package gui import ( "fmt" "math" "strings" "github.com/jesseduffield/gocui" "github.com/jesseduffield/lazygit/pkg/commands/git_commands" "github.com/jesseduffield/lazygit/pkg/gui/types" "github.com/jesseduffield/lazygit/pkg/utils" ) const HORIZONTAL_SCROLL_FACTOR = 3 // these views need to be re-rendered when the screen mode changes. The commits view, // for example, will show authorship information in half and full screen mode. func (gui *Gui) rerenderViewsWithScreenModeDependentContent() error { for _, view := range []*gocui.View{gui.Views.Branches, gui.Views.Commits} { if err := gui.rerenderView(view); err != nil { return err } } return nil } // TODO: GENERICS func nextIntInCycle(sl []WindowMaximisation, current WindowMaximisation) WindowMaximisation { for i, val := range sl { if val == current { if i == len(sl)-1 { return sl[0] } return sl[i+1] } } return sl[0] } // TODO: GENERICS func prevIntInCycle(sl []WindowMaximisation, current WindowMaximisation) WindowMaximisation { for i, val := range sl { if val == current { if i > 0 { return sl[i-1] } return sl[len(sl)-1] } } return sl[len(sl)-1] } func (gui *Gui) nextScreenMode() error { gui.State.ScreenMode = nextIntInCycle([]WindowMaximisation{SCREEN_NORMAL, SCREEN_HALF, SCREEN_FULL}, gui.State.ScreenMode) return gui.rerenderViewsWithScreenModeDependentContent() } func (gui *Gui) prevScreenMode() error { gui.State.ScreenMode = prevIntInCycle([]WindowMaximisation{SCREEN_NORMAL, SCREEN_HALF, SCREEN_FULL}, gui.State.ScreenMode) return gui.rerenderViewsWithScreenModeDependentContent() } func (gui *Gui) scrollUpView(view *gocui.View) error { ox, oy := view.Origin() newOy := int(math.Max(0, float64(oy-gui.c.UserConfig.Gui.ScrollHeight))) return view.SetOrigin(ox, newOy) } func (gui *Gui) scrollDownView(view *gocui.View) error { ox, oy := view.Origin() scrollHeight := gui.linesToScrollDown(view) if scrollHeight > 0 { if err := view.SetOrigin(ox, oy+scrollHeight); err != nil { return err } } if manager, ok := gui.viewBufferManagerMap[view.Name()]; ok { manager.ReadLines(scrollHeight) } return nil } func (gui *Gui) linesToScrollDown(view *gocui.View) int { _, oy := view.Origin() y := oy canScrollPastBottom := gui.c.UserConfig.Gui.ScrollPastBottom if !canScrollPastBottom { _, sy := view.Size() y += sy } scrollHeight := gui.c.UserConfig.Gui.ScrollHeight scrollableLines := view.ViewLinesHeight() - y if scrollableLines < 0 { return 0 } // margin is about how many lines must still appear if you scroll // all the way down. In practice every file ends in a newline so it will really // just show a single line margin := 1 if canScrollPastBottom { margin = 2 } if scrollableLines-margin < scrollHeight { scrollHeight = scrollableLines - margin } if oy+scrollHeight < 0 { return 0 } else { return scrollHeight } } func (gui *Gui) scrollUpMain() error { if gui.renderingConflicts() { gui.State.Panels.Merging.UserVerticalScrolling = true } return gui.scrollUpView(gui.Views.Main) } func (gui *Gui) scrollDownMain() error { if gui.renderingConflicts() { gui.State.Panels.Merging.UserVerticalScrolling = true } return gui.scrollDownView(gui.Views.Main) } func (gui *Gui) scrollLeftMain() error { gui.scrollLeft(gui.Views.Main) return nil } func (gui *Gui) scrollRightMain() error { gui.scrollRight(gui.Views.Main) return nil } func (gui *Gui) scrollLeft(view *gocui.View) { newOriginX := utils.Max(view.OriginX()-view.InnerWidth()/HORIZONTAL_SCROLL_FACTOR, 0) _ = view.SetOriginX(newOriginX) } func (gui *Gui) scrollRight(view *gocui.View) { _ = view.SetOriginX(view.OriginX() + view.InnerWidth()/HORIZONTAL_SCROLL_FACTOR) } func (gui *Gui) scrollUpSecondary() error { return gui.scrollUpView(gui.Views.Secondary) } func (gui *Gui) scrollDownSecondary() error { return gui.scrollDownView(gui.Views.Secondary) } func (gui *Gui) scrollUpConfirmationPanel() error { if gui.Views.Confirmation.Editable { return nil } return gui.scrollUpView(gui.Views.Confirmation) } func (gui *Gui) scrollDownConfirmationPanel() error { if gui.Views.Confirmation.Editable { return nil } return gui.scrollDownView(gui.Views.Confirmation) } func (gui *Gui) handleRefresh() error { return gui.c.Refresh(types.RefreshOptions{Mode: types.ASYNC}) } func (gui *Gui) handleMouseDownMain() error { switch gui.currentSideContext() { case gui.State.Contexts.CommitFiles: return gui.enterCommitFile(types.OnFocusOpts{ClickedViewName: "main", ClickedViewLineIdx: gui.Views.Main.SelectedLineIdx()}) } return nil } func (gui *Gui) fetch() (err error) { gui.c.LogAction("Fetch") err = gui.git.Sync.Fetch(git_commands.FetchOptions{}) if err != nil && strings.Contains(err.Error(), "exit status 128") { _ = gui.c.ErrorMsg(gui.c.Tr.PassUnameWrong) } _ = gui.c.Refresh(types.RefreshOptions{Scope: []types.RefreshableView{types.BRANCHES, types.COMMITS, types.REMOTES, types.TAGS}, Mode: types.ASYNC}) return err } func (gui *Gui) backgroundFetch() (err error) { err = gui.git.Sync.Fetch(git_commands.FetchOptions{Background: true}) _ = gui.c.Refresh(types.RefreshOptions{Scope: []types.RefreshableView{types.BRANCHES, types.COMMITS, types.REMOTES, types.TAGS}, Mode: types.ASYNC}) return err } func (gui *Gui) handleCopySelectedSideContextItemToClipboard() error { // important to note that this assumes we've selected an item in a side context itemId := gui.getSideContextSelectedItemId() if itemId == "" { return nil } gui.c.LogAction(gui.c.Tr.Actions.CopyToClipboard) if err := gui.os.CopyToClipboard(itemId); err != nil { return gui.c.Error(err) } truncatedItemId := utils.TruncateWithEllipsis(strings.Replace(itemId, "\n", " ", -1), 50) gui.c.Toast(fmt.Sprintf("'%s' %s", truncatedItemId, gui.c.Tr.LcCopiedToClipboard)) return nil }