mirror of
https://github.com/prurigro/vnotes.git
synced 2025-01-22 05:41:01 -05:00
Initial commit
This commit is contained in:
commit
c8746d8405
2 changed files with 369 additions and 0 deletions
24
readme.md
Normal file
24
readme.md
Normal file
|
@ -0,0 +1,24 @@
|
|||
# vnotes
|
||||
|
||||
Manage your notes with flat files and your system text editor
|
||||
|
||||
## Dependencies
|
||||
|
||||
* [bash](https://www.gnu.org/software/bash/bash.html)
|
||||
* [kbd](http://www.kbd-project.org)
|
||||
* [ncurses](https://invisible-island.net/ncurses/ncurses.html)
|
||||
|
||||
## Configuration
|
||||
|
||||
Configuration is handled using environment variables:
|
||||
|
||||
* `EDITOR`: The text editor that should be used to open notes (default: `/usr/bin/vim`)
|
||||
* `VNOTES_FOLDER`: The folder notes should be managed in (default: `$HOME/Notes`)
|
||||
* `VNOTES_EXTENSION`: The file extension notes should have (default: `md`)
|
||||
|
||||
## Usage
|
||||
|
||||
* `vnotes [PATTERN]`: If multiple notes include `[PATTERN]` a list of matching notes is presented to choose from, otherwise it opens the closest matching note
|
||||
* `-c|--create [NAME]`: Creates a new note named `[NAME]`
|
||||
* `-d|--delete [NAME]`: Deletes the note named `[NAME]`
|
||||
* `-h|--help`: Shows the help text
|
345
vnotes
Executable file
345
vnotes
Executable file
|
@ -0,0 +1,345 @@
|
|||
#!/usr/bin/env bash
|
||||
|
||||
#
|
||||
# vnotes: manage your notes
|
||||
#
|
||||
# Version: 1.0
|
||||
#
|
||||
# Written by Kevin MacMartin <prurigro@gmail.com>
|
||||
# Released under the MIT license
|
||||
#
|
||||
|
||||
shopt -s extglob
|
||||
|
||||
# Environment config
|
||||
EDITOR=${EDITOR:='/usr/bin/vim'}
|
||||
VNOTES_FOLDER=${VNOTES_FOLDER:="$HOME/Notes"}
|
||||
VNOTES_EXTENSION=${NOTES_SUFFIX:='md'}
|
||||
|
||||
# Dependencies
|
||||
deps=('getkeycodes' 'stty' 'tput')
|
||||
|
||||
declare -a documents=()
|
||||
max_docwidth=0
|
||||
|
||||
# Colour Scheme
|
||||
[[ -t 1 ]] && {
|
||||
c_d=$'\e[1;30m' # DARK GREY
|
||||
c_r=$'\e[1;31m' # RED
|
||||
c_g=$'\e[1;32m' # GREEN
|
||||
c_y=$'\e[1;33m' # YELLOW
|
||||
c_b=$'\e[1;34m' # BLUE
|
||||
c_m=$'\e[1;35m' # VIOLET
|
||||
c_t=$'\e[1;36m' # TEAL
|
||||
c_w=$'\e[1;37m' # WHITE
|
||||
c_u=$'\e[1;4;37m' # UNDERLINE WHITE
|
||||
c_h=$'\e[1;41m' # HIGHLIGHT RED
|
||||
c_c=$'\e[0m' # DISABLES COLOUR
|
||||
}
|
||||
|
||||
# Display help text
|
||||
function help {
|
||||
script_name="${0//*\/}"
|
||||
printf '\n%s\n\n' "${c_m}vnotes${c_w} - manage your notes$c_c"
|
||||
printf '%s\n' "${c_b}USAGE$c_c"
|
||||
printf ' %-43s%s\n' "${c_d}[${c_y}PATTERN$c_d]$c_c" ' — opens single match or lists multiple'
|
||||
printf ' %-43s%s\n' "$c_w-c$c_d|$c_w--create ${c_d}[${c_y}NAME$c_d]$c_c" ' — create a new note'
|
||||
printf ' %-43s%s\n' "$c_w-d$c_d|$c_w--delete ${c_d}[${c_y}NAME$c_d]$c_c" ' — deletes a note'
|
||||
printf ' %-43s%s\n\n' "$c_w-h$c_d|$c_w--help$c_c" ' — show this help text'
|
||||
exit
|
||||
}
|
||||
|
||||
# Function to process key codes
|
||||
function getkeycode {
|
||||
while read -r; do
|
||||
[[ "$REPLY" =~ ^[0-9][0-9]*\ (.*)$ ]] && {
|
||||
printf '%s\n' "${BASH_REMATCH[1]}"
|
||||
break
|
||||
}
|
||||
done <<< "$1"
|
||||
}
|
||||
|
||||
function prompt {
|
||||
printf '%s ' "${c_w}Please choose a number from the list ${c_d}[$c_b$one_display-$upto$c_d] $c_d(${c_m}Q$c_w to quit$c_d)$c_w:$c_c"
|
||||
}
|
||||
|
||||
function power {
|
||||
local number="$1" power="$2" total=1
|
||||
|
||||
for (( x=0; x<"$power"; x++ )); do
|
||||
total=$(( total * number ))
|
||||
done
|
||||
|
||||
printf '%s\n' "$total"
|
||||
}
|
||||
|
||||
function open_note {
|
||||
printf '%s %s\n' "${c_y}Opening:" "$c_m${documents[$1]}$c_c"
|
||||
[[ -n "$tty_state" ]] && stty "$tty_state"
|
||||
"$EDITOR" "$VNOTES_FOLDER/${documents[$1]}.$VNOTES_EXTENSION"
|
||||
exit
|
||||
}
|
||||
|
||||
function create_note {
|
||||
if [[ -n "$1" ]]; then
|
||||
note="$VNOTES_FOLDER/${1}.$VNOTES_EXTENSION"
|
||||
"$EDITOR" "$note"
|
||||
else
|
||||
printf '%s %s\n' "${c_r}ERROR:$c_c" 'No name given for the new note' >&2
|
||||
exit 1
|
||||
fi
|
||||
}
|
||||
|
||||
function delete_note {
|
||||
if [[ -n "$1" ]]; then
|
||||
note="$VNOTES_FOLDER/${1}.$VNOTES_EXTENSION"
|
||||
|
||||
if [[ -f "$note" ]]; then
|
||||
rm "$note"
|
||||
printf '%s\n' "${c_w}The note '${c_m}$1${c_w}' has been ${c_r}DELETED$c_c"
|
||||
else
|
||||
printf '%s %s\n' "${c_r}ERROR:$c_c" "${c_w}The note '${c_m}$1${c_w}' doesn't exist" >&2
|
||||
exit 1
|
||||
fi
|
||||
else
|
||||
printf '%s %s\n' "${c_r}ERROR:$c_c" 'No name given for the new note' >&2
|
||||
exit 1
|
||||
fi
|
||||
}
|
||||
|
||||
# Search for notes and open the closest match, or list out options if more than one exists
|
||||
function search_notes {
|
||||
# Take the first input as the key
|
||||
key="$1"
|
||||
|
||||
# Define Q, q, and Return respectively
|
||||
alpha_Q=$(getkeycode "$(printf '%s' 'Q' | od -t o1)")
|
||||
alpha_q=$(getkeycode "$(printf '%s' 'q' | od -t o1)")
|
||||
key_cr=$(getkeycode "$(printf '\n' | od -t o1)")
|
||||
key_0=$(getkeycode "$(printf '%s' '0' | od -t o1)")
|
||||
key_1=$(getkeycode "$(printf '%s' '1' | od -t o1)")
|
||||
key_2=$(getkeycode "$(printf '%s' '2' | od -t o1)")
|
||||
key_3=$(getkeycode "$(printf '%s' '3' | od -t o1)")
|
||||
key_4=$(getkeycode "$(printf '%s' '4' | od -t o1)")
|
||||
key_5=$(getkeycode "$(printf '%s' '5' | od -t o1)")
|
||||
key_6=$(getkeycode "$(printf '%s' '6' | od -t o1)")
|
||||
key_7=$(getkeycode "$(printf '%s' '7' | od -t o1)")
|
||||
key_8=$(getkeycode "$(printf '%s' '8' | od -t o1)")
|
||||
key_9=$(getkeycode "$(printf '%s' '9' | od -t o1)")
|
||||
|
||||
# Build the list of notes and track the longest document name
|
||||
if [[ -n "$key" ]]; then
|
||||
for each in "$VNOTES_FOLDER"/*."$VNOTES_EXTENSION"; do
|
||||
[[ "$each" =~ $key ]] && {
|
||||
each="${each/*\/}"
|
||||
each="${each/\.$VNOTES_EXTENSION}"
|
||||
documents=("${documents[@]}" "$each")
|
||||
(( max_docwidth < ${#each} )) && max_docwidth=${#each}
|
||||
}
|
||||
done
|
||||
else
|
||||
for each in "$VNOTES_FOLDER"/*."$VNOTES_EXTENSION"; do
|
||||
each="${each/*\/}"
|
||||
each="${each/\.$VNOTES_EXTENSION}"
|
||||
documents=("${documents[@]}" "$each")
|
||||
(( max_docwidth < ${#each} )) && max_docwidth=${#each}
|
||||
done
|
||||
fi
|
||||
|
||||
# Exit with an error if the list of notes is empty
|
||||
(( ${#documents[*]} )) || {
|
||||
printf '%s\n' "Can't find $key"
|
||||
exit 1
|
||||
}
|
||||
|
||||
if (( ${#documents[*]} == 1 )); then
|
||||
[[ "${documents[0]}" =~ ^\ *\*\ *$ ]] && {
|
||||
printf '%s %s\n' "${c_r}ERROR:$c_c" "${c_w}No notes in $c_m$VNOTES_FOLDER$c_c" >&2
|
||||
exit 1
|
||||
}
|
||||
|
||||
# Open the single document matching the search term
|
||||
open_note 0
|
||||
else
|
||||
# Display the multiple matching documents (or the full list if no term was passed)
|
||||
screen_width=$(tput cols)
|
||||
|
||||
if [[ -n "$screen_width" ]]; then
|
||||
maxcolumns=0
|
||||
|
||||
while (( 1 )); do
|
||||
(( (( max_docwidth + 8) * maxcolumns) > screen_width )) && {
|
||||
(( maxcolumns-- ))
|
||||
break
|
||||
}
|
||||
|
||||
(( maxcolumns++ ))
|
||||
done
|
||||
else
|
||||
maxcolumns=3
|
||||
fi
|
||||
|
||||
# Get the number of values
|
||||
upto=${#documents[@]}
|
||||
|
||||
# Calculate the number of digits in $upto
|
||||
digits=1
|
||||
|
||||
while (( 1 )); do
|
||||
digits_power=$(power 10 $digits)
|
||||
(( (upto / digits_power) < 1 )) && break
|
||||
(( digits++ ))
|
||||
done
|
||||
|
||||
unset digits_power
|
||||
|
||||
# Draw the list
|
||||
printf '\n'
|
||||
cnt=1
|
||||
|
||||
for (( x=0; x<${#documents[*]}; x++ )); do
|
||||
count_display=$(printf "%${digits}s" "$((x+1))")
|
||||
count_display="${count_display//\ /0}"
|
||||
(( x )) || one_display="$count_display"
|
||||
printf "${c_b}[%${digits}s]$c_c$c_u%$(( max_docwidth + 1 ))s$c_c" "$count_display" "${documents[$x]}"
|
||||
|
||||
if (( (cnt + 1) > maxcolumns )); then
|
||||
cnt=1
|
||||
printf '\n'
|
||||
else
|
||||
(( cnt++ ))
|
||||
printf '%4s' ' '
|
||||
(( (x + 1) >= ${#documents[*]} )) && printf '\n'
|
||||
fi
|
||||
done
|
||||
|
||||
unset cnt
|
||||
printf '\n'
|
||||
|
||||
# Initialize the interactive entry line
|
||||
tput init
|
||||
prompt
|
||||
tput sc
|
||||
tput rc
|
||||
|
||||
# Save the tty state
|
||||
tty_state="$(stty -g)"
|
||||
|
||||
# Trap a function to restore the tty on exit
|
||||
trap 'stty "$tty_state"; exit' SIGINT SIGQUIT SIGTERM
|
||||
|
||||
# Begin to capture input
|
||||
stty cs8 -icanon -echo min 1 time 1
|
||||
stty intr '' susp ''
|
||||
|
||||
# Loop the input loop and a set of input testing commands
|
||||
while (( 1 )); do
|
||||
# The input loop which accepts 0-9, q/Q/Ctrl-C to quit, Backspace and Enter
|
||||
value=''
|
||||
|
||||
while (( 1 )); do
|
||||
# Break out of the loop if $value is as long as the largest number to select from
|
||||
(( ${#value} >= digits )) && break
|
||||
|
||||
# Wait for and accept input
|
||||
keypress=$(getkeycode "$(dd bs=10 count=1 2> /dev/null | od -t o1)")
|
||||
|
||||
case "$keypress" in
|
||||
'177'|'010'|'033 133 063 176')
|
||||
# Backspace
|
||||
[[ -n "$value" ]] && value="${value:0:$((${#value}-1))}"
|
||||
;;
|
||||
|
||||
"$alpha_q"|"$alpha_Q") value='-1'; break ;;
|
||||
"$key_cr") [[ -n "$value" ]] && break ;;
|
||||
'003') value='-1'; break ;; # Ctrl-C
|
||||
"$key_0") value="${value}0" ;; # 0
|
||||
"$key_1") value="${value}1" ;; # 1
|
||||
"$key_2") value="${value}2" ;; # 2
|
||||
"$key_3") value="${value}3" ;; # 3
|
||||
"$key_4") value="${value}4" ;; # 4
|
||||
"$key_5") value="${value}5" ;; # 5
|
||||
"$key_6") value="${value}6" ;; # 6
|
||||
"$key_7") value="${value}7" ;; # 7
|
||||
"$key_8") value="${value}8" ;; # 8
|
||||
"$key_9") value="${value}9" ;; # 9
|
||||
esac
|
||||
|
||||
# Redraw the input line with an up to date $value
|
||||
tput el1
|
||||
tput cr
|
||||
prompt
|
||||
printf '%s' "$value"
|
||||
done
|
||||
|
||||
# Remove leading 0s from $value
|
||||
value="${value##+(0)}"
|
||||
|
||||
# Quit if input was given to do so
|
||||
(( value == -1 )) && {
|
||||
printf '\n%s\n' "${c_r}Quitting!$c_c"
|
||||
[[ -n "$tty_state" ]] && stty "$tty_state"
|
||||
exit
|
||||
}
|
||||
|
||||
# Evaluate the numbers and exit if valid, otherwise display an error and start again
|
||||
if (( 1 <= value && value <= upto )); then
|
||||
printf '\n'
|
||||
open_note $(( value - 1 ))
|
||||
else
|
||||
tput el1
|
||||
tput cr
|
||||
printf '%s' "$c_h Invalid number: $value $c_c"
|
||||
sleep .8
|
||||
tput el1
|
||||
tput cr
|
||||
prompt
|
||||
fi
|
||||
done
|
||||
fi
|
||||
|
||||
# Restore the tty state
|
||||
[[ -n "$tty_state" ]] && stty "$tty_state"
|
||||
}
|
||||
|
||||
# Check for missing dependencies
|
||||
declare -a missing_deps=()
|
||||
|
||||
for dep in "${deps[@]}"; do
|
||||
type -P "$dep" >/dev/null \
|
||||
|| missing_deps=( "${missing_deps[@]}" "$dep" )
|
||||
done
|
||||
|
||||
[[ -n "${missing_deps[*]}" ]] && {
|
||||
printf '%s\n' "${c_w}missing dependencies ($(
|
||||
for (( x=0; x < ${#missing_deps[@]}; x++ )); do
|
||||
printf '%s' "$c_m${missing_deps[$x]}$c_c"
|
||||
(( (x + 1) < ${#missing_deps[@]} )) && printf '%s' ', '
|
||||
done
|
||||
)$c_w)"
|
||||
|
||||
exit 1
|
||||
}
|
||||
|
||||
# Create the VNOTES_FOLDER if it doesn't exist
|
||||
[[ -d "$VNOTES_FOLDER" ]] || install -d "$VNOTES_FOLDER"
|
||||
|
||||
# Parse for command line arguments
|
||||
case $1 in
|
||||
-c|--create)
|
||||
shift
|
||||
create_note "$*"
|
||||
;;
|
||||
|
||||
-d|--delete)
|
||||
shift
|
||||
delete_note "$*"
|
||||
;;
|
||||
|
||||
-h|--help)
|
||||
help
|
||||
;;
|
||||
|
||||
*)
|
||||
search_notes "$*"
|
||||
esac
|
Loading…
Reference in a new issue