Blob Blame Raw


// Toonz includes
#include "tvectorimage.h"
#include "trasterimage.h"
#include "tlevel_io.h"
#include "tofflinegl.h"
#include "tropcm.h"
#include "tvectorrenderdata.h"
#include "tsystem.h"
#include "tvectorgl.h"
#include "tcolorstyles.h"

// TnzCore includes
#include "tfiletype.h"
#include "tvectorbrushstyle.h"

// TnzLib includes
#include "toonz/imagestyles.h"
#include "toonz/toonzfolders.h"

// Qt includes
#include <QDir>
#include <QImage>
#include <QOffscreenSurface>
#include <QThread>
#include <QGuiApplication>
#include <QOpenGLContext>
#include <QOpenGLFramebufferObject>

#include "toonz/stylemanager.h"

#include <QVector>

//********************************************************************************
//    Local namespace stuff
//********************************************************************************

namespace {

void convertRaster32ToImage(TRaster32P ras, QImage *image) {
  int lx = ras->getLx();
  int ly = ras->getLy();
  int i, j;
  ras->lock();
  for (i = 0; i < lx; i++)
    for (j = 0; j < ly; j++) {
      TPixel32 pix = ras->pixels(ly - 1 - j)[i];
      QRgb value;
      value = qRgba(pix.r, pix.g, pix.b, pix.m);
      image->setPixel(i, j, value);
    }
  ras->unlock();
}

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

QImage rasterToQImage(const TRasterP &ras, bool premultiplied = true,
                      bool mirrored = true) {
  if (TRaster32P ras32 = ras) {
    QImage image(ras->getRawData(), ras->getLx(), ras->getLy(),
                 premultiplied ? QImage::Format_ARGB32_Premultiplied
                               : QImage::Format_ARGB32);
    if (mirrored) return image.mirrored();
    return image;
  } else if (TRasterGR8P ras8 = ras) {
    QImage image(ras->getRawData(), ras->getLx(), ras->getLy(), ras->getWrap(),
                 QImage::Format_Indexed8);
    static QVector<QRgb> colorTable;
    if (colorTable.size() == 0) {
      int i;
      for (i = 0; i < 256; i++) colorTable.append(QColor(i, i, i).rgb());
    }
    image.setColorTable(colorTable);
    if (mirrored) return image.mirrored();
    return image;
  }
  return QImage();
}

}  // namespace

//********************************************************************************
//    FavoritesManager implementation
//********************************************************************************

FavoritesManager::FavoritesManager() {
  m_fpPinsToTop = ToonzFolder::getMyModuleDir() + "pintotopbrushes.txt";
  m_xxPinsToTop = false;
  loadPinsToTop();
}

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

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

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

bool FavoritesManager::loadPinsToTop() {
  if (!TFileStatus(m_fpPinsToTop).doesExist()) return false;

  TIStream is(m_fpPinsToTop);
  if (!is) throw TException("Can't read XML");
  std::string tagName;

  if (!is.matchTag(tagName) || tagName != "PinsToTop") return false;

  m_pinsToTop.clear();
  while (!is.matchEndTag()) {
    if (!is.matchTag(tagName)) throw TException("Expected tag");
    if (tagName == "BrushIdName") {
      std::string brushname;
      is >> brushname;
      m_pinsToTop.push_back(brushname);
      if (!is.matchEndTag()) throw TException("Expected end tag");
    }
  }
  m_xxPinsToTop = false;

  return true;
}

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

void FavoritesManager::savePinsToTop() {
  if (!m_xxPinsToTop) return;

  TOStream os(m_fpPinsToTop);
  if (!os) throw TException("Can't write XML");

  os.openChild("PinsToTop");
  for (auto &idname : m_pinsToTop) {
    os.openChild("BrushIdName", {});
    os << idname;
    os.closeChild();
  }
  os.closeChild();
}

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

bool FavoritesManager::getPinToTop(std::string idname) const {
  return m_pinsToTop.contains(idname);
}

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

void FavoritesManager::setPinToTop(std::string idname, bool state) {
  int index = m_pinsToTop.indexOf(idname);
  if (state && index == -1) {
    m_xxPinsToTop = true;
    m_pinsToTop.append(idname);
  } else if (!state && index != -1) {
    m_xxPinsToTop = true;
    m_pinsToTop.removeAll(idname);
  }
}

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

