1
0
mirror of https://github.com/jesseduffield/lazygit.git synced 2024-12-12 11:15:00 +02:00
lazygit/pkg/gui/line_by_line_panel.go

293 lines
7.8 KiB
Go
Raw Normal View History

package gui
import (
"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"
)
// 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
}
2021-04-18 08:30:34 +02:00
state := lbl.NewState(diff, selectedLineIdx, oldState, gui.Log)
if state == nil {
return true, nil
}
2021-04-18 08:30:34 +02:00
gui.State.Panels.LineByLine = &LblPanelState{
State: state,
SecondaryFocused: secondaryFocused,
}
2021-04-18 08:30:34 +02:00
if err := gui.refreshMainViewForLineByLine(gui.State.Panels.LineByLine); err != nil {
return false, err
}
2021-04-18 08:30:34 +02:00
if err := gui.focusSelection(gui.State.Panels.LineByLine); err != nil {
return false, err
}
2021-04-04 15:51:59 +02:00
gui.Views.Secondary.Highlight = true
gui.Views.Secondary.Wrap = false
2021-04-18 08:30:34 +02:00
secondaryPatchParser := patch.NewPatchParser(gui.Log, secondaryDiff)
2021-04-11 02:05:19 +02:00
gui.setViewContent(gui.Views.Secondary, secondaryPatchParser.Render(-1, -1, nil))
return false, nil
}
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)
})
}
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)
})
}
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)
})
}
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)
})
}
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 {
return err
}
2021-04-18 08:30:34 +02:00
return gui.focusSelection(state)
}
func (gui *Gui) handleLBLMouseDown() error {
2021-04-18 08:30:34 +02:00
return gui.withLBLActiveCheck(func(state *LblPanelState) error {
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)
})
2019-11-10 07:20:35 +02:00
}
func (gui *Gui) handleMouseDrag() error {
2021-04-18 08:30:34 +02:00
return gui.withLBLActiveCheck(func(state *LblPanelState) error {
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)
})
}
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()
}
2021-04-18 08:30:34 +02:00
func (gui *Gui) refreshMainViewForLineByLine(state *LblPanelState) error {
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() {
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
}
}
2021-04-18 08:30:34 +02:00
colorDiff := state.RenderForLineIndices(includedLineIndices)
2021-04-04 15:51:59 +02:00
gui.Views.Main.Highlight = true
gui.Views.Main.Wrap = false
2021-04-11 02:05:19 +02:00
gui.setViewContent(gui.Views.Main, colorDiff)
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
_, viewHeight := stagingView.Size()
bufferHeight := viewHeight - 1
_, origin := stagingView.Origin()
2021-06-05 05:18:53 +02:00
selectedLineIdx := state.GetSelectedLineIdx()
2021-06-05 05:18:53 +02:00
newOrigin := state.CalculateOrigin(origin, bufferHeight)
2022-01-15 03:04:00 +02:00
if err := stagingView.SetOriginY(newOrigin); err != nil {
return err
}
2022-01-15 03:04:00 +02:00
return stagingView.SetCursor(0, selectedLineIdx-newOrigin)
}
func (gui *Gui) handleToggleSelectRange() error {
2021-04-18 08:30:34 +02:00
return gui.withLBLActiveCheck(func(state *LblPanelState) error {
state.ToggleSelectRange()
2020-10-07 23:01:04 +02:00
return gui.refreshMainViewForLineByLine(state)
})
}
func (gui *Gui) handleToggleSelectHunk() error {
2021-04-18 08:30:34 +02:00
return gui.withLBLActiveCheck(func(state *LblPanelState) error {
state.ToggleSelectHunk()
2021-04-18 08:30:34 +02:00
return gui.refreshAndFocusLblPanel(state)
})
}
func (gui *Gui) escapeLineByLinePanel() {
2020-10-07 23:01:04 +02:00
gui.State.Panels.LineByLine = nil
}
func (gui *Gui) handleOpenFileAtLine() error {
2021-04-18 08:30:34 +02:00
return gui.withLBLActiveCheck(func(state *LblPanelState) error {
// 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():
filename = gui.getSelectedCommitFileName()
2021-04-03 06:56:11 +02:00
case gui.State.Contexts.Staging.GetKey():
file := gui.getSelectedFile()
if file == nil {
return nil
}
filename = file.Name
default:
return errors.Errorf("unknown main context: %s", gui.State.MainContext)
}
// 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()
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))
2021-04-18 08:30:34 +02:00
return gui.refreshAndFocusLblPanel(state)
})
}
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))
2021-04-18 08:30:34 +02:00
return gui.refreshAndFocusLblPanel(state)
})
}
func (gui *Gui) handleLineByLineGotoBottom() error {
2021-04-18 08:30:34 +02:00
return gui.withLBLActiveCheck(func(state *LblPanelState) error {
state.SelectBottom()
2021-04-18 08:30:34 +02:00
return gui.refreshAndFocusLblPanel(state)
})
}
func (gui *Gui) handleLineByLineGotoTop() error {
2021-04-18 08:30:34 +02:00
return gui.withLBLActiveCheck(func(state *LblPanelState) error {
state.SelectTop()
2021-04-18 08:30:34 +02:00
return gui.refreshAndFocusLblPanel(state)
})
}
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)
2021-04-18 08:30:34 +02:00
return gui.refreshAndFocusLblPanel(state)
})
}
2021-04-18 08:30:34 +02:00
func (gui *Gui) withLBLActiveCheck(f func(*LblPanelState) error) error {
gui.Mutexes.LineByLinePanelMutex.Lock()
defer gui.Mutexes.LineByLinePanelMutex.Unlock()
state := gui.State.Panels.LineByLine
if state == nil {
return nil
}
return f(state)
}
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()
return gui.editFileAtLine(file.Name, lineNumber)
}