shell动画

时间:2023-03-10 00:04:43
shell动画

在印象中,好像终端就是黑白界面,加扁平输出。是不是很乏味?其实现在 Linux/Unix 系统中带的终端模拟器是支持动画和彩色输出的。下面,一起来看看字符界面下的动画魅力!

1 定点输出

1.1 回车符(carriage return)

在这之前,我一直下意识的以为回车符和换行符是一个意思,相信有很多人也有这种错误的理解。其实不然,回车符(\r)是把光标返回到行首,而换行符(\n)才是把光标移到下一行。尽管在 Linux 中,是采用换行符作为新行的标识,但终端模拟器中还是会响应回车符 [1]。

OK,在理解了回车符和换行符的区别后,我们考虑一种情况:当在一行结束的时候,只输出回车符,而没有换行符,会发生什么?由于没有换行,只回到了行首,所以新打印的内容会把当前行覆盖。而我们可以基于这个特性,在脚本中做一些很有意思的动画。这里我写了一个例子,可以尝试一下:

  1. #!/bin/bash
  2. spin=('\' '|' '/' '-')
  3. cnt=0
  4. while(true)
  5. do
  6. echo -n "handling $((cnt++)), please wait... ${spin[$((cnt % 4))]}"
  7. echo -n -e \\r
  8. sleep 0.2
  9. done

这里的关键就是 -n,强制 echo 不输出换行符。

1.2 光标移动(CSI码)

单纯在同一行做文章,可能还无法满足一些动画。这时候,可以采用 CSI 码来手动移动光标。CSI(private mode character) 是用来格式化终端的输出。其序列定义为 [ESC][ + N1; N2; ... S,而 ESC 可以用 \e\033 和\x1B 来表示。这里我们简单的控制一下输出:

  1. echo -en '\e[2J\e[7;40f this is a test\e[6;38f this is another teset\n\n'

这条命令首先用 \e[2J 来清屏,\e[7;40f 把光标移到到 7,40,输出内容,然后 \e[6;38f 把光标移动到 6,38,再输出内容。具体的 CSI 码可以在 wiki 找到,发挥你的想象力吧。

Linux 下有一个 tput 命令,可以设置终端的属性,若不想用 CSI 码,可以用 tput 代替,具体可以查看 tput 联机文档。

2 色彩化输出

现在输出位置可以*控制了,那就只剩下颜色了。终端中的颜色也是通过 SGR(select graphic rendition) 码来控制。SGR 是 CSI 的一个子集,其格式为 CSI + N + m。其中 N 就是指定颜色的。可以利用 SGR 码来设定字体、加粗、下划线、背景色和前景色。具体效果可以在 flogisoft 查看,做的很全。这里只举一些基本的例子。

  1. echo -e 'this is a \e[1mtest'
  2. echo -e 'this is a \e[1;31mtest'
  3. echo -e 'this is a \e[1;4;31mtest'

可以看到,有一些效果是可以叠加的。只需中间用分号隔开就可以。

3 一些有趣的例子和工具

在分析完基本的东西后,一起来看看世界各地蛋疼的人做的成果。这里不是否定他们的工作,事实上,我也是这类人。

首先看一下我刚写的,打印一个倒三角形,然后加上一些颜色:

  1. #!/bin/bash
  2. while true
  3. do
  4. for i in {20..1}
  5. do
  6. echo -en "\e[38;5;$((RANDOM % 256))m"
  7. # expand twice
  8. eval printf \' %.0s\' {1..$((41 - i))}
  9. eval printf \''#%.0s'\' {1..$((2*i -1))}
  10. echo -e "\e[0m"
  11. done
  12. echo -en "\e[20F"
  13. sleep 0.2
  14. done

这里在 for 循环中打印三角形,然后用 \e[20F 向上移动 20 行,再继续打印三角形。而\e[38;5;$((RANDOM % 256))m 会在 256 中颜色中随机选择一种颜色。来看一下具体效果:

如果你喜欢《黑客帝国》,可以安装 cmatrix(apt-get install cmatrix),体验一下字符刷屏的效果。如果你是《星球大战》的粉丝,可以运行一下这个命令 telnet towel.blinkenlights.nl,体验一下字符界面下的电影。如果你想要震撼的效果,字符界面下的分形图像,可以运行 bb(apt-get install bb)

另外,如果你想把自己的文字变的很绚丽,可以试一下 toilet,你没看错,就这名字。

  1. toilet -f term --gay this is a test

【编者注:如果想查看更多的控制码,可以 man console_codes