2019-01-13 17:39:06 +00:00
|
|
|
package widgets
|
|
|
|
|
|
|
|
import (
|
2019-07-10 00:04:21 +00:00
|
|
|
"errors"
|
2019-01-13 18:33:43 +00:00
|
|
|
"fmt"
|
2019-01-13 18:03:28 +00:00
|
|
|
"log"
|
2022-03-18 21:35:33 +00:00
|
|
|
"time"
|
2019-01-13 18:25:56 +00:00
|
|
|
|
2020-11-30 22:07:03 +00:00
|
|
|
"github.com/gdamore/tcell/v2"
|
2019-01-13 18:03:28 +00:00
|
|
|
|
2021-11-05 09:19:46 +00:00
|
|
|
"git.sr.ht/~rjarry/aerc/config"
|
|
|
|
"git.sr.ht/~rjarry/aerc/lib"
|
|
|
|
"git.sr.ht/~rjarry/aerc/lib/sort"
|
2022-03-18 21:35:33 +00:00
|
|
|
"git.sr.ht/~rjarry/aerc/lib/statusline"
|
2021-11-05 09:19:46 +00:00
|
|
|
"git.sr.ht/~rjarry/aerc/lib/ui"
|
|
|
|
"git.sr.ht/~rjarry/aerc/models"
|
|
|
|
"git.sr.ht/~rjarry/aerc/worker"
|
|
|
|
"git.sr.ht/~rjarry/aerc/worker/types"
|
2019-01-13 17:39:06 +00:00
|
|
|
)
|
|
|
|
|
2019-12-18 05:33:57 +00:00
|
|
|
var _ ProvidesMessages = (*AccountView)(nil)
|
|
|
|
|
2019-01-13 17:39:06 +00:00
|
|
|
type AccountView struct {
|
2019-07-21 21:39:36 +00:00
|
|
|
acct *config.AccountConfig
|
2019-09-05 22:32:36 +00:00
|
|
|
aerc *Aerc
|
2019-07-21 21:39:36 +00:00
|
|
|
conf *config.AercConfig
|
2022-02-20 23:18:42 +00:00
|
|
|
dirlist DirectoryLister
|
2019-12-21 15:21:25 +00:00
|
|
|
labels []string
|
2019-07-21 21:39:36 +00:00
|
|
|
grid *ui.Grid
|
|
|
|
host TabHost
|
|
|
|
logger *log.Logger
|
|
|
|
msglist *MessageList
|
|
|
|
worker *types.Worker
|
2022-03-18 21:35:33 +00:00
|
|
|
state *statusline.State
|
2022-03-19 12:30:07 +00:00
|
|
|
update bool
|
2019-01-13 17:39:06 +00:00
|
|
|
}
|
|
|
|
|
2020-01-23 12:56:48 +00:00
|
|
|
func (acct *AccountView) UiConfig() config.UIConfig {
|
2020-08-08 09:38:38 +00:00
|
|
|
var folder string
|
|
|
|
if dirlist := acct.Directories(); dirlist != nil {
|
|
|
|
folder = dirlist.Selected()
|
|
|
|
}
|
2020-01-24 17:18:49 +00:00
|
|
|
return acct.conf.GetUiConfig(map[config.ContextType]string{
|
2020-01-23 12:56:48 +00:00
|
|
|
config.UI_CONTEXT_ACCOUNT: acct.AccountConfig().Name,
|
2020-08-08 09:38:38 +00:00
|
|
|
config.UI_CONTEXT_FOLDER: folder,
|
2020-01-23 12:56:48 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2019-09-05 22:32:36 +00:00
|
|
|
func NewAccountView(aerc *Aerc, conf *config.AercConfig, acct *config.AccountConfig,
|
2020-08-08 09:38:38 +00:00
|
|
|
logger *log.Logger, host TabHost) (*AccountView, error) {
|
2019-01-13 17:39:06 +00:00
|
|
|
|
2020-01-24 17:18:49 +00:00
|
|
|
acctUiConf := conf.GetUiConfig(map[config.ContextType]string{
|
2020-01-23 12:56:48 +00:00
|
|
|
config.UI_CONTEXT_ACCOUNT: acct.Name,
|
|
|
|
})
|
|
|
|
|
2020-05-31 11:37:46 +00:00
|
|
|
view := &AccountView{
|
|
|
|
acct: acct,
|
|
|
|
aerc: aerc,
|
|
|
|
conf: conf,
|
|
|
|
host: host,
|
|
|
|
logger: logger,
|
2022-03-18 21:35:33 +00:00
|
|
|
state: statusline.NewState(acct.Name, len(conf.Accounts) > 1, " | "),
|
2020-05-31 11:37:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
view.grid = ui.NewGrid().Rows([]ui.GridSpec{
|
2022-03-18 08:53:02 +00:00
|
|
|
{Strategy: ui.SIZE_WEIGHT, Size: ui.Const(1)},
|
2019-01-13 17:39:06 +00:00
|
|
|
}).Columns([]ui.GridSpec{
|
2022-03-18 08:53:02 +00:00
|
|
|
{Strategy: ui.SIZE_EXACT, Size: func() int {
|
2020-05-31 11:37:46 +00:00
|
|
|
return view.UiConfig().SidebarWidth
|
|
|
|
}},
|
2022-03-18 08:53:02 +00:00
|
|
|
{Strategy: ui.SIZE_WEIGHT, Size: ui.Const(1)},
|
2019-01-13 17:39:06 +00:00
|
|
|
})
|
2019-01-13 18:25:56 +00:00
|
|
|
|
2019-03-15 05:46:14 +00:00
|
|
|
worker, err := worker.NewWorker(acct.Source, logger)
|
2019-01-13 18:25:56 +00:00
|
|
|
if err != nil {
|
2020-07-27 08:03:55 +00:00
|
|
|
host.SetError(fmt.Sprintf("%s: %s", acct.Name, err))
|
2020-08-08 09:38:38 +00:00
|
|
|
logger.Printf("%s: %s\n", acct.Name, err)
|
|
|
|
return view, err
|
2019-01-13 18:25:56 +00:00
|
|
|
}
|
2020-05-31 11:37:46 +00:00
|
|
|
view.worker = worker
|
2019-01-13 18:03:28 +00:00
|
|
|
|
2020-05-31 11:37:46 +00:00
|
|
|
view.dirlist = NewDirectoryList(conf, acct, logger, worker)
|
2020-01-23 12:56:48 +00:00
|
|
|
if acctUiConf.SidebarWidth > 0 {
|
2020-07-27 08:03:55 +00:00
|
|
|
view.grid.AddChild(ui.NewBordered(view.dirlist, ui.BORDER_RIGHT, acctUiConf))
|
2019-06-05 21:20:27 +00:00
|
|
|
}
|
2019-01-13 19:25:46 +00:00
|
|
|
|
2020-05-31 11:37:46 +00:00
|
|
|
view.msglist = NewMessageList(conf, logger, aerc)
|
|
|
|
view.grid.AddChild(view.msglist).At(0, 1)
|
2019-01-13 18:03:28 +00:00
|
|
|
|
|
|
|
go worker.Backend.Run()
|
|
|
|
|
2019-03-15 05:46:14 +00:00
|
|
|
worker.PostAction(&types.Configure{Config: acct}, nil)
|
2021-11-01 20:38:26 +00:00
|
|
|
worker.PostAction(&types.Connect{}, nil)
|
2022-03-18 21:35:33 +00:00
|
|
|
view.SetStatus(statusline.ConnectionActivity("Connecting..."))
|
2019-01-13 18:03:28 +00:00
|
|
|
|
2020-08-08 09:38:38 +00:00
|
|
|
return view, nil
|
2019-01-13 18:03:28 +00:00
|
|
|
}
|
|
|
|
|
2019-05-19 09:49:57 +00:00
|
|
|
func (acct *AccountView) Tick() bool {
|
2019-05-20 23:20:20 +00:00
|
|
|
if acct.worker == nil {
|
|
|
|
return false
|
|
|
|
}
|
2019-05-19 09:49:57 +00:00
|
|
|
select {
|
|
|
|
case msg := <-acct.worker.Messages:
|
|
|
|
msg = acct.worker.ProcessMessage(msg)
|
|
|
|
acct.onMessage(msg)
|
|
|
|
return true
|
|
|
|
default:
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-03-18 21:35:33 +00:00
|
|
|
func (acct *AccountView) SetStatus(setters ...statusline.SetStateFunc) {
|
|
|
|
for _, fn := range setters {
|
|
|
|
fn(acct.state)
|
|
|
|
}
|
2022-03-19 12:30:07 +00:00
|
|
|
acct.update = true
|
2022-03-18 21:35:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (acct *AccountView) UpdateStatus() {
|
|
|
|
acct.host.SetStatus(acct.state.String())
|
|
|
|
}
|
|
|
|
|
|
|
|
func (acct *AccountView) PushStatus(status string, expiry time.Duration) {
|
|
|
|
acct.aerc.PushStatus(fmt.Sprintf("%s: %v", acct.acct.Name, status), expiry)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (acct *AccountView) PushError(err error) {
|
|
|
|
acct.aerc.PushError(fmt.Sprintf("%s: %v", acct.acct.Name, err))
|
2021-11-01 20:38:26 +00:00
|
|
|
}
|
|
|
|
|
2019-05-13 20:04:01 +00:00
|
|
|
func (acct *AccountView) AccountConfig() *config.AccountConfig {
|
|
|
|
return acct.acct
|
|
|
|
}
|
|
|
|
|
2019-05-16 16:15:34 +00:00
|
|
|
func (acct *AccountView) Worker() *types.Worker {
|
|
|
|
return acct.worker
|
|
|
|
}
|
|
|
|
|
|
|
|
func (acct *AccountView) Logger() *log.Logger {
|
|
|
|
return acct.logger
|
|
|
|
}
|
|
|
|
|
2019-03-15 02:34:34 +00:00
|
|
|
func (acct *AccountView) Name() string {
|
2019-03-15 05:46:14 +00:00
|
|
|
return acct.acct.Name
|
2019-03-15 02:34:34 +00:00
|
|
|
}
|
|
|
|
|
2019-02-10 21:46:13 +00:00
|
|
|
func (acct *AccountView) Children() []ui.Drawable {
|
2019-01-20 20:08:30 +00:00
|
|
|
return acct.grid.Children()
|
|
|
|
}
|
|
|
|
|
2019-01-13 17:39:06 +00:00
|
|
|
func (acct *AccountView) OnInvalidate(onInvalidate func(d ui.Drawable)) {
|
2019-01-13 18:25:56 +00:00
|
|
|
acct.grid.OnInvalidate(func(_ ui.Drawable) {
|
|
|
|
onInvalidate(acct)
|
|
|
|
})
|
2019-01-13 17:39:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (acct *AccountView) Invalidate() {
|
|
|
|
acct.grid.Invalidate()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (acct *AccountView) Draw(ctx *ui.Context) {
|
2022-03-19 12:30:07 +00:00
|
|
|
if acct.update {
|
|
|
|
acct.UpdateStatus()
|
|
|
|
acct.update = false
|
|
|
|
}
|
2019-01-13 17:39:06 +00:00
|
|
|
acct.grid.Draw(ctx)
|
|
|
|
}
|
2019-01-13 18:33:43 +00:00
|
|
|
|
2019-09-05 22:32:36 +00:00
|
|
|
func (acct *AccountView) MouseEvent(localX int, localY int, event tcell.Event) {
|
|
|
|
acct.grid.MouseEvent(localX, localY, event)
|
|
|
|
}
|
|
|
|
|
2019-03-17 20:19:15 +00:00
|
|
|
func (acct *AccountView) Focus(focus bool) {
|
|
|
|
// TODO: Unfocus children I guess
|
2019-01-13 18:33:43 +00:00
|
|
|
}
|
2019-01-13 19:25:46 +00:00
|
|
|
|
2022-02-20 23:18:42 +00:00
|
|
|
func (acct *AccountView) Directories() DirectoryLister {
|
2019-03-11 01:15:24 +00:00
|
|
|
return acct.dirlist
|
|
|
|
}
|
|
|
|
|
2019-12-21 15:21:25 +00:00
|
|
|
func (acct *AccountView) Labels() []string {
|
|
|
|
return acct.labels
|
|
|
|
}
|
|
|
|
|
2019-03-15 03:41:25 +00:00
|
|
|
func (acct *AccountView) Messages() *MessageList {
|
|
|
|
return acct.msglist
|
|
|
|
}
|
|
|
|
|
2019-06-02 05:15:04 +00:00
|
|
|
func (acct *AccountView) Store() *lib.MessageStore {
|
2019-08-08 10:48:51 +00:00
|
|
|
if acct.msglist == nil {
|
|
|
|
return nil
|
|
|
|
}
|
2019-06-02 05:15:04 +00:00
|
|
|
return acct.msglist.Store()
|
|
|
|
}
|
|
|
|
|
2019-07-05 16:21:12 +00:00
|
|
|
func (acct *AccountView) SelectedAccount() *AccountView {
|
|
|
|
return acct
|
|
|
|
}
|
|
|
|
|
2020-04-24 09:42:22 +00:00
|
|
|
func (acct *AccountView) SelectedDirectory() string {
|
|
|
|
return acct.dirlist.Selected()
|
|
|
|
}
|
|
|
|
|
2019-07-10 00:04:21 +00:00
|
|
|
func (acct *AccountView) SelectedMessage() (*models.MessageInfo, error) {
|
2019-07-17 07:35:50 +00:00
|
|
|
if len(acct.msglist.Store().Uids()) == 0 {
|
2019-07-10 00:04:21 +00:00
|
|
|
return nil, errors.New("no message selected")
|
|
|
|
}
|
2019-09-10 13:53:40 +00:00
|
|
|
msg := acct.msglist.Selected()
|
|
|
|
if msg == nil {
|
|
|
|
return nil, errors.New("message not loaded")
|
|
|
|
}
|
|
|
|
return msg, nil
|
2019-06-02 05:15:04 +00:00
|
|
|
}
|
|
|
|
|
2020-05-09 09:50:31 +00:00
|
|
|
func (acct *AccountView) MarkedMessages() ([]uint32, error) {
|
2019-12-18 05:33:57 +00:00
|
|
|
store := acct.Store()
|
2020-05-09 09:50:31 +00:00
|
|
|
return store.Marked(), nil
|
2019-12-18 05:33:57 +00:00
|
|
|
}
|
|
|
|
|
2019-07-05 16:21:12 +00:00
|
|
|
func (acct *AccountView) SelectedMessagePart() *PartInfo {
|
|
|
|
return nil
|
2019-06-02 05:15:04 +00:00
|
|
|
}
|
|
|
|
|
2019-03-11 01:15:24 +00:00
|
|
|
func (acct *AccountView) onMessage(msg types.WorkerMessage) {
|
2019-03-11 03:45:00 +00:00
|
|
|
switch msg := msg.(type) {
|
|
|
|
case *types.Done:
|
|
|
|
switch msg.InResponseTo().(type) {
|
2022-01-19 12:18:10 +00:00
|
|
|
case *types.Connect, *types.Reconnect:
|
2022-03-18 21:35:33 +00:00
|
|
|
acct.SetStatus(statusline.ConnectionActivity("Listing mailboxes..."))
|
2021-11-01 20:38:26 +00:00
|
|
|
acct.logger.Println("Listing mailboxes...")
|
|
|
|
acct.dirlist.UpdateList(func(dirs []string) {
|
|
|
|
var dir string
|
|
|
|
for _, _dir := range dirs {
|
|
|
|
if _dir == acct.acct.Default {
|
|
|
|
dir = _dir
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if dir == "" && len(dirs) > 0 {
|
|
|
|
dir = dirs[0]
|
|
|
|
}
|
|
|
|
if dir != "" {
|
|
|
|
acct.dirlist.Select(dir)
|
|
|
|
}
|
|
|
|
acct.msglist.SetInitDone()
|
|
|
|
acct.logger.Println("Connected.")
|
2022-03-18 21:35:33 +00:00
|
|
|
acct.SetStatus(statusline.Connected(true))
|
2021-11-01 20:38:26 +00:00
|
|
|
})
|
|
|
|
case *types.Disconnect:
|
|
|
|
acct.dirlist.UpdateList(nil)
|
|
|
|
acct.msglist.SetStore(nil)
|
|
|
|
acct.logger.Println("Disconnected.")
|
2022-03-18 21:35:33 +00:00
|
|
|
acct.SetStatus(statusline.Connected(false))
|
2019-03-11 03:45:00 +00:00
|
|
|
case *types.OpenDirectory:
|
2019-07-21 21:39:36 +00:00
|
|
|
if store, ok := acct.dirlist.SelectedMsgStore(); ok {
|
2019-03-15 02:41:43 +00:00
|
|
|
// 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-06-08 17:41:56 +00:00
|
|
|
case *types.CreateDirectory:
|
|
|
|
acct.dirlist.UpdateList(nil)
|
2020-08-18 20:27:23 +00:00
|
|
|
case *types.RemoveDirectory:
|
|
|
|
acct.dirlist.UpdateList(nil)
|
2019-03-11 03:45:00 +00:00
|
|
|
}
|
|
|
|
case *types.DirectoryInfo:
|
2019-07-21 21:39:36 +00:00
|
|
|
if store, ok := acct.dirlist.MsgStore(msg.Info.Name); ok {
|
2019-03-11 03:45:00 +00:00
|
|
|
store.Update(msg)
|
|
|
|
} else {
|
2019-07-21 20:01:51 +00:00
|
|
|
store = lib.NewMessageStore(acct.worker, msg.Info,
|
2020-02-19 07:37:20 +00:00
|
|
|
acct.getSortCriteria(),
|
2021-11-12 17:12:02 +00:00
|
|
|
acct.UiConfig().ThreadingEnabled,
|
2019-07-21 20:01:51 +00:00
|
|
|
func(msg *models.MessageInfo) {
|
|
|
|
acct.conf.Triggers.ExecNewEmail(acct.acct,
|
|
|
|
acct.conf, msg)
|
2019-07-29 14:50:02 +00:00
|
|
|
}, func() {
|
2020-01-23 12:56:48 +00:00
|
|
|
if acct.UiConfig().NewMessageBell {
|
2019-07-29 14:50:02 +00:00
|
|
|
acct.host.Beep()
|
|
|
|
}
|
2019-07-21 20:01:51 +00:00
|
|
|
})
|
2019-07-21 21:39:36 +00:00
|
|
|
acct.dirlist.SetMsgStore(msg.Info.Name, store)
|
2019-03-11 03:45:00 +00:00
|
|
|
}
|
|
|
|
case *types.DirectoryContents:
|
2019-07-21 21:39:36 +00:00
|
|
|
if store, ok := acct.dirlist.SelectedMsgStore(); ok {
|
2020-02-29 01:53:32 +00:00
|
|
|
if acct.msglist.Store() == nil {
|
|
|
|
acct.msglist.SetStore(store)
|
|
|
|
}
|
2019-06-02 08:48:03 +00:00
|
|
|
store.Update(msg)
|
|
|
|
}
|
2021-11-12 17:12:02 +00:00
|
|
|
case *types.DirectoryThreaded:
|
|
|
|
if store, ok := acct.dirlist.SelectedMsgStore(); ok {
|
|
|
|
if acct.msglist.Store() == nil {
|
|
|
|
acct.msglist.SetStore(store)
|
|
|
|
}
|
|
|
|
store.Update(msg)
|
|
|
|
}
|
2019-03-31 16:35:51 +00:00
|
|
|
case *types.FullMessage:
|
2019-07-21 21:39:36 +00:00
|
|
|
if store, ok := acct.dirlist.SelectedMsgStore(); ok {
|
2019-06-02 08:48:03 +00:00
|
|
|
store.Update(msg)
|
|
|
|
}
|
2019-03-11 03:45:00 +00:00
|
|
|
case *types.MessageInfo:
|
2019-07-21 21:39:36 +00:00
|
|
|
if store, ok := acct.dirlist.SelectedMsgStore(); ok {
|
2019-06-02 08:48:03 +00:00
|
|
|
store.Update(msg)
|
|
|
|
}
|
2019-03-21 03:23:38 +00:00
|
|
|
case *types.MessagesDeleted:
|
2019-07-21 21:39:36 +00:00
|
|
|
if store, ok := acct.dirlist.SelectedMsgStore(); ok {
|
2019-06-02 08:48:03 +00:00
|
|
|
store.Update(msg)
|
|
|
|
}
|
2019-12-21 15:21:25 +00:00
|
|
|
case *types.LabelList:
|
|
|
|
acct.labels = msg.Labels
|
2022-01-19 12:18:09 +00:00
|
|
|
case *types.ConnError:
|
2022-03-18 21:35:33 +00:00
|
|
|
acct.logger.Printf("connection error: [%s] %v", acct.acct.Name, msg.Error)
|
|
|
|
acct.SetStatus(statusline.Connected(false))
|
|
|
|
acct.PushError(msg.Error)
|
2022-02-04 23:07:16 +00:00
|
|
|
acct.msglist.SetStore(nil)
|
2022-01-19 12:18:10 +00:00
|
|
|
acct.worker.PostAction(&types.Reconnect{}, nil)
|
2019-03-11 03:45:00 +00:00
|
|
|
case *types.Error:
|
|
|
|
acct.logger.Printf("%v", msg.Error)
|
2022-03-18 21:35:33 +00:00
|
|
|
acct.PushError(msg.Error)
|
2019-03-11 03:45:00 +00:00
|
|
|
}
|
2019-03-11 01:15:24 +00:00
|
|
|
}
|
2019-09-19 22:37:44 +00:00
|
|
|
|
|
|
|
func (acct *AccountView) getSortCriteria() []*types.SortCriterion {
|
2020-01-23 12:56:48 +00:00
|
|
|
if len(acct.UiConfig().Sort) == 0 {
|
2019-09-19 22:37:44 +00:00
|
|
|
return nil
|
|
|
|
}
|
2020-01-23 12:56:48 +00:00
|
|
|
criteria, err := sort.GetSortCriteria(acct.UiConfig().Sort)
|
2019-09-19 22:37:44 +00:00
|
|
|
if err != nil {
|
2022-03-18 21:35:33 +00:00
|
|
|
acct.PushError(fmt.Errorf("ui sort: %v", err))
|
2019-09-19 22:37:44 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return criteria
|
|
|
|
}
|