#!/usr/bin/env bash


# usage info
usage() {
  cat <<EOF
  Usage: git recall [options]
  Options:
    -d, --date              Show logs for last n days.
    -a, --author            Author name.
    -f, --fetch             fetch commits.
    -h, --help              This message.
    -v, --version           Show version.
    --                      End of options.
EOF
}


# Global variables
SINCE="1 days ago" # show logs for last day by default
AUTHOR=""
FETCH=false
GIT_FORMAT=""
GIT_LOG=""
COMMITS=""
COMMITS_UNCOL=() # commits without colors
LESSKEY=false
SED_BIN="" # Sed option to use according OS.
VERSION="1.1.10"

# Are we in a git repo?
if [[ ! -d ".git" ]] && ! git rev-parse --git-dir &>/dev/null; then
  echo "abort: not a git repository." 1>&2
  exit 1
fi

# Parse options
while [[ "$1" =~ ^- && ! "$1" == "--" ]]; do
  case $1 in
    -v | --version )
      echo "$version"
      exit
      ;;
    -d | --date )
      SINCE="$2 days ago"
      shift;
      ;;
    -a | --author )
      AUTHOR="$2"
      shift
      ;;
    -f | --fetch )
      FETCH=true
      ;;
    -h | --help )
      usage
      exit
      ;;
    * )
      echo "abort: unknown argument" 1>&2
      exit 1
  esac
  shift
done
if [[ "$1" == "--" ]]; then shift; fi


# Colored output.
function colored() {
  GREEN=$(tput setaf 4)
  YELLOW=$(tput setaf 3)
  NORMAL=$(tput sgr0)
  REVERSE=$(tput rev)
} 

# Uncolored output.
function uncolored() {
  GREEN=""
  YELLOW=""
  NORMAL=""
  REVERSE=""
}

# Enable colors if supported by terminal.
if [[ -t 1 ]] && [[ -n "$TERM" ]] && which tput &>/dev/null && tput colors &>/dev/null; then
  ncolors=$(tput colors)
  if [[ -n "$ncolors" ]] && [[ "$ncolors" -ge 8 ]] ; then
    colored
  else
    uncolored
  fi
else
  uncolored
fi

# Check if lesskey is installed.
if command -v lesskey &> /dev/null; then
    LESSKEY=true
fi

# Set AUTHOR to current user if no param passed or display for all users if param equal to "all". 
if [[ ! -n $AUTHOR ]]; then
    AUTHOR=$(git config user.name 2>/dev/null)
elif [[ $AUTHOR = "all" ]]; then
    AUTHOR=".*"
fi

# Fetch changes before.
if [[ $FETCH == true ]]; then
    echo "${GREEN}Fetching changes...${NORMAL}"
    git fetch --all &> /dev/null
    tput cuu1
    tput ed # clear screen
fi

# Log template.
GIT_FORMAT="%Cred%h%Creset -%C(yellow)%d%Creset %s %Cgreen(%cr) %C(bold blue)<%an>%Creset"

# Log command.
GIT_LOG="git log --pretty=format:'${GIT_FORMAT}'  
           --author \"$AUTHOR\"
           --since \"$SINCE\" --abbrev-commit"

# Change temporary the IFS to store GIT_LOG's output into an array.
IFS=$'\n'
COMMITS=($(eval ${GIT_LOG} 2>/dev/null))
unset IFS

NI=${#COMMITS[@]} # Total number of items.
SN=$(( `tput lines` - 1 )) # Screen's number of lines.
CN=$(tput cols) # Screen's number of columns.
TN=$(( $NI < $((SN -1)) ? $NI : $((SN -1)))) # Number of lines that we will display.
OFFSET=0 #Incremented by one each time a commit's length is higher than teminal width.

# If there is no items, exit.
if [[ $NI = 0 ]]; then
    if [[ $AUTHOR = ".*" ]]; then
	echo "${YELLOW}All contributors did nothing during this period.${NORMAL}" && exit 0
    else
	echo "${YELLOW}The contributor \"${AUTHOR}\" did nothing during this period.${NORMAL}" && exit 0
    fi
fi

# Set correct sed option according OS's type
case "$OSTYPE" in
    darwin*) SED_BIN="sed -E" ;;
    *)       SED_BIN="sed -r" ;;
