/*==LICENSE==*

CyanWorlds.com Engine - MMOG client, server and tools
Copyright (C) 2011  Cyan Worlds, Inc.

This program 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 3 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, see <http://www.gnu.org/licenses/>.

Additional permissions under GNU GPL version 3 section 7

If you modify this Program, or any covered work, by linking or
combining it with any of RAD Game Tools Bink SDK, Autodesk 3ds Max SDK,
NVIDIA PhysX SDK, Microsoft DirectX SDK, OpenSSL library, Independent
JPEG Group JPEG library, Microsoft Windows Media SDK, or Apple QuickTime SDK
(or a modified version of those libraries),
containing parts covered by the terms of the Bink SDK EULA, 3ds Max EULA,
PhysX SDK EULA, DirectX SDK EULA, OpenSSL and SSLeay licenses, IJG
JPEG Library README, Windows Media SDK EULA, or QuickTime SDK EULA, the
licensors of this Program grant you additional
permission to convey the resulting work. Corresponding Source for a
non-source form of such a combination shall include the source code for
the parts of OpenSSL and IJG JPEG Library used as well as that of the covered
work.

You can contact Cyan Worlds, Inc. by email legal@cyan.com
 or by snail mail at:
      Cyan Worlds, Inc.
      14617 N Newport Hwy
      Mead, WA   99021

*==LICENSE==*/

#include "HeadSpin.h"
#include "plMorphSequence.h"
#include "plMorphSequenceSDLMod.h"

#include "plAccessGeometry.h"
#include "plAccessVtxSpan.h"

#include "pnSceneObject/plSceneObject.h"
#include "pnSceneObject/plDrawInterface.h"
#include "pnSceneObject/plCoordinateInterface.h"

#include "plDrawableSpans.h"
#include "plInstanceDrawInterface.h"

#include "hsResMgr.h"
#include "plgDispatch.h"
#include "plMessage/plRenderMsg.h"

#include "plSharedMesh.h"

#include "plTweak.h"
#include "hsTimer.h"
#include "hsFastMath.h"

///////////////////////////////////////////////////////////////////////////

void plMorphDataSet::Read(hsStream* s, hsResMgr* mgr)
{
    hsKeyedObject::Read(s, mgr);
    
    int n = s->ReadLE32();
    fMorphs.SetCount(n);
    int i;
    for( i = 0; i < n; i++ )
        fMorphs[i].Read(s, mgr);
}

void plMorphDataSet::Write(hsStream* s, hsResMgr* mgr)
{
    hsKeyedObject::Write(s, mgr);

    s->WriteLE32(fMorphs.GetCount());
    int i;
    for( i = 0; i < fMorphs.GetCount(); i++ )
        fMorphs[i].Write(s, mgr);
}   

///////////////////////////////////////////////////////////////////////////

const uint32_t kChanMask = plAccessVtxSpan::kPositionMask
                        | plAccessVtxSpan::kNormalMask
                        | plAccessVtxSpan::kUVWMask;

// Probably want another type which is just initialize/override
// so we can set the mesh to whatever base set we want, as if
// that's what got loaded from disk.
// Use Access RW. That might already be handled in the customization stuff.

plConst(float)   kMorphTime(0.5);

class plMorphTarget
{
public:
    uint16_t      fLayer;
    uint16_t      fDelta;
    float    fWeight;
};

hsTArray<plMorphTarget> fTgtWgts;

plMorphSequence::plMorphSequence()
:   fMorphFlags(0),
    fMorphSDLMod(nil),
    fGlobalLayerRef(-1)
{
}

plMorphSequence::~plMorphSequence()
{
    DeInit();
}

