Como posso fazer o cd para o diretório irmão anterior / próximo?

9

Muitas vezes tenho um layout de diretório de projeto como este

project
'-- component-a
|   '-- files...
'-- component-b
|   '-- files...
'-- component-c
    '-- files...

Geralmente estarei trabalhando em um dos diretórios component , porque é onde estão os arquivos. Quando eu, em seguida, retornar ao shell, muitas vezes preciso simplesmente mover para um diretório secundário, especialmente quando preciso fazer algumas alterações não-scriptáveis em todos os componentes. Nesses casos, eu nem vou me importar com o diretório irmão anterior em que eu trabalharei, ou o próximo diretório irmão.

Posso definir um comando prev ou next que simplesmente cd me no diretório anterior ou no próximo diretório (por alfabeto ou qualquer outro)? Porque digitar cd ../com<TAB><Arrow keys> o tempo todo está ficando um pouco antigo.

    
por Esteis 20.06.2012 / 21:04

3 respostas

8

Não use a solução commandlinefu da outra resposta : é insegura¹e ineficiente.² Em vez disso, se você está usando bash , apenas use as seguintes funções. Para torná-los persistentes, coloque-os no seu .bashrc . Note que eu uso glob order porque é embutido e fácil. Em geral, a ordem de glob é alfabética na maioria das localidades. Você receberá uma mensagem de erro se não houver nenhum diretório próximo ou anterior para ir. Em particular, você verá o erro se tentar next ou prev no diretório raiz, / .

## bash and zsh only!
# functions to cd to the next or previous sibling directory, in glob order

prev () {
    # default to current directory if no previous
    local prevdir="./"
    local cwd=${PWD##*/}
    if [[ -z $cwd ]]; then
        # $PWD must be /
        echo 'No previous directory.' >&2
        return 1
    fi
    for x in ../*/; do
        if [[ ${x#../} == ${cwd}/ ]]; then
            # found cwd
            if [[ $prevdir == ./ ]]; then
                echo 'No previous directory.' >&2
                return 1
            fi
            cd "$prevdir"
            return
        fi
        if [[ -d $x ]]; then
            prevdir=$x
        fi
    done
    # Should never get here.
    echo 'Directory not changed.' >&2
    return 1
}

next () {
    local foundcwd=
    local cwd=${PWD##*/}
    if [[ -z $cwd ]]; then
        # $PWD must be /
        echo 'No next directory.' >&2
        return 1
    fi
    for x in ../*/; do
        if [[ -n $foundcwd ]]; then
            if [[ -d $x ]]; then
                cd "$x"
                return
            fi
        elif [[ ${x#../} == ${cwd}/ ]]; then
            foundcwd=1
        fi
    done
    echo 'No next directory.' >&2
    return 1
}

¹ Não manipula todos os nomes de diretórios possíveis. A análise da saída ls nunca é segura .

² cd provavelmente não precisa ser muito eficiente, mas 6 processos são um pouco excessivos.

    
por 21.06.2012 / 00:35
4

A seguinte função permite mudar para diretórios irmãos (função bash)

function sib() {
    ## sib  search sibling directories 
    ##   prompt for choice (when two or more directories are found, current dir is removed from choices) 
    ##   change to directory after selection 
    local substr=$1
    local curdir=$(pwd)
    local choices=$(find .. -maxdepth 1 -type d -name "*${substr}*" | grep -vE '^..$' | sed -e 's:../::' | grep -vE "^${curdir##*/}$" | sort)
    if [ -z "$choices" ]; then
        echo "Sibling directory not found!"
        return
    fi
    local count=$(echo "$choices" | wc -l)
    if [[ $count -eq 1 ]]; then
        cd ../$choices
        return 
    fi
    select dir in $choices; do
        if [ -n "$dir" ]; then
            cd ../$dir
        fi
        break
    done
}

Um exemplo de uso:

$ tree
  .
  ├── component-aaa-01
  ├── component-aaa-02
  ├── component-bbb-01
  ├── component-bbb-02
  ├── component-ccc-01
  ├── component-ccc-02
  └── component-ccc-03
  7 directories, 0 files
  $ cd component-aaa-01/
  $ sib bbb-01
  $ pwd
  component-bbb-01
  $ sib bbb
  $ pwd
  component-bbb-02
  $ sib ccc
  1) component-ccc-01
  2) component-ccc-02
  3) component-ccc-03
  #? 3
  $ pwd
  component-ccc-03
  $ sib 01
  1) component-aaa-01
  2) component-bbb-01
  3) component-ccc-01
  #? 2
  $ pwd
  component-bbb-01
    
por 18.04.2014 / 13:44
2

Eu encontrei uma dica em commandlinefu .com Estou repostando isso aqui para torná-lo mais localizável, com uma explicação e um comando next adicionado enquanto estou nisso.

alias prev='cd ../"$(ls -F .. | grep '/' | grep -B1 -xF "${PWD##*/}/" | head -n 1)"'
alias next='cd ../"$(ls -F .. | grep '/' | grep -A1 -xF "${PWD##*/}/" | tail -n 1)"'

A mágica está no bloco '$ (...). Ele canaliza alguns comandos um para o outro da seguinte forma:

ls -F .. |   # list items in parent dir; '-F' requests filetype indicators
grep '/' |   # select the directories (written as  'mydir/')
grep -B1 -xF "${PWD##*/}/" |   # search for the name of the current directory in the output;
                               # print it and the line preceding it
head -n 1    # the first of those two lines contains the name of the previous sibling
    
por 20.06.2012 / 21:04