Como eu corrijo minha quebra de alerta de cor bash?

9

Eu defini um prompt do bash (usando PROMPT_FUNCTION) da seguinte forma:

function get_hg_prompt_prefix() {
    local APPLIED_COLOR=$1; shift
    local UNAPPLIED_COLOR=$1; shift
    local ALERT_COLOUR=$1; shift
    local TEXTCOLOR=$1; shift
    local mercurial_prompt_line="{{patches|join(:)|pre_applied(${APPLIED_COLOR})|post_applied(${TEXTCOLOR})|pre_unapplied(${UNAPPLIED_COLOR})|post_unapplied(${TEXTCOLOR})}\n\r}"
    local mercurial_status_prompt="{ ${ALERT_COLOUR}{status}${TEXTCOLOR}}"

    echo "$(hg prompt "${mercurial_prompt_line}" 2>/dev/null)$(hg prompt "${mercurial_status_prompt}" 2>/dev/null)"
}

function set_prompt() {
    bright='\[[01m\]'
    colors_reset='\[[00m\]'
    HOSTCOLOR=${colors_reset}='\[[34m\]'
    USERCOLOR=${colors_reset}='\[[01m\]'
    TEXTCOLOR=${colors_reset}='\[[32m\]'
    APPLIED_COLOR=${colors_reset}='\[[32m\]'
    UNAPPLIED_COLOR=${colors_reset}='\[[37m\]'
    ALERT_COLOUR=${colors_reset}='\[[31m\]'

    hg_status="$(get_hg_prompt_prefix $APPLIED_COLOR $UNAPPLIED_COLOR $ALERT_COLOUR $TEXTCOLOR)"
    ps1_prefix="${hg_status}$colors_reset($bright$(basename $VIRTUAL_ENV)$colors_reset) "
    PROMPTEND='$'
    PS1="${ps1_prefix}${USERCOLOR}\u${colors_reset}${TEXTCOLOR}@${colors_reset}${HOSTCOLOR}\h${colors_reset}${TEXTCOLOR} (\W) ${PROMPTEND}${colors_reset} "
}

PROMPT_COMMAND=set_prompt

Em geral, isso me dá um prompt de várias linhas que exibe algumas informações de status do hg, bem como o meu virtualenv atual, parecendo (sem cor) assim:

buggy-wins.patch
 ! (saas) user@computer (~) $ 

O problema é que isso está estragando o cálculo do tamanho do prompt (eu acho!) e causando estranhos problemas de quebra de terminal e posicionamento do cursor. Por exemplo, em um terminal de 80 caracteres, aqui está o prompt que vejo (o caractere cercado por ** é o local do cursor):

~) $ **a**nis) crose@chris-rose (~

Em terminais largos o suficiente para exibir o prompt, a quebra de linha ocorre muito antes do que deveria; aqui está o maior texto que eu posso colocar na linha primeiro do prompt em uma janela de terminal com largura de 108 caracteres (novamente, o ** marca a localização do meu cursor):

 **(**advanis) crose@chris-rose (~) $ sdkfjlskdjflksdjff

Quando a linha termina, ela sobrescreve o aviso. A segunda linha de entrada é executada diretamente na borda do terminal e, em seguida, é envolvida corretamente.

Então, claramente, algo está bagunçando a largura do prompt. Como posso causar bash para determinar o comprimento da seqüência PS1 não de acordo com os códigos de escape ANSI, mas de acordo com o comprimento real exibido do prompt?

    
por Chris R 17.12.2010 / 17:06

1 resposta

21

bash usa \[ \] para determinar o "comprimento exibido": o texto entre esses dois escapes é considerado não imprimível e não é contado no comprimento total; tudo o mais é.

Parece haver um problema com suas variáveis: bright='\[[01m\]' na verdade não inclui um caractere ESC, portanto, [01m é impresso como texto normal, mas não é contado no comprimento. Deve ser '\[\e[01m\]' . O mesmo para todas as outras variáveis.

Relacionados:

  • no Bash, você pode colocar \$(hg_status) em $PS1 diretamente, sem a necessidade de um PROMPT_COMMAND separado.
por 17.12.2010 / 17:53

Tags