欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  移动技术

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

程序员文章站 2023-11-19 21:42:34
listview嵌套gridview的简单实例 我的项目想实现一个listview里面的每个item都嵌套一个gridview,顶部还有主题等内容,如 总所周...

listview嵌套gridview的简单实例

我的项目想实现一个listview里面的每个item都嵌套一个gridview,顶部还有主题等内容,如

Android仿贴吧内容下的简单ListView嵌套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的“控件大小”被固定,这样就不会与另一个滑动事件冲突了。

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