QTextLayout::drawCursor() 在子类化 QAbstractTextDocumentLayout 时不起作用

问题描述 投票:0回答:3

目前

QTextEdit
QPlainTextEdit
不符合我的要求,所以我需要子类化
QAbstractTextDocumentLayout
来提供文档的自定义布局。我参考了很多
QPlainTextDocumentLayout
QTextDocumentLayout
,最终得到了一个简单的布局来显示文本。但是,我在
QTextEdit
中看不到光标,它应该在闪烁。我需要帮助来解决这个问题。

我使用的是Qt 5.9.1。简单的项目是here

draw()
VTextDocumentLayout
函数如下所示:

void VTextDocumentLayout::draw(QPainter *p_painter, const PaintContext &p_context)
{
    qDebug() << "VTextDocumentLayout draw()" << p_context.clip << p_context.cursorPosition << p_context.selections.size();
    // Find out the blocks.
    int first, last;
    blockRangeFromRect(p_context.clip, first, last);
    if (first == -1) {
        return;
    }

    QTextDocument *doc = document();
    QPointF offset(m_margin, m_blocks[first].m_offset);
    QTextBlock block = doc->findBlockByNumber(first);
    QTextBlock lastBlock = doc->findBlockByNumber(last);
    qreal maximumWidth = m_width;

    while (block.isValid()) {
        const BlockInfo &info = m_blocks[block.blockNumber()];
        const QRectF &rect = info.m_rect;
        QTextLayout *layout = block.layout();

        if (!block.isVisible()) {
            offset.ry() += rect.height();
            if (block == lastBlock) {
                break;
            }

            block = block.next();
            continue;
        }

        QTextBlockFormat blockFormat = block.blockFormat();
        QBrush bg = blockFormat.background();
        if (bg != Qt::NoBrush) {
            fillBackground(p_painter, rect, bg);
        }

        auto selections = formatRangeFromSelection(block, p_context.selections);

        QPen oldPen = p_painter->pen();
        p_painter->setPen(p_context.palette.color(QPalette::Text));

        layout->draw(p_painter,
                     offset,
                     selections,
                     p_context.clip.isValid() ? p_context.clip : QRectF());

        // Draw the cursor.
        int blpos = block.position();
        int bllen = block.length();
        bool drawCursor = p_context.cursorPosition >= blpos
                          && p_context.cursorPosition < blpos + bllen;

        Q_ASSERT(p_context.cursorPosition >= -1);
        if (drawCursor) {
            int cpos = p_context.cursorPosition;
            cpos -= blpos;

            qDebug() << "draw cursor" << block.blockNumber() << blpos << bllen << p_context.cursorPosition << cpos;
            layout->drawCursor(p_painter, offset, cpos);
        }

        p_painter->setPen(oldPen);

        offset.ry() += rect.height();
        if (block == lastBlock) {
            break;
        }

        block = block.next();
    }
}

我调用了

layout->drawCursor()
来绘制光标,但这个函数似乎什么也没做。

如有任何帮助,我们将不胜感激!谢谢!

更新: 添加日志如下:

VTextDocumentLayout draw() QRectF(67,0 9x13) 19 0
block range 0 1
draw cursor 1 5 15 19 14
blockBoundingRect() 1 13 QRectF(0,0 75x17)
VTextDocumentLayout draw() QRectF(67,0 9x13) -1 0
block range 0 1
blockBoundingRect() 1 13 QRectF(0,0 75x17)
VTextDocumentLayout draw() QRectF(67,0 9x13) 19 0

在Linux中运行这个项目时,我看不到光标。然而,当在 Windows 中运行它时,我可以看到光标但不闪烁。

更新: 似乎

QTextEdit
将错误的剪辑矩形传递给布局。如果我只插入一行文本(文档中仅插入一个块),我可以看到闪烁的光标。很奇怪!!!

qt qtextedit qtextdocument
3个回答
0
投票

PaintContext
cursorPosition
的默认值为-1。

http://doc.qt.io/qt-4.8/qabstracttextdocumentlayout-paintcontext.html#cursorPosition-var

如果没有为光标位置提供任何默认值(它是一个公共变量),那么我的猜测是值保持为-1,并且

drawCursor
始终为
false
(因为块位置索引最小值为零)最坏的情况)。

尝试将一些默认值设置为

PaintContext::cursorPosition
,这可能会显示光标。


0
投票

子类化

QAbstractTextDocumentLayout
时,
blockBoundingRect()
应返回块的 geometry,而不仅仅是像
QPlainTextDocumentLayout
那样返回矩形。

总而言之,

QPlainTextDocumentLayout
提供了一个用于子类化QAbstractTextDocumentLayout
BAD
示例。


0
投票

我的操作系统是Windows。

在Linux中运行这个项目时,我看不到光标。然而,当在 Windows 中运行它时,我可以看到光标但不闪烁。

更新:QTextEdit 似乎将错误的剪辑矩形传递给布局。如果我只插入一行文本(文档中仅插入一个块),我可以看到闪烁的光标。很奇怪!!!

您需要在

draw
函数的最后调用更新信号。

emit QAbstactTextDocumentLayout::update();

是的,正如你所说,这很奇怪。先看几行,没有问题。但是,如果您对文档进行了一定程度的编辑,绘图功能就会停止。那是因为你的光标不会闪烁。

© www.soinside.com 2019 - 2024. All rights reserved.