lazarus/ide/codetemplatesdlg.pas
2023-06-12 06:49:22 +02:00

1271 lines
39 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
LCLProc, Forms, Controls, Dialogs, ClipBrd, StdCtrls, ExtCtrls, Menus,
ButtonPanel, EditBtn,
// LazUtils
FileUtil, LazFileUtils, LazLoggerBase, LazStringUtils, LazUTF8,
// synedit
SynEdit, SynHighlighterPas, SynEditAutoComplete,
// codetools
CodeToolManager, CodeCache, KeywordFuncLists, BasicCodeTools, PascalParserTool,
// 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;
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 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 OnCopyMenuItem(Sender: TObject);
procedure OnCutMenuItem(Sender: TObject);
procedure OnInsertMacroMenuItem(Sender: TObject);
procedure OnPasteMenuItem(Sender: TObject);
procedure TemplateListBoxSelectionChange(Sender: TObject; {%H-}User: boolean);
procedure UseMacrosCheckBoxChange(Sender: TObject);
private
SynAutoComplete: TSynEditAutoComplete;
LastTemplate: integer;
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; AIndex: integer): boolean;
var
n: integer;
begin
n:=ASynAutoComplete.Completions.IndexOf(AToken);
if (n<0) or (n=AIndex) then
Result:= true
else
begin
Result:= false;
IDEMessageDialog(
lisCodeTemplError,
Format(lisCodeTemplATokenAlreadyExists, [AToken]),
mtError, [mbOK]);
end;
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], ASynAutoComplete.Completions.Count) then
begin
Result:= mrOk;
AToken:= Str[0];
AComment:= Str[1];
end;
end;
function EditCodeTemplate(ASynAutoComplete: TSynEditAutoComplete;
AIndex: integer): TModalResult;
var
Str: array of string;
begin
Result:= mrCancel;
if (AIndex<0) or (AIndex>=ASynAutoComplete.Completions.Count) then exit;
SetLength(Str{%H-}, 2);
Str[0]:= ASynAutoComplete.Completions[AIndex];
Str[1]:= ASynAutoComplete.CompletionComments[AIndex];
if not InputQuery(lisCodeTemplEditCodeTemplate,
[lisCodeTemplToken, lisCodeTemplComment], Str) then exit;
if not IsCodeTemplateOk(ASynAutoComplete, Str[0], AIndex) then exit;
ASynAutoComplete.Completions[AIndex]:= Str[0];
ASynAutoComplete.CompletionComments[AIndex]:= 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,450);
SynAutoComplete:=TSynEditAutoComplete.Create(Self);
LastTemplate:=-1;
// 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;
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.GetLanguageName);
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;
// init SynAutoComplete
EditorOpts.LoadCodeTemplates(SynAutoComplete);
// init listbox
FillCodeTemplateListBox;
with TemplateListBox do
if Items.Count>0 then begin
ItemIndex:=0;
ShowCurCodeTemplate;
end;
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.OnCopyMenuItem(Sender: TObject);
begin
TemplateSynEdit.CopyToClipboard;
end;
procedure TCodeTemplateDialog.OnCutMenuItem(Sender: TObject);
begin
TemplateSynEdit.CutToClipboard;
end;
procedure TCodeTemplateDialog.OnInsertMacroMenuItem(Sender: TObject);
begin
DoInsertMacro;
end;
procedure TCodeTemplateDialog.OnPasteMenuItem(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.Completions.IndexOf(Token);
if Index >= 0
then Index := TemplateListBox.Items.IndexOfObject(TObject({%H-}Pointer(Index)));
if Index >= 0
then TemplateListBox.ItemIndex:=Index;
ShowCurCodeTemplate;
end;
end;
procedure TCodeTemplateDialog.DeleteButtonClick(Sender: TObject);
var
a, idx: LongInt;
begin
idx := TemplateListBox.ItemIndex;
if idx < 0 then exit;
a := PtrInt(TemplateListBox.Items.Objects[idx]);
if a < 0 then exit;
if IDEMessageDialog(lisConfirm, dlgDelTemplate
+'"'+SynAutoComplete.Completions[a]+' - '
+SynAutoComplete.CompletionComments[a]+'"'
+'?',mtConfirmation,[mbOk,mbCancel])=mrOK
then begin
SynAutoComplete.DeleteCompletion(a);
LastTemplate := -1; // to prevent the saving of the deleted template
FillCodeTemplateListBox;
if idx < TemplateListBox.Items.Count then begin
TemplateListBox.ItemIndex := idx;
end;
ShowCurCodeTemplate;
end;
TemplateListBox.OnSelectionChange(Self, false); //update btn state
end;
procedure TCodeTemplateDialog.FormShow(Sender: TObject);
begin
TemplateListBox.OnSelectionChange(Self, true); //update btn states
end;
procedure TCodeTemplateDialog.RenameButtonClick(Sender: TObject);
var
a, idx: LongInt;
begin
idx := TemplateListBox.ItemIndex;
if idx < 0 then exit;
a := PtrInt(TemplateListBox.Items.Objects[idx]);
if a < 0 then exit;
if EditCodeTemplate(SynAutoComplete, a)=mrOk then begin
TemplateListBox.Items[idx]:=
SynAutoComplete.Completions[a]
+' - "'+SynAutoComplete.CompletionComments[a]+'"';
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;
end;
procedure TCodeTemplateDialog.BuildPopupMenu;
begin
CodeTemplateCopyIDEMenuCommand.OnClick:=@OnCopyMenuItem;
CodeTemplateCutIDEMenuCommand.OnClick:=@OnCutMenuItem;
CodeTemplatePasteIDEMenuCommand.OnClick:=@OnPasteMenuItem;
CodeTemplateInsertMacroIDEMenuCommand.OnClick:=@OnInsertMacroMenuItem;
// 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 ');
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;
begin
sl:=TStringListUTF8Fast.Create;
try
for a:=0 to SynAutoComplete.Completions.Count-1 do begin
// Add the index in SynAutoComplete as Object, since both indexes won't
// be in sync after sorting
sl.AddObject(SynAutoComplete.Completions[a]
+' - "'+SynAutoComplete.CompletionComments[a]+'"', 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;
procedure AddLine(const s: string);
begin
TemplateSynEdit.Lines.Add(s);
inc(LineCount);
end;
var
idx, a, sp, ep: integer;
s: string;
AutoOnCat: array[TAutoCompleteOption] of Boolean;
Attributes: TStrings;
c: TAutoCompleteOption;
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
EditTemplateGroupBox.Caption:=dbgstr(SynAutoComplete.Completions[a])
+' - '+dbgstr(SynAutoComplete.CompletionComments[a]);
Attributes:=SynAutoComplete.CompletionAttributes[a];
EnableMacros:=Attributes.IndexOfName(CodeTemplateEnableMacros)>=0;
KeepSubIndent:=Attributes.IndexOfName(CodeTemplateKeepSubIndent)>=0;
for c:=Low(TAutoCompleteOption) to High(TAutoCompleteOption) do
AutoOnCat[c]:=Attributes.IndexOfName(AutoCompleteOptionNames[c])>=0;
LastTemplate := -1;
s:=SynAutoComplete.CompletionValues[a];
//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;
KeepSubIndentCheckBox.Checked:=KeepSubIndent;
for c:=Low(TAutoCompleteOption) to High(TAutoCompleteOption) do
AutoOnOptionsCheckGroup.Checked[ord(c)]:=AutoOnCat[c];
end;
procedure TCodeTemplateDialog.SaveCurCodeTemplate;
var
a: LongInt;
procedure SetBooleanAttribute(const AttrName: string; NewValue: boolean);
var
Attributes: TStrings;
l: LongInt;
begin
Attributes:=SynAutoComplete.CompletionAttributes[a];
if NewValue then
Attributes.Values[AttrName]:='true'
else begin
l:=Attributes.IndexOfName(AttrName);
if l>=0 then
Attributes.Delete(l);
end;
end;
var
NewValue: string;
l: integer;
c: TAutoCompleteOption;
begin
if LastTemplate<0 then exit;
a := LastTemplate;
//DebugLn('TCodeTemplateDialog.SaveCurCodeTemplate A a=',dbgs(a));
NewValue:=TemplateSynEdit.Lines.Text;
// remove last EOL
if NewValue<>'' then begin
l:=length(NewValue);
if NewValue[l] in [#10,#13] then begin
dec(l);
if (l>0) and (NewValue[l] in [#10,#13])
and (NewValue[l]<>NewValue[l+1]) then
dec(l);
SetLength(NewValue,l);
end;
end;
SynAutoComplete.CompletionValues[a]:=NewValue;
SetBooleanAttribute(CodeTemplateEnableMacros,UseMacrosCheckBox.Checked);
SetBooleanAttribute(CodeTemplateKeepSubIndent,KeepSubIndentCheckBox.Checked);
for c:=low(TAutoCompleteOption) to High(TAutoCompleteOption) do
SetBooleanAttribute(AutoCompleteOptionNames[c],AutoOnOptionsCheckGroup.Checked[ord(c)]);
//DebugLn('TCodeTemplateDialog.SaveCurCodeTemplate NewValue="',NewValue,'" SynAutoComplete.CompletionValues[a]="',SynAutoComplete.CompletionValues[a],'"');
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.