local from_entry = require('telescope.from_entry') local path = require('telescope.path') local utils = require('telescope.utils') local putils = require('telescope.previewers.utils') local Previewer = require('telescope.previewers.previewer') local conf = require('telescope.config').values local pfiletype = require('plenary.filetype') local buf_delete = utils.buf_delete local defaulter = utils.make_default_callable local previewers = {} local ns_previewer = vim.api.nvim_create_namespace('telescope.previewers') previewers.file_maker = function(filepath, bufnr, bufname, use_ft_detect, callback) if use_ft_detect == nil then use_ft_detect = true end local ft = use_ft_detect and pfiletype.detect(filepath) if bufname ~= filepath then path.read_file_async(filepath, vim.schedule_wrap(function(data) vim.api.nvim_buf_set_lines(bufnr, 0, -1, false, vim.split(data, "\n")) if callback then callback(bufnr) end end)) else if callback then callback(bufnr) end end putils.highlighter(bufnr, ft) end previewers.new_buffer_previewer = function(opts) opts = opts or {} assert(opts.define_preview, "define_preview is a required function") assert(not opts.preview_fn, "preview_fn not allowed") local opt_setup = opts.setup local opt_teardown = opts.teardown local old_bufs = {} local bufname_table = {} local global_state = require'telescope.state' local function get_bufnr(self) if not self.state then return nil end return self.state.bufnr end local function set_bufnr(self, value) if get_bufnr(self) then table.insert(old_bufs, get_bufnr(self)) end if self.state then self.state.bufnr = value end end local function get_bufnr_by_bufname(self, value) if not self.state then return nil end return bufname_table[value] end local function set_bufname(self, value) if get_bufnr(self) then bufname_table[value] = get_bufnr(self) end if self.state then self.state.bufname = value end end function opts.setup(self) local state = {} if opt_setup then vim.tbl_deep_extend("force", state, opt_setup(self)) end return state end function opts.teardown(self) if opt_teardown then opt_teardown(self) end local last_nr if opts.keep_last_buf then last_nr = global_state.get_global_key('last_preview_bufnr') -- Push in another buffer so the last one will not be cleaned up local bufnr = vim.api.nvim_create_buf(false, true) vim.api.nvim_win_set_buf(self.state.winid, bufnr) end set_bufnr(self, nil) set_bufname(self, nil) for _, bufnr in ipairs(old_bufs) do if bufnr ~= last_nr then buf_delete(bufnr) end end end function opts.preview_fn(self, entry, status) if get_bufnr(self) == nil then set_bufnr(self, vim.api.nvim_win_get_buf(status.preview_win)) end if opts.get_buffer_by_name and get_bufnr_by_bufname(self, opts.get_buffer_by_name(self, entry)) then self.state.bufname = opts.get_buffer_by_name(self, entry) self.state.bufnr = get_bufnr_by_bufname(self, self.state.bufname) vim.api.nvim_win_set_buf(status.preview_win, self.state.bufnr) else local bufnr = vim.api.nvim_create_buf(false, true) set_bufnr(self, bufnr) vim.api.nvim_win_set_buf(status.preview_win, bufnr) -- TODO(conni2461): We only have to set options once. Right? vim.api.nvim_win_set_option(status.preview_win, 'winhl', 'Normal:Normal') vim.api.nvim_win_set_option(status.preview_win, 'signcolumn', 'no') vim.api.nvim_win_set_option(status.preview_win, 'foldlevel', 100) vim.api.nvim_win_set_option(status.preview_win, 'scrolloff', 999) vim.api.nvim_win_set_option(status.preview_win, 'wrap', false) self.state.winid = status.preview_win self.state.bufname = nil end if opts.keep_last_buf then global_state.set_global_key("last_preview_bufnr", self.state.bufnr) end opts.define_preview(self, entry, status) putils.with_preview_window(status, nil, function() vim.cmd'do User TelescopePreviewerLoaded' end) if opts.get_buffer_by_name then set_bufname(self, opts.get_buffer_by_name(self, entry)) end end if not opts.scroll_fn then function opts.scroll_fn(self, direction) local input = direction > 0 and "d" or "u" local count = math.abs(direction) self:send_input({ count = count, input = input }) end end if not opts.send_input then function opts.send_input(self, input) if not self.state then return end local max_line = vim.fn.getbufinfo(self.state.bufnr)[1].linecount local line = vim.api.nvim_win_get_cursor(self.state.winid)[1] if input.input == 'u' then line = (line - input.count) > 0 and (line - input.count) or 1 else line = (line + input.count) <= max_line and (line + input.count) or max_line end vim.api.nvim_win_set_cursor(self.state.winid, { line, 1 }) end end return Previewer:new(opts) end previewers.cat = defaulter(function(_) return previewers.new_buffer_previewer { get_buffer_by_name = function(_, entry) return from_entry.path(entry, true) end, define_preview = function(self, entry, status) putils.with_preview_window(status, nil, function() local p = from_entry.path(entry, true) if p == nil or p == '' then return end conf.buffer_previewer_maker(p, self.state.bufnr, self.state.bufname) end) end } end, {}) previewers.vimgrep = defaulter(function(_) return previewers.new_buffer_previewer { setup = function() return { last_set_bufnr = nil } end, teardown = function(self) if self.state and self.state.last_set_bufnr and vim.api.nvim_buf_is_valid(self.state.last_set_bufnr) then vim.api.nvim_buf_clear_namespace(self.state.last_set_bufnr, ns_previewer, 0, -1) end end, get_buffer_by_name = function(_, entry) return from_entry.path(entry, true) end, define_preview = function(self, entry, status) putils.with_preview_window(status, nil, function() local lnum = entry.lnum or 0 local p = from_entry.path(entry, true) if p == nil or p == '' then return end if self.state.last_set_bufnr then pcall(vim.api.nvim_buf_clear_namespace, self.state.last_set_bufnr, ns_previewer, 0, -1) end conf.buffer_previewer_maker(p, self.state.bufnr, self.state.bufname, true, function(bufnr) if lnum ~= 0 then pcall(vim.api.nvim_buf_add_highlight, bufnr, ns_previewer, "TelescopePreviewLine", lnum - 1, 0, -1) pcall(vim.api.nvim_win_set_cursor, self.state.winid, {lnum, 0}) end self.state.last_set_bufnr = bufnr end) end) end } end, {}) previewers.qflist = previewers.vimgrep previewers.ctags = defaulter(function(_) return previewers.new_buffer_previewer { teardown = function(self) if self.state and self.state.hl_id then pcall(vim.fn.matchdelete, self.state.hl_id, self.state.hl_win) self.state.hl_id = nil end end, get_buffer_by_name = function(_, entry) return entry.filename end, define_preview = function(self, entry, status) putils.with_preview_window(status, nil, function() local scode = string.gsub(entry.scode, '[$]$', '') scode = string.gsub(scode, [[\\]], [[\]]) scode = string.gsub(scode, [[\/]], [[/]]) scode = string.gsub(scode, '[*]', [[\*]]) conf.buffer_previewer_maker(entry.filename, self.state.bufnr, self.state.bufname, true, function(bufnr) vim.api.nvim_buf_call(bufnr, function() pcall(vim.fn.matchdelete, self.state.hl_id, self.state.winid) vim.cmd "norm! gg" vim.fn.search(scode, "W") vim.cmd "norm! zz" self.state.hl_id = vim.fn.matchadd('TelescopePreviewMatch', scode) end) end) end) end } end, {}) previewers.builtin = defaulter(function(_) return previewers.new_buffer_previewer { setup = function() return {} end, teardown = function(self) if self.state and self.state.hl_id then pcall(vim.fn.matchdelete, self.state.hl_id, self.state.hl_win) self.state.hl_id = nil end end, get_buffer_by_name = function(_, entry) return entry.filename end, define_preview = function(self, entry, status) putils.with_preview_window(status, nil, function() local module_name = vim.fn.fnamemodify(entry.filename, ':t:r') local text if entry.text:sub(1, #module_name) ~= module_name then text = module_name .. '.' .. entry.text else text = entry.text:gsub('_', '.', 1) end conf.buffer_previewer_maker(entry.filename, self.state.bufnr, self.state.bufname, true, function(bufnr) vim.api.nvim_buf_call(bufnr, function() pcall(vim.fn.matchdelete, self.state.hl_id, self.state.winid) vim.cmd "norm! gg" vim.fn.search(text, "W") vim.cmd "norm! zz" self.state.hl_id = vim.fn.matchadd('TelescopePreviewMatch', text) end) end) end) end } end, {}) previewers.help = defaulter(function(_) return previewers.new_buffer_previewer { setup = function() return {} end, teardown = function(self) if self.state and self.state.hl_id then pcall(vim.fn.matchdelete, self.state.hl_id, self.state.hl_win) self.state.hl_id = nil end end, get_buffer_by_name = function(_, entry) return entry.filename end, define_preview = function(self, entry, status) putils.with_preview_window(status, nil, function() local query = entry.cmd query = query:sub(2) query = [[\V]] .. query conf.buffer_previewer_maker(entry.filename, self.state.bufnr, self.state.bufname, false, function(bufnr) vim.api.nvim_buf_call(bufnr, function() vim.cmd(':ownsyntax help') pcall(vim.fn.matchdelete, self.state.hl_id, self.state.winid) vim.cmd "norm! gg" vim.fn.search(query, "W") vim.cmd "norm! zz" self.state.hl_id = vim.fn.matchadd('TelescopePreviewMatch', query) end) end) end) end } end, {}) previewers.man = defaulter(function(_) return previewers.new_buffer_previewer { get_buffer_by_name = function(_, entry) return entry.value end, define_preview = function(self, entry, status) putils.with_preview_window(status, nil, function() local win_width = vim.api.nvim_win_get_width(self.state.winid) putils.job_maker({'man', '-P', 'cat', entry.value}, self.state.bufnr, { env = { ["MANWIDTH"] = win_width }, value = entry.value, bufname = self.state.bufname }) putils.regex_highlighter(_, 'man') end) end } end) previewers.git_branch_log = defaulter(function(_) return previewers.new_buffer_previewer { get_buffer_by_name = function(_, entry) return entry.value end, define_preview = function(self, entry, status) local highlight_buffer = function(bufnr, content) if content and table.getn(content) == 0 then vim.api.nvim_buf_set_lines(bufnr, 0, -1, false, { "No log found for branch: " .. entry.value }) elseif content and table.getn(content) > 1 then for i = 1, #content do local line = content[i] local _, hash_start = line:find('[%*%s|]*') if hash_start then local hash_end = hash_start + 7 if hash_end < #line then vim.api.nvim_buf_add_highlight(bufnr, ns_previewer, "TelescopeResultsIdentifier", i - 1, hash_start - 1, hash_end ) end end local _, cur_start = line:find('- %(') if cur_start then local cur_end = string.find(line, '%) ') vim.api.nvim_buf_add_highlight(bufnr, ns_previewer, "TelescopeResultsConstant", i - 1, cur_start - 1, cur_end ) end local date_start, _ = line:find(' %(%d') if date_start then vim.api.nvim_buf_add_highlight(bufnr, ns_previewer, "TelescopeResultsSpecialComment", i - 1, date_start, #line ) end end end end putils.with_preview_window(status, nil, function() local cmd = { 'git', '-P', 'log', '--graph', '--pretty=format:%h -%d %s (%cr)', '--abbrev-commit', '--date=relative', entry.value } putils.job_maker(cmd, self.state.bufnr, { value = entry.value, bufname = self.state.bufname, callback = highlight_buffer }) end) end } end, {}) previewers.git_commit_diff = defaulter(function(_) return previewers.new_buffer_previewer { get_buffer_by_name = function(_, entry) return entry.value end, define_preview = function(self, entry, status) putils.with_preview_window(status, nil, function() putils.job_maker({ 'git', '-P', 'diff', entry.value .. '^!' }, self.state.bufnr, { value = entry.value, bufname = self.state.bufname }) putils.regex_highlighter(_, 'diff') end) end } end, {}) previewers.git_file_diff = defaulter(function(_) return previewers.new_buffer_previewer { get_buffer_by_name = function(_, entry) return entry.value end, define_preview = function(self, entry, status) putils.with_preview_window(status, nil, function() putils.job_maker({ 'git', '-P', 'diff', entry.value }, self.state.bufnr, { value = entry.value, bufname = self.state.bufname }) putils.regex_highlighter(_, 'diff') end) end } end, {}) previewers.autocommands = defaulter(function(_) return previewers.new_buffer_previewer { teardown = function(self) if self.state and self.state.last_set_bufnr and vim.api.nvim_buf_is_valid(self.state.last_set_bufnr) then pcall(vim.api.nvim_buf_clear_namespace, self.state.last_set_bufnr, ns_previewer, 0, -1) end end, get_buffer_by_name = function(_, entry) return entry.group end, define_preview = function(self, entry, status) local results = vim.tbl_filter(function (x) return x.group == entry.group end, status.picker.finder.results) if self.state.last_set_bufnr then pcall(vim.api.nvim_buf_clear_namespace, self.state.last_set_bufnr, ns_previewer, 0, -1) end local selected_row = 0 if self.state.bufname ~= entry.group then local display = {} table.insert(display, string.format(" augroup: %s - [ %d entries ]", entry.group, #results)) -- TODO: calculate banner width/string in setup() -- TODO: get column characters to be the same HL group as border table.insert(display, string.rep("─", vim.fn.getwininfo(status.preview_win)[1].width)) for idx, item in ipairs(results) do if item == entry then selected_row = idx end table.insert(display, string.format(" %-14sā–%-08s %s", item.event, item.ft_pattern, item.command) ) end vim.api.nvim_buf_set_option(self.state.bufnr, "filetype", "vim") vim.api.nvim_buf_set_lines(self.state.bufnr, 0, -1, false, display) vim.api.nvim_buf_add_highlight(self.state.bufnr, 0, "TelescopeBorder", 1, 0, -1) else for idx, item in ipairs(results) do if item == entry then selected_row = idx break end end end vim.api.nvim_buf_add_highlight(self.state.bufnr, ns_previewer, "TelescopePreviewLine", selected_row + 1, 0, -1) vim.api.nvim_win_set_cursor(status.preview_win, {selected_row + 1, 0}) self.state.last_set_bufnr = self.state.bufnr end, } end, {}) previewers.highlights = defaulter(function(_) return previewers.new_buffer_previewer { teardown = function(self) if self.state and self.state.last_set_bufnr and vim.api.nvim_buf_is_valid(self.state.last_set_bufnr) then vim.api.nvim_buf_clear_namespace(self.state.last_set_bufnr, ns_previewer, 0, -1) end end, get_buffer_by_name = function() return "highlights" end, define_preview = function(self, entry, status) putils.with_preview_window(status, nil, function() if not self.state.bufname then local output = vim.split(vim.fn.execute('highlight'), '\n') local hl_groups = {} for _, v in ipairs(output) do if v ~= '' then if v:sub(1, 1) == ' ' then local part_of_old = v:match('%s+(.*)') hl_groups[table.getn(hl_groups)] = hl_groups[table.getn(hl_groups)] .. part_of_old else table.insert(hl_groups, v) end end end vim.api.nvim_buf_set_lines(self.state.bufnr, 0, -1, false, hl_groups) for k, v in ipairs(hl_groups) do local startPos = string.find(v, 'xxx', 1, true) - 1 local endPos = startPos + 3 local hlgroup = string.match(v, '([^ ]*)%s+.*') pcall(vim.api.nvim_buf_add_highlight, self.state.bufnr, 0, hlgroup, k - 1, startPos, endPos) end end pcall(vim.api.nvim_buf_clear_namespace, self.state.bufnr, ns_previewer, 0, -1) vim.cmd "norm! gg" vim.fn.search(entry.value .. ' ') local lnum = vim.fn.line('.') -- That one is actually a match but its better to use it like that then matchadd vim.api.nvim_buf_add_highlight(self.state.bufnr, ns_previewer, "TelescopePreviewMatch", lnum - 1, 0, #entry.value) end) end, } end, {}) previewers.display_content = defaulter(function(_) return previewers.new_buffer_previewer { define_preview = function(self, entry, status) putils.with_preview_window(status, nil, function() assert(type(entry.preview_command) == 'function', 'entry must provide a preview_command function which will put the content into the buffer') entry.preview_command(entry, self.state.bufnr) end) end } end, {}) return previewers