/*==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==*/
///////////////////////////////////////////////////////////////////////////////
//                                                                           //
//  plRealTimeLights.cpp - Implementation for some MAX RT lights             //
//  Cyan, Inc.                                                               //
//                                                                           //
//// Version History //////////////////////////////////////////////////////////
//                                                                           //
//  8.2.2001 mcn - Created.                                                  //
//                                                                           //
///////////////////////////////////////////////////////////////////////////////

#include "HeadSpin.h"
#include "plRealTimeLights.h"
#include "iparamm2.h"
#include "MaxPlasmaMtls/Layers/plLayerTex.h"
#include "MaxPlasmaMtls/Layers/plLayerTexBitmapPB.h"
#include "MaxComponent/plMaxAnimUtils.h"
#include "plRTLightBaseAnimDlgProc.h"
#include "plGLight/plLightKonstants.h"

void DummyLightCodeIncludeFunc() {}


//// Static ClassDesc2 Get Functions //////////////////////////////////////////

plRTOmniLightDesc   plRTOmniLightDesc::fStaticDesc;
plRTSpotLightDesc   plRTSpotLightDesc::fStaticDesc;
plRTDirLightDesc    plRTDirLightDesc::fStaticDesc;


//// ParamBlock2 DlgProc Functions ////////////////////////////////////////////

const char* DecayLevel[] = {
                            "None",
                            "Inverse",
                            "Inverse Square",
                            NULL
                            };

const char* ShadowState[] = {
                            "Shadow Map",
                            NULL
                            };


class LightDlgProc : public plBaseLightProc
{
protected:
    void    IValidateSpinners( TimeValue t, WPARAM wParam, IParamBlock2 *pb, IParamMap2 *map )
    {
        /// Make sure falloff is >= hotspot (adjust the one we're not editing)
        float   hotspot, falloff;
        hotspot = pb->GetFloat( plRTLightBase::kHotSpot, t );
        falloff = pb->GetFloat( plRTLightBase::kFallOff, t );

        if( falloff < hotspot )
        {
            if( LOWORD( wParam ) == IDC_LHOTSIZESPINNER )
                pb->SetValue( plRTLightBase::kFallOff, t, hotspot );
            else
                pb->SetValue( plRTLightBase::kHotSpot, t, falloff );

            map->Invalidate( plRTLightBase::kHotSpot );
            map->Invalidate( plRTLightBase::kFallOff );
        }
        IBuildLightMesh( (plRTLightBase *)pb->GetOwner(), falloff );
    }

public:
    BOOL DlgProc(TimeValue t, IParamMap2 *map, HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam)
    {
        IParamBlock2 *pb = map->GetParamBlock();
        plRTLightBase *gl = (plRTLightBase *) pb->GetOwner();

        switch (msg)
        {
            case WM_COMMAND:
                if(LOWORD(wParam) == IDC_PROJ_MAPNAME )
                {
                    if( gl->ClassID() != RTOMNI_LIGHT_CLASSID )
//                  if(pb->GetInt(plRTLightBase::kLightType) != plRTLightBase::RT_OMNI)
                    
                    //  map->SetValue(plRTSpotLight::kProjMapTexButton, t, 
                        //gl->SetProjMap(
                        map->Invalidate(plRTSpotLight::kProjMapTexButton);
                    return false;
                }
                else if( LOWORD( wParam ) == IDC_LHOTSIZE || LOWORD( wParam ) == IDC_LFALLOFF )
                {
                    if( HIWORD( wParam ) == EN_CHANGE )
                        IValidateSpinners( t, wParam, pb, map );
                }
                break;

            case CC_SPINNER_CHANGE:      
                if( LOWORD( wParam ) == IDC_LHOTSIZESPINNER || LOWORD( wParam ) == IDC_LFALLOFFSPINNER )
                    IValidateSpinners( t, wParam, pb, map );
                break;
        }

        return plBaseLightProc::DlgProc( t, map, hWnd, msg, wParam, lParam );;
    }
    void DeleteThis() {};
};
static LightDlgProc gLiteDlgProc;


