add:addserver

This commit is contained in:
2026-04-06 01:57:00 +09:00
parent ed19e0ba4e
commit 8dff078cea
5 changed files with 173 additions and 17 deletions

View File

@@ -11,22 +11,24 @@ const (
pageHome page = iota
pageConfig
pageAddServer
pageSelectServer
)
type TUIInterface struct {
Services *services.ServicesStore
Page page
MenuItems []pages.MenuItem
Selected int
Servers map[string]services.Server
NoServers bool
InitErr error
FlashMsg string
Form addServerForm
FormErr string // inline field error (e.g. duplicate name)
Quitting bool
WindowWidth int
WindowHeight int
Services *services.ServicesStore
Page page
MenuItems []pages.MenuItem
Selected int
Servers map[string]services.Server
ServerNames []string // sorted, stable order for list rendering
NoServers bool
InitErr error
FlashMsg string
Form addServerForm
FormErr string // inline field error (e.g. duplicate name)
Quitting bool
WindowWidth int
WindowHeight int
}
func NewTUIInterface(store *services.ServicesStore) TUIInterface {

View File

@@ -1,6 +1,7 @@
package tui
import (
"sort"
"strings"
"time"
@@ -28,6 +29,16 @@ func clearFlashAfter(d time.Duration) tea.Cmd {
})
}
// 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
@@ -66,17 +77,24 @@ func (m TUIInterface) Update(msg tea.Msg) (tea.Model, tea.Cmd) {
m.FormErr = ""
return m, nil
case pages.SelectServerPageMsg:
m.Page = pageSelectServer
m.Selected = 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()
@@ -98,6 +116,10 @@ func (m TUIInterface) Update(msg tea.Msg) (tea.Model, tea.Cmd) {
if m.Page == pageAddServer {
return m.updateAddServer(msg)
}
// server list has its own key handling
if m.Page == pageSelectServer {
return m.updateSelectServer(msg)
}
switch msg.String() {
case "up", "k":
@@ -118,7 +140,9 @@ func (m TUIInterface) Update(msg tea.Msg) (tea.Model, tea.Cmd) {
return m, func() tea.Msg { return pages.HomePageMsg{} }
case "add":
return m, func() tea.Msg { return pages.AddServerPageMsg{} }
// TODO: "server", "edit", "remove"
case "server":
return m, func() tea.Msg { return pages.SelectServerPageMsg{} }
// TODO: "edit", "remove"
}
case "ctrl+c":
m.Quitting = true
@@ -130,6 +154,16 @@ func (m TUIInterface) Update(msg tea.Msg) (tea.Model, tea.Cmd) {
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
@@ -166,6 +200,13 @@ func (m TUIInterface) updateAddServer(msg tea.KeyPressMsg) (tea.Model, tea.Cmd)
m.Quitting = true
return m, tea.Quit
case "ctrl+v":
// OSC52 clipboard read; result arrives as tea.ClipboardMsg
if f.focused < len(f.inputs) {
return m, tea.ReadClipboard
}
return m, nil
case "esc":
return m, func() tea.Msg { return pages.ConfigPageMsg{} }
}
@@ -185,6 +226,42 @@ func (m TUIInterface) updateAddServer(msg tea.KeyPressMsg) (tea.Model, tea.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) 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":
// TODO: connect to selected server
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) submitAddServer() (tea.Model, tea.Cmd) {
f := m.Form
name := strings.TrimSpace(f.inputs[fieldName].Value())

View File

@@ -23,6 +23,8 @@ func (m TUIInterface) subtitle() string {
return "Configuration"
case pageAddServer:
return "Add Server"
case pageSelectServer:
return "Select Server"
default:
return "Secure file transfer"
}
@@ -46,6 +48,8 @@ func (m TUIInterface) View() tea.View {
switch m.Page {
case pageAddServer:
body = m.viewAddServer()
case pageSelectServer:
body = m.viewSelectServer()
default:
body = m.viewMenu()
}
@@ -60,13 +64,22 @@ func (m TUIInterface) View() tea.View {
)
var footerStr string
if m.Page == pageAddServer {
switch m.Page {
case pageAddServer:
footerStr = footerHint("tab/↑↓", "navigate") +
footerSep() +
footerHint("enter", "confirm") +
footerSep() +
footerHint("ctrl+v", "paste") +
footerSep() +
footerHint("esc", "back")
} else {
case pageSelectServer:
footerStr = footerHint("↑↓", "navigate") +
footerSep() +
footerHint("enter", "connect") +
footerSep() +
footerHint("esc", "back")
default:
footerStr = footerHint("↑↓", "navigate") +
footerSep() +
footerHint("enter", "select") +
@@ -117,7 +130,23 @@ func (m TUIInterface) viewMenu() string {
return menu
}
func (m TUIInterface) viewAddServer() string {
func (m TUIInterface) viewSelectServer() string {
if len(m.ServerNames) == 0 {
return styles.StatusWarnStyle.Render("⚠ No servers configured.")
}
var rows []string
for i, name := range m.ServerNames {
srv := m.Servers[name]
detail := srv.User + "@" + srv.Host
if srv.Port != "" {
detail += ":" + srv.Port
}
row := styles.ServerRowStyle(i == m.Selected, name, detail)
rows = append(rows, row)
}
return lipgloss.JoinVertical(lipgloss.Left, rows...)
}
f := m.Form
labels := []string{"Name", "Host", "User", "Private Key Path", "Port"}
required := []bool{true, true, true, true, false}