Як я можу виділити відповідні імена "%" (наприклад, якщо / end, for / end), визначені matchit.vim у виборі?


10

Наразі мій Vim виділяє відповідні дужки, дужки, лапки тощо з блакитним фоном та білим переднім планом - курсор можна переміщувати між ними %. Завдяки моєму matchit.vim, я також можу переключатися %між if / end, for / end і т. Д. - однак вони не виділяються під час вибору.

Як я можу автоматично виділити ці відповідні пари при виборі, як, наприклад, це робиться автоматично в дужках?

Крім того, як я можу змінити колір фону, який використовується для цих пар, використовуючи :highlight?

Заздалегідь спасибі.


Я оновив відповідь @Tommy A нижче, щоб пояснити погано вказані matchit.vimгрупи та інші ситуації, коли %оператор ніколи не повертає курсор у вихідне положення. Перевірте відмінності в циклі "while". Кожен, хто читає цю тему, рекомендується використовувати цю версію, щоб уникнути нескінченних циклів:

function! s:get_match_lines(line) abort
  " Loop until `%` returns the original line number; abort if
  " (1) the % operator keeps us on the same line, or
  " (2) the % operator doesn't return us to the same line after some nubmer of jumps
  let a:tolerance=25
  let a:badbreak=1
  let a:linebefore=-1
  let lines = []
  while a:tolerance && a:linebefore != line('.')
    let a:linebefore=line('.')
    let a:tolerance-=1
    normal %
    if line('.') == a:line
      " Note that the current line number is never added to the `lines`
      " list. a:line is the input argument 'line'; a is the FUNCTION BUFFER
      let a:badbreak=0
      break
    endif
    call add(lines, line('.'))
  endwhile
  "Return to original line no matter what, return list of lines to highlight
  execute "normal ".a:line."gg"
  if a:badbreak==1
    return []
  else
    return lines
  endif
endfunction

function! s:hl_matching_lines() abort
  " `b:hl_last_line` prevents running the script again while the cursor is
  " moved on the same line.  Otherwise, the cursor won't move if the current
  " line has matching pairs of something.
  if exists('b:hl_last_line') && b:hl_last_line == line('.')
    return
  endif
  let b:hl_last_line = line('.')
  " Save the window's state.
  let view = winsaveview()
  " Delete a previous match highlight.  `12345` is used for the match ID.
  " It can be anything as long as it's unique.
  silent! call matchdelete(12345)
  " Try to get matching lines from the current cursor position.
  let lines = s:get_match_lines(view.lnum)
  if empty(lines)
    " It's possible that the line has another matching line, but can't be
    " matched at the current column.  Move the cursor to column 1 to try
    " one more time.
    call cursor(view.lnum, 1)
    let lines = s:get_match_lines(view.lnum)
  endif
  if len(lines)
    " Since the current line is not in the `lines` list, only the other
    " lines are highlighted.  If you want to highlight the current line as
    " well:
    " call add(lines, view.lnum)
    if exists('*matchaddpos')
      " If matchaddpos() is availble, use it to highlight the lines since it's
      " faster than using a pattern in matchadd().
      call matchaddpos('MatchLine', lines, 0, 12345)
    else
      " Highlight the matching lines using the \%l atom.  The `MatchLine`
      " highlight group is used.
      call matchadd('MatchLine', join(map(lines, '''\%''.v:val.''l'''), '\|'), 0, 12345)
    endif
  endif
  " Restore the window's state.
  call winrestview(view)
endfunction
function! s:hl_matching_lines_clear() abort
  silent! call matchdelete(12345)
  unlet! b:hl_last_line
endfunction

" The highlight group that's used for highlighting matched lines.  By
" default, it will be the same as the `MatchParen` group.
highlight default link MatchLine MatchParen
augroup matching_lines
  autocmd!
  " Highlight lines as the cursor moves.
  autocmd CursorMoved * call s:hl_matching_lines()
  " Remove the highlight while in insert mode.
  autocmd InsertEnter * call s:hl_matching_lines_clear()
  " Remove the highlight after TextChanged.
  autocmd TextChanged,TextChangedI * call s:hl_matching_lines_clear()
augroup END

2
Я знаю, що це старе питання, але я щойно побачив, як воно з’явилося на головній сторінці мить назад. Просто хочу згадати про мій новий збіг плагінів, призначений зробити саме це, більш надійним способом: github.com/andymass/vim-matchup (разом із багатьма іншими вдосконаленнями щодо matchit).
Маса

Виглядає дуже корисно, дякую, що зробили це! Я спробую це.
Люк Девіс

Відповіді:


12

Я думав, що ця ідея цікава, тому я її сфотографував. Це буде особливо корисно у щільних файлах, таких як HTML.

