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

Unified Diff: pkg/csslib/lib/src/analyzer.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/csslib/lib/parser.dart ('k') | pkg/csslib/lib/src/css_printer.dart » ('j') | no next file with comments »
Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
Index: pkg/csslib/lib/src/analyzer.dart
diff --git a/pkg/csslib/lib/src/analyzer.dart b/pkg/csslib/lib/src/analyzer.dart
deleted file mode 100644
index 6758a2d66013532ae6675094dbe14ce4d88ed798..0000000000000000000000000000000000000000
--- a/pkg/csslib/lib/src/analyzer.dart
+++ /dev/null
@@ -1,1017 +0,0 @@
-// Copyright (c) 2012, 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.
-
-part of csslib.parser;
-
-
-// TODO(terry): Add optimizing phase to remove duplicated selectors in the same
-// selector group (e.g., .btn, .btn { color: red; }). Also, look
-// at simplifying selectors expressions too (much harder).
-// TODO(terry): Detect invalid directive usage. All @imports must occur before
-// all rules other than @charset directive. Any @import directive
-// after any non @charset or @import directive are ignored. e.g.,
-// @import "a.css";
-// div { color: red; }
-// @import "b.css";
-// becomes:
-// @import "a.css";
-// div { color: red; }
-// <http://www.w3.org/TR/css3-syntax/#at-rules>
-
-/**
- * Analysis phase will validate/fixup any new CSS feature or any SASS style
- * feature.
- */
-class Analyzer {
- final List<StyleSheet> _styleSheets;
- final Messages _messages;
-
- Analyzer(this._styleSheets, this._messages);
-
- // TODO(terry): Currently each feature walks the AST each time. Once we have
- // our complete feature set consider benchmarking the cost and
- // possibly combine in one walk.
- void run() {
- // Expand top-level @include.
- _styleSheets.forEach((styleSheet) =>
- TopLevelIncludes.expand(_messages, _styleSheets));
-
- // Expand @include in declarations.
- _styleSheets.forEach((styleSheet) =>
- DeclarationIncludes.expand(_messages, _styleSheets));
-
- // Remove all @mixin and @include
- _styleSheets.forEach((styleSheet) => MixinsAndIncludes.remove(styleSheet));
-
- // Expand any nested selectors using selector desendant combinator to
- // signal CSS inheritance notation.
- _styleSheets.forEach((styleSheet) => new ExpandNestedSelectors()
- ..visitStyleSheet(styleSheet)
- ..flatten(styleSheet));
-
- // Expand any @extend.
- _styleSheets.forEach((styleSheet) {
- var allExtends = new AllExtends()..visitStyleSheet(styleSheet);
- new InheritExtends(_messages, allExtends)..visitStyleSheet(styleSheet);
- });
- }
-}
-
-/**
- * Traverse all rulesets looking for nested ones. If a ruleset is in a
- * declaration group (implies nested selector) then generate new ruleset(s) at
- * level 0 of CSS using selector inheritance syntax (flattens the nesting).
- *
- * How the AST works for a rule [RuleSet] and nested rules. First of all a
- * CSS rule [RuleSet] consist of a selector and a declaration e.g.,
- *
- * selector {
- * declaration
- * }
- *
- * AST structure of a [RuleSet] is:
- *
- * RuleSet
- * SelectorGroup
- * List<Selector>
- * List<SimpleSelectorSequence>
- * Combinator // +, >, ~, DESCENDENT, or NONE
- * SimpleSelector // class, id, element, namespace, attribute
- * DeclarationGroup
- * List // Declaration or RuleSet
- *
- * For the simple rule:
- *
- * div + span { color: red; }
- *
- * the AST [RuleSet] is:
- *
- * RuleSet
- * SelectorGroup
- * List<Selector>
- * [0]
- * List<SimpleSelectorSequence>
- * [0] Combinator = COMBINATOR_NONE
- * ElementSelector (name = div)
- * [1] Combinator = COMBINATOR_PLUS
- * ElementSelector (name = span)
- * DeclarationGroup
- * List // Declarations or RuleSets
- * [0]
- * Declaration (property = color, expression = red)
- *
- * Usually a SelectorGroup contains 1 Selector. Consider the selectors:
- *
- * div { color: red; }
- * a { color: red; }
- *
- * are equivalent to
- *
- * div, a { color : red; }
- *
- * In the above the RuleSet would have a SelectorGroup with 2 selectors e.g.,
- *
- * RuleSet
- * SelectorGroup
- * List<Selector>
- * [0]
- * List<SimpleSelectorSequence>
- * [0] Combinator = COMBINATOR_NONE
- * ElementSelector (name = div)
- * [1]
- * List<SimpleSelectorSequence>
- * [0] Combinator = COMBINATOR_NONE
- * ElementSelector (name = a)
- * DeclarationGroup
- * List // Declarations or RuleSets
- * [0]
- * Declaration (property = color, expression = red)
- *
- * For a nested rule e.g.,
- *
- * div {
- * color : blue;
- * a { color : red; }
- * }
- *
- * Would map to the follow CSS rules:
- *
- * div { color: blue; }
- * div a { color: red; }
- *
- * The AST for the former nested rule is:
- *
- * RuleSet
- * SelectorGroup
- * List<Selector>
- * [0]
- * List<SimpleSelectorSequence>
- * [0] Combinator = COMBINATOR_NONE
- * ElementSelector (name = div)
- * DeclarationGroup
- * List // Declarations or RuleSets
- * [0]
- * Declaration (property = color, expression = blue)
- * [1]
- * RuleSet
- * SelectorGroup
- * List<Selector>
- * [0]
- * List<SimpleSelectorSequence>
- * [0] Combinator = COMBINATOR_NONE
- * ElementSelector (name = a)
- * DeclarationGroup
- * List // Declarations or RuleSets
- * [0]
- * Declaration (property = color, expression = red)
- *
- * Nested rules is a terse mechanism to describe CSS inheritance. The analyzer
- * will flatten and expand the nested rules to it's flatten strucure. Using the
- * all parent [RuleSets] (selector expressions) and applying each nested
- * [RuleSet] to the list of [Selectors] in a [SelectorGroup].
- *
- * Then result is a style sheet where all nested rules have been flatten and
- * expanded.
- */
-class ExpandNestedSelectors extends Visitor {
- /** Parent [RuleSet] if a nested rule otherwise [:null:]. */
- RuleSet _parentRuleSet;
-
- /** Top-most rule if nested rules. */
- SelectorGroup _topLevelSelectorGroup;
-
- /** SelectorGroup at each nesting level. */
- SelectorGroup _nestedSelectorGroup;
-
- /** Declaration (sans the nested selectors). */
- DeclarationGroup _flatDeclarationGroup;
-
- /** Each nested selector get's a flatten RuleSet. */
- List<RuleSet> _expandedRuleSets = [];
-
- /** Maping of a nested rule set to the fully expanded list of RuleSet(s). */
- final Map<RuleSet, List<RuleSet>> _expansions = new Map();
-
- void visitRuleSet(RuleSet node) {
- final oldParent = _parentRuleSet;
-
- var oldNestedSelectorGroups = _nestedSelectorGroup;
-
- if (_nestedSelectorGroup == null) {
- // Create top-level selector (may have nested rules).
- final newSelectors = node.selectorGroup.selectors.toList();
- _topLevelSelectorGroup = new SelectorGroup(newSelectors, node.span);
- _nestedSelectorGroup = _topLevelSelectorGroup;
- } else {
- // Generate new selector groups from the nested rules.
- _nestedSelectorGroup = _mergeToFlatten(node);
- }
-
- _parentRuleSet = node;
-
- super.visitRuleSet(node);
-
- _parentRuleSet = oldParent;
-
- // Remove nested rules; they're all flatten and in the _expandedRuleSets.
- node.declarationGroup.declarations.removeWhere((declaration) =>
- declaration is RuleSet);
-
- _nestedSelectorGroup = oldNestedSelectorGroups;
-
- // If any expandedRuleSets and we're back at the top-level rule set then
- // there were nested rule set(s).
- if (_parentRuleSet == null) {
- if (!_expandedRuleSets.isEmpty) {
- // Remember ruleset to replace with these flattened rulesets.
- _expansions[node] = _expandedRuleSets;
- _expandedRuleSets = [];
- }
- assert(_flatDeclarationGroup == null);
- assert(_nestedSelectorGroup == null);
- }
- }
-
- /**
- * Build up the list of all inherited sequences from the parent selector
- * [node] is the current nested selector and it's parent is the last entry in
- * the [_nestedSelectorGroup].
- */
- SelectorGroup _mergeToFlatten(RuleSet node) {
- // Create a new SelectorGroup for this nesting level.
- var nestedSelectors = _nestedSelectorGroup.selectors;
- var selectors = node.selectorGroup.selectors;
-
- // Create a merged set of previous parent selectors and current selectors.
- var newSelectors = [];
- for (Selector selector in selectors) {
- for (Selector nestedSelector in nestedSelectors) {
- var seq = _mergeNestedSelector(nestedSelector.simpleSelectorSequences,
- selector.simpleSelectorSequences);
- newSelectors.add(new Selector(seq, node.span));
- }
- }
-
- return new SelectorGroup(newSelectors, node.span);
- }
-
- /**
- * Merge the nested selector sequences [current] to the [parent] sequences or
- * substitue any & with the parent selector.
- */
- List<SimpleSelectorSequence> _mergeNestedSelector(
- List<SimpleSelectorSequence> parent,
- List<SimpleSelectorSequence> current) {
-
- // If any & operator then the parent selector will be substituted otherwise
- // the parent selector is pre-pended to the current selector.
- var hasThis = current.any((s) => s.simpleSelector.isThis);
-
- var newSequence = [];
-
- if (!hasThis) {
- // If no & in the sector group then prefix with the parent selector.
- newSequence.addAll(parent);
- newSequence.addAll(_convertToDescendentSequence(current));
- } else {
- for (var sequence in current) {
- if (sequence.simpleSelector.isThis) {
- // Substitue the & with the parent selector and only use a combinator
- // descendant if & is prefix by a sequence with an empty name e.g.,
- // "... + &", "&", "... ~ &", etc.
- var hasPrefix = !newSequence.isEmpty &&
- !newSequence.last.simpleSelector.name.isEmpty;
- newSequence.addAll(
- hasPrefix ? _convertToDescendentSequence(parent) : parent);
- } else {
- newSequence.add(sequence);
- }
- }
- }
-
- return newSequence;
- }
-
- /**
- * Return selector sequences with first sequence combinator being a
- * descendant. Used for nested selectors when the parent selector needs to
- * be prefixed to a nested selector or to substitute the this (&) with the
- * parent selector.
- */
- List<SimpleSelectorSequence> _convertToDescendentSequence(
- List<SimpleSelectorSequence> sequences) {
- if (sequences.isEmpty) return sequences;
-
- var newSequences = [];
- var first = sequences.first;
- newSequences.add(new SimpleSelectorSequence(first.simpleSelector,
- first.span, TokenKind.COMBINATOR_DESCENDANT));
- newSequences.addAll(sequences.skip(1));
-
- return newSequences;
- }
-
- void visitDeclarationGroup(DeclarationGroup node) {
- var span = node.span;
-
- var currentGroup = new DeclarationGroup([], span);
-
- var oldGroup = _flatDeclarationGroup;
- _flatDeclarationGroup = currentGroup;
-
- var expandedLength = _expandedRuleSets.length;
-
- super.visitDeclarationGroup(node);
-
- // We're done with the group.
- _flatDeclarationGroup = oldGroup;
-
- // No nested rule to process it's a top-level rule.
- if (_nestedSelectorGroup == _topLevelSelectorGroup) return;
-
- // If flatten selector's declaration is empty skip this selector, no need
- // to emit an empty nested selector.
- if (currentGroup.declarations.isEmpty) return;
-
- var selectorGroup = _nestedSelectorGroup;
-
- // Build new rule set from the nested selectors and declarations.
- var newRuleSet = new RuleSet(selectorGroup, currentGroup, span);
-
- // Place in order so outer-most rule is first.
- if (expandedLength == _expandedRuleSets.length) {
- _expandedRuleSets.add(newRuleSet);
- } else {
- _expandedRuleSets.insert(expandedLength, newRuleSet);
- }
- }
-
- // Record all declarations in a nested selector (Declaration, VarDefinition
- // and MarginGroup) but not the nested rule in the Declaration.
-
- void visitDeclaration(Declaration node) {
- if (_parentRuleSet != null) {
- _flatDeclarationGroup.declarations.add(node);
- }
- super.visitDeclaration(node);
- }
-
- void visitVarDefinition(VarDefinition node) {
- if (_parentRuleSet != null) {
- _flatDeclarationGroup.declarations.add(node);
- }
- super.visitVarDefinition(node);
- }
-
- void visitExtendDeclaration(ExtendDeclaration node) {
- if (_parentRuleSet != null) {
- _flatDeclarationGroup.declarations.add(node);
- }
- super.visitExtendDeclaration(node);
- }
-
- void visitMarginGroup(MarginGroup node) {
- if (_parentRuleSet != null) {
- _flatDeclarationGroup.declarations.add(node);
- }
- super.visitMarginGroup(node);
- }
-
- /**
- * Replace the rule set that contains nested rules with the flatten rule sets.
- */
- void flatten(StyleSheet styleSheet) {
- // TODO(terry): Iterate over topLevels instead of _expansions it's already
- // a map (this maybe quadratic).
- _expansions.forEach((RuleSet ruleSet, List<RuleSet> newRules) {
- var index = styleSheet.topLevels.indexOf(ruleSet);
- if (index == -1) {
- // Check any @media directives for nested rules and replace them.
- var found = _MediaRulesReplacer.replace(styleSheet, ruleSet, newRules);
- assert(found);
- } else {
- styleSheet.topLevels.insertAll(index + 1, newRules);
- }
- });
- _expansions.clear();
- }
-}
-
-class _MediaRulesReplacer extends Visitor {
- RuleSet _ruleSet;
- List<RuleSet> _newRules;
- bool _foundAndReplaced = false;
-
- /**
- * Look for the [ruleSet] inside of an @media directive; if found then replace
- * with the [newRules]. If [ruleSet] is found and replaced return true.
- */
- static bool replace(StyleSheet styleSheet, RuleSet ruleSet,
- List<RuleSet>newRules) {
- var visitor = new _MediaRulesReplacer(ruleSet, newRules);
- visitor.visitStyleSheet(styleSheet);
- return visitor._foundAndReplaced;
- }
-
- _MediaRulesReplacer(this._ruleSet, this._newRules);
-
- visitMediaDirective(MediaDirective node) {
- var index = node.rulesets.indexOf(_ruleSet);
- if (index != -1) {
- node.rulesets.insertAll(index + 1, _newRules);
- _foundAndReplaced = true;
- }
- }
-}
-
-/**
- * Expand all @include at the top-level the ruleset(s) associated with the
- * mixin.
- */
-class TopLevelIncludes extends Visitor {
- StyleSheet _styleSheet;
- final Messages _messages;
- /** Map of variable name key to it's definition. */
- final Map<String, MixinDefinition> map = new Map<String, MixinDefinition>();
- MixinDefinition currDef;
-
- static void expand(Messages messages, List<StyleSheet> styleSheets) {
- new TopLevelIncludes(messages, styleSheets);
- }
-
- bool _anyRulesets(MixinRulesetDirective def) =>
- def.rulesets.any((rule) => rule is RuleSet);
-
- TopLevelIncludes(this._messages, List<StyleSheet> styleSheets) {
- for (var styleSheet in styleSheets) {
- visitTree(styleSheet);
- }
- }
-
- void visitStyleSheet(StyleSheet ss) {
- _styleSheet = ss;
- super.visitStyleSheet(ss);
- _styleSheet = null;
- }
-
- void visitIncludeDirective(IncludeDirective node) {
- if (map.containsKey(node.name)) {
- var mixinDef = map[node.name];
- if (mixinDef is MixinRulesetDirective) {
- _TopLevelIncludeReplacer.replace(_messages, _styleSheet, node,
- mixinDef.rulesets);
- } else if (currDef is MixinRulesetDirective && _anyRulesets(currDef)) {
- // currDef is MixinRulesetDirective
- MixinRulesetDirective mixinRuleset = currDef;
- int index = mixinRuleset.rulesets.indexOf(node as dynamic);
- mixinRuleset.rulesets.replaceRange(index, index + 1, [new NoOp()]);
- _messages.warning(
- 'Using declaration mixin ${node.name} as top-level mixin',
- node.span);
- }
- } else {
- if (currDef is MixinRulesetDirective) {
- MixinRulesetDirective rulesetDirect = currDef as MixinRulesetDirective;
- var index = 0;
- rulesetDirect.rulesets.forEach((entry) {
- if (entry == node) {
- rulesetDirect.rulesets.replaceRange(index, index + 1, [new NoOp()]);
- _messages.warning('Undefined mixin ${node.name}', node.span);
- }
- index++;
- });
- }
- }
- super.visitIncludeDirective(node);
- }
-
- void visitMixinRulesetDirective(MixinRulesetDirective node) {
- currDef = node;
-
- super.visitMixinRulesetDirective(node);
-
- // Replace with latest top-level mixin definition.
- map[node.name] = node;
- currDef = null;
- }
-
- void visitMixinDeclarationDirective(MixinDeclarationDirective node) {
- currDef = node;
-
- super.visitMixinDeclarationDirective(node);
-
- // Replace with latest mixin definition.
- map[node.name] = node;
- currDef = null;
- }
-}
-
-/** @include as a top-level with ruleset(s). */
-class _TopLevelIncludeReplacer extends Visitor {
- final Messages _messages;
- final IncludeDirective _include;
- final List<RuleSet> _newRules;
- bool _foundAndReplaced = false;
-
- /**
- * Look for the [ruleSet] inside of an @media directive; if found then replace
- * with the [newRules]. If [ruleSet] is found and replaced return true.
- */
- static bool replace(Messages messages, StyleSheet styleSheet,
- IncludeDirective include, List<RuleSet>newRules) {
- var visitor = new _TopLevelIncludeReplacer(messages, include, newRules);
- visitor.visitStyleSheet(styleSheet);
- return visitor._foundAndReplaced;
- }
-
- _TopLevelIncludeReplacer(this._messages, this._include, this._newRules);
-
- visitStyleSheet(StyleSheet node) {
- var index = node.topLevels.indexOf(_include);
- if (index != -1) {
- node.topLevels.insertAll(index + 1, _newRules);
- node.topLevels.replaceRange(index, index + 1, [new NoOp()]);
- _foundAndReplaced = true;
- }
- super.visitStyleSheet(node);
- }
-
- void visitMixinRulesetDirective(MixinRulesetDirective node) {
- var index = node.rulesets.indexOf(_include as dynamic);
- if (index != -1) {
- node.rulesets.insertAll(index + 1, _newRules);
- // Only the resolve the @include once.
- node.rulesets.replaceRange(index, index + 1, [new NoOp()]);
- _foundAndReplaced = true;
- }
- super.visitMixinRulesetDirective(node);
- }
-}
-
-/**
- * Utility function to match an include to a list of either Declarations or
- * RuleSets, depending on type of mixin (ruleset or declaration). The include
- * can be an include in a declaration or an include directive (top-level).
- */
-int _findInclude(List list, var node) {
- IncludeDirective matchNode = (node is IncludeMixinAtDeclaration) ?
- node.include : node;
-
- var index = 0;
- for (var item in list) {
- var includeNode = (item is IncludeMixinAtDeclaration) ?
- item.include : item;
- if (includeNode == matchNode) return index;
- index++;
- }
- return -1;
-}
-
-/**
- * Stamp out a mixin with the defined args substituted with the user's
- * parameters.
- */
-class CallMixin extends Visitor {
- final MixinDefinition mixinDef;
- List _definedArgs;
- Expressions _currExpressions;
- int _currIndex = -1;
-
- final varUsages = new Map<String, Map<Expressions, Set<int>>>();
-
- /** Only var defs with more than one expression (comma separated). */
- final Map<String, VarDefinition> varDefs;
-
- CallMixin(this.mixinDef, [this.varDefs]) {
- if (mixinDef is MixinRulesetDirective) {
- visitMixinRulesetDirective(mixinDef);
- } else {
- visitMixinDeclarationDirective(mixinDef);
- }
- }
-
- /**
- * Given a mixin's defined arguments return a cloned mixin defintion that has
- * replaced all defined arguments with user's supplied VarUsages.
- */
- MixinDefinition transform(List<TreeNode> callArgs) {
- // TODO(terry): Handle default arguments and varArgs.
- // Transform mixin with callArgs.
- var index = 0;
- for (var index = 0; index < _definedArgs.length; index++) {
- var definedArg = _definedArgs[index];
- VarDefinition varDef;
- if (definedArg is VarDefinition) {
- varDef = definedArg;
- } else if (definedArg is VarDefinitionDirective) {
- VarDefinitionDirective varDirective = definedArg;
- varDef = varDirective.def;
- }
- var callArg = callArgs[index];
-
- // Is callArg a var definition with multi-args (expressions > 1).
- var defArgs = _varDefsAsCallArgs(callArg);
- if (defArgs.isNotEmpty) {
- // Replace call args with the var def parameters.
- callArgs.insertAll(index, defArgs);
- callArgs.removeAt(index + defArgs.length);
- callArg = callArgs[index];
- }
-
- var expressions = varUsages[varDef.definedName];
- expressions.forEach((k, v) {
- for (var usagesIndex in v) {
- k.expressions.replaceRange(usagesIndex, usagesIndex + 1, callArg);
- }
- });
- }
-
- // Clone the mixin
- return mixinDef.clone();
- }
-
- /** Rip apart var def with multiple parameters. */
- List<List<TreeNode>> _varDefsAsCallArgs(var callArg) {
- var defArgs = [];
- if (callArg is List && callArg[0] is VarUsage) {
- var varDef = varDefs[callArg[0].name];
- var expressions = varDef.expression.expressions;
- assert(expressions.length > 1);
- for (var expr in expressions) {
- if (expr is! OperatorComma) {
- defArgs.add([expr]);
- }
- }
- }
- return defArgs;
- }
-
- void visitExpressions(Expressions node) {
- var oldExpressions = _currExpressions;
- var oldIndex = _currIndex;
-
- _currExpressions = node;
- for (_currIndex = 0; _currIndex < node.expressions.length; _currIndex++) {
- node.expressions[_currIndex].visit(this);
- }
-
- _currIndex = oldIndex;
- _currExpressions = oldExpressions;
- }
-
- void _addExpression(Map<Expressions, Set<int>> expressions) {
- var indexSet = new Set<int>();
- indexSet.add(_currIndex);
- expressions[_currExpressions] = indexSet;
- }
-
- void visitVarUsage(VarUsage node) {
- assert(_currIndex != -1);
- assert(_currExpressions != null);
- if (varUsages.containsKey(node.name)) {
- Map<Expressions, Set<int>> expressions = varUsages[node.name];
- Set<int> allIndexes = expressions[_currExpressions];
- if (allIndexes == null) {
- _addExpression(expressions);
- } else {
- allIndexes.add(_currIndex);
- }
- } else {
- var newExpressions = new Map<Expressions, Set<int>>();
- _addExpression(newExpressions);
- varUsages[node.name] = newExpressions;
- }
- super.visitVarUsage(node);
- }
-
- void visitMixinDeclarationDirective(MixinDeclarationDirective node) {
- _definedArgs = node.definedArgs;
- super.visitMixinDeclarationDirective(node);
- }
-
- void visitMixinRulesetDirective(MixinRulesetDirective node) {
- _definedArgs = node.definedArgs;
- super.visitMixinRulesetDirective(node);
- }
-}
-
-/** Expand all @include inside of a declaration associated with a mixin. */
-class DeclarationIncludes extends Visitor {
- StyleSheet _styleSheet;
- final Messages _messages;
- /** Map of variable name key to it's definition. */
- final Map<String, MixinDefinition> map = new Map<String, MixinDefinition>();
- /** Cache of mixin called with parameters. */
- final Map<String, CallMixin> callMap = new Map<String, CallMixin>();
- MixinDefinition currDef;
- DeclarationGroup currDeclGroup;
-
- /** Var definitions with more than 1 expression. */
- final Map<String, VarDefinition> varDefs = new Map<String, VarDefinition>();
-
- static void expand(Messages messages, List<StyleSheet> styleSheets) {
- new DeclarationIncludes(messages, styleSheets);
- }
-
- DeclarationIncludes(this._messages, List<StyleSheet> styleSheets) {
- for (var styleSheet in styleSheets) {
- visitTree(styleSheet);
- }
- }
-
- bool _allIncludes(rulesets) =>
- rulesets.every((rule) => rule is IncludeDirective || rule is NoOp);
-
- CallMixin _createCallDeclMixin(MixinDefinition mixinDef) {
- callMap.putIfAbsent(mixinDef.name, () =>
- callMap[mixinDef.name] = new CallMixin(mixinDef, varDefs));
- return callMap[mixinDef.name];
- }
-
- void visitStyleSheet(StyleSheet ss) {
- _styleSheet = ss;
- super.visitStyleSheet(ss);
- _styleSheet = null;
- }
-
- void visitDeclarationGroup(DeclarationGroup node) {
- currDeclGroup = node;
- super.visitDeclarationGroup(node);
- currDeclGroup = null;
- }
-
- void visitIncludeMixinAtDeclaration(IncludeMixinAtDeclaration node) {
- if (map.containsKey(node.include.name)) {
- var mixinDef = map[node.include.name];
-
- // Fix up any mixin that is really a Declaration but has includes.
- if (mixinDef is MixinRulesetDirective) {
- if (!_allIncludes(mixinDef.rulesets) && currDeclGroup != null) {
- var index = _findInclude(currDeclGroup.declarations, node);
- if (index != -1) {
- currDeclGroup.declarations.replaceRange(index, index + 1,
- [new NoOp()]);
- }
- _messages.warning(
- "Using top-level mixin ${node.include.name} as a declaration",
- node.span);
- } else {
- // We're a list of @include(s) inside of a mixin ruleset - convert
- // to a list of IncludeMixinAtDeclaration(s).
- var origRulesets = mixinDef.rulesets;
- var rulesets = [];
- if (origRulesets.every((ruleset) => ruleset is IncludeDirective)) {
- origRulesets.forEach((ruleset) {
- rulesets.add(new IncludeMixinAtDeclaration(ruleset,
- ruleset.span));
- });
- _IncludeReplacer.replace(_styleSheet, node, rulesets);
- }
- }
- }
-
- if ( mixinDef.definedArgs.length > 0 && node.include.args.length > 0) {
- var callMixin = _createCallDeclMixin(mixinDef);
- mixinDef = callMixin.transform(node.include.args);
- }
-
- if (mixinDef is MixinDeclarationDirective) {
- _IncludeReplacer.replace(_styleSheet, node,
- mixinDef.declarations.declarations);
- }
- } else {
- _messages.warning("Undefined mixin ${node.include.name}", node.span);
- }
-
- super.visitIncludeMixinAtDeclaration(node);
- }
-
- void visitIncludeDirective(IncludeDirective node) {
- if (map.containsKey(node.name)) {
- var mixinDef = map[node.name];
- if (currDef is MixinDeclarationDirective &&
- mixinDef is MixinDeclarationDirective) {
- _IncludeReplacer.replace(_styleSheet, node,
- mixinDef.declarations.declarations);
- } else if (currDef is MixinDeclarationDirective) {
- var decls = (currDef as MixinDeclarationDirective)
- .declarations.declarations;
- var index = _findInclude(decls, node);
- if (index != -1) {
- decls.replaceRange(index, index + 1, [new NoOp()]);
- }
- }
- }
-
- super.visitIncludeDirective(node);
- }
-
- void visitMixinRulesetDirective(MixinRulesetDirective node) {
- currDef = node;
-
- super.visitMixinRulesetDirective(node);
-
- // Replace with latest top-level mixin definition.
- map[node.name] = node;
- currDef = null;
- }
-
- void visitMixinDeclarationDirective(MixinDeclarationDirective node) {
- currDef = node;
-
- super.visitMixinDeclarationDirective(node);
-
- // Replace with latest mixin definition.
- map[node.name] = node;
- currDef = null;
- }
-
- void visitVarDefinition(VarDefinition node) {
- // Only record var definitions that have multiple expressions (comma
- // separated for mixin parameter substitution.
- var exprs = (node.expression as Expressions).expressions;
- if (exprs.length > 1) {
- varDefs[node.definedName] = node;
- }
- super.visitVarDefinition(node);
- }
-
- void visitVarDefinitionDirective(VarDefinitionDirective node) {
- visitVarDefinition(node.def);
- }
-}
-
-/** @include as a top-level with ruleset(s). */
-class _IncludeReplacer extends Visitor {
- final _include;
- final List<Declaration> _newDeclarations;
- bool _foundAndReplaced = false;
-
- /**
- * Look for the [ruleSet] inside of a @media directive; if found then replace
- * with the [newRules].
- */
- static void replace(StyleSheet ss, var include,
- List<Declaration> newDeclarations) {
- var visitor = new _IncludeReplacer(include, newDeclarations);
- visitor.visitStyleSheet(ss);
- }
-
- _IncludeReplacer(this._include, this._newDeclarations);
-
- void visitDeclarationGroup(DeclarationGroup node) {
- var index = _findInclude(node.declarations, _include);
- if (index != -1) {
- node.declarations.insertAll(index + 1, _newDeclarations);
- // Change @include to NoOp so it's processed only once.
- node.declarations.replaceRange(index, index + 1, [new NoOp()]);
- _foundAndReplaced = true;
- }
- super.visitDeclarationGroup(node);
- }
-}
-
-/**
- * Remove all @mixin and @include and any NoOp used as placeholder for @include.
- */
-class MixinsAndIncludes extends Visitor {
- static void remove(StyleSheet styleSheet) {
- new MixinsAndIncludes()..visitStyleSheet(styleSheet);
- }
-
- bool _nodesToRemove(node) =>
- node is IncludeDirective || node is MixinDefinition || node is NoOp;
-
- void visitStyleSheet(StyleSheet ss) {
- var index = ss.topLevels.length;
- while (--index >= 0) {
- if (_nodesToRemove(ss.topLevels[index])) {
- ss.topLevels.removeAt(index);
- }
- }
- super.visitStyleSheet(ss);
- }
-
- void visitDeclarationGroup(DeclarationGroup node) {
- var index = node.declarations.length;
- while (--index >= 0) {
- if (_nodesToRemove(node.declarations[index])) {
- node.declarations.removeAt(index);
- }
- }
- super.visitDeclarationGroup(node);
- }
-}
-
-/** Find all @extend to create inheritance. */
-class AllExtends extends Visitor {
- final Map<String, List<SelectorGroup>> inherits =
- new Map<String, List<SelectorGroup>>();
-
- SelectorGroup _currSelectorGroup;
- List _currDecls;
- int _currDeclIndex;
- List<int> _extendsToRemove = [];
-
- void visitRuleSet(RuleSet node) {
- var oldSelectorGroup = _currSelectorGroup;
- _currSelectorGroup = node.selectorGroup;
-
- super.visitRuleSet(node);
-
- _currSelectorGroup = oldSelectorGroup;
- }
-
- void visitExtendDeclaration(ExtendDeclaration node) {
- var inheritName = "";
- for (var selector in node.selectors) {
- inheritName += selector.toString();
- }
- if (inherits.containsKey(inheritName)) {
- inherits[inheritName].add(_currSelectorGroup);
- } else {
- inherits[inheritName] = [_currSelectorGroup];
- }
-
- // Remove this @extend
- _extendsToRemove.add(_currDeclIndex);
-
- super.visitExtendDeclaration(node);
- }
-
- void visitDeclarationGroup(DeclarationGroup node) {
- var oldDeclIndex = _currDeclIndex;
-
- var decls = node.declarations;
- for (_currDeclIndex = 0; _currDeclIndex < decls.length; _currDeclIndex++) {
- decls[_currDeclIndex].visit(this);
- }
-
- if (_extendsToRemove.isNotEmpty) {
- var removeTotal = _extendsToRemove.length - 1;
- for (var index = removeTotal; index >= 0; index--) {
- decls.removeAt(_extendsToRemove[index]);
- }
- _extendsToRemove.clear();
- }
-
- _currDeclIndex = oldDeclIndex;
- }
-}
-
-// TODO(terry): Need to handle merging selector sequences
-// TODO(terry): Need to handle @extend-Only selectors.
-// TODO(terry): Need to handle !optional glag.
-/**
- * Changes any selector that matches @extend.
- */
-class InheritExtends extends Visitor {
- final Messages _messages;
- final AllExtends _allExtends;
-
- InheritExtends(this._messages, this._allExtends);
-
- void visitSelectorGroup(SelectorGroup node) {
- for (var selectorsIndex = 0; selectorsIndex < node.selectors.length;
- selectorsIndex++) {
- var selectors = node.selectors[selectorsIndex];
- var isLastNone = false;
- var selectorName = "";
- for (var index = 0; index < selectors.simpleSelectorSequences.length;
- index++) {
- var simpleSeq = selectors.simpleSelectorSequences[index];
- var namePart = simpleSeq.simpleSelector.toString();
- selectorName = (isLastNone) ? (selectorName + namePart) : namePart;
- List<SelectorGroup> matches = _allExtends.inherits[selectorName];
- if (matches != null) {
- for (var match in matches) {
- // Create a new group.
- var newSelectors = selectors.clone();
- var newSeq = match.selectors[0].clone();
- if (isLastNone) {
- // Add the inherited selector.
- node.selectors.add(newSeq);
- } else {
- // Replace the selector sequence to the left of the pseudo class
- // or pseudo element.
-
- // Make new selector seq combinator the same as the original.
- var orgCombinator =
- newSelectors.simpleSelectorSequences[index].combinator;
- newSeq.simpleSelectorSequences[0].combinator = orgCombinator;
-
- newSelectors.simpleSelectorSequences.replaceRange(index,
- index + 1, newSeq.simpleSelectorSequences);
- node.selectors.add(newSelectors);
- }
- isLastNone = false;
- }
- } else {
- isLastNone = simpleSeq.isCombinatorNone;
- }
- }
- }
- super.visitSelectorGroup(node);
- }
-}
« no previous file with comments | « pkg/csslib/lib/parser.dart ('k') | pkg/csslib/lib/src/css_printer.dart » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698