pyxllib 0.3.197__py3-none-any.whl → 0.3.200__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (126) hide show
  1. pyxllib/__init__.py +21 -21
  2. pyxllib/algo/__init__.py +8 -8
  3. pyxllib/algo/disjoint.py +54 -54
  4. pyxllib/algo/geo.py +541 -541
  5. pyxllib/algo/intervals.py +964 -964
  6. pyxllib/algo/matcher.py +389 -389
  7. pyxllib/algo/newbie.py +166 -166
  8. pyxllib/algo/pupil.py +629 -629
  9. pyxllib/algo/shapelylib.py +67 -67
  10. pyxllib/algo/specialist.py +241 -241
  11. pyxllib/algo/stat.py +494 -494
  12. pyxllib/algo/treelib.py +149 -149
  13. pyxllib/algo/unitlib.py +66 -66
  14. pyxllib/autogui/__init__.py +5 -5
  15. pyxllib/autogui/activewin.py +246 -246
  16. pyxllib/autogui/all.py +9 -9
  17. pyxllib/autogui/autogui.py +852 -852
  18. pyxllib/autogui/uiautolib.py +362 -362
  19. pyxllib/autogui/virtualkey.py +102 -102
  20. pyxllib/autogui/wechat.py +827 -827
  21. pyxllib/autogui/wechat_msg.py +421 -421
  22. pyxllib/autogui/wxautolib.py +84 -84
  23. pyxllib/cv/__init__.py +5 -5
  24. pyxllib/cv/expert.py +267 -267
  25. pyxllib/cv/imfile.py +159 -159
  26. pyxllib/cv/imhash.py +39 -39
  27. pyxllib/cv/pupil.py +9 -9
  28. pyxllib/cv/rgbfmt.py +1525 -1525
  29. pyxllib/cv/slidercaptcha.py +137 -137
  30. pyxllib/cv/trackbartools.py +251 -251
  31. pyxllib/cv/xlcvlib.py +1040 -1040
  32. pyxllib/cv/xlpillib.py +423 -423
  33. pyxllib/data/echarts.py +240 -240
  34. pyxllib/data/jsonlib.py +89 -89
  35. pyxllib/data/oss.py +72 -72
  36. pyxllib/data/pglib.py +1127 -1127
  37. pyxllib/data/sqlite.py +568 -568
  38. pyxllib/data/sqllib.py +297 -297
  39. pyxllib/ext/JLineViewer.py +505 -505
  40. pyxllib/ext/__init__.py +6 -6
  41. pyxllib/ext/demolib.py +246 -246
  42. pyxllib/ext/drissionlib.py +277 -277
  43. pyxllib/ext/kq5034lib.py +12 -12
  44. pyxllib/ext/old.py +663 -663
  45. pyxllib/ext/qt.py +449 -449
  46. pyxllib/ext/robustprocfile.py +497 -497
  47. pyxllib/ext/seleniumlib.py +76 -76
  48. pyxllib/ext/tk.py +173 -173
  49. pyxllib/ext/unixlib.py +827 -827
  50. pyxllib/ext/utools.py +351 -351
  51. pyxllib/ext/webhook.py +124 -119
  52. pyxllib/ext/win32lib.py +40 -40
  53. pyxllib/ext/wjxlib.py +88 -88
  54. pyxllib/ext/wpsapi.py +124 -124
  55. pyxllib/ext/xlwork.py +9 -9
  56. pyxllib/ext/yuquelib.py +1105 -1105
  57. pyxllib/file/__init__.py +17 -17
  58. pyxllib/file/docxlib.py +761 -761
  59. pyxllib/file/gitlib.py +309 -309
  60. pyxllib/file/libreoffice.py +165 -165
  61. pyxllib/file/movielib.py +148 -148
  62. pyxllib/file/newbie.py +10 -10
  63. pyxllib/file/onenotelib.py +1469 -1469
  64. pyxllib/file/packlib/__init__.py +330 -330
  65. pyxllib/file/packlib/zipfile.py +2441 -2441
  66. pyxllib/file/pdflib.py +426 -426
  67. pyxllib/file/pupil.py +185 -185
  68. pyxllib/file/specialist/__init__.py +685 -685
  69. pyxllib/file/specialist/dirlib.py +799 -799
  70. pyxllib/file/specialist/download.py +193 -193
  71. pyxllib/file/specialist/filelib.py +2829 -2829
  72. pyxllib/file/xlsxlib.py +3131 -3131
  73. pyxllib/file/xlsyncfile.py +341 -341
  74. pyxllib/prog/__init__.py +5 -5
  75. pyxllib/prog/cachetools.py +64 -64
  76. pyxllib/prog/deprecatedlib.py +233 -233
  77. pyxllib/prog/filelock.py +42 -42
  78. pyxllib/prog/ipyexec.py +253 -253
  79. pyxllib/prog/multiprogs.py +940 -940
  80. pyxllib/prog/newbie.py +451 -451
  81. pyxllib/prog/pupil.py +1197 -1197
  82. pyxllib/prog/sitepackages.py +33 -33
  83. pyxllib/prog/specialist/__init__.py +391 -391
  84. pyxllib/prog/specialist/bc.py +203 -203
  85. pyxllib/prog/specialist/browser.py +497 -497
  86. pyxllib/prog/specialist/common.py +347 -347
  87. pyxllib/prog/specialist/datetime.py +198 -198
  88. pyxllib/prog/specialist/tictoc.py +240 -240
  89. pyxllib/prog/specialist/xllog.py +180 -180
  90. pyxllib/prog/xlosenv.py +108 -108
  91. pyxllib/stdlib/__init__.py +17 -17
  92. pyxllib/stdlib/tablepyxl/__init__.py +10 -10
  93. pyxllib/stdlib/tablepyxl/style.py +303 -303
  94. pyxllib/stdlib/tablepyxl/tablepyxl.py +130 -130
  95. pyxllib/text/__init__.py +8 -8
  96. pyxllib/text/ahocorasick.py +39 -39
  97. pyxllib/text/airscript.js +744 -744
  98. pyxllib/text/charclasslib.py +121 -121
  99. pyxllib/text/jiebalib.py +267 -267
  100. pyxllib/text/jinjalib.py +32 -32
  101. pyxllib/text/jsa_ai_prompt.md +271 -271
  102. pyxllib/text/jscode.py +922 -922
  103. pyxllib/text/latex/__init__.py +158 -158
  104. pyxllib/text/levenshtein.py +303 -303
  105. pyxllib/text/nestenv.py +1215 -1215
  106. pyxllib/text/newbie.py +300 -300
  107. pyxllib/text/pupil/__init__.py +8 -8
  108. pyxllib/text/pupil/common.py +1121 -1121
  109. pyxllib/text/pupil/xlalign.py +326 -326
  110. pyxllib/text/pycode.py +47 -47
  111. pyxllib/text/specialist/__init__.py +8 -8
  112. pyxllib/text/specialist/common.py +112 -112
  113. pyxllib/text/specialist/ptag.py +186 -186
  114. pyxllib/text/spellchecker.py +172 -172
  115. pyxllib/text/templates/echart_base.html +10 -10
  116. pyxllib/text/templates/highlight_code.html +16 -16
  117. pyxllib/text/templates/latex_editor.html +102 -102
  118. pyxllib/text/vbacode.py +17 -17
  119. pyxllib/text/xmllib.py +747 -747
  120. pyxllib/xl.py +42 -39
  121. pyxllib/xlcv.py +17 -17
  122. {pyxllib-0.3.197.dist-info → pyxllib-0.3.200.dist-info}/METADATA +1 -1
  123. pyxllib-0.3.200.dist-info/RECORD +126 -0
  124. {pyxllib-0.3.197.dist-info → pyxllib-0.3.200.dist-info}/licenses/LICENSE +190 -190
  125. pyxllib-0.3.197.dist-info/RECORD +0 -126
  126. {pyxllib-0.3.197.dist-info → pyxllib-0.3.200.dist-info}/WHEEL +0 -0
