---@class LazyUtil: LazyUtilCore local M = setmetatable({}, { __index = require("lazy.core.util") }) function M.file_exists(file) return vim.loop.fs_stat(file) ~= nil end ---@param opts? LazyFloatOptions function M.float(opts) opts = opts or {} if require("lazy.view").visible() then opts = vim.tbl_deep_extend("force", { zindex = 60, border = "none", margin = { top = 3, left = 2, right = 2 }, }, opts) end return require("lazy.view.float")(opts) end function M.open(uri) if M.file_exists(uri) then return M.float({ style = "", file = uri }) end local Config = require("lazy.core.config") local cmd if Config.options.ui.browser then cmd = { Config.options.ui.browser, uri } elseif vim.fn.has("win32") == 1 then cmd = { "explorer", uri } elseif vim.fn.has("macunix") == 1 then cmd = { "open", uri } else if vim.fn.executable("xdg-open") then cmd = { "xdg-open", uri } else cmd = { "open", uri } end end local ret = vim.fn.jobstart(cmd, { detach = true }) if ret <= 0 then local msg = { "Failed to open uri", ret, vim.inspect(cmd), } vim.notify(table.concat(msg, "\n"), vim.log.levels.ERROR) end end function M.read_file(file) local fd = assert(io.open(file, "r")) ---@type string local data = fd:read("*a") fd:close() return data end function M.write_file(file, contents) local fd = assert(io.open(file, "w+")) fd:write(contents) fd:close() end ---@generic F: fun() ---@param ms number ---@param fn F ---@return F function M.throttle(ms, fn) local timer = vim.loop.new_timer() local running = false local first = true return function(...) local args = { ... } local wrapped = function() fn(unpack(args)) end if not running then if first then wrapped() first = false end timer:start(ms, 0, function() running = false vim.schedule(wrapped) end) running = true end end end ---@class LazyCmdOpts ---@field cwd? string ---@field env? table ---@field float? LazyFloatOptions ---@param cmd string[] ---@param opts? {cwd:string, filetype:string, terminal?:boolean, close_on_exit?:boolean, enter?:boolean, float?:LazyFloatOptions} function M.open_cmd(cmd, opts) opts = opts or {} local float = M.float(opts.float) if opts.filetype then vim.bo[float.buf].filetype = opts.filetype end if opts.terminal then opts.terminal = nil vim.fn.termopen(cmd, opts) if opts.enter then vim.cmd.startinsert() end if opts.close_on_exit then vim.api.nvim_create_autocmd("TermClose", { once = true, buffer = float.buf, callback = function() float:close() vim.cmd.checktime() end, }) end else local Process = require("lazy.manage.process") local lines = Process.exec(cmd, { cwd = opts.cwd }) vim.api.nvim_buf_set_lines(float.buf, 0, -1, false, lines) vim.bo[float.buf].modifiable = false end end ---@return string? function M.head(file) local f = io.open(file) if f then local ret = f:read() f:close() return ret end end ---@return {branch: string, hash:string}? function M.git_info(dir) local line = M.head(dir .. "/.git/HEAD") if line then ---@type string, string local ref, branch = line:match("ref: (refs/heads/(.*))") if ref then return { branch = branch, hash = M.head(dir .. "/.git/" .. ref), } end end end ---@param msg string|string[] ---@param opts? table function M.markdown(msg, opts) if type(msg) == "table" then msg = table.concat(msg, "\n") or msg end vim.notify( msg, vim.log.levels.INFO, vim.tbl_deep_extend("force", { title = "lazy.nvim", on_open = function(win) vim.wo[win].conceallevel = 3 vim.wo[win].concealcursor = "n" vim.wo[win].spell = false vim.treesitter.start(vim.api.nvim_win_get_buf(win), "markdown") end, }, opts or {}) ) end function M._dump(value, result) local t = type(value) if t == "number" or t == "boolean" then table.insert(result, tostring(value)) elseif t == "string" then table.insert(result, ("%q"):format(value)) elseif t == "table" then table.insert(result, "{") local i = 1 ---@diagnostic disable-next-line: no-unknown for k, v in pairs(value) do if k == i then elseif type(k) == "string" then table.insert(result, ("[%q]="):format(k)) else table.insert(result, k .. "=") end M._dump(v, result) table.insert(result, ",") i = i + 1 end table.insert(result, "}") else error("Unsupported type " .. t) end end function M.dump(value) local result = {} M._dump(value, result) return table.concat(result, "") end ---@generic V ---@param t table ---@param fn fun(key:string, value:V) function M.foreach(t, fn) ---@type string[] local keys = vim.tbl_keys(t) pcall(table.sort, keys, function(a, b) return a:lower() < b:lower() end) for _, key in ipairs(keys) do fn(key, t[key]) end end return M