unit ReplaceFuncsUnit; {$mode objfpc}{$H+} interface uses Classes, SysUtils, RegExpr, // LCL Forms, Controls, Graphics, Dialogs, ExtCtrls, Buttons, ButtonPanel, Grids, CheckLst, Menus, StdCtrls, // LazUtils FileUtil, // IdePackager IdePackagerStrConsts, // IDE, converter IdeIntfStrConsts, LazarusIDEStrConsts, ConverterTypes; type { TReplacementParam } TReplacementParam = class private fParamNum: Integer; // Sequence number of the parameter. fParamLen: Integer; // Parameter's length in the replacement string ($+num). fStrPosition: Integer; // Character index in replacement string. public constructor Create(aParamNum, aParamLen, aStrPosition: Integer); destructor Destroy; override; property ParamNum: Integer read fParamNum; property ParamLen: Integer read fParamLen; property StrPosition: Integer read fStrPosition; end; { TFuncReplacement } TFuncReplacement = class private // Defined in UI: fCategory: string; fFuncName: string; fReplClause: string; fPackageName: string; fUnitName: string; // Calculated for each actual replacement: fReplFunc: string; // May be extracted from a conditional expression. fStartPos: Integer; // Start and end positions of original func+params. fEndPos: Integer; fInclEmptyBrackets: string; // '()' is included in the replacement. fInclSemiColon: string; // Ending semiColon is included in the replacement. fParams: TStringList; // Parameters of the original function call. function ParseIf(var aStart: integer): boolean; public constructor Create(const aCategory, aFuncName, aReplacement, aPackageName, aUnitName: string); constructor Create(aFuncRepl: TFuncReplacement); destructor Destroy; override; procedure UpdateReplacement; public property Category: string read fCategory; property FuncName: string read fFuncName; property ReplClause: string read fReplClause; property ReplFunc: string read fReplFunc; // The actual replacement. property PackageName: string read fPackageName; property UnitName: string read fUnitName; property StartPos: Integer read fStartPos write fStartPos; property EndPos: Integer read fEndPos write fEndPos; property InclEmptyBrackets: string read fInclEmptyBrackets write fInclEmptyBrackets; property InclSemiColon: string read fInclSemiColon write fInclSemiColon; property Params: TStringList read fParams; end; { TFuncsAndCategories } TFuncsAndCategories = class private // Delphi func names, objects property has TFuncReplacement items. fFuncs: TStringList; fMinFuncLen: Integer; // Category names, objects property has boolean info Used/Not used. fCategories: TStringList; public constructor Create; destructor Destroy; override; procedure Clear; function AddFunc(aCategory, aDelphiFunc, aReplaceFunc, aPackage, aUnitName: string): integer; function FuncAtInd(Ind: integer): TFuncReplacement; function MinFuncLen: Integer; function AddCategory(aCategory: string; aUsed: Boolean): integer; function CategoryIsUsed(Ind: integer): Boolean; public property Funcs: TStringList read fFuncs; property Categories: TStringList read fCategories; end; { TReplaceFuncsForm } TReplaceFuncsForm = class(TForm) ButtonPanel: TButtonPanel; CategoryListBox: TCheckListBox; DeleteRow1: TMenuItem; Grid: TStringGrid; InsertRow1: TMenuItem; CategoriesLabel: TLabel; PopupMenu1: TPopupMenu; Splitter1: TSplitter; procedure CategoryListBoxClickCheck(Sender: TObject); procedure FormCreate(Sender: TObject); procedure PopupMenu1Popup(Sender: TObject); procedure InsertRow1Click(Sender: TObject); procedure DeleteRow1Click(Sender: TObject); procedure GridPrepareCanvas(sender: TObject; {%H-}aCol, aRow: Integer; {%H-}aState: TGridDrawState); procedure OKButtonClick(Sender: TObject); public function FromFuncListToUI(aFuncsAndCateg: TFuncsAndCategories): boolean; function FromUIToFuncList(aFuncsAndCateg: TFuncsAndCategories): boolean; end; function EditFuncReplacements(aFuncsAndCateg: TFuncsAndCategories; aTitle: string): TModalResult; implementation {$R *.lfm} function EditFuncReplacements(aFuncsAndCateg: TFuncsAndCategories; aTitle: string): TModalResult; var RFForm: TReplaceFuncsForm; begin RFForm:=TReplaceFuncsForm.Create(nil); try RFForm.Caption:=aTitle; RFForm.CategoriesLabel.Caption:=lisConvDelphiCategories; RFForm.Grid.Columns[0].Title.Caption:=lisCEOModeCategory; RFForm.Grid.Columns[1].Title.Caption:=lisConvDelphiFunc; RFForm.Grid.Columns[2].Title.Caption:=lisReplacement; RFForm.Grid.Columns[3].Title.Caption:=lisPackage; RFForm.Grid.Columns[4].Title.Caption:=lisUnit; RFForm.FromFuncListToUI(aFuncsAndCateg); Result:=RFForm.ShowModal; if Result=mrOK then RFForm.FromUIToFuncList(aFuncsAndCateg); finally RFForm.Free; end; end; { TReplacementParam } constructor TReplacementParam.Create(aParamNum, aParamLen, aStrPosition: Integer); begin inherited Create; fParamNum:=aParamNum; fParamLen:=aParamLen; fStrPosition:=aStrPosition; end; destructor TReplacementParam.Destroy; begin inherited Destroy; end; { TFuncReplacement } constructor TFuncReplacement.Create(const aCategory, aFuncName, aReplacement, aPackageName, aUnitName: string); begin inherited Create; fCategory:=aCategory; fFuncName:=aFuncName; fReplClause:=aReplacement; fPackageName:=aPackageName; fUnitName:=aUnitName; fParams:=TStringList.Create; end; constructor TFuncReplacement.Create(aFuncRepl: TFuncReplacement); // Copy constructor. begin Create(aFuncRepl.fCategory, aFuncRepl.fFuncName, aFuncRepl.fReplClause, aFuncRepl.fPackageName, aFuncRepl.fUnitName); end; destructor TFuncReplacement.Destroy; begin fParams.Free; inherited Destroy; end; function TFuncReplacement.ParseIf(var aStart: integer): boolean; // Parse a clause starting with "if" and set fReplFunc if the condition matches. // Example: 'if $3 match ":/" then OpenURL($3); OpenDocument($3)' // Return true if the condition matched. procedure ReadWhiteSpace(NewStartPos: integer); begin aStart:=NewStartPos; while (aStart<=Length(fReplClause)) and (fReplClause[aStart]=' ') do inc(aStart); end; function ParseParamNum: integer; var EndPos: Integer; s: String; begin if fReplClause[aStart]<>'$' then raise EDelphiConverterError.Create(Format('$ expected, %s found.', [fReplClause[aStart]])); Inc(aStart); // Skip $ EndPos:=aStart; while (EndPos<=Length(fReplClause)) and (fReplClause[EndPos] in ['0'..'9']) do Inc(EndPos); s:=Copy(fReplClause, aStart, EndPos-aStart); Result:=StrToInt(s); ReadWhiteSpace(EndPos); end; procedure ParseString(aStr: string); var EndPos: Integer; s: String; begin EndPos:=aStart; while (EndPos<=Length(fReplClause)) and (fReplClause[EndPos] in ['a'..'z','A'..'Z','_']) do Inc(EndPos); s:=Copy(fReplClause, aStart, EndPos-aStart); if s<>aStr then raise EDelphiConverterError.Create(Format('%s expected, %s found.', [aStr, s])); ReadWhiteSpace(EndPos); end; function ParseDoubleQuoted: string; var EndPos: Integer; begin if fReplClause[aStart]<>'"' then raise EDelphiConverterError.Create(Format('" expected, %s found.', [fReplClause[aStart]])); Inc(aStart); // Skip " EndPos:=aStart; while (EndPos<=Length(fReplClause)) and (fReplClause[EndPos]<>'"') do inc(EndPos); Result:=Copy(fReplClause, aStart, EndPos-aStart); ReadWhiteSpace(EndPos+1); end; function GetReplacement: string; var EndPos: Integer; begin EndPos:=aStart; while (EndPos<=Length(fReplClause)) and (fReplClause[EndPos]<>';') do inc(EndPos); Result:=Copy(fReplClause, aStart, EndPos-aStart); aStart:=EndPos+1; // Skip ';' end; var ParamPos: integer; RE: TRegExpr; Str, Param: String; Repl: String; begin // "if " is already skipped when coming here. ReadWhiteSpace(aStart); // Possible space in the beginning. ParamPos:=ParseParamNum; ParseString('match'); Str:=ParseDoubleQuoted; ParseString('then'); Repl:=GetReplacement; Result:=False; if ParamPos<=fParams.Count then begin Param:=fParams[ParamPos-1]; RE:=TRegExpr.Create; try RE.ModifierI:=True; RE.Expression:=Str; if RE.Exec(Param) then begin fReplFunc:=Repl; Result:=True; end; finally RE.Free; end; end; end; procedure TFuncReplacement.UpdateReplacement; // Parse fReplClause and set fReplFunc, maybe conditionally based on parameters. var xStart, xEnd: Integer; begin xStart:=1; while true do begin // xStart<=Length(fReplClause) // "If" condition can match or not. Continue if it didn't match. if Copy(fReplClause, xStart, 3) = 'if ' then begin Inc(xStart, 3); if ParseIf(xStart) then Break; end else begin // Replacement without conditions. Copy it and stop. xEnd:=xStart; while (xEnd<=Length(fReplClause)) and (fReplClause[xEnd]<>';') do inc(xEnd); fReplFunc:=Copy(fReplClause, xStart, xEnd-xStart); Break; end; end; end; { TFuncsAndCategories } constructor TFuncsAndCategories.Create; begin inherited Create; fFuncs:=TStringList.Create; fFuncs.Sorted:=True; fFuncs.CaseSensitive:=False; fCategories:=TStringList.Create; fCategories.Sorted:=True; fCategories.Duplicates:=dupIgnore; end; destructor TFuncsAndCategories.Destroy; begin fCategories.Free; fFuncs.Free; inherited Destroy; end; procedure TFuncsAndCategories.Clear; var i: Integer; begin for i := 0 to fFuncs.Count-1 do fFuncs.Objects[i].Free; fFuncs.Clear; fCategories.Clear; end; function TFuncsAndCategories.AddFunc( aCategory, aDelphiFunc, aReplaceFunc, aPackage, aUnitName: string): integer; // This is called when settings are read or when user made changes in GUI. // Returns index for the added func replacement, or -1 if not added (duplicate). var FuncRepl: TFuncReplacement; x: integer; begin Result:=-1; if (aDelphiFunc<>'') and not fFuncs.Find(aDelphiFunc, x) then begin FuncRepl:=TFuncReplacement.Create(aCategory, aDelphiFunc, aReplaceFunc, aPackage, aUnitName); Result:=fFuncs.AddObject(aDelphiFunc, FuncRepl); end; end; function TFuncsAndCategories.FuncAtInd(Ind: integer): TFuncReplacement; begin Result:=nil; if fFuncs[Ind]<>'' then Result:=TFuncReplacement(fFuncs.Objects[Ind]); end; function TFuncsAndCategories.AddCategory(aCategory: string; aUsed: Boolean): integer; var CategUsed: PtrInt; begin CategUsed:=0; if aUsed then CategUsed:=1; Result:=fCategories.AddObject(aCategory, TObject(CategUsed)); end; function TFuncsAndCategories.CategoryIsUsed(Ind: integer): Boolean; var CategUsed: PtrInt; begin CategUsed:=0; if fCategories[Ind]<>'' then CategUsed:=PtrInt(fCategories.Objects[Ind]); Result:=CategUsed=1; end; function TFuncsAndCategories.MinFuncLen: Integer; var i, Len: Integer; begin if (fMinFuncLen=0) and (fFuncs.Count>0) then begin fMinFuncLen:=10000; // First a big enough length for i:=0 to fFuncs.Count-1 do begin if fFuncs[i]='Ptr' then Continue; // 'Ptr' is short and is a special case. Len:=Length(fFuncs[i]); if Len-1) and not CategoryListBox.Checked[i] then SGrid.Canvas.Font.Color:= clGrayText; end; procedure TReplaceFuncsForm.OKButtonClick(Sender: TObject); begin ModalResult:=mrOK; end; function TReplaceFuncsForm.FromFuncListToUI(aFuncsAndCateg: TFuncsAndCategories): boolean; // Copy strings from Map to Grid. var FuncRepl: TFuncReplacement; i: Integer; begin Result:=true; Grid.BeginUpdate; // Skip the fixed row in grid. for i:=1 to aFuncsAndCateg.fFuncs.Count do begin if Grid.RowCount'' then // Delphi function name must have something. aFuncsAndCateg.AddFunc(Grid.Cells[0,i], Grid.Cells[1,i], Grid.Cells[2,i], Grid.Cells[3,i], Grid.Cells[4,i]); // Copy checked (used) categories. for i:=0 to CategoryListBox.Count-1 do aFuncsAndCateg.AddCategory(CategoryListBox.Items[i], CategoryListBox.Checked[i]); end; end.