mirror of
https://github.com/kristoferssolo/telescope-frecency.nvim.git
synced 2025-10-21 20:10:38 +00:00
* feat: allow user to disable database validation safe_mode * fix safe_mode logic * fix: fix user confirmation * feat: allow disabling of auto-validation
280 lines
8.9 KiB
Lua
280 lines
8.9 KiB
Lua
local has_telescope, telescope = pcall(require, "telescope")
|
|
|
|
-- TODO: make sure scandir unindexed have opts.ignore_patterns applied
|
|
-- TODO: make filters handle mulitple directories
|
|
|
|
if not has_telescope then
|
|
error("This plugin requires telescope.nvim (https://github.com/nvim-telescope/telescope.nvim)")
|
|
end
|
|
|
|
local actions = require('telescope.actions')
|
|
local conf = require('telescope.config').values
|
|
local entry_display = require "telescope.pickers.entry_display"
|
|
local finders = require "telescope.finders"
|
|
local Path = require('plenary.path')
|
|
local pickers = require "telescope.pickers"
|
|
local sorters = require "telescope.sorters"
|
|
local utils = require('telescope.utils')
|
|
local db_client = require("telescope._extensions.frecency.db_client")
|
|
|
|
local os_home = vim.loop.os_homedir()
|
|
local os_path_sep = utils.get_separator()
|
|
|
|
local state = {
|
|
results = {},
|
|
active_filter = nil,
|
|
active_filter_tag = nil,
|
|
last_filter = nil,
|
|
previous_buffer = nil,
|
|
cwd = nil,
|
|
show_scores = false,
|
|
user_workspaces = {},
|
|
lsp_workspaces = {},
|
|
picker = {}
|
|
}
|
|
|
|
local function format_filepath(filename, opts)
|
|
local original_filename = filename
|
|
|
|
if state.active_filter then
|
|
filename = Path:new(filename):make_relative(state.active_filter)
|
|
else
|
|
filename = Path:new(filename):make_relative(state.cwd)
|
|
-- check relative to home/current
|
|
if vim.startswith(filename, os_home) then
|
|
filename = "~/" .. Path:new(filename):make_relative(os_home)
|
|
elseif filename ~= original_filename then
|
|
filename = "./" .. filename
|
|
end
|
|
end
|
|
|
|
if opts.tail_path then
|
|
filename = utils.path_tail(filename)
|
|
elseif opts.shorten_path then
|
|
filename = utils.path_shorten(filename)
|
|
end
|
|
|
|
return filename
|
|
end
|
|
|
|
local function get_workspace_tags()
|
|
|
|
-- Add user config workspaces. TODO: validate that workspaces are existing directories
|
|
local tags = {}
|
|
for k,_ in pairs(state.user_workspaces) do
|
|
table.insert(tags, k)
|
|
end
|
|
|
|
-- Add CWD filter
|
|
table.insert(tags, 'CWD')
|
|
|
|
-- Add LSP workpace(s)
|
|
local lsp_workspaces = vim.api.nvim_buf_call(state.previous_buffer, vim.lsp.buf.list_workspace_folders)
|
|
if not vim.tbl_isempty(lsp_workspaces) then
|
|
state.lsp_workspaces = lsp_workspaces
|
|
table.insert(tags, 'LSP')
|
|
else
|
|
state.lsp_workspaces = {}
|
|
end
|
|
|
|
-- print(vim.inspect(tags))
|
|
-- TODO: sort tags - by collective frecency?
|
|
return tags
|
|
end
|
|
|
|
local frecency = function(opts)
|
|
opts = opts or {}
|
|
|
|
state.previous_buffer = vim.fn.bufnr('%')
|
|
state.cwd = vim.fn.expand(opts.cwd or vim.fn.getcwd())
|
|
|
|
local function get_display_cols()
|
|
local directory_col_width = 0
|
|
if state.active_filter then
|
|
if state.active_filter_tag == "LSP" then
|
|
-- TODO: Only add +1 if opts.show_filter_thing is true, +1 is for the trailing slash
|
|
directory_col_width = #(utils.path_tail(state.active_filter)) + 1
|
|
else
|
|
directory_col_width = #(Path:new(state.active_filter):make_relative(os_home)) + 1
|
|
end
|
|
end
|
|
local res = {}
|
|
res[1] = state.show_scores and {width = 8} or nil
|
|
if state.show_filter_column then
|
|
table.insert(res, {width = directory_col_width})
|
|
end
|
|
table.insert(res, {remaining = true})
|
|
return res
|
|
end
|
|
|
|
local displayer = entry_display.create {
|
|
separator = "",
|
|
hl_chars = {[os_path_sep] = "TelescopePathSeparator"},
|
|
items = get_display_cols()
|
|
}
|
|
|
|
local bufnr, buf_is_loaded, filename, hl_filename, display_items
|
|
local make_display = function(entry)
|
|
bufnr = vim.fn.bufnr
|
|
buf_is_loaded = vim.api.nvim_buf_is_loaded
|
|
|
|
filename = entry.name
|
|
hl_filename = buf_is_loaded(bufnr(filename)) and "TelescopeBufferLoaded" or ""
|
|
filename = format_filepath(filename, opts)
|
|
|
|
display_items = state.show_scores and {{entry.score, "TelescopeFrecencyScores"}} or {}
|
|
|
|
-- TODO: store the column lengths here, rather than recalculating in get_display_cols()
|
|
-- TODO: only include filter_paths column if opts.show_filter_col is true
|
|
local filter_path = ""
|
|
if state.active_filter then
|
|
if state.active_filter_tag == "LSP" or state.active_filter_tag == "CWD" then
|
|
filter_path = utils.path_tail(state.active_filter) .. os_path_sep
|
|
else
|
|
filter_path = Path:new(state.active_filter):make_relative(os_home) .. os_path_sep
|
|
end
|
|
end
|
|
table.insert(display_items, {filter_path, "Directory"})
|
|
table.insert(display_items, {filename, hl_filename})
|
|
|
|
return displayer(display_items)
|
|
end
|
|
|
|
local update_results = function(filter)
|
|
local filter_updated = false
|
|
|
|
-- validate tag
|
|
local ws_dir = filter and state.user_workspaces[filter]
|
|
if filter == "LSP" and not vim.tbl_isempty(state.lsp_workspaces) then
|
|
ws_dir = state.lsp_workspaces[1]
|
|
end
|
|
|
|
if filter == "CWD" then
|
|
ws_dir = state.cwd
|
|
end
|
|
|
|
if ws_dir ~= state.active_filter then
|
|
filter_updated = true
|
|
state.active_filter = ws_dir
|
|
state.active_filter_tag = filter
|
|
end
|
|
|
|
if vim.tbl_isempty(state.results) or filter_updated then
|
|
state.results = db_client.get_file_scores(state.show_unindexed, ws_dir)
|
|
end
|
|
return filter_updated
|
|
end
|
|
|
|
-- populate initial results
|
|
update_results()
|
|
|
|
local entry_maker = function(entry)
|
|
return {
|
|
filename = entry.filename,
|
|
display = make_display,
|
|
ordinal = entry.filename,
|
|
name = entry.filename,
|
|
score = entry.score
|
|
}
|
|
end
|
|
|
|
local delim = opts.filter_delimiter or ":"
|
|
local filter_re = "^(" .. delim .. "(%S+)" .. delim .. ")"
|
|
|
|
state.picker = pickers.new(opts, {
|
|
prompt_title = "Frecency",
|
|
on_input_filter_cb = function(query_text)
|
|
-- check for :filter: in query text
|
|
local matched, new_filter = query_text:match(filter_re)
|
|
if matched then
|
|
query_text = query_text:sub(matched:len() + 1)
|
|
end
|
|
|
|
local new_finder
|
|
local results_updated = update_results(new_filter)
|
|
if results_updated then
|
|
displayer = entry_display.create {
|
|
separator = "",
|
|
hl_chars = {[os_path_sep] = "TelescopePathSeparator"},
|
|
items = get_display_cols()
|
|
}
|
|
|
|
state.last_filter = new_filter
|
|
new_finder = finders.new_table {
|
|
results = state.results,
|
|
entry_maker = entry_maker
|
|
}
|
|
end
|
|
|
|
return {prompt = query_text, updated_finder = new_finder}
|
|
end,
|
|
attach_mappings = function(prompt_bufnr)
|
|
actions.select_default:replace_if(function()
|
|
local compinfo = vim.fn.complete_info()
|
|
return compinfo.pum_visible == 1
|
|
end, function()
|
|
local compinfo = vim.fn.complete_info()
|
|
local keys = compinfo.selected == -1 and "<C-e><Bs><Right>" or "<C-y><Right>:"
|
|
local accept_completion = vim.api.nvim_replace_termcodes(keys, true, false, true)
|
|
vim.api.nvim_feedkeys(accept_completion, "n", true)
|
|
end)
|
|
|
|
return true
|
|
end,
|
|
finder = finders.new_table {
|
|
results = state.results,
|
|
entry_maker = entry_maker
|
|
},
|
|
previewer = conf.file_previewer(opts),
|
|
sorter = sorters.get_substr_matcher(opts),
|
|
})
|
|
state.picker:find()
|
|
|
|
-- restore last filter
|
|
if state.persistent_filter and state.last_filter then
|
|
vim.api.nvim_feedkeys(":" .. state.last_filter .. ":", "n", true)
|
|
end
|
|
|
|
vim.api.nvim_buf_set_option(state.picker.prompt_bufnr, "filetype", "frecency")
|
|
vim.api.nvim_buf_set_option(state.picker.prompt_bufnr, "completefunc", "frecency#FrecencyComplete")
|
|
vim.api.nvim_buf_set_keymap(state.picker.prompt_bufnr, "i", "<Tab>", "pumvisible() ? '<C-n>' : '<C-x><C-u>'", {expr = true, noremap = true})
|
|
vim.api.nvim_buf_set_keymap(state.picker.prompt_bufnr, "i", "<S-Tab>", "pumvisible() ? '<C-p>' : ''", {expr = true, noremap = true})
|
|
end
|
|
|
|
|
|
local function set_config_state(opt_name, value, default)
|
|
state[opt_name] = value == nil and default or value
|
|
end
|
|
|
|
local health_ok = vim.fn["health#report_ok"]
|
|
local health_error = vim.fn["health#report_error"]
|
|
|
|
local function checkhealth()
|
|
local has_sql, _ = pcall(require, "sql")
|
|
if has_sql then
|
|
health_ok("sql.nvim installed.")
|
|
-- return "MOOP"
|
|
else
|
|
health_error("NOOO")
|
|
end
|
|
end
|
|
|
|
return telescope.register_extension {
|
|
setup = function(ext_config)
|
|
set_config_state('show_scores', ext_config.show_scores, false)
|
|
set_config_state('show_unindexed', ext_config.show_unindexed, true)
|
|
set_config_state('show_filter_column', ext_config.show_filter_column, true)
|
|
set_config_state('persistent_filter', ext_config.persistent_filter, true)
|
|
set_config_state('user_workspaces', ext_config.workspaces, {})
|
|
|
|
-- start the database client
|
|
db_client.init(ext_config.ignore_patterns, ext_config.db_safe_mode or true, ext_config.auto_validate or true)
|
|
end,
|
|
exports = {
|
|
frecency = frecency,
|
|
get_workspace_tags = get_workspace_tags,
|
|
validate_db = db_client.validate,
|
|
health = checkhealth,
|
|
},
|
|
}
|