it-roy-ru.com

Как создать меню выбора в сценарии оболочки?

Я создаю простой скрипт bash и хочу создать в нем меню выбора, например:

$./script

echo "Choose your option:"

1) Option 1  
2) Option 2  
3) Option 3  
4) Quit  

И по выбору пользователя я хочу, чтобы выполнялись разные действия. Я новичок в bash Shell, я искал ответы в Интернете, но ничего конкретного не получил.

122
Daniel Rodrigues
#!/bin/bash
# Bash Menu Script Example

PS3='Please enter your choice: '
options=("Option 1" "Option 2" "Option 3" "Quit")
select opt in "${options[@]}"
do
    case $opt in
        "Option 1")
            echo "you chose choice 1"
            ;;
        "Option 2")
            echo "you chose choice 2"
            ;;
        "Option 3")
            echo "you chose choice $REPLY which is $opt"
            ;;
        "Quit")
            break
            ;;
        *) echo "invalid option $REPLY";;
    esac
done

Добавьте операторы break везде, где вам нужен цикл select для выхода. Если break не выполняется, оператор select зацикливается, и меню отображается повторно.

В третий вариант я включил переменные, которые устанавливаются оператором select, чтобы продемонстрировать, что у вас есть доступ к этим значениям. Если вы выберете его, он выведет:

you chose choice 3 which is Option 3

Вы можете видеть, что $REPLY содержит строку, введенную вами в приглашении. Он используется в качестве индекса в массиве ${options[@]}, как если бы массив был основан на 1. Переменная $opt содержит строку из этого индекса в массиве.

Обратите внимание, что выбор может быть простым списком непосредственно в операторе select, например так:

select opt in foo bar baz 'multi Word choice'

но вы не можете поместить такой список в скалярную переменную из-за пробелов в одном из вариантов.

Вы также можете использовать глобирование файлов, если вы выбираете среди файлов:

select file in *.tar.gz
137
Dennis Williamson

Не новый ответ как таковой, но так как нет никакого принятого ответа, вот несколько советов и приемов кодирования, как для выбора, так и для zenity:

title="Select example"
Prompt="Pick an option:"
options=("A" "B" "C")

echo "$title"
PS3="$Prompt "
select opt in "${options[@]}" "Quit"; do 

    case "$REPLY" in

    1 ) echo "You picked $opt which is option $REPLY";;
    2 ) echo "You picked $opt which is option $REPLY";;
    3 ) echo "You picked $opt which is option $REPLY";;

    $(( ${#options[@]}+1 )) ) echo "Goodbye!"; break;;
    *) echo "Invalid option. Try another one.";continue;;

    esac

done

while opt=$(zenity --title="$title" --text="$Prompt" --list \
                   --column="Options" "${options[@]}"); do

    case "$opt" in
    "${options[0]}" ) zenity --info --text="You picked $opt, option 1";;
    "${options[1]}" ) zenity --info --text="You picked $opt, option 2";;
    "${options[2]}" ) zenity --info --text="You picked $opt, option 3";;
    *) zenity --error --text="Invalid option. Try another one.";;
    esac

done

Стоит отметить:

  • Оба будут зациклены, пока пользователь явно не выберет Выход (или Отмена для zenity). Это хороший подход для интерактивных меню сценариев: после выбора и выполнения действия снова открывается меню для другого выбора. Если выбор подразумевается только один раз, просто используйте break после esac (подход zenity может быть дополнительно сокращен)

  • Оба case основаны на индексах, а не на значениях. Я думаю, что это легче кодировать и поддерживать

  • Массив также используется для подхода zenity.

  • Опция "Выйти" не входит в число первоначальных, оригинальных опций. Он добавляется при необходимости, поэтому ваш массив остается чистым. В конце концов, "Выход" не нужен для zenity, пользователь может просто нажать "Отмена" (или закрыть окно), чтобы выйти. Обратите внимание, как оба используют один и тот же нетронутый массив параметров.

  • Варианты PS3 и REPLY можно не переименовать. select жестко запрограммирован для их использования. Все остальные переменные в скрипте (opt, options, Prompt, title) могут иметь любые имена, которые вы хотите, при условии, что вы делаете корректировки

54
MestreLion

Используя dialog , команда будет выглядеть так:

dialog --clear --backtitle "Backtitle here" --title "Заголовок здесь" --menu "Выберите один из следующих параметров:" 15 40 4\
 1 "Вариант 1"\
 2 "Вариант 2"\
 3 "Вариант 3" 

enter image description here

Поместив это в скрипт:

#!/bin/bash

HEIGHT=15
WIDTH=40
CHOICE_HEIGHT=4
BACKTITLE="Backtitle here"
TITLE="Title here"
MENU="Choose one of the following options:"

