com.google.gwt.dev.js.JsSourceGenerationVisitorWithSizeBreakdown Maven / Gradle / Ivy
/*
* Copyright 2009 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.dev.js;
import com.google.gwt.core.ext.soyc.Range;
import com.google.gwt.dev.jjs.SourceInfo;
import com.google.gwt.dev.jjs.ast.JClassType;
import com.google.gwt.dev.jjs.ast.JDeclaredType;
import com.google.gwt.dev.jjs.ast.JField;
import com.google.gwt.dev.jjs.ast.JMethod;
import com.google.gwt.dev.jjs.impl.codesplitter.FragmentExtractor;
import com.google.gwt.dev.jjs.impl.JavaToJavaScriptMap;
import com.google.gwt.dev.js.ast.JsBlock;
import com.google.gwt.dev.js.ast.JsContext;
import com.google.gwt.dev.js.ast.JsName;
import com.google.gwt.dev.js.ast.JsProgram;
import com.google.gwt.dev.js.ast.JsProgramFragment;
import com.google.gwt.dev.js.ast.JsStatement;
import com.google.gwt.dev.js.ast.JsVars.JsVar;
import com.google.gwt.dev.js.ast.JsVisitable;
import com.google.gwt.dev.util.TextOutput;
import com.google.gwt.dev.util.collect.HashMap;
import java.util.List;
import java.util.Map;
/**
* A version of {@link JsSourceGenerationVisitor} that records a
* {@link SizeBreakdown} as it goes.
*/
public class JsSourceGenerationVisitorWithSizeBreakdown extends
JsSourceGenerationVisitor {
private JavaToJavaScriptMap map;
private JsName billedAncestor; // non-null when an ancestor is also being billed
private TextOutput out;
private final Map sizeMap = new HashMap();
public JsSourceGenerationVisitorWithSizeBreakdown(TextOutput out,
JavaToJavaScriptMap javaToJavaScriptMap) {
super(out);
this.out = out;
this.map = javaToJavaScriptMap;
}
public SizeBreakdown getSizeBreakdown() {
return new SizeBreakdown(out.getPosition(), sizeMap);
}
public Map getSourceInfoMap() {
// override if your child class creates sourceinfo
return null;
}
@Override
public boolean visit(JsBlock x, JsContext ctx) {
printJsBlock(x, false, true);
return false;
}
@Override
public boolean visit(JsProgram x, JsContext ctx) {
// Descend naturally.
return true;
}
@Override
public boolean visit(JsProgramFragment x, JsContext ctx) {
// Descend naturally.
return true;
}
@Override
protected final T doAccept(T node) {
JsName newName = nameToBillTo(node, billedAncestor != null);
return generateAndBill(node, newName);
}
@Override
protected void doAcceptList(List collection) {
for (T t : collection) {
doAccept(t);
}
}
@Override
protected void doAcceptWithInsertRemove(
List collection) {
for (T t : collection) {
doAccept(t);
}
}
/**
* Generate some JavaScript and bill the number of characters generated to the given name.
*/
protected T generateAndBill(T node, JsName nameToBillTo) {
if (nameToBillTo == null) {
return super.doAccept(node);
} else {
int start = out.getPosition();
JsName savedAncestor = billedAncestor;
billedAncestor = nameToBillTo;
T retValue = super.doAccept(node);
billedAncestor = savedAncestor;
billChars(nameToBillTo, out.getPosition() - start);
return retValue;
}
}
protected JDeclaredType getDirectlyEnclosingType(JsName nameToBillTo) {
if (nameToBillTo == null) {
return null;
}
JDeclaredType type = map.nameToType(nameToBillTo);
if (type != null) {
return type;
}
JMethod method = map.nameToMethod(nameToBillTo);
if (method != null) {
return method.getEnclosingType();
}
JField field = map.nameToField(nameToBillTo);
if (field != null) {
return field.getEnclosingType();
}
return null;
}
private void billChars(JsName nameToBillTo, int chars) {
Integer oldSize = sizeMap.get(nameToBillTo);
if (oldSize == null) {
oldSize = 0;
}
sizeMap.put(nameToBillTo, oldSize + chars);
}
/**
* Returns the type, function, or variable name where this node's character count
* should be added, or null to bill to nobody.
*/
private JsName nameToBillTo(JsVisitable node, boolean isAncestorBilled) {
if (node instanceof JsStatement) {
JsStatement stat = (JsStatement) node;
JClassType type = map.typeForStatement(stat);
if (type != null) {
return map.nameForType(type);
}
JMethod method = FragmentExtractor.methodFor(stat, map);
if (method != null) {
return map.nameForMethod(method);
}
return null;
} else if (node instanceof JsVar) {
return isAncestorBilled ? null : ((JsVar) node).getName(); // handle top-level vars
}
return null;
}
}