Merge pull request #1137 from bundyo/implement-some-popup-methods

Implement FakeWindow.postMessage, window.opener.postMessage and window.closed flag for popup windows
This commit is contained in:
Cheng Zhao 2015-03-04 07:28:33 -08:00
commit fac9ea3356
3 changed files with 75 additions and 3 deletions

View file

@ -17,9 +17,11 @@ createGuest = (embedder, url, frameName, options) ->
# guest is closed by user then we should prevent |embedder| from double # guest is closed by user then we should prevent |embedder| from double
# closing guest. # closing guest.
closedByEmbedder = -> closedByEmbedder = ->
embedder.send 'ATOM_SHELL_GUEST_WINDOW_MANAGER_WINDOW_CLOSED', guest.id
guest.removeListener 'closed', closedByUser guest.removeListener 'closed', closedByUser
guest.destroy() unless guest.isClosed() guest.destroy() unless guest.isClosed()
closedByUser = -> closedByUser = ->
embedder.send 'ATOM_SHELL_GUEST_WINDOW_MANAGER_WINDOW_CLOSED', guest.id
embedder.removeListener 'render-view-deleted', closedByEmbedder embedder.removeListener 'render-view-deleted', closedByEmbedder
embedder.once 'render-view-deleted', closedByEmbedder embedder.once 'render-view-deleted', closedByEmbedder
guest.once 'closed', closedByUser guest.once 'closed', closedByUser
@ -30,6 +32,10 @@ createGuest = (embedder, url, frameName, options) ->
guest.once 'closed', -> guest.once 'closed', ->
delete frameToGuest[frameName] delete frameToGuest[frameName]
ipc.on 'ATOM_SHELL_GUEST_WINDOW_MANAGER_WINDOW_OPENER_POSTMESSAGE', (event, message, targetOrigin) ->
if embedder.getUrl().indexOf(targetOrigin) is 0 or targetOrigin is '*'
embedder.send 'ATOM_SHELL_GUEST_WINDOW_MANAGER_WINDOW_POSTMESSAGE', message, targetOrigin
guest.id guest.id
# Routed window.open messages. # Routed window.open messages.
@ -49,6 +55,12 @@ ipc.on 'ATOM_SHELL_GUEST_WINDOW_MANAGER_WINDOW_METHOD', (event, guestId, method,
return unless BrowserWindow.windows.has guestId return unless BrowserWindow.windows.has guestId
BrowserWindow.windows.get(guestId)[method] args... BrowserWindow.windows.get(guestId)[method] args...
ipc.on 'ATOM_SHELL_GUEST_WINDOW_MANAGER_WINDOW_POSTMESSAGE', (event, guestId, message, targetOrigin) ->
return unless BrowserWindow.windows.has guestId
window = BrowserWindow.windows.get(guestId)
if window.getUrl().indexOf(targetOrigin) is 0 or targetOrigin is '*'
window.send 'ATOM_SHELL_GUEST_WINDOW_MANAGER_WINDOW_POSTMESSAGE', message, targetOrigin
ipc.on 'ATOM_SHELL_GUEST_WINDOW_MANAGER_WEB_CONTENTS_METHOD', (event, guestId, method, args...) -> ipc.on 'ATOM_SHELL_GUEST_WINDOW_MANAGER_WEB_CONTENTS_METHOD', (event, guestId, method, args...) ->
return unless BrowserWindow.windows.has guestId return unless BrowserWindow.windows.has guestId
BrowserWindow.windows.get(guestId).webContents?[method] args... BrowserWindow.windows.get(guestId).webContents?[method] args...

View file

@ -5,6 +5,9 @@ remote = require 'remote'
# Window object returned by "window.open". # Window object returned by "window.open".
class FakeWindow class FakeWindow
constructor: (@guestId) -> constructor: (@guestId) ->
ipc.on 'ATOM_SHELL_GUEST_WINDOW_MANAGER_WINDOW_CLOSED', (guestId) =>
if guestId is @guestId
@closed = true
close: -> close: ->
ipc.send 'ATOM_SHELL_GUEST_WINDOW_MANAGER_WINDOW_CLOSE', @guestId ipc.send 'ATOM_SHELL_GUEST_WINDOW_MANAGER_WINDOW_CLOSE', @guestId
@ -15,6 +18,9 @@ class FakeWindow
blur: -> blur: ->
ipc.send 'ATOM_SHELL_GUEST_WINDOW_MANAGER_WINDOW_METHOD', @guestId, 'blur' ipc.send 'ATOM_SHELL_GUEST_WINDOW_MANAGER_WINDOW_METHOD', @guestId, 'blur'
postMessage: (message, targetOrigin) ->
ipc.send 'ATOM_SHELL_GUEST_WINDOW_MANAGER_WINDOW_POSTMESSAGE', @guestId, message, targetOrigin
eval: (args...) -> eval: (args...) ->
ipc.send 'ATOM_SHELL_GUEST_WINDOW_MANAGER_WEB_CONTENTS_METHOD', @guestId, 'executeJavaScript', args... ipc.send 'ATOM_SHELL_GUEST_WINDOW_MANAGER_WEB_CONTENTS_METHOD', @guestId, 'executeJavaScript', args...
@ -67,3 +73,10 @@ window.confirm = (message, title='') ->
# But we do not support prompt(). # But we do not support prompt().
window.prompt = -> window.prompt = ->
throw new Error('prompt() is and will not be supported in atom-shell.') throw new Error('prompt() is and will not be supported in atom-shell.')
window.opener =
postMessage: (message, targetOrigin) ->
ipc.send 'ATOM_SHELL_GUEST_WINDOW_MANAGER_WINDOW_OPENER_POSTMESSAGE', message, targetOrigin
ipc.on 'ATOM_SHELL_GUEST_WINDOW_MANAGER_WINDOW_POSTMESSAGE', (message, targetOrigin) ->
window.postMessage(message, targetOrigin)

View file

@ -641,9 +641,7 @@ Emitted when a redirect was received while requesting a resource.
Emitted when the page requested to open a new window for `url`. It could be Emitted when the page requested to open a new window for `url`. It could be
requested by `window.open` or an external link like `<a target='_blank'>`. requested by `window.open` or an external link like `<a target='_blank'>`.
Check the next section [Handling Child Windows](#handling-child-windows) for more information.
By default a new `BrowserWindow` will be created for the `url`, and a proxy
will be returned to `window.open` to let you have limited control of it.
Calling `event.preventDefault()` can prevent creating new windows. Calling `event.preventDefault()` can prevent creating new windows.
@ -844,3 +842,52 @@ app.on('ready', function() {
is different from the handlers on browser side. is different from the handlers on browser side.
2. There is no way to send synchronous messages from browser side to web pages, 2. There is no way to send synchronous messages from browser side to web pages,
because it would be very easy to cause dead locks. because it would be very easy to cause dead locks.
## Handling Child Windows
When the page contents request opening a new window, either by invoking
`window.open` or by an external link such as `<a target='_blank'>`, by
default a new `BrowserWindow` will be created for the `url`, and a proxy
will be returned to `window.open` to let the page to have limited control over it.
The proxy has some limited standard functionality implemented, which will help
the page to interact with it. The following methods are avaialable:
### Window.blur()
Removes focus from the child window.
### Window.close()
Forcefully closes the child window without calling its unload event.
### Window.closed
Set to true after the child window gets closed.
### Window.eval(code)
* `code` String
Evaluates the code in the child window.
### Window.focus()
Focuses the child window (brings the window to front).
### Window.postMessage(message, targetOrigin)
* `message` String
* `targetOrigin` String
Sends a message to the child window with the specified origin or "*" for no origin preference.
In addition to these methods, the child window implements `window.opener` object with no properties
and a single method:
### window.opener.postMessage(message, targetOrigin)
* `message` String
* `targetOrigin` String
Sends a message to the parent window with the specified origin or "*" for no origin preference.