//========= Copyright © 1996-2002, Valve LLC, All rights reserved. ============ // // Purpose: // // $NoKeywords: $ //============================================================================= #ifndef ELEMENTPROPERTIESTREE_H #define ELEMENTPROPERTIESTREE_H #ifdef _WIN32 #pragma once #endif #include "vgui_controls/Frame.h" #include "dme_controls/AttributeWidgetFactory.h" #include "vgui_controls/TreeView.h" #include "vgui_controls/TreeViewListControl.h" #include "datamodel/dmelement.h" #include "datamodel/dmattribute.h" #include "datamodel/dmattributevar.h" #include "datamodel/dmehandle.h" #include "datamodel/dmattributevar.h" #include "tier1/utlntree.h" #include "tier1/utlstring.h" #include "tier1/utlvector.h" #include "vgui_controls/InputDialog.h" #include "vgui/KeyCode.h" #include "dme_controls/inotifyui.h" //----------------------------------------------------------------------------- // Forward declarations //----------------------------------------------------------------------------- class CDmElement; class IDmNotify; class CDocAllElements; class IAttributeWidgetFactory; class CDmeEditorTypeDictionary; class CPropertiesTreeToolbar; namespace vgui { class TextEntry; class ComboBox; class Button; class PanelListPanel; class Menu; } //----------------------------------------------------------------------------- // CElementTreeViewListControl //----------------------------------------------------------------------------- class CElementTreeViewListControl : public vgui::CTreeViewListControl { DECLARE_CLASS_SIMPLE( CElementTreeViewListControl, CTreeViewListControl ); public: CElementTreeViewListControl( Panel *pParent, const char *pName ); virtual void ApplySchemeSettings( vgui::IScheme *pScheme ); virtual int AddItem( KeyValues *data, bool allowLabelEditing, int parentItemIndex, CUtlVector< vgui::Panel * >& columnPanels ); virtual void RemoveItem( int nItemIndex ); virtual void PerformLayout(); virtual void RemoveAll(); virtual vgui::HFont GetFont( int size ); virtual void SetFont( vgui::HFont font ); virtual int GetFontSize(); virtual void SetFontSize( int size ); virtual void Paint(); virtual void PostChildPaint(); virtual void ExpandItem( int itemIndex, bool bExpand ); virtual bool IsItemExpanded( int itemIndex ); virtual bool IsItemSelected( int itemIndex ); virtual KeyValues *GetItemData( int itemIndex ); virtual int GetTreeColumnWidth(); virtual void SetTreeColumnWidth( int w ); virtual void OnCursorMoved( int x, int y ); virtual void OnMousePressed( vgui::MouseCode code ); virtual void OnMouseReleased( vgui::MouseCode code ); virtual void OnMouseDoublePressed( vgui::MouseCode code ); virtual void OnMouseWheeled( int delta ); virtual int GetScrollBarSize(); virtual void ToggleDrawGrid(); virtual bool IsDrawingGrid(); virtual void ToggleDrawAlternatingRowColors(); virtual bool IsDrawingAlternatingRowColors(); virtual bool IsHidingTypeSubColumn(); virtual void ToggleHideSubColumn(); void ResizeTreeToExpandedWidth(); private: struct ColumnPanels_t { ColumnPanels_t() : treeViewItem( -1 ) { } ColumnPanels_t( const ColumnPanels_t& src ) { treeViewItem = src.treeViewItem; int i, c; c = src.m_Columns.Count(); for ( i = 0; i < c; ++i ) { m_Columns.AddToTail( src.m_Columns[ i ] ); } } void SetList( CUtlVector< vgui::Panel * >& list ) { m_Columns.RemoveAll(); int c = list.Count(); for ( int i = 0; i < c; ++i ) { m_Columns.AddToTail( list[ i ] ); } } int treeViewItem; CUtlVector< vgui::Panel * > m_Columns; }; // Removes an item from the tree recursively void RemoveItem_R( int nItemIndex ); void HideAll(); static bool PanelsLessFunc( const ColumnPanels_t& lhs, const ColumnPanels_t& rhs ) { return lhs.treeViewItem < rhs.treeViewItem; } int m_iTreeColumnWidth; int m_iFontSize; // 1 = verySmall, small, normal, large, verylarge bool m_bMouseLeftIsDown; bool m_bMouseIsDragging; bool m_bDrawGrid; bool m_bDrawAlternatingRowColors; bool m_bHideTypeSubColumn; CUtlRBTree< ColumnPanels_t, int > m_Panels; }; //----------------------------------------------------------------------------- // CElementPropertiesTreeInternal //----------------------------------------------------------------------------- class CElementPropertiesTreeInternal : public vgui::EditablePanel { DECLARE_CLASS_SIMPLE( CElementPropertiesTreeInternal, vgui::EditablePanel ); public: enum RefreshType_t { REFRESH_REBUILD = 0, // Close the entire tree REFRESH_VALUES_ONLY, // Tree topology hasn't changed; only update values REFRESH_TREE_VIEW, // Tree topology changed; some attributes may be added or removed }; CElementPropertiesTreeInternal( vgui::Panel *parent, IDmNotify *pNotify, CDmElement *pObject, bool autoApply = true, CDmeEditorTypeDictionary *pDict = NULL ); ~CElementPropertiesTreeInternal(); virtual void Init( ); virtual void Refresh( RefreshType_t rebuild = REFRESH_TREE_VIEW, bool preservePrevSelectedItem = false ); virtual void ApplyChanges(); virtual void GenerateChildrenOfNode( int itemIndex ); virtual void GenerateContextMenu( int itemIndex, int x, int y ); virtual void GenerateDragDataForItem( int itemIndex, KeyValues *msg ); virtual void OnLabelChanged( int itemIndex, char const *oldString, char const *newString ); virtual bool IsItemDroppable( int itemIndex, bool bInsertBefore, CUtlVector< KeyValues * >& msglist ); virtual void OnItemDropped( int itemIndex, bool bInsertBefore, CUtlVector< KeyValues * >& msglist ); virtual bool GetItemDropContextMenu( int itemIndex, vgui::Menu *menu, CUtlVector< KeyValues * >& msglist ); virtual vgui::HCursor GetItemDropCursor( int itemIndex, CUtlVector< KeyValues * >& msglist ); virtual void SetObject( CDmElement *object ); virtual void OnCommand( const char *cmd ); MESSAGE_FUNC( OnToggleShowMemoryUsage, "OnToggleShowMemoryUsage" ); MESSAGE_FUNC( OnToggleShowUniqueID, "OnToggleShowUniqueID" ); virtual bool IsShowingMemoryUsage(); virtual bool IsShowingUniqueID(); CDmElement *GetObject(); bool IsLabelBeingEdited() const; bool HasItemsSelected() const; enum { DME_PROPERTIESTREE_MENU_BACKWARD = 0, DME_PROPERTIESTREE_MENU_FORWARD, DME_PROPERTIESTREE_MENU_SEARCHHSITORY, DME_PROPERTIESTREE_MENU_UP, }; virtual void PopulateHistoryMenu( int whichMenu, vgui::Menu *menu ); virtual int GetHistoryMenuItemCount( int whichMenu ); void AddToSearchHistory( char const *str ); void SetTypeDictionary( CDmeEditorTypeDictionary *pDict ); void SetSortAttributesByName( bool bSortAttributesByName ); MESSAGE_FUNC_CHARPTR( OnNavSearch, "OnNavigateSearch", text ); protected: KeyValues *GetTreeItemData( int itemIndex ); protected: struct AttributeWidgets_t { vgui::Panel *m_pValueWidget; bool operator==( const AttributeWidgets_t &src ) const { return m_pValueWidget == src.m_pValueWidget; } }; enum { EP_EXPANDED = (1<<0), EP_SELECTED = (1<<1), }; struct TreeItem_t { TreeItem_t() : m_pElement( 0 ), m_pAttributeName(), m_pArrayElement( 0 ) { } CDmElement *m_pElement; CUtlString m_pAttributeName; CDmElement *m_pArrayElement; // points to the element referenced in an element array }; // Used to build a list of open element for refresh struct TreeInfo_t { TreeInfo_t() : m_nFlags( 0 ) { } TreeItem_t m_Item; // points to the element referenced in an element array int m_nFlags; TreeItem_t m_Preserved; }; typedef CUtlNTree< TreeInfo_t, int > OpenItemTree_t; struct SearchResult_t { CDmeHandle< CDmElement > handle; CUtlString attributeName; bool operator == ( const SearchResult_t &other ) const { if ( &other == this ) return true; if ( other.handle != handle ) return false; if ( other.attributeName != attributeName ) return false; return true; } }; bool BuildExpansionListToFindElement_R( CUtlRBTree< CDmElement *, int >& visited, int depth, SearchResult_t& sr, CDmElement *owner, CDmElement *element, char const *attributeName, int arrayIndex, CUtlVector< int >& expandIndices ); void FindMatchingElements_R( CUtlRBTree< CDmElement *, int >& visited, char const *searchstr, const DmObjectId_t *pSearchId, CDmElement *root, CUtlVector< SearchResult_t >& list ); void NavigateToSearchResult(); void SpewOpenItems( int depth, OpenItemTree_t &tree, int nOpenTreeIndex, int nItemIndex ); // Finds the tree index of a child matching the particular element + attribute int FindTreeItem( int nParentIndex, const TreeItem_t &info ); // Expands all items in the open item tree if they exist void ExpandOpenItems( OpenItemTree_t &tree, int nOpenTreeIndex, int nItemIndex, bool makeVisible ); // Builds a list of open items void BuildOpenItemList( OpenItemTree_t &tree, int nParent, int nItemIndex, bool preservePrevSelectedItem ); void FillInDataForItem( TreeItem_t &item, int nItemIndex ); // Removes an item from the tree void RemoveItem( int nItemIndex ); // Removes an item recursively void RemoveItem_R( int nItemIndex ); // Adds a single entry into the tree void CreateTreeEntry( int parentNodeIndex, CDmElement* obj, CDmAttribute *pAttribute, int nArrayIndex, AttributeWidgets_t &entry ); // populate the menu with the element hierarchy of "element_" commands void PopulateMenuWithElementHierarchy_R( vgui::Menu *pMenu, const char *pElementType, CDmElementFactoryHelper *pChildFactory = NULL ); void PopulateMenuWithElementHierarchy_R( vgui::Menu *pMenu, CDmElementFactoryHelper *pFactory ); // Sets up the attribute widget init info for a particular attribute virtual void SetupWidgetInfo( AttributeWidgetInfo_t *pInfo, CDmElement *obj, CDmAttribute *pAttribute, int nArrayIndex = -1 ); // Creates an attribute data widget using a specifically requested widget vgui::Panel *CreateAttributeDataWidget( CDmElement *pElement, const char *pWidgetName, CDmElement *obj, CDmAttribute *pAttribute, int nArrayIndex = -1 ); void UpdateTree(); void InsertAttributes( int parentNodeIndex, CDmElement *obj ); void InsertAttributeArrayMembers( int parentNodeIndex, CDmElement *obj, CDmAttribute *array ); // Adds a single editable attribute of the element to the tree void InsertSingleAttribute( int parentNodeIndex, CDmElement *obj, CDmAttribute *pAttribute, int nArrayIndex = -1 ); // Refreshes the tree view void RefreshTreeView( bool preservePrevSelectedItem = false ); // Gets tree view text void GetTreeViewText( CDmElement* obj, CDmAttribute *pAttribute, int nArrayIndex, char *pBuffer, int nMaxLen, bool& editableText ); void RemoveSelected( bool selectLeft ); void AddToHistory( CDmElement *element ); void ValidateHistory(); void SpewHistory(); void JumpToHistoryItem(); void UpdateButtonState(); void UpdateReferences(); KEYBINDING_FUNC( ondelete, KEY_DELETE, 0, OnKeyDelete, "#elementpropertiestree_ondelete_help", 0 ); KEYBINDING_FUNC( onbackspace, KEY_BACKSPACE, 0, OnKeyBackspace, "#elementpropertiestree_ondelete_help", 0 ); KEYBINDING_FUNC_NODECLARE( onrefresh, KEY_F5, 0, OnRefresh, "#elementpropertiestree_onrefresh_help", 0 ); MESSAGE_FUNC( OnEstimateMemory, "OnEstimateMemory" ); MESSAGE_FUNC( OnRename, "OnRename" ); MESSAGE_FUNC( OnRemove, "OnRemove" ); MESSAGE_FUNC( OnClear, "OnClear" ); MESSAGE_FUNC( OnSortByName, "OnSortByName" ); MESSAGE_FUNC( OnCut, "OnCut" ); MESSAGE_FUNC( OnCopy, "OnCopy" ); MESSAGE_FUNC( OnPaste, "OnPaste" ); MESSAGE_FUNC( OnPasteReference, "OnPasteReference" ); MESSAGE_FUNC( OnPasteInsert, "OnPasteInsert" ); MESSAGE_FUNC_INT( OnElementChangedExternally, "ElementChangedExternally", valuesOnly ); MESSAGE_FUNC_INT( OnNavUp, "OnNavigateUp", item ); MESSAGE_FUNC_INT( OnNavBack, "OnNavigateBack", item ); MESSAGE_FUNC_INT( OnNavForward, "OnNavigateForward", item ); MESSAGE_FUNC_INT( OnNavigateSearchAgain, "OnNavigateSearchAgain", direction ); MESSAGE_FUNC( OnShowSearchResults, "OnShowSearchResults" ); MESSAGE_FUNC( OnRefresh, "OnRefresh" ); protected: MESSAGE_FUNC_PARAMS( OnInputCompleted, "InputCompleted", params ); MESSAGE_FUNC( OnAddItem, "OnAddItem" ); MESSAGE_FUNC_PARAMS( OnSetShared, "OnSetShared", pParams ); MESSAGE_FUNC_PARAMS( OnChangeFile, "OnChangeFile", pParams ); MESSAGE_FUNC_PARAMS( OnShowFileDialog, "OnShowFileDialog", pParams ); MESSAGE_FUNC_PARAMS( OnFileSelected, "FileSelected", params ); enum DropOperation_t { DO_MOVE, DO_LINK, DO_COPY, DO_UNKNOWN, }; DropOperation_t GetDropOperation( int itemIndex, CUtlVector< KeyValues * >& msglist ); void DropItemsIntoArray( CDmrElementArray<> &array, CUtlVector< KeyValues* > &msglist, CUtlVector< CDmElement* > &list, int nArrayIndex, DropOperation_t op ); bool OnRemoveFromData( CUtlVector< KeyValues * >& list ); bool OnRemoveFromData( KeyValues *item ); void OnPaste_( bool reference ); bool ShowAddAttributeDialog( CDmElement *pElement, const char *pAttributeType ); void AddAttribute( const char *pAttributeName, KeyValues *pContext ); bool ShowSetElementAttributeDialog( CDmElement *pOwner, const char *pAttributeName, int nArrayItem, const char *pElementType ); void SetElementAttribute( const char *pElementName, KeyValues *pContext ); void OnImportElement( const char *pFullPath, KeyValues *pContext ); void OnExportElement( const char *pFullPath, KeyValues *pContext ); template < class C > void CollectSelectedElements( C &container ); void GetPathToItem( CUtlVector< TreeItem_t > &path, int itemIndex ); int OpenPath( const CUtlVector< TreeItem_t > &path ); // Refreshes the color state of the tree void RefreshTreeItemState( int nItemID ); // Refreshes the color state of the tree void SetTreeItemColor( int nItemID, CDmElement *pEntryElement, bool bIsElementArrayItem, bool bEditableLabel ); CDmeHandle< CDmeEditorTypeDictionary > m_hTypeDictionary; CUtlVector< AttributeWidgets_t > m_AttributeWidgets; IDmNotify *m_pNotify; CDmeHandle< CDmElement > m_hObject; CElementTreeViewListControl *m_pTree; bool m_bAutoApply; bool m_bShowMemoryUsage; bool m_bShowUniqueID; vgui::DHANDLE< vgui::Menu > m_hContextMenu; CPropertiesTreeToolbar *m_pToolBar; // Forward/backward navigation and search fields enum { DME_PROPERTIESTREE_MAXHISTORYITEMS = 32, DME_PROPERTIESTREE_MAXSEARCHHISTORYITEMS = 32, }; // Most recent are at the head CUtlVector< CDmeHandle< CDmElement > > m_hHistory; int m_nCurrentHistoryPosition; bool m_bSuppressHistoryUpdates; char m_szSearchStr[ 128 ]; CUtlVector< SearchResult_t > m_SearchResults; int m_nCurrentSearchResult; CUtlVector< CUtlString > m_SearchHistory; CDmeHandle< CDmElement > m_SearchResultsRoot; vgui::HCursor m_hDragCopyCursor; vgui::HCursor m_hDragLinkCursor; vgui::HCursor m_hDragMoveCursor; bool m_bSortAttributesByName; CUtlVector< CDmeHandle< CDmElement > > m_vecDmeReferencesToObject; }; //----------------------------------------------------------------------------- // CElementPropertiesTree //----------------------------------------------------------------------------- class CElementPropertiesTree : public vgui::Frame { DECLARE_CLASS_SIMPLE( CElementPropertiesTree, vgui::Frame ); public: CElementPropertiesTree( vgui::Panel *parent, IDmNotify *pNotify, CDmElement *pObject, CDmeEditorTypeDictionary *pDict = NULL ); virtual void Init( ); virtual void Refresh( CElementPropertiesTreeInternal::RefreshType_t rebuild = CElementPropertiesTreeInternal::REFRESH_REBUILD, bool preservePrevSelectedItem = false ); virtual void GenerateChildrenOfNode( int itemIndex ); virtual void SetObject( CDmElement *object ); virtual void OnCommand( const char *cmd ); virtual void ActivateBuildMode(); CElementPropertiesTreeInternal *GetInternal(); protected: CElementPropertiesTreeInternal *m_pProperties; vgui::Button *m_pOK; vgui::Button *m_pApply; vgui::Button *m_pCancel; }; //----------------------------------------------------------------------------- // Inline methods //----------------------------------------------------------------------------- inline CElementPropertiesTreeInternal *CElementPropertiesTree::GetInternal() { return m_pProperties; } //----------------------------------------------------------------------------- // Wrapper panel to hook into DmePanels //----------------------------------------------------------------------------- class CDmeElementPanel : public CElementPropertiesTreeInternal, public IDmNotify { DECLARE_CLASS_SIMPLE( CDmeElementPanel, CElementPropertiesTreeInternal ); public: CDmeElementPanel( vgui::Panel *pParent, const char *pPanelName ); void SetDmeElement( CDmElement *pElement ); // Inherited from IDmNotify virtual void NotifyDataChanged( const char *pReason, int nNotifySource, int nNotifyFlags ); }; #endif // ELEMENTPROPERTIESTREE_H