2022-08-06 10:05:00 +02:00
|
|
|
package controllers
|
|
|
|
|
|
|
|
import (
|
2022-09-13 12:11:03 +02:00
|
|
|
"os"
|
2022-08-06 10:50:52 +02:00
|
|
|
|
|
|
|
"github.com/jesseduffield/gocui"
|
2022-08-06 10:05:00 +02:00
|
|
|
"github.com/jesseduffield/lazygit/pkg/gui/context"
|
2022-08-06 10:50:52 +02:00
|
|
|
"github.com/jesseduffield/lazygit/pkg/gui/mergeconflicts"
|
2022-08-06 10:05:00 +02:00
|
|
|
"github.com/jesseduffield/lazygit/pkg/gui/types"
|
|
|
|
)
|
|
|
|
|
|
|
|
type MergeConflictsController struct {
|
|
|
|
baseController
|
2023-03-23 09:47:29 +02:00
|
|
|
c *ControllerCommon
|
2022-08-06 10:05:00 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
var _ types.IController = &MergeConflictsController{}
|
|
|
|
|
|
|
|
func NewMergeConflictsController(
|
2023-03-23 09:47:29 +02:00
|
|
|
common *ControllerCommon,
|
2022-08-06 10:05:00 +02:00
|
|
|
) *MergeConflictsController {
|
|
|
|
return &MergeConflictsController{
|
2023-03-23 09:47:29 +02:00
|
|
|
baseController: baseController{},
|
|
|
|
c: common,
|
2022-08-06 10:05:00 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (self *MergeConflictsController) GetKeybindings(opts types.KeybindingsOpts) []*types.Binding {
|
|
|
|
bindings := []*types.Binding{
|
|
|
|
{
|
|
|
|
Key: opts.GetKey(opts.Config.Universal.Edit),
|
2022-08-06 10:50:52 +02:00
|
|
|
Handler: self.HandleEditFile,
|
2023-05-25 13:11:51 +02:00
|
|
|
Description: self.c.Tr.EditFile,
|
2022-08-06 10:05:00 +02:00
|
|
|
},
|
2022-08-06 10:50:52 +02:00
|
|
|
{
|
|
|
|
Key: opts.GetKey(opts.Config.Universal.OpenFile),
|
|
|
|
Handler: self.HandleOpenFile,
|
2023-05-25 13:11:51 +02:00
|
|
|
Description: self.c.Tr.OpenFile,
|
2022-08-06 10:50:52 +02:00
|
|
|
},
|
|
|
|
{
|
|
|
|
Key: opts.GetKey(opts.Config.Universal.PrevBlock),
|
|
|
|
Handler: self.withRenderAndFocus(self.PrevConflict),
|
|
|
|
Description: self.c.Tr.PrevConflict,
|
2023-03-21 11:57:52 +02:00
|
|
|
Display: true,
|
2022-08-06 10:50:52 +02:00
|
|
|
},
|
|
|
|
{
|
|
|
|
Key: opts.GetKey(opts.Config.Universal.NextBlock),
|
|
|
|
Handler: self.withRenderAndFocus(self.NextConflict),
|
|
|
|
Description: self.c.Tr.NextConflict,
|
2023-03-21 11:57:52 +02:00
|
|
|
Display: true,
|
2022-08-06 10:50:52 +02:00
|
|
|
},
|
|
|
|
{
|
|
|
|
Key: opts.GetKey(opts.Config.Universal.PrevItem),
|
|
|
|
Handler: self.withRenderAndFocus(self.PrevConflictHunk),
|
|
|
|
Description: self.c.Tr.SelectPrevHunk,
|
2023-03-21 11:57:52 +02:00
|
|
|
Display: true,
|
2022-08-06 10:50:52 +02:00
|
|
|
},
|
|
|
|
{
|
|
|
|
Key: opts.GetKey(opts.Config.Universal.NextItem),
|
|
|
|
Handler: self.withRenderAndFocus(self.NextConflictHunk),
|
|
|
|
Description: self.c.Tr.SelectNextHunk,
|
2023-03-21 11:57:52 +02:00
|
|
|
Display: true,
|
2022-08-06 10:50:52 +02:00
|
|
|
},
|
|
|
|
{
|
|
|
|
Key: opts.GetKey(opts.Config.Universal.PrevBlockAlt),
|
|
|
|
Handler: self.withRenderAndFocus(self.PrevConflict),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Key: opts.GetKey(opts.Config.Universal.NextBlockAlt),
|
|
|
|
Handler: self.withRenderAndFocus(self.NextConflict),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Key: opts.GetKey(opts.Config.Universal.PrevItemAlt),
|
|
|
|
Handler: self.withRenderAndFocus(self.PrevConflictHunk),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Key: opts.GetKey(opts.Config.Universal.NextItemAlt),
|
|
|
|
Handler: self.withRenderAndFocus(self.NextConflictHunk),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Key: opts.GetKey(opts.Config.Universal.ScrollLeft),
|
|
|
|
Handler: self.withRenderAndFocus(self.HandleScrollLeft),
|
2023-05-25 13:11:51 +02:00
|
|
|
Description: self.c.Tr.ScrollLeft,
|
2022-08-06 10:50:52 +02:00
|
|
|
Tag: "navigation",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Key: opts.GetKey(opts.Config.Universal.ScrollRight),
|
|
|
|
Handler: self.withRenderAndFocus(self.HandleScrollRight),
|
2023-05-25 13:11:51 +02:00
|
|
|
Description: self.c.Tr.ScrollRight,
|
2022-08-06 10:50:52 +02:00
|
|
|
Tag: "navigation",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Key: opts.GetKey(opts.Config.Universal.Undo),
|
|
|
|
Handler: self.withRenderAndFocus(self.HandleUndo),
|
2023-05-25 13:11:51 +02:00
|
|
|
Description: self.c.Tr.Undo,
|
2023-03-21 11:57:52 +02:00
|
|
|
Display: true,
|
2022-08-06 10:50:52 +02:00
|
|
|
},
|
|
|
|
{
|
|
|
|
Key: opts.GetKey(opts.Config.Files.OpenMergeTool),
|
2023-03-23 09:47:29 +02:00
|
|
|
Handler: self.c.Helpers().WorkingTree.OpenMergeTool,
|
2023-05-25 13:11:51 +02:00
|
|
|
Description: self.c.Tr.OpenMergeTool,
|
2022-08-06 10:50:52 +02:00
|
|
|
},
|
|
|
|
{
|
|
|
|
Key: opts.GetKey(opts.Config.Universal.Select),
|
|
|
|
Handler: self.withRenderAndFocus(self.HandlePickHunk),
|
|
|
|
Description: self.c.Tr.PickHunk,
|
2023-03-21 11:57:52 +02:00
|
|
|
Display: true,
|
2022-08-06 10:50:52 +02:00
|
|
|
},
|
|
|
|
{
|
|
|
|
Key: opts.GetKey(opts.Config.Main.PickBothHunks),
|
|
|
|
Handler: self.withRenderAndFocus(self.HandlePickAllHunks),
|
|
|
|
Description: self.c.Tr.PickAllHunks,
|
2023-03-21 11:57:52 +02:00
|
|
|
Display: true,
|
2022-08-06 10:50:52 +02:00
|
|
|
},
|
|
|
|
{
|
|
|
|
Key: opts.GetKey(opts.Config.Universal.Return),
|
|
|
|
Handler: self.Escape,
|
|
|
|
Description: self.c.Tr.ReturnToFilesPanel,
|
|
|
|
},
|
2022-08-06 10:05:00 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return bindings
|
|
|
|
}
|
|
|
|
|
2022-08-06 10:50:52 +02:00
|
|
|
func (self *MergeConflictsController) GetMouseKeybindings(opts types.KeybindingsOpts) []*gocui.ViewMouseBinding {
|
|
|
|
return []*gocui.ViewMouseBinding{
|
|
|
|
{
|
|
|
|
ViewName: self.context().GetViewName(),
|
|
|
|
Key: gocui.MouseWheelUp,
|
|
|
|
Handler: func(gocui.ViewMouseBindingOpts) error {
|
|
|
|
return self.HandleScrollUp()
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
ViewName: self.context().GetViewName(),
|
|
|
|
Key: gocui.MouseWheelDown,
|
|
|
|
Handler: func(gocui.ViewMouseBindingOpts) error {
|
|
|
|
return self.HandleScrollDown()
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-12-30 14:24:24 +02:00
|
|
|
func (self *MergeConflictsController) GetOnFocus() func(types.OnFocusOpts) error {
|
|
|
|
return func(types.OnFocusOpts) error {
|
|
|
|
self.c.Views().MergeConflicts.Wrap = false
|
|
|
|
|
2023-03-23 09:47:29 +02:00
|
|
|
return self.c.Helpers().MergeConflicts.Render(true)
|
2022-12-30 14:24:24 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (self *MergeConflictsController) GetOnFocusLost() func(types.OnFocusLostOpts) error {
|
|
|
|
return func(types.OnFocusLostOpts) error {
|
|
|
|
self.context().SetUserScrolling(false)
|
|
|
|
self.context().GetState().ResetConflictSelection()
|
|
|
|
self.c.Views().MergeConflicts.Wrap = true
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-08-06 10:50:52 +02:00
|
|
|
func (self *MergeConflictsController) HandleScrollUp() error {
|
|
|
|
self.context().SetUserScrolling(true)
|
|
|
|
self.context().GetViewTrait().ScrollUp(self.c.UserConfig.Gui.ScrollHeight)
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (self *MergeConflictsController) HandleScrollDown() error {
|
|
|
|
self.context().SetUserScrolling(true)
|
|
|
|
self.context().GetViewTrait().ScrollDown(self.c.UserConfig.Gui.ScrollHeight)
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2022-08-06 10:05:00 +02:00
|
|
|
func (self *MergeConflictsController) Context() types.Context {
|
|
|
|
return self.context()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (self *MergeConflictsController) context() *context.MergeConflictsContext {
|
2023-03-23 04:04:57 +02:00
|
|
|
return self.c.Contexts().MergeConflicts
|
2022-08-06 10:05:00 +02:00
|
|
|
}
|
|
|
|
|
2022-08-06 10:50:52 +02:00
|
|
|
func (self *MergeConflictsController) Escape() error {
|
2023-03-04 15:43:20 +02:00
|
|
|
return self.c.PopContext()
|
2022-08-06 10:50:52 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (self *MergeConflictsController) HandleEditFile() error {
|
|
|
|
lineNumber := self.context().GetState().GetSelectedLine()
|
2023-03-23 09:47:29 +02:00
|
|
|
return self.c.Helpers().Files.EditFileAtLine(self.context().GetState().GetPath(), lineNumber)
|
2022-08-06 10:50:52 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (self *MergeConflictsController) HandleOpenFile() error {
|
2023-03-23 09:47:29 +02:00
|
|
|
return self.c.Helpers().Files.OpenFile(self.context().GetState().GetPath())
|
2022-08-06 10:50:52 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (self *MergeConflictsController) HandleScrollLeft() error {
|
|
|
|
self.context().GetViewTrait().ScrollLeft()
|
|
|
|
|
|
|
|
return nil
|
2022-08-06 10:05:00 +02:00
|
|
|
}
|
|
|
|
|
2022-08-06 10:50:52 +02:00
|
|
|
func (self *MergeConflictsController) HandleScrollRight() error {
|
|
|
|
self.context().GetViewTrait().ScrollRight()
|
2022-08-06 10:05:00 +02:00
|
|
|
|
2022-08-06 10:50:52 +02:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (self *MergeConflictsController) HandleUndo() error {
|
|
|
|
state := self.context().GetState()
|
|
|
|
|
|
|
|
ok := state.Undo()
|
|
|
|
if !ok {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
self.c.LogAction("Restoring file to previous state")
|
2023-07-31 03:33:47 +02:00
|
|
|
self.c.LogCommand(self.c.Tr.Actions.LogHandleUndo, false)
|
2022-09-13 12:18:22 +02:00
|
|
|
if err := os.WriteFile(state.GetPath(), []byte(state.GetContent()), 0o644); err != nil {
|
2022-08-06 10:50:52 +02:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (self *MergeConflictsController) PrevConflictHunk() error {
|
|
|
|
self.context().SetUserScrolling(false)
|
|
|
|
self.context().GetState().SelectPrevConflictHunk()
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (self *MergeConflictsController) NextConflictHunk() error {
|
|
|
|
self.context().SetUserScrolling(false)
|
|
|
|
self.context().GetState().SelectNextConflictHunk()
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (self *MergeConflictsController) NextConflict() error {
|
|
|
|
self.context().SetUserScrolling(false)
|
|
|
|
self.context().GetState().SelectNextConflict()
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (self *MergeConflictsController) PrevConflict() error {
|
|
|
|
self.context().SetUserScrolling(false)
|
|
|
|
self.context().GetState().SelectPrevConflict()
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (self *MergeConflictsController) HandlePickHunk() error {
|
|
|
|
return self.pickSelection(self.context().GetState().Selection())
|
|
|
|
}
|
|
|
|
|
|
|
|
func (self *MergeConflictsController) HandlePickAllHunks() error {
|
|
|
|
return self.pickSelection(mergeconflicts.ALL)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (self *MergeConflictsController) pickSelection(selection mergeconflicts.Selection) error {
|
|
|
|
ok, err := self.resolveConflict(selection)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if !ok {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
if self.context().GetState().AllConflictsResolved() {
|
|
|
|
return self.onLastConflictResolved()
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (self *MergeConflictsController) resolveConflict(selection mergeconflicts.Selection) (bool, error) {
|
|
|
|
self.context().SetUserScrolling(false)
|
|
|
|
|
|
|
|
state := self.context().GetState()
|
|
|
|
|
|
|
|
ok, content, err := state.ContentAfterConflictResolve(selection)
|
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if !ok {
|
|
|
|
return false, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
var logStr string
|
|
|
|
switch selection {
|
|
|
|
case mergeconflicts.TOP:
|
|
|
|
logStr = "Picking top hunk"
|
|
|
|
case mergeconflicts.MIDDLE:
|
|
|
|
logStr = "Picking middle hunk"
|
|
|
|
case mergeconflicts.BOTTOM:
|
|
|
|
logStr = "Picking bottom hunk"
|
|
|
|
case mergeconflicts.ALL:
|
|
|
|
logStr = "Picking all hunks"
|
|
|
|
}
|
|
|
|
self.c.LogAction("Resolve merge conflict")
|
|
|
|
self.c.LogCommand(logStr, false)
|
|
|
|
state.PushContent(content)
|
2022-09-13 12:11:03 +02:00
|
|
|
return true, os.WriteFile(state.GetPath(), []byte(content), 0o644)
|
2022-08-06 10:50:52 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (self *MergeConflictsController) onLastConflictResolved() error {
|
|
|
|
// as part of refreshing files, we handle the situation where a file has had
|
|
|
|
// its merge conflicts resolved.
|
|
|
|
return self.c.Refresh(types.RefreshOptions{Mode: types.ASYNC, Scope: []types.RefreshableView{types.FILES}})
|
|
|
|
}
|
|
|
|
|
|
|
|
func (self *MergeConflictsController) isFocused() bool {
|
|
|
|
return self.c.CurrentContext().GetKey() == self.context().GetKey()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (self *MergeConflictsController) withRenderAndFocus(f func() error) func() error {
|
|
|
|
return self.withLock(func() error {
|
|
|
|
if err := f(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return self.context().RenderAndFocus(self.isFocused())
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func (self *MergeConflictsController) withLock(f func() error) func() error {
|
|
|
|
return func() error {
|
|
|
|
self.context().GetMutex().Lock()
|
|
|
|
defer self.context().GetMutex().Unlock()
|
|
|
|
|
|
|
|
if self.context().GetState() == nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return f()
|
|
|
|
}
|
2022-08-06 10:05:00 +02:00
|
|
|
}
|