mirror of
https://gitlab.com/freepascal.org/lazarus/lazarus.git
synced 2025-04-05 15:37:50 +02:00
1271 lines
39 KiB
ObjectPascal
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.
|