在 QPlainTextEdit 中突出显示部分文本

Posted

技术标签:

【中文标题】在 QPlainTextEdit 中突出显示部分文本【英文标题】:Highlighting portions of text in QPlainTextEdit 【发布时间】:2019-12-29 08:54:05 【问题描述】:

我在 QPlainTextEdit 中有一个列表,想要突出显示特定的行,比如第 5 行。

我查看了各种教程和示例,但 PyQt5 的文档似乎相当稀疏​​。

可行的代码,我想突出显示行。 对于另一个小部件的建议,我也很感激。我不想编辑这个列表,只是显示它并突出显示行。

import sys
from PyQt5.QtWidgets import *

app = QApplication(sys.argv)
main = QWidget()
main.resize(250, 150)
main.size
tbox = QPlainTextEdit(main)
for nr in range(1,5):
    tbox.appendPlainText('%d'%nr)

## highlight line 2
## wait a while
## unhighlight line 2
## highlight line 4
main.show()
sys.exit(app.exec_())

【问题讨论】:

查看编辑后的帖子。 【参考方案1】:

如果突出显示取决于文本,那么@oetzi 的解决方案就是指定的解决方案,因为尽管您删除了行,但相同的文本仍将保持突出显示,如果相反,突出显示仅取决于行的位置,那么可能的解决方案是使用 QSyntaxHighlighter。

在下面的示例中,您可以输入以空格分隔的整数,以指示将突出显示的行的位置(位置从 0 开始):

import sys
from PyQt5.QtCore import pyqtSlot, QRegExp
from PyQt5.QtGui import QColor, QRegExpValidator, QSyntaxHighlighter, QTextCharFormat
from PyQt5.QtWidgets import (
    QApplication,
    QLineEdit,
    QPlainTextEdit,
    QVBoxLayout,
    QWidget,
)


class SyntaxHighlighter(QSyntaxHighlighter):
    def __init__(self, parent):
        super(SyntaxHighlighter, self).__init__(parent)
        self._highlight_lines = dict()

    def highlight_line(self, line, fmt):
        if isinstance(line, int) and line >= 0 and isinstance(fmt, QTextCharFormat):
            self._highlight_lines[line] = fmt
            tb = self.document().findBlockByLineNumber(line)
            self.rehighlightBlock(tb)

    def clear_highlight(self):
        self._highlight_lines = dict()
        self.rehighlight()

    def highlightBlock(self, text):
        line = self.currentBlock().blockNumber()
        fmt = self._highlight_lines.get(line)
        if fmt is not None:
            self.setFormat(0, len(text), fmt)


class Widget(QWidget):
    def __init__(self, parent=None):
        super(Widget, self).__init__(parent)

        self._lineedit = QLineEdit(textChanged=self.onTextChanged)
        regex_validator = QRegExpValidator(QRegExp(r"[0-9 ]+"))
        self._lineedit.setValidator(regex_validator)

        self._plaintextedit = QPlainTextEdit()

        self._highlighter = SyntaxHighlighter(self._plaintextedit.document())

        lay = QVBoxLayout(self)
        lay.addWidget(self._lineedit)
        lay.addWidget(self._plaintextedit)

        for i in range(10):
            self._plaintextedit.appendPlainText("line %d" % i)

        self.resize(320, 240)

    @pyqtSlot(str)
    def onTextChanged(self, text):
        fmt = QTextCharFormat()
        fmt.setBackground(QColor("yellow"))
        self._highlighter.clear_highlight()
        for e in text.split():
            line = int(e)
            self._highlighter.highlight_line(line, fmt)


if __name__ == "__main__":

    app = QApplication(sys.argv)
    w = Widget()
    w.show()
    sys.exit(app.exec_())

【讨论】:

【参考方案2】:
import sys
from PyQt5.QtWidgets import (QPlainTextEdit, QApplication, QWidget)

app = QApplication(sys.argv)
main = QWidget()
main.resize(250, 150)
main.size
tbox = QPlainTextEdit(main)
condition = 0
for nr in range(1, 5):
    if condition % 2 == 0:
        tbox.appendhtml(f"<span style='background-color: yellow;'>nr</p>")
    else:
        tbox.appendHtml(f"<span style='background-color: white;'>nr</p>")
    # tbox.appendPlainText('%d' % nr)
    condition = condition + 1
main.show()
sys.exit(app.exec_())

【讨论】:

eyllanesc 的解决方案符合我的想法。非常感谢大家。

以上是关于在 QPlainTextEdit 中突出显示部分文本的主要内容,如果未能解决你的问题,请参考以下文章

QPlainTextEdit 强制重绘

突出显示文本区域内的文本

如何在文本突出显示期间保留语法突出显示

如何在textarea中突出显示部分文本

Qt突出显示选定的行会覆盖单个单词的突出显示

在文本区域中突出显示简单的 Javascript?