Files
nvim-cmp/lua/cmp/menu.lua

233 lines
5.8 KiB
Lua

local debug = require('cmp.utils.debug')
local types = require('cmp.types')
local async = require('cmp.utils.async')
local float = require('cmp.float')
local config = require('cmp.config')
local autocmd = require('cmp.utils.autocmd')
---@class cmp.MenuOption
---@field on_select fun(e: cmp.Entry)
---@class cmp.Menu
---@field public float cmp.Float
---@field public cache cmp.Cache
---@field public offset number
---@field public on_select fun(e: cmp.Entry)
---@field public items vim.CompletedItem[]
---@field public entries cmp.Entry[]
---@field public deduped_entries cmp.Entry[]
---@field public selected_entry cmp.Entry|nil
---@field public context cmp.Context
---@field public resolve_dedup fun(callback: function)
local menu = {}
---Create menu
---@param opts cmp.MenuOption
---@return cmp.Menu
menu.new = function(opts)
local self = setmetatable({}, { __index = menu })
self.float = float.new()
self.resolve_dedup = async.dedup()
self.on_select = opts.on_select or function() end
self:reset()
autocmd.subscribe('CompleteChanged', function()
local e = self:get_selected_entry()
if e then
self:select(e)
else
self:unselect()
end
end)
return self
end
---Close menu
menu.close = function(self)
vim.schedule(function()
debug.log('menu.close', vim.fn.pumvisible())
if vim.fn.pumvisible() == 1 then
-- TODO: Is it safe to call...?
local line = vim.api.nvim_win_get_cursor(0)[1]
vim.fn.complete(#vim.api.nvim_buf_get_lines(0, line - 1, line, true)[1] + 1, {})
end
self:unselect()
end)
end
---Reset menu
menu.reset = function(self)
self.offset = nil
self.items = {}
self.entries = {}
self.deduped_entries = {}
self.context = nil
self.preselect = 0
self:close()
end
---Update menu
---@param ctx cmp.Context
---@param sources cmp.Source[]
---@return cmp.Menu
menu.update = function(self, ctx, sources)
local entries = {}
-- check the source triggered by character
local has_triggered_by_symbol_source = false
for _, s in ipairs(sources) do
if #s:get_entries(ctx) > 0 then
if s.is_triggered_by_symbol then
has_triggered_by_symbol_source = true
break
end
end
end
-- create filtered entries.
local offset = ctx.cursor.col
for i, s in ipairs(sources) do
if s.offset <= offset then
if not has_triggered_by_symbol_source or s.is_triggered_by_symbol then
-- source order priority bonus.
local priority = s:get_config().priority or ((#sources - (i - 1)) * config.get().sorting.priority_weight)
for _, e in ipairs(s:get_entries(ctx)) do
e.score = e.score + priority
table.insert(entries, e)
offset = math.min(offset, e:get_offset())
end
end
end
end
-- sort.
table.sort(entries, function(e1, e2)
for _, fn in ipairs(config.get().sorting.comparators) do
local diff = fn(e1, e2)
if diff ~= nil then
return diff
end
end
end)
-- create vim items.
local items = {}
local deduped_entries = {}
local deduped_words = {}
local preselect = 0
for _, e in ipairs(entries) do
local item = e:get_vim_item(offset)
if item.dup == 1 or not deduped_words[item.word] then
deduped_words[item.word] = true
-- We have done deduplication already, no need to force Vim to repeat it.
item.dup = 1
table.insert(items, item)
table.insert(deduped_entries, e)
if preselect == 0 and e.completion_item.preselect and config.get().preselect ~= types.cmp.PreselectMode.None then
preselect = #deduped_entries
end
end
end
-- save recent pum state.
self.offset = offset
self.items = items
self.entries = entries
self.deduped_entries = deduped_entries
self.preselect = preselect
self.context = ctx
self:show()
end
---Restore previous menu
---@param ctx cmp.Context
menu.restore = function(self, ctx)
if not ctx.pumvisible then
if #self.items > 0 then
if self.offset <= ctx.cursor.col then
debug.log('menu/restore')
self:show()
end
end
end
end
---Show completion item
menu.show = function(self)
if #self.deduped_entries == 0 then
self:close()
return
end
debug.log('menu.show', #self.deduped_entries)
local completeopt = vim.o.completeopt
if self.preselect == 1 then
vim.opt.completeopt = { 'menuone', 'noinsert' }
else
vim.opt.completeopt = config.get().completion.completeopt
end
vim.fn.complete(self.offset, self.items)
if self.preselect > 1 then
vim.api.nvim_select_popupmenu_item(self.preselect - 1, false, false, {})
end
vim.opt.completeopt = completeopt
end
---Select current item
---@param e cmp.Entry
menu.select = function(self, e)
-- Documentation (always invoke to follow to the pum position)
e:resolve(self.resolve_dedup(vim.schedule_wrap(function()
if self:get_selected_entry() == e then
self.float:show(e)
end
end)))
self.on_select(e)
end
---Select current item
menu.unselect = function(self)
self.float:close()
end
---Geta current active entry
---@return cmp.Entry|nil
menu.get_active_entry = function(self)
if vim.fn.pumvisible() == 0 or not (vim.v.completed_item or {}).user_data then
return nil
end
return self:get_selected_entry()
end
---Get current selected entry
---@return cmp.Entry|nil
menu.get_selected_entry = function(self)
if not self:is_valid_mode() then
return nil
end
local selected = vim.fn.complete_info({ 'selected' }).selected
if selected == -1 then
return nil
end
return self.deduped_entries[math.max(selected, 0) + 1]
end
---Get first entry
---@param self cmp.Entry|nil
menu.get_first_entry = function(self)
if not self:is_valid_mode() then
return nil
end
return self.deduped_entries[1]
end
---Return the completion menu is visible or not.
---@return boolean
menu.is_valid_mode = function()
return vim.fn.complete_info({ 'mode' }).mode == 'eval'
end
return menu