2020-05-17 21:32:17 +10:00
|
|
|
package gui
|
|
|
|
|
2020-05-18 22:21:36 +10:00
|
|
|
import (
|
2022-10-09 08:31:14 -07:00
|
|
|
"github.com/jesseduffield/lazycore/pkg/boxlayout"
|
2022-01-29 19:15:46 +11:00
|
|
|
"github.com/jesseduffield/lazygit/pkg/gui/context"
|
2022-01-16 14:46:53 +11:00
|
|
|
"github.com/jesseduffield/lazygit/pkg/gui/types"
|
2020-05-18 22:21:36 +10:00
|
|
|
"github.com/jesseduffield/lazygit/pkg/utils"
|
2022-05-07 17:10:38 +09:00
|
|
|
"github.com/mattn/go-runewidth"
|
2020-05-18 22:21:36 +10:00
|
|
|
)
|
|
|
|
|
2022-05-07 15:42:36 +10:00
|
|
|
// In this file we use the boxlayout package, along with knowledge about the app's state,
|
|
|
|
// to arrange the windows (i.e. panels) on the screen.
|
|
|
|
|
2021-04-11 21:52:04 +10:00
|
|
|
const INFO_SECTION_PADDING = " "
|
|
|
|
|
2022-05-08 11:41:13 +10:00
|
|
|
func (gui *Gui) getWindowDimensions(informationStr string, appStatus string) map[string]boxlayout.Dimensions {
|
|
|
|
width, height := gui.g.Size()
|
|
|
|
|
|
|
|
sideSectionWeight, mainSectionWeight := gui.getMidSectionWeights()
|
|
|
|
|
|
|
|
sidePanelsDirection := boxlayout.COLUMN
|
|
|
|
portraitMode := width <= 84 && height > 45
|
|
|
|
if portraitMode {
|
|
|
|
sidePanelsDirection = boxlayout.ROW
|
|
|
|
}
|
|
|
|
|
|
|
|
mainPanelsDirection := boxlayout.ROW
|
|
|
|
if gui.splitMainPanelSideBySide() {
|
|
|
|
mainPanelsDirection = boxlayout.COLUMN
|
|
|
|
}
|
|
|
|
|
|
|
|
extrasWindowSize := gui.getExtrasWindowSize(height)
|
|
|
|
|
2022-11-14 00:53:55 +03:00
|
|
|
showInfoSection := gui.c.UserConfig.Gui.ShowBottomLine || gui.State.Searching.isSearching || gui.isAnyModeActive() || gui.statusManager.showStatus()
|
2022-05-08 11:41:13 +10:00
|
|
|
infoSectionSize := 0
|
|
|
|
if showInfoSection {
|
|
|
|
infoSectionSize = 1
|
|
|
|
}
|
|
|
|
|
|
|
|
root := &boxlayout.Box{
|
|
|
|
Direction: boxlayout.ROW,
|
|
|
|
Children: []*boxlayout.Box{
|
|
|
|
{
|
|
|
|
Direction: sidePanelsDirection,
|
|
|
|
Weight: 1,
|
|
|
|
Children: []*boxlayout.Box{
|
|
|
|
{
|
|
|
|
Direction: boxlayout.ROW,
|
|
|
|
Weight: sideSectionWeight,
|
|
|
|
ConditionalChildren: gui.sidePanelChildren,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Direction: boxlayout.ROW,
|
|
|
|
Weight: mainSectionWeight,
|
|
|
|
Children: []*boxlayout.Box{
|
|
|
|
{
|
|
|
|
Direction: mainPanelsDirection,
|
|
|
|
Children: gui.mainSectionChildren(),
|
|
|
|
Weight: 1,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Window: "extras",
|
|
|
|
Size: extrasWindowSize,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Direction: boxlayout.COLUMN,
|
|
|
|
Size: infoSectionSize,
|
|
|
|
Children: gui.infoSectionChildren(informationStr, appStatus),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
layerOneWindows := boxlayout.ArrangeWindows(root, 0, 0, width, height)
|
|
|
|
limitWindows := boxlayout.ArrangeWindows(&boxlayout.Box{Window: "limit"}, 0, 0, width, height)
|
|
|
|
|
|
|
|
return MergeMaps(layerOneWindows, limitWindows)
|
|
|
|
}
|
|
|
|
|
|
|
|
func MergeMaps[K comparable, V any](maps ...map[K]V) map[K]V {
|
|
|
|
result := map[K]V{}
|
|
|
|
for _, currMap := range maps {
|
|
|
|
for key, value := range currMap {
|
|
|
|
result[key] = value
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return result
|
|
|
|
}
|
|
|
|
|
2020-08-15 08:28:02 +10:00
|
|
|
func (gui *Gui) mainSectionChildren() []*boxlayout.Box {
|
2020-08-21 19:53:45 +10:00
|
|
|
currentWindow := gui.currentWindow()
|
2020-05-18 22:00:07 +10:00
|
|
|
|
|
|
|
// if we're not in split mode we can just show the one main panel. Likewise if
|
|
|
|
// the main panel is focused and we're in full-screen mode
|
2020-08-21 19:53:45 +10:00
|
|
|
if !gui.isMainPanelSplit() || (gui.State.ScreenMode == SCREEN_FULL && currentWindow == "main") {
|
2020-08-15 08:28:02 +10:00
|
|
|
return []*boxlayout.Box{
|
2020-05-18 22:00:07 +10:00
|
|
|
{
|
2020-08-21 19:53:45 +10:00
|
|
|
Window: "main",
|
|
|
|
Weight: 1,
|
2020-05-18 22:00:07 +10:00
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
2020-05-17 21:32:17 +10:00
|
|
|
|
2020-08-15 08:28:02 +10:00
|
|
|
return []*boxlayout.Box{
|
2020-05-17 21:32:17 +10:00
|
|
|
{
|
2022-06-13 11:01:26 +10:00
|
|
|
Window: "main",
|
2020-08-21 19:53:45 +10:00
|
|
|
Weight: 1,
|
2020-05-17 21:32:17 +10:00
|
|
|
},
|
2020-05-18 22:00:07 +10:00
|
|
|
{
|
2022-06-13 11:01:26 +10:00
|
|
|
Window: "secondary",
|
2020-08-21 19:53:45 +10:00
|
|
|
Weight: 1,
|
2020-05-18 22:00:07 +10:00
|
|
|
},
|
2020-05-17 21:32:17 +10:00
|
|
|
}
|
2020-05-18 22:00:07 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
func (gui *Gui) getMidSectionWeights() (int, int) {
|
2020-08-21 19:53:45 +10:00
|
|
|
currentWindow := gui.currentWindow()
|
2020-05-17 21:32:17 +10:00
|
|
|
|
2020-05-17 21:44:59 +10:00
|
|
|
// we originally specified this as a ratio i.e. .20 would correspond to a weight of 1 against 4
|
2022-01-16 14:46:53 +11:00
|
|
|
sidePanelWidthRatio := gui.c.UserConfig.Gui.SidePanelWidth
|
2020-05-17 21:44:59 +10:00
|
|
|
// we could make this better by creating ratios like 2:3 rather than always 1:something
|
|
|
|
mainSectionWeight := int(1/sidePanelWidthRatio) - 1
|
|
|
|
sideSectionWeight := 1
|
|
|
|
|
2020-10-13 07:25:56 +11:00
|
|
|
if gui.splitMainPanelSideBySide() {
|
2020-05-17 21:44:59 +10:00
|
|
|
mainSectionWeight = 5 // need to shrink side panel to make way for main panels if side-by-side
|
|
|
|
}
|
2020-05-18 22:00:07 +10:00
|
|
|
|
2020-08-21 19:53:45 +10:00
|
|
|
if currentWindow == "main" {
|
2020-05-17 21:44:59 +10:00
|
|
|
if gui.State.ScreenMode == SCREEN_HALF || gui.State.ScreenMode == SCREEN_FULL {
|
|
|
|
sideSectionWeight = 0
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if gui.State.ScreenMode == SCREEN_HALF {
|
|
|
|
mainSectionWeight = 1
|
|
|
|
} else if gui.State.ScreenMode == SCREEN_FULL {
|
|
|
|
mainSectionWeight = 0
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-18 22:00:07 +10:00
|
|
|
return sideSectionWeight, mainSectionWeight
|
|
|
|
}
|
|
|
|
|
2020-08-15 08:28:02 +10:00
|
|
|
func (gui *Gui) infoSectionChildren(informationStr string, appStatus string) []*boxlayout.Box {
|
2020-05-18 22:21:36 +10:00
|
|
|
if gui.State.Searching.isSearching {
|
2020-08-15 08:28:02 +10:00
|
|
|
return []*boxlayout.Box{
|
2020-05-18 22:21:36 +10:00
|
|
|
{
|
2020-08-21 19:53:45 +10:00
|
|
|
Window: "searchPrefix",
|
2022-05-07 17:10:38 +09:00
|
|
|
Size: runewidth.StringWidth(SEARCH_PREFIX),
|
2020-05-18 22:21:36 +10:00
|
|
|
},
|
|
|
|
{
|
2020-08-21 19:53:45 +10:00
|
|
|
Window: "search",
|
|
|
|
Weight: 1,
|
2020-05-18 22:21:36 +10:00
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-11-14 01:04:56 +03:00
|
|
|
appStatusBox := &boxlayout.Box{Window: "appStatus"}
|
|
|
|
optionsBox := &boxlayout.Box{Window: "options"}
|
2020-05-18 22:21:36 +10:00
|
|
|
|
2022-11-14 01:04:56 +03:00
|
|
|
if !gui.c.UserConfig.Gui.ShowBottomLine {
|
|
|
|
optionsBox.Weight = 0
|
|
|
|
appStatusBox.Weight = 1
|
|
|
|
} else {
|
|
|
|
optionsBox.Weight = 1
|
|
|
|
appStatusBox.Size = runewidth.StringWidth(INFO_SECTION_PADDING) + runewidth.StringWidth(appStatus)
|
2020-05-18 22:21:36 +10:00
|
|
|
}
|
|
|
|
|
2022-11-14 01:04:56 +03:00
|
|
|
result := []*boxlayout.Box{appStatusBox, optionsBox}
|
|
|
|
|
|
|
|
if gui.c.UserConfig.Gui.ShowBottomLine || gui.isAnyModeActive() {
|
|
|
|
result = append(result, &boxlayout.Box{
|
|
|
|
Window: "information",
|
|
|
|
// unlike appStatus, informationStr has various colors so we need to decolorise before taking the length
|
|
|
|
Size: runewidth.StringWidth(INFO_SECTION_PADDING) + runewidth.StringWidth(utils.Decolorise(informationStr)),
|
|
|
|
})
|
|
|
|
}
|
2020-05-18 22:21:36 +10:00
|
|
|
|
|
|
|
return result
|
|
|
|
}
|
|
|
|
|
2020-10-13 07:25:56 +11:00
|
|
|
func (gui *Gui) splitMainPanelSideBySide() bool {
|
|
|
|
if !gui.isMainPanelSplit() {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2022-01-16 14:46:53 +11:00
|
|
|
mainPanelSplitMode := gui.c.UserConfig.Gui.MainPanelSplitMode
|
2020-10-13 07:25:56 +11:00
|
|
|
width, height := gui.g.Size()
|
|
|
|
|
|
|
|
switch mainPanelSplitMode {
|
|
|
|
case "vertical":
|
|
|
|
return false
|
|
|
|
case "horizontal":
|
|
|
|
return true
|
|
|
|
default:
|
|
|
|
if width < 200 && height > 30 { // 2 80 character width panels + 40 width for side panel
|
|
|
|
return false
|
|
|
|
} else {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-04-19 18:04:00 +10:00
|
|
|
func (gui *Gui) getExtrasWindowSize(screenHeight int) int {
|
|
|
|
if !gui.ShowExtrasWindow {
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
|
|
|
|
var baseSize int
|
2022-01-29 19:15:46 +11:00
|
|
|
if gui.currentStaticContext().GetKey() == context.COMMAND_LOG_CONTEXT_KEY {
|
2021-04-19 18:04:00 +10:00
|
|
|
baseSize = 1000 // my way of saying 'fill the available space'
|
|
|
|
} else if screenHeight < 40 {
|
|
|
|
baseSize = 1
|
|
|
|
} else {
|
2022-01-16 14:46:53 +11:00
|
|
|
baseSize = gui.c.UserConfig.Gui.CommandLogSize
|
2021-04-19 18:04:00 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
frameSize := 2
|
|
|
|
return baseSize + frameSize
|
|
|
|
}
|
|
|
|
|
2020-08-23 15:01:02 +10:00
|
|
|
// The stash window by default only contains one line so that it's not hogging
|
|
|
|
// too much space, but if you access it it should take up some space. This is
|
2021-09-01 22:51:24 +02:00
|
|
|
// the default behaviour when accordion mode is NOT in effect. If it is in effect
|
2020-08-23 15:01:02 +10:00
|
|
|
// then when it's accessed it will have weight 2, not 1.
|
|
|
|
func (gui *Gui) getDefaultStashWindowBox() *boxlayout.Box {
|
2021-04-06 16:01:07 +10:00
|
|
|
gui.State.ContextManager.RLock()
|
|
|
|
defer gui.State.ContextManager.RUnlock()
|
2021-04-03 13:43:43 +11:00
|
|
|
|
2020-08-23 15:01:02 +10:00
|
|
|
box := &boxlayout.Box{Window: "stash"}
|
|
|
|
stashWindowAccessed := false
|
2021-04-03 13:43:43 +11:00
|
|
|
for _, context := range gui.State.ContextManager.ContextStack {
|
2020-08-23 15:01:02 +10:00
|
|
|
if context.GetWindowName() == "stash" {
|
|
|
|
stashWindowAccessed = true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// if the stash window is anywhere in our stack we should enlargen it
|
|
|
|
if stashWindowAccessed {
|
|
|
|
box.Weight = 1
|
|
|
|
} else {
|
|
|
|
box.Size = 3
|
|
|
|
}
|
|
|
|
|
|
|
|
return box
|
|
|
|
}
|
|
|
|
|
2020-08-15 08:28:02 +10:00
|
|
|
func (gui *Gui) sidePanelChildren(width int, height int) []*boxlayout.Box {
|
2020-08-21 19:53:45 +10:00
|
|
|
currentWindow := gui.currentSideWindowName()
|
2020-05-17 21:32:17 +10:00
|
|
|
|
|
|
|
if gui.State.ScreenMode == SCREEN_FULL || gui.State.ScreenMode == SCREEN_HALF {
|
2020-08-21 19:53:45 +10:00
|
|
|
fullHeightBox := func(window string) *boxlayout.Box {
|
|
|
|
if window == currentWindow {
|
2020-08-15 08:28:02 +10:00
|
|
|
return &boxlayout.Box{
|
2020-08-21 19:53:45 +10:00
|
|
|
Window: window,
|
|
|
|
Weight: 1,
|
2020-05-17 21:32:17 +10:00
|
|
|
}
|
|
|
|
} else {
|
2020-08-15 08:28:02 +10:00
|
|
|
return &boxlayout.Box{
|
2020-08-21 19:53:45 +10:00
|
|
|
Window: window,
|
|
|
|
Size: 0,
|
2020-05-17 21:32:17 +10:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-15 08:28:02 +10:00
|
|
|
return []*boxlayout.Box{
|
2020-05-17 21:32:17 +10:00
|
|
|
fullHeightBox("status"),
|
|
|
|
fullHeightBox("files"),
|
|
|
|
fullHeightBox("branches"),
|
|
|
|
fullHeightBox("commits"),
|
|
|
|
fullHeightBox("stash"),
|
|
|
|
}
|
2020-05-17 21:44:59 +10:00
|
|
|
} else if height >= 28 {
|
2022-01-16 14:46:53 +11:00
|
|
|
accordionMode := gui.c.UserConfig.Gui.ExpandFocusedSidePanel
|
2021-09-01 22:51:24 +02:00
|
|
|
accordionBox := func(defaultBox *boxlayout.Box) *boxlayout.Box {
|
|
|
|
if accordionMode && defaultBox.Window == currentWindow {
|
2020-08-15 08:28:02 +10:00
|
|
|
return &boxlayout.Box{
|
2020-08-21 19:53:45 +10:00
|
|
|
Window: defaultBox.Window,
|
|
|
|
Weight: 2,
|
2020-08-12 22:06:37 +10:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return defaultBox
|
|
|
|
}
|
|
|
|
|
2020-08-15 08:28:02 +10:00
|
|
|
return []*boxlayout.Box{
|
2020-05-17 21:32:17 +10:00
|
|
|
{
|
2020-08-21 19:53:45 +10:00
|
|
|
Window: "status",
|
|
|
|
Size: 3,
|
2020-05-17 21:32:17 +10:00
|
|
|
},
|
2021-09-01 22:51:24 +02:00
|
|
|
accordionBox(&boxlayout.Box{Window: "files", Weight: 1}),
|
|
|
|
accordionBox(&boxlayout.Box{Window: "branches", Weight: 1}),
|
|
|
|
accordionBox(&boxlayout.Box{Window: "commits", Weight: 1}),
|
|
|
|
accordionBox(gui.getDefaultStashWindowBox()),
|
2020-05-17 21:32:17 +10:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
squashedHeight := 1
|
|
|
|
if height >= 21 {
|
|
|
|
squashedHeight = 3
|
|
|
|
}
|
|
|
|
|
2020-08-21 19:53:45 +10:00
|
|
|
squashedSidePanelBox := func(window string) *boxlayout.Box {
|
|
|
|
if window == currentWindow {
|
2020-08-15 08:28:02 +10:00
|
|
|
return &boxlayout.Box{
|
2020-08-21 19:53:45 +10:00
|
|
|
Window: window,
|
|
|
|
Weight: 1,
|
2020-05-17 21:32:17 +10:00
|
|
|
}
|
|
|
|
} else {
|
2020-08-15 08:28:02 +10:00
|
|
|
return &boxlayout.Box{
|
2020-08-21 19:53:45 +10:00
|
|
|
Window: window,
|
|
|
|
Size: squashedHeight,
|
2020-05-17 21:32:17 +10:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-15 08:28:02 +10:00
|
|
|
return []*boxlayout.Box{
|
2020-05-17 21:32:17 +10:00
|
|
|
squashedSidePanelBox("status"),
|
|
|
|
squashedSidePanelBox("files"),
|
|
|
|
squashedSidePanelBox("branches"),
|
|
|
|
squashedSidePanelBox("commits"),
|
|
|
|
squashedSidePanelBox("stash"),
|
|
|
|
}
|
|
|
|
}
|
2020-05-17 21:44:59 +10:00
|
|
|
}
|
2020-05-17 21:32:17 +10:00
|
|
|
|
2022-02-13 10:48:41 +11:00
|
|
|
func (gui *Gui) getCyclableWindows() []string {
|
|
|
|
return []string{"status", "files", "branches", "commits", "stash"}
|
|
|
|
}
|
|
|
|
|
2020-08-21 19:53:45 +10:00
|
|
|
func (gui *Gui) currentSideWindowName() string {
|
|
|
|
// there is always one and only one cyclable context in the context stack. We'll look from top to bottom
|
2021-04-06 16:01:07 +10:00
|
|
|
gui.State.ContextManager.RLock()
|
|
|
|
defer gui.State.ContextManager.RUnlock()
|
2021-04-03 13:43:43 +11:00
|
|
|
|
|
|
|
for idx := range gui.State.ContextManager.ContextStack {
|
|
|
|
reversedIdx := len(gui.State.ContextManager.ContextStack) - 1 - idx
|
|
|
|
context := gui.State.ContextManager.ContextStack[reversedIdx]
|
2020-08-16 13:58:29 +10:00
|
|
|
|
2022-01-16 14:46:53 +11:00
|
|
|
if context.GetKind() == types.SIDE_CONTEXT {
|
2020-08-21 19:53:45 +10:00
|
|
|
return context.GetWindowName()
|
2020-08-16 13:58:29 +10:00
|
|
|
}
|
2020-05-17 21:32:17 +10:00
|
|
|
}
|
|
|
|
|
2020-08-16 13:58:29 +10:00
|
|
|
return "files" // default
|
2020-05-17 21:32:17 +10:00
|
|
|
}
|