昨天看完老崔的文章,随机研究运行了下代码。今天就发朋友圈撒了一波狗粮!不料好多小伙伴儿问实现方法,现在就写下来。给需要的朋友。

原文是明哥写的,完了老崔做了下修改,然后我在老崔的基础上稍微改动了下,因为老崔的代码我这边运行报错!

以下是明哥的原文:

作者:@明哥
公众号:Python编程时光

2020年,这个看起来如此浪漫的年份,你还是一个人吗?

难不成我还能是一条狗?

18年的时候,写过一篇介绍如何使用 Python 来表白的文章。

虽然创意和使用效果都不错,但有一缺点,这是那个exe文件,女神需要打开电脑,才有可能参与进来,进而被你成功"调戏”。

由于是很早期的文章了,应该有很多人没有看过。

没有看过的,你可以点击这里查看:用Python写一个表白神器让你脱离单身

提醒你一下,后天就是 2月14日了。什么?还是一条狗呢?

行吧,那你赶上了,今天的文章,就是为你而写。

明哥今天来教你如何使用 Python 来向心中的女神表白。

前段时间,在微博上刷到了一条推荐。内容是这样的

出于好奇,我点开了,放大再放大,emmm,有点意思吖…

这四个字,对于像我这样腼腆的DS男来说,还真不好意思说,说出来,万一被拒绝了咋办?

使用套路来表白,并观察对方的反应,你大概能清楚对方是否对你也有好感,先测试下自己有几成的把握再下手或许更稳妥。

今天就教大家一个这样的套路:如何使用 Python 来做出来这样的图,有点浪漫,又有点极客。能不能拿下你女神,就要靠你(命)了。(๑•́₃ •̀๑)

首先,你得先找到一张你女神的高清图片(尽量分辨率高点的吧,效果会好点)。

这里我以一张高圆圆的图来做一下演示,原图是这样的(分辨率是:2000*1328)。

使用我写好的脚本运行后,就生成了这样一张图,请你点击,放大再放大。惊喜?

然后将这张图片发给你的女神,具体话术你自己想咯。


好吧,相比女神来说,你可能更在意这是如何实现的(活该你单身)。

其实原理很简单,代码也还不到20 行。

首先,来讲讲原理。

事实上,每一张图片都是由一个一个的像素点所组成的。而每个像素点,都有自己的颜色,其颜色可以用一个数组来表示:(a,b,c),其中每位数的取值范围都是 0-255。

比如(0,0,0)代表黑色,(255,255,255)代表白色。

当像素点足够多的时候,这张照片就是我们所说的高清照片。

而如果当像素点太少,我们的肉眼就能感知到明显的锯齿感。

用 Excel 画了个图,每一方格代表一个像素,其中若我的字体的大小设置 5(非字号5,而是每个字占用5个像素),效果大概就是如下这样子。

我只要每个像素取出一个像素值,并使用这个像素做为该字的颜色即可,在像素量够多的情况下,从远处看,是能看到我们原来图像的轮廓的。

有了思路,就可以开始我们的代码。

首先,使用 pillow.Image读取图像,并使用load函数获取到每一个像素值。

img_raw = Image.open(img_path)
img_array = img_raw.load()

然后新建一张画布,并选好你要使用的字体和字体大小。

img_new = Image.new("RGB", img_raw.size, (0, 0, 0))
draw = ImageDraw.Draw(img_new)
font = ImageFont.truetype('C:/Windows/fonts/Dengl.ttf', font_size)

由于需要不断循环 “我喜欢你!”,这五个字符。所以这里可以while循环 yield 来实现一个生成器。

def character_generator(text):
    while True:
        for i in range(len(text)):
            yield text[i]

最后,要给这些字加上相应的颜色,写入新创建的画布中。

for y in range(0, img_raw.size[1], font_size):
    for x in range(0, img_raw.size[0], font_size):
        draw.text((x, y), next(ch_gen), font=font, fill=img_array[x, y], direction=None)

最后将成品保存

img_new.convert('RGB').save("F://gyy_save.jpeg")

完整代码如下,供你参考

from PIL import Image, ImageDraw, ImageFont

font_size = 7
text = "我喜欢你!"
img_path = "F://gyy.jpeg"

img_raw = Image.open(img_path)
img_array = img_raw.load()

img_new = Image.new("RGB", img_raw.size, (0, 0, 0))
draw = ImageDraw.Draw(img_new)
font = ImageFont.truetype('C:/Windows/fonts/Dengl.ttf', font_size)

def character_generator(text):
    while True:
        for i in range(len(text)):
            yield text[i]

ch_gen = character_generator(text)

for y in range(0, img_raw.size[1], font_size):
    for x in range(0, img_raw.size[0], font_size):
        draw.text((x, y), next(ch_gen), font=font, fill=img_array[x, y], direction=None)

img_new.convert('RGB').save("F://save.jpeg")

最后再多上几张效果图吧。

接下来是老崔(崔庆才)补充的内容了。

看完之后,我复制下来也来了一段。其中明哥里面有使用一个字体文件,我这边没有这个字体,于是我就自己下载了一个「阿里巴巴普惠体」,大家如果没有的话可以自行来下载,或者大家有中文字体文件也可以直接用,或者直接到我改好的源代码这里来取:https://github.com/Germey/TextImage,里面我上传了这个字体文件。

这里我提供了字体CDN下载链接:点击下载阿里巴巴普惠字体

另外我修改成了支持批量处理的方式,如果你有很多张图片,可以放到 inputs 文件夹下,它会顺次处理并输出到 outputs 文件夹下,就不用再一个个定义图片名称啦。

我还增加了一个参数叫做 font_space,原来的代码生成的字可能有点紧凑,这里我增加这个参数可以控制字和字的距离,比如 font_space = 1,字间距就是字的大小,就是紧密排布的,如果改大一点,比如 font_space = 1.2,字间距就会变成字的 1.2 倍,就留有字大小的 0.2 倍的空隙。

最终代码如下(我已经格式化修改过):

from PIL import Image, ImageDraw, ImageFont
from glob import glob
from os.path import basename, join

# 字体大小
font_size = 10
# 字体间距,1 即间距正好为字体大小,紧凑排布,1.2 为字体大小的 1.2 倍
font_space = 1.2
# 绘制的文本
text = '入目无他人四目皆是你情人节快乐我爱你'
# 字体文件的路径
font_file = 'Alibaba-PuHuiTi-Regular.ttf'
# 输入图片路径
inputs_folder = 'inputs'
# 输出图片路径
outputs_folder = 'outputs'
def process(path):
    img_raw = Image.open(path)
    img_array = img_raw.load()
    img_new = Image.new('RGB', img_raw.size, (0, 0, 0))
    draw = ImageDraw.Draw(img_new)
    font = ImageFont.truetype(font_file, size=font_size)
    
    def character_generator(text):
        while True:
            for i in range(len(text)):
                yield text[i]
    
    ch_gen = character_generator(text)
    
    for y in range(0, img_raw.size[1], int(font_size * font_space)):
        for x in range(0, img_raw.size[0], int(font_size * font_space)):
            draw.text((x, y), next(ch_gen), font=font, fill=img_array[x, y], direction=None)
    
    img_new.convert('RGB').save(join(outputs_folder, basename(path)))

if __name__ == '__main__':
    for path in glob(join(inputs_folder, '*')):
        print('starting processing', path)
        process(path)
        print('finished processing', path)

上图测试:

效果如下:

有能力的小伙伴儿赶紧来试试吧!

发表回复

您的电子邮箱地址不会被公开。 必填项已用 * 标注

评论(3)