IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 游戏开发 -> 第一个unity游戏项目-2048 -> 正文阅读

[游戏开发]第一个unity游戏项目-2048

此文记录我这个小白没跟教程做的第一个游戏项目的过程。

主要步骤中有我一步步的探索思路,最后面有完整脚本代码(只写了一个脚本),场景有截图分享介绍,需要自己搭建。

目录

第一步:准备素材

第二步:给方格绑定变量(获取16个方格对象)

?第三步:切换方格

第四步:监听滑动

第五步:空格随机产生新数字

第六步:合并

第七步:检测游戏结束以及控制游戏重启

完整代码和总结


第一步:准备素材

找到了一张2048的素材图片,使用sprite editor分割出各单元,然后把4x4个空图片拖进场景。

第二步:给方格绑定变量(获取16个方格对象)

需要记录每个方格对应的指数,才能确定方格显示。

  private SpriteRenderer[] cell_list;
    void Start()
    {
        // 获取了所有子对象
        cell_list = GetComponentsInChildren<SpriteRenderer>();

        //遍历
        foreach (SpriteRenderer t in cell_list)
        {
            //打印拿到的子对象
            Debug.Log("t的值为:"+t);
        }
    }

?将脚本绑定给16个方格的父对象,使用GetComponentsInChildren获取方格对象

?第三步:切换方格

点击放进去的方格素材,发现它是sprite renderer对象,查询这个对象更换sprite的方法就可以替换方格。因为素材中已经有数字了,就不需要添加text。

使用Resources.LoadAll获取方格资源,然后赋值给对象。

特别注意:资源文件需要放在Assets下面的Resources文件夹中。

?代码

private Sprite[] sprite2048 = new Sprite[12];
    void Start()
    {
        //加载sprite资源,即分割的数字方格
        sprite2048 = Resources.LoadAll<Sprite>("2048");

        // 获取所有方格
        cell_list = GetComponentsInChildren<SpriteRenderer>();

        // 更改第一个方格测试
        cell_list[0].sprite = sprite2048[3];
    }

第四步:监听滑动

这段代码是在网上找的,前半部分安卓的没有尝试,后面的是电脑键盘上下左右四个键控制,Merge_Grid()函数是后面对数字的移动和处理。

void Update()
    {
        if (!gameover) {
            Press();
        }
    }


    void Press(){
		if (Application.platform == RuntimePlatform.Android && Input.touchCount == 1 && Input.touches [0].phase == TouchPhase.Moved) {
			float s01=Input.GetAxis("Mouse X");    
			float s02=Input.GetAxis("Mouse Y");    
			float distence = 5f;
			if (s01 < -1 * distence && s02>-1*distence && s02<distence) {
				move_state=-2;
			} else if (s01 > distence  && s02>-1*distence && s02<distence) {
				move_state=2;
			} else if (s02 > distence  && s01>-1*distence && s01<distence) {
				move_state=1;
			} else if (s02 < -1 * distence  && s01>-1*distence && s02<distence) {
				move_state=-1;
			}
		}
		else if(Application.platform==RuntimePlatform.WindowsEditor){
			if (Input.GetKeyDown (KeyCode.UpArrow))
				move_state=1;
			else if(Input.GetKeyDown (KeyCode.DownArrow))
				move_state=-1;
			else if(Input.GetKeyDown (KeyCode.LeftArrow))
				move_state=-2;
			else if (Input.GetKeyDown (KeyCode.RightArrow))
				move_state=2;	
		}
        if(move_state != 0)
		    Merge_Grid();    #合并数字
	}

第五步:空格随机产生新数字

准备一个数组,记录空的格子位置,然后产生随机数确定位置。查询了C#相关数据结构后,发现ArrayList和python里的list很像,能直接访问到数组的长度,就用了ArrayList,其实用普通的数组也能做到。

