本文实例为大家分享了Android二级横向菜单的实现过程.效果如上图:

这种横向的二级菜单在很多的app都有所应用.效果看起来还是非常的美观的.也算是项目需要,自己也就学了一下这个效果,首先说一下逻辑.实现的方式其实并不是很难..只不过逻辑上可能有点复杂.原理其实就是一个按钮.当触发按钮的时候弹出PopWindow.PopWindow由两个ListView构成..对两个ListView适当的适配.就可以实现这个效果了..

实现这种效果可以有两种不同的方式..一种是直接在布局文件layout.xml中写..最上方的可以是一个按钮.也可以是多个按钮..多个按钮就可以使用RadioGroup去实现..下方则采用ScrollView去实现也是可以的..

不过我还是说一下第二种方式..直接用Java去写这个布局..通过使用自定义控件的方式实现这个效果..既然是自定义,那么首先我们需要继承一个布局.布局可以使用LinearLayout或者RelativeLayout.

setValue()方法..

setValue()方法是自定义的方法..主要是用于加载布局.以及在布局当中添加相关的View.没有加载任何的xml文件..

/**

* @param textArray: ListView中item对应的text值的集合..

* @param viewArray: 当前Layout中需要加入的View..

* */

@SuppressLint("ResourceAsColor")

public void setValue(ArrayList textArray, ArrayList viewArray) {

if (mContext == null) {

return;

}

LayoutInflater inflater = (LayoutInflater) mContext.getSystemService(Context.LAYOUT_INFLATER_SERVICE);

mTextList = textArray;

for (int i = 0; i < viewArray.size(); i++) {

//这里就添加了一个View..

final RelativeLayout r = new RelativeLayout(mContext);

int maxHeight = (int) (displayHeight * 0.5); //定义布局的高度..

RelativeLayout.LayoutParams rl = new RelativeLayout.LayoutParams(RelativeLayout.LayoutParams.MATCH_PARENT, maxHeight);

r.addView(viewArray.get(i), rl); //在布局中添加View并指定参数

mViewList.add(r);

r.setTag(SMALL);

//定义最上方的按钮,并在布局中添加这个按钮。并设置按钮的text

ToggleButton tButton = (ToggleButton) inflater.inflate(R.layout.toggle_button, this, false);

addView(tButton);

mToggleList.add(tButton);

tButton.setTag(i);

tButton.setText(mTextList.get(i));

//用于实现当PopWindow显示时.再次点击收回PopWindow

r.setOnClickListener(new OnClickListener() {

@Override

public void onClick(View v) {

onPressBack();

}

});

r.setBackgroundColor(mContext.getResources().getColor(R.color.popup_main_background));

//当按钮被点击后需要触发的监听

tButton.setOnClickListener(new OnClickListener() {

@Override

public void onClick(View view) {

ToggleButton tButton = (ToggleButton) view;

/** 如果当前点击的按钮与上次的点击不同.则设置当前的按钮处于点击状态 */

if (selectedButton != null && selectedButton != tButton) {

selectedButton.setChecked(false);

}

selectedButton = tButton;

selectPosition = (Integer) selectedButton.getTag();

/** 按钮被点击后,需要触发对应的监听事件.*/

startAnimation();

if (mOnButtonClickListener != null && tButton.isChecked()) {

mOnButtonClickListener.onClick(selectPosition);

}

}

});

}

}

那么设置完了布局的样式后..只有一个ToggleButton按钮.点击后没有任何的效果.我们需要去定义一个新的View视图.用于点击按钮后需要显示的弹出窗.那么这个弹出窗也需要自定义..

弹出窗则采用两个ListView的形式进行显示.在布局中将两个ListView进行添加.对每一个ListView设置相应的适配器.然后将这个View添加到上面的主View当中.就可以实现当button被点击后,弹出窗在下方进行显示的效果..

ChildView() 弹出窗View的布局实现方式..

这里定义了这个View,并完成相应的初始化操作.设置对应的适配器也就完成了..

package com.example.view;

import java.util.ArrayList;

import java.util.LinkedList;

import android.content.Context;

import android.util.AttributeSet;

import android.util.SparseArray;

import android.view.LayoutInflater;

import android.view.View;

import android.widget.LinearLayout;

import android.widget.ListView;

import com.example.adapter.TextAdapter;

