2019-03-22 16:25:09 +01:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"os/exec"
|
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
)
|
|
|
|
|
2019-05-07 11:59:09 +02:00
|
|
|
func selector(history []string, max int, tool string) error {
|
2019-03-22 16:25:09 +01:00
|
|
|
// reverse the history
|
|
|
|
for i, j := 0, len(history)-1; i < j; i, j = i+1, j-1 {
|
|
|
|
history[i], history[j] = history[j], history[i]
|
|
|
|
}
|
|
|
|
|
2019-05-07 11:59:09 +02:00
|
|
|
selected, err := dmenu(history, max, tool)
|
2019-03-22 16:25:09 +01:00
|
|
|
if err != nil {
|
2019-05-09 18:46:16 +02:00
|
|
|
return err
|
2019-03-22 16:25:09 +01:00
|
|
|
}
|
|
|
|
|
2019-03-25 19:13:19 +01:00
|
|
|
// serve selection to the OS
|
2019-04-18 23:07:34 +02:00
|
|
|
err = exec.Command("wl-copy", []string{"--", selected}...).Run()
|
2019-03-22 16:25:09 +01:00
|
|
|
|
2019-04-18 23:07:34 +02:00
|
|
|
return err
|
2019-03-22 16:25:09 +01:00
|
|
|
}
|
|
|
|
|
2019-05-07 11:59:09 +02:00
|
|
|
func dmenu(list []string, max int, tool string) (string, error) {
|
2019-04-04 10:21:20 +02:00
|
|
|
if len(list) == 0 {
|
|
|
|
return "", nil
|
|
|
|
}
|
|
|
|
|
2019-06-17 08:40:45 +02:00
|
|
|
bin, err := exec.LookPath(tool)
|
2019-05-09 18:46:16 +02:00
|
|
|
if err != nil {
|
|
|
|
return "", fmt.Errorf("%s is not installed", tool)
|
|
|
|
}
|
|
|
|
|
2019-05-09 08:42:43 +02:00
|
|
|
var args []string
|
|
|
|
if tool == "dmenu" {
|
|
|
|
args = []string{"dmenu", "-b",
|
|
|
|
"-fn",
|
|
|
|
"-misc-dejavu sans mono-medium-r-normal--17-120-100-100-m-0-iso8859-16",
|
|
|
|
"-l",
|
|
|
|
strconv.Itoa(max)}
|
|
|
|
} else {
|
|
|
|
args = []string{"rofi", "-dmenu",
|
|
|
|
"-lines",
|
|
|
|
strconv.Itoa(max)}
|
|
|
|
}
|
2019-03-22 16:25:09 +01:00
|
|
|
|
|
|
|
// dmenu will break if items contain newlines, so we must pass them as literals.
|
2019-05-10 09:20:43 +02:00
|
|
|
// however, when it sends them back, we need a way to restore them
|
2019-09-07 19:41:41 +02:00
|
|
|
var escaped []string
|
2019-03-22 22:05:37 +01:00
|
|
|
guide := make(map[string]string)
|
2019-03-22 22:08:54 +01:00
|
|
|
for _, original := range list {
|
|
|
|
repr := fmt.Sprintf("%#v", original)
|
2019-05-10 09:20:43 +02:00
|
|
|
|
|
|
|
// dmenu will split lines longer than 1200 something; we cut at 400 to spare memory
|
2019-04-04 09:04:57 +02:00
|
|
|
max := len(repr) - 1 // drop right quote
|
2019-04-04 10:50:15 +02:00
|
|
|
maxChars := 400
|
|
|
|
if max > maxChars {
|
|
|
|
max = maxChars
|
2019-04-04 09:04:57 +02:00
|
|
|
}
|
|
|
|
repr = repr[1:max] // drop left quote
|
2019-05-10 09:20:43 +02:00
|
|
|
|
2019-03-22 22:08:54 +01:00
|
|
|
guide[repr] = original
|
2019-05-10 09:20:43 +02:00
|
|
|
escaped = append(escaped, repr)
|
2019-03-22 16:25:09 +01:00
|
|
|
}
|
|
|
|
|
2019-05-10 09:20:43 +02:00
|
|
|
input := strings.NewReader(strings.Join(escaped, "\n"))
|
2019-03-22 16:25:09 +01:00
|
|
|
|
2019-05-09 18:46:16 +02:00
|
|
|
cmd := exec.Cmd{Path: bin, Args: args, Stdin: input}
|
2019-03-22 16:25:09 +01:00
|
|
|
selected, err := cmd.Output()
|
|
|
|
if err != nil {
|
2019-05-10 09:20:43 +02:00
|
|
|
if err.Error() == "exit status 1" {
|
|
|
|
// dmenu exits with this error when no selection done
|
|
|
|
return "", nil
|
|
|
|
}
|
2019-03-22 16:25:09 +01:00
|
|
|
return "", err
|
|
|
|
}
|
2019-05-10 09:20:43 +02:00
|
|
|
trimmed := selected[:len(selected)-1] // drop newline added by dmenu
|
2019-03-22 16:25:09 +01:00
|
|
|
|
2019-04-04 09:04:57 +02:00
|
|
|
sel, ok := guide[string(trimmed)]
|
|
|
|
if !ok {
|
|
|
|
return "", fmt.Errorf("couldn't recover original string; please report this bug along with a copy of your clipman.json")
|
|
|
|
}
|
|
|
|
|
|
|
|
return sel, nil
|
2019-03-22 16:25:09 +01:00
|
|
|
}
|