使用 SaveGameFree 实现 Unity 存档/读档/删档

SaveGameFree


初始化项目

新建项目
替换原有场景
添加物体至场景
调整相机角度

安装插件 SaveGameFree


安装插件 Input System

导入
激活

新建脚本 PlayerController.cs

using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using UnityEngine.InputSystem;
using BayatGames.SaveGameFree;


public class PlayerController : MonoBehaviour
{

    public float speed = 10;
    private Rigidbody rb;
    private float movementX;
    private float movementY;
    private string identifier = "playerPosition";

    // 自定义数据类,保存 player 位置信息
    public class PlayerData
    {
        public float x;
        public float y;
    }

    void Start()
    {
        GetData();
        
        rb = GetComponent<Rigidbody>();
    }

    void Update()
    {
        // 存档快捷键 q(方便调试,不与 IDE 键位冲突)
        if (Input.GetKeyDown(KeyCode.Q))
        {
            SetData();
        }

        // 删档快捷键 e(方便调试,不与 IDE 键位冲突)
        if (Input.GetKeyDown(KeyCode.E))
        {
            DeleteData();
        }
    }

    void FixedUpdate()
    {
        // 组装 3D 向量
        Vector3 vector3D = new Vector3(movementX, 0.5f, movementY);

        // 为刚体施加力
        rb.AddForce(vector3D * speed);
    }

    void OnMove(InputValue movement)
    {
        Vector2 vector2D = movement.Get<Vector2>();
        movementX = vector2D.x;
        movementY = vector2D.y;
    }

    // 设置 postion
    void SetPlayerPosition(float x, float y)
    {
        transform.position = new Vector3(x, transform.position.y, y);
    }

    // 存档
    void SetData()
    {
        PlayerData data = new PlayerData();
        data.x = transform.position.x;
        data.y = transform.position.z;

        SaveGame.Save<PlayerData>(identifier, data);
    }

    // 读档
    void GetData()
    {
        // 校验数据是否存在
        if (SaveGame.Exists(identifier))
        {
            PlayerData playerPosition = SaveGame.Load<PlayerData>(identifier);
            SetPlayerPosition(playerPosition.x, playerPosition.y);
        }
    }

    // 删档
    void DeleteData()
    {
        SaveGame.Clear();
    }
}

编辑球体组件

添加组件,以及修改 transform 值
编辑 Player Input 组件
  • 新建 Inputs 目录,并保存 input action
  • 绑定 input action

运行项目

移动:WSAD / 方向键
读档:自动
存档:Q
删档:E


升级为云存储

云存储功能建议自己实现,重写 SaveGameWeb 类,以及选择更合适的服务端技术选型。以下是官方文档的实现

配置数据库(MySQL)

Git 仓库 Assets/BayatGames/SaveGameFree/Web/Save Game Free - Web/savegamefree.sql

CREATE DATABASE IF NOT EXISTS `savegamefree`;

USE savegamefree;

CREATE TABLE IF NOT EXISTS savegames (
  id VARCHAR(255) CHARACTER SET utf8 NOT NULL PRIMARY KEY,
  data LONGTEXT CHARACTER SET utf8 NOT NULL
);
初始化项目(Nodejs)
mkdir SaveGameFreeServer
cd SaveGameFreeServer

npm init -y
npm i koa koa-router koa-bodyparser mysql

touch index.js
编写入口文件(index.js)

Git 仓库 Assets/BayatGames/SaveGameFree/Web/Save Game Free - Web/savegamefree.php
以下是基于上述 PHP 脚本改造的 Nodejs 实现

const Koa = require('koa')
const Router = require('koa-router')
const bodyParser = require('koa-bodyparser')
const mysql = require('mysql')
 


const config = {
  portServer: 3000,
  database: {
    DATABASE: 'savegamefree',
    TABLE: 'savegames',
    USERNAME: '数据库账号',
    PASSWORD: '数据库密码',
    HOST: '数据库域名'
  }
}

const pool = mysql.createPool({
  database: config.database.DATABASE,
  user: config.database.USERNAME,
  password: config.database.PASSWORD,
  host: config.database.HOST
})

