2019-05-12 06:06:09 +02:00
|
|
|
package widgets
|
|
|
|
|
|
|
|
import (
|
2019-07-16 22:48:25 +02:00
|
|
|
"bufio"
|
2019-05-14 19:07:48 +02:00
|
|
|
"io"
|
2019-05-13 22:04:01 +02:00
|
|
|
"io/ioutil"
|
2019-07-16 22:48:25 +02:00
|
|
|
"mime"
|
|
|
|
"net/http"
|
2019-05-14 19:07:48 +02:00
|
|
|
gomail "net/mail"
|
2019-05-13 22:04:01 +02:00
|
|
|
"os"
|
2019-05-12 06:06:09 +02:00
|
|
|
"os/exec"
|
2019-07-16 22:48:25 +02:00
|
|
|
"path/filepath"
|
2019-05-14 19:07:48 +02:00
|
|
|
"time"
|
2019-05-12 06:06:09 +02:00
|
|
|
|
2019-05-14 19:07:48 +02:00
|
|
|
"github.com/emersion/go-message"
|
|
|
|
"github.com/emersion/go-message/mail"
|
2019-05-12 06:06:09 +02:00
|
|
|
"github.com/gdamore/tcell"
|
|
|
|
"github.com/mattn/go-runewidth"
|
2019-05-25 17:56:56 +02:00
|
|
|
"github.com/pkg/errors"
|
2019-05-12 06:06:09 +02:00
|
|
|
|
2019-05-18 02:57:10 +02:00
|
|
|
"git.sr.ht/~sircmpwn/aerc/config"
|
|
|
|
"git.sr.ht/~sircmpwn/aerc/lib/ui"
|
|
|
|
"git.sr.ht/~sircmpwn/aerc/worker/types"
|
2019-05-12 06:06:09 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
type Composer struct {
|
|
|
|
headers struct {
|
|
|
|
from *headerEditor
|
|
|
|
subject *headerEditor
|
|
|
|
to *headerEditor
|
|
|
|
}
|
|
|
|
|
2019-05-26 17:58:14 +02:00
|
|
|
acct *config.AccountConfig
|
|
|
|
config *config.AercConfig
|
2019-05-13 22:04:01 +02:00
|
|
|
|
2019-07-16 22:48:25 +02:00
|
|
|
defaults map[string]string
|
|
|
|
editor *Terminal
|
|
|
|
email *os.File
|
|
|
|
attachments []string
|
|
|
|
grid *ui.Grid
|
|
|
|
review *reviewMessage
|
|
|
|
worker *types.Worker
|
2019-05-12 06:06:09 +02:00
|
|
|
|
|
|
|
focusable []ui.DrawableInteractive
|
|
|
|
focused int
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: Let caller configure headers, initial body (for replies), etc
|
2019-05-14 21:25:30 +02:00
|
|
|
func NewComposer(conf *config.AercConfig,
|
2019-05-16 01:41:21 +02:00
|
|
|
acct *config.AccountConfig, worker *types.Worker) *Composer {
|
|
|
|
|
2019-05-12 06:06:09 +02:00
|
|
|
grid := ui.NewGrid().Rows([]ui.GridSpec{
|
|
|
|
{ui.SIZE_EXACT, 3},
|
|
|
|
{ui.SIZE_WEIGHT, 1},
|
|
|
|
}).Columns([]ui.GridSpec{
|
|
|
|
{ui.SIZE_WEIGHT, 1},
|
|
|
|
})
|
|
|
|
|
|
|
|
// TODO: let user specify extra headers to edit by default
|
|
|
|
headers := ui.NewGrid().Rows([]ui.GridSpec{
|
|
|
|
{ui.SIZE_EXACT, 1}, // To/From
|
|
|
|
{ui.SIZE_EXACT, 1}, // Subject
|
|
|
|
{ui.SIZE_EXACT, 1}, // [spacer]
|
|
|
|
}).Columns([]ui.GridSpec{
|
|
|
|
{ui.SIZE_WEIGHT, 1},
|
|
|
|
{ui.SIZE_WEIGHT, 1},
|
|
|
|
})
|
|
|
|
|
2019-05-12 06:38:48 +02:00
|
|
|
to := newHeaderEditor("To", "")
|
2019-05-14 21:25:30 +02:00
|
|
|
from := newHeaderEditor("From", acct.From)
|
2019-05-12 06:38:48 +02:00
|
|
|
subject := newHeaderEditor("Subject", "")
|
|
|
|
headers.AddChild(to).At(0, 0)
|
|
|
|
headers.AddChild(from).At(0, 1)
|
|
|
|
headers.AddChild(subject).At(1, 0).Span(1, 2)
|
2019-05-12 06:06:09 +02:00
|
|
|
headers.AddChild(ui.NewFill(' ')).At(2, 0).Span(1, 2)
|
|
|
|
|
2019-05-13 22:04:01 +02:00
|
|
|
email, err := ioutil.TempFile("", "aerc-compose-*.eml")
|
|
|
|
if err != nil {
|
|
|
|
// TODO: handle this better
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-05-12 06:06:09 +02:00
|
|
|
grid.AddChild(headers).At(0, 0)
|
|
|
|
|
2019-05-13 22:24:05 +02:00
|
|
|
c := &Composer{
|
2019-05-26 17:58:14 +02:00
|
|
|
acct: acct,
|
|
|
|
config: conf,
|
2019-05-13 22:04:01 +02:00
|
|
|
email: email,
|
|
|
|
grid: grid,
|
2019-05-16 01:41:21 +02:00
|
|
|
worker: worker,
|
2019-05-12 06:38:48 +02:00
|
|
|
// You have to backtab to get to "From", since you usually don't edit it
|
2019-05-13 22:04:01 +02:00
|
|
|
focused: 1,
|
2019-05-26 17:58:14 +02:00
|
|
|
focusable: []ui.DrawableInteractive{from, to, subject},
|
2019-05-12 06:06:09 +02:00
|
|
|
}
|
2019-05-14 19:07:48 +02:00
|
|
|
c.headers.to = to
|
|
|
|
c.headers.from = from
|
|
|
|
c.headers.subject = subject
|
2019-05-26 17:58:14 +02:00
|
|
|
c.ShowTerminal()
|
2019-05-13 22:24:05 +02:00
|
|
|
|
|
|
|
return c
|
2019-05-12 06:06:09 +02:00
|
|
|
}
|
|
|
|
|
2019-05-16 16:49:50 +02:00
|
|
|
// Sets additional headers to be added to the outgoing email (e.g. In-Reply-To)
|
|
|
|
func (c *Composer) Defaults(defaults map[string]string) *Composer {
|
|
|
|
c.defaults = defaults
|
|
|
|
if to, ok := defaults["To"]; ok {
|
|
|
|
c.headers.to.input.Set(to)
|
|
|
|
delete(defaults, "To")
|
|
|
|
}
|
|
|
|
if from, ok := defaults["From"]; ok {
|
|
|
|
c.headers.from.input.Set(from)
|
|
|
|
delete(defaults, "From")
|
|
|
|
}
|
|
|
|
if subject, ok := defaults["Subject"]; ok {
|
|
|
|
c.headers.subject.input.Set(subject)
|
|
|
|
delete(defaults, "Subject")
|
|
|
|
}
|
|
|
|
return c
|
|
|
|
}
|
|
|
|
|
2019-05-16 18:39:22 +02: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 {
|
|
|
|
c.email.Seek(0, os.SEEK_SET)
|
|
|
|
io.Copy(c.email, reader)
|
2019-05-16 20:09:57 +02:00
|
|
|
c.email.Sync()
|
2019-05-16 18:39:22 +02:00
|
|
|
c.email.Seek(0, os.SEEK_SET)
|
|
|
|
return c
|
|
|
|
}
|
|
|
|
|
2019-05-16 18:15:34 +02:00
|
|
|
func (c *Composer) FocusTerminal() *Composer {
|
2019-05-26 17:58:14 +02:00
|
|
|
if c.editor == nil {
|
|
|
|
return c
|
|
|
|
}
|
2019-05-16 18:15:34 +02:00
|
|
|
c.focusable[c.focused].Focus(false)
|
|
|
|
c.focused = 3
|
|
|
|
c.focusable[c.focused].Focus(true)
|
|
|
|
return c
|
|
|
|
}
|
|
|
|
|
2019-07-19 20:15:48 +02:00
|
|
|
func (c *Composer) FocusSubject() *Composer {
|
|
|
|
c.focusable[c.focused].Focus(false)
|
|
|
|
c.focused = 2
|
|
|
|
c.focusable[c.focused].Focus(true)
|
|
|
|
return c
|
|
|
|
}
|
|
|
|
|
2019-05-14 22:18:21 +02:00
|
|
|
func (c *Composer) OnSubjectChange(fn func(subject string)) {
|
|
|
|
c.headers.subject.OnChange(func() {
|
|
|
|
fn(c.headers.subject.input.String())
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2019-05-12 06:06:09 +02:00
|
|
|
func (c *Composer) Draw(ctx *ui.Context) {
|
|
|
|
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 20:05:29 +02:00
|
|
|
func (c *Composer) Close() {
|
|
|
|
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 20:27:28 +02: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 06:06:09 +02:00
|
|
|
func (c *Composer) Event(event tcell.Event) bool {
|
2019-07-16 17:33:47 +02:00
|
|
|
if c.editor != nil {
|
|
|
|
return c.focusable[c.focused].Event(event)
|
|
|
|
}
|
|
|
|
return false
|
2019-05-12 06:06:09 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Composer) Focus(focus bool) {
|
2019-05-12 06:38:48 +02:00
|
|
|
c.focusable[c.focused].Focus(focus)
|
2019-05-12 06:06:09 +02:00
|
|
|
}
|
|
|
|
|
2019-05-14 19:07:48 +02:00
|
|
|
func (c *Composer) Config() *config.AccountConfig {
|
2019-05-26 17:58:14 +02:00
|
|
|
return c.acct
|
2019-05-14 19:07:48 +02:00
|
|
|
}
|
|
|
|
|
2019-05-16 01:41:21 +02:00
|
|
|
func (c *Composer) Worker() *types.Worker {
|
|
|
|
return c.worker
|
|
|
|
}
|
|
|
|
|
2019-05-16 16:49:50 +02:00
|
|
|
func (c *Composer) PrepareHeader() (*mail.Header, []string, error) {
|
2019-05-14 19:07:48 +02:00
|
|
|
// Extract headers from the email, if present
|
|
|
|
c.email.Seek(0, os.SEEK_SET)
|
|
|
|
var (
|
|
|
|
rcpts []string
|
|
|
|
header mail.Header
|
|
|
|
)
|
|
|
|
reader, err := mail.CreateReader(c.email)
|
|
|
|
if err == nil {
|
|
|
|
header = reader.Header
|
|
|
|
defer reader.Close()
|
|
|
|
} else {
|
|
|
|
c.email.Seek(0, os.SEEK_SET)
|
|
|
|
}
|
|
|
|
// Update headers
|
|
|
|
mhdr := (*message.Header)(&header.Header)
|
2019-07-03 16:24:11 +02:00
|
|
|
mhdr.SetText("Message-Id", mail.GenerateMessageID())
|
2019-05-14 19:07:48 +02:00
|
|
|
if subject, _ := header.Subject(); subject == "" {
|
|
|
|
header.SetSubject(c.headers.subject.input.String())
|
|
|
|
}
|
2019-05-17 17:05:21 +02:00
|
|
|
if date, err := header.Date(); err != nil || date == (time.Time{}) {
|
2019-05-14 19:07:48 +02:00
|
|
|
header.SetDate(time.Now())
|
|
|
|
}
|
2019-06-05 19:58:07 +02:00
|
|
|
from := c.headers.from.input.String()
|
|
|
|
from_addrs, err := gomail.ParseAddressList(from)
|
|
|
|
if err != nil {
|
|
|
|
return nil, nil, errors.Wrapf(err, "ParseAddressList(%s)", from)
|
|
|
|
} else {
|
|
|
|
var simon_from []*mail.Address
|
|
|
|
for _, addr := range from_addrs {
|
|
|
|
simon_from = append(simon_from, (*mail.Address)(addr))
|
|
|
|
}
|
|
|
|
header.SetAddressList("From", simon_from)
|
2019-05-14 19:07:48 +02:00
|
|
|
}
|
2019-06-21 20:33:09 +02:00
|
|
|
// Merge in additional headers
|
|
|
|
txthdr := mhdr.Header
|
|
|
|
for key, value := range c.defaults {
|
|
|
|
if !txthdr.Has(key) && value != "" {
|
|
|
|
mhdr.SetText(key, value)
|
|
|
|
}
|
|
|
|
}
|
2019-05-14 19:07:48 +02:00
|
|
|
if to := c.headers.to.input.String(); to != "" {
|
|
|
|
// Dammit Simon, this branch is 3x as long as it ought to be because
|
|
|
|
// your types aren't compatible enough with each other
|
|
|
|
to_rcpts, err := gomail.ParseAddressList(to)
|
|
|
|
if err != nil {
|
2019-05-25 17:56:56 +02:00
|
|
|
return nil, nil, errors.Wrapf(err, "ParseAddressList(%s)", to)
|
2019-05-14 19:07:48 +02:00
|
|
|
}
|
|
|
|
ed_rcpts, err := header.AddressList("To")
|
|
|
|
if err != nil {
|
2019-05-25 17:56:56 +02:00
|
|
|
return nil, nil, errors.Wrap(err, "AddressList(To)")
|
2019-05-14 19:07:48 +02:00
|
|
|
}
|
|
|
|
for _, addr := range to_rcpts {
|
|
|
|
ed_rcpts = append(ed_rcpts, (*mail.Address)(addr))
|
|
|
|
}
|
|
|
|
header.SetAddressList("To", ed_rcpts)
|
|
|
|
for _, addr := range ed_rcpts {
|
|
|
|
rcpts = append(rcpts, addr.Address)
|
|
|
|
}
|
|
|
|
}
|
2019-06-02 15:40:47 +02:00
|
|
|
if cc, _ := mhdr.Text("Cc"); cc != "" {
|
|
|
|
cc_rcpts, err := gomail.ParseAddressList(cc)
|
|
|
|
if err != nil {
|
|
|
|
return nil, nil, errors.Wrapf(err, "ParseAddressList(%s)", cc)
|
|
|
|
}
|
|
|
|
// TODO: Update when the user inputs Cc's through the UI
|
|
|
|
for _, addr := range cc_rcpts {
|
|
|
|
rcpts = append(rcpts, addr.Address)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if bcc, _ := mhdr.Text("Bcc"); bcc != "" {
|
|
|
|
bcc_rcpts, err := gomail.ParseAddressList(bcc)
|
|
|
|
if err != nil {
|
|
|
|
return nil, nil, errors.Wrapf(err, "ParseAddressList(%s)", bcc)
|
|
|
|
}
|
|
|
|
// TODO: Update when the user inputs Bcc's through the UI
|
|
|
|
for _, addr := range bcc_rcpts {
|
|
|
|
rcpts = append(rcpts, addr.Address)
|
|
|
|
}
|
|
|
|
}
|
2019-05-14 20:05:29 +02:00
|
|
|
return &header, rcpts, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Composer) WriteMessage(header *mail.Header, writer io.Writer) error {
|
2019-06-27 11:06:50 +02:00
|
|
|
name := c.email.Name()
|
|
|
|
c.email.Close()
|
|
|
|
file, err := os.Open(name)
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "FileOpen")
|
|
|
|
}
|
|
|
|
c.email = file
|
2019-05-14 20:05:29 +02:00
|
|
|
var body io.Reader
|
|
|
|
reader, err := mail.CreateReader(c.email)
|
|
|
|
if err == nil {
|
|
|
|
// TODO: Do we want to let users write a full blown multipart email
|
|
|
|
// into the editor? If so this needs to change
|
|
|
|
part, err := reader.NextPart()
|
|
|
|
if err != nil {
|
2019-05-25 17:56:56 +02:00
|
|
|
return errors.Wrap(err, "reader.NextPart")
|
2019-05-14 20:05:29 +02:00
|
|
|
}
|
|
|
|
body = part.Body
|
|
|
|
defer reader.Close()
|
|
|
|
} else {
|
|
|
|
c.email.Seek(0, os.SEEK_SET)
|
|
|
|
body = c.email
|
|
|
|
}
|
2019-07-16 22:48:25 +02:00
|
|
|
|
|
|
|
if len(c.attachments) == 0 {
|
|
|
|
// don't create a multipart email if we only have text
|
|
|
|
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()
|
|
|
|
|
|
|
|
return writeBody(body, w)
|
|
|
|
}
|
|
|
|
|
|
|
|
// otherwise create a multipart email,
|
|
|
|
// with a multipart/alternative part for the text
|
|
|
|
w, err := mail.CreateWriter(writer, *header)
|
2019-05-14 19:07:48 +02:00
|
|
|
if err != nil {
|
2019-07-16 22:48:25 +02:00
|
|
|
return errors.Wrap(err, "CreateWriter")
|
2019-05-14 19:07:48 +02:00
|
|
|
}
|
2019-05-14 20:05:29 +02:00
|
|
|
defer w.Close()
|
2019-07-16 22:48:25 +02: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()
|
|
|
|
|
|
|
|
if err := writeBody(body, bw); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, a := range c.attachments {
|
|
|
|
writeAttachment(a, w)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func writeBody(body io.Reader, w io.Writer) error {
|
2019-05-25 17:56:56 +02:00
|
|
|
if _, err := io.Copy(w, body); err != nil {
|
|
|
|
return errors.Wrap(err, "io.Copy")
|
|
|
|
}
|
2019-07-16 22:48:25 +02:00
|
|
|
|
2019-05-25 17:56:56 +02:00
|
|
|
return nil
|
2019-05-14 19:07:48 +02:00
|
|
|
}
|
|
|
|
|
2019-07-16 22:48:25 +02:00
|
|
|
// write the attachment specified by path to the message
|
|
|
|
func writeAttachment(path string, writer *mail.Writer) error {
|
|
|
|
filename := filepath.Base(path)
|
|
|
|
|
|
|
|
f, err := os.Open(path)
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "os.Open")
|
|
|
|
}
|
|
|
|
defer f.Close()
|
|
|
|
|
|
|
|
reader := bufio.NewReader(f)
|
|
|
|
|
|
|
|
// determine the MIME type
|
|
|
|
// http.DetectContentType only cares about the first 512 bytes
|
|
|
|
head, err := reader.Peek(512)
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "Peek")
|
|
|
|
}
|
|
|
|
|
|
|
|
mimeString := http.DetectContentType(head)
|
|
|
|
// 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")
|
|
|
|
}
|
|
|
|
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
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Composer) AddAttachment(path string) {
|
|
|
|
c.attachments = append(c.attachments, path)
|
|
|
|
if c.review != nil {
|
|
|
|
c.grid.RemoveChild(c.review)
|
|
|
|
c.review = newReviewMessage(c, nil)
|
|
|
|
c.grid.AddChild(c.review).At(1, 0)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-13 22:24:05 +02:00
|
|
|
func (c *Composer) termClosed(err error) {
|
|
|
|
c.grid.RemoveChild(c.editor)
|
2019-05-26 17:58:14 +02:00
|
|
|
c.review = newReviewMessage(c, err)
|
|
|
|
c.grid.AddChild(c.review).At(1, 0)
|
2019-05-13 22:24:05 +02:00
|
|
|
c.editor.Destroy()
|
2019-05-14 20:05:29 +02:00
|
|
|
c.editor = nil
|
2019-05-26 17:58:14 +02: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 16:15:35 +02:00
|
|
|
editor := exec.Command("/bin/sh", "-c", editorName+" "+c.email.Name())
|
2019-05-26 17:58:14 +02:00
|
|
|
c.editor, _ = NewTerminal(editor) // TODO: handle error
|
|
|
|
c.editor.OnClose = c.termClosed
|
|
|
|
c.grid.AddChild(c.editor).At(1, 0)
|
|
|
|
c.focusable = append(c.focusable, c.editor)
|
2019-05-13 22:24:05 +02:00
|
|
|
}
|
|
|
|
|
2019-05-12 17:21:28 +02: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)
|
|
|
|
}
|
|
|
|
|
2019-05-13 22:24:05 +02:00
|
|
|
type headerEditor struct {
|
|
|
|
name string
|
|
|
|
input *ui.TextInput
|
|
|
|
}
|
|
|
|
|
2019-05-12 06:06:09 +02:00
|
|
|
func newHeaderEditor(name string, value string) *headerEditor {
|
|
|
|
return &headerEditor{
|
|
|
|
input: ui.NewTextInput(value),
|
|
|
|
name: name,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (he *headerEditor) Draw(ctx *ui.Context) {
|
|
|
|
name := he.name + " "
|
|
|
|
size := runewidth.StringWidth(name)
|
|
|
|
ctx.Fill(0, 0, size, ctx.Height(), ' ', tcell.StyleDefault)
|
|
|
|
ctx.Printf(0, 0, tcell.StyleDefault.Bold(true), "%s", name)
|
|
|
|
he.input.Draw(ctx.Subcontext(size, 0, ctx.Width()-size, 1))
|
|
|
|
}
|
|
|
|
|
|
|
|
func (he *headerEditor) Invalidate() {
|
2019-05-12 06:38:48 +02: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) {
|
|
|
|
he.input.Focus(focused)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (he *headerEditor) Event(event tcell.Event) bool {
|
|
|
|
return he.input.Event(event)
|
2019-05-12 06:06:09 +02:00
|
|
|
}
|
2019-05-13 22:24:05 +02:00
|
|
|
|
2019-05-14 22:18:21 +02:00
|
|
|
func (he *headerEditor) OnChange(fn func()) {
|
|
|
|
he.input.OnChange(func(_ *ui.TextInput) {
|
|
|
|
fn()
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2019-05-13 22:24:05 +02:00
|
|
|
type reviewMessage struct {
|
|
|
|
composer *Composer
|
|
|
|
grid *ui.Grid
|
|
|
|
}
|
|
|
|
|
2019-05-26 17:58:14 +02:00
|
|
|
func newReviewMessage(composer *Composer, err error) *reviewMessage {
|
2019-07-16 22:48:25 +02:00
|
|
|
spec := []ui.GridSpec{{ui.SIZE_EXACT, 2}, {ui.SIZE_EXACT, 1}}
|
|
|
|
for range composer.attachments {
|
|
|
|
spec = append(spec, ui.GridSpec{ui.SIZE_EXACT, 1})
|
|
|
|
}
|
|
|
|
// make the last element fill remaining space
|
|
|
|
spec = append(spec, ui.GridSpec{ui.SIZE_WEIGHT, 1})
|
|
|
|
|
|
|
|
grid := ui.NewGrid().Rows(spec).Columns([]ui.GridSpec{
|
2019-05-13 22:24:05 +02:00
|
|
|
{ui.SIZE_WEIGHT, 1},
|
|
|
|
})
|
2019-07-16 22:48:25 +02:00
|
|
|
|
2019-05-26 17:58:14 +02:00
|
|
|
if err != nil {
|
|
|
|
grid.AddChild(ui.NewText(err.Error()).
|
|
|
|
Color(tcell.ColorRed, tcell.ColorDefault))
|
|
|
|
grid.AddChild(ui.NewText("Press [q] to close this tab.")).At(1, 0)
|
|
|
|
} else {
|
|
|
|
// TODO: source this from actual keybindings?
|
|
|
|
grid.AddChild(ui.NewText(
|
2019-07-19 22:12:26 +02:00
|
|
|
"Send this email? [y]es/[n]o/[e]dit/[a]ttach")).At(0, 0)
|
2019-05-26 17:58:14 +02:00
|
|
|
grid.AddChild(ui.NewText("Attachments:").
|
|
|
|
Reverse(true)).At(1, 0)
|
2019-07-16 22:48:25 +02:00
|
|
|
if len(composer.attachments) == 0 {
|
|
|
|
grid.AddChild(ui.NewText("(none)")).At(2, 0)
|
|
|
|
} else {
|
|
|
|
for i, a := range composer.attachments {
|
|
|
|
grid.AddChild(ui.NewText(a)).At(i+2, 0)
|
|
|
|
}
|
|
|
|
}
|
2019-05-26 17:58:14 +02:00
|
|
|
}
|
2019-05-13 22:24:05 +02: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)
|
|
|
|
}
|