/*==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 "max.h"
#include "resource.h"
#include "hsTemplates.h"
#include "plComponent.h"
#include "plComponentReg.h"
#include "plMiscComponents.h"
#include "plAnimComponent.h"
#include "plNotetrackAnim.h"

#include "plGUIComponents.h"

#include "plAudioComponents.h"

#include "../MaxMain/plPlasmaRefMsgs.h"

#include "../pnSceneObject/plSceneObject.h"
#include "../pnSceneObject/plCoordinateInterface.h"
#include "../pnSceneObject/plDrawInterface.h"
#include "../plDrawable/plDrawableSpans.h"
#include "../plDrawable/plGeometrySpan.h"
#include "../plSurface/plLayerInterface.h"
#include "../plSurface/plLayer.h"
#include "../plSurface/hsGMaterial.h"
#include "../plGImage/plMipmap.h"
#include "../plGImage/plDynamicTextMap.h"

#include "../plMessage/plLayRefMsg.h"
#include "../plMessage/plMatRefMsg.h"

#include "../MaxMain/plPluginResManager.h"


#include "plgDispatch.h"
#include "../pnMessage/plObjRefMsg.h"
#include "../pnMessage/plIntRefMsg.h"
#include "../pnMessage/plNodeRefMsg.h"


#include "../plScene/plSceneNode.h"
#include "../MaxConvert/hsConverterUtils.h"
#include "../MaxConvert/hsControlConverter.h"
#include "../MaxConvert/hsMaterialConverter.h"
#include "../MaxConvert/plLayerConverter.h"
#include "../plInterp/plController.h"
#include "../plInterp/plAnimEaseTypes.h"
#include "../MaxMain/plMaxNode.h"
#include "../pnKeyedObject/plKey.h"

// GUIDialog component.
#include "../plScene/plPostEffectMod.h"
#include "../pfGameGUIMgr/pfGameGUIMgr.h"
#include "../pfGameGUIMgr/pfGUIDialogMod.h"
#include "../pfGameGUIMgr/pfGUIControlMod.h"
#include "../pfGameGUIMgr/pfGUIControlHandlers.h"
#include "../pfGameGUIMgr/pfGUIButtonMod.h"
#include "../pfGameGUIMgr/pfGUIDraggableMod.h"
#include "../pfGameGUIMgr/pfGUIListBoxMod.h"
#include "../pfGameGUIMgr/pfGUITextBoxMod.h"
#include "../pfGameGUIMgr/pfGUIEditBoxMod.h"
#include "../pfGameGUIMgr/pfGUIUpDownPairMod.h"
#include "../pfGameGUIMgr/pfGUIKnobCtrl.h"
#include "../pfGameGUIMgr/pfGUITagDefs.h"
#include "../pfGameGUIMgr/pfGUIDragBarCtrl.h"
#include "../pfGameGUIMgr/pfGUICheckBoxCtrl.h"
#include "../pfGameGUIMgr/pfGUIRadioGroupCtrl.h"
#include "../pfGameGUIMgr/pfGUIDynDisplayCtrl.h"
#include "../pfGameGUIMgr/pfGUIMultiLineEditCtrl.h"
#include "../pfGameGUIMgr/pfGUIProgressCtrl.h"
#include "../pfGameGUIMgr/pfGUIClickMapCtrl.h"
#include "../pfGameGUIMgr/pfGUIPopUpMenu.h"

// Location Related
#include "../plAgeDescription/plAgeDescription.h"
#include "../MaxMain/plMaxCFGFile.h"
#include "../MaxMain/plAgeDescInterface.h"
#include "../plFile/hsFiles.h"

#include "../MaxConvert/plConvert.h"
#include "../MaxPlasmaMtls/Layers/plDynamicTextLayer.h"
#include "../MaxPlasmaMtls/Layers/plLayerTexBitmapPB.h"

#include "../MaxMain/plMaxAccelerators.h"

#include "plPickMaterialMap.h"

#include "../plInterp/plController.h"
#include "../plAvatar/plMatrixChannel.h"

#include "../MaxPlasmaMtls/Layers/plLayerTex.h"

#include "pfGUISkinComp.h"

#include "../plResMgr/plLocalization.h"

#include "plPickLocalizationDlg.h"

#include <vector>
#include <string>


void DummyCodeIncludeFuncGUI() {}



/////////////////////////////////////////////////////////////////////////////////////////////////
//// Helper Classes /////////////////////////////////////////////////////////////////////////////

//// Hit Callback for GUI Controls //////////////////////////////////////////////////////////////

class plGUICtrlHitCallback : public HitByNameDlgCallback
{
protected:
	INode*			fOwner;
	IParamBlock2*	fPB;
	ParamID			fNodeListID;
	BOOL			fRestrict;
	hsTArray<Class_ID>		fRestrictedIDs;
	TCHAR			fTitle[ 128 ];
	BOOL			fSingle;

public:
	plGUICtrlHitCallback( INode* owner, IParamBlock2 *pb, ParamID nodeListID, TCHAR *title = nil,
							BOOL restricted = FALSE, Class_ID rID = GUI_BUTTON_CLASSID, BOOL single = TRUE )
		: fOwner( owner ), fPB( pb ), fNodeListID( nodeListID ), fRestrict( restricted ), fSingle( single )

	{
		fRestrictedIDs.Append( rID );
		strcpy( fTitle, title );
	}

	plGUICtrlHitCallback( INode* owner, IParamBlock2 *pb, ParamID nodeListID, TCHAR *title, 
							hsTArray<Class_ID> &rID )
		: fOwner( owner ), fPB( pb ), fNodeListID( nodeListID ), fRestrict( true ), fSingle(TRUE)

	{
		for( int i = 0; i < rID.GetCount(); i++ )
			fRestrictedIDs.Append( rID[ i ] );

		strcpy( fTitle, title );
	}

	virtual TCHAR *dialogTitle() { return fTitle; }
	virtual TCHAR *buttonText() { return "OK"; }

	virtual int filter(INode *node)
	{
		if( node == fOwner )
			return FALSE;

		plComponentBase *comp = ((plMaxNodeBase*)node)->ConvertToComponent();

		// If this is an activator type component
		if( comp )
		{
			if( ( fRestrict && fRestrictedIDs.Find( comp->ClassID() ) != fRestrictedIDs.kMissingIndex )
				|| ( !fRestrict && plGUIControlBase::GetGUIComp( comp ) != nil ) )
			{

				// And this wouldn't create a cyclical reference (Max doesn't like those)
				if (comp->TestForLoop(FOREVER, fPB) == REF_FAIL)
					return FALSE;

				return TRUE;
			}
		}
		else if( fRestrict && fRestrictedIDs.Find( node->ClassID() ) != fRestrictedIDs.kMissingIndex )
		{
			return TRUE;
		}

		return FALSE;
	}

	virtual void proc(INodeTab &nodeTab)
	{
		if ( nodeTab.Count() > 0 )
		{
			if( fSingle )
				fPB->SetValue( fNodeListID, TimeValue(0), nodeTab[0] );
			else
				fPB->Append( fNodeListID, nodeTab.Count(), &nodeTab[0] );
		}
	}

	virtual BOOL showHiddenAndFrozen() { return TRUE; }
	virtual BOOL singleSelect() { return TRUE; }
};

//// Single GUI Control Dialog Proc /////////////////////////////////////////////////////////////

class plGUISingleCtrlDlgProc : public ParamMap2UserDlgProc
{
protected:
	ParamID			fNodeID;
	int				fDlgItem;
	TCHAR			fTitle[ 128 ];
	hsTArray<Class_ID>	fClassesToSelect;

	ParamMap2UserDlgProc	*fProcChain;

public:

	int		GetHandledDlgItem( void ) const { return fDlgItem; }

	static const Class_ID		kEndClassList; 

	plGUISingleCtrlDlgProc( ParamID nodeID, int dlgItem, TCHAR *title, Class_ID *restrict, ParamMap2UserDlgProc *parentProc = nil )
	{
		fNodeID = nodeID;
		fDlgItem = dlgItem;
		for( int i = 0; restrict[ i ] != kEndClassList; i++ )
			fClassesToSelect.Append( restrict[ i ] );
//		fClassToSelect = restrict;
		strcpy( fTitle, title );
		fProcChain = parentProc;
	}

	BOOL DlgProc( TimeValue t, IParamMap2 *map, HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam )
	{
		switch ( msg )
		{
			case WM_INITDIALOG:
				{
					IParamBlock2 *pb = map->GetParamBlock();

					INode *node = pb->GetINode( fNodeID );
					TSTR newName( node ? node->GetName() : "Pick" );
					::SetWindowText( ::GetDlgItem( hWnd, fDlgItem ), newName );
				}
				break;

			case WM_COMMAND:
				if( ( HIWORD( wParam ) == BN_CLICKED ) )
				{
					if( LOWORD( wParam ) == fDlgItem )
					{
						IParamBlock2 *pb = map->GetParamBlock();
						plGUICtrlHitCallback hitCB( (INode *)pb->GetOwner(), pb, fNodeID, fTitle, fClassesToSelect );
						GetCOREInterface()->DoHitByNameDialog( &hitCB );

						INode* node = pb->GetINode( fNodeID );
						TSTR newName( node ? node->GetName() : "Pick" );
						::SetWindowText( ::GetDlgItem(hWnd, fDlgItem ), newName );
						map->Invalidate( fNodeID );
						::InvalidateRect( hWnd, NULL, TRUE );
						return true;
					}
				}
				break;
		}

		if( fProcChain )
			fProcChain->DlgProc( t, map, hWnd, msg, wParam, lParam );

		return false;
	}

	void DeleteThis() {}
};

const Class_ID		plGUISingleCtrlDlgProc::kEndClassList = Class_ID(); 

Class_ID	sSkinClassesToSelect[] = { GUI_SKIN_CLASSID, plGUISingleCtrlDlgProc::kEndClassList };


//// Multiple GUI Control Dialog Proc ///////////////////////////////////////////////////////////

class plGUIMultipleCtrlDlgProc : public ParamMap2UserDlgProc
{
protected:
	hsTArray<plGUISingleCtrlDlgProc *>	fSingleProcs;
	hsTArray<ParamMap2UserDlgProc *>	fProcs;

public:

	plGUIMultipleCtrlDlgProc( plGUISingleCtrlDlgProc **singleProcs, ParamMap2UserDlgProc **procs=nil )
	{
		for( int i = 0; singleProcs[ i ] != nil; i++ )
			fSingleProcs.Append( singleProcs[ i ] );
		if ( procs )
		{
			for( int i = 0; procs[ i ] != nil; i++ )
				fProcs.Append( procs[ i ] );
		}
	}

	BOOL DlgProc( TimeValue t, IParamMap2 *map, HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam )
	{
		int		i;


		switch ( msg )
		{
			case WM_INITDIALOG:
				for( i = 0; i < fSingleProcs.GetCount(); i++ )
					fSingleProcs[ i ]->DlgProc( t, map, hWnd, msg, wParam, lParam );

				for( i = 0; i < fProcs.GetCount(); i++ )
					fProcs[ i ]->DlgProc( t, map, hWnd, msg, wParam, lParam );

				return true;

			case WM_COMMAND:
				for( i = 0; i < fSingleProcs.GetCount(); i++ )
				{
					if( fSingleProcs[ i ]->GetHandledDlgItem() == LOWORD( wParam ) )
					{
						fSingleProcs[ i ]->DlgProc( t, map, hWnd, msg, wParam, lParam );
						break;
					}
				}
				// and now do the procs that want more control
				for( i = 0; i < fProcs.GetCount(); i++ )
					fProcs[ i ]->DlgProc( t, map, hWnd, msg, wParam, lParam );

				return true;
		}

		return false;
	}

	void DeleteThis() {}
};


/////////////////////////////////////////////////////////////////////////////////////////////////
//// plGUITag Component /////////////////////////////////////////////////////////////////////
//
//	GUI element that can be dragged anywhere in the 2D viewing plane.
//
/////////////////////////////////////////////////////////////////////////////////////////////////


class plGUITagProc : public ParamMap2UserDlgProc
{
protected:

	void	ILoadTags( HWND hWnd, IParamBlock2 *pb );

public:

	void DeleteThis() {}

	BOOL DlgProc( TimeValue t, IParamMap2 *map, HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam );
};

static plGUITagProc gGUITagProc;

// Class that accesses the paramblock below.
class plGUITagComponent : public plComponent
{
public:
	plGUITagComponent();
	void DeleteThis() { delete this; }

	// SetupProperties - Internal setup and write-only set properties on the MaxNode. No reading
	// of properties on the MaxNode, as it's still indeterminant.
	hsBool SetupProperties(plMaxNode *pNode, plErrorMsg *pErrMsg);

	hsBool PreConvert(plMaxNode *pNode, plErrorMsg *pErrMsg);
	hsBool Convert(plMaxNode *node, plErrorMsg *pErrMsg);

	enum
	{
		kRefCurrIDSel = 64		// So we can share it among other components
	};

	static UInt32 GetTagIDOnNode( plMaxNode *node );
};

//Max desc stuff necessary below.
#define kGUITagClassID		Class_ID(0x77276e84, 0x24f360c5)
CLASS_DESC(plGUITagComponent, gGUITagDesc, "GUI ID Tag",  "GUITag", COMP_TYPE_GUI, kGUITagClassID )

ParamBlockDesc2 gGUITagBk
(	// KLUDGE: not the defined block ID, but kept for backwards compat.
 plComponent::kBlkComp, _T("GUITag"), 0, &gGUITagDesc, P_AUTO_CONSTRUCT + P_AUTO_UI, plComponent::kRefComp,

	IDD_COMP_GUITAG, IDS_COMP_GUITAG, 0, 0, &gGUITagProc,

		plGUITagComponent::kRefCurrIDSel,	_T("currSel"),		TYPE_INT, 		0, 0,
		end,

	end
);

void	plGUITagProc::ILoadTags( HWND hWnd, IParamBlock2 *pb )
{
	int		idx, idx2 = 0;
	char	str[] = "(none)";


	SendMessage( hWnd, CB_RESETCONTENT, 0, 0 );
	idx2 = idx = SendMessage( hWnd, CB_ADDSTRING, 0, (LPARAM)str );
	SendMessage( hWnd, CB_SETITEMDATA, (WPARAM)idx, (LPARAM)0 );

	for( UInt32 i = 0; i < pfGameGUIMgr::GetNumTags(); i++ )
	{
		pfGUITag *tag = pfGameGUIMgr::GetTag( i );
		idx = SendMessage( hWnd, CB_ADDSTRING, 0, (LPARAM)tag->fName );
		SendMessage( hWnd, CB_SETITEMDATA, (WPARAM)idx, (LPARAM)tag->fID );

		if( tag->fID == pb->GetInt( plGUITagComponent::kRefCurrIDSel ) )
			idx2 = idx;
	}

	if( idx2 == 0 && pb->GetInt( plGUITagComponent::kRefCurrIDSel ) != 0 )
	{
		char	str[ 32 ];
		sprintf( str, "%d", pb->GetInt( plGUITagComponent::kRefCurrIDSel ) );
		SendMessage( hWnd, WM_SETTEXT, 0, (LPARAM)str );
	}
	else
		SendMessage( hWnd, CB_SETCURSEL, idx2, 0 );
}

// Callback enum proc for below
BOOL CALLBACK	GetEditCtrlEnumProc( HWND hWnd, LPARAM lParam )
{
	char	className[ 128 ];


	// ICK
	GetClassName( hWnd, className, sizeof( className ) - 1 );
	if( stricmp( className, "EDIT" ) == 0 )
	{
		HWND	*ptr = (HWND *)lParam;
		*ptr = hWnd;
		return FALSE;
	}
	return TRUE;
}

// Small proc that, given the handle of a combo box, returns the handle of the edit window for it
static HWND		GetEditCtrlFromComboBox( HWND combo )
{
	HWND	toReturn;

	EnumChildWindows( combo, GetEditCtrlEnumProc, (LPARAM)&toReturn );
	return toReturn;
}

// Small proc to only allow numbers in an edit box
static WNDPROC		sOriginalProc = nil;
LRESULT CALLBACK	SubclassedEditProc( HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam )
{
	switch( msg )
	{
		case WM_CHAR:
			if( !isdigit( (TCHAR)wParam ) )
				return 0;
			break;

		case WM_GETDLGCODE:
			return DLGC_WANTALLKEYS;

		case WM_KEYDOWN:
			if( wParam == VK_RETURN )
			{
				// Do the same thing as when we lose focus--check our int value and make
				// sure it's big enough (don't worry about setting the paramBlock value,
				// that'll happen when the control loses focus)

				char	str[ 32 ];
				GetWindowText( hWnd, str, sizeof( str ) - 1 );
				int id = atoi( str );

				if( id < pfGameGUIMgr::GetHighestTag() + 1 )
				{
					id = pfGameGUIMgr::GetHighestTag() + 1;
					sprintf( str, "%d", id );
					SetWindowText( hWnd, str );
				}
				SendMessage( hWnd, EM_SETSEL, 0, (LPARAM)-1 );
				return 0;
			}
			break;
	}

	return CallWindowProc( sOriginalProc, hWnd, msg, wParam, lParam );
}

BOOL plGUITagProc::DlgProc( TimeValue t, IParamMap2 *pmap, HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam )
{
	HWND	edit;
	BOOL	dummy1;


	switch( msg )
	{
		case WM_INITDIALOG:
			ILoadTags( GetDlgItem( hWnd, IDC_GUI_TAGCOMBO ), pmap->GetParamBlock() );

			// Set the edit control of the combo box to only accept number characters
			edit = GetEditCtrlFromComboBox( GetDlgItem( hWnd, IDC_GUI_TAGCOMBO ) );
			SetWindowLong( edit, GWL_STYLE, GetWindowLong( edit, GWL_STYLE ) | ES_WANTRETURN );
			sOriginalProc = (WNDPROC)SetWindowLong( edit, GWL_WNDPROC, (DWORD)SubclassedEditProc );
            
			return true;

		case WM_DESTROY:
			SetWindowLong( GetDlgItem( hWnd, IDC_GUI_TAGCOMBO ), GWL_WNDPROC, (DWORD)sOriginalProc );
			break;

		case WM_COMMAND:
			if( LOWORD( wParam ) == IDC_GUI_TAGCOMBO )
			{
				if( HIWORD( wParam ) == CBN_SELCHANGE )
				{
					int idx = SendDlgItemMessage( hWnd, IDC_GUI_TAGCOMBO, CB_GETCURSEL, 0, 0 );
					if( idx == CB_ERR )
					{
						// Must be a custom one
						int id = GetDlgItemInt( hWnd, IDC_GUI_TAGCOMBO, &dummy1, false );
						pmap->GetParamBlock()->SetValue( plGUITagComponent::kRefCurrIDSel, 0, id );
					}
					else
					{
						pmap->GetParamBlock()->SetValue( plGUITagComponent::kRefCurrIDSel, 0, 
									SendDlgItemMessage( hWnd, IDC_GUI_TAGCOMBO, CB_GETITEMDATA, idx, 0 ) );
					}
				}
				else if( HIWORD( wParam ) == CBN_KILLFOCUS )
				{
					plMaxAccelerators::Enable();

					// Make sure the number inside is valid
					if( SendDlgItemMessage( hWnd, IDC_GUI_TAGCOMBO, CB_GETCURSEL, 0, 0 ) == CB_ERR )
					{
						int id = GetDlgItemInt( hWnd, IDC_GUI_TAGCOMBO, &dummy1, false );
						if( id < pfGameGUIMgr::GetHighestTag() + 1 )
						{
							id = pfGameGUIMgr::GetHighestTag() + 1;
							SetDlgItemInt( hWnd, IDC_GUI_TAGCOMBO, id, false );
						}

						pmap->GetParamBlock()->SetValue( plGUITagComponent::kRefCurrIDSel, 0, id );
					}
				}
				else if( HIWORD( wParam ) == CBN_SETFOCUS )
				{
					plMaxAccelerators::Disable();
				}
			}
			break;
	}
	return false;
}

plGUITagComponent::plGUITagComponent()
{
	fClassDesc = &gGUITagDesc;
	fClassDesc->MakeAutoParamBlocks(this);
}

// SetupProperties - Internal setup and write-only set properties on the MaxNode. No reading
// of properties on the MaxNode, as it's still indeterminant.
hsBool plGUITagComponent::SetupProperties(plMaxNode *node,  plErrorMsg *pErrMsg)
{
	return true;
}

hsBool plGUITagComponent::PreConvert(plMaxNode *node,  plErrorMsg *pErrMsg)
{

	return true;
}

hsBool plGUITagComponent::Convert(plMaxNode *node, plErrorMsg *pErrMsg)
{
	return true;
}

UInt32 plGUITagComponent::GetTagIDOnNode( plMaxNode *node )
{
	UInt32	i;


	for( i = 0; i < node->NumAttachedComponents( false ); i++ )
	{
		plComponentBase *comp = node->GetAttachedComponent( i, false );
		if( comp->ClassID() == kGUITagClassID )
		{
			plGUITagComponent *tag = (plGUITagComponent *)comp;
			return tag->GetParamBlockByID( plComponent::kBlkComp )->GetInt( kRefCurrIDSel );
		}
	}

	return 0;
}

/////////////////////////////////////////////////////////////////////////////////////////////////
//// plGUIColorScheme Component /////////////////////////////////////////////////////////////
//
//	Defines the color scheme for a single control or an entire dialog
//
/////////////////////////////////////////////////////////////////////////////////////////////////


class plGUIColorSchemeProc : public ParamMap2UserDlgProc
{
protected:

	void	ILoadFonts( HWND hWnd, IParamBlock2 *pb );

	static int CALLBACK IMyFontEnumProc( const ENUMLOGFONTEX *logFontData, const NEWTEXTMETRICEX *physFontData,
													unsigned long fontType, LPARAM lParam );

public:

	void DeleteThis() {}

	virtual void	Update( TimeValue t, Interval &valid, IParamMap2 *map );

	BOOL DlgProc( TimeValue t, IParamMap2 *map, HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam );

};

static plGUIColorSchemeProc gGUIColorSchemeProc;

// Class that accesses the paramblock below.
class plGUIColorSchemeComp : public plComponent
{
public:
	plGUIColorSchemeComp();
	void DeleteThis() { delete this; }

	// SetupProperties - Internal setup and write-only set properties on the MaxNode. No reading
	// of properties on the MaxNode, as it's still indeterminant.
	hsBool SetupProperties(plMaxNode *pNode, plErrorMsg *pErrMsg);

	hsBool PreConvert(plMaxNode *pNode, plErrorMsg *pErrMsg);
	hsBool Convert(plMaxNode *node, plErrorMsg *pErrMsg);

	enum
	{
		kRefForeColor = 128,		// So we can share it among other components
		kRefBackColor,
		kRefSelForeColor,
		kRefSelBackColor,
		kRefUseAlphas,
		kRefFontFace,
		kRefFontSize,
		kRefFontBold,
		kRefFontItalic,

		kRefForeAlpha,
		kRefBackAlpha,
		kRefSelForeAlpha,
		kRefSelBackAlpha,

		kRefFontShadowed
	};

	static void		ConvertScheme( IParamBlock2 *pb, pfGUIColorScheme *destScheme, plErrorMsg *pErrMsg );
};

//Max desc stuff necessary below.
CLASS_DESC(plGUIColorSchemeComp, gGUIColorSchemeDesc, "GUI Color Scheme",  "GUIColorScheme", COMP_TYPE_GUI, GUI_COLORSCHEME_CLASSID )

static ParamBlockDesc2	gGUIColorSchemeBk
(
	/// Main def
	 plComponent::kBlkComp, _T("GUIColorScheme"), 0, &gGUIColorSchemeDesc, P_AUTO_CONSTRUCT + P_AUTO_UI + P_MULTIMAP, plComponent::kRefComp,

	1, 
	plGUIDialogComponent::kSchemeRollout, IDD_COMP_GUISCHEME, IDS_COMP_GUISCHEME, 0, 0, &gGUIColorSchemeProc,	

	plGUIColorSchemeComp::kRefForeColor,	_T("foreColor"),		TYPE_RGBA, 		0, 0,
		p_ui, plGUIDialogComponent::kSchemeRollout, TYPE_COLORSWATCH, IDC_GUI_FGCOLOR,
		p_default, Color( 1.f, 1.f, 1.f ),
		end,

	plGUIColorSchemeComp::kRefBackColor,	_T("backColor"),		TYPE_RGBA, 		0, 0,
		p_ui, plGUIDialogComponent::kSchemeRollout, TYPE_COLORSWATCH, IDC_GUI_BGCOLOR,
		p_default, Color( 0.f, 0.f, 0.f ),
		end,

	plGUIColorSchemeComp::kRefSelForeColor,	_T("selForeColor"),		TYPE_RGBA, 		0, 0,
		p_ui, plGUIDialogComponent::kSchemeRollout, TYPE_COLORSWATCH, IDC_GUI_SFGCOLOR,
		p_default, Color( 1.f, 1.f, 1.f ),
		end,

	plGUIColorSchemeComp::kRefSelBackColor,	_T("selBackColor"),		TYPE_RGBA, 		0, 0,
		p_ui, plGUIDialogComponent::kSchemeRollout, TYPE_COLORSWATCH, IDC_GUI_SBGCOLOR,
		p_default, Color( 0.f, 0.f, 1.f ),
		end,


	plGUIColorSchemeComp::kRefForeAlpha,	_T("foreAlpha"),		TYPE_FLOAT, 		0, 0,
		p_ui, plGUIDialogComponent::kSchemeRollout, TYPE_SLIDER, EDITTYPE_FLOAT, IDC_GUI_FGAEDIT, IDC_GUI_FGALPHA, 4,
		p_range, 0.f, 1.f,
		p_default, 1.f,
		end,

	plGUIColorSchemeComp::kRefBackAlpha,	_T("backAlpha"),		TYPE_FLOAT, 		0, 0,
		p_ui, plGUIDialogComponent::kSchemeRollout, TYPE_SLIDER, EDITTYPE_FLOAT, IDC_GUI_BGAEDIT, IDC_GUI_BGALPHA, 4,
		p_range, 0.f, 1.f,
		p_default, 1.f,
		end,

	plGUIColorSchemeComp::kRefSelForeAlpha,	_T("selForeAlpha"),		TYPE_FLOAT, 		0, 0,
		p_ui, plGUIDialogComponent::kSchemeRollout, TYPE_SLIDER, EDITTYPE_FLOAT, IDC_GUI_SFGAEDIT, IDC_GUI_SFGALPHA, 4,
		p_range, 0.f, 1.f,
		p_default, 1.f,
		end,

	plGUIColorSchemeComp::kRefSelBackAlpha,	_T("selBackAlpha"),		TYPE_FLOAT, 		0, 0,
		p_ui, plGUIDialogComponent::kSchemeRollout, TYPE_SLIDER, EDITTYPE_FLOAT, IDC_GUI_SBGAEDIT, IDC_GUI_SBGALPHA, 4,
		p_range, 0.f, 1.f,
		p_default, 1.f,
		end,


	plGUIColorSchemeComp::kRefUseAlphas, _T("useAlphas"),		TYPE_BOOL, 		0, 0,
		p_default,	FALSE,
		p_ui, plGUIDialogComponent::kSchemeRollout, TYPE_SINGLECHEKBOX, IDC_GUI_USEALPHAS,
		p_enable_ctrls, 4, plGUIColorSchemeComp::kRefForeAlpha, plGUIColorSchemeComp::kRefBackAlpha, 
							plGUIColorSchemeComp::kRefSelForeAlpha, plGUIColorSchemeComp::kRefSelBackAlpha,
		end,		


	plGUIColorSchemeComp::kRefFontFace, _T("fontFace"),		TYPE_STRING, 		0, 0,
		p_default,	_T( "Times New Roman" ),
		end,		

	plGUIColorSchemeComp::kRefFontSize,	_T("fontSize"),		TYPE_INT, 		0, 0,
		p_ui,	plGUIDialogComponent::kSchemeRollout, TYPE_SPINNER, EDITTYPE_POS_INT, IDC_GUI_FONTSIZE, IDC_GUI_FONTSIZE_SPIN, SPIN_AUTOSCALE,
		p_default, 10,
		end,

	plGUIColorSchemeComp::kRefFontBold, _T("fontBold"),		TYPE_BOOL, 		0, 0,
		p_default,	FALSE,
		p_ui, plGUIDialogComponent::kSchemeRollout, TYPE_SINGLECHEKBOX, IDC_GUI_FONTBOLD,
		end,		

	plGUIColorSchemeComp::kRefFontItalic, _T("fontItalic"),		TYPE_BOOL, 		0, 0,
		p_default,	FALSE,
		p_ui, plGUIDialogComponent::kSchemeRollout, TYPE_SINGLECHEKBOX, IDC_GUI_FONTITALIC,
		end,		

	plGUIColorSchemeComp::kRefFontShadowed, _T("fontShadowed"),		TYPE_BOOL, 		0, 0,
		p_default,	FALSE,
		p_ui, plGUIDialogComponent::kSchemeRollout, TYPE_SINGLECHEKBOX, IDC_GUI_FONTSHADOWED,
		end,		

	end
);

int CALLBACK plGUIColorSchemeProc::IMyFontEnumProc( const ENUMLOGFONTEX *logFontData, const NEWTEXTMETRICEX *physFontData,
													unsigned long fontType, LPARAM lParam )
{
	HWND	combo = (HWND)lParam;


	if( SendMessage( combo, CB_FINDSTRINGEXACT, (WPARAM)-1, (LPARAM)logFontData->elfLogFont.lfFaceName ) == CB_ERR )
		SendMessage( combo, CB_ADDSTRING, 0, (LPARAM)logFontData->elfLogFont.lfFaceName );

	return -1;
}

void	plGUIColorSchemeProc::ILoadFonts( HWND hWnd, IParamBlock2 *pb )
{
	LOGFONT	logFont;


	logFont.lfCharSet = DEFAULT_CHARSET;
	strcpy( logFont.lfFaceName, "" );
	logFont.lfPitchAndFamily = 0;

	SendMessage( hWnd, CB_RESETCONTENT, 0, 0 );

	HDC hDC = GetDC( nil );
	EnumFontFamiliesEx( hDC, &logFont, (FONTENUMPROC)IMyFontEnumProc, (LPARAM)hWnd, 0 );
	ReleaseDC( nil, hDC );

	SendMessage( hWnd, CB_SELECTSTRING, (WPARAM)-1, (LPARAM)pb->GetStr( plGUIColorSchemeComp::kRefFontFace ) );
}

#define MAXTOCOLORREF( max ) RGB( max.r * 255.f, max.g * 255.f, max.b * 255.f )

void	plGUIColorSchemeProc::Update( TimeValue t, Interval &valid, IParamMap2 *pmap )
{
}

BOOL plGUIColorSchemeProc::DlgProc( TimeValue t, IParamMap2 *pmap, HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam )
{
	char			str[ 256 ];
	HWND			placeCtrl;
	PAINTSTRUCT		paintInfo;
	RECT			previewRect, r;
	HBRUSH			bgPattBrush = nil;
	Color			fgColor, bgColor, selFgColor, selBgColor;
	Color			hatchColor = Color( 0.4f, 0.4f, 0.4f ), blendedColor, whiteColor = Color( 0.7f, 0.7f, 0.7f );
	Color			blackColor = Color( 0, 0, 0 ), blendedColor2;
	float			fgAlpha, bgAlpha, selFgAlpha, selBgAlpha;
	char			previewString[] = "Preview";
	HFONT			font;


	switch( msg )
	{
		case WM_INITDIALOG:
			ILoadFonts( GetDlgItem( hWnd, IDC_GUI_FONTFACE ), pmap->GetParamBlock() );           
			return true;

		case WM_DESTROY:
			break;

		case WM_COMMAND:
			if( LOWORD( wParam ) == IDC_GUI_FONTFACE )
			{
				if( HIWORD( wParam ) == CBN_SELCHANGE )
				{
					int idx = SendDlgItemMessage( hWnd, IDC_GUI_FONTFACE, CB_GETCURSEL, 0, 0 );

					SendDlgItemMessage( hWnd, IDC_GUI_FONTFACE, CB_GETLBTEXT, idx, (LPARAM)str );

					pmap->GetParamBlock()->SetValue( plGUIColorSchemeComp::kRefFontFace, 0, str );
				}
			}
			break;

		case CC_COLOR_CHANGE:
		case CC_COLOR_DROP:
			placeCtrl = ::GetDlgItem( hWnd, IDC_GUI_SCHEMEPREV );
			::GetClientRect( placeCtrl, &previewRect );
			::MapWindowPoints( placeCtrl, hWnd, (POINT *)&previewRect, 2 );
			::InvalidateRect( hWnd, &previewRect, FALSE );
			break;

		case WM_PAINT:

			fgColor = pmap->GetParamBlock()->GetColor( plGUIColorSchemeComp::kRefForeColor );
			bgColor = pmap->GetParamBlock()->GetColor( plGUIColorSchemeComp::kRefBackColor );
			selFgColor = pmap->GetParamBlock()->GetColor( plGUIColorSchemeComp::kRefSelForeColor );
			selBgColor = pmap->GetParamBlock()->GetColor( plGUIColorSchemeComp::kRefSelBackColor );

			fgAlpha = pmap->GetParamBlock()->GetFloat( plGUIColorSchemeComp::kRefForeAlpha );
			bgAlpha = pmap->GetParamBlock()->GetFloat( plGUIColorSchemeComp::kRefBackAlpha );
			selFgAlpha = pmap->GetParamBlock()->GetFloat( plGUIColorSchemeComp::kRefSelForeAlpha );
			selBgAlpha = pmap->GetParamBlock()->GetFloat( plGUIColorSchemeComp::kRefSelBackAlpha );
			if( pmap->GetParamBlock()->GetInt( plGUIColorSchemeComp::kRefUseAlphas ) == 0 )
				fgAlpha = bgAlpha = selFgAlpha = selBgAlpha = 1.f;

			placeCtrl = ::GetDlgItem( hWnd, IDC_GUI_SCHEMEPREV );
			::GetClientRect( placeCtrl, &previewRect );
			::MapWindowPoints( placeCtrl, hWnd, (POINT *)&previewRect, 2 );

			::BeginPaint( hWnd, &paintInfo );
			::SetBkMode( paintInfo.hdc, TRANSPARENT );

			int weight = pmap->GetParamBlock()->GetInt( plGUIColorSchemeComp::kRefFontBold ) ? FW_BOLD : FW_NORMAL;
			bool italic = pmap->GetParamBlock()->GetInt( plGUIColorSchemeComp::kRefFontItalic ) ? true : false;
			int nHeight = -MulDiv( pmap->GetParamBlock()->GetInt( plGUIColorSchemeComp::kRefFontSize ), GetDeviceCaps( paintInfo.hdc, LOGPIXELSY ), 72 );
			const char *face = pmap->GetParamBlock()->GetStr( plGUIColorSchemeComp::kRefFontFace );

			font = ::CreateFont( nHeight, 0, 0, 0, weight, italic, FALSE, FALSE, DEFAULT_CHARSET, OUT_DEFAULT_PRECIS,
							CLIP_DEFAULT_PRECIS, ANTIALIASED_QUALITY, VARIABLE_PITCH, face );
			SelectObject( paintInfo.hdc, font );

			// Left side
			r = previewRect;
			r.right = ( r.right + r.left ) >> 1;

			blendedColor = bgColor * bgAlpha + ( whiteColor * ( 1.f - bgAlpha ) );

			// doesn't like the Color to DWORD operator, so duplicating it here
			#define ColorToDWORD(color) RGB(FLto255(color.r),FLto255(color.g),FLto255(color.b))
			::SetBkColor( paintInfo.hdc, ColorToDWORD(blendedColor) );

			blendedColor = bgColor * bgAlpha + ( hatchColor * ( 1.f - bgAlpha ) );
			bgPattBrush = CreateHatchBrush( HS_DIAGCROSS, MAXTOCOLORREF( blendedColor ) );

			::FillRect( paintInfo.hdc, &r, bgPattBrush );
			if( pmap->GetParamBlock()->GetInt( plGUIColorSchemeComp::kRefFontShadowed ) )
			{
				blendedColor2 = blackColor * fgAlpha + ( blendedColor * ( 1.f - fgAlpha ) );
				::SetTextColor( paintInfo.hdc, MAXTOCOLORREF( blendedColor2 ) );
				::OffsetRect( &r, 1, 1 );
				::DrawText( paintInfo.hdc, previewString, strlen( previewString ), &r, DT_CENTER | DT_VCENTER );
				::OffsetRect( &r, -1, -1 );
			}

			blendedColor = fgColor * fgAlpha + ( blendedColor * ( 1.f - fgAlpha ) );
			::SetTextColor( paintInfo.hdc, MAXTOCOLORREF( blendedColor ) );
			::DrawText( paintInfo.hdc, previewString, strlen( previewString ), &r, DT_CENTER | DT_VCENTER );
			
			::DeleteObject( bgPattBrush );

			// Right side
			r.left = r.right;
			r.right = previewRect.right;

			blendedColor = selBgColor * selBgAlpha + ( whiteColor * ( 1.f - selBgAlpha ) );
			::SetBkColor( paintInfo.hdc, ColorToDWORD(blendedColor) );
			blendedColor = selBgColor * selBgAlpha + ( hatchColor * ( 1.f - selBgAlpha ) );
			bgPattBrush = CreateHatchBrush( HS_DIAGCROSS, MAXTOCOLORREF( blendedColor ) );

			::FillRect( paintInfo.hdc, &r, bgPattBrush );
			if( pmap->GetParamBlock()->GetInt( plGUIColorSchemeComp::kRefFontShadowed ) )
			{
				blendedColor2 = blackColor * selFgAlpha + ( blendedColor * ( 1.f - selFgAlpha ) );
				::SetTextColor( paintInfo.hdc, MAXTOCOLORREF( blendedColor2 ) );
				::OffsetRect( &r, 1, 1 );
				::DrawText( paintInfo.hdc, previewString, strlen( previewString ), &r, DT_CENTER | DT_VCENTER );
				::OffsetRect( &r, -1, -1 );
			}
			blendedColor = selFgColor * selFgAlpha + ( blendedColor * ( 1.f - selFgAlpha ) );
			::SetTextColor( paintInfo.hdc, MAXTOCOLORREF( blendedColor ) );
			::DrawText( paintInfo.hdc, previewString, strlen( previewString ), &r, DT_CENTER | DT_VCENTER );
			
			::DeleteObject( bgPattBrush );

			::DeleteObject( font );

			::EndPaint( hWnd, &paintInfo );

			return true;
	}
	return false;
}

plGUIColorSchemeComp::plGUIColorSchemeComp()
{
	fClassDesc = &gGUIColorSchemeDesc;
	fClassDesc->MakeAutoParamBlocks(this);
}

// SetupProperties - Internal setup and write-only set properties on the MaxNode. No reading
// of properties on the MaxNode, as it's still indeterminant.
hsBool plGUIColorSchemeComp::SetupProperties(plMaxNode *node,  plErrorMsg *pErrMsg)
{
	return true;
}

hsBool plGUIColorSchemeComp::PreConvert(plMaxNode *node,  plErrorMsg *pErrMsg)
{

	return true;
}

hsBool plGUIColorSchemeComp::Convert(plMaxNode *node, plErrorMsg *pErrMsg)
{
	pfGUIControlMod *ctrl = plGUIControlBase::GrabControlFromObject( node );
	if( ctrl != nil )
	{
		pfGUIColorScheme *cs = TRACKED_NEW pfGUIColorScheme;
		ConvertScheme( fCompPB, cs, pErrMsg );
		ctrl->SetColorScheme( cs );
	}
	else
	{
		pErrMsg->Set( true, "GUI Color Scheme Error", "You have applied a GUI color scheme to an object (%s) without a GUI control. This scheme will be ignored.", node->GetName()).Show();
		pErrMsg->Set( false );
		return false;	
	}

	return true;
}

void	SMaxRGBAToPlasmaRGBA( Color maxRGB, hsColorRGBA &plasmaRGBA )
{
	plasmaRGBA.Set( maxRGB.r, maxRGB.g, maxRGB.b, 1.f );
}

void	plGUIColorSchemeComp::ConvertScheme( IParamBlock2 *pb, pfGUIColorScheme *destScheme, plErrorMsg *pErrMsg )
{
	SMaxRGBAToPlasmaRGBA( pb->GetColor( kRefForeColor ), destScheme->fForeColor );
	SMaxRGBAToPlasmaRGBA( pb->GetColor( kRefBackColor ), destScheme->fBackColor );
	SMaxRGBAToPlasmaRGBA( pb->GetColor( kRefSelForeColor ), destScheme->fSelForeColor );
	SMaxRGBAToPlasmaRGBA( pb->GetColor( kRefSelBackColor ), destScheme->fSelBackColor );

	destScheme->fForeColor.a = pb->GetFloat( kRefForeAlpha );
	destScheme->fBackColor.a = pb->GetFloat( kRefBackAlpha );
	destScheme->fSelForeColor.a = pb->GetFloat( kRefSelForeAlpha );
	destScheme->fSelBackColor.a = pb->GetFloat( kRefSelBackAlpha );

	destScheme->fTransparent = pb->GetInt( kRefUseAlphas ) ? true : false;
	
	destScheme->SetFontFace( pb->GetStr( kRefFontFace ) );
	destScheme->fFontSize = pb->GetInt( kRefFontSize );
	destScheme->fFontFlags = 0;
	if( pb->GetInt( kRefFontBold ) )
		destScheme->fFontFlags |= pfGUIColorScheme::kFontBold;
	if( pb->GetInt( kRefFontItalic ) )
		destScheme->fFontFlags |= pfGUIColorScheme::kFontItalic;
	if( pb->GetInt( kRefFontShadowed ) )
		destScheme->fFontFlags |= pfGUIColorScheme::kFontShadowed;
}

/////////////////////////////////////////////////////////////////////////////////////////////////
//// plGUIProxy Rollout /////////////////////////////////////////////////////////////////////////
//
//	Defines a proxy object to be used when calculating mouse-down locations and dynamic text
//	sizing. 
//
/////////////////////////////////////////////////////////////////////////////////////////////////

enum plProxyRefs
{
	kRefProxyNode = 196,
	kRefHideProxy,
	kRefBetterHitTests
};

//// DialogProc /////////////////////////////////////////////////////////////////////////////////

class plGUIProxyDlgProc : public ParamMap2UserDlgProc
{
public:

	plGUIProxyDlgProc()
	{
	}

	void DeleteThis() {}

	BOOL DlgProc(TimeValue t, IParamMap2 *map, HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam)
	{
		IParamBlock2 *pblock = map->GetParamBlock();

		switch( msg )
		{
			case WM_COMMAND:
//				if( LOWORD( wParam ) == IDC_GUI_CLEAR )
//				{
//					pblock->Reset( (ParamID)kRefProxyNode );
//					return true;
//				}
				break;
		}	
		return false;
	}

};	

static plGUIProxyDlgProc	sGUIProxyDlgProc;

//// ParamBlock /////////////////////////////////////////////////////////////////////////////////
//	Note: we can't make this a real ParamBlock and do P_INCLUDE_PARAMS because, in Discreet's 
//	amazing method of doing things, we can't INCLUDE more than one ParamBlock in any other PB.
//	So either we chain them together here (and thus make them dependent on one another, which
//	is lame) or we just make the whole damned thing a #define, which is all P_INCLUDE_PARAMS
//	really does anyway.

#define sGUIProxyParamHeader		plGUIControlBase::kRollProxy, IDD_COMP_GUIPROXY, IDS_COMP_GUIPROXY, 0, APPENDROLL_CLOSED, &sGUIProxyDlgProc
//static ParamBlockDesc2	sSndEAXPropsParamTemplate
//(
	/// Main def
//	plComponent::kBlkComp + 1, _T("sndEAXProps"), 0, nil, P_AUTO_UI + P_MULTIMAP + P_AUTO_CONSTRUCT, plComponent::kRefComp, 

//	1, 
//	kSndEAXParams, IDD_COMP_EAXBUFFER, IDS_COMP_EAXBUFFER, 0, 0, nil,	

#define sGUIProxyParamTemplate \
																											\
	kRefBetterHitTests, _T("guiBetterHitTests"), TYPE_BOOL, 0, 0,														\
		p_ui, plGUIControlBase::kRollProxy, TYPE_SINGLECHEKBOX, IDC_GUI_BETTERHIT,							\
		p_default, false,																					\
		end																								

//	, end
//);


/////////////////////////////////////////////////////////////////////////////////////////////////
//// plGUIDialog Component //////////////////////////////////////////////////////////////////////
//
//	Defines a dialog box (i.e. a collection of controls) to be defined with the GUI manager at
//	runtime. Acts a lot like a CamView component, but it additionally handles a few other things.
//
/////////////////////////////////////////////////////////////////////////////////////////////////

class plGUIDialogProc : public ParamMap2UserDlgProc
{
protected:

	void	ILoadPages( HWND hWnd, IParamBlock2 *pb );

public:

	void DeleteThis() {}

	BOOL DlgProc( TimeValue t, IParamMap2 *map, HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam );
};
static plGUIDialogProc gGUIDialogProc;

//Max desc stuff necessary below.
CLASS_DESC(plGUIDialogComponent, gGUIDialogDesc, "GUI Dialog",  "GUIDialog", COMP_TYPE_GUI, GUI_DIALOG_COMP_CLASS_ID )

ParamBlockDesc2 gGUIDialogBk
(	// KLUDGE: not the defined block ID, but kept for backwards compat.
 plComponent::kBlkComp, _T("GUIDialog"), 0, &gGUIDialogDesc, P_AUTO_CONSTRUCT + P_AUTO_UI + P_MULTIMAP + P_INCLUDE_PARAMS, plComponent::kRefComp,

	3, 
	plGUIDialogComponent::kMainRollout,		IDD_COMP_GUIDIALOG, IDS_COMP_GUIDIALOG, 0, 0, &gGUIDialogProc,
	plGUIDialogComponent::kTagIDRollout,	IDD_COMP_GUITAG,	IDS_COMP_GUITAG,	0, 0, &gGUITagProc,
	plGUIDialogComponent::kSchemeRollout,	IDD_COMP_GUISCHEME, IDS_COMP_GUISCHEME, 0, 0, &gGUIColorSchemeProc,	

	&gGUIColorSchemeBk,

		plGUIDialogComponent::kRefDialogName,	_T("DialogName"),		TYPE_STRING, 		0, 0,
//			p_ui, plGUIDialogComponent::kMainRollout, TYPE_EDITBOX, IDC_GUIDLG_NAME,
			end,

		plGUIDialogComponent::kRefAgeName,	_T("ageName"),		TYPE_STRING, 		0, 0,
			p_default, _T( "GUI" ),
			end,
			
		plGUIDialogComponent::kRefIsModal, _T("isModal"),		TYPE_BOOL, 		0, 0,
			p_default,	FALSE,
			p_ui, plGUIDialogComponent::kMainRollout, TYPE_SINGLECHEKBOX, IDC_COMP_GUI_MODAL,
			end,		

		plGUIDialogComponent::kRefVersion,	_T("version"),		TYPE_INT, 		0, 0,
			p_ui,	plGUIDialogComponent::kMainRollout, TYPE_SPINNER, EDITTYPE_POS_INT, IDC_GUI_VERSION, IDC_GUI_VERSION_SPIN, SPIN_AUTOSCALE,
			p_default, 0,
			end,

		plGUITagComponent::kRefCurrIDSel,	_T("currSel"),		TYPE_INT, 		0, 0,
			end,

	end
);

plGUIDialogComponent::plGUIDialogComponent( hsBool dontInit )
{
	if( !dontInit )
	{
		fClassDesc = &gGUIDialogDesc;
		fClassDesc->MakeAutoParamBlocks(this);
	}
	fDialogMod = nil;
	fProcReceiver = nil;
}

pfGUIDialogMod	*plGUIDialogComponent::IMakeDialog( void )
{
	return TRACKED_NEW pfGUIDialogMod();
}

// SetupProperties - Internal setup and write-only set properties on the MaxNode. No reading
// of properties on the MaxNode, as it's still indeterminant.
hsBool plGUIDialogComponent::SetupProperties(plMaxNode *node,  plErrorMsg *pErrMsg)
{
	TimeValue timeVal( 0 );
	Object* obj = node->EvalWorldState( timeVal ).obj;

	fDialogMod = nil;

	if( obj->CanConvertToType( Class_ID( LOOKAT_CAM_CLASS_ID, 0 ) ) ||
		obj->CanConvertToType( Class_ID( SIMPLE_CAM_CLASS_ID, 0 ) ) )
	{
		// We're applied to a camera. Do our camera stuff
		node->SetForceLocal( true );
	}
	else
	{
		// We're applied to a normal object.
		node->SetNoSpanReSort(true);
		node->SetNoSpanSort(true);
	}

	/// Either way, we mangle our own location component. None of this user-defined-location stuff.

	char *dialogName = fCompPB->GetStr( kRefDialogName );
	if( dialogName == nil || *dialogName == 0 )
	{
		pErrMsg->Set(true, "GUI Dialog Component Error", "No dialog name specified on GUI Dialog component (object: %s)", node->GetName()).Show();
		return false;	
	}

	char *ageName = fCompPB->GetStr(kRefAgeName);
	Int32 seqNum = plPageInfoUtils::GetSeqNumFromAgeDesc( ageName, dialogName );
	Int32 newNum = plPluginResManager::ResMgr()->VerifySeqNumber( seqNum, ageName, dialogName );
	if( newNum != seqNum )
	{
		if( !fSeqNumValidated )
		{
			plLocation pageLoc = plPluginResManager::ResMgr()->FindLocation( ageName, dialogName );
			Int32 pageSeqNum = pageLoc.GetSequenceNumber();
			char errMsg[ 512 ];
			sprintf( errMsg, "The sequence number stored by the resource manager (0x%X) for page %s, District, %s does not match\n"
							"the sequence number stored in the .age file (0x%X). Forcing it to use the one in the .age file", 
								pageSeqNum, ageName, dialogName, seqNum );
			pErrMsg->Set( true, "PageInfo Convert Error", errMsg ).Show(); 
			pErrMsg->Set( false );
			fSeqNumValidated = true;
		}
		// force the component to use the sequence number in the .age file
		//seqNum = newNum;
	}

	plKey roomKey = plPluginResManager::ResMgr()->NameToLoc( fCompPB->GetStr( kRefAgeName ), fCompPB->GetStr( kRefDialogName ), seqNum );
	if( !roomKey )
	{
		pErrMsg->Set( true, "GUI Dialog Component Error", "GUI Dialog Component %s has a Missing Location.  Nuke the files in the dat directory and re-export.",((INode*)node)->GetName()).Show(); 
		return false;
	}

	node->SetRoomKey( roomKey );

	// Also, we make sure this node will never be fogged (affects material convert)
	node->SetIsGUI( true );

	return true;
}

hsBool plGUIDialogComponent::Convert(plMaxNode *node, plErrorMsg *pErrMsg)
{
	TimeValue timeVal(0);

	Object* obj = node->EvalWorldState(timeVal).obj;

	GenCamera* cam = nil;
	if( obj->CanConvertToType(Class_ID(LOOKAT_CAM_CLASS_ID, 0)) )
		cam = (GenCamera *) obj->ConvertToType(timeVal, Class_ID(LOOKAT_CAM_CLASS_ID, 0));
	else 
	if( obj->CanConvertToType(Class_ID(SIMPLE_CAM_CLASS_ID, 0)) ) 
		cam = (GenCamera *) obj->ConvertToType(timeVal, Class_ID(SIMPLE_CAM_CLASS_ID, 0));

	if( !cam )
	{
		// Not applied to a camera, so applied to a normal object. Since this is valid (we also act
		// as a location component), just return
		return true;
	}

	plPostEffectMod* mod = TRACKED_NEW plPostEffectMod;

	float hither = cam->GetEnvRange(timeVal, ENV_NEAR_RANGE); 
	if( hither < 0.5f )
		hither = 0.5f;
	float yon = cam->GetEnvRange(timeVal, ENV_FAR_RANGE);
	mod->SetHither(hither);
	mod->SetYon(yon);

	// radians
	float fov = cam->GetFOV(timeVal);
	// convert
	int FOVType = cam->GetFOVType();
	hsScalar fovX, fovY;
	switch(FOVType)
	{
	case 0: // FOV_W
		{
			fovX = fov;
			fovY = fovX *3.f / 4.f;
		}
		break;
	case 1: // FOV_H
		{
			fovY = fov;
			fovX = fovY * 4.f / 3.f;
		}
		break;
	}
	fovX *= 180.f / hsScalarPI;
	fovY *= 180.f / hsScalarPI;
	mod->SetFovX(fovX);
	mod->SetFovY(fovY);

	// Should already be created from SetupProperties...
	// Note: can't just grab the node's room key, 'cause we might not be on the right node!
	plKey sceneNodeKey = plPluginResManager::ResMgr()->NameToLoc( fCompPB->GetStr( kRefAgeName ), 
														fCompPB->GetStr( kRefDialogName ), (UInt32)-1 );
	mod->SetNodeKey( sceneNodeKey );

//	node->AddModifier(mod);
	// Note: we do NOT add this to the sceneObject, we don't want it actually associated with
	// a sceneObject. Instead, we just grab the LocalToWorld() off the sceneObject, since that's 
	// all we want
	hsMatrix44 l2w = node->GetLocalToWorld44();
	hsMatrix44 w2l = node->GetWorldToLocal44();
	mod->SetWorldToCamera( w2l, l2w );

	// Add it to the sceneNode as a generic interface, so it gets loaded with the sceneNode
	plLocation nodeLoc = sceneNodeKey->GetUoid().GetLocation();

	plKey modKey = hsgResMgr::ResMgr()->NewKey( fCompPB->GetStr( kRefDialogName ), mod, nodeLoc );
	hsgResMgr::ResMgr()->AddViaNotify( modKey, TRACKED_NEW plNodeRefMsg( sceneNodeKey, plRefMsg::kOnCreate, -1, plNodeRefMsg::kGeneric ), plRefFlags::kActiveRef );

	// Also add our dialog mod to the scene node in the same way
	hsgResMgr::ResMgr()->AddViaNotify( fDialogMod->GetKey(), TRACKED_NEW plNodeRefMsg( sceneNodeKey, plRefMsg::kOnCreate, -1, plNodeRefMsg::kGeneric ), plRefFlags::kActiveRef );

	/// Already created our mod, just gotta fill it out
	fDialogMod->SetRenderMod( mod );
	fDialogMod->SetName( fCompPB->GetStr( kRefDialogName ) );
	if( fCompPB->GetInt( kRefIsModal ) )
		fDialogMod->SetFlag( pfGUIDialogMod::kModal );
	fDialogMod->SetProcReceiver(fProcReceiver);
	fDialogMod->SetVersion( fCompPB->GetInt( kRefVersion ) );

	plGUIColorSchemeComp::ConvertScheme( fCompPB, fDialogMod->GetColorScheme(), pErrMsg );

	return true;
}

hsBool plGUIDialogComponent::PreConvert(plMaxNode *node,  plErrorMsg *pErrMsg)
{
	TimeValue timeVal(0);
	Object* obj = node->EvalWorldState(timeVal).obj;

	if( obj->CanConvertToType(Class_ID(LOOKAT_CAM_CLASS_ID, 0))
		|| obj->CanConvertToType(Class_ID(SIMPLE_CAM_CLASS_ID, 0)) )
	{
		// Don't do this. -mf
//		IMakeEveryoneOpaque(node);

		// Make a blank dialog modifier, which will be filled in on convert. Do 
		// this as a separate step so the dialog controls can query and get the dialog
		// mod to store a ref to

		fDialogMod = IMakeDialog();

		// Note: can't just grab the node's room key, 'cause we might not be on the right node!
		plKey sceneNodeKey = plPluginResManager::ResMgr()->NameToLoc( fCompPB->GetStr( kRefAgeName ),
															fCompPB->GetStr( kRefDialogName ), (UInt32)-1 );

		plLocation nodeLoc = sceneNodeKey->GetUoid().GetLocation();
		plKey dlgKey = hsgResMgr::ResMgr()->NewKey( fCompPB->GetStr( kRefDialogName ), fDialogMod, nodeLoc );

		fDialogMod->SetSceneNodeKey( sceneNodeKey );

		// See if there's a tag to be had
		UInt32 id = fCompPB->GetInt( plGUITagComponent::kRefCurrIDSel );
		if( id > 0 )
			fDialogMod->SetTagID( id );

		fProcReceiver = nil;
	}
	else
	{
	}

	return true;
}

void plGUIDialogComponent::IMakeEveryoneOpaque(plMaxNode* node)
{
	plMaxNode* root = (plMaxNode *)node->GetInterface()->GetRootNode();

	int i;
	for( i = 0; i < root->NumberOfChildren(); i++ )
		IMakeEveryoneOpaqueRecur((plMaxNode*)(root->GetChildNode(i)));

}

void plGUIDialogComponent::IMakeEveryoneOpaqueRecur(plMaxNode* node)
{
	if( node->CanConvert() )
	{
		node->SetNoSpanReSort(true);
		node->SetNoSpanSort(true);

		int i;
		for( i = 0; i < node->NumberOfChildren(); i++ )
		{
			IMakeEveryoneOpaqueRecur((plMaxNode *)(node->GetChildNode(i)));
		}
	}
}

plKey	plGUIDialogComponent::GetModifierKey( void )
{
	if( fDialogMod != nil )
		return fDialogMod->GetKey();

	return nil;
}

bool	plGUIDialogComponent::SetNotifyReceiver( plKey key )
{
	if( fProcReceiver != nil )
		return false;

	fProcReceiver = key;
	return true;
}

pfGUIDialogMod	*plGUIDialogComponent::GetNodeDialog( plMaxNode *childNode )
{
	UInt32 i, numComp = childNode->NumAttachedComponents( false );
	for( i = 0; i < numComp; i++ )
	{
		plComponentBase *comp = childNode->GetAttachedComponent( i );
		if( comp->ClassID() == GUI_DIALOG_COMP_CLASS_ID )
			return ( (plGUIDialogComponent *)comp )->GetModifier();
	}

	return nil;
}

void	plGUIDialogProc::ILoadPages( HWND hWnd, IParamBlock2 *pb )
{
	plAgeDescription	*aged = plPageInfoUtils::GetAgeDesc( pb->GetStr( plGUIDialogComponent::kRefAgeName ) );

	if( aged == nil )
		return;

	plAgePage	*page;
	char	*selPageName = pb->GetStr( plGUIDialogComponent::kRefDialogName );
	aged->SeekFirstPage();
	ComboBox_ResetContent( hWnd );

	while( ( page = aged->GetNextPage() ) != nil )
	{
		int idx = ComboBox_AddString( hWnd, page->GetName() );
		if( selPageName && stricmp( page->GetName(), selPageName ) == 0 )
			ComboBox_SetCurSel( hWnd, idx );
	}

	delete aged;
}

BOOL plGUIDialogProc::DlgProc( TimeValue t, IParamMap2 *pmap, HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam )
{
	switch( msg )
	{
		case WM_INITDIALOG:
			// Load the age combo box
			{
				int					i, idx, selIdx = 0;
				HWND				ageCombo = GetDlgItem( hWnd, IDC_GUIDLG_AGE );
				hsTArray<char *>	ageList;

				plAgeDescInterface::BuildAgeFileList( ageList );
				ComboBox_ResetContent( ageCombo );
				for( i = 0; i < ageList.GetCount(); i++ )
				{
					char ageName[ _MAX_FNAME ];
					_splitpath( ageList[ i ], nil, nil, ageName, nil );

					idx = ComboBox_AddString( ageCombo, ageName );
					if( stricmp( ageName, pmap->GetParamBlock()->GetStr( plGUIDialogComponent::kRefAgeName ) ) == 0 )
					{
						selIdx = idx;
					}
				}
				ComboBox_SetCurSel( ageCombo, selIdx );
			}

			ILoadPages( GetDlgItem( hWnd, IDC_GUIDLG_NAME ), pmap->GetParamBlock() );           
			return true;

		case WM_DESTROY:
			break;

		case WM_COMMAND:
			if( HIWORD( wParam ) == CBN_SELCHANGE )
			{
				if( LOWORD( wParam ) == IDC_GUIDLG_NAME )
				{
					int idx = SendDlgItemMessage( hWnd, IDC_GUIDLG_NAME, CB_GETCURSEL, 0, 0 );
					if( idx != CB_ERR )
					{
						char	name[ 256 ];
						ComboBox_GetLBText( GetDlgItem( hWnd, IDC_GUIDLG_NAME ), idx, name );
						pmap->GetParamBlock()->SetValue( plGUIDialogComponent::kRefDialogName, 0, name );
					}
				}
				else if( LOWORD( wParam ) == IDC_GUIDLG_AGE )
				{
					int idx = SendDlgItemMessage( hWnd, IDC_GUIDLG_AGE, CB_GETCURSEL, 0, 0 );
					if( idx != CB_ERR )
					{
						char	name[ 256 ];
						ComboBox_GetLBText( GetDlgItem( hWnd, IDC_GUIDLG_AGE ), idx, name );
						pmap->GetParamBlock()->SetValue( plGUIDialogComponent::kRefAgeName, 0, name );
					}

					ILoadPages( GetDlgItem( hWnd, IDC_GUIDLG_NAME ), pmap->GetParamBlock() );           
				}
			}
			break;
	}
	return false;
}


/////////////////////////////////////////////////////////////////////////////////////////////////
//// plGUIControl Component Base Class //////////////////////////////////////////////////////////
//
//	Defines a base class for all GUI control components.
//
/////////////////////////////////////////////////////////////////////////////////////////////////

void	plGUIControlBase::CollectNonDrawables( INodeTab &nonDrawables )
{
/*	if( ICanHaveProxy() )
	{
		bool hideProxy = fCompPB->GetInt( (ParamID)kRefHideProxy ) ? true : false;
		if( hideProxy )
		{
			INode	*node = fCompPB->GetINode( (ParamID)kRefProxyNode );
			if( node != nil )
				nonDrawables.Append( 1, &node );
		}
	}
*/
}