bool plMorphSequence::MsgReceive(plMessage* msg)
{
    plRenderMsg* rend = plRenderMsg::ConvertNoRef(msg);
    if( rend )
    {
        // For now, I'm ignoring the target weight stuff for shared meshes.
        // Can always add it in later if desired.
        if( fTgtWgts.GetCount() )
        {
            float delWgt = hsTimer::GetDelSysSeconds() / (kMorphTime > 0 ? kMorphTime : 1.e-3f);
            int i;
            for( i = 0; i < fTgtWgts.GetCount(); i++ )
            {
                float currWgt = GetWeight(fTgtWgts[i].fLayer, fTgtWgts[i].fDelta);
                if( fTgtWgts[i].fWeight < currWgt )
                {
                    if( fTgtWgts[i].fWeight >= (currWgt -= delWgt) )
                        currWgt = fTgtWgts[i].fWeight;
                }
                else if( fTgtWgts[i].fWeight > currWgt )
                {
                    if( fTgtWgts[i].fWeight <= (currWgt += delWgt) )
                        currWgt = fTgtWgts[i].fWeight;
                }
                
                fMorphs[fTgtWgts[i].fLayer].SetWeight(fTgtWgts[i].fDelta, currWgt);

                if( fTgtWgts[i].fWeight == currWgt )
                {
                    fTgtWgts.Remove(i);
                    i--;
                }
            }
            ISetDirty(true);
        }

        if( !(fMorphFlags & kDirty) )
        {
            // We went a whole frame without getting dirty, 
            // we can stop refreshing now.
            plgDispatch::Dispatch()->UnRegisterForExactType(plRenderMsg::Index(), GetKey());

            return true;
        }
        ISetDirty(false);
        if( fMorphFlags & kDirtyIndices )
            IFindIndices();

        if( fMorphFlags & kHaveShared )
        {
            IApplyShared();
        }
        else
        {
            Apply();
        }
        return true;
    }

    plSharedMeshBCMsg *smMsg = plSharedMeshBCMsg::ConvertNoRef(msg);
    if (smMsg)
    {
        if (IGetDrawInterface()->GetKey() == smMsg->GetSender() || IIsUsingDrawable(smMsg->fDraw))
            fMorphFlags |= kDirtyIndices;
    }           

    plGenRefMsg *refMsg = plGenRefMsg::ConvertNoRef(msg);
    if (refMsg)
    {
        plSharedMesh *mesh = plSharedMesh::ConvertNoRef(refMsg->GetRef());
        if (mesh)
        {
            if( refMsg->GetContext() & (plRefMsg::kOnCreate|plRefMsg::kOnRequest) )
            {
                AddSharedMesh(mesh);
            }
            else if( refMsg->GetContext() & plRefMsg::kOnReplace)
            {
                plSharedMesh *oldMesh = plSharedMesh::ConvertNoRef(refMsg->GetOldRef());
                if (oldMesh)
                    RemoveSharedMesh(oldMesh);
                AddSharedMesh(mesh);
            }
            else if( refMsg->GetContext() & (plRefMsg::kOnDestroy|plRefMsg::kOnRemove) )
                RemoveSharedMesh(mesh);             
            
            return true;
        }
    }
    
    return plSingleModifier::MsgReceive(msg);
}

int plMorphSequence::GetNumLayers(plKey meshKey /* = nil */) const
{
    int index = IFindSharedMeshIndex(meshKey);
    if (index < 0)
        return fMorphs.GetCount();
    else
        return fSharedMeshes[index].fMesh->fMorphSet->fMorphs.GetCount();
}

int plMorphSequence::GetNumDeltas(int iLay, plKey meshKey /* = nil */) const 
{ 
    int index = IFindSharedMeshIndex(meshKey);
    if (index < 0)
        return fMorphs[iLay].GetNumDeltas(); 
    else
        return fSharedMeshes[index].fMesh->fMorphSet->fMorphs[iLay].GetNumDeltas();
}

