electron/atom/browser/lib/guest-view-manager.js

267 lines
8.2 KiB
JavaScript
Raw Normal View History

2016-01-14 21:21:11 +00:00
const ipcMain = require('electron').ipcMain;
const webContents = require('electron').webContents;
2016-01-12 02:40:23 +00:00
2016-01-14 21:21:11 +00:00
var slice = [].slice;
2016-01-12 02:40:23 +00:00
2016-01-14 18:35:29 +00:00
// Doesn't exist in early initialization.
2016-01-14 21:21:11 +00:00
var webViewManager = null;
2016-01-12 02:40:23 +00:00
2016-01-25 16:37:15 +00:00
var supportedWebViewEvents = [
'load-commit',
'did-finish-load',
'did-fail-load',
'did-frame-finish-load',
'did-start-loading',
'did-stop-loading',
'did-get-response-details',
'did-get-redirect-request',
'dom-ready',
'console-message',
'devtools-opened',
'devtools-closed',
'devtools-focused',
'new-window',
'will-navigate',
'did-navigate',
'did-navigate-in-page',
'close',
'crashed',
'gpu-crashed',
'plugin-crashed',
'destroyed',
'page-title-updated',
'page-favicon-updated',
'enter-html-full-screen',
'leave-html-full-screen',
'media-started-playing',
'media-paused',
'found-in-page',
'did-change-theme-color',
'permission-request'
];
2016-01-12 02:40:23 +00:00
2016-01-14 21:21:11 +00:00
var nextInstanceId = 0;
2016-01-23 13:29:47 +00:00
var permissionRequests;
2016-01-14 21:21:11 +00:00
var guestInstances = {};
2016-01-23 13:29:47 +00:00
var guestPermissionRequestsMap = {};
2016-01-14 21:21:11 +00:00
var embedderElementsMap = {};
var reverseEmbedderElementsMap = {};
2016-01-12 02:40:23 +00:00
2016-01-14 18:35:29 +00:00
// Moves the last element of array to the first one.
2016-01-14 21:21:11 +00:00
var moveLastToFirst = function(list) {
2016-01-12 02:40:23 +00:00
return list.unshift(list.pop());
};
2016-01-14 18:35:29 +00:00
// Generate guestInstanceId.
2016-01-19 22:49:40 +00:00
var getNextInstanceId = function() {
2016-01-12 02:40:23 +00:00
return ++nextInstanceId;
};
2016-01-14 18:35:29 +00:00
// Create a new guest instance.
2016-01-14 21:21:11 +00:00
var createGuest = function(embedder, params) {
2016-01-12 02:40:23 +00:00
var destroy, destroyEvents, event, fn, guest, i, id, j, len, len1, listeners;
if (webViewManager == null) {
webViewManager = process.atomBinding('web_view_manager');
}
id = getNextInstanceId(embedder);
guest = webContents.create({
isGuest: true,
partition: params.partition,
embedder: embedder
});
guestInstances[id] = {
guest: guest,
embedder: embedder
};
2016-01-14 18:35:29 +00:00
// Destroy guest when the embedder is gone or navigated.
destroyEvents = ['will-destroy', 'crashed', 'did-navigate'];
2016-01-12 02:40:23 +00:00
destroy = function() {
if (guestInstances[id] != null) {
return destroyGuest(embedder, id);
}
};
for (i = 0, len = destroyEvents.length; i < len; i++) {
event = destroyEvents[i];
embedder.once(event, destroy);
2016-01-14 18:44:21 +00:00
// Users might also listen to the crashed event, so We must ensure the guest
// is destroyed before users' listener gets called. It is done by moving our
// listener to the first one in queue.
2016-01-12 02:40:23 +00:00
listeners = embedder._events[event];
if (Array.isArray(listeners)) {
moveLastToFirst(listeners);
}
}
guest.once('destroyed', function() {
var j, len1, results;
results = [];
for (j = 0, len1 = destroyEvents.length; j < len1; j++) {
event = destroyEvents[j];
results.push(embedder.removeListener(event, destroy));
}
return results;
});
2016-01-14 18:35:29 +00:00
// Init guest web view after attached.
2016-01-12 02:40:23 +00:00
guest.once('did-attach', function() {
var opts;
params = this.attachParams;
delete this.attachParams;
this.viewInstanceId = params.instanceId;
this.setSize({
normal: {
width: params.elementWidth,
height: params.elementHeight
},
enableAutoSize: params.autosize,
min: {
width: params.minwidth,
height: params.minheight
},
max: {
width: params.maxwidth,
height: params.maxheight
}
});
if (params.src) {
opts = {};
if (params.httpreferrer) {
opts.httpReferrer = params.httpreferrer;
}
if (params.useragent) {
opts.userAgent = params.useragent;
}
this.loadURL(params.src, opts);
}
if (params.allowtransparency != null) {
this.setAllowTransparency(params.allowtransparency);
}
return guest.allowPopups = params.allowpopups;
});
2016-01-14 18:35:29 +00:00
// Dispatch events to embedder.
2016-01-12 02:40:23 +00:00
fn = function(event) {
return guest.on(event, function() {
2016-01-19 22:49:40 +00:00
var args = 2 <= arguments.length ? slice.call(arguments, 1) : [];
2016-01-23 13:29:47 +00:00
if (event === 'permission-request') {
if (!guestPermissionRequestsMap[guest.viewInstanceId])
guestPermissionRequestsMap[guest.viewInstanceId] = {};
var permission = args[0];
guestPermissionRequestsMap[guest.viewInstanceId][permission] = args[1];
args.pop();
}
2016-01-12 02:40:23 +00:00
return embedder.send.apply(embedder, ["ATOM_SHELL_GUEST_VIEW_INTERNAL_DISPATCH_EVENT-" + guest.viewInstanceId, event].concat(slice.call(args)));
});
};
for (j = 0, len1 = supportedWebViewEvents.length; j < len1; j++) {
event = supportedWebViewEvents[j];
fn(event);
}
2016-01-14 18:35:29 +00:00
// Dispatch guest's IPC messages to embedder.
2016-01-12 02:40:23 +00:00
guest.on('ipc-message-host', function(_, packed) {
var args, channel;
channel = packed[0], args = 2 <= packed.length ? slice.call(packed, 1) : [];
return embedder.send.apply(embedder, ["ATOM_SHELL_GUEST_VIEW_INTERNAL_IPC_MESSAGE-" + guest.viewInstanceId, channel].concat(slice.call(args)));
});
2016-01-14 18:35:29 +00:00
// Autosize.
2016-01-12 02:40:23 +00:00
guest.on('size-changed', function() {
2016-01-19 22:49:40 +00:00
var args = 2 <= arguments.length ? slice.call(arguments, 1) : [];
2016-01-12 02:40:23 +00:00
return embedder.send.apply(embedder, ["ATOM_SHELL_GUEST_VIEW_INTERNAL_SIZE_CHANGED-" + guest.viewInstanceId].concat(slice.call(args)));
});
return id;
};
2016-01-14 18:35:29 +00:00
// Attach the guest to an element of embedder.
2016-01-14 21:21:11 +00:00
var attachGuest = function(embedder, elementInstanceId, guestInstanceId, params) {
2016-01-12 02:40:23 +00:00
var guest, key, oldGuestInstanceId, ref1, webPreferences;
guest = guestInstances[guestInstanceId].guest;
2016-01-14 18:35:29 +00:00
// Destroy the old guest when attaching.
2016-01-12 02:40:23 +00:00
key = (embedder.getId()) + "-" + elementInstanceId;
oldGuestInstanceId = embedderElementsMap[key];
if (oldGuestInstanceId != null) {
2016-01-14 18:35:29 +00:00
// Reattachment to the same guest is not currently supported.
2016-01-12 02:40:23 +00:00
if (oldGuestInstanceId === guestInstanceId) {
return;
}
if (guestInstances[oldGuestInstanceId] == null) {
return;
}
destroyGuest(embedder, oldGuestInstanceId);
}
webPreferences = {
guestInstanceId: guestInstanceId,
nodeIntegration: (ref1 = params.nodeintegration) != null ? ref1 : false,
plugins: params.plugins,
2016-01-21 10:13:41 +00:00
webSecurity: !params.disablewebsecurity,
2016-01-23 13:29:47 +00:00
blinkFeatures: params.blinkfeatures,
webNotification: !params.disablewebnotification,
2016-01-12 02:40:23 +00:00
};
if (params.preload) {
webPreferences.preloadURL = params.preload;
}
webViewManager.addGuest(guestInstanceId, elementInstanceId, embedder, guest, webPreferences);
guest.attachParams = params;
embedderElementsMap[key] = guestInstanceId;
return reverseEmbedderElementsMap[guestInstanceId] = key;
};
2016-01-14 18:35:29 +00:00
// Destroy an existing guest instance.
2016-01-14 21:21:11 +00:00
var destroyGuest = function(embedder, id) {
2016-01-12 02:40:23 +00:00
var key;
webViewManager.removeGuest(embedder, id);
guestInstances[id].guest.destroy();
delete guestInstances[id];
2016-01-25 16:37:15 +00:00
delete guestPermissionRequestsMap[id];
2016-01-12 02:40:23 +00:00
key = reverseEmbedderElementsMap[id];
if (key != null) {
delete reverseEmbedderElementsMap[id];
return delete embedderElementsMap[key];
}
};
ipcMain.on('ATOM_SHELL_GUEST_VIEW_MANAGER_CREATE_GUEST', function(event, params, requestId) {
return event.sender.send("ATOM_SHELL_RESPONSE_" + requestId, createGuest(event.sender, params));
});
ipcMain.on('ATOM_SHELL_GUEST_VIEW_MANAGER_ATTACH_GUEST', function(event, elementInstanceId, guestInstanceId, params) {
return attachGuest(event.sender, elementInstanceId, guestInstanceId, params);
});
ipcMain.on('ATOM_SHELL_GUEST_VIEW_MANAGER_DESTROY_GUEST', function(event, id) {
return destroyGuest(event.sender, id);
});
2016-01-23 13:29:47 +00:00
ipcMain.on('ATOM_SHELL_GUEST_VIEW_MANAGER_SET_PERMISSION_RESPONSE', function(event, viewInstanceId, permission, allowed) {
permissionRequests = guestPermissionRequestsMap[viewInstanceId];
if (permissionRequests && permissionRequests[permission] !== null) {
permissionRequests[permission].apply(null, [allowed]);
}
});
2016-01-12 02:40:23 +00:00
ipcMain.on('ATOM_SHELL_GUEST_VIEW_MANAGER_SET_SIZE', function(event, id, params) {
var ref1;
return (ref1 = guestInstances[id]) != null ? ref1.guest.setSize(params) : void 0;
});
ipcMain.on('ATOM_SHELL_GUEST_VIEW_MANAGER_SET_ALLOW_TRANSPARENCY', function(event, id, allowtransparency) {
var ref1;
return (ref1 = guestInstances[id]) != null ? ref1.guest.setAllowTransparency(allowtransparency) : void 0;
});
2016-01-14 18:35:29 +00:00
// Returns WebContents from its guest id.
2016-01-12 02:40:23 +00:00
exports.getGuest = function(id) {
var ref1;
return (ref1 = guestInstances[id]) != null ? ref1.guest : void 0;
};
2016-01-14 18:35:29 +00:00
// Returns the embedder of the guest.
2016-01-12 02:40:23 +00:00
exports.getEmbedder = function(id) {
var ref1;
return (ref1 = guestInstances[id]) != null ? ref1.embedder : void 0;
};