@@ -1,246 +1,246 @@
1
- #!/usr/bin/env python3
2
- # -*- coding: utf-8 -*-
3
- # @Author : 陈坤泽
4
- # @Email : 877362867@qq.com
5
- # @Date : 2024/10/29
6
-
7
- import platform
8
- import sys
9
- import time
10
-
11
- import cv2
12
- import psutil
13
- import numpy as np
14
-
15
- if sys.platform == 'win32':
16
- import win32gui
17
- import win32process
18
- # from pywinauto import Desktop
19
- import uiautomation
20
-
21
- from pyxllib.prog.cachetools import xlcache
22
- from pyxllib.cv.xlcvlib import CvImg
23
- from pyxlpr.ai.clientlib import XlAiClient
24
-
25
- # 根据平台扩展相应的类
26
- if platform.system().lower() == "darwin":
27
- from mss.darwin import MSS as OriginalMSS
28
- elif platform.system().lower() == "linux":
29
- from mss.linux import MSS as OriginalMSS
30
- elif platform.system().lower() == "windows":
31
- from mss.windows import MSS as OriginalMSS
32
- else:
33
- raise RuntimeError("Unsupported platform")
34
-
35
-
36
- def adjust_monitors(monitors, scale_factor):
37
- # 调整所有物理显示器的尺寸和位置
38
- for i in range(1, len(monitors)):
39
- monitors[i]['width'] = int(monitors[i]['width'] / scale_factor)
40
- monitors[i]['height'] = int(monitors[i]['height'] / scale_factor)
41
- monitors[i]['top'] = int(monitors[i]['top'] / scale_factor)
42
- monitors[i]['left'] = int(monitors[i]['left'] / scale_factor)
43
-
44
- # 计算虚拟桌面的边界
45
- left_bound = min(monitor['left'] for monitor in monitors[1:])
46
- top_bound = min(monitor['top'] for monitor in monitors[1:])
47
- right_bound = max(monitor['left'] + monitor['width'] for monitor in monitors[1:])
48
- bottom_bound = max(monitor['top'] + monitor['height'] for monitor in monitors[1:])
49
-
50
- # 设置虚拟桌面的尺寸和位置
51
- monitors[0]['width'] = right_bound - left_bound
52
- monitors[0]['height'] = bottom_bound - top_bound
53
- monitors[0]['top'] = top_bound
54
- monitors[0]['left'] = left_bound
55
-
56
- return monitors
57
-
58
-
59
- class ActiveWindowCapture(OriginalMSS):
60
- """
61
- 基于`MSS`库的一个扩展,用于实现跨平台的多屏幕截图,并添加了以下独特功能:
62
- 1. 实时窗口捕获:能够获取当前激活的窗口信息,并关联到对应的进程对象。
63
- 这一特性使得该类在需要实时监控、分析特定应用窗口时尤为实用。
64
- 2. 缓存机制:该类采用了缓存策略,通过`xlcache`装饰器缓存截取的屏幕和窗口数据,避免频繁重复捕获,提升性能。
65
- 缓存机制同时确保在多窗口环境中,当前获取的窗口信息是实际的活动窗口,防止数据滞后或错位。
66
- 3. 多屏幕拼接与坐标调整:支持多屏幕拼接功能,能够调整不同显示器的坐标和尺寸,以生成全屏拼接的截图画布。
67
- 对于分辨率缩放和多屏偏移,也可以进行自适应调整。
68
- 4. 标记窗口功能:在截屏时,可以选择性地标记当前激活窗口的位置,支持自定义标记的颜色、厚度和偏移量,便于在截图中直观地显示当前活跃窗口。
69
-
70
- 适合需要频繁截图、实时监控活动窗口和对应进程的场景,尤其在多屏和分辨率适配方面提供了便捷的解决方案。
71
- """
72
-
73
- def __1_动态成员变量(self):
74
- pass
75
-
76
- def __init__(self, *, scale=1):
77
- """
78
- :param scale: 可适当兼容主屏幕做了缩放处理的情景,但太复杂的情况下也不一定对
79
- """
80
- super().__init__()
81
- self.scale = scale
82
-
83
- @xlcache(property=True)
84
- def monitors2(self):
85
- """ 修正后的各个屏幕坐标 """
86
- monitors = self.monitors
87
- if self.scale != 1:
88
- monitors = adjust_monitors(monitors, self.scale)
89
- return monitors
90
-
91
- @xlcache(property=True)
92
- def window(self):
93
- """ 当前激活窗口 """
94
- # 可以小等一会等到有效窗口
95
- for i in range(4):
96
- win = uiautomation.GetForegroundControl()
97
- if win:
98
- return win
99
- else:
100
- time.sleep(0.5)
101
- return
102
-
103
- @xlcache(property=True)
104
- def process(self):
105
- """ 当前激活窗口 """
106
- hwnd = self.window.NativeWindowHandle
107
- thread_id, process_id = win32process.GetWindowThreadProcessId(hwnd)
108
- return psutil.Process(process_id)
109
-
110
- @xlcache(property=True)
111
- def xlapi(self):
112
- return XlAiClient()
113
-
114
- def __2_截图数据(self):
115
- pass
116
-
117
- @xlcache()
118
- def _capture_single_monitor(self, order):
119
- """ 缓存单个屏幕的截图 """
120
- if order == 0:
121
- # order=0时,拼接所有屏幕图像。虽然mss其实可以直接提供,但是这样就没有每个屏幕缓存的图片了,还是我自己处理一遍更好。
122
-
123
- # 计算坐标偏移量,将所有坐标调整至非负
124
- min_left = min(monitor['left'] for monitor in self.monitors2)
125
- min_top = min(monitor['top'] for monitor in self.monitors2)
126
-
127
- # 获取总画布大小(根据所有屏幕的范围计算)
128
- max_right = max(monitor['left'] + monitor['width'] for monitor in self.monitors2)
129
- max_bottom = max(monitor['top'] + monitor['height'] for monitor in self.monitors2)
130
- canvas_width = max_right - min_left
131
- canvas_height = max_bottom - min_top
132
-
133
- # 使用numpy创建空白画布 (BGR格式)
134
- img_canvas = np.zeros((canvas_height, canvas_width, 3), dtype=np.uint8)
135
-
136
- # 获取并缓存每个子屏幕截图,按调整后的坐标拼接到总画布
137
- for i in range(1, len(self.monitors2)):
138
- monitor = self.monitors2[i]
139
- single_img = self._capture_single_monitor(i) # 获取单屏截图 (cv2格式)
140
- # 将子屏幕截图粘贴到总画布的指定位置,应用坐标偏移
141
- x_offset = monitor['left'] - min_left
142
- y_offset = monitor['top'] - min_top
143
- img_canvas[y_offset:y_offset + single_img.shape[0],
144
- x_offset:x_offset + single_img.shape[1]] = single_img
145
-
146
- img_cv2 = img_canvas
147
- else:
148
- if order == -1:
149
- # 截取当前活动窗口的截图
150
- active_window = self.window
151
- rect = active_window.BoundingRectangle
152
- monitor = {
153
- 'left': int(rect.left),
154
- 'top': int(rect.top),
155
- 'width': int(rect.right - rect.left),
156
- 'height': int(rect.bottom - rect.top),
157
- }
158
- sct_img = self.grab(monitor)
159
- else:
160
- sct_img = self.grab(self.monitors2[order])
161
-
162
- # 将截图的原始数据转换成numpy数组
163
- img_np = np.frombuffer(sct_img.bgra, dtype=np.uint8)
164
- img_np = img_np.reshape((sct_img.height, sct_img.width, 4)) # 高、宽、4通道
165
- # 转换成BGR通道顺序
166
- img_cv2 = cv2.cvtColor(img_np, cv2.COLOR_BGRA2BGR)
167
-
168
- return img_cv2
169
-
170
- def _mark_window_on_screenshot(self, screenshot, window_rect, order=0, mark_params=None):
171
- """
172
- 在截图中标记窗口,并根据 order 参数自动调整坐标偏移。
173
-
174
- :param screenshot: 要标记的截图图像
175
- :param window_rect: 当前激活窗口的原始坐标 (left, top, right, bottom)
176
- :param order: 屏幕索引 (0 表示拼接所有屏幕,全图模式)
177
- :param mark_params: 标记参数字典,包含 color, base, thickness 等
178
- :return: 标记后的截图
179
- """
180
- # 1 设定默认标记参数,若传入 mark_params 则进行更新
181
- default_params = {
182
- "color": (0, 255, 0), # 默认绿色
183
- "base": 10, # 默认缩进 10 像素
184
- "thickness": 3 # 默认厚度 3 像素
185
- }
186
- mark_params = mark_params if isinstance(mark_params, dict) else None
187
- if mark_params:
188
- default_params.update(mark_params)
189
-
190
- color = default_params["color"]
191
- base = default_params["base"]
192
- thickness = default_params["thickness"]
193
-
194
- # 2 获取对应 order 的屏幕原点坐标进行偏差修正
195
- if order == 0:
196
- origin_x = self.monitors2[0]['left']
197
- origin_y = self.monitors2[0]['top']
198
- else:
199
- origin_x = self.monitors2[order]['left']
200
- origin_y = self.monitors2[order]['top']
201
-
202
- # 计算相对于指定屏幕或全图的窗口坐标
203
- adjusted_coords = (
204
- int(window_rect.left) - origin_x,
205
- int(window_rect.top) - origin_y,
206
- int(window_rect.right) - origin_x,
207
- int(window_rect.bottom) - origin_y,
208
- )
209
-
210
- # 3 绘制矩形标记
211
- cv2.rectangle(
212
- screenshot,
213
- (adjusted_coords[0] + base, adjusted_coords[1] + base),
214
- (adjusted_coords[2] - base, adjusted_coords[3] - base),
215
- color, # 使用自定义颜色
216
- thickness # 使用自定义厚度
217
- )
218
-
219
- return screenshot
220
-
221
- @xlcache()
222
- def screenshot(self, order=0, *, to_pil=False, mark_active_window=False):
223
- """ 截屏
224
-
225
- :param int order:
226
- -1, 截取当前激活窗口
227
- 0,默认值,截取全部屏幕
228
- 1,截取第1个屏幕
229
- 2,截取第2个屏幕
230
- ...
231
- :param to_pil: 是否转换成 PIL 格式
232
- :param mark_active_window: 是否标记当前激活窗口,若为字典可设置颜色、厚度等参数
233
- :return: np矩阵或PIL图片,取决于to_cv2参数
234
- """
235
- assert order < len(self.monitors), '屏幕下标越界'
236
- img = self._capture_single_monitor(order) # 调用子函数处理所有屏幕拼接
237
-
238
- # 若需要标记当前激活窗口
239
- if mark_active_window:
240
- img = self._mark_window_on_screenshot(img, self.window.BoundingRectangle, order,
241
- mark_params=mark_active_window)
242
-
243
- if to_pil:
244
- img = CvImg.read(img).to_pil_image() # 转成pil矩阵格式
245
-
246
- return img
1
+ #!/usr/bin/env python3
2
+ # -*- coding: utf-8 -*-
3
+ # @Author : 陈坤泽
4
+ # @Email : 877362867@qq.com
5
+ # @Date : 2024/10/29
6
+
7
+ import platform
8
+ import sys
9
+ import time
10
+
11
+ import cv2
12
+ import psutil
13
+ import numpy as np
14
+
15
+ if sys.platform == 'win32':
16
+ import win32gui
17
+ import win32process
18
+ # from pywinauto import Desktop
19
+ import uiautomation
20
+
21
+ from pyxllib.prog.cachetools import xlcache
22
+ from pyxllib.cv.xlcvlib import CvImg
23
+ from pyxlpr.ai.clientlib import XlAiClient
24
+
25
+ # 根据平台扩展相应的类
26
+ if platform.system().lower() == "darwin":
27
+ from mss.darwin import MSS as OriginalMSS
28
+ elif platform.system().lower() == "linux":
29
+ from mss.linux import MSS as OriginalMSS
30
+ elif platform.system().lower() == "windows":
31
+ from mss.windows import MSS as OriginalMSS
32
+ else:
33
+ raise RuntimeError("Unsupported platform")
34
+
35
+
36
+ def adjust_monitors(monitors, scale_factor):
37
+ # 调整所有物理显示器的尺寸和位置
38
+ for i in range(1, len(monitors)):
39
+ monitors[i]['width'] = int(monitors[i]['width'] / scale_factor)
40
+ monitors[i]['height'] = int(monitors[i]['height'] / scale_factor)
41
+ monitors[i]['top'] = int(monitors[i]['top'] / scale_factor)
42
+ monitors[i]['left'] = int(monitors[i]['left'] / scale_factor)
43
+
44
+ # 计算虚拟桌面的边界
45
+ left_bound = min(monitor['left'] for monitor in monitors[1:])
46
+ top_bound = min(monitor['top'] for monitor in monitors[1:])
47
+ right_bound = max(monitor['left'] + monitor['width'] for monitor in monitors[1:])
48
+ bottom_bound = max(monitor['top'] + monitor['height'] for monitor in monitors[1:])
49
+
50
+ # 设置虚拟桌面的尺寸和位置
51
+ monitors[0]['width'] = right_bound - left_bound
52
+ monitors[0]['height'] = bottom_bound - top_bound
53
+ monitors[0]['top'] = top_bound
54
+ monitors[0]['left'] = left_bound
55
+
56
+ return monitors
57
+
58
+
59
+ class ActiveWindowCapture(OriginalMSS):
60
+ """
61
+ 基于`MSS`库的一个扩展,用于实现跨平台的多屏幕截图,并添加了以下独特功能:
62
+ 1. 实时窗口捕获:能够获取当前激活的窗口信息,并关联到对应的进程对象。
63
+ 这一特性使得该类在需要实时监控、分析特定应用窗口时尤为实用。
64
+ 2. 缓存机制:该类采用了缓存策略,通过`xlcache`装饰器缓存截取的屏幕和窗口数据,避免频繁重复捕获,提升性能。
65
+ 缓存机制同时确保在多窗口环境中,当前获取的窗口信息是实际的活动窗口,防止数据滞后或错位。
66
+ 3. 多屏幕拼接与坐标调整:支持多屏幕拼接功能,能够调整不同显示器的坐标和尺寸,以生成全屏拼接的截图画布。
67
+ 对于分辨率缩放和多屏偏移,也可以进行自适应调整。
68
+ 4. 标记窗口功能:在截屏时,可以选择性地标记当前激活窗口的位置,支持自定义标记的颜色、厚度和偏移量,便于在截图中直观地显示当前活跃窗口。
69
+
70
+ 适合需要频繁截图、实时监控活动窗口和对应进程的场景,尤其在多屏和分辨率适配方面提供了便捷的解决方案。
71
+ """
72
+
73
+ def __1_动态成员变量(self):
74
+ pass
75
+
76
+ def __init__(self, *, scale=1):
77
+ """
78
+ :param scale: 可适当兼容主屏幕做了缩放处理的情景,但太复杂的情况下也不一定对
79
+ """
80
+ super().__init__()
81
+ self.scale = scale
82
+
83
+ @xlcache(property=True)
84
+ def monitors2(self):
85
+ """ 修正后的各个屏幕坐标 """
86
+ monitors = self.monitors
87
+ if self.scale != 1:
88
+ monitors = adjust_monitors(monitors, self.scale)
89
+ return monitors
90
+
91
+ @xlcache(property=True)
92
+ def window(self):
93
+ """ 当前激活窗口 """
94
+ # 可以小等一会等到有效窗口
95
+ for i in range(4):
96
+ win = uiautomation.GetForegroundControl()
97
+ if win:
98
+ return win
99
+ else:
100
+ time.sleep(0.5)
101
+ return
102
+
103
+ @xlcache(property=True)
104
+ def process(self):
105
+ """ 当前激活窗口 """
106
+ hwnd = self.window.NativeWindowHandle
107
+ thread_id, process_id = win32process.GetWindowThreadProcessId(hwnd)
108
+ return psutil.Process(process_id)
109
+
110
+ @xlcache(property=True)
111
+ def xlapi(self):
112
+ return XlAiClient()
113
+
114
+ def __2_截图数据(self):
115
+ pass
116
+
117
+ @xlcache()
118
+ def _capture_single_monitor(self, order):
119
+ """ 缓存单个屏幕的截图 """
120
+ if order == 0:
121
+ # order=0时,拼接所有屏幕图像。虽然mss其实可以直接提供,但是这样就没有每个屏幕缓存的图片了,还是我自己处理一遍更好。
122
+
123
+ # 计算坐标偏移量,将所有坐标调整至非负
124
+ min_left = min(monitor['left'] for monitor in self.monitors2)
125
+ min_top = min(monitor['top'] for monitor in self.monitors2)
126
+
127
+ # 获取总画布大小(根据所有屏幕的范围计算)
128
+ max_right = max(monitor['left'] + monitor['width'] for monitor in self.monitors2)
129
+ max_bottom = max(monitor['top'] + monitor['height'] for monitor in self.monitors2)
130
+ canvas_width = max_right - min_left
131
+ canvas_height = max_bottom - min_top
132
+
133
+ # 使用numpy创建空白画布 (BGR格式)
134
+ img_canvas = np.zeros((canvas_height, canvas_width, 3), dtype=np.uint8)
135
+
136
+ # 获取并缓存每个子屏幕截图,按调整后的坐标拼接到总画布
137
+ for i in range(1, len(self.monitors2)):
138
+ monitor = self.monitors2[i]
139
+ single_img = self._capture_single_monitor(i) # 获取单屏截图 (cv2格式)
140
+ # 将子屏幕截图粘贴到总画布的指定位置,应用坐标偏移
141
+ x_offset = monitor['left'] - min_left
142
+ y_offset = monitor['top'] - min_top
143
+ img_canvas[y_offset:y_offset + single_img.shape[0],
144
+ x_offset:x_offset + single_img.shape[1]] = single_img
145
+
146
+ img_cv2 = img_canvas
147
+ else:
148
+ if order == -1:
149
+ # 截取当前活动窗口的截图
150
+ active_window = self.window
151
+ rect = active_window.BoundingRectangle
152
+ monitor = {
153
+ 'left': int(rect.left),
154
+ 'top': int(rect.top),
155
+ 'width': int(rect.right - rect.left),
156
+ 'height': int(rect.bottom - rect.top),
157
+ }
158
+ sct_img = self.grab(monitor)
159
+ else:
160
+ sct_img = self.grab(self.monitors2[order])
161
+
162
+ # 将截图的原始数据转换成numpy数组
163
+ img_np = np.frombuffer(sct_img.bgra, dtype=np.uint8)
164
+ img_np = img_np.reshape((sct_img.height, sct_img.width, 4)) # 高、宽、4通道
165
+ # 转换成BGR通道顺序
166
+ img_cv2 = cv2.cvtColor(img_np, cv2.COLOR_BGRA2BGR)
167
+
168
+ return img_cv2
169
+
170
+ def _mark_window_on_screenshot(self, screenshot, window_rect, order=0, mark_params=None):
171
+ """
172
+ 在截图中标记窗口,并根据 order 参数自动调整坐标偏移。
173
+
174
+ :param screenshot: 要标记的截图图像
175
+ :param window_rect: 当前激活窗口的原始坐标 (left, top, right, bottom)
176
+ :param order: 屏幕索引 (0 表示拼接所有屏幕,全图模式)
177
+ :param mark_params: 标记参数字典,包含 color, base, thickness 等
178
+ :return: 标记后的截图
179
+ """
180
+ # 1 设定默认标记参数,若传入 mark_params 则进行更新
181
+ default_params = {
182
+ "color": (0, 255, 0), # 默认绿色
183
+ "base": 10, # 默认缩进 10 像素
184
+ "thickness": 3 # 默认厚度 3 像素
185
+ }
186
+ mark_params = mark_params if isinstance(mark_params, dict) else None
187
+ if mark_params:
188
+ default_params.update(mark_params)
189
+
190
+ color = default_params["color"]
191
+ base = default_params["base"]
192
+ thickness = default_params["thickness"]
193
+
194
+ # 2 获取对应 order 的屏幕原点坐标进行偏差修正
195
+ if order == 0:
196
+ origin_x = self.monitors2[0]['left']
197
+ origin_y = self.monitors2[0]['top']
198
+ else:
199
+ origin_x = self.monitors2[order]['left']
200
+ origin_y = self.monitors2[order]['top']
201
+
202
+ # 计算相对于指定屏幕或全图的窗口坐标
203
+ adjusted_coords = (
204
+ int(window_rect.left) - origin_x,
205
+ int(window_rect.top) - origin_y,
206
+ int(window_rect.right) - origin_x,
207
+ int(window_rect.bottom) - origin_y,
208
+ )
209
+
210
+ # 3 绘制矩形标记
211
+ cv2.rectangle(
212
+ screenshot,
213
+ (adjusted_coords[0] + base, adjusted_coords[1] + base),
214
+ (adjusted_coords[2] - base, adjusted_coords[3] - base),
215
+ color, # 使用自定义颜色
216
+ thickness # 使用自定义厚度
217
+ )
218
+
219
+ return screenshot
220
+
221
+ @xlcache()
222
+ def screenshot(self, order=0, *, to_pil=False, mark_active_window=False):
223
+ """ 截屏
224
+
225
+ :param int order:
226
+ -1, 截取当前激活窗口
227
+ 0,默认值,截取全部屏幕
228
+ 1,截取第1个屏幕
229
+ 2,截取第2个屏幕
230
+ ...
231
+ :param to_pil: 是否转换成 PIL 格式
232
+ :param mark_active_window: 是否标记当前激活窗口,若为字典可设置颜色、厚度等参数
233
+ :return: np矩阵或PIL图片,取决于to_cv2参数
234
+ """
235
+ assert order < len(self.monitors), '屏幕下标越界'
236
+ img = self._capture_single_monitor(order) # 调用子函数处理所有屏幕拼接
237
+
238
+ # 若需要标记当前激活窗口
239
+ if mark_active_window:
240
+ img = self._mark_window_on_screenshot(img, self.window.BoundingRectangle, order,
241
+ mark_params=mark_active_window)
242
+
243
+ if to_pil:
244
+ img = CvImg.read(img).to_pil_image() # 转成pil矩阵格式
245
+
246
+ return img
pyxllib/autogui/all.py CHANGED
@@ -1,9 +1,9 @@
1
- #!/usr/bin/env python3
2
- # -*- coding: utf-8 -*-
3
- # @Author : 陈坤泽
4
- # @Email : 877362867@qq.com
5
- # @Date : 2024/10/29
6
-
7
- from pyxllib.autogui.autogui import *
8
- from pyxllib.autogui.virtualkey import *
9
- from pyxllib.autogui.activewin import *
1
+ #!/usr/bin/env python3
2
+ # -*- coding: utf-8 -*-
3
+ # @Author : 陈坤泽
4
+ # @Email : 877362867@qq.com
5
+ # @Date : 2024/10/29
6
+
7
+ from pyxllib.autogui.autogui import *
8
+ from pyxllib.autogui.virtualkey import *
9
+ from pyxllib.autogui.activewin import *