Skip to content

Commit 89b8de2

Browse files
committed
DWaterProgress
1 parent 1fa64ba commit 89b8de2

File tree

1 file changed

+243
-0
lines changed

1 file changed

+243
-0
lines changed

Diff for: QProgressBar/Lib/DWaterProgress.py

+243
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,243 @@
1+
#!/usr/bin/env python
2+
# -*- coding: utf-8 -*-
3+
4+
"""
5+
Created on 2021/1/1
6+
@author: Irony
7+
@site: https://pyqt5.com , https://github.com/892768447
8+
9+
@file: DWaterProgress
10+
@see https://github.com/linuxdeepin/dtkwidget/blob/master/src/widgets/dwaterprogress.cpp
11+
@description:
12+
"""
13+
import math
14+
15+
from PyQt5.QtCore import pyqtSlot, QTimer, QSizeF, Qt, QRectF, QPointF, QRect, QPoint, QSize
16+
from PyQt5.QtGui import QImage, QColor, QPainter, QLinearGradient, QGradient, QPainterPath, QPixmap, QBrush, QPen
17+
from PyQt5.QtSvg import QSvgRenderer
18+
from PyQt5.QtWidgets import QProgressBar, QGraphicsDropShadowEffect
19+
20+
WATER_FRONT = """<svg xmlns="http://www.w3.org/2000/svg" width="383" height="115" viewBox="0 0 383 115">
21+
<path fill="#01C4FF" fill-rule="evenodd" d="M383,115 L383,14.1688789 C380.269872,14.0716143 377.092672,13.5814974 373.063461,12.4722672 C368.696509,11.2699114 362.241136,10.1727531 357.649256,10.1227411 C347.007291,10.0071963 342.744795,10.6014761 332.930121,12.0276784 C326.157898,13.0120512 317.51313,12.4953762 311.375303,10.33762 C305.58601,8.30230681 299.587109,8.09191178 293.164466,8.16675723 C284.09108,8.27264456 276.303198,11.8021073 267.219716,11.3406179 C260.695053,11.0091595 256.565913,8.56512814 248.546835,8.86450991 C241.871757,9.11387975 235.569934,13.1896798 228.881972,13.3297132 C219.538394,13.525622 215.498041,10.7384053 208.282229,8.42337018 C201.688974,6.30769299 190.725982,6.45048568 185.454442,8.65549452 C170.142255,15.0597811 162.05946,9.31703167 150.536236,5.36712375 C147.826999,4.43862637 144.672431,3.20971247 141.663406,2.90998579 C135.153716,2.26155522 129.812539,3.9788615 123.613779,5.46231888 C115.747555,7.3451819 106.643181,6.73503633 99.4869089,3.84572629 C96.4124243,2.60474055 93.6255416,0.951587506 90.1882469,0.261077932 C79.652131,-1.85528907 69.7970674,9.59778831 58.8051757,9.35186757 C49.4744806,9.14319709 42.6942497,2.4740197 33.3934986,1.93078665 C20.5224457,1.17888312 19.3845731,15.343297 0,13.8463882 L0,115 L383,115 Z"/>
22+
</svg>
23+
"""
24+
WATER_BACK = """<svg xmlns="http://www.w3.org/2000/svg" width="383" height="115" viewBox="0 0 383 115">
25+
<path fill="#007DFF" fill-rule="evenodd" d="M383,115 L383,14.1688789 C380.269872,14.0716143 377.092672,13.5814974 373.063461,12.4722672 C368.696509,11.2699114 362.241136,10.1727531 357.649256,10.1227411 C347.007291,10.0071963 342.744795,10.6014761 332.930121,12.0276784 C326.157898,13.0120512 317.51313,12.4953762 311.375303,10.33762 C305.58601,8.30230681 299.587109,8.09191178 293.164466,8.16675723 C284.09108,8.27264456 276.303198,11.8021073 267.219716,11.3406179 C260.695053,11.0091595 256.565913,8.56512814 248.546835,8.86450991 C241.871757,9.11387975 235.569934,13.1896798 228.881972,13.3297132 C219.538394,13.525622 215.498041,10.7384053 208.282229,8.42337018 C201.688974,6.30769299 190.725982,6.45048568 185.454442,8.65549452 C170.142255,15.0597811 162.05946,9.31703167 150.536236,5.36712375 C147.826999,4.43862637 144.672431,3.20971247 141.663406,2.90998579 C135.153716,2.26155522 129.812539,3.9788615 123.613779,5.46231888 C115.747555,7.3451819 106.643181,6.73503633 99.4869089,3.84572629 C96.4124243,2.60474055 93.6255416,0.951587506 90.1882469,0.261077932 C79.652131,-1.85528907 69.7970674,9.59778831 58.8051757,9.35186757 C49.4744806,9.14319709 42.6942497,2.4740197 33.3934986,1.93078665 C20.5224457,1.17888312 19.3845731,15.343297 0,13.8463882 L0,115 L383,115 Z"/>
26+
</svg>
27+
"""
28+
29+
30+
class Pop:
31+
# https://github.com/linuxdeepin/dtkwidget/blob/master/src/widgets/dwaterprogress.cpp#L36
32+
33+
def __init__(self, size, xs, ys, xo=0, yo=0):
34+
self.size = size
35+
self.xSpeed = xs
36+
self.ySpeed = ys
37+
self.xOffset = xo
38+
self.yOffset = yo
39+
40+
41+
class DWaterProgress(QProgressBar):
42+
43+
def __init__(self, *args, **kwargs):
44+
super(DWaterProgress, self).__init__(*args, **kwargs)
45+
self.waterFrontImage = QImage()
46+
self.waterBackImage = QImage()
47+
self.waterFrontSvg = QSvgRenderer(WATER_FRONT.encode())
48+
self.waterBackSvg = QSvgRenderer(WATER_BACK.encode())
49+
self.pops = []
50+
self.initPops()
51+
self.setTextVisible(True)
52+
self.interval = 33
53+
self.timer = QTimer(self)
54+
self.timer.setInterval(self.interval)
55+
self.timer.timeout.connect(self.onTimerOut)
56+
self.resizePixmap(self.size())
57+
self.frontXOffset = self.width()
58+
self.backXOffset = 0
59+
effect = QGraphicsDropShadowEffect(self)
60+
effect.setOffset(0, 6)
61+
effect.setColor(QColor(1, 153, 248, 255 * 5 / 20))
62+
effect.setBlurRadius(12)
63+
self.setGraphicsEffect(effect)
64+
65+
def initPops(self):
66+
self.pops = [Pop(7, -1.8, 0.6), Pop(8, 1.2, 1.0), Pop(11, 0.8, 1.6)]
67+
68+
@pyqtSlot()
69+
def start(self):
70+
self.timer.start()
71+
72+
@pyqtSlot()
73+
def stop(self):
74+
self.timer.stop()
75+
76+
def resizePixmap(self, sz):
77+
# https://github.com/linuxdeepin/dtkwidget/blob/master/src/widgets/dwaterprogress.cpp#L192
78+
# resize water
79+
waterWidth = 500 * sz.width() / 100
80+
waterHeight = 110 * sz.height() / 100
81+
waterSize = QSizeF(waterWidth, waterHeight).toSize()
82+
83+
if self.waterFrontImage.size() != waterSize:
84+
image = QImage(waterWidth, waterHeight, QImage.Format_ARGB32)
85+
image.fill(Qt.transparent)
86+
waterPainter = QPainter(image)
87+
self.waterFrontSvg.render(waterPainter)
88+
self.waterFrontImage = image
89+
90+
if self.waterBackImage.size() != waterSize:
91+
image = QImage(waterWidth, waterHeight, QImage.Format_ARGB32)
92+
image.fill(Qt.transparent) # partly transparent red-ish background
93+
waterPainter = QPainter(image)
94+
self.waterBackSvg.render(waterPainter)
95+
self.waterBackImage = image
96+
97+
def onTimerOut(self):
98+
# interval can not be zero, and limit to 1
99+
self.interval = max(1, self.interval)
100+
# move 60% per second
101+
frontXDeta = 40.0 / (1000.0 / self.interval)
102+
# move 90% per second
103+
backXDeta = 60.0 / (1000.0 / self.interval)
104+
105+
canvasWidth = int(self.width() * self.devicePixelRatioF())
106+
self.frontXOffset -= frontXDeta * canvasWidth / 100
107+
self.backXOffset += backXDeta * canvasWidth / 100
108+
109+
if self.frontXOffset > canvasWidth:
110+
self.frontXOffset = canvasWidth
111+
112+
if self.frontXOffset < - (self.waterFrontImage.width() - canvasWidth):
113+
self.frontXOffset = canvasWidth
114+
115+
if self.backXOffset > self.waterBackImage.width():
116+
self.backXOffset = 0
117+
118+
# update pop
119+
# move 25% per second default
120+
speed = 25 / (1000.0 / self.interval) # 100 / self.height()
121+
for pop in self.pops:
122+
# yOffset 0 ~ 100
123+
pop.yOffset += speed * pop.ySpeed
124+
if pop.yOffset < 0:
125+
pass
126+
if pop.yOffset > self.value():
127+
pop.yOffset = 0
128+
pop.xOffset = math.sin((pop.yOffset / 100) * 2 * 3.14) * 18 * pop.xSpeed + 50
129+
self.update()
130+
131+
def paint(self, painter):
132+
painter.setRenderHint(QPainter.Antialiasing)
133+
134+
pixelRatio = self.devicePixelRatioF()
135+
rect = QRectF(0, 0, self.width() * pixelRatio, self.height() * pixelRatio)
136+
sz = QSizeF(self.width() * pixelRatio, self.height() * pixelRatio).toSize()
137+
138+
self.resizePixmap(sz)
139+
140+
yOffset = rect.toRect().topLeft().y() + (100 - self.value() - 10) * sz.height() / 100
141+
142+
# draw water
143+
waterImage = QImage(sz, QImage.Format_ARGB32_Premultiplied)
144+
waterPainter = QPainter()
145+
waterPainter.begin(waterImage)
146+
waterPainter.setRenderHint(QPainter.Antialiasing)
147+
waterPainter.setCompositionMode(QPainter.CompositionMode_Source)
148+
149+
pointStart = QPointF(sz.width() / 2, 0)
150+
pointEnd = QPointF(sz.width() / 2, sz.height())
151+
linear = QLinearGradient(pointStart, pointEnd)
152+
startColor = QColor('#1F08FF')
153+
startColor.setAlphaF(1)
154+
endColor = QColor('#50FFF7')
155+
endColor.setAlphaF(0.28)
156+
linear.setColorAt(0, startColor)
157+
linear.setColorAt(1, endColor)
158+
linear.setSpread(QGradient.PadSpread)
159+
waterPainter.setPen(Qt.NoPen)
160+
waterPainter.setBrush(linear)
161+
waterPainter.drawEllipse(waterImage.rect().center(), sz.width() / 2 + 1, sz.height() / 2 + 1)
162+
163+
waterPainter.setCompositionMode(QPainter.CompositionMode_SourceOver)
164+
waterPainter.drawImage(int(self.backXOffset), yOffset, self.waterBackImage)
165+
waterPainter.drawImage(int(self.backXOffset) - self.waterBackImage.width(), yOffset, self.waterBackImage)
166+
waterPainter.drawImage(int(self.frontXOffset), yOffset, self.waterFrontImage)
167+
waterPainter.drawImage(int(self.frontXOffset) - self.waterFrontImage.width(), yOffset, self.waterFrontImage)
168+
169+
# draw pop
170+
if self.value() > 30:
171+
for pop in self.pops:
172+
popPath = QPainterPath()
173+
popPath.addEllipse(pop.xOffset * sz.width() / 100, (100 - pop.yOffset) * sz.height() / 100,
174+
pop.size * sz.width() / 100, pop.size * sz.height() / 100)
175+
waterPainter.fillPath(popPath, QColor(255, 255, 255, 255 * 0.3))
176+
177+
if self.isTextVisible():
178+
font = waterPainter.font()
179+
rectValue = QRect()
180+
progressText = self.text().strip('%')
181+
182+
if progressText == '100':
183+
font.setPixelSize(sz.height() * 35 / 100)
184+
waterPainter.setFont(font)
185+
186+
rectValue.setWidth(sz.width() * 60 / 100)
187+
rectValue.setHeight(sz.height() * 35 / 100)
188+
rectValue.moveCenter(rect.center().toPoint())
189+
waterPainter.setPen(Qt.white)
190+
waterPainter.drawText(rectValue, Qt.AlignCenter, progressText)
191+
else:
192+
font.setPixelSize(sz.height() * 40 / 100)
193+
waterPainter.setFont(font)
194+
195+
rectValue.setWidth(sz.width() * 45 / 100)
196+
rectValue.setHeight(sz.height() * 40 / 100)
197+
rectValue.moveCenter(rect.center().toPoint())
198+
rectValue.moveLeft(rect.left() + rect.width() * 0.45 * 0.5)
199+
200+
waterPainter.setPen(Qt.white)
201+
waterPainter.drawText(rectValue, Qt.AlignCenter, progressText)
202+
font.setPixelSize(font.pixelSize() / 2)
203+
waterPainter.setFont(font)
204+
rectPerent = QRect(QPoint(rectValue.right(), rectValue.bottom() - rect.height() * 20 / 100),
205+
QPoint(rectValue.right() + rect.width() * 20 / 100, rectValue.bottom()))
206+
207+
waterPainter.drawText(rectPerent, Qt.AlignCenter, '%')
208+
209+
waterPainter.end()
210+
211+
maskPixmap = QPixmap(sz)
212+
maskPixmap.fill(Qt.transparent)
213+
path = QPainterPath()
214+
path.addEllipse(QRectF(0, 0, sz.width(), sz.height()))
215+
maskPainter = QPainter()
216+
maskPainter.begin(maskPixmap)
217+
maskPainter.setRenderHint(QPainter.Antialiasing)
218+
maskPainter.setPen(QPen(Qt.white, 1))
219+
maskPainter.fillPath(path, QBrush(Qt.white))
220+
maskPainter.end()
221+
222+
mode = QPainter.CompositionMode_SourceIn
223+
contentImage = QImage(sz, QImage.Format_ARGB32_Premultiplied)
224+
contentPainter = QPainter()
225+
contentPainter.begin(contentImage)
226+
contentPainter.setCompositionMode(QPainter.CompositionMode_Source)
227+
contentPainter.fillRect(contentImage.rect(), Qt.transparent)
228+
contentPainter.setCompositionMode(QPainter.CompositionMode_SourceOver)
229+
contentPainter.drawImage(0, 0, maskPixmap.toImage())
230+
contentPainter.setCompositionMode(mode)
231+
contentPainter.drawImage(0, 0, waterImage)
232+
contentPainter.setCompositionMode(QPainter.CompositionMode_DestinationOver)
233+
contentPainter.end()
234+
235+
contentImage.setDevicePixelRatio(pixelRatio)
236+
painter.drawImage(self.rect(), contentImage)
237+
238+
def paintEvent(self, event):
239+
painter = QPainter(self)
240+
self.paint(painter)
241+
242+
def sizeHint(self):
243+
return QSize(100, 100)

0 commit comments

Comments
 (0)