You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
go.nvim/lua/go/impl.lua

184 lines
4.9 KiB
Lua

-- local ts_utils = require 'nvim-treesitter.ts_utils'
local utils = require("go.utils")
local log = utils.log
local vfn = vim.fn
local impl = "impl" -- GoImpl f *Foo io.Writer
-- use ts to get name
local function get_type_name()
local name = require("go.ts.go").get_struct_node_at_pos()
if name == nil then
name = require("go.ts.go").get_type_node_at_pos()
end
utils.log(name)
if name == nil then
return ""
end
local node_name = name.name
-- let move the cursor to end of line of struct name
local dim = name.dim.e
-- let move cursor
local r, c = dim.r, dim.c
utils.log("move cusror to ", r, c)
vim.api.nvim_win_set_cursor(0, { r, c })
return node_name
end
local function get_interface_name()
local name = require("go.ts.go").get_interface_node_at_pos()
utils.log(name)
if name == nil then
return nil
end
local node_name = name.name
-- let move the cursor to end of line of struct name
local dim = name.dim.e
-- let move cursor
local r, c = dim.r, dim.c
utils.log("move cusror to ", r, c)
vim.api.nvim_win_set_cursor(0, { r, c })
local pkg = require("go.package").pkg_from_path(nil, vim.api.nvim_get_current_buf())
log(pkg[1])
if pkg then
return pkg[1] .. "." .. node_name
end
end
local run = function(...)
require("go.install").install(impl)
local impl_cmd = "impl"
local iface = ""
local recv_name = ""
local arg = { ... }
utils.log(#arg, arg)
local recv = get_type_name()
iface = get_interface_name()
if #arg == 0 then
iface = vfn.input("Impl: generating method stubs for interface: ")
vim.cmd("redraw!")
if iface == "" then
utils.notify("Impl: please input interface name e.g. io.Reader or receiver name e.g. GoImpl MyType")
-- print("Usage: GoImpl f *File io.Reader")
end
elseif #arg == 1 then
-- " i.e: ':GoImpl io.Writer'
if iface ~= nil then
recv = select(1, ...)
recv = string.lower(recv) .. " *" .. recv
else
recv = string.lower(recv) .. " *" .. recv
iface = select(1, ...)
end
if recv == "" and iface == "" then
vim.notify("put cursor on struct or a interface or specify a receiver & interface")
end
utils.log(recv)
vim.cmd("redraw!")
elseif #arg == 2 then
utils.log(recv)
if iface ~= nil then
-- " i.e: ':GoImpl s TypeName'
recv = select(1, ...)
local recv_type = select(2, ...)
recv = string.lower(recv) .. " *" .. recv_type
else
recv_name = select(1, ...)
recv = string.format("%s *%s", recv_name, recv)
local l = #arg
iface = select(l, ...)
end
elseif #arg > 2 then
local l = #arg
iface = select(l, ...)
recv = select(l - 1, ...)
recv_name = select(l - 2, ...)
recv = string.format("%s %s", recv_name, recv)
end
utils.log(#arg, recv_name, recv, iface)
local dir = vfn.fnameescape(vfn.expand("%:p:h"))
impl_cmd = { impl_cmd, "-dir", dir, recv, iface }
utils.log(impl_cmd)
-- vim.cmd("normal! $%") -- do a bracket match. changed to treesitter
local opts = {
update_buffer = true,
on_exit = function(code, signal, data)
if code ~= 0 or signal ~= 0 then
utils.warn("impl failed" .. vim.inspect(data))
return
end
data = vim.split(data, "\n")
data = utils.handle_job_data(data)
if not data then
return
end
--
vim.schedule(function()
local pos = vfn.getcurpos()[2]
table.insert(data, 1, "")
vfn.append(pos, data)
end)
end,
}
local runner = require("go.runner")
runner.run(impl_cmd, opts)
end
local function match_iface_name(part)
local pkg, iface = string.match(part, "^(.*)%.(.*)$")
utils.log(pkg, iface)
local cmd = string.format("go doc %s", pkg)
local doc = vfn.systemlist(cmd)
if vim.v.shell_error ~= 0 then
utils.warn("go doc failed" .. vim.inspect(doc))
return
end
local ifaces = {}
local pat = string.format("^type (%s.*) interface", iface)
for _, line in ipairs(doc) do
local m = string.match(line, pat)
if m ~= nil then
table.insert(ifaces, string.format("%s.%s", pkg, m))
end
end
return ifaces
end
-- function complete(arglead, cmdline, cursorpos)
local function complete(_, cmdline, _)
local words = vim.split(cmdline, [[%s+]])
local gopls = require("go.gopls")
local last = words[#words]
local iface = get_interface_name()
local query = require("go.ts.go").query_type_declaration
local bufnr = vim.api.nvim_get_current_buf()
if iface ~= nil then
local nodes = require("go.ts.nodes").nodes_in_buf(query, "go", nil, bufnr, 100000, 100000)
local ns = {}
log(nodes)
for _, node in ipairs(nodes) do
table.insert(ns, node.name)
end
log(ns)
return vfn.uniq(ns)
end
if string.match(last, "^.+%..*") ~= nil then
local part = match_iface_name(last)
if part ~= nil then
return part
end
end
return vfn.uniq(vfn.sort(gopls.list_pkgs()))
end
return { run = run, complete = complete }