float plMorphSequence::GetWeight(int iLay, int iDel, plKey meshKey /* = nil */) const 
{ 
    int index = IFindSharedMeshIndex(meshKey);
    if (index == -1)
        return fMorphs[iLay].GetWeight(iDel); 
    else
        return fSharedMeshes[index].fArrayWeights[iLay].fDeltaWeights[iDel];
}

void plMorphSequence::SetWeight(int iLay, int iDel, float w, plKey meshKey /* = nil */)
{
    int index = IFindSharedMeshIndex(meshKey);

    // Only dirty if the weight isn't for a pending mesh
    if(meshKey == nil || index >= 0)
        ISetDirty(true);

    if (meshKey == nil)
    {
        if( iLay < fMorphs.GetCount() )
        {
            if( kMorphTime > 0 )
            {
                plMorphTarget tgt;
                tgt.fLayer = iLay;
                tgt.fDelta = iDel;
                tgt.fWeight = w;

                fTgtWgts.Append(tgt);
            }
            else
            {
                fMorphs[iLay].SetWeight(iDel, w);
            }
        }
    }
    else if (index >= 0)
    {
        fSharedMeshes[index].fArrayWeights[iLay].fDeltaWeights[iDel] = w;
        fSharedMeshes[index].fFlags |= plSharedMeshInfo::kInfoDirtyMesh;
        if (index == fGlobalLayerRef)
            ISetAllSharedToGlobal();
    }
    else
    {
        // Setting weight for a mesh we haven't added yet (loading state)
        index = IFindPendingStateIndex(meshKey);
        if (index < 0)
        {
            fPendingStates.Push();
            index = fPendingStates.GetCount() - 1;
            fPendingStates[index].fSharedMeshKey = meshKey;
        }
        if (fPendingStates[index].fArrayWeights.GetCount() <= iLay)
        {
            int had = fPendingStates[index].fArrayWeights.GetCount();
            hsTArray<plMorphArrayWeights> temp(iLay + 1);
            temp = fPendingStates[index].fArrayWeights;
            temp.SetCount(iLay + 1);

            fPendingStates[index].fArrayWeights.Swap(temp);
            int i;
            for( i = had; i < iLay+1; i++ )
                fPendingStates[index].fArrayWeights[i].fDeltaWeights.Reset();
        }
        if (fPendingStates[index].fArrayWeights[iLay].fDeltaWeights.GetCount() <= iDel)
            fPendingStates[index].fArrayWeights[iLay].fDeltaWeights.ExpandAndZero(iDel + 1);

        fPendingStates[index].fArrayWeights[iLay].fDeltaWeights[iDel] = w;
    }
}

void plMorphSequence::ISetDirty(bool on) 
{ 
    if( on )
    {
        if( !(fMorphFlags & kDirty) )
            plgDispatch::Dispatch()->RegisterForExactType(plRenderMsg::Index(), GetKey());
        fMorphFlags |= kDirty; 

        // Even if we know we're already dirty, there could be new info this frame.
        // Need to tell our scene object
        //
        // Actually, in the case of the avatar, this sends an insane flurry of messages to the server 
        // when we drag (or even just hold the mouse button down on) the morph scrollbars. 
        // These messages are completely unneccessary. We broadcast our state when we join a new age, and that's enough.
        // Non-avatar related morphs (if they ever exist) will need to call DirtySynchState some special
        // way. Not here.
        //
        //if (GetTarget(0))
        //  GetTarget(0)->DirtySynchState(kSDLMorphSequence, 0);
    }
    else
    {
        fMorphFlags &= ~kDirty; 
    }
}

// MorphSequence - Init
void plMorphSequence::Init()
{
    if( fMorphFlags & kHaveShared )
    {
        IFindIndices();
    }
    else
    {
        if( !GetHaveSnap() )
        {
            // Take an access snapshot of all our spans
            const plDrawInterface* di = IGetDrawInterface();

            plAccessGeometry::Instance()->TakeSnapShot(di, kChanMask);

            ISetHaveSnap(true);
        }
    }
}