#include "plRealTimeLightsPBDec.h"


#include "plRTObjLightDesc.h"


///////////////////////////////////////////////////////////////////////
//
//  Light Descriptors declared below for the different plRTLights...
//
//
//
//

//--- Base Light Class derived from the ObjLightDesc 

#define COS_45 0.7071067f
#define COS_45_2X 1.4142136f

static float stepFactor[] = {50.0f,80.0f,140.0f};
#define MAXSTEPS 1000


BaseObjLight::BaseObjLight(INode *n) : ObjLightDesc(n) 
{
    ObjectState os = n->EvalWorldState(TimeValue(0));
    assert(os.obj->SuperClassID()==LIGHT_CLASS_ID);
    gl = (os.obj->GetInterface(I_MAXSCRIPTPLUGIN) != NULL) ? (plRTLightBase*)os.obj->GetReference(0) : (plRTLightBase*)os.obj;  // JBW 4/7/99
}   

static Color blackCol(0,0,0);

int BaseObjLight::Update(TimeValue t, const RendContext& rc, RenderGlobalContext * rgc, BOOL shadows, BOOL shadowGeomChanged) {
    ObjLightDesc::Update(t,rc,rgc,shadows,shadowGeomChanged);
    intensCol = ls.intens*ls.color*rc.GlobalLightLevel();
    ObjectState os = inode->EvalWorldState(t);
    plRTLightBase* lob = (plRTLightBase *)os.obj;       
    contrast = 0; //lob->GetParamBlock2()->GetFloat(plRTLightBase::kContrast, t);
    diffSoft = 0; //lob->GetDiffuseSoft(t)/100.0f;

    float a = .5; //contrast/204.0f + 0.5f; // so "a" varies from .5 to .99
    kA = (2.0f-1.0f/a);
    kB = 1.0f-kA;
    diffSoften = false; //lob->GetParamBlock2()->GetInt(plRTLightBase::kDiffOn, t);     //GetSoftenDiffuse();

    return 1;
}






//--- Omni Light ------------------------------------------------


OmniLight::OmniLight(INode *inode, BOOL forceShadowBuf ) : BaseObjLight(inode){

    //projector = /*doShadows =  shadowRay =*/  FALSE; 
    //projMap = NULL;
    needMultiple = FALSE;

}

OmniLight::~OmniLight() 
{

}




int OmniLight::UpdateViewDepParams(const Matrix3& worldToCam) {
    BaseObjLight::UpdateViewDepParams(worldToCam);
    return 1;
    }

static Point3 MapToDir(Point3 p, int k) {
    switch(k) {
        case 0: return Point3(  p.z, p.y, -p.x); // +X
        case 1: return Point3( -p.z, p.y,  p.x); // -X
        case 2: return Point3(  p.x, p.z, -p.y); // +Y 
        case 3: return Point3(  p.x,-p.z,  p.y); // -Y
        case 4: return Point3( -p.x, p.y, -p.z); // +Z
        case 5: return p;                        // -Z
        }
    return p;
    }

static void GetMatrixForDir(Matrix3 &origm, Matrix3 &tm, int k ) {
    tm = origm;
    switch(k) {
        case 0: tm.PreRotateY(-HALFPI); break;  // Map 0: +X axis   
        case 1: tm.PreRotateY( HALFPI); break;  // Map 1: -X axis   
        case 2: tm.PreRotateX( HALFPI); break;  // Map 2: +Y axis   
        case 3: tm.PreRotateX(-HALFPI); break;  // Map 3: -Y axis   
        case 4: tm.PreRotateY(   PI  ); break;  // Map 4: +Z axis   
        case 5:                         break;  // Map 5: -Z axis   
        }
    }

static int WhichDir(Point3 &p) {
    int j = MaxComponent(p);  // the component with the maximum abs value
    return  (p[j]<0.0f) ? 2*j+1 : 2*j;
    }

