Counter Strike : Global Offensive Source Code
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

1208 lines
28 KiB

  1. //========= Copyright � 1996-2005, Valve Corporation, All rights reserved. ============//
  2. //
  3. // Purpose:
  4. //
  5. // $NoKeywords: $
  6. //=============================================================================//
  7. #include "hlfaceposer.h"
  8. #include <windows.h>
  9. #include <stdio.h>
  10. #include <mxtk/mxWindow.h>
  11. #include <mxtk/mxScrollBar.h>
  12. #include "mxexpressiontray.h"
  13. #include "expressions.h"
  14. #include "expclass.h"
  15. #include "ControlPanel.h"
  16. #include "FlexPanel.h"
  17. #include <mxtk/mxPopupMenu.h>
  18. #include "ChoreoView.h"
  19. #include "StudioModel.h"
  20. #include "ExpressionProperties.h"
  21. #include "InputProperties.h"
  22. #include "ViewerSettings.h"
  23. #include "mxExpressionTab.h"
  24. #include "choreowidgetdrawhelper.h"
  25. #include "ExpressionTool.h"
  26. #include "faceposer_models.h"
  27. #include "tier0/icommandline.h"
  28. #include "FileSystem.h"
  29. #define MAX_THUMBNAILSIZE 256
  30. #define MIN_THUMBNAILSIZE 64
  31. #define THUMBNAIL_SIZE_STEP 4
  32. #define DEFAULT_THUMBNAIL_SIZE 128
  33. #define TOP_GAP 45
  34. mxExpressionTray *g_pExpressionTrayTool = 0;
  35. mxExpressionTray::mxExpressionTray( mxWindow *parent, int id /*=0*/ )
  36. : IFacePoserToolWindow( "ExpressionTrayTool", "Expressions" ), mxWindow( parent, 0, 0, 0, 0, "ExpressionTrayTool", id )
  37. {
  38. setId( id );
  39. m_nTopOffset = 0;
  40. slScrollbar = new mxScrollbar( this, 0, 0, 18, 100, IDC_TRAYSCROLL, mxScrollbar::Vertical );
  41. m_nLastNumExpressions = -1;
  42. m_nGranularity = 10;
  43. m_nPrevCell = -1;
  44. m_nCurCell = -1;
  45. m_nClickedCell = -1;
  46. m_nButtonSquare = 16;
  47. m_nGap = 4;
  48. m_nDescriptionHeight = 34;
  49. m_nSnapshotWidth = g_viewerSettings.thumbnailsize;
  50. m_nSnapshotWidth = max( MIN_THUMBNAILSIZE, m_nSnapshotWidth );
  51. m_nSnapshotWidth = min( MAX_THUMBNAILSIZE, m_nSnapshotWidth );
  52. g_viewerSettings.thumbnailsize = m_nSnapshotWidth;
  53. m_nSnapshotHeight = m_nSnapshotWidth + m_nDescriptionHeight;
  54. m_pButtons = NULL;
  55. m_nPreviousExpressionCount = -1;
  56. m_bDragging = false;
  57. m_nDragCell = -1;
  58. CreateButtons();
  59. g_pExpressionClass = new mxExpressionTab( this, 5, 5, 500, 20, IDC_EXPRESSIONCLASS );
  60. m_pABButton = new mxButton( this, 520, 8, 50, 18, "A/B", IDC_AB );
  61. m_pThumbnailIncreaseButton = new mxButton( this, 0, 0, 18, 18, "+", IDC_THUMBNAIL_INCREASE );
  62. m_pThumbnailDecreaseButton = new mxButton( this, 0, 0, 18, 18, "-", IDC_THUMBNAIL_DECREASE );
  63. }
  64. //-----------------------------------------------------------------------------
  65. // Purpose:
  66. // Output : mxExpressionTray::~mxExpressionTray
  67. //-----------------------------------------------------------------------------
  68. mxExpressionTray::~mxExpressionTray ( void )
  69. {
  70. DeleteAllButtons();
  71. g_pExpressionTrayTool = NULL;
  72. }
  73. //-----------------------------------------------------------------------------
  74. // Purpose:
  75. // Input : cellsize -
  76. //-----------------------------------------------------------------------------
  77. void mxExpressionTray::SetCellSize( int cellsize )
  78. {
  79. m_nSnapshotWidth = cellsize;
  80. m_nSnapshotHeight = cellsize + m_nDescriptionHeight;
  81. DeleteAllButtons();
  82. CreateButtons();
  83. redraw();
  84. }
  85. //-----------------------------------------------------------------------------
  86. // Purpose:
  87. //-----------------------------------------------------------------------------
  88. void mxExpressionTray::Deselect( void )
  89. {
  90. CExpClass *active = expressions->GetActiveClass();
  91. if ( active )
  92. {
  93. for ( int i = 0 ; i < active->GetNumExpressions(); i++ )
  94. {
  95. CExpression *exp = active->GetExpression( i );
  96. if ( exp )
  97. {
  98. exp->SetSelected( false );
  99. }
  100. }
  101. }
  102. m_nCurCell = m_nPrevCell = -1;
  103. redraw();
  104. }
  105. //-----------------------------------------------------------------------------
  106. // Purpose:
  107. // Input : exp -
  108. //-----------------------------------------------------------------------------
  109. void mxExpressionTray::Select( int exp, bool deselect /*=true*/ )
  110. {
  111. int oldcell = m_nCurCell;
  112. if ( deselect )
  113. {
  114. Deselect();
  115. }
  116. m_nPrevCell = oldcell;
  117. m_nCurCell = exp;
  118. if ( m_nCurCell >= 0 )
  119. {
  120. CExpClass *active = expressions->GetActiveClass();
  121. if ( active )
  122. {
  123. CExpression *exp = active->GetExpression( m_nCurCell );
  124. if ( exp )
  125. {
  126. exp->SetSelected( true );
  127. }
  128. }
  129. }
  130. redraw();
  131. }
  132. //-----------------------------------------------------------------------------
  133. // Purpose:
  134. // Input : *btn -
  135. //-----------------------------------------------------------------------------
  136. void mxExpressionTray::AddButton( const char *name, const char *tooltip, const char *bitmap, ETMEMBERFUNC pfnCallback,
  137. bool active, int x, int y, int w, int h )
  138. {
  139. mxETButton *btn = new mxETButton;
  140. strcpy( btn->m_szName, name );
  141. strcpy( btn->m_szToolTip, tooltip );
  142. btn->m_bActive = active;
  143. btn->m_rc.left = x;
  144. btn->m_rc.top = y;
  145. btn->m_rc.right = x + w;
  146. btn->m_rc.bottom = y + h;
  147. btn->m_pImage = new mxbitmapdata_t;
  148. Assert( btn->m_pImage );
  149. btn->m_pImage->valid = false;
  150. LoadBitmapFromFile( bitmap, *btn->m_pImage );
  151. btn->m_fnCallback = pfnCallback;
  152. btn->next = m_pButtons;
  153. m_pButtons = btn;
  154. }
  155. //-----------------------------------------------------------------------------
  156. // Purpose:
  157. //-----------------------------------------------------------------------------
  158. void mxExpressionTray::CreateButtons( void )
  159. {
  160. int x = m_nSnapshotWidth - 2 * ( m_nButtonSquare + 4 );
  161. int y = 4;
  162. AddButton( "undo", "Undo", "gfx/hlfaceposer/undo.bmp", &mxExpressionTray::ET_Undo, true, x, y, m_nButtonSquare, m_nButtonSquare );
  163. x += ( m_nButtonSquare + 4 );
  164. AddButton( "redo", "Redo", "gfx/hlfaceposer/redo.bmp", &mxExpressionTray::ET_Redo, true, x, y, m_nButtonSquare, m_nButtonSquare );
  165. }
  166. void mxExpressionTray::ActivateButton( const char *name, bool active )
  167. {
  168. mxETButton *btn = FindButton( name );
  169. if ( !name )
  170. return;
  171. btn->m_bActive = active;
  172. }
  173. mxExpressionTray::mxETButton *mxExpressionTray::FindButton( const char *name )
  174. {
  175. mxETButton *p = m_pButtons;
  176. while ( p )
  177. {
  178. if ( !stricmp( p->m_szName, name ) )
  179. return p;
  180. p = p->next;
  181. }
  182. return NULL;
  183. }
  184. //-----------------------------------------------------------------------------
  185. // Purpose:
  186. //-----------------------------------------------------------------------------
  187. void mxExpressionTray::DeleteAllButtons( void )
  188. {
  189. mxETButton *p = m_pButtons, *n;
  190. while ( p )
  191. {
  192. n = p->next;
  193. delete p->m_pImage;
  194. delete p;
  195. p = n;
  196. }
  197. m_pButtons = NULL;
  198. }
  199. //-----------------------------------------------------------------------------
  200. // Purpose:
  201. // Input : x -
  202. // y -
  203. // Output : mxExpressionTray::mxETButton
  204. //-----------------------------------------------------------------------------
  205. mxExpressionTray::mxETButton *mxExpressionTray::GetItemUnderCursor( int x, int y )
  206. {
  207. // Convert to cell space
  208. int cell = GetCellUnderPosition( x, y );
  209. if ( cell == -1 )
  210. {
  211. return NULL;
  212. }
  213. // Cell is off screen?
  214. int cx, cy, cw, ch;
  215. if ( !ComputeRect( cell, cx, cy, cw, ch ) )
  216. {
  217. return NULL;
  218. }
  219. mxETButton *p = m_pButtons;
  220. while ( p )
  221. {
  222. if ( p->m_bActive &&
  223. x >= cx &&
  224. x <= cx + cw &&
  225. y >= cy &&
  226. y <= cy + ch )
  227. {
  228. // In-side cell
  229. int cellx = x - cx;
  230. int celly = y - cy;
  231. if ( cellx >= p->m_rc.left &&
  232. cellx <= p->m_rc.right &&
  233. celly >= p->m_rc.top &&
  234. celly <= p->m_rc.bottom )
  235. {
  236. return p;
  237. }
  238. }
  239. p = p->next;
  240. }
  241. return NULL;
  242. }
  243. void mxExpressionTray::DrawButton( CChoreoWidgetDrawHelper& helper, int cell, mxETButton *btn )
  244. {
  245. if ( !btn || !btn->m_pImage || !btn->m_pImage->valid )
  246. return;
  247. if ( !btn->m_bActive )
  248. return;
  249. int x, y, w, h;
  250. if ( !ComputeRect( cell, x, y, w, h ) )
  251. return;
  252. x += btn->m_rc.left;
  253. y += btn->m_rc.top;
  254. w = btn->m_rc.right - btn->m_rc.left;
  255. h = btn->m_rc.bottom - btn->m_rc.top;
  256. HDC dc = helper.GrabDC();
  257. DrawBitmapToDC( dc, x, y, w, h, *btn->m_pImage );
  258. helper.DrawOutlinedRect( Color( 170, 170, 170 ), PS_SOLID, 1, x, y, x + w, y + h );
  259. }
  260. //-----------------------------------------------------------------------------
  261. // Purpose:
  262. // Output : int
  263. //-----------------------------------------------------------------------------
  264. int mxExpressionTray::ComputePixelsNeeded( void )
  265. {
  266. CExpClass *active = expressions->GetActiveClass();
  267. if ( !active )
  268. return 100;
  269. // Remove scroll bar
  270. int w = this->w2() - 16;
  271. int colsperrow;
  272. colsperrow = ( w - m_nGap ) / ( m_nSnapshotWidth + m_nGap );
  273. // At least one
  274. colsperrow = max( 1, colsperrow );
  275. int rowsneeded = ( ( active->GetNumExpressions() + colsperrow - 1 ) / colsperrow );
  276. return rowsneeded * ( m_nSnapshotHeight + m_nGap ) + m_nGap + TOP_GAP + GetCaptionHeight();
  277. }
  278. bool mxExpressionTray::ComputeRect( int cell, int& rcx, int& rcy, int& rcw, int& rch )
  279. {
  280. // Remove scroll bar
  281. int w = this->w2() - 16;
  282. int colsperrow;
  283. colsperrow = ( w - m_nGap ) / ( m_nSnapshotWidth + m_nGap );
  284. // At least one
  285. colsperrow = max( 1, colsperrow );
  286. int row, col;
  287. row = cell / colsperrow;
  288. col = cell % colsperrow;
  289. // don't allow partial columns
  290. rcx = m_nGap + col * ( m_nSnapshotWidth + m_nGap );
  291. rcy = GetCaptionHeight() + TOP_GAP + ( -m_nTopOffset * m_nGranularity ) + m_nGap + row * ( m_nSnapshotHeight + m_nGap );
  292. // Starts off screen
  293. if ( rcx < 0 )
  294. return false;
  295. // Ends off screen
  296. if ( rcx + m_nSnapshotWidth + m_nGap > this->w2() )
  297. return false;
  298. // Allow partial in y direction
  299. if ( rcy > this->h2() )
  300. return false;
  301. if ( rcy + m_nSnapshotHeight + m_nGap < 0 )
  302. return false;
  303. // Some portion is onscreen
  304. rcw = m_nSnapshotWidth;
  305. rch = m_nSnapshotHeight;
  306. return true;
  307. }
  308. void mxExpressionTray::DrawExpressionFocusRect( CChoreoWidgetDrawHelper& helper, int x, int y, int w, int h, const Color& clr )
  309. {
  310. helper.DrawOutlinedRect( clr, PS_SOLID, 4, x, y, x + w, y + h );
  311. }
  312. void mxExpressionTray::DrawExpressionDescription( CChoreoWidgetDrawHelper& helper, int x, int y, int w, int h, const char *expressionname, const char *description )
  313. {
  314. int textheight = 15;
  315. RECT textRect;
  316. textRect.left = x + 5;
  317. textRect.top = y + h - 2 * textheight - 12;
  318. textRect.right = x + w - 10;
  319. textRect.bottom = y + h - 12;
  320. helper.DrawColoredText( "Arial", 9, FW_NORMAL, Color( 0, 0, 0 ), textRect, "%s", expressionname );
  321. // DrawText( hdc, expressionname, strlen( expressionname ), &textRect, DT_NOPREFIX | DT_SINGLELINE | DT_LEFT | DT_VCENTER | DT_WORD_ELLIPSIS );
  322. OffsetRect( &textRect, 0, textheight );
  323. helper.DrawColoredText( "Arial", 9, FW_NORMAL, Color( 63, 63, 63 ), textRect, "%s", description );
  324. // DrawText( hdc, description, strlen( description ), &textRect, DT_NOPREFIX | DT_SINGLELINE | DT_LEFT | DT_VCENTER | DT_WORD_ELLIPSIS );
  325. }
  326. //-----------------------------------------------------------------------------
  327. // Purpose:
  328. // Input : dc -
  329. // current -
  330. // rcx -
  331. // rcy -
  332. // rcw -
  333. // rch -
  334. //-----------------------------------------------------------------------------
  335. void mxExpressionTray::DrawDirtyFlag( CChoreoWidgetDrawHelper& helper, CExpression *current, int rcx, int rcy, int rcw, int rch )
  336. {
  337. // Not dirty
  338. if ( !current || ( !current->CanUndo() && !current->GetDirty() ) )
  339. return;
  340. int fontsize = 14;
  341. RECT textRect;
  342. textRect.left = rcx + 5;
  343. textRect.right = rcx + rcw;
  344. textRect.top = rcy + 5;
  345. textRect.bottom = textRect.top + fontsize + 2;
  346. helper.DrawColoredText( "Arial", fontsize, FW_NORMAL, Color( 100, 240, 255 ), textRect, "*" );
  347. }
  348. bool mxExpressionTray::PaintBackground( void )
  349. {
  350. redraw();
  351. return false;
  352. }
  353. void mxExpressionTray::DrawThumbNail( CExpClass *active, CExpression *current, CChoreoWidgetDrawHelper& helper, int rcx, int rcy, int rcw, int rch, int c, int selected, bool updateselection )
  354. {
  355. if ( !current )
  356. return;
  357. HDC dc = helper.GrabDC();
  358. helper.DrawFilledRect( RGBToColor( GetSysColor( COLOR_BTNFACE ) ), rcx, rcy, rcw + rcx, rch + rcy );
  359. if ( current->m_Bitmap[ models->GetActiveModelIndex() ].valid )
  360. {
  361. DrawBitmapToDC( dc, rcx, rcy, rcw, rch - m_nDescriptionHeight, current->m_Bitmap[ models->GetActiveModelIndex() ] );
  362. helper.DrawOutlinedRect( Color( 127, 127, 127 ), PS_SOLID, 1, rcx, rcy, rcx + rcw, rcy + rch - m_nDescriptionHeight );
  363. }
  364. DrawDirtyFlag( helper, current, rcx, rcy, rcw, rch );
  365. DrawExpressionDescription( helper, rcx, rcy, rcw, rch, current->name, current->description );
  366. if ( c == selected )
  367. {
  368. DrawExpressionFocusRect( helper, rcx, rcy, rcw, rch - m_nDescriptionHeight, Color( 255, 100, 63 ) );
  369. if ( updateselection )
  370. {
  371. m_nPrevCell = -1;
  372. m_nCurCell = c;
  373. }
  374. if ( current->CanUndo() || current->CanRedo() )
  375. {
  376. if ( current->CanUndo() )
  377. {
  378. DrawButton( helper, c, FindButton( "undo" ) );
  379. }
  380. if ( current->CanRedo() )
  381. {
  382. DrawButton( helper, c, FindButton( "redo" ) );
  383. }
  384. RECT rc;
  385. rc.left = rcx + rcw - 2 * ( m_nButtonSquare + 4 );
  386. rc.top = rcy + m_nButtonSquare + 6;
  387. rc.right = rc.left + 2 * ( m_nButtonSquare + 4 );
  388. rc.bottom = rc.top + 15;
  389. helper.DrawColoredText( "Arial", 9, FW_NORMAL, Color( 200, 200, 200 ), rc,
  390. "%i/%i", current->UndoCurrent(), current->UndoLevels() );
  391. }
  392. }
  393. else
  394. {
  395. if ( current->GetSelected() )
  396. {
  397. DrawExpressionFocusRect( helper, rcx, rcy, rcw, rch - m_nDescriptionHeight, Color( 127, 127, 220 ) );
  398. }
  399. }
  400. }
  401. void mxExpressionTray::redraw()
  402. {
  403. if ( !ToolCanDraw() )
  404. return;
  405. bool updateSelection = false;
  406. CExpClass *active = expressions->GetActiveClass();
  407. if ( active && active->GetNumExpressions() != m_nPreviousExpressionCount )
  408. {
  409. m_nTopOffset = 0;
  410. RepositionSlider();
  411. m_nPreviousExpressionCount = active->GetNumExpressions();
  412. }
  413. CChoreoWidgetDrawHelper helper( this, RGBToColor( GetSysColor( COLOR_BTNFACE ) ) );
  414. HandleToolRedraw( helper );
  415. int w, h;
  416. w = w2();
  417. h = h2();
  418. if ( active )
  419. {
  420. RECT clipRect;
  421. helper.GetClientRect( clipRect );
  422. clipRect.top += TOP_GAP + GetCaptionHeight();
  423. helper.StartClipping( clipRect );
  424. if ( m_nLastNumExpressions != active->GetNumExpressions() )
  425. {
  426. m_nTopOffset = 0;
  427. m_nLastNumExpressions = active->GetNumExpressions();
  428. RepositionSlider();
  429. updateSelection = true;
  430. }
  431. int selected = active->GetSelectedExpression();
  432. int rcx, rcy, rcw, rch;
  433. int c = 0;
  434. while ( c < active->GetNumExpressions() )
  435. {
  436. if ( !ComputeRect( c, rcx, rcy, rcw, rch ) )
  437. {
  438. c++;
  439. continue;
  440. }
  441. CExpression *current = active->GetExpression( c );
  442. if ( !current )
  443. break;
  444. DrawThumbNail( active, current, helper, rcx, rcy, rcw, rch, c, selected, updateSelection );
  445. c++;
  446. }
  447. helper.StopClipping();
  448. }
  449. else
  450. {
  451. RECT rc;
  452. helper.GetClientRect( rc );
  453. // Arial 36 normal
  454. char sz[ 256 ];
  455. sprintf( sz, "No expression file loaded" );
  456. int pointsize = 18;
  457. int textlen = helper.CalcTextWidth( "Arial", pointsize, FW_NORMAL, sz );
  458. RECT rcText;
  459. rcText.top = ( rc.bottom - rc.top ) / 2 - pointsize / 2;
  460. rcText.bottom = rcText.top + pointsize + 10;
  461. int fullw = rc.right - rc.left;
  462. rcText.left = rc.left + ( fullw - textlen ) / 2;
  463. rcText.right = rcText.left + textlen;
  464. helper.DrawColoredText( "Arial", pointsize, FW_NORMAL, Color( 80, 80, 80 ), rcText, sz );
  465. }
  466. }
  467. int mxExpressionTray::GetCellUnderPosition( int x, int y )
  468. {
  469. CExpClass *active = expressions->GetActiveClass();
  470. if ( !active )
  471. return -1;
  472. int rcx, rcy, rcw, rch;
  473. int c = 0;
  474. while ( c < active->GetNumExpressions() )
  475. {
  476. if ( !ComputeRect( c, rcx, rcy, rcw, rch ) )
  477. {
  478. c++;
  479. continue;
  480. }
  481. if ( x >= rcx && x <= rcx + rcw &&
  482. y >= rcy && y <= rcy + rch )
  483. {
  484. return c;
  485. }
  486. c++;
  487. }
  488. return -1;
  489. }
  490. void mxExpressionTray::RepositionSlider( void )
  491. {
  492. int trueh = h2() - GetCaptionHeight();
  493. int heightpixels = trueh / m_nGranularity;
  494. int rangepixels = ComputePixelsNeeded() / m_nGranularity;
  495. if ( rangepixels < heightpixels )
  496. {
  497. m_nTopOffset = 0;
  498. slScrollbar->setVisible( false );
  499. }
  500. else
  501. {
  502. slScrollbar->setVisible( true );
  503. }
  504. slScrollbar->setBounds( w2() - 16, GetCaptionHeight() + TOP_GAP, 16, trueh - TOP_GAP );
  505. m_nTopOffset = max( 0, m_nTopOffset );
  506. m_nTopOffset = min( rangepixels, m_nTopOffset );
  507. slScrollbar->setRange( 0, rangepixels );
  508. slScrollbar->setValue( m_nTopOffset );
  509. slScrollbar->setPagesize( heightpixels );
  510. }
  511. void mxExpressionTray::AB( void )
  512. {
  513. if ( m_nPrevCell == -1 && m_nCurCell == -1 )
  514. return;
  515. CExpClass *active = expressions->GetActiveClass();
  516. if ( !active )
  517. return;
  518. if ( m_nPrevCell >= 0 && m_nPrevCell < active->GetNumExpressions() )
  519. {
  520. active->SelectExpression( m_nPrevCell );
  521. }
  522. }
  523. int mxExpressionTray::CountSelected( void )
  524. {
  525. CExpClass *active = expressions->GetActiveClass();
  526. if ( !active )
  527. return 0;
  528. int c = 0;
  529. for ( int i = 0; i < active->GetNumExpressions(); i++ )
  530. {
  531. CExpression *exp = active->GetExpression( i );
  532. if ( !exp )
  533. continue;
  534. if ( exp->GetSelected() )
  535. {
  536. c++;
  537. }
  538. }
  539. return c;
  540. }
  541. void mxExpressionTray::SetClickedCell( int cell )
  542. {
  543. m_nClickedCell = cell;
  544. }
  545. void mxExpressionTray::ShowRightClickMenu( int mx, int my )
  546. {
  547. CExpClass *active = expressions->GetActiveClass();
  548. if ( !active )
  549. return;
  550. mxPopupMenu *pop = new mxPopupMenu();
  551. Assert( pop );
  552. CExpression *exp = NULL;
  553. if ( m_nClickedCell != -1 )
  554. {
  555. exp = active->GetExpression( m_nClickedCell );
  556. }
  557. pop->add( "New Expression...", IDC_CONTEXT_NEWEXP );
  558. if ( exp )
  559. {
  560. pop->addSeparator();
  561. pop->add( va( "Edit '%s'...", exp->name ), IDC_CONTEXT_EDITEXP );
  562. pop->add( va( "Save '%s'", exp->name ), IDC_CONTEXT_SAVEEXP );
  563. if ( exp->CanUndo() || exp->CanRedo() )
  564. {
  565. pop->add( va( "Revert '%s'", exp->name ), IDC_CONTEXT_REVERT );
  566. }
  567. pop->addSeparator();
  568. pop->add( va( "Delete '%s'", exp->name ), IDC_CONTEXT_DELETEXP );
  569. pop->addSeparator();
  570. pop->add( va( "Re-create thumbnail for '%s'", exp->name ), IDC_CONTEXT_CREATEBITMAP );
  571. }
  572. pop->popup( this, mx, my );
  573. }
  574. //-----------------------------------------------------------------------------
  575. // Purpose:
  576. //-----------------------------------------------------------------------------
  577. void mxExpressionTray::DrawFocusRect( void )
  578. {
  579. HDC dc = GetDC( NULL );
  580. ::DrawFocusRect( dc, &m_rcFocus );
  581. ReleaseDC( NULL, dc );
  582. }
  583. static bool IsWindowOrChild( mxWindow *parent, HWND test )
  584. {
  585. HWND parentHwnd = (HWND)parent->getHandle();
  586. if ( test == parentHwnd ||
  587. IsChild( parentHwnd, test ) )
  588. {
  589. return true;
  590. }
  591. return false;
  592. }
  593. int mxExpressionTray::handleEvent (mxEvent *event)
  594. {
  595. MDLCACHE_CRITICAL_SECTION_( g_pMDLCache );
  596. int iret = 0;
  597. if ( HandleToolEvent( event ) )
  598. {
  599. return iret;
  600. }
  601. switch ( event->event )
  602. {
  603. case mxEvent::Action:
  604. {
  605. iret = 1;
  606. switch ( event->action )
  607. {
  608. default:
  609. iret = 0;
  610. break;
  611. case IDC_EXPRESSIONCLASS:
  612. {
  613. int index = g_pExpressionClass->getSelectedIndex();
  614. if ( index >= 0 )
  615. {
  616. CExpClass *current = expressions->GetClass( index );
  617. if ( current )
  618. {
  619. // Switch classname
  620. expressions->ActivateExpressionClass( current );
  621. current->SelectExpression( 0 );
  622. }
  623. }
  624. }
  625. break;
  626. case IDC_CONTEXT_NEWEXP:
  627. g_pFlexPanel->NewExpression();
  628. break;
  629. case IDC_CONTEXT_EDITEXP:
  630. if ( m_nClickedCell != -1 )
  631. {
  632. g_pFlexPanel->EditExpression();
  633. }
  634. break;
  635. case IDC_CONTEXT_REVERT:
  636. if ( m_nClickedCell != -1 )
  637. {
  638. g_pFlexPanel->RevertExpression( m_nClickedCell );
  639. }
  640. break;
  641. case IDC_CONTEXT_SAVEEXP:
  642. if ( m_nClickedCell != -1 )
  643. {
  644. g_pFlexPanel->SaveExpression( m_nClickedCell );
  645. }
  646. break;
  647. case IDC_CONTEXT_DELETEXP:
  648. if ( m_nClickedCell != -1 )
  649. {
  650. g_pControlPanel->DeleteExpression( m_nClickedCell );
  651. }
  652. break;
  653. case IDC_TRAYSCROLL:
  654. {
  655. if (event->modifiers == SB_THUMBTRACK)
  656. {
  657. int offset = event->height;
  658. slScrollbar->setValue( offset );
  659. m_nTopOffset = offset;
  660. redraw();
  661. }
  662. else if ( event->modifiers == SB_PAGEUP )
  663. {
  664. int offset = slScrollbar->getValue();
  665. offset -= m_nGranularity;
  666. offset = max( offset, slScrollbar->getMinValue() );
  667. slScrollbar->setValue( offset );
  668. InvalidateRect( (HWND)slScrollbar->getHandle(), NULL, TRUE );
  669. m_nTopOffset = offset;
  670. redraw();
  671. }
  672. else if ( event->modifiers == SB_PAGEDOWN )
  673. {
  674. int offset = slScrollbar->getValue();
  675. offset += m_nGranularity;
  676. offset = min( offset, slScrollbar->getMaxValue() );
  677. slScrollbar->setValue( offset );
  678. InvalidateRect( (HWND)slScrollbar->getHandle(), NULL, TRUE );
  679. m_nTopOffset = offset;
  680. redraw();
  681. }
  682. }
  683. break;
  684. case IDC_AB:
  685. {
  686. AB();
  687. }
  688. break;
  689. case IDC_THUMBNAIL_INCREASE:
  690. {
  691. ThumbnailIncrease();
  692. }
  693. break;
  694. case IDC_THUMBNAIL_DECREASE:
  695. {
  696. ThumbnailDecrease();
  697. }
  698. break;
  699. case IDC_CONTEXT_CREATEBITMAP:
  700. {
  701. if ( m_nClickedCell >= 0 )
  702. {
  703. CExpClass *active = expressions->GetActiveClass();
  704. if ( active )
  705. {
  706. CExpression *exp = active->GetExpression( m_nClickedCell );
  707. if ( exp )
  708. {
  709. active->SelectExpression( m_nClickedCell );
  710. exp->CreateNewBitmap( models->GetActiveModelIndex() );
  711. redraw();
  712. }
  713. }
  714. }
  715. }
  716. break;
  717. }
  718. break;
  719. }
  720. case mxEvent::MouseDown:
  721. {
  722. if ( !( event->buttons & mxEvent::MouseRightButton ) )
  723. {
  724. // Figure out cell #
  725. int cell = GetCellUnderPosition( event->x, event->y );
  726. CExpClass *active = expressions->GetActiveClass();
  727. if ( active )
  728. {
  729. if ( cell == m_nCurCell && cell >= 0 && cell < active->GetNumExpressions() )
  730. {
  731. mxETButton *btn = GetItemUnderCursor( event->x, event->y );
  732. if ( btn && btn->m_fnCallback )
  733. {
  734. (this->*(btn->m_fnCallback))( cell );
  735. return iret;
  736. }
  737. }
  738. if ( cell >= 0 && cell < active->GetNumExpressions() )
  739. {
  740. active->SelectExpression( cell, event->modifiers & mxEvent::KeyShift ? false : true );
  741. int cx, cy, cw, ch;
  742. if ( ComputeRect( cell, cx, cy, cw, ch ) )
  743. {
  744. m_bDragging = true;
  745. m_nDragCell = cell;
  746. m_nXStart = (short)event->x;
  747. m_nYStart = (short)event->y;
  748. m_rcFocus.left = cx;
  749. m_rcFocus.top = cy;
  750. m_rcFocus.right = cx + cw;
  751. m_rcFocus.bottom = cy + ch - m_nDescriptionHeight;
  752. POINT pt;
  753. pt.x = pt.y = 0;
  754. ClientToScreen( (HWND)getHandle(), &pt );
  755. OffsetRect( &m_rcFocus, pt.x, pt.y );
  756. m_rcOrig = m_rcFocus;
  757. DrawFocusRect();
  758. }
  759. }
  760. else
  761. {
  762. Deselect();
  763. active->DeselectExpression();
  764. redraw();
  765. }
  766. }
  767. }
  768. iret = 1;
  769. }
  770. break;
  771. case mxEvent::MouseDrag:
  772. {
  773. if ( m_bDragging )
  774. {
  775. // Draw drag line of some kind
  776. DrawFocusRect();
  777. // update pos
  778. m_rcFocus = m_rcOrig;
  779. OffsetRect( &m_rcFocus, ( (short)event->x - m_nXStart ),
  780. ( (short)event->y - m_nYStart ) );
  781. DrawFocusRect();
  782. }
  783. iret = 1;
  784. }
  785. break;
  786. case mxEvent::MouseUp:
  787. {
  788. iret = 1;
  789. if ( event->buttons & mxEvent::MouseRightButton )
  790. {
  791. SetClickedCell( GetCellUnderPosition( (short)event->x, (short)event->y ) );
  792. ShowRightClickMenu( (short)event->x, (short)event->y );
  793. return iret;
  794. }
  795. int cell = GetCellUnderPosition( event->x, event->y );
  796. CExpClass *active = expressions->GetActiveClass();
  797. if ( m_bDragging )
  798. {
  799. DrawFocusRect();
  800. m_bDragging = false;
  801. // See if we let go on top of the choreo view
  802. if ( active )
  803. {
  804. // Convert x, y to screen space
  805. POINT pt;
  806. pt.x = (short)event->x;
  807. pt.y = (short)event->y;
  808. ClientToScreen( (HWND)getHandle(), &pt );
  809. HWND maybeTool = WindowFromPoint( pt );
  810. // Now tell choreo view
  811. CExpression *exp = active->GetExpression( m_nDragCell );
  812. if ( exp && maybeTool )
  813. {
  814. if ( IsWindowOrChild( g_pChoreoView, maybeTool ) )
  815. {
  816. if ( g_pChoreoView->CreateExpressionEvent( pt.x, pt.y, active, exp ) )
  817. {
  818. return iret;
  819. }
  820. }
  821. if ( IsWindowOrChild( g_pExpressionTool, maybeTool ) )
  822. {
  823. if ( g_pExpressionTool->SetFlexAnimationTrackFromExpression( pt.x, pt.y, active, exp ) )
  824. {
  825. return iret;
  826. }
  827. }
  828. }
  829. }
  830. }
  831. if ( active )
  832. {
  833. // Over a new cell
  834. if ( cell >= 0 &&
  835. cell < active->GetNumExpressions() &&
  836. cell != m_nCurCell &&
  837. m_nCurCell != -1 )
  838. {
  839. // Swap cells
  840. CExpression *exp = active->GetExpression( m_nCurCell );
  841. if ( exp )
  842. {
  843. active->SwapExpressionOrder( m_nCurCell, cell );
  844. active->SetDirty( true );
  845. active->SelectExpression( cell );
  846. }
  847. }
  848. }
  849. }
  850. break;
  851. case mxEvent::Size:
  852. {
  853. int width = w2();
  854. int ch = GetCaptionHeight();
  855. g_pExpressionClass->setBounds( 5, 5 + ch, width - 120, 20 );
  856. m_pABButton->setBounds( width - 60, 4 + ch, 60, 16 );
  857. m_pThumbnailIncreaseButton->setBounds( width - 60 - 40, 4 + ch, 16, 16 );
  858. m_pThumbnailDecreaseButton->setBounds( width - 60 - 20, 4 + ch, 16, 16 );
  859. m_nTopOffset = 0;
  860. RepositionSlider();
  861. redraw();
  862. iret = 1;
  863. }
  864. break;
  865. case mxEvent::MouseWheeled:
  866. {
  867. // Figure out cell #
  868. POINT pt;
  869. pt.x = event->x;
  870. pt.y = event->y;
  871. ScreenToClient( (HWND)getHandle(), &pt );
  872. if ( event->height < 0 )
  873. {
  874. m_nTopOffset = min( m_nTopOffset + 10, slScrollbar->getMaxValue() );
  875. }
  876. else
  877. {
  878. m_nTopOffset = max( m_nTopOffset - 10, 0 );
  879. }
  880. RepositionSlider();
  881. redraw();
  882. iret = 1;
  883. }
  884. break;
  885. };
  886. if ( iret )
  887. {
  888. SetActiveTool( this );
  889. }
  890. return iret;
  891. }
  892. void mxExpressionTray::ET_Undo( int cell )
  893. {
  894. g_pControlPanel->UndoExpression( cell );
  895. }
  896. void mxExpressionTray::ET_Redo( int cell )
  897. {
  898. g_pControlPanel->RedoExpression( cell );
  899. }
  900. //-----------------------------------------------------------------------------
  901. // Purpose:
  902. //-----------------------------------------------------------------------------
  903. void mxExpressionTray::ThumbnailIncrease( void )
  904. {
  905. if ( m_nSnapshotWidth + THUMBNAIL_SIZE_STEP <= MAX_THUMBNAILSIZE )
  906. {
  907. m_nSnapshotWidth += THUMBNAIL_SIZE_STEP;
  908. g_viewerSettings.thumbnailsize = m_nSnapshotWidth;
  909. m_nSnapshotHeight = m_nSnapshotWidth + m_nDescriptionHeight;
  910. Con_Printf( "Thumbnail size %i x %i\n", m_nSnapshotWidth, m_nSnapshotWidth );
  911. redraw();
  912. }
  913. }
  914. //-----------------------------------------------------------------------------
  915. // Purpose:
  916. //-----------------------------------------------------------------------------
  917. void mxExpressionTray::ThumbnailDecrease( void )
  918. {
  919. if ( m_nSnapshotWidth - THUMBNAIL_SIZE_STEP >= MIN_THUMBNAILSIZE )
  920. {
  921. m_nSnapshotWidth -= THUMBNAIL_SIZE_STEP;
  922. g_viewerSettings.thumbnailsize = m_nSnapshotWidth;
  923. m_nSnapshotHeight = m_nSnapshotWidth + m_nDescriptionHeight;
  924. Con_Printf( "Thumbnail size %i x %i\n", m_nSnapshotWidth, m_nSnapshotWidth );
  925. redraw();
  926. }
  927. }
  928. //-----------------------------------------------------------------------------
  929. // Purpose:
  930. //-----------------------------------------------------------------------------
  931. void mxExpressionTray::RestoreThumbnailSize( void )
  932. {
  933. m_nSnapshotWidth = g_viewerSettings.thumbnailsize;
  934. m_nSnapshotWidth = max( MIN_THUMBNAILSIZE, m_nSnapshotWidth );
  935. m_nSnapshotWidth = min( MAX_THUMBNAILSIZE, m_nSnapshotWidth );
  936. g_viewerSettings.thumbnailsize = m_nSnapshotWidth;
  937. m_nSnapshotHeight = m_nSnapshotWidth + m_nDescriptionHeight;
  938. redraw();
  939. }
  940. void mxExpressionTray::ReloadBitmaps( void )
  941. {
  942. CExpClass *cl;
  943. int c = expressions->GetNumClasses();
  944. EnableStickySnapshotMode();
  945. for ( int i = 0 ; i < c; i++ )
  946. {
  947. cl = expressions->GetClass( i );
  948. if ( !cl )
  949. continue;
  950. cl->ReloadBitmaps();
  951. }
  952. DisableStickySnapshotMode();
  953. redraw();
  954. }
  955. bool IsUsingPerPlayerExpressions()
  956. {
  957. bool bPerPlayerExpressions = false;
  958. if ( CommandLine()->CheckParm( "-perplayerexpressions" ) )
  959. {
  960. bPerPlayerExpressions = true;
  961. }
  962. else
  963. {
  964. // Returns the search path, each path is separated by ;s. Returns the length of the string returned
  965. char pSearchPath[2048];
  966. if ( g_pFullFileSystem->GetSearchPath( "GAME", false, pSearchPath, sizeof(pSearchPath) ) )
  967. {
  968. Q_FixSlashes( pSearchPath );
  969. if ( Q_stristr( pSearchPath, "\\tf" ) )
  970. {
  971. bPerPlayerExpressions = true;
  972. }
  973. }
  974. }
  975. return bPerPlayerExpressions;
  976. }
  977. void mxExpressionTray::OnModelChanged()
  978. {
  979. if ( IsUsingPerPlayerExpressions() )
  980. {
  981. Msg( "Closing current phoneme set\n" );
  982. if ( !g_pControlPanel->Closeall() )
  983. return;
  984. // See if per-model overrides exist for this model
  985. char fn[ MAX_PATH ];
  986. Q_snprintf( fn, sizeof( fn ), "expressions/%s/phonemes/phonemes.txt", models->GetActiveModelName() );
  987. // Load appropriate classes
  988. char rootDir[ MAX_PATH ];
  989. Q_snprintf( rootDir, sizeof( rootDir ), "%s/phonemes/", models->GetActiveModelName() );
  990. FacePoser_SetPhonemeRootDir( rootDir );
  991. FacePoser_EnsurePhonemesLoaded();
  992. }
  993. ReloadBitmaps();
  994. RestoreThumbnailSize();
  995. }