这段代码是在空的格子中随机一个格子产生2、4、8中的一个,里面的tmp=1、2、3分别代表实际的2、4、8。

  // 从空格中随机一个出数字
    void Creat_Num() {
        // Thread.Sleep(500);
        Debug.Log("create num");
        //出现的数字为几,2、4、8
        int tmp = 2;
        float random = Random.value;
        if(random < 0.33)
            tmp = 1;
        else if(random > 0.66)
            random = 3;

        //获取空格
        ArrayList a = new ArrayList();
        for(int i = 0; i < 16; i++){
            if(array16[i] == 0){
                a.Add(i);
            }
        }
        if(a.Count<1) return;
        int change = (int) a[Random.Range(0, a.Count)];
        array16[change] = tmp;
        cell_list[change].sprite = sprite2048[tmp];
    }

第六步:合并

void Merge_Grid(){
        // 1=up, -1=down, 2=right, -2=left
		switch (move_state) {
		case 1:
            Debug.Log("up");
            for(int i=0; i<12; i++){
                //向上紧凑
                int k = i;
                for(int j=1; i+j*4<16;j++){
                    if(array16[i+j*4] == 0 && array16[k] != 0) k = i+j*4;
                    if(array16[i+j*4]!=0 && array16[k] == 0){
                        array16[k] = array16[i+j*4];
                        array16[i+j*4] = 0;
                        k +=4 ;
                    }
                }
                if(array16[i] == 0) continue;
                //合并
                if(array16[i] == array16[i+4]){
                    array16[i]++;
                    array16[i+4] = 0;
                }
            }
			break;
		case -1:
			break;	
        #略。。。。。
		}
		move_state = 0;
        Set_Cell();    #保持array16和界面数字显示一致
        
        Creat_Num ();
        gameover = Game_Over ();
    }


    void Set_Cell(){
        //遍历
        Debug.Log("set cell");
        for (int i=0; i<16; i++){
            cell_list[i].sprite = sprite2048[array16[i]];
        }
    }

第七步:检测游戏结束以及控制游戏重启

在场景里先添加text对象,在hierarchy空白处右键->UI->Text TextMeshPro.修改text内容为YOULOSE,其他属性自行探索修改。

?在start函数里面绑定按钮,设置监听,并且设置游戏结束框的隐藏


        btn_Restart = GameObject.Find("Button").GetComponent<Button>();
        btn_Restart.onClick.AddListener(RestartGame);
        
        gameOverUI.SetActive(false);

?game OverUI的绑定,声明? ? public GameObject gameOverUI;然后添加脚本组件的地方会出现以下变量,将hierarchy中的对象拖进变量后的选择框即可,此处也绑定了gameovertext,用来更改输赢的不同文本。

?下面是游戏结束和重启游戏的代码:


    bool Game_Over() {
        for (int i = 0; i < 16; i++){
            if(array16[i] > 10){
                gameoverText.text = "You Win!";
                gameOverUI.SetActive(true);
                return true;
            }
        }
        for (int i = 0; i < 16; i++){
            if (array16[i] < 1){
                return false;
            }
            //无空格 有可合并
            if(i<12 && array16[i] == array16[i+4]){
                return false;
            }else if( (i+1)%4 != 0 && array16[i] == array16[i+1]){
                return false;
            }
        }
        
        gameOverUI.SetActive(true);
        // Time.timeScale = 0f;
        return true;
    }
    
    public void RestartGame(){
        SceneManager.LoadScene(SceneManager.GetActiveScene().name);
        // Time.timeScale = 1;
    }

完整代码和总结

实现的功能比较精简,有游戏结束的胜负两种情况,和游戏结束后的再来一次按钮,其余功能没有做。有一点补充下,我想实现先合并数字,过一点时间再出现新的数字的效果,但使用sleep会阻塞整个程序,不能完成这种效果,如果想实现,应该使用协程来处理,可以自行查找,暂时不想补充这个项目了。

完整代码如下,只有一个脚本文件。

