local actions = require('telescope.actions') local action_state = require('telescope.actions.state') local finders = require('telescope.finders') local make_entry = require('telescope.make_entry') local pickers = require('telescope.pickers') local previewers = require('telescope.previewers') local utils = require('telescope.utils') local conf = require('telescope.config').values local git = {} git.files = function(opts) local show_untracked = utils.get_default(opts.show_untracked, true) local recurse_submodules = utils.get_default(opts.recurse_submodules, false) if show_untracked and recurse_submodules then error("Git does not suppurt both --others and --recurse-submodules") end -- By creating the entry maker after the cwd options, -- we ensure the maker uses the cwd options when being created. opts.entry_maker = opts.entry_maker or make_entry.gen_from_file(opts) pickers.new(opts, { prompt_title = 'Git Files', finder = finders.new_oneshot_job( vim.tbl_flatten( { "git", "ls-files", "--exclude-standard", "--cached", show_untracked and "--others" or nil, recurse_submodules and "--recurse-submodules" or nil } ), opts ), previewer = conf.file_previewer(opts), sorter = conf.file_sorter(opts), }):find() end git.commits = function(opts) local results = utils.get_os_command_output({ 'git', 'log', '--pretty=oneline', '--abbrev-commit', '--', '.' }, opts.cwd) pickers.new(opts, { prompt_title = 'Git Commits', finder = finders.new_table { results = results, entry_maker = opts.entry_maker or make_entry.gen_from_git_commits(opts), }, previewer = previewers.git_commit_diff.new(opts), sorter = conf.file_sorter(opts), attach_mappings = function() actions.select_default:replace(actions.git_checkout) return true end }):find() end git.bcommits = function(opts) local results = utils.get_os_command_output({ 'git', 'log', '--pretty=oneline', '--abbrev-commit', vim.fn.expand('%') }, opts.cwd) pickers.new(opts, { prompt_title = 'Git BCommits', finder = finders.new_table { results = results, entry_maker = opts.entry_maker or make_entry.gen_from_git_commits(opts), }, previewer = previewers.git_commit_diff.new(opts), sorter = conf.file_sorter(opts), attach_mappings = function() actions.select_default:replace(actions.git_checkout) return true end }):find() end git.branches = function(opts) local output = utils.get_os_command_output({ 'git', 'branch', '--all' }, opts.cwd) local results = {} for _, v in ipairs(output) do if not string.match(v, 'HEAD') and v ~= '' then if vim.startswith(v, '*') then table.insert(results, 1, v) else table.insert(results, v) end end end pickers.new(opts, { prompt_title = 'Git Branches', finder = finders.new_table { results = results, entry_maker = function(entry) local addition = vim.startswith(entry, '*') and '* ' or ' ' entry = entry:gsub('[* ] ', '') entry = entry:gsub('^remotes/', '') return { value = entry, ordinal = addition .. entry, display = addition .. entry } end }, previewer = previewers.git_branch_log.new(opts), sorter = conf.file_sorter(opts), attach_mappings = function(_, map) actions.select_default:replace(actions.git_checkout) map('i', '', actions.git_track_branch) map('n', '', actions.git_track_branch) map('i', '', actions.git_rebase_branch) map('n', '', actions.git_rebase_branch) map('i', '', actions.git_delete_branch) map('n', '', actions.git_delete_branch) map('i', '', false) map('n', '', false) return true end }):find() end git.status = function(opts) local gen_new_finder = function() local expand_dir = utils.if_nil(opts.expand_dir, true, opts.expand_dir) local git_cmd = {'git', 'status', '-s', '--', '.'} if expand_dir then table.insert(git_cmd, table.getn(git_cmd) - 1, '-u') end local output = utils.get_os_command_output(git_cmd, opts.cwd) if table.getn(output) == 0 then print('No changes found') return end return finders.new_table { results = output, entry_maker = opts.entry_maker or make_entry.gen_from_git_status(opts) } end local initial_finder = gen_new_finder() if not initial_finder then return end pickers.new(opts, { prompt_title = 'Git Status', finder = initial_finder, previewer = previewers.git_file_diff.new(opts), sorter = conf.file_sorter(opts), attach_mappings = function(prompt_bufnr, map) actions.git_staging_toggle:enhance { post = function() action_state.get_current_picker(prompt_bufnr):refresh(gen_new_finder(), { reset_prompt = true }) end, } map('i', '', actions.git_staging_toggle) map('n', '', actions.git_staging_toggle) return true end }):find() end local set_opts_cwd = function(opts) if opts.cwd then opts.cwd = vim.fn.expand(opts.cwd) else opts.cwd = vim.loop.cwd() end -- Find root of git directory and remove trailing newline characters local git_root, ret = utils.get_os_command_output({ "git", "rev-parse", "--show-toplevel" }, opts.cwd) local use_git_root = utils.get_default(opts.use_git_root, true) if ret ~= 0 then error(opts.cwd .. ' is not a git directory') else if use_git_root then opts.cwd = git_root[1] end end end local function apply_checks(mod) for k, v in pairs(mod) do mod[k] = function(opts) opts = opts or {} set_opts_cwd(opts) v(opts) end end return mod end return apply_checks(git)