forked from coyim/coyim
-
Notifications
You must be signed in to change notification settings - Fork 0
/
ui_reader.go
126 lines (103 loc) · 2.76 KB
/
ui_reader.go
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
package gui
import (
"fmt"
"io/ioutil"
"log"
"os"
"path/filepath"
"strings"
"reflect"
"github.com/twstrike/coyim/Godeps/_workspace/src/github.com/twstrike/gotk3adapter/glibi"
"github.com/twstrike/coyim/Godeps/_workspace/src/github.com/twstrike/gotk3adapter/gtki"
"github.com/twstrike/coyim/gui/definitions"
)
const (
defsFolder = "gui/definitions"
xmlExtension = ".xml"
)
func getActualDefsFolder() string {
wd, _ := os.Getwd()
if strings.HasSuffix(wd, "/gui") {
return "definitions"
}
return "gui/definitions"
}
func getDefinitionWithFileFallback(uiName string) string {
// this makes sure a missing definition wont break only when the app is released
uiDef := getDefinition(uiName)
fileName := filepath.Join(getActualDefsFolder(), uiName+xmlExtension)
if fileNotFound(fileName) {
log.Printf("gui: loading compiled definition %q\n", uiName)
return uiDef.String()
}
return readFile(fileName)
}
// This must be called from the UI thread - otherwise bad things will happen sooner or later
func builderForDefinition(uiName string) gtki.Builder {
template := getDefinitionWithFileFallback(uiName)
builder, err := g.gtk.BuilderNew()
if err != nil {
//We cant recover from this
panic(err)
}
//XXX Why are we using AddFromString rather than NewFromString
err = builder.AddFromString(template)
if err != nil {
//This is a programming error
panic(fmt.Sprintf("gui: failed load %s: %s\n", uiName, err.Error()))
}
return builder
}
func fileNotFound(fileName string) bool {
_, fnf := os.Stat(fileName)
return os.IsNotExist(fnf)
}
func readFile(fileName string) string {
data, _ := ioutil.ReadFile(fileName)
return string(data)
}
func getDefinition(uiName string) fmt.Stringer {
def, ok := definitions.Get(uiName)
if !ok {
panic(fmt.Sprintf("No definition found for %s", uiName))
}
return def
}
type builder struct {
gtki.Builder
}
func newBuilder(filename string) *builder {
return newBuilderFromString(filename)
}
func newBuilderFromString(uiName string) *builder {
return &builder{builderForDefinition(uiName)}
}
func (b *builder) getObj(name string) glibi.Object {
obj, _ := b.GetObject(name)
return obj
}
func (b *builder) getItem(name string, target interface{}) {
v := reflect.ValueOf(target)
if v.Kind() != reflect.Ptr {
panic("builder.getItem() target argument must be a pointer")
}
elem := v.Elem()
elem.Set(reflect.ValueOf(b.get(name)))
}
func (b *builder) getItems(args ...interface{}) {
for len(args) >= 2 {
name, ok := args[0].(string)
if !ok {
panic("string argument expected in builder.getItems()")
}
b.getItem(name, args[1])
args = args[2:]
}
}
func (b *builder) get(name string) glibi.Object {
obj, err := b.GetObject(name)
if err != nil {
panic("builder.GetObject() failed: " + err.Error())
}
return obj
}