| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215 | /**************************************************************************** Copyright (c) 2010-2012 cocos2d-x.org Copyright (c) 2008-2010 Ricardo Quesada Copyright (c) 2011      Zynga Inc. http://www.cocos2d-x.org Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. ****************************************************************************/cc.RESOLUTION_POLICY = {    // The entire application is visible in the specified area without trying to preserve the original aspect ratio.    // Distortion can occur, and the application may appear stretched or compressed.    EXACT_FIT: 0,    // The entire application fills the specified area, without distortion but possibly with some cropping,    // while maintaining the original aspect ratio of the application.    NO_BORDER: 1,    // The entire application is visible in the specified area without distortion while maintaining the original    // aspect ratio of the application. Borders can appear on two sides of the application.    SHOW_ALL: 2,    // The application takes the height of the design resolution size and modifies the width of the internal    // canvas so that it fits the aspect ratio of the device    // no distortion will occur however you must make sure your application works on different    // aspect ratios    FIXED_HEIGHT: 3,    // The application takes the width of the design resolution size and modifies the height of the internal    // canvas so that it fits the aspect ratio of the device    // no distortion will occur however you must make sure your application works on different    // aspect ratios    FIXED_WIDTH: 4,    UNKNOWN: 5};cc.Touches = [];cc.TouchesIntergerDict = {};/** * @class * @extends cc.Class */cc.EGLView = cc.Class.extend(/** @lends cc.EGLView# */{    _delegate: null,    // Size of parent node that contains cc.container and cc.canvas    _frameSize: null,    // resolution size, it is the size appropriate for the app resources.    _designResolutionSize: null,    _originalDesignResolutionSize: null,    // Viewport is the container's rect related to content's coordinates in pixel    _viewPortRect: null,    // The visible rect in content's coordinate in point    _visibleRect: null,    // The device's pixel ratio (for retina displays)    _devicePixelRatio: 1,    // the view name    _viewName: "",    // Custom callback for resize event    _resizeCallback: null,    _scaleX: 1,    _originalScaleX: 1,    _scaleY: 1,    _originalScaleY: 1,    _indexBitsUsed: 0,    _maxTouches: 5,    _resolutionPolicy: null,    _rpExactFit: null,    _rpShowAll: null,    _rpNoBorder: null,    _rpFixedHeight: null,    _rpFixedWidth: null,    _initialized: false,    _captured: false,    _wnd: null,    _hDC: null,    _hRC: null,    _accelerometerKeyHook: null,    _supportTouch: false,    _contentTranslateLeftTop: null,    _menu: null,    // Parent node that contains cc.container and cc.canvas    _frame: null,    _frameZoomFactor: 1.0,    __resizeWithBrowserSize: false,    _isAdjustViewPort: true,    ctor: function () {        this._frame = (cc.container.parentNode === document.body) ? document.documentElement : cc.container.parentNode;        this._frameSize = cc.size(0, 0);        this._initFrameSize();        var w = cc.canvas.width, h = cc.canvas.height;        this._designResolutionSize = cc.size(w, h);        this._originalDesignResolutionSize = cc.size(w, h);        this._viewPortRect = cc.rect(0, 0, w, h);        this._visibleRect = cc.rect(0, 0, w, h);        this._delegate = cc.Director.getInstance().getTouchDispatcher();        this._contentTranslateLeftTop = {left: 0, top: 0};        this._viewName = "Cocos2dHTML5";        cc.VisibleRect.init(this._designResolutionSize);        // Setup system default resolution policies        this._rpExactFit = new cc.ResolutionPolicy(cc.ContainerStrategy.EQUAL_TO_FRAME, cc.ContentStrategy.EXACT_FIT);        this._rpShowAll = new cc.ResolutionPolicy(cc.ContainerStrategy.PROPORTION_TO_FRAME, cc.ContentStrategy.SHOW_ALL);        this._rpNoBorder = new cc.ResolutionPolicy(cc.ContainerStrategy.EQUAL_TO_FRAME, cc.ContentStrategy.NO_BORDER);        this._rpFixedHeight = new cc.ResolutionPolicy(cc.ContainerStrategy.EQUAL_TO_FRAME, cc.ContentStrategy.FIXED_HEIGHT);        this._rpFixedWidth = new cc.ResolutionPolicy(cc.ContainerStrategy.EQUAL_TO_FRAME, cc.ContentStrategy.FIXED_WIDTH);        this._hDC = cc.canvas;        this._hRC = cc.renderContext;    },    // Resize helper functions    _resizeEvent: function () {        var width = this._originalDesignResolutionSize.width;        var height = this._originalDesignResolutionSize.height;        if (this._resizeCallback) {            this._initFrameSize();            this._resizeCallback.call();        }        if (width > 0)            this.setDesignResolutionSize(width, height, this._resolutionPolicy);    },    resizeWithBrowserSize: function (enabled) {        var adjustSize;        if (enabled) {            //enable            if (!this.__resizeWithBrowserSize) {                this.__resizeWithBrowserSize = true;                adjustSize = this._resizeEvent.bind(this);                window.addEventListener('resize', adjustSize, false);            }        } else {            //disable            if (this.__resizeWithBrowserSize) {                this.__resizeWithBrowserSize = true;                adjustSize = this._resizeEvent.bind(this);                window.removeEventListener('resize', adjustSize, false);            }        }    },    setResizeCallback: function (callback) {        if (typeof callback == "function" || callback == null) {            this._resizeCallback = callback;        }    },    _initFrameSize: function () {        var locFrameSize = this._frameSize;        locFrameSize.width = this._frame.clientWidth;        locFrameSize.height = this._frame.clientHeight;    },    // hack    _adjustSizeKeepCanvasSize: function (width, height) {        var designWidth = this._originalDesignResolutionSize.width;        var designHeight = this._originalDesignResolutionSize.height;        if (designWidth > 0)            this.setDesignResolutionSize(designWidth, designHeight, this._resolutionPolicy);    },    _setViewPortMeta: function (width, height) {        if (this._isAdjustViewPort) {	        var viewportMetas = {"user-scalable": "no", "maximum-scale": "1.0", "initial-scale": "1.0"}, elems = document.getElementsByName("viewport"), vp, content;            if (elems.length == 0) {                vp = document.createElement("meta");                vp.name = "viewport";                vp.content = "";                document.head.appendChild(vp);            }            else vp = elems[0];	        // For avoiding Android Firefox issue, to remove once firefox fixes its issue.	        if (cc.Browser.isMobile && cc.Browser.type == "firefox") {		        vp.content = "initial-scale:1";		        return;	        }            content = vp.content;            for (var key in viewportMetas) {                var pattern = new RegExp(key);                if (!pattern.test(content)) {                    content += (content == "" ? "" : ",") + key + "=" + viewportMetas[key];                }            }            /*            if(width<=320){                width = 321;            }            if(height)                content ="height="+height+","+content;            if(width)                content ="width="+width+","+content;            */            vp.content = content;        }    },    // RenderTexture hacker    _setScaleXYForRenderTexture: function () {        //hack for RenderTexture on canvas mode when adapting multiple resolution resources        var scaleFactor = cc.CONTENT_SCALE_FACTOR();        this._scaleX = scaleFactor;        this._scaleY = scaleFactor;    },    // Other helper functions    _resetScale: function () {        this._scaleX = this._originalScaleX;        this._scaleY = this._originalScaleY;    },    _getUnUsedIndex: function () {        var i;        var temp = this._indexBitsUsed;        for (i = 0; i < this._maxTouches; i++) {            if (!(temp & 0x00000001)) {                this._indexBitsUsed |= (1 << i);                return i;            }            temp >>= 1;        }        // all bits are used        return -1;    },    _removeUsedIndexBit: function (index) {        if (index < 0 || index >= this._maxTouches) {            return;        }        var temp = 1 << index;        temp = ~temp;        this._indexBitsUsed &= temp;    },    // Useless, just make sure the compatibility temporarily, should be removed    _adjustSizeToBrowser: function () {    },    /**     * init     */    initialize: function () {        this._initialized = true;    },    adjustViewPort: function (enabled) {        this._isAdjustViewPort = enabled;    },    /**     * Force destroying EGL view, subclass must implement this method.     */    end: function () {    },    /**     * Get whether render system is ready(no matter opengl or canvas),     * this name is for the compatibility with cocos2d-x, subclass must implement this method.     * @return {Boolean}     */    isOpenGLReady: function () {        return (this._hDC != null && this._hRC != null);    },    /*     * Set zoom factor for frame. This method is for debugging big resolution (e.g.new ipad) app on desktop.     * @param {Number} zoomFactor     */    setFrameZoomFactor: function (zoomFactor) {        this._frameZoomFactor = zoomFactor;        this.centerWindow();        cc.Director.getInstance().setProjection(cc.Director.getInstance().getProjection());    },    /**     * Exchanges the front and back buffers, subclass must implement this method.     */    swapBuffers: function () {    },    /**     * Open or close IME keyboard , subclass must implement this method.     */    setIMEKeyboardState: function (isOpen) {        if (isOpen) {            // [EAGLView sharedEGLView] becomeFirstResponder        } else {            //  [EAGLView sharedEGLView] resignFirstResponder        }    },    /**     * <p>     *   The resolution translate on EGLView     * </p>     * @param {Number} offsetLeft     * @param {Number} offsetTop     */    setContentTranslateLeftTop: function (offsetLeft, offsetTop) {        this._contentTranslateLeftTop = {left: offsetLeft, top: offsetTop};    },    /**     * <p>     *   get the resolution translate on EGLView     * </p>     * @return {cc.Size|Object}     */    getContentTranslateLeftTop: function () {        return this._contentTranslateLeftTop;    },    /**     * Get the frame size of EGL view.     * In general, it returns the screen size since the EGL view is a fullscreen view.     * @return {cc.Size}     */    getFrameSize: function () {        return cc.size(this._frameSize.width, this._frameSize.height);    },    /**     * Set the frame size of EGL view.     * @param {Number} width     * @param {Number} height     */    setFrameSize: function (width, height) {        this._frameSize.width = width;        this._frameSize.height = height;        this._frame.style.width = width + "px";        this._frame.style.height = height + "px";        //this.centerWindow();        this._resizeEvent();        cc.Director.getInstance().setProjection(cc.Director.getInstance().getProjection());    },    centerWindow: function () {    },    setAccelerometerKeyHook: function (accelerometerKeyHook) {        this._accelerometerKeyHook = accelerometerKeyHook;    },    /**     * Get the visible area size of opengl viewport.     * @return {cc.Size}     */    getVisibleSize: function () {        return this._visibleRect._size;    },    /**     * Get the visible origin povar of opengl viewport.     * @return {cc.Point}     */    getVisibleOrigin: function () {        return this._visibleRect._origin;    },    canSetContentScaleFactor: function () {        return true;    },    /**     * Get the current resolution policy     * @return {cc.ResolutionPolicy}     */    getResolutionPolicy: function () {        return this._resolutionPolicy;    },    /**     * Set the current resolution policy     * @param {cc.ResolutionPolicy|Number} resolutionPolicy     */    setResolutionPolicy: function (resolutionPolicy) {        if (resolutionPolicy instanceof cc.ResolutionPolicy) {            this._resolutionPolicy = resolutionPolicy;        }        // Ensure compatibility with JSB        else {            switch (resolutionPolicy) {                case cc.RESOLUTION_POLICY.EXACT_FIT:                    this._resolutionPolicy = this._rpExactFit;                    break;                case cc.RESOLUTION_POLICY.SHOW_ALL:                    this._resolutionPolicy = this._rpShowAll;                    break;                case cc.RESOLUTION_POLICY.NO_BORDER:                    this._resolutionPolicy = this._rpNoBorder;                    break;                case cc.RESOLUTION_POLICY.FIXED_HEIGHT:                    this._resolutionPolicy = this._rpFixedHeight;                    break;                case cc.RESOLUTION_POLICY.FIXED_WIDTH:                    this._resolutionPolicy = this._rpFixedWidth;                    break;            }        }    },    /**     * Set the design resolution size.     * @param {Number} width Design resolution width.     * @param {Number} height Design resolution height.     * @param {cc.ResolutionPolicy|Number} resolutionPolicy The resolution policy desired, you may choose:     * [1] ResolutionExactFit       Fill screen by stretch-to-fit: if the design resolution ratio of width to height is different from the screen resolution ratio, your game view will be stretched.     * [2] ResolutionNoBorder       Full screen without black border: if the design resolution ratio of width to height is different from the screen resolution ratio, two areas of your game view will be cut.     * [3] ResolutionShowAll        Full screen with black border: if the design resolution ratio of width to height is different from the screen resolution ratio, two black borders will be shown.     * [4] ResolutionFixedHeight    Scale the content's height to screen's height and proportionally scale its width     * [5] ResolutionFixedWidth     Scale the content's width to screen's width and proportionally scale its height     * [cc.ResolutionPolicy]        Custom resolution policy, constructed by cc.ResolutionPolicy     */    setDesignResolutionSize: function (width, height, resolutionPolicy) {        // Defensive code        if (isNaN(width) || width == 0 || isNaN(height) || height == 0) {            cc.log("Resolution not valid");            return;        }        this.setResolutionPolicy(resolutionPolicy);        var policy = this._resolutionPolicy;        if (policy)            policy.preApply(this);        else {            cc.log("should set resolutionPolicy");            return;        }        // Reinit frame size        var frameW = this._frameSize.width, frameH = this._frameSize.height;        if (cc.Browser.isMobile)            this._setViewPortMeta(this._frameSize.width, this._frameSize.height);        this._initFrameSize();        // No change        if (resolutionPolicy == this._resolutionPolicy            && width == this._originalDesignResolutionSize.width && height == this._originalDesignResolutionSize.height            && frameW == this._frameSize.width && frameH == this._frameSize.height)            return;        this._designResolutionSize = cc.size(width, height);        this._originalDesignResolutionSize = cc.size(width, height);        var result = policy.apply(this, this._designResolutionSize);        if (result.scale && result.scale.length == 2) {            this._scaleX = result.scale[0];            this._scaleY = result.scale[1];        }        if (result.viewport instanceof cc.Rect) {            var vp = this._viewPortRect = result.viewport, visible = this._visibleRect;            visible._size.width = cc.canvas.width / this._scaleX;            visible._size.height = cc.canvas.height / this._scaleY;            visible._origin.x = -vp.x / this._scaleX;            visible._origin.y = -vp.y / this._scaleY;        }        // reset director's member variables to fit visible rect        var director = cc.Director.getInstance();        director._winSizeInPoints = this.getDesignResolutionSize();        policy.postApply(this);        if (cc.renderContextType == cc.WEBGL) {            // reset director's member variables to fit visible rect            director._createStatsLabel();            director.setGLDefaultValues();        }        this._originalScaleX = this._scaleX;        this._originalScaleY = this._scaleY;        // For editbox        if (cc.DOM) {            cc.DOM._resetEGLViewDiv();        }        cc.VisibleRect.init(this.getVisibleSize());    },    /**     * Get design resolution size.     * Default resolution size is the same as 'getFrameSize'.     * @return {cc.Size}     */    getDesignResolutionSize: function () {        return cc.size(this._designResolutionSize.width, this._designResolutionSize.height);    },    /**     * set touch delegate     * @param {cc.TouchDispatcher} delegate     */    setTouchDelegate: function (delegate) {        this._delegate = delegate;    },    /**     * Set opengl view port rectangle with points.     * @param {Number} x     * @param {Number} y     * @param {Number} w width     * @param {Number} h height     */    setViewPortInPoints: function (x, y, w, h) {        var locFrameZoomFactor = this._frameZoomFactor, locScaleX = this._scaleX, locScaleY = this._scaleY;        cc.renderContext.viewport((x * locScaleX * locFrameZoomFactor + this._viewPortRect.x * locFrameZoomFactor),            (y * locScaleY * locFrameZoomFactor + this._viewPortRect.y * locFrameZoomFactor),            (w * locScaleX * locFrameZoomFactor),            (h * locScaleY * locFrameZoomFactor));    },    /**     * Set Scissor rectangle with points.     * @param {Number} x     * @param {Number} y     * @param {Number} w     * @param {Number} h     */    setScissorInPoints: function (x, y, w, h) {        var locFrameZoomFactor = this._frameZoomFactor, locScaleX = this._scaleX, locScaleY = this._scaleY;        cc.renderContext.scissor((x * locScaleX * locFrameZoomFactor + this._viewPortRect.x * locFrameZoomFactor),            (y * locScaleY * locFrameZoomFactor + this._viewPortRect.y * locFrameZoomFactor),            (w * locScaleX * locFrameZoomFactor),            (h * locScaleY * locFrameZoomFactor));    },    /**     * Get whether GL_SCISSOR_TEST is enable     */    isScissorEnabled: function () {        var gl = cc.renderContext;        return gl.isEnabled(gl.SCISSOR_TEST);    },    /**     * Get the current scissor rectangle     * @return {cc.Rect}     */    getScissorRect: function () {        var gl = cc.renderContext, scaleX = this._scaleX, scaleY = this._scaleY;        var boxArr = gl.getParameter(gl.SCISSOR_BOX);        return cc.rect((boxArr[0] - this._viewPortRect.x) / scaleX, (boxArr[1] - this._viewPortRect.y) / scaleY,            boxArr[2] / scaleX, boxArr[3] / scaleY);    },    /**     * @param {String} viewName     */    setViewName: function (viewName) {        if (viewName != null && viewName.length > 0) {            this._viewName = viewName;        }    },    /**     * get view name     * @return {String}     */    getViewName: function () {        return this._viewName;    },    /**     * Get the opengl view port rectangle.     */    getViewPortRect: function () {        return this._viewPortRect;    },    /**     * Get scale factor of the horizontal direction.     */    getScaleX: function () {        return this._scaleX;    },    /**     * Get scale factor of the vertical direction.     */    getScaleY: function () {        return this._scaleY;    },    /**     * Get device pixel ratio for retina display.     */    getDevicePixelRatio: function() {        return this._devicePixelRatio;    },    /**     * Get the real location in view     */    convertToLocationInView: function (tx, ty, relatedPos) {        return {x: this._devicePixelRatio * (tx - relatedPos.left), y: this._devicePixelRatio * (relatedPos.top + relatedPos.height - ty)};    },    /**     * Touch events are handled by default; if you want to customize your handlers, please override these functions:     * @param {Number} num     * @param {Array} ids     * @param {Array} xs     * @param {Array} ys     */    handleTouchesBegin: function (num, ids, xs, ys) {        var arr = [], locViewPortRect = this._viewPortRect, locScaleX = this._scaleX, locScaleY = this._scaleY;        for (var i = 0; i < num; ++i) {            var id = ids[i];            var x = xs[i];            var y = ys[i];            var index = cc.TouchesIntergerDict[id];            var unusedIndex = 0;            // it is a new touch            if (index == null) {                unusedIndex = this._getUnUsedIndex();                // The touches is more than MAX_TOUCHES ?                if (unusedIndex == -1) {                    cc.log("The touches is more than MAX_TOUCHES, nUnusedIndex = " + unusedIndex);                    continue;                }                var touch = cc.Touches[unusedIndex] = new cc.Touch();                touch.setTouchInfo(unusedIndex, (x - locViewPortRect.x) / locScaleX,                    (y - locViewPortRect.y) / locScaleY);                cc.TouchesIntergerDict[id] = 0 | unusedIndex;                arr.push(touch);            }        }        if (arr.length !== 0)            this._delegate.touchesBegan(arr, null);    },    /**     * @param {Number} num     * @param {Number} ids     * @param {Number} xs     * @param {Number} ys     */    handleTouchesMove: function (num, ids, xs, ys) {        var arr = [];        var locScaleX = this._scaleX, locScaleY = this._scaleY, locViewPortX = this._viewPortRect.x, locViewPortY = this._viewPortRect.y;        for (var i = 0; i < num; ++i) {            var id = ids[i];            var x = xs[i];            var y = ys[i];            var index = cc.TouchesIntergerDict[id];            if (index == null) {                //cc.log("if the index doesn't exist, it is an error");                continue;            }            var touch = cc.Touches[index];            if (touch) {                touch.setTouchInfo(index, (x - locViewPortX) / locScaleX,                    (y - locViewPortY) / locScaleY);                arr.push(touch);            }            else {                // It is error, should return.                //cc.log("Moving touches with id: " + id + " error");                return;            }        }        if (arr.length == 0) {            //cc.log("touchesMoved: count = 0");            return;        }        this._delegate.touchesMoved(arr, null);    },    /**     * @param {Number} num     * @param {Number} ids     * @param {Number} xs     * @param {Number} ys     */    handleTouchesEnd: function (num, ids, xs, ys) {        var arr = [];        this.getSetOfTouchesEndOrCancel(arr, num, ids, xs, ys);        this._delegate.touchesEnded(arr, null);    },    /**     * @param {Number} num     * @param {Number} ids     * @param {Number} xs     * @param {Number} ys     */    handleTouchesCancel: function (num, ids, xs, ys) {        var arr = [];        this.getSetOfTouchesEndOrCancel(arr, num, ids, xs, ys);        this._delegate.touchesCancelled(arr, null);    },    /**     * @param {Array} arr     * @param {Number} num     * @param {Number} ids     * @param {Number} xs     * @param {Number} ys     */    getSetOfTouchesEndOrCancel: function (arr, num, ids, xs, ys) {        var locScaleX = this._scaleX, locScaleY = this._scaleY, locViewPortRect = this._viewPortRect;        for (var i = 0; i < num; ++i) {            var id = ids[i];            var x = xs[i];            var y = ys[i];            var index = cc.TouchesIntergerDict[id];            if (index == null) {                //cc.log("if the index doesn't exist, it is an error");                continue;            }            /* Add to the set to send to the director */            var touch = cc.Touches[index];            if (touch) {                //cc.log("Ending touches with id: " + id + ", x=" + x + ", y=" + y);                touch.setTouchInfo(index, (x - locViewPortRect.x) / locScaleX,                    (y - locViewPortRect.y) / locScaleY);                arr.push(touch);                // release the object                cc.Touches[index] = null;                this._removeUsedIndexBit(index);                delete cc.TouchesIntergerDict[id];            } else {                //cc.log("Ending touches with id: " + id + " error");                return;            }        }    },    // Pass the touches to the superview    touchesBegan: function (touches, event) {        var ids = [];        var xs = [];        var ys = [];        var i = 0;        var touch;        for (var j = 0; j < touches.length; j++) {            touch = touches[j];            ids[i] = touch.getId() || j;            xs[i] = touch.getLocation().x;            ys[i] = touch.getLocation().y;            ++i;        }        this.handleTouchesBegin(i, ids, xs, ys);    },    touchesMoved: function (touches, event) {        var ids = [];        var xs = [];        var ys = [];        var i = 0;        var touch;        for (var j = 0; j < touches.length; j++) {            touch = touches[j];            ids[i] = touch.getId() || j;            xs[i] = touch.getLocation().x;            ys[i] = touch.getLocation().y;            ++i;        }        this.handleTouchesMove(i, ids, xs, ys);    },    touchesEnded: function (touches, event) {        var ids = [];        var xs = [];        var ys = [];        var i = 0;        var touch;        for (var j = 0; j < touches.length; j++) {            touch = touches[j];            ids[i] = touch.getId() || j;            xs[i] = touch.getLocation().x;            ys[i] = touch.getLocation().y;            ++i;        }        this.handleTouchesEnd(i, ids, xs, ys);    },    touchesCancelled: function (touches, event) {        var ids = [];        var xs = [];        var ys = [];        var i = 0;        var touch;        for (var j = 0; j < touches.length; j++) {            touch = touches[j];            ids[i] = touch.getId() || j;            xs[i] = touch.getLocation().x;            ys[i] = touch.getLocation().y;            ++i;        }        this.handleTouchesCancel(i, ids, xs, ys);    }});cc.EGLView.getInstance = function () {    if (!this._instance) {	    // First init director	    cc.Director.getInstance();        this._instance = this._instance || new cc.EGLView();        this._instance.initialize();    }    return this._instance;};/** * <p>cc.ContainerStrategy class is the root strategy class of container's scale strategy, * it controls the behavior of how to scale the cc.container and cc.canvas object</p> * * @class * @extends cc.Class */cc.ContainerStrategy = cc.Class.extend({    // Adjust canvas's size for retina display    _adjustRetina: false,    /**     * Manipulation before appling the strategy     * @param {cc.EGLView} The target view     */    preApply: function (view) {	    if(sys.os == "iOS" || sys.os == "OS X")		    this._adjustRetina = true;    },    /**     * Function to apply this strategy     * @param {cc.EGLView} view     * @param {cc.Size} designedResolution     */    apply: function (view, designedResolution) {    },    /**     * Manipulation after appling the strategy     * @param {cc.EGLView} The target view     */    postApply: function (view) {    },    _setupContainer: function (view, w, h) {        var frame = view._frame;        if (cc.Browser.isMobile && frame == document.documentElement) {            // Automatically full screen when user touches on mobile version            cc.Screen.getInstance().autoFullScreen(frame);        }        var locCanvasElement = cc.canvas, locContainer = cc.container;        // Setup container        locContainer.style.width = locCanvasElement.style.width = w + "px";        locContainer.style.height = locCanvasElement.style.height = h + "px";        // Setup pixel ratio for retina display        var devicePixelRatio = view._devicePixelRatio = 1;        if (this._adjustRetina)            devicePixelRatio = view._devicePixelRatio = window.devicePixelRatio || 1;        // Setup canvas        locCanvasElement.width = w * devicePixelRatio;        locCanvasElement.height = h * devicePixelRatio;        var body = document.body, style;        if (body && (style = body.style)) {            style.paddingTop = style.paddingTop || "0px";            style.paddingRight = style.paddingRight || "0px";            style.paddingBottom = style.paddingBottom || "0px";            style.paddingLeft = style.paddingLeft || "0px";            style.borderTop = style.borderTop || "0px";            style.borderRight = style.borderRight || "0px";            style.borderBottom = style.borderBottom || "0px";            style.borderLeft = style.borderLeft || "0px";            style.marginTop = style.marginTop || "0px";            style.marginRight = style.marginRight || "0px";            style.marginBottom = style.marginBottom || "0px";            style.marginLeft = style.marginLeft || "0px";        }    },    _fixContainer: function () {        // Add container to document body        document.body.insertBefore(cc.container, document.body.firstChild);        // Set body's width height to window's size, and forbid overflow, so that game will be centered        var bs = document.body.style;        bs.width = window.innerWidth + "px";        bs.height = window.innerHeight + "px";        bs.overflow = "hidden";        // Body size solution doesn't work on all mobile browser so this is the aleternative: fixed container        var contStyle = cc.container.style;        contStyle.position = "fixed";        contStyle.left = contStyle.top = "0px";        // Reposition body        document.body.scrollTop = 0;    }});/** * <p>cc.ContentStrategy class is the root strategy class of content's scale strategy, * it controls the behavior of how to scale the scene and setup the viewport for the game</p> * * @class * @extends cc.Class */cc.ContentStrategy = cc.Class.extend({    _result: {        scale: [1, 1],        viewport: null    },    _buildResult: function (containerW, containerH, contentW, contentH, scaleX, scaleY) {	    // Makes content fit better the canvas	    Math.abs(containerW - contentW) < 2 && (contentW = containerW);	    Math.abs(containerH - contentH) < 2 && (contentH = containerH);        var viewport = cc.rect(Math.round((containerW - contentW) / 2),                               Math.round((containerH - contentH) / 2),                               contentW, contentH);        // Translate the content        if (cc.renderContextType == cc.CANVAS)            cc.renderContext.translate(viewport.x, viewport.y + contentH);        this._result.scale = [scaleX, scaleY];        this._result.viewport = viewport;        return this._result;    },    /**     * Manipulation before appling the strategy     * @param {cc.EGLView} The target view     */    preApply: function (view) {    },    /**     * Function to apply this strategy     * The return value is {scale: [scaleX, scaleY], viewport: {cc.Rect}},     * The target view can then apply these value to itself, it's prefered not to modify directly its private variables     * @param {cc.EGLView} view     * @param {cc.Size} designedResolution     * @return {object} scaleAndViewportRect     */    apply: function (view, designedResolution) {        return {"scale": [1, 1]};    },    /**     * Manipulation after appling the strategy     * @param {cc.EGLView} The target view     */    postApply: function (view) {    }});(function () {// Container scale strategys    var EqualToFrame = cc.ContainerStrategy.extend({        apply: function (view) {            this._setupContainer(view, view._frameSize.width, view._frameSize.height);        }    });    var ProportionalToFrame = cc.ContainerStrategy.extend({        apply: function (view, designedResolution) {            var frameW = view._frameSize.width, frameH = view._frameSize.height, containerStyle = cc.container.style,                designW = designedResolution.width, designH = designedResolution.height,                scaleX = frameW / designW, scaleY = frameH / designH,                containerW, containerH;            scaleX < scaleY ? (containerW = frameW, containerH = designH * scaleX) : (containerW = designW * scaleY, containerH = frameH);            // Adjust container size with integer value            var offx = Math.round((frameW - containerW) / 2);            var offy = Math.round((frameH - containerH) / 2);            containerW = frameW - 2 * offx;            containerH = frameH - 2 * offy;            this._setupContainer(view, containerW, containerH);            // Setup container's margin            containerStyle.marginLeft = offx + "px";            containerStyle.marginRight = offx + "px";            containerStyle.marginTop = offy + "px";            containerStyle.marginBottom = offy + "px";        }    });    var EqualToWindow = EqualToFrame.extend({        preApply: function (view) {	        this._super(view);            view._frame = document.documentElement;        },        apply: function (view) {            this._super(view);            this._fixContainer();        }    });    var ProportionalToWindow = ProportionalToFrame.extend({        preApply: function (view) {	        this._super(view);            view._frame = document.documentElement;        },        apply: function (view, designedResolution) {            this._super(view, designedResolution);            this._fixContainer();        }    });    var OriginalContainer = cc.ContainerStrategy.extend({        apply: function (view) {            this._setupContainer(view, cc.canvas.width, cc.canvas.height);        }    });// #NOT STABLE on Android# Alias: Strategy that makes the container's size equals to the window's size//    cc.ContainerStrategy.EQUAL_TO_WINDOW = new EqualToWindow();// #NOT STABLE on Android# Alias: Strategy that scale proportionally the container's size to window's size//    cc.ContainerStrategy.PROPORTION_TO_WINDOW = new ProportionalToWindow();// Alias: Strategy that makes the container's size equals to the frame's size    cc.ContainerStrategy.EQUAL_TO_FRAME = new EqualToFrame();// Alias: Strategy that scale proportionally the container's size to frame's size    cc.ContainerStrategy.PROPORTION_TO_FRAME = new ProportionalToFrame();// Alias: Strategy that keeps the original container's size    cc.ContainerStrategy.ORIGINAL_CONTAINER = new OriginalContainer();// Content scale strategys    var ExactFit = cc.ContentStrategy.extend({        apply: function (view, designedResolution) {            var containerW = cc.canvas.width, containerH = cc.canvas.height,                scaleX = containerW / designedResolution.width, scaleY = containerH / designedResolution.height;            return this._buildResult(containerW, containerH, containerW, containerH, scaleX, scaleY);        }    });    var ShowAll = cc.ContentStrategy.extend({        apply: function (view, designedResolution) {            var containerW = cc.canvas.width, containerH = cc.canvas.height,                designW = designedResolution.width, designH = designedResolution.height,                scaleX = containerW / designW, scaleY = containerH / designH, scale = 0,                contentW, contentH;	        scaleX < scaleY ? (scale = scaleX, contentW = containerW, contentH = designH * scale)                : (scale = scaleY, contentW = designW * scale, contentH = containerH);            return this._buildResult(containerW, containerH, contentW, contentH, scale, scale);        }    });    var NoBorder = cc.ContentStrategy.extend({        apply: function (view, designedResolution) {            var containerW = cc.canvas.width, containerH = cc.canvas.height,                designW = designedResolution.width, designH = designedResolution.height,                scaleX = containerW / designW, scaleY = containerH / designH, scale,                contentW, contentH;            scaleX < scaleY ? (scale = scaleY, contentW = designW * scale, contentH = containerH)                : (scale = scaleX, contentW = containerW, contentH = designH * scale);            return this._buildResult(containerW, containerH, contentW, contentH, scale, scale);        }    });    var FixedHeight = cc.ContentStrategy.extend({        apply: function (view, designedResolution) {            var containerW = cc.canvas.width, containerH = cc.canvas.height,                designH = designedResolution.height, scale = containerH / designH,                contentW = containerW, contentH = containerH;            return this._buildResult(containerW, containerH, contentW, contentH, scale, scale);        },        postApply: function (view) {            cc.Director.getInstance()._winSizeInPoints = view.getVisibleSize();        }    });    var FixedWidth = cc.ContentStrategy.extend({        apply: function (view, designedResolution) {            var containerW = cc.canvas.width, containerH = cc.canvas.height,                designW = designedResolution.width, scale = containerW / designW,                contentW = containerW, contentH = containerH;            return this._buildResult(containerW, containerH, contentW, contentH, scale, scale);        },        postApply: function (view) {            cc.Director.getInstance()._winSizeInPoints = view.getVisibleSize();        }    });// Alias: Strategy to scale the content's size to container's size, non proportional    cc.ContentStrategy.EXACT_FIT = new ExactFit();// Alias: Strategy to scale the content's size proportionally to maximum size and keeps the whole content area to be visible    cc.ContentStrategy.SHOW_ALL = new ShowAll();// Alias: Strategy to scale the content's size proportionally to fill the whole container area    cc.ContentStrategy.NO_BORDER = new NoBorder();// Alias: Strategy to scale the content's height to container's height and proportionally scale its width    cc.ContentStrategy.FIXED_HEIGHT = new FixedHeight();// Alias: Strategy to scale the content's width to container's width and proportionally scale its height    cc.ContentStrategy.FIXED_WIDTH = new FixedWidth();})();/** * <p>cc.ResolutionPolicy class is the root strategy class of scale strategy, * its main task is to maintain the compatibility with Cocos2d-x</p> * * @class * @extends cc.Class */cc.ResolutionPolicy = cc.Class.extend({    _containerStrategy: null,    _contentStrategy: null,    ctor: function (containerStg, contentStg) {        this.setContainerStrategy(containerStg);        this.setContentStrategy(contentStg);    },    /**     * Manipulation before appling the resolution policy     * @param {cc.EGLView} The target view     */    preApply: function (view) {        this._containerStrategy.preApply(view);        this._contentStrategy.preApply(view);    },    /**     * Function to apply this resolution policy     * The return value is {scale: [scaleX, scaleY], viewport: {cc.Rect}},     * The target view can then apply these value to itself, it's prefered not to modify directly its private variables     * @param {cc.EGLView} The target view     * @param {cc.Size} The user defined design resolution     * @return {object} An object contains the scale X/Y values and the viewport rect     */    apply: function (view, designedResolution) {        this._containerStrategy.apply(view, designedResolution);        return this._contentStrategy.apply(view, designedResolution);    },    /**     * Manipulation after appling the strategy     * @param {cc.EGLView} The target view     */    postApply: function (view) {        this._containerStrategy.postApply(view);        this._contentStrategy.postApply(view);    },    /**     * Setup the container's scale strategy     * @param {cc.ContainerStrategy} containerStg     */    setContainerStrategy: function (containerStg) {        if (containerStg instanceof cc.ContainerStrategy)            this._containerStrategy = containerStg;    },    /**     * Setup the content's scale strategy     * @param {cc.ContentStrategy} contentStg     */    setContentStrategy: function (contentStg) {        if (contentStg instanceof cc.ContentStrategy)            this._contentStrategy = contentStg;    }});
 |