fix: notify focus change right away rather not on next tick (#14453)

* fix: Notify focus change right away, not on next tick

* fix: emit the JS blur/focus events on next tick to avoid race condition

* address feedback from review

* fix: bind deferred Emit() calls to a WeakPtr

This is so that the deferred Emit() calls will be canceled
if the TopLevelWindow is destroyed.

* chore: remove wip/test code cruft

* fix: make linter happy

* Enable disabled tests

* refactor: cleaner impl of EmitEventSoon()

* Revert "Merge branch 'fix-win-focus' of github.com:electron/electron into fix-win-focus"

This reverts commit 90576806eb271d059f0a215c67e9b4b04f7396a4, reversing
changes made to 9c13e47779a3af78fe0970c1f3d6cd040a5354e6.

* Restore 704722c1, which was removed in error.

We apologise again for the fault in the subtitles. Those responsible for sacking the people who have just been sacked have been sacked.
This commit is contained in:
Nitish Sakhawalkar 2018-09-07 11:21:58 -07:00 committed by Charles Kerr
parent e96433243c
commit a2ab0d8ebe
5 changed files with 21 additions and 21 deletions

View file

@ -163,11 +163,11 @@ void TopLevelWindow::OnWindowEndSession() {
} }
void TopLevelWindow::OnWindowBlur() { void TopLevelWindow::OnWindowBlur() {
Emit("blur"); EmitEventSoon("blur");
} }
void TopLevelWindow::OnWindowFocus() { void TopLevelWindow::OnWindowFocus() {
Emit("focus"); EmitEventSoon("focus");
} }
void TopLevelWindow::OnWindowShow() { void TopLevelWindow::OnWindowShow() {

View file

@ -14,6 +14,7 @@
#include "atom/browser/native_window.h" #include "atom/browser/native_window.h"
#include "atom/browser/native_window_observer.h" #include "atom/browser/native_window_observer.h"
#include "atom/common/api/atom_api_native_image.h" #include "atom/common/api/atom_api_native_image.h"
#include "content/public/browser/browser_thread.h"
#include "native_mate/handle.h" #include "native_mate/handle.h"
namespace atom { namespace atom {
@ -222,6 +223,14 @@ class TopLevelWindow : public mate::TrackableObject<TopLevelWindow>,
// Remove this window from parent window's |child_windows_|. // Remove this window from parent window's |child_windows_|.
void RemoveFromParentChildWindows(); void RemoveFromParentChildWindows();
template<typename... Args>
void EmitEventSoon(base::StringPiece eventName) {
content::BrowserThread::PostTask(
content::BrowserThread::UI, FROM_HERE,
base::BindOnce(base::IgnoreResult(&TopLevelWindow::Emit<Args...>),
weak_factory_.GetWeakPtr(), eventName));
}
#if defined(OS_WIN) #if defined(OS_WIN)
typedef std::map<UINT, MessageCallback> MessageCallbackMap; typedef std::map<UINT, MessageCallback> MessageCallbackMap;
MessageCallbackMap messages_callback_map_; MessageCallbackMap messages_callback_map_;

View file

@ -1109,12 +1109,10 @@ void NativeWindowViews::OnWidgetActivationChanged(views::Widget* changed_widget,
if (changed_widget != widget()) if (changed_widget != widget())
return; return;
// Post the notification to next tick. if (active)
content::BrowserThread::PostTask( NativeWindow::NotifyWindowFocus();
content::BrowserThread::UI, FROM_HERE, else
base::Bind(active ? &NativeWindow::NotifyWindowFocus NativeWindow::NotifyWindowBlur();
: &NativeWindow::NotifyWindowBlur,
GetWeakPtr()));
// Hide menu bar when window is blured. // Hide menu bar when window is blured.
if (!active && IsMenuBarAutoHide() && IsMenuBarVisible()) if (!active && IsMenuBarAutoHide() && IsMenuBarVisible())

View file

@ -410,8 +410,7 @@ describe('BrowserWindow module', () => {
}) })
}) })
// TODO(alexeykuzmin): [Ch66] Enable the test. Passes locally. describe('BrowserWindow.getFocusedWindow()', (done) => {
xdescribe('BrowserWindow.getFocusedWindow()', (done) => {
it('returns the opener window when dev tools window is focused', (done) => { it('returns the opener window when dev tools window is focused', (done) => {
w.show() w.show()
w.webContents.once('devtools-focused', () => { w.webContents.once('devtools-focused', () => {
@ -728,8 +727,7 @@ describe('BrowserWindow module', () => {
}) })
}) })
// TODO(alexeykuzmin): [Ch66] Enable the test. Passes locally. describe('BrowserWindow.alwaysOnTop() resets level on minimize', () => {
xdescribe('BrowserWindow.alwaysOnTop() resets level on minimize', () => {
before(function () { before(function () {
if (process.platform !== 'darwin') { if (process.platform !== 'darwin') {
this.skip() this.skip()
@ -1966,8 +1964,7 @@ describe('BrowserWindow module', () => {
}) })
}) })
// TODO(alexeykuzmin): [Ch66] Enable the tests. They pass locally. describe('document.visibilityState/hidden', () => {
xdescribe('document.visibilityState/hidden', () => {
beforeEach(() => { w.destroy() }) beforeEach(() => { w.destroy() })
function onVisibilityChange (callback) { function onVisibilityChange (callback) {
@ -2571,7 +2568,6 @@ describe('BrowserWindow module', () => {
}) })
}) })
// TODO(alexeykuzmin): [Ch66] Enable the test. It passes locally.
describe('kiosk state', () => { describe('kiosk state', () => {
before(function () { before(function () {
// Only implemented on macOS. // Only implemented on macOS.
@ -2619,8 +2615,7 @@ describe('BrowserWindow module', () => {
}) })
}) })
// TODO(alexeykuzmin): [Ch66] Enable the tests. They pass locally. describe('fullscreen state', () => {
xdescribe('fullscreen state', () => {
before(function () { before(function () {
// Only implemented on macOS. // Only implemented on macOS.
if (process.platform !== 'darwin') { if (process.platform !== 'darwin') {
@ -2768,8 +2763,7 @@ describe('BrowserWindow module', () => {
} }
}) })
// TODO(alexeykuzmin): [Ch66] Enable the test. Fails on CI bots, passes locally. it('exits HTML fullscreen when window leaves fullscreen', (done) => {
xit('exits HTML fullscreen when window leaves fullscreen', (done) => {
w.destroy() w.destroy()
w = new BrowserWindow() w = new BrowserWindow()
w.webContents.once('did-finish-load', () => { w.webContents.once('did-finish-load', () => {

View file

@ -363,8 +363,7 @@ describe('webContents module', () => {
}) })
}) })
// TODO(alexeykuzmin): [Ch66] Enable the test. Passes locally. describe('focus()', () => {
xdescribe('focus()', () => {
describe('when the web contents is hidden', () => { describe('when the web contents is hidden', () => {
it('does not blur the focused window', (done) => { it('does not blur the focused window', (done) => {
ipcMain.once('answer', (event, parentFocused, childFocused) => { ipcMain.once('answer', (event, parentFocused, childFocused) => {