Compare commits

...

3 Commits

Author SHA1 Message Date
David Jenkins
8273a08530 Merge branch 'NewPostSyncCheckEvent' into 'main'
Cocoa: Hybrid lclSyncCheck posting for performance and app response during modal loops

See merge request freepascal.org/lazarus/lazarus!384
2025-04-03 20:21:19 +00:00
zeljan1
91fbaca370 Qt5,Qt6: Fixed QLineEdit behaviour and automatic selectAll() by Qt when control is focused.Related issue #10155 and issue #41562 2025-04-03 21:57:50 +02:00
David Jenkins
b692996666 Cocoa: Hybrid lclSyncCheck posting for performance and app response during modal loops
hole bunches of revisions on this one: r24031, 24036, 24045, 24048, 24053, 24054, 24055, 24059

We were trying to solved issues we say with mouse tracking and button clicking while BC was performing background tasks and also the rest of BC becoming non-responsive to the user when modal-loops where in effect.

Original code posted app specific syncCheckEvents to the main queue in TCocoaWidgetSet.SendCheckSyncronizeMessage.  When BC was running background tasks we would fill up the queue with those events and end up causing flaking mouse drag / button click performance.

Laz trunks response to that was to switch away from posting the events and instead in SendCheckSynchronizeMessage() do a call to performSelectorOnMainThread('lclSyncCheck').

This gives good performance when running our background tasks for performSelectorOnMainThread does not allow for calls to lclSyncCheck when modal loops are running.  Dmitri was not concerned as that is standard Mac behavior and so the trunk continues with that method.

However, our users expect BC to still be responsive in the main text window when other forms are open.  So we have implemented a hybrid method:

We post events to the main queue so that are seen even when in modal loops but we only post one at a time so the queue isn't stormed.  When there is an existing event in the queue we get better responsiveness by calling performSelectOrOnMainThread.

When we send an event through the queue we need to recognize it nextEventMatching so that we can do the call to lclSyncEvent and we need to know that an event has been queued.

The event queue receives NSEvents but stores Carbon Events - dropping the initial NSEvent instance.  So the NSEvent that is dequeued is not the same instance of the one that entered.   So a direct EventSent == EventDqueued does not work.  Nor does sending a certaint Type of event (we cannot guarantee it is the exact one we sent).

However, the event queue does carry around a static event ref that we can use.  syncCheckEventRef is the EventRef of the NSEvent we create and post.  Inside of nextEventMatching we check the dq'd event ref to find our event when it has come through.  Meanwhile syncCheckEventRef <> nil is our indication that a event is on the queue and no more should be queued until that one clears.

It is possible for the queue to be flushed and our event disappears. This would be a problem as we would never set syncCheckeventRef to nil and never post new events.  So we use an event of type NSPeriodic and even if it is flushed off the queue, the NSPeriodic mask will still be set and we can use:

(Event = nil) and (Mask == NSPeriodic)

to indicate that an event 'was' there but should now be cleared.

lclPostSyncCheckEvent handles either posting an event or calling performSelectOrOnMainThread based on whether there is an event in the queue or not.  This is called in SendCheckSynchronizeMessage instead of the direct call to performSelectOrOnMainThread.

nextEventMatching() now checks for an Event with ref that matches syncCheckEventRef or a nil event with mask == NSPeriodic to indicate that syncCheckEventRef should be cleared and lclSyncCheck() called.
2024-11-15 13:28:23 +00:00
4 changed files with 37 additions and 5 deletions

View File

