QDial 是一个非常奇特的控件。虽然它仍然受支持,但实现得很差,我相信它是一种选择:由于它的性质,很难添加更多功能。我有相当多的经验,我知道这不是一个容易处理的元素。
它的一个问题是它代表一个单维范围,但从视觉和 UI 来说,它是一个二维对象。
您试图实现的目标是可能的,但请考虑 UI 元素应始终以清晰的方式显示其状态并具有相应的适当行为;这是 UI 可以告诉用户状态的唯一方式。物理表盘没有这个问题:你还有一个触觉反应,告诉你齿轮何时到达终点。
根据我的经验,我可以告诉您,您应该尽可能避免使用它:它看起来是一个不错且直观的小部件,但实际上很难获得对用户而言实际上直观的正确结果。在某些情况下使用它是有意义的(在我的例子中,代表电子乐器的物理旋钮)。我建议您对拟态和 UX 方面进行一些研究。
也就是说,这是一个可能的原始实现。我已经覆盖了一些方面(最重要的是valueChanged
信号,为了命名一致性),但为了正确实现,你应该做更多的工作(和测试)。
诀窍是根据“转数”来设置范围:如果最大值为 500 并且选择了 5 转,那么表盘的实际最大值为 100。然后,每当值发生变化时,我们检查是否之前的值低于或高于实际范围的最小值/最大值,并相应地更改转数。
两个重要说明:
- 由于 QDial 继承自 QAbstractSlider,它有一个范围(最小值,最大值 + 1),并且由于除法可以有一些休息,“最后”旋转将有不同的范围;
- 我没有实现车轮事件,因为这需要进一步检查并根据“先前”值和革命选择适当的行为;
class SpecialDial(QDial):
_cycleValueChange = pyqtSignal(int)
def __init__(self, minimum=0, maximum=100, cycleCount=2):
super().__init__()
assert cycleCount > 1, 'cycles must be 2 or more'
self.setWrapping(True)
self.cycle = 0
self.cycleCount = cycleCount
self._minimum = minimum
self._maximum = maximum
self._normalMaximum = (maximum - minimum) // cycleCount
self._lastMaximum = self._normalMaximum + (maximum - minimum) % self._normalMaximum
self._previousValue = super().value()
self._valueChanged = self.valueChanged
self.valueChanged = self._cycleValueChange
self._valueChanged.connect(self.adjustValueChanged)
self.setRange(0, self._normalMaximum)
def value(self):
return super().value() + self._normalMaximum * self.cycle
def minimum(self):
return self._minimum
def maximum(self):
return self._maximum()
def dialMinimum(self):
return super().minimum()
def dialMaximum(self):
return super().maximum()
def adjustValueChanged(self, value):
if value < self._previousValue:
if (value < self.dialMaximum() * .3 and self._previousValue > self.dialMaximum() * .6 and
self.cycle + 1 < self.cycleCount):
self.cycle += 1
if self.cycle == self.cycleCount - 1:
self.setMaximum(self._lastMaximum)
elif (value > self.dialMaximum() * .6 and self._previousValue < self.dialMaximum() * .3 and
self.cycle > 0):
self.cycle -= 1
if self.cycle == 0:
self.setMaximum(self._normalMaximum)
new = self.value()
if self._previousValue != new:
self._previousValue = value
self.valueChanged.emit(self.value())
def setValue(self, value):
value = max(self._minimum, min(self._maximum, value))
if value == self.value():
return
block = self.blockSignals(True)
self.cycle, value = divmod(value, self._normalMaximum)
if self.dialMaximum() == self._normalMaximum and self.cycle == self.cycleCount - 1:
self.setMaximum(self._lastMaximum)
elif self.dialMaximum() == self._lastMaximum and self.cycle < self.cycleCount - 1:
self.setMaximum(self._normalMaximum)
super().setValue(value)
self.blockSignals(block)
self._previousValue = self.value()
self.valueChanged.emit(self._previousValue)
def keyPressEvent(self, event):
key = event.key()
if key in (Qt.Key_Right, Qt.Key_Up):
step = self.singleStep()
elif key in (Qt.Key_Left, Qt.Key_Down):
step = -self.singleStep()
elif key == Qt.Key_PageUp:
step = self.pageStep()
elif key == Qt.Key_PageDown:
step = -self.pageStep()
elif key in (Qt.Key_Home, Qt.Key_End):
if key == Qt.Key_Home or self.invertedControls():
if super().value() > 0:
self.cycle = 0
block = self.blockSignals(True)
super().setValue(0)
self.blockSignals(block)
self.valueChanged.emit(self.value())
else:
if self.cycle != self.cycleCount - 1:
self.setMaximum(self._lastMaximum)
self.cycle = self.cycleCount - 1
if super().value() != self._lastMaximum:
block = self.blockSignals(True)
super().setValue(self._lastMaximum)
self.blockSignals(block)
self.valueChanged.emit(self.value())
return
else:
super().keyPressEvent(event)
return
if self.invertedControls():
step *= -1
current = self.value()
new = max(self._minimum, min(self._maximum, current + step))
if current != new:
super().setValue(super().value() + (new - current))
class Window(QWidget):
def __init__(self):
QWidget.__init__(self)
layout = QGridLayout()
self.setLayout(layout)
self.dial = SpecialDial()
self.dial.valueChanged.connect(self.sliderMoved)
self.text=QLabel()
layout.addWidget(self.dial)
layout.addWidget(self.text)
def sliderMoved(self):
self.text.setText(str(self.dial.value()))
我强烈建议您花时间:
- 考虑一下这是否真的是您想要的,因为如上所述,从用户体验的角度来看,这种控制可能非常棘手;
- 仔细阅读代码并理解其逻辑;