Settings一级界面显示分析

Settings是系统提供各种设置的一个应用,源码位于packages/apps/Settings,平板上界面如下:

从上面抓取的布局来看,整体布局是RecyclerView+Fragment。

主要类

  • SettingsHomepageActivity.java:主界面,采用RecyclerView+Fragment结构
  • settings_homepage_container.xml:主界面布局文件
  • TopLevelSettings.java:主界面中Fragment
  • DashboardFragment.java:Settings中一个fragment基类,用来显示Prefrence列表
  • top_level_settings.xml: 主界面Prefrence布局文件
  • SubSettingLauncher.java:SubSettings启动类,点击Prefrence是启动SubSettings,SubSettings是一个Activity
  • SubSettings.java/SettingsActivity.java:展示Prefrence对应的fragment

创建主界面Activity

Mainfest中设置的LauncherActivity为Settings,这个是一个别名,实际指向的是SettingsHomepageActivity

// Settings\AndroidManifest.xml
<activity-alias android:name="Settings"
        android:label="@string/settings_label_launcher"
        android:taskAffinity="com.android.settings.root"
        android:launchMode="singleTask"
        android:exported="true"
        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>

SettingsHomepageActivity extends FragmentActivity,onCreate()中主要进行加载布局,创建Fragment,布局文件settings_homepage_container.xml和上面抓取的布局是可以对上的

// Settings\src\com\android\settings\homepage\SettingsHomepageActivity.java
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    // 1. 是否嵌入显示Activity
    mIsEmbeddingActivityEnabled = ActivityEmbeddingUtils.isEmbeddingActivityEnabled(this);
  
    // 2. 设置布局文件
    setContentView(R.layout.settings_homepage_container);
   
    // 3. 设置主界面背景、appbar高度、搜索栏等
    updateAppBarMinHeight();
    initHomepageContainer();
    updateHomepageAppBar();
    updateHomepageBackground();
    initSearchBarView();

    // 4. 不是低内存设备,显示建议条目
    if (!getSystemService(ActivityManager.class).isLowRamDevice()) {
        initAvatarView();
        showSuggestionFragment(scrollNeeded);
        
    }

    // 5.创建fragment
    mMainFragment = showFragment(() -> {
        final TopLevelSettings fragment = new TopLevelSettings();
        fragment.getArguments().putString(SettingsActivity.EXTRA_FRAGMENT_ARG_KEY,
                highlightMenuKey);
        return fragment;
    }, R.id.main_content);
}

创建主界面PrefrenceFragment

TopLevelSettings extends DashboardFragment,DashboardFragment是Settings中一个fragment基类,用来显示Prefrence列表。

TopLevelSettings主要是实现DashboardFragment的一些方法,如设置布局文件、左侧prefrence点击事件等,部分代码如下:

// Settings\src\com\android\settings\homepage\TopLevelSettings.java
// 设置prefrence布局
protected int getPreferenceScreenResId() {
    return R.xml.top_level_settings;
}
// onAttach
public void onAttach(Context context) {
    super.onAttach(context);
    HighlightableMenu.fromXml(context, getPreferenceScreenResId());
    use(SupportPreferenceController.class).setActivity(getActivity());
}
// 左侧prefrence点击事件
public boolean onPreferenceTreeClick(Preference preference) {
    if (isDuplicateClick(preference)) {
        return true;
    }
    ActivityEmbeddingRulesController.registerSubSettingsPairRule(getContext(),
            true /* clearTop */);
    setHighlightPreferenceKey(preference.getKey());
    return super.onPreferenceTreeClick(preference);
}
// 创建fragment
public boolean onPreferenceStartFragment(PreferenceFragmentCompat caller, Preference pref) {
    new SubSettingLauncher(getActivity())
            .setDestination(pref.getFragment())
            .setArguments(pref.getExtras())
            .setSourceMetricsCategory(caller instanceof Instrumentable
                    ? ((Instrumentable) caller).getMetricsCategory()
                    : Instrumentable.METRICS_CATEGORY_UNKNOWN)
            .setTitleRes(-1)
            .setIsSecondLayerPage(true)
            .launch();
    return true;
}

接下来,我们具体来看DashboardFragment,DashboardFragment继承层次比较多,具体如下,最终基类是PreferenceFragment,说明采用的Prefrence组件。Prefrence组件用法可以参考最后面博客。

TopLevelSettings --> DashboardFragment --> SettingsPreferenceFragment --> InstrumentedPreferenceFragment --> ObservablePreferenceFragment --> PreferenceFragmentCompat

onPreferenceTreeClick()回调逻辑:在InstrumentedPreferenceFragment给所有的Preference设置了点击事件,点击后会回调
onPreferenceTreeClick(),所以可以在该方法中判断相应的Prefrence进行对应的处理。

