Android Custom View-SlideListView

Android Custom View-SlideListView

ListView is one of the most used controls in Android. ListView has many usages and handles events, such as item click and long press events. In many applications, click is jump, long press will pop up some selection menus, etc. Here I want to introduce a custom control for ListView to slide out menus.

The effect diagram is as follows:
Normal state

Slide out menu status

analyze

The Scroller sliding class is mainly used. At the beginning, when the touch event is intercepted at action == MotionEvent.ACTION_DOWN, the itemView we clicked is obtained according to the exit point, and then the left or right width is automatically obtained according to the sliding mode (left sliding or right sliding);

In action == MotionEvent.ACTION_MOVE, determine whether it is possible to slide and the direction of the slide based on the movement, and use itemView.scrollTo(deltaX, 0); to move itemView;

*** In action == MotionEvent.ACTION_UP, determine the mode and moving distance to complete the slide or restore to the initial state.
accomplish

The first step is to initialize Scroller

  1. scroller = new Scroller(context);
  2. mTouchSlop = ViewConfiguration .get(getContext()).getScaledTouchSlop();

Step 2 action == MotionEvent.ACTION_DOWN

  1. case MotionEvent.ACTION_DOWN:
  2. if ( this.mode == MOD_FORBID) {
  3. return super.onTouchEvent(ev);
  4. }
  5. // If the slide is completed, slide back and return directly
  6. if (isSlided) {
  7. scrollBack();
  8. return false;
  9. }
  10. // If the scroller has not finished scrolling, we return directly
  11. if (!scroller.isFinished()) {
  12. return false;
  13. }
  14.    
  15.                  downX = (int) ev.getX();
  16.                  downY = (int) ev.getY();
  17.    
  18.                  slidePosition = pointToPosition (downX, downY);
  19. // Invalid position, no processing
  20. if ( slidePosition == AdapterView.INVALID_POSITION) {
  21. return super.onTouchEvent(ev);
  22. }
  23.    
  24. // Get the item view we clicked
  25.                  itemView = getChildAt (slidePosition - getFirstVisiblePosition());
  26. /*Here, automatically get the length of the left or right menu according to the set sliding mode*/
  27. if ( this.mode == MOD_BOTH) {
  28.                      this.leftLength = -itemView.getPaddingLeft();
  29.                      this.rightLength = -itemView.getPaddingRight();
  30. } else if ( this.mode == MOD_LEFT) {
  31.                      this.leftLength = -itemView.getPaddingLeft();
  32. } else if ( this.mode == MOD_RIGHT) {
  33.                      this.rightLength = -itemView.getPaddingRight();
  34. }
  35. break;

Step 3 action == MotionEvent.ACTION_MOVE

  1. case MotionEvent.ACTION_MOVE:
  2. if (!canMove
  3. && slidePosition != AdapterView.INVALID_POSITION
  4. && (Math.abs(ev.getX() - downX) > mTouchSlop && Math.abs(ev
  5. .getY() - downY) <   mTouchSlop )) {
  6. if (mSwipeLayout != null)
  7. mSwipeLayout.setEnabled(false);
  8. int offsetX = downX - lastX;
  9. if (offsetX > 0 && ( this.mode == MOD_BOTH || this.mode == MOD_RIGHT)) {
  10. /*Slide from right to left*/
  11.                          canMove = true ;
  12. } else if (offsetX <   0 && ( this.mode == MOD_BOTH || this.mode == MOD_LEFT)) {
  13. /*Slide from left to right*/
  14.                          canMove = true ;
  15. } else {
  16.                          canMove = false ;
  17. }
  18. /*This code is to avoid triggering the OnItemClickListener event of ListView when sliding sideways*/
  19. MotionEvent cancelEvent = MotionEvent .obtain(ev);
  20. cancelEvent
  21. .setAction(MotionEvent.ACTION_CANCEL
  22. | (ev.getActionIndex() < <   MotionEvent.ACTION_POINTER_INDEX_SHIFT ));
  23. onTouchEvent(cancelEvent);
  24. }
  25. if (canMove) {
  26. /*Set this property to prevent the ListView from scrolling up and down when sliding sideways*/
  27. requestDisallowInterceptTouchEvent(true);
  28. // Drag itemView with your finger to scroll, if deltaX is greater than 0, scroll left, and if deltaX is less than 0, scroll right
  29. int deltaX = downX - lastX;
  30. if (deltaX <   0 && ( this.mode == MOD_BOTH || this.mode == MOD_LEFT)) {
  31. /*Slide left*/
  32. itemView.scrollTo(deltaX, 0);
  33. } else if (deltaX > 0 && ( this.mode == MOD_BOTH || this.mode == MOD_RIGHT)) {
  34. /*Slide right*/
  35. itemView.scrollTo(deltaX, 0);
  36. } else {
  37. itemView.scrollTo(0, 0);
  38. }
  39. return true;
  40. }
  41. break;

