Полоса прокрутки, чтобы всегда показывать нижнюю часть потокового текста QTextBrowser - PullRequest
0 голосов
/ 26 февраля 2019

Я передаю "stdout" и "stderr" во встроенный виджет QTextBrowser в графическом интерфейсе pyqt5.Все работает, но если я передаю очень длинный вывод (что происходит постоянно), полоса прокрутки виджета всегда находится в последней верхней позиции, и, таким образом, я не могу увидеть новый вывод в реальном времени.Это действительно раздражает!Я перепробовал все, что мог найти в интернете, но безуспешно.Это не может быть слишком сложно ... Пожалуйста, откройте мои глаза!

Код, который я использую, здесь (я думаю, он был найден в StackOverflow):

import sys
from PyQt4 import QtCore, QtGui

import logging
logger = logging.getLogger(__name__)

class QtHandler(logging.Handler):

    def __init__(self):
        logging.Handler.__init__(self)

    def emit(self, record):
        record = self.format(record)
        XStream.stdout().write("{}\n".format(record))

handler = QtHandler()
handler.setFormatter(logging.Formatter("%(levelname)s: %(message)s"))
logger.addHandler(handler)
logger.setLevel(logging.DEBUG)



class XStream(QtCore.QObject):
    _stdout = None
    _stderr = None
    messageWritten = QtCore.pyqtSignal(str)
    def flush( self ):
        pass
    def fileno( self ):
        return -1
    def write( self, msg ):
        if ( not self.signalsBlocked() ):
            self.messageWritten.emit(msg)
    @staticmethod
    def stdout():
        if ( not XStream._stdout ):
            XStream._stdout = XStream()
            sys.stdout = XStream._stdout
        return XStream._stdout
    @staticmethod
    def stderr():
        if ( not XStream._stderr ):
            XStream._stderr = XStream()
            sys.stderr = XStream._stderr

        return XStream._stderr

class MyDialog(QtGui.QDialog):
    def __init__( self, parent = None ):
        super(MyDialog, self).__init__(parent)

        self._console = QtGui.QTextBrowser(self)      
        self._console.moveCursor(QtGui.QTextCursor.End)

        layout = QtGui.QVBoxLayout()
        layout.addWidget(self._console)
        self.setLayout(layout)

        XStream.stdout().messageWritten.connect(self._console.insertPlainText)
        XStream.stderr().messageWritten.connect(self._console.insertPlainText)

        self.pipe_output()


    def pipe_output( self ):
        logger.debug('debug message')
        logger.info('info message')
        logger.warning('warning message')
        logger.error('error message')
        #print('Old school hand made print message')

if ( __name__ == '__main__' ):
    #app = None
    # if ( not QtGui.QApplication.instance() ):
    app = QtGui.QApplication([])
    dlg = MyDialog()
    dlg.show()

    #if ( app ):
    app.exec_()

Вот распечаткаэкран

enter image description here

Ответы [ 2 ]

0 голосов
/ 26 февраля 2019

Благодаря titusjan и eyllanesc я смог получить то, что хочу!

Код, который работает:

import sys
from PyQt5 import QtCore, QtGui

import logging
logger = logging.getLogger(__name__)

class QtHandler(logging.Handler):

    def __init__(self):
        logging.Handler.__init__(self)

    def emit(self, record):
        record = self.format(record)
        XStream.stdout().write("{}\n".format(record))

handler = QtHandler()
handler.setFormatter(logging.Formatter("%(levelname)s: %(message)s"))
logger.addHandler(handler)
logger.setLevel(logging.DEBUG)

class XStream(QtCore.QObject):
    _stdout = None
    _stderr = None
    messageWritten = QtCore.pyqtSignal(str)
    def flush( self ):
        pass
    def fileno( self ):
        return -1
    def write( self, msg ):
        if ( not self.signalsBlocked() ):
            self.messageWritten.emit(msg)
    @staticmethod
    def stdout():
        if ( not XStream._stdout ):
            XStream._stdout = XStream()
            sys.stdout = XStream._stdout
        return XStream._stdout
    @staticmethod
    def stderr():
        if ( not XStream._stderr ):
            XStream._stderr = XStream()
            sys.stderr = XStream._stderr

        return XStream._stderr

class LogMessageViewer(QtGui.QTextBrowser):

    def __init__(self, parent=None):
        super(LogMessageViewer,self).__init__(parent)
        self.setReadOnly(True)
        #self.setLineWrapMode(QtGui.QTextEdit.NoWrap)


    @QtCore.pyqtSlot(str)
    def appendLogMessage(self, msg):
        horScrollBar = self.horizontalScrollBar()
        verScrollBar = self.verticalScrollBar()
        scrollIsAtEnd = verScrollBar.maximum() - verScrollBar.value() <= 10

        self.insertPlainText(msg)

        if scrollIsAtEnd:
            verScrollBar.setValue(verScrollBar.maximum()) # Scrolls to the bottom
            horScrollBar.setValue(0) # scroll to the left

class MyDialog(QtGui.QDialog):
    def __init__( self, parent = None ):
        super(MyDialog, self).__init__(parent)

        self._console = LogMessageViewer(self)

        layout = QtGui.QVBoxLayout()
        layout.addWidget(self._console)
        self.setLayout(layout)

        XStream.stdout().messageWritten.connect(self._console.appendLogMessage)
        XStream.stderr().messageWritten.connect(self._console.appendLogMessage)


if ( __name__ == '__main__' ):

    app = QtGui.QApplication([])
    dlg = MyDialog()
    dlg.show()

    app.exec_()

И:

enter image description here

0 голосов
/ 26 февраля 2019

В прошлом я делал нечто похожее, просмотрщик сообщений журнала, который прокручивается вниз при добавлении новых сообщений.Он основан на QTextEdit, но так как QTextBrowser также имеет метод verticalScrolBar, я полагаю, вы можете легко заставить его работать для QTextBrower.

class LogMessageViewer(QtWidgets.QTextEdit):

    def __init__(self, parent=None):
        super().__init__(parent=parent)
        self.setReadOnly(True)
        self.setLineWrapMode(QtWidgets.QTextEdit.NoWrap)


    @pyqtSlot(str)
    def appendLogMessage(self, msg):
        horScrollBar = self.horizontalScrollBar()
        verScrollBar = self.verticalScrollBar()
        scrollIsAtEnd = verScrollBar.maximum() - verScrollBar.value() <= 10

        self.append(msg)

        if scrollIsAtEnd:
            verScrollBar.setValue(verScrollBar.maximum()) # Scrolls to the bottom
            horScrollBar.setValue(0) # scroll to the left

Обратите внимание, что он толькопрокручивается автоматически, если текущая позиция прокрутки уже находится в пределах 10 пикселей от дна.Это позволяет вам прокручивать вверх и видеть предыдущий текст, не прерываясь новым выводом.Просто прокрутите вниз, чтобы снова получать живые обновления.

...