1477 lines
49 KiB
JavaScript
Raw Permalink Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

/****************************************************************************
Copyright (c) 2013-2016 Chukong Technologies Inc.
Copyright (c) 2017-2018 Xiamen Yaji Software Co., Ltd.
https://www.cocos.com/
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated engine source code (the "Software"), a limited,
worldwide, royalty-free, non-assignable, revocable and non-exclusive license
to use Cocos Creator solely to develop games on your target platforms. You shall
not use Cocos Creator software for developing other software or tools that's
used for developing games. You are not granted to publish, distribute,
sublicense, and/or sell copies of Cocos Creator.
The software or tools in this License Agreement are licensed, not sold.
Xiamen Yaji Software Co., Ltd. reserves all rights not expressly granted to you.
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.
****************************************************************************/
const TrackEntryListeners = require('./track-entry-listeners');
const RenderComponent = require('../../cocos2d/core/components/CCRenderComponent');
const spine = require('./lib/spine');
const Graphics = require('../../cocos2d/core/graphics/graphics');
const RenderFlow = require('../../cocos2d/core/renderer/render-flow');
const FLAG_POST_RENDER = RenderFlow.FLAG_POST_RENDER;
let SkeletonCache = require('./skeleton-cache');
let AttachUtil = require('./AttachUtil');
/**
* @module sp
*/
let DefaultSkinsEnum = cc.Enum({ 'default': -1 });
let DefaultAnimsEnum = cc.Enum({ '<None>': 0 });
/**
* !#en Enum for animation cache mode type.
* !#zh Spine动画缓存类型
* @enum Skeleton.AnimationCacheMode
*/
let AnimationCacheMode = cc.Enum({
/**
* !#en The realtime mode.
* !#zh 实时计算模式。
* @property {Number} REALTIME
*/
REALTIME: 0,
/**
* !#en The shared cache mode.
* !#zh 共享缓存模式。
* @property {Number} SHARED_CACHE
*/
SHARED_CACHE: 1,
/**
* !#en The private cache mode.
* !#zh 私有缓存模式。
* @property {Number} PRIVATE_CACHE
*/
PRIVATE_CACHE: 2
});
function setEnumAttr (obj, propName, enumDef) {
cc.Class.Attr.setClassAttr(obj, propName, 'type', 'Enum');
cc.Class.Attr.setClassAttr(obj, propName, 'enumList', cc.Enum.getList(enumDef));
}
/**
* !#en
* The skeleton of Spine <br/>
* <br/>
* (Skeleton has a reference to a SkeletonData and stores the state for skeleton instance,
* which consists of the current pose's bone SRT, slot colors, and which slot attachments are visible. <br/>
* Multiple skeletons can use the same SkeletonData which includes all animations, skins, and attachments.) <br/>
* !#zh
* Spine 骨骼动画 <br/>
* <br/>
* (Skeleton 具有对骨骼数据的引用并且存储了骨骼实例的状态,
* 它由当前的骨骼动作slot 颜色,和可见的 slot attachments 组成。<br/>
* 多个 Skeleton 可以使用相同的骨骼数据,其中包括所有的动画,皮肤和 attachments。
*
* @class Skeleton
* @extends RenderComponent
*/
sp.Skeleton = cc.Class({
name: 'sp.Skeleton',
extends: RenderComponent,
editor: CC_EDITOR && {
menu: 'i18n:MAIN_MENU.component.renderers/Spine Skeleton',
help: 'app://docs/html/components/spine.html',
inspector: 'packages://inspector/inspectors/comps/skeleton2d.js',
},
statics: {
AnimationCacheMode: AnimationCacheMode,
},
properties: {
/**
* !#en The skeletal animation is paused?
* !#zh 该骨骼动画是否暂停。
* @property paused
* @type {Boolean}
* @readOnly
* @default false
*/
paused: {
default: false,
visible: false
},
/**
* !#en
* The skeleton data contains the skeleton information (bind pose bones, slots, draw order,
* attachments, skins, etc) and animations but does not hold any state.<br/>
* Multiple skeletons can share the same skeleton data.
* !#zh
* 骨骼数据包含了骨骼信息绑定骨骼动作slots渲染顺序
* attachments皮肤等等和动画但不持有任何状态。<br/>
* 多个 Skeleton 可以共用相同的骨骼数据。
* @property {sp.SkeletonData} skeletonData
*/
skeletonData: {
default: null,
type: sp.SkeletonData,
notify () {
if (CC_EDITOR) {
this._resetDefaultAnim();
this._resetDefaultSkin();
this._refreshInspector();
} else {
this.defaultSkin = '';
this.defaultAnimation = '';
}
this._updateSkeletonData();
},
tooltip: CC_DEV && 'i18n:COMPONENT.skeleton.skeleton_data'
},
// 由于 spine 的 skin 是无法二次替换的,所以只能设置默认的 skin
/**
* !#en The name of default skin.
* !#zh 默认的皮肤名称。
* @property {String} defaultSkin
*/
defaultSkin: {
default: '',
visible: false
},
/**
* !#en The name of default animation.
* !#zh 默认的动画名称。
* @property {String} defaultAnimation
*/
defaultAnimation: {
default: '',
visible: false
},
/**
* !#en The name of current playing animation.
* !#zh 当前播放的动画名称。
* @property {String} animation
*/
animation: {
get () {
if (this.isAnimationCached()) {
return this._animationName;
} else {
var entry = this.getCurrent(0);
return (entry && entry.animation.name) || "";
}
},
set (value) {
this.defaultAnimation = value;
if (value) {
this.setAnimation(0, value, this.loop);
}
else if (!this.isAnimationCached()) {
this.clearTrack(0);
this.setToSetupPose();
}
},
visible: false
},
/**
* @property {Number} _defaultSkinIndex
*/
_defaultSkinIndex: {
get () {
if (this.skeletonData) {
var skinsEnum = this.skeletonData.getSkinsEnum();
if(skinsEnum) {
if(this.defaultSkin === "") {
if(skinsEnum.hasOwnProperty(0)) {
this._defaultSkinIndex = 0;
return 0;
}
} else {
var skinIndex = skinsEnum[this.defaultSkin];
if (skinIndex !== undefined) {
return skinIndex;
}
}
}
}
return 0;
},
set (value) {
var skinsEnum;
if (this.skeletonData) {
skinsEnum = this.skeletonData.getSkinsEnum();
}
if ( !skinsEnum ) {
return cc.errorID('',
this.name);
}
var skinName = skinsEnum[value];
if (skinName !== undefined) {
this.defaultSkin = skinName;
this.setSkin(this.defaultSkin);
if (CC_EDITOR && !cc.engine.isPlaying) {
this._refreshInspector();
}
}
else {
cc.errorID(7501, this.name);
}
},
type: DefaultSkinsEnum,
visible: true,
animatable: false,
displayName: "Default Skin",
tooltip: CC_DEV && 'i18n:COMPONENT.skeleton.default_skin'
},
// value of 0 represents no animation
_animationIndex: {
get () {
var animationName = (!CC_EDITOR || cc.engine.isPlaying) ? this.animation : this.defaultAnimation;
if (this.skeletonData && animationName) {
var animsEnum = this.skeletonData.getAnimsEnum();
if (animsEnum) {
var animIndex = animsEnum[animationName];
if (animIndex !== undefined) {
return animIndex;
}
}
}
return 0;
},
set (value) {
if (value === 0) {
this.animation = '';
return;
}
var animsEnum;
if (this.skeletonData) {
animsEnum = this.skeletonData.getAnimsEnum();
}
if ( !animsEnum ) {
return cc.errorID(7502, this.name);
}
var animName = animsEnum[value];
if (animName !== undefined) {
this.animation = animName;
}
else {
cc.errorID(7503, this.name);
}
},
type: DefaultAnimsEnum,
visible: true,
animatable: false,
displayName: 'Animation',
tooltip: CC_DEV && 'i18n:COMPONENT.skeleton.animation'
},
// Record pre cache mode.
_preCacheMode: -1,
_cacheMode: AnimationCacheMode.REALTIME,
_defaultCacheMode: {
default: 0,
type: AnimationCacheMode,
notify () {
this.setAnimationCacheMode(this._defaultCacheMode);
},
editorOnly: true,
visible: true,
animatable: false,
displayName: "Animation Cache Mode",
tooltip: CC_DEV && 'i18n:COMPONENT.skeleton.animation_cache_mode'
},
/**
* !#en TODO
* !#zh 是否循环播放当前骨骼动画。
* @property {Boolean} loop
* @default true
*/
loop: {
default: true,
tooltip: CC_DEV && 'i18n:COMPONENT.skeleton.loop'
},
/**
* !#en Indicates whether to enable premultiplied alpha.
* You should disable this option when image's transparent area appears to have opaque pixels,
* or enable this option when image's half transparent area appears to be darken.
* !#zh 是否启用贴图预乘。
* 当图片的透明区域出现色块时需要关闭该选项,当图片的半透明区域颜色变黑时需要启用该选项。
* @property {Boolean} premultipliedAlpha
* @default true
*/
premultipliedAlpha: {
default: true,
tooltip: CC_DEV && 'i18n:COMPONENT.skeleton.premultipliedAlpha'
},
/**
* !#en The time scale of this skeleton.
* !#zh 当前骨骼中所有动画的时间缩放率。
* @property {Number} timeScale
* @default 1
*/
timeScale: {
default: 1,
tooltip: CC_DEV && 'i18n:COMPONENT.skeleton.time_scale'
},
/**
* !#en Indicates whether open debug slots.
* !#zh 是否显示 slot 的 debug 信息。
* @property {Boolean} debugSlots
* @default false
*/
debugSlots: {
default: false,
editorOnly: true,
tooltip: CC_DEV && 'i18n:COMPONENT.skeleton.debug_slots',
notify () {
this._updateDebugDraw();
}
},
/**
* !#en Indicates whether open debug bones.
* !#zh 是否显示 bone 的 debug 信息。
* @property {Boolean} debugBones
* @default false
*/
debugBones: {
default: false,
editorOnly: true,
tooltip: CC_DEV && 'i18n:COMPONENT.skeleton.debug_bones',
notify () {
this._updateDebugDraw();
}
},
/**
* !#en Indicates whether open debug mesh.
* !#zh 是否显示 mesh 的 debug 信息。
* @property {Boolean} debugMesh
* @default false
*/
debugMesh: {
default: false,
editorOnly: true,
tooltip: CC_DEV && 'i18n:COMPONENT.skeleton.debug_mesh',
notify () {
this._updateDebugDraw();
}
},
/**
* !#en Enabled two color tint.
* !#zh 是否启用染色效果。
* @property {Boolean} useTint
* @default false
*/
useTint: {
default: false,
tooltip: CC_DEV && 'i18n:COMPONENT.skeleton.use_tint',
notify () {
this._updateUseTint();
}
},
/**
* !#en Enabled batch model, if skeleton is complex, do not enable batch, or will lower performance.
* !#zh 开启合批如果渲染大量相同纹理且结构简单的骨骼动画开启合批可以降低drawcall否则请不要开启cpu消耗会上升。
* @property {Boolean} enableBatch
* @default false
*/
enableBatch: {
default: false,
notify () {
this._updateBatch();
},
tooltip: CC_DEV && 'i18n:COMPONENT.skeleton.enabled_batch'
},
// Below properties will effect when cache mode is SHARED_CACHE or PRIVATE_CACHE.
// accumulate time
_accTime: 0,
// Play times counter
_playCount: 0,
// Frame cache
_frameCache: null,
// Cur frame
_curFrame: null,
// Skeleton cache
_skeletonCache : null,
// Aimation name
_animationName : "",
// Animation queue
_animationQueue : [],
// Head animation info of
_headAniInfo : null,
// Play times
_playTimes : 0,
// Is animation complete.
_isAniComplete: true,
autoSwitchMaterial: {
type: RenderComponent.EnableType,
default: RenderComponent.EnableType.GLOBAL,
notify(oldValue) {
if (this.autoSwitchMaterial === oldValue) return;
this.setVertsDirty();
},
},
allowDynamicAtlas: {
type: RenderComponent.EnableType,
default: RenderComponent.EnableType.GLOBAL,
notify(oldValue) {
if (this.allowDynamicAtlas === oldValue) return;
this.setVertsDirty();
},
},
},
// CONSTRUCTOR
ctor () {
this._effectDelegate = null;
this._skeleton = null;
this._rootBone = null;
this._listener = null;
this._materialCache = {};
this._debugRenderer = null;
this._startSlotIndex = -1;
this._endSlotIndex = -1;
this._startEntry = {animation : {name : ""}, trackIndex : 0};
this._endEntry = {animation : {name : ""}, trackIndex : 0};
this.attachUtil = new AttachUtil();
this._dataDirty = true;
},
setVertsDirty() {
this._dataDirty = true;
this.invalidAnimationCache();
this._materialCache = {};
this._super();
},
// override base class _getDefaultMaterial to modify default material
_getDefaultMaterial () {
return cc.Material.getBuiltinMaterial('2d-spine');
},
// override base class _updateMaterial to set define value and clear material cache
_updateMaterial () {
let useTint = this.useTint;
let baseMaterial = this.getMaterial(0);
if (baseMaterial) {
const isMultiSupport = baseMaterial.material.isMultiSupport();
if (isMultiSupport) {
this._defineMaterialTint(baseMaterial, useTint);
} else {
baseMaterial.define('USE_TINT', useTint);
baseMaterial.define('CC_USE_MODEL', !this.enableBatch);
}
let srcBlendFactor = this.premultipliedAlpha ? cc.gfx.BLEND_ONE : cc.gfx.BLEND_SRC_ALPHA;
let dstBlendFactor = cc.gfx.BLEND_ONE_MINUS_SRC_ALPHA;
baseMaterial.setBlend(
true,
cc.gfx.BLEND_FUNC_ADD,
srcBlendFactor, srcBlendFactor,
cc.gfx.BLEND_FUNC_ADD,
dstBlendFactor, dstBlendFactor
);
if (isMultiSupport) {
if (!this.enableBatch) this.enableBatch = true;
}
if (CC_JSB) {
if (this._nativeSkeleton) {
this._nativeSkeleton.setUseMulti(isMultiSupport);
}
}
}
this._materialCache = {};
},
// override base class disableRender to clear post render flag
disableRender () {
this._super();
this.node._renderFlag &= ~FLAG_POST_RENDER;
},
// override base class disableRender to add post render flag
markForRender (enable) {
this._super(enable);
if (enable) {
this.node._renderFlag |= FLAG_POST_RENDER;
} else {
this.node._renderFlag &= ~FLAG_POST_RENDER;
}
},
// if change use tint mode, just clear material cache
_updateUseTint () {
let baseMaterial = this.getMaterial(0);
if (baseMaterial) {
let useTint = this.useTint;
if (baseMaterial.material.isMultiSupport()) {
this._defineMaterialTint(baseMaterial, useTint);
} else {
baseMaterial.define('USE_TINT', useTint);
}
}
this._materialCache = {};
},
_defineMaterialTint(material, useTint) {
const passes = material._effect._passes;
if (passes && passes.length > 0) {
if (passes[0]._defines['USE_TINT'] != useTint) {
material.define('USE_TINT', useTint);
}
}
},
// if change use batch mode, just clear material cache
_updateBatch () {
let baseMaterial = this.getMaterial(0);
if (baseMaterial) {
if (!baseMaterial.material.isMultiSupport()) {
baseMaterial.define('CC_USE_MODEL', !this.enableBatch);
} else {
if (!this.enableBatch) this.enableBatch = true;
}
}
this._materialCache = {};
},
_validateRender () {
let skeletonData = this.skeletonData;
if (!skeletonData || !skeletonData.isTexturesLoaded()) {
this.disableRender();
return;
}
this._super();
},
/**
* !#en
* Sets runtime skeleton data to sp.Skeleton.<br>
* This method is different from the `skeletonData` property. This method is passed in the raw data provided by the Spine runtime, and the skeletonData type is the asset type provided by Creator.
* !#zh
* 设置底层运行时用到的 SkeletonData。<br>
* 这个接口有别于 `skeletonData` 属性,这个接口传入的是 Spine runtime 提供的原始数据,而 skeletonData 的类型是 Creator 提供的资源类型。
* @method setSkeletonData
* @param {sp.spine.SkeletonData} skeletonData
*/
setSkeletonData (skeletonData) {
if (skeletonData.width != null && skeletonData.height != null) {
this.node.setContentSize(skeletonData.width, skeletonData.height);
}
if (!CC_EDITOR) {
if (this._cacheMode === AnimationCacheMode.SHARED_CACHE) {
this._skeletonCache = SkeletonCache.sharedCache;
} else if (this._cacheMode === AnimationCacheMode.PRIVATE_CACHE) {
this._skeletonCache = new SkeletonCache;
this._skeletonCache.enablePrivateMode();
}
}
if (this.isAnimationCached()) {
if (this.debugBones || this.debugSlots) {
cc.warn("Debug bones or slots is invalid in cached mode");
}
let skeletonInfo = this._skeletonCache.getSkeletonCache(this.skeletonData._uuid, skeletonData);
this._skeleton = skeletonInfo.skeleton;
this._clipper = skeletonInfo.clipper;
this._rootBone = this._skeleton.getRootBone();
} else {
this._skeleton = new spine.Skeleton(skeletonData);
this._clipper = new spine.SkeletonClipping();
this._rootBone = this._skeleton.getRootBone();
}
this.markForRender(true);
},
/**
* !#en Sets slots visible range.
* !#zh 设置骨骼插槽可视范围。
* @method setSlotsRange
* @param {Number} startSlotIndex
* @param {Number} endSlotIndex
*/
setSlotsRange (startSlotIndex, endSlotIndex) {
if (this.isAnimationCached()) {
cc.warn("Slots visible range can not be modified in cached mode.");
} else {
this._startSlotIndex = startSlotIndex;
this._endSlotIndex = endSlotIndex;
}
},
/**
* !#en Sets animation state data.<br>
* The parameter type is {{#crossLinkModule "sp.spine"}}sp.spine{{/crossLinkModule}}.AnimationStateData.
* !#zh 设置动画状态数据。<br>
* 参数是 {{#crossLinkModule "sp.spine"}}sp.spine{{/crossLinkModule}}.AnimationStateData。
* @method setAnimationStateData
* @param {sp.spine.AnimationStateData} stateData
*/
setAnimationStateData (stateData) {
if (this.isAnimationCached()) {
cc.warn("'setAnimationStateData' interface can not be invoked in cached mode.");
} else {
var state = new spine.AnimationState(stateData);
if (this._listener) {
if (this._state) {
this._state.removeListener(this._listener);
}
state.addListener(this._listener);
}
this._state = state;
}
},
// IMPLEMENT
__preload () {
this._super();
if (CC_EDITOR) {
var Flags = cc.Object.Flags;
this._objFlags |= (Flags.IsAnchorLocked | Flags.IsSizeLocked);
this._refreshInspector();
}
var children = this.node.children;
for (var i = 0, n = children.length; i < n; i++) {
var child = children[i];
if (child && child._name === "DEBUG_DRAW_NODE" ) {
child.destroy();
}
}
this._updateSkeletonData();
this._updateDebugDraw();
this._updateUseTint();
this._updateBatch();
},
/**
* !#en
* It's best to set cache mode before set property 'dragonAsset', or will waste some cpu time.
* If set the mode in editor, then no need to worry about order problem.
* !#zh
* 若想切换渲染模式,最好在设置'dragonAsset'之前,先设置好渲染模式,否则有运行时开销。
* 若在编辑中设置渲染模式,则无需担心设置次序的问题。
*
* @method setAnimationCacheMode
* @param {AnimationCacheMode} cacheMode
* @example
* skeleton.setAnimationCacheMode(sp.Skeleton.AnimationCacheMode.SHARED_CACHE);
*/
setAnimationCacheMode (cacheMode) {
if (this._preCacheMode !== cacheMode) {
this._cacheMode = cacheMode;
this._updateSkeletonData();
this._updateUseTint();
}
},
/**
* !#en Whether in cached mode.
* !#zh 当前是否处于缓存模式。
* @method isAnimationCached
* @return {Boolean}
*/
isAnimationCached () {
if (CC_EDITOR) return false;
return this._cacheMode !== AnimationCacheMode.REALTIME;
},
update (dt) {
if (CC_EDITOR) return;
if (this.paused) return;
dt *= this.timeScale * sp.timeScale;
if (this.isAnimationCached()) {
if (this._assembler) this._assembler.handleDynamicAtlasAndSwitchMaterial(this);
// Cache mode and has animation queue.
if (this._isAniComplete) {
if (this._animationQueue.length === 0 && !this._headAniInfo) {
let frameCache = this._frameCache;
if (frameCache && frameCache.isInvalid()) {
frameCache.updateToFrame();
let frames = frameCache.frames;
this._curFrame = frames[frames.length - 1];
}
return;
}
if (!this._headAniInfo) {
this._headAniInfo = this._animationQueue.shift();
}
this._accTime += dt;
if (this._accTime > this._headAniInfo.delay) {
let aniInfo = this._headAniInfo;
this._headAniInfo = null;
this.setAnimation (0, aniInfo.animationName, aniInfo.loop);
}
return;
}
this._updateCache(dt);
} else {
this._updateRealtime(dt);
}
},
_emitCacheCompleteEvent () {
if (!this._listener) return;
this._endEntry.animation.name = this._animationName;
this._listener.complete && this._listener.complete(this._endEntry);
this._listener.end && this._listener.end(this._endEntry);
},
_updateCache (dt) {
let frameCache = this._frameCache;
if (!frameCache.isInited()) {
return;
}
let frames = frameCache.frames;
let frameTime = SkeletonCache.FrameTime;
// Animation Start, the event diffrent from dragonbones inner event,
// It has no event object.
if (this._accTime == 0 && this._playCount == 0) {
this._startEntry.animation.name = this._animationName;
this._listener && this._listener.start && this._listener.start(this._startEntry);
}
this._accTime += dt;
let frameIdx = Math.floor(this._accTime / frameTime);
if (!frameCache.isCompleted) {
frameCache.updateToFrame(frameIdx);
}
if (frameCache.isCompleted && frameIdx >= frames.length) {
this._playCount ++;
if (this._playTimes > 0 && this._playCount >= this._playTimes) {
// set frame to end frame.
this._curFrame = frames[frames.length - 1];
this._accTime = 0;
this._playCount = 0;
this._isAniComplete = true;
this._emitCacheCompleteEvent();
return;
}
this._accTime = 0;
frameIdx = 0;
this._emitCacheCompleteEvent();
}
this._curFrame = frames[frameIdx];
},
_updateRealtime (dt) {
let skeleton = this._skeleton;
let state = this._state;
if (skeleton) {
skeleton.update(dt);
if (state) {
state.update(dt);
state.apply(skeleton);
}
}
},
/**
* !#en Sets vertex effect delegate.
* !#zh 设置顶点动画代理
* @method setVertexEffectDelegate
* @param {sp.VertexEffectDelegate} effectDelegate
*/
setVertexEffectDelegate (effectDelegate) {
this._effectDelegate = effectDelegate;
},
// RENDERER
/**
* !#en Computes the world SRT from the local SRT for each bone.
* !#zh 重新更新所有骨骼的世界 Transform
* 当获取 bone 的数值未更新时,即可使用该函数进行更新数值。
* @method updateWorldTransform
* @example
* var bone = spine.findBone('head');
* cc.log(bone.worldX); // return 0;
* spine.updateWorldTransform();
* bone = spine.findBone('head');
* cc.log(bone.worldX); // return -23.12;
*/
updateWorldTransform () {
if (!this.isAnimationCached()) return;
if (this._skeleton) {
this._skeleton.updateWorldTransform();
}
},
/**
* !#en Sets the bones and slots to the setup pose.
* !#zh 还原到起始动作
* @method setToSetupPose
*/
setToSetupPose () {
if (this._skeleton) {
this._skeleton.setToSetupPose();
}
},
/**
* !#en
* Sets the bones to the setup pose,
* using the values from the `BoneData` list in the `SkeletonData`.
* !#zh
* 设置 bone 到起始动作
* 使用 SkeletonData 中的 BoneData 列表中的值。
* @method setBonesToSetupPose
*/
setBonesToSetupPose () {
if (this._skeleton) {
this._skeleton.setBonesToSetupPose();
}
},
/**
* !#en
* Sets the slots to the setup pose,
* using the values from the `SlotData` list in the `SkeletonData`.
* !#zh
* 设置 slot 到起始动作。
* 使用 SkeletonData 中的 SlotData 列表中的值。
* @method setSlotsToSetupPose
*/
setSlotsToSetupPose () {
if (this._skeleton) {
this._skeleton.setSlotsToSetupPose();
}
},
/**
* !#en
* Updating an animation cache to calculate all frame data in the animation is a cost in
* performance due to calculating all data in a single frame.
* To update the cache, use the invalidAnimationCache method with high performance.
* !#zh
* 更新某个动画缓存, 预计算动画中所有帧数据,由于在单帧计算所有数据,所以较消耗性能。
* 若想更新缓存,可使用 invalidAnimationCache 方法,具有较高性能。
* @method updateAnimationCache
* @param {String} animName
*/
updateAnimationCache (animName) {
if (!this.isAnimationCached()) return;
let uuid = this.skeletonData._uuid;
if (this._skeletonCache) {
this._skeletonCache.updateAnimationCache(uuid, animName);
}
},
/**
* !#en
* Invalidates the animation cache, which is then recomputed on each frame..
* !#zh
* 使动画缓存失效,之后会在每帧重新计算。
* @method invalidAnimationCache
*/
invalidAnimationCache () {
if (!this.isAnimationCached()) return;
if (this._skeletonCache) {
this._skeletonCache.invalidAnimationCache(this.skeletonData._uuid);
}
},
/**
* !#en
* Finds a bone by name.
* This does a string comparison for every bone.<br>
* Returns a {{#crossLinkModule "sp.spine"}}sp.spine{{/crossLinkModule}}.Bone object.
* !#zh
* 通过名称查找 bone。
* 这里对每个 bone 的名称进行了对比。<br>
* 返回一个 {{#crossLinkModule "sp.spine"}}sp.spine{{/crossLinkModule}}.Bone 对象。
*
* @method findBone
* @param {String} boneName
* @return {sp.spine.Bone}
*/
findBone (boneName) {
if (this._skeleton) {
return this._skeleton.findBone(boneName);
}
return null;
},
/**
* !#en
* Finds a slot by name. This does a string comparison for every slot.<br>
* Returns a {{#crossLinkModule "sp.spine"}}sp.spine{{/crossLinkModule}}.Slot object.
* !#zh
* 通过名称查找 slot。这里对每个 slot 的名称进行了比较。<br>
* 返回一个 {{#crossLinkModule "sp.spine"}}sp.spine{{/crossLinkModule}}.Slot 对象。
*
* @method findSlot
* @param {String} slotName
* @return {sp.spine.Slot}
*/
findSlot (slotName) {
if (this._skeleton) {
return this._skeleton.findSlot(slotName);
}
return null;
},
/**
* !#en
* Finds a skin by name and makes it the active skin.
* This does a string comparison for every skin.<br>
* Note that setting the skin does not change which attachments are visible.<br>
* Returns a {{#crossLinkModule "sp.spine"}}sp.spine{{/crossLinkModule}}.Skin object.
* !#zh
* 按名称查找皮肤,激活该皮肤。这里对每个皮肤的名称进行了比较。<br>
* 注意:设置皮肤不会改变 attachment 的可见性。<br>
* 返回一个 {{#crossLinkModule "sp.spine"}}sp.spine{{/crossLinkModule}}.Skin 对象。
*
* @method setSkin
* @param {String} skinName
*/
setSkin (skinName) {
if (this._skeleton) {
this._skeleton.setSkinByName(skinName);
this._skeleton.setSlotsToSetupPose();
}
this.invalidAnimationCache();
},
/**
* !#en
* Returns the attachment for the slot and attachment name.
* The skeleton looks first in its skin, then in the skeleton datas default skin.<br>
* Returns a {{#crossLinkModule "sp.spine"}}sp.spine{{/crossLinkModule}}.Attachment object.
* !#zh
* 通过 slot 和 attachment 的名称获取 attachment。Skeleton 优先查找它的皮肤,然后才是 Skeleton Data 中默认的皮肤。<br>
* 返回一个 {{#crossLinkModule "sp.spine"}}sp.spine{{/crossLinkModule}}.Attachment 对象。
*
* @method getAttachment
* @param {String} slotName
* @param {String} attachmentName
* @return {sp.spine.Attachment}
*/
getAttachment (slotName, attachmentName) {
if (this._skeleton) {
return this._skeleton.getAttachmentByName(slotName, attachmentName);
}
return null;
},
/**
* !#en
* Sets the attachment for the slot and attachment name.
* The skeleton looks first in its skin, then in the skeleton datas default skin.
* !#zh
* 通过 slot 和 attachment 的名字来设置 attachment。
* Skeleton 优先查找它的皮肤,然后才是 Skeleton Data 中默认的皮肤。
* @method setAttachment
* @param {String} slotName
* @param {String} attachmentName
*/
setAttachment (slotName, attachmentName) {
if (this._skeleton) {
this._skeleton.setAttachment(slotName, attachmentName);
}
this.invalidAnimationCache();
},
/**
* 获取 attachment 的 region 数据
*/
getRegionData(slotName, attachmentName) {
const attachment = this.getAttachment(slotName, attachmentName);
if (attachment) return new sp.RegionData(attachment);
return null;
},
/**
* 修改 attachment 的 region 数据
*/
setRegionData(slotName, attachmentName, regionData) {
const attachment = this.getAttachment(slotName, attachmentName);
if (attachment) {
regionData.assignToAttachment(attachment);
this.setVertsDirty();
return true;
}
return false;
},
/**
* Return the renderer of attachment.
* @method getTextureAtlas
* @param {sp.spine.RegionAttachment|spine.BoundingBoxAttachment} regionAttachment
* @return {sp.spine.TextureAtlasRegion}
*/
getTextureAtlas (regionAttachment) {
return regionAttachment.region;
},
// ANIMATION
/**
* !#en
* Mix applies all keyframe values,
* interpolated for the specified time and mixed with the current values.
* !#zh 为所有关键帧设定混合及混合时间(从当前值开始差值)。
* @method setMix
* @param {String} fromAnimation
* @param {String} toAnimation
* @param {Number} duration
*/
setMix (fromAnimation, toAnimation, duration) {
if (this._state) {
this._state.data.setMix(fromAnimation, toAnimation, duration);
}
},
/**
* !#en Set the current animation. Any queued animations are cleared.<br>
* Returns a {{#crossLinkModule "sp.spine"}}sp.spine{{/crossLinkModule}}.TrackEntry object.
* !#zh 设置当前动画。队列中的任何的动画将被清除。<br>
* 返回一个 {{#crossLinkModule "sp.spine"}}sp.spine{{/crossLinkModule}}.TrackEntry 对象。
* @method setAnimation
* @param {Number} trackIndex
* @param {String} name
* @param {Boolean} loop
* @return {sp.spine.TrackEntry}
*/
setAnimation (trackIndex, name, loop) {
this._playTimes = loop ? 0 : 1;
this._animationName = name;
if (this.isAnimationCached()) {
if (trackIndex !== 0) {
cc.warn("Track index can not greater than 0 in cached mode.");
}
if (!this._skeletonCache) return null;
let cache = this._skeletonCache.getAnimationCache(this.skeletonData._uuid, name);
if (!cache) {
cache = this._skeletonCache.initAnimationCache(this.skeletonData._uuid, name);
}
if (cache) {
this._isAniComplete = false;
this._accTime = 0;
this._playCount = 0;
this._frameCache = cache;
if (this.attachUtil._hasAttachedNode()) {
this._frameCache.enableCacheAttachedInfo();
}
if (this._assembler) this._assembler.handleDynamicAtlasAndSwitchMaterial(this);
this._frameCache.updateToFrame(0);
this._curFrame = this._frameCache.frames[0];
}
} else {
if (this._skeleton) {
var animation = this._skeleton.data.findAnimation(name);
if (!animation) {
cc.logID(7509, name);
return null;
}
var res = this._state.setAnimationWith(trackIndex, animation, loop);
this._state.apply(this._skeleton);
return res;
}
}
return null;
},
/**
* !#en Adds an animation to be played delay seconds after the current or last queued animation.<br>
* Returns a {{#crossLinkModule "sp.spine"}}sp.spine{{/crossLinkModule}}.TrackEntry object.
* !#zh 添加一个动画到动画队列尾部,还可以延迟指定的秒数。<br>
* 返回一个 {{#crossLinkModule "sp.spine"}}sp.spine{{/crossLinkModule}}.TrackEntry 对象。
* @method addAnimation
* @param {Number} trackIndex
* @param {String} name
* @param {Boolean} loop
* @param {Number} [delay=0]
* @return {sp.spine.TrackEntry}
*/
addAnimation (trackIndex, name, loop, delay) {
delay = delay || 0;
if (this.isAnimationCached()) {
if (trackIndex !== 0) {
cc.warn("Track index can not greater than 0 in cached mode.");
}
this._animationQueue.push({animationName : name, loop: loop, delay : delay});
} else {
if (this._skeleton) {
var animation = this._skeleton.data.findAnimation(name);
if (!animation) {
cc.logID(7510, name);
return null;
}
return this._state.addAnimationWith(trackIndex, animation, loop, delay);
}
}
return null;
},
/**
* !#en Find animation with specified name.
* !#zh 查找指定名称的动画
* @method findAnimation
* @param {String} name
* @returns {sp.spine.Animation}
*/
findAnimation (name) {
if (this._skeleton) {
return this._skeleton.data.findAnimation(name);
}
return null;
},
/**
* !#en Returns track entry by trackIndex.<br>
* Returns a {{#crossLinkModule "sp.spine"}}sp.spine{{/crossLinkModule}}.TrackEntry object.
* !#zh 通过 track 索引获取 TrackEntry。<br>
* 返回一个 {{#crossLinkModule "sp.spine"}}sp.spine{{/crossLinkModule}}.TrackEntry 对象。
* @method getCurrent
* @param trackIndex
* @return {sp.spine.TrackEntry}
*/
getCurrent (trackIndex) {
if (this.isAnimationCached()) {
cc.warn("'getCurrent' interface can not be invoked in cached mode.");
} else {
if (this._state) {
return this._state.getCurrent(trackIndex);
}
}
return null;
},
/**
* !#en Clears all tracks of animation state.
* !#zh 清除所有 track 的动画状态。
* @method clearTracks
*/
clearTracks () {
if (this.isAnimationCached()) {
cc.warn("'clearTracks' interface can not be invoked in cached mode.");
} else {
if (this._state) {
this._state.clearTracks();
}
}
},
/**
* !#en Clears track of animation state by trackIndex.
* !#zh 清除出指定 track 的动画状态。
* @method clearTrack
* @param {number} trackIndex
*/
clearTrack (trackIndex) {
if (this.isAnimationCached()) {
cc.warn("'clearTrack' interface can not be invoked in cached mode.");
} else {
if (this._state) {
this._state.clearTrack(trackIndex);
if (CC_EDITOR && !cc.engine.isPlaying) {
this._state.update(0);
this.setToSetupPose();
}
}
}
},
/**
* !#en Set the start event listener.
* !#zh 用来设置开始播放动画的事件监听。
* @method setStartListener
* @param {function} listener
*/
setStartListener (listener) {
this._ensureListener();
this._listener.start = listener;
},
/**
* !#en Set the interrupt event listener.
* !#zh 用来设置动画被打断的事件监听。
* @method setInterruptListener
* @param {function} listener
*/
setInterruptListener (listener) {
this._ensureListener();
this._listener.interrupt = listener;
},
/**
* !#en Set the end event listener.
* !#zh 用来设置动画播放完后的事件监听。
* @method setEndListener
* @param {function} listener
*/
setEndListener (listener) {
this._ensureListener();
this._listener.end = listener;
},
/**
* !#en Set the dispose event listener.
* !#zh 用来设置动画将被销毁的事件监听。
* @method setDisposeListener
* @param {function} listener
*/
setDisposeListener (listener) {
this._ensureListener();
this._listener.dispose = listener;
},
/**
* !#en Set the complete event listener.
* !#zh 用来设置动画播放一次循环结束后的事件监听。
* @method setCompleteListener
* @param {function} listener
*/
setCompleteListener (listener) {
this._ensureListener();
this._listener.complete = listener;
},
/**
* !#en Set the animation event listener.
* !#zh 用来设置动画播放过程中帧事件的监听。
* @method setEventListener
* @param {function} listener
*/
setEventListener (listener) {
this._ensureListener();
this._listener.event = listener;
},
/**
* !#en Set the start event listener for specified TrackEntry.
* !#zh 用来为指定的 TrackEntry 设置动画开始播放的事件监听。
* @method setTrackStartListener
* @param {sp.spine.TrackEntry} entry
* @param {function} listener
*/
setTrackStartListener (entry, listener) {
TrackEntryListeners.getListeners(entry).start = listener;
},
/**
* !#en Set the interrupt event listener for specified TrackEntry.
* !#zh 用来为指定的 TrackEntry 设置动画被打断的事件监听。
* @method setTrackInterruptListener
* @param {sp.spine.TrackEntry} entry
* @param {function} listener
*/
setTrackInterruptListener (entry, listener) {
TrackEntryListeners.getListeners(entry).interrupt = listener;
},
/**
* !#en Set the end event listener for specified TrackEntry.
* !#zh 用来为指定的 TrackEntry 设置动画播放结束的事件监听。
* @method setTrackEndListener
* @param {sp.spine.TrackEntry} entry
* @param {function} listener
*/
setTrackEndListener (entry, listener) {
TrackEntryListeners.getListeners(entry).end = listener;
},
/**
* !#en Set the dispose event listener for specified TrackEntry.
* !#zh 用来为指定的 TrackEntry 设置动画即将被销毁的事件监听。
* @method setTrackDisposeListener
* @param {sp.spine.TrackEntry} entry
* @param {function} listener
*/
setTrackDisposeListener(entry, listener){
TrackEntryListeners.getListeners(entry).dispose = listener;
},
/**
* !#en Set the complete event listener for specified TrackEntry.
* !#zh 用来为指定的 TrackEntry 设置动画一次循环播放结束的事件监听。
* @method setTrackCompleteListener
* @param {sp.spine.TrackEntry} entry
* @param {function} listener
* @param {sp.spine.TrackEntry} listener.entry
* @param {Number} listener.loopCount
*/
setTrackCompleteListener (entry, listener) {
TrackEntryListeners.getListeners(entry).complete = function (trackEntry) {
var loopCount = Math.floor(trackEntry.trackTime / trackEntry.animationEnd);
listener(trackEntry, loopCount);
};
},
/**
* !#en Set the event listener for specified TrackEntry.
* !#zh 用来为指定的 TrackEntry 设置动画帧事件的监听。
* @method setTrackEventListener
* @param {sp.spine.TrackEntry} entry
* @param {function} listener
*/
setTrackEventListener (entry, listener) {
TrackEntryListeners.getListeners(entry).event = listener;
},
/**
* !#en Get the animation state object
* !#zh 获取动画状态
* @method getState
* @return {sp.spine.AnimationState} state
*/
getState () {
return this._state;
},
_resetDefaultAnim: CC_EDITOR && function () {
if (this.skeletonData) {
let animEnum = this.skeletonData.getAnimsEnum();
if (!animEnum.hasOwnProperty(this.defaultAnimation)) {
this.defaultAnimation = '';
}
}
},
_resetDefaultSkin: CC_EDITOR && function () {
if (this.skeletonData) {
let skinEnum = this.skeletonData.getSkinsEnum();
if(!skinEnum.hasOwnProperty(this.defaultSkin)) {
this.defaultSkin = '';
}
}
},
// update animation list for editor
_updateAnimEnum: CC_EDITOR && function () {
var animEnum;
if (this.skeletonData) {
animEnum = this.skeletonData.getAnimsEnum();
}
// change enum
setEnumAttr(this, '_animationIndex', animEnum || DefaultAnimsEnum);
},
// update skin list for editor
_updateSkinEnum: CC_EDITOR && function () {
var skinEnum;
if (this.skeletonData) {
skinEnum = this.skeletonData.getSkinsEnum();
}
// change enum
setEnumAttr(this, '_defaultSkinIndex', skinEnum || DefaultSkinsEnum);
},
_ensureListener () {
if (!this._listener) {
this._listener = new TrackEntryListeners();
if (this._state) {
this._state.addListener(this._listener);
}
}
},
_updateSkeletonData() {
this._dataDirty = true;
if (!this.skeletonData) {
this.disableRender();
return;
}
let data = this.skeletonData.getRuntimeData();
if (!data) {
this.disableRender();
return;
}
try {
this.setSkeletonData(data);
if (!this.isAnimationCached()) {
this.setAnimationStateData(new spine.AnimationStateData(this._skeleton.data));
}
this.defaultSkin && this.setSkin(this.defaultSkin);
}
catch (e) {
cc.warn(e);
}
this.attachUtil.init(this);
this.attachUtil._associateAttachedNode();
this._preCacheMode = this._cacheMode;
this.animation = this.defaultAnimation;
},
_refreshInspector () {
// update inspector
this._updateAnimEnum();
this._updateSkinEnum();
Editor.Utils.refreshSelectedInspector('node', this.node.uuid);
},
_updateDebugDraw: function () {
if (this.debugBones || this.debugSlots) {
if (!this._debugRenderer) {
let debugDrawNode = new cc.PrivateNode();
debugDrawNode.name = 'DEBUG_DRAW_NODE';
let debugDraw = debugDrawNode.addComponent(Graphics);
debugDraw.lineWidth = 1;
debugDraw.strokeColor = cc.color(255, 0, 0, 255);
this._debugRenderer = debugDraw;
}
this._debugRenderer.node.parent = this.node;
if (this.isAnimationCached()) {
cc.warn("Debug bones or slots is invalid in cached mode");
}
}
else if (this._debugRenderer) {
this._debugRenderer.node.parent = null;
}
},
});
module.exports = sp.Skeleton;