mirror of
https://github.com/JakubMelka/PDF4QT.git
synced 2025-01-23 13:50:39 +01:00
381 lines
15 KiB
C++
381 lines
15 KiB
C++
// Copyright (C) 2020-2021 Jakub Melka
|
|
//
|
|
// This file is part of PDF4QT.
|
|
//
|
|
// PDF4QT is free software: you can redistribute it and/or modify
|
|
// it under the terms of the GNU Lesser General Public License as published by
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
// with the written consent of the copyright owner, any later version.
|
|
//
|
|
// PDF4QT is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
// GNU Lesser General Public License for more details.
|
|
//
|
|
// You should have received a copy of the GNU Lesser General Public License
|
|
// along with PDF4QT. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
#include "dimensionsplugin.h"
|
|
#include "pdfdrawwidget.h"
|
|
#include "pdfwidgetutils.h"
|
|
#include "pdfcms.h"
|
|
#include "settingsdialog.h"
|
|
|
|
#include <QPainter>
|
|
#include <QFontMetricsF>
|
|
|
|
namespace pdfplugin
|
|
{
|
|
|
|
DimensionsPlugin::DimensionsPlugin() :
|
|
pdf::PDFPlugin(nullptr),
|
|
m_dimensionTools(),
|
|
m_showDimensionsAction(nullptr),
|
|
m_clearDimensionsAction(nullptr),
|
|
m_settingsAction(nullptr)
|
|
{
|
|
|
|
}
|
|
|
|
void DimensionsPlugin::setWidget(pdf::PDFWidget* widget)
|
|
{
|
|
Q_ASSERT(!m_widget);
|
|
|
|
BaseClass::setWidget(widget);
|
|
|
|
QAction* horizontalDimensionAction = new QAction(QIcon(":/pdfplugins/dimensiontool/linear-horizontal.svg"), tr("&Horizontal Dimension"), this);
|
|
QAction* verticalDimensionAction = new QAction(QIcon(":/pdfplugins/dimensiontool/linear-vertical.svg"), tr("&Vertical Dimension"), this);
|
|
QAction* linearDimensionAction = new QAction(QIcon(":/pdfplugins/dimensiontool/linear.svg"), tr("&Linear Dimension"), this);
|
|
QAction* perimeterDimensionAction = new QAction(QIcon(":/pdfplugins/dimensiontool/perimeter.svg"), tr("&Perimeter"), this);
|
|
QAction* rectanglePerimeterDimensionAction = new QAction(QIcon(":/pdfplugins/dimensiontool/rectangle-perimeter.svg"), tr("&Rectangle Perimeter"), this);
|
|
QAction* areaDimensionAction = new QAction(QIcon(":/pdfplugins/dimensiontool/area.svg"), tr("&Area"), this);
|
|
QAction* rectangleAreaDimensionAction = new QAction(QIcon(":/pdfplugins/dimensiontool/rectangle-area.svg"), tr("R&ectangle Area"), this);
|
|
QAction* angleAction = new QAction(QIcon(":/pdfplugins/dimensiontool/angle.svg"), tr("An&gle"), this);
|
|
|
|
horizontalDimensionAction->setObjectName("dimensiontool_LinearHorizontalAction");
|
|
verticalDimensionAction->setObjectName("dimensiontool_LinearVerticalAction");
|
|
linearDimensionAction->setObjectName("dimensiontool_LinearAction");
|
|
perimeterDimensionAction->setObjectName("dimensiontool_PerimeterAction");
|
|
rectanglePerimeterDimensionAction->setObjectName("dimensiontool_RectanglePerimeterAction");
|
|
areaDimensionAction->setObjectName("dimensiontool_AreaAction");
|
|
rectangleAreaDimensionAction->setObjectName("dimensiontool_RectangleAreaAction");
|
|
angleAction->setObjectName("dimensiontool_AngleAction");
|
|
|
|
horizontalDimensionAction->setCheckable(true);
|
|
verticalDimensionAction->setCheckable(true);
|
|
linearDimensionAction->setCheckable(true);
|
|
perimeterDimensionAction->setCheckable(true);
|
|
rectanglePerimeterDimensionAction->setCheckable(true);
|
|
areaDimensionAction->setCheckable(true);
|
|
rectangleAreaDimensionAction->setCheckable(true);
|
|
angleAction->setCheckable(true);
|
|
|
|
m_dimensionTools[DimensionTool::LinearHorizontal] = new DimensionTool(DimensionTool::LinearHorizontal, widget->getDrawWidgetProxy(), horizontalDimensionAction, this);
|
|
m_dimensionTools[DimensionTool::LinearVertical] = new DimensionTool(DimensionTool::LinearVertical, widget->getDrawWidgetProxy(), verticalDimensionAction, this);
|
|
m_dimensionTools[DimensionTool::Linear] = new DimensionTool(DimensionTool::Linear, widget->getDrawWidgetProxy(), linearDimensionAction, this);
|
|
m_dimensionTools[DimensionTool::Perimeter] = new DimensionTool(DimensionTool::Perimeter, widget->getDrawWidgetProxy(), perimeterDimensionAction, this);
|
|
m_dimensionTools[DimensionTool::RectanglePerimeter] = new DimensionTool(DimensionTool::RectanglePerimeter, widget->getDrawWidgetProxy(), rectanglePerimeterDimensionAction, this);
|
|
m_dimensionTools[DimensionTool::Area] = new DimensionTool(DimensionTool::Area, widget->getDrawWidgetProxy(), areaDimensionAction, this);
|
|
m_dimensionTools[DimensionTool::RectangleArea] = new DimensionTool(DimensionTool::RectangleArea, widget->getDrawWidgetProxy(), rectangleAreaDimensionAction, this);
|
|
m_dimensionTools[DimensionTool::Angle] = new DimensionTool(DimensionTool::Angle, widget->getDrawWidgetProxy(), angleAction, this);
|
|
|
|
pdf::PDFToolManager* toolManager = widget->getToolManager();
|
|
for (DimensionTool* tool : m_dimensionTools)
|
|
{
|
|
toolManager->addTool(tool);
|
|
connect(tool, &DimensionTool::dimensionCreated, this, &DimensionsPlugin::onDimensionCreated);
|
|
}
|
|
|
|
m_showDimensionsAction = new QAction(QIcon(":/pdfplugins/dimensiontool/show-dimensions.svg"), tr("&Show Dimensions"), this);
|
|
m_clearDimensionsAction = new QAction(QIcon(":/pdfplugins/dimensiontool/clear-dimensions.svg"), tr("&Clear Dimensions"), this);
|
|
m_settingsAction = new QAction(QIcon(":/pdfplugins/dimensiontool/settings.svg"), tr("Se&ttings"), this);
|
|
|
|
m_showDimensionsAction->setCheckable(true);
|
|
m_showDimensionsAction->setChecked(true);
|
|
|
|
connect(m_showDimensionsAction, &QAction::triggered, this, &DimensionsPlugin::onShowDimensionsTriggered);
|
|
connect(m_clearDimensionsAction, &QAction::triggered, this, &DimensionsPlugin::onClearDimensionsTriggered);
|
|
connect(m_settingsAction, &QAction::triggered, this, &DimensionsPlugin::onSettingsTriggered);
|
|
|
|
m_lengthUnit = DimensionUnit::getLengthUnits().front();
|
|
m_areaUnit = DimensionUnit::getAreaUnits().front();
|
|
m_angleUnit = DimensionUnit::getAngleUnits().front();
|
|
|
|
m_widget->getDrawWidgetProxy()->registerDrawInterface(this);
|
|
|
|
updateActions();
|
|
}
|
|
|
|
void DimensionsPlugin::setDocument(const pdf::PDFModifiedDocument& document)
|
|
{
|
|
BaseClass::setDocument(document);
|
|
|
|
if (document.hasReset())
|
|
{
|
|
m_dimensions.clear();
|
|
updateActions();
|
|
}
|
|
}
|
|
|
|
std::vector<QAction*> DimensionsPlugin::getActions() const
|
|
{
|
|
std::vector<QAction*> result;
|
|
|
|
for (DimensionTool* tool : m_dimensionTools)
|
|
{
|
|
if (tool)
|
|
{
|
|
result.push_back(tool->getAction());
|
|
}
|
|
}
|
|
|
|
if (!result.empty())
|
|
{
|
|
result.push_back(nullptr);
|
|
result.push_back(m_showDimensionsAction);
|
|
result.push_back(m_clearDimensionsAction);
|
|
result.push_back(m_settingsAction);
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
QString DimensionsPlugin::getPluginMenuName() const
|
|
{
|
|
return tr("&Dimensions");
|
|
}
|
|
|
|
void DimensionsPlugin::drawPage(QPainter* painter,
|
|
pdf::PDFInteger pageIndex,
|
|
const pdf::PDFPrecompiledPage* compiledPage,
|
|
pdf::PDFTextLayoutGetter& layoutGetter,
|
|
const QTransform& pagePointToDevicePointMatrix,
|
|
const pdf::PDFColorConvertor& convertor,
|
|
QList<pdf::PDFRenderError>& errors) const
|
|
{
|
|
Q_UNUSED(compiledPage);
|
|
Q_UNUSED(layoutGetter);
|
|
Q_UNUSED(errors);
|
|
|
|
if (!m_showDimensionsAction || !m_showDimensionsAction->isChecked() || m_dimensions.empty())
|
|
{
|
|
// Nothing to draw
|
|
return;
|
|
}
|
|
|
|
QLocale locale;
|
|
painter->setRenderHint(QPainter::Antialiasing, true);
|
|
|
|
for (const Dimension& dimension : m_dimensions)
|
|
{
|
|
if (pageIndex != dimension.getPageIndex())
|
|
{
|
|
continue;
|
|
}
|
|
|
|
switch (dimension.getType())
|
|
{
|
|
case Dimension::Linear:
|
|
{
|
|
QPointF p1 = pagePointToDevicePointMatrix.map(dimension.getPolygon().front());
|
|
QPointF p2 = pagePointToDevicePointMatrix.map(dimension.getPolygon().back());
|
|
|
|
// Swap so p1 is to the left of the page, before p2 (for correct determination of angle)
|
|
if (p1.x() > p2.x())
|
|
{
|
|
qSwap(p1, p2);
|
|
}
|
|
|
|
QLineF line(p1, p2);
|
|
|
|
if (qFuzzyIsNull(line.length()))
|
|
{
|
|
// If we have zero line, then do nothing
|
|
continue;
|
|
}
|
|
|
|
QLineF unitVectorLine = line.normalVector().unitVector();
|
|
QPointF unitVector = unitVectorLine.p2() - unitVectorLine.p1();
|
|
qreal extensionLineSize = pdf::PDFWidgetUtils::scaleDPI_y(painter->device(), 5);
|
|
|
|
painter->setPen(convertor.convert(QColor(Qt::black), false, true));
|
|
painter->drawLine(line);
|
|
|
|
QLineF extensionLineLeft(p1 - unitVector * extensionLineSize, p1 + unitVector * extensionLineSize);
|
|
QLineF extensionLineRight(p2 - unitVector * extensionLineSize, p2 + unitVector * extensionLineSize);
|
|
|
|
painter->drawLine(extensionLineLeft);
|
|
painter->drawLine(extensionLineRight);
|
|
|
|
QPointF textPoint = line.center();
|
|
qreal angle = line.angle();
|
|
QFontMetricsF fontMetrics(painter->font());
|
|
QRectF textRect(-line.length() * 0.5, -fontMetrics.lineSpacing(), line.length(), fontMetrics.lineSpacing());
|
|
|
|
QString text = QString("%1 %2").arg(locale.toString(dimension.getMeasuredValue() * m_lengthUnit.scale, 'f', 2), m_lengthUnit.symbol);
|
|
|
|
painter->save();
|
|
painter->translate(textPoint);
|
|
painter->rotate(-angle);
|
|
painter->drawText(textRect, Qt::AlignCenter | Qt::TextDontClip, text);
|
|
painter->restore();
|
|
|
|
break;
|
|
}
|
|
|
|
case Dimension::Perimeter:
|
|
case Dimension::Area:
|
|
{
|
|
const bool isArea = dimension.getType() == Dimension::Type::Area;
|
|
const std::vector<QPointF>& polygon = dimension.getPolygon();
|
|
|
|
qreal lineSize = pdf::PDFWidgetUtils::scaleDPI_x(painter->device(), 1);
|
|
qreal pointSize = pdf::PDFWidgetUtils::scaleDPI_x(painter->device(), 5);
|
|
|
|
painter->save();
|
|
QPen pen(Qt::black);
|
|
pen.setWidthF(lineSize);
|
|
pen.setCosmetic(true);
|
|
|
|
QColor brushColor = Qt::black;
|
|
brushColor.setAlphaF(0.1f);
|
|
painter->setPen(convertor.convert(pen));
|
|
painter->setBrush(convertor.convert(QBrush(brushColor, isArea ? Qt::SolidPattern : Qt::DiagCrossPattern)));
|
|
|
|
painter->setTransform(QTransform(pagePointToDevicePointMatrix), true);
|
|
painter->drawPolygon(polygon.data(), int(polygon.size()), Qt::OddEvenFill);
|
|
painter->restore();
|
|
|
|
QPen penPoint(convertor.convert(QColor(Qt::black), false, true));
|
|
penPoint.setCapStyle(Qt::RoundCap);
|
|
penPoint.setWidthF(pointSize);
|
|
painter->setPen(penPoint);
|
|
|
|
QPointF centerPoint(0, 0);
|
|
for (const QPointF& point : polygon)
|
|
{
|
|
QPointF mappedPoint = pagePointToDevicePointMatrix.map(point);
|
|
centerPoint += mappedPoint;
|
|
painter->drawPoint(mappedPoint);
|
|
}
|
|
|
|
centerPoint *= 1.0 / qreal(polygon.size());
|
|
|
|
QString text;
|
|
|
|
if (isArea)
|
|
{
|
|
text = tr("A = %1 %2").arg(locale.toString(dimension.getMeasuredValue() * m_areaUnit.scale, 'f', 2), m_areaUnit.symbol);
|
|
}
|
|
else
|
|
{
|
|
text = tr("p = %1 %2").arg(locale.toString(dimension.getMeasuredValue() * m_lengthUnit.scale, 'f', 2), m_lengthUnit.symbol);
|
|
}
|
|
painter->drawText(centerPoint, text);
|
|
|
|
break;
|
|
}
|
|
|
|
case Dimension::Angular:
|
|
{
|
|
const std::vector<QPointF>& polygon = dimension.getPolygon();
|
|
QLineF line1(pagePointToDevicePointMatrix.map(polygon[1]), pagePointToDevicePointMatrix.map(polygon.front()));
|
|
QLineF line2(pagePointToDevicePointMatrix.map(polygon[1]), pagePointToDevicePointMatrix.map(polygon.back()));
|
|
|
|
qreal lineSize = pdf::PDFWidgetUtils::scaleDPI_x(painter->device(), 1);
|
|
qreal pointSize = pdf::PDFWidgetUtils::scaleDPI_x(painter->device(), 5);
|
|
|
|
qreal maxLength = qMax(line1.length(), line2.length());
|
|
line1.setLength(maxLength);
|
|
line2.setLength(maxLength);
|
|
|
|
QPen pen(convertor.convert(QColor(Qt::black), false, true));
|
|
pen.setWidthF(lineSize);
|
|
painter->setPen(qMove(pen));
|
|
|
|
painter->drawLine(line1);
|
|
painter->drawLine(line2);
|
|
|
|
qreal startAngle = line1.angle() * 16;
|
|
qreal angleLength = dimension.getMeasuredValue() * 16;
|
|
|
|
QRectF rect(-maxLength * 0.5, -maxLength * 0.5, maxLength, maxLength);
|
|
rect.translate(line1.p1());
|
|
painter->drawArc(rect, startAngle - angleLength, angleLength);
|
|
|
|
QPen penPoint(convertor.convert(QColor(Qt::black), false, true));
|
|
penPoint.setCapStyle(Qt::RoundCap);
|
|
penPoint.setWidthF(pointSize);
|
|
painter->setPen(penPoint);
|
|
|
|
painter->drawPoint(line1.p1());
|
|
painter->drawPoint(line1.p2());
|
|
painter->drawPoint(line2.p2());
|
|
|
|
QTransform textMatrix;
|
|
textMatrix.translate(line1.x1(), line1.y1());
|
|
textMatrix.rotate(-line1.angle() + dimension.getMeasuredValue() * 0.5);
|
|
|
|
QPointF textPoint = textMatrix.map(QPointF(maxLength * 0.25, 0.0));
|
|
QString text = QString("%1 %2").arg(locale.toString(dimension.getMeasuredValue() * m_angleUnit.scale, 'f', 2), m_angleUnit.symbol);
|
|
painter->drawText(textPoint, text);
|
|
|
|
break;
|
|
}
|
|
|
|
default:
|
|
Q_ASSERT(false);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
void DimensionsPlugin::onShowDimensionsTriggered()
|
|
{
|
|
updateGraphics();
|
|
}
|
|
|
|
void DimensionsPlugin::onClearDimensionsTriggered()
|
|
{
|
|
m_dimensions.clear();
|
|
updateActions();
|
|
updateGraphics();
|
|
}
|
|
|
|
void DimensionsPlugin::onSettingsTriggered()
|
|
{
|
|
SettingsDialog dialog(m_widget, m_lengthUnit, m_areaUnit, m_angleUnit);
|
|
dialog.exec();
|
|
|
|
updateGraphics();
|
|
}
|
|
|
|
void DimensionsPlugin::onDimensionCreated(Dimension dimension)
|
|
{
|
|
m_dimensions.emplace_back(qMove(dimension));
|
|
updateActions();
|
|
updateGraphics();
|
|
}
|
|
|
|
void DimensionsPlugin::updateActions()
|
|
{
|
|
if (m_showDimensionsAction)
|
|
{
|
|
m_showDimensionsAction->setEnabled(!m_dimensions.empty());
|
|
}
|
|
if (m_clearDimensionsAction)
|
|
{
|
|
m_clearDimensionsAction->setEnabled(!m_dimensions.empty());
|
|
}
|
|
}
|
|
|
|
void DimensionsPlugin::updateGraphics()
|
|
{
|
|
if (m_widget)
|
|
{
|
|
m_widget->getDrawWidget()->getWidget()->update();
|
|
}
|
|
}
|
|
|
|
}
|