在 PyQt5 中获取滚动条拇指的大小/宽度?

Posted

技术标签:

【中文标题】在 PyQt5 中获取滚动条拇指的大小/宽度?【英文标题】:Getting the size/width of a scrollbar thumb in PyQt5? 【发布时间】:2019-01-27 18:38:20 【问题描述】:

说,你有一个 PyQt5 滚动条:

我想以编程方式找到滚动条“拇指”的宽度(编辑:http://pyqt.sourceforge.net/Docs/PyQt4/qscrollbar.html 中称为“滑块”)及其位置;我猜像素一开始会是“自然”单位。

从上图中,我猜start 始终为 0 - 但我可以获取 leftrightwidth 的值吗? 987654330@ 和 rightwidth 微不足道 right-left);和end 以像素为单位?

我知道存在QScrollBar.value()pageStep()singleStep(),但我无法确定它们是否与上述像素尺寸有关。

(如果滚动条需要 PyQt5 示例,请参阅我在 Have tabs keep focus on mousewheel over them in a PyQt5 scrollarea? 中发布的代码)

【问题讨论】:

QStyle 类提供了所有这些功能。当我尝试了各种方法(例如 pixelMetricsubControlRect)时,似乎没有什么能像我预期的那样工作 - 但希望你能有更多的运气。 【参考方案1】:

在这种情况下,你必须使用QStyle()的subControlRect()方法来获取矩形,从中可以得到你想要的信息。

在下面的示例中,我创建了一个自定义类 QScrollBar,每当滑块的位置改变或改变 QScrollBar 的大小时,它都会通过一个信号发出该信息。

from PyQt5 import QtCore, QtGui, QtWidgets

class Scrollbar(QtWidgets.QScrollBar):
    geometryChanged = QtCore.pyqtSignal(int, int, int, int, int)

    def __init__(self, parent=None):
        super(Scrollbar, self).__init__(parent)
        self.sliderMoved.connect(self.calculate_geometry)

    def resizeEvent(self, event):
        self.calculate_geometry()
        super(Scrollbar, self).resizeEvent(event)

    def calculate_geometry(self):
        opt = QtWidgets.QStyleOptionSlider()
        self.initStyleOption(opt)
        gr = self.style().subControlRect(QtWidgets.QStyle.CC_ScrollBar, opt,
            QtWidgets.QStyle.SC_ScrollBarGroove, self)
        sr = self.style().subControlRect(QtWidgets.QStyle.CC_ScrollBar, opt,
            QtWidgets.QStyle.SC_ScrollBarSlider, self)

        start = gr.left() if self.orientation() == QtCore.Qt.Horizontal else gr.top()
        left = sr.left() if self.orientation() == QtCore.Qt.Horizontal else sr.top()
        right = sr.right() if self.orientation() == QtCore.Qt.Horizontal else sr.bottom()
        width = sr.width() if self.orientation() == QtCore.Qt.Horizontal else sr.height()
        end = gr.right() if self.orientation() == QtCore.Qt.Horizontal else gr.bottom()
        self.geometryChanged.emit(start, left, width, right, end)

