mirror of
https://gitlab.winehq.org/wine/wine-gecko.git
synced 2024-09-13 09:24:08 -07:00
330 lines
11 KiB
JavaScript
330 lines
11 KiB
JavaScript
/* -*- Mode: Java; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- /
|
|
/* vim: set shiftwidth=2 tabstop=2 autoindent cindent expandtab: */
|
|
/* ***** BEGIN LICENSE BLOCK *****
|
|
* Version: MPL 1.1/GPL 2.0/LGPL 2.1
|
|
*
|
|
* The contents of this file are subject to the Mozilla Public License Version
|
|
* 1.1 (the "License"); you may not use this file except in compliance with
|
|
* the License. You may obtain a copy of the License at
|
|
* http://www.mozilla.org/MPL/
|
|
*
|
|
* Software distributed under the License is distributed on an "AS IS" basis,
|
|
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
|
|
* for the specific language governing rights and limitations under the
|
|
* License.
|
|
*
|
|
* The Original Code is B2G.
|
|
*
|
|
* The Initial Developer of the Original Code is
|
|
* Mozilla Foundation.
|
|
* Portions created by the Initial Developer are Copyright (C) 2011
|
|
* the Initial Developer. All Rights Reserved.
|
|
*
|
|
* Contributor(s):
|
|
*
|
|
* Alternatively, the contents of this file may be used under the terms of
|
|
* either the GNU General Public License Version 2 or later (the "GPL"), or
|
|
* the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
|
|
* in which case the provisions of the GPL or the LGPL are applicable instead
|
|
* of those above. If you wish to allow use of your version of this file only
|
|
* under the terms of either the GPL or the LGPL, and not to allow others to
|
|
* use your version of this file under the terms of the MPL, indicate your
|
|
* decision by deleting the provisions above and replace them with the notice
|
|
* and other provisions required by the GPL or the LGPL. If you do not delete
|
|
* the provisions above, a recipient may use your version of this file under
|
|
* the terms of any one of the MPL, the GPL or the LGPL.
|
|
*
|
|
* ***** END LICENSE BLOCK ***** */
|
|
|
|
const Cc = Components.classes;
|
|
const Ci = Components.interfaces;
|
|
const Cu = Components.utils;
|
|
const CC = Components.Constructor;
|
|
|
|
const LocalFile = CC('@mozilla.org/file/local;1',
|
|
'nsILocalFile',
|
|
'initWithPath');
|
|
|
|
Cu.import('resource://gre/modules/XPCOMUtils.jsm');
|
|
Cu.import('resource://gre/modules/Services.jsm');
|
|
|
|
XPCOMUtils.defineLazyGetter(Services, 'env', function() {
|
|
return Cc['@mozilla.org/process/environment;1']
|
|
.getService(Ci.nsIEnvironment);
|
|
});
|
|
|
|
XPCOMUtils.defineLazyGetter(Services, 'ss', function() {
|
|
return Cc['@mozilla.org/content/style-sheet-service;1']
|
|
.getService(Ci.nsIStyleSheetService);
|
|
});
|
|
|
|
// In order to use http:// scheme instead of file:// scheme
|
|
// (that is much more restricted) the following code kick-off
|
|
// a local http server listening on http://127.0.0.1:7777 and
|
|
// http://localhost:7777.
|
|
function startupHttpd(baseDir, port) {
|
|
const httpdURL = 'chrome://browser/content/httpd.js';
|
|
let httpd = {};
|
|
Services.scriptloader.loadSubScript(httpdURL, httpd);
|
|
let server = new httpd.nsHttpServer();
|
|
server.registerDirectory('/', new LocalFile(baseDir));
|
|
server.registerContentType('appcache', 'text/cache-manifest');
|
|
server.start(port);
|
|
}
|
|
|
|
// FIXME Bug 707625
|
|
// until we have a proper security model, add some rights to
|
|
// the pre-installed web applications
|
|
function addPermissions(urls) {
|
|
let permissions = [
|
|
'indexedDB', 'indexedDB-unlimited', 'webapps-manage', 'offline-app'
|
|
];
|
|
urls.forEach(function(url) {
|
|
let uri = Services.io.newURI(url, null, null);
|
|
let allow = Ci.nsIPermissionManager.ALLOW_ACTION;
|
|
|
|
permissions.forEach(function(permission) {
|
|
Services.perms.add(uri, permission, allow);
|
|
});
|
|
});
|
|
}
|
|
|
|
|
|
var shell = {
|
|
// FIXME/bug 678695: this should be a system setting
|
|
preferredScreenBrightness: 1.0,
|
|
|
|
get contentBrowser() {
|
|
delete this.contentBrowser;
|
|
return this.contentBrowser = document.getElementById('homescreen');
|
|
},
|
|
|
|
get homeURL() {
|
|
try {
|
|
let homeSrc = Services.env.get('B2G_HOMESCREEN');
|
|
if (homeSrc)
|
|
return homeSrc;
|
|
} catch (e) {}
|
|
|
|
let urls = Services.prefs.getCharPref('browser.homescreenURL').split(',');
|
|
for (let i = 0; i < urls.length; i++) {
|
|
let url = urls[i];
|
|
if (url.substring(0, 7) != 'file://')
|
|
return url;
|
|
|
|
let file = new LocalFile(url.substring(7, url.length));
|
|
if (file.exists())
|
|
return url;
|
|
}
|
|
return null;
|
|
},
|
|
|
|
start: function shell_init() {
|
|
let homeURL = this.homeURL;
|
|
if (!homeURL) {
|
|
let msg = 'Fatal error during startup: [No homescreen found]';
|
|
return alert(msg);
|
|
}
|
|
|
|
window.controllers.appendController(this);
|
|
window.addEventListener('keypress', this);
|
|
window.addEventListener('MozApplicationManifest', this);
|
|
this.contentBrowser.addEventListener('load', this, true);
|
|
|
|
try {
|
|
Services.io.offline = false;
|
|
|
|
let fileScheme = 'file://';
|
|
if (homeURL.substring(0, fileScheme.length) == fileScheme) {
|
|
homeURL = homeURL.replace(fileScheme, '');
|
|
|
|
let baseDir = homeURL.split('/');
|
|
baseDir.pop();
|
|
baseDir = baseDir.join('/');
|
|
|
|
const SERVER_PORT = 6666;
|
|
startupHttpd(baseDir, SERVER_PORT);
|
|
|
|
let baseHost = 'http://localhost';
|
|
homeURL = homeURL.replace(baseDir, baseHost + ':' + SERVER_PORT);
|
|
}
|
|
addPermissions([homeURL]);
|
|
} catch (e) {
|
|
let msg = 'Fatal error during startup: [' + e + '[' + homeURL + ']';
|
|
return alert(msg);
|
|
}
|
|
|
|
// Load webapi+apps.js as a frame script
|
|
let frameScriptUrl = 'chrome://browser/content/webapi.js';
|
|
try {
|
|
messageManager.loadFrameScript(frameScriptUrl, true);
|
|
} catch (e) {
|
|
dump('Error when loading ' + frameScriptUrl + ' as a frame script: ' + e + '\n');
|
|
}
|
|
|
|
let browser = this.contentBrowser;
|
|
browser.homePage = homeURL;
|
|
browser.goHome();
|
|
},
|
|
|
|
stop: function shell_stop() {
|
|
window.controllers.removeController(this);
|
|
window.removeEventListener('keypress', this);
|
|
window.removeEventListener('MozApplicationManifest', this);
|
|
},
|
|
|
|
supportsCommand: function shell_supportsCommand(cmd) {
|
|
let isSupported = false;
|
|
switch (cmd) {
|
|
case 'cmd_close':
|
|
isSupported = true;
|
|
break;
|
|
default:
|
|
isSupported = false;
|
|
break;
|
|
}
|
|
return isSupported;
|
|
},
|
|
|
|
isCommandEnabled: function shell_isCommandEnabled(cmd) {
|
|
return true;
|
|
},
|
|
|
|
doCommand: function shell_doCommand(cmd) {
|
|
switch (cmd) {
|
|
case 'cmd_close':
|
|
content.postMessage('appclose', '*');
|
|
break;
|
|
}
|
|
},
|
|
|
|
handleEvent: function shell_handleEvent(evt) {
|
|
switch (evt.type) {
|
|
case 'keypress':
|
|
switch (evt.keyCode) {
|
|
case evt.DOM_VK_HOME:
|
|
this.sendEvent(content, 'home');
|
|
break;
|
|
case evt.DOM_VK_SLEEP:
|
|
this.toggleScreen();
|
|
|
|
let details = {
|
|
'enabled': screen.mozEnabled
|
|
};
|
|
this.sendEvent(content, 'sleep', details);
|
|
break;
|
|
case evt.DOM_VK_ESCAPE:
|
|
if (evt.defaultPrevented)
|
|
return;
|
|
this.doCommand('cmd_close');
|
|
break;
|
|
}
|
|
break;
|
|
case 'load':
|
|
this.contentBrowser.removeEventListener('load', this, true);
|
|
this.turnScreenOn();
|
|
|
|
let chromeWindow = window.QueryInterface(Ci.nsIDOMChromeWindow);
|
|
chromeWindow.browserDOMWindow = new nsBrowserAccess();
|
|
|
|
this.sendEvent(window, 'ContentStart');
|
|
break;
|
|
case 'MozApplicationManifest':
|
|
try {
|
|
if (!Services.prefs.getBoolPref('browser.cache.offline.enable'))
|
|
return;
|
|
|
|
let contentWindow = evt.originalTarget.defaultView;
|
|
let documentElement = contentWindow.document.documentElement;
|
|
if (!documentElement)
|
|
return;
|
|
|
|
let manifest = documentElement.getAttribute('manifest');
|
|
if (!manifest)
|
|
return;
|
|
|
|
let documentURI = contentWindow.document.documentURIObject;
|
|
if (!Services.perms.testPermission(documentURI, 'offline-app')) {
|
|
if (Services.prefs.getBoolPref('browser.offline-apps.notify')) {
|
|
// FIXME Bug 710729 - Add a UI for offline cache notifications
|
|
return;
|
|
}
|
|
return;
|
|
}
|
|
|
|
Services.perms.add(documentURI, 'offline-app',
|
|
Ci.nsIPermissionManager.ALLOW_ACTION);
|
|
|
|
let manifestURI = Services.io.newURI(manifest, null, documentURI);
|
|
let updateService = Cc['@mozilla.org/offlinecacheupdate-service;1']
|
|
.getService(Ci.nsIOfflineCacheUpdateService);
|
|
updateService.scheduleUpdate(manifestURI, documentURI, window);
|
|
} catch (e) {
|
|
dump('Error while creating offline cache: ' + e + '\n');
|
|
}
|
|
break;
|
|
}
|
|
},
|
|
sendEvent: function shell_sendEvent(content, type, details) {
|
|
let event = content.document.createEvent('CustomEvent');
|
|
event.initCustomEvent(type, true, true, details ? details : {});
|
|
content.dispatchEvent(event);
|
|
},
|
|
toggleScreen: function shell_toggleScreen() {
|
|
if (screen.mozEnabled)
|
|
this.turnScreenOff();
|
|
else
|
|
this.turnScreenOn();
|
|
},
|
|
turnScreenOff: function shell_turnScreenOff() {
|
|
screen.mozEnabled = false;
|
|
screen.mozBrightness = 0.0;
|
|
},
|
|
turnScreenOn: function shell_turnScreenOn() {
|
|
screen.mozEnabled = true;
|
|
screen.mozBrightness = this.preferredScreenBrightness;
|
|
}
|
|
};
|
|
|
|
|
|
|
|
function nsBrowserAccess() {
|
|
}
|
|
|
|
nsBrowserAccess.prototype = {
|
|
QueryInterface: XPCOMUtils.generateQI([Ci.nsIBrowserDOMWindow]),
|
|
|
|
openURI: function openURI(uri, opener, where, context) {
|
|
// TODO This should be replaced by an 'open-browser-window' intent
|
|
let contentWindow = content.wrappedJSObject;
|
|
if (!('getApplicationManager' in contentWindow))
|
|
return null;
|
|
|
|
let applicationManager = contentWindow.getApplicationManager();
|
|
if (!applicationManager)
|
|
return null;
|
|
|
|
let url = uri ? uri.spec : 'about:blank';
|
|
let window = applicationManager.launch(url, where);
|
|
return window.contentWindow;
|
|
},
|
|
|
|
openURIInFrame: function openURIInFrame(uri, opener, where, context) {
|
|
throw new Error('Not Implemented');
|
|
},
|
|
|
|
isTabContentWindow: function isTabContentWindow(contentWindow) {
|
|
return contentWindow == window;
|
|
}
|
|
};
|
|
|
|
// Pipe `console` log messages to the nsIConsoleService which writes them
|
|
// to logcat.
|
|
Services.obs.addObserver(function onConsoleAPILogEvent(subject, topic, data) {
|
|
let message = subject.wrappedJSObject;
|
|
let prefix = "Content JS " + message.level.toUpperCase() +
|
|
" at " + message.filename + ":" + message.lineNumber +
|
|
" in " + (message.functionName || "anonymous") + ": ";
|
|
Services.console.logStringMessage(prefix + Array.join(message.arguments, " "));
|
|
}, "console-api-log-event", false);
|