mirror of
https://github.com/moparisthebest/Simba
synced 2025-01-12 06:08:23 -05:00
1432 lines
40 KiB
ObjectPascal
1432 lines
40 KiB
ObjectPascal
unit v_ideCodeInsight;
|
|
|
|
{$include ValistusDefines.inc}
|
|
|
|
interface
|
|
|
|
uses
|
|
SysUtils, Classes,
|
|
CastaliaPasLex,
|
|
CastaliaPasLexTypes,
|
|
v_ideCodeParser;
|
|
|
|
type
|
|
TCodeInsight = class;
|
|
TCodeInsightArray = array of TCodeInsight;
|
|
|
|
TOnFindInclude = function(Sender: TObject; var FileName: string): Boolean of object;
|
|
TOnLoadLibrary = function(Sender: TObject; var FileName: string; out ci: TCodeInsight): Boolean of object;
|
|
|
|
{ TCodeInsight }
|
|
|
|
TCodeInsight = class(TCodeParser)
|
|
protected
|
|
fFileName: string;
|
|
fMemoryStream: TMemoryStream;
|
|
fOwnStream: Boolean;
|
|
|
|
fPos: Integer;
|
|
fDeclarationAtPos: TDeclaration;
|
|
|
|
fOnFindInclude: TOnFindInclude;
|
|
fOnLoadLibrary: TOnLoadLibrary;
|
|
fIncludes: TCodeInsightArray;
|
|
|
|
InFunc: TDeclarationArray;
|
|
InWith: TDeclarationArray;
|
|
InClassFunction: Integer;
|
|
|
|
Proposal_Filled: Boolean;
|
|
Proposal_ItemList: TStrings;
|
|
Proposal_InsertList: TStrings;
|
|
|
|
procedure SetPos(APos: Integer);
|
|
|
|
procedure Reset;
|
|
procedure Init;
|
|
|
|
function FindInclude(var FileName: string): Boolean;
|
|
procedure ParseInclude(FileName: string);
|
|
function LoadLibrary(var LibName: string): Boolean;
|
|
procedure OnInclude(Sender: TmwBasePasLex); override;
|
|
|
|
function GetVarType(s: string; out Decl: TDeclaration; Return: TVarBase): Boolean;
|
|
function GetFuncType(FuncName, FuncClass: string; out Decl: TDeclaration; Return: TVarBase): Boolean;
|
|
function FindStruct(s: string; out Decl: TDeclaration; Return: TVarBase; var ArrayCount: Integer): Boolean;
|
|
public
|
|
function GetExpressionAtPos(var BraceCount, BracketCount, CommaCount: Integer; out sp, ssp, bp: Integer; IgnoreBrackets: Boolean = False): string; overload;
|
|
function GetExpressionAtPos(var BraceCount, BracketCount, CommaCount: Integer; IgnoreBrackets: Boolean = False): string; overload;
|
|
function GetExpressionAtPos: string; overload;
|
|
function FindVarBase(s: string; GetStruct: Boolean = False; Return: TVarBase = vbName): TDeclaration;
|
|
|
|
constructor Create(FileName: string = ''); reintroduce;
|
|
destructor Destroy; override;
|
|
procedure Assign(From: TObject); override;
|
|
procedure Run(SourceStream: TCustomMemoryStream = nil; BaseDefines: TStringList = nil; MaxPos: Integer = -1; ManageStream: Boolean = False); reintroduce;
|
|
|
|
procedure Proposal_AddDeclaration(Item: TDeclaration; ItemList, InsertList: TStrings);
|
|
procedure FillProposal;
|
|
procedure FillSynCompletionProposal(ItemList, InsertList: TStrings; Prefix: string = '');
|
|
|
|
property OnFindInclude: TOnFindInclude read fOnFindInclude write fOnFindInclude;
|
|
property OnLoadLibrary: TOnLoadLibrary read fOnLoadLibrary write fOnLoadLibrary;
|
|
property FileName: string read fFileName write fFileName;
|
|
property Position: Integer read fPos write SetPos;
|
|
property DeclarationAtPos: TDeclaration read fDeclarationAtPos;
|
|
property Includes: TCodeInsightArray read fIncludes;
|
|
end;
|
|
|
|
TIncludeBuffer = record
|
|
Script: string;
|
|
DefinesIn, DefinesOut: TSaveDefinesRec;
|
|
LastChanged: Integer;
|
|
CodeInsight: TCodeInsight;
|
|
end;
|
|
TIncludeBufferArray = array of TIncludeBuffer;
|
|
|
|
const
|
|
LibPrefix = 'lib:';
|
|
|
|
var
|
|
CoreBuffer: TCodeInsightArray;
|
|
IncludeBuffer: TIncludeBufferArray;
|
|
CoreDefines: TStringList;
|
|
|
|
implementation
|
|
|
|
uses
|
|
v_Constants, v_MiscFunctions;
|
|
|
|
procedure ClearCoreBuffer;
|
|
var
|
|
i: Integer;
|
|
begin
|
|
for i := 0 to High(CoreBuffer) do
|
|
FreeAndNil(CoreBuffer[i]);
|
|
SetLength(IncludeBuffer, 0);
|
|
end;
|
|
|
|
procedure DeleteIncludeBufferIndex(Index: Integer);
|
|
var
|
|
i: Integer;
|
|
tmp : TCodeInsight;
|
|
begin
|
|
tmp := IncludeBuffer[Index].CodeInsight;
|
|
for i := Index to High(IncludeBuffer) - 1 do
|
|
IncludeBuffer[i] := IncludeBuffer[i + 1];
|
|
SetLength(IncludeBuffer, Length(IncludeBuffer) - 1);
|
|
tmp.free;
|
|
end;
|
|
|
|
procedure ClearIncludeBuffer;
|
|
var
|
|
i: Integer;
|
|
begin
|
|
for i := 0 to High(IncludeBuffer) do
|
|
IncludeBuffer[i].CodeInsight.Free;
|
|
|
|
SetLength(IncludeBuffer, 0);
|
|
end;
|
|
|
|
function GetIncludeBuffer(FileName: string; ci: TCodeInsight): TIncludeBuffer;
|
|
var
|
|
i, l, lc: Integer;
|
|
Defines: TSaveDefinesRec;
|
|
DefineMatch: Boolean;
|
|
NewBuf : TIncludeBuffer;
|
|
CS : TRTLCriticalSection;
|
|
begin
|
|
lc := 1;//FileAge(FileName);
|
|
Defines := ci.Lexer.SaveDefines;
|
|
l := Length(IncludeBuffer);
|
|
|
|
for i := l - 1 downto 0 do
|
|
begin
|
|
if (IncludeBuffer[i].CodeInsight <> nil) and (IncludeBuffer[i].CodeInsight.FileName = FileName) then
|
|
begin
|
|
DefineMatch := (IncludeBuffer[i].DefinesIn.Defines = Defines.Defines) and (IncludeBuffer[i].DefinesIn.Stack = Defines.Stack);
|
|
|
|
if (ci.FileName = IncludeBuffer[i].Script) and (not DefineMatch) then
|
|
begin
|
|
DeleteIncludeBufferIndex(i);
|
|
Dec(l);
|
|
Break;
|
|
end
|
|
else if DefineMatch then
|
|
begin
|
|
//if (IncludeBuffer[i].LastChanged = lc) then
|
|
begin
|
|
ci.Lexer.LoadDefines(IncludeBuffer[i].DefinesOut);
|
|
Result := IncludeBuffer[i];
|
|
Exit;
|
|
end;
|
|
|
|
DeleteIncludeBufferIndex(i);
|
|
Dec(l);
|
|
Break;
|
|
end;
|
|
end;
|
|
end;
|
|
with NewBuf do
|
|
begin
|
|
Script := ci.FileName;
|
|
DefinesIn := Defines;
|
|
LastChanged := lc;
|
|
CodeInsight := TCodeInsight.Create(FileName);
|
|
|
|
with CodeInsight do
|
|
begin
|
|
Assign(ci);
|
|
//Lexer.CloneDefinesFrom(ci.Lexer);
|
|
|
|
if (ci.Lexer.Defines.IndexOf('IS_INCLUDE') < 0) then
|
|
i := ci.Lexer.Defines.Add('IS_INCLUDE')
|
|
else
|
|
i := -1;
|
|
Run;
|
|
if (i > -1) then
|
|
begin
|
|
i := ci.Lexer.Defines.IndexOf('IS_INCLUDE');
|
|
if (i > -1) then
|
|
ci.Lexer.Defines.Delete(i);
|
|
end;
|
|
//DefinesOut := Lexer.SaveDefines; Weird bug, so moved out of the with statement
|
|
ci.Lexer.CloneDefinesFrom(Lexer);
|
|
end;
|
|
end;
|
|
InitCriticalSection(cs);
|
|
EnterCriticalsection(cs);
|
|
try
|
|
SetLength(IncludeBuffer, l + 1);
|
|
IncludeBuffer[l] := NewBuf;
|
|
finally
|
|
LeaveCriticalsection(cs);
|
|
end;
|
|
DoneCriticalsection(cs);
|
|
IncludeBuffer[l].DefinesOut := IncludeBuffer[l].CodeInsight.Lexer.SaveDefines;
|
|
Result := IncludeBuffer[l];
|
|
end;
|
|
|
|
function TCodeInsight.FindInclude(var FileName: string): Boolean;
|
|
var
|
|
s: string;
|
|
begin
|
|
s := FileName;
|
|
if Assigned(OnFindInclude) and OnFindInclude(Self, s) then
|
|
begin
|
|
FileName := s;
|
|
Exit(True);
|
|
end;
|
|
|
|
s := ExtractFilePath(fFileName);
|
|
if (s <> '') and FileExists(s + FileName) then
|
|
begin
|
|
FileName := s + FileName;
|
|
Exit(True);
|
|
end;
|
|
|
|
{s := ExtractFilePath(ParamStr(0));
|
|
if (s <> '') and FileExists(s + FileName) then
|
|
begin
|
|
FileName := s + FileName;
|
|
Exit(True);
|
|
end;
|
|
|
|
Result := FileExists(FileName);}
|
|
|
|
Result := False;
|
|
end;
|
|
|
|
procedure TCodeInsight.ParseInclude(FileName: string);
|
|
var
|
|
l: Integer;
|
|
begin
|
|
l := Length(fIncludes);
|
|
SetLength(fIncludes, l + 1);
|
|
fIncludes[l] := GetIncludeBuffer(FileName, Self).CodeInsight;
|
|
end;
|
|
|
|
function TCodeInsight.LoadLibrary(var LibName: string): Boolean;
|
|
var
|
|
i: Integer;
|
|
s: string;
|
|
ci: TCodeInsight;
|
|
tmp : TIncludeBuffer;
|
|
CS : TRTLCriticalSection;
|
|
begin
|
|
Result := False;
|
|
|
|
for i := High(fIncludes) downto 0 do
|
|
if (fIncludes[i].FileName = LibPrefix+LibName) then
|
|
Exit(True);
|
|
|
|
for i := High(IncludeBuffer) downto 0 do
|
|
if (IncludeBuffer[i].Script = LibPrefix+LibName) then
|
|
begin
|
|
SetLength(fIncludes, Length(fIncludes) + 1);
|
|
fIncludes[High(fIncludes)] := IncludeBuffer[i].CodeInsight;
|
|
|
|
Exit(True);
|
|
end;
|
|
|
|
s := LibName;
|
|
if Assigned(OnLoadLibrary) and OnLoadLibrary(Self, s, ci) and (ci <> nil) then
|
|
begin
|
|
LibName := s;
|
|
|
|
SetLength(fIncludes, Length(fIncludes) + 1);
|
|
fIncludes[High(fIncludes)] := ci;
|
|
|
|
with tmp do
|
|
begin
|
|
Script := LibPrefix+LibName;
|
|
CodeInsight := ci;
|
|
CodeInsight.FileName := LibPrefix+LibName;
|
|
end;
|
|
InitCriticalSection(cs);
|
|
EnterCriticalsection(cs);
|
|
try
|
|
SetLength(IncludeBuffer, Length(IncludeBuffer) + 1);
|
|
IncludeBuffer[high(IncludeBuffer)] := tmp;
|
|
finally
|
|
LeaveCriticalsection(cs);
|
|
end;
|
|
DoneCriticalsection(cs);
|
|
Exit(True);
|
|
end;
|
|
end;
|
|
|
|
procedure TCodeInsight.OnInclude(Sender: TmwBasePasLex);
|
|
var
|
|
Param: string;
|
|
i, p: Integer;
|
|
begin
|
|
Param := Sender.DirectiveParamOriginal;
|
|
{$IFDEF FPC}
|
|
Param := SetDirSeparators(param);
|
|
{$ELSE}
|
|
Param := StringReplace(Param, '/', '\', [rfReplaceAll]);
|
|
{$ENDIF}
|
|
if (not Sender.IsJunk) and (Param <> '') then
|
|
begin
|
|
p := Pos('loadlib', LowerCase(Sender.Token));
|
|
if (p > 0) and (p <= 3) then
|
|
begin
|
|
if LoadLibrary(Param) then
|
|
Param := '';
|
|
end
|
|
else if FindInclude(Param) then
|
|
begin
|
|
p := Pos('include_once', LowerCase(Sender.Token));
|
|
if (p > 0) and (p <= 3) then
|
|
for i := High(fIncludes) downto 0 do
|
|
if (fIncludes[i].FileName = Param) then
|
|
begin
|
|
Param := '';
|
|
Break;
|
|
end;
|
|
|
|
if (Param <> '') then
|
|
begin
|
|
ParseInclude(Param);
|
|
Param := '';
|
|
end;
|
|
end;
|
|
|
|
if (Param <> '') and Assigned(OnMessage) then
|
|
OnMessage(Self, meError, Format(ci_UnknownInclude, [Param]), Sender.PosXY.X, Sender.PosXY.Y);
|
|
end;
|
|
|
|
inherited;
|
|
end;
|
|
|
|
procedure TCodeInsight.SetPos(APos: Integer);
|
|
begin
|
|
if (fPos = APos) then
|
|
Exit;
|
|
Reset;
|
|
fPos := APos;
|
|
Init;
|
|
end;
|
|
|
|
procedure TCodeInsight.Init;
|
|
var
|
|
a, b: TDeclarationArray;
|
|
d: TDeclaration;
|
|
i, ii: Integer;
|
|
s: string;
|
|
begin
|
|
if (fPos >= 0) then
|
|
fDeclarationAtPos := fItems.GetItemInPos(fPos, fPos, True)
|
|
else
|
|
fDeclarationAtPos := nil;
|
|
|
|
(*if (fDeclarationatPos is TciJunk) and (fDeclarationatPos.Owner <> nil) {and (fDeclarationatPos.EndPos = fPos)} then
|
|
fDeclarationAtPos := fDeclarationatPos.Owner;*)
|
|
|
|
if (fDeclarationAtPos <> nil) and (not (fDeclarationAtPos is TciJunk)) then
|
|
begin
|
|
InFunc := fDeclarationAtPos.GetOwnersOfClass(TciProcedureDeclaration);
|
|
if (fDeclarationAtPos is TciProcedureDeclaration) then
|
|
begin
|
|
SetLength(InFunc, Length(InFunc) + 1);
|
|
InFunc[High(InFunc)] := fDeclarationAtPos;
|
|
end;
|
|
|
|
if (Length(InFunc) > 0) and (InFunc[0].Owner = nil) then
|
|
begin
|
|
d := InFunc[0].Items.GetFirstItemOfClass(TciProcedureClassName);
|
|
if (d <> nil) then
|
|
d := FindVarBase(d.CleanText, True, vbType);
|
|
if (d <> nil) and (d is TciStruct) then
|
|
begin
|
|
SetLength(InWith, Length(InWith) + 1);
|
|
InClassFunction := High(InWith);
|
|
InWith[InClassFunction] := d;
|
|
end;
|
|
end;
|
|
|
|
a := fDeclarationAtPos.GetOwnersOfClass(TciWithStatement);
|
|
if (fDeclarationAtPos is TciWithStateMent) then
|
|
begin
|
|
SetLength(a, Length(a) + 1);
|
|
a[High(a)] := fDeclarationAtPos;
|
|
end
|
|
else if fDeclarationAtPos.HasOwnerClass(TciClassType, d, True) then
|
|
begin
|
|
SetLength(InWith, Length(InWith) + 1);
|
|
InWith[High(InWith)] := d;
|
|
end;
|
|
|
|
for i := High(a) downto Low(a) do
|
|
begin
|
|
b := a[i].Items.GetItemsOfClass(TciVariable);
|
|
for ii := Low(b) to High(b) do
|
|
begin
|
|
if (fDeclarationAtPos is TciVariable) and (b[ii].EndPos >= fPos) then
|
|
Continue;
|
|
s := b[ii].CleanText;
|
|
d := FindVarBase(s, True, vbType);
|
|
if (d <> nil) and (d is TciStruct) then
|
|
begin
|
|
SetLength(InWith, Length(InWith) + 1);
|
|
InWith[High(InWith)] := d;
|
|
end;
|
|
end;
|
|
end;
|
|
end;
|
|
end;
|
|
|
|
procedure TCodeInsight.Reset;
|
|
begin
|
|
Lexer.Init;
|
|
|
|
//SetLength(fIncludes, 0);
|
|
SetLength(InFunc, 0);
|
|
SetLength(InWith, 0);
|
|
InClassFunction := -1;
|
|
|
|
Proposal_ItemList.Clear;
|
|
Proposal_InsertList.Clear;
|
|
Proposal_Filled := False;
|
|
|
|
fDeclarationAtPos := nil;
|
|
end;
|
|
|
|
function TCodeInsight.GetVarType(s: string; out Decl: TDeclaration; Return: TVarBase): Boolean;
|
|
var
|
|
a, b: TDeclarationArray;
|
|
i, ii, iii, iiii, FuncOffset, SetOffset: Integer;
|
|
c: array [1..3] of TDeclarationClass;
|
|
d: TDeclaration;
|
|
begin
|
|
Result := False;
|
|
FuncOffset := 0;
|
|
SetOffset := 0;
|
|
|
|
if (s = 'RESULT') and (Length(InFunc) > 0) then
|
|
begin
|
|
Decl := InFunc[Low(InFunc)].Items.GetFirstItemOfClass(TciReturnType);
|
|
if (Decl <> nil) then
|
|
begin
|
|
Result := True;
|
|
Exit;
|
|
end;
|
|
end;
|
|
|
|
if (s = 'SELF') and (InClassFunction <> -1) and InWith[InClassFunction].HasOwnerClass(TciTypeDeclaration, d, True) then
|
|
begin
|
|
Decl := d.Items.GetFirstItemOfClass(TciTypeName);
|
|
if (Decl <> nil) then
|
|
begin
|
|
Result := True;
|
|
Exit;
|
|
end;
|
|
end;
|
|
|
|
for i := Low(InFunc) to High(InFunc) + 1 do
|
|
begin
|
|
for iiii := 1 to 3 do
|
|
begin
|
|
case iiii of
|
|
1:
|
|
begin
|
|
c[1] := TciVarDeclaration;
|
|
c[2] := TciVarName;
|
|
c[3] := TciTypeKind;
|
|
end;
|
|
2:
|
|
begin
|
|
if (Return = vbType) then
|
|
Continue;
|
|
c[1] := TciConstantDeclaration;
|
|
c[2] := TciConstantName;
|
|
c[3] := TciExpression;
|
|
end;
|
|
3:
|
|
begin
|
|
if (Return = vbType) then
|
|
Continue;
|
|
c[1] := TciLabelDeclaration;
|
|
c[2] := TciLabelName;
|
|
c[3] := nil;
|
|
end;
|
|
else
|
|
Break;
|
|
end;
|
|
if (i = High(InFunc) + 1) then
|
|
a := fItems.GetItemsOfClass(c[1])
|
|
else
|
|
a := InFunc[i].Items.GetItemsOfClass(c[1]);
|
|
for ii := Low(a) to High(a) do
|
|
begin
|
|
b := a[ii].Items.GetItemsOfClass(c[2]);
|
|
for iii := Low(b) to High(b) do
|
|
if (PrepareString(b[iii].CleanText) = s) then
|
|
begin
|
|
Result := True;
|
|
if (Return = vbType) and (c[3] <> nil) then
|
|
Decl := a[ii].Items.GetFirstItemOfClass(c[3])
|
|
else
|
|
Decl := b[iii];
|
|
Exit;
|
|
end;
|
|
end;
|
|
end;
|
|
|
|
if (Return = vbName) then
|
|
begin
|
|
if (i = High(InFunc) + 1) then
|
|
a := fItems.GetItemsOfClass(TciQualifiedIdentifier, True)
|
|
else
|
|
a := InFunc[i].Items.GetItemsOfClass(TciQualifiedIdentifier, True);
|
|
for ii := Low(a) to High(a) - SetOffset do
|
|
begin
|
|
if (i = High(InFunc) + 1) then
|
|
if a[ii].HasOwnerClass(TciTypeDeclaration, d, True) then
|
|
begin
|
|
if d.HasOwnerClass(TciProcedureDeclaration, d, True) then
|
|
Continue;
|
|
end
|
|
else if a[ii].HasOwnerClass(TciProcedureDeclaration, d, True) and (d.Owner <> nil) then
|
|
Continue;
|
|
|
|
if (PrepareString(a[ii].CleanText) = s) then
|
|
begin
|
|
Result := True;
|
|
Decl := a[ii];
|
|
Exit;
|
|
end;
|
|
end;
|
|
SetOffset := Length(a);
|
|
end;
|
|
|
|
if (i <= High(InFunc)) then
|
|
begin
|
|
a := InFunc[i].Items.GetItemsOfClass(TciParameterName, True);
|
|
for ii := Low(a) to High(a) - FuncOffset do
|
|
if (PrepareString(a[ii].CleanText) = s) then
|
|
begin
|
|
Result := True;
|
|
if (Return = vbType) then
|
|
Decl := a[ii].Owner.Items.GetFirstItemOfClass(TciParameterType)
|
|
else
|
|
Decl := a[ii];
|
|
Exit;
|
|
end;
|
|
FuncOffset := Length(a);
|
|
end;
|
|
end;
|
|
end;
|
|
|
|
function TCodeInsight.GetFuncType(FuncName, FuncClass: string; out Decl: TDeclaration; Return: TVarBase): Boolean;
|
|
var
|
|
a: TDeclarationArray;
|
|
b: TDeclaration;
|
|
i, ii: Integer;
|
|
begin
|
|
Result := False;
|
|
|
|
for ii := Low(InFunc) to High(InFunc) + 1 do
|
|
begin
|
|
if (ii = High(InFunc) + 1) then
|
|
a := fItems.GetItemsOfClass(TciProcedureDeclaration)
|
|
else
|
|
a := InFunc[ii].Items.GetItemsOfClass(TciProcedureDeclaration);
|
|
for i := Low(a) to High(a) do
|
|
begin
|
|
b := nil;
|
|
if (FuncClass <> '') then
|
|
b := a[i].Items.GetFirstItemOfClass(TciProcedureClassName);
|
|
if ((FuncClass = '') and (b = nil)) or ((b <> nil) and (PrepareString(b.CleanText) = FuncClass)) then
|
|
begin
|
|
b := a[i].Items.GetFirstItemOfClass(TciProcedureName);
|
|
if (b <> nil) and (PrepareString(b.CleanText) = FuncName) then
|
|
begin
|
|
Result := True;
|
|
if (Return = vbType) then
|
|
Decl := a[i].Items.GetFirstItemOfClass(TciReturnType)
|
|
else
|
|
Decl := b;
|
|
|
|
if (a[i].Items.GetFirstItemOfClass(TciForward) = nil) then
|
|
Exit;
|
|
end;
|
|
end;
|
|
end;
|
|
end;
|
|
end;
|
|
|
|
function TCodeInsight.FindStruct(s: string; out Decl: TDeclaration; Return: TVarBase; var ArrayCount: Integer): Boolean;
|
|
var
|
|
a, b: TDeclarationArray;
|
|
i, ii: Integer;
|
|
begin
|
|
Result := False;
|
|
s := PrepareString(s);
|
|
a := fItems.GetItemsOfClass(TciTypeDeclaration);
|
|
|
|
for i := Low(a) to High(a) do
|
|
begin
|
|
b := a[i].Items.GetItemsOfClass(TciTypeName);
|
|
for ii := Low(b) to High(b) do
|
|
if (PrepareString(b[ii].CleanText) = s) then
|
|
begin
|
|
Result := True;
|
|
if (Return = vbType) then
|
|
begin
|
|
Decl := a[i].Items.GetFirstItemOfClass(TciTypeKind);
|
|
if (Decl = nil) then
|
|
Decl := a[i].Items.GetFirstItemOfClass(TciClassType);
|
|
if (Decl is TciTypeKind) then
|
|
Decl := TciTypeKind(Decl).GetRealType(ArrayCount);
|
|
if (Decl is TciStruct) then
|
|
Result := True
|
|
else
|
|
Result := (Decl <> nil) and (PrepareString(Decl.CleanText) <> s) and FindStruct(Decl.CleanText, Decl, Return, ArrayCount);
|
|
end
|
|
else
|
|
Decl := b[ii];
|
|
if (Decl <> nil) then
|
|
Exit;
|
|
end;
|
|
end;
|
|
end;
|
|
|
|
function TCodeInsight.GetExpressionAtPos(var BraceCount, BracketCount, CommaCount: Integer; out sp, ssp, bp: Integer; IgnoreBrackets: Boolean): string;
|
|
var
|
|
i, StartPos, EndPos, Dif: Integer;
|
|
s: string;
|
|
a: TDeclarationArray;
|
|
d: TDeclaration;
|
|
LastWasDot: Boolean;
|
|
begin
|
|
Result := '';
|
|
d := nil;
|
|
sp := -1;
|
|
ssp := -1;
|
|
bp := -1;
|
|
if (fDeclarationAtPos = nil) or
|
|
(fDeclarationAtPos is TciJunk) or
|
|
(not (
|
|
(fDeclarationAtPos.HasOwnerClass(TciCompoundStatement, d, True) or (fDeclarationAtPos is TciCompoundStatement)) or
|
|
(fDeclarationAtPos.HasOwnerClass(TciConstantDeclaration, d, True) or (fDeclarationAtPos is TciConstantDeclaration)) or
|
|
(fDeclarationAtPos.HasOwnerClass(TciVarDeclaration, d, True) or (fDeclarationAtPos is TciVarDeclaration)) or
|
|
(fDeclarationAtPos.HasOwnerClass(TciLabelDeclaration, d, True) or (fDeclarationAtPos is TciLabelDeclaration)) or
|
|
(fDeclarationAtPos.HasOwnerClass(TciTypeDeclaration, d, True) or (fDeclarationAtPos is TciTypeDeclaration)) or
|
|
(fDeclarationAtPos.HasOwnerClass(TciProcedureDeclaration, d, True) or (fDeclarationAtPos is TciProcedureDeclaration))
|
|
)) then
|
|
Exit;
|
|
if (d = nil) then
|
|
d := fDeclarationAtPos;
|
|
a := d.Items.GetItemsOfClass(TciJunk, True);
|
|
EndPos := fPos - d.StartPos;
|
|
s := d.CleanText;
|
|
|
|
Dif := EndPos;
|
|
for i := Low(a) to High(a) do
|
|
if (fPos > a[i].EndPos) then
|
|
if (Pos(LineEnding, a[i].RawText) > 0) then
|
|
EndPos := EndPos - a[i].EndPos + a[i].StartPos + Length(LineEnding)
|
|
else
|
|
EndPos := EndPos - a[i].EndPos + a[i].StartPos + 1;
|
|
Dif := Dif - EndPos;
|
|
|
|
StartPos := EndPos;
|
|
LastWasDot := False;
|
|
if (StartPos <= Length(s)) then
|
|
while (StartPos > 0) do
|
|
begin
|
|
if (BraceCount = 0) and ((BracketCount = 0) or IgnoreBrackets) and (s[StartPos] in ['a'..'z', 'A'..'Z', '0'..'9', '_']) then
|
|
{nothing}
|
|
else if (BraceCount = 0) and ((BracketCount = 0) or IgnoreBrackets) and (s[StartPos] in [#10, #11, #13, #32]) then
|
|
begin
|
|
i := StartPos;
|
|
Dec(StartPos);
|
|
while (StartPos > 0) and (s[StartPos] in [#10, #11, #13, #32]) do
|
|
Dec(StartPos);
|
|
if (StartPos > 0) and (not ((LastWasDot and (s[StartPos] in ['a'..'z', 'A'..'Z', '0'..'9', '_', ']', ')'])) or ((not LastWasDot) and (s[StartPos] = '.')))) then
|
|
begin
|
|
StartPos := i - BracketCount - BraceCount;
|
|
if (ssp = -1) then ssp := StartPos;
|
|
Break;
|
|
end;
|
|
Inc(StartPos);
|
|
end
|
|
else if (s[StartPos] = '.') then
|
|
begin
|
|
if (ssp = -1) then ssp := StartPos;
|
|
LastWasDot := True;
|
|
Dec(StartPos);
|
|
Continue;
|
|
end
|
|
else if (s[StartPos] = ']') then
|
|
Inc(BracketCount)
|
|
else if (s[StartPos] = '[') then
|
|
begin
|
|
Dec(BracketCount);
|
|
LastWasDot := True;
|
|
Dec(StartPos);
|
|
Continue;
|
|
end
|
|
else if (s[StartPos] = ')') then
|
|
Inc(BraceCount)
|
|
else if (s[StartPos] = '(') then
|
|
begin
|
|
Dec(BraceCount);
|
|
LastWasDot := True;
|
|
Dec(StartPos);
|
|
if (BraceCount = 0) then
|
|
bp := StartPos;
|
|
Continue;
|
|
end
|
|
else if (BraceCount = 1) and (BracketCount = 0) and (s[StartPos] = ',') then
|
|
Inc(CommaCount)
|
|
else if (BraceCount = 0) and ((BracketCount = 0) or IgnoreBrackets) then
|
|
Break;
|
|
|
|
if (BraceCount < 0) or ((BracketCount < 0) and (not IgnoreBrackets)) then
|
|
begin
|
|
Dec(StartPos, BraceCount);
|
|
Dec(StartPos, BracketCount);
|
|
Break;
|
|
end;
|
|
|
|
LastWasDot := False;
|
|
Dec(StartPos);
|
|
end;
|
|
if (ssp = -1) then ssp := StartPos;
|
|
if (bp = -1) then bp := EndPos;
|
|
sp := StartPos + d.StartPos + Dif + 1;
|
|
ssp := ssp + d.StartPos + Dif + 1;
|
|
bp := bp + d.StartPos + Dif + 1;
|
|
if (EndPos > Length(s)) then
|
|
Result := ''
|
|
else
|
|
Result := CompressWhiteSpace(Copy(s, StartPos + 1, EndPos - StartPos));
|
|
end;
|
|
|
|
function TCodeInsight.GetExpressionAtPos(var BraceCount, BracketCount, CommaCount: Integer; IgnoreBrackets: Boolean = False): string;
|
|
var
|
|
sp, ssp, bp: integer;
|
|
begin
|
|
result := GetExpressionAtPos(bracecount,bracketcount,commacount,sp,ssp,bp,ignorebrackets);
|
|
end;
|
|
|
|
function TCodeInsight.GetExpressionAtPos: string;
|
|
var
|
|
bcc, bkc, cc: Integer;
|
|
begin
|
|
bcc := 0;
|
|
bkc := 0;
|
|
cc := 0;
|
|
Result := GetExpressionAtPos(bcc, bkc, cc);
|
|
end;
|
|
|
|
function TCodeInsight.FindVarBase(s: string; GetStruct: Boolean = False; Return: TVarBase = vbName): TDeclaration;
|
|
|
|
function PartOfWith(s: string; out Decl: TDeclaration; Return: TVarBase; CheckClass: Boolean; var ArrayCount: Integer): Boolean;
|
|
var
|
|
i: Integer;
|
|
begin
|
|
Result := False;
|
|
for i := High(InWith) downto Low(InWith) do
|
|
if CheckClass xor (i <> InClassFunction) then
|
|
if TciStruct(InWith[i]).HasField(s, Decl, Return, ArrayCount) then
|
|
begin
|
|
Result := True;
|
|
Break;
|
|
end;
|
|
end;
|
|
|
|
function DoGetVarType(s: string; out Decl: TDeclaration; Return: TVarBase): Boolean;
|
|
|
|
function CheckIt(s: string; Item: TCodeInsight; out Decl: TDeclaration; Return: TVarBase; const CheckCore: Boolean = True): Boolean;
|
|
var
|
|
i: Integer;
|
|
begin
|
|
Result := False;
|
|
if Item.GetVarType(s, Decl, Return) then
|
|
Exit(True);
|
|
|
|
for i := High(Item.Includes) downto Low(Item.Includes) do
|
|
if CheckIt(s, Item.Includes[i], Decl, Return) then
|
|
Exit(True);
|
|
|
|
if (not CheckCore) then
|
|
Exit;
|
|
for i := High(CoreBuffer) downto Low(CoreBuffer) do
|
|
if CheckIt(s, CoreBuffer[i], Decl, Return, False) then
|
|
Exit(True);
|
|
end;
|
|
|
|
begin
|
|
Result := CheckIt(s, Self, Decl, Return);
|
|
end;
|
|
|
|
function DoGetFuncType(FuncName, FuncClass: string; out Decl: TDeclaration; Return: TVarBase): Boolean;
|
|
|
|
function CheckIt(FuncName, FuncClass: string; Item: TCodeInsight; out Decl: TDeclaration; Return: TVarBase; const CheckCore: Boolean = True): Boolean;
|
|
var
|
|
i: Integer;
|
|
begin
|
|
Result := False;
|
|
if Item.GetFuncType(FuncName, FuncClass, Decl, Return) then
|
|
Exit(True);
|
|
|
|
for i := High(Item.Includes) downto Low(Item.Includes) do
|
|
if CheckIt(FuncName, FuncClass, Item.Includes[i], Decl, Return) then
|
|
Exit(True);
|
|
|
|
if (not CheckCore) then
|
|
Exit;
|
|
for i := High(CoreBuffer) downto Low(CoreBuffer) do
|
|
if CheckIt(FuncName, FuncClass, CoreBuffer[i], Decl, Return, False) then
|
|
Exit(True);
|
|
end;
|
|
|
|
begin
|
|
Result := CheckIt(FuncName, FuncClass, Self, Decl, Return);
|
|
end;
|
|
|
|
function DoFindStruct(s: string; out Decl: TDeclaration; Return: TVarBase; var ArrayCount: Integer): Boolean;
|
|
|
|
function CheckIt(s: string; Item: TCodeInsight; out Decl: TDeclaration; Return: TVarBase; var ArrayCount: Integer; const CheckCore: Boolean = True): Boolean;
|
|
var
|
|
i: Integer;
|
|
begin
|
|
Result := False;
|
|
if Item.FindStruct(s, Decl, Return, ArrayCount) then
|
|
Exit(True);
|
|
|
|
for i := High(Item.Includes) downto Low(Item.Includes) do
|
|
if CheckIt(s, Item.Includes[i], Decl, Return, ArrayCount) then
|
|
Exit(True);
|
|
|
|
if (not CheckCore) then
|
|
Exit;
|
|
for i := High(CoreBuffer) downto Low(CoreBuffer) do
|
|
if CheckIt(s, CoreBuffer[i], Decl, Return, ArrayCount, False) then
|
|
Exit(True);
|
|
end;
|
|
|
|
begin
|
|
Result := CheckIt(s, Self, Decl, Return, ArrayCount);
|
|
end;
|
|
|
|
var
|
|
f, NameClass: string;
|
|
sa: TStringArray;
|
|
d: TDeclaration;
|
|
i, NeedArrayCount, ArrayCount: Integer;
|
|
InStruct: TciStruct;
|
|
VarBase: TVarBase;
|
|
Found, CheckVar, IsForward: Boolean;
|
|
begin
|
|
Result := nil;
|
|
if (fDeclarationAtPos is TciJunk) or (PrepareString(s) = '') then
|
|
Exit;
|
|
|
|
InStruct := nil;
|
|
NameClass := '';
|
|
IsForward := False;
|
|
CheckVar := (fPos < 0);
|
|
sa := Explode('.', PrePrepareString(s));
|
|
|
|
if (fDeclarationAtPos <> nil) then
|
|
begin
|
|
if (fDeclarationAtPos is TciProcedureName) and (fDeclarationAtPos.Owner <> nil) then
|
|
begin
|
|
if fDeclarationAtPos.HasOwnerClass(TciClassType, d, True) and (d.Owner <> nil) then
|
|
begin
|
|
d := d.Owner.Items.GetFirstItemOfClass(TciTypeName);
|
|
if (d <> nil) then
|
|
NameClass := PrepareString(d.CleanText);
|
|
end
|
|
else if (fDeclarationAtPos.Owner.Items.GetFirstItemOfClass(TciForward) <> nil) then
|
|
IsForward := True;
|
|
end;
|
|
|
|
if (Return = vbName) and (
|
|
(fDeclarationAtPos is TciVarName) or
|
|
(fDeclarationAtPos is TciConstantName) or
|
|
(fDeclarationAtPos is TciTypeName) or
|
|
(fDeclarationAtPos is TciLabelName) or
|
|
((fDeclarationAtPos is TciProcedureName) and
|
|
(NameClass = '') and
|
|
(fDeclarationAtPos.Owner <> nil) and
|
|
(not IsForward) and
|
|
(fDeclarationAtPos.Owner.Items.GetFirstItemOfClass(TciProcedureClassName) = nil)) or
|
|
(fDeclarationAtPos is TciParameterName) or
|
|
(fDeclarationAtPos is TciFieldName) or
|
|
(fDeclarationAtPos is TciQualifiedIdentifier)) then
|
|
Exit;
|
|
CheckVar := (NameClass = '') and ((
|
|
(Length(InFunc) > 0) or
|
|
(Length(InWith) > 0) or
|
|
(fDeclarationAtPos is TciCompoundStatement) or
|
|
(fDeclarationAtPos is TciExpression) or
|
|
fDeclarationAtPos.HasOwnerClass(TciCompoundStatement, d, True)) or (
|
|
(Return = vbType) and (
|
|
(fDeclarationAtPos is TciTypeKind) or
|
|
(fDeclarationAtPos is TciParameterType) or
|
|
(fDeclarationAtPos is TciReturnType) or
|
|
(fDeclarationAtPos is TciQualifiedIdentifier) or
|
|
(fDeclarationAtPos is TciExpression))));
|
|
end;
|
|
|
|
for i := Low(sa) to High(sa) do
|
|
begin
|
|
f := PrepareString(sa[i], ArrayCount);
|
|
NeedArrayCount := 0;
|
|
|
|
if (Return = vbName) and (i = High(sa)) then
|
|
VarBase := vbName
|
|
else
|
|
VarBase := vbType;
|
|
|
|
if (InStruct <> nil) then
|
|
Found := InStruct.HasField(f, Result, VarBase, NeedArrayCount)
|
|
else
|
|
begin
|
|
Found := CheckVar and PartOfWith(f, Result, VarBase, False, NeedArrayCount);
|
|
if (not Found) and (i = Low(sa)) then
|
|
begin
|
|
Found :=
|
|
(CheckVar and DoGetVarType(f, Result, VarBase)) or
|
|
((CheckVar or (NameClass <> '') or IsForward) and DoGetFuncType(f, NameClass, Result, VarBase)) or
|
|
DoFindStruct(f, Result, VarBase, NeedArrayCount);
|
|
end;
|
|
if (not Found) and CheckVar then
|
|
Found := PartOfWith(f, Result, VarBase, True, NeedArrayCount);
|
|
end;
|
|
|
|
if Found and (Result is TciTypeKind) then
|
|
Result := TciTypeKind(Result).GetRealType(NeedArrayCount);
|
|
|
|
if Found and (Result <> nil) and (not (Result is TciStruct)) and ((i < High(sa)) or GetStruct) then
|
|
Found := DoFindStruct(Result.CleanText, Result, VarBase, NeedArrayCount);
|
|
|
|
if Found and (Result <> nil) and (ArrayCount <> NeedArrayCount) and ((Result is TciStruct) or (InStruct <> nil)) and ((i < High(sa)) or GetStruct) then
|
|
begin
|
|
if (InStruct <> nil) then
|
|
Result := InStruct.GetDefault(VarBase)
|
|
else
|
|
Result := TciStruct(Result).GetDefault(VarBase);
|
|
|
|
Found := (Result <> nil);
|
|
|
|
if Found and (Result is TciTypeKind) then
|
|
Result := TciTypeKind(Result).GetRealType(NeedArrayCount);
|
|
|
|
if Found and (not (Result is TciStruct)) and ((i < High(sa)) or GetStruct) then
|
|
Found := DoFindStruct(Result.CleanText, Result, VarBase, NeedArrayCount);
|
|
|
|
if (not Found) then
|
|
begin
|
|
if Assigned(OnMessage) then
|
|
OnMessage(Self, meError, Format(ci_RangeError, [sa[i], s]), -1, -1);
|
|
Result := nil;
|
|
Exit;
|
|
end;
|
|
end;
|
|
|
|
if (not Found) or (Result = nil) then
|
|
begin
|
|
if Assigned(OnMessage) then
|
|
if (i > Low(sa)) then
|
|
OnMessage(Self, meError, Format(ci_UnknownMember, [sa[i], s]), -1, -1)
|
|
else
|
|
OnMessage(Self, meError, Format(ci_UnknownStruct, [sa[i], s]), -1, -1);
|
|
Result := nil;
|
|
Exit;
|
|
end;
|
|
|
|
InStruct := TciStruct(Result);
|
|
end;
|
|
end;
|
|
|
|
constructor TCodeInsight.Create(FileName: string = '');
|
|
begin
|
|
inherited Create;
|
|
|
|
Proposal_InsertList := TStringList.Create;
|
|
TStringList(Proposal_InsertList).Sorted := True;
|
|
Proposal_ItemList := TStringList.Create;
|
|
|
|
fOnFindInclude := nil;
|
|
fOnLoadLibrary := nil;
|
|
fFileName := FileName;
|
|
if (fFileName <> '') and (not FileExists(fFileName)) then
|
|
fFileName := '';
|
|
fPos := -1;
|
|
Reset;
|
|
|
|
Lexer.Defines.AddStrings(CoreDefines);
|
|
|
|
fOwnStream := (fFileName <> '');
|
|
if fOwnStream then
|
|
begin
|
|
fMemoryStream := TMemoryStream.Create;
|
|
with TStringList.Create do
|
|
begin
|
|
LoadFromFile(filename); //Converts the line-endings.
|
|
SaveToStream(fMemoryStream);
|
|
Free;
|
|
end;
|
|
end
|
|
else
|
|
fMemoryStream := nil;
|
|
end;
|
|
|
|
destructor TCodeInsight.Destroy;
|
|
begin
|
|
if fOwnStream then
|
|
FreeAndNil(fMemoryStream);
|
|
|
|
FreeAndNil(Proposal_InsertList);
|
|
FreeAndNil(Proposal_ItemList);
|
|
|
|
inherited;
|
|
end;
|
|
|
|
procedure TCodeInsight.Assign(From: TObject);
|
|
begin
|
|
if (From is TCodeInsight) then
|
|
with From as TCodeInsight do
|
|
begin
|
|
Self.OnFindInclude := OnFindInclude;
|
|
Self.OnLoadLibrary := OnLoadLibrary;
|
|
end;
|
|
|
|
inherited;
|
|
end;
|
|
|
|
procedure TCodeInsight.Run(SourceStream: TCustomMemoryStream = nil; BaseDefines: TStringList = nil; MaxPos: Integer = -1; ManageStream: Boolean = False);
|
|
begin
|
|
if Assigned(BaseDefines) then
|
|
begin
|
|
Lexer.ClearDefines;
|
|
Lexer.Defines.AddStrings(CoreDefines);
|
|
Lexer.Defines.AddStrings(BaseDefines);
|
|
end;
|
|
SetLength(fIncludes, 0);
|
|
|
|
if ManageStream then
|
|
begin
|
|
if (SourceStream <> nil) then
|
|
begin
|
|
if fOwnStream then
|
|
FreeAndNil(fMemoryStream);
|
|
fMemoryStream := TMemoryStream(SourceStream);
|
|
end;
|
|
fOwnStream := True;
|
|
end;
|
|
|
|
if fOwnStream then
|
|
inherited Run(fMemoryStream, MaxPos - 1)
|
|
else if Assigned(SourceStream) then
|
|
inherited Run(SourceStream, MaxPos - 1);
|
|
|
|
if (MaxPos > -1) then
|
|
Position := MaxPos;
|
|
end;
|
|
|
|
procedure TCodeInsight.Proposal_AddDeclaration(Item: TDeclaration; ItemList, InsertList: TStrings);
|
|
|
|
function FormatFirstColumn(s: string): string; inline;
|
|
begin
|
|
Result := '{#clNavy}' + LowerCase(s) + '{#0} {|}'
|
|
end;
|
|
|
|
function FormatTypeName(s: string): string; inline;
|
|
begin
|
|
Result := '{#clBlue}' + s;
|
|
end;
|
|
|
|
function FormatMainName(s: string): string; inline;
|
|
begin
|
|
Result := '{B+}' + s + '{B-}';
|
|
end;
|
|
|
|
function FormatMainExtra(s: string): string; inline;
|
|
begin
|
|
Result := '{I+}' + s + '{I-}'
|
|
end;
|
|
|
|
function GetEnumTypeName(Item: TDeclaration): string;
|
|
begin
|
|
Result := 'Enumeration';
|
|
if (Item.Items.Count > 0) then
|
|
begin
|
|
Result := Result + '(' + Item.Items[0].ShortText;
|
|
if (Item.Items.Count > 1) then
|
|
Result := Result + '..' + Item.Items[Item.Items.Count - 1].ShortText;
|
|
Result := Result + ')';
|
|
end;
|
|
end;
|
|
|
|
procedure AddEnums(Item: {TCodeInsight}TDeclaration; ItemList, InsertList: TStrings); overload;
|
|
var
|
|
{i,} ii, SetOffset: Integer;
|
|
a: TDeclarationArray;
|
|
d: TDeclaration;
|
|
s: string;
|
|
begin
|
|
SetOffset := 0;
|
|
//for i := Low(Item.InFunc) to High(Item.InFunc) + 1 do
|
|
begin
|
|
//if (i = High(Item.InFunc) + 1) then
|
|
a := Item.Items.GetItemsOfClass(TciQualifiedIdentifier, True);
|
|
//else
|
|
// a := Item.InFunc[i].Items.GetItemsOfClass(TciQualifiedIdentifier, True);
|
|
for ii := Low(a) to High(a) - SetOffset do
|
|
begin
|
|
{if (i = High(Item.InFunc) + 1) then
|
|
if a[ii].HasOwnerClass(TciTypeDeclaration, d, True) then
|
|
begin
|
|
if d.HasOwnerClass(TciProcedureDeclaration, d, True) then
|
|
Continue;
|
|
end
|
|
else if a[ii].HasOwnerClass(TciProcedureDeclaration, d, True) and (d.Owner <> nil) then
|
|
Continue;}
|
|
|
|
{$IFDEF ciCHECKDUPLICATES}
|
|
if (InsertList.IndexOf(a[ii].ShortText) > -1) then
|
|
Continue;
|
|
{$ENDIF}
|
|
|
|
s := FormatFirstColumn('enum') + FormatMainName(a[ii].ShortText);
|
|
if a[ii].HasOwnerClass(TciTypeDeclaration, d, True) then
|
|
begin
|
|
d := d.Items.GetFirstItemOfClass(TciTypeName);
|
|
if (d <> nil) then
|
|
s := s + ' ' + FormatMainExtra('(' + d.ShortText + ')');
|
|
end;
|
|
ItemList.Insert(InsertList.Add(a[ii].ShortText), s);
|
|
end;
|
|
//SetOffset := Length(a);
|
|
end;
|
|
end;
|
|
|
|
procedure AddEnums(a: TDeclarationArray; ItemList, InsertList: TStrings); overload;
|
|
var
|
|
i: Integer;
|
|
begin
|
|
for i := Low(a) to High(a) do
|
|
AddEnums(a[i], ItemList, InsertList);
|
|
end;
|
|
|
|
procedure AddFuncDeclaration(Item: TciProcedureDeclaration; ItemList, InsertList: TStrings);
|
|
var
|
|
s, n: string;
|
|
d: TDeclaration;
|
|
begin
|
|
s := FormatFirstColumn(Item.ProcType);
|
|
d := Item.Items.GetFirstItemOfClass(TciProcedureName);
|
|
if (d = nil) then
|
|
Exit;
|
|
n := d.ShortText;
|
|
|
|
{$IFDEF ciCHECKDUPLICATES}
|
|
if (InsertList.IndexOf(n) > -1) then
|
|
Exit;
|
|
{$ENDIF}
|
|
|
|
s := s + FormatMainName(n);
|
|
if (Item.Params <> '') then
|
|
begin
|
|
s := s + FormatMainExtra('(' + Item.Params + ')');
|
|
//n := n {+ '('};
|
|
end;
|
|
d := Item.Items.GetFirstItemOfClass(TciReturnType);
|
|
if (d <> nil) then
|
|
s := s + ': ' + FormatTypeName(d.ShortText)
|
|
{else
|
|
s := s + ';'};
|
|
ItemList.Insert(InsertList.Add(n), s);
|
|
|
|
AddEnums(Item.GetParamDeclarations, ItemList, InsertList);
|
|
if (d <> nil) then
|
|
AddEnums(d, ItemList, InsertList);
|
|
end;
|
|
|
|
function PropertyIndex(Item: TciClassProperty): string;
|
|
var
|
|
i: Integer;
|
|
d: TDeclaration;
|
|
a: TDeclarationArray;
|
|
begin
|
|
d := Item.Items.GetFirstItemOfClass(TciPropertyParameterList);
|
|
Result := '';
|
|
|
|
if (d <> nil) then
|
|
begin
|
|
a := d.Items.GetItemsOfClass(TciIdentifier);
|
|
for i := Low(a) to High(a) do
|
|
begin
|
|
if (Result <> '') then
|
|
Result := Result + ', ';
|
|
Result := Result + a[i].ShortText;
|
|
end;
|
|
|
|
d := d.Items.GetFirstItemOfClass(TciTypeKind);
|
|
if (d <> nil) then
|
|
begin
|
|
if (Result <> '') then
|
|
Result := Result + ': ';
|
|
Result := Result + d.ShortText;
|
|
end;
|
|
end;
|
|
|
|
if (Result <> '') then
|
|
Result := '['+Result+']';
|
|
end;
|
|
|
|
var
|
|
i: Integer;
|
|
FirstColumn, s, n: string;
|
|
a: TDeclarationArray;
|
|
b: array[1..2] of TDeclaration;
|
|
c: array[0..2] of TDeclarationClass;
|
|
begin
|
|
if item = nil then
|
|
exit;
|
|
if (Item is TciProcedureDeclaration) then
|
|
begin
|
|
AddFuncDeclaration(TciProcedureDeclaration(Item), ItemList, InsertList);
|
|
Exit;
|
|
end;
|
|
|
|
c[0] := nil;
|
|
c[1] := nil;
|
|
c[2] := nil;
|
|
if (Item is TciVarDeclaration) then
|
|
begin
|
|
FirstColumn := FormatFirstColumn('variable');
|
|
c[0] := TciVarName;
|
|
c[2] := TciTypeKind;
|
|
end
|
|
else if (Item is TciConstantDeclaration) then
|
|
begin
|
|
FirstColumn := FormatFirstColumn('constant');
|
|
c[0] := TciConstantName;
|
|
c[1] := TciExpression;
|
|
end
|
|
else if (Item is TciLabelDeclaration) then
|
|
begin
|
|
FirstColumn := FormatFirstColumn('label');
|
|
c[0] := TciLabelName;
|
|
end
|
|
else if (Item is TciTypeDeclaration) then
|
|
begin
|
|
FirstColumn := FormatFirstColumn('type');
|
|
c[0] := TciTypeName;
|
|
c[2] := TciTypeKind;
|
|
end
|
|
else if (Item is TciClassField) then
|
|
begin
|
|
FirstColumn := FormatFirstColumn('variable');
|
|
c[0] := TciFieldName;
|
|
c[2] := TciTypeKind;
|
|
end
|
|
else if (Item is TciClassProperty) then
|
|
begin
|
|
FirstColumn := FormatFirstColumn('property');
|
|
c[0] := TciFieldName;
|
|
c[2] := TciTypeKind;
|
|
end
|
|
else if (Item is TciConstParameter) or
|
|
(Item is TciOutParameter) or
|
|
(Item is TciFormalParameter) or
|
|
(Item is TciInParameter) or
|
|
(Item is TciVarParameter) then
|
|
begin
|
|
FirstColumn := FormatFirstColumn('param');
|
|
c[0] := TciParameterName;
|
|
c[2] := TciParameterType;
|
|
end;
|
|
|
|
if (c[0] = nil) then
|
|
Exit;
|
|
|
|
a := Item.Items.GetItemsOfClass(c[0]);
|
|
if (c[1] <> nil) then
|
|
b[1] := Item.Items.GetFirstItemOfClass(c[1])
|
|
else
|
|
b[1] := nil;
|
|
if (c[2] <> nil) then
|
|
begin
|
|
b[2] := Item.Items.GetFirstItemOfClass(c[2]);
|
|
if (b[2] is TciTypeKind) and (b[2].Items.Count > 0) and (not (b[2].Items[0] is TciArrayType)) then
|
|
b[2] := b[2].Items[0];
|
|
end
|
|
else
|
|
b[2] := nil;
|
|
for i := Low(a) to High(a) do
|
|
begin
|
|
n := a[i].ShortText;
|
|
|
|
{$IFDEF ciCHECKDUPLICATES}
|
|
if (InsertList.IndexOf(n) > -1) then
|
|
Continue;
|
|
{$ENDIF}
|
|
s := FirstColumn + FormatMainName(n);
|
|
if (Item is TciClassProperty) then
|
|
s := s + FormatMainExtra(PropertyIndex(TciClassProperty(Item)));
|
|
if (b[1] <> nil) then
|
|
s := s + FormatMainExtra(' = ' + b[1].ShortText);
|
|
if (b[2] <> nil) then
|
|
if (b[2] is TciEnumType) then
|
|
s := s + ': ' + FormatTypeName(GetEnumTypeName(b[2]))
|
|
else
|
|
s := s + ': ' + FormatTypeName(b[2].ShortText)
|
|
{else
|
|
s := s + ';'};
|
|
|
|
ItemList.Insert(InsertList.Add(n), s);
|
|
if (b[2] <> nil) and (b[2].Items.Count > 0) then
|
|
AddEnums(b[2], ItemList, InsertList);
|
|
end;
|
|
end;
|
|
|
|
procedure TCodeInsight.FillProposal;
|
|
var
|
|
i: Integer;
|
|
begin
|
|
if (not Proposal_Filled) then
|
|
begin
|
|
Proposal_ItemList.BeginUpdate;
|
|
Proposal_InsertList.BeginUpdate;
|
|
|
|
try
|
|
for i := 0 to Items.Count - 1 do
|
|
Proposal_AddDeclaration(Items[i], Proposal_ItemList, Proposal_InsertList);
|
|
Proposal_Filled := True;
|
|
finally
|
|
Proposal_ItemList.EndUpdate;
|
|
Proposal_InsertList.EndUpdate;
|
|
end;
|
|
end;
|
|
end;
|
|
|
|
procedure TCodeInsight.FillSynCompletionProposal(ItemList, InsertList: TStrings; Prefix: string = '');
|
|
|
|
procedure AddFile(Item: TCodeInsight; ItemList, InsertList: TStrings);
|
|
var
|
|
i: Integer;
|
|
begin
|
|
if (item = nil) or (ItemList = nil) or (InsertList = nil) or (Item.Proposal_InsertList = nil) or (Item.Proposal_ItemList = nil) then
|
|
exit;
|
|
if (not Item.Proposal_Filled) then
|
|
Item.FillProposal;
|
|
|
|
{$IFNDEF ciCHECKDUPLICATES}
|
|
ItemList.AddStrings(Item.Proposal_ItemList);
|
|
InsertList.AddStrings(Item.Proposal_InsertList);
|
|
{$ELSE}
|
|
for i := 0 to Item.Proposal_InsertList.Count - 1 do
|
|
if (InsertList.IndexOf(Item.Proposal_InsertList[i]) = -1) then
|
|
begin
|
|
ItemList.Insert(InsertList.Add(Item.Proposal_InsertList[i]), Item.Proposal_ItemList[i]);
|
|
//ItemList.Add(Item.Proposal_ItemList[i]);
|
|
end;
|
|
{$ENDIF}
|
|
|
|
//AddEnums(Item, ItemList, InsertList);
|
|
for i := Low(Item.Includes) to High(Item.Includes) do
|
|
AddFile(Item.Includes[i], ItemList, InsertList);
|
|
end;
|
|
|
|
var
|
|
i, ii: Integer;
|
|
d: TDeclaration;
|
|
begin
|
|
ItemList.BeginUpdate;
|
|
InsertList.BeginUpdate;
|
|
try
|
|
ItemList.Clear;
|
|
InsertList.Clear;
|
|
|
|
if (PrepareString(Prefix) <> '') then
|
|
begin
|
|
d := FindVarBase(Prefix, True, vbType);
|
|
if (d <> nil) then
|
|
for i := 0 to d.Items.Count - 1 do
|
|
Proposal_AddDeclaration(d.Items[i], ItemList, InsertList);
|
|
end
|
|
else
|
|
begin
|
|
for i := High(InWith) downto Low(InWith) do
|
|
if (i <> InClassFunction) then
|
|
for ii := 0 to InWith[i].Items.Count - 1 do
|
|
Proposal_AddDeclaration(InWith[i].Items[ii], ItemList, InsertList);
|
|
|
|
for i := Low(InFunc) to High(InFunc) do
|
|
for ii := 0 to InFunc[i].Items.Count - 1 do
|
|
Proposal_AddDeclaration(InFunc[i].Items[ii], ItemList, InsertList);
|
|
|
|
if (InClassFunction <> -1) then
|
|
for i := 0 to InWith[InClassFunction].Items.Count - 1 do
|
|
Proposal_AddDeclaration(InWith[InClassFunction].Items[i], ItemList, InsertList);
|
|
|
|
AddFile(Self, ItemList, InsertList);
|
|
for i := 0 to High(CoreBuffer) do
|
|
AddFile(CoreBuffer[i], ItemList, InsertList);
|
|
end;
|
|
finally
|
|
ItemList.EndUpdate;
|
|
InsertList.EndUpdate;
|
|
end;
|
|
end;
|
|
|
|
initialization
|
|
{nothing}
|
|
CoreDefines := TStringList.Create;
|
|
finalization
|
|
ClearIncludeBuffer;
|
|
ClearCoreBuffer;
|
|
CoreDefines.Free;
|
|
|
|
end.
|