esac

# Create array with uncolred commits (removing escape sequences using sed)
for elt in "${COMMITS[@]}"
do
    ELT="$(echo "$elt" | $SED_BIN "s/\x1B\[([0-9]{1,2}(;[0-9]{1,2})?)?[m|K]//g")" # remove colors escape codes
    COMMITS_UNCOL+=("$ELT")
done
# Add +1 to OFFSET if a commit's length is bigger than the current terminal session's width. (This is to fix a redraw issue)
for C in "${COMMITS_UNCOL[@]}"
do
    if [[ ${#C} -gt $CN ]]; then
	OFFSET=$(( OFFSET + 1 ))
    fi
done

# Set keys.
au="`echo -e '\e[A'`" # arrow up
au_1="k"              # arrow up
ad="`echo -e '\e[B'`" # arrow down
ad_1="j"              # arrow down
ec="`echo -e '\e'`"   # escape
nl="`echo -e '\n'`"   # newline
nl_1="e"              # expand
co="c"                # checkout

# Create a temporary lesskey file to change the keybindings so the user can use the TAB key to quit less. (more convenient)  
if [[ $LESSKEY = true ]]; then
    echo "\t quit" | lesskey -o $HOME/.lsh_less_keys_tmp -- - &> /dev/null
fi

## Get commit's diff.
function get_diff() {
    ELT="$(echo "${COMMITS_UNCOL[$CP-1]}")"
    DIFF_TIP=${ELT:0:7}
    DIFF_CMD="git show $DIFF_TIP --color=always"
    DIFF=$(eval ${DIFF_CMD} 2>/dev/null)
    tmp_diff="$(echo "$DIFF" | $SED_BIN "s/\x1B\[([0-9]{1,2}(;[0-9]{1,2})?)?[m|K]//g")" # remove colors escape codes
    off=$(echo "$tmp_diff" | grep -c ".\{$CN\}") #Number of lines in the diff that are longer than terminal width.
    DIFF_LINES_NUMBER="$(echo "$DIFF" | wc -l)"
    DIFF_LINES_NUMBER=$(( DIFF_LINES_NUMBER + off ))
}

## This function will print the diff according the commit's tip. If the diff is too long, the result will be displayed using 'less'. 
function print_diff() {
    get_diff # get commit's diff
    if [[ $(( TN + DIFF_LINES_NUMBER + OFFSET )) -ge $(( `tput lines` - 1 )) ]]; then
	if [[ $LESSKEY = true ]]; then
	    echo "$DIFF" | less -r -k $HOME/.lsh_less_keys_tmp
	else 
	    echo "$DIFF" | less -r
	fi
	tput ed # Clear screen
    else 
	stop=false
	tput ed
	for i in `seq 1 $TN`
	do
	    echo -n "$NORMAL"
	    [[ $CP == "$i" ]] && echo -n "$REVERSE"
	    echo "${COMMITS[$i - 1]}"
	    [[ $CP == "$i" ]] && echo "$DIFF"
	done
	# Wait for user action.
	while ! $stop
	do
	    read -sn 1 key
	    case "$key" in
		"$nl" | "$nl_1")
		    stop=true
		    ;;
		"q")
		    stop=true
		    END=true
		    ;;
	    esac
	done
	[[ $END = false ]] && tput cuu $(( TN + DIFF_LINES_NUMBER + OFFSET )) && tput ed
	[[ $END = true ]] && tput cuu 1
    fi
}

function do_checkout(){
    ELT="$(echo "${COMMITS_UNCOL[$CP-1]}")"
    DIFF_TIP=${ELT:0:7}
	eval "git checkout $DIFF_TIP 2> /dev/null"
}

# Calculate OFFSET to avoid bad redraw.
function calculate_offset {
    tmp=1
    index=$(( SI -1 ))
    while [[ $tmp -lt $SN ]]
    do
    	el=${COMMITS_UNCOL[$index]}
    	if [[ ${#el} -gt $CN ]] && [[ $CP -lt $((SN -1)) ]]; then
    	    OFFSET_2=$(( OFFSET_2 + 1 ))
    	    tmp=$(( tmp + 1 ))
    	fi
    	tmp=$(( tmp + 1 ))
    	index=$(( index + 1 ))
    done
}

{ # capture stdout to stderr

tput civis # hide cursor.
CP=1 # current position
SI=1 # index
END=false # end while loop
EXT=0 # Used to extend the number of lines to display.

## Loops, reads inputs and prints commits until user presses 'q' to exit or TAB to show the diff.
while ! $END
do
   END_INDEX=0 # Index for the last item to display
   # When the number of item is higher than screen's number of lines, OFFSET_2 is recalculated each time we select a new item
   # Set last index to print. (based on OFFSET)
   if [[  $TN == $NI ]]; then 
       END_INDEX=$TN
       OFFSET_2=$OFFSET
   elif [[  $TN == $(( SN - 1 )) ]]; then
       # Calculate new OFFSET.
       if [[ $OFFSET != 0 ]]; then 
	   [[ $CP -lt $((SN -1)) ]] && OFFSET_2=0
	   EXT=1
	   calculate_offset
       fi
       END_INDEX=$(( TN + SI -1 + EXT - OFFSET_2 ))
   fi

   # Loop and echo commits
   for i in `seq $SI $END_INDEX`
   do
      echo -n "$NORMAL"
      [[ $CP == $i ]] && echo -n "$REVERSE"
      echo "${COMMITS[$i - 1]}"
   done

   read -sn 1 key
   [[ "$key" == "$ec" ]] &&
   {
      read -sn 2 k2
      key="$key$k2"
   }

   case "$key" in

      "$au" | "$au_1")
         CP=$(( CP - 1 ))
         [[ $CP == 0 ]] && [[ $SI == 1 ]] && [[ $TN == $(( SN - 1 )) ]] && CP=$NI && SI=$(( NI - SN + 2 + OFFSET_2 ))
         [[ $CP == 0 ]] && [[ $SI == 1 ]] && [[ $TN == $NI ]] && CP=$TN
         [[ $CP == $(( SI - 1 )) ]] && [[ $SI != 1 ]] && SI=$(( SI - 1 ))

	 [[ $TN != $(( SN - 1 )) ]] && tput cuu $(( TN + OFFSET_2 ))
	 [[ $TN == $(( SN - 1 )) ]] && tput cuu $(( TN + EXT ))
         [[ $SI != 1 ]] && tput ed # clear screen
         ;;

      "$ad" | "$ad_1")
	 CP=$(( CP + 1 ))
         [[ $CP == $(( NI + 1 )) ]] && CP=1 && SI=1 
         [[ $CP == $(( SN + SI - 1 + EXT - OFFSET_2 )) ]] && [[ $TN == $(( SN - 1 )) ]] && SI=$(( SI + 1 ))

	 [[ $TN != $(( SN - 1 )) ]] && tput cuu $(( TN + OFFSET_2 ))
	 [[ $TN == $(( SN - 1 )) ]] && tput cuu $(( TN + EXT ))
         [[ $SI != 1 ]] && tput ed # clear screen
         [[ $SI = 1 ]] && [[ $CP = 1 ]] && tput ed # clear screen
         ;;

      "$nl" | "$nl_1")
	 [[  $TN == $NI ]] && tput cuu $(( TN + OFFSET_2 ))
	 [[  $TN != $NI ]] && tput cuu $(( TN + EXT ))
         print_diff 
         ;;
	  "$co")
		 si=false
		 END=true
		 do_checkout
		 tput cuu 1 #move cursor up one line. (remove extra line)
	     ;;

      "q")
         si=false
         END=true
	 tput cuu 1 #move cursor up one line. (remove extra line)
         ;;

      * )
	 tput cuu $(( TN + OFFSET_2 ))
   esac

done

# remove temporary less keybindings
[[ $LESSKEY = true ]] && rm $HOME/.lsh_less_keys_tmp 

tput cnorm # unhide cursor
echo "$NORMAL" # normal colors

} >&2 # END capture