Python 实现图片转字符画,静态图、GIF 都能转

来源:Python 技术「ID: pythonall」

字符画是一种由字母、标点或其他字符组成的图画,它产生于互联网时代,在聊天软件中使用较多,本文我们看一下如何将自己喜欢的图片转成字符画。

静态图片

首先,我们来演示将静态图片转为字符画,功能实现主要用到的 Python 库为 OpenCV,安装使用 pip install opencv-python 命令即可。

功能实现的基本思路为:利用聚类将像素信息聚为 3 或 5 类,颜色最深的一类用数字密集度表示,阴影的一类用横杠(-)表示,明亮部分用空白表示。

主要代码实现如下:

def img2strimg(frame, K=5):   
    if type(frame) != np.ndarray:
        frame = np.array(frame)
    height, width, *_ = frame.shape  
    frame_gray = cv2.cvtColor(frame, cv2.COLOR_BGR2GRAY)
    frame_array = np.float32(frame_gray.reshape(-1))
    criteria = (cv2.TERM_CRITERIA_EPS + cv2.TERM_CRITERIA_MAX_ITER, 10, 1.0)
    flags = cv2.KMEANS_RANDOM_CENTERS
    # 得到 labels(类别)、centroids(矩心)
    compactness, labels, centroids = cv2.kmeans(frame_array, K, None, criteria, 10, flags)
    centroids = np.uint8(centroids)
    # labels 的数个矩心以随机顺序排列,所以需要简单处理矩心
    centroids = centroids.flatten()
    centroids_sorted = sorted(centroids)
    # 获得不同 centroids 的明暗程度,0 为最暗
    centroids_index = np.array([centroids_sorted.index(value) for value in centroids])
    bright = [abs((3 * i - 2 * K) / (3 * K)) for i in range(1, 1 + K)]
    bright_bound = bright.index(np.min(bright))
    shadow = [abs((3 * i - K) / (3 * K)) for i in range(1, 1 + K)]
    shadow_bound = shadow.index(np.min(shadow))
    labels = labels.flatten()
    # 将 labels 转变为实际的明暗程度列表
    labels = centroids_index[labels]
    # 解析列表
    labels_picked = [labels[rows * width:(rows + 1) * width:2] for rows in range(0, height, 2)]
    canvas = np.zeros((3 * height, 3 * width, 3), np.uint8)
 # 创建长宽为原图三倍的白色画布
    canvas.fill(255)
    y = 8
    for rows in labels_picked:
        x = 0
        for cols in rows:
            if cols <= shadow_bound:
                cv2.putText(canvas, str(random.randint(2, 9)),
                            (x, y), cv2.FONT_HERSHEY_PLAIN, 0.45, 1)
            elif cols <= bright_bound:
                cv2.putText(canvas, "-", (x, y),
                            cv2.FONT_HERSHEY_PLAIN, 0.4, 0, 1)
            x += 6
        y += 6
    return canvas

原图如下:

效果图如下:

GIF 动图

接下来我们演示将 GIF 转为字符画,功能实现主要用到的 Python 库为 imageio、Pillow,安装使用 pip install imageio/Pillow 命令即可。

功能实现的基本思路如下:

  • 将 gif 图片的每一帧拆分为静态图片
  • 将所有静态图片变为字符画
  • 将所有字符画重新合成 gif

主要代码实现如下:

# 拆分 gif 将每一帧处理成字符画
def gif2pic(file, ascii_chars, isgray, font, scale):
    '''
    file: gif 文件
    ascii_chars: 灰度值对应的字符串
    isgray: 是否黑白
    font: ImageFont 对象
    scale: 缩放比例
    '''
    im = Image.open(file)
    path = os.getcwd()
    if(not os.path.exists(path+"/tmp")):
        os.mkdir(path+"/tmp")
    os.chdir(path+"/tmp")
    # 清空 tmp 目录下内容
    for f in os.listdir(path+"/tmp"):
        os.remove(f)
    try:
        while 1:
            current = im.tell()
            name = file.split('.')[0]+'_tmp_'+str(current)+'.png'
            # 保存每一帧图片
            im.save(name)
            # 将每一帧处理为字符画
            img2ascii(name, ascii_chars, isgray, font, scale)
            # 继续处理下一帧
            im.seek(current+1)
    except:
        os.chdir(path)

