Files
q-tools/qcd_function

406 lines
11 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 ncd() {
# Next cd: cd to ../[next_folder]. ncd - to go to previous
local _current_pwd _iter_d _iter_prev _current_found
_current_pwd=$( basename $( pwd ) )
\cd ..
for _iter_d in *; do
if [ ! -d "$_iter_d" ]; then continue; fi
if [ -n "$_current_found" ]; then
\cd "$_iter_d"
return
fi
if [ "$_iter_d" = "$_current_pwd" ]; then
_current_found=true
if [ "$1" = "-" ]; then
if [ -z "$_iter_prev" ]; then
echo $_current_pwd was the first folder.
else
\cd "$_iter_prev";
return
fi
fi
fi
_iter_prev="$_iter_d"
done
if [ -n "$_current_found" ]; then
echo "$_current_pwd" was the last folder.
else
echo "$_current_pwd" was not found in ../
fi
}
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 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
}