відповідні лінії

Наступний сценарій просто дозволяє matchit.vimробити те, що він робить, записуючи номери рядків. Пояснення містяться в коментарях сценарію.

matchlines.vim

function! s:get_match_lines(line) abort
  let lines = []

  " Loop until `%` returns the original line number
  while 1
    normal %
    if line('.') == a:line
      " Note that the current line number is never added to the `lines`
      " list.
      break
    endif
    call add(lines, line('.'))
  endwhile

  return lines
endfunction

function! s:hl_matching_lines() abort
  " `b:hl_last_line` prevents running the script again while the cursor is
  " moved on the same line.  Otherwise, the cursor won't move if the current
  " line has matching pairs of something.
  if exists('b:hl_last_line') && b:hl_last_line == line('.')
    return
  endif

  let b:hl_last_line = line('.')

  " Save the window's state.
  let view = winsaveview()

  " Delete a previous match highlight.  `12345` is used for the match ID.
  " It can be anything as long as it's unique.
  silent! call matchdelete(12345)

  " Try to get matching lines from the current cursor position.
  let lines = s:get_match_lines(view.lnum)

  if empty(lines)
    " It's possible that the line has another matching line, but can't be
    " matched at the current column.  Move the cursor to column 1 to try
    " one more time.
    call cursor(view.lnum, 1)
    let lines = s:get_match_lines(view.lnum)
  endif

  if len(lines)
    " Since the current line is not in the `lines` list, only the other
    " lines are highlighted.  If you want to highlight the current line as
    " well:
    " call add(lines, view.lnum)
    if exists('*matchaddpos')
      " If matchaddpos() is availble, use it to highlight the lines since it's
      " faster than using a pattern in matchadd().
      call matchaddpos('MatchLine', lines, 0, 12345)
    else
      " Highlight the matching lines using the \%l atom.  The `MatchLine`
      " highlight group is used.
      call matchadd('MatchLine', join(map(lines, '''\%''.v:val.''l'''), '\|'), 0, 12345)
    endif
  endif

  " Restore the window's state.
  call winrestview(view)
endfunction

function! s:hl_matching_lines_clear() abort
  silent! call matchdelete(12345)
  unlet! b:hl_last_line
endfunction


" The highlight group that's used for highlighting matched lines.  By
" default, it will be the same as the `MatchParen` group.
highlight default link MatchLine MatchParen

augroup matching_lines
  autocmd!
  " Highlight lines as the cursor moves.
  autocmd CursorMoved * call s:hl_matching_lines()
  " Remove the highlight while in insert mode.
  autocmd InsertEnter * call s:hl_matching_lines_clear()
  " Remove the highlight after TextChanged.
  autocmd TextChanged,TextChangedI * call s:hl_matching_lines_clear()
augroup END

Мені не дуже подобається, що це відбувається далі CursorMoved. Я думаю, що це краще як ключова карта, яку можна використовувати, коли мені це потрібно:

nnoremap <silent> <leader>l :<c-u>call <sid>hl_matching_lines()<cr>

Ви можете використовувати matchaddposфункцію замість цього. Це трохи швидше, і якщо ви все одно виділите всю лінію, це трохи спростить речі.
Карл Інгве Лервег

1
@ KarlYngveLervåg Добрий момент. Я підсвідомо уникаю цього, оскільки це все ще відносно нова функція (я думаю, v7.4.330), і це мене один раз вкусило в дупу. Я оновлю відповідь, щоб використовувати її.
Tommy A

Це абсолютно ідеально, велике спасибі! Гарна практика Vimscript також; спробує зрозуміти кожен рядок. Я думаю, що це може бути досить популярним, якби ви вперше написали подібну утиліту.
Люк Девіс

@LukeDavis Від цього спостерігається небажаний ефект, який я помітив: він розкрутить список стрибків. Я придумав спосіб її виправити, використовуючи <c-o>кількість разів, коли знайдено збіг, і він працює певним чином. Проблема в тому, що у matchit.vim є помилка, яка додає верхній рядок вікна до списку переходів. Це було визнано , але, схоже, немає поспіху виправити це.
Tommy A

@TommyA Привіт, ще раз дякую за цю утиліту. Я фактично знаходжу на своєму комп’ютері затримку з autocmd CursorMove досить мізерно. Я оновив вашу функцію, s:get_match_lines(line)щоб захистити від нескінченних циклів, що стало для мене великою проблемою в певних дивних контекстах. На жаль matchit.vim, повно вад. Дивіться мою редакцію вище та повідомте мені, чи є у вас якісь пропозиції; Я початківець вимскрипт.
Люк Девіс
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.