1
0
mirror of https://github.com/SpaceVim/SpaceVim.git synced 2025-01-25 09:10:03 +08:00
SpaceVim/lua/spacevim/plugin/flygrep.lua

847 lines
24 KiB
Lua
Raw Normal View History

2022-10-10 23:03:23 +08:00
--=============================================================================
-- flygrep.lua --- grep on the fly in SpaceVim
2023-03-26 13:44:47 +08:00
-- Copyright (c) 2016-2023 Wang Shidong & Contributors
2022-10-10 23:03:23 +08:00
-- Author: Wang Shidong < wsdjeg@outlook.com >
-- URL: https://spacevim.org
-- License: GPLv3
--=============================================================================
local M = {}
2020-09-19 22:57:11 +08:00
2022-10-04 21:25:05 +08:00
local logger = require('spacevim.logger').derive('flygrep')
local mpt = require('spacevim.api').import('prompt')
local hi = require('spacevim.api').import('vim.highlight')
local regex = require('spacevim.api').import('vim.regex')
local Key = require('spacevim.api').import('vim.keys')
2022-10-07 11:08:33 +08:00
local buffer = require('spacevim.api').import('vim.buffer')
2022-10-10 13:58:09 +08:00
local window = require('spacevim.api').import('vim.window')
local sl = require('spacevim.api').import('vim.statusline')
local nt = require('spacevim.api').import('notify')
2022-10-04 21:25:05 +08:00
-- set commandline mpt
mpt._prompt.mpt = vim.g.spacevim_commandline_prompt .. ' '
-- compatibility functions
2022-10-04 21:25:05 +08:00
local jobstart = vim.fn.jobstart
local jobstop = vim.fn.jobstop
local function empty(expr)
2023-06-16 23:30:23 +08:00
return vim.fn.empty(expr) == 1
end
local function isdirectory(dir)
2023-06-16 23:30:23 +08:00
return vim.fn.isdirectory(dir) == 1
end
local function noautocmd(f) -- {{{
local ei = vim.o.eventignore
vim.o.eventignore = 'all'
pcall(f)
vim.o.eventignore = ei
end
-- }}}
2022-10-04 21:25:05 +08:00
local timer_start = vim.fn.timer_start
local timer_stop = vim.fn.timer_stop
-- the script local values, same as s: in vim script
local previous_winid = -1
local grep_expr = ''
2023-06-16 23:30:23 +08:00
local grep_default_exe, grep_default_opt, grep_default_ropt, grep_default_expr_opt, grep_default_fix_string_opt, grep_default_ignore_case, grep_default_smart_case =
require('spacevim.plugin.search').default_tool()
local grep_timer_id = -1
local preview_timer_id = -1
2023-06-17 00:26:53 +08:00
local preview_bufnr = -1
local grepid = 0
local mode = ''
local buffer_id = -1
local flygrep_win_id = -1
2023-06-19 21:44:24 +08:00
--- @type string|table
local grep_files = {}
local grep_dir = ''
local grep_exe = ''
local grep_opt = {}
local grep_ropt = {}
local grep_ignore_case = {}
local grep_smart_case = {}
local grep_expr_opt = {}
local search_hi_id = -1
local filter_hi_id = -1
local grep_mode = 'expr'
local filename_pattern = [[[^:]*:\d\+:\d\+:]]
2022-10-10 13:58:09 +08:00
local preview_able = false
2022-10-07 11:08:33 +08:00
local grep_history = {}
2022-10-10 13:58:09 +08:00
local preview_win_id = -1
2022-10-10 19:23:12 +08:00
local filter_file = ''
2023-06-20 21:30:09 +08:00
--- @return table # a list of searching pattern history
local function read_histroy()
2023-06-16 23:30:23 +08:00
if
vim.fn.filereadable(vim.fn.expand(vim.g.spacevim_data_dir .. 'SpaceVim/flygrep_history'))
== 1
then
local _his = vim.fn.json_decode(
vim.fn.join(
vim.fn.readfile(vim.fn.expand(vim.g.spacevim_data_dir .. 'SpaceVim/flygrep_history'), ''),
''
)
)
2023-06-20 21:30:09 +08:00
if type(_his) == table then
return _his or {}
else
2023-06-16 23:30:23 +08:00
return {}
end
2023-06-16 23:30:23 +08:00
else
return {}
end
end
2022-10-07 11:08:33 +08:00
grep_history = read_histroy()
local function update_history()
2023-06-16 23:30:23 +08:00
if vim.fn.index(grep_history, grep_expr) >= 0 then
vim.fn.remove(grep_history, vim.fn.index(grep_history, grep_expr))
end
2023-06-18 20:24:32 +08:00
table.insert(grep_history, grep_expr)
2023-06-16 23:30:23 +08:00
if vim.fn.isdirectory(vim.fn.expand(vim.g.spacevim_data_dir .. 'SpaceVim')) == 0 then
vim.fn.mkdir(vim.fn.expand(vim.g.spacevim_data_dir .. 'SpaceVim'))
end
if
vim.fn.filereadable(vim.fn.expand(vim.g.spacevim_data_dir .. 'SpaceVim/flygrep_history'))
== 1
then
vim.fn.writefile(
{ vim.fn.json_encode(grep_history) },
vim.fn.expand(vim.g.spacevim_data_dir .. 'SpaceVim/flygrep_history')
)
end
end
local function append(t1, t2)
2023-06-16 23:30:23 +08:00
for _, v in pairs(t2) do
table.insert(t1, v)
end
end
2023-06-20 21:30:09 +08:00
--- @param expr string # searching pattern
--- @return table<string> # searching command
local function get_search_cmd(expr)
2023-06-16 23:30:23 +08:00
local cmd = { grep_exe }
append(cmd, grep_opt)
if vim.o.ignorecase then
append(cmd, grep_ignore_case)
end
if vim.o.smartcase then
append(cmd, grep_smart_case)
end
if grep_mode == 'string' then
append(cmd, grep_default_fix_string_opt)
end
append(cmd, grep_expr_opt)
if not empty(grep_files) and vim.fn.type(grep_files) == 3 then
append(cmd, { expr })
append(cmd, grep_files)
elseif not empty(grep_files) and vim.fn.type(grep_files) == 1 then
append(cmd, { expr })
append(cmd, { grep_files })
elseif not empty(grep_dir) then
if grep_exe == 'findstr' then
append(cmd, { grep_dir, expr, [[%CD%\*]] })
else
2023-06-16 23:30:23 +08:00
append(cmd, { expr, grep_dir })
end
2023-06-16 23:30:23 +08:00
else
append(cmd, { expr })
if grep_exe == 'rg' or grep_exe == 'ag' or grep_exe == 'pt' then
append(cmd, { '.' })
end
append(cmd, grep_ropt)
end
return cmd
end
2023-06-27 17:31:47 +08:00
local function update_statusline()
if sl.support_float() and vim.fn.win_id2tabwin(flygrep_win_id)[1] == vim.fn.tabpagenr() then
sl.open_float({
{ 'FlyGrep ', 'SpaceVim_statusline_a_bold' },
{ '', 'SpaceVim_statusline_a_SpaceVim_statusline_b' },
{ M.mode() .. ' ', 'SpaceVim_statusline_b' },
{ '', 'SpaceVim_statusline_b_SpaceVim_statusline_c' },
{ vim.fn.getcwd() .. ' ', 'SpaceVim_statusline_c' },
{ '', 'SpaceVim_statusline_c_SpaceVim_statusline_b' },
{ M.lineNr() .. ' ', 'SpaceVim_statusline_b' },
{ '', 'SpaceVim_statusline_b_SpaceVim_statusline_z' },
{ vim.fn['repeat'](' ', vim.o.columns - 11), 'SpaceVim_statusline_z' },
})
end
end
2023-06-16 23:30:23 +08:00
local complete_input_history_num = { 0, 0 }
2023-06-19 21:44:24 +08:00
local function grep_stdout(id, data, _)
2023-06-16 23:30:23 +08:00
-- ignore previous result
if id ~= grepid then
return
end
noautocmd(function()
local datas = vim.fn.filter(data, '!empty(v:val)')
if vim.fn.getbufline(buffer_id, 1)[1] == '' then
vim.api.nvim_buf_set_lines(buffer_id, 0, -1, false, datas)
else
vim.api.nvim_buf_set_lines(buffer_id, -1, -1, false, datas)
end
2023-06-27 17:31:47 +08:00
vim.cmd('redraw')
update_statusline()
end)
end
2023-06-19 21:44:24 +08:00
local function grep_stderr(_, data, _)
2023-06-16 23:30:23 +08:00
for _, d in pairs(data) do
logger.info('grep stderr:' .. d)
end
end
local function close_statusline()
2023-06-16 23:30:23 +08:00
sl.close_float()
end
2023-06-19 21:44:24 +08:00
local function grep_exit(id, data, _)
2023-06-18 00:28:46 +08:00
if id ~= grepid then
return
end
2023-06-16 23:30:23 +08:00
logger.info('grep exit:' .. data)
update_statusline()
vim.cmd('redraw')
mpt._build_prompt()
grepid = 0
end
2022-10-04 21:25:05 +08:00
-- The available options are:
-- - input: string, the default input pattern
-- - files: a list of string or `@buffers`
-- - cmd: list
-- - opt: list
-- - ropt: list
-- - ignore_case: boolean
-- - smart_case: boolean
2023-06-16 23:30:23 +08:00
-- - expr_opt:
2020-09-19 22:57:11 +08:00
local current_grep_pattern = ''
2023-06-19 21:44:24 +08:00
local function grep_timer(_)
2023-06-16 23:30:23 +08:00
if grep_mode == 'expr' then
current_grep_pattern = vim.fn.join(vim.fn.split(grep_expr), '.*')
else
current_grep_pattern = grep_expr
end
local cmd = get_search_cmd(current_grep_pattern)
logger.info('grep cmd:' .. vim.inspect(cmd))
grepid = jobstart(cmd, {
on_stdout = grep_stdout,
on_stderr = grep_stderr,
on_exit = grep_exit,
})
logger.info('flygrep job id is:' .. grepid)
end
local function matchadd(group, pattern, p)
2023-06-19 21:44:24 +08:00
local _, id = pcall(vim.fn.matchadd, group, pattern, p)
return id
end
2022-10-04 21:25:05 +08:00
local function expr_to_pattern(expr)
2023-06-16 23:30:23 +08:00
if grep_mode == 'expr' then
local items = vim.fn.split(expr)
local pattern = vim.fn.join(items, '.*')
local ignorecase = ''
if vim.o.ignorecase then
ignorecase = [[\c]]
else
2023-06-16 23:30:23 +08:00
ignorecase = [[\C]]
end
2023-06-16 23:30:23 +08:00
pattern = filename_pattern .. [[.*\zs]] .. ignorecase .. regex.parser(pattern, false)
logger.info('matchadd pattern: ' .. pattern)
return pattern
else
return expr
end
2022-10-04 21:25:05 +08:00
end
local function flygrep(t)
2023-06-18 00:28:46 +08:00
-- if the insert text is empty, clear grepid
grepid = 0
update_statusline()
2023-06-16 23:30:23 +08:00
mpt._build_prompt()
if t == '' then
return
end
pcall(vim.fn.matchdelete, search_hi_id)
search_hi_id = matchadd('FlygrepSearchPattern', expr_to_pattern(t), 2)
2023-06-16 23:30:23 +08:00
grep_expr = t
timer_stop(grep_timer_id)
grep_timer_id = timer_start(200, grep_timer, { ['repeat'] = 1 })
end
2022-10-04 21:25:05 +08:00
local function close_flygrep_win()
pcall(vim.api.nvim_win_close, flygrep_win_id, true)
2023-06-16 23:30:23 +08:00
vim.fn.win_gotoid(previous_winid)
end
2022-10-10 13:58:09 +08:00
local function get_file_pos(line)
2023-06-16 23:30:23 +08:00
local filename = vim.fn.fnameescape(vim.fn.split(line, [[:\d\+:]])[1])
local linenr = vim.fn.str2nr(string.sub(vim.fn.matchstr(line, [[:\d\+:]]), 2, -2))
local colum = vim.fn.str2nr(string.sub(vim.fn.matchstr(line, [[\(:\d\+\)\@<=:\d\+:]]), 2, -2))
return filename, linenr, colum
2022-10-10 13:58:09 +08:00
end
2023-06-19 21:44:24 +08:00
local function preview_timer(_)
2023-06-16 23:30:23 +08:00
local cursor = vim.api.nvim_win_get_cursor(flygrep_win_id)
local line = vim.api.nvim_buf_get_lines(buffer_id, cursor[1] - 1, cursor[1], false)[1]
if line == '' then
return
end
local filename, liner, colum = get_file_pos(line)
2023-06-17 00:26:53 +08:00
if vim.fn.bufexists(preview_bufnr) ~= 1 then
preview_bufnr = vim.api.nvim_create_buf(false, true)
end
2023-06-16 23:30:23 +08:00
local flygrep_win_height = 16
2023-06-17 00:26:53 +08:00
if not window.is_float(preview_win_id) then
noautocmd(function()
preview_win_id = vim.api.nvim_open_win(preview_bufnr, false, {
relative = 'editor',
width = vim.o.columns,
height = 8,
row = vim.o.lines - flygrep_win_height - 2 - 8,
col = 0,
})
end)
2023-06-16 23:30:23 +08:00
end
2023-06-17 00:26:53 +08:00
vim.api.nvim_buf_set_lines(preview_bufnr, 0, -1, false, vim.fn.readfile(filename, ''))
local ft = vim.filetype.match({ filename = filename })
if ft then
vim.api.nvim_buf_set_option(preview_bufnr, 'syntax', ft)
else
local ftdetect_autocmd = vim.api.nvim_get_autocmds({
group = 'filetypedetect',
event = 'BufRead',
pattern = '*.' .. vim.fn.fnamemodify(filename, ':e'),
})
-- logger.info(vim.inspect(ftdetect_autocmd))
if ftdetect_autocmd[1] then
if
ftdetect_autocmd[1].command and vim.startswith(ftdetect_autocmd[1].command, 'set filetype=')
then
ft = ftdetect_autocmd[1].command:gsub('set filetype=', '')
vim.api.nvim_buf_set_option(preview_bufnr, 'syntax', ft)
end
end
end
2023-06-19 21:44:24 +08:00
vim.api.nvim_win_set_cursor(preview_win_id, { liner, colum })
2023-06-16 23:30:23 +08:00
mpt._build_prompt()
2022-10-10 13:58:09 +08:00
end
local function preview()
2023-06-16 23:30:23 +08:00
timer_stop(preview_timer_id)
preview_timer_id = timer_start(200, preview_timer, { ['repeat'] = 1 })
2022-10-10 13:58:09 +08:00
end
local function close_preview_win()
2023-06-16 23:30:23 +08:00
pcall(vim.api.nvim_win_close, preview_win_id, true)
2022-10-10 13:58:09 +08:00
end
local function close_buffer()
2023-06-16 23:30:23 +08:00
if grepid > 0 then
grepid = 0
2023-06-16 23:30:23 +08:00
jobstop(grepid)
end
timer_stop(grep_timer_id)
timer_stop(preview_timer_id)
if preview_able then
close_preview_win()
preview_able = false
end
close_flygrep_win()
vim.cmd('noautocmd normal :')
2022-10-04 21:25:05 +08:00
end
mpt._onclose = close_buffer
local function close_grep_job()
2023-06-16 23:30:23 +08:00
if grepid > 0 then
pcall(jobstop, grepid)
end
timer_stop(grep_timer_id)
timer_stop(preview_timer_id)
vim.api.nvim_buf_set_lines(buffer_id, 0, -1, false, {})
update_statusline()
complete_input_history_num = { 0, 0 }
end
mpt._oninputpro = close_grep_job
local function next_item()
2023-06-16 23:30:23 +08:00
local cursor = vim.api.nvim_win_get_cursor(flygrep_win_id)
if cursor[1] >= vim.api.nvim_buf_line_count(buffer_id) then
cursor[1] = 1
else
cursor[1] = cursor[1] + 1
end
vim.api.nvim_win_set_cursor(flygrep_win_id, cursor)
if preview_able then
preview()
end
update_statusline()
vim.cmd('redraw')
mpt._build_prompt()
end
local function previous_item()
2023-06-16 23:30:23 +08:00
local cursor = vim.api.nvim_win_get_cursor(flygrep_win_id)
if cursor[1] == 1 then
cursor[1] = vim.api.nvim_buf_line_count(buffer_id)
else
cursor[1] = cursor[1] - 1
end
vim.api.nvim_win_set_cursor(flygrep_win_id, cursor)
if preview_able then
preview()
end
2023-06-16 23:30:23 +08:00
update_statusline()
vim.cmd('redraw')
mpt._build_prompt()
end
local function open_item(...)
2023-06-16 23:30:23 +08:00
local argv = { ... }
local edit_command = argv[1] or 'edit'
mpt._handle_fly = flygrep
local cursor = vim.api.nvim_win_get_cursor(flygrep_win_id)
local line = vim.api.nvim_buf_get_lines(buffer_id, cursor[1] - 1, cursor[1], false)[1]
-- print(vim.inspect(line))
if line ~= '' then
if grepid ~= 0 then
-- change grepid to 0, and callback function will be skipped
grepid = 0
2023-06-16 23:30:23 +08:00
jobstop(grepid)
end
mpt._clear_prompt()
mpt._quit = true
local filename, liner, colum = get_file_pos(line)
if preview_able then
close_preview_win()
2022-10-07 11:08:33 +08:00
end
2023-06-16 23:30:23 +08:00
preview_able = false
close_flygrep_win()
update_history()
buffer.open_pos(edit_command, filename, liner, colum)
vim.cmd('noautocmd normal! :')
end
2022-10-07 11:08:33 +08:00
end
local function open_item_in_tab()
2023-06-16 23:30:23 +08:00
open_item('tabedit')
end
local function open_item_vertically()
2023-06-16 23:30:23 +08:00
open_item('vsplit')
end
local function open_item_horizontally()
2023-06-16 23:30:23 +08:00
open_item('split')
end
local function move_cursor()
2023-06-16 23:30:23 +08:00
if vim.v.mouse_winid == flygrep_win_id then
vim.api.nvim_win_set_cursor(flygrep_win_id, { vim.v.mouse_lnum, 0 })
end
mpt._build_prompt()
end
local function double_click()
2023-06-16 23:30:23 +08:00
if vim.v.mouse_winid == flygrep_win_id then
vim.api.nvim_win_set_cursor(flygrep_win_id, { vim.v.mouse_lnum, 0 })
end
open_item()
end
local function toggle_expr_mode()
2023-06-16 23:30:23 +08:00
if grep_mode == 'expr' then
grep_mode = 'string'
else
grep_mode = 'expr'
end
mpt._oninputpro()
mpt._handle_fly(mpt._prompt.cursor_begin .. mpt._prompt.cursor_char .. mpt._prompt.cursor_end)
end
local function apply_to_quickfix()
2023-06-16 23:30:23 +08:00
mpt._handle_fly = flygrep
if vim.fn.getbufline(buffer_id, 1)[1] ~= '' then
if grepid ~= 0 then
-- stop job, and skip callback function
grepid = 0
2023-06-16 23:30:23 +08:00
jobstop(grepid)
end
mpt._quit = true
if preview_able then
close_preview_win()
end
2023-06-16 23:30:23 +08:00
preview_able = false
local searching_result = vim.api.nvim_buf_get_lines(buffer_id, 0, -1, false)
close_flygrep_win()
update_history()
if vim.fn.empty(searching_result) == 0 then
-- vim.cmd('cgetexpr ' .. vim.fn.join(searching_result, "\n"))
-- vim.cmd([[
-- cgetexpr join(luaeval(searching_result), "\n")
-- ]])
-- vim.fn.setqflist({})
vim.fn.setqflist({}, 'r', {
title = 'FlyGrep partten:'
.. mpt._prompt.cursor_begin
.. mpt._prompt.cursor_char
.. mpt._prompt.cursor_end,
lines = searching_result,
})
mpt._clear_prompt()
-- use botright to make sure quicfix windows width same as screen
vim.cmd('botright copen')
2023-06-16 23:30:23 +08:00
end
vim.cmd('noautocmd normal! :')
end
end
2022-10-10 13:58:09 +08:00
local function toggle_preview()
2023-06-16 23:30:23 +08:00
if not preview_able then
preview_able = true
preview()
else
close_preview_win()
preview_able = false
end
vim.cmd('redraw')
mpt._build_prompt()
2022-10-10 13:58:09 +08:00
end
2022-10-10 19:23:12 +08:00
local function get_filter_cmd(expr)
2023-06-16 23:30:23 +08:00
local cmd = { grep_exe }
append(cmd, require('spacevim.plugin.search').getFopt(grep_exe))
append(cmd, { expr, filter_file })
return cmd
2022-10-10 19:23:12 +08:00
end
2023-06-19 21:44:24 +08:00
local function filter_timer(_)
2023-06-16 23:30:23 +08:00
local cmd = get_filter_cmd(vim.fn.join(vim.fn.split(grep_expr), '.*'))
grepid = jobstart(cmd, {
on_stdout = grep_stdout,
on_exit = grep_exit,
})
2022-10-10 19:23:12 +08:00
end
local function filter(expr)
2023-06-16 23:30:23 +08:00
mpt._build_prompt()
pcall(vim.fn.matchdelete, filter_hi_id)
2023-06-16 23:30:23 +08:00
if expr == '' then
-- if the mpt is empty, put context in filter_file into flygrep buffer
if vim.fn.filereadable(filter_file) then
vim.api.nvim_buf_set_lines(buffer_id, 0, -1, false, vim.fn.readfile(filter_file, ''))
vim.cmd('redraw')
end
2023-06-16 23:30:23 +08:00
return
end
filter_hi_id = matchadd('FlygrepFilterPattern', expr_to_pattern(expr), 2)
2023-06-16 23:30:23 +08:00
grep_expr = expr
grep_timer_id = timer_start(200, filter_timer, { ['repeat'] = 1 })
2022-10-10 19:23:12 +08:00
end
local function start_filter()
2023-06-16 23:30:23 +08:00
mode = 'f'
update_statusline()
mpt._handle_fly = filter
mpt._clear_prompt()
filter_file = vim.fn.tempname()
local context = vim.api.nvim_buf_get_lines(buffer_id, 0, -1, false)
2023-06-19 21:44:24 +08:00
local ok, _ = pcall(vim.fn.writefile, context, filter_file, 'b')
2023-06-16 23:30:23 +08:00
if not ok then
logger.info('Failed to write filter content to temp file')
end
mpt._build_prompt()
2022-10-10 19:23:12 +08:00
end
local function tbl_filter(func, t) -- {{{
local rettab = {}
for _, entry in pairs(t) do
if func(entry) then
table.insert(rettab, entry)
end
end
return rettab
end
-- }}}
2023-06-17 22:20:00 +08:00
local function complete_input_history(str, num) -- {{{
-- logger.info(vim.inspect(grep_history))
-- local results = vim.fn.filter(, "v:val =~# '^' . a:str")
2023-06-18 00:28:46 +08:00
local results = tbl_filter(function(node)
2023-06-17 22:20:00 +08:00
-- here the note sometimes do not have title, then it is nil
2023-06-18 00:28:46 +08:00
if type(node) ~= 'string' then
return false
end
return vim.startswith(node, str)
end, vim.deepcopy(grep_history))
2023-06-18 00:28:46 +08:00
logger.info(vim.inspect(results))
2023-06-17 22:20:00 +08:00
local complete_items
if not empty(results) and results[-1] ~= str then
2023-06-18 00:28:46 +08:00
complete_items = results
table.insert(complete_items, str)
2023-06-17 22:20:00 +08:00
elseif empty(results) then
complete_items = { str }
else
complete_items = results
end
-- 5 0 6
local patch = (num[1] - num[2]) % vim.fn.len(complete_items)
local index
if patch >= 0 then
index = vim.fn.len(complete_items) - patch
else
index = vim.fn.abs(patch)
end
return complete_items[index]
end
-- }}}
local complete_input_history_base = ''
local function previous_match_history()
if complete_input_history_num[1] == 0 and complete_input_history_num[2] == 0 then
complete_input_history_base = mpt._prompt.cursor_begin
mpt._prompt.cursor_char = ''
mpt._prompt.cursor_end = ''
end
complete_input_history_num[1] = complete_input_history_num[1] + 1
mpt._prompt.cursor_begin =
complete_input_history(complete_input_history_base, complete_input_history_num)
2023-06-18 00:28:46 +08:00
vim.api.nvim_buf_set_lines(buffer_id, 0, -1, false, {})
2023-06-17 22:20:00 +08:00
mpt._handle_fly(mpt._prompt.cursor_begin .. mpt._prompt.cursor_char .. mpt._prompt.cursor_end)
end
local function next_match_history()
if complete_input_history_num[1] == 0 and complete_input_history_num[2] == 0 then
complete_input_history_base = mpt._prompt.cursor_begin
mpt._prompt.cursor_char = ''
mpt._prompt.cursor_end = ''
end
complete_input_history_num[2] = complete_input_history_num[2] + 1
mpt._prompt.cursor_begin =
complete_input_history(complete_input_history_base, complete_input_history_num)
2023-06-18 00:28:46 +08:00
vim.api.nvim_buf_set_lines(buffer_id, 0, -1, false, {})
2023-06-17 22:20:00 +08:00
mpt._handle_fly(mpt._prompt.cursor_begin .. mpt._prompt.cursor_char .. mpt._prompt.cursor_end)
end
local function page_up()
-- exe "noautocmd normal! \<PageUp>"
vim.api.nvim_win_call(flygrep_win_id, function()
vim.api.nvim_feedkeys(Key.t('<PageUp>'), 'x', false)
end)
if preview_able then
preview()
end
update_statusline()
vim.cmd('redraw')
mpt._build_prompt()
end
local function page_down()
-- exe "noautocmd normal! \<PageUp>"
vim.api.nvim_win_call(flygrep_win_id, function()
vim.api.nvim_feedkeys(Key.t('<PageDown>'), 'x', false)
end)
if preview_able then
preview()
end
update_statusline()
vim.cmd('redraw')
mpt._build_prompt()
end
2023-06-17 22:20:00 +08:00
2023-06-19 16:58:42 +08:00
local function update_files(f) -- {{{
end
-- }}}
local function flygrep_result_to_files() -- {{{
end
-- }}}
local function start_replace()
mode = 'r'
pcall(vim.fn.matchdelete, search_hi_id)
if grepid ~= 0 then
jobstop(grepid)
end
local replace_text = current_grep_pattern
local rst
if not empty(replace_text) then
rst = require('spacevim.plugin.iedit').start({ expr = replace_text }, 1, vim.fn.line('$'))
end
search_hi_id = vim.fn.matchadd('FlyGrepPattern', expr_to_pattern(rst), 2)
update_statusline()
if rst ~= replace_text then
update_files(flygrep_result_to_files())
vim.cmd('checktime')
end
end
mpt._function_key = {
2023-06-16 23:30:23 +08:00
[Key.t('<Tab>')] = next_item,
[Key.t('<C-j>')] = next_item,
[Key.t('<ScrollWheelDown>')] = next_item,
[Key.t('<S-tab>')] = previous_item,
[Key.t('<C-k>')] = previous_item,
[Key.t('<ScrollWheelUp>')] = previous_item,
[Key.t('<Return>')] = open_item,
[Key.t('<C-t>')] = open_item_in_tab,
[Key.t('<LeftMouse>')] = move_cursor,
[Key.t('<2-LeftMouse>')] = double_click,
[Key.t('<C-f>')] = start_filter,
[Key.t('<C-v>')] = open_item_vertically,
[Key.t('<C-s>')] = open_item_horizontally,
[Key.t('<C-q>')] = apply_to_quickfix,
[Key.t('<M-r>')] = start_replace,
[Key.t('<C-p>')] = toggle_preview,
[Key.t('<C-e>')] = toggle_expr_mode,
[Key.t('<Up>')] = previous_match_history,
[Key.t('<Down>')] = next_match_history,
[Key.t('<PageDown>')] = page_down,
[Key.t('<PageUp>')] = page_up,
2023-06-19 21:44:24 +08:00
-- [Key.t('<C-End>')] = page_end,
-- [Key.t('<C-Home>')] = page_home,
2023-06-16 23:30:23 +08:00
[Key.t('x80\xfdK')] = previous_item,
[Key.t('x80\xfc \x80\xfdK')] = previous_item,
[Key.t('x80\xfc@\x80\xfdK')] = previous_item,
[Key.t('x80\xfc`\x80\xfdK')] = previous_item,
[Key.t('x80\xfdL')] = next_item,
[Key.t('x80\xfc \x80\xfdL')] = next_item,
[Key.t('x80\xfc@\x80\xfdL')] = next_item,
[Key.t('x80\xfc`\x80\xfdL')] = next_item,
}
function M.mode()
2023-06-19 21:44:24 +08:00
local _, iedit_mode = pcall(vim.api.nvim_win_get_var, flygrep_win_id, 'spacevim_iedit_mode')
2023-06-19 16:58:42 +08:00
if iedit_mode == 'n' then
return 'iedit-normal'
elseif iedit_mode == 'i' then
return 'iedit-insert'
2023-06-16 23:30:23 +08:00
else
2023-06-19 16:58:42 +08:00
if mode == '' then
return grep_mode
else
return grep_mode .. '(' .. mode .. ')'
end
2023-06-16 23:30:23 +08:00
end
end
function M.lineNr()
2023-06-16 23:30:23 +08:00
if vim.fn.getbufline(buffer_id, 1)[1] == '' then
return 'no result'
else
local current = vim.api.nvim_win_get_cursor(flygrep_win_id)[1]
local total = vim.api.nvim_buf_line_count(buffer_id)
return current .. '/' .. total
end
end
function M.open(argv)
2023-06-16 23:30:23 +08:00
previous_winid = vim.fn.win_getid()
if empty(grep_default_exe) then
logger.warn('make sure you have one search tool in your PATH')
nt.notify('make sure you have one search tool in your PATH')
return
elseif type(argv.cmd) == "string" and vim.fn.empty(argv.cmd) == 0 and vim.fn.executable(argv.cmd) == 0 then
logger.warn(argv.cmd .. ' is not executable, make sure ' .. argv.cmd .. ' is in your PATH')
nt.notify(argv.cmd .. ' is not executable,\nmake sure ' .. argv.cmd .. ' is in your PATH', 'WarningMsg')
2023-06-16 23:30:23 +08:00
return
end
mode = ''
mpt._handle_fly = flygrep
buffer_id = vim.api.nvim_create_buf(false, true)
local flygrep_win_height = 16
noautocmd(function()
flygrep_win_id = vim.api.nvim_open_win(buffer_id, true, {
relative = 'editor',
width = vim.o.columns,
height = flygrep_win_height,
row = vim.o.lines - flygrep_win_height - 2,
col = 0,
})
end)
2023-06-16 23:30:23 +08:00
if vim.fn.exists('&winhighlight') == 1 then
vim.cmd('set winhighlight=Normal:Pmenu,EndOfBuffer:Pmenu,CursorLine:PmenuSel')
end
vim.cmd(
'setlocal buftype=nofile bufhidden=wipe nobuflisted nolist noswapfile nowrap cursorline nospell nonu norelativenumber'
)
vim.opt_local.fillchars = { eob = ' ' }
local save_tve = vim.o.t_ve
vim.cmd('setlocal t_ve=')
local cursor_hi = {}
cursor_hi = hi.group2dict('Cursor')
local lcursor_hi = {}
lcursor_hi = hi.group2dict('lCursor')
local guicursor = vim.o.guicursor
hi.hide_in_normal('Cursor')
hi.hide_in_normal('lCursor')
if vim.fn.has('nvim') == 1 then
vim.cmd('set guicursor+=a:Cursor/lCursor')
end
vim.cmd('setf SpaceVimFlyGrep')
update_statusline()
matchadd('FileName', filename_pattern, 3)
vim.cmd('hi def link FlygrepSearchPattern MoreMsg')
vim.cmd('hi def link FlygrepFilterPattern Question')
2023-06-16 23:30:23 +08:00
mpt._prompt.cursor_begin = argv.input or ''
local fs = argv.files or ''
if fs == '@buffers' then
grep_files = vim.fn.map(buffer.listed_buffers(), 'bufname(v:val)')
elseif not empty(fs) then
grep_files = fs
else
grep_files = ''
end
local dir = vim.fn.expand(argv.dir or '')
if not empty(dir) and isdirectory(dir) then
grep_dir = dir
else
grep_dir = ''
end
grep_exe = argv.cmd or grep_default_exe
if empty(grep_dir) and empty(grep_files) and grep_exe == 'findstr' then
grep_files = '*.*'
elseif grep_exe == 'findstr' and not empty(grep_dir) then
grep_dir = '/D:' .. grep_dir
end
grep_opt = argv.opt or grep_default_opt
grep_ropt = argv.ropt or grep_default_ropt
grep_ignore_case = argv.ignore_case or grep_default_ignore_case
grep_smart_case = argv.smart_case or grep_default_smart_case
grep_expr_opt = argv.expr_opt or grep_default_expr_opt
logger.info('FlyGrep startting ===========================')
logger.info(' executable : ' .. grep_exe)
logger.info(' option : ' .. vim.fn.string(grep_opt))
logger.info(' r_option : ' .. vim.fn.string(grep_ropt))
logger.info(' files : ' .. vim.fn.string(grep_files))
logger.info(' dir : ' .. vim.fn.string(grep_dir))
logger.info(' ignore_case : ' .. vim.fn.string(grep_ignore_case))
logger.info(' smart_case : ' .. vim.fn.string(grep_smart_case))
logger.info(' expr opt : ' .. vim.fn.string(grep_expr_opt))
mpt.open()
if sl.support_float() then
close_statusline()
end
logger.info('FlyGrep ending =====================')
vim.o.t_ve = save_tve
hi.hi(cursor_hi)
hi.hi(lcursor_hi)
vim.o.guicursor = guicursor
2020-09-19 22:57:11 +08:00
end
2022-10-04 21:25:05 +08:00
return M