termsupport.zsh 4.53 KB
Newer Older
1
2
3
4
5
6
7
8
# Set terminal window and tab/icon title
#
# usage: title short_tab_title [long_window_title]
#
# See: http://www.faqs.org/docs/Linux-mini/Xterm-Title.html#ss3.1
# Fully supports screen, iterm, and probably most modern xterm and rxvt
# (In screen, only short_tab_title is used)
# Limited support for Apple Terminal (Terminal can't set window and tab separately)
9
function title {
10
  emulate -L zsh
11
  setopt prompt_subst
12

Marc Cornellà's avatar
Marc Cornellà committed
13
14
15
16
17
18
  [[ "$EMACS" == *term* ]] && return

  # if $2 is unset use $1 as default
  # if it is set and empty, leave it as is
  : ${2=$1}

19
20
21
22
23
  case "$TERM" in
    cygwin|xterm*|putty*|rxvt*|ansi)
      print -Pn "\e]2;$2:q\a" # set window name
      print -Pn "\e]1;$1:q\a" # set tab name
      ;;
24
    screen*|tmux*)
25
26
27
28
29
30
      print -Pn "\ek$1:q\e\\" # set screen hardstatus
      ;;
    *)
      if [[ "$TERM_PROGRAM" == "iTerm.app" ]]; then
        print -Pn "\e]2;$2:q\a" # set window name
        print -Pn "\e]1;$1:q\a" # set tab name
31
32
33
34
      else
        # Try to use terminfo to set the title
        # If the feature is available set title
        if [[ -n "$terminfo[fsl]" ]] && [[ -n "$terminfo[tsl]" ]]; then
Marc Cornellà's avatar
Marc Cornellà committed
35
36
37
38
          echoti tsl
          print -Pn "$1"
          echoti fsl
        fi
39
40
41
      fi
      ;;
  esac
42
43
44
45
}

ZSH_THEME_TERM_TAB_TITLE_IDLE="%15<..<%~%<<" #15 char left truncated PWD
ZSH_THEME_TERM_TITLE_IDLE="%n@%m: %~"
46
# Avoid duplication of directory in terminals with independent dir display
47
if [[ "$TERM_PROGRAM" == Apple_Terminal ]]; then
48
49
  ZSH_THEME_TERM_TITLE_IDLE="%n@%m"
fi
50

51
# Runs before showing the prompt
52
function omz_termsupport_precmd {
Marc Cornellà's avatar
Marc Cornellà committed
53
  [[ "$DISABLE_AUTO_TITLE" == true ]] && return
54
  title $ZSH_THEME_TERM_TAB_TITLE_IDLE $ZSH_THEME_TERM_TITLE_IDLE
55
56
}

57
# Runs before executing the command
58
function omz_termsupport_preexec {
Marc Cornellà's avatar
Marc Cornellà committed
59
60
  [[ "$DISABLE_AUTO_TITLE" == true ]] && return

61
  emulate -L zsh
62
63
  setopt extended_glob

64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
  # split command into array of arguments
  local -a cmdargs
  cmdargs=("${(z)2}")
  # if running fg, extract the command from the job description
  if [[ "${cmdargs[1]}" = fg ]]; then
    # get the job id from the first argument passed to the fg command
    local job_id jobspec="${cmdargs[2]#%}"
    # logic based on jobs arguments:
    # http://zsh.sourceforge.net/Doc/Release/Jobs-_0026-Signals.html#Jobs
    # https://www.zsh.org/mla/users/2007/msg00704.html
    case "$jobspec" in
      <->) # %number argument:
        # use the same <number> passed as an argument
        job_id=${jobspec} ;;
      ""|%|+) # empty, %% or %+ argument:
        # use the current job, which appears with a + in $jobstates:
        # suspended:+:5071=suspended (tty output)
        job_id=${(k)jobstates[(r)*:+:*]} ;;
      -) # %- argument:
        # use the previous job, which appears with a - in $jobstates:
        # suspended:-:6493=suspended (signal)
        job_id=${(k)jobstates[(r)*:-:*]} ;;
      [?]*) # %?string argument:
        # use $jobtexts to match for a job whose command *contains* <string>
        job_id=${(k)jobtexts[(r)*${(Q)jobspec}*]} ;;
      *) # %string argument:
        # use $jobtexts to match for a job whose command *starts with* <string>
        job_id=${(k)jobtexts[(r)${(Q)jobspec}*]} ;;
    esac
Marc Cornellà's avatar
Marc Cornellà committed
93

94
    # override preexec function arguments with job command
Marc Cornellà's avatar
Marc Cornellà committed
95
96
97
    if [[ -n "${jobtexts[$job_id]}" ]]; then
      1="${jobtexts[$job_id]}"
      2="${jobtexts[$job_id]}"
98
    fi
99
100
  fi

Marc Cornellà's avatar
Marc Cornellà committed
101
  # cmd name only, or if this is sudo or ssh, the next cmd
102
  local CMD=${1[(wr)^(*=*|sudo|ssh|mosh|rake|-*)]:gs/%/%%}
103
  local LINE="${2:gs/%/%%}"
Marc Cornellà's avatar
Marc Cornellà committed
104

105
  title '$CMD' '%100>...>$LINE%<<'
106
}
107

108
109
110
autoload -U add-zsh-hook
add-zsh-hook precmd omz_termsupport_precmd
add-zsh-hook preexec omz_termsupport_preexec
111
112


113
# Keep Apple Terminal.app's current working directory updated
Janosch Schwalm's avatar
Janosch Schwalm committed
114
# Based on this answer: https://superuser.com/a/315029
115
# With extra fixes to handle multibyte chars and non-UTF-8 locales
116
117

if [[ "$TERM_PROGRAM" == "Apple_Terminal" ]] && [[ -z "$INSIDE_EMACS" ]]; then
118
  # Emits the control sequence to notify Terminal.app of the cwd
119
120
  # Identifies the directory using a file: URI scheme, including
  # the host name to disambiguate local vs. remote paths.
121
  function update_terminalapp_cwd() {
122
    emulate -L zsh
123

124
125
126
127
    # Percent-encode the host and path names.
    local URL_HOST URL_PATH
    URL_HOST="$(omz_urlencode -P $HOST)" || return 1
    URL_PATH="$(omz_urlencode -P $PWD)" || return 1
128

129
    # Undocumented Terminal.app-specific control sequence
130
    printf '\e]7;%s\a' "file://$URL_HOST$URL_PATH"
131
132
133
  }

  # Use a precmd hook instead of a chpwd hook to avoid contaminating output
134
  add-zsh-hook precmd update_terminalapp_cwd
135
136
  # Run once to get initial cwd set
  update_terminalapp_cwd
137
fi