1 /*
2  * Copyright (C) 2016 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 package com.android.incallui.incall.impl;
18 
19 import android.support.annotation.NonNull;
20 import android.support.v4.util.ArrayMap;
21 import android.util.ArraySet;
22 import com.android.dialer.common.Assert;
23 import com.android.incallui.incall.protocol.InCallButtonIds;
24 import com.android.incallui.incall.protocol.InCallButtonIdsExtension;
25 import com.google.auto.value.AutoValue;
26 import java.util.ArrayList;
27 import java.util.Collections;
28 import java.util.Comparator;
29 import java.util.List;
30 import java.util.Map;
31 import java.util.Map.Entry;
32 import java.util.Set;
33 import javax.annotation.concurrent.Immutable;
34 
35 /**
36  * Determines logical button slot and ordering based on a provided mapping.
37  *
38  * <p>The provided mapping is declared with the following pieces of information: key, the {@link
39  * InCallButtonIds} for which the mapping applies; {@link MappingInfo#getSlot()}, the arbitrarily
40  * indexed slot into which the InCallButtonId desires to be placed; {@link
41  * MappingInfo#getSlotOrder()}, the slotOrder, used to choose the correct InCallButtonId when
42  * multiple desire to be placed in the same slot; and {@link MappingInfo#getConflictOrder()}, the
43  * conflictOrder, used to determine the overall order for InCallButtonIds that weren't chosen for
44  * their desired slot.
45  */
46 @Immutable
47 final class MappedButtonConfig {
48 
49   @NonNull private final Map<Integer, MappingInfo> mapping;
50   @NonNull private final List<Integer> orderedMappedSlots;
51 
52   /**
53    * Creates this MappedButtonConfig with the given mapping of {@link InCallButtonIds} to their
54    * corresponding slots and order.
55    *
56    * @param mapping the mapping.
57    */
MappedButtonConfig(@onNull Map<Integer, MappingInfo> mapping)58   public MappedButtonConfig(@NonNull Map<Integer, MappingInfo> mapping) {
59     this.mapping = new ArrayMap<>();
60     this.mapping.putAll(Assert.isNotNull(mapping));
61     this.orderedMappedSlots = findOrderedMappedSlots();
62   }
63 
findOrderedMappedSlots()64   private List<Integer> findOrderedMappedSlots() {
65     Set<Integer> slots = new ArraySet<>();
66     for (Entry<Integer, MappingInfo> entry : mapping.entrySet()) {
67       slots.add(entry.getValue().getSlot());
68     }
69     List<Integer> orderedSlots = new ArrayList<>(slots);
70     Collections.sort(orderedSlots);
71     return orderedSlots;
72   }
73 
74   /** Returns an immutable list of the slots for which this class has button mapping. */
75   @NonNull
getOrderedMappedSlots()76   public List<Integer> getOrderedMappedSlots() {
77     if (mapping.isEmpty()) {
78       return Collections.emptyList();
79     }
80     return Collections.unmodifiableList(orderedMappedSlots);
81   }
82 
83   /**
84    * Returns a list of {@link InCallButtonIds} that are configured to be placed in the given ui
85    * slot. The slot can be based from any index, as long as it matches the provided mapping.
86    */
87   @NonNull
getButtonsForSlot(int slot)88   public List<Integer> getButtonsForSlot(int slot) {
89     List<Integer> buttons = new ArrayList<>();
90     for (Entry<Integer, MappingInfo> entry : mapping.entrySet()) {
91       if (entry.getValue().getSlot() == slot) {
92         buttons.add(entry.getKey());
93       }
94     }
95     return buttons;
96   }
97 
98   /**
99    * Returns a {@link Comparator} capable of ordering {@link InCallButtonIds} that are configured to
100    * be placed in the same slot. InCallButtonIds are sorted based on the natural ordering of {@link
101    * MappingInfo#getSlotOrder()}.
102    *
103    * <p>Note: the returned Comparator's compare method will throw an {@link
104    * IllegalArgumentException} if called with InCallButtonIds that have no configuration or are not
105    * to be placed in the same slot.
106    */
107   @NonNull
getSlotComparator()108   public Comparator<Integer> getSlotComparator() {
109     return new Comparator<Integer>() {
110       @Override
111       public int compare(Integer lhs, Integer rhs) {
112         MappingInfo lhsInfo = lookupMappingInfo(lhs);
113         MappingInfo rhsInfo = lookupMappingInfo(rhs);
114         if (lhsInfo.getSlot() != rhsInfo.getSlot()) {
115           throw new IllegalArgumentException("lhs and rhs don't go in the same slot");
116         }
117         return lhsInfo.getSlotOrder() - rhsInfo.getSlotOrder();
118       }
119     };
120   }
121 
122   /**
123    * Returns a {@link Comparator} capable of ordering {@link InCallButtonIds} by their conflict
124    * score. This comparator should be used when multiple InCallButtonIds could have been shown in
125    * the same slot. InCallButtonIds are sorted based on the natural ordering of {@link
126    * MappingInfo#getConflictOrder()}.
127    *
128    * <p>Note: the returned Comparator's compare method will throw an {@link
129    * IllegalArgumentException} if called with InCallButtonIds that have no configuration.
130    */
131   @NonNull
132   public Comparator<Integer> getConflictComparator() {
133     return new Comparator<Integer>() {
134       @Override
135       public int compare(Integer lhs, Integer rhs) {
136         MappingInfo lhsInfo = lookupMappingInfo(lhs);
137         MappingInfo rhsInfo = lookupMappingInfo(rhs);
138         return lhsInfo.getConflictOrder() - rhsInfo.getConflictOrder();
139       }
140     };
141   }
142 
143   @NonNull
144   public MappingInfo lookupMappingInfo(@InCallButtonIds int button) {
145     MappingInfo info = mapping.get(button);
146     if (info == null) {
147       throw new IllegalArgumentException(
148           "Unknown InCallButtonId: " + InCallButtonIdsExtension.toString(button));
149     }
150     return info;
151   }
152 
153   /** Holds information about button mapping. */
154   @AutoValue
155   abstract static class MappingInfo {
156 
157     public static final int NO_MUTUALLY_EXCLUSIVE_BUTTON_SET = -1;
158 
159     /** The Ui slot into which a given button desires to be placed. */
160     public abstract int getSlot();
161 
162     /**
163      * Returns an integer used to determine which button is chosen for a slot when multiple buttons
164      * desire to be placed in the same slot. Follows from the natural ordering of integers, i.e. a
165      * lower slotOrder results in the button being chosen.
166      */
167     public abstract int getSlotOrder();
168 
169     /**
170      * Returns an integer used to determine the order in which buttons that weren't chosen for their
171      * desired slot are placed into the Ui. Follows from the natural ordering of integers, i.e. a
172      * lower conflictOrder results in the button being chosen.
173      */
174     public abstract int getConflictOrder();
175 
176     /**
177      * Returns an integer representing a button for which the given button conflicts. Defaults to
178      * {@link NO_MUTUALLY_EXCLUSIVE_BUTTON_SET}.
179      *
180      * <p>If the mutually exclusive button is chosen, the associated button should never be chosen.
181      */
182     public abstract @InCallButtonIds int getMutuallyExclusiveButton();
183 
184     static Builder builder(int slot) {
185       return new AutoValue_MappedButtonConfig_MappingInfo.Builder()
186           .setSlot(slot)
187           .setSlotOrder(Integer.MAX_VALUE)
188           .setConflictOrder(Integer.MAX_VALUE)
189           .setMutuallyExclusiveButton(NO_MUTUALLY_EXCLUSIVE_BUTTON_SET);
190     }
191 
192     /** Class used to build instances of {@link MappingInfo}. */
193     @AutoValue.Builder
194     abstract static class Builder {
195       public abstract Builder setSlot(int slot);
196 
197       public abstract Builder setSlotOrder(int slotOrder);
198 
199       public abstract Builder setConflictOrder(int conflictOrder);
200 
201       public abstract Builder setMutuallyExclusiveButton(@InCallButtonIds int button);
202 
203       public abstract MappingInfo build();
204     }
205   }
206 }
207