1 /* 2 * Copyright (C) 2011, Google Inc. All rights reserved. 3 * 4 * Redistribution and use in source and binary forms, with or without 5 * modification, are permitted provided that the following conditions 6 * are met: 7 * 1. Redistributions of source code must retain the above copyright 8 * notice, this list of conditions and the following disclaimer. 9 * 2. Redistributions in binary form must reproduce the above copyright 10 * notice, this list of conditions and the following disclaimer in the 11 * documentation and/or other materials provided with the distribution. 12 * 13 * THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS'' AND ANY 14 * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED 15 * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE 16 * DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS BE LIABLE FOR ANY 17 * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES 18 * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; 19 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON 20 * ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT 21 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS 22 * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 23 */ 24 25 #include "config.h" 26 #include "core/inspector/InspectorStyleTextEditor.h" 27 28 #include "core/css/CSSPropertySourceData.h" 29 #include "core/html/parser/HTMLParserIdioms.h" 30 #include "core/inspector/InspectorStyleSheet.h" 31 32 namespace WebCore { 33 34 InspectorStyleTextEditor::InspectorStyleTextEditor(Vector<InspectorStyleProperty>* allProperties, const String& styleText, const NewLineAndWhitespace& format) 35 : m_allProperties(allProperties) 36 , m_styleText(styleText) 37 , m_format(format) 38 { 39 } 40 41 void InspectorStyleTextEditor::insertProperty(unsigned index, const String& propertyText, unsigned styleBodyLength) 42 { 43 long propertyStart = 0; 44 45 bool insertLast = true; 46 if (index < m_allProperties->size()) { 47 const InspectorStyleProperty& property = m_allProperties->at(index); 48 if (property.hasSource) { 49 propertyStart = property.sourceData.range.start; 50 // If inserting before a disabled property, it should be shifted, too. 51 insertLast = false; 52 } 53 } 54 55 bool insertFirstInSource = !m_allProperties->size() || !m_allProperties->at(0).hasSource; 56 bool insertLastInSource = true; 57 for (unsigned i = index, size = m_allProperties->size(); i < size; ++i) { 58 const InspectorStyleProperty& property = m_allProperties->at(i); 59 if (property.hasSource) { 60 insertLastInSource = false; 61 break; 62 } 63 } 64 65 String textToSet = propertyText; 66 67 int formattingPrependOffset = 0; 68 if (insertLast && !insertFirstInSource) { 69 propertyStart = styleBodyLength; 70 if (propertyStart && textToSet.length()) { 71 long curPos = propertyStart - 1; // The last position of style declaration, since propertyStart points past one. 72 while (curPos && isHTMLSpace<UChar>(m_styleText[curPos])) 73 --curPos; 74 if (curPos) { 75 bool terminated = m_styleText[curPos] == ';' || (m_styleText[curPos] == '/' && m_styleText[curPos - 1] == '*'); 76 if (!terminated) { 77 // Prepend a ";" to the property text if appending to a style declaration where 78 // the last property has no trailing ";". 79 textToSet.insert(";", 0); 80 formattingPrependOffset = 1; 81 } 82 } 83 } 84 } 85 86 const String& formatLineFeed = m_format.first; 87 const String& formatPropertyPrefix = m_format.second; 88 if (insertLastInSource) { 89 long formatPropertyPrefixLength = formatPropertyPrefix.length(); 90 if (!formattingPrependOffset && (propertyStart < formatPropertyPrefixLength || m_styleText.substring(propertyStart - formatPropertyPrefixLength, formatPropertyPrefixLength) != formatPropertyPrefix)) { 91 textToSet.insert(formatPropertyPrefix, formattingPrependOffset); 92 if (!propertyStart || !isHTMLLineBreak(m_styleText[propertyStart - 1])) 93 textToSet.insert(formatLineFeed, formattingPrependOffset); 94 } 95 if (!isHTMLLineBreak(m_styleText[propertyStart])) 96 textToSet.append(formatLineFeed); 97 } else { 98 String fullPrefix = formatLineFeed + formatPropertyPrefix; 99 long fullPrefixLength = fullPrefix.length(); 100 textToSet.append(fullPrefix); 101 if (insertFirstInSource && (propertyStart < fullPrefixLength || m_styleText.substring(propertyStart - fullPrefixLength, fullPrefixLength) != fullPrefix)) 102 textToSet.insert(fullPrefix, formattingPrependOffset); 103 } 104 m_styleText.insert(textToSet, propertyStart); 105 } 106 107 void InspectorStyleTextEditor::replaceProperty(unsigned index, const String& newText) 108 { 109 ASSERT_WITH_SECURITY_IMPLICATION(index < m_allProperties->size()); 110 internalReplaceProperty(m_allProperties->at(index), newText); 111 } 112 113 void InspectorStyleTextEditor::removeProperty(unsigned index) 114 { 115 replaceProperty(index, ""); 116 } 117 118 void InspectorStyleTextEditor::enableProperty(unsigned index) 119 { 120 InspectorStyleProperty& disabledProperty = m_allProperties->at(index); 121 ASSERT(disabledProperty.sourceData.disabled); 122 internalReplaceProperty(disabledProperty, disabledProperty.rawText.substring(2, disabledProperty.rawText.length() - 4).stripWhiteSpace()); 123 } 124 125 void InspectorStyleTextEditor::disableProperty(unsigned index) 126 { 127 ASSERT(!m_allProperties->at(index).sourceData.disabled); 128 129 InspectorStyleProperty& property = m_allProperties->at(index); 130 property.setRawTextFromStyleDeclaration(m_styleText); 131 property.sourceData.disabled = true; 132 133 internalReplaceProperty(property, "/* " + property.rawText + " */"); 134 } 135 136 void InspectorStyleTextEditor::internalReplaceProperty(const InspectorStyleProperty& property, const String& newText) 137 { 138 const SourceRange& range = property.sourceData.range; 139 long replaceRangeStart = range.start; 140 long replaceRangeEnd = range.end; 141 long newTextLength = newText.length(); 142 String finalNewText = newText; 143 144 // Removing a property - remove preceding prefix. 145 String fullPrefix = m_format.first + m_format.second; 146 long fullPrefixLength = fullPrefix.length(); 147 if (!newTextLength && fullPrefixLength) { 148 if (replaceRangeStart >= fullPrefixLength && m_styleText.substring(replaceRangeStart - fullPrefixLength, fullPrefixLength) == fullPrefix) 149 replaceRangeStart -= fullPrefixLength; 150 } else if (newTextLength) { 151 if (isHTMLLineBreak(newText[newTextLength - 1])) { 152 // Coalesce newlines of the original and new property values (to avoid a lot of blank lines while incrementally applying property values). 153 bool foundNewline = false; 154 bool isLastNewline = false; 155 int i; 156 int textLength = m_styleText.length(); 157 for (i = replaceRangeEnd; i < textLength && isSpaceOrNewline(m_styleText[i]); ++i) { 158 isLastNewline = isHTMLLineBreak(m_styleText[i]); 159 if (isLastNewline) 160 foundNewline = true; 161 else if (foundNewline && !isLastNewline) { 162 replaceRangeEnd = i; 163 break; 164 } 165 } 166 if (foundNewline && isLastNewline) 167 replaceRangeEnd = i; 168 } 169 170 if (fullPrefixLength > replaceRangeStart || m_styleText.substring(replaceRangeStart - fullPrefixLength, fullPrefixLength) != fullPrefix) 171 finalNewText.insert(fullPrefix, 0); 172 } 173 174 int replacedLength = replaceRangeEnd - replaceRangeStart; 175 m_styleText.replace(replaceRangeStart, replacedLength, finalNewText); 176 } 177 178 } // namespace WebCore 179 180