summaryrefslogtreecommitdiff
path: root/home/.oh-my-zsh/plugins/bgnotify
diff options
context:
space:
mode:
Diffstat (limited to 'home/.oh-my-zsh/plugins/bgnotify')
-rw-r--r--home/.oh-my-zsh/plugins/bgnotify/README.md64
-rw-r--r--home/.oh-my-zsh/plugins/bgnotify/bgnotify.plugin.zsh141
2 files changed, 205 insertions, 0 deletions
diff --git a/home/.oh-my-zsh/plugins/bgnotify/README.md b/home/.oh-my-zsh/plugins/bgnotify/README.md
new file mode 100644
index 0000000..1389def
--- /dev/null
+++ b/home/.oh-my-zsh/plugins/bgnotify/README.md
@@ -0,0 +1,64 @@
+# bgnotify zsh plugin
+
+cross-platform background notifications for long running commands! Supports OSX and Linux.
+
+Standalone homepage: [t413/zsh-background-notify](https://github.com/t413/zsh-background-notify)
+
+---
+
+## How to use
+
+Just add bgnotify to your plugins list in your `.zshrc`
+
+- On OS X you'll need [terminal-notifier](https://github.com/alloy/terminal-notifier)
+ * `brew install terminal-notifier` (or `gem install terminal-notifier`)
+- On Linux, make sure you have `notify-send` or `kdialog` installed. If you're using Ubuntu you should already be all set!
+- On Windows you can use [notifu](https://www.paralint.com/projects/notifu/) or the Cygwin Ports libnotify package
+
+
+## Screenshots
+
+**Linux**
+
+![screenshot from 2014-11-07 15 58 36](https://cloud.githubusercontent.com/assets/326829/4962187/256b465c-66da-11e4-927d-cc2fc105e31f.png)
+
+**OS X**
+
+![screenshot 2014-11-08 14 15 12](https://cloud.githubusercontent.com/assets/326829/4965780/19fa3eac-6795-11e4-8ed6-0355711123a9.png)
+
+**Windows**
+
+![screenshot from 2014-11-07 15 55 00](https://cloud.githubusercontent.com/assets/326829/4962159/a2625ca0-66d9-11e4-9e91-c5834913190e.png)
+
+
+## Configuration
+
+One can configure a few things:
+
+- `bgnotify_bell` enabled or disables the terminal bell (default true)
+- `bgnotify_threshold` sets the notification threshold time (default 6 seconds)
+- `function bgnotify_formatted` lets you change the notification. You can for instance customize the message and pass in an icon.
+- `bgnotify_extraargs` appends extra args to notifier (e.g. `-e` for notify-send to create a transient notification)
+
+Use these by adding a function definition before the your call to source. Example:
+
+```sh
+bgnotify_bell=false ## disable terminal bell
+bgnotify_threshold=4 ## set your own notification threshold
+
+function bgnotify_formatted {
+ ## $1=exit_status, $2=command, $3=elapsed_time
+
+ # Humanly readable elapsed time
+ local elapsed="$(( $3 % 60 ))s"
+ (( $3 < 60 )) || elapsed="$((( $3 % 3600) / 60 ))m $elapsed"
+ (( $3 < 3600 )) || elapsed="$(( $3 / 3600 ))h $elapsed"
+
+ [ $1 -eq 0 ] && title="Holy Smokes Batman" || title="Holy Graf Zeppelin"
+ [ $1 -eq 0 ] && icon="$HOME/icons/success.png" || icon="$HOME/icons/fail.png"
+ bgnotify "$title - took ${elapsed}" "$2" "$icon"
+}
+
+plugins=(git bgnotify) ## add to plugins list
+source $ZSH/oh-my-zsh.sh ## existing source call
+```
diff --git a/home/.oh-my-zsh/plugins/bgnotify/bgnotify.plugin.zsh b/home/.oh-my-zsh/plugins/bgnotify/bgnotify.plugin.zsh
new file mode 100644
index 0000000..dca8250
--- /dev/null
+++ b/home/.oh-my-zsh/plugins/bgnotify/bgnotify.plugin.zsh
@@ -0,0 +1,141 @@
+#!/usr/bin/env zsh
+
+## Setup
+
+[[ -o interactive ]] || return # don't load on non-interactive shells
+[[ -z "$SSH_CLIENT" && -z "$SSH_TTY" ]] || return # don't load on a SSH connection
+
+zmodload zsh/datetime # faster than `date`
+
+
+## Zsh Hooks
+
+function bgnotify_begin {
+ bgnotify_timestamp=$EPOCHSECONDS
+ bgnotify_lastcmd="${1:-$2}"
+}
+
+function bgnotify_end {
+ {
+ local exit_status=$?
+ local elapsed=$(( EPOCHSECONDS - bgnotify_timestamp ))
+
+ # check time elapsed
+ [[ $bgnotify_timestamp -gt 0 ]] || return 0
+ [[ $elapsed -ge $bgnotify_threshold ]] || return 0
+
+ # check if Terminal app is not active
+ [[ $(bgnotify_appid) != "$bgnotify_termid" ]] || return 0
+
+ bgnotify_formatted "$exit_status" "$bgnotify_lastcmd" "$elapsed"
+ } always {
+ bgnotify_timestamp=0
+ }
+}
+
+autoload -Uz add-zsh-hook
+add-zsh-hook preexec bgnotify_begin
+add-zsh-hook precmd bgnotify_end
+
+
+## Functions
+
+# allow custom function override
+(( ${+functions[bgnotify_formatted]} )) || \
+function bgnotify_formatted {
+ local exit_status=$1
+ local cmd="$2"
+
+ # humanly readable elapsed time
+ local elapsed="$(( $3 % 60 ))s"
+ (( $3 < 60 )) || elapsed="$((( $3 % 3600) / 60 ))m $elapsed"
+ (( $3 < 3600 )) || elapsed="$(( $3 / 3600 ))h $elapsed"
+
+ [[ $bgnotify_bell = true ]] && printf '\a' # beep sound
+ if [[ $exit_status -eq 0 ]]; then
+ bgnotify "#win (took $elapsed)" "$cmd"
+ else
+ bgnotify "#fail (took $elapsed)" "$cmd"
+ fi
+}
+
+function bgnotify_appid {
+ if (( ${+commands[osascript]} )); then
+ osascript -e "tell application id \"$(bgnotify_programid)\" to get the {id, frontmost, id of front window, visible of front window}" 2>/dev/null
+ elif [[ -n $WAYLAND_DISPLAY ]] && ([[ -n $SWAYSOCK ]] || [[ -n $I3SOCK ]]) && (( ${+commands[swaymsg]} )); then # wayland+sway
+ local app_id=$(bgnotify_find_sway_appid)
+ [[ -n "$app_id" ]] && echo "$app_id" || echo $EPOCHSECONDS
+ elif [[ -z $WAYLAND_DISPLAY ]] && [[ -n $DISPLAY ]] && (( ${+commands[xprop]} )); then
+ xprop -root _NET_ACTIVE_WINDOW 2>/dev/null | cut -d' ' -f5
+ else
+ echo $EPOCHSECONDS
+ fi
+}
+
+
+function bgnotify_find_sway_appid {
+ # output is "app_id,container_id", for example "Alacritty,1694"
+ # see example swaymsg output: https://github.com/ohmyzsh/ohmyzsh/files/13463939/output.json
+ if (( ${+commands[jq]} )); then
+ swaymsg -t get_tree | jq '.. | select(.type?) | select(.focused==true) | {app_id, id} | join(",")'
+ else
+ swaymsg -t get_tree | awk '
+ BEGIN { Id = ""; Appid = ""; FocusNesting = -1; Nesting = 0 }
+ {
+ # Enter a block
+ if ($0 ~ /.*{$/) Nesting++
+
+ # Exit a block. If Nesting is now less than FocusNesting, we have the data we are looking for
+ if ($0 ~ /^[[:blank:]]*}.*/) { Nesting--; if (FocusNesting > 0 && Nesting < FocusNesting) exit 0 }
+
+ # Save the Id, it is potentially what we are looking for
+ if ($0 ~ /^[[:blank:]]*"id": [0-9]*,?$/) { sub(/^[[:blank:]]*"id": /, ""); sub(/,$/, ""); Id = $0 }
+
+ # Save the Appid, it is potentially what we are looking for
+ if ($0 ~ /^[[:blank:]]*"app_id": ".*",?$/) { sub(/^[[:blank:]]*"app_id": "/, ""); sub(/",$/, ""); Appid = $0 }
+
+ # Window is focused, this nesting block contains the Id and Appid we want!
+ if ($0 ~ /^[[:blank:]]*"focused": true,?$/) { FocusNesting = Nesting }
+ }
+ END {
+ if (Appid != "" && Id != "" && FocusNesting != -1) print Appid "," Id
+ else print ""
+ }'
+ fi
+}
+
+function bgnotify_programid {
+ case "$TERM_PROGRAM" in
+ iTerm.app) echo 'com.googlecode.iterm2' ;;
+ Apple_Terminal) echo 'com.apple.terminal' ;;
+ esac
+}
+
+function bgnotify {
+ local title="$1"
+ local message="$2"
+ local icon="$3"
+ if (( ${+commands[terminal-notifier]} )); then # macOS
+ local term_id=$(bgnotify_programid)
+ terminal-notifier -message "$message" -title "$title" ${=icon:+-appIcon "$icon"} ${=term_id:+-activate "$term_id"} ${=bgnotify_extraargs:-} &>/dev/null
+ elif (( ${+commands[growlnotify]} )); then # macOS growl
+ growlnotify -m "$title" "$message" ${=bgnotify_extraargs:-}
+ elif (( ${+commands[notify-send]} )); then
+ notify-send "$title" "$message" ${=icon:+--icon "$icon"} ${=bgnotify_extraargs:-}
+ elif (( ${+commands[kdialog]} )); then # KDE
+ kdialog --title "$title" --passivepopup "$message" 5 ${=bgnotify_extraargs:-}
+ elif (( ${+commands[notifu]} )); then # cygwin
+ notifu /m "$message" /p "$title" ${=icon:+/i "$icon"} ${=bgnotify_extraargs:-}
+ fi
+}
+
+## Defaults
+
+# enable terminal bell on notify by default
+bgnotify_bell=${bgnotify_bell:-true}
+
+# notify if command took longer than 5s by default
+bgnotify_threshold=${bgnotify_threshold:-5}
+
+# bgnotify_appid is slow in macOS and the terminal ID won't change, so cache it at startup
+bgnotify_termid="$(bgnotify_appid)"