1*f585d8a3SJacky Wang /* 2*f585d8a3SJacky Wang * Copyright (C) 2016 The Dagger Authors. 3*f585d8a3SJacky Wang * 4*f585d8a3SJacky Wang * Licensed under the Apache License, Version 2.0 (the "License"); 5*f585d8a3SJacky Wang * you may not use this file except in compliance with the License. 6*f585d8a3SJacky Wang * You may obtain a copy of the License at 7*f585d8a3SJacky Wang * 8*f585d8a3SJacky Wang * http://www.apache.org/licenses/LICENSE-2.0 9*f585d8a3SJacky Wang * 10*f585d8a3SJacky Wang * Unless required by applicable law or agreed to in writing, software 11*f585d8a3SJacky Wang * distributed under the License is distributed on an "AS IS" BASIS, 12*f585d8a3SJacky Wang * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13*f585d8a3SJacky Wang * See the License for the specific language governing permissions and 14*f585d8a3SJacky Wang * limitations under the License. 15*f585d8a3SJacky Wang */ 16*f585d8a3SJacky Wang 17*f585d8a3SJacky Wang package dagger.producers.internal; 18*f585d8a3SJacky Wang 19*f585d8a3SJacky Wang import static dagger.internal.Providers.asDaggerProvider; 20*f585d8a3SJacky Wang import static dagger.producers.internal.Producers.entryPointViewOf; 21*f585d8a3SJacky Wang import static dagger.producers.internal.Producers.nonCancellationPropagatingViewOf; 22*f585d8a3SJacky Wang 23*f585d8a3SJacky Wang import com.google.common.base.Function; 24*f585d8a3SJacky Wang import com.google.common.collect.ImmutableMap; 25*f585d8a3SJacky Wang import com.google.common.collect.Maps; 26*f585d8a3SJacky Wang import com.google.common.util.concurrent.Futures; 27*f585d8a3SJacky Wang import com.google.common.util.concurrent.ListenableFuture; 28*f585d8a3SJacky Wang import dagger.internal.Provider; 29*f585d8a3SJacky Wang import dagger.producers.Producer; 30*f585d8a3SJacky Wang import java.util.Map; 31*f585d8a3SJacky Wang 32*f585d8a3SJacky Wang /** 33*f585d8a3SJacky Wang * A {@link Producer} implementation used to implement {@link Map} bindings. This factory returns an 34*f585d8a3SJacky Wang * immediate future of {@code Map<K, Producer<V>>} when calling {@link #get}. 35*f585d8a3SJacky Wang */ 36*f585d8a3SJacky Wang public final class MapOfProducerProducer<K, V> extends AbstractMapProducer<K, V, Producer<V>> { 37*f585d8a3SJacky Wang /** Returns a new {@link Builder}. */ builder(int size)38*f585d8a3SJacky Wang public static <K, V> Builder<K, V> builder(int size) { 39*f585d8a3SJacky Wang return new Builder<>(size); 40*f585d8a3SJacky Wang } 41*f585d8a3SJacky Wang MapOfProducerProducer(ImmutableMap<K, Producer<V>> contributingMap)42*f585d8a3SJacky Wang private MapOfProducerProducer(ImmutableMap<K, Producer<V>> contributingMap) { 43*f585d8a3SJacky Wang super(contributingMap); 44*f585d8a3SJacky Wang } 45*f585d8a3SJacky Wang 46*f585d8a3SJacky Wang @Override compute()47*f585d8a3SJacky Wang public ListenableFuture<Map<K, Producer<V>>> compute() { 48*f585d8a3SJacky Wang return Futures.<Map<K, Producer<V>>>immediateFuture(contributingMap()); 49*f585d8a3SJacky Wang } 50*f585d8a3SJacky Wang 51*f585d8a3SJacky Wang /** A builder for {@link MapOfProducerProducer} */ 52*f585d8a3SJacky Wang public static final class Builder<K, V> extends AbstractMapProducer.Builder<K, V, Producer<V>> { Builder(int size)53*f585d8a3SJacky Wang private Builder(int size) { 54*f585d8a3SJacky Wang super(size); 55*f585d8a3SJacky Wang } 56*f585d8a3SJacky Wang 57*f585d8a3SJacky Wang @Override put(K key, Producer<V> producerOfValue)58*f585d8a3SJacky Wang public Builder<K, V> put(K key, Producer<V> producerOfValue) { 59*f585d8a3SJacky Wang super.put(key, producerOfValue); 60*f585d8a3SJacky Wang return this; 61*f585d8a3SJacky Wang } 62*f585d8a3SJacky Wang 63*f585d8a3SJacky Wang @Override put(K key, Provider<V> providerOfValue)64*f585d8a3SJacky Wang public Builder<K, V> put(K key, Provider<V> providerOfValue) { 65*f585d8a3SJacky Wang super.put(key, providerOfValue); 66*f585d8a3SJacky Wang return this; 67*f585d8a3SJacky Wang } 68*f585d8a3SJacky Wang 69*f585d8a3SJacky Wang /** 70*f585d8a3SJacky Wang * Legacy javax version of the method to support libraries compiled with an older version of 71*f585d8a3SJacky Wang * Dagger. Do not use directly. 72*f585d8a3SJacky Wang */ 73*f585d8a3SJacky Wang @Deprecated put(K key, javax.inject.Provider<V> providerOfValue)74*f585d8a3SJacky Wang public Builder<K, V> put(K key, javax.inject.Provider<V> providerOfValue) { 75*f585d8a3SJacky Wang return put(key, asDaggerProvider(providerOfValue)); 76*f585d8a3SJacky Wang } 77*f585d8a3SJacky Wang 78*f585d8a3SJacky Wang @Override putAll(Producer<Map<K, Producer<V>>> mapOfProducerProducer)79*f585d8a3SJacky Wang public Builder<K, V> putAll(Producer<Map<K, Producer<V>>> mapOfProducerProducer) { 80*f585d8a3SJacky Wang super.putAll(mapOfProducerProducer); 81*f585d8a3SJacky Wang return this; 82*f585d8a3SJacky Wang } 83*f585d8a3SJacky Wang 84*f585d8a3SJacky Wang /** Returns a new {@link MapOfProducerProducer}. */ build()85*f585d8a3SJacky Wang public MapOfProducerProducer<K, V> build() { 86*f585d8a3SJacky Wang return new MapOfProducerProducer<>(mapBuilder.build()); 87*f585d8a3SJacky Wang } 88*f585d8a3SJacky Wang } 89*f585d8a3SJacky Wang 90*f585d8a3SJacky Wang @Override newDependencyView()91*f585d8a3SJacky Wang public Producer<Map<K, Producer<V>>> newDependencyView() { 92*f585d8a3SJacky Wang return newTransformedValuesView(MapOfProducerProducer.<V>toDependencyView()); 93*f585d8a3SJacky Wang } 94*f585d8a3SJacky Wang 95*f585d8a3SJacky Wang @Override newEntryPointView( CancellationListener cancellationListener)96*f585d8a3SJacky Wang public Producer<Map<K, Producer<V>>> newEntryPointView( 97*f585d8a3SJacky Wang CancellationListener cancellationListener) { 98*f585d8a3SJacky Wang return newTransformedValuesView( 99*f585d8a3SJacky Wang MapOfProducerProducer.<V>toEntryPointView(cancellationListener)); 100*f585d8a3SJacky Wang } 101*f585d8a3SJacky Wang newTransformedValuesView( Function<Producer<V>, Producer<V>> valueTransformationFunction)102*f585d8a3SJacky Wang private Producer<Map<K, Producer<V>>> newTransformedValuesView( 103*f585d8a3SJacky Wang Function<Producer<V>, Producer<V>> valueTransformationFunction) { 104*f585d8a3SJacky Wang return dagger.producers.Producers.<Map<K, Producer<V>>>immediateProducer( 105*f585d8a3SJacky Wang ImmutableMap.copyOf(Maps.transformValues(contributingMap(), valueTransformationFunction))); 106*f585d8a3SJacky Wang } 107*f585d8a3SJacky Wang 108*f585d8a3SJacky Wang @SuppressWarnings("unchecked") toDependencyView()109*f585d8a3SJacky Wang private static <T> Function<Producer<T>, Producer<T>> toDependencyView() { 110*f585d8a3SJacky Wang return (Function) TO_DEPENDENCY_VIEW; 111*f585d8a3SJacky Wang } 112*f585d8a3SJacky Wang toEntryPointView( final CancellationListener cancellationListener)113*f585d8a3SJacky Wang private static <T> Function<Producer<T>, Producer<T>> toEntryPointView( 114*f585d8a3SJacky Wang final CancellationListener cancellationListener) { 115*f585d8a3SJacky Wang return new Function<Producer<T>, Producer<T>>() { 116*f585d8a3SJacky Wang @Override 117*f585d8a3SJacky Wang public Producer<T> apply(Producer<T> input) { 118*f585d8a3SJacky Wang return entryPointViewOf(input, cancellationListener); 119*f585d8a3SJacky Wang } 120*f585d8a3SJacky Wang }; 121*f585d8a3SJacky Wang } 122*f585d8a3SJacky Wang 123*f585d8a3SJacky Wang private static final Function<Producer<?>, Producer<?>> TO_DEPENDENCY_VIEW = 124*f585d8a3SJacky Wang new Function<Producer<?>, Producer<?>>() { 125*f585d8a3SJacky Wang @Override 126*f585d8a3SJacky Wang public Producer<?> apply(Producer<?> input) { 127*f585d8a3SJacky Wang return nonCancellationPropagatingViewOf(input); 128*f585d8a3SJacky Wang } 129*f585d8a3SJacky Wang }; 130*f585d8a3SJacky Wang } 131