mirror of
https://github.com/jesseduffield/lazygit.git
synced 2025-01-06 03:53:59 +02:00
266 lines
6.6 KiB
Go
266 lines
6.6 KiB
Go
package gui
|
|
|
|
import (
|
|
"github.com/jesseduffield/lazygit/pkg/gui/boxlayout"
|
|
"github.com/jesseduffield/lazygit/pkg/utils"
|
|
)
|
|
|
|
func (gui *Gui) mainSectionChildren() []*boxlayout.Box {
|
|
currentViewName := gui.currentViewName()
|
|
|
|
// 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
|
|
if !gui.isMainPanelSplit() || (gui.State.ScreenMode == SCREEN_FULL && currentViewName == "main") {
|
|
return []*boxlayout.Box{
|
|
{
|
|
ViewName: "main",
|
|
Weight: 1,
|
|
},
|
|
}
|
|
}
|
|
|
|
main := "main"
|
|
secondary := "secondary"
|
|
if gui.secondaryViewFocused() {
|
|
// when you think you've focused the secondary view, we've actually just swapped them around in the layout
|
|
main, secondary = secondary, main
|
|
}
|
|
|
|
return []*boxlayout.Box{
|
|
{
|
|
ViewName: main,
|
|
Weight: 1,
|
|
},
|
|
{
|
|
ViewName: secondary,
|
|
Weight: 1,
|
|
},
|
|
}
|
|
}
|
|
|
|
func (gui *Gui) getMidSectionWeights() (int, int) {
|
|
currentViewName := gui.currentViewName()
|
|
|
|
// we originally specified this as a ratio i.e. .20 would correspond to a weight of 1 against 4
|
|
sidePanelWidthRatio := gui.Config.GetUserConfig().GetFloat64("gui.sidePanelWidth")
|
|
// we could make this better by creating ratios like 2:3 rather than always 1:something
|
|
mainSectionWeight := int(1/sidePanelWidthRatio) - 1
|
|
sideSectionWeight := 1
|
|
|
|
if gui.isMainPanelSplit() {
|
|
mainSectionWeight = 5 // need to shrink side panel to make way for main panels if side-by-side
|
|
}
|
|
|
|
if currentViewName == "main" {
|
|
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
|
|
}
|
|
}
|
|
|
|
return sideSectionWeight, mainSectionWeight
|
|
}
|
|
|
|
func (gui *Gui) infoSectionChildren(informationStr string, appStatus string) []*boxlayout.Box {
|
|
if gui.State.Searching.isSearching {
|
|
return []*boxlayout.Box{
|
|
{
|
|
ViewName: "searchPrefix",
|
|
Size: len(SEARCH_PREFIX),
|
|
},
|
|
{
|
|
ViewName: "search",
|
|
Weight: 1,
|
|
},
|
|
}
|
|
}
|
|
|
|
result := []*boxlayout.Box{}
|
|
|
|
if len(appStatus) > 0 {
|
|
result = append(result,
|
|
&boxlayout.Box{
|
|
ViewName: "appStatus",
|
|
Size: len(appStatus) + len(INFO_SECTION_PADDING),
|
|
},
|
|
)
|
|
}
|
|
|
|
result = append(result,
|
|
[]*boxlayout.Box{
|
|
{
|
|
ViewName: "options",
|
|
Weight: 1,
|
|
},
|
|
{
|
|
ViewName: "information",
|
|
// unlike appStatus, informationStr has various colors so we need to decolorise before taking the length
|
|
Size: len(INFO_SECTION_PADDING) + len(utils.Decolorise(informationStr)),
|
|
},
|
|
}...,
|
|
)
|
|
|
|
return result
|
|
}
|
|
|
|
func (gui *Gui) getViewDimensions(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
|
|
}
|
|
|
|
root := &boxlayout.Box{
|
|
Direction: boxlayout.ROW,
|
|
Children: []*boxlayout.Box{
|
|
{
|
|
Direction: sidePanelsDirection,
|
|
Weight: 1,
|
|
Children: []*boxlayout.Box{
|
|
{
|
|
Direction: boxlayout.ROW,
|
|
Weight: sideSectionWeight,
|
|
ConditionalChildren: gui.sidePanelChildren,
|
|
},
|
|
{
|
|
ConditionalDirection: func(width int, height int) int {
|
|
mainPanelSplitMode := gui.Config.GetUserConfig().GetString("gui.mainPanelSplitMode")
|
|
|
|
switch mainPanelSplitMode {
|
|
case "vertical":
|
|
return boxlayout.ROW
|
|
case "horizontal":
|
|
return boxlayout.COLUMN
|
|
default:
|
|
if width < 160 && height > 30 { // 2 80 character width panels
|
|
return boxlayout.ROW
|
|
} else {
|
|
return boxlayout.COLUMN
|
|
}
|
|
}
|
|
},
|
|
Direction: boxlayout.COLUMN,
|
|
Weight: mainSectionWeight,
|
|
Children: gui.mainSectionChildren(),
|
|
},
|
|
},
|
|
},
|
|
{
|
|
Direction: boxlayout.COLUMN,
|
|
Size: 1,
|
|
Children: gui.infoSectionChildren(informationStr, appStatus),
|
|
},
|
|
},
|
|
}
|
|
|
|
return boxlayout.ArrangeViews(root, 0, 0, width, height)
|
|
}
|
|
|
|
func (gui *Gui) sidePanelChildren(width int, height int) []*boxlayout.Box {
|
|
currentCyclableViewName := gui.currentCyclableViewName()
|
|
|
|
if gui.State.ScreenMode == SCREEN_FULL || gui.State.ScreenMode == SCREEN_HALF {
|
|
fullHeightBox := func(viewName string) *boxlayout.Box {
|
|
if viewName == currentCyclableViewName {
|
|
return &boxlayout.Box{
|
|
ViewName: viewName,
|
|
Weight: 1,
|
|
}
|
|
} else {
|
|
return &boxlayout.Box{
|
|
ViewName: viewName,
|
|
Size: 0,
|
|
}
|
|
}
|
|
}
|
|
|
|
return []*boxlayout.Box{
|
|
fullHeightBox("status"),
|
|
fullHeightBox("files"),
|
|
fullHeightBox("branches"),
|
|
fullHeightBox("commits"),
|
|
fullHeightBox("stash"),
|
|
}
|
|
} else if height >= 28 {
|
|
accordianMode := gui.Config.GetUserConfig().GetBool("gui.expandFocusedSidePanel")
|
|
accordianBox := func(defaultBox *boxlayout.Box) *boxlayout.Box {
|
|
if accordianMode && defaultBox.ViewName == currentCyclableViewName {
|
|
return &boxlayout.Box{
|
|
ViewName: defaultBox.ViewName,
|
|
Weight: 2,
|
|
}
|
|
}
|
|
|
|
return defaultBox
|
|
}
|
|
|
|
return []*boxlayout.Box{
|
|
{
|
|
ViewName: "status",
|
|
Size: 3,
|
|
},
|
|
accordianBox(&boxlayout.Box{ViewName: "files", Weight: 1}),
|
|
accordianBox(&boxlayout.Box{ViewName: "branches", Weight: 1}),
|
|
accordianBox(&boxlayout.Box{ViewName: "commits", Weight: 1}),
|
|
accordianBox(&boxlayout.Box{ViewName: "stash", Size: 3}),
|
|
}
|
|
} else {
|
|
squashedHeight := 1
|
|
if height >= 21 {
|
|
squashedHeight = 3
|
|
}
|
|
|
|
squashedSidePanelBox := func(viewName string) *boxlayout.Box {
|
|
if viewName == currentCyclableViewName {
|
|
return &boxlayout.Box{
|
|
ViewName: viewName,
|
|
Weight: 1,
|
|
}
|
|
} else {
|
|
return &boxlayout.Box{
|
|
ViewName: viewName,
|
|
Size: squashedHeight,
|
|
}
|
|
}
|
|
}
|
|
|
|
return []*boxlayout.Box{
|
|
squashedSidePanelBox("status"),
|
|
squashedSidePanelBox("files"),
|
|
squashedSidePanelBox("branches"),
|
|
squashedSidePanelBox("commits"),
|
|
squashedSidePanelBox("stash"),
|
|
}
|
|
}
|
|
}
|
|
|
|
func (gui *Gui) currentCyclableViewName() string {
|
|
// there is always a cyclable context in the context stack. We'll look from top to bottom
|
|
for idx := range gui.State.ContextStack {
|
|
reversedIdx := len(gui.State.ContextStack) - 1 - idx
|
|
context := gui.State.ContextStack[reversedIdx]
|
|
|
|
if context.GetKind() == SIDE_CONTEXT {
|
|
viewName := context.GetViewName()
|
|
|
|
// unfortunate result of the fact that these are separate views, have to map explicitly
|
|
if viewName == "commitFiles" {
|
|
return "commits"
|
|
}
|
|
|
|
return viewName
|
|
}
|
|
}
|
|
|
|
return "files" // default
|
|
}
|