Blob Blame Raw


#include "toonzqt/styleeditor.h"
#include "toonzqt/colorfield.h"
#include "toonzqt/dvdialog.h"
#include "toonzqt/gutil.h"
#include "toonzqt/menubarcommand.h"
#include "toonz/cleanupcolorstyles.h"
#include "tconvert.h"
#include "tcolorstyles.h"
#include "trop.h"
#include "toonzqt/lutcalibrator.h"
#include "styledata.h"

// Qt includes
#include <QApplication>
#include <QClipboard>
#include <QLayout>
#include <QPainter>
#include <QMouseEvent>
#include <QLabel>

using namespace DVGui;

namespace {

void drawChessBoard(QPainter &p) {
  for (int x = 0; x < 4; x++) {
    for (int y = 0; y < 4; y++) {
      QColor col((x + y) % 2 == 0 ? Qt::black : Qt::white);
      p.fillRect(x * 4, y * 4, 4, 4, col);
    }
  }
}

QPixmap getIconPm(const QColor &color) {
  QPixmap retPm(16, 16);
  if (color.alpha() == 255) {
    retPm.fill(color);
    return retPm;
  }
  QPainter p(&retPm);
  drawChessBoard(p);
  p.fillRect(0, 0, 16, 16, color);
  return retPm;
}

}  // namespace

//=============================================================================

CommonChessboard *CommonChessboard::instance() {
  static CommonChessboard _instance;
  return &_instance;
}

CommonChessboard::CommonChessboard() : m_bgRas(40.0, 40.0) { update(); }

void CommonChessboard::setChessboardColors(const TPixel32 &col1,
                                           const TPixel32 &col2) {
  TRop::checkBoard(m_bgRas, col1, col2,
                   TDimensionD(m_bgRas->getLx() / 8, m_bgRas->getLy() / 8),
                   TPointD(0, 0));
  QImage img(m_bgRas->getRawData(), m_bgRas->getLx(), m_bgRas->getLy(),
             QImage::Format_ARGB32);
  m_bgPix = QPixmap::fromImage(img);
}

void CommonChessboard::update() {
  TPixel32 col1, col2;
  Preferences::instance()->getChessboardColors(col1, col2);
  setChessboardColors(col1, col2);
}

//=============================================================================
/*! \class DVGui::StyleSample
                \brief The StyleSample class provides to view a square color.

                Inherits \b QWidget.

                By default square color is set to \b TPixel32(235,235,235,255),
   you
                can set other color using setColor(); you can also define
   current color
                with a style \b TColorStyle, \b getStyle(), using setStyle().
                You can pass to constructor square size.

                StyleSample permit to manage click event, it's possile to enable
   this
                feature setting enableClick(bool on) to true.
                If it is enable when click in square class emit the signal
                clicked(const TColorStyle &style).
*/
/*!	\fn void DVGui::StyleSample::clicked(const TColorStyle &style)
                This signal is emitted when click event is enable and a style is
   set.
*/
/*!	\fn void DVGui::StyleSample::enableClick(bool on)
                Set click event enable if \b is true, disable otherwise.
                If true setStyle store current style and buttonPress emit signal
                clicked(const TColorStyle &style).
*/
StyleSample::StyleSample(QWidget *parent, int sizeX, int sizeY)
    : m_samplePixmap(sizeX, sizeY, QImage::Format_ARGB32)
    , m_bgRas(sizeX, sizeY)
    , m_style(0)
    , m_clickEnabled(false)
    , m_chessColor1(0, 0, 0)
    , m_chessColor2(255, 255, 255)
    , m_sysChessboard(false)
    , m_stretch(true)
    , m_isEditing(false) {
  setMinimumSize(sizeX, sizeY);
  setColor(TPixel32::Transparent);
  TRop::checkBoard(m_bgRas, m_chessColor1, m_chessColor2,
                   TDimensionD(sizeX / 8, sizeX / 8), TPointD(0, 0));
  setEnable(true);
}

//-----------------------------------------------------------------------------

StyleSample::~StyleSample() {
  if (m_style) delete m_style;
  m_style = 0;
}

//-----------------------------------------------------------------------------
/*! Return current StyleSample \b TColorStyle style.
 */
TColorStyle *StyleSample::getStyle() const { return m_style; }

