2018-09-19 19:15:29 +10:00
|
|
|
package gui
|
|
|
|
|
|
|
|
import (
|
2022-06-17 21:10:19 +02:00
|
|
|
"fmt"
|
2018-09-19 19:15:29 +10:00
|
|
|
"os"
|
|
|
|
"path/filepath"
|
2022-06-17 21:10:19 +02:00
|
|
|
"strings"
|
2022-07-29 22:53:05 +02:00
|
|
|
"sync"
|
2018-09-19 19:15:29 +10:00
|
|
|
|
2022-03-19 19:12:58 +11:00
|
|
|
"github.com/jesseduffield/generics/slices"
|
2022-08-09 21:27:12 +10:00
|
|
|
appTypes "github.com/jesseduffield/lazygit/pkg/app/types"
|
2018-09-19 19:15:29 +10:00
|
|
|
"github.com/jesseduffield/lazygit/pkg/commands"
|
2020-09-27 16:17:26 +10:00
|
|
|
"github.com/jesseduffield/lazygit/pkg/env"
|
2022-06-17 21:10:19 +02:00
|
|
|
"github.com/jesseduffield/lazygit/pkg/gui/presentation/icons"
|
2021-07-27 15:00:37 +02:00
|
|
|
"github.com/jesseduffield/lazygit/pkg/gui/style"
|
2022-01-29 19:09:20 +11:00
|
|
|
"github.com/jesseduffield/lazygit/pkg/gui/types"
|
2022-07-31 08:44:42 +02:00
|
|
|
"github.com/jesseduffield/lazygit/pkg/utils"
|
2018-09-19 19:15:29 +10:00
|
|
|
)
|
|
|
|
|
2022-06-17 21:10:19 +02:00
|
|
|
func (gui *Gui) getCurrentBranch(path string) string {
|
2022-07-30 19:02:19 +02:00
|
|
|
readHeadFile := func(path string) (string, error) {
|
2022-09-13 18:11:03 +08:00
|
|
|
headFile, err := os.ReadFile(filepath.Join(path, "HEAD"))
|
2022-07-30 19:02:19 +02:00
|
|
|
if err == nil {
|
|
|
|
content := strings.TrimSpace(string(headFile))
|
2022-07-31 08:44:42 +02:00
|
|
|
refsPrefix := "ref: refs/heads/"
|
2023-03-19 11:20:29 +11:00
|
|
|
var branchDisplay string
|
2022-07-31 08:44:42 +02:00
|
|
|
if strings.HasPrefix(content, refsPrefix) {
|
|
|
|
// is a branch
|
|
|
|
branchDisplay = strings.TrimPrefix(content, refsPrefix)
|
|
|
|
} else {
|
|
|
|
// detached HEAD state, displaying short SHA
|
|
|
|
branchDisplay = utils.ShortSha(content)
|
|
|
|
}
|
|
|
|
return branchDisplay, nil
|
2022-07-30 19:02:19 +02:00
|
|
|
}
|
|
|
|
return "", err
|
2022-06-17 21:10:19 +02:00
|
|
|
}
|
2022-07-30 19:02:19 +02:00
|
|
|
|
2022-07-31 08:43:31 +02:00
|
|
|
gitDirPath := filepath.Join(path, ".git")
|
2022-07-30 19:02:19 +02:00
|
|
|
|
|
|
|
if gitDir, err := os.Stat(gitDirPath); err == nil {
|
|
|
|
if gitDir.IsDir() {
|
|
|
|
// ordinary repo
|
|
|
|
if branch, err := readHeadFile(gitDirPath); err == nil {
|
|
|
|
return branch
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// worktree
|
2022-09-13 18:11:03 +08:00
|
|
|
if worktreeGitDir, err := os.ReadFile(gitDirPath); err == nil {
|
2022-07-30 19:02:19 +02:00
|
|
|
content := strings.TrimSpace(string(worktreeGitDir))
|
|
|
|
worktreePath := strings.TrimPrefix(content, "gitdir: ")
|
|
|
|
if branch, err := readHeadFile(worktreePath); err == nil {
|
|
|
|
return branch
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-07-31 08:57:57 +02:00
|
|
|
return gui.c.Tr.LcBranchUnknown
|
2022-06-17 21:10:19 +02:00
|
|
|
}
|
|
|
|
|
2020-08-15 17:23:16 +10:00
|
|
|
func (gui *Gui) handleCreateRecentReposMenu() error {
|
2022-08-01 22:10:08 +10:00
|
|
|
// we'll show an empty panel if there are no recent repos
|
|
|
|
recentRepoPaths := []string{}
|
|
|
|
if len(gui.c.GetAppState().RecentRepos) > 0 {
|
|
|
|
// we skip the first one because we're currently in it
|
|
|
|
recentRepoPaths = gui.c.GetAppState().RecentRepos[1:]
|
|
|
|
}
|
2022-06-17 21:10:19 +02:00
|
|
|
|
2022-07-29 22:53:05 +02:00
|
|
|
currentBranches := sync.Map{}
|
|
|
|
|
|
|
|
wg := sync.WaitGroup{}
|
|
|
|
wg.Add(len(recentRepoPaths))
|
|
|
|
|
|
|
|
for _, path := range recentRepoPaths {
|
|
|
|
go func(path string) {
|
|
|
|
defer wg.Done()
|
|
|
|
currentBranches.Store(path, gui.getCurrentBranch(path))
|
|
|
|
}(path)
|
|
|
|
}
|
|
|
|
|
|
|
|
wg.Wait()
|
|
|
|
|
2022-06-17 21:10:19 +02:00
|
|
|
menuItems := slices.Map(recentRepoPaths, func(path string) *types.MenuItem {
|
|
|
|
branchName, _ := currentBranches.Load(path)
|
|
|
|
if icons.IsIconEnabled() {
|
2022-07-29 22:53:05 +02:00
|
|
|
branchName = icons.BRANCH_ICON + " " + fmt.Sprintf("%v", branchName)
|
2022-06-17 21:10:19 +02:00
|
|
|
}
|
2021-07-27 15:00:37 +02:00
|
|
|
|
2022-03-19 19:12:58 +11:00
|
|
|
return &types.MenuItem{
|
2022-05-08 14:23:32 +10:00
|
|
|
LabelColumns: []string{
|
2020-09-29 09:02:44 +10:00
|
|
|
filepath.Base(path),
|
2022-06-17 21:10:19 +02:00
|
|
|
style.FgCyan.Sprint(branchName),
|
2021-07-27 15:00:37 +02:00
|
|
|
style.FgMagenta.Sprint(path),
|
2020-02-14 23:26:09 +11:00
|
|
|
},
|
2022-01-28 20:44:36 +11:00
|
|
|
OnPress: func() error {
|
2021-04-03 13:43:43 +11:00
|
|
|
// if we were in a submodule, we want to forget about that stack of repos
|
|
|
|
// so that hitting escape in the new repo does nothing
|
2022-01-28 20:44:36 +11:00
|
|
|
gui.RepoPathStack.Clear()
|
2021-04-05 13:22:03 +10:00
|
|
|
return gui.dispatchSwitchToRepo(path, false)
|
2020-02-14 23:26:09 +11:00
|
|
|
},
|
2018-09-19 19:15:29 +10:00
|
|
|
}
|
2022-03-19 19:12:58 +11:00
|
|
|
})
|
2018-09-19 19:15:29 +10:00
|
|
|
|
2022-01-29 19:09:20 +11:00
|
|
|
return gui.c.Menu(types.CreateMenuOptions{Title: gui.c.Tr.RecentRepos, Items: menuItems})
|
2018-09-19 19:15:29 +10:00
|
|
|
}
|
|
|
|
|
2020-11-27 16:07:14 +09:00
|
|
|
func (gui *Gui) handleShowAllBranchLogs() error {
|
2022-01-16 14:46:53 +11:00
|
|
|
cmdObj := gui.git.Branch.AllBranchesLogCmdObj()
|
2022-08-07 11:34:53 +10:00
|
|
|
task := types.NewRunPtyTask(cmdObj.GetCmd())
|
2020-11-27 16:07:14 +09:00
|
|
|
|
2022-08-07 11:34:53 +10:00
|
|
|
return gui.c.RenderToMainViews(types.RefreshMainOpts{
|
|
|
|
Pair: gui.c.MainViewPairs().Normal,
|
|
|
|
Main: &types.ViewUpdateOpts{
|
|
|
|
Title: gui.c.Tr.LogTitle,
|
|
|
|
Task: task,
|
2020-11-27 16:07:14 +09:00
|
|
|
},
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2021-04-05 13:22:03 +10:00
|
|
|
func (gui *Gui) dispatchSwitchToRepo(path string, reuse bool) error {
|
2020-09-29 09:02:44 +10:00
|
|
|
env.UnsetGitDirEnvs()
|
2021-03-30 22:17:42 +11:00
|
|
|
originalPath, err := os.Getwd()
|
|
|
|
if err != nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-09-29 09:02:44 +10:00
|
|
|
if err := os.Chdir(path); err != nil {
|
2021-03-30 22:17:42 +11:00
|
|
|
if os.IsNotExist(err) {
|
2022-01-16 14:46:53 +11:00
|
|
|
return gui.c.ErrorMsg(gui.c.Tr.ErrRepositoryMovedOrDeleted)
|
2021-03-30 22:17:42 +11:00
|
|
|
}
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2022-02-06 13:42:17 +11:00
|
|
|
if err := commands.VerifyInGitRepo(gui.os); err != nil {
|
2021-03-30 22:17:42 +11:00
|
|
|
if err := os.Chdir(originalPath); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2020-09-29 09:02:44 +10:00
|
|
|
return err
|
|
|
|
}
|
2021-03-30 22:17:42 +11:00
|
|
|
|
2022-01-31 22:11:34 +11:00
|
|
|
if err := gui.recordCurrentDirectory(); err != nil {
|
2020-09-29 09:02:44 +10:00
|
|
|
return err
|
|
|
|
}
|
2021-04-03 13:43:43 +11:00
|
|
|
|
2022-01-15 12:04:00 +11:00
|
|
|
// these two mutexes are used by our background goroutines (triggered via `gui.goEvery`. We don't want to
|
|
|
|
// switch to a repo while one of these goroutines is in the process of updating something
|
2022-01-29 11:22:35 +11:00
|
|
|
gui.Mutexes.SyncMutex.Lock()
|
|
|
|
defer gui.Mutexes.SyncMutex.Unlock()
|
2021-04-03 19:35:45 +11:00
|
|
|
|
2022-01-15 12:04:00 +11:00
|
|
|
gui.Mutexes.RefreshingFilesMutex.Lock()
|
|
|
|
defer gui.Mutexes.RefreshingFilesMutex.Unlock()
|
2021-04-03 19:35:45 +11:00
|
|
|
|
2022-08-09 21:27:12 +10:00
|
|
|
return gui.onNewRepo(appTypes.StartArgs{}, reuse)
|
2020-09-29 09:02:44 +10:00
|
|
|
}
|
|
|
|
|
2018-09-19 19:15:29 +10:00
|
|
|
// updateRecentRepoList registers the fact that we opened lazygit in this repo,
|
|
|
|
// so that we can open the same repo via the 'recent repos' menu
|
|
|
|
func (gui *Gui) updateRecentRepoList() error {
|
2022-08-15 13:59:34 +01:00
|
|
|
isBareRepo, err := gui.git.Status.IsBareRepo()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if isBareRepo {
|
2020-09-27 16:02:20 +10:00
|
|
|
// we could totally do this but it would require storing both the git-dir and the
|
|
|
|
// worktree in our recent repos list, which is a change that would need to be
|
|
|
|
// backwards compatible
|
2022-01-16 14:46:53 +11:00
|
|
|
gui.c.Log.Info("Not appending bare repo to recent repo list")
|
2020-09-27 16:02:20 +10:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2022-01-16 14:46:53 +11:00
|
|
|
recentRepos := gui.c.GetAppState().RecentRepos
|
2018-09-19 19:15:29 +10:00
|
|
|
currentRepo, err := os.Getwd()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2018-12-10 13:45:03 +01:00
|
|
|
known, recentRepos := newRecentReposList(recentRepos, currentRepo)
|
2021-12-29 11:50:20 +11:00
|
|
|
gui.IsNewRepo = known
|
2022-01-16 14:46:53 +11:00
|
|
|
gui.c.GetAppState().RecentRepos = recentRepos
|
|
|
|
return gui.c.SaveAppState()
|
2018-09-19 19:15:29 +10:00
|
|
|
}
|
|
|
|
|
2018-12-06 22:05:16 +01:00
|
|
|
// newRecentReposList returns a new repo list with a new entry but only when it doesn't exist yet
|
2018-12-10 13:45:03 +01:00
|
|
|
func newRecentReposList(recentRepos []string, currentRepo string) (bool, []string) {
|
|
|
|
isNew := true
|
2018-09-19 19:15:29 +10:00
|
|
|
newRepos := []string{currentRepo}
|
|
|
|
for _, repo := range recentRepos {
|
|
|
|
if repo != currentRepo {
|
2022-06-17 21:20:37 +02:00
|
|
|
if _, err := os.Stat(filepath.Join(repo, ".git")); err != nil {
|
2022-06-09 15:57:54 +02:00
|
|
|
continue
|
|
|
|
}
|
2018-09-19 19:15:29 +10:00
|
|
|
newRepos = append(newRepos, repo)
|
2018-12-10 13:45:03 +01:00
|
|
|
} else {
|
|
|
|
isNew = false
|
2018-09-19 19:15:29 +10:00
|
|
|
}
|
|
|
|
}
|
2018-12-10 13:45:03 +01:00
|
|
|
return isNew, newRepos
|
2018-09-19 19:15:29 +10:00
|
|
|
}
|