void plMorphSequence::Activate()
{
    Init();
    ISetDirty(true);
}

void plMorphSequence::DeActivate()
{
    if( fMorphFlags & kHaveShared )
    {
        IReleaseIndices();
    }
    plgDispatch::Dispatch()->UnRegisterForExactType(plRenderMsg::Index(), GetKey());
}

// MorphSequence - DeInit
void plMorphSequence::DeInit()
{
    if( fMorphFlags & kHaveShared )
    {
        IReleaseIndices();
    }
    else
    {
        if( GetHaveSnap() )
        {
            // Release all our snapshot data
            const plDrawInterface* di = IGetDrawInterface();

            if( di ) 
                plAccessGeometry::Instance()->ReleaseSnapShot(di);

            ISetHaveSnap(false);
        }
    }
}

void plMorphSequence::IRenormalize(hsTArray<plAccessSpan>& dst) const
{
    int i;
    for( i = 0; i < dst.GetCount(); i++ )
    {
        hsAssert(dst[i].HasAccessVtx(), "Come on, everyone has vertices");
        plAccessVtxSpan& accVtx = dst[i].AccessVtx();
        int j;
        for( j = 0; j < accVtx.VertCount(); j++ )
        {
            hsFastMath::Normalize(accVtx.Normal(j));
        }
    }
}

// MorphSequence - Apply
void plMorphSequence::Apply() const
{
    const plDrawInterface* di = IGetDrawInterface();
    if( !di )
        return;

    // Reset to initial.
    Reset(di);

    // We'll be accumulating into the buffer, so open RW
    hsTArray<plAccessSpan> dst;
    plAccessGeometry::Instance()->OpenRW(di, dst);

    // For each MorphArray
    int i;
    for( i = 0; i < fMorphs.GetCount(); i++ )
    {
        // Apply Delta
        fMorphs[i].Apply(dst);
    }

    IRenormalize(dst);

    // Close up the access spans
    plAccessGeometry::Instance()->Close(dst);
}

// MorphSequence - Reset to initial
void plMorphSequence::Reset(const plDrawInterface* di) const
{
    if( !di )
    {
        di = IGetDrawInterface();
    }

    // Use access span RestoreSnapshot
    if( di )
        plAccessGeometry::Instance()->RestoreSnapShot(di, kChanMask);
}

const plDrawInterface* plMorphSequence::IGetDrawInterface() const
{
    plSceneObject* so = GetTarget();
    if( !so )
        return nil;

    const plDrawInterface* di = so->GetDrawInterface();

    return di;
}

void plMorphSequence::AddTarget(plSceneObject* so)
{
    plSingleModifier::AddTarget(so);

    if (!fMorphSDLMod)
    {
        fMorphSDLMod = new plMorphSequenceSDLMod;
        so->AddModifier(fMorphSDLMod);
    }
}

void plMorphSequence::RemoveTarget(plSceneObject *so)
{
    plSingleModifier::RemoveTarget(so);

    if (so)
        if (fMorphSDLMod)
            so->RemoveModifier(fMorphSDLMod);

    delete fMorphSDLMod;
    fMorphSDLMod = nil;     
}
    
void plMorphSequence::Read(hsStream* s, hsResMgr* mgr)
{
    plSingleModifier::Read(s, mgr);

    fMorphFlags = 0;

    int n = s->ReadLE32();
    fMorphs.SetCount(n);
    int i;
    for( i = 0; i < n; i++ )
        fMorphs[i].Read(s, mgr);
    
    n = s->ReadLE32();
    for( i = 0; i < n; i++)
        mgr->ReadKeyNotifyMe(s, new plGenRefMsg(GetKey(), plRefMsg::kOnCreate, -1, -1), plRefFlags::kActiveRef);
}

