├── .gitignore ├── .idea ├── .name ├── compiler.xml ├── copyright │ └── profiles_settings.xml ├── encodings.xml ├── gradle.xml ├── misc.xml ├── modules.xml ├── runConfigurations.xml └── vcs.xml ├── README.md ├── app ├── .gitignore ├── build.gradle ├── proguard-rules.pro └── src │ └── main │ ├── AndroidManifest.xml │ ├── java │ └── deadline │ │ └── swiperecyclerview │ │ ├── MainActivity.java │ │ ├── SwipeRecyclerView.java │ │ └── footerView │ │ ├── BaseFooterView.java │ │ ├── FooterViewListener.java │ │ └── SimpleFooterView.java │ └── res │ ├── layout │ ├── demo_activity_main.xml │ ├── demo_layout_item.xml │ ├── layout_footer_view.xml │ └── layout_swipe_recyclerview.xml │ ├── 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 │ ├── values-w820dp │ └── dimens.xml │ └── values │ ├── colors.xml │ ├── dimens.xml │ ├── strings.xml │ └── styles.xml ├── build.gradle ├── gradle.properties ├── gradle └── wrapper │ ├── gradle-wrapper.jar │ └── gradle-wrapper.properties ├── gradlew ├── gradlew.bat ├── settings.gradle └── swipeRecyclerView.gif /.gitignore: -------------------------------------------------------------------------------- 1 | *.iml 2 | .gradle 3 | /local.properties 4 | /.idea/workspace.xml 5 | /.idea/libraries 6 | .DS_Store 7 | /build 8 | /captures 9 | -------------------------------------------------------------------------------- /.idea/.name: -------------------------------------------------------------------------------- 1 | SwipeRecyclerView -------------------------------------------------------------------------------- /.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 | -------------------------------------------------------------------------------- /.idea/copyright/profiles_settings.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | -------------------------------------------------------------------------------- /.idea/encodings.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | -------------------------------------------------------------------------------- /.idea/gradle.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 24 | 25 | -------------------------------------------------------------------------------- /.idea/misc.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 19 | 29 | 30 | 31 | 32 | 33 | 34 | 35 | 36 | 37 | 38 | 39 | 40 | 41 | 42 | 43 | 44 | 46 | 47 | 48 | 49 | 50 | 1.8 51 | 52 | 57 | 58 | 59 | 60 | 61 | 62 | -------------------------------------------------------------------------------- /.idea/modules.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | -------------------------------------------------------------------------------- /.idea/runConfigurations.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 11 | 12 | -------------------------------------------------------------------------------- /.idea/vcs.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | #SwipeRecyclerView 2 | SwipeRefreshLayout + RecyclerView 实现的下拉刷新,上拉加载更多 3 | 4 | #ScreenShot(效果图是循环播放的,所以感觉有点诡异) 5 | ![](https://github.com/niniloveyou/SwipeRecyclerView/blob/master/swipeRecyclerView.gif) 6 | 7 | #导航 8 | 9 | ###1. 支持自动下拉刷新 10 | 11 | 设置自动下拉刷新,切记要在recyclerView.setOnLoadListener()之后调用 12 | 因为在没有设置监听接口的情况下,setRefreshing(true),调用不到OnLoadListener 13 | mSwipeRecyclerView.setRefreshing(true); 14 | 15 | ###2. 支持emptyView 16 | 17 | mSwipeRecyclerView.setEmptyView(View emptyView); 18 | 19 | ###3. 支持禁止上拉加载更多/下拉刷新 20 | 21 | //禁止下拉刷新 22 | mSwipeRecyclerView.setRefreshEnable(false); 23 | 24 | //禁止加载更多 25 | mSwipeRecyclerView.setLoadMoreEnable(false); 26 | 27 | ###4.支持自定义footer view 28 | 29 | //设置footerView 30 | //但是自定义的footerView必须继承BaseFooterView 31 | mSwipeRecyclerView.setFooterView(new SimpleFooterView(this)); 32 | 33 | ###5.支持GridLayoutManager的SpanSizeLookup 34 | 35 | //由于SwipeRecyclerView中对GridLayoutManager的SpanSizeLookup做了处理,因此对于使用了 36 | //GridLayoutManager又要使用SpanSizeLookup的情况,可以这样使用! 37 | mSwipeRecyclerView.setSpanSizeLookup(new GridLayoutManager.SpanSizeLookup() { 38 | @Override 39 | public int getSpanSize(int position) { 40 | return 3; 41 | } 42 | }); 43 | 44 | ###6.关于footerView的分割线 获取childCount - 1 不包含footerView即可 45 | 46 | //设置去除footerView 的分割线 47 | mSwipeRecyclerView.getRecyclerView().addItemDecoration(new RecyclerView.ItemDecoration() { 48 | @Override 49 | public void onDraw(Canvas c, RecyclerView parent, RecyclerView.State state) { 50 | super.onDraw(c, parent, state); 51 | Paint paint = new Paint(Paint.ANTI_ALIAS_FLAG); 52 | paint.setColor(0xFFEECCCC); 53 | 54 | Rect rect = new Rect(); 55 | int left = parent.getPaddingLeft(); 56 | int right = parent.getWidth() - parent.getPaddingRight(); 57 | final int childCount = parent.getChildCount() - 1; 58 | for (int i = 0; i < childCount; i++) { 59 | final View child = parent.getChildAt(i); 60 | 61 | //获得child的布局信息 62 | final RecyclerView.LayoutParams params = (RecyclerView.LayoutParams) child.getLayoutParams(); 63 | final int top = child.getBottom() + params.bottomMargin; 64 | final int itemDividerHeight = 1;//px 65 | rect.set(left + 50, top, right - 50, top + itemDividerHeight); 66 | c.drawRect(rect, paint); 67 | } 68 | } 69 | }); 70 | 71 | ###7.需要对SwipeRefreshLayout或RecyclerView做其他的设置 72 | 73 | mSwipeRecyclerView.getSwipeRefreshLayout() 74 | mSwipeRecyclerView.getRecyclerView() 75 | 76 | 77 | 78 | ###8.可能存在的问题 79 | 80 |     由于Recycler.Adapter中关于数据集更新的方法全是final的,无法重写,并且自定义的DataObserver也没法实现的方法 如:notifyItemMoved方法 81 | 因此使用除SwipeRecyclerView中DataObserver的方法之外的更新数据集的方法,可能会有问题所以更新数据集建议采用DataObserver中有的方法。 82 | 83 | 84 | #Usage 85 | 86 | 由于并没有放到jCenter 87 | 88 |    所以如果需要使用:请自行把layout目录下layout_swipe_recyclerview, layout_footer_view copy跟正常一样控件使用,没有自定义属性 89 | 90 | 94 | 95 | mSwipeRecyclerView = (SwipeRecyclerView) findViewById(R.id.swipeRecyclerView); 96 | 97 | 98 | #... 99 |     如果你觉得还可以star一下吧! 100 | 101 | #about me 102 | 我的博客:http://www.jianshu.com/users/25e80ace21b8/latest_articles 103 | -------------------------------------------------------------------------------- /app/.gitignore: -------------------------------------------------------------------------------- 1 | /build 2 | -------------------------------------------------------------------------------- /app/build.gradle: -------------------------------------------------------------------------------- 1 | apply plugin: 'com.android.application' 2 | 3 | android { 4 | compileSdkVersion 24 5 | buildToolsVersion "24.0.0" 6 | 7 | defaultConfig { 8 | applicationId "deadline.swiperecyclerview" 9 | minSdkVersion 11 10 | targetSdkVersion 24 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(include: ['*.jar'], dir: 'libs') 24 | testCompile 'junit:junit:4.12' 25 | compile 'com.android.support:appcompat-v7:24.2.1' 26 | compile 'com.android.support:recyclerview-v7:24.2.1' 27 | } 28 | -------------------------------------------------------------------------------- /app/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 F:\AndroidSDK/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 | -------------------------------------------------------------------------------- /app/src/main/AndroidManifest.xml: -------------------------------------------------------------------------------- 1 | 2 | 4 | 5 | 11 | 12 | 13 | 14 | 15 | 16 | 17 | 18 | 19 | 20 | 21 | -------------------------------------------------------------------------------- /app/src/main/java/deadline/swiperecyclerview/MainActivity.java: -------------------------------------------------------------------------------- 1 | package deadline.swiperecyclerview; 2 | 3 | import android.os.Handler; 4 | import android.support.v7.app.AppCompatActivity; 5 | import android.os.Bundle; 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 | import java.util.ArrayList; 14 | import java.util.List; 15 | 16 | public class MainActivity extends AppCompatActivity { 17 | 18 | private SwipeRecyclerView recyclerView; 19 | private List data; 20 | private RecyclerViewAdapter adapter; 21 | private int pagerSize = 10; 22 | 23 | @Override 24 | protected void onCreate(Bundle savedInstanceState) { 25 | super.onCreate(savedInstanceState); 26 | setContentView(R.layout.demo_activity_main); 27 | 28 | recyclerView = (SwipeRecyclerView) findViewById(R.id.swipeRecyclerView); 29 | 30 | //set color 31 | recyclerView.getSwipeRefreshLayout() 32 | .setColorSchemeColors(getResources().getColor(R.color.colorPrimary)); 33 | 34 | //set layoutManager 35 | recyclerView.getRecyclerView().setLayoutManager(new LinearLayoutManager(this)); 36 | //recyclerView.getRecyclerView().setLayoutManager(new GridLayoutManager(this, 3)); 37 | //recyclerView.getRecyclerView().setLayoutManager(new StaggeredGridLayoutManager(2, StaggeredGridLayoutManager.VERTICAL)); 38 | 39 | //禁止下拉刷新 40 | // recyclerView.setRefreshEnable(false); 41 | 42 | //禁止加载更多 43 | //recyclerView.setLoadMoreEnable(false); 44 | 45 | //设置emptyView 46 | /*TextView textView = new TextView(this); 47 | textView.setText("empty view"); 48 | recyclerView.setEmptyView(textView);*/ 49 | 50 | //设置footerView 51 | //recyclerView.setFooterView(new SimpleFooterView(this)); 52 | 53 | //由于SwipeRecyclerView中对GridLayoutManager的SpanSizeLookup做了处理,因此对于使用了 54 | //GridLayoutManager又要使用SpanSizeLookup的情况,可以这样使用! 55 | /*recyclerView.setSpanSizeLookup(new GridLayoutManager.SpanSizeLookup() { 56 | @Override 57 | public int getSpanSize(int position) { 58 | return 3; 59 | } 60 | });*/ 61 | 62 | //设置去除footerView 的分割线 63 | /* recyclerView.getRecyclerView().addItemDecoration(new RecyclerView.ItemDecoration() { 64 | @Override 65 | public void onDraw(Canvas c, RecyclerView parent, RecyclerView.State state) { 66 | super.onDraw(c, parent, state); 67 | Paint paint = new Paint(Paint.ANTI_ALIAS_FLAG); 68 | paint.setColor(0xFFEECCCC); 69 | 70 | Rect rect = new Rect(); 71 | int left = parent.getPaddingLeft(); 72 | int right = parent.getWidth() - parent.getPaddingRight(); 73 | final int childCount = parent.getChildCount() - 1; 74 | for (int i = 0; i < childCount; i++) { 75 | final View child = parent.getChildAt(i); 76 | 77 | //获得child的布局信息 78 | final RecyclerView.LayoutParams params = (RecyclerView.LayoutParams) child.getLayoutParams(); 79 | final int top = child.getBottom() + params.bottomMargin; 80 | final int itemDividerHeight = 1;//px 81 | rect.set(left + 50, top, right - 50, top + itemDividerHeight); 82 | c.drawRect(rect, paint); 83 | } 84 | } 85 | });*/ 86 | 87 | //设置noMore 88 | // recyclerView.onNoMore("-- end --"); 89 | 90 | //设置网络处理 91 | //recyclerView.onNetChange(true); 92 | 93 | //设置错误信息 94 | //recyclerView.onError("error"); 95 | 96 | data = new ArrayList<>(); 97 | adapter = new RecyclerViewAdapter(); 98 | recyclerView.setAdapter(adapter); 99 | 100 | recyclerView.setOnLoadListener(new SwipeRecyclerView.OnLoadListener() { 101 | @Override 102 | public void onRefresh() { 103 | 104 | new Handler().postDelayed(new Runnable() { 105 | @Override 106 | public void run() { 107 | data.clear(); 108 | for (int i = 0; i < pagerSize; i++) { 109 | data.add(String.valueOf(i)); 110 | } 111 | 112 | recyclerView.complete(); 113 | adapter.notifyDataSetChanged(); 114 | 115 | } 116 | }, 1000); 117 | 118 | } 119 | 120 | @Override 121 | public void onLoadMore() { 122 | new Handler().postDelayed(new Runnable() { 123 | @Override 124 | public void run() { 125 | for (int i = 0; i < pagerSize; i++) { 126 | data.add(String.valueOf(i)); 127 | } 128 | 129 | if(data.size() > 20){ 130 | recyclerView.onNoMore("-- the end --"); 131 | }else { 132 | recyclerView.stopLoadingMore(); 133 | adapter.notifyDataSetChanged(); 134 | } 135 | } 136 | }, 1000); 137 | } 138 | }); 139 | 140 | //设置自动下拉刷新,切记要在recyclerView.setOnLoadListener()之后调用 141 | //因为在没有设置监听接口的情况下,setRefreshing(true),调用不到OnLoadListener 142 | recyclerView.setRefreshing(true); 143 | } 144 | 145 | 146 | private class RecyclerViewAdapter extends RecyclerView.Adapter { 147 | 148 | @Override 149 | public int getItemCount() { 150 | return data == null ? 0 : data.size(); 151 | } 152 | 153 | @Override 154 | public ItemViewHolder onCreateViewHolder(ViewGroup parent, int viewType) { 155 | View view = LayoutInflater.from(MainActivity.this).inflate(R.layout.demo_layout_item, parent, false); 156 | return new ItemViewHolder(view); 157 | } 158 | 159 | 160 | @Override 161 | public void onBindViewHolder(final ItemViewHolder holder, final int position) { 162 | 163 | holder.tv.setText("my position is" + data.get(position)); 164 | 165 | //for test item click listener 166 | holder.tv.setOnClickListener(new View.OnClickListener() { 167 | @Override 168 | public void onClick(View v) { 169 | Toast.makeText(MainActivity.this, "快来咬我啊,我是" + position + "号", Toast.LENGTH_SHORT).show(); 170 | } 171 | }); 172 | } 173 | } 174 | 175 | 176 | static class ItemViewHolder extends RecyclerView.ViewHolder { 177 | 178 | TextView tv; 179 | 180 | public ItemViewHolder(View view) { 181 | super(view); 182 | tv = (TextView) view.findViewById(R.id.tv); 183 | } 184 | } 185 | } -------------------------------------------------------------------------------- /app/src/main/java/deadline/swiperecyclerview/SwipeRecyclerView.java: -------------------------------------------------------------------------------- 1 | package deadline.swiperecyclerview; 2 | 3 | import android.content.Context; 4 | import android.support.annotation.Nullable; 5 | import android.support.v4.widget.SwipeRefreshLayout; 6 | import android.support.v7.widget.GridLayoutManager; 7 | import android.support.v7.widget.LinearLayoutManager; 8 | import android.support.v7.widget.RecyclerView; 9 | import android.support.v7.widget.RecyclerView.OnScrollListener; 10 | import android.support.v7.widget.RecyclerView.LayoutManager; 11 | import android.support.v7.widget.StaggeredGridLayoutManager; 12 | import android.support.v7.widget.GridLayoutManager.SpanSizeLookup; 13 | import android.util.AttributeSet; 14 | import android.view.Gravity; 15 | import android.view.LayoutInflater; 16 | import android.view.View; 17 | import android.view.ViewGroup; 18 | import android.widget.FrameLayout; 19 | 20 | import deadline.swiperecyclerview.footerView.BaseFooterView; 21 | import deadline.swiperecyclerview.footerView.FooterViewListener; 22 | import deadline.swiperecyclerview.footerView.SimpleFooterView; 23 | 24 | /** 25 | * @auther deadline 26 | * @time 2016/10/22 27 | * SwipeRefreshLayout + recyclerView 28 | */ 29 | public class SwipeRecyclerView extends FrameLayout 30 | implements SwipeRefreshLayout.OnRefreshListener{ 31 | 32 | private View mEmptyView; 33 | private BaseFooterView mFootView; 34 | private RecyclerView recyclerView; 35 | private SwipeRefreshLayout mRefreshLayout; 36 | 37 | private LayoutManager mLayoutManager; 38 | private OnLoadListener mListener; 39 | private SpanSizeLookup mSpanSizeLookup; 40 | private DataObserver mDataObserver; 41 | private WrapperAdapter mWrapperAdapter; 42 | 43 | private boolean isEmptyViewShowing; 44 | private boolean isLoadingMore; 45 | private boolean isLoadMoreEnable; 46 | private boolean isRefreshEnable; 47 | 48 | private int lastVisiablePosition = 0; 49 | 50 | public SwipeRecyclerView(Context context) { 51 | this(context, null); 52 | } 53 | 54 | public SwipeRecyclerView(Context context, @Nullable AttributeSet attrs) { 55 | this(context, attrs, 0); 56 | } 57 | 58 | public SwipeRecyclerView(Context context, @Nullable AttributeSet attrs, int defStyle) { 59 | super(context, attrs, defStyle); 60 | setupSwipeRecyclerView(); 61 | } 62 | 63 | private void setupSwipeRecyclerView() { 64 | 65 | isEmptyViewShowing = false; 66 | isRefreshEnable = true; 67 | isLoadingMore = false; 68 | isLoadMoreEnable = true; 69 | 70 | mFootView = new SimpleFooterView(getContext()); 71 | 72 | View view = LayoutInflater.from(getContext()).inflate(R.layout.layout_swipe_recyclerview, this); 73 | mRefreshLayout = (SwipeRefreshLayout) view.findViewById(R.id.SwipeRefreshLayout); 74 | recyclerView = (RecyclerView) view.findViewById(R.id.RecyclerView); 75 | mLayoutManager = recyclerView.getLayoutManager(); 76 | 77 | mRefreshLayout.setOnRefreshListener(this); 78 | recyclerView.setOnScrollListener(new OnScrollListener() { 79 | @Override 80 | public void onScrollStateChanged(RecyclerView recyclerView, int newState) { 81 | super.onScrollStateChanged(recyclerView, newState); 82 | } 83 | 84 | @Override 85 | public void onScrolled(RecyclerView recyclerView, int dx, int dy) { 86 | super.onScrolled(recyclerView, dx, dy); 87 | // do nothing if load more is not enable or refreshing or loading more 88 | if(!isLoadMoreEnable || isRefreshing() || isLoadingMore){ 89 | return; 90 | } 91 | 92 | //get the lastVisiablePosition 93 | mLayoutManager = recyclerView.getLayoutManager(); 94 | if(mLayoutManager instanceof LinearLayoutManager){ 95 | lastVisiablePosition = ((LinearLayoutManager)mLayoutManager).findLastVisibleItemPosition(); 96 | }else if(mLayoutManager instanceof GridLayoutManager){ 97 | lastVisiablePosition = ((GridLayoutManager)mLayoutManager).findLastCompletelyVisibleItemPosition(); 98 | }else if(mLayoutManager instanceof StaggeredGridLayoutManager){ 99 | int[] into = new int[((StaggeredGridLayoutManager) mLayoutManager).getSpanCount()]; 100 | ((StaggeredGridLayoutManager) mLayoutManager).findLastVisibleItemPositions(into); 101 | lastVisiablePosition = findMax(into); 102 | } 103 | 104 | int childCount = mWrapperAdapter == null ? 0 : mWrapperAdapter.getItemCount(); 105 | if(childCount > 1 && lastVisiablePosition == childCount - 1){ 106 | 107 | if(mListener != null){ 108 | isLoadingMore = true; 109 | mListener.onLoadMore(); 110 | } 111 | } 112 | } 113 | }); 114 | } 115 | 116 | private int findMax(int[] lastPositions) { 117 | int max = lastPositions[0]; 118 | for (int value : lastPositions) { 119 | if (value > max) { 120 | max = value; 121 | } 122 | } 123 | return max; 124 | } 125 | 126 | /** 127 | * set is enable pull to refresh 128 | * @param refreshEnable 129 | */ 130 | public void setRefreshEnable(boolean refreshEnable){ 131 | isRefreshEnable = refreshEnable; 132 | mRefreshLayout.setEnabled(isRefreshEnable); 133 | } 134 | 135 | public boolean getRefreshEnable(){ 136 | return isRefreshEnable; 137 | } 138 | 139 | /** 140 | * set is loading more enable 141 | * @param loadMoreEnable 142 | * if true when recyclerView scroll to bottom load more action will be trigger 143 | */ 144 | public void setLoadMoreEnable(boolean loadMoreEnable) { 145 | if(!loadMoreEnable){ 146 | stopLoadingMore(); 147 | } 148 | isLoadMoreEnable = loadMoreEnable; 149 | } 150 | 151 | public boolean getLoadMoreEnable(){ 152 | return isLoadMoreEnable; 153 | } 154 | 155 | /** 156 | * get is refreshing 157 | * @return 158 | */ 159 | public boolean isRefreshing(){ 160 | return mRefreshLayout.isRefreshing(); 161 | } 162 | 163 | /** 164 | * get is loading more 165 | * @return 166 | */ 167 | public boolean isLoadingMore(){ 168 | return isLoadingMore; 169 | } 170 | 171 | /** 172 | * is empty view showing 173 | * @return 174 | */ 175 | public boolean isEmptyViewShowing(){ 176 | return isEmptyViewShowing; 177 | } 178 | 179 | /** 180 | * you may need set some other attributes of swipeRefreshLayout 181 | * @return 182 | * swipeRefreshLayout 183 | */ 184 | public SwipeRefreshLayout getSwipeRefreshLayout(){ 185 | return mRefreshLayout; 186 | } 187 | 188 | /** 189 | * you may need set some other attributes of RecyclerView 190 | * @return 191 | * RecyclerView 192 | */ 193 | public RecyclerView getRecyclerView(){ 194 | return recyclerView; 195 | } 196 | 197 | /** 198 | * set load more listener 199 | * @param listener 200 | */ 201 | public void setOnLoadListener(OnLoadListener listener){ 202 | mListener = listener; 203 | } 204 | 205 | /** 206 | * support for GridLayoutManager 207 | * @param spanSizeLookup 208 | */ 209 | public void setSpanSizeLookup(SpanSizeLookup spanSizeLookup){ 210 | this.mSpanSizeLookup = spanSizeLookup; 211 | } 212 | 213 | /** 214 | * set the footer view 215 | * @param footerView 216 | * the view to be showing when pull up 217 | */ 218 | public void setFooterView(BaseFooterView footerView){ 219 | if(footerView != null) { 220 | this.mFootView = footerView; 221 | } 222 | } 223 | 224 | /** 225 | * set a empty view like listview 226 | * @param emptyView 227 | * the view to be showing when the data set size is zero 228 | */ 229 | public void setEmptyView(View emptyView){ 230 | if(mEmptyView != null){ 231 | removeView(mEmptyView); 232 | } 233 | this.mEmptyView = emptyView; 234 | 235 | if(mDataObserver != null) { 236 | mDataObserver.onChanged(); 237 | } 238 | } 239 | 240 | /** 241 | * set adapter to recyclerView 242 | * @param adapter 243 | */ 244 | public void setAdapter(RecyclerView.Adapter adapter){ 245 | if(adapter != null) { 246 | if(mDataObserver == null){ 247 | mDataObserver = new DataObserver(); 248 | } 249 | mWrapperAdapter = new WrapperAdapter(adapter); 250 | recyclerView.setAdapter(mWrapperAdapter); 251 | adapter.registerAdapterDataObserver(mDataObserver); 252 | mDataObserver.onChanged(); 253 | } 254 | } 255 | 256 | /** 257 | * refresh or load more completed 258 | */ 259 | public void complete(){ 260 | mRefreshLayout.setRefreshing(false); 261 | stopLoadingMore(); 262 | } 263 | 264 | /** 265 | * set refreshing 266 | * if you want load data when first in, you can setRefreshing(true) 267 | * after {@link #setOnLoadListener(OnLoadListener)} 268 | * @param refreshing 269 | */ 270 | public void setRefreshing(boolean refreshing){ 271 | mRefreshLayout.setRefreshing(refreshing); 272 | if(refreshing && !isLoadingMore && mListener != null){ 273 | mListener.onRefresh(); 274 | } 275 | } 276 | 277 | /** 278 | * stop loading more without animation 279 | */ 280 | public void stopLoadingMore(){ 281 | isLoadingMore = false; 282 | if(mWrapperAdapter != null) { 283 | mWrapperAdapter.notifyItemRemoved(mWrapperAdapter.getItemCount()); 284 | } 285 | } 286 | 287 | /** 288 | * call method {@link OnLoadListener#onRefresh()} 289 | */ 290 | @Override 291 | public void onRefresh() { 292 | if(mListener != null){ 293 | 294 | //reset footer view status loading 295 | if(mFootView != null){ 296 | mFootView.onLoadingMore(); 297 | } 298 | mListener.onRefresh(); 299 | } 300 | } 301 | 302 | /** 303 | * {@link FooterViewListener#onNetChange(boolean isAvailable)} 304 | * call when network is available or not available 305 | */ 306 | public void onNetChange(boolean isAvailable) { 307 | if(mFootView != null){ 308 | mFootView.onNetChange(isAvailable); 309 | } 310 | } 311 | 312 | /** 313 | * {@link FooterViewListener#onLoadingMore()} 314 | * call when you need change footer view to loading status 315 | */ 316 | public void onLoadingMore() { 317 | if(mFootView != null){ 318 | mFootView.onLoadingMore(); 319 | } 320 | } 321 | 322 | /** 323 | * {@link FooterViewListener#onNoMore(CharSequence message)} 324 | * call when no more data add to list 325 | */ 326 | public void onNoMore(CharSequence message) { 327 | if(mFootView != null){ 328 | mFootView.onNoMore(message); 329 | } 330 | } 331 | 332 | /** 333 | * {@link FooterViewListener#onError(CharSequence message)} 334 | * call when you need show error message 335 | */ 336 | public void onError(CharSequence message) { 337 | if(mFootView != null){ 338 | mFootView.onError(message); 339 | } 340 | } 341 | 342 | 343 | private class WrapperAdapter extends RecyclerView.Adapter{ 344 | 345 | public static final int TYPE_FOOTER = 0x100; 346 | 347 | RecyclerView.Adapter mInnerAdapter; 348 | 349 | public WrapperAdapter(RecyclerView.Adapter adapter){ 350 | this.mInnerAdapter = adapter; 351 | } 352 | 353 | public boolean isLoadMoreItem(int position){ 354 | return isLoadMoreEnable && position == getItemCount() - 1; 355 | } 356 | 357 | @Override 358 | public RecyclerView.ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) { 359 | if(TYPE_FOOTER == viewType){ 360 | return new FooterViewHolder(mFootView); 361 | } 362 | return mInnerAdapter.onCreateViewHolder(parent, viewType); 363 | } 364 | 365 | @Override 366 | public void onBindViewHolder(RecyclerView.ViewHolder holder, int position) { 367 | if(isLoadMoreItem(position)){ 368 | return; 369 | } 370 | mInnerAdapter.onBindViewHolder(holder, position); 371 | } 372 | 373 | 374 | @Override 375 | public int getItemViewType(int position) { 376 | if(isLoadMoreItem(position)){ 377 | return TYPE_FOOTER; 378 | }else{ 379 | return mInnerAdapter.getItemViewType(position); 380 | } 381 | } 382 | 383 | @Override 384 | public int getItemCount() { 385 | int count = mInnerAdapter == null ? 0 : mInnerAdapter.getItemCount(); 386 | 387 | //without loadingMore when adapter size is zero 388 | if(count == 0){ 389 | return 0; 390 | } 391 | return isLoadMoreEnable ? count + 1 : count; 392 | } 393 | 394 | @Override 395 | public long getItemId(int position) { 396 | return mInnerAdapter.getItemId(position); 397 | } 398 | 399 | @Override 400 | public void onViewAttachedToWindow(RecyclerView.ViewHolder holder) { 401 | ViewGroup.LayoutParams lp = holder.itemView.getLayoutParams(); 402 | if (lp != null 403 | && lp instanceof StaggeredGridLayoutManager.LayoutParams 404 | && isLoadMoreItem(holder.getLayoutPosition())) 405 | { 406 | StaggeredGridLayoutManager.LayoutParams p = (StaggeredGridLayoutManager.LayoutParams) lp; 407 | p.setFullSpan(true); 408 | } 409 | mInnerAdapter.onViewAttachedToWindow(holder); 410 | } 411 | 412 | @Override 413 | public void onViewDetachedFromWindow(RecyclerView.ViewHolder holder) { 414 | mInnerAdapter.onViewDetachedFromWindow(holder); 415 | } 416 | 417 | @Override 418 | public void onAttachedToRecyclerView(RecyclerView recyclerView) { 419 | RecyclerView.LayoutManager manager = recyclerView.getLayoutManager(); 420 | if (manager instanceof GridLayoutManager) { 421 | final GridLayoutManager gridManager = ((GridLayoutManager) manager); 422 | gridManager.setSpanSizeLookup(new GridLayoutManager.SpanSizeLookup() { 423 | @Override 424 | public int getSpanSize(int position) { 425 | boolean isLoadMore = isLoadMoreItem(position); 426 | if(mSpanSizeLookup != null && !isLoadMore){ 427 | return mSpanSizeLookup.getSpanSize(position); 428 | } 429 | return isLoadMore ? gridManager.getSpanCount() : 1; 430 | } 431 | }); 432 | } 433 | mInnerAdapter.onAttachedToRecyclerView(recyclerView); 434 | } 435 | 436 | @Override 437 | public void onDetachedFromRecyclerView(RecyclerView recyclerView) { 438 | mInnerAdapter.onDetachedFromRecyclerView(recyclerView); 439 | } 440 | 441 | @Override 442 | public boolean onFailedToRecycleView(RecyclerView.ViewHolder holder) { 443 | return mInnerAdapter.onFailedToRecycleView(holder); 444 | } 445 | 446 | @Override 447 | public void registerAdapterDataObserver(RecyclerView.AdapterDataObserver observer) { 448 | mInnerAdapter.registerAdapterDataObserver(observer); 449 | } 450 | 451 | @Override 452 | public void unregisterAdapterDataObserver(RecyclerView.AdapterDataObserver observer) { 453 | mInnerAdapter.unregisterAdapterDataObserver(observer); 454 | } 455 | 456 | @Override 457 | public void onViewRecycled(RecyclerView.ViewHolder holder) { 458 | mInnerAdapter.onViewRecycled(holder); 459 | } 460 | } 461 | 462 | /** 463 | * ViewHolder of footerView 464 | */ 465 | private class FooterViewHolder extends RecyclerView.ViewHolder { 466 | public FooterViewHolder(View itemView) { 467 | super(itemView); 468 | } 469 | } 470 | 471 | /** 472 | * a inner class used to monitor the dataSet change 473 | *

