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

  MultiChoiceArrayAdapter其实就是可以多选的ArrayAdapter了,ArrayAdpter我们已经很熟悉了。MultiChoiceArrayAdapter这个类是抽象类,所以使用前必须要继承。下面是使用MultiChoiceArrayAdapter的步骤:

0.用自定义的控件来写一个layout

1.写一个类来继承MultiChoiceArrayAdapter

2.实例化这个类

3.用setAdapterView()来设置要加载适配器的控件。

4.写上保存的方法

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

零、写布局文件

listview_actionmode_layout.xml

没什么特别的,就一个listview

<?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/actionMode_listView"
        android:layout_width="match_parent"
        android:layout_height="wrap_content">
    </ListView>

</LinearLayout>

item.xml

注意这里要用自己定义的实现了Checkable接口的控件

<?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" />

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

二、继承MultiChoiceAdapter

这个是重点,每个方法中实现了不同的功能。其中discardSelectedItems()是自己写的,用来移除已经选中的items。

注意:onCreateActionMode()必须返回true,否则报错!

    private class TestAdapter extends MultiChoiceArrayAdapter<String>{

        String tag = getClass().getSimpleName().toString();

        /**
         * @param savedInstanceState
         * @param context
         * @param resource:Item的布局文件,即 R.layout.xxx
         * @param textViewResourceId:显示文字的textview的id,即R.id.xxxx
         * @param objects:一个String数组
         */
        public TestAdapter(Bundle savedInstanceState, Context context,
                int resource, int textViewResourceId, String[] objects) {
            //为了避免错误,这里将String数组转为ArrayList对象
            this(savedInstanceState,context,resource,textViewResourceId,
                    new ArrayList<String>(Arrays.asList(objects)));
        }

        /**
         * @param savedInstanceState
         * @param context
         * @param resource:Item的布局文件,即 R.layout.xxx
         * @param textViewResourceId:显示文字的textview的id,即R.id.xxxx
         * @param items:一个list<String>对象
         */
        public TestAdapter(Bundle savedInstanceState, Context context,
                int resource, int textViewResourceId, List<String> items) {
            super(savedInstanceState, context, resource, textViewResourceId, items);
        }

        /**
         * 传入actionMode对象,进行设置图标和操作
         * 注意要返回true!
         */
        @Override
        public boolean onCreateActionMode(ActionMode mode, Menu menu) {
            Log.i(tag, "onCreateActionMode");
            MenuInflater inflater = mode.getMenuInflater();
            inflater.inflate(R.menu.my_action_mode, menu);
            //这里必须要写true,否则会报空指针!!!
            return true;
        }

        /**
         * 进行AndroidMode上图标操作的设置
         */
        @Override
        public boolean onActionItemClicked(ActionMode mode, MenuItem item) {
            if (item.getItemId() == R.id.menu_share) {
                Toast.makeText(getContext(), "分享了" + getCheckedItemCount()+"个item", Toast.LENGTH_SHORT).show();
                return true;
            }
            if (item.getItemId() == R.id.menu_discard) {
                //丢弃选中的items
                discardSelectedItems();
                return true;
            }
            return false;
        }

        @Override
        public boolean onPrepareActionMode(ActionMode arg0, Menu arg1) {
            Log.i(tag, "onPrepareActionMode");
            return false;
        }

        /**
         * (非必须)
         * 设置点击、选中效果,非必须。如果你在item的layout设置了 android:background那么下面就不用进行设置背景了
         * 个人推荐在item的layout中设置背景色,example:android:background="@drawable/custom_list_item_background"
         */
        /*@Override
        protected View getViewImpl(int position, View convertView, ViewGroup parent) {
            //Log.i(tag, "getViewImpl");
            View view = super.getViewImpl(position, convertView, parent);
            view.setBackgroundResource(R.drawable.custom_list_item_background);
            return view;
        }*/

        /**
         * 从适配器中移除某些item
         */
        private void discardSelectedItems() {
            //得到选中的items
            Set<Long> selection = getCheckedItems();
            String[] items = new String[selection.size()];
            int i = 0;
            for (long position : selection) {
                items[i++] = getItem((int)position);
            }

            //通过判断名字来remove掉这些items
            for (String item : items) {
                /**
                 * 这里用remove时要注意传入适配器的不能是String[] items对象。否则会报错
                 * 这里我已经在构造函数中进行了处理,传入String数组也不会出错了~
                 */
                remove(item);
            }

            finishActionMode();
        }    

    }

