开源项目MultiChoiceAdapter详解(五)——可扩展的MultiChoiceBaseAdapter

  上次写到了开源项目MultiChoiceAdapter详解(四)——MultiChoiceBaseAdapter的使用,其实我们仍旧可以不使用ActionMode的,所以这里就写一个自己扩展的方法。

一、布局文件

listview_normal_layout.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical" >

    <ListView
         android:id="@+id/normal_listView"
         android:layout_width="match_parent"
         android:layout_height="wrap_content"
         android:layout_weight="1" >
    </ListView>

    <LinearLayout
        android:id="@+id/setting_linearLayout"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_gravity="bottom">

        <Button
            android:id="@+id/selectAll_button"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="全选"
            android:layout_weight="1"
            android:onClick="buttonListener"/>

        <Button
            android:id="@+id/cancle_button"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="取消"
            android:layout_weight="1"
            android:onClick="buttonListener"/>

        <Button
            android:id="@+id/delete_button"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_weight="1"
            android:text="删除"
            android:onClick="buttonListener"/>

        <Button
            android:id="@+id/share_button"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="分享"
            android:layout_weight="1"
            android:onClick="buttonListener"/>

    </LinearLayout>

</LinearLayout>

item.xml

<?xml version="1.0" encoding="utf-8"?>
<com.manuelpeinado.multichoiceadapter.view.CheckableLinearLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"

    android:background="@drawable/custom_list_item_background"

    android:orientation="horizontal">
    <!-- 上面必须要用自定义的layout,否则不会有选中的效果!!! -->

    <LinearLayout
        xmlns:android="http://schemas.android.com/apk/res/android"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:padding="16dp"
        android:orientation="horizontal" >

        <TextView
            android:id="@+id/item_textView"
            android:layout_width="200dp"
            android:layout_height="wrap_content"
            android:layout_marginLeft="30dp"
            android:layout_gravity="center_vertical"
            android:textColor="#000000"
            android:layout_weight="1"
            android:textAppearance="?android:attr/textAppearanceLarge" />

        <!--
            CheckBox中一定要写上focusable="false"否则无法相应点击事件
            @android:id/checkbox这个也一定要写,如果不写的话点击checkbox就不会触发多选状态
            如果你不想要选中checkbox就触发多选状态的话,这里可以用自己定义的id就行
         -->
        <CheckBox
            android:id="@android:id/checkbox"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_gravity="center_vertical|right"
            android:layout_weight="1"
            android:layout_marginLeft="40dp"
            android:textColor="#000000"
            android:focusable="false"
            android:visibility="gone"
            android:text="" />

    </LinearLayout>
</com.manuelpeinado.multichoiceadapter.view.CheckableLinearLayout> 

view_header.xml

这个是用于给listview添加一个头部视图的,顺带学习下listview添加顶部视图的方法呗~

<?xml version="1.0" encoding="utf-8"?>
<TextView xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:paddingBottom="16dp"
    android:paddingTop="16dp"
    android:text="这是给listview添加的顶部视图"
    android:gravity="center_horizontal"
    android:textStyle="bold" />

二、继承类然后实现