pfGUIDialogMod	*plGUIControlBase::IGetDialogMod( plMaxNode *node )
{
	UInt32		i;


	for( i = 0; i < node->NumAttachedComponents( false ); i++ )
	{
		plComponentBase *comp = node->GetAttachedComponent( i, false );
		if( comp->ClassID() == GUI_DIALOG_COMP_CLASS_ID )
		{
			// Found it!
			pfGUIDialogMod	*dlgMod = ((plGUIDialogComponent *)comp)->GetModifier();
			return dlgMod;
		}
	}

	return nil;
}

hsBool plGUIControlBase::SetupProperties( plMaxNode *pNode, plErrorMsg *pErrMsg )
{
	if( INeedsDynamicText() )
	{
		// If we're going to be using a dynamic text layer, we need to make sure the material
		// is unique for every node we're applied to
		pNode->SetForceMaterialCopy( true );
	}

	return true;
}

hsBool plGUIControlBase::PreConvert(plMaxNode *node,  plErrorMsg *pErrMsg)
{
	// Create a new control
	fControl = IGetNewControl();

	// Add it as a modifier to this node
	node->AddModifier( fControl, IGetUniqueName(node) );

	// Look for any tag IDs
	UInt32 id = plGUITagComponent::GetTagIDOnNode( node );
	if( id > 0 )
		fControl->SetTagID( id );

	// Now add it to our list of converted nodes
	UInt32 i = fTargetNodes.Find( node );
	if( i == fTargetNodes.kMissingIndex )
	{
		fTargetNodes.Append( node );
		fTargetControls.Append( fControl );
	}
	else
	{
		fTargetControls[ i ] = fControl;
	}

	return true;
}

