Android 二級(jí)列表控件ExpandableListView 的簡(jiǎn)單使用

簡(jiǎn)述

在Android 開發(fā)中多多少少會(huì)碰到需要二級(jí)列表源请,之前已經(jīng)寫過一篇RecyclerView 二級(jí)列表 其實(shí)現(xiàn)方式是通過根據(jù)不同ViewHolder 來顯示是一級(jí)還是二級(jí)列表冒黑,想起谷歌官方自己就有自帶二級(jí)列表控件ExpandableListView,如果不需要復(fù)雜效果秉继,建議直接使用官方控件杠茬,故有了今天這一篇文章。

老規(guī)矩宁赤,先上圖:


視頻錄制效果.gif

如圖所見滑動(dòng)出屏幕或者點(diǎn)擊checkbox時(shí)會(huì)出現(xiàn)錯(cuò)位等一些問題也解決了,具體方法請(qǐng)往下瀏覽(文末附上github 地址)

頁面布局

布局很簡(jiǎn)單佛猛,ExpandableListView 加底部一個(gè)Button擦秽,直接上布局截圖缩搅,相信各位能看懂


activity_main.png

由于其控件會(huì)默認(rèn)自帶箭頭(如下圖)


默認(rèn)自帶的指示器箭頭.png

我們可以通過XML中在ExpandableListView控件加上

     android:groupIndicator="@null"

取消掉其自帶的指示器箭頭究飞,當(dāng)然除了在xml上瘟栖,也可通過在代碼中寓涨,當(dāng)綁定完控件后調(diào)用代碼也可實(shí)現(xiàn)取消效果

 expandableListView.setGroupIndicator(null);

接下來是我們重點(diǎn)要研究的適配器StudentExpandableAdapter,繼承并重寫了BaseExpandableListAdapter這個(gè)類的相關(guān)函數(shù),其中注釋我已經(jīng)詳細(xì)寫在代碼中樟插,若是不懂或者寫錯(cuò)食拜,希望各位可以交流或指出,大家一起加深對(duì)其認(rèn)識(shí)奏篙。

public class StudentExpandableAdapter extends BaseExpandableListAdapter {

    private Context context;
    private List<DataEntity> dataEntity;
    private CheckBoxListener checkBoxListener;

    public StudentExpandableAdapter(Context context, List<DataEntity> dataEntity) {
        this.context = context;
        this.dataEntity = dataEntity;
    }

    /**
     * 獲取組的數(shù)目
     *
     * @return 返回一級(jí)列表組的數(shù)量
     */
    @Override
    public int getGroupCount() {
        return dataEntity == null ? 0 : dataEntity.size();
    }

    /**
     * 獲取指定組中的子節(jié)點(diǎn)數(shù)量
     *
     * @param groupPosition 子元素組所在的位置
     * @return 返回指定組中的子數(shù)量
     */
    @Override
    public int getChildrenCount(int groupPosition) {
        return dataEntity.get(groupPosition).getChildrenDataList().size();
    }

    /**
     * 獲取與給定組相關(guān)聯(lián)的對(duì)象
     *
     * @param groupPosition 子元素組所在的位置
     * @return 返回指定組的子數(shù)據(jù)
     */
    @Override
    public Object getGroup(int groupPosition) {
        return dataEntity.get(groupPosition).getTitle();
    }


    /**
     * 獲取與給定組中的給定子元素關(guān)聯(lián)的數(shù)據(jù)
     *
     * @param groupPosition 子元素組所在的位置
     * @param childPosition 子元素的位置
     * @return 返回子元素的對(duì)象
     */
    @Override
    public Object getChild(int groupPosition, int childPosition) {
        return dataEntity.get(groupPosition).getChildrenDataList().get(childPosition);
    }

    /**
     * 獲取組在給定位置的ID(唯一的)
     *
     * @param groupPosition 子元素組所在的位置
     * @return 返回關(guān)聯(lián)組ID
     */
    @Override
    public long getGroupId(int groupPosition) {
        return groupPosition;
    }


