<Excerpt in index | 首页摘要>
Android拖拽排序功能实现
<The rest of contents | 余下全文>
先来看一下效果:
##思路
Android本身就给我们提供了拖拽的实现.
在View身上都有一个startDrag()方法.
想要实现拖拽效果,还需要让拖拽的父View实现一个onDragListener.
##实现概述
首先,我选择继承自GradLayout,然后手动创建child.
然后在每个item身上添加长按事件.
长按时间内容为启动拖拽.
1
| v.startDrag(null, new DragShadowBuilder(v), null, 0);
|
new DragShadowBuilder(view)是创建一个用于拖拽的虚影,它会自动跟随手势移动.
然后在父容器实现OnDragListener,在开始拖拽的时候实现获取每个View的坐标,放置在集合中.
然后在手势移动的时候,判断当前手势的坐标点是否在某个item上,如果在就移除当前要拖拽的view,然后添加到获取到的坐标点上.
##具体代码实现
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176
| package com.qingxiang.ui.view;
import android.animation.LayoutTransition; import android.content.Context; import android.graphics.Rect; import android.util.AttributeSet; import android.view.DragEvent; import android.view.LayoutInflater; import android.view.View; import android.widget.GridLayout;
import com.bumptech.glide.Glide; import com.qingxiang.ui.R; import com.qingxiang.ui.bean.TagGroupBean; import com.qingxiang.ui.common.CommonViewHolder; import com.qingxiang.ui.utils.DensityUtils; import com.qingxiang.ui.utils.L; import com.qingxiang.ui.utils.QNUtils; import com.qingxiang.ui.utils.Utils;
import java.util.ArrayList; import java.util.List;
public class DragSortView extends GridLayout { private static final String TAG = "DragSortView"; private static final int COLUMN_COUNT = 4; private int itemSize; private View dragView; private List<Rect> viewRects; private int margin;
public DragSortView(Context context) { this(context, null); }
public DragSortView(Context context, AttributeSet attrs) { this(context, attrs, 0); }
public DragSortView(Context context, AttributeSet attrs, int defStyleAttr) { super(context, attrs, defStyleAttr); init(); }
private void init() { setColumnCount(COLUMN_COUNT); LayoutTransition transition = new LayoutTransition(); transition.setDuration(150); setLayoutTransition(transition); int width = Utils.getScreenWidth(); width -= DensityUtils.dp2px(getContext(), 20 + ((COLUMN_COUNT - 1) * 3)); itemSize = (int) (width * 1.0f / COLUMN_COUNT + 0.5f); viewRects = new ArrayList<>(); margin = DensityUtils.dp2px(getContext(), 3); }
public void setData(List<TagGroupBean> mDatas) { removeAllViews(); int i = 0; for (TagGroupBean mData : mDatas) { addView(i, mData); i++; } setOnDragListener(new DragListener()); }
private void addView(int pos, TagGroupBean mData) { LayoutInflater inflater = LayoutInflater.from(getContext()); View view = inflater.inflate(R.layout.item_find_edit_sub, this, false); CommonViewHolder holder = new CommonViewHolder(view); holder.getTextView(R.id.item_tv).setText(mData.getTag()); Glide.with(getContext()) .load(QNUtils.formatUrl(mData.getIcon(), 1, 400, 400, false)) .placeholder(R.mipmap.area_img1) .into(holder.getIv(R.id.item_iv)); holder.getContentView().setOnLongClickListener(v -> { dragView = v; v.startDrag(null, new DragShadowBuilder(v), null, 0); return true; }); addView(view, getParams(pos)); }
private class DragListener implements OnDragListener {
int lastIndex = -1;
@Override public boolean onDrag(View v, DragEvent event) { switch (event.getAction()) { case DragEvent.ACTION_DRAG_STARTED: initRects(); lastIndex = indexOfChild(dragView); if (dragView != null) dragView.setEnabled(false); return true; case DragEvent.ACTION_DRAG_ENDED: lastIndex = -1; if (dragView != null) { dragView.setEnabled(true); dragView = null; } return true; case DragEvent.ACTION_DRAG_LOCATION: int moveX = (int) event.getX(); int moveY = (int) event.getY(); L.i(TAG, "location X:" + moveX + " Y:" + moveY); int findIndex = findTouchChildIndex(moveX, moveY); if (findIndex != -1 && findIndex != lastIndex && dragView != null) { removeView(dragView); addView(dragView, findIndex); lastIndex = findIndex; } return true; } return false; } }
private void initRects() { viewRects.clear(); for (int i = 0; i < getChildCount(); i++) { View child = getChildAt(i); Rect object = new Rect(child.getLeft(), child.getTop(), child.getRight(), child.getBottom()); viewRects.add(object); } }
private int findTouchChildIndex(int moveX, int moveY) { for (int i = 0; i < viewRects.size(); i++) { boolean isFind = viewRects.get(i).contains(moveX, moveY); if (isFind) { return i; } } return -1; }
private GridLayout.LayoutParams getParams(int pos) { GridLayout.LayoutParams params = new GridLayout.LayoutParams();
params.height = itemSize; params.width = itemSize; params.leftMargin = margin;
if (pos < 4) { params.bottomMargin = margin; }
return params; } }
|