2013-01-22 10:32:26 -08:00
|
|
|
/**
|
|
|
|
* Dumb HTML views. Mostly to demonstrate how the visitor pattern over these
|
|
|
|
* views works, as driven by validation. I'm not convinced it's actually a good
|
|
|
|
* idea to do validation.
|
|
|
|
*
|
|
|
|
* @provides javelin-view-html
|
|
|
|
* @requires javelin-install
|
2013-05-18 17:04:22 -07:00
|
|
|
* javelin-dom
|
|
|
|
* javelin-view-visitor
|
|
|
|
* javelin-util
|
2013-01-22 10:32:26 -08:00
|
|
|
*/
|
|
|
|
|
|
|
|
JX.install('HTMLView', {
|
|
|
|
extend: 'View',
|
|
|
|
members : {
|
|
|
|
render: function(rendered_children) {
|
|
|
|
return JX.$N(this.getName(), this.getAllAttributes(), rendered_children);
|
|
|
|
},
|
|
|
|
validate: function() {
|
|
|
|
this.accept(JX.HTMLView.getValidatingVisitor());
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
statics: {
|
|
|
|
getValidatingVisitor: function() {
|
|
|
|
return new JX.ViewVisitor(JX.HTMLView.validate);
|
|
|
|
},
|
|
|
|
|
2015-01-14 07:59:56 +11:00
|
|
|
validate: function(view) {
|
2013-01-22 10:32:26 -08:00
|
|
|
var spec = this._getHTMLSpec();
|
2013-05-18 17:04:22 -07:00
|
|
|
if (!(view.getName() in spec)) {
|
2015-01-14 07:59:56 +11:00
|
|
|
throw new Error('invalid tag');
|
2013-01-22 10:32:26 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
var tag_spec = spec[view.getName()];
|
|
|
|
|
|
|
|
var attrs = view.getAllAttributes();
|
|
|
|
for (var attr in attrs) {
|
|
|
|
if (!(attr in tag_spec)) {
|
2015-01-14 07:59:56 +11:00
|
|
|
throw new Error('invalid attr');
|
2013-01-22 10:32:26 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
var validator = tag_spec[attr];
|
2015-01-14 07:59:56 +11:00
|
|
|
if (typeof validator === 'function') {
|
2013-01-22 10:32:26 -08:00
|
|
|
return validator(attrs[attr]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
},
|
|
|
|
|
|
|
|
_validateRel: function(target) {
|
|
|
|
return target in {
|
2015-01-14 07:59:56 +11:00
|
|
|
'_blank': 1,
|
|
|
|
'_self': 1,
|
|
|
|
'_parent': 1,
|
|
|
|
'_top': 1
|
2013-01-22 10:32:26 -08:00
|
|
|
};
|
|
|
|
},
|
|
|
|
_getHTMLSpec: function() {
|
|
|
|
var attrs_any_can_have = {
|
|
|
|
className: 1,
|
|
|
|
id: 1,
|
|
|
|
sigil: 1
|
|
|
|
};
|
|
|
|
|
|
|
|
var form_elem_attrs = {
|
|
|
|
name: 1,
|
|
|
|
value: 1
|
|
|
|
};
|
|
|
|
|
|
|
|
var spec = {
|
|
|
|
a: { href: 1, target: JX.HTMLView._validateRel },
|
|
|
|
b: {},
|
|
|
|
blockquote: {},
|
|
|
|
br: {},
|
|
|
|
button: JX.copy({}, form_elem_attrs),
|
|
|
|
canvas: {},
|
|
|
|
code: {},
|
|
|
|
dd: {},
|
|
|
|
div: {},
|
|
|
|
dl: {},
|
|
|
|
dt: {},
|
|
|
|
em: {},
|
|
|
|
embed: {},
|
|
|
|
fieldset: {},
|
|
|
|
form: { type: 1 },
|
|
|
|
h1: {},
|
|
|
|
h2: {},
|
|
|
|
h3: {},
|
|
|
|
h4: {},
|
|
|
|
h5: {},
|
|
|
|
h6: {},
|
|
|
|
hr: {},
|
|
|
|
i: {},
|
|
|
|
iframe: { src: 1 },
|
|
|
|
img: { src: 1, alt: 1 },
|
|
|
|
input: JX.copy({}, form_elem_attrs),
|
|
|
|
label: {'for': 1},
|
|
|
|
li: {},
|
|
|
|
ol: {},
|
|
|
|
optgroup: {},
|
|
|
|
option: JX.copy({}, form_elem_attrs),
|
|
|
|
p: {},
|
|
|
|
pre: {},
|
|
|
|
q: {},
|
|
|
|
select: {},
|
|
|
|
span: {},
|
|
|
|
strong: {},
|
|
|
|
sub: {},
|
|
|
|
sup: {},
|
|
|
|
table: {},
|
|
|
|
tbody: {},
|
|
|
|
td: {},
|
|
|
|
textarea: {},
|
|
|
|
tfoot: {},
|
|
|
|
th: {},
|
|
|
|
thead: {},
|
|
|
|
tr: {},
|
|
|
|
ul: {}
|
|
|
|
};
|
|
|
|
|
|
|
|
for (var k in spec) {
|
|
|
|
JX.copy(spec[k], attrs_any_can_have);
|
|
|
|
}
|
|
|
|
|
|
|
|
return spec;
|
|
|
|
},
|
|
|
|
registerToInterpreter: function(view_interpreter) {
|
|
|
|
var spec = this._getHTMLSpec();
|
|
|
|
for (var tag in spec) {
|
|
|
|
view_interpreter.register(tag, JX.HTMLView);
|
|
|
|
}
|
|
|
|
return view_interpreter;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|