Как минимум пять ответов на один общий вопрос.
В зависимости от
- posix совместимый: может работать на слабых системах с общей оболочкой средами
- bash специфично: использование так называемых bashisms
и если хотите
- простой `` in line '' вопрос / ответ (общие решения)
- довольно отформатированные интерфейсы, такие как ncurses или более графические с использованием libgtk или libqt ...
- использовать мощные возможности истории readline
1. Общие решения POSIX
Вы можете использовать команду read
, за которой следует if ... then ... else
:
echo -n "Is this a good question (y/n)? "
read answer
# if echo "$answer" | grep -iq "^y" ;then
if [ "$answer" != "${answer#[Yy]}" ] ;then
echo Yes
else
echo No
fi
(Благодаря комментарию Адама Каца : заменен тест, приведенный выше, на более переносимый и исключающий одну развилку:)
POSIX, но функция с одним ключом
Но если вы не хотите, чтобы пользователь нажимал Return , вы можете написать:
( Отредактировано: Как справедливо предположил @JonathanLeffler, сохранение конфигурации stty может быть лучше, чем просто принудительно заставить их sane .)
echo -n "Is this a good question (y/n)? "
old_stty_cfg=$(stty -g)
stty raw -echo ; answer=$(head -c 1) ; stty $old_stty_cfg # Careful playing with stty
if echo "$answer" | grep -iq "^y" ;then
echo Yes
else
echo No
fi
Примечание: Это было проверено под sh , bash , ksh , dash и BusyBox !
То же самое, но явно ожидает y или n :
#/bin/sh
echo -n "Is this a good question (y/n)? "
old_stty_cfg=$(stty -g)
stty raw -echo
answer=$( while ! head -c 1 | grep -i '[ny]' ;do true ;done )
stty $old_stty_cfg
if echo "$answer" | grep -iq "^y" ;then
echo Yes
else
echo No
fi
Использование специальных инструментов
Существует множество инструментов, созданных с использованием libncurses
, libgtk
, libqt
или других графических библиотек. Например, используя whiptail
:
if whiptail --yesno "Is this a good question" 20 60 ;then
echo Yes
else
echo No
fi
В зависимости от вашей системы, вам может потребоваться заменить whiptail
другим похожим инструментом:
dialog --yesno "Is this a good question" 20 60 && echo Yes
gdialog --yesno "Is this a good question" 20 60 && echo Yes
kdialog --yesno "Is this a good question" 20 60 && echo Yes
где 20
- высота диалогового окна в количестве строк, а 60
- ширина диалогового окна. Все эти инструменты имеют почти одинаковый синтаксис .
DIALOG=whiptail
if [ -x /usr/bin/gdialog ] ;then DIALOG=gdialog ; fi
if [ -x /usr/bin/xdialog ] ;then DIALOG=xdialog ; fi
...
$DIALOG --yesno ...
2. Bash конкретные решения
Базовый в строке метод
read -p "Is this a good question (y/n)? " answer
case ${answer:0:1} in
y|Y )
echo Yes
;;
* )
echo No
;;
esac
Я предпочитаю использовать case
, чтобы я мог даже проверить yes | ja | si | oui
при необходимости ...
в строке с одной клавишей функция
В bash мы можем указать длину предполагаемого ввода для команды read
:
read -n 1 -p "Is this a good question (y/n)? " answer
В bash команда read
принимает параметр timeout , который может быть полезен.
read -t 3 -n 1 -p "Is this a good question (y/n)? " answer
[ -z "$answer" ] && answer="Yes" # if 'yes' have to be default choice
Некоторые хитрости для специализированных инструментов
Более сложные диалоговые окна, кроме простых yes - no
целей:
dialog --menu "Is this a good question" 20 60 12 y Yes n No m Maybe
Индикатор выполнения:
dialog --gauge "Filling the tank" 20 60 0 < <(
for i in {1..100};do
printf "XXX\n%d\n%(%a %b %T)T progress: %d\nXXX\n" $i -1 $i
sleep .033
done
)
Маленькая демонстрация:
#!/bin/sh
while true ;do
[ -x "$(which ${DIALOG%% *})" ] || DIALOG=dialog
DIALOG=$($DIALOG --menu "Which tool for next run?" 20 60 12 2>&1 \
whiptail "dialog boxes from shell scripts" >/dev/tty \
dialog "dialog boxes from shell with ncurses" \
gdialog "dialog boxes from shell with Gtk" \
kdialog "dialog boxes from shell with Kde" ) || exit
clear;echo "Choosed: $DIALOG."
for i in `seq 1 100`;do
date +"`printf "XXX\n%d\n%%a %%b %%T progress: %d\nXXX\n" $i $i`"
sleep .0125
done | $DIALOG --gauge "Filling the tank" 20 60 0
$DIALOG --infobox "This is a simple info box\n\nNo action required" 20 60
sleep 3
if $DIALOG --yesno "Do you like this demo?" 20 60 ;then
AnsYesNo=Yes; else AnsYesNo=No; fi
AnsInput=$($DIALOG --inputbox "A text:" 20 60 "Text here..." 2>&1 >/dev/tty)
AnsPass=$($DIALOG --passwordbox "A secret:" 20 60 "First..." 2>&1 >/dev/tty)
$DIALOG --textbox /etc/motd 20 60
AnsCkLst=$($DIALOG --checklist "Check some..." 20 60 12 \
Correct "This demo is useful" off \
Fun "This demo is nice" off \
Strong "This demo is complex" on 2>&1 >/dev/tty)
AnsRadio=$($DIALOG --radiolist "I will:" 20 60 12 \
" -1" "Downgrade this answer" off \
" 0" "Not do anything" on \
" +1" "Upgrade this anser" off 2>&1 >/dev/tty)
out="Your answers:\nLike: $AnsYesNo\nInput: $AnsInput\nSecret: $AnsPass"
$DIALOG --msgbox "$out\nAttribs: $AnsCkLst\nNote: $AnsRadio" 20 60
done
Еще образец? Взгляните на Использование whiptail для выбора USB-устройства и Селектор USB для съемной памяти: USBKeyChooser
5. Использование истории readline
* * Пример тысячу сто пятьдесят шесть: * * одна тысяча сто пятьдесят семь
#!/bin/bash
set -i
HISTFILE=~/.myscript.history
history -c
history -r
myread() {
read -e -p '> ' $1
history -s ${!1}
}
trap 'history -a;exit' 0 1 2 3 6
while myread line;do
case ${line%% *} in
exit ) break ;;
* ) echo "Doing something with '$line'" ;;
esac
done
Это создаст файл .myscript.history
в вашем каталоге $HOME
, чем вы можете использовать команды истории readline, такие как Вверх , Вниз , Ctrl + r и др.