//-----------------------------------------------------------------------------
/*! Update current square colore and, if click event is enable set current
                StyleSample \b TColorStyle style to \b style.
*/
void StyleSample::setStyle(TColorStyle &style, int colorParameterIndex) {
  // Store current color
  TPixel32 color = style.getColorParamValue(colorParameterIndex);
  m_currentColor = QColor(color.r, color.g, color.b, color.m);
  if (LutManager::instance()->isValid())
    LutManager::instance()->convert(m_currentColor);

  /*-- TSolidColorStyleの場合のみ、単色塗りつぶし --*/
  if (style.getTagId() == 3) {
    setColor(style.getMainColor());
    m_stretch = true;
  } else {
    TDimension iconDim(width(), height());

    // obtain square icon for the TMyPaintBrushStyle
    // so that the checkerboard color will become consistent with solido style
    // when the main color is semi-transparent.
    if (style.getTagId() == 4001) {
      int d   = std::min(width(), height());
      iconDim = TDimension(d, d);
    }

    TRaster32P icon = style.getIcon(iconDim);
    // TRaster32P icon =
    //  style.getIcon(qsize2Dimension(m_samplePixmap.rect().size()));
    m_samplePixmap = rasterToQImage(icon, false);  // modified in 6.2
    m_stretch      = false;
    update();
  }
  if (m_cloneStyle) {
    if (m_style) delete m_style;  // avoid memory leak
    m_style = style.clone();
  }
}

//-----------------------------------------------------------------------------
/*! Update current square colore to \b TPixel32 \b color.
                Useful for efficiency if click event is disable.
*/
void StyleSample::setColor(const TPixel32 &pixel) {
  QColor color(pixel.r, pixel.g, pixel.b, pixel.m);
  if (LutManager::instance()->isValid()) LutManager::instance()->convert(color);

  m_samplePixmap.fill(color.rgba());
  update();
}

//-----------------------------------------------------------------------------

void StyleSample::setChessboardColors(const TPixel32 &col1,
                                      const TPixel32 &col2) {
  m_chessColor1 = col1;
  m_chessColor2 = col2;
  TRop::checkBoard(m_bgRas, m_chessColor1, m_chessColor2,
                   TDimensionD(m_bgRas->getLx() / 8, m_bgRas->getLy() / 8),
                   TPointD(0, 0));
  update();
}

//-----------------------------------------------------------------------------
/*! Paint square color.
 */
void StyleSample::paintEvent(QPaintEvent *event) {
  if (!isEnable()) return;
  QPainter painter(this);
  if (m_sysChessboard) {
    painter.drawTiledPixmap(rect(),
                            DVGui::CommonChessboard::instance()->getPixmap());
  } else {
    QImage img(m_bgRas->getRawData(), m_bgRas->getLx(), m_bgRas->getLy(),
               QImage::Format_ARGB32);
    painter.drawImage(0, 0, img.scaled(size()));
  }
  if (m_stretch) {
    painter.drawImage(0, 0, m_samplePixmap.scaled(size()));
  } else {
    // put the icon on the left
    int x = 0;
    // int x = (width() - m_samplePixmap.width()) / 2;
    int y = (height() - m_samplePixmap.height()) / 2;
    painter.fillRect(rect(), m_currentColor);
    painter.drawImage(x, y, m_samplePixmap);
  }
  if (m_isEditing) {
    // QRect rect(0,0,m_bgRas->getLx(),m_bgRas->getLy());
    painter.setPen(Qt::white);
    painter.drawRect(rect().adjusted(0, 0, -1, -1));
    painter.drawRect(rect().adjusted(2, 2, -3, -3));
    painter.setPen(QColor(180, 210, 255));
    painter.drawRect(rect().adjusted(1, 1, -2, -2));
  }
}

//-----------------------------------------------------------------------------
/*! If exist current style and event click is enable emit signal
                clicked(const TColorStyle &style).
*/
void StyleSample::mousePressEvent(QMouseEvent *event) {
  if (m_clickEnabled)
    emit clicked();
  else
    event->ignore();
}

//-----------------------------------------------------------------------------

void StyleSample::mouseDoubleClickEvent(QMouseEvent *event) { event->ignore(); }

