// Copyright (C) 2020-2022 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 . #include "pdfwidgettool.h" #include "pdfdrawwidget.h" #include "pdfcompiler.h" #include "pdfwidgetutils.h" #include "pdfpainterutils.h" #include "pdfcms.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include "pdfdbgheap.h" namespace pdf { PDFWidgetTool::PDFWidgetTool(PDFDrawWidgetProxy* proxy, QObject* parent) : BaseClass(parent), m_active(false), m_document(nullptr), m_action(nullptr), m_proxy(proxy) { } PDFWidgetTool::PDFWidgetTool(PDFDrawWidgetProxy* proxy, QAction* action, QObject* parent) : BaseClass(parent), m_active(false), m_document(nullptr), m_action(action), m_proxy(proxy) { updateActions(); } PDFWidgetTool::~PDFWidgetTool() { } void PDFWidgetTool::setDocument(const PDFModifiedDocument& document) { if (m_document != document) { // We must turn off the tool, if we are changing the document. We turn off tool, // only if whole document is being reset. if (document.hasReset()) { setActive(false); } m_document = document; for (PDFWidgetTool* tool : m_toolStack) { tool->setDocument(document); } updateActions(); } } void PDFWidgetTool::setActive(bool active) { if (m_active != active) { m_active = active; if (active) { m_proxy->registerDrawInterface(this); } else { m_proxy->unregisterDrawInterface(this); } setActiveImpl(active); updateActions(); Q_EMIT m_proxy->repaintNeeded(); Q_EMIT toolActivityChanged(active); } } void PDFWidgetTool::shortcutOverrideEvent(QWidget* widget, QKeyEvent* event) { if (PDFWidgetTool* tool = getTopToolstackTool()) { tool->shortcutOverrideEvent(widget, event); } } void PDFWidgetTool::keyPressEvent(QWidget* widget, QKeyEvent* event) { if (PDFWidgetTool* tool = getTopToolstackTool()) { tool->keyPressEvent(widget, event); } } void PDFWidgetTool::keyReleaseEvent(QWidget* widget, QKeyEvent* event) { if (PDFWidgetTool* tool = getTopToolstackTool()) { tool->keyReleaseEvent(widget, event); } } void PDFWidgetTool::mousePressEvent(QWidget* widget, QMouseEvent* event) { if (PDFWidgetTool* tool = getTopToolstackTool()) { tool->mousePressEvent(widget, event); } } void PDFWidgetTool::mouseDoubleClickEvent(QWidget* widget, QMouseEvent* event) { if (PDFWidgetTool* tool = getTopToolstackTool()) { tool->mouseDoubleClickEvent(widget, event); } } void PDFWidgetTool::mouseReleaseEvent(QWidget* widget, QMouseEvent* event) { if (PDFWidgetTool* tool = getTopToolstackTool()) { tool->mouseReleaseEvent(widget, event); } } void PDFWidgetTool::mouseMoveEvent(QWidget* widget, QMouseEvent* event) { if (PDFWidgetTool* tool = getTopToolstackTool()) { tool->mouseMoveEvent(widget, event); } } void PDFWidgetTool::wheelEvent(QWidget* widget, QWheelEvent* event) { if (PDFWidgetTool* tool = getTopToolstackTool()) { tool->wheelEvent(widget, event); } } const std::optional& PDFWidgetTool::getCursor() const { // If we have active subtool, return its mouse cursor if (PDFWidgetTool* tool = getTopToolstackTool()) { return tool->getCursor(); } return m_cursor; } void PDFWidgetTool::setActiveImpl(bool active) { for (PDFWidgetTool* tool : m_toolStack) { tool->setActive(active); } } void PDFWidgetTool::updateActions() { if (m_action) { m_action->setChecked(isActive()); m_action->setEnabled(m_document); } } PDFWidgetTool* PDFWidgetTool::getTopToolstackTool() const { if (!m_toolStack.empty()) { return m_toolStack.back(); } return nullptr; } void PDFWidgetTool::addTool(PDFWidgetTool* tool) { tool->setActive(isActive()); m_toolStack.push_back(tool); } void PDFWidgetTool::removeTool() { m_toolStack.back()->setActive(false); m_toolStack.pop_back(); } void PDFFindTextToolDialog::paintEvent(QPaintEvent* event) { QDialog::paintEvent(event); QStylePainter painter(this); // Dialog rectangle QRect rect = this->rect(); const int titleBarHeight = style()->pixelMetric(QStyle::PM_TitleBarHeight); QRect titleBarRect = rect; titleBarRect.setHeight(titleBarHeight); QStyleOptionTitleBar titleOption; titleOption.initFrom(this); titleOption.text = windowTitle(); titleOption.rect = titleBarRect; titleOption.titleBarState = windowState() | Qt::WindowActive; titleOption.titleBarFlags = Qt::Popup | Qt::CustomizeWindowHint | Qt::WindowTitleHint; painter.drawComplexControl(QStyle::CC_TitleBar, titleOption); QStyleOptionFrame frameOption; frameOption.initFrom(this); frameOption.rect = QRect(rect.x(), rect.y() + titleBarHeight, rect.width(), rect.height() - titleBarHeight); painter.drawPrimitive(QStyle::PE_Frame, frameOption); } PDFFindTextToolDialog::PDFFindTextToolDialog(PDFDrawWidgetProxy* proxy, QWidget* parent, Qt::WindowFlags f) : QDialog(parent, f), m_proxy(proxy) { } bool PDFFindTextToolDialog::event(QEvent* event) { switch (event->type()) { case QEvent::Wheel: { PDFWidget* widget = m_proxy->getWidget(); QWheelEvent* oldEvent = dynamic_cast(event); if (!rect().contains(oldEvent->position().toPoint(), false)) { IDrawWidget* pdfDrawWidget = widget->getDrawWidget(); QPointF position = pdfDrawWidget->getWidget()->mapFromGlobal(oldEvent->globalPosition()); QWheelEvent wheelEvent(position, oldEvent->globalPosition(), oldEvent->pixelDelta(), oldEvent->angleDelta(), oldEvent->buttons(), oldEvent->modifiers(), oldEvent->phase(), oldEvent->isInverted(), oldEvent->source(), oldEvent->pointingDevice()); return pdfDrawWidget->doEvent(&wheelEvent); } break; } case QEvent::KeyPress: { QKeyEvent* keyEvent = dynamic_cast(event); if (!keyEvent->modifiers() || ((keyEvent->modifiers() & Qt::KeypadModifier) && keyEvent->key() == Qt::Key_Enter)) { switch (keyEvent->key()) { case Qt::Key_Return: case Qt::Key_Enter: case Qt::Key_Home: { keyEvent->accept(); Q_EMIT goToFirstResult(); return true; } case Qt::Key_End: { keyEvent->accept(); Q_EMIT goToLastResult(); return true; } default: break; } } break; } default: break; } return QDialog::event(event); } PDFFindTextTool::PDFFindTextTool(PDFDrawWidgetProxy* proxy, QAction* prevAction, QAction* nextAction, QObject* parent, QWidget* parentDialog) : BaseClass(proxy, parent), m_prevAction(prevAction), m_nextAction(nextAction), m_parentDialog(parentDialog), m_dialog(nullptr), m_caseSensitiveCheckBox(nullptr), m_wholeWordsCheckBox(nullptr), m_findTextEdit(nullptr), m_previousButton(nullptr), m_nextButton(nullptr), m_selectedResultIndex(0) { PDFAsynchronousTextLayoutCompiler* compiler = getProxy()->getTextLayoutCompiler(); connect(compiler, &PDFAsynchronousTextLayoutCompiler::textLayoutChanged, this, &PDFFindTextTool::performSearch); connect(m_prevAction, &QAction::triggered, this, &PDFFindTextTool::onActionPrevious); connect(m_nextAction, &QAction::triggered, this, &PDFFindTextTool::onActionNext); updateActions(); } void PDFFindTextTool::drawPage(QPainter* painter, PDFInteger pageIndex, const PDFPrecompiledPage* compiledPage, PDFTextLayoutGetter& layoutGetter, const QTransform& pagePointToDevicePointMatrix, const PDFColorConvertor& convertor, QList& errors) const { Q_UNUSED(compiledPage); Q_UNUSED(errors); const pdf::PDFTextSelection& textSelection = getTextSelection(); pdf::PDFTextSelectionPainter textSelectionPainter(&textSelection); textSelectionPainter.draw(painter, pageIndex, layoutGetter, pagePointToDevicePointMatrix, convertor); } void PDFFindTextTool::clearResults() { m_findResults.clear(); m_selectedResultIndex = 0; m_textSelection.dirty(); getProxy()->repaintNeeded(); } void PDFFindTextTool::goToCurrentResult() { PDFTextSelection textSelection = getTextSelectionSelectedResultOnly(); if (!textSelection.isEmpty()) { const PDFTextSelectionColoredItem& firstItem = *textSelection.begin(); PDFTextLayoutGetter textLayoutGetter = getProxy()->getTextLayoutCompiler()->getTextLayoutLazy(firstItem.start.pageIndex); pdf::PDFTextSelectionPainter textSelectionPainter(&textSelection); QPainterPath painterPath = textSelectionPainter.prepareGeometry(firstItem.start.pageIndex, textLayoutGetter, QTransform(), nullptr); if (!painterPath.isEmpty()) { getProxy()->goToPageAndEnsureVisible(firstItem.start.pageIndex, painterPath.boundingRect()); } } } void PDFFindTextTool::setActiveImpl(bool active) { BaseClass::setActiveImpl(active); if (active) { Q_ASSERT(!m_dialog); // For find, we will need text layout getProxy()->getTextLayoutCompiler()->makeTextLayout(); // Create dialog m_dialog = new PDFFindTextToolDialog(getProxy(), m_parentDialog, Qt::Popup); m_dialog->setWindowTitle(tr("Find")); QGridLayout* layout = new QGridLayout(m_dialog); m_dialog->setLayout(layout); // Jakub Melka: we will create following widgets: // - text with label // - line edit, where user can enter search text // - 2 checkbox for settings // - 2 push buttons (previous/next) m_findTextEdit = new QLineEdit(m_dialog); m_caseSensitiveCheckBox = new QCheckBox(tr("Case sensitive"), m_dialog); m_wholeWordsCheckBox = new QCheckBox(tr("Whole words only"), m_dialog); m_previousButton = new QPushButton(tr("Previous"), m_dialog); m_nextButton = new QPushButton(tr("Next"), m_dialog); m_findTextEdit->setText(m_savedText); m_caseSensitiveCheckBox->setChecked(m_savedIsCaseSensitive); m_wholeWordsCheckBox->setChecked(m_savedIsWholeWords); m_previousButton->setDefault(false); m_nextButton->setDefault(true); m_previousButton->setShortcut(m_prevAction->shortcut()); m_nextButton->setShortcut(m_nextAction->shortcut()); connect(m_previousButton, &QPushButton::clicked, m_prevAction, &QAction::trigger); connect(m_nextButton, &QPushButton::clicked, m_nextAction, &QAction::trigger); connect(m_findTextEdit, &QLineEdit::editingFinished, this, &PDFFindTextTool::onSearchText); connect(m_caseSensitiveCheckBox, &QCheckBox::clicked, this, &PDFFindTextTool::onSearchText); connect(m_wholeWordsCheckBox, &QCheckBox::clicked, this, &PDFFindTextTool::onSearchText); connect(m_dialog, &PDFFindTextToolDialog::goToFirstResult, this, &PDFFindTextTool::onActionFirst); connect(m_dialog, &PDFFindTextToolDialog::goToLastResult, this, &PDFFindTextTool::onActionLast); QMargins margins = layout->contentsMargins(); margins.setTop(margins.top() + m_dialog->style()->pixelMetric(QStyle::PM_TitleBarHeight)); layout->setContentsMargins(margins); layout->addWidget(new QLabel(tr("Search text"), m_dialog), 0, 0, 1, -1, Qt::AlignLeft); layout->addWidget(m_findTextEdit, 1, 0, 1, -1); layout->addWidget(m_caseSensitiveCheckBox, 2, 0, 1, -1, Qt::AlignLeft); layout->addWidget(m_wholeWordsCheckBox, 3, 0, 1, -1, Qt::AlignLeft); layout->addWidget(m_previousButton, 4, 0); layout->addWidget(m_nextButton, 4, 1); m_dialog->setFixedSize(m_dialog->sizeHint()); PDFWidget* widget = getProxy()->getWidget(); QPoint topRight = widget->mapToGlobal(widget->rect().topRight()); QPoint topRightParent = m_parentDialog->mapFromGlobal(topRight); m_dialog->show(); m_dialog->move(topRightParent - QPoint(m_dialog->width() * 1.1, 0)); m_dialog->setFocus(); m_findTextEdit->setFocus(); m_findTextEdit->selectAll(); connect(m_dialog, &QDialog::rejected, this, &PDFFindTextTool::onDialogRejected); onSearchText(); } else { Q_ASSERT(m_dialog); m_savedText = m_findTextEdit->text(); m_savedIsCaseSensitive = m_caseSensitiveCheckBox->isChecked(); m_savedIsWholeWords = m_wholeWordsCheckBox->isChecked(); m_dialog->deleteLater(); m_dialog = nullptr; m_caseSensitiveCheckBox = nullptr; m_wholeWordsCheckBox = nullptr; m_findTextEdit = nullptr; m_previousButton = nullptr; m_nextButton = nullptr; clearResults(); } } void PDFFindTextTool::onSearchText() { if (!isActive()) { return; } m_parameters.phrase = m_findTextEdit->text(); m_parameters.isCaseSensitive = m_caseSensitiveCheckBox->isChecked(); m_parameters.isWholeWordsOnly = m_wholeWordsCheckBox->isChecked(); m_parameters.isSearchFinished = m_parameters.phrase.isEmpty(); clearResults(); updateResultsUI(); if (m_parameters.isSearchFinished) { // We have nothing to search for return; } pdf::PDFAsynchronousTextLayoutCompiler* compiler = getProxy()->getTextLayoutCompiler(); if (compiler->isTextLayoutReady()) { performSearch(); } else { compiler->makeTextLayout(); } } void PDFFindTextTool::onActionFirst() { if (!m_findResults.empty()) { setCurrentResultIndex(0); } } void PDFFindTextTool::onActionLast() { if (!m_findResults.empty()) { setCurrentResultIndex(m_findResults.size() - 1); } } void PDFFindTextTool::onActionPrevious() { if (!m_findResults.empty()) { setCurrentResultIndex(m_selectedResultIndex == 0 ? m_findResults.size() - 1 : m_selectedResultIndex - 1); } } void PDFFindTextTool::onActionNext() { if (!m_findResults.empty()) { setCurrentResultIndex((m_selectedResultIndex + 1) % m_findResults.size()); } } void PDFFindTextTool::setCurrentResultIndex(size_t index) { if (!m_findResults.empty()) { m_selectedResultIndex = index; m_textSelection.dirty(); getProxy()->repaintNeeded(); goToCurrentResult(); updateTitle(); } } void PDFFindTextTool::onDialogRejected() { setActive(false); } void PDFFindTextTool::performSearch() { if (m_parameters.isSearchFinished) { return; } clearResults(); m_parameters.isSearchFinished = true; if (m_parameters.phrase.isEmpty()) { return; } PDFAsynchronousTextLayoutCompiler* compiler = getProxy()->getTextLayoutCompiler(); if (!compiler->isTextLayoutReady()) { // Text layout is not ready yet return; } // Prepare string to search QString expression = m_parameters.phrase; bool useRegularExpression = false; if (m_parameters.isWholeWordsOnly) { expression = QString("\\b%1\\b").arg(QRegularExpression::escape(expression)); useRegularExpression = true; } pdf::PDFTextFlow::FlowFlags flowFlags = pdf::PDFTextFlow::SeparateBlocks; const pdf::PDFTextLayoutStorage* textLayoutStorage = compiler->getTextLayoutStorage(); if (!useRegularExpression) { // Use simple text search Qt::CaseSensitivity caseSensitivity = m_parameters.isCaseSensitive ? Qt::CaseSensitive : Qt::CaseInsensitive; m_findResults = textLayoutStorage->find(expression, caseSensitivity, flowFlags); } else { // Use regular expression search QRegularExpression::PatternOptions patternOptions = QRegularExpression::UseUnicodePropertiesOption; if (!m_parameters.isCaseSensitive) { patternOptions |= QRegularExpression::CaseInsensitiveOption; } QRegularExpression regularExpression(expression, patternOptions); m_findResults = textLayoutStorage->find(regularExpression, flowFlags); } std::sort(m_findResults.begin(), m_findResults.end()); m_selectedResultIndex = 0; m_textSelection.dirty(); getProxy()->repaintNeeded(); updateResultsUI(); } void PDFFindTextTool::updateActions() { BaseClass::updateActions(); const bool isActive = this->isActive(); const bool hasResults = !m_findResults.empty(); const bool enablePrevious = isActive && hasResults; const bool enableNext = isActive && hasResults; m_prevAction->setEnabled(enablePrevious); m_nextAction->setEnabled(enableNext); } void PDFFindTextTool::updateResultsUI() { m_selectedResultIndex = qBound(size_t(0), m_selectedResultIndex, m_findResults.size()); updateActions(); updateTitle(); } void PDFFindTextTool::updateTitle() { if (!m_dialog) { return; } if (m_findResults.empty()) { m_dialog->setWindowTitle(tr("Find")); } else { m_dialog->setWindowTitle(tr("Find (%1/%2)").arg(m_selectedResultIndex + 1).arg(m_findResults.size())); } m_dialog->update(); } PDFTextSelection PDFFindTextTool::getTextSelectionImpl() const { pdf::PDFTextSelection result; for (size_t i = 0; i < m_findResults.size(); ++i) { const pdf::PDFFindResult& findResult = m_findResults[i]; QColor color(Qt::blue); if (i == m_selectedResultIndex) { color = QColor(Qt::yellow); } result.addItems(findResult.textSelectionItems, color); } result.build(); return result; } PDFTextSelection PDFFindTextTool::getTextSelectionSelectedResultOnly() const { pdf::PDFTextSelection result; if (m_selectedResultIndex < m_findResults.size()) { const pdf::PDFFindResult& findResult = m_findResults[m_selectedResultIndex]; result.addItems(findResult.textSelectionItems, Qt::transparent); } result.build(); return result; } PDFSelectTextTool::PDFSelectTextTool(PDFDrawWidgetProxy* proxy, QAction* action, QAction* copyTextAction, QAction* selectAllAction, QAction* deselectAction, QObject* parent) : BaseClass(proxy, action, parent), m_copyTextAction(copyTextAction), m_selectAllAction(selectAllAction), m_deselectAction(deselectAction), m_isCursorOverText(false) { connect(copyTextAction, &QAction::triggered, this, &PDFSelectTextTool::onActionCopyText); connect(selectAllAction, &QAction::triggered, this, &PDFSelectTextTool::onActionSelectAll); connect(deselectAction, &QAction::triggered, this, &PDFSelectTextTool::onActionDeselect); updateActions(); } void PDFSelectTextTool::drawPage(QPainter* painter, PDFInteger pageIndex, const PDFPrecompiledPage* compiledPage, PDFTextLayoutGetter& layoutGetter, const QTransform& pagePointToDevicePointMatrix, const PDFColorConvertor& convertor, QList& errors) const { Q_UNUSED(compiledPage); Q_UNUSED(errors); pdf::PDFTextSelectionPainter textSelectionPainter(&m_textSelection); textSelectionPainter.draw(painter, pageIndex, layoutGetter, pagePointToDevicePointMatrix, convertor); } void PDFSelectTextTool::mousePressEvent(QWidget* widget, QMouseEvent* event) { Q_UNUSED(widget); if (event->button() == Qt::LeftButton) { QPointF pagePoint; const PDFInteger pageIndex = getProxy()->getPageUnderPoint(event->pos(), &pagePoint); if (pageIndex != -1) { m_selectionInfo.pageIndex = pageIndex; m_selectionInfo.selectionStartPoint = pagePoint; event->accept(); } else { m_selectionInfo = SelectionInfo(); } setSelection(pdf::PDFTextSelection()); updateCursor(); } } void PDFSelectTextTool::mouseReleaseEvent(QWidget* widget, QMouseEvent* event) { Q_UNUSED(widget); if (event->button() == Qt::LeftButton) { if (m_selectionInfo.pageIndex != -1) { QPointF pagePoint; const PDFInteger pageIndex = getProxy()->getPageUnderPoint(event->pos(), &pagePoint); if (m_selectionInfo.pageIndex == pageIndex) { // Jakub Melka: handle the selection PDFTextLayout textLayout = getProxy()->getTextLayoutCompiler()->getTextLayoutLazy(pageIndex); setSelection(textLayout.createTextSelection(pageIndex, m_selectionInfo.selectionStartPoint, pagePoint)); } else { setSelection(pdf::PDFTextSelection()); } m_selectionInfo = SelectionInfo(); event->accept(); updateCursor(); } } } void PDFSelectTextTool::mouseMoveEvent(QWidget* widget, QMouseEvent* event) { Q_UNUSED(widget); // We must make text layout. This is fast, because text layout is being // created only, if it doesn't exist. This function is also called only, // if tool is active. getProxy()->getTextLayoutCompiler()->makeTextLayout(); QPointF pagePoint; const PDFInteger pageIndex = getProxy()->getPageUnderPoint(event->pos(), &pagePoint); PDFTextLayout textLayout = getProxy()->getTextLayoutCompiler()->getTextLayoutLazy(pageIndex); m_isCursorOverText = textLayout.isHoveringOverTextBlock(pagePoint); if (m_selectionInfo.pageIndex != -1) { if (m_selectionInfo.pageIndex == pageIndex) { // Jakub Melka: handle the selection setSelection(textLayout.createTextSelection(pageIndex, m_selectionInfo.selectionStartPoint, pagePoint)); } else { setSelection(pdf::PDFTextSelection()); } event->accept(); } updateCursor(); } void PDFSelectTextTool::setActiveImpl(bool active) { BaseClass::setActiveImpl(active); if (active) { pdf::PDFAsynchronousTextLayoutCompiler* compiler = getProxy()->getTextLayoutCompiler(); if (!compiler->isTextLayoutReady()) { compiler->makeTextLayout(); } } else { // Just clear the text selection setSelection(PDFTextSelection()); } } void PDFSelectTextTool::updateActions() { BaseClass::updateActions(); const bool isActive = this->isActive(); const bool hasSelection = !m_textSelection.isEmpty(); m_selectAllAction->setEnabled(isActive); m_deselectAction->setEnabled(isActive && hasSelection); m_copyTextAction->setEnabled(isActive && hasSelection); } void PDFSelectTextTool::updateCursor() { if (isActive()) { if (m_isCursorOverText) { setCursor(QCursor(Qt::IBeamCursor)); } else { setCursor(QCursor(Qt::ArrowCursor)); } } } void PDFSelectTextTool::onActionCopyText() { if (isActive()) { // Jakub Melka: we must obey document permissions if (getDocument()->getStorage().getSecurityHandler()->isAllowed(PDFSecurityHandler::Permission::CopyContent)) { QStringList result; auto it = m_textSelection.begin(); auto itEnd = m_textSelection.nextPageRange(it); while (it != m_textSelection.end()) { const PDFInteger pageIndex = it->start.pageIndex; PDFTextLayout textLayout = getProxy()->getTextLayoutCompiler()->getTextLayoutLazy(pageIndex); result << textLayout.getTextFromSelection(it, itEnd, pageIndex); it = itEnd; itEnd = m_textSelection.nextPageRange(it); } QString text = result.join("\n\n"); if (!text.isEmpty()) { QApplication::clipboard()->setText(text, QClipboard::Clipboard); } } } } void PDFSelectTextTool::onActionSelectAll() { if (isActive()) { setSelection(getProxy()->getTextLayoutCompiler()->getTextSelectionAll(Qt::yellow)); } } void PDFSelectTextTool::onActionDeselect() { if (isActive()) { setSelection(pdf::PDFTextSelection()); } } void PDFSelectTextTool::setSelection(PDFTextSelection&& textSelection) { if (m_textSelection != textSelection) { m_textSelection = qMove(textSelection); getProxy()->repaintNeeded(); updateActions(); } } PDFToolManager::PDFToolManager(PDFDrawWidgetProxy* proxy, Actions actions, QObject* parent, QWidget* parentDialog) : BaseClass(parent), m_predefinedTools() { auto pickTool = new PDFPickTool(proxy, PDFPickTool::Mode::Rectangles, this); auto pickPageTool = new PDFPickTool(proxy, PDFPickTool::Mode::Pages, this); m_predefinedTools[PickRectangleTool] = pickTool; m_predefinedTools[PickPageTool] = pickPageTool; m_predefinedTools[FindTextTool] = new PDFFindTextTool(proxy, actions.findPrevAction, actions.findNextAction, this, parentDialog); m_predefinedTools[SelectTextTool] = new PDFSelectTextTool(proxy, actions.selectTextToolAction, actions.copyTextAction, actions.selectAllAction, actions.deselectAction, this); m_predefinedTools[SelectTableTool] = new PDFSelectTableTool(proxy, actions.selectTableToolAction, this); m_predefinedTools[MagnifierTool] = new PDFMagnifierTool(proxy, actions.magnifierAction, this); m_predefinedTools[ScreenshotTool] = new PDFScreenshotTool(proxy, actions.screenshotToolAction, this); m_predefinedTools[ExtractImageTool] = new PDFExtractImageTool(proxy, actions.extractImageAction, this); for (PDFWidgetTool* tool : m_predefinedTools) { addTool(tool); } connect(pickTool, &PDFPickTool::rectanglePicked, this, &PDFToolManager::onRectanglePicked); connect(pickPageTool, &PDFPickTool::pagePicked, this, &PDFToolManager::onPagePicked); } void PDFToolManager::addTool(PDFWidgetTool* tool) { m_tools.insert(tool); connect(tool, &PDFWidgetTool::messageDisplayRequest, this, &PDFToolManager::messageDisplayRequest); if (QAction* action = tool->getAction()) { m_actionsToTools[action] = tool; connect(action, &QAction::triggered, this, &PDFToolManager::onToolActionTriggered); } connect(tool, &PDFWidgetTool::toolActivityChanged, this, &PDFToolManager::onToolActivityChanged); } void PDFToolManager::pickRectangle(std::function callback) { setActiveTool(nullptr); m_pickRectangleCallback = callback; setActiveTool(m_predefinedTools[PickRectangleTool]); } void PDFToolManager::pickPage(std::function callback) { setActiveTool(nullptr); m_pickPageCallback = callback; setActiveTool(m_predefinedTools[PickPageTool]); } void PDFToolManager::setDocument(const PDFModifiedDocument& document) { for (PDFWidgetTool* tool : m_tools) { tool->setDocument(document); } } void PDFToolManager::setActiveTool(PDFWidgetTool* tool) { PDFWidgetTool* activeTool = getActiveTool(); if (activeTool && activeTool != tool) { activeTool->setActive(false); } Q_ASSERT(!getActiveTool()); if (tool) { tool->setActive(true); } } PDFWidgetTool* PDFToolManager::getActiveTool() const { for (PDFWidgetTool* tool : m_tools) { if (tool->isActive()) { return tool; } } return nullptr; } PDFFindTextTool* PDFToolManager::getFindTextTool() const { return qobject_cast(m_predefinedTools[FindTextTool]); } PDFMagnifierTool* PDFToolManager::getMagnifierTool() const { return qobject_cast(m_predefinedTools[MagnifierTool]); } void PDFToolManager::shortcutOverrideEvent(QWidget* widget, QKeyEvent* event) { event->ignore(); if (PDFWidgetTool* activeTool = getActiveTool()) { activeTool->shortcutOverrideEvent(widget, event); } } void PDFToolManager::keyPressEvent(QWidget* widget, QKeyEvent* event) { event->ignore(); // Escape key cancels current tool PDFWidgetTool* activeTool = getActiveTool(); if (event->key() == Qt::Key_Escape && activeTool) { activeTool->setActive(false); event->accept(); return; } if (activeTool) { activeTool->keyPressEvent(widget, event); } } void PDFToolManager::keyReleaseEvent(QWidget* widget, QKeyEvent* event) { event->ignore(); if (PDFWidgetTool* activeTool = getActiveTool()) { activeTool->keyReleaseEvent(widget, event); } } void PDFToolManager::mousePressEvent(QWidget* widget, QMouseEvent* event) { event->ignore(); if (PDFWidgetTool* activeTool = getActiveTool()) { activeTool->mousePressEvent(widget, event); } } void PDFToolManager::mouseDoubleClickEvent(QWidget* widget, QMouseEvent* event) { event->ignore(); if (PDFWidgetTool* activeTool = getActiveTool()) { activeTool->mouseDoubleClickEvent(widget, event); } } void PDFToolManager::mouseReleaseEvent(QWidget* widget, QMouseEvent* event) { event->ignore(); if (PDFWidgetTool* activeTool = getActiveTool()) { activeTool->mouseReleaseEvent(widget, event); } } void PDFToolManager::mouseMoveEvent(QWidget* widget, QMouseEvent* event) { event->ignore(); if (PDFWidgetTool* activeTool = getActiveTool()) { activeTool->mouseMoveEvent(widget, event); } } void PDFToolManager::wheelEvent(QWidget* widget, QWheelEvent* event) { event->ignore(); if (PDFWidgetTool* activeTool = getActiveTool()) { activeTool->wheelEvent(widget, event); } } const std::optional& PDFToolManager::getCursor() const { if (PDFWidgetTool* tool = getActiveTool()) { return tool->getCursor(); } static const std::optional dummy; return dummy; } void PDFToolManager::onToolActivityChanged(bool active) { PDFWidgetTool* tool = qobject_cast(sender()); if (active) { // When tool is activated outside, we must deactivate old active tool for (PDFWidgetTool* currentTool : m_tools) { if (currentTool->isActive() && currentTool != tool) { currentTool->setActive(false); } } } else { // Clear callback, if we are deactivating a tool if (tool == m_predefinedTools[PickRectangleTool]) { m_pickRectangleCallback = nullptr; } if (tool == m_predefinedTools[PickPageTool]) { m_pickPageCallback = nullptr; } } } void PDFToolManager::onToolActionTriggered(bool checked) { PDFWidgetTool* tool = m_actionsToTools.at(qobject_cast(sender())); if (checked) { setActiveTool(tool); } else { tool->setActive(false); } } void PDFToolManager::onRectanglePicked(PDFInteger pageIndex, QRectF pageRectangle) { if (m_pickRectangleCallback) { m_pickRectangleCallback(pageIndex, pageRectangle); } setActiveTool(nullptr); } void PDFToolManager::onPagePicked(PDFInteger pageIndex) { if (m_pickPageCallback) { m_pickPageCallback(pageIndex); } setActiveTool(nullptr); } PDFMagnifierTool::PDFMagnifierTool(PDFDrawWidgetProxy* proxy, QAction* action, QObject* parent) : BaseClass(proxy, action, parent), m_magnifierSize(200), m_magnifierZoom(2.0) { setCursor(Qt::BlankCursor); } void PDFMagnifierTool::mousePressEvent(QWidget* widget, QMouseEvent* event) { Q_UNUSED(widget); event->accept(); } void PDFMagnifierTool::mouseReleaseEvent(QWidget* widget, QMouseEvent* event) { Q_UNUSED(widget); event->accept(); } void PDFMagnifierTool::mouseMoveEvent(QWidget* widget, QMouseEvent* event) { Q_UNUSED(widget); event->accept(); QPoint mousePos = event->pos(); if (m_mousePos != mousePos) { m_mousePos = mousePos; Q_EMIT getProxy()->repaintNeeded(); } } void PDFMagnifierTool::drawPostRendering(QPainter* painter, QRect rect) const { if (!m_mousePos.isNull()) { QPainterPath path; path.addEllipse(m_mousePos, m_magnifierSize, m_magnifierSize); painter->save(); // Clip the painter path for magnifier painter->setClipPath(path, Qt::IntersectClip); painter->fillRect(rect, getProxy()->getPaperColor()); painter->scale(m_magnifierZoom, m_magnifierZoom); // Jakub Melka: this must be explained. We want to display the origin (mouse position) // at the same position to remain under scaling. If scale == 1, then we translate // by -m_mousePos + m_mousePos = (0, 0). Otherwise we are m_mousePos / scale away // from the original position. Example: // m_mousePos = (100, 100), scale = 2 // we are translating by -(100, 100) + (50, 50) = -(50, 50), // because origin at (100, 100) is now at position (50, 50) after scale. So, if it has to remain // the same, we must translate by -(50, 50). painter->translate(m_mousePos * (1.0 / m_magnifierZoom - 1.0)); getProxy()->drawPages(painter, rect, getProxy()->getFeatures()); painter->restore(); painter->setPen(Qt::black); painter->setBrush(Qt::NoBrush); painter->drawPath(path); } } void PDFMagnifierTool::setActiveImpl(bool active) { BaseClass::setActiveImpl(active); if (!active) { m_mousePos = QPoint(); } } PDFReal PDFMagnifierTool::getMagnifierZoom() const { return m_magnifierZoom; } void PDFMagnifierTool::setMagnifierZoom(const PDFReal& magnifierZoom) { m_magnifierZoom = magnifierZoom; } int PDFMagnifierTool::getMagnifierSize() const { return m_magnifierSize; } void PDFMagnifierTool::setMagnifierSize(int magnifierSize) { m_magnifierSize = magnifierSize; } PDFPickTool::PDFPickTool(PDFDrawWidgetProxy* proxy, PDFPickTool::Mode mode, QObject* parent) : BaseClass(proxy, parent), m_mode(mode), m_pageIndex(-1), m_drawSelectionRectangle(true), m_selectionRectangleColor(Qt::blue) { switch (m_mode) { case pdf::PDFPickTool::Mode::Pages: setCursor(Qt::ArrowCursor); break; case pdf::PDFPickTool::Mode::Points: case pdf::PDFPickTool::Mode::Rectangles: setCursor(Qt::BlankCursor); break; case pdf::PDFPickTool::Mode::Images: setCursor(Qt::CrossCursor); break; default: Q_ASSERT(false); break; } m_snapper.setSnapPointPixelSize(PDFWidgetUtils::scaleDPI_x(proxy->getWidget(), 10)); m_snapper.setSnapPointTolerance(m_snapper.getSnapPointPixelSize()); m_selectionRectangleColor.setAlphaF(0.25); connect(proxy, &PDFDrawWidgetProxy::drawSpaceChanged, this, &PDFPickTool::buildSnapData); connect(proxy, &PDFDrawWidgetProxy::pageImageChanged, this, &PDFPickTool::buildSnapData); } void PDFPickTool::drawPage(QPainter* painter, PDFInteger pageIndex, const PDFPrecompiledPage* compiledPage, PDFTextLayoutGetter& layoutGetter, const QTransform& pagePointToDevicePointMatrix, const PDFColorConvertor& convertor, QList& errors) const { Q_UNUSED(compiledPage); Q_UNUSED(layoutGetter); Q_UNUSED(errors); if (!isActive()) { return; } // If we are picking rectangles, then draw current selection rectangle if (m_mode == Mode::Rectangles && m_drawSelectionRectangle && m_pageIndex == pageIndex && !m_pickedPoints.empty()) { QPoint p1 = pagePointToDevicePointMatrix.map(m_pickedPoints.back()).toPoint(); QPoint p2 = m_snapper.getSnappedPoint().toPoint(); int xMin = qMin(p1.x(), p2.x()); int xMax = qMax(p1.x(), p2.x()); int yMin = qMin(p1.y(), p2.y()); int yMax = qMax(p1.y(), p2.y()); QRect selectionRectangle(xMin, yMin, xMax - xMin, yMax - yMin); if (selectionRectangle.isValid()) { painter->fillRect(selectionRectangle, convertor.convert(m_selectionRectangleColor, false, true)); } } if (m_mode == Mode::Images && m_snapper.getSnappedImage()) { const PDFSnapper::ViewportSnapImage* snappedImage = m_snapper.getSnappedImage(); painter->fillPath(snappedImage->viewportPath, convertor.convert(m_selectionRectangleColor, false, true)); } } void PDFPickTool::drawPostRendering(QPainter* painter, QRect rect) const { if (!isActive()) { return; } if (m_mode != Mode::Images) { m_snapper.drawSnapPoints(painter); QPoint snappedPoint = m_snapper.getSnappedPoint().toPoint(); QPoint hleft = snappedPoint; QPoint hright = snappedPoint; QPoint vtop = snappedPoint; QPoint vbottom = snappedPoint; hleft.setX(0); hright.setX(rect.width()); vtop.setY(0); vbottom.setY(rect.height()); painter->setPen(Qt::black); painter->drawLine(hleft, hright); painter->drawLine(vtop, vbottom); } if (m_mode == Mode::Pages && m_pageIndex != -1) { PDFWidgetSnapshot snapshot = getProxy()->getSnapshot(); if (snapshot.hasPage(m_pageIndex)) { const PDFWidgetSnapshot::SnapshotItem* snapshotItem = snapshot.getPageSnapshot(m_pageIndex); painter->fillRect(snapshotItem->rect, m_selectionRectangleColor); } } } void PDFPickTool::mousePressEvent(QWidget* widget, QMouseEvent* event) { Q_UNUSED(widget); event->accept(); if (event->button() == Qt::LeftButton) { switch (m_mode) { case Mode::Pages: { QPointF pagePoint; PDFInteger pageIndex = getProxy()->getPageUnderPoint(m_snapper.getSnappedPoint().toPoint(), &pagePoint); if (pageIndex != -1) { m_pageIndex = pageIndex; Q_EMIT pagePicked(pageIndex); } break; } case Mode::Points: case Mode::Rectangles: { // Try to perform pick point QPointF pagePoint; PDFInteger pageIndex = getProxy()->getPageUnderPoint(m_snapper.getSnappedPoint().toPoint(), &pagePoint); if (pageIndex != -1 && // We have picked some point on page (m_pageIndex == -1 || m_pageIndex == pageIndex)) // We are under current page { m_pageIndex = pageIndex; m_pickedPoints.push_back(pagePoint); m_snapper.setReferencePoint(pageIndex, pagePoint); // Emit signal about picked point Q_EMIT pointPicked(pageIndex, pagePoint); if (m_mode == Mode::Rectangles && m_pickedPoints.size() == 2) { QPointF first = m_pickedPoints.front(); QPointF second = m_pickedPoints.back(); const qreal xMin = qMin(first.x(), second.x()); const qreal xMax = qMax(first.x(), second.x()); const qreal yMin = qMin(first.y(), second.y()); const qreal yMax = qMax(first.y(), second.y()); QRectF pageRectangle(xMin, yMin, xMax - xMin, yMax - yMin); Q_EMIT rectanglePicked(pageIndex, pageRectangle); // We must reset tool, to pick next rectangle resetTool(); } buildSnapData(); Q_EMIT getProxy()->repaintNeeded(); } break; } case pdf::PDFPickTool::Mode::Images: { // Try to perform pick image if (const PDFSnapper::ViewportSnapImage* snappedImage = m_snapper.getSnappedImage()) { Q_EMIT imagePicked(snappedImage->image); } break; } default: { Q_ASSERT(false); break; } } } else if (event->button() == Qt::RightButton && m_mode != Mode::Images) { // Reset tool to enable new picking (right button means reset the tool) resetTool(); } } void PDFPickTool::mouseReleaseEvent(QWidget* widget, QMouseEvent* event) { Q_UNUSED(widget); event->accept(); } void PDFPickTool::mouseMoveEvent(QWidget* widget, QMouseEvent* event) { Q_UNUSED(widget); event->accept(); QPoint mousePos = event->pos(); if (m_mousePosition != mousePos) { m_mousePosition = mousePos; m_snapper.updateSnappedPoint(m_mousePosition); // Update page index, if we are picking pages. if (m_mode == Mode::Pages) { QPointF pagePoint; m_pageIndex = getProxy()->getPageUnderPoint(m_snapper.getSnappedPoint().toPoint(), &pagePoint); } Q_EMIT getProxy()->repaintNeeded(); } } QPointF PDFPickTool::getSnappedPoint() const { return m_snapper.getSnappedPoint(); } void PDFPickTool::setCustomSnapPoints(PDFInteger pageIndex, const std::vector& snapPoints) { if (m_pageIndex == pageIndex) { m_snapper.setCustomSnapPoints(snapPoints); } } void PDFPickTool::setActiveImpl(bool active) { BaseClass::setActiveImpl(active); if (active) { buildSnapData(); } else { // Reset tool to reinitialize it for future use. If tool // is activated, then it should be in initial state. resetTool(); m_snapper.clear(); } } void PDFPickTool::resetTool() { m_pickedPoints.clear(); m_pageIndex = -1; m_snapper.clearReferencePoint(); buildSnapData(); Q_EMIT getProxy()->repaintNeeded(); } void PDFPickTool::buildSnapData() { if (!isActive() || m_mode == Mode::Pages) { return; } if (m_mode == Mode::Images) { // Snap images m_snapper.buildSnapImages(getProxy()->getSnapshot()); } else { // Snap points m_snapper.buildSnapPoints(getProxy()->getSnapshot()); } } QColor PDFPickTool::getSelectionRectangleColor() const { return m_selectionRectangleColor; } void PDFPickTool::setSelectionRectangleColor(QColor selectionRectangleColor) { m_selectionRectangleColor = selectionRectangleColor; } void PDFPickTool::setDrawSelectionRectangle(bool drawSelectionRectangle) { m_drawSelectionRectangle = drawSelectionRectangle; } PDFScreenshotTool::PDFScreenshotTool(PDFDrawWidgetProxy* proxy, QAction* action, QObject* parent) : BaseClass(proxy, action, parent), m_pickTool(nullptr) { m_pickTool = new PDFPickTool(proxy, PDFPickTool::Mode::Rectangles, this); addTool(m_pickTool); connect(m_pickTool, &PDFPickTool::rectanglePicked, this, &PDFScreenshotTool::onRectanglePicked); } void PDFScreenshotTool::onRectanglePicked(PDFInteger pageIndex, QRectF pageRectangle) { PDFWidgetSnapshot snapshot = getProxy()->getSnapshot(); if (const PDFWidgetSnapshot::SnapshotItem* pageSnapshot = snapshot.getPageSnapshot(pageIndex)) { QRect selectedRectangle = pageSnapshot->pageToDeviceMatrix.mapRect(pageRectangle).toRect(); if (selectedRectangle.isValid()) { QImage image(selectedRectangle.size(), QImage::Format_RGB888); { QPainter painter(&image); painter.translate(-selectedRectangle.topLeft()); getProxy()->drawPages(&painter, getProxy()->getWidget()->rect(), getProxy()->getFeatures() | PDFRenderer::DenyExtraGraphics); } QApplication::clipboard()->setImage(image, QClipboard::Clipboard); Q_EMIT messageDisplayRequest(tr("Page contents of size %1 x %2 pixels were copied to the clipboard.").arg(image.width()).arg(image.height()), 5000); } } } PDFExtractImageTool::PDFExtractImageTool(PDFDrawWidgetProxy* proxy, QAction* action, QObject* parent) : BaseClass(proxy, action, parent), m_pickTool(nullptr) { m_pickTool = new PDFPickTool(proxy, PDFPickTool::Mode::Images, this); addTool(m_pickTool); connect(m_pickTool, &PDFPickTool::imagePicked, this, &PDFExtractImageTool::onImagePicked); } void PDFExtractImageTool::updateActions() { // Jakub Melka: We do not call base class implementation here, because // we must verify we have right to extract content (this tool extracts content) if (QAction* action = getAction()) { action->setChecked(isActive()); action->setEnabled(getDocument() && getDocument()->getStorage().getSecurityHandler()->isAllowed(PDFSecurityHandler::Permission::CopyContent)); } } void PDFExtractImageTool::onImagePicked(const QImage& image) { if (!image.isNull()) { QApplication::clipboard()->setImage(image, QClipboard::Clipboard); Q_EMIT messageDisplayRequest(tr("Image of size %1 x %2 pixels was copied to the clipboard.").arg(image.width()).arg(image.height()), 5000); } } PDFSelectTableTool::PDFSelectTableTool(PDFDrawWidgetProxy* proxy, QAction* action, QObject* parent) : BaseClass(proxy, action, parent), m_pickTool(nullptr), m_pageIndex(-1), m_isTransposed(false), m_rotation(PageRotation::None) { m_pickTool = new PDFPickTool(proxy, PDFPickTool::Mode::Rectangles, this); connect(m_pickTool, &PDFPickTool::rectanglePicked, this, &PDFSelectTableTool::onRectanglePicked); setCursor(Qt::CrossCursor); updateActions(); } void PDFSelectTableTool::drawPage(QPainter* painter, PDFInteger pageIndex, const PDFPrecompiledPage* compiledPage, PDFTextLayoutGetter& layoutGetter, const QTransform& pagePointToDevicePointMatrix, const PDFColorConvertor& convertor, QList& errors) const { BaseClass::drawPage(painter, pageIndex, compiledPage, layoutGetter, pagePointToDevicePointMatrix, convertor, errors); if (isTablePicked() && pageIndex == m_pageIndex) { PDFPainterStateGuard guard(painter); QColor color = QColor::fromRgbF(0.0f, 0.0f, 0.5f, 0.2f); QRectF rectangle = pagePointToDevicePointMatrix.mapRect(m_pickedRectangle); const PDFReal lineWidth = PDFWidgetUtils::scaleDPI_x(getProxy()->getWidget(), 2.0); QPen pen(Qt::SolidLine); pen.setWidthF(lineWidth); painter->setPen(convertor.convert(pen)); painter->setBrush(convertor.convert(QBrush(color))); painter->drawRect(rectangle); for (const PDFReal columnPosition : m_horizontalBreaks) { QPointF startPoint(columnPosition, m_pickedRectangle.top()); QPointF endPoint(columnPosition, m_pickedRectangle.bottom()); painter->drawLine(pagePointToDevicePointMatrix.map(startPoint), pagePointToDevicePointMatrix.map(endPoint)); } for (const PDFReal rowPosition : m_verticalBreaks) { QPointF startPoint(m_pickedRectangle.left(), rowPosition); QPointF endPoint(m_pickedRectangle.right(), rowPosition); painter->drawLine(pagePointToDevicePointMatrix.map(startPoint), pagePointToDevicePointMatrix.map(endPoint)); } } } void PDFSelectTableTool::mousePressEvent(QWidget* widget, QMouseEvent* event) { BaseClass::mousePressEvent(widget, event); if (event->isAccepted() || !isTablePicked()) { return; } if (event->button() == Qt::LeftButton || event->button() == Qt::RightButton) { QPointF pagePoint; const PDFInteger pageIndex = getProxy()->getPageUnderPoint(event->pos(), &pagePoint); if (pageIndex != -1 && pageIndex == m_pageIndex && m_pickedRectangle.contains(pagePoint)) { const PDFPage* page = getDocument()->getCatalog()->getPage(pageIndex); bool isSelectingColumns = false; const PageRotation rotation = getPageRotationCombined(page->getPageRotation(), getProxy()->getPageRotation()); switch (rotation) { case pdf::PageRotation::None: case pdf::PageRotation::Rotate180: isSelectingColumns = event->button() == Qt::LeftButton; break; case pdf::PageRotation::Rotate90: case pdf::PageRotation::Rotate270: isSelectingColumns = event->button() == Qt::RightButton; break; default: Q_ASSERT(false); break; } const PDFReal distanceThresholdPixels = PDFWidgetUtils::scaleDPI_x(widget, 7.0); const PDFReal distanceThreshold = getProxy()->transformPixelToDeviceSpace(distanceThresholdPixels); if (isSelectingColumns) { auto it = std::find_if(m_horizontalBreaks.begin(), m_horizontalBreaks.end(), [distanceThreshold, pagePoint](const PDFReal value) { return qAbs(value - pagePoint.x()) < distanceThreshold; }); if (it != m_horizontalBreaks.end()) { m_horizontalBreaks.erase(it); } else if (pagePoint.x() > m_pickedRectangle.left() + distanceThreshold && pagePoint.x() < m_pickedRectangle.right() - distanceThreshold) { m_horizontalBreaks.insert(std::lower_bound(m_horizontalBreaks.begin(), m_horizontalBreaks.end(), pagePoint.x()), pagePoint.x()); } } else { auto it = std::find_if(m_verticalBreaks.begin(), m_verticalBreaks.end(), [distanceThreshold, pagePoint](const PDFReal value) { return qAbs(value - pagePoint.y()) < distanceThreshold; }); if (it != m_verticalBreaks.end()) { m_verticalBreaks.erase(it); } else if (pagePoint.y() > m_pickedRectangle.top() + distanceThreshold && pagePoint.y() < m_pickedRectangle.bottom() - distanceThreshold) { m_verticalBreaks.insert(std::lower_bound(m_verticalBreaks.begin(), m_verticalBreaks.end(), pagePoint.y()), pagePoint.y()); } } Q_EMIT getProxy()->repaintNeeded(); event->accept(); } } } void PDFSelectTableTool::mouseMoveEvent(QWidget* widget, QMouseEvent* event) { BaseClass::mouseMoveEvent(widget, event); if (!event->isAccepted() && isTablePicked()) { QPointF pagePoint; const PDFInteger pageIndex = getProxy()->getPageUnderPoint(event->pos(), &pagePoint); if (pageIndex != -1 && pageIndex == m_pageIndex && m_pickedRectangle.contains(pagePoint)) { setCursor(Qt::CrossCursor); } else { setCursor(Qt::ArrowCursor); } } } void PDFSelectTableTool::shortcutOverrideEvent(QWidget* widget, QKeyEvent* event) { Q_UNUSED(widget); if (event == QKeySequence::Copy) { event->accept(); return; } } void PDFSelectTableTool::keyPressEvent(QWidget* widget, QKeyEvent* event) { Q_UNUSED(widget); if (event == QKeySequence::Copy || event->key() == Qt::Key_Return || event->key() == Qt::Key_Enter) { // Create table cells struct TableCell { size_t row = 0; size_t column = 0; QRectF rectangle; QString text; }; std::vector tableCells; std::vector horizontalBreaks = m_horizontalBreaks; std::vector verticalBreaks = m_verticalBreaks; horizontalBreaks.insert(horizontalBreaks.begin(), m_pickedRectangle.left()); horizontalBreaks.push_back(m_pickedRectangle.right()); verticalBreaks.insert(verticalBreaks.begin(), m_pickedRectangle.top()); verticalBreaks.push_back(m_pickedRectangle.bottom()); tableCells.reserve((horizontalBreaks.size() - 1) * (verticalBreaks.size() - 1)); for (size_t rowIndex = 1; rowIndex < verticalBreaks.size(); ++rowIndex) { const PDFReal top = verticalBreaks[rowIndex - 1]; const PDFReal bottom = verticalBreaks[rowIndex]; const PDFReal height = bottom - top; for (size_t columnIndex = 1; columnIndex < horizontalBreaks.size(); ++columnIndex) { const PDFReal left = horizontalBreaks[columnIndex - 1]; const PDFReal right = horizontalBreaks[columnIndex]; const PDFReal width = right - left; TableCell cell; cell.row = rowIndex; cell.column = columnIndex; cell.rectangle = QRectF(left, top, width, height); PDFTextSelection textSelection = m_textLayout.createTextSelection(m_pageIndex, cell.rectangle.bottomLeft(), cell.rectangle.topRight(), Qt::yellow, true); cell.text = m_textLayout.getTextFromSelection(textSelection, m_pageIndex).trimmed(); cell.text = cell.text.remove(QChar('\n')); tableCells.push_back(cell); } } switch (m_rotation) { case PageRotation::None: { for (TableCell& cell : tableCells) { cell.row = verticalBreaks.size() - cell.row; } break; } default: break; } if (m_isTransposed) { auto comparator = [](const TableCell& left, const TableCell right) { return std::make_pair(left.column, left.row) < std::make_pair(right.column, right.row); }; std::sort(tableCells.begin(), tableCells.end(), comparator); } else { auto comparator = [](const TableCell& left, const TableCell right) { return std::make_pair(left.row, left.column) < std::make_pair(right.row, right.column); }; std::sort(tableCells.begin(), tableCells.end(), comparator); } // Make CSV string QString string; { QTextStream stream(&string, QIODevice::WriteOnly | QIODevice::Text); bool isFirst = true; for (const TableCell& tableCell : tableCells) { if ((!m_isTransposed && tableCell.column == 1) || (m_isTransposed && tableCell.row == 1)) { if (isFirst) { isFirst = false; } else { stream << Qt::endl; } } stream << tableCell.text << ";"; } } QApplication::clipboard()->setText(string); setActive(false); event->accept(); } } void PDFSelectTableTool::setActiveImpl(bool active) { BaseClass::setActiveImpl(active); if (active) { addTool(m_pickTool); } else { // Clear all data setPageIndex(-1); setPickedRectangle(QRectF()); setTextLayout(PDFTextLayout()); m_isTransposed = false; m_horizontalBreaks.clear(); m_verticalBreaks.clear(); m_rotation = PageRotation::None; if (getTopToolstackTool()) { removeTool(); } } } void PDFSelectTableTool::onRectanglePicked(PDFInteger pageIndex, QRectF pageRectangle) { removeTool(); setPageIndex(pageIndex); setPickedRectangle(pageRectangle); setTextLayout(getProxy()->getTextLayoutCompiler()->createTextLayout(pageIndex)); const PDFPage* page = getDocument()->getCatalog()->getPage(pageIndex); const PageRotation rotation = getPageRotationCombined(page->getPageRotation(), getProxy()->getPageRotation()); switch (rotation) { case pdf::PageRotation::None: case pdf::PageRotation::Rotate180: m_isTransposed = false; break; case pdf::PageRotation::Rotate90: case pdf::PageRotation::Rotate270: m_isTransposed = true; break; default: Q_ASSERT(false); break; } m_rotation = rotation; autodetectTableGeometry(); Q_EMIT messageDisplayRequest(tr("Table region was selected. Use left/right mouse buttons to add/remove rows/columns, then use Enter key to copy the table."), 5000); } void PDFSelectTableTool::autodetectTableGeometry() { // Strategy: for horizontal/vertical direction, // detect columns/rows as follow: create overlap // graph for each direction, detect overlap components. // Then, remove "bridges" - rectangles overlapping // two other rectangles, and these two rectangles // does not overlap. These bridges are often header // rows / columns. // Detect text rectangles - divide them by lines std::vector rectangles; for (const PDFTextBlock& textBlock : m_textLayout.getTextBlocks()) { for (const PDFTextLine& textLine : textBlock.getLines()) { QRectF boundingRect = textLine.getBoundingBox().boundingRect(); if (!m_pickedRectangle.contains(boundingRect)) { continue; } rectangles.push_back(boundingRect); } } auto createComponents = [&](bool isHorizontal) -> std::vector { std::vector resultComponents; std::map> isOverlappedGraph; // Create graph of overlapped rectangles for (size_t i = 0; i < rectangles.size(); ++i) { isOverlappedGraph[i].insert(i); for (size_t j = i + 1; j < rectangles.size(); ++j) { const QRectF& leftRect = rectangles[i]; const QRectF& rightRect = rectangles[j]; if ((isHorizontal && isRectangleHorizontallyOverlapped(leftRect, rightRect)) || (!isHorizontal && isRectangleVerticallyOverlapped(leftRect, rightRect))) { isOverlappedGraph[i].insert(j); isOverlappedGraph[j].insert(i); } } } std::set bridges; // Detect bridges, i bridge <=> exist k,j, where isOverlappedGraph[i] has { k, j }, // and isOverlappedGraph[k] has not j. Second check is not neccessary, because // graph is undirectional - if j is in isOverlappedGraph[k], then k is in isOverlappedGraph[j]. for (size_t i = 0; i < rectangles.size(); ++i) { bool isBridge = false; for (size_t k : isOverlappedGraph[i]) { if (k == i) { continue; } for (size_t j : isOverlappedGraph[i]) { if (k == j) { continue; } if (!isOverlappedGraph[k].count(j)) { isBridge = true; break; } } if (isBridge) { break; } } if (isBridge) { bridges.insert(i); } } // Remove bridges from overlapped graph for (const size_t i : bridges) { isOverlappedGraph.erase(i); } for (auto& item : isOverlappedGraph) { std::set result; std::set_difference(item.second.begin(), item.second.end(), bridges.begin(), bridges.end(), std::inserter(result, result.end())); item.second = std::move(result); } // Now, each component is a clique std::set visited; for (auto& item : isOverlappedGraph) { if (visited.count(item.first)) { continue; } visited.insert(item.second.begin(), item.second.end()); QRectF boundingRectangle; for (size_t i : item.second) { boundingRectangle = boundingRectangle.united(rectangles[i]); } if (!boundingRectangle.isEmpty()) { resultComponents.push_back(boundingRectangle); } } return resultComponents; }; // Columns m_horizontalBreaks.clear(); std::vector columnComponents = createComponents(true); std::sort(columnComponents.begin(), columnComponents.end(), [](const auto& left, const auto& right) { return left.center().x() < right.center().x(); }); for (size_t i = 1; i < columnComponents.size(); ++i) { const qreal start = columnComponents[i - 1].right(); const qreal end = columnComponents[i].left(); const qreal middle = (start + end) * 0.5; m_horizontalBreaks.push_back(middle); } // Rows m_verticalBreaks.clear(); std::vector rowComponents = createComponents(false); std::sort(rowComponents.begin(), rowComponents.end(), [](const auto& left, const auto& right) { return left.center().y() < right.center().y(); }); for (size_t i = 1; i < rowComponents.size(); ++i) { const qreal start = rowComponents[i - 1].bottom(); const qreal end = rowComponents[i].top(); const qreal middle = (start + end) * 0.5; m_verticalBreaks.push_back(middle); } } bool PDFSelectTableTool::isTablePicked() const { return m_pageIndex != -1 && !m_pickedRectangle.isEmpty(); } void PDFSelectTableTool::setTextLayout(PDFTextLayout&& newTextLayout) { m_textLayout = std::move(newTextLayout); } void PDFSelectTableTool::setPageIndex(PDFInteger newPageIndex) { m_pageIndex = newPageIndex; } void PDFSelectTableTool::setPickedRectangle(const QRectF& newPickedRectangle) { m_pickedRectangle = newPickedRectangle; } } // namespace pdf