mirror of
https://github.com/jesseduffield/lazygit.git
synced 2025-04-23 12:18:51 +02:00
219 lines
5.8 KiB
Go
219 lines
5.8 KiB
Go
package context
|
|
|
|
import (
|
|
"errors"
|
|
|
|
"github.com/jesseduffield/lazygit/pkg/gui/keybindings"
|
|
"github.com/jesseduffield/lazygit/pkg/gui/style"
|
|
"github.com/jesseduffield/lazygit/pkg/gui/types"
|
|
"github.com/jesseduffield/lazygit/pkg/utils"
|
|
"github.com/samber/lo"
|
|
)
|
|
|
|
type MenuContext struct {
|
|
c *ContextCommon
|
|
|
|
*MenuViewModel
|
|
*ListContextTrait
|
|
}
|
|
|
|
var _ types.IListContext = (*MenuContext)(nil)
|
|
|
|
func NewMenuContext(
|
|
c *ContextCommon,
|
|
) *MenuContext {
|
|
viewModel := NewMenuViewModel(c)
|
|
|
|
return &MenuContext{
|
|
c: c,
|
|
MenuViewModel: viewModel,
|
|
ListContextTrait: &ListContextTrait{
|
|
Context: NewSimpleContext(NewBaseContext(NewBaseContextOpts{
|
|
View: c.Views().Menu,
|
|
WindowName: "menu",
|
|
Key: "menu",
|
|
Kind: types.TEMPORARY_POPUP,
|
|
Focusable: true,
|
|
HasUncontrolledBounds: true,
|
|
})),
|
|
ListRenderer: ListRenderer{
|
|
list: viewModel,
|
|
getDisplayStrings: viewModel.GetDisplayStrings,
|
|
getColumnAlignments: func() []utils.Alignment { return viewModel.columnAlignment },
|
|
getNonModelItems: viewModel.GetNonModelItems,
|
|
},
|
|
c: c,
|
|
},
|
|
}
|
|
}
|
|
|
|
type MenuViewModel struct {
|
|
c *ContextCommon
|
|
menuItems []*types.MenuItem
|
|
prompt string
|
|
promptLines []string
|
|
columnAlignment []utils.Alignment
|
|
*FilteredListViewModel[*types.MenuItem]
|
|
}
|
|
|
|
func NewMenuViewModel(c *ContextCommon) *MenuViewModel {
|
|
self := &MenuViewModel{
|
|
menuItems: nil,
|
|
c: c,
|
|
}
|
|
|
|
self.FilteredListViewModel = NewFilteredListViewModel(
|
|
func() []*types.MenuItem { return self.menuItems },
|
|
func(item *types.MenuItem) []string { return item.LabelColumns },
|
|
)
|
|
|
|
return self
|
|
}
|
|
|
|
func (self *MenuViewModel) SetMenuItems(items []*types.MenuItem, columnAlignment []utils.Alignment) {
|
|
self.menuItems = items
|
|
self.columnAlignment = columnAlignment
|
|
}
|
|
|
|
func (self *MenuViewModel) GetPrompt() string {
|
|
return self.prompt
|
|
}
|
|
|
|
func (self *MenuViewModel) SetPrompt(prompt string) {
|
|
self.prompt = prompt
|
|
self.promptLines = nil
|
|
}
|
|
|
|
func (self *MenuViewModel) GetPromptLines() []string {
|
|
return self.promptLines
|
|
}
|
|
|
|
func (self *MenuViewModel) SetPromptLines(promptLines []string) {
|
|
self.promptLines = promptLines
|
|
}
|
|
|
|
// TODO: move into presentation package
|
|
func (self *MenuViewModel) GetDisplayStrings(_ int, _ int) [][]string {
|
|
menuItems := self.FilteredListViewModel.GetItems()
|
|
|
|
return lo.Map(menuItems, func(item *types.MenuItem, _ int) []string {
|
|
displayStrings := item.LabelColumns
|
|
if item.DisabledReason != nil {
|
|
displayStrings[0] = style.FgDefault.SetStrikethrough().Sprint(displayStrings[0])
|
|
}
|
|
|
|
keyLabel := ""
|
|
if item.Key != nil {
|
|
keyLabel = style.FgCyan.Sprint(keybindings.LabelFromKey(item.Key))
|
|
}
|
|
|
|
checkMark := ""
|
|
switch item.Widget {
|
|
case types.MenuWidgetNone:
|
|
// do nothing
|
|
case types.MenuWidgetRadioButtonSelected:
|
|
checkMark = "(•)"
|
|
case types.MenuWidgetRadioButtonUnselected:
|
|
checkMark = "( )"
|
|
case types.MenuWidgetCheckboxSelected:
|
|
checkMark = "[✓]"
|
|
case types.MenuWidgetCheckboxUnselected:
|
|
checkMark = "[ ]"
|
|
}
|
|
|
|
displayStrings = utils.Prepend(displayStrings, keyLabel, checkMark)
|
|
return displayStrings
|
|
})
|
|
}
|
|
|
|
func (self *MenuViewModel) GetNonModelItems() []*NonModelItem {
|
|
result := []*NonModelItem{}
|
|
result = append(result, lo.Map(self.promptLines, func(line string, _ int) *NonModelItem {
|
|
return &NonModelItem{
|
|
Index: 0,
|
|
Column: 0,
|
|
Content: line,
|
|
}
|
|
})...)
|
|
|
|
// Don't display section headers when we are filtering, and the filter mode
|
|
// is fuzzy. The reason is that filtering changes the order of the items
|
|
// (they are sorted by best match), so all the sections would be messed up.
|
|
if self.FilteredListViewModel.IsFiltering() && self.c.UserConfig().Gui.UseFuzzySearch() {
|
|
return result
|
|
}
|
|
|
|
menuItems := self.FilteredListViewModel.GetItems()
|
|
var prevSection *types.MenuSection = nil
|
|
for i, menuItem := range menuItems {
|
|
if menuItem.Section != nil && menuItem.Section != prevSection {
|
|
if prevSection != nil {
|
|
result = append(result, &NonModelItem{
|
|
Index: i,
|
|
Column: 1,
|
|
Content: "",
|
|
})
|
|
}
|
|
|
|
result = append(result, &NonModelItem{
|
|
Index: i,
|
|
Column: 1,
|
|
Content: style.FgGreen.SetBold().Sprintf("--- %s ---", menuItem.Section.Title),
|
|
})
|
|
prevSection = menuItem.Section
|
|
}
|
|
}
|
|
|
|
return result
|
|
}
|
|
|
|
func (self *MenuContext) GetKeybindings(opts types.KeybindingsOpts) []*types.Binding {
|
|
basicBindings := self.ListContextTrait.GetKeybindings(opts)
|
|
menuItemsWithKeys := lo.Filter(self.menuItems, func(item *types.MenuItem, _ int) bool {
|
|
return item.Key != nil
|
|
})
|
|
|
|
menuItemBindings := lo.Map(menuItemsWithKeys, func(item *types.MenuItem, _ int) *types.Binding {
|
|
return &types.Binding{
|
|
Key: item.Key,
|
|
Handler: func() error { return self.OnMenuPress(item) },
|
|
}
|
|
})
|
|
|
|
// appending because that means the menu item bindings have lower precedence.
|
|
// So if a basic binding is to escape from the menu, we want that to still be
|
|
// what happens when you press escape. This matters when we're showing the menu
|
|
// for all keybindings of say the files context.
|
|
return append(basicBindings, menuItemBindings...)
|
|
}
|
|
|
|
func (self *MenuContext) OnMenuPress(selectedItem *types.MenuItem) error {
|
|
if selectedItem != nil && selectedItem.DisabledReason != nil {
|
|
if selectedItem.DisabledReason.ShowErrorInPanel {
|
|
return errors.New(selectedItem.DisabledReason.Text)
|
|
}
|
|
|
|
self.c.ErrorToast(self.c.Tr.DisabledMenuItemPrefix + selectedItem.DisabledReason.Text)
|
|
return nil
|
|
}
|
|
|
|
if err := self.c.Context().Pop(); err != nil {
|
|
return err
|
|
}
|
|
|
|
if selectedItem == nil {
|
|
return nil
|
|
}
|
|
|
|
if err := selectedItem.OnPress(); err != nil {
|
|
return err
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// There is currently no need to use range-select in a menu so we're disabling it.
|
|
func (self *MenuContext) RangeSelectEnabled() bool {
|
|
return false
|
|
}
|