2020-09-30 16:53:55 +02:00

514 lines
10 KiB
C++

//+-------------------------------------------------------------------------
//
// Microsoft Windows
//
// Copyright (C) Microsoft Corporation, 1997 - 1997
//
// File: bnstr.cpp
//
//--------------------------------------------------------------------------
//
// BNSTR.CPP
//
#include <stdarg.h>
#include <ctype.h>
#include "bnstr.h"
SZC BNSTR :: _pmt = "" ;
static SZ SzCopy(SZC szc)
{
SZ szNew = szc ? new char[::strlen(szc) + 1] : NULL;
return szNew ? ::strcpy(szNew, szc) : NULL;
}
BNSTR :: BNSTR ( SZC sz )
: _cchMax( 0 ),
_cchStr( 0 ),
_sz( const_cast<SZ>(_pmt) )
{
if ( sz )
{
Update( sz ) ;
}
}
BNSTR :: BNSTR ( const BNSTR & str )
: _cchMax( str._cchStr ),
_cchStr( str._cchStr ),
_sz( const_cast<SZ>(_pmt) )
{
if ( str._sz != _pmt )
{
_sz = ::SzCopy( str._sz ) ;
}
}
BNSTR :: ~ BNSTR ()
{
Reset() ;
}
void BNSTR :: Reset ()
{
DeleteSz() ;
_sz = const_cast<SZ>(_pmt) ;
_cchStr = 0 ;
_cchMax = 0 ;
}
// Protectively delete either the given string or the
// private string.
void BNSTR :: DeleteSz ()
{
if ( _sz != NULL && _sz != _pmt )
{
delete [] _sz ;
_sz = NULL ;
}
}
// Release the current buffer; reset the BNSTR.
SZ BNSTR::Transfer ()
{
SZ sz = _sz ;
_sz = NULL ;
Reset() ;
return sz = _pmt ? NULL : sz ;
}
// Give the current buffer to a new string, reset *this.
void BNSTR :: Transfer ( BNSTR & str )
{
str.Reset() ;
str._sz = _sz ;
str._cchMax = _cchMax ;
str._cchStr = _cchStr ;
_sz = NULL ;
Reset() ;
}
void BNSTR :: Trunc ( UINT cchLen )
{
if ( _sz == _pmt )
return ;
if ( cchLen < _cchStr )
_sz[cchLen] = 0 ;
}
// Update the pointed string. Since this routine is
// used by the assignment operator, it's written to allow
// for the new string being part of the old string.
bool BNSTR :: Update ( SZC sz )
{
bool bResult = true ;
UINT cch = sz ? ::strlen( sz ) : 0 ;
if ( cch > _cchMax )
{
SZ szNew = ::SzCopy( sz ) ;
if ( bResult = szNew != NULL )
{
DeleteSz() ;
_sz = szNew ;
_cchMax = _cchStr = cch ;
}
}
else
if ( cch == 0 )
{
Reset() ;
}
else
{
// REVIEW: this assumes that ::strcpy() handles overlapping regions correctly.
::strcpy( _sz, sz ) ;
_cchStr = cch ;
}
return bResult ;
}
// Grow the string. if 'cchNewSize' == 0, expand by 50%.
// If 'ppszNew' is given, store the new string there (for efficiency in
// Prefix); note that this requires that we reallocate.
bool BNSTR :: Grow ( UINT cchNewSize, SZ * ppszNew )
{
UINT cchNew = cchNewSize == 0
? (_cchMax + (_cchMax/2))
: cchNewSize ;
bool bResult = true ;
if ( cchNew > _cchMax || ppszNew )
{
SZ sz = new char [cchNew+1] ;
if ( bResult = sz != NULL )
{
_cchMax = cchNew ;
if ( ppszNew )
{
*ppszNew = sz ;
}
else
{
::strcpy( sz, _sz ) ;
DeleteSz() ;
_sz = sz ;
}
}
}
return bResult ;
}
// Expand the string to the given length; make it a blank, null terminated
// string.
bool BNSTR :: Pad ( UINT cchLength )
{
// Expand as necessary
if ( ! Grow( cchLength + 1 ) )
return false ;
// If expanding, pad the string with spaces.
while ( _cchStr < cchLength )
{
_sz[_cchStr++] = ' ' ;
}
// Truncate to proper length
_sz[_cchStr = cchLength] = 0 ;
return true ;
}
bool BNSTR :: Assign ( SZC szcData, UINT cchLen )
{
if ( ! Grow( cchLen + 1 ) )
return false ;
::memcpy( _sz, szcData, cchLen ) ;
_sz[cchLen] = 0 ;
_cchMax = _cchStr = cchLen ;
return true ;
}
SZC BNSTR :: Prefix ( SZC szPrefix )
{
assert( szPrefix != NULL ) ;
UINT cch = ::strlen( szPrefix ) ;
SZ sz ;
if ( ! Grow( _cchStr + cch + 1, & sz ) )
return NULL ;
::strcpy( sz, szPrefix ) ;
::strcpy( sz + cch, _sz ) ;
DeleteSz();
_cchStr += cch ;
return _sz = sz ;
}
SZC BNSTR :: Suffix ( SZC szSuffix )
{
if ( szSuffix )
{
UINT cch = ::strlen( szSuffix ) ;
if ( ! Grow( _cchStr + cch + 1 ) )
return NULL ;
::strcpy( _sz + _cchStr, szSuffix ) ;
_cchStr += cch ;
}
return *this ;
}
SZC BNSTR :: Suffix ( char chSuffix )
{
char rgch[2] ;
rgch[0] = chSuffix ;
rgch[1] = 0 ;
return Suffix( rgch );
}
INT BNSTR :: Compare ( SZC szSource, bool bIgnoreCase ) const
{
return bIgnoreCase
? ::stricmp( _sz, szSource )
: ::strcmp( _sz, szSource );
}
// Comparison
bool BNSTR :: operator == ( SZC szcSource ) const
{
return Compare( szcSource ) == 0 ;
}
bool BNSTR :: operator != ( SZC szSource ) const
{
return ! ((*this) == szSource) ;
}
char BNSTR :: operator [] ( UINT iChar ) const
{
assert( iChar < Length() ) ;
return _sz[iChar] ;
}
bool BNSTR :: Vsprintf ( SZC szcFmt, va_list valist )
{
// Attempt to "sprintf" the buffer. If it fails, reallocate
// a larger buffer and try again.
UINT cbMaxNew = ( _cchMax < 50
? 50
: _cchMax ) + 1 ;
do {
if ( ! Grow( cbMaxNew ) )
{
Reset() ;
return false ;
}
// Cause buffer to grow by 50% on the next cycle (if necessary)
cbMaxNew = 0 ;
// Problem: If the buffer is not big enough, _sz may not have a '\0', and Grow()
// will subsequently barf on the ::strcpy(). Quick fix:
_sz[_cchMax] = '\0';
} while ( ::_vsnprintf( _sz, _cchMax, szcFmt, valist ) < 0 ) ;
_sz[ _cchMax ] = '\0' ; // 'cause _vsnprintf, like _strncpy, doesn't always append this
// Update the string length member
_cchStr = ::strlen( _sz ) ;
return true ;
}
bool BNSTR :: Sprintf ( SZC szcFmt, ... )
{
va_list valist;
va_start( valist, szcFmt );
bool bOk = Vsprintf( szcFmt, valist ) ;
va_end( valist );
return bOk ;
}
bool BNSTR :: SprintfAppend ( SZC szcFmt, ... )
{
BNSTR strTemp ;
va_list valist;
va_start( valist, szcFmt );
bool bOk = strTemp.Vsprintf( szcFmt, valist ) ;
va_end( valist );
if ( bOk )
bOk = Suffix( strTemp ) != NULL ;
return bOk ;
}
// Cr/Lf expansion or contraction
bool BNSTR :: ExpandNl ()
{
UINT iCh ;
BNSTR str ;
Transfer( str ) ;
for ( iCh = 0 ; iCh < str.Length() ; iCh++ )
{
char ch = str[iCh];
if ( ch == '\n' )
{
if ( Suffix( '\r' ) == NULL )
return false ;
}
if ( Suffix( ch ) == NULL )
return false ;
}
return true ;
}
bool BNSTR :: ContractNl ()
{
UINT iCh ;
BNSTR str ;
Transfer( str ) ;
for ( iCh = 0 ; iCh < str.Length() ; iCh++ )
{
char ch = str[iCh];
if ( ch != '\r' )
{
if ( Suffix( ch ) == NULL )
return false ;
}
}
return true ;
}
static char rgchEsc [][2] =
{
{ '\a', 'a' },
{ '\b', 'b' },
{ '\f', 'f' },
{ '\n', 'n' },
{ '\r', 'r' },
{ '\t', 't' },
{ '\v', 'v' },
{ '\'', '\'' },
{ '\"', '\"' },
{ '\?', '\?' },
{ '\\', '\\' },
{ 0, 0 }
};
bool BNSTR :: ContractEscaped ()
{
UINT iCh ;
BNSTR str ;
Transfer( str ) ;
for ( iCh = 0 ; iCh < str.Length() ; iCh++ )
{
char ch = str[iCh];
if ( ch == '\\' && str.Length() - iCh > 1 )
{
char chEsc = 0;
for ( UINT ie = 0 ; rgchEsc[ie][0] ; ie++ )
{
if ( rgchEsc[ie][1] == ch )
break;
}
if ( chEsc = rgchEsc[ie][0] )
{
iCh++;
ch = chEsc;
}
}
if ( Suffix( ch ) == NULL )
return false ;
}
return true ;
}
// Convert unprintable characters to their escaped versions
bool BNSTR :: ExpandEscaped ()
{
UINT iCh ;
BNSTR str ;
Transfer( str ) ;
for ( iCh = 0 ; iCh < str.Length() ; iCh++ )
{
char ch = str[iCh];
if ( ! isalnum(ch) )
{
char chEsc = 0;
for ( UINT ie = 0 ; rgchEsc[ie][0] ; ie++ )
{
if ( rgchEsc[ie][0] == ch )
break;
}
if ( chEsc = rgchEsc[ie][1] )
{
if ( Suffix('\\') == NULL )
return false;
ch = chEsc;
}
}
if ( Suffix( ch ) == NULL )
return false ;
}
return true ;
}
// Change all alphabetic characters to the given case
void BNSTR :: UpCase ( bool bToUpper )
{
if ( bToUpper )
::strupr( _sz );
else
::strlwr( _sz );
}
//
// If the given expression string contains the symbolic name,
// reconstruct it with the replacement name.
bool BNSTR :: ReplaceSymName (
SZC szcSymName,
SZC szcSymNameNew,
bool bCaseInsensitive )
{
SZC szcFound ;
int cFound = 0 ;
UINT cchOffset = 0 ;
// Make a working copy of the sought symbolic name
BNSTR strSym( szcSymName );
if ( bCaseInsensitive )
strSym.UpCase();
do
{
BNSTR strTemp( Szc() );
if ( bCaseInsensitive )
strTemp.UpCase() ;
// Locate the symbolic name in the temporary copy.
szcFound = ::strstr( strTemp.Szc()+cchOffset, strSym ) ;
// If not found, we're done
if ( szcFound == NULL )
break ;
// Check to see if it's really a valid token; i.e., it's delimited.
if ( ( szcFound == strTemp.Szc()
|| ! iscsym(*(szcFound-1)) )
&& ( szcFound >= strTemp.Szc()+strTemp.Length()-strSym.Length()
|| ! iscsym(*(szcFound+strSym.Length())) )
)
{
// Build new string from preceding characters, the new sym name
// and trailing chars.
BNSTR strExprNew ;
UINT cchFound = szcFound - strTemp.Szc() ;
strExprNew.Assign( Szc(), cchFound );
strExprNew += szcSymNameNew ;
cchOffset = strExprNew.Length();
strExprNew += Szc() + cchFound + strSym.Length() ;
Assign( strExprNew );
cFound++ ;
}
else
{
// It was imbedded in another token. Skip over it.
cchOffset = szcFound - strTemp.Szc() + strSym.Length() ;
}
} while ( true );
return cFound > 0 ;
}
// Find the next occurrence of the given character in the string;
// Return -1 if not found.
INT BNSTR :: Index ( char chFind, UINT uiOffset ) const
{
if ( uiOffset >= _cchStr )
return -1 ;
SZC szcFound = ::strchr( _sz, chFind );
return szcFound
? szcFound - _sz
: -1 ;
}
// Convert the string to a floating-point number.
double BNSTR :: Atof ( UINT uiOffset ) const
{
return uiOffset < _cchStr
? ::atof( _sz + uiOffset )
: -1 ;
}
// End of BNSTR.CXX