    /**
     * 獲取給定組中給定子元素的ID(唯一的)
     *
     * @param groupPosition 子元素組所在的位置
     * @param childPosition 子元素的位置
     * @return 返回子元素關(guān)聯(lián)的ID
     */
    @Override
    public long getChildId(int groupPosition, int childPosition) {
        return childPosition;
    }

    /**
     * @return 確定id 是否總是指向同一個(gè)對(duì)象
     */
    @Override
    public boolean hasStableIds() {
        return true;
    }

    /**
     * @return 返回指定組的對(duì)應(yīng)的視圖 (一級(jí)列表樣式)
     */
    @Override
    public View getGroupView(final int groupPosition, boolean isExpanded, View convertView, ViewGroup parent) {
        ParentHolder parentHolder;
        if (convertView == null) {
            convertView = LayoutInflater.from(context).inflate(R.layout.parent_item, null);
            parentHolder = new ParentHolder();
            parentHolder.tvParent = convertView.findViewById(R.id.tv_parent);
            parentHolder.img_right = convertView.findViewById(R.id.img_right);
            convertView.setTag(parentHolder);
        } else {
            parentHolder = (ParentHolder) convertView.getTag();
        }
        parentHolder.tvParent.setText(dataEntity.get(groupPosition).getTitle());
     

        //共用一個(gè)右箭頭,如果展開則順時(shí)針旋轉(zhuǎn)90°選擇盹靴,否則不旋轉(zhuǎn)
        if (isExpanded) parentHolder.img_right.setRotation(90F);
        else parentHolder.img_right.setRotation(0F);

        return convertView;
    }

    /**
     * @return 返回指定位置對(duì)應(yīng)子視圖的視圖
     */
    @Override
    public View getChildView(final int groupPosition, final int childPosition, boolean isLastChild, View convertView, ViewGroup parent) {
        final ChildrenHolder childrenHolder;
        if (convertView == null) {
            convertView = LayoutInflater.from(context).inflate(R.layout.childrens_item, null);
            childrenHolder = new ChildrenHolder();
            childrenHolder.tvChild = convertView.findViewById(R.id.tv_child);
            childrenHolder.checkBox = convertView.findViewById(R.id.checkbox);
            convertView.setTag(childrenHolder);
        } else {
            childrenHolder = (ChildrenHolder) convertView.getTag();
        }


        //Log.e("666","班級(jí):"+dataEntity.get(groupPosition).getTitle()+"    學(xué)生:"+dataEntity.get(groupPosition).getChildrenDataList().get(childPosition).getSubContent()+"   isChecked:"+dataEntity.get(groupPosition).getChildrenDataList().get(childPosition).isSelect());
        childrenHolder.checkBox.setChecked(dataEntity.get(groupPosition).getChildrenDataList().get(childPosition).isSelect());
        childrenHolder.tvChild.setText(dataEntity.get(groupPosition).getChildrenDataList().get(childPosition).getSubContent());
        childrenHolder.checkBox.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                boolean isChecked = !dataEntity.get(groupPosition).getChildrenDataList().get(childPosition).isSelect();
                dataEntity.get(groupPosition).getChildrenDataList().get(childPosition).setSelect(!isChecked);
                Log.e("groupPosition:" + groupPosition, "childPosition:" + childPosition + " isChecked:" + isChecked);
                checkBoxListener.checkStateListener(groupPosition, childPosition, isChecked);
            }
        });


        return convertView;
    }

    /**
     * 指定位置的子元素是否可選
     *
     * @param groupPosition 子元素組所在的位置
     * @param childPosition 子元素的位置
     * @return 返回是否可選
     */

    @Override
    public boolean isChildSelectable(int groupPosition, int childPosition) {
        return true;
    }


    class ParentHolder {
        TextView tvParent;
        ImageView img_right;
    }


    class ChildrenHolder {
        TextView tvChild;
        CheckBox checkBox;
    }


    /**
     * 用于提供對(duì)外復(fù)選框修改通知接口
     */
    public interface CheckBoxListener {
        void checkStateListener(int groupPosition, int childPosition, boolean isChecked);
    }

    public void setCheckBoxListener(CheckBoxListener checkBoxListener) {
        this.checkBoxListener = checkBoxListener;
    }


    /**
     * 用于刷新更新后的數(shù)據(jù)
     */
    public void reFreshData(List<DataEntity> dataEntity) {
        this.dataEntity = dataEntity;
        notifyDataSetChanged();
    }


}