hsBool plGUIControlBase::Convert(plMaxNode *node, plErrorMsg *pErrMsg)
{
	// Error check--make sure we're in the same room as our parent (can get confusing with the wrong
	// parent-child relationships)
	if( !node->GetParentNode()->IsRootNode() )
	{
		plMaxNode *parent = (plMaxNode *)node->GetParentNode();
		if( parent->GetRoomKey() != node->GetRoomKey() )
		{
			pErrMsg->Set( true, "GUI Control Component Error", "The object %s is assigned to a different GUI dialog than its parent. Make sure both this object and its parent belong to the same GUI dialog (this control will be ignored).", node->GetName() ).Show(); 
			pErrMsg->Set( false );
			return false;
		}
	}

	pfGUIDialogMod *dialog = IGetDialogMod( node );
	if( dialog == nil )
	{
		pErrMsg->Set( true, "GUI Control Component Error", "The object %s has a GUI control applied but not a GUI Dialog Component. Apply a GUI Dialog Component to this object.", node->GetName() ).Show(); 
		pErrMsg->Set( false );
		return false;
	}

	// Grab fControl from the modifier list on the node, since fControl isn't valid
	// between PreConvert() and Convert() (it might get called multiple times, once per node applied)
	UInt32 i = fTargetNodes.Find( node );
	if( i == fTargetNodes.kMissingIndex )
	{
		pErrMsg->Set( true, "GUI Control Component Error", "The object %s somehow skipped the GUI control Pre-convert stage. Inform a programmer immediately and seek shelter.", node->GetName() ).Show(); 
		pErrMsg->Set( false );
		return false;
	}
	
	fControl = fTargetControls[ i ];

	dialog->AddControlOnExport( fControl );

	if( IHasProcRollout() )
	{
		// Also common for all controls: process the Procedure rollout--i.e. what kind of control proc do we get?
		switch( fCompPB->GetInt( kRefChoice ) )
		{
			case 0:
				// Console command
				fControl->SetHandler( TRACKED_NEW pfGUIConsoleCmdProc( fCompPB->GetStr( kRefConsoleCmd ) ) );
				break;

			case 1:
				// Inherit from parent dialog - this is a runtime flag, so we don't bother actually setting
				// a handler here, except to ensure it's nil
				fControl->SetHandler( nil );
				fControl->SetFlag( pfGUIControlMod::kInheritProcFromDlg );
				break;

			case 2:
				fControl->SetHandler( TRACKED_NEW pfGUICloseDlgProc() );
				break;

			case 3:
				// Do nothing. Just set a nil proc, but do NOT inherit from the dialog
				fControl->SetHandler( nil );
				fControl->ClearFlag( pfGUIControlMod::kInheritProcFromDlg );
				break;
		}
	}

	if( INeedsDynamicText() )
	{
		// We're a control that dynamically creates text, so look for the first dynamic layer 
		// (and hopefully the ONLY one) and store it on the control
		Mtl *maxMaterial = hsMaterialConverter::Instance().GetBaseMtl( node );
		hsTArray<plExportMaterialData> *mtlArray = hsMaterialConverter::Instance().CreateMaterialArray( maxMaterial, node, 0 );
		
		UInt32 i, j;
		plDynamicTextMap *dynText = nil;
		plLayerInterface *layerIFace = nil;

		for( i = 0; i < mtlArray->GetCount() && dynText == nil; i++ )
		{
			hsGMaterial	*plasmaMat = (*mtlArray)[ 0 ].fMaterial;

			for( j = 0; j < plasmaMat->GetNumLayers(); j++ )
			{
				layerIFace = plasmaMat->GetLayer( j );
				dynText = plDynamicTextMap::ConvertNoRef( layerIFace->GetTexture() );
				if( dynText != nil )
					break;
			}
		}

		if( dynText == nil )
		{
			pErrMsg->Set( true, "GUI Component Error", "The object %s needs a Plasma Dynamic Text Layer in its material. "
				"This control will not function properly until you apply one.", node->GetName() ).Show();
			pErrMsg->Set( false );
		}
		else
			fControl->SetDynTextMap( layerIFace, dynText );

		delete mtlArray;
	}

	if( ICanHaveProxy() )
	{
		// No proxy objects just yet, just options for better hit testing
		if( fCompPB->GetInt( kRefBetterHitTests ) )
			fControl->SetFlag( pfGUIControlMod::kBetterHitTesting );
	}

	return true;
}

pfGUIControlMod	*plGUIControlBase::GrabControlFromObject( INode *node )
{
	UInt32	i;
	plMaxNodeBase	*maxNode = (plMaxNodeBase *)node;


	for( i = 0; i < maxNode->NumAttachedComponents( false ); i++ )
	{
		plComponentBase *comp = maxNode->GetAttachedComponent( i, false );
		pfGUIControlMod	*ctrl = ConvertCompToControl( comp, maxNode );
		if( ctrl != nil )
			return ctrl;
	}

	return nil;	
}

// Given an INode, gives you a pointer to the GUI component if it actually is one, nil otherwise
plGUIControlBase	*plGUIControlBase::GetGUIComp( INode *node )
{
	if( node == nil )
		return nil;

	return GetGUIComp( ( ( plMaxNodeBase *)node )->ConvertToComponent() );
}

plGUIControlBase	*plGUIControlBase::GetGUIComp( plComponentBase *comp )
{
	if( comp == nil )
		return nil;

	if( comp->ClassID() == GUI_UPDOWNPAIR_CLASSID ||
		comp->ClassID() == GUI_BUTTON_CLASSID ||
		comp->ClassID() == GUI_DRAGGABLE_CLASSID ||
		comp->ClassID() == GUI_LISTBOX_CLASSID ||
		comp->ClassID() == GUI_TEXTBOX_CLASSID ||
		comp->ClassID() == GUI_EDITBOX_CLASSID ||
		comp->ClassID() == GUI_KNOBCTRL_CLASSID ||
		comp->ClassID() == GUI_DRAGBAR_CLASSID ||
		comp->ClassID() == GUI_CHECKBOX_CLASSID ||
		comp->ClassID() == GUI_RADIOGROUP_CLASSID ||
		comp->ClassID() == GUI_DYNDISPLAY_CLASSID ||
		comp->ClassID() == GUI_MULTILINE_CLASSID ||
		comp->ClassID() == GUI_PROGRESS_CLASSID ||
		comp->ClassID() == GUI_CLICKMAP_CLASSID )
	{
		return (plGUIControlBase *)comp;
	}

	return nil;
}

pfGUIControlMod *plGUIControlBase::GrabControlMod( INode *node, INode *sceneObjectNode )
{
	if( node == nil )
		return nil;

	plComponentBase *comp = ( ( plMaxNodeBase *)node )->ConvertToComponent();
	return ConvertCompToControl( comp, sceneObjectNode );
}

pfGUIControlMod *plGUIControlBase::ConvertCompToControl( plComponentBase *comp, INode *sceneObjectNode )
{
	plGUIControlBase	*base = GetGUIComp( comp );
	if( base != nil )
	{
		if( sceneObjectNode == nil )
		{
			// Not good, but if you select a component like this, it better only be applied to one object,
			// hence will only have one fTargetControl
			if( base->fTargetControls.GetCount() > 0 )
				return base->fTargetControls[ 0 ];
		}
		else
		{
			UInt32 i = base->fTargetNodes.Find( (plMaxNode *)sceneObjectNode );
			if( i == base->fTargetNodes.kMissingIndex )
				return nil;

			return base->fTargetControls[ i ];
		}
	}

	return nil;
}

const char	*plGUIControlBase::ISetSoundIndex( ParamID checkBoxID, ParamID sndCompID, UInt8 guiCtrlEvent, plMaxNode *maxNode )
{
	if( fCompPB->GetInt( checkBoxID ) )
	{
		plMaxNode *sndNode = (plMaxNode *)fCompPB->GetReferenceTarget( sndCompID );
		if( sndNode != nil )
		{
			plComponentBase *comp = sndNode->ConvertToComponent();
			if( comp != nil )
			{
				int idx = plAudioComp::GetSoundModIdx( comp, maxNode );
				if( idx != -1 )
				{
					fControl->SetSoundIndex( guiCtrlEvent, idx );
					return nil;
				}
				else
					return "The selected sound component could not be found on GUI control %s. Make sure you have a sound component on the same object selected.";
			}
			else
				return "The selected sound node on GUI control %s could not be converted to a component. Make sure you have a sound component selected.";
		}
		else
			return "The GUI control %s has a sound event enabled but no sound component selected. Make sure you have a sound component on the same object selected.";
	}

	return nil;
}


//// ParamBlock for Control Proc Rollout ////////////////////////////////////////////////////////

static ParamBlockDesc2	sGUIControlProcParamTemplate
(
	/// Main def
	plGUIControlBase::kBlkProc, _T("GUIControlProc"), 0, nil, P_AUTO_UI + P_MULTIMAP + P_AUTO_CONSTRUCT, plComponent::kRefComp, 

	1, 
	plGUIControlBase::kRollProc, IDD_COMP_GUIPROCROLLOUT, IDS_COMP_GUIPROCROLLOUT, 0, 0, nil,	

	plGUIControlBase::kRefChoice,	_T("which"), TYPE_INT, 		0, 0,
		p_ui,	plGUIControlBase::kRollProc, TYPE_RADIO, 4, IDC_GUI_CONRADIO, IDC_GUI_INHERITRADIO, IDC_GUI_CLOSERADIO, IDC_GUI_NILRADIO,
		p_default, 1,
		end,

	plGUIControlBase::kRefConsoleCmd,	_T("ConsoleCmd"),		TYPE_STRING, 		0, 0,
		p_ui,	plGUIControlBase::kRollProc, TYPE_EDITBOX, IDC_GUI_CONCMD,
		end,

	end
);


/////////////////////////////////////////////////////////////////////////////////////////////////
//// plGUIButton Component //////////////////////////////////////////////////////////////////////
//
//	Defines a dialog button to be defined with the GUI manager at runtime. Belongs to exactly 
//	one dialog, defined by parent-child relationship, also at runtime.
//
/////////////////////////////////////////////////////////////////////////////////////////////////

class plGUIButtonProc : public ParamMap2UserDlgProc
{
public:

	void DeleteThis() {}

	BOOL DlgProc( TimeValue t, IParamMap2 *map, HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam );
};


static plGUIButtonProc gGUIButtonProc;


// Class that accesses the paramblock below.
class plGUIButtonComponent : public plGUIControlBase
{
protected:

	virtual pfGUIControlMod	*IGetNewControl( void ) { return TRACKED_NEW pfGUIButtonMod; }
	virtual bool			ICanHaveProxy( void ) { return true; }

public:
	plGUIButtonComponent();
	void DeleteThis() { delete this; }

	// SetupProperties - Internal setup and write-only set properties on the MaxNode. No reading
	// of properties on the MaxNode, as it's still indeterminant.
	hsBool SetupProperties(plMaxNode *pNode, plErrorMsg *pErrMsg);

	hsBool PreConvert(plMaxNode *pNode, plErrorMsg *pErrMsg);
	hsBool Convert(plMaxNode *node, plErrorMsg *pErrMsg);

	enum
	{
		kRefConCmdRadio,
		kRefPythonRadio,
		kRefConsoleCmd,
		kRefAnimate,
		kRefAnimation,
		kRefMouseOverAnimate,
		kRefMouseOverAnimation,
		kRefMouseDownSound,
		kRefMouseDownSoundComp,
		kRefMouseUpSound,
		kRefMouseUpSoundComp,
		kRefMouseOverSound,
		kRefMouseOverSoundComp,
		kRefMouseOffSound,
		kRefMouseOffSoundComp,
		kRefAnimationNode,
		kRefAnimationNodeType,
		kRefMouseOverAnimationNode,
		kRefMouseOverAnimationNodeType,
		kRefDraggableChild,
		kRefUseDraggableChild,
		kRefNotifyType
	};
};

BOOL plGUIButtonProc::DlgProc( TimeValue t, IParamMap2 *pmap, HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam )
{

	switch( msg )
	{
		case WM_INITDIALOG:
			SendMessage( GetDlgItem( hWnd, IDC_COMBO_BUTTON_NOTIFYTYPE ), CB_RESETCONTENT, 0, 0 );
			SendMessage( GetDlgItem( hWnd, IDC_COMBO_BUTTON_NOTIFYTYPE ), CB_ADDSTRING, 0, (LPARAM)"Button Up" );
			SendMessage( GetDlgItem( hWnd, IDC_COMBO_BUTTON_NOTIFYTYPE ), CB_ADDSTRING, 0, (LPARAM)"Button Down" );
			SendMessage( GetDlgItem( hWnd, IDC_COMBO_BUTTON_NOTIFYTYPE ), CB_ADDSTRING, 0, (LPARAM)"Button Down and Up" );
			SendMessage( GetDlgItem( hWnd, IDC_COMBO_BUTTON_NOTIFYTYPE ), CB_SETCURSEL, pmap->GetParamBlock()->GetInt( plGUIButtonComponent::kRefNotifyType ), 0 );
			return true;

		case WM_DESTROY:
			break;

		case WM_COMMAND:
			if( LOWORD( wParam ) == IDC_COMBO_BUTTON_NOTIFYTYPE )
			{
				if( HIWORD( wParam ) == CBN_SELCHANGE )
				{
					int idx = SendDlgItemMessage( hWnd, IDC_COMBO_BUTTON_NOTIFYTYPE, CB_GETCURSEL, 0, 0 );
					pmap->GetParamBlock()->SetValue( plGUIButtonComponent::kRefNotifyType, 0, idx );
				}
			}
			break;
	}
	return false;
}

class plGUIButtonAccessor : public PBAccessor
{
public:
	void Set( PB2Value& v, ReferenceMaker* owner, ParamID id, int tabIndex, TimeValue t )
	{
		if( id == plGUIButtonComponent::kRefAnimation ||
			id == plGUIButtonComponent::kRefMouseOverAnimation ||
			id == plGUIButtonComponent::kRefMouseDownSoundComp ||
			id == plGUIButtonComponent::kRefMouseUpSoundComp ||
			id == plGUIButtonComponent::kRefMouseOverSoundComp ||
			id == plGUIButtonComponent::kRefMouseOffSoundComp ||
			id == plGUIButtonComponent::kRefAnimationNode ||
			id == plGUIButtonComponent::kRefMouseOverAnimationNode )
		{
			plGUIButtonComponent *comp = (plGUIButtonComponent *)owner;
			comp->NotifyDependents( FOREVER, PART_ALL, REFMSG_USER_COMP_REF_CHANGED );
		}
	}
};

Class_ID	sBtnClassesToSelect[] = { ANIM_COMP_CID, ANIM_GROUP_COMP_CID, plGUISingleCtrlDlgProc::kEndClassList };
Class_ID	sBtnSndClassesToSelect[] = { GUI_SOUND_COMPONENT_ID, plGUISingleCtrlDlgProc::kEndClassList };

Class_ID	sBtnDragClassesToSelect[] = { GUI_DRAGGABLE_CLASSID, plGUISingleCtrlDlgProc::kEndClassList };

static plGUIButtonAccessor sGUIButtonAccessor;

static plGUISingleCtrlDlgProc sGUIButtonSndAProc( plGUIButtonComponent::kRefMouseDownSoundComp, IDC_GUI_MDOWNSNDCOMP,
											"Select the sound to play when the mouse clicks this button", sBtnSndClassesToSelect );
static plGUISingleCtrlDlgProc sGUIButtonSndBProc( plGUIButtonComponent::kRefMouseUpSoundComp, IDC_GUI_MUPSNDCOMP,
											"Select the sound to play when the mouse lets up on this button", sBtnSndClassesToSelect );
static plGUISingleCtrlDlgProc sGUIButtonSndCProc( plGUIButtonComponent::kRefMouseOverSoundComp, IDC_GUI_MOVERSNDCOMP,
											"Select the sound to play when the mouse moves over this button", sBtnSndClassesToSelect );
static plGUISingleCtrlDlgProc sGUIButtonSndDProc( plGUIButtonComponent::kRefMouseOffSoundComp, IDC_GUI_MOFFSNDCOMP,
											"Select the sound to play when the mouse moves off of this button", sBtnSndClassesToSelect );

static plGUISingleCtrlDlgProc sGUIButtonDragChildProc( plGUIButtonComponent::kRefDraggableChild, IDC_GUI_DRAGCHILD,
											"Select the draggable to use when the mouse is dragged off of this button", sBtnDragClassesToSelect );

static plGUISingleCtrlDlgProc	*sGUIButtonSubProcs[] = { &sGUIButtonSndAProc, &sGUIButtonSndBProc, 
														  &sGUIButtonSndCProc, &sGUIButtonSndDProc, 
														  &sGUIButtonDragChildProc, nil };
static ParamMap2UserDlgProc	*sGUIButtonSubSubProcs[] = { &gGUIButtonProc, nil };

static plGUIMultipleCtrlDlgProc	sGUIButtonSels( sGUIButtonSubProcs, sGUIButtonSubSubProcs );

static plPlasmaAnimSelectDlgProc	sGUIButtonAnimA( plGUIButtonComponent::kRefAnimation, IDC_GUI_COMPSELBTN, 
													plGUIButtonComponent::kRefAnimationNode, plGUIButtonComponent::kRefAnimationNodeType, IDC_GUI_ANIMNODESEL, 
													"Select the animation to play when this button is clicked", &sGUIButtonSels );
static plPlasmaAnimSelectDlgProc	sGUIButtonProc( plGUIButtonComponent::kRefMouseOverAnimation, IDC_GUI_COMPSELBTN2, 
													plGUIButtonComponent::kRefMouseOverAnimationNode, plGUIButtonComponent::kRefMouseOverAnimationNodeType, IDC_GUI_ANIMNODESEL2, 
													"Select the animation to play when the mouse moves over this button", &sGUIButtonAnimA );


#define GUI_SOUND_REF( comp, evt, allCapsEvt )		\
	 	comp##::kRefMouse##evt##Sound,	_T( "mouse##evt##Sound" ), TYPE_BOOL, 0, 0,					\
			p_ui, plGUIControlBase::kRollMain, TYPE_SINGLECHEKBOX, IDC_GUI_M##allCapsEvt##SND,		\
			p_default, FALSE,																		\
			p_enable_ctrls, 1, comp##::kRefMouse##evt##SoundComp,									\
			end,																					\
		comp##::kRefMouse##evt##SoundComp, _T("mouse##evt##SoundComp"),	TYPE_INODE,		0, 0,		\
			p_accessor, &sGUIButtonAccessor,														\
			end

//Max desc stuff necessary below.
CLASS_DESC(plGUIButtonComponent, gGUIButtonDesc, "GUI Button",  "GUIButton", COMP_TYPE_GUI, GUI_BUTTON_CLASSID )

