mirror of
				https://gitlab.com/freepascal.org/lazarus/lazarus.git
				synced 2025-10-26 02:21:42 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			265 lines
		
	
	
		
			8.8 KiB
		
	
	
	
		
			ObjectPascal
		
	
	
	
	
	
			
		
		
	
	
			265 lines
		
	
	
		
			8.8 KiB
		
	
	
	
		
			ObjectPascal
		
	
	
	
	
	
| {
 | |
|  ***************************************************************************
 | |
|  *                                                                         *
 | |
|  *   This source is free software; you can redistribute it and/or modify   *
 | |
|  *   it under the terms of the GNU General Public License as published by  *
 | |
|  *   the Free Software Foundation; either version 2 of the License, or     *
 | |
|  *   (at your option) any later version.                                   *
 | |
|  *                                                                         *
 | |
|  *   This code 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.  See the GNU     *
 | |
|  *   General Public License for more details.                              *
 | |
|  *                                                                         *
 | |
|  *   A copy of the GNU General Public License is available on the World    *
 | |
|  *   Wide Web at <http://www.gnu.org/copyleft/gpl.html>. You can also      *
 | |
|  *   obtain it by writing to the Free Software Foundation,                 *
 | |
|  *   Inc., 51 Franklin Street - Fifth Floor, Boston, MA 02110-1335, USA.   *
 | |
|  *                                                                         *
 | |
|  ***************************************************************************
 | |
| 
 | |
|   Author: Mattias Gaertner
 | |
| 
 | |
|   Abstract:
 | |
|     The TResourceCodeTool provides functions to find, add and delete resources
 | |
|     in resource files.
 | |
| }
 | |
| unit ResourceCodeTool;
 | |
| 
 | |
| {$mode objfpc}{$H+}
 | |
| 
 | |
| interface
 | |
| 
 | |
| uses
 | |
|   Classes, SysUtils, KeywordFuncLists, MultiKeyWordListTool, CodeCache,
 | |
|   CodeAtom, BasicCodeTools;
 | |
| 
 | |
| type
 | |
|   TResourceCodeTool = class(TMultiKeyWordListCodeTool)
 | |
|   protected
 | |
|     procedure SetSource(ACode: TCodeBuffer);
 | |
|   public
 | |
|     // lazarus resources
 | |
|     function FindLazarusResourceHeaderComment(ResourceCode: TCodeBuffer
 | |
|           ): TAtomPosition;
 | |
|     function AddLazarusResourceHeaderComment(ResourceCode: TCodeBuffer;
 | |
|           const Comment: string): boolean;
 | |
|     function FindLazarusResource(ResourceCode: TCodeBuffer;
 | |
|           const ResourceName: string; StartPos: integer): TAtomPosition;
 | |
|     function FindAllLazarusResource(ResourceCode: TCodeBuffer;
 | |
|           const ResourceName: string; StartPos: integer): TAtomList;
 | |
|     function AddLazarusResource(ResourceCode: TCodeBuffer;
 | |
|           const ResourceName, ResourceData: string): boolean;
 | |
|     function RemoveLazarusResource(ResourceCode: TCodeBuffer;
 | |
|           const ResourceName: string): boolean;
 | |
|     function RemoveLazarusResourceEx(ResourceCode: TCodeBuffer;
 | |
|           const ResourceName: string; AllExceptFirst: boolean;
 | |
|           out First: TAtomPosition): boolean;
 | |
|   end;
 | |
|   
 | |
|   TResourceCodeToolError = class(Exception)
 | |
|   end;
 | |
| 
 | |
| implementation
 | |
| 
 | |
| { TResourceCodeTool }
 | |
| 
 | |
| procedure TResourceCodeTool.SetSource(ACode: TCodeBuffer);
 | |
| begin
 | |
|   ClearLastError;
 | |
|   Src:=ACode.Source;
 | |
|   SrcLen:=length(Src);
 | |
|   CurPos:=StartAtomPosition;
 | |
|   LastAtoms.Clear;
 | |
|   CurNode:=nil;
 | |
|   DoDeleteNodes(Tree.Root);
 | |
| end;
 | |
| 
 | |
| function TResourceCodeTool.FindLazarusResourceHeaderComment(
 | |
|   ResourceCode: TCodeBuffer): TAtomPosition;
 | |
| begin
 | |
|   Result.StartPos:=-1;
 | |
|   Result.EndPos:=-1;
 | |
|   Result.Flag:=cafNone;
 | |
|   SetSource(ResourceCode);
 | |
| 
 | |
|   Result.StartPos:=FindNextNonSpace(Src,1);
 | |
