2019-03-11 02:15:24 +01:00
|
|
|
package commands
|
|
|
|
|
|
|
|
import (
|
|
|
|
"errors"
|
2019-12-21 16:21:27 +01:00
|
|
|
"fmt"
|
2019-07-26 15:29:40 +02:00
|
|
|
"sort"
|
2019-06-27 19:33:11 +02:00
|
|
|
"strings"
|
2019-07-03 18:54:10 +02:00
|
|
|
"unicode"
|
2019-03-11 02:15:24 +01:00
|
|
|
|
2019-03-11 02:23:22 +01:00
|
|
|
"github.com/google/shlex"
|
|
|
|
|
2021-11-05 10:19:46 +01:00
|
|
|
"git.sr.ht/~rjarry/aerc/widgets"
|
2019-03-11 02:15:24 +01:00
|
|
|
)
|
|
|
|
|
2019-06-27 19:33:11 +02:00
|
|
|
type Command interface {
|
|
|
|
Aliases() []string
|
|
|
|
Execute(*widgets.Aerc, []string) error
|
|
|
|
Complete(*widgets.Aerc, []string) []string
|
|
|
|
}
|
2019-03-11 02:15:24 +01:00
|
|
|
|
2019-06-27 19:33:11 +02:00
|
|
|
type Commands map[string]Command
|
2019-03-11 02:15:24 +01:00
|
|
|
|
2019-03-21 21:30:23 +01:00
|
|
|
func NewCommands() *Commands {
|
2019-06-27 19:33:11 +02:00
|
|
|
cmds := Commands(make(map[string]Command))
|
2019-03-21 21:30:23 +01:00
|
|
|
return &cmds
|
|
|
|
}
|
|
|
|
|
2019-06-27 19:33:11 +02:00
|
|
|
func (cmds *Commands) dict() map[string]Command {
|
|
|
|
return map[string]Command(*cmds)
|
2019-03-21 21:30:23 +01:00
|
|
|
}
|
|
|
|
|
2019-06-27 19:33:11 +02:00
|
|
|
func (cmds *Commands) Names() []string {
|
|
|
|
names := make([]string, 0)
|
|
|
|
|
|
|
|
for k := range cmds.dict() {
|
|
|
|
names = append(names, k)
|
|
|
|
}
|
|
|
|
return names
|
|
|
|
}
|
|
|
|
|
|
|
|
func (cmds *Commands) Register(cmd Command) {
|
|
|
|
// TODO enforce unique aliases, until then, duplicate each
|
|
|
|
if len(cmd.Aliases()) < 1 {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
for _, alias := range cmd.Aliases() {
|
|
|
|
cmds.dict()[alias] = cmd
|
|
|
|
}
|
2019-03-21 21:30:23 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
type NoSuchCommand string
|
|
|
|
|
|
|
|
func (err NoSuchCommand) Error() string {
|
|
|
|
return "Unknown command " + string(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
type CommandSource interface {
|
|
|
|
Commands() *Commands
|
2019-03-11 02:15:24 +01:00
|
|
|
}
|
|
|
|
|
2019-07-21 22:01:51 +02:00
|
|
|
func (cmds *Commands) ExecuteCommand(aerc *widgets.Aerc, args []string) error {
|
2019-03-11 02:23:22 +01:00
|
|
|
if len(args) == 0 {
|
|
|
|
return errors.New("Expected a command.")
|
|
|
|
}
|
2019-06-27 19:33:11 +02:00
|
|
|
if cmd, ok := cmds.dict()[args[0]]; ok {
|
|
|
|
return cmd.Execute(aerc, args)
|
2019-03-11 02:15:24 +01:00
|
|
|
}
|
2019-03-21 21:30:23 +01:00
|
|
|
return NoSuchCommand(args[0])
|
2019-03-11 02:15:24 +01:00
|
|
|
}
|
2019-06-27 19:33:11 +02:00
|
|
|
|
|
|
|
func (cmds *Commands) GetCompletions(aerc *widgets.Aerc, cmd string) []string {
|
|
|
|
args, err := shlex.Split(cmd)
|
|
|
|
if err != nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(args) == 0 {
|
2019-07-26 15:29:40 +02:00
|
|
|
names := cmds.Names()
|
|
|
|
sort.Strings(names)
|
|
|
|
return names
|
2019-06-27 19:33:11 +02:00
|
|
|
}
|
|
|
|
|
2019-07-26 15:29:40 +02:00
|
|
|
if len(args) > 1 || cmd[len(cmd)-1] == ' ' {
|
2019-06-27 19:33:11 +02:00
|
|
|
if cmd, ok := cmds.dict()[args[0]]; ok {
|
2019-07-26 15:29:40 +02:00
|
|
|
var completions []string
|
|
|
|
if len(args) > 1 {
|
|
|
|
completions = cmd.Complete(aerc, args[1:])
|
|
|
|
} else {
|
|
|
|
completions = cmd.Complete(aerc, []string{})
|
|
|
|
}
|
2019-06-27 19:33:11 +02:00
|
|
|
if completions != nil && len(completions) == 0 {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
options := make([]string, 0)
|
|
|
|
for _, option := range completions {
|
|
|
|
options = append(options, args[0]+" "+option)
|
|
|
|
}
|
|
|
|
return options
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
names := cmds.Names()
|
|
|
|
options := make([]string, 0)
|
|
|
|
for _, name := range names {
|
|
|
|
if strings.HasPrefix(name, args[0]) {
|
|
|
|
options = append(options, name)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(options) > 0 {
|
|
|
|
return options
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
2019-07-03 18:54:10 +02:00
|
|
|
|
|
|
|
func GetFolders(aerc *widgets.Aerc, args []string) []string {
|
|
|
|
out := make([]string, 0)
|
2022-02-25 00:21:06 +01:00
|
|
|
acct := aerc.SelectedAccount()
|
|
|
|
if acct == nil {
|
|
|
|
return out
|
|
|
|
}
|
2019-07-26 15:29:40 +02:00
|
|
|
if len(args) == 0 {
|
2022-02-25 00:21:06 +01:00
|
|
|
return acct.Directories().List()
|
2019-07-26 15:29:40 +02:00
|
|
|
}
|
2022-02-25 00:21:06 +01:00
|
|
|
for _, dir := range acct.Directories().List() {
|
|
|
|
if foundInString(dir, args[0], acct.UiConfig().FuzzyFolderComplete) {
|
2019-07-03 18:54:10 +02:00
|
|
|
out = append(out, dir)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return out
|
|
|
|
}
|
2019-08-02 19:10:42 +02:00
|
|
|
|
2020-04-11 04:12:38 +02:00
|
|
|
// CompletionFromList provides a convenience wrapper for commands to use in the
|
|
|
|
// Complete function. It simply matches the items provided in valid
|
|
|
|
func CompletionFromList(valid []string, args []string) []string {
|
|
|
|
out := make([]string, 0)
|
|
|
|
if len(args) == 0 {
|
|
|
|
return valid
|
|
|
|
}
|
|
|
|
for _, v := range valid {
|
|
|
|
if hasCaseSmartPrefix(v, args[0]) {
|
|
|
|
out = append(out, v)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return out
|
|
|
|
}
|
|
|
|
|
2019-12-21 16:21:27 +01:00
|
|
|
func GetLabels(aerc *widgets.Aerc, args []string) []string {
|
2022-02-25 00:21:06 +01:00
|
|
|
acct := aerc.SelectedAccount()
|
|
|
|
if acct == nil {
|
|
|
|
return make([]string, 0)
|
|
|
|
}
|
2019-12-21 16:21:27 +01:00
|
|
|
if len(args) == 0 {
|
2022-02-25 00:21:06 +01:00
|
|
|
return acct.Labels()
|
2019-12-21 16:21:27 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// + and - are used to denote tag addition / removal and need to be striped
|
|
|
|
// only the last tag should be completed, so that multiple labels can be
|
|
|
|
// selected
|
|
|
|
last := args[len(args)-1]
|
|
|
|
others := strings.Join(args[:len(args)-1], " ")
|
|
|
|
var prefix string
|
|
|
|
switch last[0] {
|
|
|
|
case '+':
|
|
|
|
prefix = "+"
|
|
|
|
case '-':
|
|
|
|
prefix = "-"
|
|
|
|
default:
|
|
|
|
prefix = ""
|
|
|
|
}
|
|
|
|
trimmed := strings.TrimLeft(last, "+-")
|
|
|
|
|
|
|
|
out := make([]string, 0)
|
2022-02-25 00:21:06 +01:00
|
|
|
for _, label := range acct.Labels() {
|
2019-12-21 16:21:27 +01:00
|
|
|
if hasCaseSmartPrefix(label, trimmed) {
|
|
|
|
var prev string
|
|
|
|
if len(others) > 0 {
|
|
|
|
prev = others + " "
|
|
|
|
}
|
|
|
|
out = append(out, fmt.Sprintf("%v%v%v", prev, prefix, label))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return out
|
|
|
|
}
|
|
|
|
|
2022-01-31 17:32:19 +01:00
|
|
|
func foundInString(s, substring string, fuzzy bool) bool {
|
|
|
|
if fuzzy {
|
|
|
|
return caseInsensitiveContains(s, substring)
|
|
|
|
} else {
|
|
|
|
return hasCaseSmartPrefix(s, substring)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-08-02 19:10:42 +02:00
|
|
|
// hasCaseSmartPrefix checks whether s starts with prefix, using a case
|
|
|
|
// sensitive match if and only if prefix contains upper case letters.
|
|
|
|
func hasCaseSmartPrefix(s, prefix string) bool {
|
|
|
|
if hasUpper(prefix) {
|
|
|
|
return strings.HasPrefix(s, prefix)
|
|
|
|
}
|
|
|
|
return strings.HasPrefix(strings.ToLower(s), strings.ToLower(prefix))
|
|
|
|
}
|
|
|
|
|
2022-01-31 17:32:19 +01:00
|
|
|
func caseInsensitiveContains(s, substr string) bool {
|
|
|
|
s, substr = strings.ToUpper(s), strings.ToUpper(substr)
|
|
|
|
return strings.Contains(s, substr)
|
|
|
|
}
|
|
|
|
|
2019-08-02 19:10:42 +02:00
|
|
|
func hasUpper(s string) bool {
|
|
|
|
for _, r := range s {
|
|
|
|
if unicode.IsUpper(r) {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|