ParamBlockDesc2 gGUIButtonBk
(	// KLUDGE: not the defined block ID, but kept for backwards compat.
 plComponent::kBlkComp, _T("GUIButton"), 0, &gGUIButtonDesc, P_AUTO_CONSTRUCT + P_AUTO_UI + P_INCLUDE_PARAMS + P_MULTIMAP, plComponent::kRefComp,

	3,
	plGUIControlBase::kRollMain, IDD_COMP_GUIBUTTON, IDS_COMP_GUIBUTTON, 0, 0, &sGUIButtonProc,
	plGUIControlBase::kRollProc, IDD_COMP_GUIPROCROLLOUT, IDS_COMP_GUIPROCROLLOUT, 0, 0, nil,	
	sGUIProxyParamHeader,

	&sGUIControlProcParamTemplate,

		plGUIButtonComponent::kRefAnimate,	_T( "animate" ), TYPE_BOOL, 0, 0,
			p_ui, plGUIControlBase::kRollMain, TYPE_SINGLECHEKBOX, IDC_GUI_ANIMATE,
			p_default, FALSE,
			p_enable_ctrls, 1, plGUIButtonComponent::kRefAnimation,
			end,

		plGUIButtonComponent::kRefAnimation, _T("animation"),	TYPE_INODE,		0, 0,
			p_prompt, IDS_COMP_GUI_SELECTANIM,
			p_accessor, &sGUIButtonAccessor,
			end,

		plGUIButtonComponent::kRefMouseOverAnimate,	_T( "mouseOverAnimate" ), TYPE_BOOL, 0, 0,
			p_ui, plGUIControlBase::kRollMain, TYPE_SINGLECHEKBOX, IDC_GUI_MOUSEOVERANIM,
			p_default, FALSE,
			p_enable_ctrls, 1, plGUIButtonComponent::kRefMouseOverAnimation,
			end,

		plGUIButtonComponent::kRefMouseOverAnimation, _T("mouseOverAnimation"),	TYPE_INODE,		0, 0,
			p_prompt, IDS_COMP_GUI_SELECTMOUSEOVERANIM,
			p_accessor, &sGUIButtonAccessor,
			end,

		GUI_SOUND_REF( plGUIButtonComponent, Down, DOWN ),
		GUI_SOUND_REF( plGUIButtonComponent, Up, UP ),
		GUI_SOUND_REF( plGUIButtonComponent, Over, OVER ),
		GUI_SOUND_REF( plGUIButtonComponent, Off, OFF ),

	sGUIProxyParamTemplate,

		plGUIButtonComponent::kRefAnimationNode, _T("animationNode"),	TYPE_INODE,		0, 0,
			p_accessor, &sGUIButtonAccessor,
			end,

		plGUIButtonComponent::kRefAnimationNodeType, _T("animationNodeType"),	TYPE_INT,		0, 0,
			p_default, plAnimObjInterface::kUseOwnerNode,
			end,

		plGUIButtonComponent::kRefMouseOverAnimationNode, _T("moAnimationNode"),	TYPE_INODE,		0, 0,
			p_accessor, &sGUIButtonAccessor,
			end,

		plGUIButtonComponent::kRefMouseOverAnimationNodeType, _T("moAnimationNodeType"),	TYPE_INT,		0, 0,
			p_default, plAnimObjInterface::kUseOwnerNode,
			end,

		plGUIButtonComponent::kRefUseDraggableChild,	_T( "useDragChild" ), TYPE_BOOL, 0, 0,
			p_ui, plGUIControlBase::kRollMain, TYPE_SINGLECHEKBOX, IDC_GUI_USEDRAGCHILD,
			p_default, FALSE,
			p_enable_ctrls, 1, plGUIButtonComponent::kRefDraggableChild,
			end,

		plGUIButtonComponent::kRefDraggableChild,	_T( "dragChild" ), TYPE_INODE, 0, 0,
			p_accessor, &sGUIButtonAccessor,
			end,

		plGUIButtonComponent::kRefNotifyType,	_T("notifyType"),		TYPE_INT, 		0, 0,
			p_default, pfGUIButtonMod::kNotifyOnUp,
			end,
	end
);

plGUIButtonComponent::plGUIButtonComponent()
{
	fClassDesc = &gGUIButtonDesc;
	fClassDesc->MakeAutoParamBlocks(this);
}

// SetupProperties - Internal setup and write-only set properties on the MaxNode. No reading
// of properties on the MaxNode, as it's still indeterminant.
hsBool plGUIButtonComponent::SetupProperties(plMaxNode *node,  plErrorMsg *pErrMsg)
{
	if( fCompPB->GetInt( kRefAnimate ) )
	{
		plAnimObjInterface *iface = plAnimComponentBase::GetAnimInterface( fCompPB->GetINode( kRefAnimation ) );
		if( iface != nil && iface->MightRequireSeparateMaterial() )
		{
			INode *restrict = ( fCompPB->GetInt( kRefAnimationNodeType ) == plAnimObjInterface::kUseParamBlockNode ) 
									? fCompPB->GetINode( kRefAnimationNode )
									: (INode *)node;

			if( restrict != nil )
			{
				node->SetForceMaterialCopy( true );
			}
		}
	}

	if( fCompPB->GetInt( kRefMouseOverAnimate ) )
	{
		plAnimObjInterface *iface = plAnimComponentBase::GetAnimInterface( fCompPB->GetINode( kRefMouseOverAnimation ) );
		if( iface != nil && iface->MightRequireSeparateMaterial() )
		{
			INode *restrict = ( fCompPB->GetInt( kRefMouseOverAnimationNodeType ) == plAnimObjInterface::kUseParamBlockNode ) 
									? fCompPB->GetINode( kRefMouseOverAnimationNode )
									: (INode *)node;

			if( restrict != nil )
			{
				node->SetForceMaterialCopy( true );
			}
		}
	}

	return plGUIControlBase::SetupProperties( node, pErrMsg );
}

hsBool plGUIButtonComponent::PreConvert(plMaxNode *node,  plErrorMsg *pErrMsg)
{
	return plGUIControlBase::PreConvert( node, pErrMsg );
}

hsBool plGUIButtonComponent::Convert(plMaxNode *node, plErrorMsg *pErrMsg)
{
	if( !plGUIControlBase::Convert( node, pErrMsg ) )
		return false;

	pfGUIButtonMod *button = (pfGUIButtonMod *)fControl;

	// set the notify type
	button->SetNotifyType(fCompPB->GetInt( kRefNotifyType ));

	if( fCompPB->GetInt( kRefAnimate ) )
	{
		plAnimObjInterface *iface = plAnimComponentBase::GetAnimInterface( fCompPB->GetINode( kRefAnimation ) );
		if( iface != nil )
		{
			INode *restrict = ( fCompPB->GetInt( kRefAnimationNodeType ) == plAnimObjInterface::kUseParamBlockNode ) 
									? fCompPB->GetINode( kRefAnimationNode )
									: (INode *)node;


			hsTArray<plKey> keys;
			if( iface->GetKeyList( restrict, keys ) && keys.GetCount() > 0 )
				button->SetAnimationKeys( keys, iface->GetIfaceSegmentName( false ) );
		}
	}

	if( fCompPB->GetInt( kRefMouseOverAnimate ) )
	{
		plAnimObjInterface *iface = plAnimComponentBase::GetAnimInterface( fCompPB->GetINode( kRefMouseOverAnimation ) );
		if( iface != nil )
		{
			INode *restrict = ( fCompPB->GetInt( kRefMouseOverAnimationNodeType ) == plAnimObjInterface::kUseParamBlockNode ) 
									? fCompPB->GetINode( kRefMouseOverAnimationNode )
									: (INode *)node;


			hsTArray<plKey> keys;
			if( iface->GetKeyList( restrict, keys ) && keys.GetCount() > 0 )
				button->SetMouseOverAnimKeys( keys, iface->GetIfaceSegmentName( false ) );
		}
	}

	// Do sound stuff
	const char *errMsg1 = ISetSoundIndex( kRefMouseDownSound, kRefMouseDownSoundComp, pfGUIButtonMod::kMouseDown, node );
	const char *errMsg2 = ISetSoundIndex( kRefMouseUpSound, kRefMouseUpSoundComp, pfGUIButtonMod::kMouseUp, node );
	const char *errMsg3 = ISetSoundIndex( kRefMouseOverSound, kRefMouseOverSoundComp, pfGUIButtonMod::kMouseOver, node );
	const char *errMsg4 = ISetSoundIndex( kRefMouseOffSound, kRefMouseOffSoundComp, pfGUIButtonMod::kMouseOff, node );

	const char *errMsg = ( errMsg1 != nil ) ? errMsg1 : ( errMsg2 != nil ) ? errMsg2 : ( errMsg3 != nil ) ? errMsg3 : errMsg4;
	if( errMsg != nil )
	{
		pErrMsg->Set( true, "GUI Sound Event Error", errMsg, node->GetName() ).Show();
		pErrMsg->Set( false );
	}

	if( fCompPB->GetInt( kRefUseDraggableChild ) )
	{
		pfGUIDraggableMod *dragChild = pfGUIDraggableMod::ConvertNoRef( GrabControlMod( fCompPB->GetINode( kRefDraggableChild ) ) );
		if( dragChild != nil )
		{
			hsgResMgr::ResMgr()->AddViaNotify( dragChild->GetKey(), 
								new plGenRefMsg( button->GetKey(), plRefMsg::kOnCreate, -1, pfGUIButtonMod::kRefDraggable ), plRefFlags::kActiveRef );
		}
	}

	return true;
}

/////////////////////////////////////////////////////////////////////////////////////////////////
//// plGUICheckBox Component //////////////////////////////////////////////////////////////////////
//
//	Defines a dialog button to be defined with the GUI manager at runtime. Belongs to exactly 
//	one dialog, defined by parent-child relationship, also at runtime.
//
/////////////////////////////////////////////////////////////////////////////////////////////////


// Class that accesses the paramblock below.
class plGUICheckBoxComponent : public plGUIControlBase
{
protected:

	virtual pfGUIControlMod	*IGetNewControl( void ) { return TRACKED_NEW pfGUICheckBoxCtrl; }
	virtual bool			ICanHaveProxy( void ) { return true; }

public:
	plGUICheckBoxComponent();
	void DeleteThis() { delete this; }

	// SetupProperties - Internal setup and write-only set properties on the MaxNode. No reading
	// of properties on the MaxNode, as it's still indeterminant.
	hsBool SetupProperties(plMaxNode *pNode, plErrorMsg *pErrMsg);

	hsBool PreConvert(plMaxNode *pNode, plErrorMsg *pErrMsg);
	hsBool Convert(plMaxNode *node, plErrorMsg *pErrMsg);

	enum
	{
		kRefConCmdRadio,
		kRefPythonRadio,
		kRefConsoleCmd,
		kRefAnimate,
		kRefAnimation,
		kRefAnimationNode,
		kRefAnimationNodeType,
		kRefMouseDownSound,
		kRefMouseDownSoundComp,
		kRefMouseUpSound,
		kRefMouseUpSoundComp,
		kRefMouseOverSound,
		kRefMouseOverSoundComp,
		kRefMouseOffSound,
		kRefMouseOffSoundComp,
	};
};

class plGUICheckBoxAccessor : public PBAccessor
{
public:
	void Set( PB2Value& v, ReferenceMaker* owner, ParamID id, int tabIndex, TimeValue t )
	{
		if( id == plGUICheckBoxComponent::kRefAnimation ||
			id == plGUICheckBoxComponent::kRefMouseDownSoundComp ||
			id == plGUICheckBoxComponent::kRefMouseUpSoundComp ||
			id == plGUICheckBoxComponent::kRefMouseOverSoundComp ||
			id == plGUICheckBoxComponent::kRefMouseOffSoundComp )
		{
			plGUICheckBoxComponent *comp = (plGUICheckBoxComponent *)owner;
			comp->NotifyDependents( FOREVER, PART_ALL, REFMSG_USER_COMP_REF_CHANGED );
		}
	}
};
static plGUICheckBoxAccessor	sGUICheckBoxAccessor;

static plGUISingleCtrlDlgProc sGUICheckSndAProc( plGUICheckBoxComponent::kRefMouseDownSoundComp, IDC_GUI_MDOWNSNDCOMP,
											"Select the sound to play when the mouse clicks this button", sBtnSndClassesToSelect );
static plGUISingleCtrlDlgProc sGUICheckSndBProc( plGUICheckBoxComponent::kRefMouseUpSoundComp, IDC_GUI_MUPSNDCOMP,
											"Select the sound to play when the mouse lets up on this button", sBtnSndClassesToSelect );
static plGUISingleCtrlDlgProc sGUICheckSndCProc( plGUICheckBoxComponent::kRefMouseOverSoundComp, IDC_GUI_MOVERSNDCOMP,
											"Select the sound to play when the mouse moves over this button", sBtnSndClassesToSelect );
static plGUISingleCtrlDlgProc sGUICheckSndDProc( plGUICheckBoxComponent::kRefMouseOffSoundComp, IDC_GUI_MOFFSNDCOMP,
											"Select the sound to play when the mouse moves off of this button", sBtnSndClassesToSelect );

static plGUISingleCtrlDlgProc	*sGUICheckSubProcs[] = { &sGUICheckSndAProc, &sGUICheckSndBProc, 
														  &sGUICheckSndCProc, &sGUICheckSndDProc, nil };

static plGUIMultipleCtrlDlgProc	sGUICheckSels( sGUICheckSubProcs );

static plPlasmaAnimSelectDlgProc	sGUICheckBoxProc( plGUICheckBoxComponent::kRefAnimation, IDC_GUI_COMPSELBTN, 
													plGUICheckBoxComponent::kRefAnimationNode, plGUICheckBoxComponent::kRefAnimationNodeType, IDC_GUI_ANIMNODESEL, 
													"Select the animation to play when this check box is clicked", &sGUICheckSels );

//Max desc stuff necessary below.
CLASS_DESC(plGUICheckBoxComponent, gGUICheckBoxDesc, "GUI CheckBox",  "GUICheckBox", COMP_TYPE_GUI, GUI_CHECKBOX_CLASSID )

ParamBlockDesc2 gGUICheckBoxBk
(	// KLUDGE: not the defined block ID, but kept for backwards compat.
 plComponent::kBlkComp, _T("GUICheckBox"), 0, &gGUICheckBoxDesc, P_AUTO_CONSTRUCT + P_AUTO_UI + P_MULTIMAP + P_INCLUDE_PARAMS, plComponent::kRefComp,

	3,
	plGUIControlBase::kRollMain, IDD_COMP_GUIBUTTON, IDS_COMP_GUICHECKBOX, 0, 0, &sGUICheckBoxProc,
	plGUIControlBase::kRollProc, IDD_COMP_GUIPROCROLLOUT, IDS_COMP_GUIPROCROLLOUT, 0, 0, nil,	
	
	sGUIProxyParamHeader,

	&sGUIControlProcParamTemplate,

		plGUICheckBoxComponent::kRefAnimate,	_T( "animate" ), TYPE_BOOL, 0, 0,
			p_ui, plGUIControlBase::kRollMain, TYPE_SINGLECHEKBOX, IDC_GUI_ANIMATE,
			p_default, FALSE,
			p_enable_ctrls, 1, plGUIButtonComponent::kRefAnimation,
			end,

		plGUICheckBoxComponent::kRefAnimation, _T("animation"),	TYPE_INODE,		0, 0,
			p_prompt, IDS_COMP_GUI_SELECTANIM,
			p_accessor, &sGUICheckBoxAccessor,
			end,

	sGUIProxyParamTemplate,

		plGUICheckBoxComponent::kRefAnimationNode, _T("animationNode"),	TYPE_INODE,		0, 0,
			p_accessor, &sGUIButtonAccessor,
			end,

		plGUICheckBoxComponent::kRefAnimationNodeType, _T("animationNodeType"),	TYPE_INT,		0, 0,
			p_default, plAnimObjInterface::kUseOwnerNode,
			end,

		GUI_SOUND_REF( plGUICheckBoxComponent, Down, DOWN ),
		GUI_SOUND_REF( plGUICheckBoxComponent, Up, UP ),
		GUI_SOUND_REF( plGUICheckBoxComponent, Over, OVER ),
		GUI_SOUND_REF( plGUICheckBoxComponent, Off, OFF ),

	end
);

plGUICheckBoxComponent::plGUICheckBoxComponent()
{
	fClassDesc = &gGUICheckBoxDesc;
	fClassDesc->MakeAutoParamBlocks(this);
}

// SetupProperties - Internal setup and write-only set properties on the MaxNode. No reading
// of properties on the MaxNode, as it's still indeterminant.
hsBool plGUICheckBoxComponent::SetupProperties(plMaxNode *node,  plErrorMsg *pErrMsg)
{
	if( fCompPB->GetInt( kRefAnimate ) )
	{
		plAnimObjInterface *iface = plAnimComponentBase::GetAnimInterface( fCompPB->GetINode( kRefAnimation ) );
		if( iface != nil && iface->MightRequireSeparateMaterial() )
		{
			INode *restrict = ( fCompPB->GetInt( kRefAnimationNodeType ) == plAnimObjInterface::kUseParamBlockNode ) 
									? fCompPB->GetINode( kRefAnimationNode )
									: (INode *)node;

			if( restrict != nil )
			{
				node->SetForceMaterialCopy( true );
			}
		}
	}

	return plGUIControlBase::SetupProperties( node, pErrMsg );
}

hsBool plGUICheckBoxComponent::PreConvert(plMaxNode *node,  plErrorMsg *pErrMsg)
{
	return plGUIControlBase::PreConvert( node, pErrMsg );
}

hsBool plGUICheckBoxComponent::Convert(plMaxNode *node, plErrorMsg *pErrMsg)
{
	if( !plGUIControlBase::Convert( node, pErrMsg ) )
		return false;

	pfGUICheckBoxCtrl *button = (pfGUICheckBoxCtrl *)fControl;
	
	if( fCompPB->GetInt( kRefAnimate ) )
	{
		plAnimObjInterface *iface = plAnimComponentBase::GetAnimInterface( fCompPB->GetINode( kRefAnimation ) );
		if( iface != nil )
		{
			INode *restrict = ( fCompPB->GetInt( kRefAnimationNodeType ) == plAnimObjInterface::kUseParamBlockNode ) 
									? fCompPB->GetINode( kRefAnimationNode )
									: (INode *)node;


			hsTArray<plKey> keys;
			if( iface->GetKeyList( restrict, keys ) && keys.GetCount() > 0 )
				button->SetAnimationKeys( keys, iface->GetIfaceSegmentName( false ) );
		}
	}

	// Do sound stuff
	const char *errMsg1 = ISetSoundIndex( kRefMouseDownSound, kRefMouseDownSoundComp, pfGUICheckBoxCtrl::kMouseDown, node );
	const char *errMsg2 = ISetSoundIndex( kRefMouseUpSound, kRefMouseUpSoundComp, pfGUICheckBoxCtrl::kMouseUp, node );
	const char *errMsg3 = ISetSoundIndex( kRefMouseOverSound, kRefMouseOverSoundComp, pfGUICheckBoxCtrl::kMouseOver, node );
	const char *errMsg4 = ISetSoundIndex( kRefMouseOffSound, kRefMouseOffSoundComp, pfGUICheckBoxCtrl::kMouseOff, node );

	const char *errMsg = ( errMsg1 != nil ) ? errMsg1 : ( errMsg2 != nil ) ? errMsg2 : ( errMsg3 != nil ) ? errMsg3 : errMsg4;
	if( errMsg != nil )
	{
		pErrMsg->Set( true, "GUI Sound Event Error", errMsg, node->GetName() ).Show();
		pErrMsg->Set( false );
	}

	return true;
}

/////////////////////////////////////////////////////////////////////////////////////////////////
//// plGUIDraggable Component ///////////////////////////////////////////////////////////////////
//
//	GUI element that can be dragged anywhere in the 2D viewing plane.
//
/////////////////////////////////////////////////////////////////////////////////////////////////


// Class that accesses the paramblock below.
class plGUIDraggableComponent : public plGUIControlBase
{
protected:

	virtual pfGUIControlMod	*IGetNewControl( void ) { return TRACKED_NEW pfGUIDraggableMod; }
	virtual bool			ICanHaveProxy( void ) { return true; }

public:
	plGUIDraggableComponent();
	void DeleteThis() { delete this; }

	// SetupProperties - Internal setup and write-only set properties on the MaxNode. No reading
	// of properties on the MaxNode, as it's still indeterminant.
	hsBool SetupProperties(plMaxNode *pNode, plErrorMsg *pErrMsg);

	hsBool PreConvert(plMaxNode *pNode, plErrorMsg *pErrMsg);
	hsBool Convert(plMaxNode *node, plErrorMsg *pErrMsg);

	enum
	{
		kRefReportDragging,
		kRefHideCursor,
		kRefAlwaysSnap
	};
};

//Max desc stuff necessary below.
CLASS_DESC(plGUIDraggableComponent, gGUIDraggableDesc, "GUI Draggable",  "GUIDraggable", COMP_TYPE_GUI, GUI_DRAGGABLE_CLASSID )

ParamBlockDesc2 gGUIDraggableBk
(	// KLUDGE: not the defined block ID, but kept for backwards compat.
 plComponent::kBlkComp, _T("GUIDraggable"), 0, &gGUIDraggableDesc, P_AUTO_CONSTRUCT + P_AUTO_UI + P_MULTIMAP + P_INCLUDE_PARAMS, plComponent::kRefComp,

	3,
	plGUIControlBase::kRollMain, IDD_COMP_GUIDRAGGABLE, IDS_COMP_GUIDRAGGABLE, 0, 0, NULL,
	plGUIControlBase::kRollProc, IDD_COMP_GUIPROCROLLOUT, IDS_COMP_GUIPROCROLLOUT, 0, 0, nil,	
	
	sGUIProxyParamHeader,
	
	&sGUIControlProcParamTemplate,

	sGUIProxyParamTemplate,

	plGUIDraggableComponent::kRefReportDragging, _T("reportWhileDragging"), TYPE_BOOL, 0, 0,
		p_ui, plGUIControlBase::kRollMain, TYPE_SINGLECHEKBOX, IDC_GUI_REPORTDRAG,
		p_default, FALSE,
		end,

	plGUIDraggableComponent::kRefHideCursor, _T("hideCursorWhileDragging"), TYPE_BOOL, 0, 0,
		p_ui, plGUIControlBase::kRollMain, TYPE_SINGLECHEKBOX, IDC_GUI_HIDECURSOR,
		p_default, FALSE,
		end,

	plGUIDraggableComponent::kRefAlwaysSnap, _T("alwaysSnapBackToStart"), TYPE_BOOL, 0, 0,
		p_ui, plGUIControlBase::kRollMain, TYPE_SINGLECHEKBOX, IDC_GUI_SNAPSTART,
		p_default, FALSE,
		end,

	end
);

plGUIDraggableComponent::plGUIDraggableComponent()
{
	fClassDesc = &gGUIDraggableDesc;
	fClassDesc->MakeAutoParamBlocks(this);
}

// SetupProperties - Internal setup and write-only set properties on the MaxNode. No reading
// of properties on the MaxNode, as it's still indeterminant.
hsBool plGUIDraggableComponent::SetupProperties(plMaxNode *node,  plErrorMsg *pErrMsg)
{
	node->SetForceLocal( true );
	return plGUIControlBase::SetupProperties( node, pErrMsg );
}

hsBool plGUIDraggableComponent::PreConvert(plMaxNode *node,  plErrorMsg *pErrMsg)
{
	return plGUIControlBase::PreConvert( node, pErrMsg );
}

hsBool plGUIDraggableComponent::Convert(plMaxNode *node, plErrorMsg *pErrMsg)
{
	if( !plGUIControlBase::Convert( node, pErrMsg ) )
		return false;

	pfGUIDraggableMod *ctrl = (pfGUIDraggableMod *)fControl;
	
	if( fCompPB->GetInt( kRefReportDragging ) )
		ctrl->SetFlag( pfGUIDraggableMod::kReportDragging );

	if( fCompPB->GetInt( kRefHideCursor ) )
		ctrl->SetFlag( pfGUIDraggableMod::kHideCursorWhileDragging );

	if( fCompPB->GetInt( kRefAlwaysSnap ) )
		ctrl->SetFlag( pfGUIDraggableMod::kAlwaysSnapBackToStart );

	return true;
}


/////////////////////////////////////////////////////////////////////////////////////////////////
//// plGUIKnobCtrl Component ///////////////////////////////////////////////////////////////////
//
//	GUI element that can be dragged anywhere in the 2D viewing plane.
//
/////////////////////////////////////////////////////////////////////////////////////////////////


// Class that accesses the paramblock below.
class plGUIKnobCtrlComponent : public plGUIControlBase
{
protected:

	virtual pfGUIControlMod	*IGetNewControl( void ) { return TRACKED_NEW pfGUIKnobCtrl; }
	virtual bool			ICanHaveProxy( void ) { return true; }

	hsBool	IGrabAnimationRange( plMaxNode *node, plErrorMsg *pErrMsg, hsMatrix44 &startL2W, hsMatrix44 &endL2W );

public:
	plGUIKnobCtrlComponent();
	void DeleteThis() { delete this; }

	// SetupProperties - Internal setup and write-only set properties on the MaxNode. No reading
	// of properties on the MaxNode, as it's still indeterminant.
	hsBool SetupProperties(plMaxNode *pNode, plErrorMsg *pErrMsg);

	hsBool PreConvert(plMaxNode *pNode, plErrorMsg *pErrMsg);
	hsBool Convert(plMaxNode *node, plErrorMsg *pErrMsg);

	enum
	{
		kRefMinValue,
		kRefMaxValue,
		kRefStep,
		kReverseValues,
		kRefOrientation,
		kRefMouseMapping,
		kRefTriggerOnMouseUp,
		kRefAnimation,
		kRefAnimationNode,
		kRefAnimationNodeType
	};
};

static plPlasmaAnimSelectDlgProc	sGUIKnobCtrlProc( plGUIKnobCtrlComponent::kRefAnimation, IDC_GUI_COMPSELBTN, 
													plGUIKnobCtrlComponent::kRefAnimationNode, plGUIKnobCtrlComponent::kRefAnimationNodeType, IDC_GUI_ANIMNODESEL, 
													"Select the animation to use when displaying this knob control", nil );

//Max desc stuff necessary below.
CLASS_DESC(plGUIKnobCtrlComponent, gGUIKnobCtrlDesc, "GUI Knob Control",  "GUIKnobCtrl", COMP_TYPE_GUI, GUI_KNOBCTRL_CLASSID )

