/**
* Copyright (C) 2006-2017 INRIA and contributors
* Spoon - http://spoon.gforge.inria.fr/
*
* This software is governed by the CeCILL-C License under French law and
* abiding by the rules of distribution of free software. You can use, modify
* and/or redistribute the software under the terms of the CeCILL-C license as
* circulated by CEA, CNRS and INRIA at http://www.cecill.info.
*
* This program is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the CeCILL-C License for more details.
*
* The fact that you are presently reading this means that you have had
* knowledge of the CeCILL-C license and that you accept its terms.
*/
package spoon.support.reflect.declaration;
import spoon.reflect.declaration.CtElement;
import spoon.reflect.declaration.CtFormalTypeDeclarer;
import spoon.reflect.declaration.CtMethod;
import spoon.reflect.declaration.CtModifiable;
import spoon.reflect.declaration.CtShadowable;
import spoon.reflect.declaration.CtTypeParameter;
import spoon.reflect.declaration.CtTypedElement;
import spoon.reflect.declaration.ModifierKind;
import spoon.reflect.reference.CtTypeReference;
import spoon.reflect.visitor.CtVisitor;
import spoon.support.visitor.MethodTypingContext;
import java.util.ArrayList;
import java.util.EnumSet;
import java.util.List;
import java.util.Set;
import static spoon.reflect.ModelElementContainerDefaultCapacities.TYPE_TYPE_PARAMETERS_CONTAINER_DEFAULT_CAPACITY;
/**
* The implementation for {@link spoon.reflect.declaration.CtMethod}.
*
* @author Renaud Pawlak
*/
public class CtMethodImpl<T> extends CtExecutableImpl<T> implements CtMethod<T> {
private static final long serialVersionUID = 1L;
CtTypeReference<T> returnType;
boolean defaultMethod = false;
List<CtTypeParameter> formalCtTypeParameters = emptyList();
Set<ModifierKind> modifiers = CtElementImpl.emptySet();
public CtMethodImpl() {
super();
}
@Override
public void accept(CtVisitor v) {
v.visitCtMethod(this);
}
@Override
public CtTypeReference<T> getType() {
return returnType;
}
@Override
public <C extends CtTypedElement> C setType(CtTypeReference<T> type) {
if (type != null) {
type.setParent(this);
}
this.returnType = type;
return (C) this;
}
@Override
public boolean isDefaultMethod() {
return defaultMethod;
}
@Override
public <C extends CtMethod<T>> C setDefaultMethod(boolean defaultMethod) {
this.defaultMethod = defaultMethod;
return (C) this;
}
@Override
public List<CtTypeParameter> getFormalCtTypeParameters() {
return formalCtTypeParameters;
}
@Override
public <C extends CtFormalTypeDeclarer> C setFormalCtTypeParameters(List<CtTypeParameter> formalTypeParameters) {
if (formalTypeParameters == null || formalTypeParameters.isEmpty()) {
this.formalCtTypeParameters = CtElementImpl.emptyList();
return (C) this;
}
if (this.formalCtTypeParameters == CtElementImpl.<CtTypeParameter>emptyList()) {
this.formalCtTypeParameters = new ArrayList<>(TYPE_TYPE_PARAMETERS_CONTAINER_DEFAULT_CAPACITY);
}
this.formalCtTypeParameters.clear();
for (CtTypeParameter formalTypeParameter : formalTypeParameters) {
addFormalCtTypeParameter(formalTypeParameter);
}
return (C) this;
}
@Override
public <C extends CtFormalTypeDeclarer> C addFormalCtTypeParameter(CtTypeParameter formalTypeParameter) {
if (formalTypeParameter == null) {
return (C) this;
}
if (formalCtTypeParameters == CtElementImpl.<CtTypeParameter>emptyList()) {
formalCtTypeParameters = new ArrayList<>(TYPE_TYPE_PARAMETERS_CONTAINER_DEFAULT_CAPACITY);
}
formalTypeParameter.setParent(this);
formalCtTypeParameters.add(formalTypeParameter);
return (C) this;
}
@Override
public boolean removeFormalCtTypeParameter(CtTypeParameter formalTypeParameter) {
return formalCtTypeParameters.contains(formalTypeParameter) && formalCtTypeParameters.remove(formalTypeParameter);
}
@Override
public Set<ModifierKind> getModifiers() {
return modifiers;
}
@Override
public boolean hasModifier(ModifierKind modifier) {
return getModifiers().contains(modifier);
}
@Override
public <C extends CtModifiable> C setModifiers(Set<ModifierKind> modifiers) {
if (modifiers.size() > 0) {
this.modifiers = EnumSet.copyOf(modifiers);
}
return (C) this;
}
@Override
public <C extends CtModifiable> C addModifier(ModifierKind modifier) {
if (modifiers == CtElementImpl.<ModifierKind>emptySet()) {
this.modifiers = EnumSet.noneOf(ModifierKind.class);
}
modifiers.add(modifier);
return (C) this;
}
@Override
public boolean removeModifier(ModifierKind modifier) {
return !modifiers.isEmpty() && modifiers.remove(modifier);
}
@Override
public <C extends CtModifiable> C setVisibility(ModifierKind visibility) {
if (modifiers == CtElementImpl.<ModifierKind>emptySet()) {
this.modifiers = EnumSet.noneOf(ModifierKind.class);
}
getModifiers().remove(ModifierKind.PUBLIC);
getModifiers().remove(ModifierKind.PROTECTED);
getModifiers().remove(ModifierKind.PRIVATE);
getModifiers().add(visibility);
return (C) this;
}
@Override
public ModifierKind getVisibility() {
if (getModifiers().contains(ModifierKind.PUBLIC)) {
return ModifierKind.PUBLIC;
}
if (getModifiers().contains(ModifierKind.PROTECTED)) {
return ModifierKind.PROTECTED;
}
if (getModifiers().contains(ModifierKind.PRIVATE)) {
return ModifierKind.PRIVATE;
}
return null;
}
@Override
public <R extends T> void replace(CtMethod<T> element) {
replace((CtElement) element);
}
@Override
public boolean isOverriding(CtMethod<?> superMethod) {
return new MethodTypingContext().setMethod(this).isOverriding(superMethod);
}
boolean isShadow;
@Override
public boolean isShadow() {
return isShadow;
}
@Override
public <E extends CtShadowable> E setShadow(boolean isShadow) {
this.isShadow = isShadow;
return (E) this;
}
@Override
public CtMethod<T> clone() {
return (CtMethod<T>) super.clone();
}
}