2019-01-13 18:39:06 +01:00
|
|
|
package widgets
|
|
|
|
|
|
|
|
import (
|
2019-01-13 19:33:43 +01:00
|
|
|
"fmt"
|
2019-01-13 19:03:28 +01:00
|
|
|
"log"
|
2019-01-13 19:25:56 +01:00
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/gdamore/tcell"
|
2019-01-13 19:03:28 +01:00
|
|
|
|
2019-01-13 18:39:06 +01:00
|
|
|
"git.sr.ht/~sircmpwn/aerc2/config"
|
|
|
|
"git.sr.ht/~sircmpwn/aerc2/lib/ui"
|
2019-01-13 19:03:28 +01:00
|
|
|
"git.sr.ht/~sircmpwn/aerc2/worker"
|
|
|
|
"git.sr.ht/~sircmpwn/aerc2/worker/types"
|
2019-01-13 18:39:06 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
type AccountView struct {
|
2019-03-15 06:46:14 +01:00
|
|
|
acct *config.AccountConfig
|
|
|
|
conf *config.AercConfig
|
2019-01-13 20:25:46 +01:00
|
|
|
dirlist *DirectoryList
|
2019-01-13 18:39:06 +01:00
|
|
|
grid *ui.Grid
|
2019-01-13 19:03:28 +01:00
|
|
|
logger *log.Logger
|
2019-01-13 19:33:43 +01:00
|
|
|
interactive ui.Interactive
|
2019-01-13 18:39:06 +01:00
|
|
|
onInvalidate func(d ui.Drawable)
|
2019-01-14 14:14:03 +01:00
|
|
|
runCmd func(cmd string) error
|
2019-03-15 02:37:00 +01:00
|
|
|
msglist *MessageList
|
2019-03-11 04:45:00 +01:00
|
|
|
msgStores map[string]*MessageStore
|
2019-03-15 06:46:14 +01:00
|
|
|
pendingKeys []config.KeyStroke
|
2019-01-13 19:33:43 +01:00
|
|
|
statusline *StatusLine
|
|
|
|
statusbar *ui.Stack
|
2019-01-13 19:03:28 +01:00
|
|
|
worker *types.Worker
|
2019-01-13 18:39:06 +01:00
|
|
|
}
|
|
|
|
|
2019-03-15 06:46:14 +01:00
|
|
|
func NewAccountView(conf *config.AercConfig, acct *config.AccountConfig,
|
2019-01-14 14:14:03 +01:00
|
|
|
logger *log.Logger, runCmd func(cmd string) error) *AccountView {
|
2019-01-13 19:03:28 +01:00
|
|
|
|
2019-01-13 19:33:43 +01:00
|
|
|
statusbar := ui.NewStack()
|
|
|
|
statusline := NewStatusLine()
|
|
|
|
statusbar.Push(statusline)
|
2019-01-13 18:39:06 +01:00
|
|
|
|
|
|
|
grid := ui.NewGrid().Rows([]ui.GridSpec{
|
|
|
|
{ui.SIZE_WEIGHT, 1},
|
|
|
|
{ui.SIZE_EXACT, 1},
|
|
|
|
}).Columns([]ui.GridSpec{
|
|
|
|
{ui.SIZE_EXACT, 20},
|
|
|
|
{ui.SIZE_WEIGHT, 1},
|
|
|
|
})
|
2019-01-13 19:33:43 +01:00
|
|
|
grid.AddChild(statusbar).At(1, 1)
|
2019-01-13 19:25:56 +01:00
|
|
|
|
2019-03-15 06:46:14 +01:00
|
|
|
worker, err := worker.NewWorker(acct.Source, logger)
|
2019-01-13 19:25:56 +01:00
|
|
|
if err != nil {
|
2019-01-14 02:06:18 +01:00
|
|
|
statusline.Set(fmt.Sprintf("%s", err))
|
2019-01-13 19:33:43 +01:00
|
|
|
return &AccountView{
|
2019-03-15 06:46:14 +01:00
|
|
|
acct: acct,
|
2019-01-13 19:33:43 +01:00
|
|
|
grid: grid,
|
|
|
|
logger: logger,
|
|
|
|
statusline: statusline,
|
|
|
|
}
|
2019-01-13 19:25:56 +01:00
|
|
|
}
|
2019-01-13 19:03:28 +01:00
|
|
|
|
2019-03-15 06:46:14 +01:00
|
|
|
dirlist := NewDirectoryList(acct, logger, worker)
|
2019-01-13 20:25:46 +01:00
|
|
|
grid.AddChild(ui.NewBordered(dirlist, ui.BORDER_RIGHT)).Span(2, 1)
|
|
|
|
|
2019-03-15 02:51:29 +01:00
|
|
|
msglist := NewMessageList(logger)
|
2019-03-15 02:37:00 +01:00
|
|
|
grid.AddChild(msglist).At(0, 1)
|
|
|
|
|
2019-03-15 06:46:14 +01:00
|
|
|
view := &AccountView{
|
|
|
|
acct: acct,
|
2019-01-13 19:33:43 +01:00
|
|
|
conf: conf,
|
2019-01-13 20:25:46 +01:00
|
|
|
dirlist: dirlist,
|
2019-01-13 19:33:43 +01:00
|
|
|
grid: grid,
|
|
|
|
logger: logger,
|
2019-03-15 02:37:00 +01:00
|
|
|
msglist: msglist,
|
2019-03-11 04:45:00 +01:00
|
|
|
msgStores: make(map[string]*MessageStore),
|
2019-01-14 14:14:03 +01:00
|
|
|
runCmd: runCmd,
|
2019-01-13 19:33:43 +01:00
|
|
|
statusbar: statusbar,
|
2019-03-11 04:45:00 +01:00
|
|
|
statusline: statusline,
|
2019-01-13 19:33:43 +01:00
|
|
|
worker: worker,
|
2019-01-13 19:03:28 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
go worker.Backend.Run()
|
|
|
|
go func() {
|
|
|
|
for {
|
|
|
|
msg := <-worker.Messages
|
|
|
|
msg = worker.ProcessMessage(msg)
|
2019-03-15 06:46:14 +01:00
|
|
|
view.onMessage(msg)
|
2019-01-13 19:03:28 +01:00
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
2019-03-15 06:46:14 +01:00
|
|
|
worker.PostAction(&types.Configure{Config: acct}, nil)
|
|
|
|
worker.PostAction(&types.Connect{}, view.connected)
|
2019-01-14 01:41:21 +01:00
|
|
|
statusline.Set("Connecting...")
|
2019-01-13 19:03:28 +01:00
|
|
|
|
2019-03-15 06:46:14 +01:00
|
|
|
return view
|
2019-01-13 19:03:28 +01:00
|
|
|
}
|
|
|
|
|
2019-03-15 03:34:34 +01:00
|
|
|
func (acct *AccountView) Name() string {
|
2019-03-15 06:46:14 +01:00
|
|
|
return acct.acct.Name
|
2019-03-15 03:34:34 +01:00
|
|
|
}
|
|
|
|
|
2019-02-10 22:46:13 +01:00
|
|
|
func (acct *AccountView) Children() []ui.Drawable {
|
2019-01-20 21:08:30 +01:00
|
|
|
return acct.grid.Children()
|
|
|
|
}
|
|
|
|
|
2019-01-13 18:39:06 +01:00
|
|
|
func (acct *AccountView) OnInvalidate(onInvalidate func(d ui.Drawable)) {
|
2019-01-13 19:25:56 +01:00
|
|
|
acct.grid.OnInvalidate(func(_ ui.Drawable) {
|
|
|
|
onInvalidate(acct)
|
|
|
|
})
|
2019-01-13 18:39:06 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
func (acct *AccountView) Invalidate() {
|
|
|
|
acct.grid.Invalidate()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (acct *AccountView) Draw(ctx *ui.Context) {
|
|
|
|
acct.grid.Draw(ctx)
|
|
|
|
}
|
2019-01-13 19:33:43 +01:00
|
|
|
|
2019-03-15 06:46:14 +01:00
|
|
|
func (acct *AccountView) beginExCommand() {
|
|
|
|
exline := NewExLine(func(command string) {
|
|
|
|
err := acct.runCmd(command)
|
|
|
|
if err != nil {
|
|
|
|
acct.statusline.Push(" "+err.Error(), 10*time.Second).
|
|
|
|
Color(tcell.ColorRed, tcell.ColorWhite)
|
|
|
|
}
|
|
|
|
acct.statusbar.Pop()
|
|
|
|
acct.interactive = nil
|
|
|
|
}, func() {
|
|
|
|
acct.statusbar.Pop()
|
|
|
|
acct.interactive = nil
|
|
|
|
})
|
|
|
|
acct.interactive = exline
|
|
|
|
acct.statusbar.Push(exline)
|
|
|
|
}
|
|
|
|
|
2019-01-13 19:33:43 +01:00
|
|
|
func (acct *AccountView) Event(event tcell.Event) bool {
|
|
|
|
if acct.interactive != nil {
|
|
|
|
return acct.interactive.Event(event)
|
|
|
|
}
|
2019-03-15 06:46:14 +01:00
|
|
|
|
2019-01-13 19:33:43 +01:00
|
|
|
switch event := event.(type) {
|
|
|
|
case *tcell.EventKey:
|
2019-03-15 06:46:14 +01:00
|
|
|
acct.pendingKeys = append(acct.pendingKeys, config.KeyStroke{
|
|
|
|
Key: event.Key(),
|
|
|
|
Rune: event.Rune(),
|
|
|
|
})
|
|
|
|
result, output := acct.conf.Lbinds.GetBinding(acct.pendingKeys)
|
|
|
|
switch result {
|
|
|
|
case config.BINDING_FOUND:
|
|
|
|
acct.pendingKeys = []config.KeyStroke{}
|
|
|
|
for _, stroke := range output {
|
|
|
|
simulated := tcell.NewEventKey(
|
|
|
|
stroke.Key, stroke.Rune, tcell.ModNone)
|
|
|
|
acct.Event(simulated)
|
|
|
|
}
|
|
|
|
case config.BINDING_INCOMPLETE:
|
|
|
|
return false
|
|
|
|
case config.BINDING_NOT_FOUND:
|
|
|
|
acct.pendingKeys = []config.KeyStroke{}
|
|
|
|
if event.Rune() == ':' {
|
|
|
|
acct.beginExCommand()
|
|
|
|
return true
|
|
|
|
}
|
2019-01-13 19:33:43 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|
2019-01-13 20:25:46 +01:00
|
|
|
|
|
|
|
func (acct *AccountView) connected(msg types.WorkerMessage) {
|
|
|
|
switch msg := msg.(type) {
|
|
|
|
case *types.Done:
|
2019-01-13 21:32:52 +01:00
|
|
|
acct.statusline.Set("Listing mailboxes...")
|
|
|
|
acct.logger.Println("Listing mailboxes...")
|
|
|
|
acct.dirlist.UpdateList(func(dirs []string) {
|
|
|
|
var dir string
|
|
|
|
for _, _dir := range dirs {
|
|
|
|
if _dir == "INBOX" {
|
|
|
|
dir = _dir
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if dir == "" {
|
|
|
|
dir = dirs[0]
|
|
|
|
}
|
|
|
|
acct.dirlist.Select(dir)
|
|
|
|
acct.logger.Println("Connected.")
|
|
|
|
acct.statusline.Set("Connected.")
|
|
|
|
})
|
2019-01-13 20:25:46 +01:00
|
|
|
case *types.CertificateApprovalRequest:
|
|
|
|
// TODO: Ask the user
|
|
|
|
acct.worker.PostAction(&types.ApproveCertificate{
|
|
|
|
Message: types.RespondTo(msg),
|
|
|
|
Approved: true,
|
|
|
|
}, acct.connected)
|
|
|
|
}
|
|
|
|
}
|
2019-03-11 02:15:24 +01:00
|
|
|
|
|
|
|
func (acct *AccountView) Directories() *DirectoryList {
|
|
|
|
return acct.dirlist
|
|
|
|
}
|
|
|
|
|
2019-03-15 04:41:25 +01:00
|
|
|
func (acct *AccountView) Messages() *MessageList {
|
|
|
|
return acct.msglist
|
|
|
|
}
|
|
|
|
|
2019-03-11 02:15:24 +01:00
|
|
|
func (acct *AccountView) onMessage(msg types.WorkerMessage) {
|
2019-03-11 04:45:00 +01:00
|
|
|
switch msg := msg.(type) {
|
|
|
|
case *types.Done:
|
|
|
|
switch msg.InResponseTo().(type) {
|
|
|
|
case *types.OpenDirectory:
|
2019-03-15 03:41:43 +01:00
|
|
|
if store, ok := acct.msgStores[acct.dirlist.selected]; ok {
|
|
|
|
// If we've opened this dir before, we can re-render it from
|
|
|
|
// memory while we wait for the update and the UI feels
|
|
|
|
// snappier. If not, we'll unset the store and show the spinner
|
|
|
|
// while we download the UID list.
|
|
|
|
acct.msglist.SetStore(store)
|
|
|
|
} else {
|
|
|
|
acct.msglist.SetStore(nil)
|
|
|
|
}
|
2019-03-11 04:45:00 +01:00
|
|
|
acct.worker.PostAction(&types.FetchDirectoryContents{},
|
|
|
|
func(msg types.WorkerMessage) {
|
2019-03-15 02:37:00 +01:00
|
|
|
store := acct.msgStores[acct.dirlist.selected]
|
|
|
|
acct.msglist.SetStore(store)
|
2019-03-11 04:45:00 +01:00
|
|
|
})
|
|
|
|
}
|
|
|
|
case *types.DirectoryInfo:
|
|
|
|
if store, ok := acct.msgStores[msg.Name]; ok {
|
|
|
|
store.Update(msg)
|
|
|
|
} else {
|
2019-03-15 02:51:29 +01:00
|
|
|
acct.msgStores[msg.Name] = NewMessageStore(acct.worker, msg)
|
2019-03-11 04:45:00 +01:00
|
|
|
}
|
|
|
|
case *types.DirectoryContents:
|
|
|
|
store := acct.msgStores[acct.dirlist.selected]
|
|
|
|
store.Update(msg)
|
|
|
|
case *types.MessageInfo:
|
|
|
|
store := acct.msgStores[acct.dirlist.selected]
|
|
|
|
store.Update(msg)
|
|
|
|
case *types.Error:
|
|
|
|
acct.logger.Printf("%v", msg.Error)
|
|
|
|
acct.statusline.Set(fmt.Sprintf("%v", msg.Error)).
|
|
|
|
Color(tcell.ColorRed, tcell.ColorDefault)
|
|
|
|
}
|
2019-03-11 02:15:24 +01:00
|
|
|
}
|