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
pyxllib/text/newbie.py CHANGED
@@ -1,300 +1,300 @@
1
- #!/usr/bin/env python3
2
- # -*- coding: utf-8 -*-
3
- # @Author : 陈坤泽
4
- # @Email : 877362867@qq.com
5
- # @Date : 2021/06/06 10:51
6
-
7
- from pyxllib.prog.newbie import round_int
8
-
9
-
10
- class StrDecorator:
11
- """将函数的返回值字符串化,仅调用朴素的str字符串化
12
-
13
- 装饰器开发可参考: https://mp.weixin.qq.com/s/Om98PpncG52Ba1ZQ8NIjLA
14
- """
15
-
16
- def __init__(self, func):
17
- self.func = func # 使用self.func可以索引回原始函数名称
18
- self.last_raw_res = None # last raw result,上一次执行函数的原始结果
19
-
20
- def __call__(self, *args, **kwargs):
21
- self.last_raw_res = self.func(*args, **kwargs)
22
- return str(self.last_raw_res)
23
-
24
-
25
- class PrintDecorator:
26
- """将函数返回结果直接输出"""
27
-
28
- def __init__(self, func):
29
- self.func = func
30
-
31
- def __call__(self, *args, **kwargs):
32
- s = self.func(*args, **kwargs)
33
- print(s)
34
- return s # 输出后仍然会返回原函数运行值
35
-
36
-
37
- def binary_cut_str(s, fmt='0'):
38
- """180801坤泽:“二分”切割字符串
39
- :param s: 要截取的全字符串
40
- :param fmt: 截取格式,本来是想只支持0、1的,后来想想支持23456789也行
41
- 0:左边一半
42
- 1:右边的1/2
43
- 2:右边的1/3
44
- 3:右边的1/4
45
- ...
46
- 9:右边的1/10
47
- :return: 截取后的字符串
48
-
49
- >>> binary_cut_str('1234', '0')
50
- '12'
51
- >>> binary_cut_str('1234', '1')
52
- '34'
53
- >>> binary_cut_str('1234', '10')
54
- '3'
55
- >>> binary_cut_str('123456789', '20')
56
- '7'
57
- >>> binary_cut_str('123456789', '210') # 向下取整,'21'获得了9,然后'0'取到空字符串
58
- ''
59
- """
60
- for t in fmt:
61
- t = int(t)
62
- n = len(s) // (1 + max(1, t))
63
- if t == 0:
64
- s = s[:n]
65
- else:
66
- s = s[(len(s) - n):]
67
- return s
68
-
69
-
70
- def digits2roman(d):
71
- """
72
- >>> digits2roman(2)
73
- 'Ⅱ'
74
- >>> digits2roman(12)
75
- 'Ⅻ'
76
- """
77
- rmn = '~ⅠⅡⅢⅣⅤⅥⅦⅧⅨⅩⅪⅫ' # roman数字number的缩写
78
-
79
- d = int(d) # 确保是整数类型
80
- if d <= 12:
81
- return rmn[d]
82
- else:
83
- raise NotImplementedError
84
-
85
-
86
- def roman2digits(d):
87
- """
88
- >>> roman2digits('Ⅱ')
89
- 2
90
- >>> roman2digits('Ⅻ')
91
- 12
92
- """
93
- rmn = '~ⅠⅡⅢⅣⅤⅥⅦⅧⅨⅩⅪⅫ'
94
- if d in rmn:
95
- return rmn.index(d)
96
- else:
97
- raise NotImplemented
98
-
99
-
100
- def digits2circlednumber(d):
101
- d = int(d)
102
- if 0 < d <= 20:
103
- return '①②③④⑤⑥⑦⑧⑨⑩⑪⑫⑬⑭⑮⑯⑰⑱⑲⑳'[d - 1]
104
- else:
105
- raise NotImplemented
106
-
107
-
108
- def circlednumber2digits(d):
109
- t = '①②③④⑤⑥⑦⑧⑨⑩⑪⑫⑬⑭⑮⑯⑰⑱⑲⑳'
110
- if d in t:
111
- return t.index(d) + 1
112
- else:
113
- raise NotImplemented
114
-
115
-
116
- def endswith(s, tags):
117
- """除了模拟str.endswith方法,输入的tag也可以是可迭代对象
118
-
119
- >>> endswith('a.dvi', ('.log', '.aux', '.dvi', 'busy'))
120
- True
121
- """
122
- if isinstance(tags, str):
123
- return s.endswith(tags)
124
- elif isinstance(tags, (list, tuple)):
125
- for t in tags:
126
- if s.endswith(t):
127
- return True
128
- else:
129
- raise TypeError
130
- return False
131
-
132
-
133
- def xldictstr(d, key_value_delimit='=', item_delimit=' '):
134
- """将一个字典转成字符串"""
135
- res = []
136
- for k, v in d.items():
137
- res.append(str(k) + key_value_delimit + str(v).replace('\n', r'\n'))
138
- res = item_delimit.join(res)
139
- return res
140
-
141
-
142
- def findnth(haystack, needle, n):
143
- """https://stackoverflow.com/questions/1883980/find-the-nth-occurrence-of-substring-in-a-string"""
144
- if n < 0:
145
- n += haystack.count(needle)
146
- if n < 0:
147
- return -1
148
-
149
- parts = haystack.split(needle, n + 1)
150
- if len(parts) <= n + 1:
151
- return -1
152
- return len(haystack) - len(parts[-1]) - len(needle)
153
-
154
-
155
- def refine_digits_set(digits):
156
- """美化连续数字的输出效果
157
-
158
- >>> refine_digits_set([210, 207, 207, 208, 211, 212])
159
- '207,208,210-212'
160
- """
161
- arr = sorted(list(set(digits))) # 去重
162
- n = len(arr)
163
- res = ''
164
- i = 0
165
- while i < n:
166
- j = i + 2
167
- if j < n and arr[i] + 2 == arr[j]:
168
- while j < n and arr[j] - arr[i] == j - i:
169
- j += 1
170
- j = j if j < n else n - 1
171
- res += str(arr[i]) + '-' + str(arr[j]) + ','
172
- i = j + 1
173
- else:
174
- res += str(arr[i]) + ','
175
- i += 1
176
- return res[:-1] # -1是去掉最后一个','
177
-
178
-
179
- def del_tail_newline(s):
180
- """删除末尾的换行"""
181
- if len(s) > 1 and s[-1] == '\n':
182
- s = s[:-1]
183
- return s
184
-
185
-
186
- def bracket_match(s, idx):
187
- """括号匹配位置
188
- 这里以{、}为例,注意也要适用于'[]', '()'
189
- >>> bracket_match('{123}', 0)
190
- 4
191
- >>> bracket_match('0{23{5}}89', 1)
192
- 7
193
- >>> bracket_match('0{23{5}}89', 7)
194
- 1
195
- >>> bracket_match('0{23{5}78', 1) is None
196
- True
197
- >>> bracket_match('0{23{5}78', 20) is None
198
- True
199
- >>> bracket_match('0[2[4]{7}]01', 9)
200
- 1
201
- >>> bracket_match('0{[34{6}89}', -4)
202
- 5
203
- """
204
- key = '{[(<>)]}'
205
- try:
206
- if idx < 0:
207
- idx += len(s)
208
- ch1 = s[idx]
209
- idx1 = key.index(ch1)
210
- except ValueError: # 找不到ch1
211
- return None
212
- except IndexError: # 下标越界,表示没有匹配到右括号
213
- return None
214
- idx2 = len(key) - idx1 - 1
215
- ch2 = key[idx2]
216
- step = 1 if idx2 > idx1 else -1
217
- cnt = 1
218
- i = idx + step
219
- if i < 0:
220
- i += len(s)
221
- while 0 <= i < len(s):
222
- if s[i] == ch1:
223
- cnt += 1
224
- elif s[i] == ch2:
225
- cnt -= 1
226
- if cnt == 0:
227
- return i
228
- i += step
229
- return None
230
-
231
-
232
- def bracket_match2(s, idx):
233
- r"""与“bracket_match”相比,会考虑"\{"转义字符的影响
234
-
235
- >>> bracket_match2('a{b{}b}c', 1)
236
- 6
237
- >>> bracket_match2('a{b{\}b}c}d', 1)
238
- 9
239
- """
240
- key = '{[(<>)]}'
241
- try:
242
- if idx < 0:
243
- idx += len(s)
244
- ch1 = s[idx]
245
- idx1 = key.index(ch1)
246
- except ValueError: # 找不到ch1
247
- return None
248
- except IndexError: # 下标越界,表示没有匹配到右括号
249
- return None
250
- idx2 = len(key) - idx1 - 1
251
- ch2 = key[idx2]
252
- step = 1 if idx2 > idx1 else -1
253
- cnt = 1
254
- i = idx + step
255
- if i < 0:
256
- i += len(s)
257
- while 0 <= i < len(s):
258
- if i and s[i - 1] == '\\':
259
- pass
260
- elif s[i] == ch1:
261
- cnt += 1
262
- elif s[i] == ch2:
263
- cnt -= 1
264
- if cnt == 0:
265
- return i
266
- i += step
267
- return None
268
-
269
-
270
- def latexstrip(s):
271
- """latex版的strip"""
272
- return s.strip('\t\n ~')
273
-
274
-
275
- def add_quote(s):
276
- return f'"{s}"'
277
-
278
-
279
- def fold_dict(d, m=5):
280
- """ 将字典折叠为更紧凑的排版格式
281
-
282
- :param d: 一个字典对象
283
- :param m: 按照每行放m个元素重排
284
- :return: 重排后的字典内容
285
- """
286
- vals = [f"'{k}': {v}" for k, v in d.items()]
287
- line = [', '.join(vals[i:i + 5]) for i in range(0, len(vals), m)]
288
- return '{' + ',\n'.join(line) + '}'
289
-
290
-
291
- def remove_prefix(original_string, prefix):
292
- if original_string.startswith(prefix):
293
- return original_string[len(prefix):]
294
- return original_string
295
-
296
-
297
- def remove_suffix(original_string, suffix):
298
- if original_string.endswith(suffix):
299
- return original_string[:-len(suffix)]
300
- return original_string
1
+ #!/usr/bin/env python3
2
+ # -*- coding: utf-8 -*-
3
+ # @Author : 陈坤泽
4
+ # @Email : 877362867@qq.com
5
+ # @Date : 2021/06/06 10:51
6
+
7
+ from pyxllib.prog.newbie import round_int
8
+
9
+
10
+ class StrDecorator:
11
+ """将函数的返回值字符串化,仅调用朴素的str字符串化
12
+
13
+ 装饰器开发可参考: https://mp.weixin.qq.com/s/Om98PpncG52Ba1ZQ8NIjLA
14
+ """
15
+
16
+ def __init__(self, func):
17
+ self.func = func # 使用self.func可以索引回原始函数名称
18
+ self.last_raw_res = None # last raw result,上一次执行函数的原始结果
19
+
20
+ def __call__(self, *args, **kwargs):
21
+ self.last_raw_res = self.func(*args, **kwargs)
22
+ return str(self.last_raw_res)
23
+
24
+
25
+ class PrintDecorator:
26
+ """将函数返回结果直接输出"""
27
+
28
+ def __init__(self, func):
29
+ self.func = func
30
+
31
+ def __call__(self, *args, **kwargs):
32
+ s = self.func(*args, **kwargs)
33
+ print(s)
34
+ return s # 输出后仍然会返回原函数运行值
35
+
36
+
37
+ def binary_cut_str(s, fmt='0'):
38
+ """180801坤泽:“二分”切割字符串
39
+ :param s: 要截取的全字符串
40
+ :param fmt: 截取格式,本来是想只支持0、1的,后来想想支持23456789也行
41
+ 0:左边一半
42
+ 1:右边的1/2
43
+ 2:右边的1/3
44
+ 3:右边的1/4
45
+ ...
46
+ 9:右边的1/10
47
+ :return: 截取后的字符串
48
+
49
+ >>> binary_cut_str('1234', '0')
50
+ '12'
51
+ >>> binary_cut_str('1234', '1')
52
+ '34'
53
+ >>> binary_cut_str('1234', '10')
54
+ '3'
55
+ >>> binary_cut_str('123456789', '20')
56
+ '7'
57
+ >>> binary_cut_str('123456789', '210') # 向下取整,'21'获得了9,然后'0'取到空字符串
58
+ ''
59
+ """
60
+ for t in fmt:
61
+ t = int(t)
62
+ n = len(s) // (1 + max(1, t))
63
+ if t == 0:
64
+ s = s[:n]
65
+ else:
66
+ s = s[(len(s) - n):]
67
+ return s
68
+
69
+
70
+ def digits2roman(d):
71
+ """
72
+ >>> digits2roman(2)
73
+ 'Ⅱ'
74
+ >>> digits2roman(12)
75
+ 'Ⅻ'
76
+ """
77
+ rmn = '~ⅠⅡⅢⅣⅤⅥⅦⅧⅨⅩⅪⅫ' # roman数字number的缩写
78
+
79
+ d = int(d) # 确保是整数类型
80
+ if d <= 12:
81
+ return rmn[d]
82
+ else:
83
+ raise NotImplementedError
84
+
85
+
86
+ def roman2digits(d):
87
+ """
88
+ >>> roman2digits('Ⅱ')
89
+ 2
90
+ >>> roman2digits('Ⅻ')
91
+ 12
92
+ """
93
+ rmn = '~ⅠⅡⅢⅣⅤⅥⅦⅧⅨⅩⅪⅫ'
94
+ if d in rmn:
95
+ return rmn.index(d)
96
+ else:
97
+ raise NotImplemented
98
+
99
+
100
+ def digits2circlednumber(d):
101
+ d = int(d)
102
+ if 0 < d <= 20:
103
+ return '①②③④⑤⑥⑦⑧⑨⑩⑪⑫⑬⑭⑮⑯⑰⑱⑲⑳'[d - 1]
104
+ else:
105
+ raise NotImplemented
106
+
107
+
108
+ def circlednumber2digits(d):
109
+ t = '①②③④⑤⑥⑦⑧⑨⑩⑪⑫⑬⑭⑮⑯⑰⑱⑲⑳'
110
+ if d in t:
111
+ return t.index(d) + 1
112
+ else:
113
+ raise NotImplemented
114
+
115
+
116
+ def endswith(s, tags):
117
+ """除了模拟str.endswith方法,输入的tag也可以是可迭代对象
118
+
119
+ >>> endswith('a.dvi', ('.log', '.aux', '.dvi', 'busy'))
120
+ True
121
+ """
122
+ if isinstance(tags, str):
123
+ return s.endswith(tags)
124
+ elif isinstance(tags, (list, tuple)):
125
+ for t in tags:
126
+ if s.endswith(t):
127
+ return True
128
+ else:
129
+ raise TypeError
130
+ return False
131
+
132
+
133
+ def xldictstr(d, key_value_delimit='=', item_delimit=' '):
134
+ """将一个字典转成字符串"""
135
+ res = []
136
+ for k, v in d.items():
137
+ res.append(str(k) + key_value_delimit + str(v).replace('\n', r'\n'))
138
+ res = item_delimit.join(res)
139
+ return res
140
+
141
+
142
+ def findnth(haystack, needle, n):
143
+ """https://stackoverflow.com/questions/1883980/find-the-nth-occurrence-of-substring-in-a-string"""
144
+ if n < 0:
145
+ n += haystack.count(needle)
146
+ if n < 0:
147
+ return -1
148
+
149
+ parts = haystack.split(needle, n + 1)
150
+ if len(parts) <= n + 1:
151
+ return -1
152
+ return len(haystack) - len(parts[-1]) - len(needle)
153
+
154
+
155
+ def refine_digits_set(digits):
156
+ """美化连续数字的输出效果
157
+
158
+ >>> refine_digits_set([210, 207, 207, 208, 211, 212])
159
+ '207,208,210-212'
160
+ """
161
+ arr = sorted(list(set(digits))) # 去重
162
+ n = len(arr)
163
+ res = ''
164
+ i = 0
165
+ while i < n:
166
+ j = i + 2
167
+ if j < n and arr[i] + 2 == arr[j]:
168
+ while j < n and arr[j] - arr[i] == j - i:
169
+ j += 1
170
+ j = j if j < n else n - 1
171
+ res += str(arr[i]) + '-' + str(arr[j]) + ','
172
+ i = j + 1
173
+ else:
174
+ res += str(arr[i]) + ','
175
+ i += 1
176
+ return res[:-1] # -1是去掉最后一个','
177
+
178
+
179
+ def del_tail_newline(s):
180
+ """删除末尾的换行"""
181
+ if len(s) > 1 and s[-1] == '\n':
182
+ s = s[:-1]
183
+ return s
184
+
185
+
186
+ def bracket_match(s, idx):
187
+ """括号匹配位置
188
+ 这里以{、}为例,注意也要适用于'[]', '()'
189
+ >>> bracket_match('{123}', 0)
190
+ 4
191
+ >>> bracket_match('0{23{5}}89', 1)
192
+ 7
193
+ >>> bracket_match('0{23{5}}89', 7)
194
+ 1
195
+ >>> bracket_match('0{23{5}78', 1) is None
196
+ True
197
+ >>> bracket_match('0{23{5}78', 20) is None
198
+ True
199
+ >>> bracket_match('0[2[4]{7}]01', 9)
200
+ 1
201
+ >>> bracket_match('0{[34{6}89}', -4)
202
+ 5
203
+ """
204
+ key = '{[(<>)]}'
205
+ try:
206
+ if idx < 0:
207
+ idx += len(s)
208
+ ch1 = s[idx]
209
+ idx1 = key.index(ch1)
210
+ except ValueError: # 找不到ch1
211
+ return None
212
+ except IndexError: # 下标越界,表示没有匹配到右括号
213
+ return None
214
+ idx2 = len(key) - idx1 - 1
215
+ ch2 = key[idx2]
216
+ step = 1 if idx2 > idx1 else -1
217
+ cnt = 1
218
+ i = idx + step
219
+ if i < 0:
220
+ i += len(s)
221
+ while 0 <= i < len(s):
222
+ if s[i] == ch1:
223
+ cnt += 1
224
+ elif s[i] == ch2:
225
+ cnt -= 1
226
+ if cnt == 0:
227
+ return i
228
+ i += step
229
+ return None
230
+
231
+
232
+ def bracket_match2(s, idx):
233
+ r"""与“bracket_match”相比,会考虑"\{"转义字符的影响
234
+
235
+ >>> bracket_match2('a{b{}b}c', 1)
236
+ 6
237
+ >>> bracket_match2('a{b{\}b}c}d', 1)
238
+ 9
239
+ """
240
+ key = '{[(<>)]}'
241
+ try:
242
+ if idx < 0:
243
+ idx += len(s)
244
+ ch1 = s[idx]
245
+ idx1 = key.index(ch1)
246
+ except ValueError: # 找不到ch1
247
+ return None
248
+ except IndexError: # 下标越界,表示没有匹配到右括号
249
+ return None
250
+ idx2 = len(key) - idx1 - 1
251
+ ch2 = key[idx2]
252
+ step = 1 if idx2 > idx1 else -1
253
+ cnt = 1
254
+ i = idx + step
255
+ if i < 0:
256
+ i += len(s)
257
+ while 0 <= i < len(s):
258
+ if i and s[i - 1] == '\\':
259
+ pass
260
+ elif s[i] == ch1:
261
+ cnt += 1
262
+ elif s[i] == ch2:
263
+ cnt -= 1
264
+ if cnt == 0:
265
+ return i
266
+ i += step
267
+ return None
268
+
269
+
270
+ def latexstrip(s):
271
+ """latex版的strip"""
272
+ return s.strip('\t\n ~')
273
+
274
+
275
+ def add_quote(s):
276
+ return f'"{s}"'
277
+
278
+
279
+ def fold_dict(d, m=5):
280
+ """ 将字典折叠为更紧凑的排版格式
281
+
282
+ :param d: 一个字典对象
283
+ :param m: 按照每行放m个元素重排
284
+ :return: 重排后的字典内容
285
+ """
286
+ vals = [f"'{k}': {v}" for k, v in d.items()]
287
+ line = [', '.join(vals[i:i + 5]) for i in range(0, len(vals), m)]
288
+ return '{' + ',\n'.join(line) + '}'
289
+
290
+
291
+ def remove_prefix(original_string, prefix):
292
+ if original_string.startswith(prefix):
293
+ return original_string[len(prefix):]
294
+ return original_string
295
+
296
+
297
+ def remove_suffix(original_string, suffix):
298
+ if original_string.endswith(suffix):
299
+ return original_string[:-len(suffix)]
300
+ return original_string
@@ -1,8 +1,8 @@
1
- #!/usr/bin/env python3
2
- # -*- coding: utf-8 -*-
3
- # @Author : 陈坤泽
4
- # @Email : 877362867@qq.com
5
- # @Date : 2021/06/06 17:20
6
-
7
- from pyxllib.text.pupil.common import *
8
- from pyxllib.text.pupil.xlalign import *
1
+ #!/usr/bin/env python3
2
+ # -*- coding: utf-8 -*-
3
+ # @Author : 陈坤泽
4
+ # @Email : 877362867@qq.com
5
+ # @Date : 2021/06/06 17:20
6
+
7
+ from pyxllib.text.pupil.common import *
8
+ from pyxllib.text.pupil.xlalign import *