A lightweight bundle of commands focussed on swift and streamlined git operations.
- Smart-Commit: Open a popup to enter a commit message with syntax highlighting,
commit preview, automatic issue number insertion, and overlength indicators.
If there are no staged changes, stages all changes before doing so (
git add -A
). Optionally trigger agit push
afterward. - Quick commands for amend, stash, fixup, or undoing commits.
- Search issues & PRs. Open the selected issue or PR in the browser.
- Open the GitHub URL of the current line or selection.
- Explore the git history: Search the file for a string ("git pickaxe"), or examine a function's or line's history. Displays the results in a diff view with syntax highlighting, correctly following file renamings.
- Statusline components:
git blame
and branch state. - Streamlined workflow: operations are smartly combined to minimize friction. For instance, the smart-commit command combines staging, committing, and pushing, and searching the file history combines unshallowing, searching, and navigating diffs.
- Installation
- Commands
- Statusline Components
- Configuration
- Comparison to existing git plugins
- About me
Hard Requirements
- nvim 0.10 or higher
dressing.nvim
Optional/Recommended Requirements
- Treesitter parser for syntax highlighting:
TSInstall gitcommit
nvim-notify
for the commit preview, issue number insertion, and various notifications- telescope.nvim or fzf-lua for better UI when selecting commits or issues/PRs.
-- lazy.nvim
{
"chrisgrieser/nvim-tinygit",
dependencies = "stevearc/dressing.nvim",
},
-- packer
use {
"chrisgrieser/nvim-tinygit",
requires = "stevearc/dressing.nvim",
}
- Open a commit popup, alongside a preview of what is going to be committed. If
there are no staged changes, stage all changes (
git add --all
) before the commit. - Input field contents of aborted commits are briefly kept, if you just want to fix a detail.
- Optionally run
git push
if the repo is clean after committing. - The title of the input field displays what actions are going to be performed.
You can see at glance whether all changes are going to be committed, or whether
there a
git push
is triggered afterward, so there are no surprises. - Typing
#
inserts the most recent issue number,<Tab>
cycles through the issues (currently opt-in, see plugin configuration). - Only supports the commit subject line (no commit body).
-- values shown are the defaults
require("tinygit").smartCommit { pushIfClean = false, pullBeforePush = true }
Example Workflow Assuming these keybindings:
vim.keymap.set("n", "ga", "<cmd>Gitsigns add_hunk<CR>") -- gitsigns.nvim
vim.keymap.set("n", "gc", function() require("tinygit").smartCommit() end)
vim.keymap.set("n", "gp", function() require("tinygit").push() end)
- Stage some hunks (changes) via
ga
. - Use
gc
to enter a commit message. - Repeat 1 and 2.
- When done,
gp
to push the commits.
Using pushIfClean = true
allows you to combine staging, committing, and
pushing into a single step, when it is the last commit you intend to make.
amendOnlyMsg
just opens the commit popup to change the last commit message, and does not stage any changes.amendNoEdit
keeps the last commit message; if there are no staged changes, stages all changes (git add --all
), likesmartCommit
.- Optionally runs
git push --force-with-lease
afterward, if the branch has diverged (that is, the amended commit was already pushed).
-- options default to `false`
require("tinygit").amendOnlyMsg { forcePushIfDiverged = false }
require("tinygit").amendNoEdit { forcePushIfDiverged = false }
fixupCommit
lets you select a commit from the last X commits and runsgit commit --fixup
on the selected commit.- If there are no staged changes, stages all changes (
git add --all
), likesmartCommit
. - Use
squashInstead = true
to squash instead of fixup (git commit --squash
). autoRebase = true
automatically runs rebase with--autosquash
and--autostash
afterward, confirming all fixups and squashes without opening a rebase view. (Note that this can potentially result in conflicts.)
-- options show default values
require("tinygit").fixupCommit {
selectFromLastXCommits = 15,
squashInstead = false,
autoRebase = false,
}
require("tinygit").undoLastCommitOrAmend()
- Changes in the working directory are kept, but unstaged. (In the background,
this uses
git reset --mixed
.) - If there was a
push
operation done as a followup (such as.smartCommit { pushIfClean = false }
), the last commit is not undone.
- Search issues & PRs.
- Requires
curl
.
-- state: all|closed|open (default: all)
-- type: all|issue|pr (default: all)
require("tinygit").issuesAndPrs { type = "all", state = "all" }
-- alternative: if the word under the cursor is of the form `#123`,
-- just open that issue/PR
require("tinygit").openIssueUnderCursor()
- Open the current file at GitHub in the browser and copy the URL to the system clipboard.
- Normal mode: open the current file or repo.
- Visual mode: open the current selection.
-- file|repo (default: file)
require("tinygit").githubUrl("file")
push
can be combined with other actions, depending on the options.createGitHubPr
opens a PR from the current branch browser.- This requires the repo to be a fork with sufficient information on the remote.
- This does not require the
gh
cli, as it uses a GitHub web feature.
-- options default to `false`
require("tinygit").push {
pullBefore = false,
forceWithLease = false,
createGitHubPr = false,
}
require("tinygit").createGitHubPr()
Search the git history. Select from the matching commits to open a popup with a diffview of the changes.
- Search the git history of the current file (
git log -G
).- The search is case-insensitive and supports regex.
- Correctly follows file renamings, and displays past file names in the commit selection.
- Leave the input field empty to display all commits that changed the current file.
- Explore the history of a function in the current file (
git log -L
).- The search is literal.
- If the current buffer has an LSP with support for document symbols attached, you can select a function. (Otherwise, you are prompted to enter a function name.)
- Note that
git
uses heuristics to determine the enclosing function of a change, so this is not 100% perfect and has varying reliability across languages.
- Go through the history of the current line (range) (
git log -L
).- In normal mode, searches the history of the line under the cursor.
- When called in visual mode, searches the history of the selected line range.
Keymaps in the diff popup
<Tab>
: show older commit<S-Tab>
: show newer commityh
: yank the commit hash to the system clipboardR
: restore file to state at commitn
/N
: go to the next/previous occurrence of the query (only file history)
require("tinygit").searchFileHistory()
require("tinygit").functionHistory()
require("tinygit").lineHistory()
Simple wrappers around git stash push
and git stash pop
.
require("tinygit").stashPush()
require("tinygit").stashPop()
Shows the message and date (git blame
) of the last commit that changed the
current file (not line).
require("tinygit.statusline").blame()
Tip
Some status line plugins also allow you to put components into the tabline or winbar. If your status line is too crowded, you can add the blame-component to the one of those bars instead.
The component can be configured with the statusline.blame
options in the plugin
configuration.
Shows whether the local branch is ahead or behind of its remote counterpart.
(Note that this component does not run git fetch
for performance reasons, so
the information may not be up-to-date with remote changes.)
require("tinygit.statusline").branchState()
The setup
call is optional. These are the default settings:
local defaultConfig = {
commitMsg = {
commitPreview = true, -- requires nvim-notify
spellcheck = false,
keepAbortedMsgSecs = 300,
inputFieldWidth = 72, -- `false` to use dressing.nvim config
conventionalCommits = {
enforce = false,
-- stylua: ignore
keywords = {
"fix", "feat", "chore", "docs", "refactor", "build", "test",
"perf", "style", "revert", "ci", "break", "improv",
},
},
openReferencedIssue = false, -- if message has issue/PR, open in browser afterwards
insertIssuesOnHash = {
-- Experimental. Typing `#` will insert the most recent open issue.
-- Requires nvim-notify.
enabled = false,
next = "<Tab>", -- insert & normal mode
prev = "<S-Tab>",
issuesToFetch = 20,
},
},
backdrop = {
enabled = true,
blend = 60, -- 0-100
},
push = {
preventPushingFixupOrSquashCommits = true,
confirmationSound = true, -- currently macOS only, PRs welcome
},
issueIcons = {
openIssue = "🟢",
closedIssue = "🟣",
openPR = "🟩",
mergedPR = "🟪",
closedPR = "🟥",
},
historySearch = {
diffPopup = {
width = 0.8, -- float, 0 to 1
height = 0.8,
border = "single",
},
autoUnshallowIfNeeded = false,
},
statusline = {
blame = {
ignoreAuthors = {}, -- hide component if these authors (useful for bots)
hideAuthorNames = {}, -- show component, but hide names (useful for your own name)
maxMsgLen = 40,
icon = "ﰖ ",
},
branchState = {
icons = {
ahead = "",
behind = "",
diverge = "",
},
},
},
}
The appearance of the commit preview and notifications is determined by nvim-notify. To change for example the width of the preview, use:
require("notify").setup {
max_width = 60,
}
gitsigns.nvim
: No feature overlap.tinygit
rather complementsgitsigns
as the latter is used to stage changes (:GitSigns stage_hunk
) quickly, and the former allows you to commit (and push) those changes quickly.Neogit
/Fugitive
: These two probably cover much more features thantinygit
has, but with much more configuration options. The benefit oftinygit
is that it is more lightweight and aims to streamline common actions by smartly combining operations. For instance, the smart-commit command combines staging, committing, and pushing. As such,tinygit
is mostly for those people that do want a more nimble and lightweight git integration.diffview.nvim
: No overlap, except for the command to search the file history.tinygit
's version of file history search should be easier to use and has a few more quality-of-life features, such as automatically jumping to occurrences of the search term. As opposed todiffview
, the diff is not presented in a side-by-side-diff, but in a unified view.
In my day job, I am a sociologist studying the social mechanisms underlying the digital economy. For my PhD project, I investigate the governance of the app economy and how software ecosystems manage the tension between innovation and compatibility. If you are interested in this subject, feel free to get in touch.
I also occasionally blog about vim: Nano Tips for Vim