BASH-скрипты: выбор файла whiptail
Я нашел отличную небольшую программу, которая позволит мне добавлять удобные графические интерфейсы в мои Bash Scripts;
Хлыстохвост
Тем не менее, справочная страница whiptail не так уж полезна и не содержит примеров. После некоторых поисков в Google я понимаю, как создать простое меню да / нет, используя whiptail:
#! /bin/bash
# http://archives.seul.org/seul/project/Feb-1998/msg00069.html
if (whiptail --title "PPP Configuration" --backtitle "Welcome to SEUL" --yesno "
Do you want to configure your PPP connection?" 10 40 )
then
echo -e "\nWell, you better get busy!\n"
elif (whiptail --title "PPP Configuration" --backtitle "Welcome to
SEUL" --yesno " Are you sure?" 7 40)
then
echo -e "\nGood, because I can't do that yet!\n"
else
echo -e "\nToo bad, I can't do that yet\n"
fi
Но то, что я действительно хотел бы построить меню выбора файла, используя whiptail, чтобы заменить старый код, который у меня есть, в нескольких различных скриптах bash резервного копирования / восстановления, которые у меня есть:
#!/bin/bash
#This script allows you to select a file ending in the .tgz extension (in the current directory)
echo "Please Select the RESTORE FILE you would like to restore: "
select RESTOREFILE in *.tgz; do
break #Nothing
done
echo "The Restore File you selected was: ${RESTOREFILE}"
Я предполагаю, что это должно быть сделано с помощью опции '--menu', но я не знаю, как это сделать? Есть указатели? Или вы можете указать мне на некоторые примеры из кнута?
4 ответа
Создайте массив имен файлов и тегов выбора меню:
i=0
s=65 # decimal ASCII "A"
for f in *.tgz
do
# convert to octal then ASCII character for selection tag
files[i]=$(echo -en "\0$(( $s / 64 * 100 + $s % 64 / 8 * 10 + $s % 8 ))")
files[i+1]="$f" # save file name
((i+=2))
((s++))
done
Такой метод будет работать, даже если есть имена файлов с пробелами. Если количество файлов велико, возможно, вам придется разработать другую стратегию тегов.
Использование буквенных символов для тегов позволяет нажимать букву для перехода к элементу. Числовые теги, кажется, не делают этого. Если вам не нужно такое поведение, то вы можете устранить некоторые сложности.
Показать меню:
whiptail --backtitle "Welcome to SEUL" --title "Restore Files" \
--menu "Please select the file to restore" 14 40 6 "${files[@]}"
Если код выхода 255, диалог был отменен.
if [[ $? == 255 ]]
then
do cancel stuff
fi
Чтобы поймать выделение в переменной, используйте эту структуру (замените свою команду whiptail на "whiptail-command"):
result=$(whiptail-command 2>&1 >/dev/tty)
Или же
result=$(whiptail-command 3>&2 2>&1 1>&3-)
Переменная $result
будет содержать букву алфавита, которая соответствует файлу в массиве. К сожалению, Bash до версии 4 не поддерживает ассоциативные массивы. Вы можете вычислить индекс в массив файла из буквы следующим образом (обратите внимание на "лишние" одинарные кавычки):
((index = 2 * ( $( printf "%d" "'$result" ) - 65 ) + 1 ))
Пример:
Welcome to SEUL
┌──────────┤ Restore Files ├───────────┐
│ Please select the file to restore │
│ │
│ A one.tgz ↑ │
│ B two.tgz ▮ │
│ C three.tgz ▒ │
│ D another.tgz ▒ │
│ E more.tgz ▒ │
│ F sp ac es.tgz ↓ │
│ │
│ │
│ <Ok> <Cancel> │
│ │
└──────────────────────────────────────┘
Whiptail - это легкое переопределение наиболее популярных функций диалога с использованием библиотеки Newt. Я сделал быструю проверку, и многие функции в Whiptail, похоже, ведут себя как их аналоги в диалоге. Итак, руководство по диалогу должно помочь вам начать. Вы можете найти его здесь, но Google, конечно, ваш друг. С другой стороны, расширенный пример, вероятно, содержит много вдохновения для вашей проблемы.
Эта функция является частью моего репозитория функций для whiptail
# ----------------------------------------------------------------------
# File selection dialog
#
# Arguments
# 1 Dialog title
# 2 Source path to list files and directories
# 3 File mask (by default *)
# 4 "yes" to allow go back in the file system.
#
# Returns
# 0 if a file was selected and loads the FILE_SELECTED variable
# with the selected file.
# 1 if the user cancels.
# ----------------------------------------------------------------------
function dr_file_select
{
local TITLE=${1:-$MSG_INFO_TITLE}
local LOCAL_PATH=${2:-$(pwd)}
local FILE_MASK=${3:-"*"}
local ALLOW_BACK=${4:-no}
local FILES=()
[ "$ALLOW_BACK" != "no" ] && FILES+=(".." "..")
# First add folders
for DIR in $(find $LOCAL_PATH -maxdepth 1 -mindepth 1 -type d -printf "%f " 2> /dev/null)
do
FILES+=($DIR "folder")
done
# Then add the files
for FILE in $(find $LOCAL_PATH -maxdepth 1 -type f -name "$FILE_MASK" -printf "%f %s " 2> /dev/null)
do
FILES+=($FILE)
done
while true
do
FILE_SELECTED=$(whiptail --clear --backtitle "$BACK_TITLE" --title "$TITLE" --menu "$LOCAL_PATH" 38 80 30 ${FILES[@]} 3>&1 1>&2 2>&3)
if [ -z "$FILE_SELECTED" ]; then
return 1
else
if [ "$FILE_SELECTED" = ".." ] && [ "$ALLOW_BACK" != "no" ]; then
return 0
elif [ -d "$LOCAL_PATH/$FILE_SELECTED" ] ; then
if dr_file_select "$TITLE" "$LOCAL_PATH/$FILE_SELECTED" "$FILE_MASK" "yes" ; then
if [ "$FILE_SELECTED" != ".." ]; then
return 0
fi
else
return 1
fi
elif [ -f "$LOCAL_PATH/$FILE_SELECTED" ] ; then
FILE_SELECTED="$LOCAL_PATH/$FILE_SELECTED"
return 0
fi
fi
done
}
Использование простое
if dr_file_select "Please, select a file" /home/user ; then
echo "File Selected: \"$FILE_SELECTED\"."
else
echo "Cancelled!"
fi
Похоже, это один из лучших результатов при поиске кнута, и ни один из предыдущих результатов не помог мне. Вот что я использовал:
#! /bin/bash
shopt -s nullglob
dir=`pwd`
cd /path/to/files
arr=(*.tgz)
for ((i=0; i<${#arr[@]}; i++)); do j=$((2*$i+2)); a[j]="${arr[$i]}"; a[j+1]=""; done
a[0]=""
# Next line has extra spaces at right to try to center it:
a[1]="Please make a selection from the files below "
result=$(whiptail --ok-button "OK button text" --cancel-button "Cancel Button Text" --title "Title Text" --backtitle "Text at upper left corner of page" --menu "Menu Text (not used??)" 30 160 24 "${a[@]}" 2>&1 >/dev/tty)
if [[ $? = 0 ]]
then
# ge 5 in next line should be length of file extension including . character, plus 1
[ ${#result} -ge 5 ] && outfile="/path/to/files/$result" || echo "Selection not made"
fi
cd "$dir"
$ result будет пустым, если не был сделан правильный выбор. Я добавил фиктивную выборку вверху списка, которая в результате возвращает пустую строку, чтобы случайно не выбрать неправильный файл, случайно нажав Enter сразу после появления меню. Если вы не хотите этого, то в строке "for" уберите +2 в "do j=$((2*$i+2))", а также следующие две строки, которые устанавливают [0] и a[1] явно.
Вводящий в заблуждение факт заключается в том, что при чтении из массива в подобной ситуации он ожидает два элемента данных в строке, каждый из которых отображается на экране, причем первым является результат, который вы хотите вернуть, если ожидается строка (что в некоторых ситуациях может привести к будь то буква или цифра), а второй - любой описательный текст, который вы захотите. Вот почему для первой строки я использую [0], чтобы дать пустую строку в качестве результата, и a[1] в качестве описательного текста, но оттуда первый элемент в паре содержит имя файла (это то, что я на самом деле хочу вернуть), а вторая - пустая строка, так как я не хочу отображать в этих строках любой текст, кроме имени файла.
Также в предыдущем посте говорилось, что whiptail вернул код ошибки 255, если была нажата кнопка отмены, но это не относится к имеющейся у меня версии - он возвращает 1. Поэтому я просто проверяю код ошибки 0, и если это я Предположим, что это может быть допустимая запись, затем я проверяю правильную длину строки (больше, чем просто количество символов в расширении файла, включая символ.), чтобы быть уверенным.
Я попробовал следующее, которое сработало:
whiptail --title "PPP Config" --backtitle "Welcome to SEUL" --menu YourTitle 20 80 10 `for x in $(ls -1 *.tgz); do echo $x "-"; done`
Вы также можете изменить это на несколько строк, я добавил проверку на пустой список:
MYLIST=`for x in $(ls -1 *.tgz); do echo $x "-"; done`
WC=`echo $MYLIST | wc -l`
if [[WC -ne 0]]; then
whiptail --title "PPP Config" --backtitle "Welcome to SEUL" --menu YourTitle 20 80 10 $MYLIST
fi
вам нужно настроить цифры, чтобы получить чистый интерфейс. И вы можете заменить "-"
чем угодно, если хотите. Но если вы этого не сделаете, вы увидите 2 записи в строке.
Кстати: выбранная запись печатается на stderr
,
Это может нуждаться в некотором улучшении, но для основной идеи я думаю, что этого достаточно.