为什么 QColorDialog.getColor 会意外崩溃?

Why is QColorDialog.getColor crashing unexpectedly?

我得到了这个小 mcve 代码:

import sys
import re

from PyQt5 import QtGui, QtWidgets, QtCore
from PyQt5.Qsci import QsciScintilla
from PyQt5 import Qsci


class SimpleEditor(QsciScintilla):

    def __init__(self, language=None, parent=None):
        super().__init__(parent)

        font = QtGui.QFont()
        font.setFamily('Courier')
        font.setFixedPitch(True)
        font.setPointSize(10)
        self.setFont(font)
        self.setMarginsFont(font)
        fontmetrics = QtGui.QFontMetrics(font)
        self.setMarginsFont(font)
        self.setMarginWidth(0, fontmetrics.width("00000") + 6)
        self.setMarginLineNumbers(0, True)
        self.setMarginsBackgroundColor(QtGui.QColor("#cccccc"))
        self.setBraceMatching(QsciScintilla.SloppyBraceMatch)
        self.setCaretLineVisible(True)
        self.setCaretLineBackgroundColor(QtGui.QColor("#E8E8FF"))

        if language:
            self.lexer = getattr(Qsci, 'QsciLexer' + language)()
            self.setLexer(self.lexer)

        self.SendScintilla(QsciScintilla.SCI_FOLDALL, True)
        self.setAutoCompletionThreshold(1)
        self.setAutoCompletionSource(QsciScintilla.AcsAPIs)
        self.setFolding(QsciScintilla.BoxedTreeFoldStyle)

        # Signals/Slots
        self.cursorPositionChanged.connect(self.on_cursor_position_changed)
        self.copyAvailable.connect(self.on_copy_available)
        self.indicatorClicked.connect(self.on_indicator_clicked)
        self.indicatorReleased.connect(self.on_indicator_released)
        self.linesChanged.connect(self.on_lines_changed)
        self.marginClicked.connect(self.on_margin_clicked)
        self.modificationAttempted.connect(self.on_modification_attempted)
        self.modificationChanged.connect(self.on_modification_changed)
        self.selectionChanged.connect(self.on_selection_changed)
        self.textChanged.connect(self.on_text_changed)
        self.userListActivated.connect(self.on_user_list_activated)

    def on_cursor_position_changed(self, line, index):
        text = self.text(line)
        for match in re.finditer('(?:^|(?<=\W))\d+(?:\.\d+)?(?=$|\W)', text):
            start, end = match.span()
            if start <= index <= end:
                pos = self.positionFromLineIndex(line, start)
                x = self.SendScintilla(
                    QsciScintilla.SCI_POINTXFROMPOSITION, 0, pos)
                y = self.SendScintilla(
                    QsciScintilla.SCI_POINTYFROMPOSITION, 0, pos)
                point = self.mapToGlobal(QtCore.QPoint(x, y))
                num = float(match.group())
                message = 'number: %s' % num
                break
        else:
            point = QtCore.QPoint(0, 0)
            message = ''
        QtWidgets.QToolTip.showText(point, message)
        color = QtWidgets.QColorDialog.getColor()

    def on_copy_available(self, yes):
        print('-' * 80)
        print("on_copy_available")

    def on_indicator_clicked(self, line, index, state):
        print("on_indicator_clicked")

    def on_indicator_released(self, line, index, state):
        print("on_indicator_released")

    def on_lines_changed(self):
        print("on_lines_changed")

    def on_margin_clicked(self, margin, line, state):
        print("on_margin_clicked")

    def on_modification_attempted(self):
        print("on_modification_attempted")

    def on_modification_changed(self):
        print("on_modification_changed")

    def on_selection_changed(self):
        print("on_selection_changed")

    def on_text_changed(self):
        print("on_text_changed")

    def on_user_list_activated(self, id, text):
        print("on_user_list_activated")


def show_requirements():
    print(sys.version)
    print(QtCore.QT_VERSION_STR)
    print(QtCore.PYQT_VERSION_STR)

if __name__ == "__main__":
    show_requirements()

    app = QtWidgets.QApplication(sys.argv)

    ex = QtWidgets.QWidget()
    hlayout = QtWidgets.QHBoxLayout()
    ed = SimpleEditor("JavaScript")

    hlayout.addWidget(ed)

    ed.setText("""#ifdef GL_ES
precision mediump float;
#endif

#extension GL_OES_standard_derivatives : enable

uniform float time;
uniform vec2 mouse;
uniform vec2 resolution;

void main( void ) {

    vec2 st = ( gl_FragCoord.xy / resolution.xy );
    vec2 lefbot = step(vec2(0.1), st);
    float pct = lefbot.x*lefbot.y;
    vec2 rigtop = step(vec2(0.1), 1.-st);
    pct *= rigtop.x*rigtop.y;
    vec3 color = vec3(pct);

    gl_FragColor = vec4( color, 1.0 );""")

    ex.setLayout(hlayout)
    ex.show()
    ex.resize(800, 600)

    sys.exit(app.exec_())

由于某些未知原因,当我在拾取颜色后按“确定”时,脚本会意外崩溃,video 显示了我的意思。

知道为什么会这样吗?

在光标更改处理程序中打开一个带有阻塞 exec() 的对话框似乎是个坏主意,所以我对它崩溃并不感到惊讶。异步打开对话框并使用 signal/slot 来处理结果会安全得多。使用带有 show() 的闭包似乎是最好的,因为它将提供对当前局部变量的访问:

class SimpleEditor(QsciScintilla):
    def __init__(self, language=None, parent=None):
        super().__init__(parent)
        ...
        self.colors = QtWidgets.QColorDialog(self)

    def on_cursor_position_changed(self, line, index):
        text = self.text(line)
        for match in re.finditer('(?:^|(?<=\W))\d+(?:\.\d+)?(?=$|\W)', text):
            start, end = match.span()
            if start <= index <= end:
                pos = self.positionFromLineIndex(line, start)
                x = self.SendScintilla(
                    QsciScintilla.SCI_POINTXFROMPOSITION, 0, pos)
                y = self.SendScintilla(
                    QsciScintilla.SCI_POINTYFROMPOSITION, 0, pos)
                point = self.mapToGlobal(QtCore.QPoint(x, y))
                num = float(match.group())
                message = 'number: %s' % num

                def handler():
                    print(message)
                    print(self.colors.selectedColor().name())
                    self.colors.finished.disconnect()

                self.colors.finished.connect(handler)
                self.colors.show()

                break