Windows2003-3790/sdktools/restools/rltools/common/toktrans.c
2020-09-30 16:53:55 +02:00

1018 lines
25 KiB
C

#include <windows.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <ctype.h>
#include <tchar.h>
#ifdef RLDOS
#include "dosdefs.h"
#else
#include "windefs.h"
#endif
#include "restok.h"
#include "resread.h"
#include "toklist.h"
#include "commbase.h"
#define MAXLINE 1024
#define MAXTERM 512
extern UCHAR szDHW[];
extern PROJDATA gProj;
extern MSTRDATA gMstr;
#ifdef WIN32
extern HINSTANCE hInst; // Instance of the main window
#else
extern HWND hInst; // Instance of the main window
#endif
static fUnicodeGlossary = FALSE;
static long GetGlossaryIndex( FILE *, TCHAR, long []);
static void ParseGlossEntry( TCHAR *, TCHAR *, TCHAR[], TCHAR *, TCHAR[]);
static void ParseTextHotKeyToBuf( TCHAR *, TCHAR, TCHAR *);
static void ParseBufToTextHotKey( TCHAR *, TCHAR[], TCHAR *);
static WORD NormalizeIndex( TCHAR);
static int MyPutGlossStr( TCHAR *, FILE *);
static TCHAR *MyGetGlossStr( TCHAR *, int, FILE *);
static void BuildGlossEntry( TCHAR *, TCHAR *, TCHAR, TCHAR *, TCHAR);
static BOOL NotAMember( TRANSLIST *, TCHAR *);
FILE * OpenGlossary( CHAR *szGlossFile, CHAR chAccessType)
{
CHAR * szRW[4] = {"rb", "rt", "wb", "wt"};
int nRW = 0; // assume access type is 'r' (read)
FILE *fpRC = NULL;
if ( chAccessType == 'w' ) // is access type 'w' (write)?
{
nRW = fUnicodeGlossary ? 2 : 3; // yes (Unicode file or not?)
}
fpRC = fopen( szGlossFile, szRW[ nRW]);
if ( fpRC && chAccessType == 'r' )
{
USHORT usMark = GetWord( fpRC, NULL);
if ( usMark == 0xfeff )
{
fUnicodeGlossary = TRUE; // it's a Unicode text file
}
else if ( usMark == 0xfffe )
{
QuitA( IDS_WRONGENDIAN, szGlossFile, NULL);
}
else
{
fclose( fpRC);
fpRC = fopen( szGlossFile, szRW[ ++nRW]); // it's an ANSI text file
}
}
return( fpRC);
}
/**
*
*
* Function:
*
*
* Arguments:
*
* Returns:
*
* Errors Codes:
*
* History:
*
*
**/
int MakeGlossIndex( LONG * lFilePointer)
{
TCHAR szGlossEntry[MAXLINE] = TEXT("");
WORD iCurrent = 0;
LONG lFPointer = -1;
FILE *pFile = NULL;
pFile = OpenGlossary( gProj.szGlo, 'r');
if ( pFile == NULL )
{
return( 1);
}
// Glossaries some times have this bogus header at the begining.
// which we want to skip if it exists
if ( ! MyGetGlossStr( szGlossEntry, MAXLINE, pFile) )
{
// Error during first read from the glossary.
fclose( pFile);
return( 1);
}
lFPointer = ftell( pFile);
// check for glossary header
if ( lstrlen( szGlossEntry) >= 7 )
{
// lstrcpy( (TCHAR *)szDHW, szGlossEntry);
// szDHW[ MEMSIZE( 7)] = szDHW[ MEMSIZE( 7) + 1] = '\0';
// CharLower( (TCHAR *)szDHW);
//
// if ( lstrcmp( (TCHAR *)szDHW, TEXT("english")) == 0 )
if ( CompareStringW( MAKELCID( gMstr.wLanguageID, SORT_DEFAULT),
SORT_STRINGSORT | NORM_IGNORECASE,
szGlossEntry,
7,
TEXT("ENGLISH"),
7) == 2 )
{
lFPointer = ftell (pFile);
if ( ! MyGetGlossStr( szGlossEntry, MAXLINE, pFile) )
{
fclose( pFile);
return (1);
}
}
}
// now assume we are at the correct location in glossary
// file to begin generating the index, we want to save
// this location
lFilePointer[0] = lFPointer;
// glossary file is sorted so, any non letter items
// in the glossary would be first. Index into this location
// using the 1st position
// 1st lets make sure we have non letters items in
// the glossary
// now skip ( if any ) the non letter entries in the glossary
while( (WORD) szGlossEntry[0] < (WORD) TEXT('A' ) )
{
if ( ! MyGetGlossStr( szGlossEntry, MAXLINE, pFile) )
{
fclose( pFile);
return( 1);
}
}
// now position at alpha characters
iCurrent = NormalizeIndex( szGlossEntry[0] );
// now we read through the remaining glossary entries
// and save the offsets for each index as we go
do
{
if ( NormalizeIndex( szGlossEntry[0] ) > iCurrent )
{
// we passed the region for our current index
// so save the location, and move to the next index.
// note we may be skiping indexs,
lFilePointer[ iCurrent] = lFPointer;
iCurrent = NormalizeIndex( szGlossEntry[0] );
}
lFPointer = ftell( pFile );
// otherwise the current index is valied for this
// section of the glossary indexes, so just continue
} while ( MyGetGlossStr( szGlossEntry, MAXLINE, pFile) );
fclose( pFile);
return( 0);
}
/**
*
*
* Function: TransString
* Builds a circular linked list containing all translations of a string.
* The first entry in the list is the untranslated string.
*
* Arguments:
* fpGlossFile, handle to open glossary file
* szKeyText, string with the text to build translation table
* szCurrentText, text currently in the box.
* ppTransList, pointer to a pointer to a node in a circular linked list
* lFilePointer, pointer to index table for glossary file
*
* Returns:
* number of nodes in list
*
* Errors Codes:
*
* History:
* Recoded by SteveBl, 3/92
*
**/
/* Translate the string, if possible. */
int TransString(
TCHAR *szKeyText,
TCHAR *szCurrentText,
TRANSLIST **ppTransList,
LONG *lFilePointer)
{
int n = 0;
long lFileIndex;
TRANSLIST **ppCurrentPointer;
static TCHAR szGlossEntry[MAXLINE];
static TCHAR szEngText[260];
static TCHAR szIntlText[260];
TCHAR *szCurText = NULL;
TCHAR cEngHotKey = TEXT('\0');
TCHAR cIntlHotKey = TEXT('\0');
TCHAR cCurHotKey = TEXT('\0');
FILE *fpGlossFile = NULL;
// *Is* there a glossary file?
if ( (fpGlossFile = OpenGlossary( gProj.szGlo, 'r')) == NULL )
{
return( 0);
}
// FIRST let's erase the list
if ( *ppTransList )
{
(*ppTransList)->pPrev->pNext = NULL; // so we can find the end of the list
}
while ( *ppTransList )
{
TRANSLIST *pTemp;
pTemp = *ppTransList;
*ppTransList = pTemp->pNext;
RLFREE( pTemp->sz);
RLFREE( pTemp);
}
ppCurrentPointer = ppTransList;
// DONE removing the list
// Now make the first node (which is the untranslated string)
{
TCHAR * psz;
psz = (TCHAR *)FALLOC( MEMSIZE( lstrlen( szCurrentText) + 1));
lstrcpy( psz,szCurrentText);
*ppTransList = ( TRANSLIST *)FALLOC( sizeof( TRANSLIST));
(*ppTransList)->pPrev = (*ppTransList)->pNext = *ppTransList;
(*ppTransList)->sz = psz;
ppCurrentPointer = ppTransList;
n++;
}
szCurText = (TCHAR *)FALLOC( MEMSIZE( lstrlen( szKeyText) + 1) );
ParseBufToTextHotKey( szCurText, &cCurHotKey, szKeyText);
lFileIndex = GetGlossaryIndex( fpGlossFile, szCurText[0], lFilePointer);
fseek (fpGlossFile, lFileIndex, SEEK_SET);
while ( TRUE)
{
if ( ! MyGetGlossStr( szGlossEntry, MAXLINE, fpGlossFile) )
{
// Reached end of glossary file
RLFREE( szCurText);
fclose( fpGlossFile);
return n;
}
ParseGlossEntry( szGlossEntry,
szEngText,
&cEngHotKey,
szIntlText,
&cIntlHotKey);
// make comparision, using text, and hot keys
// if ( ( ! lstrcmp( szCurText, szEngText )) && cCurHotKey == cEngHotKey )
if ( CompareStringW( MAKELCID( gMstr.wLanguageID, SORT_DEFAULT),
SORT_STRINGSORT,
szCurText,
-1,
szEngText,
-1) == 2
&& cCurHotKey == cEngHotKey )
{
TCHAR * psz;
static TCHAR szTemp[ MAXINPUTBUFFER];
// we have a match, put translated text into token
if ( cIntlHotKey )
{
ParseTextHotKeyToBuf( szIntlText, cIntlHotKey, szTemp);
}
else
{
lstrcpy( szTemp, szIntlText);
}
if ( NotAMember( *ppTransList, szTemp) )
{
// add matched glossary text to circular list of matches
psz = (TCHAR *) FALLOC( MEMSIZE( lstrlen( szTemp) + 1));
lstrcpy( psz,szTemp);
(*ppCurrentPointer)->pNext = (TRANSLIST *)
FALLOC( sizeof( TRANSLIST));
((*ppCurrentPointer)->pNext)->pPrev = *ppCurrentPointer;
ppCurrentPointer = (TRANSLIST **)&((*ppCurrentPointer)->pNext);
(*ppCurrentPointer)->pPrev->pNext = *ppCurrentPointer;
(*ppCurrentPointer)->pNext = *ppTransList;
(*ppTransList)->pPrev = *ppCurrentPointer;
(*ppCurrentPointer)->sz = psz;
++n;
}
}
else
{
// can we terminate search?
// if( lstrcmpi( szEngText, szCurText ) > 0 )
if ( CompareStringW( MAKELCID( gMstr.wLanguageID, SORT_DEFAULT),
SORT_STRINGSORT,
szEngText,
-1,
szCurText,
-1) == 3 )
{
// went past index section
RLFREE( szCurText);
fclose( fpGlossFile);
return( n);
}
}
}
RLFREE( szCurText);
fclose( fpGlossFile);
return( n);
} // TransString
/**
*
*
* Function: NormalizeIndex
*
*
* Arguments:
*
* Returns:
*
* Errors Codes:
*
* History:
*
*
**/
static WORD NormalizeIndex( TCHAR chIndex )
{
TCHAR chTmp = chIndex;
CharLowerBuff( &chTmp, 1);
return( (chTmp != TEXT('"') && chTmp >= TEXT('a') && chTmp <= TEXT('z'))
? chTmp - TEXT('a') + 1
: 0);
}
/*
* Function:NotAMember
*
* Arguments:
* pList, pointer to a TRANSLIST node
* sz, string to find
*
* Returns:
* TRUE if not found in the list else FALSE
*
* History:
* 3/92, implemented SteveBl
**/
static BOOL NotAMember( TRANSLIST *pList, TCHAR *sz)
{
TRANSLIST *pCurrent = pList;
if ( ! pList )
{
return( TRUE); // empty list
}
do
{
// if ( lstrcmp( sz, pCurrent->sz) == 0 )
if ( CompareStringW( MAKELCID( gMstr.wLanguageID, SORT_DEFAULT),
SORT_STRINGSORT,
sz,
-1,
pCurrent->sz,
-1) == 2 )
{
return( FALSE); // found in list
}
pCurrent = pCurrent->pNext;
}while ( pList != pCurrent );
return( TRUE); // not found
}
/**
*
*
* Function:
*
*
* Arguments:
*
* Returns:
*
* Errors Codes:
*
* History:
*
*
**/
static void ParseGlossEntry(
TCHAR szGlossEntry[],
TCHAR szEngText[],
TCHAR cEngHotKey[1],
TCHAR szIntlText[],
TCHAR cIntlHotKey[1])
{
WORD wIndex, wIndex2;
// format is:
// <eng text><tab><eng hot key><tab><loc text><tab><loc hot key>
// Any field could be null and if there aren't the right amount of
// tabs we'll just assume that the remaining fields are empty.
wIndex=wIndex2=0;
// first get the english text
while ( szGlossEntry[wIndex2] != TEXT('\t')
&& szGlossEntry[wIndex2] != TEXT('\0') )
{
szEngText[ wIndex++] = szGlossEntry[ wIndex2++];
}
szEngText[wIndex]=TEXT('\0');
if ( szGlossEntry[ wIndex2] == TEXT('\t') )
{
++wIndex2; // skip the tab
}
// now get the eng hot key
if ( szGlossEntry[wIndex2] != TEXT('\t')
&& szGlossEntry[wIndex2] != TEXT('\0') )
{
*cEngHotKey = szGlossEntry[wIndex2++];
}
else
{
*cEngHotKey = TEXT('\0');
}
while ( szGlossEntry[ wIndex2] != TEXT('\t')
&& szGlossEntry[ wIndex2] != TEXT('\0') )
{
++wIndex2; // make sure the hot key field doesn't hold more than one char
}
if ( szGlossEntry[ wIndex2] == TEXT('\t') )
{
++wIndex2; // skip the tab
}
wIndex = 0;
// now get the intl text
while ( szGlossEntry[ wIndex2] != TEXT('\t')
&& szGlossEntry[ wIndex2] != TEXT('\0') )
{
szIntlText[wIndex++]=szGlossEntry[wIndex2++];
}
szIntlText[wIndex]='\0';
if ( szGlossEntry[ wIndex2] == TEXT('\t') )
{
++wIndex2; // skip the tab
}
// now get the intl hot key
if ( szGlossEntry[ wIndex2] != TEXT('\t')
&& szGlossEntry[ wIndex2] != TEXT('\0') )
{
*cIntlHotKey = szGlossEntry[ wIndex2++];
}
else
{
*cIntlHotKey = TEXT('\0');
}
}
/**
*
*
* Function:
*
*
* Arguments:
*
* Returns:
*
* Errors Codes:
*
* History:
*
*
**/
static void ParseBufToTextHotKey(
TCHAR *szText,
TCHAR cHotKey[1],
TCHAR *szBuf)
{
WORD wIndexBuf = 0;
WORD wIndexText = 0;
*cHotKey = TEXT('\0');
while( szBuf[ wIndexBuf] )
{
if ( szBuf[ wIndexBuf ] == TEXT('&') )
{
*cHotKey = szBuf[ ++wIndexBuf];
}
else
{
szText[ wIndexText++] = szBuf[ wIndexBuf++];
}
}
szText[ wIndexText] = TEXT('\0');
}
/**
*
*
* Function:
*
*
* Arguments:
*
* Returns:
*
* Errors Codes:
*
* History:
*
*
**/
static void ParseTextHotKeyToBuf(
TCHAR *szText,
TCHAR cHotKey,
TCHAR *szBuf )
{
WORD wIndexBuf = 0;
WORD wIndexText = 0;
// TCHAR cTmp;
while ( szText[ wIndexText] )
{
// cTmp = szText[ wIndexText];
//
// CharUpperBuff( &cTmp, 1);
//
// if ( cTmp == cHotKey )
if ( szText[ wIndexText] == cHotKey )
{
szBuf[ wIndexBuf++] = TEXT('&');
szBuf[ wIndexBuf++] = szText[ wIndexText++];
break;
}
else
{
szBuf[ wIndexBuf++] = szText[ wIndexText++];
}
}
// copy remaining string
while( szText[ wIndexText] )
{
szBuf[ wIndexBuf++] = szText[ wIndexText++];
}
szBuf[ wIndexBuf] = TEXT('\0');
}
static long GetGlossaryIndex(
FILE *fpGlossFile,
TCHAR c,
long *lGlossaryIndex )
{
int i = 0;
TCHAR cTmp = c;
CharLowerBuff( &cTmp, 1);
if ( cTmp >= TEXT('a')
&& cTmp <= TEXT('z') )
{
i = NormalizeIndex( c );
return( lGlossaryIndex[ i > 0 ? i - 1 : 0]);
}
else
{
return( 0);
}
}
/*******************************************************************************
* PROCEDURE: BuildGlossEntry
* Builds a glossary entry line.
*
* Parameters:
* sz, line buffer
* sz1, untranslated text
* c1, untranslated hot key (or 0 if no hot key)
* sz2, translated text
* c2, translated hot key (or 0 if no hot key)
*
* Returns:
* nothing. sz contains the line. (assumes there is room in the buffer)
*
* History:
* 3/93 - initial implementation - SteveBl
*******************************************************************************/
static void BuildGlossEntry(
TCHAR *sz,
TCHAR *sz1,
TCHAR c1,
TCHAR *sz2,
TCHAR c2)
{
*sz = TEXT('\0');
wsprintf( sz, TEXT("%s\t%c\t%s\t%c"), sz1, c1, sz2, c2);
}
/******************************************************************************
* PROCEDURE: AddTranslation
* Adds a translation to a glossary file.
*
* PARAMETERS:
* szGlossFile, path to the glossary
* szKey, untranslated text
* szTranslation, translated text
* lFilePointer, pointer to index hash table for glossary
*
* RETURNS:
* nothing. Key is added to glossary if no errors are encountered else
* file is left unchanged.
*
* COMMENTS:
* rebuilds the global pointer list lFilePointer
*
* HISTORY: *
* 3/92 - initial implementation - SteveBl
******************************************************************************/
void AddTranslation(
TCHAR *szKey,
TCHAR *szTranslation,
LONG *lFilePointer)
{
// DBCS begin
TCHAR szCurText [520];
TCHAR szTransText [520];
// DBCS end
TCHAR cTransHot = TEXT('\0');
TCHAR cCurHotKey = TEXT('\0');
CHAR szTempFileName [255];
FILE *fTemp = NULL;
FILE *fpGlossFile = NULL;
TCHAR szTempText [MAXLINE];
// DBCS begin
TCHAR szNewText [MAXLINE * 2];
// DBCS end
TCHAR *r = NULL;
TCHAR chTmp = TEXT('\0');
MyGetTempFileName( 0, "", 0, szTempFileName);
if ( (fTemp = OpenGlossary( szTempFileName, 'w')) != NULL )
{
if ( fUnicodeGlossary )
{
fprintf( fTemp, "%hu", 0xfeff); // Mark new one as Unicode
}
ParseBufToTextHotKey( szCurText, &cCurHotKey, szKey);
ParseBufToTextHotKey( szTransText, &cTransHot, szTranslation);
BuildGlossEntry( szNewText,
szCurText,
cCurHotKey,
szTransText,
cTransHot);
// If the glossary file exists, get its first
// line. If it doesn't exist, we'll create it
// (via CopyFile) at the end of this function.
if ( (fpGlossFile = OpenGlossary( gProj.szGlo, 'r')) != NULL )
{
if ( (r = MyGetGlossStr( szTempText,
TCHARSIN( sizeof( szTempText)),
fpGlossFile)) )
{
// lstrcpy( (TCHAR *)szDHW, szTempText);
// szDHW[ MEMSIZE( 7)] = szDHW[ MEMSIZE( 7) + 1] = '\0';
// CharLower( (TCHAR *)szDHW);
//
// if ( lstrcmpi( (TCHAR *)szDHW, TEXT("ENGLISH")) == 0 )
if ( CompareStringW( MAKELCID( gMstr.wLanguageID, SORT_DEFAULT),
SORT_STRINGSORT | NORM_IGNORECASE,
szTempText,
7,
TEXT("ENGLISH"),
7) == 2 )
{
// skip first line
MyPutGlossStr( szTempText, fTemp);
r = MyGetGlossStr( szTempText, TCHARSIN( sizeof( szTempText)), fpGlossFile);
}
}
}
else
{
r = NULL;
}
// if ( r )
// {
// chTmp = szTempText[0];
// CharLowerBuff( &chTmp, 1);
// }
// else
// {
// chTmp = szTempText[0] = TEXT('\0');
// }
// // Does the new text begin with a letter?
//
// if ( chTmp >= TEXT('a') )
// {
// // begins with a letter, we need to find where to put it
//
// while ( r && chTmp < TEXT('a') )
while ( r && CompareStringW( MAKELCID( gMstr.wLanguageID, SORT_DEFAULT),
SORT_STRINGSORT,
szTempText,
-1,
szNewText,
-1) == 1 )
{
// skip the non letter section
MyPutGlossStr( szTempText, fTemp);
r = MyGetGlossStr( szTempText,
TCHARSIN( sizeof( szTempText)),
fpGlossFile);
// if ( (r = MyGetGlossStr( szTempText,
// TCHARSIN( sizeof( szTempText)),
// fpGlossFile)) )
// {
// chTmp = szTempText[0];
// CharLowerBuff( &chTmp, 1);
// }
}
// while ( r && _tcsicmp( szTempText, szNewText) < 0 )
// {
// // skip anything smaller than me
//
// MyPutGlossStr( szTempText, fTemp);
// r = MyGetGlossStr( szTempText, TCHARSIN( sizeof( szTempText)), fpGlossFile);
// }
// }
// else
// {
// // doesn't begin with a letter, we need to insert it before
// // the letter sections begin but it still must be sorted
//
// while ( r
// && chTmp < TEXT('a')
// && _tcsicmp( szTempText, szNewText) < 0 )
// {
// MyPutGlossStr( szTempText, fTemp);
//
// if ( (r = MyGetGlossStr( szTempText,
// TCHARSIN( sizeof( szTempText)),
// fpGlossFile)) )
// {
// chTmp = szTempText[0];
// CharLowerBuff( &chTmp, 1);
// }
// }
// }
MyPutGlossStr( szNewText, fTemp);
while ( r )
{
MyPutGlossStr( szTempText,fTemp);
r = MyGetGlossStr( szTempText, TCHARSIN( sizeof( szTempText)), fpGlossFile);
}
fclose( fTemp);
if ( fpGlossFile )
{
fclose( fpGlossFile);
}
// This call will create the glossary file
// if it didn't already exist.
if ( ! CopyFileA( szTempFileName, gProj.szGlo, FALSE) )
{
QuitA( IDS_COPYFILE_FAILED, szTempFileName, gProj.szGlo);
}
remove( szTempFileName);
MakeGlossIndex( lFilePointer);
}
else
{
QuitA( IDS_NO_TMP_GLOSS, szTempFileName, NULL);
}
}
/**
*
*
* Function: MyGetGlossStr
* Replaces C runtime fgets function.
*
* History:
* 5/92, Implemented. TerryRu.
*
*
**/
static TCHAR *MyGetGlossStr( TCHAR * ptszStr, int nCount, FILE * fIn)
{
int i = 0;
#ifdef RLRES32
// It this a Unicode glossary file?
TCHAR tCh = TEXT('\0');
if ( fUnicodeGlossary )
{
do // Yes
{
tCh = ptszStr[ i++] = (TCHAR)GetWord( fIn, NULL);
} while ( i < nCount && tCh != TEXT('\n') );
if ( tCh == TEXT('\0') || feof( fIn) )
{
return( NULL);
}
ptszStr[i] = TEXT('\0');
StripNewLineW( ptszStr);
}
else // No, it's an ANSI glossary file
{
if ( fgets( szDHW, DHWSIZE, fIn) != NULL )
{
StripNewLineA( szDHW);
_MBSTOWCS( ptszStr, szDHW, nCount, (UINT)-1);
}
else
{
return( NULL);
}
}
return( ptszStr);
#else //RLRES32
if ( fgets( ptszStr, nCount, fIn) )
{
StripNewLineA( ptszStr);
}
else
{
return( NULL);
}
#endif //RLRES32
}
/**
*
*
* Function: MyPutGlossStr
* Replaces C runtime fputs function.
* History:
* 6/92, Implemented. TerryRu.
*
*
**/
static int MyPutGlossStr( TCHAR * ptszStr, FILE * fOut)
{
#ifdef RLRES32
int i = 0;
// It this a Unicode glossary file?
if ( fUnicodeGlossary )
{
do // Yes
{
PutWord( fOut, ptszStr[i], NULL);
} while ( ptszStr[ i++] );
PutWord( fOut, TEXT('\r'), NULL);
PutWord( fOut, TEXT('\n'), NULL);
i += 2;
}
else // No, it's an ANSI glossary file
{
_WCSTOMBS( szDHW, ptszStr, DHWSIZE, lstrlen( ptszStr) + 1);
i = fputs( szDHW, fOut);
fputs( "\n", fOut);
}
#else //RLRES32
i = fputs( ptszStr, fOut);
fputs( "\n", fOut);
#endif //RLRES32
return(i);
}