Compare commits

...

6 Commits

Author SHA1 Message Date
David Jenkins
ef065bd6d7 Merge branch 'LoopHijackUndoRedo' into 'main'
Cocoa: Improve Undo/Redo handling for when COCOALOOPHIJACK is defined.

See merge request freepascal.org/lazarus/lazarus!361
2025-04-03 19:45:33 +00:00
Martin
439afd6033 Codetools: fix parsing of "class of" with deprecated or unit-name prefix. 2025-04-03 21:18:45 +02:00
Martin
4027fdb098 IDE, SynEdit: refactor some color settings for Pas-Highlighter 2025-04-03 18:32:24 +02:00
zeljan1
1158c4ab59 ListViewFilterEdit: must clear previous items. issue #41559 2025-04-03 18:15:08 +02:00
zeljan1
33c532e60e Qt,Qt5,Qt6: fixed focus indication. issue #41562 2025-04-03 17:48:00 +02:00
David Jenkins
6955f7af5a Cocoa: Improve Undo/Redo handling for when COCOALOOPHIJACK is defined.
If COCOALOOPHIJACK is defined, we override the default NSTextView/NSTextField undoManager and return a custom one that manages 'groupsByEvent' behavior manually.  This appears to fix the overly broad grouping for most cases, though I have seen some oddities in testing, so there may be a hole related to either timing or interaction ordering.  It does rely on overloading all of the undo manager's register* events, so it's more fragile than I'd like, and if the hijack behavior is ever removed this should be pulled as well.

Reported to Lazarus as issue #36073

https://gitlab.com/freepascal.org/lazarus/lazarus/-/issues/36073

Our app is operating with COCOALOOPHIJACK
2024-10-21 13:22:10 -05:00
31 changed files with 477 additions and 56 deletions

View File

@ -4708,8 +4708,16 @@ begin
CreateChildNode; CreateChildNode;
CurNode.Desc:=ctnIdentifier; CurNode.Desc:=ctnIdentifier;
CurNode.EndPos:=CurPos.EndPos; CurNode.EndPos:=CurPos.EndPos;
EndChildNode;
ReadNextAtom; ReadNextAtom;
while CurPos.Flag = cafPoint do begin
// TMyClassClass = class of unit1.TMyClass
ReadNextAtom;
AtomIsIdentifierSaveE(20250403202500);
CurNode.EndPos:=CurPos.EndPos;
ReadNextAtom;
end;
EndChildNode;
ReadHintModifiers(False);
if CurPos.Flag<>cafSemicolon then if CurPos.Flag<>cafSemicolon then
SaveRaiseCharExpectedButAtomFound(20170421195756,';'); SaveRaiseCharExpectedButAtomFound(20170421195756,';');
end else begin end else begin

View File

@ -6,6 +6,7 @@ interface
type type
TClassOfMy = class of TMy{declaration:fdt_classof.TMy}; TClassOfMy = class of TMy{declaration:fdt_classof.TMy};
TClass1OfMy = class of TMy{declaration:fdt_classof.TMy} deprecated 'abc' experimental;
{ TMy } { TMy }
@ -14,6 +15,8 @@ type
class procedure Run; class procedure Run;
end; end;
TClass2OfMy = class of fdt_classof.TMy{declaration!:fdt_classof.TMy};
procedure DoIt; procedure DoIt;
implementation implementation

View File

@ -156,6 +156,7 @@ begin
fFilteredListview.RemoveFreeNotification(Self); fFilteredListview.RemoveFreeNotification(Self);
ForceFilter(''); ForceFilter('');
end; end;
fOriginalData.Clear;
fFilteredListview := AValue; fFilteredListview := AValue;
if Assigned(fFilteredListview) then if Assigned(fFilteredListview) then
begin begin

View File

@ -93,6 +93,18 @@ msgstr ""
msgid "Comment" msgid "Comment"
msgstr "" msgstr ""
#: syneditstrconst.syns_attrcommentansi
msgid "Ansi Comment"
msgstr ""
#: syneditstrconst.syns_attrcommentcurly
msgid "Curly Comment"
msgstr ""
#: syneditstrconst.syns_attrcommentslash
msgid "Slash Comment"
msgstr ""
#: syneditstrconst.syns_attrcondition #: syneditstrconst.syns_attrcondition
msgid "Condition" msgid "Condition"
msgstr "" msgstr ""

View File

@ -95,6 +95,18 @@ msgstr "Třída"
msgid "Comment" msgid "Comment"
msgstr "Komentář" msgstr "Komentář"
#: syneditstrconst.syns_attrcommentansi
msgid "Ansi Comment"
msgstr ""
#: syneditstrconst.syns_attrcommentcurly
msgid "Curly Comment"
msgstr ""
#: syneditstrconst.syns_attrcommentslash
msgid "Slash Comment"
msgstr ""
#: syneditstrconst.syns_attrcondition #: syneditstrconst.syns_attrcondition
msgid "Condition" msgid "Condition"
msgstr "Podmínka" msgstr "Podmínka"

View File

@ -97,6 +97,18 @@ msgstr "Klasse"
msgid "Comment" msgid "Comment"
msgstr "Kommentar" msgstr "Kommentar"
#: syneditstrconst.syns_attrcommentansi
msgid "Ansi Comment"
msgstr ""
#: syneditstrconst.syns_attrcommentcurly
msgid "Curly Comment"
msgstr ""
#: syneditstrconst.syns_attrcommentslash
msgid "Slash Comment"
msgstr ""
#: syneditstrconst.syns_attrcondition #: syneditstrconst.syns_attrcondition
msgid "Condition" msgid "Condition"
msgstr "Bedingung" msgstr "Bedingung"

View File

@ -95,6 +95,18 @@ msgstr "Clase"
msgid "Comment" msgid "Comment"
msgstr "Comentario" msgstr "Comentario"
#: syneditstrconst.syns_attrcommentansi
msgid "Ansi Comment"
msgstr ""
#: syneditstrconst.syns_attrcommentcurly
msgid "Curly Comment"
msgstr ""
#: syneditstrconst.syns_attrcommentslash
msgid "Slash Comment"
msgstr ""
#: syneditstrconst.syns_attrcondition #: syneditstrconst.syns_attrcondition
msgid "Condition" msgid "Condition"
msgstr "Condición" msgstr "Condición"

