Files
UnrealEngineUWP/Engine/Source/Developer/Localization/Private/PortableObjectFormatDOM.cpp
Luke Thatcher 42f1c1f96b Copying //UE4/Dev-Console to //UE4/Dev-Main (Source: //UE4/Dev-Console @ 3483086)
#lockdown Nick.Penwarden
#rb none

==========================
MAJOR FEATURES + CHANGES
==========================

Change 3389969 on 2017/04/12 by Guillaume.Abadie

	Implements FDebug::DumpStackTraceToLog() debugging utility.

Change 3391579 on 2017/04/12 by Joe.Barnes

	Fix minor spacing issue.

Change 3402629 on 2017/04/20 by Ben.Marsh

	Build: Remove job to populate the DDC. Trying out a new system for this.

Change 3417501 on 2017/05/01 by Joe.Barnes

	IWYU - Missing #include files.

Change 3419927 on 2017/05/02 by Joe.Barnes

	- Support custom LOD for shadow map generation only (r.ForceLODShadow)
	- New #define to expose forceLOD and forceLODShadow also in Test/Ship builds (#define EXPOSE_FORCE_LOD 1 in RenderCore.cpp). Not exposed by default.

Change 3420964 on 2017/05/03 by Jonathan.Fitzpatrick

	Fixed null dereference of LineBatcher when using DrawDebugSphere and DrawDebugAltCone
	#jira UE-30213

Change 3423470 on 2017/05/04 by Luke.Thatcher

	[CONSOLE] [STREAMS] [^] Merging //UE4/Dev-Main (CL 3391974) to Dev-Console (//UE4/Dev-Console)
	 - Compile errors in Switch, to be fixed after check-in.

Change 3430410 on 2017/05/09 by Ben.Woodhouse

	Fix uninitialized local variable causing crashes in Test

	#jira UE-44832

Change 3430506 on 2017/05/09 by Josh.Adams

	- Fixed up the editor platforms' method of loading TargetSettings objects so that we don't need any manual parsing of .ini files to fill out the class defaults

Change 3434035 on 2017/05/10 by Ben.Woodhouse

	Integrate updated FortGPUTestbed from Fortnite/Main

Change 3437046 on 2017/05/12 by Joe.Barnes

	Fix for clang producing a warning when not all specializations of a templated function are marked FORCEINLINE.
	Also, switch a specialization of BlendTransform() from a function with a check to just a declaration so compiler will catch error instead of a runtime catch.

Change 3437259 on 2017/05/12 by Joe.Barnes

	Fix for clang producing a warning when not all specializations of a templated function are marked FORCEINLINE.
	Also, switch a specialization of BlendTransform() from a function with a check to just a declaration so compiler will catch error instead of a runtime catch.

Change 3440758 on 2017/05/16 by Ben.Woodhouse

	Simple threaded CSV Profiler
	To capture:
	- On the commandline, add -csvCaptureFrames=N to capture N frames from startup
	- On the console, use csvprofile start, csvprofile stop or csvprofile frames=N to capture a fixed number of frames
	- Instrument with CSV_SCOPED_STAT(statname), CSV_CUSTOM_STAT(statname,value).

	CSV capture is enabled in all builds except shipping
	- Please do not check in the instrumentation √ we don╞t want to pollute the engine with lots of additional instrumentation. We may add some minimal level of instrumentation at some point

Change 3440954 on 2017/05/16 by Josh.Adams

	- Cleaned up some DeviceProfiles in BaseDP.ini

Change 3443778 on 2017/05/17 by Ben.Woodhouse

	Aliasing for transient resources + new high level API

	Changelists integrated:
	3368830
	3368887
	3377762
	3377763
	3379513
	3381840
	3382046
	3382138
	3385390
	3385391
	3385531
	3396613
	3388752
	3396756
	3397007
	3397059
	3397780
	3397883
	3401716
	3415179

Change 3451460 on 2017/05/22 by Ben.Woodhouse

	Fix editor crash (NULL dereference of ScreenSpaceShadowTexture) when moving the camera around in tm-shadermodels, probably fallout from the VRAM aliasing merge. Not sure if this is the correct fix, but it prevents the crash for now

Change 3451601 on 2017/05/22 by Josh.Adams

	- Track idle time from MaxTickRate, so that stat unit is accurate on Game: thread

Change 3452025 on 2017/05/22 by Ben.Woodhouse

	Integrate (as edit) CL 3378734 (editor crash fix)
	Also add a check for null in LightFunctionRendering.cpp

Change 3452389 on 2017/05/22 by Josh.Adams

	- Replaced POCulturePluralForms with a static array, instead of TMapBuilder (was blowing stack or similar on Switch Debug). Code courtesy of Jamie Dale.

Change 3452758 on 2017/05/22 by Joe.Barnes

	Add FindFirstClearBit() and FindFirstSetBit() to TStaticBitArray.

Change 3455889 on 2017/05/23 by Ben.Woodhouse

	Integrate from //UE4/Main/...@3453436 to //UE4/Dev-Console/...

Change 3458654 on 2017/05/25 by Joe.Conley

	Attempting to fix Static Analysis warning.

Change 3462710 on 2017/05/26 by Ben.Woodhouse

	Integrate from //UE4/Main/...@3461688 to //UE4/Dev-Console/...

