- 
                Notifications
    You must be signed in to change notification settings 
- Fork 541
8359899: Stage.isFocused() returns invalid value when Stage fails to receive focus #1849
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
base: master
Are you sure you want to change the base?
Conversation
| 👋 Welcome back lkostyra! A progress list of the required criteria for merging this PR into  | 
| ❗ This change is not yet ready to be integrated. | 
| @lukostyra This pull request has been inactive for more than 4 weeks and will be automatically closed if another 4 weeks passes without any activity. To avoid this, simply issue a  | 
| /keepalive | 
| @lukostyra The pull request is being re-evaluated and the inactivity timeout has been reset. | 
| I’ve been testing this PR by launching the manual KeyboardTest app. The call to SetForegroundWindow fails but not before activating the window. Since it’s not the foreground window this PR correctly sets the JavaFX focused property back to false. But when I later click on the title bar to bring the window to the foreground the focused property is not updated. I’m not sure how we’re expected to detect that our HWND has come to the foreground. There’s no specific message sent when this happens and since the OS thinks the window is already active and focused we don’t get WM_ACTIVATE or WM_SETFOCUS. There’s some messages we could use heuristically (like WM_NCACTIVATE) but I couldn’t find anything more clear cut. I'll keep looking but it doesn't look like Windows makes this easy. | 
| @beldenfox thanks for checking. I remember doing a fair share of looking for a reliable answer to this and couldn't find anything specific, other that SetForegroundWindow being able to fail and having to "handle it". I'll get back to testing this starting next week as well, maybe we'll manage to track down some more reasonable solution. | 
| @lukostyra This pull request has been inactive for more than 4 weeks and will be automatically closed if another 4 weeks passes without any activity. To avoid this, simply issue a  | 
| /touch I didn't get a chance to look into other solutions yet, but I will be investigating that soon. | 
| @lukostyra The pull request is being re-evaluated and the inactivity timeout has been reset. | 
This PR fixes
isFocused()returning invalid value when Stage fails to receive focus after callingStage.show(). This problem is Windows-only.In Windows the
SetForegroundWindow()API lists a set of conditions to successfully grant focus to a Window. If any of the conditions are not met, the API will return FALSE. JavaFX did not respect that and instead assumed that receivingWM_ACTIVATEwith our Window being activated is enough to assume the Window is in focus (which in some cases is not true).I first tried reacting to
WM_SETFOCUSandWM_KILLFOCUSbut it seems those messages are not sent when the window is shown for the first time (insteadWM_ACTIVATEis used).To correct this behavior, I noticed the following path is the most reliable:
ShowWindow()usingSW_SHOWNAinstead ofSW_SHOW- that makes the window visible but does NOT activate itSetForegroundWindow()- that will attempt to give the Window focus and will also activate it if it is successfulnotifyFocuscallback will be called viaWM_ACTIVATEhandlernotifyFocuscallback manually informing the upper layers the focus is lost. This establishes the correct state ofWindow.focusedproperty.With this change I observed that all tests pass as intended as long as two conditions are met (these are needed to satisfy
SetForegroundWindow()restrictions):If any of above two conditions is not met, some tests (including canary test from #1804) now timeout/fail when checking whether
Window.isFocused()is true.Manually started JavaFX apps (ex. Ensemble) run as they used to and still receive focus upon Stage showing.
Progress
Issue
Reviewing
Using
gitCheckout this PR locally:
$ git fetch https://git.openjdk.org/jfx.git pull/1849/head:pull/1849$ git checkout pull/1849Update a local copy of the PR:
$ git checkout pull/1849$ git pull https://git.openjdk.org/jfx.git pull/1849/headUsing Skara CLI tools
Checkout this PR locally:
$ git pr checkout 1849View PR using the GUI difftool:
$ git pr show -t 1849Using diff file
Download this PR as a diff file:
https://git.openjdk.org/jfx/pull/1849.diff
Using Webrev
Link to Webrev Comment