首页 > 解决方案 > PyQt Slider 没有到达我单击的特定位置,而是移动到某个步幅

问题描述

例如,我们有一个 PYQT5 的 QSlider 实例,从左到右,从 0 到 100% 当我点击 50% 的位置时,手柄不会直接移动到 50%,而只是移动一个恒定的步幅。我该怎么办?

标签: pythonqtpyqt

解决方案


您指的是“绝对设置”选项,它完全依赖于当前样式。滑块通过查询有关 的 QStyle 来检查该行为styleHint()SH_Slider_AbsoluteSetButtons它以(可能为空的)Qt.MouseButton掩码回复。

默认情况下,使用左键在手柄外部按下只会从当前滑块位置向鼠标光标重复滚动,而使用中键按下它会将滑块准确地放置在光标所在的位置(取决于操作系统和 QStyle)。

如果你想覆盖它,有两种可能性。

代理样式覆盖

这通过使用 QProxyStyle 并覆盖上述styleHint. 不幸的是,QSlider 只是在没有提供小部件参数的情况下查询有关提示的样式,因此无法知道哪个滑块发送了请求。结果是该行为将成为应用程序中所有QSlider 的全局行为,除非您仅将样式应用于您想要此行为的那些滑块,但这可能会导致一些问题和不一致,特别是如果您已经需要使用代理风格。

class ProxyStyle(QtWidgets.QProxyStyle):
    def styleHint(self, hint, opt=None, widget=None, returnData=None):
        res = super().styleHint(hint, opt, widget, returnData)
        if hint == self.SH_Slider_AbsoluteSetButtons:
            res |= QtCore.Qt.LeftButton
        return res

app = QtWidgets.QApplication(sys.argv)
# set the style globally for the application
app.setStyle(ProxyStyle())

slider = QtWidgets.QSlider()
# or just for the slider
slider.setStyle(ProxyStyle())

覆盖mouseButtonPress

此选项依赖于子类化和部分覆盖鼠标按钮按下事件。诀窍是检查按下的按钮,如果当前没有按下滑块(以避免按下多个按钮时出现意外行为),然后将滑块移动到鼠标位置,最后调用基本mouseButtonPress实现:因为此时句柄将是在鼠标下方,滑块将“相信”手柄已经存在,从而开始实际的滑块移动。

class SliderCustom(QtWidgets.QSlider):
    def mousePressEvent(self, event):
        if event.button() == QtCore.Qt.LeftButton and not self.isSliderDown():
            opt = QtWidgets.QStyleOptionSlider()
            self.initStyleOption(opt)
            sliderRect = self.style().subControlRect(
                QtWidgets.QStyle.CC_Slider, opt, 
                QtWidgets.QStyle.SC_SliderHandle, self)
            if event.pos() not in sliderRect:
                # the mouse is not over the handle, let's move it; this is based
                # on the original C++ code that moves the handle when the
                # "absolute button" is pressed
                grooveRect = self.style().subControlRect(
                    QtWidgets.QStyle.CC_Slider, opt, 
                    QtWidgets.QStyle.SC_SliderGroove, self)
                center = sliderRect.center() - sliderRect.topLeft()
                pos = event.pos() - center
                if self.orientation() == QtCore.Qt.Horizontal:
                    sliderLength = sliderRect.width()
                    sliderMin = grooveRect.x()
                    sliderMax = grooveRect.right() - sliderLength + 1
                    pos = pos.x()
                else:
                    sliderLength = sliderRect.height()
                    sliderMin = grooveRect.y()
                    sliderMax = grooveRect.bottom() - sliderLength + 1
                    pos = pos.y()
                value = self.style().sliderValueFromPosition(
                    self.minimum(), self.maximum(), pos - sliderMin, 
                    sliderMax - sliderMin, opt.upsideDown
                )
                self.setSliderPosition(value)
        super().mousePressEvent(event)

推荐阅读