CocosCreator大厅+子游戏(v1.10.2)

基于热更新的基础上,将子游戏构建生成的main.js文件一并移入src目录,在运行子游戏的时候,我们只需要require main.js这个文件即可。

大厅跳到子游戏

首先是大厅封装好的子游戏管理类,包括子游戏下载、更新、进入

export  class SubgameManager  {

    private static serverUrl;
    private static storagePath:[] = [];


    private static assertsMg;
    private static jsbCallback;

    private static subgameUpdateCallback;
    private static progressCallback;
    private static finishCallback;

    public static init(serverUrl:string){
        this.serverUrl = serverUrl;
    }

    public static isSubgameDownload(name:string):boolean{

        let file = (jsb.fileUtils ? jsb.fileUtils.getWritablePath() : '/') + 'ALLGame/' + name + '/project.manifest';
        if (jsb.fileUtils.isFileExist(file)) {
            return true;
        } else {
            return false;
        }
    }

    public static isNeedUpdateSubgame(name:string,subgameUpdateCallback?:Function){
        this.prepareJsb(name);
        this.subgameUpdateCallback = subgameUpdateCallback;
        this.jsbCallback =  new jsb.EventListenerAssetsManager(this.assertsMg, this.needUpdateCallback.bind(this));
        cc.eventManager.addListener(this.jsbCallback, 1);
        this.assertsMg.checkUpdate();
    }

    private static needUpdateCallback(event){
        let self = this;
        switch (event.getEventCode()) {
            case jsb.EventAssetsManager.ALREADY_UP_TO_DATE:
                cc.log('子游戏已经是最新的,不需要更新');
                self.subgameUpdateCallback && self.subgameUpdateCallback(false);
                break;

            case jsb.EventAssetsManager.NEW_VERSION_FOUND:
                cc.log('子游戏需要更新');
                self.subgameUpdateCallback && self.subgameUpdateCallback(true);
                break;

            // 检查是否更新出错
            case jsb.EventAssetsManager.ERROR_DOWNLOAD_MANIFEST:
            case jsb.EventAssetsManager.ERROR_PARSE_MANIFEST:
            case jsb.EventAssetsManager.ERROR_UPDATING:
            case jsb.EventAssetsManager.UPDATE_FAILED:
                //self._downloadCallback();
                break;
        }
    }

    public static downloadSubgame(name:string,progressCallback?:Function,finishCallback?:Function){
        this.prepareJsb(name);
        this.progressCallback = progressCallback;
        this.finishCallback = finishCallback;
        this.jsbCallback =  new jsb.EventListenerAssetsManager(this.assertsMg, this.downloadCallback.bind(this));
        cc.eventManager.addListener(this.jsbCallback, 1);
        this.assertsMg.update();
    }

    private static downloadCallback(event) {
        var failed = false;
        let self = this;
        switch (event.getEventCode()) {
            case jsb.EventAssetsManager.ERROR_NO_LOCAL_MANIFEST:
                /*0 本地没有配置文件*/
                cc.log('updateCb本地没有配置文件');
                failed = true;
                break;

            case jsb.EventAssetsManager.ERROR_DOWNLOAD_MANIFEST:
                /*1下载配置文件错误*/
                cc.log('updateCb下载配置文件错误');
                failed = true;
                break;

            case jsb.EventAssetsManager.ERROR_PARSE_MANIFEST:
                /*2 解析文件错误*/
                cc.log('updateCb解析文件错误');
                failed = true;
                break;

            case jsb.EventAssetsManager.NEW_VERSION_FOUND:
                /*3发现新的更新*/
                cc.log('updateCb发现新的更新');
              
                break;

            case jsb.EventAssetsManager.ALREADY_UP_TO_DATE:
                /*4 已经是最新的*/
                cc.log('updateCb已经是最新的');
                failed = true;
                break;

            case jsb.EventAssetsManager.UPDATE_PROGRESSION:
                /*5 最新进展 */
                cc.log("event.getPercentByFile()"+event.getPercentByFile());
                self.progressCallback && self.progressCallback(event.getPercentByFile());
                break;


            case jsb.EventAssetsManager.ASSET_UPDATED:
                /*6需要更新*/
                break;

            case jsb.EventAssetsManager.ERROR_UPDATING:
                /*7更新错误*/
                cc.log('updateCb更新错误');
                break;

            case jsb.EventAssetsManager.UPDATE_FINISHED:
                /*8更新完成*/
                cc.log("UPDATE_FINISHED");
                self.finishCallback && self.finishCallback(true);
                break;

            case jsb.EventAssetsManager.UPDATE_FAILED:
                /*9更新失败*/
                cc.log('UPDATE_FAILED');
                self.assertsMg.downloadFailedAssets();
                
                break;

            case jsb.EventAssetsManager.ERROR_DECOMPRESS:
                /*10解压失败*/
                cc.log('解压失败');
                break;
        }

        if (failed) {
            cc.eventManager.removeListener(self.jsbCallback);
            self.jsbCallback = null;
            self.finishCallback && self.finishCallback(false);
        }
    }

