2011-08-03 20:05:33 +00:00
|
|
|
/*
|
|
|
|
Copyright (C) 2007, 2010 - Bit-Blot
|
|
|
|
|
|
|
|
This file is part of Aquaria.
|
|
|
|
|
|
|
|
Aquaria is free software; you can redistribute it and/or
|
|
|
|
modify it under the terms of the GNU General Public License
|
|
|
|
as published by the Free Software Foundation; either version 2
|
|
|
|
of the License, or (at your option) any later version.
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
|
|
|
|
|
|
|
See the GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with this program; if not, write to the Free Software
|
|
|
|
Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
|
|
|
*/
|
2012-09-23 02:51:13 +00:00
|
|
|
#ifndef SKELETALSPRITE_H
|
|
|
|
#define SKELETALSPRITE_H
|
|
|
|
|
2011-08-03 20:05:33 +00:00
|
|
|
#include "Quad.h"
|
|
|
|
#include "SimpleIStringStream.h"
|
2021-01-11 23:26:44 +00:00
|
|
|
#include <map>
|
2022-09-23 16:10:44 +00:00
|
|
|
#include "Interpolators.h"
|
2011-08-03 20:05:33 +00:00
|
|
|
// for 2d system only
|
|
|
|
|
|
|
|
enum AnimationCommand
|
|
|
|
{
|
|
|
|
AC_PRT_LOAD =0,
|
|
|
|
AC_PRT_START ,
|
|
|
|
AC_PRT_STOP ,
|
|
|
|
AC_SEGS_START ,
|
|
|
|
AC_FRM_SHOW ,
|
|
|
|
AC_SND_PLAY ,
|
2016-04-17 12:33:23 +00:00
|
|
|
AC_SEGS_STOP,
|
|
|
|
AC_SET_PASS,
|
2023-12-21 05:04:09 +00:00
|
|
|
AC_RESET_PASS,
|
|
|
|
AC_SET_FH
|
2011-08-03 20:05:33 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
class ParticleEffect;
|
|
|
|
class SkeletalSprite;
|
|
|
|
|
2022-05-18 17:44:42 +00:00
|
|
|
class Bone : public CollideQuad
|
2011-08-03 20:05:33 +00:00
|
|
|
{
|
2016-07-09 02:18:40 +00:00
|
|
|
friend class SkeletalSprite;
|
2011-08-03 20:05:33 +00:00
|
|
|
public:
|
|
|
|
Bone();
|
2022-07-18 21:00:22 +00:00
|
|
|
virtual ~Bone();
|
2011-08-03 20:05:33 +00:00
|
|
|
void setAnimated(int a);
|
|
|
|
|
|
|
|
enum {
|
2022-09-14 03:20:37 +00:00
|
|
|
ANIM_NONE = 0x00,
|
|
|
|
ANIM_POS = 0x01,
|
|
|
|
ANIM_ROT = 0x02,
|
|
|
|
ANIM_ALL = ANIM_POS | ANIM_ROT
|
2011-08-03 20:05:33 +00:00
|
|
|
};
|
|
|
|
void createStrip(bool vert, int num);
|
2023-12-21 05:04:09 +00:00
|
|
|
void addFrame(const std::string &gfx);
|
2011-08-03 20:05:33 +00:00
|
|
|
void showFrame(int i);
|
2023-05-25 14:58:08 +00:00
|
|
|
void destroy() OVERRIDE;
|
2011-08-03 20:05:33 +00:00
|
|
|
std::string gfx;
|
|
|
|
std::string name;
|
2017-01-14 17:10:20 +00:00
|
|
|
size_t boneIdx;
|
2017-01-14 17:10:20 +00:00
|
|
|
int pidx, rbp;
|
2011-08-03 20:05:33 +00:00
|
|
|
std::string prt;
|
|
|
|
std::vector<Vector> changeStrip;
|
2016-05-05 17:40:28 +00:00
|
|
|
|
2011-08-03 20:05:33 +00:00
|
|
|
bool generateCollisionMask;
|
2022-05-17 23:18:27 +00:00
|
|
|
bool enableCollision;
|
2011-08-03 20:05:33 +00:00
|
|
|
int animated;
|
|
|
|
Vector originalScale;
|
|
|
|
|
2022-05-17 23:18:27 +00:00
|
|
|
bool canCollide() const;
|
|
|
|
|
2011-08-03 20:05:33 +00:00
|
|
|
void addSegment(Bone *b);
|
2016-07-09 02:18:40 +00:00
|
|
|
ParticleEffect *getEmitter(unsigned slot) const;
|
2011-08-03 20:05:33 +00:00
|
|
|
|
|
|
|
int segmentChain;
|
|
|
|
|
|
|
|
void updateSegments();
|
|
|
|
void updateSegment(Bone *b, const Vector &diff);
|
|
|
|
|
|
|
|
SkeletalSprite *skeleton;
|
|
|
|
|
2016-05-05 17:40:28 +00:00
|
|
|
|
2011-08-03 20:05:33 +00:00
|
|
|
void setSegmentProps(int minDist, int maxDist, bool reverse);
|
|
|
|
Vector segmentOffset;
|
|
|
|
|
2022-05-21 15:31:50 +00:00
|
|
|
bool stripVert;
|
2011-08-03 20:05:33 +00:00
|
|
|
bool fileRenderQuad;
|
2016-04-18 20:08:36 +00:00
|
|
|
bool selectable;
|
2016-04-17 12:33:23 +00:00
|
|
|
int originalRenderPass; // stores the render pass originally set in the XML file. For AC_RESET_PASS.
|
2023-12-21 05:04:09 +00:00
|
|
|
bool originalFH;
|
2011-08-03 20:05:33 +00:00
|
|
|
|
2022-05-17 23:18:27 +00:00
|
|
|
void spawnParticlesFromCollisionMask(const char *p, unsigned intv, int layer, float rotz = 0);
|
2023-04-17 23:54:12 +00:00
|
|
|
Vector getCollisionMaskNormal(Vector pos, float dist) const;
|
2022-05-18 00:15:19 +00:00
|
|
|
|
2022-05-19 23:04:19 +00:00
|
|
|
virtual void renderCollision(const RenderState& rs) const OVERRIDE;
|
2022-05-17 23:18:27 +00:00
|
|
|
|
2011-08-03 20:05:33 +00:00
|
|
|
protected:
|
2016-07-09 02:18:40 +00:00
|
|
|
std::vector<ParticleEffect*> emitters;
|
2011-08-03 20:05:33 +00:00
|
|
|
int minDist, maxDist, reverse;
|
|
|
|
std::vector<Bone*> segments;
|
2022-05-18 00:15:19 +00:00
|
|
|
public:
|
|
|
|
std::vector<Vector> collisionMask;
|
|
|
|
std::vector<Vector> transformedCollisionMask;
|
|
|
|
float collisionMaskRadius;
|
2023-12-21 05:04:09 +00:00
|
|
|
std::vector<std::string> framegfx;
|
|
|
|
|
2011-08-03 20:05:33 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
class BoneCommand
|
|
|
|
{
|
|
|
|
public:
|
2016-04-17 13:16:55 +00:00
|
|
|
bool parse(Bone *b, SimpleIStringStream &is);
|
2011-08-03 20:05:33 +00:00
|
|
|
void run();
|
|
|
|
AnimationCommand command;
|
|
|
|
Bone *b;
|
|
|
|
|
|
|
|
int slot;
|
|
|
|
std::string file;
|
|
|
|
};
|
|
|
|
|
|
|
|
class BoneKeyframe
|
|
|
|
{
|
|
|
|
public:
|
2017-01-12 21:51:46 +00:00
|
|
|
BoneKeyframe() : idx(0), x(0), y(0), rot(0), sx(1), sy(1), doScale(0) {}
|
2017-01-14 17:10:20 +00:00
|
|
|
size_t idx;
|
|
|
|
int x, y, rot;
|
2011-08-03 20:05:33 +00:00
|
|
|
float sx, sy;
|
|
|
|
bool doScale;
|
2022-09-13 16:38:44 +00:00
|
|
|
std::vector<Vector> grid;
|
|
|
|
std::vector<Vector> controlpoints;
|
2011-08-03 20:05:33 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
class SkeletalKeyframe
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
SkeletalKeyframe()
|
|
|
|
{
|
|
|
|
lerpType = 0;
|
|
|
|
t = 0;
|
|
|
|
}
|
|
|
|
int lerpType;
|
|
|
|
float t;
|
|
|
|
std::string sound;
|
|
|
|
std::vector<BoneKeyframe> keyframes;
|
2017-01-14 17:10:20 +00:00
|
|
|
BoneKeyframe *getBoneKeyframe(size_t idx);
|
2011-08-03 20:05:33 +00:00
|
|
|
std::string cmd;
|
|
|
|
std::vector<BoneCommand> commands;
|
|
|
|
|
|
|
|
void copyAllButTime(SkeletalKeyframe *copy);
|
|
|
|
};
|
|
|
|
|
2022-09-13 16:38:44 +00:00
|
|
|
class BoneGridInterpolator
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
size_t idx;
|
2022-09-23 16:10:44 +00:00
|
|
|
BSpline2D bsp;
|
|
|
|
bool storeBoneByIdx;
|
|
|
|
void updateGridOnly(BoneKeyframe& bk, const Bone *bone);
|
|
|
|
void updateGridAndBone(BoneKeyframe& bk, Bone *bone);
|
2022-09-13 16:38:44 +00:00
|
|
|
};
|
|
|
|
|
2011-08-03 20:05:33 +00:00
|
|
|
class Animation
|
|
|
|
{
|
|
|
|
public:
|
2016-04-17 12:33:23 +00:00
|
|
|
Animation();
|
2011-08-03 20:05:33 +00:00
|
|
|
std::string name;
|
|
|
|
typedef std::vector <SkeletalKeyframe> Keyframes;
|
|
|
|
Keyframes keyframes;
|
2017-01-14 17:10:20 +00:00
|
|
|
SkeletalKeyframe *getKeyframe(size_t key);
|
2011-08-03 20:05:33 +00:00
|
|
|
SkeletalKeyframe *getLastKeyframe();
|
|
|
|
SkeletalKeyframe *getFirstKeyframe();
|
Fixes to skel keyframe handling that would usually skip the first keyframe
This didn't seem to cause problems since apparently nothing ever depended
on the first keyframe being processed properly (ie. animationKey(),
bone commands, sounds, particle spawning, etc).
Curiously this was never an issue in the editor since the first keyframe
is hit immediately (since it keeps the time clamped to 0 when not animating),
and the first keyframe triggered ONLY when t==0 exactly.
With this change no keyframes are skipped, and during an animation loop,
the last and the first keyframe are hit together.
entity_setAnimationTime() got an extra param whether to skip keyframes
when setting the time manually (if not skipping, it'll catch up).
This commit also fixes a bug(?) that animationKey() was called with key -1
when changing animations, which was a side-effect of how the old code worked.
(I hope nothing depended on this behavior, i'm not sure but i think it's safe to fix)
2025-01-17 04:37:55 +00:00
|
|
|
size_t getKeyframeIndexBefore(float t) const; // last anim that's <= t
|
|
|
|
|
2017-01-14 17:10:20 +00:00
|
|
|
void cloneKey(size_t key, float toffset);
|
|
|
|
void deleteKey(size_t key);
|
2011-08-03 20:05:33 +00:00
|
|
|
void reorderKeyframes();
|
|
|
|
float getAnimationLength();
|
2017-01-14 17:10:20 +00:00
|
|
|
size_t getSkeletalKeyframeIndex(SkeletalKeyframe *skey);
|
Fixes to skel keyframe handling that would usually skip the first keyframe
This didn't seem to cause problems since apparently nothing ever depended
on the first keyframe being processed properly (ie. animationKey(),
bone commands, sounds, particle spawning, etc).
Curiously this was never an issue in the editor since the first keyframe
is hit immediately (since it keeps the time clamped to 0 when not animating),
and the first keyframe triggered ONLY when t==0 exactly.
With this change no keyframes are skipped, and during an animation loop,
the last and the first keyframe are hit together.
entity_setAnimationTime() got an extra param whether to skip keyframes
when setting the time manually (if not skipping, it'll catch up).
This commit also fixes a bug(?) that animationKey() was called with key -1
when changing animations, which was a side-effect of how the old code worked.
(I hope nothing depended on this behavior, i'm not sure but i think it's safe to fix)
2025-01-17 04:37:55 +00:00
|
|
|
size_t getNumKeyframes() const;
|
2011-08-03 20:05:33 +00:00
|
|
|
void reverse();
|
2023-12-21 05:04:09 +00:00
|
|
|
bool resetOnEnd;
|
2022-09-13 16:38:44 +00:00
|
|
|
|
2022-09-23 16:10:44 +00:00
|
|
|
BoneGridInterpolator *getBoneGridInterpolator(size_t boneIdx);
|
2022-09-13 16:38:44 +00:00
|
|
|
typedef std::vector <BoneGridInterpolator> Interpolators;
|
|
|
|
Interpolators interpolators;
|
2011-08-03 20:05:33 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
class SkeletalSprite;
|
|
|
|
|
|
|
|
class AnimationLayer
|
|
|
|
{
|
|
|
|
public:
|
2016-05-05 17:40:28 +00:00
|
|
|
|
2011-08-03 20:05:33 +00:00
|
|
|
//----
|
|
|
|
AnimationLayer();
|
2016-05-05 17:40:28 +00:00
|
|
|
void setSkeletalSprite(SkeletalSprite *s);
|
2011-08-03 20:05:33 +00:00
|
|
|
Animation *getCurrentAnimation();
|
2025-01-12 01:15:19 +00:00
|
|
|
Animation *getCurrentAnimationOrNull();
|
2011-08-03 20:05:33 +00:00
|
|
|
void animate(const std::string &animation, int loop);
|
|
|
|
void update(float dt);
|
|
|
|
void updateBones();
|
Fixes to skel keyframe handling that would usually skip the first keyframe
This didn't seem to cause problems since apparently nothing ever depended
on the first keyframe being processed properly (ie. animationKey(),
bone commands, sounds, particle spawning, etc).
Curiously this was never an issue in the editor since the first keyframe
is hit immediately (since it keeps the time clamped to 0 when not animating),
and the first keyframe triggered ONLY when t==0 exactly.
With this change no keyframes are skipped, and during an animation loop,
the last and the first keyframe are hit together.
entity_setAnimationTime() got an extra param whether to skip keyframes
when setting the time manually (if not skipping, it'll catch up).
This commit also fixes a bug(?) that animationKey() was called with key -1
when changing animations, which was a side-effect of how the old code worked.
(I hope nothing depended on this behavior, i'm not sure but i think it's safe to fix)
2025-01-17 04:37:55 +00:00
|
|
|
void keyframeReached(SkeletalKeyframe *k, size_t idx);
|
2011-08-03 20:05:33 +00:00
|
|
|
void stopAnimation();
|
|
|
|
float getAnimationLength();
|
2022-11-16 21:46:56 +00:00
|
|
|
void createTransitionAnimation(Animation& to, float time);
|
2011-08-03 20:05:33 +00:00
|
|
|
void playAnimation(int idx, int loop);
|
|
|
|
void playCurrentAnimation(int loop);
|
2013-02-03 15:13:07 +00:00
|
|
|
void enqueueAnimation(const std::string& anim, int loop);
|
2011-08-03 20:05:33 +00:00
|
|
|
float transitionAnimate(std::string anim, float time, int loop);
|
|
|
|
void setTimeMultiplier(float t);
|
2025-01-12 02:30:08 +00:00
|
|
|
bool isAnimating() const;
|
2016-04-17 12:33:23 +00:00
|
|
|
bool contains(const Bone *b) const;
|
|
|
|
void resetPass();
|
Fixes to skel keyframe handling that would usually skip the first keyframe
This didn't seem to cause problems since apparently nothing ever depended
on the first keyframe being processed properly (ie. animationKey(),
bone commands, sounds, particle spawning, etc).
Curiously this was never an issue in the editor since the first keyframe
is hit immediately (since it keeps the time clamped to 0 when not animating),
and the first keyframe triggered ONLY when t==0 exactly.
With this change no keyframes are skipped, and during an animation loop,
the last and the first keyframe are hit together.
entity_setAnimationTime() got an extra param whether to skip keyframes
when setting the time manually (if not skipping, it'll catch up).
This commit also fixes a bug(?) that animationKey() was called with key -1
when changing animations, which was a side-effect of how the old code worked.
(I hope nothing depended on this behavior, i'm not sure but i think it's safe to fix)
2025-01-17 04:37:55 +00:00
|
|
|
void setTimer(float t, bool runKeyframes);
|
|
|
|
|
2016-05-05 17:40:28 +00:00
|
|
|
|
2011-08-03 20:05:33 +00:00
|
|
|
//----
|
|
|
|
float fallThru;
|
|
|
|
float fallThruSpeed;
|
|
|
|
std::string name;
|
|
|
|
std::vector<int> ignoreBones;
|
|
|
|
std::vector<int> includeBones;
|
|
|
|
SkeletalSprite *s;
|
|
|
|
|
Fixes to skel keyframe handling that would usually skip the first keyframe
This didn't seem to cause problems since apparently nothing ever depended
on the first keyframe being processed properly (ie. animationKey(),
bone commands, sounds, particle spawning, etc).
Curiously this was never an issue in the editor since the first keyframe
is hit immediately (since it keeps the time clamped to 0 when not animating),
and the first keyframe triggered ONLY when t==0 exactly.
With this change no keyframes are skipped, and during an animation loop,
the last and the first keyframe are hit together.
entity_setAnimationTime() got an extra param whether to skip keyframes
when setting the time manually (if not skipping, it'll catch up).
This commit also fixes a bug(?) that animationKey() was called with key -1
when changing animations, which was a side-effect of how the old code worked.
(I hope nothing depended on this behavior, i'm not sure but i think it's safe to fix)
2025-01-17 04:37:55 +00:00
|
|
|
int lastKeyframeIndex;
|
2016-05-05 17:40:28 +00:00
|
|
|
|
2011-08-03 20:05:33 +00:00
|
|
|
float timer;
|
|
|
|
int loop;
|
|
|
|
Animation blendAnimation;
|
|
|
|
std::string enqueuedAnimation;
|
|
|
|
int enqueuedAnimationLoop;
|
|
|
|
//float timeMultiplier;
|
|
|
|
//HACK: should be a lerped float
|
|
|
|
InterpolatedVector timeMultiplier;
|
|
|
|
float animationLength;
|
2017-01-14 17:10:20 +00:00
|
|
|
size_t currentAnimation;
|
2011-08-03 20:05:33 +00:00
|
|
|
bool animating;
|
|
|
|
|
2016-05-05 17:40:28 +00:00
|
|
|
|
2011-08-03 20:05:33 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
class SkeletalSprite : public RenderObject
|
|
|
|
{
|
|
|
|
public:
|
2016-05-05 17:40:28 +00:00
|
|
|
|
2011-08-03 20:05:33 +00:00
|
|
|
SkeletalSprite();
|
2022-05-20 01:27:15 +00:00
|
|
|
virtual ~SkeletalSprite();
|
2022-07-18 21:00:22 +00:00
|
|
|
virtual void destroy();
|
2022-05-20 01:27:15 +00:00
|
|
|
|
2011-08-03 20:05:33 +00:00
|
|
|
void loadSkeletal(const std::string &fn);
|
2013-12-28 03:42:02 +00:00
|
|
|
bool saveSkeletal(const std::string &fn);
|
2011-08-03 20:05:33 +00:00
|
|
|
void loadSkin(const std::string &fn);
|
|
|
|
|
2017-01-14 17:10:20 +00:00
|
|
|
Bone *getBoneByIdx(size_t idx);
|
2011-08-03 20:05:33 +00:00
|
|
|
Bone *getBoneByName(const std::string &name);
|
|
|
|
void animate(const std::string &animation, int loop = 0, int layer=0);
|
|
|
|
|
2016-05-05 17:40:28 +00:00
|
|
|
|
|
|
|
|
2017-01-14 17:10:20 +00:00
|
|
|
void setTime(float time, size_t layer=0);
|
2011-08-03 20:05:33 +00:00
|
|
|
|
|
|
|
void updateBones();
|
|
|
|
void playCurrentAnimation(int loop=0, int layer=0);
|
|
|
|
void stopAnimation(int layer=0);
|
|
|
|
void stopAllAnimations();
|
|
|
|
|
2012-03-13 23:58:59 +00:00
|
|
|
float transitionAnimate(const std::string& anim, float time, int loop=0, int layer=0);
|
2011-08-03 20:05:33 +00:00
|
|
|
|
2025-01-12 02:30:08 +00:00
|
|
|
bool isAnimating(size_t layer=0) const;
|
2011-08-03 20:05:33 +00:00
|
|
|
|
2016-05-05 17:40:28 +00:00
|
|
|
void setTimeMultiplier(float t, int layer=0);
|
2011-08-03 20:05:33 +00:00
|
|
|
|
2025-01-13 01:53:33 +00:00
|
|
|
int findSelectableBoneIdxClosestTo(const Vector& pos, bool mustBeInBox = true) const; // -1 if none found, otherwise bones[idx] to get the bone
|
2017-01-14 17:10:20 +00:00
|
|
|
Animation *getCurrentAnimation(size_t layer=0);
|
2025-01-12 01:15:19 +00:00
|
|
|
Animation *getCurrentAnimationOrNull(size_t layer=0);
|
2011-08-03 20:05:33 +00:00
|
|
|
|
2016-05-05 17:40:28 +00:00
|
|
|
|
2011-08-03 20:05:33 +00:00
|
|
|
void nextAnimation();
|
|
|
|
void prevAnimation();
|
|
|
|
void lastAnimation();
|
|
|
|
void firstAnimation();
|
2022-11-18 00:52:46 +00:00
|
|
|
bool selectAnimation(const char *name);
|
2011-08-03 20:05:33 +00:00
|
|
|
|
|
|
|
|
|
|
|
void setFreeze(bool f);
|
2016-05-05 17:40:28 +00:00
|
|
|
|
2013-02-03 15:13:07 +00:00
|
|
|
Animation *getAnimation(const std::string& anim);
|
2011-08-03 20:05:33 +00:00
|
|
|
|
|
|
|
std::vector<Animation> animations;
|
|
|
|
std::vector<Bone*> bones;
|
|
|
|
|
|
|
|
bool isLoaded();
|
2017-01-20 03:51:38 +00:00
|
|
|
size_t getNumAnimLayers() const { return animLayers.size(); }
|
2011-08-03 20:05:33 +00:00
|
|
|
|
2017-01-14 17:10:20 +00:00
|
|
|
AnimationLayer* getAnimationLayer(size_t l);
|
|
|
|
size_t getBoneIdx(Bone *b);
|
|
|
|
void toggleBone(size_t idx, int v);
|
2011-08-03 20:05:33 +00:00
|
|
|
|
|
|
|
void setAnimationKeyNotify(RenderObject *r);
|
|
|
|
|
|
|
|
std::string filenameLoaded;
|
|
|
|
|
|
|
|
static std::string animationPath, skinPath, secondaryAnimationPath;
|
2012-02-19 03:57:04 +00:00
|
|
|
static void clearCache();
|
2016-05-05 17:40:28 +00:00
|
|
|
|
2011-08-03 20:05:33 +00:00
|
|
|
protected:
|
|
|
|
bool frozen;
|
|
|
|
RenderObject *animKeyNotify;
|
|
|
|
bool loaded;
|
|
|
|
friend class AnimationLayer;
|
|
|
|
std::vector<AnimationLayer> animLayers;
|
2017-01-19 22:44:30 +00:00
|
|
|
Bone* initBone(int idx, std::string gfx, int pidx, bool rbp=false, std::string name="", float cr=0, bool fh=false, bool fv=false);
|
2011-08-03 20:05:33 +00:00
|
|
|
void deleteBones();
|
|
|
|
void onUpdate(float dt);
|
|
|
|
};
|
|
|
|
|
2012-09-23 02:51:13 +00:00
|
|
|
#endif
|