2020-11-01 16:09:48 +01:00
|
|
|
package desktop
|
|
|
|
|
|
|
|
import (
|
2021-01-28 21:19:31 +01:00
|
|
|
"bytes"
|
2021-02-14 14:40:17 +01:00
|
|
|
"fmt"
|
2021-01-28 21:12:35 +01:00
|
|
|
"os/exec"
|
|
|
|
"strings"
|
2021-01-29 20:05:13 +01:00
|
|
|
|
2022-07-14 00:58:22 +02:00
|
|
|
"github.com/demodesk/neko/pkg/types"
|
|
|
|
"github.com/demodesk/neko/pkg/xevent"
|
2020-11-01 16:09:48 +01:00
|
|
|
)
|
|
|
|
|
2021-01-29 20:05:13 +01:00
|
|
|
func (manager *DesktopManagerCtx) ClipboardGetText() (*types.ClipboardText, error) {
|
|
|
|
text, err := manager.ClipboardGetBinary("STRING")
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Rich text must not always be available, can fail silently.
|
|
|
|
html, _ := manager.ClipboardGetBinary("text/html")
|
|
|
|
|
|
|
|
return &types.ClipboardText{
|
|
|
|
Text: string(text),
|
|
|
|
HTML: string(html),
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (manager *DesktopManagerCtx) ClipboardSetText(data types.ClipboardText) error {
|
|
|
|
// TODO: Refactor.
|
|
|
|
// Current implementation is unable to set multiple targets. HTML
|
|
|
|
// is set, if available. Otherwise plain text.
|
|
|
|
|
|
|
|
if data.HTML != "" {
|
|
|
|
return manager.ClipboardSetBinary("text/html", []byte(data.HTML))
|
|
|
|
}
|
|
|
|
|
|
|
|
return manager.ClipboardSetBinary("STRING", []byte(data.Text))
|
|
|
|
}
|
|
|
|
|
2021-01-28 22:51:55 +01:00
|
|
|
func (manager *DesktopManagerCtx) ClipboardGetBinary(mime string) ([]byte, error) {
|
2021-01-29 00:03:22 +01:00
|
|
|
cmd := exec.Command("xclip", "-selection", "clipboard", "-out", "-target", mime)
|
2021-01-28 21:19:31 +01:00
|
|
|
|
|
|
|
var stdout, stderr bytes.Buffer
|
|
|
|
cmd.Stdout = &stdout
|
|
|
|
cmd.Stderr = &stderr
|
|
|
|
|
|
|
|
err := cmd.Run()
|
|
|
|
if err != nil {
|
2021-02-01 18:29:39 +01:00
|
|
|
msg := strings.TrimSpace(stderr.String())
|
2021-01-28 22:51:55 +01:00
|
|
|
return nil, fmt.Errorf("%s", msg)
|
2021-01-28 21:19:31 +01:00
|
|
|
}
|
|
|
|
|
2021-01-28 22:51:55 +01:00
|
|
|
return stdout.Bytes(), nil
|
2020-11-01 16:09:48 +01:00
|
|
|
}
|
|
|
|
|
2021-01-28 22:51:55 +01:00
|
|
|
func (manager *DesktopManagerCtx) ClipboardSetBinary(mime string, data []byte) error {
|
2021-01-29 00:03:22 +01:00
|
|
|
cmd := exec.Command("xclip", "-selection", "clipboard", "-in", "-target", mime)
|
2021-01-28 21:19:31 +01:00
|
|
|
|
|
|
|
var stderr bytes.Buffer
|
|
|
|
cmd.Stderr = &stderr
|
|
|
|
|
2021-02-14 14:40:17 +01:00
|
|
|
stdin, err := cmd.StdinPipe()
|
2021-01-29 00:03:22 +01:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2022-05-03 19:57:31 +00:00
|
|
|
// TODO: Refactor.
|
|
|
|
// We need to wait until the data came to the clipboard.
|
|
|
|
wait := make(chan struct{})
|
2022-07-28 12:20:20 +02:00
|
|
|
xevent.Emmiter.Once("clipboard-updated", func(payload ...any) {
|
2022-05-03 19:57:31 +00:00
|
|
|
wait <- struct{}{}
|
|
|
|
})
|
|
|
|
|
2021-01-29 00:03:22 +01:00
|
|
|
err = cmd.Start()
|
2021-01-28 21:19:31 +01:00
|
|
|
if err != nil {
|
2021-02-01 18:29:39 +01:00
|
|
|
msg := strings.TrimSpace(stderr.String())
|
2021-01-28 21:19:31 +01:00
|
|
|
return fmt.Errorf("%s", msg)
|
|
|
|
}
|
2021-02-01 18:29:39 +01:00
|
|
|
|
|
|
|
_, err = stdin.Write(data)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2021-01-29 00:03:22 +01:00
|
|
|
stdin.Close()
|
2021-01-28 21:19:31 +01:00
|
|
|
|
2021-01-29 00:03:22 +01:00
|
|
|
// TODO: Refactor.
|
|
|
|
// cmd.Wait()
|
2022-05-03 19:57:31 +00:00
|
|
|
<-wait
|
|
|
|
|
2021-01-28 21:19:31 +01:00
|
|
|
return nil
|
2020-11-01 16:09:48 +01:00
|
|
|
}
|
2021-01-28 22:51:55 +01:00
|
|
|
|
|
|
|
func (manager *DesktopManagerCtx) ClipboardGetTargets() ([]string, error) {
|
2021-01-29 00:03:22 +01:00
|
|
|
cmd := exec.Command("xclip", "-selection", "clipboard", "-out", "-target", "TARGETS")
|
2021-01-28 22:51:55 +01:00
|
|
|
|
|
|
|
var stdout, stderr bytes.Buffer
|
|
|
|
cmd.Stdout = &stdout
|
|
|
|
cmd.Stderr = &stderr
|
|
|
|
|
|
|
|
err := cmd.Run()
|
|
|
|
if err != nil {
|
2021-02-01 18:29:39 +01:00
|
|
|
msg := strings.TrimSpace(stderr.String())
|
2021-01-28 22:51:55 +01:00
|
|
|
return nil, fmt.Errorf("%s", msg)
|
|
|
|
}
|
|
|
|
|
|
|
|
var response []string
|
2021-02-01 18:29:39 +01:00
|
|
|
targets := strings.Split(stdout.String(), "\n")
|
2021-01-28 22:51:55 +01:00
|
|
|
for _, target := range targets {
|
|
|
|
if target == "" {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
if !strings.Contains(target, "/") {
|
|
|
|
continue
|
|
|
|
}
|
2021-02-02 18:28:32 +01:00
|
|
|
|
2021-01-28 22:51:55 +01:00
|
|
|
response = append(response, target)
|
|
|
|
}
|
|
|
|
|
|
|
|
return response, nil
|
|
|
|
}
|