2019-11-05 05:21:19 +02:00
|
|
|
package gui
|
|
|
|
|
|
|
|
import (
|
2020-08-15 02:58:29 +02:00
|
|
|
"fmt"
|
|
|
|
|
|
|
|
"github.com/go-errors/errors"
|
2020-08-15 03:18:40 +02:00
|
|
|
"github.com/jesseduffield/lazygit/pkg/commands/patch"
|
2021-04-18 08:30:34 +02:00
|
|
|
"github.com/jesseduffield/lazygit/pkg/gui/lbl"
|
2019-11-05 05:21:19 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
// Currently there are two 'pseudo-panels' that make use of this 'pseudo-panel'.
|
|
|
|
// One is the staging panel where we stage files line-by-line, the other is the
|
|
|
|
// patch building panel where we add lines of an old commit's file to a patch.
|
|
|
|
// This file contains the logic around selecting lines and displaying the diffs
|
|
|
|
// staging_panel.go and patch_building_panel.go have functions specific to their
|
|
|
|
// use cases
|
|
|
|
|
|
|
|
// returns whether the patch is empty so caller can escape if necessary
|
|
|
|
// both diffs should be non-coloured because we'll parse them and colour them here
|
2021-04-18 08:30:34 +02:00
|
|
|
func (gui *Gui) refreshLineByLinePanel(diff string, secondaryDiff string, secondaryFocused bool, selectedLineIdx int) (bool, error) {
|
2021-04-04 16:06:20 +02:00
|
|
|
gui.splitMainPanel(true)
|
|
|
|
|
2021-04-18 08:30:34 +02:00
|
|
|
var oldState *lbl.State
|
|
|
|
if gui.State.Panels.LineByLine != nil {
|
|
|
|
oldState = gui.State.Panels.LineByLine.State
|
2019-11-05 05:21:19 +02:00
|
|
|
}
|
|
|
|
|
2021-04-18 08:30:34 +02:00
|
|
|
state := lbl.NewState(diff, selectedLineIdx, oldState, gui.Log)
|
|
|
|
if state == nil {
|
2019-11-05 05:21:19 +02:00
|
|
|
return true, nil
|
|
|
|
}
|
|
|
|
|
2021-04-18 08:30:34 +02:00
|
|
|
gui.State.Panels.LineByLine = &LblPanelState{
|
|
|
|
State: state,
|
2019-11-05 05:21:19 +02:00
|
|
|
SecondaryFocused: secondaryFocused,
|
|
|
|
}
|
|
|
|
|
2021-04-18 08:30:34 +02:00
|
|
|
if err := gui.refreshMainViewForLineByLine(gui.State.Panels.LineByLine); err != nil {
|
2019-11-05 05:21:19 +02:00
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
|
2021-04-18 08:30:34 +02:00
|
|
|
if err := gui.focusSelection(gui.State.Panels.LineByLine); err != nil {
|
2019-11-05 05:21:19 +02:00
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
|
2021-04-04 15:51:59 +02:00
|
|
|
gui.Views.Secondary.Highlight = true
|
|
|
|
gui.Views.Secondary.Wrap = false
|
2019-11-05 05:21:19 +02:00
|
|
|
|
2021-04-18 08:30:34 +02:00
|
|
|
secondaryPatchParser := patch.NewPatchParser(gui.Log, secondaryDiff)
|
2019-11-05 05:21:19 +02:00
|
|
|
|
2021-04-11 02:05:19 +02:00
|
|
|
gui.setViewContent(gui.Views.Secondary, secondaryPatchParser.Render(-1, -1, nil))
|
2019-11-05 05:21:19 +02:00
|
|
|
|
|
|
|
return false, nil
|
|
|
|
}
|
|
|
|
|
2020-10-01 23:56:14 +02:00
|
|
|
func (gui *Gui) handleSelectPrevLine() error {
|
2021-04-18 08:30:34 +02:00
|
|
|
return gui.withLBLActiveCheck(func(state *LblPanelState) error {
|
|
|
|
state.CycleSelection(false)
|
|
|
|
|
|
|
|
return gui.refreshAndFocusLblPanel(state)
|
2020-10-01 23:56:14 +02:00
|
|
|
})
|
2019-11-05 05:21:19 +02:00
|
|
|
}
|
|
|
|
|
2020-10-01 23:56:14 +02:00
|
|
|
func (gui *Gui) handleSelectNextLine() error {
|
2021-04-18 08:30:34 +02:00
|
|
|
return gui.withLBLActiveCheck(func(state *LblPanelState) error {
|
|
|
|
state.CycleSelection(true)
|
|
|
|
|
|
|
|
return gui.refreshAndFocusLblPanel(state)
|
2020-10-01 23:56:14 +02:00
|
|
|
})
|
2019-11-05 05:21:19 +02:00
|
|
|
}
|
|
|
|
|
2020-10-01 23:56:14 +02:00
|
|
|
func (gui *Gui) handleSelectPrevHunk() error {
|
2021-04-18 08:30:34 +02:00
|
|
|
return gui.withLBLActiveCheck(func(state *LblPanelState) error {
|
|
|
|
state.CycleHunk(false)
|
2019-11-10 07:20:35 +02:00
|
|
|
|
2021-04-18 08:30:34 +02:00
|
|
|
return gui.refreshAndFocusLblPanel(state)
|
2020-10-01 23:56:14 +02:00
|
|
|
})
|
2019-11-05 05:21:19 +02:00
|
|
|
}
|
|
|
|
|
2020-10-01 23:56:14 +02:00
|
|
|
func (gui *Gui) handleSelectNextHunk() error {
|
2021-04-18 08:30:34 +02:00
|
|
|
return gui.withLBLActiveCheck(func(state *LblPanelState) error {
|
|
|
|
state.CycleHunk(true)
|
2019-11-10 07:20:35 +02:00
|
|
|
|
2021-04-18 08:30:34 +02:00
|
|
|
return gui.refreshAndFocusLblPanel(state)
|
2020-10-01 23:56:14 +02:00
|
|
|
})
|
2019-11-05 05:21:19 +02:00
|
|
|
}
|
|
|
|
|
2021-10-02 09:50:26 +02:00
|
|
|
func (gui *Gui) copySelectedToClipboard() error {
|
|
|
|
return gui.withLBLActiveCheck(func(state *LblPanelState) error {
|
2021-10-16 04:49:40 +02:00
|
|
|
selected := state.PlainRenderSelected()
|
2021-10-02 09:50:26 +02:00
|
|
|
|
2022-01-05 03:01:59 +02:00
|
|
|
gui.logAction(gui.Tr.Actions.CopySelectedTextToClipboard)
|
2022-01-05 02:57:32 +02:00
|
|
|
if err := gui.OSCommand.CopyToClipboard(selected); err != nil {
|
2021-10-02 09:50:26 +02:00
|
|
|
return gui.surfaceError(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2021-04-18 08:30:34 +02:00
|
|
|
func (gui *Gui) refreshAndFocusLblPanel(state *LblPanelState) error {
|
2020-10-07 23:01:04 +02:00
|
|
|
if err := gui.refreshMainViewForLineByLine(state); err != nil {
|
2019-11-05 05:21:19 +02:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2021-04-18 08:30:34 +02:00
|
|
|
return gui.focusSelection(state)
|
2019-11-05 05:21:19 +02:00
|
|
|
}
|
|
|
|
|
2021-04-02 10:20:40 +02:00
|
|
|
func (gui *Gui) handleLBLMouseDown() error {
|
2021-04-18 08:30:34 +02:00
|
|
|
return gui.withLBLActiveCheck(func(state *LblPanelState) error {
|
2020-10-01 23:56:14 +02:00
|
|
|
if gui.popupPanelFocused() {
|
|
|
|
return nil
|
|
|
|
}
|
2019-11-10 07:20:35 +02:00
|
|
|
|
2021-04-18 08:30:34 +02:00
|
|
|
state.SelectNewLineForRange(gui.Views.Main.SelectedLineIdx())
|
2019-11-10 07:20:35 +02:00
|
|
|
|
2021-04-18 08:30:34 +02:00
|
|
|
return gui.refreshAndFocusLblPanel(state)
|
2020-10-01 23:56:14 +02:00
|
|
|
})
|
2019-11-10 07:20:35 +02:00
|
|
|
}
|
|
|
|
|
2021-04-02 10:20:40 +02:00
|
|
|
func (gui *Gui) handleMouseDrag() error {
|
2021-04-18 08:30:34 +02:00
|
|
|
return gui.withLBLActiveCheck(func(state *LblPanelState) error {
|
2020-10-01 23:56:14 +02:00
|
|
|
if gui.popupPanelFocused() {
|
|
|
|
return nil
|
|
|
|
}
|
2019-11-10 07:20:35 +02:00
|
|
|
|
2021-04-18 08:30:34 +02:00
|
|
|
state.SelectLine(gui.Views.Main.SelectedLineIdx())
|
|
|
|
|
|
|
|
return gui.refreshAndFocusLblPanel(state)
|
2021-04-02 10:20:40 +02:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2020-02-29 09:44:08 +02:00
|
|
|
func (gui *Gui) getSelectedCommitFileName() string {
|
2021-03-31 13:08:55 +02:00
|
|
|
idx := gui.State.Panels.CommitFiles.SelectedLineIdx
|
|
|
|
|
2022-01-21 15:13:51 +02:00
|
|
|
return gui.State.CommitFileTreeViewModel.GetItemAtIndex(idx).GetPath()
|
2020-02-29 09:44:08 +02:00
|
|
|
}
|
|
|
|
|
2021-04-18 08:30:34 +02:00
|
|
|
func (gui *Gui) refreshMainViewForLineByLine(state *LblPanelState) error {
|
2019-11-05 05:21:19 +02:00
|
|
|
var includedLineIndices []int
|
|
|
|
// I'd prefer not to have knowledge of contexts using this file but I'm not sure
|
|
|
|
// how to get around this
|
2021-04-03 06:56:11 +02:00
|
|
|
if gui.currentContext().GetKey() == gui.State.Contexts.PatchBuilding.GetKey() {
|
2020-02-29 09:44:08 +02:00
|
|
|
filename := gui.getSelectedCommitFileName()
|
2020-08-22 08:46:19 +02:00
|
|
|
var err error
|
2022-01-08 05:10:01 +02:00
|
|
|
includedLineIndices, err = gui.Git.Patch.PatchManager.GetFileIncLineIndices(filename)
|
2020-08-22 08:46:19 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2019-11-05 05:21:19 +02:00
|
|
|
}
|
2021-04-18 08:30:34 +02:00
|
|
|
colorDiff := state.RenderForLineIndices(includedLineIndices)
|
2019-11-05 05:21:19 +02:00
|
|
|
|
2021-04-04 15:51:59 +02:00
|
|
|
gui.Views.Main.Highlight = true
|
|
|
|
gui.Views.Main.Wrap = false
|
2019-11-05 05:21:19 +02:00
|
|
|
|
2021-04-11 02:05:19 +02:00
|
|
|
gui.setViewContent(gui.Views.Main, colorDiff)
|
2019-11-05 05:21:19 +02:00
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// focusSelection works out the best focus for the staging panel given the
|
|
|
|
// selected line and size of the hunk
|
2021-04-18 08:30:34 +02:00
|
|
|
func (gui *Gui) focusSelection(state *LblPanelState) error {
|
2021-04-04 15:51:59 +02:00
|
|
|
stagingView := gui.Views.Main
|
2019-11-05 05:21:19 +02:00
|
|
|
|
|
|
|
_, viewHeight := stagingView.Size()
|
|
|
|
bufferHeight := viewHeight - 1
|
|
|
|
_, origin := stagingView.Origin()
|
|
|
|
|
2021-06-05 05:18:53 +02:00
|
|
|
selectedLineIdx := state.GetSelectedLineIdx()
|
2019-11-05 05:21:19 +02:00
|
|
|
|
2021-06-05 05:18:53 +02:00
|
|
|
newOrigin := state.CalculateOrigin(origin, bufferHeight)
|
2019-11-05 05:21:19 +02:00
|
|
|
|
2022-01-15 03:04:00 +02:00
|
|
|
if err := stagingView.SetOriginY(newOrigin); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2019-11-05 05:21:19 +02:00
|
|
|
|
2022-01-15 03:04:00 +02:00
|
|
|
return stagingView.SetCursor(0, selectedLineIdx-newOrigin)
|
2019-11-05 05:21:19 +02:00
|
|
|
}
|
|
|
|
|
2020-10-01 23:56:14 +02:00
|
|
|
func (gui *Gui) handleToggleSelectRange() error {
|
2021-04-18 08:30:34 +02:00
|
|
|
return gui.withLBLActiveCheck(func(state *LblPanelState) error {
|
|
|
|
state.ToggleSelectRange()
|
2019-11-05 05:21:19 +02:00
|
|
|
|
2020-10-07 23:01:04 +02:00
|
|
|
return gui.refreshMainViewForLineByLine(state)
|
2020-10-01 23:56:14 +02:00
|
|
|
})
|
2019-11-05 05:21:19 +02:00
|
|
|
}
|
|
|
|
|
2020-10-01 23:56:14 +02:00
|
|
|
func (gui *Gui) handleToggleSelectHunk() error {
|
2021-04-18 08:30:34 +02:00
|
|
|
return gui.withLBLActiveCheck(func(state *LblPanelState) error {
|
|
|
|
state.ToggleSelectHunk()
|
2019-11-05 05:21:19 +02:00
|
|
|
|
2021-04-18 08:30:34 +02:00
|
|
|
return gui.refreshAndFocusLblPanel(state)
|
2020-10-01 23:56:14 +02:00
|
|
|
})
|
2019-11-05 05:21:19 +02:00
|
|
|
}
|
2020-01-30 23:07:34 +02:00
|
|
|
|
2020-10-01 23:56:14 +02:00
|
|
|
func (gui *Gui) escapeLineByLinePanel() {
|
2020-10-07 23:01:04 +02:00
|
|
|
gui.State.Panels.LineByLine = nil
|
2020-01-30 23:07:34 +02:00
|
|
|
}
|
2020-08-15 02:58:29 +02:00
|
|
|
|
|
|
|
func (gui *Gui) handleOpenFileAtLine() error {
|
2021-04-18 08:30:34 +02:00
|
|
|
return gui.withLBLActiveCheck(func(state *LblPanelState) error {
|
2020-10-01 23:56:14 +02:00
|
|
|
// again, would be good to use inheritance here (or maybe even composition)
|
|
|
|
var filename string
|
|
|
|
switch gui.State.MainContext {
|
2021-04-03 06:56:11 +02:00
|
|
|
case gui.State.Contexts.PatchBuilding.GetKey():
|
2020-10-01 23:56:14 +02:00
|
|
|
filename = gui.getSelectedCommitFileName()
|
2021-04-03 06:56:11 +02:00
|
|
|
case gui.State.Contexts.Staging.GetKey():
|
2020-10-01 23:56:14 +02:00
|
|
|
file := gui.getSelectedFile()
|
|
|
|
if file == nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
filename = file.Name
|
|
|
|
default:
|
|
|
|
return errors.Errorf("unknown main context: %s", gui.State.MainContext)
|
2020-08-15 02:58:29 +02:00
|
|
|
}
|
|
|
|
|
2020-10-01 23:56:14 +02:00
|
|
|
// need to look at current index, then work out what my hunk's header information is, and see how far my line is away from the hunk header
|
2021-08-04 11:43:34 +02:00
|
|
|
lineNumber := state.CurrentLineNumber()
|
2020-10-01 23:56:14 +02:00
|
|
|
filenameWithLineNum := fmt.Sprintf("%s:%d", filename, lineNumber)
|
|
|
|
if err := gui.OSCommand.OpenFile(filenameWithLineNum); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func (gui *Gui) handleLineByLineNextPage() error {
|
2021-04-18 08:30:34 +02:00
|
|
|
return gui.withLBLActiveCheck(func(state *LblPanelState) error {
|
|
|
|
state.SetLineSelectMode()
|
|
|
|
state.AdjustSelectedLineIdx(gui.pageDelta(gui.Views.Main))
|
2020-10-01 23:56:14 +02:00
|
|
|
|
2021-04-18 08:30:34 +02:00
|
|
|
return gui.refreshAndFocusLblPanel(state)
|
2020-10-01 23:56:14 +02:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func (gui *Gui) handleLineByLinePrevPage() error {
|
2021-04-18 08:30:34 +02:00
|
|
|
return gui.withLBLActiveCheck(func(state *LblPanelState) error {
|
|
|
|
state.SetLineSelectMode()
|
|
|
|
state.AdjustSelectedLineIdx(-gui.pageDelta(gui.Views.Main))
|
2020-10-01 23:56:14 +02:00
|
|
|
|
2021-04-18 08:30:34 +02:00
|
|
|
return gui.refreshAndFocusLblPanel(state)
|
2020-10-01 23:56:14 +02:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func (gui *Gui) handleLineByLineGotoBottom() error {
|
2021-04-18 08:30:34 +02:00
|
|
|
return gui.withLBLActiveCheck(func(state *LblPanelState) error {
|
|
|
|
state.SelectBottom()
|
2020-10-01 23:56:14 +02:00
|
|
|
|
2021-04-18 08:30:34 +02:00
|
|
|
return gui.refreshAndFocusLblPanel(state)
|
2020-10-01 23:56:14 +02:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func (gui *Gui) handleLineByLineGotoTop() error {
|
2021-04-18 08:30:34 +02:00
|
|
|
return gui.withLBLActiveCheck(func(state *LblPanelState) error {
|
|
|
|
state.SelectTop()
|
2020-10-01 23:56:14 +02:00
|
|
|
|
2021-04-18 08:30:34 +02:00
|
|
|
return gui.refreshAndFocusLblPanel(state)
|
2020-10-01 23:56:14 +02:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2021-04-18 08:30:34 +02:00
|
|
|
func (gui *Gui) handlelineByLineNavigateTo(selectedLineIdx int) error {
|
|
|
|
return gui.withLBLActiveCheck(func(state *LblPanelState) error {
|
|
|
|
state.SetLineSelectMode()
|
|
|
|
state.SelectLine(selectedLineIdx)
|
2020-10-01 23:56:14 +02:00
|
|
|
|
2021-04-18 08:30:34 +02:00
|
|
|
return gui.refreshAndFocusLblPanel(state)
|
|
|
|
})
|
2020-10-01 23:56:14 +02:00
|
|
|
}
|
|
|
|
|
2021-04-18 08:30:34 +02:00
|
|
|
func (gui *Gui) withLBLActiveCheck(f func(*LblPanelState) error) error {
|
2020-10-07 12:45:57 +02:00
|
|
|
gui.Mutexes.LineByLinePanelMutex.Lock()
|
|
|
|
defer gui.Mutexes.LineByLinePanelMutex.Unlock()
|
2020-10-01 23:56:14 +02:00
|
|
|
|
|
|
|
state := gui.State.Panels.LineByLine
|
|
|
|
if state == nil {
|
|
|
|
return nil
|
2020-08-15 02:58:29 +02:00
|
|
|
}
|
|
|
|
|
2020-10-01 23:56:14 +02:00
|
|
|
return f(state)
|
2020-08-15 02:58:29 +02:00
|
|
|
}
|
2021-08-03 14:38:03 +02:00
|
|
|
|
|
|
|
func (gui *Gui) handleLineByLineEdit() error {
|
|
|
|
file := gui.getSelectedFile()
|
|
|
|
if file == nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2021-08-04 11:43:34 +02:00
|
|
|
lineNumber := gui.State.Panels.LineByLine.CurrentLineNumber()
|
2021-08-03 14:38:03 +02:00
|
|
|
return gui.editFileAtLine(file.Name, lineNumber)
|
|
|
|
}
|