日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術文章
文章詳情頁

分析Android 11.0Settings源碼之主界面加載

瀏覽:92日期:2022-09-19 15:18:58

本篇主要記錄AndroidR Settings源碼主界面加載流程,方便后續工作調試其流程。

Settings代碼路徑:

packages/app/Settings/

Settings代碼獲?。?p>Setting 源碼下載地址:https://github.com/aosp-mirror/platform_packages_apps_settingsgit地址:https://github.com/aosp-mirror/platform_packages_apps_settings.git

主界面加載:

首先我們來看 Settings 模塊中的 AndroidManifest.xml 文件,找到默認啟動入口Activity信息:

<activity android:name='.homepage.SettingsHomepageActivity' android:label='@string/settings_label_launcher' android:theme='@style/Theme.Settings.Home' android:taskAffinity='com.android.settings.root' android:launchMode='singleTask' android:configChanges='keyboard|keyboardHidden'><intent-filter android:priority='1'> <action android:name='android.settings.SETTINGS' /> <category android:name='android.intent.category.DEFAULT' /></intent-filter><meta-data android:name='com.android.settings.PRIMARY_PROFILE_CONTROLLED' android:value='true' /></activity>//activity-alias可用來設置某個Activity的快捷入口,可以放在桌面上或者通過該別名被其他組件快速調起。 //android:targetActivity為目標Activity. <!-- Alias for launcher activity only, as this belongs to each profile. --><activity-alias android:name='Settings' android:label='@string/settings_label_launcher' android:taskAffinity='com.android.settings.root' android:launchMode='singleTask' android:targetActivity='.homepage.SettingsHomepageActivity'><intent-filter> <action android:name='android.intent.action.MAIN' /> <category android:name='android.intent.category.DEFAULT' /> <category android:name='android.intent.category.LAUNCHER' /></intent-filter><meta-data android:name='android.app.shortcuts' android:resource='@xml/shortcuts'/></activity-alias>

可以看到Settings的桌面圖標啟動的主界面是Settings.java,但其xml定義了targetActivity屬性,實質應是SettingsHomepageActivity.java,從onCreate()方法開始:

