学习一个知识python匿名函数(lambda表达式)
参见python基础教程
了解一个函数point
from PIL import Image
im = Image.open('35.jpg').convert('RGBA')
_, _, _, alpha = im.split()
alpha = alpha.point(lambda i: 150)
im.putalpha(alpha)
im.save('luozi.png')
准备
1.字体文件demo.ttf
链接:https://pan.baidu.com/s/1wLKMd0VoO7C5jCAAfpjBEw
提取码:ub9l
2.项目目录下新建文件夹out,存放最终生成的图片
代码:
from PIL import Image, ImageDraw, ImageFont
import os
def gen_text_img(text, font_size=20, font_path=None):
# 从文字生成图像,输入:文字内容,文字字体大小,字体路径
font = ImageFont.truetype(font_path, font_size) if font_path is not None else None
(width, length) = font.getsize(text) # 获取文字大小
text_img = Image.new('RGBA', (width, length))
draw = ImageDraw.Draw(text_img)
# 第一个tuple表示未知(left,up),之后是文字,然后颜色,最后设置字体
draw.text((0, 0), text, fill=(0, 0, 0), font=font)
text_img.save('testtext.png')
return text_img
def trans_alpha(img, pixel):
'''根据rgba的pixel调节img的透明度
这里传进来的pixel是一个四元组(r,g,b,alpha)
'''
_, _, _, alpha = img.split()
alpha = alpha.point(lambda i: pixel[-1]*10)
img.putalpha(alpha)
return img
def picture_wall_mask(text_img, edge_len, pic_dir="./user"):
# 根据文字图像生成对应的照片墙,输入:文字图像,各个照片边长,照片所在路径
new_img = Image.new('RGBA', (text_img.size[0] * edge_len, text_img.size[1] * edge_len))
file_list = os.listdir(pic_dir)
img_index = 0
for x in range(0, text_img.size[0]):
for y in range(0, text_img.size[1]):
pixel = text_img.getpixel((x, y))
file_name = file_list[img_index % len(file_list)]
try:
img = Image.open(os.path.join(pic_dir, file_name)).convert('RGBA')
img = img.resize((edge_len, edge_len))
img = trans_alpha(img, pixel)
new_img.paste(img, (x * edge_len, y * edge_len))
img_index += 1
except Exception as e:
print(f"open file {file_name} failed! {e}")
return new_img
def main(text='', font_size = 20, edge_len = 60,pic_dir = "./user", out_dir = "./out/", font_path = './demo.ttf'):
'''生成照片墙
:param text: Text of picture wall, if not defined this will generage a rectangle picture wall
:param font_size: font size of a clear value
:param edge_len: sub picture's egde length
'''
if len(text) >= 1:
text_ = ' '.join(text)#将字符串用空格分隔开
print(f"generate text wall for '{text_}' with picture path:{pic_dir}")
text_img = gen_text_img(text_, font_size, font_path)
# text_img.show()
img_ascii = picture_wall_mask(text_img, edge_len, pic_dir)
# img_ascii.show()
img_ascii.save(out_dir + os.path.sep + '_'.join(text) + '.png')
if __name__ == '__main__':
main(text='东软')
代码原理:
1.gen_text_img函数根据设定的文字以及字体大小生成一张图片(包含设定的文字),如下图所示:
image.png
先用给定的字符串生成一张图片(如上图),然后将该图片的每个像素的宽扩张edge_len倍,高也扩张edge_len倍,假设edge_len=60,那么原文字图片(如上图)的每个像素就变成了60*60像素的一个图片(我们会将每个朋友圈头像放进去);原文字图片的每个像素的透明度不同,比如显示东软这2个字的地方,透明度低(不透明),这2个字周边的地方,透明度高(透明),我们根据原文字图片每个像素的透明度,来设定放到这个像素(其实宽高已经扩大了60倍)位置的微信好友头像的透明度(trans_alpha方法实现)。
最终效果
image.png