session.go 7.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280
  1. package dialog
  2. import (
  3. "context"
  4. "strings"
  5. "slices"
  6. tea "github.com/charmbracelet/bubbletea/v2"
  7. "github.com/muesli/reflow/truncate"
  8. "github.com/sst/opencode-sdk-go"
  9. "github.com/sst/opencode/internal/app"
  10. "github.com/sst/opencode/internal/components/list"
  11. "github.com/sst/opencode/internal/components/modal"
  12. "github.com/sst/opencode/internal/components/toast"
  13. "github.com/sst/opencode/internal/layout"
  14. "github.com/sst/opencode/internal/styles"
  15. "github.com/sst/opencode/internal/theme"
  16. "github.com/sst/opencode/internal/util"
  17. )
  18. // SessionDialog interface for the session switching dialog
  19. type SessionDialog interface {
  20. layout.Modal
  21. }
  22. // sessionItem is a custom list item for sessions that can show delete confirmation
  23. type sessionItem struct {
  24. title string
  25. isDeleteConfirming bool
  26. isCurrentSession bool
  27. }
  28. func (s sessionItem) Render(
  29. selected bool,
  30. width int,
  31. isFirstInViewport bool,
  32. baseStyle styles.Style,
  33. ) string {
  34. t := theme.CurrentTheme()
  35. var text string
  36. if s.isDeleteConfirming {
  37. text = "Press again to confirm delete"
  38. } else {
  39. if s.isCurrentSession {
  40. text = "● " + s.title
  41. } else {
  42. text = s.title
  43. }
  44. }
  45. truncatedStr := truncate.StringWithTail(text, uint(width-1), "...")
  46. var itemStyle styles.Style
  47. if selected {
  48. if s.isDeleteConfirming {
  49. // Red background for delete confirmation
  50. itemStyle = baseStyle.
  51. Background(t.Error()).
  52. Foreground(t.BackgroundElement()).
  53. Width(width).
  54. PaddingLeft(1)
  55. } else if s.isCurrentSession {
  56. // Different style for current session when selected
  57. itemStyle = baseStyle.
  58. Background(t.Primary()).
  59. Foreground(t.BackgroundElement()).
  60. Width(width).
  61. PaddingLeft(1).
  62. Bold(true)
  63. } else {
  64. // Normal selection
  65. itemStyle = baseStyle.
  66. Background(t.Primary()).
  67. Foreground(t.BackgroundElement()).
  68. Width(width).
  69. PaddingLeft(1)
  70. }
  71. } else {
  72. if s.isDeleteConfirming {
  73. // Red text for delete confirmation when not selected
  74. itemStyle = baseStyle.
  75. Foreground(t.Error()).
  76. PaddingLeft(1)
  77. } else if s.isCurrentSession {
  78. // Highlight current session when not selected
  79. itemStyle = baseStyle.
  80. Foreground(t.Primary()).
  81. PaddingLeft(1).
  82. Bold(true)
  83. } else {
  84. itemStyle = baseStyle.
  85. PaddingLeft(1)
  86. }
  87. }
  88. return itemStyle.Render(truncatedStr)
  89. }
  90. func (s sessionItem) Selectable() bool {
  91. return true
  92. }
  93. type sessionDialog struct {
  94. width int
  95. height int
  96. modal *modal.Modal
  97. sessions []opencode.Session
  98. list list.List[sessionItem]
  99. app *app.App
  100. deleteConfirmation int // -1 means no confirmation, >= 0 means confirming deletion of session at this index
  101. }
  102. func (s *sessionDialog) Init() tea.Cmd {
  103. return nil
  104. }
  105. func (s *sessionDialog) Update(msg tea.Msg) (tea.Model, tea.Cmd) {
  106. switch msg := msg.(type) {
  107. case tea.WindowSizeMsg:
  108. s.width = msg.Width
  109. s.height = msg.Height
  110. s.list.SetMaxWidth(layout.Current.Container.Width - 12)
  111. case tea.KeyPressMsg:
  112. switch msg.String() {
  113. case "enter":
  114. if s.deleteConfirmation >= 0 {
  115. s.deleteConfirmation = -1
  116. s.updateListItems()
  117. return s, nil
  118. }
  119. if _, idx := s.list.GetSelectedItem(); idx >= 0 && idx < len(s.sessions) {
  120. selectedSession := s.sessions[idx]
  121. return s, tea.Sequence(
  122. util.CmdHandler(modal.CloseModalMsg{}),
  123. util.CmdHandler(app.SessionSelectedMsg(&selectedSession)),
  124. )
  125. }
  126. case "n":
  127. return s, tea.Sequence(
  128. util.CmdHandler(modal.CloseModalMsg{}),
  129. util.CmdHandler(app.SessionClearedMsg{}),
  130. )
  131. case "x", "delete", "backspace":
  132. if _, idx := s.list.GetSelectedItem(); idx >= 0 && idx < len(s.sessions) {
  133. if s.deleteConfirmation == idx {
  134. // Second press - actually delete the session
  135. sessionToDelete := s.sessions[idx]
  136. return s, tea.Sequence(
  137. func() tea.Msg {
  138. s.sessions = slices.Delete(s.sessions, idx, idx+1)
  139. s.deleteConfirmation = -1
  140. s.updateListItems()
  141. return nil
  142. },
  143. s.deleteSession(sessionToDelete.ID),
  144. )
  145. } else {
  146. // First press - enter delete confirmation mode
  147. s.deleteConfirmation = idx
  148. s.updateListItems()
  149. return s, nil
  150. }
  151. }
  152. case "esc":
  153. if s.deleteConfirmation >= 0 {
  154. s.deleteConfirmation = -1
  155. s.updateListItems()
  156. return s, nil
  157. }
  158. }
  159. }
  160. var cmd tea.Cmd
  161. listModel, cmd := s.list.Update(msg)
  162. s.list = listModel.(list.List[sessionItem])
  163. return s, cmd
  164. }
  165. func (s *sessionDialog) Render(background string) string {
  166. listView := s.list.View()
  167. t := theme.CurrentTheme()
  168. keyStyle := styles.NewStyle().Foreground(t.Text()).Background(t.BackgroundPanel()).Render
  169. mutedStyle := styles.NewStyle().Foreground(t.TextMuted()).Background(t.BackgroundPanel()).Render
  170. leftHelp := keyStyle("n") + mutedStyle(" new session")
  171. rightHelp := keyStyle("x/del") + mutedStyle(" delete session")
  172. bgColor := t.BackgroundPanel()
  173. helpText := layout.Render(layout.FlexOptions{
  174. Direction: layout.Row,
  175. Justify: layout.JustifySpaceBetween,
  176. Width: layout.Current.Container.Width - 14,
  177. Background: &bgColor,
  178. }, layout.FlexItem{View: leftHelp}, layout.FlexItem{View: rightHelp})
  179. helpText = styles.NewStyle().PaddingLeft(1).PaddingTop(1).Render(helpText)
  180. content := strings.Join([]string{listView, helpText}, "\n")
  181. return s.modal.Render(content, background)
  182. }
  183. func (s *sessionDialog) updateListItems() {
  184. _, currentIdx := s.list.GetSelectedItem()
  185. var items []sessionItem
  186. for i, sess := range s.sessions {
  187. item := sessionItem{
  188. title: sess.Title,
  189. isDeleteConfirming: s.deleteConfirmation == i,
  190. isCurrentSession: s.app.Session != nil && s.app.Session.ID == sess.ID,
  191. }
  192. items = append(items, item)
  193. }
  194. s.list.SetItems(items)
  195. s.list.SetSelectedIndex(currentIdx)
  196. }
  197. func (s *sessionDialog) deleteSession(sessionID string) tea.Cmd {
  198. return func() tea.Msg {
  199. ctx := context.Background()
  200. if err := s.app.DeleteSession(ctx, sessionID); err != nil {
  201. return toast.NewErrorToast("Failed to delete session: " + err.Error())()
  202. }
  203. return nil
  204. }
  205. }
  206. func (s *sessionDialog) Close() tea.Cmd {
  207. return nil
  208. }
  209. // NewSessionDialog creates a new session switching dialog
  210. func NewSessionDialog(app *app.App) SessionDialog {
  211. sessions, _ := app.ListSessions(context.Background())
  212. var filteredSessions []opencode.Session
  213. var items []sessionItem
  214. for _, sess := range sessions {
  215. if sess.ParentID != "" {
  216. continue
  217. }
  218. filteredSessions = append(filteredSessions, sess)
  219. items = append(items, sessionItem{
  220. title: sess.Title,
  221. isDeleteConfirming: false,
  222. isCurrentSession: app.Session != nil && app.Session.ID == sess.ID,
  223. })
  224. }
  225. listComponent := list.NewListComponent(
  226. list.WithItems(items),
  227. list.WithMaxVisibleHeight[sessionItem](10),
  228. list.WithFallbackMessage[sessionItem]("No sessions available"),
  229. list.WithAlphaNumericKeys[sessionItem](true),
  230. list.WithRenderFunc(
  231. func(item sessionItem, selected bool, width int, baseStyle styles.Style) string {
  232. return item.Render(selected, width, false, baseStyle)
  233. },
  234. ),
  235. list.WithSelectableFunc(func(item sessionItem) bool {
  236. return true
  237. }),
  238. )
  239. listComponent.SetMaxWidth(layout.Current.Container.Width - 12)
  240. return &sessionDialog{
  241. sessions: filteredSessions,
  242. list: listComponent,
  243. app: app,
  244. deleteConfirmation: -1,
  245. modal: modal.New(
  246. modal.WithTitle("Switch Session"),
  247. modal.WithMaxWidth(layout.Current.Container.Width-8),
  248. ),
  249. }
  250. }