【Python AsciiArt】利用命令行打印出字符图案

利用字符串生成工具可以方便的打印出自己想要字符图案,以及如何将图片转换为ASCII ART输出。
最终可以用命令行输出各种彩色图像---->>
利用背景显示,参数48,占位符grays[-2]


一般字符图案

我们在使用一些开源软件的时候,启动打开的字符图形很好看。于是搜索到了一些工具来实现:

"""
 _              _   _                        
| |            | | ( )                       
| |       ___  | |_|/   ___    __ _    ___   
| |      / _ \ | __|   / __|  / _` |  / _ \  
| |____ |  __/ | |_    \__ \ | (_| | | (_) | 
\_____/  \___|  \__|   |___/  \__, |  \___/  
                               __/ |         
                              |___/                                                                                                                                                                                         
"""

1.网站taag

在这里插入图片描述
可以随意输入内容,调节字体、宽、高。

同时作者还有另外一个ascii图形生成器:http://patorjk.com/arial-ascii-art

#一条小鳄鱼
#                    __  __
#       _ _        /_@)_@) \          /^^\ /^\ /^^\_ 
#    _/oo \____/~''. . .  '~\       /'\''  ~ ''~~' -'\_ 
#   / '.'. ~.~.~.       .'    ~ |     /'\~~..''''.'' ''  ~\_ 
#  ('_'_'_'_'_'_'_'_  ' :   '     \_/' '.''  . '.   .''  '.  ~\_ 
#  ~V~V~V~V  \   ~\  '' '~  '   '' ~   `   ~  ''   ~\_ 
#    /\~/\~/\~/\~/|/  '   ''  _   ' ~ ''  '    ~  '' __  '  ..  \_ 
# <-- --- ---.---.--/'   ''   /'  '\_ '' ': ~ ;;''    ' /''; \ ;'''''' '' ~\ _ 
#    \~ '. . : .:: ~. :.  /_'''_'' \_' :'''_ : _ ''/''_' '_ \:_ '''' #''..\/\/\/~/\~ ''~~~~~O
# ~~ \-~ `---~~~---- \(_)(_)(_)/ ~ ~~' ~\(_)(_)(_)\_~_~_~_~_~/˜¤¹

如果想要在python中输出,只需要把上面的字符串赋值然后使用print函数打印即可,需要用多行注释来包含这些字符:
在这里插入图片描述

2.命令行工具figlet

figlet [ -cklnoprstvxDELNRSWX ] [ -d fontdirectory ]
      [ -f fontfile ] [ -m layoutmode ]
      [ -w outputwidth ] [ -C controlfile ]
      [ -I infocode ] [ message ]

安转后直接在命令行中使用即可。更多高级用法参考doc


图片字符串图案

3.在python中显示字符串图片

这种方法的主要原理是利用一组视觉密度不同的字符,按照灰度去替换每一个像素:

  • 可以将图像的灰度定义为不同的级别来显示:
    gscale1 = "$@B%8&WM#*oahkbdpqwmZO0QLCJUYXzcvunxrjft/\|()1{}[]?-_+~i!lI;:,\"^.
    gscale2 = "@%#*+=-:. " 这种灰度级别少一些
  • 然后读入图像,将图像映射为长宽等比的矩阵;
  • 然后将颜色灰度值映射到定义的灰度级别上来。
import sys
import cv2

grays = "@%#*+=-:. "   #由于控制台是白色背景,所以先密后疏/黑色背景要转置一下
gs = 10                #10级灰度
#grays2 = "$@B%8&WM#*oahkbdpqwmZO0QLCJUYXzcvunxrjft/\|()1{}[]?-_+~i!lI;:,\"^.` "
#gs2 = 67

img = cv2.imread('where/your/img.jpg',0)  #读入灰度图

#宽(列)和高(行数)
w = img.shape[1]
h = img.shape[0]
ratio = float(w)/h  #调整长宽比 (**注:此比例为win cmd,ratio需要根据不同终端的字符长宽调整)   

scale = w // 50    #缩放尺度,向下取整,每50个像素取一个 值越小图越小(scale 越大)

for y in range(0, h, int(scale*ratio)):  #根据缩放长度 遍历高度 y对于h,x对应w
    for x in range(0, w, scale):  #根据缩放长度 遍历宽度
        idx=img[y][x] * gs // 255  #获取每个点的灰度  根据不同的灰度填写相应的 替换字符
        if idx==gs:
            idx=gs-1
        sys.stdout.write(grays[idx])  #写入控制台
    sys.stdout.write('\n')
    sys.stdout.flush()