可以看见这里关于ActionMode有关的回调方法我都是空实现

    private class MultiBaseAdapter extends MultiChoiceBaseAdapter{

        private String[] mData;

        /**
         * 构造函数
         * @param savedInstanceState
         */
        public MultiBaseAdapter(Bundle savedInstanceState,String[] data) {
            super(savedInstanceState);
            // TODO 自动生成的构造函数存根
            mData = data;
        }

        @Override
        public boolean onCreateActionMode(ActionMode mode, Menu menu) {
            return true;
        }

        @Override
        public boolean onActionItemClicked(ActionMode mode, MenuItem item) {
            return false;
        }

        @Override
        public boolean onPrepareActionMode(ActionMode arg0, Menu arg1) {
            // TODO 自动生成的方法存根
            return false;
        }

        /**
         * 看适配器中有多少元素需要加载
         */
        @Override
        public int getCount() {
            // TODO 自动生成的方法存根
            return mData.length;
        }

        /**
         * 通过position来得到相应的item,这里返回object对象
         */
        @Override
        public Object getItem(int position) {
            // TODO 自动生成的方法存根
            return mData[position];
        }

        /**
         * 通过position得到id
         */
        @Override
        public long getItemId(int position) {
            // TODO 自动生成的方法存根
            return position;
        }

        /**
         * 返回item的view对象
         */
        @Override
        protected View getViewImpl(int position, View convertView, ViewGroup parent) {
            if (convertView == null) {
                int layout = R.layout.item;
                LayoutInflater inflater = LayoutInflater.from(getContext());
                convertView = inflater.inflate(layout, parent, false);
            }
            ViewGroup group = (ViewGroup)convertView;
            ((TextView)group.findViewById(R.id.item_textView)).setText(mData[position]);
            ((CheckBox)group.findViewById(android.R.id.checkbox)).setVisibility(View.VISIBLE);
            return group;
        }

    }

三、配置适配器和相应的监听器

listView.addHeaderView(createHeaderView(), null, false);是给listview添加头视图的方法,传入false表示头图不能点击

    private MultiBaseAdapter adapter;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        // TODO 自动生成的方法存根
        super.onCreate(savedInstanceState);
        setContentView(R.layout.listview_normal_layout);

        String[] data = {"android","ios","wp","c++",
                 "java","c#","javascript","vb",
                 "delphi","PB","ASP","SQL"};

        final LinearLayout settingLL = (LinearLayout)findViewById(R.id.setting_linearLayout);
        settingLL.setVisibility(View.GONE);

        ListView listView = (ListView)findViewById(R.id.normal_listView);

        /**
         * 给listview顶部添加2个额外视图,设置顶部视图不可点击
         */
        listView.addHeaderView(createHeaderView(), null, false);
        listView.addHeaderView(createHeaderView(), null, false);
        //实例化适配器
        adapter = new MultiBaseAdapter(savedInstanceState, data);
        //添加视图
        adapter.setAdapterView(listView);
        //设置不显示actionMode
        adapter.showActionMode(false);
        //点击事件
        adapter.setOnItemClickListener(new MyItemClick(adapter));
        //监听选中的状态
        adapter.setOnSelectedStateChangeListener(new OnSelectedStateChangeListener() {
            /**
             * checkedItemCount = 已经选中的item数目
             */
            @Override
            public void onSelectedStateChanged(int checkedItemCount) {
                if (checkedItemCount != 0) {
                    settingLL.setVisibility(View.VISIBLE);
                }
                else {
                    settingLL.setVisibility(View.GONE);
                }
            }
        });
    }
    private View createHeaderView() {
        return LayoutInflater.from(this).inflate(R.layout.view_header, null);
    }

点击事件的监听器:

    /**
     * @author:Jack Tony
     * @tips  :点击事件的监听器
     * @date  :2014-10-20
     */
    private class MyItemClick implements OnItemClickListener{

        private MultiChoiceBaseAdapter mAdapter;

        public MyItemClick(MultiChoiceBaseAdapter adapter) {
            mAdapter = adapter;
        }

        @Override
        public void onItemClick(AdapterView<?> arg0, View arg1, int position, long arg3) {
            //因为在顶部添加了2个视图,所以这里的位置要下移两位

            Toast.makeText(getApplicationContext(), "点击了: " + mAdapter.getItem(position - 2), Toast.LENGTH_SHORT).show();
        }

    }

