2011-10-08 32 views

Trả lời

0

Một cách sẽ được sử dụng tput đếm cột của thiết bị đầu cuối của bạn, và subtrack số lượng ký tự mà sẽ được in trái và bên phải, sau đó sử dụng con số đó là số lượng khoảng trống giữa văn bản bên trái và bên phải. Sử dụng printf để tạo đường.

nhanh Ví dụ:

left="[${status}]\[email protected]\h:\w\$ " 
right="$(git symbolic-ref HEAD) $(date +%T)" 
spaces="$(($(tput cols) - ${#left} - ${#right}))" 
export PS1="$(printf "%s%${spaces}s\n" "$left" "$right")" 
+0

mmm, tôi chỉ thấy đây là trên line_ _same. Tôi không nghĩ rằng^sẽ làm việc. Âm thanh giống như bạn cần để có được bàn tay của bạn nhiều hơn trên '[n] curses' stuff. – c00kiemon5ter

9

Hãy thử như sau:

PS1='$(printf "%*s\r%s" $((COLUMNS-1)) "[$(git branch 2>/dev/null | grep '^*' | sed s/..//)] $(date +%H:%M:%S)" "[email protected]:$PWD$ ")' 

Lưu ý rằng bạn sẽ không bao giờ nhận được hành vi mà chính xác phù hợp với một zsh với chỉ bash. Trong trường hợp trên, tôi thấy các sự khác biệt sau đây:

  1. Phần bên phải của lời nhắc sẽ không bị xóa khi bạn chạy lệnh (accept-line sự kiện về zsh).
  2. Phần bên phải của lời nhắc sẽ bị xóa nếu bạn nhập nội dung nào đó và sau đó nhấn <C-u> hoặc <BS>.
  3. Phần bên phải của lời nhắc sẽ không được khôi phục nếu bạn nhập nội dung nào đó lên đó rồi xóa văn bản.
  4. Phần bên phải của lời nhắc sẽ không biến mất nếu bạn nhập thứ gì đó vào nó, mặc dù văn bản trong phần này sẽ bị ghi đè.
1

Hôm nay tôi đã tạo một cái gì đó tương tự theo cách sau. kiểm tra kỹ lưỡng đã không được thực hiện chưa ...

preprompt() { 
    rc=$? 
    c=31 
    [ $rc -eq 0 ] && c=32 

    PS1="\[$(color $c)\]$rc\[$(color 0)\] \t \w \$ " 
    # right "prompt" 
    # We cannot use $COLUMNS here, since in new shells the first prompt 
    # will get garbled then. Seems like the correct value of COLUMNS is 
    # in the shell init. 
    printf "%`tput cols`s`tput cr`" "${USER}@${HOST}" 
} 

PROMPT_COMMAND=preprompt 
1

Đoạn code dưới đây sẽ tạo ra một dấu nhắc mà trông giống như:

bash prompt with git status on right

Đó là phi trival để làm điều này trong bash do:

  • Readline mode string takes up characters before the prompt is printed, có nghĩa là các giải pháp printf sẽ không hoạt động trong một số trường hợp.Bởi vì điều này:
  • Loại bỏ tất cả (ví dụ như màu sắc) để tính toán một cách chính xác chiều dài của in bên tay phải phía nhắc
  • Cần phải sử dụng __git_ps1 để đối phó với git trường hợp cạnh
  • __git_ps1 chỉ xuất ra màu trong một số hoàn cảnh, và chỉ trong $PS1
  • phép màu trong __git_ps1 đầu ra trong khi loại bỏ các \[\] nhân vật từ đầu ra của nó (mà không thể được lồng vào nhau)
  • Bao bì toàn nhanh chóng RHS trong \[\] để đảm bảo rằng dấu nhắc không làm những điều kỳ lạ khi duyệt/chỉnh sửa/hoàn lệnh

#!/bin/bash 
# _options=$(shopt -op); set -exu # Save and set shell options for testing 
################## 
# Set the prompt # Sourced from .bashrc 
################## 

