当前位置: 首页 > news >正文

基于扑克牌分发效果制作时的问题总结

其基本效果如图
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

1. 在overlay模式下直接使用position来移动

实现代码

public class Card : MonoBehaviour
{public RectTransform target;public Button cardButton;private bool isPack = false;public List<RectTransform> cards = new List<RectTransform>();public List<Vector3> positions = new List<Vector3>();   //记录每张牌的原始屏幕位置void Start(){for (int i = 0; i < cards.Count; i++){//positions.Add(cards[i].anchoredPosition);positions.Add(cards[i].position);}cardButton.onClick.AddListener(CardClicked);}private void CardClicked()  //这里的关键是坐标转换,源UI的坐标系和目标UI的坐标系不一样,因为锚点位置不一样{ DOTween.KillAll();if (!isPack)    {isPack = true;Sequence se = DOTween.Sequence();for (int i = 0; i < cards.Count; i++){se.Append(cards[i].transform.DOMove(target.position, 0.2f));se.AppendInterval(0.1f);}}else{isPack = false;Sequence se = DOTween.Sequence();for (int i = positions.Count - 1; i >= 0; i--){//se.Append(cards[i].DOAnchorPos(positions[i], 0.2f));se.Append(cards[i].transform.DOMove(positions[i], 0.2f));se.AppendInterval(0.1f);}}}
}

使用DoTween来实现移动的动画。
positions 列表用来记录每张牌的原始屏幕坐标,因为Canvas是overlay模式,所以每个Image的position就是屏幕坐标,而这里的target就是中心点Center,注意,Center的锚点是在中心且聚拢的,而对于牌,因为要适配不同分辨率的屏幕,所以锚点都做了自适应。
对于将牌移动到中心点,直接将目标点设置为target.position,对于将牌还原,这里将目标点设置为提前记录的每张牌的原始position,以上成立的逻辑是因为在Canvas是overlay模式,UI元素的position就是屏幕坐标

2. 使用anchoredPosition移动

而如果要用anchoredPosition来做移动,那么情况就有些不同了,因为anchoredPosition的本质是从Anchor到Pivot的向量,即当锚点聚拢时,RectTransform的PosX和PosY就是从Anchor到Pivot的距离
在这里插入图片描述
在这里插入图片描述
此时PosY的值为200
如果将Pivot的位置移动到正方形Image的最底边
在这里插入图片描述

在这里插入图片描述
发现值变成了150,即原先的200 - 50,因为正方形边长的一半是50,此时Pivot的值是(0.5,0)

实现代码

public class Card : MonoBehaviour
{public RectTransform target;public Button cardButton;private bool isPack = false;public List<RectTransform> cards = new List<RectTransform>();public List<Vector3> positions = new List<Vector3>();     //记录每张牌的原始屏幕位置void Start(){for (int i = 0; i < cards.Count; i++){positions.Add(cards[i].anchoredPosition);//positions.Add(cards[i].position);}cardButton.onClick.AddListener(CardClicked);}private void CardClicked()  //这里的关键是坐标转换,源UI的坐标系和目标UI的坐标系不一样{ DOTween.KillAll();if (!isPack)    {isPack = true;Sequence se = DOTween.Sequence();for (int i = 0; i < cards.Count; i++){//se.Append(cards[i].transform.DOMove(target.position, 0.2f));GetTargetAnchorPos(cards[i].anchorMin);se.Append(cards[i].DOAnchorPos(target.anchoredPosition, 0.2f));se.AppendInterval(0.1f);}}else{isPack = false;Sequence se = DOTween.Sequence();for (int i = positions.Count - 1; i >= 0; i--){se.Append(cards[i].DOAnchorPos(positions[i], 0.2f));//se.Append(cards[i].transform.DOMove(positions[i], 0.2f));se.AppendInterval(0.1f);}}}/// <summary>/// 得到锚点修改后的taregt的anchoredPosition/// </summary>/// <param name="anchorCenter"></param>public void GetTargetAnchorPos(Vector2 anchorCenter){Vector2 oldAnchorCenter = (target.anchorMin + target.anchorMax) / 2;Vector2 oldAnchorPos = target.anchoredPosition;target.anchorMin = anchorCenter;target.anchorMax = anchorCenter;Vector2 deltaAnchor = oldAnchorCenter - anchorCenter;    //计算锚点的偏移Vector2 deltaMove = new Vector2(deltaAnchor.x * 1125, deltaAnchor.y * 2436);  //计算位置的偏移target.anchoredPosition = oldAnchorPos + deltaMove;}
}

增加了一个GetTargetAnchorPos方法,用来重新计算在当前这张牌所在的锚点下,中心点需要更改为这个牌的锚点,并且重新计算anchoredPosition ,前提是这些牌的锚点都是聚拢的,然后移动的时候将目标点都设置为对应点的anchoredPosition

http://www.lryc.cn/news/534608.html

相关文章:

  • 老榕树的Java专题:Redis 从入门到实践
  • 【玩转 Postman 接口测试与开发2_019】第15章:利用 Postman 初探 API 性能测试(含实战截图)
  • 在 Qt 开发中,可以将 QML 封装成库
  • 换电脑了如何快速导出vscode里的插件
  • 点大商城V2-2.6.6源码全开源uniapp +搭建教程
  • 9 Pydantic复杂数据结构的处理
  • springboot+redis实现将树形结构存储到redis
  • 6、使用one-api管理统一管理大模型,并开始使用本地大模型
  • Windows安装Lyx
  • 一文讲透大模型部署工具ollama--结合本地化部署deepseek实战
  • 网络防御高级
  • 使用PyCharm进行Django项目开发环境搭建
  • 如何定义“破坏环境”
  • 现代前端开发的演进与未来趋势:从工具革新到技术突破
  • 活动预告 |【Part1】Microsoft 安全在线技术公开课:安全性、合规性和身份基础知识
  • idea Ai工具通义灵码,Copilot我的使用方法以及比较
  • 【JavaScript】《JavaScript高级程序设计 (第4版) 》笔记-Chapter8-对象、类与面向对象编程
  • 介绍下SpringBoot常用的依赖项
  • 深度解析策略模式:从理论到企业级实战应用
  • 【Linux】深入理解linux权限
  • C++STL(六)——list模拟
  • 网络安全与AI:数字经济发展双引擎
  • WPS接入DeepSeek模型
  • 深度学习之神经网络框架搭建及模型优化
  • 采用分步式无线控制架构实现水池液位自动化管理
  • OpenEuler学习笔记(二十三):在OpenEuler上部署开源MES系统
  • SpringSecurity:授权服务器与客户端应用(入门案例)
  • 没用的文章又➕1
  • BiGRU双向门控循环单元多变量多步预测,光伏功率预测(Matlab完整源码和数据)
  • 谷歌浏览器多开指南:如何完成独立IP隔离?