CocosCreator 源码.CCActionManager详解

require('../core/platform/CCClass');

var js = require('../core/platform/js');

/* 动作管理类 */

/*

* @class HashElement

* @constructor--构造函数

* @private--私有

*/

var HashElement = function () {

    this.actions = [];//存储具体的action

    this.target = null; //ccobject

    this.actionIndex = 0;

    this.currentAction = null; //CCAction

    this.paused = false;

    this.lock = false;

};

/**

* !#en

* cc.ActionManager is a class that can manage actions.<br/>

* Normally you won't need to use this class directly. 99% of the cases you will use the CCNode interface,

* which uses this class's singleton object.

* But there are some cases where you might need to use this class. <br/>

* Examples:<br/>

* - When you want to run an action where the target is different from a CCNode.<br/>

* - When you want to pause / resume the actions<br/>

* !#zh

* cc.ActionManager 是可以管理动作的单例类。<br/>

* 通常你并不需要直接使用这个类,99%的情况您将使用 CCNode 的接口。<br/>

* 但也有一些情况下,您可能需要使用这个类。 <br/>

* 例如:

*  - 当你想要运行一个动作,但目标不是 CCNode 类型时。 <br/>

*  - 当你想要暂停/恢复动作时。 <br/>

* @class ActionManager

* @example {@link cocos2d/core/CCActionManager/ActionManager.js}

*/

cc.ActionManager = function () {

    /* object 容器,存储了所有的动作node 的 uuid,*/

    /* 哈希map-object 存储node */

    this._hashTargets = js.createMap(true);

    /* 数组存储node */

    this._arrayTargets = [];

    /* 所有的存储的arr中的node,都在时时update,update时时运行的node,遍历的其中一个 */

    this._currentTarget = null;

    cc.director._scheduler && cc.director._scheduler.enableForTarget(this);

};

/* 设置cc.ActionManager的原型对象 */

