/*******************************************************************************
* Copyright (c) 2012 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 dtool.util.NewUtils.assertCast;
import static melnorme.utilbox.core.Assert.AssertNamespace.assertTrue;
import melnorme.lang.tooling.ast.CommonASTNode;
import melnorme.lang.tooling.ast.IASTNode;
import melnorme.lang.tooling.ast.IASTVisitor;
import melnorme.lang.tooling.ast.util.ASTCodePrinter;
import melnorme.lang.tooling.ast.util.NodeVector;
import melnorme.lang.tooling.ast_actual.ASTNodeTypes;
import melnorme.lang.tooling.engine.PickedElement;
import melnorme.lang.tooling.engine.resolver.AliasSemantics.RefBasedAliasSemantics;
import melnorme.lang.tooling.engine.resolver.NamedElementSemantics;
import melnorme.lang.tooling.engine.scoping.INonScopedContainer;
import melnorme.utilbox.misc.IteratorUtil;
import dtool.ast.declarations.Attribute;
import dtool.ast.declarations.IDeclaration;
import dtool.ast.references.Reference;
import dtool.ast.statements.IStatement;
import dtool.parser.common.Token;
/**
* A definition of an alias, in the old syntax:
* <code>alias StorageClasses BasicType Declarators</code>
* when declarator declares one or more variable-like aliases.
*
* @see http://dlang.org/declaration.html#AliasDeclaration
*/
// Note implementation similarities with {@link DefinitionVariable} and {@link DefVarFragment}
public class DefinitionAliasVarDecl extends CommonDefinition implements IDeclaration, IStatement, INonScopedContainer {
public final NodeVector<Attribute> aliasedAttributes;
public final Reference target;
public final Reference cstyleSuffix;
public final NodeVector<AliasVarDeclFragment> fragments;
public DefinitionAliasVarDecl(Token[] comments, NodeVector<Attribute> aliasedAttributes, Reference target,
DefSymbol defName, Reference cstyleSuffix, NodeVector<AliasVarDeclFragment> fragments) {
super(comments, defName);
this.aliasedAttributes = parentize(aliasedAttributes);
this.target = parentize(target);
this.cstyleSuffix = parentize(cstyleSuffix);
this.fragments = parentize(fragments);
assertTrue(fragments == null || fragments.size() > 0);
}
@Override
public ASTNodeTypes getNodeType() {
return ASTNodeTypes.DEFINITION_ALIAS_VAR_DECL;
}
@Override
public void visitChildren(IASTVisitor visitor) {
acceptVisitor(visitor, aliasedAttributes);
acceptVisitor(visitor, target);
acceptVisitor(visitor, defName);
acceptVisitor(visitor, cstyleSuffix);
acceptVisitor(visitor, fragments);
}
@Override
protected CommonASTNode doCloneTree() {
return new DefinitionAliasVarDecl(comments, clone(aliasedAttributes), clone(target), clone(defName),
clone(cstyleSuffix), clone(fragments));
}
@Override
public void toStringAsCode(ASTCodePrinter cp) {
cp.append("alias ");
cp.appendList(aliasedAttributes, " ", true);
cp.append(target, " ");
cp.append(defName);
cp.append(cstyleSuffix);
cp.appendList(", ", fragments, ", ", "");
cp.append(";");
}
@Override
public EArcheType getArcheType() {
return EArcheType.Alias;
}
/* ----------------- ----------------- */
@Override
public Iterable<? extends IASTNode> getMembersIterable() {
return IteratorUtil.nonNullIterable(fragments);
}
@Override
protected NamedElementSemantics doCreateSemantics(PickedElement<?> pickedElement) {
return new RefBasedAliasSemantics(this, pickedElement) {
@Override
protected Reference getAliasTarget() {
return target;
}
};
}
/* ----------------- ----------------- */
public static class AliasVarDeclFragment extends DefUnit {
public AliasVarDeclFragment(DefSymbol defName) {
super(defName);
}
@Override
public ASTNodeTypes getNodeType() {
return ASTNodeTypes.ALIAS_VAR_DECL_FRAGMENT;
}
@Override
public DefinitionAliasVarDecl getParent_Concrete() {
return assertCast(parent, DefinitionAliasVarDecl.class);
}
@Override
public void visitChildren(IASTVisitor visitor) {
acceptVisitor(visitor, defName);
}
@Override
protected CommonASTNode doCloneTree() {
return new AliasVarDeclFragment(clone(defName));
}
@Override
public void toStringAsCode(ASTCodePrinter cp) {
cp.append(defName);
}
@Override
public EArcheType getArcheType() {
return EArcheType.Alias;
}
public Reference getAliasTarget() {
return getParent_Concrete().target;
}
/* ----------------- ----------------- */
@Override
protected NamedElementSemantics doCreateSemantics(PickedElement<?> pickedElement) {
return new RefBasedAliasSemantics(this, pickedElement) {
@Override
protected Reference getAliasTarget() {
return AliasVarDeclFragment.this.getAliasTarget();
}
};
}
}
}