| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724 | {%MainUnit sysutils.pp}{ TGenericStringBuilder }constructor TGenericStringBuilder.Create;begin  Create(DefaultCapacity,Maxint);end;constructor TGenericStringBuilder.Create(const AValue: SBString; aCapacity: Integer);begin  Create(aCapacity,Maxint);  if (system.Length(AValue)>0) then    Append(AValue);end;constructor TGenericStringBuilder.Create(const AValue: SBString; StartIndex, Alength,  aCapacity: Integer);begin  Create(Copy(AValue,StartIndex+1,Alength), aCapacity);end;constructor TGenericStringBuilder.Create(aCapacity, aMaxCapacity: Integer);begin  FMaxCapacity:=aMaxCapacity;  Capacity:=aCapacity;  FLength:=0;end;constructor TGenericStringBuilder.Create(aCapacity: Integer);begin  Create(aCapacity,MaxInt);end;constructor TGenericStringBuilder.Create(const AValue: SBString);begin  Create(aValue,DefaultCapacity);end;{ Enumerator }function TGenericStringBuilder.TCharEnumerator.GetCurrent: SBChar;begin  Result := FCurrentPosition^;  Inc(FCurrentPosition);end;function TGenericStringBuilder.TCharEnumerator.MoveNext: Boolean;begin  Result := FCurrentPosition < FEndPosition;end;function TGenericStringBuilder.GetEnumerator: TCharEnumerator;var  StartPosition: PSBChar;begin  StartPosition := @FData[0];  Result.FCurrentPosition := StartPosition;  Result.FEndPosition := StartPosition + Length;end;{ Property getter/setter }function TGenericStringBuilder.GetLength: Integer;begin  Result:=FLength;end;function TGenericStringBuilder.GetCapacity: Integer;begin  Result:=System.Length(FData);end;function TGenericStringBuilder.GetC(Index: Integer): SBChar;begin  CheckNegative(Index,'Index');  CheckRange(Index,0,Length);  Result:=FData[Index];end;procedure TGenericStringBuilder.SetC(Index: Integer; AValue: SBChar);begin  CheckNegative(Index,'Index');  CheckRange(Index,0,Length-1);  FData[Index]:=AValue;end;procedure TGenericStringBuilder.SetLength(AValue: Integer);begin  CheckNegative(AValue,'AValue');  CheckRange(AValue,0,MaxCapacity);  While AValue>Capacity do    Grow;  Flength:=AValue;end;{ Check functions }procedure TGenericStringBuilder.CheckRange(Idx, Count, MaxLen: Integer);begin  if (Idx<0) or (Idx+Count>MaxLen) then    Raise ERangeError.CreateFmt(SListIndexError,[Idx]);end;procedure TGenericStringBuilder.CheckNegative(const AValue: Integer;  const AName: SBString);begin  if (AValue<0) then    Raise ERangeError.CreateFmt(SParamIsNegative,[AName])end;{  These do the actual Appending/Inserting }procedure TGenericStringBuilder.DoAppend(const S: {$IFDEF SBUNICODE}SBString{$ELSE}RawByteString{$ENDIF});Var  L,SL : Integer;begin  SL:=System.Length(S);  if SL>0 then    begin    L:=Length;    Length:=L+SL;    Move(S[1], FData[L],SL*SizeOf(SBChar));    end;end;procedure TGenericStringBuilder.DoAppend(const AValue: TSBCharArray; Idx, aCount: Integer  );Var  L : integer;begin  L:=Length;  CheckRange(Idx,aCount,System.Length(AValue));  Length:=L+aCount;  Move(AValue[Idx],FData[L],aCount*SizeOf(SBChar));end;procedure TGenericStringBuilder.DoInsert(Index: Integer; const AValue: SBString);Var  ShiftLen,LV : Integer;begin  CheckRange(Index,0,Length-1);  LV:=System.Length(AValue);  ShiftLen:=Length-Index;  Length:=Length+LV;  Move(FData[Index],FData[Index+LV],ShiftLen*SizeOf(SBChar));  Move(AValue[1],FData[Index],LV*SizeOf(SBChar));end;procedure TGenericStringBuilder.DoInsert(Index: Integer; const AValue: TSBCharArray;  StartIndex, SBCharCount: Integer);Var  ShiftLen : Integer;begin  CheckRange(Index,0,Length-1);  CheckNegative(StartIndex,'StartIndex');  CheckNegative(SBCharCount,'SBCharCount');  CheckRange(StartIndex,SBCharCount,System.Length(AValue));  Length:=Length+SBCharCount;  ShiftLen:=Length-Index;  if ShiftLen> 0 then    Move(FData[Index], FData[Index+SBCharCount],ShiftLen*SizeOf(SBChar));  Move(AValue[StartIndex],FData[Index],SBCharCount*SizeOf(SBChar));end;{ Public routines for appending }function TGenericStringBuilder.Append(const AValue: UInt64): TGenericStringBuilder;begin  DoAppend(IntToStr(AValue));  Result:=self;end;function TGenericStringBuilder.Append(const AValue: TSBCharArray): TGenericStringBuilder;var  I,L: Integer;begin  I:=-1;  L:=System.Length(AValue);  If L=0 then    Exit(Self);  Repeat    Inc(I);  Until (I>=L) or (AValue[I]=#0);  DoAppend(AValue,0,I);  Result:=Self;end;function TGenericStringBuilder.Append(const AValue: Single): TGenericStringBuilder;begin  DoAppend(FloatToStr(AValue));  Result:=self;end;function TGenericStringBuilder.Append(const AValue: Word): TGenericStringBuilder;begin  Append(IntToStr(AValue));  Result:=self;end;function TGenericStringBuilder.Append(const AValue: Cardinal): TGenericStringBuilder;begin  DoAppend(IntToStr(AValue));  Result:=self;end;function TGenericStringBuilder.Append(const AValue: SBChar; RepeatCount: Integer  ): TGenericStringBuilder;begin  DoAppend(StringOfChar(AValue,RepeatCount));  Result:=Self;end;function TGenericStringBuilder.Append(const AValue: Shortint): TGenericStringBuilder;begin  DoAppend(IntToStr(AValue));  Result:=Self;end;function TGenericStringBuilder.Append(const AValue: SBChar): TGenericStringBuilder;begin  DoAppend(AValue);  Result:=Self;end;{$IFDEF SBUNICODE}function TGenericStringBuilder.Append(const AValue: AnsiChar): TGenericStringBuilder;begin  DoAppend(WideChar(AValue));  Result:=Self;end;{$ELSE}function TGenericStringBuilder.Append(const AValue: WideChar): TGenericStringBuilder;begin  // We don't know what the target encoding is ?  DoAppend(AnsiChar(AValue));  Result:=Self;end;function TGenericStringBuilder.Append(const AValue: UnicodeString): TGenericStringBuilder;begin  // We don't know what the target encoding is ?  DoAppend(AnsiString(AValue));  Result:=Self;end;{$ENDIF}function TGenericStringBuilder.Append(const AValue: Currency): TGenericStringBuilder;begin  DoAppend(CurrToStr(AValue));  Result:=Self;end;function TGenericStringBuilder.Append(const AValue: Boolean): TGenericStringBuilder;begin  DoAppend(BoolToStr(AValue, True));  Result:=Self;end;function TGenericStringBuilder.Append(const AValue: Byte): TGenericStringBuilder;begin  DoAppend(IntToStr(AValue));  Result:=Self;end;function TGenericStringBuilder.Append(const AValue: Double): TGenericStringBuilder;begin  DoAppend(FloatToStr(AValue));  Result:=Self;end;function TGenericStringBuilder.Append(const AValue: Int64): TGenericStringBuilder;begin  DoAppend(IntToStr(AValue));  Result:=Self;end;function TGenericStringBuilder.Append(const AValue: TObject): TGenericStringBuilder;begin  DoAppend(AValue.ToString);  Result:=Self;end;function TGenericStringBuilder.Append(const AValue: Smallint): TGenericStringBuilder;begin  DoAppend(IntToStr(AValue));  Result:=Self;end;function TGenericStringBuilder.Append(const AValue: LongInt): TGenericStringBuilder;begin  DoAppend(IntToStr(AValue));  Result:=Self;end;Function TGenericStringBuilder.Append(const AValue: TSBCharArray; StartIndex, SBCharCount: Integer): TGenericStringBuilder;begin  DoAppend(AValue,StartIndex,SBCharCount);  Result:=Self;end;Function TGenericStringBuilder.Append(const AValue: SBString; StartIndex, Count: Integer): TGenericStringBuilder;begin  CheckRange(StartIndex,Count,System.Length(AValue));  DoAppend(Copy(AValue,StartIndex+1,Count));  Result:=Self;end;function TGenericStringBuilder.Append(const AValue: PSBChar): TGenericStringBuilder;begin  DoAppend(AnsiString(AValue));  Result:=Self;end;{$IFDEF SBUNICODE}function TGenericStringBuilder.Append(const AValue: SBString): TGenericStringBuilder;begin  DoAppend(AValue);  Result:=Self;end;{$ENDIF}function TGenericStringBuilder.Append(const AValue: RawByteString): TGenericStringBuilder;begin  {$IFDEF SBUNICODE}  DoAppend(SBString(AValue));  {$ELSE}  DoAppend(AValue);  {$ENDIF}  Result:=Self;end;function TGenericStringBuilder.AppendFormat(const Fmt: SBString;  const Args: array of const): TGenericStringBuilder;begin  DoAppend(Format(Fmt,Args));  Result:=Self;end;function TGenericStringBuilder.Append(const Fmt: SBString;  const Args: array of const): TGenericStringBuilder;begin  DoAppend(Format(Fmt,Args));  Result:=Self;end;function TGenericStringBuilder.AppendLine: TGenericStringBuilder;begin  DoAppend(sLineBreak);  Result:=Self;end;function TGenericStringBuilder.AppendLine(const AValue: RawByteString): TGenericStringBuilder;begin  DoAppend(AValue);  Result:=AppendLine();end;procedure TGenericStringBuilder.Clear;begin  Length:=0;  Capacity:=DefaultCapacity;end;procedure TGenericStringBuilder.CopyTo(SourceIndex: Integer;  Var Destination: TSBCharArray; DestinationIndex: Integer; Count: Integer);begin  CheckNegative(Count,'Count');  CheckNegative(DestinationIndex,'DestinationIndex');  CheckRange(DestinationIndex,Count,System.Length(Destination));  if Count>0 then    begin    CheckRange(SourceIndex,Count,Length);    Move(FData[SourceIndex],Destination[DestinationIndex],Count * SizeOf(SBChar));    end;end;function TGenericStringBuilder.EnsureCapacity(aCapacity: Integer): Integer;begin  CheckRange(aCapacity,0,MaxCapacity);  if Capacity<aCapacity then    Capacity:=aCapacity;  Result:=Capacity;end;function TGenericStringBuilder.Equals(StringBuilder: TGenericStringBuilder): Boolean;begin  Result:=(StringBuilder<>nil);  if Result then    Result:=(Length=StringBuilder.Length)             and (MaxCapacity=StringBuilder.MaxCapacity)             and CompareMem(@FData[0],@StringBuilder.FData[0],Length*SizeOf(SBChar));end;procedure TGenericStringBuilder.Grow;var  NewCapacity: SizeInt;begin  NewCapacity:=Capacity*2;  if NewCapacity>MaxCapacity then    NewCapacity:=MaxCapacity;  Capacity:=NewCapacity;end;function TGenericStringBuilder.Insert(Index: Integer; const AValue: TObject  ): TGenericStringBuilder;begin  DoInsert(Index,AValue.ToString());  Result:=Self;end;function TGenericStringBuilder.Insert(Index: Integer; const AValue: Int64  ): TGenericStringBuilder;begin  DoInsert(Index,IntToStr(AValue));  Result:=Self;end;function TGenericStringBuilder.Insert(Index: Integer; const AValue: Single  ): TGenericStringBuilder;begin  DoInsert(Index,FloatToStr(AValue));  Result:=Self;end;function TGenericStringBuilder.Insert(Index: Integer; const AValue: SBString  ): TGenericStringBuilder;begin  DoInsert(Index,AValue);  Result:=Self;end;function TGenericStringBuilder.Insert(Index: Integer; const AValue: Word  ): TGenericStringBuilder;begin  DoInsert(Index,IntToStr(AValue));  Result:=Self;end;function TGenericStringBuilder.Insert(Index: Integer; const AValue: Shortint  ): TGenericStringBuilder;begin  DoInsert(Index, IntToStr(AValue));  Result:=Self;end;function TGenericStringBuilder.Insert(Index: Integer; const AValue: Currency  ): TGenericStringBuilder;begin  DoInsert(Index,CurrToStr(AValue));  Result:=Self;end;function TGenericStringBuilder.Insert(Index: Integer; const AValue: SBChar  ): TGenericStringBuilder;begin  DoInsert(Index,AValue);  Result:=Self;end;function TGenericStringBuilder.Insert(Index: Integer; const AValue: Byte  ): TGenericStringBuilder;begin  DoInsert(Index,IntToStr(AValue));  Result:=Self;end;function TGenericStringBuilder.Insert(Index: Integer; const AValue: Double  ): TGenericStringBuilder;begin  DoInsert(Index,FloatToStr(AValue));  Result:=Self;end;function TGenericStringBuilder.Insert(Index: Integer; const AValue: LongInt  ): TGenericStringBuilder;begin  DoInsert(Index,IntToStr(AValue));  Result:=Self;end;function TGenericStringBuilder.Insert(Index: Integer; const AValue: Smallint  ): TGenericStringBuilder;begin  DoInsert(Index,IntToStr(AValue));  Result:=Self;end;function TGenericStringBuilder.Insert(Index: Integer; const AValue: Boolean  ): TGenericStringBuilder;begin  DoInsert(Index,BoolToStr(AValue,True));  Result:=Self;end;function TGenericStringBuilder.Insert(Index: Integer; const AValue: SBString;  const aRepeatCount: Integer): TGenericStringBuilder;var  I: Integer;begin  for I:=0 to aRepeatCount-1 do    DoInsert(Index,AValue);  Result:=Self;end;function TGenericStringBuilder.Insert(Index: Integer; const AValue: TSBCharArray  ): TGenericStringBuilder;begin  DoInsert(Index,AValue,0,System.Length(AValue));  Result:=Self;end;function TGenericStringBuilder.Insert(Index: Integer; const AValue: TSBCharArray;  startIndex: Integer; SBCharCount: Integer): TGenericStringBuilder;begin  DoInsert(Index,AValue,StartIndex,SBCharCount);  Result:=Self;end;function TGenericStringBuilder.Insert(Index: Integer; const AValue: Cardinal  ): TGenericStringBuilder;begin  DoInsert(Index,IntToStr(AValue));  Result:=self;end;function TGenericStringBuilder.Insert(Index: Integer; const AValue: UInt64  ): TGenericStringBuilder;begin  DoInsert(Index,IntToStr(AValue));  Result:=self;end;procedure TGenericStringBuilder.Shrink;begin  if (Capacity div 4)>=Length then    Capacity:=Capacity div 2;end;function TGenericStringBuilder.Remove(StartIndex: Integer; RemLength: Integer  ): TGenericStringBuilder;Var  MoveIndex : Integer;begin  if (RemLength=0) then    exit(Self);  CheckNegative(RemLength,'RemLength');  CheckRange(StartIndex,0,Length);  MoveIndex:=StartIndex+RemLength;  CheckRange(MoveIndex,0,Length);  if (Length-Moveindex)>0 then    Move(FData[MoveIndex],FData[StartIndex],(Length-MoveIndex)*SizeOf(SBChar));  Length:=Length-RemLength;  Shrink;  Result:=Self;end;Function TGenericStringBuilder.Replace(const OldValue, NewValue: SBRawString; StartIndex, Count: Integer): TGenericStringBuilder;var  Cur : PSBChar;  CurIndex,MaxIndex : Integer;  OldLen, NewLen, Delta : Integer;  BC : SBChar;begin  if Count=0 then    Exit(Self);  // Some checks.  CheckNegative(StartIndex,'StartIndex');  CheckNegative(Count,'Count');  CheckRange(Startindex,Count,Length);  // Init  OldLen:=System.Length(OldValue);  NewLen:=System.Length(NewValue);  Delta:=NewLen-OldLen;  MaxIndex:=StartIndex+Count;  CurIndex:=StartIndex;  BC:=OldValue[1];  Cur:=@FData[StartIndex];  // Loop  while (CurIndex<Length-OldLen+1) and (CurIndex<MaxIndex) do    begin    if (Cur^=BC) then      begin      if 0=StrLComp(@FData[CurIndex],PSBChar(OldValue),OldLen) then        begin        // Do actual replace.        DoReplace(CurIndex,OldValue,NewValue);        Inc(CurIndex,NewLen-1);        // DoReplace may have reallocated memory, so changed pointers, reset pointer        Cur:=@FData[CurIndex];        // The max index must be increased/decreased with Delta        //  0123456789012        // 'zzbczzedeafzz' replace('e','qqqq',6,3) -> zzbczzqqqqdqqqqafzz        Inc(MaxIndex,Delta);        end;      end;    Inc(CurIndex);    Inc(Cur);    end;  Result:=Self;end;Function TGenericStringBuilder.Replace(const OldChar, NewChar: SBChar; StartIndex,  Count: Integer): TGenericStringBuilder;var  I : Integer;  Cur : PSBChar;begin  if Count=0 then    Exit(Self);  CheckNegative(StartIndex,'StartIndex');  CheckNegative(Count,'Count');  CheckRange(StartIndex,Count-1,Length);  Cur:=@FData[StartIndex];  For I:=1 to Count do    begin    if Cur^=OldChar then      Cur^:=NewChar;    Inc(Cur);    end;  Result:=Self;end;Function TGenericStringBuilder.Replace(const OldChar, NewChar: SBChar): TGenericStringBuilder;begin  Result:=Replace(OldChar,NewChar,0,Length);end;Function TGenericStringBuilder.Replace(const OldValue, NewValue: SBRawString): TGenericStringBuilder;begin  Result:=Replace(OldValue,NewValue,0,Length);end;procedure TGenericStringBuilder.SetCapacity(AValue: Integer);begin  if (AValue>FMaxCapacity) then    Raise ERangeError.CreateFmt(SListCapacityError,[AValue]);  if (AValue<Length) then    Raise ERangeError.CreateFmt(SListCapacityError,[AValue]);  System.SetLength(FData,AValue);end;{$IFDEF UNICODERTL}function TGenericStringBuilder.ToString: RTLString;{$ELSE}function TGenericStringBuilder.ToString: SBString;{$ENDIF}begin  Result:=ToString(0,Length);end;function TGenericStringBuilder.ToString(aStartIndex: Integer; aLength: Integer  ): SBString;begin  if (aLength=0) then    Result:=''  else    begin    CheckNegative(aStartIndex,'aStartIndex');    CheckNegative(aLength,'aLength');    CheckRange(aStartIndex,aLength,Length);    System.SetLength(Result,aLength);    Move(FData[aStartIndex],Result[1],aLength*SizeOf(SBChar));    end;end;procedure TGenericStringBuilder.DoReplace(Index: Integer; const Old, New: SBString);var  NVLen,OVLen,OLen,Delta,TailStart: Integer;begin  NVLen:=System.Length(New);  OVLen:=System.Length(Old);  Delta:=NVLen-OVLen;  if (Delta<>0) then    begin    OLen:=Length;    if (Delta>0) then      Length:=OLen+Delta;    TailStart:=Index+OVlen;    Move(FData[TailStart],FData[Index+NVLen],(OLen-TailStart)*SizeOf(SBChar));    if (Delta<0) then      Length:=OLen+Delta;    end;  Move(New[1],FData[Index],NVLen*SizeOf(SBChar));end;
 |