View File

@ -88,6 +88,18 @@ msgstr "Luokka"
msgid "Comment" msgid "Comment"
msgstr "Kommentti" msgstr "Kommentti"
#: syneditstrconst.syns_attrcommentansi
msgid "Ansi Comment"
msgstr ""
#: syneditstrconst.syns_attrcommentcurly
msgid "Curly Comment"
msgstr ""
#: syneditstrconst.syns_attrcommentslash
msgid "Slash Comment"
msgstr ""
#: syneditstrconst.syns_attrcondition #: syneditstrconst.syns_attrcondition
msgid "Condition" msgid "Condition"
msgstr "Ehdot" msgstr "Ehdot"

View File

@ -95,6 +95,18 @@ msgstr "Classe"
msgid "Comment" msgid "Comment"
msgstr "Commentaire" msgstr "Commentaire"
#: syneditstrconst.syns_attrcommentansi
msgid "Ansi Comment"
msgstr ""
#: syneditstrconst.syns_attrcommentcurly
msgid "Curly Comment"
msgstr ""
#: syneditstrconst.syns_attrcommentslash
msgid "Slash Comment"
msgstr ""
#: syneditstrconst.syns_attrcondition #: syneditstrconst.syns_attrcondition
msgid "Condition" msgid "Condition"
msgstr "Condition" msgstr "Condition"

View File

@ -93,6 +93,18 @@ msgstr "מחלקה"
msgid "Comment" msgid "Comment"
msgstr "הערה" msgstr "הערה"
#: syneditstrconst.syns_attrcommentansi
msgid "Ansi Comment"
msgstr ""
#: syneditstrconst.syns_attrcommentcurly
msgid "Curly Comment"
msgstr ""
#: syneditstrconst.syns_attrcommentslash
msgid "Slash Comment"
msgstr ""
#: syneditstrconst.syns_attrcondition #: syneditstrconst.syns_attrcondition
msgid "Condition" msgid "Condition"
msgstr "תנאי" msgstr "תנאי"

View File

@ -95,6 +95,18 @@ msgstr "Osztály"
msgid "Comment" msgid "Comment"
msgstr "Megjegyzés" msgstr "Megjegyzés"
#: syneditstrconst.syns_attrcommentansi
msgid "Ansi Comment"
msgstr ""
#: syneditstrconst.syns_attrcommentcurly
msgid "Curly Comment"
msgstr ""
#: syneditstrconst.syns_attrcommentslash
msgid "Slash Comment"
msgstr ""
#: syneditstrconst.syns_attrcondition #: syneditstrconst.syns_attrcondition
msgid "Condition" msgid "Condition"
msgstr "Feltétel" msgstr "Feltétel"

View File

@ -96,6 +96,18 @@ msgstr ""
msgid "Comment" msgid "Comment"
msgstr "" msgstr ""
#: syneditstrconst.syns_attrcommentansi
msgid "Ansi Comment"
msgstr ""
#: syneditstrconst.syns_attrcommentcurly
msgid "Curly Comment"
msgstr ""
#: syneditstrconst.syns_attrcommentslash
msgid "Slash Comment"
msgstr ""
#: syneditstrconst.syns_attrcondition #: syneditstrconst.syns_attrcondition
msgid "Condition" msgid "Condition"
msgstr "" msgstr ""

View File

@ -96,6 +96,18 @@ msgstr "Classe"
msgid "Comment" msgid "Comment"
msgstr "Commento" msgstr "Commento"
#: syneditstrconst.syns_attrcommentansi
msgid "Ansi Comment"
msgstr ""
#: syneditstrconst.syns_attrcommentcurly
msgid "Curly Comment"
msgstr ""
#: syneditstrconst.syns_attrcommentslash
msgid "Slash Comment"
msgstr ""
#: syneditstrconst.syns_attrcondition #: syneditstrconst.syns_attrcondition
msgid "Condition" msgid "Condition"
msgstr "Condizione" msgstr "Condizione"

View File

@ -96,6 +96,18 @@ msgstr "Klasė"
msgid "Comment" msgid "Comment"
msgstr "Komentaras" msgstr "Komentaras"
#: syneditstrconst.syns_attrcommentansi
msgid "Ansi Comment"
msgstr ""
#: syneditstrconst.syns_attrcommentcurly
msgid "Curly Comment"
msgstr ""
#: syneditstrconst.syns_attrcommentslash
msgid "Slash Comment"
msgstr ""
#: syneditstrconst.syns_attrcondition #: syneditstrconst.syns_attrcondition
msgid "Condition" msgid "Condition"
msgstr "Sąlyga" msgstr "Sąlyga"

View File

@ -93,6 +93,18 @@ msgstr ""
msgid "Comment" msgid "Comment"
msgstr "" msgstr ""
#: syneditstrconst.syns_attrcommentansi
msgid "Ansi Comment"
msgstr ""
#: syneditstrconst.syns_attrcommentcurly
msgid "Curly Comment"
msgstr ""
#: syneditstrconst.syns_attrcommentslash
msgid "Slash Comment"
msgstr ""
#: syneditstrconst.syns_attrcondition #: syneditstrconst.syns_attrcondition
msgid "Condition" msgid "Condition"
msgstr "" msgstr ""

View File

@ -96,6 +96,18 @@ msgstr "Klasa"
msgid "Comment" msgid "Comment"
msgstr "Komentarz" msgstr "Komentarz"
#: syneditstrconst.syns_attrcommentansi
msgid "Ansi Comment"
msgstr ""
#: syneditstrconst.syns_attrcommentcurly
msgid "Curly Comment"
msgstr ""
#: syneditstrconst.syns_attrcommentslash
msgid "Slash Comment"
msgstr ""
#: syneditstrconst.syns_attrcondition #: syneditstrconst.syns_attrcondition
msgid "Condition" msgid "Condition"
msgstr "Warunek" msgstr "Warunek"

View File

@ -85,6 +85,18 @@ msgstr ""
msgid "Comment" msgid "Comment"
msgstr "" msgstr ""
#: syneditstrconst.syns_attrcommentansi
msgid "Ansi Comment"
msgstr ""
#: syneditstrconst.syns_attrcommentcurly
msgid "Curly Comment"
msgstr ""
#: syneditstrconst.syns_attrcommentslash
msgid "Slash Comment"
msgstr ""
#: syneditstrconst.syns_attrcondition #: syneditstrconst.syns_attrcondition
msgid "Condition" msgid "Condition"
msgstr "" msgstr ""

