| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897 | {$IFNDEF FPC_DOTTEDUNITS}unit fpwidestring;{$ENDIF FPC_DOTTEDUNITS}{$mode objfpc}interface{$IFDEF FPC_DOTTEDUNITS}uses  System.CodePages.unicodedata;{$ELSE FPC_DOTTEDUNITS}uses  unicodedata;{$ENDIF FPC_DOTTEDUNITS}{$i rtldefs.inc}  function SetActiveCollation(const AName : UnicodeString) : Boolean;  function SetActiveCollation(const ACollation : PUCA_DataBook) : Boolean;  function GetActiveCollation() : PUCA_DataBook;var  DefaultCollationName : UnicodeString = '';implementation{$IFDEF FPC_DOTTEDUNITS}uses{$ifdef MSWINDOWS}  WinApi.Windows,{$endif MSWINDOWS}{$ifdef Unix}  UnixApi.CP,{$endif}  System.CharSet;{$ELSE FPC_DOTTEDUNITS}uses{$ifdef MSWINDOWS}  Windows,{$endif MSWINDOWS}{$ifdef Unix}  unixcp,{$endif}  charset;{$ENDIF FPC_DOTTEDUNITS}procedure fpc_rangeerror; [external name 'FPC_RANGEERROR'];{$ifdef MSWINDOWS}  function GetACP:UINT; external 'kernel32' name 'GetACP';{$endif MSWINDOWS}const  IgnoreInvalidSequenceFlag = True;var  OldManager : TUnicodeStringManager;{$ifdef FPC_HAS_FEATURE_THREADING}ThreadVar{$else FPC_HAS_FEATURE_THREADING}Var{$endif FPC_HAS_FEATURE_THREADING}  current_DefaultSystemCodePage : TSystemCodePage;  current_Map : punicodemap;  current_Collation : record    DataPtr : PUCA_DataBook;    Data    : TUCA_DataBook;  end;function SetActiveCollation(const ACollation : PUCA_DataBook) : Boolean;begin  Result := (ACollation <> nil);  if Result then begin    current_Collation.Data := ACollation^;    current_Collation.DataPtr := @current_Collation.Data;  end;end;function SetActiveCollation(const AName : UnicodeString) : Boolean;var  c : PUCA_DataBook;begin  c:=FindCollation(AName);  Result := (c <> nil);  if Result then    Result := SetActiveCollation(c);end;function GetActiveCollation() : PUCA_DataBook;begin  Result := current_Collation.DataPtr;end;{procedure error_CpNotFound(ACodePage:TSystemCodePage);begin  System.error(reCodesetConversion);end;}procedure InitThread;var  c : PUCA_DataBook;begin  current_DefaultSystemCodePage:=DefaultSystemCodePage;  current_Map:=getmap(current_DefaultSystemCodePage);  c:=nil;  if (DefaultCollationName<>'') then    c:=FindCollation(DefaultCollationName);  if (c=nil) and (GetCollationCount()>0) then    c:=FindCollation(0);  if (c<>nil) then    SetActiveCollation(c);end;procedure FiniThread;begin  current_Map:=nil;end;function FindMap(const cp: TSystemCodePage): punicodemap;inline;begin  if (cp=DefaultSystemCodePage) then    begin      { update current_Map in case the DefaultSystemCodePage has been changed }      if (current_DefaultSystemCodePage<>DefaultSystemCodePage) or not Assigned(current_Map) then        begin          FiniThread;          InitThread;        end;      Result:=current_Map;    end  else    Result:=getmap(cp);end;{ return value:  -1 if incomplete or invalid code point  0 if NULL character,  > 0 if that's the length in bytes of the code point }function UTF8CodePointLength(const Str: PAnsiChar; MaxLookAead: PtrInt): Ptrint;{... taken from ustrings.inc}var  p: PByte;  TempBYTE: Byte;  CharLen: SizeUint;  LookAhead: SizeUInt;  UC: SizeUInt;begin  if (Str=nil) then    exit(0);  p:=PByte(Str);  if (p^=0) then    exit(0);  p:=PByte(Str);  if (p^ and $80) = 0 then //One character US-ASCII,    exit(1);  TempByte:=p^;  CharLen:=0;  while (TempByte and $80)<>0 do    begin      TempByte:=(TempByte shl 1) and $FE;      Inc(CharLen);    end;  //Test for the "CharLen" conforms UTF-8 string  //This means the 10xxxxxx pattern.  if SizeUInt(CharLen-1)>MaxLookAead then //Insuficient chars in string to decode UTF-8 array    exit(-1);  for LookAhead := 1 to CharLen-1 do    begin      if ((p[LookAhead] and $80)<>$80) or         ((p[LookAhead] and $40)<>$00)      then        begin          //Invalid UTF-8 sequence, fallback.          exit(-1);        end;    end;  Result:=CharLen;  case CharLen of    1:  begin          //Not valid UTF-8 sequence          Result:=-1;        end;    2:  begin          //Two bytes UTF, convert it          UC:=(p^ and $1F) shl 6;          UC:=UC or (p[1] and $3F);          if UC <= $7F then            begin              //Invalid UTF sequence.              Result:=-1;            end;        end;    3:  begin          //Three bytes, convert it to unicode          UC:= (p^ and $0F) shl 12;          UC:= UC or ((p[1] and $3F) shl 6);          UC:= UC or ((p[2] and $3F));          if (UC <= $7FF) or (UC >= $FFFE) or ((UC >= $D800) and (UC <= $DFFF)) then            begin              //Invalid UTF-8 sequence              Result:=-1;            End;        end;    4:  begin          //Four bytes, convert it to two unicode characters          UC:= (p^ and $07) shl 18;          UC:= UC or ((p[1] and $3F) shl 12);          UC:= UC or ((p[2] and $3F) shl 6);          UC:= UC or ((p[3] and $3F));          if (UC < $10000) or (UC > $10FFFF) then            begin              Result:=-1;            end        end;    5,6,7:  begin              //Invalid UTF8 to unicode conversion,              //mask it as invalid UNICODE too.              Result:=-1;            end;  end;end;{ return value:  -1 if incomplete or invalid code point  0 if NULL character,  > 0 if that's the length in bytes of the code point }function CodePointLength(const Str: PAnsiChar; MaxLookAead: PtrInt): PtrInt;var  p : PByte;begin  if (current_DefaultSystemCodePage=CP_UTF8) then    exit(UTF8CodePointLength(Str,MaxLookAead));  if (Str=nil) then    exit(0);  p:=PByte(Str);  if (p^=0) then    exit(0);  if (current_Map=nil) then    exit(1);  if (p^>current_Map^.lastchar) then    exit(-1);  case current_Map^.map[p^].flag of    umf_undefined : Result:=-1;    umf_leadbyte  :      begin        if (MaxLookAead>0) then          Result:=2        else          Result:=-1;      end;    else      Result:=1;  end;end;procedure Unicode2AnsiMove(source:punicodechar;var dest:RawByteString;cp : TSystemCodePage;len:SizeInt);var  locSource : punicodechar;  locMap : punicodemap;  destBuffer : PAnsiChar;  destLen,actualLen, i : SizeInt;  blockLen : SizeInt;begin  if (len=0) then    begin      SetLength(dest,0);      exit;    end;  if (cp=CP_UTF8) then    begin      destLen:=UnicodeToUtf8(nil,High(SizeUInt),source,len);      SetLength(dest,destLen-1);      UnicodeToUtf8(@dest[1],destLen,source,len);      SetCodePage(dest,cp,False);      exit;    end;  if (cp=CP_UTF16) then    begin      destLen:=len*SizeOf(UnicodeChar);      SetLength(dest,destLen);      Move(source^,dest[1],destLen);      SetCodePage(dest,cp,False);      exit;    end;  locMap:=FindMap(cp);  if (locMap=nil) then    begin      DefaultUnicode2AnsiMove(source,dest,DefaultSystemCodePage,len);      exit;    end;  destLen:=3*len;  SetLength(dest,destLen);  destBuffer:=@dest[1];  actualLen:=0;  locSource:=source;  for i:=1 to len do    begin      blockLen:=getascii(tunicodechar(locSource^),locMap,destBuffer,(destLen-actualLen));      if (blockLen<0) then        begin          destLen:=destLen + 3*(1+len-i);          SetLength(dest,destLen);          destBuffer:=@dest[1];          blockLen:=getascii(tunicodechar(locSource^),locMap,destBuffer,(destLen-actualLen));        end;      Inc(destBuffer,blockLen);      actualLen:=actualLen+blockLen;      Inc(locSource);    end;  if (actualLen<>Length(dest)) then    SetLength(dest,actualLen);  if (Length(dest)>0) then    SetCodePage(dest,cp,False);end;procedure Ansi2UnicodeMove(source:PAnsiChar; cp:TSystemCodePage; var dest:UnicodeString; len:SizeInt);var  locMap : punicodemap;  destLen : SizeUInt;begin  if (len<=0) then    begin      SetLength(dest,0);      exit;    end;  if (cp=CP_UTF8) then    begin      destLen:=Utf8ToUnicode(nil,high(SizeUint),source,len);      if destLen > 0 then        SetLength(dest,destLen-1)      else        SetLength(dest,0);      Utf8ToUnicode(@dest[1],destLen,source,len);      exit;    end;  if (cp=CP_UTF16) then    begin      //what if (len mod 2) > 0 ?      destLen:=len div SizeOf(UnicodeChar);      SetLength(dest,destLen);      Move(source^,dest[1],(destLen*SizeOf(UnicodeChar)));      exit;    end;  locMap:=FindMap(cp);  if (locMap=nil) then    begin      DefaultAnsi2UnicodeMove(source,DefaultSystemCodePage,dest,len);      exit;    end;  destLen:=getunicode(source,len,locMap,nil);  SetLength(dest,destLen);  getunicode(source,len,locMap,tunicodestring(@dest[1]));end;{$ifdef MSWINDOWS}procedure Ansi2WideMove(source:PAnsiChar; cp:TSystemCodePage; var dest:WideString; len:SizeInt);var  locMap : punicodemap;  destLen : SizeInt;begin  if (len<=0) then    begin      SetLength(dest,0);      exit;    end;  locMap:=FindMap(cp);  if (locMap=nil) then    begin      DefaultAnsi2WideMove(source,DefaultSystemCodePage,dest,len);      exit;    end;  destLen:=getunicode(source,len,locMap,nil);  SetLength(dest,destLen);  getunicode(source,len,locMap,tunicodestring(@dest[1]));end;{$endif MSWINDOWS}function UpperUnicodeString(const S: UnicodeString): UnicodeString;begin  if (UnicodeToUpper(S,IgnoreInvalidSequenceFlag,Result) <> 0) then    system.error(reRangeError);end;function UpperWideString(const S: WideString): WideString;var  u : UnicodeString;begin  u:=s;  Result:=UpperUnicodeString(u);end;function LowerUnicodeString(const S: UnicodeString): UnicodeString;begin  if (UnicodeToLower(S,IgnoreInvalidSequenceFlag,Result) <> 0) then    system.error(reRangeError);end;function LowerWideString(const S: WideString): WideString;var  u : UnicodeString;begin  u:=s;  Result:=LowerUnicodeString(u);end;function CompareUnicodeStringUCA(p1,p2:PUnicodeChar; l1,l2:PtrInt) : PtrInt;inline;begin  Result := IncrementalCompareString(p1,l1,p2,l2,current_Collation.DataPtr);end;function CompareUnicodeString(p1,p2:PUnicodeChar; l1,l2:PtrInt) : PtrInt;inline;begin  if (Pointer(p1)=Pointer(p2)) then    exit(0);  if (l1=0) then    exit(-l2);  if (l2=0) then    exit(l1);  Result := CompareUnicodeStringUCA(p1,p2,l1,l2);end;type  TChangedPropsRecord = record    ComparisonStrength : Byte;  end;const  SECONDARY_STRENGTH_LEVEL = 2;function CompareUnicodeString(const s1, s2 : UnicodeString;Options : TCompareOptions) : PtrInt;  function DoCompare() : PtrInt;  var    changedProps : TChangedPropsRecord;  begin    changedProps.ComparisonStrength := current_Collation.Data.ComparisonStrength;    try      if (coIgnoreCase in Options) then        current_Collation.Data.ComparisonStrength := SECONDARY_STRENGTH_LEVEL;      Result:=CompareUnicodeString(                PUnicodeChar(Pointer(s1)),                PUnicodeChar(Pointer(s2)),                Length(s1),Length(s2)              );    finally      current_Collation.Data.ComparisonStrength := changedProps.ComparisonStrength;    end;  end;begin  if (current_Collation.DataPtr=nil) then    exit(OldManager.CompareUnicodeStringProc(s1,s2,Options));  if (Options=[]) then begin    exit(      CompareUnicodeString(           PUnicodeChar(Pointer(s1)),           PUnicodeChar(Pointer(s2)),           Length(s1),Length(s2)      )    );  end;  Result:=DoCompare();end;function CompareWideString(const s1, s2 : WideString; Options : TCompareOptions) : PtrInt;  function DoCompare() : PtrInt;  var    changedProps : TChangedPropsRecord;  begin    changedProps.ComparisonStrength := current_Collation.Data.ComparisonStrength;    try      if (coIgnoreCase in Options) then        current_Collation.Data.ComparisonStrength := SECONDARY_STRENGTH_LEVEL;      Result:=CompareUnicodeString(                PUnicodeChar(Pointer(s1)),                PUnicodeChar(Pointer(s2)),                Length(s1),Length(s2)              );    finally      current_Collation.Data.ComparisonStrength := changedProps.ComparisonStrength;    end;  end;begin  if (current_Collation.DataPtr=nil) then    exit(OldManager.CompareUnicodeStringProc(s1,s2,Options));  if (Options=[]) then begin    exit(      CompareUnicodeString(           PUnicodeChar(Pointer(s1)),           PUnicodeChar(Pointer(s2)),           Length(s1),Length(s2)      )    );  end;  Result:=DoCompare();end;function CompareTextUnicodeString(const s1, s2 : UnicodeString) : PtrInt;begin  Result:=CompareUnicodeString(s1,s2,[coIgnoreCase]);end;function CompareTextWideString(const s1, s2 : WideString) : PtrInt;begin  Result:=CompareWideString(s1,s2,[coIgnoreCase]);end;procedure EnsureAnsiLen(var S: AnsiString; const len: SizeInt); inline;begin  if (len>length(s)) then    if (length(s) < 10*256) then      setlength(s,length(s)+10)    else      setlength(s,length(s)+length(s) shr 8);end;procedure ConcatCharToAnsiStr(const c: AnsiChar; var S: AnsiString; var index: SizeInt);begin  EnsureAnsiLen(s,index);  pansichar(@s[index])^:=c;  inc(index);end;function UpperAnsiString(const s : ansistring) : ansistring;var  p        : PAnsiChar;  i,resindex : SizeInt;  mblen    : SizeInt;  us,usl   : UnicodeString;  locMap   : punicodemap;  ulen,slen : SizeUint;  k,aalen,ai : SizeInt;  aa       : array[0..8] of AnsiChar;begin  if (Length(s)=0) then    exit('');  if (DefaultSystemCodePage=CP_UTF8) then    begin      //convert to UnicodeString,uppercase,convert back to utf8      ulen:=Utf8ToUnicode(nil,high(SizeUint),@s[1],Length(s));      if ulen>0 then        SetLength(us,ulen-1);      Utf8ToUnicode(@us[1],ulen,@s[1],Length(s));      us:=UpperUnicodeString(us);      ulen:=Length(us);      slen:=UnicodeToUtf8(nil,high(SizeUInt),@us[1],ulen);      SetLength(Result,slen);      UnicodeToUtf8(@Result[1],slen,@us[1],ulen);      exit;    end;  locMap:=FindMap(DefaultSystemCodePage);  if (locMap=nil) then    exit(System.UpCase(s));  SetLength(us,2);  p:=@s[1];  slen:=length(s);  SetLength(result,slen+10);  i:=1;  resindex:=1;  while (i<=slen) do    begin      mblen:=CodePointLength(p,slen-i);      if (mblen<=0) then        begin          ConcatCharToAnsiStr(p^,result,resindex);          mblen:=1;        end      else        begin          SetLength(us,2);          ulen:=getunicode(p,mblen,locMap,@us[1]);          if (Length(us)<>ulen) then            SetLength(us,ulen);          usl:=UpperUnicodeString(us);          for k:=1 to Length(usl) do            begin              aalen:=getascii(tunicodechar(us[k]),locMap,@aa[Low(aa)],Length(aa));              for ai:=0 to aalen-1 do                ConcatCharToAnsiStr(aa[ai],result,resindex);            end;        end;      Inc(p,mblen);    end;  SetLength(result,resindex-1);end;function LowerAnsiString(const s : ansistring) : ansistring;var  p        : PAnsiChar;  i,resindex : SizeInt;  mblen    : SizeInt;  us,usl   : UnicodeString;  locMap   : punicodemap;  k,aalen,ai : SizeInt;  slen, ulen : SizeUInt;  aa       : array[0..8] of AnsiChar;begin  if (Length(s)=0) then    exit('');  if (DefaultSystemCodePage=CP_UTF8) then    begin      //convert to UnicodeString,lowercase,convert back to utf8      ulen:=Utf8ToUnicode(nil,high(SizeUInt),@s[1],Length(s));      if ulen>0 then        SetLength(us,ulen-1);      Utf8ToUnicode(@us[1],ulen,@s[1],Length(s));      us:=LowerUnicodeString(us);      ulen:=Length(us);      slen:=UnicodeToUtf8(nil,high(SizeUInt),@us[1],ulen);      SetLength(Result,slen);      UnicodeToUtf8(@Result[1],slen,@us[1],ulen);      exit;    end;  locMap:=FindMap(DefaultSystemCodePage);  if (locMap=nil) then    exit(System.LowerCase(s));  SetLength(us,2);  p:=@s[1];  slen:=length(s);  SetLength(result,slen+10);  i:=1;  resindex:=1;  while (i<=slen) do    begin      mblen:=CodePointLength(p,slen-i);      if (mblen<=0) then        begin          ConcatCharToAnsiStr(p^,result,resindex);          mblen:=1;        end      else        begin          SetLength(us,2);          ulen:=getunicode(p,mblen,locMap,@us[1]);          if (Length(us)<>ulen) then            SetLength(us,ulen);          usl:=LowerUnicodeString(us);          for k:=1 to Length(usl) do            begin              aalen:=getascii(tunicodechar(us[k]),locMap,@aa[Low(aa)],Length(aa));              for ai:=0 to aalen-1 do                ConcatCharToAnsiStr(aa[ai],result,resindex);            end;        end;      Inc(p,mblen);    end;  SetLength(result,resindex-1);end;procedure ansi2pchar(const s: ansistring; const orgp: pansichar; out p: pansichar);var  newlen: sizeint;begin  newlen:=length(s);  if newlen>strlen(orgp) then    fpc_rangeerror;  p:=orgp;  if (newlen>0) then    move(s[1],p[0],newlen);  p[newlen]:=#0;end;function AnsiStrLower(Str: PAnsiChar): PAnsiChar;var  temp: ansistring;begin  temp:=LowerAnsiString(str);  ansi2pchar(temp,str,result);end;function AnsiStrUpper(Str: PAnsiChar): PAnsiChar;var  temp: ansistring;begin  temp:=UpperAnsiString(str);  ansi2pchar(temp,str,result);end;function CharLengthPChar(const Str: PAnsiChar): PtrInt;var  len:PtrInt;  nextlen: ptrint;  s: PAnsiChar;begin  Result:=0;  if (Str=nil) or (Byte(Str^)=0) then    exit;  s:=str;  len:=strlen(s);  repeat    nextlen:=CodePointLength(s,len);    { skip invalid/incomplete sequences }    if (nextlen<0) then      nextlen:=1;    Inc(result,nextlen);    Inc(s,nextlen);    Dec(len,nextlen);  until (nextlen=0);end;function InternalCompareStrAnsiString(  const S1, S2     : PAnsiChar;  const Len1, Len2 : PtrUInt) : PtrInt;inline;var  a, b : UnicodeString;begin  a := '';  Ansi2UnicodeMove(S1,DefaultSystemCodePage,a,Len1);  b := '';  Ansi2UnicodeMove(S2,DefaultSystemCodePage,b,Len2);  Result := CompareUnicodeString(a,b,[]);end;function StrLCompAnsiString(S1, S2: PAnsiChar; MaxLen: PtrUInt): PtrInt;begin  if (current_Collation.DataPtr=nil) then    exit(OldManager.StrLCompAnsiStringProc(s1,s2,MaxLen));  if (MaxLen=0) then    exit(0);  Result := InternalCompareStrAnsiString(S1,S2,MaxLen,MaxLen);end;function CompareStrAnsiString(const S1, S2: ansistring): PtrInt;var  l1, l2 : PtrInt;begin  if (current_Collation.DataPtr=nil) and Assigned(OldManager.CompareStrAnsiStringProc) then    Exit(OldManager.CompareStrAnsiStringProc(s1,s2));  if (Pointer(S1)=Pointer(S2)) then    Exit(0);  l1:=Length(S1);  l2:=Length(S2);  if (l1=0) or (l2=0) then    Exit(l1-l2);  Result := InternalCompareStrAnsiString(@S1[1],@S2[1],l1,l2);end;function CompareTextAnsiString(const S1, S2: ansistring): PtrInt;var  a,b : ansistring;begin  a:=UpperAnsistring(s1);  b:=UpperAnsistring(s2);  Result:=CompareStrAnsiString(a,b);end;function StrCompAnsiString(S1, S2: PAnsiChar): PtrInt;var  l1,l2 : PtrInt;begin  if (current_Collation.DataPtr=nil) then    exit(OldManager.StrCompAnsiStringProc(s1,s2));  l1:=strlen(S1);  l2:=strlen(S2);  Result := InternalCompareStrAnsiString(S1,S2,l1,l2);end;function StrLICompAnsiString(S1, S2: PAnsiChar; MaxLen: PtrUInt): PtrInt;var  a, b: ansistring;begin  if (MaxLen=0) then    exit(0);  SetLength(a,MaxLen);  Move(s1^,a[1],MaxLen);  SetLength(b,MaxLen);  Move(s2^,b[1],MaxLen);  Result:=CompareTextAnsiString(a,b);end;function StrICompAnsiString(S1, S2: PAnsiChar): PtrInt;begin  Result:=CompareTextAnsiString(ansistring(s1),ansistring(s2));end;function StrLowerAnsiString(Str: PAnsiChar): PAnsiChar;var  temp: ansistring;begin  temp:=LowerAnsiString(str);  ansi2pchar(temp,str,result);end;function StrUpperAnsiString(Str: PAnsiChar): PAnsiChar;var  temp: ansistring;begin  temp:=UpperAnsiString(str);  ansi2pchar(temp,str,result);end;//------------------------------------------------------------------------------procedure SetPascalWideStringManager();var  locWideStringManager : TUnicodeStringManager;begin  OldManager := widestringmanager;  locWideStringManager:=widestringmanager;  With locWideStringManager do    begin      Wide2AnsiMoveProc:=@Unicode2AnsiMove;{$ifndef FPC_WIDESTRING_EQUAL_UNICODESTRING}      Ansi2WideMoveProc:=@Ansi2WideMove;      UpperWideStringProc:=@UpperWideString;      LowerWideStringProc:=@LowerWideString;      CompareWideStringProc:=@CompareWideString;{$else FPC_WIDESTRING_EQUAL_UNICODESTRING}      Ansi2WideMoveProc:=@Ansi2UnicodeMove;      UpperWideStringProc:=@UpperUnicodeString;      LowerWideStringProc:=@LowerUnicodeString;      CompareWideStringProc:=@CompareUnicodeString;{$endif FPC_WIDESTRING_EQUAL_UNICODESTRING}      CharLengthPCharProc:=@CharLengthPChar;      CodePointLengthProc:=@CodePointLength;      UpperAnsiStringProc:=@UpperAnsiString;      LowerAnsiStringProc:=@LowerAnsiString;      CompareStrAnsiStringProc:=@CompareStrAnsiString;      CompareTextAnsiStringProc:=@CompareTextAnsiString;      StrCompAnsiStringProc:=@StrCompAnsiString;      StrICompAnsiStringProc:=@StrICompAnsiString;      StrLCompAnsiStringProc:=@StrLCompAnsiString;      StrLICompAnsiStringProc:=@StrLICompAnsiString;      StrLowerAnsiStringProc:=@StrLowerAnsiString;      StrUpperAnsiStringProc:=@StrUpperAnsiString;      ThreadInitProc:=@InitThread;      ThreadFiniProc:=@FiniThread;      { Unicode }      Unicode2AnsiMoveProc:=@Unicode2AnsiMove;      Ansi2UnicodeMoveProc:=@Ansi2UnicodeMove;      UpperUnicodeStringProc:=@UpperUnicodeString;      LowerUnicodeStringProc:=@LowerUnicodeString;      CompareUnicodeStringProc:=@CompareUnicodeString;    end;  SetUnicodeStringManager(locWideStringManager);  DefaultUnicodeCodePage:=CP_UTF16;{$ifdef MSWINDOWS}  DefaultSystemCodePage:=GetACP();{$ELSE MSWINDOWS} {$ifdef UNIX}  DefaultSystemCodePage:=GetSystemCodepage;  if (DefaultSystemCodePage = CP_NONE) then    DefaultSystemCodePage:=CP_UTF8;  {$ifdef FPCRTL_FILESYSTEM_UTF8}  DefaultFileSystemCodePage:=CP_UTF8;  {$else}  DefaultFileSystemCodePage:=DefaultSystemCodepage;  {$endif}  DefaultRTLFileSystemCodePage:=DefaultFileSystemCodePage; {$ELSE UNIX}  if Assigned (WideStringManager.GetStandardCodePageProc) then   DefaultSystemCodePage := WideStringManager.GetStandardCodePageProc (scpAnsi)  else   DefaultSystemCodePage := CP_NONE;  DefaultFileSystemCodePage := DefaultSystemCodePage;  DefaultRTLFileSystemCodePage := DefaultSystemCodePage; {$endif UNIX}{$endif MSWINDOWS}end;initialization  current_Collation.DataPtr := nil;  SetPascalWideStringManager();  InitThread();finalization  FiniThread();end.
 |