void plMorphSequence::Write(hsStream* s, hsResMgr* mgr)
{
    plSingleModifier::Write(s, mgr);

    s->WriteLE32(fMorphs.GetCount());
    int i;
    for( i = 0; i < fMorphs.GetCount(); i++ )
        fMorphs[i].Write(s, mgr);

    s->WriteLE32(fSharedMeshes.GetCount());
    for( i = 0; i < fSharedMeshes.GetCount(); i++ )
        mgr->WriteKey(s, fSharedMeshes[i].fMesh);
}

// Normal sequence of calls:
// 1) on notification that meshes have changed (or activate)
//      IFindIndices() - Sets up indices 
//      IApplyShared() - Find which mesh is active and
//          IResetShared(iActive);
//          IApplyShared(iActive);
//      go dormant
// 2) on weight change
//      SetWeight() - Register for render message and set the new weight
// 3) on render msg:
//      if dirty
//          IApplyShared() - Find which mesh is active and
//              IResetShared(iActive);
//              IApplyShared(iActive);
//      else
//          Unregister for render message.
// 4) on deinit
//      IReleaseIndices() - Just let's us know there's nothing much to be done.
//

void plMorphSequence::IResetShared()
{
    int i;
    for (i = 0; i < fSharedMeshes.GetCount(); i++)
        IResetShared(i);
}

void plMorphSequence::IApplyShared()
{
    int i;
    for (i = 0; i < fSharedMeshes.GetCount(); i++)
    {
        IResetShared(i);
        IApplyShared(i);
    }
}

void plMorphSequence::IFindIndices()
{
    fMorphFlags &= ~kDirtyIndices;

    int i;
    for( i = 0; i < fSharedMeshes.GetCount(); i++ )
        IFindIndices(i);
}

void plMorphSequence::IReleaseIndices()
{
    int i;
    for( i = 0; i < fSharedMeshes.GetCount(); i++ )
        IReleaseIndices(i);
}


void plMorphSequence::IApplyShared(int iShare)
{
    if( iShare >= fSharedMeshes.GetCount() || fSharedMeshes[iShare].fCurrDraw == nil)
        return;

    plSharedMeshInfo& mInfo = fSharedMeshes[iShare];

    hsTArray<plAccessSpan> dst;
    // Now copy each shared mesh geometryspan into the drawable
    // to get it back to it's pristine condition.
    int i;
    for( i = 0; i < mInfo.fMesh->fSpans.GetCount(); i++ )
    {
        plAccessSpan dstAcc;
        plAccessGeometry::Instance()->OpenRW(mInfo.fCurrDraw, mInfo.fCurrIdx[i], dstAcc);

        dst.Append(dstAcc);
    }

    // For each MorphArray
    for( i = 0; i < mInfo.fMesh->fMorphSet->fMorphs.GetCount(); i++ )
    {
        // Apply Delta
        mInfo.fMesh->fMorphSet->fMorphs[i].Apply(dst, &mInfo.fArrayWeights[i].fDeltaWeights);
    }

    IRenormalize(dst);

    // Close up the access spans
    plAccessGeometry::Instance()->Close(dst);
    mInfo.fFlags &= ~plSharedMeshInfo::kInfoDirtyMesh;
}

bool plMorphSequence::IResetShared(int iShare)
{
    if( iShare >= fSharedMeshes.GetCount() || fSharedMeshes[iShare].fCurrDraw == nil)
        return false;

    plSharedMeshInfo& mInfo = fSharedMeshes[iShare];

    // Now copy each shared mesh geometryspan into the drawable
    // to get it back to it's pristine condition.
    int i;
    for( i = 0; i < mInfo.fMesh->fSpans.GetCount(); i++ )
    {
        plAccessSpan srcAcc;
        plAccessGeometry::Instance()->AccessSpanFromGeometrySpan(srcAcc, mInfo.fMesh->fSpans[i]);
        plAccessSpan dstAcc;
        plAccessGeometry::Instance()->OpenRW(mInfo.fCurrDraw, mInfo.fCurrIdx[i], dstAcc);
        
        plAccPosNormUVWIterator srcIter(&srcAcc.AccessVtx());
        plAccPosNormUVWIterator dstIter(&dstAcc.AccessVtx());

        const int numUVWs = srcAcc.AccessVtx().NumUVWs();

        for( srcIter.Begin(), dstIter.Begin(); srcIter.More(); srcIter.Advance(), dstIter.Advance() )
        {
            *dstIter.Position() = *srcIter.Position();
            *dstIter.Normal() = *srcIter.Normal();

            int j;
            for( j = 0; j < numUVWs; j++ )
                *dstIter.UVW(j) = *srcIter.UVW(j);
        }
    }

    return true;
}