int OmniLight::Update(TimeValue t, const RendContext & rc,
        RenderGlobalContext *rgc, BOOL shadows, BOOL shadowGeomChanged)
{
    BaseObjLight::Update(t,rc,rgc,shadows,shadowGeomChanged);   

    ObjectState os = inode->EvalWorldState(t);
    LightObject* lob = (LightObject *)os.obj;       
    assert(os.obj->SuperClassID()==LIGHT_CLASS_ID);
    plRTOmniLight* gl = (lob->GetInterface(I_MAXSCRIPTPLUGIN) != NULL) ? (plRTOmniLight*)lob->GetReference(0) : (plRTOmniLight*)lob;  // JBW 4/7/99

    decayType = gl->GetDecayType(); 
    decayRadius = gl->GetDecayRadius(t);

    fov = HALFPI; // 90 degree fov
    int res=1;
    if(gl->GetTex())
        gl->GetTex()->Update(t, FOREVER);
    //projector =  gl->GetProjector();
    //if (projector){
    //  projMap = gl->GetProjMap();
    //  if( projMap ) projMap->Update(t,FOREVER);
    //}

    return res;
}



////------------------------------------------------------------------
//      
//
//          SpotLight descriptors.....
//
//
//
//
//

SpotLight::SpotLight(INode *inode, BOOL forceShadowBuf ):BaseObjLight(inode) 
{
    projMap = NULL;
}

int SpotLight::Update(TimeValue t, const RendContext &rc, RenderGlobalContext *rgc, BOOL shadows, BOOL shadowGeomChanged)
{
    int res = 1;
    BaseObjLight::Update(t,rc,rgc,shadows, shadowGeomChanged);

    float hs = DegToRad(ls.hotsize);
    float fs = DegToRad(ls.fallsize);
    fall_tan = (float)tan(fs/2.0f);
    hot_cos = (float)cos(hs/2.0f);
    fall_cos =(float)cos(fs/2.0f);
    fall_sin = (float)sin(fs/2.0f);
    hotpct = ls.hotsize/ls.fallsize;
    ihotpct = 1.0f - hotpct;        

    ObjectState os = inode->EvalWorldState(t);
    LightObject* lob = (LightObject *)os.obj;       
    assert(os.obj->SuperClassID()==LIGHT_CLASS_ID);
    plRTLightBase* gl = (lob->GetInterface(I_MAXSCRIPTPLUGIN) != NULL) ? (plRTLightBase*)lob->GetReference(0) : (plRTLightBase*)lob;  // JBW 4/7/99

    decayType = gl->GetDecayType(); 
    decayRadius = gl->GetDecayRadius(t);

    projector =  gl->GetProjector();
    fov = hsMaximum(fs,hs);

    float aspect = 1.0f;
     
    fov = 2.0f* (float)atan(tan(fov*0.5f)*sqrt(aspect));
    zfac = -sz2 /(float)tan(0.5*(double)fov);
    xscale = zfac;                              
    yscale = -zfac*aspect;
    curve =(float)fabs(1.0f/xscale); 

    rectv0.y = fall_sin * (float)sqrt(aspect);
    rectv1.y = fall_sin / (float)sqrt(aspect);

    rectv0.x = rectv1.x = fall_cos;
    rectv0 = Normalize(rectv0);
    rectv1 = Normalize(rectv1);

    Interval v;
    if (projector){
        projMap = gl->GetProjMap();
        if( projMap ) projMap->Update(t,v);
    }

    return res;
}

int  SpotLight::UpdateViewDepParams(const Matrix3& worldToCam) 
{
    BaseObjLight::UpdateViewDepParams(worldToCam);
    lightDir = -FNormalize(lightToCam.GetRow(2));
    return 1;
}

BOOL SpotLight::IsFacingLight(Point3 &dir)
{
    return dir.z>0.0f;
}



//--- Directional Light ------------------------------------------------

DirLight::DirLight(INode *inode, BOOL forceShadowBuf ) : BaseObjLight(inode)
{
    projMap = NULL;
}