void FavoritesManager::togglePinToTop(std::string idname) {
  int index = m_pinsToTop.indexOf(idname);
  if (index != -1)
    m_pinsToTop.removeAt(index);
  else
    m_pinsToTop.append(idname);
  m_xxPinsToTop = true;
}

//********************************************************************************
//    BaseStyleManager implementation
//********************************************************************************

TFilePath BaseStyleManager::s_rootPath;
BaseStyleManager::ChipData BaseStyleManager::s_emptyChipData;

BaseStyleManager::BaseStyleManager(const TFilePath &stylesFolder,
                                   QString filters, QSize chipSize)
    : m_stylesFolder(stylesFolder)
    , m_filters(filters)
    , m_chipSize(chipSize)
    , m_loaded(false)
    , m_isIndexed(false) {}

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

const BaseStyleManager::ChipData &BaseStyleManager::getData(int index) const {
  if (m_isIndexed) {
    return (index < 0 || index >= m_indexes.count())
               ? s_emptyChipData
               : m_chips[m_indexes[index]];
  } else {
    return (index < 0 || index >= m_chips.count()) ? s_emptyChipData
                                                   : m_chips[index];
  }
}

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

void BaseStyleManager::applyFilter() {
  FavoritesManager *favMan = FavoritesManager::instance();
  QList<int> indexes;

  m_indexes.clear();
  int len = m_chips.count();
  for (int i = 0; i < len; i++) {
    auto &chip = m_chips[i];
    if (chip.desc.indexOf(m_searchText, 0, Qt::CaseInsensitive) >= 0) {
      if (favMan->getPinToTop(chip.idname)) {
        chip.markPinToTop = true;
        m_indexes.append(i);
      } else {
        chip.markPinToTop = false;
        indexes.append(i);
      }
    }
  }

  bool hasPinsToTop = m_indexes.count() > 0;
  m_indexes.append(indexes);
  m_isIndexed = (m_indexes.count() != len) || hasPinsToTop;
}

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

TFilePath BaseStyleManager::getRootPath() { return s_rootPath; }

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

void BaseStyleManager::setRootPath(const TFilePath &rootPath) {
  s_rootPath = rootPath;
}

//********************************************************************************
//    StyleLoaderTask definition
//********************************************************************************

class CustomStyleManager::StyleLoaderTask final : public TThread::Runnable {
  CustomStyleManager *m_manager;
  TFilePath m_fp;
  ChipData m_data;
  std::shared_ptr<QOffscreenSurface> m_offScreenSurface;

public:
  StyleLoaderTask(CustomStyleManager *manager, const TFilePath &fp);

  void run() override;

  void onFinished(TThread::RunnableP sender) override;
};

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

CustomStyleManager::StyleLoaderTask::StyleLoaderTask(
    CustomStyleManager *manager, const TFilePath &fp)
    : m_manager(manager), m_fp(fp) {
  connect(this, SIGNAL(finished(TThread::RunnableP)), this,
          SLOT(onFinished(TThread::RunnableP)));

  if (QThread::currentThread() == qGuiApp->thread()) {
    m_offScreenSurface.reset(new QOffscreenSurface());
    m_offScreenSurface->setFormat(QSurfaceFormat::defaultFormat());
    m_offScreenSurface->create();
  }
}

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

void CustomStyleManager::StyleLoaderTask::run() {
  m_data = m_manager->createPattern(m_fp, m_offScreenSurface);
}

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

void CustomStyleManager::StyleLoaderTask::onFinished(
    TThread::RunnableP sender) {
  // On the main thread...
  if (!m_data.image.isNull())  // Everything went ok
  {
    m_manager->m_chips.append(m_data);
    emit m_manager->patternAdded();
  }
}

//********************************************************************************
//    CustomStyleManager implementation
//********************************************************************************

CustomStyleManager::CustomStyleManager(std::string rasterIdName,
                                       std::string vectorIdName,
                                       const TFilePath &stylesFolder,
                                       QString filters, QSize chipSize)
    : BaseStyleManager(stylesFolder, filters, chipSize)
    , m_started(false)
    , m_rasterIdName(rasterIdName)
    , m_vectorIdName(vectorIdName) {
  m_executor.setMaxActiveTasks(1);
}

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

