UGUI进阶知识[二十九]循环GridView

节省内存的常用滑动列表还有一种形式,上下滑动的GridView。这种格式的滑动列表可用于移动设备的背包,仓库,商店UI等数据可能海量从而导致产生特别多但又看不见的UI的情况。

于是基于 UGUI进阶知识[八]循环利用滑动列表的循环ListView工程做了个更改。

主要工作在格子的初始化排布,整体区域的确定, 进入重新展示的循环之后,水平位置和对应数据的确定。

在listview总体逻辑思路确定了之后,根据思路在原代码基础上对应做这种改动还是比较容易的。

请添加图片描述

初始化排布

可见总项数的确定

将之前的水平计算变为水平和垂直计算,垂直计数+1是为了上下滑动的时候,格子位置突变的时候,还没有给用户划到,否则格子闪现会比较很违和。当滑动列表是水平滑动,则考虑水平计数加一。

 /// <summary>
    ///     获得总共需要的项数 包括多余的一项
    /// </summary>
    /// <param name="itemHeight"></param>
    /// <param name="verticalOffset"></param>
    /// <returns></returns>
    private int GetTotalShowItemNum(float itemHeight, float itemWidth,
        float verticalOffset, float horizontalOffset)
    {
        var height = GetComponent<RectTransform>().rect.height;
        var width = GetComponent<RectTransform>().rect.width;

        
        //这里多加的1是在显示区域外部 用来准备显示数据的一项
        verticalItemCount = Mathf.CeilToInt(height / (itemHeight + verticalOffset)) + 1;
        horizontalItemCount = Mathf.CeilToInt(width / (itemWidth + horizontalOffset)) ;

        int totalCount = horizontalItemCount * verticalItemCount;
        
        return totalCount;
    }

总大小确定

总数据数除以单行显示数据数等于数据行数,数据行数*单个item高度等于总高度,另外计入间隙。

 /// <summary>
    ///     设置总内容区域的大小为所有数据加载之后的大小
    ///     只是超出可显示部分是没有LoopGridListItem的
    /// </summary>
    private void SetContentSize()
    {
        int dataRowCount = loopGridListData.GetDataCount() / horizontalItemCount;
        
        var y = dataRowCount * _itemHeight +
                (dataRowCount - 1) * itemHorizonInterval;
         
        
        scrollRectContent.sizeDelta = new Vector2(scrollRectContent.sizeDelta.x, y);
    }

单个item的计算

是比起上面稍微烧脑一点的地方,按照习惯默认从上到下,从左到右id增大。计算的id用来确定位置和读取对应数据。

可见最大id和最小id的计算

在这里插入图片描述

   private void UpdateIdRange(out int toppestLeftId,  out int lowestRightId)
    {
       
        
        //当scrollRectContent对齐ScrollRect的时候 scrollRectContent.anchoredPosition.y是0
        //当scrollRectContent往上超出ScrollRect的时候 scrollRectContent.anchoredPosition.y大于0
        //toppestVisibleId表示 scrollRectContent滑动到当前位置 时候
        //可见的最顶部的item的id 部分可见也是可见 不是超出的item的id
        //FloorToInt是考虑到了顶部只显示部分item的情况 
        //此时toppestVisibleId应该是显示部分的item的时候,最左边的id
        toppestLeftId = Mathf.FloorToInt(scrollRectContent.anchoredPosition.y / 
                                            (itemHeight + itemVerticalInterval));
        
        //这里的计算表示的是最左边的id
        toppestLeftId *= horizontalItemNum;
        
        //lowestVisibleId表示可见的最底部的item的id
        //越往下 item的id越大
        lowestRightId = toppestLeftId + totalItemNum - 1;
        
        
    }

重新计算id

在这里插入图片描述

如图,顶部的三个是超出范围用户已经不可见的,下面是对这三个超出重新分配的id,根据id再去拿数据和设置位置。方法在OverflowRemedy中。

根据id重新设置位置

根据id对水平item个数求余求对应的列数,以及对水平item个数做除法求对应的行数。
然后根据UI的排布计算位置

  private void SetPos()
    {

        int horizonIndex = selfId % horizontalItemNum;
        int verticalIndex = Mathf.FloorToInt(selfId / horizontalItemNum) ;
            
            
        //每个LoopGridListItem所挂物体即预制体的Pivot是在自身的左上角 ,即滑动列表Content的原点在Content的左上角
        //每个LoopGridListItem从上到下的selfId是递增的
        //每个LoopGridListItem从上到下的anchoredPosition的y值是递减的
        //每个LoopGridListItem从左到右的anchoredPosition的x值是递增的
        //所以采用了这个计算方法
        Rect.anchoredPosition = new Vector2(horizonIndex * (itemWidth + itemHorizontalInterval),
            - verticalIndex * (itemHeight + itemVerticalInterval));
    }

