-
Notifications
You must be signed in to change notification settings - Fork 1.4k
Description
As part of my previous work on #17215 and #17960 I keep stumbling on the fact that on WPAndroid, instead of using the AndroidX Preference library for anything settings related, it is still using the legacy and long deprecated Android Preference library.
API 29), the legacy android.preference library, referred to as the platform android.preference library, is deprecated. For consistent behavior across all devices it is recommended to use the androidx.preference library.
Also, in addition to the above, the WPAndroid preference configuration and usage just that bit more complicated. The complication comes from the fact that WPAndroid used both android.preference and androidx.preference (see #17215).
androidx.preferenceis used only forPreferenceManagerrelated work, whileandroid.preferenceis used for everything else.
The above makes that bit more complicated to update the AndroidX Preference library to newer versions, and that is, because as one would need to know all the above before deciding towards the updating strategy.
For example, while working on #17960 I was tempted to also try and migrated at least one screen from android.preference and into androidx.preference. I tried with the seemingly simple AccountSettingsFragment.kt class, which is related to the very simple Account Settings screen, but without success, not at all. After a while, I realized that everything is interconnected in such a degree that it is impossible to complete this migration without affecting any other screen (more on that below).
Technical Details
While trying to migrate AccountSettingsFragment.kt class to androidx.preference, during my work on #17960 I ended up mapping down every usage of android.preference and anything that is depending on it, via extending those classes, or using those via composition.
Below is a summary of that, you can expand and see the details per class, as well as how to test each class:
1. AccountSettingsFragment.kt
Res: account_settings.xml
Extends from: PreferenceFragmentLifeCycleOwner.kt
Uses:
EditTextPreferenceWithValidation.java, which extends fromSummaryEditTextPreference.javaDetailListPreference.java
To test:
- Go to
Mescreen. - Click on the
Account Settingsbutton. - Verify that the
Account Settingsscreen is displayed. - Click on each of the settings within the
Account Settingsscreen and verify that every setting works as expected.
2. AppSettingsFragment.java
Res: app_settings.xml
Uses:
WPActivityUtils.java, which usesadd/removeToolbarFromDialog(...)WPSwitchPreference.java, which usesSummaryEditTextPreference.javaDetailListPreference.javaLearnMorePreference.javaWPPreference.java, which usesDetailListPreference.java`WPPrefUtils.java
To test:
- Go to
Mescreen. - Click on the
App Settingsbutton. - Verify that the
App Settingsscreen is displayed. - Click on each of the settings within the
App Settingsscreen and verify that every setting works as expected, including the inner settings like thePrivacy SettingsandDebug Settingsscreens.
3. SiteSettingsFragment.java
Res: site_settings.xml
Uses:
WPActivityUtils.java, which usesadd/removeToolbarFromDialog(...)WPSwitchPreference.java, which usesSummaryEditTextPreference.javaEditTextPreferenceWithValidation.java, which extends fromSummaryEditTextPreference.javaSummaryEditTextPreference.javaDetailListPreference.javaLearnMorePreference.javaWPPreference.java, which usesDetailListPreference.java`WPStartOverPreference.java, which extends from WPPreference.javaWPPrefUtils.java
To test:
- While on the
My Site/MENUtab. - Click on the
Site Settingsbutton. - Verify that the
Site Settingsscreen is displayed. - Click on each of the settings within the
Site Settingsscreen and verify that every setting works as expected.
4. JetpackSecuritySettingsFragment.java
Res: jetpack_settings.xml
Uses:
WPSwitchPreference.java, which usesSummaryEditTextPreference.javaLearnMorePreference.javaWPPrefUtils.java
To test:
- Prerequisite: You must have a Jetpack connected site, which displays this setting.
- While on the
My Site/MENUtab. - Click on the
Jetpack Settingsbutton. - Verify that the
Securitysetting screen is displayed. - Click on each of the settings within the
Securitysettings screen and verify that every setting works as expected.
5. NotificationSettingsFragment.java
Res: notifications_settings.xml
Uses:
NotificationsSettingsDialogPreference.javaWPActivityUtils.java, which usesadd/removeToolbarFromDialog(...)WPSwitchPreference.java, which usesSummaryEditTextPreference.java
To test:
- While on the
Notificationstab. - Click on the
Gearsetting button (top-right). - Verify that the
Notification Settingsscreen is displayed. - Click on each of the settings within the
Notification Settingssettings screen and verify that every setting works as expected.
Also, consider the below usages of android.preference in non-setting screen related part of WPAndroid's codebase:
1. EditPostActivity.java
Uses:
EditTextPreferenceWithValidation.java, which extends fromSummaryEditTextPreference.java
To test:
- Go to
Postscreen. - Edit a new post, which uses
PreferenceManagertosetDefaultValues(...)forAccount Settings, add a few of the main blocks and verify that everything is workings as expected.
2. MySitesPage.java (UI Test)
To test:
- Run the
StatsTestsUI test suite, which uses theMySitesPage.javaclass, and verify that all tests pass.
From the above you will notice the below:
- There are 5 settings screens overall:
Account Settings, which corresponds toAccountSettingsFragment.ktandaccount_settings.xmlApp Settings, which corresponds toAppSettingsFragment.javaandapp_settings.xmlSite Settings, which corresponds toSiteSettingsFragment.javaandsite_settings.xmlJetpack Security Settings, which corresponds toJetpackSecuritySettingsFragment.javaandjetpack_settings.xmlNotifications Settings, which corresponds toNotificationsSettingsFragment.javaandnotifications_settings.xml
- There are 8 custom preference classes:
WPPreference.java, which usesDetailListPreference.java`WPStartOverPreference.java, which extends from WPPreference.javaand only used bySiteSettingsFragment.javaWPSwitchPreference, which usesSummaryEditTextPreference.javaSummaryEditTextPreferenceEditTextPreferenceWithValidation, which extends fromSummaryEditTextPreferenceDetailListPreferenceLearnMorePreferenceNotificationsSettingsDialogPreference, which is only used byNotificationSettingsFragment.java
- There are 3 extra classes related to preferences, utils or otherwise:
WPPrefUtils, which usesadd/removeToolbarFromDialog(...)WPActivityUtilsPreferenceFragmentLifeCycleOwner, which is extended only byAccountSettingsFragment.kt
Recommendation & Suggested Strategy
From the technical details above you will notice how interconnected everything is.
As such, my recommendation is instead of migrating all 5 settings screens, and thus, forcing us to do that change in one go, most probably even in one PR (as I am not seeing another alternative), we could create a project which will rewrite each of those screens instead. This will make this change more manageable, enable the creation of multiple PRs, each tested in isolation, and which, will target trunk directly, that is, without the need for a parent intermediate feature branch.
Per the above, my suggested strategy would be:
- To create 5 individual PRs, per the 5 settings screens above.
- For each PR, create a completely new screen with
androidx.preference, along with creating any custom preference class that this screen might required to become fully operational.
In more detail, I would have followed the below PR chain strategy:
- First, start with a new
Account Settingsscreen, along with creating 3 new custom preference classes (SummaryEditTextPreference.kt,EditTextPreferenceWithValidation.ktandDetailListPreference.kt). - Then, continue with a new
Jetpack Security Settingsscreen, along with creating 2 new custom preference classes (WPSwitchPreference.ktandLearnMorePreference.kt). - Then, continue with a new
Notifications Settingsscreen, along with creating 1 new custom preference class (NotificationsSettingsDialogPreference.kt). - Then, continue with a new
App Settingsscreen, along with creating 1 new custom preference class (WPPreference.kt). - Then, end with a new
Site Settingsscreen, along with creating 1 new custom preference class (WPStartOverPreference). - Finally, if not already done by all the above, remove any remaining usage of
android.preferencefrom anywhere else (ie.PreferenceFragmentLifeCycleOwner,WPPrefUtils,WPActivityUtils,EditPostActivity,MySitesPage, etc)
Let's discuss, let me know your thoughts on all the above! 🙇