blob: 9babe3d3c0911ab077ba6e69e438a30785ed6159 [file] [log] [blame]
// Copyright (c) 2011, the Dart project authors. Please see the AUTHORS file
// for details. All rights reserved. Use of this source code is governed by a
// BSD-style license that can be found in the LICENSE file.
class Validate {
static int _classNameCheck(var selector, int matches) {
if (selector.isCombinatorDescendant() ||
(selector.isCombinatorNone() && matches == 0)) {
if (matches < 0) {
String tooMany = selector.simpleSelector.toString();
throw new CssSelectorException(
'Can not mix Id selector with class selector(s). Id ' +
'selector must be singleton too many starting at $tooMany');
}
return matches + 1;
} else {
String error = selector.toString();
throw new CssSelectorException(
'Selectors can not have combinators (>, +, or ~) before $error');
}
}
static int _elementIdCheck(var selector, int matches) {
if (selector.isCombinatorNone() && matches == 0) {
// Perfect just one element id returns matches of -1.
return -1;
} else if (selector.isCombinatorDescendant()) {
String tooMany = selector.simpleSelector.toString();
throw new CssSelectorException(
'Use of Id selector must be singleton starting at $tooMany');
} else {
String error = selector.simpleSelector.toString();
throw new CssSelectorException(
'Selectors can not have combinators (>, +, or ~) before $error');
}
}
// Validate the @{css expression} only .class and #elementId are valid inside
// of @{...}.
static template(List<ASTNode> selectors, CssWorld cssWorld) {
var errorSelector; // signal which selector didn't match.
bool found = false; // signal if a selector is matched.
int matches = 0; // < 0 IdSelectors, > 0 ClassSelector
// At most one selector group (any number of simple selector sequences).
assert(selectors.length <= 1);
for (final sels in selectors) {
for (final selector in sels.simpleSelectorSequences) {
found = false;
var simpleSelector = selector.simpleSelector;
if (simpleSelector is ClassSelector) {
// Any class name starting with an underscore is a private class name
// that doesn't have to match the world of known classes.
if (!simpleSelector.name.startsWith('_')) {
// TODO(terry): For now iterate through all classes look for faster
// mechanism hash map, etc.
for (final className in cssWorld.classes) {
if (selector.simpleSelector.name == className) {
matches = _classNameCheck(selector, matches);
found = true; // .class found.
break;
}
for (final className2 in cssWorld.classes) {
print(className2);
}
}
} else {
// Don't check any class name that is prefixed with an underscore.
// However, signal as found and bump up matches; it's a valid class
// name.
matches = _classNameCheck(selector, matches);
found = true; // ._class are always okay.
}
} else if (simpleSelector is IdSelector) {
// Any element id starting with an underscore is a private element id
// that doesn't have to match the world of known elemtn ids.
if (!simpleSelector.name.startsWith('_')) {
for (final id in cssWorld.ids) {
if (simpleSelector.name == id) {
matches = _elementIdCheck(selector, matches);
found = true; // #id found.
break;
}
}
} else {
// Don't check any element ID that is prefixed with an underscore.
// Signal as found and bump up matches; it's a valid element ID.
matches = _elementIdCheck(selector, matches);
found = true; // #_id are always okay
}
} else {
String badSelector = simpleSelector.toString();
throw new CssSelectorException(
'Invalid template selector $badSelector');
}
if (!found) {
String unknownName = simpleSelector.toString();
throw new CssSelectorException('Unknown selector name $unknownName');
}
}
}
// Every selector must match.
var selector = selectors[0];
assert((matches >= 0 ? matches : -matches) ==
selector.simpleSelectorSequences.length);
}
}