@ -69,6 +69,7 @@ type
{ TCocoaApplication } { TCocoaApplication }
TCocoaApplication = objcclass(NSApplication) TCocoaApplication = objcclass(NSApplication)
syncCheckEventRef: EventRef;
aloop : TApplicationMainLoop; aloop : TApplicationMainLoop;
isrun : Boolean; isrun : Boolean;
modals : NSMutableDictionary; modals : NSMutableDictionary;
@ -89,6 +90,7 @@ type
function nextEventMatchingMask_untilDate_inMode_dequeue(mask: NSUInteger; expiration: NSDate; mode: NSString; deqFlag: LCLObjCBoolean): NSEvent; override; function nextEventMatchingMask_untilDate_inMode_dequeue(mask: NSUInteger; expiration: NSDate; mode: NSString; deqFlag: LCLObjCBoolean): NSEvent; override;
function runModalForWindow(theWindow: NSWindow): NSInteger; override; function runModalForWindow(theWindow: NSWindow): NSInteger; override;
procedure lclPostSyncCheckEvent; message 'lclPostSyncCheckEvent';
procedure lclSyncCheck(arg: id); message 'lclSyncCheck:'; procedure lclSyncCheck(arg: id); message 'lclSyncCheck:';
{$ifdef COCOAPPRUNNING_OVERRIDEPROPERTY} {$ifdef COCOAPPRUNNING_OVERRIDEPROPERTY}
function isRunning: objc.ObjCBOOL; override; function isRunning: objc.ObjCBOOL; override;
@ -687,6 +689,13 @@ begin
Exit; Exit;
end; end;
if (not Assigned(Result) and ((mask and NSPeriodicMask) <> 0)) or
(Assigned(Result) and (Result.eventRef = syncCheckEventRef)) then
begin
syncCheckEventRef := nil;
lclSyncCheck(nil);
end;
if (Result.type_=NSApplicationDefined) and (Result.subtype=LazarusApplicationDefinedSubtypeWakeup) then if (Result.type_=NSApplicationDefined) and (Result.subtype=LazarusApplicationDefinedSubtypeWakeup) then
Result:= nil; Result:= nil;
@ -730,6 +739,21 @@ begin
Result:=inherited runModalForWindow(theWindow); Result:=inherited runModalForWindow(theWindow);
end; end;
procedure TCocoaApplication.lclPostSyncCheckEvent;
var ev: NSEvent;
begin
if not Assigned(syncCheckEventRef) then
begin
ev := NSEvent.otherEventWithType_location_modifierFlags_timestamp_windowNumber_context_subtype_data1_data2(
NSPeriodic, NSPoint(CGPointZero), 0, 0, 0, nil, 0, 0, 0);
syncCheckEventRef := ev.eventRef;
postEvent_atStart(ev, False);
end
else
performSelectorOnMainThread_withObject_waitUntilDone(
ObjCSelector('lclSyncCheck:'), nil, false);
end;
procedure TCocoaApplication.lclSyncCheck(arg: id); procedure TCocoaApplication.lclSyncCheck(arg: id);
begin begin
{$ifdef COCOALOOPNATIVE} {$ifdef COCOALOOPNATIVE}

View File

