dagger.internal.codegen.binding.ContributionBinding 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.
The newest version!
/*
* 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 com.google.common.base.Preconditions.checkState;
import static dagger.internal.codegen.xprocessing.XElements.asMethod;
import static dagger.internal.codegen.xprocessing.XElements.isAbstract;
import static dagger.internal.codegen.xprocessing.XElements.isStatic;
import androidx.room.compiler.processing.XAnnotation;
import androidx.room.compiler.processing.XElement;
import androidx.room.compiler.processing.XElementKt;
import androidx.room.compiler.processing.XType;
import androidx.room.compiler.processing.XTypeElement;
import com.google.common.collect.ImmutableSet;
import com.google.errorprone.annotations.CanIgnoreReturnValue;
import com.google.errorprone.annotations.CheckReturnValue;
import dagger.internal.codegen.base.ContributionType.HasContributionType;
import dagger.internal.codegen.base.MapType;
import dagger.internal.codegen.base.SetType;
import dagger.internal.codegen.compileroption.CompilerOptions;
import dagger.internal.codegen.model.BindingKind;
import dagger.internal.codegen.model.Key;
import dagger.internal.codegen.model.Scope;
import dagger.internal.codegen.xprocessing.Nullability;
import dagger.internal.codegen.xprocessing.XTypes;
import java.util.Optional;
/**
* An abstract class for a value object representing the mechanism by which a {@link Key} can be
* contributed to a dependency graph.
*/
@CheckReturnValue
public abstract class ContributionBinding extends Binding implements HasContributionType {
/** Returns the nullability of this binding. */
public abstract Nullability nullability();
private static final ImmutableSet KINDS_TO_CHECK_FOR_NULL =
ImmutableSet.of(BindingKind.PROVISION, BindingKind.COMPONENT_PROVISION);
public boolean shouldCheckForNull(CompilerOptions compilerOptions) {
return KINDS_TO_CHECK_FOR_NULL.contains(kind())
&& contributedPrimitiveType().isEmpty()
&& !isNullable()
&& compilerOptions.doCheckForNulls();
}
/** Returns the map key if this is a {@code Map} multibinding contribution. */
public Optional mapKey() {
return bindingElement().flatMap(MapKeys::getMapKey);
}
/** If {@link #bindingElement()} is a method that returns a primitive type, returns that type. */
public final Optional contributedPrimitiveType() {
return bindingElement()
.filter(XElementKt::isMethod)
.map(bindingElement -> asMethod(bindingElement).getReturnType())
.filter(XTypes::isPrimitive);
}
@Override
public boolean requiresModuleInstance() {
return contributingModule().isPresent()
&& bindingElement().isPresent()
&& !isAbstract(bindingElement().get())
&& !isStatic(bindingElement().get())
&& !isContributingModuleKotlinObject();
}
@Override
public final boolean isNullable() {
return nullability().isNullable();
}
/**
* Returns {@code true} if the contributing module is a Kotlin object. Note that a companion
* object is also considered a Kotlin object.
*/
private boolean isContributingModuleKotlinObject() {
return contributingModule().isPresent()
&& (contributingModule().get().isKotlinObject()
|| contributingModule().get().isCompanionObject());
}
/**
* The {@link XType type} for the {@code Factory} or {@code Producer} which is created for
* this binding. Uses the binding's key, V in the case of {@code Map>>}, and
* E {@code Set} for {@link dagger.multibindings.IntoSet @IntoSet} methods.
*/
public final XType contributedType() {
switch (contributionType()) {
case MAP:
return MapType.from(key()).unwrappedFrameworkValueType();
case SET:
return SetType.from(key()).elementType();
case SET_VALUES:
case UNIQUE:
return key().type().xprocessing();
}
throw new AssertionError();
}
public abstract Builder, ?> toBuilder();
/** Returns a new {@link ContributionBinding} with the given {@link BindingType}. */
final ContributionBinding withBindingType(BindingType bindingType) {
checkState(optionalBindingType().isEmpty());
switch (kind()) {
case DELEGATE:
return ((DelegateBinding) this).toBuilder()
.optionalBindingType(Optional.of(bindingType))
.build();
case OPTIONAL:
return ((OptionalBinding) this).toBuilder()
.optionalBindingType(Optional.of(bindingType))
.build();
case MULTIBOUND_MAP:
return ((MultiboundMapBinding) this).toBuilder()
.optionalBindingType(Optional.of(bindingType))
.build();
case MULTIBOUND_SET:
return ((MultiboundSetBinding) this).toBuilder()
.optionalBindingType(Optional.of(bindingType))
.build();
default:
throw new AssertionError("Unexpected binding kind: " + kind());
}
}
/**
* Base builder for {@link com.google.auto.value.AutoValue @AutoValue} subclasses of {@link
* ContributionBinding}.
*/
abstract static class Builder> {
@CanIgnoreReturnValue
abstract B unresolved(Optional extends Binding> unresolved);
@CanIgnoreReturnValue
abstract B bindingElement(XElement bindingElement);
@CanIgnoreReturnValue
abstract B bindingElement(Optional bindingElement);
@CanIgnoreReturnValue
final B clearBindingElement() {
return bindingElement(Optional.empty());
};
@CanIgnoreReturnValue
abstract B contributingModule(XTypeElement contributingModule);
@CanIgnoreReturnValue
abstract B key(Key key);
@CanIgnoreReturnValue
abstract B scope(Optional scope);
abstract C build();
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy