研究了很久的拖拽ListView的实现,受益良多,特此与尔共飨。
鉴于这部分内容网上的资料少而简陋,而具体的实现过程或许对大家才有帮助,为了详尽而不失真,我们一步一步分析,分成两篇文章。
一、准备。
1.需求问题
初步:实现列表的拖拽效果(可参考Android源码下packages/apps/Music中的播放列表TouchInterceptor.java源码)。
(提前说明一下,本文不是完全按照Music中实现的,代码实现方式做了一些调整,去掉来很多无关的东西,方便大家理解,效果上也修改成了另外一种 个人认为是更简单更高效的一套。)
拓展:借鉴上一篇文章Android学习系列(9)--App列表之分组ListView,实现分组列表的拖拽效果。
下面以初步实现为例子,逐步展开实现步骤。
2.搭建主界面DragListActivity.java和主布局drag_list_activity.xml。
01 |
public class DragListActivity extends Activity { |
04 |
private List<String> list = null; |
07 |
private DragListAdapter adapter = null; |
10 |
public static List<String> groupKey= new ArrayList<String>(); |
12 |
private List<String> navList = new ArrayList<String>(); |
14 |
private List<String> moreList = new ArrayList<String>(); |
17 |
public void onCreate(Bundle savedInstanceState) { |
18 |
super.onCreate(savedInstanceState); |
19 |
setContentView(R.layout.drag_list_activity); |
25 |
DragListView dragListView = (DragListView)findViewById(R.id.drag_list); |
26 |
adapter = new DragListAdapter(this, list); |
27 |
dragListView.setAdapter(adapter); |
3.列表项的布局drag_list_item.xml。
01 |
<?xml version="1.0" encoding="utf-8"?> |
04 |
android:layout_width="fill_parent" |
05 |
android:layout_height="wrap_content"> |
07 |
android:id="@+id/drag_list_item_text" |
08 |
android:layout_width="wrap_content" |
09 |
android:layout_height="@dimen/drag_item_normal_height" |
10 |
android:paddingLeft="5dip" |
11 |
android:layout_alignParentLeft="true" |
12 |
android:layout_centerVertical="true" |
13 |
android:gravity="center_vertical"/> |
14 |
<ImageView android:id="@+id/drag_list_item_image" |
15 |
android:src="@drawable/list_icon" |
16 |
android:layout_alignParentRight="true" |
17 |
android:layout_centerVertical="true" |
18 |
android:layout_width="wrap_content" |
19 |
android:layout_height="@dimen/drag_item_normal_height"/> |
4.准备样本数据。
我已经准备好了两组数据,在前面提到的initData()方法中执行初始化。
01 |
public void initData(){ |
03 |
list = new ArrayList<String>(); |
09 |
for(int i=0; i<5; i++){ |
15 |
for(int i=0; i<8; i++){ |
16 |
moreList.add("B选项"+i); |
19 |
list.addAll(moreList); |
这里定义了分组标签集合groupKey后面分组的时候会用到。
5.自定义适配器类DragListAdapter。
接着我们搭建数据适配器,负责把list的数据填充到ListView中。
01 |
public static class DragListAdapter extends ArrayAdapter<String>{ |
02 |
public DragListAdapter(Context context, List<String> objects) { |
03 |
super(context, 0, objects); |
06 |
public View getView(int position, View convertView, ViewGroup parent) { |
07 |
View view = convertView; |
10 |
view = LayoutInflater.from(getContext()).inflate(R.layout.drag_list_item, null); |
12 |
TextView textView = (TextView)view.findViewById(R.id.drag_list_item_text); |
13 |
textView.setText(getItem(position)); |
注意getItem(position)会取得数组适配器中position位置的T(这里是字符串),比较好用的一个方法。
至此,我们准备了一个正常的数据列表,效果如下:
二、实现
上面部分是我们的一个 准备工作,接下来我们通过自定义ListView,重写ListView中onInterceptTouchEvent(),onTouchEvent()方法来响应触控事件做相应的界面调整(选中,拖动,数据更改后刷新界面)等等。
6.自定义视图类。
03 |
public class DragListView extends ListView { |
05 |
private int scaledTouchSlop; |
06 |
public DragListView(Context context, AttributeSet attrs) { |
07 |
super(context, attrs); |
08 |
scaledTouchSlop = ViewConfiguration.get(context).getScaledTouchSlop(); |
7.重写触控拦截事件方法onInterceptTouchEvent()。
为了能在子控件响应触摸事件的情况下此ListView也能监听到触摸事件,我们把重写这个方法,做一些初始化工作。我们在这里捕获down事件,在down事件中,我们做一些拖动的准备工作:
1)获取点击数据项,初始化一些变量;
2)判断是否是拖动还是仅仅点击;
3)如果是拖动,建立拖动影像;
这些工作是我们后面拖动的一个执行基础,非常重要。
02 |
private ImageView dragImageView; |
03 |
private int dragSrcPosition; |
04 |
private int dragPosition; |
06 |
private int dragPoint; |
07 |
private int dragOffset; |
09 |
private WindowManager windowManager; |
10 |
private WindowManager.LayoutParams windowParams; |
12 |
private int scaledTouchSlop; |
13 |
private int upScrollBounce; |
14 |
private int downScrollBounce; |
17 |
public boolean onInterceptTouchEvent(MotionEvent ev) { |
19 |
if(ev.getAction()==MotionEvent.ACTION_DOWN){ |
20 |
int x = (int)ev.getX(); |
21 |
int y = (int)ev.getY(); |
24 |
dragSrcPosition = dragPosition = pointToPosition(x, y); |
26 |
if(dragPosition==AdapterView.INVALID_POSITION){ |
27 |
return super.onInterceptTouchEvent(ev); |
33 |
ViewGroup itemView = (ViewGroup) getChildAt(dragPosition-getFirstVisiblePosition()); |
38 |
dragPoint = y - itemView.getTop(); |
39 |
dragOffset = (int) (ev.getRawY() - y); |
42 |
View dragger = itemView.findViewById(R.id.drag_list_item_image); |
44 |
if(dragger!=null&&x>dragger.getLeft()-20){ |
49 |
upScrollBounce = Math.min(y-scaledTouchSlop, getHeight()/3); |
50 |
downScrollBounce = Math.max(y+scaledTouchSlop, getHeight()*2/3); |
53 |
itemView.setDrawingCacheEnabled(true); |
54 |
Bitmap bm = Bitmap.createBitmap(itemView.getDrawingCache()); |
61 |
return super.onInterceptTouchEvent(ev); |
看到上面的一大堆变量和操作,你可能有些眼花缭乱,在后面使用的时候回头再去理解也可。
开始拖动影像startDrag()方法:
06 |
public void startDrag(Bitmap bm ,int y){ |
10 |
windowParams = new WindowManager.LayoutParams(); |
12 |
windowParams.gravity = Gravity.TOP; |
14 |
windowParams.y = y - dragPoint + dragOffset; |
15 |
windowParams.width = WindowManager.LayoutParams.WRAP_CONTENT; |
16 |
windowParams.height = WindowManager.LayoutParams.WRAP_CONTENT; |
18 |
windowParams.flags = WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE |
19 |
| WindowManager.LayoutParams.FLAG_NOT_TOUCHABLE |
20 |
| WindowManager.LayoutParams.FLAG_KEEP_SCREEN_ON |
21 |
| WindowManager.LayoutParams.FLAG_LAYOUT_IN_SCREEN; |
22 |
windowParams.format = PixelFormat.TRANSLUCENT; |
23 |
windowParams.windowAnimations = 0; |
26 |
ImageView imageView = new ImageView(getContext()); |
27 |
imageView.setImageBitmap(bm); |
28 |
windowManager = (WindowManager)getContext().getSystemService("window"); |
29 |
windowManager.addView(imageView, windowParams); |
31 |
dragImageView = imageView; |
stopDrag()方法如下:
4 |
public void stopDrag(){ |
5 |
if(dragImageView!=null){ |
6 |
windowManager.removeView(dragImageView); |
运行看看,我们点击一项的时候没有什么反应,但是细心观察的话,其实点击项上有一层淡淡的重影,这就是我们定义的点击项的影像,后面要做的就是拖动这个影 像,放下影像以及放下位置的数据项插入和原数据项位置的删除,这部分内容以及后续扩展的内容我们放到下篇中继续分析并实现。
未完待续...