clipman/selector.go

94 lines
2.1 KiB
Go
Raw Normal View History

package main
import (
2019-09-17 09:46:30 +02:00
"errors"
"fmt"
"os"
"os/exec"
"strconv"
"strings"
)
2019-09-17 09:46:30 +02:00
func selector(data []string, max int, tool string) (string, error) {
if len(data) == 0 {
return "", errors.New("No history available")
}
2019-09-17 09:46:30 +02:00
// output to stdout and return
if tool == "-" {
2019-09-17 09:46:30 +02:00
escaped, _ := preprocessData(data, false)
os.Stdout.WriteString(strings.Join(escaped, "\n"))
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
switch tool {
case "dmenu":
2019-05-09 08:42:43 +02:00
args = []string{"dmenu", "-b",
"-fn",
"-misc-dejavu sans mono-medium-r-normal--17-120-100-100-m-0-iso8859-16",
"-l",
strconv.Itoa(max)}
case "rofi":
2019-05-09 08:42:43 +02:00
args = []string{"rofi", "-dmenu",
"-lines",
strconv.Itoa(max)}
default:
2019-09-17 09:46:30 +02:00
return "", fmt.Errorf("Unsupported tool: %s", tool)
2019-05-09 08:42:43 +02:00
}
2019-09-17 09:46:30 +02:00
processed, guide := preprocessData(data, true)
2019-09-17 09:46:30 +02:00
cmd := exec.Cmd{Path: bin, Args: args, Stdin: strings.NewReader(strings.Join(processed, "\n"))}
b, err := cmd.CombinedOutput()
if err != nil {
2019-05-10 09:20:43 +02:00
if err.Error() == "exit status 1" {
2019-09-17 09:46:30 +02:00
// dmenu/rofi exits with this error when no selection done
2019-05-10 09:20:43 +02:00
return "", nil
}
return "", err
}
2019-09-17 09:46:30 +02:00
selected := string(b[:len(b)-1]) // drop newline added by dmenu/rofi
2019-09-17 09:46:30 +02:00
sel, ok := guide[selected]
2019-04-04 09:04:57 +02:00
if !ok {
2019-09-17 09:46:30 +02:00
return "", errors.New("couldn't recover original string")
2019-04-04 09:04:57 +02:00
}
return sel, nil
}
2019-09-17 09:46:30 +02:00
// preprocessData:
// - reverses the data
// - escapes special characters (like newlines) that would break external selectors;
// - optionally it cuts items longer than 400 bytes (dmenu doesn't allow more than ~1200).
// A guide is created to allow restoring the selected item.
func preprocessData(data []string, cutting bool) ([]string, map[string]string) {
2019-09-15 09:34:04 +02:00
var escaped []string
guide := make(map[string]string)
2019-09-17 09:46:30 +02:00
for i := len(data) - 1; i >= 0; i-- { // reverse slice
original := data[i]
2019-09-15 09:34:04 +02:00
2019-09-17 09:46:30 +02:00
repr := fmt.Sprintf("%#v", original)
size := len(repr) - 1
2019-09-15 09:34:04 +02:00
if cutting {
2019-09-17 09:46:30 +02:00
const maxChars = 400
if size > maxChars {
size = maxChars
2019-09-15 09:34:04 +02:00
}
}
2019-09-17 09:46:30 +02:00
repr = repr[1:size] // drop left and right quotes
2019-09-15 09:34:04 +02:00
guide[repr] = original
escaped = append(escaped, repr)
}
return escaped, guide
}