Created
October 29, 2016 21:23
-
-
Save RamonGilabert/e45b61a0f20bed95993a0b8679e036c5 to your computer and use it in GitHub Desktop.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
# COPY OF: https://github.com/sjl/z-fish JUST CHANGING Z FOR S. ALL THE CREDIT GOES TO SJL. | |
# Maintains a jump-list of the directories you actually use | |
# | |
# Usage: | |
# * s foo # goes to most frecent dir matching foo | |
# * s foo bar # goes to most frecent dir matching foo and bar | |
# * s -r foo # goes to highest ranked dir matching foo | |
# * s -t foo # goes to most recently accessed dir matching foo | |
# * s -l foo # list all dirs matching foo (by frecency) | |
function s -d "Jump to a recent directory." | |
set -l datafile "$HOME/.s" | |
# add entries | |
if [ "$argv[1]" = "--add" ] | |
set -e argv[1] | |
# $HOME isn't worth matching | |
[ "$argv" = "$HOME" ]; and return | |
set -l tempfile (mktemp $datafile.XXXXXX) | |
test -f $tempfile; or return | |
# maintain the file | |
awk -v path="$argv" -v now=(date +%s) -F"|" ' | |
BEGIN { | |
rank[path] = 1 | |
time[path] = now | |
} | |
$2 >= 1 { | |
if( $1 == path ) { | |
rank[$1] = $2 + 1 | |
time[$1] = now | |
} else { | |
rank[$1] = $2 | |
time[$1] = $3 | |
} | |
count += $2 | |
} | |
END { | |
if( count > 1000 ) { | |
for( i in rank ) print i "|" 0.9*rank[i] "|" time[i] # aging | |
} else for( i in rank ) print i "|" rank[i] "|" time[i] | |
} | |
' $datafile ^/dev/null > $tempfile | |
mv -f $tempfile $datafile | |
# tab completion | |
else | |
if [ "$argv[1]" = "--complete" ] | |
awk -v q="$argv[2]" -F"|" ' | |
BEGIN { | |
if( q == tolower(q) ) nocase = 1 | |
split(q,fnd," ") | |
} | |
{ | |
if( system("test -d \"" $1 "\"") ) next | |
if( nocase ) { | |
for( i in fnd ) tolower($1) !~ tolower(fnd[i]) && $1 = "" | |
if( $1 ) print $1 | |
} else { | |
for( i in fnd ) $1 !~ fnd[i] && $1 = "" | |
if( $1 ) print $1 | |
} | |
} | |
' "$datafile" 2>/dev/null | |
else | |
# list/go | |
set -l last '' | |
set -l list 0 | |
set -l typ '' | |
set -l fnd '' | |
while [ (count $argv) -gt 0 ] | |
switch "$argv[1]" | |
case -- '-h' | |
echo "s [-h][-l][-r][-t] args" >&2 | |
return | |
case -- '-l' | |
set list 1 | |
case -- '-r' | |
set typ "rank" | |
case -- '-t' | |
set typ "recent" | |
case -- '--' | |
while [ "$argv[1]" ] | |
set -e argv[1] | |
set fnd "$fnd $argv[1]" | |
end | |
case '*' | |
set fnd "$fnd $argv[1]" | |
end | |
set last $1 | |
set -e argv[1] | |
end | |
[ "$fnd" ]; or set list 1 | |
# if we hit enter on a completion just go there | |
[ -d "$last" ]; and cd "$last"; and return | |
# no file yet | |
[ -f "$datafile" ]; or return | |
set -l tempfile (mktemp $datafile.XXXXXX) | |
test -f $tempfile; or return | |
set -l target (awk -v t=(date +%s) -v list="$list" -v typ="$typ" -v q="$fnd" -v tmpfl="$tempfile" -F"|" ' | |
function frecent(rank, time) { | |
dx = t-time | |
if( dx < 3600 ) return rank*4 | |
if( dx < 86400 ) return rank*2 | |
if( dx < 604800 ) return rank/2 | |
return rank/4 | |
} | |
function output(files, toopen, override) { | |
if( list ) { | |
if( typ == "recent" ) { | |
cmd = "sort -nr >&2" | |
} else cmd = "sort -n >&2" | |
for( i in files ) if( files[i] ) printf "%-10s %s\n", files[i], i | cmd | |
if( override ) printf "%-10s %s\n", "common:", override > "/dev/stderr" | |
} else { | |
if( override ) toopen = override | |
print toopen | |
} | |
} | |
function common(matches, fnd, nc) { | |
for( i in matches ) { | |
if( matches[i] && (!short || length(i) < length(short)) ) short = i | |
} | |
if( short == "/" ) return | |
for( i in matches ) if( matches[i] && i !~ short ) x = 1 | |
if( x ) return | |
if( nc ) { | |
for( i in fnd ) if( tolower(short) !~ tolower(fnd[i]) ) x = 1 | |
} else for( i in fnd ) if( short !~ fnd[i] ) x = 1 | |
if( !x ) return short | |
} | |
BEGIN { split(q, a, " ") } | |
{ | |
if( system("test -d \"" $1 "\"") ) next | |
print $0 >> tmpfl | |
if( typ == "rank" ) { | |
f = $2 | |
} else if( typ == "recent" ) { | |
f = t-$3 | |
} else f = frecent($2, $3) | |
wcase[$1] = nocase[$1] = f | |
for( i in a ) { | |
if( $1 !~ a[i] ) delete wcase[$1] | |
if( tolower($1) !~ tolower(a[i]) ) delete nocase[$1] | |
} | |
if( wcase[$1] > oldf ) { | |
cx = $1 | |
oldf = wcase[$1] | |
} else if( nocase[$1] > noldf ) { | |
ncx = $1 | |
noldf = nocase[$1] | |
} | |
} | |
END { | |
if( cx ) { | |
output(wcase, cx, common(wcase, a, 0)) | |
} else if( ncx ) output(nocase, ncx, common(nocase, a, 1)) | |
} | |
' "$datafile") | |
if [ $status -gt 0 ] | |
rm -f "$tempfile" | |
else | |
mv -f "$tempfile" "$datafile" | |
[ "$target" ]; and cd "$target" | |
end | |
end | |
end | |
end | |
function __s_init -d 'Set up automatic population of the directory list for s' | |
functions fish_prompt | grep -q 's --add' | |
if [ $status -gt 0 ] | |
functions fish_prompt | sed -e '$ i\\ | |
s --add "$PWD"' | . | |
end | |
end | |
__s_init |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment