Unity组件开发--升降梯

        我开发的升降梯由三个部分组成,反正适用于我的需求了,其他人想复用到自己的项目的话,不一定。写的也不是很好,感觉搞的有点复杂啦。完全可以在优化一下,项目赶工期,就先这样吧。能用就行,其他的再说。

1.升降梯基类:

using System.Collections;
using System.Collections.Generic;
using UnityEngine;

public abstract class LiftObjCtrBase : MonoBehaviour
{
    public float speed = 5f;  // 上升和下降的速度
    public LiftObjCtrBase targetObject;  // 指定的物体高度

    [HideInInspector]
    public bool isRising = false;  // 是否正在上升
    [HideInInspector]
    public bool isFalling = false;  // 是否正在下降
    [HideInInspector]
    public bool isPlayerStay = false; //玩家是否在上面
    [HideInInspector]
    public Vector3 initialPosition;  // 初始位置
    [HideInInspector]
    public float offsetY = 0;
    [HideInInspector]
    public Transform playerTra;
    [HideInInspector]
    public Transform targetTra;
    [HideInInspector]
    public float radius = 0;
    // Start is called before the first frame update
    [HideInInspector]
    public bool isReach = false;

    public virtual bool isTriggerEnter(Transform playerTra) {
        if (Vector3.Distance(gameObject.transform.position, playerTra.position) > radius)
        {
            return false;
        }
        else
        {

            return true;
        }
    }

    public virtual void startRising() {
    
    }

}

2.玩家接触到的头一个升降梯:第一个升降梯的碰撞组件,必须开启isTrigger属性;注意:所有升降梯都需要挂碰撞组件

using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using UnityEngine.TextCore.Text;

public class LiftFirstObjCtr : LiftObjCtrBase
{
    
    //private Transform player;  // 角色
    private void Start()
    {
        initialPosition = transform.position;  // 记录初始位置
        targetTra = targetObject.gameObject.transform;
        MeshCollider meshCollider = gameObject.GetComponent<MeshCollider>();
        radius = meshCollider.bounds.extents.magnitude;
    }

    private void Update()
    {
        if (targetObject == null) return;

        if (playerTra != null && !isTriggerEnter(playerTra) && !isFalling)
        {

            isPlayerStay = false;
            isFalling = true;
            isRising = false;
            
        }

        if (playerTra !=null && targetObject.isTriggerEnter(playerTra) && !targetObject.isRising && !targetObject.isReach)
        {

            isPlayerStay = false;
            isFalling = true;
            targetObject.startRising();
            
        }
        
        if (isRising && isPlayerStay)
        {
            // 上升逻辑

            if (playerTra == null || offsetY == 0) return;
            transform.Translate(Vector3.up * speed * Time.deltaTime);
            playerTra.position = new Vector3(playerTra.position.x, transform.position.y+ offsetY+0.4f, playerTra.position.z);
            
            // 到达目标高度后开始下降
            if (transform.position.y >= targetTra.position.y)
            {

                gameObject.GetComponent<Collider>().isTrigger = false;
                
                isRising = false;
                
            }
        }
        else if (isFalling && !isPlayerStay)
        {
            // 下降逻辑
            transform.Translate(Vector3.down * speed * Time.deltaTime);

            // 返回初始位置后停止下降
            if (transform.position.y <= initialPosition.y)
            {
                transform.position = initialPosition;
                isFalling = false;
                gameObject.GetComponent<Collider>().isTrigger = true;
            }
        }
       

    }

    

    

    private void OnTriggerEnter(UnityEngine.Collider other)
    {
        
        if (other.gameObject.tag == "Player")
        {
            Debug.Log("玩家撞到升降梯");
            
        }
    }


    public override bool isTriggerEnter(Transform playerTra) {
        // 检测角色是否在碰撞器范围内
        if (Vector3.Distance(gameObject.transform.position, playerTra.position) > radius)
        {
            return false;
        }
        else {

            return true;
        }
    }
   