|   if (Result.StartPos<=SrcLen) and (Src[Result.StartPos]='{') then
 | |
|     Result.EndPos:=FindCommentEnd(Src,Result.StartPos,false)
 | |
|   else
 | |
|     Result.StartPos:=-1;
 | |
| end;
 | |
| 
 | |
| function TResourceCodeTool.AddLazarusResourceHeaderComment(
 | |
|   ResourceCode: TCodeBuffer; const Comment: string): boolean;
 | |
| var
 | |
|   InsertPos: TAtomPosition;
 | |
| begin
 | |
|   Result:=true;
 | |
|   
 | |
|   // find existing one
 | |
|   InsertPos:=FindLazarusResourceHeaderComment(ResourceCode);
 | |
|   if InsertPos.StartPos>0 then begin
 | |
|     // there is already a comment
 | |
|     // -> don't touch it
 | |
|   end else
 | |
|     ResourceCode.Insert(1,Comment);
 | |
| end;
 | |
| 
 | |
| function TResourceCodeTool.FindLazarusResource(
 | |
|   ResourceCode: TCodeBuffer; const ResourceName: string;
 | |
|   StartPos: integer): TAtomPosition;
 | |
| var
 | |
|   ResourceNameInPascal: string;
 | |
|   ResStartPos: integer;
 | |
| begin
 | |
|   Result.StartPos:=-1;
 | |
|   Result.EndPos:=-1;
 | |
|   SetSource(ResourceCode);
 | |
|   if StartPos>=1 then
 | |
|     MoveCursorToCleanPos(StartPos);
 | |
| 
 | |
|   // search "LAZARUSRESOURCES.ADD('ResourceName',"
 | |
|   ResourceNameInPascal:=''''+UpperCaseStr(ResourceName)+'''';
 | |
|   repeat
 | |
|     ReadNextAtom;
 | |
|     if UpAtomIs('LAZARUSRESOURCES') then begin
 | |
|       ResStartPos:=CurPos.StartPos;
 | |
|       ReadNextAtom;
 | |
|       if CurPos.Flag<>cafPoint then continue;
 | |
|       ReadNextAtom;
 | |
|       if not UpAtomIs('ADD') then continue;
 | |
|       ReadNextAtom;
 | |
|       if CurPos.Flag<>cafRoundBracketOpen then continue;
 | |
|       ReadNextAtom;
 | |
|       if UpAtomIs(ResourceNameInPascal) then begin
 | |
|         // resource start found
 | |
|         Result.StartPos:=ResStartPos;
 | |
|       end;
 | |
|       UndoReadNextAtom;
 | |
|       ReadTilBracketClose(false);
 | |
|       if CurPos.Flag<>cafRoundBracketClose then begin
 | |
|         // syntax error
 | |
|         Result.StartPos:=-1;
 | |
|         exit;
 | |
|       end;
 | |
|       if (Result.StartPos>0) then begin
 | |
|         // resource end found
 | |
|         Result.EndPos:=CurPos.EndPos;
 | |
|         ReadNextAtom;
 | |
|         if CurPos.Flag=cafSemicolon then
 | |
|           Result.EndPos:=CurPos.EndPos;
 | |
|         exit;
 | |
|       end;
 | |
|     end;
 | |
|   until CurPos.StartPos>SrcLen;
 | |
| end;
 | |
| 
 | |
| function TResourceCodeTool.FindAllLazarusResource(ResourceCode: TCodeBuffer;
 | |
|   const ResourceName: string; StartPos: integer): TAtomList;
 | |
| var
 | |
|   ResourcePos: TAtomPosition;
 | |
| begin
 | |
|   Result:=TAtomList.Create;
 | |
|   repeat
 | |
|     ResourcePos:=FindLazarusResource(ResourceCode,ResourceName,StartPos);
 | |
|     if ResourcePos.StartPos<1 then break;
 | |
|     Result.Add(ResourcePos);
 | |
|     StartPos:=ResourcePos.EndPos;
 | |
|   until false;
 | |
| end;
 | |
| 
 | |
| function TResourceCodeTool.AddLazarusResource(ResourceCode: TCodeBuffer;
 | |
|   const ResourceName, ResourceData: string): boolean;
 | |
| var
 | |
|   InsertAtom: TAtomPosition;
 | |
|   NeededLineEnds, i: integer;
 | |
|   NewResData: string;
 | |
| begin
 | |
|   Result:=false;
 | |
|   // try to find an old resource and delete all doubles
 | |
|   Result:=RemoveLazarusResourceEx(ResourceCode,ResourceName,true,InsertAtom);
 | |