在这里插入图片描述在这里插入图片描述


4.输出图像

首先我们将上面的灰度图转字符的代码封装成一个函数img_ascii

def img_ascii(img,r=3):
    #img: input img
    #r:  raito params #由于不同控制台的字符长宽比不同,所以比例需要适当调整。
    #window cmd:r=3/linux console r=
    
    grays = "@%#*+=-:. "   #由于控制台是白色背景,所以先密后疏/黑色背景要转置一下
    gs = 10                #10级灰度
    #grays2 = "$@B%8&WM#*oahkbdpqwmZO0QLCJUYXzcvunxrjft/\|()1{}[]?-_+~i!lI;:,\"^.` "
    #gs2 = 67              #67级灰度

    #宽(列)和高(行数)
    w = img.shape[1]
    h = img.shape[0]
    ratio = r*float(w)/h  #调整长宽比-根据终端改变r

    scale = w // 100    #缩放尺度/取值步长,向下取整,每100/50个像素取一个 值越小图越小(scale 越大)

    for y in range(0, h, int(scale*ratio)):  #根据缩放长度 遍历高度 y对于h,x对应w
        strline=''
        for x in range(0, w, scale):  #根据缩放长度 遍历宽度
            idx=img[y][x] * gs // 255  #获取每个点的灰度  根据不同的灰度填写相应的 替换字符
            if idx==gs:
                idx=gs-1
            strline+=grays[idx]  #写入控制台
        print(strline)
        #sys.stdout.flush()

然后利用控制台输出彩色的功能

#控制带自带的256色输出功能,demo如下
#from:#https://askubuntu.com/questions/821157/print-a-256-color-test-pattern-in-the-terminal

print("Color indexes should be drawn in bold text of the same color.")

colored = [0] + [0x5f + 40 * n for n in range(0, 5)]    #array combined [0, 95, 135, 175, 215, 255]
colored_palette = [
    "%02x/%02x/%02x" % (r, g, b)    #转为16进制
    for r in colored
    for g in colored
    for b in colored
]

grayscale = [0x08 + 10 * n for n in range(0, 24)]
grayscale_palette = [
    "%02x/%02x/%02x" % (a, a, a)
    for a in grayscale 
]

normal = "\033[38;5;%sm" 
bold = "\033[1;38;5;%sm"
reset = "\033[0m"

for (i, color) in enumerate(colored_palette + grayscale_palette, 16):
    index = (bold + "%4s" + reset) % (i, str(i) + ':')
    hex   = (normal + "%s" + reset) % (i, color)
    newline = '\n' if i % 6 == 3 else ''
    print(index, hex, newline,)
##ref
#https://en.wikipedia.org/wiki/ANSI_escape_code
#https://github.com/grawity/code/blob/master/term/xterm-color-chooser
#https://unix.stackexchange.com/questions/404414/print-true-color-24-bit-test-pattern/404415#404415

可见核心是利用格式化输出ASCII转义码(ANSI escape code)来实现的:
print("\033[38;5;%sm #\n"%'1') #其中%s对应的就是256位颜色的一种
在这里插入图片描述

我们需要根据rgb值和对应的颜色构建查表或转换函数来将图形中的颜色(r,g,b)转为n[0,255]:

based = range(0,16)
based_palette = [
    "%02x" %l    #转为16进制
    for l in based
]

colored = [0] + [0x5f + 40 * n for n in range(0, 5)]    #array combined [0, 95, 135, 175, 215, 255]
colored_palette = [
    "%02x/%02x/%02x" % (r, g, b)    #转为16进制
    for r in colored
    for g in colored
    for b in colored
]

grayscale = [0x08 + 10 * n for n in range(0, 24)]
grayscale_palette = [
    "%02x/%02x/%02x" % (a, a, a)
    for a in grayscale 
]

color_256 = based_palette + colored_palette + grayscale_palette 
#生成一个字典
color_dict={color:i for (i,color) in enumerate(color_256)}
#通过rgb值近似到00/5f/87/af/d7/ff来得到彩色值

#输出显示各种颜色
index =''
for (i,color) in enumerate(color_256):
    index +=  "\033[38;5;%sm#"%i   #其中#为各个颜色的输出显示
print(index)

在这里插入图片描述
或者可以选择公式来直接转换216种彩色:
16-231: 6 × 6 × 6 cube (216 colors): 16 + 36 × r + 6 × g + b (0 ≤ r, g, b ≤ 5)(待验证)