四、用回调方法来优化

    @Override
    protected void onSaveInstanceState(Bundle outState) {
        adapter.save(outState);
    }

    @Override
    public boolean onKeyDown(int keyCode, KeyEvent event) {
        if (keyCode == KeyEvent.KEYCODE_BACK ) {
            if (adapter.getCheckedItemCount() > 0) {
                cancleAll(adapter);
                return true;
            }
        }
        return super.onKeyDown(keyCode, event);
    }  
    public void buttonListener(View v) {
        switch (v.getId()) {
        case R.id.selectAll_button:
            selectAll(adapter);
            break;
        case R.id.cancle_button:
            cancleAll(adapter);
            break;
        case R.id.delete_button:
            delectItems(adapter);
            break;
        case R.id.share_button:
            Toast.makeText(getApplicationContext(), "分享"+Arrays.toString(getSelectedItems(adapter)), 1).show();
            cancleAll(adapter);
            break;
        default:
            break;
        }
    }

    /**
     * 全选
     * @param adapter
     */
    private void selectAll(MultiChoiceBaseAdapter  adapter) {
        for (int i = 0; i < adapter.getCount(); ++i) {
            adapter.setItemChecked(i, true);
        }
    }

    /**
     * 取消所有选择效果
     * @param adapter
     */
    private void cancleAll(MultiChoiceBaseAdapter  adapter) {
        for (int i = 0; i < adapter.getCount(); ++i) {
            adapter.setItemChecked(i, false);
        }
    }

    /**
     * 得到已经选中的items
     * @param adapter
     * @return
     */
    private String[] getSelectedItems(MultiChoiceBaseAdapter adapter) {
        //得到选中的items
        Set<Long> selection = adapter.getCheckedItems();
        String[] items = new String[selection.size()];
        int i = 0;
        for (long position : selection) {
            items[i++] = (String)adapter.getItem((int)position);
        }
        return items;
    }

    /**
     * 删除已经选中的items
     * @param adapter
     */
    private void delectItems(MultiChoiceBaseAdapter  adapter) {
        //通过判断名字来remove掉这些items
        // TODO:删除某些元素,因为这里的数据源是String[]所以没有链表那样好删除,就没去实现。
        //实际中:推荐用链表来动态删除元素,在删除时需要注意的是最好以唯一的id,如position来进行删除
        cancleAll(adapter);
    }

搞定了!!!

全部代码:

package com.kale.multichoiceadaptertest;

import java.util.Arrays;
import java.util.Set;

import android.app.Activity;
import android.os.Bundle;
import android.support.v7.view.ActionMode;
import android.view.KeyEvent;
import android.view.LayoutInflater;
import android.view.Menu;
import android.view.MenuItem;
import android.view.View;
import android.view.ViewGroup;
import android.widget.AdapterView;
import android.widget.AdapterView.OnItemClickListener;
import android.widget.CheckBox;
import android.widget.LinearLayout;
import android.widget.ListView;
import android.widget.TextView;
import android.widget.Toast;

import com.manuelpeinado.multichoiceadapter.base.OnSelectedStateChangeListener;
import com.manuelpeinado.multichoiceadapter.compat.MultiChoiceBaseAdapter;

public class BaseAdapterNormalTestActivity extends Activity{

    private MultiBaseAdapter adapter;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        // TODO 自动生成的方法存根
        super.onCreate(savedInstanceState);
        setContentView(R.layout.listview_normal_layout);

        String[] data = {"android","ios","wp","c++",
                 "java","c#","javascript","vb",
                 "delphi","PB","ASP","SQL"};

        final LinearLayout settingLL = (LinearLayout)findViewById(R.id.setting_linearLayout);
        settingLL.setVisibility(View.GONE);

        ListView listView = (ListView)findViewById(R.id.normal_listView);