    private void OnTriggerStay(UnityEngine.Collider other)
    {
        
        if (other.gameObject.tag == "Player" && !isPlayerStay)
        {
            if (Vector3.Distance(other.gameObject.transform.position, gameObject.transform.position)< radius) {
                Debug.Log("玩家待在升降梯");
                isRising = true;

                playerFollow(other.gameObject);
                gameObject.GetComponent<Collider>().isTrigger = false;
            }
            
        }

    }

    


    

    private void OnTriggerExit(UnityEngine.Collider other)
    {
        
        
    }

    private void playerFollow(GameObject player) {

        
        offsetY = player.transform.position.y - transform.position.y;
        playerTra = player.transform;
        isPlayerStay = true;
    }

    public override void startRising()
    {
        
    }
}

3.中间的其他升降梯,其他中间的升降梯由于是悬浮在空中的,所以需要不能开启isTrigger属性,否则玩家碰到后就会掉下去:

using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using UnityEngine.TextCore.Text;

public class LiftMiddleObjCtr : LiftObjCtrBase
{
    public LiftObjCtrBase preLiftObj;
    // Update is called once per frame
    
    private void Start()
    {
        initialPosition = transform.position;  // 记录初始位置
        targetTra = targetObject.transform;
        MeshCollider meshCollider = gameObject.GetComponent<MeshCollider>();
        radius = meshCollider.bounds.extents.magnitude;
    }


    void Update()
    {

        if (targetObject == null) return;

        if (playerTra != null && !isTriggerEnter(playerTra) && !isFalling)
        {

            isPlayerStay = false;
            isFalling = true;
            isRising = false;
            return;
        }

        if (playerTra != null && targetObject.isTriggerEnter(playerTra) && !targetObject.isRising)
        {

            isPlayerStay = false;
            isFalling = true;
            targetObject.startRising();
            playerTra = null;
        }

        if (isRising && isPlayerStay)
        {
            // 上升逻辑
            if (playerTra == null || offsetY == 0) return;

            transform.Translate(Vector3.up * speed * Time.deltaTime);
            playerTra.position = new Vector3(playerTra.position.x, transform.position.y + offsetY + 0.4f, playerTra.position.z);
            
            // 到达目标高度后开始下降
            if (transform.position.y >= targetTra.position.y)
            {

                isRising = false;
                isReach = true;
            }
        }
        else if (isFalling && !isPlayerStay)
        {
            // 下降逻辑
            transform.Translate(Vector3.down * speed * Time.deltaTime);

            // 返回初始位置后停止下降
            if (transform.position.y <= initialPosition.y)
            {
                transform.position = initialPosition;
                isFalling = false;
                isReach = false;
                isRising = false;
            }
        }
        
    }

    public override bool isTriggerEnter(Transform playerTra)
    {

        if (Vector3.Distance(gameObject.transform.position, playerTra.position) > radius)
        {
            return false;
        }
        else
        {

            return true;
        }
    }

    public override void startRising() {

        
        isRising = true;

        playerFollow(preLiftObj.playerTra.gameObject);
        gameObject.GetComponent<Collider>().isTrigger = false;
    }

    private void playerFollow(GameObject player)
    {


        offsetY = player.transform.position.y - transform.position.y;
        playerTra = player.transform;
        isPlayerStay = true;
    }
}

4.最后一个升降梯,是玩家最终达到的升降梯,所以可以不用升降,他是出发倒数第二个升降梯的下降:感觉有点奇怪,反正整个组件都开发的感觉奇奇怪怪的,算了算了,咱也没啥要求,混口饭吃,能较差就行

using System.Collections;
using System.Collections.Generic;
using UnityEngine;

public class LiftLastObjCtr : LiftObjCtrBase
{
    // Start is called before the first frame update
    public LiftMiddleObjCtr middleObject;

    public override bool isTriggerEnter(Transform playerTra)
    {
        return false;
    }

    public override void startRising()
    {
        
    }



