仿支付宝扣款顺序,动态改变ListView各Item次序

1. 概述

在开发过程中,有时候遇到这样的需求:点击ListView的item,可以动态的上移和下移,就和支付宝的扣款顺序一样,如下图所示:


图片.png
效果如下:
1>:银行卡列表为listview,默认向上向下箭头隐藏;
2>:点击某个item,就让向上向下箭头显示;
3>:第一个item的向上箭头不能点击,最后一个item的向下箭头不能点击;
4>:对于中间的item:
点击向上箭头:和上边item数据交换位置;
点击向下箭头:和下边item数据交换位置;
说明:这个效果自己也是参照网上别人写的,自己根据自己理解有重新写了一遍,然后贴出思路和代码
2. 实现思路

1. 利用 ListView+BaseAdapter 来布局;
2. 在 BaseAdapter中 的 getView() 方法中,需要设置3个点击事件:
   1>:开始时:默认向上向下箭头隐藏;
   2>:设置 listview 的 setOnItemClickListener 点击事件,然后将点击的 item 的 向上向下箭头显示;
   3>:向上箭头设置点击事件,点击时将该 item 上移;
   4>:向下箭头设置点击事件,点击时将该 item 下移;
3. item上移下移的原理:
   1>:点击向上箭头时,将当前item的数据值与前一个 item 的数据值交换,并调用adapter.notifyDataSetChanged()刷新数据;
   2>:点击向下箭头时,将当前item的数据值与后一个 item 的数据值交换,并调用adapter.notifyDataSetChanged()刷新数据;
3. 代码如下

1>:DefaultPayStyleActivity代码如下:
/**
 * Email: 2185134304@qq.com
 * Created by Novate 2018/8/21 9:42
 * Version 1.0
 * Params:
 * Description:    默认付款方式
*/

public class DefaultPayStyleActivity extends BaseActivity implements ChangeAdapter.Callback {

    private ListView lv;
    private ChangeAdapter adapter;
    private ArrayList<String> itemList;
    private int currentPosition = -1;

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_default_pay_style);
    }

    @Override
    public void initView() {
        lv = (ListView) findViewById(R.id.lv_change);
        initDatas() ;
        adapter = new ChangeAdapter(this, itemList, this, currentPosition);
        lv.setAdapter(adapter);
    }

    private void initDatas() {
        itemList = new ArrayList<String>();
        itemList.add("余额宝");
        itemList.add("蚂蚁花呗");
        itemList.add("余额");
        itemList.add("工商银行储蓄卡(1689)");
        itemList.add("花呗分期");
    }

    /**
     * ChangeAdapter中的3个点击事件
     * 1. 整个item点击事件的处理逻辑
     * 2. 向上图标按键点击事件的处理逻辑
     * 3. 向下图标按键点击事件的处理逻辑
     */
    @Override
    public void click(View v) {
        int curPosition;
        int mCurPosition;
        switch (v.getId()){
            //整个item点击事件的处理逻辑
            case R.id.item:
                // 通过getTag(key)方法, 强制类型转换获取该item的position,然后调用refresh()方法通知adapter当前position数据更新了,
                // 然后adapter就会根据 currentPosition处理向上向下箭头的显示与隐藏
                mCurPosition = (int) v.getTag(R.id.tag_item_click);
                currentPosition = mCurPosition;
                adapter.refresh(currentPosition);
                break;

            //向上图标按键点击事件的处理逻辑
            case R.id.up:
                // 调用getTag获取点击的position,调换当前位置与上一个位置的数据值,就实现了item上移效果,对于第一个item,不处理上移,
                // 然后调用refresh()方法刷新数据即可
                curPosition = (int) v.getTag();
                if (curPosition != 0) {
                    String upFirst = itemList.get(curPosition); // 当前item位置对应数据
                    String upSecond = itemList.get(curPosition - 1); // 上一个位置数据值
                    itemList.remove(curPosition);
                    itemList.remove(curPosition - 1);
                    itemList.add(curPosition - 1, upFirst);
                    itemList.add(curPosition, upSecond);
                    currentPosition = curPosition - 1;
                    adapter.refresh(currentPosition);
                }
                break;

            //向下图标按键点击事件的处理逻辑
            case R.id.down:
                // 调用getTag获取点击的position,调用当前位置与下一个位置的数据值,就实现item下移效果,对于最后一个item,不处理下移,
                // 然后调用refresh()方法刷新数据即可
                curPosition = (int) v.getTag();
                if (curPosition != itemList.size() - 1) {
                    String downFirst = itemList.get(curPosition);
                    String downSecond = itemList.get(curPosition + 1);
                    itemList.remove(curPosition + 1);
                    itemList.remove(curPosition);
                    itemList.add(curPosition, downSecond);
                    itemList.add(curPosition + 1, downFirst);
                    currentPosition = curPosition + 1;
                    adapter.refresh(currentPosition);
                }
                break;
            default:
                break;
        }
    }
}
2>:ChangeAdapter代码如下:
/**
 * Email: 2185134304@qq.com
 * Created by Novate 2018/8/21 10:24
 * Version 1.0
 * Params:
 * Description:    默认付款方式扣款
*/