// Settings\src\com\android\settings\core\InstrumentedPreferenceFragment.java
public void onStart() {
    super.onStart();
    // Override the OnPreferenceTreeClickListener in super.onStart() to inject jank detection.
    getPreferenceManager().setOnPreferenceTreeClickListener((preference) -> {
        if (preference instanceof SwitchPreference) {
            SettingsJankMonitor.detectSwitchPreferenceClickJank(
                    getListView(), (SwitchPreference) preference);
        }
        return onPreferenceTreeClick(preference);
    });
}

getPreferenceScreenResId()逻辑:将getPreferenceScreenResId()返回的布局文件设置给了PreferenceFragment。所以,R.xml.top_level_settings就是主界面布局文件

// Settings\src\com\android\settings\core\InstrumentedPreferenceFragment.java
public void onCreatePreferences(Bundle savedInstanceState, String rootKey) {
    final int resId = getPreferenceScreenResId();
    if (resId > 0) {
        addPreferencesFromResource(resId);
    }
}

onPreferenceStartFragment()逻辑:onPreferenceStartFragment是PreferenceFragmentCompat.OnPreferenceStartFragmentCallback接口的方法,TopLevelSettings implements PreferenceFragmentCompat.OnPreferenceStartFragmentCallback,但好像没有看到registerOnPreferenceStartFragmentCallback,其实这个代码是在PreferenceFragmentCompat中,在Preference点击的时候,执行了onPreferenceStartFragment

// androidx/preference/preference/src/main/java/androidx/preference/PreferenceFragmentCompat.java
public boolean onPreferenceTreeClick(@NonNull Preference preference) {
    if (preference.getFragment() != null) {
        // 在Preference点击的时候,执行了onPreferenceStartFragment
        boolean handled = false;
        if (getCallbackFragment() instanceof OnPreferenceStartFragmentCallback) {
            handled = ((OnPreferenceStartFragmentCallback) getCallbackFragment())
                    .onPreferenceStartFragment(this, preference);
        }
        

        Fragment callbackFragment = this;
        while (!handled && callbackFragment != null) {
            if (callbackFragment instanceof OnPreferenceStartFragmentCallback) {
                handled = ((OnPreferenceStartFragmentCallback) callbackFragment)
                        .onPreferenceStartFragment(this, preference);
            }
            callbackFragment = callbackFragment.getParentFragment();
        }
        if (!handled && getContext() instanceof OnPreferenceStartFragmentCallback) {
            handled = ((OnPreferenceStartFragmentCallback) getContext())
                    .onPreferenceStartFragment(this, preference);
        }
        ...
        return true;
    }
    return false;
}

至此,我们已经清楚主界面左侧Preference列表的显示流程,但是右侧的Fragment的怎么显示出来的,我们还不知道。我们想一下,如果我们来用Preference+Fragment来实现左右双列显示,应该也是可以做的,比如两个FrameLayout,一个显示Preference列表Fragment,一个显示对应Preference条目Fragment

启动SubSettings

上面我们知道点击左侧preference条目时会执行onPreferenceStartFragment,接下来具体看代码

// Settings\src\com\android\settings\homepage\TopLevelSettings.java
// 创建fragment
public boolean onPreferenceStartFragment(PreferenceFragmentCompat caller, Preference pref) {
    new SubSettingLauncher(getActivity())
            .setDestination(pref.getFragment())
            .setArguments(pref.getExtras())
            .setSourceMetricsCategory(caller instanceof Instrumentable
                    ? ((Instrumentable) caller).getMetricsCategory()
                    : Instrumentable.METRICS_CATEGORY_UNKNOWN)
            .setTitleRes(-1)
            .setIsSecondLayerPage(true)
            .launch();
    return true;
}

SubSettingLauncher中将传过来的fragment等信息设置给intent,然后通过startActivity启动SubSettings

// Settings\src\com\android\settings\core\SubSettingLauncher.java
public class SubSettingLauncher {
    // 内部参数类
    private final LaunchRequest mLaunchRequest;
    // launch()通过startActivity启动SubSettings
    public void launch() {
        final Intent intent = toIntent();
        boolean launchAsUser = mLaunchRequest.mUserHandle != null
                && mLaunchRequest.mUserHandle.getIdentifier() != UserHandle.myUserId();
        boolean launchForResult = mLaunchRequest.mResultListener != null;
        if (launchAsUser && launchForResult) {
            launchForResultAsUser(intent, mLaunchRequest.mUserHandle,
                    mLaunchRequest.mResultListener, mLaunchRequest.mRequestCode);
        } else if (launchAsUser && !launchForResult) {
            launchAsUser(intent, mLaunchRequest.mUserHandle);
        } else if (!launchAsUser && launchForResult) {
            launchForResult(mLaunchRequest.mResultListener, intent, mLaunchRequest.mRequestCode);
        } else {
            launch(intent);
        }
    }
    // 生成intent
    public Intent toIntent() {
        final Intent intent = new Intent(Intent.ACTION_MAIN);
        copyExtras(intent);
        intent.setClass(mContext, SubSettings.class); // 注意这里,启动了SubSettings
        if (TextUtils.isEmpty(mLaunchRequest.mDestinationName)) {
            throw new IllegalArgumentException("Destination fragment must be set");
        }
        intent.putExtra(SettingsActivity.EXTRA_SHOW_FRAGMENT, mLaunchRequest.mDestinationName);
        return intent;
    }
    // 启动Activity
    void launchAsUser(Intent intent, UserHandle userHandle) {
        mContext.startActivityAsUser(intent, userHandle);
    }
}

