使用gulp對微信小游戲進(jìn)行分包

方案

  1. 生成未壓縮混淆的游戲整包 wxgame.js
  2. 截取中間部分標(biāo)記切割wxgame.js為兩個(gè)wxgame.js
  3. 把依賴包含到window上

示例

gulp task配置


/**
 * 編譯成單個(gè)js  不創(chuàng)建 window.a =a ;
 * 
 * @param {*} fileName 生成的名字
 * @param {*} outPath 輸出路徑
 * @returns
 */
function compileWXTSFile(outPath, fileName) {
    // 混淆參數(shù)
    const uglifyOptions = {
        mangle: true,// 混淆
        compress: false // 不壓縮
    }
    return gulp.src([
        "libs/*.ts",
        "src/**/*.ts"
    ])
        .pipe(sorter(false)) // 排序  false 不顯示日志
        .pipe(tsProj()) // 編譯TS文件
        .pipe(concat(fileName)) // 合并編譯后的js到一個(gè)文件
        .pipe(gulp.dest(outPath))
}

/**
 *  篩選出 window.a=a 生成獨(dú)立js  
 */
function createWXExport() {
    // 進(jìn)行編譯 不進(jìn)行混淆
    return gulp.src([
        "libs/*.ts",
        "src/**/*.ts"
    ])
        .pipe(sorter(false)) // 排序  false 不顯示日志
        .pipe(tsProj()) // 編譯TS文件 // 讀取已經(jīng)混淆好的整包
        .pipe(wxTools(true)) //  篩選頂層引用 用 window.a = a 包含
        .pipe(concat("wxgame_export.js"))
        .pipe(gulp.dest("wxgame_out/"))
}


/**
 * 添加  window.a=a 到尾部
 * @param { } inFile 
 * @param {*} outPath 
 * @param {*} outFileName 
 */
var conactExport = function (inPath, outPath, fileName) {
    return gulp.src([
        inPath + fileName,
        gameJSExport,
    ])
        .pipe(concat(fileName))
        .pipe(gulp.dest(outPath))
        .on("end", () => {
            console.log("sub success =>> ", outPath, fileName)
        })
}


/**
 *  分包
 * @param {string } path  切割后輸出路徑
 * @param {string} fileName  生成文件名
 * @param {boolean} isLast 
 * @param {string } head 
 */
function subWX(tempPath, outPath, fileName, isLast, head) {
    var key = `define("game/battle/control/BattleDataControl"`
    var uglifyArgs = {
        mangle: false, // 分包不再進(jìn)行混淆 只壓縮空白
        compress: {}
    }

    return gulp.src([gameJS,])
        .pipe(wxSub(key, isLast, head))
        .pipe(wxTools()) //  篩選頂層引用 用 window.a = a 包含
        .pipe(concat(fileName))
        // .pipe(uglify(uglifyArgs)) //打開則進(jìn)行混淆這里方便調(diào)試
        .pipe(gulp.dest(tempPath))
        .on("end", () => { // 結(jié)束后
            return conactExport(tempPath, outPath, fileName);
        })
}


// 微信分包
gulp.task('wxsub', function (args, args2) {

    var head = `var Laya = window.Laya;\n`;
    var head2 = `  var Laya = window.Laya;var __extends = window.__extends;\n`;
    // 編譯
    if (compile && compile.c != null) {
        console.log("start wx compile");
        return compileWXTSFile("wxgame_out", "wxgame.js").on("end", () => {
            console.log("compileWXTSFile Finish");
            createWXExport().on("end", () => {
                subWX("wxgame_out/sub1/", "wxgame/sub1/", "wxgame.js", false, head);
                subWX("wxgame_out/sub2/", "wxgame/sub2/", "wxgame.js", true, head2);
            })
        })
    } else {
       // 不編譯 直接拿已經(jīng)編譯的進(jìn)行切割
        subWX("wxgame_out/sub1/", "wxgame/sub1/", "wxgame.js", false, head);
        subWX("wxgame_out/sub2/", "wxgame/sub2/", "wxgame.js", true, head2);
    }

});

gulp 插件

根據(jù)key截取代碼

'use strict';

var through = require('through2');
var path = require('path');
var File = require('vinyl');
var Concat = require('concat-with-sourcemaps');
var gutil = require('gulp-util');
var PluginError = gutil.PluginError;

/**
 *  微信分包工具 被冒,基于amd模式下開發(fā)的游戲
 *  將 var  XXX 包裝到 window.XXX =XXX;
 * @param key 分包關(guān)鍵位置
 * @param   isLast  是否是取后面的
 * @param isMain  是否是主包 主包不分包 但是加頭部
 */
var wxgameSubTools = function (key, isLast, head, isMain) {

    function onFile(file, enc, cb) {
        if (file.isStream()) {
            console.error('Streams are not supported!');
            return cb();
        }
        let contents = "" + file.contents;
        let outStr;

        if (!isMain) {
            if (key == null || key == "") {
                console.error("key ", 'key= ' + key);
                return cb();
            }
            const index = contents.indexOf(key);
            if (index == -1) {
                console.error('not find key ! key= ' + key);
                return cb();
            }
            if (!isLast) { // 
                outStr = contents.substr(0, contents.indexOf(key));
            } else {
                outStr = contents.substr(contents.indexOf(key));
            }
        }else{
            outStr = contents;
        }
        // 頭部添加的文本
        if (head) {
            outStr = head + outStr;
        }
        file.contents = new Buffer(outStr);
        cb();
        this.emit("data", file);
    }
    // 不處理end 使用默認(rèn)的end
    return through.obj(onFile);
};

