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)这是默认的效果 ![](https://github.com/android-cjj/Android-RecyclerViewWithFooter/blob/master/img/cjj2.jpg) (2)这是Material风格的 ![](https://github.com/android-cjj/Android-RecyclerViewWithFooter/blob/master/img/cjj1.jpg) (3)你也可以自己定义 ![](https://github.com/android-cjj/Android-RecyclerViewWithFooter/blob/master/img/cjj.jpg) 差不多就这些了,想了解更多自己看源码。还有项目肯定存在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-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'