//=============================================================================
/*! \class DVGui::ChannelField
                \brief The ChannelField class is used to view an object to
   manage a color
                                         value, red, green, blue or transparency
   channel.

                Inherits \b QWidget.

                The object is composed of grid layout \b QGridLayout which
   contains a label
                in first row, first column, to identify channel, a text field \b
   IntLineEdit
                in first row, second column, and a slider in second row, second
   column.
                Texf field and slider are connected, so if you change one the
   other automatically
                change. You can set current value getChannel(), using
   setChannel().
                This two object is used to manage channel value, them range is
   fixed to [0,255].
                This object size is fixed, [50, 2*DVGui::WidgetHeight].

                To know when channel parameter value change class provides a
   signal, valueChanged(int value);
                class emit signal when slider value change or when text field is
   editing,
                see SLOT: onSliderChanged(int value) and onEditChanged(const
   QString &str)
                to know when signal is emitted.
*/
/*!	\fn void DVGui::ChannelField::valueChanged(int value)
                This signal is emitted when ChannelField, slider or text field,
   value change;
                if slider position change or text field is editing.
                \sa onEditChanged(const QString &str) and onSliderChanged(int
   value).
*/
ChannelField::ChannelField(QWidget *parent, const QString &string, int value,
                           int maxValue, bool horizontal, int labelWidth,
                           int sliderWidth)
    : QWidget(parent), m_maxValue(maxValue) {
  assert(maxValue > 0);
  assert(0 <= value && value <= m_maxValue);

  QLabel *channelName = new QLabel(string, this);
  m_channelEdit       = new DVGui::IntLineEdit(this, value, 0, maxValue);
  m_channelSlider     = new QSlider(Qt::Horizontal, this);
  m_channelSlider->setFocusPolicy(Qt::NoFocus);
  channelName->setAlignment(Qt::AlignRight | Qt::AlignVCenter);
  channelName->setFixedWidth(labelWidth);

  m_channelSlider->setRange(0, maxValue);
  m_channelSlider->setValue(value);
  if (sliderWidth > 0) m_channelSlider->setFixedWidth(sliderWidth);

  //----layout
  QGridLayout *mainLayout = new QGridLayout(this);
  mainLayout->setMargin(0);
  mainLayout->setSpacing(3);
  {
    mainLayout->addWidget(channelName, 0, 0);
    mainLayout->addWidget(m_channelEdit, 0, 1);

    mainLayout->addWidget(m_channelSlider, horizontal ? 0 : 1,
                          horizontal ? 2 : 1);
  }
  mainLayout->setColumnStretch(0, 0);
  mainLayout->setColumnStretch(1, 1);
  mainLayout->setRowStretch(2, 1);
  setLayout(mainLayout);

  //----singnal-slot connections

  bool ret = connect(m_channelEdit, SIGNAL(textChanged(const QString &)),
                     SLOT(onEditChanged(const QString &)));
  ret      = ret && connect(m_channelEdit, SIGNAL(editingFinished()),
                       SLOT(onEditFinished()));
  ret      = ret && connect(m_channelSlider, SIGNAL(valueChanged(int)),
                       SLOT(onSliderChanged(int)));
  ret      = ret && connect(m_channelSlider, SIGNAL(sliderReleased()),
                       SLOT(onSliderReleased()));
  assert(ret);
}

//-----------------------------------------------------------------------------
/*! Set current value to \b value.
                \sa getChannel()
*/
void ChannelField::setChannel(int value) {
  if (getChannel() == value) return;
  assert(0 <= value && value <= m_maxValue);
  m_channelSlider->setValue(value);
  m_channelEdit->setValue(value);
}

//-----------------------------------------------------------------------------
/*! Return current channel value.
                \sa setChannel()
*/
int ChannelField::getChannel() {
  int value = m_channelEdit->getValue();
  assert(m_channelSlider->value() == value);
  return value;
}

//-----------------------------------------------------------------------------
/*!	Set slider value to new string \b str value.
                Verify if value is lower than 255 or greater than 0, range,
   otherwise set
                current value to 255 or 0. If slider value is different from
   value in \b str
                emit signal valueChanged(int value).
*/
void ChannelField::onEditChanged(const QString &str) {
  int value = str.toInt();
  if (value < 0) value = 0;
  if (value > m_maxValue) value = m_maxValue;
  assert(0 <= value && value <= m_maxValue);
  if (str.toInt() != value) m_channelEdit->setValue(value);
  if (m_channelSlider->value() == value) return;
  m_channelSlider->setValue(value);
  emit valueChanged(value, true);
}

//-----------------------------------------------------------------------------

void ChannelField::onEditFinished() {
  emit valueChanged(m_channelEdit->getValue(), false);
}

//-----------------------------------------------------------------------------
/*! Set text field value to \b value. If text field value is different from \b
   value
                emit signal valueChanged(int value).
*/
void ChannelField::onSliderChanged(int value) {
  assert(0 <= value && value <= m_maxValue);
  if (m_channelEdit->getValue() == value) return;
  m_channelEdit->setText(QString(std::to_string(value).c_str()));
  emit valueChanged(value, true);
}

//-----------------------------------------------------------------------------

void ChannelField::onSliderReleased() {
  emit valueChanged(m_channelSlider->value(), false);
}

