pyqt中的CodeEditor示例

时间:2015-10-20 18:12:35

标签: python qt python-3.x pyqt4

我试着学习一些Qt(PyQt)。为此,我使用了文档的Code Editor example。当前行的突出显示工作正常。但是,行号不显示。

实际上甚至没有调用LineNumberArea.paintEvent。很容易也不是CodeEditor.lineNumberAreaPaintEvent。据我所知,数字条的paintEvent应定期调用。或者至少在有updateRequest或滚动事件时(由CodeEditor.updateLineNumberArea调用)。

这是我将代码从c ++移植到python:

import sys
from PyQt4.QtGui import *
from PyQt4.QtCore import *

import numpy as np

class LineNumberArea(QWidget):
    def __init__(self, editor):
        super().__init__()
        self.editor = editor


    def sizeHint(self):
        return Qsize(self.editor.lineNumberAreaWidth(), 0)


    def paintEvent(self, event):
        print('LineNumberArea.paintEvent')
        self.editor.lineNumberAreaPaintEvent(event)


class CodeEditor(QPlainTextEdit):
    def __init__(self):
        super().__init__()
        self.lineNumberArea = LineNumberArea(self)

        self.connect(self, SIGNAL('blockCountChanged(int)'), self.updateLineNumberAreaWidth)
        self.connect(self, SIGNAL('updateRequest(QRect,int)'), self.updateLineNumberArea)
        self.connect(self, SIGNAL('cursorPositionChanged()'), self.highlightCurrentLine)

        self.updateLineNumberAreaWidth(0)


    def lineNumberAreaWidth(self):
        """ This method has been slightly modified (use of log and uses actual
        font rather than standart.) """
        n_lines = self.blockCount()
        digits = np.ceil(np.log10(n_lines)) + 1
        return digits * QFontMetrics(self.font()).width('9') + 3


    def updateLineNumberAreaWidth(self, _):
        print('CodeEditor.updateLineNumberAreaWidth: margin = {}'.format(self.lineNumberAreaWidth()))
        self.setViewportMargins(self.lineNumberAreaWidth(), 0, 0, 0)


    def updateLineNumberArea(self, rect, dy):
        print('CodeEditor.updateLineNumberArea: rect = {}, dy = {}'.format(rect, dy))

        if dy:
            self.lineNumberArea.scroll(0, dy)
        else:
            self.lineNumberArea.update(0, rect.y(), self.lineNumberArea.width(),
                                       rect.height())

        print('CodeEditor.updateLineNumberArea: rect.contains(self.viewport().rect()) = {}'.format(rect.contains(self.viewport().rect())))
        if rect.contains(self.viewport().rect()):
            self.updateLineNumberAreaWidth(0)


    def resizeEvent(self, event):
        super().resizeEvent(event)

        cr = self.contentsRect();
        self.lineNumberArea.setGeometry(QRect(cr.left(), cr.top(),
                                        self.lineNumberAreaWidth(), cr.height()))

    def lineNumberAreaPaintEvent(self, event):
        print('CodeEditor.lineNumberAreaPaintEvent')
        painter(self.lineNumberArea)
        painter.fillRect(event.rect(), Qt.lightGray)

        block = self.firstVisibleBlock()
        blockNumber = block.blockNumber()
        top = self.blockBoundingGeometry(block).translated(self.contentOffset()).top()
        bottom = top + self.blockBoundingRect(block).height()

        # Just to make sure I use the right font
        height = QFontMetrics(self.font()).height()
        while block.isValid() and (top <= event.rect().bottom()):
            if block.isVisible() and (bottom >= event.rect().top()):
                number = str(blockNumber + 1)
                painter.setPen(Qt.black)
                painter.drawText(0, top, lineNumberArea.width(), height,
                                 Qt.AlignRight, number)

            block = block.next()
            top = bottom
            bottom = top + self.blockBoundingRect(block).height()
            blockNumber += 1


    def highlightCurrentLine(self):
        extraSelections = []

        if not self.isReadOnly():
            selection = QTextEdit.ExtraSelection()

            lineColor = QColor(Qt.yellow).lighter(160)

            selection.format.setBackground(lineColor)
            selection.format.setProperty(QTextFormat.FullWidthSelection, True)
            selection.cursor = self.textCursor()
            selection.cursor.clearSelection()
            extraSelections.append(selection)
        self.setExtraSelections(extraSelections)


if __name__ == "__main__":
    app = QApplication(sys.argv)

    txt = CodeEditor()
    txt.show()

    sys.exit(app.exec_())

非常感谢任何帮助。

如果重要: python:3.4.3,PyQt:4.8.6,操作系统:RHEL 6

1 个答案:

答案 0 :(得分:3)

你的例子看起来大多正确。缺少绘制事件的问题是由于未在LineNumberArea小部件上设置父级而引起的。所以你只需要:

class LineNumberArea(QWidget):
    def __init__(self, editor):
        super().__init__(editor)

此外,lineNumberAreaPaintEvent方法存在一些问题,但很容易修复:

def lineNumberAreaPaintEvent(self, event):
    # missing constructor
    painter = QPainter(self.lineNumberArea)
    ...
    # no need to use QFontMetrics
    height = self.fontMetrics().height()
    ...
            # missing self
            painter.drawText(0, top, self.lineNumberArea.width(), height,
                             Qt.AlignRight, number)

我不确定您要对lineNumberAreaWidth的实施做些什么,因为它似乎没有给出正确的结果。原始实现完全按预期工作(但也快得多):

def lineNumberAreaWidth(self):
    digits = 1
    count = max(1, self.blockCount())
    while count >= 10:
        count /= 10
        digits += 1
    space = 3 + self.fontMetrics().width('9') * digits
    return space