2011-11-10 12:09:18 +02:00
|
|
|
/* -*- Mode: javascript; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
|
|
/* vim: set ts=2 et sw=2 tw=80: */
|
2012-05-21 12:12:37 +01:00
|
|
|
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
2011-11-10 12:09:18 +02:00
|
|
|
"use strict";
|
|
|
|
|
2013-04-11 13:59:08 -07:00
|
|
|
const {Cu} = require("chrome");
|
|
|
|
|
|
|
|
let {TiltVisualizer} = require("devtools/tilt/tilt-visualizer");
|
|
|
|
let TiltGL = require("devtools/tilt/tilt-gl");
|
|
|
|
let TiltUtils = require("devtools/tilt/tilt-utils");
|
|
|
|
let EventEmitter = require("devtools/shared/event-emitter");
|
2013-05-24 11:26:17 +01:00
|
|
|
let Telemetry = require("devtools/shared/telemetry");
|
2013-04-11 13:59:08 -07:00
|
|
|
|
|
|
|
Cu.import("resource://gre/modules/Services.jsm");
|
2011-11-10 12:09:18 +02:00
|
|
|
|
|
|
|
// Tilt notifications dispatched through the nsIObserverService.
|
|
|
|
const TILT_NOTIFICATIONS = {
|
2013-04-10 02:05:09 +03:00
|
|
|
// Called early in the startup of a new tilt instance
|
|
|
|
STARTUP: "tilt-startup",
|
2011-11-10 12:09:18 +02:00
|
|
|
|
2012-01-19 17:48:22 +02:00
|
|
|
// Fires when Tilt starts the initialization.
|
|
|
|
INITIALIZING: "tilt-initializing",
|
|
|
|
|
|
|
|
// Fires immediately after initialization is complete.
|
|
|
|
// (when the canvas overlay is visible and the 3D mesh is completely created)
|
2011-11-10 12:09:18 +02:00
|
|
|
INITIALIZED: "tilt-initialized",
|
|
|
|
|
2012-01-19 17:48:22 +02:00
|
|
|
// Fires immediately before the destruction is started.
|
|
|
|
DESTROYING: "tilt-destroying",
|
|
|
|
|
|
|
|
// Fires immediately before the destruction is finished.
|
|
|
|
// (just before the canvas overlay is removed from its parent node)
|
|
|
|
BEFORE_DESTROYED: "tilt-before-destroyed",
|
|
|
|
|
|
|
|
// Fires when Tilt is completely destroyed.
|
2011-11-10 12:09:18 +02:00
|
|
|
DESTROYED: "tilt-destroyed",
|
|
|
|
|
|
|
|
// Fires when Tilt is shown (after a tab-switch).
|
|
|
|
SHOWN: "tilt-shown",
|
|
|
|
|
|
|
|
// Fires when Tilt is hidden (after a tab-switch).
|
2012-01-19 17:48:22 +02:00
|
|
|
HIDDEN: "tilt-hidden",
|
|
|
|
|
|
|
|
// Fires once Tilt highlights an element in the page.
|
|
|
|
HIGHLIGHTING: "tilt-highlighting",
|
|
|
|
|
|
|
|
// Fires once Tilt stops highlighting any element.
|
|
|
|
UNHIGHLIGHTING: "tilt-unhighlighting",
|
|
|
|
|
|
|
|
// Fires when a node is removed from the 3D mesh.
|
|
|
|
NODE_REMOVED: "tilt-node-removed"
|
2011-11-10 12:09:18 +02:00
|
|
|
};
|
|
|
|
|
2013-04-11 13:59:08 -07:00
|
|
|
let TiltManager = {
|
2012-11-30 08:07:59 +00:00
|
|
|
_instances: new WeakMap(),
|
|
|
|
getTiltForBrowser: function(aChromeWindow)
|
|
|
|
{
|
|
|
|
if (this._instances.has(aChromeWindow)) {
|
|
|
|
return this._instances.get(aChromeWindow);
|
|
|
|
} else {
|
|
|
|
let tilt = new Tilt(aChromeWindow);
|
|
|
|
this._instances.set(aChromeWindow, tilt);
|
|
|
|
return tilt;
|
|
|
|
}
|
|
|
|
},
|
|
|
|
}
|
2011-11-10 12:09:18 +02:00
|
|
|
|
2013-04-11 13:59:08 -07:00
|
|
|
exports.TiltManager = TiltManager;
|
|
|
|
|
2011-11-10 12:09:18 +02:00
|
|
|
/**
|
|
|
|
* Object managing instances of the visualizer.
|
|
|
|
*
|
|
|
|
* @param {Window} aWindow
|
|
|
|
* the chrome window used by each visualizer instance
|
|
|
|
*/
|
2013-04-11 13:59:08 -07:00
|
|
|
function Tilt(aWindow)
|
2011-11-10 12:09:18 +02:00
|
|
|
{
|
|
|
|
/**
|
|
|
|
* Save a reference to the top-level window.
|
|
|
|
*/
|
|
|
|
this.chromeWindow = aWindow;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* All the instances of TiltVisualizer.
|
|
|
|
*/
|
|
|
|
this.visualizers = {};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Shortcut for accessing notifications strings.
|
|
|
|
*/
|
|
|
|
this.NOTIFICATIONS = TILT_NOTIFICATIONS;
|
2012-11-30 08:07:59 +00:00
|
|
|
|
2012-12-19 02:17:39 +05:30
|
|
|
EventEmitter.decorate(this);
|
|
|
|
|
2012-11-30 08:07:59 +00:00
|
|
|
this.setup();
|
2013-05-24 11:26:17 +01:00
|
|
|
|
|
|
|
this._telemetry = new Telemetry();
|
2011-11-10 12:09:18 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
Tilt.prototype = {
|
|
|
|
|
|
|
|
/**
|
2012-11-23 10:54:42 +02:00
|
|
|
* Initializes a visualizer for the current tab or closes it if already open.
|
2011-11-10 12:09:18 +02:00
|
|
|
*/
|
2012-11-23 10:54:42 +02:00
|
|
|
toggle: function T_toggle()
|
2011-11-10 12:09:18 +02:00
|
|
|
{
|
2012-11-30 08:07:59 +00:00
|
|
|
let contentWindow = this.chromeWindow.gBrowser.selectedBrowser.contentWindow;
|
2011-11-10 12:09:18 +02:00
|
|
|
let id = this.currentWindowId;
|
2012-11-30 08:07:59 +00:00
|
|
|
let self = this;
|
|
|
|
|
|
|
|
contentWindow.addEventListener("beforeunload", function onUnload() {
|
|
|
|
contentWindow.removeEventListener("beforeunload", onUnload, false);
|
|
|
|
self.destroy(id, true);
|
|
|
|
}, false);
|
2011-11-10 12:09:18 +02:00
|
|
|
|
|
|
|
// if the visualizer for the current tab is already open, destroy it now
|
|
|
|
if (this.visualizers[id]) {
|
2012-01-16 09:00:56 +02:00
|
|
|
this.destroy(id, true);
|
2013-05-24 11:26:17 +01:00
|
|
|
this._telemetry.toolClosed("tilt");
|
2011-11-10 12:09:18 +02:00
|
|
|
return;
|
2013-05-24 11:26:17 +01:00
|
|
|
} else {
|
|
|
|
this._telemetry.toolOpened("tilt");
|
2011-11-10 12:09:18 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// create a visualizer instance for the current tab
|
|
|
|
this.visualizers[id] = new TiltVisualizer({
|
2012-01-20 13:36:48 +02:00
|
|
|
chromeWindow: this.chromeWindow,
|
2013-04-10 02:05:09 +03:00
|
|
|
contentWindow: contentWindow,
|
2012-01-20 13:36:48 +02:00
|
|
|
parentNode: this.chromeWindow.gBrowser.selectedBrowser.parentNode,
|
2012-11-30 08:07:59 +00:00
|
|
|
notifications: this.NOTIFICATIONS,
|
|
|
|
tab: this.chromeWindow.gBrowser.selectedTab
|
2011-11-10 12:09:18 +02:00
|
|
|
});
|
|
|
|
|
2013-04-10 02:05:09 +03:00
|
|
|
Services.obs.notifyObservers(contentWindow, TILT_NOTIFICATIONS.STARTUP, null);
|
|
|
|
this.visualizers[id].init();
|
|
|
|
|
2011-11-10 12:09:18 +02:00
|
|
|
// make sure the visualizer object was initialized properly
|
|
|
|
if (!this.visualizers[id].isInitialized()) {
|
|
|
|
this.destroy(id);
|
2012-06-03 20:15:41 +03:00
|
|
|
this.failureCallback && this.failureCallback();
|
2011-11-10 12:09:18 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2013-04-10 02:05:09 +03:00
|
|
|
this.lastInstanceId = id;
|
2012-12-19 02:17:39 +05:30
|
|
|
this.emit("change", this.chromeWindow.gBrowser.selectedTab);
|
2013-04-10 02:05:09 +03:00
|
|
|
Services.obs.notifyObservers(contentWindow, TILT_NOTIFICATIONS.INITIALIZING, null);
|
2011-11-10 12:09:18 +02:00
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
2012-02-07 11:44:55 +02:00
|
|
|
* Starts destroying a specific instance of the visualizer.
|
2011-11-10 12:09:18 +02:00
|
|
|
*
|
|
|
|
* @param {String} aId
|
|
|
|
* the identifier of the instance in the visualizers array
|
2012-01-16 09:00:56 +02:00
|
|
|
* @param {Boolean} aAnimateFlag
|
|
|
|
* optional, set to true to display a destruction transition
|
2011-11-10 12:09:18 +02:00
|
|
|
*/
|
2012-01-16 09:00:56 +02:00
|
|
|
destroy: function T_destroy(aId, aAnimateFlag)
|
2011-11-10 12:09:18 +02:00
|
|
|
{
|
2012-02-07 11:44:55 +02:00
|
|
|
// if the visualizer is destroyed or destroying, don't do anything
|
|
|
|
if (!this.visualizers[aId] || this._isDestroying) {
|
2011-11-10 12:09:18 +02:00
|
|
|
return;
|
|
|
|
}
|
2012-02-07 11:44:55 +02:00
|
|
|
this._isDestroying = true;
|
2011-11-10 12:09:18 +02:00
|
|
|
|
2012-02-07 11:44:55 +02:00
|
|
|
let controller = this.visualizers[aId].controller;
|
|
|
|
let presenter = this.visualizers[aId].presenter;
|
2012-01-16 09:00:56 +02:00
|
|
|
|
2012-02-07 11:44:55 +02:00
|
|
|
let content = presenter.contentWindow;
|
|
|
|
let pageXOffset = content.pageXOffset * presenter.transforms.zoom;
|
|
|
|
let pageYOffset = content.pageYOffset * presenter.transforms.zoom;
|
|
|
|
TiltUtils.setDocumentZoom(this.chromeWindow, presenter.transforms.zoom);
|
2012-01-16 09:00:56 +02:00
|
|
|
|
2012-02-07 11:44:55 +02:00
|
|
|
// if we're not doing any outro animation, just finish destruction directly
|
|
|
|
if (!aAnimateFlag) {
|
|
|
|
this._finish(aId);
|
|
|
|
return;
|
|
|
|
}
|
2012-01-16 09:00:56 +02:00
|
|
|
|
2012-02-07 11:44:55 +02:00
|
|
|
// otherwise, trigger the outro animation and notify necessary observers
|
2013-04-10 02:05:09 +03:00
|
|
|
Services.obs.notifyObservers(content, TILT_NOTIFICATIONS.DESTROYING, null);
|
2012-01-16 09:00:57 +02:00
|
|
|
|
2012-02-07 11:44:55 +02:00
|
|
|
controller.removeEventListeners();
|
|
|
|
controller.arcball.reset([-pageXOffset, -pageYOffset]);
|
|
|
|
presenter.executeDestruction(this._finish.bind(this, aId));
|
|
|
|
},
|
2012-01-19 17:48:22 +02:00
|
|
|
|
2012-02-07 11:44:55 +02:00
|
|
|
/**
|
|
|
|
* Finishes detroying a specific instance of the visualizer.
|
|
|
|
*
|
|
|
|
* @param {String} aId
|
|
|
|
* the identifier of the instance in the visualizers array
|
|
|
|
*/
|
|
|
|
_finish: function T__finish(aId)
|
|
|
|
{
|
2013-04-10 02:05:09 +03:00
|
|
|
let contentWindow = this.visualizers[aId].presenter.contentWindow;
|
2012-02-07 11:44:55 +02:00
|
|
|
this.visualizers[aId].removeOverlay();
|
|
|
|
this.visualizers[aId].cleanup();
|
|
|
|
this.visualizers[aId] = null;
|
2012-01-16 09:00:56 +02:00
|
|
|
|
2012-02-07 11:44:55 +02:00
|
|
|
this._isDestroying = false;
|
|
|
|
this.chromeWindow.gBrowser.selectedBrowser.focus();
|
2012-12-19 02:17:39 +05:30
|
|
|
this.emit("change", this.chromeWindow.gBrowser.selectedTab);
|
2013-04-10 02:05:09 +03:00
|
|
|
Services.obs.notifyObservers(contentWindow, TILT_NOTIFICATIONS.DESTROYED, null);
|
2011-11-10 12:09:18 +02:00
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Handles the event fired when a tab is selected.
|
|
|
|
*/
|
|
|
|
_onTabSelect: function T__onTabSelect()
|
|
|
|
{
|
2013-04-10 02:05:09 +03:00
|
|
|
if (this.visualizers[this.lastInstanceId]) {
|
|
|
|
let contentWindow = this.visualizers[this.lastInstanceId].presenter.contentWindow;
|
|
|
|
Services.obs.notifyObservers(contentWindow, TILT_NOTIFICATIONS.HIDDEN, null);
|
|
|
|
}
|
|
|
|
|
2012-01-27 23:44:59 +02:00
|
|
|
if (this.currentInstance) {
|
2013-04-10 02:05:09 +03:00
|
|
|
let contentWindow = this.currentInstance.presenter.contentWindow;
|
|
|
|
Services.obs.notifyObservers(contentWindow, TILT_NOTIFICATIONS.SHOWN, null);
|
2011-11-10 12:09:18 +02:00
|
|
|
}
|
2013-04-10 02:05:09 +03:00
|
|
|
|
|
|
|
this.lastInstanceId = this.currentWindowId;
|
2011-11-10 12:09:18 +02:00
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Add the browser event listeners to handle state changes.
|
|
|
|
*/
|
|
|
|
setup: function T_setup()
|
|
|
|
{
|
|
|
|
// load the preferences from the devtools.tilt branch
|
|
|
|
TiltVisualizer.Prefs.load();
|
|
|
|
|
2012-11-30 08:07:59 +00:00
|
|
|
this.chromeWindow.gBrowser.tabContainer.addEventListener(
|
|
|
|
"TabSelect", this._onTabSelect.bind(this), false);
|
2011-11-10 12:09:18 +02:00
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns true if this tool is enabled.
|
|
|
|
*/
|
|
|
|
get enabled()
|
|
|
|
{
|
|
|
|
return (TiltVisualizer.Prefs.enabled &&
|
2012-01-13 15:27:12 +02:00
|
|
|
(TiltGL.isWebGLForceEnabled() || TiltGL.isWebGLSupported()));
|
2011-11-10 12:09:18 +02:00
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets the ID of the current window object to identify the visualizer.
|
|
|
|
*/
|
|
|
|
get currentWindowId()
|
|
|
|
{
|
2012-01-27 23:44:59 +02:00
|
|
|
return TiltUtils.getWindowId(
|
|
|
|
this.chromeWindow.gBrowser.selectedBrowser.contentWindow);
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets the visualizer instance for the current tab.
|
|
|
|
*/
|
|
|
|
get currentInstance()
|
|
|
|
{
|
|
|
|
return this.visualizers[this.currentWindowId];
|
2011-11-10 12:09:18 +02:00
|
|
|
},
|
|
|
|
};
|