mirror of
https://gitlab.com/freepascal.org/lazarus/lazarus.git
synced 2025-04-05 16:37:54 +02:00
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:
commit
cd7aa23874
@ -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;
|
||||||
|
@ -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.
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user