ParamBlockDesc2 gGUIKnobCtrlBk
(	// KLUDGE: not the defined block ID, but kept for backwards compat.
 plComponent::kBlkComp, _T("GUIKnobCtrl"), 0, &gGUIKnobCtrlDesc, P_AUTO_CONSTRUCT + P_AUTO_UI + P_MULTIMAP + P_INCLUDE_PARAMS, plComponent::kRefComp,

	3,
	plGUIControlBase::kRollMain, IDD_COMP_GUIKNOB, IDS_COMP_GUIKNOB, 0, 0, &sGUIKnobCtrlProc,
	plGUIControlBase::kRollProc, IDD_COMP_GUIPROCROLLOUT, IDS_COMP_GUIPROCROLLOUT, 0, 0, nil,	
	
	sGUIProxyParamHeader,
	
	&sGUIControlProcParamTemplate,
	
	plGUIKnobCtrlComponent::kRefMinValue, _T("minValue"),	TYPE_FLOAT,	0, 0,	
		p_default, 0.0f,
		p_ui, plGUIControlBase::kRollMain, TYPE_SPINNER,	EDITTYPE_POS_FLOAT, IDC_GUI_LOWER, IDC_GUI_LOWER_SPIN, SPIN_AUTOSCALE,
		end,

	plGUIKnobCtrlComponent::kRefMaxValue, _T("maxValue"),	TYPE_FLOAT,	0, 0,	
		p_default, 10.0f,
		p_range, -10000.f, 10000.f,			// WHY do we even need to specify this?
		p_ui, plGUIControlBase::kRollMain, TYPE_SPINNER,	EDITTYPE_POS_FLOAT, IDC_GUI_UPPER, IDC_GUI_UPPER_SPIN, SPIN_AUTOSCALE,
		end,

	plGUIKnobCtrlComponent::kRefStep, _T("step"),	TYPE_FLOAT,	0, 0,	
		p_default, 1.0f,
		p_ui, plGUIControlBase::kRollMain, TYPE_SPINNER,	EDITTYPE_POS_FLOAT, IDC_GUI_STEP, IDC_GUI_STEP_SPIN, SPIN_AUTOSCALE,
		end,

	plGUIKnobCtrlComponent::kReverseValues, _T("reverseValues"), TYPE_BOOL, 0, 0,
		p_ui, plGUIControlBase::kRollMain, TYPE_SINGLECHEKBOX, IDC_GUI_REVERSE,
		p_default, FALSE,
		end,

	plGUIKnobCtrlComponent::kRefOrientation,	_T("orientation"), TYPE_INT, 		0, 0,
		p_ui,	plGUIControlBase::kRollMain, TYPE_RADIO, 2, IDC_ORIENTATION_RADIO, IDC_ORIENTATION_RADIO2,
		p_default, 0,
		end,
	
	plGUIKnobCtrlComponent::kRefMouseMapping, _T("mouseMapping"), TYPE_INT, 0, 0,
		p_ui, plGUIControlBase::kRollMain, TYPE_RADIO, 3, IDC_GUI_MOUSEMAPREL, IDC_GUI_MOUSEMAPANIM, IDC_GUI_MOUSEMAPSCRN,
		p_default, 0,
		end,

	plGUIKnobCtrlComponent::kRefTriggerOnMouseUp, _T("triggerOnMouseUp"), TYPE_BOOL, 0, 0,
		p_ui, plGUIControlBase::kRollMain, TYPE_SINGLECHEKBOX, IDC_GUI_TRIGGERONUP,
		p_default, FALSE,
		end,
		
	sGUIProxyParamTemplate,

	plGUIKnobCtrlComponent::kRefAnimation, _T("animation"),	TYPE_INODE,		0, 0,
		p_prompt, IDS_COMP_GUI_SELECTANIM,
		end,

	plGUIKnobCtrlComponent::kRefAnimationNode, _T("animationNode"),	TYPE_INODE,		0, 0,
		end,

	plGUIKnobCtrlComponent::kRefAnimationNodeType, _T("animationNodeType"),	TYPE_INT,		0, 0,
		p_default, plAnimObjInterface::kUseOwnerNode,
		end,

	end
);

plGUIKnobCtrlComponent::plGUIKnobCtrlComponent()
{
	fClassDesc = &gGUIKnobCtrlDesc;
	fClassDesc->MakeAutoParamBlocks(this);
}

hsBool	plGUIKnobCtrlComponent::IGrabAnimationRange( plMaxNode *node, plErrorMsg *pErrMsg, hsMatrix44 &startL2W, hsMatrix44 &endL2W )
{
	hsBool	result = false;


	// Get the affine parts and the TM Controller
	plSceneObject *obj = node->GetSceneObject();
	hsAffineParts * parts = TRACKED_NEW hsAffineParts;
	plController* tmc = hsControlConverter::Instance().ConvertTMAnim(obj, node, parts);

	if (tmc)
	{
		plMatrixControllerChannel *channel = TRACKED_NEW plMatrixControllerChannel(tmc, parts);

		hsScalar length = tmc->GetLength();

		startL2W = channel->Value( 0.f );
		endL2W = channel->Value( length );

		delete channel;
		result = true;
	}

	delete parts;	// We copy this over, so no need to keep it around
	return result;
}

// SetupProperties - Internal setup and write-only set properties on the MaxNode. No reading
// of properties on the MaxNode, as it's still indeterminant.
hsBool plGUIKnobCtrlComponent::SetupProperties(plMaxNode *node,  plErrorMsg *pErrMsg)
{
	node->SetForceLocal( true );

	plAnimObjInterface *iface = plAnimComponentBase::GetAnimInterface( fCompPB->GetINode( kRefAnimation ) );
	if( iface != nil && iface->MightRequireSeparateMaterial() )
	{
		INode *restrict = ( fCompPB->GetInt( kRefAnimationNodeType ) == plAnimObjInterface::kUseParamBlockNode ) 
								? fCompPB->GetINode( kRefAnimationNode )
								: (INode *)node;

		if( restrict != nil )
		{
			node->SetForceMaterialCopy( true );
		}
	}

	return plGUIControlBase::SetupProperties( node, pErrMsg );
}

hsBool plGUIKnobCtrlComponent::PreConvert(plMaxNode *node,  plErrorMsg *pErrMsg)
{
	return plGUIControlBase::PreConvert( node, pErrMsg );
}

// For hackery below (see warning below)
#include "../plAvatar/plAGMasterMod.h"

hsBool plGUIKnobCtrlComponent::Convert(plMaxNode *node, plErrorMsg *pErrMsg)
{
	if( !plGUIControlBase::Convert( node, pErrMsg ) )
		return false;

	pfGUIKnobCtrl *ctrl = (pfGUIKnobCtrl *)fControl;
	
	ctrl->SetRange( fCompPB->GetFloat( kRefMinValue ), fCompPB->GetFloat( kRefMaxValue ) );
	ctrl->SetStep( fCompPB->GetFloat( kRefStep ) );

	if( fCompPB->GetInt( kReverseValues ) )
		ctrl->SetFlag( pfGUIKnobCtrl::kReverseValues );

	// Get the animation to use
	plAnimObjInterface *iface = plAnimComponentBase::GetAnimInterface( fCompPB->GetINode( kRefAnimation ) );
	if( iface != nil )
	{
		INode *restrict = ( fCompPB->GetInt( kRefAnimationNodeType ) == plAnimObjInterface::kUseParamBlockNode ) 
								? fCompPB->GetINode( kRefAnimationNode )
								: (INode *)node;


		hsTArray<plKey> keys;
		if( iface->GetKeyList( restrict, keys ) && keys.GetCount() > 0 )
			ctrl->SetAnimationKeys( keys, iface->GetIfaceSegmentName( false ) );
	}
	else
	{
		// HACKERY WARNING: Old knobs assumed the animation was just the same one applied to our node,
		// so to avoid breaking old formats, if we can't grab an animObjInterface, we just grab the key
		// of the master mod of our node, like we would've before
		plAGMasterMod	*master = node->GetAGMasterMod();
		hsTArray<plKey> keys;
		keys.Append( master->GetKey() );
		ctrl->SetAnimationKeys( keys, ENTIRE_ANIMATION_NAME );
	}

	if( fCompPB->GetInt( kRefOrientation ) == 1 )
		ctrl->SetFlag( pfGUIKnobCtrl::kLeftRightOrientation );

	hsMatrix44 startL2W, endL2W;
	switch( fCompPB->GetInt( kRefMouseMapping ) )
	{
		case 0:		// Default, normal (old) relative behavior
			break;
		case 1:		// Map to the range of animation positions
			if( !IGrabAnimationRange( node, pErrMsg, startL2W, endL2W ) )
			{
				pErrMsg->Set( true, "Unable to grab animation range for the GUI Knob Control %s. The Map-To-Screen-Range feature will be disabled.", node->GetName() ).Show();
				pErrMsg->Set( false );
			}
			else
			{
				hsPoint3 startPos = startL2W.GetTranslate();
				hsPoint3 endPos = endL2W.GetTranslate();

				ctrl->SetScreenRange( startPos, endPos );
				ctrl->SetFlag( pfGUIKnobCtrl::kMapToAnimationRange );
			}
			break;
		case 2:		// Map to a range on the screen
			ctrl->SetFlag( pfGUIKnobCtrl::kMapToScreenRange );
			break;
	}

	if( fCompPB->GetInt( kRefTriggerOnMouseUp ) )
		ctrl->SetFlag( pfGUIKnobCtrl::kTriggerOnlyOnMouseUp );

	return true;
}

/////////////////////////////////////////////////////////////////////////////////////////////////
//// plGUIListBox Component /////////////////////////////////////////////////////////////////////
//
//	GUI element that can be dragged anywhere in the 2D viewing plane.
//
/////////////////////////////////////////////////////////////////////////////////////////////////


// Class that accesses the paramblock below.
class plGUIListBoxComponent : public plGUIControlBase
{
protected:

	virtual pfGUIControlMod	*IGetNewControl( void ) { return TRACKED_NEW pfGUIListBoxMod; }
	virtual bool			INeedsDynamicText( void ) { return true; }

public:
	plGUIListBoxComponent();
	void DeleteThis() { delete this; }

	// SetupProperties - Internal setup and write-only set properties on the MaxNode. No reading
	// of properties on the MaxNode, as it's still indeterminant.
	hsBool SetupProperties(plMaxNode *pNode, plErrorMsg *pErrMsg);

	hsBool PreConvert(plMaxNode *pNode, plErrorMsg *pErrMsg);
	hsBool Convert(plMaxNode *node, plErrorMsg *pErrMsg);

	enum
	{
		kRefUseScroll,
		kRefScrollCtrl,
		kRefSingleSelect,
		kRefXparentBgnd,
		kRefDragDropSource,
		kRefDisableKeys,
		kRefAllow2DElementGrid,
		kRefScrollLeftToRight,
		kRefScaleWithRes,
		kRefPassClicksThrough,
		kRefEnableTreeBehavior,
		kRefSkin,
		kRefHandsOffMultiSelect
	};
};

// When one of our parameters that is a ref changes, send out the component ref
// changed message.  Normally, messages from component refs are ignored since
// they pass along all the messages of the ref, which generates a lot of false
// converts.
class plGUIListBoxAccessor : public PBAccessor
{
public:
	void Set( PB2Value& v, ReferenceMaker* owner, ParamID id, int tabIndex, TimeValue t )
	{
		if( id == plGUIListBoxComponent::kRefScrollCtrl )
		{
			plGUIListBoxComponent *comp = (plGUIListBoxComponent *)owner;
			comp->NotifyDependents( FOREVER, PART_ALL, REFMSG_USER_COMP_REF_CHANGED );
		}
	}
};

Class_ID	sScrollingClassesToSelect[] = { GUI_UPDOWNPAIR_CLASSID, GUI_KNOBCTRL_CLASSID, plGUISingleCtrlDlgProc::kEndClassList };

static plGUIListBoxAccessor sGUIListBoxAccessor;
static plGUISingleCtrlDlgProc sGUIListBoxProc( plGUIListBoxComponent::kRefScrollCtrl, IDC_GUI_COMPSELBTN,
											"Select the control to use for scrolling this list box", sScrollingClassesToSelect );

static plGUISingleCtrlDlgProc sGUILBSkinSelectProc( plGUIListBoxComponent::kRefSkin, IDC_GUI_SKIN,
											"Select the skin to use for this list box", sSkinClassesToSelect,
											&sGUIListBoxProc );

//Max desc stuff necessary below.
CLASS_DESC(plGUIListBoxComponent, gGUIListBoxDesc, "GUI List Box",  "GUIListBox", COMP_TYPE_GUI, GUI_LISTBOX_CLASSID )

ParamBlockDesc2 gGUIListBoxBk
(	// KLUDGE: not the defined block ID, but kept for backwards compat.
 plComponent::kBlkComp, _T("GUIListBox"), 0, &gGUIListBoxDesc, P_AUTO_CONSTRUCT + P_AUTO_UI + P_MULTIMAP + P_INCLUDE_PARAMS, plComponent::kRefComp,

	2,
	plGUIControlBase::kRollMain, IDD_COMP_GUILISTBOX, IDS_COMP_GUILISTBOX, 0, 0, &sGUILBSkinSelectProc,
	plGUIControlBase::kRollProc, IDD_COMP_GUIPROCROLLOUT, IDS_COMP_GUIPROCROLLOUT, 0, 0, nil,	
	
	&sGUIControlProcParamTemplate,	

	plGUIListBoxComponent::kRefUseScroll,	_T( "enableScrolling" ), TYPE_BOOL, 0, 0,
		p_ui, plGUIControlBase::kRollMain, TYPE_SINGLECHEKBOX, IDC_GUI_SCROLLCTRL,
		p_default, FALSE,
		p_enable_ctrls, 1, plGUIListBoxComponent::kRefScrollCtrl,
		end,

	plGUIListBoxComponent::kRefScrollCtrl, _T("scrollControl"),	TYPE_INODE,		0, 0,
		p_prompt, IDS_COMP_GUI_SELECTSCROLL,
		p_accessor, &sGUIListBoxAccessor,
		end,

	plGUIListBoxComponent::kRefSingleSelect,	_T( "singleSelect" ), TYPE_BOOL, 0, 0,
		p_ui, plGUIControlBase::kRollMain, TYPE_SINGLECHEKBOX, IDC_GUI_SINGLESEL,
		p_default, FALSE,
		end,

	plGUIListBoxComponent::kRefXparentBgnd,	_T( "xparentBgnd" ), TYPE_BOOL, 0, 0,
		p_ui, plGUIControlBase::kRollMain, TYPE_SINGLECHEKBOX, IDC_GUI_XPARENT,
		p_default, FALSE,
		end,
	
	plGUIListBoxComponent::kRefDragDropSource,	_T( "dragDropCapable" ), TYPE_BOOL, 0, 0,
		p_ui, plGUIControlBase::kRollMain, TYPE_SINGLECHEKBOX, IDC_GUI_DRAGDROPSRC,
		p_default, FALSE,
		end,

	plGUIListBoxComponent::kRefDisableKeys,	_T( "disableKeys" ), TYPE_BOOL, 0, 0,
		p_ui, plGUIControlBase::kRollMain, TYPE_SINGLECHEKBOX, IDC_GUI_DISABLEKEYS,
		p_default, FALSE,
		end,
	
	plGUIListBoxComponent::kRefAllow2DElementGrid,	_T( "allow2DElementGrid" ), TYPE_BOOL, 0, 0,
		p_ui, plGUIControlBase::kRollMain, TYPE_SINGLECHEKBOX, IDC_GUI_ALLOWMULTIROW,
		p_default, FALSE,
		end,
	
	plGUIListBoxComponent::kRefScrollLeftToRight,	_T( "scrollLeftToRight" ), TYPE_BOOL, 0, 0,
		p_ui, plGUIControlBase::kRollMain, TYPE_SINGLECHEKBOX, IDC_GUI_SCROLLL2R,
		p_default, FALSE,
		end,

	plGUIListBoxComponent::kRefScaleWithRes,	_T( "scaleWithRes" ), TYPE_BOOL, 0, 0,
		p_ui, plGUIControlBase::kRollMain, TYPE_SINGLECHEKBOX, IDC_GUI_SCALERES,
		p_default, FALSE,
		end,
				
	plGUIListBoxComponent::kRefPassClicksThrough,	_T( "passClicksThru" ), TYPE_BOOL, 0, 0,
		p_ui, plGUIControlBase::kRollMain, TYPE_SINGLECHEKBOX, IDC_GUI_PASSTHRU,
		p_default, FALSE,
		end,
		
	plGUIListBoxComponent::kRefEnableTreeBehavior,	_T( "makeLikeATree" ), TYPE_BOOL, 0, 0,
		p_ui, plGUIControlBase::kRollMain, TYPE_SINGLECHEKBOX, IDC_GUI_ENABLETREE,
		p_default, FALSE,
		end,
				
	plGUIListBoxComponent::kRefSkin, _T("skin"),	TYPE_INODE,		0, 0,
		end,

	plGUIListBoxComponent::kRefHandsOffMultiSelect,	_T( "handsOffMultiSelect" ), TYPE_BOOL, 0, 0,
		p_ui, plGUIControlBase::kRollMain, TYPE_SINGLECHEKBOX, IDC_GUI_HANDSOFF,
		p_default, FALSE,
		end,
		
	end
);

plGUIListBoxComponent::plGUIListBoxComponent()
{
	fClassDesc = &gGUIListBoxDesc;
	fClassDesc->MakeAutoParamBlocks(this);
}

// SetupProperties - Internal setup and write-only set properties on the MaxNode. No reading
// of properties on the MaxNode, as it's still indeterminant.
hsBool plGUIListBoxComponent::SetupProperties(plMaxNode *node,  plErrorMsg *pErrMsg)
{
	return plGUIControlBase::SetupProperties( node, pErrMsg );
}

hsBool plGUIListBoxComponent::PreConvert(plMaxNode *node,  plErrorMsg *pErrMsg)
{
	return plGUIControlBase::PreConvert( node, pErrMsg );
}

hsBool plGUIListBoxComponent::Convert(plMaxNode *node, plErrorMsg *pErrMsg)
{
	if( !plGUIControlBase::Convert( node, pErrMsg ) )
		return false;

	pfGUIListBoxMod *ctrl = (pfGUIListBoxMod *)fControl;
	
	if( fCompPB->GetInt( kRefUseScroll ) )
	{
		// Get the scrolling control to use
		pfGUIValueCtrl *scroll = pfGUIValueCtrl::ConvertNoRef( GrabControlMod( fCompPB->GetINode( kRefScrollCtrl ) ) );
		if( scroll != nil )
		{
			hsgResMgr::ResMgr()->AddViaNotify( scroll->GetKey(), TRACKED_NEW plGenRefMsg( ctrl->GetKey(), 
										plRefMsg::kOnCreate, -1, pfGUIListBoxMod::kRefScrollCtrl ), plRefFlags::kActiveRef );
		}
	}

	if( fCompPB->GetInt( kRefSingleSelect ) )
		ctrl->SetSingleSelect( true );

	if( fCompPB->GetInt( kRefXparentBgnd ) )
		ctrl->SetFlag( pfGUIListBoxMod::kXparentBgnd );

	if( fCompPB->GetInt( kRefDragDropSource ) )
		ctrl->SetFlag( pfGUIListBoxMod::kDragAndDropCapable );

	if( fCompPB->GetInt( kRefDisableKeys ) )
		ctrl->SetFlag( pfGUIListBoxMod::kDisableKeyActions );

	if( fCompPB->GetInt( kRefAllow2DElementGrid ) )
		ctrl->SetFlag( pfGUIListBoxMod::kAllowMultipleElementsPerRow );

	if( fCompPB->GetInt( kRefScrollLeftToRight ) )
		ctrl->SetFlag( pfGUIListBoxMod::kScrollLeftToRight );
	
	if( fCompPB->GetInt( kRefScaleWithRes ) )
		ctrl->SetFlag( pfGUIControlMod::kScaleTextWithResolution );

	if( fCompPB->GetInt( kRefPassClicksThrough ) )
		ctrl->SetFlag( pfGUIListBoxMod::kAllowMousePassThrough );

	if( fCompPB->GetInt( kRefEnableTreeBehavior ) )
		ctrl->SetFlag( pfGUIListBoxMod::kGrowLeavesAndProcessOxygen );

	if( fCompPB->GetInt( kRefHandsOffMultiSelect ) )
		ctrl->SetFlag( pfGUIListBoxMod::kHandsOffMultiSelect );
	
	INode *sNode = fCompPB->GetINode( kRefSkin );
	if( sNode != nil )
	{
		plComponentBase *comp = ( (plMaxNode *)sNode )->ConvertToComponent();
		if( comp != nil )
		{
			Class_ID nodeID = comp->ClassID();
			hsAssert( nodeID == GUI_SKIN_CLASSID, "Bad node param in GUIMenu::Convert()" );

			plGUISkinComp *skin = (plGUISkinComp *)comp;
			hsgResMgr::ResMgr()->AddViaNotify( skin->GetConvertedSkin()->GetKey(), TRACKED_NEW plGenRefMsg( ctrl->GetKey(), plRefMsg::kOnCreate, -1, pfGUIControlMod::kRefSkin ), plRefFlags::kActiveRef );
		}
	}

	return true;
}

/////////////////////////////////////////////////////////////////////////////////////////////////
//// plGUITextBox Component /////////////////////////////////////////////////////////////////////
//
//	GUI element that displays a block of wrapped text.
//
/////////////////////////////////////////////////////////////////////////////////////////////////


// Class that accesses the paramblock below.
class plGUITextBoxComponent : public plGUIControlBase
{
protected:

	virtual pfGUIControlMod	*IGetNewControl( void ) { return TRACKED_NEW pfGUITextBoxMod; }
	virtual bool			INeedsDynamicText( void ) { return true; }

public:
	plGUITextBoxComponent();
	void DeleteThis() { delete this; }

	// SetupProperties - Internal setup and write-only set properties on the MaxNode. No reading
	// of properties on the MaxNode, as it's still indeterminant.
	hsBool SetupProperties(plMaxNode *pNode, plErrorMsg *pErrMsg);

	hsBool PreConvert(plMaxNode *pNode, plErrorMsg *pErrMsg);
	hsBool Convert(plMaxNode *node, plErrorMsg *pErrMsg);

	enum
	{
		kRefInitText,
		kRefFontSize,
		kRefXparentBgnd,
		kRefJustify,
		kRefScaleWithRes,
		kRefUseLocalization,
		kRefLocalizationPath
	};
};

class plGUITextBoxProc : public ParamMap2UserDlgProc
{
private:
	std::vector<std::string> fTranslations;
	int fCurLanguage;
	void ISetTranslation(int lang, std::string text)
	{
		while (lang >= fTranslations.size())
			fTranslations.push_back("");
		fTranslations[lang] = text;
	}
protected:

public:

	void DeleteThis() {}

	BOOL DlgProc( TimeValue t, IParamMap2 *pmap, HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam )
	{
		int i;
		switch( msg )
		{
			case WM_INITDIALOG:
				// make sure there is a string to get
				if ( pmap->GetParamBlock()->GetStr( plGUITextBoxComponent::kRefInitText ) )
				{
					fTranslations = plLocalization::StringToLocal(pmap->GetParamBlock()->GetStr( plGUITextBoxComponent::kRefInitText ) );
					SetDlgItemText( hWnd, IDC_GUI_INITTEXT, fTranslations[0].c_str() );
				}
				else
					// if there is no text, then there is nothing to translate
					SetDlgItemText( hWnd, IDC_GUI_INITTEXT, pmap->GetParamBlock()->GetStr( plGUITextBoxComponent::kRefInitText ) );
				SendMessage( GetDlgItem( hWnd, IDC_GUI_LANGUAGE ), CB_RESETCONTENT, 0, 0 );
				for (i=0; i<plLocalization::kNumLanguages; i++)
					SendMessage( GetDlgItem( hWnd, IDC_GUI_LANGUAGE ), CB_ADDSTRING, 0, (LPARAM)plLocalization::GetLanguageName((plLocalization::Language)i) );
				SendMessage( GetDlgItem( hWnd, IDC_GUI_LANGUAGE ), CB_SETCURSEL, 0, 0 );
				fCurLanguage = 0;

				SetDlgItemText( hWnd, IDC_GUI_LOCALIZATION_PATH, pmap->GetParamBlock()->GetStr( plGUITextBoxComponent::kRefLocalizationPath ) );

				if ( pmap->GetParamBlock()->GetInt( plGUITextBoxComponent::kRefUseLocalization ) != 0 )
				{
					// disable standard text, enable loc path
					EnableWindow( GetDlgItem( hWnd, IDC_GUI_INITTEXT ), false );
					EnableWindow( GetDlgItem( hWnd, IDC_GUI_LANGUAGE ), false );
					EnableWindow( GetDlgItem( hWnd, IDC_GUI_LOCALIZATION_PATH ), true );
					EnableWindow( GetDlgItem( hWnd, IDC_GUI_SELECT_LOC_PATH ), true );
					CheckDlgButton( hWnd, IDC_GUI_USE_LOCALIZATION, BST_CHECKED );
				}
				else
				{
					// enable standard text, disable loc path
					EnableWindow( GetDlgItem( hWnd, IDC_GUI_INITTEXT ), true );
					EnableWindow( GetDlgItem( hWnd, IDC_GUI_LANGUAGE ), true );
					EnableWindow( GetDlgItem( hWnd, IDC_GUI_LOCALIZATION_PATH ), false );
					EnableWindow( GetDlgItem( hWnd, IDC_GUI_SELECT_LOC_PATH ), false );
					CheckDlgButton( hWnd, IDC_GUI_USE_LOCALIZATION, BST_UNCHECKED );
				}
				return true;

			case WM_DESTROY:
				break;

			case WM_COMMAND:
				if( LOWORD( wParam ) == IDC_GUI_INITTEXT )
				{
					if( HIWORD( wParam ) == EN_CHANGE )
					{
						int strLen = SendDlgItemMessage( hWnd, IDC_GUI_INITTEXT, WM_GETTEXTLENGTH, 0, 0 );
						if( strLen > 0 )
						{
							char *str = TRACKED_NEW char[ strLen + 1 ];
							GetDlgItemText( hWnd, IDC_GUI_INITTEXT, str, strLen + 1 );
							str[ strLen ] = 0;
							ISetTranslation(fCurLanguage,str);
							delete [] str;

							std::string translation = plLocalization::LocalToString(fTranslations);
							str = TRACKED_NEW char[ translation.length() + 1 ];
							strcpy(str,translation.c_str());
							str[translation.length()] = 0;
				
							pmap->GetParamBlock()->SetValue( plGUITextBoxComponent::kRefInitText, 0, str );
							delete [] str;
						}
					}
					else if( HIWORD( wParam ) == EN_KILLFOCUS )
					{
						plMaxAccelerators::Enable();
					}
					else if( HIWORD( wParam ) == EN_SETFOCUS )
					{
						plMaxAccelerators::Disable();
					}
				}
				else if( LOWORD( wParam ) == IDC_GUI_LOCALIZATION_PATH )
				{
					if( HIWORD( wParam ) == EN_CHANGE )
					{
						int strLen = SendDlgItemMessage( hWnd, IDC_GUI_LOCALIZATION_PATH, WM_GETTEXTLENGTH, 0, 0 );
						if( strLen > 0 )
						{
							char *str = TRACKED_NEW char[ strLen + 1 ];
							GetDlgItemText( hWnd, IDC_GUI_LOCALIZATION_PATH, str, strLen + 1 );
							str[ strLen ] = 0;
							pmap->GetParamBlock()->SetValue( plGUITextBoxComponent::kRefLocalizationPath, 0, str );
							delete [] str;
						}
					}
					else if( HIWORD( wParam ) == EN_KILLFOCUS )
					{
						plMaxAccelerators::Enable();
					}
					else if( HIWORD( wParam ) == EN_SETFOCUS )
					{
						plMaxAccelerators::Disable();
					}
				}
				else if( LOWORD( wParam ) == IDC_GUI_LANGUAGE )
				{
					if( HIWORD( wParam ) == CBN_SELCHANGE )
					{
						int idx = SendDlgItemMessage( hWnd, IDC_GUI_LANGUAGE, CB_GETCURSEL, 0, 0 );
						if (idx >= fTranslations.size())
							SetDlgItemText( hWnd, IDC_GUI_INITTEXT, "" );
						else
							SetDlgItemText( hWnd, IDC_GUI_INITTEXT, fTranslations[idx].c_str() );
						fCurLanguage = idx;
					}
				}
				else if( LOWORD( wParam ) == IDC_GUI_SELECT_LOC_PATH )
				{
					char value[512];
					GetDlgItemText( hWnd, IDC_GUI_LOCALIZATION_PATH, value, 512 );
					plPickLocalizationDlg dlg( value );
					if( dlg.DoPick() )
					{
						pmap->GetParamBlock()->SetValue( plGUITextBoxComponent::kRefLocalizationPath, 0, (char*)dlg.GetValue() );
						SetDlgItemText( hWnd, IDC_GUI_LOCALIZATION_PATH, (char*)dlg.GetValue() );
					}
				}
				else if( LOWORD( wParam ) == IDC_GUI_USE_LOCALIZATION )
				{
					// enable/disable the appropriate values
					bool useLoc = ( IsDlgButtonChecked( hWnd, IDC_GUI_USE_LOCALIZATION ) == BST_CHECKED );
					pmap->GetParamBlock()->SetValue( plGUITextBoxComponent::kRefUseLocalization, 0, useLoc ? 1 : 0 );

					if ( useLoc )
					{
						// disable standard text, enable loc path
						EnableWindow( GetDlgItem( hWnd, IDC_GUI_INITTEXT ), false );
						EnableWindow( GetDlgItem( hWnd, IDC_GUI_LANGUAGE ), false );
						EnableWindow( GetDlgItem( hWnd, IDC_GUI_LOCALIZATION_PATH ), true );
						EnableWindow( GetDlgItem( hWnd, IDC_GUI_SELECT_LOC_PATH ), true );
					}
					else
					{
						// enable standard text, disable loc path
						EnableWindow( GetDlgItem( hWnd, IDC_GUI_INITTEXT ), true );
						EnableWindow( GetDlgItem( hWnd, IDC_GUI_LANGUAGE ), true );
						EnableWindow( GetDlgItem( hWnd, IDC_GUI_LOCALIZATION_PATH ), false );
						EnableWindow( GetDlgItem( hWnd, IDC_GUI_SELECT_LOC_PATH ), false );
					}
				}
				break;
		}
		return false;
	}
};
static plGUITextBoxProc gGUITextBoxProc;