        /**
         * 给listview顶部添加2个额外视图,设置顶部视图不可点击
         */
        listView.addHeaderView(createHeaderView(), null, false);
        listView.addHeaderView(createHeaderView(), null, false);
        //实例化适配器
        adapter = new MultiBaseAdapter(savedInstanceState, data);
        //添加视图
        adapter.setAdapterView(listView);
        //设置不显示actionMode
        adapter.showActionMode(false);
        //点击事件
        adapter.setOnItemClickListener(new MyItemClick(adapter));
        //监听选中的状态
        adapter.setOnSelectedStateChangeListener(new OnSelectedStateChangeListener() {
            /**
             * checkedItemCount = 已经选中的item数目
             */
            @Override
            public void onSelectedStateChanged(int checkedItemCount) {
                if (checkedItemCount != 0) {
                    settingLL.setVisibility(View.VISIBLE);
                }
                else {
                    settingLL.setVisibility(View.GONE);
                }
            }
        });
    }

    @Override
    protected void onSaveInstanceState(Bundle outState) {
        adapter.save(outState);
    }

    @Override
    public boolean onKeyDown(int keyCode, KeyEvent event) {
        if (keyCode == KeyEvent.KEYCODE_BACK ) {
            if (adapter.getCheckedItemCount() > 0) {
                cancleAll(adapter);
                return true;
            }
        }
        return super.onKeyDown(keyCode, event);
    }  

    private View createHeaderView() {
        return LayoutInflater.from(this).inflate(R.layout.view_header, null);
    }

    private class MultiBaseAdapter extends MultiChoiceBaseAdapter{

        private String[] mData;

        /**
         * 构造函数
         * @param savedInstanceState
         */
        public MultiBaseAdapter(Bundle savedInstanceState,String[] data) {
            super(savedInstanceState);
            // TODO 自动生成的构造函数存根
            mData = data;
        }

        @Override
        public boolean onCreateActionMode(ActionMode mode, Menu menu) {
            return true;
        }

        @Override
        public boolean onActionItemClicked(ActionMode mode, MenuItem item) {
            return false;
        }

        @Override
        public boolean onPrepareActionMode(ActionMode arg0, Menu arg1) {
            // TODO 自动生成的方法存根
            return false;
        }

        /**
         * 看适配器中有多少元素需要加载
         */
        @Override
        public int getCount() {
            // TODO 自动生成的方法存根
            return mData.length;
        }

        /**
         * 通过position来得到相应的item,这里返回object对象
         */
        @Override
        public Object getItem(int position) {
            // TODO 自动生成的方法存根
            return mData[position];
        }

        /**
         * 通过position得到id
         */
        @Override
        public long getItemId(int position) {
            // TODO 自动生成的方法存根
            return position;
        }

        /**
         * 返回item的view对象
         */
        @Override
        protected View getViewImpl(int position, View convertView, ViewGroup parent) {
            if (convertView == null) {
                int layout = R.layout.item;
                LayoutInflater inflater = LayoutInflater.from(getContext());
                convertView = inflater.inflate(layout, parent, false);
            }
            ViewGroup group = (ViewGroup)convertView;
            ((TextView)group.findViewById(R.id.item_textView)).setText(mData[position]);
            ((CheckBox)group.findViewById(android.R.id.checkbox)).setVisibility(View.VISIBLE);
            return group;
        }

    }

    public void buttonListener(View v) {
        switch (v.getId()) {
        case R.id.selectAll_button:
            selectAll(adapter);
            break;
        case R.id.cancle_button:
            cancleAll(adapter);
            break;
        case R.id.delete_button:
            delectItems(adapter);
            break;
        case R.id.share_button:
            Toast.makeText(getApplicationContext(), "分享"+Arrays.toString(getSelectedItems(adapter)), 1).show();
            cancleAll(adapter);
            break;
        default:
            break;
        }
    }

    /**
     * 全选
     * @param adapter
     */
    private void selectAll(MultiChoiceBaseAdapter  adapter) {
        for (int i = 0; i < adapter.getCount(); ++i) {
            adapter.setItemChecked(i, true);
        }
    }

    /**
     * 取消所有选择效果
     * @param adapter
     */
    private void cancleAll(MultiChoiceBaseAdapter  adapter) {
        for (int i = 0; i < adapter.getCount(); ++i) {
            adapter.setItemChecked(i, false);
        }
    }

    /**
     * 得到已经选中的items
     * @param adapter
     * @return
     */
    private String[] getSelectedItems(MultiChoiceBaseAdapter adapter) {
        //得到选中的items
        Set<Long> selection = adapter.getCheckedItems();
        String[] items = new String[selection.size()];
        int i = 0;
        for (long position : selection) {
            items[i++] = (String)adapter.getItem((int)position);
        }
        return items;
    }

    /**
     * 删除已经选中的items
     * @param adapter
     */
    private void delectItems(MultiChoiceBaseAdapter  adapter) {
        //通过判断名字来remove掉这些items
        // TODO:删除某些元素,因为这里的数据源是String[]所以没有链表那样好删除,就没去实现。
        //实际中:推荐用链表来动态删除元素,在删除时需要注意的是最好以唯一的id,如position来进行删除
        cancleAll(adapter);
    }

    /**
     * @author:Jack Tony
     * @tips  :点击事件的监听器
     * @date  :2014-10-20
     */
    private class MyItemClick implements OnItemClickListener{

        private MultiChoiceBaseAdapter mAdapter;

        public MyItemClick(MultiChoiceBaseAdapter adapter) {
            mAdapter = adapter;
        }

        @Override
        public void onItemClick(AdapterView<?> arg0, View arg1, int position, long arg3) {
            //因为在顶部添加了2个视图,所以这里的位置要下移两位

            Toast.makeText(getApplicationContext(), "点击了: " + mAdapter.getItem(position - 2), Toast.LENGTH_SHORT).show();
        }

    }
}

