diff options
author | Lorry Tar Creator <lorry-tar-importer@lorry> | 2017-06-27 06:07:23 +0000 |
---|---|---|
committer | Lorry Tar Creator <lorry-tar-importer@lorry> | 2017-06-27 06:07:23 +0000 |
commit | 1bf1084f2b10c3b47fd1a588d85d21ed0eb41d0c (patch) | |
tree | 46dcd36c86e7fbc6e5df36deb463b33e9967a6f7 /Source/WebInspectorUI/UserInterface/Models/CSSCompletions.js | |
parent | 32761a6cee1d0dee366b885b7b9c777e67885688 (diff) | |
download | WebKitGtk-tarball-master.tar.gz |
webkitgtk-2.16.5HEADwebkitgtk-2.16.5master
Diffstat (limited to 'Source/WebInspectorUI/UserInterface/Models/CSSCompletions.js')
-rw-r--r-- | Source/WebInspectorUI/UserInterface/Models/CSSCompletions.js | 298 |
1 files changed, 298 insertions, 0 deletions
diff --git a/Source/WebInspectorUI/UserInterface/Models/CSSCompletions.js b/Source/WebInspectorUI/UserInterface/Models/CSSCompletions.js new file mode 100644 index 000000000..8cab966b6 --- /dev/null +++ b/Source/WebInspectorUI/UserInterface/Models/CSSCompletions.js @@ -0,0 +1,298 @@ +/* + * Copyright (C) 2010 Nikita Vasilyev. All rights reserved. + * Copyright (C) 2010 Joseph Pecoraro. All rights reserved. + * Copyright (C) 2010 Google Inc. All rights reserved. + * Copyright (C) 2013 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: + * + * * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following disclaimer + * in the documentation and/or other materials provided with the + * distribution. + * * Neither the name of Google Inc. nor the names of its + * contributors may be used to endorse or promote products derived from + * this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR + * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT + * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, + * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT + * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, + * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY + * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT + * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE + * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + */ + +WebInspector.CSSCompletions = class CSSCompletions +{ + constructor(properties, acceptEmptyPrefix) + { + this._values = []; + this._longhands = {}; + this._shorthands = {}; + + // The `properties` parameter can be either a list of objects with 'name' / 'longhand' + // properties when initialized from the protocol for CSSCompletions.cssNameCompletions. + // Or it may just a list of strings when quickly initialized for other completion purposes. + if (properties.length && typeof properties[0] === "string") + this._values = this._values.concat(properties); + else { + for (var property of properties) { + var propertyName = property.name; + console.assert(propertyName); + + this._values.push(propertyName); + + var longhands = property.longhands; + if (longhands) { + this._longhands[propertyName] = longhands; + + for (var j = 0; j < longhands.length; ++j) { + var longhandName = longhands[j]; + + var shorthands = this._shorthands[longhandName]; + if (!shorthands) { + shorthands = []; + this._shorthands[longhandName] = shorthands; + } + + shorthands.push(propertyName); + } + } + } + } + + this._values.sort(); + + this._acceptEmptyPrefix = acceptEmptyPrefix; + } + + // Static + + static requestCSSCompletions() + { + if (WebInspector.CSSCompletions.cssNameCompletions) + return; + + function propertyNamesCallback(error, names) + { + if (error) + return; + + WebInspector.CSSCompletions.cssNameCompletions = new WebInspector.CSSCompletions(names, false); + + WebInspector.CSSKeywordCompletions.addCustomCompletions(names); + + // CodeMirror is not included by tests so we shouldn't assume it always exists. + // If it isn't available we skip MIME type associations. + if (!window.CodeMirror) + return; + + var propertyNamesForCodeMirror = {}; + var valueKeywordsForCodeMirror = {"inherit": true, "initial": true, "unset": true, "revert": true, "var": true}; + var colorKeywordsForCodeMirror = {}; + + function nameForCodeMirror(name) + { + // CodeMirror parses the vendor prefix separate from the property or keyword name, + // so we need to strip vendor prefixes from our names. Also strip function parenthesis. + return name.replace(/^-[^-]+-/, "").replace(/\(\)$/, "").toLowerCase(); + } + + function collectPropertyNameForCodeMirror(propertyName) + { + // Properties can also be value keywords, like when used in a transition. + // So we add them to both lists. + var codeMirrorPropertyName = nameForCodeMirror(propertyName); + propertyNamesForCodeMirror[codeMirrorPropertyName] = true; + valueKeywordsForCodeMirror[codeMirrorPropertyName] = true; + } + + for (var property of names) + collectPropertyNameForCodeMirror(property.name); + + for (var propertyName in WebInspector.CSSKeywordCompletions._propertyKeywordMap) { + var keywords = WebInspector.CSSKeywordCompletions._propertyKeywordMap[propertyName]; + for (var i = 0; i < keywords.length; ++i) { + // Skip numbers, like the ones defined for font-weight. + if (!isNaN(Number(keywords[i]))) + continue; + valueKeywordsForCodeMirror[nameForCodeMirror(keywords[i])] = true; + } + } + + WebInspector.CSSKeywordCompletions._colors.forEach(function(colorName) { + colorKeywordsForCodeMirror[nameForCodeMirror(colorName)] = true; + }); + + function updateCodeMirrorCSSMode(mimeType) + { + var modeSpec = CodeMirror.resolveMode(mimeType); + + console.assert(modeSpec.propertyKeywords); + console.assert(modeSpec.valueKeywords); + console.assert(modeSpec.colorKeywords); + + modeSpec.propertyKeywords = propertyNamesForCodeMirror; + modeSpec.valueKeywords = valueKeywordsForCodeMirror; + modeSpec.colorKeywords = colorKeywordsForCodeMirror; + + CodeMirror.defineMIME(mimeType, modeSpec); + } + + updateCodeMirrorCSSMode("text/css"); + updateCodeMirrorCSSMode("text/x-scss"); + } + + function fontFamilyNamesCallback(error, fontFamilyNames) + { + if (error) + return; + + WebInspector.CSSKeywordCompletions.addPropertyCompletionValues("font-family", fontFamilyNames); + WebInspector.CSSKeywordCompletions.addPropertyCompletionValues("font", fontFamilyNames); + } + + if (window.CSSAgent) { + CSSAgent.getSupportedCSSProperties(propertyNamesCallback); + + // COMPATIBILITY (iOS 9): CSS.getSupportedSystemFontFamilyNames did not exist. + if (CSSAgent.getSupportedSystemFontFamilyNames) + CSSAgent.getSupportedSystemFontFamilyNames(fontFamilyNamesCallback); + } + } + + // Public + + get values() + { + return this._values; + } + + startsWith(prefix) + { + var firstIndex = this._firstIndexOfPrefix(prefix); + if (firstIndex === -1) + return []; + + var results = []; + while (firstIndex < this._values.length && this._values[firstIndex].startsWith(prefix)) + results.push(this._values[firstIndex++]); + return results; + } + + _firstIndexOfPrefix(prefix) + { + if (!this._values.length) + return -1; + if (!prefix) + return this._acceptEmptyPrefix ? 0 : -1; + + var maxIndex = this._values.length - 1; + var minIndex = 0; + var foundIndex; + + do { + var middleIndex = (maxIndex + minIndex) >> 1; + if (this._values[middleIndex].startsWith(prefix)) { + foundIndex = middleIndex; + break; + } + if (this._values[middleIndex] < prefix) + minIndex = middleIndex + 1; + else + maxIndex = middleIndex - 1; + } while (minIndex <= maxIndex); + + if (foundIndex === undefined) + return -1; + + while (foundIndex && this._values[foundIndex - 1].startsWith(prefix)) + foundIndex--; + + return foundIndex; + } + + keySet() + { + if (!this._keySet) + this._keySet = this._values.keySet(); + return this._keySet; + } + + next(str, prefix) + { + return this._closest(str, prefix, 1); + } + + previous(str, prefix) + { + return this._closest(str, prefix, -1); + } + + _closest(str, prefix, shift) + { + if (!str) + return ""; + + var index = this._values.indexOf(str); + if (index === -1) + return ""; + + if (!prefix) { + index = (index + this._values.length + shift) % this._values.length; + return this._values[index]; + } + + var propertiesWithPrefix = this.startsWith(prefix); + var j = propertiesWithPrefix.indexOf(str); + j = (j + propertiesWithPrefix.length + shift) % propertiesWithPrefix.length; + return propertiesWithPrefix[j]; + } + + isShorthandPropertyName(shorthand) + { + return shorthand in this._longhands; + } + + shorthandsForLonghand(longhand) + { + return this._shorthands[longhand] || []; + } + + isValidPropertyName(name) + { + return this._values.includes(name); + } + + propertyRequiresWebkitPrefix(name) + { + return this._values.includes("-webkit-" + name) && !this._values.includes(name); + } + + getClosestPropertyName(name) + { + var bestMatches = [{distance: Infinity, name: null}]; + + for (var property of this._values) { + var distance = name.levenshteinDistance(property); + + if (distance < bestMatches[0].distance) + bestMatches = [{distance, name: property}]; + else if (distance === bestMatches[0].distance) + bestMatches.push({distance, name: property}); + } + + return bestMatches.length < 3 ? bestMatches[0].name : false; + } +}; + +WebInspector.CSSCompletions.cssNameCompletions = null; |