    public static enterSubgame(name) {
        if (!this.storagePath[name]) {
            this.downloadSubgame(name);
            return;
        }

        window.require(this.storagePath[name] + '/src/main.js');
    }


    private static prepareJsb(name:string){
        this.storagePath[name] = ((jsb.fileUtils ? jsb.fileUtils.getWritablePath() : '/') + 'ALLGame/' + name);
      
        var UIRLFILE = this.serverUrl +"/"+ name;
    
        var customManifestStr = JSON.stringify({
            'packageUrl': UIRLFILE,
            'remoteManifestUrl': UIRLFILE + '/project.manifest',
            'remoteVersionUrl': UIRLFILE + '/version.manifest',
            'version': '0.0.1',
            'assets': {},
            'searchPaths': []
        });

        this.assertsMg = new jsb.AssetsManager('', this.storagePath[name], this.versionCompare);

        if (!cc.sys.ENABLE_GC_FOR_NATIVE_OBJECTS) {
            this.assertsMg.retain();
        }

        this.assertsMg.setVerifyCallback(function(path, asset) {
            var compressed = asset.compressed;
            if (compressed) {
                return true;
            } else {
                return true;
            }
        });

        if (cc.sys.os === cc.sys.OS_ANDROID) {
            this.assertsMg.setMaxConcurrentTask(2);
        }

        if (this.assertsMg.getState() === jsb.AssetsManager.State.UNINITED) {
            var manifest = new jsb.Manifest(customManifestStr, this.storagePath[name]);
            this.assertsMg.loadLocalManifest(manifest, this.storagePath[name]);
        }

    }

    private static versionCompare(versionA, versionB):number{

        var vA = versionA.split('.');
            var vB = versionB.split('.');
            for (var i = 0; i < vA.length; ++i) {
                var a = parseInt(vA[i]);
                var b = parseInt(vB[i] || 0);
                if (a === b) {
                    continue;
                } else {
                    return a - b;
                }
            }
            if (vB.length > vA.length) {
                return -1;
            } else {
                return 0;
            }
    }
 
}

接着看使用这个类:

import { SubgameManager } from "./SubgameManager";

const {ccclass, property} = cc._decorator;

@ccclass
export default class Subgame extends cc.Component {

    @property(cc.Label)
    label: cc.Label = null;

    private subgame = "Niuniu"

    onLoad(){

        SubgameManager.init("http://192.168.0.136:8000");

        if(SubgameManager.isSubgameDownload(this.subgame)){

            SubgameManager.isNeedUpdateSubgame(this.subgame,(isSuccess)=>{
                this.label.string = isSuccess ? "子游戏需要更新" : "子游戏不需要更新";
            });

        }else{
            this.label.string = "子游戏未下载";
        }

    }

    click(){
        SubgameManager.downloadSubgame(this.subgame,(progress)=>{
            if (isNaN(progress)) {
                progress = 0;
            }
            this.label.string = "资源下载中   " + ~~(progress * 100) + "%";
        },(success)=>{
            if (success) {
                SubgameManager.enterSubgame(this.subgame);
            } 
        })
    }
}

