lazarus/components/synedit/syneditautocomplete.pp

677 lines
20 KiB
ObjectPascal

{-------------------------------------------------------------------------------
The contents of this file are subject to the Mozilla Public License
Version 1.1 (the "License"); you may not use this file except in compliance
with the License. You may obtain a copy of the License at
http://www.mozilla.org/MPL/
Software distributed under the License is distributed on an "AS IS" basis,
WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License for
the specific language governing rights and limitations under the License.
The Original Code is: SynEditAutoComplete.pas, released 2000-06-25.
The Initial Author of the Original Code is Michael Hieke.
Portions written by Michael Hieke are Copyright 2000 Michael Hieke.
Portions written by Cyrille de Brebisson (from mwCompletionProposal.pas) are
Copyright 1999 Cyrille de Brebisson.
All Rights Reserved.
Contributors to the SynEdit and mwEdit projects are listed in the
Contributors.txt file.
Alternatively, the contents of this file may be used under the terms of the
GNU General Public License Version 2 or later (the "GPL"), in which case
the provisions of the GPL are applicable instead of those above.
If you wish to allow use of your version of this file only under the terms
of the GPL and not to allow others to use your version of this file
under the MPL, indicate your decision by deleting the provisions above and
replace them with the notice and other provisions required by the GPL.
If you do not delete the provisions above, a recipient may use your version
of this file under either the MPL or the GPL.
$Id$
You may retrieve the latest version of this file at the SynEdit home page,
located at http://SynEdit.SourceForge.net
Known Issues:
-------------------------------------------------------------------------------}
unit SynEditAutoComplete;
{$I synedit.inc}
interface
uses
Classes, SysUtils, StrUtils,
LCLIntf, LCLType, Controls,
LazStringUtils, LazUTF8,
SynEdit, SynEditKeyCmds, SynEditPlugins;
const
CodeTemplateMacroMagic = '$(EnableMakros)';
CodeTemplateEnableMacros = 'EnableMakros';
CodeTemplateKeepSubIndent = 'KeepSubIndent';
CodeTemplateAttributesStartMagic = '$(AttributesStart)';
CodeTemplateAttributesEndMagic = '$(AttributesEnd)';
type
TCustomSynAutoComplete = class;
TOnTokenNotFound = procedure(Sender: TObject; AToken: string;
AEditor: TCustomSynEdit; var Index:integer) of object;
TOnExecuteCompletion = procedure(ASynAutoComplete: TCustomSynAutoComplete;
Index: integer) of object;
// Data for a single template item.
{ TTemplate }
TTemplate = class
private
fKey: string;
fValue: string;
fComment: string;
fAttributes: TStringList; // List of attributes.
public
constructor Create;
constructor Create(aTemplate: TTemplate);
constructor Create(aKey, aValue, aComment: string);
destructor Destroy; override;
procedure SetBooleanAttribute(aName: string; aValue: Boolean);
procedure SetValueWithoutLastEOL(aValue: string);
property Key: string read fKey write fKey;
property Value: string read fValue write fValue;
property Comment: string read fComment write fComment;
property Attributes: TStringList read fAttributes;
end;
{ TTemplateList }
TTemplateList = class
private
fList: TStringListUTF8Fast;
function GetSorted: Boolean;
function GetTemplate(Index: Integer): TTemplate;
procedure SetSorted(AValue: Boolean);
procedure SetTemplate(Index: Integer; AValue: TTemplate);
public
constructor Create;
destructor Destroy; override;
function Add(aTemplate: TTemplate): Integer;
function ByKey(aKey: string): TTemplate;
function Count: Integer;
procedure Clear;
procedure Delete(Index: Integer);
function IndexOf(aKey: string): Integer;
procedure Sort;
public
property Sorted: Boolean read GetSorted write SetSorted;
property Templates[Index: Integer]: TTemplate read GetTemplate
write SetTemplate; default;
end;
{ TCustomSynAutoComplete }
TCustomSynAutoComplete = class(TLazSynMultiEditPlugin)
private
fOnTokenNotFound: TOnTokenNotFound;
fIndentToTokenStart: boolean;
FOnExecuteCompletion: TOnExecuteCompletion;
protected
fCodeTemplSource: TStrings;
fCodeTemplates: TTemplateList;
fEditor: TCustomSynEdit;
fEditors: TList;
fEOTokenChars: string;
fCaseSensitive: boolean;
fParsed: boolean;
procedure CompletionListChanged(Sender: TObject);
function GetCodeTemplates: TTemplateList;
procedure ParseCompletionList; virtual;
procedure SetCodeTemplSource(Value: TStrings); virtual;
procedure SynEditCommandHandler(Sender: TObject; AfterProcessing: boolean;
var Handled: boolean; var Command: TSynEditorCommand;
var AChar: TUTF8Char;
Data: pointer; HandlerData: pointer);
procedure DoEditorAdded(AValue: TCustomSynEdit); override;
procedure DoEditorRemoving(AValue: TCustomSynEdit); override;
public
constructor Create(AOwner: TComponent); override;
destructor Destroy; override;
procedure AddCompletion(ATemplate: TTemplate);
procedure AddCompletion(AToken, AValue, AComment: string);
procedure Execute(AEditor: TCustomSynEdit); virtual;
procedure ExecuteCompletion(AToken: string; AEditor: TCustomSynEdit); virtual;
public
property CodeTemplSource: TStrings read fCodeTemplSource write SetCodeTemplSource;
property CaseSensitive: boolean read fCaseSensitive write fCaseSensitive;
property CodeTemplates: TTemplateList read GetCodeTemplates;
property EndOfTokenChr: string read fEOTokenChars write fEOTokenChars;
property OnTokenNotFound: TOnTokenNotFound read fOnTokenNotFound write fOnTokenNotFound;
property IndentToTokenStart: boolean read fIndentToTokenStart write fIndentToTokenStart;
property OnExecuteCompletion: TOnExecuteCompletion read FOnExecuteCompletion
write FOnExecuteCompletion;
end;
{ TSynEditAutoComplete }
TSynEditAutoComplete = class(TCustomSynAutoComplete)
published
property CodeTemplSource;
property CaseSensitive;
property Editor;
property EndOfTokenChr;
property OnTokenNotFound;
property IndentToTokenStart;
property OnExecuteCompletion;
end;
implementation
{ TTemplate }
constructor TTemplate.Create;
begin
inherited Create;
fAttributes := TStringListUTF8Fast.Create;
fAttributes.UseLocale := False;
end;
constructor TTemplate.Create(aTemplate: TTemplate);
begin // A copy constructor
Create;
fKey := aTemplate.fKey;
fValue := aTemplate.fValue;
fComment := aTemplate.fComment;
fAttributes.Assign(aTemplate.Attributes);
end;
constructor TTemplate.Create(aKey, aValue, aComment: string);
begin
Create;
fKey := aKey;
fValue := aValue;
fComment := aComment;
end;
destructor TTemplate.Destroy;
begin
fAttributes.Free;
inherited Destroy;
end;
procedure TTemplate.SetBooleanAttribute(aName: string; aValue: Boolean);
var
i: Integer;
begin
if aValue then
fAttributes.Values[aName] := 'true'
else begin
i := fAttributes.IndexOfName(aName);
if i >= 0 then
fAttributes.Delete(i);
end;
end;
procedure TTemplate.SetValueWithoutLastEOL(aValue: string);
var
l: Integer;
begin // Remove last EOL from Value.
if aValue <> '' then begin
l := Length(aValue);
if aValue[l] in [#10,#13] then begin
Dec(l);
if (l > 0) and (aValue[l] in [#10,#13])
and (aValue[l] <> aValue[l+1]) then
Dec(l);
SetLength(aValue, l);
end;
end;
fValue := aValue;
end;
{ TTemplateList }
constructor TTemplateList.Create;
begin
inherited Create;
fList := TStringListUTF8Fast.Create;
fList.OwnsObjects := True;
end;
destructor TTemplateList.Destroy;
begin
fList.Free;
inherited Destroy;
end;
function TTemplateList.Add(aTemplate: TTemplate): Integer;
begin
Result := fList.AddObject(aTemplate.fKey, aTemplate);
end;
function TTemplateList.ByKey(aKey: string): TTemplate;
var
i: Integer;
begin
i := fList.IndexOf(aKey);
if i >= 0 then
Result := TTemplate(fList.Objects[i])
else
Result := Nil;
end;
function TTemplateList.Count: Integer;
begin
Result := fList.Count;
end;
procedure TTemplateList.Clear;
begin
fList.Clear;
end;
procedure TTemplateList.Delete(Index: Integer);
begin
fList.Delete(Index);
end;
function TTemplateList.IndexOf(aKey: string): Integer;
begin
Result := fList.IndexOf(aKey);
end;
procedure TTemplateList.Sort;
begin
fList.Sort;
end;
function TTemplateList.GetSorted: Boolean;
begin
Result := fList.Sorted;
end;
function TTemplateList.GetTemplate(Index: Integer): TTemplate;
begin
Result := TTemplate(fList.Objects[Index]);
end;
procedure TTemplateList.SetSorted(AValue: Boolean);
begin
fList.Sorted := AValue;
end;
procedure TTemplateList.SetTemplate(Index: Integer; AValue: TTemplate);
begin
fList.Objects[Index] := AValue;
end;
{ TCustomSynAutoComplete }
procedure TCustomSynAutoComplete.AddCompletion(ATemplate: TTemplate);
var
NewTemplate: TTemplate;
begin
Assert(ATemplate.Key<>'', 'TCustomSynAutoComplete.AddCompletion: Key is empty.');
if not fParsed then
ParseCompletionList;
NewTemplate := TTemplate.Create(ATemplate);
fCodeTemplates.Add(NewTemplate);
end;
procedure TCustomSynAutoComplete.AddCompletion(AToken, AValue, AComment: string);
var
NewTemplate: TTemplate;
begin
Assert(AToken<>'', 'TCustomSynAutoComplete.AddCompletion: Key is empty.');
if not fParsed then
ParseCompletionList;
NewTemplate := TTemplate.Create(AToken, AValue, AComment);
fCodeTemplates.Add(NewTemplate);
end;
procedure TCustomSynAutoComplete.CompletionListChanged(Sender: TObject);
begin
fParsed := FALSE;
end;
constructor TCustomSynAutoComplete.Create(AOwner: TComponent);
begin
inherited Create(AOwner);
fCodeTemplSource := TStringList.Create;
TStringList(fCodeTemplSource).OnChange := @CompletionListChanged;
fCodeTemplates := TTemplateList.Create;
fEditors := TList.Create;
fEOTokenChars := '()[]{}.';
end;
destructor TCustomSynAutoComplete.Destroy;
begin
fEditors.Free;
fCodeTemplates.Free;
fCodeTemplSource.Free;
inherited Destroy;
end;
procedure TCustomSynAutoComplete.Execute(AEditor: TCustomSynEdit);
var
s: string;
i, j: integer;
begin
if AEditor <> nil then begin
// get token
s := AEditor.LineText;
j := AEditor.LogicalCaretXY.x;
i := j - 1;
if i <= Length(s) then begin
while (i > 0) and (s[i] > ' ') and (Pos(s[i], fEOTokenChars) = 0) do
Dec(i);
Inc(i);
s := Copy(s, i, j - i);
ExecuteCompletion(s, AEditor);
end else begin
i:=-1;
if Assigned(OnTokenNotFound) then
OnTokenNotFound(Self,'',AEditor,i);
if i>=0 then
ExecuteCompletion(fCodeTemplates[i].Key, AEditor);
end;
end;
end;
procedure TCustomSynAutoComplete.ExecuteCompletion(AToken: string;
AEditor: TCustomSynEdit);
var
i, j, Len, IndentLen, TokenStartX: integer;
s: string;
IdxMaybe, NumMaybe: integer;
p, p2: TPoint;
NewCaretPos: boolean;
Temp: TStringList;
Template: TTemplate;
begin
if not fParsed then
ParseCompletionList;
Len := Length(AToken);
if (Len=0) and Assigned(OnTokenNotFound) then
begin
i := 0;
OnTokenNotFound(Self,AToken,AEditor,i);
end;
if (Len > 0) and (AEditor <> nil) and not AEditor.ReadOnly
and (fCodeTemplates.Count > 0)
then begin
// find completion for this token - not all chars necessary if unambiguous
i := fCodeTemplates.Count - 1;
IdxMaybe := -1;
NumMaybe := 0;
if fCaseSensitive then begin
while i > -1 do begin
s := fCodeTemplates[i].Key;
if s = AToken then
break
else if LazStartsStr(AToken, s) then begin
Inc(NumMaybe);
IdxMaybe := i;
end;
Dec(i);
end;
end else begin
while i > -1 do begin
s := fCodeTemplates[i].Key;
if AnsiCompareText(s, AToken) = 0 then
break
else if AnsiCompareText(Copy(s, 1, Len), AToken) = 0 then begin
Inc(NumMaybe);
IdxMaybe := i;
end;
Dec(i);
end;
end;
if (i = -1) and (NumMaybe = 1) then
i := IdxMaybe;
if (i < 0) and Assigned(fOnTokenNotFound) then
fOnTokenNotFound(Self,AToken,AEditor,i);
if i > -1 then begin
// select token in editor
p := AEditor.LogicalCaretXY;
if Assigned(OnExecuteCompletion) then
OnExecuteCompletion(Self,i)
else begin
AEditor.BeginUpdate;
try
TokenStartX:=p.x;
s:=AEditor.Lines[p.y-1];
if TokenStartX>length(s) then TokenStartX:=length(s);
while (TokenStartX > 1) and (s[TokenStartX-1] > ' ')
and (Pos(s[TokenStartX-1], fEOTokenChars) = 0) do
Dec(TokenStartX);
AEditor.BlockBegin := Point(TokenStartX, p.y);
AEditor.BlockEnd := p;
// indent the completion string if necessary, determine the caret pos
if IndentToTokenStart then begin
IndentLen := p.x - Len - 1;
end else begin
// indent the same as the first line
IndentLen:=1;
if (p.y>0) and (p.y<=AEditor.Lines.Count) then begin
s:=AEditor.Lines[p.y-1];
while (IndentLen<p.x)
and ((IndentLen>length(s)) or (s[IndentLen]<=' ')) do
inc(IndentLen);
end;
dec(IndentLen);
end;
p := AEditor.BlockBegin;
NewCaretPos := FALSE;
Temp := TStringList.Create;
try
Template := fCodeTemplates[i];
s:=Template.fValue;
Temp.Text := s;
if (s<>'') and (s[length(s)] in [#10,#13]) then
Temp.Add('');
// indent lines
if (IndentLen > 0) and (Temp.Count > 1) then
begin
s := StringOfChar(' ', IndentLen);
for i := 1 to Temp.Count - 1 do
Temp[i] := s + Temp[i];
end;
// find first '|' and use it as caret position
for i := 0 to Temp.Count - 1 do
begin
s := Temp[i];
j := Pos('|', s);
if j > 0 then
begin
Delete(s, j, 1);
Temp[i] := s;
// if j > 1 then
// Dec(j);
NewCaretPos := TRUE;
Inc(p.y, i);
if i = 0 then
// Inc(p.x, j)
Inc(p.x, j - 1)
else
p.x := j;
break;
end;
end;
s := Temp.Text;
// strip the trailing #13#10 that was appended by the stringlist
i := Length(s);
if (i>=1) and (s[i] in [#10,#13]) then begin
dec(i);
if (i>=1) and (s[i] in [#10,#13]) and (s[i]<>s[i+1]) then
dec(i);
SetLength(s, i);
end;
finally
Temp.Free;
end;
// replace the selected text and position the caret
p2 := AEditor.BlockBegin;
AEditor.SelText := '';
AEditor.SetTextBetweenPoints(p2, p2, s, [], scamEnd);
if NewCaretPos then
AEditor.LogicalCaretXY := p;
AEditor.EnsureCursorPosVisible;
finally
AEditor.EndUpdate;
end;
end;
end;
end;
end;
function TCustomSynAutoComplete.GetCodeTemplates: TTemplateList;
begin
if not fParsed then
ParseCompletionList;
Result := fCodeTemplates;
end;
procedure TCustomSynAutoComplete.ParseCompletionList;
var
sKey, sValue, sComment: string;
procedure SaveEntry;
var
StartI, EndI: Integer;
Template: TTemplate;
begin
Template := TTemplate.Create;
Template.Key := sKey;
Template.Comment := sComment;
Assert(not LazStartsStr(CodeTemplateMacroMagic, sValue), 'SaveEntry: Found '+CodeTemplateMacroMagic);
if LazStartsStr(CodeTemplateAttributesStartMagic, sValue) then
begin
// Start of attributes
StartI := Length(CodeTemplateAttributesStartMagic) + 1;
while (StartI <= Length(sValue)) and (sValue[StartI] in [#10,#13]) do
Inc(StartI);
EndI := PosEx(CodeTemplateAttributesEndMagic, sValue, StartI);
if EndI = 0 then
raise Exception.Create('TCustomSynAutoComplete.ParseCompletionList: "'
+ CodeTemplateAttributesEndMagic + '" not found.');
Template.Attributes.Text := Copy(sValue, StartI, EndI-StartI);
// Start of value
StartI := EndI + Length(CodeTemplateAttributesEndMagic);
while (StartI <= Length(sValue)) and (sValue[StartI] in [#10,#13]) do
Inc(StartI);
Template.Value := Copy(sValue, StartI, Length(sValue));
end;
fCodeTemplates.Add(Template);
sKey := '';
sValue := '';
sComment := '';
end;
var
BorlandDCI: boolean;
i, j, Len: integer;
S: string;
TemplateStarted: Boolean;
begin
fCodeTemplates.Clear;
sKey := '';
sValue := '';
sComment := '';
if fCodeTemplSource.Count > 0 then begin
S := fCodeTemplSource[0];
BorlandDCI := (S <> '') and (S[1] = '[');
TemplateStarted:=false;
for i := 0 to fCodeTemplSource.Count - 1 do begin
S := fCodeTemplSource[i];
Len := Length(S);
if BorlandDCI then begin
// the style of the Delphi32.dci file
if (Len > 0) and (S[1] = '[') then begin
// save last entry
if sKey <> '' then
SaveEntry;
// new completion entry
j := 2;
while (j <= Len) and (S[j] > ' ') do
Inc(j);
sKey := Copy(S, 2, j - 2);
// start of comment in DCI file
while (j <= Len) and (S[j] <= ' ') do
Inc(j);
if (j <= Len) and (S[j] = '|') then
Inc(j);
while (j <= Len) and (S[j] <= ' ') do
Inc(j);
sComment := Copy(S, j, Len);
if sComment[Length(sComment)] = ']' then
SetLength(sComment, Length(sComment) - 1);
TemplateStarted:=true;
end else begin
if not TemplateStarted then
sValue := sValue + LineEnding;
TemplateStarted:=false;
sValue := sValue + S;
end;
end else begin
// the original style
if (Len > 0) and (S[1] <> '=') then begin
// save last entry
if sKey <> '' then
SaveEntry;
// new completion entry
sKey := S;
TemplateStarted:=true;
end else if (Len > 0) and (S[1] = '=') then begin
if not TemplateStarted then
sValue := sValue + LineEnding;
TemplateStarted:=false;
sValue := sValue + Copy(S, 2, Len);
end;
end;
end;
if sKey <> '' then
SaveEntry;
end;
fCodeTemplates.Sorted := True;
fParsed:=true;
end;
procedure TCustomSynAutoComplete.SetCodeTemplSource(Value: TStrings);
begin
fCodeTemplSource.Assign(Value);
fParsed := FALSE;
end;
procedure TCustomSynAutoComplete.SynEditCommandHandler(Sender: TObject;
AfterProcessing: boolean; var Handled: boolean;
var Command: TSynEditorCommand;
var AChar: TUTF8Char; Data: pointer;
HandlerData: pointer);
begin
if not AfterProcessing and not Handled and (Command = ecAutoCompletion) then
begin
Handled := TRUE;
Execute(Sender as TCustomSynEdit);
end;
end;
procedure TCustomSynAutoComplete.DoEditorAdded(AValue: TCustomSynEdit);
begin
inherited DoEditorAdded(AValue);
AValue.RegisterCommandHandler(@SynEditCommandHandler, nil);
end;
procedure TCustomSynAutoComplete.DoEditorRemoving(AValue: TCustomSynEdit);
begin
inherited DoEditorRemoving(AValue);
AValue.UnregisterCommandHandler(@SynEditCommandHandler);
end;
end.