474 | * because wrapperAdapter do not know when wrapperAdapter.mInnerAdapter 475 | *

476 | * dataSet changed, these method are final 477 | */ 478 | class DataObserver extends RecyclerView.AdapterDataObserver{ 479 | 480 | @Override 481 | public void onChanged() { 482 | super.onChanged(); 483 | RecyclerView.Adapter adapter = recyclerView.getAdapter(); 484 | if(adapter != null && mEmptyView != null){ 485 | 486 | int count = 0; 487 | if(isLoadMoreEnable && adapter.getItemCount() != 0){ 488 | count ++; 489 | } 490 | if(adapter.getItemCount() == count){ 491 | isEmptyViewShowing = true; 492 | if(mEmptyView.getParent() == null){ 493 | FrameLayout.LayoutParams params = new LayoutParams( 494 | ViewGroup.LayoutParams.WRAP_CONTENT, ViewGroup.LayoutParams.WRAP_CONTENT); 495 | params.gravity = Gravity.CENTER; 496 | 497 | addView(mEmptyView, params); 498 | } 499 | 500 | recyclerView.setVisibility(GONE); 501 | mEmptyView.setVisibility(VISIBLE); 502 | }else{ 503 | isEmptyViewShowing = false; 504 | mEmptyView.setVisibility(GONE); 505 | recyclerView.setVisibility(VISIBLE); 506 | } 507 | } 508 | mWrapperAdapter.notifyDataSetChanged(); 509 | } 510 | 511 | @Override 512 | public void onItemRangeChanged(int positionStart, int itemCount) { 513 | super.onItemRangeChanged(positionStart, itemCount); 514 | mWrapperAdapter.notifyItemRangeChanged(positionStart, itemCount); 515 | } 516 | 517 | @Override 518 | public void onItemRangeChanged(int positionStart, int itemCount, Object payload) { 519 | super.onItemRangeChanged(positionStart, itemCount, payload); 520 | mWrapperAdapter.notifyItemRangeChanged(positionStart, itemCount, payload); 521 | } 522 | 523 | @Override 524 | public void onItemRangeInserted(int positionStart, int itemCount) { 525 | super.onItemRangeInserted(positionStart, itemCount); 526 | mWrapperAdapter.notifyItemRangeInserted(positionStart, itemCount); 527 | } 528 | 529 | @Override 530 | public void onItemRangeMoved(int fromPosition, int toPosition, int itemCount) { 531 | super.onItemRangeMoved(fromPosition, toPosition, itemCount); 532 | mWrapperAdapter.notifyItemRangeRemoved(fromPosition, itemCount); 533 | } 534 | 535 | @Override 536 | public void onItemRangeRemoved(int positionStart, int itemCount) { 537 | super.onItemRangeRemoved(positionStart, itemCount); 538 | mWrapperAdapter.notifyItemRangeRemoved(positionStart, itemCount); 539 | } 540 | 541 | } 542 | 543 | public interface OnLoadListener { 544 | 545 | void onRefresh(); 546 | 547 | void onLoadMore(); 548 | } 549 | } 550 | -------------------------------------------------------------------------------- /app/src/main/java/deadline/swiperecyclerview/footerView/BaseFooterView.java: -------------------------------------------------------------------------------- 1 | package deadline.swiperecyclerview.footerView; 2 | 3 | import android.content.Context; 4 | import android.util.AttributeSet; 5 | import android.widget.FrameLayout; 6 | 7 | /** 8 | * @auther deadline 9 | * @time 2016/10/22 10 | */ 11 | public abstract class BaseFooterView extends FrameLayout implements FooterViewListener{ 12 | 13 | public BaseFooterView(Context context) { 14 | super(context); 15 | } 16 | 17 | public BaseFooterView(Context context, AttributeSet attrs) { 18 | super(context, attrs); 19 | } 20 | 21 | public BaseFooterView(Context context, AttributeSet attrs, int defStyleAttr) { 22 | super(context, attrs, defStyleAttr); 23 | } 24 | } 25 | -------------------------------------------------------------------------------- /app/src/main/java/deadline/swiperecyclerview/footerView/FooterViewListener.java: -------------------------------------------------------------------------------- 1 | package deadline.swiperecyclerview.footerView; 2 | 3 | /** 4 | * @author deadline 5 | * @time 2016/10/22 6 | */ 7 | public interface FooterViewListener { 8 | 9 | /** 10 | * 网络不好的时候想要展示的UI 11 | */ 12 | void onNetChange(boolean isAvailable); 13 | 14 | /** 15 | * 正常的loading的View 16 | */ 17 | void onLoadingMore(); 18 | 19 | /** 20 | * 没有更多数据 21 | */ 22 | void onNoMore(CharSequence message); 23 | 24 | /** 25 | * 错误时展示的View 26 | */ 27 | void onError(CharSequence message); 28 | } 29 | -------------------------------------------------------------------------------- /app/src/main/java/deadline/swiperecyclerview/footerView/SimpleFooterView.java: -------------------------------------------------------------------------------- 1 | package deadline.swiperecyclerview.footerView; 2 | 3 | import android.content.Context; 4 | import android.util.AttributeSet; 5 | import android.view.LayoutInflater; 6 | import android.view.View; 7 | import android.view.ViewGroup; 8 | import android.widget.FrameLayout; 9 | import android.widget.ProgressBar; 10 | import android.widget.TextView; 11 | 12 | import deadline.swiperecyclerview.R; 13 | 14 | /** 15 | * @author deadline 16 | * @time 2016/10/22 17 | */ 18 | public class SimpleFooterView extends BaseFooterView{ 19 | 20 | private TextView mText; 21 | 22 | private ProgressBar progressBar; 23 | 24 | public SimpleFooterView(Context context) { 25 | this(context, null); 26 | } 27 | 28 | public SimpleFooterView(Context context, AttributeSet attrs) { 29 | this(context, attrs, 0); 30 | } 31 | 32 | public SimpleFooterView(Context context, AttributeSet attrs, int defStyleAttr) { 33 | super(context, attrs, defStyleAttr); 34 | 35 | setLayoutParams(new FrameLayout.LayoutParams( 36 | ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.WRAP_CONTENT)); 37 | View view = LayoutInflater.from(getContext()).inflate(R.layout.layout_footer_view, this); 38 | progressBar = (ProgressBar) view.findViewById(R.id.footer_view_progressbar); 39 | mText = (TextView) view.findViewById(R.id.footer_view_tv); 40 | } 41 | 42 | 43 | 44 | @Override 45 | public void onLoadingMore() { 46 | progressBar.setVisibility(VISIBLE); 47 | mText.setVisibility(GONE); 48 | } 49 | 50 | public void showText(){ 51 | progressBar.setVisibility(GONE); 52 | mText.setVisibility(VISIBLE); 53 | } 54 | 55 | /**************文字自行修改或根据传入的参数动态修改****************/ 56 | 57 | @Override 58 | public void onNoMore(CharSequence message) { 59 | showText(); 60 | mText.setText("-- the end --"); 61 | } 62 | 63 | @Override 64 | public void onError(CharSequence message) { 65 | showText(); 66 | mText.setText("啊哦,好像哪里不对劲!"); 67 | } 68 | 69 | @Override 70 | public void onNetChange(boolean isAvailable) { 71 | showText(); 72 | mText.setText("网络连接不通畅!"); 73 | } 74 | } 75 | -------------------------------------------------------------------------------- /app/src/main/res/layout/demo_activity_main.xml: -------------------------------------------------------------------------------- 1 | 2 | 8 | 9 | 13 | 14 | 15 | -------------------------------------------------------------------------------- /app/src/main/res/layout/demo_layout_item.xml: -------------------------------------------------------------------------------- 1 | 2 | 6 | 7 | 14 | -------------------------------------------------------------------------------- /app/src/main/res/layout/layout_footer_view.xml: -------------------------------------------------------------------------------- 1 | 2 | 8 | 9 | 14 | 15 | 16 | 21 | -------------------------------------------------------------------------------- /app/src/main/res/layout/layout_swipe_recyclerview.xml: -------------------------------------------------------------------------------- 1 | 2 | 8 | 9 | 15 | 16 | -------------------------------------------------------------------------------- /app/src/main/res/mipmap-hdpi/ic_launcher.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/niniloveyou/SwipeRecyclerView/8a9ebbdf196cd9619cb79a809c25fcdd4324438f/app/src/main/res/mipmap-hdpi/ic_launcher.png -------------------------------------------------------------------------------- /app/src/main/res/mipmap-mdpi/ic_launcher.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/niniloveyou/SwipeRecyclerView/8a9ebbdf196cd9619cb79a809c25fcdd4324438f/app/src/main/res/mipmap-mdpi/ic_launcher.png -------------------------------------------------------------------------------- /app/src/main/res/mipmap-xhdpi/ic_launcher.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/niniloveyou/SwipeRecyclerView/8a9ebbdf196cd9619cb79a809c25fcdd4324438f/app/src/main/res/mipmap-xhdpi/ic_launcher.png -------------------------------------------------------------------------------- /app/src/main/res/mipmap-xxhdpi/ic_launcher.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/niniloveyou/SwipeRecyclerView/8a9ebbdf196cd9619cb79a809c25fcdd4324438f/app/src/main/res/mipmap-xxhdpi/ic_launcher.png -------------------------------------------------------------------------------- /app/src/main/res/mipmap-xxxhdpi/ic_launcher.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/niniloveyou/SwipeRecyclerView/8a9ebbdf196cd9619cb79a809c25fcdd4324438f/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png -------------------------------------------------------------------------------- /app/src/main/res/values-w820dp/dimens.xml: -------------------------------------------------------------------------------- 1 | 2 | 5 | 64dp 6 | 7 | -------------------------------------------------------------------------------- /app/src/main/res/values/colors.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | #3F51B5 4 | #303F9F 5 | #FF4081 6 | 7 | -------------------------------------------------------------------------------- /app/src/main/res/values/dimens.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 16dp 4 | 16dp 5 | 6 | -------------------------------------------------------------------------------- /app/src/main/res/values/strings.xml: -------------------------------------------------------------------------------- 1 | 2 | SwipeRecyclerView 3 | 4 | -------------------------------------------------------------------------------- /app/src/main/res/values/styles.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 10 | 11 | 12 | -------------------------------------------------------------------------------- /build.gradle: -------------------------------------------------------------------------------- 1 | // Top-level build file where you can add configuration options common to all sub-projects/modules. 2 | 3 | buildscript { 4 | repositories { 5 | jcenter() 6 | } 7 | dependencies { 8 | classpath 'com.android.tools.build:gradle:1.3.0' 9 | 10 | // NOTE: Do not place your application dependencies here; they belong 11 | // in the individual module build.gradle files 12 | } 13 | } 14 | 15 | allprojects { 16 | repositories { 17 | jcenter() 18 | } 19 | } 20 | 21 | task clean(type: Delete) { 22 | delete rootProject.buildDir 23 | } 24 | -------------------------------------------------------------------------------- /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 -------------------------------------------------------------------------------- /gradle/wrapper/gradle-wrapper.jar: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/niniloveyou/SwipeRecyclerView/8a9ebbdf196cd9619cb79a809c25fcdd4324438f/gradle/wrapper/gradle-wrapper.jar -------------------------------------------------------------------------------- /gradle/wrapper/gradle-wrapper.properties: -------------------------------------------------------------------------------- 1 | #Wed Oct 21 11:34:03 PDT 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.8-all.zip 7 | -------------------------------------------------------------------------------- /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 | # Attempt to set APP_HOME 46 | # Resolve links: $0 may be a link 47 | PRG="$0" 48 | # Need this for relative symlinks. 49 | while [ -h "$PRG" ] ; do 50 | ls=`ls -ld "$PRG"` 51 | link=`expr "$ls" : '.*-> \(.*\)$'` 52 | if expr "$link" : '/.*' > /dev/null; then 53 | PRG="$link" 54 | else 55 | PRG=`dirname "$PRG"`"/$link" 56 | fi 57 | done 58 | SAVED="`pwd`" 59 | cd "`dirname \"$PRG\"`/" >/dev/null 60 | APP_HOME="`pwd -P`" 61 | cd "$SAVED" >/dev/null 62 | 63 | CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar 64 | 65 | # Determine the Java command to use to start the JVM. 66 | if [ -n "$JAVA_HOME" ] ; then 67 | if [ -x "$JAVA_HOME/jre/sh/java" ] ; then 68 | # IBM's JDK on AIX uses strange locations for the executables 69 | JAVACMD="$JAVA_HOME/jre/sh/java" 70 | else 71 | JAVACMD="$JAVA_HOME/bin/java" 72 | fi 73 | if [ ! -x "$JAVACMD" ] ; then 74 | die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME 75 | 76 | Please set the JAVA_HOME variable in your environment to match the 77 | location of your Java installation." 78 | fi 79 | else 80 | JAVACMD="java" 81 | which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. 82 | 83 | Please set the JAVA_HOME variable in your environment to match the 84 | location of your Java installation." 85 | fi 86 | 87 | # Increase the maximum file descriptors if we can. 88 | if [ "$cygwin" = "false" -a "$darwin" = "false" ] ; then 89 | MAX_FD_LIMIT=`ulimit -H -n` 90 | if [ $? -eq 0 ] ; then 91 | if [ "$MAX_FD" = "maximum" -o "$MAX_FD" = "max" ] ; then 92 | MAX_FD="$MAX_FD_LIMIT" 93 | fi 94 | ulimit -n $MAX_FD 95 | if [ $? -ne 0 ] ; then 96 | warn "Could not set maximum file descriptor limit: $MAX_FD" 97 | fi 98 | else 99 | warn "Could not query maximum file descriptor limit: $MAX_FD_LIMIT" 100 | fi 101 | fi 102 | 103 | # For Darwin, add options to specify how the application appears in the dock 104 | if $darwin; then 105 | GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\"" 106 | fi 107 | 108 | # For Cygwin, switch paths to Windows format before running java 109 | if $cygwin ; then 110 | APP_HOME=`cygpath --path --mixed "$APP_HOME"` 111 | CLASSPATH=`cygpath --path --mixed "$CLASSPATH"` 112 | JAVACMD=`cygpath --unix "$JAVACMD"` 113 | 114 | # We build the pattern for arguments to be converted via cygpath 115 | ROOTDIRSRAW=`find -L / -maxdepth 1 -mindepth 1 -type d 2>/dev/null` 116 | SEP="" 117 | for dir in $ROOTDIRSRAW ; do 118 | ROOTDIRS="$ROOTDIRS$SEP$dir" 119 | SEP="|" 120 | done 121 | OURCYGPATTERN="(^($ROOTDIRS))" 122 | # Add a user-defined pattern to the cygpath arguments 123 | if [ "$GRADLE_CYGPATTERN" != "" ] ; then 124 | OURCYGPATTERN="$OURCYGPATTERN|($GRADLE_CYGPATTERN)" 125 | fi 126 | # Now convert the arguments - kludge to limit ourselves to /bin/sh 127 | i=0 128 | for arg in "$@" ; do 129 | CHECK=`echo "$arg"|egrep -c "$OURCYGPATTERN" -` 130 | CHECK2=`echo "$arg"|egrep -c "^-"` ### Determine if an option 131 | 132 | if [ $CHECK -ne 0 ] && [ $CHECK2 -eq 0 ] ; then ### Added a condition 133 | eval `echo args$i`=`cygpath --path --ignore --mixed "$arg"` 134 | else 135 | eval `echo args$i`="\"$arg\"" 136 | fi 137 | i=$((i+1)) 138 | done 139 | case $i in 140 | (0) set -- ;; 141 | (1) set -- "$args0" ;; 142 | (2) set -- "$args0" "$args1" ;; 143 | (3) set -- "$args0" "$args1" "$args2" ;; 144 | (4) set -- "$args0" "$args1" "$args2" "$args3" ;; 145 | (5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;; 146 | (6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;; 147 | (7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;; 148 | (8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;; 149 | (9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;; 150 | esac 151 | fi 152 | 153 | # Split up the JVM_OPTS And GRADLE_OPTS values into an array, following the shell quoting and substitution rules 154 | function splitJvmOpts() { 155 | JVM_OPTS=("$@") 156 | } 157 | eval splitJvmOpts $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS 158 | JVM_OPTS[${#JVM_OPTS[*]}]="-Dorg.gradle.appname=$APP_BASE_NAME" 159 | 160 | exec "$JAVACMD" "${JVM_OPTS[@]}" -classpath "$CLASSPATH" org.gradle.wrapper.GradleWrapperMain "$@" 161 | -------------------------------------------------------------------------------- /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 | -------------------------------------------------------------------------------- /settings.gradle: -------------------------------------------------------------------------------- 1 | include ':app' 2 | -------------------------------------------------------------------------------- /swipeRecyclerView.gif: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/niniloveyou/SwipeRecyclerView/8a9ebbdf196cd9619cb79a809c25fcdd4324438f/swipeRecyclerView.gif --------------------------------------------------------------------------------