我试着学习一些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
答案 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