QDoubleSpinBox 退格键的编辑行为

Editing behavior of “backspace” key of QDoubleSpinBox

我遇到了 QDoubleSpinBox 的问题。 “退格”键的编辑行为在某种程度上取决于后缀的大小。 如果我将“m”设置为后缀,然后将光标设置在旋转框的末尾并按“退格键”,光标会跳过“m”后缀到可以使用更多“退格键”进行编辑的值。 如果我将后缀设置为“mm”或任何双字母单词,无论我按下多少个“退格键”,光标都会停留在旋转框的末尾。

我尝试调试进入“validate”方法的内容,得到了一个奇怪的结果: 当光标位于“0,00m”的末尾时按下“退格键”,验证接收到“0,00m”。 当光标位于“0,00_m”末尾时按下“退格键”时,验证接收到“0,00__m” 当光标位于“0,00_mm”末尾时按下“退格键”时,验证接收到“0,00_m_mm”

这种行为的原因是什么,我该如何克服?

# coding=utf-8
from PyQt5 import QtWidgets


class SpinBox(QtWidgets.QDoubleSpinBox):
    def __init__(self):
        super().__init__()

    def validate(self, text, index):
        res = super().validate(text, index)
        print(text, res, self.text())
        return res

if __name__ == "__main__":
    q_app = QtWidgets.QApplication([])
    sb = SpinBox()
    sb.setSuffix(" m")
    sb.show()
    q_app.exec_()

QDoubleSpinBox/QAbstractSpinBox 的源代码在关键事件处理方面极其复杂 - 我无法弄清楚默认行为应该是什么,甚至无法确定它可能在何处实现。某处可能存在错误,但我不想打赌。

看来唯一的选择是重新实现 keyPressEvent:

class SpinBox(QtWidgets.QDoubleSpinBox):
    def keyPressEvent(self, event):
        if event.key() == QtCore.Qt.Key_Backspace:
            suffix = self.suffix()
            if suffix:
                edit = self.lineEdit()
                text = edit.text()
                if (text.endswith(suffix) and
                    text != self.specialValueText()):
                    pos = edit.cursorPosition()
                    end = len(text) - len(suffix)
                    if pos > end:
                        edit.setCursorPosition(end)
                        return
        super().keyPressEvent(event)