using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using System;
using Random = UnityEngine.Random;
using System.Threading;
using UnityEngine.UI;
using UnityEngine.SceneManagement;
using TMPro;

public class GameManager : MonoBehaviour
{
    // 
    private GameObject cell16;
    private SpriteRenderer[] cell_list;
    private Sprite[] sprite2048 = new Sprite[12];
    private int move_state;  //1:上移 -1:下移 2:右移 -2:左移
    private int[] array16 = new int [16] {0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0};
    private bool gameover = false;

    private Button btn_Restart;
    public GameObject gameOverUI;
    public TMP_Text gameoverText;
    void Start()
    {
        Debug.Log("start");
        //加载sprite资源,即分割的数字方格
        sprite2048 = Resources.LoadAll<Sprite>("2048");

        // 获取所有方格对象
        cell_list = GetComponentsInChildren<SpriteRenderer>(); 

        cell_list[0].sprite = sprite2048[array16[0]];
        Creat_Num ();
        Creat_Num ();

        btn_Restart = GameObject.Find("Button").GetComponent<Button>();
        btn_Restart.onClick.AddListener(RestartGame);
        
        gameOverUI.SetActive(false);
    }

    // Update is called once per frame
    void Update()
    {
        if (!gameover) {
            Press();
        }
    }


    void Press(){
		if (Application.platform == RuntimePlatform.Android && Input.touchCount == 1 && Input.touches [0].phase == TouchPhase.Moved) {
			float s01=Input.GetAxis("Mouse X");    
			float s02=Input.GetAxis("Mouse Y");    
			float distence = 5f;
			if (s01 < -1 * distence && s02>-1*distence && s02<distence) {
				move_state=-2;
			} else if (s01 > distence  && s02>-1*distence && s02<distence) {
				move_state=2;
			} else if (s02 > distence  && s01>-1*distence && s01<distence) {
				move_state=1;
			} else if (s02 < -1 * distence  && s01>-1*distence && s02<distence) {
				move_state=-1;
			}
		}
		else if(Application.platform==RuntimePlatform.WindowsEditor){
			if (Input.GetKeyDown (KeyCode.UpArrow))
				move_state=1;
			else if(Input.GetKeyDown (KeyCode.DownArrow))
				move_state=-1;
			else if(Input.GetKeyDown (KeyCode.LeftArrow))
				move_state=-2;
			else if (Input.GetKeyDown (KeyCode.RightArrow))
				move_state=2;	
		}
        if(move_state != 0)
		    Merge_Grid();
	}

    // 从空格中随机一个出数字
    void Creat_Num() {
        // Thread.Sleep(500);
        Debug.Log("create num");
        //出现的数字为几,2、4、8
        int tmp = 2;
        float random = Random.value;
        if(random < 0.33)
            tmp = 1;
        else if(random > 0.66)
            random = 3;

        //获取空格
        ArrayList a = new ArrayList();
        for(int i = 0; i < 16; i++){
            if(array16[i] == 0){
                a.Add(i);
            }
        }
        if(a.Count<1) return;
        int change = (int) a[Random.Range(0, a.Count)];
        array16[change] = tmp;
        cell_list[change].sprite = sprite2048[tmp];
    }

