python3+PyQt5實現(xiàn)自定義分?jǐn)?shù)滑塊部件
更新時間:2018年04月24日 14:28:51 作者:basisworker
這篇文章主要為大家詳細(xì)介紹了python3+PyQt5實現(xiàn)自定義分?jǐn)?shù)滑塊部件,具有一定的參考價值,感興趣的小伙伴們可以參考一下
本文通過Python3+PyQt5實現(xiàn)自定義部件–分?jǐn)?shù)滑塊。它既能支持鍵盤也支持鼠標(biāo),使用物理(視口)坐標(biāo)通過繪制方式顯示。
#!/usr/bin/env python3
import platform
from PyQt5.QtCore import (QPointF, QRectF, QSize, Qt,pyqtSignal)
from PyQt5.QtWidgets import (QApplication, QDialog,QSizePolicy,
QGridLayout, QLCDNumber, QLabel,
QSpinBox, QWidget)
from PyQt5.QtGui import QColor,QFont,QPainter,QFontMetricsF,QPalette, QPolygonF
X11 = True
try:
from PyQt5.QtGui import qt_x11_wait_for_window_manager
except ImportError:
X11 = False
class FractionSlider(QWidget):
XMARGIN = 12.0
YMARGIN = 5.0
WSTRING = "999"
valueChanged = pyqtSignal(int,int)
def __init__(self, numerator=0, denominator=10, parent=None):
super(FractionSlider, self).__init__(parent)
self.__numerator = numerator
self.__denominator = denominator
self.setFocusPolicy(Qt.WheelFocus)
self.setSizePolicy(QSizePolicy(QSizePolicy.MinimumExpanding,
QSizePolicy.Fixed))
def decimal(self):
return self.__numerator / float(self.__denominator)
def fraction(self):
return self.__numerator, self.__denominator
def sizeHint(self):
return self.minimumSizeHint()
def minimumSizeHint(self):
font = QFont(self.font())
font.setPointSize(font.pointSize() - 1)
fm = QFontMetricsF(font)
return QSize(fm.width(FractionSlider.WSTRING) *
self.__denominator,
(fm.height() * 4) + FractionSlider.YMARGIN)
def setFraction(self, numerator, denominator=None):
if denominator is not None:
if 3 <= denominator <= 60:
self.__denominator = denominator
else:
raise ValueError("denominator out of range")
if 0 <= numerator <= self.__denominator:
self.__numerator = numerator
else:
raise ValueError("numerator out of range")
self.update()
self.updateGeometry()
def mousePressEvent(self, event):
if event.button() == Qt.LeftButton:
self.moveSlider(event.x())
event.accept()
else:
QWidget.mousePressEvent(self, event)
def mouseMoveEvent(self, event):
self.moveSlider(event.x())
def moveSlider(self, x):
span = self.width() - (FractionSlider.XMARGIN * 2)
offset = span - x + FractionSlider.XMARGIN
numerator = int(round(self.__denominator *
(1.0 - (offset / span))))
numerator = max(0, min(numerator, self.__denominator))
if numerator != self.__numerator:
self.__numerator = numerator
#self.emit(SIGNAL("valueChanged(int,int)"),
# self.__numerator, self.__denominator)
self.valueChanged.emit(self.__numerator, self.__denominator)
self.update()
def keyPressEvent(self, event):
change = 0
if event.key() == Qt.Key_Home:
change = -self.__denominator
elif event.key() in (Qt.Key_Up, Qt.Key_Right):
change = 1
elif event.key() == Qt.Key_PageUp:
change = (self.__denominator // 10) + 1
elif event.key() in (Qt.Key_Down, Qt.Key_Left):
change = -1
elif event.key() == Qt.Key_PageDown:
change = -((self.__denominator // 10) + 1)
elif event.key() == Qt.Key_End:
change = self.__denominator
if change:
numerator = self.__numerator
numerator += change
numerator = max(0, min(numerator, self.__denominator))
if numerator != self.__numerator:
self.__numerator = numerator
#self.emit(SIGNAL("valueChanged(int,int)"),
# self.__numerator, self.__denominator)
self.valueChanged.emit(self.__numerator, self.__denominator)
self.update()
event.accept()
else:
QWidget.keyPressEvent(self, event)
def paintEvent(self, event=None):
font = QFont(self.font())
font.setPointSize(font.pointSize() - 1)
fm = QFontMetricsF(font)
fracWidth = fm.width(FractionSlider.WSTRING)
indent = fm.boundingRect("9").width() / 2.0
if not X11:
fracWidth *= 1.5
span = self.width() - (FractionSlider.XMARGIN * 2)
value = self.__numerator / float(self.__denominator)
painter = QPainter(self)
painter.setRenderHint(QPainter.Antialiasing)
painter.setRenderHint(QPainter.TextAntialiasing)
painter.setPen(self.palette().color(QPalette.Mid))
painter.setBrush(self.palette().brush(
QPalette.AlternateBase))
painter.drawRect(self.rect())
segColor = QColor(Qt.green).darker(120)
segLineColor = segColor.darker()
painter.setPen(segLineColor)
painter.setBrush(segColor)
painter.drawRect(FractionSlider.XMARGIN,
FractionSlider.YMARGIN, span, fm.height())
textColor = self.palette().color(QPalette.Text)
segWidth = span / self.__denominator
segHeight = fm.height() * 2
nRect = fm.boundingRect(FractionSlider.WSTRING)
x = FractionSlider.XMARGIN
yOffset = segHeight + fm.height()
for i in range(self.__denominator + 1):
painter.setPen(segLineColor)
painter.drawLine(x, FractionSlider.YMARGIN, x, segHeight)
painter.setPen(textColor)
y = segHeight
rect = QRectF(nRect)
rect.moveCenter(QPointF(x, y + fm.height() / 2.0))
#painter.drawText(rect, Qt.AlignCenter,
#QString.number(i))
painter.drawText(rect, Qt.AlignCenter,str(i))
y = yOffset
rect.moveCenter(QPointF(x, y + fm.height() / 2.0))
painter.drawText(rect, Qt.AlignCenter,
str(self.__denominator))
painter.drawLine(QPointF(rect.left() + indent, y),
QPointF(rect.right() - indent, y))
x += segWidth
span = int(span)
y = FractionSlider.YMARGIN - 0.5
triangle = [QPointF(value * span, y),
QPointF((value * span) +
(2 * FractionSlider.XMARGIN), y),
QPointF((value * span) +
FractionSlider.XMARGIN, fm.height())]
painter.setPen(Qt.yellow)
painter.setBrush(Qt.darkYellow)
painter.drawPolygon(QPolygonF(triangle))
if __name__ == "__main__":
import sys
app = QApplication(sys.argv)
form = QDialog()
sliderLabel = QLabel("&Fraction")
slider = FractionSlider(denominator=12)
sliderLabel.setBuddy(slider)
denominatorLabel = QLabel("&Denominator")
denominatorSpinBox = QSpinBox()
denominatorLabel.setBuddy(denominatorSpinBox)
denominatorSpinBox.setRange(3, 60)
denominatorSpinBox.setValue(slider.fraction()[1])
denominatorSpinBox.setAlignment(Qt.AlignRight|Qt.AlignVCenter)
numeratorLabel = QLabel("Numerator")
numeratorLCD = QLCDNumber()
numeratorLCD.setSegmentStyle(QLCDNumber.Flat)
layout = QGridLayout()
layout.addWidget(sliderLabel, 0, 0)
layout.addWidget(slider, 0, 1, 1, 5)
layout.addWidget(numeratorLabel, 1, 0)
layout.addWidget(numeratorLCD, 1, 1)
layout.addWidget(denominatorLabel, 1, 2)
layout.addWidget(denominatorSpinBox, 1, 3)
form.setLayout(layout)
def valueChanged(denominator):
numerator = int(slider.decimal() * denominator)
slider.setFraction(numerator, denominator)
numeratorLCD.display(numerator)
#form.connect(slider, SIGNAL("valueChanged(int,int)"),
#numeratorLCD, SLOT("display(int)"))
slider.valueChanged[int,int].connect(numeratorLCD.display)
#form.connect(denominatorSpinBox, SIGNAL("valueChanged(int)"),
#valueChanged)
denominatorSpinBox.valueChanged[int].connect(valueChanged)
form.setWindowTitle("Fraction Slider")
form.show()
app.exec_()
運行結(jié)果:

以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Python中使用format函數(shù)的小結(jié)
在Python中,format()函數(shù)是一種用于格式化字符串的方法主要介紹了Python中使用format函數(shù)的小結(jié),本文就來介紹一下format()函數(shù)的使用示例,感興趣的可以了解一下2023-08-08
基于Python+Appium實現(xiàn)京東雙十一自動領(lǐng)金幣功能
一年一度的雙十一即將來臨,各大平臺都在搞活動,京東天貓忙的不易樂乎,做任務(wù)領(lǐng)金幣的過程真的好無聊,今天小編給大家分享一篇教程通關(guān)Python+Appium實現(xiàn)京東雙十一自動領(lǐng)金幣功能,需要的朋友可以參考下2019-10-10
Pandas中常用的七個時間戳處理函數(shù)使用總結(jié)
在零售、經(jīng)濟(jì)和金融等行業(yè),數(shù)據(jù)總是由于貨幣和銷售而不斷變化,生成的所有數(shù)據(jù)都高度依賴于時間。如果這些數(shù)據(jù)沒有時間戳或標(biāo)記,實際上很難管理所有收集的數(shù)據(jù)。本文為大家準(zhǔn)備了Pandas中常用的七個時間戳處理函數(shù),需要的可以參考一下2022-04-04
PyTorch加載預(yù)訓(xùn)練模型實例(pretrained)
今天小編就為大家分享一篇PyTorch加載預(yù)訓(xùn)練模型實例(pretrained),具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-01-01
基于PyQt5制作Excel文件數(shù)據(jù)去重小工具
這篇文章主要介紹了如何利用PyQt5模塊制作一個Excel文件數(shù)據(jù)去重小工具,可以將單個或者多個Excel文件數(shù)據(jù)進(jìn)行去重操作,去重的列可以通過自定義制定,需要的可以參考一下2022-04-04

