2019-03-11 03:45:00 +00:00
|
|
|
package widgets
|
|
|
|
|
|
|
|
import (
|
2019-06-07 20:22:04 +00:00
|
|
|
"fmt"
|
2019-03-15 01:37:00 +00:00
|
|
|
"log"
|
2020-06-09 19:13:14 +00:00
|
|
|
"math"
|
2021-11-12 17:12:02 +00:00
|
|
|
"strings"
|
2019-03-15 01:37:00 +00:00
|
|
|
|
2021-11-12 17:12:02 +00:00
|
|
|
sortthread "github.com/emersion/go-imap-sortthread"
|
2020-11-30 22:07:03 +00:00
|
|
|
"github.com/gdamore/tcell/v2"
|
2019-06-07 20:22:04 +00:00
|
|
|
"github.com/mattn/go-runewidth"
|
2019-03-15 01:37:00 +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/format"
|
|
|
|
"git.sr.ht/~rjarry/aerc/lib/ui"
|
|
|
|
"git.sr.ht/~rjarry/aerc/models"
|
2021-11-12 17:12:02 +00:00
|
|
|
"git.sr.ht/~rjarry/aerc/worker/types"
|
2019-03-11 03:45:00 +00:00
|
|
|
)
|
|
|
|
|
2019-03-15 01:37:00 +00:00
|
|
|
type MessageList struct {
|
2019-04-27 16:47:59 +00:00
|
|
|
ui.Invalidatable
|
2022-02-25 16:53:33 +00:00
|
|
|
Scrollable
|
2019-07-31 07:50:07 +00:00
|
|
|
conf *config.AercConfig
|
|
|
|
logger *log.Logger
|
|
|
|
height int
|
|
|
|
nmsgs int
|
|
|
|
spinner *Spinner
|
|
|
|
store *lib.MessageStore
|
|
|
|
isInitalizing bool
|
2019-09-05 22:32:36 +00:00
|
|
|
aerc *Aerc
|
2019-03-15 01:37:00 +00:00
|
|
|
}
|
|
|
|
|
2019-09-05 22:32:36 +00:00
|
|
|
func NewMessageList(conf *config.AercConfig, logger *log.Logger, aerc *Aerc) *MessageList {
|
2019-03-15 01:37:00 +00:00
|
|
|
ml := &MessageList{
|
2019-07-31 07:50:07 +00:00
|
|
|
conf: conf,
|
|
|
|
logger: logger,
|
2019-08-30 01:30:35 +00:00
|
|
|
spinner: NewSpinner(&conf.Ui),
|
2019-07-31 07:50:07 +00:00
|
|
|
isInitalizing: true,
|
2019-09-05 22:32:36 +00:00
|
|
|
aerc: aerc,
|
2019-03-15 01:37:00 +00:00
|
|
|
}
|
|
|
|
ml.spinner.OnInvalidate(func(_ ui.Drawable) {
|
|
|
|
ml.Invalidate()
|
|
|
|
})
|
|
|
|
// TODO: stop spinner, probably
|
|
|
|
ml.spinner.Start()
|
|
|
|
return ml
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ml *MessageList) Invalidate() {
|
2019-04-27 16:47:59 +00:00
|
|
|
ml.DoInvalidate(ml)
|
2019-03-15 01:37:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (ml *MessageList) Draw(ctx *ui.Context) {
|
2019-03-16 01:41:18 +00:00
|
|
|
ml.height = ctx.Height()
|
2022-02-24 23:21:06 +00:00
|
|
|
uiConfig := ml.aerc.SelectedAccountUiConfig()
|
2020-07-27 08:03:55 +00:00
|
|
|
ctx.Fill(0, 0, ctx.Width(), ctx.Height(), ' ',
|
2022-02-24 23:21:06 +00:00
|
|
|
uiConfig.GetStyle(config.STYLE_MSGLIST_DEFAULT))
|
2019-03-15 01:37:00 +00:00
|
|
|
|
2022-02-24 23:21:06 +00:00
|
|
|
acct := ml.aerc.SelectedAccount()
|
2019-04-28 13:26:22 +00:00
|
|
|
store := ml.Store()
|
2022-02-24 23:21:06 +00:00
|
|
|
if store == nil || acct == nil {
|
2019-07-31 07:50:07 +00:00
|
|
|
if ml.isInitalizing {
|
|
|
|
ml.spinner.Draw(ctx)
|
|
|
|
return
|
|
|
|
} else {
|
|
|
|
ml.spinner.Stop()
|
|
|
|
ml.drawEmptyMessage(ctx)
|
|
|
|
return
|
|
|
|
}
|
2019-03-15 01:37:00 +00:00
|
|
|
}
|
|
|
|
|
2022-02-25 16:53:33 +00:00
|
|
|
ml.UpdateScroller(ml.height, len(store.Uids()))
|
|
|
|
if store := ml.Store(); store != nil && len(store.Uids()) > 0 {
|
|
|
|
ml.EnsureScroll(store.SelectedIndex())
|
2020-06-09 19:13:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
textWidth := ctx.Width()
|
2022-02-25 16:53:33 +00:00
|
|
|
if ml.NeedScrollbar() {
|
2020-06-09 19:13:14 +00:00
|
|
|
textWidth -= 1
|
|
|
|
}
|
|
|
|
if textWidth < 0 {
|
|
|
|
textWidth = 0
|
|
|
|
}
|
|
|
|
|
2019-03-15 01:37:00 +00:00
|
|
|
var (
|
2019-03-15 01:51:29 +00:00
|
|
|
needsHeaders []uint32
|
2019-03-15 01:37:00 +00:00
|
|
|
row int = 0
|
|
|
|
)
|
2019-03-15 02:19:04 +00:00
|
|
|
|
2022-02-24 23:21:06 +00:00
|
|
|
if uiConfig.ThreadingEnabled || store.BuildThreads() {
|
2021-11-12 17:12:02 +00:00
|
|
|
threads := store.Threads
|
|
|
|
counter := len(store.Uids())
|
2019-03-15 02:19:04 +00:00
|
|
|
|
2021-11-12 17:12:02 +00:00
|
|
|
for i := len(threads) - 1; i >= 0; i-- {
|
|
|
|
var lastSubject string
|
|
|
|
threads[i].Walk(func(t *types.Thread, _ int, currentErr error) error {
|
|
|
|
if currentErr != nil {
|
|
|
|
return currentErr
|
|
|
|
}
|
|
|
|
if t.Hidden || t.Deleted {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
counter--
|
2022-02-25 16:53:33 +00:00
|
|
|
if counter > len(store.Uids())-1-ml.Scroll() {
|
2021-11-12 17:12:02 +00:00
|
|
|
//skip messages which are higher than the viewport
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
msg := store.Messages[t.Uid]
|
|
|
|
var prefix string
|
|
|
|
var subject string
|
|
|
|
var normalizedSubject string
|
|
|
|
if msg != nil {
|
|
|
|
prefix = threadPrefix(t)
|
|
|
|
if msg.Envelope != nil {
|
|
|
|
subject = msg.Envelope.Subject
|
|
|
|
normalizedSubject, _ = sortthread.GetBaseSubject(subject)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
fmtCtx := format.Ctx{
|
2022-02-24 23:21:06 +00:00
|
|
|
FromAddress: acct.acct.From,
|
|
|
|
AccountName: acct.Name(),
|
2021-11-12 17:12:02 +00:00
|
|
|
MsgInfo: msg,
|
|
|
|
MsgNum: row,
|
|
|
|
MsgIsMarked: store.IsMarked(t.Uid),
|
|
|
|
ThreadPrefix: prefix,
|
|
|
|
ThreadSameSubject: normalizedSubject == lastSubject,
|
|
|
|
}
|
|
|
|
if ml.drawRow(textWidth, ctx, t.Uid, row, &needsHeaders, fmtCtx) {
|
|
|
|
return types.ErrSkipThread
|
|
|
|
}
|
|
|
|
lastSubject = normalizedSubject
|
|
|
|
row++
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
if row >= ctx.Height() {
|
|
|
|
break
|
2019-06-08 13:10:14 +00:00
|
|
|
}
|
|
|
|
}
|
2021-11-12 17:12:02 +00:00
|
|
|
} else {
|
|
|
|
uids := store.Uids()
|
2022-02-25 16:53:33 +00:00
|
|
|
for i := len(uids) - 1 - ml.Scroll(); i >= 0; i-- {
|
2021-11-12 17:12:02 +00:00
|
|
|
uid := uids[i]
|
|
|
|
msg := store.Messages[uid]
|
|
|
|
fmtCtx := format.Ctx{
|
2022-02-24 23:21:06 +00:00
|
|
|
FromAddress: acct.acct.From,
|
|
|
|
AccountName: acct.Name(),
|
2020-10-14 06:42:26 +00:00
|
|
|
MsgInfo: msg,
|
2021-11-12 17:12:02 +00:00
|
|
|
MsgNum: row,
|
2020-10-14 06:42:26 +00:00
|
|
|
MsgIsMarked: store.IsMarked(uid),
|
2021-11-12 17:12:02 +00:00
|
|
|
}
|
|
|
|
if ml.drawRow(textWidth, ctx, uid, row, &needsHeaders, fmtCtx) {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
row += 1
|
2019-06-07 19:35:23 +00:00
|
|
|
}
|
2019-03-15 01:37:00 +00:00
|
|
|
}
|
|
|
|
|
2022-02-25 16:53:33 +00:00
|
|
|
if ml.NeedScrollbar() {
|
2020-06-09 19:13:14 +00:00
|
|
|
scrollbarCtx := ctx.Subcontext(ctx.Width()-1, 0, 1, ctx.Height())
|
2022-02-25 16:53:33 +00:00
|
|
|
ml.drawScrollbar(scrollbarCtx)
|
2020-06-09 19:13:14 +00:00
|
|
|
}
|
|
|
|
|
2021-11-12 17:12:02 +00:00
|
|
|
if len(store.Uids()) == 0 {
|
2020-02-29 04:03:09 +00:00
|
|
|
if store.Sorting {
|
|
|
|
ml.spinner.Start()
|
|
|
|
ml.spinner.Draw(ctx)
|
|
|
|
return
|
|
|
|
} else {
|
|
|
|
ml.drawEmptyMessage(ctx)
|
|
|
|
}
|
2019-04-04 18:25:51 +00:00
|
|
|
}
|
|
|
|
|
2019-03-15 01:37:00 +00:00
|
|
|
if len(needsHeaders) != 0 {
|
2019-04-28 13:26:22 +00:00
|
|
|
store.FetchHeaders(needsHeaders, nil)
|
2019-03-15 01:37:00 +00:00
|
|
|
ml.spinner.Start()
|
|
|
|
} else {
|
|
|
|
ml.spinner.Stop()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-11-12 17:12:02 +00:00
|
|
|
func (ml *MessageList) drawRow(textWidth int, ctx *ui.Context, uid uint32, row int, needsHeaders *[]uint32, fmtCtx format.Ctx) bool {
|
|
|
|
store := ml.store
|
|
|
|
msg := store.Messages[uid]
|
2022-02-24 23:21:06 +00:00
|
|
|
acct := ml.aerc.SelectedAccount()
|
2021-11-12 17:12:02 +00:00
|
|
|
|
2022-02-24 23:21:06 +00:00
|
|
|
if row >= ctx.Height() || acct == nil {
|
2021-11-12 17:12:02 +00:00
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
if msg == nil {
|
|
|
|
*needsHeaders = append(*needsHeaders, uid)
|
|
|
|
ml.spinner.Draw(ctx.Subcontext(0, row, textWidth, 1))
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
confParams := map[config.ContextType]string{
|
2022-02-24 23:21:06 +00:00
|
|
|
config.UI_CONTEXT_ACCOUNT: acct.AccountConfig().Name,
|
|
|
|
config.UI_CONTEXT_FOLDER: acct.Directories().Selected(),
|
2021-11-12 17:12:02 +00:00
|
|
|
}
|
|
|
|
if msg.Envelope != nil {
|
|
|
|
confParams[config.UI_CONTEXT_SUBJECT] = msg.Envelope.Subject
|
|
|
|
}
|
|
|
|
uiConfig := ml.conf.GetUiConfig(confParams)
|
|
|
|
|
|
|
|
msg_styles := []config.StyleObject{}
|
|
|
|
// unread message
|
|
|
|
seen := false
|
|
|
|
flagged := false
|
|
|
|
for _, flag := range msg.Flags {
|
|
|
|
switch flag {
|
|
|
|
case models.SeenFlag:
|
|
|
|
seen = true
|
|
|
|
case models.FlaggedFlag:
|
|
|
|
flagged = true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if seen {
|
|
|
|
msg_styles = append(msg_styles, config.STYLE_MSGLIST_READ)
|
|
|
|
} else {
|
|
|
|
msg_styles = append(msg_styles, config.STYLE_MSGLIST_UNREAD)
|
|
|
|
}
|
|
|
|
|
|
|
|
if flagged {
|
|
|
|
msg_styles = append(msg_styles, config.STYLE_MSGLIST_FLAGGED)
|
|
|
|
}
|
|
|
|
|
|
|
|
// deleted message
|
|
|
|
if _, ok := store.Deleted[msg.Uid]; ok {
|
|
|
|
msg_styles = append(msg_styles, config.STYLE_MSGLIST_DELETED)
|
|
|
|
}
|
|
|
|
|
|
|
|
// marked message
|
|
|
|
if store.IsMarked(msg.Uid) {
|
|
|
|
msg_styles = append(msg_styles, config.STYLE_MSGLIST_MARKED)
|
|
|
|
}
|
|
|
|
|
|
|
|
var style tcell.Style
|
|
|
|
// current row
|
2022-02-25 16:53:33 +00:00
|
|
|
if row == ml.store.SelectedIndex()-ml.Scroll() {
|
2021-11-12 17:12:02 +00:00
|
|
|
style = uiConfig.GetComposedStyleSelected(config.STYLE_MSGLIST_DEFAULT, msg_styles)
|
|
|
|
} else {
|
|
|
|
style = uiConfig.GetComposedStyle(config.STYLE_MSGLIST_DEFAULT, msg_styles)
|
|
|
|
}
|
|
|
|
|
|
|
|
ctx.Fill(0, row, ctx.Width(), 1, ' ', style)
|
|
|
|
fmtStr, args, err := format.ParseMessageFormat(
|
|
|
|
uiConfig.IndexFormat, uiConfig.TimestampFormat,
|
|
|
|
uiConfig.ThisDayTimeFormat,
|
|
|
|
uiConfig.ThisWeekTimeFormat,
|
|
|
|
uiConfig.ThisYearTimeFormat,
|
|
|
|
fmtCtx)
|
|
|
|
if err != nil {
|
|
|
|
ctx.Printf(0, row, style, "%v", err)
|
|
|
|
} else {
|
|
|
|
line := fmt.Sprintf(fmtStr, args...)
|
|
|
|
line = runewidth.Truncate(line, textWidth, "…")
|
|
|
|
ctx.Printf(0, row, style, "%s", line)
|
|
|
|
}
|
|
|
|
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2022-02-25 16:53:33 +00:00
|
|
|
func (ml *MessageList) drawScrollbar(ctx *ui.Context) {
|
2020-06-09 19:13:14 +00:00
|
|
|
gutterStyle := tcell.StyleDefault
|
|
|
|
pillStyle := tcell.StyleDefault.Reverse(true)
|
|
|
|
|
|
|
|
// gutter
|
|
|
|
ctx.Fill(0, 0, 1, ctx.Height(), ' ', gutterStyle)
|
|
|
|
|
|
|
|
// pill
|
2022-02-25 16:53:33 +00:00
|
|
|
pillSize := int(math.Ceil(float64(ctx.Height()) * ml.PercentVisible()))
|
|
|
|
pillOffset := int(math.Floor(float64(ctx.Height()) * ml.PercentScrolled()))
|
2020-06-09 19:13:14 +00:00
|
|
|
ctx.Fill(0, pillOffset, 1, pillSize, ' ', pillStyle)
|
|
|
|
}
|
|
|
|
|
2019-09-05 22:32:36 +00:00
|
|
|
func (ml *MessageList) MouseEvent(localX int, localY int, event tcell.Event) {
|
|
|
|
switch event := event.(type) {
|
|
|
|
case *tcell.EventMouse:
|
|
|
|
switch event.Buttons() {
|
|
|
|
case tcell.Button1:
|
|
|
|
if ml.aerc == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
selectedMsg, ok := ml.Clicked(localX, localY)
|
|
|
|
if ok {
|
|
|
|
ml.Select(selectedMsg)
|
|
|
|
acct := ml.aerc.SelectedAccount()
|
2022-02-24 23:21:06 +00:00
|
|
|
if acct == nil || acct.Messages().Empty() {
|
2019-09-05 22:32:36 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
store := acct.Messages().Store()
|
|
|
|
msg := acct.Messages().Selected()
|
|
|
|
if msg == nil {
|
|
|
|
return
|
|
|
|
}
|
2020-03-03 21:20:07 +00:00
|
|
|
lib.NewMessageStoreView(msg, store, ml.aerc.DecryptKeys,
|
2020-05-19 11:06:46 +00:00
|
|
|
func(view lib.MessageView, err error) {
|
|
|
|
if err != nil {
|
2020-05-28 14:32:42 +00:00
|
|
|
ml.aerc.PushError(err.Error())
|
2020-05-19 11:06:46 +00:00
|
|
|
return
|
|
|
|
}
|
2020-03-03 21:20:07 +00:00
|
|
|
viewer := NewMessageViewer(acct, ml.aerc.Config(), view)
|
|
|
|
ml.aerc.NewTab(viewer, msg.Envelope.Subject)
|
|
|
|
})
|
2019-09-05 22:32:36 +00:00
|
|
|
}
|
|
|
|
case tcell.WheelDown:
|
2020-02-22 16:05:46 +00:00
|
|
|
if ml.store != nil {
|
|
|
|
ml.store.Next()
|
|
|
|
}
|
2020-06-09 19:13:13 +00:00
|
|
|
ml.Invalidate()
|
2019-09-05 22:32:36 +00:00
|
|
|
case tcell.WheelUp:
|
2020-02-22 16:05:46 +00:00
|
|
|
if ml.store != nil {
|
|
|
|
ml.store.Prev()
|
|
|
|
}
|
2020-06-09 19:13:13 +00:00
|
|
|
ml.Invalidate()
|
2019-09-05 22:32:36 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ml *MessageList) Clicked(x, y int) (int, bool) {
|
|
|
|
store := ml.Store()
|
|
|
|
if store == nil || ml.nmsgs == 0 || y >= ml.nmsgs {
|
|
|
|
return 0, false
|
|
|
|
}
|
2022-02-25 16:53:33 +00:00
|
|
|
return y + ml.Scroll(), true
|
2019-09-05 22:32:36 +00:00
|
|
|
}
|
|
|
|
|
2019-03-16 01:41:18 +00:00
|
|
|
func (ml *MessageList) Height() int {
|
|
|
|
return ml.height
|
|
|
|
}
|
|
|
|
|
2019-03-21 03:23:38 +00:00
|
|
|
func (ml *MessageList) storeUpdate(store *lib.MessageStore) {
|
2019-04-28 13:26:22 +00:00
|
|
|
if ml.Store() != store {
|
2019-03-21 03:23:38 +00:00
|
|
|
return
|
|
|
|
}
|
2019-07-17 07:35:50 +00:00
|
|
|
uids := store.Uids()
|
2019-04-28 13:26:22 +00:00
|
|
|
|
2019-07-17 07:35:50 +00:00
|
|
|
if len(uids) > 0 {
|
2019-05-19 22:08:41 +00:00
|
|
|
// When new messages come in, advance the cursor accordingly
|
|
|
|
// Note that this assumes new messages are appended to the top, which
|
|
|
|
// isn't necessarily true once we implement SORT... ideally we'd look
|
|
|
|
// for the previously selected UID.
|
2019-07-17 07:35:50 +00:00
|
|
|
if len(uids) > ml.nmsgs && ml.nmsgs != 0 {
|
|
|
|
for i := 0; i < len(uids)-ml.nmsgs; i++ {
|
2019-06-11 14:08:44 +00:00
|
|
|
ml.Store().Next()
|
2019-05-19 22:08:41 +00:00
|
|
|
}
|
|
|
|
}
|
2019-07-17 07:35:50 +00:00
|
|
|
if len(uids) < ml.nmsgs && ml.nmsgs != 0 {
|
|
|
|
for i := 0; i < ml.nmsgs-len(uids); i++ {
|
2019-06-11 14:08:44 +00:00
|
|
|
ml.Store().Prev()
|
2019-05-19 22:21:02 +00:00
|
|
|
}
|
|
|
|
}
|
2019-07-17 07:35:50 +00:00
|
|
|
ml.nmsgs = len(uids)
|
2019-03-21 03:23:38 +00:00
|
|
|
}
|
2019-04-28 13:26:22 +00:00
|
|
|
|
2019-03-21 03:23:38 +00:00
|
|
|
ml.Invalidate()
|
|
|
|
}
|
|
|
|
|
2019-03-16 01:36:06 +00:00
|
|
|
func (ml *MessageList) SetStore(store *lib.MessageStore) {
|
2019-05-19 22:09:16 +00:00
|
|
|
if ml.Store() != store {
|
2022-02-25 16:53:33 +00:00
|
|
|
ml.Scrollable = Scrollable{}
|
2019-03-16 01:49:40 +00:00
|
|
|
}
|
2019-05-19 09:49:57 +00:00
|
|
|
ml.store = store
|
2019-03-15 01:37:00 +00:00
|
|
|
if store != nil {
|
|
|
|
ml.spinner.Stop()
|
2019-07-17 07:35:50 +00:00
|
|
|
ml.nmsgs = len(store.Uids())
|
2019-04-28 13:26:22 +00:00
|
|
|
store.OnUpdate(ml.storeUpdate)
|
2022-03-20 08:47:52 +00:00
|
|
|
store.OnFilterChange(func(store *lib.MessageStore) {
|
|
|
|
if ml.Store() != store {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
ml.nmsgs = len(store.Uids())
|
|
|
|
})
|
2019-03-15 01:37:00 +00:00
|
|
|
} else {
|
|
|
|
ml.spinner.Start()
|
|
|
|
}
|
|
|
|
ml.Invalidate()
|
|
|
|
}
|
2019-03-15 03:41:25 +00:00
|
|
|
|
2019-07-31 07:50:07 +00:00
|
|
|
func (ml *MessageList) SetInitDone() {
|
|
|
|
ml.isInitalizing = false
|
|
|
|
}
|
|
|
|
|
2019-03-21 03:23:38 +00:00
|
|
|
func (ml *MessageList) Store() *lib.MessageStore {
|
2019-05-19 09:49:57 +00:00
|
|
|
return ml.store
|
2019-03-21 03:23:38 +00:00
|
|
|
}
|
|
|
|
|
2019-04-09 03:14:14 +00:00
|
|
|
func (ml *MessageList) Empty() bool {
|
2019-04-28 13:26:22 +00:00
|
|
|
store := ml.Store()
|
2019-07-17 07:35:50 +00:00
|
|
|
return store == nil || len(store.Uids()) == 0
|
2019-04-09 03:14:14 +00:00
|
|
|
}
|
|
|
|
|
2019-07-08 02:43:56 +00:00
|
|
|
func (ml *MessageList) Selected() *models.MessageInfo {
|
2019-04-28 13:26:22 +00:00
|
|
|
store := ml.Store()
|
2019-07-17 07:35:50 +00:00
|
|
|
uids := store.Uids()
|
|
|
|
return store.Messages[uids[len(uids)-ml.store.SelectedIndex()-1]]
|
2019-03-21 03:23:38 +00:00
|
|
|
}
|
|
|
|
|
2019-03-16 02:01:20 +00:00
|
|
|
func (ml *MessageList) Select(index int) {
|
2019-04-28 13:26:22 +00:00
|
|
|
store := ml.Store()
|
2019-06-11 05:05:55 +00:00
|
|
|
store.Select(index)
|
2020-06-09 19:13:13 +00:00
|
|
|
ml.Invalidate()
|
2019-03-16 02:01:20 +00:00
|
|
|
}
|
|
|
|
|
2019-07-31 07:50:07 +00:00
|
|
|
func (ml *MessageList) drawEmptyMessage(ctx *ui.Context) {
|
2022-02-24 23:21:06 +00:00
|
|
|
uiConfig := ml.aerc.SelectedAccountUiConfig()
|
2020-07-27 08:03:55 +00:00
|
|
|
msg := uiConfig.EmptyMessage
|
2019-07-31 07:50:07 +00:00
|
|
|
ctx.Printf((ctx.Width()/2)-(len(msg)/2), 0,
|
2020-07-27 08:03:55 +00:00
|
|
|
uiConfig.GetStyle(config.STYLE_MSGLIST_DEFAULT), "%s", msg)
|
2019-07-31 07:50:07 +00:00
|
|
|
}
|
2021-11-12 17:12:02 +00:00
|
|
|
|
|
|
|
func threadPrefix(t *types.Thread) string {
|
|
|
|
var arrow string
|
|
|
|
if t.Parent != nil {
|
|
|
|
if t.NextSibling != nil {
|
|
|
|
arrow = "├─>"
|
|
|
|
} else {
|
|
|
|
arrow = "└─>"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
var prefix []string
|
|
|
|
for n := t; n.Parent != nil; n = n.Parent {
|
|
|
|
if n.Parent.NextSibling != nil {
|
|
|
|
prefix = append(prefix, "│ ")
|
|
|
|
} else {
|
|
|
|
prefix = append(prefix, " ")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// prefix is now in a reverse order (inside --> outside), so turn it
|
|
|
|
for i, j := 0, len(prefix)-1; i < j; i, j = i+1, j-1 {
|
|
|
|
prefix[i], prefix[j] = prefix[j], prefix[i]
|
|
|
|
}
|
|
|
|
|
|
|
|
// we don't want to indent the first child, hence we strip that level
|
|
|
|
if len(prefix) > 0 {
|
|
|
|
prefix = prefix[1:]
|
|
|
|
}
|
|
|
|
ps := strings.Join(prefix, "")
|
|
|
|
return fmt.Sprintf("%v%v", ps, arrow)
|
|
|
|
}
|