пейджерная программа вроде меньше, способная повторять верхние N строк

15

Есть ли способ заставить lessпрограмму повторять первую строку (или первые 2 строки) на каждой отображаемой странице?

Есть ли другая пейджерная программа, которая может это сделать?

Это будет приложение-убийца для просмотра таблиц базы данных, думаю mysqlили psqlили gqlplus...

Для тех из вас, кто не понимает, см. Скриншот внизу этой страницы . Я хочу повторить строку заголовка + горизонтальную полосу ascii.

filiprem
источник
О, мальчик, который просит расширения до меньшего, как точка Freeze Pane. Например, --freeze-pane 10,2 будет содержать 1 строку заголовков столбцов и 10 строк заголовков столбцов. Горизонтальная и вертикальная прокрутка сохранит заголовки строк и столбцов соответственно. Это было бы очень здорово использовать для psql пейджера (merlinmoncure.blogspot.com/2007/10/better-psql-with-less.html)
Гюнтер Шадов

Ответы:

12

Есть решение с использованием Vim.

Во-первых, нам нужен макрос Vim, который выполнит большую часть работы, я сохраню его в ~/.vim/plugin/less.vim:

" :Less
" turn vim into a pager for psql aligned results 
fun! Less()
  set nocompatible
  set nowrap
  set scrollopt=hor
  set scrollbind
  set number
  execute 'above split'
  " resize upper window to one line; two lines are not needed because vim adds separating line
  execute 'resize 1'
  " switch to lower window and scroll 2 lines down 
  wincmd j
  execute 'norm! 2^E'
  " hide statusline in lower window
  set laststatus=0
  " hide contents of upper statusline. editor note: do not remove trailing spaces in next line!
  set statusline=\  
  " arrows do scrolling instead of moving
  nmap ^[OC zL
  nmap ^[OB ^E
  nmap ^[OD zH
  nmap ^[OA ^Y
  nmap <Space> <PageDown>
  " faster quit (I tend to forget about the upper panel)
  nmap q :qa^M
  nmap Q :qa^M
endfun
command! -nargs=0 Less call Less()

Во-вторых, чтобы эмулировать пейджер, мне нужно вызвать vim, чтобы он:

  • читать стандартный ввод
  • но если аргумент задан в командной строке, прочитайте, что там происходит
  • работать в режиме только для чтения
  • пропустите все сценарии инициализации, но вместо этого выполните макрос Less, определенный выше

Я собрал это вместе как вспомогательный скрипт ~/bin/vimpager:

#!/bin/bash
what=-
test "$@" && what="$@"
exec vim -u NONE -R -S ~/.vim/plugin/less.vim -c Less $what

В-третьих, мне нужно переопределить переменную окружения $ PAGER, но только для psql (добавьте это к моему ~/.bash_aliases):

if which vimpager &>/dev/null; then
  alias psql='PAGER=vimpager psql';
fi
filiprem
источник
Это прекрасно, я попробовал. Теперь было бы хорошо, если бы вместо этих номеров строк можно было добавлять также заголовки строк.
Гюнтер Шадов
4

Вы пробовали режим SQL в Emacs / XEmacs?

Это, конечно, не так просто, как moreили less, но он делает то, о чем вы просили, оставляя строку заголовка при прокрутке результатов по вертикали и горизонтали.

BSD
источник
Спасибо, я не знаю Emacs, но это звучит интересно. В конечном итоге мне понадобится сценарий оболочки, который будет запускать emacs, запускать там psql (с заданными коннект-параметрами), включать sql-режим и делать то, что я хочу (всегда замораживать верхние 2 строки, когда результаты запроса превышают размер экрана) , какие-нибудь намеки на это?
filiprem
3

Это очень сильно заимствует из принятого ответа, но добавляет ...

  • Быстрая прокрутка
  • Невозможно случайно прокрутить в шапку
  • Подсветка синтаксиса (некоторые кредиты принадлежат здесь )
    • Положительные / отрицательные числа, даты, время, NULLИстина / Ложь (и T / F, Y / N, Да / Нет)
    • Номера строк, если они у вас есть перед символом трубы.
  • Текст справки
  • Поддержка Vim, которая включена в Git для Windows
  • Не угрожайте обновить представление, если буфер стандартного ввода изменяется

Некоторые порции, возможно, придется настроить для вашего конкретного вывода, так как я не использую psql. У меня также есть немного отличающиеся вспомогательные функции для моих целей, но они похожи на те, что приняты в ответе.

Пример ввода

  | ID |   First   |     Last     | Member | Balance |
--+----+-----------+--------------+--------+---------+
 1|  4 | Tom       | Hanks        | False  |    0.00 |
 2| 12 | Susan     | Patterson    | True   |   10.00 |
 3| 23 | Harriet   | Langford-Wat | False  |    0.00 |
 4|  8 | Jerry     |     NULL     | True   | -382.94 |
[… More rows …]
10| 87 | Horace    | Weaver       | False  |   47.52 |

Код

" :HeadPager
" Turn vim into a pager with a header row
" Adapted from /unix//a/27840/143088
fun! HeadPager()
    " If you didn't get three lines, shortcut out
    if line('$') < 3
        set nocompatible
        nmap <silent> q :qa!<c-M>
        nmap <silent> Q :qa!<c-M>
        return
    endif

    set noswapfile
    set nocompatible
    set nowrap
    set scrollopt=hor
    set scrollbind

    " Hide statusline in lower window
    set laststatus=0
    " Explain mapped chars in status line.
    set statusline=\ \ \ Q\ to\ quit\.\ Arrows\ or\ mousewheel\ to\ scroll\.\ \(Vim\ commands\ work\,\ too\.\)

    " Delete/copy header lines
    silent execute '1,2d'

    " Split screen with new buffer (opens at top)
    execute 'new'

    " Switch to upper split
    wincmd k

    " Paste the header over the blank line
    execute 'norm! Vp'

    " Header highlighting
    syn match Pipe "|"
    hi def Pipe ctermfg=blue
    syn match Any /[^|]\+/
    hi def Any ctermfg=yellow

    " Switch back to lower split for scrolling
    wincmd j

    " Set lower split height to maximum
    execute "norm! \<c-W>_"

    " Syntax highlighting
    syn cluster CellContents contains=None
    syn match Pipe "|" contained nextgroup=@CellContents skipwhite
    hi def Pipe ctermfg=blue

    " Start with newline or |. End right before next | or EOL
    syn region Cell start=/\v(^|\|)\s*/ end=/\v(\||$)\@=/ contains=LineNumber,Pipe

    syn match NumPos /\v\+?\d+(,?\d{3})*\.?\d*\ze *(\||$)\@=/ contained
    syn match NumNeg   /\v-\d+(,?\d{3})*\.?\d*\ze *(\||$)\@=/ contained
    syn match NumZero         /\v[+-]?0+\.?0*\ze *(\||$)\@=/  contained
    hi def NumPos ctermfg=cyan
    hi def NumNeg ctermfg=red
    hi def NumZero ctermfg=NONE
    syn cluster CellContents add=NumPos,NumNeg,NumZero

    syn match DateVal /\v\d{4}-\d{2}-\d{2}/ contained nextgroup=TimeVal skipwhite
    syn match TimeVal /\v\d{1,2}:\d{2}(:\d{2})?(\.\d+)?(Z| ?\c[AP]M)?\ze *(\||$)\@=/ contained
    hi def DateVal ctermfg=magenta
    hi def TimeVal ctermfg=magenta
    syn cluster CellContents add=DateVal,TimeVal

    syn match TrueVal /\v\c(t(rue)?|y(es)?)\ze *(\||$)\@=/ contained
    syn match FalseVal /\v\c(f(alse)?|no?)\ze *(\||$)\@=/ contained
    hi def TrueVal ctermfg=green
    hi def FalseVal ctermfg=red
    syn match NullVal /\v\cnull?\ze *(\||$)\@=/ contained
    hi def NullVal ctermbg=gray ctermfg=black
    syn cluster CellContents add=TrueVal,FalseVal,NullVal

    syn match LineNumber /^ *\d\+/ contained
    hi def LineNumber ctermfg=yellow

    " Arrows do scrolling instead of moving
    nmap <silent> <Up> 3<c-Y>
    nmap <silent> <Down> 3<c-E>
    nmap <silent> <Left> zH
    nmap <silent> <Right> zL
    nmap <Space> <PageDown>
    " Faster quit (I tend to forget about the upper panel)
    nmap <silent> q :qa!<c-M>
    nmap <silent> Q :qa!<c-M>

    " Ignore external updates to the buffer
    autocmd! FileChangedShell */fd/*
    autocmd! FileChangedRO */fd/*
endfun
command! -nargs=0 HeadPager call HeadPager()
Майкл - Где Клэй Ширки
источник
2

Вы можете использовать несколько «регионов» в screen:

$ cat screenrc.sql
escape ^aa  # adjust as needed
bind q quit # to quickly exit
screen 0 less ${FILE}
screen 1 less ${FILE}
split  # create two regions
focus top # starting with the top region
resize 4  # make it four lines (one for screen line, one for less prompt)
select 0  # display window 0
focus bottom  # in the bottom region
select 1  # display window 1 and focus here

Тогда вам просто нужно установить переменную окружения $ FILE:

$ FILE=$HOME/.bash_profile screen -mc screenrc.sql
Arcege
источник
1
это почти то, что я хотел, но (а) верхнее окно не прокручивается вправо, поэтому оно бесполезно для широких столов
filiprem
Не уверен, что вы подразумеваете под «бесполезным для широких столов»; Экран может расширяться до размера терминала (если он не запускает команду fitэкрана). Я думал, что вы не хотите, чтобы верх прокручивался. Когда я проверил это самостоятельно, оба окна прокручиваются как они должны. Верхняя прокручивает две строки (1-2, 3-4, 5-6 и т. Д.), А нижняя прокручивает по мере необходимости. Какое поведение вы видите /
Arcege
0

Вы можете добавить число перед «вперед», и оно будет прокручивать N строк, а не всю длину. Поэтому, если в окне вашего терминала 40 строк, наберите, 38fчтобы начать прокрутку только 38 строк, оставив последние 2 строки с последней «страницы». Из справочной страницы:

   SPACE or ^V or f or ^F
          Scroll forward N  lines,  default  one  window  (see  option  -z
          below).   If  N  is  more  than  the screen size, only the final
          screenful is displayed.  Warning: some systems use ^V as a  spe‐
          cial literalization character.

   z      Like  SPACE,  but  if  N is specified, it becomes the new window
          size.

   b or ^B or ESC-v
          Scroll backward N lines,  default  one  window  (see  option  -z
          below).   If  N  is  more  than  the screen size, only the final
          screenful is displayed.
Arcege
источник
1
Мне нужно сохранить первые , а не последние N строк. Как "заморозить верхние N строк" в Google Spreadsheet.
filiprem
Ах, прости. Я не знаю, будет ли это делать утилита. Тогда я предлагаю следующее: используйте screenили tmuxдля создания двух панелей, измените размер первой или двух строк ( resize 2), а затем запустите lessна второй панели, lessкак обычно. Вы можете настроить это как скрипт с конкретным файлом .screenrc. Смотрите альтернативный ответ.
Arcege