module.exports = wxgameSubTools;

生成分包 ,提取依賴到window

'use strict';

var through = require('through2');
var path = require('path');
var File = require('vinyl');
var Concat = require('concat-with-sourcemaps');

/**
 *  微信分包工具 军掂,基于amd模式下開發(fā)的游戲
 *  將 var  XXX 包裝到 window.XXX =XXX;
 * @param {boolean } isExport  是否是只輸出window.a=a
 */
var wxgameTools = function (isExport) {

    /**
     * 流處理  匹配 /(?:^|\n)(function|var) [_0-9a-zA-Z]+/ig; 添加為 window.XXX = XXX;
     *
     * @param {*} file 流文件 
     * @returns
     */
    function onFile(file, enc, cb) {
        if (file.isStream()) {
            this.emit('error', new PluginError(PLUGIN_NAME, 'Streams are not supported!'));
            return cb();
        }

        var contents = "" + file.contents;
        // 匹配fuanction XXX   或者 var XXX 
        var reg = /(?:^|\n)(function|var) [_0-9a-zA-Z]+/ig;
        // 匹配fuanction   或者 var  去掉 function 或者 var 
        var reg2 = /(?:^|\n)(function|var) /ig;
        var match = contents.match(reg)
        if (match && match.length > 0) {
            var strs = ""
            for (var i = 0; i < match.length; i++) {
                var str = match[i].replace(reg2, "");
                strs += "\n try{window." + str + "=" + str + ";}catch(e){console.warn("+'"'+str+'")}';
            }
            var bf = new Buffer( strs ); // 提前分配
            if(isExport){
                file.contents = bf;
            }else{
                file.contents = Buffer.concat([file.contents, bf])
            }
            
        }
        cb();
        this.emit("data",file);
    }
    // 不處理end 使用默認(rèn)的end
    return through.obj(onFile);
};

module.exports = wxgameTools;
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市昨悼,隨后出現(xiàn)的幾起案子蝗锥,更是在濱河造成了極大的恐慌,老刑警劉巖幔戏,帶你破解...
    沈念sama閱讀 217,277評論 6 503
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件玛追,死亡現(xiàn)場離奇詭異,居然都是意外死亡闲延,警方通過查閱死者的電腦和手機(jī)痊剖,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,689評論 3 393
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來垒玲,“玉大人陆馁,你說我怎么就攤上這事『嫌” “怎么了叮贩?”我有些...
    開封第一講書人閱讀 163,624評論 0 353
  • 文/不壞的土叔 我叫張陵击狮,是天一觀的道長。 經(jīng)常有香客問我益老,道長彪蓬,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,356評論 1 293
  • 正文 為了忘掉前任捺萌,我火速辦了婚禮档冬,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘桃纯。我一直安慰自己酷誓,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,402評論 6 392
  • 文/花漫 我一把揭開白布态坦。 她就那樣靜靜地躺著盐数,像睡著了一般。 火紅的嫁衣襯著肌膚如雪伞梯。 梳的紋絲不亂的頭發(fā)上玫氢,一...
    開封第一講書人閱讀 51,292評論 1 301
  • 那天,我揣著相機(jī)與錄音谜诫,去河邊找鬼琐旁。 笑死,一個(gè)胖子當(dāng)著我的面吹牛猜绣,可吹牛的內(nèi)容都是我干的灰殴。 我是一名探鬼主播,決...
    沈念sama閱讀 40,135評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼掰邢,長吁一口氣:“原來是場噩夢啊……” “哼牺陶!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起辣之,我...
    開封第一講書人閱讀 38,992評論 0 275
  • 序言:老撾萬榮一對情侶失蹤掰伸,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后怀估,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體狮鸭,經(jīng)...
    沈念sama閱讀 45,429評論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,636評論 3 334
  • 正文 我和宋清朗相戀三年多搀,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了歧蕉。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,785評論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡康铭,死狀恐怖惯退,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情从藤,我是刑警寧澤催跪,帶...
    沈念sama閱讀 35,492評論 5 345
  • 正文 年R本政府宣布锁蠕,位于F島的核電站,受9級特大地震影響懊蒸,放射性物質(zhì)發(fā)生泄漏荣倾。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,092評論 3 328
  • 文/蒙蒙 一骑丸、第九天 我趴在偏房一處隱蔽的房頂上張望逃呼。 院中可真熱鬧,春花似錦者娱、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,723評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至平匈,卻和暖如春框沟,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背增炭。 一陣腳步聲響...
    開封第一講書人閱讀 32,858評論 1 269
  • 我被黑心中介騙來泰國打工忍燥, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人隙姿。 一個(gè)月前我還...
    沈念sama閱讀 47,891評論 2 370
  • 正文 我出身青樓梅垄,卻偏偏與公主長得像,于是被迫代替她去往敵國和親输玷。 傳聞我的和親對象是個(gè)殘疾皇子队丝,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,713評論 2 354

推薦閱讀更多精彩內(nèi)容