2012-10-02 09:12:39 +02:00
|
|
|
/****************************************************************************
|
2010-09-24 20:16:34 +02:00
|
|
|
**
|
2016-01-15 14:57:40 +01:00
|
|
|
** Copyright (C) 2016 The Qt Company Ltd.
|
|
|
|
|
** Contact: https://www.qt.io/licensing/
|
2010-09-24 20:16:34 +02:00
|
|
|
**
|
2012-10-02 09:12:39 +02:00
|
|
|
** This file is part of Qt Creator.
|
2010-09-24 20:16:34 +02:00
|
|
|
**
|
2012-10-02 09:12:39 +02:00
|
|
|
** 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
|
2016-01-15 14:57:40 +01:00
|
|
|
** a written agreement between you and The Qt Company. For licensing terms
|
|
|
|
|
** and conditions see https://www.qt.io/terms-conditions. For further
|
|
|
|
|
** information use the contact form at https://www.qt.io/contact-us.
|
2010-09-24 20:16:34 +02:00
|
|
|
**
|
2016-01-15 14:57:40 +01:00
|
|
|
** GNU General Public License Usage
|
|
|
|
|
** Alternatively, this file may be used under the terms of the GNU
|
|
|
|
|
** General Public License version 3 as published by the Free Software
|
|
|
|
|
** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT
|
|
|
|
|
** included in the packaging of this file. Please review the following
|
|
|
|
|
** information to ensure the GNU General Public License requirements will
|
|
|
|
|
** be met: https://www.gnu.org/licenses/gpl-3.0.html.
|
2010-12-17 17:14:20 +01:00
|
|
|
**
|
2012-10-02 09:12:39 +02:00
|
|
|
****************************************************************************/
|
2010-09-24 20:16:34 +02:00
|
|
|
|
2017-09-21 12:35:24 +02:00
|
|
|
#include "textutils.h"
|
2010-09-24 20:16:34 +02:00
|
|
|
|
2012-02-15 10:42:41 +01:00
|
|
|
#include <QTextDocument>
|
|
|
|
|
#include <QTextBlock>
|
2010-09-24 20:16:34 +02:00
|
|
|
|
2017-09-21 12:35:24 +02:00
|
|
|
namespace Utils {
|
|
|
|
|
namespace Text {
|
2011-04-15 16:19:23 +02:00
|
|
|
|
|
|
|
|
bool convertPosition(const QTextDocument *document, int pos, int *line, int *column)
|
|
|
|
|
{
|
|
|
|
|
QTextBlock block = document->findBlock(pos);
|
|
|
|
|
if (!block.isValid()) {
|
|
|
|
|
(*line) = -1;
|
|
|
|
|
(*column) = -1;
|
|
|
|
|
return false;
|
|
|
|
|
} else {
|
2018-09-25 16:19:41 +02:00
|
|
|
// line and column are both 1-based
|
2011-04-15 16:19:23 +02:00
|
|
|
(*line) = block.blockNumber() + 1;
|
2018-09-25 16:19:41 +02:00
|
|
|
(*column) = pos - block.position() + 1;
|
2011-04-15 16:19:23 +02:00
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
}
|
2010-09-24 20:16:34 +02:00
|
|
|
|
2018-07-17 23:21:05 +03:00
|
|
|
OptionalLineColumn convertPosition(const QTextDocument *document, int pos)
|
2017-11-23 14:30:09 +01:00
|
|
|
{
|
2018-07-17 23:21:05 +03:00
|
|
|
OptionalLineColumn optional;
|
2017-11-23 14:30:09 +01:00
|
|
|
|
|
|
|
|
QTextBlock block = document->findBlock(pos);
|
|
|
|
|
|
|
|
|
|
if (block.isValid())
|
|
|
|
|
optional.emplace(block.blockNumber() + 1, pos - block.position());
|
|
|
|
|
|
|
|
|
|
return optional;
|
|
|
|
|
}
|
|
|
|
|
|
2018-07-13 12:33:46 +02:00
|
|
|
int positionInText(const QTextDocument *textDocument, int line, int column)
|
2018-06-15 14:33:01 +02:00
|
|
|
{
|
|
|
|
|
// Deduct 1 from line and column since they are 1-based.
|
|
|
|
|
// Column should already be converted from UTF-8 byte offset to the TextEditor column.
|
|
|
|
|
return textDocument->findBlockByNumber(line - 1).position() + column - 1;
|
|
|
|
|
}
|
|
|
|
|
|
2011-04-15 16:19:23 +02:00
|
|
|
QString textAt(QTextCursor tc, int pos, int length)
|
2010-09-24 20:16:34 +02:00
|
|
|
{
|
2011-04-15 16:19:23 +02:00
|
|
|
if (pos < 0)
|
|
|
|
|
pos = 0;
|
|
|
|
|
tc.movePosition(QTextCursor::End);
|
|
|
|
|
if (pos + length > tc.position())
|
|
|
|
|
length = tc.position() - pos;
|
2010-09-24 20:16:34 +02:00
|
|
|
|
2011-04-15 16:19:23 +02:00
|
|
|
tc.setPosition(pos);
|
|
|
|
|
tc.setPosition(pos + length, QTextCursor::KeepAnchor);
|
2010-09-24 20:16:34 +02:00
|
|
|
|
2014-03-22 20:24:55 +02:00
|
|
|
// selectedText() returns U+2029 (PARAGRAPH SEPARATOR) instead of newline
|
|
|
|
|
return tc.selectedText().replace(QChar::ParagraphSeparator, QLatin1Char('\n'));
|
2011-04-15 16:19:23 +02:00
|
|
|
}
|
2010-09-24 20:16:34 +02:00
|
|
|
|
2016-08-04 15:26:53 +02:00
|
|
|
QTextCursor selectAt(QTextCursor textCursor, uint line, uint column, uint length)
|
|
|
|
|
{
|
|
|
|
|
if (line < 1)
|
|
|
|
|
line = 1;
|
|
|
|
|
|
|
|
|
|
if (column < 1)
|
|
|
|
|
column = 1;
|
|
|
|
|
|
|
|
|
|
textCursor.setPosition(0);
|
|
|
|
|
textCursor.movePosition(QTextCursor::NextBlock, QTextCursor::MoveAnchor, line - 1);
|
|
|
|
|
textCursor.movePosition(QTextCursor::NextCharacter,QTextCursor::MoveAnchor, column + length - 1 );
|
|
|
|
|
|
|
|
|
|
textCursor.movePosition(QTextCursor::PreviousCharacter,QTextCursor::KeepAnchor, length);
|
|
|
|
|
|
|
|
|
|
return textCursor;
|
|
|
|
|
}
|
|
|
|
|
|
2015-12-20 19:59:00 +01:00
|
|
|
QTextCursor flippedCursor(const QTextCursor &cursor)
|
|
|
|
|
{
|
|
|
|
|
QTextCursor flipped = cursor;
|
|
|
|
|
flipped.clearSelection();
|
|
|
|
|
flipped.setPosition(cursor.anchor(), QTextCursor::KeepAnchor);
|
|
|
|
|
return flipped;
|
|
|
|
|
}
|
|
|
|
|
|
2017-06-29 16:35:48 +02:00
|
|
|
static bool isValidIdentifierChar(const QChar &c)
|
|
|
|
|
{
|
|
|
|
|
return c.isLetter()
|
|
|
|
|
|| c.isNumber()
|
|
|
|
|
|| c == QLatin1Char('_')
|
|
|
|
|
|| c.isHighSurrogate()
|
|
|
|
|
|| c.isLowSurrogate();
|
|
|
|
|
}
|
|
|
|
|
|
2017-08-03 16:43:38 +02:00
|
|
|
static bool isAfterOperatorKeyword(QTextCursor cursor)
|
|
|
|
|
{
|
|
|
|
|
cursor.movePosition(QTextCursor::PreviousWord);
|
|
|
|
|
cursor.movePosition(QTextCursor::EndOfWord, QTextCursor::KeepAnchor);
|
|
|
|
|
return cursor.selectedText() == "operator";
|
|
|
|
|
}
|
|
|
|
|
|
2017-06-29 16:35:48 +02:00
|
|
|
QTextCursor wordStartCursor(const QTextCursor &textCursor)
|
|
|
|
|
{
|
|
|
|
|
const int originalPosition = textCursor.position();
|
|
|
|
|
QTextCursor cursor(textCursor);
|
|
|
|
|
cursor.movePosition(QTextCursor::StartOfWord);
|
|
|
|
|
const int wordStartPosition = cursor.position();
|
|
|
|
|
|
|
|
|
|
if (originalPosition == wordStartPosition) {
|
|
|
|
|
// Cursor is not on an identifier, check whether we are right after one.
|
|
|
|
|
const QChar c = textCursor.document()->characterAt(originalPosition - 1);
|
|
|
|
|
if (isValidIdentifierChar(c))
|
|
|
|
|
cursor.movePosition(QTextCursor::PreviousWord);
|
|
|
|
|
}
|
2017-08-03 16:43:38 +02:00
|
|
|
if (isAfterOperatorKeyword(cursor))
|
|
|
|
|
cursor.movePosition(QTextCursor::PreviousWord);
|
2017-06-29 16:35:48 +02:00
|
|
|
|
|
|
|
|
return cursor;
|
|
|
|
|
}
|
|
|
|
|
|
2017-09-21 12:35:24 +02:00
|
|
|
} // Text
|
|
|
|
|
} // Utils
|