cc.ActionManager.prototype = {

    constructor: cc.ActionManager,

    /* 元素池,数组类型 */

    _elementPool: [],

    /* 根据target查找element */

    _searchElementByTarget: function (arr, target) {

        for (var k = 0; k < arr.length; k++) {

            if (target === arr[k].target)

                return arr[k];

        }

        return null;

    },

    /* 获取一个element info,设置target,设置pause属性 */

    _getElement: function (target, paused) {

        var element = this._elementPool.pop();

        /* 当element不存在的时候,new一个 */

        if (!element) {

            element = new HashElement();

        }

        element.target = target;

        /* !!强制转换成布尔类型 */

        element.paused = !!paused;

        /* 返回element */

        return element;

    },

    /* 把element元素直接放到容器,重置element属性,回收 */

    _putElement: function (element) {

        /* node的actions数组的长度为0 */

        element.actions.length = 0;

        /*  */

        element.actionIndex = 0;

        /* 元素的当前action为空 */

        element.currentAction = null;

        /* 设置暂停属性为false */

        element.paused = false;

        /* 设置target为空 */

        element.target = null;

        /* 设置lock属性为false */

        element.lock = false;

        /* 回收info,放到pool池 */

        this._elementPool.push(element);

    },

    /**

    * !#en

    * Adds an action with a target.<br/>

    * If the target is already present, then the action will be added to the existing target.

    * If the target is not present, a new instance of this target will be created either paused or not, and the action will be added to the newly created target.

    * When the target is paused, the queued actions won't be 'ticked'.

    * !#zh

    * 增加一个动作,同时还需要提供动作的目标对象,目标对象是否暂停作为参数。<br/>

    * 如果目标已存在,动作将会被直接添加到现有的节点中。<br/>

    * 如果目标不存在,将为这一目标创建一个新的实例,并将动作添加进去。<br/>

    * 当目标状态的 paused 为 true,动作将不会被执行

    *

    * @method addAction

    * @param {Action} action

    * @param {Node} target

    * @param {Boolean} paused

    */

    addAction: function (action, target, paused) {

        if (!action || !target) {

            cc.errorID(1000);

            return;

        }

        //check if the action target already exists

        /* 检查操作目标是否已存在 */

        /* * !#zh 主要用于编辑器的 uuid,在编辑器下可用于持久化存储,在项目构建之后将变成自增的 id。

        */

        var element = this._hashTargets[target._id];

        //if doesn't exists, create a hashelement and push in mpTargets

        /* 如果不存在,则创建一个 hashelement 并推入 mpTargets */

        if (!element) {

            element = this._getElement(target, paused);//获取一个element

            this._hashTargets[target._id] = element;//根据node uuid 存储info object存储

            this._arrayTargets.push(element);//array add info

        }

        else if (!element.actions) {//当element存在,并且element的actions-动作array为空

            element.actions = [];

        }

        element.actions.push(action);//info的action容器添加一个action

        action.startWithTarget(target);//action 设置target,

    },

    /**

    * !#en Removes all actions from all the targets.

    * !#zh 移除所有对象的所有动作。

    * @method removeAllActions

    */

    removeAllActions: function () {

        var locTargets = this._arrayTargets;

        /* 遍历数组-info */

        for (var i = 0; i < locTargets.length; i++) {

            var element = locTargets[i];

            if (element)

                this._putElement(element);//回收elementinfo

        }

        /* 清空数组 */

        this._arrayTargets.length = 0;

        //重新创建一个map 给hashTargets赋值

        this._hashTargets = js.createMap(true);

    },

    /**

    * !#en

    * Removes all actions from a certain target. <br/>

    * All the actions that belongs to the target will be removed.

    * !#zh

    * 移除指定对象上的所有动作。<br/>

    * 属于该目标的所有的动作将被删除。

    * @method removeAllActionsFromTarget

    * @param {Node} target

    * @param {Boolean} forceDelete

    */

    removeAllActionsFromTarget: function (target, forceDelete) {

        // explicit null handling

        if (target == null)

            return;

        /* 从map中找到对应的info */

        var element = this._hashTargets[target._id];

        if (element) {

            /* 清空数组 */

            element.actions.length = 0;

            this._deleteHashElement(element);

        }

    },

    /**

    * !#en Removes an action given an action reference.

    * !#zh 移除指定的动作。

    * @method removeAction

    * @param {Action} action

    */

    removeAction: function (action) {

        // explicit null handling

        /* 空action处理 */

        if (!action) {

            return;

        }

        /* 获取原始目标节点。 */

        var target = action.getOriginalTarget();

        /* 从map中获取info--element */

        var element = this._hashTargets[target._id];

        /* info不存在则return */

        if (!element) {

            return;

        }

        /* 遍历action数组 找到对应的action */

        for (var i = 0; i < element.actions.length; i++) {

            /* 找到对应的action */

            if (element.actions[i] === action) {

                /* 删除action */

                element.actions.splice(i, 1);

                // update actionIndex in case we are in tick. looping over the actions

                /* 如果我们处于勾选状态,请更新actionIndex。循环操作 */

                if (element.actionIndex >= i)

                    element.actionIndex--;

                break;

            }

        }

    },

    /* 根据action的tag,移除某个node的action */

    _removeActionByTag(tag, element, target) {

        /* 遍历某个具体info下的所有action */

        for (var i = 0, l = element.actions.length; i < l; ++i) {

            var action = element.actions[i];

            /* 某个action的tag和要找的tag一致 */

            if (action && action.getTag() === tag) {

                /* 如果target和action的target属性不一致,则跳过 */

                if (target && action.getOriginalTarget() !== target) {

                    continue;

                }

                /* 从action数组中删除具体的action */

                this._removeActionAtIndex(i, element);

                break;

            }

        }

    },

    /**

    * !#en Removes an action given its tag and the target.

    * !#zh 删除指定对象下特定标签的一个动作,将删除首个匹配到的动作。

    * @method removeActionByTag

    * @param {Number} tag

    * @param {Node} [target]

    */

    removeActionByTag: function (tag, target) {

        /* tag不合法 */

        if (tag === cc.Action.TAG_INVALID)

            cc.logID(1002);

        /* 哈希map容器 */

        let hashTargets = this._hashTargets;

        if (target) {

            var element = hashTargets[target._id];

            /* info找到 */

            if (element) {

                this._removeActionByTag(tag, element, target);

            }

        }/* 假如target参数为空,则遍历所有匹配key 根据找到的info进行传参删除 */

        else {

            for (let name in hashTargets) {

                let element = hashTargets[name];

                this._removeActionByTag(tag, element);

            }

        }

    },

    /**

    * !#en Gets an action given its tag an a target.

    * !#zh 通过目标对象和标签获取一个动作。

    * @method getActionByTag

    * @param {Number} tag

    * @param {Node} target

    * @return {Action|Null}  return the Action with the given tag on success

    */

    getActionByTag: function (tag, target) {

        /* tag不合法报错 */

        if (tag === cc.Action.TAG_INVALID)

            cc.logID(1004);

        /* 根据targed的id获取到info */

        var element = this._hashTargets[target._id];

        if (element) {

            /* info的actions数组不为空,进行遍历actions,进行tag比较 */

            if (element.actions != null) {

                for (var i = 0; i < element.actions.length; ++i) {

                    var action = element.actions[i];

                    if (action && action.getTag() === tag)

                        return action;

                }

            }

            cc.logID(1005, tag);

        }

        return null;

    },

    /**

    * !#en

    * Returns the numbers of actions that are running in a certain target. <br/>

    * Composable actions are counted as 1 action. <br/>

    * Example: <br/>

    * - If you are running 1 Sequence of 7 actions, it will return 1. <br/>

    * - If you are running 7 Sequences of 2 actions, it will return 7.

    * !#zh

    * 返回指定对象下所有正在运行的动作数量。 <br/>

    * 组合动作被算作一个动作。<br/>

    * 例如:<br/>

    *  - 如果您正在运行 7 个动作组成的序列动作(Sequence),这个函数将返回 1。<br/>

    *  - 如果你正在运行 2 个序列动作(Sequence)和 5 个普通动作,这个函数将返回 7。<br/>

    *

    * @method getNumberOfRunningActionsInTarget

    * @param {Node} target

    * @return {Number}

    */

    /* 返回指定对象下所有正在运行的动作数量 */

    getNumberOfRunningActionsInTarget: function (target) {

        /* 找对应的info */

        var element = this._hashTargets[target._id];

        /* 返回数组的长度 */

        if (element)

            return (element.actions) ? element.actions.length : 0;

        return 0;

    },

    /**

    * !#en Pauses the target: all running actions and newly added actions will be paused.

    * !#zh 暂停指定对象:所有正在运行的动作和新添加的动作都将会暂停。

    * @method pauseTarget

    * @param {Node} target

    */

    /* 暂停某个对象的所有action */

    pauseTarget: function (target) {

        var element = this._hashTargets[target._id];

        if (element)

            element.paused = true;

    },

    /**

    * !#en Resumes the target. All queued actions will be resumed.

    * !#zh 让指定目标恢复运行。在执行序列中所有被暂停的动作将重新恢复运行。

    * @method resumeTarget

    * @param {Node} target

    */

    /* 回复action的运行 */

    resumeTarget: function (target) {

        /* 根据对象id获取info */

        var element = this._hashTargets[target._id];

        if (element)

            element.paused = false;

    },

    /**

    * !#en Pauses all running actions, returning a list of targets whose actions were paused.

    * !#zh 暂停所有正在运行的动作,返回一个包含了那些动作被暂停了的目标对象的列表。

    * @method pauseAllRunningActions

    * @return {Array}  a list of targets whose actions were paused.

    */

    /* 暂停所有正在运行的动作,返回一个包含了那些动作被暂停了的目标对象的列表。 */

    pauseAllRunningActions: function () {

        var idsWithActions = [];

        /* 存储info的数组 */

        var locTargets = this._arrayTargets;

        /* 遍历容器 */

        for (var i = 0; i < locTargets.length; i++) {

            var element = locTargets[i];

            /* info没pause的,设置为pause为true */

            if (element && !element.paused) {

                element.paused = true;

                /* 把对应都放到数组 */

                idsWithActions.push(element.target);

            }

        }

        /* 返回数组 */

        return idsWithActions;

    },

    /**

    * !#en Resume a set of targets (convenience function to reverse a pauseAllRunningActions or pauseTargets call).

    * !#zh 让一组指定对象恢复运行(用来逆转 pauseAllRunningActions 效果的便捷函数)。

    * @method resumeTargets

    * @param {Array} targetsToResume

    */

    resumeTargets: function (targetsToResume) {

        /* 参数必须是数组,不为空 */

        if (!targetsToResume)

            return;

        /* 遍历数组,获取所有具体对象 */

        for (var i = 0; i < targetsToResume.length; i++) {

            if (targetsToResume[i])

                this.resumeTarget(targetsToResume[i]);

        }

    },

    /**

    * !#en Pause a set of targets.

    * !#zh 暂停一组指定对象。

    * @method pauseTargets

    * @param {Array} targetsToPause

    */

    pauseTargets: function (targetsToPause) {

        if (!targetsToPause)

            return;

        /* 遍历存储对象的数组 */

        for (var i = 0; i < targetsToPause.length; i++) {

            if (targetsToPause[i])

                this.pauseTarget(targetsToPause[i]);

        }

    },

    /**

    * !#en

    * purges the shared action manager. It releases the retained instance. <br/>

    * because it uses this, so it can not be static.

    * !#zh

    * 清除共用的动作管理器。它释放了持有的实例。 <br/>

    * 因为它使用 this,因此它不能是静态的。

    * @method purgeSharedManager

    */

    purgeSharedManager: function () {

        /* 取消指定对象的 update 定时器。 */

        cc.director.getScheduler().unscheduleUpdate(this);

    },

    //protected

    /* 删除固定数组下标的info */

    _removeActionAtIndex: function (index, element) {

        /* 获取固定的action */

        var action = element.actions[index];

        /* 删除某个index的属性 */

        element.actions.splice(index, 1);

        // update actionIndex in case we are in tick. looping over the actions

        /* 如果我们处于勾选状态,请更新actionIndex。循环操作 */

        if (element.actionIndex >= index)

            element.actionIndex--;

            /* 当action数组为空的时候,删除这个info */

        if (element.actions.length === 0) {

            this._deleteHashElement(element);

        }

    },

    /* 删除map内存储的这个info */

    _deleteHashElement: function (element) {

        var ret = false;

        if (element && !element.lock) {

            if (this._hashTargets[element.target._id]) {

                /* 从object 删除info */

                delete this._hashTargets[element.target._id];

                var targets = this._arrayTargets;

                /* 遍历数组 */

                for (var i = 0, l = targets.length; i < l; i++) {

                    if (targets[i] === element) {

                        targets.splice(i, 1);//删除这个元素,从数组中

                        break;

                    }

                }

                /* 回收info */

                this._putElement(element);

                ret = true;

            }

        }

        return ret;

    },

    /**

    * !#en The ActionManager update。

    * !#zh ActionManager 主循环。

    * @method update

    * @param {Number} dt delta time in seconds

    */

    update: function (dt) {

        /* 声明locCurrTarget 临时变量存储的是info  locTargets是数组this._arrayTargets的重新赋值 */

        var locTargets = this._arrayTargets, locCurrTarget;

        for (var elt = 0; elt < locTargets.length; elt++) {

            this._currentTarget = locTargets[elt];

            /* 设置当前target info */

            locCurrTarget = this._currentTarget;

            /* action没有被暂停,并且存在actions */

            if (!locCurrTarget.paused && locCurrTarget.actions) {

                locCurrTarget.lock = true;

                // The 'actions' CCMutableArray may change while inside this loop.

                /* 在此循环内,“动作”CCMutableArray 可能会发生变化。 */

                for (locCurrTarget.actionIndex = 0; locCurrTarget.actionIndex < locCurrTarget.actions.length; locCurrTarget.actionIndex++) {

                    locCurrTarget.currentAction = locCurrTarget.actions[locCurrTarget.actionIndex];

                    if (!locCurrTarget.currentAction)

                        continue;

                    //use for speed

                    locCurrTarget.currentAction.step(dt * (locCurrTarget.currentAction._speedMethod ? locCurrTarget.currentAction._speed : 1));

                    if (locCurrTarget.currentAction && locCurrTarget.currentAction.isDone()) {

                        locCurrTarget.currentAction.stop();

                        var action = locCurrTarget.currentAction;

                        // Make currentAction nil to prevent removeAction from salvaging it.

                        /* 将 currentAction 设置为 nil 以防止 removeAction 抢救它。 */

                        locCurrTarget.currentAction = null;

                        /* 移除指定动作 */

                        this.removeAction(action);

                    }

                    locCurrTarget.currentAction = null;

                }

                locCurrTarget.lock = false;

            }

            // only delete currentTarget if no actions were scheduled during the cycle (issue #481)

            /* 仅当周期内未安排任何操作时才删除 currentTarget(问题 #481) */

            if (locCurrTarget.actions.length === 0) {//回收info

                this._deleteHashElement(locCurrTarget) && elt--;

            }

        }

    }

};

/* 测试环境,拓展属性isTargetPaused_TEST ,判断某个node的动画是否是paused */

if (CC_TEST) {

    cc.ActionManager.prototype.isTargetPaused_TEST = function (target) {

        /* 获取某个info */

        var element = this._hashTargets[target._id];

        return element.paused;

    };

}

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

推荐阅读更多精彩内容