//=============================================================================

ColorField::ColorFieldEditorController *ColorField::m_editorController = 0;
//																							new
// ColorField::ColorFieldEditorController();

//=============================================================================
/*! \class DVGui::ColorField
                \brief The ColorField class is used to view an object to manage
   a color.

                Inherits \b QWidget.

                The object is composed of a horizontal layout \b QHBoxLayout
   which contains
                a StyleSample, and three or four ChannelField, it depend if
   transparency is
                count, to manage color channel.
                You can pass to constructor current color value, getColor(), or
   set it
                calling setColor(). You can also pass to constructor a boolean
   to know if
                manage transparency channel or not, and an integer for
   StyleSample size.

                To know when color value change class provides a signal,
   colorChanged(const TPixel32 &);
                class emit signal when one ChannelField value change.
                See SLOT: onRedChannelChanged(int value),
   onGreenChannelChanged(int value),
                onBlueChannelChanged(int value) and onAlphaChannelChanged(int
   value) to know
                when signal is emitted.

                \b Example: initialize a ColorField with transparency channel.
                \code
                        ColorField* colorFld = new
   ColorField(0,true,TPixel32(0,0,255,255),50);
                \endcode
                \b Result:
                        \image html ColorField.jpg
*/
/*!	\fn void DVGui::ColorField::colorChanged(const TPixel32 &)
                This signal is emitted when a channel value of current color
   change.
*/
/*!	\fn TPixel32  DVGui::ColorField::getColor() const
                Return ColorField current color.
*/
ColorField::ColorField(QWidget *parent, bool isAlphaActive, TPixel32 color,
                       int squareSize, bool useStyleEditor, int sliderWidth)
    : QWidget(parent)
    , m_color(color)
    , m_notifyEditingChange(true)
    , m_useStyleEditor(useStyleEditor) {
  setMaximumHeight(squareSize);
  QHBoxLayout *layout = new QHBoxLayout(this);
  layout->setMargin(0);
  layout->setSpacing(5);

  layout->setSizeConstraint(QLayout::SetFixedSize);

  int h = WidgetHeight;

  m_colorSample = new StyleSample(this, squareSize, squareSize);
  m_colorSample->setColor(m_color);
  m_redChannel =
      new ChannelField(this, tr("R:"), m_color.r, 255, false, 13, sliderWidth);
  connect(m_redChannel, SIGNAL(valueChanged(int, bool)),
          SLOT(onRedChannelChanged(int, bool)));
  m_greenChannel =
      new ChannelField(this, tr("G:"), m_color.g, 255, false, 13, sliderWidth);
  connect(m_greenChannel, SIGNAL(valueChanged(int, bool)),
          SLOT(onGreenChannelChanged(int, bool)));
  m_blueChannel =
      new ChannelField(this, tr("B:"), m_color.b, 255, false, 13, sliderWidth);
  connect(m_blueChannel, SIGNAL(valueChanged(int, bool)),
          SLOT(onBlueChannelChanged(int, bool)));
  m_alphaChannel =
      new ChannelField(this, tr("A:"), m_color.m, 255, false, 13, sliderWidth);
  connect(m_alphaChannel, SIGNAL(valueChanged(int, bool)),
          SLOT(onAlphaChannelChanged(int, bool)));

  layout->addWidget(m_colorSample);
  layout->addWidget(m_redChannel);
  layout->addWidget(m_greenChannel);
  layout->addWidget(m_blueChannel);
  layout->addWidget(m_alphaChannel);

  if (!isAlphaActive) m_alphaChannel->hide();
  setLayout(layout);
}

//-----------------------------------------------------------------------------
/*! Set ColorField current color to \b color. Update channel value of
                \b ChannelField and \b StyleSample color.
*/

void ColorField::setAlphaActive(bool active) {
  if (active && !m_alphaChannel->isVisible()) {
    m_alphaChannel->show();
    connect(m_alphaChannel, SIGNAL(valueChanged(int, bool)),
            SLOT(onAlphaChannelChanged(int, bool)));
    assert(m_color.m == 255);
    m_alphaChannel->setChannel(0);
    m_color.m = 0;
    m_colorSample->setColor(m_color);
    emit colorChanged(m_color, false);
  } else if (!active && m_alphaChannel->isVisible()) {
    m_alphaChannel->hide();
    disconnect(m_alphaChannel, SIGNAL(valueChanged(int, bool)), this,
               SLOT(onAlphaChannelChanged(int, bool)));
    if (m_color.m != 255) {
      m_alphaChannel->setChannel(255);
      m_color.m = 255;
      m_colorSample->setColor(m_color);
      emit colorChanged(m_color, false);
    }
  }
}

