Android GridView同一行的Item高度不同,使其高度平齐,自动适应高度最大的item

看到这篇文章,你可能会问————你搞这玩意有个P的意思,用处在哪呢。别急,请看我下面实际遇到的场景。

这是我项目做的一个底部选项弹窗,没经过处理是这样的,可能有些人说就是要这种凌乱的效果,那这篇文章你可以到此结束了。



下面经过行高处理之后是这样的


是不是觉得突然顺眼了很多,来介绍一下具体的实现。

首先在Adapter中传入一个GridView实例,或者直接getView函数里获取ViewGroup强转,这就不用我多说了吧,下面开始我的表演

这是Adapter中ViewHolder关键函数,用于更新item高度

class ViewHolder{
        LinearLayout llView;//视图项
        TextView tvText;//文字

        /**
         * 更新视图高度
         */
        private void updateHeight() {
            //当一个视图create还没有开始绘制,获取不到宽高,但是视图绘制完成,视图树的布局发生改变时,可以被ViewTreeObserver监听到
            itemView.getViewTreeObserver().addOnGlobalLayoutListener(
                    new ViewTreeObserver.OnGlobalLayoutListener() {
                        public void onGlobalLayout() {
                            //当前位置
                            int position = (Integer) tvText.getTag();
                            //当前行
                            int curRow = position / gv.getNumColumns();
                            //当前view高度
                            int height = itemView.getHeight();
                            //初始最大行高度
                            if (mostRowHeight[curRow] == 0){
                                mostRowHeight[curRow] = height;
                            }else {
                                //当前View高度小于最大行高
                                if (height < mostRowHeight[curRow]){
                                    //设置当前View高度=最大行高
                                    itemView.setLayoutParams(new GridView.LayoutParams(
                                            GridView.LayoutParams.MATCH_PARENT,
                                            mostRowHeight[curRow]));
                                    //设置当前View高度>最大行高
                                }else if(height > mostRowHeight[curRow]) {
                                    //更新最大行高
                                    mostRowHeight[curRow] = height;
                                    //更新所在行所有在此之前的view的行高
                                    for (int i = curRow * gv.getNumColumns(); i < getCount() && i < position; i++){
                                        View view = gv.getChildAt(i);
                                        if (view.getHeight() != mostRowHeight[curRow]){
                                            view.setLayoutParams(new GridView.LayoutParams(
                                                    GridView.LayoutParams.MATCH_PARENT,
                                                    mostRowHeight[curRow]));
                                        }
                                    }
                                }
                            }
                            if (position == getCount() - 1){
                                try {
                                    setGridViewHeight();
                                } catch (Exception e) {
                                    e.printStackTrace();
                                }
                            }
                        }
                    });
        }
    }


由于我这里GridView嵌套在一个layout布局中,所以最后要计算GridView总高度,要不然显示不正常,你们可以自己试试

/**
     * 设置gridView的高度
     * @throws Exception
     */
    private void setGridViewHeight() throws Exception{
        ViewGroup.LayoutParams params = gv.getLayoutParams();
        int totalHeight = 0;
        //计算行总高度
        for (int i = 0; i < mostRowHeight.length; i++){
            totalHeight += mostRowHeight[i];
        }

        Class<?> clazz=gv.getClass();
        //利用反射,取得纵向分割线高度
        Field horizontalSpacing = clazz.getDeclaredField("mRequestedHorizontalSpacing");
        horizontalSpacing.setAccessible(true);
        int horizontalBorderHeight = (Integer)horizontalSpacing.get(gv);
        //高度等于行总高 + 行间距
        params.height = totalHeight + horizontalBorderHeight  * (mostRowHeight.length - 1);
        gv.setLayoutParams(params);
    }

在getView中获取gridview实体以及行数的计算


 
   private GridView gv;
    private int[] mostRowHeight;
    public View getView(int position, @Nullable View convertView, @NonNull ViewGroup parent) {
       
        if (convertView == null) {
                  //...viewHolder实例化
                              

                if (gv == null){
                //获取gridview实例
                gv = (GridView) viewGroup;
                //计算行数
                int row = getCount() / gv.getNumColumns();
                if (getCount() % gv.getNumColumns() != 0){
                    row += 1;
                }
                mostRowHeight = new int[row];
            }
                //缓存视图位置
               viewHolder.tvText.setTag(position);
                //设置视图高度
               viewHolder.updateHeight();
        } else {
            viewHolder = (ViewHolder) convertView.getTag();
        }
        //initView(viewHolder, position);视图数据初始化,自行定义
        return convertView;
    }

 
 


发布了3 篇原创文章 · 获赞 6 · 访问量 1万+

猜你喜欢

转载自blog.csdn.net/qq_26851595/article/details/78689090