/**************************************************************************** Copyright (c) 2013-2016 Chukong Technologies Inc. Copyright (c) 2017-2018 Xiamen Yaji Software Co., Ltd. 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. ****************************************************************************/ 'use strict'; import { Mat4, Vec2, Vec3, Quat, Trs } from './value-types'; import { approx } from './value-types/utils' const BaseNode = require('./utils/base-node'); const PrefabHelper = require('./utils/prefab-helper'); const nodeMemPool = require('./utils/trans-pool').NodeMemPool; const AffineTrans = require('./utils/affine-transform'); const eventManager = require('./event-manager'); const macro = require('./platform/CCMacro'); const js = require('./platform/js'); const Event = require('./event/event'); const EventTarget = require('./event/event-target'); const RenderFlow = require('./renderer/render-flow'); const Flags = cc.Object.Flags; const Destroying = Flags.Destroying; const ERR_INVALID_NUMBER = CC_EDITOR && 'The %s is invalid'; const ONE_DEGREE = Math.PI / 180; var ActionManagerExist = !!cc.ActionManager; var emptyFunc = function () {}; // getWorldPosition temp var var _gwpVec3 = new Vec3(); var _gwpQuat = new Quat(); // _invTransformPoint temp var var _tpVec3a = new Vec3(); var _tpVec3b = new Vec3(); var _tpQuata = new Quat(); var _tpQuatb = new Quat(); // setWorldPosition temp var var _swpVec3 = new Vec3(); // getWorldScale temp var var _gwsVec3 = new Vec3(); // setWorldScale temp var var _swsVec3 = new Vec3(); // getWorldRT temp var var _gwrtVec3a = new Vec3(); var _gwrtVec3b = new Vec3(); var _gwrtQuata = new Quat(); var _gwrtQuatb = new Quat(); // lookAt temp var var _laVec3 = new Vec3(); var _laQuat = new Quat(); //up、right、forward temp var var _urfVec3 = new Vec3(); var _urfQuat = new Quat(); // _hitTest temp var var _htVec3a = new Vec3(); var _htVec3b = new Vec3(); // getWorldRotation temp var var _gwrQuat = new Quat(); // setWorldRotation temp var var _swrQuat = new Quat(); var _quata = new Quat(); var _mat4_temp = cc.mat4(); var _vec3_temp = new Vec3(); const POSITION_ON = 1 << 0; const SCALE_ON = 1 << 1; const ROTATION_ON = 1 << 2; const SIZE_ON = 1 << 3; const ANCHOR_ON = 1 << 4; const COLOR_ON = 1 << 5; let _cachedPool = new js.Pool(); _cachedPool.get = function () { return this._get() || []; }; let BuiltinGroupIndex = cc.Enum({ DEBUG: 31 }); /** * !#en Node's local dirty properties flag * !#zh Node 的本地属性 dirty 状态位 * @enum Node._LocalDirtyFlag * @static * @private * @namespace Node */ var LocalDirtyFlag = cc.Enum({ /** * !#en Flag for position dirty * !#zh 位置 dirty 的标记位 * @property {Number} POSITION * @static */ POSITION: 1 << 0, /** * !#en Flag for scale dirty * !#zh 缩放 dirty 的标记位 * @property {Number} SCALE * @static */ SCALE: 1 << 1, /** * !#en Flag for rotation dirty * !#zh 旋转 dirty 的标记位 * @property {Number} ROTATION * @static */ ROTATION: 1 << 2, /** * !#en Flag for skew dirty * !#zh skew dirty 的标记位 * @property {Number} SKEW * @static */ SKEW: 1 << 3, /** * !#en Flag for rotation, scale or position dirty * !#zh 旋转,缩放,或位置 dirty 的标记位 * @property {Number} TRS * @static */ TRS: 1 << 0 | 1 << 1 | 1 << 2, /** * !#en Flag for rotation or scale dirty * !#zh 旋转或缩放 dirty 的标记位 * @property {Number} RS * @static */ RS: 1 << 1 | 1 << 2, /** * !#en Flag for rotation, scale, position, skew dirty * !#zh 旋转,缩放,位置,或斜角 dirty 的标记位 * @property {Number} TRS * @static */ TRSS: 1 << 0 | 1 << 1 | 1 << 2 | 1 << 3, /** * !#en Flag for physics position dirty * !#zh 物理位置 dirty 的标记位 * @property {Number} PHYSICS_POSITION * @static */ PHYSICS_POSITION: 1 << 4, /** * !#en Flag for physics scale dirty * !#zh 物理缩放 dirty 的标记位 * @property {Number} PHYSICS_SCALE * @static */ PHYSICS_SCALE: 1 << 5, /** * !#en Flag for physics rotation dirty * !#zh 物理旋转 dirty 的标记位 * @property {Number} PHYSICS_ROTATION * @static */ PHYSICS_ROTATION: 1 << 6, /** * !#en Flag for physics trs dirty * !#zh 物理位置旋转缩放 dirty 的标记位 * @property {Number} PHYSICS_TRS * @static */ PHYSICS_TRS: 1 << 4 | 1 << 5 | 1 << 6, /** * !#en Flag for physics rs dirty * !#zh 物理旋转缩放 dirty 的标记位 * @property {Number} PHYSICS_RS * @static */ PHYSICS_RS: 1 << 5 | 1 << 6, /** * !#en Flag for node and physics position dirty * !#zh 所有位置 dirty 的标记位 * @property {Number} ALL_POSITION * @static */ ALL_POSITION: 1 << 0 | 1 << 4, /** * !#en Flag for node and physics scale dirty * !#zh 所有缩放 dirty 的标记位 * @property {Number} ALL_SCALE * @static */ ALL_SCALE: 1 << 1 | 1 << 5, /** * !#en Flag for node and physics rotation dirty * !#zh 所有旋转 dirty 的标记位 * @property {Number} ALL_ROTATION * @static */ ALL_ROTATION: 1 << 2 | 1 << 6, /** * !#en Flag for node and physics trs dirty * !#zh 所有trs dirty 的标记位 * @property {Number} ALL_TRS * @static */ ALL_TRS: 1 << 0 | 1 << 1 | 1 << 2 | 1 << 4 | 1 << 5 | 1 << 6, /** * !#en Flag for all dirty properties * !#zh 覆盖所有 dirty 状态的标记位 * @property {Number} ALL * @static */ ALL: 0xffff, }); /** * !#en The event type supported by Node * !#zh Node 支持的事件类型 * @class Node.EventType * @static * @namespace Node */ // Why EventType defined as class, because the first parameter of Node.on method needs set as 'string' type. var EventType = cc.Enum({ /** * !#en The event type for touch start event, you can use its value directly: 'touchstart' * !#zh 当手指触摸到屏幕时。 * @property {String} TOUCH_START * @static */ TOUCH_START: 'touchstart', /** * !#en The event type for touch move event, you can use its value directly: 'touchmove' * !#zh 当手指在屏幕上移动时。 * @property {String} TOUCH_MOVE * @static */ TOUCH_MOVE: 'touchmove', /** * !#en The event type for touch end event, you can use its value directly: 'touchend' * !#zh 当手指在目标节点区域内离开屏幕时。 * @property {String} TOUCH_END * @static */ TOUCH_END: 'touchend', /** * !#en The event type for touch end event, you can use its value directly: 'touchcancel' * !#zh 当手指在目标节点区域外离开屏幕时。 * @property {String} TOUCH_CANCEL * @static */ TOUCH_CANCEL: 'touchcancel', /** * !#en The event type for mouse down events, you can use its value directly: 'mousedown' * !#zh 当鼠标按下时触发一次。 * @property {String} MOUSE_DOWN * @static */ MOUSE_DOWN: 'mousedown', /** * !#en The event type for mouse move events, you can use its value directly: 'mousemove' * !#zh 当鼠标在目标节点在目标节点区域中移动时,不论是否按下。 * @property {String} MOUSE_MOVE * @static */ MOUSE_MOVE: 'mousemove', /** * !#en The event type for mouse enter target events, you can use its value directly: 'mouseenter' * !#zh 当鼠标移入目标节点区域时,不论是否按下。 * @property {String} MOUSE_ENTER * @static */ MOUSE_ENTER: 'mouseenter', /** * !#en The event type for mouse leave target events, you can use its value directly: 'mouseleave' * !#zh 当鼠标移出目标节点区域时,不论是否按下。 * @property {String} MOUSE_LEAVE * @static */ MOUSE_LEAVE: 'mouseleave', /** * !#en The event type for mouse up events, you can use its value directly: 'mouseup' * !#zh 当鼠标从按下状态松开时触发一次。 * @property {String} MOUSE_UP * @static */ MOUSE_UP: 'mouseup', /** * !#en The event type for mouse wheel events, you can use its value directly: 'mousewheel' * !#zh 当鼠标滚轮滚动时。 * @property {String} MOUSE_WHEEL * @static */ MOUSE_WHEEL: 'mousewheel', /** * !#en The event type for position change events. * Performance note, this event will be triggered every time corresponding properties being changed, * if the event callback have heavy logic it may have great performance impact, try to avoid such scenario. * !#zh 当节点位置改变时触发的事件。 * 性能警告:这个事件会在每次对应的属性被修改时触发,如果事件回调损耗较高,有可能对性能有很大的负面影响,请尽量避免这种情况。 * @property {String} POSITION_CHANGED * @static */ POSITION_CHANGED: 'position-changed', /** * !#en The event type for rotation change events. * Performance note, this event will be triggered every time corresponding properties being changed, * if the event callback have heavy logic it may have great performance impact, try to avoid such scenario. * !#zh 当节点旋转改变时触发的事件。 * 性能警告:这个事件会在每次对应的属性被修改时触发,如果事件回调损耗较高,有可能对性能有很大的负面影响,请尽量避免这种情况。 * @property {String} ROTATION_CHANGED * @static */ ROTATION_CHANGED: 'rotation-changed', /** * !#en The event type for scale change events. * Performance note, this event will be triggered every time corresponding properties being changed, * if the event callback have heavy logic it may have great performance impact, try to avoid such scenario. * !#zh 当节点缩放改变时触发的事件。 * 性能警告:这个事件会在每次对应的属性被修改时触发,如果事件回调损耗较高,有可能对性能有很大的负面影响,请尽量避免这种情况。 * @property {String} SCALE_CHANGED * @static */ SCALE_CHANGED: 'scale-changed', /** * !#en The event type for size change events. * Performance note, this event will be triggered every time corresponding properties being changed, * if the event callback have heavy logic it may have great performance impact, try to avoid such scenario. * !#zh 当节点尺寸改变时触发的事件。 * 性能警告:这个事件会在每次对应的属性被修改时触发,如果事件回调损耗较高,有可能对性能有很大的负面影响,请尽量避免这种情况。 * @property {String} SIZE_CHANGED * @static */ SIZE_CHANGED: 'size-changed', /** * !#en The event type for anchor point change events. * Performance note, this event will be triggered every time corresponding properties being changed, * if the event callback have heavy logic it may have great performance impact, try to avoid such scenario. * !#zh 当节点锚点改变时触发的事件。 * 性能警告:这个事件会在每次对应的属性被修改时触发,如果事件回调损耗较高,有可能对性能有很大的负面影响,请尽量避免这种情况。 * @property {String} ANCHOR_CHANGED * @static */ ANCHOR_CHANGED: 'anchor-changed', /** * !#en The event type for color change events. * Performance note, this event will be triggered every time corresponding properties being changed, * if the event callback have heavy logic it may have great performance impact, try to avoid such scenario. * !#zh 当节点颜色改变时触发的事件。 * 性能警告:这个事件会在每次对应的属性被修改时触发,如果事件回调损耗较高,有可能对性能有很大的负面影响,请尽量避免这种情况。 * @property {String} COLOR_CHANGED * @static */ COLOR_CHANGED: 'color-changed', /** * !#en The event type for new child added events. * !#zh 当新的子节点被添加时触发的事件。 * @property {String} CHILD_ADDED * @static */ CHILD_ADDED: 'child-added', /** * !#en The event type for child removed events. * !#zh 当子节点被移除时触发的事件。 * @property {String} CHILD_REMOVED * @static */ CHILD_REMOVED: 'child-removed', /** * !#en The event type for children reorder events. * !#zh 当子节点顺序改变时触发的事件。 * @property {String} CHILD_REORDER * @static */ CHILD_REORDER: 'child-reorder', /** * !#en The event type for node group changed events. * !#zh 当节点归属群组发生变化时触发的事件。 * @property {String} GROUP_CHANGED * @static */ GROUP_CHANGED: 'group-changed', /** * !#en The event type for node's sibling order changed. * !#zh 当节点在兄弟节点中的顺序发生变化时触发的事件。 * @property {String} SIBLING_ORDER_CHANGED * @static */ SIBLING_ORDER_CHANGED: 'sibling-order-changed', }); var _touchEvents = [ EventType.TOUCH_START, EventType.TOUCH_MOVE, EventType.TOUCH_END, EventType.TOUCH_CANCEL, ]; var _mouseEvents = [ EventType.MOUSE_DOWN, EventType.MOUSE_ENTER, EventType.MOUSE_MOVE, EventType.MOUSE_LEAVE, EventType.MOUSE_UP, EventType.MOUSE_WHEEL, ]; var _skewNeedWarn = true; var _skewWarn = function (value, node) { if (value !== 0) { var nodePath = ""; if (CC_EDITOR) { var NodeUtils = Editor.require('scene://utils/node'); nodePath = `Node: ${NodeUtils.getNodePath(node)}.` } _skewNeedWarn && cc.warn("`cc.Node.skewX/Y` is deprecated since v2.2.1, please use 3D node instead.", nodePath); !CC_EDITOR && (_skewNeedWarn = false); } } var _currentHovered = null; var _touchStartHandler = function (touch, event) { var pos = touch.getLocation(); var node = this.owner; if (node._hitTest(pos, this)) { event.type = EventType.TOUCH_START; event.touch = touch; event.bubbles = true; node.dispatchEvent(event); return true; } return false; }; var _touchMoveHandler = function (touch, event) { var node = this.owner; event.type = EventType.TOUCH_MOVE; event.touch = touch; event.bubbles = true; node.dispatchEvent(event); }; var _touchEndHandler = function (touch, event) { var pos = touch.getLocation(); var node = this.owner; if (node._hitTest(pos, this)) { event.type = EventType.TOUCH_END; } else { event.type = EventType.TOUCH_CANCEL; } event.touch = touch; event.bubbles = true; node.dispatchEvent(event); }; var _touchCancelHandler = function (touch, event) { var pos = touch.getLocation(); var node = this.owner; event.type = EventType.TOUCH_CANCEL; event.touch = touch; event.bubbles = true; node.dispatchEvent(event); }; var _mouseDownHandler = function (event) { var pos = event.getLocation(); var node = this.owner; if (node._hitTest(pos, this)) { event.type = EventType.MOUSE_DOWN; event.bubbles = true; node.dispatchEvent(event); } }; var _mouseMoveHandler = function (event) { var pos = event.getLocation(); var node = this.owner; var hit = node._hitTest(pos, this); if (hit) { if (!this._previousIn) { // Fix issue when hover node switched, previous hovered node won't get MOUSE_LEAVE notification if (_currentHovered && _currentHovered._mouseListener) { event.type = EventType.MOUSE_LEAVE; _currentHovered.dispatchEvent(event); _currentHovered._mouseListener._previousIn = false; } _currentHovered = this.owner; event.type = EventType.MOUSE_ENTER; node.dispatchEvent(event); this._previousIn = true; } event.type = EventType.MOUSE_MOVE; event.bubbles = true; node.dispatchEvent(event); } else if (this._previousIn) { event.type = EventType.MOUSE_LEAVE; node.dispatchEvent(event); this._previousIn = false; _currentHovered = null; } else { // continue dispatching return; } // Event processed, cleanup event.stopPropagation(); }; var _mouseUpHandler = function (event) { var pos = event.getLocation(); var node = this.owner; if (node._hitTest(pos, this)) { event.type = EventType.MOUSE_UP; event.bubbles = true; node.dispatchEvent(event); event.stopPropagation(); } }; var _mouseWheelHandler = function (event) { var pos = event.getLocation(); var node = this.owner; if (node._hitTest(pos, this)) { event.type = EventType.MOUSE_WHEEL; event.bubbles = true; node.dispatchEvent(event); event.stopPropagation(); } }; function _searchComponentsInParent (node, comp) { if (comp) { let index = 0; let list = null; for (var curr = node; curr && cc.Node.isNode(curr); curr = curr._parent, ++index) { if (curr.getComponent(comp)) { let next = { index: index, node: curr, }; if (list) { list.push(next); } else { list = [next]; } } } return list; } return null; } function _checkListeners (node, events) { if (!(node._objFlags & Destroying)) { if (node._bubblingListeners) { for (let i = 0, l = events.length; i < l; ++i) { if (node._bubblingListeners.hasEventListener(events[i])) { return true; } } } if (node._capturingListeners) { for (let i = 0, l = events.length; i < l; ++i) { if (node._capturingListeners.hasEventListener(events[i])) { return true; } } } return false; } return true; } function _doDispatchEvent (owner, event, cachedArray) { var target, i; event.target = owner; // Event.CAPTURING_PHASE cachedArray.length = 0; owner._getCapturingTargets(event.type, cachedArray); // capturing event.eventPhase = 1; for (i = cachedArray.length - 1; i >= 0; --i) { target = cachedArray[i]; if (target._capturingListeners) { event.currentTarget = target; // fire event target._capturingListeners.emit(event.type, event, cachedArray); // check if propagation stopped if (event._propagationStopped) { cachedArray.length = 0; return; } } } cachedArray.length = 0; // Event.AT_TARGET // checks if destroyed in capturing callbacks event.eventPhase = 2; event.currentTarget = owner; if (owner._capturingListeners) { owner._capturingListeners.emit(event.type, event); } if (!event._propagationImmediateStopped && owner._bubblingListeners) { owner._bubblingListeners.emit(event.type, event); } if (!event._propagationStopped && event.bubbles) { // Event.BUBBLING_PHASE owner._getBubblingTargets(event.type, cachedArray); // propagate event.eventPhase = 3; for (i = 0; i < cachedArray.length; ++i) { target = cachedArray[i]; if (target._bubblingListeners) { event.currentTarget = target; // fire event target._bubblingListeners.emit(event.type, event); // check if propagation stopped if (event._propagationStopped) { cachedArray.length = 0; return; } } } } cachedArray.length = 0; } // traversal the node tree, child cullingMask must keep the same with the parent. function _getActualGroupIndex (node) { let groupIndex = node.groupIndex; if (groupIndex === 0 && node.parent) { groupIndex = _getActualGroupIndex(node.parent); } return groupIndex; } function _updateCullingMask (node) { let index = _getActualGroupIndex(node); node._cullingMask = 1 << index; if (CC_JSB && CC_NATIVERENDERER) { node._proxy && node._proxy.updateCullingMask(); } for (let i = 0; i < node._children.length; i++) { _updateCullingMask(node._children[i]); } } // 2D/3D matrix functions function updateLocalMatrix3D () { if (this._localMatDirty & LocalDirtyFlag.TRSS) { // Update transform let t = this._matrix; let tm = t.m; Trs.toMat4(t, this._trs); // skew if (this._skewX || this._skewY) { let a = tm[0], b = tm[1], c = tm[4], d = tm[5]; let skx = Math.tan(this._skewX * ONE_DEGREE); let sky = Math.tan(this._skewY * ONE_DEGREE); if (skx === Infinity) skx = 99999999; if (sky === Infinity) sky = 99999999; tm[0] = a + c * sky; tm[1] = b + d * sky; tm[4] = c + a * skx; tm[5] = d + b * skx; } this._localMatDirty &= ~LocalDirtyFlag.TRSS; // Register dirty status of world matrix so that it can be recalculated this._worldMatDirty = true; } } function updateLocalMatrix2D () { let dirtyFlag = this._localMatDirty; if (!(dirtyFlag & LocalDirtyFlag.TRSS)) return; // Update transform let t = this._matrix; let tm = t.m; let trs = this._trs; if (dirtyFlag & (LocalDirtyFlag.RS | LocalDirtyFlag.SKEW)) { let rotation = -this._eulerAngles.z; let hasSkew = this._skewX || this._skewY; let sx = trs[7], sy = trs[8]; if (rotation || hasSkew) { let a = 1, b = 0, c = 0, d = 1; // rotation if (rotation) { let rotationRadians = rotation * ONE_DEGREE; c = Math.sin(rotationRadians); d = Math.cos(rotationRadians); a = d; b = -c; } // scale tm[0] = a *= sx; tm[1] = b *= sx; tm[4] = c *= sy; tm[5] = d *= sy; // skew if (hasSkew) { let a = tm[0], b = tm[1], c = tm[4], d = tm[5]; let skx = Math.tan(this._skewX * ONE_DEGREE); let sky = Math.tan(this._skewY * ONE_DEGREE); if (skx === Infinity) skx = 99999999; if (sky === Infinity) sky = 99999999; tm[0] = a + c * sky; tm[1] = b + d * sky; tm[4] = c + a * skx; tm[5] = d + b * skx; } } else { tm[0] = sx; tm[1] = 0; tm[4] = 0; tm[5] = sy; } } // position tm[12] = trs[0]; tm[13] = trs[1]; this._localMatDirty &= ~LocalDirtyFlag.TRSS; // Register dirty status of world matrix so that it can be recalculated this._worldMatDirty = true; } function calculWorldMatrix3D () { // Avoid as much function call as possible if (this._localMatDirty & LocalDirtyFlag.TRSS) { this._updateLocalMatrix(); } if (this._parent) { let parentMat = this._parent._worldMatrix; Mat4.mul(this._worldMatrix, parentMat, this._matrix); } else { Mat4.copy(this._worldMatrix, this._matrix); } this._worldMatDirty = false; } function calculWorldMatrix2D () { // Avoid as much function call as possible if (this._localMatDirty & LocalDirtyFlag.TRSS) { this._updateLocalMatrix(); } // Assume parent world matrix is correct let parent = this._parent; if (parent) { this._mulMat(this._worldMatrix, parent._worldMatrix, this._matrix); } else { Mat4.copy(this._worldMatrix, this._matrix); } this._worldMatDirty = false; } function mulMat2D (out, a, b) { let am = a.m, bm = b.m, outm = out.m; let aa=am[0], ab=am[1], ac=am[4], ad=am[5], atx=am[12], aty=am[13]; let ba=bm[0], bb=bm[1], bc=bm[4], bd=bm[5], btx=bm[12], bty=bm[13]; if (ab !== 0 || ac !== 0) { outm[0] = ba * aa + bb * ac; outm[1] = ba * ab + bb * ad; outm[4] = bc * aa + bd * ac; outm[5] = bc * ab + bd * ad; outm[12] = aa * btx + ac * bty + atx; outm[13] = ab * btx + ad * bty + aty; } else { outm[0] = ba * aa; outm[1] = bb * ad; outm[4] = bc * aa; outm[5] = bd * ad; outm[12] = aa * btx + atx; outm[13] = ad * bty + aty; } } const mulMat3D = Mat4.mul; /** * !#en * Class of all entities in Cocos Creator scenes.
* For events supported by Node, please refer to {{#crossLink "Node.EventType"}}{{/crossLink}} * !#zh * Cocos Creator 场景中的所有节点类。
* 支持的节点事件,请参阅 {{#crossLink "Node.EventType"}}{{/crossLink}}。 * @class Node * @extends _BaseNode */ let NodeDefines = { name: 'cc.Node', extends: BaseNode, properties: { // SERIALIZABLE _opacity: 255, _color: cc.Color.WHITE, _contentSize: cc.Size, _anchorPoint: cc.v2(0.5, 0.5), _position: undefined, _scale: undefined, _trs: null, _eulerAngles: cc.Vec3, _skewX: 0.0, _skewY: 0.0, _zIndex: { default: undefined, type: cc.Integer }, _localZOrder: { default: 0, serializable: false }, _is3DNode: false, // internal properties /** * !#en * Group index of node.
* Which Group this node belongs to will resolve that this node's collision components can collide with which other collision componentns.
* !#zh * 节点的分组索引。
* 节点的分组将关系到节点的碰撞组件可以与哪些碰撞组件相碰撞。
* @property groupIndex * @type {Integer} * @default 0 */ _groupIndex: { default: 0, formerlySerializedAs: 'groupIndex' }, groupIndex: { get () { return this._groupIndex; }, set (value) { this._groupIndex = value; _updateCullingMask(this); this.emit(EventType.GROUP_CHANGED, this); } }, /** * !#en * Group of node.
* Which Group this node belongs to will resolve that this node's collision components can collide with which other collision componentns.
* !#zh * 节点的分组。
* 节点的分组将关系到节点的碰撞组件可以与哪些碰撞组件相碰撞。
* @property group * @type {String} */ group: { get () { return cc.game.groupList[this.groupIndex] || ''; }, set (value) { // update the groupIndex this.groupIndex = cc.game.groupList.indexOf(value); } }, //properties moved from base node begin /** * !#en The position (x, y) of the node in its parent's coordinates. * !#zh 节点在父节点坐标系中的位置(x, y)。 * @property {Vec3} position * @example * cc.log("Node Position: " + node.position); */ /** * !#en x axis position of node. * !#zh 节点 X 轴坐标。 * @property x * @type {Number} * @example * node.x = 100; * cc.log("Node Position X: " + node.x); */ x: { get () { return this._trs[0]; }, set (value) { let trs = this._trs; if (value !== trs[0]) { if (!CC_EDITOR || isFinite(value)) { let oldValue; if (CC_EDITOR) { oldValue = trs[0]; } trs[0] = value; this.setLocalDirty(LocalDirtyFlag.ALL_POSITION); // fast check event if (this._eventMask & POSITION_ON) { // send event if (CC_EDITOR) { this.emit(EventType.POSITION_CHANGED, new cc.Vec3(oldValue, trs[1], trs[2])); } else { this.emit(EventType.POSITION_CHANGED); } } } else { cc.error(ERR_INVALID_NUMBER, 'new x'); } } }, }, /** * !#en y axis position of node. * !#zh 节点 Y 轴坐标。 * @property y * @type {Number} * @example * node.y = 100; * cc.log("Node Position Y: " + node.y); */ y: { get () { return this._trs[1]; }, set (value) { let trs = this._trs; if (value !== trs[1]) { if (!CC_EDITOR || isFinite(value)) { let oldValue; if (CC_EDITOR) { oldValue = trs[1]; } trs[1] = value; this.setLocalDirty(LocalDirtyFlag.ALL_POSITION); // fast check event if (this._eventMask & POSITION_ON) { // send event if (CC_EDITOR) { this.emit(EventType.POSITION_CHANGED, new cc.Vec3(trs[0], oldValue, trs[2])); } else { this.emit(EventType.POSITION_CHANGED); } } } else { cc.error(ERR_INVALID_NUMBER, 'new y'); } } }, }, /** * !#en z axis position of node. * !#zh 节点 Z 轴坐标。 * @property z * @type {Number} */ z: { get () { return this._trs[2]; }, set (value) { let trs = this._trs; if (value !== trs[2]) { if (!CC_EDITOR || isFinite(value)) { let oldValue; if (CC_EDITOR) { oldValue = trs[2]; } trs[2] = value; this.setLocalDirty(LocalDirtyFlag.ALL_POSITION); !CC_NATIVERENDERER && (this._renderFlag |= RenderFlow.FLAG_WORLD_TRANSFORM); // fast check event if (this._eventMask & POSITION_ON) { if (CC_EDITOR) { this.emit(EventType.POSITION_CHANGED, new cc.Vec3(trs[0], trs[1], oldValue)); } else { this.emit(EventType.POSITION_CHANGED); } } } else { cc.error(ERR_INVALID_NUMBER, 'new z'); } } } }, /** * !#en Rotation of node. * !#zh 该节点旋转角度。 * @property rotation * @type {Number} * @deprecated since v2.1 * @example * node.rotation = 90; * cc.log("Node Rotation: " + node.rotation); */ rotation: { get () { if (CC_DEBUG) { cc.warn("`cc.Node.rotation` is deprecated since v2.1.0, please use `-angle` instead. (`this.node.rotation` -> `-this.node.angle`)"); } return -this.angle; }, set (value) { if (CC_DEBUG) { cc.warn("`cc.Node.rotation` is deprecated since v2.1.0, please set `-angle` instead. (`this.node.rotation = x` -> `this.node.angle = -x`)"); } this.angle = -value; } }, /** * !#en * Angle of node, the positive value is anti-clockwise direction. * !#zh * 该节点的旋转角度,正值为逆时针方向。 * @property angle * @type {Number} */ angle: { get () { return this._eulerAngles.z; }, set (value) { Vec3.set(this._eulerAngles, 0, 0, value); Trs.fromAngleZ(this._trs, value); this.setLocalDirty(LocalDirtyFlag.ALL_ROTATION); if (this._eventMask & ROTATION_ON) { this.emit(EventType.ROTATION_CHANGED); } } }, /** * !#en The rotation as Euler angles in degrees, used in 3D node. * !#zh 该节点的欧拉角度,用于 3D 节点。 * @property eulerAngles * @type {Vec3} * @example * node.is3DNode = true; * node.eulerAngles = cc.v3(45, 45, 45); * cc.log("Node eulerAngles (X, Y, Z): " + node.eulerAngles.toString()); */ /** * !#en Rotation on x axis. * !#zh 该节点 X 轴旋转角度。 * @property rotationX * @type {Number} * @deprecated since v2.1 * @example * node.is3DNode = true; * node.eulerAngles = cc.v3(45, 0, 0); * cc.log("Node eulerAngles X: " + node.eulerAngles.x); */ rotationX: { get () { if (CC_DEBUG) { cc.warn("`cc.Node.rotationX` is deprecated since v2.1.0, please use `eulerAngles.x` instead. (`this.node.rotationX` -> `this.node.eulerAngles.x`)"); } return this._eulerAngles.x; }, set (value) { if (CC_DEBUG) { cc.warn("`cc.Node.rotationX` is deprecated since v2.1.0, please set `eulerAngles` instead. (`this.node.rotationX = x` -> `this.node.is3DNode = true; this.node.eulerAngles = cc.v3(x, 0, 0)`"); } if (this._eulerAngles.x !== value) { this._eulerAngles.x = value; // Update quaternion from rotation if (this._eulerAngles.x === this._eulerAngles.y) { Trs.fromAngleZ(this._trs, -value); } else { Trs.fromEulerNumber(this._trs, value, this._eulerAngles.y, 0); } this.setLocalDirty(LocalDirtyFlag.ALL_ROTATION); if (this._eventMask & ROTATION_ON) { this.emit(EventType.ROTATION_CHANGED); } } }, }, /** * !#en Rotation on y axis. * !#zh 该节点 Y 轴旋转角度。 * @property rotationY * @type {Number} * @deprecated since v2.1 * @example * node.is3DNode = true; * node.eulerAngles = cc.v3(0, 45, 0); * cc.log("Node eulerAngles Y: " + node.eulerAngles.y); */ rotationY: { get () { if (CC_DEBUG) { cc.warn("`cc.Node.rotationY` is deprecated since v2.1.0, please use `eulerAngles.y` instead. (`this.node.rotationY` -> `this.node.eulerAngles.y`)"); } return this._eulerAngles.y; }, set (value) { if (CC_DEBUG) { cc.warn("`cc.Node.rotationY` is deprecated since v2.1.0, please set `eulerAngles` instead. (`this.node.rotationY = y` -> `this.node.is3DNode = true; this.node.eulerAngles = cc.v3(0, y, 0)`"); } if (this._eulerAngles.y !== value) { this._eulerAngles.y = value; // Update quaternion from rotation if (this._eulerAngles.x === this._eulerAngles.y) { Trs.fromAngleZ(this._trs, -value); } else { Trs.fromEulerNumber(this._trs, this._eulerAngles.x, value, 0); } this.setLocalDirty(LocalDirtyFlag.ALL_ROTATION); if (this._eventMask & ROTATION_ON) { this.emit(EventType.ROTATION_CHANGED); } } }, }, eulerAngles: { get () { if (CC_EDITOR) { return this._eulerAngles; } else { return Trs.toEuler(this._eulerAngles, this._trs); } }, set (v) { if (CC_EDITOR) { this._eulerAngles.set(v); } Trs.fromEuler(this._trs, v); this.setLocalDirty(LocalDirtyFlag.ALL_ROTATION); !CC_NATIVERENDERER && (this._renderFlag |= RenderFlow.FLAG_TRANSFORM); if (this._eventMask & ROTATION_ON) { this.emit(EventType.ROTATION_CHANGED); } } }, // This property is used for Mesh Skeleton Animation // Should be removed when node.rotation upgrade to quaternion value quat: { get () { let trs = this._trs; return new Quat(trs[3], trs[4], trs[5], trs[6]); }, set (v) { this.setRotation(v); } }, /** * !#en The local scale relative to the parent. * !#zh 节点相对父节点的缩放。 * @property scale * @type {Number} * @example * node.scale = 1; */ scale: { get () { return this._trs[7]; }, set (v) { this.setScale(v); } }, /** * !#en Scale on x axis. * !#zh 节点 X 轴缩放。 * @property scaleX * @type {Number} * @example * node.scaleX = 0.5; * cc.log("Node Scale X: " + node.scaleX); */ scaleX: { get () { return this._trs[7]; }, set (value) { if (this._trs[7] !== value) { this._trs[7] = value; this.setLocalDirty(LocalDirtyFlag.ALL_SCALE); if (this._eventMask & SCALE_ON) { this.emit(EventType.SCALE_CHANGED); } } }, }, /** * !#en Scale on y axis. * !#zh 节点 Y 轴缩放。 * @property scaleY * @type {Number} * @example * node.scaleY = 0.5; * cc.log("Node Scale Y: " + node.scaleY); */ scaleY: { get () { return this._trs[8]; }, set (value) { if (this._trs[8] !== value) { this._trs[8] = value; this.setLocalDirty(LocalDirtyFlag.ALL_SCALE); if (this._eventMask & SCALE_ON) { this.emit(EventType.SCALE_CHANGED); } } }, }, /** * !#en Scale on z axis. * !#zh 节点 Z 轴缩放。 * @property scaleZ * @type {Number} */ scaleZ: { get () { return this._trs[9]; }, set (value) { if (this._trs[9] !== value) { this._trs[9] = value; this.setLocalDirty(LocalDirtyFlag.ALL_SCALE); !CC_NATIVERENDERER && (this._renderFlag |= RenderFlow.FLAG_TRANSFORM); if (this._eventMask & SCALE_ON) { this.emit(EventType.SCALE_CHANGED); } } } }, /** * !#en Skew x * !#zh 该节点 X 轴倾斜角度。 * @property skewX * @type {Number} * @example * node.skewX = 0; * cc.log("Node SkewX: " + node.skewX); * @deprecated since v2.2.1 */ skewX: { get () { return this._skewX; }, set (value) { _skewWarn(value, this); this._skewX = value; this.setLocalDirty(LocalDirtyFlag.SKEW); if (CC_JSB && CC_NATIVERENDERER) { this._proxy.updateSkew(); } } }, /** * !#en Skew y * !#zh 该节点 Y 轴倾斜角度。 * @property skewY * @type {Number} * @example * node.skewY = 0; * cc.log("Node SkewY: " + node.skewY); * @deprecated since v2.2.1 */ skewY: { get () { return this._skewY; }, set (value) { _skewWarn(value, this); this._skewY = value; this.setLocalDirty(LocalDirtyFlag.SKEW); if (CC_JSB && CC_NATIVERENDERER) { this._proxy.updateSkew(); } } }, /** * !#en Opacity of node, default value is 255. * !#zh 节点透明度,默认值为 255。 * @property opacity * @type {Number} * @example * node.opacity = 255; */ opacity: { get () { return this._opacity; }, set (value) { value = cc.misc.clampf(value, 0, 255); if (this._opacity !== value) { this._opacity = value; if (CC_JSB && CC_NATIVERENDERER) { this._proxy.updateOpacity(); } this._renderFlag |= RenderFlow.FLAG_OPACITY_COLOR; } }, range: [0, 255] }, /** * !#en Color of node, default value is white: (255, 255, 255). * !#zh 节点颜色。默认为白色,数值为:(255,255,255)。 * @property color * @type {Color} * @example * node.color = new cc.Color(255, 255, 255); */ color: { get () { return this._color.clone() }, set (value) { if (!this._color.equals(value)) { this._color.set(value); if (CC_DEV && value.a !== 255) { cc.warnID(1626); } this._renderFlag |= RenderFlow.FLAG_COLOR; if (this._eventMask & COLOR_ON) { this.emit(EventType.COLOR_CHANGED, value); } } }, }, /** * !#en Anchor point's position on x axis. * !#zh 节点 X 轴锚点位置。 * @property anchorX * @type {Number} * @example * node.anchorX = 0; */ anchorX: { get () { return this._anchorPoint.x; }, set (value) { var anchorPoint = this._anchorPoint; if (anchorPoint.x !== value) { anchorPoint.x = value; if (this._eventMask & ANCHOR_ON) { this.emit(EventType.ANCHOR_CHANGED); } } }, }, /** * !#en Anchor point's position on y axis. * !#zh 节点 Y 轴锚点位置。 * @property anchorY * @type {Number} * @example * node.anchorY = 0; */ anchorY: { get () { return this._anchorPoint.y; }, set (value) { var anchorPoint = this._anchorPoint; if (anchorPoint.y !== value) { anchorPoint.y = value; if (this._eventMask & ANCHOR_ON) { this.emit(EventType.ANCHOR_CHANGED); } } }, }, /** * !#en Width of node. * !#zh 节点宽度。 * @property width * @type {Number} * @example * node.width = 100; */ width: { get () { return this._contentSize.width; }, set (value) { if (value !== this._contentSize.width) { if (CC_EDITOR) { var clone = cc.size(this._contentSize.width, this._contentSize.height); } this._contentSize.width = value; if (this._eventMask & SIZE_ON) { if (CC_EDITOR) { this.emit(EventType.SIZE_CHANGED, clone); } else { this.emit(EventType.SIZE_CHANGED); } } } }, }, /** * !#en Height of node. * !#zh 节点高度。 * @property height * @type {Number} * @example * node.height = 100; */ height: { get () { return this._contentSize.height; }, set (value) { if (value !== this._contentSize.height) { if (CC_EDITOR) { var clone = cc.size(this._contentSize.width, this._contentSize.height); } this._contentSize.height = value; if (this._eventMask & SIZE_ON) { if (CC_EDITOR) { this.emit(EventType.SIZE_CHANGED, clone); } else { this.emit(EventType.SIZE_CHANGED); } } } }, }, /** * !#en zIndex is the 'key' used to sort the node relative to its siblings.
* The value of zIndex should be in the range between cc.macro.MIN_ZINDEX and cc.macro.MAX_ZINDEX.
* The Node's parent will sort all its children based on the zIndex value and the arrival order.
* Nodes with greater zIndex will be sorted after nodes with smaller zIndex.
* If two nodes have the same zIndex, then the node that was added first to the children's array will be in front of the other node in the array.
* Node's order in children list will affect its rendering order. Parent is always rendering before all children. * !#zh zIndex 是用来对节点进行排序的关键属性,它决定一个节点在兄弟节点之间的位置。
* zIndex 的取值应该介于 cc.macro.MIN_ZINDEX 和 cc.macro.MAX_ZINDEX 之间 * 父节点主要根据节点的 zIndex 和添加次序来排序,拥有更高 zIndex 的节点将被排在后面,如果两个节点的 zIndex 一致,先添加的节点会稳定排在另一个节点之前。
* 节点在 children 中的顺序决定了其渲染顺序。父节点永远在所有子节点之前被渲染 * @property zIndex * @type {Number} * @example * node.zIndex = 1; * cc.log("Node zIndex: " + node.zIndex); */ zIndex: { get () { return this._localZOrder >> 16; }, set (value) { if (value > macro.MAX_ZINDEX) { cc.warnID(1636); value = macro.MAX_ZINDEX; } else if (value < macro.MIN_ZINDEX) { cc.warnID(1637); value = macro.MIN_ZINDEX; } if (this.zIndex !== value) { this._localZOrder = (this._localZOrder & 0x0000ffff) | (value << 16); this.emit(EventType.SIBLING_ORDER_CHANGED); this._onSiblingIndexChanged(); } } }, /** * !#en * Switch 2D/3D node. The 2D nodes will run faster. * !#zh * 切换 2D/3D 节点,2D 节点会有更高的运行效率 * @property {Boolean} is3DNode * @default false */ is3DNode: { get () { return this._is3DNode; }, set (v) { if (this._is3DNode === v) { return; } this._is3DNode = v; this._update3DFunction(); } }, /** * !#en Returns a normalized vector representing the up direction (Y axis) of the node in world space. * !#zh 获取节点正上方(y 轴)面对的方向,返回值为世界坐标系下的归一化向量 * * @property up * @type {Vec3} */ up: { get () { var _up = Vec3.transformQuat(_urfVec3, Vec3.UP, this.getWorldRotation(_urfQuat)); return _up.clone(); } }, /** * !#en Returns a normalized vector representing the right direction (X axis) of the node in world space. * !#zh 获取节点正右方(x 轴)面对的方向,返回值为世界坐标系下的归一化向量 * * @property right * @type {Vec3} */ right: { get () { var _right = Vec3.transformQuat(_urfVec3, Vec3.RIGHT, this.getWorldRotation(_urfQuat)); return _right.clone(); } }, /** * !#en Returns a normalized vector representing the forward direction (Z axis) of the node in world space. * !#zh 获取节点正前方(z 轴)面对的方向,返回值为世界坐标系下的归一化向量 * * @property forward * @type {Vec3} */ forward: { get () { var _forward = Vec3.transformQuat(_urfVec3, Vec3.FORWARD, this.getWorldRotation(_urfQuat)); return _forward.clone(); } }, }, /** * @method constructor * @param {String} [name] */ ctor () { this._reorderChildDirty = false; // cache component this._widget = null; // fast render component access this._renderComponent = null; // Event listeners this._capturingListeners = null; this._bubblingListeners = null; // Touch event listener this._touchListener = null; // Mouse event listener this._mouseListener = null; this._initDataFromPool(); this._eventMask = 0; this._cullingMask = 1; this._childArrivalOrder = 1; // Proxy if (CC_JSB && CC_NATIVERENDERER) { this._proxy = new renderer.NodeProxy(this._spaceInfo.unitID, this._spaceInfo.index, this._id, this._name); this._proxy.init(this); } // should reset _renderFlag for both web and native this._renderFlag = RenderFlow.FLAG_TRANSFORM | RenderFlow.FLAG_OPACITY_COLOR; }, statics: { EventType, _LocalDirtyFlag: LocalDirtyFlag, // is node but not scene isNode (obj) { return obj instanceof Node && (obj.constructor === Node || !(obj instanceof cc.Scene)); }, BuiltinGroupIndex }, // OVERRIDES _onSiblingIndexChanged () { // update rendering scene graph, sort them by arrivalOrder if (this._parent) { this._parent._delaySort(); } }, _onPreDestroy () { var destroyByParent = this._onPreDestroyBase(); // Actions if (ActionManagerExist) { cc.director.getActionManager().removeAllActionsFromTarget(this); } // Remove Node.currentHovered if (_currentHovered === this) { _currentHovered = null; } this._bubblingListeners && this._bubblingListeners.clear(); this._capturingListeners && this._capturingListeners.clear(); // Remove all event listeners if necessary if (this._touchListener || this._mouseListener) { eventManager.removeListeners(this); if (this._touchListener) { this._touchListener.owner = null; this._touchListener.mask = null; this._touchListener = null; } if (this._mouseListener) { this._mouseListener.owner = null; this._mouseListener.mask = null; this._mouseListener = null; } } if (CC_JSB && CC_NATIVERENDERER) { this._proxy.destroy(); this._proxy = null; } this._backDataIntoPool(); if (this._reorderChildDirty) { cc.director.__fastOff(cc.Director.EVENT_AFTER_UPDATE, this.sortAllChildren, this); } if (!destroyByParent) { // simulate some destruct logic to make undo system work correctly if (CC_EDITOR) { // ensure this node can reattach to scene by undo system this._parent = null; } } }, _onPostActivated (active) { var actionManager = ActionManagerExist ? cc.director.getActionManager() : null; if (active) { // Refresh transform this._renderFlag |= RenderFlow.FLAG_WORLD_TRANSFORM; // ActionManager & EventManager actionManager && actionManager.resumeTarget(this); eventManager.resumeTarget(this); // Search Mask in parent this._checkListenerMask(); } else { // deactivate actionManager && actionManager.pauseTarget(this); eventManager.pauseTarget(this); } }, _onHierarchyChanged (oldParent) { this._updateOrderOfArrival(); // Fixed a bug where children and parent node groups were forced to synchronize, instead of only synchronizing `_cullingMask` value _updateCullingMask(this); if (this._parent) { this._parent._delaySort(); } this._renderFlag |= RenderFlow.FLAG_WORLD_TRANSFORM | RenderFlow.FLAG_OPACITY_COLOR; this._onHierarchyChangedBase(oldParent); if (cc._widgetManager) { cc._widgetManager._nodesOrderDirty = true; } if (oldParent && this._activeInHierarchy) { //TODO: It may be necessary to update the listener mask of all child nodes. this._checkListenerMask(); } // Node proxy if (CC_JSB && CC_NATIVERENDERER) { this._proxy.updateParent(); } }, // INTERNAL _update3DFunction () { if (this._is3DNode) { this._updateLocalMatrix = updateLocalMatrix3D; this._calculWorldMatrix = calculWorldMatrix3D; this._mulMat = mulMat3D; } else { this._updateLocalMatrix = updateLocalMatrix2D; this._calculWorldMatrix = calculWorldMatrix2D; this._mulMat = mulMat2D; } if (this._renderComponent && this._renderComponent._on3DNodeChanged) { this._renderComponent._on3DNodeChanged(); } this._renderFlag |= RenderFlow.FLAG_TRANSFORM; this._localMatDirty = LocalDirtyFlag.ALL; if (CC_JSB && CC_NATIVERENDERER) { this._proxy.update3DNode(); } }, _initDataFromPool () { if (!this._spaceInfo) { if (CC_EDITOR || CC_TEST) { this._spaceInfo = { trs: new Float64Array(10), localMat: new Float64Array(16), worldMat: new Float64Array(16), }; } else { this._spaceInfo = nodeMemPool.pop(); } } let spaceInfo = this._spaceInfo; this._matrix = cc.mat4(spaceInfo.localMat); Mat4.identity(this._matrix); this._worldMatrix = cc.mat4(spaceInfo.worldMat); Mat4.identity(this._worldMatrix); this._localMatDirty = LocalDirtyFlag.ALL; this._worldMatDirty = true; let trs = this._trs = spaceInfo.trs; trs[0] = 0; // position.x trs[1] = 0; // position.y trs[2] = 0; // position.z trs[3] = 0; // rotation.x trs[4] = 0; // rotation.y trs[5] = 0; // rotation.z trs[6] = 1; // rotation.w trs[7] = 1; // scale.x trs[8] = 1; // scale.y trs[9] = 1; // scale.z }, _backDataIntoPool () { if (!(CC_EDITOR || CC_TEST)) { // push back to pool nodeMemPool.push(this._spaceInfo); this._matrix = null; this._worldMatrix = null; this._trs = null; this._spaceInfo = null; } }, _toEuler () { if (this.is3DNode) { Trs.toEuler(this._eulerAngles, this._trs); } else { let z = Math.asin(this._trs[5]) / ONE_DEGREE * 2; Vec3.set(this._eulerAngles, 0, 0, z); } }, _fromEuler () { if (this.is3DNode) { Trs.fromEuler(this._trs, this._eulerAngles); } else { Trs.fromAngleZ(this._trs, this._eulerAngles.z); } }, _initProperties () { if (this._is3DNode) { this._update3DFunction(); } let trs = this._trs; if (trs) { let desTrs = trs; trs = this._trs = this._spaceInfo.trs; // just adapt to old trs if (desTrs.length === 11) { trs.set(desTrs.subarray(1)); } else { trs.set(desTrs); } } else { trs = this._trs = this._spaceInfo.trs; } if (CC_EDITOR) { if (this._skewX !== 0 || this._skewY !== 0) { var NodeUtils = Editor.require('scene://utils/node'); cc.warn("`cc.Node.skewX/Y` is deprecated since v2.2.1, please use 3D node instead.", `Node: ${NodeUtils.getNodePath(this)}.`); } } this._fromEuler(); if (CC_JSB && CC_NATIVERENDERER) { this._renderFlag |= RenderFlow.FLAG_TRANSFORM | RenderFlow.FLAG_OPACITY_COLOR; } }, /* * The initializer for Node which will be called before all components onLoad */ _onBatchCreated (dontSyncChildPrefab) { this._initProperties(); // Fixed a bug where children and parent node groups were forced to synchronize, instead of only synchronizing `_cullingMask` value this._cullingMask = 1 << _getActualGroupIndex(this); if (CC_JSB && CC_NATIVERENDERER) { this._proxy && this._proxy.updateCullingMask(); } if (!this._activeInHierarchy) { if (CC_EDITOR ? cc.director.getActionManager() : ActionManagerExist) { // deactivate ActionManager and EventManager by default cc.director.getActionManager().pauseTarget(this); } eventManager.pauseTarget(this); } let children = this._children; for (let i = 0, len = children.length; i < len; i++) { let child = children[i]; if (!dontSyncChildPrefab) { // sync child prefab let prefabInfo = child._prefab; if (prefabInfo && prefabInfo.sync && prefabInfo.root === child) { PrefabHelper.syncWithPrefab(child); } child._updateOrderOfArrival(); } child._onBatchCreated(dontSyncChildPrefab); } if (children.length > 0) { this._renderFlag |= RenderFlow.FLAG_CHILDREN; } if (CC_JSB && CC_NATIVERENDERER) { this._proxy.initNative(); } }, // EVENT TARGET _checkListenerMask () { // Because Mask may be nested, need to find all the Mask components in the parent node. // The click area must satisfy all Masks to trigger the click. if (this._touchListener) { var mask = this._touchListener.mask = _searchComponentsInParent(this, cc.Mask); if (this._mouseListener) { this._mouseListener.mask = mask; } } else if (this._mouseListener) { this._mouseListener.mask = _searchComponentsInParent(this, cc.Mask); } }, _checknSetupSysEvent (type) { let newAdded = false; let forDispatch = false; if (_touchEvents.indexOf(type) !== -1) { if (!this._touchListener) { this._touchListener = cc.EventListener.create({ event: cc.EventListener.TOUCH_ONE_BY_ONE, swallowTouches: true, owner: this, mask: _searchComponentsInParent(this, cc.Mask), onTouchBegan: _touchStartHandler, onTouchMoved: _touchMoveHandler, onTouchEnded: _touchEndHandler, onTouchCancelled: _touchCancelHandler }); eventManager.addListener(this._touchListener, this); newAdded = true; } forDispatch = true; } else if (_mouseEvents.indexOf(type) !== -1) { if (!this._mouseListener) { this._mouseListener = cc.EventListener.create({ event: cc.EventListener.MOUSE, _previousIn: false, owner: this, mask: _searchComponentsInParent(this, cc.Mask), onMouseDown: _mouseDownHandler, onMouseMove: _mouseMoveHandler, onMouseUp: _mouseUpHandler, onMouseScroll: _mouseWheelHandler, }); eventManager.addListener(this._mouseListener, this); newAdded = true; } forDispatch = true; } if (newAdded && !this._activeInHierarchy) { cc.director.getScheduler().schedule(function () { if (!this._activeInHierarchy) { eventManager.pauseTarget(this); } }, this, 0, 0, 0, false); } return forDispatch; }, /** * !#en * Register a callback of a specific event type on Node.
* Use this method to register touch or mouse event permit propagation based on scene graph,
* These kinds of event are triggered with dispatchEvent, the dispatch process has three steps:
* 1. Capturing phase: dispatch in capture targets (`_getCapturingTargets`), e.g. parents in node tree, from root to the real target
* 2. At target phase: dispatch to the listeners of the real target
* 3. Bubbling phase: dispatch in bubble targets (`_getBubblingTargets`), e.g. parents in node tree, from the real target to root
* In any moment of the dispatching process, it can be stopped via `event.stopPropagation()` or `event.stopPropagationImmidiate()`.
* It's the recommended way to register touch/mouse event for Node,
* please do not use cc.eventManager directly for Node.
* You can also register custom event and use `emit` to trigger custom event on Node.
* For such events, there won't be capturing and bubbling phase, your event will be dispatched directly to its listeners registered on the same node.
* You can also pass event callback parameters with `emit` by passing parameters after `type`. * !#zh * 在节点上注册指定类型的回调函数,也可以设置 target 用于绑定响应函数的 this 对象。
* 鼠标或触摸事件会被系统调用 dispatchEvent 方法触发,触发的过程包含三个阶段:
* 1. 捕获阶段:派发事件给捕获目标(通过 `_getCapturingTargets` 获取),比如,节点树中注册了捕获阶段的父节点,从根节点开始派发直到目标节点。
* 2. 目标阶段:派发给目标节点的监听器。
* 3. 冒泡阶段:派发事件给冒泡目标(通过 `_getBubblingTargets` 获取),比如,节点树中注册了冒泡阶段的父节点,从目标节点开始派发直到根节点。
* 同时您可以将事件派发到父节点或者通过调用 stopPropagation 拦截它。
* 推荐使用这种方式来监听节点上的触摸或鼠标事件,请不要在节点上直接使用 cc.eventManager。
* 你也可以注册自定义事件到节点上,并通过 emit 方法触发此类事件,对于这类事件,不会发生捕获冒泡阶段,只会直接派发给注册在该节点上的监听器
* 你可以通过在 emit 方法调用时在 type 之后传递额外的参数作为事件回调的参数列表 * @method on * @param {String|Node.EventType} type - A string representing the event type to listen for.
See {{#crossLink "Node/EventTyupe/POSITION_CHANGED"}}Node Events{{/crossLink}} for all builtin events. * @param {Function} callback - The callback that will be invoked when the event is dispatched. The callback is ignored if it is a duplicate (the callbacks are unique). * @param {Event|any} [callback.event] event or first argument when emit * @param {any} [callback.arg2] arg2 * @param {any} [callback.arg3] arg3 * @param {any} [callback.arg4] arg4 * @param {any} [callback.arg5] arg5 * @param {Object} [target] - The target (this object) to invoke the callback, can be null * @param {Boolean} [useCapture=false] - When set to true, the listener will be triggered at capturing phase which is ahead of the final target emit, otherwise it will be triggered during bubbling phase. * @return {Function} - Just returns the incoming callback so you can save the anonymous function easier. * @typescript * on(type: string, callback: T, target?: any, useCapture?: boolean): T * @example * this.node.on(cc.Node.EventType.TOUCH_START, this.memberFunction, this); // if "this" is component and the "memberFunction" declared in CCClass. * node.on(cc.Node.EventType.TOUCH_START, callback, this); * node.on(cc.Node.EventType.TOUCH_MOVE, callback, this); * node.on(cc.Node.EventType.TOUCH_END, callback, this); * node.on(cc.Node.EventType.TOUCH_CANCEL, callback, this); * node.on(cc.Node.EventType.ANCHOR_CHANGED, callback); * node.on(cc.Node.EventType.COLOR_CHANGED, callback); */ on (type, callback, target, useCapture) { let forDispatch = this._checknSetupSysEvent(type); if (forDispatch) { return this._onDispatch(type, callback, target, useCapture); } else { switch (type) { case EventType.POSITION_CHANGED: this._eventMask |= POSITION_ON; break; case EventType.SCALE_CHANGED: this._eventMask |= SCALE_ON; break; case EventType.ROTATION_CHANGED: this._eventMask |= ROTATION_ON; break; case EventType.SIZE_CHANGED: this._eventMask |= SIZE_ON; break; case EventType.ANCHOR_CHANGED: this._eventMask |= ANCHOR_ON; break; case EventType.COLOR_CHANGED: this._eventMask |= COLOR_ON; break; } if (!this._bubblingListeners) { this._bubblingListeners = new EventTarget(); } return this._bubblingListeners.on(type, callback, target); } }, /** * !#en * Register an callback of a specific event type on the Node, * the callback will remove itself after the first time it is triggered. * !#zh * 注册节点的特定事件类型回调,回调会在第一时间被触发后删除自身。 * * @method once * @param {String} type - A string representing the event type to listen for. * @param {Function} callback - The callback that will be invoked when the event is dispatched. * The callback is ignored if it is a duplicate (the callbacks are unique). * @param {Event|any} [callback.event] event or first argument when emit * @param {any} [callback.arg2] arg2 * @param {any} [callback.arg3] arg3 * @param {any} [callback.arg4] arg4 * @param {any} [callback.arg5] arg5 * @param {Object} [target] - The target (this object) to invoke the callback, can be null * @typescript * once(type: string, callback: T, target?: any, useCapture?: boolean): T * @example * node.once(cc.Node.EventType.ANCHOR_CHANGED, callback); */ once (type, callback, target, useCapture) { let forDispatch = this._checknSetupSysEvent(type); let listeners = null; if (forDispatch && useCapture) { listeners = this._capturingListeners = this._capturingListeners || new EventTarget(); } else { listeners = this._bubblingListeners = this._bubblingListeners || new EventTarget(); } listeners.once(type, callback, target); listeners.once(type, () => { this.off(type, callback, target); }, undefined); }, _onDispatch (type, callback, target, useCapture) { // Accept also patameters like: (type, callback, useCapture) if (typeof target === 'boolean') { useCapture = target; target = undefined; } else useCapture = !!useCapture; if (!callback) { cc.errorID(6800); return; } var listeners = null; if (useCapture) { listeners = this._capturingListeners = this._capturingListeners || new EventTarget(); } else { listeners = this._bubblingListeners = this._bubblingListeners || new EventTarget(); } if ( !listeners.hasEventListener(type, callback, target) ) { listeners.on(type, callback, target); if (target && target.__eventTargets) { target.__eventTargets.push(this); } } return callback; }, /** * !#en * Removes the callback previously registered with the same type, callback, target and or useCapture. * This method is merely an alias to removeEventListener. * !#zh 删除之前与同类型,回调,目标或 useCapture 注册的回调。 * @method off * @param {String} type - A string representing the event type being removed. * @param {Function} [callback] - The callback to remove. * @param {Object} [target] - The target (this object) to invoke the callback, if it's not given, only callback without target will be removed * @param {Boolean} [useCapture=false] - When set to true, the listener will be triggered at capturing phase which is ahead of the final target emit, otherwise it will be triggered during bubbling phase. * @example * this.node.off(cc.Node.EventType.TOUCH_START, this.memberFunction, this); * node.off(cc.Node.EventType.TOUCH_START, callback, this.node); * node.off(cc.Node.EventType.ANCHOR_CHANGED, callback, this); */ off (type, callback, target, useCapture) { let touchEvent = _touchEvents.indexOf(type) !== -1; let mouseEvent = !touchEvent && _mouseEvents.indexOf(type) !== -1; if (touchEvent || mouseEvent) { this._offDispatch(type, callback, target, useCapture); if (touchEvent) { if (this._touchListener && !_checkListeners(this, _touchEvents)) { eventManager.removeListener(this._touchListener); this._touchListener = null; } } else if (mouseEvent) { if (this._mouseListener && !_checkListeners(this, _mouseEvents)) { eventManager.removeListener(this._mouseListener); this._mouseListener = null; } } } else if (this._bubblingListeners) { this._bubblingListeners.off(type, callback, target); var hasListeners = this._bubblingListeners.hasEventListener(type); // All listener removed if (!hasListeners) { switch (type) { case EventType.POSITION_CHANGED: this._eventMask &= ~POSITION_ON; break; case EventType.SCALE_CHANGED: this._eventMask &= ~SCALE_ON; break; case EventType.ROTATION_CHANGED: this._eventMask &= ~ROTATION_ON; break; case EventType.SIZE_CHANGED: this._eventMask &= ~SIZE_ON; break; case EventType.ANCHOR_CHANGED: this._eventMask &= ~ANCHOR_ON; break; case EventType.COLOR_CHANGED: this._eventMask &= ~COLOR_ON; break; } } } }, _offDispatch (type, callback, target, useCapture) { // Accept also patameters like: (type, callback, useCapture) if (typeof target === 'boolean') { useCapture = target; target = undefined; } else useCapture = !!useCapture; if (!callback) { this._capturingListeners && this._capturingListeners.removeAll(type); this._bubblingListeners && this._bubblingListeners.removeAll(type); } else { var listeners = useCapture ? this._capturingListeners : this._bubblingListeners; if (listeners) { listeners.off(type, callback, target); if (target && target.__eventTargets) { js.array.fastRemove(target.__eventTargets, this); } } } }, /** * !#en Removes all callbacks previously registered with the same target. * !#zh 移除目标上的所有注册事件。 * @method targetOff * @param {Object} target - The target to be searched for all related callbacks * @example * node.targetOff(target); */ targetOff (target) { let listeners = this._bubblingListeners; if (listeners) { listeners.targetOff(target); // Check for event mask reset if ((this._eventMask & POSITION_ON) && !listeners.hasEventListener(EventType.POSITION_CHANGED)) { this._eventMask &= ~POSITION_ON; } if ((this._eventMask & SCALE_ON) && !listeners.hasEventListener(EventType.SCALE_CHANGED)) { this._eventMask &= ~SCALE_ON; } if ((this._eventMask & ROTATION_ON) && !listeners.hasEventListener(EventType.ROTATION_CHANGED)) { this._eventMask &= ~ROTATION_ON; } if ((this._eventMask & SIZE_ON) && !listeners.hasEventListener(EventType.SIZE_CHANGED)) { this._eventMask &= ~SIZE_ON; } if ((this._eventMask & ANCHOR_ON) && !listeners.hasEventListener(EventType.ANCHOR_CHANGED)) { this._eventMask &= ~ANCHOR_ON; } if ((this._eventMask & COLOR_ON) && !listeners.hasEventListener(EventType.COLOR_CHANGED)) { this._eventMask &= ~COLOR_ON; } } if (this._capturingListeners) { this._capturingListeners.targetOff(target); } if (target && target.__eventTargets) { js.array.fastRemove(target.__eventTargets, this); } if (this._touchListener && !_checkListeners(this, _touchEvents)) { eventManager.removeListener(this._touchListener); this._touchListener = null; } if (this._mouseListener && !_checkListeners(this, _mouseEvents)) { eventManager.removeListener(this._mouseListener); this._mouseListener = null; } }, /** * !#en Checks whether the EventTarget object has any callback registered for a specific type of event. * !#zh 检查事件目标对象是否有为特定类型的事件注册的回调。 * @method hasEventListener * @param {String} type - The type of event. * @return {Boolean} True if a callback of the specified type is registered; false otherwise. */ hasEventListener (type) { let has = false; if (this._bubblingListeners) { has = this._bubblingListeners.hasEventListener(type); } if (!has && this._capturingListeners) { has = this._capturingListeners.hasEventListener(type); } return has; }, /** * !#en * Trigger an event directly with the event name and necessary arguments. * !#zh * 通过事件名发送自定义事件 * * @method emit * @param {String} type - event type * @param {*} [arg1] - First argument in callback * @param {*} [arg2] - Second argument in callback * @param {*} [arg3] - Third argument in callback * @param {*} [arg4] - Fourth argument in callback * @param {*} [arg5] - Fifth argument in callback * @example * * eventTarget.emit('fire', event); * eventTarget.emit('fire', message, emitter); */ emit (type, arg1, arg2, arg3, arg4, arg5) { if (this._bubblingListeners) { this._bubblingListeners.emit(type, arg1, arg2, arg3, arg4, arg5); } }, /** * !#en * Dispatches an event into the event flow. * The event target is the EventTarget object upon which the dispatchEvent() method is called. * !#zh 分发事件到事件流中。 * * @method dispatchEvent * @param {Event} event - The Event object that is dispatched into the event flow */ dispatchEvent (event) { var _array = _cachedPool.get(); _doDispatchEvent(this, event, _array); _cachedPool.put(_array); }, /** * !#en Pause node related system events registered with the current Node. Node system events includes touch and mouse events. * If recursive is set to true, then this API will pause the node system events for the node and all nodes in its sub node tree. * Reference: http://docs.cocos2d-x.org/editors_and_tools/creator-chapters/scripting/internal-events/ * !#zh 暂停当前节点上注册的所有节点系统事件,节点系统事件包含触摸和鼠标事件。 * 如果传递 recursive 为 true,那么这个 API 将暂停本节点和它的子树上所有节点的节点系统事件。 * 参考:https://www.cocos.com/docs/creator/scripting/internal-events.html * @method pauseSystemEvents * @param {Boolean} recursive - Whether to pause node system events on the sub node tree. * @example * node.pauseSystemEvents(true); */ pauseSystemEvents (recursive) { eventManager.pauseTarget(this, recursive); }, /** * !#en Resume node related system events registered with the current Node. Node system events includes touch and mouse events. * If recursive is set to true, then this API will resume the node system events for the node and all nodes in its sub node tree. * Reference: http://docs.cocos2d-x.org/editors_and_tools/creator-chapters/scripting/internal-events/ * !#zh 恢复当前节点上注册的所有节点系统事件,节点系统事件包含触摸和鼠标事件。 * 如果传递 recursive 为 true,那么这个 API 将恢复本节点和它的子树上所有节点的节点系统事件。 * 参考:https://www.cocos.com/docs/creator/scripting/internal-events.html * @method resumeSystemEvents * @param {Boolean} recursive - Whether to resume node system events on the sub node tree. * @example * node.resumeSystemEvents(true); */ resumeSystemEvents (recursive) { eventManager.resumeTarget(this, recursive); }, _hitTest (point, listener) { let w = this._contentSize.width, h = this._contentSize.height, cameraPt = _htVec3a, testPt = _htVec3b; let camera = cc.Camera.findCamera(this); if (camera) { camera.getScreenToWorldPoint(point, cameraPt); } else { cameraPt.set(point); } this._updateWorldMatrix(); // If scale is 0, it can't be hit. if (!Mat4.invert(_mat4_temp, this._worldMatrix)) { return false; } Vec2.transformMat4(testPt, cameraPt, _mat4_temp); testPt.x += this._anchorPoint.x * w; testPt.y += this._anchorPoint.y * h; let hit = false; if (testPt.x >= 0 && testPt.y >= 0 && testPt.x <= w && testPt.y <= h) { hit = true; if (listener && listener.mask) { let mask = listener.mask; let parent = this; let length = mask ? mask.length : 0; // find mask parent, should hit test it for (let i = 0, j = 0; parent && j < length; ++i, parent = parent.parent) { let temp = mask[j]; if (i === temp.index) { if (parent === temp.node) { let comp = parent.getComponent(cc.Mask); if (comp && comp._enabled && !comp._hitTest(cameraPt)) { hit = false; break } j++; } else { // mask parent no longer exists mask.length = j; break } } else if (i > temp.index) { // mask parent no longer exists mask.length = j; break } } } } return hit; }, /** * Get all the targets listening to the supplied type of event in the target's capturing phase. * The capturing phase comprises the journey from the root to the last node BEFORE the event target's node. * The result should save in the array parameter, and MUST SORT from child nodes to parent nodes. * * Subclasses can override this method to make event propagable. * @method _getCapturingTargets * @private * @param {String} type - the event type * @param {Array} array - the array to receive targets * @example {@link cocos2d/core/event/_getCapturingTargets.js} */ _getCapturingTargets (type, array) { var parent = this.parent; while (parent) { if (parent._capturingListeners && parent._capturingListeners.hasEventListener(type)) { array.push(parent); } parent = parent.parent; } }, /** * Get all the targets listening to the supplied type of event in the target's bubbling phase. * The bubbling phase comprises any SUBSEQUENT nodes encountered on the return trip to the root of the tree. * The result should save in the array parameter, and MUST SORT from child nodes to parent nodes. * * Subclasses can override this method to make event propagable. * @method _getBubblingTargets * @private * @param {String} type - the event type * @param {Array} array - the array to receive targets */ _getBubblingTargets (type, array) { var parent = this.parent; while (parent) { if (parent._bubblingListeners && parent._bubblingListeners.hasEventListener(type)) { array.push(parent); } parent = parent.parent; } }, // ACTIONS /** * !#en * Executes an action, and returns the action that is executed.
* The node becomes the action's target. Refer to cc.Action's getTarget()
* Calling runAction while the node is not active won't have any effect.
* Note:You shouldn't modify the action after runAction, that won't take any effect.
* if you want to modify, when you define action plus. * !#zh * 执行并返回该执行的动作。该节点将会变成动作的目标。
* 调用 runAction 时,节点自身处于不激活状态将不会有任何效果。
* 注意:你不应该修改 runAction 后的动作,将无法发挥作用,如果想进行修改,请在定义 action 时加入。 * @method runAction * @param {Action} action * @return {Action} An Action pointer * @example * var action = cc.scaleTo(0.2, 1, 0.6); * node.runAction(action); * node.runAction(action).repeatForever(); // fail * node.runAction(action.repeatForever()); // right */ runAction: ActionManagerExist ? function (action) { if (!this.active) return; cc.assertID(action, 1618); let am = cc.director.getActionManager(); if (!am._suppressDeprecation) { am._suppressDeprecation = true; cc.warnID(1639); } am.addAction(action, this, false); return action; } : emptyFunc, /** * !#en Pause all actions running on the current node. Equals to `cc.director.getActionManager().pauseTarget(node)`. * !#zh 暂停本节点上所有正在运行的动作。和 `cc.director.getActionManager().pauseTarget(node);` 等价。 * @method pauseAllActions * @example * node.pauseAllActions(); */ pauseAllActions: ActionManagerExist ? function () { cc.director.getActionManager().pauseTarget(this); } : emptyFunc, /** * !#en Resume all paused actions on the current node. Equals to `cc.director.getActionManager().resumeTarget(node)`. * !#zh 恢复运行本节点上所有暂停的动作。和 `cc.director.getActionManager().resumeTarget(node);` 等价。 * @method resumeAllActions * @example * node.resumeAllActions(); */ resumeAllActions: ActionManagerExist ? function () { cc.director.getActionManager().resumeTarget(this); } : emptyFunc, /** * !#en Stops and removes all actions from the running action list . * !#zh 停止并且移除所有正在运行的动作列表。 * @method stopAllActions * @example * node.stopAllActions(); */ stopAllActions: ActionManagerExist ? function () { cc.director.getActionManager().removeAllActionsFromTarget(this); } : emptyFunc, /** * !#en Stops and removes an action from the running action list. * !#zh 停止并移除指定的动作。 * @method stopAction * @param {Action} action An action object to be removed. * @example * var action = cc.scaleTo(0.2, 1, 0.6); * node.stopAction(action); */ stopAction: ActionManagerExist ? function (action) { cc.director.getActionManager().removeAction(action); } : emptyFunc, /** * !#en Removes an action from the running action list by its tag. * !#zh 停止并且移除指定标签的动作。 * @method stopActionByTag * @param {Number} tag A tag that indicates the action to be removed. * @example * node.stopActionByTag(1); */ stopActionByTag: ActionManagerExist ? function (tag) { if (tag === cc.Action.TAG_INVALID) { cc.logID(1612); return; } cc.director.getActionManager().removeActionByTag(tag, this); } : emptyFunc, /** * !#en Returns an action from the running action list by its tag. * !#zh 通过标签获取指定动作。 * @method getActionByTag * @see cc.Action#getTag and cc.Action#setTag * @param {Number} tag * @return {Action} The action object with the given tag. * @example * var action = node.getActionByTag(1); */ getActionByTag: ActionManagerExist ? function (tag) { if (tag === cc.Action.TAG_INVALID) { cc.logID(1613); return null; } return cc.director.getActionManager().getActionByTag(tag, this); } : function () { return null; }, /** * !#en * Returns the numbers of actions that are running plus the ones that are schedule to run (actions in actionsToAdd and actions arrays).
* Composable actions are counted as 1 action. Example:
* If you are running 1 Sequence of 7 actions, it will return 1.
* If you are running 7 Sequences of 2 actions, it will return 7.

* !#zh * 获取运行着的动作加上正在调度运行的动作的总数。
* 例如:
* - 如果你正在运行 7 个动作中的 1 个 Sequence,它将返回 1。
* - 如果你正在运行 2 个动作中的 7 个 Sequence,它将返回 7。
* * @method getNumberOfRunningActions * @return {Number} The number of actions that are running plus the ones that are schedule to run * @example * var count = node.getNumberOfRunningActions(); * cc.log("Running Action Count: " + count); */ getNumberOfRunningActions: ActionManagerExist ? function () { return cc.director.getActionManager().getNumberOfRunningActionsInTarget(this); } : function () { return 0; }, // TRANSFORM RELATED /** * !#en * Returns a copy of the position (x, y, z) of the node in its parent's coordinates. * You can pass a cc.Vec2 or cc.Vec3 as the argument to receive the return values. * !#zh * 获取节点在父节点坐标系中的位置(x, y, z)。 * 你可以传一个 cc.Vec2 或者 cc.Vec3 作为参数来接收返回值。 * @method getPosition * @param {Vec2|Vec3} [out] - The return value to receive position * @return {Vec2|Vec3} The position (x, y, z) of the node in its parent's coordinates * @example * cc.log("Node Position: " + node.getPosition()); */ getPosition (out) { out = out || new Vec3(); return Trs.toPosition(out, this._trs); }, /** * !#en * Sets the position (x, y, z) of the node in its parent's coordinates.
* Usually we use cc.v2(x, y) to compose cc.Vec2 object, in this case, position.z will become 0.
* and passing two numbers (x, y) is more efficient than passing cc.Vec2 object, in this case, position.z will remain unchanged. * For 3D node we can use cc.v3(x, y, z) to compose cc.Vec3 object,
* and passing three numbers (x, y, z) is more efficient than passing cc.Vec3 object. * !#zh * 设置节点在父节点坐标系中的位置。
* 可以通过下面的方式设置坐标点:
* 1. 传入 2 个数值 x, y (此时不会改变 position.z 的值)。
* 2. 传入 cc.v2(x, y) 类型为 cc.Vec2 的对象 (此时 position.z 的值将被设置为0)。 * 3. 对于 3D 节点可以传入 3 个数值 x, y, z。
* 4. 对于 3D 节点可以传入 cc.v3(x, y, z) 类型为 cc.Vec3 的对象。 * @method setPosition * @param {Vec2|Vec3|Number} x - X coordinate for position or the position object * @param {Number} [y] - Y coordinate for position * @param {Number} [z] - Z coordinate for position */ setPosition (newPosOrX, y, z) { let x; if (y === undefined) { x = newPosOrX.x; y = newPosOrX.y; z = newPosOrX.z; } else { x = newPosOrX; } let trs = this._trs; if (z === undefined) { z = trs[2]; } if (trs[0] === x && trs[1] === y && trs[2] === z) { return; } if (CC_EDITOR) { var oldPosition = new cc.Vec3(trs[0], trs[1], trs[2]); } trs[0] = x; trs[1] = y; trs[2] = z; this.setLocalDirty(LocalDirtyFlag.ALL_POSITION); !CC_NATIVERENDERER && (this._renderFlag |= RenderFlow.FLAG_WORLD_TRANSFORM); // fast check event if (this._eventMask & POSITION_ON) { if (CC_EDITOR) { this.emit(EventType.POSITION_CHANGED, oldPosition); } else { this.emit(EventType.POSITION_CHANGED); } } }, /** * !#en * Returns the scale factor of the node. * Need pass a cc.Vec2 or cc.Vec3 as the argument to receive the return values. * !#zh 获取节点的缩放,需要传一个 cc.Vec2 或者 cc.Vec3 作为参数来接收返回值。 * @method getScale * @param {Vec2|Vec3} out * @return {Vec2|Vec3} The scale factor * @example * cc.log("Node Scale: " + node.getScale(cc.v3())); */ getScale (out) { if (out !== undefined) { return Trs.toScale(out, this._trs); } else { cc.errorID(1400, 'cc.Node.getScale', 'cc.Node.scale or cc.Node.getScale(cc.Vec3)'); return this._trs[7]; } }, /** * !#en * Sets the scale of axis in local coordinates of the node. * You can operate 2 axis in 2D node, and 3 axis in 3D node. * When only (x, y) is passed, the value of scale.z will not be changed. * When a Vec2 is passed in, the value of scale.z will be set to 0. * !#zh * 设置节点在本地坐标系中坐标轴上的缩放比例。 * 2D 节点可以操作两个坐标轴,而 3D 节点可以操作三个坐标轴。 * 当只传入 (x, y) 时,scale.z 的值不会被改变。 * 当只传入 Vec2 对象时,scale.z 的值将被设置为0。 * @method setScale * @param {Number|Vec2|Vec3} x - scaleX or scale object * @param {Number} [y] * @param {Number} [z] * @example * node.setScale(cc.v2(2, 2)); // Notice: scaleZ will be 0 * node.setScale(cc.v3(2, 2, 2)); // for 3D node * node.setScale(2); */ setScale (newScaleOrX, y, z) { let x; // only one parameter, and it's a Vec2/Vec3: if (newScaleOrX && typeof newScaleOrX !== 'number') { x = newScaleOrX.x; y = newScaleOrX.y; z = newScaleOrX.z; } // only one parameter, and it's a number: else if (newScaleOrX !== undefined && y === undefined) { x = newScaleOrX; y = newScaleOrX; z = newScaleOrX; } // two or three paramters: else { x = newScaleOrX; } let trs = this._trs; if (z === undefined) { z = trs[9]; } if (trs[7] !== x || trs[8] !== y || trs[9] !== z) { trs[7] = x; trs[8] = y; trs[9] = z; this.setLocalDirty(LocalDirtyFlag.ALL_SCALE); !CC_NATIVERENDERER && (this._renderFlag |= RenderFlow.FLAG_TRANSFORM); if (this._eventMask & SCALE_ON) { this.emit(EventType.SCALE_CHANGED); } } }, /** * !#en * Get rotation of node (in quaternion). * Need pass a cc.Quat as the argument to receive the return values. * !#zh * 获取该节点的 quaternion 旋转角度,需要传一个 cc.Quat 作为参数来接收返回值。 * @method getRotation * @param {Quat} out * @return {Quat} Quaternion object represents the rotation */ getRotation (out) { if (out instanceof Quat) { return Trs.toRotation(out, this._trs); } else { if (CC_DEBUG) { cc.warn("`cc.Node.getRotation()` is deprecated since v2.1.0, please use `-cc.Node.angle` instead. (`this.node.getRotation()` -> `-this.node.angle`)"); } return -this.angle; } }, /** * !#en Set rotation of node (in quaternion). * !#zh 设置该节点的 quaternion 旋转角度。 * @method setRotation * @param {cc.Quat|Number} quat Quaternion object represents the rotation or the x value of quaternion * @param {Number} [y] y value of quternion * @param {Number} [z] z value of quternion * @param {Number} [w] w value of quternion */ setRotation (rotation, y, z, w) { if (typeof rotation === 'number' && y === undefined) { if (CC_DEBUG) { cc.warn("`cc.Node.setRotation(degree)` is deprecated since v2.1.0, please set `-cc.Node.angle` instead. (`this.node.setRotation(x)` -> `this.node.angle = -x`)"); } this.angle = -rotation; } else { let x = rotation; if (y === undefined) { x = rotation.x; y = rotation.y; z = rotation.z; w = rotation.w; } let trs = this._trs; if (trs[3] !== x || trs[4] !== y || trs[5] !== z || trs[6] !== w) { trs[3] = x; trs[4] = y; trs[5] = z; trs[6] = w; this.setLocalDirty(LocalDirtyFlag.ALL_ROTATION); if (this._eventMask & ROTATION_ON) { this.emit(EventType.ROTATION_CHANGED); } if (CC_EDITOR) { this._toEuler(); } } } }, /** * !#en * Returns a copy the untransformed size of the node.
* The contentSize remains the same no matter the node is scaled or rotated.
* All nodes has a size. Layer and Scene has the same size of the screen by default.
* !#zh 获取节点自身大小,不受该节点是否被缩放或者旋转的影响。 * @method getContentSize * @return {Size} The untransformed size of the node. * @example * cc.log("Content Size: " + node.getContentSize()); */ getContentSize () { return cc.size(this._contentSize.width, this._contentSize.height); }, /** * !#en * Sets the untransformed size of the node.
* The contentSize remains the same no matter the node is scaled or rotated.
* All nodes has a size. Layer and Scene has the same size of the screen. * !#zh 设置节点原始大小,不受该节点是否被缩放或者旋转的影响。 * @method setContentSize * @param {Size|Number} size - The untransformed size of the node or The untransformed size's width of the node. * @param {Number} [height] - The untransformed size's height of the node. * @example * node.setContentSize(cc.size(100, 100)); * node.setContentSize(100, 100); */ setContentSize (size, height) { var locContentSize = this._contentSize; var clone; if (height === undefined) { if (approx(size.width, locContentSize.width) && approx(size.height, locContentSize.height)) return; if (CC_EDITOR) { clone = cc.size(locContentSize.width, locContentSize.height); } locContentSize.width = size.width; locContentSize.height = size.height; } else { if (approx(size, locContentSize.width) && approx(height, locContentSize.height)) return; if (CC_EDITOR) { clone = cc.size(locContentSize.width, locContentSize.height); } locContentSize.width = size; locContentSize.height = height; } if (this._eventMask & SIZE_ON) { if (CC_EDITOR) { this.emit(EventType.SIZE_CHANGED, clone); } else { this.emit(EventType.SIZE_CHANGED); } } }, /** * !#en * Returns a copy of the anchor point.
* Anchor point is the point around which all transformations and positioning manipulations take place.
* It's like a pin in the node where it is "attached" to its parent.
* The anchorPoint is normalized, like a percentage. (0,0) means the bottom-left corner and (1,1) means the top-right corner.
* But you can use values higher than (1,1) and lower than (0,0) too.
* The default anchor point is (0.5,0.5), so it starts at the center of the node. * !#zh * 获取节点锚点,用百分比表示。
* 锚点应用于所有变换和坐标点的操作,它就像在节点上连接其父节点的大头针。
* 锚点是标准化的,就像百分比一样。(0,0) 表示左下角,(1,1) 表示右上角。
* 但是你可以使用比(1,1)更高的值或者比(0,0)更低的值。
* 默认的锚点是(0.5,0.5),因此它开始于节点的中心位置。
* 注意:Creator 中的锚点仅用于定位所在的节点,子节点的定位不受影响。 * @method getAnchorPoint * @return {Vec2} The anchor point of node. * @example * cc.log("Node AnchorPoint: " + node.getAnchorPoint()); */ getAnchorPoint () { return cc.v2(this._anchorPoint); }, /** * !#en * Sets the anchor point in percent.
* anchor point is the point around which all transformations and positioning manipulations take place.
* It's like a pin in the node where it is "attached" to its parent.
* The anchorPoint is normalized, like a percentage. (0,0) means the bottom-left corner and (1,1) means the top-right corner.
* But you can use values higher than (1,1) and lower than (0,0) too.
* The default anchor point is (0.5,0.5), so it starts at the center of the node. * !#zh * 设置锚点的百分比。
* 锚点应用于所有变换和坐标点的操作,它就像在节点上连接其父节点的大头针。
* 锚点是标准化的,就像百分比一样。(0,0) 表示左下角,(1,1) 表示右上角。
* 但是你可以使用比(1,1)更高的值或者比(0,0)更低的值。
* 默认的锚点是(0.5,0.5),因此它开始于节点的中心位置。
* 注意:Creator 中的锚点仅用于定位所在的节点,子节点的定位不受影响。 * @method setAnchorPoint * @param {Vec2|Number} point - The anchor point of node or The x axis anchor of node. * @param {Number} [y] - The y axis anchor of node. * @example * node.setAnchorPoint(cc.v2(1, 1)); * node.setAnchorPoint(1, 1); */ setAnchorPoint (point, y) { var locAnchorPoint = this._anchorPoint; if (y === undefined) { if ((point.x === locAnchorPoint.x) && (point.y === locAnchorPoint.y)) return; locAnchorPoint.x = point.x; locAnchorPoint.y = point.y; } else { if ((point === locAnchorPoint.x) && (y === locAnchorPoint.y)) return; locAnchorPoint.x = point; locAnchorPoint.y = y; } this.setLocalDirty(LocalDirtyFlag.ALL_POSITION); if (this._eventMask & ANCHOR_ON) { this.emit(EventType.ANCHOR_CHANGED); } }, /* * Transforms position from world space to local space. * @method _invTransformPoint * @param {Vec3} out * @param {Vec3} vec3 */ _invTransformPoint (out, pos) { if (this._parent) { this._parent._invTransformPoint(out, pos); } else { Vec3.copy(out, pos); } let ltrs = this._trs; // out = parent_inv_pos - pos Trs.toPosition(_tpVec3a, ltrs); Vec3.sub(out, out, _tpVec3a); // out = inv(rot) * out Trs.toRotation(_tpQuata, ltrs); Quat.conjugate(_tpQuatb, _tpQuata); Vec3.transformQuat(out, out, _tpQuatb); // out = (1/scale) * out Trs.toScale(_tpVec3a, ltrs); Vec3.inverseSafe(_tpVec3b, _tpVec3a); Vec3.mul(out, out, _tpVec3b); return out; }, /* * Calculate and return world position. * This is not a public API yet, its usage could be updated * @method getWorldPosition * @param {Vec3} out * @return {Vec3} */ getWorldPosition (out) { Trs.toPosition(out, this._trs); let curr = this._parent; let ltrs; while (curr) { ltrs = curr._trs; // out = parent_scale * pos Trs.toScale(_gwpVec3, ltrs); Vec3.mul(out, out, _gwpVec3); // out = parent_quat * out Trs.toRotation(_gwpQuat, ltrs); Vec3.transformQuat(out, out, _gwpQuat); // out = out + pos Trs.toPosition(_gwpVec3, ltrs); Vec3.add(out, out, _gwpVec3); curr = curr._parent; } return out; }, /* * Set world position. * This is not a public API yet, its usage could be updated * @method setWorldPosition * @param {Vec3} pos */ setWorldPosition (pos) { let ltrs = this._trs; if (CC_EDITOR) { var oldPosition = new cc.Vec3(ltrs[0], ltrs[1], ltrs[2]); } // NOTE: this is faster than invert world matrix and transform the point if (this._parent) { this._parent._invTransformPoint(_swpVec3, pos); } else { Vec3.copy(_swpVec3, pos); } Trs.fromPosition(ltrs, _swpVec3); this.setLocalDirty(LocalDirtyFlag.ALL_POSITION); // fast check event if (this._eventMask & POSITION_ON) { // send event if (CC_EDITOR) { this.emit(EventType.POSITION_CHANGED, oldPosition); } else { this.emit(EventType.POSITION_CHANGED); } } }, /* * Calculate and return world rotation * This is not a public API yet, its usage could be updated * @method getWorldRotation * @param {Quat} out * @return {Quat} */ getWorldRotation (out) { Trs.toRotation(_gwrQuat, this._trs); Quat.copy(out, _gwrQuat); let curr = this._parent; while (curr) { Trs.toRotation(_gwrQuat, curr._trs); Quat.mul(out, _gwrQuat, out); curr = curr._parent; } return out; }, /* * Set world rotation with quaternion * This is not a public API yet, its usage could be updated * @method setWorldRotation * @param {Quat} val */ setWorldRotation (val) { if (this._parent) { this._parent.getWorldRotation(_swrQuat); Quat.conjugate(_swrQuat, _swrQuat); Quat.mul(_swrQuat, _swrQuat, val); } else { Quat.copy(_swrQuat, val); } Trs.fromRotation(this._trs, _swrQuat); if (CC_EDITOR) { this._toEuler(); } this.setLocalDirty(LocalDirtyFlag.ALL_ROTATION); }, /* * Calculate and return world scale * This is not a public API yet, its usage could be updated * @method getWorldScale * @param {Vec3} out * @return {Vec3} */ getWorldScale (out) { Trs.toScale(_gwsVec3, this._trs); Vec3.copy(out, _gwsVec3); let curr = this._parent; while (curr) { Trs.toScale(_gwsVec3, curr._trs); Vec3.mul(out, out, _gwsVec3); curr = curr._parent; } return out; }, /* * Set world scale with vec3 * This is not a public API yet, its usage could be updated * @method setWorldScale * @param {Vec3} scale */ setWorldScale (scale) { if (this._parent) { this._parent.getWorldScale(_swsVec3); Vec3.div(_swsVec3, scale, _swsVec3); } else { Vec3.copy(_swsVec3, scale); } Trs.fromScale(this._trs, _swsVec3); this.setLocalDirty(LocalDirtyFlag.ALL_SCALE); }, getWorldRT (out) { let opos = _gwrtVec3a; let orot = _gwrtQuata; let ltrs = this._trs; Trs.toPosition(opos, ltrs); Trs.toRotation(orot, ltrs); let curr = this._parent; while (curr) { ltrs = curr._trs; // opos = parent_lscale * lpos Trs.toScale(_gwrtVec3b, ltrs); Vec3.mul(opos, opos, _gwrtVec3b); // opos = parent_lrot * opos Trs.toRotation(_gwrtQuatb, ltrs); Vec3.transformQuat(opos, opos, _gwrtQuatb); // opos = opos + lpos Trs.toPosition(_gwrtVec3b, ltrs); Vec3.add(opos, opos, _gwrtVec3b); // orot = lrot * orot Quat.mul(orot, _gwrtQuatb, orot); curr = curr._parent; } Mat4.fromRT(out, orot, opos); return out; }, /** * !#en Set rotation by lookAt target point, normally used by Camera Node * !#zh 通过观察目标来设置 rotation,一般用于 Camera Node 上 * @method lookAt * @param {Vec3} pos * @param {Vec3} [up] - default is (0,1,0) */ lookAt (pos, up) { this.getWorldPosition(_laVec3); Vec3.sub(_laVec3, _laVec3, pos); // NOTE: we use -z for view-dir Vec3.normalize(_laVec3, _laVec3); Quat.fromViewUp(_laQuat, _laVec3, up); this.setWorldRotation(_laQuat); }, _updateLocalMatrix: updateLocalMatrix2D, _calculWorldMatrix () { // Avoid as much function call as possible if (this._localMatDirty & LocalDirtyFlag.TRSS) { this._updateLocalMatrix(); } // Assume parent world matrix is correct let parent = this._parent; if (parent) { this._mulMat(this._worldMatrix, parent._worldMatrix, this._matrix); } else { Mat4.copy(this._worldMatrix, this._matrix); } this._worldMatDirty = false; }, _mulMat: mulMat2D, _updateWorldMatrix () { if (this._parent) { this._parent._updateWorldMatrix(); } if (this._worldMatDirty) { this._calculWorldMatrix(); // Sync dirty to children let children = this._children; for (let i = 0, l = children.length; i < l; i++) { children[i]._worldMatDirty = true; } } }, setLocalDirty (flag) { this._localMatDirty |= flag; this._worldMatDirty = true; if (flag === LocalDirtyFlag.ALL_POSITION || flag === LocalDirtyFlag.POSITION) { this._renderFlag |= RenderFlow.FLAG_WORLD_TRANSFORM; } else { this._renderFlag |= RenderFlow.FLAG_TRANSFORM; } }, setWorldDirty () { this._worldMatDirty = true; }, /** * !#en * Get the local transform matrix (4x4), based on parent node coordinates * !#zh 返回局部空间坐标系的矩阵,基于父节点坐标系。 * @method getLocalMatrix * @param {Mat4} out The matrix object to be filled with data * @return {Mat4} Same as the out matrix object * @example * let mat4 = cc.mat4(); * node.getLocalMatrix(mat4); */ getLocalMatrix (out) { this._updateLocalMatrix(); return Mat4.copy(out, this._matrix); }, /** * !#en * Get the world transform matrix (4x4) * !#zh 返回世界空间坐标系的矩阵。 * @method getWorldMatrix * @param {Mat4} out The matrix object to be filled with data * @return {Mat4} Same as the out matrix object * @example * let mat4 = cc.mat4(); * node.getWorldMatrix(mat4); */ getWorldMatrix (out) { this._updateWorldMatrix(); return Mat4.copy(out, this._worldMatrix); }, /** * !#en * Converts a Point to node (local) space coordinates. * !#zh * 将一个点转换到节点 (局部) 空间坐标系。 * @method convertToNodeSpaceAR * @param {Vec3|Vec2} worldPoint * @param {Vec3|Vec2} [out] * @return {Vec3|Vec2} * @typescript * convertToNodeSpaceAR(worldPoint: T, out?: T): T * @example * var newVec2 = node.convertToNodeSpaceAR(cc.v2(100, 100)); * var newVec3 = node.convertToNodeSpaceAR(cc.v3(100, 100, 100)); */ convertToNodeSpaceAR (worldPoint, out) { this._updateWorldMatrix(); Mat4.invert(_mat4_temp, this._worldMatrix); if (worldPoint instanceof cc.Vec2) { out = out || new cc.Vec2(); return Vec2.transformMat4(out, worldPoint, _mat4_temp); } else { out = out || new cc.Vec3(); return Vec3.transformMat4(out, worldPoint, _mat4_temp); } }, /** * !#en * Converts a Point in node coordinates to world space coordinates. * !#zh * 将节点坐标系下的一个点转换到世界空间坐标系。 * @method convertToWorldSpaceAR * @param {Vec3|Vec2} nodePoint * @param {Vec3|Vec2} [out] * @return {Vec3|Vec2} * @typescript * convertToWorldSpaceAR(nodePoint: T, out?: T): T * @example * var newVec2 = node.convertToWorldSpaceAR(cc.v2(100, 100)); * var newVec3 = node.convertToWorldSpaceAR(cc.v3(100, 100, 100)); */ convertToWorldSpaceAR (nodePoint, out) { this._updateWorldMatrix(); if (nodePoint instanceof cc.Vec2) { out = out || new cc.Vec2(); return Vec2.transformMat4(out, nodePoint, this._worldMatrix); } else { out = out || new cc.Vec3(); return Vec3.transformMat4(out, nodePoint, this._worldMatrix); } }, // OLD TRANSFORM ACCESS APIs /** * !#en Converts a Point to node (local) space coordinates then add the anchor point position. * So the return position will be related to the left bottom corner of the node's bounding box. * This equals to the API behavior of cocos2d-x, you probably want to use convertToNodeSpaceAR instead * !#zh 将一个点转换到节点 (局部) 坐标系,并加上锚点的坐标。
* 也就是说返回的坐标是相对于节点包围盒左下角的坐标。
* 这个 API 的设计是为了和 cocos2d-x 中行为一致,更多情况下你可能需要使用 convertToNodeSpaceAR。 * @method convertToNodeSpace * @deprecated since v2.1.3 * @param {Vec2} worldPoint * @return {Vec2} * @example * var newVec2 = node.convertToNodeSpace(cc.v2(100, 100)); */ convertToNodeSpace (worldPoint) { this._updateWorldMatrix(); Mat4.invert(_mat4_temp, this._worldMatrix); let out = new cc.Vec2(); Vec2.transformMat4(out, worldPoint, _mat4_temp); out.x += this._anchorPoint.x * this._contentSize.width; out.y += this._anchorPoint.y * this._contentSize.height; return out; }, /** * !#en Converts a Point related to the left bottom corner of the node's bounding box to world space coordinates. * This equals to the API behavior of cocos2d-x, you probably want to use convertToWorldSpaceAR instead * !#zh 将一个相对于节点左下角的坐标位置转换到世界空间坐标系。 * 这个 API 的设计是为了和 cocos2d-x 中行为一致,更多情况下你可能需要使用 convertToWorldSpaceAR * @method convertToWorldSpace * @deprecated since v2.1.3 * @param {Vec2} nodePoint * @return {Vec2} * @example * var newVec2 = node.convertToWorldSpace(cc.v2(100, 100)); */ convertToWorldSpace (nodePoint) { this._updateWorldMatrix(); let out = new cc.Vec2( nodePoint.x - this._anchorPoint.x * this._contentSize.width, nodePoint.y - this._anchorPoint.y * this._contentSize.height ); return Vec2.transformMat4(out, out, this._worldMatrix); }, /** * !#en * Returns the matrix that transform the node's (local) space coordinates into the parent's space coordinates.
* The matrix is in Pixels. * !#zh 返回这个将节点(局部)的空间坐标系转换成父节点的空间坐标系的矩阵。这个矩阵以像素为单位。 * @method getNodeToParentTransform * @deprecated since v2.0 * @param {AffineTransform} [out] The affine transform object to be filled with data * @return {AffineTransform} Same as the out affine transform object * @example * let affineTransform = cc.AffineTransform.create(); * node.getNodeToParentTransform(affineTransform); */ getNodeToParentTransform (out) { if (!out) { out = AffineTrans.identity(); } this._updateLocalMatrix(); var contentSize = this._contentSize; _vec3_temp.x = -this._anchorPoint.x * contentSize.width; _vec3_temp.y = -this._anchorPoint.y * contentSize.height; Mat4.copy(_mat4_temp, this._matrix); Mat4.transform(_mat4_temp, _mat4_temp, _vec3_temp); return AffineTrans.fromMat4(out, _mat4_temp); }, /** * !#en * Returns the matrix that transform the node's (local) space coordinates into the parent's space coordinates.
* The matrix is in Pixels.
* This method is AR (Anchor Relative). * !#zh * 返回这个将节点(局部)的空间坐标系转换成父节点的空间坐标系的矩阵。
* 这个矩阵以像素为单位。
* 该方法基于节点坐标。 * @method getNodeToParentTransformAR * @deprecated since v2.0 * @param {AffineTransform} [out] The affine transform object to be filled with data * @return {AffineTransform} Same as the out affine transform object * @example * let affineTransform = cc.AffineTransform.create(); * node.getNodeToParentTransformAR(affineTransform); */ getNodeToParentTransformAR (out) { if (!out) { out = AffineTrans.identity(); } this._updateLocalMatrix(); return AffineTrans.fromMat4(out, this._matrix); }, /** * !#en Returns the world affine transform matrix. The matrix is in Pixels. * !#zh 返回节点到世界坐标系的仿射变换矩阵。矩阵单位是像素。 * @method getNodeToWorldTransform * @deprecated since v2.0 * @param {AffineTransform} [out] The affine transform object to be filled with data * @return {AffineTransform} Same as the out affine transform object * @example * let affineTransform = cc.AffineTransform.create(); * node.getNodeToWorldTransform(affineTransform); */ getNodeToWorldTransform (out) { if (!out) { out = AffineTrans.identity(); } this._updateWorldMatrix(); var contentSize = this._contentSize; _vec3_temp.x = -this._anchorPoint.x * contentSize.width; _vec3_temp.y = -this._anchorPoint.y * contentSize.height; Mat4.copy(_mat4_temp, this._worldMatrix); Mat4.transform(_mat4_temp, _mat4_temp, _vec3_temp); return AffineTrans.fromMat4(out, _mat4_temp); }, /** * !#en * Returns the world affine transform matrix. The matrix is in Pixels.
* This method is AR (Anchor Relative). * !#zh * 返回节点到世界坐标仿射变换矩阵。矩阵单位是像素。
* 该方法基于节点坐标。 * @method getNodeToWorldTransformAR * @deprecated since v2.0 * @param {AffineTransform} [out] The affine transform object to be filled with data * @return {AffineTransform} Same as the out affine transform object * @example * let affineTransform = cc.AffineTransform.create(); * node.getNodeToWorldTransformAR(affineTransform); */ getNodeToWorldTransformAR (out) { if (!out) { out = AffineTrans.identity(); } this._updateWorldMatrix(); return AffineTrans.fromMat4(out, this._worldMatrix); }, /** * !#en * Returns the matrix that transform parent's space coordinates to the node's (local) space coordinates.
* The matrix is in Pixels. The returned transform is readonly and cannot be changed. * !#zh * 返回将父节点的坐标系转换成节点(局部)的空间坐标系的矩阵。
* 该矩阵以像素为单位。返回的矩阵是只读的,不能更改。 * @method getParentToNodeTransform * @deprecated since v2.0 * @param {AffineTransform} [out] The affine transform object to be filled with data * @return {AffineTransform} Same as the out affine transform object * @example * let affineTransform = cc.AffineTransform.create(); * node.getParentToNodeTransform(affineTransform); */ getParentToNodeTransform (out) { if (!out) { out = AffineTrans.identity(); } this._updateLocalMatrix(); Mat4.invert(_mat4_temp, this._matrix); return AffineTrans.fromMat4(out, _mat4_temp); }, /** * !#en Returns the inverse world affine transform matrix. The matrix is in Pixels. * !#en 返回世界坐标系到节点坐标系的逆矩阵。 * @method getWorldToNodeTransform * @deprecated since v2.0 * @param {AffineTransform} [out] The affine transform object to be filled with data * @return {AffineTransform} Same as the out affine transform object * @example * let affineTransform = cc.AffineTransform.create(); * node.getWorldToNodeTransform(affineTransform); */ getWorldToNodeTransform (out) { if (!out) { out = AffineTrans.identity(); } this._updateWorldMatrix(); Mat4.invert(_mat4_temp, this._worldMatrix); return AffineTrans.fromMat4(out, _mat4_temp); }, /** * !#en convenience methods which take a cc.Touch instead of cc.Vec2. * !#zh 将触摸点转换成本地坐标系中位置。 * @method convertTouchToNodeSpace * @deprecated since v2.0 * @param {Touch} touch - The touch object * @return {Vec2} * @example * var newVec2 = node.convertTouchToNodeSpace(touch); */ convertTouchToNodeSpace (touch) { return this.convertToNodeSpace(touch.getLocation()); }, /** * !#en converts a cc.Touch (world coordinates) into a local coordinate. This method is AR (Anchor Relative). * !#zh 转换一个 cc.Touch(世界坐标)到一个局部坐标,该方法基于节点坐标。 * @method convertTouchToNodeSpaceAR * @deprecated since v2.0 * @param {Touch} touch - The touch object * @return {Vec2} * @example * var newVec2 = node.convertTouchToNodeSpaceAR(touch); */ convertTouchToNodeSpaceAR (touch) { return this.convertToNodeSpaceAR(touch.getLocation()); }, /** * !#en * Returns a "local" axis aligned bounding box of the node.
* The returned box is relative only to its parent. * !#zh 返回父节坐标系下的轴向对齐的包围盒。 * @method getBoundingBox * @return {Rect} The calculated bounding box of the node * @example * var boundingBox = node.getBoundingBox(); */ getBoundingBox () { this._updateLocalMatrix(); let width = this._contentSize.width; let height = this._contentSize.height; let rect = cc.rect( -this._anchorPoint.x * width, -this._anchorPoint.y * height, width, height); return rect.transformMat4(rect, this._matrix); }, /** * !#en * Returns a "world" axis aligned bounding box of the node.
* The bounding box contains self and active children's world bounding box. * !#zh * 返回节点在世界坐标系下的对齐轴向的包围盒(AABB)。
* 该边框包含自身和已激活的子节点的世界边框。 * @method getBoundingBoxToWorld * @return {Rect} * @example * var newRect = node.getBoundingBoxToWorld(); */ getBoundingBoxToWorld () { if (this._parent) { this._parent._updateWorldMatrix(); return this._getBoundingBoxTo(); } else { return this.getBoundingBox(); } }, _getBoundingBoxTo () { let width = this._contentSize.width; let height = this._contentSize.height; let rect = cc.rect( -this._anchorPoint.x * width, -this._anchorPoint.y * height, width, height); this._calculWorldMatrix(); rect.transformMat4(rect, this._worldMatrix); //query child's BoundingBox if (!this._children) return rect; var locChildren = this._children; for (var i = 0; i < locChildren.length; i++) { var child = locChildren[i]; if (child && child.active) { var childRect = child._getBoundingBoxTo(); if (childRect) rect.union(rect, childRect); } } return rect; }, _updateOrderOfArrival () { var arrivalOrder = this._parent ? ++this._parent._childArrivalOrder : 0; this._localZOrder = (this._localZOrder & 0xffff0000) | arrivalOrder; this.emit(EventType.SIBLING_ORDER_CHANGED); }, /** * !#en * Set Group index of node without children.
* Which Group this node belongs to will resolve that this node's collision components can collide with which other collision componentns.
* !#zh * 设置节点本身的分组索引。不影响子节点
* 节点的分组将关系到节点的碰撞组件可以与哪些碰撞组件相碰撞。
* @property groupIndex * @type {Integer} * @default 0 */ setSelfGroupIndex (groupIndex) { this._groupIndex = groupIndex || 0; this._cullingMask = 1 << groupIndex; if (CC_JSB && CC_NATIVERENDERER) { this._proxy && this._proxy.updateCullingMask(); } }, /** * !#en * Adds a child to the node with z order and name. * !#zh * 添加子节点,并且可以修改该节点的 局部 Z 顺序和名字。 * @method addChild * @param {Node} child - A child node * @param {Number} [zIndex] - Z order for drawing priority. Please refer to zIndex property * @param {String} [name] - A name to identify the node easily. Please refer to name property * @example * node.addChild(newNode, 1, "node"); */ addChild (child, zIndex, name) { if (CC_DEV && !cc.Node.isNode(child)) { return cc.errorID(1634, cc.js.getClassName(child)); } cc.assertID(child, 1606); cc.assertID(child._parent === null, 1605); // invokes the parent setter child.parent = this; if (zIndex !== undefined) { child.zIndex = zIndex; } if (name !== undefined) { child.name = name; } }, /** * !#en Stops all running actions and schedulers. * !#zh 停止所有正在播放的动作和计时器。 * @method cleanup * @example * node.cleanup(); */ cleanup () { // actions ActionManagerExist && cc.director.getActionManager().removeAllActionsFromTarget(this); // event eventManager.removeListeners(this); // children var i, len = this._children.length, node; for (i = 0; i < len; ++i) { node = this._children[i]; if (node) node.cleanup(); } }, /** * !#en Sorts the children array depends on children's zIndex and arrivalOrder, * normally you won't need to invoke this function. * !#zh 根据子节点的 zIndex 和 arrivalOrder 进行排序,正常情况下开发者不需要手动调用这个函数。 * * @method sortAllChildren */ sortAllChildren () { if (this._reorderChildDirty) { this._reorderChildDirty = false; // delay update arrivalOrder before sort children var _children = this._children, child; // reset arrivalOrder before sort children this._childArrivalOrder = 1; for (let i = 0, len = _children.length; i < len; i++) { child = _children[i]; child._updateOrderOfArrival(); } // Optimize reordering event code to fix problems with setting zindex // https://github.com/cocos-creator/2d-tasks/issues/1186 eventManager._setDirtyForNode(this); if (_children.length > 1) { // insertion sort let child, child2; for (let i = 1, count = _children.length; i < count; i++) { child = _children[i]; let j = i; for (; j > 0 && (child2 = _children[j - 1])._localZOrder > child._localZOrder; j--) { _children[j] = child2; } _children[j] = child; } this.emit(EventType.CHILD_REORDER, this); } cc.director.__fastOff(cc.Director.EVENT_AFTER_UPDATE, this.sortAllChildren, this); } }, _delaySort () { if (!this._reorderChildDirty) { this._reorderChildDirty = true; cc.director.__fastOn(cc.Director.EVENT_AFTER_UPDATE, this.sortAllChildren, this); } }, _restoreProperties: CC_EDITOR && function () { /* * TODO: Refine this code after completing undo/redo 2.0. * The node will be destroyed when deleting in the editor, * but it will be reserved and reused for undo. */ // restore 3d node this.is3DNode = this.is3DNode; if (!this._matrix) { this._matrix = cc.mat4(this._spaceInfo.localMat); Mat4.identity(this._matrix); } if (!this._worldMatrix) { this._worldMatrix = cc.mat4(this._spaceInfo.worldMat); Mat4.identity(this._worldMatrix); } this._localMatDirty = LocalDirtyFlag.ALL; this._worldMatDirty = true; this._fromEuler(); this._renderFlag |= RenderFlow.FLAG_TRANSFORM | RenderFlow.FLAG_OPACITY_COLOR; if (this._renderComponent) { this._renderComponent.markForRender(true); } if (this._children.length > 0) { this._renderFlag |= RenderFlow.FLAG_CHILDREN; } }, onRestore: CC_EDITOR && function () { this._onRestoreBase(); this.emit(EventType.GROUP_CHANGED, this); this.emit(EventType.POSITION_CHANGED, this.position.clone()); this.emit(EventType.SIZE_CHANGED, this._contentSize.clone()); this.emit(EventType.ROTATION_CHANGED); this.emit(EventType.SCALE_CHANGED) this.emit(EventType.COLOR_CHANGED, this._color.clone()); this.emit(EventType.ANCHOR_CHANGED); this._restoreProperties(); var actionManager = cc.director.getActionManager(); if (this._activeInHierarchy) { actionManager && actionManager.resumeTarget(this); eventManager.resumeTarget(this); } else { actionManager && actionManager.pauseTarget(this); eventManager.pauseTarget(this); } }, }; if (CC_EDITOR) { // deprecated, only used to import old data in editor js.mixin(NodeDefines.properties, { _scaleX: { default: undefined, type: cc.Float, editorOnly: true }, _scaleY: { default: undefined, type: cc.Float, editorOnly: true }, }); } let Node = cc.Class(NodeDefines); // 3D Node Property // Node Event /** * !#en * The position changing event, you can listen to this event through the statement this.node.on(cc.Node.EventType.POSITION_CHANGED, callback, this); * !#zh * 位置变动监听事件, 通过 this.node.on(cc.Node.EventType.POSITION_CHANGED, callback, this); 进行监听。 * @event position-changed * @param {Vec2} oldPos - The old position, but this parameter is only available in editor! */ /** * !#en * The size changing event, you can listen to this event through the statement this.node.on(cc.Node.EventType.SIZE_CHANGED, callback, this); * !#zh * 尺寸变动监听事件,通过 this.node.on(cc.Node.EventType.SIZE_CHANGED, callback, this); 进行监听。 * @event size-changed * @param {Size} oldSize - The old size, but this parameter is only available in editor! */ /** * !#en * The anchor changing event, you can listen to this event through the statement this.node.on(cc.Node.EventType.ANCHOR_CHANGED, callback, this); * !#zh * 锚点变动监听事件,通过 this.node.on(cc.Node.EventType.ANCHOR_CHANGED, callback, this); 进行监听。 * @event anchor-changed */ /** * !#en * The adding child event, you can listen to this event through the statement this.node.on(cc.Node.EventType.CHILD_ADDED, callback, this); * !#zh * 增加子节点监听事件,通过 this.node.on(cc.Node.EventType.CHILD_ADDED, callback, this); 进行监听。 * @event child-added * @param {Node} child - child which have been added */ /** * !#en * The removing child event, you can listen to this event through the statement this.node.on(cc.Node.EventType.CHILD_REMOVED, callback, this); * !#zh * 删除子节点监听事件,通过 this.node.on(cc.Node.EventType.CHILD_REMOVED, callback, this); 进行监听。 * @event child-removed * @param {Node} child - child which have been removed */ /** * !#en * The reordering child event, you can listen to this event through the statement this.node.on(cc.Node.EventType.CHILD_REORDER, callback, this); * !#zh * 子节点顺序变动监听事件,通过 this.node.on(cc.Node.EventType.CHILD_REORDER, callback, this); 进行监听。 * @event child-reorder * @param {Node} node - node whose children have been reordered */ /** * !#en * The group changing event, you can listen to this event through the statement this.node.on(cc.Node.EventType.GROUP_CHANGED, callback, this); * !#zh * 节点分组变动监听事件,通过 this.node.on(cc.Node.EventType.GROUP_CHANGED, callback, this); 进行监听。 * @event group-changed * @param {Node} node - node whose group has changed */ // Deprecated APIs /** * !#en * Returns the displayed opacity of Node, * the difference between displayed opacity and opacity is that displayed opacity is calculated based on opacity and parent node's opacity when cascade opacity enabled. * !#zh * 获取节点显示透明度, * 显示透明度和透明度之间的不同之处在于当启用级连透明度时, * 显示透明度是基于自身透明度和父节点透明度计算的。 * * @method getDisplayedOpacity * @return {number} displayed opacity * @deprecated since v2.0, please use opacity property, cascade opacity is removed */ /** * !#en * Returns the displayed color of Node, * the difference between displayed color and color is that displayed color is calculated based on color and parent node's color when cascade color enabled. * !#zh * 获取节点的显示颜色, * 显示颜色和颜色之间的不同之处在于当启用级连颜色时, * 显示颜色是基于自身颜色和父节点颜色计算的。 * * @method getDisplayedColor * @return {Color} * @deprecated since v2.0, please use color property, cascade color is removed */ /** * !#en Cascade opacity is removed from v2.0 * Indicate whether node's opacity value affect its child nodes, default value is true. * !#zh 透明度级联功能从 v2.0 开始已移除 * 节点的不透明度值是否影响其子节点,默认值为 true。 * @property cascadeOpacity * @deprecated since v2.0 * @type {Boolean} */ /** * !#en Cascade opacity is removed from v2.0 * Returns whether node's opacity value affect its child nodes. * !#zh 透明度级联功能从 v2.0 开始已移除 * 返回节点的不透明度值是否影响其子节点。 * @method isCascadeOpacityEnabled * @deprecated since v2.0 * @return {Boolean} */ /** * !#en Cascade opacity is removed from v2.0 * Enable or disable cascade opacity, if cascade enabled, child nodes' opacity will be the multiplication of parent opacity and its own opacity. * !#zh 透明度级联功能从 v2.0 开始已移除 * 启用或禁用级连不透明度,如果级连启用,子节点的不透明度将是父不透明度乘上它自己的不透明度。 * @method setCascadeOpacityEnabled * @deprecated since v2.0 * @param {Boolean} cascadeOpacityEnabled */ /** * !#en Opacity modify RGB have been removed since v2.0 * Set whether color should be changed with the opacity value, * useless in ccsg.Node, but this function is override in some class to have such behavior. * !#zh 透明度影响颜色配置已经被废弃 * 设置更改透明度时是否修改RGB值, * @method setOpacityModifyRGB * @deprecated since v2.0 * @param {Boolean} opacityValue */ /** * !#en Opacity modify RGB have been removed since v2.0 * Get whether color should be changed with the opacity value. * !#zh 透明度影响颜色配置已经被废弃 * 获取更改透明度时是否修改RGB值。 * @method isOpacityModifyRGB * @deprecated since v2.0 * @return {Boolean} */ let _p = Node.prototype; js.getset(_p, 'position', _p.getPosition, _p.setPosition, false, true); if (CC_EDITOR) { let vec3_tmp = new Vec3(); cc.js.getset(_p, 'worldEulerAngles', function () { let angles = new Vec3(this._eulerAngles); let parent = this.parent; while (parent) { angles.addSelf(parent._eulerAngles); parent = parent.parent; } return angles; }, function (v) { vec3_tmp.set(v); let parent = this.parent; while (parent) { vec3_tmp.subSelf(parent._eulerAngles); parent = parent.parent; } this.eulerAngles = vec3_tmp; }); } cc.Node = module.exports = Node;