开源项目MultiChoiceAdapter详解(一)——概要介绍

开源项目MultiChoiceAdapter详解(二)——MultiChoiceArrayAdapter的使用

开源项目MultiChoiceAdapter详解(三)——MulitChoiceNormalArrayAdapter的使用

开源项目MultiChoiceAdapter详解(四)——MultiChoiceBaseAdapter的使用

开源项目MultiChoiceAdapter详解(五)——可扩展的MultiChoiceBaseAdapter

开源项目MultiChoiceAdapter详解(六)——GridView和MultiChoiceBaseAdapter配合使用

时间: 2024-08-24 07:35:49

开源项目MultiChoiceAdapter详解(五)——可扩展的MultiChoiceBaseAdapter的相关文章

开源项目MultiChoiceAdapter详解(六)——GridView和MultiChoiceBaseAdapter配合使用

这篇其实没啥重要的,主要就算是个总结吧. 一. 这里实现的是类似于上图的多图选择的效果.关键在于item布局文件的写法.这也就是这个框架奇葩的一点,莫名其妙的要在一个自定义控件里面再放一个自定义的控件,如果不这样就出不了选中的效果.分析下原因是这里整个item被图片所覆盖了,仅仅设置一个有选择效果的父控件会被图片所覆盖,所以还得用一个可以选中的iamgeview进行替换imageview. 下面就是这个布局文件 item_gridview.xml <?xml version="1.0&qu

开源项目MultiChoiceAdapter详解(三)——MulitChoiceNormalArrayAdapter的使用

MulitChoiceNormalArrayAdapter是我自己定义的一个类,其实就是实现了MulitChoiceArrayAdapter,为什么做这个简单的实现类呢,因为这样我们在不用ActionMode的时候就不用每次要写一个类来继承MulitChoiceArrayAdapter了,直接实例化MulitChoiceNormalArrayAdapter即可.下面贴一个compat包下的MulitChoiceNormalArrayAdapter的源码. MulitChoiceNormalArr

开源项目MultiChoiceAdapter详解(四)——MultiChoiceBaseAdapter的使用