    void Merge_Grid(){
        // 1=up, -1=down, 2=right, -2=left
		switch (move_state) {
		case 1:
            Debug.Log("up");
            for(int i=0; i<12; i++){
                //向上紧凑
                int k = i;
                for(int j=1; i+j*4<16;j++){
                    if(array16[i+j*4] == 0 && array16[k] != 0) k = i+j*4;
                    if(array16[i+j*4]!=0 && array16[k] == 0){
                        array16[k] = array16[i+j*4];
                        array16[i+j*4] = 0;
                        k +=4 ;
                    }
                }
                if(array16[i] == 0) continue;
                //合并
                if(array16[i] == array16[i+4]){
                    array16[i]++;
                    array16[i+4] = 0;
                }
            }
			break;
		case -1:
            Debug.Log("down");
            for(int i=15; i>3; i--){
                //向下紧凑
                int k = i;
                for(int j=1; i-j*4>=0;j++){
                    if(array16[i-j*4] == 0 && array16[k] != 0) k = i-j*4;
                    if(array16[i-j*4]!=0 && array16[k] == 0){
                        array16[k] = array16[i-j*4];
                        array16[i-j*4] = 0;
                        k -=4 ;
                    }
                }
                if(array16[i] == 0) continue;
                //合并
                if(array16[i] == array16[i-4]){
                    array16[i]++;
                    array16[i-4] = 0;
                }
            }
			break;
		case 2:
            Debug.Log("right");
            for(int i=15; i>0; i-=4){
                for(int x=0; x<3; x++){
                    //向right紧凑
                    int k = i-x;
                    for(int j=1; x+j<4;j++){
                        if(array16[i-j-x] == 0 && array16[k] != 0) k = i-j-x;
                        if(array16[i-j-x]!=0 && array16[k] == 0){
                            array16[k] = array16[i-j-x];
                            array16[i-j-x] = 0;
                            k--;
                        }
                    }
                    if(array16[i-x] == 0) continue;
                    //合并
                    if(array16[i-x] == array16[i-x-1]){
                        array16[i-x]++;
                        array16[i-x-1] = 0;
                    }
                }
            }
			break;
		case -2:
            Debug.Log("left");
            for(int i=0; i<13; i+=4){
                for(int x=0; x<3; x++){
                    //向left紧凑
                    int k = i+x;
                    for(int j=1; x+j<4;j++){
                        if(array16[i+j+x] == 0 && array16[k] != 0) k = i+j+x;
                        if(array16[i+j+x]!=0 && array16[k] == 0){
                            array16[k] = array16[i+j+x];
                            array16[i+j+x] = 0;
                            k++;
                        }
                    }
                    if(array16[i+x] == 0) continue;
                    //合并
                    if(array16[i+x] == array16[i+x+1]){
                        array16[i+x]++;
                        array16[i+x+1] = 0;
                    }
                }
            }
			break;	
		}
		move_state = 0;
        Set_Cell();
        
        Creat_Num ();
        gameover = Game_Over ();
    }

    void Set_Cell(){
        //遍历
        Debug.Log("set cell");
        for (int i=0; i<16; i++){
            cell_list[i].sprite = sprite2048[array16[i]];
        }
    }

    bool Game_Over() {
        for (int i = 0; i < 16; i++){
            if(array16[i] > 10){
                gameoverText.text = "You Win!";
                gameOverUI.SetActive(true);
                return true;
            }
        }
        for (int i = 0; i < 16; i++){
            if (array16[i] < 1){
                return false;
            }
            //无空格 有可合并
            if(i<12 && array16[i] == array16[i+4]){
                return false;
            }else if( (i+1)%4 != 0 && array16[i] == array16[i+1]){
                return false;
            }
        }
        
        gameOverUI.SetActive(true);
        // Time.timeScale = 0f;
        return true;
    }
    
    public void RestartGame(){
        SceneManager.LoadScene(SceneManager.GetActiveScene().name);
        // Time.timeScale = 1;
    }

    public void Quit(){
        Application.Quit();
    }

}

  游戏开发 最新文章
6、英飞凌-AURIX-TC3XX: PWM实验之使用 GT
泛型自动装箱
CubeMax添加Rtthread操作系统 组件STM32F10
python多线程编程:如何优雅地关闭线程
数据类型隐式转换导致的阻塞
WebAPi实现多文件上传,并附带参数
from origin ‘null‘ has been blocked by
UE4 蓝图调用C++函数(附带项目工程)
Unity学习笔记(一)结构体的简单理解与应用
【Memory As a Programming Concept in C a
上一篇文章      下一篇文章      查看所有文章
加:2022-04-06 16:21:40  更:2022-04-06 16:23:32 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/16 20:16:50-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码