父布局使用的xml:


parent_item.png

子布局使用的xml:


childrens_item.png

注意在getGroupView中g(shù)etGroupView的控件不能設(shè)置一些搶占焦點(diǎn)的事件或?qū)傩哉耄琰c(diǎn)擊事件或者在代碼布局里設(shè)置了focusable屬性為true,都會(huì)導(dǎo)致無法展開子列表稿静。

getChildView中子視圖梭冠,checkBox不調(diào)用setOnCheckedChangeListener是由于可能會(huì)因?yàn)檫x中的組展開觸發(fā)而導(dǎo)致混亂,這邊改為使用setOnClickListener改备,這是一種折中方案控漠,因?yàn)闋顟B(tài)的改變不是來自事件onClick(也就是你點(diǎn)擊了不一定知道狀態(tài)是否成功更改),OnCheckChangedListener則是監(jiān)聽CheckBox的狀態(tài)悬钳,成功后回調(diào)盐捷。

實(shí)體類DataEntity 代碼如下

public class DataEntity {
    private String title;//一級(jí)列表內(nèi)容
    private List<ChildrenData> childrenDataList;


    public DataEntity(String title, List<ChildrenData> childrenDataList) {
        this.title = title;
        this.childrenDataList = childrenDataList;
    }

    public List<ChildrenData> getChildrenDataList() {
        return childrenDataList;
    }

    public String getTitle() {
        return title;
    }

    public void setTitle(String title) {
        this.title = title;
    }

    public void setChildrenDataList(List<ChildrenData> childrenDataList) {
        this.childrenDataList = childrenDataList;
    }

    public static class ChildrenData{
        private String subContent;//子內(nèi)容
        private boolean select;//是否選中

        public ChildrenData(String subContent, boolean select) {
            this.subContent = subContent;
            this.select = select;
        }

        public String getSubContent() {
            return subContent;
        }

        public void setSubContent(String subContent) {
            this.subContent = subContent;
        }

        public boolean isSelect() {
            return select;
        }

        public void setSelect(boolean select) {
            this.select = select;
        }
    }
}

最后是在我們的主界面中實(shí)現(xiàn)代碼

public class MainActivity extends AppCompatActivity {
    private ExpandableListView expandableListView;
    private Button btn_select;
    private boolean selectAll;
    private List<DataEntity>  dataEntityList=new ArrayList<>();
    private StudentExpandableAdapter  studentExpandableAdapter;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        expandableListView =findViewById(R.id.listView);
        btn_select=findViewById(R.id.btn_select);
        initData();
        initAdapter();
        setOnClickEvent();
    }

    private void initData() {

        for(int i=0;i<5;i++){
            List<DataEntity.ChildrenData> childrenData=new ArrayList<>();
            for(int j=0;j<8;j++){
                DataEntity.ChildrenData children=new DataEntity.ChildrenData("學(xué)生"+(j+1),false);
                childrenData.add(children);
            }
            DataEntity dataEntity=new DataEntity((i+1)+"班",childrenData);
            dataEntityList.add(dataEntity);
        }
    }

    private void setOnClickEvent() {
        btn_select.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                selectAll=!selectAll;
                if(selectAll){
                    //遍歷設(shè)置全選
                    for(int i=0;i<dataEntityList.size();i++){
                        for(int j=0;j<dataEntityList.get(i).getChildrenDataList().size();j++){
                            dataEntityList.get(i).getChildrenDataList().get(j).setSelect(true);
                        }
                    }
                }else {
                    //遍歷設(shè)置取消全選
                    for(int i=0;i<dataEntityList.size();i++){
                        for(int j=0;j<dataEntityList.get(i).getChildrenDataList().size();j++){
                            dataEntityList.get(i).getChildrenDataList().get(j).setSelect(false);
                        }
                    }
                }


                studentExpandableAdapter.reFreshData(dataEntityList);

                btn_select.setText(selectAll? "取消全選":"全選");

            }
        });
    }

    private void initAdapter() {
        studentExpandableAdapter=new StudentExpandableAdapter(this,dataEntityList);
        expandableListView.setAdapter(studentExpandableAdapter);

        studentExpandableAdapter.setCheckBoxListener(new StudentExpandableAdapter.CheckBoxListener() {
            @Override
            public void checkStateListener(int groupPosition, int childPosition, boolean isChecked) {
                Log.e("MainActivity","isChecked:"+isChecked);
                dataEntityList.get(groupPosition).getChildrenDataList().get(childPosition).setSelect(isChecked);
                studentExpandableAdapter.reFreshData(dataEntityList);
            }
        });


        /**
         * 默認(rèn)展開某個(gè)item
         * */
        //expandableListView.expandGroup(1);


    }
}