bool plMorphSequence::IFindIndices(int iShare)
{
    plSharedMeshInfo& mInfo = fSharedMeshes[iShare];
    mInfo.fCurrDraw = nil; // In case we fail.

    const plInstanceDrawInterface* di = plInstanceDrawInterface::ConvertNoRef(IGetDrawInterface());
    if( !di )
        return false;

    int32_t meshIdx = di->GetSharedMeshIndex(mInfo.fMesh);
    if( meshIdx < 0 )
        return false;

    plDrawableSpans* dr = plDrawableSpans::ConvertNoRef(di->GetDrawable((uint8_t)meshIdx));
    if( !dr )
        return false;

    mInfo.fCurrDraw = dr;

    plDISpanIndex& diIndex = dr->GetDISpans(di->GetDrawableMeshIndex((uint8_t)meshIdx));

    hsAssert(mInfo.fMesh->fSpans.GetCount() == diIndex.GetCount(), "Mismatch between geometry and indices");

    mInfo.fCurrIdx.SetCount(diIndex.GetCount());

    int i;
    for( i = 0; i < diIndex.GetCount(); i++ )
        mInfo.fCurrIdx[i] = diIndex[i];

    return true;
}

void plMorphSequence::IReleaseIndices(int iShare)
{
    plSharedMeshInfo& mInfo = fSharedMeshes[iShare];
    mInfo.fCurrDraw = nil;
}

int32_t plMorphSequence::IFindSharedMeshIndex(plKey meshKey) const
{
    int i;
    for( i = 0; i < fSharedMeshes.GetCount(); i++ )
    {
        if (fSharedMeshes[i].fMesh->GetKey() == meshKey)
            return i;
    }

    return -1;
}

int32_t plMorphSequence::IFindPendingStateIndex(plKey meshKey) const
{
    int i;
    for( i = 0; i < fPendingStates.GetCount(); i++ )
    {
        if (fPendingStates[i].fSharedMeshKey == meshKey)
            return i;
    }
    
    return -1;
}

bool plMorphSequence::IIsUsingDrawable(plDrawable *draw)
{
    int i;
    for (i = 0; i < fSharedMeshes.GetCount(); i++)
    {
        if (fSharedMeshes[i].fCurrDraw == draw)
            return true;
    }

    return false;
}

