DOM improvements; fix unit tests

This commit is contained in:
Vadim Lopatin 2015-12-23 13:41:59 +03:00
parent e2a847f108
commit 34fe23adca
1 changed files with 117 additions and 25 deletions

View File

@ -11,11 +11,11 @@ import std.algorithm : equal;
// Namespace, element tag and attribute names are stored as numeric ids for better performance and lesser memory consumption. // Namespace, element tag and attribute names are stored as numeric ids for better performance and lesser memory consumption.
/// id type for interning namespaces /// id type for interning namespaces
alias ns_id = ushort; alias ns_id = short;
/// id type for interning element names /// id type for interning element names
alias elem_id = uint; alias elem_id = int;
/// id type for interning attribute names /// id type for interning attribute names
alias attr_id = ushort; alias attr_id = short;
/// Base class for DOM nodes /// Base class for DOM nodes
@ -52,6 +52,22 @@ public:
/// returns attribute count /// returns attribute count
@property int attrCount() { return 0; } @property int attrCount() { return 0; }
/// get attribute by index
Attribute attr(int index) { return null; }
/// get attribute by namespace and attribute ids
Attribute attr(ns_id nsid, attr_id attrid) { return null; }
/// get attribute by namespace and attribute names
Attribute attr(string nsname, string attrname) { return attr(_document.nsId(nsname), _document.attrId(attrname)); }
/// set attribute value by namespace and attribute ids
Attribute setAttr(ns_id nsid, attr_id attrid, string value) { assert(false); }
/// set attribute value by namespace and attribute names
Attribute setAttr(string nsname, string attrname, string value) { return setAttr(_document.nsId(nsname), _document.attrId(attrname), value); }
/// get attribute value by namespace and attribute ids
string attrValue(ns_id nsid, attr_id attrid) { return null; }
/// get attribute value by namespace and attribute ids
string attrValue(string nsname, string attrname) { return attrValue(_document.nsId(nsname), _document.attrId(attrname)); }
// child nodes // child nodes
/// returns child node count /// returns child node count
@ -71,16 +87,17 @@ public:
/// append text child /// append text child
Node appendText(dstring s, int index = -1) { assert(false); } Node appendText(dstring s, int index = -1) { assert(false); }
/// append element child - by namespace and tag names /// append element child - by namespace and tag names
Node appendElement(string ns, string tag, int index = -1) { assert(false); } Node appendElement(string ns, string tag, int index = -1) { return appendElement(_document.nsId(ns), _document.tagId(tag), index); }
/// append element child - by namespace and tag ids /// append element child - by namespace and tag ids
Node appendElement(ns_id ns, elem_id tag, int index = -1) { assert(false); } Node appendElement(ns_id ns, elem_id tag, int index = -1) { assert(false); }
// Text methods
/// node text /// node text
@property dstring text() { return null; } @property dstring text() { return null; }
/// ditto /// ditto
@property void text(dstring s) { } @property void text(dstring s) { }
} }
/// Text node /// Text node
@ -102,6 +119,7 @@ public:
class Element : Node { class Element : Node {
private: private:
Collection!Node _children; Collection!Node _children;
Collection!Attribute _attrs;
elem_id _id; // element tag id elem_id _id; // element tag id
ns_id _ns; // element namespace id ns_id _ns; // element namespace id
@ -117,6 +135,45 @@ public:
/// return element namespace id /// return element namespace id
override @property ns_id nsid() { return _ns; } override @property ns_id nsid() { return _ns; }
// Attributes
/// returns attribute count
override @property int attrCount() { return cast(int)_attrs.length; }
/// get attribute by index
override Attribute attr(int index) { return index >= 0 && index < _attrs.length ? _attrs[index] : null; }
/// get attribute by namespace and attribute ids
override Attribute attr(ns_id nsid, attr_id attrid) {
foreach (a; _attrs)
if ((nsid == Ns.any || nsid == a.nsid) && attrid == a.id)
return a;
return null;
}
/// get attribute by namespace and attribute names
override Attribute attr(string nsname, string attrname) { return attr(_document.nsId(nsname), _document.attrId(attrname)); }
/// set attribute value by namespace and attribute ids
override Attribute setAttr(ns_id nsid, attr_id attrid, string value) {
Attribute a = attr(nsid, attrid);
if (!a) {
a = new Attribute(this, nsid, attrid, value);
_attrs.add(a);
} else {
a.value = value;
}
return a;
}
/// set attribute value by namespace and attribute names
override Attribute setAttr(string nsname, string attrname, string value) { return setAttr(_document.nsId(nsname), _document.attrId(attrname), value); }
/// get attribute value by namespace and attribute ids
override string attrValue(ns_id nsid, attr_id attrid) {
if (Attribute a = attr(nsid, attrid))
return a.value;
return null;
}
/// get attribute value by namespace and attribute ids
override string attrValue(string nsname, string attrname) { return attrValue(_document.nsId(nsname), _document.attrId(attrname)); }
// child nodes // child nodes
/// returns child node count /// returns child node count
@ -141,18 +198,14 @@ public:
_children.add(item, index >= 0 ? index : size_t.max); _children.add(item, index >= 0 ? index : size_t.max);
return item; return item;
} }
/// append element child - by namespace and tag names
override Node appendElement(string ns, string tag, int index = -1) {
Node item = document.createElement(ns, tag);
_children.add(item, index >= 0 ? index : size_t.max);
return item;
}
/// append element child - by namespace and tag ids /// append element child - by namespace and tag ids
override Node appendElement(ns_id ns, elem_id tag, int index = -1) { override Node appendElement(ns_id ns, elem_id tag, int index = -1) {
Node item = document.createElement(ns, tag); Node item = document.createElement(ns, tag);
_children.add(item, index >= 0 ? index : size_t.max); _children.add(item, index >= 0 ? index : size_t.max);
return item; return item;
} }
/// append element child - by namespace and tag names
override Node appendElement(string ns, string tag, int index = -1) { return appendElement(_document.nsId(ns), _document.tagId(tag), index); }
} }
/// Document node /// Document node
@ -175,7 +228,7 @@ public:
} }
/// create element node by namespace and tag names /// create element node by namespace and tag names
Element createElement(string ns, string tag) { Element createElement(string ns, string tag) {
return new Element(this, internNs(ns), internTag(tag)); return new Element(this, nsId(ns), tagId(tag));
} }
// Ids // Ids
@ -193,19 +246,19 @@ public:
return _attrIds[id]; return _attrIds[id];
} }
/// get id for element tag name /// get id for element tag name
elem_id internTag(string s) { elem_id tagId(string s) {
if (s.empty) if (s.empty)
return 0; return 0;
return _elemIds.intern(s); return _elemIds.intern(s);
} }
/// get id for namespace name /// get id for namespace name
ns_id internNs(string s) { ns_id nsId(string s) {
if (s.empty) if (s.empty)
return 0; return 0;
return _nsIds.intern(s); return _nsIds.intern(s);
} }
/// get id for namespace name /// get id for attribute name
attr_id internAttr(string s) { attr_id attrId(string s) {
if (s.empty) if (s.empty)
return 0; return 0;
return _attrIds.intern(s); return _attrIds.intern(s);
@ -216,7 +269,38 @@ private:
IdentMap!(ns_id) _nsIds; IdentMap!(ns_id) _nsIds;
} }
class Attribute {
private:
attr_id _id;
ns_id _nsid;
string _value;
Node _parent;
this(Node parent, ns_id nsid, attr_id id, string value) {
_parent = parent;
_nsid = nsid;
_id = id;
_value = value;
}
public:
/// Parent element which owns this attribute
@property Node parent() { return _parent; }
/// Parent element document
@property Document document() { return _parent.document; }
/// get attribute id
@property attr_id id() { return _id; }
/// get attribute namespace id
@property ns_id nsid() { return _nsid; }
/// get attribute tag name
@property string name() { return document.tagName(_id); }
/// get attribute namespace name
@property string nsname() { return document.nsName(_nsid); }
/// get attribute value
@property string value() { return _value; }
/// set attribute value
@property void value(string s) { _value = s; }
}
/// remove trailing _ from string, e.g. "body_" -> "body" /// remove trailing _ from string, e.g. "body_" -> "body"
private string removeTrailingUnderscore(string s) { private string removeTrailingUnderscore(string s) {
@ -230,7 +314,7 @@ struct IdentMap(ident_t) {
/// initialize with elements of enum /// initialize with elements of enum
void init(E)() if (is(E == enum)) { void init(E)() if (is(E == enum)) {
foreach(member; EnumMembers!E) { foreach(member; EnumMembers!E) {
static if (member.to!int) { static if (member.to!int > 0) {
//pragma(msg, "interning string '" ~ removeTrailingUnderscore(member.to!string) ~ "' for " ~ E.stringof); //pragma(msg, "interning string '" ~ removeTrailingUnderscore(member.to!string) ~ "' for " ~ E.stringof);
intern(removeTrailingUnderscore(member.to!string), member); intern(removeTrailingUnderscore(member.to!string), member);
} }
@ -275,8 +359,8 @@ private:
} }
/// standard tags /// standard tags
enum Tag { enum Tag : elem_id {
NONE, none,
body_, body_,
pre, pre,
div, div,
@ -284,16 +368,17 @@ enum Tag {
} }
/// standard attributes /// standard attributes
enum Attr { enum Attr : attr_id {
NONE, none,
id, id,
class_, class_,
style style
} }
/// standard namespaces /// standard namespaces
enum Ns { enum Ns : ns_id {
NONE, any = -1,
none = 0,
xmlns, xmlns,
xs, xs,
xlink, xlink,
@ -301,7 +386,7 @@ enum Ns {
xsi xsi
} }
unittest { void testDOM() {
import std.algorithm : equal; import std.algorithm : equal;
//import std.stdio; //import std.stdio;
IdentMap!(elem_id) map; IdentMap!(elem_id) map;
@ -317,11 +402,18 @@ unittest {
assert(body_.name.equal("body")); assert(body_.name.equal("body"));
auto div = body_.appendElement(null, "div"); auto div = body_.appendElement(null, "div");
assert(body_.childCount == 1); assert(body_.childCount == 1);
assert(div.id == Tag.body_); assert(div.id == Tag.div);
assert(div.name.equal("div")); assert(div.name.equal("div"));
div.appendText("Some text"d); div.appendText("Some text"d);
assert(div.childCount == 1); assert(div.childCount == 1);
assert(div.child(0).text.equal("Some text"d)); assert(div.child(0).text.equal("Some text"d));
div.setAttr(Ns.none, Attr.id, "div_id");
assert(div.attrValue(Ns.none, Attr.id).equal("div_id"));
destroy(doc); destroy(doc);
} }
unittest {
testDOM();
}