forked from hexops/vecty
-
Notifications
You must be signed in to change notification settings - Fork 0
/
markup.go
276 lines (246 loc) · 7.76 KB
/
markup.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
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
package vecty
import (
"reflect"
"github.com/gopherjs/gopherjs/js"
)
// EventListener is markup that specifies a callback function to be invoked when
// the named DOM event is fired.
type EventListener struct {
Name string
Listener func(*Event)
callPreventDefault bool
callStopPropagation bool
wrapper func(jsEvent *js.Object)
}
// PreventDefault prevents the default behavior of the event from occurring.
//
// See https://developer.mozilla.org/en-US/docs/Web/API/Event/preventDefault.
func (l *EventListener) PreventDefault() *EventListener {
l.callPreventDefault = true
return l
}
// StopPropagation prevents further propagation of the current event in the
// capturing and bubbling phases.
//
// See https://developer.mozilla.org/en-US/docs/Web/API/Event/stopPropagation.
func (l *EventListener) StopPropagation() *EventListener {
l.callStopPropagation = true
return l
}
// Apply implements the Applyer interface.
func (l *EventListener) Apply(h *HTML) {
h.eventListeners = append(h.eventListeners, l)
}
// Event represents a DOM event.
type Event struct {
*js.Object
Target *js.Object
}
// MarkupOrChild represents one of:
//
// Component
// *HTML
// List
// KeyedList
// nil
// MarkupList
//
// An unexported method on this interface ensures at compile time that the
// underlying value must be one of these types.
type MarkupOrChild interface {
isMarkupOrChild()
}
func apply(m MarkupOrChild, h *HTML) {
switch m := m.(type) {
case MarkupList:
m.Apply(h)
case nil:
h.children = append(h.children, nil)
case Component, *HTML, List, KeyedList:
h.children = append(h.children, m.(ComponentOrHTML))
default:
panic("vecty: internal error (unexpected MarkupOrChild type " + reflect.TypeOf(m).String() + ")")
}
}
// Applyer represents some type of markup (a style, property, data, etc) which
// can be applied to a given HTML element or text node.
type Applyer interface {
// Apply applies the markup to the given HTML element or text node.
Apply(h *HTML)
}
type markupFunc func(h *HTML)
func (m markupFunc) Apply(h *HTML) { m(h) }
// Style returns an Applyer which applies the given CSS style. Generally, this
// function is not used directly but rather the style subpackage (which is type
// safe) should be used instead.
func Style(key, value string) Applyer {
return markupFunc(func(h *HTML) {
if h.styles == nil {
h.styles = make(map[string]string)
}
h.styles[key] = value
})
}
// Key returns an Applyer that uniquely identifies the HTML element amongst its
// siblings. When used, all other sibling elements and components must also be
// keyed.
func Key(key interface{}) Applyer {
return markupFunc(func(h *HTML) {
h.key = key
})
}
// Property returns an Applyer which applies the given JavaScript property to an
// HTML element or text node. Generally, this function is not used directly but
// rather the prop and style subpackages (which are type safe) should be used instead.
//
// To set style, use style package or Style. Property panics if key is "style".
func Property(key string, value interface{}) Applyer {
if key == "style" {
panic(`vecty: Property called with key "style"; style package or Style should be used instead`)
}
return markupFunc(func(h *HTML) {
if h.properties == nil {
h.properties = make(map[string]interface{})
}
h.properties[key] = value
})
}
// Attribute returns an Applyer which applies the given attribute to an element.
//
// In most situations, you should use Property function, or the prop subpackage
// (which is type-safe) instead. There are only a few attributes (aria-*, role,
// etc) which do not have equivalent properties. Always opt for the property
// first, before relying on an attribute.
func Attribute(key string, value interface{}) Applyer {
return markupFunc(func(h *HTML) {
if h.attributes == nil {
h.attributes = make(map[string]interface{})
}
h.attributes[key] = value
})
}
// Data returns an Applyer which applies the given data attribute.
func Data(key, value string) Applyer {
return markupFunc(func(h *HTML) {
if h.dataset == nil {
h.dataset = make(map[string]string)
}
h.dataset[key] = value
})
}
// Class returns an Applyer which applies the provided classes. Subsequent
// calls to this function will append additional classes. To toggle classes,
// use ClassMap instead. Each class name must be passed as a separate argument.
func Class(class ...string) Applyer {
mustValidateClassNames(class)
return markupFunc(func(h *HTML) {
if h.classes == nil {
h.classes = make(map[string]struct{})
}
for _, name := range class {
h.classes[name] = struct{}{}
}
})
}
// mustValidateClassNames ensures no class names have spaces
// and panics with clear instructions on how to fix this user error.
func mustValidateClassNames(class []string) {
for _, name := range class {
if containsSpace(name) {
panic(`vecty: invalid argument to vecty.Class "` + name + `" (string may not contain spaces)`)
}
}
}
// containsSpace reports whether s contains a space character.
func containsSpace(s string) bool {
for _, c := range s {
if c == ' ' {
return true
}
}
return false
}
// ClassMap is markup that specifies classes to be applied to an element if
// their boolean value are true.
type ClassMap map[string]bool
// Apply implements the Applyer interface.
func (m ClassMap) Apply(h *HTML) {
if h.classes == nil {
h.classes = make(map[string]struct{})
}
for name, active := range m {
if !active {
delete(h.classes, name)
continue
}
h.classes[name] = struct{}{}
}
}
// MarkupList represents a list of Applyer which is individually
// applied to an HTML element or text node.
//
// It may only be created through the Markup function.
type MarkupList struct {
list []Applyer
}
// Apply implements the Applyer interface.
func (m MarkupList) Apply(h *HTML) {
for _, a := range m.list {
if a == nil {
continue
}
a.Apply(h)
}
}
// isMarkupOrChild implements MarkupOrChild
func (m MarkupList) isMarkupOrChild() {}
// Markup wraps a list of Applyer which is individually
// applied to an HTML element or text node.
func Markup(m ...Applyer) MarkupList {
// returns public non-pointer struct value with private field so that users
// must acquire a MarkupList only from this function, and so that it can
// never be nil (which would make it indistinguishable from (*HTML)(nil) in
// a call to e.g. Tag).
return MarkupList{list: m}
}
// If returns nil if cond is false, otherwise it returns the given children.
func If(cond bool, children ...ComponentOrHTML) MarkupOrChild {
if cond {
return List(children)
}
return nil
}
// MarkupIf returns nil if cond is false, otherwise it returns the given markup.
func MarkupIf(cond bool, markup ...Applyer) Applyer {
if cond {
return Markup(markup...)
}
return nil
}
// UnsafeHTML is Applyer which unsafely sets the inner HTML of an HTML element.
//
// It is entirely up to the caller to ensure the input HTML is properly
// sanitized.
//
// It is akin to innerHTML in standard JavaScript and dangerouslySetInnerHTML
// in React, and is said to be unsafe because Vecty makes no effort to validate
// or ensure the HTML is safe for insertion in the DOM. If the HTML came from a
// user, for example, it would create a cross-site-scripting (XSS) exploit in
// the application.
//
// The returned Applyer can only be applied to HTML, not vecty.Text, or else a
// panic will occur.
func UnsafeHTML(html string) Applyer {
return markupFunc(func(h *HTML) {
h.innerHTML = html
})
}
// Namespace is Applyer which sets the namespace URI to associate with the
// created element. This is primarily used when working with, e.g., SVG.
//
// See https://developer.mozilla.org/en-US/docs/Web/API/Document/createElementNS#Valid Namespace URIs
func Namespace(uri string) Applyer {
return markupFunc(func(h *HTML) {
h.namespace = uri
})
}