    // Update is called once per frame
    void Update()
    {



        if (middleObject != null && middleObject.playerTra != null) {

            if (Vector3.Distance(middleObject.playerTra.position,gameObject.transform.position) < 1) {
                if (Vector3.Distance(middleObject.playerTra.position, middleObject.gameObject.transform.position) > 1)
                {

                    middleObject.isFalling = true;
                    middleObject.isPlayerStay = false;
                }
            }

              
        }
    }

    
}

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

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

相关文章

【EI会议征稿通知】第三届智能电网与绿色能源国际学术会议(ICSGGE 2024)

第三届智能电网与绿色能源国际学术会议&#xff08;ICSGGE 2024&#xff09; 2024 3rd International Conference on Smart Grid and Green Energy 2024年第三届智能电网与绿色能源国际学术会议&#xff08;ICSGGE 2024&#xff09;将于2024年4月19-21日在中国成都举行。会议…

GraphQL和REST API的区别

我的新书《Android App开发入门与实战》已于2020年8月由人民邮电出版社出版&#xff0c;欢迎购买。点击进入详情 GraphQL&#xff08;Graph Query Language&#xff09;和REST&#xff08;Representational State Transfer&#xff09;是两种用于构建和设计API的不同方法。以下…

基于ssm公交车信息管理系统论文

摘 要 当下&#xff0c;正处于信息化的时代&#xff0c;许多行业顺应时代的变化&#xff0c;结合使用计算机技术向数字化、信息化建设迈进。以前相关行业对于公交车信息的管理和控制&#xff0c;采用人工登记的方式保存相关数据&#xff0c;这种以人力为主的管理模式已然落后。…

vue2 element 弹出框拖拽会出现一层阴影问题

问题如图所示&#xff1a; 因增加 draggable 属性导致我弹窗表单清空文本框时&#xff0c;从右向左选中字体会出现拖拽阴影效果 去掉 draggable 即可 <template><div class"sys-jobTrigger-container"><el-dialog:visible.sync"state.isShowD…

上班经常迟到怎么办?

相信上班需要打卡的小伙伴都跟博主一样&#xff0c;经常可能因为迟一两分钟导致打开迟到而懊恼。 那么&#xff0c;如何避免这种问题呢&#xff1f;下面给大家提供一下博主自己试过的方法,效果还挺不错的 时间逆推法&#xff1a; 拿博主举例子&#xff0c;我住处到公司大概是…

react 基于Ant Desgin Upload简单实现导入 导出功能

效果图&#xff1a; 导入&#xff1a; 导出&#xff1a; 导入代码&#xff1a; const propsConfig {name: file,action: importDataExcelApi, //后端接口headers: {authorization: authorization-text,loginUserId: sessionStorage.getItem(userLogin)? JSON.parse(sessio…

C++完成使用map Update数据 二进制数据

1、在LXMysql.h和LXMysql.cpp分别定义和编写关于pin语句的代码 //获取更新数据的sql语句 where语句中用户要包含where 更新std::string GetUpdatesql(XDATA kv, std::string table, std::string where); std::string LXMysql::GetUpdatesql(XDATA kv, std::string table, std…

Spring——Spring基于注解的IOC配置

基于注解的IOC配置 学习基于注解的IOC配置&#xff0c;大家脑海里首先得有一个认知&#xff0c;即注解配置和xml配置要实现的功能都是一样的&#xff0c;都是要降低程序间的耦合。只是配置的形式不一样。 1.创建工程 1.1 pom.xml <?xml version"1.0" encoding…

基于SSM框架和Layui框架的管理系统

计算机毕业设计&#xff1a;打造安全、高效的信息管理系统在这个数字化时代&#xff0c;信息安全和高效管理是至关重要的。为了帮助学校或机构更好地管理和保护信息&#xff0c;我们为您设计了一套功能强大的信息管理系统。该系统利用先进的技术&#xff0c;结合MD5加密&#x…

excel中解决多行文本自动调整行高后打印预览还是显示不全情况

注意&#xff1a;此方法对于多行合并后单元格行高调整不适用&#xff0c;需要手动调整&#xff0c;如大家有简便方法&#xff0c;欢迎评论。 一、调整表格为自动调整行高 1&#xff09;点击此处全选表格 2&#xff09;在第一行序号单元格的下端&#xff0c;鼠标成黑十字时&am…

