gui/pkg/systray/menu.go
Snider b559562dd9
Some checks failed
Security Scan / security (pull_request) Failing after 28s
Test / test (pull_request) Failing after 1m59s
fix(dx): use coreerr.E() and go-io, update CLAUDE.md, add tests
- Replace 90+ fmt.Errorf calls with coreerr.E() from go-log across
  display, window, systray, keybinding, contextmenu, and mcp packages
- Replace os.ReadFile/WriteFile/MkdirAll with coreio.Local in
  window/layout.go and window/state.go
- Update CLAUDE.md: fix key files table for new package structure,
  document error handling and file I/O conventions, add missing deps
- Add 37 tests for window package (task handlers, persistence,
  tiling modes, snap positions, workflow layouts)
- Window coverage: 47.1% → 69.8%

Co-Authored-By: Virgil <virgil@lethean.io>
2026-03-17 09:05:35 +00:00

80 lines
1.9 KiB
Go

// pkg/systray/menu.go
package systray
import coreerr "forge.lthn.ai/core/go-log"
// SetMenu sets a dynamic menu on the tray from TrayMenuItem descriptors.
func (m *Manager) SetMenu(items []TrayMenuItem) error {
if m.tray == nil {
return coreerr.E("systray.SetMenu", "tray not initialised", nil)
}
menu := m.buildMenu(items)
m.tray.SetMenu(menu)
return nil
}
// buildMenu recursively builds a PlatformMenu from TrayMenuItem descriptors.
func (m *Manager) buildMenu(items []TrayMenuItem) PlatformMenu {
menu := m.platform.NewMenu()
for _, item := range items {
if item.Type == "separator" {
menu.AddSeparator()
continue
}
if len(item.Submenu) > 0 {
sub := m.buildMenu(item.Submenu)
mi := menu.Add(item.Label)
_ = mi.AddSubmenu()
_ = sub // TODO: wire sub into parent via platform
continue
}
mi := menu.Add(item.Label)
if item.Tooltip != "" {
mi.SetTooltip(item.Tooltip)
}
if item.Disabled {
mi.SetEnabled(false)
}
if item.Checked {
mi.SetChecked(true)
}
if item.ActionID != "" {
actionID := item.ActionID
mi.OnClick(func() {
if cb, ok := m.GetCallback(actionID); ok {
cb()
}
})
}
}
return menu
}
// RegisterCallback registers a callback for a menu action ID.
func (m *Manager) RegisterCallback(actionID string, callback func()) {
m.mu.Lock()
m.callbacks[actionID] = callback
m.mu.Unlock()
}
// UnregisterCallback removes a callback.
func (m *Manager) UnregisterCallback(actionID string) {
m.mu.Lock()
delete(m.callbacks, actionID)
m.mu.Unlock()
}
// GetCallback returns the callback for an action ID.
func (m *Manager) GetCallback(actionID string) (func(), bool) {
m.mu.RLock()
defer m.mu.RUnlock()
cb, ok := m.callbacks[actionID]
return cb, ok
}
// GetInfo returns tray status information.
func (m *Manager) GetInfo() map[string]any {
return map[string]any{
"active": m.IsActive(),
}
}