int DirLight::Update(TimeValue t, const RendContext &rc, 
        RenderGlobalContext *rgc, BOOL shadows, BOOL shadowGeomChanged)
{
    int res = 1;
    BaseObjLight::Update(t,rc,rgc,shadows,shadowGeomChanged);
    hotsz = ls.hotsize;
    fallsz = ls.fallsize;
    fallsq = fallsz*fallsz;
    hotpct = ls.hotsize/ls.fallsize;
    ihotpct = 1.0f - hotpct;

    ObjectState os = inode->EvalWorldState(t);
    LightObject* lob = (LightObject *)os.obj;       
    assert(os.obj->SuperClassID()==LIGHT_CLASS_ID);
    plRTDirLight* gl = (lob->GetInterface(I_MAXSCRIPTPLUGIN) != NULL) ? (plRTDirLight*)lob->GetReference(0) : (plRTDirLight*)lob;  // JBW 4/7/99

    //projector =  gl->GetProjector();

    aspect = 1.0f;

    //if (projector){
    //      projMap = gl->GetProjMap();
    //      if( projMap ) projMap->Update(t,FOREVER);
    //  }
    return res;
};

int DirLight::UpdateViewDepParams(const Matrix3& worldToCam) {
    BaseObjLight::UpdateViewDepParams(worldToCam);
    lightDir = FNormalize(lightToCam.GetRow(2));

    return 1;
    }







////////////////////////////////////////////////////////////////////////////////////
//
//  plRTOmni Stuff below
//
//
//
//
//

plRTOmniLight::plRTOmniLight()
{
    fIP = NULL; 
    fLightPB = NULL; 
    fClassDesc = plRTOmniLightDesc::GetDesc();
    fClassDesc->MakeAutoParamBlocks(this);

    fLightPB->SetValue(kLightColor, 0,  Color(255,255,255));
    SetHSVColor(0, Point3(255, 255, 255));
    
    fTex = NULL;

    meshBuilt = 0; 
    
    IBuildMeshes(true);
}

ObjLightDesc *plRTOmniLight::CreateLightDesc(INode *n, BOOL forceShadowBuf)
{
    return TRACKED_NEW OmniLight( n, forceShadowBuf );
}


RefTargetHandle plRTOmniLight::Clone(RemapDir &remap)
{

    plRTOmniLight *obj = TRACKED_NEW plRTOmniLight;//(plRTLightBase*) fClassDesc->Create(false);
    obj->ReplaceReference(kRefSpotLight, fLightPB->Clone(remap));
    BaseClone(this, obj, remap);

    return obj;
}

void plRTOmniLight::IBuildMeshes( BOOL isnew ) 
{
    BuildStaticMeshes();
    fMesh = staticMesh[ plRTLightBase::RT_OMNI ];
}

//// DrawConeAndLine /////////////////////////////////////////////////////////

int     plRTOmniLight::DrawConeAndLine( TimeValue t, INode* inode, GraphicsWindow *gw, int drawing ) 
{
    float   atOneHalfDist;
    Matrix3 tm = inode->GetObjectTM( t );


    gw->setTransform( tm );
    gw->clearHitCode();

    if( ( extDispFlags & EXT_DISP_ONLY_SELECTED ) )
    {
        if( GetUseAtten() )
        {
            // Draw hotspot as the point at which light is 1/2 intensity (just to help the visual)
            gw->setColor( LINE_COLOR, GetUIColor( COLOR_HOTSPOT ) );

            if( fLightPB->GetInt( kAttenTypeRadio, t ) == 0 )
                atOneHalfDist = GetAtten( t, ATTEN_END ) / ( fLightPB->GetFloat( kIntensity, t ) * plSillyLightKonstants::GetFarPowerKonst() - 1.f );
            else
                atOneHalfDist = sqrt( GetAtten( t, ATTEN_END ) * GetAtten( t, ATTEN_END ) / ( fLightPB->GetFloat( kIntensity, t ) * plSillyLightKonstants::GetFarPowerKonst() - 1.f ) );

            if( atOneHalfDist > 0.0f )
                DrawCone( t, gw, atOneHalfDist );

            gw->setColor( LINE_COLOR, GetUIColor( COLOR_FALLOFF ) );
            DrawCone( t, gw, GetAtten( t, ATTEN_END ) );
        }
        else
            DrawArrows( t, gw, 50 );
    }

    return gw->checkHitCode();
}