# Select git info displayed, see /usr/lib/git-core/git-sh-prompt for more 
export GIT_PS1_SHOWCOLORHINTS=1   # Make pretty colours inside $PS1 
export GIT_PS1_SHOWDIRTYSTATE=1   # '*'=unstaged, '+'=staged 
export GIT_PS1_SHOWSTASHSTATE=1   # '$'=stashed 
export GIT_PS1_SHOWUNTRACKEDFILES=1  # '%'=untracked 
export GIT_PS1_SHOWUPSTREAM="verbose"  # 'u='=no difference, 'u+1'=ahead by 1 commit 
export GIT_PS1_STATESEPARATOR=''   # No space between branch and index status 
export GIT_PS1_DESCRIBE_STYLE="describe" # Detached HEAD style: 
# describe  relative to older annotated tag (v1.6.3.1-13-gdd42c2f) 
# contains  relative to newer annotated tag (v1.6.3.2~35) 
# branch  relative to newer tag or branch (master~4) 
# default  exactly eatching tag 


# Sets prompt like: 
# [email protected]:~/prj/sample_app[exit]$     master*% u= | 30 Apr 22:27 
_set_bash_prompt() { 
    # Set left hand side of the prompt 
    PS1="\[email protected]\h:\w\$ " 

    # 
    # Git status 
    # 

    # Save current state of user shopt settings promptvars and extglob 
    local user_shopt 
    user_shopt=$(shopt -p promptvars extglob) 
    # __git_ps1 usually returns literal text "${__git_ps1_branch_name}" rather 
    # than the contained branch name, eg "master". This prevents calculating 
    # the length of the printable characers in the RHS string (used to move the 
    # cursor that many columns left from the terminal's right edge.) However if 
    # "shopt promptvars" is unset, __git_ps1 it will include the dereferenced 
    # branch name instead. 
    shopt -qu promptvars 
    # extglob is required for the ${variable//@(pattern)/} replacements 
    shopt -qs extglob 

    # Allow disabling git status and no error if __git_ps1 undefined 
    if [[ ! -v _disable_git_prompt && $(type -t __git_ps1 2>/dev/null) == function ]]; then 
    # __git_ps1 will only make pretty colours inside $PS1 
    local old_PS1=$PS1 
    __git_ps1 "" "" "%s" # force colour; no default round bracket (decorations) 

    # Strip "\[" and "\[": non-printable character markers. __git_ps1 outputs 
    # them however the whole of the RHS prompt needs to be included in these 
    # markers, and they can't be nested. 
    git=${PS1//@(\\@(\[|\]))/} 
    PS1=$old_PS1 
    fi 

    # 
    # Right hand side of prompt 
    # 
    local rhs="" # String to be printed on the right hand side of terminal 

    # Create a string like: "25 Apr 13:15" 
    local date_time 
    printf -v date_time "%(%e %b %H:%M)T" -1 # -1 is current time 

    # Format the RHS prompt 
    [[ -n $git ]] && rhs="$git | " #" 
    rhs+="\e[0;1;31m${date_time}" 

    # Strip ANSI CSI commands (eg colours) to enble counting the length of 
    # printable characters, giving offset of cursor from terminal RHS edge (from 
    # https://www.commandlinefu.com/commands/view/12043/remove-color-special-escape-ansi-codes-from-text-with-sed) 
    # Neither bash not sed support lookbehind zero-length assertions, so it's not 
    # possible to ignore "\\e", (ie a literal '\' followed by a literal 'e'), yet 
    # still remove "\e" (ie ESC) 
    local rhs_printable=${rhs//@(\\@(\[|]|[Ee]\[*([0-9;])[a-zA-Z]))/} 
    # or, in using sed (but requires exec): 
    # local rhs_printable=$(sed -e 's,\\[][]\|\\[Ee]\[\([0-9;]\)*[A-Za-z],,g' <<< "$rhs") 

    # Reference: https://en.wikipedia.org/wiki/ANSI_escape_code 
    local Save='\e[s' # Save cursor position 
    local Rest='\e[u' # Restore cursor to save point 

    # Save cursor position, jump to (right hand edge minus N columns) where N is 
    # the length of the printable RHS string. Print the RHS string, then return 
    # to the saved position and print the LHS prompt. 

    # Note: "\[" and "\]" are used so that bash can calculate the number of 
    # printed characters so that the prompt doesn't do strange things when 
    # command line editing/browsing/completion. Ensure that these are not nested. 
    PS1="\[\e[0m${Save}\e[$((COLUMNS - ${#rhs_printable}))G${rhs}${Rest}\]${PS1}" 

    eval "$user_shopt" 
} 

# eval "$_options"; unset _options # Restore previous shell options from line 2 
Các vấn đề liên quan