PySide2学习记录(十四):控件的风格和简单绘制

Python版本3.7
PySide2 Version: 5.14.1
官方文档:http://doc.qt.io/qtforpython/index.html

QStyle类是一个抽象类,里面包含了各个控件的外观风格,由于这方面涉及到的东西太多,有些我也没有弄太懂,只能简单记录下。
下面是关于QStyleFactory的使用,设置应用的风格,比如在Mac上实现windows风格。

from PySide2.QtWidgets import QApplication, QWidget, QPushButton, \
    QVBoxLayout, QStyleFactory, QProgressBar

app = QApplication()
# 打印支持的风格
print(QStyleFactory.keys())
# 设置应用全局风格
# app.setStyle(QStyleFactory.create('fusion'))

window = QWidget()
layout = QVBoxLayout()

btn01 = QPushButton('111')
btn02 = QPushButton('222')
btn03 = QPushButton('333')
progress = QProgressBar()

# 设置单个控件风格
btn02.setStyle(QStyleFactory.create('fusion'))
btn03.setStyle(QStyleFactory.create('windows'))
progress.setStyle(QStyleFactory.create('fusion'))

progress.setValue(60)

layout.addWidget(btn01)
layout.addWidget(btn02)
layout.addWidget(btn03)
layout.addWidget(progress)

window.setLayout(layout)
window.show()
app.exec_()

运行截图:

图1

QStyleFactory是一个独立的类,即没有父类也没有子类,它只有两个静态函数,一个是create(arg_1),传入一个字符串参数,如上面例子中所示,还有一个是keys(),用来查看支持哪些风格。每个控件都可以单独设置自己的风格,也可以直接设置全局风格。
如果不满意系统自带的风格,我们也可以自己绘制想要的风格,如下。

from PySide2.QtWidgets import QPushButton, QApplication, QVBoxLayout, QWidget
from PySide2.QtGui import QPainter, QColor, Qt, QPen

class MyPushButton(QPushButton):

    def __init__(self, s, parent=None):
        super().__init__(s, parent)
        self.is_pressed = False
        self.is_entered = False
        
        # 鼠标按压事件
    def mousePressEvent(self, event):
        self.is_pressed = True
        self.update()
        
    # 鼠标释放事件
    def mouseReleaseEvent(self, event):
        self.is_pressed = False
        self.update()
        
    # 鼠标进入事件
    def enterEvent(self, event):
        self.is_entered = True
        self.update()
        
    # 鼠标离开事件
    def leaveEvent(self, event):
        self.is_entered = False
        self.update()
        
    # 绘图事件
    def paintEvent(self, event):
        painter = QPainter(self)
        # 获取绘制区域
        rect = self.rect()
        if self.is_entered:
            # 填充区域颜色
            painter.fillRect(rect, QColor("#B8B8B8"))
            if self.is_pressed:
                painter.fillRect(rect, QColor("#FF0000"))
        else:
            painter.fillRect(rect, QColor("#E9E9E9"))
        # 绘制文本
        painter.drawText(rect, Qt.AlignCenter, self.text())
        pen = QPen(Qt.SolidLine)
        pen.setColor(QColor("#A9A9A9"))
        pen.setWidth(2)
        # 设置画笔
        painter.setPen(pen)
        # 绘制边框
        painter.drawRect(rect)


app = QApplication()
window = QWidget()
layout = QVBoxLayout()

btn = MyPushButton('我是按钮')
default_btn = QPushButton('我是默认按钮')
layout.addWidget(btn)
layout.addWidget(default_btn)

window.setLayout(layout)
window.show()
app.exec_()

运行截图:

图2

除了像上面那样进行绘制之外,还有一种方式可以进行绘制。

from PySide2.QtWidgets import QWidget, QPushButton, QApplication, \
    QVBoxLayout, QStyleOptionButton, QStyle
from PySide2.QtGui import QPainter, QColor

class MyPushButton(QPushButton):
    def __init__(self, s, parent=None):
        super().__init__(s, parent)

    def paintEvent(self, event):
        painter = QPainter(self)
        btn = QStyleOptionButton()
        btn.rect = self.rect()
        # adjusted用于对rect进行微调
        painter.fillRect(btn.rect.adjusted(7, 3, -7, -3), QColor("#FF6699"))
        btn.text = self.text()
        self.style().drawControl(QStyle.CE_PushButton, btn, painter)


app = QApplication()
window = QWidget()
layout = QVBoxLayout()

button01 = MyPushButton('按钮')
button02 = QPushButton('默认按钮')
layout.addWidget(button01)
layout.addWidget(button02)

window.setLayout(layout)
window.show()
app.exec_()

上面例子使用了QStyleOptionButton这个类(其它类似的类请在官方文档查看),注意checkbox,radiobutton和pushbutton都属于button。QStyleOptionButton里面包含了要绘制这些按钮的所有信息,我们填充了好了QStyleOptionButton的对象之后,就可以使用QStyle类里面的drawControl方法来绘制这个控件,实际上这个方法主要用于绘制控制元素,主要是与用户交互或者显示一些信息的元素(详细请参考QStyle),比如按钮,单选框,进度条等等,其中QStyle.CE_PushButton代表了这个元素的样式,点击查看元素样式注意上面两个例子都是直接重新绘制控件的外观,如果我们想像第一个例子那样,使用函数setStyle()来设置来更改外观的话,需要自己继承QStyle类或者它的子类(一般情况下是继承子类),比如QCommonStyle类,然后将这个类的对象传递给setStyle函数,具体查看文档。下面提供一个例子仅供参考。

from PySide2.QtWidgets import QApplication, QCommonStyle, QVBoxLayout, \
    QStyle, QWidget, QPushButton
from PySide2.QtGui import QBrush, QColor
from PySide2.QtCore import Qt

class MyStyle(QCommonStyle):
    def __init__(self, text):
        super().__init__()
        self.text = text

    def drawControl(self, element, option, painter, w=None):
        default_color = QBrush(QColor('#FF6699'))
        enter_color = QBrush(QColor('#FF9900'))
        press_color = QBrush(QColor('#FF0000'))

        # 在c++中,这里直接可以强制转换成QStyleOptionButton,但是Python里面并不能
        pb = option
        area = pb.rect

        if pb.state & QStyle.State_MouseOver:
            painter.fillRect(area, default_color)
        elif pb.state & QStyle.State_Raised:
            painter.fillRect(area, enter_color)
        if pb.state & QStyle.State_Sunken:
            painter.fillRect(area, press_color)
        painter.drawText(area, Qt.AlignCenter, self.text)

    def polish(self, w):
        # 设置Qt::WA_Hover 属性后,将使鼠标在进入或离开部件时产生绘制事件
        w.setAttribute(Qt.WA_Hover, True)

    def unpolish(self, w):
        w.setAttribute(Qt.WA_Hover, False)


app = QApplication()
win = QWidget()
layout = QVBoxLayout()

btn01 = QPushButton('自定义style的按钮')
btn02 = QPushButton('系统默认style的按钮')
btn01.setStyle(MyStyle(btn01.text()))

layout.addWidget(btn01)
layout.addWidget(btn02)

win.setLayout(layout)
win.show()
app.exec_()

运行截图:

图3

友情链接:
QStyleFactory
QStyle
Styles and Style Aware Widgets
QStyleOptionButton

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,332评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,508评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,812评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,607评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,728评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,919评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,071评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,802评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,256评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,576评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,712评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,389评论 4 332
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,032评论 3 316
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,798评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,026评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,473评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,606评论 2 350