Функция blink зависит от терминала (или эмулятора терминала). Большинство терминалов, которые вы будете использовать, принимают управляющие последовательности, документированные в ECMA-48, например, VT100-совместимые. Управляющая последовательность может
Приложения обычно используют описание терминала (terminfo или termcap). Если в описании терминала не сказано, как мигать, то и приложение не будет знать.
Если на вашем компьютере есть infocmp
(для terminfo), он покажет возможности, перечисленные в описании терминала. bash
ищет только blink
- используя имя termcap, поскольку это приложение termcap. В более общем случае terminfo может также описать, как выполнить blink, используя sgr
(которого нет в описаниях termcap).
Например, это terminfo описание vt100
:
> infocmp vt100
# Reconstructed via infocmp from file: /usr/local/ncurses/share/terminfo/v/vt100
vt100|vt100-am|dec vt100 (w/advanced video),
am, mc5i, msgr, xenl, xon,
cols#80, it#8, lines#24, vt#3,
acsc=``aaffggjjkkllmmnnooppqqrrssttuuvvwwxxyyzz{{||}}~~,
bel=^G, blink=\E[5m$<2>, bold=\E[1m$<2>,
clear=\E[H\E[J$<50>, cr=^M, csr=\E[%i%p1%d;%p2%dr,
cub=\E[%p1%dD, cub1=^H, cud=\E[%p1%dB, cud1=^J,
cuf=\E[%p1%dC, cuf1=\E[C$<2>,
cup=\E[%i%p1%d;%p2%dH$<5>, cuu=\E[%p1%dA,
cuu1=\E[A$<2>, ed=\E[J$<50>, el=\E[K$<3>, el1=\E[1K$<3>,
enacs=\E(B\E)0, home=\E[H, ht=^I, hts=\EH, ind=^J, ka1=\EOq,
ka3=\EOs, kb2=\EOr, kbs=^H, kc1=\EOp, kc3=\EOn, kcub1=\EOD,
kcud1=\EOB, kcuf1=\EOC, kcuu1=\EOA, kent=\EOM, kf0=\EOy,
kf1=\EOP, kf10=\EOx, kf2=\EOQ, kf3=\EOR, kf4=\EOS, kf5=\EOt,
kf6=\EOu, kf7=\EOv, kf8=\EOl, kf9=\EOw, lf1=pf1, lf2=pf2,
lf3=pf3, lf4=pf4, mc0=\E[0i, mc4=\E[4i, mc5=\E[5i, rc=\E8,
rev=\E[7m$<2>, ri=\EM$<5>, rmacs=^O, rmam=\E[?7l,
rmkx=\E[?1l\E>, rmso=\E[m$<2>, rmul=\E[m$<2>,
rs2=\E>\E[?3l\E[?4l\E[?5l\E[?7h\E[?8h, sc=\E7,
sgr=\E[0%?%p1%p6%|%t;1%;%?%p2%t;4%;%?%p1%p3%|%t;7%;%?%p4%t;5%;m%?%p9%t\016%e\017%;$<2>,
sgr0=\E[m\017$<2>, smacs=^N, smam=\E[?7h, smkx=\E[?1h\E=,
smso=\E[7m$<2>, smul=\E[4m$<2>, tbc=\E[3g,
Соответствующий termcap -
> infocmp -Cr vt100
# Reconstructed via infocmp from file: /usr/local/ncurses/share/terminfo/v/vt100
vt100|vt100-am|dec vt100 (w/advanced video):\
:5i:am:bs:ms:xn:xo:\
:co#80:it#8:li#24:vt#3:\
:@8=\EOM:DO=\E[%dB:K1=\EOq:K2=\EOr:K3=\EOs:K4=\EOp:K5=\EOn:\
:LE=\E[%dD:RA=\E[?7l:RI=\E[%dC:SA=\E[?7h:UP=\E[%dA:\
:ac=``aaffggjjkkllmmnnooppqqrrssttuuvvwwxxyyzz{{||}}~~:\
:ae=^O:as=^N:bl=^G:cb=\E[1K:cd=\E[J:ce=\E[K:cl=\E[H\E[J:\
:cm=\E[%i%d;%dH:cr=^M:cs=\E[%i%d;%dr:ct=\E[3g:do=^J:\
:eA=\E(B\E)0:ho=\E[H:k0=\EOy:k1=\EOP:k2=\EOQ:k3=\EOR:\
:k4=\EOS:k5=\EOt:k6=\EOu:k7=\EOv:k8=\EOl:k9=\EOw:k;=\EOx:\
:kb=^H:kd=\EOB:ke=\E[?1l\E>:kl=\EOD:kr=\EOC:ks=\E[?1h\E=:\
:ku=\EOA:l1=pf1:l2=pf2:l3=pf3:l4=pf4:le=^H:mb=\E[5m:\
:md=\E[1m:me=\E[0m:mr=\E[7m:nd=\E[C:pf=\E[4i:po=\E[5i:\
:ps=\E[0i:rc=\E8:rs=\E>\E[?3l\E[?4l\E[?5l\E[?7h\E[?8h:\
:..sa=\E[0%?%p1%p6%|%t;1%;%?%p2%t;4%;%?%p1%p3%|%t;7%;%?%p4%t;5%;m%?%p9%t\016%e\017%;$<2>:\
:sc=\E7:se=\E[m:sf=^J:so=\E[7m:sr=\EM:st=\EH:ta=^I:ue=\E[m:\
:up=\E[A:us=\E[4m:
(Имя termcap для blink
- mb
, что можно увидеть в описании).
Итак... если вы не видите мигающий текст, это может быть (а) сам терминал или (б) описание терминала.
Дополнительная литература:
inner.sh
может быть:
printf '%s\n' "$1" >&3
и в outer.sh
вы можете сделать:
{ inner=$(middle "$@" 3>&1 >&4 4>&-); } 4>&1
printf '%s\n' "$inner"
Внутренний текст передается по каналу (в подстановке команд )и сохраняется в переменной оболочки. Это предполагает, что middle
не закрывает этот fd 3 перед вызовомinner.sh
(, но нет причин для этого ).
Пояснение:
{... } 4>&1
. В этой группе команд изначально оба fd 1 и 4 указывают на исходный стандартный вывод. IOW, мы сделали копию stdout external.sh на fd 4, чтобы иметь возможность восстановить его внутри подстановки команды $(...)
. Внутри этой подстановки команд stdout (fd 1 )представляет собой канал. В этом смысл замены команды. Он хочет получить вывод команды. Но здесь нам не нужен стандартный вывод middle
, нам нужно то, что он (или, точнее, его дочерний элементinner.sh
)записывает в fd 3, поэтому:middle 3>&1 >&4 4>&-
:для middle
мы делаем fd 3 каналом cmdsubst, чтобы то, что inner.sh
пишет там, попадало в $inner
, и мы восстанавливаем стандартный вывод middle
на сохраненный исходный стандартный вывод. на fd 4. Мы закрываем fd 4 после того, как он выполнил свою задачу, так как middle
не нужно ничего с ним делать.