登录
首页 >  Golang >  Go问答

使用 vim 运行 gofmt,而无需插件

来源:stackoverflow

时间:2024-02-20 12:00:26 475浏览 收藏

小伙伴们有没有觉得学习Golang很有意思?有意思就对了!今天就给大家带来《使用 vim 运行 gofmt,而无需插件》,以下内容将会涉及到,若是在学习中对其中部分知识点有疑问,或许看了本文就能帮到你!

问题内容

我想在 vim 上保存时运行 gofmt,而不安装任何插件。这是我尝试过的(灵感来自https://gist.github.com/tbrisbout/a91ac3419440cde40c5f54dc32c94427):

function! GoFmt()
  let file = expand('%')
  silent execute "!gofmt -w " . file
  edit!
endfunction

command! GoFmt call GoFmt()
augroup go_autocmd
  autocmd BufWritePost *.go GoFmt
augroup END

这仅在没有格式错误时有效。但是,如果代码包含错误,它会在屏幕底部显示错误消息,并且(看起来)它以文本形式出现在缓冲区上,因此整个代码都会被破坏。 有没有一种简单的方法可以在 vim 上处理此类工作?


正确答案


gofmt 既是一个代码格式化程序一个 linter,这一设计决策使得将其与 vim 集成变得更加困难。实际上,这意味着事情可以朝三种方向发展:

  • 您的代码很完美,没有任何反应,
  • 您的代码存在格式问题,已重新格式化,
  • 您的代码存在语法问题,将输出错误列表。

例如,我们可以天真地尝试通过 :help 'formatprg'gq 和朋友使用 gofmt 但最终可能会用垃圾覆盖我们的代码,例如:

<standard input>:4:2: expected statement, found '.'
<standard input>:5:3: expected '}', found 'eof'

太高了。就像您的情况一样,我们可以执行 u 来撤消,但这并不有趣。我想我们将不得不解决 gofmt 的糟糕设计。

第一步:切换到 :help bufwritepre。我们已经看到 gofmt 可以处理 stdin,它允许我们格式化缓冲区和文件。这很方便,因为在写入文件后对其进行格式化会无缘无故地再次写入该文件,并迫使我们在 vim 中重新加载它......而这一切似乎都是浪费。 :help bufwritepost 最好保留用于不影响 vim 状态的事情。

function! gofmt()
  echomsg "hello"
endfunction

command! gofmt call gofmt()

augroup go_autocmd
  autocmd bufwritepre *.go gofmt
augroup end

第二步:通过 gofmt 过滤整个缓冲区。

function! gofmt()
  silent %!gofmt
endfunction
  • 最好的情况:什么也没有发生,或者缓冲区被格式化内容覆盖。
  • 最坏的情况:整个缓冲区被错误报告替换。

第三步:通过基本撤消“处理”最坏的情况。如果外部命令返回错误,我们可以通过 :help v:shell_error 获取错误并执行需要执行的操作。

function! gofmt()
  silent %!gofmt
  if v:shell_error > 0
    silent undo
  endif
endfunction

第四步:尝试将光标保持在原位。

function! gofmt()
  let saved_view = winsaveview()
  silent %!gofmt
  if v:shell_error > 0
    silent undo
  endif
  call winrestview(saved_view)
endfunction

请参阅 :help winsaveview():help winrestview()

第五步:如果适用,创建包含 gofmt 报告的错误的快速修复列表。 :help getline() 为我们提供了缓冲区的所有行(因此是所有错误),在列表中我们修改每个项目,以便文件名是当前文件名,而不是无用的 < 标准输入 >。我们将该列表提供给 :help :cexpr 以在撤消过滤器之前创建快速修复列表。

function! gofmt()
  let saved_view = winsaveview()
  silent %!gofmt
  if v:shell_error > 0
    cexpr getline(1, '$')->map({ idx, val -> val->substitute('<standard input>', expand('%'), '') })
    silent undo
  endif
  call winrestview(saved_view)
endfunction

这一步有一点“画出*嚎叫的其余部分”的感觉,但它实际上只是一个简单的 :help map() 中的一个简单的 :help replacement() 。有关 { foo, bar -> baz } 语法,请参阅 :help lambda

第六步也是最后一步:如果存在任何有效错误,请使用 :help :cwindow 打开快速修复窗口。

function! GoFmt()
  let saved_view = winsaveview()
  silent %!gofmt
  if v:shell_error > 0
    cexpr getline(1, '$')->map({ idx, val -> val->substitute('<standard input>', expand('%'), '') })
    silent undo
    cwindow
  endif
  call winrestview(saved_view)
endfunction

以上就是本文的全部内容了,是否有顺利帮助你解决问题?若是能给你带来学习上的帮助,请大家多多支持golang学习网!更多关于Golang的相关知识,也可关注golang学习网公众号。

声明:本文转载于:stackoverflow 如有侵犯,请联系study_golang@163.com删除
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>