1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
|
package main
import (
"image"
"image/png"
"log"
"os"
"strings"
"time"
"github.com/godbus/dbus/v5"
)
type server struct{}
func (s server) GetCapabilities() (capabilities []string, e *dbus.Error) {
// log.Print("GetCapabilities called")
return []string{
"body",
"actions",
}, nil
}
func (s server) GetServerInformation() (name, vendor, version, specVersion string, e *dbus.Error) {
// log.Print("GetServerInformation called")
return "corvid", "CartConnoisseur", "0.1.0", "1.2", nil
}
func (s server) CloseNotification(id uint32) (e *dbus.Error) {
// log.Printf("CloseNotification called: %d", id)
notification, ok := notifications.notifications[id]
if ok {
notification.close(CloseReasonClosed)
}
return nil
}
func (s server) Notify(appName string, replacesId uint32, appIcon string, summary string, body string, actions []string, hints map[string]dbus.Variant, expireTimeout int32) (id uint32, e *dbus.Error) {
// log.Print("Notify called")
notifications.mutex.Lock()
defer notifications.mutex.Unlock()
if replacesId == 0 {
id = notifications.nextId
notifications.nextId++
} else {
id = replacesId
}
actionMap := make(map[string]string)
for i := 0; i < len(actions)-1; i += 2 {
actionMap[actions[i]] = actions[i+1]
}
hintMap := make(map[string]hint)
img := ""
for key, value := range hints {
if !value.Signature().Empty() {
if strings.Contains("ybnqiuxtds", string(value.Signature().String()[0])) {
hintMap[key] = hint{Variant: value}
} else if key == "image-data" {
raw := value.Value().([]interface{})
var i image.Image
if raw[3].(bool) {
i = &image.NRGBA{
Pix: raw[6].([]uint8),
Stride: int(raw[2].(int32)),
Rect: image.Rect(0, 0, int(raw[0].(int32)), int(raw[1].(int32))),
}
} else {
rgb := raw[6].([]uint8)
rgba := make([]uint8, len(rgb)/3*4)
for i := 0; i < len(rgb)-1; i += 3 {
rgba[i/3*4] = rgb[i]
rgba[i/3*4+1] = rgb[i+1]
rgba[i/3*4+2] = rgb[i+2]
rgba[i/3*4+3] = 0xff
}
i = &image.NRGBA{
Pix: rgba,
Stride: int(raw[2].(int32)),
Rect: image.Rect(0, 0, int(raw[0].(int32)), int(raw[1].(int32))),
}
}
_ = i
f, err := os.CreateTemp(os.TempDir(), "corvid-*.png")
if err != nil {
log.Fatal(err)
}
defer f.Close()
png.Encode(f, i)
img = f.Name()
}
}
}
if expireTimeout == -1 {
expireTimeout = DEFAULT_EXPIRATION
}
notification := notification{
Id: id,
AppName: appName,
AppIcon: appIcon,
Summary: summary,
Body: body,
Actions: actionMap,
Hints: hintMap,
Timestamp: time.Now().Unix(),
Expiration: expireTimeout,
Image: img,
timer: nil,
}
if expireTimeout != 0 {
notification.timer = time.AfterFunc(time.Duration(expireTimeout)*time.Millisecond, func() {
notification.close(CloseReasonExpire)
})
}
notifications.notifications[id] = notification
output()
return id, nil
}
|