dagger.internal.codegen.binding.Binding 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 com.google.common.base.Suppliers.memoize;
import static javax.lang.model.element.Modifier.ABSTRACT;
import static javax.lang.model.element.Modifier.STATIC;
import com.google.common.base.Supplier;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableSet;
import com.google.common.collect.Lists;
import com.google.common.collect.Sets;
import dagger.internal.codegen.langmodel.DaggerTypes;
import dagger.spi.model.BindingKind;
import dagger.spi.model.DependencyRequest;
import dagger.spi.model.Scope;
import java.util.List;
import java.util.Optional;
import java.util.Set;
import javax.lang.model.element.Element;
import javax.lang.model.element.Modifier;
import javax.lang.model.element.TypeElement;
import javax.lang.model.element.TypeParameterElement;
import javax.lang.model.type.DeclaredType;
import javax.lang.model.type.TypeMirror;
import javax.lang.model.util.SimpleTypeVisitor6;
/**
* An abstract type for classes representing a Dagger binding. Particularly, contains the {@link
* Element} that generated the binding and the {@link DependencyRequest} instances that are required
* to satisfy the binding, but leaves the specifics of the mechanism of the binding to the
* subtypes.
*/
public abstract class Binding extends BindingDeclaration {
/**
* Returns {@code true} if using this binding requires an instance of the {@link
* #contributingModule()}.
*/
public boolean requiresModuleInstance() {
if (!bindingElement().isPresent() || !contributingModule().isPresent()) {
return false;
}
Set modifiers = bindingElement().get().getModifiers();
return !modifiers.contains(ABSTRACT) && !modifiers.contains(STATIC);
}
/**
* Returns {@code true} if this binding may provide {@code null} instead of an instance of {@link
* #key()}. Nullable bindings cannot be requested from {@linkplain DependencyRequest#isNullable()
* non-nullable dependency requests}.
*/
public abstract boolean isNullable();
/** The kind of binding this instance represents. */
public abstract BindingKind kind();
/** The {@link BindingType} of this binding. */
public abstract BindingType bindingType();
/** The {@link FrameworkType} of this binding. */
public final FrameworkType frameworkType() {
return FrameworkType.forBindingType(bindingType());
}
/**
* The explicit set of {@link DependencyRequest dependencies} required to satisfy this binding as
* defined by the user-defined injection sites.
*/
public abstract ImmutableSet explicitDependencies();
/**
* The set of {@link DependencyRequest dependencies} that are added by the framework rather than a
* user-defined injection site. This returns an unmodifiable set.
*/
public ImmutableSet implicitDependencies() {
return ImmutableSet.of();
}
private final Supplier> dependencies =
memoize(
() -> {
ImmutableSet implicitDependencies = implicitDependencies();
return ImmutableSet.copyOf(
implicitDependencies.isEmpty()
? explicitDependencies()
: Sets.union(implicitDependencies, explicitDependencies()));
});
/**
* The set of {@link DependencyRequest dependencies} required to satisfy this binding. This is the
* union of {@link #explicitDependencies()} and {@link #implicitDependencies()}. This returns an
* unmodifiable set.
*/
public final ImmutableSet dependencies() {
return dependencies.get();
}
/**
* If this binding's key's type parameters are different from those of the {@link
* #bindingTypeElement()}, this is the binding for the {@link #bindingTypeElement()}'s unresolved
* type.
*/
public abstract Optional extends Binding> unresolved();
public Optional scope() {
return Optional.empty();
}
// TODO(sameb): Remove the TypeElement parameter and pull it from the TypeMirror.
static boolean hasNonDefaultTypeParameters(
TypeElement element, TypeMirror type, DaggerTypes types) {
// If the element has no type parameters, nothing can be wrong.
if (element.getTypeParameters().isEmpty()) {
return false;
}
List defaultTypes = Lists.newArrayList();
for (TypeParameterElement parameter : element.getTypeParameters()) {
defaultTypes.add(parameter.asType());
}
List actualTypes =
type.accept(
new SimpleTypeVisitor6, Void>() {
@Override
protected List defaultAction(TypeMirror e, Void p) {
return ImmutableList.of();
}
@Override
public List visitDeclared(DeclaredType t, Void p) {
return ImmutableList.copyOf(t.getTypeArguments());
}
},
null);
// The actual type parameter size can be different if the user is using a raw type.
if (defaultTypes.size() != actualTypes.size()) {
return true;
}
for (int i = 0; i < defaultTypes.size(); i++) {
if (!types.isSameType(defaultTypes.get(i), actualTypes.get(i))) {
return true;
}
}
return false;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy