/* * Copyright 2014 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.gwt.resources.converter; import com.google.gwt.core.ext.TreeLogger; import com.google.gwt.core.ext.TreeLogger.Type; import com.google.gwt.resources.css.ast.Context; import com.google.gwt.resources.css.ast.CssProperty; import com.google.gwt.resources.css.ast.CssProperty.FunctionValue; import com.google.gwt.resources.css.ast.CssProperty.IdentValue; import com.google.gwt.resources.css.ast.CssProperty.ListValue; import com.google.gwt.resources.css.ast.CssProperty.Value; import com.google.gwt.resources.css.ast.CssRule; import com.google.gwt.resources.css.ast.CssVisitor; import com.google.gwt.thirdparty.guava.common.base.Preconditions; import com.google.gwt.thirdparty.guava.common.collect.Sets; import java.util.ArrayList; import java.util.List; import java.util.Locale; import java.util.Set; import java.util.regex.Matcher; import java.util.regex.Pattern; /** * Converts upper case strings used in properties to lowercase if they are not defined * with a proper @def statement. */ public class UndefinedConstantVisitor extends CssVisitor { private final Set<String> gssContantNames; private final Pattern pattern = Pattern.compile("^[A-Z_][A-Z_0-9]+$"); private final Set<String> propertyNamesToSkip = Sets.newHashSet("filter", "-ms-filter", "font-family"); private final boolean lenient; private final TreeLogger treeLogger; public UndefinedConstantVisitor(Set<String> gssContantNames, boolean lenient, TreeLogger treeLogger) { this.gssContantNames = gssContantNames; this.lenient = lenient; this.treeLogger = treeLogger; } @Override public boolean visit(CssRule x, Context ctx) { List<CssProperty> properties = x.getProperties(); for (CssProperty cssProperty : properties) { String cssPropertyName = cssProperty.getName(); if (propertyNamesToSkip.contains(cssPropertyName)) { continue; } // for logging purpose String selector = x.getSelectors().toString(); ListValue listValue = visitListValue(cssProperty.getValues(), cssPropertyName, selector); cssProperty.setValue(listValue); } return false; } private ListValue visitListValue(ListValue values, String cssPropertyName, String selector) { Preconditions.checkNotNull(values, "values cannot be null"); List<Value> cssPropertyValues = values.getValues(); List<Value> newValues = new ArrayList<Value>(cssPropertyValues.size()); for (Value value : cssPropertyValues) { if (value.isListValue() != null) { newValues.add(visitListValue(value.isListValue(), cssPropertyName, selector)); } else if (value.isFunctionValue() != null) { FunctionValue functionValue = value.isFunctionValue(); ListValue listValue = visitListValue(functionValue.getValues(), cssPropertyName, selector); newValues.add(new FunctionValue(functionValue.getName(), listValue)); } else if (value.isIdentValue() != null) { newValues.add(visitIdentValue(value.isIdentValue(), cssPropertyName, selector)); } else { newValues.add(value); } } return new ListValue(newValues); } private Value visitIdentValue(IdentValue identValue, String cssPropertyName, String selector) { Matcher matcher = pattern.matcher(identValue.getIdent()); if (matcher.matches()) { String upperCaseString = matcher.group(); if (!gssContantNames.contains(upperCaseString)) { treeLogger.log(Type.WARN, "Property '" + cssPropertyName + "' from rule '" + selector + "' uses an undefined constant: " + upperCaseString); if (lenient) { treeLogger.log(Type.WARN, "turning '" + upperCaseString + "' to lower case. This is probably not what you wanted here in the " + "first place!"); return new IdentValue(upperCaseString.toLowerCase(Locale.US)); } else { throw new Css2GssConversionException("Found undefined constant in input. " + cssPropertyName + "' from rule '" + selector + "' undefined constant: " + upperCaseString); } } } return identValue; } }