1
0
mirror of https://github.com/jesseduffield/lazygit.git synced 2025-01-24 05:36:19 +02:00
lazygit/pkg/gui/global_handlers.go

254 lines
6.0 KiB
Go
Raw Normal View History

2020-03-29 10:31:34 +11:00
package gui
import (
"fmt"
2020-03-29 10:31:34 +11:00
"math"
"strings"
2020-08-11 21:18:38 +10:00
"github.com/jesseduffield/lazygit/pkg/commands"
2020-03-29 10:31:34 +11:00
"github.com/jesseduffield/lazygit/pkg/utils"
)
2020-08-19 19:07:14 +10:00
// 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.
2020-08-19 19:26:05 +10:00
func (gui *Gui) rerenderViewsWithScreenModeDependentContent() error {
for _, viewName := range []string{"branches", "commits"} {
2020-08-19 19:07:14 +10:00
if err := gui.rerenderView(viewName); err != nil {
return err
}
2020-03-29 10:31:34 +11:00
}
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)
2020-08-19 19:26:05 +10:00
return gui.rerenderViewsWithScreenModeDependentContent()
}
func (gui *Gui) prevScreenMode() error {
gui.State.ScreenMode = prevIntInCycle([]WindowMaximisation{SCREEN_NORMAL, SCREEN_HALF, SCREEN_FULL}, gui.State.ScreenMode)
2020-08-19 19:07:14 +10:00
2020-08-19 19:26:05 +10:00
return gui.rerenderViewsWithScreenModeDependentContent()
2020-03-29 10:31:34 +11:00
}
func (gui *Gui) scrollUpView(viewName string) error {
2020-08-23 15:59:22 +10:00
mainView, err := gui.g.View(viewName)
if err != nil {
return nil
}
2020-03-29 10:31:34 +11:00
ox, oy := mainView.Origin()
2020-10-03 14:54:55 +10:00
newOy := int(math.Max(0, float64(oy-gui.Config.GetUserConfig().Gui.ScrollHeight)))
2020-03-29 10:31:34 +11:00
return mainView.SetOrigin(ox, newOy)
}
func (gui *Gui) scrollDownView(viewName string) error {
2020-08-23 15:59:22 +10:00
mainView, err := gui.g.View(viewName)
if err != nil {
return nil
}
2020-03-29 10:31:34 +11:00
ox, oy := mainView.Origin()
y := oy
2021-04-02 15:25:27 +11:00
canScrollPastBottom := gui.Config.GetUserConfig().Gui.ScrollPastBottom
if !canScrollPastBottom {
2020-03-29 10:31:34 +11:00
_, sy := mainView.Size()
y += sy
}
2020-10-03 14:54:55 +10:00
scrollHeight := gui.Config.GetUserConfig().Gui.ScrollHeight
2021-04-02 15:25:27 +11:00
scrollableLines := mainView.ViewLinesHeight() - y
if scrollableLines > 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
}
2021-04-02 19:49:23 +11:00
if oy+scrollHeight >= 0 {
if err := mainView.SetOrigin(ox, oy+scrollHeight); err != nil {
return err
}
2020-03-29 10:31:34 +11:00
}
}
if manager, ok := gui.viewBufferManagerMap[viewName]; ok {
manager.ReadLines(scrollHeight)
}
return nil
}
func (gui *Gui) scrollUpMain() error {
if gui.canScrollMergePanel() {
gui.State.Panels.Merging.UserScrolling = true
}
2020-03-29 10:31:34 +11:00
return gui.scrollUpView("main")
}
func (gui *Gui) scrollDownMain() error {
if gui.canScrollMergePanel() {
gui.State.Panels.Merging.UserScrolling = true
}
2020-03-29 10:31:34 +11:00
return gui.scrollDownView("main")
}
func (gui *Gui) scrollUpSecondary() error {
2020-03-29 10:31:34 +11:00
return gui.scrollUpView("secondary")
}
func (gui *Gui) scrollDownSecondary() error {
2020-03-29 10:31:34 +11:00
return gui.scrollDownView("secondary")
}
func (gui *Gui) scrollUpConfirmationPanel() error {
view := gui.getConfirmationView()
if view != nil || view.Editable {
2020-03-29 10:31:34 +11:00
return nil
}
2020-03-29 10:31:34 +11:00
return gui.scrollUpView("confirmation")
}
func (gui *Gui) scrollDownConfirmationPanel() error {
view := gui.getConfirmationView()
if view != nil || view.Editable {
2020-03-29 10:31:34 +11:00
return nil
}
2020-03-29 10:31:34 +11:00
return gui.scrollDownView("confirmation")
}
func (gui *Gui) handleRefresh() error {
2020-03-29 10:31:34 +11:00
return gui.refreshSidePanels(refreshOptions{mode: ASYNC})
}
func (gui *Gui) handleMouseDownMain() error {
2020-03-29 10:31:34 +11:00
if gui.popupPanelFocused() {
return nil
}
view := gui.getMainView()
switch gui.g.CurrentView().Name() {
2020-03-29 10:31:34 +11:00
case "files":
2020-08-16 18:22:55 +10:00
// set filename, set primary/secondary selected, set line number, then switch context
// I'll need to know it was changed though.
// Could I pass something along to the context change?
return gui.enterFile(false, view.SelectedLineIdx())
2020-03-29 10:31:34 +11:00
case "commitFiles":
return gui.enterCommitFile(view.SelectedLineIdx())
2020-03-29 10:31:34 +11:00
}
return nil
}
func (gui *Gui) handleMouseDownSecondary() error {
2020-03-29 10:31:34 +11:00
if gui.popupPanelFocused() {
return nil
}
view := gui.getSecondaryView()
switch gui.g.CurrentView().Name() {
2020-03-29 10:31:34 +11:00
case "files":
return gui.enterFile(true, view.SelectedLineIdx())
2020-03-29 10:31:34 +11:00
}
return nil
}
func (gui *Gui) handleInfoClick() error {
2020-03-29 10:31:34 +11:00
if !gui.g.Mouse {
return nil
}
view := gui.getInformationView()
cx, _ := view.Cursor()
width, _ := view.Size()
2020-03-29 10:31:34 +11:00
2020-08-23 09:23:59 +10:00
for _, mode := range gui.modeStatuses() {
if mode.isActive() {
2020-10-04 11:00:48 +11:00
if width-cx > len(gui.Tr.ResetInParentheses) {
2020-08-28 09:52:56 +10:00
return nil
}
2020-08-23 09:23:59 +10:00
return mode.reset()
}
}
// if we're not in an active mode we show the donate button
2020-10-04 11:00:48 +11:00
if cx <= len(gui.Tr.Donate)+len(INFO_SECTION_PADDING) {
2020-03-29 10:31:34 +11:00
return gui.OSCommand.OpenLink("https://github.com/sponsors/jesseduffield")
}
return nil
}
2020-08-11 21:18:38 +10:00
func (gui *Gui) fetch(canPromptForCredentials bool) (err error) {
gui.Mutexes.FetchMutex.Lock()
defer gui.Mutexes.FetchMutex.Unlock()
2020-08-23 21:25:39 +10:00
2020-08-11 21:18:38 +10:00
fetchOpts := commands.FetchOptions{}
if canPromptForCredentials {
fetchOpts.PromptUserForCredential = gui.promptUserForCredential
}
err = gui.GitCommand.Fetch(fetchOpts)
2020-03-29 10:31:34 +11:00
2020-08-11 21:18:38 +10:00
if canPromptForCredentials && err != nil && strings.Contains(err.Error(), "exit status 128") {
2020-11-16 20:38:26 +11:00
_ = gui.createErrorPanel(gui.Tr.PassUnameWrong)
2020-03-29 10:31:34 +11:00
}
_ = gui.refreshSidePanels(refreshOptions{scope: []RefreshableView{BRANCHES, COMMITS, REMOTES, TAGS}, mode: ASYNC})
2020-03-29 10:31:34 +11:00
2020-08-11 20:18:50 +10:00
return err
2020-03-29 10:31:34 +11:00
}
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
}
if err := gui.OSCommand.CopyToClipboard(itemId); err != nil {
return gui.surfaceError(err)
}
2020-12-24 10:21:54 +01:00
truncatedItemId := utils.TruncateWithEllipsis(strings.Replace(itemId, "\n", " ", -1), 50)
gui.raiseToast(fmt.Sprintf("'%s' %s", truncatedItemId, gui.Tr.LcCopiedToClipboard))
return nil
}