forked from MemeCrafters/meme-generator
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathutils.py
436 lines (379 loc) · 14 KB
/
utils.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
import asyncio
import hashlib
import inspect
import math
import random
import time
from dataclasses import dataclass
from enum import Enum
from functools import partial, wraps
from io import BytesIO
from typing import (
TYPE_CHECKING,
Any,
Callable,
Coroutine,
List,
Literal,
Optional,
Protocol,
Tuple,
TypeVar,
)
import httpx
from PIL.Image import Image as IMG
from pil_utils import BuildImage, Text2Image
from pil_utils.types import ColorType, FontStyle, FontWeight
from typing_extensions import ParamSpec
from .config import meme_config
from .exception import MemeGeneratorException
if TYPE_CHECKING:
from .meme import Meme
P = ParamSpec("P")
R = TypeVar("R")
def run_sync(call: Callable[P, R]) -> Callable[P, Coroutine[None, None, R]]:
"""一个用于包装 sync function 为 async function 的装饰器
参数:
call: 被装饰的同步函数
"""
@wraps(call)
async def _wrapper(*args: P.args, **kwargs: P.kwargs) -> R:
loop = asyncio.get_running_loop()
pfunc = partial(call, *args, **kwargs)
result = await loop.run_in_executor(None, pfunc)
return result
return _wrapper
def is_coroutine_callable(call: Callable[..., Any]) -> bool:
"""检查 call 是否是一个 callable 协程函数"""
if inspect.isroutine(call):
return inspect.iscoroutinefunction(call)
if inspect.isclass(call):
return False
func_ = getattr(call, "__call__", None)
return inspect.iscoroutinefunction(func_)
def save_gif(frames: List[IMG], duration: float) -> BytesIO:
output = BytesIO()
frames[0].save(
output,
format="GIF",
save_all=True,
append_images=frames[1:],
duration=duration * 1000,
loop=0,
disposal=2,
optimize=False,
)
# 没有超出最大大小,直接返回
nbytes = output.getbuffer().nbytes
if nbytes <= meme_config.gif.gif_max_size * 10**6:
return output
# 超出最大大小,帧数超出最大帧数时,缩减帧数
n_frames = len(frames)
gif_max_frames = meme_config.gif.gif_max_frames
if n_frames > gif_max_frames:
index = range(n_frames)
ratio = n_frames / gif_max_frames
index = (int(i * ratio) for i in range(gif_max_frames))
new_duration = duration * ratio
new_frames = [frames[i] for i in index]
return save_gif(new_frames, new_duration)
# 超出最大大小,帧数没有超出最大帧数时,缩小尺寸
new_frames = [
frame.resize((int(frame.width * 0.9), int(frame.height * 0.9)))
for frame in frames
]
return save_gif(new_frames, duration)
class Maker(Protocol):
def __call__(self, img: BuildImage) -> BuildImage:
...
class GifMaker(Protocol):
def __call__(self, i: int) -> Maker:
...
def get_avg_duration(image: IMG) -> float:
if not getattr(image, "is_animated", False):
return 0
total_duration = 0
for i in range(image.n_frames):
image.seek(i)
total_duration += image.info["duration"]
return total_duration / image.n_frames
def split_gif(image: IMG) -> List[IMG]:
frames: List[IMG] = []
update_mode = "full"
for i in range(image.n_frames):
image.seek(i)
if image.tile: # type: ignore
update_region = image.tile[0][1][2:] # type: ignore
if update_region != image.size:
update_mode = "partial"
break
last_frame: Optional[IMG] = None
for i in range(image.n_frames):
image.seek(i)
frame = image.copy()
if update_mode == "partial" and last_frame:
frame = last_frame.copy().paste(frame)
frames.append(frame)
image.seek(0)
if image.info.__contains__("transparency"):
frames[0].info["transparency"] = image.info["transparency"]
return frames
def make_jpg_or_gif(
img: BuildImage, func: Maker, keep_transparency: bool = False
) -> BytesIO:
"""
制作静图或者动图
:params
* ``img``: 输入图片
* ``func``: 图片处理函数,输入img,返回处理后的图片
* ``keep_transparency``: 传入gif时,是否保留该gif的透明度
"""
image = img.image
if not getattr(image, "is_animated", False):
return func(img).save_jpg()
else:
frames = split_gif(image)
duration = get_avg_duration(image) / 1000
frames = [func(BuildImage(frame)).image for frame in frames]
if keep_transparency:
image.seek(0)
if image.info.__contains__("transparency"):
frames[0].info["transparency"] = image.info["transparency"]
return save_gif(frames, duration)
def make_png_or_gif(
img: BuildImage, func: Maker, keep_transparency: bool = False
) -> BytesIO:
"""
制作静图或者动图
:params
* ``img``: 输入图片
* ``func``: 图片处理函数,输入img,返回处理后的图片
* ``keep_transparency``: 传入gif时,是否保留该gif的透明度
"""
image = img.image
if not getattr(image, "is_animated", False):
return func(img).save_png()
else:
frames = split_gif(image)
duration = get_avg_duration(image) / 1000
frames = [func(BuildImage(frame)).image for frame in frames]
if keep_transparency:
image.seek(0)
if image.info.__contains__("transparency"):
frames[0].info["transparency"] = image.info["transparency"]
return save_gif(frames, duration)
class FrameAlignPolicy(Enum):
"""
要叠加的gif长度大于基准gif时,是否延长基准gif长度以对齐两个gif
"""
no_extend = 0
"""不延长"""
extend_first = 1
"""延长第一帧"""
extend_last = 2
"""延长最后一帧"""
extend_loop = 3
"""以循环方式延长"""
def make_gif_or_combined_gif(
img: BuildImage,
maker: GifMaker,
frame_num: int,
duration: float,
frame_align: FrameAlignPolicy = FrameAlignPolicy.no_extend,
input_based: bool = False,
keep_transparency: bool = False,
) -> BytesIO:
"""
使用静图或动图制作gif
:params
* ``img``: 输入图片,如头像
* ``maker``: 图片处理函数生成,传入第几帧,返回对应的图片处理函数
* ``frame_num``: 目标gif的帧数
* ``duration``: 相邻帧之间的时间间隔,单位为秒
* ``frame_align``: 要叠加的gif长度大于基准gif时,gif长度对齐方式
* ``input_based``: 是否以输入gif为基准合成gif,默认为`False`,即以目标gif为基准
* ``keep_transparency``: 传入gif时,是否保留该gif的透明度
"""
image = img.image
if not getattr(image, "is_animated", False):
return save_gif([maker(i)(img).image for i in range(frame_num)], duration)
frame_num_in = image.n_frames
duration_in = get_avg_duration(image) / 1000
total_duration_in = frame_num_in * duration_in
total_duration = frame_num * duration
if input_based:
frame_num_base = frame_num_in
frame_num_fit = frame_num
duration_base = duration_in
duration_fit = duration
total_duration_base = total_duration_in
total_duration_fit = total_duration
else:
frame_num_base = frame_num
frame_num_fit = frame_num_in
duration_base = duration
duration_fit = duration_in
total_duration_base = total_duration
total_duration_fit = total_duration_in
frame_idxs: List[int] = list(range(frame_num_base))
diff_duration = total_duration_fit - total_duration_base
diff_num = int(diff_duration / duration_base)
if diff_duration >= duration_base:
if frame_align == FrameAlignPolicy.extend_first:
frame_idxs = [0] * diff_num + frame_idxs
elif frame_align == FrameAlignPolicy.extend_last:
frame_idxs += [frame_num_base - 1] * diff_num
elif frame_align == FrameAlignPolicy.extend_loop:
frame_num_total = frame_num_base
# 重复基准gif,直到两个gif总时长之差在1个间隔以内,或总帧数超出最大帧数
while frame_num_total + frame_num_base <= meme_config.gif.gif_max_frames:
frame_num_total += frame_num_base
frame_idxs += list(range(frame_num_base))
multiple = round(frame_num_total * duration_base / total_duration_fit)
if (
math.fabs(
total_duration_fit * multiple - frame_num_total * duration_base
)
<= duration_base
):
break
frames: List[IMG] = []
frame_idx_fit = 0
time_start = 0
for i, idx in enumerate(frame_idxs):
while frame_idx_fit < frame_num_fit:
if (
frame_idx_fit * duration_fit
<= i * duration_base - time_start
< (frame_idx_fit + 1) * duration_fit
):
if input_based:
idx_in = idx
idx_maker = frame_idx_fit
else:
idx_in = frame_idx_fit
idx_maker = idx
func = maker(idx_maker)
image.seek(idx_in)
frames.append(func(BuildImage(image.copy())).image)
break
else:
frame_idx_fit += 1
if frame_idx_fit >= frame_num_fit:
frame_idx_fit = 0
time_start += total_duration_fit
if keep_transparency:
image.seek(0)
if image.info.__contains__("transparency"):
frames[0].info["transparency"] = image.info["transparency"]
return save_gif(frames, duration)
async def translate(text: str, lang_from: str = "auto", lang_to: str = "zh") -> str:
appid = meme_config.translate.baidu_trans_appid
apikey = meme_config.translate.baidu_trans_apikey
if not appid or not apikey:
raise MemeGeneratorException(
"The `baidu_trans_appid` or `baidu_trans_apikey` is not set."
"Please check your config file!"
)
salt = str(round(time.time() * 1000))
sign_raw = appid + text + salt + apikey
sign = hashlib.md5(sign_raw.encode("utf8")).hexdigest()
params = {
"q": text,
"from": lang_from,
"to": lang_to,
"appid": appid,
"salt": salt,
"sign": sign,
}
url = "https://fanyi-api.baidu.com/api/trans/vip/translate"
async with httpx.AsyncClient() as client:
resp = await client.get(url, params=params)
result = resp.json()
return result["trans_result"][0]["dst"]
def random_text() -> str:
return random.choice(["刘一", "陈二", "张三", "李四", "王五", "赵六", "孙七", "周八", "吴九", "郑十"])
def random_image() -> BytesIO:
text = random.choice(["😂", "😅", "🤗", "🤤", "🥵", "🥰", "😍", "😭", "😋", "😏"])
return (
BuildImage.new("RGBA", (500, 500), "white")
.draw_text((0, 0, 500, 500), text, max_fontsize=400)
.save_png()
)
@dataclass
class TextProperties:
fill: ColorType = "black"
style: FontStyle = "normal"
weight: FontWeight = "normal"
stroke_width: int = 0
stroke_fill: Optional[ColorType] = None
def default_template(meme: "Meme", number: int) -> str:
return f"{number}. {'/'.join(meme.keywords)}"
def render_meme_list(
meme_list: List[Tuple["Meme", TextProperties]],
*,
template: Callable[["Meme", int], str] = default_template,
order_direction: Literal["row", "column"] = "column",
columns: int = 4,
column_align: Literal["left", "center", "right"] = "left",
item_padding: Tuple[int, int] = (15, 6),
image_padding: Tuple[int, int] = (50, 50),
bg_color: ColorType = "white",
fontsize: int = 30,
fontname: str = "",
fallback_fonts: List[str] = [],
) -> BytesIO:
item_images: List[Text2Image] = []
for i, (meme, properties) in enumerate(meme_list, start=1):
text = template(meme, i)
t2m = Text2Image.from_text(
text,
fontsize=fontsize,
style=properties.style,
weight=properties.weight,
fill=properties.fill,
stroke_width=properties.stroke_width,
stroke_fill=properties.stroke_fill,
fontname=fontname,
fallback_fonts=fallback_fonts,
)
item_images.append(t2m)
char_A = (
Text2Image.from_text(
"A", fontsize=fontsize, fontname=fontname, fallback_fonts=fallback_fonts
)
.lines[0]
.chars[0]
)
num_per_col = math.ceil(len(item_images) / columns)
column_images: List[BuildImage] = []
for col in range(columns):
if order_direction == "column":
images = item_images[col * num_per_col : (col + 1) * num_per_col]
else:
images = [
item_images[num * columns + col]
for num in range((len(item_images) - col - 1) // columns + 1)
]
img_w = max(t2m.width for t2m in images) + item_padding[0] * 2
img_h = (char_A.ascent + item_padding[1] * 2) * len(images) + char_A.descent
image = BuildImage.new("RGB", (img_w, img_h), bg_color)
y = item_padding[1]
for t2m in images:
if column_align == "left":
x = 0
elif column_align == "center":
x = (img_w - t2m.width - item_padding[0] * 2) // 2
else:
x = img_w - t2m.width - item_padding[0] * 2
t2m.draw_on_image(image.image, (x, y))
y += char_A.ascent + item_padding[1] * 2
column_images.append(image)
img_w = sum(img.width for img in column_images) + image_padding[0] * 2
img_h = max(img.height for img in column_images) + image_padding[1] * 2
image = BuildImage.new("RGB", (img_w, img_h), bg_color)
x, y = image_padding
for img in column_images:
image.paste(img, (x, y))
x += img.width
return image.save_jpg()