2018-02-27 21:17:26 -05:00
|
|
|
package widgets
|
|
|
|
|
|
|
|
import (
|
2019-07-19 14:15:48 -04:00
|
|
|
"errors"
|
2020-03-03 16:20:07 -05:00
|
|
|
"fmt"
|
2019-08-07 08:21:15 +02:00
|
|
|
"io"
|
2018-02-27 21:17:26 -05:00
|
|
|
"log"
|
2019-07-19 14:15:48 -04:00
|
|
|
"net/url"
|
|
|
|
"strings"
|
2019-03-17 16:19:15 -04:00
|
|
|
"time"
|
2018-02-27 21:17:26 -05:00
|
|
|
|
2018-06-01 09:58:00 +02:00
|
|
|
"github.com/gdamore/tcell"
|
2019-07-21 21:01:51 +01:00
|
|
|
"github.com/google/shlex"
|
2020-03-03 16:20:07 -05:00
|
|
|
"golang.org/x/crypto/openpgp"
|
2018-02-27 21:17:26 -05:00
|
|
|
|
2019-05-17 20:57:10 -04:00
|
|
|
"git.sr.ht/~sircmpwn/aerc/config"
|
2019-07-23 12:52:33 -04:00
|
|
|
"git.sr.ht/~sircmpwn/aerc/lib"
|
2019-05-17 20:57:10 -04:00
|
|
|
"git.sr.ht/~sircmpwn/aerc/lib/ui"
|
2020-01-08 21:44:14 +01:00
|
|
|
"git.sr.ht/~sircmpwn/aerc/models"
|
2018-02-27 21:17:26 -05:00
|
|
|
)
|
|
|
|
|
|
|
|
type Aerc struct {
|
2019-03-17 16:19:15 -04:00
|
|
|
accounts map[string]*AccountView
|
2019-07-21 21:01:51 +01:00
|
|
|
cmd func(cmd []string) error
|
2019-07-23 12:52:33 -04:00
|
|
|
cmdHistory lib.History
|
2019-06-27 10:33:11 -07:00
|
|
|
complete func(cmd string) []string
|
2019-03-17 16:19:15 -04:00
|
|
|
conf *config.AercConfig
|
2019-09-12 23:16:37 -03:00
|
|
|
focused ui.Interactive
|
|
|
|
grid *ui.Grid
|
2019-03-17 16:19:15 -04:00
|
|
|
logger *log.Logger
|
2019-03-21 17:49:59 -04:00
|
|
|
simulating int
|
2019-09-12 23:16:37 -03:00
|
|
|
statusbar *ui.Stack
|
2019-03-17 16:19:15 -04:00
|
|
|
statusline *StatusLine
|
|
|
|
pendingKeys []config.KeyStroke
|
2019-09-12 23:16:37 -03:00
|
|
|
prompts *ui.Stack
|
|
|
|
tabs *ui.Tabs
|
2020-03-03 16:20:07 -05:00
|
|
|
ui *ui.UI
|
2019-07-29 10:50:02 -04:00
|
|
|
beep func() error
|
2020-05-19 13:06:49 +02:00
|
|
|
dialog ui.DrawableInteractive
|
2018-02-27 21:17:26 -05:00
|
|
|
}
|
|
|
|
|
2020-04-24 11:36:16 +02:00
|
|
|
type Choice struct {
|
|
|
|
Key string
|
|
|
|
Text string
|
|
|
|
Command []string
|
|
|
|
}
|
|
|
|
|
2019-03-10 21:15:24 -04:00
|
|
|
func NewAerc(conf *config.AercConfig, logger *log.Logger,
|
2019-07-23 12:52:33 -04:00
|
|
|
cmd func(cmd []string) error, complete func(cmd string) []string,
|
|
|
|
cmdHistory lib.History) *Aerc {
|
2019-03-10 21:15:24 -04:00
|
|
|
|
2020-03-07 16:42:41 +00:00
|
|
|
tabs := ui.NewTabs(&conf.Ui)
|
2018-02-27 21:17:26 -05:00
|
|
|
|
2020-05-28 10:32:42 -04:00
|
|
|
statusbar := ui.NewStack()
|
|
|
|
statusline := NewStatusLine()
|
2019-03-17 16:19:15 -04:00
|
|
|
statusbar.Push(statusline)
|
|
|
|
|
2019-09-12 23:16:37 -03:00
|
|
|
grid := ui.NewGrid().Rows([]ui.GridSpec{
|
2020-05-31 12:37:46 +01:00
|
|
|
{ui.SIZE_EXACT, ui.Const(1)},
|
|
|
|
{ui.SIZE_WEIGHT, ui.Const(1)},
|
|
|
|
{ui.SIZE_EXACT, ui.Const(1)},
|
2019-09-12 23:16:37 -03:00
|
|
|
}).Columns([]ui.GridSpec{
|
2020-05-31 12:37:46 +01:00
|
|
|
{ui.SIZE_WEIGHT, ui.Const(1)},
|
2018-02-27 21:17:26 -05:00
|
|
|
})
|
2019-03-30 14:12:04 -04:00
|
|
|
grid.AddChild(tabs.TabStrip)
|
|
|
|
grid.AddChild(tabs.TabContent).At(1, 0)
|
|
|
|
grid.AddChild(statusbar).At(2, 0)
|
2018-02-27 21:17:26 -05:00
|
|
|
|
2019-01-14 08:14:03 -05:00
|
|
|
aerc := &Aerc{
|
2019-03-17 16:19:15 -04:00
|
|
|
accounts: make(map[string]*AccountView),
|
|
|
|
conf: conf,
|
|
|
|
cmd: cmd,
|
2019-07-23 12:52:33 -04:00
|
|
|
cmdHistory: cmdHistory,
|
2019-06-27 10:33:11 -07:00
|
|
|
complete: complete,
|
2019-03-17 16:19:15 -04:00
|
|
|
grid: grid,
|
|
|
|
logger: logger,
|
|
|
|
statusbar: statusbar,
|
|
|
|
statusline: statusline,
|
2020-05-28 10:32:42 -04:00
|
|
|
prompts: ui.NewStack(),
|
2019-03-17 16:19:15 -04:00
|
|
|
tabs: tabs,
|
2019-01-14 08:14:03 -05:00
|
|
|
}
|
2019-01-13 13:25:56 -05:00
|
|
|
|
2019-07-16 18:43:08 +01:00
|
|
|
statusline.SetAerc(aerc)
|
2019-07-21 21:01:51 +01:00
|
|
|
conf.Triggers.ExecuteCommand = cmd
|
2019-07-16 18:43:08 +01:00
|
|
|
|
2019-05-22 10:39:52 -04:00
|
|
|
for i, acct := range conf.Accounts {
|
2019-09-05 23:32:36 +01:00
|
|
|
view := NewAccountView(aerc, conf, &conf.Accounts[i], logger, aerc)
|
2019-01-14 08:14:03 -05:00
|
|
|
aerc.accounts[acct.Name] = view
|
2019-01-13 13:03:28 -05:00
|
|
|
tabs.Add(view, acct.Name)
|
2018-06-11 19:23:09 -04:00
|
|
|
}
|
2018-02-27 21:17:26 -05:00
|
|
|
|
2019-05-22 11:35:55 -04:00
|
|
|
if len(conf.Accounts) == 0 {
|
|
|
|
wizard := NewAccountWizard(aerc.Config(), aerc)
|
|
|
|
wizard.Focus(true)
|
|
|
|
aerc.NewTab(wizard, "New account")
|
|
|
|
}
|
|
|
|
|
2019-09-05 23:32:36 +01:00
|
|
|
tabs.CloseTab = func(index int) {
|
|
|
|
switch content := aerc.tabs.Tabs[index].Content.(type) {
|
|
|
|
case *AccountView:
|
|
|
|
return
|
|
|
|
case *AccountWizard:
|
|
|
|
return
|
|
|
|
case *Composer:
|
|
|
|
aerc.RemoveTab(content)
|
|
|
|
content.Close()
|
|
|
|
case *Terminal:
|
|
|
|
content.Close(nil)
|
|
|
|
case *MessageViewer:
|
|
|
|
aerc.RemoveTab(content)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-14 08:14:03 -05:00
|
|
|
return aerc
|
2018-02-27 21:17:26 -05:00
|
|
|
}
|
|
|
|
|
2019-07-29 10:50:02 -04:00
|
|
|
func (aerc *Aerc) OnBeep(f func() error) {
|
|
|
|
aerc.beep = f
|
|
|
|
}
|
|
|
|
|
|
|
|
func (aerc *Aerc) Beep() {
|
|
|
|
if aerc.beep == nil {
|
|
|
|
aerc.logger.Printf("should beep, but no beeper")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if err := aerc.beep(); err != nil {
|
|
|
|
aerc.logger.Printf("tried to beep, but could not: %v", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-19 09:49:57 +00:00
|
|
|
func (aerc *Aerc) Tick() bool {
|
|
|
|
more := false
|
|
|
|
for _, acct := range aerc.accounts {
|
|
|
|
more = acct.Tick() || more
|
|
|
|
}
|
2019-08-19 21:56:12 -04:00
|
|
|
|
|
|
|
if len(aerc.prompts.Children()) > 0 {
|
|
|
|
more = true
|
|
|
|
previous := aerc.focused
|
|
|
|
prompt := aerc.prompts.Pop().(*ExLine)
|
|
|
|
prompt.finish = func() {
|
|
|
|
aerc.statusbar.Pop()
|
|
|
|
aerc.focus(previous)
|
|
|
|
}
|
|
|
|
|
|
|
|
aerc.statusbar.Push(prompt)
|
|
|
|
aerc.focus(prompt)
|
|
|
|
}
|
|
|
|
|
2019-05-19 09:49:57 +00:00
|
|
|
return more
|
|
|
|
}
|
|
|
|
|
2019-02-10 22:46:13 +01:00
|
|
|
func (aerc *Aerc) Children() []ui.Drawable {
|
2019-01-20 15:08:30 -05:00
|
|
|
return aerc.grid.Children()
|
|
|
|
}
|
|
|
|
|
2019-09-12 23:16:37 -03:00
|
|
|
func (aerc *Aerc) OnInvalidate(onInvalidate func(d ui.Drawable)) {
|
|
|
|
aerc.grid.OnInvalidate(func(_ ui.Drawable) {
|
2019-01-13 13:25:56 -05:00
|
|
|
onInvalidate(aerc)
|
|
|
|
})
|
2018-02-27 21:17:26 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
func (aerc *Aerc) Invalidate() {
|
|
|
|
aerc.grid.Invalidate()
|
|
|
|
}
|
|
|
|
|
2019-03-17 14:02:33 -04:00
|
|
|
func (aerc *Aerc) Focus(focus bool) {
|
|
|
|
// who cares
|
|
|
|
}
|
|
|
|
|
2019-09-12 23:16:37 -03:00
|
|
|
func (aerc *Aerc) Draw(ctx *ui.Context) {
|
2018-02-27 21:17:26 -05:00
|
|
|
aerc.grid.Draw(ctx)
|
2020-05-19 13:06:49 +02:00
|
|
|
if aerc.dialog != nil {
|
|
|
|
aerc.dialog.Draw(ctx.Subcontext(4, ctx.Height()/2-2,
|
2020-03-03 21:51:27 -05:00
|
|
|
ctx.Width()-8, 4))
|
2020-03-03 16:20:07 -05:00
|
|
|
}
|
2018-02-27 21:17:26 -05:00
|
|
|
}
|
|
|
|
|
2019-03-21 17:36:42 -04:00
|
|
|
func (aerc *Aerc) getBindings() *config.KeyBindings {
|
2019-05-14 14:27:28 -04:00
|
|
|
switch view := aerc.SelectedTab().(type) {
|
2019-03-21 17:36:42 -04:00
|
|
|
case *AccountView:
|
|
|
|
return aerc.conf.Bindings.MessageList
|
2019-05-21 16:31:04 -04:00
|
|
|
case *AccountWizard:
|
|
|
|
return aerc.conf.Bindings.AccountWizard
|
2019-05-12 00:06:09 -04:00
|
|
|
case *Composer:
|
2019-05-14 14:27:28 -04:00
|
|
|
switch view.Bindings() {
|
|
|
|
case "compose::editor":
|
|
|
|
return aerc.conf.Bindings.ComposeEditor
|
|
|
|
case "compose::review":
|
|
|
|
return aerc.conf.Bindings.ComposeReview
|
|
|
|
default:
|
|
|
|
return aerc.conf.Bindings.Compose
|
|
|
|
}
|
2019-03-30 21:45:41 -04:00
|
|
|
case *MessageViewer:
|
|
|
|
return aerc.conf.Bindings.MessageView
|
2019-03-30 14:12:04 -04:00
|
|
|
case *Terminal:
|
2019-03-21 17:44:44 -04:00
|
|
|
return aerc.conf.Bindings.Terminal
|
2019-03-21 17:36:42 -04:00
|
|
|
default:
|
|
|
|
return aerc.conf.Bindings.Global
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (aerc *Aerc) simulate(strokes []config.KeyStroke) {
|
|
|
|
aerc.pendingKeys = []config.KeyStroke{}
|
2019-03-21 17:49:59 -04:00
|
|
|
aerc.simulating += 1
|
2019-03-21 17:36:42 -04:00
|
|
|
for _, stroke := range strokes {
|
|
|
|
simulated := tcell.NewEventKey(
|
|
|
|
stroke.Key, stroke.Rune, tcell.ModNone)
|
|
|
|
aerc.Event(simulated)
|
|
|
|
}
|
2019-03-21 17:49:59 -04:00
|
|
|
aerc.simulating -= 1
|
2019-03-21 17:36:42 -04:00
|
|
|
}
|
|
|
|
|
2018-06-01 09:58:00 +02:00
|
|
|
func (aerc *Aerc) Event(event tcell.Event) bool {
|
2020-05-19 13:06:49 +02:00
|
|
|
if aerc.dialog != nil {
|
|
|
|
return aerc.dialog.Event(event)
|
2020-03-03 16:20:07 -05:00
|
|
|
}
|
|
|
|
|
2019-03-17 16:19:15 -04:00
|
|
|
if aerc.focused != nil {
|
|
|
|
return aerc.focused.Event(event)
|
|
|
|
}
|
|
|
|
|
|
|
|
switch event := event.(type) {
|
|
|
|
case *tcell.EventKey:
|
2019-03-21 21:34:12 -04:00
|
|
|
aerc.statusline.Expire()
|
2019-03-17 16:19:15 -04:00
|
|
|
aerc.pendingKeys = append(aerc.pendingKeys, config.KeyStroke{
|
|
|
|
Key: event.Key(),
|
|
|
|
Rune: event.Rune(),
|
|
|
|
})
|
2019-07-16 18:43:08 +01:00
|
|
|
aerc.statusline.Invalidate()
|
2019-03-21 17:36:42 -04:00
|
|
|
bindings := aerc.getBindings()
|
|
|
|
incomplete := false
|
|
|
|
result, strokes := bindings.GetBinding(aerc.pendingKeys)
|
2019-03-17 16:19:15 -04:00
|
|
|
switch result {
|
|
|
|
case config.BINDING_FOUND:
|
2019-03-21 17:36:42 -04:00
|
|
|
aerc.simulate(strokes)
|
|
|
|
return true
|
2019-03-17 16:19:15 -04:00
|
|
|
case config.BINDING_INCOMPLETE:
|
2019-03-21 17:36:42 -04:00
|
|
|
incomplete = true
|
2019-03-17 16:19:15 -04:00
|
|
|
case config.BINDING_NOT_FOUND:
|
2019-03-21 17:36:42 -04:00
|
|
|
}
|
|
|
|
if bindings.Globals {
|
|
|
|
result, strokes = aerc.conf.Bindings.Global.
|
|
|
|
GetBinding(aerc.pendingKeys)
|
|
|
|
switch result {
|
|
|
|
case config.BINDING_FOUND:
|
|
|
|
aerc.simulate(strokes)
|
|
|
|
return true
|
|
|
|
case config.BINDING_INCOMPLETE:
|
|
|
|
incomplete = true
|
|
|
|
case config.BINDING_NOT_FOUND:
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if !incomplete {
|
2019-03-17 16:19:15 -04:00
|
|
|
aerc.pendingKeys = []config.KeyStroke{}
|
2019-03-21 17:49:59 -04:00
|
|
|
exKey := bindings.ExKey
|
|
|
|
if aerc.simulating > 0 {
|
|
|
|
// Keybindings still use : even if you change the ex key
|
|
|
|
exKey = aerc.conf.Bindings.Global.ExKey
|
|
|
|
}
|
|
|
|
if event.Key() == exKey.Key && event.Rune() == exKey.Rune {
|
2019-11-15 13:28:34 -07:00
|
|
|
aerc.BeginExCommand("")
|
2019-03-17 16:19:15 -04:00
|
|
|
return true
|
|
|
|
}
|
2019-03-17 17:39:22 -04:00
|
|
|
interactive, ok := aerc.tabs.Tabs[aerc.tabs.Selected].Content.(ui.Interactive)
|
|
|
|
if ok {
|
|
|
|
return interactive.Event(event)
|
|
|
|
}
|
|
|
|
return false
|
2019-03-17 16:19:15 -04:00
|
|
|
}
|
2019-07-11 23:15:15 +01:00
|
|
|
case *tcell.EventMouse:
|
2019-09-05 23:32:36 +01:00
|
|
|
if event.Buttons() == tcell.ButtonNone {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
x, y := event.Position()
|
|
|
|
aerc.grid.MouseEvent(x, y, event)
|
|
|
|
return true
|
2019-03-17 16:19:15 -04:00
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
func (aerc *Aerc) Config() *config.AercConfig {
|
|
|
|
return aerc.conf
|
2018-02-27 21:17:26 -05:00
|
|
|
}
|
2019-01-14 08:14:03 -05:00
|
|
|
|
2019-06-21 14:33:09 -04:00
|
|
|
func (aerc *Aerc) Logger() *log.Logger {
|
|
|
|
return aerc.logger
|
|
|
|
}
|
|
|
|
|
2019-03-10 21:15:24 -04:00
|
|
|
func (aerc *Aerc) SelectedAccount() *AccountView {
|
2019-10-01 18:01:49 +01:00
|
|
|
switch tab := aerc.SelectedTab().(type) {
|
|
|
|
case *AccountView:
|
|
|
|
return tab
|
|
|
|
case *MessageViewer:
|
|
|
|
return tab.SelectedAccount()
|
2020-04-24 11:42:21 +02:00
|
|
|
case *Composer:
|
|
|
|
return tab.Account()
|
2019-03-17 14:57:05 -04:00
|
|
|
}
|
2019-10-01 18:01:49 +01:00
|
|
|
return nil
|
2019-01-14 08:14:03 -05:00
|
|
|
}
|
2019-03-17 16:19:15 -04:00
|
|
|
|
2019-03-17 17:23:53 -04:00
|
|
|
func (aerc *Aerc) SelectedTab() ui.Drawable {
|
|
|
|
return aerc.tabs.Tabs[aerc.tabs.Selected].Content
|
|
|
|
}
|
|
|
|
|
2020-03-02 19:54:42 +00:00
|
|
|
func (aerc *Aerc) SelectedTabIndex() int {
|
|
|
|
return aerc.tabs.Selected
|
|
|
|
}
|
|
|
|
|
|
|
|
func (aerc *Aerc) NumTabs() int {
|
|
|
|
return len(aerc.tabs.Tabs)
|
|
|
|
}
|
|
|
|
|
2019-09-05 23:32:36 +01:00
|
|
|
func (aerc *Aerc) NewTab(clickable ui.Drawable, name string) *ui.Tab {
|
|
|
|
tab := aerc.tabs.Add(clickable, name)
|
2019-03-17 16:19:15 -04:00
|
|
|
aerc.tabs.Select(len(aerc.tabs.Tabs) - 1)
|
|
|
|
return tab
|
|
|
|
}
|
|
|
|
|
2019-03-17 17:23:53 -04:00
|
|
|
func (aerc *Aerc) RemoveTab(tab ui.Drawable) {
|
|
|
|
aerc.tabs.Remove(tab)
|
|
|
|
}
|
|
|
|
|
2019-06-10 22:05:56 -07:00
|
|
|
func (aerc *Aerc) ReplaceTab(tabSrc ui.Drawable, tabTarget ui.Drawable, name string) {
|
|
|
|
aerc.tabs.Replace(tabSrc, tabTarget, name)
|
|
|
|
}
|
|
|
|
|
2020-03-02 19:54:42 +00:00
|
|
|
func (aerc *Aerc) MoveTab(i int) {
|
|
|
|
aerc.tabs.MoveTab(i)
|
|
|
|
}
|
|
|
|
|
2020-03-07 16:42:41 +00:00
|
|
|
func (aerc *Aerc) PinTab() {
|
|
|
|
aerc.tabs.PinTab()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (aerc *Aerc) UnpinTab() {
|
|
|
|
aerc.tabs.UnpinTab()
|
|
|
|
}
|
|
|
|
|
2019-03-17 16:24:17 -04:00
|
|
|
func (aerc *Aerc) NextTab() {
|
2019-09-05 23:32:36 +01:00
|
|
|
aerc.tabs.NextTab()
|
2019-03-17 16:24:17 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
func (aerc *Aerc) PrevTab() {
|
2019-09-05 23:32:36 +01:00
|
|
|
aerc.tabs.PrevTab()
|
2019-03-17 16:24:17 -04:00
|
|
|
}
|
|
|
|
|
2019-07-19 18:12:57 +01:00
|
|
|
func (aerc *Aerc) SelectTab(name string) bool {
|
|
|
|
for i, tab := range aerc.tabs.Tabs {
|
|
|
|
if tab.Name == name {
|
|
|
|
aerc.tabs.Select(i)
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2019-07-26 22:41:13 +01:00
|
|
|
func (aerc *Aerc) SelectTabIndex(index int) bool {
|
2019-09-03 16:34:03 -03:00
|
|
|
for i := range aerc.tabs.Tabs {
|
2019-07-26 22:41:13 +01:00
|
|
|
if i == index {
|
|
|
|
aerc.tabs.Select(i)
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2019-07-19 18:12:57 +01:00
|
|
|
func (aerc *Aerc) TabNames() []string {
|
|
|
|
var names []string
|
|
|
|
for _, tab := range aerc.tabs.Tabs {
|
|
|
|
names = append(names, tab.Name)
|
|
|
|
}
|
|
|
|
return names
|
|
|
|
}
|
|
|
|
|
|
|
|
func (aerc *Aerc) SelectPreviousTab() bool {
|
|
|
|
return aerc.tabs.SelectPrevious()
|
|
|
|
}
|
|
|
|
|
2019-03-17 16:19:15 -04:00
|
|
|
// TODO: Use per-account status lines, but a global ex line
|
|
|
|
func (aerc *Aerc) SetStatus(status string) *StatusMessage {
|
|
|
|
return aerc.statusline.Set(status)
|
|
|
|
}
|
|
|
|
|
2020-05-28 10:32:32 -04:00
|
|
|
func (aerc *Aerc) PushStatus(text string, expiry time.Duration) *StatusMessage {
|
|
|
|
return aerc.statusline.Push(text, expiry)
|
2019-03-17 16:19:15 -04:00
|
|
|
}
|
|
|
|
|
2020-05-28 10:32:42 -04:00
|
|
|
func (aerc *Aerc) PushError(text string) {
|
|
|
|
aerc.PushStatus(text, 10*time.Second).Color(tcell.ColorDefault, tcell.ColorRed)
|
2019-05-26 17:37:39 -04:00
|
|
|
}
|
|
|
|
|
2019-09-12 23:16:37 -03:00
|
|
|
func (aerc *Aerc) focus(item ui.Interactive) {
|
2019-03-17 16:19:15 -04:00
|
|
|
if aerc.focused == item {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if aerc.focused != nil {
|
|
|
|
aerc.focused.Focus(false)
|
|
|
|
}
|
|
|
|
aerc.focused = item
|
2019-03-21 21:34:12 -04:00
|
|
|
interactive, ok := aerc.tabs.Tabs[aerc.tabs.Selected].Content.(ui.Interactive)
|
2019-03-17 16:19:15 -04:00
|
|
|
if item != nil {
|
|
|
|
item.Focus(true)
|
2019-03-21 21:34:12 -04:00
|
|
|
if ok {
|
|
|
|
interactive.Focus(false)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if ok {
|
|
|
|
interactive.Focus(true)
|
|
|
|
}
|
2019-03-17 16:19:15 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-15 13:28:34 -07:00
|
|
|
func (aerc *Aerc) BeginExCommand(cmd string) {
|
2019-03-17 16:19:15 -04:00
|
|
|
previous := aerc.focused
|
2019-12-20 13:21:33 -05:00
|
|
|
exline := NewExLine(aerc.conf, cmd, func(cmd string) {
|
2019-07-21 21:01:51 +01:00
|
|
|
parts, err := shlex.Split(cmd)
|
|
|
|
if err != nil {
|
2020-05-28 10:32:42 -04:00
|
|
|
aerc.PushError(" " + err.Error())
|
2019-07-21 21:01:51 +01:00
|
|
|
}
|
|
|
|
err = aerc.cmd(parts)
|
2019-03-17 16:19:15 -04:00
|
|
|
if err != nil {
|
2020-05-28 10:32:42 -04:00
|
|
|
aerc.PushError(" " + err.Error())
|
2019-03-17 16:19:15 -04:00
|
|
|
}
|
2019-07-23 12:52:33 -04:00
|
|
|
// only add to history if this is an unsimulated command,
|
|
|
|
// ie one not executed from a keybinding
|
|
|
|
if aerc.simulating == 0 {
|
|
|
|
aerc.cmdHistory.Add(cmd)
|
|
|
|
}
|
2019-03-17 16:19:15 -04:00
|
|
|
}, func() {
|
|
|
|
aerc.statusbar.Pop()
|
|
|
|
aerc.focus(previous)
|
2019-06-27 10:33:11 -07:00
|
|
|
}, func(cmd string) []string {
|
|
|
|
return aerc.complete(cmd)
|
2019-07-23 12:52:33 -04:00
|
|
|
}, aerc.cmdHistory)
|
2019-03-17 16:19:15 -04:00
|
|
|
aerc.statusbar.Push(exline)
|
|
|
|
aerc.focus(exline)
|
|
|
|
}
|
2019-07-19 14:15:48 -04:00
|
|
|
|
2019-08-19 21:56:12 -04:00
|
|
|
func (aerc *Aerc) RegisterPrompt(prompt string, cmd []string) {
|
2019-12-20 13:21:33 -05:00
|
|
|
p := NewPrompt(aerc.conf, prompt, func(text string) {
|
2019-08-19 21:56:12 -04:00
|
|
|
if text != "" {
|
|
|
|
cmd = append(cmd, text)
|
|
|
|
}
|
|
|
|
err := aerc.cmd(cmd)
|
|
|
|
if err != nil {
|
2020-05-28 10:32:42 -04:00
|
|
|
aerc.PushError(" " + err.Error())
|
2019-08-19 21:56:12 -04:00
|
|
|
}
|
|
|
|
}, func(cmd string) []string {
|
|
|
|
return nil // TODO: completions
|
|
|
|
})
|
|
|
|
aerc.prompts.Push(p)
|
|
|
|
}
|
|
|
|
|
2020-04-24 11:36:16 +02:00
|
|
|
func (aerc *Aerc) RegisterChoices(choices []Choice) {
|
|
|
|
cmds := make(map[string][]string)
|
|
|
|
texts := []string{}
|
|
|
|
for _, c := range choices {
|
|
|
|
text := fmt.Sprintf("[%s] %s", c.Key, c.Text)
|
|
|
|
if strings.Contains(c.Text, c.Key) {
|
2020-04-24 12:59:40 -04:00
|
|
|
text = strings.Replace(c.Text, c.Key, "["+c.Key+"]", 1)
|
2020-04-24 11:36:16 +02:00
|
|
|
}
|
|
|
|
texts = append(texts, text)
|
|
|
|
cmds[c.Key] = c.Command
|
|
|
|
}
|
|
|
|
prompt := strings.Join(texts, ", ") + "? "
|
|
|
|
p := NewPrompt(aerc.conf, prompt, func(text string) {
|
|
|
|
cmd, ok := cmds[text]
|
|
|
|
if !ok {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
err := aerc.cmd(cmd)
|
|
|
|
if err != nil {
|
2020-05-28 10:32:42 -04:00
|
|
|
aerc.PushError(" " + err.Error())
|
2020-04-24 11:36:16 +02:00
|
|
|
}
|
|
|
|
}, func(cmd string) []string {
|
|
|
|
return nil // TODO: completions
|
|
|
|
})
|
|
|
|
aerc.prompts.Push(p)
|
|
|
|
}
|
|
|
|
|
2019-07-19 14:15:48 -04:00
|
|
|
func (aerc *Aerc) Mailto(addr *url.URL) error {
|
|
|
|
acct := aerc.SelectedAccount()
|
|
|
|
if acct == nil {
|
|
|
|
return errors.New("No account selected")
|
|
|
|
}
|
|
|
|
defaults := make(map[string]string)
|
|
|
|
defaults["To"] = addr.Opaque
|
|
|
|
headerMap := map[string]string{
|
|
|
|
"cc": "Cc",
|
|
|
|
"in-reply-to": "In-Reply-To",
|
|
|
|
"subject": "Subject",
|
|
|
|
}
|
|
|
|
for key, vals := range addr.Query() {
|
|
|
|
if header, ok := headerMap[strings.ToLower(key)]; ok {
|
|
|
|
defaults[header] = strings.Join(vals, ",")
|
|
|
|
}
|
|
|
|
}
|
2020-04-24 11:42:21 +02:00
|
|
|
composer, err := NewComposer(aerc, acct, aerc.Config(),
|
2020-01-08 21:44:14 +01:00
|
|
|
acct.AccountConfig(), acct.Worker(), "", defaults, models.OriginalMail{})
|
2019-11-03 13:51:14 +01:00
|
|
|
if err != nil {
|
|
|
|
return nil
|
|
|
|
}
|
2019-07-19 14:15:48 -04:00
|
|
|
composer.FocusSubject()
|
|
|
|
title := "New email"
|
|
|
|
if subj, ok := defaults["Subject"]; ok {
|
|
|
|
title = subj
|
|
|
|
composer.FocusTerminal()
|
|
|
|
}
|
|
|
|
tab := aerc.NewTab(composer, title)
|
2019-07-22 16:29:07 -07:00
|
|
|
composer.OnHeaderChange("Subject", func(subject string) {
|
2019-07-19 14:15:48 -04:00
|
|
|
if subject == "" {
|
|
|
|
tab.Name = "New email"
|
|
|
|
} else {
|
|
|
|
tab.Name = subject
|
|
|
|
}
|
|
|
|
tab.Content.Invalidate()
|
|
|
|
})
|
|
|
|
return nil
|
|
|
|
}
|
2019-08-07 08:21:15 +02:00
|
|
|
|
|
|
|
func (aerc *Aerc) CloseBackends() error {
|
|
|
|
var returnErr error
|
|
|
|
for _, acct := range aerc.accounts {
|
|
|
|
var raw interface{} = acct.worker.Backend
|
|
|
|
c, ok := raw.(io.Closer)
|
|
|
|
if !ok {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
err := c.Close()
|
|
|
|
if err != nil {
|
|
|
|
returnErr = err
|
|
|
|
aerc.logger.Printf("Closing backend failed for %v: %v\n",
|
|
|
|
acct.Name(), err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return returnErr
|
|
|
|
}
|
2020-03-03 16:20:07 -05:00
|
|
|
|
2020-05-19 13:06:49 +02:00
|
|
|
func (aerc *Aerc) AddDialog(d ui.DrawableInteractive) {
|
|
|
|
aerc.dialog = d
|
|
|
|
aerc.dialog.OnInvalidate(func(_ ui.Drawable) {
|
2020-03-03 16:20:07 -05:00
|
|
|
aerc.Invalidate()
|
|
|
|
})
|
|
|
|
aerc.Invalidate()
|
2020-05-19 13:06:49 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
func (aerc *Aerc) CloseDialog() {
|
|
|
|
aerc.dialog = nil
|
|
|
|
aerc.Invalidate()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
func (aerc *Aerc) GetPassword(title string, prompt string) (chText chan string, chErr chan error) {
|
|
|
|
chText = make(chan string, 1)
|
|
|
|
chErr = make(chan error, 1)
|
2020-05-28 10:32:42 -04:00
|
|
|
getPasswd := NewGetPasswd(title, prompt, func(pw string, err error) {
|
2020-05-19 13:06:49 +02:00
|
|
|
defer func() {
|
|
|
|
close(chErr)
|
|
|
|
close(chText)
|
|
|
|
aerc.CloseDialog()
|
|
|
|
}()
|
|
|
|
if err != nil {
|
|
|
|
chErr <- err
|
|
|
|
return
|
|
|
|
}
|
|
|
|
chErr <- nil
|
|
|
|
chText <- pw
|
|
|
|
return
|
|
|
|
})
|
|
|
|
aerc.AddDialog(getPasswd)
|
|
|
|
|
|
|
|
return
|
2020-03-03 16:20:07 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
func (aerc *Aerc) Initialize(ui *ui.UI) {
|
|
|
|
aerc.ui = ui
|
|
|
|
}
|
|
|
|
|
2020-05-19 13:06:47 +02:00
|
|
|
func (aerc *Aerc) DecryptKeys(keys []openpgp.Key, symmetric bool) (b []byte, err error) {
|
2020-03-03 16:20:07 -05:00
|
|
|
for _, key := range keys {
|
2020-05-19 13:06:49 +02:00
|
|
|
ident := key.Entity.PrimaryIdentity()
|
|
|
|
chPass, chErr := aerc.GetPassword("Decrypt PGP private key",
|
2020-05-19 13:06:47 +02:00
|
|
|
fmt.Sprintf("Enter password for %s (%8X)\nPress <ESC> to cancel",
|
2020-05-19 13:06:49 +02:00
|
|
|
ident.Name, key.PublicKey.KeyId))
|
|
|
|
|
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case err = <-chErr:
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
2020-05-19 13:06:48 +02:00
|
|
|
}
|
2020-05-19 13:06:49 +02:00
|
|
|
pass := <-chPass
|
|
|
|
err = key.PrivateKey.Decrypt([]byte(pass))
|
|
|
|
return nil, err
|
|
|
|
default:
|
|
|
|
aerc.ui.Tick()
|
|
|
|
}
|
2020-03-03 16:20:07 -05:00
|
|
|
}
|
|
|
|
}
|
2020-05-19 13:06:47 +02:00
|
|
|
return nil, err
|
2020-03-03 16:20:07 -05:00
|
|
|
}
|