221 lines
4.8 KiB
Go
221 lines
4.8 KiB
Go
package msg
|
|
|
|
import (
|
|
"bytes"
|
|
"errors"
|
|
"fmt"
|
|
"io"
|
|
"strings"
|
|
|
|
"git.sr.ht/~sircmpwn/getopt"
|
|
|
|
"git.sr.ht/~sircmpwn/aerc/lib"
|
|
"git.sr.ht/~sircmpwn/aerc/lib/format"
|
|
"git.sr.ht/~sircmpwn/aerc/models"
|
|
"git.sr.ht/~sircmpwn/aerc/widgets"
|
|
)
|
|
|
|
type reply struct{}
|
|
|
|
func init() {
|
|
register(reply{})
|
|
}
|
|
|
|
func (reply) Aliases() []string {
|
|
return []string{"reply"}
|
|
}
|
|
|
|
func (reply) Complete(aerc *widgets.Aerc, args []string) []string {
|
|
return nil
|
|
}
|
|
|
|
func (reply) Execute(aerc *widgets.Aerc, args []string) error {
|
|
opts, optind, err := getopt.Getopts(args, "aqT:")
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if optind != len(args) {
|
|
return errors.New("Usage: reply [-aq -T <template>]")
|
|
}
|
|
var (
|
|
quote bool
|
|
replyAll bool
|
|
template string
|
|
)
|
|
for _, opt := range opts {
|
|
switch opt.Option {
|
|
case 'a':
|
|
replyAll = true
|
|
case 'q':
|
|
quote = true
|
|
case 'T':
|
|
template = opt.Value
|
|
}
|
|
}
|
|
|
|
widget := aerc.SelectedTab().(widgets.ProvidesMessage)
|
|
acct := widget.SelectedAccount()
|
|
|
|
if acct == nil {
|
|
return errors.New("No account selected")
|
|
}
|
|
conf := acct.AccountConfig()
|
|
from, err := format.ParseAddress(conf.From)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
alias_of_us, err := format.ParseAddressList(conf.Aliases)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
store := widget.Store()
|
|
if store == nil {
|
|
return errors.New("Cannot perform action. Messages still loading")
|
|
}
|
|
msg, err := widget.SelectedMessage()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
var (
|
|
to []*models.Address
|
|
cc []*models.Address
|
|
)
|
|
if args[0] == "reply" {
|
|
if len(msg.Envelope.ReplyTo) != 0 {
|
|
to = msg.Envelope.ReplyTo
|
|
} else {
|
|
to = msg.Envelope.From
|
|
}
|
|
|
|
// figure out the sending from address if we have aliases
|
|
if len(alias_of_us) != 0 {
|
|
allRecipients := append(msg.Envelope.To, msg.Envelope.Cc...)
|
|
outer:
|
|
for _, addr := range allRecipients {
|
|
if addr.Address == from.Address {
|
|
from = addr
|
|
break
|
|
}
|
|
for _, alias := range alias_of_us {
|
|
if addr.Address == alias.Address {
|
|
from = addr
|
|
break outer
|
|
}
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
isMainRecipient := func(a *models.Address) bool {
|
|
for _, ta := range to {
|
|
if ta.Address == a.Address {
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
if replyAll {
|
|
for _, addr := range msg.Envelope.Cc {
|
|
//dedupe stuff from the to/from headers
|
|
if isMainRecipient(addr) || addr.Address == from.Address {
|
|
continue
|
|
}
|
|
cc = append(cc, addr)
|
|
}
|
|
envTos := make([]*models.Address, 0, len(msg.Envelope.To))
|
|
for _, addr := range msg.Envelope.To {
|
|
if addr.Address == from.Address {
|
|
continue
|
|
}
|
|
envTos = append(envTos, addr)
|
|
}
|
|
to = append(to, envTos...)
|
|
}
|
|
}
|
|
|
|
var subject string
|
|
if !strings.HasPrefix(strings.ToLower(msg.Envelope.Subject), "re: ") {
|
|
subject = "Re: " + msg.Envelope.Subject
|
|
} else {
|
|
subject = msg.Envelope.Subject
|
|
}
|
|
|
|
defaults := map[string]string{
|
|
"To": format.FormatAddresses(to),
|
|
"Cc": format.FormatAddresses(cc),
|
|
"From": from.Format(),
|
|
"Subject": subject,
|
|
"In-Reply-To": msg.Envelope.MessageId,
|
|
}
|
|
original := models.OriginalMail{}
|
|
|
|
addTab := func() error {
|
|
if template != "" {
|
|
original.From = format.FormatAddresses(msg.Envelope.From)
|
|
original.Date = msg.Envelope.Date
|
|
}
|
|
|
|
composer, err := widgets.NewComposer(aerc, acct, aerc.Config(),
|
|
acct.AccountConfig(), acct.Worker(), template, defaults, original)
|
|
if err != nil {
|
|
aerc.PushError("Error: " + err.Error())
|
|
return err
|
|
}
|
|
|
|
if args[0] == "reply" {
|
|
composer.FocusTerminal()
|
|
}
|
|
|
|
tab := aerc.NewTab(composer, subject)
|
|
composer.OnHeaderChange("Subject", func(subject string) {
|
|
if subject == "" {
|
|
tab.Name = "New email"
|
|
} else {
|
|
tab.Name = subject
|
|
}
|
|
tab.Content.Invalidate()
|
|
})
|
|
|
|
composer.OnClose(func(c *widgets.Composer) {
|
|
store.Answered([]uint32{msg.Uid}, c.Sent(), nil)
|
|
})
|
|
|
|
return nil
|
|
}
|
|
|
|
if quote {
|
|
if template == "" {
|
|
template = aerc.Config().Templates.QuotedReply
|
|
}
|
|
|
|
part := lib.FindPlaintext(msg.BodyStructure, nil)
|
|
if part == nil {
|
|
// mkey... let's get the first thing that isn't a container
|
|
// if that's still nil it's either not a multipart msg (ok) or
|
|
// broken (containers only)
|
|
part = lib.FindFirstNonMultipart(msg.BodyStructure, nil)
|
|
}
|
|
store.FetchBodyPart(msg.Uid, part, func(reader io.Reader) {
|
|
buf := new(bytes.Buffer)
|
|
buf.ReadFrom(reader)
|
|
original.Text = buf.String()
|
|
if len(msg.BodyStructure.Parts) == 0 {
|
|
original.MIMEType = fmt.Sprintf("%s/%s",
|
|
msg.BodyStructure.MIMEType, msg.BodyStructure.MIMESubType)
|
|
} else {
|
|
// TODO: still will be "multipart/mixed" for mixed mails with
|
|
// attachments, fix this after aerc could handle responding to
|
|
// such mails
|
|
original.MIMEType = fmt.Sprintf("%s/%s",
|
|
msg.BodyStructure.Parts[0].MIMEType,
|
|
msg.BodyStructure.Parts[0].MIMESubType)
|
|
}
|
|
addTab()
|
|
})
|
|
return nil
|
|
} else {
|
|
return addTab()
|
|
}
|
|
}
|