准备好之后,开始准备小游戏,首先将小游戏构建下,模板是default,如果使用脚本加密,那么大厅与子游戏脚本加密的key一定要相同!!因为主程序是大厅,解密脚本都是用大厅的key。构建成功后,将main.js复制一份到src下,然后打开修改两个地方。无论creator哪个版本,以构建出来的main.js为主,然后同样修改这两地方就好了

 //~~~~~~~~~1、添加这段~~~~~~~~~~~~~~~
     cc.director.startAnimation();  //官方说解决个BUG
     'use strict';
     //后面的路径根据自己的游戏修改
     cc.INGAME = (jsb.fileUtils ? jsb.fileUtils.getWritablePath() : '/')+'ALLGame/Niuniu/';  
 //~~~~~~~~~~~~~~~~~~~~~~~~~~
 //~~~~~~~~~2.修改这段~~~~~~~~~~~~~~~
      require(cc.INGAME + 'src/settings.js');
      require(cc.INGAME  +window._CCSettings ? 'src/project.dev.js' : 'src/project.js');
      // require('src/jsb_polyfill.js');
 //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~

整份main.js如下(v1.10.2):

(function () {

    //~~~~~~~~~1、添加这段~~~~~~~~~~~~~~~
     cc.director.startAnimation();  //官方说解决个BUG
     'use strict';
     cc.INGAME = (jsb.fileUtils ? jsb.fileUtils.getWritablePath() : '/')+'ALLGame/Niuniu/';  
     //~~~~~~~~~~~~~~~~~~~~~~~~~~

    function boot () {

        var settings = window._CCSettings;
        window._CCSettings = undefined;

        if ( !settings.debug ) {
            var uuids = settings.uuids;

            var rawAssets = settings.rawAssets;
            var assetTypes = settings.assetTypes;
            var realRawAssets = settings.rawAssets = {};
            for (var mount in rawAssets) {
                var entries = rawAssets[mount];
                var realEntries = realRawAssets[mount] = {};
                for (var id in entries) {
                    var entry = entries[id];
                    var type = entry[1];
                    // retrieve minified raw asset
                    if (typeof type === 'number') {
                        entry[1] = assetTypes[type];
                    }
                    // retrieve uuid
                    realEntries[uuids[id] || id] = entry;
                }
            }

            var scenes = settings.scenes;
            for (var i = 0; i < scenes.length; ++i) {
                var scene = scenes[i];
                if (typeof scene.uuid === 'number') {
                    scene.uuid = uuids[scene.uuid];
                }
            }

            var packedAssets = settings.packedAssets;
            for (var packId in packedAssets) {
                var packedIds = packedAssets[packId];
                for (var j = 0; j < packedIds.length; ++j) {
                    if (typeof packedIds[j] === 'number') {
                        packedIds[j] = uuids[packedIds[j]];
                    }
                }
            }
        }

        // init engine
        var canvas;

        if (cc.sys.isBrowser) {
            canvas = document.getElementById('GameCanvas');
        }

        if (false) {
            var ORIENTATIONS = {
                'portrait': 1,
                'landscape left': 2,
                'landscape right': 3
            };
            BK.Director.screenMode = ORIENTATIONS[settings.orientation];
            initAdapter();
        }

        function setLoadingDisplay () {
            // Loading splash scene
            var splash = document.getElementById('splash');
            var progressBar = splash.querySelector('.progress-bar span');
            cc.loader.onProgress = function (completedCount, totalCount, item) {
                var percent = 100 * completedCount / totalCount;
                if (progressBar) {
                    progressBar.style.width = percent.toFixed(2) + '%';
                }
            };
            splash.style.display = 'block';
            progressBar.style.width = '0%';

            cc.director.once(cc.Director.EVENT_AFTER_SCENE_LAUNCH, function () {
                splash.style.display = 'none';
            });
        }

        var onStart = function () {
            cc.loader.downloader._subpackages = settings.subpackages;

            if (false) {
                BK.Script.loadlib();
            }

            cc.view.resizeWithBrowserSize(true);

            if (!false && !false) {
                if (cc.sys.isBrowser) {
                    setLoadingDisplay();
                }

                if (cc.sys.isMobile) {
                    if (settings.orientation === 'landscape') {
                        cc.view.setOrientation(cc.macro.ORIENTATION_LANDSCAPE);
                    }
                    else if (settings.orientation === 'portrait') {
                        cc.view.setOrientation(cc.macro.ORIENTATION_PORTRAIT);
                    }
                    cc.view.enableAutoFullScreen([
                        cc.sys.BROWSER_TYPE_BAIDU,
                        cc.sys.BROWSER_TYPE_WECHAT,
                        cc.sys.BROWSER_TYPE_MOBILE_QQ,
                        cc.sys.BROWSER_TYPE_MIUI,
                    ].indexOf(cc.sys.browserType) < 0);
                }

                // Limit downloading max concurrent task to 2,
                // more tasks simultaneously may cause performance draw back on some android system / browsers.
                // You can adjust the number based on your own test result, you have to set it before any loading process to take effect.
                if (cc.sys.isBrowser && cc.sys.os === cc.sys.OS_ANDROID) {
                    cc.macro.DOWNLOAD_MAX_CONCURRENT = 2;
                }
            }

            // init assets
            cc.AssetLibrary.init({
                libraryPath: 'res/import',
                rawAssetsBase: 'res/raw-',
                rawAssets: settings.rawAssets,
                packedAssets: settings.packedAssets,
                md5AssetsMap: settings.md5AssetsMap
            });

            if (false) {
                cc.Pipeline.Downloader.PackDownloader._doPreload("WECHAT_SUBDOMAIN", settings.WECHAT_SUBDOMAIN_DATA);
            }

            var launchScene = settings.launchScene;

            // load scene
            cc.director.loadScene(launchScene, null,
                function () {
                    if (cc.sys.isBrowser) {
                        // show canvas
                        canvas.style.visibility = '';
                        var div = document.getElementById('GameDiv');
                        if (div) {
                            div.style.backgroundImage = '';
                        }
                    }
                    cc.loader.onProgress = null;
                    console.log('Success to load scene: ' + launchScene);
                }
            );
        };

        // jsList
        var jsList = settings.jsList;

        if (!false) {
            var bundledScript = settings.debug ? 'src/project.dev.js' : 'src/project.js';
            if (jsList) {
                jsList = jsList.map(function (x) {
                    return 'src/' + x;
                });
                jsList.push(bundledScript);
            }
            else {
                jsList = [bundledScript];
            }
        }

        // anysdk scripts
        if (cc.sys.isNative && cc.sys.isMobile) {
//            jsList = jsList.concat(['src/anysdk/jsb_anysdk.js', 'src/anysdk/jsb_anysdk_constants.js']);
        }

        var option = {
            //width: width,
            //height: height,
            id: 'GameCanvas',
            scenes: settings.scenes,
            debugMode: settings.debug ? cc.DebugMode.INFO : cc.DebugMode.ERROR,
            showFPS: (!false && !false) && settings.debug,
            frameRate: 60,
            jsList: jsList,
            groupList: settings.groupList,
            collisionMatrix: settings.collisionMatrix,
            renderMode: 0
        }

        cc.game.run(option, onStart);
    }

    if (false) {
        BK.Script.loadlib('GameRes://libs/qqplay-adapter.js');
        BK.Script.loadlib('GameRes://src/settings.js');
        BK.Script.loadlib();
        BK.Script.loadlib('GameRes://libs/qqplay-downloader.js');
        qqPlayDownloader.REMOTE_SERVER_ROOT = "";
        var prevPipe = cc.loader.md5Pipe || cc.loader.assetLoader;
        cc.loader.insertPipeAfter(prevPipe, qqPlayDownloader);
        // <plugin script code>
        boot();
        return;
    }

    if (false) {
        require(window._CCSettings.debug ? 'cocos2d-js.js' : 'cocos2d-js-min.js');
        require('./libs/weapp-adapter/engine/index.js');
        var prevPipe = cc.loader.md5Pipe || cc.loader.assetLoader;
        cc.loader.insertPipeAfter(prevPipe, wxDownloader);
        boot();
        return;
    }

    if (window.jsb) {

        //~~~~~~~~~2.修改这段~~~~~~~~~~~~~~~
        require(cc.INGAME + 'src/settings.js');
        require(cc.INGAME  +window._CCSettings ? 'src/project.dev.js' : 'src/project.js');
          // require('src/jsb_polyfill.js');
        //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
      
        boot();
        return;
    }

    if (window.document) {
        var splash = document.getElementById('splash');
        splash.style.display = 'block';

        var cocos2d = document.createElement('script');
        cocos2d.async = true;
        cocos2d.src = window._CCSettings.debug ? 'cocos2d-js.js' : 'cocos2d-js-min.js';

        var engineLoaded = function () {
            document.body.removeChild(cocos2d);
            cocos2d.removeEventListener('load', engineLoaded, false);
            if (typeof VConsole !== 'undefined') {
                window.vConsole = new VConsole();
            }
            boot();
        };
        cocos2d.addEventListener('load', engineLoaded, false);
        document.body.appendChild(cocos2d);
    }

})();

