Optimize memory for selector cache
This commit is contained in:
parent
aaa6de784b
commit
b34c3eaf82
1 changed files with 80 additions and 49 deletions
|
|
@ -4,66 +4,97 @@ var Vnode = require("../render/vnode")
|
||||||
|
|
||||||
var selectorParser = /(?:(^|#|\.)([^#\.\[\]]+))|(\[(.+?)(?:\s*=\s*("|'|)((?:\\["'\]]|.)*?)\5)?\])/g
|
var selectorParser = /(?:(^|#|\.)([^#\.\[\]]+))|(\[(.+?)(?:\s*=\s*("|'|)((?:\\["'\]]|.)*?)\5)?\])/g
|
||||||
var selectorCache = {}
|
var selectorCache = {}
|
||||||
|
var hasOwn = {}.hasOwnProperty
|
||||||
|
|
||||||
|
function compileSelector(selector) {
|
||||||
|
var match, tag = "div", classes = [], attrs = {}
|
||||||
|
while (match = selectorParser.exec(selector)) {
|
||||||
|
var type = match[1], value = match[2]
|
||||||
|
if (type === "" && value !== "") tag = value
|
||||||
|
else if (type === "#") attrs.id = value
|
||||||
|
else if (type === ".") classes.push(value)
|
||||||
|
else if (match[3][0] === "[") {
|
||||||
|
var attrValue = match[6]
|
||||||
|
if (attrValue) attrValue = attrValue.replace(/\\(["'])/g, "$1").replace(/\\\\/g, "\\")
|
||||||
|
if (match[4] === "class") classes.push(attrValue)
|
||||||
|
else attrs[match[4]] = attrValue || true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (classes.length > 0) attrs.className = classes.join(" ")
|
||||||
|
return selectorCache[selector] = {tag: tag, attrs: attrs}
|
||||||
|
}
|
||||||
|
|
||||||
|
function execSelector(state, attrs, children) {
|
||||||
|
var hasAttrs = false, childList, text
|
||||||
|
var className = attrs.className || attrs.class
|
||||||
|
|
||||||
|
for (var key in state.attrs) {
|
||||||
|
if (hasOwn.call(state.attrs, key)) {
|
||||||
|
attrs[key] = state.attrs[key]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (className != null) {
|
||||||
|
if (attrs.class != null) {
|
||||||
|
attrs.class = undefined
|
||||||
|
attrs.className = className
|
||||||
|
}
|
||||||
|
|
||||||
|
if (state.attrs.className != null) {
|
||||||
|
attrs.className = state.attrs.className + " " + className
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for (var key in attrs) {
|
||||||
|
if (hasOwn.call(attrs, key) && key !== "key") {
|
||||||
|
hasAttrs = true
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (Array.isArray(children) && children.length === 1 && children[0] != null && children[0].tag === "#") {
|
||||||
|
text = children[0].children
|
||||||
|
} else {
|
||||||
|
childList = children
|
||||||
|
}
|
||||||
|
|
||||||
|
return Vnode(state.tag, attrs.key, hasAttrs ? attrs : undefined, childList, text)
|
||||||
|
}
|
||||||
|
|
||||||
function hyperscript(selector) {
|
function hyperscript(selector) {
|
||||||
|
// Because sloppy mode sucks
|
||||||
|
var attrs = arguments[1], start = 2, children
|
||||||
|
|
||||||
if (selector == null || typeof selector !== "string" && typeof selector !== "function" && typeof selector.view !== "function") {
|
if (selector == null || typeof selector !== "string" && typeof selector !== "function" && typeof selector.view !== "function") {
|
||||||
throw Error("The selector must be either a string or a component.");
|
throw Error("The selector must be either a string or a component.");
|
||||||
}
|
}
|
||||||
|
|
||||||
if (typeof selector === "string" && selectorCache[selector] === undefined) {
|
if (typeof selector === "string") {
|
||||||
var match, tag, classes = [], attributes = {}
|
var cached = selectorCache[selector] || compileSelector(selector)
|
||||||
while (match = selectorParser.exec(selector)) {
|
}
|
||||||
var type = match[1], value = match[2]
|
|
||||||
if (type === "" && value !== "") tag = value
|
|
||||||
else if (type === "#") attributes.id = value
|
|
||||||
else if (type === ".") classes.push(value)
|
|
||||||
else if (match[3][0] === "[") {
|
|
||||||
var attrValue = match[6]
|
|
||||||
if (attrValue) attrValue = attrValue.replace(/\\(["'])/g, "$1").replace(/\\\\/g, "\\")
|
|
||||||
if (match[4] === "class") classes.push(attrValue)
|
|
||||||
else attributes[match[4]] = attrValue || true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (classes.length > 0) attributes.className = classes.join(" ")
|
|
||||||
selectorCache[selector] = function(attrs, children) {
|
|
||||||
var hasAttrs = false, childList, text
|
|
||||||
var className = attrs.className || attrs.class
|
|
||||||
for (var key in attributes) attrs[key] = attributes[key]
|
|
||||||
if (className !== undefined) {
|
|
||||||
if (attrs.class !== undefined) {
|
|
||||||
attrs.class = undefined
|
|
||||||
attrs.className = className
|
|
||||||
}
|
|
||||||
if (attributes.className !== undefined) attrs.className = attributes.className + " " + className
|
|
||||||
}
|
|
||||||
for (var key in attrs) {
|
|
||||||
if (key !== "key") {
|
|
||||||
hasAttrs = true
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (Array.isArray(children) && children.length == 1 && children[0] != null && children[0].tag === "#") text = children[0].children
|
|
||||||
else childList = children
|
|
||||||
|
|
||||||
return Vnode(tag || "div", attrs.key, hasAttrs ? attrs : undefined, childList, text, undefined)
|
if (!attrs) {
|
||||||
}
|
attrs = {}
|
||||||
|
} else if (typeof attrs !== "object" || attrs.tag != null || Array.isArray(attrs)) {
|
||||||
|
attrs = {}
|
||||||
|
start = 1
|
||||||
}
|
}
|
||||||
var attrs, children, childrenIndex
|
|
||||||
if (arguments[1] == null || typeof arguments[1] === "object" && arguments[1].tag === undefined && !Array.isArray(arguments[1])) {
|
if (arguments.length === start + 1) {
|
||||||
attrs = arguments[1]
|
children = arguments[start]
|
||||||
childrenIndex = 2
|
if (!Array.isArray(children)) children = [children]
|
||||||
}
|
} else {
|
||||||
else childrenIndex = 1
|
|
||||||
if (arguments.length === childrenIndex + 1) {
|
|
||||||
children = Array.isArray(arguments[childrenIndex]) ? arguments[childrenIndex] : [arguments[childrenIndex]]
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
children = []
|
children = []
|
||||||
for (var i = childrenIndex; i < arguments.length; i++) children.push(arguments[i])
|
while (start < arguments.length) children.push(arguments[start++])
|
||||||
}
|
}
|
||||||
|
|
||||||
if (typeof selector === "string") return selectorCache[selector](attrs || {}, Vnode.normalizeChildren(children))
|
var normalized = Vnode.normalizeChildren(children)
|
||||||
|
|
||||||
return Vnode(selector, attrs && attrs.key, attrs || {}, Vnode.normalizeChildren(children), undefined, undefined)
|
if (typeof selector === "string") {
|
||||||
|
return execSelector(cached, attrs, normalized)
|
||||||
|
} else {
|
||||||
|
return Vnode(selector, attrs.key, attrs, normalized)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
module.exports = hyperscript
|
module.exports = hyperscript
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue