ユーザーが矢印キーを使用してハイライトカーソルを移動し、Enterキーを押して1つを選択する3つのオプションを表示するシェルスクリプトでメニューを作成する方法
dialog は、達成しようとしていることのための優れたツールです。簡単な3つの選択肢のメニューの例を次に示します。
dialog --menu "Choose one:" 10 30 3 \
1 Red \
2 Green \
3 Blue
構文は次のとおりです。
dialog --menu <text> <height> <width> <menu-height> [<tag><item>]
選択範囲はstderr
に送信されます。 3色を使用したサンプルスクリプトを次に示します。
#!/bin/bash
TMPFILE=$(mktemp)
dialog --menu "Choose one:" 10 30 3 \
1 Red \
2 Green \
3 Blue 2>$TMPFILE
RESULT=$(cat $TMPFILE)
case $RESULT in
1) echo "Red";;
2) echo "Green";;
3) echo "Blue";;
*) echo "Unknown color";;
esac
rm $TMPFILE
Debianでは、dialog
を 同じ名前のパッケージ でインストールできます。
以下は、select_option
関数の形式の純粋なbash
スクリプトソリューションで、 ANSIエスケープシーケンス と組み込みread
のみに依存しています。
OSX上のBash 4.2.45で動作します。私が知っているすべての環境で同じようにうまく機能しない可能性のあるファンキーな部分は、get_cursor_row()
、key_input()
(上/下キーを検出するため)およびcursor_to()
です。関数。
#!/usr/bin/env bash
# Renders a text based list of options that can be selected by the
# user using up, down and enter keys and returns the chosen option.
#
# Arguments : list of options, maximum of 256
# "opt1" "opt2" ...
# Return value: selected index (0 for opt1, 1 for opt2 ...)
function select_option {
# little helpers for terminal print control and key input
ESC=$( printf "\033")
cursor_blink_on() { printf "$ESC[?25h"; }
cursor_blink_off() { printf "$ESC[?25l"; }
cursor_to() { printf "$ESC[$1;${2:-1}H"; }
print_option() { printf " $1 "; }
print_selected() { printf " $ESC[7m $1 $ESC[27m"; }
get_cursor_row() { IFS=';' read -sdR -p $'\E[6n' ROW COL; echo ${ROW#*[}; }
key_input() { read -s -n3 key 2>/dev/null >&2
if [[ $key = $ESC[A ]]; then echo up; fi
if [[ $key = $ESC[B ]]; then echo down; fi
if [[ $key = "" ]]; then echo enter; fi; }
# initially print empty new lines (scroll down if at bottom of screen)
for opt; do printf "\n"; done
# determine current screen position for overwriting the options
local lastrow=`get_cursor_row`
local startrow=$(($lastrow - $#))
# ensure cursor and input echoing back on upon a ctrl+c during read -s
trap "cursor_blink_on; stty echo; printf '\n'; exit" 2
cursor_blink_off
local selected=0
while true; do
# print options by overwriting the last lines
local idx=0
for opt; do
cursor_to $(($startrow + $idx))
if [ $idx -eq $selected ]; then
print_selected "$opt"
else
print_option "$opt"
fi
((idx++))
done
# user key control
case `key_input` in
enter) break;;
up) ((selected--));
if [ $selected -lt 0 ]; then selected=$(($# - 1)); fi;;
down) ((selected++));
if [ $selected -ge $# ]; then selected=0; fi;;
esac
done
# cursor position back to normal
cursor_to $lastrow
printf "\n"
cursor_blink_on
return $selected
}
次に使用例を示します。
echo "Select one option using up/down keys and enter to confirm:"
echo
options=("one" "two" "three")
select_option "${options[@]}"
choice=$?
echo "Choosen index = $choice"
echo " value = ${options[$choice]}"
出力は以下のようになり、現在選択されているオプションは反転ansiカラーリングを使用して強調表示されます(ここではマークダウンで伝えるのは困難です)。これは、必要に応じてprint_selected()
関数で調整できます。
Select one option using up/down keys and enter to confirm:
[one]
two
three
更新:以下は、上記のselect_opt
関数をラップしてcase
ステートメントで簡単に使用できるようにする小さな拡張select_option
です。
function select_opt {
select_option "$@" 1>&2
local result=$?
echo $result
return $result
}
3つのリテラルオプションの使用例:
case `select_opt "Yes" "No" "Cancel"` in
0) echo "selected Yes";;
1) echo "selected No";;
2) echo "selected Cancel";;
esac
いくつかの既知のエントリがある場合は混合して(この場合は[はい]と[いいえ])、ワイルドカードの場合の終了コード$?
を利用することもできます。
options=("Yes" "No" "${array[@]}") # join arrays to add some variable array
case `select_opt "${options[@]}"` in
0) echo "selected Yes";;
1) echo "selected No";;
*) echo "selected ${options[$?]}";;
esac