SubSettings extends SettingsActivity,在onCreate()中拿到传过来的fragment className,通过反射的创建了fragment,然后进行了显示

// Settings\src\com\android\settings\SettingsActivity.java
protected void onCreate(Bundle savedState) {
    super.onCreate(savedState);
    Log.d(LOG_TAG, "Starting onCreate");
    createUiFromIntent(savedState, intent);
}
protected void createUiFromIntent(Bundle savedState, Intent intent) {
    ...
    // 从intent中拿到fragment className
    final String initialFragmentName = getInitialFragmentName(intent);
    // 设置布局文件
    setContentView(R.layout.settings_main_prefs);
    getSupportFragmentManager().addOnBackStackChangedListener(this);
    if (savedState != null) {
        
    } else {
        // 默认进来走这里
        launchSettingFragment(initialFragmentName, intent);
    }
   ...
}

void launchSettingFragment(String initialFragmentName, Intent intent) {
    if (initialFragmentName != null) {
        setTitleFromIntent(intent);
        Bundle initialArguments = intent.getBundleExtra(EXTRA_SHOW_FRAGMENT_ARGUMENTS);
        switchToFragment(initialFragmentName, initialArguments, true,
                mInitialTitleResId, mInitialTitle);
    } else {
        mInitialTitleResId = R.string.dashboard_title;
        switchToFragment(TopLevelSettings.class.getName(), null /* args */, false,
                mInitialTitleResId, mInitialTitle);
    }
}

// 创建显示fragment
private void switchToFragment(String fragmentName, Bundle args, boolean validate,
        int titleResId, CharSequence title) {
    Log.d(LOG_TAG, "Switching to fragment " + fragmentName);
    
    // 通过反射的创建fragment
    Fragment f = Utils.getTargetFragment(this, fragmentName, args);
    if (f == null) {
        return;
    }
    FragmentTransaction transaction = getSupportFragmentManager().beginTransaction();
    transaction.replace(R.id.main_content, f);
    if (titleResId > 0) {
        transaction.setBreadCrumbTitle(titleResId);
    } else if (title != null) {
        transaction.setBreadCrumbTitle(title);
    }
    transaction.commitAllowingStateLoss();
    getSupportFragmentManager().executePendingTransactions();
}

至此,Settings主界面创建显示逻辑讲完了

总结

  • Settings主界面,左侧是一个Preference列表,右侧是一个activity,activity展示Preference对应的fragment
  • 二级界面显示类似,可参考上面分析
  • 其它一些细节待后面学习,像如何进行左右两边展示的、嵌入三方apk等

参考

Prefrence介绍和用法

  • https://blog.csdn.net/crazymo_/article/details/51705942
  • https://blog.csdn.net/qq_37858386/article/details/103806336

Settings分析

  • https://cloud.tencent.com/developer/article/2349589
Android Settings模块是Android系统中的一个重要组成部分,它提供了用户对设备进行配置和管理的界面。它是一个完整的应用程序,由多个Activity和Fragment组成,主要包括以下几个部分: 1. 系统设置:包括网络、声音、显示、电池、存储、安全等设置,用户可以根据自己的需要对系统进行设置。 2. 应用设置:包括已安装应用的管理和配置,用户可以查看应用信息、权限、通知、存储、数据使用情况等。 3. 用户设置:包括用户账户和个人资料的管理和配置,用户可以添加、删除、切换用户账户,还可以修改个人资料、语言、时区等。 4. 开发者选项:为开发者提供了一些高级设置和调试工具,例如USB调试、CPU使用情况、布局边界等。 5. 关于手机:提供了设备的基本信息,包括设备型号、Android版本、内核版本、基带版本等。 Android Settings模块的实现主要依赖于Android框架中的SettingsProvider和Settings应用程序。SettingsProvider是一个ContentProvider,为Settings应用程序提供数据源,包括系统设置、应用设置、用户设置等。Settings应用程序则负责展示这些数据并提供用户交互界面。 总的来说,Android Settings模块为用户提供了方便、实用的设备配置和管理功能,同时为开发者提供了一些高级设置和调试工具,是Android系统中非常重要的一个组成部分。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值