dagger.internal.codegen.binding.FrameworkField Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of dagger-compiler Show documentation
Show all versions of dagger-compiler Show documentation
A fast dependency injector for Android and Java.
/*
* Copyright (C) 2014 The Dagger Authors.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package dagger.internal.codegen.binding;
import static androidx.room.compiler.processing.XElementKt.isConstructor;
import static androidx.room.compiler.processing.XElementKt.isMethod;
import static androidx.room.compiler.processing.XElementKt.isMethodParameter;
import static androidx.room.compiler.processing.XElementKt.isTypeElement;
import static dagger.internal.codegen.model.BindingKind.MEMBERS_INJECTOR;
import static dagger.internal.codegen.xprocessing.XElements.getSimpleName;
import androidx.room.compiler.processing.XElement;
import com.google.auto.value.AutoValue;
import com.google.common.base.CaseFormat;
import com.google.common.base.Preconditions;
import com.squareup.javapoet.ClassName;
import com.squareup.javapoet.ParameterizedTypeName;
import com.squareup.javapoet.TypeName;
import dagger.internal.codegen.base.MapType;
import dagger.internal.codegen.javapoet.TypeNames;
import java.util.Optional;
/**
* A value object that represents a field in the generated Component class.
*
* Examples:
*
*
* - {@code Provider
}
* - {@code Producer
}
* - {@code Provider
*/
@AutoValue
public abstract class FrameworkField {
/**
* Creates a framework field.
*
* @param fieldType the type of the framework field (e.g., {@code Provider}).
* @param fieldName the base name of the field. The name of the raw type of the field will be
* added as a suffix
*/
public static FrameworkField create(TypeName fieldType, String fieldName) {
Preconditions.checkState(
fieldType instanceof ClassName || fieldType instanceof ParameterizedTypeName,
"Can only create a field with a class name or parameterized type name");
String suffix = ((ClassName) TypeNames.rawTypeName(fieldType)).simpleName();
return new AutoValue_FrameworkField(
fieldType,
fieldName.endsWith(suffix) ? fieldName : fieldName + suffix);
}
/**
* A framework field for a {@link ContributionBinding}.
*
* @param frameworkClass if present, the field will use this framework class instead of the normal
* one for the binding's type.
*/
public static FrameworkField forBinding(
ContributionBinding binding, Optional frameworkClassName) {
return create(
fieldType(binding, frameworkClassName.orElse(binding.frameworkType().frameworkClassName())),
frameworkFieldName(binding));
}
private static TypeName fieldType(ContributionBinding binding, ClassName frameworkClassName) {
if (binding.contributionType().isMultibinding()) {
return ParameterizedTypeName.get(frameworkClassName, binding.contributedType().getTypeName());
}
// If the binding key type is a Map>, we need to change field type to a raw
// type. This is because it actually needs to be changed to Map>,
// but that gets into assignment issues when the field is passed to methods that expect
// Map>. We could add casts everywhere, but it is easier to just
// make the field itself a raw type.
if (MapType.isMapOfProvider(binding.contributedType())) {
return frameworkClassName;
}
return ParameterizedTypeName.get(
frameworkClassName, binding.key().type().xprocessing().getTypeName());
}
private static String frameworkFieldName(ContributionBinding binding) {
if (binding.bindingElement().isPresent()) {
String name = bindingElementName(binding.bindingElement().get());
return binding.kind().equals(MEMBERS_INJECTOR) ? name + "MembersInjector" : name;
}
return KeyVariableNamer.name(binding.key());
}
private static String bindingElementName(XElement bindingElement) {
if (isConstructor(bindingElement)) {
return bindingElementName(bindingElement.getEnclosingElement());
} else if (isMethod(bindingElement)) {
return getSimpleName(bindingElement);
} else if (isTypeElement(bindingElement)) {
return CaseFormat.UPPER_CAMEL.to(CaseFormat.LOWER_CAMEL, getSimpleName(bindingElement));
} else if (isMethodParameter(bindingElement)) {
return getSimpleName(bindingElement);
} else {
throw new IllegalArgumentException("Unexpected binding " + bindingElement);
}
}
public abstract TypeName type();
public abstract String name();
}