public class ChangeAdapter extends BaseAdapter implements View.OnClickListener {

    private ArrayList<String> itemList;
    private Context mContext;
    private Callback mCallback;
    private int mCurPosition;//定义该变量来标记当前item的点击位置

    //定义回调接口实现ListView内Item的内部控件的点击事件
    public interface Callback {
        public void click(View v);
    }

    public ChangeAdapter(Context mContext, ArrayList<String> itemList, Callback mCallback, int mCurPosition) {
        this.mContext = mContext;
        this.itemList = itemList;
        this.mCallback = mCallback;
        this.mCurPosition = mCurPosition;
    }

    @Override
    public int getCount() {
        return itemList.size();
    }

    @Override
    public Object getItem(int position) {
        return itemList.get(position);
    }

    @Override
    public long getItemId(int position) {
        return position;
    }

    @Override
    public View getView(int position, View convertView, ViewGroup parent) {

        ViewHolder viewHolder;
        if (convertView == null) {
            convertView = LayoutInflater.from(mContext).inflate(R.layout.item_change, null);
            viewHolder = new ViewHolder();
            viewHolder.itemName = (CustomTextView) convertView.findViewById(R.id.item_name);
            viewHolder.upBtn = (ImageView) convertView.findViewById(R.id.up);
            viewHolder.downBtn = (ImageView) convertView.findViewById(R.id.down);
            convertView.setTag(R.id.tag_viewholder, viewHolder);
        } else {
            viewHolder = (ViewHolder) convertView.getTag(R.id.tag_viewholder);
        }
        viewHolder.itemName.setText(itemList.get(position));


        //根据点击或者向上向下操作的item的当前位置,来控制向上和向下的按钮的可见与否
        if (mCurPosition == position && mCurPosition == 0) {
            // 自己添加
            viewHolder.upBtn.setVisibility(View.VISIBLE);
            viewHolder.upBtn.setEnabled(false);

            viewHolder.downBtn.setVisibility(View.VISIBLE);
        } else if (mCurPosition == position && mCurPosition == itemList.size() - 1) {
            viewHolder.upBtn.setVisibility(View.VISIBLE);

            // 自己添加
            viewHolder.downBtn.setVisibility(View.VISIBLE);
            viewHolder.downBtn.setEnabled(false);

        } else if (mCurPosition == position && mCurPosition != 0 && mCurPosition != itemList.size() - 1) {
            viewHolder.upBtn.setVisibility(View.VISIBLE);
            viewHolder.downBtn.setVisibility(View.VISIBLE);

            // 自己添加
            viewHolder.upBtn.setEnabled(true);
            viewHolder.upBtn.setEnabled(true);


        } else {
            viewHolder.upBtn.setVisibility(View.INVISIBLE);
            viewHolder.downBtn.setVisibility(View.INVISIBLE);
        }

        //设置item向下移动的点击事件并标志其位置,只需要设置position即可
        viewHolder.downBtn.setOnClickListener(this);
        viewHolder.downBtn.setTag(position);

        //设置item向上移动的点击事件并标志其位置,只需要设置position即可
        viewHolder.upBtn.setOnClickListener(this);
        viewHolder.upBtn.setTag(position);

        //设置整个item的点击事件并标志其位置,需要通过tag_item_click
        convertView.setOnClickListener(this);
        convertView.setTag(R.id.tag_item_click, position);

        return convertView;
    }


    class ViewHolder {
        CustomTextView itemName;
        ImageView upBtn,downBtn;
    }

    //定义item内部控件的点击事件由回调接口定义的点击方法来处理
    @Override
    public void onClick(View v) {
        mCallback.click(v);
    }

