k9s/internal/ui/action.go

221 lines
4.2 KiB
Go

// SPDX-License-Identifier: Apache-2.0
// Copyright Authors of K9s
package ui
import (
"log/slog"
"slices"
"sync"
"github.com/derailed/k9s/internal/model"
"github.com/derailed/k9s/internal/slogs"
"github.com/derailed/tcell/v2"
)
type (
// RangeFn represents a range iteration callback.
RangeFn func(tcell.Key, KeyAction)
// ActionHandler handles a keyboard command.
ActionHandler func(*tcell.EventKey) *tcell.EventKey
// ActionOpts tracks various action options.
ActionOpts struct {
Visible bool
Shared bool
Plugin bool
HotKey bool
Dangerous bool
}
// KeyAction represents a keyboard action.
KeyAction struct {
Description string
Action ActionHandler
Opts ActionOpts
}
// KeyMap tracks key to action mappings.
KeyMap map[tcell.Key]KeyAction
// KeyActions tracks mappings between keystrokes and actions.
KeyActions struct {
actions KeyMap
mx sync.RWMutex
}
)
// NewKeyAction returns a new keyboard action.
func NewKeyAction(d string, a ActionHandler, visible bool) KeyAction {
return NewKeyActionWithOpts(d, a, ActionOpts{
Visible: visible,
})
}
// NewSharedKeyAction returns a new shared keyboard action.
func NewSharedKeyAction(d string, a ActionHandler, visible bool) KeyAction {
return NewKeyActionWithOpts(d, a, ActionOpts{
Visible: visible,
Shared: true,
})
}
// NewKeyActionWithOpts returns a new keyboard action.
func NewKeyActionWithOpts(d string, a ActionHandler, opts ActionOpts) KeyAction {
return KeyAction{
Description: d,
Action: a,
Opts: opts,
}
}
// NewKeyActions returns a new instance.
func NewKeyActions() *KeyActions {
return &KeyActions{
actions: make(map[tcell.Key]KeyAction),
}
}
// NewKeyActionsFromMap construct actions from key map.
func NewKeyActionsFromMap(mm KeyMap) *KeyActions {
return &KeyActions{actions: mm}
}
// Get fetches an action given a key.
func (a *KeyActions) Get(key tcell.Key) (KeyAction, bool) {
a.mx.RLock()
defer a.mx.RUnlock()
v, ok := a.actions[key]
return v, ok
}
// Len returns action mapping count.
func (a *KeyActions) Len() int {
a.mx.RLock()
defer a.mx.RUnlock()
return len(a.actions)
}
// Reset clears out actions.
func (a *KeyActions) Reset(aa *KeyActions) {
a.Clear()
a.Merge(aa)
}
// Range ranges over all actions and triggers a given function.
func (a *KeyActions) Range(f RangeFn) {
var km KeyMap
a.mx.RLock()
km = a.actions
a.mx.RUnlock()
for k, v := range km {
f(k, v)
}
}
// Add adds a new key action.
func (a *KeyActions) Add(k tcell.Key, ka KeyAction) {
a.mx.Lock()
defer a.mx.Unlock()
a.actions[k] = ka
}
// Bulk bulk insert key mappings.
func (a *KeyActions) Bulk(aa KeyMap) {
a.mx.Lock()
defer a.mx.Unlock()
for k, v := range aa {
a.actions[k] = v
}
}
// Merge merges given actions into existing set.
func (a *KeyActions) Merge(aa *KeyActions) {
a.mx.Lock()
defer a.mx.Unlock()
for k, v := range aa.actions {
a.actions[k] = v
}
}
// Clear remove all actions.
func (a *KeyActions) Clear() {
a.mx.Lock()
defer a.mx.Unlock()
for k := range a.actions {
delete(a.actions, k)
}
}
// ClearDanger remove all dangerous actions.
func (a *KeyActions) ClearDanger() {
a.mx.Lock()
defer a.mx.Unlock()
for k, v := range a.actions {
if v.Opts.Dangerous {
delete(a.actions, k)
}
}
}
// Set replace actions with new ones.
func (a *KeyActions) Set(aa *KeyActions) {
a.mx.Lock()
defer a.mx.Unlock()
for k, v := range aa.actions {
a.actions[k] = v
}
}
// Delete deletes actions by the given keys.
func (a *KeyActions) Delete(kk ...tcell.Key) {
a.mx.Lock()
defer a.mx.Unlock()
for _, k := range kk {
delete(a.actions, k)
}
}
// Hints returns a collection of hints.
func (a *KeyActions) Hints() model.MenuHints {
a.mx.RLock()
defer a.mx.RUnlock()
kk := make([]tcell.Key, 0, len(a.actions))
for k := range a.actions {
if !a.actions[k].Opts.Shared {
kk = append(kk, k)
}
}
slices.Sort(kk)
hh := make(model.MenuHints, 0, len(kk))
for _, k := range kk {
if name, ok := tcell.KeyNames[k]; ok {
hh = append(hh,
model.MenuHint{
Mnemonic: name,
Description: a.actions[k].Description,
Visible: a.actions[k].Opts.Visible,
},
)
} else {
slog.Error("Unable to locate key name", slogs.Key, k)
}
}
return hh
}