blob: c044d2f2a6b5ec62f5d41de5d63bf4a314d6f253 [file] [log] [blame]
/*
* Copyright (C) 2015 Apple Inc. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 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.
*
* THIS SOFTWARE IS PROVIDED BY APPLE INC. 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 INC. 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.
*/
WebInspector.VisualStyleSelectorTreeItem = class VisualStyleSelectorTreeItem extends WebInspector.GeneralTreeElement
{
constructor(delegate, style, title, subtitle)
{
let iconClassName;
switch (style.type) {
case WebInspector.CSSStyleDeclaration.Type.Rule:
console.assert(style.ownerRule instanceof WebInspector.CSSRule, style.ownerRule);
if (style.inherited)
iconClassName = WebInspector.CSSStyleDeclarationSection.InheritedStyleRuleIconStyleClassName;
else if (style.ownerRule.type === WebInspector.CSSStyleSheet.Type.Author)
iconClassName = WebInspector.CSSStyleDeclarationSection.AuthorStyleRuleIconStyleClassName;
else if (style.ownerRule.type === WebInspector.CSSStyleSheet.Type.User)
iconClassName = WebInspector.CSSStyleDeclarationSection.UserStyleRuleIconStyleClassName;
else if (style.ownerRule.type === WebInspector.CSSStyleSheet.Type.UserAgent)
iconClassName = WebInspector.CSSStyleDeclarationSection.UserAgentStyleRuleIconStyleClassName;
else if (style.ownerRule.type === WebInspector.CSSStyleSheet.Type.Inspector)
iconClassName = WebInspector.CSSStyleDeclarationSection.InspectorStyleRuleIconStyleClassName;
break;
case WebInspector.CSSStyleDeclaration.Type.Inline:
case WebInspector.CSSStyleDeclaration.Type.Attribute:
if (style.inherited)
iconClassName = WebInspector.CSSStyleDeclarationSection.InheritedElementStyleRuleIconStyleClassName;
else
iconClassName = WebInspector.DOMTreeElementPathComponent.DOMElementIconStyleClassName;
break;
}
let iconClasses = [iconClassName];
if (style.ownerRule && style.ownerRule.hasMatchedPseudoElementSelector())
iconClasses.push(WebInspector.CSSStyleDeclarationSection.PseudoElementSelectorStyleClassName);
title = title.trim();
super(["visual-style-selector-item", ...iconClasses], title, subtitle, style);
this._delegate = delegate;
this._iconClasses = iconClasses;
this._lastValue = title;
this._enableEditing = true;
this._hasInvalidSelector = false;
}
// Public
get iconClassName()
{
return this._iconClasses.join(" ");
}
get selectorText()
{
let titleText = this._mainTitleElement.textContent;
if (!titleText || !titleText.length)
titleText = this._mainTitle;
return titleText.trim();
}
// Protected
onattach()
{
super.onattach();
this._listItemNode.addEventListener("mouseover", this._highlightNodesWithSelector.bind(this));
this._listItemNode.addEventListener("mouseout", this._hideDOMNodeHighlight.bind(this));
this._checkboxElement = document.createElement("input");
this._checkboxElement.type = "checkbox";
this._checkboxElement.checked = !this.representedObject[WebInspector.VisualStyleDetailsPanel.StyleDisabledSymbol];
this._updateCheckboxTitle();
this._checkboxElement.addEventListener("change", this._handleCheckboxChanged.bind(this));
this._listItemNode.insertBefore(this._checkboxElement, this._iconElement);
this._iconElement.addEventListener("click", this._handleIconElementClicked.bind(this));
this._mainTitleElement.spellcheck = false;
this._mainTitleElement.addEventListener("mousedown", this._handleMainTitleMouseDown.bind(this));
this._mainTitleElement.addEventListener("keydown", this._handleMainTitleKeyDown.bind(this));
this._mainTitleElement.addEventListener("keyup", this._highlightNodesWithSelector.bind(this));
this._mainTitleElement.addEventListener("blur", this._commitSelector.bind(this));
this.representedObject.addEventListener(WebInspector.CSSStyleDeclaration.Event.InitialTextModified, this._styleTextModified, this);
if (this.representedObject.ownerRule)
this.representedObject.ownerRule.addEventListener(WebInspector.CSSRule.Event.SelectorChanged, this._updateSelectorIcon, this);
this._styleTextModified();
}
ondeselect()
{
this._listItemNode.classList.remove("editable");
this._mainTitleElement.contentEditable = false;
}
populateContextMenu(contextMenu, event)
{
contextMenu.appendItem(WebInspector.UIString("Copy Rule"), () => {
InspectorFrontendHost.copyText(this.representedObject.generateCSSRuleString());
});
if (this.representedObject.modified) {
contextMenu.appendItem(WebInspector.UIString("Reset"), () => {
this.representedObject.resetText();
});
}
if (!this.representedObject.ownerRule)
return;
contextMenu.appendItem(WebInspector.UIString("Show Source"), () => {
const options = {
ignoreNetworkTab: true,
ignoreSearchTab: true,
};
if (event.metaKey)
WebInspector.showOriginalUnformattedSourceCodeLocation(this.representedObject.ownerRule.sourceCodeLocation, options);
else
WebInspector.showSourceCodeLocation(this.representedObject.ownerRule.sourceCodeLocation, options);
});
// Only used one colon temporarily since single-colon pseudo elements are valid CSS.
if (WebInspector.CSSStyleManager.PseudoElementNames.some((className) => this.representedObject.selectorText.includes(":" + className)))
return;
if (WebInspector.CSSStyleManager.ForceablePseudoClasses.every((className) => !this.representedObject.selectorText.includes(":" + className))) {
contextMenu.appendSeparator();
for (let pseudoClass of WebInspector.CSSStyleManager.ForceablePseudoClasses) {
if (pseudoClass === "visited" && this.representedObject.node.nodeName() !== "A")
continue;
let pseudoClassSelector = ":" + pseudoClass;
contextMenu.appendItem(WebInspector.UIString("Add %s Rule").format(pseudoClassSelector), () => {
this.representedObject.node.setPseudoClassEnabled(pseudoClass, true);
let pseudoSelectors = this.representedObject.ownerRule.selectors.map((selector) => selector.text + pseudoClassSelector);
this.representedObject.nodeStyles.addRule(pseudoSelectors.join(", "));
});
}
}
contextMenu.appendSeparator();
for (let pseudoElement of WebInspector.CSSStyleManager.PseudoElementNames) {
let pseudoElementSelector = "::" + pseudoElement;
const styleText = "content: \"\";";
let existingTreeItem = null;
if (this._delegate && typeof this._delegate.treeItemForStyle === "function") {
let selectorText = this.representedObject.ownerRule.selectorText;
let existingRules = this.representedObject.nodeStyles.rulesForSelector(selectorText + pseudoElementSelector);
if (existingRules.length) {
// There shouldn't really ever be more than one pseudo-element rule
// that is not in a media query. As such, just focus the first rule
// on the assumption that it is the only one necessary.
existingTreeItem = this._delegate.treeItemForStyle(existingRules[0].style);
}
}
let title = existingTreeItem ? WebInspector.UIString("Select %s Rule") : WebInspector.UIString("Create %s Rule");
contextMenu.appendItem(title.format(pseudoElementSelector), () => {
if (existingTreeItem) {
existingTreeItem.select(true, true);
return;
}
let pseudoSelectors = this.representedObject.ownerRule.selectors.map((selector) => selector.text + pseudoElementSelector);
this.representedObject.nodeStyles.addRule(pseudoSelectors.join(", "), styleText);
});
}
super.populateContextMenu(contextMenu, event);
}
// Private
_highlightNodesWithSelector()
{
if (!this.representedObject.ownerRule) {
WebInspector.domTreeManager.highlightDOMNode(this.representedObject.node.id);
return;
}
WebInspector.domTreeManager.highlightSelector(this.selectorText, this.representedObject.node.ownerDocument.frameIdentifier);
}
_hideDOMNodeHighlight()
{
WebInspector.domTreeManager.hideDOMNodeHighlight();
}
_handleCheckboxChanged(event)
{
this._updateCheckboxTitle();
this.dispatchEventToListeners(WebInspector.VisualStyleSelectorTreeItem.Event.CheckboxChanged, {enabled: this._checkboxElement.checked});
}
_updateCheckboxTitle()
{
if (this._checkboxElement.checked)
this._checkboxElement.title = WebInspector.UIString("Comment out rule");
else
this._checkboxElement.title = WebInspector.UIString("Uncomment rule");
}
_handleMainTitleMouseDown(event)
{
if (event.button !== 0 || event.ctrlKey)
return;
this._listItemNode.classList.toggle("editable", this.selected);
this._mainTitleElement.contentEditable = this.selected ? "plaintext-only" : false;
}
_handleMainTitleKeyDown(event)
{
this._highlightNodesWithSelector();
let enterKeyCode = WebInspector.KeyboardShortcut.Key.Enter.keyCode;
if (event.keyCode === enterKeyCode)
this._mainTitleElement.blur();
}
_commitSelector()
{
this._hideDOMNodeHighlight();
this._listItemNode.classList.remove("editable");
this._mainTitleElement.contentEditable = false;
this._updateTitleTooltip();
let value = this.selectorText;
if (value === this._lastValue && !this._hasInvalidSelector)
return;
this.representedObject.ownerRule.selectorText = value;
}
_styleTextModified()
{
this._listItemNode.classList.toggle("modified", this.representedObject.modified);
}
_updateSelectorIcon(event)
{
this._hasInvalidSelector = event && event.data && !event.data.valid;
this._listItemNode.classList.toggle("selector-invalid", !!this._hasInvalidSelector);
if (this._hasInvalidSelector) {
this._iconElement.title = WebInspector.UIString("The selector ā€œ%sā€ is invalid.\nClick to revert to the previous selector.").format(this.selectorText);
this.mainTitleElement.title = WebInspector.UIString("Using previous selector ā€œ%sā€").format(this.representedObject.ownerRule.selectorText);
return;
}
this._iconElement.title = null;
this.mainTitleElement.title = null;
let hasMatchedPseudoElementSelector = this.representedObject.ownerRule && this.representedObject.ownerRule.hasMatchedPseudoElementSelector();
this._iconClasses.toggleIncludes(WebInspector.CSSStyleDeclarationSection.PseudoElementSelectorStyleClassName, hasMatchedPseudoElementSelector);
}
_handleIconElementClicked(event)
{
if (this._hasInvalidSelector && this.representedObject.ownerRule) {
this.mainTitleElement.textContent = this._lastValue = this.representedObject.ownerRule.selectorText;
this._updateSelectorIcon();
return;
}
}
};
WebInspector.VisualStyleSelectorTreeItem.Event = {
CheckboxChanged: "visual-style-selector-item-checkbox-changed"
};