mirror of
https://gitlab.com/freepascal.org/lazarus/lazarus.git
synced 2025-05-14 19:12:54 +02:00
570 lines
18 KiB
ObjectPascal
570 lines
18 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., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. *
|
|
* *
|
|
***************************************************************************
|
|
|
|
Author: Mattias Gaertner
|
|
|
|
Abstract:
|
|
Shows all overloads of the procedure/method at sourcee editor position.
|
|
Sortable columns:
|
|
Path: unitnames.classes
|
|
(If not filtered:) Compatibility: exact, compatible, incompatible
|
|
Distance
|
|
Last visited
|
|
Filter:
|
|
Params must be compatible: Yes, No
|
|
(Only for method:) Only descendants, have same ancestor method, all
|
|
(Only for method:) Show abstract methods and interfaces
|
|
}
|
|
unit CodyFindOverloads;
|
|
|
|
{$mode objfpc}{$H+}
|
|
|
|
interface
|
|
|
|
uses
|
|
Classes, SysUtils, AVL_Tree, FileUtil, LazLoggerBase, CodyUtils,
|
|
CodeToolManager, CodeTree, CodeCache, FindDeclarationTool, PascalParserTool,
|
|
BasicCodeTools, CTUnitGraph, FileProcs, StdCodeTools, CodeGraph, LazIDEIntf,
|
|
IDEWindowIntf, ProjectIntf, Forms, Controls, Graphics, Dialogs, ExtCtrls,
|
|
StdCtrls, Grids, ComCtrls;
|
|
|
|
type
|
|
TCFOUnit = class(TUGUnit)
|
|
public
|
|
end;
|
|
|
|
TCFONode = class(TCodeGraphNode)
|
|
public
|
|
Tool: TFindDeclarationTool;
|
|
end;
|
|
|
|
TCFOEdgeType = (
|
|
cfoetReachable, // FromNode (proc) is reachable by ToNode(program)
|
|
cfoetMethodOf, // FromNode (proc) is method of ToNode (class)
|
|
cfoetDescendantOf // FromNode (class) is descendant of ToNode (class)
|
|
);
|
|
|
|
TCFOEdge = class(TCodeGraphEdge)
|
|
public
|
|
Typ: TCFOEdgeType;
|
|
end;
|
|
|
|
TCFOFlag = (
|
|
cfofParsing,
|
|
cfofGatherProcs
|
|
);
|
|
TCFOFlags = set of TCFOFlag;
|
|
|
|
{ TCodyFindOverloadsWindow }
|
|
|
|
TCodyFindOverloadsWindow = class(TForm)
|
|
BtnPanel: TPanel;
|
|
CompatibleParamsCheckBox: TCheckBox;
|
|
FilterGroupBox: TGroupBox;
|
|
HideAbstractCheckBox: TCheckBox;
|
|
JumpToButton: TButton;
|
|
ProgressBar1: TProgressBar;
|
|
RefreshButton: TButton;
|
|
RelationComboBox: TComboBox;
|
|
RelationLabel: TLabel;
|
|
ResultsGroupBox: TGroupBox;
|
|
ResultsStringGrid: TStringGrid;
|
|
Timer1: TTimer;
|
|
procedure CloseButtonClick(Sender: TObject);
|
|
procedure FormClose(Sender: TObject; var {%H-}CloseAction: TCloseAction);
|
|
procedure FormCreate(Sender: TObject);
|
|
procedure JumpToButtonClick(Sender: TObject);
|
|
procedure OnIdle(Sender: TObject; var Done: Boolean);
|
|
procedure RefreshButtonClick(Sender: TObject);
|
|
procedure Timer1Timer(Sender: TObject);
|
|
private
|
|
FIdleConnected: boolean;
|
|
FFlags: TCFOFlags;
|
|
FTargetName: string;
|
|
FTargetPath: string;
|
|
FTargetXYPosition: TCodeXYPosition;
|
|
FUsesGraph: TUsesGraph;
|
|
procedure SetIdleConnected(AValue: boolean);
|
|
procedure CreateUsesGraph(out TheUsesGraph: TUsesGraph);
|
|
procedure StartParsing;
|
|
procedure AbortParsing;
|
|
procedure AddStartAndTargetUnits;
|
|
procedure GatherProcsOfAllUnits;
|
|
procedure GatherProcsOfUnit(NodeGraph: TCodeGraph; ProgNode: TCodeTreeNode;
|
|
CurUnit: TCFOUnit);
|
|
procedure FreeUsesGraph;
|
|
function GetDefaultCaption: string;
|
|
procedure FillFilterControls(ProcTool: TFindDeclarationTool;
|
|
ProcNode: TCodeTreeNode);
|
|
protected
|
|
procedure UpdateShowing; override;
|
|
public
|
|
procedure JumpToIdentifier;
|
|
function Init: boolean;
|
|
property IdleConnected: boolean read FIdleConnected write SetIdleConnected;
|
|
property UsesGraph: TUsesGraph read FUsesGraph;
|
|
property TargetXYPosition: TCodeXYPosition read FTargetXYPosition;
|
|
property TargetName: string read FTargetName;
|
|
property TargetPath: string read FTargetPath;
|
|
end;
|
|
|
|
var
|
|
CodyFindOverloadsWindow: TCodyFindOverloadsWindow;
|
|
|
|
procedure ShowFindOverloadsClicked(Sender: TObject);
|
|
procedure ShowFindOverloads(State: TIWGetFormState = iwgfShowOnTop);
|
|
|
|
implementation
|
|
|
|
procedure ShowFindOverloadsClicked(Sender: TObject);
|
|
begin
|
|
ShowFindOverloads;
|
|
end;
|
|
|
|
procedure ShowFindOverloads(State: TIWGetFormState);
|
|
begin
|
|
if CodyFindOverloadsWindow = nil then
|
|
IDEWindowCreators.CreateForm(CodyFindOverloadsWindow,TCodyFindOverloadsWindow,
|
|
State=iwgfDisabled,LazarusIDE.OwningComponent)
|
|
else if State=iwgfDisabled then
|
|
CodyFindOverloadsWindow.DisableAlign;
|
|
if State>=iwgfShow then
|
|
IDEWindowCreators.ShowForm(CodyFindOverloadsWindow,State=iwgfShowOnTop);
|
|
end;
|
|
|
|
{$R *.lfm}
|
|
|
|
{ TCodyFindOverloadsWindow }
|
|
|
|
procedure TCodyFindOverloadsWindow.FormCreate(Sender: TObject);
|
|
begin
|
|
Caption:=GetDefaultCaption;
|
|
RefreshButton.Caption:='Refresh';
|
|
JumpToButton.Caption:='Jump to';
|
|
|
|
FilterGroupBox.Caption:='Filter';
|
|
CompatibleParamsCheckBox.Caption:='Only procedures with compatible parameters';
|
|
CompatibleParamsCheckBox.Hint:='If unchecked list also procedures with same name, but incompatible parameter lists.';
|
|
HideAbstractCheckBox.Caption:='Hide abstract methods and methods of class interfaces';
|
|
RelationLabel.Caption:='Relations:';
|
|
end;
|
|
|
|
procedure TCodyFindOverloadsWindow.CloseButtonClick(Sender: TObject);
|
|
begin
|
|
|
|
end;
|
|
|
|
procedure TCodyFindOverloadsWindow.FormClose(Sender: TObject;
|
|
var CloseAction: TCloseAction);
|
|
begin
|
|
AbortParsing;
|
|
FreeUsesGraph;
|
|
end;
|
|
|
|
procedure TCodyFindOverloadsWindow.JumpToButtonClick(Sender: TObject);
|
|
begin
|
|
|
|
end;
|
|
|
|
procedure TCodyFindOverloadsWindow.OnIdle(Sender: TObject; var Done: Boolean);
|
|
var
|
|
Completed: boolean;
|
|
begin
|
|
if cfofParsing in FFlags then begin
|
|
fUsesGraph.Parse(true,Completed,200);
|
|
if Completed then begin
|
|
FFlags:=FFlags-[cfofParsing]+[cfofGatherProcs];
|
|
end;
|
|
end else if cfofGatherProcs in FFlags then begin
|
|
GatherProcsOfAllUnits;
|
|
FFlags:=FFlags-[cfofGatherProcs];
|
|
// hide progress bar and update stats
|
|
ProgressBar1.Visible:=false;
|
|
RefreshButton.Enabled:=true;
|
|
Timer1.Enabled:=false;
|
|
ResultsGroupBox.Caption:=Format('Units: %s', [IntToStr(FUsesGraph.FilesTree.Count)]);
|
|
// update controls
|
|
//UpdateAll;
|
|
end else
|
|
IdleConnected:=false;
|
|
Done:=not IdleConnected;
|
|
end;
|
|
|
|
procedure TCodyFindOverloadsWindow.RefreshButtonClick(Sender: TObject);
|
|
begin
|
|
if cfofParsing in FFlags then exit;
|
|
Init;
|
|
end;
|
|
|
|
procedure TCodyFindOverloadsWindow.Timer1Timer(Sender: TObject);
|
|
var
|
|
Cnt: Integer;
|
|
begin
|
|
if (FUsesGraph=nil) then exit;
|
|
Cnt:=0;
|
|
if FUsesGraph.FilesTree<>nil then
|
|
Cnt:=FUsesGraph.FilesTree.Count;
|
|
ResultsGroupBox.Caption:=Format('Scanning: %s units ...', [IntToStr(Cnt)]);
|
|
end;
|
|
|
|
procedure TCodyFindOverloadsWindow.SetIdleConnected(AValue: boolean);
|
|
begin
|
|
if FIdleConnected=AValue then Exit;
|
|
FIdleConnected:=AValue;
|
|
if Application=nil then exit;
|
|
if IdleConnected then
|
|
Application.AddOnIdleHandler(@OnIdle)
|
|
else
|
|
Application.RemoveOnIdleHandler(@OnIdle);
|
|
end;
|
|
|
|
procedure TCodyFindOverloadsWindow.CreateUsesGraph(out TheUsesGraph: TUsesGraph
|
|
);
|
|
begin
|
|
TheUsesGraph:=CodeToolBoss.CreateUsesGraph;
|
|
if not TCFOUnit.InheritsFrom(TheUsesGraph.UnitClass) then
|
|
RaiseCatchableException('');
|
|
TheUsesGraph.UnitClass:=TCFOUnit;
|
|
end;
|
|
|
|
procedure TCodyFindOverloadsWindow.StartParsing;
|
|
begin
|
|
if (FUsesGraph<>nil) or (cfofParsing in FFlags) then
|
|
RaiseCatchableException('');
|
|
Include(FFlags,cfofParsing);
|
|
|
|
ProgressBar1.Visible:=true;
|
|
ProgressBar1.Style:=pbstMarquee;
|
|
ResultsGroupBox.Caption:='Scanning ...';
|
|
Timer1.Enabled:=true;
|
|
RefreshButton.Enabled:=false;
|
|
|
|
CreateUsesGraph(FUsesGraph);
|
|
|
|
LazarusIDE.BeginCodeTools;
|
|
AddStartAndTargetUnits;
|
|
|
|
IdleConnected:=true;
|
|
end;
|
|
|
|
procedure TCodyFindOverloadsWindow.AbortParsing;
|
|
begin
|
|
FFlags:=[];
|
|
IdleConnected:=false;
|
|
ProgressBar1.Visible:=false;
|
|
RefreshButton.Enabled:=true;
|
|
end;
|
|
|
|
procedure TCodyFindOverloadsWindow.AddStartAndTargetUnits;
|
|
var
|
|
aProject: TLazProject;
|
|
begin
|
|
FUsesGraph.TargetAll:=true;
|
|
// project lpr
|
|
aProject:=LazarusIDE.ActiveProject;
|
|
if (aProject<>nil) and (aProject.MainFile<>nil) then
|
|
FUsesGraph.AddStartUnit(aProject.MainFile.Filename);
|
|
end;
|
|
|
|
procedure TCodyFindOverloadsWindow.GatherProcsOfAllUnits;
|
|
var
|
|
FileNode: TAVLTreeNode;
|
|
CurUnit: TCFOUnit;
|
|
NodeGraph: TCodeGraph;
|
|
ProgNode: TCodeTreeNode;
|
|
begin
|
|
if FUsesGraph=nil then begin
|
|
Exclude(FFlags,cfofGatherProcs);
|
|
exit;
|
|
end;
|
|
debugln(['TCodyFindOverloadsWindow.GatherProcsOfAllUnits START']);
|
|
ProgNode:=TCodeTreeNode.Create;
|
|
NodeGraph:=TCodeGraph.Create(TCFONode,TCFOEdge);
|
|
try
|
|
NodeGraph.AddGraphNode(ProgNode);
|
|
|
|
FileNode:=FUsesGraph.FilesTree.FindLowest;
|
|
while FileNode<>nil do begin
|
|
CurUnit:=TCFOUnit(FileNode.Data);
|
|
GatherProcsOfUnit(NodeGraph,ProgNode,CurUnit);
|
|
FileNode:=FUsesGraph.FilesTree.FindSuccessor(FileNode);
|
|
end;
|
|
finally
|
|
NodeGraph.Free;
|
|
ProgNode.Free;
|
|
end;
|
|
debugln(['TCodyFindOverloadsWindow.GatherProcsOfAllUnits END']);
|
|
end;
|
|
|
|
procedure TCodyFindOverloadsWindow.GatherProcsOfUnit(NodeGraph: TCodeGraph;
|
|
ProgNode: TCodeTreeNode; CurUnit: TCFOUnit);
|
|
|
|
procedure AddAncestors(Tool: TFindDeclarationTool; ClassNode: TCodeTreeNode);
|
|
begin
|
|
|
|
end;
|
|
|
|
var
|
|
Tool: TStandardCodeTool;
|
|
ProcNode, ClassNode: TCodeTreeNode;
|
|
CurProcName: String;
|
|
GraphProcNode, GraphClassNode: TCFONode;
|
|
Edge: TCFOEdge;
|
|
begin
|
|
if ugufLoadError in CurUnit.Flags then exit;
|
|
if not (ugufReached in CurUnit.Flags) then exit; // this unit was not reached
|
|
if ugufIsIncludeFile in CurUnit.Flags then exit;
|
|
Tool:=CurUnit.Tool;
|
|
ProcNode:=Tool.Tree.Root;
|
|
while ProcNode<>nil do begin
|
|
if ProcNode.Desc in [ctnImplementation,ctnBeginBlock] then break;
|
|
if ProcNode.Desc=ctnProcedure then begin
|
|
CurProcName:=Tool.ExtractProcName(ProcNode,[phpWithoutClassName]);
|
|
if CompareIdentifiers(PChar(CurProcName),PChar(FTargetName))=0 then begin
|
|
debugln(['TCodyFindOverloadsWindow.GatherProcsOfUnit ',Tool.CleanPosToStr(ProcNode.StartPos,true)]);
|
|
GraphProcNode:=TCFONode(NodeGraph.AddGraphNode(ProcNode));
|
|
GraphProcNode.Tool:=Tool;
|
|
ClassNode:=ProcNode.Parent;
|
|
while ClassNode<>nil do begin
|
|
if ClassNode.Desc in AllClasses then break;
|
|
ClassNode:=ClassNode.Parent;
|
|
end;
|
|
if ClassNode<>nil then begin
|
|
GraphClassNode:=TCFONode(NodeGraph.AddGraphNode(ClassNode));
|
|
GraphClassNode.Tool:=Tool;
|
|
// create edge "is method of"
|
|
Edge:=TCFOEdge(NodeGraph.AddEdge(ProcNode,ClassNode));
|
|
Edge.Typ:=cfoetMethodOf;
|
|
// create edge "reachable", so that all nodes are reachable
|
|
Edge:=TCFOEdge(NodeGraph.AddEdge(ClassNode,ProgNode));
|
|
Edge.Typ:=cfoetReachable;
|
|
AddAncestors(Tool,ClassNode);
|
|
end else begin
|
|
// not a method
|
|
// create edge "reachable", so that all nodes are reachable
|
|
Edge:=TCFOEdge(NodeGraph.AddEdge(ProcNode,ProgNode));
|
|
Edge.Typ:=cfoetReachable;
|
|
end;
|
|
end;
|
|
end;
|
|
ProcNode:=ProcNode.Next;
|
|
end;
|
|
end;
|
|
|
|
procedure TCodyFindOverloadsWindow.FreeUsesGraph;
|
|
begin
|
|
FreeAndNil(FUsesGraph);
|
|
end;
|
|
|
|
function TCodyFindOverloadsWindow.GetDefaultCaption: string;
|
|
begin
|
|
Result:='Cody - Find Overloads';
|
|
end;
|
|
|
|
procedure TCodyFindOverloadsWindow.FillFilterControls(
|
|
ProcTool: TFindDeclarationTool; ProcNode: TCodeTreeNode);
|
|
var
|
|
sl: TStringList;
|
|
ClassNode: TCodeTreeNode;
|
|
ListOfPFindContext: TFPList;
|
|
Params: TFindDeclarationParams;
|
|
i: Integer;
|
|
aContext: PFindContext;
|
|
begin
|
|
// RelationComboBox
|
|
sl:=TStringList.Create;
|
|
try
|
|
ClassNode:=ProcNode;
|
|
while (ClassNode<>nil) and (not (ClassNode.Desc in AllClasses)) do
|
|
ClassNode:=ClassNode.Parent;
|
|
if ClassNode<>nil then begin
|
|
// method
|
|
sl.Add('Only descendants of '+ProcTool.ExtractClassName(ClassNode,false));
|
|
ListOfPFindContext:=nil;
|
|
Params:=TFindDeclarationParams.Create(nil);
|
|
try
|
|
ProcTool.FindAncestorsOfClass(ClassNode,ListOfPFindContext,Params,
|
|
false,false);
|
|
if ListOfPFindContext<>nil then begin
|
|
for i:=0 to ListOfPFindContext.Count-1 do begin
|
|
aContext:=PFindContext(ListOfPFindContext[i]);
|
|
sl.Add('Only descendants of '+aContext^.Tool.ExtractClassName(aContext^.Node,false));
|
|
end;
|
|
end;
|
|
finally
|
|
Params.Free;
|
|
FreeListOfPFindContext(ListOfPFindContext);
|
|
end;
|
|
sl.Add('Only methods');
|
|
end else begin
|
|
// procedure, non method
|
|
sl.Add('Only non methods');
|
|
end;
|
|
sl.Add('Any');
|
|
RelationComboBox.Items:=sl;
|
|
if sl.IndexOf(RelationComboBox.Text)<0 then
|
|
RelationComboBox.Text:='Any';
|
|
finally
|
|
sl.Free;
|
|
end;
|
|
end;
|
|
|
|
procedure TCodyFindOverloadsWindow.UpdateShowing;
|
|
begin
|
|
inherited UpdateShowing;
|
|
if IsVisible and (FUsesGraph=nil) then begin
|
|
Init;
|
|
end;
|
|
end;
|
|
|
|
procedure TCodyFindOverloadsWindow.JumpToIdentifier;
|
|
begin
|
|
|
|
end;
|
|
|
|
function TCodyFindOverloadsWindow.Init: boolean;
|
|
var
|
|
CurTool: TCodeTool;
|
|
CurNode: TCodeTreeNode;
|
|
CurCodePos: TCodeXYPosition;
|
|
CurCleanPos: integer;
|
|
|
|
procedure FindProcDeclaration(ProcTool: TFindDeclarationTool;
|
|
var ProcNode: TCodeTreeNode);
|
|
// find the method declaration of a method body
|
|
// find the forward or interface declaration of a proc body
|
|
var
|
|
Node: TCodeTreeNode;
|
|
begin
|
|
if ProcNode=nil then exit;
|
|
if ProcNode.Desc=ctnProcedureHead then
|
|
ProcNode:=ProcNode.Parent;
|
|
if ProcNode.Desc<>ctnProcedure then exit;
|
|
if ProcNode.Parent.Desc=ctnInterface then exit;
|
|
if (ProcNode.Parent.Desc=ctnImplementation)
|
|
or ProcTool.NodeIsMethodBody(ProcNode)
|
|
or (not ProcTool.NodeIsForwardProc(ProcNode)) then begin
|
|
Node:=ProcTool.FindCorrespondingProcNode(ProcNode,[phpWithoutClassName]);
|
|
if Node=nil then exit;
|
|
ProcNode:=Node;
|
|
end;
|
|
end;
|
|
|
|
function CheckCursorAtProcCall(StatementNode: TCodeTreeNode;
|
|
out ProcTool: TFindDeclarationTool; out ProcNode: TCodeTreeNode): boolean;
|
|
var
|
|
CurIdentStart, CurIdentEnd, NewTopLine: integer;
|
|
NewTool: TFindDeclarationTool;
|
|
NewNode: TCodeTreeNode;
|
|
NewPos: TCodeXYPosition;
|
|
begin
|
|
Result:=true;
|
|
ProcTool:=nil;
|
|
ProcNode:=nil;
|
|
if StatementNode=nil then exit;
|
|
// cursor in statement => check if on a proc call.
|
|
if (CurCodePos.Code=nil) then exit;
|
|
GetIdentStartEndAtPosition(CurTool.Src,CurCleanPos,CurIdentStart,CurIdentEnd);
|
|
if CurIdentStart>=CurIdentEnd then exit;
|
|
DebugLn(['TCodyFindOverloadsDialog.Init.IsCursorAtProcCall checking identifier "',copy(CurTool.Src,CurIdentStart,CurIdentEnd-CurIdentStart),'"']);
|
|
if not CurTool.FindDeclaration(CurCodePos,DefaultFindSmartFlags,
|
|
NewTool,NewNode,NewPos,NewTopLine)
|
|
then begin
|
|
ResultsGroupBox.Caption:='Parse error';
|
|
LazarusIDE.DoJumpToCodeToolBossError;
|
|
exit(false);
|
|
end;
|
|
if NewNode.Desc in [ctnProcedure,ctnProcedureHead] then begin
|
|
ProcTool:=NewTool;
|
|
ProcNode:=NewNode;
|
|
FindProcDeclaration(ProcTool,ProcNode);
|
|
debugln(['TCodyFindOverloadsDialog.Init.CheckCursorAtProcCall TargetProc ',ProcTool.CleanPosToStr(ProcNode.StartPos,true),' Class=',ProcTool.ExtractProcName(ProcNode,[phpAddClassName])]);
|
|
end;
|
|
end;
|
|
|
|
var
|
|
ProcNode, Node, BeginNode, TargetProcNode: TCodeTreeNode;
|
|
ErrorHandled: boolean;
|
|
CurInitError: TCUParseError;
|
|
TargetTool: TFindDeclarationTool;
|
|
begin
|
|
Result:=false;
|
|
|
|
AbortParsing;
|
|
ResultsStringGrid.Visible:=false;
|
|
FTargetName:='';
|
|
FTargetPath:='';
|
|
FTargetXYPosition:=CleanCodeXYPosition;
|
|
FreeUsesGraph;
|
|
|
|
// parse source
|
|
CurInitError:=ParseTilCursor(CurTool, CurCleanPos, CurNode, ErrorHandled, true, @CurCodePos);
|
|
if CurInitError<>cupeSuccess then begin
|
|
ResultsGroupBox.Caption:='Parse error';
|
|
exit;
|
|
end;
|
|
|
|
// find target proc node
|
|
ProcNode:=nil;
|
|
BeginNode:=nil;
|
|
Node:=CurNode;
|
|
TargetTool:=nil;
|
|
TargetProcNode:=nil;
|
|
while Node<>nil do begin
|
|
if Node.Desc=ctnProcedure then begin
|
|
ProcNode:=Node;
|
|
break;
|
|
end else if (BeginNode=nil) and (Node.Desc=ctnBeginBlock) then begin
|
|
BeginNode:=Node;
|
|
if not CheckCursorAtProcCall(BeginNode,TargetTool,TargetProcNode) then
|
|
exit;
|
|
end;
|
|
Node:=Node.Parent;
|
|
end;
|
|
|
|
FindProcDeclaration(CurTool,ProcNode);
|
|
if ProcNode<>nil then begin
|
|
// ToDo: add to visited procs
|
|
debugln(['TCodyFindOverloadsDialog.Init ContextProc ',CurTool.CleanPosToStr(ProcNode.StartPos,true),' Class=',CurTool.ExtractProcName(ProcNode,[phpAddClassName])]);
|
|
end;
|
|
|
|
if TargetProcNode=nil then begin
|
|
TargetTool:=CurTool;
|
|
TargetProcNode:=ProcNode;
|
|
end;
|
|
if TargetProcNode=nil then begin
|
|
ResultsGroupBox.Caption:='Need source editor at procedure call or declaration';
|
|
exit;
|
|
end;
|
|
|
|
FTargetName:=TargetTool.ExtractProcName(TargetProcNode,[phpWithoutClassName]);
|
|
FTargetPath:=TargetTool.ExtractProcName(TargetProcNode,[phpAddClassName]);
|
|
TargetTool.CleanPosToCaret(TargetProcNode.StartPos,FTargetXYPosition);
|
|
|
|
FillFilterControls(TargetTool,TargetProcNode);
|
|
|
|
StartParsing;
|
|
|
|
Result:=true;
|
|
end;
|
|
|
|
end.
|
|
|