#include "vptilefactory.h" #include #include "../vwidgets/vmaingraphicsscene.h" #include "layout/vpsheet.h" #include "scene/vpmaingraphicsview.h" #include "../vmisc/def.h" #include "../vmisc/vcommonsettings.h" #include "../vlayout/vprintlayout.h" namespace { const QColor tileColor(180, 180, 180); //--------------------------------------------------------------------------------------------------------------------- auto Grayscale(QImage image) -> QImage { for (int ii = 0; ii < image.height(); ii++) { uchar* scan = image.scanLine(ii); int depth = 4; for (int jj = 0; jj < image.width(); jj++) { QRgb* rgbpixel = reinterpret_cast(scan + jj * depth); int gray = qGray(*rgbpixel); *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 { const qreal opacity = watermarkData.opacity/100.; QPixmap pixmap; QString imagePath = AbsoluteMPath(watermarkPath, watermarkData.path); QString imageCacheKey = QString("puzzle=path%1+opacity%2+rotation%3+grayscale%4+xscale%5+yxcale%6") .arg(imagePath, QString::number(opacity), QString::number(watermarkData.imageRotation), watermarkData.grayscale ? trueStr : falseStr ).arg(xScale).arg(yScale); if (not QPixmapCache::find(imageCacheKey, &pixmap)) { QImageReader imageReader(imagePath); QImage watermark = imageReader.read(); if (watermark.isNull()) { error = imageReader.errorString(); return pixmap; } if (watermarkData.grayscale) { watermark = Grayscale(watermark); } // Workaround for QGraphicsPixmapItem opacity problem. // Opacity applied only if use a cached pixmap and only after first draw. First image always has opacity 1. // Preparing an image manually allows to avoid the problem. QSize scaledSize(qRound(watermark.width() * xScale), qRound(watermark.height() * yScale)); QImage tmp(scaledSize, watermark.format()); tmp = tmp.convertToFormat(QImage::Format_ARGB32); tmp.fill(Qt::transparent); QPainter p(&tmp); p.setOpacity(opacity); QTransform t; t.translate(tmp.width()/2., tmp.height()/2.); t.rotate(-watermarkData.imageRotation); t.translate(-tmp.width()/2., -tmp.height()/2.); p.setTransform(t); p.drawImage(QRectF(QPointF(), scaledSize), watermark); pixmap = QPixmap::fromImage(tmp); QPixmapCache::insert(imageCacheKey, pixmap); } return pixmap; } } // namespace //--------------------------------------------------------------------------------------------------------------------- VPTileFactory::VPTileFactory(const VPLayoutPtr &layout, VCommonSettings *commonSettings): m_layout(layout), m_commonSettings(commonSettings) { } //--------------------------------------------------------------------------------------------------------------------- void VPTileFactory::refreshTileInfos() { VPLayoutPtr layout = m_layout.toStrongRef(); if(not layout.isNull()) { QSizeF tilesSize = layout->LayoutSettings().GetTilesSize(); QMarginsF tilesMargins = layout->LayoutSettings().GetTilesMargins(); // sets the drawing height m_drawingAreaHeight = tilesSize.height(); if (not layout->LayoutSettings().IgnoreTilesMargins()) { m_drawingAreaHeight -= tilesMargins.top() + tilesMargins.bottom(); } // sets the drawing width m_drawingAreaWidth = tilesSize.width(); if (not layout->LayoutSettings().IgnoreTilesMargins()) { m_drawingAreaWidth -= tilesMargins.left() + tilesMargins.right(); } m_watermarkData = layout->WatermarkData(); } } //--------------------------------------------------------------------------------------------------------------------- void VPTileFactory::drawTile(QPainter *painter, QPrinter *printer, const VPSheetPtr &sheet, int row, int col) { SCASSERT(painter != nullptr) SCASSERT(printer != nullptr) VPLayoutPtr layout = m_layout.toStrongRef(); if(layout.isNull()) { return; } if (sheet.isNull()) { return; } QMarginsF sheetMargins; if (not sheet->IgnoreMargins()) { sheetMargins = sheet->GetSheetMargins(); } const int nbCol = ColNb(sheet); const int nbRow = RowNb(sheet); if (row < 0 || row > nbRow || col < 0 || col > nbCol) { return; } QPen penTileInfos = QPen(tileColor, m_commonSettings->WidthHairLine(), Qt::DashLine, Qt::RoundCap, Qt::RoundJoin); QPen penTileDrawing = QPen(Qt::black, m_commonSettings->WidthMainLine(), Qt::SolidLine, Qt::RoundCap, Qt::RoundJoin); painter->setPen(penTileDrawing); // paint the content of the page qreal xScale = layout->LayoutSettings().HorizontalScale(); qreal yScale = layout->LayoutSettings().VerticalScale(); QRectF source = QRectF(sheetMargins.left() + col*(m_drawingAreaWidth - tileStripeWidth) / xScale, sheetMargins.top() + row*(m_drawingAreaHeight - tileStripeWidth) / yScale, m_drawingAreaWidth / xScale, m_drawingAreaHeight / yScale); QRectF target = QRectF(0, 0, m_drawingAreaWidth, m_drawingAreaHeight); sheet->SceneData()->Scene()->render(painter, VPrintLayout::SceneTargetRect(printer, target), source, Qt::IgnoreAspectRatio); QScopedPointer svgRenderer(new QSvgRenderer()); // ------------- prepare triangles for position marks QRectF rectBasic = QRectF(-UnitConvertor(0.5, Unit::Cm, Unit::Px), 0, UnitConvertor(1, Unit::Cm, Unit::Px), UnitConvertor(0.5, Unit::Cm, Unit::Px) ); QPainterPath triangleBasic; triangleBasic.moveTo(rectBasic.topLeft()); triangleBasic.lineTo(rectBasic.topRight()); triangleBasic.lineTo(rectBasic.left() + (rectBasic.width() / 2), rectBasic.bottom()); triangleBasic.lineTo(rectBasic.topLeft()); QBrush triangleBush = QBrush(QColor(200,200,200)); // add the tiles decorations (cutting and gluing lines, scissors, infos etc.) painter->setPen(penTileInfos); if(row > 0) { // add top triangle QPainterPath triangleTop = QTransform() .translate(m_drawingAreaWidth/2, 0) .map(triangleBasic); painter->fillPath(triangleTop, triangleBush); // scissors along the top line svgRenderer->load(QStringLiteral("://puzzleicon/svg/icon_scissors_horizontal.svg")); svgRenderer->render(painter, QRectF(m_drawingAreaWidth - tileStripeWidth, 0, UnitConvertor(0.95, Unit::Cm, Unit::Px), UnitConvertor(0.56, Unit::Cm, Unit::Px))); // dashed top line (for cutting) penTileInfos.setStyle(Qt::DashLine); painter->setPen(penTileInfos); painter->drawLine(QPointF(), QPointF(m_drawingAreaWidth, 0)); } else { // solid top line stopping at the edge penTileInfos.setStyle(Qt::SolidLine); painter->setPen(penTileInfos); if(col < nbCol - 1) { painter->drawLine(QPointF(), QPointF(m_drawingAreaWidth, 0)); } else { painter->drawLine(QPointF(), QPointF(m_drawingAreaWidth - tileStripeWidth, 0)); } } if(col > 0) { // add left triangle QPainterPath triangleLeft = QTransform() .translate(0, m_drawingAreaHeight/2) .rotate(-90) .map(triangleBasic); painter->fillPath(triangleLeft, triangleBush); // scissors along the left line svgRenderer->load(QStringLiteral("://puzzleicon/svg/icon_scissors_vertical.svg")); svgRenderer->render(painter, QRectF(0, m_drawingAreaHeight - tileStripeWidth, UnitConvertor(0.56, Unit::Cm, Unit::Px), UnitConvertor(0.95, Unit::Cm, Unit::Px))); // dashed left line (for cutting) penTileInfos.setStyle(Qt::DashLine); painter->setPen(penTileInfos); painter->drawLine(QPointF(), QPointF(0, m_drawingAreaHeight)); } else { // solid left line at the edge penTileInfos.setStyle(Qt::SolidLine); painter->setPen(penTileInfos); if(row < nbRow - 1) { painter->drawLine(QPointF(), QPointF(0, m_drawingAreaHeight)); } else { painter->drawLine(QPointF(), QPointF(0, m_drawingAreaHeight - tileStripeWidth)); } } if(row < nbRow-1) { // add bottom triangle QPainterPath triangleBottom = QTransform() .translate(m_drawingAreaWidth/2, m_drawingAreaHeight - tileStripeWidth) .rotate(180) .map(triangleBasic); painter->fillPath(triangleBottom, triangleBush); // dotted bottom line (for glueing) penTileInfos.setStyle(Qt::DotLine); painter->setPen(penTileInfos); painter->drawLine(QPointF(0, m_drawingAreaHeight - tileStripeWidth), QPointF(m_drawingAreaWidth, m_drawingAreaHeight - tileStripeWidth)); } else { // solid bottom line at the edge penTileInfos.setStyle(Qt::SolidLine); painter->setPen(penTileInfos); if(col < nbCol - 1) { painter->drawLine(QPointF(0, m_drawingAreaHeight - tileStripeWidth), QPointF(m_drawingAreaWidth, m_drawingAreaHeight - tileStripeWidth)); } else { painter->drawLine(QPointF(0, m_drawingAreaHeight - tileStripeWidth), QPointF(m_drawingAreaWidth - tileStripeWidth, m_drawingAreaHeight - tileStripeWidth));; } } DrawRuler(painter); DrawWatermark(painter); if(col < nbCol-1) { // add right triangle QPainterPath triangleRight = QTransform() .translate(m_drawingAreaWidth - tileStripeWidth, m_drawingAreaHeight/2) .rotate(90) .map(triangleBasic); painter->fillPath(triangleRight, triangleBush); // dotted right line (for glueing) penTileInfos.setStyle(Qt::DotLine); painter->setPen(penTileInfos); painter->drawLine(QPointF(m_drawingAreaWidth - tileStripeWidth, 0), QPointF(m_drawingAreaWidth - tileStripeWidth, m_drawingAreaHeight)); } else { // solid right line at the edge penTileInfos.setStyle(Qt::SolidLine); painter->setPen(penTileInfos); if(row < nbRow - 1) { 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)); } } // prepare the painting for the text information 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); const QString tileColorStr = QString("%1,%2,%3").arg(tileColor.red()).arg(tileColor.green()).arg(tileColor.blue()); td.setHtml(QString("" "" "" "" "
%2
") .arg(tileColorStr, grid)); painter->setPen(penTileInfos); painter->save(); painter->translate(QPointF(UnitConvertor(1, Unit::Cm, Unit::Px), m_drawingAreaHeight - tileStripeWidth/1.3)); 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)); QFontMetrics metrix = QFontMetrics(td.defaultFont()); #if QT_VERSION >= QT_VERSION_CHECK(5, 11, 0) int maxWidth = metrix.horizontalAdvance(QString().fill('z', 50)); #else int maxWidth = metrix.width(QString().fill('z', 50)); #endif QString clippedSheetName = metrix.elidedText(sheet->GetName(), Qt::ElideMiddle, maxWidth); td.setHtml(QString("" "" "" "" "
%2 - %3
") .arg(tileColorStr).arg(page).arg(clippedSheetName)); painter->save(); painter->rotate(-90); painter->translate(QPointF(-(m_drawingAreaHeight) + UnitConvertor(1, Unit::Cm, Unit::Px), m_drawingAreaWidth - tileStripeWidth)); td.drawContents(painter); painter->restore(); } //--------------------------------------------------------------------------------------------------------------------- auto VPTileFactory::RowNb(const VPSheetPtr &sheet) const -> int { if (sheet.isNull()) { return 0; } qreal yScale = 1; VPLayoutPtr layout = m_layout.toStrongRef(); if(not layout.isNull()) { yScale = layout->LayoutSettings().VerticalScale(); } QRectF sheetSize = sheet->GetMarginsRect(); return qCeil(sheetSize.height() * yScale / (m_drawingAreaHeight - tileStripeWidth)); } //--------------------------------------------------------------------------------------------------------------------- auto VPTileFactory::ColNb(const VPSheetPtr &sheet) const -> int { if (sheet.isNull()) { return 0; } qreal xScale = 1; VPLayoutPtr layout = m_layout.toStrongRef(); if(not layout.isNull()) { xScale = layout->LayoutSettings().HorizontalScale(); } QRectF sheetSize = sheet->GetMarginsRect(); return qCeil(sheetSize.width() * xScale / (m_drawingAreaWidth - tileStripeWidth)); } //--------------------------------------------------------------------------------------------------------------------- auto VPTileFactory::DrawingAreaHeight() const -> qreal { return m_drawingAreaHeight; } //--------------------------------------------------------------------------------------------------------------------- auto VPTileFactory::DrawingAreaWidth() const -> qreal { return m_drawingAreaWidth; } //--------------------------------------------------------------------------------------------------------------------- auto VPTileFactory::WatermarkData() const -> const VWatermarkData & { return m_watermarkData; } //--------------------------------------------------------------------------------------------------------------------- void VPTileFactory::DrawRuler(QPainter *painter) { VPLayoutPtr layout = m_layout.toStrongRef(); if(layout.isNull()) { return; } QPen rulePen(tileColor, 1, Qt::SolidLine); painter->save(); painter->setPen(rulePen); const qreal notchHeight = UnitConvertor(3, Unit::Mm, Unit::Px); const qreal shortNotchHeight = UnitConvertor(1.1, Unit::Mm, Unit::Px); Unit layoutUnits = layout->LayoutSettings().GetUnit(); const qreal step = UnitConvertor(1, layoutUnits, Unit::Px); double marksCount = (m_drawingAreaWidth-tileStripeWidth) / step; int i = 0; while (i < marksCount) { if (i != 0) { // don't need 0 notch // middle ruler line painter->drawLine(QPointF(step * i - step / 2., m_drawingAreaHeight-tileStripeWidth), QPointF(step * i - step / 2., m_drawingAreaHeight - tileStripeWidth + shortNotchHeight)); // ruler line painter->drawLine(QPointF(step * i, m_drawingAreaHeight-tileStripeWidth), QPointF(step * i, m_drawingAreaHeight - tileStripeWidth + notchHeight)); } else { QString units = layoutUnits == Unit::Cm || layoutUnits == Unit::Mm ? tr("cm", "unit") : tr("in", "unit"); QFont fnt = painter->font(); fnt.setPointSize(10); qreal unitsWidth = 0; QFontMetrics fm(fnt); #if QT_VERSION >= QT_VERSION_CHECK(5, 11, 0) unitsWidth = fm.horizontalAdvance(units); #else unitsWidth = fm.width(units); #endif painter->drawText(QPointF(step*0.5-unitsWidth*0.6, m_drawingAreaHeight - tileStripeWidth + notchHeight+shortNotchHeight), units); } ++i; } painter->restore(); } //--------------------------------------------------------------------------------------------------------------------- void VPTileFactory::DrawWatermark(QPainter *painter) { SCASSERT(painter != nullptr) VPLayoutPtr layout = m_layout.toStrongRef(); if(layout.isNull()) { return; } if (m_watermarkData.opacity > 0) { QRectF img(0, 0, m_drawingAreaWidth - tileStripeWidth, m_drawingAreaHeight - tileStripeWidth); if (m_watermarkData.showImage && not m_watermarkData.path.isEmpty()) { PaintWatermarkImage(painter, img, m_watermarkData, layout->LayoutSettings().WatermarkPath(), layout->LayoutSettings().HorizontalScale(), layout->LayoutSettings().VerticalScale()); } if (m_watermarkData.showText && not m_watermarkData.text.isEmpty()) { PaintWatermarkText(painter, img, m_watermarkData, layout->LayoutSettings().HorizontalScale(), layout->LayoutSettings().VerticalScale()); } } } //--------------------------------------------------------------------------------------------------------------------- void VPTileFactory::PaintWatermarkText(QPainter *painter, const QRectF &img, const VWatermarkData &watermarkData, qreal xScale, qreal yScale) { SCASSERT(painter != nullptr) painter->save(); painter->setOpacity(watermarkData.opacity/100.); 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); QPointF center = img.center() - text.boundingRect().center(); 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, qreal xScale, qreal yScale) { SCASSERT(painter != nullptr) auto BrokenImage = [img, watermarkData, watermarkPath]() { const qreal opacity = watermarkData.opacity/100.; QPixmap watermark; QString imagePath = QString("puzzle=path%1+opacity%2_broken") .arg(AbsoluteMPath(watermarkPath, watermarkData.path), QString::number(opacity)); if (not QPixmapCache::find(imagePath, &watermark)) { QScopedPointer svgRenderer(new QSvgRenderer()); QRect imageRect(0, 0, qRound(img.width()/4.), qRound(img.width()/4.)); watermark = QPixmap(imageRect.size()); watermark.fill(Qt::transparent); QPainter imagePainter(&watermark); imagePainter.setOpacity(opacity); svgRenderer->load(QStringLiteral("://puzzleicon/svg/no_watermark_image.svg")); svgRenderer->render(&imagePainter, imageRect); QPixmapCache::insert(imagePath, watermark); return watermark; } return watermark; }; QPixmap watermark; QFileInfo f(watermarkData.path); if (f.suffix() == "png" || f.suffix() == "jpg" || f.suffix() == "jpeg" || f.suffix() == "bmp") { QString error; watermark = WatermarkImageFromCache(watermarkData, watermarkPath, xScale, yScale, error); if (watermark.isNull()) { watermark = BrokenImage(); } } else { watermark = BrokenImage(); } if (watermark.width() < img.width() && watermark.height() < img.height()) { QRect imagePosition(0, 0, watermark.width(), watermark.height()); imagePosition.translate(img.center().toPoint() - imagePosition.center()); painter->drawPixmap(imagePosition, watermark); } else { painter->drawPixmap(img.toRect(), watermark); } }