我正在尝试在vim状态行中显示实时字数。我这样做是通过在我的.vimrc中设置我的状态行并在其中插入一个函数。这个函数的想法是返回当前缓冲区中的字数。然后,此编号显示在状态行上。这应该很好地工作,因为状态线几乎在每个可能的机会更新,因此计数将始终保持“实时”。
问题是我当前定义的函数很慢,所以当它用于除最小文件之外的所有文件时,vim显然是缓慢的;由于这个功能经常执行。
总之,有没有人有一个聪明的技巧来产生一个快速计算当前缓冲区中的单词数并返回结果的函数?
答案 0 :(得分:25)
我真的很喜欢迈克尔邓恩上面的回答,但我发现当我编辑时,它导致我无法访问最后一列。所以我对这个功能进行了一些小改动:
function! WordCount()
let s:old_status = v:statusmsg
let position = getpos(".")
exe ":silent normal g\<c-g>"
let stat = v:statusmsg
let s:word_count = 0
if stat != '--No lines in buffer--'
let s:word_count = str2nr(split(v:statusmsg)[11])
let v:statusmsg = s:old_status
end
call setpos('.', position)
return s:word_count
endfunction
我已将其包含在我的状态行中而没有任何问题:
:set statusline=wc:%{WordCount()}
答案 1 :(得分:23)
这是罗德里戈奎罗的想法的可用版本。它不会更改状态栏,而是还原statusmsg变量。
function WordCount()
let s:old_status = v:statusmsg
exe "silent normal g\<c-g>"
let s:word_count = str2nr(split(v:statusmsg)[11])
let v:statusmsg = s:old_status
return s:word_count
endfunction
这似乎足够快,可以直接包含在状态行中,例如:
:set statusline=wc:%{WordCount()}
答案 2 :(得分:9)
保持当前行的计数以及其余缓冲区的单独计数。当您在当前行上键入(或删除)单词时,仅更新该计数,但显示当前行数和剩余缓冲区计数的总和。
当您更改行时,将当前行计数添加到缓冲区计数,计算当前行中的单词,a)设置当前行计数,b)从缓冲区计数中减去它。
定期重新计算缓冲区也是明智的(请注意,您不必一次计算整个缓冲区,因为您知道正在进行编辑的位置)。
答案 3 :(得分:4)
每当您停止输入一段时间(特别是updatetime
毫秒)时,这将重新计算字数。
let g:word_count="<unknown>"
fun! WordCount()
return g:word_count
endfun
fun! UpdateWordCount()
let s = system("wc -w ".expand("%p"))
let parts = split(s, ' ')
if len(parts) > 1
let g:word_count = parts[0]
endif
endfun
augroup WordCounter
au! CursorHold * call UpdateWordCount()
au! CursorHoldI * call UpdateWordCount()
augroup END
" how eager are you? (default is 4000 ms)
set updatetime=500
" modify as you please...
set statusline=%{WordCount()}\ words
享受!
答案 4 :(得分:3)
所以我写了:
func CountWords() exe "normal g\" let words = substitute(v:statusmsg, "^.*Word [^ ]* of ", "", "") let words = substitute(words, ";.*", "", "") return words endfunc
但它将信息打印到状态栏,所以我认为它不适合您的用例。但速度非常快!
答案 5 :(得分:2)
以下是对Abslom Daak的回答的改进,该回答也适用于视觉模式。
function! WordCount()
let s:old_status = v:statusmsg
let position = getpos(".")
exe ":silent normal g\<c-g>"
let stat = v:statusmsg
let s:word_count = 0
if stat != '--No lines in buffer--'
if stat =~ "^Selected"
let s:word_count = str2nr(split(v:statusmsg)[5])
else
let s:word_count = str2nr(split(v:statusmsg)[11])
end
let v:statusmsg = s:old_status
end
call setpos('.', position)
return s:word_count
endfunction
像以前一样包含在状态行中。这是一个右对齐状态行:
set statusline=%=%{WordCount()}\ words\
答案 6 :(得分:2)
我使用了稍微不同的方法。而不是确保单词计数功能特别快,我只在光标停止移动时调用它。这些命令将执行此操作:
:au CursorHold * exe "normal g\<c-g>"
:au CursorHoldI * exe "normal g\<c-g>"
也许并不是提问者想要的,但比这里的一些答案简单得多,而且对我的用例来说足够好了(在输入一两句话之后看一下字数统计)。
将updatetime
设置为较小的值也有帮助:
set updatetime=300
对于字数没有巨大的开销轮询,因为CursorHold
和CursorHoldI
仅在光标停止移动时一次,而不是每updatetime
毫秒。
答案 7 :(得分:1)
我是Vim脚本的新手,但我可能会建议
function WordCount()
redir => l:status
exe "silent normal g\<c-g>"
redir END
return str2nr(split(l:status)[11])
endfunction
因为它不会覆盖现有的状态行而更加清洁。
我发布的理由是指出这个函数有一个令人费解的错误:即它会破坏append命令。点击 A 会让你进入插入模式,光标位于行上最后一个字符的右边。但是,启用此自定义状态栏后,它会将您置于最终字符的左侧。
任何人都知道是什么原因造成的?
答案 8 :(得分:1)
因为vim现在原生支持此功能:
:echo wordcount().words
答案 9 :(得分:1)
vim
版本7.4.1042起自vim
版本7.4.1042起,您可以按照以下方式简单地更改statusline
:
set statusline+=%{wordcount().words}\ words
set laststatus=2 " enables the statusline.
vim-airline
中的字数 vim-airline
针对许多文件类型提供了标准的字数统计,截至撰写本文时:
asciidoc, help, mail, markdown, org, rst, tex ,text
如果vim-airline
中未显示字数,则通常是由于无法识别的文件类型。例如,at least for now不能由vim-airline
识别复合文件类型markdown.pandoc
来进行字数统计。可以通过如下修改.vimrc
来轻松地解决此问题:
let g:airline#extensions#wordcount#filetypes = '\vasciidoc|help|mail|markdown|markdown.pandoc|org|rst|tex|text'
set laststatus=2 " enables vim-airline.
\v
语句将覆盖默认的g:airline#extensions#wordcount#filetypes
变量。最后一行确保启用vim-airline
。
如有疑问,在发出以下命令后将返回打开文件的&filetype
:
:echo &filetype
这是一个元示例:
答案 10 :(得分:1)
这是对Michael Dunn's version的改进,缓存了字数,因此需要更少的处理。
function! WC()
if &modified || !exists("b:wordcount")
let l:old_status = v:statusmsg
execute "silent normal g\<c-g>"
let b:wordcount = str2nr(split(v:statusmsg)[11])
let v:statusmsg = l:old_status
return b:wordcount
else
return b:wordcount
endif
endfunction
答案 11 :(得分:1)
我的建议:
function! UpdateWordCount()
let b:word_count = eval(join(map(getline("1", "$"), "len(split(v:val, '\\s\\+'))"), "+"))
endfunction
augroup UpdateWordCount
au!
autocmd BufRead,BufNewFile,BufEnter,CursorHold,CursorHoldI,InsertEnter,InsertLeave * call UpdateWordCount()
augroup END
let &statusline='wc:%{get(b:, "word_count", 0)}'
我不确定这与其他一些解决方案的速度相比如何,但它肯定比大多数解决方案简单得多。
答案 12 :(得分:1)
我从写作功能的vim帮助页面中获取了大部分内容。
function! WordCount()
let lnum = 1
let n = 0
while lnum <= line('$')
let n = n + len(split(getline(lnum)))
let lnum = lnum + 1
endwhile
return n
endfunction
当然,和其他人一样,你需要:
:set statusline=wc:%{WordCount()}
我确信有人可以清理这个以使其更具vimmy(s:n而不仅仅是n?),但我相信基本功能就在那里。
编辑:
再看一遍,我真的很喜欢Mikael Jansson的解决方案。我不喜欢炮轰wc
(不便携,也许慢)。如果我们用上面的代码替换他的UpdateWordCount
函数(将我的函数重命名为UpdateWordCount
),那么我认为我们有更好的解决方案。
答案 13 :(得分:0)
如果其他人从谷歌来到这里,我修改了Abslom Daak的答案,与Airline合作。我将以下内容保存为
~/.vim/bundle/vim-airline/autoload/airline/extensions/pandoc.vim
并添加了
call airline#extensions#pandoc#init(s:ext)
到extensions.vim
let s:spc = g:airline_symbols.space
function! airline#extensions#pandoc#word_count()
if mode() == "s"
return 0
else
let s:old_status = v:statusmsg
let position = getpos(".")
let s:word_count = 0
exe ":silent normal g\<c-g>"
let stat = v:statusmsg
let s:word_count = 0
if stat != '--No lines in buffer--'
let s:word_count = str2nr(split(v:statusmsg)[11])
let v:statusmsg = s:old_status
end
call setpos('.', position)
return s:word_count
end
endfunction
function! airline#extensions#pandoc#apply(...)
if &ft == "pandoc"
let w:airline_section_x = "%{airline#extensions#pandoc#word_count()} Words"
endif
endfunction
function! airline#extensions#pandoc#init(ext)
call a:ext.add_statusline_func('airline#extensions#pandoc#apply')
endfunction
答案 14 :(得分:0)
Guy Gur-Ari的refinement的变体
function! StatuslineWordCount()
if !&l:spell
return ''
endif
if empty(getline(line('$')))
return ''
endif
let mode = mode()
if !(mode ==# 'v' || mode ==# 'V' || mode ==# "\<c-v>" || mode =~# '[ni]')
return ''
endif
let s:old_status = v:statusmsg
let position = getpos('.')
let stat = v:statusmsg
let s:word_count = 0
exe ":silent normal g\<c-g>"
try
if mode ==# 'v' || mode ==# 'V'
let s:word_count = split(split(v:statusmsg, ';')[1])[0]
elseif mode ==# "\<c-v>"
let s:word_count = split(split(v:statusmsg, ';')[2])[0]
elseif mode =~# '[ni]'
let s:word_count = split(split(v:statusmsg, ';')[2])[3]
end
" index out of range
catch /^Vim\%((\a\+)\)\=:E\%(684\|116\)/
return ''
endtry
let v:statusmsg = s:old_status
call setpos('.', position)
return "\ \|\ " . s:word_count . 'w'
endfunction
可以通过例如
添加到状态行 set statusline+=%.10{StatuslineWordCount()} " wordcount
答案 15 :(得分:0)
使用Steve Moyer提供的答案中的方法,我能够生成以下解决方案。这是一个相当不优雅的黑客,我害怕,我觉得必须有一个更简洁的解决方案,但它的工作原理,并且比每次更新状态行时简单计算缓冲区中的所有单词要快得多。我还应该注意,这个解决方案是独立于平台的,并不假设系统有'wc'或类似的东西。
我的解决方案不会定期更新缓冲区,但Mikael Jansson提供的答案将能够提供此功能。到目前为止,我还没有找到一个我的解决方案不同步的实例。但是我只是简单地对此进行了测试,因为准确的实时字数对我的需求并不重要。我用于匹配单词的模式也很简单,适用于简单的文本文档。如果有人对模式或任何其他建议有更好的想法,请随时发布答案或编辑此帖子。
我的解决方案:
"returns the count of how many words are in the entire file excluding the current line
"updates the buffer variable Global_Word_Count to reflect this
fu! OtherLineWordCount()
let data = []
"get lines above and below current line unless current line is first or last
if line(".") > 1
let data = getline(1, line(".")-1)
endif
if line(".") < line("$")
let data = data + getline(line(".")+1, "$")
endif
let count_words = 0
let pattern = "\\<\\(\\w\\|-\\|'\\)\\+\\>"
for str in data
let count_words = count_words + NumPatternsInString(str, pattern)
endfor
let b:Global_Word_Count = count_words
return count_words
endf
"returns the word count for the current line
"updates the buffer variable Current_Line_Number
"updates the buffer variable Current_Line_Word_Count
fu! CurrentLineWordCount()
if b:Current_Line_Number != line(".") "if the line number has changed then add old count
let b:Global_Word_Count = b:Global_Word_Count + b:Current_Line_Word_Count
endif
"calculate number of words on current line
let line = getline(".")
let pattern = "\\<\\(\\w\\|-\\|'\\)\\+\\>"
let count_words = NumPatternsInString(line, pattern)
let b:Current_Line_Word_Count = count_words "update buffer variable with current line count
if b:Current_Line_Number != line(".") "if the line number has changed then subtract current line count
let b:Global_Word_Count = b:Global_Word_Count - b:Current_Line_Word_Count
endif
let b:Current_Line_Number = line(".") "update buffer variable with current line number
return count_words
endf
"returns the word count for the entire file using variables defined in other procedures
"this is the function that is called repeatedly and controls the other word
"count functions.
fu! WordCount()
if exists("b:Global_Word_Count") == 0
let b:Global_Word_Count = 0
let b:Current_Line_Word_Count = 0
let b:Current_Line_Number = line(".")
call OtherLineWordCount()
endif
call CurrentLineWordCount()
return b:Global_Word_Count + b:Current_Line_Word_Count
endf
"returns the number of patterns found in a string
fu! NumPatternsInString(str, pat)
let i = 0
let num = -1
while i != -1
let num = num + 1
let i = matchend(a:str, a:pat, i)
endwhile
return num
endf
然后通过以下方式将其添加到状态行:
:set statusline=wc:%{WordCount()}
我希望这可以帮助任何在Vim中寻找实时字数的人。虽然一个并不总是准确的。或者当然g ctrl-g会为你提供Vim的字数!
答案 16 :(得分:0)
以 https://stackoverflow.com/a/60310471/11001018 为基础,我的建议是:
"new in vim 7.4.1042
let g:word_count=wordcount().words
function WordCount()
if has_key(wordcount(),'visual_words')
let g:word_count=wordcount().visual_words."/".wordcount().words
else
let g:word_count=wordcount().cursor_words."/".wordcount().words
endif
return g:word_count
endfunction
然后:
set statusline+=\ w:%{WordCount()},