Loading src/com/android/settings/slices/SliceData.java +11 −4 Original line number Diff line number Diff line Loading @@ -227,19 +227,19 @@ public class SliceData { public SliceData build() { if (TextUtils.isEmpty(mKey)) { throw new IllegalStateException("Key cannot be empty"); throw new InvalidSliceDataException("Key cannot be empty"); } if (TextUtils.isEmpty(mTitle)) { throw new IllegalStateException("Title cannot be empty"); throw new InvalidSliceDataException("Title cannot be empty"); } if (TextUtils.isEmpty(mFragmentClassName)) { throw new IllegalStateException("Fragment Name cannot be empty"); throw new InvalidSliceDataException("Fragment Name cannot be empty"); } if (TextUtils.isEmpty(mPrefControllerClassName)) { throw new IllegalStateException("Preference Controller cannot be empty"); throw new InvalidSliceDataException("Preference Controller cannot be empty"); } return new SliceData(this); Loading @@ -249,4 +249,11 @@ public class SliceData { return mKey; } } public static class InvalidSliceDataException extends RuntimeException { public InvalidSliceDataException(String message) { super(message); } } } No newline at end of file src/com/android/settings/slices/SliceDataConverter.java +10 −5 Original line number Diff line number Diff line Loading @@ -40,12 +40,12 @@ import android.util.Log; import android.util.Xml; import android.view.accessibility.AccessibilityManager; import com.android.settings.accessibility.AccessibilitySlicePreferenceController; import com.android.settings.core.PreferenceXmlParserUtils; import com.android.settings.core.PreferenceXmlParserUtils.MetadataFlag; import com.android.internal.annotations.VisibleForTesting; import com.android.settings.R; import com.android.settings.accessibility.AccessibilitySettings; import com.android.settings.accessibility.AccessibilitySlicePreferenceController; import com.android.settings.core.PreferenceXmlParserUtils; import com.android.settings.core.PreferenceXmlParserUtils.MetadataFlag; import com.android.settings.dashboard.DashboardFragment; import com.android.settings.overlay.FeatureFactory; import com.android.settings.search.DatabaseIndexingUtils; Loading Loading @@ -219,6 +219,8 @@ class SliceDataConverter { xmlSliceData.add(xmlSlice); } } catch (SliceData.InvalidSliceDataException e) { Log.w(TAG, "Invalid data when building SliceData for " + fragmentName, e); } catch (XmlPullParserException e) { Log.w(TAG, "XML Error parsing PreferenceScreen: ", e); } catch (IOException e) { Loading Loading @@ -271,8 +273,11 @@ class SliceDataConverter { .setTitle(title) .setIcon(iconResource) .setSliceType(SliceData.SliceType.SWITCH); try { sliceData.add(sliceDataBuilder.build()); } catch (SliceData.InvalidSliceDataException e) { Log.w(TAG, "Invalid data when building a11y SliceData for " + flattenedName, e); } } return sliceData; Loading tests/robotests/src/com/android/settings/slices/SliceDataTest.java +4 −4 Original line number Diff line number Diff line Loading @@ -70,7 +70,7 @@ public class SliceDataTest { assertThat(data.isPlatformDefined()).isEqualTo(IS_PLATFORM_DEFINED); } @Test(expected = IllegalStateException.class) @Test(expected = SliceData.InvalidSliceDataException.class) public void testBuilder_noKey_throwsIllegalStateException() { new SliceData.Builder() .setTitle(TITLE) Loading @@ -83,7 +83,7 @@ public class SliceDataTest { .build(); } @Test(expected = IllegalStateException.class) @Test(expected = SliceData.InvalidSliceDataException.class) public void testBuilder_noTitle_throwsIllegalStateException() { new SliceData.Builder() .setKey(KEY) Loading @@ -96,7 +96,7 @@ public class SliceDataTest { .build(); } @Test(expected = IllegalStateException.class) @Test(expected = SliceData.InvalidSliceDataException.class) public void testBuilder_noFragment_throwsIllegalStateException() { new SliceData.Builder() .setKey(KEY) Loading @@ -109,7 +109,7 @@ public class SliceDataTest { .build(); } @Test(expected = IllegalStateException.class) @Test(expected = SliceData.InvalidSliceDataException.class) public void testBuilder_noPrefController_throwsIllegalStateException() { new SliceData.Builder() .setKey(KEY) Loading tests/unit/src/com/android/settings/slices/SliceDataContractTest.java 0 → 100644 +122 −0 Original line number Diff line number Diff line /* * 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.slices; import static junit.framework.Assert.fail; import android.content.Context; import android.os.Bundle; import android.platform.test.annotations.Presubmit; import android.provider.SearchIndexableResource; import android.support.test.InstrumentationRegistry; import android.support.test.filters.MediumTest; import android.support.test.runner.AndroidJUnit4; import android.text.TextUtils; import android.util.Log; import com.android.settings.core.PreferenceXmlParserUtils; import com.android.settings.overlay.FeatureFactory; import com.android.settings.search.DatabaseIndexingUtils; import com.android.settings.search.Indexable; import com.android.settings.search.SearchIndexableResources; import org.junit.Before; import org.junit.Test; import org.junit.runner.RunWith; import org.xmlpull.v1.XmlPullParserException; import java.io.IOException; import java.util.HashSet; import java.util.List; import java.util.Set; @RunWith(AndroidJUnit4.class) @MediumTest public class SliceDataContractTest { private static final String TAG = "SliceDataContractTest"; private Context mContext; @Before public void setUp() { mContext = InstrumentationRegistry.getTargetContext(); } @Test @Presubmit public void preferenceWithControllerMustHaveNonEmptyTitle() throws IOException, XmlPullParserException { final Set<String> nullTitleFragments = new HashSet<>(); final SearchIndexableResources resources = FeatureFactory.getFactory(mContext).getSearchFeatureProvider() .getSearchIndexableResources(); for (Class<?> clazz : resources.getProviderValues()) { verifyPreferenceTitle(nullTitleFragments, clazz); } if (!nullTitleFragments.isEmpty()) { final StringBuilder error = new StringBuilder( "All preferences with a controller must have a non-empty title by default, " + "found empty title in the following fragments\n"); for (String c : nullTitleFragments) { error.append(c).append("\n"); } fail(error.toString()); } } private void verifyPreferenceTitle(Set<String> nullTitleFragments, Class<?> clazz) throws IOException, XmlPullParserException { if (clazz == null) { return; } final String className = clazz.getName(); final Indexable.SearchIndexProvider provider = DatabaseIndexingUtils.getSearchIndexProvider(clazz); final List<SearchIndexableResource> resourcesToIndex = provider.getXmlResourcesToIndex(mContext, true); if (resourcesToIndex == null) { Log.d(TAG, className + "is not providing SearchIndexableResource, skipping"); return; } for (SearchIndexableResource sir : resourcesToIndex) { final List<Bundle> metadata = PreferenceXmlParserUtils.extractMetadata(mContext, sir.xmlResId, PreferenceXmlParserUtils.MetadataFlag.FLAG_INCLUDE_PREF_SCREEN | PreferenceXmlParserUtils.MetadataFlag.FLAG_NEED_PREF_TITLE | PreferenceXmlParserUtils.MetadataFlag.FLAG_NEED_PREF_CONTROLLER); for (Bundle bundle : metadata) { final String controller = bundle.getString( PreferenceXmlParserUtils.METADATA_CONTROLLER); if (TextUtils.isEmpty(controller)) { continue; } final String title = bundle.getString(PreferenceXmlParserUtils.METADATA_TITLE); if (TextUtils.isEmpty(title)) { nullTitleFragments.add(className); } } } } } No newline at end of file Loading
src/com/android/settings/slices/SliceData.java +11 −4 Original line number Diff line number Diff line Loading @@ -227,19 +227,19 @@ public class SliceData { public SliceData build() { if (TextUtils.isEmpty(mKey)) { throw new IllegalStateException("Key cannot be empty"); throw new InvalidSliceDataException("Key cannot be empty"); } if (TextUtils.isEmpty(mTitle)) { throw new IllegalStateException("Title cannot be empty"); throw new InvalidSliceDataException("Title cannot be empty"); } if (TextUtils.isEmpty(mFragmentClassName)) { throw new IllegalStateException("Fragment Name cannot be empty"); throw new InvalidSliceDataException("Fragment Name cannot be empty"); } if (TextUtils.isEmpty(mPrefControllerClassName)) { throw new IllegalStateException("Preference Controller cannot be empty"); throw new InvalidSliceDataException("Preference Controller cannot be empty"); } return new SliceData(this); Loading @@ -249,4 +249,11 @@ public class SliceData { return mKey; } } public static class InvalidSliceDataException extends RuntimeException { public InvalidSliceDataException(String message) { super(message); } } } No newline at end of file
src/com/android/settings/slices/SliceDataConverter.java +10 −5 Original line number Diff line number Diff line Loading @@ -40,12 +40,12 @@ import android.util.Log; import android.util.Xml; import android.view.accessibility.AccessibilityManager; import com.android.settings.accessibility.AccessibilitySlicePreferenceController; import com.android.settings.core.PreferenceXmlParserUtils; import com.android.settings.core.PreferenceXmlParserUtils.MetadataFlag; import com.android.internal.annotations.VisibleForTesting; import com.android.settings.R; import com.android.settings.accessibility.AccessibilitySettings; import com.android.settings.accessibility.AccessibilitySlicePreferenceController; import com.android.settings.core.PreferenceXmlParserUtils; import com.android.settings.core.PreferenceXmlParserUtils.MetadataFlag; import com.android.settings.dashboard.DashboardFragment; import com.android.settings.overlay.FeatureFactory; import com.android.settings.search.DatabaseIndexingUtils; Loading Loading @@ -219,6 +219,8 @@ class SliceDataConverter { xmlSliceData.add(xmlSlice); } } catch (SliceData.InvalidSliceDataException e) { Log.w(TAG, "Invalid data when building SliceData for " + fragmentName, e); } catch (XmlPullParserException e) { Log.w(TAG, "XML Error parsing PreferenceScreen: ", e); } catch (IOException e) { Loading Loading @@ -271,8 +273,11 @@ class SliceDataConverter { .setTitle(title) .setIcon(iconResource) .setSliceType(SliceData.SliceType.SWITCH); try { sliceData.add(sliceDataBuilder.build()); } catch (SliceData.InvalidSliceDataException e) { Log.w(TAG, "Invalid data when building a11y SliceData for " + flattenedName, e); } } return sliceData; Loading
tests/robotests/src/com/android/settings/slices/SliceDataTest.java +4 −4 Original line number Diff line number Diff line Loading @@ -70,7 +70,7 @@ public class SliceDataTest { assertThat(data.isPlatformDefined()).isEqualTo(IS_PLATFORM_DEFINED); } @Test(expected = IllegalStateException.class) @Test(expected = SliceData.InvalidSliceDataException.class) public void testBuilder_noKey_throwsIllegalStateException() { new SliceData.Builder() .setTitle(TITLE) Loading @@ -83,7 +83,7 @@ public class SliceDataTest { .build(); } @Test(expected = IllegalStateException.class) @Test(expected = SliceData.InvalidSliceDataException.class) public void testBuilder_noTitle_throwsIllegalStateException() { new SliceData.Builder() .setKey(KEY) Loading @@ -96,7 +96,7 @@ public class SliceDataTest { .build(); } @Test(expected = IllegalStateException.class) @Test(expected = SliceData.InvalidSliceDataException.class) public void testBuilder_noFragment_throwsIllegalStateException() { new SliceData.Builder() .setKey(KEY) Loading @@ -109,7 +109,7 @@ public class SliceDataTest { .build(); } @Test(expected = IllegalStateException.class) @Test(expected = SliceData.InvalidSliceDataException.class) public void testBuilder_noPrefController_throwsIllegalStateException() { new SliceData.Builder() .setKey(KEY) Loading
tests/unit/src/com/android/settings/slices/SliceDataContractTest.java 0 → 100644 +122 −0 Original line number Diff line number Diff line /* * 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.slices; import static junit.framework.Assert.fail; import android.content.Context; import android.os.Bundle; import android.platform.test.annotations.Presubmit; import android.provider.SearchIndexableResource; import android.support.test.InstrumentationRegistry; import android.support.test.filters.MediumTest; import android.support.test.runner.AndroidJUnit4; import android.text.TextUtils; import android.util.Log; import com.android.settings.core.PreferenceXmlParserUtils; import com.android.settings.overlay.FeatureFactory; import com.android.settings.search.DatabaseIndexingUtils; import com.android.settings.search.Indexable; import com.android.settings.search.SearchIndexableResources; import org.junit.Before; import org.junit.Test; import org.junit.runner.RunWith; import org.xmlpull.v1.XmlPullParserException; import java.io.IOException; import java.util.HashSet; import java.util.List; import java.util.Set; @RunWith(AndroidJUnit4.class) @MediumTest public class SliceDataContractTest { private static final String TAG = "SliceDataContractTest"; private Context mContext; @Before public void setUp() { mContext = InstrumentationRegistry.getTargetContext(); } @Test @Presubmit public void preferenceWithControllerMustHaveNonEmptyTitle() throws IOException, XmlPullParserException { final Set<String> nullTitleFragments = new HashSet<>(); final SearchIndexableResources resources = FeatureFactory.getFactory(mContext).getSearchFeatureProvider() .getSearchIndexableResources(); for (Class<?> clazz : resources.getProviderValues()) { verifyPreferenceTitle(nullTitleFragments, clazz); } if (!nullTitleFragments.isEmpty()) { final StringBuilder error = new StringBuilder( "All preferences with a controller must have a non-empty title by default, " + "found empty title in the following fragments\n"); for (String c : nullTitleFragments) { error.append(c).append("\n"); } fail(error.toString()); } } private void verifyPreferenceTitle(Set<String> nullTitleFragments, Class<?> clazz) throws IOException, XmlPullParserException { if (clazz == null) { return; } final String className = clazz.getName(); final Indexable.SearchIndexProvider provider = DatabaseIndexingUtils.getSearchIndexProvider(clazz); final List<SearchIndexableResource> resourcesToIndex = provider.getXmlResourcesToIndex(mContext, true); if (resourcesToIndex == null) { Log.d(TAG, className + "is not providing SearchIndexableResource, skipping"); return; } for (SearchIndexableResource sir : resourcesToIndex) { final List<Bundle> metadata = PreferenceXmlParserUtils.extractMetadata(mContext, sir.xmlResId, PreferenceXmlParserUtils.MetadataFlag.FLAG_INCLUDE_PREF_SCREEN | PreferenceXmlParserUtils.MetadataFlag.FLAG_NEED_PREF_TITLE | PreferenceXmlParserUtils.MetadataFlag.FLAG_NEED_PREF_CONTROLLER); for (Bundle bundle : metadata) { final String controller = bundle.getString( PreferenceXmlParserUtils.METADATA_CONTROLLER); if (TextUtils.isEmpty(controller)) { continue; } final String title = bundle.getString(PreferenceXmlParserUtils.METADATA_TITLE); if (TextUtils.isEmpty(title)) { nullTitleFragments.add(className); } } } } } No newline at end of file