//------------------------------

void ColorField::setColor(const TPixel32 &color) {
  if (m_color == color) return;
  m_color = color;
  updateChannels();
  m_colorSample->setColor(m_color);
}

//-----------------------------------------------------------------------------
/*! Set all \b ChannelField channel value to ColorField current color.
 */
void ColorField::hideChannelsFields(bool hide) {
  if (hide) {
    m_redChannel->hide();
    m_greenChannel->hide();
    m_blueChannel->hide();
    m_alphaChannel->hide();
    disconnect(m_redChannel, SIGNAL(valueChanged(int, bool)), this,
               SLOT(onRedChannelChanged(int, bool)));
    disconnect(m_greenChannel, SIGNAL(valueChanged(int, bool)), this,
               SLOT(onGreenChannelChanged(int, bool)));
    disconnect(m_blueChannel, SIGNAL(valueChanged(int, bool)), this,
               SLOT(onBlueChannelChanged(int, bool)));
    disconnect(m_alphaChannel, SIGNAL(valueChanged(int, bool)), this,
               SLOT(onAlphaChannelChanged(int, bool)));
  } else {
    m_redChannel->show();
    m_greenChannel->show();
    m_blueChannel->show();
    m_alphaChannel->show();
    ;
    connect(m_redChannel, SIGNAL(valueChanged(int, bool)),
            SLOT(onRedChannelChanged(int, bool)));
    connect(m_greenChannel, SIGNAL(valueChanged(int, bool)),
            SLOT(onGreenChannelChanged(int, bool)));
    connect(m_blueChannel, SIGNAL(valueChanged(int, bool)),
            SLOT(onBlueChannelChanged(int, bool)));
    connect(m_alphaChannel, SIGNAL(valueChanged(int, bool)),
            SLOT(onAlphaChannelChanged(int, bool)));
  }
}

//-----------------------------------------------------------------------------
/*! Set all \b ChannelField channel value to ColorField current color.
 */
void ColorField::updateChannels() {
  m_redChannel->setChannel(m_color.r);
  m_greenChannel->setChannel(m_color.g);
  m_blueChannel->setChannel(m_color.b);
  m_alphaChannel->setChannel(m_color.m);
}

//-----------------------------------------------------------------------------

void ColorField::mousePressEvent(QMouseEvent *event) {
  if (event->button() != Qt::LeftButton) return;
  QPoint p = event->pos();
  if (!m_colorSample->visibleRegion().contains(p)) return;

  if (!m_useStyleEditor || !getEditorController()) return;

  getEditorController()->edit(this);
}

//-----------------------------------------------------------------------------

void ColorField::mouseDoubleClickEvent(QMouseEvent *event) {
  QPoint p = event->pos();
  if (!m_colorSample->visibleRegion().contains(p)) return;

  if (!m_useStyleEditor || !getEditorController()) return;

  CommandManager::instance()->execute("MI_OpenStyleControl");
  getEditorController()->edit(this);
}

//-----------------------------------------------------------------------------

void ColorField::hideEvent(QHideEvent *) {
  if (!m_useStyleEditor || !getEditorController()) return;

  getEditorController()->hide();
}

//-----------------------------------------------------------------------------

void ColorField::contextMenuEvent(QContextMenuEvent *event) {
  bool hasColor = QApplication::clipboard()->mimeData()->hasColor();
  const StyleData *data =
      dynamic_cast<const StyleData *>(QApplication::clipboard()->mimeData());

  QMenu menu(this);
  if (hasColor) {  // pasting QColor
    QColor color = qvariant_cast<QColor>(
        QApplication::clipboard()->mimeData()->colorData());
    QAction *action = new QAction(tr("Paste Color"), this);
    action->setIcon(QIcon(getIconPm(color)));
    action->setData(color);

    connect(action, SIGNAL(triggered()), this, SLOT(onPasteColor()));
    menu.addAction(action);
    menu.addSeparator();
  } else if (data && data->getStyleCount() > 0) {  // pasting styles colors
    // show 10 styles in maximum
    int styleCount = std::min(10, data->getStyleCount());
    for (int i = 0; i < styleCount; i++) {
      QString styleName = QString::fromStdWString(data->getStyle(i)->getName());
      TPixel32 color    = data->getStyle(i)->getMainColor();
      QColor _color(color.r, color.g, color.b, color.m);

      QAction *action =
          new QAction(tr("Paste Color of %1").arg(styleName), this);
      action->setIcon(QIcon(getIconPm(_color)));
      action->setData(_color);

      connect(action, SIGNAL(triggered()), this, SLOT(onPasteColor()));
      menu.addAction(action);
    }
    menu.addSeparator();
  }

  QAction *copyAction = new QAction(tr("Copy Color"), this);
  connect(copyAction, SIGNAL(triggered()), this, SLOT(onCopyColor()));
  menu.addAction(copyAction);

  menu.exec(event->globalPos());
  event->accept();
}