import com.example.expandtabview.R;

public class ChildView extends LinearLayout {

private ListView regionListView; //主ListView

private ListView plateListView; //子ListView

//主ListView每一个Item对应的text

private String LeftFaString[] = new String[] { "美食", "快餐小吃", "火锅", "海鲜/烧烤",

"特色菜", "香锅/烤鱼", "地方菜", "东南亚菜", "西餐", "日韩料理" };

//子ListView每一个Item对应的text..采用了二维数组的实现方式..

private String LeftCh1String[][] = new String[][] {

{ "全部" },

{ "全部", "中式简餐", "地方小吃", "盖浇饭", "米粉米线", "面馆", "麻辣烫", "黄焖鸡米饭",

"鸭脖卤味", "饺子馄饨", "炸鸡炸串", "包子/粥", "零食", "生煎锅贴", "冒菜" },

{ "全部", "其他火锅" }, { "全部", "小龙虾" }, { "全部" }, { "全部", "香锅", "烤鱼" },

{ "全部", "鲁菜", "川菜", "其他" }, { "全部" },

{ "全部", "意面披萨", "西式快餐", "其他西餐" }, { "全部", "韩式简餐", "韩国料理" } };

//添加主ListView中的数据信息

private ArrayList groups = new ArrayList();

//添加子ListView中的数据信息

private LinkedList childrenItem = new LinkedList();

//稀疏数组

private SparseArray> children = new SparseArray>();

//为ListView设置适配器

private TextAdapter plateListViewAdapter;

private TextAdapter earaListViewAdapter;

//监听事件的设置

private OnSelectListener mOnSelectListener;

private int tEaraPosition = 0; //用于保存当前主ListView被点击的Item对应的Position.

private int tBlockPosition = 0; //用于保存当前子ListView被点击的Item对应的Position.

private String showString = "";

public ChildView(Context context) {

super(context);

init(context);

}

public ChildView(Context context, AttributeSet attrs) {

super(context, attrs);

init(context);

}

private void init(Context context) {

LayoutInflater inflater = (LayoutInflater) context

.getSystemService(Context.LAYOUT_INFLATER_SERVICE);

//加载布局,绑定ID.

inflater.inflate(R.layout.view_region, this, true);

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

plateListView = (ListView) findViewById(R.id.listView2);

//初始化ListView中每一个item对应的text

for(int i=0;i<10;i++){

groups.add(LeftFaString[i]);

LinkedList tItem = new LinkedList();

for(int j=0;j

tItem.add(LeftCh1String[i][j]);

}

children.put(i, tItem);

}

//主ListView列表项的适配器

earaListViewAdapter = new TextAdapter(context, groups,

R.drawable.choose,

R.drawable.choose_eara_item_selector);

earaListViewAdapter.setTextSize(12);

earaListViewAdapter.setSelectedPositionNoNotify(tEaraPosition);

regionListView.setAdapter(earaListViewAdapter);

earaListViewAdapter

.setOnItemClickListener(new TextAdapter.OnItemClickListener() {

@Override

public void onItemClick(View view, int position) {

if (position < children.size()) {

childrenItem.clear();

//获取这一页的所有数据信息..然后唤醒适配器更新数据

childrenItem.addAll(children.get(position));

plateListViewAdapter.notifyDataSetChanged();

}

}

});

if (tEaraPosition < children.size())

childrenItem.addAll(children.get(tEaraPosition));

//子ListView的适配器

plateListViewAdapter = new TextAdapter(context, childrenItem,

R.drawable.choose_item_right,

R.drawable.choose_plate_item_selector);

plateListViewAdapter.setTextSize(12);

plateListViewAdapter.setSelectedPositionNoNotify(tBlockPosition);

plateListView.setAdapter(plateListViewAdapter);

//设置当Item被点击后触发的监听.

plateListViewAdapter

.setOnItemClickListener(new TextAdapter.OnItemClickListener() {

@Override

public void onItemClick(View view, final int position) {

//获取被点击的Item的文字数据

showString = childrenItem.get(position);

if (mOnSelectListener != null) {

mOnSelectListener.getValue(showString);

}

}

});

if (tBlockPosition < childrenItem.size())

showString = childrenItem.get(tBlockPosition);

setDefaultSelect();

}

//设置当前Item的Position.

public void setDefaultSelect() {

//默认选择的Item项

regionListView.setSelection(tEaraPosition);

plateListView.setSelection(tBlockPosition);

}

public String getShowText() {

return showString;

}

public void setOnSelectListener(OnSelectListener onSelectListener) {

mOnSelectListener = onSelectListener;

}

public interface OnSelectListener {

public void getValue(String showText);

}

}

