Blob Blame Raw
#include "iwa_textfx.h"

#include "tparamuiconcept.h"
#include <QFontMetrics>
#include <QImage>
#include <QPainter>

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

Iwa_TextFx::Iwa_TextFx()
    : m_text(L"Lorem ipsum")
    , m_hAlign(new TIntEnumParam(Qt::AlignLeft, "Left"))
    , m_center(TPointD(0.0, 0.0))
    , m_width(200.0)
    , m_height(60.0)
    , m_font(new TFontParam())
    , m_textColor(TPixel32::Black)
    , m_boxColor(TPixel32::Transparent)
    , m_showBorder(false) {
  m_targetType->setValue(INPUT_TEXT);

  m_hAlign->addItem(Qt::AlignRight, "Right");
  m_hAlign->addItem(Qt::AlignHCenter, "Center");
  m_hAlign->addItem(Qt::AlignJustify, "Justify");

  m_text->setMultiLineEnabled(true);

  m_center->getX()->setMeasureName("fxLength");
  m_center->getY()->setMeasureName("fxLength");
  m_width->setMeasureName("fxLength");
  m_height->setMeasureName("fxLength");

  m_width->setValueRange(1.0, (std::numeric_limits<double>::max)());
  m_height->setValueRange(1.0, (std::numeric_limits<double>::max)());

  // set the initial font size to 30
  QFont font;
  font.fromString(QString::fromStdWString(m_font->getValue()));
  font.setPixelSize(30);
  m_font->setValue(font.toString().toStdWString(), true);

  bindParam(this, "targetType", m_targetType);
  bindParam(this, "columnIndex", m_columnIndex);
  bindParam(this, "text", m_text);
  bindParam(this, "hAlign", m_hAlign);
  bindParam(this, "center", m_center);
  bindParam(this, "width", m_width);
  bindParam(this, "height", m_height);
  bindParam(this, "font", m_font);
  bindParam(this, "textColor", m_textColor);
  bindParam(this, "boxColor", m_boxColor);
  bindParam(this, "showBorder", m_showBorder);
}

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

bool Iwa_TextFx::doGetBBox(double frame, TRectD &bBox,
                           const TRenderSettings &ri) {
  bBox = TConsts::infiniteRectD;
  return true;
}

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

void Iwa_TextFx::doCompute(TTile &tile, double frame,
                           const TRenderSettings &ri) {
  QString text;
  if (m_targetType->getValue() == INPUT_TEXT)
    text = QString::fromStdWString(m_text->getValue());
  else
    text = m_noteLevelStr;
  if (text.isEmpty()) return;

  QFont font;
  font.fromString(QString::fromStdWString(m_font->getValue()));

  double fac = sqrt(fabs(ri.m_affine.det()));
  int size   = (int)(fac * std::abs(font.pixelSize()));

  TPoint center = convert(
      fac * m_center->getValue(frame) -
      (tile.m_pos + tile.getRaster()->getCenterD()) +
      TPointD(ri.m_cameraBox.getLx() / 2.0, ri.m_cameraBox.getLy() / 2.0));

  QRect textBoxRect(0, 0, fac * m_width->getValue(frame),
                    fac * m_height->getValue(frame));
  textBoxRect.moveCenter(QPoint(center.x, center.y));

  Qt::AlignmentFlag hAlignFlag = (Qt::AlignmentFlag)m_hAlign->getValue();

  // For simplification, text will always be "wrapped".
  // If user would like to make the line with no break, they can just expand the
  // boundary or set the smaller font size.
  int flag = hAlignFlag | Qt::AlignVCenter | Qt::TextWordWrap;

  QFont tmpFont(font);
  tmpFont.setPixelSize(100);
  QFontMetricsF tmpFm(tmpFont);
  QRectF bbox = tmpFm.boundingRect(textBoxRect, flag, text);

  float ratio = std::min(textBoxRect.width() / bbox.width(),
                         textBoxRect.height() / bbox.height());

  // compute the font size which will just fit the item region
  int fontSize = (int)(100.0f * ratio);
  tmpFont.setPixelSize(fontSize);
  tmpFm = QFontMetricsF(tmpFont);
  bbox  = tmpFm.boundingRect(textBoxRect, flag, text);
  bool isInRect;
  if (textBoxRect.width() >= bbox.width() &&
      textBoxRect.height() >= bbox.height())
    isInRect = true;
  else
    isInRect = false;
  while (1) {
    fontSize += (isInRect) ? 1 : -1;
    if (fontSize <= 0)  // cannot draw
      return;
    if (isInRect && fontSize >= size) break;
    tmpFont.setPixelSize(fontSize);
    tmpFm = QFontMetricsF(tmpFont);
    bbox  = tmpFm.boundingRect(textBoxRect, flag, text);

    bool newIsInRect = (textBoxRect.width() >= bbox.width() &&
                        textBoxRect.height() >= bbox.height());
    if (isInRect != newIsInRect) {
      if (isInRect) fontSize--;
      break;
    }
  }

  if (size < fontSize) {
    fontSize = size;
  }

  font.setPixelSize(fontSize);
  tmpFm = QFontMetricsF(font);
  bbox  = tmpFm.boundingRect(textBoxRect, flag, text);

  double lineWidth = 0.1 * (double)fontSize;

  // Usually the text bounding box has less horizontal margin than vertical.
  // So here I added more margin to width.
  QImage img(bbox.width() + (int)(lineWidth * 4),
             bbox.height() + (int)(lineWidth * 2),
             QImage::Format_ARGB32_Premultiplied);
  img.fill(Qt::transparent);

  bbox.moveCenter(img.rect().center());

  QPainter painter(&img);
  TPixel32 boxColor = m_boxColor->getValue(frame);
  TPixel32 color    = m_textColor->getValue(frame);

  if (boxColor.m > 0)
    painter.fillRect(img.rect(), QColor((int)boxColor.r, (int)boxColor.g,
                                        (int)boxColor.b, (int)boxColor.m));

  QPen pen(QColor((int)color.r, (int)color.g, (int)color.b, (int)color.m));
  painter.setPen(pen);
  painter.setFont(font);
  painter.drawText(bbox, flag, text);

  if (m_showBorder->getValue()) {
    pen.setWidthF(lineWidth);
    pen.setJoinStyle(Qt::MiterJoin);
    painter.setPen(pen);
    painter.drawRect(img.rect().adjusted(lineWidth / 2, lineWidth / 2,
                                         -lineWidth / 2, -lineWidth / 2));
  }

  TPoint imgRootPos = center - TPoint(img.width() / 2, img.height() / 2);

  tile.getRaster()->clear();
  TRaster32P ras32 = (TRaster32P)tile.getRaster();
  TRaster64P ras64 = (TRaster64P)tile.getRaster();
  if (ras32)
    putTextImage<TRaster32P, TPixel32>(ras32, imgRootPos, img);
  else if (ras64)
    putTextImage<TRaster64P, TPixel64>(ras64, imgRootPos, img);
  else
    throw TException("Iwa_TextFx: unsupported Pixel Type");
}

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

