python - PyQt Slider 没有到达我单击的特定位置,而是移动到某个步幅
问题描述
例如,我们有一个 PYQT5 的 QSlider 实例,从左到右,从 0 到 100% 当我点击 50% 的位置时,手柄不会直接移动到 50%,而只是移动一个恒定的步幅。我该怎么办?
解决方案
您指的是“绝对设置”选项,它完全依赖于当前样式。滑块通过查询有关 的 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)
推荐阅读
- laravel - guzzlehttp 无法解析主机
- php - 如何在 wordpress 子菜单中检索自定义链接?
- html - 将边距底部添加到边框折叠表
- android - 删除列表中长单击侦听器上的项目
- javascript - 如何将格式正确的 Javascript 对象(字符串格式)转换为对象?
- php - Laravel 5.0 表单验证每个关系唯一
- python - 如何将列表字典写入字符串而不是 CSV 文件?
- sql-server - 如何在查询中自动将 SQL 表导出到 CSV 文件?
- python - TypeError:预期的字符串或类似字节的对象python重新包
- r - 在 R 中过滤数据框:将 GREP 与变量名结合使用?