//-----------------------------------------------------------------------------
/*! If current red channel value of color is different from \b value set it,
                change \b StyleSample color and emit signal \b
   colorChanged(const TPixel32 &).
*/
void ColorField::onRedChannelChanged(int value, bool isDragging) {
  if (m_color.r == value) {
    if (!isDragging) emit colorChanged(m_color, isDragging);
    return;
  }
  m_color = TPixel32(value, m_color.g, m_color.b, m_color.m);
  m_colorSample->setColor(m_color);
  emit colorChanged(m_color, isDragging);
}

//-----------------------------------------------------------------------------
/*! If current green channel value of color is different from \b value set it,
                change \b StyleSample color and emit signal \b
   colorChanged(const TPixel32 &).
*/
void ColorField::onGreenChannelChanged(int value, bool isDragging) {
  if (m_color.g == value) {
    if (!isDragging) emit colorChanged(m_color, isDragging);
    return;
  }
  m_color = TPixel32(m_color.r, value, m_color.b, m_color.m);
  m_colorSample->setColor(m_color);
  emit colorChanged(m_color, isDragging);
}

//-----------------------------------------------------------------------------
/*! If current blue channel value of color is different from \b value set it,
                change \b StyleSample color and emit signal \b
   colorChanged(const TPixel32 &).
*/
void ColorField::onBlueChannelChanged(int value, bool isDragging) {
  if (m_color.b == value) {
    if (!isDragging) emit colorChanged(m_color, isDragging);
    return;
  }
  m_color = TPixel32(m_color.r, m_color.g, value, m_color.m);
  m_colorSample->setColor(m_color);
  emit colorChanged(m_color, isDragging);
}

//-----------------------------------------------------------------------------
/*! If current alpha channel value of color is different from \b value set it,
                change \b StyleSample color and emit signal \b
   colorChanged(const TPixel32 &).
*/
void ColorField::onAlphaChannelChanged(int value, bool isDragging) {
  if (m_color.m == value) {
    if (!isDragging) emit colorChanged(m_color, isDragging);
    return;
  }
  m_color = TPixel32(m_color.r, m_color.g, m_color.b, value);
  m_colorSample->setColor(m_color);
  emit colorChanged(m_color, isDragging);
}

//-----------------------------------------------------------------------------

void ColorField::onPasteColor() {
  QColor color = qobject_cast<QAction *>(sender())->data().value<QColor>();

  m_color = TPixel32(color.red(), color.green(), color.blue(), color.alpha());
  if (!m_alphaChannel->isVisible()) m_color.m = 255;
  m_colorSample->setColor(m_color);
  updateChannels();
  emit colorChanged(m_color, false);
}

//-----------------------------------------------------------------------------

void ColorField::onCopyColor() {
  QColor color(m_color.r, m_color.g, m_color.b, m_color.m);

  QMimeData *data = new QMimeData();
  data->setColorData(color);
  QApplication::clipboard()->setMimeData(data);
}

//-----------------------------------------------------------------------------

void ColorField::setChessboardColors(const TPixel32 &col1,
                                     const TPixel32 &col2) {
  m_colorSample->setChessboardColors(col1, col2);
}

//-----------------------------------------------------------------------------

void ColorField::setEditorController(
    ColorFieldEditorController *editorController) {
  m_editorController = editorController;
}

//-----------------------------------------------------------------------------

ColorField::ColorFieldEditorController *ColorField::getEditorController() {
  return m_editorController;
}

//-----------------------------------------------------------------------
#define SQUARESIZE 50

void CleanupColorField::onBrightnessChannelChanged(int value, bool dragging) {
  m_cleanupStyle->setBrightness(value);
  m_ph->notifyColorStyleChanged(dragging);
}

//-----------------------------------------------

void CleanupColorField::onContrastChannelChanged(int value, bool dragging) {
  m_cleanupStyle->setContrast(value);
  m_ph->notifyColorStyleChanged(dragging);
}

//-----------------------------------------------

void CleanupColorField::onCThresholdChannelChanged(int value, bool dragging) {
  ((TBlackCleanupStyle *)m_cleanupStyle)->setColorThreshold((double)value);
  m_ph->notifyColorStyleChanged(dragging);
}