void Iwa_TextFx::getParamUIs(TParamUIConcept *&concepts, int &length) {
  concepts = new TParamUIConcept[length = 2];

  concepts[0].m_type  = TParamUIConcept::POINT;
  concepts[0].m_label = "Center";
  concepts[0].m_params.push_back(m_center);

  concepts[1].m_type = TParamUIConcept::RECT;
  concepts[1].m_params.push_back(m_width);
  concepts[1].m_params.push_back(m_height);
  concepts[1].m_params.push_back(m_center);
}

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

std::string Iwa_TextFx::getAlias(double frame,
                                 const TRenderSettings &info) const {
  std::string alias = getFxType();
  alias += "[";

  std::string paramalias("");
  for (int i = 0; i < getParams()->getParamCount(); ++i) {
    TParam *param = getParams()->getParam(i);
    paramalias += param->getName() + "=" + param->getValueAlias(frame, 3);
  }

  if (m_targetType->getValue() == INPUT_TEXT)
    return alias + "," + std::to_string(getIdentifier()) + paramalias + "]";
  else
    return alias + std::to_string(frame) + "," +
           std::to_string(getIdentifier()) + paramalias + "]";
}

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

template <typename RASTER, typename PIXEL>
void Iwa_TextFx::putTextImage(const RASTER srcRas, TPoint &pos, QImage &img) {
  for (int j = 0; j < img.height(); j++) {
    int rasY = pos.y + j;
    if (rasY < 0) continue;
    if (srcRas->getLy() <= rasY) break;

    PIXEL *pix  = srcRas->pixels(rasY);
    QRgb *img_p = (QRgb *)img.scanLine(img.height() - j - 1);
    for (int i = 0; i < img.width(); i++, img_p++) {
      int rasX = pos.x + i;
      if (rasX < 0) continue;
      if (srcRas->getLx() <= rasX) break;

      pix[rasX].r = (typename PIXEL::Channel)(
          qRed(*img_p) * (int)PIXEL::maxChannelValue / (int)UCHAR_MAX);
      pix[rasX].g = (typename PIXEL::Channel)(
          qGreen(*img_p) * (int)PIXEL::maxChannelValue / (int)UCHAR_MAX);
      pix[rasX].b = (typename PIXEL::Channel)(
          qBlue(*img_p) * (int)PIXEL::maxChannelValue / (int)UCHAR_MAX);
      pix[rasX].m = (typename PIXEL::Channel)(
          qAlpha(*img_p) * (int)PIXEL::maxChannelValue / (int)UCHAR_MAX);
    }
  }
}

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

FX_PLUGIN_IDENTIFIER(Iwa_TextFx, "iwa_TextFx");