//Max desc stuff necessary below.
CLASS_DESC(plGUITextBoxComponent, gGUITextBoxDesc, "GUI Text Box",  "GUITextBox", COMP_TYPE_GUI, GUI_TEXTBOX_CLASSID )

ParamBlockDesc2 gGUITextBoxBk
(	// KLUDGE: not the defined block ID, but kept for backwards compat.
 plComponent::kBlkComp, _T("GUITextBox"), 0, &gGUITextBoxDesc, P_AUTO_CONSTRUCT + P_AUTO_UI + P_MULTIMAP + P_INCLUDE_PARAMS, plComponent::kRefComp,

	2,
	plGUIControlBase::kRollMain, IDD_COMP_GUITEXTBOX, IDS_COMP_GUITEXTBOX, 0, 0, &gGUITextBoxProc,
	plGUIControlBase::kRollProc, IDD_COMP_GUIPROCROLLOUT, IDS_COMP_GUIPROCROLLOUT, 0, 0, nil,	
	
	&sGUIControlProcParamTemplate,	

		plGUITextBoxComponent::kRefInitText,	_T("InitText"),		TYPE_STRING, 		0, 0,
//			p_ui,	plGUIControlBase::kRollMain, TYPE_EDITBOX, IDC_GUI_INITTEXT,
			end,

		plGUITextBoxComponent::kRefXparentBgnd,	_T( "xparentBgnd" ), TYPE_BOOL, 0, 0,
			p_ui, plGUIControlBase::kRollMain, TYPE_SINGLECHEKBOX, IDC_GUI_XPARENT,
			p_default, FALSE,
			end,
	
		plGUITextBoxComponent::kRefJustify,	_T("justify"), TYPE_INT, 		0, 0,
			p_ui,	plGUIControlBase::kRollMain, TYPE_RADIO, 3, IDC_JUSTIFYRADIO, IDC_JUSTRADIO2, IDC_JUSTRADIO3, 
			p_default, 0,
			end,

		plGUITextBoxComponent::kRefScaleWithRes,	_T( "scaleWithRes" ), TYPE_BOOL, 0, 0,
			p_ui, plGUIControlBase::kRollMain, TYPE_SINGLECHEKBOX, IDC_GUI_SCALERES,
			p_default, FALSE,
			end,

		plGUITextBoxComponent::kRefUseLocalization,	_T( "useLocalization" ), TYPE_BOOL, 0, 0,
			p_default, FALSE,
			end,

		plGUITextBoxComponent::kRefLocalizationPath,_T( "localizationPath" ),TYPE_STRING, 0, 0,
			end,
				
	end
);

plGUITextBoxComponent::plGUITextBoxComponent()
{
	fClassDesc = &gGUITextBoxDesc;
	fClassDesc->MakeAutoParamBlocks(this);
}

// SetupProperties - Internal setup and write-only set properties on the MaxNode. No reading
// of properties on the MaxNode, as it's still indeterminant.
hsBool plGUITextBoxComponent::SetupProperties(plMaxNode *node,  plErrorMsg *pErrMsg)
{
	return plGUIControlBase::SetupProperties( node, pErrMsg );
}

hsBool plGUITextBoxComponent::PreConvert(plMaxNode *node,  plErrorMsg *pErrMsg)
{
	return plGUIControlBase::PreConvert( node, pErrMsg );
}

hsBool plGUITextBoxComponent::Convert(plMaxNode *node, plErrorMsg *pErrMsg)
{
	if( !plGUIControlBase::Convert( node, pErrMsg ) )
		return false;

	pfGUITextBoxMod *ctrl = (pfGUITextBoxMod *)fControl;

	ctrl->SetText( fCompPB->GetStr( kRefInitText ) );

	if( fCompPB->GetInt( kRefXparentBgnd ) )
		ctrl->SetFlag( pfGUITextBoxMod::kXparentBgnd );

	int just = fCompPB->GetInt( kRefJustify );
	if( just == 1 )
		ctrl->SetFlag( pfGUITextBoxMod::kCenterJustify );
	else if( just == 2 )
		ctrl->SetFlag( pfGUITextBoxMod::kRightJustify );

	if( fCompPB->GetInt( kRefScaleWithRes ) )
		ctrl->SetFlag( pfGUIControlMod::kScaleTextWithResolution );

	ctrl->SetUseLocalizationPath( fCompPB->GetInt( kRefUseLocalization ) != 0 );
	ctrl->SetLocalizationPath( fCompPB->GetStr( kRefLocalizationPath ) );

	return true;
}

/////////////////////////////////////////////////////////////////////////////////////////////////
//// plGUIEditBox Component /////////////////////////////////////////////////////////////////////
//
//	GUI element that can be dragged anywhere in the 2D viewing plane.
//
/////////////////////////////////////////////////////////////////////////////////////////////////


// Class that accesses the paramblock below.
class plGUIEditBoxComponent : public plGUIControlBase
{
protected:

	virtual pfGUIControlMod	*IGetNewControl( void ) { return TRACKED_NEW pfGUIEditBoxMod; }
	virtual bool			INeedsDynamicText( void ) { return true; }

public:
	plGUIEditBoxComponent();
	void DeleteThis() { delete this; }

	// SetupProperties - Internal setup and write-only set properties on the MaxNode. No reading
	// of properties on the MaxNode, as it's still indeterminant.
	hsBool SetupProperties(plMaxNode *pNode, plErrorMsg *pErrMsg);

	hsBool PreConvert(plMaxNode *pNode, plErrorMsg *pErrMsg);
	hsBool Convert(plMaxNode *node, plErrorMsg *pErrMsg);

	enum
	{
		kRefXparentBgnd,
		kRefScaleWithRes
	};
};

//Max desc stuff necessary below.
CLASS_DESC(plGUIEditBoxComponent, gGUIEditBoxDesc, "GUI Edit Box",  "GUIEditBox", COMP_TYPE_GUI, GUI_EDITBOX_CLASSID )

ParamBlockDesc2 gGUIEditBoxBk
(	// KLUDGE: not the defined block ID, but kept for backwards compat.
 plComponent::kBlkComp, _T("GUIEditBox"), 0, &gGUIEditBoxDesc, P_AUTO_CONSTRUCT + P_AUTO_UI + P_MULTIMAP + P_INCLUDE_PARAMS, plComponent::kRefComp,

	2,
	plGUIControlBase::kRollMain, IDD_COMP_GUIEDITBOX, IDS_COMP_GUIEDITBOX, 0, 0, NULL,
	plGUIControlBase::kRollProc, IDD_COMP_GUIPROCROLLOUT, IDS_COMP_GUIPROCROLLOUT, 0, 0, nil,	
	
	&sGUIControlProcParamTemplate,	

	plGUIEditBoxComponent::kRefXparentBgnd,	_T( "xparentBgnd" ), TYPE_BOOL, 0, 0,
		p_ui, plGUIControlBase::kRollMain, TYPE_SINGLECHEKBOX, IDC_GUI_XPARENT,
		p_default, FALSE,
		end,
	
	plGUIEditBoxComponent::kRefScaleWithRes,	_T( "scaleWithRes" ), TYPE_BOOL, 0, 0,
		p_ui, plGUIControlBase::kRollMain, TYPE_SINGLECHEKBOX, IDC_GUI_SCALERES,
		p_default, FALSE,
		end,
			
	end
);

plGUIEditBoxComponent::plGUIEditBoxComponent()
{
	fClassDesc = &gGUIEditBoxDesc;
	fClassDesc->MakeAutoParamBlocks(this);
}

// SetupProperties - Internal setup and write-only set properties on the MaxNode. No reading
// of properties on the MaxNode, as it's still indeterminant.
hsBool plGUIEditBoxComponent::SetupProperties(plMaxNode *node,  plErrorMsg *pErrMsg)
{
	return plGUIControlBase::SetupProperties( node, pErrMsg );
}

hsBool plGUIEditBoxComponent::PreConvert(plMaxNode *node,  plErrorMsg *pErrMsg)
{
	return plGUIControlBase::PreConvert( node, pErrMsg );
}

hsBool plGUIEditBoxComponent::Convert(plMaxNode *node, plErrorMsg *pErrMsg)
{
	if( !plGUIControlBase::Convert( node, pErrMsg ) )
		return false;

	pfGUIEditBoxMod *ctrl = (pfGUIEditBoxMod *)fControl;
	
	if( fCompPB->GetInt( kRefXparentBgnd ) )
		ctrl->SetFlag( pfGUIEditBoxMod::kXparentBgnd );

	if( fCompPB->GetInt( kRefScaleWithRes ) )
		ctrl->SetFlag( pfGUIControlMod::kScaleTextWithResolution );

	return true;
}


/////////////////////////////////////////////////////////////////////////////////////////////////
//// plGUIUpDownPair Component //////////////////////////////////////////////////////////////////
//
//	GUI grouping element that uses two buttons to alter a value up and down
//
/////////////////////////////////////////////////////////////////////////////////////////////////


// Class that accesses the paramblock below.
class plGUIUpDownPairComponent : public plGUIControlBase
{
protected:

	virtual pfGUIControlMod	*IGetNewControl( void ) { return TRACKED_NEW pfGUIUpDownPairMod; }

public:
	plGUIUpDownPairComponent();
	void DeleteThis() { delete this; }

	// SetupProperties - Internal setup and write-only set properties on the MaxNode. No reading
	// of properties on the MaxNode, as it's still indeterminant.
	hsBool SetupProperties(plMaxNode *pNode, plErrorMsg *pErrMsg);

	hsBool PreConvert(plMaxNode *pNode, plErrorMsg *pErrMsg);
	hsBool Convert(plMaxNode *node, plErrorMsg *pErrMsg);

	enum
	{
		kRefMinValue,
		kRefMaxValue,
		kRefStep,
		kRefUpControl,
		kRefDownControl
	};
};

//// Dialog proc ////////////////////////////////////////////////////////////////////////////////

class plGUIUDPairDlgProc : public ParamMap2UserDlgProc
{
protected:
	ParamID			fUpNodeID, fDownNodeID;
	int				fUpDlgItem, fDownDlgItem;
	TCHAR			fTitle[ 128 ];

public:
	plGUIUDPairDlgProc( ParamID upNodeID, int upDlgItem, ParamID downNodeID, int downDlgItem, TCHAR *title )
	{
		fUpNodeID = upNodeID;
		fDownNodeID = downNodeID;
		fUpDlgItem = upDlgItem;
		fDownDlgItem = downDlgItem;
		strcpy( fTitle, title );
	}

	BOOL DlgProc( TimeValue t, IParamMap2 *map, HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam )
	{
		switch ( msg )
		{
			case WM_INITDIALOG:
				{
					IParamBlock2 *pb = map->GetParamBlock();

					INode *node = pb->GetINode( fUpNodeID );
					TSTR newName( node ? node->GetName() : "Pick" );
					::SetWindowText( ::GetDlgItem( hWnd, fUpDlgItem ), newName );

					node = pb->GetINode( fDownNodeID );
					TSTR newName2( node ? node->GetName() : "Pick" );
					::SetWindowText( ::GetDlgItem( hWnd, fDownDlgItem ), newName2 );
				}
				return true;

			case WM_COMMAND:
				if( ( HIWORD( wParam ) == BN_CLICKED ) )
				{
					if( LOWORD( wParam ) == fUpDlgItem )
					{
						IParamBlock2 *pb = map->GetParamBlock();
						plGUICtrlHitCallback hitCB( (INode *)pb->GetOwner(), pb, fUpNodeID, fTitle, true, GUI_BUTTON_CLASSID );
						GetCOREInterface()->DoHitByNameDialog( &hitCB );

						INode* node = pb->GetINode( fUpNodeID );
						TSTR newName( node ? node->GetName() : "Pick" );
						::SetWindowText( ::GetDlgItem(hWnd, fUpDlgItem ), newName );
						map->Invalidate( fUpNodeID );
						::InvalidateRect( hWnd, NULL, TRUE );
					}
					else if( LOWORD( wParam ) == fDownDlgItem )
					{
						IParamBlock2 *pb = map->GetParamBlock();
						plGUICtrlHitCallback hitCB( (INode *)pb->GetOwner(), pb, fDownNodeID, fTitle, true, GUI_BUTTON_CLASSID );
						GetCOREInterface()->DoHitByNameDialog( &hitCB );

						INode* node = pb->GetINode( fDownNodeID );
						TSTR newName( node ? node->GetName() : "Pick" );
						::SetWindowText( ::GetDlgItem(hWnd, fDownDlgItem ), newName );
						map->Invalidate( fDownDlgItem );
						::InvalidateRect( hWnd, NULL, TRUE );
					}
				}
				return true;
		}

		return false;
	}

	void DeleteThis() {}
};

// When one of our parameters that is a ref changes, send out the component ref
// changed message.  Normally, messages from component refs are ignored since
// they pass along all the messages of the ref, which generates a lot of false
// converts.
class plGUIUDAccessor : public PBAccessor
{
public:
	void Set( PB2Value& v, ReferenceMaker* owner, ParamID id, int tabIndex, TimeValue t )
	{
		if( id == plGUIUpDownPairComponent::kRefUpControl
			|| id == plGUIUpDownPairComponent::kRefDownControl )
		{
			plGUIUpDownPairComponent *comp = (plGUIUpDownPairComponent *)owner;
			comp->NotifyDependents( FOREVER, PART_ALL, REFMSG_USER_COMP_REF_CHANGED );
		}
	}
};

static plGUIUDAccessor sGUIUDAccessor;
static plGUIUDPairDlgProc sGUIUDPairDlgProc( plGUIUpDownPairComponent::kRefUpControl, IDC_GUI_COMPSELBTN,
											plGUIUpDownPairComponent::kRefDownControl, IDC_GUI_COMPSELBTN2,
											"Select the control to use in this pair" );

//Max desc stuff necessary below.
CLASS_DESC(plGUIUpDownPairComponent, gGUIUDPairDesc, "GUI Up/Down Pair",  "GUIUDPair", COMP_TYPE_GUI, GUI_UPDOWNPAIR_CLASSID )

ParamBlockDesc2 gGUIUDPairBk
(	// KLUDGE: not the defined block ID, but kept for backwards compat.
 plComponent::kBlkComp, _T("GUIUDPair"), 0, &gGUIUDPairDesc, P_AUTO_CONSTRUCT + P_AUTO_UI + P_MULTIMAP + P_INCLUDE_PARAMS, plComponent::kRefComp,

	2,
	plGUIControlBase::kRollMain, IDD_COMP_GUIUDSCROLL, IDS_COMP_GUIUDSCROLL, 0, 0, &sGUIUDPairDlgProc,
	plGUIControlBase::kRollProc, IDD_COMP_GUIPROCROLLOUT, IDS_COMP_GUIPROCROLLOUT, 0, 0, nil,	
	
	&sGUIControlProcParamTemplate,	

	plGUIUpDownPairComponent::kRefUpControl, _T("upControl"),	TYPE_INODE,		0, 0,
		p_prompt, IDS_COMP_GUI_SELECTUDCTRL,
		p_accessor, &sGUIUDAccessor,
		end,

	plGUIUpDownPairComponent::kRefDownControl, _T("downControl"),	TYPE_INODE,		0, 0,
		p_prompt, IDS_COMP_GUI_SELECTUDCTRL,
		p_accessor, &sGUIUDAccessor,
		end,

	plGUIUpDownPairComponent::kRefMinValue, _T("minValue"),	TYPE_FLOAT,	0, 0,	
		p_default, 0.0f,
		p_ui, plGUIControlBase::kRollMain, TYPE_SPINNER,	EDITTYPE_POS_FLOAT, IDC_GUI_LOWER, IDC_GUI_LOWER_SPIN, SPIN_AUTOSCALE,
		end,

	plGUIUpDownPairComponent::kRefMaxValue, _T("maxValue"),	TYPE_FLOAT,	0, 0,	
		p_default, 10.0f,
		p_ui, plGUIControlBase::kRollMain, TYPE_SPINNER,	EDITTYPE_POS_FLOAT, IDC_GUI_UPPER, IDC_GUI_UPPER_SPIN, SPIN_AUTOSCALE,
		end,

	plGUIUpDownPairComponent::kRefStep, _T("step"),	TYPE_FLOAT,	0, 0,	
		p_default, 1.0f,
		p_ui, plGUIControlBase::kRollMain, TYPE_SPINNER,	EDITTYPE_POS_FLOAT, IDC_GUI_STEP, IDC_GUI_STEP_SPIN, SPIN_AUTOSCALE,
		end,

	end
);

plGUIUpDownPairComponent::plGUIUpDownPairComponent()
{
	fClassDesc = &gGUIUDPairDesc;
	fClassDesc->MakeAutoParamBlocks(this);
}

// SetupProperties - Internal setup and write-only set properties on the MaxNode. No reading
// of properties on the MaxNode, as it's still indeterminant.
hsBool plGUIUpDownPairComponent::SetupProperties(plMaxNode *node,  plErrorMsg *pErrMsg)
{
	return plGUIControlBase::SetupProperties( node, pErrMsg );
}

hsBool plGUIUpDownPairComponent::PreConvert(plMaxNode *node,  plErrorMsg *pErrMsg)
{
	return plGUIControlBase::PreConvert( node, pErrMsg );
}

hsBool plGUIUpDownPairComponent::Convert(plMaxNode *node, plErrorMsg *pErrMsg)
{
	if( !plGUIControlBase::Convert( node, pErrMsg ) )
		return false;

	pfGUIUpDownPairMod *ctrl = (pfGUIUpDownPairMod *)fControl;

	// Get the child controls
	pfGUIButtonMod *up = pfGUIButtonMod::ConvertNoRef( GrabControlMod( fCompPB->GetINode( kRefUpControl ) ) );
	pfGUIButtonMod *down = pfGUIButtonMod::ConvertNoRef( GrabControlMod( fCompPB->GetINode( kRefDownControl ) ) );

	ctrl->SetControls( up, down );

	ctrl->SetRange( fCompPB->GetFloat( kRefMinValue ), fCompPB->GetFloat( kRefMaxValue ) );
	ctrl->SetStep( fCompPB->GetFloat( kRefStep ) );

	return true;
}


/////////////////////////////////////////////////////////////////////////////////////////////////
//// plGUIDragBar Component ///////////////////////////////////////////////////////////////////
//
//	GUI element that can be dragged anywhere in the 2D viewing plane.
//
/////////////////////////////////////////////////////////////////////////////////////////////////


// Class that accesses the paramblock below.
class plGUIDragBarComponent : public plGUIControlBase
{
protected:

	virtual pfGUIControlMod	*IGetNewControl( void ) { return TRACKED_NEW pfGUIDragBarCtrl; }
	virtual bool			ICanHaveProxy( void ) { return true; }

public:
	plGUIDragBarComponent();
	void DeleteThis() { delete this; }

	// SetupProperties - Internal setup and write-only set properties on the MaxNode. No reading
	// of properties on the MaxNode, as it's still indeterminant.
	hsBool SetupProperties(plMaxNode *pNode, plErrorMsg *pErrMsg);

	hsBool PreConvert(plMaxNode *pNode, plErrorMsg *pErrMsg);
	hsBool Convert(plMaxNode *node, plErrorMsg *pErrMsg);

	enum
	{
	};
};

//Max desc stuff necessary below.
CLASS_DESC(plGUIDragBarComponent, gGUIDragBarDesc, "GUI Dialog Drag Bar",  "GUIDragBar", COMP_TYPE_GUI, GUI_DRAGBAR_CLASSID )

ParamBlockDesc2 gGUIDragBarBk
(	// KLUDGE: not the defined block ID, but kept for backwards compat.
 plComponent::kBlkComp, _T("GUIDragBar"), 0, &gGUIDragBarDesc, P_AUTO_CONSTRUCT + P_AUTO_UI + P_MULTIMAP + P_INCLUDE_PARAMS, plComponent::kRefComp,

	3,
	plGUIControlBase::kRollMain, IDD_COMP_GUIDRAGBAR, IDS_COMP_GUIDRAGBAR, 0, 0, NULL,
	plGUIControlBase::kRollProc, IDD_COMP_GUIPROCROLLOUT, IDS_COMP_GUIPROCROLLOUT, 0, 0, nil,	
	
	sGUIProxyParamHeader,
	
	&sGUIControlProcParamTemplate,	

	sGUIProxyParamTemplate,

	end
);

plGUIDragBarComponent::plGUIDragBarComponent()
{
	fClassDesc = &gGUIDragBarDesc;
	fClassDesc->MakeAutoParamBlocks(this);
}

// SetupProperties - Internal setup and write-only set properties on the MaxNode. No reading
// of properties on the MaxNode, as it's still indeterminant.
hsBool plGUIDragBarComponent::SetupProperties(plMaxNode *node,  plErrorMsg *pErrMsg)
{
	node->SetForceLocal( true );
	return plGUIControlBase::SetupProperties( node, pErrMsg );
}

hsBool plGUIDragBarComponent::PreConvert(plMaxNode *node,  plErrorMsg *pErrMsg)
{
	return plGUIControlBase::PreConvert( node, pErrMsg );
}

hsBool plGUIDragBarComponent::Convert(plMaxNode *node, plErrorMsg *pErrMsg)
{
	if( !plGUIControlBase::Convert( node, pErrMsg ) )
		return false;

	pfGUIDragBarCtrl *ctrl = (pfGUIDragBarCtrl *)fControl;
	
	return true;
}


/////////////////////////////////////////////////////////////////////////////////////////////////
//// plGUIRadioGroup Component //////////////////////////////////////////////////////////////////
//
//	GUI grouping element that ensures that only one of a group of check boxes is checked at any
//	one time, and takes on the value of whichever one is currently checked, or -1 if none.
//
/////////////////////////////////////////////////////////////////////////////////////////////////


// Class that accesses the paramblock below.
class plGUIRadioGroupAccessor;
class plGUIRadioGroupComponent : public plGUIControlBase
{
	friend class plGUIRadioGroupAccessor;

protected:

	virtual pfGUIControlMod	*IGetNewControl( void ) { return TRACKED_NEW pfGUIRadioGroupCtrl; }

public:
	plGUIRadioGroupComponent();
	void DeleteThis() { delete this; }

	// SetupProperties - Internal setup and write-only set properties on the MaxNode. No reading
	// of properties on the MaxNode, as it's still indeterminant.
	hsBool SetupProperties(plMaxNode *pNode, plErrorMsg *pErrMsg);

	hsBool PreConvert(plMaxNode *pNode, plErrorMsg *pErrMsg);
	hsBool Convert(plMaxNode *node, plErrorMsg *pErrMsg);

	enum
	{
		kRefCheckBoxes,
		kRefDefaultSel,
		kRefAllowNoSel
	};
};

//// Dialog proc ////////////////////////////////////////////////////////////////////////////////

class plGUIRadioGroupProc : public ParamMap2UserDlgProc
{
protected:

public:
	plGUIRadioGroupProc()
	{
	}

	void	SetSpinnerRange( IParamMap2 *pMap )
	{
		if( pMap == nil )
			return;

		HWND	hWnd = pMap->GetHWnd();
		if( hWnd == nil )
			return;

		ISpinnerControl *spin = GetISpinner( GetDlgItem( hWnd, IDC_GUI_DEFSEL_SPIN ) );

		int minValue = pMap->GetParamBlock()->GetInt( plGUIRadioGroupComponent::kRefAllowNoSel ) ? -1 : 0;
		int maxValue = pMap->GetParamBlock()->Count( plGUIRadioGroupComponent::kRefCheckBoxes );

		spin->SetLimits( minValue, maxValue );

		ReleaseISpinner( spin );
	}

	BOOL DlgProc( TimeValue t, IParamMap2 *map, HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam )
	{
		switch ( msg )
		{
			case WM_INITDIALOG:
				{
					SetSpinnerRange( map );
				}
				return true;

			case WM_COMMAND:
				if( ( HIWORD( wParam ) == BN_CLICKED ) )
				{
					if( LOWORD( wParam ) == IDC_GUI_ADDCHECK )
					{
						IParamBlock2 *pb = map->GetParamBlock();

						plGUICtrlHitCallback hitCB( (INode *)pb->GetOwner(), pb, plGUIRadioGroupComponent::kRefCheckBoxes, 
									"Select a check box to add to this radio group", true, GUI_CHECKBOX_CLASSID, false );

						GetCOREInterface()->DoHitByNameDialog( &hitCB );

						map->Invalidate( plGUIRadioGroupComponent::kRefCheckBoxes );
					}
				}
				return true;
		}

		return false;
	}

	void DeleteThis() {}
};
static plGUIRadioGroupProc sGUIRadioGroupProc;

//Max desc stuff necessary below.
CLASS_DESC(plGUIRadioGroupComponent, gGUIRadioGroupDesc, "GUI Radio Group",  "GUIRadioGroup", COMP_TYPE_GUI, GUI_RADIOGROUP_CLASSID )

// When one of our parameters that is a ref changes, send out the component ref
// changed message.  Normally, messages from component refs are ignored since
// they pass along all the messages of the ref, which generates a lot of false
// converts.
class plGUIRadioGroupAccessor : public PBAccessor
{
public:
	void Set( PB2Value& v, ReferenceMaker* owner, ParamID id, int tabIndex, TimeValue t )
	{
		plGUIRadioGroupComponent *comp = (plGUIRadioGroupComponent *)owner;
		IParamBlock2	*pBlock = comp->fCompPB;

		if( id == plGUIRadioGroupComponent::kRefCheckBoxes )
		{
			comp->NotifyDependents( FOREVER, PART_ALL, REFMSG_USER_COMP_REF_CHANGED );
			sGUIRadioGroupProc.SetSpinnerRange( pBlock->GetMap( plGUIControlBase::kRollMain ) );
		}
		else if( id == plGUIRadioGroupComponent::kRefAllowNoSel )
			sGUIRadioGroupProc.SetSpinnerRange( pBlock->GetMap( plGUIControlBase::kRollMain ) );
	}

	void	TabChanged( tab_changes changeCode, Tab<PB2Value> *tab, ReferenceMaker *owner, ParamID id, int tabIndex, int count )
	{
		plGUIRadioGroupComponent *comp = (plGUIRadioGroupComponent *)owner;
		IParamBlock2	*pBlock = comp->fCompPB;

		if( id == plGUIRadioGroupComponent::kRefCheckBoxes )
		{
			comp->NotifyDependents( FOREVER, PART_ALL, REFMSG_USER_COMP_REF_CHANGED );
			sGUIRadioGroupProc.SetSpinnerRange( pBlock->GetMap( plGUIControlBase::kRollMain ) );
		}
	}

};

static plGUIRadioGroupAccessor sGUIRadioGroupAccessor;