@Overrideprotected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.settings_homepage_container); final View root = findViewById(R.id.settings_homepage_container); root.setSystemUiVisibility( View.SYSTEM_UI_FLAG_LAYOUT_HIDE_NAVIGATION | View.SYSTEM_UI_FLAG_LAYOUT_STABLE); setHomepageContainerPaddingTop(); final Toolbar toolbar = findViewById(R.id.search_action_bar); FeatureFactory.getFactory(this).getSearchFeatureProvider() .initSearchToolbar(this /* activity */, toolbar, SettingsEnums.SETTINGS_HOMEPAGE); final ImageView avatarView = findViewById(R.id.account_avatar); getLifecycle().addObserver(new AvatarViewMixin(this, avatarView)); getLifecycle().addObserver(new HideNonSystemOverlayMixin(this)); if (!getSystemService(ActivityManager.class).isLowRamDevice()) {// Only allow contextual feature on high ram devices.showFragment(new ContextualCardsFragment(), R.id.contextual_cards_content); } showFragment(new TopLevelSettings(), R.id.main_content); ((FrameLayout) findViewById(R.id.main_content)) .getLayoutTransition().enableTransitionType(LayoutTransition.CHANGING);}

可以看到主界面的layout為settings_homepage_container.xml:

<androidx.coordinatorlayout.widget.CoordinatorLayout xmlns:android='http://schemas.android.com/apk/res/android' xmlns:app='http://schemas.android.com/apk/res-auto' android: android:fitsSystemWindows='true' android:layout_width='match_parent' android:layout_height='match_parent'> <androidx.core.widget.NestedScrollViewandroid: android:layout_width='match_parent'android:layout_height='match_parent'app:layout_behavior='com.android.settings.widget.FloatingAppBarScrollingViewBehavior'><LinearLayout android: android:layout_width='match_parent' android:layout_height='wrap_content' android:orientation='vertical'> <FrameLayoutandroid: android:layout_width='match_parent'android:layout_height='wrap_content'android:layout_marginStart='@dimen/contextual_card_side_margin'android:layout_marginEnd='@dimen/contextual_card_side_margin'/> <FrameLayoutandroid: android:layout_width='match_parent'android:layout_height='wrap_content'android:animateLayoutChanges='true'android:background='?android:attr/windowBackground'/></LinearLayout> </androidx.core.widget.NestedScrollView> <com.google.android.material.appbar.AppBarLayoutandroid:layout_width='match_parent'android:layout_height='wrap_content'android:touchscreenBlocksFocus='false'android:keyboardNavigationCluster='false'><include layout='@layout/search_bar'/> </com.google.android.material.appbar.AppBarLayout></androidx.coordinatorlayout.widget.CoordinatorLayout>

主界面布局中主要包含兩部分:一個頂部快捷搜索欄,一個Id為main_content的FrameLayout就是用來顯示主設置內容的,即Settings的一級菜單項界面?;氐給nCreate()方法:

showFragment(new TopLevelSettings(), R.id.main_content);

可以看到一級菜單啟動的是TopLevelSettings,TopLevelSettings繼承于DashboardFragment.java:

public class TopLevelSettings extends DashboardFragment implementsPreferenceFragmentCompat.OnPreferenceStartFragmentCallback

TopLevelSettings的構造方法:

public TopLevelSettings() { final Bundle args = new Bundle(); // Disable the search icon because this page uses a full search view in actionbar. args.putBoolean(NEED_SEARCH_ICON_IN_ACTION_BAR, false); setArguments(args);}

可以看到通過構造方法傳遞了一個參數,從注釋中可以看出,該參數的用意是由于主界面使用完整的搜索視圖所以在主界面的actionbar中隱藏了搜索圖標。然后再根據framgments生命周期先來看onAttach()方法:

@Overridepublic void onAttach(Context context) { super.onAttach(context); use(SupportPreferenceController.class).setActivity(getActivity());}

調用父類DashboardFragment.java的onAttach()方法:

@Overridepublic void onAttach(Context context) { super.onAttach(context); mSuppressInjectedTileKeys = Arrays.asList(context.getResources().getStringArray( R.array.config_suppress_injected_tile_keys)); mDashboardFeatureProvider = FeatureFactory.getFactory(context). getDashboardFeatureProvider(context); // Load preference controllers from code final List<AbstractPreferenceController> controllersFromCode = createPreferenceControllers(context); // Load preference controllers from xml definition final List<BasePreferenceController> controllersFromXml = PreferenceControllerListHelper .getPreferenceControllersFromXml(context, getPreferenceScreenResId()); // Filter xml-based controllers in case a similar controller is created from code already. final List<BasePreferenceController> uniqueControllerFromXml = PreferenceControllerListHelper.filterControllers( controllersFromXml, controllersFromCode); // Add unique controllers to list. if (controllersFromCode != null) {mControllers.addAll(controllersFromCode); } mControllers.addAll(uniqueControllerFromXml); // And wire up with lifecycle. final Lifecycle lifecycle = getSettingsLifecycle(); uniqueControllerFromXml.forEach(controller -> {if (controller instanceof LifecycleObserver) { lifecycle.addObserver((LifecycleObserver) controller);} }); // Set metrics category for BasePreferenceController. final int metricCategory = getMetricsCategory(); mControllers.forEach(controller -> {if (controller instanceof BasePreferenceController) { ((BasePreferenceController) controller).setMetricsCategory(metricCategory);} }); mPlaceholderPreferenceController = new DashboardTilePlaceholderPreferenceController(context); mControllers.add(mPlaceholderPreferenceController); for (AbstractPreferenceController controller : mControllers) {addPreferenceController(controller); }}

通過方法注釋可以得知此方法主要是完成preference controllers的加載。DashboardFragment.java的onCreate()方法:

@Overridepublic void onCreate(Bundle icicle) { super.onCreate(icicle); // Set ComparisonCallback so we get better animation when list changes. getPreferenceManager().setPreferenceComparisonCallback( new PreferenceManager.SimplePreferenceComparisonCallback()); if (icicle != null) {// Upon rotation configuration change we need to update preference states before any// editing dialog is recreated (that would happen before onResume is called).updatePreferenceStates(); }}

設置ComparisonCallback,以便在列表更改時獲得更好的動畫效果。第一次進入時,icicle為null,根據log定位發現,其后調用DashboardFragment.java的onCreatePreferences()方法:

@Overridepublic void onCreatePreferences(Bundle savedInstanceState, String rootKey) { checkUiBlocker(mControllers); refreshAllPreferences(getLogTag()); mControllers.stream() .map(controller -> (Preference) findPreference(controller.getPreferenceKey())) .filter(Objects::nonNull) .forEach(preference -> {// Give all controllers a chance to handle click.preference.getExtras().putInt(CATEGORY, getMetricsCategory()); });}

調用refreshAllPreferences():

/** * Refresh all preference items, including both static prefs from xml, and dynamic items from * DashboardCategory. */private void refreshAllPreferences(final String tag) { final PreferenceScreen screen = getPreferenceScreen(); // First remove old preferences. if (screen != null) {// Intentionally do not cache PreferenceScreen because it will be recreated later.screen.removeAll(); } // Add resource based tiles. displayResourceTiles(); refreshDashboardTiles(tag); final Activity activity = getActivity(); if (activity != null) {Log.d(tag, 'All preferences added, reporting fully drawn');activity.reportFullyDrawn(); } updatePreferenceVisibility(mPreferenceControllers);}

刷新所有preference items,包括來自xml的靜態preference和來自DashboardCategory的動態preference,靜態xml定義的prefs(調用displayResourceTiles()方法),動態DashboardCategory動態加載(調用refreshDashboardTiles(TAG)方法,其中TAG為 “TopLevelSettings”)。displayResourceTiles():此方法主要是從xml資源文件中加載顯示prefs:

/** * Displays resource based tiles. */private void displayResourceTiles() { final int resId = getPreferenceScreenResId(); if (resId <= 0) {return; } addPreferencesFromResource(resId); final PreferenceScreen screen = getPreferenceScreen(); screen.setOnExpandButtonClickListener(this); displayResourceTilesToScreen(screen);}/** * Perform {@link AbstractPreferenceController#displayPreference(PreferenceScreen)} * on all {@link AbstractPreferenceController}s. */protected void displayResourceTilesToScreen(PreferenceScreen screen) { mPreferenceControllers.values().stream().flatMap(Collection::stream).forEach( controller -> controller.displayPreference(screen));}靜態加載

首先調用getPreferenceScreenResId()方法獲取所要加載的xml的ID,然后調用子類TopLevelSettings.java的getPreferenceScreenResId()方法:

@Overrideprotected int getPreferenceScreenResId() { return R.xml.top_level_settings;}

可以看到Settings主界面加載的xml文件是top_level_settings:

<PreferenceScreen xmlns:android='http://schemas.android.com/apk/res/android' xmlns:settings='http://schemas.android.com/apk/res-auto' android:key='top_level_settings'> <Preferenceandroid:key='top_level_network'android:title='@string/network_dashboard_title'android:summary='@string/summary_placeholder'android:icon='@drawable/ic_homepage_network'android:order='-120'android:fragment='com.android.settings.network.NetworkDashboardFragment'settings:controller='com.android.settings.network.TopLevelNetworkEntryPreferenceController'/> <Preferenceandroid:key='top_level_connected_devices'android:title='@string/connected_devices_dashboard_title'android:summary='@string/summary_placeholder'android:icon='@drawable/ic_homepage_connected_device'android:order='-110'android:fragment='com.android.settings.connecteddevice.ConnectedDeviceDashboardFragment'settings:controller='com.android.settings.connecteddevice.TopLevelConnectedDevicesPreferenceController'/> <Preferenceandroid:key='top_level_apps_and_notifs'android:title='@string/app_and_notification_dashboard_title'android:summary='@string/app_and_notification_dashboard_summary'android:icon='@drawable/ic_homepage_apps'android:order='-100'android:fragment='com.android.settings.applications.AppAndNotificationDashboardFragment'/> <Preferenceandroid:key='top_level_battery'android:title='@string/power_usage_summary_title'android:summary='@string/summary_placeholder'android:icon='@drawable/ic_homepage_battery'android:fragment='com.android.settings.fuelgauge.PowerUsageSummary'android:order='-90'settings:controller='com.android.settings.fuelgauge.TopLevelBatteryPreferenceController'/> <Preferenceandroid:key='top_level_display'android:title='@string/display_settings'android:summary='@string/summary_placeholder'android:icon='@drawable/ic_homepage_display'android:order='-80'android:fragment='com.android.settings.DisplaySettings'settings:controller='com.android.settings.display.TopLevelDisplayPreferenceController'/> <Preferenceandroid:key='top_level_sound'android:title='@string/sound_settings'android:summary='@string/sound_dashboard_summary'android:icon='@drawable/ic_homepage_sound'android:order='-70'android:fragment='com.android.settings.notification.SoundSettings'/> <Preferenceandroid:key='top_level_storage'android:title='@string/storage_settings'android:summary='@string/summary_placeholder'android:icon='@drawable/ic_homepage_storage'android:order='-60'android:fragment='com.android.settings.deviceinfo.StorageSettings'settings:controller='com.android.settings.deviceinfo.TopLevelStoragePreferenceController'/> <Preferenceandroid:key='top_level_privacy'android:title='@string/privacy_dashboard_title'android:summary='@string/privacy_dashboard_summary'android:icon='@drawable/ic_homepage_privacy'android:order='-55'android:fragment='com.android.settings.privacy.PrivacyDashboardFragment'/> <Preferenceandroid:key='top_level_location'android:title='@string/location_settings_title'android:summary='@string/location_settings_loading_app_permission_stats'android:icon='@drawable/ic_homepage_location'android:order='-50'android:fragment='com.android.settings.location.LocationSettings'settings:controller='com.android.settings.location.TopLevelLocationPreferenceController'/> <Preferenceandroid:key='top_level_security'android:title='@string/security_settings_title'android:summary='@string/summary_placeholder'android:icon='@drawable/ic_homepage_security'android:order='-40'android:fragment='com.android.settings.security.SecuritySettings'settings:controller='com.android.settings.security.TopLevelSecurityEntryPreferenceController'/> <Preferenceandroid:key='top_level_accounts'android:title='@string/account_dashboard_title'android:summary='@string/summary_placeholder'android:icon='@drawable/ic_homepage_accounts'android:order='-30'android:fragment='com.android.settings.accounts.AccountDashboardFragment'settings:controller='com.android.settings.accounts.TopLevelAccountEntryPreferenceController'/> <Preferenceandroid:key='top_level_accessibility'android:title='@string/accessibility_settings'android:summary='@string/accessibility_settings_summary'android:icon='@drawable/ic_homepage_accessibility'android:order='-20'android:fragment='com.android.settings.accessibility.AccessibilitySettings'settings:controller='com.android.settings.accessibility.TopLevelAccessibilityPreferenceController'/> <Preferenceandroid:key='top_level_system'android:title='@string/header_category_system'android:summary='@string/system_dashboard_summary'android:icon='@drawable/ic_homepage_system_dashboard'android:order='10'android:fragment='com.android.settings.system.SystemDashboardFragment'/> <Preferenceandroid:key='top_level_about_device'android:title='@string/about_settings'android:summary='@string/summary_placeholder'android:icon='@drawable/ic_homepage_about'android:order='20'android:fragment='com.android.settings.deviceinfo.aboutphone.MyDeviceInfoFragment'settings:controller='com.android.settings.deviceinfo.aboutphone.TopLevelAboutDevicePreferenceController'/> <Preferenceandroid:key='top_level_support'android:summary='@string/support_summary'android:title='@string/page_tab_title_support'android:icon='@drawable/ic_homepage_support'android:order='100'settings:controller='com.android.settings.support.SupportPreferenceController'/></PreferenceScreen>

可以看到主要配置的是一些Preference菜單項如網絡和互聯網、已連接的設備、應用和通知、電池等等,Preference的配置含義:

 key:唯一性ID; title:標題; summary:簡介; ico:圖標; order:加載顯示優先級,order為負時,絕對值越高,界面顯示越靠前;order為正時,值越高,顯示越靠后; fragment:點擊此preference所跳轉的fragment界面; controller:控制管理類。  動態加載

refreshDashboardTiles

總結: Settings的主Activity實質實現是在SettingsHomepageActivity.java內; Settings的主界面設置item的顯示是在fragment上,fragment為TopLevelSettings.java,加載顯示的布局為top_level_settings.xml; Settings主界面設置項item的加載顯示主要分為兩部分,一部分是xml定義的靜態加載,xml為top_level_settings.xml;一部分是DashboardCategory來獲取動態加載; 每個設置項item均為一個preference,通過xml定義加載時,必須要有一個controller,可以是在xml中定義'settings:controller'屬性聲明,名稱必須與類的包名路徑相同;也可直接在相關fragment中實現createPreferenceControllers()方法去調用構造相關controller。此二者存其一即可。 xml中配置preference時,必須定義”android:key“屬性;

以上就是分析Android 11.0Settings源碼之主界面加載的詳細內容,更多關于Android 11.0Settings源碼的資料請關注好吧啦網其它相關文章!

標簽: Android
相關文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
亚洲第一精品影视| 日韩一级网站| 青青草国产成人99久久| 国产精品日本| 99香蕉国产精品偷在线观看 | 色爱综合av| 久久精品动漫| 久久九九99| 亚洲香蕉网站| 中国女人久久久| 国产一区二区中文| 一本一本久久| 伊人久久亚洲| 欧美亚洲综合视频| 久久69成人| 精品日韩视频| 99国产精品| 日韩国产精品久久久| 国产亚洲字幕| 久久国产精品色av免费看| 捆绑调教美女网站视频一区| 中文字幕在线视频久| 激情欧美丁香| 蜜桃91丨九色丨蝌蚪91桃色| 欧美在线观看天堂一区二区三区| 国产欧美精品| 国产成人精品福利| 国内精品99| 亚洲精品日本| 久久精品亚洲一区二区| 亚洲美女久久精品| 国产字幕视频一区二区| 蜜桃视频在线观看一区| 国产欧美一区| 久久婷婷久久| 四虎精品一区二区免费| 麻豆国产欧美一区二区三区 | 国产一区二区三区四区| 91精品二区| 亚洲色图国产| 精品亚洲a∨一区二区三区18| 91精品一区国产高清在线gif | 最新亚洲激情| 日韩av三区| 五月激情久久| 日韩黄色免费网站| 伊人久久av| 亚洲精选av| 国产精品久久观看| 激情偷拍久久| 日韩精品免费视频一区二区三区| 国产精品videossex| 亚洲性色视频| 国产精品久久久久久久久免费高清 | 国产毛片久久久| 91亚洲国产成人久久精品| 欧美1级日本1级| 欧美综合精品| 日韩欧美字幕| 国产欧美日韩一级| 国产尤物精品| 国产精品久久久久av蜜臀| 亚洲精品2区| 精品国产乱码| 亚洲自啪免费| 日韩欧美一区二区三区在线视频| 91午夜精品| 午夜久久黄色| 国产精品啊啊啊| 视频一区在线视频| 欧产日产国产精品视频| 国产剧情一区二区在线观看| 一区二区自拍| 日本少妇精品亚洲第一区| 久久中文字幕av| 久久久国产精品网站| 亚洲青青久久| 亚洲一级高清| 国产一区二区三区四区大秀 | 国产亚洲人成a在线v网站| 四虎4545www国产精品| 国产精品永久| 免费一级片91| 欧美日韩高清| 色一区二区三区| 911精品国产| 亚洲精品成人| www成人在线视频| 精品午夜av| 国产精品传媒麻豆hd| 最新日韩欧美| 在线观看精品| 美女在线视频一区| 日本欧美一区二区| 蜜桃国内精品久久久久软件9| 国际精品欧美精品| 欧美成人aaa| 国产三级一区| 日韩一区中文| 亚洲精品第一| 视频在线观看一区| 久久一区二区中文字幕| 久久久久久色 | 欧美另类中文字幕| 亚洲精品日本| 免费精品视频在线| 激情91久久| 亚洲二区视频| 激情国产在线| 国产欧美激情| 日韩av电影一区| 亚洲精品在线a| 亚洲精品一级二级三级| 99pao成人国产永久免费视频 | 欧美午夜三级| 日本不卡在线视频| 日本不卡高清| 日韩精品免费观看视频| 免费久久精品视频| 免费在线观看精品| 一区二区三区国产盗摄| 久久一级电影| 日本精品影院| 日韩欧美精品综合| 欧美天堂视频| 深夜福利视频一区二区| 国产在线观看www| 国产一区二区亚洲| 国产一区二区三区四区大秀| 麻豆一区二区99久久久久| 久久国产精品美女| 国产成人精选| 99精品美女| 91精品观看| 亚洲中午字幕| 日韩久久99| 国产精品白丝久久av网站| 欧美视频久久| 美女高潮久久久| 狠狠躁少妇一区二区三区| 成人免费网站www网站高清| 日韩毛片在线| 久久一区二区三区喷水| av不卡免费看| 亚洲精品免费观看| 国产欧美亚洲一区| 福利一区和二区| 1000部精品久久久久久久久| 91久久亚洲| 日韩av一区二区三区| 麻豆国产精品| 亚洲www啪成人一区二区| 久久久精品午夜少妇| 水蜜桃久久夜色精品一区的特点| 欧美一区91| 水蜜桃精品av一区二区| 久久网站免费观看| 亚洲精品一二三**| 美女久久99| 婷婷成人在线| 免费成人在线影院| 91成人精品观看| 国产伦久视频在线观看| 亚洲女人av| 欧美a在线观看| 午夜日本精品| 久久av影视| 国产粉嫩在线观看| 中文亚洲欧美| 久久99性xxx老妇胖精品| 久久九九精品| 日本成人在线一区| 欧美精品高清| 男女男精品视频网| 色婷婷亚洲mv天堂mv在影片| 亚洲综合不卡| 国产精品免费99久久久| 一二三区精品| 丝袜国产日韩另类美女| 日韩国产专区| 精品久久久中文字幕| 青青草精品视频| 国产成人精品一区二区三区视频 | 国产精品色婷婷在线观看| 国产麻豆综合| 播放一区二区| 福利一区二区| 蜜桃视频一区二区| 免费视频久久| 免费视频最近日韩| 日韩亚洲国产欧美| 欧美亚洲一级| 精品黄色一级片| 久久在线免费| 精品欧美视频| 久久久久91| 1000部精品久久久久久久久| 日韩三级久久| 国产精选在线| 亚洲有吗中文字幕|