mirror of
https://gitlab.com/freepascal.org/lazarus/lazarus.git
synced 2025-04-05 10:57:55 +02:00
codetools: fix xml: option to remove unopen close tags, aissist: fix xml
This commit is contained in:
parent
1c07e853c7
commit
85dd5c0743
@ -7,7 +7,7 @@ interface
|
|||||||
uses
|
uses
|
||||||
Classes, SysUtils, Forms, Controls, Graphics, Dialogs, StdCtrls,
|
Classes, SysUtils, Forms, Controls, Graphics, Dialogs, StdCtrls,
|
||||||
// aissist
|
// aissist
|
||||||
AIClient, FrmAixplain, LazLoggerBase;
|
AIClient, FrmAixplain, StrAIssist, LazLoggerBase, CTXMLFixFragment;
|
||||||
|
|
||||||
const
|
const
|
||||||
SDescribeProcPrompt = 'Explain the following function in one sentence:';
|
SDescribeProcPrompt = 'Explain the following function in one sentence:';
|
||||||
@ -71,12 +71,33 @@ begin
|
|||||||
end;
|
end;
|
||||||
|
|
||||||
procedure TAIssistFPDocEditDlg.HandleAIResponse(Sender: TObject; aResponses: TPromptResponseArray);
|
procedure TAIssistFPDocEditDlg.HandleAIResponse(Sender: TObject; aResponses: TPromptResponseArray);
|
||||||
|
|
||||||
|
var
|
||||||
|
S : TStrings;
|
||||||
|
|
||||||
begin
|
begin
|
||||||
inherited HandleAIResponse(Sender, aResponses);
|
FBusy:=False;
|
||||||
|
ActivateResponse;
|
||||||
if Length(AResponses)=0 then
|
if Length(AResponses)=0 then
|
||||||
Description:=''
|
begin
|
||||||
|
Description:='';
|
||||||
|
mExplain.Lines.Add(SNoExplanation);
|
||||||
|
end
|
||||||
else
|
else
|
||||||
|
begin
|
||||||
Description:=aResponses[0].Response;
|
Description:=aResponses[0].Response;
|
||||||
|
|
||||||
|
FixFPDocFragment(Description,true,true,nil,[fffRemoveWrongCloseTags]);
|
||||||
|
|
||||||
|
mExplain.Lines.Add(SAIExplanation);
|
||||||
|
S:=TStringList.Create;
|
||||||
|
try
|
||||||
|
S.Text:=Description;
|
||||||
|
mExplain.Lines.AddStrings(S);
|
||||||
|
finally
|
||||||
|
S.Free;
|
||||||
|
end;
|
||||||
|
end;
|
||||||
DebugLn(['TAIssistFPDocEditDlg.HandleAIResponse Description="',Description,'"']);
|
DebugLn(['TAIssistFPDocEditDlg.HandleAIResponse Description="',Description,'"']);
|
||||||
end;
|
end;
|
||||||
|
|
||||||
|
@ -34,18 +34,29 @@ uses
|
|||||||
Classes, SysUtils, FileProcs, contnrs, BasicCodeTools;
|
Classes, SysUtils, FileProcs, contnrs, BasicCodeTools;
|
||||||
|
|
||||||
type
|
type
|
||||||
|
TFixFPDocFlag = (
|
||||||
|
fffVerbose, // write debugln to stdout
|
||||||
|
fffRemoveWrongCloseTags // remove all close tags which were not open, default: convert to text
|
||||||
|
);
|
||||||
|
TFixFPDocFlags = set of TFixFPDocFlag;
|
||||||
PObjectList = ^TObjectList;
|
PObjectList = ^TObjectList;
|
||||||
|
|
||||||
procedure FixFPDocFragment(var Fragment: string;
|
procedure FixFPDocFragment(var Fragment: string;
|
||||||
AllowTags, // for attribute values set this to false, so that all < are converted
|
AllowTags, // for attribute values set this to false, so that all < are converted
|
||||||
Fix: boolean; // fix errors using heuristics creating valid xml
|
Fix: boolean; // fix errors using heuristics creating valid xml
|
||||||
ErrorList: PObjectList = nil;
|
ErrorList: PObjectList = nil;
|
||||||
Verbose: boolean = false // write debugln to stdout
|
Flags: TFixFPDocFlags = []);
|
||||||
);
|
|
||||||
procedure FixFPDocAttributeValue(var Value: string);
|
procedure FixFPDocAttributeValue(var Value: string);
|
||||||
|
|
||||||
implementation
|
implementation
|
||||||
|
|
||||||
|
const
|
||||||
|
LowerAlpha = ['a'..'z'];
|
||||||
|
UpperAlpha = ['A'..'Z'];
|
||||||
|
Alpha = LowerAlpha+UpperAlpha;
|
||||||
|
Digit = ['0'..'9'];
|
||||||
|
InvalidChars = [#0..#8,#11,#12,#14..#31,#127];
|
||||||
|
Space = [' ',#9,#10,#13];
|
||||||
|
|
||||||
type
|
type
|
||||||
TFPDocFragmentError = class
|
TFPDocFragmentError = class
|
||||||
@ -54,9 +65,9 @@ type
|
|||||||
Msg: string;
|
Msg: string;
|
||||||
end;
|
end;
|
||||||
|
|
||||||
procedure FixFPDocFragment(var Fragment: string; AllowTags, Fix: boolean;
|
procedure FixFPDocFragment(var Fragment: string; AllowTags, Fix: boolean; ErrorList: PObjectList;
|
||||||
ErrorList: PObjectList; Verbose: boolean);
|
Flags: TFixFPDocFlags);
|
||||||
{ - Fix all tags to lowercase to reduce svn commits
|
{ - Fix all tags to lowercase to reduce git commits
|
||||||
- auto close comments
|
- auto close comments
|
||||||
- remove #0 from comments
|
- remove #0 from comments
|
||||||
- convert special characters to &x;
|
- convert special characters to &x;
|
||||||
@ -102,7 +113,7 @@ var
|
|||||||
NewError: TFPDocFragmentError;
|
NewError: TFPDocFragmentError;
|
||||||
LineStart,LineEnd: integer;
|
LineStart,LineEnd: integer;
|
||||||
begin
|
begin
|
||||||
if Verbose then begin
|
if fffVerbose in Flags then begin
|
||||||
debugln(['Error at ',LineCol(Rel(ErrorPos)),': ',ErrorMsg]);
|
debugln(['Error at ',LineCol(Rel(ErrorPos)),': ',ErrorMsg]);
|
||||||
GetLineStartEndAtPosition(Fragment,Rel(ErrorPos),LineStart,LineEnd);
|
GetLineStartEndAtPosition(Fragment,Rel(ErrorPos),LineStart,LineEnd);
|
||||||
debugln([' Line: ',copy(Fragment,LineStart,Rel(ErrorPos)-LineStart),'|',
|
debugln([' Line: ',copy(Fragment,LineStart,Rel(ErrorPos)-LineStart),'|',
|
||||||
@ -201,7 +212,7 @@ var
|
|||||||
inc(p,4);
|
inc(p,4);
|
||||||
// parse comment
|
// parse comment
|
||||||
repeat
|
repeat
|
||||||
if p^ in [#0..#8,#11,#12,#14..#31,#127] then begin
|
if p^ in InvalidChars then begin
|
||||||
// invalid character in comment => delete
|
// invalid character in comment => delete
|
||||||
if (p^=#0) and (p-PChar(Fragment)=length(Fragment)) then
|
if (p^=#0) and (p-PChar(Fragment)=length(Fragment)) then
|
||||||
begin
|
begin
|
||||||
@ -246,7 +257,7 @@ var
|
|||||||
// decimal number
|
// decimal number
|
||||||
inc(p);
|
inc(p);
|
||||||
i:=ord(p^)-ord('0');
|
i:=ord(p^)-ord('0');
|
||||||
while p^ in ['0'..'9'] do
|
while p^ in Digit do
|
||||||
begin
|
begin
|
||||||
i:=i+10+ord(p^)-ord('0');
|
i:=i+10+ord(p^)-ord('0');
|
||||||
if i>$10FFFF then
|
if i>$10FFFF then
|
||||||
@ -265,8 +276,8 @@ var
|
|||||||
// name
|
// name
|
||||||
inc(p);
|
inc(p);
|
||||||
NeedLowercase:=nil;
|
NeedLowercase:=nil;
|
||||||
while p^ in ['a'..'z','A'..'Z'] do begin
|
while p^ in Alpha do begin
|
||||||
if (NeedLowercase=nil) and (p^ in ['A'..'Z']) then
|
if (NeedLowercase=nil) and (p^ in UpperAlpha) then
|
||||||
NeedLowercase:=p;
|
NeedLowercase:=p;
|
||||||
inc(p);
|
inc(p);
|
||||||
end;
|
end;
|
||||||
@ -323,7 +334,7 @@ var
|
|||||||
begin
|
begin
|
||||||
// read attribute name
|
// read attribute name
|
||||||
LowerCaseName;
|
LowerCaseName;
|
||||||
while p^ in [' ',#9,#10,#13] do inc(p); // skip space
|
while p^ in Space do inc(p); // skip space
|
||||||
if p^<>'=' then begin
|
if p^<>'=' then begin
|
||||||
// missing value
|
// missing value
|
||||||
Error(p,'expected =');
|
Error(p,'expected =');
|
||||||
@ -333,7 +344,7 @@ var
|
|||||||
if p^='=' then begin
|
if p^='=' then begin
|
||||||
// read equal
|
// read equal
|
||||||
inc(p);
|
inc(p);
|
||||||
while p^ in [' ',#9,#10,#13] do inc(p); // skip space
|
while p^ in Space do inc(p); // skip space
|
||||||
if not (p^ in ['"','''']) then begin
|
if not (p^ in ['"','''']) then begin
|
||||||
// missing quote
|
// missing quote
|
||||||
Error(p,'expected "');
|
Error(p,'expected "');
|
||||||
@ -373,8 +384,9 @@ var
|
|||||||
procedure ParseCloseTag;
|
procedure ParseCloseTag;
|
||||||
var
|
var
|
||||||
i: LongInt;
|
i: LongInt;
|
||||||
|
EndP: PChar;
|
||||||
begin
|
begin
|
||||||
if (p^<>'<') or (p[1]<>'/') or (not (p[2] in ['a'..'z','A'..'Z'])) then
|
if (p^<>'<') or (p[1]<>'/') or (not (p[2] in Alpha)) then
|
||||||
begin
|
begin
|
||||||
HandleSpecialChar;
|
HandleSpecialChar;
|
||||||
exit;
|
exit;
|
||||||
@ -387,7 +399,14 @@ var
|
|||||||
if i<0 then begin
|
if i<0 then begin
|
||||||
// no corresponding open tag
|
// no corresponding open tag
|
||||||
Error(p,'tag was never opened');
|
Error(p,'tag was never opened');
|
||||||
HandleSpecialChar;
|
if fffRemoveWrongCloseTags in Flags then begin
|
||||||
|
EndP:=p+2;
|
||||||
|
while EndP^ in Alpha do inc(EndP);
|
||||||
|
if EndP='>' then
|
||||||
|
inc(EndP);
|
||||||
|
Replace(Rel(p),EndP-p,'');
|
||||||
|
end else
|
||||||
|
HandleSpecialChar;
|
||||||
exit;
|
exit;
|
||||||
end;
|
end;
|
||||||
TopItem^.CloseStartPos:=Rel(p);
|
TopItem^.CloseStartPos:=Rel(p);
|
||||||
@ -406,7 +425,7 @@ var
|
|||||||
dec(i);
|
dec(i);
|
||||||
end;
|
end;
|
||||||
// skip space
|
// skip space
|
||||||
while (p^ in [' ',#9,#10,#13]) do inc(p);
|
while (p^ in Space) do inc(p);
|
||||||
if p^='/' then begin
|
if p^='/' then begin
|
||||||
// uneeded close /
|
// uneeded close /
|
||||||
Error(p,'invalid close /');
|
Error(p,'invalid close /');
|
||||||
@ -432,7 +451,7 @@ var
|
|||||||
inc(p);
|
inc(p);
|
||||||
LowerCaseName;
|
LowerCaseName;
|
||||||
repeat
|
repeat
|
||||||
while p^ in [' ',#9,#10,#13] do inc(p); // skip space
|
while p^ in Space do inc(p); // skip space
|
||||||
case p^ of
|
case p^ of
|
||||||
'a'..'z','A'..'Z':
|
'a'..'z','A'..'Z':
|
||||||
ParseAttribute;
|
ParseAttribute;
|
||||||
@ -479,7 +498,7 @@ var
|
|||||||
end else if (p[1]='!') and (p[2]='-') and (p[3]='-') then begin
|
end else if (p[1]='!') and (p[2]='-') and (p[3]='-') then begin
|
||||||
// comment
|
// comment
|
||||||
ParseComment;
|
ParseComment;
|
||||||
end else if p[1] in ['a'..'z','A'..'Z'] then begin
|
end else if p[1] in Alpha then begin
|
||||||
// start tag
|
// start tag
|
||||||
ParseOpenTag;
|
ParseOpenTag;
|
||||||
end else if p[1]='/' then begin
|
end else if p[1]='/' then begin
|
||||||
|
@ -22,7 +22,7 @@ type
|
|||||||
|
|
||||||
TTestCTXMLFixFragment = class(TTestCase)
|
TTestCTXMLFixFragment = class(TTestCase)
|
||||||
protected
|
protected
|
||||||
function TestFrag(Title, Fragment, FixedFragment: string): boolean;
|
function TestFrag(Title, Fragment, FixedFragment: string; Flags: TFixFPDocFlags = []): boolean;
|
||||||
function TestAttr(Title, Value, FixedValue: string): boolean;
|
function TestAttr(Title, Value, FixedValue: string): boolean;
|
||||||
published
|
published
|
||||||
procedure TestFixXMLFragmentComment;
|
procedure TestFixXMLFragmentComment;
|
||||||
@ -39,14 +39,14 @@ implementation
|
|||||||
|
|
||||||
{ TTestCTXMLFixFragment }
|
{ TTestCTXMLFixFragment }
|
||||||
|
|
||||||
function TTestCTXMLFixFragment.TestFrag(Title, Fragment, FixedFragment: string
|
function TTestCTXMLFixFragment.TestFrag(Title, Fragment, FixedFragment: string;
|
||||||
): boolean;
|
Flags: TFixFPDocFlags): boolean;
|
||||||
var
|
var
|
||||||
s: String;
|
s: String;
|
||||||
begin
|
begin
|
||||||
Result:=true;
|
Result:=true;
|
||||||
s:=Fragment;
|
s:=Fragment;
|
||||||
FixFPDocFragment(s,true,true,nil,false);
|
FixFPDocFragment(s,true,true,nil,Flags);
|
||||||
AssertEquals(Title+' fragment: '+DbgStr(Fragment),dbgstr(FixedFragment),dbgstr(s));
|
AssertEquals(Title+' fragment: '+DbgStr(Fragment),dbgstr(FixedFragment),dbgstr(s));
|
||||||
end;
|
end;
|
||||||
|
|
||||||
@ -101,6 +101,7 @@ begin
|
|||||||
TestFrag('close open tag','<a>','<a/>');
|
TestFrag('close open tag','<a>','<a/>');
|
||||||
TestFrag('close open sub tag','<p><a></p>','<p><a/></p>');
|
TestFrag('close open sub tag','<p><a></p>','<p><a/></p>');
|
||||||
TestFrag('disable invalid close tag','</p>','</p>');
|
TestFrag('disable invalid close tag','</p>','</p>');
|
||||||
|
TestFrag('remove invalid close tag','bla</s>','bla',[fffRemoveWrongCloseTags]);
|
||||||
end;
|
end;
|
||||||
|
|
||||||
procedure TTestCTXMLFixFragment.TestFixXMLFragmentBugReports;
|
procedure TTestCTXMLFixFragment.TestFixXMLFragmentBugReports;
|
||||||
|
@ -1042,6 +1042,7 @@ begin
|
|||||||
Btn.Caption:=LazarusHelp.FPDocEditorTextBtnHandlers[i].Caption;
|
Btn.Caption:=LazarusHelp.FPDocEditorTextBtnHandlers[i].Caption;
|
||||||
Btn.Hint:=LazarusHelp.FPDocEditorTextBtnHandlers[i].Hint;
|
Btn.Hint:=LazarusHelp.FPDocEditorTextBtnHandlers[i].Hint;
|
||||||
Btn.ShowHint:=Btn.Hint>'';
|
Btn.ShowHint:=Btn.Hint>'';
|
||||||
|
Btn.Enabled:=HasEdit;
|
||||||
end;
|
end;
|
||||||
for i:=length(FCustomSpeedButtons)-1 downto Cnt do
|
for i:=length(FCustomSpeedButtons)-1 downto Cnt do
|
||||||
FCustomSpeedButtons[i].Free;
|
FCustomSpeedButtons[i].Free;
|
||||||
@ -1648,6 +1649,7 @@ begin
|
|||||||
OnExec(Params);
|
OnExec(Params);
|
||||||
if not Params.Success then exit;
|
if not Params.Success then exit;
|
||||||
|
|
||||||
|
// insert text
|
||||||
case Params.Part of
|
case Params.Part of
|
||||||
fpdepShortDesc:
|
fpdepShortDesc:
|
||||||
begin
|
begin
|
||||||
|
Loading…
Reference in New Issue
Block a user