बैश स्क्रिप्ट में बहु-चयन मेनू


28

मैं बैश नौसिखिया हूं, लेकिन मैं एक स्क्रिप्ट बनाना चाहूंगा जिसमें मैं उपयोगकर्ता को विकल्पों की सूची से कई विकल्पों का चयन करने की अनुमति दूंगा।

अनिवार्य रूप से मैं जो चाहूंगा वह नीचे दिए गए उदाहरण के समान है:

       #!/bin/bash
       OPTIONS="Hello Quit"
       select opt in $OPTIONS; do
           if [ "$opt" = "Quit" ]; then
            echo done
            exit
           elif [ "$opt" = "Hello" ]; then
            echo Hello World
           else
            clear
            echo bad option
           fi
       done

( Http://www.faqs.org/docs/Linux-HOWTO/Bash-Prog-Intro-HOWTO.html#ss9.1 से साभार )

हालाँकि मेरी स्क्रिप्ट में अधिक विकल्प होंगे, और मैं गुणकों को चयनित करने की अनुमति देना चाहूंगा। तो कुछ इस तरह से:

1) विकल्प 1
2) विकल्प 2
3) विकल्प 3
4) विकल्प 4
5) हो गया

जिन लोगों का उन्होंने चयन किया है, उन पर प्रतिक्रिया देना भी बहुत अच्छा होगा, जैसे कि पहले से चुने गए लोगों के बगल में प्लस चिन्ह। उदाहरण के लिए, यदि आप "1" चुनते हैं, तो मैं पृष्ठ को साफ़ और पुनर्मुद्रण करना चाहूंगा:

1) Option 1 +
2) Option 2
3) Option 3
4) Option 4
5) Done

यदि आप "3" का चयन करते हैं:

1) Option 1 +
2) Option 2
3) Option 3 +
4) Option 4
5) Done

इसके अलावा, अगर वे फिर से चुने गए (1) मैं इसे "रद्द" विकल्प पसंद करूंगा:

1) Option 1
2) Option 2
3) Option 3 +
4) Option 4
5) Done

और अंत में जब डन दबाया जाता है तो मैं उन लोगों की एक सूची चाहूंगा जिन्हें प्रोग्राम से बाहर होने से पहले प्रदर्शित किया जाना था, जैसे कि वर्तमान स्थिति है:

1) Option 1
2) Option 2 +
3) Option 3 + 
4) Option 4 +
5) Done

5 दबाकर प्रिंट करना चाहिए:

Option 2, Option 3, Option 4

... और स्क्रिप्ट समाप्त।

तो मेरा सवाल है - क्या यह संभव है, और यदि कोई ऐसा है जो कोड नमूना प्रदान करने में सक्षम है?

कोई भी सलाह काफी सराही जाएगी।

जवाबों:


35

मुझे लगता है कि आपको डायलॉग या व्हिपटेल पर एक नज़र डालनी चाहिए

संवाद बॉक्स

संपादित करें:

आपके प्रश्न के विकल्पों का उपयोग करके एक उदाहरण स्क्रिप्ट दी गई है:

#!/bin/bash
cmd=(dialog --separate-output --checklist "Select options:" 22 76 16)
options=(1 "Option 1" off    # any option can be set to default to "on"
         2 "Option 2" off
         3 "Option 3" off
         4 "Option 4" off)
choices=$("${cmd[@]}" "${options[@]}" 2>&1 >/dev/tty)
clear
for choice in $choices
do
    case $choice in
        1)
            echo "First Option"
            ;;
        2)
            echo "Second Option"
            ;;
        3)
            echo "Third Option"
            ;;
        4)
            echo "Fourth Option"
            ;;
    esac
done

उसके लिए धन्यवाद। जितना मैंने आशा की थी उससे अधिक जटिल लगता है, लेकिन मैं इसे बाहर की जाँच करूंगा :-)
user38939

@ am2605: मेरा संपादन देखें मैंने एक उदाहरण स्क्रिप्ट जोड़ी।
अगली सूचना तक रोक दिया गया।

