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"
|
2022-03-22 08:52:27 +00:00
|
|
|
"git.sr.ht/~rjarry/aerc/logging"
|
2021-11-05 09:19:46 +00:00
|
|
|
"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-05-30 12:34:18 +00:00
|
|
|
newConn bool // True if this is a first run after a new connection/reconnection
|
2022-07-03 15:11:13 +00:00
|
|
|
uiConf *config.UIConfig
|
2019-01-13 17:39:06 +00:00
|
|
|
}
|
|
|
|
|
2022-07-03 15:11:12 +00:00
|
|
|
func (acct *AccountView) UiConfig() *config.UIConfig {
|
2020-08-08 09:38:38 +00:00
|
|
|
if dirlist := acct.Directories(); dirlist != nil {
|
2022-07-03 15:11:13 +00:00
|
|
|
return dirlist.UiConfig()
|
2020-08-08 09:38:38 +00:00
|
|
|
}
|
2022-07-03 15:11:13 +00:00
|
|
|
return acct.uiConf
|
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,
|
2022-03-24 09:47:34 +00:00
|
|
|
logger *log.Logger, host TabHost, deferLoop chan struct{},
|
|
|
|
) (*AccountView, error) {
|
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-04-18 14:06:27 +00:00
|
|
|
state: statusline.NewState(acct.Name, len(conf.Accounts) > 1, conf.Statusline),
|
2022-07-03 15:11:13 +00:00
|
|
|
uiConf: acctUiConf,
|
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
|
|
|
|
2022-03-22 08:52:27 +00:00
|
|
|
go func() {
|
|
|
|
defer logging.PanicHandler()
|
|
|
|
|
2022-03-24 09:47:34 +00:00
|
|
|
if deferLoop != nil {
|
|
|
|
<-deferLoop
|
|
|
|
}
|
|
|
|
|
2022-03-22 08:52:27 +00:00
|
|
|
worker.Backend.Run()
|
|
|
|
}()
|
2019-01-13 18:03:28 +00:00
|
|
|
|
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..."))
|
2022-05-30 12:34:18 +00:00
|
|
|
if acct.CheckMail.Minutes() > 0 {
|
|
|
|
view.CheckMailTimer(acct.CheckMail)
|
|
|
|
}
|
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 {
|
2022-03-21 21:18:51 +00:00
|
|
|
fn(acct.state, acct.SelectedDirectory())
|
2022-03-18 21:35:33 +00:00
|
|
|
}
|
2022-03-21 21:18:51 +00:00
|
|
|
acct.UpdateStatus()
|
2022-03-18 21:35:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (acct *AccountView) UpdateStatus() {
|
2022-03-24 20:42:05 +00:00
|
|
|
if acct.isSelected() {
|
|
|
|
acct.host.SetStatus(acct.state.StatusLine(acct.SelectedDirectory()))
|
|
|
|
}
|
2022-03-18 21:35:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
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-04-18 14:06:27 +00:00
|
|
|
if acct.state.SetWidth(ctx.Width()) {
|
|
|
|
acct.UpdateStatus()
|
|
|
|
}
|
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
|
|
|
}
|
|
|
|
|
2022-03-24 20:42:05 +00:00
|
|
|
func (acct *AccountView) isSelected() bool {
|
|
|
|
return acct.aerc.NumTabs() > 0 && acct == acct.aerc.SelectedAccount()
|
|
|
|
}
|
|
|
|
|
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-05-30 12:34:18 +00:00
|
|
|
acct.SetStatus(statusline.SetConnected(true))
|
|
|
|
acct.newConn = true
|
2021-11-01 20:38:26 +00:00
|
|
|
})
|
|
|
|
case *types.Disconnect:
|
2022-06-07 15:59:05 +00:00
|
|
|
acct.dirlist.ClearList()
|
2021-11-01 20:38:26 +00:00
|
|
|
acct.msglist.SetStore(nil)
|
|
|
|
acct.logger.Println("Disconnected.")
|
2022-05-30 12:34:18 +00:00
|
|
|
acct.SetStatus(statusline.SetConnected(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)
|
2022-05-30 12:34:18 +00:00
|
|
|
case *types.FetchMessageHeaders:
|
|
|
|
if acct.newConn && acct.AccountConfig().CheckMail.Minutes() > 0 {
|
|
|
|
acct.newConn = false
|
|
|
|
acct.CheckMail()
|
|
|
|
}
|
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,
|
2022-03-24 22:12:14 +00:00
|
|
|
acct.GetSortCriteria(),
|
2021-11-12 17:12:02 +00:00
|
|
|
acct.UiConfig().ThreadingEnabled,
|
2022-07-05 19:48:39 +00:00
|
|
|
acct.UiConfig().ForceClientThreads,
|
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)
|
2022-07-05 19:48:35 +00:00
|
|
|
acct.SetStatus(statusline.Threading(store.ThreadedView()))
|
2019-06-02 08:48:03 +00:00
|
|
|
}
|
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)
|
2022-07-05 19:48:35 +00:00
|
|
|
acct.SetStatus(statusline.Threading(store.ThreadedView()))
|
2021-11-12 17:12:02 +00:00
|
|
|
}
|
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 {
|
2022-06-02 00:24:53 +00:00
|
|
|
store.DirInfo.Exists -= len(msg.Uids)
|
|
|
|
// False to trigger recount of recent/unseen
|
|
|
|
store.DirInfo.AccurateCounts = false
|
2019-06-02 08:48:03 +00:00
|
|
|
store.Update(msg)
|
|
|
|
}
|
2022-06-02 00:24:53 +00:00
|
|
|
case *types.MessagesCopied:
|
|
|
|
// Only update the destination destStore if it is initialized
|
|
|
|
if destStore, ok := acct.dirlist.MsgStore(msg.Destination); ok {
|
|
|
|
var recent, unseen int
|
|
|
|
for _, uid := range msg.Uids {
|
|
|
|
// Get the message from the originating store
|
|
|
|
msg, ok := acct.Store().Messages[uid]
|
|
|
|
if !ok {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
seen := false
|
|
|
|
for _, flag := range msg.Flags {
|
|
|
|
if flag == models.SeenFlag {
|
|
|
|
seen = true
|
|
|
|
}
|
|
|
|
if flag == models.RecentFlag {
|
|
|
|
recent = recent + 1
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if !seen {
|
|
|
|
unseen = unseen + 1
|
|
|
|
}
|
|
|
|
}
|
|
|
|
destStore.DirInfo.Recent += recent
|
|
|
|
destStore.DirInfo.Unseen += unseen
|
|
|
|
destStore.DirInfo.Exists += len(msg.Uids)
|
|
|
|
// True. For imap, we don't have the message in the store until we
|
|
|
|
// Select so we need to rely on the math we just did for accurate
|
|
|
|
// counts
|
|
|
|
destStore.DirInfo.AccurateCounts = true
|
|
|
|
}
|
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)
|
2022-05-30 12:34:18 +00:00
|
|
|
acct.SetStatus(statusline.SetConnected(false))
|
2022-03-18 21:35:33 +00:00
|
|
|
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
|
|
|
}
|
2022-03-21 21:18:51 +00:00
|
|
|
acct.UpdateStatus()
|
2019-03-11 01:15:24 +00:00
|
|
|
}
|
2019-09-19 22:37:44 +00:00
|
|
|
|
2022-03-24 22:12:14 +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
|
|
|
|
}
|
2022-05-30 12:34:18 +00:00
|
|
|
|
|
|
|
func (acct *AccountView) CheckMail() {
|
|
|
|
// Exclude selected mailbox, per IMAP specification
|
|
|
|
exclude := append(acct.AccountConfig().CheckMailExclude, acct.dirlist.Selected())
|
|
|
|
dirs := acct.dirlist.List()
|
|
|
|
dirs = acct.dirlist.FilterDirs(dirs, acct.AccountConfig().CheckMailInclude, false)
|
|
|
|
dirs = acct.dirlist.FilterDirs(dirs, exclude, true)
|
|
|
|
acct.logger.Printf("Checking for new mail on account %s", acct.Name())
|
|
|
|
acct.SetStatus(statusline.ConnectionActivity("Checking for new mail..."))
|
|
|
|
msg := &types.CheckMail{
|
|
|
|
Directories: dirs,
|
|
|
|
Command: acct.acct.CheckMailCmd,
|
|
|
|
Timeout: acct.acct.CheckMailTimeout,
|
|
|
|
}
|
|
|
|
acct.worker.PostAction(msg, func(_ types.WorkerMessage) {
|
|
|
|
acct.SetStatus(statusline.ConnectionActivity(""))
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func (acct *AccountView) CheckMailTimer(d time.Duration) {
|
|
|
|
ticker := time.NewTicker(d)
|
|
|
|
go func() {
|
|
|
|
for range ticker.C {
|
|
|
|
if !acct.state.Connected() {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
acct.CheckMail()
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
}
|