//// DrawCone ////////////////////////////////////////////////////////////////
//  Function called by MAX to render the cone shape in the viewport for this
//  light. Note that this is the cone, not the actual object itself!

void    plRTOmniLight::DrawCone( TimeValue t, GraphicsWindow *gw, float dist ) 
{
    Point3  pts[ NUM_CIRC_PTS * 3 + 1 ];


    /// Draw sphere-thingy
    GetAttenPoints( t, dist, pts );

    gw->polyline( NUM_CIRC_PTS, pts,                    nil, nil, true, nil );
    gw->polyline( NUM_CIRC_PTS, pts + NUM_CIRC_PTS,     nil, nil, true, nil );
    gw->polyline( NUM_CIRC_PTS, pts + 2 * NUM_CIRC_PTS, nil, nil, true, nil );
}


//// DrawArrows //////////////////////////////////////////////////////////////
//  Renders some arrows in all directions, to show a radiating, attenuation-less
//  omni light.

void    plRTOmniLight::DrawArrows( TimeValue t, GraphicsWindow *gw, float dist ) 
{
    Point3  directions[] = { Point3( 1, 0, 0 ), Point3( -1, 0, 0 ), Point3( 0, 1, 0 ), Point3( 0, -1, 0 ),
                             Point3( 0, 0, 1 ), Point3( 0, 0, -1 ), 
                             Point3( 2, 2, 0 ), Point3( 2, -2, 0 ), Point3( 2, 0, 2 ), Point3( 2, 0, -2 ),
                             Point3( -2, 2, 0 ), Point3( -2, -2, 0 ), Point3( -2, 0, 2 ), Point3( -2, 0, -2 ),
                             Point3( 0, 2, 2 ), Point3( 0, 2, -2 ), Point3( 0, -2, 2 ), Point3( 0, -2, -2 ),
                             Point3( 0, 0, 0 ) };
    Point3  empty( 0, 0, 0 );
    int     i;
    Point3  pts[ 5 ];


    /// Adjust directions
    for( i = 0; directions[ i ] != empty; i++ )
    {
        if( directions[ i ].x == 2.f )
            directions[ i ].x = 0.7f;
        else if( directions[ i ].x == -2.f )
            directions[ i ].x = -0.7f;

        if( directions[ i ].y == 2.f )
            directions[ i ].y = 0.7f;
        else if( directions[ i ].y == -2.f )
            directions[ i ].y = -0.7f;

        if( directions[ i ].z == 2.f )
            directions[ i ].z = 0.7f;
        else if( directions[ i ].z == -2.f )
            directions[ i ].z = -0.7f;
    }

    /// Draw da arrows
    gw->setColor( LINE_COLOR, GetUIColor( COLOR_HOTSPOT ) );
    for( i = 0; directions[ i ] != empty; i++ )
        DrawArrow( t, gw, directions[ i ], dist );
}

//// GetLocalBoundBox ////////////////////////////////////////////////////////

void    plRTOmniLight::GetLocalBoundBox( TimeValue t, INode *node, ViewExp *vpt, Box3 &box )
{
    Point3  loc = node->GetObjectTM( t ).GetTrans();
    float   scaleFactor = vpt->NonScalingObjectSize() * vpt->GetVPWorldWidth( loc ) / 360.0f;
    float   width;

    box = fMesh.getBoundingBox();
    // Because we want to scale about the origin, not the box center, we have to do this funky offset
    Point3  boxCenter = box.Center();
    box.Translate( -boxCenter );
    box.Scale( scaleFactor );
    boxCenter *= scaleFactor;
    box.Translate( boxCenter );

    // Include points for the spotlight. That means either the attenuated cone or 
    // our unattenuated cone display
    if( ( extDispFlags & EXT_DISP_ONLY_SELECTED ) )
    {
        if( GetUseAtten() ) 
            width = GetAtten( t, ATTEN_END );
        else
            width = 50.f;       // Include our arrows

        box += Point3( -width, -width, -width );
        box += Point3( width, width, width );
    }
}

