package de.plushnikov.intellij.plugin.processor.field;
import com.intellij.openapi.util.text.StringUtil;
import com.intellij.psi.PsiAnnotation;
import com.intellij.psi.PsiClass;
import com.intellij.psi.PsiCodeBlock;
import com.intellij.psi.PsiElement;
import com.intellij.psi.PsiField;
import com.intellij.psi.PsiMethod;
import com.intellij.psi.PsiModifier;
import com.intellij.psi.PsiModifierList;
import com.intellij.psi.PsiParameter;
import com.intellij.psi.PsiType;
import de.plushnikov.intellij.plugin.problem.ProblemBuilder;
import de.plushnikov.intellij.plugin.processor.LombokPsiElementUsage;
import de.plushnikov.intellij.plugin.psi.LombokLightMethodBuilder;
import de.plushnikov.intellij.plugin.quickfix.PsiQuickFixFactory;
import de.plushnikov.intellij.plugin.thirdparty.LombokUtils;
import de.plushnikov.intellij.plugin.util.LombokProcessorUtil;
import de.plushnikov.intellij.plugin.util.PsiAnnotationSearchUtil;
import de.plushnikov.intellij.plugin.util.PsiAnnotationUtil;
import de.plushnikov.intellij.plugin.util.PsiClassUtil;
import de.plushnikov.intellij.plugin.util.PsiMethodUtil;
import lombok.Setter;
import org.jetbrains.annotations.NotNull;
import java.util.Collection;
import java.util.List;
/**
* Inspect and validate @Setter lombok annotation on a field
* Creates setter method for this field
*
* @author Plushnikov Michail
*/
public class SetterFieldProcessor extends AbstractFieldProcessor {
public SetterFieldProcessor() {
super(PsiMethod.class, Setter.class);
}
@Override
protected void generatePsiElements(@NotNull PsiField psiField, @NotNull PsiAnnotation psiAnnotation, @NotNull List<? super PsiElement> target) {
final String methodVisibility = LombokProcessorUtil.getMethodModifier(psiAnnotation);
final PsiClass psiClass = psiField.getContainingClass();
if (methodVisibility != null && psiClass != null) {
target.add(createSetterMethod(psiField, psiClass, methodVisibility));
}
}
@Override
protected boolean validate(@NotNull PsiAnnotation psiAnnotation, @NotNull PsiField psiField, @NotNull ProblemBuilder builder) {
boolean result;
result = validateFinalModifier(psiAnnotation, psiField, builder);
if (result) {
result = validateVisibility(psiAnnotation);
if (result) {
result = validateExistingMethods(psiField, builder);
if (result) {
result = validateAccessorPrefix(psiField, builder);
}
}
}
return result;
}
private boolean validateFinalModifier(@NotNull PsiAnnotation psiAnnotation, @NotNull PsiField psiField, @NotNull ProblemBuilder builder) {
boolean result = true;
if (psiField.hasModifierProperty(PsiModifier.FINAL) && null != LombokProcessorUtil.getMethodModifier(psiAnnotation)) {
builder.addWarning("Not generating setter for this field: Setters cannot be generated for final fields.",
PsiQuickFixFactory.createModifierListFix(psiField, PsiModifier.FINAL, false, false));
result = false;
}
return result;
}
private boolean validateVisibility(@NotNull PsiAnnotation psiAnnotation) {
final String methodVisibility = LombokProcessorUtil.getMethodModifier(psiAnnotation);
return null != methodVisibility;
}
private boolean validateExistingMethods(@NotNull PsiField psiField, @NotNull ProblemBuilder builder) {
boolean result = true;
final PsiClass psiClass = psiField.getContainingClass();
if (null != psiClass) {
final Collection<PsiMethod> classMethods = PsiClassUtil.collectClassMethodsIntern(psiClass);
filterToleratedElements(classMethods);
final boolean isBoolean = PsiType.BOOLEAN.equals(psiField.getType());
final Collection<String> methodNames = getAllSetterNames(psiField, isBoolean);
for (String methodName : methodNames) {
if (PsiMethodUtil.hasSimilarMethod(classMethods, methodName, 1)) {
final String setterMethodName = getSetterName(psiField, isBoolean);
builder.addWarning("Not generated '%s'(): A method with similar name '%s' already exists", setterMethodName, methodName);
result = false;
}
}
}
return result;
}
private boolean validateAccessorPrefix(@NotNull PsiField psiField, @NotNull ProblemBuilder builder) {
boolean result = true;
if (!AccessorsInfo.build(psiField).prefixDefinedAndStartsWith(psiField.getName())) {
builder.addWarning("Not generating setter for this field: It does not fit your @Accessors prefix list.");
result = false;
}
return result;
}
public Collection<String> getAllSetterNames(@NotNull PsiField psiField, boolean isBoolean) {
final AccessorsInfo accessorsInfo = AccessorsInfo.build(psiField);
return LombokUtils.toAllSetterNames(accessorsInfo, psiField.getName(), isBoolean);
}
private String getSetterName(@NotNull PsiField psiField, boolean isBoolean) {
final AccessorsInfo accessorsInfo = AccessorsInfo.build(psiField);
return LombokUtils.toSetterName(accessorsInfo, psiField.getName(), isBoolean);
}
@NotNull
public PsiMethod createSetterMethod(@NotNull PsiField psiField, @NotNull PsiClass psiClass, @NotNull String methodModifier) {
final String fieldName = psiField.getName();
final PsiType psiFieldType = psiField.getType();
final PsiAnnotation setterAnnotation = PsiAnnotationSearchUtil.findAnnotation(psiField, Setter.class);
final String methodName = getSetterName(psiField, PsiType.BOOLEAN.equals(psiFieldType));
PsiType returnType = getReturnType(psiField);
LombokLightMethodBuilder method = new LombokLightMethodBuilder(psiField.getManager(), methodName)
.withMethodReturnType(returnType)
.withContainingClass(psiClass)
.withParameter(fieldName, psiFieldType)
.withNavigationElement(psiField);
if (StringUtil.isNotEmpty(methodModifier)) {
method.withModifier(methodModifier);
}
boolean isStatic = psiField.hasModifierProperty(PsiModifier.STATIC);
if (isStatic) {
method.withModifier(PsiModifier.STATIC);
}
PsiParameter methodParameter = method.getParameterList().getParameters()[0];
PsiModifierList methodParameterModifierList = methodParameter.getModifierList();
if (null != methodParameterModifierList) {
final Collection<String> annotationsToCopy = PsiAnnotationUtil.collectAnnotationsToCopy(psiField,
LombokUtils.NON_NULL_PATTERN, LombokUtils.NULLABLE_PATTERN);
for (String annotationFQN : annotationsToCopy) {
methodParameterModifierList.addAnnotation(annotationFQN);
}
addOnXAnnotations(setterAnnotation, methodParameterModifierList, "onParam");
}
method.withBody(createCodeBlock(psiField, psiClass, returnType, isStatic, methodParameter));
PsiModifierList methodModifierList = method.getModifierList();
copyAnnotations(psiField, methodModifierList, LombokUtils.DEPRECATED_PATTERN);
addOnXAnnotations(setterAnnotation, methodModifierList, "onMethod");
return method;
}
@NotNull
private PsiCodeBlock createCodeBlock(@NotNull PsiField psiField, @NotNull PsiClass psiClass, PsiType returnType, boolean isStatic, PsiParameter methodParameter) {
final String blockText;
if (isShouldGenerateFullBodyBlock()) {
final String thisOrClass = isStatic ? psiClass.getName() : "this";
blockText = String.format("%s.%s = %s; ", thisOrClass, psiField.getName(), methodParameter.getName());
} else {
blockText = "";
}
String codeBlockText = blockText;
if (!isStatic && !PsiType.VOID.equals(returnType)) {
codeBlockText += "return this;";
}
return PsiMethodUtil.createCodeBlockFromText(codeBlockText, psiClass);
}
private PsiType getReturnType(@NotNull PsiField psiField) {
PsiType result = PsiType.VOID;
if (!psiField.hasModifierProperty(PsiModifier.STATIC) && AccessorsInfo.build(psiField).isChain()) {
final PsiClass fieldClass = psiField.getContainingClass();
if (null != fieldClass) {
result = PsiClassUtil.getTypeWithGenerics(fieldClass);
}
}
return result;
}
@Override
public LombokPsiElementUsage checkFieldUsage(@NotNull PsiField psiField, @NotNull PsiAnnotation psiAnnotation) {
return LombokPsiElementUsage.WRITE;
}
}