/******************************************************************************* * Copyright (c) 2014 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 melnorme.lang.tooling.engine; import static melnorme.utilbox.core.Assert.AssertNamespace.assertTrue; import melnorme.lang.tooling.ast_actual.ElementDoc; import melnorme.lang.tooling.engine.resolver.NamedElementSemantics; import melnorme.lang.tooling.engine.resolver.NonValueConcreteElementSemantics; import melnorme.lang.tooling.engine.scoping.CommonScopeLookup; import melnorme.lang.tooling.symbols.AbstractNamedElement; import melnorme.lang.tooling.symbols.IConcreteNamedElement; import melnorme.lang.tooling.symbols.INamedElement; import melnorme.utilbox.collections.ArrayList2; import melnorme.utilbox.collections.Collection2; import dtool.ast.definitions.EArcheType; import dtool.engine.util.NamedElementUtil; /** * An overloaded named element aggregates several elements with the same name in the same scope. * This can be a semantic error or not, depending on the composition of the elements. * For example it is usually valid for functions elements with the same name to exists. */ public class OverloadedNamedElement extends AbstractNamedElement implements IConcreteNamedElement { public static final String ERROR_NAME = ErrorElement.ERROR_PREFIX + "NameConflict"; protected final ArrayList2<INamedElement> elements; protected final INamedElement firstElement; public OverloadedNamedElement(INamedElement firstElement) { super(firstElement.getName(), firstElement.getLexicalParent(), firstElement); this.firstElement = firstElement; this.elements = new ArrayList2<>(); elements.add(firstElement); } @Override protected void doSetElementSemanticReady() { checkAreSemanticReady(elements, true); } /* ----------------- ----------------- */ @Override public EArcheType getArcheType() { return EArcheType.Error; } @Override public String getNameInRegularNamespace() { return firstElement.getNameInRegularNamespace(); } @Override public String getFullyQualifiedName() { return firstElement.getFullyQualifiedName(); } @Override public String getModuleFullName() { return firstElement.getModuleFullName(); } @Override public ElementDoc resolveDDoc() { return null; } public Collection2<INamedElement> getOverloadedElements() { return elements; } public void addElement(INamedElement newElement) { assertTrue(isSemanticReady() == false); assertTrue(newElement.getNameInRegularNamespace().equals(firstElement.getNameInRegularNamespace())); elements.add(newElement); } @Override public String toString() { StringBuilder sb = new StringBuilder(); sb.append(ERROR_NAME); sb.append("["); boolean first = true; for (INamedElement namedElement : getOverloadedElements()) { if(first) { first = false; } else { sb.append("| "); } sb.append(NamedElementUtil.namedElementToString(namedElement)); } sb.append("]"); return sb.toString(); } /* ----------------- ----------------- */ @Override protected NamedElementSemantics doCreateSemantics(PickedElement<?> pickedElement) { return new NonValueConcreteElementSemantics(this, pickedElement) { @Override public void resolveSearchInMembersScope(CommonScopeLookup search) { // Do nothing. } }; } }