至此,簡(jiǎn)單講完了ExpandableListView 的基礎(chǔ)使用默勾,希望能對(duì)小伙伴們提供一點(diǎn)幫助碉渡。
最后附上該項(xiàng)目的github

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市母剥,隨后出現(xiàn)的幾起案子滞诺,更是在濱河造成了極大的恐慌,老刑警劉巖环疼,帶你破解...
    沈念sama閱讀 206,378評(píng)論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件习霹,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡炫隶,警方通過查閱死者的電腦和手機(jī)淋叶,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,356評(píng)論 2 382
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來伪阶,“玉大人煞檩,你說我怎么就攤上這事处嫌。” “怎么了形娇?”我有些...
    開封第一講書人閱讀 152,702評(píng)論 0 342
  • 文/不壞的土叔 我叫張陵锰霜,是天一觀的道長(zhǎng)。 經(jīng)常有香客問我桐早,道長(zhǎng),這世上最難降的妖魔是什么友存? 我笑而不...
    開封第一講書人閱讀 55,259評(píng)論 1 279
  • 正文 為了忘掉前任,我火速辦了婚禮屡立,結(jié)果婚禮上搀军,老公的妹妹穿的比我還像新娘。我一直安慰自己罩句,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,263評(píng)論 5 371
  • 文/花漫 我一把揭開白布乳愉。 她就那樣靜靜地躺著屯远,像睡著了一般。 火紅的嫁衣襯著肌膚如雪慨丐。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,036評(píng)論 1 285
  • 那天备闲,我揣著相機(jī)與錄音崩溪,去河邊找鬼斩松。 笑死,一個(gè)胖子當(dāng)著我的面吹牛惧盹,可吹牛的內(nèi)容都是我干的瞪讼。 我是一名探鬼主播符欠,決...
    沈念sama閱讀 38,349評(píng)論 3 400
  • 文/蒼蘭香墨 我猛地睜開眼瓶埋,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來了养筒?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 36,979評(píng)論 0 259
  • 序言:老撾萬榮一對(duì)情侶失蹤挤悉,失蹤者是張志新(化名)和其女友劉穎巫湘,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體尚氛,經(jīng)...
    沈念sama閱讀 43,469評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 35,938評(píng)論 2 323
  • 正文 我和宋清朗相戀三年畏梆,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了奈懒。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,059評(píng)論 1 333
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡溜畅,死狀恐怖极祸,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情遥金,我是刑警寧澤,帶...
    沈念sama閱讀 33,703評(píng)論 4 323
  • 正文 年R本政府宣布选泻,位于F島的核電站,受9級(jí)特大地震影響页眯,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜窝撵,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,257評(píng)論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望短曾。 院中可真熱鬧,春花似錦错英、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,262評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至碉考,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間锌仅,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,485評(píng)論 1 262
  • 我被黑心中介騙來泰國(guó)打工热芹, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留惨撇,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 45,501評(píng)論 2 354
  • 正文 我出身青樓魁衙,卻偏偏與公主長(zhǎng)得像报腔,于是被迫代替她去往敵國(guó)和親剖淀。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,792評(píng)論 2 345

推薦閱讀更多精彩內(nèi)容