学习一个知识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方法实现)。
最终效果