Files
mail/model.go
titor c9b77feabe feat: 收件箱功能新增按回车查看详情面板
- 添加邮件详情面板显示(主题、发件人、收件人、抄送、账户、时间、正文)
- 优化邮件列表卡片样式,增加选中高亮效果
- 窗口宽度 >= 80 时启用双面板布局,左侧列表右侧详情
- 简化依赖包,从 charm.land 使用统一导入路径
- 删除未使用的 golangci/goreleaser 配置文件
2026-04-10 04:41:22 +08:00

408 lines
9.6 KiB
Go

package main
import (
"os"
"strings"
"time"
"charm.land/bubbles/v2/filepicker"
"charm.land/bubbles/v2/help"
"charm.land/bubbles/v2/key"
"charm.land/bubbles/v2/list"
"charm.land/bubbles/v2/spinner"
"charm.land/bubbles/v2/textarea"
"charm.land/bubbles/v2/textinput"
tea "charm.land/bubbletea/v2"
"github.com/charmbracelet/x/exp/ordered"
"github.com/resendlabs/resend-go"
)
// State is the current state of the application.
type State int
const (
editingFrom State = iota
editingTo
editingCc
editingBcc
editingSubject
editingBody
editingAttachments
hoveringSendButton
pickingFile
sendingEmail
)
// DeliveryMethod is the method of delivery for the email.
type DeliveryMethod int
const (
// None is the default delivery method.
None DeliveryMethod = iota
// Resend uses https://resend.com to send an email.
Resend
// SMTP uses an SMTP server to send an email.
SMTP
// Unknown is set when the user has not chosen a single delivery method.
// i.e. multiple delivery methods are set.
Unknown
)
// Model is Pop's application model.
type Model struct {
// state represents the current state of the application.
state State
// DeliveryMethod is whether we are using DeliveryMethod or Resend.
DeliveryMethod DeliveryMethod
// From represents the sender's email address.
From textinput.Model
// To represents the recipient's email address.
// This can be a comma-separated list of addresses.
To textinput.Model
// Subject represents the email's subject.
Subject textinput.Model
// Body represents the email's body.
// This can be written in markdown and will be converted to HTML.
Body textarea.Model
// Attachments represents the email's attachments.
// This is a list of file paths which are picked with a filepicker.
Attachments list.Model
showCc bool
Cc textinput.Model
Bcc textinput.Model
// filepicker is used to pick file attachments.
filepicker filepicker.Model
loadingSpinner spinner.Model
help help.Model
keymap KeyMap
quitting bool
abort bool
err error
}
// NewModel returns a new model for the application.
func NewModel(defaults resend.SendEmailRequest, deliveryMethod DeliveryMethod) Model {
from := textinput.New()
from.Prompt = "From "
from.Placeholder = "me@example.com"
from.SetValue(defaults.From)
to := textinput.New()
to.Prompt = "To "
to.Placeholder = "you@example.com"
to.SetValue(strings.Join(defaults.To, ToSeparator))
cc := textinput.New()
cc.Prompt = "Cc "
cc.Placeholder = "cc@example.com"
cc.SetValue(strings.Join(defaults.Cc, ToSeparator))
bcc := textinput.New()
bcc.Prompt = "Bcc "
bcc.Placeholder = "bcc@example.com"
bcc.SetValue(strings.Join(defaults.Bcc, ToSeparator))
subject := textinput.New()
subject.Prompt = "Subject "
subject.Placeholder = "Hello!"
subject.SetValue(defaults.Subject)
body := textarea.New()
body.Placeholder = "# Email"
body.SetValue(defaults.Text)
body.CharLimit = 4000
body.Blur()
// Decide which input to focus.
var state State
switch {
case defaults.From == "":
state = editingFrom
case len(defaults.To) == 0:
state = editingTo
case defaults.Subject == "":
state = editingSubject
case defaults.Text == "":
state = editingBody
}
attachments := list.New([]list.Item{}, attachmentDelegate{}, 0, 3)
attachments.DisableQuitKeybindings()
attachments.SetShowTitle(true)
attachments.Title = "Attachments"
attachments.SetShowHelp(false)
attachments.SetShowStatusBar(false)
attachments.SetStatusBarItemName("attachment", "attachments")
attachments.SetShowPagination(false)
for _, a := range defaults.Attachments {
attachments.InsertItem(0, attachment(a.Filename))
}
picker := filepicker.New()
picker.CurrentDirectory, _ = os.UserHomeDir()
loadingSpinner := spinner.New()
loadingSpinner.Spinner = spinner.Dot
m := Model{
state: state,
From: from,
To: to,
showCc: len(cc.Value()) > 0 || len(bcc.Value()) > 0,
Cc: cc,
Bcc: bcc,
Subject: subject,
Body: body,
Attachments: attachments,
filepicker: picker,
help: help.New(),
keymap: DefaultKeybinds(),
loadingSpinner: loadingSpinner,
DeliveryMethod: deliveryMethod,
}
m.focusActiveInput()
return m
}
// Init initializes the model.
func (m Model) Init() tea.Cmd {
return nil
}
type clearErrMsg struct{}
func clearErrAfter(d time.Duration) tea.Cmd {
return tea.Tick(d, func(t time.Time) tea.Msg {
return clearErrMsg{}
})
}
// Update is the update loop for the model.
func (m Model) Update(msg tea.Msg) (tea.Model, tea.Cmd) {
switch msg := msg.(type) {
case sendEmailSuccessMsg:
m.quitting = true
return m, tea.Quit
case sendEmailFailureMsg:
m.blurInputs()
m.state = editingFrom
m.focusActiveInput()
m.err = msg
return m, clearErrAfter(10 * time.Second)
case clearErrMsg:
m.err = nil
case tea.KeyMsg:
switch {
case key.Matches(msg, m.keymap.NextInput):
m.blurInputs()
switch m.state {
case editingFrom:
m.state = editingTo
m.To.Focus()
case editingTo:
if m.showCc {
m.state = editingCc
} else {
m.state = editingSubject
}
case editingCc:
m.state = editingBcc
case editingBcc:
m.state = editingSubject
case editingSubject:
m.state = editingBody
case editingBody:
m.state = editingAttachments
case editingAttachments:
m.state = hoveringSendButton
case hoveringSendButton:
m.state = editingFrom
}
m.focusActiveInput()
case key.Matches(msg, m.keymap.PrevInput):
m.blurInputs()
switch m.state {
case editingFrom:
m.state = hoveringSendButton
case editingTo:
m.state = editingFrom
case editingCc:
m.state = editingTo
case editingBcc:
m.state = editingCc
case editingSubject:
if m.showCc {
m.state = editingBcc
} else {
m.state = editingTo
}
case editingBody:
m.state = editingSubject
case editingAttachments:
m.state = editingBody
case hoveringSendButton:
m.state = editingAttachments
}
m.focusActiveInput()
case key.Matches(msg, m.keymap.Back):
m.state = editingAttachments
m.updateKeymap()
return m, nil
case key.Matches(msg, m.keymap.Send):
m.state = sendingEmail
return m, tea.Batch(
m.loadingSpinner.Tick,
m.sendEmailCmd(),
)
case key.Matches(msg, m.keymap.Attach):
m.state = pickingFile
return m, m.filepicker.Init()
case key.Matches(msg, m.keymap.Unattach):
m.Attachments.RemoveItem(m.Attachments.Index())
m.Attachments.SetHeight(ordered.Max(len(m.Attachments.Items()), 1) + 2)
case key.Matches(msg, m.keymap.Quit):
m.quitting = true
m.abort = true
return m, tea.Quit
}
}
m.updateKeymap()
var cmds []tea.Cmd
var cmd tea.Cmd
m.From, cmd = m.From.Update(msg)
cmds = append(cmds, cmd)
m.To, cmd = m.To.Update(msg)
cmds = append(cmds, cmd)
if m.showCc {
m.Cc, cmd = m.Cc.Update(msg)
cmds = append(cmds, cmd)
m.Bcc, cmd = m.Bcc.Update(msg)
cmds = append(cmds, cmd)
}
m.Subject, cmd = m.Subject.Update(msg)
cmds = append(cmds, cmd)
m.Body, cmd = m.Body.Update(msg)
cmds = append(cmds, cmd)
m.filepicker, cmd = m.filepicker.Update(msg)
cmds = append(cmds, cmd)
switch m.state {
case pickingFile:
if didSelect, path := m.filepicker.DidSelectFile(msg); didSelect {
m.Attachments.InsertItem(0, attachment(path))
m.Attachments.SetHeight(len(m.Attachments.Items()) + 2)
m.state = editingAttachments
m.updateKeymap()
}
case editingAttachments:
m.Attachments, cmd = m.Attachments.Update(msg)
cmds = append(cmds, cmd)
case sendingEmail:
m.loadingSpinner, cmd = m.loadingSpinner.Update(msg)
cmds = append(cmds, cmd)
}
m.help, cmd = m.help.Update(msg)
cmds = append(cmds, cmd)
return m, tea.Batch(cmds...)
}
func (m *Model) blurInputs() {
m.From.Blur()
m.To.Blur()
m.Subject.Blur()
m.Body.Blur()
if m.showCc {
m.Cc.Blur()
m.Bcc.Blur()
}
m.Attachments.SetDelegate(attachmentDelegate{false})
}
func (m *Model) focusActiveInput() {
switch m.state {
case editingFrom:
m.From.Focus()
m.From.CursorEnd()
case editingTo:
m.To.Focus()
m.To.CursorEnd()
case editingCc:
m.Cc.Focus()
m.Cc.CursorEnd()
case editingBcc:
m.Bcc.Focus()
m.Bcc.CursorEnd()
case editingSubject:
m.Subject.Focus()
m.Subject.CursorEnd()
case editingBody:
m.Body.Focus()
m.Body.CursorEnd()
case editingAttachments:
m.Attachments.SetDelegate(attachmentDelegate{true})
}
}
// View displays the application.
func (m Model) View() tea.View {
if m.quitting {
return tea.NewView("")
}
switch m.state {
case pickingFile:
return tea.NewView("\n" + activeLabelStyle.Render("Attachments") + " " + commentStyle.Render(m.filepicker.CurrentDirectory) +
"\n\n" + m.filepicker.View())
case sendingEmail:
return tea.NewView("\n " + m.loadingSpinner.View() + "Sending email")
}
var s strings.Builder
s.WriteString(m.From.View())
s.WriteString("\n")
s.WriteString(m.To.View())
s.WriteString("\n")
if m.showCc {
s.WriteString(m.Cc.View())
s.WriteString("\n")
s.WriteString(m.Bcc.View())
s.WriteString("\n")
}
s.WriteString(m.Subject.View())
s.WriteString("\n\n")
s.WriteString(m.Body.View())
s.WriteString("\n\n")
s.WriteString(m.Attachments.View())
s.WriteString("\n")
if m.state == hoveringSendButton && m.canSend() {
s.WriteString(sendButtonActiveStyle.Render("Send"))
} else if m.state == hoveringSendButton {
s.WriteString(sendButtonInactiveStyle.Render("Send"))
} else {
s.WriteString(sendButtonStyle.Render("Send"))
}
s.WriteString("\n\n")
s.WriteString(m.help.View(m.keymap))
if m.err != nil {
s.WriteString("\n\n")
s.WriteString(errorStyle.Render(m.err.Error()))
}
return tea.NewView(paddedStyle.Render(s.String()))
}