pyxllib 0.3.96__py3-none-any.whl → 0.3.197__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 (306) hide show
  1. pyxllib/algo/geo.py +12 -0
  2. pyxllib/algo/intervals.py +1 -1
  3. pyxllib/algo/matcher.py +78 -0
  4. pyxllib/algo/pupil.py +187 -19
  5. pyxllib/algo/specialist.py +2 -1
  6. pyxllib/algo/stat.py +38 -2
  7. {pyxlpr → pyxllib/autogui}/__init__.py +1 -1
  8. pyxllib/autogui/activewin.py +246 -0
  9. pyxllib/autogui/all.py +9 -0
  10. pyxllib/{ext/autogui → autogui}/autogui.py +40 -11
  11. pyxllib/autogui/uiautolib.py +362 -0
  12. pyxllib/autogui/wechat.py +827 -0
  13. pyxllib/autogui/wechat_msg.py +421 -0
  14. pyxllib/autogui/wxautolib.py +84 -0
  15. pyxllib/cv/slidercaptcha.py +137 -0
  16. pyxllib/data/echarts.py +123 -12
  17. pyxllib/data/jsonlib.py +89 -0
  18. pyxllib/data/pglib.py +514 -30
  19. pyxllib/data/sqlite.py +231 -4
  20. pyxllib/ext/JLineViewer.py +14 -1
  21. pyxllib/ext/drissionlib.py +277 -0
  22. pyxllib/ext/kq5034lib.py +0 -1594
  23. pyxllib/ext/robustprocfile.py +497 -0
  24. pyxllib/ext/unixlib.py +6 -5
  25. pyxllib/ext/utools.py +108 -95
  26. pyxllib/ext/webhook.py +32 -14
  27. pyxllib/ext/wjxlib.py +88 -0
  28. pyxllib/ext/wpsapi.py +124 -0
  29. pyxllib/ext/xlwork.py +9 -0
  30. pyxllib/ext/yuquelib.py +1003 -71
  31. pyxllib/file/docxlib.py +1 -1
  32. pyxllib/file/libreoffice.py +165 -0
  33. pyxllib/file/movielib.py +9 -0
  34. pyxllib/file/packlib/__init__.py +112 -75
  35. pyxllib/file/pdflib.py +1 -1
  36. pyxllib/file/pupil.py +1 -1
  37. pyxllib/file/specialist/dirlib.py +1 -1
  38. pyxllib/file/specialist/download.py +10 -3
  39. pyxllib/file/specialist/filelib.py +266 -55
  40. pyxllib/file/xlsxlib.py +205 -50
  41. pyxllib/file/xlsyncfile.py +341 -0
  42. pyxllib/prog/cachetools.py +64 -0
  43. pyxllib/prog/filelock.py +42 -0
  44. pyxllib/prog/multiprogs.py +940 -0
  45. pyxllib/prog/newbie.py +9 -2
  46. pyxllib/prog/pupil.py +129 -60
  47. pyxllib/prog/specialist/__init__.py +176 -2
  48. pyxllib/prog/specialist/bc.py +5 -2
  49. pyxllib/prog/specialist/browser.py +11 -2
  50. pyxllib/prog/specialist/datetime.py +68 -0
  51. pyxllib/prog/specialist/tictoc.py +12 -13
  52. pyxllib/prog/specialist/xllog.py +5 -5
  53. pyxllib/prog/xlosenv.py +7 -0
  54. pyxllib/text/airscript.js +744 -0
  55. pyxllib/text/charclasslib.py +17 -5
  56. pyxllib/text/jiebalib.py +6 -3
  57. pyxllib/text/jinjalib.py +32 -0
  58. pyxllib/text/jsa_ai_prompt.md +271 -0
  59. pyxllib/text/jscode.py +159 -4
  60. pyxllib/text/nestenv.py +1 -1
  61. pyxllib/text/newbie.py +12 -0
  62. pyxllib/text/pupil/common.py +26 -0
  63. pyxllib/text/specialist/ptag.py +2 -2
  64. pyxllib/text/templates/echart_base.html +11 -0
  65. pyxllib/text/templates/highlight_code.html +17 -0
  66. pyxllib/text/templates/latex_editor.html +103 -0
  67. pyxllib/text/xmllib.py +76 -14
  68. pyxllib/xl.py +2 -1
  69. pyxllib-0.3.197.dist-info/METADATA +48 -0
  70. pyxllib-0.3.197.dist-info/RECORD +126 -0
  71. {pyxllib-0.3.96.dist-info → pyxllib-0.3.197.dist-info}/WHEEL +1 -2
  72. pyxllib/ext/autogui/__init__.py +0 -8
  73. pyxllib-0.3.96.dist-info/METADATA +0 -51
  74. pyxllib-0.3.96.dist-info/RECORD +0 -333
  75. pyxllib-0.3.96.dist-info/top_level.txt +0 -2
  76. pyxlpr/ai/__init__.py +0 -5
  77. pyxlpr/ai/clientlib.py +0 -1281
  78. pyxlpr/ai/specialist.py +0 -286
  79. pyxlpr/ai/torch_app.py +0 -172
  80. pyxlpr/ai/xlpaddle.py +0 -655
  81. pyxlpr/ai/xltorch.py +0 -705
  82. pyxlpr/data/__init__.py +0 -11
  83. pyxlpr/data/coco.py +0 -1325
  84. pyxlpr/data/datacls.py +0 -365
  85. pyxlpr/data/datasets.py +0 -200
  86. pyxlpr/data/gptlib.py +0 -1291
  87. pyxlpr/data/icdar/__init__.py +0 -96
  88. pyxlpr/data/icdar/deteval.py +0 -377
  89. pyxlpr/data/icdar/icdar2013.py +0 -341
  90. pyxlpr/data/icdar/iou.py +0 -340
  91. pyxlpr/data/icdar/rrc_evaluation_funcs_1_1.py +0 -463
  92. pyxlpr/data/imtextline.py +0 -473
  93. pyxlpr/data/labelme.py +0 -866
  94. pyxlpr/data/removeline.py +0 -179
  95. pyxlpr/data/specialist.py +0 -57
  96. pyxlpr/eval/__init__.py +0 -85
  97. pyxlpr/paddleocr.py +0 -776
  98. pyxlpr/ppocr/__init__.py +0 -15
  99. pyxlpr/ppocr/configs/rec/multi_language/generate_multi_language_configs.py +0 -226
  100. pyxlpr/ppocr/data/__init__.py +0 -135
  101. pyxlpr/ppocr/data/imaug/ColorJitter.py +0 -26
  102. pyxlpr/ppocr/data/imaug/__init__.py +0 -67
  103. pyxlpr/ppocr/data/imaug/copy_paste.py +0 -170
  104. pyxlpr/ppocr/data/imaug/east_process.py +0 -437
  105. pyxlpr/ppocr/data/imaug/gen_table_mask.py +0 -244
  106. pyxlpr/ppocr/data/imaug/iaa_augment.py +0 -114
  107. pyxlpr/ppocr/data/imaug/label_ops.py +0 -789
  108. pyxlpr/ppocr/data/imaug/make_border_map.py +0 -184
  109. pyxlpr/ppocr/data/imaug/make_pse_gt.py +0 -106
  110. pyxlpr/ppocr/data/imaug/make_shrink_map.py +0 -126
  111. pyxlpr/ppocr/data/imaug/operators.py +0 -433
  112. pyxlpr/ppocr/data/imaug/pg_process.py +0 -906
  113. pyxlpr/ppocr/data/imaug/randaugment.py +0 -143
  114. pyxlpr/ppocr/data/imaug/random_crop_data.py +0 -239
  115. pyxlpr/ppocr/data/imaug/rec_img_aug.py +0 -533
  116. pyxlpr/ppocr/data/imaug/sast_process.py +0 -777
  117. pyxlpr/ppocr/data/imaug/text_image_aug/__init__.py +0 -17
  118. pyxlpr/ppocr/data/imaug/text_image_aug/augment.py +0 -120
  119. pyxlpr/ppocr/data/imaug/text_image_aug/warp_mls.py +0 -168
  120. pyxlpr/ppocr/data/lmdb_dataset.py +0 -115
  121. pyxlpr/ppocr/data/pgnet_dataset.py +0 -104
  122. pyxlpr/ppocr/data/pubtab_dataset.py +0 -107
  123. pyxlpr/ppocr/data/simple_dataset.py +0 -372
  124. pyxlpr/ppocr/losses/__init__.py +0 -61
  125. pyxlpr/ppocr/losses/ace_loss.py +0 -52
  126. pyxlpr/ppocr/losses/basic_loss.py +0 -135
  127. pyxlpr/ppocr/losses/center_loss.py +0 -88
  128. pyxlpr/ppocr/losses/cls_loss.py +0 -30
  129. pyxlpr/ppocr/losses/combined_loss.py +0 -67
  130. pyxlpr/ppocr/losses/det_basic_loss.py +0 -208
  131. pyxlpr/ppocr/losses/det_db_loss.py +0 -80
  132. pyxlpr/ppocr/losses/det_east_loss.py +0 -63
  133. pyxlpr/ppocr/losses/det_pse_loss.py +0 -149
  134. pyxlpr/ppocr/losses/det_sast_loss.py +0 -121
  135. pyxlpr/ppocr/losses/distillation_loss.py +0 -272
  136. pyxlpr/ppocr/losses/e2e_pg_loss.py +0 -140
  137. pyxlpr/ppocr/losses/kie_sdmgr_loss.py +0 -113
  138. pyxlpr/ppocr/losses/rec_aster_loss.py +0 -99
  139. pyxlpr/ppocr/losses/rec_att_loss.py +0 -39
  140. pyxlpr/ppocr/losses/rec_ctc_loss.py +0 -44
  141. pyxlpr/ppocr/losses/rec_enhanced_ctc_loss.py +0 -70
  142. pyxlpr/ppocr/losses/rec_nrtr_loss.py +0 -30
  143. pyxlpr/ppocr/losses/rec_sar_loss.py +0 -28
  144. pyxlpr/ppocr/losses/rec_srn_loss.py +0 -47
  145. pyxlpr/ppocr/losses/table_att_loss.py +0 -109
  146. pyxlpr/ppocr/metrics/__init__.py +0 -44
  147. pyxlpr/ppocr/metrics/cls_metric.py +0 -45
  148. pyxlpr/ppocr/metrics/det_metric.py +0 -82
  149. pyxlpr/ppocr/metrics/distillation_metric.py +0 -73
  150. pyxlpr/ppocr/metrics/e2e_metric.py +0 -86
  151. pyxlpr/ppocr/metrics/eval_det_iou.py +0 -274
  152. pyxlpr/ppocr/metrics/kie_metric.py +0 -70
  153. pyxlpr/ppocr/metrics/rec_metric.py +0 -75
  154. pyxlpr/ppocr/metrics/table_metric.py +0 -50
  155. pyxlpr/ppocr/modeling/architectures/__init__.py +0 -32
  156. pyxlpr/ppocr/modeling/architectures/base_model.py +0 -88
  157. pyxlpr/ppocr/modeling/architectures/distillation_model.py +0 -60
  158. pyxlpr/ppocr/modeling/backbones/__init__.py +0 -54
  159. pyxlpr/ppocr/modeling/backbones/det_mobilenet_v3.py +0 -268
  160. pyxlpr/ppocr/modeling/backbones/det_resnet_vd.py +0 -246
  161. pyxlpr/ppocr/modeling/backbones/det_resnet_vd_sast.py +0 -285
  162. pyxlpr/ppocr/modeling/backbones/e2e_resnet_vd_pg.py +0 -265
  163. pyxlpr/ppocr/modeling/backbones/kie_unet_sdmgr.py +0 -186
  164. pyxlpr/ppocr/modeling/backbones/rec_mobilenet_v3.py +0 -138
  165. pyxlpr/ppocr/modeling/backbones/rec_mv1_enhance.py +0 -258
  166. pyxlpr/ppocr/modeling/backbones/rec_nrtr_mtb.py +0 -48
  167. pyxlpr/ppocr/modeling/backbones/rec_resnet_31.py +0 -210
  168. pyxlpr/ppocr/modeling/backbones/rec_resnet_aster.py +0 -143
  169. pyxlpr/ppocr/modeling/backbones/rec_resnet_fpn.py +0 -307
  170. pyxlpr/ppocr/modeling/backbones/rec_resnet_vd.py +0 -286
  171. pyxlpr/ppocr/modeling/heads/__init__.py +0 -54
  172. pyxlpr/ppocr/modeling/heads/cls_head.py +0 -52
  173. pyxlpr/ppocr/modeling/heads/det_db_head.py +0 -118
  174. pyxlpr/ppocr/modeling/heads/det_east_head.py +0 -121
  175. pyxlpr/ppocr/modeling/heads/det_pse_head.py +0 -37
  176. pyxlpr/ppocr/modeling/heads/det_sast_head.py +0 -128
  177. pyxlpr/ppocr/modeling/heads/e2e_pg_head.py +0 -253
  178. pyxlpr/ppocr/modeling/heads/kie_sdmgr_head.py +0 -206
  179. pyxlpr/ppocr/modeling/heads/multiheadAttention.py +0 -163
  180. pyxlpr/ppocr/modeling/heads/rec_aster_head.py +0 -393
  181. pyxlpr/ppocr/modeling/heads/rec_att_head.py +0 -202
  182. pyxlpr/ppocr/modeling/heads/rec_ctc_head.py +0 -88
  183. pyxlpr/ppocr/modeling/heads/rec_nrtr_head.py +0 -826
  184. pyxlpr/ppocr/modeling/heads/rec_sar_head.py +0 -402
  185. pyxlpr/ppocr/modeling/heads/rec_srn_head.py +0 -280
  186. pyxlpr/ppocr/modeling/heads/self_attention.py +0 -406
  187. pyxlpr/ppocr/modeling/heads/table_att_head.py +0 -246
  188. pyxlpr/ppocr/modeling/necks/__init__.py +0 -32
  189. pyxlpr/ppocr/modeling/necks/db_fpn.py +0 -111
  190. pyxlpr/ppocr/modeling/necks/east_fpn.py +0 -188
  191. pyxlpr/ppocr/modeling/necks/fpn.py +0 -138
  192. pyxlpr/ppocr/modeling/necks/pg_fpn.py +0 -314
  193. pyxlpr/ppocr/modeling/necks/rnn.py +0 -92
  194. pyxlpr/ppocr/modeling/necks/sast_fpn.py +0 -284
  195. pyxlpr/ppocr/modeling/necks/table_fpn.py +0 -110
  196. pyxlpr/ppocr/modeling/transforms/__init__.py +0 -28
  197. pyxlpr/ppocr/modeling/transforms/stn.py +0 -135
  198. pyxlpr/ppocr/modeling/transforms/tps.py +0 -308
  199. pyxlpr/ppocr/modeling/transforms/tps_spatial_transformer.py +0 -156
  200. pyxlpr/ppocr/optimizer/__init__.py +0 -61
  201. pyxlpr/ppocr/optimizer/learning_rate.py +0 -228
  202. pyxlpr/ppocr/optimizer/lr_scheduler.py +0 -49
  203. pyxlpr/ppocr/optimizer/optimizer.py +0 -160
  204. pyxlpr/ppocr/optimizer/regularizer.py +0 -52
  205. pyxlpr/ppocr/postprocess/__init__.py +0 -55
  206. pyxlpr/ppocr/postprocess/cls_postprocess.py +0 -33
  207. pyxlpr/ppocr/postprocess/db_postprocess.py +0 -234
  208. pyxlpr/ppocr/postprocess/east_postprocess.py +0 -143
  209. pyxlpr/ppocr/postprocess/locality_aware_nms.py +0 -200
  210. pyxlpr/ppocr/postprocess/pg_postprocess.py +0 -52
  211. pyxlpr/ppocr/postprocess/pse_postprocess/__init__.py +0 -15
  212. pyxlpr/ppocr/postprocess/pse_postprocess/pse/__init__.py +0 -29
  213. pyxlpr/ppocr/postprocess/pse_postprocess/pse/setup.py +0 -14
  214. pyxlpr/ppocr/postprocess/pse_postprocess/pse_postprocess.py +0 -118
  215. pyxlpr/ppocr/postprocess/rec_postprocess.py +0 -654
  216. pyxlpr/ppocr/postprocess/sast_postprocess.py +0 -355
  217. pyxlpr/ppocr/tools/__init__.py +0 -14
  218. pyxlpr/ppocr/tools/eval.py +0 -83
  219. pyxlpr/ppocr/tools/export_center.py +0 -77
  220. pyxlpr/ppocr/tools/export_model.py +0 -129
  221. pyxlpr/ppocr/tools/infer/predict_cls.py +0 -151
  222. pyxlpr/ppocr/tools/infer/predict_det.py +0 -300
  223. pyxlpr/ppocr/tools/infer/predict_e2e.py +0 -169
  224. pyxlpr/ppocr/tools/infer/predict_rec.py +0 -414
  225. pyxlpr/ppocr/tools/infer/predict_system.py +0 -204
  226. pyxlpr/ppocr/tools/infer/utility.py +0 -629
  227. pyxlpr/ppocr/tools/infer_cls.py +0 -83
  228. pyxlpr/ppocr/tools/infer_det.py +0 -134
  229. pyxlpr/ppocr/tools/infer_e2e.py +0 -122
  230. pyxlpr/ppocr/tools/infer_kie.py +0 -153
  231. pyxlpr/ppocr/tools/infer_rec.py +0 -146
  232. pyxlpr/ppocr/tools/infer_table.py +0 -107
  233. pyxlpr/ppocr/tools/program.py +0 -596
  234. pyxlpr/ppocr/tools/test_hubserving.py +0 -117
  235. pyxlpr/ppocr/tools/train.py +0 -163
  236. pyxlpr/ppocr/tools/xlprog.py +0 -748
  237. pyxlpr/ppocr/utils/EN_symbol_dict.txt +0 -94
  238. pyxlpr/ppocr/utils/__init__.py +0 -24
  239. pyxlpr/ppocr/utils/dict/ar_dict.txt +0 -117
  240. pyxlpr/ppocr/utils/dict/arabic_dict.txt +0 -162
  241. pyxlpr/ppocr/utils/dict/be_dict.txt +0 -145
  242. pyxlpr/ppocr/utils/dict/bg_dict.txt +0 -140
  243. pyxlpr/ppocr/utils/dict/chinese_cht_dict.txt +0 -8421
  244. pyxlpr/ppocr/utils/dict/cyrillic_dict.txt +0 -163
  245. pyxlpr/ppocr/utils/dict/devanagari_dict.txt +0 -167
  246. pyxlpr/ppocr/utils/dict/en_dict.txt +0 -63
  247. pyxlpr/ppocr/utils/dict/fa_dict.txt +0 -136
  248. pyxlpr/ppocr/utils/dict/french_dict.txt +0 -136
  249. pyxlpr/ppocr/utils/dict/german_dict.txt +0 -143
  250. pyxlpr/ppocr/utils/dict/hi_dict.txt +0 -162
  251. pyxlpr/ppocr/utils/dict/it_dict.txt +0 -118
  252. pyxlpr/ppocr/utils/dict/japan_dict.txt +0 -4399
  253. pyxlpr/ppocr/utils/dict/ka_dict.txt +0 -153
  254. pyxlpr/ppocr/utils/dict/korean_dict.txt +0 -3688
  255. pyxlpr/ppocr/utils/dict/latin_dict.txt +0 -185
  256. pyxlpr/ppocr/utils/dict/mr_dict.txt +0 -153
  257. pyxlpr/ppocr/utils/dict/ne_dict.txt +0 -153
  258. pyxlpr/ppocr/utils/dict/oc_dict.txt +0 -96
  259. pyxlpr/ppocr/utils/dict/pu_dict.txt +0 -130
  260. pyxlpr/ppocr/utils/dict/rs_dict.txt +0 -91
  261. pyxlpr/ppocr/utils/dict/rsc_dict.txt +0 -134
  262. pyxlpr/ppocr/utils/dict/ru_dict.txt +0 -125
  263. pyxlpr/ppocr/utils/dict/ta_dict.txt +0 -128
  264. pyxlpr/ppocr/utils/dict/table_dict.txt +0 -277
  265. pyxlpr/ppocr/utils/dict/table_structure_dict.txt +0 -2759
  266. pyxlpr/ppocr/utils/dict/te_dict.txt +0 -151
  267. pyxlpr/ppocr/utils/dict/ug_dict.txt +0 -114
  268. pyxlpr/ppocr/utils/dict/uk_dict.txt +0 -142
  269. pyxlpr/ppocr/utils/dict/ur_dict.txt +0 -137
  270. pyxlpr/ppocr/utils/dict/xi_dict.txt +0 -110
  271. pyxlpr/ppocr/utils/dict90.txt +0 -90
  272. pyxlpr/ppocr/utils/e2e_metric/Deteval.py +0 -574
  273. pyxlpr/ppocr/utils/e2e_metric/polygon_fast.py +0 -83
  274. pyxlpr/ppocr/utils/e2e_utils/extract_batchsize.py +0 -87
  275. pyxlpr/ppocr/utils/e2e_utils/extract_textpoint_fast.py +0 -457
  276. pyxlpr/ppocr/utils/e2e_utils/extract_textpoint_slow.py +0 -592
  277. pyxlpr/ppocr/utils/e2e_utils/pgnet_pp_utils.py +0 -162
  278. pyxlpr/ppocr/utils/e2e_utils/visual.py +0 -162
  279. pyxlpr/ppocr/utils/en_dict.txt +0 -95
  280. pyxlpr/ppocr/utils/gen_label.py +0 -81
  281. pyxlpr/ppocr/utils/ic15_dict.txt +0 -36
  282. pyxlpr/ppocr/utils/iou.py +0 -54
  283. pyxlpr/ppocr/utils/logging.py +0 -69
  284. pyxlpr/ppocr/utils/network.py +0 -84
  285. pyxlpr/ppocr/utils/ppocr_keys_v1.txt +0 -6623
  286. pyxlpr/ppocr/utils/profiler.py +0 -110
  287. pyxlpr/ppocr/utils/save_load.py +0 -150
  288. pyxlpr/ppocr/utils/stats.py +0 -72
  289. pyxlpr/ppocr/utils/utility.py +0 -80
  290. pyxlpr/ppstructure/__init__.py +0 -13
  291. pyxlpr/ppstructure/predict_system.py +0 -187
  292. pyxlpr/ppstructure/table/__init__.py +0 -13
  293. pyxlpr/ppstructure/table/eval_table.py +0 -72
  294. pyxlpr/ppstructure/table/matcher.py +0 -192
  295. pyxlpr/ppstructure/table/predict_structure.py +0 -136
  296. pyxlpr/ppstructure/table/predict_table.py +0 -221
  297. pyxlpr/ppstructure/table/table_metric/__init__.py +0 -16
  298. pyxlpr/ppstructure/table/table_metric/parallel.py +0 -51
  299. pyxlpr/ppstructure/table/table_metric/table_metric.py +0 -247
  300. pyxlpr/ppstructure/table/tablepyxl/__init__.py +0 -13
  301. pyxlpr/ppstructure/table/tablepyxl/style.py +0 -283
  302. pyxlpr/ppstructure/table/tablepyxl/tablepyxl.py +0 -118
  303. pyxlpr/ppstructure/utility.py +0 -71
  304. pyxlpr/xlai.py +0 -10
  305. /pyxllib/{ext/autogui → autogui}/virtualkey.py +0 -0
  306. {pyxllib-0.3.96.dist-info → pyxllib-0.3.197.dist-info/licenses}/LICENSE +0 -0