//-----------------------------------------------

void CleanupColorField::onWThresholdChannelChanged(int value, bool dragging) {
  ((TBlackCleanupStyle *)m_cleanupStyle)->setWhiteThreshold((double)value);
  m_ph->notifyColorStyleChanged(dragging);
}

void CleanupColorField::onHRangeChannelChanged(int value, bool dragging) {
  ((TColorCleanupStyle *)m_cleanupStyle)->setHRange(value);
  m_ph->notifyColorStyleChanged(dragging);
}

//-----------------------------------------------

void CleanupColorField::onLineWidthChannelChanged(int value, bool dragging) {
  ((TColorCleanupStyle *)m_cleanupStyle)->setLineWidth(value);
  m_ph->notifyColorStyleChanged(dragging);
}

//---------------------------------------------------

void CleanupColorField::mousePressEvent(QMouseEvent *event) {
  if (event->button() != Qt::LeftButton) return;

  emit StyleSelected(m_cleanupStyle);

  if (getEditorController()) getEditorController()->edit(this);
}

//-----------------------------------------------
CleanupColorField::CleanupColorField(QWidget *parent,
                                     TCleanupStyle *cleanupStyle,
                                     TPaletteHandle *ph, bool greyMode)
    : QWidget(parent)
    , m_style(cleanupStyle)
    , m_cleanupStyle(cleanupStyle)
    , m_ph(ph)
    , m_greyMode(greyMode)
    , m_notifyEditingChange(true) {
  TBlackCleanupStyle *bs = dynamic_cast<TBlackCleanupStyle *>(cleanupStyle);
  TColorCleanupStyle *cs = dynamic_cast<TColorCleanupStyle *>(cleanupStyle);
  assert(bs || cs);

  m_colorSample = new StyleSample(this, SQUARESIZE / 2, SQUARESIZE);
  m_brightnessChannel =
      new ChannelField(this, DVGui::CleanupColorField::tr("Brightness:"),
                       cleanupStyle->getBrightness(), 100, true, 75, -1);
  m_contrastChannel =
      new ChannelField(this, DVGui::CleanupColorField::tr("Contrast:"),
                       cleanupStyle->getContrast(), 100, true, 75, -1);
  if (!greyMode) {
    if (bs) {
      m_cThresholdChannel =
          new ChannelField(this, DVGui::CleanupColorField::tr("Color Thres"),
                           bs->getColorThreshold(), 100, true, 75, -1);
      m_wThresholdChannel =
          new ChannelField(this, DVGui::CleanupColorField::tr("White Thres"),
                           bs->getWhiteThreshold(), 100, true, 75, -1);
    } else  // cs
    {
      m_hRangeChannel =
          new ChannelField(this, DVGui::CleanupColorField::tr("H Range"),
                           cs->getHRange(), 120, true, 75, -1);
      m_lineWidthChannel =
          new ChannelField(this, DVGui::CleanupColorField::tr("Line Width"),
                           cs->getLineWidth(), 100, true, 75, -1);
    }
  }

  m_colorSample->setStyle(*cleanupStyle, 0);

  //---- layout

  QHBoxLayout *mainLay = new QHBoxLayout();
  mainLay->setMargin(8);
  mainLay->setSpacing(5);
  {
    mainLay->addWidget(m_colorSample, 0);

    QVBoxLayout *paramLay = new QVBoxLayout();
    paramLay->setMargin(0);
    paramLay->setSpacing(3);
    {
      paramLay->addWidget(m_brightnessChannel);
      paramLay->addWidget(m_contrastChannel);
      if (!greyMode) {
        if (bs) {
          paramLay->addWidget(m_cThresholdChannel);
          paramLay->addWidget(m_wThresholdChannel);
        } else {
          paramLay->addWidget(m_hRangeChannel);
          paramLay->addWidget(m_lineWidthChannel);
        }
      }
    }
    mainLay->addLayout(paramLay, 1);
  }
  setLayout(mainLay);

  //---- signal-slot connections

  bool ret = true;
  ret = ret && connect(m_brightnessChannel, SIGNAL(valueChanged(int, bool)),
                       SLOT(onBrightnessChannelChanged(int, bool)));
  ret = ret && connect(m_contrastChannel, SIGNAL(valueChanged(int, bool)),
                       SLOT(onContrastChannelChanged(int, bool)));
  if (!greyMode) {
    if (bs) {
      ret = ret && connect(m_cThresholdChannel, SIGNAL(valueChanged(int, bool)),
                           SLOT(onCThresholdChannelChanged(int, bool)));
      ret = ret && connect(m_wThresholdChannel, SIGNAL(valueChanged(int, bool)),
                           SLOT(onWThresholdChannelChanged(int, bool)));
    } else {
      ret = ret && connect(m_hRangeChannel, SIGNAL(valueChanged(int, bool)),
                           SLOT(onHRangeChannelChanged(int, bool)));
      ret = ret && connect(m_lineWidthChannel, SIGNAL(valueChanged(int, bool)),
                           SLOT(onLineWidthChannelChanged(int, bool)));
    }
  }
}

