仿贴吧内容下的简单ListView嵌套GridView

ListView嵌套GridView的简单实例

我的项目想实现一个listview里面的每个item都嵌套一个GridView,顶部还有主题等内容,如
这里写图片描述
总所周知,关于ListView嵌套GridView,最主要问题莫过于嵌套状态下滑动冲突问题,具体怎么解决,喜欢冗长无注释的代码的,请点击这里这篇文章跟其他的都大同小异了,不过在缺少注释的情况下,我发现了一点点小问题:

/**
 * 创建日期:2017/3/21.
 * 说明:构造方法会根据你的SDK最低版本不同而要求不同,如18的至少必须重写前
 * 三个,第四个SDK要求最低21,可以不重写,但前三个必须写,否则这个自定义的
 * MyGridView 在运用时会报错;
 * onMeasure:自定义GridView 控件,实现无法滚动(拖动)的方法
 */
public class MyGridView extends GridView {

    public MyGridView(Context context) {
        super(context);
    }

    public MyGridView(Context context, AttributeSet attrs) {
        super(context, attrs);
    }

    public MyGridView(Context context, AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
    }

//public MyGridView(Context context, AttributeSet attrs, int   //defStyleAttr, int defStyleRes) {
//       super(context, attrs, defStyleAttr, defStyleRes);
//    }

    /**
     * 重写测量GridView的内容空间(有多少数据内容)
     * @param widthMeasureSpec 占用宽度
     * @param heightMeasureSpec 占用高度
     */
    @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        // expandSpec:拓展空间,其中MeasureSpec.AT_MOST为“最大模式”
        // AT_MOST:最大模式,比喻为布局里的match_parent
        // EXACTLY:精确模式,比喻为布局里的"50dp"
        // UNSPECIFIED:未指定模式,比喻为布局里的wrap_content
        int expandSpec = MeasureSpec.makeMeasureSpec(Integer.MAX_VALUE >> 2, MeasureSpec.AT_MOST);
        super.onMeasure(widthMeasureSpec, expandSpec);
    }

    /**
     * 重写事件分发:因为两个都是ViewGroup,这个方法不知道是否可行,
     * 有兴趣的朋友自行脑补
     */
//    @Override
//    public boolean dispatchTouchEvent(MotionEvent ev) {
//        if (ev.getAction() == MotionEvent.ACTION_MOVE){
//            //返回true直接结束当前事件消费
//            return true;
//        }
//        return super.dispatchTouchEvent(ev);
//    }

     /**
     * 如果是嵌套在ScollView中的,则这样写
     * 设置是否有ScrollBar,当要在ScollView中显示时,应当设置为
     * false。 默认为 true
     */
//    boolean haveScrollbars = false;
//    public void setHaveScrollbar(boolean haveScrollbar) {
//        this.haveScrollbar = haveScrollbar;
//    }
//    @Override
//    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
//        if (haveScrollbars == false) {
//            int expandSpec = MeasureSpec.makeMeasureSpec(Integer.MAX_VALUE >> 2, MeasureSpec.AT_MOST);
//            super.onMeasure(widthMeasureSpec, expandSpec);
//        } else {
//            super.onMeasure(widthMeasureSpec, heightMeasureSpec);
//        }
//    }

}

上面这个GridView就算自定义好了,接下来我们简单贴上listview的适配器主要方法getview(…)

 @Override
    public View getView(final int position, View convertView, ViewGroup parent) {
        ViewHolder holder = null;
        if (convertView == null) {
            holder = new ViewHolder();
            convertView = mInflater.inflate(R.layout.item_lv_gridview, null);
            holder.mMyGridView = (MyGridView) convertView.findViewById(R.id.gridView_show_controller);
            convertView.setTag(holder);
        } else {
            holder = (ViewHolder) convertView.getTag();
        }

//注意:这是重要的地方
//鉴于我们想让每个item下的GridView都能独立,因此只能通过new 出适配器来
//单独定义每个item,这样才能让每个listview的item内容都有不同的GridView
        DevicesAdapter devicesAdapter = new DevicesAdapter(mContext);
        devicesAdapter.setDevicesList(deviceList);
        holder.mMyGridView.setAdapter(devicesAdapter);
        holder.mMyGridView.setOnItemClickListener(new AdapterView.OnItemClickListener() {
            @Override
            public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
//监听最后一个item(position==list.size()-1),
//动态更改其作用功能(比如增加一条数据,或者blablabla...)
                if (position == parent.getCount() - 1) {
                    mItemListener.onAddClick();
                } else {
   mItemListener.onDeviceItemClick(deviceList.get(position));
                }
            }
        });

        return convertView;
    }

    static class ViewHolder {
        MyGridView mMyGridView;
    }

只要再定义GridView的适配器就大功告成了(适配器相信到这时候大家应该都很熟了,我就不注释了哈,请原谅我比较懒)

public class DevicesAdapter extends BaseAdapter {

    private final LayoutInflater mInflater;
    private ArrayList<DeviceInfos> devicesList;

    public DevicesAdapter(Context context) {
        mInflater = LayoutInflater.from(context);
    }

    public void setDevicesList(ArrayList<DeviceInfos> devicesList) {
        this.devicesList = devicesList;
    }

    @Override
    public int getCount() {
        if (devicesList == null) {
            return 1;
        }
        return devicesList.size() + 1;
    }

    @Override
    public Object getItem(int position) {
        return null;
    }

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

    @Override
    public View getView(int position, View convertView, ViewGroup parent) {
        ViewHolder holder = null;
        if (convertView == null) {
            convertView = mInflater.inflate(R.layout.item_grid_view, null);
            holder = new ViewHolder();
            holder.mImageView = (ImageView) convertView.findViewById(R.id.img_controller);
            holder.mTextView = (TextView) convertView.findViewById(R.id.tv_controller_name);
            convertView.setTag(holder);
        } else {
            holder = (ViewHolder) convertView.getTag();
        }
        if (devicesList!=null && position < devicesList.size()) {
            String childDeviceName = devicesList.get(position).getChildDeviceName();
            holder.mImageView.setImageResource(R.drawable.huajidadi);
            holder.mTextView.setText(childDeviceName);
        } else {
            holder.mImageView.setImageResource(R.drawable.add);
            holder.mTextView.setVisibility(View.GONE);
        }
        return convertView;
    }

    static class ViewHolder {
        ImageView mImageView;
        TextView mTextView;
    }
}

布局我就不给了,就简单的几个item的布局,请同学们自己定义吧…

总结:解决滑动冲突的方法或者涉及点击、移动等属性的,用自定义控件再重写事件分发,可以很好的解决。但是对于同样是ViewGroup就没办法像View的组件那样,通过调用onInterceptTouchEvent(MotionEvent ev)进行拦截了,因此只能另寻他法,而这关键就是重写onMeasure(int widthMeasureSpec, int heightMeasureSpec)这个方法,使得GridView的“控件大小”被固定,这样就不会与另一个滑动事件冲突了。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值