这里在继承的时候其实就顺带实现了ActionMode的callback接口的内容了,如果对ActionMode熟悉的人一定很容易理解。

二、实例化这个类

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

        ListView actionModelistView = (ListView)findViewById(R.id.actionMode_listView);

        actionModeAdapter = new TestAdapter(savedInstanceState, this,R.layout.item, R.id.item_textView, data);

三、设置控件和监听器

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.listview_actionmode_layout);

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

        ListView actionModelistView = (ListView)findViewById(R.id.actionMode_listView);

        actionModeAdapter = new TestAdapter(savedInstanceState, this,R.layout.item, R.id.item_textView, data);
        actionModeAdapter.setAdapterView(actionModelistView);
        actionModeAdapter.setOnItemClickListener(new MyItemClick(actionModeAdapter));

        //listView.setAdapter(adapter);//不这么用啦

    }

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

        private TestAdapter mAdapter;

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

        @Override
        public void onItemClick(AdapterView<?> arg0, View arg1, int position, long arg3) {
            Toast.makeText(getApplicationContext(), "点击了: " + mAdapter.getItem(position), Toast.LENGTH_SHORT).show();
        }

    }

四、保存的方法

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

于是大功告成!下面是全部代码:

ArrayAdapterTestActivity.java (这个类没用兼容模式,如果用兼容模式请参考顶部的注释,其实就是导入不同的包即可)

package com.kale.multichoiceadaptertest;

import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.Set;

import android.app.Activity;
import android.content.Context;
import android.os.Bundle;
import android.util.Log;
import android.view.ActionMode;
import android.view.Menu;
import android.view.MenuInflater;
import android.view.MenuItem;
import android.view.View;
import android.widget.AdapterView;
import android.widget.AdapterView.OnItemClickListener;
import android.widget.ListView;
import android.widget.Toast;

import com.manuelpeinado.multichoiceadapter.normal.MultiChoiceArrayAdapter;

/**
 * 如果是4.0以上的版本可以直接用activity,不用actionbarActivity,导入的时候导入
 * import android.view.ActionMode;
 * import com.manuelpeinado.multichoiceadapter.normal.MultiChoiceArrayAdapter;
 * 把
 * import android.support.v7.view.ActionMode;
 * import com.manuelpeinado.multichoiceadapter.compat.MultiChoiceArrayAdapter;
 * 删掉
 */

/**
 * @author:Jack Tony
 * @tips  :ActionMode和ArrayAdapter配合实现多选
 * @date  :2014-10-21
 */
public class ArrayAdapterTestActivity extends Activity{

