DragonNest/GameCommon/DnUIString.cpp
2024-12-20 16:56:44 +08:00

471 lines
No EOL
30 KiB
C++
Raw Blame History

This file contains invisible Unicode characters

This file contains invisible Unicode characters that are indistinguishable to humans but may be processed differently by a computer. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

#include "StdAfx.h"
#include "EtUIXML.h"
#include "DnUIString.h"
#if defined(_CLIENT)
#include "DnInterfaceString.h"
#endif // #if defined(_CLIENT)
#ifdef _DEBUG
#define new new(_NORMAL_BLOCK,__FILE__,__LINE__)
#endif
#ifdef PRE_ADD_WORD_GENDER
void ChangeWordOrder( std::vector<std::string>& vecParam )
{
std::wstring strString;
std::string strSubStr;
int nAdjectiveIndex = -1;
int nNounIndex = -1;
for( int i=0; i<static_cast<int>( vecParam.size() ); i++ )
{
std::string strParm = vecParam[i];
if( strParm[0] == '{' && strParm[strParm.size()-1] == '}' )
strSubStr = strParm.substr( 1, strParm.size() - 2 );
#ifdef PRE_ADD_MULTILANGUAGE
strString = GetEtUIXML().GetUIString( CEtUIXML::idCategory1, atoi( strSubStr.c_str() ), MultiLanguage::eDefaultLanguage );
#else // PRE_ADD_MULTILANGUAGE
strString = GetEtUIXML().GetUIString( CEtUIXML::idCategory1, atoi( strSubStr.c_str() ) );
#endif // PRE_ADD_MULTILANGUAGE
std::wstring::size_type GenderTag = strString.find( ADJECTIVE_GENDER_TAGE );
if( GenderTag != std::wstring::npos )
{
nAdjectiveIndex = i;
for( int j=i+1; j<static_cast<int>( vecParam.size() ); j++ )
{
strParm = vecParam[j];
if( strParm[0] == '{' && strParm[strParm.size()-1] == '}' )
strSubStr = strParm.substr( 1, strParm.size() - 2 );
#ifdef PRE_ADD_MULTILANGUAGE
strString = GetEtUIXML().GetUIString( CEtUIXML::idCategory1, atoi( strSubStr.c_str() ), MultiLanguage::eDefaultLanguage );
#else // PRE_ADD_MULTILANGUAGE
strString = GetEtUIXML().GetUIString( CEtUIXML::idCategory1, atoi( strSubStr.c_str() ) );
#endif // PRE_ADD_MULTILANGUAGE
GenderTag = strString.find( NOUN_GENDER_TAGE );
if( GenderTag != std::wstring::npos )
{
nNounIndex = j;
break;
}
}
break;
}
}
if( nAdjectiveIndex >= 0 && nNounIndex > nAdjectiveIndex )
{
std::string strNounParam = vecParam[nNounIndex];
vecParam.erase( vecParam.begin() + nNounIndex );
vecParam.insert( vecParam.begin() + nAdjectiveIndex + 1, strNounParam );
}
}
std::wstring ApplyWordGenderString( bool bIsAdjective, std::wstring& wszParamString, std::vector<std::string>& vecParam )
{
std::wstring strApplyWordGenderString = wszParamString;
if( bIsAdjective )
{
std::wstring strNounGender;
std::string strSubStr;
for( int i=0; i<static_cast<int>( vecParam.size() ); i++ )
{
std::string strParm = vecParam[i];
if( strParm[0] == '{' && strParm[strParm.size()-1] == '}' )
strSubStr = strParm.substr( 1, strParm.size() - 2 );
#ifdef PRE_ADD_MULTILANGUAGE
strNounGender = GetEtUIXML().GetUIString( CEtUIXML::idCategory1, atoi( strSubStr.c_str() ), MultiLanguage::eDefaultLanguage );
#else // PRE_ADD_MULTILANGUAGE
strNounGender = GetEtUIXML().GetUIString( CEtUIXML::idCategory1, atoi( strSubStr.c_str() ) );
#endif // PRE_ADD_MULTILANGUAGE
std::wstring::size_type GenderTag = strNounGender.find( NOUN_GENDER_TAGE );
if( GenderTag != std::wstring::npos )
{
strNounGender.erase( 0, GenderTag + lstrlenW( NOUN_GENDER_TAGE ) );
std::transform( strNounGender.begin(), strNounGender.end(), strNounGender.begin(), towlower );
break;
}
}
if( strNounGender != L"" )
{
std::wstring strGenderTag;
std::wstring::size_type GenderTag = strApplyWordGenderString.find( ADJECTIVE_GENDER_TAGE );
if( GenderTag != std::wstring::npos )
{
strGenderTag = strApplyWordGenderString;
strGenderTag.erase( 0, GenderTag );
strApplyWordGenderString.erase( GenderTag, strApplyWordGenderString.length() - GenderTag );
if( wcscmp( strNounGender.c_str(), L"m" ) == 0 )
{
std::wstring::size_type SlashIndex = strGenderTag.find( L"/" );
if( SlashIndex != std::wstring::npos )
{
strGenderTag.erase( SlashIndex, strGenderTag.length() - SlashIndex );
strGenderTag.erase( 0, lstrlenW( NOUN_GENDER_TAGE ) );
strApplyWordGenderString += strGenderTag;
}
}
else if( wcscmp( strNounGender.c_str(), L"f" ) == 0 )
{
std::wstring::size_type SlashIndex = strGenderTag.rfind( L"/" );
if( SlashIndex != std::wstring::npos )
{
strGenderTag.erase( SlashIndex, strGenderTag.length() - SlashIndex );
SlashIndex = strGenderTag.find( L"/" );
if( SlashIndex != std::wstring::npos )
{
strGenderTag.erase( 0, SlashIndex + 1 );
strApplyWordGenderString += strGenderTag;
}
}
}
else if( wcscmp( strNounGender.c_str(), L"n" ) == 0 )
{
std::wstring::size_type SlashIndex = strGenderTag.rfind( L"/" );
if( SlashIndex != std::wstring::npos )
{
strGenderTag.erase( 0, SlashIndex + 1 );
strApplyWordGenderString += strGenderTag;
}
}
}
}
}
else
{
std::wstring::size_type GenderTag = strApplyWordGenderString.find( NOUN_GENDER_TAGE );
if( GenderTag != std::wstring::npos )
strApplyWordGenderString.erase( GenderTag, strApplyWordGenderString.length() - GenderTag );
}
return strApplyWordGenderString;
}
#endif // PRE_ADD_WORD_GENDER
#ifdef PRE_ADD_ITEMNAME_AUTOCOMPLETE
void MakeUIStringUseVariableParamByBaseMsg( std::wstring &wszStr, std::wstring& wszBaseMessage, char *szParam, std::vector <std::pair<std::wstring, int>> * pvList )
#else //#ifdef PRE_ADD_ITEMNAME_AUTOCOMPLETE
void MakeUIStringUseVariableParamByBaseMsg( std::wstring &wszStr, std::wstring& wszBaseMessage, char *szParam )
#endif //#ifdef PRE_ADD_ITEMNAME_AUTOCOMPLETE
{
// 여기에 영어, 한글, 소수가 들어갈 일은 절대 없으니, {int형숫자} 혹은 숫자만 생각하고 처리하면 된다.
// 파라미터를 스트링형태로 저장.
std::vector<std::string> vecParam;
if( szParam != NULL && (int)strlen(szParam) > 0 )
{
for( int i=0;; i++ )
{
const char *pStr = _GetSubStrByCount( i, szParam, ',' );
if( pStr == NULL || (int)strlen(pStr) == 0 ) break;
vecParam.push_back(pStr);
}
}
#ifdef PRE_ADD_WORD_GENDER
if( static_cast<int>( vecParam.size() ) > 2 )
ChangeWordOrder( vecParam );
#endif // PRE_ADD_WORD_GENDER
int nOffset = 0;
// BaseMessage에서 치환 스트링을 찾아 바꾼다.
while(1)
{
// nOffset은 치환 스트링이 대체되지 않았을 경우(파라미터가 없다거나 해서) 그 뒤부터 검색하기 위해 사용한다.
// 제대로 치환된다면 계속 맨 앞부터 찾는다.
std::wstring::size_type nPos = wszBaseMessage.find_first_of( L"{", nOffset );
std::wstring::size_type nPos2 = wszBaseMessage.find_first_of( L"}", nOffset );
if( (nPos != std::wstring::npos) && (nPos2 != std::wstring::npos) )
{
// 예외처리. }게 {보다 먼저 나올 경우, 그 지점부터 다시 찾는다.
if( nPos2 < nPos )
{
nOffset = (int)nPos2+1;
continue;
}
std::wstring wszParamIndex = wszBaseMessage.substr(nPos+1, nPos2-nPos-1);
#ifndef _FINAL_BUILD
// 예외처리. {}안에 아무것도 없을 경우,
if( wszParamIndex.size() == 0 )
_ASSERT(0&&"UIString조합 중 치환 스트링 중괄호 안에 숫자가 안적혀있습니다.");
// 예외처리. {}안에 숫자 말고 다른게 들어있을 경우,
for( int i = 0; i < (int)wszParamIndex.size(); ++i )
{
if( L'0' <= wszParamIndex[i] && wszParamIndex[i] <= L'9' )
{
}
else
{
_ASSERT(0&&"UIString조합 중 치환 스트링 중괄호 안에 숫자 말고 다른 스트링이 들어있습니다.");
}
}
#endif
int nIndex = _wtoi(wszParamIndex.c_str());
// 인덱스가 유효한지 보고, 유효하지 않다면 치환 내용을 그대로 둔다.
if( nIndex < 0 || nIndex+1 > (int)vecParam.size() )
{
nOffset = (int)nPos2+1;
continue;
}
// 파라미터 인덱스에 맞는 파라미터를 구해온다.
//std::string szParam = vecParam[nIndex]; // 지역 선언이 외부 범위에 있는 같은 이름의 선언을 숨깁니다. code analysis
std::string wszParam = vecParam[nIndex];
std::wstring wszParamString;
#ifdef PRE_ADD_WORD_GENDER
bool bIsAdjective = false;
#endif // PRE_ADD_WORD_GENDER
// 파라미터가 그냥 숫자인지, {숫자}인지 판단해 스트링으로 변환한다.(데이터에 공백 없을거라한다.)
if( wszParam[0] == '{' && wszParam[wszParam.size()-1] == '}' )
{
std::string szSubStr = wszParam.substr(1, wszParam.size()-2);
#ifndef _FINAL_BUILD
// 예외처리. {}안에 아무것도 없을 경우,
if( szSubStr.size() == 0 )
_ASSERT(0&&"UIString조합 중 파라미터 중괄호 안에 숫자가 안적혀있습니다.");
// 예외처리. {}안에 숫자 말고 다른게 들어있을 경우,
for( int i = 0; i < (int)szSubStr.size(); ++i )
{
if( L'0' <= szSubStr[i] && szSubStr[i] <= L'9' )
{
}
else
{
_ASSERT(0&&"UIString조합 중 파라미터 중괄호 안에 숫자 말고 다른 스트링이 들어있습니다.");
}
}
#endif
#if defined(PRE_ADD_MULTILANGUAGE)
//기본적인 유아이처리를 위한 문자열만드는 녀석이다 일단은 서버에서는 일단 아이템목록 생성시 사용 일단은 기본인자
wszParamString = GetEtUIXML().GetUIString( CEtUIXML::idCategory1, atoi(szSubStr.c_str()), MultiLanguage::eDefaultLanguage );
#else //#if defined(PRE_ADD_MULTILANGUAGE)
wszParamString = GetEtUIXML().GetUIString( CEtUIXML::idCategory1, atoi(szSubStr.c_str()) );
#endif //#if defined(PRE_ADD_MULTILANGUAGE)
#ifdef PRE_ADD_WORD_GENDER
std::wstring strParam;
strParam = wszParamString;
std::wstring::size_type GenderTag = strParam.find( ADJECTIVE_GENDER_TAGE );
if( GenderTag != std::wstring::npos )
{
bIsAdjective = true;
strParam.erase( GenderTag, strParam.length() - GenderTag );
}
GenderTag = strParam.find( NOUN_GENDER_TAGE );
if( GenderTag != std::wstring::npos )
strParam.erase( GenderTag, strParam.length() - GenderTag );
#endif // PRE_ADD_WORD_GENDER
#ifdef PRE_ADD_ITEMNAME_AUTOCOMPLETE
if (pvList)
{
#ifdef PRE_ADD_WORD_GENDER
pvList->push_back(std::make_pair(strParam, atoi(szSubStr.c_str())));
#else // PRE_ADD_WORD_GENDER
pvList->push_back(std::make_pair(wszParamString, atoi(szSubStr.c_str())));
#endif // PRE_ADD_WORD_GENDER
}
#endif //#ifdef PRE_ADD_ITEMNAME_AUTOCOMPLETE
#ifndef _FINAL_BUILD
// 예외처리. 파라미터에 있는 {}를 UIString으로 변환했는데, 그 안에 중괄호 {}가 있을 경우
for( int i = 0; i < (int)wszParamString.size(); ++i )
{
if( wszParamString[i] == L'{' || wszParamString[i] == L'}' )
{
_ASSERT(0&&"UIString조합 중 파라미터 중괄호의 값안에 중괄호{} 가 들어있습니다.");
}
}
#endif
}
else
{
ToWideString( wszParam, wszParamString );
}
#ifdef PRE_ADD_WORD_GENDER
wszParamString = ApplyWordGenderString( bIsAdjective, wszParamString, vecParam );
#endif // PRE_ADD_WORD_GENDER
// {숫자}위치에 파라미터 스트링값을 넣는다.
wszBaseMessage.replace( nPos, nPos2-nPos+1, wszParamString );
}
else
{
// 결과값 리턴 후 종료
wszStr = wszBaseMessage;
break;
}
}
}
#ifdef PRE_ADD_ITEMNAME_AUTOCOMPLETE
void MakeUIStringUseVariableParam( std::wstring &wszStr, int nMessageId, char *szParam, std::vector <std::pair<std::wstring, int>> * pvList )
#else //#ifdef PRE_ADD_ITEMNAME_AUTOCOMPLETE
void MakeUIStringUseVariableParam( std::wstring &wszStr, int nMessageId, char *szParam )
#endif //#ifdef PRE_ADD_ITEMNAME_AUTOCOMPLETE
{
#if defined(PRE_ADD_MULTILANGUAGE)
std::wstring wszBaseMessage = GetEtUIXML().GetUIString( CEtUIXML::idCategory1, nMessageId, MultiLanguage::eDefaultLanguage );
#else //#if defined(PRE_ADD_MULTILANGUAGE)
std::wstring wszBaseMessage = GetEtUIXML().GetUIString( CEtUIXML::idCategory1, nMessageId );
#endif //#if defined(PRE_ADD_MULTILANGUAGE)
#ifdef PRE_ADD_ITEMNAME_AUTOCOMPLETE
MakeUIStringUseVariableParamByBaseMsg(wszStr, wszBaseMessage, szParam, pvList);
#else //#ifdef PRE_ADD_ITEMNAME_AUTOCOMPLETE
MakeUIStringUseVariableParamByBaseMsg(wszStr, wszBaseMessage, szParam);
#endif //#ifdef PRE_ADD_ITEMNAME_AUTOCOMPLETE
#ifdef PRE_ADD_EXCEPTIONAL_MAKESTRING
if( szParam != NULL && static_cast<int>( strlen( szParam ) ) > 0 )
{
std::wstring strExceptional = GetEtExeptionalUIXML().GetReplacementUIString( szParam );
if( strExceptional.length() > 0 )
wszStr = strExceptional;
}
#endif // PRE_ADD_EXCEPTIONAL_MAKESTRING
}
#if defined(PRE_ADD_ITEM_GAINTABLE) && defined(_CLIENT)
void MakeUIStringItemGainDescription(std::wstring& result, const std::string& baseMessage)
{
if (baseMessage.empty())
return;
std::wstring convertBaseMessage;
DN_INTERFACE::UTIL::String2Wstring(convertBaseMessage, baseMessage);
int nOffset = 0;
while(1)
{
// nOffset은 치환 스트링이 대체되지 않았을 경우(파라미터가 없다거나 해서) 그 뒤부터 검색하기 위해 사용한다.
// 제대로 치환된다면 계속 맨 앞부터 찾는다.
std::wstring::size_type nPos = convertBaseMessage.find_first_of(L"{", nOffset);
std::wstring::size_type nPos2 = convertBaseMessage.find_first_of(L"}", nOffset);
if ((nPos != std::wstring::npos) && (nPos2 != std::wstring::npos))
{
// 예외처리. }게 {보다 먼저 나올 경우, 그 지점부터 다시 찾는다.
if (nPos2 < nPos)
{
nOffset = (int)nPos2+1;
continue;
}
std::wstring wszParamIndex = convertBaseMessage.substr(nPos+1, nPos2-nPos-1);
#ifndef _FINAL_BUILD
// 예외처리. {}안에 아무것도 없을 경우,
if( wszParamIndex.size() == 0 )
_ASSERT(0&&"UIString조합 중 치환 스트링 중괄호 안에 숫자가 안적혀있습니다.");
// 예외처리. {}안에 숫자 말고 다른게 들어있을 경우,
for( int i = 0; i < (int)wszParamIndex.size(); ++i )
{
if( L'0' <= wszParamIndex[i] && wszParamIndex[i] <= L'9' )
{
}
else
{
_ASSERT(0&&"UIString조합 중 치환 스트링 중괄호 안에 숫자 말고 다른 스트링이 들어있습니다.");
}
}
#endif
int nIndex = _wtoi(wszParamIndex.c_str());
// 인덱스가 유효한지 보고, 유효하지 않다면 치환 내용을 그대로 둔다.
if (nIndex <= 0) //rlkt fix
{
nOffset = (int)nPos2+1;
continue;
}
#ifdef PRE_ADD_MULTILANGUAGE
std::wstring wszParamString = GetEtUIXML().GetUIString(CEtUIXML::idCategory1, nIndex, MultiLanguage::eDefaultLanguage);
#else
std::wstring wszParamString = GetEtUIXML().GetUIString(CEtUIXML::idCategory1, nIndex);
#endif
if (wszParamString.empty() == false)
convertBaseMessage.replace(nPos, nPos2-nPos+1, wszParamString);
}
else
{
// 결과값 리턴 후 종료
result = convertBaseMessage;
break;
}
}
}
#endif
// 문자열파싱 - 앞의 문자열의 포맷에 맞게 뒤의 문자열을 가져와서 결합.
//
// : "{0} {1} 처치시달성" | "{가나다},{abc}" => 가나다 abc 처치시달성
//
std::wstring ParseUIParamString( int destID, char * pStrParam )
{
std::wstring strParam;
ToWideString( pStrParam, strParam );
std::vector< std::wstring > tokens;
TokenizeW( strParam, tokens, L"," );
std::wstring str( GetEtUIXML().GetUIString( CEtUIXML::idCategory1, destID ) );
std::wstring strTemp( str );
std::wstring::size_type preBegin = 0;
std::wstring::size_type begin = 0;
std::wstring::size_type end = 0;
while( 1 )
{
begin = strTemp.find_first_of( L"{" );
if( std::wstring::npos == begin )
break;
end = strTemp.find_first_of( L"}" );
if( std::wstring::npos == end )
break;
std::wstring strNum = strTemp.substr( begin + 1, (end-1) - begin );
int num = _wtoi( strNum.c_str() );
std::wstring::size_type addSize = 0;
const wchar_t * strL = NULL;
std::wstring & strToken = tokens[num];
if( std::wstring::npos == strToken.find( L"{" ) )
{
str.replace( preBegin+begin, end-begin+1, strToken );
addSize = wcslen( strToken.c_str() );
}
else if( strToken.size() > 2 )
{
std::wstring _str = strToken.substr( 1, strToken.size()-2 );
strL = GetEtUIXML().GetUIString( CEtUIXML::idCategory1, _wtoi(_str.c_str()) );
str.replace( preBegin+begin, end-begin+1, strL );
addSize = wcslen( strL );
}
strTemp = strTemp.substr( end+1, strTemp.size() );
/*if( strL )
preBegin += wcslen( strL );
else
preBegin += end;*/
preBegin += begin + addSize;
}
return str;
}