Size: 1692
Comment: mark examples by requirements, make the POSIX one really POSIX, make them functions, and show how to call them
|
Size: 1637
Comment: remove useless sleep calls
|
Deletions are marked like this. | Additions are marked like this. |
Line 54: | Line 54: |
# Bash, with GNU sleep | # Bash |
Line 57: | Line 57: |
sn=${#sp} | |
Line 58: | Line 59: |
printf "\b${sp:sc++:1}" ((sc==${#sp})) && sc=0 sleep 0.1 |
printf '\b%s' "${sp:sc++%sn:1}" |
Line 64: | Line 63: |
sleep 0.1 |
Can I do a spinner in Bash?
Sure!
# Bash, with GNU sleep spin() { local i=0 local sp='/-\|' local n=${#sp} printf ' ' sleep 0.1 while true; do printf '\b%s' "${sp:i++%n:1}" sleep 0.1 done }
Each time the loop iterates, it displays the next character in the sp string, wrapping around as it reaches the end. (i is the position of the current character to display and ${#sp} is the length of the sp string).
The \b string is replaced by a 'backspace' character. Alternatively, you could play with \r to go back to the beginning of the line.
To slow it down, the sleep command is included inside the loop (after the printf).
A POSIX equivalent would be:
# POSIX sh spin() { sp='/-\|' printf ' ' sleep 1 while true; do printf '\b%.1s' "$sp" sp=${sp#?}${sp%???} sleep 1 done }
One way to use these spinners in a script is to run them as background processes, and kill them when you're done. For example,
# POSIX sh spin & spinpid=$! # long-running commands here kill "$spinpid"
If you already have a loop which does a lot of work, you can write a function that "advances" the spinner one step at a time, and call it at the beginning of each iteration:
# Bash sp='/-\|' sc=0 sn=${#sp} spin() { printf '\b%s' "${sp:sc++%sn:1}" } endspin() { printf '\r%s\n' "$*" } until work_done; do spin some_work ... done endspin
A similar technique can be used to build progress bars.