blob: 79db544d126d70feb9f610a5486d3ab31ada84b5 [file] [log] [blame]
Jason Monk9c7844c2017-01-18 15:21:53 -05001/*
2 * Copyright (C) 2017 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file
5 * except in compliance with the License. You may obtain a copy of the License at
6 *
7 * http://www.apache.org/licenses/LICENSE-2.0
8 *
9 * Unless required by applicable law or agreed to in writing, software distributed under the
10 * License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
11 * KIND, either express or implied. See the License for the specific language governing
12 * permissions and limitations under the License.
13 */
14
15package com.android.systemui;
16
Jason Monkde850bb2017-02-01 19:26:30 -050017import android.content.Context;
Jason Monk9c7844c2017-01-18 15:21:53 -050018import android.content.res.Configuration;
19import android.os.Handler;
20import android.os.HandlerThread;
21import android.os.Looper;
22import android.os.Process;
23import android.util.ArrayMap;
24
25import com.android.internal.annotations.VisibleForTesting;
Christine Franks69c2d1d2017-01-23 14:45:29 -080026import com.android.internal.app.NightDisplayController;
Jason Monk8c09ac72017-03-16 11:53:40 -040027import com.android.internal.logging.MetricsLogger;
Adrian Roos09c43c82017-02-09 19:58:25 +010028import com.android.internal.util.Preconditions;
Jason Monk6a73e632017-03-17 11:08:30 -040029import com.android.settingslib.bluetooth.LocalBluetoothManager;
Jason Monk9c7844c2017-01-18 15:21:53 -050030import com.android.systemui.assist.AssistManager;
Jason Monk790442e2017-02-13 17:49:39 -050031import com.android.systemui.fragments.FragmentService;
Jason Monkec34da82017-02-24 15:57:05 -050032import com.android.systemui.plugins.ActivityStarter;
33import com.android.systemui.plugins.PluginDependencyProvider;
Jason Monkde850bb2017-02-01 19:26:30 -050034import com.android.systemui.plugins.PluginManager;
Jason Monk340b0e52017-03-08 14:57:56 -050035import com.android.systemui.plugins.PluginManagerImpl;
Jason Monk782cd672017-03-22 12:50:57 -040036import com.android.systemui.plugins.VolumeDialogController;
Jason Monkaa573e92017-01-27 17:00:29 -050037import com.android.systemui.statusbar.phone.ConfigurationControllerImpl;
38import com.android.systemui.statusbar.phone.DarkIconDispatcherImpl;
Jason Monk9c7844c2017-01-18 15:21:53 -050039import com.android.systemui.statusbar.phone.ManagedProfileController;
40import com.android.systemui.statusbar.phone.ManagedProfileControllerImpl;
Jason Monk421a9412017-02-06 09:15:21 -080041import com.android.systemui.statusbar.phone.StatusBarWindowManager;
Jason Monkaa573e92017-01-27 17:00:29 -050042import com.android.systemui.statusbar.phone.StatusBarIconController;
43import com.android.systemui.statusbar.phone.StatusBarIconControllerImpl;
Jason Monk9c7844c2017-01-18 15:21:53 -050044import com.android.systemui.statusbar.policy.AccessibilityController;
Jason Monk91e587e2017-04-13 13:41:23 -040045import com.android.systemui.statusbar.policy.AccessibilityManagerWrapper;
Jason Monk9c7844c2017-01-18 15:21:53 -050046import com.android.systemui.statusbar.policy.BatteryController;
47import com.android.systemui.statusbar.policy.BatteryControllerImpl;
48import com.android.systemui.statusbar.policy.BluetoothController;
49import com.android.systemui.statusbar.policy.BluetoothControllerImpl;
50import com.android.systemui.statusbar.policy.CastController;
51import com.android.systemui.statusbar.policy.CastControllerImpl;
Jason Monkaa573e92017-01-27 17:00:29 -050052import com.android.systemui.statusbar.policy.ConfigurationController;
53import com.android.systemui.statusbar.policy.DarkIconDispatcher;
Jason Monk9c7844c2017-01-18 15:21:53 -050054import com.android.systemui.statusbar.policy.DataSaverController;
55import com.android.systemui.statusbar.policy.DeviceProvisionedController;
56import com.android.systemui.statusbar.policy.DeviceProvisionedControllerImpl;
Jason Monk1d9632d2017-02-09 13:20:04 -080057import com.android.systemui.statusbar.policy.ExtensionController;
58import com.android.systemui.statusbar.policy.ExtensionControllerImpl;
Jason Monk9c7844c2017-01-18 15:21:53 -050059import com.android.systemui.statusbar.policy.FlashlightController;
60import com.android.systemui.statusbar.policy.FlashlightControllerImpl;
61import com.android.systemui.statusbar.policy.HotspotController;
62import com.android.systemui.statusbar.policy.HotspotControllerImpl;
63import com.android.systemui.statusbar.policy.KeyguardMonitor;
64import com.android.systemui.statusbar.policy.KeyguardMonitorImpl;
65import com.android.systemui.statusbar.policy.LocationController;
66import com.android.systemui.statusbar.policy.LocationControllerImpl;
67import com.android.systemui.statusbar.policy.NetworkController;
68import com.android.systemui.statusbar.policy.NetworkControllerImpl;
69import com.android.systemui.statusbar.policy.NextAlarmController;
70import com.android.systemui.statusbar.policy.NextAlarmControllerImpl;
71import com.android.systemui.statusbar.policy.RotationLockController;
72import com.android.systemui.statusbar.policy.RotationLockControllerImpl;
73import com.android.systemui.statusbar.policy.SecurityController;
74import com.android.systemui.statusbar.policy.SecurityControllerImpl;
75import com.android.systemui.statusbar.policy.UserInfoController;
76import com.android.systemui.statusbar.policy.UserInfoControllerImpl;
77import com.android.systemui.statusbar.policy.UserSwitcherController;
78import com.android.systemui.statusbar.policy.ZenModeController;
79import com.android.systemui.statusbar.policy.ZenModeControllerImpl;
Jason Monkd4afe152017-05-01 15:37:43 -040080import com.android.systemui.tuner.TunablePadding;
81import com.android.systemui.tuner.TunablePadding.TunablePaddingService;
Jason Monkde850bb2017-02-01 19:26:30 -050082import com.android.systemui.tuner.TunerService;
Jason Monk340b0e52017-03-08 14:57:56 -050083import com.android.systemui.tuner.TunerServiceImpl;
Adrian Roos91250682017-02-06 14:48:15 -080084import com.android.systemui.util.leak.GarbageMonitor;
Adrian Roose1e0b482017-02-02 16:00:59 -080085import com.android.systemui.util.leak.LeakDetector;
Adrian Roos91250682017-02-06 14:48:15 -080086import com.android.systemui.util.leak.LeakReporter;
Jason Monk782cd672017-03-22 12:50:57 -040087import com.android.systemui.volume.VolumeDialogControllerImpl;
Jason Monk9c7844c2017-01-18 15:21:53 -050088
Lucas Dupin314d41f2017-05-08 15:52:58 -070089import com.google.android.colorextraction.ColorExtractor;
90
Jason Monk9c7844c2017-01-18 15:21:53 -050091import java.io.FileDescriptor;
92import java.io.PrintWriter;
Jason Monkde850bb2017-02-01 19:26:30 -050093import java.util.HashMap;
Jason Monk790442e2017-02-13 17:49:39 -050094import java.util.function.Consumer;
Jason Monk9c7844c2017-01-18 15:21:53 -050095
96/**
97 * Class to handle ugly dependencies throughout sysui until we determine the
98 * long-term dependency injection solution.
99 *
100 * Classes added here should be things that are expected to live the lifetime of sysui,
101 * and are generally applicable to many parts of sysui. They will be lazily
102 * initialized to ensure they aren't created on form factors that don't need them
103 * (e.g. HotspotController on TV). Despite being lazily initialized, it is expected
104 * that all dependencies will be gotten during sysui startup, and not during runtime
105 * to avoid jank.
106 *
107 * All classes used here are expected to manage their own lifecycle, meaning if
108 * they have no clients they should not have any registered resources like bound
109 * services, registered receivers, etc.
110 */
111public class Dependency extends SystemUI {
112
113 /**
114 * Key for getting a background Looper for background work.
115 */
Adrian Roos09c43c82017-02-09 19:58:25 +0100116 public static final DependencyKey<Looper> BG_LOOPER = new DependencyKey<>("background_looper");
Jason Monk9c7844c2017-01-18 15:21:53 -0500117 /**
118 * Key for getting a Handler for receiving time tick broadcasts on.
119 */
Adrian Roos09c43c82017-02-09 19:58:25 +0100120 public static final DependencyKey<Handler> TIME_TICK_HANDLER =
121 new DependencyKey<>("time_tick_handler");
Jason Monk9c7844c2017-01-18 15:21:53 -0500122 /**
123 * Generic handler on the main thread.
124 */
Adrian Roos09c43c82017-02-09 19:58:25 +0100125 public static final DependencyKey<Handler> MAIN_HANDLER = new DependencyKey<>("main_handler");
Jason Monk9c7844c2017-01-18 15:21:53 -0500126
Adrian Roos91250682017-02-06 14:48:15 -0800127 /**
128 * An email address to send memory leak reports to by default.
129 */
130 public static final DependencyKey<String> LEAK_REPORT_EMAIL
131 = new DependencyKey<>("leak_report_email");
132
Adrian Roos09c43c82017-02-09 19:58:25 +0100133 private final ArrayMap<Object, Object> mDependencies = new ArrayMap<>();
134 private final ArrayMap<Object, DependencyProvider> mProviders = new ArrayMap<>();
Jason Monk9c7844c2017-01-18 15:21:53 -0500135
136 @Override
137 public void start() {
138 sDependency = this;
139 // TODO: Think about ways to push these creation rules out of Dependency to cut down
140 // on imports.
141 mProviders.put(TIME_TICK_HANDLER, () -> {
142 HandlerThread thread = new HandlerThread("TimeTick");
143 thread.start();
144 return new Handler(thread.getLooper());
145 });
146 mProviders.put(BG_LOOPER, () -> {
147 HandlerThread thread = new HandlerThread("SysUiBg",
148 Process.THREAD_PRIORITY_BACKGROUND);
149 thread.start();
150 return thread.getLooper();
151 });
152 mProviders.put(MAIN_HANDLER, () -> new Handler(Looper.getMainLooper()));
Adrian Roos09c43c82017-02-09 19:58:25 +0100153 mProviders.put(ActivityStarter.class, () -> new ActivityStarterDelegate());
154 mProviders.put(ActivityStarterDelegate.class, () ->
Jason Monk9c7844c2017-01-18 15:21:53 -0500155 getDependency(ActivityStarter.class));
156
Adrian Roos09c43c82017-02-09 19:58:25 +0100157 mProviders.put(BluetoothController.class, () ->
Jason Monk9c7844c2017-01-18 15:21:53 -0500158 new BluetoothControllerImpl(mContext, getDependency(BG_LOOPER)));
159
Adrian Roos09c43c82017-02-09 19:58:25 +0100160 mProviders.put(LocationController.class, () ->
Jason Monk9c7844c2017-01-18 15:21:53 -0500161 new LocationControllerImpl(mContext, getDependency(BG_LOOPER)));
162
Adrian Roos09c43c82017-02-09 19:58:25 +0100163 mProviders.put(RotationLockController.class, () ->
Jason Monk9c7844c2017-01-18 15:21:53 -0500164 new RotationLockControllerImpl(mContext));
165
Adrian Roos09c43c82017-02-09 19:58:25 +0100166 mProviders.put(NetworkController.class, () ->
Jason Monk9c7844c2017-01-18 15:21:53 -0500167 new NetworkControllerImpl(mContext, getDependency(BG_LOOPER),
168 getDependency(DeviceProvisionedController.class)));
169
Adrian Roos09c43c82017-02-09 19:58:25 +0100170 mProviders.put(ZenModeController.class, () ->
Jason Monk9c7844c2017-01-18 15:21:53 -0500171 new ZenModeControllerImpl(mContext, getDependency(MAIN_HANDLER)));
172
Adrian Roos09c43c82017-02-09 19:58:25 +0100173 mProviders.put(HotspotController.class, () ->
Jason Monk9c7844c2017-01-18 15:21:53 -0500174 new HotspotControllerImpl(mContext));
175
Adrian Roos09c43c82017-02-09 19:58:25 +0100176 mProviders.put(CastController.class, () ->
Jason Monk9c7844c2017-01-18 15:21:53 -0500177 new CastControllerImpl(mContext));
178
Adrian Roos09c43c82017-02-09 19:58:25 +0100179 mProviders.put(FlashlightController.class, () ->
Jason Monk9c7844c2017-01-18 15:21:53 -0500180 new FlashlightControllerImpl(mContext));
181
Adrian Roos09c43c82017-02-09 19:58:25 +0100182 mProviders.put(KeyguardMonitor.class, () ->
Jason Monk9c7844c2017-01-18 15:21:53 -0500183 new KeyguardMonitorImpl(mContext));
184
Adrian Roos09c43c82017-02-09 19:58:25 +0100185 mProviders.put(UserSwitcherController.class, () ->
Jason Monk9c7844c2017-01-18 15:21:53 -0500186 new UserSwitcherController(mContext, getDependency(KeyguardMonitor.class),
187 getDependency(MAIN_HANDLER), getDependency(ActivityStarter.class)));
188
Adrian Roos09c43c82017-02-09 19:58:25 +0100189 mProviders.put(UserInfoController.class, () ->
Jason Monk9c7844c2017-01-18 15:21:53 -0500190 new UserInfoControllerImpl(mContext));
191
Adrian Roos09c43c82017-02-09 19:58:25 +0100192 mProviders.put(BatteryController.class, () ->
Jason Monk9c7844c2017-01-18 15:21:53 -0500193 new BatteryControllerImpl(mContext));
194
Christine Franks69c2d1d2017-01-23 14:45:29 -0800195 mProviders.put(NightDisplayController.class, () ->
196 new NightDisplayController(mContext));
197
Adrian Roos09c43c82017-02-09 19:58:25 +0100198 mProviders.put(ManagedProfileController.class, () ->
Jason Monk9c7844c2017-01-18 15:21:53 -0500199 new ManagedProfileControllerImpl(mContext));
200
Adrian Roos09c43c82017-02-09 19:58:25 +0100201 mProviders.put(NextAlarmController.class, () ->
Jason Monk9c7844c2017-01-18 15:21:53 -0500202 new NextAlarmControllerImpl(mContext));
203
Adrian Roos09c43c82017-02-09 19:58:25 +0100204 mProviders.put(DataSaverController.class, () ->
Jason Monk9c7844c2017-01-18 15:21:53 -0500205 get(NetworkController.class).getDataSaverController());
206
Adrian Roos09c43c82017-02-09 19:58:25 +0100207 mProviders.put(AccessibilityController.class, () ->
Jason Monk9c7844c2017-01-18 15:21:53 -0500208 new AccessibilityController(mContext));
209
Adrian Roos09c43c82017-02-09 19:58:25 +0100210 mProviders.put(DeviceProvisionedController.class, () ->
Jason Monk9c7844c2017-01-18 15:21:53 -0500211 new DeviceProvisionedControllerImpl(mContext));
212
Adrian Roos09c43c82017-02-09 19:58:25 +0100213 mProviders.put(PluginManager.class, () ->
Jason Monk340b0e52017-03-08 14:57:56 -0500214 new PluginManagerImpl(mContext));
Jason Monkde850bb2017-02-01 19:26:30 -0500215
Adrian Roos09c43c82017-02-09 19:58:25 +0100216 mProviders.put(AssistManager.class, () ->
Jason Monk9c7844c2017-01-18 15:21:53 -0500217 new AssistManager(getDependency(DeviceProvisionedController.class), mContext));
218
Adrian Roos09c43c82017-02-09 19:58:25 +0100219 mProviders.put(SecurityController.class, () ->
Jason Monk9c7844c2017-01-18 15:21:53 -0500220 new SecurityControllerImpl(mContext));
221
Adrian Roos09c43c82017-02-09 19:58:25 +0100222 mProviders.put(LeakDetector.class, LeakDetector::create);
Adrian Roose1e0b482017-02-02 16:00:59 -0800223
Adrian Roos91250682017-02-06 14:48:15 -0800224 mProviders.put(LEAK_REPORT_EMAIL, () -> null);
225
226 mProviders.put(LeakReporter.class, () -> new LeakReporter(
227 mContext,
228 getDependency(LeakDetector.class),
229 getDependency(LEAK_REPORT_EMAIL)));
230
231 mProviders.put(GarbageMonitor.class, () -> new GarbageMonitor(
232 getDependency(BG_LOOPER),
233 getDependency(LeakDetector.class),
234 getDependency(LeakReporter.class)));
235
Adrian Roos09c43c82017-02-09 19:58:25 +0100236 mProviders.put(TunerService.class, () ->
Jason Monk340b0e52017-03-08 14:57:56 -0500237 new TunerServiceImpl(mContext));
Jason Monkde850bb2017-02-01 19:26:30 -0500238
Adrian Roos09c43c82017-02-09 19:58:25 +0100239 mProviders.put(StatusBarWindowManager.class, () ->
Jason Monk421a9412017-02-06 09:15:21 -0800240 new StatusBarWindowManager(mContext));
241
Adrian Roos09c43c82017-02-09 19:58:25 +0100242 mProviders.put(DarkIconDispatcher.class, () ->
Jason Monkaa573e92017-01-27 17:00:29 -0500243 new DarkIconDispatcherImpl(mContext));
244
Adrian Roos09c43c82017-02-09 19:58:25 +0100245 mProviders.put(ConfigurationController.class, () ->
Jason Monkaa573e92017-01-27 17:00:29 -0500246 new ConfigurationControllerImpl(mContext));
247
Adrian Roos09c43c82017-02-09 19:58:25 +0100248 mProviders.put(StatusBarIconController.class, () ->
Jason Monkaa573e92017-01-27 17:00:29 -0500249 new StatusBarIconControllerImpl(mContext));
250
Jason Monk790442e2017-02-13 17:49:39 -0500251 mProviders.put(FragmentService.class, () ->
252 new FragmentService(mContext));
253
Jason Monk1d9632d2017-02-09 13:20:04 -0800254 mProviders.put(ExtensionController.class, () ->
Jason Monk0b80c4e2017-05-01 15:38:34 -0400255 new ExtensionControllerImpl(mContext));
Jason Monk1d9632d2017-02-09 13:20:04 -0800256
Jason Monkec34da82017-02-24 15:57:05 -0500257 mProviders.put(PluginDependencyProvider.class, () ->
258 new PluginDependencyProvider(get(PluginManager.class)));
259
Jason Monk6a73e632017-03-17 11:08:30 -0400260 mProviders.put(LocalBluetoothManager.class, () ->
261 LocalBluetoothManager.getInstance(mContext, null));
262
Jason Monk782cd672017-03-22 12:50:57 -0400263 mProviders.put(VolumeDialogController.class, () ->
264 new VolumeDialogControllerImpl(mContext));
265
Jason Monk8c09ac72017-03-16 11:53:40 -0400266 mProviders.put(MetricsLogger.class, () -> new MetricsLogger());
267
Jason Monk91e587e2017-04-13 13:41:23 -0400268 mProviders.put(AccessibilityManagerWrapper.class,
269 () -> new AccessibilityManagerWrapper(mContext));
270
Lucas Dupin314d41f2017-05-08 15:52:58 -0700271 mProviders.put(ColorExtractor.class, () -> new ColorExtractor(mContext));
272
Jason Monkd4afe152017-05-01 15:37:43 -0400273 mProviders.put(TunablePaddingService.class, () -> new TunablePaddingService());
274
Jason Monk9c7844c2017-01-18 15:21:53 -0500275 // Put all dependencies above here so the factory can override them if it wants.
276 SystemUIFactory.getInstance().injectDependencies(mProviders, mContext);
277 }
278
279 @Override
Jason Monke1072022017-04-06 15:14:18 -0400280 public synchronized void dump(FileDescriptor fd, PrintWriter pw, String[] args) {
Jason Monk9c7844c2017-01-18 15:21:53 -0500281 super.dump(fd, pw, args);
282 pw.println("Dumping existing controllers:");
283 mDependencies.values().stream().filter(obj -> obj instanceof Dumpable)
284 .forEach(o -> ((Dumpable) o).dump(fd, pw, args));
285 }
286
287 @Override
Jason Monke1072022017-04-06 15:14:18 -0400288 protected synchronized void onConfigurationChanged(Configuration newConfig) {
Jason Monk9c7844c2017-01-18 15:21:53 -0500289 super.onConfigurationChanged(newConfig);
290 mDependencies.values().stream().filter(obj -> obj instanceof ConfigurationChangedReceiver)
291 .forEach(o -> ((ConfigurationChangedReceiver) o).onConfigurationChanged(newConfig));
292 }
293
294 protected final <T> T getDependency(Class<T> cls) {
Adrian Roos09c43c82017-02-09 19:58:25 +0100295 return getDependencyInner(cls);
Jason Monk9c7844c2017-01-18 15:21:53 -0500296 }
297
Adrian Roos09c43c82017-02-09 19:58:25 +0100298 protected final <T> T getDependency(DependencyKey<T> key) {
299 return getDependencyInner(key);
300 }
301
Jason Monke1072022017-04-06 15:14:18 -0400302 private synchronized <T> T getDependencyInner(Object key) {
Adrian Roos09c43c82017-02-09 19:58:25 +0100303 @SuppressWarnings("unchecked")
304 T obj = (T) mDependencies.get(key);
Jason Monk9c7844c2017-01-18 15:21:53 -0500305 if (obj == null) {
Adrian Roos09c43c82017-02-09 19:58:25 +0100306 obj = createDependency(key);
307 mDependencies.put(key, obj);
Jason Monk9c7844c2017-01-18 15:21:53 -0500308 }
309 return obj;
310 }
311
312 @VisibleForTesting
Adrian Roos09c43c82017-02-09 19:58:25 +0100313 protected <T> T createDependency(Object cls) {
314 Preconditions.checkArgument(cls instanceof DependencyKey<?> || cls instanceof Class<?>);
315
316 @SuppressWarnings("unchecked")
Jason Monk9c7844c2017-01-18 15:21:53 -0500317 DependencyProvider<T> provider = mProviders.get(cls);
318 if (provider == null) {
319 throw new IllegalArgumentException("Unsupported dependency " + cls);
320 }
321 return provider.createDependency();
322 }
323
324 private static Dependency sDependency;
325
326 public interface DependencyProvider<T> {
327 T createDependency();
328 }
329
Jason Monk790442e2017-02-13 17:49:39 -0500330 private <T> void destroyDependency(Class<T> cls, Consumer<T> destroy) {
331 T dep = (T) mDependencies.remove(cls);
332 if (dep != null && destroy != null) {
333 destroy.accept(dep);
334 }
335 }
336
Jason Monkde850bb2017-02-01 19:26:30 -0500337 /**
338 * Used in separate processes (like tuner settings) to init the dependencies.
339 */
340 public static void initDependencies(Context context) {
341 if (sDependency != null) return;
342 Dependency d = new Dependency();
Jason Monk790442e2017-02-13 17:49:39 -0500343 d.mContext = context;
Jason Monkde850bb2017-02-01 19:26:30 -0500344 d.mComponents = new HashMap<>();
345 d.start();
346 }
347
Jason Monk790442e2017-02-13 17:49:39 -0500348 /**
349 * Used in separate process teardown to ensure the context isn't leaked.
350 *
351 * TODO: Remove once PreferenceFragment doesn't reference getActivity()
352 * anymore and these context hacks are no longer needed.
353 */
354 public static void clearDependencies() {
355 sDependency = null;
356 }
357
358 /**
359 * Checks to see if a dependency is instantiated, if it is it removes it from
360 * the cache and calls the destroy callback.
361 */
362 public static <T> void destroy(Class<T> cls, Consumer<T> destroy) {
363 sDependency.destroyDependency(cls, destroy);
364 }
365
Jason Monk9c7844c2017-01-18 15:21:53 -0500366 public static <T> T get(Class<T> cls) {
Adrian Roos09c43c82017-02-09 19:58:25 +0100367 return sDependency.getDependency(cls);
Jason Monk9c7844c2017-01-18 15:21:53 -0500368 }
369
Adrian Roos09c43c82017-02-09 19:58:25 +0100370 public static <T> T get(DependencyKey<T> cls) {
Jason Monk9c7844c2017-01-18 15:21:53 -0500371 return sDependency.getDependency(cls);
372 }
Adrian Roos09c43c82017-02-09 19:58:25 +0100373
374 public static final class DependencyKey<V> {
375 private final String mDisplayName;
376
377 public DependencyKey(String displayName) {
378 mDisplayName = displayName;
379 }
380
381 @Override
382 public String toString() {
383 return mDisplayName;
384 }
385 }
Jason Monk9c7844c2017-01-18 15:21:53 -0500386}