Step 4 action == MotionEvent.ACTION_UP

  1. case MotionEvent.ACTION_UP:
  2. if (mSwipeLayout != null)
  3. mSwipeLayout.setEnabled(true);
  4. //requestDisallowInterceptTouchEvent(false);
  5. if (canMove){
  6.                      canMove = false ;
  7. scrollByDistanceX();
  8. }
  9. break;

Complete code

Here is the complete code

  1. package com.jwenfeng.fastdev.view;
  2.    
  3. import android.content.Context;
  4. import android.support.v4.widget.SwipeRefreshLayout;
  5. import android.util.AttributeSet;
  6. import android.view.MotionEvent;
  7. import android.view.View;
  8. import android.view.ViewConfiguration;
  9. import android.widget.AdapterView;
  10. import android.widget.ListView;
  11. import android.widget.Scroller;
  12.    
  13. /**
  14. * Current class comment: ListView slide-out menu
  15. * Project name: fastdev
  16. * Package name: com.jwenfeng.fastdev.view
  17. * Author: jinwenfeng on 16/4/11 10:55
  18. * Email: [email protected]
  19. * QQ: 823546371
  20. * Company: Nanjing Muzun Information Technology Co., Ltd.
  21. * © 2016 jinwenfeng
  22. * © All rights reserved. No distribution without permission
  23. */
  24. public class SlideListView extends ListView {
  25.    
  26. /**Pull down to refresh view*/
  27. private SwipeRefreshLayout mSwipeLayout;
  28. /**
  29. * Disable side sliding mode
  30. */
  31. public static int MOD_FORBID = 0 ;
  32. /**
  33. * Slide from left to right to exit menu mode
  34. */
  35. public static int MOD_LEFT = 1 ;
  36. /**
  37. * Slide out of menu mode from right to left
  38. */
  39. public static int MOD_RIGHT = 2 ;
  40. /**
  41. * You can slide out the menu mode on the left or right
  42. */
  43. public static int MOD_BOTH = 3 ;
  44. /**
  45. * Current mode
  46. */
  47. private int mode = MOD_FORBID ;
  48. /**
  49. * Length of the left menu
  50. */
  51. private int leftLength = 0 ;
  52. /**
  53. * The length of the right menu
  54. */
  55. private int rightLength = 0 ;
  56.    
  57. /**
  58. * The current sliding ListView position
  59. */
  60. private int slidePosition;
  61. /**
  62. * The X coordinate where the finger presses
  63. */
  64. private int downY;
  65. /**
  66. * The Y coordinate where the finger presses
  67. */
  68. private int downX;
  69. /**
  70. * ListView items
  71. */
  72. private View itemView;
  73. /**
  74. * Sliding
  75. */
  76. private Scroller scroller;
  77. /**
  78. * Considered to be the minimum distance that the user can slide
  79. */
  80. private int mTouchSlop;
  81.    
  82. /**
  83. * Determine whether it can slide sideways
  84. */
  85. private boolean canMove = false ;
  86. /**
  87. * Indicates whether the slide is completed
  88. */
  89. private boolean isSlided = false ;
  90.    
  91. public SlideListView(Context context) {
  92. this(context, null);
  93. }
  94.    
  95. public SlideListView(Context context, AttributeSet attrs) {
  96. this(context, attrs,0);
  97. }
  98.    
  99. public SlideListView(Context context, AttributeSet attrs, int defStyleAttr) {
  100. super(context, attrs, defStyleAttr);
  101.          scroller = new Scroller(context);
  102.          mTouchSlop = ViewConfiguration .get(getContext()).getScaledTouchSlop();
  103. }
  104.    
  105. /**
  106. * Initialize the menu's slide-out mode
  107. *
  108. * @param mode
  109. */
  110. public void initSlideMode(int mode) {
  111.          this.mode = mode;
  112. }
  113.    
  114. /**
  115. * Handle the logic of dragging ListView item
  116. */
  117. @Override
  118. public boolean onTouchEvent(MotionEvent ev) {
  119. final int action = ev .getAction();
  120. int lastX = (int) ev.getX();
  121.    
  122. switch (action) {
  123. case MotionEvent.ACTION_DOWN:
  124. if ( this.mode == MOD_FORBID) {
  125. return super.onTouchEvent(ev);
  126. }
  127. // If the slide is completed, slide back and return directly
  128. if (isSlided) {
  129. scrollBack();
  130. return false;
  131. }
  132. // If the scroller has not finished scrolling, we return directly
  133. if (!scroller.isFinished()) {
  134. return false;
  135. }
  136.    
  137.                  downX = (int) ev.getX();
  138.                  downY = (int) ev.getY();
  139.    
  140.                  slidePosition = pointToPosition (downX, downY);
  141. // Invalid position, no processing
  142. if ( slidePosition == AdapterView.INVALID_POSITION) {
  143. return super.onTouchEvent(ev);
  144. }
  145.    
  146. // Get the item view we clicked
  147.                  itemView = getChildAt (slidePosition - getFirstVisiblePosition());
  148. /*Here, automatically get the length of the left or right menu according to the set sliding mode*/
  149. if ( this.mode == MOD_BOTH) {
  150.                      this.leftLength = -itemView.getPaddingLeft();
  151.                      this.rightLength = -itemView.getPaddingRight();
  152. } else if ( this.mode == MOD_LEFT) {
  153.                      this.leftLength = -itemView.getPaddingLeft();
  154. } else if ( this.mode == MOD_RIGHT) {
  155.                      this.rightLength = -itemView.getPaddingRight();
  156. }
  157. break;
  158. case MotionEvent.ACTION_MOVE:
  159. if (!canMove
  160. && slidePosition != AdapterView.INVALID_POSITION
  161. && (Math.abs(ev.getX() - downX) > mTouchSlop && Math.abs(ev
  162. .getY() - downY) <   mTouchSlop )) {
  163. if (mSwipeLayout != null)
  164. mSwipeLayout.setEnabled(false);
  165. int offsetX = downX - lastX;
  166. if (offsetX > 0 && ( this.mode == MOD_BOTH || this.mode == MOD_RIGHT)) {
  167. /*Slide from right to left*/
  168.                          canMove = true ;
  169. } else if (offsetX <   0 && ( this.mode == MOD_BOTH || this.mode == MOD_LEFT)) {
  170. /*Slide from left to right*/
  171.                          canMove = true ;
  172. } else {
  173.                          canMove = false ;
  174. }
  175. /*This code is to avoid triggering the OnItemClickListener event of ListView when sliding sideways*/
  176. MotionEvent cancelEvent = MotionEvent .obtain(ev);
  177. cancelEvent
  178. .setAction(MotionEvent.ACTION_CANCEL
  179. | (ev.getActionIndex() < <   MotionEvent.ACTION_POINTER_INDEX_SHIFT ));
  180. onTouchEvent(cancelEvent);
  181. }
  182. if (canMove) {
  183. /*Set this property to prevent the ListView from scrolling up and down when sliding sideways*/
  184. requestDisallowInterceptTouchEvent(true);
  185. // Drag itemView with your finger to scroll, if deltaX is greater than 0, scroll left, and if deltaX is less than 0, scroll right
  186. int deltaX = downX - lastX;
  187. if (deltaX <   0 && ( this.mode == MOD_BOTH || this.mode == MOD_LEFT)) {
  188. /*Slide left*/
  189. itemView.scrollTo(deltaX, 0);
  190. } else if (deltaX > 0 && ( this.mode == MOD_BOTH || this.mode == MOD_RIGHT)) {
  191. /*Slide right*/
  192. itemView.scrollTo(deltaX, 0);
  193. } else {
  194. itemView.scrollTo(0, 0);
  195. }
  196. return true;
  197. }
  198. break;
  199.    
  200. case MotionEvent.ACTION_UP:
  201. if (mSwipeLayout != null)
  202. mSwipeLayout.setEnabled(true);
  203. //requestDisallowInterceptTouchEvent(false);
  204. if (canMove){
  205.                      canMove = false ;
  206. scrollByDistanceX();
  207. }
  208. break;
  209. }
  210.    
  211. return super.onTouchEvent(ev);
  212. }
  213.    
  214. private void scrollByDistanceX() {
  215. if( this.mode == MOD_FORBID){
  216. return;
  217. }
  218. if(itemView.getScrollX() > 0 && ( this.mode == MOD_BOTH || this.mode == MOD_RIGHT)){
  219. /*Slide from right to left*/
  220. if (itemView.getScrollX() > = rightLength / 2) {
  221. scrollLeft();
  222. } else {
  223. // Scroll back to the original position
  224. scrollBack();
  225. }
  226. }else if(itemView.getScrollX() <   0 && ( this.mode == MOD_BOTH || this.mode == MOD_LEFT)){
  227. /*Slide from left to right*/
  228. if (itemView.getScrollX() < = -leftLength / 2) {
  229. scrollRight();
  230. } else {
  231. // Scroll back to the original position
  232. scrollBack();
  233. }
  234. }else{
  235. // Scroll back to the original position
  236. scrollBack();
  237. }
  238. }
  239.    
  240. /**
  241. * When sliding to the right, getScrollX() returns the distance to the left edge, which is the distance from the left edge of the View to the start of sliding, so sliding to the right is a negative value
  242. */
  243. private void scrollRight() {
  244.          isSlided = true ;
  245. final int delta = (leftLength + itemView.getScrollX());
  246. // Call the startScroll method to set some scrolling parameters. We call scrollTo in the computeScroll() method to scroll the item.
  247. scroller.startScroll(itemView.getScrollX(), 0, -delta, 0,
  248. Math.abs(delta));
  249. postInvalidate(); // Refresh itemView
  250. }
  251.    
  252. /**
  253. * Slide to the left. According to the above, we know that sliding to the left is a positive value.
  254. */
  255. private void scrollLeft() {
  256.          isSlided = true ;
  257. final int delta = (rightLength - itemView.getScrollX());
  258. // Call the startScroll method to set some scrolling parameters. We call scrollTo in the computeScroll() method to scroll the item.
  259. scroller.startScroll(itemView.getScrollX(), 0, delta, 0,
  260. Math.abs(delta));
  261. postInvalidate(); // Refresh itemView
  262.    
  263. }
  264.    
  265. private void scrollBack() {
  266.          isSlided = false ;
  267. scroller.startScroll(itemView.getScrollX(), 0, -itemView.getScrollX(),
  268. 0, Math.abs(itemView.getScrollX()));
  269. postInvalidate(); // Refresh itemView
  270. }
  271.    
  272. @Override
  273. public void computeScroll() {
  274. // When startScroll is called, scroller.computeScrollOffset() returns true.
  275. if (scroller.computeScrollOffset()) {
  276. // Let the ListView item scroll according to the current scroll offset
  277. itemView.scrollTo(scroller.getCurrX(), scroller.getCurrY());
  278. postInvalidate();
  279. }
  280. }
  281.    
  282. /**
  283. * Provided for external calls to slide the slide back
  284. */
  285. public void slideBack() {
  286. this.scrollBack();
  287. }
  288.    
  289. public void setSwipeLayout(SwipeRefreshLayout mSwipeLayout) {
  290.          this.mSwipeLayout = mSwipeLayout;
  291. }
  292. }

<<:  Jiaxin Customer Service: A Transformer of Enterprise Customer Service in the "Internet +" Era

>>:  The standard answers to programmer interviews are not standard

Recommend

How can businesses use short video platforms for marketing in 2021?

A summary of how businesses and companies can use...

Power consumption optimization practice of Douyin

Author: Gaoyuan Tang Zhongfeng Power consumption ...

Community Operation: How do I operate the community?

The essence of a community is connection. It is a...

Create short video IP in the first year of 5G and make your fans addicted to you

2019 is known as the first year of 5G. The arriva...

How much does it cost to make a wine utensils mini program in Bayanzhuoer?

There is no doubt that the topic of mini programs...

How brands can be revitalized through private domain marketing

Since the concept of private domain marketing has...

How to use traffic dividends to acquire customers at low cost?

In an environment dominated by traffic giants suc...

Amazon Entry-level to Mastery Training Courses (26 sets)

A collection of Amazon introductory to proficient...

How to design a landing page to reduce information flow costs by 60%?

With the promotion competition so fierce today, s...

Game live streaming platform: Huya Live Development Analysis

The author of this article analyzes the main func...

2019 Google keyword search advertising skills

In fact, there are skills to optimize the number ...