3
यह केवल तब तक जटिल दिखता है जब तक आप इसे एक या दो बार उपयोग नहीं करते हैं, फिर आप कभी भी कुछ और उपयोग नहीं करेंगे ...
क्रिस एस

27

यदि आपको लगता whiptailहै कि यह जटिल है, तो यह एक बैश-केवल कोड जाता है जो वास्तव में आप चाहते हैं। यह छोटी (~ 20 पंक्तियाँ) है, लेकिन एक भिखारी के लिए थोड़ी सी गूढ़ है। चेक किए गए विकल्पों के लिए "+" दिखाने के अलावा, यह प्रत्येक उपयोगकर्ता कार्रवाई ("अमान्य विकल्प", "विकल्प X की जाँच की गई" / अनियंत्रित आदि) के लिए प्रतिक्रिया भी प्रदान करता है।

उसने कहा, तुम वहाँ जाओ!

आशा है कि आप आनंद लेंगे ... इसे बनाने के लिए काफी मजेदार चुनौती थी :)

#!/bin/bash

# customize with your own.
options=("AAA" "BBB" "CCC" "DDD")

menu() {
    echo "Avaliable options:"
    for i in ${!options[@]}; do 
        printf "%3d%s) %s\n" $((i+1)) "${choices[i]:- }" "${options[i]}"
    done
    if [[ "$msg" ]]; then echo "$msg"; fi
}

prompt="Check an option (again to uncheck, ENTER when done): "
while menu && read -rp "$prompt" num && [[ "$num" ]]; do
    [[ "$num" != *[![:digit:]]* ]] &&
    (( num > 0 && num <= ${#options[@]} )) ||
    { msg="Invalid option: $num"; continue; }
    ((num--)); msg="${options[num]} was ${choices[num]:+un}checked"
    [[ "${choices[num]}" ]] && choices[num]="" || choices[num]="+"
done

printf "You selected"; msg=" nothing"
for i in ${!options[@]}; do 
    [[ "${choices[i]}" ]] && { printf " %s" "${options[i]}"; msg=""; }
done
echo "$msg"

बहुत बढ़िया! बहुत बढ़िया!
डैनियल

4
यह एक थोड़ा गूढ़ है, लेकिन मुझे आपके जटिल ब्रेस विस्तार और गतिशील सरणियों के उपयोग से प्यार है। मुझे ऐसा करने में थोड़ा समय लगा कि मैं सबकुछ पढ़ सकूं क्योंकि ऐसा होता है लेकिन मुझे यह पसंद है। मुझे इस तथ्य से भी प्यार है कि आपने प्रिंटफ () फ़ंक्शन का उपयोग किया है। मुझे कई ऐसे नहीं मिले हैं जो इसके बारे में जानते हैं। बहुत आसान है अगर किसी को सी में कोडिंग करने के लिए उपयोग किया जाता है
योकेई

1
अगर कोई एक बार में कई विकल्प (स्पेस अलग) का चयन करने में सक्षम होना चाहता है:while menu && read -rp "$prompt" nums && [[ "$nums" ]]; do while read num; do ... done < <(echo $nums |sed "s/ /\n/g") done
TAAPSogeking

1
यह एक स्क्रिप्ट को विकसित करने में वास्तव में उपयोगी था, जिसका उपयोग कई अन्य लोगों द्वारा किया जाता है, जिनके पास व्हिपटेल या अन्य पैकेज तक पहुंच नहीं है, क्योंकि वे git bashविंडोज़ के लिए उपयोग कर रहे हैं !
डॉ। इवोल

5

यहां ठीक वैसा ही करने का एक तरीका है जैसा आप केवल बाहरी सुविधाओं के साथ बैश सुविधाओं का उपयोग करना चाहते हैं। यह वर्तमान चयनों को चिह्नित करता है और आपको उन्हें टॉगल करने की अनुमति देता है।

#!/bin/bash
# Purpose: Demonstrate usage of select and case with toggleable flags to indicate choices
# 2013-05-10 - Dennis Williamson

choice () {
    local choice=$1
    if [[ ${opts[choice]} ]] # toggle
    then
        opts[choice]=
    else
        opts[choice]=+
    fi
}

PS3='Please enter your choice: '
while :
do
    clear
    options=("Option 1 ${opts[1]}" "Option 2 ${opts[2]}" "Option 3 ${opts[3]}" "Done")
    select opt in "${options[@]}"
    do
        case $opt in
            "Option 1 ${opts[1]}")
                choice 1
                break
                ;;
            "Option 2 ${opts[2]}")
                choice 2
                break
                ;;
            "Option 3 ${opts[3]}")
                choice 3
                break
                ;;
            "Option 4 ${opts[4]}")
                choice 4
                break
                ;;
            "Done")
                break 2
                ;;
            *) printf '%s\n' 'invalid option';;
        esac
    done
