aerc/widgets/account.go

202 lines
4.7 KiB
Go
Raw Normal View History

package widgets
import (
2019-01-13 19:33:43 +01:00
"fmt"
2019-01-13 19:03:28 +01:00
"log"
"github.com/gdamore/tcell"
2019-01-13 19:03:28 +01:00
2019-05-18 02:57:10 +02:00
"git.sr.ht/~sircmpwn/aerc/config"
"git.sr.ht/~sircmpwn/aerc/lib"
"git.sr.ht/~sircmpwn/aerc/lib/ui"
"git.sr.ht/~sircmpwn/aerc/worker"
"git.sr.ht/~sircmpwn/aerc/worker/types"
)
type AccountView struct {
acct *config.AccountConfig
conf *config.AercConfig
dirlist *DirectoryList
grid *ui.Grid
host TabHost
logger *log.Logger
msglist *MessageList
msgStores map[string]*lib.MessageStore
worker *types.Worker
}
2019-03-15 06:46:14 +01:00
func NewAccountView(conf *config.AercConfig, acct *config.AccountConfig,
logger *log.Logger, host TabHost) *AccountView {
grid := ui.NewGrid().Rows([]ui.GridSpec{
{ui.SIZE_WEIGHT, 1},
}).Columns([]ui.GridSpec{
2019-03-16 01:40:28 +01:00
{ui.SIZE_EXACT, conf.Ui.SidebarWidth},
{ui.SIZE_WEIGHT, 1},
})
2019-03-15 06:46:14 +01:00
worker, err := worker.NewWorker(acct.Source, logger)
if err != nil {
host.SetStatus(fmt.Sprintf("%s: %s", acct.Name, err))
2019-01-13 19:33:43 +01:00
return &AccountView{
acct: acct,
grid: grid,
host: host,
logger: logger,
2019-01-13 19:33:43 +01:00
}
}
2019-01-13 19:03:28 +01:00
2019-03-15 06:46:14 +01:00
dirlist := NewDirectoryList(acct, logger, worker)
grid.AddChild(ui.NewBordered(dirlist, ui.BORDER_RIGHT))
2019-01-13 20:25:46 +01:00
msglist := NewMessageList(conf, logger)
grid.AddChild(msglist).At(0, 1)
2019-03-15 06:46:14 +01:00
view := &AccountView{
acct: acct,
conf: conf,
dirlist: dirlist,
grid: grid,
host: host,
logger: logger,
msglist: msglist,
msgStores: make(map[string]*lib.MessageStore),
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)
host.SetStatus("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
}
func (acct *AccountView) AccountConfig() *config.AccountConfig {
return acct.acct
}
2019-05-16 18:15:34 +02:00
func (acct *AccountView) Worker() *types.Worker {
return acct.worker
}
func (acct *AccountView) Logger() *log.Logger {
return acct.logger
}
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 {
return acct.grid.Children()
}
func (acct *AccountView) OnInvalidate(onInvalidate func(d ui.Drawable)) {
acct.grid.OnInvalidate(func(_ ui.Drawable) {
onInvalidate(acct)
})
}
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
func (acct *AccountView) Focus(focus bool) {
// TODO: Unfocus children I guess
2019-01-13 19:33:43 +01:00
}
2019-01-13 20:25:46 +01:00
func (acct *AccountView) connected(msg types.WorkerMessage) {
switch msg := msg.(type) {
case *types.Done:
acct.host.SetStatus("Listing mailboxes...")
acct.logger.Println("Listing mailboxes...")
acct.dirlist.UpdateList(func(dirs []string) {
var dir string
for _, _dir := range dirs {
2019-03-16 02:33:08 +01:00
if _dir == acct.acct.Default {
dir = _dir
break
}
}
if dir == "" {
dir = dirs[0]
}
acct.dirlist.Select(dir)
acct.logger.Println("Connected.")
acct.host.SetStatus("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) {
switch msg := msg.(type) {
case *types.Done:
switch msg.InResponseTo().(type) {
case *types.OpenDirectory:
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)
}
}
case *types.DirectoryInfo:
if store, ok := acct.msgStores[msg.Name]; ok {
store.Update(msg)
} else {
2019-05-14 02:16:55 +02:00
store = lib.NewMessageStore(acct.worker, msg)
acct.msgStores[msg.Name] = store
store.OnUpdate(func(_ *lib.MessageStore) {
store.OnUpdate(nil)
acct.msglist.SetStore(store)
})
}
case *types.DirectoryContents:
store := acct.msgStores[acct.dirlist.selected]
store.Update(msg)
2019-03-31 18:35:51 +02:00
case *types.FullMessage:
2019-03-30 03:35:53 +01:00
store := acct.msgStores[acct.dirlist.selected]
store.Update(msg)
case *types.MessageInfo:
store := acct.msgStores[acct.dirlist.selected]
store.Update(msg)
2019-03-21 04:23:38 +01:00
case *types.MessagesDeleted:
store := acct.msgStores[acct.dirlist.selected]
store.Update(msg)
case *types.Error:
acct.logger.Printf("%v", msg.Error)
acct.host.SetStatus(fmt.Sprintf("%v", msg.Error)).
2019-03-30 18:05:00 +01:00
Color(tcell.ColorDefault, tcell.ColorRed)
}
2019-03-11 02:15:24 +01:00
}