blob: 6ab7a2d8fc2ce08354d34edd85055153ccd12268 [file] [log] [blame]
/*
* Copyright (C) 2011 Google Inc. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met:
*
* * Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* * Redistributions in binary form must reproduce the above
* copyright notice, this list of conditions and the following disclaimer
* in the documentation and/or other materials provided with the
* distribution.
* * Neither the name of Google Inc. nor the names of its
* contributors may be used to endorse or promote products derived from
* this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
/**
* @constructor
* @extends {WebInspector.VBox}
*/
WebInspector.InspectorView = function()
{
WebInspector.VBox.call(this);
WebInspector.Dialog.setModalHostView(this);
this.setMinimumSize(240, 72);
// DevTools sidebar is a vertical split of panels tabbed pane and a drawer.
this._drawerSplitWidget = new WebInspector.SplitWidget(false, true, "Inspector.drawerSplitViewState", 200, 200);
this._drawerSplitWidget.hideSidebar();
this._drawerSplitWidget.enableShowModeSaving();
this._drawerSplitWidget.show(this.element);
this._tabbedPane = new WebInspector.TabbedPane();
this._tabbedPane.registerRequiredCSS("ui/inspectorViewTabbedPane.css");
this._tabbedPane.element.classList.add("inspector-view-tabbed-pane");
this._tabbedPane.setTabSlider(true);
this._tabbedPane.setAllowTabReorder(true, false);
this._tabbedPane.addEventListener(WebInspector.TabbedPane.EventTypes.TabOrderChanged, this._persistPanelOrder, this);
this._tabOrderSetting = WebInspector.settings.createSetting("InspectorView.panelOrder", {});
this._drawerSplitWidget.setMainWidget(this._tabbedPane);
this._drawer = new WebInspector.Drawer(this._drawerSplitWidget);
this._panels = {};
// Used by tests.
WebInspector["panels"] = this._panels;
this._history = [];
this._historyIterator = -1;
this._keyDownBound = this._keyDown.bind(this);
this._keyPressBound = this._keyPress.bind(this);
/** @type {!Object.<string, !WebInspector.PanelDescriptor>} */
this._panelDescriptors = {};
/** @type {!Object.<string, !Promise.<!WebInspector.Panel> >} */
this._panelPromises = {};
this._lastActivePanelSetting = WebInspector.settings.createSetting("lastActivePanel", "elements");
InspectorFrontendHost.events.addEventListener(InspectorFrontendHostAPI.Events.ShowPanel, showPanel.bind(this));
this._loadPanelDesciptors();
/**
* @this {WebInspector.InspectorView}
* @param {!WebInspector.Event} event
*/
function showPanel(event)
{
var panelName = /** @type {string} */ (event.data);
this.showPanel(panelName);
}
};
WebInspector.InspectorView.prototype = {
wasShown: function()
{
this.element.ownerDocument.addEventListener("keydown", this._keyDownBound, false);
this.element.ownerDocument.addEventListener("keypress", this._keyPressBound, false);
},
willHide: function()
{
this.element.ownerDocument.removeEventListener("keydown", this._keyDownBound, false);
this.element.ownerDocument.removeEventListener("keypress", this._keyPressBound, false);
},
_loadPanelDesciptors: function()
{
/**
* @param {!Runtime.Extension} extension
* @this {!WebInspector.InspectorView}
*/
function processPanelExtensions(extension)
{
var descriptor = new WebInspector.RuntimeExtensionPanelDescriptor(extension);
var weight = this._tabOrderSetting.get()[descriptor.name()];
if (weight === undefined)
weight = extension.descriptor()["order"];
if (weight === undefined)
weight = 10000;
panelWeights.set(descriptor, weight);
}
/**
* @param {!WebInspector.PanelDescriptor} left
* @param {!WebInspector.PanelDescriptor} right
*/
function orderComparator(left, right)
{
return panelWeights.get(left) > panelWeights.get(right);
}
WebInspector.startBatchUpdate();
/** @type {!Map<!WebInspector.PanelDescriptor, number>} */
var panelWeights = new Map();
self.runtime.extensions(WebInspector.PanelFactory).forEach(processPanelExtensions.bind(this));
var sortedPanels = panelWeights.keysArray().sort(orderComparator);
for (var panelDescriptor of sortedPanels)
this._innerAddPanel(panelDescriptor);
WebInspector.endBatchUpdate();
},
createToolbars: function()
{
this._leftToolbar = new WebInspector.ExtensibleToolbar("main-toolbar-left");
this._leftToolbar.element.classList.add("inspector-view-toolbar", "inspector-view-toolbar-left");
this._tabbedPane.insertBeforeTabStrip(this._leftToolbar.element);
var rightToolbarContainer = createElementWithClass("div", "hbox flex-none flex-centered");
this._tabbedPane.appendAfterTabStrip(rightToolbarContainer);
this._rightToolbar = new WebInspector.ExtensibleToolbar("main-toolbar-right");
this._rightToolbar.element.classList.add("inspector-view-toolbar", "flex-none");
rightToolbarContainer.appendChild(this._rightToolbar.element);
},
/**
* @param {!WebInspector.PanelDescriptor} panelDescriptor
* @param {number=} index
*/
_innerAddPanel: function(panelDescriptor, index)
{
var panelName = panelDescriptor.name();
this._panelDescriptors[panelName] = panelDescriptor;
this._tabbedPane.appendTab(panelName, panelDescriptor.title(), new WebInspector.Widget(), undefined, undefined, undefined, index);
if (this._lastActivePanelSetting.get() === panelName)
this._tabbedPane.selectTab(panelName);
},
/**
* @param {!WebInspector.PanelDescriptor} panelDescriptor
*/
addPanel: function(panelDescriptor)
{
var weight = this._tabOrderSetting.get()[panelDescriptor.name()];
// Keep in sync with _persistPanelOrder().
if (weight)
weight = Math.max(0, Math.round(weight / 10) - 1);
this._innerAddPanel(panelDescriptor, weight);
},
/**
* @param {string} panelName
* @return {boolean}
*/
hasPanel: function(panelName)
{
return !!this._panelDescriptors[panelName];
},
/**
* @param {string} panelName
* @return {!Promise.<!WebInspector.Panel>}
*/
panel: function(panelName)
{
var panelDescriptor = this._panelDescriptors[panelName];
if (!panelDescriptor)
return Promise.reject(new Error("Can't load panel without the descriptor: " + panelName));
var promise = this._panelPromises[panelName];
if (promise)
return promise;
promise = panelDescriptor.panel();
this._panelPromises[panelName] = promise;
promise.then(cachePanel.bind(this));
/**
* @param {!WebInspector.Panel} panel
* @return {!WebInspector.Panel}
* @this {WebInspector.InspectorView}
*/
function cachePanel(panel)
{
delete this._panelPromises[panelName];
this._panels[panelName] = panel;
return panel;
}
return promise;
},
/**
* @param {boolean} allTargetsSuspended
*/
onSuspendStateChanged: function(allTargetsSuspended)
{
this._currentPanelLocked = allTargetsSuspended;
this._tabbedPane.setCurrentTabLocked(this._currentPanelLocked);
if (this._leftToolbar)
this._leftToolbar.setEnabled(!this._currentPanelLocked);
if (this._rightToolbar)
this._rightToolbar.setEnabled(!this._currentPanelLocked);
},
/**
* The returned Promise is resolved with null if another showPanel()
* gets called while this.panel(panelName) Promise is in flight.
*
* @param {string} panelName
* @return {!Promise.<?WebInspector.Panel>}
*/
showPanel: function(panelName)
{
if (this._currentPanelLocked) {
if (this._currentPanel !== this._panels[panelName])
return Promise.reject(new Error("Current panel locked"));
return Promise.resolve(this._currentPanel);
}
this._panelForShowPromise = this.panel(panelName);
return this._panelForShowPromise.then(setCurrentPanelIfNecessary.bind(this, this._panelForShowPromise));
/**
* @param {!Promise.<!WebInspector.Panel>} panelPromise
* @param {!WebInspector.Panel} panel
* @return {?WebInspector.Panel}
* @this {WebInspector.InspectorView}
*/
function setCurrentPanelIfNecessary(panelPromise, panel)
{
if (this._panelForShowPromise !== panelPromise)
return null;
this.setCurrentPanel(panel);
return panel;
}
},
/**
* @param {string} panelName
* @param {string} iconType
* @param {string=} iconTooltip
*/
setPanelIcon: function(panelName, iconType, iconTooltip)
{
this._tabbedPane.setTabIcon(panelName, iconType, iconTooltip);
},
/**
* @return {!WebInspector.Panel}
*/
currentPanel: function()
{
return this._currentPanel;
},
showInitialPanel: function()
{
if (InspectorFrontendHost.isUnderTest())
return;
this._showInitialPanel();
},
_showInitialPanel: function()
{
this._tabbedPane.addEventListener(WebInspector.TabbedPane.EventTypes.TabSelected, this._tabSelected, this);
this._tabSelected();
this._drawer.initialPanelShown();
},
/**
* @param {string} panelName
*/
showInitialPanelForTest: function(panelName)
{
this._tabbedPane.addEventListener(WebInspector.TabbedPane.EventTypes.TabSelected, this._tabSelected, this);
this.setCurrentPanel(this._panels[panelName]);
this._drawer.initialPanelShown();
},
_tabSelected: function()
{
var panelName = this._tabbedPane.selectedTabId;
if (!panelName)
return;
this.showPanel(panelName);
},
/**
* @param {!WebInspector.Panel} panel
* @param {boolean=} suppressBringToFront
* @return {!WebInspector.Panel}
*/
setCurrentPanel: function(panel, suppressBringToFront)
{
delete this._panelForShowPromise;
if (this._currentPanelLocked)
return this._currentPanel;
if (!suppressBringToFront)
InspectorFrontendHost.bringToFront();
if (this._currentPanel === panel)
return panel;
this._currentPanel = panel;
if (!this._panels[panel.name])
this._panels[panel.name] = panel;
this._tabbedPane.changeTabView(panel.name, panel);
this._tabbedPane.removeEventListener(WebInspector.TabbedPane.EventTypes.TabSelected, this._tabSelected, this);
this._tabbedPane.selectTab(panel.name);
this._tabbedPane.addEventListener(WebInspector.TabbedPane.EventTypes.TabSelected, this._tabSelected, this);
this._lastActivePanelSetting.set(panel.name);
this._pushToHistory(panel.name);
WebInspector.userMetrics.panelShown(panel.name);
panel.focus();
return panel;
},
showDrawer: function()
{
this._drawer.showDrawer();
},
/**
* @return {boolean}
*/
drawerVisible: function()
{
return this._drawer.isShowing();
},
/**
* @param {string} id
* @param {boolean=} immediate
* @return {!Promise.<?WebInspector.Widget>}
*/
showViewInDrawer: function(id, immediate)
{
return this._drawer.showView(id, immediate);
},
/**
* @return {?string}
*/
selectedViewInDrawer: function()
{
return this._drawer.selectedViewId();
},
closeDrawer: function()
{
this._drawer.closeDrawer();
},
/**
* @param {boolean} minimized
*/
setDrawerMinimized: function(minimized)
{
this._drawerSplitWidget.setSidebarMinimized(minimized);
this._drawerSplitWidget.setResizable(!minimized);
},
/**
* @return {boolean}
*/
isDrawerMinimized: function()
{
return this._drawerSplitWidget.isSidebarMinimized();
},
/**
* @override
* @return {!Element}
*/
defaultFocusedElement: function()
{
if (this._drawer.hasFocus())
return this._drawer.defaultFocusedElement();
return this._currentPanel ? this._currentPanel.defaultFocusedElement() : null;
},
_keyPress: function(event)
{
// BUG 104250: Windows 7 posts a WM_CHAR message upon the Ctrl+']' keypress.
// Any charCode < 32 is not going to be a valid keypress.
if (event.charCode < 32 && WebInspector.isWin())
return;
clearTimeout(this._keyDownTimer);
delete this._keyDownTimer;
},
_keyDown: function(event)
{
if (!WebInspector.KeyboardShortcut.eventHasCtrlOrMeta(event))
return;
var keyboardEvent = /** @type {!KeyboardEvent} */ (event);
// Ctrl/Cmd + 1-9 should show corresponding panel.
var panelShortcutEnabled = WebInspector.moduleSetting("shortcutPanelSwitch").get();
if (panelShortcutEnabled && !event.shiftKey && !event.altKey) {
var panelIndex = -1;
if (event.keyCode > 0x30 && event.keyCode < 0x3A)
panelIndex = event.keyCode - 0x31;
else if (event.keyCode > 0x60 && event.keyCode < 0x6A && keyboardEvent.location === KeyboardEvent.DOM_KEY_LOCATION_NUMPAD)
panelIndex = event.keyCode - 0x61;
if (panelIndex !== -1) {
var panelName = this._tabbedPane.allTabs()[panelIndex];
if (panelName) {
if (!WebInspector.Dialog.hasInstance() && !this._currentPanelLocked)
this.showPanel(panelName);
event.consume(true);
}
return;
}
}
// BUG85312: On French AZERTY keyboards, AltGr-]/[ combinations (synonymous to Ctrl-Alt-]/[ on Windows) are used to enter ]/[,
// so for a ]/[-related keydown we delay the panel switch using a timer, to see if there is a keypress event following this one.
// If there is, we cancel the timer and do not consider this a panel switch.
if (!WebInspector.isWin() || (event.key !== "[" && event.key !== "]")) {
this._keyDownInternal(event);
return;
}
this._keyDownTimer = setTimeout(this._keyDownInternal.bind(this, event), 0);
},
_keyDownInternal: function(event)
{
if (this._currentPanelLocked)
return;
var direction = 0;
if (event.key === "[")
direction = -1;
if (event.key === "]")
direction = 1;
if (!direction)
return;
if (!event.shiftKey && !event.altKey) {
if (!WebInspector.Dialog.hasInstance())
this._changePanelInDirection(direction);
event.consume(true);
return;
}
if (event.altKey && this._moveInHistory(direction))
event.consume(true);
},
/**
* @param {number} direction
*/
_changePanelInDirection: function(direction)
{
var panelOrder = this._tabbedPane.allTabs();
var index = panelOrder.indexOf(this.currentPanel().name);
index = (index + panelOrder.length + direction) % panelOrder.length;
this.showPanel(panelOrder[index]);
},
/**
* @param {number} move
*/
_moveInHistory: function(move)
{
var newIndex = this._historyIterator + move;
if (newIndex >= this._history.length || newIndex < 0)
return false;
this._inHistory = true;
this._historyIterator = newIndex;
if (!WebInspector.Dialog.hasInstance())
this.setCurrentPanel(this._panels[this._history[this._historyIterator]]);
delete this._inHistory;
return true;
},
_pushToHistory: function(panelName)
{
if (this._inHistory)
return;
this._history.splice(this._historyIterator + 1, this._history.length - this._historyIterator - 1);
if (!this._history.length || this._history[this._history.length - 1] !== panelName)
this._history.push(panelName);
this._historyIterator = this._history.length - 1;
},
onResize: function()
{
WebInspector.Dialog.modalHostRepositioned();
},
/**
* @return {!Element}
*/
topResizerElement: function()
{
return this._tabbedPane.headerElement();
},
toolbarItemResized: function()
{
this._tabbedPane.headerResized();
},
/**
* @param {!WebInspector.Event} event
*/
_persistPanelOrder: function(event)
{
var tabs = /** @type {!Array.<!WebInspector.TabbedPaneTab>} */(event.data);
var tabOrders = this._tabOrderSetting.get();
for (var i = 0; i < tabs.length; i++)
tabOrders[tabs[i].id] = (i + 1) * 10;
this._tabOrderSetting.set(tabOrders);
},
/**
* @param {!WebInspector.SplitWidget} splitWidget
*/
setOwnerSplit: function(splitWidget)
{
this._ownerSplitWidget = splitWidget;
},
minimize: function()
{
if (this._ownerSplitWidget)
this._ownerSplitWidget.setSidebarMinimized(true);
},
restore: function()
{
if (this._ownerSplitWidget)
this._ownerSplitWidget.setSidebarMinimized(false);
},
__proto__: WebInspector.VBox.prototype
};
/**
* @type {!WebInspector.InspectorView}
*/
WebInspector.inspectorView;
/**
* @constructor
* @implements {WebInspector.ActionDelegate}
*/
WebInspector.InspectorView.DrawerToggleActionDelegate = function()
{
}
WebInspector.InspectorView.DrawerToggleActionDelegate.prototype = {
/**
* @override
* @param {!WebInspector.Context} context
* @param {string} actionId
* @return {boolean}
*/
handleAction: function(context, actionId)
{
if (WebInspector.inspectorView.drawerVisible())
WebInspector.inspectorView.closeDrawer();
else
WebInspector.inspectorView.showDrawer();
return true;
}
}