# 将不同的灰度值映射为 ASCII 字符
def get_char(ascii_chars, r, g, b):
    length = len(ascii_chars)
    gray = int(0.2126 * r + 0.7152 * g + 0.0722 * b)
    return ascii_chars[int(gray/(256/length))]

# 将图片处理成字符画
def img2ascii(img, ascii_chars, isgray, font, scale):
    scale = scale
    # 将图片转换为 RGB 模式
    im = Image.open(img).convert('RGB')
    # 设定处理后的字符画大小
    raw_width = int(im.width * scale)
    raw_height = int(im.height * scale)
    # 获取设定的字体的尺寸
    font_x, font_y = font.getsize(' ')
    # 确定单元的大小
    block_x = int(font_x * scale)
    block_y = int(font_y * scale)
    # 确定长宽各有几个单元
    w = int(raw_width/block_x)
    h = int(raw_height/block_y)
    # 将每个单元缩小为一个像素
    im = im.resize((w, h), Image.NEAREST)
    # txts 和 colors 分别存储对应块的 ASCII 字符和 RGB 值
    txts = []
    colors = []
    for i in range(h):
        line = ''
        lineColor = []
        for j in range(w):
            pixel = im.getpixel((j, i))
            lineColor.append((pixel[0], pixel[1], pixel[2]))
            line += get_char(ascii_chars, pixel[0], pixel[1], pixel[2])
        txts.append(line)
        colors.append(lineColor)
    # 创建新画布
    img_txt = Image.new('RGB', (raw_width, raw_height), (255, 255, 255))
    # 创建 ImageDraw 对象以写入 ASCII
    draw = ImageDraw.Draw(img_txt)
    for j in range(len(txts)):
        for i in range(len(txts[0])):
            if isgray:
                draw.text((i * block_x, j * block_y), txts[j][i], (119,136,153))
            else:
                draw.text((i * block_x, j * block_y), txts[j][i], colors[j][i])
    img_txt.save(img)

# 读取 tmp 目录下文件合成 gif
def pic2gif(dir_name, out_name, duration):
    path = os.getcwd()
    os.chdir(dir_name)
    dirs = os.listdir()
    images = []
    num = 0
    for d in dirs:
        images.append(imageio.imread(d))
        num += 1
    os.chdir(path)
    imageio.mimsave(out_name + '_ascii.gif',images,duration = duration)

原图如下:

黑白效果图如下:

彩色效果图如下:

总结

本文我们利用 Python 演示了将静态图和 GIF 转为字符画的方法,大家如果有兴趣的话,可以将自己喜欢的图转一下,如果对转换效果不满意,还可以修改代码,改成自己满意的效果。

(0)