修改完成后,利用上一篇热更新提到的version_generator.js,生成project. manifest和version. manifest,这里步骤不能变,一定先构建好子游戏,复制main.js到src并修改,再利用version_generator.js生成project. manifest和version. manifest。准备好之后,将src、res、project. manifest、version. manifest放在服务器:

image.png

然后可以测试跳到子游戏了。

子游戏返回大厅

在大厅跳到子游戏时,我们利用了main.js,同理的,返回大厅也是。首先准好返回大厅的代码,注意我目前的版本需要window.require,网上其他文章好像1.5.1以前只需要require

 returnHall(){
      let subgameSearchPath = (jsb.fileUtils ? jsb.fileUtils.getWritablePath() : '/')+'ALLGame/Niuniu/';
      window.require(subgameSearchPath + 'src/hall.js');
  }

然后设置好点击事件构建后,与上面的步骤一样复制main.js到src并修改,然后将修改完的main.js复制一份,改名为hall.js,修改hall.js的 cc.INGAME,这里区分Android与iOS

 if (cc.sys.os === cc.sys.OS_ANDROID) {
      cc.INGAME = 'assets/';
 }else if(cc.sys.os == cc.sys.OS_IOS){
      cc.INGAME = jsb.reflection.callStaticMethod("AppController", "getHallPath")+"/";
 }