对应数据的确定

在id计算好之后,对应的取数据逻辑不用改。

问题

  1. 在锚点设置为某种情况的时候,recttransform是没有任何属性和方法以及他们的组合计算能够获取实例化出来已经在canvas下的UI的宽高,这时的获取的值的规律和SizeDelta是一致的。 考虑的解决方法是重新临时设置锚点,在获取后再设置回去;或者在下一帧的时候再获取值,这明显是来自unity的bug了。补充在了 UGUI进阶知识[四]关于UI定位和适配。这里出问题的UI是实时实例化出来的,并非运行前摆放好。

  2. item的anchor和pivot应该调节成如下图所示,首先anchor应该聚集于一点,否则实例化item 的时候其宽高会因为父物体宽高不同而变得与预设体的宽高不一致,第一点已经验证了在这种情况下实例化UI的时候,不论anchor是否聚集还是散开,没有任何办法能拿到宽高,所以只有在预制体里面人工记录宽高。其他原因参考UGUI进阶知识[四]关于UI定位和适配
    在这里插入图片描述

  3. 在整个items的顶端或者底端,会有没有数据的部分,也就是说,已经滑动到了数据的底部部分没有数据了,这时再往上拖拽,还是会有循环的item出来。目前没有对这部分做处理,最简单的做法是限制ScrollRect在这部分不进行拉出。
    在这里插入图片描述

工程地址

Unity UGUI循环GridView

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/24474.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Tomcat服务器、Servlet生命周期、上传下载文件、使用XHR请求数据、注解使用

文章目录 Servlet认识Tomcat服务器使用Maven创建Web项目创建Servlet探究Servlet的生命周期解读和使用HttpServletWebServlet注解详解使用POST请求完成登陆上传和下载文件下载文件上传文件 使用XHR请求数据重定向与请求转发重定向请求转发 ServletContext对象初始化参数 Servlet…

Office project 2010安装教程

哈喽&#xff0c;大家好。今天一起学习的是project 2010的安装&#xff0c;Microsoft Office project项目管理工具软件&#xff0c;凝集了许多成熟的项目管理现代理论和方法&#xff0c;可以帮助项目管理者实现时间、资源、成本计划、控制。有兴趣的小伙伴也可以来一起试试手。…

在职阿里6年,一个29岁女软件测试工程师的心声

简单的先说一下&#xff0c;坐标杭州&#xff0c;14届本科毕业&#xff0c;算上年前在阿里巴巴的面试&#xff0c;一共有面试了有6家公司&#xff08;因为不想请假&#xff0c;因此只是每个晚上去其他公司面试&#xff0c;所以面试的公司比较少&#xff09; 其中成功的有4家&am…

CSAPP Lab5- MallocLab

实验目标 本实验需要用c语言实现一个动态的存储分配器&#xff0c;也就是你自己版本的malloc&#xff0c;free&#xff0c;realloc函数。 实验步骤 tar xvf malloclab-handout.tar解压文件 我们需要修改的唯一文件是mm.c&#xff0c;包含如下几个需要实现的函数 int mm_ini…

c++调用dll出现LNK2001 无法解析的外部符号

先说说下正常的dll。 动态库显试调用一般3个文件.h .lib .dll &#xff0c;隐式调用 只需要2个文件:.h&#xff08;函数定义&#xff09; .dll 静态库2个文件&#xff1a;.h .lib 先说C正常dll显式调用 #include "BYD_MES/MES2Interface.h" //#include 是以当前…

Android 12.0下拉状态栏通知栏的通知设置默认展开

1.概述 在12.0的产品定制化中,对于SystemUI的定制也是常用的功能,而在下拉状态栏中的通知栏部分也是极其重要的部分,每条通知实时更新在通知栏部分,由于通知栏高度的限制,每条通知是默认收缩的,功能开发需要要求通知默认展开,所以就要从通知的加载流程分析 如图: 2.…

【Java基础篇】运算符

作者简介&#xff1a; 辭七七&#xff0c;目前大一&#xff0c;正在学习C/C&#xff0c;Java&#xff0c;Python等 作者主页&#xff1a; 七七的个人主页 文章收录专栏&#xff1a;Java.SE&#xff0c;本专栏主要讲解运算符&#xff0c;程序逻辑控制&#xff0c;方法的使用&…

多态的原理、单继承和多继承的虚函数表、以及虚函数表的打印。

