mirror of
https://github.com/alliedmodders/hl2sdk.git
synced 2025-01-04 00:23:25 +08:00
849 lines
22 KiB
C++
849 lines
22 KiB
C++
|
//====== Copyright <20> 1996-2005, Valve Corporation, All rights reserved. =======
|
|||
|
//
|
|||
|
// Purpose:
|
|||
|
//
|
|||
|
//=============================================================================
|
|||
|
#include "vgui_controls/KeyBoardEditorDialog.h"
|
|||
|
#include "vgui_controls/ListPanel.h"
|
|||
|
#include "vgui_controls/Button.h"
|
|||
|
#include "vgui_controls/TextEntry.h"
|
|||
|
#include "vgui/ISurface.h"
|
|||
|
#include "vgui/IInput.h"
|
|||
|
#include "vgui/IVGui.h"
|
|||
|
#include "vgui/ILocalize.h"
|
|||
|
#include "KeyValues.h"
|
|||
|
#include "vgui/Cursor.h"
|
|||
|
#include "tier1/UtlDict.h"
|
|||
|
|
|||
|
using namespace vgui;
|
|||
|
|
|||
|
static char *CopyString( const char *in )
|
|||
|
{
|
|||
|
if ( !in )
|
|||
|
return NULL;
|
|||
|
|
|||
|
int len = strlen( in );
|
|||
|
char *n = new char[ len + 1 ];
|
|||
|
Q_strncpy( n, in, len + 1 );
|
|||
|
return n;
|
|||
|
}
|
|||
|
|
|||
|
CKeyBoardEditorPage::SaveMapping_t::SaveMapping_t() : map( 0 )
|
|||
|
{
|
|||
|
}
|
|||
|
|
|||
|
CKeyBoardEditorPage::SaveMapping_t::SaveMapping_t( const SaveMapping_t& src )
|
|||
|
{
|
|||
|
map = src.map;
|
|||
|
current = src.current;
|
|||
|
original = src.original;
|
|||
|
}
|
|||
|
|
|||
|
//-----------------------------------------------------------------------------
|
|||
|
// Purpose: Special list subclass to handle drawing of trap mode prompt on top of
|
|||
|
// lists client area
|
|||
|
//-----------------------------------------------------------------------------
|
|||
|
class VControlsListPanel : public ListPanel
|
|||
|
{
|
|||
|
DECLARE_CLASS_SIMPLE( VControlsListPanel, ListPanel );
|
|||
|
|
|||
|
public:
|
|||
|
// Construction
|
|||
|
VControlsListPanel( vgui::Panel *parent, const char *listName );
|
|||
|
virtual ~VControlsListPanel();
|
|||
|
|
|||
|
// Start/end capturing
|
|||
|
virtual void StartCaptureMode(vgui::HCursor hCursor = NULL);
|
|||
|
virtual void EndCaptureMode(vgui::HCursor hCursor = NULL);
|
|||
|
virtual bool IsCapturing();
|
|||
|
|
|||
|
// Set which item should be associated with the prompt
|
|||
|
virtual void SetItemOfInterest(int itemID);
|
|||
|
virtual int GetItemOfInterest();
|
|||
|
|
|||
|
virtual void OnMousePressed(vgui::MouseCode code);
|
|||
|
virtual void OnMouseDoublePressed(vgui::MouseCode code);
|
|||
|
|
|||
|
KEYBINDING_FUNC( clearbinding, KEY_DELETE, 0, OnClearBinding, 0, 0 );
|
|||
|
|
|||
|
private:
|
|||
|
void ApplySchemeSettings(vgui::IScheme *pScheme );
|
|||
|
|
|||
|
// Are we showing the prompt?
|
|||
|
bool m_bCaptureMode;
|
|||
|
// If so, where?
|
|||
|
int m_nClickRow;
|
|||
|
// Font to use for showing the prompt
|
|||
|
vgui::HFont m_hFont;
|
|||
|
// panel used to edit
|
|||
|
class CInlineEditPanel *m_pInlineEditPanel;
|
|||
|
int m_iMouseX, m_iMouseY;
|
|||
|
};
|
|||
|
|
|||
|
//-----------------------------------------------------------------------------
|
|||
|
// Purpose: panel used for inline editing of key bindings
|
|||
|
//-----------------------------------------------------------------------------
|
|||
|
class CInlineEditPanel : public vgui::Panel
|
|||
|
{
|
|||
|
DECLARE_CLASS_SIMPLE( CInlineEditPanel, vgui::Panel );
|
|||
|
|
|||
|
public:
|
|||
|
CInlineEditPanel() : vgui::Panel(NULL, "InlineEditPanel")
|
|||
|
{
|
|||
|
}
|
|||
|
|
|||
|
virtual void Paint()
|
|||
|
{
|
|||
|
int wide, tall;
|
|||
|
GetSize(wide, tall);
|
|||
|
|
|||
|
// Draw a white rectangle around that cell
|
|||
|
vgui::surface()->DrawSetColor( 63, 63, 63, 255 );
|
|||
|
vgui::surface()->DrawFilledRect( 0, 0, wide, tall );
|
|||
|
|
|||
|
vgui::surface()->DrawSetColor( 0, 255, 0, 255 );
|
|||
|
vgui::surface()->DrawOutlinedRect( 0, 0, wide, tall );
|
|||
|
}
|
|||
|
|
|||
|
virtual void OnKeyCodeTyped(KeyCode code)
|
|||
|
{
|
|||
|
// forward up
|
|||
|
if (GetParent())
|
|||
|
{
|
|||
|
GetParent()->OnKeyCodeTyped(code);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
virtual void ApplySchemeSettings(IScheme *pScheme)
|
|||
|
{
|
|||
|
Panel::ApplySchemeSettings(pScheme);
|
|||
|
SetBorder(pScheme->GetBorder("DepressedButtonBorder"));
|
|||
|
}
|
|||
|
|
|||
|
void OnMousePressed(vgui::MouseCode code)
|
|||
|
{
|
|||
|
// forward up mouse pressed messages to be handled by the key options
|
|||
|
if (GetParent())
|
|||
|
{
|
|||
|
GetParent()->OnMousePressed(code);
|
|||
|
}
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
//-----------------------------------------------------------------------------
|
|||
|
// Purpose: Construction
|
|||
|
//-----------------------------------------------------------------------------
|
|||
|
VControlsListPanel::VControlsListPanel( vgui::Panel *parent, const char *listName ) : BaseClass( parent, listName )
|
|||
|
{
|
|||
|
m_bCaptureMode = false;
|
|||
|
m_nClickRow = 0;
|
|||
|
m_pInlineEditPanel = new CInlineEditPanel();
|
|||
|
m_hFont = INVALID_FONT;
|
|||
|
}
|
|||
|
|
|||
|
//-----------------------------------------------------------------------------
|
|||
|
// Purpose: Destructor
|
|||
|
//-----------------------------------------------------------------------------
|
|||
|
VControlsListPanel::~VControlsListPanel()
|
|||
|
{
|
|||
|
m_pInlineEditPanel->MarkForDeletion();
|
|||
|
}
|
|||
|
|
|||
|
//-----------------------------------------------------------------------------
|
|||
|
// Purpose:
|
|||
|
//-----------------------------------------------------------------------------
|
|||
|
void VControlsListPanel::ApplySchemeSettings(IScheme *pScheme )
|
|||
|
{
|
|||
|
BaseClass::ApplySchemeSettings( pScheme );
|
|||
|
m_hFont = pScheme->GetFont("DefaultVerySmall", IsProportional() );
|
|||
|
}
|
|||
|
|
|||
|
//-----------------------------------------------------------------------------
|
|||
|
// Purpose: Start capture prompt display
|
|||
|
//-----------------------------------------------------------------------------
|
|||
|
void VControlsListPanel::StartCaptureMode( HCursor hCursor )
|
|||
|
{
|
|||
|
m_bCaptureMode = true;
|
|||
|
EnterEditMode(m_nClickRow, 1, m_pInlineEditPanel);
|
|||
|
input()->SetMouseFocus(m_pInlineEditPanel->GetVPanel());
|
|||
|
input()->SetMouseCapture(m_pInlineEditPanel->GetVPanel());
|
|||
|
|
|||
|
if (hCursor)
|
|||
|
{
|
|||
|
m_pInlineEditPanel->SetCursor(hCursor);
|
|||
|
|
|||
|
// save off the cursor position so we can restore it
|
|||
|
vgui::input()->GetCursorPos( m_iMouseX, m_iMouseY );
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
void VControlsListPanel::OnClearBinding()
|
|||
|
{
|
|||
|
if ( m_bCaptureMode )
|
|||
|
return;
|
|||
|
|
|||
|
if ( GetItemOfInterest() < 0 )
|
|||
|
return;
|
|||
|
|
|||
|
PostMessage( GetParent()->GetVPanel(), new KeyValues( "ClearBinding", "item", GetItemOfInterest() ) );
|
|||
|
}
|
|||
|
|
|||
|
//-----------------------------------------------------------------------------
|
|||
|
// Purpose: Finish capture prompt display
|
|||
|
//-----------------------------------------------------------------------------
|
|||
|
void VControlsListPanel::EndCaptureMode( HCursor hCursor )
|
|||
|
{
|
|||
|
m_bCaptureMode = false;
|
|||
|
input()->SetMouseCapture(NULL);
|
|||
|
LeaveEditMode();
|
|||
|
RequestFocus();
|
|||
|
input()->SetMouseFocus(GetVPanel());
|
|||
|
if (hCursor)
|
|||
|
{
|
|||
|
m_pInlineEditPanel->SetCursor(hCursor);
|
|||
|
surface()->SetCursor(hCursor);
|
|||
|
if ( hCursor != dc_none )
|
|||
|
{
|
|||
|
vgui::input()->SetCursorPos ( m_iMouseX, m_iMouseY );
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
//-----------------------------------------------------------------------------
|
|||
|
// Purpose: Set active row column
|
|||
|
//-----------------------------------------------------------------------------
|
|||
|
void VControlsListPanel::SetItemOfInterest(int itemID)
|
|||
|
{
|
|||
|
m_nClickRow = itemID;
|
|||
|
}
|
|||
|
|
|||
|
//-----------------------------------------------------------------------------
|
|||
|
// Purpose: Retrieve row, column of interest
|
|||
|
//-----------------------------------------------------------------------------
|
|||
|
int VControlsListPanel::GetItemOfInterest()
|
|||
|
{
|
|||
|
return m_nClickRow;
|
|||
|
}
|
|||
|
|
|||
|
//-----------------------------------------------------------------------------
|
|||
|
// Purpose: returns true if we're currently waiting to capture a key
|
|||
|
//-----------------------------------------------------------------------------
|
|||
|
bool VControlsListPanel::IsCapturing( void )
|
|||
|
{
|
|||
|
return m_bCaptureMode;
|
|||
|
}
|
|||
|
|
|||
|
//-----------------------------------------------------------------------------
|
|||
|
// Purpose: Forwards mouse pressed message up to keyboard page when in capture
|
|||
|
//-----------------------------------------------------------------------------
|
|||
|
void VControlsListPanel::OnMousePressed(vgui::MouseCode code)
|
|||
|
{
|
|||
|
if (IsCapturing())
|
|||
|
{
|
|||
|
// forward up mouse pressed messages to be handled by the key options
|
|||
|
if (GetParent())
|
|||
|
{
|
|||
|
GetParent()->OnMousePressed(code);
|
|||
|
}
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
BaseClass::OnMousePressed(code);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
//-----------------------------------------------------------------------------
|
|||
|
// Purpose: input handler
|
|||
|
//-----------------------------------------------------------------------------
|
|||
|
void VControlsListPanel::OnMouseDoublePressed( vgui::MouseCode code )
|
|||
|
{
|
|||
|
int c = GetSelectedItemsCount();
|
|||
|
if ( c > 0 )
|
|||
|
{
|
|||
|
// enter capture mode
|
|||
|
OnKeyCodeTyped(KEY_ENTER);
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
BaseClass::OnMouseDoublePressed(code);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
CKeyBoardEditorPage::CKeyBoardEditorPage( Panel *parent, Panel *panelToEdit, KeyBindingContextHandle_t handle )
|
|||
|
: BaseClass( parent, "KeyBoardEditorPage" ),
|
|||
|
m_pPanel( panelToEdit ),
|
|||
|
m_Handle( handle )
|
|||
|
{
|
|||
|
Assert( m_pPanel );
|
|||
|
|
|||
|
m_pList = new VControlsListPanel( this, "KeyBindings" );
|
|||
|
m_pList->SetIgnoreDoubleClick( true );
|
|||
|
m_pList->AddColumnHeader(0, "Action", "#KBEditorBindingName", 175, 0);
|
|||
|
m_pList->AddColumnHeader(1, "Binding", "#KBEditorBinding", 175, 0);
|
|||
|
m_pList->AddColumnHeader(2, "Description", "#KBEditorDescription", 300, 0);
|
|||
|
|
|||
|
LoadControlSettings( "resource/KeyBoardEditorPage.res" );
|
|||
|
|
|||
|
SaveMappings();
|
|||
|
}
|
|||
|
|
|||
|
//-----------------------------------------------------------------------------
|
|||
|
// Purpose:
|
|||
|
// Input : -
|
|||
|
//-----------------------------------------------------------------------------
|
|||
|
CKeyBoardEditorPage::~CKeyBoardEditorPage()
|
|||
|
{
|
|||
|
int c = m_Save.Count();
|
|||
|
for ( int i = 0 ; i < c; ++i )
|
|||
|
{
|
|||
|
delete m_Save[ i ];
|
|||
|
}
|
|||
|
m_Save.RemoveAll();
|
|||
|
}
|
|||
|
|
|||
|
void CKeyBoardEditorPage::ApplySchemeSettings( IScheme *scheme )
|
|||
|
{
|
|||
|
BaseClass::ApplySchemeSettings( scheme );
|
|||
|
PopulateList();
|
|||
|
}
|
|||
|
|
|||
|
//-----------------------------------------------------------------------------
|
|||
|
// Purpose:
|
|||
|
// Input : -
|
|||
|
//-----------------------------------------------------------------------------
|
|||
|
void CKeyBoardEditorPage::SaveMappings()
|
|||
|
{
|
|||
|
Assert( m_Save.Count() == 0 );
|
|||
|
|
|||
|
CUtlVector< PanelKeyBindingMap * > maps;
|
|||
|
GetMappingList( m_pPanel, maps );
|
|||
|
|
|||
|
// add header item
|
|||
|
int c = maps.Count();
|
|||
|
for ( int i = 0; i < c; ++i )
|
|||
|
{
|
|||
|
PanelKeyBindingMap *m = maps[ i ];
|
|||
|
SaveMapping_t *sm = new SaveMapping_t;
|
|||
|
sm->map = m;
|
|||
|
sm->current = m->boundkeys;
|
|||
|
sm->original = m->boundkeys;
|
|||
|
m_Save.AddToTail( sm );
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
//-----------------------------------------------------------------------------
|
|||
|
// Purpose:
|
|||
|
// Input : -
|
|||
|
//-----------------------------------------------------------------------------
|
|||
|
void CKeyBoardEditorPage::UpdateCurrentMappings()
|
|||
|
{
|
|||
|
int c = m_Save.Count();
|
|||
|
for ( int i = 0 ; i < c; ++i )
|
|||
|
{
|
|||
|
PanelKeyBindingMap *m = m_Save[ i ]->map;
|
|||
|
Assert( m );
|
|||
|
m_Save[ i ]->current = m->boundkeys;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
//-----------------------------------------------------------------------------
|
|||
|
// Purpose:
|
|||
|
// Input : -
|
|||
|
//-----------------------------------------------------------------------------
|
|||
|
void CKeyBoardEditorPage::RestoreMappings()
|
|||
|
{
|
|||
|
int c = m_Save.Count();
|
|||
|
for ( int i = 0; i < c; ++i )
|
|||
|
{
|
|||
|
SaveMapping_t *sm = m_Save[ i ];
|
|||
|
sm->current = sm->original;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
void CKeyBoardEditorPage::ApplyMappings()
|
|||
|
{
|
|||
|
int c = m_Save.Count();
|
|||
|
for ( int i = 0; i < c; ++i )
|
|||
|
{
|
|||
|
SaveMapping_t *sm = m_Save[ i ];
|
|||
|
sm->map->boundkeys = sm->current;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
//-----------------------------------------------------------------------------
|
|||
|
// Purpose: User clicked on item: remember where last active row/column was
|
|||
|
//-----------------------------------------------------------------------------
|
|||
|
void CKeyBoardEditorPage::ItemSelected()
|
|||
|
{
|
|||
|
int c = m_pList->GetSelectedItemsCount();
|
|||
|
if ( c > 0 )
|
|||
|
{
|
|||
|
m_pList->SetItemOfInterest( m_pList->GetSelectedItem( 0 ) );
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
void CKeyBoardEditorPage::BindKey( KeyCode code )
|
|||
|
{
|
|||
|
bool shift = (input()->IsKeyDown(KEY_LSHIFT) || input()->IsKeyDown(KEY_RSHIFT));
|
|||
|
bool ctrl = (input()->IsKeyDown(KEY_LCONTROL) || input()->IsKeyDown(KEY_RCONTROL));
|
|||
|
bool alt = (input()->IsKeyDown(KEY_LALT) || input()->IsKeyDown(KEY_RALT));
|
|||
|
|
|||
|
int modifiers = 0;
|
|||
|
if ( shift )
|
|||
|
{
|
|||
|
modifiers |= MODIFIER_SHIFT;
|
|||
|
}
|
|||
|
if ( ctrl )
|
|||
|
{
|
|||
|
modifiers |= MODIFIER_CONTROL;
|
|||
|
}
|
|||
|
if ( alt )
|
|||
|
{
|
|||
|
modifiers |= MODIFIER_ALT;
|
|||
|
}
|
|||
|
|
|||
|
int r = m_pList->GetItemOfInterest();
|
|||
|
|
|||
|
// Retrieve clicked row and column
|
|||
|
m_pList->EndCaptureMode(dc_arrow);
|
|||
|
|
|||
|
// Find item for this row
|
|||
|
KeyValues *item = m_pList->GetItem(r);
|
|||
|
if ( item )
|
|||
|
{
|
|||
|
BoundKey_t *kbMap = reinterpret_cast< BoundKey_t * >( item->GetPtr( "Item", 0 ) );
|
|||
|
if ( kbMap )
|
|||
|
{
|
|||
|
KeyBindingMap_t *binding = m_pPanel->LookupBindingByKeyCode( code, modifiers );
|
|||
|
if ( binding && Q_stricmp( kbMap->bindingname, binding->bindingname ) )
|
|||
|
{
|
|||
|
// Key is already rebound!!!
|
|||
|
char warning[ 512 ];
|
|||
|
Q_snprintf( warning, sizeof( warning ), "Can't bind to '%S', key is already bound to '%s'\n",
|
|||
|
Panel::KeyCodeToDisplayString( code ), binding->bindingname );
|
|||
|
Warning( warning );
|
|||
|
return;
|
|||
|
}
|
|||
|
|
|||
|
kbMap->keycode = code;
|
|||
|
kbMap->modifiers = modifiers;
|
|||
|
|
|||
|
PopulateList();
|
|||
|
}
|
|||
|
|
|||
|
KeyBindingMap_t *bindingMap = reinterpret_cast< KeyBindingMap_t * >( item->GetPtr( "Unbound", 0 ) );
|
|||
|
if ( bindingMap )
|
|||
|
{
|
|||
|
KeyBindingMap_t *binding = m_pPanel->LookupBindingByKeyCode( code, modifiers );
|
|||
|
if ( binding && Q_stricmp( bindingMap->bindingname, binding->bindingname ) )
|
|||
|
{
|
|||
|
// Key is already rebound!!!
|
|||
|
char warning[ 512 ];
|
|||
|
Q_snprintf( warning, sizeof( warning ), "Can't bind to '%S', key is already bound to '%s'\n",
|
|||
|
Panel::KeyCodeToDisplayString( code ), binding->bindingname );
|
|||
|
Warning( warning );
|
|||
|
return;
|
|||
|
}
|
|||
|
|
|||
|
// Need to add to current entries
|
|||
|
m_pPanel->AddKeyBinding( bindingMap->bindingname, code, modifiers );
|
|||
|
UpdateCurrentMappings();
|
|||
|
PopulateList();
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
void CKeyBoardEditorPage::OnPageHide()
|
|||
|
{
|
|||
|
if ( m_pList->IsCapturing() )
|
|||
|
{
|
|||
|
// Cancel capturing
|
|||
|
m_pList->EndCaptureMode(dc_arrow);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
//-----------------------------------------------------------------------------
|
|||
|
// Purpose: binds double-clicking or hitting enter in the keybind list to changing the key
|
|||
|
//-----------------------------------------------------------------------------
|
|||
|
void CKeyBoardEditorPage::OnKeyCodeTyped(vgui::KeyCode code)
|
|||
|
{
|
|||
|
switch ( code )
|
|||
|
{
|
|||
|
case KEY_ENTER:
|
|||
|
{
|
|||
|
if ( !m_pList->IsCapturing() )
|
|||
|
{
|
|||
|
OnCommand( "ChangeKey" );
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
BindKey( code );
|
|||
|
}
|
|||
|
}
|
|||
|
break;
|
|||
|
case KEY_LSHIFT:
|
|||
|
case KEY_RSHIFT:
|
|||
|
case KEY_LALT:
|
|||
|
case KEY_RALT:
|
|||
|
case KEY_LCONTROL:
|
|||
|
case KEY_RCONTROL:
|
|||
|
{
|
|||
|
// Swallow these
|
|||
|
break;
|
|||
|
}
|
|||
|
break;
|
|||
|
default:
|
|||
|
{
|
|||
|
if ( m_pList->IsCapturing() )
|
|||
|
{
|
|||
|
BindKey( code );
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
BaseClass::OnKeyCodeTyped(code);
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
void CKeyBoardEditorPage::OnCommand( char const *cmd )
|
|||
|
{
|
|||
|
if ( !m_pList->IsCapturing() && !Q_stricmp( cmd, "ChangeKey" ) )
|
|||
|
{
|
|||
|
m_pList->StartCaptureMode(dc_blank);
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
BaseClass::OnCommand( cmd );
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
void CKeyBoardEditorPage::OnSaveChanges()
|
|||
|
{
|
|||
|
ApplyMappings();
|
|||
|
}
|
|||
|
|
|||
|
void CKeyBoardEditorPage::OnRevert()
|
|||
|
{
|
|||
|
RestoreMappings();
|
|||
|
PopulateList();
|
|||
|
}
|
|||
|
|
|||
|
void CKeyBoardEditorPage::OnUseDefaults()
|
|||
|
{
|
|||
|
m_pPanel->RevertKeyBindingsToDefault();
|
|||
|
UpdateCurrentMappings();
|
|||
|
PopulateList();
|
|||
|
}
|
|||
|
|
|||
|
void CKeyBoardEditorPage::GetMappingList( Panel *panel, CUtlVector< PanelKeyBindingMap * >& maps )
|
|||
|
{
|
|||
|
PanelKeyBindingMap *map = panel->GetKBMap();
|
|||
|
while ( map )
|
|||
|
{
|
|||
|
maps.AddToTail( map );
|
|||
|
map = map->baseMap;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
static bool BindingLessFunc( KeyValues * const & lhs, KeyValues * const &rhs )
|
|||
|
{
|
|||
|
KeyValues *p1, *p2;
|
|||
|
|
|||
|
p1 = const_cast< KeyValues * >( lhs );
|
|||
|
p2 = const_cast< KeyValues * >( rhs );
|
|||
|
return ( Q_stricmp( p1->GetString( "Action" ), p2->GetString( "Action" ) ) < 0 ) ? true : false;
|
|||
|
}
|
|||
|
|
|||
|
void CKeyBoardEditorPage::AnsiText( char const *token, char *out, size_t buflen )
|
|||
|
{
|
|||
|
out[ 0 ] = 0;
|
|||
|
|
|||
|
wchar_t *str = g_pVGuiLocalize->Find( token );
|
|||
|
if ( !str )
|
|||
|
{
|
|||
|
Q_strncpy( out, token, buflen );
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
g_pVGuiLocalize->ConvertUnicodeToANSI( str, out, buflen );
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
void CKeyBoardEditorPage::PopulateList()
|
|||
|
{
|
|||
|
m_pList->DeleteAllItems();
|
|||
|
|
|||
|
int i, j;
|
|||
|
|
|||
|
CUtlRBTree< KeyValues *, int > sorted( 0, 0, BindingLessFunc );
|
|||
|
|
|||
|
// add header item
|
|||
|
int c = m_Save.Count();
|
|||
|
for ( i = 0; i < c; ++i )
|
|||
|
{
|
|||
|
SaveMapping_t* sm = m_Save[ i ];
|
|||
|
|
|||
|
PanelKeyBindingMap *m = sm->map;
|
|||
|
Assert( m );
|
|||
|
|
|||
|
int bindings = sm->current.Count();
|
|||
|
for ( j = 0; j < bindings; ++j )
|
|||
|
{
|
|||
|
BoundKey_t *kbMap = &sm->current[ j ];
|
|||
|
Assert( kbMap );
|
|||
|
|
|||
|
// Create a new: blank item
|
|||
|
KeyValues *item = new KeyValues( "Item" );
|
|||
|
|
|||
|
// Fill in data
|
|||
|
char loc[ 128 ];
|
|||
|
Q_snprintf( loc, sizeof( loc ), "#%s", kbMap->bindingname );
|
|||
|
|
|||
|
char ansi[ 256 ];
|
|||
|
AnsiText( loc, ansi, sizeof( ansi ) );
|
|||
|
|
|||
|
item->SetString( "Action", ansi );
|
|||
|
item->SetWString( "Binding", Panel::KeyCodeModifiersToDisplayString( (KeyCode)kbMap->keycode, kbMap->modifiers ) );
|
|||
|
|
|||
|
// Find the binding
|
|||
|
KeyBindingMap_t *bindingMap = m_pPanel->LookupBinding( kbMap->bindingname );
|
|||
|
if ( bindingMap &&
|
|||
|
bindingMap->helpstring )
|
|||
|
{
|
|||
|
AnsiText( bindingMap->helpstring, ansi, sizeof( ansi ) );
|
|||
|
item->SetString( "Description", ansi);
|
|||
|
}
|
|||
|
|
|||
|
item->SetPtr( "Item", kbMap );
|
|||
|
|
|||
|
sorted.Insert( item );
|
|||
|
}
|
|||
|
|
|||
|
// Now try and find any "unbound" keys...
|
|||
|
int mappings = m->entries.Count();
|
|||
|
for ( j = 0; j < mappings; ++j )
|
|||
|
{
|
|||
|
KeyBindingMap_t *kbMap = &m->entries[ j ];
|
|||
|
|
|||
|
// See if it's bound
|
|||
|
CUtlVector< BoundKey_t * > list;
|
|||
|
m_pPanel->LookupBoundKeys( kbMap->bindingname, list );
|
|||
|
if ( list.Count() > 0 )
|
|||
|
continue;
|
|||
|
|
|||
|
// Not bound, add a placeholder entry
|
|||
|
// Create a new: blank item
|
|||
|
KeyValues *item = new KeyValues( "Item" );
|
|||
|
|
|||
|
// fill in data
|
|||
|
char loc[ 128 ];
|
|||
|
Q_snprintf( loc, sizeof( loc ), "#%s", kbMap->bindingname );
|
|||
|
|
|||
|
char ansi[ 256 ];
|
|||
|
AnsiText( loc, ansi, sizeof( ansi ) );
|
|||
|
|
|||
|
item->SetString( "Action", ansi );
|
|||
|
item->SetWString( "Binding", L"" );
|
|||
|
if ( kbMap->helpstring )
|
|||
|
{
|
|||
|
AnsiText( kbMap->helpstring, ansi, sizeof( ansi ) );
|
|||
|
item->SetString( "Description", ansi );
|
|||
|
}
|
|||
|
|
|||
|
item->SetPtr( "Unbound", kbMap );
|
|||
|
|
|||
|
sorted.Insert( item );
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
for ( j = sorted.FirstInorder() ; j != sorted.InvalidIndex(); j = sorted.NextInorder( j ) )
|
|||
|
{
|
|||
|
KeyValues *item = sorted[ j ];
|
|||
|
|
|||
|
// Add to list
|
|||
|
m_pList->AddItem( item, 0, false, false );
|
|||
|
|
|||
|
item->deleteThis();
|
|||
|
}
|
|||
|
|
|||
|
sorted.RemoveAll();
|
|||
|
}
|
|||
|
|
|||
|
void CKeyBoardEditorPage::OnClearBinding( int item )
|
|||
|
{
|
|||
|
// Find item for this row
|
|||
|
KeyValues *kv = m_pList->GetItem(item );
|
|||
|
if ( !kv )
|
|||
|
{
|
|||
|
return;
|
|||
|
}
|
|||
|
|
|||
|
BoundKey_t *kbMap = reinterpret_cast< BoundKey_t * >( kv->GetPtr( "Item", 0 ) );
|
|||
|
if ( !kbMap )
|
|||
|
{
|
|||
|
return;
|
|||
|
}
|
|||
|
|
|||
|
kbMap->keycode = KEY_NONE;
|
|||
|
kbMap->modifiers = 0;
|
|||
|
|
|||
|
PopulateList();
|
|||
|
}
|
|||
|
|
|||
|
CKeyBoardEditorSheet::CKeyBoardEditorSheet( Panel *parent, Panel *panelToEdit, KeyBindingContextHandle_t handle )
|
|||
|
: BaseClass( parent, "KeyBoardEditorSheet" ),
|
|||
|
m_bSaveToExternalFile( false ),
|
|||
|
m_Handle( handle ),
|
|||
|
m_SaveFileName( UTL_INVAL_SYMBOL ),
|
|||
|
m_SaveFilePathID( UTL_INVAL_SYMBOL )
|
|||
|
{
|
|||
|
m_hPanel = panelToEdit;
|
|||
|
|
|||
|
SetSmallTabs( true );
|
|||
|
|
|||
|
// Create this sheet and add the subcontrols
|
|||
|
CKeyBoardEditorPage *active = NULL;
|
|||
|
|
|||
|
int subCount = Panel::GetPanelsWithKeyBindingsCount( handle );
|
|||
|
for ( int i = 0; i < subCount; ++i )
|
|||
|
{
|
|||
|
Panel *p = Panel::GetPanelWithKeyBindings( handle, i );
|
|||
|
if ( !p )
|
|||
|
continue;
|
|||
|
|
|||
|
// Don't display panels with no keymappings
|
|||
|
if ( p->GetKeyMappingCount() == 0 )
|
|||
|
continue;
|
|||
|
|
|||
|
CKeyBoardEditorPage *newPage = new CKeyBoardEditorPage( this, p, handle );
|
|||
|
AddPage( newPage, p->GetName() );
|
|||
|
if ( p == panelToEdit )
|
|||
|
{
|
|||
|
active = newPage;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
if ( active )
|
|||
|
{
|
|||
|
SetActivePage( active );
|
|||
|
}
|
|||
|
|
|||
|
LoadControlSettings( "resource/KeyBoardEditorSheet.res" );
|
|||
|
}
|
|||
|
|
|||
|
void CKeyBoardEditorSheet::SetKeybindingsSaveFile( char const *filename, char const *pathID /*= 0*/ )
|
|||
|
{
|
|||
|
Assert( filename );
|
|||
|
m_bSaveToExternalFile = true;
|
|||
|
m_SaveFileName = filename;
|
|||
|
if ( pathID != NULL )
|
|||
|
{
|
|||
|
m_SaveFilePathID = pathID;
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
m_SaveFilePathID = UTL_INVAL_SYMBOL;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
void CKeyBoardEditorSheet::OnSaveChanges()
|
|||
|
{
|
|||
|
int c = GetNumPages();
|
|||
|
for ( int i = 0 ; i < c; ++i )
|
|||
|
{
|
|||
|
CKeyBoardEditorPage *page = static_cast< CKeyBoardEditorPage * >( GetPage( i ) );
|
|||
|
page->OnSaveChanges();
|
|||
|
}
|
|||
|
|
|||
|
if ( m_bSaveToExternalFile )
|
|||
|
{
|
|||
|
m_hPanel->SaveKeyBindingsToFile( m_Handle, m_SaveFileName.String(), m_SaveFilePathID.IsValid() ? m_SaveFilePathID.String() : NULL );
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
m_hPanel->SaveKeyBindings( m_Handle );
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
void CKeyBoardEditorSheet::OnRevert()
|
|||
|
{
|
|||
|
int c = GetNumPages();
|
|||
|
for ( int i = 0 ; i < c; ++i )
|
|||
|
{
|
|||
|
CKeyBoardEditorPage *page = static_cast< CKeyBoardEditorPage * >( GetPage( i ) );
|
|||
|
page->OnRevert();
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
void CKeyBoardEditorSheet::OnUseDefaults()
|
|||
|
{
|
|||
|
int c = GetNumPages();
|
|||
|
for ( int i = 0 ; i < c; ++i )
|
|||
|
{
|
|||
|
CKeyBoardEditorPage *page = static_cast< CKeyBoardEditorPage * >( GetPage( i ) );
|
|||
|
page->OnUseDefaults();
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
CKeyBoardEditorDialog::CKeyBoardEditorDialog( Panel *parent, Panel *panelToEdit, KeyBindingContextHandle_t handle )
|
|||
|
: BaseClass( parent, "KeyBoardEditorDialog" )
|
|||
|
{
|
|||
|
m_pSave = new Button( this, "Save", "#KBEditorSave", this, "save" );
|
|||
|
m_pCancel = new Button( this, "Cancel", "#KBEditorCancel", this, "cancel" );
|
|||
|
m_pRevert = new Button( this, "Revert", "#KBEditorRevert", this, "revert" );
|
|||
|
m_pUseDefaults = new Button( this, "Defaults", "#KBEditorUseDefaults", this, "defaults" );
|
|||
|
|
|||
|
m_pKBEditor = new CKeyBoardEditorSheet( this, panelToEdit, handle );
|
|||
|
|
|||
|
LoadControlSettings( "resource/KeyBoardEditorDialog.res" );
|
|||
|
|
|||
|
SetTitle( "#KBEditorTitle", true );
|
|||
|
|
|||
|
SetSmallCaption( true );
|
|||
|
SetMinimumSize( 640, 200 );
|
|||
|
SetMinimizeButtonVisible( false );
|
|||
|
SetMaximizeButtonVisible( false );
|
|||
|
SetSizeable( true );
|
|||
|
SetMoveable( true );
|
|||
|
SetMenuButtonVisible( false );
|
|||
|
|
|||
|
SetVisible( true );
|
|||
|
|
|||
|
MoveToCenterOfScreen();
|
|||
|
}
|
|||
|
|
|||
|
void CKeyBoardEditorDialog::OnCommand( char const *cmd )
|
|||
|
{
|
|||
|
if ( !Q_stricmp( cmd, "save" ) )
|
|||
|
{
|
|||
|
m_pKBEditor->OnSaveChanges();
|
|||
|
MarkForDeletion();
|
|||
|
}
|
|||
|
else if ( !Q_stricmp( cmd, "cancel" ) ||
|
|||
|
!Q_stricmp( cmd, "Close" ) )
|
|||
|
{
|
|||
|
m_pKBEditor->OnRevert();
|
|||
|
MarkForDeletion();
|
|||
|
}
|
|||
|
else if ( !Q_stricmp( cmd, "revert" ) )
|
|||
|
{
|
|||
|
m_pKBEditor->OnRevert();
|
|||
|
}
|
|||
|
else if ( !Q_stricmp( cmd, "defaults" ) )
|
|||
|
{
|
|||
|
m_pKBEditor->OnUseDefaults();
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
BaseClass::OnCommand( cmd );
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
void CKeyBoardEditorDialog::SetKeybindingsSaveFile( char const *filename, char const *pathID /*= 0*/ )
|
|||
|
{
|
|||
|
m_pKBEditor->SetKeybindingsSaveFile( filename, pathID );
|
|||
|
}
|