|   if InsertAtom.StartPos<1 then begin
 | |
|     // not found -> add at end of file
 | |
|     InsertAtom.StartPos:=ResourceCode.SourceLength+1;
 | |
|     InsertAtom.EndPos:=ResourceCode.SourceLength+1;
 | |
|   end else begin
 | |
|     InsertAtom.StartPos:=BasicCodeTools.FindLineEndOrCodeInFrontOfPosition(Src,
 | |
|                                      InsertAtom.StartPos,1,false,true);
 | |
|     InsertAtom.EndPos:=BasicCodeTools.FindLineEndOrCodeAfterPosition(Src,
 | |
|                                      InsertAtom.EndPos,SrcLen,false);
 | |
|   end;
 | |
|   if CodeIsOnlySpace(Src,1,InsertAtom.StartPos-1) then
 | |
|     InsertAtom.StartPos:=1;
 | |
|   if CodeIsOnlySpace(Src,InsertAtom.EndPos+1,SrcLen) then
 | |
|     InsertAtom.EndPos:=SrcLen+1;
 | |
| 
 | |
|   NewResData:=ResourceData;
 | |
|   i:=length(NewResData);
 | |
|   while (i>1) and (NewResData[i] in [' ',#10,#13]) do
 | |
|     dec(i);
 | |
|   SetLength(NewResData,i);
 | |
|   // add front gap
 | |
|   NeededLineEnds:=CountNeededLineEndsToAddForward(ResourceData,1,2);
 | |
|   NeededLineEnds:=CountNeededLineEndsToAddBackward(Src,InsertAtom.StartPos-1,
 | |
|                                                    NeededLineEnds);
 | |
|   for i:=1 to NeededLineEnds do
 | |
|     NewResData:=LineEnding+NewResData;
 | |
|   // add start gap
 | |
|   NeededLineEnds:=CountNeededLineEndsToAddBackward(ResourceData,
 | |
|                                                    length(ResourceData),2);
 | |
|   NeededLineEnds:=CountNeededLineEndsToAddForward(Src,InsertAtom.EndPos,
 | |
|                                                   NeededLineEnds);
 | |
|   for i:=1 to NeededLineEnds do
 | |
|     NewResData:=NewResData+LineEnding;
 | |
|   // replace
 | |
|   ResourceCode.Replace(InsertAtom.StartPos,
 | |
|                        InsertAtom.EndPos-InsertAtom.StartPos,
 | |
|                        NewResData);
 | |
|   Result:=true;
 | |
| end;
 | |
| 
 | |
| function TResourceCodeTool.RemoveLazarusResource(ResourceCode: TCodeBuffer;
 | |
|   const ResourceName: string): boolean;
 | |
| var
 | |
|   FirstResPos: TAtomPosition;
 | |
| begin
 | |
|   Result:=RemoveLazarusResourceEx(ResourceCode,ResourceName,false,FirstResPos);
 | |
| end;
 | |
| 
 | |
| function TResourceCodeTool.RemoveLazarusResourceEx(ResourceCode: TCodeBuffer;
 | |
|   const ResourceName: string; AllExceptFirst: boolean; out First: TAtomPosition
 | |
|   ): boolean;
 | |
| var
 | |
|   ResourcePositions: TAtomList;
 | |
|   CurResPos: TAtomPosition;
 | |
|   i, FirstIndex: integer;
 | |
| begin
 | |
|   Result:=true;
 | |
|   ResourcePositions:=FindAllLazarusResource(ResourceCode,ResourceName,-1);
 | |
|   try
 | |
|     if AllExceptFirst then
 | |
|       FirstIndex:=1
 | |
|     else
 | |
|       FirstIndex:=0;
 | |
|     for i:=ResourcePositions.Count-1 downto FirstIndex do begin
 | |
|       CurResPos:=ResourcePositions[i];
 | |
|       CurResPos.EndPos:=BasicCodeTools.FindLineEndOrCodeAfterPosition(Src,
 | |
|                              CurResPos.EndPos,SrcLen,false);
 | |
|       ResourceCode.Delete(CurResPos.StartPos,
 | |
|                           CurResPos.EndPos-CurResPos.StartPos);
 | |
|     end;
 | |
|     if ResourcePositions.Count>0 then begin
 | |
|       First:=ResourcePositions[0];
 | |
|     end else begin
 | |
|       First.StartPos:=-1;
 | |
|       First.EndPos:=-1;
 | |
|     end;
 | |
|   finally
 | |
|     ResourcePositions.Free;
 | |
|   end;
 | |
| end;
 | |
| 
 | |
| end.
 | |
| 
 | 
