简体   繁体   English

PyQt4:使QTextEdit上的某些文本不可编辑

[英]PyQt4 : making certain text on QTextEdit uneditable

Is there any way I can make certain text on QTextEdit permanent. 有什么办法可以使QTextEdit上的某些文本永久化。 Applications like cmd.exe where the current user directory is displayed and the rest of the screen is up for input. 像cmd.exe这样的应用程序,其中显示当前用户目录,并且屏幕的其余部分可以输入。 I tried inserting a QLabel but unable to do so, here's my code, I am currently taking user input through a separate line edit. 我尝试插入QLabel,但无法这样做,这是我的代码,目前我正在通过单独的行编辑来接受用户输入。

UPDATE I had a look at Ipython QtConsole where the line number is displayed constantly, how can I do that, I am looking in the source but if anyone who already knows it, please do tell. 更新我看了Ipython QtConsole,其中行号一直显示,我该怎么做,我正在寻找源代码,但是如果有人已经知道的话,请告诉我。 Here is the QtConsole for ipython notebook, I am trying to replicate this. 这是用于ipython笔记本的QtConsole,我正在尝试复制它。

在此处输入图片说明

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


def main():
    app = QApplication(sys.argv)
    w = MyWindow()
    w.show()
    sys.exit(app.exec_())


class MyWindow(QWidget):
    def __init__(self, *args):
        QWidget.__init__(self, *args)

        # create objects
        label = QLabel(self.tr("Enter command and press Return"))
        self.le = QLineEdit()
        self.te = QTextEdit()
        self.lbl = QLabel(str(os.getcwd())+"> ")

        # layout
        layout = QVBoxLayout(self)
        layout.addWidget(label)
        layout.addWidget(self.le)
        layout.addWidget(self.te)
        self.setLayout(layout)

        # styling
        self.te.setReadOnly(True)

        # create connection
        self.mytext = str(self.le.text())
        self.connect(self.le, PyQt4.QtCore.SIGNAL("returnPressed(void)"),
                     self.display)

    def display(self):
        mytext = str(self.le.text())
        self.te.append(self.lbl +str(os.popen(mytext).read()))
        self.le.setText("")


if __name__ == "__main__":
    main()

A simple solution is to create a class that inherits from QTextEdit and overwrite and add the necessary attributes as shown below: 一个简单的解决方案是创建一个从QTextEdit继承并覆盖并添加必要属性的类,如下所示:

class TextEdit(QTextEdit):
    def __init__(self, *args, **kwargs):
        QTextEdit.__init__(self, *args, **kwargs)
        self.staticText = os.getcwd()
        self.counter = 1
        self.setReadOnly(True)

    def append(self, text):
        n_text = "{text} [{number}] > ".format(text=self.staticText, number=self.counter)
        self.counter += 1
        QTextEdit.append(self, n_text+text)

Complete Code: 完整的代码:

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

class TextEdit(QTextEdit):
    def __init__(self, *args, **kwargs):
        QTextEdit.__init__(self, *args, **kwargs)
        self.staticText = os.getcwd()
        self.counter = 1
        self.setReadOnly(True)

    def append(self, text):
        n_text = "{text} [{number}] > ".format(text=self.staticText, number=self.counter)
        self.counter += 1
        QTextEdit.append(self, n_text+text)


class MyWindow(QWidget):
    def __init__(self, *args, **kwargs):
        QWidget.__init__(self, *args, **kwargs)
        label = QLabel(self.tr("Enter command and press Return"), self)
        self.le = QLineEdit(self)
        self.te = TextEdit(self)
        # layout
        layout = QVBoxLayout(self)
        layout.addWidget(label)
        layout.addWidget(self.le)
        layout.addWidget(self.te)
        self.setLayout(layout)
        self.connect(self.le, SIGNAL("returnPressed(void)"), self.display)
        # self.le.returnPressed.connect(self.display)

    def display(self):
        command = str(self.le.text())
        resp = str(os.popen(command).read())
        self.te.append(resp)
        self.le.clear()


def main():
    app = QApplication(sys.argv)
    w = MyWindow()
    w.show()
    sys.exit(app.exec_())

if __name__ == "__main__":
    main()

To emulate QtConsole we must overwrite some methods of QTextEdit, catch some events, and verify that it does not eliminate the prefix as I show below: 要模拟QtConsole,我们必须覆盖QTextEdit的某些方法,捕获一些事件,并验证它不会消除前缀,如下所示:

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

class TextEdit(QTextEdit):
    def __init__(self, *args, **kwargs):
        QTextEdit.__init__(self, *args, **kwargs)
        self.staticText = os.getcwd()
        self.counter = 1
        self.prefix = ""
        self.callPrefix()
        self.setContextMenuPolicy(Qt.CustomContextMenu)
        self.customContextMenuRequested.connect(self.onCustomContextMenuRequest)

    def onCustomContextMenuRequest(self, point):
        menu = self.createStandardContextMenu()
        for action in menu.actions():
            if "Delete" in action.text():
                action.triggered.disconnect()
                menu.removeAction(action)
            elif "Cu&t" in action.text():
                action.triggered.disconnect()
                menu.removeAction(action)
            elif "Paste" in action.text():
                action.triggered.disconnect()

        act = menu.exec_(point)
        if act:
            if "Paste" in act.text():
                self.customPaste()


    def customPaste(self):
        self.moveCursor(QTextCursor.End)
        self.insertPlainText(QApplication.clipboard().text())
        self.moveCursor(QTextCursor.End)

    def clearCurrentLine(self):
        cs = self.textCursor()
        cs.movePosition(QTextCursor.StartOfLine)
        cs.movePosition(QTextCursor.EndOfLine)
        cs.select(QTextCursor.LineUnderCursor)
        text = cs.removeSelectedText()

    def isPrefix(self, text):
        return self.prefix == text

    def getCurrentLine(self):
        cs = self.textCursor()
        cs.movePosition(QTextCursor.StartOfLine)
        cs.movePosition(QTextCursor.EndOfLine)
        cs.select(QTextCursor.LineUnderCursor)
        text = cs.selectedText()
        return text

    def keyPressEvent(self, event):
        if event.key() == Qt.Key_Return:
            command = self.getCurrentLine()[len(self.prefix):]
            self.execute(command)
            self.callPrefix()
            return
        elif event.key() == Qt.Key_Backspace:
            if self.prefix == self.getCurrentLine():
                return
        elif event.matches(QKeySequence.Delete):
            return
        if event.matches(QKeySequence.Paste):
            self.customPaste()
            return
        elif self.textCursor().hasSelection():
            t = self.toPlainText()
            self.textCursor().clearSelection()
            QTextEdit.keyPressEvent(self, event)
            self.setPlainText(t)
            self.moveCursor(QTextCursor.End)
            return
        QTextEdit.keyPressEvent(self, event)

    def callPrefix(self):
        self.prefix = "{text} [{number}] >".format(text=self.staticText, number=self.counter)
        self.counter += 1
        self.append(self.prefix)

    def execute(self, command):
        resp = os.popen(command).read()
        self.append(resp)


class MyWindow(QWidget):
    def __init__(self, *args, **kwargs):
        QWidget.__init__(self, *args, **kwargs)
        label = QLabel(self.tr("Enter command and press Return"), self)
        self.te = TextEdit(self)
        # layout
        layout = QVBoxLayout(self)
        layout.addWidget(label)
        layout.addWidget(self.te)
        self.setLayout(layout)

def main():
    app = QApplication(sys.argv)
    w = MyWindow()
    w.show()
    sys.exit(app.exec_())

if __name__ == "__main__":
    main()

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM