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.
- pyxllib/__init__.py +21 -21
- pyxllib/algo/__init__.py +8 -8
- pyxllib/algo/disjoint.py +54 -54
- pyxllib/algo/geo.py +541 -541
- pyxllib/algo/intervals.py +964 -964
- pyxllib/algo/matcher.py +389 -389
- pyxllib/algo/newbie.py +166 -166
- pyxllib/algo/pupil.py +629 -629
- pyxllib/algo/shapelylib.py +67 -67
- pyxllib/algo/specialist.py +241 -241
- pyxllib/algo/stat.py +494 -494
- pyxllib/algo/treelib.py +149 -149
- pyxllib/algo/unitlib.py +66 -66
- pyxllib/autogui/__init__.py +5 -5
- pyxllib/autogui/activewin.py +246 -246
- pyxllib/autogui/all.py +9 -9
- pyxllib/autogui/autogui.py +852 -852
- pyxllib/autogui/uiautolib.py +362 -362
- pyxllib/autogui/virtualkey.py +102 -102
- pyxllib/autogui/wechat.py +827 -827
- pyxllib/autogui/wechat_msg.py +421 -421
- pyxllib/autogui/wxautolib.py +84 -84
- pyxllib/cv/__init__.py +5 -5
- pyxllib/cv/expert.py +267 -267
- pyxllib/cv/imfile.py +159 -159
- pyxllib/cv/imhash.py +39 -39
- pyxllib/cv/pupil.py +9 -9
- pyxllib/cv/rgbfmt.py +1525 -1525
- pyxllib/cv/slidercaptcha.py +137 -137
- pyxllib/cv/trackbartools.py +251 -251
- pyxllib/cv/xlcvlib.py +1040 -1040
- pyxllib/cv/xlpillib.py +423 -423
- pyxllib/data/echarts.py +240 -240
- pyxllib/data/jsonlib.py +89 -89
- pyxllib/data/oss.py +72 -72
- pyxllib/data/pglib.py +1127 -1127
- pyxllib/data/sqlite.py +568 -568
- pyxllib/data/sqllib.py +297 -297
- pyxllib/ext/JLineViewer.py +505 -505
- pyxllib/ext/__init__.py +6 -6
- pyxllib/ext/demolib.py +246 -246
- pyxllib/ext/drissionlib.py +277 -277
- pyxllib/ext/kq5034lib.py +12 -12
- pyxllib/ext/old.py +663 -663
- pyxllib/ext/qt.py +449 -449
- pyxllib/ext/robustprocfile.py +497 -497
- pyxllib/ext/seleniumlib.py +76 -76
- pyxllib/ext/tk.py +173 -173
- pyxllib/ext/unixlib.py +827 -827
- pyxllib/ext/utools.py +351 -351
- pyxllib/ext/webhook.py +124 -119
- pyxllib/ext/win32lib.py +40 -40
- pyxllib/ext/wjxlib.py +88 -88
- pyxllib/ext/wpsapi.py +124 -124
- pyxllib/ext/xlwork.py +9 -9
- pyxllib/ext/yuquelib.py +1105 -1105
- pyxllib/file/__init__.py +17 -17
- pyxllib/file/docxlib.py +761 -761
- pyxllib/file/gitlib.py +309 -309
- pyxllib/file/libreoffice.py +165 -165
- pyxllib/file/movielib.py +148 -148
- pyxllib/file/newbie.py +10 -10
- pyxllib/file/onenotelib.py +1469 -1469
- pyxllib/file/packlib/__init__.py +330 -330
- pyxllib/file/packlib/zipfile.py +2441 -2441
- pyxllib/file/pdflib.py +426 -426
- pyxllib/file/pupil.py +185 -185
- pyxllib/file/specialist/__init__.py +685 -685
- pyxllib/file/specialist/dirlib.py +799 -799
- pyxllib/file/specialist/download.py +193 -193
- pyxllib/file/specialist/filelib.py +2829 -2829
- pyxllib/file/xlsxlib.py +3131 -3131
- pyxllib/file/xlsyncfile.py +341 -341
- pyxllib/prog/__init__.py +5 -5
- pyxllib/prog/cachetools.py +64 -64
- pyxllib/prog/deprecatedlib.py +233 -233
- pyxllib/prog/filelock.py +42 -42
- pyxllib/prog/ipyexec.py +253 -253
- pyxllib/prog/multiprogs.py +940 -940
- pyxllib/prog/newbie.py +451 -451
- pyxllib/prog/pupil.py +1197 -1197
- pyxllib/prog/sitepackages.py +33 -33
- pyxllib/prog/specialist/__init__.py +391 -391
- pyxllib/prog/specialist/bc.py +203 -203
- pyxllib/prog/specialist/browser.py +497 -497
- pyxllib/prog/specialist/common.py +347 -347
- pyxllib/prog/specialist/datetime.py +198 -198
- pyxllib/prog/specialist/tictoc.py +240 -240
- pyxllib/prog/specialist/xllog.py +180 -180
- pyxllib/prog/xlosenv.py +108 -108
- pyxllib/stdlib/__init__.py +17 -17
- pyxllib/stdlib/tablepyxl/__init__.py +10 -10
- pyxllib/stdlib/tablepyxl/style.py +303 -303
- pyxllib/stdlib/tablepyxl/tablepyxl.py +130 -130
- pyxllib/text/__init__.py +8 -8
- pyxllib/text/ahocorasick.py +39 -39
- pyxllib/text/airscript.js +744 -744
- pyxllib/text/charclasslib.py +121 -121
- pyxllib/text/jiebalib.py +267 -267
- pyxllib/text/jinjalib.py +32 -32
- pyxllib/text/jsa_ai_prompt.md +271 -271
- pyxllib/text/jscode.py +922 -922
- pyxllib/text/latex/__init__.py +158 -158
- pyxllib/text/levenshtein.py +303 -303
- pyxllib/text/nestenv.py +1215 -1215
- pyxllib/text/newbie.py +300 -300
- pyxllib/text/pupil/__init__.py +8 -8
- pyxllib/text/pupil/common.py +1121 -1121
- pyxllib/text/pupil/xlalign.py +326 -326
- pyxllib/text/pycode.py +47 -47
- pyxllib/text/specialist/__init__.py +8 -8
- pyxllib/text/specialist/common.py +112 -112
- pyxllib/text/specialist/ptag.py +186 -186
- pyxllib/text/spellchecker.py +172 -172
- pyxllib/text/templates/echart_base.html +10 -10
- pyxllib/text/templates/highlight_code.html +16 -16
- pyxllib/text/templates/latex_editor.html +102 -102
- pyxllib/text/vbacode.py +17 -17
- pyxllib/text/xmllib.py +747 -747
- pyxllib/xl.py +42 -39
- pyxllib/xlcv.py +17 -17
- {pyxllib-0.3.197.dist-info → pyxllib-0.3.200.dist-info}/METADATA +1 -1
- pyxllib-0.3.200.dist-info/RECORD +126 -0
- {pyxllib-0.3.197.dist-info → pyxllib-0.3.200.dist-info}/licenses/LICENSE +190 -190
- pyxllib-0.3.197.dist-info/RECORD +0 -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
|
pyxllib/text/pupil/__init__.py
CHANGED
@@ -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 *
|