local log = require"guihua.log".info local trace = require"guihua.log".trace local M = {} local clone = require'guihua.util'.clone local function filename(url) return url:match("^.+/(.+)$") or url end local function extension(url) local ext = url:match("^.+(%..+)$") or "txt" return string.sub(ext, 2) end local function get_pads(win_width, text, postfix) local trim = false local margin = win_width - #text - #postfix if margin < 0 then trace('line too long', win_width, #text, #postfix) if #postfix > 1 then trim = true end end local sz = #text if sz < 30 then sz = 30 end local space local i = math.floor((sz + 10) / 10) i = i * 10 - #text trace(i, #text, #postfix, postfix, text, win_width) if i + #text + #postfix < win_width then local rem = win_width - i - #text - #postfix rem = math.floor(rem / 10) if rem > 0 then i = i + rem * 10 -- log(i) end end if i > 3 then i = i - 3 end space = string.rep(' ', i) trace(text, i, postfix, win_width) return space, trim end function M.prepare_for_render(items, opts) opts = opts or {} if items == nil or #items < 1 then print("no item found or empty fields") return end local item = clone(items[1]) local display_items = {item} local last_summary_idx = 1 local total_ref_in_file = 1 local icon = " " local lspapi = opts.api or "∑" local ok, devicons = pcall(require, "nvim-web-devicons") if ok then local fn = filename(items[1].filename) local ext = extension(fn) icon = devicons.get_icon(fn, ext) or icon end local call_by_presented = false local width = 100 opts.width = opts.width or width local win_width = opts.width -- buf for i = 1, #items do if items[i].call_by and #items[i].call_by > 0 then call_by_presented = true end end for i = 1, #items do local space local trim local lspapi_display = lspapi items[i].symbol_name = items[i].symbol_name or "" -- some LSP API does not have range for this local fn = display_items[last_summary_idx].filename local dfn = items[i].display_filename if last_summary_idx == 1 then lspapi_display = items[i].symbol_name .. ' ' .. lspapi_display trace(items[1], lspapi_display, display_items[last_summary_idx]) end display_items[last_summary_idx].filename_only = true -- trace(items[i], items[i].filename, last_summary_idx, display_items[last_summary_idx].filename) -- TODO refact display_filename generate part if items[i].filename == fn then space, trim = get_pads(opts.width, icon .. ' ' .. dfn, lspapi_display .. ' 12') if trim and opts.width > 50 and #dfn > opts.width - 20 then local fn1 = string.sub(dfn, 1, opts.width - 50) local fn2 = string.sub(dfn, #dfn - 10, #dfn) display_items[last_summary_idx].display_filename = fn1 .. "" .. fn2 space = ' ' -- log("trim", fn1, fn2) end display_items[last_summary_idx].text = string.format("%s %s%s%s %i", icon, display_items[last_summary_idx] .display_filename, space, lspapi_display, total_ref_in_file) total_ref_in_file = total_ref_in_file + 1 else lspapi_display = lspapi item = clone(items[i]) space, trim = get_pads(opts.width, icon .. ' ' .. item.display_filename, lspapi_display .. ' 12') if trim and opts.width > 52 and #item.display_filename > opts.width - 20 then item.display_filename = string.sub(item.display_filename, 1, opts.width - 52) .. "" .. string.sub(item.display_filename, #item.display_filename - 10, #item.display_filename) space = ' ' end item.text = string.format("%s %s%s%s 1", icon, item.display_filename, space, lspapi_display) trace(item.text) table.insert(display_items, item) total_ref_in_file = 1 last_summary_idx = #display_items end -- content of code lines item = clone(items[i]) item.text = require'navigator.util'.trim_and_pad(item.text) item.text = string.format("%4i: %s", item.lnum, item.text) local ts_report = "" if item.lhs then ts_report = _NgConfigValues.icons.value_changed end if item.definition then ts_report = ts_report .. _NgConfigValues.icons.value_definition .. ' ' end local header_len = #ts_report + 4 -- magic number 2 trace(ts_report, header_len) item.text = item.text:gsub('%s*[%[%(%{]*%s*$', '') if item.call_by ~= nil and #item.call_by > 0 then trace("call_by:", #item.call_by) for _, value in pairs(item.call_by) do if value.node_text then local txt = value.node_text:gsub('%s*[%[%(%{]*%s*$', '') local endwise = '{}' if value.type == 'method' or value.type == 'function' then endwise = '()' local syb = items[i].symbol_name if txt == items[i].symbol_name or (#txt > #syb and txt:sub(#txt - #syb + 1) == syb) then if ts_report ~= _NgConfigValues.icons.value_definition .. ' ' then ts_report = ts_report .. _NgConfigValues.icons.value_definition .. ' ' end header_len = #ts_report + 1 else ts_report = ts_report .. ' ' end end if #ts_report > header_len then ts_report = ts_report .. '  ' end ts_report = ts_report .. value.kind .. txt .. endwise trace(item) end end end if #ts_report > 1 then space, trim = get_pads(win_width, item.text, ts_report) if trim then item.text = string.sub(item.text, 1, opts.width - 20) .. "" end if #space + #item.text + #ts_report >= win_width then if #item.text + #ts_report > win_width then trace("exceeding", #item.text, #ts_report, win_width) space = ' ' else local remain = win_width - #item.text - #ts_report trace("remain", remain) space = string.rep(' ', remain) end end item.text = item.text .. space .. ts_report end local tail = display_items[#display_items].text if tail ~= item.text then -- deduplicate trace(item.text) trace(item.call_by) table.insert(display_items, item) end end display_items[last_summary_idx].filename_only = true -- display_items[last_summary_idx].text=string.format("%s [%i]", display_items[last_summary_idx].filename, -- total_ref_in_file) return display_items end return M