mirror of
https://github.com/nillerusr/source-engine.git
synced 2025-01-19 11:31:29 +00:00
1183 lines
25 KiB
C++
1183 lines
25 KiB
C++
//========= Copyright Valve Corporation, All rights reserved. ============//
|
|
//
|
|
// Purpose:
|
|
//
|
|
//===========================================================================//
|
|
|
|
#include "cbase.h"
|
|
#include <windows.h>
|
|
#include "resource.h"
|
|
#include "wavefile.h"
|
|
#include "wavebrowser.h"
|
|
#include "SoundEmitterSystem/isoundemittersystembase.h"
|
|
#include "ifaceposersound.h"
|
|
#include "snd_wave_source.h"
|
|
#include "filesystem.h"
|
|
#include "tabwindow.h"
|
|
#include "inputproperties.h"
|
|
#include "choreowidgetdrawhelper.h"
|
|
#include "ifileloader.h"
|
|
#include "tier2/riff.h"
|
|
#include "UtlBuffer.h"
|
|
#include "ChoreoEvent.h"
|
|
|
|
CWaveBrowser *g_pWaveBrowser = NULL;
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: Implements the RIFF i/o interface on stdio
|
|
//-----------------------------------------------------------------------------
|
|
class StdIOReadBinary : public IFileReadBinary
|
|
{
|
|
public:
|
|
int open( const char *pFileName )
|
|
{
|
|
return (int)filesystem->Open( pFileName, "rb" );
|
|
}
|
|
|
|
int read( void *pOutput, int size, int file )
|
|
{
|
|
if ( !file )
|
|
return 0;
|
|
|
|
return filesystem->Read( pOutput, size, (FileHandle_t)file );
|
|
}
|
|
|
|
void seek( int file, int pos )
|
|
{
|
|
if ( !file )
|
|
return;
|
|
|
|
filesystem->Seek( (FileHandle_t)file, pos, FILESYSTEM_SEEK_HEAD );
|
|
}
|
|
|
|
unsigned int tell( int file )
|
|
{
|
|
if ( !file )
|
|
return 0;
|
|
|
|
return filesystem->Tell( (FileHandle_t)file );
|
|
}
|
|
|
|
unsigned int size( int file )
|
|
{
|
|
if ( !file )
|
|
return 0;
|
|
|
|
return filesystem->Size( (FileHandle_t)file );
|
|
}
|
|
|
|
void close( int file )
|
|
{
|
|
if ( !file )
|
|
return;
|
|
|
|
filesystem->Close( (FileHandle_t)file );
|
|
}
|
|
};
|
|
|
|
class StdIOWriteBinary : public IFileWriteBinary
|
|
{
|
|
public:
|
|
int create( const char *pFileName )
|
|
{
|
|
return (int)filesystem->Open( pFileName, "wb" );
|
|
}
|
|
|
|
int write( void *pData, int size, int file )
|
|
{
|
|
return filesystem->Write( pData, size, (FileHandle_t)file );
|
|
}
|
|
|
|
void close( int file )
|
|
{
|
|
filesystem->Close( (FileHandle_t)file );
|
|
}
|
|
|
|
void seek( int file, int pos )
|
|
{
|
|
filesystem->Seek( (FileHandle_t)file, pos, FILESYSTEM_SEEK_HEAD );
|
|
}
|
|
|
|
unsigned int tell( int file )
|
|
{
|
|
return filesystem->Tell( (FileHandle_t)file );
|
|
}
|
|
};
|
|
|
|
static StdIOReadBinary io_in;
|
|
static StdIOWriteBinary io_out;
|
|
|
|
#define RIFF_WAVE MAKEID('W','A','V','E')
|
|
#define WAVE_FMT MAKEID('f','m','t',' ')
|
|
#define WAVE_DATA MAKEID('d','a','t','a')
|
|
#define WAVE_FACT MAKEID('f','a','c','t')
|
|
#define WAVE_CUE MAKEID('c','u','e',' ')
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose:
|
|
// Input : &walk -
|
|
//-----------------------------------------------------------------------------
|
|
static void SceneManager_ParseSentence( CSentence& sentence, IterateRIFF &walk )
|
|
{
|
|
CUtlBuffer buf( 0, 0, CUtlBuffer::TEXT_BUFFER );
|
|
|
|
buf.EnsureCapacity( walk.ChunkSize() );
|
|
walk.ChunkRead( buf.Base() );
|
|
buf.SeekPut( CUtlBuffer::SEEK_HEAD, walk.ChunkSize() );
|
|
|
|
sentence.InitFromDataChunk( buf.Base(), buf.TellPut() );
|
|
}
|
|
|
|
bool SceneManager_LoadSentenceFromWavFileUsingIO( char const *wavfile, CSentence& sentence, IFileReadBinary& io )
|
|
{
|
|
sentence.Reset();
|
|
|
|
InFileRIFF riff( wavfile, io );
|
|
|
|
// UNDONE: Don't use printf to handle errors
|
|
if ( riff.RIFFName() != RIFF_WAVE )
|
|
{
|
|
return false;
|
|
}
|
|
|
|
// set up the iterator for the whole file (root RIFF is a chunk)
|
|
IterateRIFF walk( riff, riff.RIFFSize() );
|
|
|
|
// This chunk must be first as it contains the wave's format
|
|
// break out when we've parsed it
|
|
bool found = false;
|
|
while ( walk.ChunkAvailable() && !found )
|
|
{
|
|
switch( walk.ChunkName() )
|
|
{
|
|
case WAVE_VALVEDATA:
|
|
{
|
|
found = true;
|
|
SceneManager_ParseSentence( sentence, walk );
|
|
}
|
|
break;
|
|
}
|
|
walk.ChunkNext();
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool SceneManager_LoadSentenceFromWavFile( char const *wavfile, CSentence& sentence )
|
|
{
|
|
return SceneManager_LoadSentenceFromWavFileUsingIO( wavfile, sentence, io_in );
|
|
}
|
|
|
|
enum
|
|
{
|
|
// Controls
|
|
IDC_SB_LISTVIEW = 101,
|
|
IDC_SB_FILETREE,
|
|
|
|
// Messages
|
|
IDC_SB_PLAY = 1000,
|
|
};
|
|
|
|
enum
|
|
{
|
|
COL_WAV = 0,
|
|
COL_DUCKED,
|
|
COL_PHONEMES,
|
|
COL_SENTENCE
|
|
};
|
|
|
|
class CWaveList : public mxListView
|
|
{
|
|
public:
|
|
CWaveList( mxWindow *parent, int id = 0 )
|
|
: mxListView( parent, 0, 0, 0, 0, id )
|
|
{
|
|
// Add column headers
|
|
insertTextColumn( COL_WAV, 300, "WAV" );
|
|
insertTextColumn( COL_DUCKED, 50, "Ducked" );
|
|
insertTextColumn( COL_PHONEMES, 120, "Words [ Phonemes ]" );
|
|
insertTextColumn( COL_SENTENCE, 300, "Sentence Text" );
|
|
}
|
|
};
|
|
|
|
class CWaveFileTree : public mxTreeView
|
|
{
|
|
public:
|
|
CWaveFileTree( mxWindow *parent, int id = 0 ) : mxTreeView( parent, 0, 0, 0, 0, id ),
|
|
m_Paths( 0, 0, FileTreeLessFunc )
|
|
{
|
|
}
|
|
|
|
void Clear()
|
|
{
|
|
removeAll();
|
|
m_Paths.RemoveAll();
|
|
}
|
|
|
|
void FindOrAddSubdirectory( char const *subdir )
|
|
{
|
|
FileTreePath fp;
|
|
Q_strcpy( fp.path, subdir );
|
|
|
|
if ( m_Paths.Find( fp ) != m_Paths.InvalidIndex() )
|
|
return;
|
|
|
|
m_Paths.Insert( fp );
|
|
}
|
|
|
|
mxTreeViewItem *FindOrAddChildItem( mxTreeViewItem *parent, char const *child )
|
|
{
|
|
mxTreeViewItem *p = getFirstChild( parent );
|
|
if ( !p )
|
|
{
|
|
return add( parent, child );
|
|
}
|
|
|
|
while ( p )
|
|
{
|
|
if ( !Q_stricmp( getLabel( p ), child ) )
|
|
return p;
|
|
|
|
p = getNextChild( p );
|
|
}
|
|
|
|
return add( parent, child );
|
|
}
|
|
|
|
void _PopulateTree( int pathId, char const *path )
|
|
{
|
|
char sz[ 512 ];
|
|
Q_strcpy( sz, path );
|
|
char *p = sz;
|
|
|
|
// Start at root
|
|
mxTreeViewItem *cur = NULL;
|
|
|
|
// Tokenize path
|
|
while ( p && p[0] )
|
|
{
|
|
char *slash = Q_strstr( p, "/" );
|
|
if ( !slash )
|
|
{
|
|
slash = Q_strstr( p, "\\" );
|
|
}
|
|
|
|
char *check = p;
|
|
|
|
if ( slash )
|
|
{
|
|
*slash = 0;
|
|
|
|
// see if a child of current already exists with this name
|
|
p = slash + 1;
|
|
}
|
|
else
|
|
{
|
|
p = NULL;
|
|
}
|
|
|
|
Assert( check );
|
|
|
|
cur = FindOrAddChildItem( cur, check );
|
|
}
|
|
|
|
setUserData( cur, (void *)pathId );
|
|
}
|
|
|
|
char const *GetSelectedPath( void )
|
|
{
|
|
mxTreeViewItem *tvi = getSelectedItem();
|
|
unsigned int id = (unsigned int)getUserData( tvi );
|
|
|
|
if ( id < 0 || id >= m_Paths.Count() )
|
|
{
|
|
Assert( 0 );
|
|
return "";
|
|
}
|
|
return m_Paths[ id ].path;
|
|
}
|
|
|
|
void PopulateTree()
|
|
{
|
|
int i;
|
|
for ( i = m_Paths.FirstInorder(); i != m_Paths.InvalidIndex(); i = m_Paths.NextInorder( i ) )
|
|
{
|
|
_PopulateTree( i, m_Paths[ i ].path );
|
|
}
|
|
|
|
mxTreeViewItem *p = getFirstChild( NULL );
|
|
setOpen( p, true );
|
|
}
|
|
|
|
struct FileTreePath
|
|
{
|
|
char path[ MAX_PATH ];
|
|
};
|
|
|
|
static bool FileTreeLessFunc( const FileTreePath &lhs, const FileTreePath &rhs )
|
|
{
|
|
return Q_stricmp( lhs.path, rhs.path ) < 0;
|
|
}
|
|
|
|
CUtlRBTree< FileTreePath, int > m_Paths;
|
|
};
|
|
#pragma optimize( "", off )
|
|
class CWaveOptionsWindow : public mxWindow
|
|
{
|
|
typedef mxWindow BaseClass;
|
|
public:
|
|
enum
|
|
{
|
|
IDC_PLAYSOUND = 1000,
|
|
IDC_STOP_SOUNDS,
|
|
IDC_SEARCH,
|
|
IDC_CANCELSEARCH,
|
|
};
|
|
|
|
CWaveOptionsWindow( CWaveBrowser *browser ) : BaseClass( browser, 0, 0, 0, 0 ), m_pBrowser( browser )
|
|
{
|
|
FacePoser_AddWindowStyle( this, WS_CLIPSIBLINGS | WS_CLIPCHILDREN );
|
|
|
|
m_szSearchString[0]=0;
|
|
|
|
m_pPlay = new mxButton( this, 0, 0, 0, 0, "Play", IDC_PLAYSOUND );
|
|
|
|
m_pStopSounds = new mxButton( this, 0, 0, 0, 0, "Stop Sounds", IDC_STOP_SOUNDS );
|
|
|
|
m_pSearch = new mxLineEdit( this, 0, 0, 0, 0, "", IDC_SEARCH );
|
|
|
|
m_pCancelSearch = new mxButton( this, 0, 0, 0, 0, "Cancel", IDC_CANCELSEARCH );
|
|
}
|
|
|
|
bool PaintBackground( void )
|
|
{
|
|
redraw();
|
|
return false;
|
|
}
|
|
|
|
|
|
virtual void redraw()
|
|
{
|
|
CChoreoWidgetDrawHelper drawHelper( this, GetSysColor( COLOR_BTNFACE ) );
|
|
}
|
|
virtual int handleEvent( mxEvent *event )
|
|
{
|
|
int iret = 0;
|
|
switch ( event->event )
|
|
{
|
|
default:
|
|
break;
|
|
case mxEvent::Size:
|
|
{
|
|
iret = 1;
|
|
|
|
int split = 120;
|
|
|
|
int x = 1;
|
|
|
|
m_pPlay->setBounds( x, 1, split, h2() - 2 );
|
|
|
|
x += split + 10;
|
|
|
|
m_pStopSounds->setBounds( x, 1, split, h2()-2 );
|
|
|
|
x += split + 10;
|
|
|
|
m_pCancelSearch->setBounds( x, 1, split, h2() - 2 );
|
|
|
|
x += split + 10;
|
|
|
|
m_pSearch->setBounds( x, 0, split * 3, h2() - 1 );
|
|
|
|
x += split * 3 + 10;
|
|
}
|
|
break;
|
|
case mxEvent::KeyDown:
|
|
switch ( event->action )
|
|
{
|
|
default:
|
|
break;
|
|
case IDC_SEARCH:
|
|
{
|
|
if ( event->event == mxEvent::KeyDown )
|
|
{
|
|
OnSearch();
|
|
}
|
|
iret = 1;
|
|
};
|
|
break;
|
|
}
|
|
break;
|
|
case mxEvent::Action:
|
|
{
|
|
switch ( event->action )
|
|
{
|
|
case IDC_STOP_SOUNDS:
|
|
{
|
|
iret = 1;
|
|
sound->StopAll();
|
|
}
|
|
break;
|
|
case IDC_SEARCH:
|
|
iret = 1;
|
|
break;
|
|
case IDC_PLAYSOUND:
|
|
{
|
|
iret = 1;
|
|
m_pBrowser->OnPlay();
|
|
}
|
|
break;
|
|
case IDC_CANCELSEARCH:
|
|
{
|
|
iret = 1;
|
|
OnCancelSearch();
|
|
}
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
|
|
return iret;
|
|
}
|
|
|
|
char const *GetSearchString()
|
|
{
|
|
return m_szSearchString;
|
|
}
|
|
|
|
void OnSearch()
|
|
{
|
|
m_pSearch->getText( m_szSearchString, sizeof( m_szSearchString ) );
|
|
|
|
m_pBrowser->OnSearch();
|
|
}
|
|
|
|
void OnCancelSearch()
|
|
{
|
|
m_szSearchString[ 0 ] = 0;
|
|
m_pSearch->clear();
|
|
|
|
m_pBrowser->OnCancelSearch();
|
|
}
|
|
|
|
private:
|
|
|
|
mxButton *m_pStopSounds;
|
|
mxButton *m_pPlay;
|
|
mxLineEdit *m_pSearch;
|
|
mxButton *m_pCancelSearch;
|
|
|
|
CWaveBrowser *m_pBrowser;
|
|
|
|
char m_szSearchString[ 256 ];
|
|
};
|
|
|
|
#pragma optimize( "", on )
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose:
|
|
// Input : *parent -
|
|
//-----------------------------------------------------------------------------
|
|
CWaveBrowser::CWaveBrowser( mxWindow *parent )
|
|
: IFacePoserToolWindow( "WaveBrowser", "Waves" ), mxWindow( parent, 0, 0, 0, 0 )
|
|
{
|
|
SetAutoProcess( false );
|
|
|
|
m_bTextSearch = false;
|
|
m_nPrevProcessed = -1;
|
|
|
|
m_pListView = new CWaveList( this, IDC_SB_LISTVIEW );
|
|
m_pOptions = new CWaveOptionsWindow( this );
|
|
m_pFileTree = new CWaveFileTree( this, IDC_SB_FILETREE );
|
|
|
|
//HIMAGELIST list = CreateImageList();
|
|
|
|
// Associate the image list with the tree-view control.
|
|
//m_pListView->setImageList( (void *)list );
|
|
|
|
LoadAllSounds();
|
|
|
|
PopulateTree( NULL );
|
|
}
|
|
|
|
#define CX_ICON 16
|
|
#define CY_ICON 16
|
|
|
|
HIMAGELIST CWaveBrowser::CreateImageList()
|
|
{
|
|
HIMAGELIST list;
|
|
|
|
list = ImageList_Create( CX_ICON, CY_ICON,
|
|
FALSE, NUM_IMAGES, 0 );
|
|
|
|
// Load the icon resources, and add the icons to the image list.
|
|
HICON hicon;
|
|
int slot;
|
|
#if defined( DBGFLAG_ASSERT )
|
|
int c = 0;
|
|
#endif
|
|
|
|
/*
|
|
hicon = LoadIcon( GetModuleHandle( 0 ), MAKEINTRESOURCE(IDI_WORKSPACE));
|
|
slot = ImageList_AddIcon(list, hicon);
|
|
Assert( slot == c++ );
|
|
DeleteObject( hicon );
|
|
|
|
hicon = LoadIcon( GetModuleHandle( 0 ), MAKEINTRESOURCE(IDI_WORKSPACE_CHECKEDOUT));
|
|
slot = ImageList_AddIcon(list, hicon);
|
|
Assert( slot == c++ );
|
|
DeleteObject( hicon );
|
|
|
|
hicon = LoadIcon(GetModuleHandle( 0 ), MAKEINTRESOURCE(IDI_PROJECT));
|
|
slot = ImageList_AddIcon(list, hicon);
|
|
Assert( slot == c++ );
|
|
DeleteObject( hicon );
|
|
|
|
hicon = LoadIcon(GetModuleHandle( 0 ), MAKEINTRESOURCE(IDI_PROJECT_CHECKEDOUT));
|
|
slot = ImageList_AddIcon(list, hicon);
|
|
Assert( slot == c++ );
|
|
DeleteObject( hicon );
|
|
|
|
hicon = LoadIcon(GetModuleHandle( 0 ), MAKEINTRESOURCE(IDI_SCENE));
|
|
slot = ImageList_AddIcon(list, hicon);
|
|
Assert( slot == c++ );
|
|
DeleteObject( hicon );
|
|
*/
|
|
|
|
// hicon = LoadIcon(GetModuleHandle( 0 ), MAKEINTRESOURCE(IDI_SCENE_CHECKEDOUT));
|
|
// slot = ImageList_AddIcon(list, hicon);
|
|
// Assert( slot == c++ );
|
|
// DeleteObject( hicon );
|
|
|
|
/*
|
|
hicon = LoadIcon(GetModuleHandle( 0 ), MAKEINTRESOURCE(IDI_VCD));
|
|
slot = ImageList_AddIcon(list, hicon);
|
|
Assert( slot == c++ );
|
|
DeleteObject( hicon );
|
|
|
|
hicon = LoadIcon(GetModuleHandle( 0 ), MAKEINTRESOURCE(IDI_VCD_CHECKEDOUT ));
|
|
slot = ImageList_AddIcon(list, hicon);
|
|
Assert( slot == c++ );
|
|
DeleteObject( hicon );
|
|
*/
|
|
|
|
hicon = LoadIcon(GetModuleHandle( 0 ), MAKEINTRESOURCE(IDI_WAV));
|
|
slot = ImageList_AddIcon(list, hicon);
|
|
Assert( slot == c++ );
|
|
DeleteObject( hicon );
|
|
|
|
/*
|
|
hicon = LoadIcon(GetModuleHandle( 0 ), MAKEINTRESOURCE(IDI_WAV_CHECKEDOUT));
|
|
slot = ImageList_AddIcon(list, hicon);
|
|
Assert( slot == c++ );
|
|
DeleteObject( hicon );
|
|
|
|
hicon = LoadIcon(GetModuleHandle( 0 ), MAKEINTRESOURCE(IDI_SPEAK));
|
|
slot = ImageList_AddIcon(list, hicon);
|
|
Assert( slot == c++ );
|
|
DeleteObject( hicon );
|
|
|
|
hicon = LoadIcon(GetModuleHandle( 0 ), MAKEINTRESOURCE(IDI_SPEAK_CHECKEDOUT));
|
|
slot = ImageList_AddIcon(list, hicon);
|
|
Assert( slot == c++ );
|
|
DeleteObject( hicon );
|
|
*/
|
|
|
|
return list;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose:
|
|
//-----------------------------------------------------------------------------
|
|
void CWaveBrowser::OnDelete()
|
|
{
|
|
RemoveAllSounds();
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose:
|
|
// Input : *event -
|
|
// Output : int
|
|
//-----------------------------------------------------------------------------
|
|
int CWaveBrowser::handleEvent( mxEvent *event )
|
|
{
|
|
int iret = 0;
|
|
|
|
if ( HandleToolEvent( event ) )
|
|
{
|
|
return iret;
|
|
}
|
|
|
|
switch ( event->event )
|
|
{
|
|
default:
|
|
break;
|
|
case mxEvent::Action:
|
|
{
|
|
iret = 1;
|
|
switch ( event->action )
|
|
{
|
|
default:
|
|
{
|
|
iret = 0;
|
|
}
|
|
break;
|
|
case IDC_SB_LISTVIEW:
|
|
{
|
|
SetActiveTool( this );
|
|
|
|
bool rightmouse = ( event->flags == mxEvent::RightClicked ) ? true : false;
|
|
bool doubleclicked = ( event->flags == mxEvent::DoubleClicked ) ? true : false;
|
|
|
|
if ( rightmouse )
|
|
{
|
|
ShowContextMenu();
|
|
}
|
|
else if ( doubleclicked )
|
|
{
|
|
if ( m_pListView->getNumSelected() == 1 )
|
|
{
|
|
int index = m_pListView->getNextSelectedItem( -1 );
|
|
if ( index >= 0 )
|
|
{
|
|
CWaveFile *wav = (CWaveFile *)m_pListView->getUserData( index, 0 );
|
|
if ( wav )
|
|
{
|
|
wav->Play();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
case IDC_SB_FILETREE:
|
|
{
|
|
SetActiveTool( this );
|
|
|
|
PopulateTree( m_pFileTree->GetSelectedPath() );
|
|
}
|
|
break;
|
|
case IDC_SB_PLAY:
|
|
{
|
|
OnPlay();
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
case mxEvent::Size:
|
|
{
|
|
int optionsh = 20;
|
|
|
|
m_pOptions->setBounds( 0, 0, w2(), optionsh );
|
|
|
|
int filetreewidth = 175;
|
|
|
|
m_pFileTree->setBounds( 0, optionsh, filetreewidth, h2() - optionsh );
|
|
m_pListView->setBounds( filetreewidth, optionsh, w2() - filetreewidth, h2() - optionsh );
|
|
|
|
iret = 1;
|
|
}
|
|
break;
|
|
case mxEvent::Close:
|
|
{
|
|
iret = 1;
|
|
}
|
|
break;
|
|
}
|
|
|
|
return iret;
|
|
}
|
|
|
|
static bool NameLessFunc( CWaveFile *const& name1, CWaveFile *const& name2 )
|
|
{
|
|
if ( Q_stricmp( name1->GetName(), name2->GetName() ) < 0 )
|
|
return true;
|
|
return false;
|
|
}
|
|
|
|
#define SOUND_PREFIX_LEN 6
|
|
//-----------------------------------------------------------------------------
|
|
// Finds all .wav files in a particular directory
|
|
//-----------------------------------------------------------------------------
|
|
bool CWaveBrowser::LoadWaveFilesInDirectory( CUtlDict< CWaveFile *, int >& soundlist, char const* pDirectoryName, int nDirectoryNameLen )
|
|
{
|
|
Assert( Q_strnicmp( pDirectoryName, "sound", 5 ) == 0 );
|
|
|
|
char *pWildCard;
|
|
pWildCard = ( char * )stackalloc( nDirectoryNameLen + 7 );
|
|
Q_snprintf( pWildCard, nDirectoryNameLen + 7, "%s/*.wav", pDirectoryName );
|
|
|
|
if ( !filesystem )
|
|
{
|
|
return false;
|
|
}
|
|
|
|
FileFindHandle_t findHandle;
|
|
const char *pFileName = filesystem->FindFirst( pWildCard, &findHandle );
|
|
while( pFileName )
|
|
{
|
|
if( !filesystem->FindIsDirectory( findHandle ) )
|
|
{
|
|
// Strip off the 'sound/' part of the name.
|
|
char *pFileNameWithPath;
|
|
int nAllocSize = nDirectoryNameLen + Q_strlen(pFileName) + 2;
|
|
pFileNameWithPath = (char *)stackalloc( nAllocSize );
|
|
Q_snprintf( pFileNameWithPath, nAllocSize, "%s/%s", &pDirectoryName[SOUND_PREFIX_LEN], pFileName );
|
|
Q_strnlwr( pFileNameWithPath, nAllocSize );
|
|
|
|
CWaveFile *wav = new CWaveFile( pFileNameWithPath );
|
|
soundlist.Insert( pFileNameWithPath, wav );
|
|
|
|
/*
|
|
if ( !(soundlist.Count() % 500 ) )
|
|
{
|
|
Con_Printf( "CWaveBrowser: loaded %i sounds\n", soundlist.Count() );
|
|
}
|
|
*/
|
|
}
|
|
pFileName = filesystem->FindNext( findHandle );
|
|
}
|
|
|
|
m_pFileTree->FindOrAddSubdirectory( &pDirectoryName[ SOUND_PREFIX_LEN ] );
|
|
|
|
filesystem->FindClose( findHandle );
|
|
return true;
|
|
}
|
|
|
|
bool CWaveBrowser::InitDirectoryRecursive( CUtlDict< CWaveFile *, int >& soundlist, char const* pDirectoryName )
|
|
{
|
|
// Compute directory name length
|
|
int nDirectoryNameLen = Q_strlen( pDirectoryName );
|
|
|
|
if (!LoadWaveFilesInDirectory( soundlist, pDirectoryName, nDirectoryNameLen ) )
|
|
return false;
|
|
|
|
char *pWildCard = ( char * )stackalloc( nDirectoryNameLen + 4 );
|
|
strcpy(pWildCard, pDirectoryName);
|
|
strcat(pWildCard, "/*.");
|
|
int nPathStrLen = nDirectoryNameLen + 1;
|
|
|
|
FileFindHandle_t findHandle;
|
|
const char *pFileName = filesystem->FindFirst( pWildCard, &findHandle );
|
|
while( pFileName )
|
|
{
|
|
if ((pFileName[0] != '.') || (pFileName[1] != '.' && pFileName[1] != 0))
|
|
{
|
|
if( filesystem->FindIsDirectory( findHandle ) )
|
|
{
|
|
int fileNameStrLen = Q_strlen( pFileName );
|
|
char *pFileNameWithPath = ( char * )stackalloc( nPathStrLen + fileNameStrLen + 1 );
|
|
memcpy( pFileNameWithPath, pWildCard, nPathStrLen );
|
|
pFileNameWithPath[nPathStrLen] = '\0';
|
|
strcat( pFileNameWithPath, pFileName );
|
|
|
|
if (!InitDirectoryRecursive( soundlist, pFileNameWithPath ))
|
|
return false;
|
|
}
|
|
}
|
|
pFileName = filesystem->FindNext( findHandle );
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void CWaveBrowser::LoadAllSounds()
|
|
{
|
|
RemoveAllSounds();
|
|
|
|
Con_Printf( "Building list of all .wavs in sound/ folder\n" );
|
|
|
|
InitDirectoryRecursive( m_AllSounds, "sound" );
|
|
|
|
m_pFileTree->PopulateTree();
|
|
}
|
|
|
|
void CWaveBrowser::RemoveAllSounds()
|
|
{
|
|
int c = m_AllSounds.Count();
|
|
for ( int i = 0; i < c; i++ )
|
|
{
|
|
CWaveFile *wav = m_AllSounds[ i ];
|
|
delete wav;
|
|
}
|
|
|
|
m_AllSounds.RemoveAll();
|
|
m_Scripts.RemoveAll();
|
|
m_CurrentSelection.RemoveAll();
|
|
|
|
m_pFileTree->Clear();
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose:
|
|
//-----------------------------------------------------------------------------
|
|
void CWaveBrowser::PopulateTree( char const *subdirectory )
|
|
{
|
|
int i;
|
|
|
|
CUtlRBTree< CWaveFile *, int > m_Sorted( 0, 0, NameLessFunc );
|
|
|
|
bool check_load_sentence_data = false;
|
|
|
|
char const *texttofind = NULL;
|
|
|
|
if ( m_bTextSearch )
|
|
{
|
|
subdirectory = NULL;
|
|
texttofind = GetSearchString();
|
|
}
|
|
|
|
int len = 0;
|
|
if ( subdirectory )
|
|
{
|
|
len = Q_strlen( subdirectory );
|
|
check_load_sentence_data = ( Q_strstr( subdirectory, "/" ) || subdirectory[0] ) ? true : false;
|
|
}
|
|
|
|
int c = m_AllSounds.Count();
|
|
for ( i = 0; i < c; i++ )
|
|
{
|
|
CWaveFile *wav = m_AllSounds[ i ];
|
|
char const *name = wav->GetName();
|
|
|
|
if ( subdirectory )
|
|
{
|
|
if ( Q_strnicmp( subdirectory, wav->GetName(), len ) )
|
|
continue;
|
|
}
|
|
|
|
if ( m_bTextSearch && texttofind )
|
|
{
|
|
if ( !Q_stristr( name, texttofind ) )
|
|
continue;
|
|
}
|
|
|
|
m_Sorted.Insert( wav );
|
|
}
|
|
|
|
char prevSelectedName[ 512 ];
|
|
prevSelectedName[ 0 ] = 0;
|
|
if ( m_pListView->getNumSelected() == 1 )
|
|
{
|
|
int selectedItem = m_pListView->getNextSelectedItem( 0 );
|
|
if ( selectedItem >= 0 )
|
|
{
|
|
// Grab wave name of previously selected item
|
|
Q_strcpy( prevSelectedName, m_pListView->getLabel( selectedItem, 0 ) );
|
|
}
|
|
}
|
|
|
|
// Repopulate tree
|
|
m_pListView->removeAll();
|
|
|
|
int loadcount = 0;
|
|
|
|
m_pListView->setDrawingEnabled( false );
|
|
|
|
int selectedSlot = -1;
|
|
|
|
CUtlVector< CWaveFile * > list;
|
|
|
|
|
|
for ( i = m_Sorted.FirstInorder(); i != m_Sorted.InvalidIndex(); i = m_Sorted.NextInorder( i ) )
|
|
{
|
|
CWaveFile *wav = m_Sorted[ i ];
|
|
char const *name = wav->GetName();
|
|
|
|
int slot = m_pListView->add( name );
|
|
|
|
if ( !Q_stricmp( prevSelectedName, name ) )
|
|
{
|
|
selectedSlot = slot;
|
|
}
|
|
|
|
if ( ( check_load_sentence_data || m_bTextSearch ) &&
|
|
!wav->HasLoadedSentenceInfo() && !wav->IsAsyncLoading() )
|
|
{
|
|
wav->SetAsyncLoading( true );
|
|
list.AddToTail( wav );
|
|
}
|
|
|
|
// m_pListView->setImage( slot, COL_WAV, wav->GetIconIndex() );
|
|
m_pListView->setUserData( slot, COL_WAV, (void *)wav );
|
|
|
|
if ( wav->HasLoadedSentenceInfo() )
|
|
{
|
|
m_pListView->setLabel( slot, COL_DUCKED, wav->GetVoiceDuck() ? "yes" : "no" );
|
|
m_pListView->setLabel( slot, COL_PHONEMES, wav->GetPhonemeCount() || wav->GetWordCount() ? va( "%i [ %i ]", wav->GetWordCount(), wav->GetPhonemeCount() ) : "" );
|
|
m_pListView->setLabel( slot, COL_SENTENCE, wav->GetSentenceText() );
|
|
}
|
|
else
|
|
{
|
|
m_pListView->setLabel( slot, COL_SENTENCE, "(loading...)" );
|
|
}
|
|
|
|
++loadcount;
|
|
}
|
|
|
|
m_pListView->setDrawingEnabled( true );
|
|
|
|
if ( selectedSlot != -1 )
|
|
{
|
|
m_pListView->setSelected( selectedSlot, true );
|
|
m_pListView->scrollToItem( selectedSlot );
|
|
}
|
|
|
|
if ( list.Count() > 0 )
|
|
{
|
|
fileloader->AddWaveFilesToThread( list );
|
|
}
|
|
|
|
// Con_Printf( "CWaveBrowser: selected %i sounds\n", loadcount );
|
|
}
|
|
|
|
void CWaveBrowser::RepopulateTree()
|
|
{
|
|
PopulateTree( m_pFileTree->GetSelectedPath() );
|
|
}
|
|
|
|
void CWaveBrowser::BuildSelectionList( CUtlVector< CWaveFile * >& selected )
|
|
{
|
|
selected.RemoveAll();
|
|
|
|
int idx = -1;
|
|
do
|
|
{
|
|
idx = m_pListView->getNextSelectedItem( idx );
|
|
if ( idx != -1 )
|
|
{
|
|
CWaveFile *wav = (CWaveFile *)m_pListView->getUserData( idx, 0 );
|
|
if ( wav )
|
|
{
|
|
selected.AddToTail( wav );
|
|
}
|
|
}
|
|
} while ( idx != -1 );
|
|
|
|
}
|
|
|
|
void CWaveBrowser::ShowContextMenu( void )
|
|
{
|
|
SetActiveTool( this );
|
|
|
|
BuildSelectionList( m_CurrentSelection );
|
|
if ( m_CurrentSelection.Count() <= 0 )
|
|
return;
|
|
|
|
POINT pt;
|
|
GetCursorPos( &pt );
|
|
ScreenToClient( (HWND)getHandle(), &pt );
|
|
|
|
// New scene, edit comments
|
|
mxPopupMenu *pop = new mxPopupMenu();
|
|
|
|
if ( m_CurrentSelection.Count() == 1 )
|
|
{
|
|
pop->add ("&Play", IDC_SB_PLAY );
|
|
// pop->addSeparator();
|
|
}
|
|
|
|
// pop->add( "Import Sentence Data", IDC_SB_IMPORTSENTENCE );
|
|
// pop->add( "Export Sentence Data", IDC_SB_EXPORTSENTENCE );
|
|
|
|
pop->popup( this, pt.x, pt.y );
|
|
}
|
|
|
|
void CWaveBrowser::OnPlay()
|
|
{
|
|
SetActiveTool( this );
|
|
|
|
BuildSelectionList( m_CurrentSelection );
|
|
if ( m_CurrentSelection.Count() == 1 )
|
|
{
|
|
CWaveFile *wav = m_CurrentSelection[ 0 ];
|
|
if ( wav )
|
|
{
|
|
wav->Play();
|
|
}
|
|
}
|
|
}
|
|
|
|
static void SplitFileName( char const *in, char *path, int maxpath, char *filename, int maxfilename )
|
|
{
|
|
char drive[_MAX_DRIVE];
|
|
char dir[_MAX_DIR];
|
|
char fname[_MAX_FNAME];
|
|
char ext[_MAX_EXT];
|
|
|
|
_splitpath( in, drive, dir, fname, ext );
|
|
|
|
if ( dir[0] )
|
|
{
|
|
Q_snprintf( path, maxpath, "\\%s", dir );
|
|
}
|
|
else
|
|
{
|
|
path[0] = 0;
|
|
}
|
|
Q_snprintf( filename, maxfilename, "%s%s", fname, ext );
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose:
|
|
// Input : *se -
|
|
//-----------------------------------------------------------------------------
|
|
void CWaveBrowser::JumpToItem( CWaveFile *wav )
|
|
{
|
|
|
|
SetActiveTool( this );
|
|
|
|
char path[ 256 ];
|
|
char filename[ 256 ];
|
|
|
|
SplitFileName( wav->GetFileName(), path, sizeof( path ), filename, sizeof( filename ) );
|
|
|
|
char *usepath = path + Q_strlen( "/sound/" );
|
|
PopulateTree( usepath );
|
|
|
|
int idx = 0;
|
|
int c = m_pListView->getItemCount();
|
|
for ( ; idx < c; idx++ )
|
|
{
|
|
CWaveFile *item = (CWaveFile *)m_pListView->getUserData( idx, 0 );
|
|
if ( !Q_stricmp( item->GetFileName(), wav->GetFileName() ) )
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
|
|
if ( idx < c )
|
|
{
|
|
m_pListView->scrollToItem( idx );
|
|
}
|
|
}
|
|
|
|
CWaveFile *CWaveBrowser::FindEntry( char const *wavname, bool jump /*= false*/ )
|
|
{
|
|
int idx = m_AllSounds.Find( wavname );
|
|
if ( idx != m_AllSounds.InvalidIndex() )
|
|
{
|
|
CWaveFile *wav = m_AllSounds[ idx ];
|
|
if ( jump )
|
|
{
|
|
JumpToItem( wav );
|
|
}
|
|
|
|
return wav;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
int CWaveBrowser::GetSoundCount() const
|
|
{
|
|
return m_AllSounds.Count();
|
|
}
|
|
|
|
CWaveFile *CWaveBrowser::GetSound( int index )
|
|
{
|
|
if ( index < 0 || index >= (int)m_AllSounds.Count() )
|
|
return NULL;
|
|
|
|
return m_AllSounds[ index ];
|
|
}
|
|
|
|
|
|
void CWaveBrowser::OnSearch()
|
|
{
|
|
SetActiveTool( this );
|
|
|
|
m_bTextSearch = true;
|
|
|
|
PopulateTree( GetSearchString());
|
|
}
|
|
|
|
void CWaveBrowser::OnCancelSearch()
|
|
{
|
|
SetActiveTool( this );
|
|
|
|
m_bTextSearch = false;
|
|
|
|
PopulateTree( m_pFileTree->GetSelectedPath() );
|
|
}
|
|
|
|
char const *CWaveBrowser::GetSearchString()
|
|
{
|
|
return m_pOptions->GetSearchString();
|
|
}
|
|
|
|
void CWaveBrowser::Think( float dt )
|
|
{
|
|
int pending = fileloader->GetPendingLoadCount();
|
|
if ( pending != m_nPrevProcessed )
|
|
{
|
|
m_nPrevProcessed = pending;
|
|
|
|
// Put into suffix of window title
|
|
if ( pending == 0 )
|
|
{
|
|
SetSuffix( "" );
|
|
}
|
|
else
|
|
{
|
|
SetSuffix( va( " - %i", pending ) );
|
|
}
|
|
}
|
|
|
|
int c = fileloader->ProcessCompleted();
|
|
if ( c > 0 )
|
|
{
|
|
RepopulateTree();
|
|
}
|
|
}
|
|
|
|
void CWaveBrowser::SetEvent( CChoreoEvent *event )
|
|
{
|
|
if ( event->GetType() != CChoreoEvent::SPEAK )
|
|
return;
|
|
|
|
SetCurrent( FacePoser_TranslateSoundName( event->GetParameters() ) );
|
|
}
|
|
|
|
void CWaveBrowser::SetCurrent( char const *filename )
|
|
{
|
|
// Get sound name and look up .wav from it
|
|
char const *p = filename;
|
|
if ( p &&
|
|
( !Q_strnicmp( p, "sound/", 6 ) || !Q_strnicmp( p, "sound\\", 6 ) ) )
|
|
{
|
|
p += 6;
|
|
}
|
|
|
|
char fn[ 512 ];
|
|
Q_strncpy( fn, p, sizeof( fn ) );
|
|
Q_FixSlashes( fn );
|
|
|
|
int i;
|
|
int c = m_pListView->getItemCount();
|
|
|
|
for ( i = 0; i < c; ++i )
|
|
{
|
|
CWaveFile *wav = reinterpret_cast< CWaveFile * >( m_pListView->getUserData( i, COL_WAV ) );
|
|
if ( !wav )
|
|
continue;
|
|
|
|
char fixed[ 512 ];
|
|
Q_strncpy( fixed, wav->GetName(), sizeof( fixed ) );
|
|
Q_FixSlashes( fixed );
|
|
|
|
if ( !Q_stricmp( fixed, fn ) )
|
|
{
|
|
m_pListView->scrollToItem( i );
|
|
m_pListView->setSelected( i, true );
|
|
}
|
|
else
|
|
{
|
|
m_pListView->setSelected( i, false );
|
|
}
|
|
}
|
|
}
|