ParamBlockDesc2 gGUIRadioGroupBk
(	// KLUDGE: not the defined block ID, but kept for backwards compat.
 plComponent::kBlkComp, _T("GUIRadioGroup"), 0, &gGUIRadioGroupDesc, P_AUTO_CONSTRUCT + P_AUTO_UI + P_MULTIMAP + P_INCLUDE_PARAMS, plComponent::kRefComp,

	2,
	plGUIControlBase::kRollMain, IDD_COMP_GUIRADIOGROUP, IDS_COMP_GUIRADIOGROUP, 0, 0, &sGUIRadioGroupProc,
	plGUIControlBase::kRollProc, IDD_COMP_GUIPROCROLLOUT, IDS_COMP_GUIPROCROLLOUT, 0, 0, nil,	
	
	&sGUIControlProcParamTemplate,	

	plGUIRadioGroupComponent::kRefCheckBoxes,	_T("checkBoxes"),	TYPE_INODE_TAB, 0,		0, 0,
		p_ui,			plGUIControlBase::kRollMain, TYPE_NODELISTBOX, IDC_GUI_CHECKLIST, 0, 0, IDC_GUI_DELCHECK,
		p_accessor,		&sGUIRadioGroupAccessor,
		end,

	plGUIRadioGroupComponent::kRefDefaultSel, _T("defaultSelection"),	TYPE_INT,	0, 0,	
		p_default, 0,
		p_range, 0, 0,
		p_ui, plGUIControlBase::kRollMain, TYPE_SPINNER,	EDITTYPE_INT, IDC_GUI_DEFSEL, IDC_GUI_DEFSEL_SPIN, SPIN_AUTOSCALE,
		end,

	plGUIRadioGroupComponent::kRefAllowNoSel,	_T( "allowNoSel" ), TYPE_BOOL, 0, 0,
		p_ui, plGUIControlBase::kRollMain, TYPE_SINGLECHEKBOX, IDC_GUI_ALLOWNONE,
		p_default, FALSE,
		p_accessor, &sGUIRadioGroupAccessor,
		end,

	end
);

plGUIRadioGroupComponent::plGUIRadioGroupComponent()
{
	fClassDesc = &gGUIRadioGroupDesc;
	fClassDesc->MakeAutoParamBlocks(this);
}

// SetupProperties - Internal setup and write-only set properties on the MaxNode. No reading
// of properties on the MaxNode, as it's still indeterminant.
hsBool plGUIRadioGroupComponent::SetupProperties(plMaxNode *node,  plErrorMsg *pErrMsg)
{
	return plGUIControlBase::SetupProperties( node, pErrMsg );
}


hsBool plGUIRadioGroupComponent::PreConvert(plMaxNode *node,  plErrorMsg *pErrMsg)
{
	return plGUIControlBase::PreConvert( node, pErrMsg );
}

hsBool plGUIRadioGroupComponent::Convert(plMaxNode *node, plErrorMsg *pErrMsg)
{
	if( !plGUIControlBase::Convert( node, pErrMsg ) )
		return false;

	pfGUIRadioGroupCtrl *ctrl = (pfGUIRadioGroupCtrl *)fControl;

	int	i;
	ctrl->ClearControlList();
	for( i = 0; i < fCompPB->Count( kRefCheckBoxes ); i++ )
	{
		pfGUICheckBoxCtrl *cb = pfGUICheckBoxCtrl::ConvertNoRef( GrabControlMod( fCompPB->GetINode( kRefCheckBoxes, 0, i ) ) );
		if( cb != nil )
			ctrl->AddControl( cb );
	}

	if( fCompPB->GetInt( kRefAllowNoSel ) )
		ctrl->SetFlag( pfGUIRadioGroupCtrl::kAllowNoSelection );

	ctrl->SetDefaultValue( fCompPB->GetInt( kRefDefaultSel ) );

	return true;
}


/////////////////////////////////////////////////////////////////////////////////////////////////
//// plGUIDynDisplay Component //////////////////////////////////////////////////////////////////
//
//	GUI element that can be dragged anywhere in the 2D viewing plane.
//
/////////////////////////////////////////////////////////////////////////////////////////////////


// Class that accesses the paramblock below.
class plGUIDynDisplayComponent : public plGUIControlBase
{
protected:

	virtual pfGUIControlMod	*IGetNewControl( void ) { return TRACKED_NEW pfGUIDynDisplayCtrl; }
	virtual bool			IHasProcRollout( void ) { return false; }

public:
	plGUIDynDisplayComponent();
	void DeleteThis() { delete this; }

	// SetupProperties - Internal setup and write-only set properties on the MaxNode. No reading
	// of properties on the MaxNode, as it's still indeterminant.
	hsBool SetupProperties(plMaxNode *pNode, plErrorMsg *pErrMsg);

	hsBool PreConvert(plMaxNode *pNode, plErrorMsg *pErrMsg);
	hsBool Convert(plMaxNode *node, plErrorMsg *pErrMsg);

	enum
	{
		kRefDynLayer
	};
};

//// Dialog proc ////////////////////////////////////////////////////////////////////////////////

class plGUIDynDisplayProc : public ParamMap2UserDlgProc
{
protected:

public:
	plGUIDynDisplayProc()
	{
	}

	BOOL DlgProc( TimeValue t, IParamMap2 *map, HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam )
	{
		switch ( msg )
		{
			case WM_INITDIALOG:
				{
					IParamBlock2 *pb = map->GetParamBlock();

					Texmap *tmap = pb->GetTexmap( plGUIDynDisplayComponent::kRefDynLayer );
					if( tmap != nil )
						SetDlgItemText( hWnd, IDC_GUI_PICKMAT, (const char *)tmap->GetName() );
					else
						SetDlgItemText( hWnd, IDC_GUI_PICKMAT, "Pick" );
				}
				return true;

			case WM_COMMAND:
				if( ( HIWORD( wParam ) == BN_CLICKED ) )
				{
					if( LOWORD( wParam ) == IDC_GUI_PICKMAT )
					{
						IParamBlock2 *pb = map->GetParamBlock();

						if( plPickMaterialMap::PickTexmap( pb, plGUIDynDisplayComponent::kRefDynLayer ) )
						{
							Texmap *tmap = pb->GetTexmap( plGUIDynDisplayComponent::kRefDynLayer );
							if( tmap != nil )
								SetDlgItemText( hWnd, IDC_GUI_PICKMAT, (const char *)tmap->GetName() );
							else
								SetDlgItemText( hWnd, IDC_GUI_PICKMAT, "Pick" );

							map->Invalidate( plGUIDynDisplayComponent::kRefDynLayer );
						}
					}
				}
				return true;
		}

		return false;
	}

	void DeleteThis() {}
};
static plGUIDynDisplayProc	sGUIDynDisplayProc;

//Max desc stuff necessary below.
CLASS_DESC(plGUIDynDisplayComponent, gGUIDynDisplayDesc, "GUI Dynamic Display",  "GUIDynDisplay", COMP_TYPE_GUI, GUI_DYNDISPLAY_CLASSID )

ParamBlockDesc2 gGUIDynDisplayBk
(	// KLUDGE: not the defined block ID, but kept for backwards compat.
 plComponent::kBlkComp, _T("GUIDynDisplay"), 0, &gGUIDynDisplayDesc, P_AUTO_CONSTRUCT + P_AUTO_UI + P_MULTIMAP, plComponent::kRefComp,

	1,
	plGUIControlBase::kRollMain, IDD_COMP_GUIDYNDISPLAY, IDS_COMP_GUIDYNDISPLAY, 0, 0, &sGUIDynDisplayProc,

	plGUIDynDisplayComponent::kRefDynLayer,	_T("dynLayer"),	TYPE_TEXMAP, 0, 0,
//		p_ui, plGUIControlBase::kRollMain, TYPE_TEXMAPBUTTON, IDC_GUI_COMPSELBTN,
		end,

	end
);

plGUIDynDisplayComponent::plGUIDynDisplayComponent()
{
	fClassDesc = &gGUIDynDisplayDesc;
	fClassDesc->MakeAutoParamBlocks(this);
}

// SetupProperties - Internal setup and write-only set properties on the MaxNode. No reading
// of properties on the MaxNode, as it's still indeterminant.
hsBool plGUIDynDisplayComponent::SetupProperties(plMaxNode *node,  plErrorMsg *pErrMsg)
{
	return plGUIControlBase::SetupProperties( node, pErrMsg );
}

hsBool plGUIDynDisplayComponent::PreConvert(plMaxNode *node,  plErrorMsg *pErrMsg)
{
	return plGUIControlBase::PreConvert( node, pErrMsg );
}

hsBool plGUIDynDisplayComponent::Convert(plMaxNode *node, plErrorMsg *pErrMsg)
{
	if( !plGUIControlBase::Convert( node, pErrMsg ) )
		return false;

	pfGUIDynDisplayCtrl *ctrl = (pfGUIDynDisplayCtrl *)fControl;
	
	Texmap *tmap = fCompPB->GetTexmap( plGUIDynDisplayComponent::kRefDynLayer );
	plPlasmaMAXLayer *pLayer = plPlasmaMAXLayer::GetPlasmaMAXLayer( tmap );
	if( pLayer == nil /*|| pLayer->ClassID() != DYN_TEXT_LAYER_CLASS_ID */ )
	{

		pErrMsg->Set(true, "GUI Control Component Error", "The texmap selected for the Dynamic Display Control on object \"%s\" is not a Plasma Dynamic Text Layer. Please fix.", node->GetName() ).Show();
		return false;	
	}
	
	const hsTArray<hsMaterialConverter::DoneMaterialData> &materials = hsMaterialConverter::Instance().DoneMaterials();

	UInt32 i,count = pLayer->GetNumConversionTargets();
	for( i = 0; i < count; i++ )
	{
		plLayerInterface *layIface = pLayer->GetConversionTarget( i );
		
		ctrl->AddLayer( layIface );

		plDynamicTextMap *map = plDynamicTextMap::ConvertNoRef( layIface->GetTexture() );
		if( map != nil )
			ctrl->AddMap( map );

		UInt32 mat;
		bool found = false;
		for (mat=0; mat<materials.GetCount(); mat++)
		{
			hsGMaterial *curMaterial = materials[mat].fHsMaterial;
			UInt32 lay;
			for (lay=0; lay<curMaterial->GetNumLayers(); lay++)
			{
				if (layIface->BottomOfStack() == curMaterial->GetLayer(lay))
				{
					ctrl->AddMaterial(curMaterial);
					found = true;
					break;
				}
			}
			if (found)
				break;
		}
	}

	return true;
}

/////////////////////////////////////////////////////////////////////////////////////////////////
//// plGUIMultiLineEdit Component ///////////////////////////////////////////////////////////////
//
//	GUI element that can be dragged anywhere in the 2D viewing plane.
//
/////////////////////////////////////////////////////////////////////////////////////////////////


// Class that accesses the paramblock below.
class plGUIMultiLineEditComp : public plGUIControlBase
{
protected:

	virtual pfGUIControlMod	*IGetNewControl( void ) { return TRACKED_NEW pfGUIMultiLineEditCtrl; }
	virtual bool			INeedsDynamicText( void ) { return true; }

public:
	plGUIMultiLineEditComp();
	void DeleteThis() { delete this; }

	// SetupProperties - Internal setup and write-only set properties on the MaxNode. No reading
	// of properties on the MaxNode, as it's still indeterminant.
	hsBool SetupProperties(plMaxNode *pNode, plErrorMsg *pErrMsg);

	hsBool PreConvert(plMaxNode *pNode, plErrorMsg *pErrMsg);
	hsBool Convert(plMaxNode *node, plErrorMsg *pErrMsg);

	enum
	{
		kRefXparentBgnd,
		kRefScaleWithRes,
		kRefUseScroll,
		kRefScrollCtrl,
	};
};

static plGUISingleCtrlDlgProc sGUIMultiLineProc( plGUIMultiLineEditComp::kRefScrollCtrl, IDC_GUI_COMPSELBTN,
											"Select the control to use for scrolling this multi-line edit box", sScrollingClassesToSelect );

//Max desc stuff necessary below.
CLASS_DESC(plGUIMultiLineEditComp, gGUIMultiLineEditDesc, "GUI Multi-Line Edit Box",  "GUIMultiLineEdit", COMP_TYPE_GUI, GUI_MULTILINE_CLASSID )

ParamBlockDesc2 gGUIMultiLineEditBoxBk
(	// KLUDGE: not the defined block ID, but kept for backwards compat.
 plComponent::kBlkComp, _T("GUIMultiLineEdit"), 0, &gGUIMultiLineEditDesc, P_AUTO_CONSTRUCT + P_AUTO_UI + P_MULTIMAP + P_INCLUDE_PARAMS, plComponent::kRefComp,

	2,
	plGUIControlBase::kRollMain, IDD_COMP_GUIMULTILINE, IDS_COMP_GUIMULTILINE, 0, 0, &sGUIMultiLineProc,
	plGUIControlBase::kRollProc, IDD_COMP_GUIPROCROLLOUT, IDS_COMP_GUIPROCROLLOUT, 0, 0, nil,	
	
	&sGUIControlProcParamTemplate,	

	plGUIMultiLineEditComp::kRefXparentBgnd,	_T( "xparentBgnd" ), TYPE_BOOL, 0, 0,
		p_ui, plGUIControlBase::kRollMain, TYPE_SINGLECHEKBOX, IDC_GUI_XPARENT,
		p_default, FALSE,
		end,
	
	plGUIMultiLineEditComp::kRefScaleWithRes,	_T( "scaleWithRes" ), TYPE_BOOL, 0, 0,
		p_ui, plGUIControlBase::kRollMain, TYPE_SINGLECHEKBOX, IDC_GUI_SCALERES,
		p_default, FALSE,
		end,
			
	plGUIMultiLineEditComp::kRefUseScroll,	_T( "enableScrolling" ), TYPE_BOOL, 0, 0,
		p_ui, plGUIControlBase::kRollMain, TYPE_SINGLECHEKBOX, IDC_GUI_SCROLLCTRL,
		p_default, FALSE,
		p_enable_ctrls, 1, plGUIMultiLineEditComp::kRefScrollCtrl,
		end,

	plGUIMultiLineEditComp::kRefScrollCtrl, _T("scrollControl"),	TYPE_INODE,		0, 0,
		p_prompt, IDS_COMP_GUI_SELECTSCROLL,
		p_accessor, &sGUIListBoxAccessor,
		end,

	end
);

plGUIMultiLineEditComp::plGUIMultiLineEditComp()
{
	fClassDesc = &gGUIMultiLineEditDesc;
	fClassDesc->MakeAutoParamBlocks(this);
}

// SetupProperties - Internal setup and write-only set properties on the MaxNode. No reading
// of properties on the MaxNode, as it's still indeterminant.
hsBool plGUIMultiLineEditComp::SetupProperties(plMaxNode *node,  plErrorMsg *pErrMsg)
{
	return plGUIControlBase::SetupProperties( node, pErrMsg );
}

hsBool plGUIMultiLineEditComp::PreConvert(plMaxNode *node,  plErrorMsg *pErrMsg)
{
	return plGUIControlBase::PreConvert( node, pErrMsg );
}

hsBool plGUIMultiLineEditComp::Convert(plMaxNode *node, plErrorMsg *pErrMsg)
{
	if( !plGUIControlBase::Convert( node, pErrMsg ) )
		return false;

	pfGUIMultiLineEditCtrl *ctrl = (pfGUIMultiLineEditCtrl *)fControl;
	
	if( fCompPB->GetInt( kRefXparentBgnd ) )
		ctrl->SetFlag( pfGUIControlMod::kXparentBgnd );

	if( fCompPB->GetInt( kRefScaleWithRes ) )
		ctrl->SetFlag( pfGUIControlMod::kScaleTextWithResolution );

	if( fCompPB->GetInt( kRefUseScroll ) )
	{
		// Get the scrolling control to use
		pfGUIValueCtrl *scroll = pfGUIValueCtrl::ConvertNoRef( GrabControlMod( fCompPB->GetINode( kRefScrollCtrl ) ) );
		if( scroll != nil )
		{
			hsgResMgr::ResMgr()->AddViaNotify( scroll->GetKey(), TRACKED_NEW plGenRefMsg( ctrl->GetKey(), 
										plRefMsg::kOnCreate, -1, pfGUIMultiLineEditCtrl::kRefScrollCtrl ), plRefFlags::kActiveRef );
		}
	}

	return true;
}


/////////////////////////////////////////////////////////////////////////////////////////////////
//// plGUIProgressCtrl Component ///////////////////////////////////////////////////////////////////
//
//	GUI element that can be dragged anywhere in the 2D viewing plane.
//
/////////////////////////////////////////////////////////////////////////////////////////////////


// Class that accesses the paramblock below.
class plGUIProgressCtrlComponent : public plGUIControlBase
{
protected:

	virtual pfGUIControlMod	*IGetNewControl( void ) { return TRACKED_NEW pfGUIProgressCtrl; }
	virtual bool			ICanHaveProxy( void ) { return false; }

public:
	plGUIProgressCtrlComponent();
	void DeleteThis() { delete this; }

	// SetupProperties - Internal setup and write-only set properties on the MaxNode. No reading
	// of properties on the MaxNode, as it's still indeterminant.
	hsBool SetupProperties(plMaxNode *pNode, plErrorMsg *pErrMsg);

	hsBool PreConvert(plMaxNode *pNode, plErrorMsg *pErrMsg);
	hsBool Convert(plMaxNode *node, plErrorMsg *pErrMsg);

	enum
	{
		kRefMinValue,
		kRefMaxValue,
		kRefStep,
		kReverseValues,
		kRefOrientation,
		kRefMouseMapping,
		kRefTriggerOnMouseUp,
		kRefAnimation,
		kRefAnimationNode,
		kRefAnimationNodeType,
		kRefAnimateSound,
		kRefAnimateSoundComp
	};
};

class plGUIProgressCtrlAccessor : public PBAccessor
{
public:
	void Set( PB2Value& v, ReferenceMaker* owner, ParamID id, int tabIndex, TimeValue t )
	{
		if( id == plGUIProgressCtrlComponent::kRefAnimateSoundComp )
		{
			plGUIProgressCtrlComponent *comp = (plGUIProgressCtrlComponent *)owner;
			comp->NotifyDependents( FOREVER, PART_ALL, REFMSG_USER_COMP_REF_CHANGED );
		}
	}
};

static plGUIProgressCtrlAccessor	sGUIProgressCtrlAccessor;

class plGUISoundDlgProc : public ParamMap2UserDlgProc
{
protected:
	ParamID			fSoundID;
	int				fSoundItem;
	TCHAR			fTitle[ 128 ];
	
public:
	plGUISoundDlgProc( ParamID soundID, int soundItem, TCHAR *title )
	{
		fSoundID = soundID;
		fSoundItem = soundItem;
		strcpy( fTitle, title );
	}
	
	BOOL DlgProc( TimeValue t, IParamMap2 *map, HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam )
	{
		switch ( msg )
		{
		case WM_INITDIALOG:
			{
				IParamBlock2 *pb = map->GetParamBlock();
				
				INode *node = pb->GetINode( fSoundID );
				TSTR newName( node ? node->GetName() : "Pick" );
				::SetWindowText( ::GetDlgItem( hWnd, fSoundItem ), newName );
			}
			return true;
			
		case WM_COMMAND:
			if( ( HIWORD( wParam ) == BN_CLICKED ) )
			{
				if( LOWORD( wParam ) == fSoundItem )
				{
					IParamBlock2 *pb = map->GetParamBlock();
					plGUICtrlHitCallback hitCB( (INode *)pb->GetOwner(), pb, fSoundID, fTitle, true, GUI_SOUND_COMPONENT_ID );
					GetCOREInterface()->DoHitByNameDialog( &hitCB );
					
					INode* node = pb->GetINode( fSoundID );
					TSTR newName( node ? node->GetName() : "Pick" );
					::SetWindowText( ::GetDlgItem(hWnd, fSoundItem ), newName );
					map->Invalidate( fSoundID );
					::InvalidateRect( hWnd, NULL, TRUE );
					return true;
				}
			}
			break;
		}
		
		return false;
	}
	
	void DeleteThis() {}
};

static plGUISoundDlgProc			sGUIProgressCtrlSndProc( plGUIProgressCtrlComponent::kRefAnimateSoundComp, IDC_GUI_ANIMSNDCOMP,
											"Select the sound to play when this control animates" );

static plPlasmaAnimSelectDlgProc	sGUIProgressCtrlProc( plGUIProgressCtrlComponent::kRefAnimation, IDC_GUI_COMPSELBTN, 
													plGUIProgressCtrlComponent::kRefAnimationNode, plGUIProgressCtrlComponent::kRefAnimationNodeType, IDC_GUI_ANIMNODESEL, 
													"Select the animation to use when displaying this knob control", &sGUIProgressCtrlSndProc );

//Max desc stuff necessary below.
CLASS_DESC(plGUIProgressCtrlComponent, gGUIProgressCtrlDesc, "GUI Progress Control",  "GUIProgressCtrl", COMP_TYPE_GUI, GUI_PROGRESS_CLASSID )

ParamBlockDesc2 gGUIProgressCtrlBk
(	// KLUDGE: not the defined block ID, but kept for backwards compat.
 plComponent::kBlkComp, _T("GUIProgressCtrl"), 0, &gGUIProgressCtrlDesc, P_AUTO_CONSTRUCT + P_AUTO_UI + P_MULTIMAP + P_INCLUDE_PARAMS, plComponent::kRefComp,

	2,
	plGUIControlBase::kRollMain, IDD_COMP_GUIPROGRESS, IDS_COMP_GUIPROGRESS, 0, 0, &sGUIProgressCtrlProc,
	plGUIControlBase::kRollProc, IDD_COMP_GUIPROCROLLOUT, IDS_COMP_GUIPROCROLLOUT, 0, 0, nil,	
	
	&sGUIControlProcParamTemplate,
	
	plGUIProgressCtrlComponent::kRefMinValue, _T("minValue"),	TYPE_FLOAT,	0, 0,	
		p_default, 0.0f,
		p_ui, plGUIControlBase::kRollMain, TYPE_SPINNER,	EDITTYPE_POS_FLOAT, IDC_GUI_LOWER, IDC_GUI_LOWER_SPIN, SPIN_AUTOSCALE,
		end,

	plGUIProgressCtrlComponent::kRefMaxValue, _T("maxValue"),	TYPE_FLOAT,	0, 0,	
		p_default, 10.0f,
		p_range, -10000.f, 10000.f,			// WHY do we even need to specify this?
		p_ui, plGUIControlBase::kRollMain, TYPE_SPINNER,	EDITTYPE_POS_FLOAT, IDC_GUI_UPPER, IDC_GUI_UPPER_SPIN, SPIN_AUTOSCALE,
		end,

	plGUIProgressCtrlComponent::kRefStep, _T("step"),	TYPE_FLOAT,	0, 0,	
		p_default, 1.0f,
		p_ui, plGUIControlBase::kRollMain, TYPE_SPINNER,	EDITTYPE_POS_FLOAT, IDC_GUI_STEP, IDC_GUI_STEP_SPIN, SPIN_AUTOSCALE,
		end,

	plGUIProgressCtrlComponent::kReverseValues, _T("reverseValues"), TYPE_BOOL, 0, 0,
		p_ui, plGUIControlBase::kRollMain, TYPE_SINGLECHEKBOX, IDC_GUI_REVERSE,
		p_default, FALSE,
		end,

	plGUIProgressCtrlComponent::kRefAnimation, _T("animation"),	TYPE_INODE,		0, 0,
		p_prompt, IDS_COMP_GUI_SELECTANIM,
		end,

	plGUIProgressCtrlComponent::kRefAnimationNode, _T("animationNode"),	TYPE_INODE,		0, 0,
		end,

	plGUIProgressCtrlComponent::kRefAnimationNodeType, _T("animationNodeType"),	TYPE_INT,		0, 0,
		p_default, plAnimObjInterface::kUseOwnerNode,
		end,

	plGUIProgressCtrlComponent::kRefAnimateSound,	_T( "animateSound" ), TYPE_BOOL, 0, 0,
		p_ui, plGUIControlBase::kRollMain, TYPE_SINGLECHEKBOX, IDC_GUI_ANIMSND,
		p_default, FALSE,
		p_enable_ctrls, 1, plGUIProgressCtrlComponent::kRefAnimateSoundComp,
		end,
		
	plGUIProgressCtrlComponent::kRefAnimateSoundComp, _T("animateSoundComp"),	TYPE_INODE,		0, 0,
		p_accessor, &sGUIProgressCtrlAccessor,
		end,

	end
);

plGUIProgressCtrlComponent::plGUIProgressCtrlComponent()
{
	fClassDesc = &gGUIProgressCtrlDesc;
	fClassDesc->MakeAutoParamBlocks(this);
}

// SetupProperties - Internal setup and write-only set properties on the MaxNode. No reading
// of properties on the MaxNode, as it's still indeterminant.
hsBool plGUIProgressCtrlComponent::SetupProperties(plMaxNode *node,  plErrorMsg *pErrMsg)
{
	node->SetForceLocal( true );

	plAnimObjInterface *iface = plAnimComponentBase::GetAnimInterface( fCompPB->GetINode( kRefAnimation ) );
	if( iface != nil && iface->MightRequireSeparateMaterial() )
	{
		INode *restrict = ( fCompPB->GetInt( kRefAnimationNodeType ) == plAnimObjInterface::kUseParamBlockNode ) 
								? fCompPB->GetINode( kRefAnimationNode )
								: (INode *)node;

		if( restrict != nil )
		{
			node->SetForceMaterialCopy( true );
		}
	}

	return plGUIControlBase::SetupProperties( node, pErrMsg );
}

hsBool plGUIProgressCtrlComponent::PreConvert(plMaxNode *node,  plErrorMsg *pErrMsg)
{
	return plGUIControlBase::PreConvert( node, pErrMsg );
}

// For hackery below (see warning below)
#include "../plAvatar/plAGMasterMod.h"

hsBool plGUIProgressCtrlComponent::Convert(plMaxNode *node, plErrorMsg *pErrMsg)
{
	if( !plGUIControlBase::Convert( node, pErrMsg ) )
		return false;

	pfGUIProgressCtrl *ctrl = (pfGUIProgressCtrl *)fControl;
	
	ctrl->SetRange( fCompPB->GetFloat( kRefMinValue ), fCompPB->GetFloat( kRefMaxValue ) );
	ctrl->SetStep( fCompPB->GetFloat( kRefStep ) );

	if( fCompPB->GetInt( kReverseValues ) )
		ctrl->SetFlag( pfGUIProgressCtrl::kReverseValues );

	// Get the animation to use
	plAnimObjInterface *iface = plAnimComponentBase::GetAnimInterface( fCompPB->GetINode( kRefAnimation ) );
	if( iface != nil )
	{
		INode *restrict = ( fCompPB->GetInt( kRefAnimationNodeType ) == plAnimObjInterface::kUseParamBlockNode ) 
								? fCompPB->GetINode( kRefAnimationNode )
								: (INode *)node;


		hsTArray<plKey> keys;
		if( iface->GetKeyList( restrict, keys ) && keys.GetCount() > 0 )
			ctrl->SetAnimationKeys( keys, iface->GetIfaceSegmentName( false ) );
	}
	else
	{
		// HACKERY WARNING: Old knobs assumed the animation was just the same one applied to our node,
		// so to avoid breaking old formats, if we can't grab an animObjInterface, we just grab the key
		// of the master mod of our node, like we would've before
		plAGMasterMod	*master = node->GetAGMasterMod();
		hsTArray<plKey> keys;
		keys.Append( master->GetKey() );
		ctrl->SetAnimationKeys( keys, ENTIRE_ANIMATION_NAME );
	}

	const char *errMsg = ISetSoundIndex( kRefAnimateSound, kRefAnimateSoundComp, pfGUIProgressCtrl::kAnimateSound, node );
	if( errMsg != nil )
	{
		pErrMsg->Set( true, "GUI Sound Event Error", errMsg, node->GetName() ).Show();
		pErrMsg->Set( false );
	}

	return true;
}

