forked from Robitx/gp.nvim
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathhelper.lua
296 lines (266 loc) · 8.27 KB
/
helper.lua
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
--------------------------------------------------------------------------------
-- Generic independent helper functions
--------------------------------------------------------------------------------
local logger = require("gp.logger")
local _H = {}
---@param keys string # string of keystrokes
---@param mode string # string of vim mode ('n', 'i', 'c', etc.), default is 'n'
_H.feedkeys = function(keys, mode)
mode = mode or "n"
keys = vim.api.nvim_replace_termcodes(keys, true, false, true)
vim.api.nvim_feedkeys(keys, mode, true)
end
---@param buffers table # table of buffers
---@param mode table | string # mode(s) to set keymap for
---@param key string # shortcut key
---@param callback function | string # callback or string to set keymap
---@param desc string | nil # optional description for keymap
_H.set_keymap = function(buffers, mode, key, callback, desc)
logger.debug(
"registering shortcut:"
.. " mode: "
.. vim.inspect(mode)
.. " key: "
.. key
.. " buffers: "
.. vim.inspect(buffers)
.. " callback: "
.. vim.inspect(callback)
)
for _, buf in ipairs(buffers) do
vim.keymap.set(mode, key, callback, {
noremap = true,
silent = true,
nowait = true,
buffer = buf,
desc = desc,
})
end
end
---@param events string | table # events to listen to
---@param buffers table | nil # buffers to listen to (nil for all buffers)
---@param callback function # callback to call
---@param gid number # augroup id
_H.autocmd = function(events, buffers, callback, gid)
if buffers then
for _, buf in ipairs(buffers) do
vim.api.nvim_create_autocmd(events, {
group = gid,
buffer = buf,
callback = vim.schedule_wrap(callback),
})
end
else
vim.api.nvim_create_autocmd(events, {
group = gid,
callback = vim.schedule_wrap(callback),
})
end
end
---@param file_name string # name of the file for which to delete buffers
_H.delete_buffer = function(file_name)
-- iterate over buffer list and close all buffers with the same name
for _, b in ipairs(vim.api.nvim_list_bufs()) do
if vim.api.nvim_buf_is_valid(b) and vim.api.nvim_buf_get_name(b) == file_name then
vim.api.nvim_buf_delete(b, { force = true })
end
end
end
---@param file string | nil # name of the file to delete
_H.delete_file = function(file)
logger.debug("deleting file: " .. vim.inspect(file))
if file == nil then
return
end
_H.delete_buffer(file)
os.remove(file)
end
---@param file_name string # name of the file for which to get buffer
---@return number | nil # buffer number
_H.get_buffer = function(file_name)
for _, b in ipairs(vim.api.nvim_list_bufs()) do
if vim.api.nvim_buf_is_valid(b) then
if _H.ends_with(vim.api.nvim_buf_get_name(b), file_name) then
return b
end
end
end
return nil
end
---@return string # returns unique uuid
_H.uuid = function()
local random = math.random
local template = "xxxxxxxx_xxxx_4xxx_yxxx_xxxxxxxxxxxx"
local result = string.gsub(template, "[xy]", function(c)
local v = (c == "x") and random(0, 0xf) or random(8, 0xb)
return string.format("%x", v)
end)
return result
end
---@param name string # name of the augroup
---@param opts table | nil # options for the augroup
---@return number # returns augroup id
_H.create_augroup = function(name, opts)
return vim.api.nvim_create_augroup(name .. "_" .. _H.uuid(), opts or { clear = true })
end
---@param buf number # buffer number
---@return number # returns the first line with content of specified buffer
_H.last_content_line = function(buf)
buf = buf or vim.api.nvim_get_current_buf()
-- go from end and return number of last nonwhitespace line
local line = vim.api.nvim_buf_line_count(buf)
while line > 0 do
local content = vim.api.nvim_buf_get_lines(buf, line - 1, line, false)[1]
if content:match("%S") then
return line
end
line = line - 1
end
return 0
end
---@param buf number # buffer number
---@return string # returns filetype of specified buffer
_H.get_filetype = function(buf)
return vim.api.nvim_get_option_value("filetype", { buf = buf })
end
---@param line number # line number
---@param buf number # buffer number
---@param win number | nil # window number
_H.cursor_to_line = function(line, buf, win)
-- don't manipulate cursor if user is elsewhere
if buf ~= vim.api.nvim_get_current_buf() then
return
end
-- check if win is valid
if not win or not vim.api.nvim_win_is_valid(win) then
return
end
-- move cursor to the line
vim.api.nvim_win_set_cursor(win, { line, 0 })
end
---@param str string # string to check
---@param start string # string to check for
_H.starts_with = function(str, start)
return str:sub(1, #start) == start
end
---@param str string # string to check
---@param ending string # string to check for
_H.ends_with = function(str, ending)
return ending == "" or str:sub(-#ending) == ending
end
-- helper function to find the root directory of the current git repository
---@param path string | nil # optional path to start searching from
---@return string # returns the path of the git root dir or an empty string if not found
_H.find_git_root = function(path)
logger.debug("finding git root for path: " .. vim.inspect(path))
local cwd = vim.fn.expand("%:p:h")
if path then
cwd = vim.fn.fnamemodify(path, ":p:h")
end
for _ = 0, 1000 do
local files = vim.fn.readdir(cwd)
if vim.tbl_contains(files, ".git") then
logger.debug("found git root: " .. cwd)
return cwd
end
local parent = vim.fn.fnamemodify(cwd, ":h")
if parent == cwd then
break
end
cwd = parent
end
logger.debug("git root not found")
return ""
end
---@param buf number # buffer number
_H.undojoin = function(buf)
if not buf or not vim.api.nvim_buf_is_loaded(buf) then
return
end
local status, result = pcall(vim.cmd.undojoin)
if not status then
if result:match("E790") then
return
end
logger.error("Error running undojoin: " .. vim.inspect(result))
end
end
---@param tbl table # the table to be stored
---@param file_path string # the file path where the table will be stored as json
_H.table_to_file = function(tbl, file_path)
local json = vim.json.encode(tbl)
local file = io.open(file_path, "w")
if not file then
logger.warning("Failed to open file for writing: " .. file_path)
return
end
file:write(json)
file:close()
end
---@param file_path string # the file path from where to read the json into a table
---@return table | nil # the table read from the file, or nil if an error occurred
_H.file_to_table = function(file_path)
local file, err = io.open(file_path, "r")
if not file then
logger.warning("Failed to open file for reading: " .. file_path .. "\nError: " .. err)
return nil
end
local content = file:read("*a")
file:close()
if content == nil or content == "" then
logger.warning("Failed to read any content from file: " .. file_path)
return nil
end
local tbl = vim.json.decode(content)
return tbl
end
---@param dir string # directory to prepare
---@param name string | nil # name of the directory
---@return string # returns resolved directory path
_H.prepare_dir = function(dir, name)
local odir = dir
dir = dir:gsub("/$", "")
name = name and name .. " " or ""
if vim.fn.isdirectory(dir) == 0 then
logger.debug("creating " .. name .. "directory: " .. dir)
vim.fn.mkdir(dir, "p")
end
dir = vim.fn.resolve(dir)
logger.debug("resolved " .. name .. "directory:\n" .. odir .. " -> " .. dir)
return dir
end
---@param cmd_name string # name of the command
---@param cmd_func function # function to be executed when the command is called
---@param completion function | table | nil # optional function returning table for completion
---@param desc string | nil # description of the command
_H.create_user_command = function(cmd_name, cmd_func, completion, desc)
logger.debug("creating user command: " .. cmd_name)
vim.api.nvim_create_user_command(cmd_name, cmd_func, {
nargs = "*",
range = true,
desc = desc or "Gp.nvim command",
complete = function(arg_lead, cmd_line, cursor_pos)
logger.debug(
"completing user command: "
.. cmd_name
.. "\narg_lead: "
.. arg_lead
.. "\ncmd_line: "
.. cmd_line
.. "\ncursor_pos: "
.. cursor_pos
)
if not completion then
return {}
end
if type(completion) == "function" then
return completion(arg_lead, cmd_line, cursor_pos) or {}
end
if type(completion) == "table" then
return completion
end
return {}
end,
})
end
return _H