zoukankan      html  css  js  c++  java
  • Cocos Creator 实现大厅+子游戏模式

      大厅+子游戏的模式,在棋牌类型、教育类型游戏中比较常见,通常是安装包里面只有大厅的资源和代码,然后子游戏根据需求以热更新的方式下载来提供给玩家。

      之前一直负责的是cocos2dx-lua的开发,lua作为脚本语言,非常适合做热更新及大厅+子游戏模式的开发。而cocos creator使用的是js或者ts,作为脚本语言也是很方便做热更新的,但是js编译是把所有的脚本编译成一个单独的js文件,如果不做调整,会导致所有的大厅+游戏代码编译成一个单独的js文件,是不太方便做成大厅+子游戏的模式的,这里要讨论的就是如何调整,以做成大厅+子游戏的模式。

      首先就是要将大厅、子游戏分开来构建了,要不就是创建单独的大厅+子游戏工程,要不就是将大厅或者子游戏的代码、资源拷贝到一个构建工程,然后用命令行工具或者直接使用GUI工具构建。我这边没有单独分开创建工程,选择的是在构建的时候通过脚本拷贝相关的资源,然后单独构建。

      分开构建/编译好了资源之后,就是在app中怎么使用了。在lua版本中,引擎的资源加载方式是直接读取指定目录的脚本/资源,所以我们只要先将子游戏的脚本/资源下载好,再引入指定目录的脚本就可以了。js作为脚本资源,思路上我们也是读取对应目录的脚本,但是在creator版本中,引擎封装了一套资源加载工具,每个资源对应一个uuid,访问资源的时候是使用uuid去寻找资源(我们使用cc.loader.loadRes传入的是带资源目录的url,内部会根据这个url找到uuid再来加载资源),构建/编译项目的时候,会生成一个setting.js/jsc的文件,这个文件就是uuid和实际资源路径的对应表。所以我们需要做的就是如何引入子游戏生成的这个setting文件。参考论坛网友的思路,就是引入另外一个js脚本,在该脚本中再去读取对应子游戏的setting文件,合并到大厅的setting配置中,然后再跳转到对应的游戏场景,上代码:

        // 首先还是要设置好搜索路径
        var searchPaths = jsb.fileUtils.getSearchPaths();
        searchPaths.unshift(cc.JS_DIR);
        jsb.fileUtils.setSearchPaths(searchPaths); 
        
        // 判断是否已经加载过子游戏的setting
        if (!cc.gameSetting){
            window.require(js_path + 'src/settings.js');
            settings = window._CCSettings;
            window._CCSettings = undefined;
    
            // 防止重复加载脚本
            if (!cc.jsList[js_path]){
                require(js_path + 'src/' + (settings.debug ? 'project.dev.js' : 'project.js'));
                cc.jsList[js_path] = true
            }
        }
        else{
            settings = cc.gameSetting
        }
                
        // 合并assetTypes
        var gameAssetTypes = settings.assetTypes;
        settings.assetTypes = baseSetting.assetTypes;
        if (gameAssetTypes && settings.assetTypes){
            for (var typeIndex in gameAssetTypes) {
                var type = gameAssetTypes[typeIndex];
                //不包含就塞到settings里面去
                if (settings.assetTypes.indexOf(type) == -1) {
                    settings.assetTypes.push(type);
                }
            }
            for (var uuidKey in settings.rawAssets.assets) {
                var index = settings.rawAssets.assets[uuidKey][1];
                var type1 = gameAssetTypes[index];
    
                for (var typeIndex in settings.assetTypes) {
                    var type2 = settings.assetTypes[typeIndex];
                    if (type1 == type2) {
                        settings.rawAssets.assets[uuidKey][1] = parseInt(typeIndex);
                    }
                }
            }
        }    
        // 调整资源配置
        for (var assetkey in baseSetting.packedAssets) {
            settings.packedAssets[assetkey] = baseSetting.packedAssets[assetkey];
        }
        //动态资源合并
        for (var uuidKey in baseSetting.rawAssets.assets) {        
            settings.rawAssets.assets[uuidKey] = baseSetting.rawAssets.assets[uuidKey];
        }
        //场景合并
        for (var sceneKey in baseSetting.scenes) {
            if (settings.scenes.indexOf(baseSetting.scenes[sceneKey]) == -1){
                settings.scenes.push(baseSetting.scenes[sceneKey]);
            }        
        }
        // uuid合并
        for (var uuidKey in baseSetting.uuids) {
            if (settings.uuids.indexOf(baseSetting.uuids[uuidKey]) == -1) {
                settings.uuids.push(baseSetting.uuids[uuidKey])
            }
        }

      上述代码是在论坛网友提供的demo基础上进行了部分调整,核心的逻辑还是一致的:主要就是读取对应子游戏的setting文件,然后合并到大厅的setting中,建立好子游戏资源的uuid对应关系,主要就能在游戏中引入对应的游戏资源。

      论坛网友提供的demo中,从子游戏回到大厅,需要再引入一份独立的js文件。但是我的理解是,在启动大厅的时候已经将大厅的setting加入到内存中了,资源和uuid的对应关系已经建立,这个时候其实已经没有必要再重复引入一次大厅的setting配置,再来合并。实际项目上,我也是按照我的理解,没有再单独的引入js文件实现从子游戏回到大厅,目前也暂时没有碰到问题。

      另外,因为思路上是大厅和子游戏要分开打包,在开发的过程中我们是可以大厅+子游戏一起开发。但是要注意的是,子游戏不能直接在编辑器中引用大厅的资源,比如子游戏的某个脚本是继承自大厅的,,如:

    cc.Class({
        extends:  bg.GameModel,
        ...
    })

      GamModel是大厅工程的代码,在开发的过程中,因为子游戏和大厅在一个工程,这样直接使用是没有问题的。如果将子游戏单独打包,在构建的过程中会报错,不过还是能构建成功,但是在运行的时候这个脚本组件就不会绑定到对应的节点上去,所以应该调整一下:

    cc.Class({
        extends: window.bg == undefined ? cc.Component : bg.GameModel,
        ...
    )}

      这样构建的过程中不会报错,脚本组件能正常的绑定到对应的节点上去。运行过程中因为已经加载了大厅的代码,所以这个三目运算的结果是取后面的bg.GameModel。

      

      实现环境:Cocos Creator 2.0.9 版本。之前使用2.10版本,同样的代码构建的工程,在win32模拟器上就会一直报错,自带模拟器对应的src目录下的modular.js会出现错误,后来换成2.0.9版本,自编译win32工程,再用模拟器运行就没有问题了。在安卓和ios上到是都正常的。

  • 相关阅读:
    Codeforces Round #454 Div. 2 A B C (暂时)
    Codeforces Round #453 Div. 2 A B C D (暂时)
    EOJ Monthly 2017.12 A B C D
    C++调用Matlab引擎 图像读写与处理 (知识+代码篇)
    Codeforces Round #449 Div. 2 A B C (暂时)
    AtCoder Regular Contest 077 E
    hdu 6218 Bridge 线段树 set
    hdu 2243 考研路茫茫——单词情结 AC自动机 矩阵幂次求和
    php配置php-fpm启动参数及配置详解
    PHP连接MySQL数据库的三种方式(mysql、mysqli、pdo)
  • 原文地址:https://www.cnblogs.com/zhong-dev/p/10782834.html
Copyright © 2011-2022 走看看