Change 3471711 on 2017/06/02 by Jonathan.Fitzpatrick

	Updating MallocProfiler to use the accessor for OnOutOfMemory delegate to conform with change made in CL 3415996.

Change 3473813 on 2017/06/05 by Ben.Woodhouse

	Fix streaming visibility logic bug reported on UDN

	#jira UE-43892

Change 3475298 on 2017/06/06 by Luke.Thatcher

	[CONSOLE] [!] Fix RHITransitionResources crash with more than 16 textures.
	 - Old command had a fixed sized array of 16 textures that would overflow.
	 - New command allocates an array of texture pointers inline in the command list, so any number is supported.

	#jira UE-45625

Change 3476776 on 2017/06/06 by Ben.Woodhouse

	Integrate from //UE4/Main/...@3475908

Change 3479083 on 2017/06/07 by Ben.Woodhouse

	Integrate as edit CL 3467315 from dev-animphys:
	From Alexis Matte.
	Ensure SectionMap is fixed up for old entries that are no longer valid.

	#JIRA UE-45438

	#jira UE-45735

Change 3480576 on 2017/06/08 by Ben.Woodhouse

	Integrate from //UE4/Main/...@3480024 to //UE4/Dev-Console/...

[CL 3483258 by Luke Thatcher in Main branch]
2017-06-09 17:44:13 -04:00

1002 lines
31 KiB
C++