    //在对数据进行处理后,调用该方法,通知adapter刷新数据
    public void refresh(int currentPosition) {
        mCurPosition = currentPosition;
        notifyDataSetChanged();
    }
}
3>:item_change布局文件如下:
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:id="@+id/item"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:orientation="vertical"
    >

    <RelativeLayout
        android:layout_width="match_parent"
        android:layout_height="@dimen/px70"
        android:paddingLeft="@dimen/px30"
        android:paddingRight="@dimen/px30"
        >
    <cn.uploo.yhh.view.CustomTextView
        android:id="@+id/item_name"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="余额宝"
        android:layout_centerVertical="true"
        />


    <ImageView
        android:id="@+id/up"
        android:layout_width="@dimen/px50"
        android:layout_height="@dimen/px50"
        android:src="@mipmap/up"
        android:background="@null"
        android:visibility="visible"
        android:layout_alignParentRight="true"
        android:layout_centerVertical="true"
        android:layout_marginLeft="@dimen/px60"
        />

    <ImageView
        android:id="@+id/down"
        android:layout_width="@dimen/px50"
        android:layout_height="@dimen/px50"
        android:src="@mipmap/down"
        android:background="@null"
        android:visibility="visible"
        android:layout_toLeftOf="@id/up"
        android:layout_centerVertical="true"
        android:layout_marginRight="@dimen/px30"
        />
    </RelativeLayout>
    <View
        android:layout_width="match_parent"
        android:layout_height="@dimen/px1"
        android:background="@color/bg_hui"
        />

</LinearLayout>
4>:在res -> values 创建 ids.xml文件:
<resources>
    <!-- 绑定 ChangeAdapter 中的 viewholder -->
    <item type="id" name="tag_viewholder"></item>
    <!-- 绑定 ChangeAdapter 的item 点击事件的id -->
    <item type="id" name="tag_item_click"></item>
</resources>

4. 需要注意的地方

1.  在ChangeAdapter中定义接口Callback,然后在 getView() 方法中对 item、向上箭头、向下箭头设置点击事件,
    并且通过setTag方法保存对应position位置:

    a:对于 viewHolder、整个item的点击事件需要在 res -> values -> ids中定义对应id
         来保存viewHolder和点击的item:
         convertView.setTag(R.id.tag_viewholder, viewHolder);
         convertView.setTag(R.id.tag_item_click, position);

    b:对于向上、向下箭头的点击事件直接使用setTag方法保存点击的位置就可以:
        viewHolder.downBtn.setOnClickListener(this);
        viewHolder.downBtn.setTag(position);
        viewHolder.upBtn.setOnClickListener(this);
        viewHolder.upBtn.setTag(position);

    c:在 DefaultPayStyleActivity中直接实现Callback接口,然后通过ChangeAdapter的构造方法把 callback的
       this传递给 ChangeAdapter即可

2.  从第一步可知:我们已经对item、两个箭头的点击事件通过setTag方法和setTag(R.id.tag_item_click, position)
    方法将点击的 位置 position保存了,可以直接在 Activity中听过 getTag方法获取即可;

3.  然后 在 ChangeAdapter中 定义 refresh()刷新数据的方法,当点击位置放生改变后,通知adapter刷新:

4.  在DefaultPayStyleActivity中重点就是click方法:

    1>:对于item的点击事件:首先通过 getTag方法并强制转换为点击的item的position,然后调用refresh()方法刷新
        数据;

    2>:对于向上箭头点击事件:通过getTag获取到当前点击位置,并交换当前位置和上一个位置的数据值,此时就实现了
        item上移效果,对于第一个item不处理上移效果,最后调用refresh()刷新数据即可;

    3>:对于向下箭头点击事件:通过getTag获取到当前点击位置,并交换当前位置和下一个位置的数据值,此时就实现了
        item下移效果,对于最后一个item不处理下移效果,最后调用refresh()刷新数据即可;

5.  这里没有存储item移动位置后的位置,如果需要,可以使用sp存储,这样下次再打开界面时就可以看见之前移动后的位置.

这里会涉及到ArrayList移除数据 remove方法的使用,详情见下边文章:
ArrayList中上移下移list集合中相邻两个位置数据

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,098评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,213评论 2 380
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 149,960评论 0 336
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,519评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,512评论 5 364
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,533评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,914评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,574评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,804评论 1 296
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,563评论 2 319
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,644评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,350评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,933评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,908评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,146评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,847评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,361评论 2 342

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,384评论 25 707
  • 【Android 控件 RecyclerView】 概述 RecyclerView是什么 从Android 5.0...
    Rtia阅读 307,309评论 27 439
  • 转:昨晚有幸参加格西老师《脱“瘾”而出》体验课程大收获 1、上瘾症有以下特质: 我总想得到;无法控制和停止;会伤害...
    yoyo1969001阅读 85评论 0 0
  • Anaconda使用总结 2016.7.19 PeterYuan 序 Python易用,但用好却不易,其中比较头疼...
    PeterYuan阅读 311,640评论 36 347
  • 生活里,我们都有一个错觉:幸福总是别人的,唯有烦恼属于自己!于是,我们一天天感受着所谓的烦恼,一天天寻找着,其实就...
    六两在路上阅读 124评论 0 0