class Widget(QtWidgets.QWidget):
    def __init__(self, parent=None):
        super(Widget, self).__init__(parent)

        hlayv = QtWidgets.QHBoxLayout()
        hlayv.addWidget(QtWidgets.QLabel("start: "))
        self._start_label_v = QtWidgets.QLabel()
        hlayv.addWidget(self._start_label_v)
        hlayv.addWidget(QtWidgets.QLabel("left: "))
        self._left_label_v = QtWidgets.QLabel()
        hlayv.addWidget(self._left_label_v)
        hlayv.addWidget(QtWidgets.QLabel("width: "))
        self._width_label_v = QtWidgets.QLabel()
        hlayv.addWidget(self._width_label_v)
        hlayv.addWidget(QtWidgets.QLabel("right: "))
        self._right_label_v = QtWidgets.QLabel()
        hlayv.addWidget(self._right_label_v)
        hlayv.addWidget(QtWidgets.QLabel("end: "))
        self._end_label_v = QtWidgets.QLabel()
        hlayv.addWidget(self._end_label_v)

        hlayh = QtWidgets.QHBoxLayout()
        hlayh.addWidget(QtWidgets.QLabel("start: "))
        self._start_label_h = QtWidgets.QLabel()
        hlayh.addWidget(self._start_label_h)
        hlayh.addWidget(QtWidgets.QLabel("left: "))
        self._left_label_h = QtWidgets.QLabel()
        hlayh.addWidget(self._left_label_h)
        hlayh.addWidget(QtWidgets.QLabel("width: "))
        self._width_label_h = QtWidgets.QLabel()
        hlayh.addWidget(self._width_label_h)
        hlayh.addWidget(QtWidgets.QLabel("right: "))
        self._right_label_h = QtWidgets.QLabel()
        hlayh.addWidget(self._right_label_h)
        hlayh.addWidget(QtWidgets.QLabel("end: "))
        self._end_label_h = QtWidgets.QLabel()
        hlayh.addWidget(self._end_label_h)

        self.scrollarea = QtWidgets.QScrollArea()
        content_widget = QtWidgets.QLabel()
        content_widget.setStyleSheet('''background-color : red;''')
        content_widget.setFixedSize(1000, 1000)
        self.scrollarea.setWidget(content_widget)
        hscrollbar = Scrollbar()
        hscrollbar.geometryChanged.connect(self.on_vertical_geometryChanged)
        self.scrollarea.setHorizontalScrollBar(hscrollbar)
        self.scrollarea.setHorizontalScrollBarPolicy(QtCore.Qt.ScrollBarAlwaysOn)

        vscrollbar =Scrollbar()
        self.scrollarea.setVerticalScrollBar(vscrollbar)
        vscrollbar.geometryChanged.connect(self.on_horizontal_geometryChanged)
        self.scrollarea.setHorizontalScrollBarPolicy(QtCore.Qt.ScrollBarAlwaysOn)

        vlay = QtWidgets.QVBoxLayout(self)
        vlay.addWidget(QtWidgets.QLabel("<b>Vertical: </b>"))
        vlay.addLayout(hlayv)
        vlay.addWidget(QtWidgets.QLabel("<b>Horizontal: </b>"))
        vlay.addLayout(hlayh)
        vlay.addWidget(self.scrollarea)

    @QtCore.pyqtSlot(int, int, int, int, int)
    def on_vertical_geometryChanged(self, start, left, width, right, end):
        self._start_label_v.setNum(start)
        self._left_label_v.setNum(left)
        self._width_label_v.setNum(width)
        self._right_label_v.setNum(right)
        self._end_label_v.setNum(end)

    QtCore.pyqtSlot(int, int, int, int, int)
    def on_horizontal_geometryChanged(self, start, left, width, right, end):
        self._start_label_h.setNum(start)
        self._left_label_h.setNum(left)
        self._width_label_h.setNum(width)
        self._right_label_h.setNum(right)
        self._end_label_h.setNum(end)

if __name__ == '__main__':
    import sys
    app = QtWidgets.QApplication(sys.argv)
    w = Widget()
    w.show()
    sys.exit(app.exec_())

【讨论】:

非常感谢@eyllanesc 提供的示例 - 这最终并非微不足道(与我一开始的预期不同);很高兴有这个记录!

以上是关于在 PyQt5 中获取滚动条拇指的大小/宽度?的主要内容,如果未能解决你的问题,请参考以下文章

如何获取浏览器的滚动条大小?

如何设置滚动条的宽度及颜色?

在js中怎么设置滚动条滚动的距离

获取滚动条宽度

VC中怎么滚动条宽度怎么解决

如何在 UWP ScrollViewer 中自定义水平滚动拇指