{ $Id$ This file is part of the Free Component Library (FCL) Copyright (c) 1999-2000 by the Free Pascal development team See the file COPYING.FPC, included in this distribution, for details about the copyright. 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. **********************************************************************} {****************************************************************************} {* TList *} {****************************************************************************} { TList = class(TObject) private FList: PPointerList; FCount: Integer; FCapacity: Integer; } Const // Ratio of Pointer and Word Size. WordRatio = SizeOf(Pointer) Div SizeOf(Word); function TList.Get(Index: Integer): Pointer; begin If (Index<0) or (Index>=FCount) then Error(SListIndexError,Index); Result:=FList^[Index]; end; procedure TList.Grow; begin // Only for compatibility with Delphi. Not needed. end; procedure TList.Put(Index: Integer; Item: Pointer); begin if (Index<0) or (Index>=FCount) then Error(SListIndexError,Index); Flist^[Index]:=Item; end; procedure TList.SetCapacity(NewCapacity: Integer); Var NewList,ToFree : PPointerList; begin If (NewCapacity<0) or (NewCapacity>MaxListSize) then Error (SListCapacityError,NewCapacity); if NewCapacity=FCapacity then exit; ReallocMem(FList,SizeOf(Pointer)*NewCapacity); FCapacity:=NewCapacity; end; procedure TList.SetCount(NewCount: Integer); begin If (NewCount<0) or (NewCount>MaxListSize)then Error(SListCountError,NewCount); If NewCountFCount then begin If NewCount>FCapacity then SetCapacity (NewCount); If FCount=FCount) then Error (SListIndexError,Index); FCount:=FCount-1; System.Move (FList^[Index+1],FList^[Index],(FCount-Index)*SizeOf(Pointer)); end; class procedure TList.Error(const Msg: string; Data: Integer); begin Raise EListError.CreateFmt(Msg,[Data]) at get_caller_addr(get_frame); end; procedure TList.Exchange(Index1, Index2: Integer); var Temp : Pointer; begin If ((Index1>=FCount) or (Index1<0)) then Error(SListIndexError,Index1); If ((Index2>=FCount) or (Index2<0)) then Error(SListIndexError,Index2); Temp:=FList^[Index1]; FList^[Index1]:=FList^[Index2]; FList^[Index2]:=Temp; end; function TList.Expand: TList; Var IncSize : Longint; begin if FCount3 then IncSize:=IncSize+4; if FCapacity>8 then IncSize:=IncSize+8; SetCapacity(FCapacity+IncSize); Result:=Self; end; function TList.First: Pointer; begin If FCount=0 then Result:=Nil else Result:=Items[0]; end; function TList.IndexOf(Item: Pointer): Integer; begin Result:=0; While (ResultItem) do Result:=Result+1; If Result=FCount then Result:=-1; end; procedure TList.Insert(Index: Integer; Item: Pointer); begin If (Index<0) or (Index>FCount )then Error(SlistIndexError,Index); IF FCount=FCapacity Then Self.Expand; If IndexCount-1)) then Error(SListIndexError,CurIndex); If (NewINdex<0) then Error(SlistIndexError,NewIndex); Temp:=FList^[CurIndex]; Self.Delete(CurIndex); // ?? If NewIndex>CurIndex then NewIndex:=NewIndex-1; // Newindex changes when deleting ?? Self.Insert (NewIndex,Temp); end; function TList.Remove(Item: Pointer): Integer; begin Result:=IndexOf(Item); If Result<>-1 then Self.Delete (Result); end; Procedure TList.Pack; Var {Last,I,J,}Runner : Longint; begin // Not the fastest; but surely correct For Runner:=Fcount-1 downto 0 do if Items[Runner]=Nil then Self.Delete(Runner); { The following may be faster in case of large and defragmented lists If count=0 then exit; Runner:=0;I:=0; TheLast:=Count; while runnerNil) and (RunnerNil) and (J0 Do I:=I+1; While Compare(P,FList^[J])<0 Do J:=J-1; If I<=J then begin Q:=Flist^[I]; Flist^[I]:=FList^[J]; FList^[J]:=Q; I:=I+1; J:=j-1; end; Until I>J; If L=R; end; procedure TList.Sort(Compare: TListSortCompare); begin If Not Assigned(FList) or (FCount<2) then exit; QuickSort (Flist, 0, FCount-1,Compare); end; {****************************************************************************} {* TThreadList *} {****************************************************************************} constructor TThreadList.Create; begin inherited Create; //InitializeCriticalSection(FLock); FList := TList.Create; end; destructor TThreadList.Destroy; begin LockList; try FList.Free; inherited Destroy; finally UnlockList; end; end; procedure TThreadList.Add(Item: Pointer); begin Locklist; try //make sure it's not already in the list if FList.indexof(Item) = -1 then FList.Add(Item); finally UnlockList; end; end; procedure TThreadList.Clear; begin Locklist; try FList.Clear; finally UnLockList; end; end; function TThreadList.LockList: TList; begin Result := FList; end; procedure TThreadList.Remove(Item: Pointer); begin LockList; try FList.Remove(Item); finally UnlockList; end; end; procedure TThreadList.UnlockList; begin end; { $Log$ Revision 1.4 2000-11-17 13:39:49 sg * Extended Error methods so that exceptions are raised from the caller's address instead of the Error method Revision 1.3 2000/09/14 18:39:31 michael + Fixed setcapacity Revision 1.2 2000/07/13 11:32:59 michael + removed logs }