View File

@ -95,6 +95,18 @@ msgstr "Classe"
msgid "Comment" msgid "Comment"
msgstr "Comentário" msgstr "Comentário"
#: syneditstrconst.syns_attrcommentansi
msgid "Ansi Comment"
msgstr ""
#: syneditstrconst.syns_attrcommentcurly
msgid "Curly Comment"
msgstr ""
#: syneditstrconst.syns_attrcommentslash
msgid "Slash Comment"
msgstr ""
#: syneditstrconst.syns_attrcondition #: syneditstrconst.syns_attrcondition
msgid "Condition" msgid "Condition"
msgstr "Condição" msgstr "Condição"

View File

@ -95,6 +95,18 @@ msgstr "Класс"
msgid "Comment" msgid "Comment"
msgstr "Комментарий" msgstr "Комментарий"
#: syneditstrconst.syns_attrcommentansi
msgid "Ansi Comment"
msgstr ""
#: syneditstrconst.syns_attrcommentcurly
msgid "Curly Comment"
msgstr ""
#: syneditstrconst.syns_attrcommentslash
msgid "Slash Comment"
msgstr ""
#: syneditstrconst.syns_attrcondition #: syneditstrconst.syns_attrcondition
msgid "Condition" msgid "Condition"
msgstr "Условие" msgstr "Условие"

View File

@ -99,6 +99,18 @@ msgstr "Sınıf"
msgid "Comment" msgid "Comment"
msgstr "Açıklama" msgstr "Açıklama"
#: syneditstrconst.syns_attrcommentansi
msgid "Ansi Comment"
msgstr ""
#: syneditstrconst.syns_attrcommentcurly
msgid "Curly Comment"
msgstr ""
#: syneditstrconst.syns_attrcommentslash
msgid "Slash Comment"
msgstr ""
#: syneditstrconst.syns_attrcondition #: syneditstrconst.syns_attrcondition
msgid "Condition" msgid "Condition"
msgstr "Şart" msgstr "Şart"

View File

@ -98,6 +98,18 @@ msgstr "Клас"
msgid "Comment" msgid "Comment"
msgstr "Коментар" msgstr "Коментар"
#: syneditstrconst.syns_attrcommentansi
msgid "Ansi Comment"
msgstr ""
#: syneditstrconst.syns_attrcommentcurly
msgid "Curly Comment"
msgstr ""
#: syneditstrconst.syns_attrcommentslash
msgid "Slash Comment"
msgstr ""
#: syneditstrconst.syns_attrcondition #: syneditstrconst.syns_attrcondition
msgid "Condition" msgid "Condition"
msgstr "Умова" msgstr "Умова"

View File

@ -96,6 +96,18 @@ msgstr "类"
msgid "Comment" msgid "Comment"
msgstr "注释" msgstr "注释"
#: syneditstrconst.syns_attrcommentansi
msgid "Ansi Comment"
msgstr ""
#: syneditstrconst.syns_attrcommentcurly
msgid "Curly Comment"
msgstr ""
#: syneditstrconst.syns_attrcommentslash
msgid "Slash Comment"
msgstr ""
#: syneditstrconst.syns_attrcondition #: syneditstrconst.syns_attrcondition
msgid "Condition" msgid "Condition"
msgstr "状态" msgstr "状态"

View File

@ -66,6 +66,9 @@ resourcestring
SYNS_AttrCharacter = 'Character'; SYNS_AttrCharacter = 'Character';
SYNS_AttrClass = 'Class'; SYNS_AttrClass = 'Class';
SYNS_AttrComment = 'Comment'; SYNS_AttrComment = 'Comment';
SYNS_AttrCommentAnsi = 'Ansi Comment';
SYNS_AttrCommentCurly = 'Curly Comment';
SYNS_AttrCommentSlash = 'Slash Comment';
SYNS_AttrIDEDirective = 'IDE Directive'; SYNS_AttrIDEDirective = 'IDE Directive';
SYNS_AttrCondition = 'Condition'; SYNS_AttrCondition = 'Condition';
SYNS_AttrDataType = 'Data type'; SYNS_AttrDataType = 'Data type';
@ -199,6 +202,9 @@ const
SYNS_XML_AttrCharacter = 'Character'; SYNS_XML_AttrCharacter = 'Character';
SYNS_XML_AttrClass = 'Class'; SYNS_XML_AttrClass = 'Class';
SYNS_XML_AttrComment = 'Comment'; SYNS_XML_AttrComment = 'Comment';
SYNS_XML_AttrCommentAnsi = 'Ansi Comment';
SYNS_XML_AttrCommentCurly = 'Curly Comment';
SYNS_XML_AttrCommentSlash = 'Slash Comment';
SYNS_XML_AttrIDEDirective = 'IDE Directive'; SYNS_XML_AttrIDEDirective = 'IDE Directive';
SYNS_XML_AttrCondition = 'Condition'; SYNS_XML_AttrCondition = 'Condition';
SYNS_XML_AttrDataType = 'Data type'; SYNS_XML_AttrDataType = 'Data type';

View File

