1 /*
2  * Copyright (C) 2018 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.systemui.statusbar.phone;
18 
19 import android.content.Context;
20 import android.os.Handler;
21 import android.telephony.SubscriptionInfo;
22 import android.util.ArraySet;
23 import android.util.Log;
24 
25 import com.android.systemui.Dependency;
26 import com.android.systemui.R;
27 import com.android.systemui.statusbar.policy.NetworkController;
28 import com.android.systemui.statusbar.policy.NetworkController.IconState;
29 import com.android.systemui.statusbar.policy.NetworkControllerImpl;
30 import com.android.systemui.statusbar.policy.SecurityController;
31 import com.android.systemui.tuner.TunerService;
32 import com.android.systemui.tuner.TunerService.Tunable;
33 
34 import java.util.ArrayList;
35 import java.util.List;
36 import java.util.Objects;
37 
38 
39 public class StatusBarSignalPolicy implements NetworkControllerImpl.SignalCallback,
40         SecurityController.SecurityControllerCallback, Tunable {
41     private static final String TAG = "StatusBarSignalPolicy";
42 
43     private final String mSlotAirplane;
44     private final String mSlotMobile;
45     private final String mSlotWifi;
46     private final String mSlotEthernet;
47     private final String mSlotVpn;
48 
49     private final Context mContext;
50     private final StatusBarIconController mIconController;
51     private final NetworkController mNetworkController;
52     private final SecurityController mSecurityController;
53     private final Handler mHandler = Handler.getMain();
54 
55     private boolean mBlockAirplane;
56     private boolean mBlockMobile;
57     private boolean mBlockWifi;
58     private boolean mBlockEthernet;
59     private boolean mActivityEnabled;
60     private boolean mForceBlockWifi;
61 
62     // Track as little state as possible, and only for padding purposes
63     private boolean mIsAirplaneMode = false;
64     private boolean mWifiVisible = false;
65 
66     private ArrayList<MobileIconState> mMobileStates = new ArrayList<MobileIconState>();
67     private WifiIconState mWifiIconState = new WifiIconState();
68 
StatusBarSignalPolicy(Context context, StatusBarIconController iconController)69     public StatusBarSignalPolicy(Context context, StatusBarIconController iconController) {
70         mContext = context;
71 
72         mSlotAirplane = mContext.getString(com.android.internal.R.string.status_bar_airplane);
73         mSlotMobile   = mContext.getString(com.android.internal.R.string.status_bar_mobile);
74         mSlotWifi     = mContext.getString(com.android.internal.R.string.status_bar_wifi);
75         mSlotEthernet = mContext.getString(com.android.internal.R.string.status_bar_ethernet);
76         mSlotVpn      = mContext.getString(com.android.internal.R.string.status_bar_vpn);
77         mActivityEnabled = mContext.getResources().getBoolean(R.bool.config_showActivity);
78 
79         mIconController = iconController;
80         mNetworkController = Dependency.get(NetworkController.class);
81         mSecurityController = Dependency.get(SecurityController.class);
82 
83         Dependency.get(TunerService.class).addTunable(this, StatusBarIconController.ICON_BLACKLIST);
84         mNetworkController.addCallback(this);
85         mSecurityController.addCallback(this);
86     }
87 
destroy()88     public void destroy() {
89         Dependency.get(TunerService.class).removeTunable(this);
90         mNetworkController.removeCallback(this);
91         mSecurityController.removeCallback(this);
92     }
93 
updateVpn()94     private void updateVpn() {
95         boolean vpnVisible = mSecurityController.isVpnEnabled();
96         int vpnIconId = currentVpnIconId(mSecurityController.isVpnBranded());
97 
98         mIconController.setIcon(mSlotVpn, vpnIconId,
99                 mContext.getResources().getString(R.string.accessibility_vpn_on));
100         mIconController.setIconVisibility(mSlotVpn, vpnVisible);
101     }
102 
currentVpnIconId(boolean isBranded)103     private int currentVpnIconId(boolean isBranded) {
104         return isBranded ? R.drawable.stat_sys_branded_vpn : R.drawable.stat_sys_vpn_ic;
105     }
106 
107     /**
108      * From SecurityController
109      */
110     @Override
onStateChanged()111     public void onStateChanged() {
112         mHandler.post(this::updateVpn);
113     }
114 
115     @Override
onTuningChanged(String key, String newValue)116     public void onTuningChanged(String key, String newValue) {
117         if (!StatusBarIconController.ICON_BLACKLIST.equals(key)) {
118             return;
119         }
120         ArraySet<String> blockList = StatusBarIconController.getIconBlacklist(mContext, newValue);
121         boolean blockAirplane = blockList.contains(mSlotAirplane);
122         boolean blockMobile = blockList.contains(mSlotMobile);
123         boolean blockWifi = blockList.contains(mSlotWifi);
124         boolean blockEthernet = blockList.contains(mSlotEthernet);
125 
126         if (blockAirplane != mBlockAirplane || blockMobile != mBlockMobile
127                 || blockEthernet != mBlockEthernet || blockWifi != mBlockWifi) {
128             mBlockAirplane = blockAirplane;
129             mBlockMobile = blockMobile;
130             mBlockEthernet = blockEthernet;
131             mBlockWifi = blockWifi || mForceBlockWifi;
132             // Re-register to get new callbacks.
133             mNetworkController.removeCallback(this);
134             mNetworkController.addCallback(this);
135         }
136     }
137 
138     @Override
setWifiIndicators(boolean enabled, IconState statusIcon, IconState qsIcon, boolean activityIn, boolean activityOut, String description, boolean isTransient, String statusLabel)139     public void setWifiIndicators(boolean enabled, IconState statusIcon, IconState qsIcon,
140             boolean activityIn, boolean activityOut, String description, boolean isTransient,
141             String statusLabel) {
142 
143         boolean visible = statusIcon.visible && !mBlockWifi;
144         boolean in = activityIn && mActivityEnabled && visible;
145         boolean out = activityOut && mActivityEnabled && visible;
146 
147         WifiIconState newState = mWifiIconState.copy();
148 
149         newState.visible = visible;
150         newState.resId = statusIcon.icon;
151         newState.activityIn = in;
152         newState.activityOut = out;
153         newState.slot = mSlotWifi;
154         newState.airplaneSpacerVisible = mIsAirplaneMode;
155         newState.contentDescription = statusIcon.contentDescription;
156 
157         MobileIconState first = getFirstMobileState();
158         newState.signalSpacerVisible = first != null && first.typeId != 0;
159 
160         updateWifiIconWithState(newState);
161         mWifiIconState = newState;
162     }
163 
updateShowWifiSignalSpacer(WifiIconState state)164     private void updateShowWifiSignalSpacer(WifiIconState state) {
165         MobileIconState first = getFirstMobileState();
166         state.signalSpacerVisible = first != null && first.typeId != 0;
167     }
168 
updateWifiIconWithState(WifiIconState state)169     private void updateWifiIconWithState(WifiIconState state) {
170         if (state.visible && state.resId > 0) {
171             mIconController.setSignalIcon(mSlotWifi, state);
172             mIconController.setIconVisibility(mSlotWifi, true);
173         } else {
174             mIconController.setIconVisibility(mSlotWifi, false);
175         }
176     }
177 
178     @Override
setMobileDataIndicators(IconState statusIcon, IconState qsIcon, int statusType, int qsType, boolean activityIn, boolean activityOut, CharSequence typeContentDescription, CharSequence typeContentDescriptionHtml, CharSequence description, boolean isWide, int subId, boolean roaming)179     public void setMobileDataIndicators(IconState statusIcon, IconState qsIcon, int statusType,
180             int qsType, boolean activityIn, boolean activityOut,
181             CharSequence typeContentDescription,
182             CharSequence typeContentDescriptionHtml, CharSequence description,
183             boolean isWide, int subId, boolean roaming) {
184         MobileIconState state = getState(subId);
185         if (state == null) {
186             return;
187         }
188 
189         // Visibility of the data type indicator changed
190         boolean typeChanged = statusType != state.typeId && (statusType == 0 || state.typeId == 0);
191 
192         state.visible = statusIcon.visible && !mBlockMobile;
193         state.strengthId = statusIcon.icon;
194         state.typeId = statusType;
195         state.contentDescription = statusIcon.contentDescription;
196         state.typeContentDescription = typeContentDescription;
197         state.roaming = roaming;
198         state.activityIn = activityIn && mActivityEnabled;
199         state.activityOut = activityOut && mActivityEnabled;
200 
201         // Always send a copy to maintain value type semantics
202         mIconController.setMobileIcons(mSlotMobile, MobileIconState.copyStates(mMobileStates));
203 
204         if (typeChanged) {
205             WifiIconState wifiCopy = mWifiIconState.copy();
206             updateShowWifiSignalSpacer(wifiCopy);
207             if (!Objects.equals(wifiCopy, mWifiIconState)) {
208                 updateWifiIconWithState(wifiCopy);
209                 mWifiIconState = wifiCopy;
210             }
211         }
212     }
213 
getState(int subId)214     private MobileIconState getState(int subId) {
215         for (MobileIconState state : mMobileStates) {
216             if (state.subId == subId) {
217                 return state;
218             }
219         }
220         Log.e(TAG, "Unexpected subscription " + subId);
221         return null;
222     }
223 
getFirstMobileState()224     private MobileIconState getFirstMobileState() {
225         if (mMobileStates.size() > 0) {
226             return mMobileStates.get(0);
227         }
228 
229         return null;
230     }
231 
232 
233     /**
234      * It is expected that a call to setSubs will be immediately followed by setMobileDataIndicators
235      * so we don't have to update the icon manager at this point, just remove the old ones
236      * @param subs list of mobile subscriptions, displayed as mobile data indicators (max 8)
237      */
238     @Override
setSubs(List<SubscriptionInfo> subs)239     public void setSubs(List<SubscriptionInfo> subs) {
240         if (hasCorrectSubs(subs)) {
241             return;
242         }
243 
244         mIconController.removeAllIconsForSlot(mSlotMobile);
245         mMobileStates.clear();
246         final int n = subs.size();
247         for (int i = 0; i < n; i++) {
248             mMobileStates.add(new MobileIconState(subs.get(i).getSubscriptionId()));
249         }
250     }
251 
hasCorrectSubs(List<SubscriptionInfo> subs)252     private boolean hasCorrectSubs(List<SubscriptionInfo> subs) {
253         final int N = subs.size();
254         if (N != mMobileStates.size()) {
255             return false;
256         }
257         for (int i = 0; i < N; i++) {
258             if (mMobileStates.get(i).subId != subs.get(i).getSubscriptionId()) {
259                 return false;
260             }
261         }
262         return true;
263     }
264 
265     @Override
setNoSims(boolean show, boolean simDetected)266     public void setNoSims(boolean show, boolean simDetected) {
267         // Noop yay!
268     }
269 
270 
271     @Override
setEthernetIndicators(IconState state)272     public void setEthernetIndicators(IconState state) {
273         boolean visible = state.visible && !mBlockEthernet;
274         int resId = state.icon;
275         String description = state.contentDescription;
276 
277         if (resId > 0) {
278             mIconController.setIcon(mSlotEthernet, resId, description);
279             mIconController.setIconVisibility(mSlotEthernet, true);
280         } else {
281             mIconController.setIconVisibility(mSlotEthernet, false);
282         }
283     }
284 
285     @Override
setIsAirplaneMode(IconState icon)286     public void setIsAirplaneMode(IconState icon) {
287         mIsAirplaneMode = icon.visible && !mBlockAirplane;
288         int resId = icon.icon;
289         String description = icon.contentDescription;
290 
291         if (mIsAirplaneMode && resId > 0) {
292             mIconController.setIcon(mSlotAirplane, resId, description);
293             mIconController.setIconVisibility(mSlotAirplane, true);
294         } else {
295             mIconController.setIconVisibility(mSlotAirplane, false);
296         }
297     }
298 
299     @Override
setMobileDataEnabled(boolean enabled)300     public void setMobileDataEnabled(boolean enabled) {
301         // Don't care.
302     }
303 
304     private static abstract class SignalIconState {
305         public boolean visible;
306         public boolean activityOut;
307         public boolean activityIn;
308         public String slot;
309         public String contentDescription;
310 
311         @Override
equals(Object o)312         public boolean equals(Object o) {
313             // Skipping reference equality bc this should be more of a value type
314             if (o == null || getClass() != o.getClass()) {
315                 return false;
316             }
317             SignalIconState that = (SignalIconState) o;
318             return visible == that.visible &&
319                     activityOut == that.activityOut &&
320                     activityIn == that.activityIn &&
321                     Objects.equals(contentDescription, that.contentDescription) &&
322                     Objects.equals(slot, that.slot);
323         }
324 
325         @Override
hashCode()326         public int hashCode() {
327             return Objects.hash(visible, activityOut, slot);
328         }
329 
copyTo(SignalIconState other)330         protected void copyTo(SignalIconState other) {
331             other.visible = visible;
332             other.activityIn = activityIn;
333             other.activityOut = activityOut;
334             other.slot = slot;
335             other.contentDescription = contentDescription;
336         }
337     }
338 
339     public static class WifiIconState extends SignalIconState{
340         public int resId;
341         public boolean airplaneSpacerVisible;
342         public boolean signalSpacerVisible;
343 
344         @Override
equals(Object o)345         public boolean equals(Object o) {
346             // Skipping reference equality bc this should be more of a value type
347             if (o == null || getClass() != o.getClass()) {
348                 return false;
349             }
350             if (!super.equals(o)) {
351                 return false;
352             }
353             WifiIconState that = (WifiIconState) o;
354             return resId == that.resId &&
355                     airplaneSpacerVisible == that.airplaneSpacerVisible &&
356                     signalSpacerVisible == that.signalSpacerVisible;
357         }
358 
copyTo(WifiIconState other)359         public void copyTo(WifiIconState other) {
360             super.copyTo(other);
361             other.resId = resId;
362             other.airplaneSpacerVisible = airplaneSpacerVisible;
363             other.signalSpacerVisible = signalSpacerVisible;
364         }
365 
copy()366         public WifiIconState copy() {
367             WifiIconState newState = new WifiIconState();
368             copyTo(newState);
369             return newState;
370         }
371 
372         @Override
hashCode()373         public int hashCode() {
374             return Objects.hash(super.hashCode(),
375                     resId, airplaneSpacerVisible, signalSpacerVisible);
376         }
377 
toString()378         @Override public String toString() {
379             return "WifiIconState(resId=" + resId + ", visible=" + visible + ")";
380         }
381     }
382 
383     /**
384      * A little different. This one delegates to SignalDrawable instead of a specific resId
385      */
386     public static class MobileIconState extends SignalIconState {
387         public int subId;
388         public int strengthId;
389         public int typeId;
390         public boolean roaming;
391         public boolean needsLeadingPadding;
392         public CharSequence typeContentDescription;
393 
MobileIconState(int subId)394         private MobileIconState(int subId) {
395             super();
396             this.subId = subId;
397         }
398 
399         @Override
equals(Object o)400         public boolean equals(Object o) {
401             if (o == null || getClass() != o.getClass()) {
402                 return false;
403             }
404             if (!super.equals(o)) {
405                 return false;
406             }
407             MobileIconState that = (MobileIconState) o;
408             return subId == that.subId &&
409                     strengthId == that.strengthId &&
410                     typeId == that.typeId &&
411                     roaming == that.roaming &&
412                     needsLeadingPadding == that.needsLeadingPadding &&
413                     Objects.equals(typeContentDescription, that.typeContentDescription);
414         }
415 
416         @Override
hashCode()417         public int hashCode() {
418 
419             return Objects
420                     .hash(super.hashCode(), subId, strengthId, typeId, roaming, needsLeadingPadding,
421                             typeContentDescription);
422         }
423 
copy()424         public MobileIconState copy() {
425             MobileIconState copy = new MobileIconState(this.subId);
426             copyTo(copy);
427             return copy;
428         }
429 
copyTo(MobileIconState other)430         public void copyTo(MobileIconState other) {
431             super.copyTo(other);
432             other.subId = subId;
433             other.strengthId = strengthId;
434             other.typeId = typeId;
435             other.roaming = roaming;
436             other.needsLeadingPadding = needsLeadingPadding;
437             other.typeContentDescription = typeContentDescription;
438         }
439 
copyStates(List<MobileIconState> inStates)440         private static List<MobileIconState> copyStates(List<MobileIconState> inStates) {
441             ArrayList<MobileIconState> outStates = new ArrayList<>();
442             for (MobileIconState state : inStates) {
443                 MobileIconState copy = new MobileIconState(state.subId);
444                 state.copyTo(copy);
445                 outStates.add(copy);
446             }
447 
448             return outStates;
449         }
450 
toString()451         @Override public String toString() {
452             return "MobileIconState(subId=" + subId + ", strengthId=" + strengthId + ", roaming="
453                     + roaming + ", typeId=" + typeId + ", visible=" + visible + ")";
454         }
455     }
456 }
457