void CustomStyleManager::loadItems() {
  // Build the folder to be read
  const TFilePath &rootFP(getRootPath());

  assert(rootFP != TFilePath());
  if (rootFP == TFilePath()) return;

  QDir patternDir(
      QString::fromStdWString((rootFP + m_stylesFolder).getWideString()));
  patternDir.setNameFilters(m_filters.split(' '));

  // Read the said folder
  TFilePathSet fps;
  try {
    TSystem::readDirectory(fps, patternDir);
  } catch (...) {
    return;
  }

  // Delete patterns no longer in the folder
  TFilePathSet newFps;
  TFilePathSet::iterator it;
  int i;
  for (i = 0; i < m_chips.size(); i++) {
    ChipData data = m_chips.at(i);
    for (it = fps.begin(); it != fps.end(); ++it) {
      int isVector = (it->getType() == "pli") ? 1 : 0;
      QString name = QString::fromStdWString(it->getWideName());
      if (data.name == name && data.isVector == isVector) break;
    }

    if (it == fps.end()) {
      m_chips.removeAt(i);
      i--;
    } else
      fps.erase(it);  // The style is not new, so don't generate tasks for it
  }

  // For each (now new) file entry, generate a fetching task
  for (TFilePathSet::iterator it = fps.begin(); it != fps.end(); it++)
    m_executor.addTask(new StyleLoaderTask(this, *it));
}

QImage CustomStyleManager::makeIcon(
    const TFilePath &path, const QSize &qChipSize,
    std::shared_ptr<QOffscreenSurface> offsurf) {
  try {
    // Fetch the level
    TLevelReaderP lr(path);
    TLevelP level = lr->loadInfo();
    if (!level || level->getFrameCount() == 0) return QImage();

    // Fetch the image of the first frame in the level
    TLevel::Iterator frameIt = level->begin();
    if (frameIt == level->end()) return QImage();
    TImageP img = lr->getFrameReader(frameIt->first)->load();

    // Process the image
    TDimension chipSize(qChipSize.width(), qChipSize.height());

    TVectorImageP vimg = img;
    TRasterImageP rimg = img;

    TRaster32P ras;

    QImage image;

    if (vimg) {
      assert(level->getPalette());
      TPalette *vPalette = level->getPalette();
      assert(vPalette);
      vimg->setPalette(vPalette);

#ifdef LINUX
      TOfflineGL *glContext = 0;
      glContext             = TOfflineGL::getStock(chipSize);
      glContext->clear(TPixel32::White);
#else
      QOpenGLContext *glContext = new QOpenGLContext();
      if (QOpenGLContext::currentContext())
        glContext->setShareContext(QOpenGLContext::currentContext());
      glContext->setFormat(QSurfaceFormat::defaultFormat());
      glContext->create();
      glContext->makeCurrent(offsurf.get());
      // attaching stencil buffer here as some styles use it
      QOpenGLFramebufferObject fb(
          chipSize.lx, chipSize.ly,
          QOpenGLFramebufferObject::CombinedDepthStencil);

      fb.bind();
      // Draw
      glViewport(0, 0, chipSize.lx, chipSize.ly);
      glClearColor(1.0, 1.0, 1.0, 1.0);  // clear with white color
      glClear(GL_COLOR_BUFFER_BIT);

      glMatrixMode(GL_PROJECTION);
      glLoadIdentity();
      gluOrtho2D(0, chipSize.lx, 0, chipSize.ly);

      glMatrixMode(GL_MODELVIEW);
      glLoadIdentity();
#endif

      TRectD bbox = img->getBBox();
      double scx  = 0.8 * chipSize.lx / bbox.getLx();
      double scy  = 0.8 * chipSize.ly / bbox.getLy();
      double sc   = std::min(scx, scy);
      double dx   = 0.5 * chipSize.lx;
      double dy   = 0.5 * chipSize.ly;

      TAffine aff =
          TTranslation(dx, dy) * TScale(sc) *
          TTranslation(-0.5 * (bbox.x0 + bbox.x1), -0.5 * (bbox.y0 + bbox.y1));
      TVectorRenderData rd(aff, chipSize, vPalette, 0, true);

#ifdef LINUX
      glContext->draw(img, rd);
      // No need to clone! The received raster already is a copy of the
      // context's buffer
      ras = glContext->getRaster();  //->clone();
#else
      tglDraw(rd, vimg.getPointer());

      image = QImage(fb.toImage().scaled(QSize(chipSize.lx, chipSize.ly),
                                         Qt::IgnoreAspectRatio,
                                         Qt::SmoothTransformation));
      fb.release();
      glContext->deleteLater();
#endif
    } else if (rimg) {
      TDimension size = rimg->getRaster()->getSize();
      if (size == chipSize)
        ras = rimg->getRaster()->clone();  // Yep, this may be necessary
      else {
        TRaster32P rout(chipSize);

        TRop::resample(rout, rimg->getRaster(),
                       TScale((double)chipSize.lx / size.lx,
                              (double)chipSize.ly / size.ly));

        TRop::addBackground(rout, TPixel::White);
        ras = rout;
      }
#ifndef LINUX
      // image = QImage(chipSize.lx, chipSize.ly, QImage::Format_RGB32);
      // convertRaster32ToImage(ras, &image);
      image = rasterToQImage(ras);
#endif
    } else
      assert(!"unsupported type for custom styles!");

#ifdef LINUX
    // image = QImage(chipSize.lx, chipSize.ly, QImage::Format_RGB32);
    // convertRaster32ToImage(ras, &image);
    image = rasterToQImage(ras);
#endif

    return image;
  } catch (...) {
  }

  return QImage();
}

