summaryrefslogtreecommitdiff
path: root/vcl
diff options
context:
space:
mode:
authorPatrick Luby <guibmacdev@gmail.com>2024-11-05 19:51:55 -0500
committerMichael Weghorn <m.weghorn@posteo.de>2024-11-11 13:26:32 +0100
commita684bf03e622f64729c885053969c3837dda7efd (patch)
tree3db446a78498aeff4b0a68c64a9f7c120d91baa5 /vcl
parentf761d098e9a0960554aa4fc02f84a711b50a1cff (diff)
tdf#163764 Force pending timers to run after marked text changes
During native dictation, waiting for the next native event is blocked while dictation runs in a loop within a native callback. Because of this, LibreOffice's painting timers won't fire until dictation is cancelled or the user pauses speaking. So, force any pending timers to fire after the marked text changes. Also, remove the fix for OpenOffice bug 106901 as causes any key down events to cancel dictation and removing the fix does not appear to cause the original crashing bug to reoccur. Note: inserting a character from the system Character Viewer window causes dictation to stop responding and the only way I have found to restart dictation is by toggling dictation off and then back on again. This same behavior occurs in Apple's TextEdit application so this appears to be a macOS bug. Change-Id: Iad2b54870ff1a315f2f71d72bef24af3cea808e6 Reviewed-on: https://gerrit.libreoffice.org/c/core/+/176100 Reviewed-by: Michael Weghorn <m.weghorn@posteo.de> Tested-by: Jenkins Reviewed-by: Patrick Luby <guibomacdev@gmail.com> (cherry picked from commit 62baf23796132d6c0e17ae1ff37c3b9ee0556cf0) Reviewed-on: https://gerrit.libreoffice.org/c/core/+/176366
Diffstat (limited to 'vcl')
-rw-r--r--vcl/inc/svdata.hxx1
-rw-r--r--vcl/osx/salframeview.mm92
-rw-r--r--vcl/osx/salinst.cxx4
-rw-r--r--vcl/osx/saltimer.cxx4
-rw-r--r--vcl/source/app/scheduler.cxx2
5 files changed, 59 insertions, 44 deletions
diff --git a/vcl/inc/svdata.hxx b/vcl/inc/svdata.hxx
index 3619de00d25b..cf1e707f0e26 100644
--- a/vcl/inc/svdata.hxx
+++ b/vcl/inc/svdata.hxx
@@ -275,6 +275,7 @@ struct ImplSVWinData
bool mbNoDeactivate = false; // true: do not execute Deactivate
bool mbNoSaveFocus = false; // true: menus must not save/restore focus
bool mbIsLiveResize = false; // true: skip waiting for events and low priority timers
+ bool mbIsWaitingForNativeEvent = false; // true: code is executing via a native callback while waiting for the next native event
};
typedef std::vector< std::pair< OUString, FieldUnit > > FieldUnitStringList;
diff --git a/vcl/osx/salframeview.mm b/vcl/osx/salframeview.mm
index 950cec3271eb..15228b7b5e91 100644
--- a/vcl/osx/salframeview.mm
+++ b/vcl/osx/salframeview.mm
@@ -212,6 +212,32 @@ static void updateWinDataInLiveResize(bool bInLiveResize)
}
}
+static void freezeWindowSizeAndReschedule( NSWindow *pWindow )
+{
+ if ( pWindow )
+ {
+ // Application::Reschedule() can potentially display a modal
+ // dialog which will cause a hang so temporarily disable any
+ // resizing by clamping the window's minimum and maximum sizes
+ // to the current frame size which in Application::Reschedule().
+ bool bIsLiveResize = ImplGetSVData()->mpWinData->mbIsLiveResize;
+ NSSize aMinSize = [pWindow minSize];
+ NSSize aMaxSize = [pWindow maxSize];
+ if ( bIsLiveResize )
+ {
+ NSRect aFrame = [pWindow frame];
+ [pWindow setMinSize:aFrame.size];
+ [pWindow setMaxSize:aFrame.size];
+ }
+ Application::Reschedule( true );
+ if ( bIsLiveResize )
+ {
+ [pWindow setMinSize:aMinSize];
+ [pWindow setMaxSize:aMaxSize];
+ }
+ }
+}
+
@interface NSResponder (SalFrameWindow)
-(BOOL)accessibilityIsIgnored;
@end
@@ -453,18 +479,7 @@ static void updateWinDataInLiveResize(bool bInLiveResize)
// not trigger redrawing with the new size.
// Instead, force relayout by dispatching all pending internal
// events and firing any pending timers.
- // Also, Application::Reschedule() can potentially display a
- // modal dialog which will cause a hang so temporarily disable
- // live resize by clamping the window's minimum and maximum sizes
- // to the current frame size which in Application::Reschedule().
- NSRect aFrame = [self frame];
- NSSize aMinSize = [self minSize];
- NSSize aMaxSize = [self maxSize];
- [self setMinSize:aFrame.size];
- [self setMaxSize:aFrame.size];
- Application::Reschedule( true );
- [self setMinSize:aMinSize];
- [self setMaxSize:aMaxSize];
+ freezeWindowSizeAndReschedule( self );
if ( ImplGetSVData()->mpWinData->mbIsLiveResize )
{
@@ -1930,36 +1945,11 @@ static void updateWinDataInLiveResize(bool bInLiveResize)
mbNeedSpecialKeyHandle = true;
}
- // FIXME:
- // #i106901#
- // if we come here outside of mbInKeyInput, this is likely to be because
- // of the keyboard viewer. For unknown reasons having no marked range
- // in this case causes a crash. So we say we have a marked range anyway
- // This is a hack, since it is not understood what a) causes that crash
- // and b) why we should have a marked range at this point.
- if( ! mbInKeyInput )
- bHasMarkedText = true;
-
return bHasMarkedText;
}
- (NSRange)markedRange
{
- // FIXME:
- // #i106901#
- // if we come here outside of mbInKeyInput, this is likely to be because
- // of the keyboard viewer. For unknown reasons having no marked range
- // in this case causes a crash. So we say we have a marked range anyway
- // This is a hack, since it is not understood what a) causes that crash
- // and b) why we should have a marked range at this point. Stop the native
- // input method popup from appearing in the bottom left corner of the
- // screen by returning the marked range if is valid when called outside of
- // mbInKeyInput. If a zero length range is returned, macOS won't call
- // [self firstRectForCharacterRange:actualRange:] for any newly appended
- // uncommitted text.
- if( ! mbInKeyInput )
- return mMarkedRange.location != NSNotFound ? mMarkedRange : NSMakeRange( 0, 0 );
-
return [self hasMarkedText] ? mMarkedRange : NSMakeRange( NSNotFound, 0 );
}
@@ -1988,6 +1978,7 @@ static void updateWinDataInLiveResize(bool bInLiveResize)
[self unmarkText];
int len = [aString length];
+ bool bReschedule = false;
SalExtTextInputEvent aInputEvent;
if( len > 0 ) {
// Set the marked and selected ranges to the marked text and selected
@@ -2052,16 +2043,35 @@ static void updateWinDataInLiveResize(bool bInLiveResize)
aInputEvent.mnCursorPos = nSelectionStart;
aInputEvent.mnCursorFlags = 0;
aInputEvent.mpTextAttr = aInputFlags.data();
- mpFrame->CallCallback( SalEvent::ExtTextInput, static_cast<void *>(&aInputEvent) );
+ if( AquaSalFrame::isAlive( mpFrame ) )
+ {
+ bReschedule = true;
+ mpFrame->CallCallback( SalEvent::ExtTextInput, static_cast<void *>(&aInputEvent) );
+ }
} else {
aInputEvent.maText.clear();
aInputEvent.mnCursorPos = 0;
aInputEvent.mnCursorFlags = 0;
aInputEvent.mpTextAttr = nullptr;
- mpFrame->CallCallback( SalEvent::ExtTextInput, static_cast<void *>(&aInputEvent) );
- mpFrame->CallCallback( SalEvent::EndExtTextInput, nullptr );
+ if( AquaSalFrame::isAlive( mpFrame ) )
+ {
+ bReschedule = true;
+ mpFrame->CallCallback( SalEvent::ExtTextInput, static_cast<void *>(&aInputEvent) );
+ if( AquaSalFrame::isAlive( mpFrame ) )
+ mpFrame->CallCallback( SalEvent::EndExtTextInput, nullptr );
+ }
}
- mbKeyHandled= true;
+
+ // tdf#163764 force pending timers to run after marked text changes
+ // During native dictation, waiting for the next native event is
+ // blocked while dictation runs in a loop within a native callback.
+ // Because of this, LibreOffice's painting timers won't fire until
+ // dictation is cancelled or the user pauses speaking. So, force
+ // any pending timers to fire after the marked text changes.
+ if( bReschedule && ImplGetSVData()->mpWinData->mbIsWaitingForNativeEvent )
+ freezeWindowSizeAndReschedule( [self window] );
+
+ mbKeyHandled = true;
}
- (void)unmarkText
diff --git a/vcl/osx/salinst.cxx b/vcl/osx/salinst.cxx
index 7196c2aa6c69..42a8efe029e4 100644
--- a/vcl/osx/salinst.cxx
+++ b/vcl/osx/salinst.cxx
@@ -636,6 +636,8 @@ bool AquaSalInstance::DoYield(bool bWait, bool bHandleAllCurrentEvents)
// Some events and timers call Application::Reschedule() or
// Application::Yield() so don't block and wait for events when a
// window is in live resize
+ bool bOldIsWaitingForNativeEvent = ImplGetSVData()->mpWinData->mbIsWaitingForNativeEvent;
+ ImplGetSVData()->mpWinData->mbIsWaitingForNativeEvent = !SalInstance::IsRunningUnitTest();
if( bWait && ! bHadEvent && !ImplGetSVData()->mpWinData->mbIsLiveResize )
{
SolarMutexReleaser aReleaser;
@@ -659,6 +661,8 @@ bool AquaSalInstance::DoYield(bool bWait, bool bHandleAllCurrentEvents)
[NSApp updateWindows];
}
+ ImplGetSVData()->mpWinData->mbIsWaitingForNativeEvent = bOldIsWaitingForNativeEvent;
+
// collect update rectangles
for( auto pSalFrame : GetSalData()->mpInstance->getFrames() )
{
diff --git a/vcl/osx/saltimer.cxx b/vcl/osx/saltimer.cxx
index 8af7de217678..fffd1c63b92c 100644
--- a/vcl/osx/saltimer.cxx
+++ b/vcl/osx/saltimer.cxx
@@ -83,7 +83,7 @@ void AquaSalTimer::Start( sal_uInt64 nMS )
return;
}
- m_bDirectTimeout = (0 == nMS) && !ImplGetSVData()->mpWinData->mbIsLiveResize;
+ m_bDirectTimeout = (0 == nMS) && !ImplGetSVData()->mpWinData->mbIsLiveResize && !ImplGetSVData()->mpWinData->mbIsWaitingForNativeEvent;
if ( m_bDirectTimeout )
Stop();
else
@@ -142,7 +142,7 @@ void AquaSalTimer::callTimerCallback()
void AquaSalTimer::handleTimerElapsed()
{
- if ( m_bDirectTimeout || ImplGetSVData()->mpWinData->mbIsLiveResize )
+ if ( m_bDirectTimeout || ImplGetSVData()->mpWinData->mbIsLiveResize || ImplGetSVData()->mpWinData->mbIsWaitingForNativeEvent )
{
// Stop the timer, as it is just invalidated after the firing function
Stop();
diff --git a/vcl/source/app/scheduler.cxx b/vcl/source/app/scheduler.cxx
index e9f4057cd47c..eaca69adafa1 100644
--- a/vcl/source/app/scheduler.cxx
+++ b/vcl/source/app/scheduler.cxx
@@ -389,7 +389,7 @@ void Scheduler::CallbackTaskScheduling()
// Only higher priority tasks need to be fired to redraw the window
// so skip firing potentially long-running tasks, such as the Writer
// idle layout timer, when a window is in live resize
- if ( ImplGetSVData()->mpWinData->mbIsLiveResize && nTaskPriority == static_cast<int>(TaskPriority::LOWEST) )
+ if ( nTaskPriority == static_cast<int>(TaskPriority::LOWEST) && ( ImplGetSVData()->mpWinData->mbIsLiveResize || ImplGetSVData()->mpWinData->mbIsWaitingForNativeEvent ) )
continue;
pSchedulerData = rSchedCtx.mpFirstSchedulerData[nTaskPriority];