summaryrefslogtreecommitdiff
path: root/library
diff options
context:
space:
mode:
authorMaurice Lam <yukl@google.com>2015-11-18 16:04:56 -0800
committerMaurice Lam <yukl@google.com>2015-11-18 20:25:04 -0800
commitfce4cf6161b2a1644ad21034f0afe4087d659ab4 (patch)
tree5c0e77b475e4323537fc85c25b69d530e7426470 /library
parent35cc29c9f307191962cd7124db14eb7b78c9407e (diff)
downloadsetupwizard-fce4cf6161b2a1644ad21034f0afe4087d659ab4.tar.gz
[SetupWizard] Implement GlifLayout
Bug: 25726515 Change-Id: Ib4ba64e648e52a9859fb6509a9ce41c692a20aa2
Diffstat (limited to 'library')
-rw-r--r--library/eclair-mr1/res/values/styles.xml5
-rw-r--r--library/main/res/layout/suw_glif_template.xml50
-rw-r--r--library/main/res/values/attrs.xml12
-rw-r--r--library/main/res/values/dimens.xml7
-rw-r--r--library/main/res/values/styles.xml27
-rw-r--r--library/main/src/com/android/setupwizardlib/GlifLayout.java175
-rw-r--r--library/platform/res/values-v21/styles.xml5
-rw-r--r--library/test/res/layout/test_glif_layout.xml27
-rw-r--r--library/test/src/com/android/setupwizardlib/test/GlifLayoutTest.java92
9 files changed, 390 insertions, 10 deletions
diff --git a/library/eclair-mr1/res/values/styles.xml b/library/eclair-mr1/res/values/styles.xml
index 607e948..9b08899 100644
--- a/library/eclair-mr1/res/values/styles.xml
+++ b/library/eclair-mr1/res/values/styles.xml
@@ -64,17 +64,18 @@
<!-- Specify the indeterminateTintMode to work around a bug in Lollipop -->
<item name="android:indeterminateTintMode" tools:ignore="NewApi">src_in</item>
<item name="android:navigationBarColor" tools:ignore="NewApi">@android:color/black</item>
- <item name="android:statusBarColor" tools:ignore="NewApi">@android:color/black</item>
+ <item name="android:statusBarColor" tools:ignore="NewApi">?attr/colorPrimary</item>
<item name="android:textColorLink">@color/suw_link_color_light</item>
<item name="android:windowAnimationStyle">@style/Animation.SuwWindowAnimation</item>
<item name="android:windowDisablePreview">true</item>
<item name="android:windowSoftInputMode">adjustResize</item>
<item name="colorAccent">@color/suw_color_accent_light</item>
+ <item name="colorPrimary">@color/suw_color_accent_light</item>
<item name="listPreferredItemPaddingLeft">?attr/suwMarginSides</item>
<item name="listPreferredItemPaddingRight">?attr/suwMarginSides</item>
<item name="suwCardBackground">@drawable/suw_card_bg_light</item>
- <item name="suwMarginSides">@dimen/suw_layout_margin_sides</item>
+ <item name="suwMarginSides">@dimen/suw_glif_margin_sides</item>
<item name="suwNavBarTheme">@style/SuwNavBarThemeLight</item>
<item name="textAppearanceListItemSmall">@style/TextAppearance.AppCompat.Body1</item>
</style>
diff --git a/library/main/res/layout/suw_glif_template.xml b/library/main/res/layout/suw_glif_template.xml
new file mode 100644
index 0000000..d9be0b8
--- /dev/null
+++ b/library/main/res/layout/suw_glif_template.xml
@@ -0,0 +1,50 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<!--
+ Copyright (C) 2015 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.
+-->
+
+<ScrollView
+ xmlns:android="http://schemas.android.com/apk/res/android"
+ android:id="@+id/suw_scroll_view"
+ android:layout_width="match_parent"
+ android:layout_height="match_parent"
+ android:fillViewport="true">
+
+ <LinearLayout
+ android:layout_width="match_parent"
+ android:layout_height="wrap_content"
+ android:orientation="vertical">
+
+ <ImageView
+ android:id="@+id/suw_layout_icon"
+ style="@style/SuwGlifIcon"
+ android:layout_width="wrap_content"
+ android:layout_height="wrap_content"
+ android:contentDescription="@null" />
+
+ <TextView
+ android:id="@+id/suw_layout_title"
+ style="@style/SuwGlifHeaderTitle"
+ android:layout_width="match_parent"
+ android:layout_height="wrap_content" />
+
+ <FrameLayout
+ android:id="@+id/suw_layout_content"
+ android:layout_width="match_parent"
+ android:layout_height="wrap_content" />
+
+ </LinearLayout>
+
+</ScrollView>
diff --git a/library/main/res/values/attrs.xml b/library/main/res/values/attrs.xml
index 49ad430..209f81c 100644
--- a/library/main/res/values/attrs.xml
+++ b/library/main/res/values/attrs.xml
@@ -17,6 +17,7 @@
<resources>
+ <!-- Theme attributes -->
<attr name="suwLayoutTheme" format="reference" />
<attr name="suwMarginSides" format="dimension|reference" />
@@ -26,6 +27,9 @@
<attr name="suwNavBarTextColor" format="color" />
<attr name="suwNavBarTheme" format="reference" />
+ <!-- Custom view attributes -->
+ <attr name="suwHeaderText" format="string" localization="suggested" />
+
<declare-styleable name="SuwIllustration">
<attr name="suwAspectRatio" format="float" />
</declare-styleable>
@@ -34,10 +38,16 @@
<attr name="suwHeader" format="reference" />
</declare-styleable>
+ <declare-styleable name="SuwGlifLayout">
+ <attr name="android:icon" />
+ <attr name="suwHeaderColor" format="reference|color" />
+ <attr name="suwHeaderText" />
+ </declare-styleable>
+
<declare-styleable name="SuwSetupWizardLayout">
<attr name="suwBackground" format="color|reference" />
<attr name="suwBackgroundTile" format="color|reference" />
- <attr name="suwHeaderText" format="string" localization="suggested" />
+ <attr name="suwHeaderText" />
<attr name="suwDecorPaddingTop" format="dimension|reference" />
<attr name="suwIllustration" format="color|reference" />
<attr name="suwIllustrationAspectRatio" format="float|reference" />
diff --git a/library/main/res/values/dimens.xml b/library/main/res/values/dimens.xml
index 211f01d..e955d85 100644
--- a/library/main/res/values/dimens.xml
+++ b/library/main/res/values/dimens.xml
@@ -1,4 +1,4 @@
-<?xml version="1.0" encoding="utf-8"?>
+<?xml version="1.0" encoding="UTF-8"?>
<!--
Copyright (C) 2015 The Android Open Source Project
@@ -19,6 +19,8 @@
<!-- General -->
<dimen name="suw_layout_margin_sides">40dp</dimen>
+ <dimen name="suw_glif_margin_sides">24dp</dimen>
+ <dimen name="suw_glif_margin_top">24dp</dimen>
<!-- Content styles -->
<dimen name="suw_check_box_line_spacing_extra">4sp</dimen>
@@ -69,6 +71,9 @@
<!-- This is the extra spacing required to make the leading exactly 32sp -->
<dimen name="suw_header_title_line_spacing_extra">3.67sp</dimen>
+ <dimen name="suw_glif_header_title_margin_top">24dp</dimen>
+ <dimen name="suw_glif_header_title_margin_bottom">8dp</dimen>
+
<!-- Illustration -->
<item name="suw_illustration_aspect_ratio" format="float" type="dimen">2.22</item>
diff --git a/library/main/res/values/styles.xml b/library/main/res/values/styles.xml
index a720471..787b5b9 100644
--- a/library/main/res/values/styles.xml
+++ b/library/main/res/values/styles.xml
@@ -96,19 +96,38 @@
<!-- Header layout (for phones) -->
- <style name="SuwHeaderTitle">
+ <style name="SuwBaseHeaderTitle">
<!-- Before Honeycomb, layout_gravity is needed for FrameLayout to apply the margins -->
<item name="android:layout_gravity">top</item>
+ <item name="android:ellipsize">end</item>
+ <item name="android:maxLines">2</item>
+ <item name="android:textSize">@dimen/suw_header_title_size</item>
+ </style>
+
+ <style name="SuwHeaderTitle" parent="SuwBaseHeaderTitle">
<item name="android:layout_marginBottom">@dimen/suw_header_title_margin_bottom</item>
<item name="android:layout_marginLeft">?attr/suwMarginSides</item>
<item name="android:layout_marginRight">?attr/suwMarginSides</item>
- <item name="android:ellipsize">end</item>
<item name="android:lineSpacingExtra">@dimen/suw_header_title_line_spacing_extra</item>
- <item name="android:maxLines">2</item>
<item name="android:paddingBottom">@dimen/suw_header_title_padding_bottom</item>
<item name="android:paddingTop">@dimen/suw_header_title_padding_top</item>
<item name="android:textColor">@android:color/white</item>
- <item name="android:textSize">@dimen/suw_header_title_size</item>
+ </style>
+
+ <!-- GLIF layout -->
+
+ <style name="SuwGlifHeaderTitle" parent="SuwBaseHeaderTitle">
+ <item name="android:layout_marginBottom">@dimen/suw_glif_header_title_margin_bottom</item>
+ <item name="android:layout_marginLeft">?attr/suwMarginSides</item>
+ <item name="android:layout_marginRight">?attr/suwMarginSides</item>
+ <item name="android:layout_marginTop">@dimen/suw_glif_header_title_margin_top</item>
+ <item name="android:textColor">?android:attr/textColorPrimary</item>
+ </style>
+
+ <style name="SuwGlifIcon">
+ <item name="android:layout_marginLeft">?attr/suwMarginSides</item>
+ <item name="android:layout_marginRight">?attr/suwMarginSides</item>
+ <item name="android:layout_marginTop">@dimen/suw_glif_margin_top</item>
</style>
<!-- Navigation bar styles -->
diff --git a/library/main/src/com/android/setupwizardlib/GlifLayout.java b/library/main/src/com/android/setupwizardlib/GlifLayout.java
new file mode 100644
index 0000000..09fafec
--- /dev/null
+++ b/library/main/src/com/android/setupwizardlib/GlifLayout.java
@@ -0,0 +1,175 @@
+/*
+ * Copyright (C) 2015 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.setupwizardlib;
+
+import android.annotation.TargetApi;
+import android.content.Context;
+import android.content.res.ColorStateList;
+import android.content.res.TypedArray;
+import android.graphics.drawable.Drawable;
+import android.os.Build.VERSION_CODES;
+import android.util.AttributeSet;
+import android.view.LayoutInflater;
+import android.view.View;
+import android.view.ViewGroup;
+import android.widget.ImageView;
+import android.widget.ScrollView;
+import android.widget.TextView;
+
+/**
+ * Layout for the GLIF theme used in Setup Wizard for N.
+ *
+ * <p>Example usage:
+ * <pre>{@code
+ * &lt;com.android.setupwizardlib.GlifLayout
+ * xmlns:android="http://schemas.android.com/apk/res/android"
+ * xmlns:app="http://schemas.android.com/apk/res-auto"
+ * android:layout_width="match_parent"
+ * android:layout_height="match_parent"
+ * android:icon="@drawable/my_icon"
+ * app:suwHeaderText="@string/my_title">
+ *
+ * &lt;!-- Content here -->
+ *
+ * &lt;/com.android.setupwizardlib.GlifLayout>
+ * }</pre>
+ */
+public class GlifLayout extends TemplateLayout {
+
+ private static final String TAG = "GlifLayout";
+
+ public GlifLayout(Context context) {
+ this(context, 0, 0);
+ }
+
+ public GlifLayout(Context context, int template) {
+ this(context, template, 0);
+ }
+
+ public GlifLayout(Context context, int template, int containerId) {
+ super(context, template, containerId);
+ init(null, R.attr.suwLayoutTheme);
+ }
+
+ public GlifLayout(Context context, AttributeSet attrs) {
+ super(context, attrs);
+ init(attrs, R.attr.suwLayoutTheme);
+ }
+
+ @TargetApi(VERSION_CODES.HONEYCOMB)
+ public GlifLayout(Context context, AttributeSet attrs, int defStyleAttr) {
+ super(context, attrs, defStyleAttr);
+ init(attrs, defStyleAttr);
+ }
+
+ // All the constructors delegate to this init method. The 3-argument constructor is not
+ // available in LinearLayout before v11, so call super with the exact same arguments.
+ private void init(AttributeSet attrs, int defStyleAttr) {
+ final TypedArray a = getContext().obtainStyledAttributes(attrs,
+ R.styleable.SuwGlifLayout, defStyleAttr, 0);
+
+ final Drawable icon = a.getDrawable(R.styleable.SuwGlifLayout_android_icon);
+ if (icon != null) {
+ setIcon(icon);
+ }
+
+ // Set the header color
+ final ColorStateList headerColor =
+ a.getColorStateList(R.styleable.SuwGlifLayout_suwHeaderColor);
+ if (headerColor != null) {
+ setHeaderColor(headerColor);
+ }
+
+
+ // Set the header text
+ final CharSequence headerText =
+ a.getText(R.styleable.SuwGlifLayout_suwHeaderText);
+ if (headerText != null) {
+ setHeaderText(headerText);
+ }
+
+ a.recycle();
+ }
+
+ @Override
+ protected View onInflateTemplate(LayoutInflater inflater, int template) {
+ if (template == 0) {
+ template = R.layout.suw_glif_template;
+ }
+ return super.onInflateTemplate(inflater, template);
+ }
+
+ @Override
+ protected ViewGroup findContainer(int containerId) {
+ if (containerId == 0) {
+ containerId = R.id.suw_layout_content;
+ }
+ return super.findContainer(containerId);
+ }
+
+ public ScrollView getScrollView() {
+ final View view = findViewById(R.id.suw_scroll_view);
+ return view instanceof ScrollView ? (ScrollView) view : null;
+ }
+
+ public void setHeaderText(int title) {
+ final TextView titleView = (TextView) findViewById(R.id.suw_layout_title);
+ if (titleView != null) {
+ titleView.setText(title);
+ }
+ }
+
+ public void setHeaderText(CharSequence title) {
+ final TextView titleView = (TextView) findViewById(R.id.suw_layout_title);
+ if (titleView != null) {
+ titleView.setText(title);
+ }
+ }
+
+ public CharSequence getHeaderText() {
+ final TextView titleView = (TextView) findViewById(R.id.suw_layout_title);
+ return titleView != null ? titleView.getText() : null;
+ }
+
+ public void setHeaderColor(ColorStateList color) {
+ final TextView titleView = (TextView) findViewById(R.id.suw_layout_title);
+ if (titleView != null) {
+ titleView.setTextColor(color);
+ }
+ }
+
+ public ColorStateList getHeaderColor() {
+ final TextView titleView = (TextView) findViewById(R.id.suw_layout_title);
+ return titleView != null ? titleView.getTextColors() : null;
+ }
+
+ public void setIcon(Drawable icon) {
+ final ImageView iconView = getIconView();
+ if (iconView != null) {
+ iconView.setImageDrawable(icon);
+ }
+ }
+
+ public Drawable getIcon() {
+ final ImageView iconView = getIconView();
+ return iconView != null ? iconView.getDrawable() : null;
+ }
+
+ private ImageView getIconView() {
+ return (ImageView) findViewById(R.id.suw_layout_icon);
+ }
+}
diff --git a/library/platform/res/values-v21/styles.xml b/library/platform/res/values-v21/styles.xml
index d531fab..d6e6147 100644
--- a/library/platform/res/values-v21/styles.xml
+++ b/library/platform/res/values-v21/styles.xml
@@ -66,13 +66,14 @@
<style name="SuwThemeGlif.Light" parent="android:Theme.Material.Light.NoActionBar">
<item name="android:colorAccent">@color/suw_color_accent_light</item>
+ <item name="android:colorPrimary">@color/suw_color_accent_light</item>
<item name="android:indeterminateTint">@color/suw_progress_bar_color_light</item>
<!-- Specify the indeterminateTintMode to work around a bug in Lollipop -->
<item name="android:indeterminateTintMode">src_in</item>
<item name="android:listPreferredItemPaddingEnd">?attr/suwMarginSides</item>
<item name="android:listPreferredItemPaddingStart">?attr/suwMarginSides</item>
<item name="android:navigationBarColor">@android:color/black</item>
- <item name="android:statusBarColor">@android:color/black</item>
+ <item name="android:statusBarColor">?android:attr/colorPrimary</item>
<item name="android:textAppearanceListItemSmall">@android:style/TextAppearance.Material.Body1</item>
<item name="android:textColorLink">@color/suw_link_color_light</item>
<item name="android:windowAnimationStyle">@style/Animation.SuwWindowAnimation</item>
@@ -80,7 +81,7 @@
<item name="android:windowSoftInputMode">adjustResize</item>
<item name="suwCardBackground">@drawable/suw_card_bg</item>
- <item name="suwMarginSides">@dimen/suw_layout_margin_sides</item>
+ <item name="suwMarginSides">@dimen/suw_glif_margin_sides</item>
<item name="suwNavBarTheme">@style/SuwNavBarThemeLight</item>
</style>
diff --git a/library/test/res/layout/test_glif_layout.xml b/library/test/res/layout/test_glif_layout.xml
new file mode 100644
index 0000000..3c61c65
--- /dev/null
+++ b/library/test/res/layout/test_glif_layout.xml
@@ -0,0 +1,27 @@
+<!--
+ Copyright (C) 2015 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.
+-->
+
+<com.android.setupwizardlib.GlifLayout
+ xmlns:android="http://schemas.android.com/apk/res/android"
+ android:layout_width="match_parent"
+ android:layout_height="match_parent">
+
+ <TextView
+ android:id="@+id/test_content"
+ android:layout_width="match_parent"
+ android:layout_height="wrap_content" />
+
+</com.android.setupwizardlib.GlifLayout> \ No newline at end of file
diff --git a/library/test/src/com/android/setupwizardlib/test/GlifLayoutTest.java b/library/test/src/com/android/setupwizardlib/test/GlifLayoutTest.java
new file mode 100644
index 0000000..37f254c
--- /dev/null
+++ b/library/test/src/com/android/setupwizardlib/test/GlifLayoutTest.java
@@ -0,0 +1,92 @@
+/*
+ * Copyright (C) 2015 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.setupwizardlib.test;
+
+import android.content.Context;
+import android.test.InstrumentationTestCase;
+import android.test.suitebuilder.annotation.SmallTest;
+import android.view.ContextThemeWrapper;
+import android.view.LayoutInflater;
+import android.view.View;
+import android.widget.ScrollView;
+import android.widget.TextView;
+
+import com.android.setupwizardlib.GlifLayout;
+
+public class GlifLayoutTest extends InstrumentationTestCase {
+
+ private Context mContext;
+
+ @Override
+ protected void setUp() throws Exception {
+ super.setUp();
+ mContext = new ContextThemeWrapper(getInstrumentation().getContext(),
+ R.style.SuwThemeGlif_Light);
+ }
+
+ @SmallTest
+ public void testDefaultTemplate() {
+ GlifLayout layout = new GlifLayout(mContext);
+ assertDefaultTemplateInflated(layout);
+ }
+
+ @SmallTest
+ public void testSetHeaderText() {
+ GlifLayout layout = new GlifLayout(mContext);
+ TextView title = (TextView) layout.findViewById(R.id.suw_layout_title);
+ layout.setHeaderText("Abracadabra");
+ assertEquals("Header text should be \"Abracadabra\"", "Abracadabra", title.getText());
+ }
+
+ @SmallTest
+ public void testAddView() {
+ GlifLayout layout = new GlifLayout(mContext);
+ TextView tv = new TextView(mContext);
+ tv.setId(R.id.test_view_id);
+ layout.addView(tv);
+ assertDefaultTemplateInflated(layout);
+ View view = layout.findViewById(R.id.test_view_id);
+ assertSame("The view added should be the same text view", tv, view);
+ }
+
+ @SmallTest
+ public void testInflateFromXml() {
+ LayoutInflater inflater = LayoutInflater.from(mContext);
+ GlifLayout layout = (GlifLayout) inflater.inflate(R.layout.test_glif_layout, null);
+ assertDefaultTemplateInflated(layout);
+ View content = layout.findViewById(R.id.test_content);
+ assertTrue("@id/test_content should be a TextView", content instanceof TextView);
+ }
+
+ @SmallTest
+ public void testGetScrollView() {
+ GlifLayout layout = new GlifLayout(mContext);
+ assertNotNull("Get scroll view should not be null with default template",
+ layout.getScrollView());
+ }
+
+ private void assertDefaultTemplateInflated(GlifLayout layout) {
+ View title = layout.findViewById(R.id.suw_layout_title);
+ assertNotNull("@id/suw_layout_title should not be null", title);
+
+ View icon = layout.findViewById(R.id.suw_layout_icon);
+ assertNotNull("@id/suw_layout_icon should not be null", icon);
+
+ View scrollView = layout.findViewById(R.id.suw_scroll_view);
+ assertTrue("@id/suw_scroll_view should be a ScrollView", scrollView instanceof ScrollView);
+ }
+}