/////////////////////////////////////////////////////////////////////////////////////////////////
//// plGUIClickMap Component ///////////////////////////////////////////////////////////////////
//
//	GUI element that just keeps track of where on its surface (from 0-1) that it was clicked.
//
/////////////////////////////////////////////////////////////////////////////////////////////////


// Class that accesses the paramblock below.
class plGUIClickMapComponent : public plGUIControlBase
{
protected:

	virtual pfGUIControlMod	*IGetNewControl( void ) { return TRACKED_NEW pfGUIClickMapCtrl; }
	virtual bool			ICanHaveProxy( void ) { return false; }

public:
	plGUIClickMapComponent();
	void DeleteThis() { delete this; }

	// SetupProperties - Internal setup and write-only set properties on the MaxNode. No reading
	// of properties on the MaxNode, as it's still indeterminant.
	hsBool SetupProperties(plMaxNode *pNode, plErrorMsg *pErrMsg);

	hsBool PreConvert(plMaxNode *pNode, plErrorMsg *pErrMsg);
	hsBool Convert(plMaxNode *node, plErrorMsg *pErrMsg);

	enum
	{
		kRefReportDragging
	};
};

//Max desc stuff necessary below.
CLASS_DESC(plGUIClickMapComponent, gGUIClickMapDesc, "GUI Clickable Map",  "GUIClickMap", COMP_TYPE_GUI, GUI_CLICKMAP_CLASSID )

ParamBlockDesc2 gGUIClickMapBk
(	// KLUDGE: not the defined block ID, but kept for backwards compat.
 plComponent::kBlkComp, _T("GUIClickMap"), 0, &gGUIClickMapDesc, P_AUTO_CONSTRUCT + P_AUTO_UI + P_MULTIMAP + P_INCLUDE_PARAMS, plComponent::kRefComp,

	2,
	plGUIControlBase::kRollMain, IDD_COMP_GUICLICKMAP, IDS_COMP_GUICLICKMAP, 0, 0, NULL,
	plGUIControlBase::kRollProc, IDD_COMP_GUIPROCROLLOUT, IDS_COMP_GUIPROCROLLOUT, 0, 0, nil,	
	
	&sGUIControlProcParamTemplate,

	plGUIClickMapComponent::kRefReportDragging, _T("reportWhileDragging"), TYPE_BOOL, 0, 0,
		p_ui, plGUIControlBase::kRollMain, TYPE_SINGLECHEKBOX, IDC_GUI_REPORTDRAG,
		p_default, FALSE,
		end,

	end
);

plGUIClickMapComponent::plGUIClickMapComponent()
{
	fClassDesc = &gGUIClickMapDesc;
	fClassDesc->MakeAutoParamBlocks(this);
}

// SetupProperties - Internal setup and write-only set properties on the MaxNode. No reading
// of properties on the MaxNode, as it's still indeterminant.
hsBool plGUIClickMapComponent::SetupProperties(plMaxNode *node,  plErrorMsg *pErrMsg)
{
	node->SetForceLocal( true );
	return plGUIControlBase::SetupProperties( node, pErrMsg );
}

hsBool plGUIClickMapComponent::PreConvert(plMaxNode *node,  plErrorMsg *pErrMsg)
{
	return plGUIControlBase::PreConvert( node, pErrMsg );
}

hsBool plGUIClickMapComponent::Convert(plMaxNode *node, plErrorMsg *pErrMsg)
{
	if( !plGUIControlBase::Convert( node, pErrMsg ) )
		return false;

	pfGUIClickMapCtrl *ctrl = (pfGUIClickMapCtrl *)fControl;
	
	if( fCompPB->GetInt( kRefReportDragging ) )
		ctrl->SetFlag( pfGUIClickMapCtrl::kReportDragging );

	return true;
}


/////////////////////////////////////////////////////////////////////////////////////////////////
//// plGUISkin Component ////////////////////////////////////////////////////////////////////
//
//	Defines a skin to use when rendering certain GUI controls (just menus for now)
//
/////////////////////////////////////////////////////////////////////////////////////////////////


class pfGUISkinProc : public ParamMap2UserDlgProc
{
protected:

public:

	void DeleteThis() {}

//	virtual void	Update( TimeValue t, Interval &valid, IParamMap2 *map );

	BOOL DlgProc( TimeValue t, IParamMap2 *map, HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam );
};

static pfGUISkinProc gGUISkinProc;

// Component defined in pfGUISkinProc.h

#define kDeclSkinRectValues( ref ) (plGUISkinComp::##ref + 0), _T("f##ref##.left"), TYPE_INT, 0, 0, p_default, 0, end, \
									(plGUISkinComp::##ref + 1), _T("f##ref##.top"), TYPE_INT, 0, 0, p_default, 0, end, \
									(plGUISkinComp::##ref + 2), _T("f##ref##.width"), TYPE_INT, 0, 0, p_default, 8, end, \
									(plGUISkinComp::##ref + 3), _T("f##ref##.height"), TYPE_INT, 0, 0, p_default, 8, end

#define kSetSkinRectValues( pb, ref, l, t, w, h ) { pb->SetValue( ref + 0, 0, (int) l ); \
													pb->SetValue( ref + 1, 0, (int) t ); \
													pb->SetValue( ref + 2, 0, (int) r ); \
													pb->SetValue( ref + 3, 0, (int) b ); }


//Max desc stuff necessary below.
CLASS_DESC(plGUISkinComp, gGUISkinDesc, "GUI Skin",  "GUISkin", COMP_TYPE_GUI, GUI_SKIN_CLASSID )

static ParamBlockDesc2	gGUISkinBk
(
	/// Main def
	 plComponent::kBlkComp, _T("GUISkin"), 0, &gGUISkinDesc, P_AUTO_CONSTRUCT + P_AUTO_UI, plComponent::kRefComp,

	IDD_COMP_GUISKIN, IDS_COMP_GUISKIN, 0, 0, &gGUISkinProc,	

	plGUISkinComp::kRefBitmap,	_T("bitmap"),		TYPE_TEXMAP, 		0, 0,
		end,

	kDeclSkinRectValues( kRefUpLeftCorner ),
	kDeclSkinRectValues( kRefTopSpan ),
	kDeclSkinRectValues( kRefUpRightCorner ),
	kDeclSkinRectValues( kRefRightSpan ),
	kDeclSkinRectValues( kRefLowerRightCorner ),
	kDeclSkinRectValues( kRefBottomSpan ),
	kDeclSkinRectValues( kRefLowerLeftCorner ),
	kDeclSkinRectValues( kRefLeftSpan ),
	kDeclSkinRectValues( kRefMiddleFill ),
	kDeclSkinRectValues( kRefSelectedFill ),
	kDeclSkinRectValues( kRefSubMenuArrow ),
	kDeclSkinRectValues( kRefSelectedSubMenuArrow ),
	kDeclSkinRectValues( kRefTreeButtonClosed ),
	kDeclSkinRectValues( kRefTreeButtonOpen ),

	plGUISkinComp::kRefItemMargin,	_T("itemMargin"),		TYPE_INT, 		0, 0,
		p_ui,	TYPE_SPINNER, EDITTYPE_POS_INT, IDC_GUI_IMARGIN, IDC_GUI_IMARGIN_SPIN, SPIN_AUTOSCALE,
		p_default, 1,
		end,

	plGUISkinComp::kRefBorderMargin,	_T("borderMargin"),		TYPE_INT, 		0, 0,
		p_ui,	TYPE_SPINNER, EDITTYPE_POS_INT, IDC_GUI_BMARGIN, IDC_GUI_BMARGIN_SPIN, SPIN_AUTOSCALE,
		p_default, 4,
		end,

	end
);

// Editor proc
extern HINSTANCE hInstance;

BOOL pfGUISkinProc::DlgProc( TimeValue t, IParamMap2 *pmap, HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam )
{
	IParamBlock2		*pb = pmap->GetParamBlock();
	plGUISkinComp		*comp = (plGUISkinComp *)pb->GetOwner();
	PBBitmap			*bitmap;
	plLayerTex			*layer = comp->GetSkinBitmap();
	ICustButton			*bmSelectBtn;

	switch( msg )
	{
		case WM_INITDIALOG:
			// Set projection map bitmap name
			bmSelectBtn = GetICustButton( GetDlgItem( hWnd, IDC_GUI_SKINBMAP ) );
			if( bmSelectBtn != nil )
			{
				bitmap = ( layer == nil ) ? nil : layer->GetPBBitmap();
				if( bitmap != nil )
					bmSelectBtn->SetText( (TCHAR *)bitmap->bi.Filename() );
				else
					bmSelectBtn->SetText( _T( "<none>" ) );
				ReleaseICustButton( bmSelectBtn );
			}

			return true;

		case WM_DESTROY:
			break;

		case WM_COMMAND:
			if( LOWORD( wParam ) == IDC_GUI_EDITELEM )
			{
				bitmap = ( layer == nil ) ? nil : layer->GetPBBitmap();
				if( bitmap != nil )
				{
					pfGUISkinEditProc	proc( comp );
					DialogBox( hInstance, MAKEINTRESOURCE( IDD_COMP_SKINEDIT ), GetCOREInterface()->GetMAXHWnd(), proc.DlgProc );
				}
			}

			else if( LOWORD( wParam ) == IDC_GUI_SKINBMAP )
			{
				BOOL selectedNewBitmap = layer->HandleBitmapSelection();
				if( selectedNewBitmap )
				{
					bmSelectBtn = GetICustButton( GetDlgItem( hWnd, IDC_GUI_SKINBMAP ) );
					bitmap = layer->GetPBBitmap();
					bmSelectBtn->SetText( bitmap != nil ? (TCHAR *)bitmap->bi.Filename() : "");
					ReleaseICustButton( bmSelectBtn );
				}
				return false;
			}
			break;

	}
	return false;
}

plKey	plGUISkinComp::GetConvertedSkinKey( void ) const
{
	if( fConvertedSkin != nil )
		return fConvertedSkin->GetKey();

	return nil;
}

UInt32	plGUISkinComp::GetNumMtls( void ) const
{
	return 1;
}

Texmap	*plGUISkinComp::GetMtl( UInt32 idx )
{
	return (Texmap *)GetSkinBitmap();
}

//// GetSkinBitmap ///////////////////////////////////////////////////////////

plLayerTex	*plGUISkinComp::GetSkinBitmap( void )
{  
	// If we don't have one, create one
	plLayerTex	*layer = (plLayerTex *)fCompPB->GetTexmap( kRefBitmap, 0 );
	if( layer == nil || layer->ClassID() != LAYER_TEX_CLASS_ID )
	{
		layer = TRACKED_NEW plLayerTex;

		fCompPB->SetValue( kRefBitmap, 0, (Texmap *)layer );
	}
	if( layer )
	{
		IParamBlock2* bitmapPB = layer->GetParamBlockByID( plLayerTex::kBlkBitmap );
		if( bitmapPB->GetInt(kBmpScaling) != kScalingNone )
			bitmapPB->SetValue(kBmpScaling, TimeValue(0), kScalingNone);
	}


	return layer;
}
	
plGUISkinComp::plGUISkinComp()
{
	fClassDesc = &gGUISkinDesc;
	fClassDesc->MakeAutoParamBlocks(this);
}

// SetupProperties - Internal setup and write-only set properties on the MaxNode. No reading
// of properties on the MaxNode, as it's still indeterminant.
hsBool plGUISkinComp::SetupProperties(plMaxNode *node,  plErrorMsg *pErrMsg)
{
	fConvertedSkin = nil;
	return true;
}

hsBool plGUISkinComp::PreConvert(plMaxNode *node,  plErrorMsg *pErrMsg)
{
	// Create and assign key here, so other components can grab the key later
	if( fConvertedSkin != nil )
		return true;		// Only convert once, since we don't care what node we're on

	Texmap *texture = fCompPB->GetTexmap( kRefBitmap );
	if( texture == nil || texture->ClassID() != LAYER_TEX_CLASS_ID || ( (plLayerTex *)texture )->GetPBBitmap() == nil )
	{
		pErrMsg->Set( true, "GUI Skin Convert Error", 
							"The GUI skin component %s doesn't have a mipmap associated with it. This skin will not "
							"be exported.", GetINode()->GetName() ).CheckAndAsk();
		pErrMsg->Set( false );
		return true;
	}

	fConvertedSkin = TRACKED_NEW pfGUISkin();
	hsgResMgr::ResMgr()->NewKey( IGetUniqueName(node), fConvertedSkin, node->GetLocation() );

	return true;
}

hsBool plGUISkinComp::Convert(plMaxNode *node, plErrorMsg *pErrMsg)
{
	// Actually do the work of converting all the skin data
	if( fConvertedSkin == nil )
		return true;		// Eh?


	fConvertedSkin->SetMargins( fCompPB->GetInt( kRefItemMargin ), fCompPB->GetInt( kRefBorderMargin ) );

	UInt32	i;
	for( i = 0; i < pfGUISkin::kNumElements; i++ )
	{
		ParamID		id = ( i * 4 ) + kRefUpLeftCorner;

		fConvertedSkin->SetElement( i, fCompPB->GetInt( id + 0 ), fCompPB->GetInt( id + 1 ), 
									fCompPB->GetInt( id + 2 ), fCompPB->GetInt( id + 3 ) );
	}

	plLayerTex *layer= (plLayerTex *)fCompPB->GetTexmap( kRefBitmap );
	if( layer != nil )
	{
		PBBitmap *texture = layer->GetPBBitmap();
		if( texture != nil )
		{
			plBitmap *bMap = plLayerConverter::Instance().CreateSimpleTexture( texture->bi.Name(), fConvertedSkin->GetKey()->GetUoid().GetLocation(), 0, plMipmap::kForceNonCompressed | plMipmap::kAlphaChannelFlag | plMipmap::kNoMaxSize );
			if( bMap != nil && plMipmap::ConvertNoRef( bMap ) != nil )
			{
				hsgResMgr::ResMgr()->AddViaNotify( bMap->GetKey(), TRACKED_NEW plGenRefMsg( fConvertedSkin->GetKey(), 
										plRefMsg::kOnCreate, -1, pfGUISkin::kRefMipmap ), plRefFlags::kActiveRef );
			}
		}
	}

	return true;
}

hsBool plGUISkinComp::DeInit(plMaxNode *node, plErrorMsg *pErrMsg)
{
	fConvertedSkin = nil;
	return true; 
}


/////////////////////////////////////////////////////////////////////////////////////////////////
//// plGUIPopUpMenu Component ///////////////////////////////////////////////////////////////////
//
//	Defines a pop-up menu, with an auto-anchor to the sceneObject it's attached to
//
/////////////////////////////////////////////////////////////////////////////////////////////////


/*class plGUIMenuProc : public ParamMap2UserDlgProc
{
protected:

	void	ILoadPages( HWND hWnd, IParamBlock2 *pb );

public:

	void DeleteThis() {}

	BOOL DlgProc( TimeValue t, IParamMap2 *map, HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam );
};
static plGUIMenuProc gGUIMenuProc;
*/

static plGUISingleCtrlDlgProc sGUISkinSelectProc( plGUIMenuComponent::kRefSkin, IDC_GUI_SKIN,
											"Select the skin to use for this pop-up menu", sSkinClassesToSelect,
											&gGUIDialogProc );

//Max desc stuff necessary below.
CLASS_DESC(plGUIMenuComponent, gGUIMenuDesc, "GUI Menu",  "GUIMenu", COMP_TYPE_GUI, GUI_MENUANCHOR_CLASSID )

ParamBlockDesc2 gGUIMenuBk
(	// KLUDGE: not the defined block ID, but kept for backwards compat.
 plComponent::kBlkComp, _T("GUIMenu"), 0, &gGUIMenuDesc, P_AUTO_CONSTRUCT + P_AUTO_UI + P_MULTIMAP + P_INCLUDE_PARAMS, plComponent::kRefComp,

	3, 
	plGUIMenuComponent::kMainRollout,	IDD_COMP_GUIMENUANCHOR, IDS_COMP_GUIMENUANCHOR, 0, 0, &sGUISkinSelectProc,
	plGUIMenuComponent::kTagIDRollout,	IDD_COMP_GUITAG,	IDS_COMP_GUITAG,	0, 0, &gGUITagProc,
	plGUIMenuComponent::kSchemeRollout,	IDD_COMP_GUISCHEME, IDS_COMP_GUISCHEME, 0, 0, &gGUIColorSchemeProc,	

	&gGUIColorSchemeBk,

		plGUIMenuComponent::kRefDialogName,	_T("MenuName"),		TYPE_STRING, 		0, 0,
//			p_ui, plGUIMenuComponent::kMainRollout, TYPE_EDITBOX, IDC_GUIDLG_NAME,
			end,

		plGUIMenuComponent::kRefAgeName,	_T("ageName"),		TYPE_STRING, 		0, 0,
			p_default, _T( "GUI" ),
			end,
			
		plGUIMenuComponent::kRefVersion,	_T("version"),		TYPE_INT, 		0, 0,
			p_ui,	plGUIMenuComponent::kMainRollout, TYPE_SPINNER, EDITTYPE_POS_INT, IDC_GUI_VERSION, IDC_GUI_VERSION_SPIN, SPIN_AUTOSCALE,
			p_default, 0,
			end,

		plGUITagComponent::kRefCurrIDSel,	_T("currSel"),		TYPE_INT, 		0, 0,
			end,

		plGUIMenuComponent::kRefSkin, _T("skin"),	TYPE_INODE,		0, 0,
			end,

		plGUIMenuComponent::kRefNeverClose, _T("neverClose"),		TYPE_BOOL, 		0, 0,
			p_default,	FALSE,
			p_ui, plGUIMenuComponent::kMainRollout, TYPE_SINGLECHEKBOX, IDC_GUI_NEVERCLOSE,
			end,		

		plGUIMenuComponent::kRefModalOutside, _T("modalOutside"),		TYPE_BOOL, 		0, 0,
			p_default,	FALSE,
			p_ui, plGUIMenuComponent::kMainRollout, TYPE_SINGLECHEKBOX, IDC_GUI_MODALOUTSIDE,
			end,		

		plGUIMenuComponent::kRefOpenOnHover, _T("openSubsOnHover"),		TYPE_BOOL, 		0, 0,
			p_default,	FALSE,
			p_ui, plGUIMenuComponent::kMainRollout, TYPE_SINGLECHEKBOX, IDC_GUI_HOVER,
			end,		

		plGUIMenuComponent::kRefAlignment, _T("alignment"), TYPE_INT, 0, 0,
			p_default, 3,
			p_ui, plGUIMenuComponent::kMainRollout, TYPE_RADIO, 4, IDC_ALIGNRADIO1, IDC_ALIGNRADIO2, IDC_ALIGNRADIO3, IDC_ALIGNRADIO4,
			end,

		plGUIMenuComponent::kRefScaleWithScreenRes, _T("maintainSizeAcrossRes"),		TYPE_BOOL, 		0, 0,
			p_default,	FALSE,
			p_ui, plGUIMenuComponent::kMainRollout, TYPE_SINGLECHEKBOX, IDC_GUI_SCALERES,
			end,		

	end
);

plGUIMenuComponent::plGUIMenuComponent() : plGUIDialogComponent( true )
{
	fClassDesc = &gGUIMenuDesc;
	fClassDesc->MakeAutoParamBlocks(this);
}

pfGUIDialogMod	*plGUIMenuComponent::IMakeDialog( void )
{
	return TRACKED_NEW pfGUIPopUpMenu();
}

plKey	plGUIMenuComponent::GetConvertedMenuKey( void ) const
{
	if( fConvertedMenu == nil )
		return nil;

	return fConvertedMenu->GetKey();
}

// SetupProperties - Internal setup and write-only set properties on the MaxNode. No reading
// of properties on the MaxNode, as it's still indeterminant.
hsBool plGUIMenuComponent::SetupProperties(plMaxNode *node,  plErrorMsg *pErrMsg)
{
//	return plGUIDialogComponent::SetupProperties( node, pErrMsg );
	fConvertedMenu = nil;
	return true;
}

hsBool plGUIMenuComponent::Convert(plMaxNode *node, plErrorMsg *pErrMsg)
{
	pfGUIPopUpMenu *menu = fConvertedMenu;

//	hsBool	b = plGUIDialogComponent::Convert( node, pErrMsg );
//	if( b )
	{
//		pfGUIPopUpMenu *menu = pfGUIPopUpMenu::ConvertNoRef( fDialogMod );
//		hsAssert( menu != nil, "Somehow got a bad poitner in GUIMenu::Convert()" );

		INode *sNode = fCompPB->GetINode( kRefSkin );
		if( sNode != nil )
		{
			plComponentBase *comp = ( (plMaxNode *)sNode )->ConvertToComponent();
			if( comp != nil )
			{
				Class_ID nodeID = comp->ClassID();
				hsAssert( nodeID == GUI_SKIN_CLASSID, "Bad node param in GUIMenu::Convert()" );

				plGUISkinComp *skin = (plGUISkinComp *)comp;
				menu->SetSkin( skin->GetConvertedSkin() );
			}
		}

		if( fCompPB->GetInt( kRefNeverClose ) )
			menu->SetFlag( pfGUIPopUpMenu::kStayOpenAfterClick );

		if( fCompPB->GetInt( kRefModalOutside ) )
			menu->SetFlag( pfGUIPopUpMenu::kModalOutsideMenus );

		if( fCompPB->GetInt( kRefOpenOnHover ) )
			menu->SetFlag( pfGUIPopUpMenu::kOpenSubMenusOnHover );

		if( fCompPB->GetInt( kRefScaleWithScreenRes ) )
			menu->SetFlag( pfGUIPopUpMenu::kScaleWithResolution );

		switch( fCompPB->GetInt( kRefAlignment ) )
		{
			case 0: menu->SetAlignment( pfGUIPopUpMenu::kAlignUpLeft ); break;
			case 1: menu->SetAlignment( pfGUIPopUpMenu::kAlignUpRight ); break;
			case 2: menu->SetAlignment( pfGUIPopUpMenu::kAlignDownLeft ); break;
			case 3: menu->SetAlignment( pfGUIPopUpMenu::kAlignDownRight ); break;
		}
	}

	// Note: we use the owning dialog of our anchor object as the context, i.e. who translates
	// our point at runtime into screen coordinates
	menu->SetOriginAnchor( node->GetSceneObject(), plGUIDialogComponent::GetNodeDialog( node ) );

	const plLocation &loc = menu->GetKey()->GetUoid().GetLocation();

	// Create the rendermod
	plPostEffectMod	*renderMod = TRACKED_NEW plPostEffectMod;
	hsgResMgr::ResMgr()->NewKey( IGetUniqueName(node), renderMod, loc );

	renderMod->SetHither( 0.5f );
	renderMod->SetYon( 200.f );
	renderMod->SetNodeKey( fConvertedNode );

	float scrnWidth = 20.f;

	// fovX should be such that scrnWidth is the projected width at z=100
	float fovX = atan( scrnWidth / ( 2.f * 100.f ) ) * 2.f;
	float fovY = fovX;// * 3.f / 4.f;

	renderMod->SetFovX( fovX * 180.f / hsScalarPI );
	renderMod->SetFovY( fovY * 180.f / hsScalarPI );


	hsgResMgr::ResMgr()->AddViaNotify( renderMod->GetKey(), TRACKED_NEW plNodeRefMsg( fConvertedNode, plRefMsg::kOnCreate, -1, plNodeRefMsg::kGeneric ), plRefFlags::kActiveRef );
	hsgResMgr::ResMgr()->AddViaNotify( fConvertedNode, TRACKED_NEW plGenRefMsg( renderMod->GetKey(), plRefMsg::kOnCreate, 0, plPostEffectMod::kNodeRef ), plRefFlags::kPassiveRef );		

	menu->SetRenderMod( renderMod );
	menu->SetName( fCompPB->GetStr( kRefDialogName ) );

	// Create the dummy scene object to hold the menu
	plSceneObject	*newObj = TRACKED_NEW plSceneObject;
	hsgResMgr::ResMgr()->NewKey( IGetUniqueName(node), newObj, loc );

	// *#&$(*@&#$ need a coordIface...
	plCoordinateInterface *newCI = TRACKED_NEW plCoordinateInterface;
	hsgResMgr::ResMgr()->NewKey( IGetUniqueName(node), newCI, loc );


	hsgResMgr::ResMgr()->AddViaNotify( menu->GetKey(), TRACKED_NEW plObjRefMsg( newObj->GetKey(), plRefMsg::kOnCreate, -1, plObjRefMsg::kModifier ), plRefFlags::kActiveRef );		

	hsgResMgr::ResMgr()->AddViaNotify( newCI->GetKey(), TRACKED_NEW plObjRefMsg( newObj->GetKey(), plRefMsg::kOnCreate, -1, plObjRefMsg::kInterface ), plRefFlags::kActiveRef );		
	hsgResMgr::ResMgr()->AddViaNotify( renderMod->GetKey(), TRACKED_NEW plObjRefMsg( newObj->GetKey(), plRefMsg::kOnCreate, -1, plObjRefMsg::kModifier ), plRefFlags::kActiveRef );		

	newObj->SetSceneNode( fConvertedNode );
	menu->SetSceneNodeKey( fConvertedNode );

	{
		hsMatrix44 l2w, w2l;
		l2w.Reset();
		l2w.GetInverse( &w2l );
		newObj->SetTransform( l2w, w2l );
	}

	// Should be done now...
	return true;
}

hsBool plGUIMenuComponent::PreConvert(plMaxNode *node,  plErrorMsg *pErrMsg)
{
	// Create and assign key here, so other components can grab the key later
	if( fConvertedMenu != nil )
		return true;		// Only convert once, since we don't care what node we're on

	/// Create an entirely new sceneNode for us
	Int32 seqNum = plPageInfoUtils::GetSeqNumFromAgeDesc( fCompPB->GetStr( kRefAgeName ), fCompPB->GetStr( kRefDialogName ) );
	Int32 newNum = plPluginResManager::ResMgr()->VerifySeqNumber( seqNum, fCompPB->GetStr( kRefAgeName ), fCompPB->GetStr( kRefDialogName ) );
	if( newNum != seqNum )
	{
		if( !fSeqNumValidated )
		{
			char errMsg[ 512 ];
			sprintf( errMsg, "GUI Menu Component %s has an invalid location sequence number (0x%X). Temporarily using a valid one (0x%X).", 
								node->GetName(), seqNum, newNum );
			pErrMsg->Set( true, "PageInfo Convert Error", errMsg ).Show(); 
			pErrMsg->Set( false );
			fSeqNumValidated = true;
		}
		seqNum = newNum;
	}

	fConvertedNode = plPluginResManager::ResMgr()->NameToLoc( fCompPB->GetStr( kRefAgeName ), fCompPB->GetStr( kRefDialogName ), seqNum );
	if( !fConvertedNode )
	{
		pErrMsg->Set( true, "GUI Menu Component Error", "GUI MenuComponent %s has a Missing Location.  Nuke the files in the dat directory and re-export.",((INode*)node)->GetName()).Show(); 
		return false;
	}

	fConvertedMenu = TRACKED_NEW pfGUIPopUpMenu();
	hsgResMgr::ResMgr()->NewKey( IGetUniqueName(node), fConvertedMenu, fConvertedNode->GetUoid().GetLocation() );

	return true;

//	return plGUIDialogComponent::PreConvert( node, pErrMsg );
}

hsBool plGUIMenuComponent::DeInit(plMaxNode *node, plErrorMsg *pErrMsg)
{
	fConvertedMenu = nil;
	fConvertedNode = nil;
	return true;
}