OPTIONS=(1 "Option 1"
         2 "Option 2"
         3 "Option 3")

CHOICE=$(dialog --clear \
                --backtitle "$BACKTITLE" \
                --title "$TITLE" \
                --menu "$MENU" \
                $HEIGHT $WIDTH $CHOICE_HEIGHT \
                "${OPTIONS[@]}" \
                2>&1 >/dev/tty)

clear
case $CHOICE in
        1)
            echo "You chose Option 1"
            ;;
        2)
            echo "You chose Option 2"
            ;;
        3)
            echo "You chose Option 3"
            ;;
esac
52
Alaa Ali

Вы можете использовать этот простой скрипт для создания опций

#!/bin/bash 
 echo "выберите операцию ************" 
 echo "1) операция 1" 
 echo "2) операция 2 "
 эхо" 3) операция 3 "
 эхо" 4) операция 4 " 
прочитайте n регистр $ n в 1) эхо "Вы выбрали вариант 1" ;; 2) эхо "вы выбрали вариант 2" ;; 3) эхо "Вы выбрали опцию 3" ;; 4) эхо "Вы выбрали опцию 4" ;; *) Эхо "неверная опция" ;; Esac
14
jibin

Так как это нацелено на Ubuntu, вы должны использовать любой бэкэнд debconf, настроенный для использования. Вы можете узнать бэкэнд debconf с помощью:

Sudo -s "echo get debconf/frontend | debconf-communicate"

Если он говорит "диалог", то он, вероятно, использует whiptail или dialog. На Lucid это whiptail.

Если это не поможет, используйте bash "select", как объяснил Деннис Уильямсон.

7
Li Lo
#!/bin/sh 
 show_menu () {
 normal = `echo"\033 [m "` 
 menu = `echo"\033 [36m "` #Blue 
 number = `echo"\033 [33m "` #yellow 
 bgred = `echo"\033 [41m "` 
 fgred = `echo"\033 [31m "` 
 printf "\ n $ {menu} ************************************** ******* $ {normal}\n "
 printf" $ {menu} ** $ {number} 1) $ {menu} Смонтировать Dropbox $ {normal}\n "
 printf "$ {menu} ** $ {number} 2) $ {menu} Монтирование USB 500 Gig Drive $ {normal}\n" 
 printf "$ {menu} ** $ {number} 3) $ {menu} Перезапустите Apache $ {normal}\n "
 printf" $ {menu} ** $ {number} 4) $ {menu} ssh Frost Tomcat Server $ {normal}\n "
 printf "$ {menu} ** $ {number} 5) $ {menu} Некоторые другие команды $ {normal}\n" 
 printf "$ {menu} ********* ************************************ $ {normal}\n "
 printf" Пожалуйста, введите опцию меню и введите или $ {fgred} x для выхода. $ {Normal} "
 Read opt 
} 
 
 Option_picked () {
 msgcolor = `echo"\033 [01; 31m "` # жирный красный 
 normal = `echo"\033 [00; 00m "` # normal white 
 message = $ {@: - "$ {normal} Ошибка: сообщение не передано"} 
 printf "$ {msgcolor } $ {message} $ {normal}\n "
} 
 
 clear 
 show_menu 
 while [$ opt! = ''] 
 сделать 
 if [$ opt = '']; затем 
 exit; 
 else 
 case $ opt in 
 1) clear; 
 option_picked "Option 1 Picked"; 
 printf "Sudo mount/dev/sdh1/mnt/DropBox /; # 3 терабайта"; 
 Show_menu; 
 ;; 
 2) очистить; 
 Option_picked "Option 2 Picked "; 
 Printf" Sudo mount/dev/sdi1/mnt/usbDrive; # Дисковод на 500 гигов "; 
 Show_menu; 
 ;; 
 3) clear; 
 option_picked "Option 3 Picked"; 
 printf "Sudo service Apache2 restart"; 
 show_menu; 
 ;; 
 4) clear; 
 option_picked "Option 4 Picked"; 
 printf "ssh lmesser @ -p 2010"; 
 show_menu; 
 ;; 
 x) выход; 
 ;; 
\n) выход; 
 ;; 
 *) очистить; 
 option_picked "Выбрать опцию из меню "; 
 show_menu; 
 ;; 
 esac 
 fi 
 сделано
7
Alex Lucard

У меня есть еще один вариант, который представляет собой смесь этих ответов, но что делает его приятным, так это то, что вам нужно всего лишь нажать одну клавишу, а затем сценарий продолжается благодаря опции чтения -n. В этом примере мы предлагаем выключить, перезагрузить или просто выйти из скрипта, используя ANS в качестве нашей переменной, и пользователю нужно только нажать E, R или S. Я также устанавливаю значение по умолчанию для выхода, поэтому, если нажата клавиша enter, то скрипт выйдет.