相关推荐

  • 【OpenCV 4开发详解】图像连通域分析

    小白学视觉",选择"星标"公众号 重磅干货,第一时间送达 经过几个月的努力,小白终于完成了市面上第一本OpenCV 4入门书籍<OpenCV 4开发详解>.为 ...

  • 一段蛋疼的代码:超不清视频播放器

    在控制台里播放视频,用字符来表示画面 不过我觉得它至少可以有三个作用:1.用来练习视频和图像处理的编程开发:2.在没有图形界面的服务器上播放视频(虽然效果不咋地):3.作为一种独特的艺术风格化处理 程 ...

  • 图像旋转90/180 opencv坐标系

    问题描述: 图像旋转90.180.270等 使用类似下面的代码,会有黑边,图像变形之类的问题 其实windows系统自带类似的功能,但是我需要批量处理图像,因此尝试自己写 算法基础: 这种直角的旋转, ...

  • python+opencv图像处理(四十四)

    童年记忆中的爆米花,小时候只在过年的时候才能吃到,现在偶尔能在路边看到有在爆的,看到了总忍不住想买,虽然味道没什么变化,只是再吃不出来以前的幸福了. 长大了......心也大了...... 可是,实力 ...

  • 【OpenCV 4开发详解】图像腐蚀

     小白学视觉",选择"星标"公众号 重磅干货,第一时间送达 经过几个月的努力,小白终于完成了市面上第一本OpenCV 4入门书籍<OpenCV 4开发详解>. ...

  • Base-x 编码的奥秘

    Base Encoding 是一组二进制转文本的编码模式(Encoding Scheme),常见的有 Base64.Base58.Base32.Base16.大家不仅疑惑为什么需要二进制转文本这种编码 ...

  • python+opencv图像处理(四十二)

    Kirsch算子 1.Kirsch算子 Kirsch算子是R.Kirsch提出来一种边缘检测新算法,它采用8个模板对图像上的每一个像素点进行卷积求导数,这8个模板代表8个方向,对图像上的8个特定边缘方 ...

  • 我用Python做出真的“字节跳动”!

    我用Python做出真的"字节跳动"! Python入门与进阶 2021-07-14 来源:https://blog.csdn.net/hhladminhhl/article/de ...

  • Python:图片转字符画(~情人节神器~)

    字符画真的很有意思,将图片中的像素用字符代替,就生成了字符画. 但是像素是有颜色深浅的,我们如何将带有不同颜色的像素编码为对应的字符呢? 转化方法: 将彩色图片转化为灰度图 根据颜色深浅的RGB值(值 ...

  • 如何使用 Python 将图片变为字符的模样

    我们先来看一下,小帅b原本的头像是这样的: 用 Python 写的脚本转一下,就变成这样了: 也就是说,将图片中的轮廓,用你喜欢的字符代替,是不是有点装逼了: 这是如何实现呢?接下来就是: 学习 Py ...

  • 不会画天花图的都进来看看!就这破玩意有那么难?(内附天花布置图讲解视频)

    文章下方附天花布置图讲解视频 众所周知,天花图在室内设计中是非常重要的一环,其功能的综合性比较强,除却装饰作用外,还兼有照明.音响.空调.防火等功能. 但是对于设计师,特别是对于从业不久的设计师来说, ...

  • 用 Python 教你画花样图

    来源:Python 技术「ID: pythonall」 在之前的一篇文章Python可视化神器-Plotly动画展示展现了可视化神器-Plotly的动画的基本应用,本文介绍如何在Python中使用 P ...

  • 王琦瓷板画渔翁图四条屏图片赏析

    王琦(1866-1933),为"珠山八友"之首,字:碧珍,别号:陶迷道人,斋名"匋匋斋".生于1886年,卒于1933年.祖籍安徽,后迁居江西新建县.幼年曾读私 ...

  • 欣赏 | 蒋廷锡没骨牡丹画50图 分外妖娆

    蒋廷锡(1669-1732),字扬孙,又字酉君,出身于江苏常熟名宦之家.他是江南才子,也是内廷重臣,是诗文皆长的士大夫,也是独步宫廷的词臣画家. <百种牡丹图>,雍正年间词臣画家--蒋廷锡 ...

  • 2021全新夏季周五早安句子问候祝福图片 最漂亮早上好动态表情图片大全,早晨好动图美图大全

    人生中,观众向来比朋友多,观众只会让人从视觉上舒服,朋友却会让你内心感动!朋友,不是天天见面,吃喝玩乐,互相吹捧,而是懂你,在精神上,灵魂上,支持你,鼓励你,帮助你.在你有所不足时,指正你!早安 用每 ...

  • 5月7日最漂亮早上好动态表情图片大全,早晨好动图美图大全

    清晨的第一缕阳光照亮心房,清晨的第一声问候由我轻轻送上,都说时间是种极好的东西,原谅了不可原谅的,过去了曾经过不去的,早安~ 生活总是这样,不能叫人处处都满意,但我们还要热情的活下去,人活一生,值得爱 ...

  • 5月10日最漂亮早上好动态表情图片大全,早晨好动图美图大全

    缘分天空 有缘千里来相聚 812篇原创内容 公众号 醉相思 祁隆 - 醉相思 1.天气在变,思念不变:季节在变,关怀不变:岁月在变,问候不变:距离在变,友情不变.不管世界怎么变,对你的祝福始终不变.天 ...