├── sample
├── .gitignore
├── src
│ ├── main
│ │ ├── res
│ │ │ ├── mipmap-hdpi
│ │ │ │ └── ic_launcher.png
│ │ │ ├── mipmap-mdpi
│ │ │ │ └── ic_launcher.png
│ │ │ ├── mipmap-xhdpi
│ │ │ │ └── ic_launcher.png
│ │ │ ├── mipmap-xxhdpi
│ │ │ │ └── ic_launcher.png
│ │ │ ├── mipmap-xxxhdpi
│ │ │ │ └── ic_launcher.png
│ │ │ ├── drawable-xhdpi
│ │ │ │ ├── sample_footer_error.png
│ │ │ │ └── sample_footer_loading.png
│ │ │ ├── values
│ │ │ │ ├── colors.xml
│ │ │ │ ├── strings.xml
│ │ │ │ ├── styles.xml
│ │ │ │ └── dimens.xml
│ │ │ ├── values-w820dp
│ │ │ │ └── dimens.xml
│ │ │ ├── layout
│ │ │ │ ├── sample_activity.xml
│ │ │ │ ├── sample_common_list_footer_end.xml
│ │ │ │ ├── sample_common_list_footer_network_error.xml
│ │ │ │ ├── sample_common_list_footer_loading.xml
│ │ │ │ ├── sample_common_list_footer.xml
│ │ │ │ ├── sample_footer.xml
│ │ │ │ ├── sample_header.xml
│ │ │ │ ├── sample_item_card.xml
│ │ │ │ └── sample_item_text.xml
│ │ │ └── drawable
│ │ │ │ └── sample_footer_loading_progress.xml
│ │ ├── java
│ │ │ └── com
│ │ │ │ └── cundong
│ │ │ │ └── recyclerview
│ │ │ │ └── sample
│ │ │ │ ├── ItemModel.java
│ │ │ │ ├── weight
│ │ │ │ ├── SampleFooter.java
│ │ │ │ ├── SampleHeader.java
│ │ │ │ └── LoadingFooter.java
│ │ │ │ ├── utils
│ │ │ │ ├── NetworkUtils.java
│ │ │ │ └── RecyclerViewStateUtils.java
│ │ │ │ ├── MainActivity.java
│ │ │ │ ├── LinearLayoutActivity.java
│ │ │ │ ├── EndlessGridLayoutActivity.java
│ │ │ │ ├── EndlessStaggeredGridLayoutActivity.java
│ │ │ │ └── EndlessLinearLayoutActivity.java
│ │ └── AndroidManifest.xml
│ ├── test
│ │ └── java
│ │ │ └── com
│ │ │ └── cundong
│ │ │ └── recyclerview
│ │ │ └── ExampleUnitTest.java
│ └── androidTest
│ │ └── java
│ │ └── com
│ │ └── cundong
│ │ └── recyclerview
│ │ └── ApplicationTest.java
├── proguard-rules.pro
├── build.gradle
└── sample.iml
├── library
├── .gitignore
├── src
│ ├── main
│ │ ├── res
│ │ │ └── values
│ │ │ │ └── strings.xml
│ │ ├── AndroidManifest.xml
│ │ └── java
│ │ │ └── com
│ │ │ └── cundong
│ │ │ └── recyclerview
│ │ │ ├── OnListLoadNextPageListener.java
│ │ │ ├── HeaderSpanSizeLookup.java
│ │ │ ├── ExStaggeredGridLayoutManager.java
│ │ │ ├── EndlessRecyclerOnScrollListener.java
│ │ │ ├── RecyclerViewUtils.java
│ │ │ └── HeaderAndFooterRecyclerViewAdapter.java
│ ├── test
│ │ └── java
│ │ │ └── com
│ │ │ └── cundong
│ │ │ └── recyclerview
│ │ │ └── ExampleUnitTest.java
│ └── androidTest
│ │ └── java
│ │ └── com
│ │ └── cundong
│ │ └── recyclerview
│ │ └── ApplicationTest.java
├── build.gradle
├── proguard-rules.pro
└── library.iml
├── .idea
├── .name
├── copyright
│ └── profiles_settings.xml
├── encodings.xml
├── vcs.xml
├── modules.xml
├── runConfigurations.xml
├── compiler.xml
├── gradle.xml
└── misc.xml
├── settings.gradle
├── art
├── art1.png
├── art2.png
├── art3.png
├── art4.png
└── art5.png
├── .gitignore
├── gradle
└── wrapper
│ ├── gradle-wrapper.jar
│ └── gradle-wrapper.properties
├── gradle.properties
├── HeaderAndFooterRecyclerView.iml
├── gradlew.bat
├── README.md
├── gradlew
└── LICENSE
/sample/.gitignore:
--------------------------------------------------------------------------------
1 | /build
2 |
--------------------------------------------------------------------------------
/library/.gitignore:
--------------------------------------------------------------------------------
1 | /build
2 |
--------------------------------------------------------------------------------
/.idea/.name:
--------------------------------------------------------------------------------
1 | HeaderAndFooterRecyclerView
--------------------------------------------------------------------------------
/settings.gradle:
--------------------------------------------------------------------------------
1 | include ':sample', ':library'
2 |
--------------------------------------------------------------------------------
/art/art1.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/bayshier/HeaderAndFooterRecyclerView/master/art/art1.png
--------------------------------------------------------------------------------
/art/art2.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/bayshier/HeaderAndFooterRecyclerView/master/art/art2.png
--------------------------------------------------------------------------------
/art/art3.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/bayshier/HeaderAndFooterRecyclerView/master/art/art3.png
--------------------------------------------------------------------------------
/art/art4.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/bayshier/HeaderAndFooterRecyclerView/master/art/art4.png
--------------------------------------------------------------------------------
/art/art5.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/bayshier/HeaderAndFooterRecyclerView/master/art/art5.png
--------------------------------------------------------------------------------
/.idea/copyright/profiles_settings.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
--------------------------------------------------------------------------------
/library/src/main/res/values/strings.xml:
--------------------------------------------------------------------------------
1 |
2 | Library
3 |
4 |
--------------------------------------------------------------------------------
/.gitignore:
--------------------------------------------------------------------------------
1 | .gradle
2 | /local.properties
3 | /.idea/workspace.xml
4 | /.idea/libraries
5 | .DS_Store
6 | /build
7 | /captures
8 |
--------------------------------------------------------------------------------
/gradle/wrapper/gradle-wrapper.jar:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/bayshier/HeaderAndFooterRecyclerView/master/gradle/wrapper/gradle-wrapper.jar
--------------------------------------------------------------------------------
/sample/src/main/res/mipmap-hdpi/ic_launcher.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/bayshier/HeaderAndFooterRecyclerView/master/sample/src/main/res/mipmap-hdpi/ic_launcher.png
--------------------------------------------------------------------------------
/sample/src/main/res/mipmap-mdpi/ic_launcher.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/bayshier/HeaderAndFooterRecyclerView/master/sample/src/main/res/mipmap-mdpi/ic_launcher.png
--------------------------------------------------------------------------------
/sample/src/main/res/mipmap-xhdpi/ic_launcher.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/bayshier/HeaderAndFooterRecyclerView/master/sample/src/main/res/mipmap-xhdpi/ic_launcher.png
--------------------------------------------------------------------------------
/sample/src/main/res/mipmap-xxhdpi/ic_launcher.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/bayshier/HeaderAndFooterRecyclerView/master/sample/src/main/res/mipmap-xxhdpi/ic_launcher.png
--------------------------------------------------------------------------------
/sample/src/main/res/mipmap-xxxhdpi/ic_launcher.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/bayshier/HeaderAndFooterRecyclerView/master/sample/src/main/res/mipmap-xxxhdpi/ic_launcher.png
--------------------------------------------------------------------------------
/sample/src/main/res/drawable-xhdpi/sample_footer_error.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/bayshier/HeaderAndFooterRecyclerView/master/sample/src/main/res/drawable-xhdpi/sample_footer_error.png
--------------------------------------------------------------------------------
/.idea/encodings.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
--------------------------------------------------------------------------------
/.idea/vcs.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
--------------------------------------------------------------------------------
/sample/src/main/res/drawable-xhdpi/sample_footer_loading.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/bayshier/HeaderAndFooterRecyclerView/master/sample/src/main/res/drawable-xhdpi/sample_footer_loading.png
--------------------------------------------------------------------------------
/sample/src/main/java/com/cundong/recyclerview/sample/ItemModel.java:
--------------------------------------------------------------------------------
1 | package com.cundong.recyclerview.sample;
2 |
3 | /**
4 | * Created by liucundong on 2015/11/17.
5 | */
6 | public class ItemModel {
7 | public long id;
8 | public String title;
9 | }
--------------------------------------------------------------------------------
/gradle/wrapper/gradle-wrapper.properties:
--------------------------------------------------------------------------------
1 | #Fri Oct 30 19:02:37 CST 2015
2 | distributionBase=GRADLE_USER_HOME
3 | distributionPath=wrapper/dists
4 | zipStoreBase=GRADLE_USER_HOME
5 | zipStorePath=wrapper/dists
6 | distributionUrl=https\://services.gradle.org/distributions/gradle-2.4-all.zip
7 |
--------------------------------------------------------------------------------
/library/src/main/AndroidManifest.xml:
--------------------------------------------------------------------------------
1 |
3 |
4 |
8 |
9 |
10 |
11 |
12 |
--------------------------------------------------------------------------------
/sample/src/main/res/values/colors.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 | #3F51B5
4 | #303F9F
5 | #FF4081
6 |
7 | #F0F0F0
8 | #8A000000
9 |
10 |
--------------------------------------------------------------------------------
/library/build.gradle:
--------------------------------------------------------------------------------
1 | apply plugin: 'com.android.library'
2 |
3 | android {
4 | compileSdkVersion 23
5 | buildToolsVersion "23.0.1"
6 |
7 | defaultConfig {
8 | minSdkVersion 14
9 | targetSdkVersion 23
10 | versionCode 1
11 | versionName "1.0"
12 | }
13 | }
14 |
15 | dependencies {
16 | compile 'com.android.support:recyclerview-v7:23.1.0'
17 | }
18 |
--------------------------------------------------------------------------------
/sample/src/main/res/values-w820dp/dimens.xml:
--------------------------------------------------------------------------------
1 |
2 |
5 | 64dp
6 |
7 |
--------------------------------------------------------------------------------
/library/src/test/java/com/cundong/recyclerview/ExampleUnitTest.java:
--------------------------------------------------------------------------------
1 | package com.cundong.recyclerview;
2 |
3 | import org.junit.Test;
4 |
5 | import static org.junit.Assert.*;
6 |
7 | /**
8 | * To work on unit tests, switch the Test Artifact in the Build Variants view.
9 | */
10 | public class ExampleUnitTest {
11 | @Test
12 | public void addition_isCorrect() throws Exception {
13 | assertEquals(4, 2 + 2);
14 | }
15 | }
--------------------------------------------------------------------------------
/sample/src/main/res/values/strings.xml:
--------------------------------------------------------------------------------
1 |
2 | RecyclerView
3 |
4 | The Header
5 | The Footer
6 |
7 |
8 | 正在加载中…
9 | 点击重新加载
10 | 已经到底了
11 |
12 |
--------------------------------------------------------------------------------
/sample/src/test/java/com/cundong/recyclerview/ExampleUnitTest.java:
--------------------------------------------------------------------------------
1 | package com.cundong.recyclerview;
2 |
3 | import org.junit.Test;
4 |
5 | import static org.junit.Assert.*;
6 |
7 | /**
8 | * To work on unit tests, switch the Test Artifact in the Build Variants view.
9 | */
10 | public class ExampleUnitTest {
11 | @Test
12 | public void addition_isCorrect() throws Exception {
13 | assertEquals(4, 2 + 2);
14 | }
15 | }
--------------------------------------------------------------------------------
/sample/src/main/res/values/styles.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
10 |
11 |
12 |
--------------------------------------------------------------------------------
/library/src/androidTest/java/com/cundong/recyclerview/ApplicationTest.java:
--------------------------------------------------------------------------------
1 | package com.cundong.recyclerview;
2 |
3 | import android.app.Application;
4 | import android.test.ApplicationTestCase;
5 |
6 | /**
7 | * Testing Fundamentals
8 | */
9 | public class ApplicationTest extends ApplicationTestCase {
10 | public ApplicationTest() {
11 | super(Application.class);
12 | }
13 | }
--------------------------------------------------------------------------------
/sample/src/androidTest/java/com/cundong/recyclerview/ApplicationTest.java:
--------------------------------------------------------------------------------
1 | package com.cundong.recyclerview;
2 |
3 | import android.app.Application;
4 | import android.test.ApplicationTestCase;
5 |
6 | /**
7 | * Testing Fundamentals
8 | */
9 | public class ApplicationTest extends ApplicationTestCase {
10 | public ApplicationTest() {
11 | super(Application.class);
12 | }
13 | }
--------------------------------------------------------------------------------
/library/src/main/java/com/cundong/recyclerview/OnListLoadNextPageListener.java:
--------------------------------------------------------------------------------
1 | package com.cundong.recyclerview;
2 |
3 | import android.view.View;
4 |
5 | /**
6 | * Created by cundong on 2015/10/9.
7 | * RecyclerView/ListView/GridView 滑动加载下一页时的回调接口
8 | */
9 | public interface OnListLoadNextPageListener {
10 |
11 | /**
12 | * 开始加载下一页
13 | *
14 | * @param view 当前RecyclerView/ListView/GridView
15 | */
16 | public void onLoadNextPage(View view);
17 | }
18 |
--------------------------------------------------------------------------------
/sample/src/main/res/layout/sample_activity.xml:
--------------------------------------------------------------------------------
1 |
6 |
7 |
11 |
12 |
--------------------------------------------------------------------------------
/sample/src/main/res/drawable/sample_footer_loading_progress.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 | -
5 |
12 |
13 |
14 |
--------------------------------------------------------------------------------
/.idea/modules.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
10 |
--------------------------------------------------------------------------------
/.idea/runConfigurations.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
11 |
12 |
--------------------------------------------------------------------------------
/sample/src/main/res/layout/sample_common_list_footer_end.xml:
--------------------------------------------------------------------------------
1 |
2 |
7 |
8 |
16 |
--------------------------------------------------------------------------------
/library/proguard-rules.pro:
--------------------------------------------------------------------------------
1 | # Add project specific ProGuard rules here.
2 | # By default, the flags in this file are appended to flags specified
3 | # in D:\Users\liucundong\AppData\Local\Android\sdk/tools/proguard/proguard-android.txt
4 | # You can edit the include path and order by changing the proguardFiles
5 | # directive in build.gradle.
6 | #
7 | # For more details, see
8 | # http://developer.android.com/guide/developing/tools/proguard.html
9 |
10 | # Add any project specific keep options here:
11 |
12 | # If your project uses WebView with JS, uncomment the following
13 | # and specify the fully qualified class name to the JavaScript interface
14 | # class:
15 | #-keepclassmembers class fqcn.of.javascript.interface.for.webview {
16 | # public *;
17 | #}
18 |
--------------------------------------------------------------------------------
/sample/proguard-rules.pro:
--------------------------------------------------------------------------------
1 | # Add project specific ProGuard rules here.
2 | # By default, the flags in this file are appended to flags specified
3 | # in D:\Users\liucundong\AppData\Local\Android\sdk/tools/proguard/proguard-android.txt
4 | # You can edit the include path and order by changing the proguardFiles
5 | # directive in build.gradle.
6 | #
7 | # For more details, see
8 | # http://developer.android.com/guide/developing/tools/proguard.html
9 |
10 | # Add any project specific keep options here:
11 |
12 | # If your project uses WebView with JS, uncomment the following
13 | # and specify the fully qualified class name to the JavaScript interface
14 | # class:
15 | #-keepclassmembers class fqcn.of.javascript.interface.for.webview {
16 | # public *;
17 | #}
18 |
--------------------------------------------------------------------------------
/sample/src/main/res/values/dimens.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 | 16dp
4 | 16dp
5 |
6 | 0.66dp
7 | 4dp
8 | 10dp
9 | 14dp
10 | 22dp
11 | 36dp
12 | 40dp
13 | 60dp
14 | 72dp
15 |
16 | 12sp
17 | 14sp
18 | 16sp
19 |
20 |
--------------------------------------------------------------------------------
/.idea/compiler.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
10 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
18 |
19 |
20 |
21 |
22 |
23 |
--------------------------------------------------------------------------------
/.idea/gradle.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
10 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
18 |
19 |
20 |
--------------------------------------------------------------------------------
/sample/build.gradle:
--------------------------------------------------------------------------------
1 | apply plugin: 'com.android.application'
2 |
3 | android {
4 | compileSdkVersion 23
5 | buildToolsVersion "23.0.1"
6 |
7 | defaultConfig {
8 | applicationId "com.cundong.recyclerview.sample"
9 | minSdkVersion 14
10 | targetSdkVersion 23
11 | versionCode 1
12 | versionName "1.0"
13 | }
14 | buildTypes {
15 | release {
16 | minifyEnabled false
17 | proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
18 | }
19 | }
20 | }
21 |
22 | dependencies {
23 | compile fileTree(dir: 'libs', include: ['*.jar'])
24 | compile project(path: ':library')
25 |
26 | compile 'com.android.support:design:23.1.0'
27 | compile 'com.android.support:support-v4:23.1.0'
28 | compile 'com.android.support:cardview-v7:23.1.0'
29 | }
--------------------------------------------------------------------------------
/library/src/main/java/com/cundong/recyclerview/HeaderSpanSizeLookup.java:
--------------------------------------------------------------------------------
1 | package com.cundong.recyclerview;
2 |
3 | import android.support.v7.widget.GridLayoutManager;
4 |
5 | /**
6 | * Created by cundong on 2015/10/23.
7 | *
8 | * RecyclerView为GridLayoutManager时,设置了HeaderView,就会用到这个SpanSizeLookup
9 | */
10 | public class HeaderSpanSizeLookup extends GridLayoutManager.SpanSizeLookup {
11 |
12 | private HeaderAndFooterRecyclerViewAdapter adapter;
13 | private int mSpanSize = 1;
14 |
15 | public HeaderSpanSizeLookup(HeaderAndFooterRecyclerViewAdapter adapter, int spanSize) {
16 | this.adapter = adapter;
17 | this.mSpanSize = spanSize;
18 | }
19 |
20 | @Override
21 | public int getSpanSize(int position) {
22 | boolean isHeaderOrFooter = adapter.isHeader(position) || adapter.isFooter(position);
23 | return isHeaderOrFooter ? mSpanSize : 1;
24 | }
25 | }
--------------------------------------------------------------------------------
/gradle.properties:
--------------------------------------------------------------------------------
1 | # Project-wide Gradle settings.
2 |
3 | # IDE (e.g. Android Studio) users:
4 | # Gradle settings configured through the IDE *will override*
5 | # any settings specified in this file.
6 |
7 | # For more details on how to configure your build environment visit
8 | # http://www.gradle.org/docs/current/userguide/build_environment.html
9 |
10 | # Specifies the JVM arguments used for the daemon process.
11 | # The setting is particularly useful for tweaking memory settings.
12 | # Default value: -Xmx10248m -XX:MaxPermSize=256m
13 | # org.gradle.jvmargs=-Xmx2048m -XX:MaxPermSize=512m -XX:+HeapDumpOnOutOfMemoryError -Dfile.encoding=UTF-8
14 |
15 | # When configured, Gradle will run in incubating parallel mode.
16 | # This option should only be used with decoupled projects. More details, visit
17 | # http://www.gradle.org/docs/current/userguide/multi_project_builds.html#sec:decoupled_projects
18 | # org.gradle.parallel=true
--------------------------------------------------------------------------------
/sample/src/main/res/layout/sample_common_list_footer_network_error.xml:
--------------------------------------------------------------------------------
1 |
2 |
7 |
8 |
13 |
14 |
22 |
--------------------------------------------------------------------------------
/sample/src/main/java/com/cundong/recyclerview/sample/weight/SampleFooter.java:
--------------------------------------------------------------------------------
1 | package com.cundong.recyclerview.sample.weight;
2 |
3 | import android.content.Context;
4 | import android.util.AttributeSet;
5 | import android.widget.RelativeLayout;
6 |
7 | import com.cundong.recyclerview.sample.R;
8 |
9 | /**
10 | * Created by cundong on 2015/10/9.
11 | *
12 | * RecyclerView的FooterView,简单的展示一个TextView
13 | */
14 | public class SampleFooter extends RelativeLayout {
15 |
16 | public SampleFooter(Context context) {
17 | super(context);
18 | init(context);
19 | }
20 |
21 | public SampleFooter(Context context, AttributeSet attrs) {
22 | super(context, attrs);
23 | init(context);
24 | }
25 |
26 | public SampleFooter(Context context, AttributeSet attrs, int defStyleAttr) {
27 | super(context, attrs, defStyleAttr);
28 | init(context);
29 | }
30 |
31 | public void init(Context context) {
32 |
33 | inflate(context, R.layout.sample_footer, this);
34 | }
35 | }
--------------------------------------------------------------------------------
/sample/src/main/java/com/cundong/recyclerview/sample/weight/SampleHeader.java:
--------------------------------------------------------------------------------
1 | package com.cundong.recyclerview.sample.weight;
2 |
3 | import android.content.Context;
4 | import android.util.AttributeSet;
5 | import android.widget.RelativeLayout;
6 |
7 | import com.cundong.recyclerview.sample.R;
8 |
9 | /**
10 | * Created by cundong on 2015/10/9.
11 | *
12 | * RecyclerView的HeaderView,简单的展示一个TextView
13 | */
14 | public class SampleHeader extends RelativeLayout {
15 |
16 | public SampleHeader(Context context) {
17 | super(context);
18 | init(context);
19 | }
20 |
21 | public SampleHeader(Context context, AttributeSet attrs) {
22 | super(context, attrs);
23 | init(context);
24 | }
25 |
26 | public SampleHeader(Context context, AttributeSet attrs, int defStyleAttr) {
27 | super(context, attrs, defStyleAttr);
28 | init(context);
29 | }
30 |
31 | public void init(Context context) {
32 |
33 | inflate(context, R.layout.sample_header, this);
34 | }
35 | }
--------------------------------------------------------------------------------
/HeaderAndFooterRecyclerView.iml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
10 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
18 |
19 |
--------------------------------------------------------------------------------
/sample/src/main/res/layout/sample_common_list_footer_loading.xml:
--------------------------------------------------------------------------------
1 |
2 |
8 |
9 |
15 |
16 |
24 |
--------------------------------------------------------------------------------
/sample/src/main/res/layout/sample_common_list_footer.xml:
--------------------------------------------------------------------------------
1 |
2 |
8 |
9 |
14 |
15 |
20 |
21 |
26 |
27 |
--------------------------------------------------------------------------------
/sample/src/main/res/layout/sample_footer.xml:
--------------------------------------------------------------------------------
1 |
2 |
8 |
9 |
13 |
14 |
24 |
25 |
29 |
30 |
--------------------------------------------------------------------------------
/sample/src/main/res/layout/sample_header.xml:
--------------------------------------------------------------------------------
1 |
2 |
8 |
9 |
13 |
14 |
24 |
25 |
29 |
30 |
--------------------------------------------------------------------------------
/sample/src/main/res/layout/sample_item_card.xml:
--------------------------------------------------------------------------------
1 |
2 |
7 |
8 |
18 |
19 |
25 |
26 |
27 |
28 |
--------------------------------------------------------------------------------
/sample/src/main/AndroidManifest.xml:
--------------------------------------------------------------------------------
1 |
2 |
4 |
5 |
6 |
7 |
8 |
9 |
15 |
16 |
17 |
18 |
19 |
20 |
21 |
22 |
23 |
24 |
25 |
26 |
27 |
28 |
29 |
--------------------------------------------------------------------------------
/sample/src/main/res/layout/sample_item_text.xml:
--------------------------------------------------------------------------------
1 |
2 |
7 |
8 |
18 |
19 |
27 |
28 |
29 |
--------------------------------------------------------------------------------
/.idea/misc.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
10 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
18 |
19 |
20 |
21 |
22 |
23 |
24 |
25 |
26 |
27 |
28 |
29 |
30 |
31 |
32 |
33 |
34 |
35 |
36 |
37 |
38 |
39 |
40 |
41 |
42 |
43 |
44 |
45 |
46 |
--------------------------------------------------------------------------------
/gradlew.bat:
--------------------------------------------------------------------------------
1 | @if "%DEBUG%" == "" @echo off
2 | @rem ##########################################################################
3 | @rem
4 | @rem Gradle startup script for Windows
5 | @rem
6 | @rem ##########################################################################
7 |
8 | @rem Set local scope for the variables with windows NT shell
9 | if "%OS%"=="Windows_NT" setlocal
10 |
11 | @rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
12 | set DEFAULT_JVM_OPTS=
13 |
14 | set DIRNAME=%~dp0
15 | if "%DIRNAME%" == "" set DIRNAME=.
16 | set APP_BASE_NAME=%~n0
17 | set APP_HOME=%DIRNAME%
18 |
19 | @rem Find java.exe
20 | if defined JAVA_HOME goto findJavaFromJavaHome
21 |
22 | set JAVA_EXE=java.exe
23 | %JAVA_EXE% -version >NUL 2>&1
24 | if "%ERRORLEVEL%" == "0" goto init
25 |
26 | echo.
27 | echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
28 | echo.
29 | echo Please set the JAVA_HOME variable in your environment to match the
30 | echo location of your Java installation.
31 |
32 | goto fail
33 |
34 | :findJavaFromJavaHome
35 | set JAVA_HOME=%JAVA_HOME:"=%
36 | set JAVA_EXE=%JAVA_HOME%/bin/java.exe
37 |
38 | if exist "%JAVA_EXE%" goto init
39 |
40 | echo.
41 | echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME%
42 | echo.
43 | echo Please set the JAVA_HOME variable in your environment to match the
44 | echo location of your Java installation.
45 |
46 | goto fail
47 |
48 | :init
49 | @rem Get command-line arguments, handling Windowz variants
50 |
51 | if not "%OS%" == "Windows_NT" goto win9xME_args
52 | if "%@eval[2+2]" == "4" goto 4NT_args
53 |
54 | :win9xME_args
55 | @rem Slurp the command line arguments.
56 | set CMD_LINE_ARGS=
57 | set _SKIP=2
58 |
59 | :win9xME_args_slurp
60 | if "x%~1" == "x" goto execute
61 |
62 | set CMD_LINE_ARGS=%*
63 | goto execute
64 |
65 | :4NT_args
66 | @rem Get arguments from the 4NT Shell from JP Software
67 | set CMD_LINE_ARGS=%$
68 |
69 | :execute
70 | @rem Setup the command line
71 |
72 | set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar
73 |
74 | @rem Execute Gradle
75 | "%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %CMD_LINE_ARGS%
76 |
77 | :end
78 | @rem End local scope for the variables with windows NT shell
79 | if "%ERRORLEVEL%"=="0" goto mainEnd
80 |
81 | :fail
82 | rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of
83 | rem the _cmd.exe /c_ return code!
84 | if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1
85 | exit /b 1
86 |
87 | :mainEnd
88 | if "%OS%"=="Windows_NT" endlocal
89 |
90 | :omega
91 |
--------------------------------------------------------------------------------
/library/src/main/java/com/cundong/recyclerview/ExStaggeredGridLayoutManager.java:
--------------------------------------------------------------------------------
1 | package com.cundong.recyclerview;
2 |
3 | import android.support.v7.widget.GridLayoutManager;
4 | import android.support.v7.widget.RecyclerView;
5 | import android.support.v7.widget.StaggeredGridLayoutManager;
6 | import android.view.View;
7 |
8 | /**
9 | * Created by cundong on 2015/10/9.
10 | *
11 | * 拓展的StaggeredGridLayoutManager,tks @Jack Tony
12 | */
13 | public class ExStaggeredGridLayoutManager extends StaggeredGridLayoutManager {
14 |
15 | private final String TAG = getClass().getSimpleName();
16 |
17 | GridLayoutManager.SpanSizeLookup mSpanSizeLookup;
18 |
19 | public ExStaggeredGridLayoutManager(int spanCount, int orientation) {
20 | super(spanCount, orientation);
21 | }
22 |
23 | /**
24 | * Returns the current used by the GridLayoutManager.
25 | *
26 | * @return The current used by the GridLayoutManager.
27 | */
28 | public GridLayoutManager.SpanSizeLookup getSpanSizeLookup() {
29 | return mSpanSizeLookup;
30 | }
31 |
32 | /**
33 | * 设置某个位置的item的跨列程度,这里和GridLayoutManager有点不一样,
34 | * 如果你设置某个位置的item的span>1了,那么这个item会占据所有列
35 | *
36 | * @param spanSizeLookup instance to be used to query number of spans
37 | * occupied by each item
38 | */
39 | public void setSpanSizeLookup(GridLayoutManager.SpanSizeLookup spanSizeLookup) {
40 | mSpanSizeLookup = spanSizeLookup;
41 | }
42 |
43 | @Override
44 | public void onMeasure(RecyclerView.Recycler recycler, RecyclerView.State state, int widthSpec, int heightSpec) {
45 | //Log.d(TAG, "item count = " + getItemCount());
46 | for (int i = 0; i < getItemCount(); i++) {
47 |
48 | if (mSpanSizeLookup.getSpanSize(i) > 1) {
49 | //Log.d(TAG, "lookup > 1 = " + i);
50 | try {
51 | //fix 动态添加时报IndexOutOfBoundsException
52 | View view = recycler.getViewForPosition(i);
53 | if (view != null) {
54 | /**
55 | *占用所有的列
56 | * @see https://plus.google.com/+EtienneLawlor/posts/c5T7fu9ujqi
57 | */
58 | StaggeredGridLayoutManager.LayoutParams lp = (StaggeredGridLayoutManager.LayoutParams) view.getLayoutParams();
59 | lp.setFullSpan(true);
60 | }
61 | // recycler.recycleView(view);
62 | } catch (Exception e) {
63 | e.printStackTrace();
64 | }
65 | }
66 | }
67 | super.onMeasure(recycler, state, widthSpec, heightSpec);
68 | }
69 | }
--------------------------------------------------------------------------------
/sample/src/main/java/com/cundong/recyclerview/sample/utils/NetworkUtils.java:
--------------------------------------------------------------------------------
1 | package com.cundong.recyclerview.sample.utils;
2 |
3 | import android.content.Context;
4 | import android.content.pm.PackageManager;
5 | import android.net.ConnectivityManager;
6 | import android.net.NetworkInfo;
7 |
8 | /**
9 | * Created by cundong on 2015/10/9.
10 | *
11 | * 网络使用工具类
12 | */
13 | public class NetworkUtils {
14 |
15 | /**
16 | * 判断是不是wifi网络状态
17 | *
18 | * @param paramContext
19 | * @return
20 | */
21 | public static boolean isWifi(Context paramContext) {
22 | return "2".equals(getNetType(paramContext)[0]);
23 | }
24 |
25 | /**
26 | * 判断是不是2/3G网络状态
27 | *
28 | * @param paramContext
29 | * @return
30 | */
31 | public static boolean isMobile(Context paramContext) {
32 | return "1".equals(getNetType(paramContext)[0]);
33 | }
34 |
35 | /**
36 | * 网络是否可用
37 | *
38 | * @param paramContext
39 | * @return
40 | */
41 | public static boolean isNetAvailable(Context paramContext) {
42 | if ("1".equals(getNetType(paramContext)[0]) || "2".equals(getNetType(paramContext)[0])) {
43 | return true;
44 | }
45 | return false;
46 | }
47 |
48 | /**
49 | * 获取当前网络状态 返回2代表wifi,1代表2G/3G
50 | *
51 | * @param paramContext
52 | * @return
53 | */
54 | public static String[] getNetType(Context paramContext) {
55 | String[] arrayOfString = {"Unknown", "Unknown"};
56 | PackageManager localPackageManager = paramContext.getPackageManager();
57 | if (localPackageManager.checkPermission("android.permission.ACCESS_NETWORK_STATE", paramContext.getPackageName()) != 0) {
58 | arrayOfString[0] = "Unknown";
59 | return arrayOfString;
60 | }
61 |
62 | ConnectivityManager localConnectivityManager = (ConnectivityManager) paramContext.getSystemService("connectivity");
63 | if (localConnectivityManager == null) {
64 | arrayOfString[0] = "Unknown";
65 | return arrayOfString;
66 | }
67 |
68 | NetworkInfo localNetworkInfo1 = localConnectivityManager.getNetworkInfo(1);
69 | if (localNetworkInfo1 != null && localNetworkInfo1.getState() == NetworkInfo.State.CONNECTED) {
70 | arrayOfString[0] = "2";
71 | return arrayOfString;
72 | }
73 |
74 | NetworkInfo localNetworkInfo2 = localConnectivityManager.getNetworkInfo(0);
75 | if (localNetworkInfo2 != null && localNetworkInfo2.getState() == NetworkInfo.State.CONNECTED) {
76 | arrayOfString[0] = "1";
77 | arrayOfString[1] = localNetworkInfo2.getSubtypeName();
78 | return arrayOfString;
79 | }
80 |
81 | return arrayOfString;
82 | }
83 | }
--------------------------------------------------------------------------------
/sample/src/main/java/com/cundong/recyclerview/sample/MainActivity.java:
--------------------------------------------------------------------------------
1 | package com.cundong.recyclerview.sample;
2 |
3 | import android.content.Context;
4 | import android.content.Intent;
5 | import android.os.Bundle;
6 | import android.support.v7.app.AppCompatActivity;
7 | import android.support.v7.widget.LinearLayoutManager;
8 | import android.support.v7.widget.RecyclerView;
9 | import android.view.LayoutInflater;
10 | import android.view.View;
11 | import android.view.ViewGroup;
12 | import android.widget.TextView;
13 |
14 | import com.cundong.recyclerview.RecyclerViewUtils;
15 |
16 | import java.util.ArrayList;
17 |
18 | /**
19 | * Created by cundong on 2015/11/10.
20 | *
21 | * Sample入口
22 | */
23 | public class MainActivity extends AppCompatActivity {
24 |
25 | private static final Class>[] ACTIVITY = {LinearLayoutActivity.class, EndlessLinearLayoutActivity.class, EndlessGridLayoutActivity.class, EndlessStaggeredGridLayoutActivity.class};
26 | private static final String[] TITLE = {"LinearLayoutSample", "EndlessLinearLayoutActivity", "EndlessGridLayoutActivity", "EndlessStaggeredGridLayoutActivity"};
27 |
28 | private RecyclerView mRecyclerView = null;
29 |
30 | private DataAdapter mDataAdapter = null;
31 | private ArrayList mDataList = null;
32 |
33 | @Override
34 | protected void onCreate(Bundle savedInstanceState) {
35 | super.onCreate(savedInstanceState);
36 | setContentView(R.layout.sample_activity);
37 |
38 | mRecyclerView = (RecyclerView) findViewById(R.id.list);
39 | mRecyclerView.setLayoutManager(new LinearLayoutManager(this));
40 |
41 | mDataList = new ArrayList<>();
42 | for (int i = 0; i < TITLE.length; i++) {
43 |
44 | ListItem item = new ListItem();
45 | item.title = TITLE[i];
46 | item.activity = ACTIVITY[i];
47 | mDataList.add(item);
48 | }
49 |
50 | mDataAdapter = new DataAdapter(this);
51 | mDataAdapter.setData(mDataList);
52 | mRecyclerView.setAdapter(mDataAdapter);
53 | }
54 |
55 | private static class ListItem {
56 | public String title;
57 | public Class> activity;
58 | }
59 |
60 | private class DataAdapter extends RecyclerView.Adapter {
61 |
62 | private LayoutInflater mLayoutInflater;
63 | private ArrayList mDataList = new ArrayList<>();
64 |
65 | public DataAdapter(Context context) {
66 | mLayoutInflater = LayoutInflater.from(context);
67 | }
68 |
69 | public void setData(ArrayList list) {
70 | this.mDataList = list;
71 | notifyDataSetChanged();
72 | }
73 |
74 | @Override
75 | public RecyclerView.ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
76 | return new ViewHolder(mLayoutInflater.inflate(R.layout.sample_item_text, parent, false));
77 | }
78 |
79 | @Override
80 | public void onBindViewHolder(RecyclerView.ViewHolder holder, int position) {
81 |
82 | ListItem listItem = mDataList.get(position);
83 |
84 | ViewHolder viewHolder = (ViewHolder) holder;
85 | viewHolder.textView.setText(listItem.title);
86 | }
87 |
88 | @Override
89 | public int getItemCount() {
90 | return mDataList.size();
91 | }
92 |
93 | private class ViewHolder extends RecyclerView.ViewHolder {
94 |
95 | private TextView textView;
96 |
97 | public ViewHolder(View itemView) {
98 | super(itemView);
99 | textView = (TextView) itemView.findViewById(R.id.info_text);
100 |
101 | textView.setOnClickListener(new View.OnClickListener() {
102 | @Override
103 | public void onClick(View v) {
104 |
105 | ListItem listItem = mDataList.get(RecyclerViewUtils.getAdapterPosition(mRecyclerView, ViewHolder.this));
106 | startActivity(new Intent(MainActivity.this, listItem.activity));
107 | }
108 | });
109 | }
110 | }
111 | }
112 | }
--------------------------------------------------------------------------------
/library/src/main/java/com/cundong/recyclerview/EndlessRecyclerOnScrollListener.java:
--------------------------------------------------------------------------------
1 | package com.cundong.recyclerview;
2 |
3 | import android.support.v7.widget.GridLayoutManager;
4 | import android.support.v7.widget.LinearLayoutManager;
5 | import android.support.v7.widget.RecyclerView;
6 | import android.support.v7.widget.StaggeredGridLayoutManager;
7 | import android.view.View;
8 |
9 | /**
10 | * Created by cundong on 2015/10/9.
11 | *
12 | * 继承自RecyclerView.OnScrollListener,可以监听到是否滑动到页面最低部
13 | */
14 | public class EndlessRecyclerOnScrollListener extends RecyclerView.OnScrollListener implements OnListLoadNextPageListener {
15 |
16 | /**
17 | * 当前RecyclerView类型
18 | */
19 | protected LayoutManagerType layoutManagerType;
20 |
21 | /**
22 | * 最后一个的位置
23 | */
24 | private int[] lastPositions;
25 |
26 | /**
27 | * 最后一个可见的item的位置
28 | */
29 | private int lastVisibleItemPosition;
30 |
31 | /**
32 | * 当前滑动的状态
33 | */
34 | private int currentScrollState = 0;
35 |
36 | @Override
37 | public void onScrolled(RecyclerView recyclerView, int dx, int dy) {
38 | super.onScrolled(recyclerView, dx, dy);
39 |
40 | RecyclerView.LayoutManager layoutManager = recyclerView.getLayoutManager();
41 |
42 | if (layoutManagerType == null) {
43 | if (layoutManager instanceof LinearLayoutManager) {
44 | layoutManagerType = LayoutManagerType.LinearLayout;
45 | } else if (layoutManager instanceof GridLayoutManager) {
46 | layoutManagerType = LayoutManagerType.GridLayout;
47 | } else if (layoutManager instanceof StaggeredGridLayoutManager) {
48 | layoutManagerType = LayoutManagerType.StaggeredGridLayout;
49 | } else {
50 | throw new RuntimeException(
51 | "Unsupported LayoutManager used. Valid ones are LinearLayoutManager, GridLayoutManager and StaggeredGridLayoutManager");
52 | }
53 | }
54 |
55 | switch (layoutManagerType) {
56 | case LinearLayout:
57 | lastVisibleItemPosition = ((LinearLayoutManager) layoutManager).findLastVisibleItemPosition();
58 | break;
59 | case GridLayout:
60 | lastVisibleItemPosition = ((GridLayoutManager) layoutManager).findLastVisibleItemPosition();
61 | break;
62 | case StaggeredGridLayout:
63 | StaggeredGridLayoutManager staggeredGridLayoutManager = (StaggeredGridLayoutManager) layoutManager;
64 | if (lastPositions == null) {
65 | lastPositions = new int[staggeredGridLayoutManager.getSpanCount()];
66 | }
67 | staggeredGridLayoutManager.findLastVisibleItemPositions(lastPositions);
68 | lastVisibleItemPosition = findMax(lastPositions);
69 | break;
70 | }
71 | }
72 |
73 | @Override
74 | public void onScrollStateChanged(RecyclerView recyclerView, int newState) {
75 | super.onScrollStateChanged(recyclerView, newState);
76 | currentScrollState = newState;
77 | RecyclerView.LayoutManager layoutManager = recyclerView.getLayoutManager();
78 | int visibleItemCount = layoutManager.getChildCount();
79 | int totalItemCount = layoutManager.getItemCount();
80 | if ((visibleItemCount > 0 && currentScrollState == RecyclerView.SCROLL_STATE_IDLE && (lastVisibleItemPosition) >= totalItemCount - 1)) {
81 | onLoadNextPage(recyclerView);
82 | }
83 | }
84 |
85 | /**
86 | * 取数组中最大值
87 | *
88 | * @param lastPositions
89 | * @return
90 | */
91 | private int findMax(int[] lastPositions) {
92 | int max = lastPositions[0];
93 | for (int value : lastPositions) {
94 | if (value > max) {
95 | max = value;
96 | }
97 | }
98 |
99 | return max;
100 | }
101 |
102 | @Override
103 | public void onLoadNextPage(final View view) {
104 | }
105 |
106 | public static enum LayoutManagerType {
107 | LinearLayout,
108 | StaggeredGridLayout,
109 | GridLayout
110 | }
111 | }
112 |
--------------------------------------------------------------------------------
/sample/src/main/java/com/cundong/recyclerview/sample/LinearLayoutActivity.java:
--------------------------------------------------------------------------------
1 | package com.cundong.recyclerview.sample;
2 |
3 | import android.content.Context;
4 | import android.os.Bundle;
5 | import android.support.v7.app.AppCompatActivity;
6 | import android.support.v7.widget.LinearLayoutManager;
7 | import android.support.v7.widget.RecyclerView;
8 | import android.view.LayoutInflater;
9 | import android.view.View;
10 | import android.view.ViewGroup;
11 | import android.widget.TextView;
12 | import android.widget.Toast;
13 |
14 | import com.cundong.recyclerview.HeaderAndFooterRecyclerViewAdapter;
15 | import com.cundong.recyclerview.RecyclerViewUtils;
16 | import com.cundong.recyclerview.sample.weight.SampleFooter;
17 | import com.cundong.recyclerview.sample.weight.SampleHeader;
18 |
19 | import java.util.ArrayList;
20 |
21 | /**
22 | * Created by cundong on 2015/10/29.
23 | *
24 | * 带HeaderView、FooterView的LinearLayout RecyclerView
25 | */
26 | public class LinearLayoutActivity extends AppCompatActivity {
27 |
28 | private RecyclerView mRecyclerView = null;
29 |
30 | private DataAdapter mDataAdapter = null;
31 |
32 | private HeaderAndFooterRecyclerViewAdapter mHeaderAndFooterRecyclerViewAdapter = null;
33 |
34 | @Override
35 | public void onCreate(Bundle savedInstanceState) {
36 | super.onCreate(savedInstanceState);
37 | setContentView(R.layout.sample_activity);
38 |
39 | mRecyclerView = (RecyclerView) findViewById(R.id.list);
40 |
41 | //init data
42 | ArrayList dataList = new ArrayList<>();
43 | for (int i = 0; i < 5; i++) {
44 | dataList.add("item" + i);
45 | }
46 |
47 | mDataAdapter = new DataAdapter(this);
48 | mDataAdapter.setData(dataList);
49 |
50 | mHeaderAndFooterRecyclerViewAdapter = new HeaderAndFooterRecyclerViewAdapter(mDataAdapter);
51 | mRecyclerView.setAdapter(mHeaderAndFooterRecyclerViewAdapter);
52 |
53 | mRecyclerView.setLayoutManager(new LinearLayoutManager(this));
54 |
55 | //add a HeaderView
56 | RecyclerViewUtils.setHeaderView(mRecyclerView, new SampleHeader(this));
57 |
58 | //add a FooterView
59 | RecyclerViewUtils.setFooterView(mRecyclerView, new SampleFooter(this));
60 | }
61 |
62 | private class DataAdapter extends RecyclerView.Adapter {
63 |
64 | private LayoutInflater mLayoutInflater;
65 | private ArrayList mDataList = new ArrayList<>();
66 |
67 | public DataAdapter(Context context) {
68 | mLayoutInflater = LayoutInflater.from(context);
69 | }
70 |
71 | public void setData(ArrayList list) {
72 | this.mDataList = list;
73 | notifyDataSetChanged();
74 | }
75 |
76 | @Override
77 | public RecyclerView.ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
78 | return new ViewHolder(mLayoutInflater.inflate(R.layout.sample_item_text, parent, false));
79 | }
80 |
81 | @Override
82 | public void onBindViewHolder(RecyclerView.ViewHolder holder, int position) {
83 |
84 | String item = mDataList.get(position);
85 |
86 | ViewHolder viewHolder = (ViewHolder) holder;
87 | viewHolder.textView.setText(item);
88 | }
89 |
90 | @Override
91 | public int getItemCount() {
92 | return mDataList.size();
93 | }
94 |
95 | private class ViewHolder extends RecyclerView.ViewHolder {
96 |
97 | private TextView textView;
98 |
99 | public ViewHolder(View itemView) {
100 | super(itemView);
101 | textView = (TextView) itemView.findViewById(R.id.info_text);
102 |
103 | textView.setOnClickListener(new View.OnClickListener() {
104 | @Override
105 | public void onClick(View v) {
106 | String text = mDataList.get(RecyclerViewUtils.getAdapterPosition(mRecyclerView, ViewHolder.this));
107 | Toast.makeText(LinearLayoutActivity.this, text, Toast.LENGTH_SHORT).show();
108 | }
109 | });
110 | }
111 | }
112 | }
113 | }
--------------------------------------------------------------------------------
/sample/src/main/java/com/cundong/recyclerview/sample/utils/RecyclerViewStateUtils.java:
--------------------------------------------------------------------------------
1 | package com.cundong.recyclerview.sample.utils;
2 |
3 | import android.app.Activity;
4 | import android.support.v7.widget.RecyclerView;
5 | import android.view.View;
6 |
7 | import com.cundong.recyclerview.HeaderAndFooterRecyclerViewAdapter;
8 | import com.cundong.recyclerview.sample.weight.LoadingFooter;
9 |
10 | /**
11 | * Created by cundong on 2015/11/9.
12 | *
13 | * 分页展示数据时,RecyclerView的FooterView State 操作工具类
14 | *
15 | * RecyclerView一共有几种State:Normal/Loading/Error/TheEnd
16 | */
17 | public class RecyclerViewStateUtils {
18 |
19 | /**
20 | * 设置headerAndFooterAdapter的FooterView State
21 | *
22 | * @param instance context
23 | * @param recyclerView recyclerView
24 | * @param pageSize 分页展示时,recyclerView每一页的数量
25 | * @param state FooterView State
26 | * @param errorListener FooterView处于Error状态时的点击事件
27 | */
28 | public static void setFooterViewState(Activity instance, RecyclerView recyclerView, int pageSize, LoadingFooter.State state, View.OnClickListener errorListener) {
29 |
30 | if(instance==null || instance.isFinishing()) {
31 | return;
32 | }
33 |
34 | RecyclerView.Adapter outerAdapter = recyclerView.getAdapter();
35 |
36 | if (outerAdapter == null || !(outerAdapter instanceof HeaderAndFooterRecyclerViewAdapter)) {
37 | return;
38 | }
39 |
40 | HeaderAndFooterRecyclerViewAdapter headerAndFooterAdapter = (HeaderAndFooterRecyclerViewAdapter) outerAdapter;
41 |
42 | //只有一页的时候,就别加什么FooterView了
43 | if (headerAndFooterAdapter.getInnerAdapter().getItemCount() < pageSize) {
44 | return;
45 | }
46 |
47 | LoadingFooter footerView;
48 |
49 | //已经有footerView了
50 | if (headerAndFooterAdapter.getFooterViewsCount() > 0) {
51 | footerView = (LoadingFooter) headerAndFooterAdapter.getFooterView();
52 | footerView.setState(state);
53 |
54 | if (state == LoadingFooter.State.NetWorkError) {
55 | footerView.setOnClickListener(errorListener);
56 | }
57 | recyclerView.scrollToPosition(headerAndFooterAdapter.getItemCount() - 1);
58 | } else {
59 | footerView = new LoadingFooter(instance);
60 | footerView.setState(state);
61 |
62 | if (state == LoadingFooter.State.NetWorkError) {
63 | footerView.setOnClickListener(errorListener);
64 | }
65 |
66 | headerAndFooterAdapter.addFooterView(footerView);
67 | recyclerView.scrollToPosition(headerAndFooterAdapter.getItemCount() - 1);
68 | }
69 | }
70 |
71 | /**
72 | * 获取当前RecyclerView.FooterView的状态
73 | *
74 | * @param recyclerView
75 | */
76 | public static LoadingFooter.State getFooterViewState(RecyclerView recyclerView) {
77 |
78 | RecyclerView.Adapter outerAdapter = recyclerView.getAdapter();
79 | if (outerAdapter != null && outerAdapter instanceof HeaderAndFooterRecyclerViewAdapter) {
80 | if (((HeaderAndFooterRecyclerViewAdapter) outerAdapter).getFooterViewsCount() > 0) {
81 | LoadingFooter footerView = (LoadingFooter) ((HeaderAndFooterRecyclerViewAdapter) outerAdapter).getFooterView();
82 | return footerView.getState();
83 | }
84 | }
85 |
86 | return LoadingFooter.State.Normal;
87 | }
88 |
89 | /**
90 | * 设置当前RecyclerView.FooterView的状态
91 | *
92 | * @param recyclerView
93 | * @param state
94 | */
95 | public static void setFooterViewState(RecyclerView recyclerView, LoadingFooter.State state) {
96 | RecyclerView.Adapter outerAdapter = recyclerView.getAdapter();
97 | if (outerAdapter != null && outerAdapter instanceof HeaderAndFooterRecyclerViewAdapter) {
98 | if (((HeaderAndFooterRecyclerViewAdapter) outerAdapter).getFooterViewsCount() > 0) {
99 | LoadingFooter footerView = (LoadingFooter) ((HeaderAndFooterRecyclerViewAdapter) outerAdapter).getFooterView();
100 | footerView.setState(state);
101 | }
102 | }
103 | }
104 | }
105 |
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | ##HeaderAndFooterRecyclerView
2 |
3 | ------
4 |
5 | ##介绍
6 |
7 | HeaderAndFooterRecyclerView是支持addHeaderView、 addFooterView、分页加载的RecyclerView解决方案。
8 |
9 | 它可以对 RecyclerView 控件进行拓展(通过RecyclerView.Adapter实现),给RecyclerView增加HeaderView、FooterView,并且**不需要**对你的具体业务逻辑Adapter做任何修改。
10 |
11 | 同时,通过修改 FooterView State,可以动态 FooterView 赋予不同状态(加载中、加载失败、滑到最底等),可以实现 RecyclerView 分页加载数据时的 Loading/TheEnd/NetWorkError 效果。
12 |
13 | ##使用
14 |
15 | * 添加HeaderView、FooterView
16 | ```java
17 | mHeaderAndFooterRecyclerViewAdapter = new HeaderAndFooterRecyclerViewAdapter(mDataAdapter);
18 | mRecyclerView.setAdapter(mHeaderAndFooterRecyclerViewAdapter);
19 |
20 | mRecyclerView.setLayoutManager(new LinearLayoutManager(this));
21 |
22 | //add a HeaderView
23 | RecyclerViewUtils.setHeaderView(mRecyclerView, new SampleHeader(this));
24 |
25 | //add a FooterView
26 | RecyclerViewUtils.setFooterView(mRecyclerView, new SampleFooter(this));
27 | ```
28 |
29 | * LinearLayout/GridLayout/StaggeredGridLayout布局的RecyclerView分页加载
30 |
31 | ```java
32 | mRecyclerView.addOnScrollListener(mOnScrollListener);
33 | ```
34 |
35 | ```java
36 | private EndlessRecyclerOnScrollListener mOnScrollListener = new EndlessRecyclerOnScrollListener() {
37 |
38 | @Override
39 | public void onLoadNextPage(View view) {
40 | super.onLoadNextPage(view);
41 |
42 | LoadingFooter.State state = RecyclerViewStateUtils.getFooterViewState(mRecyclerView);
43 | if(state == LoadingFooter.State.Loading) {
44 | Log.d("@Cundong", "the state is Loading, just wait..");
45 | return;
46 | }
47 |
48 | mCurrentCounter = mDataList.size();
49 |
50 | if (mCurrentCounter < TOTAL_COUNTER) {
51 | // loading more
52 | RecyclerViewStateUtils.setFooterViewState(EndlessLinearLayoutActivity.this, mRecyclerView, REQUEST_COUNT, LoadingFooter.State.Loading, null);
53 | requestData();
54 | } else {
55 | //the end
56 | RecyclerViewStateUtils.setFooterViewState(EndlessLinearLayoutActivity.this, mRecyclerView, REQUEST_COUNT, LoadingFooter.State.TheEnd, null);
57 | }
58 | }
59 | };
60 | ```
61 | ## 注意事项
62 |
63 | 如果已经使用 ```RecyclerViewUtils.setHeaderView(mRecyclerView, view);``` 为RecyclerView添加了HeaderView,那么再调用ViewHolder类的```getAdapterPosition()```、```getLayoutPosition()```时返回的值就会因为增加了Header而受影响(返回的position等于真实的position+headerCounter)。
64 |
65 | 因此,这种情况下请使用
66 | ```RecyclerViewUtils.getAdapterPosition(mRecyclerView, ViewHolder.this)```、```RecyclerViewUtils.getLayoutPosition(mRecyclerView, ViewHolder.this)``` 两个方法来替代。
67 |
68 | ## Demo
69 |
70 | * 添加HeaderView、FooterView
71 |
72 | ![截屏][1]
73 |
74 | * 支持分页加载的LinearLayout布局RecyclerView
75 |
76 | ![截屏][2]
77 |
78 | * 支持分页加载的GridLayout布局RecyclerView
79 |
80 | ![截屏][3]
81 |
82 | * 支持分页加载的StaggeredGridLayout布局RecyclerView
83 |
84 | ![截屏][4]
85 |
86 | * 分页加载失败时的GridLayout布局RecyclerView
87 |
88 | ![截屏][5]
89 |
90 | ## 关于我
91 |
92 | * Blog: [http://my.oschina.net/liucundong/blog][6]
93 | * Mail: cundong.liu#gmail.com
94 |
95 | ## License
96 |
97 | Copyright 2015 Cundong
98 |
99 | Licensed under the Apache License, Version 2.0 (the "License");
100 | you may not use this file except in compliance with the License.
101 | You may obtain a copy of the License at
102 |
103 | http://www.apache.org/licenses/LICENSE-2.0
104 |
105 | Unless required by applicable law or agreed to in writing, software
106 | distributed under the License is distributed on an "AS IS" BASIS,
107 | WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
108 | See the License for the specific language governing permissions and
109 | limitations under the License.
110 |
111 | [1]: https://raw.githubusercontent.com/cundong/HeaderAndFooterRecyclerView/master/art/art1.png
112 | [2]: https://raw.githubusercontent.com/cundong/HeaderAndFooterRecyclerView/master/art/art2.png
113 | [3]: https://raw.githubusercontent.com/cundong/HeaderAndFooterRecyclerView/master/art/art3.png
114 | [4]: https://raw.githubusercontent.com/cundong/HeaderAndFooterRecyclerView/master/art/art4.png
115 | [5]: https://raw.githubusercontent.com/cundong/HeaderAndFooterRecyclerView/master/art/art5.png
116 | [6]: http://my.oschina.net/liucundong/blog
117 |
--------------------------------------------------------------------------------
/library/src/main/java/com/cundong/recyclerview/RecyclerViewUtils.java:
--------------------------------------------------------------------------------
1 | package com.cundong.recyclerview;
2 |
3 | import android.support.v7.widget.RecyclerView;
4 | import android.view.View;
5 |
6 | /**
7 | * Created by cundong on 2015/10/22.
8 | *
9 | * RecyclerView设置Header/Footer所用到的工具类
10 | */
11 | public class RecyclerViewUtils {
12 |
13 | /**
14 | * 设置HeaderView
15 | *
16 | * @param recyclerView
17 | * @param view
18 | */
19 | public static void setHeaderView(RecyclerView recyclerView, View view) {
20 | RecyclerView.Adapter outerAdapter = recyclerView.getAdapter();
21 |
22 | if (outerAdapter == null || !(outerAdapter instanceof HeaderAndFooterRecyclerViewAdapter)) {
23 | return;
24 | }
25 |
26 | HeaderAndFooterRecyclerViewAdapter headerAndFooterAdapter = (HeaderAndFooterRecyclerViewAdapter) outerAdapter;
27 | if (headerAndFooterAdapter.getHeaderViewsCount() == 0) {
28 | headerAndFooterAdapter.addHeaderView(view);
29 | }
30 | }
31 |
32 | /**
33 | * 设置FooterView
34 | *
35 | * @param recyclerView
36 | * @param view
37 | */
38 | public static void setFooterView(RecyclerView recyclerView, View view) {
39 | RecyclerView.Adapter outerAdapter = recyclerView.getAdapter();
40 |
41 | if (outerAdapter == null || !(outerAdapter instanceof HeaderAndFooterRecyclerViewAdapter)) {
42 | return;
43 | }
44 |
45 | HeaderAndFooterRecyclerViewAdapter headerAndFooterAdapter = (HeaderAndFooterRecyclerViewAdapter) outerAdapter;
46 | if (headerAndFooterAdapter.getFooterViewsCount() == 0) {
47 | headerAndFooterAdapter.addFooterView(view);
48 | }
49 | }
50 |
51 | /**
52 | * 移除FooterView
53 | *
54 | * @param recyclerView
55 | */
56 | public static void removeFooterView(RecyclerView recyclerView) {
57 |
58 | RecyclerView.Adapter outerAdapter = recyclerView.getAdapter();
59 |
60 | if (outerAdapter != null && outerAdapter instanceof HeaderAndFooterRecyclerViewAdapter) {
61 |
62 | int footerViewCounter = ((HeaderAndFooterRecyclerViewAdapter) outerAdapter).getFooterViewsCount();
63 | if (footerViewCounter > 0) {
64 | View footerView = ((HeaderAndFooterRecyclerViewAdapter) outerAdapter).getFooterView();
65 | ((HeaderAndFooterRecyclerViewAdapter) outerAdapter).removeFooterView(footerView);
66 | }
67 | }
68 | }
69 |
70 | /**
71 | * 移除HeaderView
72 | *
73 | * @param recyclerView
74 | */
75 | public static void removeHeaderView(RecyclerView recyclerView) {
76 |
77 | RecyclerView.Adapter outerAdapter = recyclerView.getAdapter();
78 |
79 | if (outerAdapter != null && outerAdapter instanceof HeaderAndFooterRecyclerViewAdapter) {
80 |
81 | int headerViewCounter = ((HeaderAndFooterRecyclerViewAdapter) outerAdapter).getHeaderViewsCount();
82 | if (headerViewCounter > 0) {
83 | View headerView = ((HeaderAndFooterRecyclerViewAdapter) outerAdapter).getHeaderView();
84 | ((HeaderAndFooterRecyclerViewAdapter) outerAdapter).removeFooterView(headerView);
85 | }
86 | }
87 | }
88 |
89 | /**
90 | * 请使用本方法替代RecyclerView.ViewHolder的getLayoutPosition()方法
91 | *
92 | * @param recyclerView
93 | * @param holder
94 | * @return
95 | */
96 | public static int getLayoutPosition(RecyclerView recyclerView, RecyclerView.ViewHolder holder) {
97 | RecyclerView.Adapter outerAdapter = recyclerView.getAdapter();
98 | if (outerAdapter != null && outerAdapter instanceof HeaderAndFooterRecyclerViewAdapter) {
99 |
100 | int headerViewCounter = ((HeaderAndFooterRecyclerViewAdapter) outerAdapter).getHeaderViewsCount();
101 | if (headerViewCounter > 0) {
102 | return holder.getLayoutPosition() - headerViewCounter;
103 | }
104 | }
105 |
106 | return holder.getLayoutPosition();
107 | }
108 |
109 | /**
110 | * 请使用本方法替代RecyclerView.ViewHolder的getAdapterPosition()方法
111 | *
112 | * @param recyclerView
113 | * @param holder
114 | * @return
115 | */
116 | public static int getAdapterPosition(RecyclerView recyclerView, RecyclerView.ViewHolder holder) {
117 | RecyclerView.Adapter outerAdapter = recyclerView.getAdapter();
118 | if (outerAdapter != null && outerAdapter instanceof HeaderAndFooterRecyclerViewAdapter) {
119 |
120 | int headerViewCounter = ((HeaderAndFooterRecyclerViewAdapter) outerAdapter).getHeaderViewsCount();
121 | if (headerViewCounter > 0) {
122 | return holder.getAdapterPosition() - headerViewCounter;
123 | }
124 | }
125 |
126 | return holder.getAdapterPosition();
127 | }
128 | }
--------------------------------------------------------------------------------
/sample/src/main/java/com/cundong/recyclerview/sample/weight/LoadingFooter.java:
--------------------------------------------------------------------------------
1 | package com.cundong.recyclerview.sample.weight;
2 |
3 | import android.content.Context;
4 | import android.util.AttributeSet;
5 | import android.view.View;
6 | import android.view.ViewStub;
7 | import android.widget.ProgressBar;
8 | import android.widget.RelativeLayout;
9 | import android.widget.TextView;
10 |
11 | import com.cundong.recyclerview.sample.R;
12 |
13 | /**
14 | * Created by cundong on 2015/10/9.
15 | *
16 | * ListView/GridView/RecyclerView 分页加载时使用到的FooterView
17 | */
18 | public class LoadingFooter extends RelativeLayout {
19 |
20 | protected State mState = State.Normal;
21 | private View mLoadingView;
22 | private View mNetworkErrorView;
23 | private View mTheEndView;
24 | private ProgressBar mLoadingProgress;
25 | private TextView mLoadingText;
26 |
27 | public LoadingFooter(Context context) {
28 | super(context);
29 | init(context);
30 | }
31 |
32 | public LoadingFooter(Context context, AttributeSet attrs) {
33 | super(context, attrs);
34 | init(context);
35 | }
36 |
37 | public LoadingFooter(Context context, AttributeSet attrs, int defStyleAttr) {
38 | super(context, attrs, defStyleAttr);
39 | init(context);
40 | }
41 |
42 | public void init(Context context) {
43 |
44 | inflate(context, R.layout.sample_common_list_footer, this);
45 | setOnClickListener(null);
46 |
47 | setState(State.Normal, true);
48 | }
49 |
50 | public State getState() {
51 | return mState;
52 | }
53 |
54 | public void setState(State status ) {
55 | setState(status, true);
56 | }
57 |
58 | /**
59 | * 设置状态
60 | *
61 | * @param status
62 | * @param showView 是否展示当前View
63 | */
64 | public void setState(State status, boolean showView) {
65 | if (mState == status) {
66 | return;
67 | }
68 | mState = status;
69 |
70 | switch (status) {
71 |
72 | case Normal:
73 | setOnClickListener(null);
74 | if (mLoadingView != null) {
75 | mLoadingView.setVisibility(GONE);
76 | }
77 |
78 | if (mTheEndView != null) {
79 | mTheEndView.setVisibility(GONE);
80 | }
81 |
82 | if (mNetworkErrorView != null) {
83 | mNetworkErrorView.setVisibility(GONE);
84 | }
85 |
86 | break;
87 | case Loading:
88 | setOnClickListener(null);
89 | if (mTheEndView != null) {
90 | mTheEndView.setVisibility(GONE);
91 | }
92 |
93 | if (mNetworkErrorView != null) {
94 | mNetworkErrorView.setVisibility(GONE);
95 | }
96 |
97 | if (mLoadingView == null) {
98 | ViewStub viewStub = (ViewStub) findViewById(R.id.loading_viewstub);
99 | mLoadingView = viewStub.inflate();
100 |
101 | mLoadingProgress = (ProgressBar) mLoadingView.findViewById(R.id.loading_progress);
102 | mLoadingText = (TextView) mLoadingView.findViewById(R.id.loading_text);
103 | } else {
104 | mLoadingView.setVisibility(VISIBLE);
105 | }
106 |
107 | mLoadingView.setVisibility(showView ? VISIBLE : GONE);
108 |
109 | mLoadingProgress.setVisibility(View.VISIBLE);
110 | mLoadingText.setText(R.string.list_footer_loading);
111 | break;
112 | case TheEnd:
113 | setOnClickListener(null);
114 | if (mLoadingView != null) {
115 | mLoadingView.setVisibility(GONE);
116 | }
117 |
118 | if (mNetworkErrorView != null) {
119 | mNetworkErrorView.setVisibility(GONE);
120 | }
121 |
122 | if (mTheEndView == null) {
123 | ViewStub viewStub = (ViewStub) findViewById(R.id.end_viewstub);
124 | mTheEndView = viewStub.inflate();
125 | } else {
126 | mTheEndView.setVisibility(VISIBLE);
127 | }
128 |
129 | mTheEndView.setVisibility(showView ? VISIBLE : GONE);
130 | break;
131 | case NetWorkError:
132 |
133 | if (mLoadingView != null) {
134 | mLoadingView.setVisibility(GONE);
135 | }
136 |
137 | if (mTheEndView != null) {
138 | mTheEndView.setVisibility(GONE);
139 | }
140 |
141 | if (mNetworkErrorView == null) {
142 | ViewStub viewStub = (ViewStub) findViewById(R.id.network_error_viewstub);
143 | mNetworkErrorView = viewStub.inflate();
144 | } else {
145 | mNetworkErrorView.setVisibility(VISIBLE);
146 | }
147 |
148 | mNetworkErrorView.setVisibility(showView ? VISIBLE : GONE);
149 | break;
150 | default:
151 |
152 | break;
153 | }
154 | }
155 |
156 | public static enum State {
157 | Normal/**正常*/, TheEnd/**加载到最底了*/, Loading/**加载中..*/, NetWorkError/**网络异常*/
158 | }
159 | }
--------------------------------------------------------------------------------
/gradlew:
--------------------------------------------------------------------------------
1 | #!/usr/bin/env bash
2 |
3 | ##############################################################################
4 | ##
5 | ## Gradle start up script for UN*X
6 | ##
7 | ##############################################################################
8 |
9 | # Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
10 | DEFAULT_JVM_OPTS=""
11 |
12 | APP_NAME="Gradle"
13 | APP_BASE_NAME=`basename "$0"`
14 |
15 | # Use the maximum available, or set MAX_FD != -1 to use that value.
16 | MAX_FD="maximum"
17 |
18 | warn ( ) {
19 | echo "$*"
20 | }
21 |
22 | die ( ) {
23 | echo
24 | echo "$*"
25 | echo
26 | exit 1
27 | }
28 |
29 | # OS specific support (must be 'true' or 'false').
30 | cygwin=false
31 | msys=false
32 | darwin=false
33 | case "`uname`" in
34 | CYGWIN* )
35 | cygwin=true
36 | ;;
37 | Darwin* )
38 | darwin=true
39 | ;;
40 | MINGW* )
41 | msys=true
42 | ;;
43 | esac
44 |
45 | # For Cygwin, ensure paths are in UNIX format before anything is touched.
46 | if $cygwin ; then
47 | [ -n "$JAVA_HOME" ] && JAVA_HOME=`cygpath --unix "$JAVA_HOME"`
48 | fi
49 |
50 | # Attempt to set APP_HOME
51 | # Resolve links: $0 may be a link
52 | PRG="$0"
53 | # Need this for relative symlinks.
54 | while [ -h "$PRG" ] ; do
55 | ls=`ls -ld "$PRG"`
56 | link=`expr "$ls" : '.*-> \(.*\)$'`
57 | if expr "$link" : '/.*' > /dev/null; then
58 | PRG="$link"
59 | else
60 | PRG=`dirname "$PRG"`"/$link"
61 | fi
62 | done
63 | SAVED="`pwd`"
64 | cd "`dirname \"$PRG\"`/" >&-
65 | APP_HOME="`pwd -P`"
66 | cd "$SAVED" >&-
67 |
68 | CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar
69 |
70 | # Determine the Java command to use to start the JVM.
71 | if [ -n "$JAVA_HOME" ] ; then
72 | if [ -x "$JAVA_HOME/jre/sh/java" ] ; then
73 | # IBM's JDK on AIX uses strange locations for the executables
74 | JAVACMD="$JAVA_HOME/jre/sh/java"
75 | else
76 | JAVACMD="$JAVA_HOME/bin/java"
77 | fi
78 | if [ ! -x "$JAVACMD" ] ; then
79 | die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME
80 |
81 | Please set the JAVA_HOME variable in your environment to match the
82 | location of your Java installation."
83 | fi
84 | else
85 | JAVACMD="java"
86 | which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
87 |
88 | Please set the JAVA_HOME variable in your environment to match the
89 | location of your Java installation."
90 | fi
91 |
92 | # Increase the maximum file descriptors if we can.
93 | if [ "$cygwin" = "false" -a "$darwin" = "false" ] ; then
94 | MAX_FD_LIMIT=`ulimit -H -n`
95 | if [ $? -eq 0 ] ; then
96 | if [ "$MAX_FD" = "maximum" -o "$MAX_FD" = "max" ] ; then
97 | MAX_FD="$MAX_FD_LIMIT"
98 | fi
99 | ulimit -n $MAX_FD
100 | if [ $? -ne 0 ] ; then
101 | warn "Could not set maximum file descriptor limit: $MAX_FD"
102 | fi
103 | else
104 | warn "Could not query maximum file descriptor limit: $MAX_FD_LIMIT"
105 | fi
106 | fi
107 |
108 | # For Darwin, add options to specify how the application appears in the dock
109 | if $darwin; then
110 | GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\""
111 | fi
112 |
113 | # For Cygwin, switch paths to Windows format before running java
114 | if $cygwin ; then
115 | APP_HOME=`cygpath --path --mixed "$APP_HOME"`
116 | CLASSPATH=`cygpath --path --mixed "$CLASSPATH"`
117 |
118 | # We build the pattern for arguments to be converted via cygpath
119 | ROOTDIRSRAW=`find -L / -maxdepth 1 -mindepth 1 -type d 2>/dev/null`
120 | SEP=""
121 | for dir in $ROOTDIRSRAW ; do
122 | ROOTDIRS="$ROOTDIRS$SEP$dir"
123 | SEP="|"
124 | done
125 | OURCYGPATTERN="(^($ROOTDIRS))"
126 | # Add a user-defined pattern to the cygpath arguments
127 | if [ "$GRADLE_CYGPATTERN" != "" ] ; then
128 | OURCYGPATTERN="$OURCYGPATTERN|($GRADLE_CYGPATTERN)"
129 | fi
130 | # Now convert the arguments - kludge to limit ourselves to /bin/sh
131 | i=0
132 | for arg in "$@" ; do
133 | CHECK=`echo "$arg"|egrep -c "$OURCYGPATTERN" -`
134 | CHECK2=`echo "$arg"|egrep -c "^-"` ### Determine if an option
135 |
136 | if [ $CHECK -ne 0 ] && [ $CHECK2 -eq 0 ] ; then ### Added a condition
137 | eval `echo args$i`=`cygpath --path --ignore --mixed "$arg"`
138 | else
139 | eval `echo args$i`="\"$arg\""
140 | fi
141 | i=$((i+1))
142 | done
143 | case $i in
144 | (0) set -- ;;
145 | (1) set -- "$args0" ;;
146 | (2) set -- "$args0" "$args1" ;;
147 | (3) set -- "$args0" "$args1" "$args2" ;;
148 | (4) set -- "$args0" "$args1" "$args2" "$args3" ;;
149 | (5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;;
150 | (6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;;
151 | (7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;;
152 | (8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;;
153 | (9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;;
154 | esac
155 | fi
156 |
157 | # Split up the JVM_OPTS And GRADLE_OPTS values into an array, following the shell quoting and substitution rules
158 | function splitJvmOpts() {
159 | JVM_OPTS=("$@")
160 | }
161 | eval splitJvmOpts $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS
162 | JVM_OPTS[${#JVM_OPTS[*]}]="-Dorg.gradle.appname=$APP_BASE_NAME"
163 |
164 | exec "$JAVACMD" "${JVM_OPTS[@]}" -classpath "$CLASSPATH" org.gradle.wrapper.GradleWrapperMain "$@"
165 |
--------------------------------------------------------------------------------
/library/library.iml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
10 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
18 | generateDebugAndroidTestSources
19 | generateDebugSources
20 |
21 |
22 |
23 |
24 |
25 |
26 |
27 |
28 |
29 |
30 |
31 |
32 |
33 |
34 |
35 |
36 |
37 |
38 |
39 |
40 |
41 |
42 |
43 |
44 |
45 |
46 |
47 |
48 |
49 |
50 |
51 |
52 |
53 |
54 |
55 |
56 |
57 |
58 |
59 |
60 |
61 |
62 |
63 |
64 |
65 |
66 |
67 |
68 |
69 |
70 |
71 |
72 |
73 |
74 |
75 |
76 |
77 |
78 |
79 |
80 |
81 |
82 |
83 |
84 |
85 |
86 |
87 |
88 |
89 |
90 |
91 |
92 |
93 |
94 |
95 |
96 |
97 |
98 |
--------------------------------------------------------------------------------
/library/src/main/java/com/cundong/recyclerview/HeaderAndFooterRecyclerViewAdapter.java:
--------------------------------------------------------------------------------
1 | package com.cundong.recyclerview;
2 |
3 | import android.support.v7.widget.RecyclerView;
4 | import android.support.v7.widget.StaggeredGridLayoutManager;
5 | import android.view.View;
6 | import android.view.ViewGroup;
7 |
8 | import java.util.ArrayList;
9 |
10 | /**
11 | * Created by cundong on 2015/10/9.
12 | *
13 | * RecyclerView.Adapter with Header and Footer
14 | */
15 | public class HeaderAndFooterRecyclerViewAdapter extends RecyclerView.Adapter {
16 |
17 | private static final int TYPE_HEADER_VIEW = Integer.MIN_VALUE;
18 | private static final int TYPE_FOOTER_VIEW = Integer.MIN_VALUE + 1;
19 |
20 | /**
21 | * RecyclerView使用的,真正的Adapter
22 | */
23 | private RecyclerView.Adapter mInnerAdapter;
24 |
25 | private ArrayList mHeaderViews = new ArrayList<>();
26 | private ArrayList mFooterViews = new ArrayList<>();
27 |
28 | private RecyclerView.AdapterDataObserver mDataObserver = new RecyclerView.AdapterDataObserver() {
29 |
30 | @Override
31 | public void onChanged() {
32 | super.onChanged();
33 | notifyDataSetChanged();
34 | }
35 |
36 | @Override
37 | public void onItemRangeChanged(int positionStart, int itemCount) {
38 | super.onItemRangeChanged(positionStart, itemCount);
39 | notifyItemRangeChanged(positionStart + getHeaderViewsCount(), itemCount);
40 | }
41 |
42 | @Override
43 | public void onItemRangeInserted(int positionStart, int itemCount) {
44 | super.onItemRangeInserted(positionStart, itemCount);
45 | notifyItemRangeInserted(positionStart + getHeaderViewsCount(), itemCount);
46 | }
47 |
48 | @Override
49 | public void onItemRangeRemoved(int positionStart, int itemCount) {
50 | super.onItemRangeRemoved(positionStart, itemCount);
51 | notifyItemRangeRemoved(positionStart + getHeaderViewsCount(), itemCount);
52 | }
53 |
54 | @Override
55 | public void onItemRangeMoved(int fromPosition, int toPosition, int itemCount) {
56 | super.onItemRangeMoved(fromPosition, toPosition, itemCount);
57 | int headerViewsCountCount = getHeaderViewsCount();
58 | notifyItemRangeChanged(fromPosition + headerViewsCountCount, toPosition + headerViewsCountCount + itemCount);
59 | }
60 | };
61 |
62 | public HeaderAndFooterRecyclerViewAdapter() {
63 | }
64 |
65 | public HeaderAndFooterRecyclerViewAdapter(RecyclerView.Adapter innerAdapter) {
66 | setAdapter(innerAdapter);
67 | }
68 |
69 | /**
70 | * 设置adapter
71 | * @param adapter
72 | */
73 | public void setAdapter(RecyclerView.Adapter adapter) {
74 |
75 | if (adapter != null) {
76 | if (!(adapter instanceof RecyclerView.Adapter))
77 | throw new RuntimeException("your adapter must be a RecyclerView.Adapter");
78 | }
79 |
80 | if (mInnerAdapter != null) {
81 | notifyItemRangeRemoved(getHeaderViewsCount(), mInnerAdapter.getItemCount());
82 | mInnerAdapter.unregisterAdapterDataObserver(mDataObserver);
83 | }
84 |
85 | this.mInnerAdapter = adapter;
86 | mInnerAdapter.registerAdapterDataObserver(mDataObserver);
87 | notifyItemRangeInserted(getHeaderViewsCount(), mInnerAdapter.getItemCount());
88 | }
89 |
90 | public RecyclerView.Adapter getInnerAdapter() {
91 | return mInnerAdapter;
92 | }
93 |
94 | public void addHeaderView(View header) {
95 |
96 | if (header == null) {
97 | throw new RuntimeException("header is null");
98 | }
99 |
100 | mHeaderViews.add(header);
101 | this.notifyDataSetChanged();
102 | }
103 |
104 | public void addFooterView(View footer) {
105 |
106 | if (footer == null) {
107 | throw new RuntimeException("footer is null");
108 | }
109 |
110 | mFooterViews.add(footer);
111 | this.notifyDataSetChanged();
112 | }
113 |
114 | /**
115 | * 返回第一个FoView
116 | * @return
117 | */
118 | public View getFooterView() {
119 | return getFooterViewsCount()>0 ? mFooterViews.get(0) : null;
120 | }
121 |
122 | /**
123 | * 返回第一个HeaderView
124 | * @return
125 | */
126 | public View getHeaderView() {
127 | return getHeaderViewsCount()>0 ? mHeaderViews.get(0) : null;
128 | }
129 |
130 | public void removeHeaderView(View view) {
131 | mHeaderViews.remove(view);
132 | this.notifyDataSetChanged();
133 | }
134 |
135 | public void removeFooterView(View view) {
136 | mFooterViews.remove(view);
137 | this.notifyDataSetChanged();
138 | }
139 |
140 | public int getHeaderViewsCount() {
141 | return mHeaderViews.size();
142 | }
143 |
144 | public int getFooterViewsCount() {
145 | return mFooterViews.size();
146 | }
147 |
148 | public boolean isHeader(int position) {
149 | return getHeaderViewsCount() > 0 && position == 0;
150 | }
151 |
152 | public boolean isFooter(int position) {
153 | int lastPosition = getItemCount() - 1;
154 | return getFooterViewsCount() > 0 && position == lastPosition;
155 | }
156 |
157 | @Override
158 | public RecyclerView.ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
159 | int headerViewsCountCount = getHeaderViewsCount();
160 | if (viewType < TYPE_HEADER_VIEW + headerViewsCountCount) {
161 | return new ViewHolder(mHeaderViews.get(viewType - TYPE_HEADER_VIEW));
162 | } else if (viewType >= TYPE_FOOTER_VIEW && viewType < Integer.MAX_VALUE / 2) {
163 | return new ViewHolder(mFooterViews.get(viewType - TYPE_FOOTER_VIEW));
164 | } else {
165 | return mInnerAdapter.onCreateViewHolder(parent, viewType - Integer.MAX_VALUE / 2);
166 | }
167 | }
168 |
169 | @Override
170 | public void onBindViewHolder(RecyclerView.ViewHolder holder, int position) {
171 | int headerViewsCountCount = getHeaderViewsCount();
172 | if (position >= headerViewsCountCount && position < headerViewsCountCount + mInnerAdapter.getItemCount()) {
173 | mInnerAdapter.onBindViewHolder(holder, position - headerViewsCountCount);
174 | } else {
175 | ViewGroup.LayoutParams layoutParams = holder.itemView.getLayoutParams();
176 | if(layoutParams instanceof StaggeredGridLayoutManager.LayoutParams) {
177 | ((StaggeredGridLayoutManager.LayoutParams) layoutParams).setFullSpan(true);
178 | }
179 | }
180 | }
181 |
182 | @Override
183 | public int getItemCount() {
184 | return getHeaderViewsCount() + getFooterViewsCount() + mInnerAdapter.getItemCount();
185 | }
186 |
187 | @Override
188 | public int getItemViewType(int position) {
189 | int innerCount = mInnerAdapter.getItemCount();
190 | int headerViewsCountCount = getHeaderViewsCount();
191 | if (position < headerViewsCountCount) {
192 | return TYPE_HEADER_VIEW + position;
193 | } else if (headerViewsCountCount <= position && position < headerViewsCountCount + innerCount) {
194 |
195 | int innerItemViewType = mInnerAdapter.getItemViewType(position - headerViewsCountCount);
196 | if(innerItemViewType >= Integer.MAX_VALUE / 2) {
197 | throw new IllegalArgumentException("your adapter's return value of getViewTypeCount() must < Integer.MAX_VALUE / 2");
198 | }
199 | return innerItemViewType + Integer.MAX_VALUE / 2;
200 | } else {
201 | return TYPE_FOOTER_VIEW + position - headerViewsCountCount - innerCount;
202 | }
203 | }
204 |
205 | public static class ViewHolder extends RecyclerView.ViewHolder {
206 | public ViewHolder(View itemView) {
207 | super(itemView);
208 | }
209 | }
210 | }
211 |
--------------------------------------------------------------------------------
/sample/sample.iml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
10 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
18 | generateDebugAndroidTestSources
19 | generateDebugSources
20 |
21 |
22 |
23 |
24 |
25 |
26 |
27 |
28 |
29 |
30 |
31 |
32 |
33 |
34 |
35 |
36 |
37 |
38 |
39 |
40 |
41 |
42 |
43 |
44 |
45 |
46 |
47 |
48 |
49 |
50 |
51 |
52 |
53 |
54 |
55 |
56 |
57 |
58 |
59 |
60 |
61 |
62 |
63 |
64 |
65 |
66 |
67 |
68 |
69 |
70 |
71 |
72 |
73 |
74 |
75 |
76 |
77 |
78 |
79 |
80 |
81 |
82 |
83 |
84 |
85 |
86 |
87 |
88 |
89 |
90 |
91 |
92 |
93 |
94 |
95 |
96 |
97 |
98 |
99 |
100 |
101 |
102 |
--------------------------------------------------------------------------------
/sample/src/main/java/com/cundong/recyclerview/sample/EndlessGridLayoutActivity.java:
--------------------------------------------------------------------------------
1 | package com.cundong.recyclerview.sample;
2 |
3 | import android.content.Context;
4 | import android.os.Bundle;
5 | import android.os.Handler;
6 | import android.os.Message;
7 | import android.support.v7.app.AppCompatActivity;
8 | import android.support.v7.widget.GridLayoutManager;
9 | import android.support.v7.widget.LinearLayoutManager;
10 | import android.support.v7.widget.RecyclerView;
11 | import android.util.Log;
12 | import android.view.LayoutInflater;
13 | import android.view.View;
14 | import android.view.ViewGroup;
15 | import android.widget.TextView;
16 | import android.widget.Toast;
17 |
18 | import com.cundong.recyclerview.EndlessRecyclerOnScrollListener;
19 | import com.cundong.recyclerview.HeaderAndFooterRecyclerViewAdapter;
20 | import com.cundong.recyclerview.HeaderSpanSizeLookup;
21 | import com.cundong.recyclerview.RecyclerViewUtils;
22 | import com.cundong.recyclerview.sample.utils.NetworkUtils;
23 | import com.cundong.recyclerview.sample.utils.RecyclerViewStateUtils;
24 | import com.cundong.recyclerview.sample.weight.LoadingFooter;
25 | import com.cundong.recyclerview.sample.weight.SampleHeader;
26 |
27 | import java.lang.ref.WeakReference;
28 | import java.util.ArrayList;
29 |
30 | /**
31 | * Created by cundong on 2015/10/29.
32 | *
33 | * 带HeaderView的分页加载GridLayout RecyclerView
34 | */
35 | public class EndlessGridLayoutActivity extends AppCompatActivity {
36 |
37 | /**服务器端一共多少条数据*/
38 | private static final int TOTAL_COUNTER = 64;
39 |
40 | /**每一页展示多少条数据*/
41 | private static final int REQUEST_COUNT = 10;
42 |
43 | /**已经获取到多少条数据了*/
44 | private int mCurrentCounter = 0;
45 |
46 | private RecyclerView mRecyclerView = null;
47 |
48 | private DataAdapter mDataAdapter = null;
49 |
50 | private PreviewHandler mHandler = new PreviewHandler(this);
51 | private HeaderAndFooterRecyclerViewAdapter mHeaderAndFooterRecyclerViewAdapter = null;
52 |
53 | @Override
54 | public void onCreate(Bundle savedInstanceState) {
55 | super.onCreate(savedInstanceState);
56 | setContentView(R.layout.sample_activity);
57 |
58 | mRecyclerView = (RecyclerView) findViewById(R.id.list);
59 |
60 | //init data
61 | ArrayList dataList = new ArrayList<>();
62 | for (int i = 0; i < 10; i++) {
63 | dataList.add("item" + i);
64 | }
65 |
66 | mCurrentCounter = dataList.size();
67 |
68 | mDataAdapter = new DataAdapter(this);
69 | mDataAdapter.addAll(dataList);
70 |
71 | mHeaderAndFooterRecyclerViewAdapter = new HeaderAndFooterRecyclerViewAdapter(mDataAdapter);
72 | mRecyclerView.setAdapter(mHeaderAndFooterRecyclerViewAdapter);
73 |
74 | //setLayoutManager
75 | GridLayoutManager manager = new GridLayoutManager(this, 2);
76 | manager.setSpanSizeLookup(new HeaderSpanSizeLookup((HeaderAndFooterRecyclerViewAdapter) mRecyclerView.getAdapter(), manager.getSpanCount()));
77 | mRecyclerView.setLayoutManager(manager);
78 |
79 | RecyclerViewUtils.setHeaderView(mRecyclerView, new SampleHeader(this));
80 |
81 | mRecyclerView.addOnScrollListener(mOnScrollListener);
82 | }
83 |
84 | private void notifyDataSetChanged() {
85 | mHeaderAndFooterRecyclerViewAdapter.notifyDataSetChanged();
86 | }
87 |
88 | private void addItems(ArrayList list) {
89 | mDataAdapter.addAll(list);
90 | mCurrentCounter += list.size();
91 | }
92 |
93 | private EndlessRecyclerOnScrollListener mOnScrollListener = new EndlessRecyclerOnScrollListener() {
94 |
95 | @Override
96 | public void onLoadNextPage(View view) {
97 | super.onLoadNextPage(view);
98 |
99 | LoadingFooter.State state = RecyclerViewStateUtils.getFooterViewState(mRecyclerView);
100 | if(state == LoadingFooter.State.Loading) {
101 | Log.d("@Cundong", "the state is Loading, just wait..");
102 | return;
103 | }
104 |
105 | if (mCurrentCounter < TOTAL_COUNTER) {
106 | // loading more
107 | RecyclerViewStateUtils.setFooterViewState(EndlessGridLayoutActivity.this, mRecyclerView, REQUEST_COUNT, LoadingFooter.State.Loading, null);
108 | requestData();
109 | } else {
110 | //the end
111 | RecyclerViewStateUtils.setFooterViewState(EndlessGridLayoutActivity.this, mRecyclerView, REQUEST_COUNT, LoadingFooter.State.TheEnd, null);
112 | }
113 | }
114 | };
115 |
116 | private static class PreviewHandler extends Handler {
117 |
118 | private WeakReference ref;
119 |
120 | PreviewHandler(EndlessGridLayoutActivity activity) {
121 | ref = new WeakReference<>(activity);
122 | }
123 |
124 | @Override
125 | public void handleMessage(Message msg) {
126 | final EndlessGridLayoutActivity activity = ref.get();
127 | if (activity == null || activity.isFinishing()) {
128 | return;
129 | }
130 |
131 | switch (msg.what) {
132 | case -1:
133 | int currentSize = activity.mDataAdapter.getItemCount();
134 |
135 | //模拟组装10个数据
136 | ArrayList newList = new ArrayList<>();
137 | for (int i = 0; i < 10; i++) {
138 | if (newList.size() + currentSize >= TOTAL_COUNTER) {
139 | break;
140 | }
141 |
142 | newList.add("item" + (currentSize + i));
143 | }
144 |
145 | activity.addItems(newList);
146 | RecyclerViewStateUtils.setFooterViewState(activity.mRecyclerView, LoadingFooter.State.Normal);
147 | break;
148 | case -2:
149 | activity.notifyDataSetChanged();
150 | break;
151 | case -3:
152 | RecyclerViewStateUtils.setFooterViewState(activity, activity.mRecyclerView, REQUEST_COUNT, LoadingFooter.State.NetWorkError, activity.mFooterClick);
153 | break;
154 | }
155 | }
156 | }
157 |
158 | private View.OnClickListener mFooterClick = new View.OnClickListener() {
159 | @Override
160 | public void onClick(View v) {
161 | RecyclerViewStateUtils.setFooterViewState(EndlessGridLayoutActivity.this, mRecyclerView, REQUEST_COUNT, LoadingFooter.State.Loading, null);
162 | requestData();
163 | }
164 | };
165 |
166 | /**
167 | * 模拟请求网络
168 | */
169 | private void requestData() {
170 |
171 | new Thread() {
172 |
173 | @Override
174 | public void run() {
175 | super.run();
176 |
177 | try {
178 | Thread.sleep(1000);
179 | } catch (InterruptedException e) {
180 | e.printStackTrace();
181 | }
182 |
183 | //模拟一下网络请求失败的情况
184 | if(NetworkUtils.isNetAvailable(EndlessGridLayoutActivity.this)) {
185 | mHandler.sendEmptyMessage(-1);
186 | } else {
187 | mHandler.sendEmptyMessage(-3);
188 | }
189 | }
190 | }.start();
191 | }
192 |
193 | private class DataAdapter extends RecyclerView.Adapter {
194 |
195 | private LayoutInflater mLayoutInflater;
196 | private ArrayList mDataList = new ArrayList<>();
197 |
198 | public DataAdapter(Context context) {
199 | mLayoutInflater = LayoutInflater.from(context);
200 | }
201 |
202 | private void addAll(ArrayList list) {
203 | int lastIndex = this.mDataList.size();
204 | if (this.mDataList.addAll(list)) {
205 | notifyItemRangeInserted(lastIndex, list.size());
206 | }
207 | }
208 |
209 | @Override
210 | public RecyclerView.ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
211 | return new ViewHolder(mLayoutInflater.inflate(R.layout.sample_item_card, parent, false));
212 | }
213 |
214 | @Override
215 | public void onBindViewHolder(RecyclerView.ViewHolder holder, int position) {
216 |
217 | String item = mDataList.get(position);
218 |
219 | ViewHolder viewHolder = (ViewHolder) holder;
220 | viewHolder.textView.setText(item);
221 | }
222 |
223 | @Override
224 | public int getItemCount() {
225 | return mDataList.size();
226 | }
227 |
228 | private class ViewHolder extends RecyclerView.ViewHolder {
229 |
230 | private TextView textView;
231 |
232 | public ViewHolder(View itemView) {
233 | super(itemView);
234 | textView = (TextView) itemView.findViewById(R.id.info_text);
235 |
236 | textView.setOnClickListener( new View.OnClickListener() {
237 | @Override
238 | public void onClick(View v) {
239 | String text = mDataList.get(RecyclerViewUtils.getAdapterPosition(mRecyclerView, ViewHolder.this));
240 | Toast.makeText(EndlessGridLayoutActivity.this, text, Toast.LENGTH_SHORT).show();
241 | }
242 | });
243 | }
244 | }
245 | }
246 | }
--------------------------------------------------------------------------------
/sample/src/main/java/com/cundong/recyclerview/sample/EndlessStaggeredGridLayoutActivity.java:
--------------------------------------------------------------------------------
1 | package com.cundong.recyclerview.sample;
2 |
3 | import android.content.Context;
4 | import android.os.Bundle;
5 | import android.os.Handler;
6 | import android.os.Message;
7 | import android.support.v7.app.AppCompatActivity;
8 | import android.support.v7.widget.CardView;
9 | import android.support.v7.widget.RecyclerView;
10 | import android.support.v7.widget.StaggeredGridLayoutManager;
11 | import android.util.Log;
12 | import android.view.LayoutInflater;
13 | import android.view.View;
14 | import android.view.ViewGroup;
15 | import android.widget.TextView;
16 | import android.widget.Toast;
17 |
18 | import com.cundong.recyclerview.EndlessRecyclerOnScrollListener;
19 | import com.cundong.recyclerview.ExStaggeredGridLayoutManager;
20 | import com.cundong.recyclerview.HeaderAndFooterRecyclerViewAdapter;
21 | import com.cundong.recyclerview.HeaderSpanSizeLookup;
22 | import com.cundong.recyclerview.RecyclerViewUtils;
23 | import com.cundong.recyclerview.sample.utils.NetworkUtils;
24 | import com.cundong.recyclerview.sample.utils.RecyclerViewStateUtils;
25 | import com.cundong.recyclerview.sample.weight.LoadingFooter;
26 | import com.cundong.recyclerview.sample.weight.SampleHeader;
27 |
28 | import java.lang.ref.WeakReference;
29 | import java.util.ArrayList;
30 |
31 | /**
32 | * Created by cundong on 2015/10/29.
33 | *
34 | * 带HeaderView的分页加载GridLayout RecyclerView
35 | */
36 | public class EndlessStaggeredGridLayoutActivity extends AppCompatActivity {
37 |
38 | /**服务器端一共多少条数据*/
39 | private static final int TOTAL_COUNTER = 64;
40 |
41 | /**每一页展示多少条数据*/
42 | private static final int REQUEST_COUNT = 10;
43 |
44 | /**已经获取到多少条数据了*/
45 | private int mCurrentCounter = 0;
46 |
47 | private RecyclerView mRecyclerView = null;
48 |
49 | private DataAdapter mDataAdapter = null;
50 |
51 | private PreviewHandler mHandler = new PreviewHandler(this);
52 | private HeaderAndFooterRecyclerViewAdapter mHeaderAndFooterRecyclerViewAdapter = null;
53 |
54 | @Override
55 | public void onCreate(Bundle savedInstanceState) {
56 | super.onCreate(savedInstanceState);
57 | setContentView(R.layout.sample_activity);
58 |
59 | mRecyclerView = (RecyclerView) findViewById(R.id.list);
60 |
61 | //init data
62 | ArrayList dataList = new ArrayList<>();
63 | for (int i = 0; i < 10; i++) {
64 | dataList.add("item" + i);
65 | }
66 |
67 | mCurrentCounter = dataList.size();
68 |
69 | mDataAdapter = new DataAdapter(this);
70 | mDataAdapter.addAll(dataList);
71 |
72 | mHeaderAndFooterRecyclerViewAdapter = new HeaderAndFooterRecyclerViewAdapter(mDataAdapter);
73 | mRecyclerView.setAdapter(mHeaderAndFooterRecyclerViewAdapter);
74 |
75 | //setLayoutManager
76 | ExStaggeredGridLayoutManager manager = new ExStaggeredGridLayoutManager (2, StaggeredGridLayoutManager.VERTICAL);
77 | manager.setSpanSizeLookup(new HeaderSpanSizeLookup((HeaderAndFooterRecyclerViewAdapter) mRecyclerView.getAdapter(), manager.getSpanCount()));
78 | mRecyclerView.setLayoutManager(manager);
79 |
80 | RecyclerViewUtils.setHeaderView(mRecyclerView, new SampleHeader(this));
81 |
82 | mRecyclerView.addOnScrollListener(mOnScrollListener);
83 | }
84 |
85 | private void addItems(ArrayList list) {
86 | mDataAdapter.addAll(list);
87 | mCurrentCounter += list.size();
88 | }
89 |
90 | private void notifyDataSetChanged() {
91 | mHeaderAndFooterRecyclerViewAdapter.notifyDataSetChanged();
92 | }
93 |
94 | private EndlessRecyclerOnScrollListener mOnScrollListener = new EndlessRecyclerOnScrollListener() {
95 |
96 | @Override
97 | public void onLoadNextPage(View view) {
98 | super.onLoadNextPage(view);
99 |
100 | LoadingFooter.State state = RecyclerViewStateUtils.getFooterViewState(mRecyclerView);
101 | if(state == LoadingFooter.State.Loading) {
102 | Log.d("@Cundong", "the state is Loading, just wait..");
103 | return;
104 | }
105 |
106 | if (mCurrentCounter < TOTAL_COUNTER) {
107 | // loading more
108 | RecyclerViewStateUtils.setFooterViewState(EndlessStaggeredGridLayoutActivity.this, mRecyclerView, REQUEST_COUNT, LoadingFooter.State.Loading, null);
109 | requestData();
110 | } else {
111 | //the end
112 | RecyclerViewStateUtils.setFooterViewState(EndlessStaggeredGridLayoutActivity.this, mRecyclerView, REQUEST_COUNT, LoadingFooter.State.TheEnd, null);
113 | }
114 | }
115 | };
116 |
117 | private static class PreviewHandler extends Handler {
118 |
119 | private WeakReference ref;
120 |
121 | PreviewHandler(EndlessStaggeredGridLayoutActivity activity) {
122 | ref = new WeakReference<>(activity);
123 | }
124 |
125 | @Override
126 | public void handleMessage(Message msg) {
127 | final EndlessStaggeredGridLayoutActivity activity = ref.get();
128 | if (activity == null || activity.isFinishing()) {
129 | return;
130 | }
131 |
132 | switch (msg.what) {
133 | case -1:
134 | int currentSize = activity.mDataAdapter.getItemCount();
135 |
136 | //模拟组装10个数据
137 | ArrayList newList = new ArrayList<>();
138 | for (int i = 0; i < 10; i++) {
139 | if (newList.size() + currentSize >= TOTAL_COUNTER) {
140 | break;
141 | }
142 |
143 | newList.add("item" + (currentSize + i));
144 | }
145 |
146 | activity.addItems(newList);
147 | RecyclerViewStateUtils.setFooterViewState(activity.mRecyclerView, LoadingFooter.State.Normal);
148 | break;
149 | case -2:
150 | activity.notifyDataSetChanged();
151 | break;
152 | case -3:
153 | RecyclerViewStateUtils.setFooterViewState(activity, activity.mRecyclerView, REQUEST_COUNT, LoadingFooter.State.NetWorkError, activity.mFooterClick);
154 | break;
155 | }
156 | }
157 | }
158 |
159 | private View.OnClickListener mFooterClick = new View.OnClickListener() {
160 | @Override
161 | public void onClick(View v) {
162 | RecyclerViewStateUtils.setFooterViewState(EndlessStaggeredGridLayoutActivity.this, mRecyclerView, REQUEST_COUNT, LoadingFooter.State.Loading, null);
163 | requestData();
164 | }
165 | };
166 |
167 | /**
168 | * 模拟请求网络
169 | */
170 | private void requestData() {
171 |
172 | new Thread() {
173 |
174 | @Override
175 | public void run() {
176 | super.run();
177 |
178 | try {
179 | Thread.sleep(1000);
180 | } catch (InterruptedException e) {
181 | e.printStackTrace();
182 | }
183 |
184 | //模拟一下网络请求失败的情况
185 | if(NetworkUtils.isNetAvailable(EndlessStaggeredGridLayoutActivity.this)) {
186 | mHandler.sendEmptyMessage(-1);
187 | } else {
188 | mHandler.sendEmptyMessage(-3);
189 | }
190 | }
191 | }.start();
192 | }
193 |
194 | private class DataAdapter extends RecyclerView.Adapter {
195 |
196 | private LayoutInflater mLayoutInflater;
197 | private ArrayList mDataList = new ArrayList<>();
198 |
199 | private int largeCardHeight, smallCardHeight;
200 |
201 | public DataAdapter(Context context) {
202 | mLayoutInflater = LayoutInflater.from(context);
203 | largeCardHeight = (int)context.getResources().getDisplayMetrics().density * 300;
204 | smallCardHeight = (int)context.getResources().getDisplayMetrics().density * 200;
205 | }
206 |
207 | private void addAll(ArrayList list) {
208 | int lastIndex = this.mDataList.size();
209 | if (this.mDataList.addAll(list)) {
210 | notifyItemRangeInserted(lastIndex, list.size());
211 | }
212 | }
213 |
214 | @Override
215 | public RecyclerView.ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
216 | return new ViewHolder(mLayoutInflater.inflate(R.layout.sample_item_card, parent, false));
217 | }
218 |
219 | @Override
220 | public void onBindViewHolder(RecyclerView.ViewHolder holder, int position) {
221 |
222 | String item = mDataList.get(position);
223 |
224 | ViewHolder viewHolder = (ViewHolder) holder;
225 | viewHolder.textView.setText(item);
226 |
227 | //修改高度,模拟交错效果
228 | viewHolder.cardView.getLayoutParams().height = position % 2 != 0 ? largeCardHeight : smallCardHeight;
229 | }
230 |
231 | @Override
232 | public int getItemCount() {
233 | return mDataList.size();
234 | }
235 |
236 | private class ViewHolder extends RecyclerView.ViewHolder {
237 |
238 | private CardView cardView;
239 | private TextView textView;
240 |
241 | public ViewHolder(View itemView) {
242 | super(itemView);
243 | cardView = (CardView) itemView.findViewById(R.id.card_view);
244 | textView = (TextView) itemView.findViewById(R.id.info_text);
245 |
246 | textView.setOnClickListener( new View.OnClickListener() {
247 | @Override
248 | public void onClick(View v) {
249 | String text = mDataList.get(RecyclerViewUtils.getAdapterPosition(mRecyclerView, ViewHolder.this));
250 | Toast.makeText(EndlessStaggeredGridLayoutActivity.this, text, Toast.LENGTH_SHORT).show();
251 | }
252 | });
253 | }
254 | }
255 | }
256 | }
--------------------------------------------------------------------------------
/sample/src/main/java/com/cundong/recyclerview/sample/EndlessLinearLayoutActivity.java:
--------------------------------------------------------------------------------
1 | package com.cundong.recyclerview.sample;
2 |
3 | import android.content.Context;
4 | import android.os.Bundle;
5 | import android.os.Handler;
6 | import android.os.Message;
7 | import android.support.v7.app.AppCompatActivity;
8 | import android.support.v7.util.SortedList;
9 | import android.support.v7.widget.LinearLayoutManager;
10 | import android.support.v7.widget.RecyclerView;
11 | import android.util.Log;
12 | import android.view.LayoutInflater;
13 | import android.view.View;
14 | import android.view.ViewGroup;
15 | import android.widget.TextView;
16 | import android.widget.Toast;
17 |
18 | import com.cundong.recyclerview.EndlessRecyclerOnScrollListener;
19 | import com.cundong.recyclerview.HeaderAndFooterRecyclerViewAdapter;
20 | import com.cundong.recyclerview.RecyclerViewUtils;
21 | import com.cundong.recyclerview.sample.utils.NetworkUtils;
22 | import com.cundong.recyclerview.sample.utils.RecyclerViewStateUtils;
23 | import com.cundong.recyclerview.sample.weight.LoadingFooter;
24 | import com.cundong.recyclerview.sample.weight.SampleHeader;
25 |
26 | import java.lang.ref.WeakReference;
27 | import java.util.ArrayList;
28 |
29 | /**
30 | * Created by cundong on 2015/10/29.
31 | *
32 | * 带HeaderView的分页加载LinearLayout RecyclerView
33 | */
34 | public class EndlessLinearLayoutActivity extends AppCompatActivity {
35 |
36 | /**服务器端一共多少条数据*/
37 | private static final int TOTAL_COUNTER = 64;
38 |
39 | /**每一页展示多少条数据*/
40 | private static final int REQUEST_COUNT = 10;
41 |
42 | /**已经获取到多少条数据了*/
43 | private int mCurrentCounter = 0;
44 |
45 | private RecyclerView mRecyclerView = null;
46 |
47 | private DataAdapter mDataAdapter = null;
48 |
49 | private PreviewHandler mHandler = new PreviewHandler(this);
50 | private HeaderAndFooterRecyclerViewAdapter mHeaderAndFooterRecyclerViewAdapter = null;
51 |
52 | @Override
53 | public void onCreate(Bundle savedInstanceState) {
54 | super.onCreate(savedInstanceState);
55 | setContentView(R.layout.sample_activity);
56 |
57 | mRecyclerView = (RecyclerView) findViewById(R.id.list);
58 |
59 | //init data
60 | ArrayList dataList = new ArrayList<>();
61 |
62 | for (int i = 0; i < 10; i++) {
63 |
64 | ItemModel item = new ItemModel();
65 | item.id = i;
66 | item.title = "item" + i;
67 | dataList.add(item);
68 | }
69 |
70 | mCurrentCounter = dataList.size();
71 |
72 | mDataAdapter = new DataAdapter(this);
73 | mDataAdapter.addItems(dataList);
74 |
75 | mHeaderAndFooterRecyclerViewAdapter = new HeaderAndFooterRecyclerViewAdapter(mDataAdapter);
76 | mRecyclerView.setAdapter(mHeaderAndFooterRecyclerViewAdapter);
77 |
78 | mRecyclerView.setLayoutManager(new LinearLayoutManager(this));
79 |
80 | RecyclerViewUtils.setHeaderView(mRecyclerView, new SampleHeader(this));
81 |
82 | mRecyclerView.addOnScrollListener(mOnScrollListener);
83 | }
84 |
85 | private void notifyDataSetChanged() {
86 | mHeaderAndFooterRecyclerViewAdapter.notifyDataSetChanged();
87 | }
88 |
89 | private void addItems(ArrayList list) {
90 |
91 | mDataAdapter.addItems(list);
92 | mCurrentCounter += list.size();
93 | }
94 |
95 | private EndlessRecyclerOnScrollListener mOnScrollListener = new EndlessRecyclerOnScrollListener() {
96 |
97 | @Override
98 | public void onLoadNextPage(View view) {
99 | super.onLoadNextPage(view);
100 |
101 | LoadingFooter.State state = RecyclerViewStateUtils.getFooterViewState(mRecyclerView);
102 | if(state == LoadingFooter.State.Loading) {
103 | Log.d("@Cundong", "the state is Loading, just wait..");
104 | return;
105 | }
106 |
107 | if (mCurrentCounter < TOTAL_COUNTER) {
108 | // loading more
109 | RecyclerViewStateUtils.setFooterViewState(EndlessLinearLayoutActivity.this, mRecyclerView, REQUEST_COUNT, LoadingFooter.State.Loading, null);
110 | requestData();
111 | } else {
112 | //the end
113 | RecyclerViewStateUtils.setFooterViewState(EndlessLinearLayoutActivity.this, mRecyclerView, REQUEST_COUNT, LoadingFooter.State.TheEnd, null);
114 | }
115 | }
116 | };
117 |
118 | private static class PreviewHandler extends Handler {
119 |
120 | private WeakReference ref;
121 |
122 | PreviewHandler(EndlessLinearLayoutActivity activity) {
123 | ref = new WeakReference<>(activity);
124 | }
125 |
126 | @Override
127 | public void handleMessage(Message msg) {
128 | final EndlessLinearLayoutActivity activity = ref.get();
129 | if (activity == null || activity.isFinishing()) {
130 | return;
131 | }
132 |
133 | switch (msg.what) {
134 | case -1:
135 | int currentSize = activity.mDataAdapter.getItemCount();
136 |
137 | //模拟组装10个数据
138 | ArrayList newList = new ArrayList<>();
139 | for (int i = 0; i < 10; i++) {
140 | if (newList.size() + currentSize >= TOTAL_COUNTER) {
141 | break;
142 | }
143 |
144 | ItemModel item = new ItemModel();
145 | item.id = currentSize + i;
146 | item.title = "item" + (item.id);
147 |
148 | newList.add(item);
149 | }
150 |
151 | activity.addItems(newList);
152 | RecyclerViewStateUtils.setFooterViewState(activity.mRecyclerView, LoadingFooter.State.Normal);
153 | break;
154 | case -2:
155 | activity.notifyDataSetChanged();
156 | break;
157 | case -3:
158 | RecyclerViewStateUtils.setFooterViewState(activity, activity.mRecyclerView, REQUEST_COUNT, LoadingFooter.State.NetWorkError, activity.mFooterClick);
159 | break;
160 | }
161 | }
162 | }
163 |
164 | private View.OnClickListener mFooterClick = new View.OnClickListener() {
165 | @Override
166 | public void onClick(View v) {
167 | RecyclerViewStateUtils.setFooterViewState(EndlessLinearLayoutActivity.this, mRecyclerView, REQUEST_COUNT, LoadingFooter.State.Loading, null);
168 | requestData();
169 | }
170 | };
171 |
172 | /**
173 | * 模拟请求网络
174 | */
175 | private void requestData() {
176 |
177 | new Thread() {
178 |
179 | @Override
180 | public void run() {
181 | super.run();
182 |
183 | try {
184 | Thread.sleep(1000);
185 | } catch (InterruptedException e) {
186 | e.printStackTrace();
187 | }
188 |
189 | //模拟一下网络请求失败的情况
190 | if(NetworkUtils.isNetAvailable(EndlessLinearLayoutActivity.this)) {
191 | mHandler.sendEmptyMessage(-1);
192 | } else {
193 | mHandler.sendEmptyMessage(-3);
194 | }
195 | }
196 | }.start();
197 | }
198 |
199 | private class DataAdapter extends RecyclerView.Adapter {
200 |
201 | private LayoutInflater mLayoutInflater;
202 | private SortedList mSortedList;
203 |
204 | public DataAdapter(Context context) {
205 | mLayoutInflater = LayoutInflater.from(context);
206 | mSortedList = new SortedList<>(ItemModel.class, new SortedList.Callback() {
207 |
208 | /**
209 | * 返回一个负整数(第一个参数小于第二个)、零(相等)或正整数(第一个参数大于第二个)
210 | */
211 | @Override
212 | public int compare(ItemModel o1, ItemModel o2) {
213 |
214 | if (o1.id < o2.id) {
215 | return -1;
216 | } else if (o1.id > o2.id) {
217 | return 1;
218 | }
219 |
220 | return 0;
221 | }
222 |
223 | @Override
224 | public boolean areContentsTheSame(ItemModel oldItem, ItemModel newItem) {
225 | return oldItem.title.equals(newItem.title);
226 | }
227 |
228 | @Override
229 | public boolean areItemsTheSame(ItemModel item1, ItemModel item2) {
230 | return item1.id == item2.id;
231 | }
232 |
233 | @Override
234 | public void onInserted(int position, int count) {
235 | notifyItemRangeInserted(position, count);
236 | }
237 |
238 | @Override
239 | public void onRemoved(int position, int count) {
240 | notifyItemRangeRemoved(position, count);
241 | }
242 |
243 | @Override
244 | public void onMoved(int fromPosition, int toPosition) {
245 | notifyItemMoved(fromPosition, toPosition);
246 | }
247 |
248 | @Override
249 | public void onChanged(int position, int count) {
250 | notifyItemRangeChanged(position, count);
251 | }
252 | });
253 | }
254 |
255 | public void addItems(ArrayList list) {
256 | mSortedList.beginBatchedUpdates();
257 |
258 | for(ItemModel itemModel : list) {
259 | mSortedList.add(itemModel);
260 | }
261 |
262 | mSortedList.endBatchedUpdates();
263 | }
264 |
265 | public void deleteItems(ArrayList items) {
266 | mSortedList.beginBatchedUpdates();
267 | for (ItemModel item : items) {
268 | mSortedList.remove(item);
269 | }
270 | mSortedList.endBatchedUpdates();
271 | }
272 |
273 | @Override
274 | public RecyclerView.ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
275 | return new ViewHolder(mLayoutInflater.inflate(R.layout.sample_item_text, parent, false));
276 | }
277 |
278 | @Override
279 | public void onBindViewHolder(RecyclerView.ViewHolder holder, int position) {
280 |
281 | ItemModel item = mSortedList.get(position);
282 |
283 | ViewHolder viewHolder = (ViewHolder) holder;
284 | viewHolder.textView.setText(item.title);
285 | }
286 |
287 | @Override
288 | public int getItemCount() {
289 | return mSortedList.size();
290 | }
291 |
292 | private class ViewHolder extends RecyclerView.ViewHolder {
293 |
294 | private TextView textView;
295 |
296 | public ViewHolder(View itemView) {
297 | super(itemView);
298 | textView = (TextView) itemView.findViewById(R.id.info_text);
299 |
300 | textView.setOnClickListener( new View.OnClickListener() {
301 | @Override
302 | public void onClick(View v) {
303 | ItemModel item = mSortedList.get(RecyclerViewUtils.getAdapterPosition(mRecyclerView, ViewHolder.this));
304 | Toast.makeText(EndlessLinearLayoutActivity.this, item.title, Toast.LENGTH_SHORT).show();
305 | }
306 | });
307 | }
308 | }
309 | }
310 | }
--------------------------------------------------------------------------------
/LICENSE:
--------------------------------------------------------------------------------
1 | Apache License
2 | Version 2.0, January 2004
3 | http://www.apache.org/licenses/
4 |
5 | TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION
6 |
7 | 1. Definitions.
8 |
9 | "License" shall mean the terms and conditions for use, reproduction,
10 | and distribution as defined by Sections 1 through 9 of this document.
11 |
12 | "Licensor" shall mean the copyright owner or entity authorized by
13 | the copyright owner that is granting the License.
14 |
15 | "Legal Entity" shall mean the union of the acting entity and all
16 | other entities that control, are controlled by, or are under common
17 | control with that entity. For the purposes of this definition,
18 | "control" means (i) the power, direct or indirect, to cause the
19 | direction or management of such entity, whether by contract or
20 | otherwise, or (ii) ownership of fifty percent (50%) or more of the
21 | outstanding shares, or (iii) beneficial ownership of such entity.
22 |
23 | "You" (or "Your") shall mean an individual or Legal Entity
24 | exercising permissions granted by this License.
25 |
26 | "Source" form shall mean the preferred form for making modifications,
27 | including but not limited to software source code, documentation
28 | source, and configuration files.
29 |
30 | "Object" form shall mean any form resulting from mechanical
31 | transformation or translation of a Source form, including but
32 | not limited to compiled object code, generated documentation,
33 | and conversions to other media types.
34 |
35 | "Work" shall mean the work of authorship, whether in Source or
36 | Object form, made available under the License, as indicated by a
37 | copyright notice that is included in or attached to the work
38 | (an example is provided in the Appendix below).
39 |
40 | "Derivative Works" shall mean any work, whether in Source or Object
41 | form, that is based on (or derived from) the Work and for which the
42 | editorial revisions, annotations, elaborations, or other modifications
43 | represent, as a whole, an original work of authorship. For the purposes
44 | of this License, Derivative Works shall not include works that remain
45 | separable from, or merely link (or bind by name) to the interfaces of,
46 | the Work and Derivative Works thereof.
47 |
48 | "Contribution" shall mean any work of authorship, including
49 | the original version of the Work and any modifications or additions
50 | to that Work or Derivative Works thereof, that is intentionally
51 | submitted to Licensor for inclusion in the Work by the copyright owner
52 | or by an individual or Legal Entity authorized to submit on behalf of
53 | the copyright owner. For the purposes of this definition, "submitted"
54 | means any form of electronic, verbal, or written communication sent
55 | to the Licensor or its representatives, including but not limited to
56 | communication on electronic mailing lists, source code control systems,
57 | and issue tracking systems that are managed by, or on behalf of, the
58 | Licensor for the purpose of discussing and improving the Work, but
59 | excluding communication that is conspicuously marked or otherwise
60 | designated in writing by the copyright owner as "Not a Contribution."
61 |
62 | "Contributor" shall mean Licensor and any individual or Legal Entity
63 | on behalf of whom a Contribution has been received by Licensor and
64 | subsequently incorporated within the Work.
65 |
66 | 2. Grant of Copyright License. Subject to the terms and conditions of
67 | this License, each Contributor hereby grants to You a perpetual,
68 | worldwide, non-exclusive, no-charge, royalty-free, irrevocable
69 | copyright license to reproduce, prepare Derivative Works of,
70 | publicly display, publicly perform, sublicense, and distribute the
71 | Work and such Derivative Works in Source or Object form.
72 |
73 | 3. Grant of Patent License. Subject to the terms and conditions of
74 | this License, each Contributor hereby grants to You a perpetual,
75 | worldwide, non-exclusive, no-charge, royalty-free, irrevocable
76 | (except as stated in this section) patent license to make, have made,
77 | use, offer to sell, sell, import, and otherwise transfer the Work,
78 | where such license applies only to those patent claims licensable
79 | by such Contributor that are necessarily infringed by their
80 | Contribution(s) alone or by combination of their Contribution(s)
81 | with the Work to which such Contribution(s) was submitted. If You
82 | institute patent litigation against any entity (including a
83 | cross-claim or counterclaim in a lawsuit) alleging that the Work
84 | or a Contribution incorporated within the Work constitutes direct
85 | or contributory patent infringement, then any patent licenses
86 | granted to You under this License for that Work shall terminate
87 | as of the date such litigation is filed.
88 |
89 | 4. Redistribution. You may reproduce and distribute copies of the
90 | Work or Derivative Works thereof in any medium, with or without
91 | modifications, and in Source or Object form, provided that You
92 | meet the following conditions:
93 |
94 | (a) You must give any other recipients of the Work or
95 | Derivative Works a copy of this License; and
96 |
97 | (b) You must cause any modified files to carry prominent notices
98 | stating that You changed the files; and
99 |
100 | (c) You must retain, in the Source form of any Derivative Works
101 | that You distribute, all copyright, patent, trademark, and
102 | attribution notices from the Source form of the Work,
103 | excluding those notices that do not pertain to any part of
104 | the Derivative Works; and
105 |
106 | (d) If the Work includes a "NOTICE" text file as part of its
107 | distribution, then any Derivative Works that You distribute must
108 | include a readable copy of the attribution notices contained
109 | within such NOTICE file, excluding those notices that do not
110 | pertain to any part of the Derivative Works, in at least one
111 | of the following places: within a NOTICE text file distributed
112 | as part of the Derivative Works; within the Source form or
113 | documentation, if provided along with the Derivative Works; or,
114 | within a display generated by the Derivative Works, if and
115 | wherever such third-party notices normally appear. The contents
116 | of the NOTICE file are for informational purposes only and
117 | do not modify the License. You may add Your own attribution
118 | notices within Derivative Works that You distribute, alongside
119 | or as an addendum to the NOTICE text from the Work, provided
120 | that such additional attribution notices cannot be construed
121 | as modifying the License.
122 |
123 | You may add Your own copyright statement to Your modifications and
124 | may provide additional or different license terms and conditions
125 | for use, reproduction, or distribution of Your modifications, or
126 | for any such Derivative Works as a whole, provided Your use,
127 | reproduction, and distribution of the Work otherwise complies with
128 | the conditions stated in this License.
129 |
130 | 5. Submission of Contributions. Unless You explicitly state otherwise,
131 | any Contribution intentionally submitted for inclusion in the Work
132 | by You to the Licensor shall be under the terms and conditions of
133 | this License, without any additional terms or conditions.
134 | Notwithstanding the above, nothing herein shall supersede or modify
135 | the terms of any separate license agreement you may have executed
136 | with Licensor regarding such Contributions.
137 |
138 | 6. Trademarks. This License does not grant permission to use the trade
139 | names, trademarks, service marks, or product names of the Licensor,
140 | except as required for reasonable and customary use in describing the
141 | origin of the Work and reproducing the content of the NOTICE file.
142 |
143 | 7. Disclaimer of Warranty. Unless required by applicable law or
144 | agreed to in writing, Licensor provides the Work (and each
145 | Contributor provides its Contributions) on an "AS IS" BASIS,
146 | WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
147 | implied, including, without limitation, any warranties or conditions
148 | of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A
149 | PARTICULAR PURPOSE. You are solely responsible for determining the
150 | appropriateness of using or redistributing the Work and assume any
151 | risks associated with Your exercise of permissions under this License.
152 |
153 | 8. Limitation of Liability. In no event and under no legal theory,
154 | whether in tort (including negligence), contract, or otherwise,
155 | unless required by applicable law (such as deliberate and grossly
156 | negligent acts) or agreed to in writing, shall any Contributor be
157 | liable to You for damages, including any direct, indirect, special,
158 | incidental, or consequential damages of any character arising as a
159 | result of this License or out of the use or inability to use the
160 | Work (including but not limited to damages for loss of goodwill,
161 | work stoppage, computer failure or malfunction, or any and all
162 | other commercial damages or losses), even if such Contributor
163 | has been advised of the possibility of such damages.
164 |
165 | 9. Accepting Warranty or Additional Liability. While redistributing
166 | the Work or Derivative Works thereof, You may choose to offer,
167 | and charge a fee for, acceptance of support, warranty, indemnity,
168 | or other liability obligations and/or rights consistent with this
169 | License. However, in accepting such obligations, You may act only
170 | on Your own behalf and on Your sole responsibility, not on behalf
171 | of any other Contributor, and only if You agree to indemnify,
172 | defend, and hold each Contributor harmless for any liability
173 | incurred by, or claims asserted against, such Contributor by reason
174 | of your accepting any such warranty or additional liability.
175 |
176 | END OF TERMS AND CONDITIONS
177 |
178 | APPENDIX: How to apply the Apache License to your work.
179 |
180 | To apply the Apache License to your work, attach the following
181 | boilerplate notice, with the fields enclosed by brackets "{}"
182 | replaced with your own identifying information. (Don't include
183 | the brackets!) The text should be enclosed in the appropriate
184 | comment syntax for the file format. We also recommend that a
185 | file or class name and description of purpose be included on the
186 | same "printed page" as the copyright notice for easier
187 | identification within third-party archives.
188 |
189 | Copyright {yyyy} {name of copyright owner}
190 |
191 | Licensed under the Apache License, Version 2.0 (the "License");
192 | you may not use this file except in compliance with the License.
193 | You may obtain a copy of the License at
194 |
195 | http://www.apache.org/licenses/LICENSE-2.0
196 |
197 | Unless required by applicable law or agreed to in writing, software
198 | distributed under the License is distributed on an "AS IS" BASIS,
199 | WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
200 | See the License for the specific language governing permissions and
201 | limitations under the License.
202 |
203 |
--------------------------------------------------------------------------------