////////////////////////////////////////////////////////////////////////////////////////
//  
//
//          SpotLight Stuff
//
//


//// Local Copy of shared Anim PB /////////////////////////////////////////////

plRTSpotLight::plRTSpotLight()
{
    fIP = NULL; 
    fLightPB = NULL; 
    fClassDesc = plRTSpotLightDesc::GetDesc();
    fClassDesc->MakeAutoParamBlocks(this);

    fLightPB->SetValue(kLightColor, 0,  Color(255,255,255));
    SetHSVColor(0, Point3(255, 255, 255));
    
    fTex = NULL;
    meshBuilt = 0; 
    
    IBuildMeshes(true);
}

ObjLightDesc *plRTSpotLight::CreateLightDesc(INode *n, BOOL forceShadowBuf)
{
    return TRACKED_NEW SpotLight( n, forceShadowBuf );
}


RefTargetHandle plRTSpotLight::Clone(RemapDir &remap)
{

    plRTSpotLight *obj = TRACKED_NEW plRTSpotLight;//(plRTLightBase*) fClassDesc->Create(false);
    obj->ReplaceReference(kRefSpotLight, fLightPB->Clone(remap));
    BaseClone(this, obj, remap);
    return obj;
}

Texmap  *plRTSpotLight::GetProjMap()
{
    if( !fLightPB->GetInt( kUseProjectorBool ) )
        return nil;

    Interval valid = Interval(0,0); 
    if( !GetTex() )
    {
        if( fLightPB->GetInt( kUseProjectorBool ) )
        {
            PBBitmap* bitmap = fLightPB->GetBitmap( kProjMapTexButton, 0 );         
            SetProjMap( &bitmap->bi );
        }
    }

    if( GetTex() )
    {
        const char* dbgTexName = GetTex()->GetName();

        IParamBlock2 *bitmapPB = fTex->GetParamBlockByID(plLayerTex::kBlkBitmap);
        hsAssert(bitmapPB, "LayerTex with no param block");

        int noCompress = fLightPB->GetInt(kProjNoCompress);
        int noMip = fLightPB->GetInt(kProjNoMip);
        bitmapPB->SetValue(kBmpNonCompressed, TimeValue(0), noCompress);
        bitmapPB->SetValue(kBmpNoFilter, TimeValue(0), noMip);
    }

    return (Texmap *)GetTex();
}

void    plRTSpotLight::IBuildMeshes( BOOL isnew ) 
{
    float val = fLightPB->GetFloat( kHotSpot, TimeValue(0) );       //Init val of HotSpot
    if( isnew ) 
    {
        val = fLightPB->GetFloat(kHotSpot, TimeValue(0));
        SetHotspot(  TimeValue(0), val);
        //val = 45.0;
        val = fLightPB->GetFloat(kFallOff, TimeValue(0));
        SetFallsize(  TimeValue(0), val);
        val = fLightPB->GetFloat(kAttenMaxFalloffEdit, TimeValue(0)); //fLightPB->GetFloat(kTargetDist, TimeValue(0));
        if(val < 1.0f)
            SetTDist(  TimeValue(0), DEF_TDIST);
        else
            SetTDist(  TimeValue(0), val);

        val = fLightPB->GetFloat(kHotSpot, TimeValue(0));
    
    }

    BuildSpotMesh( val );

    fMesh = spotMesh;
}

//// DrawConeAndLine /////////////////////////////////////////////////////////

int     plRTSpotLight::DrawConeAndLine( TimeValue t, INode* inode, GraphicsWindow *gw, int drawing ) 
{
    Matrix3 tm = inode->GetObjectTM( t );


    gw->setTransform( tm );
    gw->clearHitCode();

    if( extDispFlags & EXT_DISP_ONLY_SELECTED )
        DrawCone( t, gw, GetAtten( t, ATTEN_END ) );

    return gw->checkHitCode();
}

