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
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
|
--- Module containing simple utility functions for vis-lspc/
-- @module util
-- @author Florian Fischer
-- @author git-bruh <prathamIN@proton.me>
-- @license GPL-3
-- @copyright 2024-2025 Florian Fischer
-- @copyright 2024 git-bruh <prathamIN@proton.me>
local util = {}
local source_str = debug.getinfo(1, 'S').source:sub(2)
local source_path = source_str:match('(.*/)')
local lspc
function util.init(lspc_)
lspc = lspc_
return util
end
--- Execute a command and capture its output.
-- @param cmd the command to execute
-- @return the output of the command written to stdout
function util.capture_cmd(cmd)
local p = assert(io.popen(cmd, 'r'))
local s = assert(p:read('*a'))
local success, _, status = p:close()
if not success then
local err = cmd .. ' failed with exit code: ' .. status
lspc:err(err)
end
return s
end
local vis_supports_pipe_buf = pcall(vis.pipe, vis, 'foo', 'true', false)
--- Wrapper for the two vis:pipe variants.
-- If vis does not support vis:pipe(input, cmd), prefix the command
-- with a printf call piping the result to the original command.
-- @param input The input to pipe to the command
-- @param cmd The external command to pipe the input to
function util.vis_pipe(input, cmd, fullscreen)
if vis_supports_pipe_buf then
return vis:pipe(input, cmd, fullscreen or false)
end
local escaped_input = input:gsub('\'', '\'"\'"\'')
cmd = 'printf %s \'' .. escaped_input .. '\' | ' .. cmd
return vis:pipe(vis.win.file, {start = 0, finish = 0}, cmd)
end
--- Split a path into its components
-- @param path the path to split into components
-- @return a table containing the path components
function util.split_path_into_components(path)
local components = {}
if #path == 1 then
return nil
end
-- Skip the initial '/'
local start_idx = 2
while true do
local slash = path:find('/', start_idx + 1)
if slash == nil then
table.insert(components, path:sub(start_idx, #path))
return components
else
table.insert(components, path:sub(start_idx, slash - 1))
start_idx = slash + 1
end
end
end
--- Get a path relative to the current working directory
-- @param cwd_components Table of the path components of the CWD
-- @param absolute_path_or_components absolute path or table of its path components
-- @return the relative path
function util.get_relative_path(cwd_components, absolute_path_or_components)
local absolute_components
if type(absolute_path_or_components) == 'string' then
absolute_components = util.split_path_into_components(absolute_path_or_components)
else
absolute_components = absolute_path_or_components
end
for idx = 1, #cwd_components do
local cwd = cwd_components[idx]
local absolute = absolute_components[idx]
if cwd ~= absolute then
local dir = ''
-- Atleast the first component must match for us to convert
-- it to a relative path
if idx ~= 1 then
for _ = idx, #cwd_components do
dir = dir .. '..' .. '/'
end
-- Skip trailing '/'
dir = dir:sub(1, #dir - 1)
end
for i = idx, #absolute_components do
dir = dir .. '/' .. absolute_components[i]
end
return dir
end
end
-- cwd shorter than absolute path
local dir = ''
for i = #cwd_components + 1, #absolute_components do
dir = dir .. '/' .. absolute_components[i]
end
-- Skip leading '/'
return dir:sub(2)
end
--- Strip the last component from a pathname
-- @param the pathname
-- @return the pathname up to the last '/'
function util.dirname(name)
if name == '.' or name == '..' or name == '/' then
return name
end
-- strip a trailing path separator
if name:sub(#name, #name) == '/' then
name = name:sub(1, #name - 1)
end
local dirname = name:match('(.*)[/]')
-- There was no path separator in name.
if not dirname then
return '.'
end
-- The name started with the root dir.
if dirname == '' then
return '/'
end
return dirname
end
--- Create an iterator yielding the nth line of a file
--
-- @param path The path to the file
function util.file_line_iterator_to_n(path)
local file = assert(io.open(path, 'r'))
local lines = file:lines()
local last_line = nil
local last_n = 1
return function(n)
if n == -1 then
file:close()
return nil
end
if n < last_n then
-- We might have multiple references on the same line, so we can
-- get called again with the previous line number
if (n + 1) == last_n then
return last_line
end
return nil
end
for line in lines do
if n == last_n then
last_n = last_n + 1
last_line = line
return line
end
last_n = last_n + 1
end
-- Iterator exhausted
return nil
end
end
--- Find file based on globs in the parent file system tree
-- @param globs a new line separated string of file globs
-- @param start the starting path
function util.find_upwards(globs, start)
local status, out = util.vis_pipe(globs, '\'' .. source_path:gsub('\'', '\'\\\'\'') ..
'/tools/find-upwards\' "' .. start .. '"')
if status ~= 0 or out == nil then
return nil
end
-- Skip trailing newline
return out:sub(1, #out - 1)
end
-- get the vis_selection from current primary selection
local function get_selection(win)
return {line = win.selection.line, col = win.selection.col}
end
--- Calculate the 0-based byte offsets from multiple sorted selections
-- @param file the file to calculate the positions in
-- @param sorted_selections a table of sorted vis_selections
-- @return a table of positions
function util.vis_sorted_selections_to_pos(file, sorted_selections)
local positions = {}
if file.pos_by_linecol then
for _, sel in ipairs(sorted_selections) do
table.insert(positions, file:pos_by_linecol(sel.line, sel.col))
end
return positions
end
local line_count = 0
local pos = 0
local sel_i = 1
local sel = sorted_selections[sel_i]
for line in file:lines_iterator() do
line_count = line_count + 1
while line_count == sel.line do
table.insert(positions, pos + (sel.col - 1))
sel_i = sel_i + 1
-- no more selections to convert
if sel_i > #sorted_selections then
break
end
sel = sorted_selections[sel_i]
end
pos = pos + #line + 1
end
-- Some language servers (pylsp) send ranges including the first character after the last line.
-- e.g. [{"line":1, "col":1}, {"line":#lines+1, "col":1}]
-- But this selection can not be handled by iterating all lines.
-- Special case ranges ensing after the last line.
-- Additionaly the start and end can overlap.
-- Gopls sends the range [{"line":1, "col":1}, {"line":1, "col":1}] when editing an empty file.
while sel and sel.line == line_count + 1 and sel.col == 1 do
table.insert(positions, pos)
sel_i = sel_i + 1
sel = sorted_selections[sel_i]
end
return positions
end
local function vis_pos_before(p1, p2)
return p1.line < p2.line or (p1.line == p2.line and p1.col < p2.col)
end
--- Calculate the 0-based byte offsets from multiple selections
-- @param file the file to calculate the positions in
-- @param selections a table of selections
-- @return a table of positions
function util.vis_selections_to_pos(file, selections)
table.sort(selections, vis_pos_before)
return util.vis_sorted_selections_to_pos(file, selections)
end
--- Get the line and column from a 0-based byte offset
-- ATTENTION: the fallback version of this function modifies the primary
-- selection so it is not safe to call it for example during WIN_HIGHLIGHT events
-- @param pos the 0-based byte offset into the file
-- @return the 1-based line number
-- @return the 1-based column
function util.vis_pos_to_sel(win, pos)
if win.file.linecol_by_pos then
local lineno, col = win.file:linecol_by_pos(pos)
return {line = lineno, col = col}
end
local old_selection = get_selection(win)
-- move primary selection
win.selection.pos = pos
local sel = get_selection(win)
-- restore old primary selection
win.selection:to(old_selection.line, old_selection.col)
return sel
end
--- Count the visual characters in a line
-- This is useful to detect wrapped lines including tabs which may add a
-- unspecified amount of white space to a line depending on their position and
-- the used tabwidth.
-- @param win the window containing the line
-- @param line the line to count
-- @param nchars the number of characters in the line
-- @return the number of visual characters
util.visual_chars_in_line = function(win, line, nchars)
-- fast string iteration inspired by:
-- https://stackoverflow.com/a/49222705
local l = {string.byte(line, 1, nchars)}
local line_len = 0
for i = 1, nchars do
local c = l[i] -- Note: produces char codes instead of chars.
if c == 9 then -- '\t'
local chars_to_tab_stop = win.options.tabwidth - (line_len % win.options.tabwidth)
line_len = line_len + chars_to_tab_stop
else
line_len = line_len + 1
end
end
return line_len
end
--- Utility table function
util.table = {}
--- Recursively copy a table and all its members.
-- @param tbl the table to copy
-- @return a deep copy of the table
function util.table.deep_copy(tbl)
local cpy = {}
for k, v in pairs(tbl) do
if type(v) ~= 'table' then
cpy[k] = v
else
cpy[k] = util.table.deep_copy(v)
end
end
for i, v in ipairs(tbl) do
if type(v) ~= 'table' then
cpy[i] = v
else
cpy[i] = util.table.deep_copy(v)
end
end
return cpy
end
--- Recursively merge two tables.
-- This method modifies the weak table, if this is not intended make sure you
-- use a deep copy of the weak table.
-- Named members of the strong table override members of the weak one.
-- Numbered members get inserted into the weak table.
-- @param weak table which's members are potentially overridden
-- @param strong table which's members override ones from the weak table
-- @return the merged table
function util.table.merge(weak, strong)
for k, v in pairs(strong) do
if type(v) ~= 'table' then
if tonumber(k) then -- append numeric members
table.insert(weak, v)
else -- potentially override the weak member
weak[k] = v
end
else
weak[k] = util.table.merge(weak[k] or {}, v)
end
end
return weak
end
return util
|