MultiChoiceBaseAdapter是一个可以多选的BaseAdapter,使用的方式相比来说扩展性更强! 使用方式: 1.布局文件 2.写一个类继承MultiChoiceBaseAdapter 3.实现内部的各个方法 4.设置数据源和视图 5.完成保存的回调方法 一.布局文件 listview_actionmode_layout.xml <?xml version="1.0" encoding="utf-8"?> <LinearLayou

开源项目MultiChoiceAdapter详解(二)——MultiChoiceArrayAdapter的使用

MultiChoiceArrayAdapter其实就是可以多选的ArrayAdapter了,ArrayAdpter我们已经很熟悉了.MultiChoiceArrayAdapter这个类是抽象类,所以使用前必须要继承.下面是使用MultiChoiceArrayAdapter的步骤: 0.用自定义的控件来写一个layout 1.写一个类来继承MultiChoiceArrayAdapter 2.实例化这个类 3.用setAdapterView()来设置要加载适配器的控件. 4.写上保存的方法 @Ove

开源项目使用详解过程

开源项目使用详解过程Q-Q:971-041-894定位[手機系列找回刪除等信息]这件事说来也是巧了,也算是他们上级领导的矛盾吧,因为公司与另一个公司之前有一点合同上的纠葛,所以老板在发函以前之前让我跟对方公司现承认一下,看是不是真的要闹到这个地步,因为我是担任这个项目的,相对在状况上回对比的了解,那个时分我刚好在外面,所以只能用自己的手机打以前了,这也没什么,不便是知会一声吗,当然的,两头的利益不是那么快就可以到达的,所以发函这个工作仍是进行了请问我的苹果5s手机被偷关还能机么定位吗.手机没电话

开源项目PullToRefresh详解(二)——PullToRefreshGridView

这里介绍的是PullToRefreshGridView的使用方法,和之前的PullToRefreshListView方法如出一辙,因为这个开源项目模块化很棒,所以很容易实现.等于说我们可以按照之前使用控件的方式来操作,不用考虑其他的问题. 思路: 1.写布局文件,放入可以下拉刷新的控件 2.找到下拉刷新的控件,设置监听器,并且在刷新方法中开启一个异步任务来操作 3.通过这个下拉刷新控件的getRefreshableView()方法来得到GridView对象,按照正常的操作来设置适配器 4.在异步

Android 开源项目DiskLruCache 详解

有兴趣的同学可以读完这篇文章以后 可以看看这个硬盘缓存和volley 或者是其他 图片缓存框架中使用的硬盘缓存有什么异同点. 讲道理的话,其实硬盘缓存这个模块并不难写,难就难在 你要考虑到百分之0.1的那种情况,比如写文件的时候 手机突然没电了 之类的,你得保证文件正确性,唯一性等等.今天就来看看这个DiskLruCache是怎么实现这些内容的. 用法大家就自己去谷歌吧,在这里提一句,DiskLruCache 在4.0以上的源码中被编译到了platform 下面的libcore.io这个包路径下

开源项目PullToRefresh详解(三)——PullToRefreshScrollView

和前几篇文章一样,这里还是先设置布局文件,然后找到这个控件.只不过这里要简单很多. 1.布局文件 <?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="fill_parent" android:la

转:Windows下的PHP开发环境搭建——PHP线程安全与非线程安全、Apache版本选择,及详解五种运行模式。

原文来自于:http://www.ituring.com.cn/article/128439 Windows下的PHP开发环境搭建——PHP线程安全与非线程安全.Apache版本选择,及详解五种运行模式. 今天为在Windows下建立PHP开发环境,在考虑下载何种PHP版本时,遭遇一些让我困惑的情况,为了解决这些困惑,不出意料地牵扯出更多让我困惑的问题. 为了将这些困惑一网打尽,我花了一下午加一晚上的时间查阅了大量资料,并做了一番实验后,终于把这些困惑全都搞得清清楚楚了. 说实话,之所以花了这么