A.首先我们需要定义一本RGB**到真彩的颜色字典,利用前面生成的color_256来定义:

#先定义颜色和对应的16进制值
based = range(0,16)
based_palette = [
    "%02x" %l    #转为16进制
    for l in based
]

colored = [0] + [0x5f + 40 * n for n in range(0, 5)]    #array combined [0, 95, 135, 175, 215, 255]
colored_palette = [
    "%02x%02x%02x" % (r, g, b)    #转为16进制
    for r in colored
    for g in colored
    for b in colored
]

grayscale = [0x08 + 10 * n for n in range(0, 24)]
grayscale_palette = [
    "%02x%02x%02x" % (a, a, a)
    for a in grayscale 
]

color_256 = based_palette + colored_palette + grayscale_palette 
#生成一个字典
color_dict={color:i for (i,color) in enumerate(color_256)}
#color_dict={}
#for index,name in enumerate(color_256):
#    color_dict[name]=index

B.随后我们需要定义函数将图像的RGB转换为对应的真彩像素:

#首先定义函数,利用颜色字典将RGB颜色转换为真彩对应数值
def cvtrgb(rgb,color_dict):
    xx=''
    #根据上面生成的颜色字典来,对于不同取值区间赋予不同的值
    for i in range(3):
        if rgb[i]<95:
            xx+= '00'
        elif rgb[i]<135:
            xx+= '5f'
        elif rgb[i]<175:
            xx+= '87'
        elif rgb[i]<215:
            xx+= 'af'
        elif rgb[i]<225:
            xx+= 'd7'
        else:
            xx+= 'ff'
        name =  ''.join(xx)
    
    value = color_dict[name]
    return value 

#随后对输入图进行遍历,将所有的RGB值转换为相应的真彩值
def cvtimg(img,color_dict):
    ascii_img = np.array(img[:,:,0],dtype=np.string_)
    for h in range(img.shape[0]):
        for w in range(img.shape[1]):
            ascii_img[h,w] = cvtrgb(img[h,w,:],color_dict)   #调用换色函数
    return ascii_img  #返回值中每一个像素已经是真彩值

C.最后重新定义一个真彩ASCII彩色绘图函数来绘制,将原来的绘图函数略微修改即可:

def img_color_ascii(img,r=3):
    #img: input img
    #r:  raito params #由于不同控制台的字符长宽比不同,所以比例需要适当调整。
    #window cmd:r=3/linux console r=
    
    grays = "@%#*+=-:. "   #由于控制台是白色背景,所以先密后疏/黑色背景要转置一下
    gs = 10                #10级灰度
    #grays2 = "$@B%8&WM#*oahkbdpqwmZO0QLCJUYXzcvunxrjft/\|()1{}[]?-_+~i!lI;:,\"^.` "
    #gs2 = 67              #67级灰度

    #宽(列)和高(行数)
    w = img.shape[1]
    h = img.shape[0]
    ratio = r*float(w)/h  #调整长宽比-根据终端改变r

    scale = w // 100    #缩放尺度/取值步长,向下取整,每100/50个像素取一个 值越小图越小(scale 越大)

    for y in range(0, h, int(scale*ratio)):  #根据缩放长度 遍历高度 y对于h,x对应w
        strline=''
        for x in range(0, w, scale):  #根据缩放长度 遍历宽度
            idx=img[y][x] * gs // 255  #获取每个点的灰度  根据不同的灰度填写相应的 替换字符
            if idx==gs:
                idx=gs-1  #防止溢出
			######改变这里,将真彩值利用命令行格式化输出赋予
            color_id = "\033[38;5;%sm%s"%(img[y][x],grays[2])      #输出!
            strline+= color_id #按行写入控制台
        print(strline)

D.下面就可以使用了:

#导入图片
import cv2
import matplotlib.pyplot as plt
import numpy as np
img0 = cv2.imread('img2.png')
img =cv2.cvtColor(img0,cv2.COLOR_BGR2RGB)
plt.imshow(img)
plt.axis('off')
plt.show()

在这里插入图片描述
随后利用前面定义的颜色转换函数得到ascii编码的颜色值:

#使用前面定义的颜色字典,颜色转换函数cvtrgb和图像映射哈数cvtimg
ass = cvtimg(img,color_dict)
ass = np.array(ass,dtype=np.int)  #将array转化为int类型
img_color_ascii(ass,2.5)          #彩色绘图函数,r=2.5调整比例,由于命令行行距存在需要微调r因子