iOS还需要在xcode中,AppController类下加入方法getHallPath:

+ (NSString *)getHallPath
{
    return [[NSBundle mainBundle] bundlePath];
}

解决游戏之间cid、classname冲突问题

A Class already exists with the same cid
cid冲突可能是复制原因造成的,解决的方法是把冲突的脚本移出工程,再等creator刷新后,重新导入进来。

A Class already exists with the same classname
classname冲突,如果是公用的脚本,比如一些通用类,在各个游戏一样的话,可以忽略,creator不会重新加载,但那些有区别的类名又相同的,目前的做法是每个游戏都类名都加游戏前缀。

解决内存问题

已知的问题:

假如进去子游戏一次,退出到大厅,发现更新了,更新子游戏了,再进去子游戏没有更新到,因为子游戏的数据还在内存,不会再去重新load。

子游戏退出到大厅,内存数据还在,下次进入子游戏的数据还是最后一次修改的数据,不会重置。

目前没有很好的方案,我们用了一种偏方,返回大厅都用cc.game.restart,黑屏的问题,利用原生交互弹一张loading,因为cc.game.restart不会重启应用,用一张loading图先盖住creator,等大厅onenable是时候隐藏了loading。

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

推荐阅读更多精彩内容

  • 一、前言 根据上一篇(Cocos Creator热更新),可以看出以下几点:build-default目录下的ma...
    陌上冰火阅读 17,125评论 43 27
  • creator生成的项目是纯粹的数据驱动而非代码驱动,一切皆为数据,包括脚本、图片、音频、动画等,而数据驱动需要一...
    Dane_404阅读 630评论 0 0
  • 版权声明:本文为博主原创文章,未经博主允许不得转载。 webpack介绍和使用 一、webpack介绍 1、由来 ...
    it筱竹阅读 11,028评论 0 21
  • 丘奇先生 最近偏爱温暖的影片。 阿米尔汗的摔跤吧爸爸、神秘巨星;迪士尼的寻梦环游记,还有前几天看的返老还童,都是这...
    宁黛阅读 257评论 0 0
  • 1 那时候的我,不过是六七岁的黄毛丫头。 那也不过是平凡得不能再平凡的川东北冬日的黎明时分,鸡刚鸣过第三遍,用现在...
    阿鹿在写阅读 836评论 1 3