/**
* 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.code;
import spoon.SpoonException;
import spoon.reflect.code.CtBlock;
import spoon.reflect.code.CtBodyHolder;
import spoon.reflect.code.CtExpression;
import spoon.reflect.code.CtLambda;
import spoon.reflect.code.CtStatement;
import spoon.reflect.declaration.CtExecutable;
import spoon.reflect.declaration.CtMethod;
import spoon.reflect.declaration.CtNamedElement;
import spoon.reflect.declaration.CtParameter;
import spoon.reflect.declaration.CtType;
import spoon.reflect.declaration.ModifierKind;
import spoon.reflect.reference.CtExecutableReference;
import spoon.reflect.reference.CtTypeReference;
import spoon.reflect.visitor.CtVisitor;
import spoon.support.reflect.declaration.CtElementImpl;
import spoon.support.util.QualifiedNameBasedSortedSet;
import spoon.support.visitor.SignaturePrinter;
import java.util.ArrayList;
import java.util.List;
import java.util.Set;
import static spoon.reflect.ModelElementContainerDefaultCapacities.PARAMETERS_CONTAINER_DEFAULT_CAPACITY;
public class CtLambdaImpl<T> extends CtExpressionImpl<T> implements CtLambda<T> {
String simpleName = "";
CtExpression<T> expression;
CtBlock<?> body;
List<CtParameter<?>> parameters = emptyList();
Set<CtTypeReference<? extends Throwable>> thrownTypes = emptySet();
@Override
public void accept(CtVisitor visitor) {
visitor.visitCtLambda(this);
}
@Override
public String getSimpleName() {
return simpleName;
}
@Override
public <C extends CtNamedElement> C setSimpleName(String simpleName) {
this.simpleName = simpleName;
return (C) this;
}
@Override
@SuppressWarnings("unchecked")
public CtBlock<T> getBody() {
return (CtBlock<T>) body;
}
@Override
public <C extends CtBodyHolder> C setBody(CtStatement statement) {
if (statement != null) {
CtBlock<?> body = getFactory().Code().getOrCreateCtBlock(statement);
if (expression != null && body != null) {
throw new SpoonException("A lambda can't have two bodys.");
}
if (body != null) {
body.setParent(this);
}
this.body = body;
} else {
this.body = null;
}
return (C) this;
}
@SuppressWarnings("unchecked")
@Override
public <R> CtMethod<R> getOverriddenMethod() {
//The type of this lambda expression. For example: `Consumer<Integer>`
CtTypeReference<T> lambdaTypeRef = getType();
if (lambdaTypeRef == null) {
//it can be null in noclasspath mode, so we do not know which method is called, by lambda
return null;
}
CtType<T> lambdaType = lambdaTypeRef.getTypeDeclaration();
if (lambdaType.isInterface() == false) {
throw new SpoonException("The lambda can be based on interface only. But type " + lambdaTypeRef.getQualifiedName() + " is not an interface");
}
Set<CtMethod<?>> lambdaTypeMethods = lambdaType.getAllMethods();
CtMethod<?> lambdaExecutableMethod = null;
if (lambdaTypeMethods.size() == 1) {
//even the default method can be used, if it is the only one
lambdaExecutableMethod = lambdaTypeMethods.iterator().next();
} else {
for (CtMethod<?> method : lambdaTypeMethods) {
if (method.isDefaultMethod() || method.hasModifier(ModifierKind.PRIVATE) || method.hasModifier(ModifierKind.STATIC)) {
continue;
}
if (lambdaExecutableMethod != null) {
throw new SpoonException("The lambda can be based on interface, which has only one method. But " + lambdaTypeRef.getQualifiedName() + " has at least two: " + lambdaExecutableMethod.getSignature() + " and " + method.getSignature());
}
lambdaExecutableMethod = method;
}
}
if (lambdaExecutableMethod == null) {
throw new SpoonException("The lambda can be based on interface, which has one method. But " + lambdaTypeRef.getQualifiedName() + " has no one");
}
return (CtMethod<R>) lambdaExecutableMethod;
}
@Override
public List<CtParameter<?>> getParameters() {
return unmodifiableList(parameters);
}
@Override
public <C extends CtExecutable<T>> C setParameters(List<CtParameter<?>> params) {
if (params == null || params.isEmpty()) {
this.parameters = CtElementImpl.emptyList();
return (C) this;
}
if (this.parameters == CtElementImpl.<CtParameter<?>>emptyList()) {
this.parameters = new ArrayList<>(PARAMETERS_CONTAINER_DEFAULT_CAPACITY);
}
this.parameters.clear();
for (CtParameter<?> p : params) {
addParameter(p);
}
return (C) this;
}
@Override
public <C extends CtExecutable<T>> C addParameter(CtParameter<?> parameter) {
if (parameter == null) {
return (C) this;
}
if (parameters == CtElementImpl.<CtParameter<?>>emptyList()) {
parameters = new ArrayList<>(PARAMETERS_CONTAINER_DEFAULT_CAPACITY);
}
parameter.setParent(this);
parameters.add(parameter);
return (C) this;
}
@Override
public boolean removeParameter(CtParameter<?> parameter) {
return parameters != CtElementImpl.<CtParameter<?>>emptyList()
&& parameters.remove(parameter);
}
@Override
public Set<CtTypeReference<? extends Throwable>> getThrownTypes() {
return thrownTypes;
}
@Override
public <C extends CtExecutable<T>> C setThrownTypes(Set<CtTypeReference<? extends Throwable>> thrownTypes) {
if (thrownTypes == null || thrownTypes.isEmpty()) {
this.thrownTypes = CtElementImpl.emptySet();
return (C) this;
}
if (this.thrownTypes == CtElementImpl.<CtTypeReference<? extends Throwable>>emptySet()) {
this.thrownTypes = new QualifiedNameBasedSortedSet<>();
}
this.thrownTypes.clear();
for (CtTypeReference<? extends Throwable> thrownType : thrownTypes) {
addThrownType(thrownType);
}
return (C) this;
}
@Override
public <C extends CtExecutable<T>> C addThrownType(CtTypeReference<? extends Throwable> throwType) {
if (throwType == null) {
return (C) this;
}
if (thrownTypes == CtElementImpl.<CtTypeReference<? extends Throwable>>emptySet()) {
thrownTypes = new QualifiedNameBasedSortedSet<>();
}
throwType.setParent(this);
thrownTypes.add(throwType);
return (C) this;
}
@Override
public boolean removeThrownType(CtTypeReference<? extends Throwable> throwType) {
return thrownTypes.remove(throwType);
}
@Override
public String getSignature() {
final SignaturePrinter pr = new SignaturePrinter();
pr.scan(this);
return pr.getSignature();
}
@Override
public CtExecutableReference<T> getReference() {
return getFactory().Executable().createReference(this);
}
@Override
public CtExpression<T> getExpression() {
return expression;
}
@Override
public <C extends CtLambda<T>> C setExpression(CtExpression<T> expression) {
if (body != null && expression != null) {
throw new SpoonException("A lambda can't have two bodys.");
} else {
if (expression != null) {
expression.setParent(this);
}
this.expression = expression;
}
return (C) this;
}
@Override
public CtLambda<T> clone() {
return (CtLambda<T>) super.clone();
}
}