//// DrawCone ////////////////////////////////////////////////////////////////
//  Function called by MAX to render the cone shape in the viewport for this
//  light. Note that this is the cone, not the actual object itself!

void    plRTSpotLight::DrawCone( TimeValue t, GraphicsWindow *gw, float dist ) 
{
    int     i;
    Point3  pts[ NUM_CIRC_PTS + 1 ], u[ 3 ];


    if( !GetUseAtten() )
    {
        /// Don't use atten, but still want a cone, so draw the cone w/ a dist of 100
        /// and the lines extending past it (thus indicating that it keeps going)
        dist = 100;
    }

    /// Draw hotspot cone
    gw->setColor( LINE_COLOR, GetUIColor( COLOR_HOTSPOT ) );
    GetConePoints( t, -1.0f, GetHotspot( t ), dist, pts );
    gw->polyline( NUM_CIRC_PTS, pts, nil, nil, true, nil );

    if( GetUseAtten() )
    {
        u[ 0 ] = Point3( 0, 0, 0 );
        for( i = 0; i < NUM_CIRC_PTS; i += SEG_INDEX )
        {
            u[ 1 ] = pts[ i ];
            gw->polyline( 2, u, nil, nil, true, nil );
        }
    }
    else
    {
        for( i = 0; i < NUM_CIRC_PTS; i += SEG_INDEX )
        {
            pts[ i ] = pts[ i ].Normalize();
            DrawArrow( t, gw, pts[ i ], dist + 50.f );
        }
    }

    /// Draw falloff cone if necessary
    if( GetHotspot( t ) < GetFallsize( t ) )
    {
        gw->setColor( LINE_COLOR, GetUIColor( COLOR_FALLOFF ) );
        GetConePoints( t, -1.0f, GetFallsize( t ), dist, pts );
        gw->polyline( NUM_CIRC_PTS, pts, nil, nil, true, nil );

        if( GetUseAtten() )
        {
            u[ 0 ] = Point3( 0, 0, 0 );
            for( i = 0; i < NUM_CIRC_PTS; i += SEG_INDEX )
            {
                u[ 1 ] = pts[ i ];
                gw->polyline( 2, u, nil, nil, true, nil );
            }
        }
        else
        {
            for( i = 0; i < NUM_CIRC_PTS; i += SEG_INDEX )
            {
                pts[ i ] = pts[ i ].Normalize();
                DrawArrow( t, gw, pts[ i ], dist + 50.f );
            }
        }
    }
}

//// GetLocalBoundBox ////////////////////////////////////////////////////////

void    plRTSpotLight::GetLocalBoundBox( TimeValue t, INode *node, ViewExp *vpt, Box3 &box )
{
    Point3  loc = node->GetObjectTM( t ).GetTrans();
    float   scaleFactor = vpt->NonScalingObjectSize() * vpt->GetVPWorldWidth( loc ) / 360.0f;
    float   width, depth;

    box = fMesh.getBoundingBox();
    // Because we want to scale about the origin, not the box center, we have to do this funky offset
    Point3  boxCenter = box.Center();
    box.Translate( -boxCenter );
    box.Scale( scaleFactor );
    boxCenter *= scaleFactor;
    box.Translate( boxCenter );

    // Include points for the spotlight. That means either the attenuated cone or 
    // our unattenuated cone display
    if( ( extDispFlags & EXT_DISP_ONLY_SELECTED ) )
    {
        if( GetUseAtten() ) 
            depth = GetAtten( t, ATTEN_END );
        else
            depth = 100.f + 50.f;   // Include arrows

        width = depth * tan( DegToRad( GetFallsize( t ) / 2.f ) );

        box += Point3( -width, -width, 0.f );
        box += Point3( width, width, -depth );
    }
}

///////////////////////////////////////////////////////////////////////////////
//// Directional Light ////////////////////////////////////////////////////////
///////////////////////////////////////////////////////////////////////////////