@ -188,7 +188,11 @@ type
reaDeclVarName, reaDeclVarName,
reaDeclTypeName, reaDeclTypeName,
reaDeclType, reaDeclType,
reaDeclValue reaDeclValue,
// other
reCommentAnsi,
reCommentCurly,
reCommentSlash
); );
TRequiredStates = set of TRequiredState; TRequiredStates = set of TRequiredState;
@ -595,6 +599,9 @@ type
PSynPasSynCustomTokenInfoListEx = ^TSynPasSynCustomTokenInfoListEx; PSynPasSynCustomTokenInfoListEx = ^TSynPasSynCustomTokenInfoListEx;
private private
FCaseLabelAttriMatchesElseOtherwise: Boolean; FCaseLabelAttriMatchesElseOtherwise: Boolean;
FCommentAnsiAttri: TSynHighlighterAttributesModifier;
FCommentCurlyAttri: TSynHighlighterAttributesModifier;
FCommentSlashAttri: TSynHighlighterAttributesModifier;
FNestedBracketAttribs: TSynHighlighterAttributesModifierCollection; FNestedBracketAttribs: TSynHighlighterAttributesModifierCollection;
FNestedBracketMergedMarkup: TSynSelectedColorMergeResult; FNestedBracketMergedMarkup: TSynSelectedColorMergeResult;
FHighNestedBracketAttrib: integer; FHighNestedBracketAttrib: integer;
@ -605,7 +612,6 @@ type
Lists: array of TSynPasSynCustomTokenInfoListEx; Lists: array of TSynPasSynCustomTokenInfoListEx;
end; end;
FCustomTokenMarkup, FCustomCommentTokenMarkup: TSynHighlighterAttributesModifier; FCustomTokenMarkup, FCustomCommentTokenMarkup: TSynHighlighterAttributesModifier;
FCustomTokenMarkupSlash, FCustomTokenMarkupAnsi, FCustomTokenMarkupBor: TSynHighlighterAttributesModifier;
FCustomTokenMergedMarkup, FCustomCommentTokenMergedMarkup: TSynSelectedColorMergeResult; FCustomTokenMergedMarkup, FCustomCommentTokenMergedMarkup: TSynSelectedColorMergeResult;
FCurIDEDirectiveAttri: TSynSelectedColorMergeResult; FCurIDEDirectiveAttri: TSynSelectedColorMergeResult;
@ -987,6 +993,9 @@ type
property AsmAttri: TSynHighlighterAttributes read fAsmAttri write fAsmAttri; property AsmAttri: TSynHighlighterAttributes read fAsmAttri write fAsmAttri;
property CommentAttri: TSynHighlighterAttributes read fCommentAttri property CommentAttri: TSynHighlighterAttributes read fCommentAttri
write fCommentAttri; write fCommentAttri;
property CommentAnsiAttri: TSynHighlighterAttributesModifier read FCommentAnsiAttri write FCommentAnsiAttri;
property CommentCurlyAttri: TSynHighlighterAttributesModifier read FCommentCurlyAttri write FCommentCurlyAttri;
property CommentSlashAttri: TSynHighlighterAttributesModifier read FCommentSlashAttri write FCommentSlashAttri;
property IDEDirectiveAttri: TSynHighlighterAttributesModifier read FIDEDirectiveAttri property IDEDirectiveAttri: TSynHighlighterAttributesModifier read FIDEDirectiveAttri
write FIDEDirectiveAttri; write FIDEDirectiveAttri;
property IdentifierAttri: TSynHighlighterAttributes read fIdentifierAttri property IdentifierAttri: TSynHighlighterAttributes read fIdentifierAttri
@ -1596,9 +1605,6 @@ begin
FNeedCustomTokenBuild := False; FNeedCustomTokenBuild := False;
FCustomTokenMarkup := nil; FCustomTokenMarkup := nil;
FCustomCommentTokenMarkup := nil; FCustomCommentTokenMarkup := nil;
FCustomTokenMarkupSlash := nil;
FCustomTokenMarkupAnsi := nil;
FCustomTokenMarkupBor := nil;
for i := 0 to 255 do begin for i := 0 to 255 do begin
for j := 0 to length(FCustomTokenInfo[i].Lists) - 1 do for j := 0 to length(FCustomTokenInfo[i].Lists) - 1 do
FreeAndNil(FCustomTokenInfo[i].Lists[j].List); FreeAndNil(FCustomTokenInfo[i].Lists[j].List);
@ -1625,23 +1631,6 @@ begin
FCustomTokenInfo[h].MatchTokenKinds := FCustomTokenInfo[h].MatchTokenKinds + mtk; FCustomTokenInfo[h].MatchTokenKinds := FCustomTokenInfo[h].MatchTokenKinds + mtk;
for tk in mtk do begin for tk in mtk do begin
case tk of
tkSlashComment:
if t = '*' then begin
FCustomTokenMarkupSlash := FSynCustomTokens[i].Markup;
continue;
end;
tkAnsiComment:
if t = '*' then begin
FCustomTokenMarkupAnsi := FSynCustomTokens[i].Markup;
continue;
end;
tkBorComment:
if t = '*' then begin
FCustomTokenMarkupBor := FSynCustomTokens[i].Markup;
continue;
end;
end;
Lst := FindList(h, tk); Lst := FindList(h, tk);
Lst^.List.AddObject(UpperCase(t), FSynCustomTokens[i]); Lst^.List.AddObject(UpperCase(t), FSynCustomTokens[i]);
end; end;
@ -3943,7 +3932,7 @@ end;
constructor TSynPasSyn.Create(AOwner: TComponent); constructor TSynPasSyn.Create(AOwner: TComponent);
begin begin
inherited Create(AOwner); inherited Create(AOwner);
FCaseLabelAttriMatchesElseOtherwise := True;; FCaseLabelAttriMatchesElseOtherwise := True;
FStringKeywordMode := spsmDefault; FStringKeywordMode := spsmDefault;
FExtendedKeywordsMode := False; FExtendedKeywordsMode := False;
CreateDividerDrawConfig; CreateDividerDrawConfig;
@ -3953,6 +3942,12 @@ begin
fCommentAttri := TSynHighlighterAttributes.Create(@SYNS_AttrComment, SYNS_XML_AttrComment); fCommentAttri := TSynHighlighterAttributes.Create(@SYNS_AttrComment, SYNS_XML_AttrComment);
fCommentAttri.Style:= [fsItalic]; fCommentAttri.Style:= [fsItalic];
AddAttribute(fCommentAttri); AddAttribute(fCommentAttri);
FCommentAnsiAttri := TSynHighlighterAttributesModifier.Create(@SYNS_AttrCommentAnsi, SYNS_XML_AttrCommentAnsi);
AddAttribute(FCommentAnsiAttri);
FCommentCurlyAttri := TSynHighlighterAttributesModifier.Create(@SYNS_AttrCommentCurly, SYNS_XML_AttrCommentCurly);
AddAttribute(FCommentCurlyAttri);
FCommentSlashAttri := TSynHighlighterAttributesModifier.Create(@SYNS_AttrCommentSlash, SYNS_XML_AttrCommentSlash);
AddAttribute(FCommentSlashAttri);
FIDEDirectiveAttri := TSynHighlighterAttributesModifier.Create(@SYNS_AttrIDEDirective, SYNS_XML_AttrIDEDirective); FIDEDirectiveAttri := TSynHighlighterAttributesModifier.Create(@SYNS_AttrIDEDirective, SYNS_XML_AttrIDEDirective);
AddAttribute(FIDEDirectiveAttri); AddAttribute(FIDEDirectiveAttri);
// FCurIDEDirectiveAttri, FCurCaseLabelAttri, FCurProcTypeDeclExtraAttr // FCurIDEDirectiveAttri, FCurCaseLabelAttri, FCurProcTypeDeclExtraAttr
@ -4182,7 +4177,8 @@ var
p: LongInt; p: LongInt;
IsInWord, WasInWord, ct: Boolean; IsInWord, WasInWord, ct: Boolean;
begin begin
FCustomCommentTokenMarkup := FCustomTokenMarkupBor; if reCommentCurly in FRequiredStates then
FCustomCommentTokenMarkup := FCommentCurlyAttri;
fTokenID := tkComment; fTokenID := tkComment;
if rsIDEDirective in fRange then if rsIDEDirective in fRange then
fTokenID := tkIDEDirective; fTokenID := tkIDEDirective;
@ -4500,7 +4496,8 @@ begin
dec(Run); dec(Run);
StartPascalCodeFoldBlock(cfbtBorCommand); StartPascalCodeFoldBlock(cfbtBorCommand);
FCustomCommentTokenMarkup := FCustomTokenMarkupBor; if reCommentCurly in FRequiredStates then
FCustomCommentTokenMarkup := FCommentCurlyAttri;
if not (FIsInNextToEOL or IsScanning) then if not (FIsInNextToEOL or IsScanning) then
GetCustomSymbolToken(tkBorComment, 1, FCustomTokenMarkup); GetCustomSymbolToken(tkBorComment, 1, FCustomTokenMarkup);
@ -4737,7 +4734,8 @@ var
IsInWord, WasInWord, ct: Boolean; IsInWord, WasInWord, ct: Boolean;
begin begin
fTokenID := tkComment; fTokenID := tkComment;
FCustomCommentTokenMarkup := FCustomTokenMarkupAnsi; if reCommentAnsi in FRequiredStates then
FCustomCommentTokenMarkup := FCommentAnsiAttri;
if (not (FIsInNextToEOL or IsScanning)) then begin if (not (FIsInNextToEOL or IsScanning)) then begin
if FUsePasDoc and (fLine[Run] = '@') then begin if FUsePasDoc and (fLine[Run] = '@') then begin
@ -4863,7 +4861,8 @@ begin
Dec(Run); Dec(Run);
StartPascalCodeFoldBlock(cfbtAnsiComment); StartPascalCodeFoldBlock(cfbtAnsiComment);
FCustomCommentTokenMarkup := FCustomTokenMarkupAnsi; if reCommentAnsi in FRequiredStates then
FCustomCommentTokenMarkup := FCommentAnsiAttri;
if not (FIsInNextToEOL or IsScanning) then if not (FIsInNextToEOL or IsScanning) then
GetCustomSymbolToken(tkAnsiComment, 2, FCustomTokenMarkup); GetCustomSymbolToken(tkAnsiComment, 2, FCustomTokenMarkup);
@ -5061,7 +5060,8 @@ end;
procedure TSynPasSyn.SlashProc; procedure TSynPasSyn.SlashProc;
begin begin
if fLine[Run+1] = '/' then begin if fLine[Run+1] = '/' then begin
FCustomCommentTokenMarkup := FCustomTokenMarkupSlash; if reCommentSlash in FRequiredStates then
FCustomCommentTokenMarkup := FCommentSlashAttri;
FIsInSlash := True; FIsInSlash := True;
fTokenID := tkComment; fTokenID := tkComment;
@ -5093,7 +5093,8 @@ var
AtSlashOpen: Boolean; AtSlashOpen: Boolean;
begin begin
if FIsInSlash and (not (FIsInNextToEOL or IsScanning)) then begin if FIsInSlash and (not (FIsInNextToEOL or IsScanning)) then begin
FCustomCommentTokenMarkup := FCustomTokenMarkupSlash; if reCommentSlash in FRequiredStates then
FCustomCommentTokenMarkup := FCommentSlashAttri;
fTokenID := tkComment; fTokenID := tkComment;
if (fLine[Run] = '@') then begin if (fLine[Run] = '@') then begin
@ -5113,7 +5114,8 @@ begin
AtSlashOpen := (fLine[Run] = '/') and (fLine[Run + 1] = '/') and not FIsInSlash; AtSlashOpen := (fLine[Run] = '/') and (fLine[Run + 1] = '/') and not FIsInSlash;
if FIsInSlash or AtSlashOpen then begin if FIsInSlash or AtSlashOpen then begin
FIsInSlash := True; FIsInSlash := True;
FCustomCommentTokenMarkup := FCustomTokenMarkupSlash; if reCommentSlash in FRequiredStates then
FCustomCommentTokenMarkup := FCommentSlashAttri;
// Continue fold block // Continue fold block
fTokenID := tkComment; fTokenID := tkComment;
@ -7309,6 +7311,10 @@ begin
FRequiredStates := []; FRequiredStates := [];
if FCommentAnsiAttri.IsEnabled then Include(FRequiredStates, reCommentAnsi);
if FCommentCurlyAttri.IsEnabled then Include(FRequiredStates, reCommentCurly);
if FCommentSlashAttri.IsEnabled then Include(FRequiredStates, reCommentSlash);
//if fPasDocKeyWordAttri.IsEnabled then begin //if fPasDocKeyWordAttri.IsEnabled then begin
//end; //end;
//if fPasDocSymbolAttri.IsEnabled then begin //if fPasDocSymbolAttri.IsEnabled then begin

