@ -41,250 +41,339 @@ You can contact Cyan Worlds, Inc. by email legal@cyan.com
* = = LICENSE = = */
* = = LICENSE = = */
// Basic edit dialog stuff
// Basic edit dialog stuff
# include "plEditDlg.h"
# include "plEditDlg.h"
# include "res/resource.h"
# include "plLocTreeView.h"
# include "plLocTreeView.h"
# include "plAddDlgs.h"
# include "plAddDlgs.h"
# include "pfLocalizationMgr/pfLocalizationMgr.h"
# include "pfLocalizationMgr/pfLocalizationDataMgr.h"
# include "pfLocalizationMgr/pfLocalizationDataMgr.h"
# include <map>
# include <QDialog>
# include <QMessageBox>
# include <QFileDialog>
# include "ui_EditDialog.h"
HWND gEditDlg = NULL ;
# include <functional>
extern HINSTANCE gInstance ;
extern HWND gTreeView ;
// global data for this dialog
# define ABOUT_TEXT R"(plLocalizationEditor
plString gCurrentPath ;
A basic editor for Plasma 21 localization resource files
Copyright ( C ) 2004 Cyan Worlds , Inc . ) "
// split a subtitle path up into its component parts
static void IAboutDialog ( QWidget * parent )
void SplitLocalizationPath ( plString path , plString & ageName , plString & setName , plString & locName , plString & locLanguage )
{
{
ageName = setName = locName = locLanguage = " " ;
QDialog dlg ( parent ) ;
QLabel * image = new QLabel ( & dlg ) ;
std : : vector < plString > tokens = path . Tokenize ( " . " ) ;
image - > setPixmap ( QPixmap ( " :/icon1.ico " ) ) ;
if ( tokens . size ( ) > = 1 )
QLabel * text = new QLabel ( QObject : : tr ( ABOUT_TEXT ) , & dlg ) ;
ageName = tokens [ 0 ] ;
QPushButton * ok = new QPushButton ( QObject : : tr ( " OK " ) , & dlg ) ;
if ( tokens . size ( ) > = 2 )
ok - > setDefault ( true ) ;
setName = tokens [ 1 ] ;
if ( tokens . size ( ) > = 3 )
QHBoxLayout * layout = new QHBoxLayout ( & dlg ) ;
locName = tokens [ 2 ] ;
layout - > setMargin ( 8 ) ;
if ( tokens . size ( ) > = 4 )
layout - > setSpacing ( 10 ) ;
locLanguage = tokens [ 3 ] ;
layout - > addWidget ( image ) ;
layout - > addWidget ( text ) ;
layout - > addWidget ( ok ) ;
dlg . connect ( ok , & QPushButton : : clicked , & dlg , & QDialog : : accept ) ;
dlg . exec ( ) ;
}
}
// saves the current localization text to the data manager
EditDialog : : EditDialog ( )
void SaveLocalizationText ( )
: fEditMode ( kEditNothing )
{
{
if ( gCurrentPath . IsEmpty ( ) )
fUI = new Ui_EditDialog ;
return ; // no path to save
fUI - > setupUi ( this ) ;
uint32_t textLen = ( uint32_t ) SendMessage ( GetDlgItem ( gEditDlg , IDC_LOCALIZATIONTEXT ) , WM_GETTEXTLENGTH , ( WPARAM ) 0 , ( LPARAM ) 0 ) ;
connect ( fUI - > fOpenAction , SIGNAL ( triggered ( ) ) , SLOT ( OpenDataDirectory ( ) ) ) ;
wchar_t * buffer = new wchar_t [ textLen + 2 ] ;
connect ( fUI - > fSaveCurrentAction , SIGNAL ( triggered ( ) ) , SLOT ( SaveToCurrent ( ) ) ) ;
GetDlgItemTextW ( gEditDlg , IDC_LOCALIZATIONTEXT , buffer , textLen + 1 ) ;
connect ( fUI - > fSaveOtherAction , SIGNAL ( triggered ( ) ) , SLOT ( SaveToDirectory ( ) ) ) ;
buffer [ textLen + 1 ] = 0 ;
connect ( fUI - > fExitAction , SIGNAL ( triggered ( ) ) , SLOT ( close ( ) ) ) ;
plString plainTextData = plString : : FromWchar ( buffer ) ;
connect ( fUI - > fAboutAction , & QAction : : triggered , std : : bind ( & IAboutDialog , this ) ) ;
delete [ ] buffer ;
plString ageName , setName , elementName , elementLanguage ;
SplitLocalizationPath ( gCurrentPath , ageName , setName , elementName , elementLanguage ) ;
plString name = plString : : Format ( " %s.%s.%s " , ageName . c_str ( ) , setName . c_str ( ) , elementName . c_str ( ) ) ;
connect ( fUI - > fLocalizationTree , SIGNAL ( currentItemChanged ( QTreeWidgetItem * , QTreeWidgetItem * ) ) ,
pfLocalizationDataMgr : : Instance ( ) . SetElementPlainTextData ( name , elementLanguage , plainTextData ) ;
SLOT ( LocPathChanged ( QTreeWidgetItem * , QTreeWidgetItem * ) ) ) ;
connect ( fUI - > fAddButton , SIGNAL ( clicked ( ) ) , SLOT ( AddClicked ( ) ) ) ;
connect ( fUI - > fDeleteButton , SIGNAL ( clicked ( ) ) , SLOT ( DeleteClicked ( ) ) ) ;
EnableEdit ( false ) ;
}
}
// Reset all controls to their default values (except for static controls)
EditDialog : : ~ EditDialog ( )
void ResetDlgDefaults ( )
{
{
SetDlgItemTextW ( gEditDlg , IDC_LOCALIZATIONTEXT , L " " ) ;
pfLocalizationMgr : : Shutdown ( ) ;
delete fUI ;
}
}
// Enable/disable all edit controls (some won't enable/disable unless an audio file is loaded and the subtitle is timed)
// saves the current localization text to the data manager
void EnableDlg ( BOOL enable )
void EditDialog : : SaveLocalizationText ( )
{
{
if ( ! enable )
if ( fCurrentLocPath . IsEmpty ( ) )
ResetDlgDefaults ( ) ; // reset controls to defaults
return ; // no path to save
EnableWindow ( GetDlgItem ( gEditDlg , IDC_LOCALIZATIONTEXT ) , enable ) ;
plString text = fUI - > fLocalizationText - > toPlainText ( ) . toUtf8 ( ) . constData ( ) ;
plString ageName , setName , elementName , elementLanguage ;
SplitLocalizationPath ( fCurrentLocPath , ageName , setName , elementName , elementLanguage ) ;
plString name = plString : : Format ( " %s.%s.%s " , ageName . c_str ( ) , setName . c_str ( ) , elementName . c_str ( ) ) ;
pfLocalizationDataMgr : : Instance ( ) . SetElementPlainTextData ( name , elementLanguage , text ) ;
}
}
// updates the edit dialog based on the path specified
void EditDialog : : LoadLocalization ( const plString & locPath )
void UpdateEditDlg ( plString locPath )
{
{
if ( locPath = = gCurrentPath )
if ( locPath = = fCurrentLoc Path)
return ;
return ;
gCurrentPath = locPath ;
fCurrentLocPath = locPath ;
fUI - > fTextPathLabel - > setText ( QString ( " Text (%1): " ) . arg ( locPath . c_str ( ) ) ) ;
plString itemText = plString : : Format ( " Text (%s): " , locPath . c_str ( ) ) ;
SetDlgItemTextW ( gEditDlg , IDC_LOCPATH , itemText . ToWchar ( ) ) ;
plString ageName , setName , elementName , elementLanguage ;
plString ageName , setName , elementName , elementLanguage ;
SplitLocalizationPath ( locPath , ageName , setName , elementName , elementLanguage ) ;
SplitLocalizationPath ( locPath , ageName , setName , elementName , elementLanguage ) ;
// now make sure they've drilled down deep enough to enable the dialog
// now make sure they've drilled down deep enough to enable the dialog
if ( elementLanguage . IsEmpty ( ) ) // not deep enough
if ( elementLanguage . IsEmpty ( ) ) // not deep enough
EnableDlg ( FALSE ) ;
EnableEdit ( false ) ;
else
else
{
{
EnableDlg ( TRUE ) ;
EnableEdit ( true ) ;
plString key = plString : : Format ( " %s.%s.%s " , ageName . c_str ( ) , setName . c_str ( ) , elementName . c_str ( ) ) ;
plString key = plString : : Format ( " %s.%s.%s " , ageName . c_str ( ) , setName . c_str ( ) , elementName . c_str ( ) ) ;
plString elementText = pfLocalizationDataMgr : : Instance ( ) . GetElementPlainTextData ( key , elementLanguage ) ;
plString elementText = pfLocalizationDataMgr : : Instance ( ) . GetElementPlainTextData ( key , elementLanguage ) ;
SetDlgItemTextW ( gEditDlg , IDC_LOCALIZATIONTEXT , elementText . ToWcha r( ) ) ;
fUI - > fLocalizationText - > setPlainText ( elementText . c_st r( ) ) ;
}
}
// now to setup the add/delete buttons
// now to setup the add/delete buttons
if ( ! elementLanguage . IsEmpty ( ) ) // they have selected a language
if ( ! elementLanguage . IsEmpty ( ) ) // they have selected a language
{
{
SetDlgItemText ( gEditDlg , IDC_ADD , L " Add Localization " ) ;
fEditMode = kEditLocalization ;
EnableWindow ( GetDlgItem ( gEditDlg , IDC_ADD ) , TRUE ) ;
fUI - > fAddButton - > setText ( tr ( " Add Localization " ) ) ;
SetDlgItemText ( gEditDlg , IDC_DELETE , L " Delete Localization " ) ;
fUI - > fAddButton - > setEnabled ( true ) ;
if ( elementLanguage ! = " English " ) // don't allow them to delete the default language
fUI - > fDeleteButton - > setText ( tr ( " Delete Localization " ) ) ;
EnableWindow ( GetDlgItem ( gEditDlg , IDC_DELETE ) , TRUE ) ;
else
// don't allow them to delete the default language
EnableWindow ( GetDlgItem ( gEditDlg , IDC_DELETE ) , FALSE ) ;
fUI - > fDeleteButton - > setEnabled ( elementLanguage ! = " English " ) ;
}
}
else // they have selected something else
else // they have selected something else
{
{
SetDlgItemText ( gEditDlg , IDC_ADD , L " Add Element " ) ;
fEditMode = kEditElement ;
EnableWindow ( GetDlgItem ( gEditDlg , IDC_ADD ) , TRUE ) ;
fUI - > fAddButton - > setText ( tr ( " Add Element " ) ) ;
SetDlgItemText ( gEditDlg , IDC_DELETE , L " Delete Element " ) ;
fUI - > fAddButton - > setEnabled ( true ) ;
fUI - > fDeleteButton - > setText ( tr ( " Delete Element " ) ) ;
if ( ! elementName . IsEmpty ( ) ) // they have selected an individual element
if ( ! elementName . IsEmpty ( ) ) // they have selected an individual element
{
{
std : : vector < plString > elementNames = pfLocalizationDataMgr : : Instance ( ) . GetElementList ( ageName , setName ) ;
std : : vector < plString > elementNames = pfLocalizationDataMgr : : Instance ( ) . GetElementList ( ageName , setName ) ;
if ( elementNames . size ( ) > 1 ) // they can't delete the only subtitle in a set
EnableWindow ( GetDlgItem ( gEditDlg , IDC_DELETE ) , TRUE ) ;
// they can't delete the only subtitle in a set
else
fUI - > fDeleteButton - > setEnabled ( elementNames . size ( ) > 1 ) ;
EnableWindow ( GetDlgItem ( gEditDlg , IDC_DELETE ) , FALSE ) ;
}
}
else
else
EnableWindow ( GetDlgItem ( gEditDlg , IDC_DELETE ) , FALSE ) ;
fUI - > fDeleteButton - > setEnabled ( false ) ;
}
}
}
}
BOOL HandleCommandMessage ( HWND hWnd , UINT msg , WPARAM wParam , LPARAM lParam )
void EditDialog : : EnableEdit ( bool enable )
{
{
int wmID , wmEvent ;
if ( ! enable )
wmID = LOWORD ( wParam ) ;
fUI - > fLocalizationText - > setPlainText ( " " ) ;
wmEvent = HIWORD ( wParam ) ;
switch ( wmEvent )
fUI - > fLocalizationText - > setEnabled ( enable ) ;
}
void EditDialog : : closeEvent ( QCloseEvent * event )
{
if ( fCurrentSavePath . isEmpty ( ) ) // no data open
{
{
case BN_CLICKED :
event - > accept ( ) ;
switch ( wmID )
return ;
}
SaveLocalizationText ( ) ; // make sure any changed text is saved to the manager
QMessageBox : : StandardButton result = QMessageBox : : question ( this , tr ( " Save Changes " ) ,
tr ( " Do you wish to save your changes? " ) ,
QMessageBox : : Yes | QMessageBox : : No | QMessageBox : : Cancel ) ;
if ( result = = QMessageBox : : Yes )
SaveToDirectory ( ) ;
if ( result = = QMessageBox : : Cancel )
event - > ignore ( ) ;
else
event - > accept ( ) ;
}
void EditDialog : : OpenDataDirectory ( )
{
QString path = QFileDialog : : getExistingDirectory ( this ,
tr ( " Select a localization data directory: " ) ,
QDir : : current ( ) . absolutePath ( ) ,
QFileDialog : : ShowDirsOnly | QFileDialog : : ReadOnly ) ;
if ( ! path . isEmpty ( ) )
{
plWaitCursor waitCursor ( this ) ;
pfLocalizationMgr : : Shutdown ( ) ;
fCurrentSavePath = path ;
pfLocalizationMgr : : Initialize ( fCurrentSavePath . toUtf8 ( ) . constData ( ) ) ;
fUI - > fLocalizationTree - > clear ( ) ;
fUI - > fLocalizationTree - > LoadData ( " " ) ;
SetTitle ( path ) ;
fUI - > fSaveCurrentAction - > setEnabled ( true ) ;
fUI - > fSaveOtherAction - > setEnabled ( true ) ;
}
}
void EditDialog : : SaveToCurrent ( )
{
SaveLocalizationText ( ) ; // make sure any changed text is saved to the manager
// save it to our current directory
QMessageBox : : StandardButton result = QMessageBox : : question ( this , tr ( " Save to Current Directory " ) ,
tr ( " Are you sure you want to save to the current directory? Current data will be overwritten! " ) ,
QMessageBox : : Yes | QMessageBox : : No | QMessageBox : : Cancel ) ;
if ( result = = QMessageBox : : Yes )
{
plWaitCursor waitCursor ( this ) ;
pfLocalizationDataMgr : : Instance ( ) . WriteDatabaseToDisk ( fCurrentSavePath . toUtf8 ( ) . constData ( ) ) ;
}
else if ( result = = QMessageBox : : No )
SaveToDirectory ( ) ;
// and if it's cancel we don't do anything
}
void EditDialog : : SaveToDirectory ( )
{
SaveLocalizationText ( ) ; // make sure any changed text is saved to the manager
QString path = QFileDialog : : getExistingDirectory ( this ,
tr ( " Select a directory to save the localization data to: " ) ,
fCurrentSavePath , QFileDialog : : ShowDirsOnly ) ;
// save it to a new directory
if ( ! path . isEmpty ( ) )
{
plWaitCursor waitCursor ( this ) ;
fCurrentSavePath = path ;
SetTitle ( path ) ;
pfLocalizationDataMgr : : Instance ( ) . WriteDatabaseToDisk ( fCurrentSavePath . toUtf8 ( ) . constData ( ) ) ;
}
}
void EditDialog : : LocPathChanged ( QTreeWidgetItem * current , QTreeWidgetItem * )
{
SaveLocalizationText ( ) ; // save any current changes to the database
LoadLocalization ( fUI - > fLocalizationTree - > CurrentPath ( ) ) ;
}
void EditDialog : : AddClicked ( )
{
SaveLocalizationText ( ) ; // save any current changes to the database
if ( fEditMode = = kEditElement )
{
plAddElementDlg dlg ( fCurrentLocPath , this ) ;
if ( dlg . DoPick ( ) )
{
{
case IDC_ADD :
plString path = dlg . GetValue ( ) ; // path is age.set.name
if ( ! pfLocalizationDataMgr : : Instance ( ) . AddElement ( path ) )
{
{
SaveLocalizationText ( ) ; // save any current changes to the database
QMessageBox : : critical ( this , tr ( " Error " ) ,
tr ( " Couldn't add new element because one already exists with that name! " ) ) ;
plString buttonText ;
wchar_t buff [ 256 ] ;
GetDlgItemText ( gEditDlg , IDC_ADD , buff , 256 ) ;
buttonText = plString : : FromWchar ( buff ) ;
if ( buttonText = = " Add Element " )
{
plAddElementDlg dlg ( gCurrentPath ) ;
if ( dlg . DoPick ( gEditDlg ) )
{
plString path = dlg . GetValue ( ) ; // path is age.set.name
if ( ! pfLocalizationDataMgr : : Instance ( ) . AddElement ( path ) )
MessageBox ( gEditDlg , L " Couldn't add new element because one already exists with that name! " , L " Error " , MB_ICONERROR | MB_OK ) ;
else
{
gCurrentPath = " " ;
plLocTreeView : : ClearTreeView ( gTreeView ) ;
plLocTreeView : : FillTreeViewFromData ( gTreeView , path ) ;
UpdateEditDlg ( path ) ;
}
}
}
else if ( buttonText = = " Add Localization " )
{
plAddLocalizationDlg dlg ( gCurrentPath ) ;
if ( dlg . DoPick ( gEditDlg ) )
{
plString newLanguage = dlg . GetValue ( ) ;
plString ageName , setName , elementName , elementLanguage ;
SplitLocalizationPath ( gCurrentPath , ageName , setName , elementName , elementLanguage ) ;
plString key = plString : : Format ( " %s.%s.%s " , ageName . c_str ( ) , setName . c_str ( ) , elementName . c_str ( ) ) ;
if ( ! pfLocalizationDataMgr : : Instance ( ) . AddLocalization ( key , newLanguage ) )
MessageBox ( gEditDlg , L " Couldn't add additional localization! " , L " Error " , MB_ICONERROR | MB_OK ) ;
else
{
plString path = plString : : Format ( " %s.%s " , key . c_str ( ) , newLanguage . c_str ( ) ) ;
gCurrentPath = " " ;
plLocTreeView : : ClearTreeView ( gTreeView ) ;
plLocTreeView : : FillTreeViewFromData ( gTreeView , path ) ;
UpdateEditDlg ( path ) ;
}
}
}
return FALSE ;
}
}
case IDC_DELETE :
else
{
fCurrentLocPath = " " ;
fUI - > fLocalizationTree - > clear ( ) ;
fUI - > fLocalizationTree - > LoadData ( path ) ;
LoadLocalization ( path ) ;
}
}
}
else if ( fEditMode = = kEditLocalization )
{
plAddLocalizationDlg dlg ( fCurrentLocPath , this ) ;
if ( dlg . DoPick ( ) )
{
plString newLanguage = dlg . GetValue ( ) ;
plString ageName , setName , elementName , elementLanguage ;
SplitLocalizationPath ( fCurrentLocPath , ageName , setName , elementName , elementLanguage ) ;
plString key = plString : : Format ( " %s.%s.%s " , ageName . c_str ( ) , setName . c_str ( ) , elementName . c_str ( ) ) ;
if ( ! pfLocalizationDataMgr : : Instance ( ) . AddLocalization ( key , newLanguage ) )
QMessageBox : : critical ( this , tr ( " Error " ) , tr ( " Couldn't add additional localization! " ) ) ;
else
{
{
SaveLocalizationText ( ) ; // save any current changes to the database
plString path = plString : : Format ( " %s.%s " , key . c_str ( ) , newLanguage . c_str ( ) ) ;
fCurrentLocPath = " " ;
plString messageText = plString : : Format ( " Are you sure that you want to delete %s? " , gCurrentPath . c_str ( ) ) ;
fUI - > fLocalizationTree - > clear ( ) ;
int res = MessageBoxW ( gEditDlg , messageText . ToWchar ( ) , L " Delete " , MB_ICONQUESTION | MB_YESNO ) ;
fUI - > fLocalizationTree - > LoadData ( path ) ;
if ( res = = IDYES )
LoadLocalization ( path ) ;
{
plString buttonText ;
wchar_t buff [ 256 ] ;
GetDlgItemText ( gEditDlg , IDC_DELETE , buff , 256 ) ;
buttonText = plString : : FromWchar ( buff ) ;
if ( buttonText = = " Delete Element " )
{
if ( ! pfLocalizationDataMgr : : Instance ( ) . DeleteElement ( gCurrentPath ) )
MessageBox ( gEditDlg , L " Couldn't delete element! " , L " Error " , MB_ICONERROR | MB_OK ) ;
else
{
plString path = gCurrentPath ;
gCurrentPath = " " ;
plLocTreeView : : ClearTreeView ( gTreeView ) ;
plLocTreeView : : FillTreeViewFromData ( gTreeView , path ) ;
UpdateEditDlg ( path ) ;
}
}
else if ( buttonText = = " Delete Localization " )
{
plString ageName , setName , elementName , elementLanguage ;
SplitLocalizationPath ( gCurrentPath , ageName , setName , elementName , elementLanguage ) ;
plString key = plString : : Format ( " %s.%s.%s " , ageName . c_str ( ) , setName . c_str ( ) , elementName . c_str ( ) ) ;
if ( ! pfLocalizationDataMgr : : Instance ( ) . DeleteLocalization ( key , elementLanguage ) )
MessageBox ( gEditDlg , L " Couldn't delete localization! " , L " Error " , MB_ICONERROR | MB_OK ) ;
else
{
plString path = gCurrentPath ;
gCurrentPath = " " ;
plLocTreeView : : ClearTreeView ( gTreeView ) ;
plLocTreeView : : FillTreeViewFromData ( gTreeView , path ) ;
UpdateEditDlg ( path ) ;
}
}
}
}
}
return FALSE ;
}
}
}
}
return ( BOOL ) DefWindowProc ( hWnd , msg , wParam , lParam ) ;
}
}
// our dialog's window procedure
void EditDialog : : DeleteClicked ( )
INT_PTR CALLBACK EditDlgProc ( HWND hWnd , UINT msg , WPARAM wParam , LPARAM lParam )
{
{
switch ( msg )
SaveLocalizationText ( ) ; // save any current changes to the database
QMessageBox : : StandardButton reply = QMessageBox : : question ( this , tr ( " Delete " ) ,
tr ( " Are you sure that you want to delete %1? " ) . arg ( fCurrentLocPath . c_str ( ) ) ) ;
if ( reply = = QMessageBox : : Yes )
{
{
case WM_INITDIALOG :
if ( fEditMode = = kEditElement )
{
if ( ! pfLocalizationDataMgr : : Instance ( ) . DeleteElement ( fCurrentLocPath ) )
QMessageBox : : critical ( this , tr ( " Error " ) , tr ( " Couldn't delete element! " ) ) ;
else
{
plString path = fCurrentLocPath ;
fCurrentLocPath = " " ;
fUI - > fLocalizationTree - > clear ( ) ;
fUI - > fLocalizationTree - > LoadData ( path ) ;
LoadLocalization ( path ) ;
}
}
else if ( fEditMode = = kEditLocalization )
{
{
gEditDlg = hWnd ;
plString ageName , setName , elementName , elementLanguage ;
EnableDlg ( FALSE ) ;
SplitLocalizationPath ( fCurrentLocPath , ageName , setName , elementName , elementLanguage ) ;
plString key = plString : : Format ( " %s.%s.%s " , ageName . c_str ( ) , setName . c_str ( ) , elementName . c_str ( ) ) ;
if ( ! pfLocalizationDataMgr : : Instance ( ) . DeleteLocalization ( key , elementLanguage ) )
QMessageBox : : critical ( this , tr ( " Error " ) , tr ( " Couldn't delete localization! " ) ) ;
else
{
plString path = fCurrentLocPath ;
fCurrentLocPath = " " ;
fUI - > fLocalizationTree - > clear ( ) ;
fUI - > fLocalizationTree - > LoadData ( path ) ;
LoadLocalization ( path ) ;
}
}
}
break ;
case WM_COMMAND :
return HandleCommandMessage ( hWnd , msg , wParam , lParam ) ;
}
}
}
return FALSE ;
// split a subtitle path up into its component parts
void SplitLocalizationPath ( const plString & path , plString & ageName ,
plString & setName , plString & locName , plString & locLanguage )
{
ageName = setName = locName = locLanguage = " " ;
std : : vector < plString > tokens = path . Tokenize ( " . " ) ;
if ( tokens . size ( ) > = 1 )
ageName = tokens [ 0 ] ;
if ( tokens . size ( ) > = 2 )
setName = tokens [ 1 ] ;
if ( tokens . size ( ) > = 3 )
locName = tokens [ 2 ] ;
if ( tokens . size ( ) > = 4 )
locLanguage = tokens [ 3 ] ;
}
}