一、多态原理 1、下面这个结果是多少&#xff1f; class A { public:virtual void func(){cout << "func()" << endl;}private:int _a 1; };int main() {printf("%d\n", sizeof(A));return 0; } 是 4&#xff1f;8&#xff1f;还是多少&am…

MVC 接收不到参数? —— 看我如何给你安排得明明白白

文章结构 问题背景&#xff1a;问题处理总结 问题背景&#xff1a; 现有如下代码&#xff1a; PostMapping(value "/payment/create") ResponseBody public CommonResult create(Payment payment) {}乍眼看去是不是很好&#xff0c;至少没啥问题很自然&#xff0c…

什么是日志关联

什么是日志关联 日志关联是一种分析来自不同源的日志数据以识别事件模式的技术。它用于更好地了解网络的活动&#xff0c;从而有效地保护网络免受漏洞和威胁。 日志关联是日志管理过程的关键部分。收集和存储日志后&#xff0c;集中式日志服务器将执行分析以检测特定事件。日…

LC-3 机器码编程实验

一、实验目的 分析和理解试验指定的需解决问题。利用LC-3的机器代码设计实现相关程序。通过LC-3仿真器调试和运行相关程序并得到正确的结果。 二、实验内容 利用LC-3的机器代码计算一个16位的字中有多少位是“1”&#xff0c;程序从x3000开始&#xff0c;需计算的字存储在x3…

c++ 11标准模板(STL) std::map(九)

定义于头文件<map> template< class Key, class T, class Compare std::less<Key>, class Allocator std::allocator<std::pair<const Key, T> > > class map;(1)namespace pmr { template <class Key, class T, clas…

知识图谱实战应用12-食谱领域智能问答系统,实现菜谱问答

大家好,我是微学AI,今天给大家介绍一下知识图谱实战应用12-食谱领域智能问答系统,实现菜谱问答,本项目基于py2neo和neo4j图数据库,将知识图谱应用于菜谱领域。通过构建菜谱知识图谱,实现简单的菜谱食材问答系统。用户可以通过问答系统,快速获取简单的菜谱食材信息。 一…

悦灵犀-全新的智能AI工具

最近一段时间&#xff0c;人工智能再次成为人类创新的焦点&#xff0c;不得不说&#xff0c;人工智能正在以一种全新的方式改变人们的生活&#xff0c;这是一个以大模型为核心的人工智能新时代&#xff0c;大模型的出现让千行百业将迎来新的机遇。 悦享星光作为国内高新技术企…

VC GDI双缓冲绘图

VC GDI双缓冲绘图 VC GDI双缓冲绘图创建内存DC和内存图片&#xff0c;缺一不可最好是封装一下内存绘制绘制效果 关键是不闪烁PS 重绘机制 VC GDI双缓冲绘图 双缓冲绘图&#xff0c;知道这个知识点&#xff0c;每次用的时候还得踩一遍坑&#xff0c;真是服&#xff0c;总结记录…

如何实现 ESP 设备多证书管理?

设置特定分区存储证书文件多证书文件管理证书格式转换证书下载使用证书文件 1、设置特定分区存储证书文件 在项目工程下分区表文件下定义证书分区表文件&#xff0c;如下&#xff1a; 如上&#xff0c;转换的 certificate.bin 下载地址就为 0x41000证书分区文件的大小可不做设…

Typora改变字体颜色

方法一&#xff1a;下载AutoHotkey并创建快捷键的方法&#xff08;推荐&#xff09; 第一步&#xff1a;在官网&#xff08;https://www.autohotkey.com/&#xff09;下载 AutoHotkey并傻瓜式安装&#xff0c;安装在任意盘符下均可&#xff1b; 第二步&#xff1a;在安装目录…

JavaScript 基础 DOM (三)

日期对象 实例化 获得当前时间 const date new Date() 获得指定时间 const date1 new Date( 指定时间) 方法 // 1. 实例化const date new Date();// 2. 调用时间对象方法// 通过方法分别获取年、月、日&#xff0c;时、分、秒const year date.getFullYear(); // 四位年份 时…

docker的基本相关知识和操作

镜像相关操作命令&#xff1a; 访问DockerHub搜索镜像&#xff0c;https://hub.docker.com/ 查看本地镜像&#xff1a;docker images 搜索镜像 docker search redis &#xff08;搜索redis&#xff09; 拉取镜像&#xff1a;docker pull redis &#xff08;默认版本&#x…

【Python】列表和字典

知识目录 一、写在前面✨二、列表应用三、字典应用四、总结撒花&#x1f60a; 一、写在前面✨ 大家好&#xff01;我是初心&#xff0c;希望我们一路走来能坚守初心&#xff01; 今天跟大家分享的文章是 Python函数式编程第二弹&#xff0c;再次以两个简单的例子带大家更好的…