Tecla de seta / menu Enter

8

Como criar um menu em um script de shell que exibirá 3 opções que um usuário usará as teclas de setas para mover o cursor de destaque e pressionar Enter para selecionar um?

    
por Mrplow911 25.07.2014 / 16:37

2 respostas

6
O diálogo é uma ótima ferramenta para o que você está tentando alcançar. Aqui está o exemplo de um simples menu de 3 opções:

dialog --menu "Choose one:" 10 30 3 \
    1 Red \
    2 Green \
    3 Blue

A sintaxe é a seguinte:

dialog --menu <text> <height> <width> <menu-height> [<tag><item>]

A seleção será enviada para stderr . Aqui está um exemplo de script usando 3 cores.

#!/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

No Debian, você pode instalar o dialog através do pacote com o mesmo nome .

    
por 25.07.2014 / 18:27
9

Aqui está uma solução de script bash pura na forma da função select_option , contando apenas com seqüências de escape ANSI e o read incorporado.

Funciona no Bash 4.2.45 no OSX. As partes engraçadas que podem não funcionar igualmente bem em todos os ambientes, a partir do que eu sei, são as funções get_cursor_row() , key_input() (para detectar as teclas para cima / baixo) e 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 "3")
    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
}

Aqui está um exemplo de uso:

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]}"

A saída é semelhante a abaixo, com a opção atualmente selecionada destacada usando a coloração ansi inversa (difícil de transmitir aqui no markdown). Isso pode ser adaptado na função print_selected() , se desejado.

Select one option using up/down keys and enter to confirm:

  [one] 
   two 
   three 

Atualização: Aqui está uma pequena extensão select_opt envolvendo a função select_option acima para facilitar o uso em uma instrução case :

function select_opt {
    select_option "$@" 1>&2
    local result=$?
    echo $result
    return $result
}

Exemplo de uso com 3 opções literais:

case 'select_opt "Yes" "No" "Cancel"' in
    0) echo "selected Yes";;
    1) echo "selected No";;
    2) echo "selected Cancel";;
esac

Você também pode misturar se houver algumas entradas conhecidas (Sim e Não neste caso) e aproveitar o código de saída $? para o caso curinga:

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
    
por 06.01.2018 / 08:04