主要思路是通过ScrollView借助实现的,这个主要以只能垂直拉动为例。
为了方便理解,我们可以先创建一个ScrollView,然后新建几个Image放在Scrollview的Content物体之下,自己随意拖动,将其摆好,新建出来的场景图如下图所示。
当我向下拖动时,如下图所示,Content 的anchoredPosition.y会发生变化,而这个值与拖动长度有关,一个Image我设置的大小是宽高分别是60X60,我刚刚差不多拖动了1个Image还要多一些的距离,现在值Conten的Y值增加到100左右,因此可以看出,我可利用Content的Y值,从而大概知道自己向上拉动了几个Image。如下图所示。
当我们拖动时,ScrollView的OnVauleChanged事件就会被触发。大体思路是当我们每生成一个Image时,我们就给他赋予一个Index值来记录它。每当玩家拖动滑动条,触发OnVauleChange事件时,我们计算当前应该的Index值,从而来刷新界面。具体代码如下图所示。
using System;
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using UnityEngine.UI;
public class MyLoopScrollvewMgr : MonoBehaviour {
private ScrollRect myScrollView;
private RectTransform myContent;
public GameObject CellPerfab;
///
/// 当前场景中生成的Cell对象列表
///
public List CellItemList = new List();
///
/// Cell与分配给他的Index对应表
///
public Dictionary CellIndexDic = new Dictionary();
[Header ("参数设置")]
public RectOffset MyGridPadding;
public Vector2 MyGridSpacing;
public Vector2 MyGridCellSize;
///
/// 创建在场景中的Item数量个数参数
///
private int creatCount=15;
///
/// 当前数据的Item总数
///
private int dataCount=68;
///
///非法的起始索引
///
private const int invalidStartIndex = -1;
///
/// 当前Content的Y值下第一个开始的Index值
///
private int StartIndex;
///
/// 一行Cell的个数
///
public int rowCellNum=3;
///
/// 上方缓存行数
///
private int OffsetTopLineNum = 2;
///
/// 下方缓存行数
///
private int OffsetBottomLineNum=2;
void Start () {
Init();
}
void Init()
{
CellItemList.Clear();
CellIndexDic.Clear();
myScrollView = GetComponent();
myContent = myScrollView.content;
MyGridCellSize = CellPerfab.GetComponent().sizeDelta;
#region Content参数初始化
rowCellNum = (int)(myScrollView .GetComponent().sizeDelta.x / MyGridCellSize.x);
SetContentAnchors();
myContent .sizeDelta = SetContentSize();
myContent.anchoredPosition = Vector2.zero;
#endregion
StartIndex = 0;
for (int i = 0; i < creatCount ; i++)
{
CreatCell(i);
}
myScrollView.onValueChanged.RemoveAllListeners();
myScrollView.onValueChanged.AddListener(OnValueChanged);
}
private void OnValueChanged(Vector2 arg0)
{
int CurrentIndex = GetCurrentIndex();
if (CurrentIndex !=StartIndex &&CurrentIndex >invalidStartIndex )
{
StartIndex = CurrentIndex;
//为了将需要缓存的也先生出来,以免玩家滑动太快,显示不过来。
//如果不考虑缓存,min应该为StartIndex,max应该为StartIndex+creatCount
int min = Mathf.Max(StartIndex - OffsetTopLineNum * rowCellNum, 0);
int max = Mathf.Min( StartIndex + creatCount + OffsetBottomLineNum * rowCellNum,dataCount );
for (int i = CellItemList.Count-1; i >=0 ; i--)
{
GameObject go = CellItemList[i];
int index = CellIndexDic[go];
if(index max )
{
ReturnCell(go);
}
}
float maxCreat = Mathf.Min(StartIndex + creatCount, dataCount);
bool isExit = false;
for (int i = min; i < max; i++)
{
isExit = false;
for (int j = 0; j < CellItemList.Count; j++)
{
GameObject cell = CellItemList[j];
if (CellIndexDic[cell] == i)
{
isExit = true;
break;
}
}
if (isExit)
{
continue;
}
CreatCell(i);
}
#region
/* for (int i = StartIndex ; i < maxCreat; i++)
{
isExit = false;
for (int j = 0; j < CellItemList .Count ; j++)
{
GameObject cell = CellItemList[j];
if(CellIndexDic [cell]==i )
{
isExit = true;
break;
}
}
if(isExit )
{
continue;
}
CreatCell(i);
}*/
#endregion
}
}
private int GetCurrentIndex()
{
int index = 0;
if(myScrollView .vertical )
{
index =Mathf .FloorToInt ( myContent.anchoredPosition .y / (MyGridCellSize.y + MyGridSpacing .y));
}
return index*rowCellNum ;
}
private void CreatCell(int cellIndex)
{
GameObject cell = GetItemInPool();
cell.SetActive(true);
cell.transform.name = cellIndex.ToString();
cell.transform.Find("Image/ID").gameObject.GetComponent().text = cellIndex.ToString();
cell.transform.parent = myContent;
RectTransform rect = cell .GetComponent();
rect.pivot = new Vector2(0, 1);
//垂直拉动以左上为锚点
rect.anchorMin = new Vector2(0, 1);
rect.anchorMax = new Vector2(0, 1);
cell.transform.localPosition = GetCellPosition(cellIndex);
CellItemList.Add(cell);
CellIndexDic.Add(cell, cellIndex);
}
private Vector3 GetCellPosition(int cellIndex)
{
Vector3 tempvec = Vector3 .zero;
if(myScrollView .vertical && myScrollView.horizontal == false)
{
int row =cellIndex / rowCellNum;
int column = cellIndex% rowCellNum;
tempvec.x = MyGridCellSize.x * column + MyGridSpacing.x * (column)+MyGridPadding .left ;
tempvec.y =-( MyGridCellSize.y * row + MyGridSpacing.y * (row ) + MyGridPadding.top);
}
return tempvec;
}
private void ReturnCell(GameObject cell)
{
cell.SetActive(false);
CellItemList.Remove(cell);
CellIndexDic[cell] = invalidStartIndex;
ReturnItemToPool(cell);
CellIndexDic.Remove(cell);
}
#region Content相关设置方法
private Vector2 SetContentSize()
{
Vector2 tempContentSize = Vector2.zero;
if (myScrollView == null) return tempContentSize;
if (myScrollView.vertical && myScrollView.horizontal == false)
{
int row =Mathf .CeilToInt ((float)dataCount / rowCellNum);
tempContentSize.x = myContent.sizeDelta.x;
tempContentSize.y = (MyGridCellSize.y *(row) + MyGridSpacing .y *(row )+MyGridPadding.top );
}
return tempContentSize;
}
private void SetContentAnchors()
{
if (myScrollView == null) return;
//仅仅考虑当前只能垂直拉动或水平拉动情况
if(myScrollView .vertical&&myScrollView .horizontal ==false )
{
//垂直拉动以上方为锚点
myContent.anchorMin = new Vector2(0, 1);
myContent.anchorMax = new Vector2(1, 1);
}else if((!myScrollView.vertical) && myScrollView.horizontal)
{
//水平拉动以左边为锚点
myContent.anchorMin = new Vector2(0, 0);
myContent.anchorMax = new Vector2(0, 1);
}
}
#endregion
#region 简单对象池
private Stack itemPools = new Stack();
private int poolsMaxCount = 40;
public GameObject GetItemInPool()
{
GameObject item = null;
if(itemPools.Count >0)
{
item = itemPools.Pop();
}
if(item ==null )
{
item = Instantiate(CellPerfab);
}
return item;
}
public void ReturnItemToPool(GameObject item)
{
DestroyObject(item);
//if (item == null) return;
//if (itemPools.Count > poolsMaxCount)
//{
// DestroyObject(item);
//}
//else
//{
// itemPools.Push(item);
//}
}
#endregion
}
几个方法做一下解释
1、就是初始化时候步骤大概是这样的
1.1、获取ScrollView,以及他的Content,
1.2、对Content进行锚点重置,和sizeDelta的大小重置,以及anchoredPosition归零。
1.3、根据初始要求创建Item,修改他们的锚点,分配他们Index,和计算他们应该的位置。
2、Content的sizeDelta大小是一开始设定好的,他是根据你所要创建的所有数据总数还确定大小的。不过也可以后面做更改,为了方便,我一开始就全部算好了。
3、每个Item的位置计算,创建时都会分配一个Index,根据这个Index,让其除以每行Item的总个数可以得到他在几行,取余可以得到他在几列。从而再根据Item的Y大小和每行Y的间隔,算出他应该的位置。
4.当拖动时,改变时,计算当前Index值,就是用content的Y除以()