blob: 5989e4c6a266cd23dd24c8dbba1e58aba6d75dfd [file] [log] [blame]
Daniel Sandlercc8befa2013-06-11 14:45:48 -04001/*
2 * Copyright (C) 2013 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
17package com.android.launcher3;
18
kholoud mohamedd5a4d602022-04-28 15:55:40 +010019import static android.app.admin.DevicePolicyManager.ACTION_DEVICE_POLICY_RESOURCE_UPDATED;
Sunny Goyalb02dafc2023-06-22 11:17:29 -070020import static android.content.Context.RECEIVER_EXPORTED;
kholoud mohamedd5a4d602022-04-28 15:55:40 +010021
fbaron9d08e0f2024-04-24 11:26:01 -070022import static com.android.launcher3.Flags.enableSmartspaceRemovalToggle;
Charlie Andersonbad2be42024-12-06 16:32:03 -050023import static com.android.launcher3.InvariantDeviceProfile.GRID_NAME_PREFS_KEY;
24import static com.android.launcher3.LauncherPrefs.DB_FILE;
25import static com.android.launcher3.LauncherPrefs.GRID_NAME;
Stefan Andoniand1b33b32022-12-16 21:22:27 +000026import static com.android.launcher3.LauncherPrefs.ICON_STATE;
27import static com.android.launcher3.LauncherPrefs.THEMED_ICONS;
Charlie Andersonbad2be42024-12-06 16:32:03 -050028import static com.android.launcher3.model.DeviceGridState.KEY_DB_FILE;
fbaronc0e687b2023-10-04 12:11:20 -070029import static com.android.launcher3.model.LoaderTask.SMARTSPACE_ON_HOME_SCREEN;
Sunny Goyal045b4fa2019-09-20 12:51:37 -070030import static com.android.launcher3.util.Executors.MODEL_EXECUTOR;
Fengjiang Lib87ad6f2024-07-09 10:15:43 -070031import static com.android.launcher3.util.Executors.UI_HELPER_EXECUTOR;
Jon Mirandaec1277e2021-03-25 10:41:54 -040032import static com.android.launcher3.util.SettingsCache.NOTIFICATION_BADGING_URI;
Himanshu Gupta739b3c92023-11-06 17:47:29 +000033import static com.android.launcher3.util.SettingsCache.PRIVATE_SPACE_HIDE_WHEN_LOCKED_URI;
Sunny Goyald0e360a2018-06-29 14:40:18 -070034
Tonyd48710c2017-07-27 23:23:58 -070035import android.content.ComponentName;
Sunny Goyale755d462014-07-22 13:48:29 -070036import android.content.Context;
Sunny Goyal01c32a62021-05-17 13:23:08 -070037import android.content.SharedPreferences;
38import android.content.SharedPreferences.OnSharedPreferenceChangeListener;
Sunny Goyale7b00122019-10-02 16:13:34 -070039import android.content.pm.LauncherApps;
Jakob Schneider94b5f172024-01-31 19:17:10 +000040import android.content.pm.LauncherApps.ArchiveCompatibilityParams;
Sunny Goyalb47172b2021-05-03 19:59:51 -070041import android.os.UserHandle;
Daniel Sandlerb9eb2862013-06-14 20:17:30 -040042import android.util.Log;
Sunny Goyal71b5c0b2015-01-08 16:59:04 -080043
Tracy Zhoube13d102020-01-12 01:07:59 -080044import androidx.annotation.Nullable;
Anton Vayvodee382242024-05-28 21:26:25 -040045import androidx.core.os.BuildCompat;
Tracy Zhoube13d102020-01-12 01:07:59 -080046
Sunny Goyalb47172b2021-05-03 19:59:51 -070047import com.android.launcher3.graphics.IconShape;
Sunny Goyalf840f102018-09-21 14:41:05 -070048import com.android.launcher3.icons.IconCache;
Sunny Goyal14168432019-10-24 15:59:49 -070049import com.android.launcher3.icons.IconProvider;
Sunny Goyal68af5492021-12-24 01:08:31 +053050import com.android.launcher3.icons.LauncherIconProvider;
Sunny Goyal87dc48b2018-10-12 11:42:33 -070051import com.android.launcher3.icons.LauncherIcons;
Charlie Andersonbad2be42024-12-06 16:32:03 -050052import com.android.launcher3.logging.FileLog;
Sunny Goyal6449a212024-01-12 09:40:43 -080053import com.android.launcher3.model.ModelLauncherCallbacks;
Shamali Pea078cb2024-11-04 21:35:38 +000054import com.android.launcher3.model.WidgetsFilterDataProvider;
Tonyd48710c2017-07-27 23:23:58 -070055import com.android.launcher3.notification.NotificationListener;
Sunny Goyal73b5a272019-12-09 14:55:56 -080056import com.android.launcher3.pm.InstallSessionHelper;
Sunny Goyal045b4fa2019-09-20 12:51:37 -070057import com.android.launcher3.pm.InstallSessionTracker;
Sunny Goyal337c81f2019-12-10 12:19:13 -080058import com.android.launcher3.pm.UserCache;
Stefan Andonian5bd9a222023-02-23 00:58:33 +000059import com.android.launcher3.util.LockedUserState;
Sunny Goyald0e360a2018-06-29 14:40:18 -070060import com.android.launcher3.util.MainThreadInitializedObject;
Winson Chung94e8ad02024-05-08 21:14:57 +000061import com.android.launcher3.util.PackageManagerHelper;
Sunny Goyalfdbef272017-02-01 12:52:54 -080062import com.android.launcher3.util.Preconditions;
Sunny Goyalb47172b2021-05-03 19:59:51 -070063import com.android.launcher3.util.RunnableList;
Sunny Goyal14168432019-10-24 15:59:49 -070064import com.android.launcher3.util.SafeCloseable;
Jon Mirandaec1277e2021-03-25 10:41:54 -040065import com.android.launcher3.util.SettingsCache;
Sunny Goyale7b00122019-10-02 16:13:34 -070066import com.android.launcher3.util.SimpleBroadcastReceiver;
Sunny Goyal01c32a62021-05-17 13:23:08 -070067import com.android.launcher3.util.Themes;
Sunny Goyal31e27ed2024-01-09 15:45:14 -080068import com.android.launcher3.util.TraceHelper;
Pinyao Tingc7a6c292019-08-26 14:36:02 -070069import com.android.launcher3.widget.custom.CustomWidgetManager;
Sunny Goyal71b5c0b2015-01-08 16:59:04 -080070
Sunny Goyal777d4902021-08-27 21:22:17 +000071public class LauncherAppState implements SafeCloseable {
Chris Wrenaeff7ea2014-02-14 16:59:24 -050072
Charlie Andersonbad2be42024-12-06 16:32:03 -050073 public static final String TAG = "LauncherAppState";
Sunny Goyal29947f02017-12-18 13:49:44 -080074 public static final String ACTION_FORCE_ROLOAD = "force-reload-launcher";
75
Sunny Goyalfdbef272017-02-01 12:52:54 -080076 // We do not need any synchronization for this variable as its only written on UI thread.
Sunny Goyalcf845f02019-09-25 09:08:16 -070077 public static final MainThreadInitializedObject<LauncherAppState> INSTANCE =
Sunny Goyale529a862019-08-06 09:48:36 -070078 new MainThreadInitializedObject<>(LauncherAppState::new);
Daniel Sandlercc8befa2013-06-11 14:45:48 -040079
Sunny Goyalfdbef272017-02-01 12:52:54 -080080 private final Context mContext;
81 private final LauncherModel mModel;
Sunny Goyal68af5492021-12-24 01:08:31 +053082 private final LauncherIconProvider mIconProvider;
Sunny Goyalfdbef272017-02-01 12:52:54 -080083 private final IconCache mIconCache;
Sunny Goyalfdbef272017-02-01 12:52:54 -080084 private final InvariantDeviceProfile mInvariantDeviceProfile;
Sunny Goyal31e27ed2024-01-09 15:45:14 -080085 private boolean mIsSafeModeEnabled;
86
Sunny Goyalb47172b2021-05-03 19:59:51 -070087 private final RunnableList mOnTerminateCallback = new RunnableList();
Sunny Goyal045b4fa2019-09-20 12:51:37 -070088
Sunny Goyal10fa0162024-04-21 00:13:35 -070089 public static LauncherAppState getInstance(Context context) {
Sunny Goyald0e360a2018-06-29 14:40:18 -070090 return INSTANCE.get(context);
Daniel Sandlercc8befa2013-06-11 14:45:48 -040091 }
92
Daniel Sandlercc8befa2013-06-11 14:45:48 -040093 public Context getContext() {
Sunny Goyalfdbef272017-02-01 12:52:54 -080094 return mContext;
Daniel Sandlercc8befa2013-06-11 14:45:48 -040095 }
96
Jakob Schneider2f609d02024-01-10 18:03:35 +000097 @SuppressWarnings("NewApi")
Tracy Zhoube13d102020-01-12 01:07:59 -080098 public LauncherAppState(Context context) {
99 this(context, LauncherFiles.APP_ICONS_DB);
Sunny Goyaldf4241c2021-05-19 19:42:14 -0700100 Log.v(Launcher.TAG, "LauncherAppState initiated");
101 Preconditions.assertUIThread();
Sunny Goyal27595792015-03-19 13:18:44 -0700102
Sunny Goyal31e27ed2024-01-09 15:45:14 -0800103 mIsSafeModeEnabled = TraceHelper.allowIpcs("isSafeMode",
104 () -> context.getPackageManager().isSafeMode());
Sunny Goyal6e6f7992021-08-24 16:23:29 -0700105 mInvariantDeviceProfile.addOnChangeListener(modelPropertiesChanged -> {
106 if (modelPropertiesChanged) {
107 refreshAndReloadLauncher();
108 }
109 });
Daniel Sandlercc8befa2013-06-11 14:45:48 -0400110
Sunny Goyal6449a212024-01-12 09:40:43 -0800111 ModelLauncherCallbacks callbacks = mModel.newModelCallbacks();
Jakob Schneider2f609d02024-01-10 18:03:35 +0000112 LauncherApps launcherApps = mContext.getSystemService(LauncherApps.class);
113 launcherApps.registerCallback(callbacks);
Sunny Goyale274d972023-05-01 16:55:59 -0700114 mOnTerminateCallback.add(() ->
Sunny Goyal6449a212024-01-12 09:40:43 -0800115 mContext.getSystemService(LauncherApps.class).unregisterCallback(callbacks));
Jakob Schneider328c3222024-02-01 15:39:11 +0000116
Anton Vayvodee382242024-05-28 21:26:25 -0400117 if (BuildCompat.isAtLeastV() && Flags.enableSupportForArchiving()) {
Jakob Schneider94b5f172024-01-31 19:17:10 +0000118 ArchiveCompatibilityParams params = new ArchiveCompatibilityParams();
119 params.setEnableUnarchivalConfirmation(false);
Charlie Anderson6b8e3622024-08-01 15:19:27 -0400120 params.setEnableIconOverlay(!Flags.useNewIconForArchivedApps());
Jakob Schneider94b5f172024-01-31 19:17:10 +0000121 launcherApps.setArchiveCompatibility(params);
Jakob Schneider2f609d02024-01-10 18:03:35 +0000122 }
Sunny Goyalb47172b2021-05-03 19:59:51 -0700123
124 SimpleBroadcastReceiver modelChangeReceiver =
Fengjiang Lib87ad6f2024-07-09 10:15:43 -0700125 new SimpleBroadcastReceiver(UI_HELPER_EXECUTOR, mModel::onBroadcastIntent);
Fengjiang Lib87ad6f2024-07-09 10:15:43 -0700126 modelChangeReceiver.register(
127 mContext,
kholoud mohamedd5a4d602022-04-28 15:55:40 +0100128 ACTION_DEVICE_POLICY_RESOURCE_UPDATED);
Sunny Goyalb02dafc2023-06-22 11:17:29 -0700129 if (BuildConfig.IS_STUDIO_BUILD) {
Sunny Goyale79d4532024-12-31 00:10:20 -0800130 modelChangeReceiver.register(mContext, RECEIVER_EXPORTED, ACTION_FORCE_ROLOAD);
Sunny Goyal29947f02017-12-18 13:49:44 -0800131 }
Fengjiang Lib87ad6f2024-07-09 10:15:43 -0700132 mOnTerminateCallback.add(() -> modelChangeReceiver.unregisterReceiverSafely(mContext));
Sunny Goyal14168432019-10-24 15:59:49 -0700133
Sunny Goyalb47172b2021-05-03 19:59:51 -0700134 SafeCloseable userChangeListener = UserCache.INSTANCE.get(mContext)
Sunny Goyalc80e60d2023-06-07 12:52:30 -0700135 .addUserEventListener(mModel::onUserEvent);
Sunny Goyalb47172b2021-05-03 19:59:51 -0700136 mOnTerminateCallback.add(userChangeListener::close);
Sunny Goyal337c81f2019-12-10 12:19:13 -0800137
fbaron9d08e0f2024-04-24 11:26:01 -0700138 if (enableSmartspaceRemovalToggle()) {
fbaronc0e687b2023-10-04 12:11:20 -0700139 OnSharedPreferenceChangeListener firstPagePinnedItemListener =
140 new OnSharedPreferenceChangeListener() {
141 @Override
142 public void onSharedPreferenceChanged(
143 SharedPreferences sharedPreferences, String key) {
144 if (SMARTSPACE_ON_HOME_SCREEN.equals(key)) {
145 mModel.forceReload();
146 }
147 }
148 };
149 LauncherPrefs.getPrefs(mContext).registerOnSharedPreferenceChangeListener(
150 firstPagePinnedItemListener);
151 mOnTerminateCallback.add(() -> LauncherPrefs.getPrefs(mContext)
152 .unregisterOnSharedPreferenceChangeListener(firstPagePinnedItemListener));
153 }
154
Stefan Andonian5bd9a222023-02-23 00:58:33 +0000155 LockedUserState.get(context).runOnUserUnlocked(() -> {
Sunny Goyale274d972023-05-01 16:55:59 -0700156 CustomWidgetManager cwm = CustomWidgetManager.INSTANCE.get(mContext);
Sunny Goyal6bbd8052024-10-24 12:38:33 -0700157 mOnTerminateCallback.add(cwm.addWidgetRefreshCallback(mModel::rebindCallbacks)::close);
Tony Wickham827cef22016-03-17 15:39:39 -0700158
Stefan Andonian5bd9a222023-02-23 00:58:33 +0000159 IconObserver observer = new IconObserver();
160 SafeCloseable iconChangeTracker = mIconProvider.registerIconChangeListener(
161 observer, MODEL_EXECUTOR.getHandler());
162 mOnTerminateCallback.add(iconChangeTracker::close);
163 MODEL_EXECUTOR.execute(observer::verifyIconChanged);
164 LauncherPrefs.get(context).addListener(observer, THEMED_ICONS);
165 mOnTerminateCallback.add(
166 () -> LauncherPrefs.get(mContext).removeListener(observer, THEMED_ICONS));
167
168 InstallSessionTracker installSessionTracker =
Sunny Goyalc3632952024-10-09 15:15:42 -0700169 InstallSessionHelper.INSTANCE.get(context).registerInstallTracker(callbacks);
Stefan Andonian5bd9a222023-02-23 00:58:33 +0000170 mOnTerminateCallback.add(installSessionTracker::unregister);
171 });
Sunny Goyal045b4fa2019-09-20 12:51:37 -0700172
Sunny Goyaleaf7a952020-07-29 16:54:20 -0700173 // Register an observer to rebind the notification listener when dots are re-enabled.
Sunny Goyalb47172b2021-05-03 19:59:51 -0700174 SettingsCache settingsCache = SettingsCache.INSTANCE.get(mContext);
175 SettingsCache.OnChangeListener notificationLister = this::onNotificationSettingsChanged;
176 settingsCache.register(NOTIFICATION_BADGING_URI, notificationLister);
177 onNotificationSettingsChanged(settingsCache.getValue(NOTIFICATION_BADGING_URI));
178 mOnTerminateCallback.add(() ->
179 settingsCache.unregister(NOTIFICATION_BADGING_URI, notificationLister));
Himanshu Gupta739b3c92023-11-06 17:47:29 +0000180 // Register an observer to notify Launcher about Private Space settings toggle.
Pechetty Sravani (xWF)41743bd2024-11-14 06:11:08 +0000181 registerPrivateSpaceHideWhenLockListener(settingsCache);
Sunny Goyal420d5452018-10-17 10:25:14 -0700182 }
183
Tracy Zhoube13d102020-01-12 01:07:59 -0800184 public LauncherAppState(Context context, @Nullable String iconCacheFileName) {
Tracy Zhoube13d102020-01-12 01:07:59 -0800185 mContext = context;
186
187 mInvariantDeviceProfile = InvariantDeviceProfile.INSTANCE.get(context);
Sunny Goyal68af5492021-12-24 01:08:31 +0530188 mIconProvider = new LauncherIconProvider(context);
Sunny Goyalb47172b2021-05-03 19:59:51 -0700189 mIconCache = new IconCache(mContext, mInvariantDeviceProfile,
190 iconCacheFileName, mIconProvider);
Shamali Pea078cb2024-11-04 21:35:38 +0000191 mModel = new LauncherModel(context, this, mIconCache,
192 WidgetsFilterDataProvider.Companion.newInstance(context), new AppFilter(mContext),
Winson Chung94e8ad02024-05-08 21:14:57 +0000193 PackageManagerHelper.INSTANCE.get(context), iconCacheFileName != null);
Sunny Goyaldf4241c2021-05-19 19:42:14 -0700194 mOnTerminateCallback.add(mIconCache::close);
Sunny Goyale274d972023-05-01 16:55:59 -0700195 mOnTerminateCallback.add(mModel::destroy);
Tracy Zhoube13d102020-01-12 01:07:59 -0800196 }
197
Sunny Goyalb47172b2021-05-03 19:59:51 -0700198 private void onNotificationSettingsChanged(boolean areNotificationDotsEnabled) {
Tony Wickhamf34bee82018-12-03 18:11:39 -0800199 if (areNotificationDotsEnabled) {
Sunny Goyal420d5452018-10-17 10:25:14 -0700200 NotificationListener.requestRebind(new ComponentName(
201 mContext, NotificationListener.class));
Tonyd48710c2017-07-27 23:23:58 -0700202 }
Daniel Sandlercc8befa2013-06-11 14:45:48 -0400203 }
Kenny Guy1317e2d2014-05-08 18:52:50 +0100204
Himanshu Gupta739b3c92023-11-06 17:47:29 +0000205 private void registerPrivateSpaceHideWhenLockListener(SettingsCache settingsCache) {
206 SettingsCache.OnChangeListener psHideWhenLockChangedListener =
207 this::onPrivateSpaceHideWhenLockChanged;
208 settingsCache.register(PRIVATE_SPACE_HIDE_WHEN_LOCKED_URI, psHideWhenLockChangedListener);
209 mOnTerminateCallback.add(() -> settingsCache.unregister(PRIVATE_SPACE_HIDE_WHEN_LOCKED_URI,
210 psHideWhenLockChangedListener));
211 }
212
213 private void onPrivateSpaceHideWhenLockChanged(boolean isPrivateSpaceHideOnLockEnabled) {
214 mModel.forceReload();
215 }
216
Sunny Goyalb47172b2021-05-03 19:59:51 -0700217 private void refreshAndReloadLauncher() {
Sunny Goyalbf3efe82024-05-07 10:19:57 -0700218 LauncherIcons.clearPool(mContext);
Sunny Goyalb47172b2021-05-03 19:59:51 -0700219 mIconCache.updateIconParams(
220 mInvariantDeviceProfile.fillResIconDpi, mInvariantDeviceProfile.iconBitmapSize);
Sunny Goyal87dc48b2018-10-12 11:42:33 -0700221 mModel.forceReload();
222 }
223
Daniel Sandlercc8befa2013-06-11 14:45:48 -0400224 /**
225 * Call from Application.onTerminate(), which is not guaranteed to ever be called.
226 */
Sunny Goyal777d4902021-08-27 21:22:17 +0000227 @Override
228 public void close() {
Sunny Goyalb47172b2021-05-03 19:59:51 -0700229 mOnTerminateCallback.executeAllAndDestroy();
230 }
Sunny Goyal01615a62019-09-20 12:00:07 -0700231
Sunny Goyalb47172b2021-05-03 19:59:51 -0700232 public IconProvider getIconProvider() {
233 return mIconProvider;
Daniel Sandlercc8befa2013-06-11 14:45:48 -0400234 }
235
Sunny Goyal34942622014-08-29 17:20:55 -0700236 public IconCache getIconCache() {
Daniel Sandlercc8befa2013-06-11 14:45:48 -0400237 return mIconCache;
238 }
239
Sunny Goyal18bf8e22015-04-08 18:13:46 -0700240 public LauncherModel getModel() {
Daniel Sandlercc8befa2013-06-11 14:45:48 -0400241 return mModel;
242 }
243
Adam Cohen2e6da152015-05-06 11:42:25 -0700244 public InvariantDeviceProfile getInvariantDeviceProfile() {
245 return mInvariantDeviceProfile;
246 }
Sunny Goyal87f784c2017-01-11 10:48:34 -0800247
Sunny Goyal31e27ed2024-01-09 15:45:14 -0800248 public boolean isSafeModeEnabled() {
249 return mIsSafeModeEnabled;
250 }
251
Sunny Goyal87f784c2017-01-11 10:48:34 -0800252 /**
253 * Shorthand for {@link #getInvariantDeviceProfile()}
254 */
255 public static InvariantDeviceProfile getIDP(Context context) {
Sunny Goyald0e360a2018-06-29 14:40:18 -0700256 return InvariantDeviceProfile.INSTANCE.get(context);
Sunny Goyal87f784c2017-01-11 10:48:34 -0800257 }
Sunny Goyalb47172b2021-05-03 19:59:51 -0700258
Sunny Goyal01c32a62021-05-17 13:23:08 -0700259 private class IconObserver
Brian Isganitis9cbc4782024-10-08 14:47:17 -0400260 implements IconProvider.IconChangeListener, LauncherPrefChangeListener {
Sunny Goyalb47172b2021-05-03 19:59:51 -0700261
262 @Override
263 public void onAppIconChanged(String packageName, UserHandle user) {
264 mModel.onAppIconChanged(packageName, user);
265 }
266
267 @Override
268 public void onSystemIconStateChanged(String iconState) {
Sunny Goyal638a6872024-04-25 15:55:11 -0700269 IconShape.INSTANCE.get(mContext).pickBestShape(mContext);
Sunny Goyalb47172b2021-05-03 19:59:51 -0700270 refreshAndReloadLauncher();
Stefan Andoniand1b33b32022-12-16 21:22:27 +0000271 LauncherPrefs.get(mContext).put(ICON_STATE, iconState);
Sunny Goyalb47172b2021-05-03 19:59:51 -0700272 }
273
274 void verifyIconChanged() {
275 String iconState = mIconProvider.getSystemIconState();
Stefan Andoniand1b33b32022-12-16 21:22:27 +0000276 if (!iconState.equals(LauncherPrefs.get(mContext).get(ICON_STATE))) {
Sunny Goyalb47172b2021-05-03 19:59:51 -0700277 onSystemIconStateChanged(iconState);
278 }
279 }
Sunny Goyal01c32a62021-05-17 13:23:08 -0700280
281 @Override
Brian Isganitis9cbc4782024-10-08 14:47:17 -0400282 public void onPrefChanged(String key) {
Sunny Goyal01c32a62021-05-17 13:23:08 -0700283 if (Themes.KEY_THEMED_ICONS.equals(key)) {
284 mIconProvider.setIconThemeSupported(Themes.isThemedIconEnabled(mContext));
285 verifyIconChanged();
Charlie Andersonbad2be42024-12-06 16:32:03 -0500286 } else if (GRID_NAME_PREFS_KEY.equals(key)) {
287 FileLog.d(TAG, "onPrefChanged GRID_NAME changed: "
288 + LauncherPrefs.get(mContext).get(GRID_NAME));
289 } else if (KEY_DB_FILE.equals(key)) {
290 FileLog.d(TAG, "onPrefChanged DB_FILE changed: "
291 + LauncherPrefs.get(mContext).get(DB_FILE));
Sunny Goyal01c32a62021-05-17 13:23:08 -0700292 }
293 }
Sunny Goyalb47172b2021-05-03 19:59:51 -0700294 }
Daniel Sandlercc8befa2013-06-11 14:45:48 -0400295}