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
|
import { capitalize } from './utils.js'
import { ns } from '../modules/core/namespaces.js'
import Base from '../types/Base.js'
const elements = {}
export const root = Symbol('root')
// Method for element creation
export function makeNode (name) {
// create element
return document.createElementNS(ns, name)
}
export function makeInstance (element) {
if (element instanceof Base) return element
if (typeof element === 'object') {
return adopt(element)
}
if (element == null) {
return new elements[root]()
}
if (typeof element === 'string' && element.charAt(0) !== '<') {
return adopt(document.querySelector(element))
}
var node = makeNode('svg')
node.innerHTML = element
// We can use firstChild here because we know,
// that the first char is < and thus an element
element = adopt(node.firstChild)
return element
}
export function nodeOrNew (name, node) {
return node instanceof window.Node ? node : makeNode(name)
}
// Adopt existing svg elements
export function adopt (node) {
// check for presence of node
if (!node) return null
// make sure a node isn't already adopted
if (node.instance instanceof Base) return node.instance
if (!(node instanceof window.SVGElement)) {
return new elements.HtmlNode(node)
}
// initialize variables
var element
// adopt with element-specific settings
if (node.nodeName === 'svg') {
element = new elements[root](node)
} else if (node.nodeName === 'linearGradient' || node.nodeName === 'radialGradient') {
element = new elements.Gradient(node)
} else if (elements[capitalize(node.nodeName)]) {
element = new elements[capitalize(node.nodeName)](node)
} else {
element = new elements.Bare(node)
}
return element
}
export function register (element, name = element.name, asRoot = false) {
elements[name] = element
if (asRoot) elements[root] = element
return element
}
export function getClass (name) {
return elements[name]
}
// Element id sequence
let did = 1000
// Get next named element id
export function eid (name) {
return 'Svgjs' + capitalize(name) + (did++)
}
// Deep new id assignment
export function assignNewId (node) {
// do the same for SVG child nodes as well
for (var i = node.children.length - 1; i >= 0; i--) {
assignNewId(node.children[i])
}
if (node.id) {
return adopt(node).id(eid(node.nodeName))
}
return adopt(node)
}
// Method for extending objects
export function extend (modules, methods, attrCheck) {
var key, i
modules = Array.isArray(modules) ? modules : [modules]
for (i = modules.length - 1; i >= 0; i--) {
for (key in methods) {
let method = methods[key]
if (attrCheck) {
method = wrapWithAttrCheck(methods[key])
}
modules[i].prototype[key] = method
}
}
}
export function extendWithAttrCheck (...args) {
extend(...args, true)
}
export function wrapWithAttrCheck (fn) {
return function (...args) {
let o = args[args.length - 1]
if (o && o.constructor === Object && !(o instanceof Array)) {
return fn.apply(this, args.slice(0, -1)).attr(o)
} else {
return fn.apply(this, args)
}
}
}
|