lazarus/ide/codetemplatesdlg.pas

1359 lines
41 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., 51 Franklin Street - Fifth Floor, Boston, MA 02110-1335, USA. *
* *
***************************************************************************
Author: Mattias Gaertner
Abstract:
A dialog for adding and editing code templates
}
unit CodeTemplatesDlg;
{$mode objfpc}{$H+}
interface
uses
Classes, SysUtils, RegExpr,
// LCL
Forms, Controls, Dialogs, ClipBrd, StdCtrls, ExtCtrls, Menus,
ButtonPanel, EditBtn, LCLType,
// LazUtils
FileUtil, LazFileUtils, LazLoggerBase, LazStringUtils, LazUtilities, LazTracer,
LazUTF8,
// synedit
SynEdit, SynHighlighterPas, SynEditAutoComplete, SynEditTypes,
// codetools
CodeToolManager, CodeCache, KeywordFuncLists, BasicCodeTools, PascalParserTool,
// IdeUtils
IdeUtilsPkgStrConsts,
// IDEIntf
SrcEditorIntf, MenuIntf, IDEWindowIntf, LazIDEIntf, IDEHelpIntf, IDEDialogs,
// IDE
IdeIntfStrConsts, LazarusIDEStrConsts, EditorOptions, CodeMacroSelect, CodeMacroPrompt;
type
TAutoCompleteOption = (
acoLineBreak,
acoSpace,
acoTab,
acoWordEnd,
acoIgnoreForSelection,
acoRemoveChar
);
const
AutoCompleteOptionNames: array[TAutoCompleteOption] of shortstring = (
'AutoOnLineBreak',
'AutoOnSpace',
'AutoOnTab',
'AutoOnWordEnd',
'IgnoreForSelection',
'RemoveChar' // do not add the typed character
);
type
{ TCodeTemplateDialog }
TCodeTemplateDialog = class(TForm)
AddButton: TButton;
ASynPasSyn: TSynFreePascalSyn;
AutoOnOptionsCheckGroup: TCheckGroup;
ButtonPanel: TButtonPanel;
EditTemplateGroupBox: TGroupBox;
FilenameEdit: TFileNameEdit;
InsertMacroButton: TButton;
KeepSubIndentCheckBox: TCheckBox;
OptionsPanel: TPanel;
ControlPanel: TPanel;
Splitter1: TSplitter;
UseMacrosCheckBox: TCheckBox;
RenameButton: TButton;
DeleteButton: TButton;
TemplateListBox: TListBox;
TemplateSynEdit: TSynEdit;
TemplatesGroupBox: TGroupBox;
FilenameGroupBox: TGroupBox;
MainPopupMenu: TPopupMenu;
procedure AddButtonClick(Sender: TObject);
procedure DeleteButtonClick(Sender: TObject);
procedure FormKeyDown(Sender: TObject; var Key: Word; Shift: TShiftState);
procedure FormShow(Sender: TObject);
procedure RenameButtonClick(Sender: TObject);
procedure FormClose(Sender: TObject; var {%H-}CloseAction: TCloseAction);
procedure FormCreate(Sender: TObject);
procedure HelpButtonClick(Sender: TObject);
procedure InsertMacroButtonClick(Sender: TObject);
procedure OkButtonClick(Sender: TObject);
procedure TemplateListBoxSelectionChange(Sender: TObject; {%H-}User: boolean);
procedure UseMacrosCheckBoxChange(Sender: TObject);
private
SynAutoComplete: TSynEditAutoComplete;
LastTemplate: integer;
procedure CopyMenuItem(Sender: TObject);
procedure CutMenuItem(Sender: TObject);
procedure InsertMacroMenuItem(Sender: TObject);
procedure PasteMenuItem(Sender: TObject);
procedure BuildPopupMenu;
procedure DoInsertMacro;
public
procedure FillCodeTemplateListBox;
procedure ShowCurCodeTemplate;
procedure SaveCurCodeTemplate;
end;
{ TLazCodeMacros }
TLazCodeMacros = class(TIDECodeMacros)
private
FItems: TFPList; // list of TIDECodeMacro
protected
function GetItems(Index: integer): TIDECodeMacro; override;
public
constructor Create;
destructor Destroy; override;
procedure Clear;
property Items[Index: integer]: TIDECodeMacro read GetItems; default;
function Count: integer; override;
function Add(Macro: TIDECodeMacro): integer; override;
function FindByName(const AName: string): TIDECodeMacro; override;
function CreateUniqueName(const AName: string): string; override;
end;
function ShowCodeTemplateDialog: TModalResult;
function AddCodeTemplate(ASynAutoComplete: TSynEditAutoComplete;
var AToken, AComment: string): TModalResult;
function EditCodeTemplate(ASynAutoComplete: TSynEditAutoComplete;
AIndex: integer): TModalResult;
procedure CreateStandardCodeMacros;
// standard code macros
function CodeMacroUpper(const Parameter: string; {%H-}InteractiveValue: TPersistent;
{%H-}SrcEdit: TSourceEditorInterface;
var Value, {%H-}ErrorMsg: string): boolean;
function CodeMacroLower(const Parameter: string; {%H-}InteractiveValue: TPersistent;
{%H-}SrcEdit: TSourceEditorInterface;
var Value, {%H-}ErrorMsg: string): boolean;
function CodeMacroPaste(const {%H-}Parameter: string; {%H-}InteractiveValue: TPersistent;
{%H-}SrcEdit: TSourceEditorInterface;
var Value, {%H-}ErrorMsg: string): boolean;
function CodeMacroProcedureHead(const Parameter: string;
{%H-}InteractiveValue: TPersistent;
SrcEdit: TSourceEditorInterface;
var Value, ErrorMsg: string): boolean;
function CodeMacroProcedureName(const {%H-}Parameter: string;
InteractiveValue: TPersistent;
SrcEdit: TSourceEditorInterface;
var Value, ErrorMsg: string): boolean;
function CodeMacroDate(const Parameter: string; {%H-}InteractiveValue: TPersistent;
{%H-}SrcEdit: TSourceEditorInterface;
var Value, {%H-}ErrorMsg: string): boolean;
function CodeMacroTime(const Parameter: string; {%H-}InteractiveValue: TPersistent;
{%H-}SrcEdit: TSourceEditorInterface;
var Value, {%H-}ErrorMsg: string): boolean;
function CodeMacroDateTime(const Parameter: string; {%H-}InteractiveValue: TPersistent;
{%H-}SrcEdit: TSourceEditorInterface;
var Value, {%H-}ErrorMsg: string): boolean;
function CodeMacroAddMissingEnd(const {%H-}Parameter: string;
{%H-}InteractiveValue: TPersistent;
SrcEdit: TSourceEditorInterface;
var Value, {%H-}ErrorMsg: string): boolean;
function CodeMacroAddSemicolon(const {%H-}Parameter: string;
{%H-}InteractiveValue: TPersistent;
SrcEdit: TSourceEditorInterface;
var Value, {%H-}ErrorMsg: string): boolean;
function CodeMacroOfAll(const {%H-}Parameter: string; {%H-}InteractiveValue: TPersistent;
SrcEdit: TSourceEditorInterface;
var Value, ErrorMsg: string): boolean;
function CodeMacroPrevWord(const Parameter: string;
{%H-}InteractiveValue: TPersistent;
SrcEdit: TSourceEditorInterface;
var Value, {%H-}ErrorMsg: string): boolean;
function CodeMacroWordAtCursor(const {%H-}Parameter: string; {%H-}InteractiveValue: TPersistent;
SrcEdit: TSourceEditorInterface;
var Value, {%H-}ErrorMsg: string): boolean;
const
CodeTemplatesMenuRootName = 'CodeTemplates';
var
CodeTemplateCopyIDEMenuCommand: TIDEMenuCommand;
CodeTemplateCutIDEMenuCommand: TIDEMenuCommand;
CodeTemplatePasteIDEMenuCommand: TIDEMenuCommand;
CodeTemplateInsertMacroIDEMenuCommand: TIDEMenuCommand;
procedure RegisterStandardCodeTemplatesMenuItems;
implementation
{$R *.lfm}
function ShowCodeTemplateDialog: TModalResult;
var
CodeTemplateDialog: TCodeTemplateDialog;
begin
CodeTemplateDialog:=TCodeTemplateDialog.Create(nil);
Result:=CodeTemplateDialog.ShowModal;
CodeTemplateDialog.Free;
end;
function IsCodeTemplateOk(ASynAutoComplete: TSynEditAutoComplete;
const AToken: string; ASkipExistCheck: boolean = false): boolean;
var
i: integer;
begin
result := true;
// empty
if AToken = '' then
begin
IDEMessageDialog(lisCodeTemplError, lisCodeTemplErrorEmptyName, mtError, [mbOK]);
exit(false);
end;
// exists
if not ASkipExistCheck then
if ASynAutoComplete.CodeTemplates.ByKey(AToken) <> Nil then
begin
IDEMessageDialog(lisCodeTemplError, lisCodeTemplErrorAlreadyExists, mtError, [mbOK]);
exit(false);
end;
// first symbol
if AToken[1] in ['0'..'9'] then
result := false;
// all symbols
i := 1;
while (i <= length(AToken)) and (result = true) do
begin
if not (AToken[i] in ['0'..'9', 'A'..'Z', 'a'..'z', '_']) then
result := false;
inc(i);
end;
if result = false then
IDEMessageDialog(lisCodeTemplError, lisCodeTemplErrorInvalidName, mtError, [mbOK]);
end;
function AddCodeTemplate(ASynAutoComplete: TSynEditAutoComplete;
var AToken, AComment: string): TModalResult;
var
Str: array of string;
begin
Result:= mrCancel;
SetLength(Str{%H-}, 2);
Str[0]:= AToken;
Str[1]:= AComment;
if InputQuery(lisCodeTemplAddCodeTemplate,
[lisCodeTemplToken, lisCodeTemplComment], Str) then
if IsCodeTemplateOk(ASynAutoComplete, Str[0]) then
begin
Result:= mrOk;
AToken:= Str[0];
AComment:= Str[1];
end;
end;
function EditCodeTemplate(ASynAutoComplete: TSynEditAutoComplete;
AIndex: integer): TModalResult;
var
Template: TTemplate;
Str: array of string;
begin
Result:= mrCancel;
if (AIndex<0) or (AIndex>=ASynAutoComplete.CodeTemplates.Count) then exit;
Template := ASynAutoComplete.CodeTemplates[AIndex];
SetLength(Str{%H-}, 2);
Str[0] := Template.Key;
Str[1] := Template.Comment;
if not InputQuery(lisCodeTemplEditCodeTemplate,
[lisCodeTemplToken, lisCodeTemplComment], Str) then exit;
if not IsCodeTemplateOk(ASynAutoComplete, Str[0], true) then exit;
Template.Key := Str[0];
Template.Comment := Str[1];
Result:= mrOk;
end;
function CodeMacroUpper(const Parameter: string; InteractiveValue: TPersistent;
SrcEdit: TSourceEditorInterface;
var Value, ErrorMsg: string): boolean;
begin
Value:=UpperCase(Parameter);
Result:=true;
end;
function CodeMacroLower(const Parameter: string; InteractiveValue: TPersistent;
SrcEdit: TSourceEditorInterface;
var Value, ErrorMsg: string): boolean;
begin
Value:=LowerCase(Parameter);
Result:=true;
end;
function CodeMacroPaste(const Parameter: string; InteractiveValue: TPersistent;
SrcEdit: TSourceEditorInterface;
var Value, ErrorMsg: string): boolean;
begin
Value:=Clipboard.AsText;
Result:=true;
end;
function CodeMacroProcedureHead(const Parameter: string;
InteractiveValue: TPersistent; SrcEdit: TSourceEditorInterface; var Value,
ErrorMsg: string): boolean;
var
Params: TStrings;
Param: string;
i: Integer;
Attributes: TProcHeadAttributes;
CodeBuf: TCodeBuffer;
XY: TPoint;
p: integer;
StartPos: Integer;
begin
//debugln('CodeMacroProcedureHead A ',Parameter);
// parse attributes
Params:=SplitString(Parameter,',');
if Params<>nil then begin
try
Attributes:=[];
for i:=0 to Params.Count-1 do begin
Param:=Params[i];
if SysUtils.CompareText(Param,'WithStart')=0 then
Include(Attributes,phpWithStart)
else if SysUtils.CompareText(Param,'WithStart')=0 then
Include(Attributes,phpWithStart)
else if SysUtils.CompareText(Param,'WithoutClassKeyword')=0 then
Include(Attributes,phpWithoutClassKeyword)
else if SysUtils.CompareText(Param,'AddClassName')=0 then
Include(Attributes,phpAddClassName)
else if SysUtils.CompareText(Param,'WithoutClassName')=0 then
Include(Attributes,phpWithoutClassName)
else if SysUtils.CompareText(Param,'WithoutName')=0 then
Include(Attributes,phpWithoutName)
else if SysUtils.CompareText(Param,'WithoutParamList')=0 then
Include(Attributes,phpWithoutParamList)
else if SysUtils.CompareText(Param,'WithVarModifiers')=0 then
Include(Attributes,phpWithVarModifiers)
else if SysUtils.CompareText(Param,'WithParameterNames')=0 then
Include(Attributes,phpWithParameterNames)
else if SysUtils.CompareText(Param,'WithoutParamTypes')=0 then
Include(Attributes,phpWithoutParamTypes)
else if SysUtils.CompareText(Param,'WithDefaultValues')=0 then
Include(Attributes,phpWithDefaultValues)
else if SysUtils.CompareText(Param,'WithResultType')=0 then
Include(Attributes,phpWithResultType)
else if SysUtils.CompareText(Param,'WithOfObject')=0 then
Include(Attributes,phpWithOfObject)
else if SysUtils.CompareText(Param,'WithCallingSpecs')=0 then
Include(Attributes,phpWithCallingSpecs)
else if SysUtils.CompareText(Param,'WithProcModifiers')=0 then
Include(Attributes,phpWithProcModifiers)
else if SysUtils.CompareText(Param,'WithComments')=0 then
Include(Attributes,phpWithComments)
else if SysUtils.CompareText(Param,'InUpperCase')=0 then
Include(Attributes,phpInUpperCase)
else if SysUtils.CompareText(Param,'CommentsToSpace')=0 then
Include(Attributes,phpCommentsToSpace)
else if SysUtils.CompareText(Param,'WithoutBrackets')=0 then
Include(Attributes,phpWithoutBrackets)
else if SysUtils.CompareText(Param,'WithoutSemicolon')=0 then
Include(Attributes,phpWithoutSemicolon)
else begin
Result:=false;
ErrorMsg:='Unknown Option: "'+Param+'"';
exit;
end;
end;
finally
Params.Free;
end;
end;
//debugln('CodeMacroProcedureHead B ',dbgs(Attributes));
CodeBuf:=SrcEdit.CodeToolsBuffer as TCodeBuffer;
XY:=SrcEdit.CursorTextXY;
CodeBuf.LineColToPosition(XY.Y,XY.X,p);
if p>0 then begin
StartPos:=GetIdentStartPosition(CodeBuf.Source,p);
XY.X := XY.X + StartPos-p;
end;
if not CodeToolBoss.ExtractProcedureHeader(CodeBuf,XY.X,XY.Y,Attributes,Value)
then begin
Result:=false;
ErrorMsg:=CodeToolBoss.ErrorMessage;
LazarusIDE.DoJumpToCodeToolBossError;
exit;
end;
//debugln('CodeMacroProcedureHead C Value="',Value,'"');
Result:=true;
end;
function CodeMacroProcedureName(const Parameter: string;
InteractiveValue: TPersistent; SrcEdit: TSourceEditorInterface; var Value,
ErrorMsg: string): boolean;
begin
Result:=CodeMacroProcedureHead(
'WithoutParamList,WithoutBrackets,WithoutSemicolon',
InteractiveValue,SrcEdit,Value,ErrorMsg);
end;
function CodeMacroDate(const Parameter: string; InteractiveValue: TPersistent;
SrcEdit: TSourceEditorInterface; var Value, ErrorMsg: string): boolean;
begin
if Parameter<>'' then
Value:=FormatDateTime(Parameter,Now)
else
Value:=DateToStr(Now);
Result:=true;
end;
function CodeMacroTime(const Parameter: string; InteractiveValue: TPersistent;
SrcEdit: TSourceEditorInterface; var Value, ErrorMsg: string): boolean;
begin
if Parameter<>'' then
Value:=FormatDateTime(Parameter,Now)
else
Value:=TimeToStr(Now);
Result:=true;
end;
function CodeMacroDateTime(const Parameter: string;
InteractiveValue: TPersistent; SrcEdit: TSourceEditorInterface; var Value,
ErrorMsg: string): boolean;
begin
if Parameter<>'' then
Value:=FormatDateTime(Parameter,Now)
else
Value:=DateTimeToStr(Now);
Result:=true;
end;
function CodeMacroAddMissingEnd(const Parameter: string;
InteractiveValue: TPersistent; SrcEdit: TSourceEditorInterface; var Value,
ErrorMsg: string): boolean;
{ checks if at current position a block end should be inserted
Examples:
No block end required:
begin|
end
repeat|
until
begin|
repeat
Block end required:
begin
begin|
end;
}
var
Line: String;
p: TPoint;
CodeXYPos: TCodeXYPosition;
begin
Result:=true;
Value:='';
Line:=SrcEdit.CurrentLineText;
p:=SrcEdit.CursorTextXY;
if p.y<1 then exit;
CodeXYPos.X:=p.x;
CodeXYPos.Y:=p.y;
CodeXYPos.Code:=SrcEdit.CodeToolsBuffer as TCodeBuffer;
if CodeXYPos.Code=nil then exit;
// ToDo
while (p.y<=SrcEdit.LineCount) do begin
Line:=SrcEdit.Lines[p.y-1];
while (p.x<=length(Line)) do begin
if IsSpaceChar[Line[p.x]] then
inc(p.x)
else begin
if CompareIdentifiers(@Line[p.x],'end')=0 then begin
// has already an end
exit;
end else begin
// missing end
Value:=LineEnding+'end;'+LineEnding;
end;
end;
end;
inc(p.y);
p.x:=1;
end;
end;
function CodeMacroAddSemicolon(const Parameter: string;
InteractiveValue: TPersistent; SrcEdit: TSourceEditorInterface; var Value,
ErrorMsg: string): boolean;
var
XY: TPoint;
Code: TCodeBuffer;
p, AtomStart: integer;
Src: String;
begin
Result:=true;
Value:='';
XY:=SrcEdit.CursorTextXY;
if XY.y<1 then exit;
Code:=SrcEdit.CodeToolsBuffer as TCodeBuffer;
Code.LineColToPosition(XY.y,XY.x,p);
Src:=Code.Source;
ReadRawNextPascalAtom(Src,p,AtomStart,true,true);
if StringCase(copy(Src,AtomStart,p-AtomStart),
[ 'else', 'do', ';', ')', ']' ], True, False) >= 0 then
exit;
Value:=';';
end;
function CodeMacroOfAll(const Parameter: string; InteractiveValue: TPersistent;
SrcEdit: TSourceEditorInterface; var Value, ErrorMsg: string): boolean;
// completes
// case SomeEnum of
// <list of enums>
// end;
var
List, Params: TStrings;
Code: TCodeBuffer;
CaretXY: TPoint;
p: integer;
i: Integer;
Indent, Param: String;
WithoutExtraIndent: Boolean;
begin
WithoutExtraIndent := False;
Params:=SplitString(Parameter,',');
if Params<>nil then
begin
try
for i:=0 to Params.Count-1 do
begin
Param:=Params[i];
if SysUtils.CompareText(Param,'WithoutExtraIndent')=0 then
WithoutExtraIndent := True
else begin
Result:=false;
ErrorMsg:='Unknown Option: "'+Param+'"';
exit;
end;
end;
finally
Params.Free;
end;
end;
List:=TStringList.Create;
try
CaretXY:=SrcEdit.CursorTextXY;
Code:=SrcEdit.CodeToolsBuffer as TCodeBuffer;
Code.LineColToPosition(CaretXY.Y,CaretXY.X,p);
if p<1 then begin
ErrorMsg:='outside of code';
exit(false);
end;
while (p>1) and (IsIdentChar[Code.Source[p-1]]) do
begin
dec(p);
dec(CaretXY.X);
end;
if not CodeToolBoss.GetValuesOfCaseVariable(
SrcEdit.CodeToolsBuffer as TCodeBuffer,
CaretXY.X,CaretXY.Y,List) then
begin
Result:=false;
ErrorMsg:=CodeToolBoss.ErrorMessage;
if ErrorMsg='' then
ErrorMsg:='missing case variable';
LazarusIDE.DoJumpToCodeToolBossError;
exit;
end;
Indent := StringOfChar(' ',CodeToolBoss.IndentSize);
if not WithoutExtraIndent then
begin
Indent := Indent
+StringOfChar(#9,EditorOptions.EditorOpts.BlockTabIndent)
+StringOfChar(' ',EditorOptions.EditorOpts.BlockIndent);
end;
Value:='';
for i:=0 to List.Count-1 do
Value:=Value+ Indent + List[i]+': ;'+LineEnding;
finally
List.Free;
end;
Result:=true;
end;
function CodeMacroPrevWord(const Parameter: string;
InteractiveValue: TPersistent; SrcEdit: TSourceEditorInterface; var Value,
ErrorMsg: string): boolean;
{ gets word previous to the cursor position in current line
Examples:
line
i 0 count-1 forb|
with code template
for $PrevWord(1) := $PrevWord(2) to $PrevWord(3) do // template:$PrevWord(0)
begin
|
end;$PrevWord(-1)
is expanded to
for i := 0 to count-1 do // template:forb
begin
|
end;
if $PrevWord(2) is empty, then template
is expanded to
for i := | to do // template:forb
begin
end;
$PrevWord(0) expands to template itself, i.e. 'forb'
$PrevWord(-1) expands to empty string and is used in the end
if macro to delete words in the beginning of the line
}
var
Line: String;
p: TPoint;
CodeXYPos: TCodeXYPosition;
re : TRegExpr;
iParam,lastword,firstword,Position : Integer;
st: TStringList;
begin
iParam:=StrToIntDef(Parameter,-1);
Result:=true;
Value:='';
Line:=SrcEdit.CurrentLineText;
p:=SrcEdit.CursorTextXY;
if p.y<1 then exit;
CodeXYPos.X:=p.x;
CodeXYPos.Y:=p.y;
CodeXYPos.Code:=SrcEdit.CodeToolsBuffer as TCodeBuffer;
if CodeXYPos.Code=nil then exit;
st:=TStringList.Create;
re:=TRegExpr.Create;
re.Expression:='[\w\-+*\(\)\[\].^@]+';
if(re.Exec(Line))then
begin
firstword:=re.MatchPos[0];
repeat
st.Add(re.Match[0]);
lastword:=re.MatchPos[0];
until (not re.ExecNext);
end;
if st.Count>1 then
st.Move(st.count-1, 0);
if(iParam<0)then
begin
p.X:=SrcEdit.CursorTextXY.x;
CodeXYPos.Code.LineColToPosition(CodeXYPos.Y,firstword,Position);
CodeXYPos.Code.Delete(Position,lastword-firstword);
p.X:=p.X-(lastword-firstword);
SrcEdit.CursorTextXY:=p;
Value:='';
end
else
begin
if(iParam<st.count)then
Value:=st[iParam]
else
Value:='|';
end;
st.Free;
re.Free;
end;
function CodeMacroWordAtCursor(const Parameter: string;
InteractiveValue: TPersistent; SrcEdit: TSourceEditorInterface; var Value,
ErrorMsg: string): boolean;
var
SynEditor: TSynEdit;
begin
SynEditor:=SrcEdit.EditorControl as TSynEdit;
Value:=SynEditor.GetWordAtRowCol(SynEditor.LogicalCaretXY);
Result:=true;
end;
function CodeMacroEditParam(const Parameter: string;
{%H-}InteractiveValue: TPersistent; {%H-}SrcEdit: TSourceEditorInterface; var Value,
{%H-}ErrorMsg: string; TemplateParser: TIDETemplateParser): boolean;
var
p: TLazTemplateParser;
temp: TStringList;
i, g: Integer;
s: String;
begin
p := TLazTemplateParser(TemplateParser);
Value := Parameter;
g := -1;
temp := TStringList.Create;
try
s := Parameter;
while length(s) > 0 do begin
if s[1] = '"' then begin
System.Delete(s, 1, 1);
i := pos('"', s);
end
else
i := pos(',', s);
if i < 1 then
i := length(s) + 1;
temp.add(copy(s, 1, i - 1));
System.Delete(s, 1, i);
end;
//temp.CommaText := Parameter;
if temp.Count > 0 then begin
Value := temp[0];
temp.Delete(0);
i := temp.IndexOfName('Sync');
if i < 0 then
i := temp.IndexOfName('S');
if i >= 0 then
i := StrToIntDef(temp.ValueFromIndex[i], -1)
else
if (temp.IndexOf('Sync') >= 0) or (temp.IndexOf('S') >= 0) then begin
i := p.EditCellList.Count - 1;
while i >= 0 do begin
if TLazSynPluginSyncronizedEditCell(p.EditCellList[i]).CellValue = Value then
break;
dec(i);
end;
end;
dec(i);
if (i >= 0) and (i < p.EditCellList.Count) then begin
Value := TLazSynPluginSyncronizedEditCell(p.EditCellList[i]).CellValue;
g := TLazSynPluginSyncronizedEditCell(p.EditCellList[i]).Group;
end;
end;
finally
temp.Free;
end;
with TLazSynPluginSyncronizedEditCell(p.EditCellList.AddNew) do begin
LogStart := Point(p.DestPosX, p.DestPosY);
LogEnd := Point(p.DestPosX + length(Value), p.DestPosY);
if g < 0 then begin
Group := p.EditCellList.Count;
FirstInGroup := True;
end
else
Group := g;
CellValue := Value;
end;
Result := True;
end;
procedure RegisterStandardCodeTemplatesMenuItems;
var
Path: string;
begin
CodeTemplatesMenuRoot := RegisterIDEMenuRoot(CodeTemplatesMenuRootName);
Path := CodeTemplatesMenuRoot.Name;
CodeTemplateCutIDEMenuCommand := RegisterIDEMenuCommand(Path, 'Cut', lisCut);
CodeTemplateCopyIDEMenuCommand := RegisterIDEMenuCommand(Path, 'Copy', lisCopy);
CodeTemplatePasteIDEMenuCommand := RegisterIDEMenuCommand(Path, 'Paste', lisPaste);
CodeTemplateInsertMacroIDEMenuCommand := RegisterIDEMenuCommand(Path,
'InsertMacro', lisInsertMacro);
end;
procedure CreateStandardCodeMacros;
begin
IDECodeMacros:=TLazCodeMacros.Create;
RegisterCodeMacro('Upper', lisUppercaseString,
lisUppercaseStringGivenAsParameter,
@CodeMacroUpper,nil);
RegisterCodeMacro('Lower', lisLowercaseString,
lisLowercaseStringGivenAsParameter,
@CodeMacroLower,nil);
RegisterCodeMacro('Paste', lisPasteClipboard,
lisPasteFromClipboard,
@CodeMacroPaste,nil);
RegisterCodeMacro('ProcedureHead', lisInsertProcedureHead,
lisInsertHeaderOfCurrentProcedure,
@CodeMacroProcedureHead,nil);
RegisterCodeMacro('ProcedureName', lisInsertProcedureName,
lisInsertNameOfCurrentProcedure,
@CodeMacroProcedureName,nil);
RegisterCodeMacro('Date', lisInsertDate,
lisInsertDateOptionalFormatString,
@CodeMacroDate,nil);
RegisterCodeMacro('Time', lisInsertTime,
lisInsertTimeOptionalFormatString,
@CodeMacroTime,nil);
RegisterCodeMacro('DateTime', lisInsertDateAndTime,
lisInsertDateAndTimeOptionalFormatString,
@CodeMacroDateTime,nil);
RegisterCodeMacro('AddMissingEnd', lisInsertEndIfNeeded,
lisCheckIfTheNextTokenInSourceIsAnEndAndIfNotReturnsL,
@CodeMacroAddMissingEnd,nil);
RegisterCodeMacro('AddSemicolon', lisInsertSemicolonIfNeeded,
lisCheckTheNextTokenInSourceAndAddASemicolonIfNeeded,
@CodeMacroAddSemicolon,nil);
RegisterCodeMacro('OfAll', lisListOfAllCaseValues,
lisReturnsListOfAllValuesOfCaseVariableInFrontOfVaria,
@CodeMacroOfAll,nil);
RegisterCodeMacro('WordAtCursor', lisGetWordAtCurrentCursorPosition,
lisGetWordAtCurrentCursorPosition2,
@CodeMacroWordAtCursor,nil);
RegisterCodeMacro('PrevWord', lisPrecedingWord,
lisReturnParameterIndexedWord,
@CodeMacroPrevWord,nil);
RegisterCodeMacroEx('Param', lisTemplateEditParamCell,
Format(lisTemplateEditParamCellHelp, [LineEnding]),
@CodeMacroEditParam,nil);
end;
{ TCodeTemplateDialog }
procedure TCodeTemplateDialog.FormCreate(Sender: TObject);
var
ColorScheme: String;
begin
IDEDialogLayoutList.ApplyLayout(Self,600,550);
// init captions
Caption:=dlgEdCodeTempl;
AddButton.Caption:=lisAdd;
RenameButton.Caption:=lisRename;
DeleteButton.Caption:=lisDelete;
TemplatesGroupBox.Caption:=lisCTDTemplates;
ButtonPanel.OKButton.Caption:=lisBtnOk;
ButtonPanel.HelpButton.Caption:=lisMenuHelp;
ButtonPanel.CancelButton.Caption:=lisCancel;
FilenameGroupBox.Caption:=lisDebugOptionsFrmModule;
UseMacrosCheckBox.Caption:=lisEnableMacros;
InsertMacroButton.Caption:=lisInsertMacro;
KeepSubIndentCheckBox.Caption:=lisKeepSubIndentation;
KeepSubIndentCheckBox.Hint:=lisKeepRelativeIndentationOfMultiLineTemplate;
AutoOnOptionsCheckGroup.Caption:=lisCodeTemplAutoCompleteOn;
// the order matches the TAutoCompleteOption enumeration!
AutoOnOptionsCheckGroup.Items.Add(lisAutomaticallyOnLineBreak);
AutoOnOptionsCheckGroup.Items.Add(lisAutomaticallyOnSpace);
AutoOnOptionsCheckGroup.Items.Add(lisAutomaticallyOnTab);
AutoOnOptionsCheckGroup.Items.Add(lisAutomaticallyOnWordEnd);
AutoOnOptionsCheckGroup.Items.Add(lisAutomaticallyIgnoreForSelection);
AutoOnOptionsCheckGroup.Items.Add(lisAutomaticallyRemoveCharacter);
FilenameEdit.Text:=EditorOpts.CodeTemplateFileNameRaw;
FilenameEdit.InitialDir:=ExtractFilePath(EditorOpts.CodeTemplateFileNameExpand);
FilenameEdit.DialogTitle:=dlgChsCodeTempl;
FilenameEdit.Filter:=dlgFilterDciFile + '|*.dci|' + dlgFilterAll + '|' + GetAllFilesMask;
// init synedit
ColorScheme:=EditorOpts.ReadColorScheme(ASynPasSyn.LanguageName);
EditorOpts.ReadHighlighterSettings(ASynPasSyn,ColorScheme);
if EditorOpts.UseSyntaxHighlight then
TemplateSynEdit.Highlighter:=ASynPasSyn
else
TemplateSynEdit.Highlighter:=nil;
EditorOpts.SetMarkupColors(TemplateSynEdit);
EditorOpts.GetSynEditSettings(TemplateSynEdit);
EditorOpts.AssignKeyMapTo(TemplateSynEdit);
TemplateSynEdit.Gutter.Visible:=false;
TemplateSynEdit.WantTabs := false;
TemplateSynEdit.ScrollBars := ssAutoBoth;
TemplateSynEdit.Options := TemplateSynEdit.Options - [eoScrollPastEof];
TemplateSynEdit.Options := TemplateSynEdit.Options - [eoScrollPastEol];
TemplateSynEdit.Options2 := TemplateSynEdit.Options2 - [eoScrollPastEolAddPage];
TemplateSynEdit.Options2 := TemplateSynEdit.Options2 - [eoScrollPastEolAutoCaret];
// init SynAutoComplete
SynAutoComplete:=TSynEditAutoComplete.Create(Self);
EditorOpts.LoadCodeTemplates(SynAutoComplete);
// init listbox
LastTemplate:=-1;
FillCodeTemplateListBox;
if TemplateListBox.Items.Count>0 then
TemplateListBox.ItemIndex:=0; // this call ShowCurCodeTemplate
BuildPopupMenu;
end;
procedure TCodeTemplateDialog.HelpButtonClick(Sender: TObject);
begin
LazarusHelp.ShowHelpForIDEControl(Self);
end;
procedure TCodeTemplateDialog.InsertMacroButtonClick(Sender: TObject);
begin
DoInsertMacro;
end;
procedure TCodeTemplateDialog.OkButtonClick(Sender: TObject);
var
Res: TModalResult;
begin
SaveCurCodeTemplate;
EditorOpts.CodeTemplateFileNameRaw:=FilenameEdit.Text;
//EditorOpts.CodeTemplateIndentToTokenStart:=
// (CodeTemplateIndentTypeRadioGroup.ItemIndex=0);
EditorOpts.Save;
if BuildBorlandDCIFile(SynAutoComplete) then begin
Res:=mrOk;
repeat
res := EditorOpts.SaveCodeTemplates(SynAutoComplete);
if res <> mrOK then begin
res:=IDEMessageDialog(lisCCOErrorCaption, 'Unable to write code '
+'templates to file '''
+EditorOpts.CodeTemplateFileNameExpand+'''! ',mtError
,[mbAbort, mbIgnore, mbRetry]);
if res=mrAbort then exit;
end;
until Res<>mrRetry;
end;
ModalResult:=mrOk;
end;
procedure TCodeTemplateDialog.CopyMenuItem(Sender: TObject);
begin
TemplateSynEdit.CopyToClipboard;
end;
procedure TCodeTemplateDialog.CutMenuItem(Sender: TObject);
begin
TemplateSynEdit.CutToClipboard;
end;
procedure TCodeTemplateDialog.InsertMacroMenuItem(Sender: TObject);
begin
DoInsertMacro;
end;
procedure TCodeTemplateDialog.PasteMenuItem(Sender: TObject);
begin
TemplateSynEdit.PasteFromClipboard;
end;
procedure TCodeTemplateDialog.AddButtonClick(Sender: TObject);
var
Token: String;
Comment: String;
Index: PtrInt;
begin
SaveCurCodeTemplate;
Token:='new';
Comment:='(custom)';
if AddCodeTemplate(SynAutoComplete,Token,Comment)=mrOk then begin
SynAutoComplete.AddCompletion(Token, '', Comment);
FillCodeTemplateListBox;
Index := SynAutoComplete.CodeTemplates.IndexOf(Token);
if Index >= 0 then
Index := TemplateListBox.Items.IndexOfObject(TObject({%H-}Pointer(Index)));
if Index >= 0 then
TemplateListBox.ItemIndex:=Index;
ShowCurCodeTemplate;
UseMacrosCheckBox.Checked:=true;
if TemplateSynEdit.CanSetFocus then
TemplateSynEdit.SetFocus;
end;
end;
procedure TCodeTemplateDialog.DeleteButtonClick(Sender: TObject);
var
a, idx: LongInt;
Template: TTemplate;
begin
idx := TemplateListBox.ItemIndex;
if idx < 0 then exit;
a := PtrInt(TemplateListBox.Items.Objects[idx]);
if a < 0 then exit;
Template := SynAutoComplete.CodeTemplates[a];
if IDEMessageDialog(lisConfirm, dlgDelTemplate
+'"'+Template.Key+' - '+Template.Comment+'"?',
mtConfirmation,[mbOk,mbCancel])=mrOK
then begin
SynAutoComplete.CodeTemplates.Delete(a);
LastTemplate := -1; // to prevent the saving of the deleted template
FillCodeTemplateListBox;
if idx < TemplateListBox.Items.Count then
TemplateListBox.ItemIndex := idx;
ShowCurCodeTemplate;
end;
TemplateListBox.OnSelectionChange(Self, false); //update btn state
end;
procedure TCodeTemplateDialog.FormKeyDown(Sender: TObject; var Key: Word; Shift: TShiftState);
begin
// cancel
if (Key = VK_ESCAPE) and (Shift = []) then
begin
Key := 0;
Close;
end
// apply
else if (Key = VK_RETURN) and (Shift = [ssCtrl]) then
begin
Key := 0;
OkButtonClick(Sender);
end
// call help
else if (Key = VK_F1) and (Shift = []) then
begin
HelpButtonClick(Sender);
Key := 0;
end
// open file
else if (Key = VK_O) and (Shift = [ssCtrl]) then
begin
FilenameEdit.RunDialog;
Key := 0;
end
// create a new template
else if (Key = VK_N) and (Shift = [ssCtrl]) then
begin
AddButtonClick(Sender);
Key := 0;
end
// delete current template
else if (Key = VK_DELETE) and (Shift = [ssCtrl]) then
begin
DeleteButtonClick(Sender);
Key := 0;
end
// rename current template
else if (Key = VK_F2) and (Shift = []) then
begin
RenameButtonClick(Sender);
Key := 0;
end
// select next template
else if (Key = VK_DOWN) and (Shift = [ssCtrl, ssShift]) then
begin
with TemplateListBox do
if ItemIndex >= 0 then
begin
if ItemIndex + 1 < Items.Count then
ItemIndex := ItemIndex + 1;
end else begin
if Items.Count > 0 then
ItemIndex := 0;
end;
Key := 0;
end
// select previous template
else if (Key = VK_UP) and (Shift = [ssCtrl, ssShift]) then
begin
with TemplateListBox do
if ItemIndex >= 0 then
begin
if ItemIndex - 1 >= 0 then
ItemIndex := ItemIndex - 1;
end else begin
if Items.Count > 0 then
ItemIndex := Items.Count - 1;
end;
Key := 0;
end;
end;
procedure TCodeTemplateDialog.FormShow(Sender: TObject);
begin
TemplateListBox.OnSelectionChange(Self, true); //update btn states
end;
procedure TCodeTemplateDialog.RenameButtonClick(Sender: TObject);
var
a, idx: LongInt;
Template: TTemplate;
begin
idx := TemplateListBox.ItemIndex;
if idx < 0 then exit;
a := PtrInt(TemplateListBox.Items.Objects[idx]);
if a < 0 then exit;
SaveCurCodeTemplate;
Template := SynAutoComplete.CodeTemplates[a];
if EditCodeTemplate(SynAutoComplete, a)=mrOk then begin
TemplateListBox.Items[idx] := Template.Key+' - "'+Template.Comment+'"';
ShowCurCodeTemplate;
end;
end;
procedure TCodeTemplateDialog.FormClose(Sender: TObject;
var CloseAction: TCloseAction);
begin
IDEDialogLayoutList.SaveLayout(Self);
end;
procedure TCodeTemplateDialog.TemplateListBoxSelectionChange(Sender: TObject;
User: boolean);
var
en: boolean;
begin
en := TemplateListBox.ItemIndex>=0;
DeleteButton.Enabled := en;
RenameButton.Enabled := en;
EditTemplateGroupBox.Enabled := en;
SaveCurCodeTemplate;
ShowCurCodeTemplate;
end;
procedure TCodeTemplateDialog.UseMacrosCheckBoxChange(Sender: TObject);
begin
InsertMacroButton.Enabled:=UseMacrosCheckBox.Checked;
CodeTemplateInsertMacroIDEMenuCommand.Enabled:=UseMacrosCheckBox.Checked;
end;
procedure TCodeTemplateDialog.BuildPopupMenu;
begin
CodeTemplateCopyIDEMenuCommand.OnClick:=@CopyMenuItem;
CodeTemplateCutIDEMenuCommand.OnClick:=@CutMenuItem;
CodeTemplatePasteIDEMenuCommand.OnClick:=@PasteMenuItem;
CodeTemplateInsertMacroIDEMenuCommand.OnClick:=@InsertMacroMenuItem;
// assign the root TMenuItem to the registered menu root.
MainPopupMenu:=TPopupMenu.Create(Self);
// This will automatically create all registered items
CodeTemplatesMenuRoot.MenuItem := MainPopupMenu.Items;
//MainPopupMenu.Items.WriteDebugReport('TMessagesView.Create ');
TemplateSynEdit.PopupMenu:=MainPopupMenu;
end;
procedure TCodeTemplateDialog.DoInsertMacro;
var
Macro: TIDECodeMacro;
Parameter: string;
begin
Macro:=ShowCodeMacroSelectDialog(Parameter);
if Macro<>nil then begin
TemplateSynEdit.SelText:='$'+Macro.Name+'('+Parameter+')';
end;
end;
procedure TCodeTemplateDialog.FillCodeTemplateListBox;
var
a: PtrInt;
sl: TStringListUTF8Fast;
Template: TTemplate;
begin
sl:=TStringListUTF8Fast.Create;
try
for a:=0 to SynAutoComplete.CodeTemplates.Count-1 do begin
// Add the index in SynAutoComplete as Object, since both indexes won't
// be in sync after sorting
Template := SynAutoComplete.CodeTemplates[a];
sl.AddObject(Template.Key+' - "'+Template.Comment+'"',
TObject({%H-}Pointer(a)));
end;
sl.Sort;
TemplateListBox.Items.Assign(sl);
finally
sl.Free;
end;
end;
procedure TCodeTemplateDialog.ShowCurCodeTemplate;
var
EnableMacros, KeepSubIndent: boolean;
LineCount: integer;
Template: TTemplate;
idx, a, sp, ep: integer;
s: string;
AutoOnCat: array[TAutoCompleteOption] of Boolean;
c: TAutoCompleteOption;
//
procedure AddLine(const s: string);
begin
TemplateSynEdit.Lines.Add(s);
inc(LineCount);
end;
//
function GetBooleanAttribute(const AttrName: string): boolean; inline;
begin
result:=StrToBoolDef(Template.Attributes.Values[AttrName], false);
end;
//
begin
EnableMacros:=false;
KeepSubIndent:=false;
for c:=Low(TAutoCompleteOption) to High(TAutoCompleteOption) do
AutoOnCat[c]:=false;
LineCount := 0;
idx := TemplateListBox.ItemIndex;
// search template
if idx >= 0
then a := PtrInt(TemplateListBox.Items.Objects[idx])
else a := -1;
TemplateSynEdit.Lines.BeginUpdate;
TemplateSynEdit.Lines.Clear;
// debugln('TCodeTemplateDialog.ShowCurCodeTemplate A a=',dbgs(a));
if a >= 0
then begin
Template:=SynAutoComplete.CodeTemplates[a];
EditTemplateGroupBox.Caption:=dbgstr(Template.Key)
+' - '+dbgstr(Template.Comment);
EnableMacros:=GetBooleanAttribute(CodeTemplateEnableMacros);
KeepSubIndent:=GetBooleanAttribute(CodeTemplateKeepSubIndent);
for c:=Low(TAutoCompleteOption) to High(TAutoCompleteOption) do
AutoOnCat[c]:=GetBooleanAttribute(AutoCompleteOptionNames[c]);
LastTemplate := -1;
s:=Template.Value;
//debugln('TCodeTemplateDialog.ShowCurCodeTemplate s="',s,'"');
sp:=1;
ep:=1;
while ep<=length(s) do begin
if s[ep] in [#10,#13] then begin
AddLine(copy(s,sp,ep-sp));
inc(ep);
if (ep<=length(s)) and (s[ep] in [#10,#13]) and (s[ep-1]<>s[ep]) then
inc(ep);
sp:=ep;
end else inc(ep);
end;
if (ep>sp) or ((s<>'') and (s[length(s)] in [#10,#13])) then
AddLine(copy(s,sp,ep-sp));
end else begin
EditTemplateGroupBox.Caption:=lisNoTemplateSelected;
end;
LastTemplate := a;
TemplateSynEdit.Lines.EndUpdate;
TemplateSynEdit.Invalidate;
UseMacrosCheckBox.Checked:=EnableMacros;
InsertMacroButton.Enabled:=EnableMacros;
CodeTemplateInsertMacroIDEMenuCommand.Enabled:=EnableMacros;
KeepSubIndentCheckBox.Checked:=KeepSubIndent;
for c:=Low(TAutoCompleteOption) to High(TAutoCompleteOption) do
AutoOnOptionsCheckGroup.Checked[ord(c)]:=AutoOnCat[c];
end;
procedure TCodeTemplateDialog.SaveCurCodeTemplate;
var
Templ: TTemplate;
c: TAutoCompleteOption;
begin
if LastTemplate<0 then exit;
Templ:=SynAutoComplete.CodeTemplates[LastTemplate];
Templ.SetValueWithoutLastEOL(TemplateSynEdit.Lines.Text);
Templ.SetBooleanAttribute(CodeTemplateEnableMacros, UseMacrosCheckBox.Checked);
Templ.SetBooleanAttribute(CodeTemplateKeepSubIndent, KeepSubIndentCheckBox.Checked);
for c:=low(TAutoCompleteOption) to High(TAutoCompleteOption) do
Templ.SetBooleanAttribute(AutoCompleteOptionNames[c],
AutoOnOptionsCheckGroup.Checked[ord(c)]);
end;
{ TLazCodeMacros }
function TLazCodeMacros.GetItems(Index: integer): TIDECodeMacro;
begin
Result:=TIDECodeMacro(FItems[Index]);
end;
constructor TLazCodeMacros.Create;
begin
FItems:=TFPList.Create;
end;
destructor TLazCodeMacros.Destroy;
begin
Clear;
FreeAndNil(FItems);
inherited Destroy;
end;
procedure TLazCodeMacros.Clear;
var
i: Integer;
begin
for i:=0 to FItems.Count-1 do TObject(FItems[i]).Free;
FItems.Clear;
end;
function TLazCodeMacros.Count: integer;
begin
Result:=FItems.Count;
end;
function TLazCodeMacros.Add(Macro: TIDECodeMacro): integer;
begin
if FindByName(Macro.Name)<>nil then
RaiseGDBException('TLazCodeMacros.Add Name already exists');
Result:=FItems.Add(Macro);
end;
function TLazCodeMacros.FindByName(const AName: string): TIDECodeMacro;
var
i: LongInt;
begin
i:=Count-1;
while (i>=0) do begin
Result:=Items[i];
if (SysUtils.CompareText(Result.Name,AName)=0) then exit;
dec(i);
end;
Result:=nil;
end;
function TLazCodeMacros.CreateUniqueName(const AName: string): string;
begin
Result:=AName;
if FindByName(Result)=nil then exit;
Result:=CreateFirstIdentifier(Result);
while FindByName(Result)<>nil do
Result:=CreateNextIdentifier(Result);
end;
end.