382 lines
10 KiB
Plaintext
382 lines
10 KiB
Plaintext
function cd_history () {
|
|
# Function that replaces "cd". It stores visited folder in ~/.bash_cdhistory
|
|
local old
|
|
local p
|
|
local b
|
|
if [ -z "$1" ]
|
|
then \cd "$HOME"
|
|
return
|
|
fi
|
|
touch "$HOME/.bash_cdhistory"
|
|
old=$( tail -n 149 "$HOME/.bash_cdhistory" )
|
|
echo "$old" > "$HOME/.bash_cdhistory"
|
|
\cd "$1"
|
|
p=$( pwd )
|
|
b=$( basename "$p" )
|
|
echo $b":"$p >> "$HOME/.bash_cdhistory"
|
|
}
|
|
alias cd=cd_history
|
|
|
|
function hcd() {
|
|
# History cd. Run without arguments to see list of entries, give number to change cwd.
|
|
local d
|
|
|
|
if [ -z "$1" ]
|
|
then tail -n 20 "$HOME/.bash_cdhistory" | head -n 19 | cut -d: -f2 | tac | cat -n | sed 's,^ \+,,' | tac
|
|
return
|
|
fi
|
|
d=$( tail -n 20 "$HOME/.bash_cdhistory" | head -n 19 | cut -d: -f2 | tac | cat -n | sed 's,^ \+,,' | grep -h "^$1 " )
|
|
d=${d/* /}
|
|
if [ ! -z "$d" ]
|
|
then \cd "$d"
|
|
fi
|
|
}
|
|
|
|
function qcd() {
|
|
# cd command, that jumps to folders visited in the near history, or user bookmarked folders
|
|
local OPTIND
|
|
local OPTARG
|
|
local opt
|
|
local case
|
|
local d
|
|
[ -e ~/.qcd ] || touch ~/.qcd
|
|
[ -e ~/.bash_cdhistory ] || touch ~/.bash_cdhistory
|
|
while getopts ae:hiILl:m opt
|
|
do case "$opt" in
|
|
a)
|
|
# Adding
|
|
local name=${!OPTIND}
|
|
if [ -z "$name" ]
|
|
then name=$( basename $( pwd ))
|
|
fi
|
|
|
|
echo "$name":$( pwd )
|
|
echo "$name":$( pwd ) >> ~/.qcd
|
|
return
|
|
;;
|
|
i)
|
|
case="-i"
|
|
;;
|
|
I)
|
|
echo "See ~/.bash_completion for possible duplicates"
|
|
echo '
|
|
have qcd &&
|
|
_qcd()
|
|
{
|
|
local cur
|
|
local d
|
|
COMPREPLY=()
|
|
cur=${COMP_WORDS[COMP_CWORD]}
|
|
if [[ "$cur" == -* ]]; then
|
|
COMPREPLY=( $( compgen -W "-a -i -h -l -L -m" -- $cur ) )
|
|
return 0
|
|
fi
|
|
COMPREPLY=( $( compgen -W "$( grep -h ^"$cur" ~/.qcd <( tac ~/.bash_cdhistory ) | cut -d: -f1 )" ) )
|
|
}
|
|
complete -F _qcd qcd
|
|
' >> ~/.bash_completion
|
|
. /etc/bash_completion
|
|
touch ~/.qcd ~/.bash_cdhistory
|
|
return
|
|
;;
|
|
L)
|
|
echo ==History ~/.bash_cdhistory:
|
|
cat ~/.bash_cdhistory
|
|
echo
|
|
echo ==Saved ~/.qcd:
|
|
cat ~/.qcd
|
|
return
|
|
;;
|
|
l)
|
|
local d=$( grep $case -h ^"$OPTARG" ~/.qcd <( tac ~/.bash_cdhistory ) | head -n 1 )
|
|
d=${d/*:/}
|
|
echo $d
|
|
return
|
|
;;
|
|
e)
|
|
local d=$( grep $case -h ^"$OPTARG" ~/.qcd <( tac ~/.bash_cdhistory ) | head -n 1 )
|
|
d="${d/*:/}"
|
|
echo QCD=$d
|
|
QCD="$d"
|
|
return
|
|
;;
|
|
m)
|
|
local IFS=$'\n'
|
|
touch ~/.qcd ~/.bash_cdhistory
|
|
touch ~/.qcd.tmp ~/.bash_cdhistory.tmp
|
|
for line in $( cat ~/.bash_cdhistory );
|
|
do if [ -d "${line/*:/}" ];
|
|
then echo "$line" >> ~/.bash_cdhistory.tmp
|
|
fi
|
|
done
|
|
mv ~/.bash_cdhistory.tmp ~/.bash_cdhistory
|
|
for line in $( cat ~/.qcd );
|
|
do if [ -d "${line/*:/}" ];
|
|
then echo "$line" >> ~/.qcd.tmp
|
|
fi
|
|
done
|
|
mv ~/.qcd.tmp ~/.qcd
|
|
return
|
|
;;
|
|
h)
|
|
echo 'qcd [-hiLm]|[-al] [name]
|
|
Version: 2015-10-13
|
|
Change current working path based on the list ~/.qcd
|
|
Keeps a history of folders visited in ~/.bash_cdhistory
|
|
|
|
-a [name] Adds the path to the list
|
|
You may add the name of the path, but when omitted
|
|
the basename will be used
|
|
-e [name] Show the match, store in variable QCD
|
|
-i Case insensitive search, must come first.
|
|
-I Install, with autocompletion (use only once)
|
|
-l [name] Show the match, but do not change
|
|
-L Lists the paths
|
|
-m Maintain the list, deleting non-existing entries'
|
|
return
|
|
;;
|
|
esac
|
|
done
|
|
shift $(($OPTIND - 1))
|
|
if [ -z "$1" ]
|
|
then [[ $OPTIND -gt 1 ]] && return
|
|
\cd; return
|
|
fi
|
|
unset OPTSTRING
|
|
unset OPTIND
|
|
if [ "$1" = "-" ]
|
|
then d=$( tail -n 1 ~/.bash_cdhistory )
|
|
d=${d/*:/}
|
|
\cd "$d"
|
|
return
|
|
fi
|
|
d=$( grep $case -h ^"$1" ~/.qcd <( tac ~/.bash_cdhistory ) | head -n 1 )
|
|
d=${d/*:/}
|
|
if [ ! -z "$d" ]
|
|
then \cd "$d"
|
|
fi
|
|
|
|
}
|
|
|
|
function whenfilechanges() {
|
|
# Run a command when file(s) time stamp changes
|
|
echo "Version: 2015-10-13"
|
|
[ -z "$2" ] && { echo 'Usage: whenfilechanges "file" "some" "command"
|
|
Follows the modification time of the "file" and runs the command
|
|
at every change.
|
|
The "file" may also be a glob e.g. "*tex" '
|
|
return
|
|
}
|
|
|
|
local fname
|
|
local forced
|
|
local otime
|
|
local ntime
|
|
local i
|
|
|
|
fname=($1)
|
|
echo Waiting for ${#fname[@]} files to change: ${fname[@]}
|
|
shift 1
|
|
echo "Command: \"$@\""
|
|
otime=()
|
|
for ((i=0;i<${#fname[@]};i++))
|
|
do [ -e "${fname[$i]}" ] || {
|
|
echo "File: ${fname[$i]} not found!"
|
|
return
|
|
}
|
|
otime[$i]=$( stat -c %Z "${fname[$i]}" )
|
|
done
|
|
while :
|
|
do ntime=()
|
|
for ((i=0;i<${#fname[@]};i++))
|
|
do ntime[$i]=$( stat -c %Z "${fname[$i]}" )
|
|
[ "${ntime[$i]}" -ne "${otime[$i]}" ] && {
|
|
echo "${fname[$i]} changed:"
|
|
otime[$i]=$( stat -c %Z "${fname[$i]}" )
|
|
eval "$@"
|
|
echo -n "Waiting for file changes... "
|
|
date
|
|
}
|
|
done
|
|
echo -ne Waiting.. $( date )\\r
|
|
read -t 2 forced
|
|
[ $? -eq 0 ] && eval "$@"
|
|
done
|
|
|
|
}
|
|
|
|
function gcd() {
|
|
# guess cd, find first match in folder, or ask if multiple hits
|
|
local cdto
|
|
{ cdto=$( python -c '
|
|
import sys,os,re
|
|
if len(sys.argv)==1 or sys.argv[-1]=="-h":
|
|
print("Arguments: [dir/]pattern_to_match_folders")
|
|
if sys.argv[-1]=="-h":
|
|
sys.exit(0)
|
|
def G():
|
|
return "\033[32;1m"
|
|
def R():
|
|
return "\033[31;1m"
|
|
def Z():
|
|
return "\033[0m"
|
|
def CS():
|
|
return "1234567890qwertyuiop"
|
|
class getch:
|
|
def get(self):
|
|
fd = sys.stdin.fileno()
|
|
old_settings = termios.tcgetattr(fd)
|
|
try:
|
|
tty.setraw(sys.stdin.fileno())
|
|
ch = sys.stdin.read(1)
|
|
finally:
|
|
termios.tcsetattr(fd, termios.TCSADRAIN, old_settings)
|
|
return ch
|
|
|
|
pattern=".".join(sys.argv[1:])
|
|
|
|
pat_dir=os.path.dirname(pattern)
|
|
if pat_dir=="":
|
|
pat_dir="."
|
|
pat_base=os.path.basename(pattern)
|
|
if not os.path.exists(pat_dir):
|
|
print(pat_dir+": "+R()+"No such folder"+Z())
|
|
sys.exit(1)
|
|
current_folders=[d for d in os.listdir(pat_dir) if os.path.isdir(d)]
|
|
current_folders=[d for d in current_folders if not d.startswith(".")]
|
|
current_folders.sort()
|
|
|
|
match_case=re.compile(".*"+pat_base+".*")
|
|
repl_case=re.compile("("+pat_base+")")
|
|
|
|
matching_folders=[d for d in current_folders if match_case.match(d)]
|
|
matches=[repl_case.sub(G()+"\g<1>"+Z(),d) for d in matching_folders]
|
|
|
|
if len(matching_folders)==0:
|
|
match_nocase=re.compile(".*"+pat_base+".*",re.I)
|
|
repl_nocase=re.compile("("+pat_base+")",re.I)
|
|
matching_folders=[d for d in current_folders if match_nocase.match(d)]
|
|
matches=[repl_nocase.sub(G()+"\g<1>"+Z(),d) for d in matching_folders]
|
|
|
|
if len(matching_folders)==1:
|
|
print(matches[0])
|
|
sys.stderr.write(os.path.join(pat_dir,matching_folders[0]))
|
|
sys.exit(0)
|
|
if len(matching_folders)==0:
|
|
print(R()+"no matches"+Z())
|
|
sys.exit(1)
|
|
for i,m in enumerate(matches[0:20]):
|
|
print(CS()[i]+": "+m)
|
|
if len(matches)>20:
|
|
print("Skipping rest...")
|
|
import termios,tty
|
|
ch=getch()
|
|
key_in=ch.get()
|
|
try:
|
|
key_index=CS().index(key_in)
|
|
key_match=matching_folders[key_index]
|
|
except ValueError,IndexError:
|
|
sys.exit(1)
|
|
sys.stderr.write(os.path.join(pat_dir,key_match))
|
|
' "$@" 2>&1 1>&$out); } {out}>&1 ; \cd "$cdto"
|
|
}
|
|
|
|
function rmv () {
|
|
# mv files/folders with rsync (shows speed and progress)
|
|
local sources
|
|
sources=()
|
|
# remove / from ends, if target is an existing folder
|
|
for (( i=1; i<=$(($#-1)); i++ ))
|
|
do if [ -d "${@: -1}" ]
|
|
then sources+=("${!i%/}")
|
|
else sources+=("${!i}")
|
|
fi
|
|
done
|
|
rsync -vaP --remove-source-files "${sources[@]}" "${@: -1}"
|
|
# remove empty folders from sources (not last argument)
|
|
for (( i=1; i<=$(($#-1)); i++ ))
|
|
do if [ -d "${!i}" ]
|
|
then find "${!i}" -depth -type d -exec rmdir \{\} \;
|
|
fi
|
|
done
|
|
}
|
|
|
|
function igrep () {
|
|
# Interactive grep
|
|
local args
|
|
args="-i -e "
|
|
while true
|
|
do read -e -i "$args" args
|
|
echo $@
|
|
grep --color=auto $args "$@"
|
|
if [ "$?" -ne 0 ]
|
|
then echo _no_matches_ | grep --color=auto ".*"
|
|
fi
|
|
done
|
|
|
|
}
|
|
|
|
function ised () {
|
|
# Interactive sed
|
|
local args
|
|
if [ ! -f "$1" ]; then echo must give atleast one file; return 1;fi;
|
|
args="-e s/string//g"
|
|
while true
|
|
do read -e -i "$args" args
|
|
echo input files: "$@"
|
|
eval sed $args "$@"
|
|
if [ "$?" -ne 0 ]
|
|
then echo Error output | grep --color=auto ".*"
|
|
fi
|
|
done
|
|
|
|
}
|
|
|
|
function foldermenu_prompt {
|
|
# Function to add in PS prompt variable, enabling foldermenu support
|
|
[ -f .foldermenu ] && {
|
|
foldermenu -lf 10 $@ || echo -n "*"
|
|
} || {
|
|
true
|
|
}
|
|
}
|
|
|
|
function _tools-update {
|
|
# update self
|
|
[[ -z "$TOOLSPATH" ]] && return
|
|
[[ -w "$TOOLSPATH" ]] || return
|
|
"$TOOLSPATH"/rc -u
|
|
}
|
|
|
|
function qbg {
|
|
# Run a program quiet, and backgrounded
|
|
"$@" &> /dev/null &
|
|
}
|
|
|
|
function set_term_title {
|
|
# set term in byobu/screen xterm etc..
|
|
[ -z "$1" ] && {
|
|
echo -e '\033k'$HOSTNAME'\033\\'
|
|
} || {
|
|
echo -e '\033k'$1'\033\\'
|
|
}
|
|
}
|
|
|
|
function path_add_current {
|
|
# Add current or given folder to PATH
|
|
[[ "$1" = "-h" ]] && {
|
|
echo 'Add the current folder in PATH, or pass a folder name to be added'
|
|
return
|
|
}
|
|
local p=$(pwd)
|
|
[[ -z "$1" ]] || {
|
|
p=$( readlink -e "$1" )
|
|
}
|
|
[[ -z "$p" ]] && {
|
|
echo Path "$1" not found >&2
|
|
return
|
|
}
|
|
export PATH=$( echo ${p}:$PATH | awk -F: '{for (i=1;i<=NF;i++) { if ( !x[$i]++ ) printf("%s:",$i); }}' | sed 's,:\+$,,g' )
|
|
echo PATH=$PATH
|
|
}
|