valentina/src/app/puzzle/vptilefactory.cpp

801 lines
29 KiB
C++
Raw Normal View History

#include "vptilefactory.h"
#include <QtSvg>
2023-06-21 09:24:51 +02:00
#include "../vlayout/vprintlayout.h"
#include "../vmisc/compatibility.h"
2021-08-09 14:09:10 +02:00
#include "../vmisc/def.h"
2022-02-18 09:36:38 +01:00
#include "../vmisc/literals.h"
2021-08-09 14:09:10 +02:00
#include "../vmisc/vcommonsettings.h"
2023-06-21 09:24:51 +02:00
#include "../vwidgets/vmaingraphicsscene.h"
2023-07-15 12:47:14 +02:00
#include "layout/vplayout.h"
2023-06-21 09:24:51 +02:00
#include "layout/vpsheet.h"
#include "theme/vtheme.h"
2023-06-21 09:24:51 +02:00
using namespace Qt::Literals::StringLiterals;
2021-09-08 09:41:35 +02:00
namespace
{
2023-06-21 09:24:51 +02:00
QT_WARNING_PUSH
QT_WARNING_DISABLE_CLANG("-Wunused-member-function")
Q_GLOBAL_STATIC_WITH_ARGS(QColor, tileColor, (180, 180, 180)) // NOLINT
Q_GLOBAL_STATIC_WITH_ARGS(QBrush, triangleBush, (QColor(200, 200, 200))) // NOLINT
QT_WARNING_POP
2021-09-11 18:39:38 +02:00
//---------------------------------------------------------------------------------------------------------------------
auto Grayscale(QImage image) -> QImage
{
for (int ii = 0; ii < image.height(); ii++)
{
2023-06-21 09:24:51 +02:00
uchar *scan = image.scanLine(ii);
2024-02-19 17:09:56 +01:00
int const depth = 4;
2021-09-11 18:39:38 +02:00
for (int jj = 0; jj < image.width(); jj++)
{
2023-06-21 09:24:51 +02:00
auto *rgbpixel = reinterpret_cast<QRgb *>(scan + jj * depth); // NOLINT
2024-02-19 17:09:56 +01:00
int const gray = qGray(*rgbpixel);
2021-09-11 18:39:38 +02:00
*rgbpixel = QColor(gray, gray, gray, qAlpha(*rgbpixel)).rgba();
}
}
return image;
}
//---------------------------------------------------------------------------------------------------------------------
auto WatermarkImageFromCache(const VWatermarkData &watermarkData, const QString &watermarkPath, qreal xScale,
qreal yScale, QString &error) -> QPixmap
{
QPixmap pixmap;
2024-02-19 17:09:56 +01:00
QString const imagePath = AbsoluteMPath(watermarkPath, watermarkData.path);
2021-09-11 18:39:38 +02:00
if (QString const imageCacheKey = QStringLiteral("puzzle=path%1+rotation%3+grayscale%4+xscale%5+yxcale%6")
.arg(imagePath, QString::number(watermarkData.imageRotation),
watermarkData.grayscale ? trueStr : falseStr)
.arg(xScale)
.arg(yScale);
not QPixmapCache::find(imageCacheKey, &pixmap))
2021-09-11 18:39:38 +02:00
{
QImageReader imageReader(imagePath);
QImage watermark = imageReader.read();
if (watermark.isNull())
{
error = imageReader.errorString();
return pixmap;
}
if (watermarkData.grayscale)
{
watermark = Grayscale(watermark);
}
QTransform t;
t.scale(1 / xScale, 1 / yScale);
watermark = watermark.transformed(t);
2021-09-11 18:39:38 +02:00
t = QTransform();
2023-06-21 09:24:51 +02:00
t.translate(watermark.width() / 2., watermark.height() / 2.);
t.rotate(-watermarkData.imageRotation);
2023-06-21 09:24:51 +02:00
t.translate(-watermark.width() / 2., -watermark.height() / 2.);
watermark = watermark.transformed(t);
2021-09-11 18:39:38 +02:00
pixmap = QPixmap::fromImage(watermark);
2021-09-11 18:39:38 +02:00
QPixmapCache::insert(imageCacheKey, pixmap);
}
return pixmap;
2021-09-08 09:41:35 +02:00
}
2022-02-12 11:01:07 +01:00
//---------------------------------------------------------------------------------------------------------------------
// ------------- prepare triangles for position marks
auto TriangleBasic() -> QPainterPath
{
// ------------- prepare triangles for position marks
2024-02-19 17:09:56 +01:00
QRectF const rectBasic = QRectF(-UnitConvertor(0.5, Unit::Cm, Unit::Px), 0, UnitConvertor(1, Unit::Cm, Unit::Px),
UnitConvertor(0.5, Unit::Cm, Unit::Px));
2022-02-12 11:01:07 +01:00
QPainterPath triangleBasic;
triangleBasic.moveTo(rectBasic.topLeft());
triangleBasic.lineTo(rectBasic.topRight());
triangleBasic.lineTo(rectBasic.left() + (rectBasic.width() / 2.), rectBasic.bottom());
triangleBasic.lineTo(rectBasic.topLeft());
return triangleBasic;
}
2023-06-21 09:24:51 +02:00
} // namespace
2021-09-08 09:41:35 +02:00
//---------------------------------------------------------------------------------------------------------------------
2023-06-21 09:24:51 +02:00
VPTileFactory::VPTileFactory(const VPLayoutPtr &layout, VCommonSettings *commonSettings, QObject *parent)
: QObject(parent),
m_layout(layout),
2021-09-07 19:26:35 +02:00
m_commonSettings(commonSettings)
{
}
//---------------------------------------------------------------------------------------------------------------------
void VPTileFactory::RefreshTileInfos()
{
2024-02-19 17:09:56 +01:00
VPLayoutPtr const layout = m_layout.toStrongRef();
2023-06-21 09:24:51 +02:00
if (not layout.isNull())
{
2024-02-19 17:09:56 +01:00
QSizeF const tilesSize = layout->LayoutSettings().GetTilesSize();
QMarginsF const tilesMargins = layout->LayoutSettings().GetTilesMargins();
2020-11-20 15:51:24 +01:00
// sets the drawing height
2021-08-25 15:58:50 +02:00
m_drawingAreaHeight = tilesSize.height();
2020-11-20 15:51:24 +01:00
2021-08-25 15:58:50 +02:00
if (not layout->LayoutSettings().IgnoreTilesMargins())
{
2021-09-07 19:26:35 +02:00
m_drawingAreaHeight -= tilesMargins.top() + tilesMargins.bottom();
2021-08-25 15:58:50 +02:00
}
2020-11-20 15:51:24 +01:00
2021-08-25 15:58:50 +02:00
// sets the drawing width
m_drawingAreaWidth = tilesSize.width();
2020-11-20 15:51:24 +01:00
2021-08-25 15:58:50 +02:00
if (not layout->LayoutSettings().IgnoreTilesMargins())
2020-11-20 15:51:24 +01:00
{
2021-09-07 19:26:35 +02:00
m_drawingAreaWidth -= tilesMargins.left() + tilesMargins.right();
2020-11-20 15:51:24 +01:00
}
}
}
2021-09-11 18:39:38 +02:00
//---------------------------------------------------------------------------------------------------------------------
void VPTileFactory::RefreshWatermarkData()
{
2024-02-19 17:09:56 +01:00
VPLayoutPtr const layout = m_layout.toStrongRef();
2023-06-21 09:24:51 +02:00
if (not layout.isNull())
{
2021-09-11 18:39:38 +02:00
m_watermarkData = layout->WatermarkData();
}
else
{
m_watermarkData = VWatermarkData();
}
}
//---------------------------------------------------------------------------------------------------------------------
2021-09-06 14:31:19 +02:00
void VPTileFactory::drawTile(QPainter *painter, QPrinter *printer, const VPSheetPtr &sheet, int row, int col)
{
2021-09-06 14:31:19 +02:00
SCASSERT(painter != nullptr)
SCASSERT(printer != nullptr)
if (sheet.isNull())
{
return;
}
const int nbCol = ColNb(sheet);
const int nbRow = RowNb(sheet);
if (row < 0 || row > nbRow || col < 0 || col > nbCol)
{
return;
2021-08-17 17:49:28 +02:00
}
2021-09-06 14:31:19 +02:00
2022-02-12 11:01:07 +01:00
DrawTilePageContent(painter, sheet, row, col, printer);
// add the tiles decorations (cutting and gluing lines, scissors, infos etc.)
2022-02-12 11:01:07 +01:00
painter->setPen(PenTileInfos());
2022-08-18 16:10:42 +02:00
painter->save();
const QPair<qreal, qreal> scale = VPrintLayout::PrinterScaleDiff(printer);
painter->scale(scale.first, scale.second);
2023-06-21 09:24:51 +02:00
if (row > 0)
{
// add top triangle
2022-02-12 11:01:07 +01:00
DrawTopTriangle(painter);
// scissors along the top line
DrawTopLineScissors(painter);
// dashed top line (for cutting)
2022-02-12 11:01:07 +01:00
DrawTopCuttingLine(painter);
}
else
{
// solid top line stopping at the edge
2022-02-12 11:01:07 +01:00
DrawSolidTopLine(painter, col, nbCol);
}
2023-06-21 09:24:51 +02:00
if (col > 0)
{
// add left triangle
2022-02-12 11:01:07 +01:00
DrawLeftTriangle(painter);
// scissors along the left line
DrawLeftLineScissors(painter);
// dashed left line (for cutting)
2022-02-12 11:01:07 +01:00
DrawLeftCuttingLine(painter);
}
else
{
// solid left line at the edge
2022-02-12 11:01:07 +01:00
DrawSolidLeftLine(painter, row, nbRow);
}
2023-06-21 09:24:51 +02:00
if (row < nbRow - 1)
{
// add bottom triangle
2022-02-12 11:01:07 +01:00
DrawBottomTriangle(painter);
// dotted bottom line (for glueing)
2022-02-12 11:01:07 +01:00
DrawBottomCuttingLine(painter);
2021-09-06 14:31:19 +02:00
}
else
{
// solid bottom line at the edge
2022-02-12 11:01:07 +01:00
DrawSolidBottomLine(painter, col, nbCol);
}
2022-08-18 16:10:42 +02:00
DrawRuler(painter, scale.first);
2021-09-11 18:39:38 +02:00
DrawWatermark(painter);
2021-09-08 09:41:35 +02:00
2023-06-21 09:24:51 +02:00
if (col < nbCol - 1)
{
// add right triangle
2022-02-12 11:01:07 +01:00
DrawRightTriangle(painter);
// dotted right line (for glueing)
2022-02-12 11:01:07 +01:00
DrawRightCuttingLine(painter);
}
else
{
// solid right line at the edge
2022-02-12 11:01:07 +01:00
DrawSolidRightLine(painter, row, nbRow);
}
// prepare the painting for the text information
2022-02-12 11:01:07 +01:00
DrawTextInformation(painter, row, col, nbRow, nbCol, sheet->GetName());
2022-08-18 16:10:42 +02:00
painter->restore();
}
//---------------------------------------------------------------------------------------------------------------------
2021-09-06 14:31:19 +02:00
auto VPTileFactory::RowNb(const VPSheetPtr &sheet) const -> int
{
2021-09-06 14:31:19 +02:00
if (sheet.isNull())
{
return 0;
}
2021-09-06 15:56:56 +02:00
qreal yScale = 1;
if (VPLayoutPtr const layout = m_layout.toStrongRef(); not layout.isNull())
2021-09-06 15:56:56 +02:00
{
yScale = layout->LayoutSettings().VerticalScale();
}
2024-02-19 17:09:56 +01:00
QRectF const sheetSize = sheet->GetMarginsRect();
2021-09-07 19:26:35 +02:00
return qCeil(sheetSize.height() * yScale / (m_drawingAreaHeight - tileStripeWidth));
}
//---------------------------------------------------------------------------------------------------------------------
2021-09-06 14:31:19 +02:00
auto VPTileFactory::ColNb(const VPSheetPtr &sheet) const -> int
{
2021-09-06 14:31:19 +02:00
if (sheet.isNull())
{
return 0;
}
2021-09-06 15:56:56 +02:00
qreal xScale = 1;
if (VPLayoutPtr const layout = m_layout.toStrongRef(); not layout.isNull())
2021-09-06 15:56:56 +02:00
{
xScale = layout->LayoutSettings().HorizontalScale();
}
2024-02-19 17:09:56 +01:00
QRectF const sheetSize = sheet->GetMarginsRect();
2021-09-07 19:26:35 +02:00
return qCeil(sheetSize.width() * xScale / (m_drawingAreaWidth - tileStripeWidth));
}
//---------------------------------------------------------------------------------------------------------------------
2021-09-06 14:31:19 +02:00
auto VPTileFactory::DrawingAreaHeight() const -> qreal
{
return m_drawingAreaHeight;
}
//---------------------------------------------------------------------------------------------------------------------
2021-09-06 14:31:19 +02:00
auto VPTileFactory::DrawingAreaWidth() const -> qreal
{
return m_drawingAreaWidth;
}
2021-09-08 09:41:35 +02:00
2021-09-11 18:39:38 +02:00
//---------------------------------------------------------------------------------------------------------------------
auto VPTileFactory::WatermarkData() const -> const VWatermarkData &
{
return m_watermarkData;
}
2021-09-08 09:41:35 +02:00
//---------------------------------------------------------------------------------------------------------------------
2022-08-18 16:10:42 +02:00
void VPTileFactory::DrawRuler(QPainter *painter, qreal scale) const
2021-09-08 09:41:35 +02:00
{
2024-02-19 17:09:56 +01:00
VPLayoutPtr const layout = m_layout.toStrongRef();
2023-06-21 09:24:51 +02:00
if (layout.isNull())
2021-09-08 09:41:35 +02:00
{
return;
}
2024-02-19 17:09:56 +01:00
QPen const rulePen(*tileColor, 1, Qt::SolidLine);
2021-09-08 09:41:35 +02:00
painter->save();
painter->setPen(rulePen);
const qreal notchHeight = UnitConvertor(3, Unit::Mm, Unit::Px);
const qreal shortNotchHeight = UnitConvertor(1.1, Unit::Mm, Unit::Px);
2024-02-19 17:09:56 +01:00
Unit const layoutUnits = layout->LayoutSettings().GetUnit();
Unit const rulerUnits = layoutUnits == Unit::Inch ? layoutUnits : Unit::Cm;
const qreal step = UnitConvertor(1, rulerUnits, Unit::Px);
2024-02-19 17:09:56 +01:00
double const marksCount = (m_drawingAreaWidth - tileStripeWidth) / step;
2021-09-08 09:41:35 +02:00
int i = 0;
while (i < marksCount)
{
if (i != 0)
{ // don't need 0 notch
// middle ruler line
2023-06-21 09:24:51 +02:00
painter->drawLine(QPointF(step * i - step / 2., m_drawingAreaHeight - tileStripeWidth),
2021-09-08 09:41:35 +02:00
QPointF(step * i - step / 2., m_drawingAreaHeight - tileStripeWidth + shortNotchHeight));
// ruler line
2023-06-21 09:24:51 +02:00
painter->drawLine(QPointF(step * i, m_drawingAreaHeight - tileStripeWidth),
2021-09-08 09:41:35 +02:00
QPointF(step * i, m_drawingAreaHeight - tileStripeWidth + notchHeight));
}
else
{
2022-08-18 16:10:42 +02:00
painter->save();
2021-09-08 09:41:35 +02:00
QFont fnt = painter->font();
2023-06-21 09:24:51 +02:00
const int size = qRound(10 / scale);
2022-08-18 16:10:42 +02:00
size > 0 ? fnt.setPointSize(size) : fnt.setPointSize(10);
painter->setFont(fnt);
2021-09-08 09:41:35 +02:00
qreal unitsWidth = 0;
2024-02-19 17:09:56 +01:00
QFontMetrics const fm(fnt);
QString const units = rulerUnits != Unit::Inch ? tr("cm", "unit") : tr("in", "unit");
unitsWidth = fm.horizontalAdvance(units);
2023-06-21 09:24:51 +02:00
painter->drawText(QPointF(step * 0.5 - unitsWidth * 0.6,
m_drawingAreaHeight - tileStripeWidth + notchHeight + shortNotchHeight),
units);
2022-08-18 16:10:42 +02:00
painter->restore();
2021-09-08 09:41:35 +02:00
}
++i;
}
painter->restore();
}
2021-09-11 18:39:38 +02:00
//---------------------------------------------------------------------------------------------------------------------
2022-02-12 11:01:07 +01:00
void VPTileFactory::DrawWatermark(QPainter *painter) const
2021-09-11 18:39:38 +02:00
{
SCASSERT(painter != nullptr)
2024-02-19 17:09:56 +01:00
VPLayoutPtr const layout = m_layout.toStrongRef();
2023-06-21 09:24:51 +02:00
if (layout.isNull())
2021-09-11 18:39:38 +02:00
{
return;
}
if (m_watermarkData.opacity > 0)
{
2024-02-19 17:09:56 +01:00
QRectF const img(0, 0, m_drawingAreaWidth - tileStripeWidth, m_drawingAreaHeight - tileStripeWidth);
2021-09-11 18:39:38 +02:00
if (m_watermarkData.showImage && not m_watermarkData.path.isEmpty())
{
PaintWatermarkImage(painter, img, m_watermarkData, layout->LayoutSettings().WatermarkPath(), false);
2021-09-11 18:39:38 +02:00
}
if (m_watermarkData.showText && not m_watermarkData.text.isEmpty())
{
PaintWatermarkText(painter, img, m_watermarkData);
2021-09-11 18:39:38 +02:00
}
}
}
2022-02-12 11:01:07 +01:00
//---------------------------------------------------------------------------------------------------------------------
inline auto VPTileFactory::PenTileInfos() const -> QPen
{
return {*tileColor, m_commonSettings->WidthHairLine(), Qt::DashLine, Qt::RoundCap, Qt::RoundJoin};
}
//---------------------------------------------------------------------------------------------------------------------
void VPTileFactory::DrawTilePageContent(QPainter *painter, const VPSheetPtr &sheet, int row, int col,
QPrinter *printer) const
{
2024-02-19 17:09:56 +01:00
VPLayoutPtr const layout = m_layout.toStrongRef();
2023-06-21 09:24:51 +02:00
if (layout.isNull())
2022-02-12 11:01:07 +01:00
{
return;
}
QMarginsF sheetMargins;
if (not sheet->IgnoreMargins())
{
sheetMargins = sheet->GetSheetMargins();
}
2024-02-19 17:09:56 +01:00
QPen const penTileDrawing =
2023-06-21 09:24:51 +02:00
QPen(Qt::black, m_commonSettings->WidthMainLine(), Qt::SolidLine, Qt::RoundCap, Qt::RoundJoin);
2022-02-12 11:01:07 +01:00
painter->setPen(penTileDrawing);
// paint the content of the page
const qreal xScale = layout->LayoutSettings().HorizontalScale();
const qreal yScale = layout->LayoutSettings().VerticalScale();
2024-02-19 17:09:56 +01:00
QRectF const source = QRectF(sheetMargins.left() + col * (m_drawingAreaWidth - tileStripeWidth) / xScale,
sheetMargins.top() + row * (m_drawingAreaHeight - tileStripeWidth) / yScale,
m_drawingAreaWidth / xScale, m_drawingAreaHeight / yScale);
2023-06-21 09:24:51 +02:00
2024-02-19 17:09:56 +01:00
QRectF const target = QRectF(0, 0, m_drawingAreaWidth, m_drawingAreaHeight);
2022-02-12 11:01:07 +01:00
sheet->SceneData()->Scene()->render(painter, VPrintLayout::SceneTargetRect(printer, target), source,
Qt::IgnoreAspectRatio);
}
//---------------------------------------------------------------------------------------------------------------------
void VPTileFactory::DrawTopTriangle(QPainter *painter) const
{
2024-02-19 17:09:56 +01:00
QPainterPath const triangleTop = QTransform().translate(m_drawingAreaWidth / 2, 0).map(TriangleBasic());
2022-02-12 11:01:07 +01:00
painter->fillPath(triangleTop, *triangleBush);
}
//---------------------------------------------------------------------------------------------------------------------
void VPTileFactory::DrawLeftTriangle(QPainter *painter) const
{
2024-02-19 17:09:56 +01:00
QPainterPath const triangleLeft =
QTransform().translate(0, m_drawingAreaHeight / 2).rotate(-90).map(TriangleBasic());
2022-02-12 11:01:07 +01:00
painter->fillPath(triangleLeft, *triangleBush);
}
//---------------------------------------------------------------------------------------------------------------------
void VPTileFactory::DrawBottomTriangle(QPainter *painter) const
{
2024-02-19 17:09:56 +01:00
QPainterPath const triangleBottom = QTransform()
.translate(m_drawingAreaWidth / 2, m_drawingAreaHeight - tileStripeWidth)
.rotate(180)
.map(TriangleBasic());
2022-02-12 11:01:07 +01:00
painter->fillPath(triangleBottom, *triangleBush);
}
//---------------------------------------------------------------------------------------------------------------------
void VPTileFactory::DrawRightTriangle(QPainter *painter) const
{
2024-02-19 17:09:56 +01:00
QPainterPath const triangleRight = QTransform()
.translate(m_drawingAreaWidth - tileStripeWidth, m_drawingAreaHeight / 2)
.rotate(90)
.map(TriangleBasic());
2022-02-12 11:01:07 +01:00
painter->fillPath(triangleRight, *triangleBush);
}
//---------------------------------------------------------------------------------------------------------------------
void VPTileFactory::DrawTopLineScissors(QPainter *painter) const
{
QSvgRenderer svgRenderer(QStringLiteral("://puzzleicon/svg/icon_scissors_horizontal.svg"));
2023-06-21 09:24:51 +02:00
svgRenderer.render(painter, QRectF(m_drawingAreaWidth - tileStripeWidth, 0, UnitConvertor(0.95, Unit::Cm, Unit::Px),
2022-02-12 11:01:07 +01:00
UnitConvertor(0.56, Unit::Cm, Unit::Px)));
}
//---------------------------------------------------------------------------------------------------------------------
void VPTileFactory::DrawLeftLineScissors(QPainter *painter) const
{
QSvgRenderer svgRenderer(QStringLiteral("://puzzleicon/svg/icon_scissors_vertical.svg"));
2023-06-21 09:24:51 +02:00
svgRenderer.render(painter,
QRectF(0, m_drawingAreaHeight - tileStripeWidth, UnitConvertor(0.56, Unit::Cm, Unit::Px),
UnitConvertor(0.95, Unit::Cm, Unit::Px)));
2022-02-12 11:01:07 +01:00
}
//---------------------------------------------------------------------------------------------------------------------
void VPTileFactory::DrawTopCuttingLine(QPainter *painter) const
{
QPen penTileInfos = PenTileInfos();
penTileInfos.setStyle(Qt::DashLine);
painter->setPen(penTileInfos);
painter->drawLine(QPointF(), QPointF(m_drawingAreaWidth, 0));
}
//---------------------------------------------------------------------------------------------------------------------
void VPTileFactory::DrawLeftCuttingLine(QPainter *painter) const
{
QPen penTileInfos = PenTileInfos();
penTileInfos.setStyle(Qt::DashLine);
painter->setPen(penTileInfos);
painter->drawLine(QPointF(), QPointF(0, m_drawingAreaHeight));
}
//---------------------------------------------------------------------------------------------------------------------
void VPTileFactory::DrawBottomCuttingLine(QPainter *painter) const
{
QPen penTileInfos = PenTileInfos();
penTileInfos.setStyle(Qt::DotLine);
painter->setPen(penTileInfos);
painter->drawLine(QPointF(0, m_drawingAreaHeight - tileStripeWidth),
QPointF(m_drawingAreaWidth, m_drawingAreaHeight - tileStripeWidth));
}
//---------------------------------------------------------------------------------------------------------------------
void VPTileFactory::DrawRightCuttingLine(QPainter *painter) const
{
QPen penTileInfos = PenTileInfos();
penTileInfos.setStyle(Qt::DotLine);
painter->setPen(penTileInfos);
painter->drawLine(QPointF(m_drawingAreaWidth - tileStripeWidth, 0),
QPointF(m_drawingAreaWidth - tileStripeWidth, m_drawingAreaHeight));
}
//---------------------------------------------------------------------------------------------------------------------
void VPTileFactory::DrawSolidTopLine(QPainter *painter, int col, int nbCol) const
{
QPen penTileInfos = PenTileInfos();
penTileInfos.setStyle(Qt::SolidLine);
painter->setPen(penTileInfos);
2023-06-21 09:24:51 +02:00
if (col < nbCol - 1)
2022-02-12 11:01:07 +01:00
{
painter->drawLine(QPointF(), QPointF(m_drawingAreaWidth, 0));
}
else
{
painter->drawLine(QPointF(), QPointF(m_drawingAreaWidth - tileStripeWidth, 0));
}
}
//---------------------------------------------------------------------------------------------------------------------
void VPTileFactory::DrawSolidLeftLine(QPainter *painter, int row, int nbRow) const
{
QPen penTileInfos = PenTileInfos();
penTileInfos.setStyle(Qt::SolidLine);
painter->setPen(penTileInfos);
2023-06-21 09:24:51 +02:00
if (row < nbRow - 1)
2022-02-12 11:01:07 +01:00
{
painter->drawLine(QPointF(), QPointF(0, m_drawingAreaHeight));
}
else
{
painter->drawLine(QPointF(), QPointF(0, m_drawingAreaHeight - tileStripeWidth));
}
}
//---------------------------------------------------------------------------------------------------------------------
void VPTileFactory::DrawSolidBottomLine(QPainter *painter, int col, int nbCol) const
{
QPen penTileInfos = PenTileInfos();
penTileInfos.setStyle(Qt::SolidLine);
painter->setPen(penTileInfos);
2023-06-21 09:24:51 +02:00
if (col < nbCol - 1)
2022-02-12 11:01:07 +01:00
{
painter->drawLine(QPointF(0, m_drawingAreaHeight - tileStripeWidth),
QPointF(m_drawingAreaWidth, m_drawingAreaHeight - tileStripeWidth));
}
else
{
painter->drawLine(QPointF(0, m_drawingAreaHeight - tileStripeWidth),
2023-06-21 09:24:51 +02:00
QPointF(m_drawingAreaWidth - tileStripeWidth, m_drawingAreaHeight - tileStripeWidth));
;
2022-02-12 11:01:07 +01:00
}
}
//---------------------------------------------------------------------------------------------------------------------
void VPTileFactory::DrawSolidRightLine(QPainter *painter, int row, int nbRow) const
{
QPen penTileInfos = PenTileInfos();
penTileInfos.setStyle(Qt::SolidLine);
painter->setPen(penTileInfos);
2023-06-21 09:24:51 +02:00
if (row < nbRow - 1)
2022-02-12 11:01:07 +01:00
{
painter->drawLine(QPointF(m_drawingAreaWidth - tileStripeWidth, 0),
QPointF(m_drawingAreaWidth - tileStripeWidth, m_drawingAreaHeight));
}
else
{
painter->drawLine(QPointF(m_drawingAreaWidth - tileStripeWidth, 0),
QPointF(m_drawingAreaWidth - tileStripeWidth, m_drawingAreaHeight - tileStripeWidth));
}
}
//---------------------------------------------------------------------------------------------------------------------
void VPTileFactory::DrawTextInformation(QPainter *painter, int row, int col, int nbRow, int nbCol,
const QString &sheetName) const
{
QTextDocument td;
td.setPageSize(QSizeF(m_drawingAreaWidth - UnitConvertor(2, Unit::Cm, Unit::Px), m_drawingAreaHeight));
// paint the grid information
const QString grid = tr("Grid ( %1 , %2 )").arg(row + 1).arg(col + 1);
2023-06-21 09:24:51 +02:00
const QString tileColorStr =
QStringLiteral("%1,%2,%3").arg(tileColor->red()).arg(tileColor->green()).arg(tileColor->blue());
2022-02-12 11:01:07 +01:00
td.setHtml(u"<table width='100%' style='color:rgb(%1);'>"
2023-10-09 11:45:34 +02:00
u"<tr>"
u"<td align='center'>%2</td>"
u"</tr>"
u"</table>"_s.arg(tileColorStr, grid));
2022-02-12 11:01:07 +01:00
painter->setPen(PenTileInfos());
painter->save();
2023-06-21 09:24:51 +02:00
painter->translate(QPointF(UnitConvertor(1, Unit::Cm, Unit::Px), m_drawingAreaHeight - tileStripeWidth / 1.3));
2022-02-12 11:01:07 +01:00
td.drawContents(painter);
painter->restore();
// paint the page information
const QString page = tr("Page %1 of %2").arg(row * nbCol + col + 1).arg(nbCol * nbRow);
td.setPageSize(QSizeF(m_drawingAreaHeight - UnitConvertor(2, Unit::Cm, Unit::Px), m_drawingAreaWidth));
2024-02-19 17:09:56 +01:00
QFontMetrics const metrix = QFontMetrics(td.defaultFont());
int const maxWidth = metrix.horizontalAdvance(QString().fill('z', 50));
QString const clippedSheetName = metrix.elidedText(sheetName, Qt::ElideMiddle, maxWidth);
2022-02-12 11:01:07 +01:00
td.setHtml(QStringLiteral("<table width='100%' style='color:rgb(%1);'>"
"<tr>"
"<td align='center'>%2 - %3</td>"
"</tr>"
2023-06-21 09:24:51 +02:00
"</table>")
.arg(tileColorStr, page, clippedSheetName));
2022-02-12 11:01:07 +01:00
painter->save();
painter->rotate(-90);
2023-06-21 09:24:51 +02:00
painter->translate(
QPointF(-(m_drawingAreaHeight) + UnitConvertor(1, Unit::Cm, Unit::Px), m_drawingAreaWidth - tileStripeWidth));
2022-02-12 11:01:07 +01:00
td.drawContents(painter);
painter->restore();
}
2021-09-11 18:39:38 +02:00
//---------------------------------------------------------------------------------------------------------------------
void VPTileFactory::PaintWatermarkText(QPainter *painter, const QRectF &img, const VWatermarkData &watermarkData,
qreal xScale, qreal yScale)
{
SCASSERT(painter != nullptr)
painter->save();
2023-06-21 09:24:51 +02:00
painter->setOpacity(watermarkData.opacity / 100.);
2021-09-11 18:39:38 +02:00
QPen pen = painter->pen();
pen.setWidth(1);
pen.setColor(watermarkData.textColor);
pen.setStyle(Qt::SolidLine);
painter->setPen(pen);
painter->setBrush(watermarkData.textColor);
QTransform t;
t.translate(img.center().x(), img.center().y());
t.rotate(-watermarkData.textRotation);
t.translate(-img.center().x(), -img.center().y());
t.scale(xScale, yScale);
QPainterPath text;
text.addText(img.center(), watermarkData.font, watermarkData.text);
text = t.map(text);
2024-02-19 17:09:56 +01:00
QPointF const center = img.center() - text.boundingRect().center();
2021-09-11 18:39:38 +02:00
t = QTransform();
t.translate(center.x(), center.y());
text = t.map(text);
painter->drawPath(text);
painter->restore();
}
//---------------------------------------------------------------------------------------------------------------------
void VPTileFactory::PaintWatermarkImage(QPainter *painter, const QRectF &img, const VWatermarkData &watermarkData,
const QString &watermarkPath, bool folowColorScheme, qreal xScale, qreal yScale)
2021-09-11 18:39:38 +02:00
{
SCASSERT(painter != nullptr)
2023-06-21 09:24:51 +02:00
const qreal opacity = watermarkData.opacity / 100.;
auto BrokenImage = [&img, &watermarkData, watermarkPath, opacity, folowColorScheme]()
2021-09-11 18:39:38 +02:00
{
QString colorScheme = QStringLiteral("light");
if (folowColorScheme)
{
colorScheme =
(VTheme::ColorSheme() == VColorSheme::Light ? QStringLiteral("light") : QStringLiteral("dark"));
}
2021-09-11 18:39:38 +02:00
QPixmap watermark;
if (QString const imagePath =
QStringLiteral("puzzle=colorScheme%1+path%2+opacity%3_broken")
.arg(colorScheme, AbsoluteMPath(watermarkPath, watermarkData.path), QString::number(opacity));
not QPixmapCache::find(imagePath, &watermark))
2021-09-11 18:39:38 +02:00
{
2024-02-19 17:09:56 +01:00
QScopedPointer<QSvgRenderer> const svgRenderer(new QSvgRenderer());
2021-09-11 18:39:38 +02:00
2024-02-19 17:09:56 +01:00
QRect const imageRect(0, 0, qRound(img.width() / 4.), qRound(img.width() / 4.));
2021-09-11 18:39:38 +02:00
watermark = QPixmap(imageRect.size());
watermark.fill(Qt::transparent);
QPainter imagePainter(&watermark);
imagePainter.setOpacity(opacity);
svgRenderer->load(QStringLiteral("://puzzleicon/svg/%1/no_watermark_image.svg").arg(colorScheme));
2021-09-11 18:39:38 +02:00
svgRenderer->render(&imagePainter, imageRect);
QPixmapCache::insert(imagePath, watermark);
return watermark;
}
return watermark;
};
2024-02-19 17:09:56 +01:00
QString const imagePath = AbsoluteMPath(watermarkPath, watermarkData.path);
QFileInfo const f(imagePath);
QImageReader imageReader(imagePath);
2024-02-19 17:09:56 +01:00
QImage const watermarkImage = imageReader.read();
if (watermarkImage.isNull())
{
if (QPixmap const watermarkPixmap = BrokenImage();
watermarkPixmap.width() < img.width() && watermarkPixmap.height() < img.height())
{
QRect imagePosition(0, 0, watermarkPixmap.width(), watermarkPixmap.height());
imagePosition.translate(img.center().toPoint() - imagePosition.center());
painter->drawPixmap(imagePosition, watermarkPixmap);
}
else
{
painter->drawPixmap(img.toRect(), watermarkPixmap);
}
return;
}
2024-02-19 17:09:56 +01:00
qint64 const fileSize = watermarkImage.sizeInBytes();
qint64 const pixelSize = fileSize / watermarkImage.height() / watermarkImage.width();
QSize const scaledSize(qRound(watermarkImage.width() / xScale), qRound(watermarkImage.height() / yScale));
qint64 const scaledImageSize = pixelSize * scaledSize.width() * scaledSize.height() / 1024;
if (int const limit = QPixmapCache::cacheLimit(); scaledImageSize > limit && (xScale < 1 || yScale < 1))
{
2024-02-19 17:09:56 +01:00
QScopedPointer<QSvgRenderer> const svgRenderer(new QSvgRenderer());
painter->save();
painter->setOpacity(opacity);
painter->restore();
2024-02-19 17:09:56 +01:00
QString const grayscale = watermarkData.grayscale ? QStringLiteral("_grayscale") : QString();
svgRenderer->load(QStringLiteral("://puzzleicon/svg/watermark_placeholder%1.svg").arg(grayscale));
QRect imageRect(0, 0, qRound(watermarkImage.width() / xScale), qRound(watermarkImage.height() / yScale));
imageRect.translate(img.center().toPoint() - imageRect.center());
svgRenderer->render(painter, imageRect);
return;
}
2021-09-11 18:39:38 +02:00
QPixmap watermark;
if (f.suffix() == "png"_L1 || f.suffix() == "jpg"_L1 || f.suffix() == "jpeg"_L1 || f.suffix() == "bmp"_L1)
2021-09-11 18:39:38 +02:00
{
QString error;
watermark = WatermarkImageFromCache(watermarkData, watermarkPath, xScale, yScale, error);
if (watermark.isNull())
{
watermark = BrokenImage();
}
}
else
{
watermark = BrokenImage();
}
painter->save();
2023-06-21 09:24:51 +02:00
painter->setOpacity(watermarkData.opacity / 100.);
2021-09-28 15:10:55 +02:00
QRect imagePosition(0, 0, watermark.width(), watermark.height());
imagePosition.translate(img.center().toPoint() - imagePosition.center());
2021-09-11 18:39:38 +02:00
if (watermark.width() < img.width() && watermark.height() < img.height())
{
painter->drawPixmap(imagePosition, watermark);
}
else
{
2024-02-19 17:09:56 +01:00
QRect const croppedRect = imagePosition.intersected(img.toRect());
QPixmap const cropped = watermark.copy(croppedRect.translated(-imagePosition.x(), -imagePosition.y()));
2021-09-28 15:10:55 +02:00
painter->drawPixmap(croppedRect, cropped);
2021-09-11 18:39:38 +02:00
}
painter->restore();
2021-09-11 18:39:38 +02:00
}