Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(199)

Unified Diff: pkg/third_party/html5lib/lib/dom.dart

Issue 814113004: Pull args, intl, logging, shelf, and source_maps out of the SDK. (Closed) Base URL: https://dart.googlecode.com/svn/branches/bleeding_edge/dart
Patch Set: Also csslib. Created 6 years ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
« no previous file with comments | « pkg/third_party/html5lib/README.md ('k') | pkg/third_party/html5lib/lib/dom_parsing.dart » ('j') | no next file with comments »
Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
Index: pkg/third_party/html5lib/lib/dom.dart
diff --git a/pkg/third_party/html5lib/lib/dom.dart b/pkg/third_party/html5lib/lib/dom.dart
deleted file mode 100644
index 3c5b332911444c43719e9f463946a2040f16b6e0..0000000000000000000000000000000000000000
--- a/pkg/third_party/html5lib/lib/dom.dart
+++ /dev/null
@@ -1,1020 +0,0 @@
-/// A simple tree API that results from parsing html. Intended to be compatible
-/// with dart:html, but it is missing many types and APIs.
-library dom;
-
-// TODO(jmesserly): lots to do here. Originally I wanted to generate this using
-// our Blink IDL generator, but another idea is to directly use the excellent
-// http://dom.spec.whatwg.org/ and http://html.spec.whatwg.org/ and just
-// implement that.
-
-import 'dart:collection';
-import 'package:source_span/source_span.dart';
-
-import 'src/constants.dart';
-import 'src/css_class_set.dart';
-import 'src/list_proxy.dart';
-import 'src/query_selector.dart' as query;
-import 'src/token.dart';
-import 'src/tokenizer.dart';
-import 'dom_parsing.dart';
-import 'parser.dart';
-
-export 'src/css_class_set.dart' show CssClassSet;
-
-// TODO(jmesserly): this needs to be replaced by an AttributeMap for attributes
-// that exposes namespace info.
-class AttributeName implements Comparable {
- /// The namespace prefix, e.g. `xlink`.
- final String prefix;
-
- /// The attribute name, e.g. `title`.
- final String name;
-
- /// The namespace url, e.g. `http://www.w3.org/1999/xlink`
- final String namespace;
-
- const AttributeName(this.prefix, this.name, this.namespace);
-
- String toString() {
- // Implement:
- // http://www.whatwg.org/specs/web-apps/current-work/multipage/the-end.html#serializing-html-fragments
- // If we get here we know we are xml, xmlns, or xlink, because of
- // [HtmlParser.adjustForeignAttriubtes] is the only place we create
- // an AttributeName.
- return prefix != null ? '$prefix:$name' : name;
- }
-
- int get hashCode {
- int h = prefix.hashCode;
- h = 37 * (h & 0x1FFFFF) + name.hashCode;
- h = 37 * (h & 0x1FFFFF) + namespace.hashCode;
- return h & 0x3FFFFFFF;
- }
-
- int compareTo(other) {
- // Not sure about this sort order
- if (other is! AttributeName) return 1;
- int cmp = (prefix != null ? prefix : "").compareTo(
- (other.prefix != null ? other.prefix : ""));
- if (cmp != 0) return cmp;
- cmp = name.compareTo(other.name);
- if (cmp != 0) return cmp;
- return namespace.compareTo(other.namespace);
- }
-
- bool operator ==(x) {
- if (x is! AttributeName) return false;
- return prefix == x.prefix && name == x.name && namespace == x.namespace;
- }
-}
-
-// http://dom.spec.whatwg.org/#parentnode
-abstract class _ParentNode implements Node {
- // TODO(jmesserly): this is only a partial implementation
-
- /// Seaches for the first descendant node matching the given selectors, using
- /// a preorder traversal.
- ///
- /// NOTE: Not all selectors from
- /// [selectors level 4](http://dev.w3.org/csswg/selectors-4/)
- /// are implemented. For example, nth-child does not implement An+B syntax
- /// and *-of-type is not implemented. If a selector is not implemented this
- /// method will throw [UniplmentedError].
- Element querySelector(String selector) =>
- query.querySelector(this, selector);
-
- /// Returns all descendant nodes matching the given selectors, using a
- /// preorder traversal.
- ///
- /// NOTE: Not all selectors from
- /// [selectors level 4](http://dev.w3.org/csswg/selectors-4/)
- /// are implemented. For example, nth-child does not implement An+B syntax
- /// and *-of-type is not implemented. If a selector is not implemented this
- /// method will throw [UniplmentedError].
- List<Element> querySelectorAll(String selector) =>
- query.querySelectorAll(this, selector);
-}
-
-// http://dom.spec.whatwg.org/#interface-nonelementparentnode
-abstract class _NonElementParentNode implements _ParentNode {
- // TODO(jmesserly): could be faster, should throw on invalid id.
- Element getElementById(String id) => querySelector('#$id');
-}
-
-// This doesn't exist as an interface in the spec, but it's useful to merge
-// common methods from these:
-// http://dom.spec.whatwg.org/#interface-document
-// http://dom.spec.whatwg.org/#element
-abstract class _ElementAndDocument implements _ParentNode {
- // TODO(jmesserly): could be faster, should throw on invalid tag/class names.
-
- List<Element> getElementsByTagName(String localName) =>
- querySelectorAll(localName);
-
- List<Element> getElementsByClassName(String classNames) =>
- querySelectorAll(classNames.splitMapJoin(' ',
- onNonMatch: (m) => m.isNotEmpty ? '.$m' : m,
- onMatch: (m) => ''));
-}
-
-/// Really basic implementation of a DOM-core like Node.
-abstract class Node {
- static const int ATTRIBUTE_NODE = 2;
- static const int CDATA_SECTION_NODE = 4;
- static const int COMMENT_NODE = 8;
- static const int DOCUMENT_FRAGMENT_NODE = 11;
- static const int DOCUMENT_NODE = 9;
- static const int DOCUMENT_TYPE_NODE = 10;
- static const int ELEMENT_NODE = 1;
- static const int ENTITY_NODE = 6;
- static const int ENTITY_REFERENCE_NODE = 5;
- static const int NOTATION_NODE = 12;
- static const int PROCESSING_INSTRUCTION_NODE = 7;
- static const int TEXT_NODE = 3;
-
- /// The parent of the current node (or null for the document node).
- Node parentNode;
-
- /// The parent element of this node.
- ///
- /// Returns null if this node either does not have a parent or its parent is
- /// not an element.
- Element get parent => parentNode is Element ? parentNode : null;
-
- // TODO(jmesserly): should move to Element.
- /// A map holding name, value pairs for attributes of the node.
- ///
- /// Note that attribute order needs to be stable for serialization, so we use
- /// a LinkedHashMap. Each key is a [String] or [AttributeName].
- LinkedHashMap<dynamic, String> attributes = new LinkedHashMap();
-
- /// A list of child nodes of the current node. This must
- /// include all elements but not necessarily other node types.
- final NodeList nodes = new NodeList._();
-
- List<Element> _elements;
-
- // TODO(jmesserly): consider using an Expando for this, and put it in
- // dom_parsing. Need to check the performance affect.
- /// The source span of this node, if it was created by the [HtmlParser].
- FileSpan sourceSpan;
-
- /// The attribute spans if requested. Otherwise null.
- LinkedHashMap<dynamic, FileSpan> _attributeSpans;
- LinkedHashMap<dynamic, FileSpan> _attributeValueSpans;
-
- Node._() {
- nodes._parent = this;
- }
-
- /// If [sourceSpan] is available, this contains the spans of each attribute.
- /// The span of an attribute is the entire attribute, including the name and
- /// quotes (if any). For example, the span of "attr" in `<a attr="value">`
- /// would be the text `attr="value"`.
- LinkedHashMap<dynamic, FileSpan> get attributeSpans {
- _ensureAttributeSpans();
- return _attributeSpans;
- }
-
- /// If [sourceSpan] is available, this contains the spans of each attribute's
- /// value. Unlike [attributeSpans], this span will inlcude only the value.
- /// For example, the value span of "attr" in `<a attr="value">` would be the
- /// text `value`.
- LinkedHashMap<dynamic, FileSpan> get attributeValueSpans {
- _ensureAttributeSpans();
- return _attributeValueSpans;
- }
-
- List<Element> get children {
- if (_elements == null) {
- _elements = new FilteredElementList(this);
- }
- return _elements;
- }
-
- /// Returns a copy of this node.
- ///
- /// If [deep] is `true`, then all of this node's children and decendents are
- /// copied as well. If [deep] is `false`, then only this node is copied.
- Node clone(bool deep);
-
- int get nodeType;
-
- // http://domparsing.spec.whatwg.org/#extensions-to-the-element-interface
- String get _outerHtml {
- var str = new StringBuffer();
- _addOuterHtml(str);
- return str.toString();
- }
-
- String get _innerHtml {
- var str = new StringBuffer();
- _addInnerHtml(str);
- return str.toString();
- }
-
- // Implemented per: http://dom.spec.whatwg.org/#dom-node-textcontent
- String get text => null;
- set text(String value) {}
-
- void append(Node node) => nodes.add(node);
-
- Node get firstChild => nodes.isNotEmpty ? nodes[0] : null;
-
- void _addOuterHtml(StringBuffer str);
-
- void _addInnerHtml(StringBuffer str) {
- for (Node child in nodes) child._addOuterHtml(str);
- }
-
- Node remove() {
- // TODO(jmesserly): is parent == null an error?
- if (parentNode != null) {
- parentNode.nodes.remove(this);
- }
- return this;
- }
-
- /// Insert [node] as a child of the current node, before [refNode] in the
- /// list of child nodes. Raises [UnsupportedOperationException] if [refNode]
- /// is not a child of the current node. If refNode is null, this adds to the
- /// end of the list.
- void insertBefore(Node node, Node refNode) {
- if (refNode == null) {
- nodes.add(node);
- } else {
- nodes.insert(nodes.indexOf(refNode), node);
- }
- }
-
- /// Replaces this node with another node.
- Node replaceWith(Node otherNode) {
- if (parentNode == null) {
- throw new UnsupportedError('Node must have a parent to replace it.');
- }
- parentNode.nodes[parentNode.nodes.indexOf(this)] = otherNode;
- return this;
- }
-
- // TODO(jmesserly): should this be a property or remove?
- /// Return true if the node has children or text.
- bool hasContent() => nodes.length > 0;
-
- /// Move all the children of the current node to [newParent].
- /// This is needed so that trees that don't store text as nodes move the
- /// text in the correct way.
- void reparentChildren(Node newParent) {
- newParent.nodes.addAll(nodes);
- nodes.clear();
- }
-
- bool hasChildNodes() => !nodes.isEmpty;
-
- bool contains(Node node) => nodes.contains(node);
-
- /// Checks if this is a type selector.
- /// See <http://www.w3.org/TR/CSS2/grammar.html>.
- /// Note: this doesn't support '*', the universal selector, non-ascii chars or
- /// escape chars.
- bool _isTypeSelector(String selector) {
- // Parser:
-
- // element_name
- // : IDENT | '*'
- // ;
-
- // Lexer:
-
- // nmstart [_a-z]|{nonascii}|{escape}
- // nmchar [_a-z0-9-]|{nonascii}|{escape}
- // ident -?{nmstart}{nmchar}*
- // nonascii [\240-\377]
- // unicode \\{h}{1,6}(\r\n|[ \t\r\n\f])?
- // escape {unicode}|\\[^\r\n\f0-9a-f]
-
- // As mentioned above, no nonascii or escape support yet.
- int len = selector.length;
- if (len == 0) return false;
-
- int i = 0;
- const int DASH = 45;
- if (selector.codeUnitAt(i) == DASH) i++;
-
- if (i >= len || !isLetter(selector[i])) return false;
- i++;
-
- for (; i < len; i++) {
- if (!isLetterOrDigit(selector[i]) && selector.codeUnitAt(i) != DASH) {
- return false;
- }
- }
-
- return true;
- }
-
- /// Initialize [attributeSpans] using [sourceSpan].
- void _ensureAttributeSpans() {
- if (_attributeSpans != null) return;
-
- _attributeSpans = new LinkedHashMap<dynamic, FileSpan>();
- _attributeValueSpans = new LinkedHashMap<dynamic, FileSpan>();
-
- if (sourceSpan == null) return;
-
- var tokenizer = new HtmlTokenizer(sourceSpan.text, generateSpans: true,
- attributeSpans: true);
-
- tokenizer.moveNext();
- var token = tokenizer.current as StartTagToken;
-
- if (token.attributeSpans == null) return; // no attributes
-
- for (var attr in token.attributeSpans) {
- var offset = sourceSpan.start.offset;
- _attributeSpans[attr.name] = sourceSpan.file.span(
- offset + attr.start, offset + attr.end);
- if (attr.startValue != null) {
- _attributeValueSpans[attr.name] = sourceSpan.file.span(
- offset + attr.startValue, offset + attr.endValue);
- }
- }
- }
-
- _clone(Node shallowClone, bool deep) {
- if (deep) {
- for (var child in nodes) {
- shallowClone.append(child.clone(true));
- }
- }
- return shallowClone;
- }
-}
-
-class Document extends Node
- with _ParentNode, _NonElementParentNode, _ElementAndDocument {
-
- Document() : super._();
- factory Document.html(String html) => parse(html);
-
- int get nodeType => Node.DOCUMENT_NODE;
-
- // TODO(jmesserly): optmize this if needed
- Element get documentElement => querySelector('html');
- Element get head => documentElement.querySelector('head');
- Element get body => documentElement.querySelector('body');
-
- /// Returns a fragment of HTML or XML that represents the element and its
- /// contents.
- // TODO(jmesserly): this API is not specified in:
- // <http://domparsing.spec.whatwg.org/> nor is it in dart:html, instead
- // only Element has outerHtml. However it is quite useful. Should we move it
- // to dom_parsing, where we keep other custom APIs?
- String get outerHtml => _outerHtml;
-
- String toString() => "#document";
-
- void _addOuterHtml(StringBuffer str) => _addInnerHtml(str);
-
- Document clone(bool deep) => _clone(new Document(), deep);
-
- Element createElement(String tag) => new Element.tag(tag);
-
- // TODO(jmesserly): this is only a partial implementation of:
- // http://dom.spec.whatwg.org/#dom-document-createelementns
- Element createElementNS(String namespaceUri, String tag) {
- if (namespaceUri == '') namespaceUri = null;
- return new Element._(tag, namespaceUri);
- }
-
- DocumentFragment createDocumentFragment() => new DocumentFragment();
-}
-
-class DocumentFragment extends Node
- with _ParentNode, _NonElementParentNode {
-
- DocumentFragment() : super._();
- factory DocumentFragment.html(String html) => parseFragment(html);
-
- int get nodeType => Node.DOCUMENT_FRAGMENT_NODE;
-
- /// Returns a fragment of HTML or XML that represents the element and its
- /// contents.
- // TODO(jmesserly): this API is not specified in:
- // <http://domparsing.spec.whatwg.org/> nor is it in dart:html, instead
- // only Element has outerHtml. However it is quite useful. Should we move it
- // to dom_parsing, where we keep other custom APIs?
- String get outerHtml => _outerHtml;
-
- String toString() => "#document-fragment";
-
- DocumentFragment clone(bool deep) => _clone(new DocumentFragment(), deep);
-
- void _addOuterHtml(StringBuffer str) => _addInnerHtml(str);
-
- String get text => _getText(this);
- set text(String value) => _setText(this, value);
-}
-
-class DocumentType extends Node {
- final String name;
- final String publicId;
- final String systemId;
-
- DocumentType(String name, this.publicId, this.systemId)
- // Note: once Node.tagName is removed, don't pass "name" to super
- : name = name, super._();
-
- int get nodeType => Node.DOCUMENT_TYPE_NODE;
-
- String toString() {
- if (publicId != null || systemId != null) {
- // TODO(jmesserly): the html5 serialization spec does not add these. But
- // it seems useful, and the parser can handle it, so for now keeping it.
- var pid = publicId != null ? publicId : '';
- var sid = systemId != null ? systemId : '';
- return '<!DOCTYPE $name "$pid" "$sid">';
- } else {
- return '<!DOCTYPE $name>';
- }
- }
-
-
- void _addOuterHtml(StringBuffer str) {
- str.write(toString());
- }
-
- DocumentType clone(bool deep) => new DocumentType(name, publicId, systemId);
-}
-
-class Text extends Node {
- String data;
-
- Text(this.data) : super._();
-
- int get nodeType => Node.TEXT_NODE;
-
- String toString() => '"$data"';
-
- void _addOuterHtml(StringBuffer str) => writeTextNodeAsHtml(str, this);
-
- Text clone(bool deep) => new Text(data);
-
- String get text => data;
- set text(String value) { data = value; }
-}
-
-// TODO(jmesserly): Elements should have a pointer back to their document
-class Element extends Node with _ParentNode, _ElementAndDocument {
- final String namespaceUri;
-
- /// The [local name](http://dom.spec.whatwg.org/#concept-element-local-name)
- /// of this element.
- final String localName;
-
- Element._(this.localName, [this.namespaceUri]) : super._();
-
- Element.tag(this.localName) : namespaceUri = Namespaces.html, super._();
-
- static final _START_TAG_REGEXP = new RegExp('<(\\w+)');
-
- static final _CUSTOM_PARENT_TAG_MAP = const {
- 'body': 'html',
- 'head': 'html',
- 'caption': 'table',
- 'td': 'tr',
- 'colgroup': 'table',
- 'col': 'colgroup',
- 'tr': 'tbody',
- 'tbody': 'table',
- 'tfoot': 'table',
- 'thead': 'table',
- 'track': 'audio',
- };
-
- // TODO(jmesserly): this is from dart:html _ElementFactoryProvider...
- // TODO(jmesserly): have a look at fixing some things in dart:html, in
- // particular: is the parent tag map complete? Is it faster without regexp?
- // TODO(jmesserly): for our version we can do something smarter in the parser.
- // All we really need is to set the correct parse state.
- factory Element.html(String html) {
-
- // TODO(jacobr): this method can be made more robust and performant.
- // 1) Cache the dummy parent elements required to use innerHTML rather than
- // creating them every call.
- // 2) Verify that the html does not contain leading or trailing text nodes.
- // 3) Verify that the html does not contain both <head> and <body> tags.
- // 4) Detatch the created element from its dummy parent.
- String parentTag = 'div';
- String tag;
- final match = _START_TAG_REGEXP.firstMatch(html);
- if (match != null) {
- tag = match.group(1).toLowerCase();
- if (_CUSTOM_PARENT_TAG_MAP.containsKey(tag)) {
- parentTag = _CUSTOM_PARENT_TAG_MAP[tag];
- }
- }
-
- var fragment = parseFragment(html, container: parentTag);
- Element element;
- if (fragment.children.length == 1) {
- element = fragment.children[0];
- } else if (parentTag == 'html' && fragment.children.length == 2) {
- // You'll always get a head and a body when starting from html.
- element = fragment.children[tag == 'head' ? 0 : 1];
- } else {
- throw new ArgumentError('HTML had ${fragment.children.length} '
- 'top level elements but 1 expected');
- }
- element.remove();
- return element;
- }
-
- int get nodeType => Node.ELEMENT_NODE;
-
- // TODO(jmesserly): we can make this faster
- Element get previousElementSibling {
- if (parentNode == null) return null;
- var siblings = parentNode.nodes;
- for (int i = siblings.indexOf(this) - 1; i >= 0; i--) {
- var s = siblings[i];
- if (s is Element) return s;
- }
- return null;
- }
-
- Element get nextElementSibling {
- if (parentNode == null) return null;
- var siblings = parentNode.nodes;
- for (int i = siblings.indexOf(this) + 1; i < siblings.length; i++) {
- var s = siblings[i];
- if (s is Element) return s;
- }
- return null;
- }
-
- String toString() {
- var prefix = Namespaces.getPrefix(namespaceUri);
- return "<${prefix == null ? '' : '$prefix '}$localName>";
- }
-
- String get text => _getText(this);
- set text(String value) => _setText(this, value);
-
- /// Returns a fragment of HTML or XML that represents the element and its
- /// contents.
- String get outerHtml => _outerHtml;
-
- /// Returns a fragment of HTML or XML that represents the element's contents.
- /// Can be set, to replace the contents of the element with nodes parsed from
- /// the given string.
- String get innerHtml => _innerHtml;
- // TODO(jmesserly): deprecate in favor of:
- // <https://api.dartlang.org/apidocs/channels/stable/#dart-dom-html.Element@id_setInnerHtml>
- set innerHtml(String value) {
- nodes.clear();
- // TODO(jmesserly): should be able to get the same effect by adding the
- // fragment directly.
- nodes.addAll(parseFragment(value, container: localName).nodes);
- }
-
- void _addOuterHtml(StringBuffer str) {
- // http://www.whatwg.org/specs/web-apps/current-work/multipage/the-end.html#serializing-html-fragments
- // Element is the most complicated one.
- str.write('<${_getSerializationPrefix(namespaceUri)}$localName');
-
- if (attributes.length > 0) {
- attributes.forEach((key, v) {
- // Note: AttributeName.toString handles serialization of attribute
- // namespace, if needed.
- str.write(' $key="${htmlSerializeEscape(v, attributeMode: true)}"');
- });
- }
-
- str.write('>');
-
- if (nodes.length > 0) {
- if (localName == 'pre' || localName == 'textarea' ||
- localName == 'listing') {
- final first = nodes[0];
- if (first is Text && first.data.startsWith('\n')) {
- // These nodes will remove a leading \n at parse time, so if we still
- // have one, it means we started with two. Add it back.
- str.write('\n');
- }
- }
-
- _addInnerHtml(str);
- }
-
- // void elements must not have an end tag
- // http://dev.w3.org/html5/markup/syntax.html#void-elements
- if (!isVoidElement(localName)) str.write('</$localName>');
- }
-
- static String _getSerializationPrefix(String uri) {
- if (uri == null ||
- uri == Namespaces.html ||
- uri == Namespaces.mathml ||
- uri == Namespaces.svg) {
- return '';
- }
- var prefix = Namespaces.getPrefix(uri);
- // TODO(jmesserly): the spec doesn't define "qualified name".
- // I'm not sure if this is correct, but it should parse reasonably.
- return prefix == null ? '' : '$prefix:';
- }
-
- Element clone(bool deep) {
- var result = new Element._(localName, namespaceUri)
- ..attributes = new LinkedHashMap.from(attributes);
- return _clone(result, deep);
- }
-
- // http://dom.spec.whatwg.org/#dom-element-id
- String get id {
- var result = attributes['id'];
- return result != null ? result : '';
- }
-
- set id(String value) {
- attributes['id'] = '$value';
- }
-
- // http://dom.spec.whatwg.org/#dom-element-classname
- String get className {
- var result = attributes['class'];
- return result != null ? result : '';
- }
-
- set className(String value) {
- attributes['class'] = '$value';
- }
-
- /**
- * The set of CSS classes applied to this element.
- *
- * This set makes it easy to add, remove or toggle the classes applied to
- * this element.
- *
- * element.classes.add('selected');
- * element.classes.toggle('isOnline');
- * element.classes.remove('selected');
- */
- CssClassSet get classes => new ElementCssClassSet(this);
-}
-
-class Comment extends Node {
- String data;
-
- Comment(this.data) : super._();
-
- int get nodeType => Node.COMMENT_NODE;
-
- String toString() => "<!-- $data -->";
-
- void _addOuterHtml(StringBuffer str) {
- str.write("<!--$data-->");
- }
-
- Comment clone(bool deep) => new Comment(data);
-
- String get text => data;
- set text(String value) {
- this.data = value;
- }
-}
-
-
-// TODO(jmesserly): fix this to extend one of the corelib classes if possible.
-// (The requirement to remove the node from the old node list makes it tricky.)
-// TODO(jmesserly): is there any way to share code with the _NodeListImpl?
-class NodeList extends ListProxy<Node> {
- // Note: this is conceptually final, but because of circular reference
- // between Node and NodeList we initialize it after construction.
- Node _parent;
-
- NodeList._();
-
- Node get first => this[0];
-
- Node _setParent(Node node) {
- // Note: we need to remove the node from its previous parent node, if any,
- // before updating its parent pointer to point at our parent.
- node.remove();
- node.parentNode = _parent;
- return node;
- }
-
- void add(Node value) {
- if (value is DocumentFragment) {
- addAll(value.nodes);
- } else {
- super.add(_setParent(value));
- }
- }
-
- void addLast(Node value) => add(value);
-
- void addAll(Iterable<Node> collection) {
- // Note: we need to be careful if collection is another NodeList.
- // In particular:
- // 1. we need to copy the items before updating their parent pointers,
- // _flattenDocFragments does a copy internally.
- // 2. we should update parent pointers in reverse order. That way they
- // are removed from the original NodeList (if any) from the end, which
- // is faster.
- var list = _flattenDocFragments(collection);
- for (var node in list.reversed) _setParent(node);
- super.addAll(list);
- }
-
- void insert(int index, Node value) {
- if (value is DocumentFragment) {
- insertAll(index, value.nodes);
- } else {
- super.insert(index, _setParent(value));
- }
- }
-
- Node removeLast() => super.removeLast()..parentNode = null;
-
- Node removeAt(int i) => super.removeAt(i)..parentNode = null;
-
- void clear() {
- for (var node in this) node.parentNode = null;
- super.clear();
- }
-
- void operator []=(int index, Node value) {
- if (value is DocumentFragment) {
- removeAt(index);
- insertAll(index, value.nodes);
- } else {
- this[index].parentNode = null;
- super[index] = _setParent(value);
- }
- }
-
- // TODO(jmesserly): These aren't implemented in DOM _NodeListImpl, see
- // http://code.google.com/p/dart/issues/detail?id=5371
- void setRange(int start, int rangeLength, List<Node> from,
- [int startFrom = 0]) {
- if (from is NodeList) {
- // Note: this is presumed to make a copy
- from = from.sublist(startFrom, startFrom + rangeLength);
- }
- // Note: see comment in [addAll]. We need to be careful about the order of
- // operations if [from] is also a NodeList.
- for (int i = rangeLength - 1; i >= 0; i--) {
- this[start + i] = from[startFrom + i];
- }
- }
-
- void replaceRange(int start, int end, Iterable<Node> newContents) {
- removeRange(start, end);
- insertAll(start, newContents);
- }
-
- void removeRange(int start, int rangeLength) {
- for (int i = start; i < rangeLength; i++) this[i].parentNode = null;
- super.removeRange(start, rangeLength);
- }
-
- void removeWhere(bool test(Element e)) {
- for (var node in where(test)) {
- node.parentNode = null;
- }
- super.removeWhere(test);
- }
-
- void retainWhere(bool test(Element e)) {
- for (var node in where((n) => !test(n))) {
- node.parentNode = null;
- }
- super.retainWhere(test);
- }
-
- void insertAll(int index, Iterable<Node> collection) {
- // Note: we need to be careful how we copy nodes. See note in addAll.
- var list = _flattenDocFragments(collection);
- for (var node in list.reversed) _setParent(node);
- super.insertAll(index, list);
- }
-
- _flattenDocFragments(Iterable<Node> collection) {
- // Note: this function serves two purposes:
- // * it flattens document fragments
- // * it creates a copy of [collections] when `collection is NodeList`.
- var result = [];
- for (var node in collection) {
- if (node is DocumentFragment) {
- result.addAll(node.nodes);
- } else {
- result.add(node);
- }
- }
- return result;
- }
-}
-
-
-/// An indexable collection of a node's descendants in the document tree,
-/// filtered so that only elements are in the collection.
-// TODO(jmesserly): this was copied from dart:html
-// TODO(jmesserly): "implements List<Element>" is a workaround for analyzer bug.
-class FilteredElementList extends IterableBase<Element> with ListMixin<Element>
- implements List<Element> {
-
- final Node _node;
- final List<Node> _childNodes;
-
- /// Creates a collection of the elements that descend from a node.
- ///
- /// Example usage:
- ///
- /// var filteredElements = new FilteredElementList(query("#container"));
- /// // filteredElements is [a, b, c].
- FilteredElementList(Node node): _childNodes = node.nodes, _node = node;
-
- // We can't memoize this, since it's possible that children will be messed
- // with externally to this class.
- //
- // TODO(nweiz): we don't always need to create a new list. For example
- // forEach, every, any, ... could directly work on the _childNodes.
- List<Element> get _filtered =>
- new List<Element>.from(_childNodes.where((n) => n is Element));
-
- void forEach(void f(Element element)) {
- _filtered.forEach(f);
- }
-
- void operator []=(int index, Element value) {
- this[index].replaceWith(value);
- }
-
- void set length(int newLength) {
- final len = this.length;
- if (newLength >= len) {
- return;
- } else if (newLength < 0) {
- throw new ArgumentError("Invalid list length");
- }
-
- removeRange(newLength, len);
- }
-
- String join([String separator = ""]) => _filtered.join(separator);
-
- void add(Element value) {
- _childNodes.add(value);
- }
-
- void addAll(Iterable<Element> iterable) {
- for (Element element in iterable) {
- add(element);
- }
- }
-
- bool contains(Element element) {
- return element is Element && _childNodes.contains(element);
- }
-
- Iterable<Element> get reversed => _filtered.reversed;
-
- void sort([int compare(Element a, Element b)]) {
- throw new UnsupportedError('TODO(jacobr): should we impl?');
- }
-
- void setRange(int start, int end, Iterable<Element> iterable,
- [int skipCount = 0]) {
- throw new UnimplementedError();
- }
-
- void fillRange(int start, int end, [Element fillValue]) {
- throw new UnimplementedError();
- }
-
- void replaceRange(int start, int end, Iterable<Element> iterable) {
- throw new UnimplementedError();
- }
-
- void removeRange(int start, int end) {
- _filtered.sublist(start, end).forEach((el) => el.remove());
- }
-
- void clear() {
- // Currently, ElementList#clear clears even non-element nodes, so we follow
- // that behavior.
- _childNodes.clear();
- }
-
- Element removeLast() {
- final result = this.last;
- if (result != null) {
- result.remove();
- }
- return result;
- }
-
- Iterable map(f(Element element)) => _filtered.map(f);
- Iterable<Element> where(bool f(Element element)) => _filtered.where(f);
- Iterable expand(Iterable f(Element element)) => _filtered.expand(f);
-
- void insert(int index, Element value) {
- _childNodes.insert(index, value);
- }
-
- void insertAll(int index, Iterable<Element> iterable) {
- _childNodes.insertAll(index, iterable);
- }
-
- Element removeAt(int index) {
- final result = this[index];
- result.remove();
- return result;
- }
-
- bool remove(Object element) {
- if (element is! Element) return false;
- for (int i = 0; i < length; i++) {
- Element indexElement = this[i];
- if (identical(indexElement, element)) {
- indexElement.remove();
- return true;
- }
- }
- return false;
- }
-
- Element reduce(Element combine(Element value, Element element)) {
- return _filtered.reduce(combine);
- }
-
- dynamic fold(dynamic initialValue,
- dynamic combine(dynamic previousValue, Element element)) {
- return _filtered.fold(initialValue, combine);
- }
-
- bool every(bool f(Element element)) => _filtered.every(f);
- bool any(bool f(Element element)) => _filtered.any(f);
- List<Element> toList({ bool growable: true }) =>
- new List<Element>.from(this, growable: growable);
- Set<Element> toSet() => new Set<Element>.from(this);
- Element firstWhere(bool test(Element value), {Element orElse()}) {
- return _filtered.firstWhere(test, orElse: orElse);
- }
-
- Element lastWhere(bool test(Element value), {Element orElse()}) {
- return _filtered.lastWhere(test, orElse: orElse);
- }
-
- Element singleWhere(bool test(Element value)) {
- return _filtered.singleWhere(test);
- }
-
- Element elementAt(int index) {
- return this[index];
- }
-
- bool get isEmpty => _filtered.isEmpty;
- int get length => _filtered.length;
- Element operator [](int index) => _filtered[index];
- Iterator<Element> get iterator => _filtered.iterator;
- List<Element> sublist(int start, [int end]) =>
- _filtered.sublist(start, end);
- Iterable<Element> getRange(int start, int end) =>
- _filtered.getRange(start, end);
- int indexOf(Element element, [int start = 0]) =>
- _filtered.indexOf(element, start);
-
- int lastIndexOf(Element element, [int start = null]) {
- if (start == null) start = length - 1;
- return _filtered.lastIndexOf(element, start);
- }
-
- Element get first => _filtered.first;
-
- Element get last => _filtered.last;
-
- Element get single => _filtered.single;
-}
-
-// http://dom.spec.whatwg.org/#dom-node-textcontent
-// For Element and DocumentFragment
-String _getText(Node node) =>
- (new _ConcatTextVisitor()..visit(node)).toString();
-
-void _setText(Node node, String value) {
- node.nodes.clear();
- node.append(new Text(value));
-}
-
-class _ConcatTextVisitor extends TreeVisitor {
- final _str = new StringBuffer();
-
- String toString() => _str.toString();
-
- visitText(Text node) {
- _str.write(node.data);
- }
-}
« no previous file with comments | « pkg/third_party/html5lib/README.md ('k') | pkg/third_party/html5lib/lib/dom_parsing.dart » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698