summaryrefslogtreecommitdiffstats
path: root/pkg/gui/staging_panel.go
blob: a24becbd24cdc51c1b8e484930f19f1f98892269 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
package gui

import (
	"github.com/jesseduffield/gocui"
	"github.com/jesseduffield/lazygit/pkg/git"
	"github.com/jesseduffield/lazygit/pkg/utils"
)

func (gui *Gui) refreshStagingPanel() error {
	file, err := gui.getSelectedFile(gui.g)
	if err != nil {
		if err != gui.Errors.ErrNoFiles {
			return err
		}
		return gui.handleStagingEscape(gui.g, nil)
	}

	if !file.HasUnstagedChanges {
		return gui.handleStagingEscape(gui.g, nil)
	}

	// note for custom diffs, we'll need to send a flag here saying not to use the custom diff
	diff := gui.GitCommand.Diff(file, true)
	colorDiff := gui.GitCommand.Diff(file, false)

	if len(diff) < 2 {
		return gui.handleStagingEscape(gui.g, nil)
	}

	// parse the diff and store the line numbers of hunks and stageable lines
	// TODO: maybe instantiate this at application start
	p, err := git.NewPatchParser(gui.Log)
	if err != nil {
		return nil
	}
	hunkStarts, stageableLines, err := p.ParsePatch(diff)
	if err != nil {
		return nil
	}

	var selectedLine int
	if gui.State.Panels.Staging != nil {
		end := len(stageableLines) - 1
		if end < gui.State.Panels.Staging.SelectedLine {
			selectedLine = end
		} else {
			selectedLine = gui.State.Panels.Staging.SelectedLine
		}
	} else {
		selectedLine = 0
	}

	gui.State.Panels.Staging = &stagingPanelState{
		StageableLines: stageableLines,
		HunkStarts:     hunkStarts,
		SelectedLine:   selectedLine,
		Diff:           diff,
	}

	if len(stageableLines) == 0 {
		return gui.createErrorPanel(gui.g, "No lines to stage")
	}

	if err := gui.focusLineAndHunk(); err != nil {
		return err
	}

	mainView := gui.getMainView()
	mainView.Highlight = true
	mainView.Wrap = false

	gui.g.Update(func(*gocui.Gui) error {
		return gui.setViewContent(gui.g, gui.getMainView(), colorDiff)
	})

	return nil
}

func (gui *Gui) handleStagingEscape(g *gocui.Gui, v *gocui.View) error {
	gui.State.Panels.Staging = nil

	return gui.switchFocus(gui.g, nil, gui.getFilesView())
}

func (gui *Gui) handleStagingPrevLine(g *gocui.Gui, v *gocui.View) error {
	return gui.handleCycleLine(true)
}

func (gui *Gui) handleStagingNextLine(g *gocui.Gui, v *gocui.View) error {
	return gui.handleCycleLine(false)
}

func (gui *Gui) handleStagingPrevHunk(g *gocui.Gui, v *gocui.View) error {
	return gui.handleCycleHunk(true)
}

func (gui *Gui) handleStagingNextHunk(g *gocui.Gui, v *gocui.View) error {
	return gui.handleCycleHunk(false)
}

func (gui *Gui) handleCycleHunk(prev bool) error {
	state := gui.State.Panels.Staging
	lineNumbers := state.StageableLines
	currentLine := lineNumbers[state.SelectedLine]
	currentHunkIndex := utils.PrevIndex(state.HunkStarts, currentLine)
	var newHunkIndex int
	if prev {
		if currentHunkIndex == 0 {
			newHunkIndex = len(state.HunkStarts) - 1
		} else {
			newHunkIndex = currentHunkIndex - 1
		}
	} else {
		if currentHunkIndex == len(state.HunkStarts)-1 {
			newHunkIndex = 0
		} else {
			newHunkIndex = currentHunkIndex + 1
		}
	}

	state.SelectedLine = utils.NextIndex(lineNumbers, state.HunkStarts[newHunkIndex])

	return gui.focusLineAndHunk()
}

func (gui *Gui) handleCycleLine(prev bool) error {
	state := gui.State.Panels.Staging
	lineNumbers := state.StageableLines
	currentLine := lineNumbers[state.SelectedLine]
	var newIndex int
	if prev {
		newIndex = utils.PrevIndex(lineNumbers, currentLine)
	} else {
		newIndex = utils.NextIndex(lineNumbers, currentLine)
	}
	state.SelectedLine = newIndex

	return gui.focusLineAndHunk()
}

// focusLineAndHunk works out the best focus for the staging panel given the
// selected line and size of the hunk
func (gui *Gui) focusLineAndHunk() error {
	stagingView := gui.getMainView()
	state := gui.State.Panels.Staging

	lineNumber := state.StageableLines[state.SelectedLine]

	// we want the bottom line of the view buffer to ideally be the bottom line
	// of the hunk, but if the hunk is too big we'll just go three lines beyond
	// the currently selected line so that the user can see the context
	var bottomLine int
	nextHunkStartIndex := utils.NextIndex(state.HunkStarts, lineNumber)
	if nextHunkStartIndex == 0 {
		// for now linesHeight is an efficient means of getting the number of lines
		// in the patch. However if we introduce word wrap we'll need to update this
		bottomLine = stagingView.LinesHeight() - 1
	} else {
		bottomLine = state.HunkStarts[nextHunkStartIndex] - 1
	}

	hunkStartIndex := utils.PrevIndex(state.HunkStarts, lineNumber)
	hunkStart := state.HunkStarts[hunkStartIndex]
	// if it's the first hunk we'll also show the diff header
	if hunkStartIndex == 0 {
		hunkStart = 0
	}

	_, height := stagingView.Size()
	// if this hunk is too big, we will just ensure that the user can at least
	// see three lines of context below the cursor
	if bottomLine-hunkStart > height {
		bottomLine = lineNumber + 3
	}

	return gui.generalFocusLine(lineNumber, bottomLine, stagingView)
}

func (gui *Gui) handleStageHunk(g *gocui.Gui, v *gocui.View) error {
	return gui.handleStageLineOrHunk(true)
}

func (gui *Gui) handleStageLine(g *gocui.Gui, v *gocui.View) error {
	return gui.handleStageLineOrHunk(false)
}

func (gui *Gui) handleStageLineOrHunk(hunk bool) error {
	state := gui.State.Panels.Staging
	p, err := git.NewPatchModifier(gui.Log)
	if err != nil {
		return err
	}

	currentLine := state.StageableLines[state.SelectedLine]
	var patch string
	if hunk {
		patch, err = p.ModifyPatchForHunk(state.Diff, state.HunkStarts, currentLine)
	} else {
		patch, err = p.ModifyPatchForLine(state.Diff, currentLine)
	}
	if err != nil {
		return err
	}

	// for logging purposes
	// ioutil.WriteFile("patch.diff", []byte(patch), 0600)

	// apply the patch then refresh this panel
	// create a new temp file with the patch, then call git apply with that patch
	_, err = gui.GitCommand.ApplyPatch(patch)
	if err != nil {
		return err
	}

	if err := gui.refreshFiles(); err != nil {
		return err
	}
	if err := gui.refreshStagingPanel(); err != nil {
		return err
	}
	return nil
}