2019-05-12 04:06:09 +00:00
|
|
|
package widgets
|
|
|
|
|
|
|
|
import (
|
2019-07-16 20:48:25 +00:00
|
|
|
"bufio"
|
2019-09-11 19:28:14 +00:00
|
|
|
"bytes"
|
|
|
|
"fmt"
|
2019-05-14 17:07:48 +00:00
|
|
|
"io"
|
2019-05-13 20:04:01 +00:00
|
|
|
"io/ioutil"
|
2019-07-16 20:48:25 +00:00
|
|
|
"mime"
|
|
|
|
"net/http"
|
2020-11-10 19:27:30 +00:00
|
|
|
"net/textproto"
|
2019-05-13 20:04:01 +00:00
|
|
|
"os"
|
2019-05-12 04:06:09 +00:00
|
|
|
"os/exec"
|
2019-07-16 20:48:25 +00:00
|
|
|
"path/filepath"
|
2019-08-04 00:23:08 +00:00
|
|
|
"strings"
|
2019-05-14 17:07:48 +00:00
|
|
|
"time"
|
2019-05-12 04:06:09 +00:00
|
|
|
|
2019-05-14 17:07:48 +00:00
|
|
|
"github.com/emersion/go-message/mail"
|
2020-11-30 22:07:03 +00:00
|
|
|
"github.com/gdamore/tcell/v2"
|
2019-05-12 04:06:09 +00:00
|
|
|
"github.com/mattn/go-runewidth"
|
2019-09-11 19:28:14 +00:00
|
|
|
"github.com/mitchellh/go-homedir"
|
2019-05-25 15:56:56 +00:00
|
|
|
"github.com/pkg/errors"
|
2019-05-12 04:06:09 +00:00
|
|
|
|
2021-11-05 09:19:46 +00:00
|
|
|
"git.sr.ht/~rjarry/aerc/completer"
|
|
|
|
"git.sr.ht/~rjarry/aerc/config"
|
|
|
|
"git.sr.ht/~rjarry/aerc/lib/format"
|
|
|
|
"git.sr.ht/~rjarry/aerc/lib/templates"
|
|
|
|
"git.sr.ht/~rjarry/aerc/lib/ui"
|
|
|
|
"git.sr.ht/~rjarry/aerc/models"
|
|
|
|
"git.sr.ht/~rjarry/aerc/worker/types"
|
2019-05-12 04:06:09 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
type Composer struct {
|
2020-11-10 19:27:30 +00:00
|
|
|
editors map[string]*headerEditor // indexes in lower case (from / cc / bcc)
|
|
|
|
header *mail.Header
|
|
|
|
parent models.OriginalMail // parent of current message, only set if reply
|
2019-05-12 04:06:09 +00:00
|
|
|
|
2020-04-24 09:42:21 +00:00
|
|
|
acctConfig *config.AccountConfig
|
|
|
|
config *config.AercConfig
|
|
|
|
acct *AccountView
|
|
|
|
aerc *Aerc
|
2019-05-13 20:04:01 +00:00
|
|
|
|
2019-12-12 15:22:28 +00:00
|
|
|
attachments []string
|
2019-07-16 20:48:25 +00:00
|
|
|
editor *Terminal
|
|
|
|
email *os.File
|
|
|
|
grid *ui.Grid
|
2020-11-03 05:56:12 +00:00
|
|
|
heditors *ui.Grid // from, to, cc display a user can jump to
|
2019-07-16 20:48:25 +00:00
|
|
|
review *reviewMessage
|
|
|
|
worker *types.Worker
|
2019-12-20 18:21:35 +00:00
|
|
|
completer *completer.Completer
|
2021-12-30 09:25:08 +00:00
|
|
|
sign bool
|
2021-12-30 09:25:09 +00:00
|
|
|
encrypt bool
|
2019-05-12 04:06:09 +00:00
|
|
|
|
2019-08-01 18:29:21 +00:00
|
|
|
layout HeaderLayout
|
2019-09-05 22:32:36 +00:00
|
|
|
focusable []ui.MouseableDrawableInteractive
|
2019-05-12 04:06:09 +00:00
|
|
|
focused int
|
2020-05-25 14:59:48 +00:00
|
|
|
sent bool
|
2019-08-18 09:33:15 +00:00
|
|
|
|
|
|
|
onClose []func(ti *Composer)
|
2019-09-05 22:32:36 +00:00
|
|
|
|
|
|
|
width int
|
2019-05-12 04:06:09 +00:00
|
|
|
}
|
|
|
|
|
2020-04-24 09:42:21 +00:00
|
|
|
func NewComposer(aerc *Aerc, acct *AccountView, conf *config.AercConfig,
|
2020-04-24 09:42:22 +00:00
|
|
|
acctConfig *config.AccountConfig, worker *types.Worker, template string,
|
2020-11-10 19:27:30 +00:00
|
|
|
h *mail.Header, orig models.OriginalMail) (*Composer, error) {
|
2019-07-22 23:29:07 +00:00
|
|
|
|
2020-11-10 19:27:30 +00:00
|
|
|
if h == nil {
|
|
|
|
h = new(mail.Header)
|
2019-07-22 23:29:07 +00:00
|
|
|
}
|
2020-11-10 19:27:30 +00:00
|
|
|
if fl, err := h.AddressList("from"); err != nil || fl == nil {
|
|
|
|
fl, err = mail.ParseAddressList(acctConfig.From)
|
|
|
|
// realistically this blows up way before us during the config loading
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if fl != nil {
|
|
|
|
h.SetAddressList("from", fl)
|
|
|
|
|
|
|
|
}
|
2019-07-22 23:29:07 +00:00
|
|
|
}
|
|
|
|
|
2020-11-10 19:27:30 +00:00
|
|
|
templateData := templates.ParseTemplateData(h, orig)
|
2019-12-20 18:21:35 +00:00
|
|
|
cmpl := completer.New(conf.Compose.AddressBookCmd, func(err error) {
|
2020-07-27 08:03:55 +00:00
|
|
|
aerc.PushError(
|
|
|
|
fmt.Sprintf("could not complete header: %v", err))
|
2019-12-20 18:21:35 +00:00
|
|
|
worker.Logger.Printf("could not complete header: %v", err)
|
|
|
|
}, aerc.Logger())
|
2019-07-22 23:29:07 +00:00
|
|
|
|
2019-05-13 20:04:01 +00:00
|
|
|
email, err := ioutil.TempFile("", "aerc-compose-*.eml")
|
|
|
|
if err != nil {
|
|
|
|
// TODO: handle this better
|
2019-11-03 12:51:14 +00:00
|
|
|
return nil, err
|
2019-05-13 20:04:01 +00:00
|
|
|
}
|
|
|
|
|
2019-05-13 20:24:05 +00:00
|
|
|
c := &Composer{
|
2020-04-24 09:42:21 +00:00
|
|
|
acct: acct,
|
|
|
|
acctConfig: acctConfig,
|
|
|
|
aerc: aerc,
|
|
|
|
config: conf,
|
2020-11-10 19:27:30 +00:00
|
|
|
header: h,
|
|
|
|
parent: orig,
|
2020-04-24 09:42:21 +00:00
|
|
|
email: email,
|
|
|
|
worker: worker,
|
2019-05-12 04:38:48 +00:00
|
|
|
// You have to backtab to get to "From", since you usually don't edit it
|
2019-05-13 20:04:01 +00:00
|
|
|
focused: 1,
|
2019-12-20 18:21:35 +00:00
|
|
|
completer: cmpl,
|
2019-05-12 04:06:09 +00:00
|
|
|
}
|
2019-11-03 12:51:14 +00:00
|
|
|
if err := c.AddTemplate(template, templateData); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2022-02-03 08:54:23 +00:00
|
|
|
c.buildComposeHeader(aerc, cmpl)
|
|
|
|
|
2020-04-23 18:55:18 +00:00
|
|
|
c.AddSignature()
|
2019-09-11 19:28:14 +00:00
|
|
|
|
2019-08-01 18:29:21 +00:00
|
|
|
c.updateGrid()
|
2019-05-26 15:58:14 +00:00
|
|
|
c.ShowTerminal()
|
2019-05-13 20:24:05 +00:00
|
|
|
|
2019-11-03 12:51:14 +00:00
|
|
|
return c, nil
|
2019-05-12 04:06:09 +00:00
|
|
|
}
|
|
|
|
|
2020-11-10 19:27:30 +00:00
|
|
|
func (c *Composer) buildComposeHeader(aerc *Aerc, cmpl *completer.Completer) {
|
|
|
|
|
|
|
|
c.layout = aerc.conf.Compose.HeaderLayout
|
|
|
|
c.editors = make(map[string]*headerEditor)
|
|
|
|
c.focusable = make([]ui.MouseableDrawableInteractive, 0)
|
2022-02-24 23:21:06 +00:00
|
|
|
uiConfig := aerc.SelectedAccountUiConfig()
|
2019-07-22 23:29:07 +00:00
|
|
|
|
2020-11-10 19:27:30 +00:00
|
|
|
for i, row := range c.layout {
|
|
|
|
for j, h := range row {
|
|
|
|
h = strings.ToLower(h)
|
|
|
|
c.layout[i][j] = h // normalize to lowercase
|
2022-02-24 23:21:06 +00:00
|
|
|
e := newHeaderEditor(h, c.header, uiConfig)
|
2020-07-27 08:03:55 +00:00
|
|
|
if aerc.conf.Ui.CompletionPopovers {
|
2022-02-24 23:21:06 +00:00
|
|
|
e.input.TabComplete(cmpl.ForHeader(h), uiConfig.CompletionDelay)
|
2019-12-20 18:21:35 +00:00
|
|
|
}
|
2020-11-10 19:27:30 +00:00
|
|
|
c.editors[h] = e
|
2019-07-22 23:29:07 +00:00
|
|
|
switch h {
|
2020-11-10 19:27:30 +00:00
|
|
|
case "from":
|
2019-07-22 23:29:07 +00:00
|
|
|
// Prepend From to support backtab
|
2020-11-10 19:27:30 +00:00
|
|
|
c.focusable = append([]ui.MouseableDrawableInteractive{e}, c.focusable...)
|
2019-07-22 23:29:07 +00:00
|
|
|
default:
|
2020-11-10 19:27:30 +00:00
|
|
|
c.focusable = append(c.focusable, e)
|
2019-07-22 23:29:07 +00:00
|
|
|
}
|
|
|
|
}
|
2019-05-16 14:49:50 +00:00
|
|
|
}
|
2019-07-22 23:29:07 +00:00
|
|
|
|
2020-11-10 19:27:30 +00:00
|
|
|
// Add Cc/Bcc editors to layout if present in header and not already visible
|
|
|
|
for _, h := range []string{"cc", "bcc"} {
|
|
|
|
if c.header.Has(h) {
|
|
|
|
if _, ok := c.editors[h]; !ok {
|
2022-02-24 23:21:06 +00:00
|
|
|
e := newHeaderEditor(h, c.header, uiConfig)
|
2020-07-27 08:03:55 +00:00
|
|
|
if aerc.conf.Ui.CompletionPopovers {
|
2022-02-24 23:21:06 +00:00
|
|
|
e.input.TabComplete(cmpl.ForHeader(h), uiConfig.CompletionDelay)
|
2019-12-20 18:21:35 +00:00
|
|
|
}
|
2020-11-10 19:27:30 +00:00
|
|
|
c.editors[h] = e
|
|
|
|
c.focusable = append(c.focusable, e)
|
|
|
|
c.layout = append(c.layout, []string{h})
|
2019-07-22 23:29:07 +00:00
|
|
|
}
|
|
|
|
}
|
2019-05-16 14:49:50 +00:00
|
|
|
}
|
2019-07-22 23:29:07 +00:00
|
|
|
|
2020-11-10 19:27:30 +00:00
|
|
|
// load current header values into all editors
|
|
|
|
for _, e := range c.editors {
|
|
|
|
e.loadValue()
|
2019-05-16 14:49:50 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-25 14:59:48 +00:00
|
|
|
func (c *Composer) SetSent() {
|
|
|
|
c.sent = true
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Composer) Sent() bool {
|
|
|
|
return c.sent
|
|
|
|
}
|
|
|
|
|
2021-12-30 09:25:08 +00:00
|
|
|
func (c *Composer) SetSign(sign bool) *Composer {
|
|
|
|
c.sign = sign
|
|
|
|
return c
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Composer) Sign() bool {
|
|
|
|
return c.sign
|
|
|
|
}
|
|
|
|
|
2021-12-30 09:25:09 +00:00
|
|
|
func (c *Composer) SetEncrypt(encrypt bool) *Composer {
|
|
|
|
c.encrypt = encrypt
|
|
|
|
return c
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Composer) Encrypt() bool {
|
|
|
|
return c.encrypt
|
|
|
|
}
|
|
|
|
|
2019-05-16 16:39:22 +00:00
|
|
|
// Note: this does not reload the editor. You must call this before the first
|
|
|
|
// Draw() call.
|
|
|
|
func (c *Composer) SetContents(reader io.Reader) *Composer {
|
2019-09-03 19:34:08 +00:00
|
|
|
c.email.Seek(0, io.SeekStart)
|
2019-05-16 16:39:22 +00:00
|
|
|
io.Copy(c.email, reader)
|
2019-05-16 18:09:57 +00:00
|
|
|
c.email.Sync()
|
2019-09-03 19:34:08 +00:00
|
|
|
c.email.Seek(0, io.SeekStart)
|
2019-05-16 16:39:22 +00:00
|
|
|
return c
|
|
|
|
}
|
|
|
|
|
2019-09-11 19:28:14 +00:00
|
|
|
func (c *Composer) AppendContents(reader io.Reader) {
|
|
|
|
c.email.Seek(0, io.SeekEnd)
|
|
|
|
io.Copy(c.email, reader)
|
|
|
|
c.email.Sync()
|
|
|
|
}
|
|
|
|
|
2019-11-03 12:51:14 +00:00
|
|
|
func (c *Composer) AddTemplate(template string, data interface{}) error {
|
|
|
|
if template == "" {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-11-10 16:00:21 +00:00
|
|
|
templateText, err := templates.ParseTemplateFromFile(
|
|
|
|
template, c.config.Templates.TemplateDirs, data)
|
2019-11-03 12:51:14 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2020-04-23 18:55:18 +00:00
|
|
|
mr, err := mail.CreateReader(templateText)
|
2019-12-05 15:23:21 +00:00
|
|
|
if err != nil {
|
2020-04-23 18:55:18 +00:00
|
|
|
return fmt.Errorf("Template loading failed: %v", err)
|
2019-12-05 15:23:21 +00:00
|
|
|
}
|
|
|
|
|
2020-11-10 19:27:30 +00:00
|
|
|
// copy the headers contained in the template to the compose headers
|
2020-04-23 18:55:18 +00:00
|
|
|
hf := mr.Header.Fields()
|
|
|
|
for hf.Next() {
|
2020-11-10 19:27:30 +00:00
|
|
|
c.header.Set(hf.Key(), hf.Value())
|
2019-12-05 15:23:21 +00:00
|
|
|
}
|
|
|
|
|
2020-04-23 18:55:18 +00:00
|
|
|
part, err := mr.NextPart()
|
2019-12-05 15:23:21 +00:00
|
|
|
if err != nil {
|
2020-04-23 18:55:18 +00:00
|
|
|
return fmt.Errorf("Could not get body of template: %v", err)
|
2019-12-05 15:23:21 +00:00
|
|
|
}
|
|
|
|
|
2020-04-23 18:55:18 +00:00
|
|
|
c.AppendContents(part.Body)
|
2019-11-03 12:51:14 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-09-11 19:28:14 +00:00
|
|
|
func (c *Composer) AddSignature() {
|
|
|
|
var signature []byte
|
2020-04-24 09:42:21 +00:00
|
|
|
if c.acctConfig.SignatureCmd != "" {
|
2019-09-11 19:28:14 +00:00
|
|
|
var err error
|
|
|
|
signature, err = c.readSignatureFromCmd()
|
|
|
|
if err != nil {
|
|
|
|
signature = c.readSignatureFromFile()
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
signature = c.readSignatureFromFile()
|
|
|
|
}
|
|
|
|
c.AppendContents(bytes.NewReader(signature))
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Composer) readSignatureFromCmd() ([]byte, error) {
|
2020-04-24 09:42:21 +00:00
|
|
|
sigCmd := c.acctConfig.SignatureCmd
|
2019-09-11 19:28:14 +00:00
|
|
|
cmd := exec.Command("sh", "-c", sigCmd)
|
|
|
|
signature, err := cmd.Output()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return signature, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Composer) readSignatureFromFile() []byte {
|
2020-04-24 09:42:21 +00:00
|
|
|
sigFile := c.acctConfig.SignatureFile
|
2019-09-11 19:28:14 +00:00
|
|
|
if sigFile == "" {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
sigFile, err := homedir.Expand(sigFile)
|
|
|
|
if err != nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
signature, err := ioutil.ReadFile(sigFile)
|
|
|
|
if err != nil {
|
2020-07-27 08:03:55 +00:00
|
|
|
c.aerc.PushError(
|
|
|
|
fmt.Sprintf(" Error loading signature from file: %v", sigFile))
|
2019-09-11 19:28:14 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return signature
|
|
|
|
}
|
|
|
|
|
2019-05-16 16:15:34 +00:00
|
|
|
func (c *Composer) FocusTerminal() *Composer {
|
2019-05-26 15:58:14 +00:00
|
|
|
if c.editor == nil {
|
|
|
|
return c
|
|
|
|
}
|
2019-05-16 16:15:34 +00:00
|
|
|
c.focusable[c.focused].Focus(false)
|
2019-07-22 23:29:07 +00:00
|
|
|
c.focused = len(c.editors)
|
2019-05-16 16:15:34 +00:00
|
|
|
c.focusable[c.focused].Focus(true)
|
|
|
|
return c
|
|
|
|
}
|
|
|
|
|
2019-07-22 23:29:07 +00:00
|
|
|
// OnHeaderChange registers an OnChange callback for the specified header.
|
|
|
|
func (c *Composer) OnHeaderChange(header string, fn func(subject string)) {
|
2020-11-10 19:27:30 +00:00
|
|
|
if editor, ok := c.editors[strings.ToLower(header)]; ok {
|
2019-07-22 23:29:07 +00:00
|
|
|
editor.OnChange(func() {
|
|
|
|
fn(editor.input.String())
|
|
|
|
})
|
|
|
|
}
|
2019-05-14 20:18:21 +00:00
|
|
|
}
|
|
|
|
|
2019-08-18 09:33:15 +00:00
|
|
|
func (c *Composer) OnClose(fn func(composer *Composer)) {
|
|
|
|
c.onClose = append(c.onClose, fn)
|
|
|
|
}
|
|
|
|
|
2019-05-12 04:06:09 +00:00
|
|
|
func (c *Composer) Draw(ctx *ui.Context) {
|
2019-09-05 22:32:36 +00:00
|
|
|
c.width = ctx.Width()
|
2019-05-12 04:06:09 +00:00
|
|
|
c.grid.Draw(ctx)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Composer) Invalidate() {
|
|
|
|
c.grid.Invalidate()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Composer) OnInvalidate(fn func(d ui.Drawable)) {
|
|
|
|
c.grid.OnInvalidate(func(_ ui.Drawable) {
|
|
|
|
fn(c)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2019-05-14 18:05:29 +00:00
|
|
|
func (c *Composer) Close() {
|
2019-08-18 09:33:15 +00:00
|
|
|
for _, onClose := range c.onClose {
|
|
|
|
onClose(c)
|
|
|
|
}
|
2019-05-14 18:05:29 +00:00
|
|
|
if c.email != nil {
|
|
|
|
path := c.email.Name()
|
|
|
|
c.email.Close()
|
|
|
|
os.Remove(path)
|
|
|
|
c.email = nil
|
|
|
|
}
|
|
|
|
if c.editor != nil {
|
|
|
|
c.editor.Destroy()
|
|
|
|
c.editor = nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-14 18:27:28 +00:00
|
|
|
func (c *Composer) Bindings() string {
|
|
|
|
if c.editor == nil {
|
|
|
|
return "compose::review"
|
|
|
|
} else if c.editor == c.focusable[c.focused] {
|
|
|
|
return "compose::editor"
|
|
|
|
} else {
|
|
|
|
return "compose"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-12 04:06:09 +00:00
|
|
|
func (c *Composer) Event(event tcell.Event) bool {
|
2019-07-16 15:33:47 +00:00
|
|
|
if c.editor != nil {
|
|
|
|
return c.focusable[c.focused].Event(event)
|
|
|
|
}
|
|
|
|
return false
|
2019-05-12 04:06:09 +00:00
|
|
|
}
|
|
|
|
|
2019-11-06 03:43:45 +00:00
|
|
|
func (c *Composer) MouseEvent(localX int, localY int, event tcell.Event) {
|
|
|
|
c.grid.MouseEvent(localX, localY, event)
|
|
|
|
for _, e := range c.focusable {
|
|
|
|
he, ok := e.(*headerEditor)
|
|
|
|
if ok && he.focused {
|
2022-03-21 23:23:47 +00:00
|
|
|
c.FocusEditor(he.name)
|
2019-11-06 03:43:45 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-12 04:06:09 +00:00
|
|
|
func (c *Composer) Focus(focus bool) {
|
2019-05-12 04:38:48 +00:00
|
|
|
c.focusable[c.focused].Focus(focus)
|
2019-05-12 04:06:09 +00:00
|
|
|
}
|
|
|
|
|
2019-05-14 17:07:48 +00:00
|
|
|
func (c *Composer) Config() *config.AccountConfig {
|
2020-04-24 09:42:21 +00:00
|
|
|
return c.acctConfig
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Composer) Account() *AccountView {
|
2019-05-26 15:58:14 +00:00
|
|
|
return c.acct
|
2019-05-14 17:07:48 +00:00
|
|
|
}
|
|
|
|
|
2019-05-15 23:41:21 +00:00
|
|
|
func (c *Composer) Worker() *types.Worker {
|
|
|
|
return c.worker
|
|
|
|
}
|
|
|
|
|
2020-11-10 19:27:30 +00:00
|
|
|
//PrepareHeader finalizes the header, adding the value from the editors
|
|
|
|
func (c *Composer) PrepareHeader() (*mail.Header, error) {
|
|
|
|
for _, editor := range c.editors {
|
|
|
|
editor.storeValue()
|
2019-05-14 17:07:48 +00:00
|
|
|
}
|
2019-07-22 23:29:07 +00:00
|
|
|
|
2020-11-10 19:27:30 +00:00
|
|
|
// control headers not normally set by the user
|
|
|
|
// repeated calls to PrepareHeader should be a noop
|
|
|
|
if !c.header.Has("Message-Id") {
|
|
|
|
err := c.header.GenerateMessageID()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
2019-06-21 18:33:09 +00:00
|
|
|
}
|
|
|
|
}
|
2020-11-10 19:27:30 +00:00
|
|
|
if !c.header.Has("Date") {
|
|
|
|
c.header.SetDate(time.Now())
|
|
|
|
}
|
|
|
|
return c.header, nil
|
2019-05-14 18:05:29 +00:00
|
|
|
}
|
|
|
|
|
2021-12-30 09:25:08 +00:00
|
|
|
func getSenderEmail(c *Composer) (string, error) {
|
|
|
|
// add the from: field also to the 'recipients' list
|
|
|
|
if c.acctConfig.From == "" {
|
|
|
|
return "", errors.New("No 'From' configured for this account")
|
|
|
|
}
|
|
|
|
from, err := mail.ParseAddress(c.acctConfig.From)
|
|
|
|
if err != nil {
|
|
|
|
return "", errors.Wrap(err, "ParseAddress(config.From)")
|
|
|
|
}
|
|
|
|
return from.Address, nil
|
|
|
|
}
|
|
|
|
|
2021-12-30 09:25:09 +00:00
|
|
|
func getRecipientsEmail(c *Composer) ([]string, error) {
|
|
|
|
h, err := c.PrepareHeader()
|
|
|
|
if err != nil {
|
|
|
|
return nil, errors.Wrap(err, "PrepareHeader")
|
|
|
|
}
|
|
|
|
|
|
|
|
// collect all 'recipients' from header (to:, cc:, bcc:)
|
|
|
|
rcpts := make(map[string]bool)
|
|
|
|
for _, key := range []string{"to", "cc", "bcc"} {
|
|
|
|
list, err := h.AddressList(key)
|
|
|
|
if err != nil {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
for _, entry := range list {
|
|
|
|
if entry != nil {
|
|
|
|
rcpts[entry.Address] = true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// return email addresses as string slice
|
|
|
|
results := []string{}
|
|
|
|
for email, _ := range rcpts {
|
|
|
|
results = append(results, email)
|
|
|
|
}
|
|
|
|
return results, nil
|
|
|
|
}
|
|
|
|
|
2019-05-14 18:05:29 +00:00
|
|
|
func (c *Composer) WriteMessage(header *mail.Header, writer io.Writer) error {
|
2019-07-22 23:29:07 +00:00
|
|
|
if err := c.reloadEmail(); err != nil {
|
|
|
|
return err
|
2019-06-27 09:06:50 +00:00
|
|
|
}
|
2019-07-16 20:48:25 +00:00
|
|
|
|
2021-12-30 09:25:09 +00:00
|
|
|
if c.sign || c.encrypt {
|
2019-07-16 20:48:25 +00:00
|
|
|
|
2021-12-30 09:25:08 +00:00
|
|
|
var signedHeader mail.Header
|
|
|
|
signedHeader.SetContentType("text/plain", nil)
|
|
|
|
|
|
|
|
var buf bytes.Buffer
|
|
|
|
var cleartext io.WriteCloser
|
2021-12-30 09:25:09 +00:00
|
|
|
var err error
|
2019-07-31 17:33:50 +00:00
|
|
|
|
2022-04-25 13:30:44 +00:00
|
|
|
signer := ""
|
2021-12-30 09:25:09 +00:00
|
|
|
if c.sign {
|
2022-04-25 13:30:44 +00:00
|
|
|
if c.acctConfig.PgpKeyId != "" {
|
|
|
|
signer = c.acctConfig.PgpKeyId
|
|
|
|
} else {
|
|
|
|
signer, err = getSenderEmail(c)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2021-12-30 09:25:09 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if c.encrypt {
|
|
|
|
rcpts, err := getRecipientsEmail(c)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2022-04-25 13:30:44 +00:00
|
|
|
cleartext, err = c.aerc.Crypto.Encrypt(&buf, rcpts, signer, c.aerc.DecryptKeys, header)
|
2021-12-30 09:25:09 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
} else {
|
2022-04-25 13:30:44 +00:00
|
|
|
cleartext, err = c.aerc.Crypto.Sign(&buf, signer, c.aerc.DecryptKeys, header)
|
2021-12-30 09:25:09 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2019-07-31 17:33:50 +00:00
|
|
|
}
|
2021-12-30 09:25:08 +00:00
|
|
|
|
|
|
|
err = writeMsgImpl(c, &signedHeader, cleartext)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
cleartext.Close()
|
|
|
|
io.Copy(writer, &buf)
|
|
|
|
return nil
|
|
|
|
|
|
|
|
} else {
|
|
|
|
return writeMsgImpl(c, header, writer)
|
2019-07-31 17:33:50 +00:00
|
|
|
}
|
2021-12-30 09:25:08 +00:00
|
|
|
}
|
2019-07-31 17:33:50 +00:00
|
|
|
|
2021-12-30 09:25:08 +00:00
|
|
|
func writeMsgImpl(c *Composer, header *mail.Header, writer io.Writer) error {
|
|
|
|
if len(c.attachments) == 0 {
|
|
|
|
// no attachements
|
|
|
|
return writeInlineBody(header, c.email, writer)
|
|
|
|
} else {
|
|
|
|
// with attachements
|
|
|
|
w, err := mail.CreateWriter(writer, *header)
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "CreateWriter")
|
|
|
|
}
|
|
|
|
if err := writeMultipartBody(c.email, w); err != nil {
|
|
|
|
return errors.Wrap(err, "writeMultipartBody")
|
|
|
|
}
|
|
|
|
for _, a := range c.attachments {
|
|
|
|
if err := writeAttachment(a, w); err != nil {
|
|
|
|
return errors.Wrap(err, "writeAttachment")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
w.Close()
|
|
|
|
}
|
2019-07-31 17:33:50 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func writeInlineBody(header *mail.Header, body io.Reader, writer io.Writer) error {
|
|
|
|
header.SetContentType("text/plain", map[string]string{"charset": "UTF-8"})
|
|
|
|
w, err := mail.CreateSingleInlineWriter(writer, *header)
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "CreateSingleInlineWriter")
|
|
|
|
}
|
|
|
|
defer w.Close()
|
|
|
|
if _, err := io.Copy(w, body); err != nil {
|
|
|
|
return errors.Wrap(err, "io.Copy")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// write the message body to the multipart message
|
|
|
|
func writeMultipartBody(body io.Reader, w *mail.Writer) error {
|
2019-07-16 20:48:25 +00:00
|
|
|
bh := mail.InlineHeader{}
|
|
|
|
bh.SetContentType("text/plain", map[string]string{"charset": "UTF-8"})
|
|
|
|
|
|
|
|
bi, err := w.CreateInline()
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "CreateInline")
|
|
|
|
}
|
|
|
|
defer bi.Close()
|
|
|
|
|
|
|
|
bw, err := bi.CreatePart(bh)
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "CreatePart")
|
|
|
|
}
|
|
|
|
defer bw.Close()
|
2019-07-31 17:33:50 +00:00
|
|
|
if _, err := io.Copy(bw, body); err != nil {
|
2019-05-25 15:56:56 +00:00
|
|
|
return errors.Wrap(err, "io.Copy")
|
|
|
|
}
|
|
|
|
return nil
|
2019-05-14 17:07:48 +00:00
|
|
|
}
|
|
|
|
|
2019-07-16 20:48:25 +00:00
|
|
|
// write the attachment specified by path to the message
|
|
|
|
func writeAttachment(path string, writer *mail.Writer) error {
|
|
|
|
f, err := os.Open(path)
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "os.Open")
|
|
|
|
}
|
|
|
|
defer f.Close()
|
|
|
|
|
|
|
|
reader := bufio.NewReader(f)
|
|
|
|
|
2020-12-15 20:48:25 +00:00
|
|
|
// if we have an extension, prefer that instead of trying to sniff the header.
|
|
|
|
// That's generally more accurate than sniffing as lots of things are zip files
|
|
|
|
// under the hood, e.g. most office file types
|
|
|
|
ext := filepath.Ext(path)
|
|
|
|
var mimeString string
|
|
|
|
if mimeString = mime.TypeByExtension(ext); mimeString != "" {
|
|
|
|
// found it in the DB
|
|
|
|
} else {
|
|
|
|
// Sniff the mime type instead
|
|
|
|
// http.DetectContentType only cares about the first 512 bytes
|
|
|
|
head, err := reader.Peek(512)
|
|
|
|
if err != nil && err != io.EOF {
|
|
|
|
return errors.Wrap(err, "Peek")
|
|
|
|
}
|
|
|
|
mimeString = http.DetectContentType(head)
|
2019-07-16 20:48:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// mimeString can contain type and params (like text encoding),
|
|
|
|
// so we need to break them apart before passing them to the headers
|
|
|
|
mimeType, params, err := mime.ParseMediaType(mimeString)
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "ParseMediaType")
|
|
|
|
}
|
2020-12-15 20:48:25 +00:00
|
|
|
filename := filepath.Base(path)
|
2019-07-16 20:48:25 +00:00
|
|
|
params["name"] = filename
|
|
|
|
|
|
|
|
// set header fields
|
|
|
|
ah := mail.AttachmentHeader{}
|
|
|
|
ah.SetContentType(mimeType, params)
|
|
|
|
// setting the filename auto sets the content disposition
|
|
|
|
ah.SetFilename(filename)
|
|
|
|
|
|
|
|
aw, err := writer.CreateAttachment(ah)
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "CreateAttachment")
|
|
|
|
}
|
|
|
|
defer aw.Close()
|
|
|
|
|
|
|
|
if _, err := reader.WriteTo(aw); err != nil {
|
|
|
|
return errors.Wrap(err, "reader.WriteTo")
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-07-27 14:38:53 +00:00
|
|
|
func (c *Composer) GetAttachments() []string {
|
|
|
|
return c.attachments
|
|
|
|
}
|
|
|
|
|
2019-07-16 20:48:25 +00:00
|
|
|
func (c *Composer) AddAttachment(path string) {
|
|
|
|
c.attachments = append(c.attachments, path)
|
2019-07-27 14:38:53 +00:00
|
|
|
c.resetReview()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Composer) DeleteAttachment(path string) error {
|
|
|
|
for i, a := range c.attachments {
|
|
|
|
if a == path {
|
|
|
|
c.attachments = append(c.attachments[:i], c.attachments[i+1:]...)
|
|
|
|
c.resetReview()
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return errors.New("attachment does not exist")
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Composer) resetReview() {
|
2019-07-16 20:48:25 +00:00
|
|
|
if c.review != nil {
|
|
|
|
c.grid.RemoveChild(c.review)
|
|
|
|
c.review = newReviewMessage(c, nil)
|
2020-03-03 21:57:21 +00:00
|
|
|
c.grid.AddChild(c.review).At(2, 0)
|
2019-07-16 20:48:25 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-06 03:43:45 +00:00
|
|
|
func (c *Composer) termEvent(event tcell.Event) bool {
|
|
|
|
switch event := event.(type) {
|
|
|
|
case *tcell.EventMouse:
|
|
|
|
switch event.Buttons() {
|
|
|
|
case tcell.Button1:
|
|
|
|
c.FocusTerminal()
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2019-05-13 20:24:05 +00:00
|
|
|
func (c *Composer) termClosed(err error) {
|
|
|
|
c.grid.RemoveChild(c.editor)
|
2019-05-26 15:58:14 +00:00
|
|
|
c.review = newReviewMessage(c, err)
|
2020-03-03 21:57:21 +00:00
|
|
|
c.grid.AddChild(c.review).At(2, 0)
|
2019-05-13 20:24:05 +00:00
|
|
|
c.editor.Destroy()
|
2019-05-14 18:05:29 +00:00
|
|
|
c.editor = nil
|
2019-05-26 15:58:14 +00:00
|
|
|
c.focusable = c.focusable[:len(c.focusable)-1]
|
|
|
|
if c.focused >= len(c.focusable) {
|
|
|
|
c.focused = len(c.focusable) - 1
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Composer) ShowTerminal() {
|
|
|
|
if c.editor != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if c.review != nil {
|
|
|
|
c.grid.RemoveChild(c.review)
|
|
|
|
}
|
|
|
|
editorName := c.config.Compose.Editor
|
|
|
|
if editorName == "" {
|
|
|
|
editorName = os.Getenv("EDITOR")
|
|
|
|
}
|
|
|
|
if editorName == "" {
|
|
|
|
editorName = "vi"
|
|
|
|
}
|
2019-06-07 14:15:35 +00:00
|
|
|
editor := exec.Command("/bin/sh", "-c", editorName+" "+c.email.Name())
|
2019-05-26 15:58:14 +00:00
|
|
|
c.editor, _ = NewTerminal(editor) // TODO: handle error
|
2019-11-06 03:43:45 +00:00
|
|
|
c.editor.OnEvent = c.termEvent
|
2019-05-26 15:58:14 +00:00
|
|
|
c.editor.OnClose = c.termClosed
|
2020-03-03 21:57:21 +00:00
|
|
|
c.grid.AddChild(c.editor).At(2, 0)
|
2019-05-26 15:58:14 +00:00
|
|
|
c.focusable = append(c.focusable, c.editor)
|
2019-05-13 20:24:05 +00:00
|
|
|
}
|
|
|
|
|
2019-05-12 15:21:28 +00:00
|
|
|
func (c *Composer) PrevField() {
|
|
|
|
c.focusable[c.focused].Focus(false)
|
|
|
|
c.focused--
|
|
|
|
if c.focused == -1 {
|
|
|
|
c.focused = len(c.focusable) - 1
|
|
|
|
}
|
|
|
|
c.focusable[c.focused].Focus(true)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Composer) NextField() {
|
|
|
|
c.focusable[c.focused].Focus(false)
|
|
|
|
c.focused = (c.focused + 1) % len(c.focusable)
|
|
|
|
c.focusable[c.focused].Focus(true)
|
|
|
|
}
|
|
|
|
|
2022-03-21 23:23:47 +00:00
|
|
|
func (c *Composer) FocusEditor(editor string) {
|
|
|
|
editor = strings.ToLower(editor)
|
2019-08-04 04:09:13 +00:00
|
|
|
c.focusable[c.focused].Focus(false)
|
2022-03-21 23:23:47 +00:00
|
|
|
for i, f := range c.focusable {
|
|
|
|
e := f.(*headerEditor)
|
|
|
|
if strings.ToLower(e.name) == editor {
|
2019-08-04 04:09:13 +00:00
|
|
|
c.focused = i
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
c.focusable[c.focused].Focus(true)
|
|
|
|
}
|
|
|
|
|
2019-08-01 18:29:21 +00:00
|
|
|
// AddEditor appends a new header editor to the compose window.
|
2019-08-04 00:23:08 +00:00
|
|
|
func (c *Composer) AddEditor(header string, value string, appendHeader bool) {
|
2020-11-10 19:27:30 +00:00
|
|
|
var editor *headerEditor
|
|
|
|
header = strings.ToLower(header)
|
|
|
|
if e, ok := c.editors[header]; ok {
|
|
|
|
e.storeValue() // flush modifications from the user to the header
|
|
|
|
editor = e
|
|
|
|
} else {
|
2022-02-24 23:21:06 +00:00
|
|
|
uiConfig := c.aerc.SelectedAccountUiConfig()
|
|
|
|
e := newHeaderEditor(header, c.header, uiConfig)
|
|
|
|
if uiConfig.CompletionPopovers {
|
|
|
|
e.input.TabComplete(c.completer.ForHeader(header), uiConfig.CompletionDelay)
|
2019-08-04 00:23:08 +00:00
|
|
|
}
|
2020-11-10 19:27:30 +00:00
|
|
|
c.editors[header] = e
|
|
|
|
c.layout = append(c.layout, []string{header})
|
|
|
|
// Insert focus of new editor before terminal editor
|
|
|
|
c.focusable = append(
|
|
|
|
c.focusable[:len(c.focusable)-1],
|
|
|
|
e,
|
|
|
|
c.focusable[len(c.focusable)-1],
|
|
|
|
)
|
|
|
|
editor = e
|
|
|
|
}
|
|
|
|
|
|
|
|
if appendHeader {
|
|
|
|
currVal := editor.input.String()
|
|
|
|
if currVal != "" {
|
|
|
|
value = strings.TrimSpace(currVal) + ", " + value
|
2019-08-04 04:09:13 +00:00
|
|
|
}
|
2019-08-01 18:29:21 +00:00
|
|
|
}
|
2020-11-10 19:27:30 +00:00
|
|
|
if value != "" || appendHeader {
|
|
|
|
c.editors[header].input.Set(value)
|
|
|
|
editor.storeValue()
|
|
|
|
}
|
2019-08-04 04:09:13 +00:00
|
|
|
if value == "" {
|
2022-03-21 23:23:47 +00:00
|
|
|
c.FocusEditor(c.editors[header].name)
|
2019-08-04 04:09:13 +00:00
|
|
|
}
|
2020-11-10 19:27:30 +00:00
|
|
|
c.updateGrid()
|
2019-08-01 18:29:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// updateGrid should be called when the underlying header layout is changed.
|
|
|
|
func (c *Composer) updateGrid() {
|
2020-11-03 05:56:12 +00:00
|
|
|
heditors, height := c.layout.grid(
|
2020-11-10 19:27:30 +00:00
|
|
|
func(h string) ui.Drawable {
|
|
|
|
return c.editors[h]
|
|
|
|
},
|
2019-08-01 18:29:21 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
if c.grid == nil {
|
2020-05-31 11:37:46 +00:00
|
|
|
c.grid = ui.NewGrid().Columns([]ui.GridSpec{
|
2022-03-18 08:53:02 +00:00
|
|
|
{Strategy: ui.SIZE_WEIGHT, Size: ui.Const(1)},
|
2020-05-31 11:37:46 +00:00
|
|
|
})
|
2019-08-01 18:29:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
c.grid.Rows([]ui.GridSpec{
|
2022-03-18 08:53:02 +00:00
|
|
|
{Strategy: ui.SIZE_EXACT, Size: ui.Const(height)},
|
|
|
|
{Strategy: ui.SIZE_EXACT, Size: ui.Const(1)},
|
|
|
|
{Strategy: ui.SIZE_WEIGHT, Size: ui.Const(1)},
|
2019-08-01 18:29:21 +00:00
|
|
|
})
|
|
|
|
|
2020-11-03 05:56:12 +00:00
|
|
|
if c.heditors != nil {
|
|
|
|
c.grid.RemoveChild(c.heditors)
|
2019-08-01 18:29:21 +00:00
|
|
|
}
|
2021-10-26 20:42:07 +00:00
|
|
|
borderStyle := c.config.Ui.GetStyle(config.STYLE_BORDER)
|
2021-11-29 22:48:35 +00:00
|
|
|
borderChar := c.config.Ui.BorderCharHorizontal
|
2020-11-03 05:56:12 +00:00
|
|
|
c.heditors = heditors
|
|
|
|
c.grid.AddChild(c.heditors).At(0, 0)
|
2021-11-29 22:48:35 +00:00
|
|
|
c.grid.AddChild(ui.NewFill(borderChar, borderStyle)).At(1, 0)
|
2019-08-01 18:29:21 +00:00
|
|
|
}
|
|
|
|
|
2019-07-22 23:29:07 +00:00
|
|
|
func (c *Composer) reloadEmail() error {
|
|
|
|
name := c.email.Name()
|
|
|
|
c.email.Close()
|
|
|
|
file, err := os.Open(name)
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "ReloadEmail")
|
|
|
|
}
|
|
|
|
c.email = file
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-05-13 20:24:05 +00:00
|
|
|
type headerEditor struct {
|
2020-07-27 08:03:55 +00:00
|
|
|
name string
|
2020-11-10 19:27:30 +00:00
|
|
|
header *mail.Header
|
2020-07-27 08:03:55 +00:00
|
|
|
focused bool
|
|
|
|
input *ui.TextInput
|
|
|
|
uiConfig config.UIConfig
|
2019-05-13 20:24:05 +00:00
|
|
|
}
|
|
|
|
|
2020-11-10 19:27:30 +00:00
|
|
|
func newHeaderEditor(name string, h *mail.Header,
|
|
|
|
uiConfig config.UIConfig) *headerEditor {
|
|
|
|
he := &headerEditor{
|
|
|
|
input: ui.NewTextInput("", uiConfig),
|
2020-07-27 08:03:55 +00:00
|
|
|
name: name,
|
2020-11-10 19:27:30 +00:00
|
|
|
header: h,
|
2020-07-27 08:03:55 +00:00
|
|
|
uiConfig: uiConfig,
|
2019-05-12 04:06:09 +00:00
|
|
|
}
|
2020-11-10 19:27:30 +00:00
|
|
|
he.loadValue()
|
|
|
|
return he
|
|
|
|
}
|
|
|
|
|
|
|
|
//extractHumanHeaderValue extracts the human readable string for key from the
|
|
|
|
//header. If a parsing error occurs the raw value is returned
|
|
|
|
func extractHumanHeaderValue(key string, h *mail.Header) string {
|
|
|
|
var val string
|
|
|
|
var err error
|
|
|
|
switch strings.ToLower(key) {
|
|
|
|
case "to", "from", "cc", "bcc":
|
|
|
|
var list []*mail.Address
|
|
|
|
list, err = h.AddressList(key)
|
|
|
|
val = format.FormatAddresses(list)
|
|
|
|
default:
|
|
|
|
val, err = h.Text(key)
|
|
|
|
}
|
|
|
|
if err != nil {
|
|
|
|
// if we can't parse it, show it raw
|
|
|
|
val = h.Get(key)
|
|
|
|
}
|
|
|
|
return val
|
|
|
|
}
|
|
|
|
|
|
|
|
//loadValue loads the value of he.name form the underlying header
|
|
|
|
//the value is decoded and meant for human consumption.
|
|
|
|
//decoding issues are ignored and return their raw values
|
|
|
|
func (he *headerEditor) loadValue() {
|
|
|
|
he.input.Set(extractHumanHeaderValue(he.name, he.header))
|
|
|
|
he.input.Invalidate()
|
|
|
|
}
|
|
|
|
|
|
|
|
//storeValue writes the current state back to the underlying header.
|
|
|
|
//errors are ignored
|
|
|
|
func (he *headerEditor) storeValue() {
|
|
|
|
val := he.input.String()
|
|
|
|
switch strings.ToLower(he.name) {
|
|
|
|
case "to", "from", "cc", "bcc":
|
|
|
|
list, err := mail.ParseAddressList(val)
|
|
|
|
if err == nil {
|
|
|
|
he.header.SetAddressList(he.name, list)
|
|
|
|
} else {
|
|
|
|
// garbage, but it'll blow up upon sending and the user can
|
|
|
|
// fix the issue
|
|
|
|
he.header.SetText(he.name, val)
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
he.header.SetText(he.name, val)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//setValue overwrites the current value of the header editor and flushes it
|
|
|
|
//to the underlying header
|
|
|
|
func (he *headerEditor) setValue(val string) {
|
|
|
|
he.input.Set(val)
|
|
|
|
he.storeValue()
|
2019-05-12 04:06:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (he *headerEditor) Draw(ctx *ui.Context) {
|
2021-01-13 07:08:09 +00:00
|
|
|
name := textproto.CanonicalMIMEHeaderKey(he.name)
|
|
|
|
// Extra character to put a blank cell between the header and the input
|
2021-11-05 09:34:10 +00:00
|
|
|
size := runewidth.StringWidth(name+":") + 1
|
2020-07-27 08:03:55 +00:00
|
|
|
defaultStyle := he.uiConfig.GetStyle(config.STYLE_DEFAULT)
|
|
|
|
headerStyle := he.uiConfig.GetStyle(config.STYLE_HEADER)
|
|
|
|
ctx.Fill(0, 0, size, ctx.Height(), ' ', defaultStyle)
|
2021-11-05 09:34:10 +00:00
|
|
|
ctx.Printf(0, 0, headerStyle, "%s:", name)
|
2019-05-12 04:06:09 +00:00
|
|
|
he.input.Draw(ctx.Subcontext(size, 0, ctx.Width()-size, 1))
|
|
|
|
}
|
|
|
|
|
2019-09-05 22:32:36 +00:00
|
|
|
func (he *headerEditor) MouseEvent(localX int, localY int, event tcell.Event) {
|
|
|
|
switch event := event.(type) {
|
|
|
|
case *tcell.EventMouse:
|
2019-11-06 03:43:45 +00:00
|
|
|
switch event.Buttons() {
|
|
|
|
case tcell.Button1:
|
|
|
|
he.focused = true
|
|
|
|
}
|
|
|
|
|
2019-09-05 22:32:36 +00:00
|
|
|
width := runewidth.StringWidth(he.name + " ")
|
|
|
|
if localX >= width {
|
|
|
|
he.input.MouseEvent(localX-width, localY, event)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-12 04:06:09 +00:00
|
|
|
func (he *headerEditor) Invalidate() {
|
2019-05-12 04:38:48 +00:00
|
|
|
he.input.Invalidate()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (he *headerEditor) OnInvalidate(fn func(ui.Drawable)) {
|
|
|
|
he.input.OnInvalidate(func(_ ui.Drawable) {
|
|
|
|
fn(he)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func (he *headerEditor) Focus(focused bool) {
|
2019-11-06 03:43:45 +00:00
|
|
|
he.focused = focused
|
2019-05-12 04:38:48 +00:00
|
|
|
he.input.Focus(focused)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (he *headerEditor) Event(event tcell.Event) bool {
|
|
|
|
return he.input.Event(event)
|
2019-05-12 04:06:09 +00:00
|
|
|
}
|
2019-05-13 20:24:05 +00:00
|
|
|
|
2019-05-14 20:18:21 +00:00
|
|
|
func (he *headerEditor) OnChange(fn func()) {
|
|
|
|
he.input.OnChange(func(_ *ui.TextInput) {
|
|
|
|
fn()
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2019-05-13 20:24:05 +00:00
|
|
|
type reviewMessage struct {
|
|
|
|
composer *Composer
|
|
|
|
grid *ui.Grid
|
|
|
|
}
|
|
|
|
|
2022-02-02 19:47:54 +00:00
|
|
|
var reviewCommands = [][]string{
|
|
|
|
{":send<enter>", "Send"},
|
|
|
|
{":edit<enter>", "Edit"},
|
|
|
|
{":attach<space>", "Add attachment"},
|
|
|
|
{":detach<space>", "Remove attachment"},
|
|
|
|
{":postpone<enter>", "Postpone"},
|
|
|
|
{":abort<enter>", "Abort (discard message, no confirmation)"},
|
|
|
|
{":choose -o d discard abort -o p postpone postpone<enter>", "Abort or postpone"},
|
|
|
|
}
|
|
|
|
|
2019-05-26 15:58:14 +00:00
|
|
|
func newReviewMessage(composer *Composer, err error) *reviewMessage {
|
2022-02-02 19:47:54 +00:00
|
|
|
bindings := composer.config.MergeContextualBinds(
|
|
|
|
composer.config.Bindings.ComposeReview,
|
|
|
|
config.BIND_CONTEXT_ACCOUNT,
|
|
|
|
composer.acctConfig.Name,
|
|
|
|
"compose::review",
|
|
|
|
)
|
|
|
|
|
|
|
|
var actions []string
|
|
|
|
|
|
|
|
for _, command := range reviewCommands {
|
|
|
|
cmd := command[0]
|
|
|
|
name := command[1]
|
|
|
|
strokes, _ := config.ParseKeyStrokes(cmd)
|
|
|
|
var inputs []string
|
|
|
|
for _, input := range bindings.GetReverseBindings(strokes) {
|
|
|
|
inputs = append(inputs, config.FormatKeyStrokes(input))
|
|
|
|
}
|
2022-04-12 09:49:31 +00:00
|
|
|
actions = append(actions, fmt.Sprintf(" %-6s %-40s %s",
|
|
|
|
strings.Join(inputs[:], ", "), name, cmd))
|
2022-02-02 19:47:54 +00:00
|
|
|
}
|
|
|
|
|
2020-05-31 11:37:46 +00:00
|
|
|
spec := []ui.GridSpec{
|
2022-03-18 08:53:02 +00:00
|
|
|
{Strategy: ui.SIZE_EXACT, Size: ui.Const(1)},
|
2020-05-31 11:37:46 +00:00
|
|
|
}
|
2022-02-02 19:47:54 +00:00
|
|
|
for i := 0; i < len(actions)-1; i++ {
|
2022-03-18 08:53:02 +00:00
|
|
|
spec = append(spec, ui.GridSpec{Strategy: ui.SIZE_EXACT, Size: ui.Const(1)})
|
2022-02-02 19:47:54 +00:00
|
|
|
}
|
2022-03-18 08:53:02 +00:00
|
|
|
spec = append(spec, ui.GridSpec{Strategy: ui.SIZE_EXACT, Size: ui.Const(2)})
|
|
|
|
spec = append(spec, ui.GridSpec{Strategy: ui.SIZE_EXACT, Size: ui.Const(1)})
|
2019-07-27 14:38:51 +00:00
|
|
|
for i := 0; i < len(composer.attachments)-1; i++ {
|
2022-03-18 08:53:02 +00:00
|
|
|
spec = append(spec, ui.GridSpec{Strategy: ui.SIZE_EXACT, Size: ui.Const(1)})
|
2019-07-16 20:48:25 +00:00
|
|
|
}
|
|
|
|
// make the last element fill remaining space
|
2022-03-18 08:53:02 +00:00
|
|
|
spec = append(spec, ui.GridSpec{Strategy: ui.SIZE_WEIGHT, Size: ui.Const(1)})
|
2019-07-16 20:48:25 +00:00
|
|
|
|
|
|
|
grid := ui.NewGrid().Rows(spec).Columns([]ui.GridSpec{
|
2022-03-18 08:53:02 +00:00
|
|
|
{Strategy: ui.SIZE_WEIGHT, Size: ui.Const(1)},
|
2019-05-13 20:24:05 +00:00
|
|
|
})
|
2019-07-16 20:48:25 +00:00
|
|
|
|
2020-07-27 08:03:55 +00:00
|
|
|
uiConfig := composer.config.Ui
|
|
|
|
|
2019-05-26 15:58:14 +00:00
|
|
|
if err != nil {
|
2020-07-27 08:03:55 +00:00
|
|
|
grid.AddChild(ui.NewText(err.Error(), uiConfig.GetStyle(config.STYLE_ERROR)))
|
|
|
|
grid.AddChild(ui.NewText("Press [q] to close this tab.",
|
|
|
|
uiConfig.GetStyle(config.STYLE_DEFAULT))).At(1, 0)
|
2019-05-26 15:58:14 +00:00
|
|
|
} else {
|
2022-02-02 19:47:54 +00:00
|
|
|
grid.AddChild(ui.NewText("Send this email?",
|
|
|
|
uiConfig.GetStyle(config.STYLE_TITLE))).At(0, 0)
|
|
|
|
i := 1
|
|
|
|
for _, action := range actions {
|
|
|
|
grid.AddChild(ui.NewText(action,
|
|
|
|
uiConfig.GetStyle(config.STYLE_DEFAULT))).At(i, 0)
|
|
|
|
i += 1
|
|
|
|
}
|
2020-07-27 08:03:55 +00:00
|
|
|
grid.AddChild(ui.NewText("Attachments:",
|
2022-02-02 19:47:54 +00:00
|
|
|
uiConfig.GetStyle(config.STYLE_TITLE))).At(i, 0)
|
|
|
|
i += 1
|
2019-07-16 20:48:25 +00:00
|
|
|
if len(composer.attachments) == 0 {
|
2020-07-27 08:03:55 +00:00
|
|
|
grid.AddChild(ui.NewText("(none)",
|
2022-02-02 19:47:54 +00:00
|
|
|
uiConfig.GetStyle(config.STYLE_DEFAULT))).At(i, 0)
|
2019-07-16 20:48:25 +00:00
|
|
|
} else {
|
2022-02-02 19:47:54 +00:00
|
|
|
for _, a := range composer.attachments {
|
2020-07-27 08:03:55 +00:00
|
|
|
grid.AddChild(ui.NewText(a, uiConfig.GetStyle(config.STYLE_DEFAULT))).
|
2022-02-02 19:47:54 +00:00
|
|
|
At(i, 0)
|
|
|
|
i += 1
|
2019-07-16 20:48:25 +00:00
|
|
|
}
|
|
|
|
}
|
2019-05-26 15:58:14 +00:00
|
|
|
}
|
2019-05-13 20:24:05 +00:00
|
|
|
|
|
|
|
return &reviewMessage{
|
|
|
|
composer: composer,
|
|
|
|
grid: grid,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (rm *reviewMessage) Invalidate() {
|
|
|
|
rm.grid.Invalidate()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (rm *reviewMessage) OnInvalidate(fn func(ui.Drawable)) {
|
|
|
|
rm.grid.OnInvalidate(func(_ ui.Drawable) {
|
|
|
|
fn(rm)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func (rm *reviewMessage) Draw(ctx *ui.Context) {
|
|
|
|
rm.grid.Draw(ctx)
|
|
|
|
}
|