我正在尝试在 PyQt5 中制作一个具有基本语法突出显示、代码完成和可点击函数和变量的简单文本编辑器。我最大的希望是使用 PyQt5 的 QScintilla 端口
。我在 Eli Bendersky 网站 ( http://eli.thegreenplace.net/2011/04/01/sample-using-qscintilla-with-pyqt
上找到了以下基于 QScintilla 的文本编辑器示例,Victor S. 已将其改编为 PyQt5 )。我认为这个例子是一个很好的起点:
#-------------------------------------------------------------------------
# qsci_simple_pythoneditor.pyw
#
# QScintilla sample with PyQt
#
# Eli Bendersky (eliben@gmail.com)
# This code is in the public domain
#-------------------------------------------------------------------------
import sys
import sip
from PyQt5.QtWidgets import *
from PyQt5.QtCore import *
from PyQt5.QtGui import *
from PyQt5.Qsci import QsciScintilla, QsciLexerPython
class SimplePythonEditor(QsciScintilla):
ARROW_MARKER_NUM = 8
def __init__(self, parent=None):
super(SimplePythonEditor, self).__init__(parent)
# Set the default font
font = QFont()
font.setFamily('Courier')
font.setFixedPitch(True)
font.setPointSize(10)
self.setFont(font)
self.setMarginsFont(font)
# Margin 0 is used for line numbers
fontmetrics = QFontMetrics(font)
self.setMarginsFont(font)
self.setMarginWidth(0, fontmetrics.width("00000") + 6)
self.setMarginLineNumbers(0, True)
self.setMarginsBackgroundColor(QColor("#cccccc"))
# Clickable margin 1 for showing markers
self.setMarginSensitivity(1, True)
# self.connect(self,
# SIGNAL('marginClicked(int, int, Qt::KeyboardModifiers)'),
# self.on_margin_clicked)
self.markerDefine(QsciScintilla.RightArrow,
self.ARROW_MARKER_NUM)
self.setMarkerBackgroundColor(QColor("#ee1111"),
self.ARROW_MARKER_NUM)
# Brace matching: enable for a brace immediately before or after
# the current position
#
self.setBraceMatching(QsciScintilla.SloppyBraceMatch)
# Current line visible with special background color
self.setCaretLineVisible(True)
self.setCaretLineBackgroundColor(QColor("#ffe4e4"))
# Set Python lexer
# Set style for Python comments (style number 1) to a fixed-width
# courier.
#
lexer = QsciLexerPython()
lexer.setDefaultFont(font)
self.setLexer(lexer)
text = bytearray(str.encode("Arial"))
# 32, "Courier New"
self.SendScintilla(QsciScintilla.SCI_STYLESETFONT, 1, text)
# Don't want to see the horizontal scrollbar at all
# Use raw message to Scintilla here (all messages are documented
# here: http://www.scintilla.org/ScintillaDoc.html)
self.SendScintilla(QsciScintilla.SCI_SETHSCROLLBAR, 0)
# not too small
self.setMinimumSize(600, 450)
def on_margin_clicked(self, nmargin, nline, modifiers):
# Toggle marker for the line the margin was clicked on
if self.markersAtLine(nline) != 0:
self.markerDelete(nline, self.ARROW_MARKER_NUM)
else:
self.markerAdd(nline, self.ARROW_MARKER_NUM)
if __name__ == "__main__":
app = QApplication(sys.argv)
editor = SimplePythonEditor()
editor.show()
editor.setText(open(sys.argv[0]).read())
app.exec_()
只需将此代码复制粘贴到一个空.py
文件中,然后运行它。您应该会在显示屏上看到以下简单的文本编辑器:
注意语法高亮是多么完美!QScintilla 确实在后台做了一些解析来实现这一点。
是否可以为此文本编辑器制作可点击的函数和变量?每个自尊的 IDE 都有它。您单击一个函数,IDE 会跳转到函数定义。变量也一样。我想知道:
- QScintilla 是否支持可点击的函数和变量?
- 如果没有,是否可以在 QScintilla 文本编辑器中导入另一个实现此功能的 python 模块?
编辑:
λuser 注意到以下内容:
可点击的函数名称需要使用更深入的编程语言知识进行完整解析 [..]
这超出了 Scintilla/QScintilla 的范围。Scintilla 提供了一种在鼠标单击文本某处时做出反应的方法,但是“函数的定义在哪里”的逻辑不在 Scintilla 中,而且可能永远不会出现。
但是,一些项目专门用于此任务,例如ctags。您可以简单地围绕这种工具编写一个包装器。
我想为ctags编写这样的包装器现在在我的 TODO 列表中。第一步是在用户单击函数或变量时获得反应(Qt 信号)。当您将鼠标悬停在函数/变量上时,该函数/变量可能会变蓝,以通知用户它是可点击的。我已经尝试过实现这一目标,但由于缺少 QScintilla 文档而受阻。
因此,让我们将问题缩减为:您如何使 QScintilla 文本编辑器中的函数或变量可点击(可点击定义为“发生某事”)
编辑:
我现在才回到这个问题 - 几个月后。我一直在与我的朋友 Matic Kukovec 合作设计一个关于 QScintilla 的网站。这是一个关于如何使用它的初学者友好教程:
我希望这一举措能填补缺乏文档的空白。