done

printf '%s\n' 'Options chosen:'
for opt in "${!opts[@]}"
do
    if [[ ${opts[opt]} ]]
    then
        printf '%s\n' "Option $opt"
    fi
done

Ksh के लिए, फ़ंक्शन की पहली दो पंक्तियों को बदलें:

function choice {
    typeset choice=$1

और करने के लिए shebang #!/bin/ksh


अच्छा छूट! इसे केएसएच में चलाने का प्रबंधन कैसे करें?
फुआसा

1
@FuSsA: मैंने अपने जवाब को संपादित करके इसे ksh में काम करने के लिए आवश्यक परिवर्तनों को दिखाने के लिए किया।
अगली सूचना तक रोक दिया गया।

1
बैश में सरणी हैंडलिंग बहुत कट्टर है। आप केवल पहले नहीं हैं, आप पूरे त्रिमूर्ति पर 40k से ऊपर के एकमात्र हैं।
पीटर का कहना है कि मोनिका

1
@FuSsA: options=(*)(या अन्य ग्लोबिंग पैटर्न) आपको सरणी में फ़ाइलों की एक सूची मिलेगी। फिर चुनौती को चयन अंक सरणी ( ${opts[@]}) के साथ एक साथ जोड़ दिया जाएगा। यह एक forलूप के साथ किया जा सकता है , लेकिन इसे बाहरी whileलूप के माध्यम से प्रत्येक पास के लिए चलाना होगा । आप अपने अन्य उत्तर में उपयोग करने पर विचार करना चाहते हैं dialogया whiptailजैसा कि मैंने उल्लेख किया है - हालांकि ये बाहरी निर्भरताएं हैं।
अगली सूचना तक रोक दिया गया।

1
@ फुस्सा: तब आप स्ट्रिंग को किसी अन्य सरणी में सहेज सकते हैं (या स्ट्रिंग को उपयोग ${opts[@]}और सहेज सकते हैं, फ़ंक्शन के अतिरिक्त तर्क के रूप में, इसके बजाय +)।
अगली सूचना तक रोक दिया गया।

2

मैंने प्रश्नावली नामक एक पुस्तकालय लिखा , जो कमांड लाइन प्रश्नावली बनाने के लिए एक मिनी-डीएसएल है। यह उपयोगकर्ता को प्रश्नों की एक श्रृंखला का उत्तर देने के लिए प्रेरित करता है और स्टैडआउट के उत्तरों को प्रिंट करता है।

यह आपके कार्य को वास्तव में आसान बनाता है। इसे स्थापित करें pip install questionnaireऔर एक स्क्रिप्ट बनाएं, जैसे questions.py, इस तरह:

from questionnaire import Questionnaire
q = Questionnaire(out_type='plain')

q.add_question('options', prompt='Choose some options', prompter='multiple',
               options=['Option 1', 'Option 2', 'Option 3', 'Option 4'], all=None)

q.run()

फिर चला python questions.py। जब आप उन सवालों के जवाब देने के लिए तैयार हो जाते हैं जो वे स्टडआउट के लिए मुद्रित होते हैं। यह पायथन 2 और 3 के साथ काम करता है, जिनमें से एक आपके सिस्टम पर लगभग निश्चित रूप से स्थापित है।

यदि कोई ऐसा करना चाहता है तो यह और भी जटिल प्रश्नावली को संभाल सकता है। यहाँ कुछ विशेषताएं हैं:

  • प्रिंट करने के लिए JSON (या सादे पाठ के रूप में) के उत्तर प्रिंट करता है
  • उपयोगकर्ताओं को वापस जाने और पुन: प्रश्न करने की अनुमति देता है
  • सशर्त प्रश्नों का समर्थन करता है (प्रश्न पिछले उत्तरों पर निर्भर कर सकते हैं)
  • निम्नलिखित प्रकार के प्रश्नों का समर्थन करता है: कच्चा इनपुट, एक चुनें, कई चुनें
  • प्रश्न प्रस्तुति और उत्तर मूल्यों के बीच कोई अनिवार्य युग्मन नहीं

1

मैंने MestreLion से उदाहरण का उपयोग किया और नीचे दिए गए कोड का मसौदा तैयार किया। आपको केवल पहले दो खंडों में विकल्पों और कार्यों को अपडेट करना होगा।

#!/bin/bash
#title:         menu.sh
#description:   Menu which allows multiple items to be selected
#author:        Nathan Davieau
#               Based on script from MestreLion
#created:       May 19 2016
#updated:       N/A
#version:       1.0
#usage:         ./menu.sh
#==============================================================================

#Menu options
options[0]="AAA"
options[1]="BBB"
options[2]="CCC"
options[3]="DDD"
options[4]="EEE"

#Actions to take based on selection
function ACTIONS {
    if [[ ${choices[0]} ]]; then
        #Option 1 selected
        echo "Option 1 selected"
    fi
    if [[ ${choices[1]} ]]; then
        #Option 2 selected
        echo "Option 2 selected"
    fi
    if [[ ${choices[2]} ]]; then
        #Option 3 selected
        echo "Option 3 selected"
    fi
    if [[ ${choices[3]} ]]; then
        #Option 4 selected
        echo "Option 4 selected"
    fi
    if [[ ${choices[4]} ]]; then
        #Option 5 selected
        echo "Option 5 selected"
    fi
}

#Variables
ERROR=" "

#Clear screen for menu
clear

#Menu function
function MENU {
    echo "Menu Options"
    for NUM in ${!options[@]}; do
        echo "[""${choices[NUM]:- }""]" $(( NUM+1 ))") ${options[NUM]}"
    done
    echo "$ERROR"
}

#Menu loop
while MENU && read -e -p "Select the desired options using their number (again to uncheck, ENTER when done): " -n1 SELECTION && [[ -n "$SELECTION" ]]; do
    clear
    if [[ "$SELECTION" == *[[:digit:]]* && $SELECTION -ge 1 && $SELECTION -le ${#options[@]} ]]; then
        (( SELECTION-- ))
        if [[ "${choices[SELECTION]}" == "+" ]]; then
            choices[SELECTION]=""
        else
            choices[SELECTION]="+"
        fi
            ERROR=" "
    else
        ERROR="Invalid option: $SELECTION"
    fi
done

ACTIONS

बहुत बढ़िया जवाब। संख्या बढ़ाने के लिए एक नोट भी जोड़ें, e..g विकल्प 15; n1 SELECTIONअंकों की संख्या बढ़ाने के लिए महत्वपूर्ण भाग कहाँ है ..
dbf

जोड़ना भूल गए; जहां -n2 SELECTIONदो अंकों को स्वीकार किया जाएगा (जैसे 15), -n3तीन को स्वीकार करता है (उदाहरण के लिए 153), आदि
dbf

1

यहां एक बैश फ़ंक्शन है जो उपयोगकर्ता को तीर कुंजी और स्पेस के साथ कई विकल्पों का चयन करने की अनुमति देता है, और एंटर के साथ पुष्टि करता है। यह एक अच्छा मेनू की तरह लग रहा है। मैंने इसे https://unix.stackexchange.com/a/415155 की मदद से लिखा है । इसे इस तरह कहा जा सकता है:

multiselect result "Option 1;Option 2;Option 3" "true;;true"

परिणाम को चर में एक सरणी के रूप में संग्रहीत किया जाता है, जिसमें पहले तर्क के रूप में नाम दिया गया होता है। अंतिम तर्क वैकल्पिक है और डिफ़ॉल्ट रूप से चयनित कुछ विकल्पों को बनाने के लिए उपयोग किया जाता है। यह इस तरह दिख रहा है।

function prompt_for_multiselect {

    # 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_inactive()    { printf "$2   $1 "; }
    print_active()      { printf "$2  $ESC[7m $1 $ESC[27m"; }
    get_cursor_row()    { IFS=';' read -sdR -p $'\E[6n' ROW COL; echo ${ROW#*[}; }
    key_input()         {
      local key
      IFS= read -rsn1 key 2>/dev/null >&2
      if [[ $key = ""      ]]; then echo enter; fi;
      if [[ $key = $'\x20' ]]; then echo space; fi;
      if [[ $key = $'\x1b' ]]; then
        read -rsn2 key
        if [[ $key = [A ]]; then echo up;    fi;
        if [[ $key = [B ]]; then echo down;  fi;
      fi 
    }
    toggle_option()    {
      local arr_name=$1
      eval "local arr=(\"\${${arr_name}[@]}\")"
      local option=$2
      if [[ ${arr[option]} == true ]]; then
        arr[option]=
      else
        arr[option]=true
      fi
      eval $arr_name='("${arr[@]}")'
    }

    local retval=$1
    local options
    local defaults

    IFS=';' read -r -a options <<< "$2"
    if [[ -z $3 ]]; then
      defaults=()
    else
      IFS=';' read -r -a defaults <<< "$3"
    fi
    local selected=()

    for ((i=0; i<${#options[@]}; i++)); do
      selected+=("${defaults[i]}")
      printf "\n"
    done

    # determine current screen position for overwriting the options
    local lastrow=`get_cursor_row`
    local startrow=$(($lastrow - ${#options[@]}))

    # 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 active=0
    while true; do
        # print options by overwriting the last lines
        local idx=0
        for option in "${options[@]}"; do
            local prefix="[ ]"
            if [[ ${selected[idx]} == true ]]; then
              prefix="[x]"
            fi

            cursor_to $(($startrow + $idx))
            if [ $idx -eq $active ]; then
                print_active "$option" "$prefix"
            else
                print_inactive "$option" "$prefix"
            fi
            ((idx++))
        done

        # user key control
        case `key_input` in
            space)  toggle_option selected $active;;
            enter)  break;;
            up)     ((active--));
                    if [ $active -lt 0 ]; then active=$((${#options[@]} - 1)); fi;;
            down)   ((active++));
                    if [ $active -ge ${#options[@]} ]; then active=0; fi;;
        esac
    done

    # cursor position back to normal
    cursor_to $lastrow
    printf "\n"
    cursor_blink_on

    eval $retval='("${selected[@]}")'
}

आप इसे कैसे कहते हैं? फ़ाइल कैसी लगेगी?
एली

1

चूंकि मुझे प्रॉम्प्ट-टूलकिट (अजगर), डायलाउर (जंग) या जिज्ञासु (नोड) के लिए कोई उपयुक्त बीएचएएस अल्टरनेटिव नहीं मिला, इसलिए मैंने इसे अपने दम पर आजमाया:

https://i.stack.imgur.com/6AyAI.png

https://asciinema.org/a/Y4hLxnN20JtAlrn3hsC6dCRn8 https://gist.github.com/blurayne/f63c5a8521c0eeab8a9afd8baa45c65e


-1
export supermode=none

source easybashgui

list "Option 1" "Option 2" "Option 3" "Option 4"

2
हो सकता है कि आप थोड़ा सा विवरण जोड़ सकें कि यह क्या कर रहा है? भविष्य के आगंतुकों के लिए, ओपी के लिए इतना नहीं।
SLM

इसके अलावा, की उत्पत्ति के लिए एक कड़ी easybashgui
अगली सूचना तक रोक दिया गया।
हमारी साइट का प्रयोग करके, आप स्वीकार करते हैं कि आपने हमारी Cookie Policy और निजता नीति को पढ़ और समझा लिया है।
Licensed under cc by-sa 3.0 with attribution required.