#include "toonz/boardsettings.h"
// TnzLib includes
#include "toonz/toonzscene.h"
#include "toonz/tproject.h"
#include "toonz/sceneproperties.h"
#include "toonz/toonzfolders.h"
#include "toutputproperties.h"
#include "tsystem.h"
#include <QImage>
#include <QDate>
#include <QDateTime>
#include <QFontMetricsF>
#include <QMap>
namespace {
QMap<BoardItem::Type, std::wstring> strs = {
{BoardItem::FreeText, L"FreeText"},
{BoardItem::ProjectName, L"ProjectName"},
{BoardItem::SceneName, L"SceneName"},
{BoardItem::Duration_Frame, L"Duration_Frame"},
{BoardItem::Duration_SecFrame, L"Duration_SecFrame"},
{BoardItem::Duration_HHMMSSFF, L"Duration_HHMMSSFF"},
{BoardItem::CurrentDate, L"CurrentDate"},
{BoardItem::CurrentDateTime, L"CurrentDateTime"},
{BoardItem::UserName, L"UserName"},
{BoardItem::ScenePath_Aliased, L"ScenePath_Aliased"},
{BoardItem::ScenePath_Full, L"ScenePath_Full"},
{BoardItem::MoviePath_Aliased, L"MoviePath_Aliased"},
{BoardItem::MoviePath_Full, L"MoviePath_Full"},
{BoardItem::Image, L"Image"}};
std::wstring type2String(BoardItem::Type type) { return strs.value(type, L""); }
BoardItem::Type string2Type(std::wstring str) {
return strs.key(str, BoardItem::TypeCount);
}
}; // namespace
BoardItem::BoardItem() {
m_name = "Item";
m_type = ProjectName;
m_rect = QRectF(0.1, 0.1, 0.8, 0.8);
m_maximumFontSize = 300;
m_color = Qt::black;
}
QString BoardItem::getContentText(ToonzScene *scene) {
auto getDuration = [&]() {
TOutputProperties *oprop = scene->getProperties()->getOutputProperties();
int from, to, step;
if (oprop->getRange(from, to, step))
return to - from + 1;
else
return scene->getFrameCount();
};
switch (m_type) {
case FreeText:
return m_text;
break;
case ProjectName:
return scene->getProject()->getName().getQString();
break;
case SceneName:
return QString::fromStdWString(scene->getSceneName());
break;
case Duration_Frame:
return QString::number(getDuration());
break;
case Duration_SecFrame: {
TOutputProperties *oprop = scene->getProperties()->getOutputProperties();
int fps = (int)oprop->getFrameRate();
int frame = getDuration();
return QString("%1 + %2").arg(QString::number(frame / fps),
QString::number(frame % fps));
} break;
case Duration_HHMMSSFF: {
TOutputProperties *oprop = scene->getProperties()->getOutputProperties();
int fps = (int)oprop->getFrameRate();
int frame = getDuration();
int hh = frame / (fps * 60 * 60);
frame -= hh * fps * 60 * 60;
int mm = frame / (fps * 60);
frame -= mm * fps * 60;
int ss = frame / fps;
int ff = frame % fps;
return QString::number(hh).rightJustified(2, '0') + ":" +
QString::number(mm).rightJustified(2, '0') + ":" +
QString::number(ss).rightJustified(2, '0') + ":" +
QString::number(ff).rightJustified(2, '0');
} break;
case CurrentDate:
return QDate::currentDate().toString(Qt::DefaultLocaleLongDate);
break;
case CurrentDateTime:
return QDateTime::currentDateTime().toString(Qt::DefaultLocaleLongDate);
break;
case UserName:
return TSystem::getUserName();
break;
case ScenePath_Aliased:
return scene->codeFilePath(scene->getScenePath()).getQString();
break;
case ScenePath_Full:
return scene->decodeFilePath(scene->getScenePath()).getQString();
break;
case MoviePath_Aliased: {
TOutputProperties *oprop = scene->getProperties()->getOutputProperties();
return scene->codeFilePath(oprop->getPath()).getQString();
} break;
case MoviePath_Full: {
TOutputProperties *oprop = scene->getProperties()->getOutputProperties();
return scene->decodeFilePath(oprop->getPath()).getQString();
} break;
}
return QString();
}
QRectF BoardItem::getItemRect(QSize imgSize) {
QSizeF imgSizeF(imgSize);
return QRectF(
imgSizeF.width() * m_rect.left(), imgSizeF.height() * m_rect.top(),
imgSizeF.width() * m_rect.width(), imgSizeF.height() * m_rect.height());
}
void BoardItem::drawItem(QPainter &p, QSize imgSize, int shrink,
ToonzScene *scene) {
QRectF itemRect = getItemRect(imgSize);
if (m_type == Image) {
if (m_imgPath.isEmpty()) return;
TFilePath decodedPath = scene->decodeFilePath(m_imgPath);
QImage img(decodedPath.getQString());
if (m_imgARMode == Qt::KeepAspectRatio) {
float ratio = std::min((float)itemRect.width() / (float)img.width(),
(float)itemRect.height() / (float)img.height());
QSizeF imgSize((float)img.width() * ratio, (float)img.height() * ratio);
QPointF imgTopLeft =
itemRect.topLeft() +
QPointF((itemRect.width() - imgSize.width()) * 0.5f,
(itemRect.height() - imgSize.height()) * 0.5f);
p.drawImage(QRectF(imgTopLeft, imgSize), img);
} else if (m_imgARMode == Qt::IgnoreAspectRatio)
p.drawImage(itemRect, img);
return;
}
QString contentText = getContentText(scene);
QFont tmpFont(m_font);
tmpFont.setPixelSize(100);
QFontMetricsF tmpFm(tmpFont);
QRectF tmpBounding =
tmpFm.boundingRect(itemRect, Qt::AlignLeft | Qt::AlignTop, contentText);
float ratio = std::min(itemRect.width() / tmpBounding.width(),
itemRect.height() / tmpBounding.height());
// compute the font size which will just fit the item region
int fontSize = (int)(100.0f * ratio);
tmpFont.setPixelSize(fontSize);
tmpFm = QFontMetricsF(tmpFont);
tmpBounding =
tmpFm.boundingRect(itemRect, Qt::AlignLeft | Qt::AlignTop, contentText);
bool isInRect;
if (itemRect.width() >= tmpBounding.width() &&
itemRect.height() >= tmpBounding.height())
isInRect = true;
else
isInRect = false;
while (1) {
fontSize += (isInRect) ? 1 : -1;
if (fontSize <= 0) // cannot draw
return;
tmpFont.setPixelSize(fontSize);
tmpFm = QFontMetricsF(tmpFont);
tmpBounding =
tmpFm.boundingRect(itemRect, Qt::AlignLeft | Qt::AlignTop, contentText);
bool newIsInRect = (itemRect.width() >= tmpBounding.width() &&
itemRect.height() >= tmpBounding.height());
if (isInRect != newIsInRect) {
if (isInRect) fontSize--;
break;
}
}
//----
fontSize = std::min(fontSize, m_maximumFontSize / shrink);
QFont font(m_font);
font.setPixelSize(fontSize);
p.setFont(font);
p.setPen(m_color);
if (m_type == FreeText)
p.drawText(itemRect, Qt::AlignLeft | Qt::AlignTop, contentText);
else
p.drawText(itemRect, Qt::AlignCenter, contentText);
}
void BoardItem::saveData(TOStream &os) {
os.child("type") << type2String(m_type);
os.child("name") << m_name;
os.child("rect") << m_rect.x() << m_rect.y() << m_rect.width()
<< m_rect.height();
if (m_type == Image) {
// if the path is in library folder, then save the relative path
TFilePath libFp = ToonzFolder::getLibraryFolder();
if (libFp.isAncestorOf(m_imgPath))
os.child("imgPath") << 1 << m_imgPath - libFp;
else
os.child("imgPath") << 0 << m_imgPath;
os.child("imgARMode") << (int)m_imgARMode;
} else {
if (m_type == FreeText) os.child("text") << m_text;
os.child("maximumFontSize") << m_maximumFontSize;
os.child("color") << m_color.red() << m_color.green() << m_color.blue()
<< m_color.alpha();
os.child("font") << m_font.family() << (int)(m_font.bold() ? 1 : 0)
<< (int)(m_font.italic() ? 1 : 0);
}
}
void BoardItem::loadData(TIStream &is) {
std::string tagName;
while (is.matchTag(tagName)) {
if (tagName == "type") {
std::wstring typeStr;
is >> typeStr;
m_type = string2Type(typeStr);
} else if (tagName == "name") {
std::wstring str;
is >> str;
m_name = QString::fromStdWString(str);
} else if (tagName == "rect") {
double x, y, width, height;
is >> x >> y >> width >> height;
m_rect = QRectF(x, y, width, height);
} else if (tagName == "imgPath") {
int isInLibrary;
TFilePath fp;
is >> isInLibrary >> fp;
if (isInLibrary == 1)
m_imgPath = ToonzFolder::getLibraryFolder() + fp;
else
m_imgPath = fp;
} else if (tagName == "imgARMode") {
int mode;
is >> mode;
m_imgARMode = (Qt::AspectRatioMode)mode;
} else if (tagName == "text") {
std::wstring str;
is >> str;
m_text = QString::fromStdWString(str);
} else if (tagName == "maximumFontSize") {
is >> m_maximumFontSize;
} else if (tagName == "color") {
int r, g, b, a;
is >> r >> g >> b >> a;
m_color = QColor(r, g, b, a);
} else if (tagName == "font") {
QString family;
int isBold, isItalic;
is >> family >> isBold >> isItalic;
m_font.setFamily(family);
m_font.setBold(isBold == 1);
m_font.setItalic(isItalic == 1);
} else
throw TException("unexpected tag: " + tagName);
is.closeChild();
}
}
//======================================================================================
BoardSettings::BoardSettings() {
// add one item as an example
m_items.push_back(BoardItem());
}
QImage BoardSettings::getBoardImage(TDimension &dim, int shrink,
ToonzScene *scene) {
QImage img(dim.lx, dim.ly, QImage::Format_ARGB32);
QPainter painter(&img);
painter.fillRect(img.rect(), Qt::white);
// draw each item
for (int i = m_items.size() - 1; i >= 0; i--)
m_items[i].drawItem(painter, img.rect().size(), shrink, scene);
painter.end();
return img;
}
TRaster32P BoardSettings::getBoardRaster(TDimension &dim, int shrink,
ToonzScene *scene) {
QImage img = getBoardImage(dim, shrink, scene);
// convert QImage to TRaster
TRaster32P boardRas(dim);
int img_y = img.height() - 1;
for (int j = 0; j < dim.ly; j++, img_y--) {
TPixel32 *pix = boardRas->pixels(j);
QRgb *img_p = (QRgb *)img.scanLine(img_y);
for (int i = 0; i < dim.lx; i++, pix++, img_p++) {
(*pix).r = (typename TPixel32::Channel)(qRed(*img_p));
(*pix).g = (typename TPixel32::Channel)(qGreen(*img_p));
(*pix).b = (typename TPixel32::Channel)(qBlue(*img_p));
(*pix).m = (typename TPixel32::Channel)(qAlpha(*img_p));
}
}
return boardRas;
}
void BoardSettings::addNewItem(int insertAt) {
m_items.insert(insertAt, BoardItem());
}
void BoardSettings::removeItem(int index) {
if (index < 0 || index >= m_items.size()) return;
m_items.removeAt(index);
}
void BoardSettings::swapItems(int i, int j) { m_items.swap(i, j); }
void BoardSettings::saveData(TOStream &os, bool forPreset) {
if (!forPreset) os.child("active") << (int)((m_active) ? 1 : 0);
os.child("duration") << m_duration;
if (!m_items.isEmpty()) {
os.openChild("boardItems");
for (int i = 0; i < getItemCount(); i++) {
os.openChild("item");
m_items[i].saveData(os);
os.closeChild();
}
os.closeChild();
}
}
void BoardSettings::loadData(TIStream &is) {
std::string tagName;
while (is.matchTag(tagName)) {
if (tagName == "active") {
int val;
is >> val;
setActive(val == 1);
} else if (tagName == "duration") {
is >> m_duration;
} else if (tagName == "boardItems") {
m_items.clear();
while (is.matchTag(tagName)) {
if (tagName == "item") {
BoardItem item;
item.loadData(is);
m_items.append(item);
} else
throw TException("unexpected tag: " + tagName);
is.closeChild();
}
} else
throw TException("unexpected tag: " + tagName);
is.closeChild();
}
}