244 lines
No EOL
9.1 KiB
Markdown
244 lines
No EOL
9.1 KiB
Markdown
# Demo
|
||
|
||
- 目录
|
||
- [重启窗口Widget](#1重启窗口Widget)
|
||
- [简单的窗口贴边隐藏](#2简单的窗口贴边隐藏)
|
||
- [嵌入外部窗口](#3嵌入外部窗口)
|
||
- [简单跟随其它窗口](#4简单跟随其它窗口)
|
||
- [简单探测窗口和放大截图](#5简单探测窗口和放大截图)
|
||
- [无边框自定义标题栏窗口](#6无边框自定义标题栏窗口)
|
||
- [右下角弹出框](#7右下角弹出框)
|
||
- [程序重启](#8程序重启)
|
||
- [自定义属性](#9自定义属性)
|
||
- [调用截图DLL](#10调用截图DLL)
|
||
- [单实例应用](#11单实例应用)
|
||
- [简单的右下角气泡提示](#12简单的右下角气泡提示)
|
||
- [右侧消息通知栏](#13右侧消息通知栏)
|
||
- [验证码控件](#14验证码控件)
|
||
- [人脸特征点](#15人脸特征点)
|
||
- [使用Threading](#16使用Threading)
|
||
- [背景连线动画](#17背景连线动画)
|
||
- [无边框圆角对话框](#18无边框圆角对话框)
|
||
- [调整窗口显示边框](#19调整窗口显示边框)
|
||
- [判断信号是否连接](#20判断信号是否连接)
|
||
- [调用虚拟键盘](#21调用虚拟键盘)
|
||
- [动态忙碌光标](#22动态忙碌光标)
|
||
- [屏幕变动监听](#23屏幕变动监听)
|
||
- [无边框窗口](#24无边框窗口)
|
||
|
||
## 1、重启窗口Widget
|
||
[运行 RestartWindow.py](RestartWindow.py)
|
||
|
||
利用类变量对窗口的变量进行引用,防止被回收(导致窗口一闪而过),重启时先显示新窗口后关闭自己
|
||
|
||
![RestartWindow](ScreenShot/RestartWindow.gif)
|
||
|
||
## 2、简单的窗口贴边隐藏
|
||
[运行 WeltHideWindow.py](WeltHideWindow.py)
|
||
|
||
1. 大概思路
|
||
1. 思路是当窗口进入左边,顶部,右边一半时,此时判断窗口的坐标
|
||
1. 如果窗口的x坐标小于0 则需要隐藏到左边
|
||
1. 如果窗口的y坐标小于0 则需要隐藏到顶部
|
||
1. 如果窗口的x坐标大于屏幕宽度-窗口宽度/2 则需要隐藏到右边
|
||
|
||
2. 事件说明
|
||
1. `mousePressEvent`,鼠标按下事件,主要记录按下的坐标
|
||
1. `mouseMoveEvent`,鼠标移动事件,用于移动窗口
|
||
1. `mouseReleaseEvent`,鼠标弹起事件,用于判断是否需要隐藏窗口
|
||
1. `enterEvent`,鼠标进入事件,用于窗口隐藏后,是否需要暂时显示预览
|
||
1. `leaveEvent`,鼠标离开事件,用于窗口暂时显示后自动隐藏效果
|
||
|
||
![WeltHideWindow](ScreenShot/WeltHideWindow.gif)
|
||
|
||
## 3、嵌入外部窗口
|
||
[运行 EmbedWindow.py](EmbedWindow.py)
|
||
|
||
1. 使用`SetParent`函数设置外部窗口的`parent`为Qt的窗口
|
||
1. Qt使用`QWidget.createWindowContainer(QWindow.fromWinId(窗口ID))`生成QWidget
|
||
1. 使用`GetWindowLong`得到原来窗口的样式属性`style = win32gui.GetWindowLong(hwnd, win32con.GWL_STYLE)`和`exstyle = win32gui.GetWindowLong(hwnd, win32con.GWL_EXSTYLE)`
|
||
1. 这里还原窗口后不会显示,用spy++发现没有了WS_VISIBLE样式(未解决)
|
||
|
||
![EmbedWindow](ScreenShot/EmbedWindow.gif)
|
||
|
||
|
||
## 4、简单跟随其它窗口
|
||
[运行 FollowWindow.py](FollowWindow.py)
|
||
|
||
1. 利用win32gui模块获取目标窗口的句柄
|
||
1. 通过句柄获取目标窗口的大小位置,并设置自己的位置
|
||
1. 当句柄失效时关闭自己
|
||
1. 主要是检测时间,在10毫秒以下很流畅
|
||
|
||
![FollowWindow](ScreenShot/FollowWindow.gif)
|
||
|
||
|
||
## 5、简单探测窗口和放大截图
|
||
[运行 ProbeWindow.py](ProbeWindow.py)
|
||
|
||
1. 利用`win32gui`模块获取鼠标所在位置的窗口大小(未去掉边框)和rgb颜色
|
||
1. 利用一个全屏的全透明鼠标穿透的窗口(目的在于绘制矩形框和截图)
|
||
|
||
![ProbeWindow](ScreenShot/ProbeWindow.gif)
|
||
|
||
|
||
## 6、无边框自定义标题栏窗口
|
||
[运行 FramelessWindow.py](FramelessWindow.py) | [运行 NativeEvent.py](NativeEvent.py)
|
||
|
||
1. 重写鼠标事件
|
||
|
||
1. 使用一个`QWidget`(`FramelessWindow`)作为父窗口, 一个`TitleBar`作为标题栏, 一个`QWidget`作为底部容器
|
||
1. 父窗口`FramelessWindow`设置为背景透明,但是需要绘制一定宽度的透明度很高的矩形边框用来接受鼠标事件(变形鼠标样式进行调整窗口大小)
|
||
1. `TitleBar`的最小化最大化关闭等按钮事件关联到父窗口里
|
||
1. `TitleBar`中的鼠标按下移动事件得到坐标也传递到父窗口调用move方法进行窗口移动
|
||
|
||
2. windows api
|
||
|
||
1. 使用`win32gui`设置薄边框
|
||
1. 重写`nativeEvent`事件拦截边框的系统边框的显示,并返回各个方向
|
||
|
||
![FramelessWindow](ScreenShot/FramelessWindow.gif)
|
||
|
||
## 7、右下角弹出框
|
||
[运行 WindowNotify.py](WindowNotify.py)
|
||
|
||
![WindowNotify](ScreenShot/WindowNotify.gif)
|
||
|
||
## 8、程序重启
|
||
[运行 AutoRestart.py](AutoRestart.py)
|
||
|
||
![AutoRestart](ScreenShot/AutoRestart.gif)
|
||
|
||
## 9、自定义属性
|
||
[运行 CustomProperties.py](CustomProperties.py)
|
||
|
||
![CustomProperties](ScreenShot/CustomProperties.png)
|
||
|
||
## 10、调用截图DLL
|
||
[运行 ScreenShotDll.py](ScreenShotDll.py)
|
||
|
||
![ScreenShotDll](ScreenShot/ScreenShotDll.gif)
|
||
|
||
## 11、单实例应用
|
||
[运行 SingleApplication.py](SingleApplication.py) | [运行 SharedMemory.py](SharedMemory.py)
|
||
|
||
1. QSharedMemory
|
||
2. QLocalSocket, QLocalServer
|
||
|
||
## 12、简单的右下角气泡提示
|
||
[运行 BubbleTips.py](BubbleTips.py)
|
||
|
||
1. 使用 `QWidget` 包含一个 `QLabel`, 其中 `QWidget` 通过 `paintEvent` 绘制气泡形状
|
||
2. 使用 `QPropertyAnimation` 属性动画来移动气泡和改变气泡的透明度
|
||
3. 使用 `QParallelAnimationGroup` 动画组来同时运行两个动画
|
||
|
||
![BubbleTips](ScreenShot/BubbleTips.gif)
|
||
|
||
## 13、右侧消息通知栏
|
||
[运行 Notification.py](Notification.py)
|
||
|
||
![Notification](ScreenShot/Notification.gif)
|
||
|
||
## 14、验证码控件
|
||
[运行 VerificationCode.py](VerificationCode.py)
|
||
|
||
1. 更新为paintEvent方式,采用上下跳动
|
||
2. 参考网上一些代码,都是采用paintEvent绘制,这里采用QLabel显示html结合字体来显示文字<br />
|
||
然后在paintEvent中绘制噪点和线条
|
||
|
||
![VerificationCode](ScreenShot/VerificationCode.gif)
|
||
|
||
## 15、人脸特征点
|
||
[运行 FacePoints.py](FacePoints.py)
|
||
|
||
PyQt 结合 Opencv 进行人脸检测;
|
||
由于直接在主线程中进行特征点获取,效率比较低
|
||
|
||
依赖文件
|
||
|
||
1. [opencv](https://www.lfd.uci.edu/~gohlke/pythonlibs/#opencv)
|
||
2. [numpy](https://www.lfd.uci.edu/~gohlke/pythonlibs/#numpy)
|
||
3. [dlib](http://dlib.net/)
|
||
1. [dlib-19.4.0.win32-py2.7.exe](Data/dlib-19.4.0.win32-py2.7.exe)
|
||
2. [dlib-19.4.0.win32-py3.4.exe](Data/dlib-19.4.0.win32-py3.4.exe)
|
||
3. [dlib-19.4.0.win32-py3.5.exe](Data/dlib-19.4.0.win32-py3.5.exe)
|
||
4. [shape-predictor-68-face-landmarks.dat.bz2](http://dlib.net/files/shape_predictor_68_face_landmarks.dat.bz2)
|
||
|
||
![FacePoints](ScreenShot/FacePoints.png)
|
||
|
||
## 16、使用Threading
|
||
[运行 QtThreading.py](QtThreading.py)
|
||
|
||
在PyQt中使用Theading线程
|
||
|
||
![QtThreading](ScreenShot/QtThreading.gif)
|
||
|
||
## 17、背景连线动画
|
||
[运行 CircleLine.py](CircleLine.py)
|
||
|
||
主要参考 [背景连线动画.html](Data/背景连线动画.html)
|
||
|
||
![CircleLine](ScreenShot/CircleLine.gif)
|
||
|
||
|
||
## 18、无边框圆角对话框
|
||
[运行 FramelessDialog.py](FramelessDialog.py)
|
||
|
||
1. 通过设置 `self.setWindowFlags(self.windowFlags() | Qt.FramelessWindowHint)` 和 `self.setAttribute(Qt.WA_TranslucentBackground, True)` 达到无边框和背景透明
|
||
2. 在`QDialog`中放置一个`QWidget`作为背景和圆角
|
||
3. 在`QWidget`中放置其他内容
|
||
|
||
![FramelessDialog1](ScreenShot/FramelessDialog1.png)
|
||
![FramelessDialog](ScreenShot/FramelessDialog.png)
|
||
|
||
## 19、调整窗口显示边框
|
||
[运行 ShowFrameWhenDrag.py](ShowFrameWhenDrag.py)
|
||
|
||
1. 全局设置是【】在控制面板中->调整Windows的外观和性能->去掉勾选 拖动时显示窗口内容】
|
||
2. 但是为了不影响其它应用,可以在窗口处理函数wndproc中对其进行判断处理
|
||
3. 必须先要替换wndproc为自己的函数
|
||
4. 当消息事件==WM_NCLBUTTONDOWN的时候, 先强制开启,然后处理完成后再还原
|
||
|
||
好处在于可以减少窗口更新的次数(用途有频繁渲染的界面)
|
||
|
||
![ShowFrameWhenDrag](ScreenShot/ShowFrameWhenDrag.gif)
|
||
|
||
## 20、判断信号是否连接
|
||
[运行 IsSignalConnected.py](IsSignalConnected.py)
|
||
|
||
1. 通过 `isSignalConnected` 判断是否连接
|
||
2. 通过对象的 `receivers` 获取连接的数量来判断
|
||
|
||
![IsSignalConnected](ScreenShot/IsSignalConnected.png)
|
||
|
||
## 21、调用虚拟键盘
|
||
[运行 CallVirtualKeyboard.py](CallVirtualKeyboard.py)
|
||
|
||
1. Windows上调用的是`osk.exe`
|
||
2. Linux上调用的是`florence`,`onboard`,`kvkbd`,这三种屏幕键盘需要自行安装
|
||
|
||
![CallVirtualKeyboard1](ScreenShot/CallVirtualKeyboard1.png)
|
||
![CallVirtualKeyboard2](ScreenShot/CallVirtualKeyboard2.png)
|
||
|
||
## 22、动态忙碌光标
|
||
[运行 GifCursor.py](GifCursor.py)
|
||
|
||
通过定时器不停的修改光标图片来实现动态效果
|
||
|
||
![GifCursor](ScreenShot/GifCursor.gif)
|
||
|
||
## 23、屏幕变动监听
|
||
[运行 ScreenNotify.py](ScreenNotify.py)
|
||
|
||
通过定时器减少不同的变化信号,尽量保证只调用一次槽函数来获取信息
|
||
|
||
![ScreenNotify](ScreenShot/ScreenNotify.png)
|
||
|
||
## 24、无边框窗口
|
||
[运行 NewFramelessWindow.py](NewFramelessWindow.py)
|
||
|
||
1. 该方法只针对 `Qt5.15` 以上版本有效
|
||
2. 通过事件过滤器判断边缘设置鼠标样式
|
||
3. 处理点击事件交通过 `QWindow.startSystemMove` 和 `QWindow.startSystemResize` 传递给系统处理
|
||
|
||
![NewFramelessWindow](ScreenShot/NewFramelessWindow.gif) |