/*******************************************************************************
* Copyright (c) 2011 Bruno Medeiros and other Contributors.
* All rights reserved. This program and the accompanying materials
* are made available under the terms of the Eclipse Public License v1.0
* which accompanies this distribution, and is available at
* http://www.eclipse.org/legal/epl-v10.html
*
* Contributors:
* Bruno Medeiros - initial API and implementation
*******************************************************************************/
package dtool.ast.definitions;
import static melnorme.utilbox.core.Assert.AssertNamespace.assertNotNull;
import melnorme.lang.tooling.ast.CommonASTNode;
import melnorme.lang.tooling.ast.IASTVisitor;
import melnorme.lang.tooling.ast.INamedElementNode;
import melnorme.lang.tooling.ast.util.ASTCodePrinter;
import melnorme.lang.tooling.ast_actual.ASTNodeTypes;
import melnorme.lang.tooling.context.ISemanticContext;
import melnorme.lang.tooling.engine.PickedElement;
import melnorme.lang.tooling.engine.resolver.NamedElementSemantics;
import melnorme.lang.tooling.engine.resolver.ResolvableUtil;
import melnorme.lang.tooling.engine.resolver.VarSemantics;
import melnorme.lang.tooling.symbols.IConcreteNamedElement;
import melnorme.lang.tooling.symbols.ITypeNamedElement;
import melnorme.utilbox.collections.Indexable;
import dtool.ast.expressions.Expression;
import dtool.ast.expressions.Resolvable;
import dtool.ast.references.Reference;
import dtool.engine.analysis.templates.TemplateParameterAnalyser;
import dtool.engine.analysis.templates.VarElement;
public class TemplateValueParam extends DefUnit implements IConcreteNamedElement, ITemplateParameter {
public final Reference type;
public final Expression specializationValue;
public final Expression defaultValue;
public TemplateValueParam(Reference type, DefSymbol defName, Expression specializationValue,
Expression defaultValue) {
super(defName);
this.type = parentize(assertNotNull(type));
this.specializationValue = parentize(specializationValue);
this.defaultValue = parentize(defaultValue);
}
@Override
public ASTNodeTypes getNodeType() {
return ASTNodeTypes.TEMPLATE_VALUE_PARAM;
}
@Override
public void visitChildren(IASTVisitor visitor) {
acceptVisitor(visitor, type);
acceptVisitor(visitor, defName);
acceptVisitor(visitor, specializationValue);
acceptVisitor(visitor, defaultValue);
}
@Override
protected CommonASTNode doCloneTree() {
return new TemplateValueParam(clone(type), clone(defName), clone(specializationValue), clone(defaultValue));
}
@Override
public void toStringAsCode(ASTCodePrinter cp) {
cp.append(type, " ");
cp.append(defName);
cp.append(" : ", specializationValue);
cp.append(" = ", defaultValue);
}
@Override
public EArcheType getArcheType() {
return EArcheType.Variable;
}
/* ----------------- ----------------- */
@Override
protected NamedElementSemantics doCreateSemantics(PickedElement<?> pickedElement) {
return new VarSemantics(this, pickedElement) {
@Override
protected Reference getTypeReference() {
return type;
}
};
}
/* ----------------- ----------------- */
@Override
public TemplateParameterAnalyser getParameterAnalyser() {
return templateParameterAnalyser;
}
protected final TemplateParameterAnalyser templateParameterAnalyser = new TemplateParameterAnalyser() {
@Override
public TplMatchLevel getMatchPriority(Resolvable tplArg, ISemanticContext context) {
ITypeNamedElement argType = ResolvableUtil.resolveTypeOfExpression(tplArg, context);
if(argType.getArcheType() == EArcheType.Error) {
return TplMatchLevel.NONE;
}
ITypeNamedElement paramType = ResolvableUtil.resolveTargetType(type, context);
if(argType.getSemantics(context).isCompatibleWith(paramType)) {
return TplMatchLevel.VALUE;
}
return TplMatchLevel.NONE;
}
@Override
public INamedElementNode createTemplateArgument(Indexable<Resolvable> tplArgs, int argIndex,
ISemanticContext tplRefContext) {
return new VarElement(defName, type, getArgument(tplArgs, argIndex, null));
}
};
}