那么最后就剩下适配器了..

ArrayAdapter

这里使用了ArrayAdapter适配器.继承与BaseAdapter.可以用于显示文本数据..我们也都知道,适配器必须要实现的方法就是getView()方法了..那么我们就简单的看一下这个方法..

@SuppressLint("ResourceAsColor") @SuppressWarnings("deprecation")

@Override

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

TextView view;

if (convertView == null) {

view = (TextView) LayoutInflater.from(mContext).inflate(R.layout.choose_item, parent, false);

} else {

view = (TextView) convertView;

}

view.setTag(position);

String mString = "";

if (mListData != null) {

if (position < mListData.size()) {

mString = mListData.get(position);

}

} else if (mArrayData != null) {

if (position < mArrayData.length) {

mString = mArrayData[position];

}

}

if (mString.contains("不限"))

view.setText("不限");

else

view.setText(mString);

view.setTextSize(TypedValue.COMPLEX_UNIT_SP,textSize);

if (selectedText != null && selectedText.equals(mString)) {

view.setBackgroundDrawable(selectedDrawble);//设置选中的背景图片

} else {

view.setBackgroundDrawable(mContext.getResources().getDrawable(normalDrawbleId));//设置未选中状态背景图片

}

view.setPadding(20, 0, 0, 0);

view.setOnClickListener(onClickListener);

return view;

}

适配的工作还是非常的简单的.仅仅一个TextView就可以搞定了.当然我们也可以写一个比较复杂的样式.在一个Layout内部定义一些复杂的控件.就能够实现更好的效果.

最后再MainActivity中的布局文件中加载这个控件,简单的做一些初始化操作就可以实现了..

package com.example.expandtabview;

import java.util.ArrayList;

import android.app.Activity;

import android.os.Bundle;

import android.view.View;

import android.widget.Toast;

import com.example.view.ExpandTabView;

import com.example.view.ChildView;

public class MainActivity extends Activity {

private ExpandTabView expandTabView;

private ArrayList mViewArray = new ArrayList();

private ChildView viewLeft;

@Override

protected void onCreate(Bundle savedInstanceState) {

super.onCreate(savedInstanceState);

setContentView(R.layout.activity_main);

initView();

initVaule();

initListener();

}

private void initView() {

//初始化控件

expandTabView = (ExpandTabView) findViewById(R.id.expandtab_view);

viewLeft = new ChildView(this);

}

private void initVaule() {

mViewArray.add(viewLeft);

//设置顶部数据信息

ArrayList mTextArray = new ArrayList();

mTextArray.add("全部");

expandTabView.setValue(mTextArray, mViewArray);

expandTabView.setTitle(viewLeft.getShowText(), 0);

}

private void initListener() {

viewLeft.setOnSelectListener(new ChildView.OnSelectListener() {

@Override

public void getValue(String showText) {

onRefresh(viewLeft,showText);

}

});

}

//视图被点击后刷新数据

private void onRefresh(View view, String showText) {

expandTabView.onPressBack();

int position = getPositon(view);

if (position >= 0 && !expandTabView.getTitle(position).equals(showText)) {

expandTabView.setTitle(showText, position);

}

Toast.makeText(MainActivity.this, showText, Toast.LENGTH_SHORT).show();

}

//获取当前的view

private int getPositon(View tView) {

for (int i = 0; i < mViewArray.size(); i++) {

if (mViewArray.get(i) == tView) {

return i;

}

}

return -1;

}

}

这里只是贴了一些核心代码.其他的涉及的一些不重要的代码就不在这里粘贴了..最后放一张图片流程.方便大家去理解.最后给出源代码.

放一个源代码提供下载,方便去理解这个过程:Android实现横向二级菜单

以上就是本文的全部内容,希望对大家的学习有所帮助。