自定义页面,落地页面自由搭配

自定义页面 路径 应用 >> 新增自定义页面 功能简介 应用内新增「自定义页面」。 自定义页面是一个可以自由配置的落地页面&#xff0c;支持通过不同的入口设置连接到不同的链接地址&#xff0c;使得不同的应用资源可以根据业务场景化的展示。 使用场景&#xff1a; 一…

新手可理解的PyTorch线性层解析:神经网络的构建基石

目录 torch.nn子模块Linear Layers详解 nn.Identity Identity 类描述 Identity 类的功能和作用 Identity 类的参数 形状 示例代码 nn.Linear Linear 类描述 Linear 类的功能和作用 Linear 类的参数 形状 变量 示例代码 nn.Bilinear Bilinear 类的功能和作用 B…

设计一个LLMops的端到端业务流程需要哪些存储技术

周五晚上老哥们儿hengzi给我打电话说有没有大模型相关的存储相关的设计,答:没有现成的,后来我甩过去一个TP,PP,DDP的文档。仔细想想也对,也不对。对的是,确实是存储,但是只有显存里的存储,不对的是其实他想问的是端到端的,主要还是磁盘上的存储 然后我翻内网也没什么现…

进程、进程管理和计划任务

一、程序进程 1.1 什么是程序 是一组计算机能识别和执行的指令&#xff0c;运行于电子计算机上&#xff0c;满足人们某种需求的信息化工具 用于描述进程要完成的功能&#xff0c;是控制进程执行的指令集 二、进程 2.1 什么是进程&#xff1f; 记载到内存中运行&#xff…

二进制分析平台逆向编译器:Vector 35 Binary Ninja 软件详细功能介绍

Vector 35 Binary Ninja是一款为多种架构的反汇编提供第一方支持&#xff0c;包括 x86、x86-64、ARMv7&#xff08;带有 Thumb2&#xff09;、ARMv8 (AArch64)、PowerPC、6502、Z80 和 MIPS的反编译器&#xff0c;Binary Ninja为二进制分析构建了最好的自动化和 API。 Vector 3…

web期末作业动态时钟UI界面毛玻璃版

效果图 html代码奉上 <!DOCTYPE html> <html lang"zh-CN"><head><meta charset"UTF-8"><meta http-equiv"X-UA-Compatible" content"IEedge"><meta name"viewport" content"widthde…

Callback Hook

一、Callback Hook 函数名&#xff1a;useCallback 用于得到一个固定引用值的函数&#xff0c;通常用它进行性能优化。 useCallback: 该函数只需要传入两个参数&#xff1a;一个回调函数和一个依赖数组即可。 1.函数&#xff0c;useCallback会固定该函数的引用&#xff0c;…

UVa1402/LA3961 Robotic Sort

题目链接 本题是2007年ICPC欧洲区域赛中欧赛区的S题 题意 一个实验室里有 n 个长短不一的试管。你的任务是编写一段程序&#xff0c;用机器臂把它们按照高度从小到大的顺序排列。对于高度相同的试管&#xff0c;排序前后的相对位置应保持不变。排序方法如下图所示。 排序需要n…

多通道病虫害分子检测仪-百科科普知识

在农业科技日新月异的今天&#xff0c;病虫害防治已经成为现代农业的重要一环。为了更精准、更快速地检测和防治病虫害&#xff0c;多通道病虫害分子检测仪应运而生&#xff0c;成为守护绿色家园的"黑科技"。 WX-XC1多通道病虫害分子检测仪是一款集成了分子生物学、…

【React系列】网络框架axios库的使用

本文来自#React系列教程&#xff1a;https://mp.weixin.qq.com/mp/appmsgalbum?__bizMzg5MDAzNzkwNA&actiongetalbum&album_id1566025152667107329) 一. axios库的基本使用 1.1. 网络请求的选择 目前前端中发送网络请求的方式有很多种&#xff1a; 选择一:传统的Aj…