CustomStyleManager::ChipData CustomStyleManager::createPattern(
    const TFilePath &path, std::shared_ptr<QOffscreenSurface> offsurf) {
  ChipData data;

  bool isVector = (path.getType() == "pli" || path.getType() == "svg");

  // Generate preview
  try {
    data.image = makeIcon(path, getChipSize(), offsurf);
  } catch (...) {
  }

  if (!data.image.isNull()) {
    data.name     = QString::fromStdWString(path.getWideName());
    data.desc     = data.name;
    data.isVector = isVector;
    if (isVector)
      data.idname = m_vectorIdName + data.name.toStdString();
    else
      data.idname = m_rasterIdName + data.name.toStdString();
    data.hash = TColorStyle::generateHash(data.idname);
  }
  return data;
}

//********************************************************************************
//    TextureStyleManager implementation
//********************************************************************************

TextureStyleManager::TextureStyleManager(const TFilePath &stylesFolder,
                                         QSize chipSize)
    : BaseStyleManager(stylesFolder, QString(), chipSize) {}

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

void TextureStyleManager::loadTexture(const TFilePath &fp) {
  if (fp == TFilePath()) {
    TRaster32P ras(25, 25);
    TTextureStyle::fillCustomTextureIcon(ras);
    // ras->fill(TPixel::Blue);
    ChipData customText(
        QString(""), QObject::tr("Custom Texture", "TextureStyleChooserPage"),
        rasterToQImage(ras), 4, false, ras,
        TTextureStyle::staticBrushIdName(L""));
    customText.hash = TTextureStyle::generateHash(customText.idname);
    m_chips.append(customText);
    return;
  }

  TRasterP ras;
  TImageReader::load(fp, ras);
  if (!ras || ras->getLx() < 2 || ras->getLy() < 2) return;

  TRaster32P ras32 = ras;
  if (!ras32) return;

  TDimension d(2, 2);
  while (d.lx < 256 && d.lx * 2 <= ras32->getLx()) d.lx *= 2;
  while (d.ly < 256 && d.ly * 2 <= ras32->getLy()) d.ly *= 2;

  TRaster32P texture;
  if (d == ras32->getSize())
    texture = ras32;
  else {
    texture = TRaster32P(d);
    TScale sc((double)texture->getLx() / ras32->getLx(),
              (double)texture->getLy() / ras32->getLy());
    TRop::resample(texture, ras32, sc);
  }

  QString name = QString::fromStdWString(fp.getLevelNameW());
  ChipData text(name, name, rasterToQImage(ras), 4, false, texture,
                TTextureStyle::staticBrushIdName(fp.getLevelNameW()));
  text.hash = TTextureStyle::generateHash(text.idname);

  m_chips.append(text);
}

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

