磨砂效果的实现思路
这两周一直在思考怎么在pyqt上实现窗口磨砂效果,网上搜了一圈,全都是Qt的实现方法,而且都是通过调用windows的api来实现磨砂效果的。本来想试试能不能照猫画虎,把C++给翻译成python,看了网上的代码发现C++的一些数据结构python没有(也有可能是自己太菜没发现),感觉直接翻译不太行。正好今天查python的官方文档的时候看到了ctypes里面的HWND,激动地连叫woc,既然来硬♂的不行,倒不如换一条思路:让C++帮我做磨砂的事情,而我只需要在python中调用封装好的dll中的接口函数就行了。需要注意的是,要想成功调用这个dll必须安好MSVC,不想安的话可以参见我的第七篇博客《如何在pyqt中通过调用SetWindowCompositionAttribute实现Win10亚克力效果》,这里面用纯python的方式实现了窗口磨砂效果。
具体实现流程
-
先写一个实现磨砂效果的C++函数setBlur(),调用的时候只需往这个函数传递窗口句柄就行了;
-
在VS2019里面把写好的函数编译为aeroDll.dll;
-
在python通过hWnd=ctypes.wintypes.HWND(int(self.winId()));把WId类型强制转换为HWND类型,接着只需使用一条命令ctypes.cdll.LoadLibrary('Aero\\aeroDll.dll').setBlur(hWnd),就能实现动态链接库的载入和接口函数的调用,源代码如下:
import sys
from ctypes import cdll
from ctypes.wintypes import HWND
from PyQt5.QtCore import Qt
from PyQt5.QtGui import QColor, QPainter
from PyQt5.QtWidgets import QApplication, QWidget
class Demo(QWidget):
""" 磨砂效果的实现 """
def __init__(self):
super().__init__()
self.resize(500, 500)
# 去除边框
self.setWindowFlags(Qt.FramelessWindowHint)
# 背景透明
self.setAttribute(Qt.WA_TranslucentBackground)
# 设置背景色
self.bgColor = QColor(255,50,50,80) # 可以根据个人需要调节透明度
# 调用api
hWnd = HWND(int(self.winId())) # 直接HWND(self.winId())会报错
cdll.LoadLibrary('Aero\\aeroDll.dll').setBlur(hWnd) # dll和脚本放在同一个目录下会报错找不到dll
def paintEvent(self, e):
""" 绘制背景 """
painter = QPainter(self)
painter.setRenderHint(QPainter.Antialiasing)
painter.setPen(Qt.NoPen)
painter.setBrush(self.bgColor)
painter.drawRoundedRect(self.rect(), 20, 20)
if __name__ == "__main__":
app = QApplication(sys.argv)
demo = Demo()
demo.show()
sys.exit(app.exec_())
运行效果如下图所示(我老婆真美(~ ̄▽ ̄)~
源代码和编译完的dll
百度网盘(提取码:7o6r): link