blob: 70be2849ac479895ef3f126d7a7f0f6d2c4f5058 [file] [log] [blame]
/*
*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You 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 org.apache.royale.compiler.internal.codegen.js.jx;
import org.apache.royale.compiler.codegen.ISubEmitter;
import org.apache.royale.compiler.codegen.js.IJSEmitter;
import org.apache.royale.compiler.constants.IASKeywordConstants;
import org.apache.royale.compiler.definitions.IDefinition;
import org.apache.royale.compiler.internal.codegen.as.ASEmitterTokens;
import org.apache.royale.compiler.internal.codegen.js.JSSubEmitter;
import org.apache.royale.compiler.internal.codegen.js.royale.JSRoyaleEmitter;
import org.apache.royale.compiler.internal.codegen.js.utils.EmitterUtils;
import org.apache.royale.compiler.internal.projects.RoyaleJSProject;
import org.apache.royale.compiler.internal.tree.as.ChainedVariableNode;
import org.apache.royale.compiler.projects.ICompilerProject;
import org.apache.royale.compiler.tree.ASTNodeID;
import org.apache.royale.compiler.tree.as.IASNode;
import org.apache.royale.compiler.tree.as.IEmbedNode;
import org.apache.royale.compiler.tree.as.IExpressionNode;
import org.apache.royale.compiler.tree.as.IFunctionNode;
import org.apache.royale.compiler.tree.as.IVariableNode;
/**
* Local variable in a function. For member and static variables of a class, see
* FieldEmitter instead.
*/
public class VarDeclarationEmitter extends JSSubEmitter implements
ISubEmitter<IVariableNode>
{
public VarDeclarationEmitter(IJSEmitter emitter)
{
super(emitter);
}
@Override
public void emit(IVariableNode node)
{
// TODO (mschmalle) will remove this cast as more things get abstracted
JSRoyaleEmitter fjs = (JSRoyaleEmitter) getEmitter();
getModel().getVars().add(node);
boolean defaultInitializers = false;
ICompilerProject project = getProject();
if(project instanceof RoyaleJSProject)
{
RoyaleJSProject fjsProject = (RoyaleJSProject) project;
if(fjsProject.config != null)
{
defaultInitializers = fjsProject.config.getJsDefaultInitializers();
}
}
boolean needsDefaultValue = EmitterUtils.needsDefaultValue(node, defaultInitializers, getProject());
IFunctionNode parentFnNode = (IFunctionNode) node.getAncestorOfType(IFunctionNode.class);
boolean isHoisting = fjs.isEmittingHoistedNodes(parentFnNode);
if (!(node instanceof ChainedVariableNode) && !isHoisting && needsDefaultValue)
{
write("//");
}
if (!(node instanceof ChainedVariableNode) && !node.isConst())
{
fjs.emitMemberKeyword(node);
}
IExpressionNode variableTypeNode = node.getVariableTypeNode();
IDefinition variableDef = null;
boolean hasVariableType = variableTypeNode.getLine() >= 0;
if(hasVariableType)
{
variableDef = variableTypeNode.resolve(getProject());
startMapping(variableTypeNode,
variableTypeNode.getLine(),
variableTypeNode.getColumn() - 1); //include the :
}
else
{
//the result of getVariableTypeNode() may not have a line and
//column. this can happen when the type is omitted in the code, and
//the compiler generates a node for type *.
//in this case, put it at the end of the name expression.
IExpressionNode nameExpressionNode = node.getNameExpressionNode();
startMapping(variableTypeNode, nameExpressionNode.getLine(),
nameExpressionNode.getColumn() + nameExpressionNode.getAbsoluteEnd() - nameExpressionNode.getAbsoluteStart());
}
IExpressionNode avnode = node.getAssignedValueNode();
IDefinition avtypedef = null;
if (avnode != null)
{
avtypedef = avnode.resolveType(getProject());
String opcode = avnode.getNodeID().getParaphrase();
if (opcode != "AnonymousFunction")
{
fjs.getDocEmitter().emitVarDoc(node, avtypedef, getProject());
}
}
else
{
fjs.getDocEmitter().emitVarDoc(node, null, getProject());
}
endMapping(variableTypeNode);
if (!(node instanceof ChainedVariableNode) && node.isConst())
{
fjs.emitMemberKeyword(node);
}
fjs.emitDeclarationName(node);
if (avnode == null)
{
emitDefaultInitializer(node, defaultInitializers);
}
else if(!(avnode instanceof IEmbedNode))
{
if (hasVariableType)
{
startMapping(node, node.getVariableTypeNode());
}
else
{
startMapping(node, node.getNameExpressionNode());
}
write(ASEmitterTokens.SPACE);
writeToken(ASEmitterTokens.EQUAL);
endMapping(node);
getEmitter().emitAssignmentCoercion(avnode, variableDef);
}
emitChainedVariables(node, defaultInitializers, isHoisting);
}
private void emitDefaultInitializer(IVariableNode node, boolean defaultInitializers)
{
IDefinition typedef = null;
IExpressionNode enode = node.getVariableTypeNode();
if (enode != null)
{
typedef = enode.resolveType(getProject());
}
if (typedef != null)
{
if (node.getParent() != null &&
node.getParent().getParent() != null &&
node.getParent().getParent().getNodeID() != ASTNodeID.Op_InID)
{
String defName = typedef.getQualifiedName();
if (defName.equals("int") || defName.equals("uint"))
{
write(ASEmitterTokens.SPACE);
writeToken(ASEmitterTokens.EQUAL);
write("0");
}
else if (defaultInitializers)
{
if (defName.equals("Number"))
{
write(ASEmitterTokens.SPACE);
writeToken(ASEmitterTokens.EQUAL);
write(IASKeywordConstants.NA_N);
}
else if (defName.equals("Boolean"))
{
write(ASEmitterTokens.SPACE);
writeToken(ASEmitterTokens.EQUAL);
write(IASKeywordConstants.FALSE);
}
else if (!defName.equals("*"))
{
//type * is meant to default to undefined, so it
//doesn't need to be initialized, but everything
//else should default to null
write(ASEmitterTokens.SPACE);
writeToken(ASEmitterTokens.EQUAL);
write(IASKeywordConstants.NULL);
}
}
}
}
}
private void emitChainedVariables(IVariableNode node, boolean defaultInitializers, boolean isHoisting)
{
JSRoyaleEmitter fjs = (JSRoyaleEmitter) getEmitter();
if (!(node instanceof ChainedVariableNode))
{
// check for chained variables
int len = node.getChildCount();
boolean splitVariables = false;
for (int i = 0; i < len; i++)
{
IASNode child = node.getChild(i);
if (child instanceof ChainedVariableNode)
{
ChainedVariableNode varChild = (ChainedVariableNode) child;
//if any of them need a default value, they all should be
//split onto different lines
if(EmitterUtils.needsDefaultValue(varChild, defaultInitializers, getProject()))
{
splitVariables = true;
break;
}
}
}
for (int i = 0; i < len; i++)
{
IASNode child = node.getChild(i);
if (child instanceof ChainedVariableNode)
{
if (splitVariables)
{
boolean childNeedsDefault = EmitterUtils.needsDefaultValue((IVariableNode) child, defaultInitializers, getProject());
if (isHoisting && !childNeedsDefault)
{
//this one does not need to be hoisted because it
//already has a default value
continue;
}
startMapping(node, node.getChild(i - 1));
write(ASEmitterTokens.SEMICOLON);
endMapping(node);
writeNewline();
if (!isHoisting && childNeedsDefault)
{
write("//");
}
writeToken(ASEmitterTokens.VAR);
}
else
{
startMapping(node, node.getChild(i - 1));
writeToken(ASEmitterTokens.COMMA);
endMapping(node);
}
fjs.emitVarDeclaration((IVariableNode) child);
}
}
}
}
}