android 横向多级列表,Android实现横向二级菜单相关推荐

  1. Android 横向滚动列表 (类似横向ListView)

    示例: 布局代码***.xml: <?xml version="1.0" encoding="utf-8"?> <LinearLayout x ...

  2. android+完美的列表,android完美讲义.pdf

    android完美讲义 王骋 QQ Phone Mail • 感谢我的同事,陈操(横拓开源技术副总),在 他提供的资料乊上做了迚一步总结提炼. • 参考资料android sdk开发文档,IBM开发者 ...

  3. android双联动列表,Android Fragment实现列表和内容联动

    在平板上经常能看到这种的情况:左边是一个列表,右边是列表项对应的内容,当点击某一个列表时,右边内容区也会随之改变.下面使用fragment简单的demo: 思路:在mainactivity定义一个回调 ...

  4. android开发评论列表,Android类Reddit循环评论列表开发

    最近因为项目需求,需要开发类似Reddit里面无限循环的评论列表,于是就开始研究其实现方式和可用性,reddit评论如下: reddit评论 从最开始的印象看,我们可以看到这是一个树状列表,如果没有限 ...

  5. android 新闻功能列表,android listview实现新闻列表展示效果

    本文实例为大家分享了android listview列表展示效果的具体代码,供大家参考,具体内容如下 1.封装一些新闻数据 2.使用listview展示出来 3.设置条目点击事件,点击后跳转浏览器查看 ...

  6. android+获取相册列表,android 获取相册列表的实现(二)

    该项目实现的功能如下: 获取手机相册,点击每个相册之后进入该相册的图片列表界面,在图片列表界面可以实现图片多选,然后进入所选择的图片界面,在该界面内可 该项目实现的功能如下: 获取手机相册,点击每个相 ...

  7. android可扩展列表,android ExpandableListView可扩展列表

    http://leiwuluan.iteye.com/blog/1508356 先看一效果图. 列表中要有 图片和文字: 所以我们要实现一个自定义的   适配器. 介绍一个类:BaseExpandab ...

  8. android 余额宝收益列表,Android 仿支付宝中的余额宝收益进度条

    一. 看效果 二.上代码 package com.framework.widget; import android.app.Activity; import android.content.Conte ...

  9. android 新闻功能列表,android根据提供的接口获取新闻列表

    查询新闻列表: 接口名称:GetNewsInfo 接口参数:(string account(账号), string pwd(密码), string newstype(新闻类型[图文新闻(传入4),图片 ...

最新文章

  1. ECMAScript 6中的let和const关键词
  2. python 重写__repr__与__str__函数
  3. php errno 28,php7.28 编译出错 一直通不过去
  4. boost::fusion::reverse用法的测试程序
  5. Delphi面向对象学习随笔七:COM
  6. Spring AOP源码解析——AOP动态代理原理和实现方式
  7. 阿里腾讯前端一面小结
  8. leetcode article
  9. 转:全卷积网络(FCN)与图像分割
  10. IDEA上传本地项目到SVN
  11. ios无痕埋点_掌握数据生命周期:初识数据埋点
  12. linux系统win,Windows10安装运行linux系统的方法
  13. POJ 1984 Navigation Nightmare
  14. java用户登录窗口怎么删除_JAVA图形界面用户登录后如何关闭登录的对话框
  15. 【JavaScript】JavaScript高级教程
  16. DOS获取注册表键值
  17. SSM框架整合,简单案例
  18. VMware vSphere Esxi官网下载页面链接
  19. Android悬浮窗的简单实现
  20. BGP Dampening Cyrus

热门文章

  1. oracle 启动mrp进程,DataGuard MRP进程crash的处理案例 ORA-01111
  2. 校外培训行业迎来强监管,“教育+区块链”新模式试图解决行业痼疾
  3. HTML入门笔记7-粗体、斜体和下划线标签
  4. 龙石数据成功入选TGO鲲鹏会企业图谱(第一版)
  5. 基于YOLOv5的疲劳检测,可检测是否玩手机,抽烟,喝水,眨眼,打哈欠等
  6. 输入法9键 VS 26键,哪个更适合?
  7. 联想win10电池养护模式在哪_老鸟操作win10笔记本电池保养模式的具体解决伎俩...
  8. 计算机专业未来就业方向有哪些?
  9. 【青少年编程】【Scratch】01 运动模块
  10. liunx 对日志文件进行模糊匹配搜索,并查出指定范围的内容