426 lines
9.2 KiB
Go
426 lines
9.2 KiB
Go
package tui
|
|
|
|
import (
|
|
"sort"
|
|
"strings"
|
|
"time"
|
|
|
|
"filepass/internal/pages"
|
|
"filepass/internal/services"
|
|
|
|
tea "charm.land/bubbletea/v2"
|
|
)
|
|
|
|
type configLoadedMsg struct {
|
|
servers map[string]services.Server
|
|
err error
|
|
}
|
|
|
|
type serverAddedMsg struct {
|
|
name string
|
|
servers map[string]services.Server
|
|
}
|
|
|
|
type clearFlashMsg struct{}
|
|
|
|
type storageFilesMsg struct {
|
|
files []string
|
|
err error
|
|
}
|
|
|
|
func checkStorageCmd(store *services.ServicesStore, serverName string) tea.Cmd {
|
|
return func() tea.Msg {
|
|
storage, err := store.NewStorageService(serverName)
|
|
if err != nil {
|
|
return storageFilesMsg{err: err}
|
|
}
|
|
files, err := storage.Check()
|
|
return storageFilesMsg{files: files, err: err}
|
|
}
|
|
}
|
|
|
|
func clearFlashAfter(d time.Duration) tea.Cmd {
|
|
return tea.Tick(d, func(time.Time) tea.Msg {
|
|
return clearFlashMsg{}
|
|
})
|
|
}
|
|
|
|
// sortedServerNames returns the keys of servers sorted alphabetically.
|
|
func sortedServerNames(servers map[string]services.Server) []string {
|
|
names := make([]string, 0, len(servers))
|
|
for name := range servers {
|
|
names = append(names, name)
|
|
}
|
|
sort.Strings(names)
|
|
return names
|
|
}
|
|
|
|
// isDisabled reports whether a menu item is non-interactive given current state.
|
|
func (m TUIInterface) isDisabled(i int) bool {
|
|
return m.MenuItems[i].RequiresServers && m.NoServers
|
|
}
|
|
|
|
// nextSelectable finds the next non-disabled index in direction (+1 or -1).
|
|
func (m TUIInterface) nextSelectable(from, dir int) int {
|
|
i := from + dir
|
|
for i >= 0 && i < len(m.MenuItems) {
|
|
if !m.isDisabled(i) {
|
|
return i
|
|
}
|
|
i += dir
|
|
}
|
|
return from
|
|
}
|
|
|
|
func (m TUIInterface) Update(msg tea.Msg) (tea.Model, tea.Cmd) {
|
|
switch msg := msg.(type) {
|
|
|
|
case pages.HomePageMsg:
|
|
m.Page = pageHome
|
|
m.MenuItems = pages.HomeMenuItems()
|
|
m.Selected = 0
|
|
return m, nil
|
|
|
|
case pages.ConfigPageMsg:
|
|
m.Page = pageConfig
|
|
m.MenuItems = pages.ConfigMenuItems()
|
|
m.Selected = 0
|
|
return m, nil
|
|
|
|
case pages.AddServerPageMsg:
|
|
m.Page = pageAddServer
|
|
m.Form = newAddServerForm()
|
|
m.FormErr = ""
|
|
return m, nil
|
|
|
|
case pages.SelectServerPageMsg:
|
|
m.Page = pageSelectServer
|
|
m.Selected = 0
|
|
return m, nil
|
|
|
|
case pages.ServerActionsPageMsg:
|
|
m.Page = pageServerActions
|
|
m.ActiveServer = msg.ServerName
|
|
m.MenuItems = pages.ServerActionItems()
|
|
m.Selected = 0
|
|
m.FileSelected = 0
|
|
m.FileFocused = true
|
|
m.StorageFiles = nil
|
|
m.StorageErr = nil
|
|
m.StorageLoading = true
|
|
return m, checkStorageCmd(m.Services, msg.ServerName)
|
|
|
|
case pages.FileActionPageMsg:
|
|
m.Page = pageFileAction
|
|
m.ActiveFile = msg.Filename
|
|
m.MenuItems = pages.FileActionItems()
|
|
m.Selected = 0
|
|
return m, nil
|
|
|
|
case storageFilesMsg:
|
|
m.StorageLoading = false
|
|
m.StorageFiles = msg.files
|
|
m.StorageErr = msg.err
|
|
m.FileSelected = 0
|
|
return m, nil
|
|
|
|
case configLoadedMsg:
|
|
if msg.err != nil {
|
|
m.InitErr = msg.err
|
|
return m, nil
|
|
}
|
|
m.Servers = msg.servers
|
|
m.ServerNames = sortedServerNames(msg.servers)
|
|
m.NoServers = len(msg.servers) == 0
|
|
return m, nil
|
|
|
|
case serverAddedMsg:
|
|
m.Servers = msg.servers
|
|
m.ServerNames = sortedServerNames(msg.servers)
|
|
m.NoServers = len(msg.servers) == 0
|
|
m.Page = pageConfig
|
|
m.MenuItems = pages.ConfigMenuItems()
|
|
m.Selected = 0
|
|
m.FlashMsg = "✓ \"" + msg.name + "\" added successfully."
|
|
return m, clearFlashAfter(2 * time.Second)
|
|
|
|
case clearFlashMsg:
|
|
m.FlashMsg = ""
|
|
return m, nil
|
|
|
|
case tea.WindowSizeMsg:
|
|
m.WindowWidth = msg.Width
|
|
m.WindowHeight = msg.Height
|
|
return m, nil
|
|
|
|
case tea.KeyPressMsg:
|
|
if m.Page == pageAddServer {
|
|
return m.updateAddServer(msg)
|
|
}
|
|
if m.Page == pageSelectServer {
|
|
return m.updateSelectServer(msg)
|
|
}
|
|
if m.Page == pageServerActions {
|
|
return m.updateServerActions(msg)
|
|
}
|
|
if m.Page == pageFileAction {
|
|
return m.updateFileAction(msg)
|
|
}
|
|
|
|
switch msg.String() {
|
|
case "up", "k":
|
|
m.Selected = m.nextSelectable(m.Selected, -1)
|
|
case "down", "j":
|
|
m.Selected = m.nextSelectable(m.Selected, +1)
|
|
case "enter":
|
|
if m.isDisabled(m.Selected) {
|
|
return m, nil
|
|
}
|
|
switch m.MenuItems[m.Selected].Key {
|
|
case "exit":
|
|
m.Quitting = true
|
|
return m, tea.Quit
|
|
case "config":
|
|
return m, func() tea.Msg { return pages.ConfigPageMsg{} }
|
|
case "back":
|
|
return m, func() tea.Msg { return pages.HomePageMsg{} }
|
|
case "add":
|
|
return m, func() tea.Msg { return pages.AddServerPageMsg{} }
|
|
case "server":
|
|
return m, func() tea.Msg { return pages.SelectServerPageMsg{} }
|
|
// TODO: "edit", "remove"
|
|
}
|
|
case "ctrl+c":
|
|
m.Quitting = true
|
|
return m, tea.Quit
|
|
case "esc":
|
|
if m.Page == pageConfig {
|
|
return m, func() tea.Msg { return pages.HomePageMsg{} }
|
|
}
|
|
m.Quitting = true
|
|
return m, tea.Quit
|
|
}
|
|
|
|
case tea.PasteMsg:
|
|
if m.Page == pageAddServer {
|
|
return m.updateAddServerPaste(msg.Content)
|
|
}
|
|
|
|
case tea.ClipboardMsg:
|
|
if m.Page == pageAddServer {
|
|
return m.updateAddServerPaste(msg.Content)
|
|
}
|
|
}
|
|
|
|
return m, nil
|
|
}
|
|
|
|
func (m TUIInterface) updateServerActions(msg tea.KeyPressMsg) (tea.Model, tea.Cmd) {
|
|
switch msg.String() {
|
|
case "tab":
|
|
// toggle focus between action menu and file list
|
|
if len(m.StorageFiles) > 0 {
|
|
m.FileFocused = !m.FileFocused
|
|
}
|
|
|
|
case "up", "k":
|
|
if m.FileFocused {
|
|
if m.FileSelected > 0 {
|
|
m.FileSelected--
|
|
}
|
|
} else {
|
|
if m.Selected > 0 {
|
|
m.Selected--
|
|
}
|
|
}
|
|
|
|
case "down", "j":
|
|
if m.FileFocused {
|
|
if m.FileSelected < len(m.StorageFiles)-1 {
|
|
m.FileSelected++
|
|
}
|
|
} else {
|
|
if m.Selected < len(m.MenuItems)-1 {
|
|
m.Selected++
|
|
}
|
|
}
|
|
|
|
case "enter":
|
|
if m.FileFocused && len(m.StorageFiles) > 0 {
|
|
file := m.StorageFiles[m.FileSelected]
|
|
server := m.ActiveServer
|
|
return m, func() tea.Msg {
|
|
return pages.FileActionPageMsg{ServerName: server, Filename: file}
|
|
}
|
|
}
|
|
switch m.MenuItems[m.Selected].Key {
|
|
case "send":
|
|
// TODO: navigate to send page
|
|
case "clean":
|
|
// TODO: navigate to clean all page
|
|
}
|
|
|
|
case "ctrl+c":
|
|
m.Quitting = true
|
|
return m, tea.Quit
|
|
|
|
case "esc":
|
|
return m, func() tea.Msg { return pages.SelectServerPageMsg{} }
|
|
}
|
|
|
|
return m, nil
|
|
}
|
|
|
|
func (m TUIInterface) updateFileAction(msg tea.KeyPressMsg) (tea.Model, tea.Cmd) {
|
|
switch msg.String() {
|
|
case "up", "k":
|
|
if m.Selected > 0 {
|
|
m.Selected--
|
|
}
|
|
case "down", "j":
|
|
if m.Selected < len(m.MenuItems)-1 {
|
|
m.Selected++
|
|
}
|
|
case "enter":
|
|
server := m.ActiveServer
|
|
file := m.ActiveFile
|
|
switch m.MenuItems[m.Selected].Key {
|
|
case "get":
|
|
_ = server
|
|
_ = file
|
|
// TODO: implement get
|
|
case "delete":
|
|
_ = server
|
|
_ = file
|
|
// TODO: implement delete
|
|
}
|
|
case "ctrl+c":
|
|
m.Quitting = true
|
|
return m, tea.Quit
|
|
case "esc":
|
|
server := m.ActiveServer
|
|
return m, func() tea.Msg { return pages.ServerActionsPageMsg{ServerName: server} }
|
|
}
|
|
return m, nil
|
|
}
|
|
|
|
func (m TUIInterface) updateSelectServer(msg tea.KeyPressMsg) (tea.Model, tea.Cmd) {
|
|
last := len(m.ServerNames) - 1
|
|
switch msg.String() {
|
|
case "up", "k":
|
|
if m.Selected > 0 {
|
|
m.Selected--
|
|
}
|
|
case "down", "j":
|
|
if m.Selected < last {
|
|
m.Selected++
|
|
}
|
|
case "enter":
|
|
if m.Selected >= 0 && m.Selected < len(m.ServerNames) {
|
|
name := m.ServerNames[m.Selected]
|
|
return m, func() tea.Msg { return pages.ServerActionsPageMsg{ServerName: name} }
|
|
}
|
|
case "ctrl+c":
|
|
m.Quitting = true
|
|
return m, tea.Quit
|
|
case "esc":
|
|
return m, func() tea.Msg { return pages.HomePageMsg{} }
|
|
}
|
|
return m, nil
|
|
}
|
|
|
|
func (m TUIInterface) updateAddServer(msg tea.KeyPressMsg) (tea.Model, tea.Cmd) {
|
|
f := m.Form
|
|
|
|
switch msg.String() {
|
|
case "tab", "down":
|
|
m.Form = f.focusNext()
|
|
return m, nil
|
|
|
|
case "shift+tab", "up":
|
|
m.Form = f.focusPrev()
|
|
return m, nil
|
|
|
|
case "enter":
|
|
if f.focused < fieldSave {
|
|
m.Form = f.focusNext()
|
|
return m, nil
|
|
}
|
|
if f.focused == fieldSave {
|
|
return m.submitAddServer()
|
|
}
|
|
if f.focused == fieldBack {
|
|
return m, func() tea.Msg { return pages.ConfigPageMsg{} }
|
|
}
|
|
|
|
case "ctrl+c":
|
|
m.Quitting = true
|
|
return m, tea.Quit
|
|
|
|
case "ctrl+v":
|
|
if f.focused < len(f.inputs) {
|
|
return m, tea.ReadClipboard
|
|
}
|
|
return m, nil
|
|
|
|
case "esc":
|
|
return m, func() tea.Msg { return pages.ConfigPageMsg{} }
|
|
}
|
|
|
|
if f.focused < len(f.inputs) {
|
|
var cmd tea.Cmd
|
|
f.inputs[f.focused], cmd = f.inputs[f.focused].Update(msg)
|
|
if f.focused == fieldName {
|
|
m.FormErr = ""
|
|
}
|
|
m.Form = f
|
|
return m, cmd
|
|
}
|
|
|
|
return m, nil
|
|
}
|
|
|
|
func (m TUIInterface) updateAddServerPaste(text string) (tea.Model, tea.Cmd) {
|
|
f := m.Form
|
|
if f.focused >= len(f.inputs) {
|
|
return m, nil
|
|
}
|
|
var cmd tea.Cmd
|
|
f.inputs[f.focused], cmd = f.inputs[f.focused].Update(tea.PasteMsg{Content: text})
|
|
if f.focused == fieldName {
|
|
m.FormErr = ""
|
|
}
|
|
m.Form = f
|
|
return m, cmd
|
|
}
|
|
|
|
func (m TUIInterface) submitAddServer() (tea.Model, tea.Cmd) {
|
|
f := m.Form
|
|
name := strings.TrimSpace(f.inputs[fieldName].Value())
|
|
|
|
if !f.canSave() {
|
|
return m, nil
|
|
}
|
|
|
|
if m.Services.Config.HasServer(name) {
|
|
m.FormErr = "✗ \"" + name + "\" already exists."
|
|
return m, nil
|
|
}
|
|
|
|
s := services.Server{
|
|
Host: strings.TrimSpace(f.inputs[fieldHost].Value()),
|
|
User: strings.TrimSpace(f.inputs[fieldUser].Value()),
|
|
PrivateKey: strings.TrimSpace(f.inputs[fieldPrivateKey].Value()),
|
|
Port: strings.TrimSpace(f.inputs[fieldPort].Value()),
|
|
}
|
|
|
|
if err := m.Services.Config.AddServer(name, s); err != nil {
|
|
m.FormErr = "✗ " + err.Error()
|
|
return m, nil
|
|
}
|
|
|
|
return m, func() tea.Msg {
|
|
return serverAddedMsg{name: name, servers: m.Services.Config.Servers()}
|
|
}
|
|
}
|