class Mysql {
  constructor() { }
  setPositionById(id, data) {
    return new Promise((resolve, reject) => {
      pool.query(`INSERT INTO ${config.database.TABLE} (id, data) VALUES ('${id}', ${JSON.stringify(data)}) ON DUPLICATE KEY UPDATE data=VALUES (data)`, function (error, results, fields) {
        if (error) {
          throw error
        }
        resolve(results)
      })
    })
  }
  getPositionById(id) {
    return new Promise((resolve, reject) => {
      pool.query(`SELECT data FROM ${config.database.TABLE} WHERE id='${id}'`, function (error, results, fields) {
        if (error) {
          throw error
        }
        resolve(results)
      })
    })
  }
} 



const app = new Koa()
const router = new Router()
const mysqlInstance = new Mysql()

app
  .use(bodyParser())
  .use(router.routes())
  .use(router.allowedMethods())

router
  .post('/savegamefree', async (ctx, next) => {
    const postData = ctx.request.body
    if (postData.action === 'save') {
      const res = await mysqlInstance.setPositionById(postData.identifier, postData.data)
      ctx.body = res
    }
    else if (postData.action === 'load') {
      const res = await mysqlInstance.getPositionById(postData.identifier)
      ctx.body = res[0] ? res[0].data : null
    }
  })



app.listen(config.portServer)
console.log(`listening on port ${config.portServer}`)
运行项目
node index

修改脚本 PlayerController.cs(云存储版本)

using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using UnityEngine.InputSystem;
using BayatGames.SaveGameFree;


public class PlayerController : MonoBehaviour
{

    public float speed = 10;
    private Rigidbody rb;
    private float movementX;
    private float movementY;
    private string identifier = "playerPosition";

    // 自定义数据类,保存 player 位置信息
    public class PlayerData
    {
        public float x;
        public float y;
    }

    void Start()
    {

        GetDataCloud();

        rb = GetComponent<Rigidbody>();
    }

    void Update()
    {
        // 存档快捷键 q(方便调试,不与 IDE 键位冲突)
        if (Input.GetKeyDown(KeyCode.Q))
        {
            SetDataCloud();
        }

        // 删档快捷键 e(方便调试,不与 IDE 键位冲突)
        if (Input.GetKeyDown(KeyCode.E))
        {
            DeleteData();
        }
    }

    void FixedUpdate()
    {
        // 组装 3D 向量
        Vector3 vector3D = new Vector3(movementX, 0.5f, movementY);

        // 为刚体施加力
        rb.AddForce(vector3D * speed);
    }

    void OnMove(InputValue movement)
    {
        Vector2 vector2D = movement.Get<Vector2>();
        movementX = vector2D.x;
        movementY = vector2D.y;
    }

    // 设置 postion
    void SetPlayerPosition(float x, float y)
    {
        transform.position = new Vector3(x, transform.position.y, y);
    }

    // 云存档
    void SetDataCloud()
    {
        StartCoroutine(SaveEnumerator());
    }

    // 云读档
    void GetDataCloud()
    {
        transform.localScale = Vector3.zero;    // 先隐藏球体,防止异步更新位置 导致球体位置闪烁
        StartCoroutine(LoadEnumerator());
    }

    // http://localhost.charlesproxy.com:3000 映射到 http://localhost:3000
    // 防止 charles 抓不到 unity 的请求包
    IEnumerator SaveEnumerator()
    {
        SaveGameWeb web = new SaveGameWeb("usernameFaker", "passwordFaker", "http://localhost.charlesproxy.com:3000/savegamefree");

        PlayerData data = new PlayerData();
        data.x = transform.position.x;
        data.y = transform.position.z;

        yield return StartCoroutine(web.Save<PlayerData>(identifier, data));
    }

    IEnumerator LoadEnumerator()
    {
        SaveGameWeb web = new SaveGameWeb("usernameFaker", "passwordFaker", "http://localhost.charlesproxy.com:3000/savegamefree");

        yield return StartCoroutine(web.Download(identifier));

        PlayerData defaultData = new PlayerData();
        defaultData.x = 0.0f;
        defaultData.y = 0.0f;
        PlayerData playerPosition = web.Load<PlayerData>(identifier, defaultData);
        SetPlayerPosition(playerPosition.x, playerPosition.y);
        transform.localScale = Vector3.one;
    }
}


运行项目

移动:WSAD / 方向键
读档:自动
存档:Q

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,772评论 6 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,458评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,610评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,640评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,657评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,590评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,962评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,631评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,870评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,611评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,704评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,386评论 4 319
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,969评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,944评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,179评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,742评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,440评论 2 342

推荐阅读更多精彩内容