Setting模块的界面渲染的时候说过,setting的主界面(一级菜单),是通过DashboardSummary来加载的,DashboardSmmary作为设置主界面的顶部容器,自然与数据的加载也有关。DashboardSummary是Fragment的子类,所以也要执行onCreate()方法:
data:image/s3,"s3://crabby-images/03b7a/03b7ae0711d6a23d684dcb2f33af08ff4332af03" alt=""
可以看到在onCreate()方法里面有两个数据提供者mDashboardFeatureProvider,mSuggestionFeatureProvider,这两个是很重要的数据提供者,但是他们两个获取的数据是不同的,这里着重看一下mDashboardFeatureProvider,它提供的数据是一级菜单如”电池”,”显示”,”网络和互联网”等。
FeatureFactory其为单例模式,返回的是FeatureFactoryImpl对象,而FeatureFactoryImpl对象的getDashboardFeatureProvider方法,返回的是DashboardFeatureProviderImpl对象: \packages\apps\settings\src\com\android\settings\overlay\FeatureFactoryImpl.javadata:image/s3,"s3://crabby-images/4e52b/4e52b8f6a765da3c12ad41c53ca1f4e1ad6d6b7a" alt=""
然后执行:
data:image/s3,"s3://crabby-images/f8833/f883307879bcc1f1320855bbc4733265b2a9386c" alt=""
其传入的参数为 当前的activity 以及 CategoryKey.CATEGORY_HOMEPAGE字串
data:image/s3,"s3://crabby-images/4f82c/4f82c3450c5f90c8bb20e1e5553ea65da50cb668" alt=""
可以看到Category,类别对象是通过DashboardFeatureProviderImpl的getTilesForCategory方法,因为DashboardFeatureProviderImpl是mDashboardFeatureProvider的实现
packages\apps\settings\src\com\android\settings\dashboard\DashboardFeatureProviderImpl.javaDashboardFeatureProviderImpl:
data:image/s3,"s3://crabby-images/38e0f/38e0fc2c9993f0cb41c5d9d6defe4e6050d65949" alt=""
mCategoryManager的初始化时在DashboardFeatureProviderImpl的构造方法中,使用CategoryManager的静态方法get获取的:
data:image/s3,"s3://crabby-images/eecec/eecece0a5aa85d7d76788bf0305a898e94871085" alt=""
data:image/s3,"s3://crabby-images/29527/29527e462f41b5d537d168e9c2b98ddcf4e1949a" alt=""
其构造方法初始化一些参数,再看CategoryManager的getTilesByCategory方法:
data:image/s3,"s3://crabby-images/c2ece/c2ece625f8b703be4f5ec6bb8bc7562817629211" alt=""
然后调用了tryInitCategories方法:
data:image/s3,"s3://crabby-images/963a3/963a3d5ca3d9788d7e962e6443b728197b4d0f9b" alt=""
由于是第一次进入,mCategories并没有初始化过,因此会走入if条件,然后调用TileUtils.java的getCategories方法,初始化mCategories的数据,注意传入的参数为 context
frameworks\base\packages\settingslib\src\com\android\settingslib\drawer\TileUtils.javadata:image/s3,"s3://crabby-images/c7d08/c7d08281431dd825f6192c09b3ecae09c5f7604a" alt=""
data:image/s3,"s3://crabby-images/270dc/270dc037721dfbdbfcd3bcf3e7ceef3b036d4899" alt=""
现在来详细看看getCategories方法:
- 首先调用调用了getTilesForAction()方法
data:image/s3,"s3://crabby-images/9c084/9c0848b3f8663eaaf8e50dcced72e66b747dd224" alt=""
- 调用getTilesForIntent方法
data:image/s3,"s3://crabby-images/bac33/bac33bab7772a66f9b3d8bffe403d009f9e33727" alt=""
data:image/s3,"s3://crabby-images/a3437/a3437e7c22feb2b9bcad1d21380b49b6e4283989" alt=""
data:image/s3,"s3://crabby-images/aad4b/aad4bf8b52d16744d794e1f15d56d880543bd219" alt=""
- getTilesForIntent中调用updateTileData方法:
data:image/s3,"s3://crabby-images/a53bb/a53bbe5d901ce287f8a81328a60c6d68034402d3" alt=""
data:image/s3,"s3://crabby-images/7538f/7538fc2f9105db254e5c4c30740f826c97d34db3" alt=""
data:image/s3,"s3://crabby-images/5f9de/5f9de2b5600c02675620bead2f4d064b6058432c" alt=""
data:image/s3,"s3://crabby-images/e6c4f/e6c4f823c915ba69cb5a69fda5fc8409ec3281ca" alt=""
data:image/s3,"s3://crabby-images/8f0a7/8f0a773d1816cbe352e735ca919022aad54a92e1" alt=""
- getTilesForIntent中调用addedCache.put(key, tile)方法
data:image/s3,"s3://crabby-images/e3819/e381961ed35d817c2c11d55aedfd8c8f23231ee7" alt=""
- 回到getCategories方法,在getTilesForAction方法后执行了createCategory方法
data:image/s3,"s3://crabby-images/d71ec/d71ec3ca020bd6bc50d429331b5b5ac4b87645d0" alt=""
(1)创建DashboardCategory对象
(2)利用PM查询所有含有Tile对象categoriyKey生成的intent对象的ResolveInfo集合
(3)把acitivity label值赋值给category title属性
(4)把解析intent-filter标签的priority值赋值给category属性
通过createCategory方法,创建DashboardCategory,并且根据tile的category来进行分类,最终getCategories方法会返回包含多个DashboardCategory的ArrayList
data:image/s3,"s3://crabby-images/407c1/407c16ee0760718d80bb4a7e4953dadd115635dd" alt=""
除了注册了一个广播接收器,还执行了一个异步操作:new CategoriesUpdateTask().execute()
data:image/s3,"s3://crabby-images/d9a74/d9a74a0dcc47bbb229731ea58f94b3aaae950825" alt=""
可以看到在异步任务里面除了调用了reloadAllCategories()方法一直加载数据以外,还调用了接口CategoryListener的唯一方法onCategoriesChanged(),那么作为界面容器的DashboardSummary肯定重载了这个接口,实现了onCategoriesChanged()方法,回到DashboardSummary:
data:image/s3,"s3://crabby-images/80aa6/80aa6790df56ff809d4688554a03ba61a7a1c503" alt=""
在onCategoriesChanged中调用了rebuildUI方法,从注释中可以看到,其实第一次启动时,在DashboardSummary的onViewCreated的方法中也调用了rebuildUI()
data:image/s3,"s3://crabby-images/c04ce/c04ceb815cf4a1cbede1c2e0049d38d36deec7d3" alt=""
onViewCreated的两个作用
1. 初始化DashboardAdapter, 并设置到mDashboard中
2. 重新更新UI
现在就来看看rebuildUI()方法:
data:image/s3,"s3://crabby-images/e30a9/e30a9b6c57211bad8b01859339fd07f3415be548" alt=""
在rebuildUI中无论条件判断进入那个分支,都会执行updateCategoryAndSuggestion方法,不论是否支持Suggestion,最终都会调用updateCategoryAndSuggestion方法,并且传入的参数为null
data:image/s3,"s3://crabby-images/ebd3a/ebd3a0572ed8ab2b9b9f54bb0063c52e1de39a88" alt=""
调用DashboardAdapter的setCategory方法,设置获取到的DashboardCategory
data:image/s3,"s3://crabby-images/7992f/7992fff436a7f898a55c9a3f3624ddf6dce46f95" alt=""
最后调用DashboardAdapter的setCategory方法,设置获取到的DashboardCategory
可以知道第一级菜单完全是动态加载的DashboardSummary在onCreateView加载的布局R.layout.dashboard,是一个recycleview
data:image/s3,"s3://crabby-images/8f9fc/8f9fcfa3e6130fbf08f347df38322bcbfa1bfb93" alt=""
data:image/s3,"s3://crabby-images/24968/24968ec59a27e4c16ea36e917ec74b437c8231c3" alt=""
在onViewCreated方法获取到recycleview控件,然后设置获取到的DashboardAdapter
data:image/s3,"s3://crabby-images/81b4f/81b4f390eaf48885377a65b0a42ba6c397f589df" alt=""
有两种情况,一种是一级菜单是activty,还有一种是一级菜单是fragment
一级菜单项的实现是activity因为是通过activity实现的,通过直接继承activity,实现比较简单 首先,在AndroidManifest.xml清单中添加要添加的菜单项,以HardKey.java为例
data:image/s3,"s3://crabby-images/88bee/88bee2e0333440fa470d4a2dca2e5a0da205798b" alt=""
然后,再自己实现HardKey.java这个activity,自己实现他的布局和具体逻辑。
一级菜单项的实现是fragment首先也是在AndroidManifest.xml清单中添加:
data:image/s3,"s3://crabby-images/55255/55255c84ac92b58e711cf76432adb6a1b69936f4" alt=""
然后自己实现FlashSettingsActivity.java这个fragment 最后在Setting.java里面添加:
public static class FlashSettingsActivity extends SettingsActivity { /* empty */ } 总结:其实我们添加FlashSettingsActivity这个并不是一个activity,这只是SetttingsActivity的一个子类,而且是空实现,之所以要添加这个空的activity,是为了外部应用能够跳转到FlashSettingsActivity界面,如果是fragment的话,外部应用是无法直接跳转到fragment的,当点击跳转后,会执行SetttingsActivity中的方法,根据其在AndroidManifest清单文件中的注册信息,它的的mata-data信息,找到“com.android.settings.FRAGMENT_CLASS”,然后加载对应的fragment,一般在AndroidManifest中会声明称settings的内部类: android:name=”Settings$FlashSettingsActivity” 在尝试修改priority=”9”会发现,mmm模块编译安装到手机之后发现位置没有改变啊。其实这个priority=”9”的判断是在frameworks中进行处理的,所以只有编译frameworks之后才会生效。