123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154 |
- { Unicode tables unit.
- Copyright (c) 2013 by Inoussa OUEDRAOGO
- The source code is distributed under the Library GNU
- General Public License with the following modification:
- - object files and libraries linked into an application may be
- distributed without source code.
- If you didn't receive a copy of the file COPYING, contact:
- Free Software Foundation
- 675 Mass Ave
- Cambridge, MA 02139
- USA
- This program is distributed in the hope that it will be useful,
- but WITHOUT ANY WARRANTY; without even the implied warranty of
- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
- }
- unit unicodedata;
- {$mode delphi}
- {$H+}
- {$PACKENUM 1}
- {$SCOPEDENUMS ON}
- {$pointermath on}
- {$define USE_INLINE}
- {$warn 4056 off} //Conversion between ordinals and pointers is not portable
- { $define uni_debug}
- interface
- const
- MAX_WORD = High(Word);
- LOW_SURROGATE_BEGIN = Word($DC00);
- LOW_SURROGATE_END = Word($DFFF);
- HIGH_SURROGATE_BEGIN = Word($D800);
- HIGH_SURROGATE_END = Word($DBFF);
- HIGH_SURROGATE_COUNT = HIGH_SURROGATE_END - HIGH_SURROGATE_BEGIN + 1;
- UCS4_HALF_BASE = LongWord($10000);
- UCS4_HALF_MASK = Word($3FF);
- MAX_LEGAL_UTF32 = $10FFFF;
- const
- // Unicode General Category
- UGC_UppercaseLetter = 0;
- UGC_LowercaseLetter = 1;
- UGC_TitlecaseLetter = 2;
- UGC_ModifierLetter = 3;
- UGC_OtherLetter = 4;
- UGC_NonSpacingMark = 5;
- UGC_CombiningMark = 6;
- UGC_EnclosingMark = 7;
- UGC_DecimalNumber = 8;
- UGC_LetterNumber = 9;
- UGC_OtherNumber = 10;
- UGC_ConnectPunctuation = 11;
- UGC_DashPunctuation = 12;
- UGC_OpenPunctuation = 13;
- UGC_ClosePunctuation = 14;
- UGC_InitialPunctuation = 15;
- UGC_FinalPunctuation = 16;
- UGC_OtherPunctuation = 17;
- UGC_MathSymbol = 18;
- UGC_CurrencySymbol = 19;
- UGC_ModifierSymbol = 20;
- UGC_OtherSymbol = 21;
- UGC_SpaceSeparator = 22;
- UGC_LineSeparator = 23;
- UGC_ParagraphSeparator = 24;
- UGC_Control = 25;
- UGC_Format = 26;
- UGC_Surrogate = 27;
- UGC_PrivateUse = 28;
- UGC_Unassigned = 29;
- type
- TUInt24Rec = packed record
- public
- {$ifdef FPC_LITTLE_ENDIAN}
- byte0, byte1, byte2 : Byte;
- {$else FPC_LITTLE_ENDIAN}
- byte2, byte1, byte0 : Byte;
- {$endif FPC_LITTLE_ENDIAN}
- public
- class operator Implicit(a : TUInt24Rec) : Cardinal;{$ifdef USE_INLINE}inline;{$ENDIF}
- class operator Implicit(a : TUInt24Rec) : LongInt;{$ifdef USE_INLINE}inline;{$ENDIF}
- class operator Implicit(a : TUInt24Rec) : Word;{$ifdef USE_INLINE}inline;{$ENDIF}
- class operator Implicit(a : TUInt24Rec) : Byte;{$ifdef USE_INLINE}inline;{$ENDIF}
- class operator Implicit(a : Cardinal) : TUInt24Rec;{$ifdef USE_INLINE}inline;{$ENDIF}
- class operator Equal(a, b: TUInt24Rec): Boolean;{$ifdef USE_INLINE}inline;{$ENDIF}
- class operator Equal(a : TUInt24Rec; b : Cardinal): Boolean;{$ifdef USE_INLINE}inline;{$ENDIF}
- class operator Equal(a : Cardinal; b : TUInt24Rec): Boolean;{$ifdef USE_INLINE}inline;{$ENDIF}
- class operator Equal(a : TUInt24Rec; b : LongInt): Boolean;{$ifdef USE_INLINE}inline;{$ENDIF}
- class operator Equal(a : LongInt; b : TUInt24Rec): Boolean;{$ifdef USE_INLINE}inline;{$ENDIF}
- class operator Equal(a : TUInt24Rec; b : Word): Boolean;{$ifdef USE_INLINE}inline;{$ENDIF}
- class operator Equal(a : Word; b : TUInt24Rec): Boolean;{$ifdef USE_INLINE}inline;{$ENDIF}
- class operator Equal(a : TUInt24Rec; b : Byte): Boolean;{$ifdef USE_INLINE}inline;{$ENDIF}
- class operator Equal(a : Byte; b : TUInt24Rec): Boolean;{$ifdef USE_INLINE}inline;{$ENDIF}
- class operator NotEqual(a, b: TUInt24Rec): Boolean;{$ifdef USE_INLINE}inline;{$ENDIF}
- class operator NotEqual(a : TUInt24Rec; b : Cardinal): Boolean;{$ifdef USE_INLINE}inline;{$ENDIF}
- class operator NotEqual(a : Cardinal; b : TUInt24Rec): Boolean;{$ifdef USE_INLINE}inline;{$ENDIF}
- class operator GreaterThan(a, b: TUInt24Rec): Boolean;{$ifdef USE_INLINE}inline;{$ENDIF}
- class operator GreaterThan(a : TUInt24Rec; b : Cardinal): Boolean;{$ifdef USE_INLINE}inline;{$ENDIF}
- class operator GreaterThan(a : Cardinal; b : TUInt24Rec): Boolean;{$ifdef USE_INLINE}inline;{$ENDIF}
- class operator GreaterThanOrEqual(a, b: TUInt24Rec): Boolean;{$ifdef USE_INLINE}inline;{$ENDIF}
- class operator GreaterThanOrEqual(a : TUInt24Rec; b : Cardinal): Boolean;{$ifdef USE_INLINE}inline;{$ENDIF}
- class operator GreaterThanOrEqual(a : Cardinal; b : TUInt24Rec): Boolean;{$ifdef USE_INLINE}inline;{$ENDIF}
- class operator LessThan(a, b: TUInt24Rec): Boolean;{$ifdef USE_INLINE}inline;{$ENDIF}
- class operator LessThan(a : TUInt24Rec; b : Cardinal): Boolean;{$ifdef USE_INLINE}inline;{$ENDIF}
- class operator LessThan(a : Cardinal; b : TUInt24Rec): Boolean;{$ifdef USE_INLINE}inline;{$ENDIF}
- class operator LessThanOrEqual(a, b: TUInt24Rec): Boolean;{$ifdef USE_INLINE}inline;{$ENDIF}
- class operator LessThanOrEqual(a : TUInt24Rec; b : Cardinal): Boolean;{$ifdef USE_INLINE}inline;{$ENDIF}
- class operator LessThanOrEqual(a : Cardinal; b : TUInt24Rec): Boolean;{$ifdef USE_INLINE}inline;{$ENDIF}
- end;
- UInt24 = TUInt24Rec;
- PUInt24 = ^UInt24;
- const
- ZERO_UINT24 : UInt24 =
- {$ifdef FPC_LITTLE_ENDIAN}
- (byte0 : 0; byte1 : 0; byte2 : 0;);
- {$else FPC_LITTLE_ENDIAN}
- (byte2 : 0; byte1 : 0; byte0 : 0;);
- {$endif FPC_LITTLE_ENDIAN}
- type
- PUC_Prop = ^TUC_Prop;
- { TUC_Prop }
- TUC_Prop = packed record
- private
- function GetCategory : Byte;inline;
- procedure SetCategory(AValue : Byte);
- function GetWhiteSpace : Boolean;inline;
- procedure SetWhiteSpace(AValue : Boolean);
- function GetHangulSyllable : Boolean;inline;
- procedure SetHangulSyllable(AValue : Boolean);
- function GetNumericValue: Double;inline;
- public
- CategoryData : Byte;
- public
- CCC : Byte;
- NumericIndex : Byte;
- SimpleUpperCase : UInt24;
- SimpleLowerCase : UInt24;
- DecompositionID : SmallInt;
- public
- property Category : Byte read GetCategory write SetCategory;
- property WhiteSpace : Boolean read GetWhiteSpace write SetWhiteSpace;
- property HangulSyllable : Boolean read GetHangulSyllable write SetHangulSyllable;
- property NumericValue : Double read GetNumericValue;
- end;
- type
- TUCA_PropWeights = packed record
- Weights : array[0..2] of Word;
- end;
- PUCA_PropWeights = ^TUCA_PropWeights;
- TUCA_PropItemContextRec = packed record
- public
- CodePointCount : Byte;
- WeightCount : Byte;
- public
- function GetCodePoints() : PUInt24;inline;
- function GetWeights() : PUCA_PropWeights;inline;
- end;
- PUCA_PropItemContextRec = ^TUCA_PropItemContextRec;
- PUCA_PropItemContextTreeNodeRec = ^TUCA_PropItemContextTreeNodeRec;
- TUCA_PropItemContextTreeNodeRec = packed record
- public
- Left : Word;
- Right : Word;
- Data : TUCA_PropItemContextRec;
- public
- function GetLeftNode() : PUCA_PropItemContextTreeNodeRec;inline;
- function GetRightNode() : PUCA_PropItemContextTreeNodeRec;inline;
- end;
- { TUCA_PropItemContextTreeRec }
- TUCA_PropItemContextTreeRec = packed record
- public
- Size : UInt24;
- public
- function GetData:PUCA_PropItemContextTreeNodeRec;inline;
- property Data : PUCA_PropItemContextTreeNodeRec read GetData;
- function Find(
- const AChars : PUInt24;
- const ACharCount : Integer;
- out ANode : PUCA_PropItemContextTreeNodeRec
- ) : Boolean;
- end;
- PUCA_PropItemContextTreeRec = ^TUCA_PropItemContextTreeRec;
- { TUCA_PropItemRec }
- TUCA_PropItemRec = packed record
- private
- const FLAG_VALID = 0;
- const FLAG_CODEPOINT = 1;
- const FLAG_CONTEXTUAL = 2;
- const FLAG_DELETION = 3;
- const FLAG_COMPRESS_WEIGHT_1 = 6;
- const FLAG_COMPRESS_WEIGHT_2 = 7;
- private
- function GetCodePoint() : UInt24;inline;
- public
- WeightLength : Byte;
- ChildCount : Byte;
- Size : Word;
- Flags : Byte;
- public
- function HasCodePoint() : Boolean;inline;
- property CodePoint : UInt24 read GetCodePoint;
- //Weights : array[0..WeightLength] of TUCA_PropWeights;
- function IsValid() : Boolean;inline;
- //function GetWeightArray() : PUCA_PropWeights;inline;
- procedure GetWeightArray(ADest : PUCA_PropWeights);
- function GetSelfOnlySize() : Cardinal;inline;
- function GetContextual() : Boolean;inline;
- property Contextual : Boolean read GetContextual;
- function GetContext() : PUCA_PropItemContextTreeRec;
- function IsDeleted() : Boolean;inline;
- function IsWeightCompress_1() : Boolean;inline;
- function IsWeightCompress_2() : Boolean;inline;
- end;
- PUCA_PropItemRec = ^TUCA_PropItemRec;
- TUCA_VariableKind = (
- ucaShifted, ucaNonIgnorable, ucaBlanked, ucaShiftedTrimmed,
- ucaIgnoreSP // This one is not implemented !
- );
- TCollationName = string[128];
- PUCA_DataBook = ^TUCA_DataBook;
- TUCA_DataBook = packed record
- public
- Base : PUCA_DataBook;
- Version : TCollationName;
- CollationName : TCollationName;
- VariableWeight : TUCA_VariableKind;
- Backwards : array[0..3] of Boolean;
- BMP_Table1 : PByte;
- BMP_Table2 : PUInt24;
- OBMP_Table1 : PWord;
- OBMP_Table2 : PUInt24;
- PropCount : Integer;
- Props : PUCA_PropItemRec;
- VariableLowLimit : Word;
- VariableHighLimit : Word;
- public
- function IsVariable(const AWeight : PUCA_PropWeights) : Boolean; inline;
- end;
- TCollationField = (BackWard, VariableLowLimit, VariableHighLimit);
- TCollationFields = set of TCollationField;
- const
- ROOT_COLLATION_NAME = 'DUCET';
- procedure FromUCS4(const AValue : UCS4Char; var AHighS, ALowS : UnicodeChar);inline;
- function ToUCS4(const AHighS, ALowS : UnicodeChar) : UCS4Char;inline;
- function UnicodeIsSurrogatePair(
- const AHighSurrogate,
- ALowSurrogate : UnicodeChar
- ) : Boolean;inline;
- function UnicodeIsHighSurrogate(const AValue : UnicodeChar) : Boolean;inline;
- function UnicodeIsLowSurrogate(const AValue : UnicodeChar) : Boolean;inline;
- function GetProps(const ACodePoint : Word) : PUC_Prop;overload;inline;
- function GetProps(const AHighS, ALowS : UnicodeChar): PUC_Prop;overload;inline;
- function GetProps(const ACodePoint : Cardinal) : PUC_Prop;overload;inline;
- function GetPropUCA(const AHighS, ALowS : UnicodeChar; const ABook : PUCA_DataBook): PUCA_PropItemRec; inline; overload;
- function GetPropUCA(const AChar : UnicodeChar; const ABook : PUCA_DataBook) : PUCA_PropItemRec; inline; overload;
- function NormalizeNFD(const AString : UnicodeString) : UnicodeString;inline;overload;
- function NormalizeNFD(const AStr : PUnicodeChar; ALength : SizeInt) : UnicodeString;overload;
- procedure CanonicalOrder(var AString : UnicodeString);inline;overload;
- procedure CanonicalOrder(AStr : PUnicodeChar; const ALength : SizeInt);overload;
- type
- TUCASortKeyItem = Word;
- TUCASortKey = array of TUCASortKeyItem;
- function ComputeSortKey(
- const AString : UnicodeString;
- const ACollation : PUCA_DataBook
- ) : TUCASortKey;inline;overload;
- function ComputeSortKey(
- const AStr : PUnicodeChar;
- const ALength : SizeInt;
- const ACollation : PUCA_DataBook
- ) : TUCASortKey;overload;
- function CompareSortKey(const A, B : TUCASortKey) : Integer;overload;
- function CompareSortKey(const A : TUCASortKey; const B : array of Word) : Integer;overload;
- function RegisterCollation(const ACollation : PUCA_DataBook) : Boolean;
- function UnregisterCollation(const AName : ansistring): Boolean;
- function FindCollation(const AName : ansistring): PUCA_DataBook;overload;
- function FindCollation(const AIndex : Integer): PUCA_DataBook;overload;
- function GetCollationCount() : Integer;
- procedure PrepareCollation(
- ACollation : PUCA_DataBook;
- const ABaseName : ansistring;
- const AChangedFields : TCollationFields
- );
- resourcestring
- SCollationNotFound = 'Collation not found : "%s".';
- implementation
- uses
- unicodenumtable;
- type
- TCardinalRec = packed record
- {$ifdef FPC_LITTLE_ENDIAN}
- byte0, byte1, byte2, byte3 : Byte;
- {$else FPC_LITTLE_ENDIAN}
- byte3, byte2, byte1, byte0 : Byte;
- {$endif FPC_LITTLE_ENDIAN}
- end;
- TWordRec = packed record
- {$ifdef FPC_LITTLE_ENDIAN}
- byte0, byte1 : Byte;
- {$else FPC_LITTLE_ENDIAN}
- byte1, byte0 : Byte;
- {$endif FPC_LITTLE_ENDIAN}
- end;
- { TUInt24Rec }
- class operator TUInt24Rec.Implicit(a : TUInt24Rec) : Cardinal;
- begin
- TCardinalRec(Result).byte0 := a.byte0;
- TCardinalRec(Result).byte1 := a.byte1;
- TCardinalRec(Result).byte2 := a.byte2;
- TCardinalRec(Result).byte3 := 0;
- end;
- class operator TUInt24Rec.Implicit(a : TUInt24Rec) : LongInt;
- begin
- Result := Cardinal(a);
- end;
- class operator TUInt24Rec.Implicit(a : TUInt24Rec) : Word;
- begin
- {$IFOPT R+}
- if (a > $FFFF) then
- Error(reIntOverflow);
- {$ENDIF R+}
- TWordRec(Result).byte0 := a.byte0;
- TWordRec(Result).byte1 := a.byte1;
- end;
- class operator TUInt24Rec.Implicit(a : TUInt24Rec) : Byte;
- begin
- {$IFOPT R+}
- if (a > $FF) then
- Error(reIntOverflow);
- {$ENDIF R+}
- Result := a.byte0;
- end;
- class operator TUInt24Rec.Implicit(a : Cardinal) : TUInt24Rec;
- begin
- {$IFOPT R+}
- if (a > $FFFFFF) then
- Error(reIntOverflow);
- {$ENDIF R+}
- Result.byte0 := TCardinalRec(a).byte0;
- Result.byte1 := TCardinalRec(a).byte1;
- Result.byte2 := TCardinalRec(a).byte2;
- end;
- class operator TUInt24Rec.Equal(a, b : TUInt24Rec) : Boolean;
- begin
- Result := (a.byte0 = b.byte0) and (a.byte1 = b.byte1) and (a.byte2 = b.byte2);
- end;
- class operator TUInt24Rec.Equal(a : TUInt24Rec; b : Cardinal) : Boolean;
- begin
- Result := (TCardinalRec(b).byte3 = 0) and
- (a.byte0 = TCardinalRec(b).byte0) and
- (a.byte1 = TCardinalRec(b).byte1) and
- (a.byte2 = TCardinalRec(b).byte2);
- end;
- class operator TUInt24Rec.Equal(a : Cardinal; b : TUInt24Rec) : Boolean;
- begin
- Result := (b = a);
- end;
- class operator TUInt24Rec.Equal(a : TUInt24Rec; b : LongInt) : Boolean;
- begin
- Result := (LongInt(a) = b);
- end;
- class operator TUInt24Rec.Equal(a : LongInt; b : TUInt24Rec) : Boolean;
- begin
- Result := (b = a);
- end;
- class operator TUInt24Rec.Equal(a : TUInt24Rec; b : Word) : Boolean;
- begin
- Result := (a.byte2 = 0) and
- (a.byte0 = TWordRec(b).byte0) and
- (a.byte1 = TWordRec(b).byte1);
- end;
- class operator TUInt24Rec.Equal(a : Word; b : TUInt24Rec) : Boolean;
- begin
- Result := (b = a);
- end;
- class operator TUInt24Rec.Equal(a : TUInt24Rec; b : Byte) : Boolean;
- begin
- Result := (a.byte2 = 0) and
- (a.byte1 = 0) and
- (a.byte0 = b);
- end;
- class operator TUInt24Rec.Equal(a : Byte; b : TUInt24Rec) : Boolean;
- begin
- Result := (b = a);
- end;
- class operator TUInt24Rec.NotEqual(a, b : TUInt24Rec) : Boolean;
- begin
- Result := (a.byte0 <> b.byte0) or (a.byte1 <> b.byte1) or (a.byte2 <> b.byte2);
- end;
- class operator TUInt24Rec.NotEqual(a : TUInt24Rec; b : Cardinal) : Boolean;
- begin
- Result := (TCardinalRec(b).byte3 <> 0) or
- (a.byte0 <> TCardinalRec(b).byte0) or
- (a.byte1 <> TCardinalRec(b).byte1) or
- (a.byte2 <> TCardinalRec(b).byte2);
- end;
- class operator TUInt24Rec.NotEqual(a : Cardinal; b : TUInt24Rec) : Boolean;
- begin
- Result := (b <> a);
- end;
- class operator TUInt24Rec.GreaterThan(a, b: TUInt24Rec): Boolean;
- begin
- Result := (a.byte2 > b.byte2) or
- ((a.byte2 = b.byte2) and (a.byte1 > b.byte1)) or
- ((a.byte2 = b.byte2) and (a.byte1 = b.byte1) and (a.byte0 > b.byte0));
- end;
- class operator TUInt24Rec.GreaterThan(a: TUInt24Rec; b: Cardinal): Boolean;
- begin
- Result := Cardinal(a) > b;
- end;
- class operator TUInt24Rec.GreaterThan(a: Cardinal; b: TUInt24Rec): Boolean;
- begin
- Result := a > Cardinal(b);
- end;
- class operator TUInt24Rec.GreaterThanOrEqual(a, b: TUInt24Rec): Boolean;
- begin
- Result := (a.byte2 > b.byte2) or
- ((a.byte2 = b.byte2) and (a.byte1 > b.byte1)) or
- ((a.byte2 = b.byte2) and (a.byte1 = b.byte1) and (a.byte0 >= b.byte0));
- end;
- class operator TUInt24Rec.GreaterThanOrEqual(a: TUInt24Rec; b: Cardinal): Boolean;
- begin
- Result := Cardinal(a) >= b;
- end;
- class operator TUInt24Rec.GreaterThanOrEqual(a: Cardinal; b: TUInt24Rec): Boolean;
- begin
- Result := a >= Cardinal(b);
- end;
- class operator TUInt24Rec.LessThan(a, b: TUInt24Rec): Boolean;
- begin
- Result := (b > a);
- end;
- class operator TUInt24Rec.LessThan(a: TUInt24Rec; b: Cardinal): Boolean;
- begin
- Result := Cardinal(a) < b;
- end;
- class operator TUInt24Rec.LessThan(a: Cardinal; b: TUInt24Rec): Boolean;
- begin
- Result := a < Cardinal(b);
- end;
- class operator TUInt24Rec.LessThanOrEqual(a, b: TUInt24Rec): Boolean;
- begin
- Result := (b >= a);
- end;
- class operator TUInt24Rec.LessThanOrEqual(a: TUInt24Rec; b: Cardinal): Boolean;
- begin
- Result := Cardinal(a) <= b;
- end;
- class operator TUInt24Rec.LessThanOrEqual(a: Cardinal; b: TUInt24Rec): Boolean;
- begin
- Result := a <= Cardinal(b);
- end;
- var
- CollationTable : array of PUCA_DataBook;
- function IndexOfCollation(const AName : string) : Integer;
- var
- i, c : Integer;
- p : Pointer;
- begin
- c := Length(AName);
- p := @AName[1];
- for i := 0 to Length(CollationTable) - 1 do begin
- if (Length(CollationTable[i]^.CollationName) = c) and
- (CompareByte((CollationTable[i]^.CollationName[1]),p^,c)=0)
- then
- exit(i);
- end;
- Result := -1;
- end;
- function RegisterCollation(const ACollation : PUCA_DataBook) : Boolean;
- var
- i : Integer;
- begin
- Result := (IndexOfCollation(ACollation^.CollationName) = -1);
- if Result then begin
- i := Length(CollationTable);
- SetLength(CollationTable,(i+1));
- CollationTable[i] := ACollation;
- end;
- end;
- function UnregisterCollation(const AName : ansistring): Boolean;
- var
- i, c : Integer;
- begin
- i := IndexOfCollation(AName);
- Result := (i >= 0);
- if Result then begin
- c := Length(CollationTable);
- if (c = 1) then begin
- SetLength(CollationTable,0);
- end else begin
- CollationTable[i] := CollationTable[c-1];
- SetLength(CollationTable,(c-1));
- end;
- end;
- end;
- function FindCollation(const AName : ansistring): PUCA_DataBook;overload;
- var
- i : Integer;
- begin
- i := IndexOfCollation(AName);
- if (i = -1) then
- Result := nil
- else
- Result := CollationTable[i];
- end;
- function GetCollationCount() : Integer;
- begin
- Result := Length(CollationTable);
- end;
- function FindCollation(const AIndex : Integer): PUCA_DataBook;overload;
- begin
- if (AIndex < 0) or (AIndex >= Length(CollationTable)) then
- Result := nil
- else
- Result := CollationTable[AIndex];
- end;
- procedure PrepareCollation(
- ACollation : PUCA_DataBook;
- const ABaseName : ansistring;
- const AChangedFields : TCollationFields
- );
- var
- s : ansistring;
- p, base : PUCA_DataBook;
- begin
- if (ABaseName <> '') then
- s := ABaseName
- else
- s := ROOT_COLLATION_NAME;
- p := ACollation;
- base := FindCollation(s);
- if (base = nil) then
- Error(reCodesetConversion);
- p^.Base := base;
- if not(TCollationField.BackWard in AChangedFields) then
- p^.Backwards := base^.Backwards;
- if not(TCollationField.VariableLowLimit in AChangedFields) then
- p^.VariableLowLimit := base^.VariableLowLimit;
- if not(TCollationField.VariableHighLimit in AChangedFields) then
- p^.VariableLowLimit := base^.VariableHighLimit;
- end;
- {$INCLUDE unicodedata.inc}
- {$IFDEF ENDIAN_LITTLE}
- {$INCLUDE unicodedata_le.inc}
- {$ENDIF ENDIAN_LITTLE}
- {$IFDEF ENDIAN_BIG}
- {$INCLUDE unicodedata_be.inc}
- {$ENDIF ENDIAN_BIG}
- procedure FromUCS4(const AValue : UCS4Char; var AHighS, ALowS : UnicodeChar);inline;
- begin
- AHighS := UnicodeChar((AValue - $10000) shr 10 + $d800);
- ALowS := UnicodeChar((AValue - $10000) and $3ff + $dc00);
- end;
- function ToUCS4(const AHighS, ALowS : UnicodeChar) : UCS4Char;inline;
- begin
- Result := (UCS4Char(Word(AHighS)) - HIGH_SURROGATE_BEGIN) shl 10 +
- (UCS4Char(Word(ALowS)) - LOW_SURROGATE_BEGIN) + UCS4_HALF_BASE;
- end;
- function UnicodeIsSurrogatePair(
- const AHighSurrogate,
- ALowSurrogate : UnicodeChar
- ) : Boolean;
- begin
- Result :=
- ( (Word(AHighSurrogate) >= HIGH_SURROGATE_BEGIN) and
- (Word(AHighSurrogate) <= HIGH_SURROGATE_END)
- ) and
- ( (Word(ALowSurrogate) >= LOW_SURROGATE_BEGIN) and
- (Word(ALowSurrogate) <= LOW_SURROGATE_END)
- )
- end;
- function UnicodeIsHighSurrogate(const AValue : UnicodeChar) : Boolean;
- begin
- Result := (Word(AValue) >= HIGH_SURROGATE_BEGIN) and
- (Word(AValue) <= HIGH_SURROGATE_END);
- end;
- function UnicodeIsLowSurrogate(const AValue : UnicodeChar) : Boolean;
- begin
- Result := (Word(AValue) >= LOW_SURROGATE_BEGIN) and
- (Word(AValue) <= LOW_SURROGATE_END);
- end;
- function GetProps(const ACodePoint : Word) : PUC_Prop;overload;inline;
- begin
- Result:=
- @UC_PROP_ARRAY[
- UC_TABLE_3[
- UC_TABLE_2[UC_TABLE_1[hi(ACodePoint)]]
- [lo(ACodePoint) shr 4]
- ][lo(ACodePoint) and $F]
- ]; {
- @UC_PROP_ARRAY[
- UC_TABLE_2[
- (UC_TABLE_1[WordRec(ACodePoint).Hi] * 256) +
- WordRec(ACodePoint).Lo
- ]
- ];}
- end;
- function GetProps(const AHighS, ALowS : UnicodeChar): PUC_Prop;overload;inline;
- begin
- Result:=
- @UC_PROP_ARRAY[
- UCO_TABLE_3[
- UCO_TABLE_2[UCO_TABLE_1[Word(AHighS)-HIGH_SURROGATE_BEGIN]]
- [(Word(ALowS) - LOW_SURROGATE_BEGIN) div 32]
- ][(Word(ALowS) - LOW_SURROGATE_BEGIN) mod 32]
- ]; {
- Result:=
- @UC_PROP_ARRAY[
- UCO_TABLE_2[
- (UCO_TABLE_1[Word(AHighS)-HIGH_SURROGATE_BEGIN] * HIGH_SURROGATE_COUNT) +
- Word(ALowS) - LOW_SURROGATE_BEGIN
- ]
- ]; }
- end;
- function GetProps(const ACodePoint : Cardinal) : PUC_Prop;inline;
- var
- l, h : UnicodeChar;
- begin
- if (ACodePoint <= High(Word)) then
- exit(GetProps(Word(ACodePoint)));
- FromUCS4(ACodePoint,h,l);
- Result := GetProps(h,l);
- end;
- //----------------------------------------------------------------------
- function DecomposeHangul(const AChar : Cardinal; ABuffer : PCardinal) : Integer;
- const
- SBase = $AC00;
- LBase = $1100;
- VBase = $1161;
- TBase = $11A7;
- LCount = 19;
- VCount = 21;
- TCount = 28;
- NCount = VCount * TCount; // 588
- SCount = LCount * NCount; // 11172
- var
- SIndex, L, V, T : Integer;
- begin
- SIndex := AChar - SBase;
- if (SIndex < 0) or (SIndex >= SCount) then begin
- ABuffer^ := AChar;
- exit(1);
- end;
- L := LBase + SIndex div NCount;
- V := VBase + (SIndex mod NCount) div TCount;
- T := TBase + SIndex mod TCount;
- ABuffer[0] := L;
- ABuffer[1] := V;
- Result := 2;
- if (T <> TBase) then begin
- ABuffer[2] := T;
- Inc(Result);
- end;
- end;
- function Decompose(const ADecomposeIndex : Integer; ABuffer : PUnicodeChar) : Integer;
- var
- locStack : array[0..23] of Cardinal;
- locStackIdx : Integer;
- ResultBuffer : array[0..23] of Cardinal;
- ResultIdx : Integer;
- procedure AddCompositionToStack(const AIndex : Integer);
- var
- pdecIdx : ^TDecompositionIndexRec;
- k, kc : Integer;
- pu : ^UInt24;
- begin
- pdecIdx := @(UC_DEC_BOOK_DATA.Index[AIndex]);
- pu := @(UC_DEC_BOOK_DATA.CodePoints[pdecIdx^.StartPosition]);
- kc := pdecIdx^.Length;
- Inc(pu,kc);
- for k := 1 to kc do begin
- Dec(pu);
- locStack[locStackIdx + k] := pu^;
- end;
- locStackIdx := locStackIdx + kc;
- end;
- procedure AddResult(const AChar : Cardinal);inline;
- begin
- Inc(ResultIdx);
- ResultBuffer[ResultIdx] := AChar;
- end;
- function PopStack() : Cardinal;inline;
- begin
- Result := locStack[locStackIdx];
- Dec(locStackIdx);
- end;
- var
- cu : Cardinal;
- decIdx : SmallInt;
- locIsWord : Boolean;
- i : Integer;
- p : PUnicodeChar;
- begin
- ResultIdx := -1;
- locStackIdx := -1;
- AddCompositionToStack(ADecomposeIndex);
- while (locStackIdx >= 0) do begin
- cu := PopStack();
- locIsWord := (cu <= MAX_WORD);
- if locIsWord then
- decIdx := GetProps(Word(cu))^.DecompositionID
- else
- decIdx := GetProps(cu)^.DecompositionID;
- if (decIdx = -1) then
- AddResult(cu)
- else
- AddCompositionToStack(decIdx);
- end;
- p := ABuffer;
- Result := 0;
- for i := 0 to ResultIdx do begin
- cu := ResultBuffer[i];
- if (cu <= MAX_WORD) then begin
- p[0] := UnicodeChar(Word(cu));
- Inc(p);
- end else begin
- FromUCS4(cu,p[0],p[1]);
- Inc(p,2);
- Inc(Result);
- end;
- end;
- Result := Result + ResultIdx + 1;
- end;
- procedure CanonicalOrder(var AString : UnicodeString);
- begin
- CanonicalOrder(@AString[1],Length(AString));
- end;
- procedure CanonicalOrder(AStr : PUnicodeChar; const ALength : SizeInt);
- var
- i, c : SizeInt;
- p, q : PUnicodeChar;
- locIsSurrogateP, locIsSurrogateQ : Boolean;
- procedure Swap();
- var
- t, t1 : UnicodeChar;
- begin
- if not locIsSurrogateP then begin
- if not locIsSurrogateQ then begin
- t := p^;
- p^ := q^;
- q^ := t;
- exit;
- end;
- t := p^;
- p[0] := q[0];
- p[1] := q[1];
- q[1] := t;
- exit;
- end;
- if not locIsSurrogateQ then begin
- t := q[0];
- p[2] := p[1];
- p[1] := p[0];
- p[0] := t;
- exit;
- end;
- t := p[0];
- t1 := p[1];
- p[0] := q[0];
- p[1] := q[1];
- q[0] := t;
- q[1] := t1;
- end;
- var
- pu : PUC_Prop;
- cccp, cccq : Byte;
- begin
- c := ALength;
- if (c < 2) then
- exit;
- p := AStr;
- i := 1;
- while (i < c) do begin
- pu := GetProps(Word(p^));
- locIsSurrogateP := (pu^.Category = UGC_Surrogate);
- if locIsSurrogateP then begin
- if (i = (c - 1)) then
- Break;
- if not UnicodeIsSurrogatePair(p[0],p[1]) then begin
- Inc(p);
- Inc(i);
- Continue;
- end;
- pu := GetProps(p[0],p[1]);
- end;
- if (pu^.CCC > 0) then begin
- cccp := pu^.CCC;
- if locIsSurrogateP then
- q := p + 2
- else
- q := p + 1;
- pu := GetProps(Word(q^));
- locIsSurrogateQ := (pu^.Category = UGC_Surrogate);
- if locIsSurrogateQ then begin
- if (i = c) then
- Break;
- if not UnicodeIsSurrogatePair(q[0],q[1]) then begin
- Inc(p);
- Inc(i);
- Continue;
- end;
- pu := GetProps(q[0],q[1]);
- end;
- cccq := pu^.CCC;
- if (cccq > 0) and (cccp > cccq) then begin
- Swap();
- if (i > 1) then begin
- Dec(p);
- Dec(i);
- pu := GetProps(Word(p^));
- if (pu^.Category = UGC_Surrogate) then begin
- if (i > 1) then begin
- Dec(p);
- Dec(i);
- end;
- end;
- Continue;
- end;
- end;
- end;
- if locIsSurrogateP then begin
- Inc(p);
- Inc(i);
- end;
- Inc(p);
- Inc(i);
- end;
- end;
- //Canonical Decomposition
- function NormalizeNFD(const AString : UnicodeString) : UnicodeString;
- begin
- Result := NormalizeNFD(@AString[1],Length(AString));
- end;
- function NormalizeNFD(const AStr : PUnicodeChar; ALength : SizeInt) : UnicodeString;
- const MAX_EXPAND = 3;
- var
- i, c, kc, k : SizeInt;
- pp, pr : PUnicodeChar;
- pu : PUC_Prop;
- locIsSurrogate : Boolean;
- cpArray : array[0..7] of Cardinal;
- cp : Cardinal;
- begin
- c := ALength;
- SetLength(Result,(MAX_EXPAND*c));
- if (c > 0) then begin
- pp := AStr;
- pr := @Result[1];
- i := 1;
- while (i <= c) do begin
- pu := GetProps(Word(pp^));
- locIsSurrogate := (pu^.Category = UGC_Surrogate);
- if locIsSurrogate then begin
- if (i = c) then
- Break;
- if not UnicodeIsSurrogatePair(pp[0],pp[1]) then begin
- pr^ := pp^;
- Inc(pp);
- Inc(pr);
- Inc(i);
- Continue;
- end;
- pu := GetProps(pp[0],pp[1]);
- end;
- if pu^.HangulSyllable then begin
- if locIsSurrogate then begin
- cp := ToUCS4(pp[0],pp[1]);
- Inc(pp);
- Inc(i);
- end else begin
- cp := Word(pp^);
- end;
- kc := DecomposeHangul(cp,@cpArray[0]);
- for k := 0 to kc - 1 do begin
- if (cpArray[k] <= MAX_WORD) then begin
- pr^ := UnicodeChar(Word(cpArray[k]));
- pr := pr + 1;
- end else begin
- FromUCS4(cpArray[k],pr[0],pr[1]);
- pr := pr + 2;
- end;
- end;
- if (kc > 0) then
- Dec(pr);
- end else begin
- if (pu^.DecompositionID = -1) then begin
- pr^ := pp^;
- if locIsSurrogate then begin
- Inc(pp);
- Inc(pr);
- Inc(i);
- pr^ := pp^;
- end;
- end else begin
- k := Decompose(pu^.DecompositionID,pr);
- pr := pr + (k - 1);
- if locIsSurrogate then begin
- Inc(pp);
- Inc(i);
- end;
- end;
- end;
- Inc(pp);
- Inc(pr);
- Inc(i);
- end;
- Dec(pp);
- i := ((PtrUInt(pr) - PtrUInt(@Result[1])) div SizeOf(UnicodeChar));
- SetLength(Result,i);
- CanonicalOrder(@Result[1],Length(Result));
- end;
- end;
- type
- TBitOrder = 0..7;
- function IsBitON(const AData : Byte; const ABit : TBitOrder) : Boolean ;inline;
- begin
- Result := ( ( AData and ( 1 shl ABit ) ) <> 0 );
- end;
- procedure SetBit(var AData : Byte; const ABit : TBitOrder; const AValue : Boolean);inline;
- begin
- if AValue then
- AData := AData or (1 shl (ABit mod 8))
- else
- AData := AData and ( not ( 1 shl ( ABit mod 8 ) ) );
- end;
- { TUCA_PropItemContextTreeNodeRec }
- function TUCA_PropItemContextTreeNodeRec.GetLeftNode: PUCA_PropItemContextTreeNodeRec;
- begin
- if (Self.Left = 0) then
- Result := nil
- else
- Result := PUCA_PropItemContextTreeNodeRec(PtrUInt(@Self) + Self.Left);
- end;
- function TUCA_PropItemContextTreeNodeRec.GetRightNode: PUCA_PropItemContextTreeNodeRec;
- begin
- if (Self.Right = 0) then
- Result := nil
- else
- Result := PUCA_PropItemContextTreeNodeRec(PtrUInt(@Self) + Self.Right);
- end;
- { TUCA_PropItemContextRec }
- function TUCA_PropItemContextRec.GetCodePoints() : PUInt24;
- begin
- Result := PUInt24(
- PtrUInt(@Self) + SizeOf(Self.CodePointCount) +
- SizeOf(Self.WeightCount)
- );
- end;
- function TUCA_PropItemContextRec.GetWeights: PUCA_PropWeights;
- begin
- Result := PUCA_PropWeights(
- PtrUInt(@Self) +
- SizeOf(Self.CodePointCount) + SizeOf(Self.WeightCount) +
- (Self.CodePointCount*SizeOf(UInt24))
- );
- end;
- { TUCA_PropItemContextTreeRec }
- function TUCA_PropItemContextTreeRec.GetData: PUCA_PropItemContextTreeNodeRec;
- begin
- if (Size = 0) then
- Result := nil
- else
- Result := PUCA_PropItemContextTreeNodeRec(
- PtrUInt(
- PtrUInt(@Self) + SizeOf(UInt24){Size}
- )
- );
- end;
- function CompareCodePoints(
- A : PUInt24; LA : Integer;
- B : PUInt24; LB : Integer
- ) : Integer;
- var
- i, hb : Integer;
- begin
- if (A = B) then
- exit(0);
- Result := 1;
- hb := LB - 1;
- for i := 0 to LA - 1 do begin
- if (i > hb) then
- exit;
- if (A[i] < B[i]) then
- exit(-1);
- if (A[i] > B[i]) then
- exit(1);
- end;
- if (LA = LB) then
- exit(0);
- exit(-1);
- end;
- function TUCA_PropItemContextTreeRec.Find(
- const AChars : PUInt24;
- const ACharCount : Integer;
- out ANode : PUCA_PropItemContextTreeNodeRec
- ) : Boolean;
- var
- t : PUCA_PropItemContextTreeNodeRec;
- begin
- t := Data;
- while (t <> nil) do begin
- case CompareCodePoints(AChars,ACharCount,t^.Data.GetCodePoints(),t^.Data.CodePointCount) of
- 0 : Break;
- -1 : t := t^.GetLeftNode();
- else
- t := t^.GetRightNode();
- end;
- end;
- Result := (t <> nil);
- if Result then
- ANode := t;
- end;
- { TUC_Prop }
- function TUC_Prop.GetCategory: Byte;
- begin
- Result := Byte((CategoryData and Byte($F8)) shr 3);
- end;
- function TUC_Prop.GetNumericValue: Double;
- begin
- Result := UC_NUMERIC_ARRAY[NumericIndex];
- end;
- procedure TUC_Prop.SetCategory(AValue: Byte);
- begin
- CategoryData := Byte(CategoryData or Byte(AValue shl 3));
- end;
- function TUC_Prop.GetWhiteSpace: Boolean;
- begin
- Result := IsBitON(CategoryData,0);
- end;
- procedure TUC_Prop.SetWhiteSpace(AValue: Boolean);
- begin
- SetBit(CategoryData,0,AValue);
- end;
- function TUC_Prop.GetHangulSyllable: Boolean;
- begin
- Result := IsBitON(CategoryData,1);
- end;
- procedure TUC_Prop.SetHangulSyllable(AValue: Boolean);
- begin
- SetBit(CategoryData,1,AValue);
- end;
- { TUCA_DataBook }
- function TUCA_DataBook.IsVariable(const AWeight: PUCA_PropWeights): Boolean;
- begin
- Result := (AWeight^.Weights[0] >= Self.VariableLowLimit) and
- (AWeight^.Weights[0] <= Self.VariableHighLimit);
- end;
- { TUCA_PropItemRec }
- function TUCA_PropItemRec.IsWeightCompress_1 : Boolean;
- begin
- Result := IsBitON(Flags,FLAG_COMPRESS_WEIGHT_1);
- end;
- function TUCA_PropItemRec.IsWeightCompress_2 : Boolean;
- begin
- Result := IsBitON(Flags,FLAG_COMPRESS_WEIGHT_2);
- end;
- function TUCA_PropItemRec.GetCodePoint() : UInt24;
- begin
- if HasCodePoint() then begin
- if Contextual then
- Result := PUInt24(
- PtrUInt(@Self) + Self.GetSelfOnlySize()- SizeOf(UInt24) -
- Cardinal(GetContext()^.Size)
- )^
- else
- Result := PUInt24(PtrUInt(@Self) + Self.GetSelfOnlySize() - SizeOf(UInt24))^
- end else begin
- {$ifdef uni_debug}
- raise EUnicodeException.Create('TUCA_PropItemRec.GetCodePoint : "No code point available."');
- {$else uni_debug}
- Result := ZERO_UINT24;
- {$endif uni_debug}
- end
- end;
- function TUCA_PropItemRec.HasCodePoint() : Boolean;
- begin
- Result := IsBitON(Flags,FLAG_CODEPOINT);
- end;
- function TUCA_PropItemRec.IsValid() : Boolean;
- begin
- Result := IsBitON(Flags,FLAG_VALID);
- end;
- {function TUCA_PropItemRec.GetWeightArray: PUCA_PropWeights;
- begin
- Result := PUCA_PropWeights(PtrUInt(@Self) + SizeOf(TUCA_PropItemRec));
- end;}
- procedure TUCA_PropItemRec.GetWeightArray(ADest: PUCA_PropWeights);
- var
- c : Integer;
- p : PByte;
- pd : PUCA_PropWeights;
- begin
- c := WeightLength;
- p := PByte(PtrUInt(@Self) + SizeOf(TUCA_PropItemRec));
- pd := ADest;
- pd^.Weights[0] := PWord(p)^;
- p := p + 2;
- if not IsWeightCompress_1() then begin
- pd^.Weights[1] := PWord(p)^;
- p := p + 2;
- end else begin
- pd^.Weights[1] := p^;
- p := p + 1;
- end;
- if not IsWeightCompress_2() then begin
- pd^.Weights[2] := PWord(p)^;
- p := p + 2;
- end else begin
- pd^.Weights[2] := p^;
- p := p + 1;
- end;
- if (c > 1) then
- Move(p^, (pd+1)^, ((c-1)*SizeOf(TUCA_PropWeights)));
- end;
- function TUCA_PropItemRec.GetSelfOnlySize() : Cardinal;
- begin
- Result := SizeOf(TUCA_PropItemRec);
- if (WeightLength > 0) then begin
- Result := Result + (WeightLength * Sizeof(TUCA_PropWeights));
- if IsWeightCompress_1() then
- Result := Result - 1;
- if IsWeightCompress_2() then
- Result := Result - 1;
- end;
- if HasCodePoint() then
- Result := Result + SizeOf(UInt24);
- if Contextual then
- Result := Result + Cardinal(GetContext()^.Size);
- end;
- function TUCA_PropItemRec.GetContextual: Boolean;
- begin
- Result := IsBitON(Flags,FLAG_CONTEXTUAL);
- end;
- function TUCA_PropItemRec.GetContext: PUCA_PropItemContextTreeRec;
- var
- p : PtrUInt;
- begin
- if not Contextual then
- exit(nil);
- p := PtrUInt(@Self) + SizeOf(TUCA_PropItemRec);
- if IsBitON(Flags,FLAG_CODEPOINT) then
- p := p + SizeOf(UInt24);
- Result := PUCA_PropItemContextTreeRec(p);
- end;
- function TUCA_PropItemRec.IsDeleted() : Boolean;
- begin
- Result := IsBitON(Flags,FLAG_DELETION);
- end;
- function GetPropUCA(const AChar : UnicodeChar; const ABook : PUCA_DataBook) : PUCA_PropItemRec;
- var
- i : Cardinal;
- begin
- if (ABook^.BMP_Table2 = nil) then
- exit(nil);
- i := ABook^.BMP_Table2[
- (ABook^.BMP_Table1[Hi(Word(AChar))] * 256) +
- Lo(Word(AChar))
- ];
- if (i > 0) then
- Result:= PUCA_PropItemRec(PtrUInt(ABook^.Props) + i - 1)
- else
- Result := nil;
- end;
- function GetPropUCA(const AHighS, ALowS : UnicodeChar; const ABook : PUCA_DataBook): PUCA_PropItemRec;
- var
- i : Cardinal;
- begin
- if (ABook^.OBMP_Table2 = nil) then
- exit(nil);
- i := ABook^.OBMP_Table2[
- (ABook^.OBMP_Table1[Word(AHighS)-HIGH_SURROGATE_BEGIN] * HIGH_SURROGATE_COUNT) +
- Word(ALowS) - LOW_SURROGATE_BEGIN
- ];
- if (i > 0) then
- Result:= PUCA_PropItemRec(PtrUInt(ABook^.Props) + i - 1)
- else
- Result := nil;
- end;
- {$include weight_derivation.inc}
- function CompareSortKey(const A : TUCASortKey; const B : array of Word) : Integer;
- var
- bb : TUCASortKey;
- begin
- SetLength(bb,Length(B));
- if (Length(bb) > 0) then
- Move(B[0],bb[0],(Length(bb)*SizeOf(B[0])));
- Result := CompareSortKey(A,bb);
- end;
- function CompareSortKey(const A, B : TUCASortKey) : Integer;
- var
- i, hb : Integer;
- begin
- if (Pointer(A) = Pointer(B)) then
- exit(0);
- Result := 1;
- hb := Length(B) - 1;
- for i := 0 to Length(A) - 1 do begin
- if (i > hb) then
- exit;
- if (A[i] < B[i]) then
- exit(-1);
- if (A[i] > B[i]) then
- exit(1);
- end;
- if (Length(A) = Length(B)) then
- exit(0);
- exit(-1);
- end;
- type
- TUCA_PropWeightsArray = array of TUCA_PropWeights;
- function FormKeyBlanked(const ACEList : TUCA_PropWeightsArray; const ACollation : PUCA_DataBook) : TUCASortKey;
- var
- r : TUCASortKey;
- i, c, k, ral, levelCount : Integer;
- pce : PUCA_PropWeights;
- begin
- c := Length(ACEList);
- if (c = 0) then
- exit(nil);
- levelCount := Length(ACEList[0].Weights);
- SetLength(r,(levelCount*c + levelCount));
- ral := 0;
- for i := 0 to levelCount - 1 do begin
- if not ACollation^.Backwards[i] then begin
- pce := @ACEList[0];
- for k := 0 to c - 1 do begin
- if not(ACollation^.IsVariable(pce)) and (pce^.Weights[i] <> 0) then begin
- r[ral] := pce^.Weights[i];
- ral := ral + 1;
- end;
- pce := pce + 1;
- end;
- end else begin
- pce := @ACEList[c-1];
- for k := 0 to c - 1 do begin
- if not(ACollation^.IsVariable(pce)) and (pce^.Weights[i] <> 0) then begin
- r[ral] := pce^.Weights[i];
- ral := ral + 1;
- end;
- pce := pce - 1;
- end;
- end;
- r[ral] := 0;
- ral := ral + 1;
- end;
- ral := ral - 1;
- SetLength(r,ral);
- Result := r;
- end;
- function FormKeyNonIgnorable(const ACEList : TUCA_PropWeightsArray; const ACollation : PUCA_DataBook) : TUCASortKey;
- var
- r : TUCASortKey;
- i, c, k, ral, levelCount : Integer;
- pce : PUCA_PropWeights;
- begin
- c := Length(ACEList);
- if (c = 0) then
- exit(nil);
- levelCount := Length(ACEList[0].Weights);
- SetLength(r,(levelCount*c + levelCount));
- ral := 0;
- for i := 0 to levelCount - 1 do begin
- if not ACollation^.Backwards[i] then begin
- pce := @ACEList[0];
- for k := 0 to c - 1 do begin
- if (pce^.Weights[i] <> 0) then begin
- r[ral] := pce^.Weights[i];
- ral := ral + 1;
- end;
- pce := pce + 1;
- end;
- end else begin
- pce := @ACEList[c-1];
- for k := 0 to c - 1 do begin
- if (pce^.Weights[i] <> 0) then begin
- r[ral] := pce^.Weights[i];
- ral := ral + 1;
- end;
- pce := pce - 1;
- end;
- end;
- r[ral] := 0;
- ral := ral + 1;
- end;
- ral := ral - 1;
- SetLength(r,ral);
- Result := r;
- end;
- function FormKeyShifted(const ACEList : TUCA_PropWeightsArray; const ACollation : PUCA_DataBook) : TUCASortKey;
- var
- r : TUCASortKey;
- i, c, k, ral, levelCount : Integer;
- pce : PUCA_PropWeights;
- variableState : Boolean;
- begin
- c := Length(ACEList);
- if (c = 0) then
- exit(nil);
- levelCount := Length(ACEList[0].Weights);
- SetLength(r,(levelCount*c + levelCount));
- ral := 0;
- for i := 0 to levelCount - 1 do begin
- if not ACollation^.Backwards[i] then begin
- variableState := False;
- pce := @ACEList[0];
- for k := 0 to c - 1 do begin
- if not ACollation^.IsVariable(pce) then begin
- if (pce^.Weights[0] <> 0) then
- variableState := False;
- if (pce^.Weights[i] <> 0) and not(variableState) then begin
- r[ral] := pce^.Weights[i];
- ral := ral + 1;
- end;
- end else begin
- variableState := True;
- end;
- pce := pce + 1;
- end;
- end else begin
- pce := @ACEList[c-1];
- for k := 0 to c - 1 do begin
- if not ACollation^.IsVariable(pce) then begin
- if (pce^.Weights[0] <> 0) then
- variableState := False;
- if (pce^.Weights[i] <> 0) and not(variableState) then begin
- r[ral] := pce^.Weights[i];
- ral := ral + 1;
- end;
- end else begin
- variableState := True;
- end;
- pce := pce - 1;
- end;
- end;
- r[ral] := 0;
- ral := ral + 1;
- end;
- ral := ral - 1;
- SetLength(r,ral);
- Result := r;
- end;
- function FormKeyShiftedTrimmed(
- const ACEList : TUCA_PropWeightsArray;
- const ACollation : PUCA_DataBook
- ) : TUCASortKey;
- var
- i : Integer;
- p : ^TUCASortKeyItem;
- begin
- Result := FormKeyShifted(ACEList,ACollation);
- i := Length(Result) - 1;
- if (i >= 0) then begin
- p := @Result[i];
- while (i >= 0) do begin
- if (p^ <> $FFFF) then
- Break;
- Dec(i);
- Dec(p);
- end;
- if ((i+1) < Length(Result)) then
- SetLength(Result,(i+1));
- end;
- end;
- function FindChild(
- const ACodePoint : Cardinal;
- const AParent : PUCA_PropItemRec
- ) : PUCA_PropItemRec;inline;
- var
- k : Integer;
- begin
- Result := PUCA_PropItemRec(PtrUInt(AParent) + AParent^.GetSelfOnlySize());
- for k := 0 to AParent^.ChildCount - 1 do begin
- if (ACodePoint = Result^.CodePoint) then
- exit;
- Result := PUCA_PropItemRec(PtrUInt(Result) + Result^.Size);
- end;
- Result := nil;
- end;
- function ComputeSortKey(
- const AString : UnicodeString;
- const ACollation : PUCA_DataBook
- ) : TUCASortKey;
- begin
- Result := ComputeSortKey(@AString[1],Length(AString),ACollation);
- end;
- function ComputeRawSortKey(
- const AStr : PUnicodeChar;
- const ALength : SizeInt;
- const ACollation : PUCA_DataBook
- ) : TUCA_PropWeightsArray;
- var
- r : TUCA_PropWeightsArray;
- ral {used length of "r"}: Integer;
- rl {capacity of "r"} : Integer;
- procedure GrowKey(const AMinGrow : Integer = 0);inline;
- begin
- if (rl < AMinGrow) then
- rl := rl + AMinGrow
- else
- rl := 2 * rl;
- SetLength(r,rl);
- end;
- var
- i : Integer;
- s : UnicodeString;
- ps : PUnicodeChar;
- cp : Cardinal;
- cl : PUCA_DataBook;
- pp : PUCA_PropItemRec;
- ppLevel : Byte;
- removedCharIndex : array of DWord;
- removedCharIndexLength : DWord;
- locHistory : array[0..24] of record
- i : Integer;
- cl : PUCA_DataBook;
- pp : PUCA_PropItemRec;
- ppLevel : Byte;
- cp : Cardinal;
- removedCharIndexLength : DWord;
- end;
- locHistoryTop : Integer;
- suppressState : record
- cl : PUCA_DataBook;
- CharCount : Integer;
- end;
- LastKeyOwner : record
- Length : Integer;
- Chars : array[0..24] of UInt24;
- end;
- procedure SaveKeyOwner();
- var
- k : Integer;
- kppLevel : Byte;
- begin
- k := 0;
- kppLevel := High(Byte);
- while (k <= locHistoryTop) do begin
- if (kppLevel <> locHistory[k].ppLevel) then begin
- LastKeyOwner.Chars[k] := locHistory[k].cp;
- kppLevel := locHistory[k].ppLevel;
- end;
- k := k + 1;
- end;
- if (k = 0) or (kppLevel <> ppLevel) then begin
- LastKeyOwner.Chars[k] := cp;
- k := k + 1;
- end;
- LastKeyOwner.Length := k;
- end;
- procedure AddWeights(AItem : PUCA_PropItemRec);inline;
- begin
- SaveKeyOwner();
- if ((ral + AItem^.WeightLength) > rl) then
- GrowKey(AItem^.WeightLength);
- AItem^.GetWeightArray(@r[ral]);
- ral := ral + AItem^.WeightLength;
- end;
- procedure AddContextWeights(AItem : PUCA_PropItemContextRec);inline;
- begin
- if ((ral + AItem^.WeightCount) > rl) then
- GrowKey(AItem^.WeightCount);
- Move(AItem^.GetWeights()^,r[ral],(AItem^.WeightCount*SizeOf(r[0])));
- ral := ral + AItem^.WeightCount;
- end;
- procedure AddComputedWeights(ACodePoint : Cardinal);inline;
- begin
- SaveKeyOwner();
- if ((ral + 2) > rl) then
- GrowKey();
- DeriveWeight(ACodePoint,@r[ral]);
- ral := ral + 2;
- end;
- procedure RecordDeletion();inline;
- begin
- if pp^.IsValid() and pp^.IsDeleted() (*pp^.GetWeightLength() = 0*) then begin
- if (suppressState.cl = nil) or
- (suppressState.CharCount > ppLevel)
- then begin
- suppressState.cl := cl;
- suppressState.CharCount := ppLevel;
- end;
- end;
- end;
- procedure RecordStep();inline;
- begin
- Inc(locHistoryTop);
- locHistory[locHistoryTop].i := i;
- locHistory[locHistoryTop].cl := cl;
- locHistory[locHistoryTop].pp := pp;
- locHistory[locHistoryTop].ppLevel := ppLevel;
- locHistory[locHistoryTop].cp := cp;
- locHistory[locHistoryTop].removedCharIndexLength := removedCharIndexLength;
- RecordDeletion();
- end;
- procedure ClearHistory();inline;
- begin
- locHistoryTop := -1;
- end;
- function HasHistory() : Boolean;inline;
- begin
- Result := (locHistoryTop >= 0);
- end;
- function GetHistoryLength() : Integer;inline;
- begin
- Result := (locHistoryTop + 1);
- end;
- procedure GoBack();inline;
- begin
- Assert(locHistoryTop >= 0);
- i := locHistory[locHistoryTop].i;
- cp := locHistory[locHistoryTop].cp;
- cl := locHistory[locHistoryTop].cl;
- pp := locHistory[locHistoryTop].pp;
- ppLevel := locHistory[locHistoryTop].ppLevel;
- removedCharIndexLength := locHistory[locHistoryTop].removedCharIndexLength;
- ps := @s[i];
- Dec(locHistoryTop);
- end;
- var
- c : Integer;
- lastUnblockedNonstarterCCC : Byte;
- function IsUnblockedNonstarter(const AStartFrom : Integer) : Boolean;
- var
- k : DWord;
- pk : PUnicodeChar;
- puk : PUC_Prop;
- begin
- k := AStartFrom;
- if (k > c) then
- exit(False);
- if (removedCharIndexLength>0) and
- (IndexDWord(removedCharIndex[0],removedCharIndexLength,k) >= 0)
- then begin
- exit(False);
- end;
- {if (k = (i+1)) or
- ( (k = (i+2)) and UnicodeIsHighSurrogate(s[i]) )
- then
- lastUnblockedNonstarterCCC := 0;}
- pk := @s[k];
- if UnicodeIsHighSurrogate(pk^) then begin
- if (k = c) then
- exit(False);
- if UnicodeIsLowSurrogate(pk[1]) then
- puk := GetProps(pk[0],pk[1])
- else
- puk := GetProps(Word(pk^));
- end else begin
- puk := GetProps(Word(pk^));
- end;
- if (puk^.CCC = 0) or (lastUnblockedNonstarterCCC >= puk^.CCC) then
- exit(False);
- lastUnblockedNonstarterCCC := puk^.CCC;
- Result := True;
- end;
- procedure RemoveChar(APos : Integer);inline;
- begin
- if (removedCharIndexLength >= Length(removedCharIndex)) then
- SetLength(removedCharIndex,(2*removedCharIndexLength + 2));
- removedCharIndex[removedCharIndexLength] := APos;
- Inc(removedCharIndexLength);
- if UnicodeIsHighSurrogate(s[APos]) and (APos < c) and UnicodeIsLowSurrogate(s[APos+1]) then begin
- if (removedCharIndexLength >= Length(removedCharIndex)) then
- SetLength(removedCharIndex,(2*removedCharIndexLength + 2));
- removedCharIndex[removedCharIndexLength] := APos+1;
- Inc(removedCharIndexLength);
- end;
- end;
- procedure Inc_I();inline;
- begin
- if (removedCharIndexLength = 0) then begin
- Inc(i);
- Inc(ps);
- exit;
- end;
- while True do begin
- Inc(i);
- Inc(ps);
- if (IndexDWord(removedCharIndex[0],removedCharIndexLength,i) = -1) then
- Break;
- end;
- end;
- var
- surrogateState : Boolean;
- function MoveToNextChar() : Boolean;inline;
- begin
- Result := True;
- if UnicodeIsHighSurrogate(ps[0]) then begin
- if (i = c) then
- exit(False);
- if UnicodeIsLowSurrogate(ps[1]) then begin
- surrogateState := True;
- cp := ToUCS4(ps[0],ps[1]);
- end else begin
- surrogateState := False;
- cp := Word(ps[0]);
- end;
- end else begin
- surrogateState := False;
- cp := Word(ps[0]);
- end;
- end;
- procedure ClearPP(const AClearSuppressInfo : Boolean = True);inline;
- begin
- cl := nil;
- pp := nil;
- ppLevel := 0;
- if AClearSuppressInfo then begin
- suppressState.cl := nil;
- suppressState.CharCount := 0;
- end;
- end;
- function FindPropUCA() : Boolean;
- var
- candidateCL : PUCA_DataBook;
- begin
- pp := nil;
- if (cl = nil) then
- candidateCL := ACollation
- else
- candidateCL := cl;
- if surrogateState then begin
- while (candidateCL <> nil) do begin
- pp := GetPropUCA(ps[0],ps[1],candidateCL);
- if (pp <> nil) then
- break;
- candidateCL := candidateCL^.Base;
- end;
- end else begin
- while (candidateCL <> nil) do begin
- pp := GetPropUCA(ps[0],candidateCL);
- if (pp <> nil) then
- break;
- candidateCL := candidateCL^.Base;
- end;
- end;
- cl := candidateCL;
- Result := (pp <> nil);
- end;
- procedure AddWeightsAndClear();inline;
- var
- ctxNode : PUCA_PropItemContextTreeNodeRec;
- begin
- if (pp^.WeightLength > 0) then begin
- AddWeights(pp);
- end else
- if (LastKeyOwner.Length > 0) and pp^.Contextual and
- pp^.GetContext()^.Find(@LastKeyOwner.Chars[0],LastKeyOwner.Length,ctxNode) and
- (ctxNode^.Data.WeightCount > 0)
- then begin
- AddContextWeights(@ctxNode^.Data);
- end;
- //AddWeights(pp);
- ClearHistory();
- ClearPP();
- end;
- procedure StartMatch();
- procedure HandleLastChar();
- var
- ctxNode : PUCA_PropItemContextTreeNodeRec;
- begin
- while True do begin
- if pp^.IsValid() then begin
- if (pp^.WeightLength > 0) then
- AddWeights(pp)
- else
- if (LastKeyOwner.Length > 0) and pp^.Contextual and
- pp^.GetContext()^.Find(@LastKeyOwner.Chars[0],LastKeyOwner.Length,ctxNode) and
- (ctxNode^.Data.WeightCount > 0)
- then
- AddContextWeights(@ctxNode^.Data)
- else
- AddComputedWeights(cp){handle deletion of code point};
- break;
- end;
- if (cl^.Base = nil) then begin
- AddComputedWeights(cp);
- break;
- end;
- cl := cl^.Base;
- if not FindPropUCA() then begin
- AddComputedWeights(cp);
- break;
- end;
- end;
- end;
- var
- tmpCtxNode : PUCA_PropItemContextTreeNodeRec;
- begin
- ppLevel := 0;
- if not FindPropUCA() then begin
- AddComputedWeights(cp);
- ClearHistory();
- ClearPP();
- end else begin
- if (i = c) then begin
- HandleLastChar();
- end else begin
- if pp^.IsValid()then begin
- if (pp^.ChildCount = 0) then begin
- if (pp^.WeightLength > 0) then
- AddWeights(pp)
- else
- if (LastKeyOwner.Length > 0) and pp^.Contextual and
- pp^.GetContext()^.Find(@LastKeyOwner.Chars[0],LastKeyOwner.Length,tmpCtxNode) and
- (tmpCtxNode^.Data.WeightCount > 0)
- then
- AddContextWeights(@tmpCtxNode^.Data)
- else
- AddComputedWeights(cp){handle deletion of code point};
- ClearPP();
- ClearHistory();
- end else begin
- RecordStep();
- end
- end else begin
- if (pp^.ChildCount = 0) then begin
- AddComputedWeights(cp);
- ClearPP();
- ClearHistory();
- end else begin
- RecordStep();
- end;
- end ;
- end;
- end;
- end;
- function TryPermutation() : Boolean;
- var
- kk : Integer;
- b : Boolean;
- puk : PUC_Prop;
- ppk : PUCA_PropItemRec;
- begin
- Result := False;
- puk := GetProps(cp);
- if (puk^.CCC = 0) then
- exit;
- lastUnblockedNonstarterCCC := puk^.CCC;
- if surrogateState then
- kk := i + 2
- else
- kk := i + 1;
- while IsUnblockedNonstarter(kk) do begin
- b := UnicodeIsHighSurrogate(s[kk]) and (kk<c) and UnicodeIsLowSurrogate(s[kk+1]);
- if b then
- ppk := FindChild(ToUCS4(s[kk],s[kk+1]),pp)
- else
- ppk := FindChild(Word(s[kk]),pp);
- if (ppk <> nil) then begin
- pp := ppk;
- RemoveChar(kk);
- Inc(ppLevel);
- RecordStep();
- Result := True;
- if (pp^.ChildCount = 0 ) then
- Break;
- end;
- if b then
- Inc(kk);
- Inc(kk);
- end;
- end;
- procedure AdvanceCharPos();inline;
- begin
- if UnicodeIsHighSurrogate(ps[0]) and (i<c) and UnicodeIsLowSurrogate(ps[1]) then begin
- Inc(i);
- Inc(ps);
- end;
- Inc_I();
- end;
- var
- ok : Boolean;
- pp1 : PUCA_PropItemRec;
- cltemp : PUCA_DataBook;
- ctxNode : PUCA_PropItemContextTreeNodeRec;
- begin
- if (ALength = 0) then
- exit(nil);
- c := ALength;
- s := NormalizeNFD(AStr,c);
- c := Length(s);
- rl := 3*c;
- SetLength(r,rl);
- ral := 0;
- ps := @s[1];
- ClearPP();
- locHistoryTop := -1;
- removedCharIndexLength := 0;
- FillByte(suppressState,SizeOf(suppressState),0);
- LastKeyOwner.Length := 0;
- i := 1;
- while (i <= c) and MoveToNextChar() do begin
- if (pp = nil) then begin // Start Matching
- StartMatch();
- end else begin
- pp1 := FindChild(cp,pp);
- if (pp1 <> nil) then begin
- Inc(ppLevel);
- pp := pp1;
- if (pp^.ChildCount = 0) or (i = c) then begin
- ok := False;
- if pp^.IsValid() and (suppressState.CharCount = 0) then begin
- if (pp^.WeightLength > 0) then begin
- AddWeightsAndClear();
- ok := True;
- end else
- if (LastKeyOwner.Length > 0) and pp^.Contextual and
- pp^.GetContext()^.Find(@LastKeyOwner.Chars[0],LastKeyOwner.Length,ctxNode) and
- (ctxNode^.Data.WeightCount > 0)
- then begin
- AddContextWeights(@ctxNode^.Data);
- ClearHistory();
- ClearPP();
- ok := True;
- end
- end;
- if not ok then begin
- RecordDeletion();
- ok := False;
- while HasHistory() do begin
- GoBack();
- if pp^.IsValid() and
- ( ( (cl = suppressState.cl) and (ppLevel <> suppressState.CharCount) ) or
- ( (cl <> suppressState.cl) and (ppLevel < suppressState.CharCount) )
- )
- then begin
- AddWeightsAndClear();
- ok := True;
- Break;
- end;
- end;
- if not ok then begin
- cltemp := cl^.Base;
- if (cltemp <> nil) then begin
- ClearPP(False);
- cl := cltemp;
- Continue;
- end;
- end;
- if not ok then begin
- AddComputedWeights(cp);
- ClearHistory();
- ClearPP();
- end;
- end;
- end else begin
- RecordStep();
- end;
- end else begin
- // permutations !
- ok := False;
- if TryPermutation() and pp^.IsValid() then begin
- if (suppressState.CharCount = 0) then begin
- AddWeightsAndClear();
- Continue;
- end;
- while True do begin
- if pp^.IsValid() and
- (pp^.WeightLength > 0) and
- ( ( (cl = suppressState.cl) and (ppLevel <> suppressState.CharCount) ) or
- ( (cl <> suppressState.cl) and (ppLevel < suppressState.CharCount) )
- )
- then begin
- AddWeightsAndClear();
- ok := True;
- break;
- end;
- if not HasHistory() then
- break;
- GoBack();
- if (pp = nil) then
- break;
- end;
- end;
- if not ok then begin
- if pp^.IsValid() and (suppressState.CharCount = 0) then begin
- if (pp^.WeightLength > 0) then begin
- AddWeightsAndClear();
- ok := True;
- end else
- if (LastKeyOwner.Length > 0) and pp^.Contextual and
- pp^.GetContext()^.Find(@LastKeyOwner.Chars[0],LastKeyOwner.Length,ctxNode) and
- (ctxNode^.Data.WeightCount > 0)
- then begin
- AddContextWeights(@ctxNode^.Data);
- ClearHistory();
- ClearPP();
- ok := True;
- end
- end;
- if ok then
- Continue;
- end;
- if not ok then begin
- if (cl^.Base <> nil) then begin
- cltemp := cl^.Base;
- while HasHistory() do
- GoBack();
- pp := nil;
- ppLevel := 0;
- cl := cltemp;
- Continue;
- end;
- //walk back
- ok := False;
- while HasHistory() do begin
- GoBack();
- if pp^.IsValid() and
- (pp^.WeightLength > 0) and
- ( (suppressState.CharCount = 0) or
- ( ( (cl = suppressState.cl) and (ppLevel <> suppressState.CharCount) ) or
- ( (cl <> suppressState.cl) and (ppLevel < suppressState.CharCount) )
- )
- )
- then begin
- AddWeightsAndClear();
- ok := True;
- Break;
- end;
- end;
- if ok then begin
- AdvanceCharPos();
- Continue;
- end;
- if (pp <> nil) then begin
- AddComputedWeights(cp);
- ClearHistory();
- ClearPP();
- end;
- end;
- end;
- end;
- if surrogateState then begin
- Inc(ps);
- Inc(i);
- end;
- //
- Inc_I();
- end;
- SetLength(r,ral);
- Result := r;
- end;
- function ComputeSortKey(
- const AStr : PUnicodeChar;
- const ALength : SizeInt;
- const ACollation : PUCA_DataBook
- ) : TUCASortKey;
- var
- r : TUCA_PropWeightsArray;
- begin
- r := ComputeRawSortKey(AStr,ALength,ACollation);
- case ACollation^.VariableWeight of
- TUCA_VariableKind.ucaShifted : Result := FormKeyShifted(r,ACollation);
- TUCA_VariableKind.ucaBlanked : Result := FormKeyBlanked(r,ACollation);
- TUCA_VariableKind.ucaNonIgnorable : Result := FormKeyNonIgnorable(r,ACollation);
- TUCA_VariableKind.ucaShiftedTrimmed : Result := FormKeyShiftedTrimmed(r,ACollation);
- else
- Result := FormKeyShifted(r,ACollation);
- end;
- end;
- end.
|