summaryrefslogtreecommitdiffstats
path: root/chromium/third_party/WebKit/Source/devtools/front_end/elements/StylesSidebarPane.js
diff options
context:
space:
mode:
Diffstat (limited to 'chromium/third_party/WebKit/Source/devtools/front_end/elements/StylesSidebarPane.js')
-rw-r--r--chromium/third_party/WebKit/Source/devtools/front_end/elements/StylesSidebarPane.js3327
1 files changed, 3327 insertions, 0 deletions
diff --git a/chromium/third_party/WebKit/Source/devtools/front_end/elements/StylesSidebarPane.js b/chromium/third_party/WebKit/Source/devtools/front_end/elements/StylesSidebarPane.js
new file mode 100644
index 00000000000..51f5ddaf6ab
--- /dev/null
+++ b/chromium/third_party/WebKit/Source/devtools/front_end/elements/StylesSidebarPane.js
@@ -0,0 +1,3327 @@
+/*
+ * Copyright (C) 2007 Apple Inc. All rights reserved.
+ * Copyright (C) 2009 Joseph Pecoraro
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. 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.
+ * 3. Neither the name of Apple Computer, Inc. ("Apple") 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 APPLE AND ITS 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 APPLE OR ITS 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.SidebarPane}
+ * @param {!WebInspector.ComputedStyleSidebarPane} computedStylePane
+ * @param {function(!WebInspector.DOMNode, string, boolean)=} setPseudoClassCallback
+ */
+WebInspector.StylesSidebarPane = function(computedStylePane, setPseudoClassCallback)
+{
+ WebInspector.SidebarPane.call(this, WebInspector.UIString("Styles"));
+
+ this._elementStateButton = document.createElement("button");
+ this._elementStateButton.className = "pane-title-button element-state";
+ this._elementStateButton.title = WebInspector.UIString("Toggle Element State");
+ this._elementStateButton.addEventListener("click", this._toggleElementStatePane.bind(this), false);
+ this.titleElement.appendChild(this._elementStateButton);
+
+ var addButton = document.createElement("button");
+ addButton.className = "pane-title-button add";
+ addButton.id = "add-style-button-test-id";
+ addButton.title = WebInspector.UIString("New Style Rule");
+ addButton.addEventListener("click", this._createNewRule.bind(this), false);
+ this.titleElement.appendChild(addButton);
+
+ this._computedStylePane = computedStylePane;
+ computedStylePane.setHostingPane(this);
+ this._setPseudoClassCallback = setPseudoClassCallback;
+ this.element.addEventListener("contextmenu", this._contextMenuEventFired.bind(this), true);
+ WebInspector.settings.colorFormat.addChangeListener(this._colorFormatSettingChanged.bind(this));
+ WebInspector.settings.showUserAgentStyles.addChangeListener(this._showUserAgentStylesSettingChanged.bind(this));
+
+ this._createElementStatePane();
+ this.bodyElement.appendChild(this._elementStatePane);
+ this._sectionsContainer = document.createElement("div");
+ this.bodyElement.appendChild(this._sectionsContainer);
+
+ this._spectrumHelper = new WebInspector.SpectrumPopupHelper();
+ this._linkifier = new WebInspector.Linkifier(new WebInspector.Linkifier.DefaultCSSFormatter());
+
+ this.element.classList.add("styles-pane");
+ this.element.classList.toggle("show-user-styles", WebInspector.settings.showUserAgentStyles.get());
+ this.element.addEventListener("mousemove", this._mouseMovedOverElement.bind(this), false);
+ document.body.addEventListener("keydown", this._keyDown.bind(this), false);
+ document.body.addEventListener("keyup", this._keyUp.bind(this), false);
+}
+
+// Keep in sync with RenderStyleConstants.h PseudoId enum. Array below contains pseudo id names for corresponding enum indexes.
+// First item is empty due to its artificial NOPSEUDO nature in the enum.
+// FIXME: find a way of generating this mapping or getting it from combination of RenderStyleConstants and CSSSelector.cpp at
+// runtime.
+WebInspector.StylesSidebarPane.PseudoIdNames = [
+ "", "first-line", "first-letter", "before", "after", "selection", "", "-webkit-scrollbar", "-webkit-file-upload-button",
+ "-webkit-input-placeholder", "-webkit-slider-thumb", "-webkit-search-cancel-button", "-webkit-search-decoration",
+ "-webkit-search-results-decoration", "-webkit-search-results-button", "-webkit-media-controls-panel",
+ "-webkit-media-controls-play-button", "-webkit-media-controls-mute-button", "-webkit-media-controls-timeline",
+ "-webkit-media-controls-timeline-container", "-webkit-media-controls-volume-slider",
+ "-webkit-media-controls-volume-slider-container", "-webkit-media-controls-current-time-display",
+ "-webkit-media-controls-time-remaining-display", "-webkit-media-controls-fullscreen-button",
+ "-webkit-media-controls-toggle-closed-captions-button", "-webkit-media-controls-status-display", "-webkit-scrollbar-thumb",
+ "-webkit-scrollbar-button", "-webkit-scrollbar-track", "-webkit-scrollbar-track-piece", "-webkit-scrollbar-corner",
+ "-webkit-resizer", "-webkit-inner-spin-button", "-webkit-outer-spin-button"
+];
+
+WebInspector.StylesSidebarPane._colorRegex = /((?:rgb|hsl)a?\([^)]+\)|#[0-9a-fA-F]{6}|#[0-9a-fA-F]{3}|\b\w+\b(?!-))/g;
+
+/**
+ * @param {!WebInspector.CSSProperty} property
+ * @return {!Element}
+ */
+WebInspector.StylesSidebarPane.createExclamationMark = function(property)
+{
+ var exclamationElement = document.createElement("div");
+ exclamationElement.className = "exclamation-mark" + (WebInspector.StylesSidebarPane._ignoreErrorsForProperty(property) ? "" : " warning-icon-small");
+ exclamationElement.title = WebInspector.CSSMetadata.cssPropertiesMetainfo.keySet()[property.name.toLowerCase()] ? WebInspector.UIString("Invalid property value.") : WebInspector.UIString("Unknown property name.");
+ return exclamationElement;
+}
+
+/**
+ * @param {!WebInspector.Color} color
+ */
+WebInspector.StylesSidebarPane._colorFormat = function(color)
+{
+ const cf = WebInspector.Color.Format;
+ var format;
+ var formatSetting = WebInspector.settings.colorFormat.get();
+ if (formatSetting === cf.Original)
+ format = cf.Original;
+ else if (formatSetting === cf.RGB)
+ format = (color.hasAlpha() ? cf.RGBA : cf.RGB);
+ else if (formatSetting === cf.HSL)
+ format = (color.hasAlpha() ? cf.HSLA : cf.HSL);
+ else if (!color.hasAlpha())
+ format = (color.canBeShortHex() ? cf.ShortHEX : cf.HEX);
+ else
+ format = cf.RGBA;
+
+ return format;
+}
+
+/**
+ * @param {!WebInspector.CSSProperty} property
+ */
+WebInspector.StylesSidebarPane._ignoreErrorsForProperty = function(property) {
+ function hasUnknownVendorPrefix(string)
+ {
+ return !string.startsWith("-webkit-") && /^[-_][\w\d]+-\w/.test(string);
+ }
+
+ var name = property.name.toLowerCase();
+
+ // IE hack.
+ if (name.charAt(0) === "_")
+ return true;
+
+ // IE has a different format for this.
+ if (name === "filter")
+ return true;
+
+ // Common IE-specific property prefix.
+ if (name.startsWith("scrollbar-"))
+ return true;
+ if (hasUnknownVendorPrefix(name))
+ return true;
+
+ var value = property.value.toLowerCase();
+
+ // IE hack.
+ if (value.endsWith("\9"))
+ return true;
+ if (hasUnknownVendorPrefix(value))
+ return true;
+
+ return false;
+}
+
+WebInspector.StylesSidebarPane.prototype = {
+ /**
+ * @param {!WebInspector.CSSRule} editedRule
+ * @param {!WebInspector.TextRange} oldRange
+ * @param {!WebInspector.TextRange} newRange
+ */
+ _styleSheetRuleEdited: function(editedRule, oldRange, newRange)
+ {
+ var styleRuleSections = this.sections[0];
+ for (var i = 1; i < styleRuleSections.length; ++i)
+ styleRuleSections[i]._styleSheetRuleEdited(editedRule, oldRange, newRange);
+ },
+
+ /**
+ * @param {?Event} event
+ */
+ _contextMenuEventFired: function(event)
+ {
+ // We start editing upon click -> default navigation to resources panel is not available
+ // Hence we add a soft context menu for hrefs.
+ var contextMenu = new WebInspector.ContextMenu(event);
+ contextMenu.appendApplicableItems(/** @type {!Node} */ (event.target));
+ contextMenu.show();
+ },
+
+ /**
+ * @param {!Element} matchedStylesElement
+ * @param {!Element} computedStylesElement
+ */
+ setFilterBoxContainers: function(matchedStylesElement, computedStylesElement)
+ {
+ matchedStylesElement.appendChild(this._createCSSFilterControl());
+ this._computedStylePane.setFilterBoxContainer(computedStylesElement);
+ },
+
+ /**
+ * @return {!Element}
+ */
+ _createCSSFilterControl: function()
+ {
+ var filterInput = this._createPropertyFilterElement(false, searchHandler.bind(this));
+
+ /**
+ * @param {?RegExp} regex
+ * @this {WebInspector.StylesSidebarPane}
+ */
+ function searchHandler(regex)
+ {
+ this._filterRegex = regex;
+ }
+
+ return filterInput;
+ },
+
+ get _forcedPseudoClasses()
+ {
+ return this._node ? (this._node.getUserProperty(WebInspector.CSSStyleModel.PseudoStatePropertyName) || undefined) : undefined;
+ },
+
+ _updateForcedPseudoStateInputs: function()
+ {
+ if (!this._node)
+ return;
+
+ var hasPseudoType = !!this._node.pseudoType();
+ this._elementStateButton.classList.toggle("hidden", hasPseudoType);
+ this._elementStatePane.classList.toggle("expanded", !hasPseudoType && this._elementStateButton.classList.contains("toggled"));
+
+ var nodePseudoState = this._forcedPseudoClasses;
+ if (!nodePseudoState)
+ nodePseudoState = [];
+
+ var inputs = this._elementStatePane.inputs;
+ for (var i = 0; i < inputs.length; ++i)
+ inputs[i].checked = nodePseudoState.indexOf(inputs[i].state) >= 0;
+ },
+
+ /**
+ * @param {?WebInspector.DOMNode} node
+ * @param {boolean=} forceUpdate
+ */
+ update: function(node, forceUpdate)
+ {
+ this._spectrumHelper.hide();
+ this._discardElementUnderMouse();
+
+ var refresh = false;
+
+ if (forceUpdate)
+ delete this._node;
+
+ if (!forceUpdate && (node === this._node))
+ refresh = true;
+
+ if (node && node.nodeType() === Node.TEXT_NODE && node.parentNode)
+ node = node.parentNode;
+
+ if (node && node.nodeType() !== Node.ELEMENT_NODE)
+ node = null;
+
+ if (node) {
+ this._updateTarget(node.target());
+ this._node = node;
+ } else
+ node = this._node;
+
+ this._updateForcedPseudoStateInputs();
+
+ if (refresh)
+ this._refreshUpdate();
+ else
+ this._rebuildUpdate();
+ },
+
+ /**
+ * @param {!WebInspector.Target} target
+ */
+ _updateTarget: function(target)
+ {
+ if (this._target === target)
+ return;
+ if (this._target) {
+ this._target.cssModel.removeEventListener(WebInspector.CSSStyleModel.Events.StyleSheetAdded, this._styleSheetOrMediaQueryResultChanged, this);
+ this._target.cssModel.removeEventListener(WebInspector.CSSStyleModel.Events.StyleSheetRemoved, this._styleSheetOrMediaQueryResultChanged, this);
+ this._target.cssModel.removeEventListener(WebInspector.CSSStyleModel.Events.StyleSheetChanged, this._styleSheetOrMediaQueryResultChanged, this);
+ this._target.cssModel.removeEventListener(WebInspector.CSSStyleModel.Events.MediaQueryResultChanged, this._styleSheetOrMediaQueryResultChanged, this);
+ this._target.domModel.removeEventListener(WebInspector.DOMModel.Events.AttrModified, this._attributeChanged, this);
+ this._target.domModel.removeEventListener(WebInspector.DOMModel.Events.AttrRemoved, this._attributeChanged, this);
+ this._target.resourceTreeModel.removeEventListener(WebInspector.ResourceTreeModel.EventTypes.FrameResized, this._frameResized, this);
+ }
+ this._target = target;
+ this._target.cssModel.addEventListener(WebInspector.CSSStyleModel.Events.StyleSheetAdded, this._styleSheetOrMediaQueryResultChanged, this);
+ this._target.cssModel.addEventListener(WebInspector.CSSStyleModel.Events.StyleSheetRemoved, this._styleSheetOrMediaQueryResultChanged, this);
+ this._target.cssModel.addEventListener(WebInspector.CSSStyleModel.Events.StyleSheetChanged, this._styleSheetOrMediaQueryResultChanged, this);
+ this._target.cssModel.addEventListener(WebInspector.CSSStyleModel.Events.MediaQueryResultChanged, this._styleSheetOrMediaQueryResultChanged, this);
+ this._target.domModel.addEventListener(WebInspector.DOMModel.Events.AttrModified, this._attributeChanged, this);
+ this._target.domModel.addEventListener(WebInspector.DOMModel.Events.AttrRemoved, this._attributeChanged, this);
+ this._target.resourceTreeModel.addEventListener(WebInspector.ResourceTreeModel.EventTypes.FrameResized, this._frameResized, this);
+ },
+
+ /**
+ * @param {!WebInspector.StylePropertiesSection=} editedSection
+ * @param {boolean=} forceFetchComputedStyle
+ * @param {function()=} userCallback
+ */
+ _refreshUpdate: function(editedSection, forceFetchComputedStyle, userCallback)
+ {
+ var callbackWrapper = function()
+ {
+ if (this._filterRegex)
+ this._updateFilter(false);
+ if (userCallback)
+ userCallback();
+ }.bind(this);
+
+ if (this._refreshUpdateInProgress) {
+ this._lastNodeForInnerRefresh = this._node;
+ return;
+ }
+
+ var node = this._validateNode(userCallback);
+ if (!node)
+ return;
+
+ /**
+ * @param {?WebInspector.CSSStyleDeclaration} computedStyle
+ * @this {WebInspector.StylesSidebarPane}
+ */
+ function computedStyleCallback(computedStyle)
+ {
+ delete this._refreshUpdateInProgress;
+
+ if (this._lastNodeForInnerRefresh) {
+ delete this._lastNodeForInnerRefresh;
+ this._refreshUpdate(editedSection, forceFetchComputedStyle, callbackWrapper);
+ return;
+ }
+
+ if (this._node === node && computedStyle)
+ this._innerRefreshUpdate(node, computedStyle, editedSection);
+
+ callbackWrapper();
+ }
+
+ if (this._computedStylePane.isShowing() || forceFetchComputedStyle) {
+ this._refreshUpdateInProgress = true;
+ this._target.cssModel.getComputedStyleAsync(node.id, computedStyleCallback.bind(this));
+ } else {
+ this._innerRefreshUpdate(node, null, editedSection);
+ callbackWrapper();
+ }
+ },
+
+ _rebuildUpdate: function()
+ {
+ if (this._rebuildUpdateInProgress) {
+ this._lastNodeForInnerRebuild = this._node;
+ return;
+ }
+
+ var node = this._validateNode();
+ if (!node)
+ return;
+
+ this._rebuildUpdateInProgress = true;
+
+ var resultStyles = {};
+
+ /**
+ * @param {?*} matchedResult
+ * @this {WebInspector.StylesSidebarPane}
+ */
+ function stylesCallback(matchedResult)
+ {
+ delete this._rebuildUpdateInProgress;
+
+ var lastNodeForRebuild = this._lastNodeForInnerRebuild;
+ if (lastNodeForRebuild) {
+ delete this._lastNodeForInnerRebuild;
+ if (lastNodeForRebuild !== this._node) {
+ this._rebuildUpdate();
+ return;
+ }
+ }
+
+ if (matchedResult && this._node === node) {
+ resultStyles.matchedCSSRules = matchedResult.matchedCSSRules;
+ resultStyles.pseudoElements = matchedResult.pseudoElements;
+ resultStyles.inherited = matchedResult.inherited;
+ this._innerRebuildUpdate(node, resultStyles);
+ }
+
+ if (lastNodeForRebuild) {
+ // lastNodeForRebuild is the same as this.node - another rebuild has been requested.
+ this._rebuildUpdate();
+ return;
+ }
+ }
+
+ /**
+ * @param {?WebInspector.CSSStyleDeclaration} inlineStyle
+ * @param {?WebInspector.CSSStyleDeclaration} attributesStyle
+ */
+ function inlineCallback(inlineStyle, attributesStyle)
+ {
+ resultStyles.inlineStyle = inlineStyle;
+ resultStyles.attributesStyle = attributesStyle;
+ }
+
+ /**
+ * @param {?WebInspector.CSSStyleDeclaration} computedStyle
+ */
+ function computedCallback(computedStyle)
+ {
+ resultStyles.computedStyle = computedStyle;
+ }
+
+ if (this._computedStylePane.isShowing())
+ this._target.cssModel.getComputedStyleAsync(node.id, computedCallback);
+ this._target.cssModel.getInlineStylesAsync(node.id, inlineCallback);
+ this._target.cssModel.getMatchedStylesAsync(node.id, true, true, stylesCallback.bind(this));
+ },
+
+ /**
+ * @param {function()=} userCallback
+ */
+ _validateNode: function(userCallback)
+ {
+ if (!this._node) {
+ this._sectionsContainer.removeChildren();
+ this._computedStylePane.bodyElement.removeChildren();
+ this.sections = {};
+ if (userCallback)
+ userCallback();
+ return null;
+ }
+ return this._node;
+ },
+
+ _styleSheetOrMediaQueryResultChanged: function()
+ {
+ if (this._userOperation || this._isEditingStyle)
+ return;
+
+ this._rebuildUpdate();
+ },
+
+ _frameResized: function()
+ {
+ /**
+ * @this {WebInspector.StylesSidebarPane}
+ */
+ function refreshContents()
+ {
+ this._styleSheetOrMediaQueryResultChanged();
+ delete this._activeTimer;
+ }
+
+ if (this._activeTimer)
+ clearTimeout(this._activeTimer);
+
+ this._activeTimer = setTimeout(refreshContents.bind(this), 100);
+ },
+
+ _attributeChanged: function(event)
+ {
+ // Any attribute removal or modification can affect the styles of "related" nodes.
+ // Do not touch the styles if they are being edited.
+ if (this._isEditingStyle || this._userOperation)
+ return;
+
+ if (!this._canAffectCurrentStyles(event.data.node))
+ return;
+
+ this._rebuildUpdate();
+ },
+
+ _canAffectCurrentStyles: function(node)
+ {
+ return this._node && (this._node === node || node.parentNode === this._node.parentNode || node.isAncestor(this._node));
+ },
+
+ _innerRefreshUpdate: function(node, computedStyle, editedSection)
+ {
+ for (var pseudoId in this.sections) {
+ var styleRules = this._refreshStyleRules(this.sections[pseudoId], computedStyle);
+ var usedProperties = {};
+ this._markUsedProperties(styleRules, usedProperties);
+ this._refreshSectionsForStyleRules(styleRules, usedProperties, editedSection);
+ }
+ if (computedStyle)
+ this.sections[0][0].rebuildComputedTrace(this.sections[0]);
+
+ this._nodeStylesUpdatedForTest(node, false);
+ },
+
+ _innerRebuildUpdate: function(node, styles)
+ {
+ this._sectionsContainer.removeChildren();
+ this._computedStylePane.bodyElement.removeChildren();
+ this._linkifier.reset();
+
+ var styleRules = this._rebuildStyleRules(node, styles);
+ var usedProperties = {};
+ this._markUsedProperties(styleRules, usedProperties);
+ this.sections[0] = this._rebuildSectionsForStyleRules(styleRules, usedProperties, null);
+ var anchorElement = this.sections[0].inheritedPropertiesSeparatorElement;
+
+ if (styles.computedStyle)
+ this.sections[0][0].rebuildComputedTrace(this.sections[0]);
+
+ for (var i = 0; i < styles.pseudoElements.length; ++i) {
+ var pseudoElementCSSRules = styles.pseudoElements[i];
+
+ styleRules = [];
+ var pseudoId = pseudoElementCSSRules.pseudoId;
+
+ var entry = { isStyleSeparator: true, pseudoId: pseudoId };
+ styleRules.push(entry);
+
+ // Add rules in reverse order to match the cascade order.
+ for (var j = pseudoElementCSSRules.rules.length - 1; j >= 0; --j) {
+ var rule = pseudoElementCSSRules.rules[j];
+ styleRules.push({ style: rule.style, selectorText: rule.selectorText, media: rule.media, sourceURL: rule.resourceURL(), rule: rule, editable: !!(rule.style && rule.style.styleSheetId) });
+ }
+ usedProperties = {};
+ this._markUsedProperties(styleRules, usedProperties);
+ this.sections[pseudoId] = this._rebuildSectionsForStyleRules(styleRules, usedProperties, anchorElement);
+ }
+
+ if (this._filterRegex)
+ this._updateFilter(false);
+ this._nodeStylesUpdatedForTest(node, true);
+ },
+
+ _nodeStylesUpdatedForTest: function(node, rebuild)
+ {
+ // Tests override this method.
+ },
+
+ _refreshStyleRules: function(sections, computedStyle)
+ {
+ var nodeComputedStyle = computedStyle;
+ var styleRules = [];
+ for (var i = 0; sections && i < sections.length; ++i) {
+ var section = sections[i];
+ if (section.isBlank)
+ continue;
+ if (section.computedStyle)
+ section.styleRule.style = nodeComputedStyle;
+ var styleRule = { section: section, style: section.styleRule.style, computedStyle: section.computedStyle, rule: section.rule, editable: !!(section.styleRule.style && section.styleRule.style.styleSheetId),
+ isAttribute: section.styleRule.isAttribute, isInherited: section.styleRule.isInherited, parentNode: section.styleRule.parentNode };
+ styleRules.push(styleRule);
+ }
+ return styleRules;
+ },
+
+ _rebuildStyleRules: function(node, styles)
+ {
+ var nodeComputedStyle = styles.computedStyle;
+ this.sections = {};
+
+ var styleRules = [];
+
+ function addAttributesStyle()
+ {
+ if (!styles.attributesStyle)
+ return;
+ var attrStyle = { style: styles.attributesStyle, editable: false };
+ attrStyle.selectorText = node.nodeNameInCorrectCase() + "[" + WebInspector.UIString("Attributes Style") + "]";
+ styleRules.push(attrStyle);
+ }
+
+ styleRules.push({ computedStyle: true, selectorText: "", style: nodeComputedStyle, editable: false });
+
+ if (!!node.pseudoType())
+ styleRules.push({ isStyleSeparator: true, isPlaceholder: true });
+
+ // Inline style has the greatest specificity.
+ if (styles.inlineStyle && node.nodeType() === Node.ELEMENT_NODE) {
+ var inlineStyle = { selectorText: "element.style", style: styles.inlineStyle, isAttribute: true };
+ styleRules.push(inlineStyle);
+ }
+
+ // Add rules in reverse order to match the cascade order.
+ var addedAttributesStyle;
+ for (var i = styles.matchedCSSRules.length - 1; i >= 0; --i) {
+ var rule = styles.matchedCSSRules[i];
+ if ((rule.isUser || rule.isUserAgent) && !addedAttributesStyle) {
+ // Show element's Style Attributes after all author rules.
+ addedAttributesStyle = true;
+ addAttributesStyle();
+ }
+ styleRules.push({ style: rule.style, selectorText: rule.selectorText, media: rule.media, sourceURL: rule.resourceURL(), rule: rule, editable: !!(rule.style && rule.style.styleSheetId) });
+ }
+
+ if (!addedAttributesStyle)
+ addAttributesStyle();
+
+ // Walk the node structure and identify styles with inherited properties.
+ var parentNode = node.parentNode;
+ function insertInheritedNodeSeparator(node)
+ {
+ var entry = {};
+ entry.isStyleSeparator = true;
+ entry.node = node;
+ styleRules.push(entry);
+ }
+
+ for (var parentOrdinal = 0; parentOrdinal < styles.inherited.length; ++parentOrdinal) {
+ var parentStyles = styles.inherited[parentOrdinal];
+ var separatorInserted = false;
+ if (parentStyles.inlineStyle) {
+ if (this._containsInherited(parentStyles.inlineStyle)) {
+ var inlineStyle = { selectorText: WebInspector.UIString("Style Attribute"), style: parentStyles.inlineStyle, isAttribute: true, isInherited: true, parentNode: parentNode };
+ if (!separatorInserted) {
+ insertInheritedNodeSeparator(parentNode);
+ separatorInserted = true;
+ }
+ styleRules.push(inlineStyle);
+ }
+ }
+
+ for (var i = parentStyles.matchedCSSRules.length - 1; i >= 0; --i) {
+ var rulePayload = parentStyles.matchedCSSRules[i];
+ if (!this._containsInherited(rulePayload.style))
+ continue;
+ var rule = rulePayload;
+
+ if (!separatorInserted) {
+ insertInheritedNodeSeparator(parentNode);
+ separatorInserted = true;
+ }
+ styleRules.push({ style: rule.style, selectorText: rule.selectorText, media: rule.media, sourceURL: rule.resourceURL(), rule: rule, isInherited: true, parentNode: parentNode, editable: !!(rule.style && rule.style.styleSheetId) });
+ }
+ parentNode = parentNode.parentNode;
+ }
+ return styleRules;
+ },
+
+ _markUsedProperties: function(styleRules, usedProperties)
+ {
+ var foundImportantProperties = {};
+ var propertyToEffectiveRule = {};
+ var inheritedPropertyToNode = {};
+ for (var i = 0; i < styleRules.length; ++i) {
+ var styleRule = styleRules[i];
+ if (styleRule.computedStyle || styleRule.isStyleSeparator)
+ continue;
+ if (styleRule.section && styleRule.section.noAffect)
+ continue;
+
+ styleRule.usedProperties = {};
+
+ var style = styleRule.style;
+ var allProperties = style.allProperties;
+ for (var j = 0; j < allProperties.length; ++j) {
+ var property = allProperties[j];
+ if (!property.isLive || !property.parsedOk)
+ continue;
+
+ // Do not pick non-inherited properties from inherited styles.
+ if (styleRule.isInherited && !WebInspector.CSSMetadata.isPropertyInherited(property.name))
+ continue;
+
+ var canonicalName = WebInspector.CSSMetadata.canonicalPropertyName(property.name);
+ if (foundImportantProperties.hasOwnProperty(canonicalName))
+ continue;
+
+ if (!property.important && usedProperties.hasOwnProperty(canonicalName))
+ continue;
+
+ var isKnownProperty = propertyToEffectiveRule.hasOwnProperty(canonicalName);
+ if (!isKnownProperty && styleRule.isInherited && !inheritedPropertyToNode[canonicalName])
+ inheritedPropertyToNode[canonicalName] = styleRule.parentNode;
+
+ if (property.important) {
+ if (styleRule.isInherited && isKnownProperty && styleRule.parentNode !== inheritedPropertyToNode[canonicalName])
+ continue;
+
+ foundImportantProperties[canonicalName] = true;
+ if (isKnownProperty)
+ delete propertyToEffectiveRule[canonicalName].usedProperties[canonicalName];
+ }
+
+ styleRule.usedProperties[canonicalName] = true;
+ usedProperties[canonicalName] = true;
+ propertyToEffectiveRule[canonicalName] = styleRule;
+ }
+ }
+ },
+
+ _refreshSectionsForStyleRules: function(styleRules, usedProperties, editedSection)
+ {
+ // Walk the style rules and update the sections with new overloaded and used properties.
+ for (var i = 0; i < styleRules.length; ++i) {
+ var styleRule = styleRules[i];
+ var section = styleRule.section;
+ if (styleRule.computedStyle) {
+ section._usedProperties = usedProperties;
+ section.update();
+ } else {
+ section._usedProperties = styleRule.usedProperties;
+ section.update(section === editedSection);
+ }
+ }
+ },
+
+ /**
+ * @param {!Array.<!Object>} styleRules
+ * @param {!Object.<string, boolean>} usedProperties
+ * @param {?Element} anchorElement
+ */
+ _rebuildSectionsForStyleRules: function(styleRules, usedProperties, anchorElement)
+ {
+ // Make a property section for each style rule.
+ var sections = [];
+ for (var i = 0; i < styleRules.length; ++i) {
+ var styleRule = styleRules[i];
+ if (styleRule.isStyleSeparator) {
+ var separatorElement = document.createElement("div");
+ if (styleRule.isPlaceholder) {
+ separatorElement.className = "styles-sidebar-placeholder";
+ this._sectionsContainer.insertBefore(separatorElement, anchorElement);
+ continue;
+ }
+ separatorElement.className = "sidebar-separator";
+ if (styleRule.node) {
+ var link = WebInspector.DOMPresentationUtils.linkifyNodeReference(styleRule.node);
+ separatorElement.appendChild(document.createTextNode(WebInspector.UIString("Inherited from") + " "));
+ separatorElement.appendChild(link);
+ if (!sections.inheritedPropertiesSeparatorElement)
+ sections.inheritedPropertiesSeparatorElement = separatorElement;
+ } else if ("pseudoId" in styleRule) {
+ var pseudoName = WebInspector.StylesSidebarPane.PseudoIdNames[styleRule.pseudoId];
+ if (pseudoName)
+ separatorElement.textContent = WebInspector.UIString("Pseudo ::%s element", pseudoName);
+ else
+ separatorElement.textContent = WebInspector.UIString("Pseudo element");
+ } else
+ separatorElement.textContent = styleRule.text;
+ this._sectionsContainer.insertBefore(separatorElement, anchorElement);
+ continue;
+ }
+ var computedStyle = styleRule.computedStyle;
+
+ // Default editable to true if it was omitted.
+ var editable = styleRule.editable;
+ if (typeof editable === "undefined")
+ editable = true;
+
+ if (computedStyle)
+ var section = new WebInspector.ComputedStylePropertiesSection(this, styleRule, usedProperties);
+ else {
+ var section = new WebInspector.StylePropertiesSection(this, styleRule, editable, styleRule.isInherited);
+ section._markSelectorMatches();
+ }
+ section.expanded = true;
+
+ if (computedStyle)
+ this._computedStylePane.bodyElement.appendChild(section.element);
+ else
+ this._sectionsContainer.insertBefore(section.element, anchorElement);
+ sections.push(section);
+ }
+ return sections;
+ },
+
+ _containsInherited: function(style)
+ {
+ var properties = style.allProperties;
+ for (var i = 0; i < properties.length; ++i) {
+ var property = properties[i];
+ // Does this style contain non-overridden inherited property?
+ if (property.isLive && WebInspector.CSSMetadata.isPropertyInherited(property.name))
+ return true;
+ }
+ return false;
+ },
+
+ _colorFormatSettingChanged: function(event)
+ {
+ for (var pseudoId in this.sections) {
+ var sections = this.sections[pseudoId];
+ for (var i = 0; i < sections.length; ++i)
+ sections[i].update(true);
+ }
+ },
+
+ _createNewRule: function(event)
+ {
+ event.consume();
+ this.expand();
+ this.addBlankSection().startEditingSelector();
+ },
+
+ /**
+ * @return {!WebInspector.BlankStylePropertiesSection}
+ */
+ addBlankSection: function()
+ {
+ var blankSection = new WebInspector.BlankStylePropertiesSection(this, this._node ? WebInspector.DOMPresentationUtils.simpleSelector(this._node) : "");
+
+ var elementStyleSection = this.sections[0][1];
+ this._sectionsContainer.insertBefore(blankSection.element, elementStyleSection.element.nextSibling);
+
+ this.sections[0].splice(2, 0, blankSection);
+
+ return blankSection;
+ },
+
+ removeSection: function(section)
+ {
+ for (var pseudoId in this.sections) {
+ var sections = this.sections[pseudoId];
+ var index = sections.indexOf(section);
+ if (index === -1)
+ continue;
+ sections.splice(index, 1);
+ section.element.remove();
+ }
+ },
+
+ _toggleElementStatePane: function(event)
+ {
+ event.consume();
+
+ var buttonToggled = !this._elementStateButton.classList.contains("toggled");
+ if (buttonToggled)
+ this.expand();
+ this._elementStateButton.classList.toggle("toggled", buttonToggled);
+ this._elementStatePane.classList.toggle("expanded", buttonToggled);
+ },
+
+ _createElementStatePane: function()
+ {
+ this._elementStatePane = document.createElement("div");
+ this._elementStatePane.className = "styles-element-state-pane source-code";
+ var table = document.createElement("table");
+
+ var inputs = [];
+ this._elementStatePane.inputs = inputs;
+
+ /**
+ * @param {?Event} event
+ * @this {WebInspector.StylesSidebarPane}
+ */
+ function clickListener(event)
+ {
+ var node = this._validateNode();
+ if (!node)
+ return;
+ this._setPseudoClassCallback(node, event.target.state, event.target.checked);
+ }
+
+ /**
+ * @param {string} state
+ * @return {!Element}
+ * @this {WebInspector.StylesSidebarPane}
+ */
+ function createCheckbox(state)
+ {
+ var td = document.createElement("td");
+ var label = document.createElement("label");
+ var input = document.createElement("input");
+ input.type = "checkbox";
+ input.state = state;
+ input.addEventListener("click", clickListener.bind(this), false);
+ inputs.push(input);
+ label.appendChild(input);
+ label.appendChild(document.createTextNode(":" + state));
+ td.appendChild(label);
+ return td;
+ }
+
+ var tr = table.createChild("tr");
+ tr.appendChild(createCheckbox.call(this, "active"));
+ tr.appendChild(createCheckbox.call(this, "hover"));
+
+ tr = table.createChild("tr");
+ tr.appendChild(createCheckbox.call(this, "focus"));
+ tr.appendChild(createCheckbox.call(this, "visited"));
+
+ this._elementStatePane.appendChild(table);
+ },
+
+ /**
+ * @return {?RegExp}
+ */
+ filterRegex: function()
+ {
+ return this._filterRegex;
+ },
+
+ /**
+ * @param {boolean} isComputedStyleFilter
+ * @return {!Element}
+ * @param {function(?RegExp)} filterCallback
+ */
+ _createPropertyFilterElement: function(isComputedStyleFilter, filterCallback)
+ {
+ var input = document.createElement("input");
+ input.type = "text";
+ input.placeholder = isComputedStyleFilter ? WebInspector.UIString("Filter") : WebInspector.UIString("Find in Styles");
+ var boundSearchHandler = searchHandler.bind(this);
+
+ /**
+ * @this {WebInspector.StylesSidebarPane}
+ */
+ function searchHandler()
+ {
+ var regex = input.value ? new RegExp(input.value.escapeForRegExp(), "i") : null;
+ filterCallback(regex);
+ input.parentNode.classList.toggle("styles-filter-engaged", !!input.value);
+ this._updateFilter(isComputedStyleFilter);
+ }
+ input.addEventListener("input", boundSearchHandler, false);
+
+ /**
+ * @param {?Event} event
+ */
+ function keydownHandler(event)
+ {
+ var Esc = "U+001B";
+ if (event.keyIdentifier !== Esc || !input.value)
+ return;
+ event.consume(true);
+ input.value = "";
+ boundSearchHandler();
+ }
+ input.addEventListener("keydown", keydownHandler, false);
+
+ return input;
+ },
+
+ /**
+ * @param {boolean} isComputedStyleFilter
+ */
+ _updateFilter: function(isComputedStyleFilter)
+ {
+ for (var pseudoId in this.sections) {
+ var sections = this.sections[pseudoId];
+ for (var i = 0; i < sections.length; ++i) {
+ var section = sections[i];
+ if (isComputedStyleFilter !== !!section.computedStyle)
+ continue;
+ section._updateFilter();
+ }
+ }
+ },
+
+ /**
+ * @param {!WebInspector.Event} event
+ */
+ _showUserAgentStylesSettingChanged: function(event)
+ {
+ var showStyles = /** @type {boolean} */ (event.data);
+ this.element.classList.toggle("show-user-styles", showStyles);
+ },
+
+ willHide: function()
+ {
+ this._spectrumHelper.hide();
+ this._discardElementUnderMouse();
+ },
+
+ _discardElementUnderMouse: function()
+ {
+ if (this._elementUnderMouse)
+ this._elementUnderMouse.classList.remove("styles-panel-hovered");
+ delete this._elementUnderMouse;
+ },
+
+ _mouseMovedOverElement: function(e)
+ {
+ if (this._elementUnderMouse && e.target !== this._elementUnderMouse)
+ this._discardElementUnderMouse();
+ this._elementUnderMouse = e.target;
+ if (WebInspector.KeyboardShortcut.eventHasCtrlOrMeta(e))
+ this._elementUnderMouse.classList.add("styles-panel-hovered");
+ },
+
+ _keyDown: function(e)
+ {
+ if ((!WebInspector.isMac() && e.keyCode === WebInspector.KeyboardShortcut.Keys.Ctrl.code) ||
+ (WebInspector.isMac() && e.keyCode === WebInspector.KeyboardShortcut.Keys.Meta.code)) {
+ if (this._elementUnderMouse)
+ this._elementUnderMouse.classList.add("styles-panel-hovered");
+ }
+ },
+
+ _keyUp: function(e)
+ {
+ if ((!WebInspector.isMac() && e.keyCode === WebInspector.KeyboardShortcut.Keys.Ctrl.code) ||
+ (WebInspector.isMac() && e.keyCode === WebInspector.KeyboardShortcut.Keys.Meta.code)) {
+ this._discardElementUnderMouse();
+ }
+ },
+
+ __proto__: WebInspector.SidebarPane.prototype
+}
+
+/**
+ * @constructor
+ * @extends {WebInspector.SidebarPane}
+ */
+WebInspector.ComputedStyleSidebarPane = function()
+{
+ WebInspector.SidebarPane.call(this, WebInspector.UIString("Computed Style"));
+}
+
+WebInspector.ComputedStyleSidebarPane.prototype = {
+ /**
+ * @param {!WebInspector.StylesSidebarPane} pane
+ */
+ setHostingPane: function(pane)
+ {
+ this._stylesSidebarPane = pane;
+ },
+
+ setFilterBoxContainer: function(element)
+ {
+ element.appendChild(this._stylesSidebarPane._createPropertyFilterElement(true, filterCallback.bind(this)));
+
+ /**
+ * @param {?RegExp} regex
+ * @this {WebInspector.ComputedStyleSidebarPane}
+ */
+ function filterCallback(regex)
+ {
+ this._filterRegex = regex;
+ }
+ },
+
+ wasShown: function()
+ {
+ WebInspector.SidebarPane.prototype.wasShown.call(this);
+ if (!this._hasFreshContent)
+ this.prepareContent();
+ },
+
+ /**
+ * @param {function()=} callback
+ */
+ prepareContent: function(callback)
+ {
+ /**
+ * @this {WebInspector.ComputedStyleSidebarPane}
+ */
+ function wrappedCallback() {
+ this._hasFreshContent = true;
+ if (callback)
+ callback();
+ delete this._hasFreshContent;
+ }
+ this._stylesSidebarPane._refreshUpdate(null, true, wrappedCallback.bind(this));
+ },
+
+ /**
+ * @return {?RegExp}
+ */
+ filterRegex: function()
+ {
+ return this._filterRegex;
+ },
+
+ __proto__: WebInspector.SidebarPane.prototype
+}
+
+/**
+ * @constructor
+ * @extends {WebInspector.PropertiesSection}
+ * @param {!WebInspector.StylesSidebarPane} parentPane
+ * @param {!Object} styleRule
+ * @param {boolean} editable
+ * @param {boolean} isInherited
+ */
+WebInspector.StylePropertiesSection = function(parentPane, styleRule, editable, isInherited)
+{
+ WebInspector.PropertiesSection.call(this, "");
+
+ this._parentPane = parentPane;
+ this.styleRule = styleRule;
+ this.rule = this.styleRule.rule;
+ this.editable = editable;
+ this.isInherited = isInherited;
+
+ var extraClasses = (this.rule && (this.rule.isUser || this.rule.isUserAgent) ? " user-rule" : "");
+ this.element.className = "styles-section matched-styles monospace" + extraClasses;
+ // We don't really use properties' disclosure.
+ this.propertiesElement.classList.remove("properties-tree");
+
+ var selectorContainer = document.createElement("div");
+ this._selectorElement = document.createElement("span");
+ this._selectorElement.textContent = styleRule.selectorText;
+ selectorContainer.appendChild(this._selectorElement);
+
+ var openBrace = document.createElement("span");
+ openBrace.textContent = " {";
+ selectorContainer.appendChild(openBrace);
+ selectorContainer.addEventListener("mousedown", this._handleEmptySpaceMouseDown.bind(this), false);
+ selectorContainer.addEventListener("click", this._handleSelectorContainerClick.bind(this), false);
+
+ var closeBrace = document.createElement("div");
+ closeBrace.textContent = "}";
+ this.element.appendChild(closeBrace);
+
+ this._selectorElement.addEventListener("click", this._handleSelectorClick.bind(this), false);
+ this.element.addEventListener("mousedown", this._handleEmptySpaceMouseDown.bind(this), false);
+ this.element.addEventListener("click", this._handleEmptySpaceClick.bind(this), false);
+
+ if (this.rule) {
+ // Prevent editing the user agent and user rules.
+ if (this.rule.isUserAgent || this.rule.isUser)
+ this.editable = false;
+ else {
+ // Check this is a real CSSRule, not a bogus object coming from WebInspector.BlankStylePropertiesSection.
+ if (this.rule.styleSheetId)
+ this.navigable = !!this.rule.resourceURL();
+ }
+ this.titleElement.classList.add("styles-selector");
+ }
+
+ this._usedProperties = styleRule.usedProperties;
+
+ this._selectorRefElement = document.createElement("div");
+ this._selectorRefElement.className = "subtitle";
+ this._mediaListElement = this.titleElement.createChild("div", "media-list");
+ this._updateMediaList();
+ this._updateRuleOrigin();
+ selectorContainer.insertBefore(this._selectorRefElement, selectorContainer.firstChild);
+ this.titleElement.appendChild(selectorContainer);
+ this._selectorContainer = selectorContainer;
+
+ if (isInherited)
+ this.element.classList.add("styles-show-inherited"); // This one is related to inherited rules, not computed style.
+
+ if (this.navigable)
+ this.element.classList.add("navigable");
+
+ if (!this.editable)
+ this.element.classList.add("read-only");
+}
+
+WebInspector.StylePropertiesSection.prototype = {
+ /**
+ * @param {!WebInspector.CSSRule} editedRule
+ * @param {!WebInspector.TextRange} oldRange
+ * @param {!WebInspector.TextRange} newRange
+ */
+ _styleSheetRuleEdited: function(editedRule, oldRange, newRange)
+ {
+ if (!this.rule || !this.rule.styleSheetId)
+ return;
+ if (this.rule !== editedRule)
+ this.rule.sourceStyleSheetEdited(editedRule.styleSheetId, oldRange, newRange);
+ this._updateMediaList();
+ this._updateRuleOrigin();
+ },
+
+ /**
+ * @param {!Object} styleRule
+ */
+ _createMediaList: function(styleRule)
+ {
+ if (!styleRule.media)
+ return;
+ for (var i = styleRule.media.length - 1; i >= 0; --i) {
+ var media = styleRule.media[i];
+ var mediaDataElement = this._mediaListElement.createChild("div", "media");
+ var mediaText;
+ switch (media.source) {
+ case WebInspector.CSSMedia.Source.LINKED_SHEET:
+ case WebInspector.CSSMedia.Source.INLINE_SHEET:
+ mediaText = "media=\"" + media.text + "\"";
+ break;
+ case WebInspector.CSSMedia.Source.MEDIA_RULE:
+ mediaText = "@media " + media.text;
+ break;
+ case WebInspector.CSSMedia.Source.IMPORT_RULE:
+ mediaText = "@import " + media.text;
+ break;
+ }
+
+ if (media.sourceURL) {
+ var refElement = mediaDataElement.createChild("div", "subtitle");
+ var rawLocation;
+ var mediaHeader;
+ if (media.range) {
+ mediaHeader = media.header();
+ if (mediaHeader) {
+ var lineNumber = media.lineNumberInSource();
+ var columnNumber = media.columnNumberInSource();
+ console.assert(typeof lineNumber !== "undefined" && typeof columnNumber !== "undefined");
+ rawLocation = new WebInspector.CSSLocation(this._parentPane._target, media.sourceURL, lineNumber, columnNumber);
+ }
+ }
+
+ var anchor;
+ if (rawLocation)
+ anchor = this._parentPane._linkifier.linkifyCSSLocation(mediaHeader.id, rawLocation);
+ else {
+ // The "linkedStylesheet" case.
+ anchor = WebInspector.linkifyResourceAsNode(media.sourceURL, undefined, "subtitle", media.sourceURL);
+ }
+ anchor.style.float = "right";
+ refElement.appendChild(anchor);
+ }
+
+ var mediaTextElement = mediaDataElement.createChild("span");
+ mediaTextElement.textContent = mediaText;
+ mediaTextElement.title = media.text;
+ }
+ },
+
+ _updateMediaList: function()
+ {
+ this._mediaListElement.removeChildren();
+ this._createMediaList(this.styleRule);
+ },
+
+ collapse: function()
+ {
+ // Overriding with empty body.
+ },
+
+ handleClick: function()
+ {
+ // Avoid consuming events.
+ },
+
+ /**
+ * @param {string} propertyName
+ * @return {boolean}
+ */
+ isPropertyInherited: function(propertyName)
+ {
+ if (this.isInherited) {
+ // While rendering inherited stylesheet, reverse meaning of this property.
+ // Render truly inherited properties with black, i.e. return them as non-inherited.
+ return !WebInspector.CSSMetadata.isPropertyInherited(propertyName);
+ }
+ return false;
+ },
+
+ /**
+ * @param {string} propertyName
+ * @param {boolean=} isShorthand
+ * @return {boolean}
+ */
+ isPropertyOverloaded: function(propertyName, isShorthand)
+ {
+ if (!this._usedProperties || this.noAffect)
+ return false;
+
+ if (this.isInherited && !WebInspector.CSSMetadata.isPropertyInherited(propertyName)) {
+ // In the inherited sections, only show overrides for the potentially inherited properties.
+ return false;
+ }
+
+ var canonicalName = WebInspector.CSSMetadata.canonicalPropertyName(propertyName);
+ var used = (canonicalName in this._usedProperties);
+ if (used || !isShorthand)
+ return !used;
+
+ // Find out if any of the individual longhand properties of the shorthand
+ // are used, if none are then the shorthand is overloaded too.
+ var longhandProperties = this.styleRule.style.longhandProperties(propertyName);
+ for (var j = 0; j < longhandProperties.length; ++j) {
+ var individualProperty = longhandProperties[j];
+ if (WebInspector.CSSMetadata.canonicalPropertyName(individualProperty.name) in this._usedProperties)
+ return false;
+ }
+
+ return true;
+ },
+
+ /**
+ * @return {?WebInspector.StylePropertiesSection}
+ */
+ nextEditableSibling: function()
+ {
+ var curSection = this;
+ do {
+ curSection = curSection.nextSibling;
+ } while (curSection && !curSection.editable);
+
+ if (!curSection) {
+ curSection = this.firstSibling;
+ while (curSection && !curSection.editable)
+ curSection = curSection.nextSibling;
+ }
+
+ return (curSection && curSection.editable) ? curSection : null;
+ },
+
+ /**
+ * @return {?WebInspector.StylePropertiesSection}
+ */
+ previousEditableSibling: function()
+ {
+ var curSection = this;
+ do {
+ curSection = curSection.previousSibling;
+ } while (curSection && !curSection.editable);
+
+ if (!curSection) {
+ curSection = this.lastSibling;
+ while (curSection && !curSection.editable)
+ curSection = curSection.previousSibling;
+ }
+
+ return (curSection && curSection.editable) ? curSection : null;
+ },
+
+ update: function(full)
+ {
+ if (this.styleRule.selectorText)
+ this._selectorElement.textContent = this.styleRule.selectorText;
+ this._markSelectorMatches();
+ if (full) {
+ this.propertiesTreeOutline.removeChildren();
+ this.populated = false;
+ } else {
+ var child = this.propertiesTreeOutline.children[0];
+ while (child) {
+ child.overloaded = this.isPropertyOverloaded(child.name, child.isShorthand);
+ child = child.traverseNextTreeElement(false, null, true);
+ }
+ }
+ this.afterUpdate();
+ },
+
+ afterUpdate: function()
+ {
+ if (this._afterUpdate) {
+ this._afterUpdate(this);
+ delete this._afterUpdate;
+ }
+ },
+
+ onpopulate: function()
+ {
+ var style = this.styleRule.style;
+ var allProperties = style.allProperties;
+ this.uniqueProperties = [];
+
+ var styleHasEditableSource = this.editable && !!style.range;
+ if (styleHasEditableSource) {
+ for (var i = 0; i < allProperties.length; ++i) {
+ var property = allProperties[i];
+ this.uniqueProperties.push(property);
+ if (property.styleBased)
+ continue;
+
+ var isShorthand = !!WebInspector.CSSMetadata.cssPropertiesMetainfo.longhands(property.name);
+ var inherited = this.isPropertyInherited(property.name);
+ var overloaded = property.inactive || this.isPropertyOverloaded(property.name);
+ var item = new WebInspector.StylePropertyTreeElement(this._parentPane, this.styleRule, style, property, isShorthand, inherited, overloaded);
+ this.propertiesTreeOutline.appendChild(item);
+ }
+ return;
+ }
+
+ var generatedShorthands = {};
+ // For style-based properties, generate shorthands with values when possible.
+ for (var i = 0; i < allProperties.length; ++i) {
+ var property = allProperties[i];
+ this.uniqueProperties.push(property);
+ var isShorthand = !!WebInspector.CSSMetadata.cssPropertiesMetainfo.longhands(property.name);
+
+ // For style-based properties, try generating shorthands.
+ var shorthands = isShorthand ? null : WebInspector.CSSMetadata.cssPropertiesMetainfo.shorthands(property.name);
+ var shorthandPropertyAvailable = false;
+ for (var j = 0; shorthands && !shorthandPropertyAvailable && j < shorthands.length; ++j) {
+ var shorthand = shorthands[j];
+ if (shorthand in generatedShorthands) {
+ shorthandPropertyAvailable = true;
+ continue; // There already is a shorthand this longhands falls under.
+ }
+ if (style.getLiveProperty(shorthand)) {
+ shorthandPropertyAvailable = true;
+ continue; // There is an explict shorthand property this longhands falls under.
+ }
+ if (!style.shorthandValue(shorthand)) {
+ shorthandPropertyAvailable = false;
+ continue; // Never generate synthetic shorthands when no value is available.
+ }
+
+ // Generate synthetic shorthand we have a value for.
+ var shorthandProperty = new WebInspector.CSSProperty(style, style.allProperties.length, shorthand, style.shorthandValue(shorthand), false, false, true, true);
+ var overloaded = property.inactive || this.isPropertyOverloaded(property.name, true);
+ var item = new WebInspector.StylePropertyTreeElement(this._parentPane, this.styleRule, style, shorthandProperty, /* isShorthand */ true, /* inherited */ false, overloaded);
+ this.propertiesTreeOutline.appendChild(item);
+ generatedShorthands[shorthand] = shorthandProperty;
+ shorthandPropertyAvailable = true;
+ }
+ if (shorthandPropertyAvailable)
+ continue; // Shorthand for the property found.
+
+ var inherited = this.isPropertyInherited(property.name);
+ var overloaded = property.inactive || this.isPropertyOverloaded(property.name, isShorthand);
+ var item = new WebInspector.StylePropertyTreeElement(this._parentPane, this.styleRule, style, property, isShorthand, inherited, overloaded);
+ this.propertiesTreeOutline.appendChild(item);
+ }
+ },
+
+ _updateFilter: function()
+ {
+ if (this.styleRule.isAttribute)
+ return;
+ var regex = this._parentPane.filterRegex();
+ var hideRule = regex && !regex.test(this.element.textContent);
+ this.element.classList.toggle("hidden", hideRule);
+ if (hideRule)
+ return;
+
+ var children = this.propertiesTreeOutline.children;
+ for (var i = 0; i < children.length; ++i)
+ children[i]._updateFilter();
+
+ if (this.styleRule.rule)
+ this._markSelectorHighlights();
+ },
+
+ _markSelectorMatches: function()
+ {
+ var rule = this.styleRule.rule;
+ if (!rule)
+ return;
+
+ var matchingSelectors = rule.matchingSelectors;
+ // .selector is rendered as non-affecting selector by default.
+ if (this.noAffect || matchingSelectors)
+ this._selectorElement.className = "selector";
+ if (!matchingSelectors)
+ return;
+
+ var selectors = rule.selectors;
+ var fragment = document.createDocumentFragment();
+ var currentMatch = 0;
+ for (var i = 0; i < selectors.length ; ++i) {
+ if (i)
+ fragment.appendChild(document.createTextNode(", "));
+ var isSelectorMatching = matchingSelectors[currentMatch] === i;
+ if (isSelectorMatching)
+ ++currentMatch;
+ var rawLocation = new WebInspector.CSSLocation(this._parentPane._target, rule.sourceURL, rule.lineNumberInSource(i), rule.columnNumberInSource(i));
+ var matchingSelectorClass = isSelectorMatching ? " selector-matches" : "";
+ var selectorElement = document.createElement("span");
+ selectorElement.className = "simple-selector" + matchingSelectorClass;
+ if (rule.styleSheetId)
+ selectorElement._selectorIndex = i;
+ selectorElement.textContent = selectors[i].value;
+
+ fragment.appendChild(selectorElement);
+ }
+
+ this._selectorElement.removeChildren();
+ this._selectorElement.appendChild(fragment);
+ this._markSelectorHighlights();
+ },
+
+ _markSelectorHighlights: function()
+ {
+ var selectors = this._selectorElement.getElementsByClassName("simple-selector");
+ var regex = this._parentPane.filterRegex();
+ for (var i = 0; i < selectors.length; ++i) {
+ var selectorMatchesFilter = regex && regex.test(selectors[i].textContent);
+ selectors[i].classList.toggle("filter-match", selectorMatchesFilter);
+ }
+ },
+
+ _checkWillCancelEditing: function()
+ {
+ var willCauseCancelEditing = this._willCauseCancelEditing;
+ delete this._willCauseCancelEditing;
+ return willCauseCancelEditing;
+ },
+
+ _handleSelectorContainerClick: function(event)
+ {
+ if (this._checkWillCancelEditing() || !this.editable)
+ return;
+ if (event.target === this._selectorContainer)
+ this.addNewBlankProperty(0).startEditing();
+ },
+
+ /**
+ * @param {number=} index
+ * @return {!WebInspector.StylePropertyTreeElement}
+ */
+ addNewBlankProperty: function(index)
+ {
+ var style = this.styleRule.style;
+ var property = style.newBlankProperty(index);
+ var item = new WebInspector.StylePropertyTreeElement(this._parentPane, this.styleRule, style, property, false, false, false);
+ index = property.index;
+ this.propertiesTreeOutline.insertChild(item, index);
+ item.listItemElement.textContent = "";
+ item._newProperty = true;
+ item.updateTitle();
+ return item;
+ },
+
+ _createRuleOriginNode: function()
+ {
+ /**
+ * @param {string} url
+ * @param {number} line
+ */
+ function linkifyUncopyable(url, line)
+ {
+ var link = WebInspector.linkifyResourceAsNode(url, line, "", url + ":" + (line + 1));
+ link.classList.add("webkit-html-resource-link");
+ link.setAttribute("data-uncopyable", link.textContent);
+ link.textContent = "";
+ return link;
+ }
+
+ if (this.styleRule.sourceURL) {
+ var firstMatchingIndex = this.styleRule.rule.matchingSelectors && this.rule.matchingSelectors.length ? this.rule.matchingSelectors[0] : 0;
+ var matchingSelectorLocation = new WebInspector.CSSLocation(this._parentPane._target, this.styleRule.sourceURL, this.rule.lineNumberInSource(firstMatchingIndex), this.rule.columnNumberInSource(firstMatchingIndex));
+ return this._parentPane._linkifier.linkifyCSSLocation(this.rule.styleSheetId, matchingSelectorLocation) || linkifyUncopyable(this.styleRule.sourceURL, this.rule.lineNumberInSource());
+ }
+
+ if (!this.rule)
+ return document.createTextNode("");
+
+ if (this.rule.isUserAgent)
+ return document.createTextNode(WebInspector.UIString("user agent stylesheet"));
+ if (this.rule.isUser)
+ return document.createTextNode(WebInspector.UIString("user stylesheet"));
+ if (this.rule.isViaInspector)
+ return document.createTextNode(WebInspector.UIString("via inspector"));
+ return document.createTextNode("");
+ },
+
+ _handleEmptySpaceMouseDown: function()
+ {
+ this._willCauseCancelEditing = this._parentPane._isEditingStyle;
+ },
+
+ _handleEmptySpaceClick: function(event)
+ {
+ if (!this.editable)
+ return;
+
+ if (!window.getSelection().isCollapsed)
+ return;
+
+ if (this._checkWillCancelEditing())
+ return;
+
+ if (event.target.classList.contains("header") || this.element.classList.contains("read-only") || event.target.enclosingNodeOrSelfWithClass("media")) {
+ event.consume();
+ return;
+ }
+ this.expand();
+ this.addNewBlankProperty().startEditing();
+ },
+
+ _handleSelectorClick: function(event)
+ {
+ if (WebInspector.KeyboardShortcut.eventHasCtrlOrMeta(event) && this.navigable && event.target.classList.contains("simple-selector")) {
+ var index = event.target._selectorIndex;
+ var styleSheetHeader = this._parentPane._target.cssModel.styleSheetHeaderForId(this.rule.styleSheetId);
+ var uiLocation = styleSheetHeader.rawLocationToUILocation(this.rule.lineNumberInSource(index), this.rule.columnNumberInSource(index));
+ WebInspector.Revealer.reveal(uiLocation);
+ return;
+ }
+ this._startEditingOnMouseEvent();
+ event.consume(true);
+ },
+
+ _startEditingOnMouseEvent: function()
+ {
+ if (!this.editable)
+ return;
+
+ if (!this.rule && this.propertiesTreeOutline.children.length === 0) {
+ this.expand();
+ this.addNewBlankProperty().startEditing();
+ return;
+ }
+
+ if (!this.rule)
+ return;
+
+ this.startEditingSelector();
+ },
+
+ startEditingSelector: function()
+ {
+ var element = this._selectorElement;
+ if (WebInspector.isBeingEdited(element))
+ return;
+
+ element.scrollIntoViewIfNeeded(false);
+ element.textContent = element.textContent; // Reset selector marks in group.
+
+ var config = new WebInspector.InplaceEditor.Config(this.editingSelectorCommitted.bind(this), this.editingSelectorCancelled.bind(this));
+ WebInspector.InplaceEditor.startEditing(this._selectorElement, config);
+
+ window.getSelection().setBaseAndExtent(element, 0, element, 1);
+ this._parentPane._isEditingStyle = true;
+ },
+
+ _moveEditorFromSelector: function(moveDirection)
+ {
+ this._markSelectorMatches();
+
+ if (!moveDirection)
+ return;
+
+ if (moveDirection === "forward") {
+ this.expand();
+ var firstChild = this.propertiesTreeOutline.children[0];
+ while (firstChild && firstChild.inherited)
+ firstChild = firstChild.nextSibling;
+ if (!firstChild)
+ this.addNewBlankProperty().startEditing();
+ else
+ firstChild.startEditing(firstChild.nameElement);
+ } else {
+ var previousSection = this.previousEditableSibling();
+ if (!previousSection)
+ return;
+
+ previousSection.expand();
+ previousSection.addNewBlankProperty().startEditing();
+ }
+ },
+
+ editingSelectorCommitted: function(element, newContent, oldContent, context, moveDirection)
+ {
+ this._editingSelectorEnded();
+ if (newContent)
+ newContent = newContent.trim();
+ if (newContent === oldContent) {
+ // Revert to a trimmed version of the selector if need be.
+ this._selectorElement.textContent = newContent;
+ this._moveEditorFromSelector(moveDirection);
+ return;
+ }
+
+ var selectedNode = this._parentPane._node;
+
+ /**
+ * @param {!WebInspector.CSSRule} newRule
+ * @this {WebInspector.StylePropertiesSection}
+ */
+ function successCallback(newRule)
+ {
+ var doesAffectSelectedNode = newRule.matchingSelectors.length > 0;
+ if (!doesAffectSelectedNode) {
+ this.noAffect = true;
+ this.element.classList.add("no-affect");
+ } else {
+ delete this.noAffect;
+ this.element.classList.remove("no-affect");
+ }
+
+ var oldSelectorRange = this.rule.selectorRange;
+ this.rule = newRule;
+ this.styleRule = { section: this, style: newRule.style, selectorText: newRule.selectorText, media: newRule.media, sourceURL: newRule.resourceURL(), rule: newRule };
+
+ this._parentPane.update(selectedNode);
+ this._parentPane._styleSheetRuleEdited(this.rule, oldSelectorRange, this.rule.selectorRange);
+
+ finishOperationAndMoveEditor.call(this, moveDirection);
+ }
+
+ /**
+ * @this {WebInspector.StylePropertiesSection}
+ */
+ function finishOperationAndMoveEditor(direction)
+ {
+ delete this._parentPane._userOperation;
+ this._moveEditorFromSelector(direction);
+ }
+
+ // This gets deleted in finishOperationAndMoveEditor(), which is called both on success and failure.
+ this._parentPane._userOperation = true;
+ this._parentPane._target.cssModel.setRuleSelector(this.rule, selectedNode ? selectedNode.id : 0, newContent, successCallback.bind(this), finishOperationAndMoveEditor.bind(this, moveDirection));
+ },
+
+ _updateRuleOrigin: function()
+ {
+ this._selectorRefElement.removeChildren();
+ this._selectorRefElement.appendChild(this._createRuleOriginNode());
+ },
+
+ _editingSelectorEnded: function()
+ {
+ delete this._parentPane._isEditingStyle;
+ },
+
+ editingSelectorCancelled: function()
+ {
+ this._editingSelectorEnded();
+
+ // Mark the selectors in group if necessary.
+ // This is overridden by BlankStylePropertiesSection.
+ this._markSelectorMatches();
+ },
+
+ __proto__: WebInspector.PropertiesSection.prototype
+}
+
+/**
+ * @constructor
+ * @extends {WebInspector.PropertiesSection}
+ * @param {!WebInspector.StylesSidebarPane} stylesPane
+ * @param {!Object} styleRule
+ * @param {!Object.<string, boolean>} usedProperties
+ */
+WebInspector.ComputedStylePropertiesSection = function(stylesPane, styleRule, usedProperties)
+{
+ WebInspector.PropertiesSection.call(this, "");
+
+ var subtitle = this.headerElement.createChild("div", "sidebar-pane-subtitle vbox");
+ var showInheritedCheckbox = new WebInspector.Checkbox(WebInspector.UIString("Show inherited properties"), "hbox");
+ subtitle.appendChild(showInheritedCheckbox.element);
+
+ this._hasFreshContent = false;
+
+ /**
+ * @this {WebInspector.ComputedStylePropertiesSection}
+ */
+ function showInheritedToggleFunction()
+ {
+ var showInherited = showInheritedCheckbox.checked;
+ WebInspector.settings.showInheritedComputedStyleProperties.set(showInherited);
+ if (showInherited)
+ this.element.classList.add("styles-show-inherited");
+ else
+ this.element.classList.remove("styles-show-inherited");
+ }
+
+ showInheritedCheckbox.addEventListener(showInheritedToggleFunction.bind(this));
+
+ this.element.className = "styles-section monospace read-only computed-style";
+ if (WebInspector.settings.showInheritedComputedStyleProperties.get()) {
+ this.element.classList.add("styles-show-inherited");
+ showInheritedCheckbox.checked = true;
+ }
+
+ this._stylesPane = stylesPane;
+ this.styleRule = styleRule;
+ this._usedProperties = usedProperties;
+ this._alwaysShowComputedProperties = { "display": true, "height": true, "width": true };
+ this.computedStyle = true;
+ this._propertyTreeElements = {};
+ this._expandedPropertyNames = {};
+}
+
+WebInspector.ComputedStylePropertiesSection.prototype = {
+ collapse: function(dontRememberState)
+ {
+ // Overriding with empty body.
+ },
+
+ _isPropertyInherited: function(propertyName)
+ {
+ var canonicalName = WebInspector.CSSMetadata.canonicalPropertyName(propertyName);
+ return !(canonicalName in this._usedProperties) && !(canonicalName in this._alwaysShowComputedProperties);
+ },
+
+ update: function()
+ {
+ this._expandedPropertyNames = {};
+ for (var name in this._propertyTreeElements) {
+ if (this._propertyTreeElements[name].expanded)
+ this._expandedPropertyNames[name] = true;
+ }
+ this._propertyTreeElements = {};
+ this.propertiesTreeOutline.removeChildren();
+ this.populated = false;
+ },
+
+ _updateFilter: function()
+ {
+ var children = this.propertiesTreeOutline.children;
+ for (var i = 0; i < children.length; ++i)
+ children[i]._updateFilter();
+ },
+
+ onpopulate: function()
+ {
+ function sorter(a, b)
+ {
+ return a.name.compareTo(b.name);
+ }
+
+ var style = this.styleRule.style;
+ if (!style)
+ return;
+
+ var uniqueProperties = [];
+ var allProperties = style.allProperties;
+ for (var i = 0; i < allProperties.length; ++i)
+ uniqueProperties.push(allProperties[i]);
+ uniqueProperties.sort(sorter);
+
+ this._propertyTreeElements = {};
+ for (var i = 0; i < uniqueProperties.length; ++i) {
+ var property = uniqueProperties[i];
+ var inherited = this._isPropertyInherited(property.name);
+ var item = new WebInspector.ComputedStylePropertyTreeElement(this._stylesPane, this.styleRule, style, property, inherited);
+ this.propertiesTreeOutline.appendChild(item);
+ this._propertyTreeElements[property.name] = item;
+ }
+ },
+
+ rebuildComputedTrace: function(sections)
+ {
+ for (var i = 0; i < sections.length; ++i) {
+ var section = sections[i];
+ if (section.computedStyle || section.isBlank)
+ continue;
+
+ for (var j = 0; j < section.uniqueProperties.length; ++j) {
+ var property = section.uniqueProperties[j];
+ if (property.disabled)
+ continue;
+ if (section.isInherited && !WebInspector.CSSMetadata.isPropertyInherited(property.name))
+ continue;
+
+ var treeElement = this._propertyTreeElements[property.name.toLowerCase()];
+ if (treeElement) {
+ var fragment = document.createDocumentFragment();
+ var selector = fragment.createChild("span");
+ selector.style.color = "gray";
+ selector.textContent = section.styleRule.selectorText;
+ fragment.appendChild(document.createTextNode(" - " + property.value + " "));
+ var subtitle = fragment.createChild("span");
+ subtitle.style.float = "right";
+ subtitle.appendChild(section._createRuleOriginNode());
+ var childElement = new TreeElement(fragment, null, false);
+ treeElement.appendChild(childElement);
+ if (property.inactive || section.isPropertyOverloaded(property.name))
+ childElement.listItemElement.classList.add("overloaded");
+ if (!property.parsedOk) {
+ childElement.listItemElement.classList.add("not-parsed-ok");
+ childElement.listItemElement.insertBefore(WebInspector.StylesSidebarPane.createExclamationMark(property), childElement.listItemElement.firstChild);
+ if (WebInspector.StylesSidebarPane._ignoreErrorsForProperty(property))
+ childElement.listItemElement.classList.add("has-ignorable-error");
+ }
+ }
+ }
+ }
+
+ // Restore expanded state after update.
+ for (var name in this._expandedPropertyNames) {
+ if (name in this._propertyTreeElements)
+ this._propertyTreeElements[name].expand();
+ }
+ },
+
+ __proto__: WebInspector.PropertiesSection.prototype
+}
+
+/**
+ * @constructor
+ * @extends {WebInspector.StylePropertiesSection}
+ * @param {!WebInspector.StylesSidebarPane} stylesPane
+ * @param {string} defaultSelectorText
+ */
+WebInspector.BlankStylePropertiesSection = function(stylesPane, defaultSelectorText)
+{
+ WebInspector.StylePropertiesSection.call(this, stylesPane, {selectorText: defaultSelectorText, rule: {isViaInspector: true}}, true, false);
+ this.element.classList.add("blank-section");
+}
+
+WebInspector.BlankStylePropertiesSection.prototype = {
+ get isBlank()
+ {
+ return !this._normal;
+ },
+
+ expand: function()
+ {
+ if (!this.isBlank)
+ WebInspector.StylePropertiesSection.prototype.expand.call(this);
+ },
+
+ editingSelectorCommitted: function(element, newContent, oldContent, context, moveDirection)
+ {
+ if (!this.isBlank) {
+ WebInspector.StylePropertiesSection.prototype.editingSelectorCommitted.call(this, element, newContent, oldContent, context, moveDirection);
+ return;
+ }
+
+ /**
+ * @param {!WebInspector.CSSRule} newRule
+ * @this {WebInspector.StylePropertiesSection}
+ */
+ function successCallback(newRule)
+ {
+ var doesSelectorAffectSelectedNode = newRule.matchingSelectors.length > 0;
+ var styleRule = { section: this, style: newRule.style, selectorText: newRule.selectorText, sourceURL: newRule.resourceURL(), rule: newRule };
+ this.makeNormal(styleRule);
+
+ if (!doesSelectorAffectSelectedNode) {
+ this.noAffect = true;
+ this.element.classList.add("no-affect");
+ }
+
+ this._updateRuleOrigin();
+ this.expand();
+ if (this.element.parentElement) // Might have been detached already.
+ this._moveEditorFromSelector(moveDirection);
+
+ delete this._parentPane._userOperation;
+ this._editingSelectorEnded();
+ this._markSelectorMatches();
+ }
+
+ if (newContent)
+ newContent = newContent.trim();
+ this._parentPane._userOperation = true;
+
+ var cssModel = this._parentPane._target.cssModel;
+ cssModel.requestViaInspectorStylesheet(this._parentPane._node, viaInspectorCallback.bind(this));
+
+ /**
+ * @this {WebInspector.BlankStylePropertiesSection}
+ * @param {?WebInspector.CSSStyleSheetHeader} styleSheetHeader
+ */
+ function viaInspectorCallback(styleSheetHeader)
+ {
+ if (!styleSheetHeader) {
+ this.editingSelectorCancelled();
+ return;
+ }
+ cssModel.addRule(styleSheetHeader.id, this._parentPane._node, newContent, successCallback.bind(this), this.editingSelectorCancelled.bind(this));
+ }
+ },
+
+ editingSelectorCancelled: function()
+ {
+ delete this._parentPane._userOperation;
+ if (!this.isBlank) {
+ WebInspector.StylePropertiesSection.prototype.editingSelectorCancelled.call(this);
+ return;
+ }
+
+ this._editingSelectorEnded();
+ this._parentPane.removeSection(this);
+ },
+
+ makeNormal: function(styleRule)
+ {
+ this.element.classList.remove("blank-section");
+ this.styleRule = styleRule;
+ this.rule = styleRule.rule;
+
+ // FIXME: replace this instance by a normal WebInspector.StylePropertiesSection.
+ this._normal = true;
+ },
+
+ __proto__: WebInspector.StylePropertiesSection.prototype
+}
+
+/**
+ * @constructor
+ * @extends {TreeElement}
+ * @param {!Object} styleRule
+ * @param {!WebInspector.CSSStyleDeclaration} style
+ * @param {!WebInspector.CSSProperty} property
+ * @param {boolean} inherited
+ * @param {boolean} overloaded
+ * @param {boolean} hasChildren
+ */
+WebInspector.StylePropertyTreeElementBase = function(styleRule, style, property, inherited, overloaded, hasChildren)
+{
+ this._styleRule = styleRule;
+ this.style = style;
+ this.property = property;
+ this._inherited = inherited;
+ this._overloaded = overloaded;
+
+ // Pass an empty title, the title gets made later in onattach.
+ TreeElement.call(this, "", null, hasChildren);
+
+ this.selectable = false;
+}
+
+WebInspector.StylePropertyTreeElementBase.prototype = {
+ /**
+ * @return {?WebInspector.DOMNode}
+ */
+ node: function()
+ {
+ return null; // Overridden by ancestors.
+ },
+
+ /**
+ * @return {?WebInspector.StylesSidebarPane}
+ */
+ editablePane: function()
+ {
+ return null; // Overridden by ancestors.
+ },
+
+ /**
+ * @return {!WebInspector.StylesSidebarPane|!WebInspector.ComputedStyleSidebarPane}
+ */
+ parentPane: function()
+ {
+ throw "Not implemented";
+ },
+
+ get inherited()
+ {
+ return this._inherited;
+ },
+
+ /**
+ * @return {boolean}
+ */
+ hasIgnorableError: function()
+ {
+ return !this.parsedOk && WebInspector.StylesSidebarPane._ignoreErrorsForProperty(this.property);
+ },
+
+ set inherited(x)
+ {
+ if (x === this._inherited)
+ return;
+ this._inherited = x;
+ this.updateState();
+ },
+
+ get overloaded()
+ {
+ return this._overloaded;
+ },
+
+ set overloaded(x)
+ {
+ if (x === this._overloaded)
+ return;
+ this._overloaded = x;
+ this.updateState();
+ },
+
+ get disabled()
+ {
+ return this.property.disabled;
+ },
+
+ get name()
+ {
+ if (!this.disabled || !this.property.text)
+ return this.property.name;
+
+ var text = this.property.text;
+ var index = text.indexOf(":");
+ if (index < 1)
+ return this.property.name;
+
+ text = text.substring(0, index).trim();
+ if (text.startsWith("/*"))
+ text = text.substring(2).trim();
+ return text;
+ },
+
+ get value()
+ {
+ if (!this.disabled || !this.property.text)
+ return this.property.value;
+
+ var match = this.property.text.match(/(.*);\s*/);
+ if (!match || !match[1])
+ return this.property.value;
+
+ var text = match[1];
+ var index = text.indexOf(":");
+ if (index < 1)
+ return this.property.value;
+
+ return text.substring(index + 1).trim();
+ },
+
+ get parsedOk()
+ {
+ return this.property.parsedOk;
+ },
+
+ onattach: function()
+ {
+ this.updateTitle();
+ },
+
+ updateTitle: function()
+ {
+ var value = this.value;
+
+ this.updateState();
+
+ var nameElement = document.createElement("span");
+ nameElement.className = "webkit-css-property";
+ nameElement.textContent = this.name;
+ nameElement.title = this.property.propertyText;
+ this.nameElement = nameElement;
+
+ this._expandElement = document.createElement("span");
+ this._expandElement.className = "expand-element";
+
+ var valueElement = document.createElement("span");
+ valueElement.className = "value";
+ this.valueElement = valueElement;
+
+ /**
+ * @param {!RegExp} regex
+ * @param {function(string):!Node} processor
+ * @param {?function(string):!Node} nextProcessor
+ * @param {string} valueText
+ * @return {!DocumentFragment}
+ */
+ function processValue(regex, processor, nextProcessor, valueText)
+ {
+ var container = document.createDocumentFragment();
+
+ var items = valueText.replace(regex, "\0$1\0").split("\0");
+ for (var i = 0; i < items.length; ++i) {
+ if ((i % 2) === 0) {
+ if (nextProcessor)
+ container.appendChild(nextProcessor(items[i]));
+ else
+ container.appendChild(document.createTextNode(items[i]));
+ } else {
+ var processedNode = processor(items[i]);
+ if (processedNode)
+ container.appendChild(processedNode);
+ }
+ }
+
+ return container;
+ }
+
+ /**
+ * @param {string} url
+ * @return {!Node}
+ * @this {WebInspector.StylePropertyTreeElementBase}
+ */
+ function linkifyURL(url)
+ {
+ var hrefUrl = url;
+ var match = hrefUrl.match(/['"]?([^'"]+)/);
+ if (match)
+ hrefUrl = match[1];
+ var container = document.createDocumentFragment();
+ container.appendChild(document.createTextNode("url("));
+ if (this._styleRule.sourceURL)
+ hrefUrl = WebInspector.ParsedURL.completeURL(this._styleRule.sourceURL, hrefUrl);
+ else if (this.node())
+ hrefUrl = this.node().resolveURL(hrefUrl);
+ var hasResource = hrefUrl && !!WebInspector.resourceForURL(hrefUrl);
+ // FIXME: WebInspector.linkifyURLAsNode() should really use baseURI.
+ container.appendChild(WebInspector.linkifyURLAsNode(hrefUrl || url, url, undefined, !hasResource));
+ container.appendChild(document.createTextNode(")"));
+ return container;
+ }
+
+ if (value) {
+ var colorProcessor = processValue.bind(null, WebInspector.StylesSidebarPane._colorRegex, this._processColor.bind(this, nameElement, valueElement), null);
+ valueElement.appendChild(processValue(/url\(\s*([^)]+)\s*\)/g, linkifyURL.bind(this), WebInspector.CSSMetadata.isColorAwareProperty(this.name) && this.parsedOk ? colorProcessor : null, value));
+ }
+
+ this.listItemElement.removeChildren();
+ nameElement.normalize();
+ valueElement.normalize();
+
+ if (!this.treeOutline)
+ return;
+
+ if (this.disabled)
+ this.listItemElement.createChild("span", "styles-clipboard-only").createTextChild("/* ");
+ this.listItemElement.appendChild(nameElement);
+ this.listItemElement.appendChild(document.createTextNode(": "));
+ this.listItemElement.appendChild(this._expandElement);
+ this.listItemElement.appendChild(valueElement);
+ this.listItemElement.appendChild(document.createTextNode(";"));
+ if (this.disabled)
+ this.listItemElement.createChild("span", "styles-clipboard-only").createTextChild(" */");
+
+ if (!this.parsedOk) {
+ // Avoid having longhands under an invalid shorthand.
+ this.hasChildren = false;
+ this.listItemElement.classList.add("not-parsed-ok");
+
+ // Add a separate exclamation mark IMG element with a tooltip.
+ this.listItemElement.insertBefore(WebInspector.StylesSidebarPane.createExclamationMark(this.property), this.listItemElement.firstChild);
+ }
+ if (this.property.inactive)
+ this.listItemElement.classList.add("inactive");
+ this._updateFilter();
+ },
+
+ _updateFilter: function()
+ {
+ var regEx = this.parentPane().filterRegex();
+ this.listItemElement.classList.toggle("filter-match", !!regEx && (regEx.test(this.property.name) || regEx.test(this.property.value)));
+ },
+
+ /**
+ * @param {!Element} nameElement
+ * @param {!Element} valueElement
+ * @param {string} text
+ * @return {!Node}
+ */
+ _processColor: function(nameElement, valueElement, text)
+ {
+ var color = WebInspector.Color.parse(text);
+
+ // We can be called with valid non-color values of |text| (like 'none' from border style)
+ if (!color)
+ return document.createTextNode(text);
+
+ var format = WebInspector.StylesSidebarPane._colorFormat(color);
+ var spectrumHelper = this.editablePane() && this.editablePane()._spectrumHelper;
+ var spectrum = spectrumHelper ? spectrumHelper.spectrum() : null;
+
+ var isEditable = !!(this._styleRule && this._styleRule.editable !== false); // |editable| is true by default.
+ var colorSwatch = new WebInspector.ColorSwatch(!isEditable);
+ colorSwatch.setColorString(text);
+ colorSwatch.element.addEventListener("click", swatchClick.bind(this), false);
+
+ var scrollerElement;
+ var boundSpectrumChanged = spectrumChanged.bind(this);
+ var boundSpectrumHidden = spectrumHidden.bind(this);
+
+ /**
+ * @param {!WebInspector.Event} e
+ * @this {WebInspector.StylePropertyTreeElementBase}
+ */
+ function spectrumChanged(e)
+ {
+ var colorString = /** @type {string} */ (e.data);
+ spectrum.displayText = colorString;
+ colorValueElement.textContent = colorString;
+ colorSwatch.setColorString(colorString);
+ this.applyStyleText(nameElement.textContent + ": " + valueElement.textContent, false, false, false);
+ }
+
+ /**
+ * @param {!WebInspector.Event} event
+ * @this {WebInspector.StylePropertyTreeElementBase}
+ */
+ function spectrumHidden(event)
+ {
+ if (scrollerElement)
+ scrollerElement.removeEventListener("scroll", repositionSpectrum, false);
+ var commitEdit = event.data;
+ var propertyText = !commitEdit && this.originalPropertyText ? this.originalPropertyText : (nameElement.textContent + ": " + valueElement.textContent);
+ this.applyStyleText(propertyText, true, true, false);
+ spectrum.removeEventListener(WebInspector.Spectrum.Events.ColorChanged, boundSpectrumChanged);
+ spectrumHelper.removeEventListener(WebInspector.SpectrumPopupHelper.Events.Hidden, boundSpectrumHidden);
+
+ delete this.editablePane()._isEditingStyle;
+ delete this.originalPropertyText;
+ }
+
+ function repositionSpectrum()
+ {
+ spectrumHelper.reposition(colorSwatch.element);
+ }
+
+ /**
+ * @param {?Event} e
+ * @this {WebInspector.StylePropertyTreeElementBase}
+ */
+ function swatchClick(e)
+ {
+ e.consume(true);
+
+ // Shift + click toggles color formats.
+ // Click opens colorpicker, only if the element is not in computed styles section.
+ if (!spectrumHelper || e.shiftKey) {
+ changeColorDisplay();
+ return;
+ }
+
+ if (!isEditable)
+ return;
+
+ var visible = spectrumHelper.toggle(colorSwatch.element, color, format);
+ if (visible) {
+ spectrum.displayText = color.toString(format);
+ this.originalPropertyText = this.property.propertyText;
+ this.editablePane()._isEditingStyle = true;
+ spectrum.addEventListener(WebInspector.Spectrum.Events.ColorChanged, boundSpectrumChanged);
+ spectrumHelper.addEventListener(WebInspector.SpectrumPopupHelper.Events.Hidden, boundSpectrumHidden);
+
+ scrollerElement = colorSwatch.element.enclosingNodeOrSelfWithClass("scroll-target");
+ if (scrollerElement)
+ scrollerElement.addEventListener("scroll", repositionSpectrum, false);
+ else
+ console.error("Unable to handle color picker scrolling");
+ }
+ }
+
+ var colorValueElement = document.createElement("span");
+ colorValueElement.textContent = color.toString(format);
+
+ /**
+ * @param {string} curFormat
+ */
+ function nextFormat(curFormat)
+ {
+ // The format loop is as follows:
+ // * original
+ // * rgb(a)
+ // * hsl(a)
+ // * nickname (if the color has a nickname)
+ // * if the color is simple:
+ // - shorthex (if has short hex)
+ // - hex
+ var cf = WebInspector.Color.Format;
+
+ switch (curFormat) {
+ case cf.Original:
+ return !color.hasAlpha() ? cf.RGB : cf.RGBA;
+
+ case cf.RGB:
+ case cf.RGBA:
+ return !color.hasAlpha() ? cf.HSL : cf.HSLA;
+
+ case cf.HSL:
+ case cf.HSLA:
+ if (color.nickname())
+ return cf.Nickname;
+ if (!color.hasAlpha())
+ return color.canBeShortHex() ? cf.ShortHEX : cf.HEX;
+ else
+ return cf.Original;
+
+ case cf.ShortHEX:
+ return cf.HEX;
+
+ case cf.HEX:
+ return cf.Original;
+
+ case cf.Nickname:
+ if (!color.hasAlpha())
+ return color.canBeShortHex() ? cf.ShortHEX : cf.HEX;
+ else
+ return cf.Original;
+
+ default:
+ return cf.RGBA;
+ }
+ }
+
+ function changeColorDisplay()
+ {
+ do {
+ format = nextFormat(format);
+ var currentValue = color.toString(format);
+ } while (currentValue === colorValueElement.textContent);
+ colorValueElement.textContent = currentValue;
+ }
+
+ var container = document.createElement("nobr");
+ container.appendChild(colorSwatch.element);
+ container.appendChild(colorValueElement);
+ return container;
+ },
+
+ updateState: function()
+ {
+ if (!this.listItemElement)
+ return;
+
+ if (this.style.isPropertyImplicit(this.name))
+ this.listItemElement.classList.add("implicit");
+ else
+ this.listItemElement.classList.remove("implicit");
+
+ if (this.hasIgnorableError())
+ this.listItemElement.classList.add("has-ignorable-error");
+ else
+ this.listItemElement.classList.remove("has-ignorable-error");
+
+ if (this.inherited)
+ this.listItemElement.classList.add("inherited");
+ else
+ this.listItemElement.classList.remove("inherited");
+
+ if (this.overloaded)
+ this.listItemElement.classList.add("overloaded");
+ else
+ this.listItemElement.classList.remove("overloaded");
+
+ if (this.disabled)
+ this.listItemElement.classList.add("disabled");
+ else
+ this.listItemElement.classList.remove("disabled");
+ },
+
+ __proto__: TreeElement.prototype
+}
+
+/**
+ * @constructor
+ * @extends {WebInspector.StylePropertyTreeElementBase}
+ * @param {!WebInspector.StylesSidebarPane} stylesPane
+ * @param {!Object} styleRule
+ * @param {!WebInspector.CSSStyleDeclaration} style
+ * @param {!WebInspector.CSSProperty} property
+ * @param {boolean} inherited
+ */
+WebInspector.ComputedStylePropertyTreeElement = function(stylesPane, styleRule, style, property, inherited)
+{
+ WebInspector.StylePropertyTreeElementBase.call(this, styleRule, style, property, inherited, false, false);
+ this._stylesPane = stylesPane;
+}
+
+WebInspector.ComputedStylePropertyTreeElement.prototype = {
+ /**
+ * @return {?WebInspector.DOMNode}
+ */
+ node: function()
+ {
+ return this._stylesPane._node;
+ },
+
+ /**
+ * @return {?WebInspector.StylesSidebarPane}
+ */
+ editablePane: function()
+ {
+ return null;
+ },
+
+ /**
+ * @return {!WebInspector.ComputedStyleSidebarPane}
+ */
+ parentPane: function()
+ {
+ return this._stylesPane._computedStylePane;
+ },
+
+ _updateFilter: function()
+ {
+ var regEx = this.parentPane().filterRegex();
+ this.listItemElement.classList.toggle("hidden", !!regEx && (!regEx.test(this.property.name) && !regEx.test(this.property.value)));
+ },
+
+ __proto__: WebInspector.StylePropertyTreeElementBase.prototype
+}
+
+/**
+ * @constructor
+ * @extends {WebInspector.StylePropertyTreeElementBase}
+ * @param {!WebInspector.StylesSidebarPane} stylesPane
+ * @param {!Object} styleRule
+ * @param {!WebInspector.CSSStyleDeclaration} style
+ * @param {!WebInspector.CSSProperty} property
+ * @param {boolean} isShorthand
+ * @param {boolean} inherited
+ * @param {boolean} overloaded
+ */
+WebInspector.StylePropertyTreeElement = function(stylesPane, styleRule, style, property, isShorthand, inherited, overloaded)
+{
+ WebInspector.StylePropertyTreeElementBase.call(this, styleRule, style, property, inherited, overloaded, isShorthand);
+ this._parentPane = stylesPane;
+ this.isShorthand = isShorthand;
+}
+
+WebInspector.StylePropertyTreeElement.prototype = {
+ /**
+ * @return {?WebInspector.DOMNode}
+ */
+ node: function()
+ {
+ return this._parentPane._node;
+ },
+
+ /**
+ * @return {?WebInspector.StylesSidebarPane}
+ */
+ editablePane: function()
+ {
+ return this._parentPane;
+ },
+
+ /**
+ * @return {!WebInspector.StylesSidebarPane}
+ */
+ parentPane: function()
+ {
+ return this._parentPane;
+ },
+
+ /**
+ * @return {?WebInspector.StylePropertiesSection}
+ */
+ section: function()
+ {
+ return this.treeOutline && this.treeOutline.section;
+ },
+
+ /**
+ * @param {function()=} userCallback
+ */
+ _updatePane: function(userCallback)
+ {
+ var section = this.section();
+ if (section && section._parentPane)
+ section._parentPane._refreshUpdate(section, false, userCallback);
+ else {
+ if (userCallback)
+ userCallback();
+ }
+ },
+
+ /**
+ * @param {!WebInspector.CSSStyleDeclaration} newStyle
+ */
+ _applyNewStyle: function(newStyle)
+ {
+ newStyle.parentRule = this.style.parentRule;
+ var oldStyleRange = /** @type {!WebInspector.TextRange} */ (this.style.range);
+ var newStyleRange = /** @type {!WebInspector.TextRange} */ (newStyle.range);
+ this.style = newStyle;
+ this._styleRule.style = newStyle;
+ if (this.style.parentRule) {
+ this.style.parentRule.style = this.style;
+ this._parentPane._styleSheetRuleEdited(this.style.parentRule, oldStyleRange, newStyleRange);
+ }
+ },
+
+ /**
+ * @param {?Event} event
+ */
+ toggleEnabled: function(event)
+ {
+ var disabled = !event.target.checked;
+
+ /**
+ * @param {?WebInspector.CSSStyleDeclaration} newStyle
+ * @this {WebInspector.StylePropertyTreeElement}
+ */
+ function callback(newStyle)
+ {
+ delete this._parentPane._userOperation;
+
+ if (!newStyle)
+ return;
+ this._applyNewStyle(newStyle);
+
+ var section = this.section();
+ if (section && section._parentPane)
+ section._parentPane.dispatchEventToListeners("style property toggled");
+
+ this._updatePane();
+ }
+
+ this._parentPane._userOperation = true;
+ this.property.setDisabled(disabled, callback.bind(this));
+ event.consume();
+ },
+
+ onpopulate: function()
+ {
+ // Only populate once and if this property is a shorthand.
+ if (this.children.length || !this.isShorthand)
+ return;
+
+ var longhandProperties = this.style.longhandProperties(this.name);
+ for (var i = 0; i < longhandProperties.length; ++i) {
+ var name = longhandProperties[i].name;
+ var inherited = false;
+ var overloaded = false;
+
+ var section = this.section();
+ if (section) {
+ inherited = section.isPropertyInherited(name);
+ overloaded = section.isPropertyOverloaded(name);
+ }
+
+ var liveProperty = this.style.getLiveProperty(name);
+ if (!liveProperty)
+ continue;
+
+ var item = new WebInspector.StylePropertyTreeElement(this._parentPane, this._styleRule, this.style, liveProperty, false, inherited, overloaded);
+ this.appendChild(item);
+ }
+ },
+
+ onattach: function()
+ {
+ WebInspector.StylePropertyTreeElementBase.prototype.onattach.call(this);
+
+ this.listItemElement.addEventListener("mousedown", this._mouseDown.bind(this));
+ this.listItemElement.addEventListener("mouseup", this._resetMouseDownElement.bind(this));
+ this.listItemElement.addEventListener("click", this._mouseClick.bind(this));
+ },
+
+ _mouseDown: function(event)
+ {
+ if (this._parentPane) {
+ this._parentPane._mouseDownTreeElement = this;
+ this._parentPane._mouseDownTreeElementIsName = this._isNameElement(event.target);
+ this._parentPane._mouseDownTreeElementIsValue = this._isValueElement(event.target);
+ }
+ },
+
+ _resetMouseDownElement: function()
+ {
+ if (this._parentPane) {
+ delete this._parentPane._mouseDownTreeElement;
+ delete this._parentPane._mouseDownTreeElementIsName;
+ delete this._parentPane._mouseDownTreeElementIsValue;
+ }
+ },
+
+ updateTitle: function()
+ {
+ WebInspector.StylePropertyTreeElementBase.prototype.updateTitle.call(this);
+
+ if (this.parsedOk && this.section() && this.parent.root) {
+ var enabledCheckboxElement = document.createElement("input");
+ enabledCheckboxElement.className = "enabled-button";
+ enabledCheckboxElement.type = "checkbox";
+ enabledCheckboxElement.checked = !this.disabled;
+ enabledCheckboxElement.addEventListener("click", this.toggleEnabled.bind(this), false);
+ this.listItemElement.insertBefore(enabledCheckboxElement, this.listItemElement.firstChild);
+ }
+ },
+
+ _mouseClick: function(event)
+ {
+ if (!window.getSelection().isCollapsed)
+ return;
+
+ event.consume(true);
+
+ if (event.target === this.listItemElement) {
+ var section = this.section();
+ if (!section || !section.editable)
+ return;
+
+ if (section._checkWillCancelEditing())
+ return;
+ section.addNewBlankProperty(this.property.index + 1).startEditing();
+ return;
+ }
+
+ if (WebInspector.KeyboardShortcut.eventHasCtrlOrMeta(event) && this.section().navigable) {
+ this._navigateToSource(event.target);
+ return;
+ }
+
+ this.startEditing(event.target);
+ },
+
+ /**
+ * @param {!Element} element
+ */
+ _navigateToSource: function(element)
+ {
+ console.assert(this.section().navigable);
+ var propertyNameClicked = element === this.nameElement;
+ WebInspector.Revealer.reveal(this.property.uiLocation(propertyNameClicked));
+ },
+
+ /**
+ * @param {!Element} element
+ */
+ _isNameElement: function(element)
+ {
+ return element.enclosingNodeOrSelfWithClass("webkit-css-property") === this.nameElement;
+ },
+
+ /**
+ * @param {!Element} element
+ */
+ _isValueElement: function(element)
+ {
+ return !!element.enclosingNodeOrSelfWithClass("value");
+ },
+
+ /**
+ * @param {?Element=} selectElement
+ */
+ startEditing: function(selectElement)
+ {
+ // FIXME: we don't allow editing of longhand properties under a shorthand right now.
+ if (this.parent.isShorthand)
+ return;
+
+ if (selectElement === this._expandElement)
+ return;
+
+ var section = this.section();
+ if (section && !section.editable)
+ return;
+
+ if (!selectElement)
+ selectElement = this.nameElement; // No arguments passed in - edit the name element by default.
+ else
+ selectElement = selectElement.enclosingNodeOrSelfWithClass("webkit-css-property") || selectElement.enclosingNodeOrSelfWithClass("value");
+
+ if (WebInspector.isBeingEdited(selectElement))
+ return;
+
+ var isEditingName = selectElement === this.nameElement;
+ if (!isEditingName)
+ this.valueElement.textContent = restoreURLs(this.valueElement.textContent, this.value);
+
+ /**
+ * @param {string} fieldValue
+ * @param {string} modelValue
+ * @return {string}
+ */
+ function restoreURLs(fieldValue, modelValue)
+ {
+ const urlRegex = /\b(url\([^)]*\))/g;
+ var splitFieldValue = fieldValue.split(urlRegex);
+ if (splitFieldValue.length === 1)
+ return fieldValue;
+ var modelUrlRegex = new RegExp(urlRegex);
+ for (var i = 1; i < splitFieldValue.length; i += 2) {
+ var match = modelUrlRegex.exec(modelValue);
+ if (match)
+ splitFieldValue[i] = match[0];
+ }
+ return splitFieldValue.join("");
+ }
+
+ var context = {
+ expanded: this.expanded,
+ hasChildren: this.hasChildren,
+ isEditingName: isEditingName,
+ previousContent: selectElement.textContent
+ };
+
+ // Lie about our children to prevent expanding on double click and to collapse shorthands.
+ this.hasChildren = false;
+
+ if (selectElement.parentElement)
+ selectElement.parentElement.classList.add("child-editing");
+ selectElement.textContent = selectElement.textContent; // remove color swatch and the like
+
+ /**
+ * @this {WebInspector.StylePropertyTreeElement}
+ */
+ function pasteHandler(context, event)
+ {
+ var data = event.clipboardData.getData("Text");
+ if (!data)
+ return;
+ var colonIdx = data.indexOf(":");
+ if (colonIdx < 0)
+ return;
+ var name = data.substring(0, colonIdx).trim();
+ var value = data.substring(colonIdx + 1).trim();
+
+ event.preventDefault();
+
+ if (!("originalName" in context)) {
+ context.originalName = this.nameElement.textContent;
+ context.originalValue = this.valueElement.textContent;
+ }
+ this.property.name = name;
+ this.property.value = value;
+ this.nameElement.textContent = name;
+ this.valueElement.textContent = value;
+ this.nameElement.normalize();
+ this.valueElement.normalize();
+
+ this.editingCommitted(event.target.textContent, context, "forward");
+ }
+
+ /**
+ * @this {WebInspector.StylePropertyTreeElement}
+ */
+ function blurListener(context, event)
+ {
+ var treeElement = this._parentPane._mouseDownTreeElement;
+ var moveDirection = "";
+ if (treeElement === this) {
+ if (isEditingName && this._parentPane._mouseDownTreeElementIsValue)
+ moveDirection = "forward";
+ if (!isEditingName && this._parentPane._mouseDownTreeElementIsName)
+ moveDirection = "backward";
+ }
+ this.editingCommitted(event.target.textContent, context, moveDirection);
+ }
+
+ delete this.originalPropertyText;
+
+ this._parentPane._isEditingStyle = true;
+ if (selectElement.parentElement)
+ selectElement.parentElement.scrollIntoViewIfNeeded(false);
+
+ var applyItemCallback = !isEditingName ? this._applyFreeFlowStyleTextEdit.bind(this, true) : undefined;
+ this._prompt = new WebInspector.StylesSidebarPane.CSSPropertyPrompt(isEditingName ? WebInspector.CSSMetadata.cssPropertiesMetainfo : WebInspector.CSSMetadata.keywordsForProperty(this.nameElement.textContent), this, isEditingName);
+ if (applyItemCallback) {
+ this._prompt.addEventListener(WebInspector.TextPrompt.Events.ItemApplied, applyItemCallback, this);
+ this._prompt.addEventListener(WebInspector.TextPrompt.Events.ItemAccepted, applyItemCallback, this);
+ }
+ var proxyElement = this._prompt.attachAndStartEditing(selectElement, blurListener.bind(this, context));
+
+ proxyElement.addEventListener("keydown", this.editingNameValueKeyDown.bind(this, context), false);
+ proxyElement.addEventListener("keypress", this.editingNameValueKeyPress.bind(this, context), false);
+ if (isEditingName)
+ proxyElement.addEventListener("paste", pasteHandler.bind(this, context), false);
+
+ window.getSelection().setBaseAndExtent(selectElement, 0, selectElement, 1);
+ },
+
+ editingNameValueKeyDown: function(context, event)
+ {
+ if (event.handled)
+ return;
+
+ var isEditingName = context.isEditingName;
+ var result;
+
+ if (isEnterKey(event)) {
+ event.preventDefault();
+ result = "forward";
+ } else if (event.keyCode === WebInspector.KeyboardShortcut.Keys.Esc.code || event.keyIdentifier === "U+001B")
+ result = "cancel";
+ else if (!isEditingName && this._newProperty && event.keyCode === WebInspector.KeyboardShortcut.Keys.Backspace.code) {
+ // For a new property, when Backspace is pressed at the beginning of new property value, move back to the property name.
+ var selection = window.getSelection();
+ if (selection.isCollapsed && !selection.focusOffset) {
+ event.preventDefault();
+ result = "backward";
+ }
+ } else if (event.keyIdentifier === "U+0009") { // Tab key.
+ result = event.shiftKey ? "backward" : "forward";
+ event.preventDefault();
+ }
+
+ if (result) {
+ switch (result) {
+ case "cancel":
+ this.editingCancelled(null, context);
+ break;
+ case "forward":
+ case "backward":
+ this.editingCommitted(event.target.textContent, context, result);
+ break;
+ }
+
+ event.consume();
+ return;
+ }
+
+ if (!isEditingName)
+ this._applyFreeFlowStyleTextEdit(false);
+ },
+
+ editingNameValueKeyPress: function(context, event)
+ {
+ function shouldCommitValueSemicolon(text, cursorPosition)
+ {
+ // FIXME: should this account for semicolons inside comments?
+ var openQuote = "";
+ for (var i = 0; i < cursorPosition; ++i) {
+ var ch = text[i];
+ if (ch === "\\" && openQuote !== "")
+ ++i; // skip next character inside string
+ else if (!openQuote && (ch === "\"" || ch === "'"))
+ openQuote = ch;
+ else if (openQuote === ch)
+ openQuote = "";
+ }
+ return !openQuote;
+ }
+
+ var keyChar = String.fromCharCode(event.charCode);
+ var isFieldInputTerminated = (context.isEditingName ? keyChar === ":" : keyChar === ";" && shouldCommitValueSemicolon(event.target.textContent, event.target.selectionLeftOffset()));
+ if (isFieldInputTerminated) {
+ // Enter or colon (for name)/semicolon outside of string (for value).
+ event.consume(true);
+ this.editingCommitted(event.target.textContent, context, "forward");
+ return;
+ }
+ },
+
+ _applyFreeFlowStyleTextEdit: function(now)
+ {
+ if (this._applyFreeFlowStyleTextEditTimer)
+ clearTimeout(this._applyFreeFlowStyleTextEditTimer);
+
+ /**
+ * @this {WebInspector.StylePropertyTreeElement}
+ */
+ function apply()
+ {
+ var valueText = this.valueElement.textContent;
+ if (valueText.indexOf(";") === -1)
+ this.applyStyleText(this.nameElement.textContent + ": " + valueText, false, false, false);
+ }
+ if (now)
+ apply.call(this);
+ else
+ this._applyFreeFlowStyleTextEditTimer = setTimeout(apply.bind(this), 100);
+ },
+
+ kickFreeFlowStyleEditForTest: function()
+ {
+ this._applyFreeFlowStyleTextEdit(true);
+ },
+
+ editingEnded: function(context)
+ {
+ this._resetMouseDownElement();
+ if (this._applyFreeFlowStyleTextEditTimer)
+ clearTimeout(this._applyFreeFlowStyleTextEditTimer);
+
+ this.hasChildren = context.hasChildren;
+ if (context.expanded)
+ this.expand();
+ var editedElement = context.isEditingName ? this.nameElement : this.valueElement;
+ // The proxyElement has been deleted, no need to remove listener.
+ if (editedElement.parentElement)
+ editedElement.parentElement.classList.remove("child-editing");
+
+ delete this._parentPane._isEditingStyle;
+ },
+
+ editingCancelled: function(element, context)
+ {
+ this._removePrompt();
+ this._revertStyleUponEditingCanceled(this.originalPropertyText);
+ // This should happen last, as it clears the info necessary to restore the property value after [Page]Up/Down changes.
+ this.editingEnded(context);
+ },
+
+ _revertStyleUponEditingCanceled: function(originalPropertyText)
+ {
+ if (typeof originalPropertyText === "string") {
+ delete this.originalPropertyText;
+ this.applyStyleText(originalPropertyText, true, false, true);
+ } else {
+ if (this._newProperty)
+ this.treeOutline.removeChild(this);
+ else
+ this.updateTitle();
+ }
+ },
+
+ _findSibling: function(moveDirection)
+ {
+ var target = this;
+ do {
+ target = (moveDirection === "forward" ? target.nextSibling : target.previousSibling);
+ } while(target && target.inherited);
+
+ return target;
+ },
+
+ /**
+ * @param {string} userInput
+ * @param {!Object} context
+ * @param {string} moveDirection
+ */
+ editingCommitted: function(userInput, context, moveDirection)
+ {
+ this._removePrompt();
+ this.editingEnded(context);
+ var isEditingName = context.isEditingName;
+
+ // Determine where to move to before making changes
+ var createNewProperty, moveToPropertyName, moveToSelector;
+ var isDataPasted = "originalName" in context;
+ var isDirtyViaPaste = isDataPasted && (this.nameElement.textContent !== context.originalName || this.valueElement.textContent !== context.originalValue);
+ var isPropertySplitPaste = isDataPasted && isEditingName && this.valueElement.textContent !== context.originalValue;
+ var moveTo = this;
+ var moveToOther = (isEditingName ^ (moveDirection === "forward"));
+ var abandonNewProperty = this._newProperty && !userInput && (moveToOther || isEditingName);
+ if (moveDirection === "forward" && (!isEditingName || isPropertySplitPaste) || moveDirection === "backward" && isEditingName) {
+ moveTo = moveTo._findSibling(moveDirection);
+ if (moveTo)
+ moveToPropertyName = moveTo.name;
+ else if (moveDirection === "forward" && (!this._newProperty || userInput))
+ createNewProperty = true;
+ else if (moveDirection === "backward")
+ moveToSelector = true;
+ }
+
+ // Make the Changes and trigger the moveToNextCallback after updating.
+ var moveToIndex = moveTo && this.treeOutline ? this.treeOutline.children.indexOf(moveTo) : -1;
+ var blankInput = /^\s*$/.test(userInput);
+ var shouldCommitNewProperty = this._newProperty && (isPropertySplitPaste || moveToOther || (!moveDirection && !isEditingName) || (isEditingName && blankInput));
+ var section = this.section();
+ if (((userInput !== context.previousContent || isDirtyViaPaste) && !this._newProperty) || shouldCommitNewProperty) {
+ section._afterUpdate = moveToNextCallback.bind(this, this._newProperty, !blankInput, section);
+ var propertyText;
+ if (blankInput || (this._newProperty && /^\s*$/.test(this.valueElement.textContent)))
+ propertyText = "";
+ else {
+ if (isEditingName)
+ propertyText = userInput + ": " + this.property.value;
+ else
+ propertyText = this.property.name + ": " + userInput;
+ }
+ this.applyStyleText(propertyText, true, true, false);
+ } else {
+ if (isEditingName)
+ this.property.name = userInput;
+ else
+ this.property.value = userInput;
+ if (!isDataPasted && !this._newProperty)
+ this.updateTitle();
+ moveToNextCallback.call(this, this._newProperty, false, section);
+ }
+
+ /**
+ * The Callback to start editing the next/previous property/selector.
+ * @this {WebInspector.StylePropertyTreeElement}
+ */
+ function moveToNextCallback(alreadyNew, valueChanged, section)
+ {
+ if (!moveDirection)
+ return;
+
+ // User just tabbed through without changes.
+ if (moveTo && moveTo.parent) {
+ moveTo.startEditing(!isEditingName ? moveTo.nameElement : moveTo.valueElement);
+ return;
+ }
+
+ // User has made a change then tabbed, wiping all the original treeElements.
+ // Recalculate the new treeElement for the same property we were going to edit next.
+ if (moveTo && !moveTo.parent) {
+ var propertyElements = section.propertiesTreeOutline.children;
+ if (moveDirection === "forward" && blankInput && !isEditingName)
+ --moveToIndex;
+ if (moveToIndex >= propertyElements.length && !this._newProperty)
+ createNewProperty = true;
+ else {
+ var treeElement = moveToIndex >= 0 ? propertyElements[moveToIndex] : null;
+ if (treeElement) {
+ var elementToEdit = !isEditingName || isPropertySplitPaste ? treeElement.nameElement : treeElement.valueElement;
+ if (alreadyNew && blankInput)
+ elementToEdit = moveDirection === "forward" ? treeElement.nameElement : treeElement.valueElement;
+ treeElement.startEditing(elementToEdit);
+ return;
+ } else if (!alreadyNew)
+ moveToSelector = true;
+ }
+ }
+
+ // Create a new attribute in this section (or move to next editable selector if possible).
+ if (createNewProperty) {
+ if (alreadyNew && !valueChanged && (isEditingName ^ (moveDirection === "backward")))
+ return;
+
+ section.addNewBlankProperty().startEditing();
+ return;
+ }
+
+ if (abandonNewProperty) {
+ moveTo = this._findSibling(moveDirection);
+ var sectionToEdit = (moveTo || moveDirection === "backward") ? section : section.nextEditableSibling();
+ if (sectionToEdit) {
+ if (sectionToEdit.rule)
+ sectionToEdit.startEditingSelector();
+ else
+ sectionToEdit._moveEditorFromSelector(moveDirection);
+ }
+ return;
+ }
+
+ if (moveToSelector) {
+ if (section.rule)
+ section.startEditingSelector();
+ else
+ section._moveEditorFromSelector(moveDirection);
+ }
+ }
+ },
+
+ _removePrompt: function()
+ {
+ // BUG 53242. This cannot go into editingEnded(), as it should always happen first for any editing outcome.
+ if (this._prompt) {
+ this._prompt.detach();
+ delete this._prompt;
+ }
+ },
+
+ _hasBeenModifiedIncrementally: function()
+ {
+ // New properties applied via up/down or live editing have an originalPropertyText and will be deleted later
+ // on, if cancelled, when the empty string gets applied as their style text.
+ return typeof this.originalPropertyText === "string" || (!!this.property.propertyText && this._newProperty);
+ },
+
+ styleTextAppliedForTest: function()
+ {
+ },
+
+ applyStyleText: function(styleText, updateInterface, majorChange, isRevert)
+ {
+ function userOperationFinishedCallback(parentPane, updateInterface)
+ {
+ if (updateInterface)
+ delete parentPane._userOperation;
+ }
+
+ // Leave a way to cancel editing after incremental changes.
+ if (!isRevert && !updateInterface && !this._hasBeenModifiedIncrementally()) {
+ // Remember the rule's original CSS text on [Page](Up|Down), so it can be restored
+ // if the editing is canceled.
+ this.originalPropertyText = this.property.propertyText;
+ }
+
+ if (!this.treeOutline)
+ return;
+
+ var section = this.section();
+ styleText = styleText.replace(/\s/g, " ").trim(); // Replace &nbsp; with whitespace.
+ var styleTextLength = styleText.length;
+ if (!styleTextLength && updateInterface && !isRevert && this._newProperty && !this._hasBeenModifiedIncrementally()) {
+ // The user deleted everything and never applied a new property value via Up/Down scrolling/live editing, so remove the tree element and update.
+ this.parent.removeChild(this);
+ section.afterUpdate();
+ return;
+ }
+
+ var currentNode = this._parentPane._node;
+ if (updateInterface)
+ this._parentPane._userOperation = true;
+
+ /**
+ * @param {function()} userCallback
+ * @param {string} originalPropertyText
+ * @param {?WebInspector.CSSStyleDeclaration} newStyle
+ * @this {WebInspector.StylePropertyTreeElement}
+ */
+ function callback(userCallback, originalPropertyText, newStyle)
+ {
+ if (!newStyle) {
+ if (updateInterface) {
+ // It did not apply, cancel editing.
+ this._revertStyleUponEditingCanceled(originalPropertyText);
+ }
+ userCallback();
+ return;
+ }
+ this._applyNewStyle(newStyle);
+
+ if (this._newProperty)
+ this._newPropertyInStyle = true;
+
+ this.property = newStyle.propertyAt(this.property.index);
+ if (section && section._parentPane)
+ section._parentPane.dispatchEventToListeners("style edited");
+
+ if (updateInterface && currentNode === this.node()) {
+ this._updatePane(userCallback);
+ this.styleTextAppliedForTest();
+ return;
+ }
+
+ userCallback();
+ this.styleTextAppliedForTest();
+ }
+
+ // Append a ";" if the new text does not end in ";".
+ // FIXME: this does not handle trailing comments.
+ if (styleText.length && !/;\s*$/.test(styleText))
+ styleText += ";";
+ var overwriteProperty = !!(!this._newProperty || this._newPropertyInStyle);
+ this.property.setText(styleText, majorChange, overwriteProperty, callback.bind(this, userOperationFinishedCallback.bind(null, this._parentPane, updateInterface), this.originalPropertyText));
+ },
+
+ /**
+ * @return {boolean}
+ */
+ ondblclick: function()
+ {
+ return true; // handled
+ },
+
+ /**
+ * @param {?Event} event
+ * @return {boolean}
+ */
+ isEventWithinDisclosureTriangle: function(event)
+ {
+ return event.target === this._expandElement;
+ },
+
+ __proto__: WebInspector.StylePropertyTreeElementBase.prototype
+}
+
+/**
+ * @constructor
+ * @extends {WebInspector.TextPrompt}
+ * @param {!WebInspector.CSSMetadata} cssCompletions
+ * @param {!WebInspector.StylePropertyTreeElement} sidebarPane
+ * @param {boolean} isEditingName
+ */
+WebInspector.StylesSidebarPane.CSSPropertyPrompt = function(cssCompletions, sidebarPane, isEditingName)
+{
+ // Use the same callback both for applyItemCallback and acceptItemCallback.
+ WebInspector.TextPrompt.call(this, this._buildPropertyCompletions.bind(this), WebInspector.StyleValueDelimiters);
+ this.setSuggestBoxEnabled(true);
+ this._cssCompletions = cssCompletions;
+ this._sidebarPane = sidebarPane;
+ this._isEditingName = isEditingName;
+
+ if (!isEditingName)
+ this.disableDefaultSuggestionForEmptyInput();
+}
+
+WebInspector.StylesSidebarPane.CSSPropertyPrompt.prototype = {
+ /**
+ * @param {?Event} event
+ */
+ onKeyDown: function(event)
+ {
+ switch (event.keyIdentifier) {
+ case "Up":
+ case "Down":
+ case "PageUp":
+ case "PageDown":
+ if (this._handleNameOrValueUpDown(event)) {
+ event.preventDefault();
+ return;
+ }
+ break;
+ case "Enter":
+ if (this.autoCompleteElement && !this.autoCompleteElement.textContent.length) {
+ this.tabKeyPressed();
+ return;
+ }
+ break;
+ }
+
+ WebInspector.TextPrompt.prototype.onKeyDown.call(this, event);
+ },
+
+ onMouseWheel: function(event)
+ {
+ if (this._handleNameOrValueUpDown(event)) {
+ event.consume(true);
+ return;
+ }
+ WebInspector.TextPrompt.prototype.onMouseWheel.call(this, event);
+ },
+
+ /**
+ * @override
+ * @return {boolean}
+ */
+ tabKeyPressed: function()
+ {
+ this.acceptAutoComplete();
+
+ // Always tab to the next field.
+ return false;
+ },
+
+ /**
+ * @param {?Event} event
+ * @return {boolean}
+ */
+ _handleNameOrValueUpDown: function(event)
+ {
+ /**
+ * @param {string} originalValue
+ * @param {string} replacementString
+ * @this {WebInspector.StylesSidebarPane.CSSPropertyPrompt}
+ */
+ function finishHandler(originalValue, replacementString)
+ {
+ // Synthesize property text disregarding any comments, custom whitespace etc.
+ this._sidebarPane.applyStyleText(this._sidebarPane.nameElement.textContent + ": " + this._sidebarPane.valueElement.textContent, false, false, false);
+ }
+
+ // Handle numeric value increment/decrement only at this point.
+ if (!this._isEditingName && WebInspector.handleElementValueModifications(event, this._sidebarPane.valueElement, finishHandler.bind(this), this._isValueSuggestion.bind(this)))
+ return true;
+
+ return false;
+ },
+
+ /**
+ * @param {string} word
+ * @return {boolean}
+ */
+ _isValueSuggestion: function(word)
+ {
+ if (!word)
+ return false;
+ word = word.toLowerCase();
+ return this._cssCompletions.keySet().hasOwnProperty(word);
+ },
+
+ /**
+ * @param {!Element} proxyElement
+ * @param {!Range} wordRange
+ * @param {boolean} force
+ * @param {function(!Array.<string>, number=)} completionsReadyCallback
+ */
+ _buildPropertyCompletions: function(proxyElement, wordRange, force, completionsReadyCallback)
+ {
+ var prefix = wordRange.toString().toLowerCase();
+ if (!prefix && !force && (this._isEditingName || proxyElement.textContent.length)) {
+ completionsReadyCallback([]);
+ return;
+ }
+
+ var results = this._cssCompletions.startsWith(prefix);
+ var selectedIndex = this._cssCompletions.mostUsedOf(results);
+ completionsReadyCallback(results, selectedIndex);
+ },
+
+ __proto__: WebInspector.TextPrompt.prototype
+}