186 lines
5.1 KiB
Lua
186 lines
5.1 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 row, col = unpack(vim.api.nvim_win_get_cursor(0))
|
|
local name = require("go.ts.go").get_struct_node_at_pos(row, col)
|
|
if name == nil then
|
|
name = require("go.ts.go").get_type_node_at_pos(row, col)
|
|
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 row, col = unpack(vim.api.nvim_win_get_cursor(0))
|
|
local name = require("go.ts.go").get_interface_node_at_pos(row, col)
|
|
|
|
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()
|
|
local 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, ...)
|
|
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 }
|