View File

@ -25,7 +25,7 @@ unit editor_color_options;
interface interface
uses uses
Classes, Controls, Math, Types, typinfo, sysutils, Classes, Controls, Math, Types, typinfo, fgl, sysutils,
// LazUtils // LazUtils
Laz2_XMLCfg, LazFileUtils, LazUTF8, LazLoggerBase, Laz2_XMLCfg, LazFileUtils, LazUTF8, LazLoggerBase,
// LCL // LCL
@ -1162,13 +1162,53 @@ begin
end; end;
procedure TEditorColorOptionsFrame.FillColorElementListBox; procedure TEditorColorOptionsFrame.FillColorElementListBox;
function GetParentNameForLanguageElement(AnAttr: TColorSchemeAttribute;
out AParentName, AGroupName: String): boolean;
begin
AGroupName := '';
AParentName := FCurrentHighlighter.LanguageName;
Result := True;
if hafCustomWords in AnAttr.Features then begin
AParentName := AParentName + ' ' + dlgAddHiAttrGroup_Suffix_Custom;
end
else
if strlcomp(PChar(AnAttr.StoredName), PChar(NESTED_BRACKET_STOREDNAME), length(NESTED_BRACKET_STOREDNAME)) = 0
then begin
AParentName := AParentName + ' ' + dlgAddHiAttrGroup_Suffix_NBrackets;
end
else
if hafAlpha in AnAttr.Features then begin
AParentName := AParentName + ' ' + dlgAddHiAttrGroup_Suffix_Extended;
case AnAttr.StoredName of
//SYNS_XML_AttrIDEDirective,
SYNS_XML_AttrCommentAnsi, SYNS_XML_AttrCommentCurly, SYNS_XML_AttrCommentSlash,
SYNS_XML_AttrPasDocKey, SYNS_XML_AttrPasDocSymbol, SYNS_XML_AttrPasDocUnknown:
AGroupName := dlgAddHiAttrGroup_Comment;
SYNS_XML_AttrProcedureHeaderName, SYNS_XML_AttrPropertyName,
SYNS_XML_AttrProcedureHeaderParam, SYNS_XML_AttrProcedureHeaderResult,
SYNS_XML_AttrProcedureHeaderType, SYNS_XML_AttrProcedureHeaderValue:
AGroupName := dlgAddHiAttrGroup_ProgHeader;
SYNS_XML_AttrDeclarationVarConstName, SYNS_XML_AttrDeclarationTypeName,
SYNS_XML_AttrDeclarationType, SYNS_XML_AttrDeclarationValue:
AGroupName := dlgAddHiAttrGroup_DeclSection;
end;
end
else
Result := False;
end;
type
TGroupNodes = specialize TFPGMap<String, TTreeNode>;
var var
i, AttriIdx: Integer; i, AttriIdx: Integer;
ParentName: String; ParentName, GroupName: String;
ParentNode: TTreeNode; ParentNode: TTreeNode;
j: TAhaGroupName; j: TAhaGroupName;
Attr: TColorSchemeAttribute; Attr: TColorSchemeAttribute;
NewNode, DefNode, p, ComplWindowEntryParentNode: TTreeNode; NewNode, DefNode, p, ComplWindowEntryParentNode: TTreeNode;
GroupNodes: TGroupNodes;
begin begin
ColorElementTree.BeginUpdate; ColorElementTree.BeginUpdate;
ColorElementTree.Items.Clear; ColorElementTree.Items.Clear;
@ -1185,6 +1225,7 @@ begin
if not(j in [agnDefault, agnLanguage, agnRegistered]) then if not(j in [agnDefault, agnLanguage, agnRegistered]) then
ColorElementTree.Items.Add(nil, AdditionalHighlightGroupNames[j]).Visible := False; ColorElementTree.Items.Add(nil, AdditionalHighlightGroupNames[j]).Visible := False;
GroupNodes := TGroupNodes.Create;
// Fill Attributes in // Fill Attributes in
ComplWindowEntryParentNode := nil; ComplWindowEntryParentNode := nil;
DefNode := nil; DefNode := nil;
@ -1196,25 +1237,18 @@ begin
agnLanguage: agnLanguage:
begin begin
ParentNode := ColorElementTree.Items.GetFirstNode; ParentNode := ColorElementTree.Items.GetFirstNode;
if FIsEditingDefaults then begin if GetParentNameForLanguageElement(Attr, ParentName, GroupName) then begin
ParentName := AdditionalHighlightGroupNames[agnDefault];
end
else
if hafCustomWords in Attr.Features then begin
ParentName := FCurrentHighlighter.LanguageName + ' ' + dlgAddHiAttrGroup_Suffix_Custom;
ParentNode := ColorElementTree.Items.FindTopLvlNode(ParentName); ParentNode := ColorElementTree.Items.FindTopLvlNode(ParentName);
end if ParentNode = nil then
else ParentNode := ColorElementTree.Items.Add(nil, ParentName);
if strlcomp(PChar(Attr.StoredName), PChar(NESTED_BRACKET_STOREDNAME), length(NESTED_BRACKET_STOREDNAME)) = 0
then begin if GroupName <> '' then begin
ParentName := FCurrentHighlighter.LanguageName + ' ' + dlgAddHiAttrGroup_Suffix_NBrackets; ParentName := GroupName;
ParentNode := ColorElementTree.Items.FindTopLvlNode(ParentName); p := ParentNode;
end if not GroupNodes.TryGetData(GroupName, ParentNode) then begin
else begin ParentNode := ColorElementTree.Items.AddChild(p, GroupName);
ParentName := FCurrentHighlighter.LanguageName; GroupNodes.Add(GroupName, ParentNode);
if hafAlpha in Attr.Features then begin end;
ParentName := ParentName + ' ' + dlgAddHiAttrGroup_Suffix_Extended;
ParentNode := ColorElementTree.Items.FindTopLvlNode(ParentName);
end; end;
end; end;
end; end;
@ -1261,6 +1295,9 @@ begin
FindCurHighlightElement; FindCurHighlightElement;
if assigned(ComplWindowEntryParentNode) then if assigned(ComplWindowEntryParentNode) then
ComplWindowEntryParentNode.Collapse(True); ComplWindowEntryParentNode.Collapse(True);
for i := 0 to GroupNodes.Count - 1 do
GroupNodes.Data[i].Collapse(False);
GroupNodes.Free;
end; end;
procedure TEditorColorOptionsFrame.SetColorElementsToDefaults(OnlySelected: Boolean); procedure TEditorColorOptionsFrame.SetColorElementsToDefaults(OnlySelected: Boolean);

