pyxllib 0.3.197__py3-none-any.whl → 3.201.1__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 (127) hide show
  1. pyxllib/__init__.py +14 -21
  2. pyxllib/algo/__init__.py +8 -8
  3. pyxllib/algo/disjoint.py +54 -54
  4. pyxllib/algo/geo.py +537 -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 +145 -149
  13. pyxllib/algo/unitlib.py +62 -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 +846 -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 +236 -240
  34. pyxllib/data/jsonlib.py +85 -89
  35. pyxllib/data/oss.py +72 -72
  36. pyxllib/data/pglib.py +1111 -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 +251 -246
  42. pyxllib/ext/drissionlib.py +277 -277
  43. pyxllib/ext/kq5034lib.py +12 -12
  44. pyxllib/ext/qt.py +449 -449
  45. pyxllib/ext/robustprocfile.py +493 -497
  46. pyxllib/ext/seleniumlib.py +76 -76
  47. pyxllib/ext/tk.py +173 -173
  48. pyxllib/ext/unixlib.py +821 -827
  49. pyxllib/ext/utools.py +345 -351
  50. pyxllib/ext/webhook.py +124 -119
  51. pyxllib/ext/win32lib.py +40 -40
  52. pyxllib/ext/wjxlib.py +91 -88
  53. pyxllib/ext/wpsapi.py +124 -124
  54. pyxllib/ext/xlwork.py +9 -9
  55. pyxllib/ext/yuquelib.py +1110 -1105
  56. pyxllib/file/__init__.py +17 -17
  57. pyxllib/file/docxlib.py +757 -761
  58. pyxllib/file/gitlib.py +309 -309
  59. pyxllib/file/libreoffice.py +165 -165
  60. pyxllib/file/movielib.py +144 -148
  61. pyxllib/file/newbie.py +10 -10
  62. pyxllib/file/onenotelib.py +1469 -1469
  63. pyxllib/file/packlib/__init__.py +330 -330
  64. pyxllib/file/packlib/zipfile.py +2441 -2441
  65. pyxllib/file/pdflib.py +422 -426
  66. pyxllib/file/pupil.py +185 -185
  67. pyxllib/file/specialist/__init__.py +681 -685
  68. pyxllib/file/specialist/dirlib.py +799 -799
  69. pyxllib/file/specialist/download.py +193 -193
  70. pyxllib/file/specialist/filelib.py +2825 -2829
  71. pyxllib/file/xlsxlib.py +3122 -3131
  72. pyxllib/file/xlsyncfile.py +341 -341
  73. pyxllib/prog/__init__.py +5 -5
  74. pyxllib/prog/cachetools.py +58 -64
  75. pyxllib/prog/deprecatedlib.py +233 -233
  76. pyxllib/prog/filelock.py +42 -42
  77. pyxllib/prog/ipyexec.py +253 -253
  78. pyxllib/prog/multiprogs.py +940 -940
  79. pyxllib/prog/newbie.py +451 -451
  80. pyxllib/prog/pupil.py +1208 -1197
  81. pyxllib/prog/sitepackages.py +33 -33
  82. pyxllib/prog/specialist/__init__.py +348 -391
  83. pyxllib/prog/specialist/bc.py +203 -203
  84. pyxllib/prog/specialist/browser.py +497 -497
  85. pyxllib/prog/specialist/common.py +347 -347
  86. pyxllib/prog/specialist/datetime.py +198 -198
  87. pyxllib/prog/specialist/tictoc.py +240 -240
  88. pyxllib/prog/specialist/xllog.py +180 -180
  89. pyxllib/prog/xlosenv.py +110 -108
  90. pyxllib/stdlib/__init__.py +17 -17
  91. pyxllib/stdlib/tablepyxl/__init__.py +10 -10
  92. pyxllib/stdlib/tablepyxl/style.py +303 -303
  93. pyxllib/stdlib/tablepyxl/tablepyxl.py +130 -130
  94. pyxllib/text/__init__.py +8 -8
  95. pyxllib/text/ahocorasick.py +36 -39
  96. pyxllib/text/airscript.js +754 -744
  97. pyxllib/text/charclasslib.py +121 -121
  98. pyxllib/text/jiebalib.py +267 -267
  99. pyxllib/text/jinjalib.py +27 -32
  100. pyxllib/text/jsa_ai_prompt.md +271 -271
  101. pyxllib/text/jscode.py +922 -922
  102. pyxllib/text/latex/__init__.py +158 -158
  103. pyxllib/text/levenshtein.py +303 -303
  104. pyxllib/text/nestenv.py +1215 -1215
  105. pyxllib/text/newbie.py +300 -300
  106. pyxllib/text/pupil/__init__.py +8 -8
  107. pyxllib/text/pupil/common.py +1121 -1121
  108. pyxllib/text/pupil/xlalign.py +326 -326
  109. pyxllib/text/pycode.py +47 -47
  110. pyxllib/text/specialist/__init__.py +8 -8
  111. pyxllib/text/specialist/common.py +112 -112
  112. pyxllib/text/specialist/ptag.py +186 -186
  113. pyxllib/text/spellchecker.py +172 -172
  114. pyxllib/text/templates/echart_base.html +10 -10
  115. pyxllib/text/templates/highlight_code.html +16 -16
  116. pyxllib/text/templates/latex_editor.html +102 -102
  117. pyxllib/text/vbacode.py +17 -17
  118. pyxllib/text/xmllib.py +741 -747
  119. pyxllib/xl.py +42 -39
  120. pyxllib/xlcv.py +17 -17
  121. pyxllib-3.201.1.dist-info/METADATA +296 -0
  122. pyxllib-3.201.1.dist-info/RECORD +125 -0
  123. {pyxllib-0.3.197.dist-info → pyxllib-3.201.1.dist-info}/licenses/LICENSE +190 -190
  124. pyxllib/ext/old.py +0 -663
  125. pyxllib-0.3.197.dist-info/METADATA +0 -48
  126. pyxllib-0.3.197.dist-info/RECORD +0 -126
  127. {pyxllib-0.3.197.dist-info → pyxllib-3.201.1.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 *