2019-06-02 05:15:04 +00:00
|
|
|
package msg
|
2019-03-21 03:23:38 +00:00
|
|
|
|
|
|
|
import (
|
|
|
|
"errors"
|
2020-05-28 14:32:32 +00:00
|
|
|
"time"
|
2019-05-14 20:34:42 +00:00
|
|
|
|
2021-11-05 09:19:46 +00:00
|
|
|
"git.sr.ht/~rjarry/aerc/lib"
|
|
|
|
"git.sr.ht/~rjarry/aerc/models"
|
|
|
|
"git.sr.ht/~rjarry/aerc/widgets"
|
|
|
|
"git.sr.ht/~rjarry/aerc/worker/types"
|
2019-03-21 03:23:38 +00:00
|
|
|
)
|
|
|
|
|
2019-06-27 17:33:11 +00:00
|
|
|
type Delete struct{}
|
|
|
|
|
2019-03-21 03:23:38 +00:00
|
|
|
func init() {
|
2019-06-27 17:33:11 +00:00
|
|
|
register(Delete{})
|
|
|
|
}
|
|
|
|
|
2019-09-03 19:34:03 +00:00
|
|
|
func (Delete) Aliases() []string {
|
2019-06-27 17:33:11 +00:00
|
|
|
return []string{"delete", "delete-message"}
|
|
|
|
}
|
|
|
|
|
2019-09-03 19:34:03 +00:00
|
|
|
func (Delete) Complete(aerc *widgets.Aerc, args []string) []string {
|
2019-06-27 17:33:11 +00:00
|
|
|
return nil
|
2019-03-21 03:23:38 +00:00
|
|
|
}
|
|
|
|
|
2019-09-03 19:34:03 +00:00
|
|
|
func (Delete) Execute(aerc *widgets.Aerc, args []string) error {
|
2019-03-21 03:23:38 +00:00
|
|
|
if len(args) != 1 {
|
2019-05-19 22:23:34 +00:00
|
|
|
return errors.New("Usage: :delete")
|
2019-03-21 03:23:38 +00:00
|
|
|
}
|
2019-06-02 05:15:04 +00:00
|
|
|
|
2019-12-18 05:34:04 +00:00
|
|
|
h := newHelper(aerc)
|
|
|
|
store, err := h.store()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
2019-03-21 03:23:38 +00:00
|
|
|
}
|
2020-05-09 09:50:30 +00:00
|
|
|
uids, err := h.markedOrSelectedUids()
|
2019-12-18 05:34:04 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
2019-07-14 07:42:24 +00:00
|
|
|
}
|
2019-12-18 05:34:04 +00:00
|
|
|
acct, err := h.account()
|
2019-07-10 00:04:21 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
2019-07-03 01:46:05 +00:00
|
|
|
}
|
2022-08-05 02:43:48 +00:00
|
|
|
sel := store.Selected()
|
2022-07-31 20:16:40 +00:00
|
|
|
// caution, can be nil
|
2022-07-24 02:03:45 +00:00
|
|
|
next := findNextNonDeleted(uids, store)
|
2022-07-31 14:41:20 +00:00
|
|
|
store.ClearVisualMark()
|
2019-12-18 05:34:04 +00:00
|
|
|
store.Delete(uids, func(msg types.WorkerMessage) {
|
2019-05-14 20:34:42 +00:00
|
|
|
switch msg := msg.(type) {
|
|
|
|
case *types.Done:
|
2020-05-28 14:32:32 +00:00
|
|
|
aerc.PushStatus("Messages deleted.", 10*time.Second)
|
2022-07-24 02:03:45 +00:00
|
|
|
mv, isMsgView := h.msgProvider.(*widgets.MessageViewer)
|
|
|
|
if isMsgView {
|
|
|
|
if !aerc.Config().Ui.NextMessageOnDelete {
|
|
|
|
aerc.RemoveTab(h.msgProvider)
|
|
|
|
} else {
|
|
|
|
// no more messages in the list
|
|
|
|
if next == nil {
|
|
|
|
aerc.RemoveTab(h.msgProvider)
|
2022-07-26 09:30:26 +00:00
|
|
|
acct.Messages().Select(0)
|
2022-07-24 02:03:45 +00:00
|
|
|
acct.Messages().Invalidate()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
lib.NewMessageStoreView(next, store, aerc.Crypto, aerc.DecryptKeys,
|
|
|
|
func(view lib.MessageView, err error) {
|
|
|
|
if err != nil {
|
|
|
|
aerc.PushError(err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
nextMv := widgets.NewMessageViewer(acct, aerc.Config(), view)
|
|
|
|
aerc.ReplaceTab(mv, nextMv, next.Envelope.Subject)
|
|
|
|
})
|
|
|
|
}
|
2022-07-24 02:03:46 +00:00
|
|
|
} else {
|
|
|
|
if next == nil {
|
|
|
|
// We deleted the last message, select the new last message
|
|
|
|
// instead of the first message
|
2022-07-26 09:30:26 +00:00
|
|
|
acct.Messages().Select(0)
|
2022-07-24 02:03:46 +00:00
|
|
|
}
|
2022-07-24 02:03:45 +00:00
|
|
|
}
|
2019-05-14 20:34:42 +00:00
|
|
|
case *types.Error:
|
2022-07-31 14:41:20 +00:00
|
|
|
store.Remark()
|
2022-08-05 02:43:48 +00:00
|
|
|
store.Select(sel.Uid)
|
2021-01-30 12:51:32 +00:00
|
|
|
aerc.PushError(msg.Error.Error())
|
2020-07-03 06:12:19 +00:00
|
|
|
case *types.Unsupported:
|
2022-07-31 14:41:20 +00:00
|
|
|
store.Remark()
|
2022-08-05 02:43:48 +00:00
|
|
|
store.Select(sel.Uid)
|
2020-07-03 06:12:19 +00:00
|
|
|
// notmuch doesn't support it, we want the user to know
|
|
|
|
aerc.PushError(" error, unsupported for this worker")
|
2019-05-14 20:34:42 +00:00
|
|
|
}
|
|
|
|
})
|
2019-03-21 03:23:38 +00:00
|
|
|
return nil
|
|
|
|
}
|
2019-12-18 05:34:04 +00:00
|
|
|
|
|
|
|
func findNextNonDeleted(deleted []uint32, store *lib.MessageStore) *models.MessageInfo {
|
2022-07-26 09:30:27 +00:00
|
|
|
var next, previous *models.MessageInfo
|
|
|
|
stepper := []func(){store.Next, store.Prev}
|
|
|
|
for _, stepFn := range stepper {
|
|
|
|
for {
|
|
|
|
next = store.Selected()
|
|
|
|
if next != nil && !contains(deleted, next.Uid) {
|
|
|
|
return next
|
|
|
|
}
|
|
|
|
if next == nil || previous == next {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
stepFn()
|
|
|
|
previous = next
|
|
|
|
}
|
2019-12-18 05:34:04 +00:00
|
|
|
}
|
2022-03-09 21:48:00 +00:00
|
|
|
|
2022-07-26 09:30:27 +00:00
|
|
|
if next != nil {
|
|
|
|
store.Select(next.Uid)
|
2019-12-18 05:34:04 +00:00
|
|
|
}
|
2022-03-09 21:48:00 +00:00
|
|
|
return next
|
2019-12-18 05:34:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func contains(uids []uint32, uid uint32) bool {
|
|
|
|
for _, item := range uids {
|
|
|
|
if item == uid {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|