@@ -0,0 +1,362 @@
1
+ #!/usr/bin/env python3
2
+ # -*- coding: utf-8 -*-
3
+ # @Author : 陈坤泽
4
+ # @Email : 877362867@qq.com
5
+ # @Date : 2024/11/05
6
+
7
+ """
8
+ 以uiautomation为核心的相关工具库
9
+ """
10
+
11
+ import os
12
+ import sys
13
+ import textwrap
14
+ import time
15
+ from typing import Iterable, Callable, List
16
+ import subprocess
17
+ import tempfile
18
+
19
+ import psutil
20
+ import pandas as pd
21
+ from fastcore.basics import GetAttr
22
+
23
+ from loguru import logger
24
+ # ui组件大多是树形组织结构,auto库自带树形操作太弱。没有专业的树形结构库,能搞个毛线。
25
+ from anytree import NodeMixin
26
+
27
+ import ctypes
28
+ from ctypes import wintypes
29
+
30
+ if sys.platform == 'win32':
31
+ import win32con
32
+ import win32gui
33
+ import win32process
34
+ import win32clipboard
35
+
36
+ import uiautomation as uia
37
+ from uiautomation import WindowControl
38
+
39
+
40
+ def __1_clipboard_utils():
41
+ pass
42
+
43
+
44
+ def retry_on_failure(max_retries: int = 5):
45
+ """
46
+ 一个装饰器,用于在失败时重试执行被装饰的函数。
47
+
48
+ Args:
49
+ max_retries (int): 最大重试次数。
50
+
51
+ Returns:
52
+ Callable: 包装后的函数。
53
+ """
54
+
55
+ def decorator(func: Callable):
56
+ def wrapper(*args, **kwargs):
57
+ for attempt in range(max_retries):
58
+ try:
59
+ if func(*args, **kwargs):
60
+ return True
61
+ except Exception as e:
62
+ time.sleep(.05)
63
+ print(f"Attempt {attempt + 1} failed: {e}")
64
+ return False
65
+
66
+ return wrapper
67
+
68
+ return decorator
69
+
70
+
71
+ def set_clipboard_data(fmt: int, buf: ctypes.Array) -> bool:
72
+ """
73
+ 将数据设置到Windows剪切板中。
74
+
75
+ Args:
76
+ fmt (int): 数据格式,例如 win32clipboard.CF_HDROP。
77
+ buf (ctypes.Array): 要设置到剪切板的数据。
78
+
79
+ Returns:
80
+ bool: 操作成功返回 True,否则返回 False。
81
+ """
82
+ try:
83
+ win32clipboard.OpenClipboard()
84
+ win32clipboard.EmptyClipboard()
85
+ win32clipboard.SetClipboardData(fmt, buf)
86
+ return True
87
+ except Exception as e:
88
+ print(f"Error setting clipboard data: {e}")
89
+ return False
90
+ finally:
91
+ win32clipboard.CloseClipboard()
92
+
93
+
94
+ def get_clipboard_files() -> List[str]:
95
+ """
96
+ 获取剪切板中的文件路径列表。
97
+
98
+ Returns:
99
+ List[str]: 包含剪切板中文件路径的列表,如果没有文件路径或操作失败,返回空列表。
100
+ """
101
+ try:
102
+ win32clipboard.OpenClipboard()
103
+ if win32clipboard.IsClipboardFormatAvailable(win32clipboard.CF_HDROP):
104
+ return list(win32clipboard.GetClipboardData(win32clipboard.CF_HDROP))
105
+ else:
106
+ return list()
107
+ finally:
108
+ win32clipboard.CloseClipboard()
109
+
110
+
111
+ @retry_on_failure(max_retries=5)
112
+ def validate_clipboard_files(file_paths: Iterable[str], fmt: int, buf: ctypes.Array) -> bool:
113
+ """
114
+ 验证剪切板中的文件路径是否与给定的文件路径一致。
115
+
116
+ Args:
117
+ file_paths (Iterable): 一个包含文件路径的可迭代对象,每个路径都是一个字符串。
118
+ fmt (int): 数据格式,例如 win32clipboard.CF_HDROP。
119
+ buf (ctypes.Array): 要验证的剪切板数据。
120
+
121
+ Returns:
122
+ bool: 如果剪切板中的文件路径与给定的文件路径一致,则返回 True
123
+
124
+ Raises:
125
+ ValueError: 如果剪切板文件路径与给定文件路径不一致。
126
+ """
127
+ # 设置文件到剪切板
128
+ set_clipboard_data(fmt, buf)
129
+ # 验证剪切板中的文件路径是否与给定的文件路径一致
130
+ if set(get_clipboard_files()) == set(file_paths):
131
+ return True
132
+ raise ValueError("剪切板文件路径不对哇!")
133
+
134
+
135
+ def copy_files_to_clipboard(file_paths: Iterable[str]) -> bool:
136
+ """
137
+ 将一系列文件路径复制到Windows剪切板。这允许用户在其他应用程序中,如文件资源管理器中粘贴这些文件。
138
+
139
+ Args:
140
+ file_paths (Iterable): 一个包含文件路径的可迭代对象,每个路径都是一个字符串。
141
+
142
+ Returns:
143
+ bool: 如果成功将文件路径复制到剪切板,则返回 True,否则返回 False
144
+ """
145
+ # 定义所需的 Windows 结构和函数
146
+ CF_HDROP = 15
147
+
148
+ class DROPFILES(ctypes.Structure):
149
+ _fields_ = [("pFiles", wintypes.DWORD),
150
+ ("pt", wintypes.POINT),
151
+ ("fNC", wintypes.BOOL),
152
+ ("fWide", wintypes.BOOL)]
153
+
154
+ offset = ctypes.sizeof(DROPFILES)
155
+ length = sum(len(p) + 1 for p in file_paths) + 1
156
+ size = offset + length * ctypes.sizeof(wintypes.WCHAR)
157
+ buf = (ctypes.c_char * size)()
158
+ df = DROPFILES.from_buffer(buf)
159
+ df.pFiles, df.fWide = offset, True
160
+ for path in file_paths:
161
+ path = os.path.normpath(path)
162
+ array_t = ctypes.c_wchar * (len(path) + 1)
163
+ path_buf = array_t.from_buffer(buf, offset)
164
+ path_buf.value = path
165
+ offset += ctypes.sizeof(path_buf)
166
+ buf[offset:offset + ctypes.sizeof(wintypes.WCHAR)] = b'\0\0'
167
+
168
+ # 验证文件是否成功复制到剪切板
169
+ return validate_clipboard_files([os.path.normpath(file) for file in file_paths], CF_HDROP, buf=buf)
170
+
171
+
172
+ def __2_窗口功能():
173
+ pass
174
+
175
+
176
+ def get_windows_info():
177
+ """ 得到当前机器的全部窗口信息清单 """
178
+ window_list = []
179
+
180
+ def get_all_hwnd(hwnd, mouse):
181
+ thread_id, process_id = win32process.GetWindowThreadProcessId(hwnd)
182
+ proc = psutil.Process(process_id)
183
+
184
+ is_window = win32gui.IsWindow(hwnd)
185
+ is_enabled = win32gui.IsWindowEnabled(hwnd)
186
+ is_visible = win32gui.IsWindowVisible(hwnd)
187
+ text = win32gui.GetWindowText(hwnd)
188
+
189
+ data = {
190
+ 'proc_name': proc.name(),
191
+ 'process_id': process_id,
192
+ 'thread_id': thread_id,
193
+ 'hwnd': hwnd,
194
+ 'ClassName': win32gui.GetClassName(hwnd),
195
+ 'ControlTypeName': '',
196
+ 'WindowText': text,
197
+ 'IsWindow': is_window,
198
+ 'IsWindowEnabled': is_enabled,
199
+ 'IsWindowVisible': is_visible
200
+ }
201
+
202
+ if not data['proc_name'].endswith('.tmp') and is_visible:
203
+ ctrl = uia.ControlFromHandle(hwnd)
204
+ data['ControlTypeName'] = ctrl.ControlTypeName
205
+
206
+ window_list.append(data)
207
+
208
+ win32gui.EnumWindows(get_all_hwnd, 0)
209
+ return pd.DataFrame(window_list)
210
+
211
+
212
+ def find_ctrl(class_name=None, name=None, **kwargs):
213
+ if class_name is not None:
214
+ kwargs['ClassName'] = class_name
215
+ if name is not None:
216
+ kwargs['Name'] = name
217
+ ctrl = uia.WindowControl(**kwargs)
218
+ return ctrl
219
+
220
+
221
+ class UiCtrlNode(NodeMixin, GetAttr, WindowControl):
222
+ _default = 'ctrl'
223
+
224
+ def __0_构建(self):
225
+ pass
226
+
227
+ def __init__(self, ctrl, parent=None, *, build_depth=-1):
228
+ """
229
+ :param ctrl: 当前节点
230
+ :param parent: 父结点
231
+ :param build_depth: 自动构建多少层树节点,默认-1表示构建全部节点
232
+ """
233
+ # 初始化节点信息
234
+ self.ctrl = ctrl
235
+ # 试过了,没用,因为新找出来的都是新构建的类,找不到proxy的
236
+ # self.ctrl.proxy: 'UiCtrlNode' = self # 再给其扩展一个proxy属性,指向其升级过的对象
237
+ self.ctrl_type = ctrl.ControlTypeName
238
+ self.text = ctrl.Name
239
+ self.parent = parent # 指定父节点,用于形成树结构
240
+
241
+ # 自动递归创建子节点
242
+ self.build_children(build_depth)
243
+
244
+ @classmethod
245
+ def init_from_name(cls, class_name=None, name=None, *, build_depth=-1, **kwargs):
246
+ ctrl = find_ctrl(class_name=class_name, name=name, **kwargs)
247
+ return cls(ctrl, build_depth=build_depth)
248
+
249
+ def activate(self, check_seconds=2):
250
+ """ 激活当前窗口
251
+ """
252
+ while True:
253
+ # todo 这种限定情况并不严谨,有概率出现重复的~
254
+ hwnd = win32gui.FindWindow(self.ctrl.ClassName, self.text)
255
+ # logger.info(hwnd)
256
+ if not hwnd:
257
+ return
258
+
259
+ if win32gui.GetForegroundWindow() != hwnd:
260
+ win32gui.ShowWindow(hwnd, win32con.SW_RESTORE)
261
+
262
+ try:
263
+ # 在这里执行SetForegroundWindow,只有程序的第1次运行有效,之后就会被很多全屏类的应用占用最前置,覆盖不了了
264
+ # 为了解决这问题,就只能暴力每次都新开一个程序来执行这个SetForegroundWindow操作
265
+ subprocess.run(
266
+ [sys.executable, "-c", f"import win32gui; win32gui.SetForegroundWindow({hwnd})"],
267
+ stdout=subprocess.PIPE,
268
+ )
269
+ except Exception as e:
270
+ pass
271
+ # 理论上并不需要等待,但加个等待,有助于稳定性检测,如果当前窗口在check_seconds秒内频繁切换,
272
+ # 使用activate虽然激活了,但并不安全,只有check_seconds秒内维持稳定在这个窗口,再进行下游任务会更好
273
+ time.sleep(check_seconds)
274
+ else:
275
+ break
276
+
277
+ def build_children(self, build_depth=-1, child_node_class=None):
278
+ """ 创建并添加子节点到树中 """
279
+ if build_depth == 0:
280
+ return
281
+ self.children = [] # 删除现有的所有子结点
282
+ child_node_class = child_node_class or self.__class__
283
+ for child_ctrl in self.ctrl.GetChildren():
284
+ child_node_class(child_ctrl, parent=self, build_depth=build_depth - 1)
285
+
286
+ def __1_调试(self):
287
+ pass
288
+
289
+ def trace_rect(self, duration_per_circle=2, num_circles=1):
290
+ """ 用鼠标勾画出当前组件的矩形位置区域 """
291
+ from pyxllib.autogui.autogui import UiTracePath
292
+
293
+ rect = self.ctrl.BoundingRectangle
294
+ ltrb = [rect.left, rect.top, rect.right, rect.bottom]
295
+ UiTracePath.from_ltrb(ltrb,
296
+ duration_per_circle=duration_per_circle,
297
+ num_circles=num_circles)
298
+
299
+ def __2_功能(self):
300
+ pass
301
+
302
+ def __getattr__(self, item):
303
+ # 尝试从self.ctrl中获取属性
304
+ return getattr(self.ctrl, item)
305
+
306
+ def __getitem__(self, index):
307
+ """ 通过索引直接访问子节点
308
+
309
+ ui操作经常要各种结构化的访问,加个这个简化引用方式
310
+ """
311
+ try:
312
+ return self.children[index]
313
+ except IndexError: # 如果出现下标错误,需要自动重新刷新所有控件
314
+ # self.parent重建是不够的,但我也不知道为什么self.root重建后就可以了
315
+ # 我的理解是重建后self自己不是都不存在了?
316
+ self.root.build_children()
317
+ # 应该在有些情况下self.root重建还能继续使用,但有些特殊情况应该会炸
318
+ return self.children[index]
319
+
320
+ def get_ctrl_hash_tag(self, level=1):
321
+ """ 生成节点的哈希字符串,以反映子树结构,一般用来对节点做分类及映射到对应处理函数 """
322
+ # 当前节点的类型标识符
323
+ hash_strs = [f"{level}{self.ctrl_type[0].lower()}"]
324
+ # 遍历所有子节点,递归生成子节点的哈希值
325
+ for child in self.children:
326
+ hash_strs.append(f"{child.get_ctrl_hash_tag(level + 1)}")
327
+ return ''.join(hash_strs)
328
+
329
+ def __3_展示(self):
330
+ pass
331
+
332
+ def _format_text(self, text):
333
+ """ 将换行替换为空格的小工具方法 """
334
+ return text.replace('\n', ' ')
335
+
336
+ def __repr__(self):
337
+ """ 用于在打印节点时显示关键信息 """
338
+ return f"UiNode(ctrl_type={self.ctrl_type}, text={self._format_text(self.text)})"
339
+
340
+ def render_tree(self):
341
+ """ 展示以self为根节点的整体内容结构 """
342
+ # 1 渲染自身
343
+ line = [self.ctrl_type]
344
+ line.append(self._format_text(self.text))
345
+
346
+ # 加上控件的坐标信息
347
+ rect = self.ctrl.BoundingRectangle
348
+ line.append(f"[{rect.left}, {rect.top}, {rect.right}, {rect.bottom}]")
349
+
350
+ # 我的hash值
351
+ tag = self.get_ctrl_hash_tag()
352
+ if len(tag) <= 64:
353
+ line.append(tag)
354
+
355
+ lines = [' '.join(line)]
356
+
357
+ # 2 子结点情况
358
+ for child in self.children:
359
+ line = child.render_tree()
360
+ line = textwrap.indent(line, ' ')
361
+ lines.append(line)
362
+ return '\n'.join(lines)