lazarus/converter/replacefuncsunit.pas

513 lines
14 KiB
ObjectPascal

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<fMinFuncLen then
fMinFuncLen:=Len;
end;
end;
Result:=fMinFuncLen;
end;
{ TReplaceFuncsForm }
procedure TReplaceFuncsForm.FormCreate(Sender: TObject);
begin
Caption:=lisReplacementFuncs;
ButtonPanel.OKButton.Caption := lisBtnOk;
ButtonPanel.HelpButton.Caption := lisMenuHelp;
ButtonPanel.CancelButton.Caption := lisCancel;
end;
procedure TReplaceFuncsForm.CategoryListBoxClickCheck(Sender: TObject);
begin
Grid.Invalidate; // Update;
end;
procedure TReplaceFuncsForm.PopupMenu1Popup(Sender: TObject);
var
ControlCoord, NewCell: TPoint;
begin
ControlCoord:=Grid.ScreenToControl(PopupMenu1.PopupPoint);
NewCell:=Grid.MouseToCell(ControlCoord);
Grid.Col:=NewCell.X;
Grid.Row:=NewCell.Y;
end;
procedure TReplaceFuncsForm.InsertRow1Click(Sender: TObject);
begin
Grid.InsertColRow(False, Grid.Row);
end;
procedure TReplaceFuncsForm.DeleteRow1Click(Sender: TObject);
begin
Grid.DeleteColRow(False, Grid.Row);
end;
procedure TReplaceFuncsForm.GridPrepareCanvas(sender: TObject;
aCol, aRow: Integer; aState: TGridDrawState);
var
SGrid: TStringGrid;
Categ: string;
i: integer;
begin
SGrid:=Sender as TStringGrid;
Categ:=SGrid.Cells[0, aRow]; // Column 0 = category.
i:=CategoryListBox.Items.IndexOf(Categ);
if (i<>-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<i+1 then
Grid.RowCount:=i+1;
FuncRepl:=TFuncReplacement(aFuncsAndCateg.fFuncs.Objects[i-1]);
Grid.Cells[0,i]:=FuncRepl.fCategory;
Grid.Cells[1,i]:=aFuncsAndCateg.fFuncs[i-1]; // Delphi function name
Grid.Cells[2,i]:=FuncRepl.fReplClause;
Grid.Cells[3,i]:=FuncRepl.PackageName;
Grid.Cells[4,i]:=FuncRepl.fUnitName;
end;
for i:=0 to aFuncsAndCateg.fCategories.Count-1 do begin
CategoryListBox.Items.Add(aFuncsAndCateg.fCategories[i]);
CategoryListBox.Checked[i]:=aFuncsAndCateg.CategoryIsUsed(i);
end;
Grid.EndUpdate;
end;
function TReplaceFuncsForm.FromUIToFuncList(aFuncsAndCateg: TFuncsAndCategories): boolean;
var
i: Integer;
begin
Result:=true;
aFuncsAndCateg.Clear;
// Collect (maybe edited) properties from StringGrid to fStringMap.
for i:=1 to Grid.RowCount-1 do // Skip the fixed row.
if Grid.Cells[1,i]<>'' 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.