2020-03-29 10:31:34 +11:00
|
|
|
package gui
|
|
|
|
|
|
|
|
import (
|
2022-06-13 11:01:26 +10:00
|
|
|
"github.com/jesseduffield/generics/slices"
|
2020-03-29 10:31:34 +11:00
|
|
|
"github.com/jesseduffield/gocui"
|
2022-06-13 11:01:26 +10:00
|
|
|
"github.com/jesseduffield/lazygit/pkg/gui/context"
|
2020-03-29 10:31:34 +11:00
|
|
|
"github.com/jesseduffield/lazygit/pkg/theme"
|
|
|
|
)
|
|
|
|
|
2020-05-18 22:21:36 +10:00
|
|
|
const SEARCH_PREFIX = "search: "
|
2020-05-18 22:00:07 +10:00
|
|
|
|
2020-03-29 10:31:34 +11:00
|
|
|
// layout is called for every screen re-render e.g. when the screen is resized
|
|
|
|
func (gui *Gui) layout(g *gocui.Gui) error {
|
2021-04-08 23:55:18 +10:00
|
|
|
if !gui.ViewsSetup {
|
2022-02-05 16:56:36 +11:00
|
|
|
gui.printCommandLogHeader()
|
|
|
|
|
|
|
|
if _, err := gui.g.SetCurrentView(gui.defaultSideContext().GetViewName()); err != nil {
|
2021-04-08 23:55:18 +10:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-03-29 10:31:34 +11:00
|
|
|
g.Highlight = true
|
|
|
|
width, height := g.Size()
|
|
|
|
|
2020-05-18 22:21:36 +10:00
|
|
|
informationStr := gui.informationStr()
|
2020-03-29 10:31:34 +11:00
|
|
|
appStatus := gui.statusManager.getStatusString()
|
2020-05-18 22:21:36 +10:00
|
|
|
|
2020-08-21 19:53:45 +10:00
|
|
|
viewDimensions := gui.getWindowDimensions(informationStr, appStatus)
|
2020-03-29 10:31:34 +11:00
|
|
|
|
|
|
|
// reading more lines into main view buffers upon resize
|
2021-04-05 00:31:52 +10:00
|
|
|
prevMainView := gui.Views.Main
|
|
|
|
if prevMainView != nil {
|
2020-03-29 10:31:34 +11:00
|
|
|
_, prevMainHeight := prevMainView.Size()
|
2020-08-15 08:28:02 +10:00
|
|
|
newMainHeight := viewDimensions["main"].Y1 - viewDimensions["main"].Y0 - 1
|
2020-05-16 12:35:19 +10:00
|
|
|
heightDiff := newMainHeight - prevMainHeight
|
2020-03-29 10:31:34 +11:00
|
|
|
if heightDiff > 0 {
|
|
|
|
if manager, ok := gui.viewBufferManagerMap["main"]; ok {
|
|
|
|
manager.ReadLines(heightDiff)
|
|
|
|
}
|
|
|
|
if manager, ok := gui.viewBufferManagerMap["secondary"]; ok {
|
|
|
|
manager.ReadLines(heightDiff)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-06-13 11:01:26 +10:00
|
|
|
// we assume that the view has already been created.
|
|
|
|
setViewFromDimensions := func(viewName string, windowName string) (*gocui.View, error) {
|
2020-08-17 20:45:44 +10:00
|
|
|
dimensionsObj, ok := viewDimensions[windowName]
|
|
|
|
|
2022-06-13 11:01:26 +10:00
|
|
|
view, err := g.View(viewName)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2020-08-17 20:45:44 +10:00
|
|
|
if !ok {
|
|
|
|
// view not specified in dimensions object: so create the view and hide it
|
2020-08-23 11:53:05 +10:00
|
|
|
// making the view take up the whole space in the background in case it needs
|
|
|
|
// to render content as soon as it appears, because lazyloaded content (via a pty task)
|
|
|
|
// cares about the size of the view.
|
2022-06-13 11:01:26 +10:00
|
|
|
_, err := g.SetView(viewName, 0, 0, width, height, 0)
|
|
|
|
view.Visible = false
|
2021-04-04 23:51:59 +10:00
|
|
|
return view, err
|
2020-08-17 20:45:44 +10:00
|
|
|
}
|
|
|
|
|
2020-05-18 22:21:36 +10:00
|
|
|
frameOffset := 1
|
2022-06-13 11:01:26 +10:00
|
|
|
if view.Frame {
|
2020-05-18 22:21:36 +10:00
|
|
|
frameOffset = 0
|
|
|
|
}
|
2022-06-13 11:01:26 +10:00
|
|
|
_, err = g.SetView(
|
2020-05-16 12:35:19 +10:00
|
|
|
viewName,
|
2020-08-15 08:28:02 +10:00
|
|
|
dimensionsObj.X0-frameOffset,
|
|
|
|
dimensionsObj.Y0-frameOffset,
|
|
|
|
dimensionsObj.X1+frameOffset,
|
|
|
|
dimensionsObj.Y1+frameOffset,
|
2020-05-16 12:35:19 +10:00
|
|
|
0,
|
|
|
|
)
|
2022-06-13 11:01:26 +10:00
|
|
|
view.Visible = true
|
2021-04-04 23:51:59 +10:00
|
|
|
|
|
|
|
return view, err
|
2020-05-16 12:35:19 +10:00
|
|
|
}
|
|
|
|
|
2022-06-13 11:01:26 +10:00
|
|
|
for _, context := range gui.State.Contexts.Flatten() {
|
|
|
|
if !context.HasControlledBounds() {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
_, err := setViewFromDimensions(context.GetViewName(), context.GetWindowName())
|
2023-02-10 21:23:48 +09:00
|
|
|
if err != nil && !gocui.IsUnknownView(err) {
|
2020-03-29 10:31:34 +11:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-05-08 11:41:13 +10:00
|
|
|
minimumHeight := 9
|
|
|
|
minimumWidth := 10
|
|
|
|
gui.Views.Limit.Visible = height < minimumHeight || width < minimumWidth
|
|
|
|
|
2022-05-08 12:46:48 +10:00
|
|
|
gui.Views.Tooltip.Visible = gui.Views.Menu.Visible && gui.Views.Tooltip.Buffer() != ""
|
|
|
|
|
2022-03-26 14:44:30 +11:00
|
|
|
for _, context := range gui.TransientContexts() {
|
2022-03-24 22:07:30 +11:00
|
|
|
view, err := gui.g.View(context.GetViewName())
|
2023-02-10 21:23:48 +09:00
|
|
|
if err != nil && !gocui.IsUnknownView(err) {
|
2022-03-24 22:07:30 +11:00
|
|
|
return err
|
|
|
|
}
|
2022-12-30 23:24:24 +11:00
|
|
|
view.Visible = gui.helpers.Window.GetViewNameForWindow(context.GetWindowName()) == context.GetViewName()
|
2022-03-24 22:07:30 +11:00
|
|
|
}
|
2020-03-29 10:31:34 +11:00
|
|
|
|
2022-01-28 20:44:36 +11:00
|
|
|
if gui.PrevLayout.Information != informationStr {
|
2022-12-30 23:24:24 +11:00
|
|
|
gui.c.SetViewContent(gui.Views.Information, informationStr)
|
2022-01-28 20:44:36 +11:00
|
|
|
gui.PrevLayout.Information = informationStr
|
2020-03-29 10:31:34 +11:00
|
|
|
}
|
|
|
|
|
2021-04-03 15:56:11 +11:00
|
|
|
if !gui.ViewsSetup {
|
2021-04-03 13:43:43 +11:00
|
|
|
if err := gui.onInitialViewsCreation(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2021-04-03 15:56:11 +11:00
|
|
|
gui.ViewsSetup = true
|
2021-04-03 13:43:43 +11:00
|
|
|
}
|
|
|
|
|
2021-04-03 15:56:11 +11:00
|
|
|
if !gui.State.ViewsSetup {
|
|
|
|
if err := gui.onInitialViewsCreationForRepo(); err != nil {
|
2020-03-29 10:31:34 +11:00
|
|
|
return err
|
|
|
|
}
|
2021-04-03 15:56:11 +11:00
|
|
|
|
|
|
|
gui.State.ViewsSetup = true
|
2020-03-29 10:31:34 +11:00
|
|
|
}
|
|
|
|
|
2021-04-03 11:32:14 +11:00
|
|
|
for _, listContext := range gui.getListContexts() {
|
2021-11-01 09:35:54 +11:00
|
|
|
view, err := gui.g.View(listContext.GetViewName())
|
2021-04-03 11:32:14 +11:00
|
|
|
if err != nil {
|
|
|
|
continue
|
|
|
|
}
|
2020-03-29 10:31:34 +11:00
|
|
|
|
2021-11-01 09:35:54 +11:00
|
|
|
listContext.FocusLine()
|
2020-05-13 21:10:00 +10:00
|
|
|
|
2021-04-03 11:32:14 +11:00
|
|
|
view.SelBgColor = theme.GocuiSelectedLineBgColor
|
2020-08-16 09:18:57 +10:00
|
|
|
|
|
|
|
// I doubt this is expensive though it's admittedly redundant after the first render
|
2022-01-16 14:46:53 +11:00
|
|
|
view.SetOnSelectItem(gui.onSelectItemWrapper(listContext.OnSearchSelect))
|
2020-03-29 10:31:34 +11:00
|
|
|
}
|
|
|
|
|
2022-06-13 11:01:26 +10:00
|
|
|
for _, context := range gui.getPatchExplorerContexts() {
|
|
|
|
context := context
|
|
|
|
context.GetView().SetOnSelectItem(gui.onSelectItemWrapper(
|
|
|
|
func(selectedLineIdx int) error {
|
|
|
|
context.GetMutex().Lock()
|
|
|
|
defer context.GetMutex().Unlock()
|
|
|
|
return context.NavigateTo(gui.c.IsCurrentContext(context), selectedLineIdx)
|
|
|
|
}),
|
|
|
|
)
|
|
|
|
}
|
2020-10-02 07:56:14 +10:00
|
|
|
|
2021-04-04 23:51:59 +10:00
|
|
|
mainViewWidth, mainViewHeight := gui.Views.Main.Size()
|
2022-01-28 20:44:36 +11:00
|
|
|
if mainViewWidth != gui.PrevLayout.MainWidth || mainViewHeight != gui.PrevLayout.MainHeight {
|
|
|
|
gui.PrevLayout.MainWidth = mainViewWidth
|
|
|
|
gui.PrevLayout.MainHeight = mainViewHeight
|
2020-03-29 10:31:34 +11:00
|
|
|
if err := gui.onResize(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// here is a good place log some stuff
|
2020-09-26 10:23:10 +10:00
|
|
|
// if you run `lazygit --logs`
|
2020-03-29 10:31:34 +11:00
|
|
|
// this will let you see these branches as prettified json
|
2022-01-31 22:11:34 +11:00
|
|
|
// gui.c.Log.Info(utils.AsJson(gui.State.Model.Branches[0:4]))
|
2020-08-15 17:23:16 +10:00
|
|
|
return gui.resizeCurrentPopupPanel()
|
2020-03-29 10:31:34 +11:00
|
|
|
}
|
|
|
|
|
2021-04-08 23:55:18 +10:00
|
|
|
func (gui *Gui) prepareView(viewName string) (*gocui.View, error) {
|
2021-04-04 23:51:59 +10:00
|
|
|
// arbitrarily giving the view enough size so that we don't get an error, but
|
|
|
|
// it's expected that the view will be given the correct size before being shown
|
|
|
|
return gui.g.SetView(viewName, 0, 0, 10, 10, 0)
|
|
|
|
}
|
|
|
|
|
2021-04-03 15:56:11 +11:00
|
|
|
func (gui *Gui) onInitialViewsCreationForRepo() error {
|
2021-04-03 13:43:43 +11:00
|
|
|
// hide any popup views. This only applies when we've just switched repos
|
|
|
|
for _, viewName := range gui.popupViewNames() {
|
2021-04-04 23:51:59 +10:00
|
|
|
view, err := gui.g.View(viewName)
|
|
|
|
if err == nil {
|
|
|
|
view.Visible = false
|
2021-04-03 15:56:11 +11:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-12-30 23:24:24 +11:00
|
|
|
initialContext := gui.c.CurrentSideContext()
|
2022-01-16 14:46:53 +11:00
|
|
|
if err := gui.c.PushContext(initialContext); err != nil {
|
2021-04-03 15:56:11 +11:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return gui.loadNewRepo()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (gui *Gui) onInitialViewsCreation() error {
|
2021-04-04 23:51:59 +10:00
|
|
|
// now we order the views (in order of bottom first)
|
2022-05-08 11:41:13 +10:00
|
|
|
for _, view := range gui.orderedViews() {
|
2021-04-04 23:51:59 +10:00
|
|
|
if _, err := gui.g.SetViewOnTop(view.Name()); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-01 07:03:03 +10:00
|
|
|
gui.g.Mutexes.ViewsMutex.Lock()
|
2021-04-03 13:43:43 +11:00
|
|
|
// add tabs to views
|
2020-09-30 08:27:12 +10:00
|
|
|
for _, view := range gui.g.Views() {
|
2022-06-13 11:01:26 +10:00
|
|
|
// if the view is in our mapping, we'll set the tabs and the tab index
|
|
|
|
for _, values := range gui.viewTabMap() {
|
|
|
|
index := slices.IndexFunc(values, func(tabContext context.TabView) bool {
|
|
|
|
return tabContext.ViewName == view.Name()
|
|
|
|
})
|
|
|
|
|
|
|
|
if index != -1 {
|
|
|
|
view.Tabs = slices.Map(values, func(tabContext context.TabView) string {
|
|
|
|
return tabContext.Tab
|
|
|
|
})
|
|
|
|
view.TabIndex = index
|
|
|
|
}
|
2020-09-30 08:27:12 +10:00
|
|
|
}
|
|
|
|
}
|
2020-10-01 07:03:03 +10:00
|
|
|
gui.g.Mutexes.ViewsMutex.Unlock()
|
2020-09-30 08:27:12 +10:00
|
|
|
|
2022-01-16 14:46:53 +11:00
|
|
|
if !gui.c.UserConfig.DisableStartupPopups {
|
2021-04-07 22:43:19 +10:00
|
|
|
popupTasks := []func(chan struct{}) error{}
|
2022-01-16 14:46:53 +11:00
|
|
|
storedPopupVersion := gui.c.GetAppState().StartupPopupVersion
|
2021-04-07 22:43:19 +10:00
|
|
|
if storedPopupVersion < StartupPopupVersion {
|
|
|
|
popupTasks = append(popupTasks, gui.showIntroPopupMessage)
|
|
|
|
}
|
|
|
|
gui.showInitialPopups(popupTasks)
|
|
|
|
}
|
|
|
|
|
2020-08-16 22:49:37 +10:00
|
|
|
if gui.showRecentRepos {
|
2022-12-30 23:24:24 +11:00
|
|
|
if err := gui.helpers.Repos.CreateRecentReposMenu(); err != nil {
|
2020-08-16 22:49:37 +10:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
gui.showRecentRepos = false
|
|
|
|
}
|
|
|
|
|
2022-12-30 23:24:24 +11:00
|
|
|
gui.helpers.Update.CheckForUpdateInBackground()
|
2021-04-03 15:56:11 +11:00
|
|
|
|
|
|
|
gui.waitForIntro.Done()
|
|
|
|
|
|
|
|
return nil
|
2020-03-29 10:31:34 +11:00
|
|
|
}
|
2022-12-30 23:24:24 +11:00
|
|
|
|
|
|
|
// getFocusLayout returns a manager function for when view gain and lose focus
|
|
|
|
func (gui *Gui) getFocusLayout() func(g *gocui.Gui) error {
|
|
|
|
var previousView *gocui.View
|
|
|
|
return func(g *gocui.Gui) error {
|
|
|
|
newView := gui.g.CurrentView()
|
|
|
|
// for now we don't consider losing focus to a popup panel as actually losing focus
|
|
|
|
if newView != previousView && !gui.isPopupPanel(newView.Name()) {
|
|
|
|
if err := gui.onViewFocusLost(previousView); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
previousView = newView
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (gui *Gui) onViewFocusLost(oldView *gocui.View) error {
|
|
|
|
if oldView == nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
oldView.Highlight = false
|
|
|
|
|
|
|
|
_ = oldView.SetOriginX(0)
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|