View File

@ -2202,6 +2202,9 @@ resourcestring
dlgAddHiAttrGroup_Suffix_Custom = '(Custom)'; dlgAddHiAttrGroup_Suffix_Custom = '(Custom)';
dlgAddHiAttrGroup_Suffix_NBrackets = '(Nested Brackets)'; dlgAddHiAttrGroup_Suffix_NBrackets = '(Nested Brackets)';
dlgAddHiAttrGroup_Suffix_EntryType = '(entry type)'; dlgAddHiAttrGroup_Suffix_EntryType = '(entry type)';
dlgAddHiAttrGroup_Comment = 'Comments';
dlgAddHiAttrGroup_ProgHeader = 'Procedure Header';
dlgAddHiAttrGroup_DeclSection = 'Declaration Sections';
dlgEditAccessCaptionLockedInView = 'Locked, if text in view'; dlgEditAccessCaptionLockedInView = 'Locked, if text in view';

View File

@ -15,6 +15,7 @@
unit Cocoa_Extra; unit Cocoa_Extra;
{$mode objfpc}{$H+} {$mode objfpc}{$H+}
{$modeswitch cblocks}
{$modeswitch objectivec1} {$modeswitch objectivec1}
{$include cocoadefines.inc} {$include cocoadefines.inc}
@ -645,6 +646,14 @@ type
patchVersion: NSInteger; patchVersion: NSInteger;
end; end;
NSUndoManagerUndoWithTargetCBlock = reference to procedure(target: id); cblock; cdecl;
NSUndoManagerFix = objccategory external (NSUndoManager)
procedure registerUndoWithTarget_handler(target: id;
handler: NSUndoManagerUndoWithTargetCBlock);
message 'registerUndoWithTarget:handler:';
end;
const const
// defined in NSApplication.h // defined in NSApplication.h
NSAppKitVersionNumber10_5 = 949; NSAppKitVersionNumber10_5 = 949;

