/******************************************************************************* * 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)); } }; }