add:addserver
This commit is contained in:
3
internal/pages/select_server.go
Normal file
3
internal/pages/select_server.go
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
package pages
|
||||||
|
|
||||||
|
type SelectServerPageMsg struct{}
|
||||||
@@ -100,6 +100,31 @@ var (
|
|||||||
FooterDescStyle = lipgloss.NewStyle().Foreground(lipgloss.Color("243"))
|
FooterDescStyle = lipgloss.NewStyle().Foreground(lipgloss.Color("243"))
|
||||||
)
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
serverRowBase = lipgloss.NewStyle().
|
||||||
|
PaddingLeft(2).
|
||||||
|
PaddingTop(0).
|
||||||
|
Width(44)
|
||||||
|
|
||||||
|
serverRowBaseActive = lipgloss.NewStyle().
|
||||||
|
PaddingLeft(2).
|
||||||
|
Width(44)
|
||||||
|
|
||||||
|
serverRowNameStyle = lipgloss.NewStyle().
|
||||||
|
Bold(true).
|
||||||
|
Foreground(lipgloss.Color("255"))
|
||||||
|
|
||||||
|
serverRowNameActiveStyle = lipgloss.NewStyle().
|
||||||
|
Bold(true).
|
||||||
|
Foreground(lipgloss.Color("75"))
|
||||||
|
|
||||||
|
serverRowDetailStyle = lipgloss.NewStyle().
|
||||||
|
Foreground(lipgloss.Color("243"))
|
||||||
|
|
||||||
|
serverRowDetailActiveStyle = lipgloss.NewStyle().
|
||||||
|
Foreground(lipgloss.Color("117"))
|
||||||
|
)
|
||||||
|
|
||||||
func MenuItemStyle(active, disabled bool) lipgloss.Style {
|
func MenuItemStyle(active, disabled bool) lipgloss.Style {
|
||||||
switch {
|
switch {
|
||||||
case disabled:
|
case disabled:
|
||||||
@@ -130,3 +155,23 @@ func ButtonStyle(focused, enabled bool) lipgloss.Style {
|
|||||||
return buttonInactive
|
return buttonInactive
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// ServerRowStyle renders a two-line server list entry: bold name on top,
|
||||||
|
// dim "user@host[:port]" detail below.
|
||||||
|
func ServerRowStyle(active bool, name, detail string) string {
|
||||||
|
nameStyle := serverRowNameStyle
|
||||||
|
detailStyle := serverRowDetailStyle
|
||||||
|
base := serverRowBase
|
||||||
|
prefix := " "
|
||||||
|
if active {
|
||||||
|
nameStyle = serverRowNameActiveStyle
|
||||||
|
detailStyle = serverRowDetailActiveStyle
|
||||||
|
base = serverRowBaseActive
|
||||||
|
prefix = "▸ "
|
||||||
|
}
|
||||||
|
row := lipgloss.JoinVertical(lipgloss.Left,
|
||||||
|
nameStyle.Render(prefix+name),
|
||||||
|
detailStyle.Render(" "+detail),
|
||||||
|
)
|
||||||
|
return base.Render(row)
|
||||||
|
}
|
||||||
|
|||||||
@@ -11,22 +11,24 @@ const (
|
|||||||
pageHome page = iota
|
pageHome page = iota
|
||||||
pageConfig
|
pageConfig
|
||||||
pageAddServer
|
pageAddServer
|
||||||
|
pageSelectServer
|
||||||
)
|
)
|
||||||
|
|
||||||
type TUIInterface struct {
|
type TUIInterface struct {
|
||||||
Services *services.ServicesStore
|
Services *services.ServicesStore
|
||||||
Page page
|
Page page
|
||||||
MenuItems []pages.MenuItem
|
MenuItems []pages.MenuItem
|
||||||
Selected int
|
Selected int
|
||||||
Servers map[string]services.Server
|
Servers map[string]services.Server
|
||||||
NoServers bool
|
ServerNames []string // sorted, stable order for list rendering
|
||||||
InitErr error
|
NoServers bool
|
||||||
FlashMsg string
|
InitErr error
|
||||||
Form addServerForm
|
FlashMsg string
|
||||||
FormErr string // inline field error (e.g. duplicate name)
|
Form addServerForm
|
||||||
Quitting bool
|
FormErr string // inline field error (e.g. duplicate name)
|
||||||
WindowWidth int
|
Quitting bool
|
||||||
WindowHeight int
|
WindowWidth int
|
||||||
|
WindowHeight int
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewTUIInterface(store *services.ServicesStore) TUIInterface {
|
func NewTUIInterface(store *services.ServicesStore) TUIInterface {
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
package tui
|
package tui
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"sort"
|
||||||
"strings"
|
"strings"
|
||||||
"time"
|
"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.
|
// isDisabled reports whether a menu item is non-interactive given current state.
|
||||||
func (m TUIInterface) isDisabled(i int) bool {
|
func (m TUIInterface) isDisabled(i int) bool {
|
||||||
return m.MenuItems[i].RequiresServers && m.NoServers
|
return m.MenuItems[i].RequiresServers && m.NoServers
|
||||||
@@ -66,17 +77,24 @@ func (m TUIInterface) Update(msg tea.Msg) (tea.Model, tea.Cmd) {
|
|||||||
m.FormErr = ""
|
m.FormErr = ""
|
||||||
return m, nil
|
return m, nil
|
||||||
|
|
||||||
|
case pages.SelectServerPageMsg:
|
||||||
|
m.Page = pageSelectServer
|
||||||
|
m.Selected = 0
|
||||||
|
return m, nil
|
||||||
|
|
||||||
case configLoadedMsg:
|
case configLoadedMsg:
|
||||||
if msg.err != nil {
|
if msg.err != nil {
|
||||||
m.InitErr = msg.err
|
m.InitErr = msg.err
|
||||||
return m, nil
|
return m, nil
|
||||||
}
|
}
|
||||||
m.Servers = msg.servers
|
m.Servers = msg.servers
|
||||||
|
m.ServerNames = sortedServerNames(msg.servers)
|
||||||
m.NoServers = len(msg.servers) == 0
|
m.NoServers = len(msg.servers) == 0
|
||||||
return m, nil
|
return m, nil
|
||||||
|
|
||||||
case serverAddedMsg:
|
case serverAddedMsg:
|
||||||
m.Servers = msg.servers
|
m.Servers = msg.servers
|
||||||
|
m.ServerNames = sortedServerNames(msg.servers)
|
||||||
m.NoServers = len(msg.servers) == 0
|
m.NoServers = len(msg.servers) == 0
|
||||||
m.Page = pageConfig
|
m.Page = pageConfig
|
||||||
m.MenuItems = pages.ConfigMenuItems()
|
m.MenuItems = pages.ConfigMenuItems()
|
||||||
@@ -98,6 +116,10 @@ func (m TUIInterface) Update(msg tea.Msg) (tea.Model, tea.Cmd) {
|
|||||||
if m.Page == pageAddServer {
|
if m.Page == pageAddServer {
|
||||||
return m.updateAddServer(msg)
|
return m.updateAddServer(msg)
|
||||||
}
|
}
|
||||||
|
// server list has its own key handling
|
||||||
|
if m.Page == pageSelectServer {
|
||||||
|
return m.updateSelectServer(msg)
|
||||||
|
}
|
||||||
|
|
||||||
switch msg.String() {
|
switch msg.String() {
|
||||||
case "up", "k":
|
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{} }
|
return m, func() tea.Msg { return pages.HomePageMsg{} }
|
||||||
case "add":
|
case "add":
|
||||||
return m, func() tea.Msg { return pages.AddServerPageMsg{} }
|
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":
|
case "ctrl+c":
|
||||||
m.Quitting = true
|
m.Quitting = true
|
||||||
@@ -130,6 +154,16 @@ func (m TUIInterface) Update(msg tea.Msg) (tea.Model, tea.Cmd) {
|
|||||||
m.Quitting = true
|
m.Quitting = true
|
||||||
return m, tea.Quit
|
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
|
return m, nil
|
||||||
@@ -166,6 +200,13 @@ func (m TUIInterface) updateAddServer(msg tea.KeyPressMsg) (tea.Model, tea.Cmd)
|
|||||||
m.Quitting = true
|
m.Quitting = true
|
||||||
return m, tea.Quit
|
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":
|
case "esc":
|
||||||
return m, func() tea.Msg { return pages.ConfigPageMsg{} }
|
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
|
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) {
|
func (m TUIInterface) submitAddServer() (tea.Model, tea.Cmd) {
|
||||||
f := m.Form
|
f := m.Form
|
||||||
name := strings.TrimSpace(f.inputs[fieldName].Value())
|
name := strings.TrimSpace(f.inputs[fieldName].Value())
|
||||||
|
|||||||
@@ -23,6 +23,8 @@ func (m TUIInterface) subtitle() string {
|
|||||||
return "Configuration"
|
return "Configuration"
|
||||||
case pageAddServer:
|
case pageAddServer:
|
||||||
return "Add Server"
|
return "Add Server"
|
||||||
|
case pageSelectServer:
|
||||||
|
return "Select Server"
|
||||||
default:
|
default:
|
||||||
return "Secure file transfer"
|
return "Secure file transfer"
|
||||||
}
|
}
|
||||||
@@ -46,6 +48,8 @@ func (m TUIInterface) View() tea.View {
|
|||||||
switch m.Page {
|
switch m.Page {
|
||||||
case pageAddServer:
|
case pageAddServer:
|
||||||
body = m.viewAddServer()
|
body = m.viewAddServer()
|
||||||
|
case pageSelectServer:
|
||||||
|
body = m.viewSelectServer()
|
||||||
default:
|
default:
|
||||||
body = m.viewMenu()
|
body = m.viewMenu()
|
||||||
}
|
}
|
||||||
@@ -60,13 +64,22 @@ func (m TUIInterface) View() tea.View {
|
|||||||
)
|
)
|
||||||
|
|
||||||
var footerStr string
|
var footerStr string
|
||||||
if m.Page == pageAddServer {
|
switch m.Page {
|
||||||
|
case pageAddServer:
|
||||||
footerStr = footerHint("tab/↑↓", "navigate") +
|
footerStr = footerHint("tab/↑↓", "navigate") +
|
||||||
footerSep() +
|
footerSep() +
|
||||||
footerHint("enter", "confirm") +
|
footerHint("enter", "confirm") +
|
||||||
footerSep() +
|
footerSep() +
|
||||||
|
footerHint("ctrl+v", "paste") +
|
||||||
|
footerSep() +
|
||||||
footerHint("esc", "back")
|
footerHint("esc", "back")
|
||||||
} else {
|
case pageSelectServer:
|
||||||
|
footerStr = footerHint("↑↓", "navigate") +
|
||||||
|
footerSep() +
|
||||||
|
footerHint("enter", "connect") +
|
||||||
|
footerSep() +
|
||||||
|
footerHint("esc", "back")
|
||||||
|
default:
|
||||||
footerStr = footerHint("↑↓", "navigate") +
|
footerStr = footerHint("↑↓", "navigate") +
|
||||||
footerSep() +
|
footerSep() +
|
||||||
footerHint("enter", "select") +
|
footerHint("enter", "select") +
|
||||||
@@ -117,7 +130,23 @@ func (m TUIInterface) viewMenu() string {
|
|||||||
return menu
|
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
|
f := m.Form
|
||||||
labels := []string{"Name", "Host", "User", "Private Key Path", "Port"}
|
labels := []string{"Name", "Host", "User", "Private Key Path", "Port"}
|
||||||
required := []bool{true, true, true, true, false}
|
required := []bool{true, true, true, true, false}
|
||||||
|
|||||||
Reference in New Issue
Block a user