    TestAdapter actionModeAdapter;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.listview_actionmode_layout);

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

        ListView actionModelistView = (ListView)findViewById(R.id.actionMode_listView);

        actionModeAdapter = new TestAdapter(savedInstanceState, this,R.layout.item, R.id.item_textView, data);
        actionModeAdapter.setAdapterView(actionModelistView);
        actionModeAdapter.setOnItemClickListener(new MyItemClick(actionModeAdapter));

        //listView.setAdapter(adapter);//不这么用啦

    }

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

    private class TestAdapter extends MultiChoiceArrayAdapter<String>{

        String tag = getClass().getSimpleName().toString();

        /**
         * @param savedInstanceState
         * @param context
         * @param resource:Item的布局文件,即 R.layout.xxx
         * @param textViewResourceId:显示文字的textview的id,即R.id.xxxx
         * @param objects:一个String数组
         */
        public TestAdapter(Bundle savedInstanceState, Context context,
                int resource, int textViewResourceId, String[] objects) {
            //为了避免错误,这里将String数组转为ArrayList对象
            this(savedInstanceState,context,resource,textViewResourceId,
                    new ArrayList<String>(Arrays.asList(objects)));
        }

        /**
         * @param savedInstanceState
         * @param context
         * @param resource:Item的布局文件,即 R.layout.xxx
         * @param textViewResourceId:显示文字的textview的id,即R.id.xxxx
         * @param items:一个list<String>对象
         */
        public TestAdapter(Bundle savedInstanceState, Context context,
                int resource, int textViewResourceId, List<String> items) {
            super(savedInstanceState, context, resource, textViewResourceId, items);
        }

        /**
         * 传入actionMode对象,进行设置图标和操作
         * 注意要返回true!
         */
        @Override
        public boolean onCreateActionMode(ActionMode mode, Menu menu) {
            Log.i(tag, "onCreateActionMode");
            MenuInflater inflater = mode.getMenuInflater();
            inflater.inflate(R.menu.my_action_mode, menu);
            //这里必须要写true,否则会报空指针!!!
            return true;
        }

        /**
         * 进行AndroidMode上图标操作的设置
         */
        @Override
        public boolean onActionItemClicked(ActionMode mode, MenuItem item) {
            if (item.getItemId() == R.id.menu_share) {
                Toast.makeText(getContext(), "分享了" + getCheckedItemCount()+"个item", Toast.LENGTH_SHORT).show();
                return true;
            }
            if (item.getItemId() == R.id.menu_discard) {
                //丢弃选中的items
                discardSelectedItems();
                return true;
            }
            return false;
        }

        @Override
        public boolean onPrepareActionMode(ActionMode arg0, Menu arg1) {
            Log.i(tag, "onPrepareActionMode");
            return false;
        }

        /**
         * (非必须)
         * 设置点击、选中效果,非必须。如果你在item的layout设置了 android:background那么下面就不用进行设置背景了
         * 个人推荐在item的layout中设置背景色,example:android:background="@drawable/custom_list_item_background"
         */
        /*@Override
        protected View getViewImpl(int position, View convertView, ViewGroup parent) {
            //Log.i(tag, "getViewImpl");
            View view = super.getViewImpl(position, convertView, parent);
            view.setBackgroundResource(R.drawable.custom_list_item_background);
            return view;
        }*/

        /**
         * 从适配器中移除某些item
         */
        private void discardSelectedItems() {
            //得到选中的items
            Set<Long> selection = getCheckedItems();
            String[] items = new String[selection.size()];
            int i = 0;
            for (long position : selection) {
                items[i++] = getItem((int)position);
            }

            //通过判断名字来remove掉这些items
            for (String item : items) {
                /**
                 * 这里用remove时要注意传入适配器的不能是String[] items对象。否则会报错
                 * 这里我已经在构造函数中进行了处理,传入String数组也不会出错了~
                 */
                remove(item);
            }

            finishActionMode();
        }    

    }

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

        private TestAdapter mAdapter;

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

        @Override
        public void onItemClick(AdapterView<?> arg0, View arg1, int position, long arg3) {
            Toast.makeText(getApplicationContext(), "点击了: " + mAdapter.getItem(position), Toast.LENGTH_SHORT).show();
        }

    }

}
时间: 2024-10-19 02:32:03

开源项目MultiChoiceAdapter详解(二)——MultiChoiceArrayAdapter的使用的相关文章

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

上次写到了开源项目MultiChoiceAdapter详解(四)——MultiChoiceBaseAdapter的使用,其实我们仍旧可以不使用ActionMode的,所以这里就写一个自己扩展的方法. 一.布局文件 listview_normal_layout.xml <?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.andr

开源项目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

开源项目使用详解过程

开源项目使用详解过程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

UINavigationController详解二(转)页面切换和SegmentedController

原文出自:http://blog.csdn.net/totogo2010/article/details/7682433,非常感谢. 1.RootView 跳到SecondView 首先我们需要新一个View.新建SecondView,按住Command键然后按N,弹出新建页面,我们新建SecondView 2.为Button 添加点击事件,实现跳转 在RootViewController.xib中和RootViewController.h文件建立连接 在RootViewController.m