com.google.common.css.compiler.passes.CollectProvideNamespaces Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of closure-stylesheets Show documentation
Show all versions of closure-stylesheets Show documentation
Closure Stylesheets is an extension to CSS that adds variables,
functions,
conditionals, and mixins to standard CSS. The tool also supports
minification, linting, RTL flipping, and CSS class renaming.
The newest version!
/*
* Copyright 2013 Google Inc.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.google.common.css.compiler.passes;
import com.google.common.base.Preconditions;
import com.google.common.collect.ImmutableListMultimap;
import com.google.common.collect.ImmutableMap;
import com.google.common.collect.LinkedListMultimap;
import com.google.common.collect.ListMultimap;
import com.google.common.collect.Maps;
import com.google.common.css.compiler.ast.CssCompilerPass;
import com.google.common.css.compiler.ast.CssDefinitionNode;
import com.google.common.css.compiler.ast.CssMixinDefinitionNode;
import com.google.common.css.compiler.ast.CssProvideNode;
import com.google.common.css.compiler.ast.CssRequireNode;
import com.google.common.css.compiler.ast.DefaultTreeVisitor;
import com.google.common.css.compiler.ast.ErrorManager;
import com.google.common.css.compiler.ast.VisitController;
import java.util.Map;
import java.util.logging.Logger;
import java.util.regex.Pattern;
/**
* A compiler pass to help find missing {@code @require} lines for def constant references
* and mixins.
* This pass simply collects namespaces that correpond to constant definitions and mixins.
* Also see the CheckMissingRequire pass that is used in conjunction with this one.
*
*/
public final class CollectProvideNamespaces extends DefaultTreeVisitor implements CssCompilerPass {
private static final Logger logger = Logger.getLogger(CollectProvideNamespaces.class.getName());
private static final Pattern OVERRIDE_REGEX = Pattern.compile(
"/\\*\\s+@overrideSelector\\s+\\{(.*)\\}\\s+\\*/");
private final VisitController visitController;
private final ErrorManager errorManager;
// Key: filename; Value: provide namespace
private final Map filenameProvideMap = Maps.newHashMap();
// Key: filename; Value: require namespace
private final ListMultimap filenameRequireMap = LinkedListMultimap.create();
// Multiple namespaces can contain the same defs due to duplicate defs (or mods).
// Key: def name; Value: provide namespace
private final ListMultimap defProvideMap = LinkedListMultimap.create();
// Key: defmixin name; Value: provide namespace
private final ListMultimap defmixinProvideMap = LinkedListMultimap.create();
public Map getFilenameProvideMap() {
return ImmutableMap.copyOf(filenameProvideMap);
}
public ListMultimap getFilenameRequireMap() {
return ImmutableListMultimap.copyOf(filenameRequireMap);
}
public ListMultimap getDefProvideMap() {
return ImmutableListMultimap.copyOf(defProvideMap);
}
public ListMultimap getDefmixinProvideMap() {
return ImmutableListMultimap.copyOf(defmixinProvideMap);
}
public CollectProvideNamespaces(VisitController visitController,
ErrorManager errorManager) {
this.visitController = visitController;
this.errorManager = errorManager;
}
@Override
public boolean enterProvideNode(CssProvideNode node) {
Preconditions.checkState(node.getSourceCodeLocation() != null);
String filename = node.getSourceCodeLocation().getSourceCode().getFileName();
filenameProvideMap.put(filename, node.getProvide());
return true;
}
@Override
public boolean enterRequireNode(CssRequireNode node) {
Preconditions.checkState(node.getSourceCodeLocation() != null);
String filename = node.getSourceCodeLocation().getSourceCode().getFileName();
filenameRequireMap.put(filename, node.getRequire());
return true;
}
@Override
public boolean enterDefinition(CssDefinitionNode node) {
if (node.getSourceCodeLocation() == null) {
// Cannot enforce provide / require for generated GSS components.
return true;
}
String filename = node.getSourceCodeLocation().getSourceCode().getFileName();
String provideNamespace = filenameProvideMap.get(filename);
// Remove this after switching to the new syntax.
if (provideNamespace == null) { // ignore old format @provide
return true;
}
defProvideMap.put(node.getName().getValue(), provideNamespace);
return true;
}
@Override
public boolean enterMixinDefinition(CssMixinDefinitionNode node) {
if (node.getSourceCodeLocation() == null) {
// Cannot enforce provide / require for generated GSS components.
return true;
}
String filename = node.getSourceCodeLocation().getSourceCode().getFileName();
String provideNamespace = filenameProvideMap.get(filename);
// Remove this after switching to the new syntax.
if (provideNamespace == null) { // ignore old format @provide
return true;
}
Preconditions.checkArgument(provideNamespace != null);
defmixinProvideMap.put(node.getDefinitionName(), provideNamespace);
return true;
}
@Override
public void runPass() {
visitController.startVisit(this);
}
}