//--------------------------------------------------------------

void CleanupColorField::updateColor() {
  if (m_cleanupStyle->canUpdate()) {
    m_cleanupStyle->invalidateIcon();
    m_colorSample->setStyle(*m_cleanupStyle, 0);

    m_brightnessChannel->setChannel(m_cleanupStyle->getBrightness());
    if (m_cleanupStyle->isContrastEnabled())
      m_contrastChannel->setChannel(m_cleanupStyle->getContrast());

    TBlackCleanupStyle *bs;
    TColorCleanupStyle *cs;
    if ((bs = dynamic_cast<TBlackCleanupStyle *>(m_cleanupStyle)) &&
        !m_greyMode) {
      m_cThresholdChannel->setChannel(bs->getColorThreshold());
      m_wThresholdChannel->setChannel(bs->getWhiteThreshold());
    } else if ((cs = dynamic_cast<TColorCleanupStyle *>(m_cleanupStyle))) {
      m_hRangeChannel->setChannel(cs->getHRange());
      m_lineWidthChannel->setChannel(cs->getLineWidth());
    }
  }
}

//--------------------------------------------------------------

TPixel32 CleanupColorField::getColor() const {
  return m_cleanupStyle->getMainColor();
}

//--------------------------------------------------------------

void CleanupColorField::setColor(const TPixel32 &color) {
  if (m_cleanupStyle->getMainColor() == color) return;

  m_cleanupStyle->setMainColor(color);
  m_cleanupStyle->invalidateIcon();
  m_colorSample->setStyle(*m_cleanupStyle, 0);
  m_ph->notifyColorStyleChanged(false);
}

//------------------------------------------------------------

TPixel32 CleanupColorField::getOutputColor() const {
  return m_cleanupStyle->getColorParamValue(1);
}

//------------------------------------------------------------

void CleanupColorField::setOutputColor(const TPixel32 &color) {
  if (getOutputColor() == color) return;

  m_cleanupStyle->setColorParamValue(1, color);
  m_cleanupStyle->invalidateIcon();
  m_colorSample->setStyle(*m_cleanupStyle, 0);
  m_ph->notifyColorStyleChanged();
}

//------------------------------------------------------------

void CleanupColorField::setStyle(TColorStyle *style) {
  if (getColor() == style->getMainColor() &&
      getOutputColor() == style->getColorParamValue(1))
    return;

  m_cleanupStyle->setMainColor(style->getMainColor());
  m_cleanupStyle->setColorParamValue(1, style->getColorParamValue(1));
  m_cleanupStyle->invalidateIcon();
  m_colorSample->setStyle(*m_cleanupStyle, 0);
  m_ph->notifyColorStyleChanged();
}

//------------------------------------------------------------

CleanupColorField::CleanupColorFieldEditorController
    *CleanupColorField::m_editorController = 0;

CleanupColorField::CleanupColorFieldEditorController *
CleanupColorField::getEditorController() {
  return m_editorController;
}

//-----------------------------------------------------------------------------

void CleanupColorField::setEditorController(
    CleanupColorFieldEditorController *editorController) {
  m_editorController = editorController;
}

//------------------------------------------------------------------

void CleanupColorField::mouseDoubleClickEvent(QMouseEvent *event) {
  QPoint p = event->pos();
  if (!m_colorSample->visibleRegion().contains(p)) return;
  emit StyleSelected(m_cleanupStyle);
  if (!getEditorController()) return;

  CommandManager::instance()->execute("MI_OpenStyleControl");
  getEditorController()->edit(this);
}

//-----------------------------------------------------------

void CleanupColorField::hideEvent(QHideEvent *) {
  if (!getEditorController()) return;
  getEditorController()->edit(0);
  getEditorController()->hide();
  // setEditorController(0);
}

//-----------------------------------------------------------

void CleanupColorField::setContrastEnabled(bool enable) {
  m_contrastChannel->setEnabled(enable);
  m_cleanupStyle->enableContrast(enable);
}