summaryrefslogtreecommitdiff
path: root/library/src
diff options
context:
space:
mode:
authorYuutaW <17158086+Trumeet@users.noreply.github.com>2019-04-13 12:09:55 -0700
committerYuutaW <17158086+Trumeet@users.noreply.github.com>2019-04-13 12:09:55 -0700
commit7b20783dd6de98dd99aa104b2251eb43aa31cac7 (patch)
treed1a4c5da1caa09175e9d2b8199e6bc0aa481bd5b /library/src
downloadFlow-7b20783dd6de98dd99aa104b2251eb43aa31cac7.tar
Flow-7b20783dd6de98dd99aa104b2251eb43aa31cac7.tar.gz
Flow-7b20783dd6de98dd99aa104b2251eb43aa31cac7.tar.bz2
Flow-7b20783dd6de98dd99aa104b2251eb43aa31cac7.zip
First Commit
Signed-off-by: YuutaW <17158086+Trumeet@users.noreply.github.com>
Diffstat (limited to 'library/src')
-rw-r--r--library/src/androidTest/java/moe/yuuta/flow/ExampleInstrumentedTest.java26
-rw-r--r--library/src/main/AndroidManifest.xml1
-rw-r--r--library/src/main/java/moe/yuuta/flow/FlowFragment.java194
-rw-r--r--library/src/main/java/moe/yuuta/flow/FlowInfo.kt10
-rw-r--r--library/src/main/java/moe/yuuta/flow/Header.java54
-rw-r--r--library/src/main/java/moe/yuuta/flow/HeaderConfig.kt7
-rw-r--r--library/src/main/java/moe/yuuta/flow/IFlowFragment.java17
-rw-r--r--library/src/main/java/moe/yuuta/flow/NavigationBar.java82
-rw-r--r--library/src/main/java/moe/yuuta/flow/NavigationBarConfig.kt16
-rw-r--r--library/src/main/java/moe/yuuta/flow/PageFragment.java30
-rw-r--r--library/src/main/java/moe/yuuta/flow/widgets/FlowPager.java64
-rw-r--r--library/src/main/res/layout/fragment_flow.xml130
-rw-r--r--library/src/main/res/values/attrs.xml4
-rw-r--r--library/src/main/res/values/ids.xml14
-rw-r--r--library/src/main/res/values/strings.xml4
-rw-r--r--library/src/test/java/moe/yuuta/flow/ExampleUnitTest.java17
16 files changed, 670 insertions, 0 deletions
diff --git a/library/src/androidTest/java/moe/yuuta/flow/ExampleInstrumentedTest.java b/library/src/androidTest/java/moe/yuuta/flow/ExampleInstrumentedTest.java
new file mode 100644
index 0000000..435e2e8
--- /dev/null
+++ b/library/src/androidTest/java/moe/yuuta/flow/ExampleInstrumentedTest.java
@@ -0,0 +1,26 @@
+package moe.yuuta.flow;
+
+import android.content.Context;
+import android.support.test.InstrumentationRegistry;
+import android.support.test.runner.AndroidJUnit4;
+
+import org.junit.Test;
+import org.junit.runner.RunWith;
+
+import static org.junit.Assert.assertEquals;
+
+/**
+ * Instrumented test, which will execute on an Android device.
+ *
+ * @see <a href="http://d.android.com/tools/testing">Testing documentation</a>
+ */
+@RunWith(AndroidJUnit4.class)
+public class ExampleInstrumentedTest {
+ @Test
+ public void useAppContext() {
+ // Context of the app under test.
+ Context appContext = InstrumentationRegistry.getInstrumentation().getTargetContext();
+
+ assertEquals("moe.yuuta.flow.test", appContext.getPackageName());
+ }
+}
diff --git a/library/src/main/AndroidManifest.xml b/library/src/main/AndroidManifest.xml
new file mode 100644
index 0000000..f3b86f1
--- /dev/null
+++ b/library/src/main/AndroidManifest.xml
@@ -0,0 +1 @@
+<manifest package="moe.yuuta.flow" />
diff --git a/library/src/main/java/moe/yuuta/flow/FlowFragment.java b/library/src/main/java/moe/yuuta/flow/FlowFragment.java
new file mode 100644
index 0000000..d7d4776
--- /dev/null
+++ b/library/src/main/java/moe/yuuta/flow/FlowFragment.java
@@ -0,0 +1,194 @@
+package moe.yuuta.flow;
+
+import android.os.Bundle;
+import android.view.LayoutInflater;
+import android.view.View;
+import android.view.ViewGroup;
+
+import androidx.annotation.NonNull;
+import androidx.annotation.Nullable;
+import androidx.constraintlayout.widget.ConstraintLayout;
+import androidx.fragment.app.Fragment;
+import androidx.fragment.app.FragmentManager;
+import androidx.fragment.app.FragmentPagerAdapter;
+import androidx.viewpager.widget.ViewPager;
+
+import java.util.ArrayList;
+import java.util.List;
+
+/**
+ * The Host fragment
+ */
+public class FlowFragment extends Fragment implements IFlowFragment, View.OnClickListener {
+ private List<PageFragment> mPages = new ArrayList<>(0);
+ // true: after the view settles, update the UI immediately.
+ private volatile boolean mUIUpdateScheduled;
+
+ private Header mHeader;
+ private ViewPager mPager;
+ private NavigationBar mNav;
+
+ /**
+ * Should only be called once.
+ */
+ public void setPages(@NonNull List<PageFragment> pages) {
+ if (mPages != null && mPages.size() > 0) {
+ throw new IllegalStateException("This method should only be called once. Current size: " + mPages.size());
+ }
+ mPages = pages;
+ // Because this method can be called before setting up the layout, so we need to schedule it until the layout is set up.
+ notifyCurrentFlowInfoUpdated();
+ }
+
+ /**
+ * Update the WHOLE UI.
+ */
+ private void updateUI() {
+ final int currentIndex = mPager.getCurrentItem();
+ final PageFragment currentFragment = mPages.get(currentIndex);
+ if (currentFragment.mInfo == null) {
+ throw new NullPointerException("Info is null");
+ }
+ if (currentFragment.mInfo.getNavigationBarConfig() != null) {
+ mNav.applyInfo(currentFragment.mInfo.getNavigationBarConfig());
+ } else {
+ mNav.applyInfo(new NavigationBarConfig(getString(R.string.flow_nav_bar_next),
+ getString(R.string.flow_nav_bar_previous),
+ currentIndex == 0 ? View.GONE : View.VISIBLE,
+ currentIndex >= (mPages.size() - 1) ? View.GONE : View.VISIBLE,
+ View.VISIBLE,
+ this,
+ this));
+ }
+ mHeader.applyInfo(currentFragment.mInfo.getHeaderConfig());
+ }
+
+ @Override
+ public void onClick(View v) {
+ final int currentIndex = mPager.getCurrentItem();
+ if (v.getId() == R.id.flow_nav_left_button) {
+ if (currentIndex > 0) previousFlow();
+ } else if (v.getId() == R.id.flow_nav_right_button) {
+ if (currentIndex < (mPages.size() - 1)) nextFlow();
+ }
+ }
+
+ @Nullable
+ @Override
+ public View onCreateView(@NonNull LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {
+ final View view = inflater.inflate(R.layout.fragment_flow, container, false);
+ mHeader = new Header(view.<ConstraintLayout>findViewById(R.id.flow_host_header));
+ mNav = new NavigationBar(view.<ConstraintLayout>findViewById(R.id.flow_host_nav));
+ mPager = view.findViewById(R.id.flow_host_pager);
+ mPager.setAdapter(new FragmentPagerAdapter(getChildFragmentManager()) {
+ @Override
+ public Fragment getItem(int position) {
+ PageFragment fragment = mPages.get(position);
+ return fragment;
+ }
+
+ @Override
+ public int getCount() {
+ return mPages.size();
+ }
+ });
+ mPager.addOnPageChangeListener(mPageListener);
+ getChildFragmentManager().registerFragmentLifecycleCallbacks(mCallback, false);
+ return view;
+ }
+
+ @Override
+ public void onDestroyView() {
+ getChildFragmentManager().unregisterFragmentLifecycleCallbacks(mCallback);
+ mPager.removeOnPageChangeListener(mPageListener);
+ super.onDestroyView();
+ }
+
+ @Override
+ public void onViewCreated(@NonNull View view, @Nullable Bundle savedInstanceState) {
+ super.onViewCreated(view, savedInstanceState);
+ if (mUIUpdateScheduled) {
+ updateUI();
+ mUIUpdateScheduled = false;
+ }
+ }
+
+ @Override
+ public void notifyCurrentFlowInfoUpdated() {
+ if (getView() != null) {
+ mUIUpdateScheduled = false;
+ updateUI();
+ } else {
+ mUIUpdateScheduled = true;
+ }
+ }
+
+ @Override
+ public void nextFlow() {
+ switchToFlow(mPager.getCurrentItem() + 1);
+ }
+
+ @Override
+ public void previousFlow() {
+ switchToFlow(mPager.getCurrentItem() - 1);
+ }
+
+ @Override
+ public int getFlowCount() {
+ return mPages.size();
+ }
+
+ @Override
+ public void switchToFlow(int index) {
+ mPager.setCurrentItem(index, true);
+ }
+
+ private FragmentManager.FragmentLifecycleCallbacks mCallback = new FragmentManager.FragmentLifecycleCallbacks() {
+ @Override
+ public void onFragmentPreCreated(@NonNull FragmentManager fm, @NonNull Fragment f, @Nullable Bundle savedInstanceState) {
+ if (f instanceof PageFragment) {
+ final PageFragment pf = (PageFragment) f;
+ pf.setHostFragment(FlowFragment.this);
+ }
+ }
+ };
+
+ private ViewPager.OnPageChangeListener mPageListener = new ViewPager.OnPageChangeListener() {
+ @Override
+ public void onPageScrolled(int position, float positionOffset, int positionOffsetPixels) {
+
+ }
+
+ @Override
+ public void onPageSelected(int position) {
+ notifyCurrentFlowInfoUpdated();
+ }
+
+ @Override
+ public void onPageScrollStateChanged(int state) {
+
+ }
+ };
+
+ /**
+ * @return true: the fragment handled this event and you do not need to call super. false: call super.
+ */
+ public boolean onBackPressed() {
+ if (getView() == null) return false;
+ if (mPages.size() <= 0) return false;
+ final PageFragment pf = mPages.get(mPager.getCurrentItem());
+ if (pf != null) {
+ if (pf.onBackPressed()) return true;
+ }
+ // Default handling
+ if (mPager.getCurrentItem() == 0) return false;
+ mNav.getButton(NavigationBar.ButtonPosition.LEFT).performClick();
+ return true;
+ }
+
+ @Override
+ @NonNull
+ public View.OnClickListener getGeneralFlowNavListener() {
+ return this;
+ }
+}
diff --git a/library/src/main/java/moe/yuuta/flow/FlowInfo.kt b/library/src/main/java/moe/yuuta/flow/FlowInfo.kt
new file mode 100644
index 0000000..d12228c
--- /dev/null
+++ b/library/src/main/java/moe/yuuta/flow/FlowInfo.kt
@@ -0,0 +1,10 @@
+package moe.yuuta.flow;
+
+/**
+ * Stores basic info, for instance, title and subtitle for the flow. These data will
+ * be auto applied when user switch to the related flow and will be reset after leaving the flow.
+ */
+data class FlowInfo(
+ var headerConfig: HeaderConfig,
+ var navigationBarConfig: NavigationBarConfig?
+) \ No newline at end of file
diff --git a/library/src/main/java/moe/yuuta/flow/Header.java b/library/src/main/java/moe/yuuta/flow/Header.java
new file mode 100644
index 0000000..42d9124
--- /dev/null
+++ b/library/src/main/java/moe/yuuta/flow/Header.java
@@ -0,0 +1,54 @@
+package moe.yuuta.flow;
+
+import android.view.View;
+import android.view.ViewGroup;
+import android.widget.ProgressBar;
+import android.widget.TextView;
+
+import androidx.annotation.NonNull;
+
+class Header {
+ public enum WhichView {
+ TITLE,
+ SUBTITLE
+ }
+
+ private ViewGroup mRoot;
+ private TextView mTitle;
+ private TextView mSubtitle;
+ private ProgressBar mProgressBar;
+
+ Header() {}
+
+ Header(@NonNull ViewGroup root) {
+ attach(root);
+ }
+
+ /**
+ * The config will be reset after re-attaching.
+ */
+ void attach(@NonNull ViewGroup root) {
+ mRoot = root;
+ mTitle = mRoot.findViewById(R.id.flow_header_title);
+ mSubtitle = mRoot.findViewById(R.id.flow_header_subtitle);
+ mProgressBar = mRoot.findViewById(R.id.flow_header_progressbar);
+ }
+
+ @NonNull
+ private TextView getText(@NonNull WhichView position) {
+ switch (position) {
+ case TITLE:
+ return mTitle;
+ case SUBTITLE:
+ return mSubtitle;
+ default:
+ throw new IllegalArgumentException("Unexpected position");
+ }
+ }
+
+ void applyInfo(@NonNull HeaderConfig config) {
+ mTitle.setText(config.getTitleText());
+ mSubtitle.setText(config.getSubtitleText());
+ mProgressBar.setVisibility(config.getShowProgressBar() ? View.VISIBLE : View.GONE);
+ }
+}
diff --git a/library/src/main/java/moe/yuuta/flow/HeaderConfig.kt b/library/src/main/java/moe/yuuta/flow/HeaderConfig.kt
new file mode 100644
index 0000000..0133897
--- /dev/null
+++ b/library/src/main/java/moe/yuuta/flow/HeaderConfig.kt
@@ -0,0 +1,7 @@
+package moe.yuuta.flow
+
+class HeaderConfig(
+ var titleText: CharSequence,
+ var subtitleText: CharSequence?,
+ var showProgressBar: Boolean = false
+) \ No newline at end of file
diff --git a/library/src/main/java/moe/yuuta/flow/IFlowFragment.java b/library/src/main/java/moe/yuuta/flow/IFlowFragment.java
new file mode 100644
index 0000000..79f67f6
--- /dev/null
+++ b/library/src/main/java/moe/yuuta/flow/IFlowFragment.java
@@ -0,0 +1,17 @@
+package moe.yuuta.flow;
+
+import android.view.View;
+
+import androidx.annotation.NonNull;
+
+/**
+ * A bridge which is exposed to {@link PageFragment} for controlling the {@link FlowFragment}
+ */
+public interface IFlowFragment {
+ void notifyCurrentFlowInfoUpdated();
+ void nextFlow();
+ void previousFlow();
+ int getFlowCount();
+ void switchToFlow(int index);
+ @NonNull View.OnClickListener getGeneralFlowNavListener();
+}
diff --git a/library/src/main/java/moe/yuuta/flow/NavigationBar.java b/library/src/main/java/moe/yuuta/flow/NavigationBar.java
new file mode 100644
index 0000000..14fae26
--- /dev/null
+++ b/library/src/main/java/moe/yuuta/flow/NavigationBar.java
@@ -0,0 +1,82 @@
+package moe.yuuta.flow;
+
+import android.view.View;
+import android.view.ViewGroup;
+import android.widget.Button;
+
+import androidx.annotation.NonNull;
+import androidx.annotation.Nullable;
+
+class NavigationBar {
+ public enum ButtonPosition {
+ LEFT,
+ RIGHT
+ }
+
+ private ViewGroup mRoot;
+ private Button mLeftButton;
+ private Button mRightButton;
+
+ NavigationBar() {}
+
+ NavigationBar(@NonNull ViewGroup navBarRoot) {
+ attach(navBarRoot);
+ }
+
+ /**
+ * The config will be reset after re-attaching.
+ */
+ void attach(@NonNull ViewGroup navBarRoot) {
+ mRoot = navBarRoot;
+ mLeftButton = mRoot.findViewById(R.id.flow_nav_left_button);
+ mRightButton = mRoot.findViewById(R.id.flow_nav_right_button);
+ }
+
+ @NonNull
+ Button getButton(@NonNull ButtonPosition position) {
+ switch (position) {
+ case LEFT:
+ return mLeftButton;
+ case RIGHT:
+ return mRightButton;
+ default:
+ throw new IllegalArgumentException("Unexpected position");
+ }
+ }
+
+ private void setListener(@NonNull ButtonPosition position, @Nullable View.OnClickListener listener) {
+ switch (position) {
+ case LEFT:
+ mLeftButton.setOnClickListener(listener);
+ break;
+ case RIGHT:
+ mRightButton.setOnClickListener(listener);
+ break;
+ }
+ }
+
+ void applyInfo(@NonNull NavigationBarConfig config) {
+ mLeftButton.setText(config.getLeftButtonText());
+ mRightButton.setText(config.getRightButtonText());
+ setNavigationBarVisibility(config.getNavBarVisibility());
+ setButtonVisibility(ButtonPosition.LEFT, config.getLeftButtonVisibility());
+ setButtonVisibility(ButtonPosition.RIGHT, config.getRightButtonVisibility());
+ setListener(ButtonPosition.LEFT, config.getLeftListener());
+ setListener(ButtonPosition.RIGHT, config.getRightListener());
+ }
+
+ private void setNavigationBarVisibility(@View.Visibility int visibility) {
+ mRoot.setVisibility(visibility);
+ }
+
+ private void setButtonVisibility(@NonNull ButtonPosition position, @View.Visibility int visibility) {
+ switch (position) {
+ case LEFT:
+ mLeftButton.setVisibility(visibility);
+ break;
+ case RIGHT:
+ mRightButton.setVisibility(visibility);
+ break;
+ }
+ }
+}
diff --git a/library/src/main/java/moe/yuuta/flow/NavigationBarConfig.kt b/library/src/main/java/moe/yuuta/flow/NavigationBarConfig.kt
new file mode 100644
index 0000000..e027b73
--- /dev/null
+++ b/library/src/main/java/moe/yuuta/flow/NavigationBarConfig.kt
@@ -0,0 +1,16 @@
+package moe.yuuta.flow
+
+import android.view.View
+
+class NavigationBarConfig(
+ var rightButtonText: CharSequence,
+ var leftButtonText: CharSequence,
+ @field:View.Visibility
+ var leftButtonVisibility: Int,
+ @field:View.Visibility
+ var rightButtonVisibility: Int,
+ @field:View.Visibility
+ var navBarVisibility: Int,
+ var leftListener: View.OnClickListener?,
+ var rightListener: View.OnClickListener?
+) \ No newline at end of file
diff --git a/library/src/main/java/moe/yuuta/flow/PageFragment.java b/library/src/main/java/moe/yuuta/flow/PageFragment.java
new file mode 100644
index 0000000..dcc9f21
--- /dev/null
+++ b/library/src/main/java/moe/yuuta/flow/PageFragment.java
@@ -0,0 +1,30 @@
+package moe.yuuta.flow;
+
+import androidx.annotation.NonNull;
+import androidx.fragment.app.Fragment;
+
+public abstract class PageFragment extends Fragment {
+ private IFlowFragment mHostFragment;
+
+ /**
+ * Once mInfo is changed, you should call {@link IFlowFragment#notifyCurrentFlowInfoUpdated()} to publish it.
+ * Note: it will be permanently change the recorded info.
+ */
+ protected FlowInfo mInfo;
+
+ final void setHostFragment(@NonNull IFlowFragment hostFragment) {
+ this.mHostFragment = hostFragment;
+ }
+
+ @NonNull
+ protected final IFlowFragment getHostFragment() {
+ return mHostFragment;
+ }
+
+ /**
+ * @return true: the fragment handled this event and you do not need to call super. false: call super.
+ */
+ public boolean onBackPressed() {
+ return false;
+ }
+} \ No newline at end of file
diff --git a/library/src/main/java/moe/yuuta/flow/widgets/FlowPager.java b/library/src/main/java/moe/yuuta/flow/widgets/FlowPager.java
new file mode 100644
index 0000000..c821eef
--- /dev/null
+++ b/library/src/main/java/moe/yuuta/flow/widgets/FlowPager.java
@@ -0,0 +1,64 @@
+package moe.yuuta.flow.widgets;
+
+import android.content.Context;
+import android.os.Build;
+import android.util.AttributeSet;
+import android.view.MotionEvent;
+import android.view.View;
+
+import androidx.annotation.NonNull;
+import androidx.annotation.Nullable;
+import androidx.fragment.app.FragmentPagerAdapter;
+import androidx.viewpager.widget.ViewPager;
+
+public class FlowPager extends ViewPager {
+ public FlowPager(@NonNull Context context) {
+ super(context);
+ }
+
+ public FlowPager(@NonNull Context context, @Nullable AttributeSet attrs) {
+ super(context, attrs);
+ }
+
+ // Thanks to https://stackoverflow.com/a/32488566/6792243
+ @Override
+ protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
+ super.onMeasure(widthMeasureSpec, heightMeasureSpec);
+
+ if(null != getAdapter()) {
+ int height = 0;
+ View child = ((FragmentPagerAdapter) getAdapter()).getItem(getCurrentItem()).getView();
+ if (child != null) {
+ child.measure(widthMeasureSpec, MeasureSpec.makeMeasureSpec(0, MeasureSpec.UNSPECIFIED));
+ height = child.getMeasuredHeight();
+ // TODO: Support api-14 and api-15?
+ if (Build.VERSION.SDK_INT >= 16 && height < getMinimumHeight()) {
+ height = getMinimumHeight();
+ }
+ }
+
+ int newHeight = MeasureSpec.makeMeasureSpec(height, MeasureSpec.EXACTLY);
+ if (getLayoutParams().height != 0 && heightMeasureSpec != newHeight) {
+ getLayoutParams().height = height;
+
+ } else {
+ heightMeasureSpec = newHeight;
+ }
+ }
+
+ super.onMeasure(widthMeasureSpec, heightMeasureSpec);
+ }
+
+ // Thanks to https://stackoverflow.com/a/13437997/6792243
+ @Override
+ public boolean onTouchEvent(MotionEvent event) {
+ return false;
+ }
+
+ // Thanks to https://stackoverflow.com/a/13437997/6792243
+ @Override
+ public boolean onInterceptTouchEvent(MotionEvent event) {
+ return false;
+ }
+
+}
diff --git a/library/src/main/res/layout/fragment_flow.xml b/library/src/main/res/layout/fragment_flow.xml
new file mode 100644
index 0000000..98f6fba
--- /dev/null
+++ b/library/src/main/res/layout/fragment_flow.xml
@@ -0,0 +1,130 @@
+<?xml version="1.0" encoding="utf-8"?>
+<androidx.constraintlayout.widget.ConstraintLayout
+ xmlns:android="http://schemas.android.com/apk/res/android"
+ xmlns:tools="http://schemas.android.com/tools"
+ xmlns:app="http://schemas.android.com/apk/res-auto"
+ android:orientation="vertical"
+ android:layout_width="match_parent"
+ android:layout_height="match_parent">
+
+ <androidx.constraintlayout.widget.ConstraintLayout
+ android:id="@id/flow_host_header"
+ app:layout_constraintTop_toTopOf="parent"
+ app:layout_constraintStart_toStartOf="parent"
+ app:layout_constraintEnd_toEndOf="parent"
+ app:layout_constraintLeft_toLeftOf="parent"
+ app:layout_constraintRight_toRightOf="parent"
+ android:layout_width="match_parent"
+ android:layout_height="0dp">
+ <!-- TODO: firstBaselineToTopHeight compat -->
+ <androidx.constraintlayout.widget.ConstraintLayout
+ android:layout_width="0dp"
+ android:layout_height="wrap_content"
+ app:layout_constraintTop_toTopOf="parent"
+ app:layout_constraintStart_toStartOf="parent"
+ app:layout_constraintEnd_toEndOf="parent">
+ <ProgressBar
+ android:id="@+id/flow_header_progressbar"
+ android:layout_width="0dp"
+ android:layout_height="wrap_content"
+ app:layout_constraintStart_toStartOf="parent"
+ app:layout_constraintTop_toTopOf="parent"
+ app:layout_constraintEnd_toEndOf="parent"
+ app:layout_constraintBottom_toTopOf="@id/flow_header_progressbar_guideline"
+ android:indeterminate="true"
+ android:paddingTop="-4dp"
+ style="?android:attr/progressBarStyleHorizontal"/>
+
+ <androidx.constraintlayout.widget.Guideline
+ android:layout_width="wrap_content"
+ android:layout_height="wrap_content"
+ android:id="@id/flow_header_progressbar_guideline"
+ android:orientation="horizontal"
+ app:layout_constraintGuide_begin="2dp" />
+ </androidx.constraintlayout.widget.ConstraintLayout>
+
+ <TextView
+ android:id="@id/flow_header_title"
+ android:layout_width="wrap_content"
+ android:layout_height="wrap_content"
+ android:layout_marginTop="64dp"
+ android:textSize="24sp"
+ android:textColor="?android:attr/textColorPrimary"
+ android:layout_gravity="center_horizontal"
+ android:firstBaselineToTopHeight="32dp"
+ app:layout_constraintTop_toTopOf="parent"
+ app:layout_constraintStart_toStartOf="parent"
+ app:layout_constraintLeft_toLeftOf="parent"
+ app:layout_constraintEnd_toEndOf="parent"
+ app:layout_constraintRight_toRightOf="parent"
+ android:textAppearance="?android:attr/textAppearanceLarge"
+ tools:text="Title" />
+
+ <TextView
+ android:id="@id/flow_header_subtitle"
+ android:layout_width="wrap_content"
+ android:layout_height="wrap_content"
+ android:layout_marginTop="8dp"
+ android:textSize="16sp"
+ android:textColor="?android:attr/textColorPrimary"
+ android:layout_gravity="center_horizontal"
+ android:firstBaselineToTopHeight="24dp"
+ android:textScaleX="1.1"
+ app:layout_constraintTop_toBottomOf="@id/flow_header_title"
+ app:layout_constraintStart_toStartOf="parent"
+ app:layout_constraintLeft_toLeftOf="parent"
+ app:layout_constraintEnd_toEndOf="parent"
+ app:layout_constraintRight_toRightOf="parent"
+ android:textAppearance="?android:attr/textAppearanceMedium"
+ tools:text="Subtitle" />
+ </androidx.constraintlayout.widget.ConstraintLayout>
+
+ <moe.yuuta.flow.widgets.FlowPager
+ android:id="@id/flow_host_pager"
+ android:layout_width="match_parent"
+ android:layout_height="wrap_content"
+ android:layout_marginTop="44dp"
+ app:layout_constraintTop_toBottomOf="@id/flow_host_header"
+ app:layout_constraintStart_toStartOf="parent"
+ app:layout_constraintLeft_toLeftOf="parent"
+ app:layout_constraintEnd_toEndOf="parent"
+ app:layout_constraintRight_toRightOf="parent"/>
+
+ <androidx.constraintlayout.widget.ConstraintLayout
+ android:id="@id/flow_host_nav"
+ android:layout_width="match_parent"
+ android:layout_height="wrap_content"
+ android:layout_marginTop="38dp"
+ android:layout_marginStart="44dp"
+ android:layout_marginLeft="44dp"
+ android:layout_marginEnd="44dp"
+ android:layout_marginRight="44dp"
+ app:layout_constraintTop_toBottomOf="@id/flow_host_pager"
+ app:layout_constraintStart_toStartOf="parent"
+ app:layout_constraintLeft_toLeftOf="parent"
+ app:layout_constraintEnd_toEndOf="parent"
+ app:layout_constraintRight_toRightOf="parent"
+ style="?android:attr/buttonBarStyle">
+ <Button
+ android:id="@id/flow_nav_left_button"
+ android:layout_width="wrap_content"
+ android:layout_height="wrap_content"
+ app:layout_constraintStart_toStartOf="parent"
+ app:layout_constraintTop_toTopOf="parent"
+ app:layout_constraintBottom_toBottomOf="parent"
+ app:layout_constraintLeft_toLeftOf="parent"
+ tools:text="Back"
+ style="?android:attr/buttonBarButtonStyle"/>
+
+ <Button
+ android:id="@id/flow_nav_right_button"
+ android:layout_width="wrap_content"
+ android:layout_height="wrap_content"
+ app:layout_constraintRight_toRightOf="parent"
+ app:layout_constraintTop_toTopOf="parent"
+ app:layout_constraintBottom_toBottomOf="parent"
+ app:layout_constraintEnd_toEndOf="parent"
+ tools:text="Next"
+ style="?attr/positiveButtonStyle" />
+ </androidx.constraintlayout.widget.ConstraintLayout>
+</androidx.constraintlayout.widget.ConstraintLayout> \ No newline at end of file
diff --git a/library/src/main/res/values/attrs.xml b/library/src/main/res/values/attrs.xml
new file mode 100644
index 0000000..0e32e81
--- /dev/null
+++ b/library/src/main/res/values/attrs.xml
@@ -0,0 +1,4 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <attr name="positiveButtonStyle" format="reference" />
+</resources> \ No newline at end of file
diff --git a/library/src/main/res/values/ids.xml b/library/src/main/res/values/ids.xml
new file mode 100644
index 0000000..a3117c8
--- /dev/null
+++ b/library/src/main/res/values/ids.xml
@@ -0,0 +1,14 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <item name="flow_host_header" type="id" />
+ <item name="flow_host_pager" type="id" />
+ <item name="flow_host_nav" type="id" />
+
+ <item name="flow_nav_left_button" type="id" />
+ <item name="flow_nav_right_button" type="id" />
+
+ <item name="flow_header_title" type="id" />
+ <item name="flow_header_subtitle" type="id" />
+ <item name="flow_header_progressbar" type="id" />
+ <item name="flow_header_progressbar_guideline" type="id" />
+</resources> \ No newline at end of file
diff --git a/library/src/main/res/values/strings.xml b/library/src/main/res/values/strings.xml
new file mode 100644
index 0000000..ad04d0e
--- /dev/null
+++ b/library/src/main/res/values/strings.xml
@@ -0,0 +1,4 @@
+<resources>
+ <string name="flow_nav_bar_previous">Back</string>
+ <string name="flow_nav_bar_next">Next</string>
+</resources>
diff --git a/library/src/test/java/moe/yuuta/flow/ExampleUnitTest.java b/library/src/test/java/moe/yuuta/flow/ExampleUnitTest.java
new file mode 100644
index 0000000..84848a0
--- /dev/null
+++ b/library/src/test/java/moe/yuuta/flow/ExampleUnitTest.java
@@ -0,0 +1,17 @@
+package moe.yuuta.flow;
+
+import org.junit.Test;
+
+import static org.junit.Assert.assertEquals;
+
+/**
+ * Example local unit test, which will execute on the development machine (host).
+ *
+ * @see <a href="http://d.android.com/tools/testing">Testing documentation</a>
+ */
+public class ExampleUnitTest {
+ @Test
+ public void addition_isCorrect() {
+ assertEquals(4, 2 + 2);
+ }
+} \ No newline at end of file