Клавиша со стрелкой / Войти в меню

12

Как создать меню в сценарии оболочки, которое будет отображать 3 параметра, которые пользователь будет использовать клавиши со стрелками для перемещения курсора выделения и нажмите Enter, чтобы выбрать один?

Mrplow911
источник
Я думаю, что вам не повезло, WRT для функциональности клавиш со стрелками и выделения в чистом сценарии оболочки (вы можете сделать последнее с помощью tput, но я думаю, что первое невозможно), но вы можете создавать простые меню в bash с помощью select: tldp.org/LDP/Bash-Beginners-Guide/html/sect_09_06.html
Златовласка
Вы имеете в виду меню с графическим интерфейсом (используя что-то вроде [zenity] (Бен Браудер) или текстовое меню с использованием чего-то вроде ncurses ?
terdon
Я пытаюсь создать меню, похожее на то, которое вы получаете, если вам нужно было выбрать вариант загрузки для Windows («безопасный режим», «нормальный» и т. Д.)
Mrplow911
1
Существует dialogпакет, который создает базовые интерфейсы терминала faux-GUI в скриптах.
HalosGhost
@HalosGhost Вы знаете какие-нибудь примеры этого?
Mrplow911

Ответы:

10

Диалог - отличный инструмент для того, чего вы пытаетесь достичь. Вот пример простого меню с 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помощью пакета одного и того же имени .

Джон У. Смит
источник
22

Вот чистое bashрешение сценария в форме select_optionфункции, основанное исключительно на escape-последовательностях ANSI и встроенном read.

Работает на Bash 4.2.45 на OSX. Фанки части , которые могут не работать одинаково хорошо во всех средах , от всего я знаю , являются 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обертывающее вышеуказанную select_optionфункцию, чтобы упростить его использование в caseвыражении:

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
Александр Климетчек
источник
1
Это красиво и удивительно ; Большое спасибо за то, что поделились! Это ваше собственное изначально? Есть ли онлайн-репозиторий для клонирования / разветвления? Единственное, что я мог найти в контроле версий, было на GitHub в Gist Стивенма (с добавлением редактирования строк), который указывает на это, смеется. Работаю над своими собственными изменениями (в Gist, но планирую сделать репо) здесь, хотя мне все еще нужно обновиться с последними изменениями.
l3l_aze
1
Я использовал это в не публичном коде. Собрал его из различных фрагментов, найденных в сети :-)
Александр Климетчек,
Ух ты; хорошо сделано. Я начал репо со своими модификациями по адресу https://github.com/l3laze/sind . Пока что самые большие различия - улучшенная обработка ввода и добавление строки заголовка. Я надеюсь добавить однострочное и многострочное редактирование, но пока что ничего не сделало для тех, кто не рассматривал этот код
l3l_aze