@ -50,9 +50,7 @@ end;
procedure TCocoaWidgetSet.SendCheckSynchronizeMessage; procedure TCocoaWidgetSet.SendCheckSynchronizeMessage;
begin begin
InitApplication TCocoaApplication(NSApp).lclPostSyncCheckEvent;
.performSelectorOnMainThread_withObject_waitUntilDone(
ObjCSelector('lclSyncCheck:'), nil, false);
end; end;
{------------------------------------------------------------------------------ {------------------------------------------------------------------------------

View File

@ -107,6 +107,7 @@ type
TQtWidget = class(TQtObject, IUnknown) TQtWidget = class(TQtObject, IUnknown)
private private
FDefaultFocusReason: QtFocusReason;
FInResizeEvent: boolean; FInResizeEvent: boolean;
FWidgetState: TQtWidgetStates; FWidgetState: TQtWidgetStates;
FWidgetDefaultFont: TQtFont; FWidgetDefaultFont: TQtFont;
@ -319,6 +320,7 @@ type
nil): QPixmapH; nil): QPixmapH;
property ChildOfComplexWidget: TChildOfComplexWidget read FChildOfComplexWidget write FChildOfComplexWidget; property ChildOfComplexWidget: TChildOfComplexWidget read FChildOfComplexWidget write FChildOfComplexWidget;
property Context: HDC read GetContext; property Context: HDC read GetContext;
property DefaultFocusReason: QtFocusReason read FDefaultFocusReason write FDefaultFocusReason;
property HasCaret: Boolean read FHasCaret write SetHasCaret; property HasCaret: Boolean read FHasCaret write SetHasCaret;
property HasPaint: Boolean read FHasPaint write FHasPaint; property HasPaint: Boolean read FHasPaint write FHasPaint;
property InResizeEvent: boolean read FInResizeEvent write FInResizeEvent; property InResizeEvent: boolean read FInResizeEvent write FInResizeEvent;
@ -2161,6 +2163,7 @@ end;
procedure TQtWidget.InitializeWidget; procedure TQtWidget.InitializeWidget;
begin begin
FDefaultFocusReason := QtTabFocusReason;
FInResizeEvent := False; FInResizeEvent := False;
// default states // default states
FWidgetState := []; FWidgetState := [];
@ -5164,7 +5167,7 @@ end;
procedure TQtWidget.setFocus; procedure TQtWidget.setFocus;
begin begin
if getFocusPolicy <> QtNoFocus then if getFocusPolicy <> QtNoFocus then
QWidget_setFocus(Widget, QtTabFocusReason) {issue #10155} QWidget_setFocus(Widget, FDefaultFocusReason) {issue #10155}
else else
QWidget_setFocus(Widget); QWidget_setFocus(Widget);
end; end;
@ -9769,6 +9772,7 @@ begin
FCachedSelectionLen := -1; FCachedSelectionLen := -1;
FIntValidator := nil; FIntValidator := nil;
FNumbersOnly := False; FNumbersOnly := False;
FDefaultFocusReason := QtOtherFocusReason;
if AParams.WndParent <> 0 then if AParams.WndParent <> 0 then
Parent := TQtWidget(AParams.WndParent).GetContainerWidget Parent := TQtWidget(AParams.WndParent).GetContainerWidget
else else
@ -16412,6 +16416,7 @@ end;
procedure TQtMenu.InitializeWidget; procedure TQtMenu.InitializeWidget;
begin begin
FDefaultFocusReason := QtTabFocusReason;
FWidgetState := []; FWidgetState := [];
ChildOfComplexWidget := ccwNone; ChildOfComplexWidget := ccwNone;
WidgetColorRole := QPaletteWindow; WidgetColorRole := QPaletteWindow;

View File

@ -107,6 +107,7 @@ type
TQtWidget = class(TQtObject, IUnknown) TQtWidget = class(TQtObject, IUnknown)
private private
FDefaultFocusReason: QtFocusReason;
FInResizeEvent: boolean; FInResizeEvent: boolean;
FWidgetState: TQtWidgetStates; FWidgetState: TQtWidgetStates;
FWidgetDefaultFont: TQtFont; FWidgetDefaultFont: TQtFont;
@ -316,6 +317,7 @@ type
nil): QPixmapH; nil): QPixmapH;
property ChildOfComplexWidget: TChildOfComplexWidget read FChildOfComplexWidget write FChildOfComplexWidget; property ChildOfComplexWidget: TChildOfComplexWidget read FChildOfComplexWidget write FChildOfComplexWidget;
property Context: HDC read GetContext; property Context: HDC read GetContext;
property DefaultFocusReason: QtFocusReason read FDefaultFocusReason write FDefaultFocusReason;
property HasCaret: Boolean read FHasCaret write SetHasCaret; property HasCaret: Boolean read FHasCaret write SetHasCaret;
property HasPaint: Boolean read FHasPaint write FHasPaint; property HasPaint: Boolean read FHasPaint write FHasPaint;
property InResizeEvent: boolean read FInResizeEvent write FInResizeEvent; property InResizeEvent: boolean read FInResizeEvent write FInResizeEvent;
@ -2158,6 +2160,7 @@ end;
procedure TQtWidget.InitializeWidget; procedure TQtWidget.InitializeWidget;
begin begin
FDefaultFocusReason := QtTabFocusReason;
FInResizeEvent := False; FInResizeEvent := False;
// default states // default states
FWidgetState := []; FWidgetState := [];
@ -5168,7 +5171,7 @@ end;
procedure TQtWidget.setFocus; procedure TQtWidget.setFocus;
begin begin
if getFocusPolicy <> QtNoFocus then if getFocusPolicy <> QtNoFocus then
QWidget_setFocus(Widget, QtTabFocusReason) {issue #10155} QWidget_setFocus(Widget, FDefaultFocusReason) {issue #10155}
else else
QWidget_setFocus(Widget); QWidget_setFocus(Widget);
end; end;
@ -9725,6 +9728,7 @@ begin
FCachedSelectionLen := -1; FCachedSelectionLen := -1;
FIntValidator := nil; FIntValidator := nil;
FNumbersOnly := False; FNumbersOnly := False;
FDefaultFocusReason := QtOtherFocusReason;
if AParams.WndParent <> 0 then if AParams.WndParent <> 0 then
Parent := TQtWidget(AParams.WndParent).GetContainerWidget Parent := TQtWidget(AParams.WndParent).GetContainerWidget
else else
@ -16322,6 +16326,7 @@ end;
procedure TQtMenu.InitializeWidget; procedure TQtMenu.InitializeWidget;
begin begin
FDefaultFocusReason := QtTabFocusReason;
FWidgetState := []; FWidgetState := [];
ChildOfComplexWidget := ccwNone; ChildOfComplexWidget := ccwNone;
WidgetColorRole := QPaletteWindow; WidgetColorRole := QPaletteWindow;