plRTDirLight::plRTDirLight()
{
    fIP = NULL; 
    fLightPB = NULL; 
    fClassDesc = plRTDirLightDesc::GetDesc();
    fClassDesc->MakeAutoParamBlocks(this);

    fLightPB->SetValue(kLightColor, 0,  Color(255,255,255));
    SetHSVColor(0, Point3(255, 255, 255));
    
    fTex = NULL;
    meshBuilt = 0; 
    
    IBuildMeshes(true);
}

ObjLightDesc *plRTDirLight::CreateLightDesc(INode *n, BOOL forceShadowBuf)
{
    return TRACKED_NEW DirLight( n, forceShadowBuf );
}


RefTargetHandle plRTDirLight::Clone(RemapDir &remap)
{

    plRTDirLight *obj = TRACKED_NEW plRTDirLight;//(plRTLightBase*) fClassDesc->Create(false);
    obj->ReplaceReference(kRefDirLight, fLightPB->Clone(remap));
    BaseClone(this, obj, remap);
    return obj;
}

//// IBuildMeshes ////////////////////////////////////////////////////////////

void    plRTDirLight::IBuildMeshes( BOOL isnew ) 
{
    BuildStaticMeshes();

    fMesh = staticMesh[ plRTLightBase::RT_OMNI + 1 ];
}

//// DrawCone ////////////////////////////////////////////////////////////////
//  Function called by MAX to render the cone shape in the viewport for this
//  light. Note that this is the cone, not the actual object itself!

void    plRTDirLight::DrawCone( TimeValue t, GraphicsWindow *gw, float dist ) 
{
    Point3  arrow[ 7 ];
    int     i, j, r;
    float   d;
    const float spacing = 20.f;


    // Draw some funky arrows to represent our direction
    dist = 100.f;
    gw->setColor( LINE_COLOR, GetUIColor( COLOR_HOTSPOT ) );

    for( i = -2; i <= 2; i++ )
    {
        for( j = -2; j <= 2; j++ )
        {
            r = ( i * i ) + ( j * j );
            if( r <= 4 )
            {
                d = dist * ( 5 - r ) / 5;
                IBuildZArrow( i * spacing, j * spacing, -d, -10.f, arrow );
                gw->polyline( 6, arrow, nil, nil, true, nil );
            }
        }
    }
}

//// IBuildZArrow ////////////////////////////////////////////////////////////

void    plRTDirLight::IBuildZArrow( float x, float y, float zDist, float arrowSize, Point3 *pts )
{
    pts[ 0 ] = Point3( x, y, 0.f );
    pts[ 1 ] = Point3( x, y, zDist );
    pts[ 2 ] = Point3( x + arrowSize / 2.f, y, zDist - arrowSize );
    pts[ 3 ] = Point3( x, y, zDist - arrowSize );
    pts[ 4 ] = Point3( x, y + arrowSize / 2.f, zDist - arrowSize );
    pts[ 5 ] = Point3( x, y, zDist );
}

//// GetLocalBoundBox ////////////////////////////////////////////////////////

void    plRTDirLight::GetLocalBoundBox( TimeValue t, INode *node, ViewExp *vpt, Box3 &box )
{
    Point3  loc = node->GetObjectTM( t ).GetTrans();
    float   scaleFactor = vpt->NonScalingObjectSize() * vpt->GetVPWorldWidth( loc ) / 360.0f;
    float   width, height, depth;

    box = fMesh.getBoundingBox();
    // Because we want to scale about the origin, not the box center, we have to do this funky offset
    Point3  boxCenter = box.Center();
    box.Translate( -boxCenter );
    box.Scale( scaleFactor );
    boxCenter *= scaleFactor;
    box.Translate( boxCenter );

    if( ( extDispFlags & EXT_DISP_ONLY_SELECTED ) )
    {
        width = 2 * 20.f + ( 10.f / 2.f );  // Add in half arrow size
        height = 2 * 20.f + ( 10.f / 2.f );
        depth = 100.f;

        box += Point3( -width, -height, 0.f );
        box += Point3( width, height, -depth );
    }
}