void TextureStyleManager::loadItems() {
  m_chips.clear();
  if (getRootPath() == TFilePath()) return;

  TFilePath texturePath = getRootPath() + "textures";
  TFilePathSet fps;
  try {
    fps = TSystem::readDirectory(texturePath);
  } catch (...) {
    return;
  }
  if (fps.empty()) return;
  int count = 0;
  for (TFilePathSet::iterator it = fps.begin(); it != fps.end(); it++)
    if (TFileType::getInfo(*it) == TFileType::RASTER_IMAGE) {
      try {
        loadTexture(*it);
        ++count;
      } catch (...) {
      }
    }
  loadTexture(TFilePath());  // custom texture

  m_loaded = true;
}

//********************************************************************************
//    MyPaintBrushStyleManager  implementation
//********************************************************************************

MyPaintBrushStyleManager::MyPaintBrushStyleManager(QSize chipSize)
    : BaseStyleManager(TFilePath(), QString(), chipSize) {}

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

void MyPaintBrushStyleManager::loadItems() {
  m_brushes.clear();
  m_chips.clear();
  std::set<TFilePath> brushFiles;

  TFilePathSet dirs = TMyPaintBrushStyle::getBrushesDirs();
  for (TFilePathSet::iterator i = dirs.begin(); i != dirs.end(); ++i) {
    TFileStatus fs(*i);
    if (fs.doesExist() && fs.isDirectory()) {
      TFilePathSet files = TSystem::readDirectoryTree(*i, false, true);
      for (TFilePathSet::iterator j = files.begin(); j != files.end(); ++j)
        if (j->getType() == TMyPaintBrushStyle::getBrushType())
          brushFiles.insert(*j - *i);
    }
  }

  // reserve memory to avoid reallocation
  m_brushes.reserve(brushFiles.size());
  for (std::set<TFilePath>::iterator i = brushFiles.begin();
       i != brushFiles.end(); ++i) {
    TMyPaintBrushStyle style = TMyPaintBrushStyle(*i);
    m_brushes.push_back(style);

    // Generate a QImage preview to draw the chip faster at cost of few memory
    QImage previewQImage = rasterToQImage(style.getPreview());
    QString stylePath    = style.getPath().getQString();
    m_chips.append(ChipData(stylePath, stylePath, previewQImage, 4001, false,
                            TRasterP(), style.getBrushIdName(),
                            style.getBrushIdHash()));
  }

  m_loaded = true;
}

//********************************************************************************
//    SpecialStyleManager  implementation
//********************************************************************************

SpecialStyleManager::SpecialStyleManager(QSize chipSize)
    : BaseStyleManager(TFilePath(), QString(), chipSize) {}

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

void SpecialStyleManager::loadItems() {
  m_chips.clear();

  std::vector<int> tags;
  TColorStyle::getAllTags(tags);

  int chipCount = 0;

  for (int j = 0; j < (int)tags.size(); j++) {
    int tagId = tags[j];
    if (tagId == 3 ||     // solid color
        tagId == 4 ||     // texture
        tagId == 100 ||   // obsolete imagepattern id
        tagId == 2000 ||  // imagepattern
        tagId == 2800 ||  // imagepattern
        tagId == 2001 ||  // cleanup
        tagId == 2002 ||  // black cleanup
        tagId == 3000 ||  // vector brush
        tagId == 4001     // mypaint brush
    )
      continue;

    TColorStyle *style = TColorStyle::create(tagId);
    if (style->isRasterStyle()) {
      delete style;
      continue;
    }
    TDimension chipSize(getChipSize().width(), getChipSize().height());
    // QImage *image = new QImage(chipSize.lx, chipSize.ly,
    // QImage::Format_RGB32); convertRaster32ToImage(style->getIcon(chipSize),
    // image);
    TRaster32P raster = style->getIcon(chipSize);
    ChipData chip(style->getDescription(), style->getDescription(),
                  rasterToQImage(raster), tagId, true, raster,
                  style->getBrushIdName(), style->getBrushIdHash());
    m_chips.append(chip);
    delete style;
  }

  m_loaded = true;
}