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

492 lines
13 KiB
Go
Raw Normal View History

2019-02-16 12:01:17 +02:00
package gui
2020-08-16 02:05:45 +02:00
import (
"errors"
"fmt"
2022-02-13 01:48:41 +02:00
"sort"
"strings"
2020-08-16 05:58:29 +02:00
"github.com/jesseduffield/gocui"
2022-01-29 10:15:46 +02:00
"github.com/jesseduffield/lazygit/pkg/gui/context"
"github.com/jesseduffield/lazygit/pkg/gui/types"
2020-08-16 02:05:45 +02:00
)
func (gui *Gui) popupViewNames() []string {
result := []string{}
2022-02-05 07:56:36 +02:00
for _, context := range gui.State.Contexts.Flatten() {
if context.GetKind() == types.PERSISTENT_POPUP || context.GetKind() == types.TEMPORARY_POPUP {
result = append(result, context.GetViewName())
}
}
return result
}
func (gui *Gui) currentContextKeyIgnoringPopups() types.ContextKey {
gui.State.ContextManager.RLock()
defer gui.State.ContextManager.RUnlock()
stack := gui.State.ContextManager.ContextStack
2020-08-23 03:23:32 +02:00
for i := range stack {
reversedIndex := len(stack) - 1 - i
context := stack[reversedIndex]
kind := stack[reversedIndex].GetKind()
if kind != types.TEMPORARY_POPUP && kind != types.PERSISTENT_POPUP {
2020-08-23 03:23:32 +02:00
return context.GetKey()
}
}
return ""
}
// use replaceContext when you don't want to return to the original context upon
// hitting escape: you want to go that context's parent instead.
func (gui *Gui) replaceContext(c types.Context) error {
2022-01-15 03:04:00 +02:00
gui.State.ContextManager.Lock()
defer gui.State.ContextManager.Unlock()
2022-02-05 07:56:36 +02:00
if !c.IsFocusable() {
return nil
}
2022-01-15 03:04:00 +02:00
if len(gui.State.ContextManager.ContextStack) == 0 {
gui.State.ContextManager.ContextStack = []types.Context{c}
2022-01-15 03:04:00 +02:00
} else {
// replace the last item with the given item
gui.State.ContextManager.ContextStack = append(gui.State.ContextManager.ContextStack[0:len(gui.State.ContextManager.ContextStack)-1], c)
}
2022-01-15 03:04:00 +02:00
return gui.activateContext(c)
}
func (gui *Gui) pushContext(c types.Context, opts ...types.OnFocusOpts) error {
// using triple dot but you should only ever pass one of these opt structs
if len(opts) > 1 {
return errors.New("cannot pass multiple opts to pushContext")
}
2022-02-05 07:56:36 +02:00
if !c.IsFocusable() {
return nil
2022-02-05 05:42:56 +02:00
}
gui.State.ContextManager.Lock()
// push onto stack
// if we are switching to a side context, remove all other contexts in the stack
if c.GetKind() == types.SIDE_CONTEXT {
for _, stackContext := range gui.State.ContextManager.ContextStack {
if stackContext.GetKey() != c.GetKey() {
if err := gui.deactivateContext(stackContext); err != nil {
gui.State.ContextManager.Unlock()
return err
}
}
}
gui.State.ContextManager.ContextStack = []types.Context{c}
} else if len(gui.State.ContextManager.ContextStack) == 0 || gui.currentContextWithoutLock().GetKey() != c.GetKey() {
// Do not append if the one at the end is the same context (e.g. opening a menu from a menu)
// In that case we'll just close the menu entirely when the user hits escape.
// TODO: think about other exceptional cases
gui.State.ContextManager.ContextStack = append(gui.State.ContextManager.ContextStack, c)
}
gui.State.ContextManager.Unlock()
return gui.activateContext(c, opts...)
}
// asynchronous code idea: functions return an error via a channel, when done
// pushContextWithView is to be used when you don't know which context you
2020-08-16 05:58:29 +02:00
// want to switch to: you only know the view that you want to switch to. It will
// look up the context currently active for that view and switch to that context
func (gui *Gui) pushContextWithView(viewName string) error {
2022-02-05 07:56:36 +02:00
return gui.c.PushContext(gui.State.ViewContextMap.Get(viewName))
2020-08-16 05:58:29 +02:00
}
2020-08-16 10:17:16 +02:00
func (gui *Gui) returnFromContext() error {
2021-09-25 05:21:28 +02:00
gui.State.ContextManager.Lock()
2020-08-16 10:17:16 +02:00
2021-09-25 05:21:28 +02:00
if len(gui.State.ContextManager.ContextStack) == 1 {
// cannot escape from bottommost context
gui.State.ContextManager.Unlock()
return nil
}
2020-08-16 10:17:16 +02:00
2021-09-25 05:21:28 +02:00
n := len(gui.State.ContextManager.ContextStack) - 1
2020-08-16 10:17:16 +02:00
2021-09-25 05:21:28 +02:00
currentContext := gui.State.ContextManager.ContextStack[n]
newContext := gui.State.ContextManager.ContextStack[n-1]
2020-08-16 10:17:16 +02:00
2021-09-25 05:21:28 +02:00
gui.State.ContextManager.ContextStack = gui.State.ContextManager.ContextStack[:n]
2022-02-05 05:42:56 +02:00
gui.g.SetCurrentContext(string(newContext.GetKey()))
2021-09-25 05:21:28 +02:00
gui.State.ContextManager.Unlock()
2020-08-16 10:17:16 +02:00
2021-09-25 05:21:28 +02:00
if err := gui.deactivateContext(currentContext); err != nil {
return err
}
2021-09-25 05:21:28 +02:00
return gui.activateContext(newContext)
2020-08-16 05:58:29 +02:00
}
func (gui *Gui) deactivateContext(c types.Context) error {
2021-04-06 02:26:18 +02:00
view, _ := gui.g.View(c.GetViewName())
if view != nil && view.IsSearching() {
if err := gui.onSearchEscape(); err != nil {
return err
2021-04-04 15:51:59 +02:00
}
2020-08-17 13:58:30 +02:00
}
2021-04-06 02:26:18 +02:00
// if we are the kind of context that is sent to back upon deactivation, we should do that
2022-01-29 10:15:46 +02:00
if view != nil && (c.GetKind() == types.TEMPORARY_POPUP || c.GetKind() == types.PERSISTENT_POPUP || c.GetKey() == context.COMMIT_FILES_CONTEXT_KEY) {
2021-04-06 02:26:18 +02:00
view.Visible = false
}
2020-08-17 13:58:30 +02:00
if err := c.HandleFocusLost(); err != nil {
return err
}
return nil
}
2020-08-19 11:07:14 +02:00
// postRefreshUpdate is to be called on a context after the state that it depends on has been refreshed
// if the context's view is set to another context we do nothing.
// if the context's view is the current view we trigger a focus; re-selecting the current item.
func (gui *Gui) postRefreshUpdate(c types.Context) error {
2022-02-05 07:56:36 +02:00
if gui.State.ViewContextMap.Get(c.GetViewName()).GetKey() != c.GetKey() {
2020-08-19 11:07:14 +02:00
return nil
}
if err := c.HandleRender(); err != nil {
return err
}
if gui.currentViewName() == c.GetViewName() {
if err := c.HandleFocus(); err != nil {
2020-08-19 10:41:57 +02:00
return err
}
}
return nil
}
func (gui *Gui) activateContext(c types.Context, opts ...types.OnFocusOpts) error {
2020-08-16 10:17:16 +02:00
viewName := c.GetViewName()
2020-08-19 10:41:57 +02:00
v, err := gui.g.View(viewName)
2020-08-16 10:17:16 +02:00
if err != nil {
return err
2020-08-16 10:17:16 +02:00
}
2022-02-05 07:56:36 +02:00
originalViewContextKey := gui.State.ViewContextMap.Get(viewName).GetKey()
2020-08-19 10:41:57 +02:00
2022-02-05 05:42:56 +02:00
gui.setWindowContext(c)
gui.setViewTabForContext(c)
2020-08-19 11:07:14 +02:00
if viewName == "main" {
2022-02-05 05:42:56 +02:00
gui.changeMainViewsContext(c)
2020-08-19 10:41:57 +02:00
} else {
2022-02-05 05:42:56 +02:00
gui.changeMainViewsContext(gui.State.Contexts.Normal)
}
2022-02-05 05:42:56 +02:00
gui.g.SetCurrentContext(string(c.GetKey()))
2020-08-16 10:17:16 +02:00
if _, err := gui.g.SetCurrentView(viewName); err != nil {
return err
2020-08-16 05:58:29 +02:00
}
2021-04-04 15:51:59 +02:00
v.Visible = true
2020-08-16 05:58:29 +02:00
2020-08-23 03:08:51 +02:00
// if the new context's view was previously displaying another context, render the new context
if originalViewContextKey != c.GetKey() {
if err := c.HandleRender(); err != nil {
return err
}
}
2022-02-05 07:56:36 +02:00
gui.ViewContextMapSet(viewName, c)
2020-08-16 05:58:29 +02:00
2020-08-23 03:08:51 +02:00
gui.g.Cursor = v.Editable
2020-08-16 05:58:29 +02:00
// render the options available for the current context at the bottom of the screen
optionsMap := c.GetOptionsMap()
if optionsMap == nil {
optionsMap = gui.globalOptionsMap()
2020-08-16 05:58:29 +02:00
}
gui.renderOptionsMap(optionsMap)
2020-08-16 05:58:29 +02:00
if err := c.HandleFocus(opts...); err != nil {
2020-08-16 05:58:29 +02:00
return err
}
return nil
}
2022-02-13 01:48:41 +02:00
func (gui *Gui) optionsMapToString(optionsMap map[string]string) string {
optionsArray := make([]string, 0)
for key, description := range optionsMap {
optionsArray = append(optionsArray, key+": "+description)
}
sort.Strings(optionsArray)
return strings.Join(optionsArray, ", ")
}
func (gui *Gui) renderOptionsMap(optionsMap map[string]string) {
_ = gui.renderString(gui.Views.Options, gui.optionsMapToString(optionsMap))
}
2022-02-05 07:56:36 +02:00
// also setting context on view for now. We'll need to pick one of these two approaches to stick with.
func (gui *Gui) ViewContextMapSet(viewName string, c types.Context) {
gui.State.ViewContextMap.Set(viewName, c)
view, err := gui.g.View(viewName)
if err != nil {
panic(err)
}
view.Context = string(c.GetKey())
}
2021-04-05 04:45:27 +02:00
// // currently unused
// func (gui *Gui) renderContextStack() string {
// result := ""
// for _, context := range gui.State.ContextManager.ContextStack {
// result += string(context.GetKey()) + "\n"
// }
// return result
// }
2020-08-16 05:58:29 +02:00
func (gui *Gui) currentContext() types.Context {
gui.State.ContextManager.RLock()
defer gui.State.ContextManager.RUnlock()
return gui.currentContextWithoutLock()
}
func (gui *Gui) currentContextWithoutLock() types.Context {
if len(gui.State.ContextManager.ContextStack) == 0 {
return gui.defaultSideContext()
}
return gui.State.ContextManager.ContextStack[len(gui.State.ContextManager.ContextStack)-1]
2020-08-16 02:05:45 +02:00
}
2021-04-04 15:51:59 +02:00
// the status panel is not yet a list context (and may never be), so this method is not
// quite the same as currentSideContext()
func (gui *Gui) currentSideListContext() types.IListContext {
2021-04-04 15:51:59 +02:00
context := gui.currentSideContext()
listContext, ok := context.(types.IListContext)
2021-04-04 15:51:59 +02:00
if !ok {
return nil
}
return listContext
}
func (gui *Gui) currentSideContext() types.Context {
gui.State.ContextManager.RLock()
defer gui.State.ContextManager.RUnlock()
stack := gui.State.ContextManager.ContextStack
2020-08-22 00:49:02 +02:00
// on startup the stack can be empty so we'll return an empty string in that case
if len(stack) == 0 {
2021-04-04 15:51:59 +02:00
return gui.defaultSideContext()
2020-08-22 00:49:02 +02:00
}
// find the first context in the stack with the type of types.SIDE_CONTEXT
2020-08-22 00:49:02 +02:00
for i := range stack {
context := stack[len(stack)-1-i]
if context.GetKind() == types.SIDE_CONTEXT {
2021-04-04 15:51:59 +02:00
return context
2020-08-22 00:49:02 +02:00
}
}
2021-04-04 15:51:59 +02:00
return gui.defaultSideContext()
2020-08-22 00:49:02 +02:00
}
2021-04-11 07:01:49 +02:00
// static as opposed to popup
func (gui *Gui) currentStaticContext() types.Context {
2021-04-11 07:01:49 +02:00
gui.State.ContextManager.RLock()
defer gui.State.ContextManager.RUnlock()
stack := gui.State.ContextManager.ContextStack
if len(stack) == 0 {
return gui.defaultSideContext()
}
// find the first context in the stack without a popup type
for i := range stack {
context := stack[len(stack)-1-i]
if context.GetKind() != types.TEMPORARY_POPUP && context.GetKind() != types.PERSISTENT_POPUP {
2021-04-11 07:01:49 +02:00
return context
}
}
return gui.defaultSideContext()
}
func (gui *Gui) defaultSideContext() types.Context {
2021-04-04 15:51:59 +02:00
if gui.State.Modes.Filtering.Active() {
return gui.State.Contexts.BranchCommits
} else {
return gui.State.Contexts.Files
}
2020-08-22 00:49:02 +02:00
}
2020-08-16 05:58:29 +02: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()
2020-08-16 09:29:06 +02:00
if err := gui.onViewFocusChange(); err != nil {
2020-08-16 05:58:29 +02:00
return err
}
// 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, newView); err != nil {
return err
}
previousView = newView
}
return nil
}
}
2020-08-16 02:05:45 +02:00
2020-08-16 09:29:06 +02:00
func (gui *Gui) onViewFocusChange() error {
2020-09-30 22:54:29 +02:00
gui.g.Mutexes.ViewsMutex.Lock()
defer gui.g.Mutexes.ViewsMutex.Unlock()
2020-08-16 05:58:29 +02:00
currentView := gui.g.CurrentView()
for _, view := range gui.g.Views() {
2021-04-11 07:01:49 +02:00
view.Highlight = view.Name() != "main" && view.Name() != "extras" && view == currentView
2020-08-16 05:58:29 +02:00
}
return nil
}
func (gui *Gui) onViewFocusLost(oldView *gocui.View, newView *gocui.View) error {
if oldView == nil {
2020-08-16 05:58:29 +02:00
return nil
}
2021-11-02 11:35:53 +02:00
_ = oldView.SetOriginX(0)
if oldView == gui.Views.CommitFiles && newView != gui.Views.Main && newView != gui.Views.Secondary && newView != gui.Views.Search {
2022-02-05 05:42:56 +02:00
gui.resetWindowContext(gui.State.Contexts.CommitFiles)
2021-04-03 06:56:11 +02:00
if err := gui.deactivateContext(gui.State.Contexts.CommitFiles); err != nil {
return err
}
}
2020-08-16 05:58:29 +02:00
return nil
}
2020-08-16 10:17:16 +02:00
// changeContext is a helper function for when we want to change a 'main' context
// which currently just means a context that affects both the main and secondary views
// other views can have their context changed directly but this function helps
// keep the main and secondary views in sync
2022-02-05 05:42:56 +02:00
func (gui *Gui) changeMainViewsContext(c types.Context) {
if gui.State.MainContext == c.GetKey() {
2020-08-16 10:17:16 +02:00
return
}
2022-02-05 05:42:56 +02:00
switch c.GetKey() {
2022-01-29 10:15:46 +02:00
case context.MAIN_NORMAL_CONTEXT_KEY, context.MAIN_PATCH_BUILDING_CONTEXT_KEY, context.MAIN_STAGING_CONTEXT_KEY, context.MAIN_MERGING_CONTEXT_KEY:
2022-02-05 07:56:36 +02:00
gui.ViewContextMapSet(gui.Views.Main.Name(), c)
gui.ViewContextMapSet(gui.Views.Secondary.Name(), c)
default:
2022-02-05 05:42:56 +02:00
panic(fmt.Sprintf("unknown context for main: %s", c.GetKey()))
2020-08-16 10:17:16 +02:00
}
2022-02-05 05:42:56 +02:00
gui.State.MainContext = c.GetKey()
2020-08-16 10:17:16 +02:00
}
2020-08-19 10:41:57 +02:00
func (gui *Gui) viewTabNames(viewName string) []string {
tabContexts := gui.State.ViewTabContextMap[viewName]
2020-08-19 10:41:57 +02:00
2020-09-30 00:27:12 +02:00
if len(tabContexts) == 0 {
return nil
}
2020-08-19 10:41:57 +02:00
result := make([]string, len(tabContexts))
for i, tabContext := range tabContexts {
result[i] = tabContext.Tab
2020-08-19 10:41:57 +02:00
}
return result
}
func (gui *Gui) setViewTabForContext(c types.Context) {
// search for the context in our map and if we find it, set the tab for the corresponding view
tabContexts, ok := gui.State.ViewTabContextMap[c.GetViewName()]
if !ok {
return
}
for tabIndex, tabContext := range tabContexts {
for _, context := range tabContext.Contexts {
if context.GetKey() == c.GetKey() {
// get the view, set the tab
v, err := gui.g.View(c.GetViewName())
if err != nil {
gui.c.Log.Error(err)
return
}
v.TabIndex = tabIndex
return
}
}
}
}
func (gui *Gui) contextForContextKey(contextKey types.ContextKey) (types.Context, bool) {
2022-02-05 07:56:36 +02:00
for _, context := range gui.State.Contexts.Flatten() {
2020-08-19 11:07:14 +02:00
if context.GetKey() == contextKey {
2020-09-27 01:37:22 +02:00
return context, true
2020-08-19 11:07:14 +02:00
}
}
2020-09-27 01:37:22 +02:00
return nil, false
2020-08-19 11:07:14 +02:00
}
2021-04-04 16:44:13 +02:00
func (gui *Gui) rerenderView(view *gocui.View) error {
2022-02-05 07:56:36 +02:00
return gui.State.ViewContextMap.Get(view.Name()).HandleRender()
2020-08-19 10:41:57 +02:00
}
func (gui *Gui) getSideContextSelectedItemId() string {
2021-04-04 15:51:59 +02:00
currentSideContext := gui.currentSideListContext()
if currentSideContext == nil {
return ""
}
2022-01-30 11:03:08 +02:00
return currentSideContext.GetSelectedItemId()
}
2022-02-05 05:42:56 +02:00
func (gui *Gui) isContextVisible(c types.Context) bool {
2022-02-05 07:56:36 +02:00
return gui.State.WindowViewNameMap[c.GetWindowName()] == c.GetViewName() && gui.State.ViewContextMap.Get(c.GetViewName()).GetKey() == c.GetKey()
2022-02-05 05:42:56 +02:00
}
// currently unused
// func (gui *Gui) getCurrentSideView() *gocui.View {
// currentSideContext := gui.currentSideContext()
// if currentSideContext == nil {
// return nil
// }
// view, _ := gui.g.View(currentSideContext.GetViewName())
// return view
// }
// currently unused
// func (gui *Gui) renderContextStack() string {
// result := ""
// for _, context := range gui.State.ContextManager.ContextStack {
// result += context.GetViewName() + "\n"
// }
// return result
// }