void plMorphSequence::AddSharedMesh(plSharedMesh* mesh)
{
    // This happens if SDL state tells us to use a mesh without morph info
    // (Either because the object used to but no longer does, or because
    // the SDL is giving us the wrong key.)
    if (!mesh->fMorphSet)
        return;

    if (fSharedMeshes.GetCount() == 0)
        plgDispatch::Dispatch()->RegisterForExactType(plSharedMeshBCMsg::Index(), GetKey());

    if (IFindSharedMeshIndex(mesh->GetKey()) >= 0)
        return; // We already have it.

    hsAssert(fSharedMeshes.GetCount() < 127, "Too many meshes for one morph sequence.");
    SetUseSharedMesh(true); 
    int pendingIndex = IFindPendingStateIndex(mesh->GetKey());

    plSharedMeshInfo mInfo;
    mInfo.fMesh = mesh;
    mInfo.fCurrDraw = nil;

    // Intialize our weights to zero.
    mInfo.fArrayWeights.Reset();
    mInfo.fArrayWeights.SetCount(mesh->fMorphSet->fMorphs.GetCount());
    int i, j;
    for (i = 0; i < mesh->fMorphSet->fMorphs.GetCount(); i++)
        mInfo.fArrayWeights[i].fDeltaWeights.ExpandAndZero(mesh->fMorphSet->fMorphs[i].GetNumDeltas());
            
    // Aha, we have some pending weights. Copy them in!
    if (pendingIndex >= 0)
    {
        // Filter in any data that's valid
        for (i = 0; i < mInfo.fArrayWeights.GetCount() && i < fPendingStates[pendingIndex].fArrayWeights.GetCount(); i++)
        {
            for (j = 0; j < mInfo.fArrayWeights[i].fDeltaWeights.GetCount() && 
                        j < fPendingStates[pendingIndex].fArrayWeights[i].fDeltaWeights.GetCount(); j++)
            {
                mInfo.fArrayWeights[i].fDeltaWeights[j] = fPendingStates[pendingIndex].fArrayWeights[i].fDeltaWeights[j];
            }
        }
            
        fPendingStates.Remove(pendingIndex);

        mInfo.fFlags |= plSharedMeshInfo::kInfoDirtyMesh;
        ISetDirty(true);
    }

    fSharedMeshes.Append(mInfo);
    IFindIndices(fSharedMeshes.GetCount() - 1);

    if (mesh->fFlags & plSharedMesh::kLayer0GlobalToMod)
    {
        fGlobalLayerRef = fSharedMeshes.GetCount() - 1;
        ISetAllSharedToGlobal();
    }
    else 
        ISetSingleSharedToGlobal(fSharedMeshes.GetCount() - 1);
}

void plMorphSequence::RemoveSharedMesh(plSharedMesh* mesh)
{
    int idx = IFindSharedMeshIndex(mesh->GetKey());
    if (idx < 0)
        return; // Don't have it... can't remove it can we?
    
    fSharedMeshes.Remove(idx);

    fGlobalLayerRef = -1;
    int i;
    for (i = 0; i < fSharedMeshes.GetCount(); i++)
    {
        if (fSharedMeshes[i].fMesh->fFlags & plSharedMesh::kLayer0GlobalToMod)
        {
            fGlobalLayerRef = i;
            break;
        }
    }
    
    if (fSharedMeshes.GetCount() == 0)
        plgDispatch::Dispatch()->UnRegisterForExactType(plSharedMeshBCMsg::Index(), GetKey());
}

void plMorphSequence::FindMorphMods(const plSceneObject *so, hsTArray<const plMorphSequence*> &mods)
{
    const plMorphSequence *morph = plMorphSequence::ConvertNoRef(so->GetModifierByType(plMorphSequence::Index()));
    if (morph)
        mods.Append(morph);
    
    const plCoordinateInterface *ci = so->GetCoordinateInterface();
    int i;
    for (i = 0; i < ci->GetNumChildren(); i++)
        FindMorphMods(ci->GetChild(i)->GetOwner(), mods);
}

void plMorphSequence::ISetAllSharedToGlobal()
{
    int i;
    for (i = 0; i < fSharedMeshes.GetCount(); i++)
    {
        if (i != fGlobalLayerRef)
            ISetSingleSharedToGlobal(i);
    }
}

void plMorphSequence::ISetSingleSharedToGlobal(int idx)
{
    if (fGlobalLayerRef < 0)
        return;

    int i;
    for (i = 0; i < fSharedMeshes[fGlobalLayerRef].fArrayWeights[0].fDeltaWeights.GetCount(); i++)
        SetWeight(0, i, fSharedMeshes[fGlobalLayerRef].fArrayWeights[0].fDeltaWeights[i], fSharedMeshes[idx].fMesh->GetKey());
}