read -n 1 -p "Would you like to exit, reboot, or shutdown? (E/r/s) " ans;

case $ans in
    r|R)
        Sudo reboot;;
    s|S)
        Sudo poweroff;;
    *)
        exit;;
esac
6
HarlemSquirrel

Я использовал Zenity, который, кажется, всегда есть в Ubuntu, работает очень хорошо и имеет много возможностей. Это эскиз возможного меню:

#! /bin/bash

selection=$(zenity --list "Option 1" "Option 2" "Option 3" --column="" --text="Text above column(s)" --title="My menu")

case "$selection" in
"Option 1")zenity --info --text="Do something here for No1";;
"Option 2")zenity --info --text="Do something here for No2";;
"Option 3")zenity --info --text="Do something here for No3";;
esac
6
LazyEchidna

Bash необычное меню

Попробуйте сначала, а затем посетите мою страницу для подробного описания ... Нет необходимости во внешних библиотеках или программах, таких как диалог или zenity ...

#/bin/bash
# by oToGamez
# www.pro-toolz.net

      E='echo -e';e='echo -en';trap "R;exit" 2
    ESC=$( $e "\e")
   TPUT(){ $e "\e[${1};${2}H";}
  CLEAR(){ $e "\ec";}
  CIVIS(){ $e "\e[?25l";}
   DRAW(){ $e "\e%@\e(0";}
  WRITE(){ $e "\e(B";}
   MARK(){ $e "\e[7m";}
 UNMARK(){ $e "\e[27m";}
      R(){ CLEAR ;stty sane;$e "\ec\e[37;44m\e[J";};
   HEAD(){ DRAW
           for each in $(seq 1 13);do
           $E "   x                                          x"
           done
           WRITE;MARK;TPUT 1 5
           $E "BASH SELECTION MENU                       ";UNMARK;}
           i=0; CLEAR; CIVIS;NULL=/dev/null
   FOOT(){ MARK;TPUT 13 5
           printf "ENTER - SELECT,NEXT                       ";UNMARK;}
  ARROW(){ read -s -n3 key 2>/dev/null >&2
           if [[ $key = $ESC[A ]];then echo up;fi
           if [[ $key = $ESC[B ]];then echo dn;fi;}
     M0(){ TPUT  4 20; $e "Login info";}
     M1(){ TPUT  5 20; $e "Network";}
     M2(){ TPUT  6 20; $e "Disk";}
     M3(){ TPUT  7 20; $e "Routing";}
     M4(){ TPUT  8 20; $e "Time";}
     M5(){ TPUT  9 20; $e "ABOUT  ";}
     M6(){ TPUT 10 20; $e "EXIT   ";}
      LM=6
   MENU(){ for each in $(seq 0 $LM);do M${each};done;}
    POS(){ if [[ $cur == up ]];then ((i--));fi
           if [[ $cur == dn ]];then ((i++));fi
           if [[ $i -lt 0   ]];then i=$LM;fi
           if [[ $i -gt $LM ]];then i=0;fi;}
REFRESH(){ after=$((i+1)); before=$((i-1))
           if [[ $before -lt 0  ]];then before=$LM;fi
           if [[ $after -gt $LM ]];then after=0;fi
           if [[ $j -lt $i      ]];then UNMARK;M$before;else UNMARK;M$after;fi
           if [[ $after -eq 0 ]] || [ $before -eq $LM ];then
           UNMARK; M$before; M$after;fi;j=$i;UNMARK;M$before;M$after;}
   INIT(){ R;HEAD;FOOT;MENU;}
     SC(){ REFRESH;MARK;$S;$b;cur=`ARROW`;}
     ES(){ MARK;$e "ENTER = main menu ";$b;read;INIT;};INIT
  while [[ "$O" != " " ]]; do case $i in
        0) S=M0;SC;if [[ $cur == "" ]];then R;$e "\n$(w        )\n";ES;fi;;
        1) S=M1;SC;if [[ $cur == "" ]];then R;$e "\n$(ifconfig )\n";ES;fi;;
        2) S=M2;SC;if [[ $cur == "" ]];then R;$e "\n$(df -h    )\n";ES;fi;;
        3) S=M3;SC;if [[ $cur == "" ]];then R;$e "\n$(route -n )\n";ES;fi;;
        4) S=M4;SC;if [[ $cur == "" ]];then R;$e "\n$(date     )\n";ES;fi;;
        5) S=M5;SC;if [[ $cur == "" ]];then R;$e "\n$($e by oTo)\n";ES;fi;;
        6) S=M6;SC;if [[ $cur == "" ]];then R;exit 0;fi;;
 esac;POS;done
5
user360154

Уже есть тот же вопрос в serverfault ответил. Решение там использует whiptail .

2
txwikinger