summaryrefslogtreecommitdiff
path: root/lua/telescope/builtin/git.lua
diff options
context:
space:
mode:
authorSimon Hauser <Simon-Hauser@outlook.de>2020-11-16 21:10:26 +0100
committerGitHub <noreply@github.com>2020-11-16 15:10:26 -0500
commit81172630270325947d2c6b3afda5599e34b48499 (patch)
tree575dde0f6ad204926e010e3b861738d469ac6026 /lua/telescope/builtin/git.lua
parentc01b7ad024cbab82d6c54feebf1eb6f86a2e8d2e (diff)
feat: Add git builtins (#240)
Conni2461 is a great coder and contributor :)
Diffstat (limited to 'lua/telescope/builtin/git.lua')
-rw-r--r--lua/telescope/builtin/git.lua177
1 files changed, 177 insertions, 0 deletions
diff --git a/lua/telescope/builtin/git.lua b/lua/telescope/builtin/git.lua
new file mode 100644
index 0000000..0f90403
--- /dev/null
+++ b/lua/telescope/builtin/git.lua
@@ -0,0 +1,177 @@
+local actions = require('telescope.actions')
+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)
+
+ -- 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 File',
+ finder = finders.new_oneshot_job(
+ { "git", "ls-files", "--exclude-standard", "--cached", show_untracked and "--others" },
+ opts
+ ),
+ previewer = previewers.cat.new(opts),
+ sorter = conf.file_sorter(opts),
+ }):find()
+end
+
+git.commits = function(opts)
+ local cmd = 'git log --pretty=oneline --abbrev-commit'
+ local results = vim.split(utils.get_os_command_output(cmd), '\n')
+
+ pickers.new(opts, {
+ prompt_title = 'Git Commits',
+ finder = finders.new_table {
+ results = results,
+ entry_maker = make_entry.gen_from_git_commits(opts),
+ },
+ previewer = previewers.git_commit_diff.new(opts),
+ sorter = conf.file_sorter(opts),
+ attach_mappings = function()
+ actions.goto_file_selection_edit:replace(actions.git_checkout)
+ return true
+ end
+ }):find()
+end
+
+git.bcommits = function(opts)
+ local cmd = 'git log --pretty=oneline --abbrev-commit ' .. vim.fn.expand('%')
+ local results = vim.split(utils.get_os_command_output(cmd), '\n')
+
+ pickers.new(opts, {
+ prompt_title = 'Git BCommits',
+ finder = finders.new_table {
+ results = results,
+ entry_maker = make_entry.gen_from_git_commits(opts),
+ },
+ previewer = previewers.git_commit_diff.new(opts),
+ sorter = conf.file_sorter(opts),
+ attach_mappings = function()
+ actions.goto_file_selection_edit:replace(actions.git_checkout)
+ return true
+ end
+ }):find()
+end
+
+git.branches = function(opts)
+ -- Does this command in lua (hopefully):
+ -- 'git branch --all | grep -v HEAD | sed "s/.* //;s#remotes/[^/]*/##" | sort -u'
+ local output = vim.split(utils.get_os_command_output('git branch --all'), '\n')
+
+ local tmp_results = {}
+ for _, v in ipairs(output) do
+ if not string.match(v, 'HEAD') and v ~= '' then
+ v = string.gsub(v, '.* ', '')
+ v = string.gsub(v, '^remotes/.*/', '')
+ tmp_results[v] = true
+ end
+ end
+
+ local results = {}
+ for k, _ in pairs(tmp_results) do
+ table.insert(results, k)
+ end
+
+ pickers.new(opts, {
+ prompt_title = 'Git Branches',
+ finder = finders.new_table {
+ results = results,
+ entry_maker = function(entry)
+ return {
+ value = entry,
+ ordinal = entry,
+ display = entry,
+ }
+ end
+ },
+ previewer = previewers.git_branch_log.new(opts),
+ sorter = conf.file_sorter(opts),
+ attach_mappings = function()
+ actions.goto_file_selection_edit:replace(actions.git_checkout)
+ return true
+ end
+ }):find()
+end
+
+git.status = function(opts)
+ local output = vim.split(utils.get_os_command_output('git status -s'), '\n')
+ local results = {}
+ for _, v in ipairs(output) do
+ if v ~= "" then
+ local mod, fname = string.match(v, '(..)%s(.+)')
+ if mod ~= 'A ' and mod ~= 'M ' and mod ~= 'R ' and mod ~= 'D ' then
+ table.insert(results, { mod = mod, file = fname })
+ end
+ end
+ end
+
+ if vim.tbl_isempty(results) then
+ print('No changes found')
+ return
+ end
+
+ pickers.new(opts, {
+ prompt_title = 'Git Status',
+ finder = finders.new_table {
+ results = results,
+ entry_maker = function(entry)
+ return {
+ value = entry.file,
+ ordinal = entry.mod .. ' ' .. entry.file,
+ display = entry.mod .. ' ' .. entry.file,
+ }
+ end
+ },
+ previewer = previewers.git_file_diff.new(opts),
+ sorter = conf.file_sorter(opts),
+ attach_mappings = function()
+ actions.goto_file_selection_edit:replace(actions.git_add)
+ return true
+ end
+ }):find()
+end
+
+local set_opts_cwd = function(opts)
+ local is_git_dir = function(path)
+ vim.fn.system('cd ' .. path .. ' && git rev-parse HEAD >/dev/null 2>&1')
+ if vim.v.shell_error ~= 0 then
+ error(path .. ' is not a git directory')
+ end
+ end
+
+ if opts.cwd then
+ opts.cwd = vim.fn.expand(opts.cwd)
+ is_git_dir(opts)
+ else
+ is_git_dir(vim.fn.expand('%:p:h'))
+ --- Find root of git directory and remove trailing newline characters
+ opts.cwd = vim.fn.systemlist("git rev-parse --show-toplevel")[1]
+ 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)