Merge branch 'LoopHijackUndoRedo' into 'main'

Cocoa: Improve Undo/Redo handling for when COCOALOOPHIJACK is defined.

See merge request freepascal.org/lazarus/lazarus!361
This commit is contained in:
David Jenkins 2025-04-03 20:21:19 +00:00
commit cd7aa23874
2 changed files with 131 additions and 2 deletions

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.