From c2cd2c1231c7cb9c58463a62fb969e3b4909151e Mon Sep 17 00:00:00 2001 From: Kevin Rushforth Date: Thu, 4 Jan 2024 15:41:01 -0800 Subject: [PATCH] WIP: 8221261: fix IME deadlock when using JFXPanel -- call IME handler on AppKitThread --- .../sun/lwawt/macosx/CInputMethod.java | 38 ++++++++++++++++--- 1 file changed, 33 insertions(+), 5 deletions(-) diff --git a/src/java.desktop/macosx/classes/sun/lwawt/macosx/CInputMethod.java b/src/java.desktop/macosx/classes/sun/lwawt/macosx/CInputMethod.java index 64d8a101d69f6..f83bda20382a1 100644 --- a/src/java.desktop/macosx/classes/sun/lwawt/macosx/CInputMethod.java +++ b/src/java.desktop/macosx/classes/sun/lwawt/macosx/CInputMethod.java @@ -91,6 +91,29 @@ public class CInputMethod extends InputMethodAdapter { public CInputMethod() { } + private void trace(String meth) { + /* + System.err.println("CInputMethod::" + meth + " : " + + "fAwtFocussedComponent = " + fAwtFocussedComponent + " " + + "fIMContext = " + fIMContext); + */ + } + + private boolean isJFXPanel(Component component) { + // FIXME: need a more robust way to determine this + return "javafx.embed.swing.JFXPanel".equals(component.getClass().getName()); + } + + private void invokeAndWaitIME(Runnable runnable, Component component) + throws InvocationTargetException { + + if (isJFXPanel(component)) { + System.err.println("Detected JFXPanel...run directly"); + runnable.run(); + } else { + LWCToolkit.invokeAndWait(runnable, component); + } + } /** * Sets the input method context, which is used to dispatch input method @@ -587,7 +610,8 @@ private synchronized String attributedSubstringFromRange(final int locationIn, f final String[] retString = new String[1]; try { - LWCToolkit.invokeAndWait(new Runnable() { + trace("attributedSubstringFromRange"); + invokeAndWaitIME(new Runnable() { public void run() { synchronized(retString) { int location = locationIn; int length = lengthIn; @@ -639,7 +663,8 @@ private synchronized int[] selectedRange() { final int[] returnValue = new int[2]; try { - LWCToolkit.invokeAndWait(new Runnable() { + trace("selectedRange"); + invokeAndWaitIME(new Runnable() { public void run() { synchronized(returnValue) { AttributedCharacterIterator theIterator = fIMContext.getSelectedText(null); if (theIterator == null) { @@ -690,7 +715,8 @@ private synchronized int[] markedRange() { final int[] returnValue = new int[2]; try { - LWCToolkit.invokeAndWait(new Runnable() { + trace("markedRange"); + invokeAndWaitIME(new Runnable() { public void run() { synchronized(returnValue) { // The insert position is always after the composed text, so the range start is the // insert spot less the length of the composed text. @@ -714,7 +740,8 @@ private synchronized int[] firstRectForCharacterRange(final int absoluteTextOffs final int[] rect = new int[4]; try { - LWCToolkit.invokeAndWait(new Runnable() { + trace("firstRectForCharacterRange"); + invokeAndWaitIME(new Runnable() { public void run() { synchronized(rect) { int insertOffset = fIMContext.getInsertPositionOffset(); int composedTextOffset = absoluteTextOffset - insertOffset; @@ -758,7 +785,8 @@ private synchronized int characterIndexForPoint(final int screenX, final int scr final int[] insertPositionOffset = new int[1]; try { - LWCToolkit.invokeAndWait(new Runnable() { + trace("characterIndexForPoint"); + invokeAndWaitIME(new Runnable() { public void run() { synchronized(offsetInfo) { offsetInfo[0] = fIMContext.getLocationOffset(screenX, screenY); insertPositionOffset[0] = fIMContext.getInsertPositionOffset();