Repository: android-cjj/Android-RecyclerViewWithFooter
Branch: master
Commit: 637385b14e53
Files: 41
Total size: 52.0 KB
Directory structure:
gitextract_2qfdse6u/
├── .gitignore
├── README.md
├── app/
│ ├── .gitignore
│ ├── build.gradle
│ ├── proguard-rules.pro
│ └── src/
│ └── main/
│ ├── AndroidManifest.xml
│ ├── java/
│ │ └── com/
│ │ └── cjj/
│ │ └── loadmorervdemo/
│ │ ├── CustomFootItem.java
│ │ ├── DemoRvAdapter.java
│ │ └── MainActivity.java
│ └── res/
│ ├── layout/
│ │ ├── activity_main.xml
│ │ ├── content_main.xml
│ │ ├── foot_view.xml
│ │ └── item_home.xml
│ ├── menu/
│ │ └── menu_main.xml
│ ├── values/
│ │ ├── colors.xml
│ │ ├── dimens.xml
│ │ ├── strings.xml
│ │ └── styles.xml
│ ├── values-v21/
│ │ └── styles.xml
│ └── values-w820dp/
│ └── dimens.xml
├── build.gradle
├── gradle/
│ └── wrapper/
│ ├── gradle-wrapper.jar
│ └── gradle-wrapper.properties
├── gradle.properties
├── gradlew
├── gradlew.bat
├── library/
│ ├── .gitignore
│ ├── build.gradle
│ ├── proguard-rules.pro
│ └── src/
│ └── main/
│ ├── AndroidManifest.xml
│ ├── java/
│ │ └── com/
│ │ └── cjj/
│ │ └── loadmore/
│ │ ├── DefaultEmptyItem.java
│ │ ├── DefaultFootItem.java
│ │ ├── EmptyItem.java
│ │ ├── FootItem.java
│ │ ├── OnLoadMoreListener.java
│ │ ├── RecyclerViewUtils.java
│ │ └── RecyclerViewWithFooter.java
│ └── res/
│ ├── layout/
│ │ ├── rv_with_footer_empty_layout.xml
│ │ └── rv_with_footer_loading.xml
│ └── values/
│ └── rv_with_footer_strings.xml
└── settings.gradle
================================================
FILE CONTENTS
================================================
================================================
FILE: .gitignore
================================================
# Built application files
*.apk
*.ap_
# Files for the Dalvik VM
*.dex
# Java class files
*.class
# Generated files
bin/
gen/
# Gradle files
.gradle/
build/
# Local configuration file (sdk path, etc)
local.properties
# Proguard folder generated by Eclipse
proguard/
# Log Files
*.log
# Android Studio Navigation editor temp files
.navigation/
# Android Studio captures folder
captures/
# jetbrain project file
.idea/
*.iml
================================================
FILE: README.md
================================================
#RecyclerViewWithFooter
之所以会写这个库,是因为最近遇到朋友问推荐一个好的下拉刷新的库,然后我就给了我收集的[BeautifulRefreshLayout](https://github.com/android-cjj/BeautifulRefreshLayout),但是,这时候又有个问题,明明这个库写的很好,就是因为没有加载更多,而另一个库设计的没前一个库好,只是功能比上个库多,所以选择了下一个。对于这种情况我是反对的,呵呵。
举个例子吧,我觉得官方的SwipeRefreshLayout已经是个很好的控件了,就是没有上拉刷新,所以很多人没用,特别是初学android的朋友,因为我们需要有一个库可以解决一切问题,对于这种情况,我给的demo已经很好的解决。
随意说说,为什么很多人不把上拉下拉融合在一起,至少我们的理解是他们是一体的,对,你这样想是没错。但是,我觉得这要看项目的具体情况。我之前写的[MaterialRefreshLayout](https://github.com/android-cjj/Android-MaterialRefreshLayout)就具备了,但是,上拉的时候需要用户自己拉,呵呵,我觉得很多余,另外,那个项目bug很多,慎用。呵呵... 所以,加载更多就是需要滑到底部自动加载,之前是 ListView.addFooterView(view) 的方法,常用作auto loading view ,但是RecyclerView 没有提供这个方法,没事,我写了 。
说了那么多废话,我们还是说说这个项目的功能吧。顾名思义,其实它就是一个给RecyclerView加底部View的库
使用
==================
xml中的布局:
```xml
```
java中
```java
mRecyclerViewWithFooter = (RecyclerViewWithFooter) this.findViewById(R.id.rv_load_more);
mRecyclerViewWithFooter.setAdapter();
```
这样就可以了,是不是很简单,呵呵。
如果需要监听Rv滑到底部,
```java
mRecyclerViewWithFooter.setOnLoadMoreListener(new OnLoadMoreListener() {
@Override
public void onLoadMore() {
//加载数据
}
});
```
RecyclerViewWithFooter有三种类型,分别是:
```java
/**
* 表示现在是切换成 load 状态
*/
public void setLoad() {
}
/**
* 表示切换成没有更多数据状态
*
* @param end
*/
public void setEnd(CharSequence end) {
}
/**
* 表示切换成 无数据 为空状态
*
* @param empty
* @param resId
*/
public void setEmpty(CharSequence empty, @DrawableRes int resId) {
}
```
底部FootView如果不设置setFootItem,默认为DefaultFootItem,内嵌入了MaterialFootItem,效果如下:
(1)这是默认的效果

(2)这是Material风格的

(3)你也可以自己定义

差不多就这些了,想了解更多自己看源码。还有项目肯定存在bug,选择权在于你,不要出了bug就找骂原作者,你没有资格,因为如果你不用,有个屁的bug,如果你用了出问题可以加
GitHub小伙伴交流群'' 477826523
进群有要求,至少你Github有东西,别问为什么,原则问题,呵呵。我也是个菜鸟,也希望高手能帮我,呵呵
特别鸣谢:
--------------------------------
[dim](https://github.com/zzz40500),[markzhai](https://github.com/markzhai)我死基友,呵呵
我的微博:
------------------------------
[http://weibo.com/chenjijun2011](http://weibo.com/chenjijun2011)
License
=======
The MIT License (MIT)
Copyright (c) 2015 android-cjj
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all
copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
SOFTWARE.
================================================
FILE: app/.gitignore
================================================
/build
================================================
FILE: app/build.gradle
================================================
apply plugin: 'com.android.application'
android {
compileSdkVersion 23
buildToolsVersion "23.0.2"
defaultConfig {
applicationId "com.cjj.loadmorervdemo"
minSdkVersion 15
targetSdkVersion 23
versionCode 1
versionName "1.0"
}
buildTypes {
release {
minifyEnabled false
proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
}
}
}
dependencies {
compile fileTree(include: ['*.jar'], dir: 'libs')
testCompile 'junit:junit:4.12'
compile 'com.android.support:appcompat-v7:23.+'
compile 'com.android.support:design:23.+'
compile 'com.android.support:recyclerview-v7:23.1.1'
compile 'in.srain.cube:ultra-ptr:1.0.11'
compile project(':library')
}
================================================
FILE: app/proguard-rules.pro
================================================
# Add project specific ProGuard rules here.
# By default, the flags in this file are appended to flags specified
# in D:\world\android\sdk/tools/proguard/proguard-android.txt
# You can edit the include path and order by changing the proguardFiles
# directive in build.gradle.
#
# For more details, see
# http://developer.android.com/guide/developing/tools/proguard.html
# Add any project specific keep options here:
# If your project uses WebView with JS, uncomment the following
# and specify the fully qualified class name to the JavaScript interface
# class:
#-keepclassmembers class fqcn.of.javascript.interface.for.webview {
# public *;
#}
================================================
FILE: app/src/main/AndroidManifest.xml
================================================
================================================
FILE: app/src/main/java/com/cjj/loadmorervdemo/CustomFootItem.java
================================================
package com.cjj.loadmorervdemo;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import com.cjj.loadmore.FootItem;
import com.cjj.loadmore.RecyclerViewWithFooter;
/**
* Created by cjj on 2016/2/1.
*/
public class CustomFootItem extends FootItem {
@Override
public View onCreateView(ViewGroup parent) {
return LayoutInflater.from(parent.getContext()).inflate(R.layout.foot_view, parent, false);
}
@Override
public void onBindData(View view, int state) {
/**
* state 有 RecyclerViewWithFooter 定义的 STATE_END, STATE_LOADING, STATE_EMPTY, STATE_NONE
*/
if (state == RecyclerViewWithFooter.STATE_LOADING) {
} else if (state == RecyclerViewWithFooter.STATE_END) {
}
}
}
================================================
FILE: app/src/main/java/com/cjj/loadmorervdemo/DemoRvAdapter.java
================================================
package com.cjj.loadmorervdemo;
import android.content.Context;
import android.support.v7.widget.RecyclerView;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.ImageView;
import java.util.List;
public class DemoRvAdapter extends RecyclerView.Adapter {
private Context mContext;
private List mDataList;
public DemoRvAdapter(Context context, List mDatas) {
mContext = context;
mDataList = mDatas;
}
@Override
public MyViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
return new MyViewHolder(
LayoutInflater.from(mContext).inflate(R.layout.item_home, parent, false));
}
@Override
public void onBindViewHolder(MyViewHolder holder, int position) {
holder.mImageView.setImageResource(mDataList.get(position));
}
@Override
public int getItemCount() {
return mDataList.size();
}
class MyViewHolder extends RecyclerView.ViewHolder {
ImageView mImageView;
public MyViewHolder(View view) {
super(view);
mImageView = (ImageView) view.findViewById(R.id.iv_image);
}
}
}
================================================
FILE: app/src/main/java/com/cjj/loadmorervdemo/MainActivity.java
================================================
package com.cjj.loadmorervdemo;
import android.os.Bundle;
import android.os.Handler;
import android.support.v4.widget.SwipeRefreshLayout;
import android.support.v7.app.AppCompatActivity;
import android.support.v7.widget.Toolbar;
import android.view.Menu;
import android.view.MenuItem;
import com.cjj.loadmore.DefaultFootItem;
import com.cjj.loadmore.OnLoadMoreListener;
import com.cjj.loadmore.RecyclerViewWithFooter;
import java.util.ArrayList;
import java.util.List;
public class MainActivity extends AppCompatActivity {
private List mDatas;
private RecyclerViewWithFooter mRecyclerViewWithFooter;
private Handler mHandler = new Handler();
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
final Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);
setSupportActionBar(toolbar);
final SwipeRefreshLayout swipeRefreshLayout = (SwipeRefreshLayout) findViewById(R.id.swipe);
swipeRefreshLayout.setColorSchemeResources(android.R.color.holo_red_light, android.R.color.holo_blue_light, android.R.color.holo_green_light);
swipeRefreshLayout.setOnRefreshListener(new SwipeRefreshLayout.OnRefreshListener() {
@Override
public void onRefresh() {
mHandler.postDelayed(new Runnable() {
@Override
public void run() {
addData();
swipeRefreshLayout.setRefreshing(false);
}
}, 3000);
}
});
initData();
mRecyclerViewWithFooter = (RecyclerViewWithFooter) this.findViewById(R.id.rv_load_more);
mRecyclerViewWithFooter.setAdapter(new DemoRvAdapter(this, mDatas));
// mRecyclerViewWithFooter.setStaggeredGridLayoutManager(2);
mRecyclerViewWithFooter.setFootItem(new DefaultFootItem());//默认是这种
// mRecyclerViewWithFooter.setFootItem(new CustomFootItem());//自定义
mRecyclerViewWithFooter.setOnLoadMoreListener(new OnLoadMoreListener() {
@Override
public void onLoadMore() {
mRecyclerViewWithFooter.postDelayed(new Runnable() {
@Override
public void run() {
addData();
}
}, 2000);
}
});
}
protected void initData() {
mDatas = new ArrayList<>();
mDatas.add(R.mipmap.cat1);
mDatas.add(R.mipmap.cat2);
// mDatas.add(R.mipmap.cat3);
// mDatas.add(R.mipmap.cjj);
// mDatas.add(R.mipmap.cat1);
// mDatas.add(R.mipmap.cat2);
}
protected void addData() {
mDatas.add(R.mipmap.cat1);
mDatas.add(R.mipmap.cat2);
mDatas.add(R.mipmap.cat3);
mDatas.add(R.mipmap.cjj);
mDatas.add(R.mipmap.cat1);
mDatas.add(R.mipmap.cat2);
mRecyclerViewWithFooter.getAdapter().notifyDataSetChanged();
}
@Override
public boolean onCreateOptionsMenu(Menu menu) {
// Inflate the menu; this adds items to the action bar if it is present.
getMenuInflater().inflate(R.menu.menu_main, menu);
return true;
}
@Override
public boolean onOptionsItemSelected(MenuItem item) {
// Handle action bar item clicks here. The action bar will
// automatically handle clicks on the Home/Up button, so long
// as you specify a parent activity in AndroidManifest.xml.
int id = item.getItemId();
//noinspection SimplifiableIfStatement
if (id == R.id.addMoreAction) {
mRecyclerViewWithFooter.setLoading();
addData();
return true;
}
if (id == R.id.endAction) {
mRecyclerViewWithFooter.setEnd("没有更多数据了");
return true;
}
if (id == R.id.emptyAction) {
mDatas.clear();
mRecyclerViewWithFooter.setEmpty("没有数据", R.mipmap.ic_launcher);
return true;
}
return super.onOptionsItemSelected(item);
}
@Override
protected void onDestroy() {
super.onDestroy();
mHandler.removeCallbacksAndMessages(null);
}
}
================================================
FILE: app/src/main/res/layout/activity_main.xml
================================================
================================================
FILE: app/src/main/res/layout/content_main.xml
================================================
================================================
FILE: app/src/main/res/layout/foot_view.xml
================================================
================================================
FILE: app/src/main/res/layout/item_home.xml
================================================
================================================
FILE: app/src/main/res/menu/menu_main.xml
================================================
================================================
FILE: app/src/main/res/values/colors.xml
================================================
@android:color/holo_red_light
#303F9F
#FF4081
================================================
FILE: app/src/main/res/values/dimens.xml
================================================
16dp
16dp
16dp
================================================
FILE: app/src/main/res/values/strings.xml
================================================
RVWithFooter
Settings
加载中...
end
================================================
FILE: app/src/main/res/values/styles.xml
================================================
================================================
FILE: app/src/main/res/values-v21/styles.xml
================================================
================================================
FILE: app/src/main/res/values-w820dp/dimens.xml
================================================
64dp
================================================
FILE: build.gradle
================================================
// Top-level build file where you can add configuration options common to all sub-projects/modules.
buildscript {
repositories {
jcenter()
}
dependencies {
classpath 'com.android.tools.build:gradle:1.3.1'
// NOTE: Do not place your application dependencies here; they belong
// in the individual module build.gradle files
}
}
allprojects {
repositories {
jcenter()
}
}
task clean(type: Delete) {
delete rootProject.buildDir
}
================================================
FILE: gradle/wrapper/gradle-wrapper.properties
================================================
#Mon Dec 28 10:00:20 PST 2015
distributionBase=GRADLE_USER_HOME
distributionPath=wrapper/dists
zipStoreBase=GRADLE_USER_HOME
zipStorePath=wrapper/dists
distributionUrl=https\://services.gradle.org/distributions/gradle-2.8-all.zip
================================================
FILE: gradle.properties
================================================
# Project-wide Gradle settings.
# IDE (e.g. Android Studio) users:
# Gradle settings configured through the IDE *will override*
# any settings specified in this file.
# For more details on how to configure your build environment visit
# http://www.gradle.org/docs/current/userguide/build_environment.html
# Specifies the JVM arguments used for the daemon process.
# The setting is particularly useful for tweaking memory settings.
# Default value: -Xmx10248m -XX:MaxPermSize=256m
# org.gradle.jvmargs=-Xmx2048m -XX:MaxPermSize=512m -XX:+HeapDumpOnOutOfMemoryError -Dfile.encoding=UTF-8
# When configured, Gradle will run in incubating parallel mode.
# This option should only be used with decoupled projects. More details, visit
# http://www.gradle.org/docs/current/userguide/multi_project_builds.html#sec:decoupled_projects
# org.gradle.parallel=true
================================================
FILE: gradlew
================================================
#!/usr/bin/env bash
##############################################################################
##
## Gradle start up script for UN*X
##
##############################################################################
# Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
DEFAULT_JVM_OPTS=""
APP_NAME="Gradle"
APP_BASE_NAME=`basename "$0"`
# Use the maximum available, or set MAX_FD != -1 to use that value.
MAX_FD="maximum"
warn ( ) {
echo "$*"
}
die ( ) {
echo
echo "$*"
echo
exit 1
}
# OS specific support (must be 'true' or 'false').
cygwin=false
msys=false
darwin=false
case "`uname`" in
CYGWIN* )
cygwin=true
;;
Darwin* )
darwin=true
;;
MINGW* )
msys=true
;;
esac
# Attempt to set APP_HOME
# Resolve links: $0 may be a link
PRG="$0"
# Need this for relative symlinks.
while [ -h "$PRG" ] ; do
ls=`ls -ld "$PRG"`
link=`expr "$ls" : '.*-> \(.*\)$'`
if expr "$link" : '/.*' > /dev/null; then
PRG="$link"
else
PRG=`dirname "$PRG"`"/$link"
fi
done
SAVED="`pwd`"
cd "`dirname \"$PRG\"`/" >/dev/null
APP_HOME="`pwd -P`"
cd "$SAVED" >/dev/null
CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar
# Determine the Java command to use to start the JVM.
if [ -n "$JAVA_HOME" ] ; then
if [ -x "$JAVA_HOME/jre/sh/java" ] ; then
# IBM's JDK on AIX uses strange locations for the executables
JAVACMD="$JAVA_HOME/jre/sh/java"
else
JAVACMD="$JAVA_HOME/bin/java"
fi
if [ ! -x "$JAVACMD" ] ; then
die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME
Please set the JAVA_HOME variable in your environment to match the
location of your Java installation."
fi
else
JAVACMD="java"
which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
Please set the JAVA_HOME variable in your environment to match the
location of your Java installation."
fi
# Increase the maximum file descriptors if we can.
if [ "$cygwin" = "false" -a "$darwin" = "false" ] ; then
MAX_FD_LIMIT=`ulimit -H -n`
if [ $? -eq 0 ] ; then
if [ "$MAX_FD" = "maximum" -o "$MAX_FD" = "max" ] ; then
MAX_FD="$MAX_FD_LIMIT"
fi
ulimit -n $MAX_FD
if [ $? -ne 0 ] ; then
warn "Could not set maximum file descriptor limit: $MAX_FD"
fi
else
warn "Could not query maximum file descriptor limit: $MAX_FD_LIMIT"
fi
fi
# For Darwin, add options to specify how the application appears in the dock
if $darwin; then
GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\""
fi
# For Cygwin, switch paths to Windows format before running java
if $cygwin ; then
APP_HOME=`cygpath --path --mixed "$APP_HOME"`
CLASSPATH=`cygpath --path --mixed "$CLASSPATH"`
JAVACMD=`cygpath --unix "$JAVACMD"`
# We build the pattern for arguments to be converted via cygpath
ROOTDIRSRAW=`find -L / -maxdepth 1 -mindepth 1 -type d 2>/dev/null`
SEP=""
for dir in $ROOTDIRSRAW ; do
ROOTDIRS="$ROOTDIRS$SEP$dir"
SEP="|"
done
OURCYGPATTERN="(^($ROOTDIRS))"
# Add a user-defined pattern to the cygpath arguments
if [ "$GRADLE_CYGPATTERN" != "" ] ; then
OURCYGPATTERN="$OURCYGPATTERN|($GRADLE_CYGPATTERN)"
fi
# Now convert the arguments - kludge to limit ourselves to /bin/sh
i=0
for arg in "$@" ; do
CHECK=`echo "$arg"|egrep -c "$OURCYGPATTERN" -`
CHECK2=`echo "$arg"|egrep -c "^-"` ### Determine if an option
if [ $CHECK -ne 0 ] && [ $CHECK2 -eq 0 ] ; then ### Added a condition
eval `echo args$i`=`cygpath --path --ignore --mixed "$arg"`
else
eval `echo args$i`="\"$arg\""
fi
i=$((i+1))
done
case $i in
(0) set -- ;;
(1) set -- "$args0" ;;
(2) set -- "$args0" "$args1" ;;
(3) set -- "$args0" "$args1" "$args2" ;;
(4) set -- "$args0" "$args1" "$args2" "$args3" ;;
(5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;;
(6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;;
(7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;;
(8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;;
(9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;;
esac
fi
# Split up the JVM_OPTS And GRADLE_OPTS values into an array, following the shell quoting and substitution rules
function splitJvmOpts() {
JVM_OPTS=("$@")
}
eval splitJvmOpts $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS
JVM_OPTS[${#JVM_OPTS[*]}]="-Dorg.gradle.appname=$APP_BASE_NAME"
exec "$JAVACMD" "${JVM_OPTS[@]}" -classpath "$CLASSPATH" org.gradle.wrapper.GradleWrapperMain "$@"
================================================
FILE: gradlew.bat
================================================
@if "%DEBUG%" == "" @echo off
@rem ##########################################################################
@rem
@rem Gradle startup script for Windows
@rem
@rem ##########################################################################
@rem Set local scope for the variables with windows NT shell
if "%OS%"=="Windows_NT" setlocal
@rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
set DEFAULT_JVM_OPTS=
set DIRNAME=%~dp0
if "%DIRNAME%" == "" set DIRNAME=.
set APP_BASE_NAME=%~n0
set APP_HOME=%DIRNAME%
@rem Find java.exe
if defined JAVA_HOME goto findJavaFromJavaHome
set JAVA_EXE=java.exe
%JAVA_EXE% -version >NUL 2>&1
if "%ERRORLEVEL%" == "0" goto init
echo.
echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
echo.
echo Please set the JAVA_HOME variable in your environment to match the
echo location of your Java installation.
goto fail
:findJavaFromJavaHome
set JAVA_HOME=%JAVA_HOME:"=%
set JAVA_EXE=%JAVA_HOME%/bin/java.exe
if exist "%JAVA_EXE%" goto init
echo.
echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME%
echo.
echo Please set the JAVA_HOME variable in your environment to match the
echo location of your Java installation.
goto fail
:init
@rem Get command-line arguments, handling Windowz variants
if not "%OS%" == "Windows_NT" goto win9xME_args
if "%@eval[2+2]" == "4" goto 4NT_args
:win9xME_args
@rem Slurp the command line arguments.
set CMD_LINE_ARGS=
set _SKIP=2
:win9xME_args_slurp
if "x%~1" == "x" goto execute
set CMD_LINE_ARGS=%*
goto execute
:4NT_args
@rem Get arguments from the 4NT Shell from JP Software
set CMD_LINE_ARGS=%$
:execute
@rem Setup the command line
set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar
@rem Execute Gradle
"%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %CMD_LINE_ARGS%
:end
@rem End local scope for the variables with windows NT shell
if "%ERRORLEVEL%"=="0" goto mainEnd
:fail
rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of
rem the _cmd.exe /c_ return code!
if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1
exit /b 1
:mainEnd
if "%OS%"=="Windows_NT" endlocal
:omega
================================================
FILE: library/.gitignore
================================================
/build
================================================
FILE: library/build.gradle
================================================
apply plugin: 'com.android.library'
android {
compileSdkVersion 23
buildToolsVersion "23.0.2"
defaultConfig {
minSdkVersion 14
targetSdkVersion 23
versionCode 1
versionName "1.0"
}
buildTypes {
release {
minifyEnabled false
proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
}
}
}
dependencies {
compile fileTree(include: ['*.jar'], dir: 'libs')
testCompile 'junit:junit:4.12'
compile 'com.android.support:appcompat-v7:23.1.1'
compile 'com.android.support:recyclerview-v7:23.1.1'
}
================================================
FILE: library/proguard-rules.pro
================================================
# Add project specific ProGuard rules here.
# By default, the flags in this file are appended to flags specified
# in D:\world\android\sdk/tools/proguard/proguard-android.txt
# You can edit the include path and order by changing the proguardFiles
# directive in build.gradle.
#
# For more details, see
# http://developer.android.com/guide/developing/tools/proguard.html
# Add any project specific keep options here:
# If your project uses WebView with JS, uncomment the following
# and specify the fully qualified class name to the JavaScript interface
# class:
#-keepclassmembers class fqcn.of.javascript.interface.for.webview {
# public *;
#}
================================================
FILE: library/src/main/AndroidManifest.xml
================================================
================================================
FILE: library/src/main/java/com/cjj/loadmore/DefaultEmptyItem.java
================================================
package com.cjj.loadmore;
import android.text.TextUtils;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.ImageView;
import android.widget.TextView;
import com.cjj.R;
/**
* 默认的空数据视图
*
* @author cjj on 16/1/31.
*/
public class DefaultEmptyItem extends EmptyItem {
private TextView mEmptyTextView;
private ImageView mEmptyImageView;
@Override
public View onCreateView(ViewGroup parent) {
View view = LayoutInflater.from(parent.getContext()).inflate(R.layout.rv_with_footer_empty_layout, parent, false);
view.setLayoutParams(new ViewGroup.LayoutParams(parent.getMeasuredWidth(), parent.getMeasuredHeight()));
mEmptyTextView = (TextView) view.findViewById(R.id.rv_with_footer_empty_title);
mEmptyImageView = (ImageView) view.findViewById(R.id.rv_with_footer_empty_icon);
return view;
}
@Override
public void onBindData(View view) {
if (TextUtils.isEmpty(mEmptyText)) {
mEmptyTextView.setVisibility(View.GONE);
} else {
mEmptyTextView.setVisibility(View.VISIBLE);
mEmptyTextView.setText(mEmptyText);
}
if (mEmptyIconRes != -1) {
mEmptyImageView.setImageResource(mEmptyIconRes);
}
}
}
================================================
FILE: library/src/main/java/com/cjj/loadmore/DefaultFootItem.java
================================================
package com.cjj.loadmore;
import android.content.Context;
import android.text.TextUtils;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.ProgressBar;
import android.widget.TextView;
import com.cjj.R;
/**
* @author cjj
*/
public class DefaultFootItem extends FootItem {
private ProgressBar mProgressBar;
private TextView mLoadingText;
private TextView mEndTextView;
@Override
public View onCreateView(ViewGroup parent) {
LayoutInflater inflater = (LayoutInflater) parent.getContext().getSystemService(Context.LAYOUT_INFLATER_SERVICE);
View view = inflater.inflate(R.layout.rv_with_footer_loading, parent, false);
mProgressBar = (ProgressBar) view.findViewById(R.id.rv_with_footer_loading_progress);
mEndTextView = (TextView) view.findViewById(R.id.rv_with_footer_loading_end);
mLoadingText = (TextView) view.findViewById(R.id.rv_with_footer_loading_load);
return view;
}
@Override
public void onBindData(View view, int state) {
if (state == RecyclerViewWithFooter.STATE_LOADING) {
if (TextUtils.isEmpty(loadingText)) {
showProgressBar(view.getContext().getResources().getString(R.string.rv_with_footer_loading));
} else {
showProgressBar(loadingText);
}
} else if (state == RecyclerViewWithFooter.STATE_END) {
showEnd(endText);
} else if (state == RecyclerViewWithFooter.STATE_PULL_TO_LOAD) {
if (TextUtils.isEmpty(pullToLoadText)) {
showPullToLoad(view.getContext().getResources().getString(R.string.rv_with_footer_pull_load_more));
} else {
showPullToLoad(loadingText);
}
}
}
public void showPullToLoad(CharSequence message) {
showEnd(message);
}
public void showProgressBar(CharSequence load) {
mEndTextView.setVisibility(View.GONE);
mProgressBar.setVisibility(View.VISIBLE);
if (!TextUtils.isEmpty(load)) {
mLoadingText.setVisibility(View.VISIBLE);
mLoadingText.setText(load);
} else {
mLoadingText.setVisibility(View.GONE);
}
}
public void showEnd(CharSequence end) {
mEndTextView.setVisibility(View.VISIBLE);
mProgressBar.setVisibility(View.GONE);
mLoadingText.setVisibility(View.GONE);
if (!TextUtils.isEmpty(end)) {
mEndTextView.setText(end);
}
}
}
================================================
FILE: library/src/main/java/com/cjj/loadmore/EmptyItem.java
================================================
package com.cjj.loadmore;
import android.view.View;
import android.view.ViewGroup;
/**
* 没数据时候的默认View
*
* @author zzz40500 on 16/1/31.
*/
public abstract class EmptyItem {
CharSequence mEmptyText;
int mEmptyIconRes = -1;
abstract View onCreateView(ViewGroup parent);
abstract void onBindData(View view);
}
================================================
FILE: library/src/main/java/com/cjj/loadmore/FootItem.java
================================================
package com.cjj.loadmore;
import android.view.View;
import android.view.ViewGroup;
/**
* Footer item
*
* @author cjj on 2016/1/30.
*/
public abstract class FootItem {
public CharSequence loadingText;
public CharSequence endText;
public CharSequence pullToLoadText;
public abstract View onCreateView(ViewGroup parent);
public abstract void onBindData(View view, int state);
}
================================================
FILE: library/src/main/java/com/cjj/loadmore/OnLoadMoreListener.java
================================================
package com.cjj.loadmore;
/**
* Load more interface
*
* @author cjj
*/
public interface OnLoadMoreListener {
void onLoadMore();
}
================================================
FILE: library/src/main/java/com/cjj/loadmore/RecyclerViewUtils.java
================================================
package com.cjj.loadmore;
import android.support.v7.widget.GridLayoutManager;
import android.support.v7.widget.LinearLayoutManager;
import android.support.v7.widget.RecyclerView;
import android.support.v7.widget.StaggeredGridLayoutManager;
import android.util.Log;
/**
* 设置rv manager 类型
*
* @author cjj
*/
class RecyclerViewUtils {
private static final String TAG = "RecyclerViewUtils";
public static void setVerticalLinearLayout(RecyclerView rv) {
LinearLayoutManager layoutManager = new LinearLayoutManager(rv.getContext(), LinearLayoutManager.VERTICAL, false) {
@Override
public void onLayoutChildren(RecyclerView.Recycler recycler, RecyclerView.State state) {
try {
super.onLayoutChildren(recycler, state);
} catch (IndexOutOfBoundsException e) {
Log.e(TAG, "meet an IndexOutOfBoundsException in RecyclerView");
}
}
};
rv.setLayoutManager(layoutManager);
}
public static void setGridLayout(RecyclerView rv, int spanCount) {
GridLayoutManager gridLayoutManager = new GridLayoutManager(rv.getContext(), spanCount, GridLayoutManager.VERTICAL, false) {
@Override
public void onLayoutChildren(RecyclerView.Recycler recycler, RecyclerView.State state) {
try {
super.onLayoutChildren(recycler, state);
} catch (IndexOutOfBoundsException e) {
Log.e(TAG, "meet an IndexOutOfBoundsException in RecyclerView");
}
}
};
rv.setLayoutManager(gridLayoutManager);
}
public static void setStaggeredGridLayoutManager(RecyclerView rv, int spanCount) {
StaggeredGridLayoutManager staggeredGridLayoutManager = new StaggeredGridLayoutManager(spanCount, StaggeredGridLayoutManager.VERTICAL) {
@Override
public void onLayoutChildren(RecyclerView.Recycler recycler, RecyclerView.State state) {
try {
super.onLayoutChildren(recycler, state);
} catch (IndexOutOfBoundsException e) {
Log.e(RecyclerViewUtils.TAG, "meet an IndexOutOfBoundsException in RecyclerView");
}
}
};
rv.setLayoutManager(staggeredGridLayoutManager);
}
}
================================================
FILE: library/src/main/java/com/cjj/loadmore/RecyclerViewWithFooter.java
================================================
package com.cjj.loadmore;
import android.content.Context;
import android.support.annotation.DrawableRes;
import android.support.annotation.IntDef;
import android.support.v7.widget.GridLayoutManager;
import android.support.v7.widget.LinearLayoutManager;
import android.support.v7.widget.RecyclerView;
import android.support.v7.widget.StaggeredGridLayoutManager;
import android.util.AttributeSet;
import android.util.Log;
import android.view.View;
import android.view.ViewGroup;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
/**
* A {@link RecyclerView} with footer which enables load more.
*
* @author cjj
*/
public class RecyclerViewWithFooter extends RecyclerView {
private static final String TAG = "RecyclerViewWithFooter";
public static final int STATE_END = 0;
public static final int STATE_LOADING = 1;
public static final int STATE_EMPTY = 2;
public static final int STATE_NONE = 3;
public static final int STATE_PULL_TO_LOAD = 4;
private boolean mIsGetDataForNet = false;
@State
private int mState = STATE_NONE;
/**
* 默认的 FootItem;
*/
private FootItem mFootItem = new DefaultFootItem();
private EmptyItem mEmptyItem = new DefaultEmptyItem();
private AdapterDataObserver mAdapterDataObserver = new AdapterDataObserver() {
@Override
public void onChanged() {
super.onChanged();
reset();
}
private void reset() {
mIsGetDataForNet = false;
}
@Override
public void onItemRangeChanged(int positionStart, int itemCount) {
super.onItemRangeChanged(positionStart, itemCount);
reset();
}
@Override
public void onItemRangeChanged(int positionStart, int itemCount, Object payload) {
super.onItemRangeChanged(positionStart, itemCount, payload);
reset();
}
@Override
public void onItemRangeInserted(int positionStart, int itemCount) {
super.onItemRangeInserted(positionStart, itemCount);
reset();
}
@Override
public void onItemRangeRemoved(int positionStart, int itemCount) {
super.onItemRangeRemoved(positionStart, itemCount);
reset();
}
@Override
public void onItemRangeMoved(int fromPosition, int toPosition, int itemCount) {
super.onItemRangeMoved(fromPosition, toPosition, itemCount);
reset();
}
};
public RecyclerViewWithFooter(Context context) {
super(context);
init();
}
public RecyclerViewWithFooter(Context context, AttributeSet attrs) {
super(context, attrs);
init();
}
public RecyclerViewWithFooter(Context context, AttributeSet attrs, int defStyle) {
super(context, attrs, defStyle);
init();
}
private void init() {
setVerticalLinearLayout();
}
public void setVerticalLinearLayout() {
RecyclerViewUtils.setVerticalLinearLayout(this);
}
public void setGridLayout(int span) {
RecyclerViewUtils.setGridLayout(this, span);
}
public void setStaggeredGridLayoutManager(int spanCount) {
RecyclerViewUtils.setStaggeredGridLayoutManager(this, spanCount);
}
public void setOnLoadMoreListener(OnLoadMoreListener onLoadMoreListener) {
mState = STATE_PULL_TO_LOAD;
final OnLoadMoreListenerWrapper wrapper = new OnLoadMoreListenerWrapper(onLoadMoreListener);
addOnScrollListener(new RecyclerView.OnScrollListener() {
@Override
public void onScrollStateChanged(RecyclerView recyclerView, int newState) {
super.onScrollStateChanged(recyclerView, newState);
if (newState == RecyclerView.SCROLL_STATE_IDLE) {
RecyclerView.LayoutManager layoutManager = recyclerView.getLayoutManager();
if (layoutManager instanceof LinearLayoutManager) {
int lastVisiblePosition = ((LinearLayoutManager) layoutManager).findLastVisibleItemPosition();
if (lastVisiblePosition >= recyclerView.getAdapter().getItemCount() - 1) {
if (mState == STATE_PULL_TO_LOAD) {
setLoading();
}
wrapper.onLoadMore();
}
} else if (layoutManager instanceof StaggeredGridLayoutManager) {
StaggeredGridLayoutManager staggeredGridLayoutManager = (StaggeredGridLayoutManager) layoutManager;
int last[] = new int[staggeredGridLayoutManager.getSpanCount()];
staggeredGridLayoutManager.findLastVisibleItemPositions(last);
for (int aLast : last) {
Log.i(TAG, aLast + " " + recyclerView.getAdapter().getItemCount());
if (aLast >= recyclerView.getAdapter().getItemCount() - 1) {
if (mState == STATE_PULL_TO_LOAD) {
setLoading();
}
wrapper.onLoadMore();
}
}
}
}
}
});
}
@Override
public void setAdapter(Adapter adapter) {
LoadMoreAdapter loadMoreAdapter;
if (adapter instanceof LoadMoreAdapter) {
loadMoreAdapter = (LoadMoreAdapter) adapter;
loadMoreAdapter.registerAdapterDataObserver(mAdapterDataObserver);
super.setAdapter(adapter);
} else {
loadMoreAdapter = new LoadMoreAdapter(adapter);
loadMoreAdapter.registerAdapterDataObserver(mAdapterDataObserver);
super.setAdapter(loadMoreAdapter);
}
}
/**
* 设置loading提示字符串
*
* @param loadText 提示字符串
* @return {@link RecyclerViewWithFooter}
*/
public RecyclerViewWithFooter applyLoadingText(CharSequence loadText) {
mFootItem.loadingText = loadText;
return this;
}
public RecyclerViewWithFooter applyPullToLoadText(CharSequence pullToLoadText) {
mFootItem.pullToLoadText = pullToLoadText;
return this;
}
public RecyclerViewWithFooter applyEndText(CharSequence endText) {
mFootItem.endText = endText;
return this;
}
public RecyclerViewWithFooter applyEmptyText(CharSequence emptyText, @DrawableRes int drawableId) {
mEmptyItem.mEmptyIconRes = drawableId;
mEmptyItem.mEmptyText = emptyText;
return this;
}
public void setFootItem(FootItem footItem) {
if (mFootItem != null) {
if (footItem.endText == null) {
footItem.endText = mFootItem.endText;
}
if (footItem.loadingText == null) {
footItem.loadingText = mFootItem.loadingText;
}
if (footItem.pullToLoadText == null) {
footItem.pullToLoadText = mFootItem.pullToLoadText;
}
}
mFootItem = footItem;
}
public void setEmptyItem(EmptyItem emptyItem) {
if (mEmptyItem != null) {
if (emptyItem.mEmptyIconRes == -1) {
emptyItem.mEmptyIconRes = mEmptyItem.mEmptyIconRes;
}
if (emptyItem.mEmptyText == null) {
emptyItem.mEmptyText = mEmptyItem.mEmptyText;
}
}
mEmptyItem = emptyItem;
}
/**
* 切换为loading状态
*/
public void setLoading() {
if (getAdapter() != null) {
mState = STATE_LOADING;
mIsGetDataForNet = false;
getAdapter().notifyItemChanged(getAdapter().getItemCount() - 1);
}
}
/**
* 切换为没有更多数据状态
*
* @param end 提示字符串
*/
public void setEnd(CharSequence end) {
if (getAdapter() != null) {
mIsGetDataForNet = false;
mState = STATE_END;
mFootItem.endText = end;
getAdapter().notifyItemChanged(getAdapter().getItemCount() - 1);
}
}
/**
* 切换为没有更多数据状态
*/
public void setEnd() {
if (getAdapter() != null) {
mIsGetDataForNet = false;
mState = STATE_END;
getAdapter().notifyItemChanged(getAdapter().getItemCount() - 1);
}
}
/**
* 切换成无数据状态
*
* @param empty 无数据状态提示消息
* @param resId 无数据状态提示图标
*/
public void setEmpty(CharSequence empty, @DrawableRes int resId) {
if (getAdapter() != null) {
mState = STATE_EMPTY;
mEmptyItem.mEmptyText = empty;
mEmptyItem.mEmptyIconRes = resId;
if (isEmpty()) {
getAdapter().notifyDataSetChanged();
}
}
}
/**
* 切换成无数据状态
*/
public void setEmpty() {
if (getAdapter() != null) {
mState = STATE_EMPTY;
if (isEmpty()) {
getAdapter().notifyDataSetChanged();
}
}
}
/**
* 数据是否为空
*/
private boolean isEmpty() {
return (mState == STATE_NONE && getAdapter().getItemCount() == 0) ||
(mState != STATE_NONE && getAdapter().getItemCount() == 1);
}
public boolean isLoadMoreEnable() {
return mState != STATE_LOADING;
}
@IntDef({STATE_END, STATE_LOADING, STATE_EMPTY, STATE_NONE, STATE_PULL_TO_LOAD})
@Retention(RetentionPolicy.SOURCE)
public @interface State {
}
private class OnLoadMoreListenerWrapper implements OnLoadMoreListener {
private OnLoadMoreListener mOnLoadMoreListener;
public OnLoadMoreListenerWrapper(OnLoadMoreListener onLoadMoreListener) {
mOnLoadMoreListener = onLoadMoreListener;
}
@Override
public void onLoadMore() {
if (!mIsGetDataForNet && !isLoadMoreEnable()) {
mIsGetDataForNet = true;
mOnLoadMoreListener.onLoadMore();
}
}
}
public class LoadMoreAdapter extends RecyclerView.Adapter {
public static final int LOAD_MORE_VIEW_TYPE = -404;
public static final int EMPTY_VIEW_TYPE = -403;
public RecyclerView.Adapter mAdapter;
public LoadMoreAdapter(Adapter adapter) {
mAdapter = adapter;
}
@Override
public ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
if (viewType == LOAD_MORE_VIEW_TYPE) {
return new LoadMoreVH();
} else if (viewType == EMPTY_VIEW_TYPE) {
return new EmptyVH();
}
return mAdapter.onCreateViewHolder(parent, viewType);
}
@Override
public void registerAdapterDataObserver(AdapterDataObserver observer) {
super.registerAdapterDataObserver(observer);
mAdapter.registerAdapterDataObserver(observer);
}
@Override
public void unregisterAdapterDataObserver(AdapterDataObserver observer) {
super.unregisterAdapterDataObserver(observer);
mAdapter.unregisterAdapterDataObserver(observer);
}
@Override
public void onBindViewHolder(ViewHolder holder, int position) {
if (!isFootView(position)) {
mAdapter.onBindViewHolder(holder, position);
} else {
if (getLayoutManager() instanceof StaggeredGridLayoutManager) {
ViewGroup.LayoutParams layoutParams = holder.itemView.getLayoutParams();
if (layoutParams instanceof StaggeredGridLayoutManager.LayoutParams) {
((StaggeredGridLayoutManager.LayoutParams) layoutParams).setFullSpan(true);
}
} else if (getLayoutManager() instanceof GridLayoutManager) {
final GridLayoutManager layoutManager = (GridLayoutManager) getLayoutManager();
layoutManager.setSpanSizeLookup(new GridLayoutManager.SpanSizeLookup() {
@Override
public int getSpanSize(int position) {
int viewType = getAdapter().getItemViewType(position);
if (viewType < 0) {
return layoutManager.getSpanCount();
}
return 1;
}
});
}
if (holder instanceof VH) {
((VH) holder).onBindViewHolder();
}
}
}
private boolean isFootView(int position) {
return position == getItemCount() - 1 && mState != STATE_NONE;
}
@Override
public int getItemViewType(int position) {
if (!isFootView(position)) {
return mAdapter.getItemViewType(position);
} else {
if (mState == STATE_EMPTY && getItemCount() == 1) {
return EMPTY_VIEW_TYPE;
} else {
return LOAD_MORE_VIEW_TYPE;
}
}
}
@Override
public int getItemCount() {
if (mState == STATE_NONE) {
return mAdapter.getItemCount();
} else {
return mAdapter.getItemCount() + 1;
}
}
/**
* 加载更多的ViewHolder
*/
private class LoadMoreVH extends VH {
private View mItemView;
public LoadMoreVH() {
super(mFootItem.onCreateView(RecyclerViewWithFooter.this));
mItemView = itemView;
}
@Override
public void onBindViewHolder() {
super.onBindViewHolder();
if (mState == STATE_LOADING || mState == STATE_END || mState == STATE_PULL_TO_LOAD) {
mFootItem.onBindData(mItemView, mState);
}
}
}
/**
* 数据为空时的ViewHolder
*/
private class EmptyVH extends VH {
public EmptyVH() {
super(mEmptyItem.onCreateView(RecyclerViewWithFooter.this));
}
@Override
public void onBindViewHolder() {
super.onBindViewHolder();
mEmptyItem.onBindData(itemView);
}
}
class VH extends RecyclerView.ViewHolder {
public VH(View itemView) {
super(itemView);
}
public void onBindViewHolder() {
}
}
}
}
================================================
FILE: library/src/main/res/layout/rv_with_footer_empty_layout.xml
================================================
================================================
FILE: library/src/main/res/layout/rv_with_footer_loading.xml
================================================
================================================
FILE: library/src/main/res/values/rv_with_footer_strings.xml
================================================
上拉或点击加载更多
加载中…
没有数据啦
================================================
FILE: settings.gradle
================================================
include ':app', ':library'