/* * Copyright (C) 2018 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package com.android.settings.homepage; import static android.provider.Settings.ACTION_SETTINGS_EMBED_DEEP_LINK_ACTIVITY; import static android.provider.Settings.EXTRA_SETTINGS_EMBEDDED_DEEP_LINK_HIGHLIGHT_MENU_KEY; import static android.provider.Settings.EXTRA_SETTINGS_EMBEDDED_DEEP_LINK_INTENT_URI; import static com.android.settings.SettingsActivity.EXTRA_USER_HANDLE; import android.animation.LayoutTransition; import android.app.ActivityManager; import android.app.settings.SettingsEnums; import android.content.ComponentName; import android.content.Intent; import android.content.pm.ActivityInfo; import android.content.pm.PackageManager; import android.content.pm.PackageManager.ApplicationInfoFlags; import android.content.pm.UserInfo; import android.content.res.Configuration; import android.net.Uri; import android.os.Bundle; import android.os.Process; import android.os.UserHandle; import android.os.UserManager; import android.text.TextUtils; import android.util.ArraySet; import android.util.FeatureFlagUtils; import android.util.Log; import android.view.View; import android.view.Window; import android.view.WindowManager; import android.widget.FrameLayout; import android.widget.ImageView; import android.widget.Toolbar; import androidx.annotation.VisibleForTesting; import androidx.core.graphics.Insets; import androidx.core.util.Consumer; import androidx.core.view.ViewCompat; import androidx.core.view.WindowCompat; import androidx.core.view.WindowInsetsCompat; import androidx.fragment.app.Fragment; import androidx.fragment.app.FragmentActivity; import androidx.fragment.app.FragmentManager; import androidx.fragment.app.FragmentTransaction; import androidx.window.embedding.SplitController; import androidx.window.embedding.SplitInfo; import androidx.window.embedding.SplitRule; import androidx.window.java.embedding.SplitControllerCallbackAdapter; import com.android.settings.R; import com.android.settings.Settings; import com.android.settings.SettingsActivity; import com.android.settings.SettingsApplication; import com.android.settings.accounts.AvatarViewMixin; import com.android.settings.activityembedding.ActivityEmbeddingRulesController; import com.android.settings.activityembedding.ActivityEmbeddingUtils; import com.android.settings.activityembedding.EmbeddedDeepLinkUtils; import com.android.settings.core.CategoryMixin; import com.android.settings.core.FeatureFlags; import com.android.settings.flags.Flags; import com.android.settings.homepage.contextualcards.ContextualCardsFragment; import com.android.settings.overlay.FeatureFactory; import com.android.settings.safetycenter.SafetyCenterManagerWrapper; import com.android.settingslib.Utils; import com.android.settingslib.core.lifecycle.HideNonSystemOverlayMixin; import com.google.android.setupcompat.util.WizardManagerHelper; import java.net.URISyntaxException; import java.util.List; import java.util.Set; /** Settings homepage activity */ public class SettingsHomepageActivity extends FragmentActivity implements CategoryMixin.CategoryHandler { private static final String TAG = "SettingsHomepageActivity"; // Additional extra of Settings#ACTION_SETTINGS_LARGE_SCREEN_DEEP_LINK. // Put true value to the intent when startActivity for a deep link intent from this Activity. public static final String EXTRA_IS_FROM_SETTINGS_HOMEPAGE = "is_from_settings_homepage"; // Additional extra of Settings#ACTION_SETTINGS_LARGE_SCREEN_DEEP_LINK. // Set & get Uri of the Intent separately to prevent failure of Intent#ParseUri. public static final String EXTRA_SETTINGS_LARGE_SCREEN_DEEP_LINK_INTENT_DATA = "settings_large_screen_deep_link_intent_data"; // The referrer who fires the initial intent to start the homepage @VisibleForTesting static final String EXTRA_INITIAL_REFERRER = "initial_referrer"; static final int DEFAULT_HIGHLIGHT_MENU_KEY = R.string.menu_key_network; private static final long HOMEPAGE_LOADING_TIMEOUT_MS = 300; private TopLevelSettings mMainFragment; private View mHomepageView; private View mSuggestionView; private View mTwoPaneSuggestionView; private CategoryMixin mCategoryMixin; private Set mLoadedListeners; private boolean mIsEmbeddingActivityEnabled; private boolean mIsTwoPane; // A regular layout shows icons on homepage, whereas a simplified layout doesn't. private boolean mIsRegularLayout = true; private SplitControllerCallbackAdapter mSplitControllerAdapter; private SplitInfoCallback mCallback; private boolean mAllowUpdateSuggestion = true; /** A listener receiving homepage loaded events. */ public interface HomepageLoadedListener { /** Called when the homepage is loaded. */ void onHomepageLoaded(); } private interface FragmentCreator { T create(); /** To initialize after {@link #create} */ default void init(Fragment fragment) {} } /** * Try to add a {@link HomepageLoadedListener}. If homepage is already loaded, the listener * will not be notified. * * @return Whether the listener is added. */ public boolean addHomepageLoadedListener(HomepageLoadedListener listener) { if (mHomepageView == null) { return false; } else { if (!mLoadedListeners.contains(listener)) { mLoadedListeners.add(listener); } return true; } } /** * Shows the homepage and shows/hides the suggestion together. Only allows to be executed once * to avoid the flicker caused by the suggestion suddenly appearing/disappearing. */ public void showHomepageWithSuggestion(boolean showSuggestion) { if (mAllowUpdateSuggestion) { Log.i(TAG, "showHomepageWithSuggestion: " + showSuggestion); mAllowUpdateSuggestion = false; if (Flags.homepageRevamp()) { mSuggestionView.setVisibility(showSuggestion ? View.VISIBLE : View.GONE); } else { mSuggestionView.setVisibility(showSuggestion ? View.VISIBLE : View.GONE); mTwoPaneSuggestionView.setVisibility(showSuggestion ? View.VISIBLE : View.GONE); } } if (mHomepageView == null) { return; } final View homepageView = mHomepageView; mHomepageView = null; mLoadedListeners.forEach(listener -> listener.onHomepageLoaded()); mLoadedListeners.clear(); homepageView.setVisibility(View.VISIBLE); } /** Returns the main content fragment */ public TopLevelSettings getMainFragment() { return mMainFragment; } @Override public CategoryMixin getCategoryMixin() { return mCategoryMixin; } @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); // Ensure device is provisioned in order to access Settings home // TODO(b/331254029): This should later be replaced in favor of an allowlist boolean unprovisioned = android.provider.Settings.Global.getInt(getContentResolver(), android.provider.Settings.Global.DEVICE_PROVISIONED, 0) == 0; if (unprovisioned) { Log.e(TAG, "Device is not provisioned, exiting Settings"); finish(); return; } // Settings homepage should be the task root, otherwise there will be UI issues. boolean isTaskRoot = isTaskRoot(); mIsEmbeddingActivityEnabled = ActivityEmbeddingUtils.isEmbeddingActivityEnabled(this); if (mIsEmbeddingActivityEnabled) { final UserManager um = getSystemService(UserManager.class); final UserInfo userInfo = um.getUserInfo(getUserId()); if (EmbeddedDeepLinkUtils.isSubProfile(userInfo)) { final Intent intent = new Intent(getIntent()) .addFlags(Intent.FLAG_ACTIVITY_FORWARD_RESULT) .putExtra(EXTRA_USER_HANDLE, getUser()) .putExtra(EXTRA_INITIAL_REFERRER, getCurrentReferrer()); if (TextUtils.equals(intent.getAction(), ACTION_SETTINGS_EMBED_DEEP_LINK_ACTIVITY) && this instanceof DeepLinkHomepageActivity) { intent.setClass(this, DeepLinkHomepageActivityInternal.class); } else { intent.setPackage(getPackageName()); } if (!isTaskRoot) { intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK); } else { intent.removeFlags(Intent.FLAG_ACTIVITY_NEW_TASK); } startActivityAsUser(intent, um.getProfileParent(userInfo.id).getUserHandle()); finish(); return; } } if (!isTaskRoot) { if ((getIntent().getFlags() & Intent.FLAG_ACTIVITY_NEW_TASK) != 0) { Log.i(TAG, "Activity has been started, finishing"); } else { Log.i(TAG, "Homepage should be started with FLAG_ACTIVITY_NEW_TASK, restarting"); Intent intent = new Intent(getIntent()) .setPackage(getPackageName()) .addFlags(Intent.FLAG_ACTIVITY_NEW_TASK | Intent.FLAG_ACTIVITY_FORWARD_RESULT) .putExtra(EXTRA_USER_HANDLE, getUser()) .putExtra(EXTRA_INITIAL_REFERRER, getCurrentReferrer()); startActivity(intent); } finish(); return; } setupEdgeToEdge(); setContentView( Flags.homepageRevamp() ? R.layout.settings_homepage_container_v2 : R.layout.settings_homepage_container); mIsTwoPane = ActivityEmbeddingUtils.isAlreadyEmbedded(this); updateAppBarMinHeight(); initHomepageContainer(); updateHomepageAppBar(); updateHomepageBackground(); mLoadedListeners = new ArraySet<>(); initSearchBarView(); getLifecycle().addObserver(new HideNonSystemOverlayMixin(this)); mCategoryMixin = new CategoryMixin(this); getLifecycle().addObserver(mCategoryMixin); final String highlightMenuKey = getHighlightMenuKey(); // Only allow features on high ram devices. if (!getSystemService(ActivityManager.class).isLowRamDevice()) { initAvatarView(); final boolean scrollNeeded = mIsEmbeddingActivityEnabled && !TextUtils.equals(getString(DEFAULT_HIGHLIGHT_MENU_KEY), highlightMenuKey); showSuggestionFragment(scrollNeeded); if (FeatureFlagUtils.isEnabled(this, FeatureFlags.CONTEXTUAL_HOME)) { showFragment(() -> new ContextualCardsFragment(), R.id.contextual_cards_content); ((FrameLayout) findViewById(R.id.main_content)) .getLayoutTransition().enableTransitionType(LayoutTransition.CHANGING); } } mMainFragment = showFragment(() -> { final TopLevelSettings fragment = new TopLevelSettings(); fragment.getArguments().putString(SettingsActivity.EXTRA_FRAGMENT_ARG_KEY, highlightMenuKey); return fragment; }, R.id.main_content); // Launch the intent from deep link for large screen devices. if (shouldLaunchDeepLinkIntentToRight()) { launchDeepLinkIntentToRight(); } // Settings app may be launched on an existing task. Reset SplitPairRule of SubSettings here // to prevent SplitPairRule of an existing task applied on a new started Settings app. if (mIsEmbeddingActivityEnabled && (getIntent().getFlags() & Intent.FLAG_ACTIVITY_CLEAR_TOP) != 0) { initSplitPairRules(); } updateHomepagePaddings(); updateSplitLayout(); enableTaskLocaleOverride(); } @VisibleForTesting void initSplitPairRules() { new ActivityEmbeddingRulesController(getApplicationContext()).initRules(); } @Override protected void onStart() { ((SettingsApplication) getApplication()).setHomeActivity(this); super.onStart(); if (mIsEmbeddingActivityEnabled) { final SplitController splitController = SplitController.getInstance(this); mSplitControllerAdapter = new SplitControllerCallbackAdapter(splitController); mCallback = new SplitInfoCallback(this); mSplitControllerAdapter.addSplitListener(this, Runnable::run, mCallback); } } @Override protected void onStop() { super.onStop(); mAllowUpdateSuggestion = true; if (mSplitControllerAdapter != null && mCallback != null) { mSplitControllerAdapter.removeSplitListener(mCallback); mCallback = null; mSplitControllerAdapter = null; } } @Override protected void onNewIntent(Intent intent) { super.onNewIntent(intent); // When it's large screen 2-pane and Settings app is in the background, receiving an Intent // will not recreate this activity. Update the intent for this case. setIntent(intent); reloadHighlightMenuKey(); if (isFinishing()) { return; } // Launch the intent from deep link for large screen devices. if (shouldLaunchDeepLinkIntentToRight()) { launchDeepLinkIntentToRight(); } } @Override public void onConfigurationChanged(Configuration newConfig) { super.onConfigurationChanged(newConfig); updateHomepageUI(); } private void updateSplitLayout() { if (!mIsEmbeddingActivityEnabled) { return; } if (mIsTwoPane) { if (mIsRegularLayout == ActivityEmbeddingUtils.isRegularHomepageLayout(this)) { // Layout unchanged return; } } else if (mIsRegularLayout) { // One pane mode with the regular layout, not needed to change return; } mIsRegularLayout = !mIsRegularLayout; // Update search title padding View searchTitle = findViewById(R.id.search_bar_title); if (searchTitle != null) { int paddingStart = getResources().getDimensionPixelSize( mIsRegularLayout ? R.dimen.search_bar_title_padding_start_regular_two_pane : R.dimen.search_bar_title_padding_start); searchTitle.setPaddingRelative(paddingStart, 0, 0, 0); } // Notify fragments getSupportFragmentManager().getFragments().forEach(fragment -> { if (fragment instanceof SplitLayoutListener) { ((SplitLayoutListener) fragment).onSplitLayoutChanged(mIsRegularLayout); } }); } private void setupEdgeToEdge() { WindowCompat.setDecorFitsSystemWindows(getWindow(), false); ViewCompat.setOnApplyWindowInsetsListener(findViewById(android.R.id.content), (v, windowInsets) -> { Insets insets = windowInsets.getInsets(WindowInsetsCompat.Type.systemBars()); // Apply the insets paddings to the view. v.setPadding(insets.left, insets.top, insets.right, insets.bottom); // Return CONSUMED if you don't want the window insets to keep being // passed down to descendant views. return WindowInsetsCompat.CONSUMED; }); } private void initSearchBarView() { if (Flags.homepageRevamp()) { View toolbar = findViewById(R.id.search_action_bar); FeatureFactory.getFeatureFactory().getSearchFeatureProvider() .initSearchToolbar(this /* activity */, toolbar, SettingsEnums.SETTINGS_HOMEPAGE); } else { final Toolbar toolbar = findViewById(R.id.search_action_bar); FeatureFactory.getFeatureFactory().getSearchFeatureProvider() .initSearchToolbar(this /* activity */, toolbar, SettingsEnums.SETTINGS_HOMEPAGE); if (mIsEmbeddingActivityEnabled) { final Toolbar toolbarTwoPaneVersion = findViewById(R.id.search_action_bar_two_pane); FeatureFactory.getFeatureFactory().getSearchFeatureProvider() .initSearchToolbar(this /* activity */, toolbarTwoPaneVersion, SettingsEnums.SETTINGS_HOMEPAGE); } } } private void initAvatarView() { if (Flags.homepageRevamp()) { return; } final ImageView avatarView = findViewById(R.id.account_avatar); final ImageView avatarTwoPaneView = findViewById(R.id.account_avatar_two_pane_version); if (AvatarViewMixin.isAvatarSupported(this)) { avatarView.setVisibility(View.VISIBLE); getLifecycle().addObserver(new AvatarViewMixin(this, avatarView)); if (mIsEmbeddingActivityEnabled) { avatarTwoPaneView.setVisibility(View.VISIBLE); getLifecycle().addObserver(new AvatarViewMixin(this, avatarTwoPaneView)); } } } private void updateHomepageUI() { final boolean newTwoPaneState = ActivityEmbeddingUtils.isAlreadyEmbedded(this); if (mIsTwoPane != newTwoPaneState) { mIsTwoPane = newTwoPaneState; updateHomepageAppBar(); updateHomepageBackground(); updateHomepagePaddings(); } updateSplitLayout(); } private void updateHomepageBackground() { if (!Flags.homepageRevamp() && !mIsEmbeddingActivityEnabled) { return; } final Window window = getWindow(); final int color = mIsTwoPane ? getColor(R.color.settings_two_pane_background_color) : Utils.getColorAttrDefaultColor(this, android.R.attr.colorBackground); window.addFlags(WindowManager.LayoutParams.FLAG_DRAWS_SYSTEM_BAR_BACKGROUNDS); // Update status bar color window.setStatusBarColor(color); // Update content background. findViewById(android.R.id.content).setBackgroundColor(color); if (Flags.homepageRevamp()) { //Update search bar background findViewById(R.id.app_bar_container).setBackgroundColor(color); } } private void showSuggestionFragment(boolean scrollNeeded) { final Class fragmentClass = FeatureFactory.getFeatureFactory() .getSuggestionFeatureProvider().getContextualSuggestionFragment(); if (fragmentClass == null) { return; } if (Flags.homepageRevamp()) { mSuggestionView = findViewById(R.id.suggestion_content); } else { mSuggestionView = findViewById(R.id.suggestion_content); mTwoPaneSuggestionView = findViewById(R.id.two_pane_suggestion_content); } mHomepageView = findViewById(R.id.settings_homepage_container); // Hide the homepage for preparing the suggestion. If scrolling is needed, the list views // should be initialized in the invisible homepage view to prevent a scroll flicker. mHomepageView.setVisibility(scrollNeeded ? View.INVISIBLE : View.GONE); // Schedule a timer to show the homepage and hide the suggestion on timeout. mHomepageView.postDelayed(() -> showHomepageWithSuggestion(false), HOMEPAGE_LOADING_TIMEOUT_MS); if (Flags.homepageRevamp()) { showFragment(new SuggestionFragCreator(fragmentClass, true), R.id.suggestion_content); } else { showFragment(new SuggestionFragCreator(fragmentClass, /* isTwoPaneLayout= */ false), R.id.suggestion_content); if (mIsEmbeddingActivityEnabled) { showFragment(new SuggestionFragCreator(fragmentClass, /* isTwoPaneLayout= */ true), R.id.two_pane_suggestion_content); } } } private T showFragment(FragmentCreator fragmentCreator, int id) { final FragmentManager fragmentManager = getSupportFragmentManager(); final FragmentTransaction fragmentTransaction = fragmentManager.beginTransaction(); T showFragment = (T) fragmentManager.findFragmentById(id); if (showFragment == null) { showFragment = fragmentCreator.create(); fragmentCreator.init(showFragment); fragmentTransaction.add(id, showFragment); } else { fragmentCreator.init(showFragment); fragmentTransaction.show(showFragment); } fragmentTransaction.commit(); return showFragment; } private boolean shouldLaunchDeepLinkIntentToRight() { if (!ActivityEmbeddingUtils.isSettingsSplitEnabled(this) || !FeatureFlagUtils.isEnabled(this, FeatureFlagUtils.SETTINGS_SUPPORT_LARGE_SCREEN)) { return false; } Intent intent = getIntent(); return intent != null && TextUtils.equals(intent.getAction(), ACTION_SETTINGS_EMBED_DEEP_LINK_ACTIVITY); } private void launchDeepLinkIntentToRight() { if (!(this instanceof DeepLinkHomepageActivity || this instanceof DeepLinkHomepageActivityInternal)) { Log.e(TAG, "Not a deep link component"); finish(); return; } if (!WizardManagerHelper.isUserSetupComplete(this)) { Log.e(TAG, "Cancel deep link before SUW completed"); finish(); return; } final Intent intent = getIntent(); final String intentUriString = intent.getStringExtra( EXTRA_SETTINGS_EMBEDDED_DEEP_LINK_INTENT_URI); if (TextUtils.isEmpty(intentUriString)) { Log.e(TAG, "No EXTRA_SETTINGS_EMBEDDED_DEEP_LINK_INTENT_URI to deep link"); finish(); return; } final Intent targetIntent; try { targetIntent = Intent.parseUri(intentUriString, Intent.URI_INTENT_SCHEME); } catch (URISyntaxException e) { Log.e(TAG, "Failed to parse deep link intent: " + e); finish(); return; } targetIntent.setData(intent.getParcelableExtra( SettingsHomepageActivity.EXTRA_SETTINGS_LARGE_SCREEN_DEEP_LINK_INTENT_DATA)); final ComponentName targetComponentName = targetIntent.resolveActivity(getPackageManager()); if (targetComponentName == null) { Log.e(TAG, "No valid target for the deep link intent: " + targetIntent); finish(); return; } ActivityInfo targetActivityInfo; try { targetActivityInfo = getPackageManager().getActivityInfo(targetComponentName, /* flags= */ 0); } catch (PackageManager.NameNotFoundException e) { Log.e(TAG, "Failed to get target ActivityInfo: " + e); finish(); return; } UserHandle user = intent.getParcelableExtra(EXTRA_USER_HANDLE, UserHandle.class); String caller = getInitialReferrer(); int callerUid = -1; if (caller != null) { try { callerUid = getPackageManager().getApplicationInfoAsUser(caller, ApplicationInfoFlags.of(/* flags= */ 0), user != null ? user.getIdentifier() : getUserId()).uid; } catch (PackageManager.NameNotFoundException e) { Log.e(TAG, "Not able to get callerUid: " + e); finish(); return; } } if (!hasPrivilegedAccess(caller, callerUid, targetActivityInfo.packageName)) { if (!targetActivityInfo.exported) { Log.e(TAG, "Target Activity is not exported"); finish(); return; } if (!isCallingAppPermitted(targetActivityInfo.permission, callerUid)) { Log.e(TAG, "Calling app must have the permission of deep link Activity"); finish(); return; } } // Only allow FLAG_GRANT_READ/WRITE_URI_PERMISSION if calling app has the permission to // access specified Uri. int uriPermissionFlags = targetIntent.getFlags() & (Intent.FLAG_GRANT_READ_URI_PERMISSION | Intent.FLAG_GRANT_WRITE_URI_PERMISSION); if (targetIntent.getData() != null && uriPermissionFlags != 0 && checkUriPermission(targetIntent.getData(), /* pid= */ -1, callerUid, uriPermissionFlags) == PackageManager.PERMISSION_DENIED) { Log.e(TAG, "Calling app must have the permission to access Uri and grant permission"); finish(); return; } targetIntent.setComponent(targetComponentName); // To prevent launchDeepLinkIntentToRight again for configuration change. intent.setAction(null); targetIntent.removeFlags(Intent.FLAG_ACTIVITY_NEW_TASK | Intent.FLAG_ACTIVITY_NEW_DOCUMENT); targetIntent.addFlags(Intent.FLAG_ACTIVITY_FORWARD_RESULT); // Sender of intent may want to send intent extra data to the destination of targetIntent. targetIntent.replaceExtras(intent); targetIntent.putExtra(EXTRA_IS_FROM_SETTINGS_HOMEPAGE, true); targetIntent.putExtra(SettingsActivity.EXTRA_IS_FROM_SLICE, false); // Set 2-pane pair rule for the deep link page. ActivityEmbeddingRulesController.registerTwoPanePairRule(this, new ComponentName(getApplicationContext(), getClass()), targetComponentName, targetIntent.getAction(), SplitRule.FinishBehavior.ALWAYS, SplitRule.FinishBehavior.ALWAYS, true /* clearTop */); ActivityEmbeddingRulesController.registerTwoPanePairRule(this, new ComponentName(getApplicationContext(), Settings.class), targetComponentName, targetIntent.getAction(), SplitRule.FinishBehavior.ALWAYS, SplitRule.FinishBehavior.ALWAYS, true /* clearTop */); if (user != null) { startActivityAsUser(targetIntent, user); } else { startActivity(targetIntent); } } // Check if the caller has privileged access to launch the target page. private boolean hasPrivilegedAccess(String callerPkg, int callerUid, String targetPackage) { if (TextUtils.equals(callerPkg, getPackageName())) { return true; } int targetUid = -1; try { targetUid = getPackageManager().getApplicationInfo(targetPackage, ApplicationInfoFlags.of(/* flags= */ 0)).uid; } catch (PackageManager.NameNotFoundException e) { Log.e(TAG, "Not able to get targetUid: " + e); return false; } // When activityInfo.exported is false, Activity still can be launched if applications have // the same user ID. if (UserHandle.isSameApp(callerUid, targetUid)) { return true; } // When activityInfo.exported is false, Activity still can be launched if calling app has // root or system privilege. int callingAppId = UserHandle.getAppId(callerUid); if (callingAppId == Process.ROOT_UID || callingAppId == Process.SYSTEM_UID) { return true; } return false; } @VisibleForTesting String getInitialReferrer() { String referrer = getCurrentReferrer(); if (!TextUtils.equals(referrer, getPackageName())) { return referrer; } String initialReferrer = getIntent().getStringExtra(EXTRA_INITIAL_REFERRER); return TextUtils.isEmpty(initialReferrer) ? referrer : initialReferrer; } @VisibleForTesting String getCurrentReferrer() { Intent intent = getIntent(); // Clear extras to get the real referrer intent.removeExtra(Intent.EXTRA_REFERRER); intent.removeExtra(Intent.EXTRA_REFERRER_NAME); Uri referrer = getReferrer(); return referrer != null ? referrer.getHost() : null; } @VisibleForTesting boolean isCallingAppPermitted(String permission, int callerUid) { return TextUtils.isEmpty(permission) || checkPermission(permission, /* pid= */ -1, callerUid) == PackageManager.PERMISSION_GRANTED; } private String getHighlightMenuKey() { final Intent intent = getIntent(); if (intent != null && TextUtils.equals(intent.getAction(), ACTION_SETTINGS_EMBED_DEEP_LINK_ACTIVITY)) { final String menuKey = intent.getStringExtra( EXTRA_SETTINGS_EMBEDDED_DEEP_LINK_HIGHLIGHT_MENU_KEY); if (!TextUtils.isEmpty(menuKey)) { return maybeRemapMenuKey(menuKey); } } return getString(DEFAULT_HIGHLIGHT_MENU_KEY); } private String maybeRemapMenuKey(String menuKey) { boolean isPrivacyOrSecurityMenuKey = getString(R.string.menu_key_privacy).equals(menuKey) || getString(R.string.menu_key_security).equals(menuKey); boolean isSafetyCenterMenuKey = getString(R.string.menu_key_safety_center).equals(menuKey); if (isPrivacyOrSecurityMenuKey && SafetyCenterManagerWrapper.get().isEnabled(this)) { return getString(R.string.menu_key_safety_center); } if (isSafetyCenterMenuKey && !SafetyCenterManagerWrapper.get().isEnabled(this)) { // We don't know if security or privacy, default to security as it is above. return getString(R.string.menu_key_security); } return menuKey; } private void reloadHighlightMenuKey() { mMainFragment.getArguments().putString(SettingsActivity.EXTRA_FRAGMENT_ARG_KEY, getHighlightMenuKey()); mMainFragment.reloadHighlightMenuKey(); } private void initHomepageContainer() { final View view = findViewById(R.id.homepage_container); // Prevent inner RecyclerView gets focus and invokes scrolling. view.setFocusableInTouchMode(true); view.requestFocus(); } private void updateHomepageAppBar() { if (Flags.homepageRevamp() || !mIsEmbeddingActivityEnabled) { return; } updateAppBarMinHeight(); if (mIsTwoPane) { findViewById(R.id.homepage_app_bar_regular_phone_view).setVisibility(View.GONE); findViewById(R.id.homepage_app_bar_two_pane_view).setVisibility(View.VISIBLE); findViewById(R.id.suggestion_container_two_pane).setVisibility(View.VISIBLE); } else { findViewById(R.id.homepage_app_bar_regular_phone_view).setVisibility(View.VISIBLE); findViewById(R.id.homepage_app_bar_two_pane_view).setVisibility(View.GONE); findViewById(R.id.suggestion_container_two_pane).setVisibility(View.GONE); } } private void updateHomepagePaddings() { if (Flags.homepageRevamp() || !mIsEmbeddingActivityEnabled) { return; } if (mIsTwoPane) { int padding = getResources().getDimensionPixelSize( R.dimen.homepage_padding_horizontal_two_pane); mMainFragment.setPaddingHorizontal(padding); } else { mMainFragment.setPaddingHorizontal(0); } mMainFragment.updatePreferencePadding(mIsTwoPane); } private void updateAppBarMinHeight() { if (Flags.homepageRevamp()) { return; } final int searchBarHeight = getResources().getDimensionPixelSize(R.dimen.search_bar_height); final int margin = getResources().getDimensionPixelSize( mIsEmbeddingActivityEnabled && mIsTwoPane ? R.dimen.homepage_app_bar_padding_two_pane : R.dimen.search_bar_margin); findViewById(R.id.app_bar_container).setMinimumHeight(searchBarHeight + margin * 2); } private static class SuggestionFragCreator implements FragmentCreator { private final Class mClass; private final boolean mIsTwoPaneLayout; SuggestionFragCreator(Class clazz, boolean isTwoPaneLayout) { mClass = clazz; mIsTwoPaneLayout = isTwoPaneLayout; } @Override public Fragment create() { try { Fragment fragment = mClass.getConstructor().newInstance(); return fragment; } catch (Exception e) { Log.w(TAG, "Cannot show fragment", e); } return null; } @Override public void init(Fragment fragment) { if (fragment instanceof SplitLayoutListener) { ((SplitLayoutListener) fragment).setSplitLayoutSupported(mIsTwoPaneLayout); } } } /** The callback invoked while AE splitting. */ private static class SplitInfoCallback implements Consumer> { private final SettingsHomepageActivity mActivity; private boolean mIsSplitUpdatedUI = false; SplitInfoCallback(SettingsHomepageActivity activity) { mActivity = activity; } @Override public void accept(List splitInfoList) { if (!splitInfoList.isEmpty() && !mIsSplitUpdatedUI && !mActivity.isFinishing() && ActivityEmbeddingUtils.isAlreadyEmbedded(mActivity)) { mIsSplitUpdatedUI = true; mActivity.updateHomepageUI(); } } } }