// Copyright 1998-2017 Epic Games, Inc. All Rights Reserved.
#include "PortableObjectFormatDOM.h"
#include "Internationalization/Culture.h"
#include "Containers/MapBuilder.h"
static const TCHAR* NewLineDelimiter = TEXT("\n");
/**
* Default culture plural rules. Culture names are in the following format: Language_Country@Variant
* See references: http://www.unicode.org/cldr/charts/latest/supplemental/language_plural_rules.html
http://docs.translatehouse.org/projects/localization-guide/en/latest/l10n/pluralforms.html
*/
const TCHAR* GetPluralForm(const TCHAR* InCulture)
{
struct FPOCulturePluralForm
{
const TCHAR* CultureName;
const TCHAR* PluralForm;
};
static const FPOCulturePluralForm POCulturePluralForms[] = {
{ TEXT("ach"), TEXT("nplurals=2; plural=(n > 1);") },
{ TEXT("af"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("ak"), TEXT("nplurals=2; plural=(n > 1);") },
{ TEXT("aln"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("am"), TEXT("nplurals=2; plural=(n > 1);") },
{ TEXT("am_ET"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("an"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("ar"), TEXT("nplurals=6; plural=(n==0 ? 0 : n==1 ? 1 : n==2 ? 2 : n%100>=3 && n%100<=10 ? 3 : n%100>=11 && n%100<=99 ? 4 : 5);") },
{ TEXT("ar_SA"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("arn"), TEXT("nplurals=2; plural=(n > 1);") },
{ TEXT("as"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("ast"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("ay"), TEXT("nplurals=1; plural=0;") },
{ TEXT("az"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("bal"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("be"), TEXT("nplurals=3; plural=(n%10==1 && n%100!=11 ? 0 : n%10>=2 && n%10<=4 && (n%100<10 || n%100>=20) ? 1 : 2);") },
{ TEXT("bg"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("bn"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("bo"), TEXT("nplurals=1; plural=0;") },
{ TEXT("br"), TEXT("nplurals=2; plural=(n > 1);") },
{ TEXT("brx"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("bs"), TEXT("nplurals=3; plural=(n%10==1 && n%100!=11 ? 0 : n%10>=2 && n%10<=4 && (n%100<10 || n%100>=20) ? 1 : 2);") },
{ TEXT("ca"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("cgg"), TEXT("nplurals=1; plural=0;") },
{ TEXT("crh"), TEXT("nplurals=1; plural=0;") },
{ TEXT("cs"), TEXT("nplurals=3; plural=(n==1) ? 0 : (n>=2 && n<=4) ? 1 : 2;") },
{ TEXT("csb"), TEXT("nplurals=3; plural=(n==1 ? 0 : n%10>=2 && n%10<=4 && (n%100<10 || n%100>=20) ? 1 : 2;") },
{ TEXT("cy"), TEXT("nplurals=4; plural=(n==1) ? 0 : (n==2) ? 1 : (n != 8 && n != 11) ? 2 : 3;") },
{ TEXT("da"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("de"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("doi"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("dz"), TEXT("nplurals=1; plural=0;") },
{ TEXT("el"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("en"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("eo"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("es"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("es_ar"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("et"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("eu"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("fa"), TEXT("nplurals=1; plural=0;") },
{ TEXT("fi"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("fil"), TEXT("nplurals=2; plural=(n > 1);") },
{ TEXT("fo"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("fr"), TEXT("nplurals=2; plural=(n > 1);") },
{ TEXT("frp"), TEXT("nplurals=2; plural=(n > 1);") },
{ TEXT("fur"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("fy"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("ga"), TEXT("nplurals=5; plural=(n==1 ? 0 : n==2 ? 1 : n<7 ? 2 : n<11 ? 3 : 4);") },
{ TEXT("gd"), TEXT("nplurals=3; plural=(n < 2 ? 0 : n == 2 ? 1 : 2);") },
{ TEXT("gl"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("gu"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("gun"), TEXT("nplurals=2; plural=(n > 1);") },
{ TEXT("ha"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("he"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("hi"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("hne"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("hr"), TEXT("nplurals=3; plural=(n%10==1 && n%100!=11 ? 0 : n%10>=2 && n%10<=4 && (n%100<10 || n%100>=20) ? 1 : 2);") },
{ TEXT("hsb"), TEXT("nplurals=4; plural=(n%100==1 ? 0 : n%100==2 ? 1 : n%100==3 || n%100==4 ? 2 : 3);") },
{ TEXT("ht"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("hu"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("hy"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("ia"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("id"), TEXT("nplurals=1; plural=0;") },
{ TEXT("ig"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("ilo"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("is"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("it"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("ja"), TEXT("nplurals=1; plural=0;") },
{ TEXT("jv"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("ka"), TEXT("nplurals=1; plural=0;") },
{ TEXT("kk"), TEXT("nplurals=1; plural=0;") },
{ TEXT("km"), TEXT("nplurals=1; plural=0;") },
{ TEXT("kn"), TEXT("nplurals=1; plural=0;") },
{ TEXT("ko"), TEXT("nplurals=1; plural=0;") },
{ TEXT("ks"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("ku"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("kw"), TEXT("nplurals=4; plural=(n==1) ? 0 : (n==2) ? 1 : (n == 3) ? 2 : 3;") },
{ TEXT("ky"), TEXT("nplurals=1; plural=0;") },
{ TEXT("la"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("lb"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("li"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("ln"), TEXT("nplurals=2; plural=(n > 1);") },
{ TEXT("lo"), TEXT("nplurals=1; plural=0;") },
{ TEXT("lt"), TEXT("nplurals=3; plural=(n%10==1 && n%100!=11 ? 0 : n%10>=2 && (n%100<10 || n%100>=20) ? 1 : 2);") },
{ TEXT("lv"), TEXT("nplurals=3; plural=(n%10==1 && n%100!=11 ? 0 : n != 0 ? 1 : 2);") },
{ TEXT("mai"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("mg"), TEXT("nplurals=2; plural=(n > 1);") },
{ TEXT("mi"), TEXT("nplurals=2; plural=(n > 1);") },
{ TEXT("mk"), TEXT("nplurals=2; plural=(n % 10 == 1 && n % 100 != 11) ? 0 : 1;") },
{ TEXT("ml"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("mn"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("mr"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("ms"), TEXT("nplurals=1; plural=0;") },
{ TEXT("mt"), TEXT("nplurals=4; plural=(n==1 ? 0 : n==0 || ( n%100>1 && n%100<11) ? 1 : (n%100>10 && n%100<20 ) ? 2 : 3);") },
{ TEXT("my"), TEXT("nplurals=1; plural=0;") },
{ TEXT("nah"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("nap"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("nb"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("nds"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("ne"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("nl"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("nn"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("no"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("nr"), TEXT("nplurals=1; plural=0;") },
{ TEXT("nso"), TEXT("nplurals=2; plural=(n > 1);") },
{ TEXT("oc"), TEXT("nplurals=2; plural=(n > 1);") },
{ TEXT("or"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("pa"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("pap"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("pl"), TEXT("nplurals=3; plural=(n==1 ? 0 : n%10>=2 && n%10<=4 && (n%100<10 || n%100>=20) ? 1 : 2);") },
{ TEXT("pms"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("ps"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("pt"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("pt_BR"), TEXT("nplurals=2; plural=(n > 1);") },
{ TEXT("rm"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("ro"), TEXT("nplurals=3; plural=(n==1?0:(((n%100>19)||((n%100==0)&&(n!=0)))?2:1));") },
{ TEXT("ru"), TEXT("nplurals=3; plural=(n%10==1 && n%100!=11 ? 0 : n%10>=2 && n%10<=4 && (n%100<10 || n%100>=20) ? 1 : 2);") },
{ TEXT("rw"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("sc"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("sco"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("se"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("si"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("sk"), TEXT("nplurals=3; plural=(n==1) ? 0 : (n>=2 && n<=4) ? 1 : 2;") },
{ TEXT("sl"), TEXT("nplurals=4; plural=(n%100==1 ? 0 : n%100==2 ? 1 : n%100==3 || n%100==4 ? 2 : 3);") },
{ TEXT("sm"), TEXT("nplurals=1; plural=0;") },
{ TEXT("sn"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("so"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("son"), TEXT("nplurals=1; plural=0;") },
{ TEXT("sq"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("sr"), TEXT("nplurals=3; plural=(n%10==1 && n%100!=11 ? 0 : n%10>=2 && n%10<=4 && (n%100<10 || n%100>=20) ? 1 : 2);") },
{ TEXT("st"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("su"), TEXT("nplurals=1; plural=0;") },
{ TEXT("sv"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("sw"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("ta"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("te"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("tg"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("th"), TEXT("nplurals=1; plural=0;") },
{ TEXT("ti"), TEXT("nplurals=2; plural=(n > 1);") },
{ TEXT("tk"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("tl"), TEXT("nplurals=2; plural=(n > 1);") },
{ TEXT("tlh"), TEXT("nplurals=2; plural=(n > 1);") },
{ TEXT("to"), TEXT("nplurals=1; plural=0;") },
{ TEXT("tr"), TEXT("nplurals=1; plural=0;") },
{ TEXT("tt"), TEXT("nplurals=1; plural=0;") },
{ TEXT("udm"), TEXT("nplurals=1; plural=0;") },
{ TEXT("ug"), TEXT("nplurals=1; plural=0;") },
{ TEXT("uk"), TEXT("nplurals=3; plural=(n%10==1 && n%100!=11 ? 0 : n%10>=2 && n%10<=4 && (n%100<10 || n%100>=20) ? 1 : 2);") },
{ TEXT("ur"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("uz"), TEXT("nplurals=1; plural=0;") },
{ TEXT("ve"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("vi"), TEXT("nplurals=1; plural=0;") },
{ TEXT("vls"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("wa"), TEXT("nplurals=2; plural=(n > 1);") },
{ TEXT("wo"), TEXT("nplurals=1; plural=0;") },
{ TEXT("xh"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("yi"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("yo"), TEXT("nplurals=2; plural=(n != 1);") },
{ TEXT("zh"), TEXT("nplurals=1; plural=0;") },
{ TEXT("zu"), TEXT("nplurals=2; plural=(n != 1);") },
};
for (const FPOCulturePluralForm& POCulturePluralForm : POCulturePluralForms)
{
if (FCString::Stricmp(POCulturePluralForm.CultureName, InCulture) == 0)
{
return POCulturePluralForm.PluralForm;
}
}
return nullptr;
}
bool FindDelimitedString(const FString& InStr, const FString& LeftDelim, const FString& RightDelim, FString& Result)
{
int32 Start = InStr.Find(LeftDelim, ESearchCase::CaseSensitive);
int32 End = InStr.Find(RightDelim, ESearchCase::CaseSensitive, ESearchDir::FromEnd);
if (Start == INDEX_NONE || !(End > Start))
{
return false;
}
Start += LeftDelim.Len();
if (Start >= End)
{
Result = TEXT("");
}
else
{
Result = InStr.Mid(Start, End - Start);
}
return true;
}
FPortableObjectCulture::FPortableObjectCulture( const FString& LangCode, const FString& PluralForms )
: LanguageCode( LangCode )
, LanguagePluralForms( PluralForms )
, Culture( FInternationalization::Get().GetCulture( LangCode ) )
{
}
FPortableObjectCulture::FPortableObjectCulture( const FPortableObjectCulture& Other )
: LanguageCode( Other.LanguageCode )
, LanguagePluralForms( Other.LanguagePluralForms )
, Culture( FInternationalization::Get().GetCulture( Other.LanguageCode ) )
{
}
void FPortableObjectCulture::SetLanguageCode( const FString& LangCode )
{
LanguageCode = LangCode;
Culture = FInternationalization::Get().GetCulture( LangCode );
}
FString FPortableObjectCulture::Language() const
{
FString Result;
if( Culture.IsValid() )
{
// NOTE: The below function name may be a bit misleading because it will return three letter language codes if needed.
Result = Culture->GetTwoLetterISOLanguageName();
}
return Result;
}
FString FPortableObjectCulture::Country() const
{
FString Result;
if( Culture.IsValid() )
{
Result = Culture->GetRegion();
}
return Result;
}
FString FPortableObjectCulture::Variant() const
{
FString Result;
if( Culture.IsValid() )
{
Result = Culture->GetVariant();
}
return Result;
}
FString FPortableObjectCulture::DisplayName() const
{
FString Result;
if( Culture.IsValid() )
{
Result = Culture->GetDisplayName();
}
return Result;
}
FString FPortableObjectCulture::EnglishName() const
{
FString Result;
if( Culture.IsValid() )
{
Result = Culture->GetEnglishName();
}
return Result;
}
FString FPortableObjectCulture::GetPluralForms() const
{
if( !LanguagePluralForms.IsEmpty() )
{
return LanguagePluralForms;
}
return GetDefaultPluralForms();
}
FString FPortableObjectCulture::GetDefaultPluralForms() const
{
FString Result;
if( LanguageCode.IsEmpty() )
{
return Result;
}
if( const TCHAR* LanguageCodePair = GetPluralForm( *GetLanguageCode() ) )
{
Result = LanguageCodePair;
}
else if( const TCHAR* LangCountryVariantPair = GetPluralForm( *( Language() + TEXT("_") + Country() + TEXT("@") + Variant() ) ) )
{
Result = LangCountryVariantPair;
}
else if( const TCHAR* LangCountryPair = GetPluralForm( *( Language() + TEXT("_") + Country() ) ) )
{
Result = LangCountryPair;
}
else if( const TCHAR* LangPair = GetPluralForm( *Language() ) )
{
Result = LangPair;
}
else
{
const TCHAR* Fallback = GetPluralForm( TEXT("en") );
if( Fallback )
{
Result = Fallback;
}
else
{
Result = TEXT("nplurals=2; plural=(n != 1);");
}
}
return Result;
}
FString FPortableObjectHeader::ToString() const
{
FString Result;
for( const FString& Comment : Comments )
{
Result += FString::Printf(TEXT("# %s%s"), *Comment, NewLineDelimiter);
}
Result += FString::Printf(TEXT("msgid \"\"%s"), NewLineDelimiter);
Result += FString::Printf(TEXT("msgstr \"\"%s"), NewLineDelimiter);
for( auto Entry : HeaderEntries )
{
const FString& Key = Entry.Key;
const FString& Value = Entry.Value;
Result += FString::Printf( TEXT("\"%s: %s\\n\"%s"), *Key, *Value, NewLineDelimiter );
}
return Result;
}
bool FPortableObjectHeader::FromLocPOEntry( const TSharedRef<const FPortableObjectEntry> LocEntry )
{
if( !LocEntry->MsgId.IsEmpty() || LocEntry->MsgStr.Num() != 1 )
{
return false;
}
Clear();
Comments = LocEntry->TranslatorComments;
// The POEntry would store header info inside the MsgStr[0]
TArray<FString> HeaderLinesToProcess;
LocEntry->MsgStr[0].ReplaceEscapedCharWithChar().ParseIntoArray( HeaderLinesToProcess, NewLineDelimiter, true );
for( const FString& PotentialHeaderEntry : HeaderLinesToProcess )
{
int32 SplitIndex;
if( PotentialHeaderEntry.FindChar( TCHAR(':'), SplitIndex ) )
{
// Looks like a header entry so we add it
const FString& Key = PotentialHeaderEntry.LeftChop( PotentialHeaderEntry.Len() - SplitIndex ).Trim().TrimTrailing();
FString Value = PotentialHeaderEntry.RightChop( SplitIndex+1 ).Trim().TrimTrailing();
HeaderEntries.Emplace( Key, MoveTemp(Value) );
}
}
return true;
}
FPortableObjectHeader::FPOHeaderEntry* FPortableObjectHeader::FindEntry( const FString& EntryKey )
{
for( auto& Entry : HeaderEntries )
{
if( Entry.Key == EntryKey )
{
return &Entry;
}
}
return NULL;
}
const FPortableObjectHeader::FPOHeaderEntry* FPortableObjectHeader::FindEntry( const FString& EntryKey ) const
{
for( auto& Entry : HeaderEntries )
{
if( Entry.Key == EntryKey )
{
return &Entry;
}
}
return NULL;
}
FString FPortableObjectHeader::GetEntryValue( const FString& EntryKey ) const
{
FString Result;
const FPOHeaderEntry* Entry = FindEntry( EntryKey );
if( Entry )
{
Result = Entry->Value;
}
return Result;
}
bool FPortableObjectHeader::HasEntry( const FString& EntryKey ) const
{
return ( FindEntry( EntryKey ) != NULL);
}
void FPortableObjectHeader::SetEntryValue( const FString& EntryKey, const FString& EntryValue )
{
FPOHeaderEntry* Entry = FindEntry( EntryKey );
if( Entry )
{
Entry->Value = EntryValue;
}
else
{
HeaderEntries.Emplace( EntryKey, EntryValue );
}
}
void FPortableObjectHeader::UpdateTimeStamp()
{
// @TODO: Time format not exactly correct. We have something like this: 2014-02-07 20:06 This is what it should be: 2014-02-07 14:12-0600
FString Time = *FDateTime::UtcNow().ToString(TEXT("%Y-%m-%d %H:%M"));
SetEntryValue( TEXT("POT-Creation-Date"), Time );
SetEntryValue( TEXT("PO-Revision-Date"), Time );
}
FString FPortableObjectFormatDOM::ToString()
{
FString Result;
Header.UpdateTimeStamp();
Result += Header.ToString();
Result += NewLineDelimiter;
for( auto Entry : Entries )
{
Result += Entry->ToString();
Result += NewLineDelimiter;
}
return Result;
}
bool FPortableObjectFormatDOM::FromString( const FString& InStr )
{
if( InStr.IsEmpty() )
{
return false;
}
bool bSuccess = true;
FString ParseString = InStr.Replace(TEXT("\r\n"), NewLineDelimiter);
TArray<FString> LinesToProcess;
ParseString.ParseIntoArray( LinesToProcess, NewLineDelimiter, false );
TSharedRef<FPortableObjectEntry> ProcessedEntry = MakeShareable( new FPortableObjectEntry );
bool bHasMsgId = false;
uint32 NumFileLines = LinesToProcess.Num();
for( uint32 LineIdx = 0; LineIdx < NumFileLines; ++LineIdx )
{
const FString& Line = LinesToProcess[LineIdx];
if( Line.IsEmpty() )
{
// When we encounter a blank line, we will either ignore it, or consider it the boundary of
// a LocPOEntry or FPortableObjectHeader if we processed any valid data before encountering the blank.
// If this entry is valid we'll check it and process it further.
if( bHasMsgId && ProcessedEntry->MsgStr.Num() > 0 )
{
// Check if we are dealing with a header entry
if( ProcessedEntry->MsgId.IsEmpty() && ProcessedEntry->MsgStr.Num() == 1 )
{
// This is a header
bool bAddedHeader = Header.FromLocPOEntry( ProcessedEntry );
if( !bAddedHeader )
{
return false;
}
ProjectName = Header.GetEntryValue(TEXT("Project-Id-Version"));
}
else
{
bool bAddEntry = AddEntry( ProcessedEntry );
if( !bAddEntry )
{
return false;
}
}
}
// Now starting a new entry and reset any flags we have been tracking
ProcessedEntry = MakeShareable( new FPortableObjectEntry );
bHasMsgId = false;
}
else if( Line.StartsWith( TEXT("#,") ) )
{
// Flags
FString Flag;
if( FindDelimitedString(Line, TEXT("#,"), NewLineDelimiter, Flag ) )
{
if( !Flag.IsEmpty() )
{
ProcessedEntry->Flags.Add( Flag );
}
}
}
else if( Line.StartsWith( TEXT("#.") ) )
{
// Extracted comments
FString Comment;
ProcessedEntry->ExtractedComments.Add( Line.RightChop( FString(TEXT("#. ")).Len() ) );
}
else if( Line.StartsWith( TEXT("#:") ) )
{
// Reference
FString Reference;
ProcessedEntry->AddReference( Line.RightChop( FString(TEXT("#: ")).Len() ) );
}
else if( Line.StartsWith( TEXT(":|") ) )
{
// This represents previous messages. We just drop this in unknown since we don't handle it
ProcessedEntry->UnknownElements.Add( Line );
}
else if( Line.StartsWith( TEXT("# ") ) || Line.StartsWith( TEXT("#\t") ) )
{
FString Comment;
ProcessedEntry->TranslatorComments.Add( Line.RightChop( FString(TEXT("# ")).Len() ) );
}
else if( Line == TEXT("#") )
{
ProcessedEntry->TranslatorComments.Add( TEXT("") );
}
else if( Line.StartsWith( TEXT("msgctxt") ) )
{
FString RawMsgCtxt;
if( !FindDelimitedString(Line, TEXT("\""), TEXT("\""), RawMsgCtxt ) )
{
bSuccess = false;
break;
}
for( uint32 NextLineIdx = LineIdx + 1; NextLineIdx < NumFileLines && LinesToProcess[NextLineIdx].Trim().TrimTrailing().StartsWith(TEXT("\"")); ++NextLineIdx)
{
FString Tmp;
if (FindDelimitedString(Line, TEXT("\""), TEXT("\""), Tmp))
{
RawMsgCtxt += Tmp;
}
}
// If the following line contains more info for this element we continue to parse it out
uint32 NextLineIdx = LineIdx + 1;
while( NextLineIdx < NumFileLines )
{
const FString& NextLine = LinesToProcess[NextLineIdx].TrimTrailing().Trim();
if( NextLine.StartsWith("\"") && NextLine.EndsWith("\"") )
{
RawMsgCtxt += NextLine.Mid( 1, NextLine.Len()-2 );
}
else
{
break;
}
LineIdx = NextLineIdx;
NextLineIdx++;
}
ProcessedEntry->MsgCtxt = RawMsgCtxt;
}
else if( Line.StartsWith( TEXT("msgid") ) )
{
FString RawMsgId;
if( !FindDelimitedString(Line, TEXT("\""), TEXT("\""), RawMsgId ) )
{
bSuccess = false;
break;
}
// If the following line contains more info for this element we continue to parse it out
uint32 NextLineIdx = LineIdx + 1;
while( NextLineIdx < NumFileLines )
{
const FString& NextLine = LinesToProcess[NextLineIdx].TrimTrailing().Trim();
if( NextLine.StartsWith("\"") && NextLine.EndsWith("\"") )
{
RawMsgId += NextLine.Mid( 1, NextLine.Len()-2 );
}
else
{
break;
}
LineIdx = NextLineIdx;
NextLineIdx++;
}
ProcessedEntry->MsgId = RawMsgId;
bHasMsgId = true;
}
else if( Line.StartsWith( TEXT("msgid_plural") ) )
{
FString RawMsgIdPlural;
if( !FindDelimitedString(Line, TEXT("\""), TEXT("\""), RawMsgIdPlural ) )
{
bSuccess = false;
break;
}
// If the following line contains more info for this element we continue to parse it out
uint32 NextLineIdx = LineIdx + 1;
while( NextLineIdx < NumFileLines )
{
const FString& NextLine = LinesToProcess[NextLineIdx].TrimTrailing().Trim();
if( NextLine.StartsWith("\"") && NextLine.EndsWith("\"") )
{
RawMsgIdPlural += NextLine.Mid( 1, NextLine.Len()-2 );
}
else
{
break;
}
LineIdx = NextLineIdx;
NextLineIdx++;
}
ProcessedEntry->MsgIdPlural = RawMsgIdPlural;
}
else if( Line.StartsWith( TEXT("msgstr[") ) )
{
FString IndexStr;
if( !FindDelimitedString(Line, TEXT("["), TEXT("]"), IndexStr ) )
{
bSuccess = false;
break;
}
int32 Index = -1;
TTypeFromString<int32>::FromString(Index, *IndexStr );
check(Index >= 0);
FString RawMsgStr;
if( !FindDelimitedString(Line, TEXT("\""), TEXT("\""), RawMsgStr ) )
{
bSuccess = false;
break;
}
// If the following line contains more info for this element we continue to parse it out
uint32 NextLineIdx = LineIdx + 1;
while( NextLineIdx < NumFileLines )
{
const FString& NextLine = LinesToProcess[NextLineIdx].TrimTrailing().Trim();
if( NextLine.StartsWith("\"") && NextLine.EndsWith("\"") )
{
RawMsgStr += NextLine.Mid( 1, NextLine.Len()-2 );
}
else
{
break;
}
LineIdx = NextLineIdx;
NextLineIdx++;
}
if( ProcessedEntry->MsgStr.Num() > Index )
{
ProcessedEntry->MsgStr[Index] = RawMsgStr;
}
else
{
ProcessedEntry->MsgStr.Insert( RawMsgStr, Index );
}
}
else if( Line.StartsWith( TEXT("msgstr") ) )
{
FString RawMsgStr;
if( !FindDelimitedString(Line, TEXT("\""), TEXT("\""), RawMsgStr ) )
{
bSuccess = false;
break;
}
// If the following line contains more info for this element we continue to parse it out
uint32 NextLineIdx = LineIdx + 1;
while( NextLineIdx < NumFileLines )
{
const FString& NextLine = LinesToProcess[NextLineIdx].TrimTrailing().Trim();
if( NextLine.StartsWith("\"") && NextLine.EndsWith("\"") )
{
RawMsgStr += NextLine.Mid( 1, NextLine.Len()-2 );
}
else
{
break;
}
LineIdx = NextLineIdx;
NextLineIdx++;
}
FString MsgStr = RawMsgStr;
if( ProcessedEntry->MsgStr.Num() > 0 )
{
ProcessedEntry->MsgStr[0] = MsgStr;
}
else
{
ProcessedEntry->MsgStr.Add( MsgStr );
}
}
else
{
ProcessedEntry->UnknownElements.Add( Line );
}
}
return bSuccess;
}
void FPortableObjectFormatDOM::CreateNewHeader()
{
// Reference: http://www.gnu.org/software/gettext/manual/gettext.html#Header-Entry
// Reference: http://www.gnu.org/software/gettext/manual/html_node/Header-Entry.html
//Hard code some header entries for now in the following format
/*
# Engine English translation
# Copyright 1998-2017 Epic Games, Inc. All Rights Reserved.
#
msgid ""
msgstr ""
"Project-Id-Version: Engine\n"
"POT-Creation-Date: 2014-1-31 04:16+0000\n"
"PO-Revision-Date: 2014-1-31 04:16+0000\n"
"Language-Team: \n"
"Language: en-us\n"
"MIME-Version: 1.0\n"
"Content-Type: text/plain; charset=UTF-8\n"
"Content-Transfer-Encoding: 8bit\n"
"Plural-Forms: nplurals=2; plural=(n != 1);\n"
*/
Header.Clear();
Header.SetEntryValue( TEXT("Project-Id-Version"), *GetProjectName() );
// Setup header entries
Header.UpdateTimeStamp();
Header.SetEntryValue( TEXT("Language-Team"), TEXT("") );
Header.SetEntryValue( TEXT("Language"), Language.GetLanguageCode() );
Header.SetEntryValue( TEXT("MIME-Version"), TEXT("1.0") );
Header.SetEntryValue( TEXT("Content-Type"), TEXT("text/plain; charset=UTF-8") );
Header.SetEntryValue( TEXT("Content-Transfer-Encoding"), TEXT("8bit") );
Header.SetEntryValue( TEXT("Plural-Forms"), Language.GetPluralForms() );
Header.Comments.Add( FString::Printf(TEXT("%s %s translation."), *GetProjectName(), *Language.EnglishName() ) );
Header.Comments.Add( TEXT("Copyright 1998-2017 Epic Games, Inc. All Rights Reserved.") );
Header.Comments.Add( FString(TEXT("")) );
}
bool FPortableObjectFormatDOM::SetLanguage( const FString& LanguageCode, const FString& LangPluralForms )
{
FPortableObjectCulture NewLang( LanguageCode, LangPluralForms );
if( NewLang.IsValid() )
{
Language = NewLang;
return true;
}
return false;
}
bool FPortableObjectFormatDOM::AddEntry( const TSharedRef< FPortableObjectEntry> LocEntry )
{
TSharedPtr<FPortableObjectEntry> ExistingEntry = FindEntry( LocEntry );
if( ExistingEntry.IsValid() )
{
ExistingEntry->AddReferences( LocEntry->ReferenceComments );
ExistingEntry->AddExtractedComments( LocEntry->ExtractedComments );
// Checks for a situation that we don't know how to handle yet. ex. What happens when we match all the IDs for an entry but another param differs?
check( LocEntry->TranslatorComments == ExistingEntry->TranslatorComments );
}
else
{
Entries.Add( LocEntry );
}
return true;
}
TSharedPtr<FPortableObjectEntry> FPortableObjectFormatDOM::FindEntry( const TSharedRef<const FPortableObjectEntry> LocEntry ) const
{
for( auto Entry : Entries )
{
if( *Entry == *LocEntry )
{
return Entry;
}
}
return NULL;
}
TSharedPtr<FPortableObjectEntry> FPortableObjectFormatDOM::FindEntry( const FString& MsgId, const FString& MsgIdPlural, const FString& MsgCtxt ) const
{
TSharedRef<FPortableObjectEntry> TempEntry = MakeShareable( new FPortableObjectEntry );
TempEntry->MsgId = MsgId;
TempEntry->MsgIdPlural = MsgIdPlural;
TempEntry->MsgCtxt = MsgCtxt;
return FindEntry( TempEntry );
}
void FPortableObjectFormatDOM::SortEntries()
{
// Sort keys.
for (const TSharedPtr<FPortableObjectEntry>& Entry : Entries)
{
Entry->ReferenceComments.Sort();
}
// Sort by namespace, then keys, then source text.
const auto& SortingPredicate = [](const TSharedPtr<FPortableObjectEntry>& A, const TSharedPtr<FPortableObjectEntry>& B) -> bool
{
// Compare namespace
if (A->MsgCtxt < B->MsgCtxt)
{
return true;
}
else if (A->MsgCtxt > B->MsgCtxt)
{
return false;
}
// Compare keys
const int32 ExtractedCommentCount = FMath::Max(A->ExtractedComments.Num(), B->ExtractedComments.Num());
for (int32 i = 0; i < ExtractedCommentCount; ++i)
{
// If A has no more comments, it is before B.
if (!A->ExtractedComments.IsValidIndex(i) && B->ExtractedComments.IsValidIndex(i))
{
return true;
}
// If B has no more comments, it is before A.
if (A->ExtractedComments.IsValidIndex(i) && !B->ExtractedComments.IsValidIndex(i))
{
return false;
}
check(A->ExtractedComments.IsValidIndex(i) && B->ExtractedComments.IsValidIndex(i));
// If A's comment is lexicographically less, it is before B.
if (A->ExtractedComments[i] < B->ExtractedComments[i])
{
return true;
}
// If B's comment is lexicographically less, it is before A.
if (A->ExtractedComments[i] > B->ExtractedComments[i])
{
return false;
}
}
// Compare source string
if (A->MsgId < B->MsgId)
{
return true;
}
else if (A->MsgId > B->MsgId)
{
return false;
}
return A.Get() < B.Get();
};
Entries.Sort(SortingPredicate);
}
void FPortableObjectEntry::AddExtractedComment( const FString& InComment )
{
if(!InComment.IsEmpty())
{
ExtractedComments.AddUnique(InComment);
}
//// Extracted comments can contain multiple references in a single line so we parse those out.
//TArray<FString> CommentsToProcess;
//InComment.ParseIntoArray( CommentsToProcess, TEXT(" "), true );
//for( const FString& ExtractedComment : CommentsToProcess )
//{
// ExtractedComments.AddUnique( ExtractedComment );
//}
}
void FPortableObjectEntry::AddReference( const FString& InReference )
{
if(!InReference.IsEmpty())
{
ReferenceComments.AddUnique(InReference);
}
//// Reference comments can contain multiple references in a single line so we parse those out.
//TArray<FString> ReferencesToProcess;
//InReference.ParseIntoArray( ReferencesToProcess, TEXT(" "), true );
//for( const FString& Reference : ReferencesToProcess )
//{
// ReferenceComments.AddUnique( Reference );
//}
}
void FPortableObjectEntry::AddExtractedComments( const TArray<FString>& InComments )
{
for( const FString& ExtractedComment : InComments )
{
AddExtractedComment( ExtractedComment );
}
}
void FPortableObjectEntry::AddReferences( const TArray<FString>& InReferences )
{
for( const FString& Reference : InReferences )
{
AddReference( Reference );
}
}
FString FPortableObjectEntry::ToString() const
{
check( !MsgId.IsEmpty() );
FString Result;
for( const FString& TranslatorComment : TranslatorComments )
{
Result += FString::Printf( TEXT("# %s%s"), *TranslatorComment, NewLineDelimiter );
}
for( const FString& Comment : ExtractedComments )
{
if( !Comment.IsEmpty() )
{
Result += FString::Printf( TEXT("#. %s%s"), *Comment, NewLineDelimiter );
}
else
{
Result += FString::Printf( TEXT("#.%s"), NewLineDelimiter );
}
}
for( const FString& ReferenceComment : ReferenceComments )
{
Result += FString::Printf( TEXT("#: %s%s"), *ReferenceComment, NewLineDelimiter );
}
if( !MsgCtxt.IsEmpty() )
{
Result += FString::Printf(TEXT("msgctxt \"%s\"%s"), *MsgCtxt, NewLineDelimiter);
}
Result += FString::Printf(TEXT("msgid \"%s\"%s"), *MsgId, NewLineDelimiter);
if( MsgStr.Num() == 0)
{
Result += FString::Printf(TEXT("msgstr \"\"%s"), NewLineDelimiter);
}
else if( MsgStr.Num() == 1 )
{
Result += FString::Printf(TEXT("msgstr \"%s\"%s"), *MsgStr[0], NewLineDelimiter);
}
else
{
for( int32 Idx = 0; Idx < MsgStr.Num(); ++Idx )
{
Result += FString::Printf(TEXT("msgstr[%d] \"%s\"%s"), Idx, *MsgStr[Idx], NewLineDelimiter);
}
}
return Result;
}