View File

@ -23,6 +23,9 @@ unit CocoaTextEdits;
{.$DEFINE COCOA_DEBUG_SETBOUNDS} {.$DEFINE COCOA_DEBUG_SETBOUNDS}
{.$DEFINE COCOA_SPIN_DEBUG} {.$DEFINE COCOA_SPIN_DEBUG}
{.$DEFINE COCOA_SPINEDIT_INSIDE_CONTAINER} {.$DEFINE COCOA_SPINEDIT_INSIDE_CONTAINER}
{$IFDEF COCOALOOPHIJACK}
{$DEFINE COCOA_OVERRIDE_UNDOMANAGER}
{$ENDIF}
interface interface
@ -31,7 +34,7 @@ uses
Math, // needed for MinDouble, MaxDouble Math, // needed for MinDouble, MaxDouble
LCLType, LCLType,
MacOSAll, CocoaAll, CocoaConfig, CocoaUtils, CocoaGDIObjects, MacOSAll, CocoaAll, CocoaConfig, CocoaUtils, CocoaGDIObjects,
CocoaPrivate, CocoaCallback; CocoaPrivate, CocoaCallback, Cocoa_Extra;
const const
SPINEDIT_DEFAULT_STEPPER_WIDTH = 15; SPINEDIT_DEFAULT_STEPPER_WIDTH = 15;
@ -64,7 +67,10 @@ type
callback: ICommonCallback; callback: ICommonCallback;
maxLength: Integer; maxLength: Integer;
fixedInitSetting: Boolean; fixedInitSetting: Boolean;
{$IFDEF COCOA_OVERRIDE_UNDOMANAGER}
FUndoManager: NSUndoManager;
procedure dealloc; override;
{$ENDIF}
function acceptsFirstResponder: LCLObjCBoolean; override; function acceptsFirstResponder: LCLObjCBoolean; override;
function lclGetCallback: ICommonCallback; override; function lclGetCallback: ICommonCallback; override;
procedure lclClearCallback; override; procedure lclClearCallback; override;
@ -83,6 +89,10 @@ type
procedure scrollWheel(event: NSEvent); override; procedure scrollWheel(event: NSEvent); override;
procedure lclSetMaxLength(amax: integer); procedure lclSetMaxLength(amax: integer);
{$IFDEF COCOA_OVERRIDE_UNDOMANAGER}
function undoManagerForTextView(view: NSTextView): NSUndoManager; message 'undoManagerForTextView:';
{$ENDIF}
end; end;
{ TCocoaSecureTextField } { TCocoaSecureTextField }
@ -91,6 +101,10 @@ type
public public
maxLength: Integer; maxLength: Integer;
callback: ICommonCallback; callback: ICommonCallback;
{$IFDEF COCOA_OVERRIDE_UNDOMANAGER}
FUndoManager: NSUndoManager;
procedure dealloc; override;
{$ENDIF}
function acceptsFirstResponder: LCLObjCBoolean; override; function acceptsFirstResponder: LCLObjCBoolean; override;
function lclGetCallback: ICommonCallback; override; function lclGetCallback: ICommonCallback; override;
procedure lclClearCallback; override; procedure lclClearCallback; override;
@ -108,6 +122,9 @@ type
procedure scrollWheel(event: NSEvent); override; procedure scrollWheel(event: NSEvent); override;
procedure lclSetMaxLength(amax: integer); procedure lclSetMaxLength(amax: integer);
{$IFDEF COCOA_OVERRIDE_UNDOMANAGER}
function undoManagerForTextView(view: NSTextView): NSUndoManager; message 'undoManagerForTextView:';
{$ENDIF}
end; end;
{ TCocoaTextView } { TCocoaTextView }
@ -458,6 +475,20 @@ type
end; end;
{$ENDIF} {$ENDIF}
{ TCocoaUndoManager }
{$IFDEF COCOA_OVERRIDE_UNDOMANAGER}
TCocoaUndoManager = objcclass(NSUndoManager)
lastEvent: NSEvent; // weak reference
function init: id; override;
procedure undo; override;
procedure registerUndoWithTarget_selector_object(target: id; selector: SEL;
anObject: id); override;
procedure registerUndoWithTarget_handler(target: id;
handler: NSUndoManagerUndoWithTargetCBlock); override;
procedure lclCheckGrouping; message 'lclCheckGrouping';
end;
{$ENDIF}
// these constants are missing from CocoaAll for some reason // these constants are missing from CocoaAll for some reason
const const
NSTextAlignmentLeft = 0; NSTextAlignmentLeft = 0;
@ -983,6 +1014,15 @@ end;
{ TCocoaTextField } { TCocoaTextField }
{$IFDEF COCOA_OVERRIDE_UNDOMANAGER}
procedure TCocoaTextField.dealloc;
begin
if Assigned(FUndoManager) then
FUndoManager.release;
inherited dealloc;
end;
{$ENDIF}
function TCocoaTextField.acceptsFirstResponder: LCLObjCBoolean; function TCocoaTextField.acceptsFirstResponder: LCLObjCBoolean;
begin begin
Result := NSViewCanFocus(Self); Result := NSViewCanFocus(Self);
@ -1090,6 +1130,15 @@ begin
maxLength := amax; maxLength := amax;
end; end;
{$IFDEF COCOA_OVERRIDE_UNDOMANAGER}
function TCocoaTextField.undoManagerForTextView(view: NSTextView): NSUndoManager;
begin
if not Assigned(FUndoManager) then
FUndoManager := TCocoaUndoManager.alloc.init;
Result := FUndoManager;
end;
{$ENDIF}
{ TCocoaTextView } { TCocoaTextView }
procedure TCocoaTextView.changeColor(sender: id); procedure TCocoaTextView.changeColor(sender: id);
@ -1244,12 +1293,25 @@ end;
function TCocoaTextView.undoManagerForTextView(view: NSTextView): NSUndoManager; function TCocoaTextView.undoManagerForTextView(view: NSTextView): NSUndoManager;
begin begin
if not Assigned(FUndoManager) then if not Assigned(FUndoManager) then
{$IFDEF COCOA_OVERRIDE_UNDOMANAGER}
FUndoManager := TCocoaUndoManager.alloc.init;
{$ELSE}
FUndoManager := NSUndoManager.alloc.init; FUndoManager := NSUndoManager.alloc.init;
{$ENDIF}
Result := FUndoManager; Result := FUndoManager;
end; end;
{ TCocoaSecureTextField } { TCocoaSecureTextField }
{$IFDEF COCOA_OVERRIDE_UNDOMANAGER}
procedure TCocoaSecureTextField.dealloc;
begin
if Assigned(FUndoManager) then
FUndoManager.release;
inherited dealloc;
end;
{$ENDIF}
function TCocoaSecureTextField.acceptsFirstResponder: LCLObjCBoolean; function TCocoaSecureTextField.acceptsFirstResponder: LCLObjCBoolean;
begin begin
Result := NSViewCanFocus(Self); Result := NSViewCanFocus(Self);
@ -1338,6 +1400,15 @@ begin
MaxLength := amax; MaxLength := amax;
end; end;
{$IFDEF COCOA_OVERRIDE_UNDOMANAGER}
function TCocoaSecureTextField.undoManagerForTextView(view: NSTextView): NSUndoManager;
begin
if not Assigned(FUndoManager) then
FUndoManager := TCocoaUndoManager.alloc.init;
Result := FUndoManager;
end;
{$ENDIF}
{ TCocoaEditComboBoxList } { TCocoaEditComboBoxList }
procedure TCocoaEditComboBoxList.InsertItem(Index: Integer; const S: string; procedure TCocoaEditComboBoxList.InsertItem(Index: Integer; const S: string;
@ -2369,5 +2440,54 @@ end;
{$ENDIF} {$ENDIF}
{$IFDEF COCOA_OVERRIDE_UNDOMANAGER}
{ TCocoaUndoManager }
function TCocoaUndoManager.init: id;
begin
// This manages top-level undo groups automatically to work around an issue
// where, if we hijack the run loop, all undoable actions are combined into a
// single undo group. It isn't necessary for correct behavior in the other
// modes.
Result := inherited init;
Result.setGroupsByEvent(False);
end;
procedure TCocoaUndoManager.undo;
begin
if not groupsByEvent and (groupingLevel = 1) then
endUndoGrouping;
inherited;
end;
procedure TCocoaUndoManager.registerUndoWithTarget_selector_object(target: id;
selector: SEL; anObject: id);
begin
lclCheckGrouping;
inherited;
end;
procedure TCocoaUndoManager.registerUndoWithTarget_handler(target: id;
handler: NSUndoManagerUndoWithTargetCBlock);
begin
lclCheckGrouping;
inherited registerUndoWithTarget_handler(target, handler);
end;
procedure TCocoaUndoManager.lclCheckGrouping;
begin
if groupsByEvent or isUndoing or isRedoing then
Exit;
if (groupingLevel = 1) and (lastEvent <> NSApp.currentEvent) then
endUndoGrouping;
if groupingLevel = 0 then begin
lastEvent := NSApp.currentEvent;
beginUndoGrouping;
end;
end;
{$ENDIF}
end. end.

View File

@ -5008,7 +5008,7 @@ end;
procedure TQtWidget.setFocus; procedure TQtWidget.setFocus;
begin begin
if getFocusPolicy <> QtNoFocus then if getFocusPolicy <> QtNoFocus then
QWidget_setFocus(Widget, QtOtherFocusReason) {issue #10155} QWidget_setFocus(Widget, QtTabFocusReason) {issue #10155}
else else
QWidget_setFocus(Widget); QWidget_setFocus(Widget);
end; end;

View File

@ -5164,7 +5164,7 @@ end;
procedure TQtWidget.setFocus; procedure TQtWidget.setFocus;
begin begin
if getFocusPolicy <> QtNoFocus then if getFocusPolicy <> QtNoFocus then
QWidget_setFocus(Widget, QtOtherFocusReason) {issue #10155} QWidget_setFocus(Widget, QtTabFocusReason) {issue #10155}
else else
QWidget_setFocus(Widget); QWidget_setFocus(Widget);
end; end;

View File

@ -5168,7 +5168,7 @@ end;
procedure TQtWidget.setFocus; procedure TQtWidget.setFocus;
begin begin
if getFocusPolicy <> QtNoFocus then if getFocusPolicy <> QtNoFocus then
QWidget_setFocus(Widget, QtOtherFocusReason) {issue #10155} QWidget_setFocus(Widget, QtTabFocusReason) {issue #10155}
else else
QWidget_setFocus(Widget); QWidget_setFocus(Widget);
end; end;