在这里插入图片描述

下面就是一些转换的图像,一起来感受一下自己的ASCII ART。(由于截图,尺度可能有些许扭曲)
在这里插入图片描述在这里插入图片描述
在这里插入图片描述在这里插入图片描述
在这里插入图片描述在这里插入图片描述
在这里插入图片描述在这里插入图片描述

5. 彩蛋

在这里插入图片描述
在做完了上面所有的代码之后,发现了控制台的输出还可以直接使用RGB来表示:
"\033[38;2;r;g;bm "
所以无需进行颜色空间转换,直接利用RGB即可!
将上面的绘图函数稍加改变:

def img_RGBcolor_ascii(img,r=3):
    #img: input img img here is 3channel!
    #r:  raito params #由于不同控制台的字符长宽比不同,所以比例需要适当调整。
    #window cmd:r=3/linux console r=
    
    grays = "@%#*+=-:. "   #由于控制台是白色背景,所以先密后疏/黑色背景要转置一下
    gs = 10                #10级灰度
    #grays2 = "$@B%8&WM#*oahkbdpqwmZO0QLCJUYXzcvunxrjft/\|()1{}[]?-_+~i!lI;:,\"^.` "
    #gs2 = 67              #67级灰度

    #宽(列)和高(行数)
    w = img.shape[1]
    h = img.shape[0]
    ratio = r*float(w)/h  #调整长宽比-根据终端改变r

    scale = w // 100    #缩放尺度/取值步长,向下取整,每100/50个像素取一个 值越小图越小(scale 越大)

    for y in range(0, h, int(scale*ratio)):  #根据缩放长度 遍历高度 y对于h,x对应w
        strline=''
        for x in range(0, w, scale):  #根据缩放长度 遍历宽度
            idx=int(img[100][100].mean()) * gs // 255  #获取每个点的灰度  根据不同的灰度填写相应的 替换字符
            if idx==gs:
                idx=gs-1  #防止溢出
			######改变这里,将RGB值,利用2控制参数直接输入
            color_id = "\033[38;2;%d;%d;%dm%s"%(img[y][x][0],img[y][x][1],img[y][x][2],grays[2])      #输出!  
            #38为前景  ->> 48为背景 ,使用grays[-1/-2]输出
            strline+= color_id #按行写入控制台
        print(strline)

img_RGBcolor_ascii(img)
就可以愉快绘图了!
利用背景显示,参数48,占位符grays[-2]


在这里插入图片描述
pic from pexels.com

TODO

#https://en.wikipedia.org/wiki/ANSI_escape_code
#opencv:https://docs.opencv.org/3.1.0/de/d25/imgproc_color_conversions.html
def cvtRGB2ITU([r,g,b]):
    [r,g,b] = [r,g,b]/255*5
    color_v = 16 + 36 × r + 6 × g + b
    return color_v
def cvtRGB2ITU(color):
    #array combined [0, 95, 135, 175, 215, 255]  find the region
    color = np.array(color)/40
    color_v = color
    return color_v

ref:
https://en.wikipedia.org/wiki/ASCII_art
https://blog.csdn.net/qq_19655383/article/details/70176349
https://blog.csdn.net/su_bao/article/details/80355001
color:
https://www.text-image.com/index.html
https://www.maketecheasier.com/convert-pictures-ascii-art/
https://fossbytes.com/linux-distribution-logo-ascii-art-terminal/
https://gallery.technet.microsoft.com/scriptcenter/bc15444a-9490-4115-aa40-76d898041724
https://stackoverflow.com/questions/40754673/python-fails-to-output-ansi-color-codes-for-the-terminal
https://en.wikipedia.org/wiki/8-bit_color
changechar:https://www.geeksforgeeks.org/converting-image-ascii-image-python/
++||| https://stackoverflow.com/questions/287871/print-in-terminal-with-colors
modules:
https://pypi.org/project/colorama/
https://pypi.org/project/termcolor/
ternima:
https://unix.stackexchange.com/questions/404414/print-true-color-24-bit-test-pattern/404415#404415
++YY$$ python:https://askubuntu.com/questions/821157/print-a-256-color-test-pattern-in-the-terminal
https://segmentfault.com/q/1010000016688755

posted @ 2018-12-01 13:06  hitrjj  Views(4795)  Comments(0Edit  收藏  举报