summaryrefslogtreecommitdiffstats
path: root/src/pdfquick/qquickpdfselection.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'src/pdfquick/qquickpdfselection.cpp')
-rw-r--r--src/pdfquick/qquickpdfselection.cpp172
1 files changed, 89 insertions, 83 deletions
diff --git a/src/pdfquick/qquickpdfselection.cpp b/src/pdfquick/qquickpdfselection.cpp
index 48ff0d8bd..4776cb8b4 100644
--- a/src/pdfquick/qquickpdfselection.cpp
+++ b/src/pdfquick/qquickpdfselection.cpp
@@ -1,38 +1,5 @@
-/****************************************************************************
-**
-** Copyright (C) 2020 The Qt Company Ltd.
-** Contact: http://www.qt.io/licensing/
-**
-** This file is part of the QtPDF module of the Qt Toolkit.
-**
-** $QT_BEGIN_LICENSE:LGPL3$
-** Commercial License Usage
-** Licensees holding valid commercial Qt licenses may use this file in
-** accordance with the commercial license agreement provided with the
-** Software or, alternatively, in accordance with the terms contained in
-** a written agreement between you and The Qt Company. For licensing terms
-** and conditions see http://www.qt.io/terms-conditions. For further
-** information use the contact form at http://www.qt.io/contact-us.
-**
-** GNU Lesser General Public License Usage
-** Alternatively, this file may be used under the terms of the GNU Lesser
-** General Public License version 3 as published by the Free Software
-** Foundation and appearing in the file LICENSE.LGPLv3 included in the
-** packaging of this file. Please review the following information to
-** ensure the GNU Lesser General Public License version 3 requirements
-** will be met: https://www.gnu.org/licenses/lgpl.html.
-**
-** GNU General Public License Usage
-** Alternatively, this file may be used under the terms of the GNU
-** General Public License version 2.0 or later as published by the Free
-** Software Foundation and appearing in the file LICENSE.GPL included in
-** the packaging of this file. Please review the following information to
-** ensure the GNU General Public License version 2.0 requirements will be
-** met: http://www.gnu.org/licenses/gpl-2.0.html.
-**
-** $QT_END_LICENSE$
-**
-****************************************************************************/
+// Copyright (C) 2020 The Qt Company Ltd.
+// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only
#include "qquickpdfselection_p.h"
#include "qquickpdfdocument_p.h"
@@ -56,13 +23,14 @@ static const QRegularExpression WordDelimiter(QStringLiteral("\\s"));
//! \instantiates QQuickPdfSelection
\inqmlmodule QtQuick.Pdf
\ingroup pdf
+ \inherits Item
\brief A representation of a text selection within a PDF Document.
\since 5.15
PdfSelection provides the text string and its geometry within a bounding box
from one point to another.
- To modify the selection using the mouse, bind \l fromPoint and \l toPoint
+ To modify the selection using the mouse, bind \l from and \l to
to the suitable properties of an input handler so that they will be set to
the positions where the drag gesture begins and ends, respectively; and
bind the \l hold property so that it will be set to \c true during the drag
@@ -78,11 +46,19 @@ QQuickPdfSelection::QQuickPdfSelection(QQuickItem *parent)
{
#if QT_CONFIG(im)
setFlags(ItemIsFocusScope | ItemAcceptsInputMethod);
- // workaround to get Copy instead of Paste on the popover menu (QTBUG-83811)
- setProperty("qt_im_readonly", QVariant(true));
#endif
}
+/*!
+ \internal
+*/
+QQuickPdfSelection::~QQuickPdfSelection() = default;
+
+/*!
+ \qmlproperty PdfDocument PdfSelection::document
+
+ This property holds the PDF document in which to select text.
+*/
QQuickPdfDocument *QQuickPdfSelection::document() const
{
return m_document;
@@ -118,8 +94,8 @@ void QQuickPdfSelection::setDocument(QQuickPdfDocument *document)
PdfSelection {
id: selection
document: doc
- fromPoint: textSelectionDrag.centroid.pressPosition
- toPoint: textSelectionDrag.centroid.position
+ from: textSelectionDrag.centroid.pressPosition
+ to: textSelectionDrag.centroid.position
hold: !textSelectionDrag.active
}
Shape {
@@ -143,27 +119,39 @@ QList<QPolygonF> QQuickPdfSelection::geometry() const
return m_geometry;
}
+/*!
+ \qmlmethod void PdfSelection::clear()
+
+ Clears the current selection.
+*/
void QQuickPdfSelection::clear()
{
m_hitPoint = QPointF();
- m_fromPoint = QPointF();
- m_toPoint = QPointF();
+ m_from = QPointF();
+ m_to = QPointF();
m_heightAtAnchor = 0;
m_heightAtCursor = 0;
m_fromCharIndex = -1;
m_toCharIndex = -1;
m_text.clear();
m_geometry.clear();
- emit fromPointChanged();
- emit toPointChanged();
+ emit fromChanged();
+ emit toChanged();
emit textChanged();
emit selectedAreaChanged();
QGuiApplication::inputMethod()->update(Qt::ImQueryInput);
}
+/*!
+ \qmlmethod void PdfSelection::selectAll()
+
+ Selects all text on the current \l page.
+*/
void QQuickPdfSelection::selectAll()
{
- QPdfSelection sel = m_document->m_doc.getAllText(m_page);
+ if (!m_document)
+ return;
+ QPdfSelection sel = m_document->document()->getAllText(m_page);
if (sel.text() != m_text) {
m_text = sel.text();
if (QGuiApplication::clipboard()->supportsSelection())
@@ -179,11 +167,11 @@ void QQuickPdfSelection::selectAll()
m_fromCharIndex = sel.startIndex();
m_toCharIndex = sel.endIndex();
if (sel.bounds().isEmpty()) {
- m_fromPoint = QPointF();
- m_toPoint = QPointF();
+ m_from = QPointF();
+ m_to = QPointF();
} else {
- m_fromPoint = sel.bounds().first().boundingRect().topLeft() * m_renderScale;
- m_toPoint = sel.bounds().last().boundingRect().bottomRight() * m_renderScale - QPointF(0, m_heightAtCursor);
+ m_from = sel.bounds().first().boundingRect().topLeft() * m_renderScale;
+ m_to = sel.bounds().last().boundingRect().bottomRight() * m_renderScale - QPointF(0, m_heightAtCursor);
}
QGuiApplication::inputMethod()->update(Qt::ImCursorRectangle | Qt::ImAnchorRectangle);
@@ -196,21 +184,25 @@ void QQuickPdfSelection::keyReleaseEvent(QKeyEvent *ev)
qCDebug(qLcIm) << "release" << ev;
const auto &allText = pageText();
if (ev == QKeySequence::MoveToPreviousWord) {
+ if (!m_document)
+ return;
// iOS sends MoveToPreviousWord first to get to the beginning of the word,
// and then SelectNextWord to select the whole word.
- int i = allText.lastIndexOf(WordDelimiter, m_fromCharIndex - allText.length());
+ int i = allText.lastIndexOf(WordDelimiter, m_fromCharIndex - allText.size());
if (i < 0)
i = 0;
else
i += 1; // don't select the space before the word
- auto sel = m_document->m_doc.getSelectionAtIndex(m_page, i, m_text.length() + m_fromCharIndex - i);
+ auto sel = m_document->document()->getSelectionAtIndex(m_page, i, m_text.size() + m_fromCharIndex - i);
update(sel);
QGuiApplication::inputMethod()->update(Qt::ImAnchorRectangle);
} else if (ev == QKeySequence::SelectNextWord) {
+ if (!m_document)
+ return;
int i = allText.indexOf(WordDelimiter, m_toCharIndex);
if (i < 0)
- i = allText.length(); // go to the end of m_textAfter
- auto sel = m_document->m_doc.getSelectionAtIndex(m_page, m_fromCharIndex, m_text.length() + i - m_toCharIndex);
+ i = allText.size(); // go to the end of m_textAfter
+ auto sel = m_document->document()->getSelectionAtIndex(m_page, m_fromCharIndex, m_text.size() + i - m_toCharIndex);
update(sel);
QGuiApplication::inputMethod()->update(Qt::ImCursorRectangle);
} else if (ev == QKeySequence::Copy) {
@@ -226,7 +218,9 @@ void QQuickPdfSelection::inputMethodEvent(QInputMethodEvent *event)
qCDebug(qLcIm) << "QInputMethodEvent::Cursor: moved to" << attr.start << "len" << attr.length;
break;
case QInputMethodEvent::Selection: {
- auto sel = m_document->m_doc.getSelectionAtIndex(m_page, attr.start, attr.length);
+ if (!m_document)
+ return;
+ auto sel = m_document->document()->getSelectionAtIndex(m_page, attr.start, attr.length);
update(sel);
qCDebug(qLcIm) << "QInputMethodEvent::Selection: from" << attr.start << "len" << attr.length
<< "result:" << m_fromCharIndex << "->" << m_toCharIndex << sel.boundingRectangle();
@@ -247,16 +241,18 @@ QVariant QQuickPdfSelection::inputMethodQuery(Qt::InputMethodQuery query, const
if (!argument.isNull()) {
qCDebug(qLcIm) << "IM query" << query << "with arg" << argument;
if (query == Qt::ImCursorPosition) {
+ if (!m_document)
+ return {};
// If it didn't move since last time, return the same result.
if (m_hitPoint == argument.toPointF())
return inputMethodQuery(query);
m_hitPoint = argument.toPointF();
- auto tp = m_document->m_doc.d->hitTest(m_page, m_hitPoint / m_renderScale);
+ auto tp = m_document->document()->d->hitTest(m_page, m_hitPoint / m_renderScale);
qCDebug(qLcIm) << "ImCursorPosition hit testing in px" << m_hitPoint << "pt" << (m_hitPoint / m_renderScale)
<< "got char index" << tp.charIndex << "@" << tp.position << "pt," << tp.position * m_renderScale << "px";
if (tp.charIndex >= 0) {
m_toCharIndex = tp.charIndex;
- m_toPoint = tp.position * m_renderScale - QPointF(0, m_heightAtCursor);
+ m_to = tp.position * m_renderScale - QPointF(0, m_heightAtCursor);
m_heightAtCursor = tp.height * m_renderScale;
if (qFuzzyIsNull(m_heightAtAnchor))
m_heightAtAnchor = m_heightAtCursor;
@@ -289,10 +285,10 @@ QVariant QQuickPdfSelection::inputMethodQuery(Qt::InputMethodQuery query) const
ret = m_toCharIndex;
break;
case Qt::ImAnchorRectangle:
- ret = QRectF(m_fromPoint, QSizeF(1, m_heightAtAnchor));
+ ret = QRectF(m_from, QSizeF(1, m_heightAtAnchor));
break;
case Qt::ImCursorRectangle:
- ret = QRectF(m_toPoint, QSizeF(1, m_heightAtCursor));
+ ret = QRectF(m_to, QSizeF(1, m_heightAtCursor));
break;
case Qt::ImSurroundingText:
ret = QVariant(pageText());
@@ -321,6 +317,7 @@ QVariant QQuickPdfSelection::inputMethodQuery(Qt::InputMethodQuery query) const
case Qt::ImPlatformData:
break;
case Qt::ImReadOnly:
+ ret = true;
break;
case Qt::ImQueryInput:
case Qt::ImQueryAll:
@@ -335,7 +332,9 @@ QVariant QQuickPdfSelection::inputMethodQuery(Qt::InputMethodQuery query) const
const QString &QQuickPdfSelection::pageText() const
{
if (m_pageTextDirty) {
- m_pageText = m_document->m_doc.getAllText(m_page).text();
+ if (!m_document)
+ return m_pageText;
+ m_pageText = m_document->document()->getAllText(m_page).text();
m_pageTextDirty = false;
}
return m_pageText;
@@ -345,8 +344,8 @@ void QQuickPdfSelection::resetPoints()
{
bool wasHolding = m_hold;
m_hold = false;
- setFromPoint(QPointF());
- setToPoint(QPointF());
+ setFrom(QPointF());
+ setTo(QPointF());
m_hold = wasHolding;
}
@@ -377,7 +376,7 @@ void QQuickPdfSelection::setPage(int page)
\qmlproperty real PdfSelection::renderScale
\brief The ratio from points to pixels at which the page is rendered.
- This is used to scale \l fromPoint and \l toPoint to find ranges of
+ This is used to scale \l from and \l to to find ranges of
selected characters in the document, because positions within the document
are always given in points.
*/
@@ -388,6 +387,11 @@ qreal QQuickPdfSelection::renderScale() const
void QQuickPdfSelection::setRenderScale(qreal scale)
{
+ if (qFuzzyIsNull(scale)) {
+ qWarning() << "PdfSelection.renderScale cannot be set to 0.";
+ return;
+ }
+
if (qFuzzyCompare(scale, m_renderScale))
return;
@@ -397,7 +401,7 @@ void QQuickPdfSelection::setRenderScale(qreal scale)
}
/*!
- \qmlproperty point PdfSelection::fromPoint
+ \qmlproperty point PdfSelection::from
The beginning location, in pixels from the upper-left corner of the page,
from which to find selected text. This can be bound to the
@@ -405,41 +409,41 @@ void QQuickPdfSelection::setRenderScale(qreal scale)
the position where the user presses the mouse button and begins dragging,
for example.
*/
-QPointF QQuickPdfSelection::fromPoint() const
+QPointF QQuickPdfSelection::from() const
{
- return m_fromPoint;
+ return m_from;
}
-void QQuickPdfSelection::setFromPoint(QPointF fromPoint)
+void QQuickPdfSelection::setFrom(QPointF from)
{
- if (m_hold || m_fromPoint == fromPoint)
+ if (m_hold || m_from == from)
return;
- m_fromPoint = fromPoint;
- emit fromPointChanged();
+ m_from = from;
+ emit fromChanged();
updateResults();
}
/*!
- \qmlproperty point PdfSelection::toPoint
+ \qmlproperty point PdfSelection::to
The ending location, in pixels from the upper-left corner of the page,
from which to find selected text. This can be bound to the
\c centroid.position of a \l DragHandler to end selection of text at the
position where the user is currently dragging the mouse, for example.
*/
-QPointF QQuickPdfSelection::toPoint() const
+QPointF QQuickPdfSelection::to() const
{
- return m_toPoint;
+ return m_to;
}
-void QQuickPdfSelection::setToPoint(QPointF toPoint)
+void QQuickPdfSelection::setTo(QPointF to)
{
- if (m_hold || m_toPoint == toPoint)
+ if (m_hold || m_to == to)
return;
- m_toPoint = toPoint;
- emit toPointChanged();
+ m_to = to;
+ emit toChanged();
updateResults();
}
@@ -447,7 +451,7 @@ void QQuickPdfSelection::setToPoint(QPointF toPoint)
\qmlproperty bool PdfSelection::hold
Controls whether to hold the existing selection regardless of changes to
- \l fromPoint and \l toPoint. This property can be set to \c true when the mouse
+ \l from and \l to. This property can be set to \c true when the mouse
or touchpoint is released, so that the selection is not lost due to the
point bindings changing.
*/
@@ -491,8 +495,8 @@ void QQuickPdfSelection::updateResults()
{
if (!m_document)
return;
- QPdfSelection sel = m_document->document().getSelection(m_page,
- m_fromPoint / m_renderScale, m_toPoint / m_renderScale);
+ QPdfSelection sel = m_document->document()->getSelection(m_page,
+ m_from / m_renderScale, m_to / m_renderScale);
update(sel, true);
}
@@ -516,12 +520,12 @@ void QQuickPdfSelection::update(const QPdfSelection &sel, bool textAndGeometryOn
m_fromCharIndex = sel.startIndex();
m_toCharIndex = sel.endIndex();
if (sel.bounds().isEmpty()) {
- m_fromPoint = sel.boundingRectangle().topLeft() * m_renderScale;
- m_toPoint = m_fromPoint;
+ m_from = sel.boundingRectangle().topLeft() * m_renderScale;
+ m_to = m_from;
} else {
Qt::InputMethodQueries toUpdate = {};
QRectF firstLineBounds = sel.bounds().first().boundingRect();
- m_fromPoint = firstLineBounds.topLeft() * m_renderScale;
+ m_from = firstLineBounds.topLeft() * m_renderScale;
if (!qFuzzyCompare(m_heightAtAnchor, firstLineBounds.height())) {
m_heightAtAnchor = firstLineBounds.height() * m_renderScale;
toUpdate.setFlag(Qt::ImAnchorRectangle);
@@ -531,10 +535,12 @@ void QQuickPdfSelection::update(const QPdfSelection &sel, bool textAndGeometryOn
m_heightAtCursor = lastLineBounds.height() * m_renderScale;
toUpdate.setFlag(Qt::ImCursorRectangle);
}
- m_toPoint = lastLineBounds.topRight() * m_renderScale;
+ m_to = lastLineBounds.topRight() * m_renderScale;
if (toUpdate)
QGuiApplication::inputMethod()->update(toUpdate);
}
}
QT_END_NAMESPACE
+
+#include "moc_qquickpdfselection_p.cpp"