fix/rewrite/integrate colorpicker
* full rewrite of colorview.js to make it 10-100 times faster (render on demand via extendMode) * full rewrite of colorpicker.js to simplify CSS * automatic light/dark theme based on current color of the editor * fixes, tweaks, speedups * color spot will always be on the left of its text i.e. no line break on wrapping * support #RRGGBBAA and #RGBA hex colors * support "transparent" as rgba(0, 0, 0, 0) * HEX/hex toggle * fix HSLA regexp * Esc/Enter key to close * innerHTML -> textContent * toggle the feature correctly * fade out before autohiding * always show alpha 1 like devtools does * set cursor:pointer only on the clickable part of the sliders * bigger color format switcher with a tooltip * autofocus input on open, disable spellcheck * try not to obscure the source color spot & text * restore focus without scrolling
This commit is contained in:
parent
c50b251ae4
commit
37a62b0407
|
@ -1,2 +1,3 @@
|
|||
vendor/
|
||||
vendor-overwrites/
|
||||
vendor-overwrites/*
|
||||
!vendor-overwrites/colorpicker
|
||||
|
|
|
@ -127,6 +127,10 @@
|
|||
"message": "Autocomplete on typing",
|
||||
"description": "Label for the checkbox in the style editor."
|
||||
},
|
||||
"cm_colorpicker": {
|
||||
"message": "Colorpickers for CSS colors",
|
||||
"description": "Label for the checkbox controlling colorpicker option for the style editor."
|
||||
},
|
||||
"cm_indentWithTabs": {
|
||||
"message": "Use tabs with smart indentation",
|
||||
"description": "Label for the checkbox controlling tabs with smart indentation option for the style editor."
|
||||
|
@ -171,6 +175,14 @@
|
|||
"message": "Theme",
|
||||
"description": "Label for the style editor's CSS theme."
|
||||
},
|
||||
"colorpickerSwitchFormatTooltip": {
|
||||
"message": "Switch formats: HEX -> RGB -> HSL",
|
||||
"description": "Tooltip for the switch button in the color picker popup in the style editor."
|
||||
},
|
||||
"colorpickerTooltip": {
|
||||
"message": "Open color picker",
|
||||
"description": "Tooltip for the colored squares shown before CSS colors in the style editor."
|
||||
},
|
||||
"dysfunctional": {
|
||||
"message": "Stylus cannot function in private windows because Firefox disallows direct connection to the internal background page context of the extension.",
|
||||
"description": "Displayed in Firefox when its settings make Stylus dysfunctional"
|
||||
|
|
|
@ -190,6 +190,10 @@
|
|||
<input id="editor.autocompleteOnTyping" type="checkbox">
|
||||
<label for="editor.autocompleteOnTyping" i18n-text="cm_autocompleteOnTyping"></label>
|
||||
</div>
|
||||
<div class="option">
|
||||
<input id="editor.colorpicker" type="checkbox">
|
||||
<label for="editor.colorpicker" i18n-text="cm_colorpicker"></label>
|
||||
</div>
|
||||
<div class="option aligned">
|
||||
<label id="tabSize-label" for="editor.tabSize" i18n-text="cm_tabSize"></label>
|
||||
<input id="editor.tabSize" type="number" min="0">
|
||||
|
|
44
edit/edit.js
44
edit/edit.js
|
@ -7,6 +7,12 @@
|
|||
/* global closeCurrentTab regExpTester messageBox */
|
||||
'use strict';
|
||||
|
||||
onDOMready()
|
||||
.then(() => Promise.all([
|
||||
onColorpickerReady(),
|
||||
]))
|
||||
.then(init);
|
||||
|
||||
let styleId = null;
|
||||
// only the actually dirty items here
|
||||
let dirty = {};
|
||||
|
@ -362,6 +368,8 @@ function acmeEventListener(event) {
|
|||
}
|
||||
option = 'highlightSelectionMatches';
|
||||
break;
|
||||
case 'colorpicker':
|
||||
return;
|
||||
}
|
||||
CodeMirror.setOption(option, value);
|
||||
}
|
||||
|
@ -1298,8 +1306,6 @@ function beautify(event) {
|
|||
}
|
||||
}
|
||||
|
||||
onDOMready().then(init);
|
||||
|
||||
function init() {
|
||||
initCodeMirror();
|
||||
getStyle().then(style => {
|
||||
|
@ -2065,3 +2071,37 @@ function setGlobalProgress(done, total) {
|
|||
progressElement.remove();
|
||||
}
|
||||
}
|
||||
|
||||
function onColorpickerReady() {
|
||||
const scripts = [
|
||||
'/vendor-overwrites/colorpicker/colorpicker.css',
|
||||
'/vendor-overwrites/colorpicker/colorpicker.js',
|
||||
'/vendor-overwrites/colorpicker/colorview.js',
|
||||
];
|
||||
prefs.subscribe(['editor.colorpicker'], colorpickerOnDemand);
|
||||
return prefs.get('editor.colorpicker') && colorpickerOnDemand(null, true);
|
||||
|
||||
function colorpickerOnDemand(id, enabled) {
|
||||
return loadScript(enabled && scripts)
|
||||
.then(() => setColorpickerOption(id, enabled));
|
||||
}
|
||||
|
||||
function setColorpickerOption(id, enabled) {
|
||||
CodeMirror.defaults.colorpicker = enabled && {
|
||||
forceUpdate: editors.length > 0,
|
||||
tooltip: t('colorpickerTooltip'),
|
||||
popupOptions: {
|
||||
tooltipForSwitcher: t('colorpickerSwitchFormatTooltip'),
|
||||
hexUppercase: prefs.get('editor.colorpicker.hexUppercase'),
|
||||
hideDelay: 5000,
|
||||
embedderCallback: state => {
|
||||
if (state && state.hexUppercase !== prefs.get('editor.colorpicker.hexUppercase')) {
|
||||
prefs.set('editor.colorpicker.hexUppercase', state.hexUppercase);
|
||||
}
|
||||
},
|
||||
},
|
||||
};
|
||||
// on page load runs before CodeMirror.setOption is defined
|
||||
editors.forEach(cm => cm.setOption('colorpicker', CodeMirror.defaults.colorpicker));
|
||||
}
|
||||
}
|
||||
|
|
|
@ -56,6 +56,11 @@ var prefs = new function Prefs() {
|
|||
|
||||
'editor.appliesToLineWidget': true, // show applies-to line widget on the editor
|
||||
|
||||
// show CSS colors as clickable colored rectangles
|
||||
'editor.colorpicker': true,
|
||||
// #DEAD or #beef
|
||||
'editor.colorpicker.hexUppercase': false,
|
||||
|
||||
'iconset': 0, // 0 = dark-themed icon
|
||||
// 1 = light-themed icon
|
||||
|
||||
|
|
|
@ -1,57 +1,151 @@
|
|||
|
||||
/* codemirror colorview */
|
||||
|
||||
.codemirror-colorview {
|
||||
border : 1px solid #8e8e8e;
|
||||
position: relative;
|
||||
display : inline-block;
|
||||
box-sizing : border-box;
|
||||
margin : 0px 2px;
|
||||
width : 10px;
|
||||
height : 10px;
|
||||
cursor: pointer;
|
||||
background-image : url("");
|
||||
background-repeat: repeat;
|
||||
}
|
||||
|
||||
.codemirror-colorview .codemirror-colorview-background {
|
||||
content: "";
|
||||
position: absolute;
|
||||
left:0px;
|
||||
right:0px;
|
||||
bottom:0px;
|
||||
top:0px;
|
||||
}
|
||||
|
||||
.codemirror-colorview:hover {
|
||||
border-color: #494949;
|
||||
}
|
||||
|
||||
|
||||
/* codemirror-colorpicker */
|
||||
|
||||
.codemirror-colorpicker {
|
||||
.cm-colorview {
|
||||
position: relative;
|
||||
width: 226px;
|
||||
z-index: 1000;
|
||||
white-space: nowrap;
|
||||
}
|
||||
.codemirror-colorpicker > .color {
|
||||
|
||||
.cm-colorview::before {
|
||||
content: "";
|
||||
position: relative;
|
||||
display: inline-block;
|
||||
box-sizing: content-box;
|
||||
margin: 0 3px;
|
||||
width: 8px;
|
||||
height: 8px;
|
||||
background-image: url("");
|
||||
background-repeat: repeat;
|
||||
}
|
||||
|
||||
.cm-colorview + .cm-colorview.cm-overlay::before,
|
||||
.cm-colorview.cm-overlay + .cm-colorview::before {
|
||||
content: none;
|
||||
}
|
||||
|
||||
.codemirror-colorview-background {
|
||||
position: absolute;
|
||||
left: 2px;
|
||||
top: 2px;
|
||||
width: 10px;
|
||||
height: 10px;
|
||||
box-sizing: border-box;
|
||||
border: 1px solid #8e8e8e;
|
||||
content: "";
|
||||
cursor: pointer;
|
||||
}
|
||||
|
||||
.codemirror-colorview-background:hover {
|
||||
border-color: #494949;
|
||||
}
|
||||
|
||||
/* colorpicker */
|
||||
|
||||
.colorpicker-theme-light {
|
||||
--main-background-color: #fff;
|
||||
--main-border-color: #ccc;
|
||||
|
||||
--label-color: #666;
|
||||
--label-color-hover: #000;
|
||||
|
||||
--input-background-color: #fff;
|
||||
--input-background-color-hover: #ddd;
|
||||
--input-background-color-focus: #fff;
|
||||
|
||||
--input-color: #444;
|
||||
--input-color-focus: #000;
|
||||
|
||||
--input-border-color: #bbb;
|
||||
--input-border-color-focus: #888;
|
||||
--input-border-color-hover: #444;
|
||||
|
||||
--invalid-border-color: hsl(0, 100%, 50%);
|
||||
--invalid-background-color: hsla(0, 100%, 50%, 0.15);
|
||||
--invalid-color: hsl(0, 100%, 40%);
|
||||
}
|
||||
|
||||
.colorpicker-theme-dark {
|
||||
--main-background-color: #242424;
|
||||
--main-border-color: #888;
|
||||
|
||||
--label-color: #aaa;
|
||||
--label-color-hover: #eee;
|
||||
|
||||
--input-background-color: #222;
|
||||
--input-background-color-hover: #222;
|
||||
--input-background-color-focus: #383838;
|
||||
|
||||
--input-color: #ddd;
|
||||
--input-color-focus: #fff;
|
||||
|
||||
--input-border-color: #505050;
|
||||
--input-border-color-focus: #777;
|
||||
--input-border-color-hover: #888;
|
||||
|
||||
--invalid-border-color: hsl(0, 100%, 27%);
|
||||
--invalid-background-color: hsla(0, 100%, 50%, 0.3);
|
||||
--invalid-color: hsl(0, 100%, 75%);
|
||||
}
|
||||
|
||||
.colorpicker-popup {
|
||||
--switcher-width: 30px;
|
||||
position: relative;
|
||||
width: 350px;
|
||||
z-index: 1000;
|
||||
transition: opacity .5s;
|
||||
color: var(--label-color);
|
||||
border: 1px solid var(--main-border-color);
|
||||
background-color: var(--main-background-color);
|
||||
box-shadow: 0 0 10px 0 rgba(0, 0, 0, 0.12);
|
||||
-webkit-user-select: none;
|
||||
-moz-user-select: none;
|
||||
-ms-user-select: none;
|
||||
-o-user-select: none;
|
||||
user-select: none;
|
||||
}
|
||||
|
||||
.colorpicker-popup[data-fading="1"] {
|
||||
opacity: .75;
|
||||
}
|
||||
|
||||
.colorpicker-popup[data-fading="2"] {
|
||||
opacity: 0;
|
||||
}
|
||||
|
||||
.colorpicker-saturation-container {
|
||||
position: relative;
|
||||
height: 120px;
|
||||
overflow: hidden;
|
||||
cursor: pointer;
|
||||
}
|
||||
.codemirror-colorpicker > .color > .saturation {
|
||||
|
||||
.colorpicker-opacity-bar {
|
||||
position: absolute;
|
||||
display: block;
|
||||
content: "";
|
||||
left: 0;
|
||||
right: 0;
|
||||
bottom: 0;
|
||||
top: 0;
|
||||
background: linear-gradient(to right, rgba(232, 232, 232, 0), rgba(232, 232, 232, 1));
|
||||
}
|
||||
|
||||
.colorpicker-saturation {
|
||||
position: relative;
|
||||
width: 100%;
|
||||
height: 100%;
|
||||
background-color: rgba(204, 154, 129, 0);
|
||||
background-image: linear-gradient(to right, #FFF, rgba(204, 154, 129, 0));
|
||||
background-repeat: repeat-x;
|
||||
}
|
||||
.codemirror-colorpicker > .color > .saturation > .value {
|
||||
|
||||
.colorpicker-value {
|
||||
position: relative;
|
||||
width: 100%;
|
||||
height: 100%;
|
||||
background-image: linear-gradient(to top, #000, rgba(204, 154, 129, 0));
|
||||
}
|
||||
.codemirror-colorpicker > .color > .saturation > .value > .drag-pointer {
|
||||
|
||||
.colorpicker-drag-pointer {
|
||||
position: absolute;
|
||||
width: 10px;
|
||||
height: 10px;
|
||||
|
@ -60,21 +154,25 @@
|
|||
border-radius: 50%;
|
||||
left: -5px;
|
||||
top: -5px;
|
||||
border: 1px solid #fff;
|
||||
box-shadow: 0 0 2px 0 rgba(0, 0, 0, 0.05);
|
||||
}
|
||||
.codemirror-colorpicker > .control {
|
||||
|
||||
.colorpicker-sliders {
|
||||
position: relative;
|
||||
padding: 18px 0px 14px 0px;
|
||||
-webkit-user-select: none;
|
||||
-moz-user-select: none;
|
||||
-ms-user-select: none;
|
||||
-o-user-select: none;
|
||||
user-select: none;
|
||||
padding: 10px 0 6px 0;
|
||||
border-top: 1px solid transparent;
|
||||
}
|
||||
.codemirror-colorpicker > .control > .color,
|
||||
.codemirror-colorpicker > .control > .empty {
|
||||
|
||||
.colorpicker-theme-dark .colorpicker-sliders {
|
||||
border-color: var(--input-border-color);
|
||||
}
|
||||
|
||||
.colorpicker-swatch,
|
||||
.colorpicker-empty {
|
||||
position: absolute;
|
||||
left: 11px;
|
||||
top: 24px;
|
||||
top: 17px;
|
||||
width: 30px;
|
||||
height: 30px;
|
||||
-webkit-border-radius: 50%;
|
||||
|
@ -84,33 +182,41 @@
|
|||
-moz-box-sizing: border-box;
|
||||
box-sizing: border-box;
|
||||
}
|
||||
.codemirror-colorpicker > .control > .hue {
|
||||
|
||||
.colorpicker-empty {
|
||||
background: url("") repeat;
|
||||
}
|
||||
|
||||
.colorpicker-hue {
|
||||
position: relative;
|
||||
padding: 6px 12px;
|
||||
margin: 0px 0px 0px 45px;
|
||||
margin: 0 0 0 45px;
|
||||
-webkit-box-sizing: border-box;
|
||||
-moz-box-sizing: border-box;
|
||||
box-sizing: border-box;
|
||||
cursor: pointer;
|
||||
}
|
||||
.codemirror-colorpicker > .control > .hue > .hue-container {
|
||||
|
||||
.colorpicker-hue-container {
|
||||
position: relative;
|
||||
width: 100%;
|
||||
height: 10px;
|
||||
-webkit-border-radius: 3px;
|
||||
-moz-border-radius: 3px;
|
||||
border-radius: 3px;
|
||||
cursor: pointer;
|
||||
background: linear-gradient(to right, #ff0000 0%, #ffff00 17%, #00ff00 33%, #00ffff 50%, #0000ff 67%, #ff00ff 83%, #ff0000 100%);
|
||||
}
|
||||
.codemirror-colorpicker > .control > .opacity {
|
||||
|
||||
.colorpicker-opacity {
|
||||
position: relative;
|
||||
padding: 3px 12px;
|
||||
margin: 0px 0px 0px 45px;
|
||||
margin: 0 0 0 45px;
|
||||
-webkit-box-sizing: border-box;
|
||||
-moz-box-sizing: border-box;
|
||||
box-sizing: border-box;
|
||||
cursor: pointer;
|
||||
}
|
||||
.codemirror-colorpicker > .control > .opacity > .opacity-container {
|
||||
|
||||
.colorpicker-opacity-container {
|
||||
position: relative;
|
||||
width: 100%;
|
||||
height: 10px;
|
||||
|
@ -118,10 +224,13 @@
|
|||
-webkit-border-radius: 3px;
|
||||
-moz-border-radius: 3px;
|
||||
border-radius: 3px;
|
||||
cursor: pointer;
|
||||
background-image: url("");
|
||||
background-repeat: repeat;
|
||||
}
|
||||
|
||||
.codemirror-colorpicker > .control .drag-bar,
|
||||
.codemirror-colorpicker > .control .drag-bar2 {
|
||||
.colorpicker-hue-knob,
|
||||
.colorpicker-opacity-knob {
|
||||
position: absolute;
|
||||
cursor: pointer;
|
||||
top: 50% !important;
|
||||
|
@ -132,179 +241,148 @@
|
|||
-webkit-border-radius: 50px;
|
||||
-moz-border-radius: 50px;
|
||||
border-radius: 50px;
|
||||
border: 1px solid rgba(0, 0, 0, 0.5);
|
||||
box-shadow: 0 0 2px 0 rgba(0, 0, 0, 0.1);
|
||||
background-color: #fff;
|
||||
}
|
||||
.codemirror-colorpicker > .information {
|
||||
|
||||
.colorpicker-input-container {
|
||||
position: relative;
|
||||
-webkit-box-sizing: padding-box;
|
||||
-moz-box-sizing: padding-box;
|
||||
box-sizing: padding-box;
|
||||
}
|
||||
|
||||
.codemirror-colorpicker > .information.hex > .information-item.hex {
|
||||
display: flex;
|
||||
}
|
||||
|
||||
.codemirror-colorpicker > .information.rgb > .information-item.rgb {
|
||||
display: flex;
|
||||
}
|
||||
|
||||
.codemirror-colorpicker > .information.hsl > .information-item.hsl {
|
||||
display: flex;
|
||||
}
|
||||
|
||||
.codemirror-colorpicker > .information > .information-item {
|
||||
display:none;
|
||||
.colorpicker-input-group {
|
||||
display: none;
|
||||
position: relative;
|
||||
padding: 0px 5px;
|
||||
padding: 0 5px;
|
||||
-webkit-box-sizing: border-box;
|
||||
-moz-box-sizing: border-box;
|
||||
box-sizing: border-box;
|
||||
margin-right:40px;
|
||||
margin-right: calc(var(--switcher-width) - 10px);
|
||||
}
|
||||
|
||||
.codemirror-colorpicker > .information > .information-change {
|
||||
position:absolute;
|
||||
display:block;
|
||||
width:40px;
|
||||
top:0px;
|
||||
right:0px;
|
||||
bottom:0px;
|
||||
.colorpicker-input-group[data-active] {
|
||||
display: flex;
|
||||
}
|
||||
|
||||
.codemirror-colorpicker > .information > .information-change > .format-change-button {
|
||||
width:100%;
|
||||
height:100%;
|
||||
background:transparent;
|
||||
border:0px;
|
||||
cursor:pointer;
|
||||
outline:none;
|
||||
}
|
||||
|
||||
.codemirror-colorpicker > .information > .information-item > .input-field {
|
||||
display:block;
|
||||
flex:1;
|
||||
.colorpicker-input-field {
|
||||
display: block;
|
||||
position: relative;
|
||||
flex: 1;
|
||||
padding: 5px;
|
||||
-webkit-box-sizing: border-box;
|
||||
-moz-box-sizing: border-box;
|
||||
box-sizing: border-box;
|
||||
}
|
||||
|
||||
.codemirror-colorpicker > .information > .information-item > .input-field > input {
|
||||
text-align: center;
|
||||
width:100%;
|
||||
padding:3px 5px;
|
||||
font-size:11px;
|
||||
color: #333;
|
||||
-webkit-box-sizing: border-box;
|
||||
-moz-box-sizing: border-box;
|
||||
box-sizing: border-box;
|
||||
-webkit-user-select: text;
|
||||
-moz-user-select: text;
|
||||
-ms-user-select: text;
|
||||
-o-user-select: text;
|
||||
user-select: text;
|
||||
border: 1px solid #cbcbcb;
|
||||
.colorpicker-input-field[class$="-a"] {
|
||||
flex-grow: 1.5;
|
||||
}
|
||||
|
||||
.codemirror-colorpicker > .information > .information-item > .input-field > .title {
|
||||
text-align:center;
|
||||
font-size:12px;
|
||||
color:#a9a9a9;
|
||||
}
|
||||
|
||||
.codemirror-colorpicker > .information > input {
|
||||
.colorpicker-hsl-h::before {
|
||||
content: "\b0"; /* degree */
|
||||
position: absolute;
|
||||
right: -2px;
|
||||
top: 8px;
|
||||
}
|
||||
|
||||
.colorpicker-hsl-s::before,
|
||||
.colorpicker-hsl-l::before {
|
||||
content: "%";
|
||||
position: absolute;
|
||||
right: -1ex;
|
||||
top: 8px;
|
||||
font-size: 10px;
|
||||
height: 20px;
|
||||
bottom: 20px;
|
||||
padding: 0 0 0 2px;
|
||||
-webkit-box-sizing: border-box;
|
||||
-moz-box-sizing: border-box;
|
||||
}
|
||||
|
||||
.colorpicker-input {
|
||||
text-align: center;
|
||||
width: 100%;
|
||||
padding: 3px 5px;
|
||||
font-size: 11px;
|
||||
font-weight: bold;
|
||||
box-sizing: border-box;
|
||||
-webkit-user-select: text;
|
||||
-moz-user-select: text;
|
||||
-ms-user-select: text;
|
||||
-o-user-select: text;
|
||||
user-select: text;
|
||||
border: 1px solid var(--input-border-color);
|
||||
background-color: var(--input-background-color);
|
||||
color: var(--input-color);
|
||||
}
|
||||
|
||||
|
||||
.codemirror-colorpicker {
|
||||
border: 1px solid #ececec;
|
||||
background-color: #fff;
|
||||
-webkit-box-shadow: 0 0px 10px 0px rgba(0, 0, 0, 0.12);
|
||||
-moz-box-shadow: 0 0px 10px 0px rgba(0, 0, 0, 0.12);
|
||||
box-shadow: 0 0px 10px 0px rgba(0, 0, 0, 0.12);
|
||||
}
|
||||
.codemirror-colorpicker > .color > .saturation {
|
||||
background-color: rgba(204, 154, 129, 0);
|
||||
background-image: -moz-linear-gradient(left, #FFF, rgba(204, 154, 129, 0));
|
||||
background-image: -webkit-gradient(linear, 0 0, 100% 0, from(#FFF), to(rgba(204, 154, 129, 0)));
|
||||
background-image: -webkit-linear-gradient(left, #FFF, rgba(204, 154, 129, 0));
|
||||
background-image: -o-linear-gradient(left, #FFF, rgba(204, 154, 129, 0));
|
||||
background-image: linear-gradient(to right, #FFF, rgba(204, 154, 129, 0));
|
||||
background-repeat: repeat-x;
|
||||
filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#ffffffff', endColorstr='#00cc9a81', GradientType=1);
|
||||
}
|
||||
.codemirror-colorpicker > .color > .saturation > .value {
|
||||
background-image: -webkit-gradient(linear, 0 100%, 0 0, from(#000000), to(rgba(204, 154, 129, 0)));
|
||||
background-image: -webkit-linear-gradient(bottom, #000000, rgba(204, 154, 129, 0));
|
||||
background-image: -moz-linear-gradient(bottom, #000000, rgba(204, 154, 129, 0));
|
||||
background-image: -o-linear-gradient(bottom, #000000, rgba(204, 154, 129, 0));
|
||||
background-image: -ms-linear-gradient(bottom, #000000, rgba(204, 154, 129, 0));
|
||||
background-image: linear-gradient(to top, #000000, rgba(204, 154, 129, 0));
|
||||
}
|
||||
.codemirror-colorpicker > .color > .saturation > .value > .drag-pointer {
|
||||
border: 1px solid #fff;
|
||||
-webkit-box-shadow: 0 0 2px 0 rgba(0, 0, 0, 0.05);
|
||||
-moz-box-shadow: 0 0 2px 0 rgba(0, 0, 0, 0.05);
|
||||
box-shadow: 0 0 2px 0 rgba(0, 0, 0, 0.05);
|
||||
}
|
||||
.codemirror-colorpicker > .control > .hue > .hue-container {
|
||||
background: -moz-linear-gradient(left, #ff0000 0%, #ffff00 17%, #00ff00 33%, #00ffff 50%, #0000ff 67%, #ff00ff 83%, #ff0000 100%);
|
||||
background: -ms-linear-gradient(left, #ff0000 0%, #ffff00 17%, #00ff00 33%, #00ffff 50%, #0000ff 67%, #ff00ff 83%, #ff0000 100%);
|
||||
background: -o-linear-gradient(left, #ff0000 0%, #ffff00 17%, #00ff00 33%, #00ffff 50%, #0000ff 67%, #ff00ff 83%, #ff0000 100%);
|
||||
background: -webkit-gradient(linear, left top, right top, from(#ff0000), color-stop(0.17, #ffff00), color-stop(0.33, #00ff00), color-stop(0.5, #00ffff), color-stop(0.67, #0000ff), color-stop(0.83, #ff00ff), to(#ff0000));
|
||||
background: -webkit-linear-gradient(left, #ff0000 0%, #ffff00 17%, #00ff00 33%, #00ffff 50%, #0000ff 67%, #ff00ff 83%, #ff0000 100%);
|
||||
background: linear-gradient(to right, #ff0000 0%, #ffff00 17%, #00ff00 33%, #00ffff 50%, #0000ff 67%, #ff00ff 83%, #ff0000 100%);
|
||||
}
|
||||
.codemirror-colorpicker > .control > .opacity > .opacity-container > .color-bar {
|
||||
position:absolute;
|
||||
display:block;
|
||||
content:"";
|
||||
left:0px;
|
||||
right:0px;
|
||||
bottom:0px;
|
||||
top:0px;
|
||||
background: -moz-linear-gradient(left, rgba(232, 232, 232, 0), rgba(232, 232, 232, 1));
|
||||
background: -ms-linear-gradient(left, rgba(232, 232, 232, 0), rgba(232, 232, 232, 1));
|
||||
background: -o-linear-gradient(left, rgba(232, 232, 232, 0), rgba(232, 232, 232, 1));
|
||||
background: -webkit-gradient(linear, left top, right top, from(rgba(232, 232, 232, 0)), to(rgba(232, 232, 232, 1)));
|
||||
background: -webkit-linear-gradient(left, rgba(232, 232, 232, 0), rgba(232, 232, 232, 1));
|
||||
background: linear-gradient(to right, rgba(232, 232, 232, 0) , rgba(232, 232, 232, 1) );
|
||||
}
|
||||
.codemirror-colorpicker > .control > .opacity > .opacity-container {
|
||||
background-image : url("");
|
||||
background-repeat: repeat;
|
||||
.colorpicker-theme-dark .colorpicker-input::-webkit-inner-spin-button {
|
||||
-webkit-filter: invert(1);
|
||||
filter: invert(1);
|
||||
}
|
||||
|
||||
.codemirror-colorpicker > .control > .empty {
|
||||
background: url("") repeat;
|
||||
.colorpicker-input:hover {
|
||||
border-color: var(--input-border-color-hover);
|
||||
}
|
||||
.codemirror-colorpicker > .control .drag-bar,
|
||||
.codemirror-colorpicker > .control .drag-bar2 {
|
||||
border: 1px solid rgba(0, 0, 0, 0.5);
|
||||
-webkit-box-shadow: 0 0 2px 0 rgba(0, 0, 0, 0.1);
|
||||
-moz-box-shadow: 0 0 2px 0 rgba(0, 0, 0, 0.1);
|
||||
box-shadow: 0 0 2px 0 rgba(0, 0, 0, 0.1);
|
||||
background-color: #FFFFFF;
|
||||
|
||||
.colorpicker-input:focus {
|
||||
color: var(--input-color-focus);
|
||||
border-color: var(--input-border-color-focus);
|
||||
background-color: var(--input-background-color-focus);
|
||||
}
|
||||
.codemirror-colorpicker > .information {
|
||||
/*border-top: 1px solid #e8e8e8;*/
|
||||
|
||||
.colorpicker-theme-dark input:focus {
|
||||
outline: none !important;
|
||||
}
|
||||
.codemirror-colorpicker > .information > .title {
|
||||
color: #a3a3a3;
|
||||
|
||||
.colorpicker-input:invalid {
|
||||
border-color: var(--invalid-border-color);
|
||||
background-color: var(--invalid-background-color);
|
||||
color: var(--invalid-color);
|
||||
}
|
||||
.codemirror-colorpicker > .information > .input {
|
||||
color: #333;
|
||||
|
||||
.colorpicker-title {
|
||||
text-align: center;
|
||||
font-size: 12px;
|
||||
font-family: monospace;
|
||||
display: flex;
|
||||
justify-content: center;
|
||||
color: var(--label-color);
|
||||
}
|
||||
|
||||
.colorpicker-title-action {
|
||||
cursor: pointer;
|
||||
}
|
||||
|
||||
.colorpicker-title-action[data-active] {
|
||||
font-weight: bold;
|
||||
color: var(--input-color);
|
||||
cursor: default;
|
||||
pointer-events: none;
|
||||
}
|
||||
|
||||
.colorpicker-format-change {
|
||||
position: absolute;
|
||||
display: block;
|
||||
width: var(--switcher-width);
|
||||
top: 0;
|
||||
right: 0;
|
||||
bottom: 0;
|
||||
overflow: hidden;
|
||||
}
|
||||
|
||||
.colorpicker-format-change-button {
|
||||
width: 100%;
|
||||
height: 100%;
|
||||
background: transparent;
|
||||
border: 0;
|
||||
cursor: pointer;
|
||||
outline: none;
|
||||
font-family: monospace !important;
|
||||
font-size: var(--switcher-width) !important;
|
||||
margin-top: -5px;
|
||||
color: var(--label-color);
|
||||
text-align: center;
|
||||
}
|
||||
|
||||
.colorpicker-format-change-button:hover {
|
||||
color: var(--label-color-hover);
|
||||
}
|
File diff suppressed because it is too large
Load Diff
|
@ -1,434 +1,457 @@
|
|||
(function(mod) {
|
||||
if (typeof exports == "object" && typeof module == "object") // CommonJS
|
||||
mod(require("codemirror"));
|
||||
else if (typeof define == "function" && define.amd) // AMD
|
||||
define(["codemirror"], mod);
|
||||
else // Plain browser env
|
||||
mod(CodeMirror);
|
||||
})(function(CodeMirror) {
|
||||
"use strict";
|
||||
/* global CodeMirror */
|
||||
'use strict';
|
||||
|
||||
// color names
|
||||
var color_names={aliceblue:"rgb(240, 248, 255)",antiquewhite:"rgb(250, 235, 215)",aqua:"rgb(0, 255, 255)",aquamarine:"rgb(127, 255, 212)",azure:"rgb(240, 255, 255)",beige:"rgb(245, 245, 220)",bisque:"rgb(255, 228, 196)",black:"rgb(0, 0, 0)",blanchedalmond:"rgb(255, 235, 205)",blue:"rgb(0, 0, 255)",blueviolet:"rgb(138, 43, 226)",brown:"rgb(165, 42, 42)",burlywood:"rgb(222, 184, 135)",cadetblue:"rgb(95, 158, 160)",chartreuse:"rgb(127, 255, 0)",chocolate:"rgb(210, 105, 30)",coral:"rgb(255, 127, 80)",cornflowerblue:"rgb(100, 149, 237)",cornsilk:"rgb(255, 248, 220)",crimson:"rgb(237, 20, 61)",cyan:"rgb(0, 255, 255)",darkblue:"rgb(0, 0, 139)",darkcyan:"rgb(0, 139, 139)",darkgoldenrod:"rgb(184, 134, 11)",darkgray:"rgb(169, 169, 169)",darkgrey:"rgb(169, 169, 169)",darkgreen:"rgb(0, 100, 0)",darkkhaki:"rgb(189, 183, 107)",darkmagenta:"rgb(139, 0, 139)",darkolivegreen:"rgb(85, 107, 47)",darkorange:"rgb(255, 140, 0)",darkorchid:"rgb(153, 50, 204)",darkred:"rgb(139, 0, 0)",darksalmon:"rgb(233, 150, 122)",darkseagreen:"rgb(143, 188, 143)",darkslateblue:"rgb(72, 61, 139)",darkslategray:"rgb(47, 79, 79)",darkslategrey:"rgb(47, 79, 79)",darkturquoise:"rgb(0, 206, 209)",darkviolet:"rgb(148, 0, 211)",deeppink:"rgb(255, 20, 147)",deepskyblue:"rgb(0, 191, 255)",dimgray:"rgb(105, 105, 105)",dimgrey:"rgb(105, 105, 105)",dodgerblue:"rgb(30, 144, 255)",firebrick:"rgb(178, 34, 34)",floralwhite:"rgb(255, 250, 240)",forestgreen:"rgb(34, 139, 34)",fuchsia:"rgb(255, 0, 255)",gainsboro:"rgb(220, 220, 220)",ghostwhite:"rgb(248, 248, 255)",gold:"rgb(255, 215, 0)",goldenrod:"rgb(218, 165, 32)",gray:"rgb(128, 128, 128)",grey:"rgb(128, 128, 128)",green:"rgb(0, 128, 0)",greenyellow:"rgb(173, 255, 47)",honeydew:"rgb(240, 255, 240)",hotpink:"rgb(255, 105, 180)",indianred:"rgb(205, 92, 92)",indigo:"rgb(75, 0, 130)",ivory:"rgb(255, 255, 240)",khaki:"rgb(240, 230, 140)",lavender:"rgb(230, 230, 250)",lavenderblush:"rgb(255, 240, 245)",lawngreen:"rgb(124, 252, 0)",lemonchiffon:"rgb(255, 250, 205)",lightblue:"rgb(173, 216, 230)",lightcoral:"rgb(240, 128, 128)",lightcyan:"rgb(224, 255, 255)",lightgoldenrodyellow:"rgb(250, 250, 210)",lightgreen:"rgb(144, 238, 144)",lightgray:"rgb(211, 211, 211)",lightgrey:"rgb(211, 211, 211)",lightpink:"rgb(255, 182, 193)",lightsalmon:"rgb(255, 160, 122)",lightseagreen:"rgb(32, 178, 170)",lightskyblue:"rgb(135, 206, 250)",lightslategray:"rgb(119, 136, 153)",lightslategrey:"rgb(119, 136, 153)",lightsteelblue:"rgb(176, 196, 222)",lightyellow:"rgb(255, 255, 224)",lime:"rgb(0, 255, 0)",limegreen:"rgb(50, 205, 50)",linen:"rgb(250, 240, 230)",magenta:"rgb(255, 0, 255)",maroon:"rgb(128, 0, 0)",mediumaquamarine:"rgb(102, 205, 170)",mediumblue:"rgb(0, 0, 205)",mediumorchid:"rgb(186, 85, 211)",mediumpurple:"rgb(147, 112, 219)",mediumseagreen:"rgb(60, 179, 113)",mediumslateblue:"rgb(123, 104, 238)",mediumspringgreen:"rgb(0, 250, 154)",mediumturquoise:"rgb(72, 209, 204)",mediumvioletred:"rgb(199, 21, 133)",midnightblue:"rgb(25, 25, 112)",mintcream:"rgb(245, 255, 250)",mistyrose:"rgb(255, 228, 225)",moccasin:"rgb(255, 228, 181)",navajowhite:"rgb(255, 222, 173)",navy:"rgb(0, 0, 128)",oldlace:"rgb(253, 245, 230)",olive:"rgb(128, 128, 0)",olivedrab:"rgb(107, 142, 35)",orange:"rgb(255, 165, 0)",orangered:"rgb(255, 69, 0)",orchid:"rgb(218, 112, 214)",palegoldenrod:"rgb(238, 232, 170)",palegreen:"rgb(152, 251, 152)",paleturquoise:"rgb(175, 238, 238)",palevioletred:"rgb(219, 112, 147)",papayawhip:"rgb(255, 239, 213)",peachpuff:"rgb(255, 218, 185)",peru:"rgb(205, 133, 63)",pink:"rgb(255, 192, 203)",plum:"rgb(221, 160, 221)",powderblue:"rgb(176, 224, 230)",purple:"rgb(128, 0, 128)",rebeccapurple:"rgb(102, 51, 153)",red:"rgb(255, 0, 0)",rosybrown:"rgb(188, 143, 143)",royalblue:"rgb(65, 105, 225)",saddlebrown:"rgb(139, 69, 19)",salmon:"rgb(250, 128, 114)",sandybrown:"rgb(244, 164, 96)",seagreen:"rgb(46, 139, 87)",seashell:"rgb(255, 245, 238)",sienna:"rgb(160, 82, 45)",silver:"rgb(192, 192, 192)",skyblue:"rgb(135, 206, 235)",slateblue:"rgb(106, 90, 205)",slategray:"rgb(112, 128, 144)",slategrey:"rgb(112, 128, 144)",snow:"rgb(255, 250, 250)",springgreen:"rgb(0, 255, 127)",steelblue:"rgb(70, 130, 180)",tan:"rgb(210, 180, 140)",teal:"rgb(0, 128, 128)",thistle:"rgb(216, 191, 216)",tomato:"rgb(255, 99, 71)",turquoise:"rgb(64, 224, 208)",violet:"rgb(238, 130, 238)",wheat:"rgb(245, 222, 179)",white:"rgb(255, 255, 255)",whitesmoke:"rgb(245, 245, 245)",yellow:"rgb(255, 255, 0)",yellowgreen:"rgb(154, 205, 50)",transparent:"rgba(0, 0, 0, 0)"};
|
||||
(() => {
|
||||
const OWN_TOKEN_NAME = 'colorview';
|
||||
const OWN_DOM_CLASS = 'cm-' + OWN_TOKEN_NAME;
|
||||
const OWN_BACKGROUND_CLASS = 'codemirror-colorview-background';
|
||||
const HOOKED_TOKEN = new Map([
|
||||
['atom', colorizeAtom],
|
||||
['keyword', colorizeKeyword],
|
||||
].map(([name, fn]) => [name, {override: name + ' ' + OWN_TOKEN_NAME, process: fn}]));
|
||||
|
||||
var colorpicker_class = 'codemirror-colorview';
|
||||
var colorpicker_background_class = 'codemirror-colorview-background';
|
||||
const NAMED_COLORS = getNamedColorsMap();
|
||||
const TRANSPARENT = {
|
||||
color: 'transparent',
|
||||
colorValue: 'rgba(0, 0, 0, 0)', // as per the CSS spec
|
||||
};
|
||||
const RX_COLOR = {
|
||||
hex: /#(?:[a-f\d]{3,4}|[a-f\d]{6}|[a-f\d]{8})\b/yi,
|
||||
rgb: /rgb\((?:\s*\d{1,3}\s*,\s*){2}\d{1,3}\s*\)/yi,
|
||||
rgba: /rgba\((?:\s*\d{1,3}\s*,\s*){3}\d*\.?\d+\s*\)/yi,
|
||||
hsl: /hsl\(\s*(?:-?\d+|-?\d*\.\d+)\s*(?:,\s*(?:-?\d+|-?\d*\.\d+)%\s*){2}\)/yi,
|
||||
hsla: /hsla\(\s*(?:-?\d+|-?\d*\.\d+)\s*(?:,\s*(?:-?\d+|-?\d*\.\d+)%\s*){2},\s*(?:-?\d+|-?\d*\.\d+)\s*\)/yi,
|
||||
named: new RegExp([...NAMED_COLORS.keys()].join('|'), 'i'),
|
||||
};
|
||||
|
||||
// Excluded tokens do not show color views..
|
||||
var excluded_token = ['comment'];
|
||||
const CodeMirrorEvents = {
|
||||
update(cm) {
|
||||
if (cm.state.colorpicker.cache.size) {
|
||||
renderVisibleTokens(cm);
|
||||
}
|
||||
},
|
||||
keyup(cm) {
|
||||
const popup = cm.state.colorpicker.popup;
|
||||
if (popup && popup.options.isShortCut === false) {
|
||||
popup.hide();
|
||||
}
|
||||
},
|
||||
mousedown(cm, event) {
|
||||
const self = cm.state.colorpicker;
|
||||
const isMarker = event.button === 0 && event.target.classList.contains(OWN_BACKGROUND_CLASS);
|
||||
window.dispatchEvent(new CustomEvent('close-colorpicker-popup', {detail: isMarker && self.popup}));
|
||||
if (isMarker) {
|
||||
event.preventDefault();
|
||||
self.openPopupForToken(event.target.parentNode);
|
||||
}
|
||||
},
|
||||
};
|
||||
|
||||
CodeMirror.defineOption("colorpicker", false, function (cm, val, old) {
|
||||
function registerEvents(cm) {
|
||||
Object.keys(CodeMirrorEvents).forEach(name => cm.on(name, CodeMirrorEvents[name]));
|
||||
}
|
||||
|
||||
if (old && old != CodeMirror.Init) {
|
||||
function unregisterEvents(cm) {
|
||||
Object.keys(CodeMirrorEvents).forEach(name => cm.off(name, CodeMirrorEvents[name]));
|
||||
}
|
||||
|
||||
if (cm.state.colorpicker)
|
||||
{
|
||||
cm.state.colorpicker.destroy();
|
||||
cm.state.colorpicker = null;
|
||||
function registerHooks() {
|
||||
const mx = CodeMirror.modeExtensions.css;
|
||||
if (!mx || mx.token !== colorizeToken) {
|
||||
CodeMirror.extendMode('css', {
|
||||
token: colorizeToken,
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
// remove event listener
|
||||
function unregisterHooks() {
|
||||
const mx = CodeMirror.modeExtensions.css;
|
||||
if (mx && mx.token === colorizeToken) {
|
||||
delete mx.token;
|
||||
}
|
||||
}
|
||||
|
||||
function resetMode(cm) {
|
||||
cm.setOption('mode', cm.getMode().name);
|
||||
}
|
||||
|
||||
function colorizeToken(stream, state) {
|
||||
const token = this._token.apply(this, arguments);
|
||||
const hookedToken = token && HOOKED_TOKEN.get(token);
|
||||
if (!token || !hookedToken) {
|
||||
return token;
|
||||
}
|
||||
const data = state.colorpicker = (state.colorpicker || {});
|
||||
const cache = data.cache = (data.cache || stream.lineOracle.doc.cm.state.colorpicker.cache);
|
||||
const string = stream.string;
|
||||
const sameString = string === data.lastString;
|
||||
|
||||
data.lastString = string;
|
||||
|
||||
let lineCache = data.lineCache = (sameString ? data.lineCache : cache.get(string));
|
||||
if (lineCache && lineCache.get(stream.start)) {
|
||||
return hookedToken.override;
|
||||
}
|
||||
|
||||
const color = hookedToken.process(stream);
|
||||
if (color) {
|
||||
if (!lineCache) {
|
||||
lineCache = data.lineCache = new Map();
|
||||
cache.set(string, lineCache);
|
||||
}
|
||||
lineCache.set(stream.start, color);
|
||||
lineCache.set('lastAccessTime', performance.now());
|
||||
return hookedToken.override;
|
||||
}
|
||||
|
||||
return token;
|
||||
}
|
||||
|
||||
function colorizeAtom(stream) {
|
||||
const {start, pos, string} = stream;
|
||||
const c1 = string.charAt(start);
|
||||
if ((c1 === 't' || c1 === 'T') && string.slice(start, pos).toLowerCase() === 'transparent') {
|
||||
return TRANSPARENT;
|
||||
}
|
||||
const maybeHex = c1 === '#';
|
||||
const s = !maybeHex && string.charAt(pos) === '(' && string.slice(start, pos).toLowerCase();
|
||||
if (maybeHex || (s === 'rgb' || s === 'rgba' || s === 'hsl' || s === 'hsla')) {
|
||||
const rx = maybeHex ? RX_COLOR.hex : RX_COLOR[s];
|
||||
rx.lastIndex = start;
|
||||
const match = rx.exec(string);
|
||||
return match && {color: match[0]};
|
||||
}
|
||||
}
|
||||
|
||||
function colorizeKeyword(stream) {
|
||||
const {start, pos, string} = stream;
|
||||
if (string.charAt(start) !== '!') {
|
||||
const color = string.slice(start, pos);
|
||||
const colorValue = NAMED_COLORS.get(color.toLowerCase());
|
||||
return colorValue && {color, colorValue};
|
||||
}
|
||||
}
|
||||
|
||||
function renderVisibleTokens(cm) {
|
||||
const {cache, options} = cm.state.colorpicker;
|
||||
let line = cm.display.viewFrom - 1;
|
||||
for (const {line: lineHandle, text} of cm.display.renderedView) {
|
||||
if (!lineHandle.parent) {
|
||||
continue;
|
||||
}
|
||||
line++;
|
||||
const styles = lineHandle.styles;
|
||||
if (!styles) {
|
||||
continue;
|
||||
}
|
||||
const lineCache = cache.get(lineHandle.text);
|
||||
if (!lineCache) {
|
||||
continue;
|
||||
}
|
||||
let lineCacheAlive = false;
|
||||
let elementIndex = 0;
|
||||
let elements;
|
||||
for (let i = 1; i < styles.length; i += 2) {
|
||||
const token = styles[i + 1];
|
||||
if (!token || !token.includes(OWN_TOKEN_NAME)) {
|
||||
continue;
|
||||
}
|
||||
|
||||
if (val)
|
||||
{
|
||||
cm.state.colorpicker = new codemirror_colorpicker(cm, val);
|
||||
const start = styles[i - 2] || 0;
|
||||
const data = lineCache.get(start);
|
||||
if (!data) {
|
||||
continue;
|
||||
}
|
||||
});
|
||||
|
||||
function onChange(cm, evt) {
|
||||
if (evt.origin == 'setValue') { // if content is changed by setValue method, it initialize markers
|
||||
cm.state.colorpicker.close_color_picker();
|
||||
cm.state.colorpicker.init_color_update();
|
||||
cm.state.colorpicker.style_color_update();
|
||||
} else {
|
||||
cm.state.colorpicker.style_color_update(cm.getCursor().line);
|
||||
elements = elements || text.getElementsByClassName(OWN_DOM_CLASS);
|
||||
const el = elements[elementIndex++];
|
||||
if (el.colorpickerData && el.colorpickerData.color === data.color) {
|
||||
continue;
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
function onUpdate(cm, evt) {
|
||||
if (!cm.state.colorpicker.isUpdate) {
|
||||
cm.state.colorpicker.isUpdate = true;
|
||||
cm.state.colorpicker.close_color_picker();
|
||||
cm.state.colorpicker.init_color_update();
|
||||
cm.state.colorpicker.style_color_update();
|
||||
el.colorpickerData = Object.assign({line, ch: start}, data);
|
||||
let bg = el.firstElementChild;
|
||||
if (!bg) {
|
||||
bg = document.createElement('div');
|
||||
bg.className = OWN_BACKGROUND_CLASS;
|
||||
bg.title = options.tooltip;
|
||||
el.appendChild(bg);
|
||||
}
|
||||
bg.style.setProperty('background-color', data.color, 'important');
|
||||
lineCacheAlive = true;
|
||||
}
|
||||
if (lineCacheAlive) {
|
||||
lineCache.set('lastAccessTime', performance.now());
|
||||
}
|
||||
}
|
||||
trimCache(cm);
|
||||
}
|
||||
|
||||
function trimCache(cm, debounced) {
|
||||
if (!debounced) {
|
||||
clearTimeout(trimCache.timer);
|
||||
trimCache.timer = setTimeout(trimCache, 20e3, cm, true);
|
||||
return;
|
||||
}
|
||||
const cutoff = performance.now() - 60e3;
|
||||
const {cache} = cm.state.colorpicker;
|
||||
const textToKeep = new Set();
|
||||
cm.doc.iter(({text}) => textToKeep.add(text));
|
||||
for (const [text, lineCache] of cache.entries()) {
|
||||
if (lineCache.get('lastAccessTime') < cutoff && !textToKeep.has(text)) {
|
||||
cache.delete(text);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
function getNamedColorsMap() {
|
||||
return new Map([
|
||||
['aliceblue', '#f0f8ff'],
|
||||
['antiquewhite', '#faebd7'],
|
||||
['aqua', '#00ffff'],
|
||||
['aquamarine', '#7fffd4'],
|
||||
['azure', '#f0ffff'],
|
||||
['beige', '#f5f5dc'],
|
||||
['bisque', '#ffe4c4'],
|
||||
['black', '#000000'],
|
||||
['blanchedalmond', '#ffebcd'],
|
||||
['blue', '#0000ff'],
|
||||
['blueviolet', '#8a2be2'],
|
||||
['brown', '#a52a2a'],
|
||||
['burlywood', '#deb887'],
|
||||
['cadetblue', '#5f9ea0'],
|
||||
['chartreuse', '#7fff00'],
|
||||
['chocolate', '#d2691e'],
|
||||
['coral', '#ff7f50'],
|
||||
['cornflowerblue', '#6495ed'],
|
||||
['cornsilk', '#fff8dc'],
|
||||
['crimson', '#dc143c'],
|
||||
['cyan', '#00ffff'],
|
||||
['darkblue', '#00008b'],
|
||||
['darkcyan', '#008b8b'],
|
||||
['darkgoldenrod', '#b8860b'],
|
||||
['darkgray', '#a9a9a9'],
|
||||
['darkgreen', '#006400'],
|
||||
['darkgrey', '#a9a9a9'],
|
||||
['darkkhaki', '#bdb76b'],
|
||||
['darkmagenta', '#8b008b'],
|
||||
['darkolivegreen', '#556b2f'],
|
||||
['darkorange', '#ff8c00'],
|
||||
['darkorchid', '#9932cc'],
|
||||
['darkred', '#8b0000'],
|
||||
['darksalmon', '#e9967a'],
|
||||
['darkseagreen', '#8fbc8f'],
|
||||
['darkslateblue', '#483d8b'],
|
||||
['darkslategray', '#2f4f4f'],
|
||||
['darkslategrey', '#2f4f4f'],
|
||||
['darkturquoise', '#00ced1'],
|
||||
['darkviolet', '#9400d3'],
|
||||
['deeppink', '#ff1493'],
|
||||
['deepskyblue', '#00bfff'],
|
||||
['dimgray', '#696969'],
|
||||
['dimgrey', '#696969'],
|
||||
['dodgerblue', '#1e90ff'],
|
||||
['firebrick', '#b22222'],
|
||||
['floralwhite', '#fffaf0'],
|
||||
['forestgreen', '#228b22'],
|
||||
['fuchsia', '#ff00ff'],
|
||||
['gainsboro', '#dcdcdc'],
|
||||
['ghostwhite', '#f8f8ff'],
|
||||
['gold', '#ffd700'],
|
||||
['goldenrod', '#daa520'],
|
||||
['gray', '#808080'],
|
||||
['green', '#008000'],
|
||||
['greenyellow', '#adff2f'],
|
||||
['grey', '#808080'],
|
||||
['honeydew', '#f0fff0'],
|
||||
['hotpink', '#ff69b4'],
|
||||
['indianred', '#cd5c5c'],
|
||||
['indigo', '#4b0082'],
|
||||
['ivory', '#fffff0'],
|
||||
['khaki', '#f0e68c'],
|
||||
['lavender', '#e6e6fa'],
|
||||
['lavenderblush', '#fff0f5'],
|
||||
['lawngreen', '#7cfc00'],
|
||||
['lemonchiffon', '#fffacd'],
|
||||
['lightblue', '#add8e6'],
|
||||
['lightcoral', '#f08080'],
|
||||
['lightcyan', '#e0ffff'],
|
||||
['lightgoldenrodyellow', '#fafad2'],
|
||||
['lightgray', '#d3d3d3'],
|
||||
['lightgreen', '#90ee90'],
|
||||
['lightgrey', '#d3d3d3'],
|
||||
['lightpink', '#ffb6c1'],
|
||||
['lightsalmon', '#ffa07a'],
|
||||
['lightseagreen', '#20b2aa'],
|
||||
['lightskyblue', '#87cefa'],
|
||||
['lightslategray', '#778899'],
|
||||
['lightslategrey', '#778899'],
|
||||
['lightsteelblue', '#b0c4de'],
|
||||
['lightyellow', '#ffffe0'],
|
||||
['lime', '#00ff00'],
|
||||
['limegreen', '#32cd32'],
|
||||
['linen', '#faf0e6'],
|
||||
['magenta', '#ff00ff'],
|
||||
['maroon', '#800000'],
|
||||
['mediumaquamarine', '#66cdaa'],
|
||||
['mediumblue', '#0000cd'],
|
||||
['mediumorchid', '#ba55d3'],
|
||||
['mediumpurple', '#9370db'],
|
||||
['mediumseagreen', '#3cb371'],
|
||||
['mediumslateblue', '#7b68ee'],
|
||||
['mediumspringgreen', '#00fa9a'],
|
||||
['mediumturquoise', '#48d1cc'],
|
||||
['mediumvioletred', '#c71585'],
|
||||
['midnightblue', '#191970'],
|
||||
['mintcream', '#f5fffa'],
|
||||
['mistyrose', '#ffe4e1'],
|
||||
['moccasin', '#ffe4b5'],
|
||||
['navajowhite', '#ffdead'],
|
||||
['navy', '#000080'],
|
||||
['oldlace', '#fdf5e6'],
|
||||
['olive', '#808000'],
|
||||
['olivedrab', '#6b8e23'],
|
||||
['orange', '#ffa500'],
|
||||
['orangered', '#ff4500'],
|
||||
['orchid', '#da70d6'],
|
||||
['palegoldenrod', '#eee8aa'],
|
||||
['palegreen', '#98fb98'],
|
||||
['paleturquoise', '#afeeee'],
|
||||
['palevioletred', '#db7093'],
|
||||
['papayawhip', '#ffefd5'],
|
||||
['peachpuff', '#ffdab9'],
|
||||
['peru', '#cd853f'],
|
||||
['pink', '#ffc0cb'],
|
||||
['plum', '#dda0dd'],
|
||||
['powderblue', '#b0e0e6'],
|
||||
['purple', '#800080'],
|
||||
['rebeccapurple', '#663399'],
|
||||
['red', '#ff0000'],
|
||||
['rosybrown', '#bc8f8f'],
|
||||
['royalblue', '#4169e1'],
|
||||
['saddlebrown', '#8b4513'],
|
||||
['salmon', '#fa8072'],
|
||||
['sandybrown', '#f4a460'],
|
||||
['seagreen', '#2e8b57'],
|
||||
['seashell', '#fff5ee'],
|
||||
['sienna', '#a0522d'],
|
||||
['silver', '#c0c0c0'],
|
||||
['skyblue', '#87ceeb'],
|
||||
['slateblue', '#6a5acd'],
|
||||
['slategray', '#708090'],
|
||||
['slategrey', '#708090'],
|
||||
['snow', '#fffafa'],
|
||||
['springgreen', '#00ff7f'],
|
||||
['steelblue', '#4682b4'],
|
||||
['tan', '#d2b48c'],
|
||||
['teal', '#008080'],
|
||||
['thistle', '#d8bfd8'],
|
||||
['tomato', '#ff6347'],
|
||||
['turquoise', '#40e0d0'],
|
||||
['violet', '#ee82ee'],
|
||||
['wheat', '#f5deb3'],
|
||||
['white', '#ffffff'],
|
||||
['whitesmoke', '#f5f5f5'],
|
||||
['yellow', '#ffff00'],
|
||||
['yellowgreen', '#9acd32'],
|
||||
]);
|
||||
}
|
||||
|
||||
class ColorMarker {
|
||||
constructor(cm, {
|
||||
tooltip = 'Open color picker',
|
||||
popupOptions = {},
|
||||
colorpicker,
|
||||
forceUpdate,
|
||||
} = {}) {
|
||||
this.cm = cm;
|
||||
this.options = {
|
||||
tooltip,
|
||||
popup: Object.assign({
|
||||
hideDelay: 2000,
|
||||
hexUppercase: false,
|
||||
tooltipForSwitcher: 'Switch formats: HEX -> RGB -> HSL',
|
||||
}, popupOptions),
|
||||
};
|
||||
this.popup = cm.colorpicker ? cm.colorpicker() : colorpicker;
|
||||
this.cache = new Map();
|
||||
registerHooks(cm);
|
||||
registerEvents(cm);
|
||||
if (forceUpdate) {
|
||||
resetMode(cm);
|
||||
}
|
||||
}
|
||||
|
||||
function onRefresh(cm, evt) {
|
||||
onChange(cm, { origin : 'setValue'});
|
||||
destroy() {
|
||||
unregisterHooks(this.cm);
|
||||
unregisterEvents(this.cm);
|
||||
resetMode(this.cm);
|
||||
this.cm.state.colorpicker = null;
|
||||
}
|
||||
|
||||
function onKeyup(cm, evt) {
|
||||
cm.state.colorpicker.keyup(evt);
|
||||
}
|
||||
|
||||
function onMousedown(cm, evt) {
|
||||
if (cm.state.colorpicker.is_edit_mode())
|
||||
{
|
||||
cm.state.colorpicker.check_mousedown(evt);
|
||||
openPopup(defaultColor = '#FFFFFF') {
|
||||
const cursor = this.cm.getCursor();
|
||||
const data = {
|
||||
line: cursor.line,
|
||||
ch: cursor.ch,
|
||||
color: defaultColor,
|
||||
isShortCut: true,
|
||||
};
|
||||
for (const {from, marker} of this.cm.getLineHandle(cursor.line).markedSpans || []) {
|
||||
if (from <= data.ch && (marker.replacedWith || {}).colorpickerData) {
|
||||
const {color, colorValue} = marker.replacedWith.colorpickerData;
|
||||
if (data.ch <= from + color.length) {
|
||||
data.ch = from;
|
||||
data.color = color;
|
||||
data.colorValue = colorValue;
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
this.openPopupForToken({colorpickerData: data});
|
||||
}
|
||||
|
||||
function onPaste (cm, evt) {
|
||||
onChange(cm, { origin : 'setValue'});
|
||||
openPopupForToken({colorpickerData: data}) {
|
||||
if (this.popup) {
|
||||
const {left, bottom: top} = this.cm.charCoords(data, 'window');
|
||||
this.popup.show(Object.assign(this.options.popup, data, {
|
||||
top,
|
||||
left,
|
||||
cm: this.cm,
|
||||
color: data.colorValue || data.color,
|
||||
prevColor: data.color,
|
||||
isShortCut: false,
|
||||
callback: ColorMarker.popupOnChange,
|
||||
}));
|
||||
}
|
||||
}
|
||||
|
||||
function onScroll (cm) {
|
||||
cm.state.colorpicker.close_color_picker();
|
||||
closePopup() {
|
||||
if (this.popup) {
|
||||
this.popup.hide();
|
||||
}
|
||||
}
|
||||
|
||||
function debounce (callback, delay) {
|
||||
|
||||
var t = undefined;
|
||||
|
||||
return function (cm, e) {
|
||||
if (t) {
|
||||
clearTimeout(t);
|
||||
}
|
||||
|
||||
t = setTimeout(function () {
|
||||
callback(cm, e);
|
||||
}, delay || 300);
|
||||
}
|
||||
static popupOnChange(newColor) {
|
||||
const {cm, line, ch, embedderCallback} = this;
|
||||
const to = {line, ch: ch + this.prevColor.length};
|
||||
if (cm.getRange(this, to) !== newColor) {
|
||||
this.prevColor = newColor;
|
||||
cm.replaceRange(newColor, this, to, '*colorpicker');
|
||||
}
|
||||
if (typeof embedderCallback === 'function') {
|
||||
embedderCallback(this);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
function has_class(el, cls) {
|
||||
if (!el || !el.className) {
|
||||
return false;
|
||||
} else {
|
||||
var newClass = ' ' + el.className + ' ';
|
||||
return newClass.indexOf(' ' + cls + ' ') > -1;
|
||||
}
|
||||
CodeMirror.defineOption('colorpicker', false, (cm, value, oldValue) => {
|
||||
if (oldValue && oldValue !== CodeMirror.Init && cm.state.colorpicker) {
|
||||
cm.state.colorpicker.destroy();
|
||||
}
|
||||
|
||||
function codemirror_colorpicker (cm, opt) {
|
||||
var self = this;
|
||||
|
||||
if (typeof opt == 'boolean')
|
||||
{
|
||||
opt = { mode : 'view' };
|
||||
} else {
|
||||
opt = Object.assign({ mode: 'view' }, opt || {});
|
||||
}
|
||||
|
||||
this.opt = opt;
|
||||
this.cm = cm;
|
||||
this.markers = {};
|
||||
|
||||
// set excluded token
|
||||
excluded_token = this.opt.excluded_token || excluded_token;
|
||||
|
||||
if (this.cm.colorpicker) {
|
||||
this.colorpicker = this.cm.colorpicker();
|
||||
} else if (this.opt.colorpicker) {
|
||||
this.colorpicker = this.opt.colorpicker;
|
||||
}
|
||||
|
||||
this.init_event();
|
||||
|
||||
if (value) {
|
||||
cm.state.colorpicker = new ColorMarker(cm, value);
|
||||
}
|
||||
});
|
||||
|
||||
codemirror_colorpicker.prototype.init_event = function () {
|
||||
|
||||
this.cm.on('mousedown', onMousedown);
|
||||
this.cm.on('keyup', onKeyup);
|
||||
this.cm.on('change', onChange);
|
||||
this.cm.on('update', onUpdate);
|
||||
this.cm.on('refresh', onRefresh);
|
||||
|
||||
// create paste callback
|
||||
this.onPasteCallback = (function (cm, callback) {
|
||||
return function (evt) {
|
||||
callback.call(this, cm, evt);
|
||||
}
|
||||
})(this.cm, onPaste);
|
||||
|
||||
this.cm.getWrapperElement().addEventListener('paste', this.onPasteCallback);
|
||||
|
||||
if (this.is_edit_mode())
|
||||
{
|
||||
this.cm.on('scroll', debounce(onScroll, 50));
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
codemirror_colorpicker.prototype.is_edit_mode = function () {
|
||||
return this.opt.mode == 'edit';
|
||||
}
|
||||
|
||||
codemirror_colorpicker.prototype.is_view_mode = function () {
|
||||
return this.opt.mode == 'view';
|
||||
}
|
||||
|
||||
codemirror_colorpicker.prototype.destroy = function () {
|
||||
this.cm.off('mousedown', onMousedown);
|
||||
this.cm.off('keyup', onKeyup);
|
||||
this.cm.off('change', onChange)
|
||||
this.cm.getWrapperElement().removeEventListener('paste', this.onPasteCallback);
|
||||
|
||||
if (this.is_edit_mode())
|
||||
{
|
||||
this.cm.off('scroll');
|
||||
}
|
||||
}
|
||||
|
||||
codemirror_colorpicker.prototype.hasClass = function (el, className) {
|
||||
if (!el.className)
|
||||
{
|
||||
return false;
|
||||
} else {
|
||||
var newClass = ' ' + el.className + ' ';
|
||||
return newClass.indexOf(' ' + className + ' ') > -1;
|
||||
}
|
||||
}
|
||||
|
||||
codemirror_colorpicker.prototype.check_mousedown = function (evt) {
|
||||
if (this.hasClass(evt.target, colorpicker_background_class) )
|
||||
{
|
||||
this.open_color_picker(evt.target.parentNode);
|
||||
} else {
|
||||
this.close_color_picker();
|
||||
}
|
||||
}
|
||||
|
||||
codemirror_colorpicker.prototype.popup_color_picker = function (defalutColor) {
|
||||
var cursor = this.cm.getCursor();
|
||||
var self = this;
|
||||
var colorMarker = {
|
||||
lineNo : cursor.line,
|
||||
ch : cursor.ch,
|
||||
color: defalutColor || '#FFFFFF',
|
||||
isShortCut : true
|
||||
};
|
||||
|
||||
Object.keys(this.markers).forEach(function(key) {
|
||||
var searchKey = "#" + key;
|
||||
if (searchKey.indexOf( "#" + colorMarker.lineNo + ":") > -1) {
|
||||
var marker = self.markers[key];
|
||||
|
||||
if (marker.ch <= colorMarker.ch && colorMarker.ch <= marker.ch + marker.color.length) {
|
||||
// when cursor has marker
|
||||
colorMarker.ch = marker.ch;
|
||||
colorMarker.color = marker.color;
|
||||
colorMarker.nameColor = marker.nameColor;
|
||||
}
|
||||
|
||||
}
|
||||
});
|
||||
|
||||
this.open_color_picker(colorMarker);
|
||||
}
|
||||
|
||||
codemirror_colorpicker.prototype.open_color_picker = function (el) {
|
||||
var lineNo = el.lineNo;
|
||||
var ch = el.ch;
|
||||
var nameColor = el.nameColor;
|
||||
var color = el.color;
|
||||
|
||||
|
||||
if (this.colorpicker) {
|
||||
var self = this;
|
||||
var prevColor = color;
|
||||
var pos = this.cm.charCoords({line : lineNo, ch : ch });
|
||||
this.colorpicker.show({
|
||||
left : pos.left,
|
||||
top : pos.bottom,
|
||||
isShortCut : el.isShortCut || false,
|
||||
hideDelay : self.opt.hideDelay || 2000
|
||||
}, nameColor || color, function (newColor) {
|
||||
self.cm.replaceRange(newColor, { line : lineNo, ch : ch } , { line : lineNo, ch : ch + prevColor.length }, '*colorpicker');
|
||||
prevColor = newColor;
|
||||
});
|
||||
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
codemirror_colorpicker.prototype.close_color_picker = function (el) {
|
||||
if (this.colorpicker)
|
||||
{
|
||||
this.colorpicker.hide();
|
||||
}
|
||||
}
|
||||
|
||||
codemirror_colorpicker.prototype.key = function (lineNo, ch) {
|
||||
return [lineNo, ch].join(":");
|
||||
}
|
||||
|
||||
|
||||
codemirror_colorpicker.prototype.keyup = function (evt) {
|
||||
|
||||
if (this.colorpicker ) {
|
||||
if (evt.key == 'Escape') {
|
||||
this.colorpicker.hide();
|
||||
} else if (this.colorpicker.isShortCut() == false) {
|
||||
this.colorpicker.hide();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
codemirror_colorpicker.prototype.init_color_update = function () {
|
||||
this.markers = {}; // initialize marker list
|
||||
}
|
||||
|
||||
codemirror_colorpicker.prototype.style_color_update = function (lineHandle) {
|
||||
|
||||
if (lineHandle) {
|
||||
this.match(lineHandle);
|
||||
} else {
|
||||
var max = this.cm.lineCount();
|
||||
|
||||
for(var lineNo = 0; lineNo < max; lineNo++) {
|
||||
this.match(lineNo);
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
codemirror_colorpicker.prototype.empty_marker = function (lineNo, lineHandle) {
|
||||
var list = lineHandle.markedSpans || [];
|
||||
|
||||
for(var i = 0, len = list.length; i < len; i++) {
|
||||
var key = this.key(lineNo, list[i].from);
|
||||
|
||||
if (key && has_class(list[i].marker.replacedWith, colorpicker_class)) {
|
||||
delete this.markers[key];
|
||||
list[i].marker.clear();
|
||||
}
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
codemirror_colorpicker.prototype.color_regexp = /(#(?:[\da-f]{3}){1,2}|rgb\((?:\s*\d{1,3},\s*){2}\d{1,3}\s*\)|rgba\((?:\s*\d{1,3},\s*){3}\d*\.?\d+\s*\)|hsl\(\s*\d{1,3}(?:,\s*\d{1,3}%){2}\s*\)|hsla\(\s*\d{1,3}(?:,\s*\d{1,3}%){2},\s*\d*\.?\d+\s*\)|([\w_\-]+))/gi;
|
||||
|
||||
codemirror_colorpicker.prototype.match_result = function (lineHandle) {
|
||||
return lineHandle.text.match(this.color_regexp);
|
||||
}
|
||||
|
||||
codemirror_colorpicker.prototype.match = function (lineNo) {
|
||||
var lineHandle = this.cm.getLineHandle(lineNo);
|
||||
|
||||
this.empty_marker(lineNo, lineHandle);
|
||||
|
||||
var result = this.match_result(lineHandle);
|
||||
if (result)
|
||||
{
|
||||
var obj = { next : 0 };
|
||||
for(var i = 0, len = result.length; i < len; i++) {
|
||||
|
||||
if (result[i].indexOf('#') > -1 || result[i].indexOf('rgb') > -1 || result[i].indexOf('hsl') > -1) {
|
||||
this.render(obj, lineNo, lineHandle, result[i]);
|
||||
} else {
|
||||
var nameColor = color_names[result[i]];
|
||||
if (nameColor) {
|
||||
this.render(obj, lineNo, lineHandle, result[i], nameColor);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
codemirror_colorpicker.prototype.make_element = function () {
|
||||
var el = document.createElement('div');
|
||||
|
||||
el.className = colorpicker_class;
|
||||
|
||||
if (this.is_edit_mode())
|
||||
{
|
||||
el.title ="open color picker";
|
||||
} else {
|
||||
el.title ="";
|
||||
}
|
||||
|
||||
el.back_element = this.make_background_element();
|
||||
el.appendChild(el.back_element);
|
||||
|
||||
return el;
|
||||
}
|
||||
|
||||
codemirror_colorpicker.prototype.make_background_element = function () {
|
||||
var el = document.createElement('div');
|
||||
|
||||
el.className = colorpicker_background_class;
|
||||
|
||||
return el;
|
||||
}
|
||||
|
||||
codemirror_colorpicker.prototype.set_state = function (lineNo, start, color, nameColor) {
|
||||
var marker = this.create_marker(lineNo, start);
|
||||
|
||||
|
||||
marker.lineNo = lineNo;
|
||||
marker.ch = start;
|
||||
marker.color = color;
|
||||
marker.nameColor = nameColor;
|
||||
|
||||
return marker;
|
||||
}
|
||||
|
||||
codemirror_colorpicker.prototype.create_marker = function (lineNo, start) {
|
||||
|
||||
var key = this.key(lineNo,start);
|
||||
|
||||
if (!this.markers[key]) {
|
||||
this.markers[key] = this.make_element();
|
||||
}
|
||||
|
||||
|
||||
return this.markers[key];
|
||||
|
||||
}
|
||||
|
||||
codemirror_colorpicker.prototype.has_marker = function (lineNo, start) {
|
||||
var key = this.key(lineNo,start);
|
||||
return !!(this.markers[key])
|
||||
}
|
||||
|
||||
codemirror_colorpicker.prototype.update_element = function (el, color) {
|
||||
el.back_element.style.backgroundColor = color;
|
||||
}
|
||||
|
||||
codemirror_colorpicker.prototype.set_mark = function (line, ch, el) {
|
||||
this.cm.setBookmark({ line : line, ch : ch}, { widget : el, handleMouseEvents : true} );
|
||||
|
||||
}
|
||||
|
||||
codemirror_colorpicker.prototype.is_excluded_token = function (line, ch) {
|
||||
var token = this.cm.getTokenAt({line : line, ch : ch});
|
||||
var count = 0;
|
||||
for(var i = 0, len = excluded_token.length; i < len; i++) {
|
||||
if (token.type === excluded_token[i]) {
|
||||
count++;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
return count > 0; // true is that it has a excluded token
|
||||
}
|
||||
|
||||
codemirror_colorpicker.prototype.render = function (cursor, lineNo, lineHandle, color, nameColor) {
|
||||
var start = lineHandle.text.indexOf(color, cursor.next);
|
||||
|
||||
if (this.is_excluded_token(lineNo, start) === true) {
|
||||
// excluded token do not show.
|
||||
return;
|
||||
}
|
||||
|
||||
cursor.next = start + color.length;
|
||||
|
||||
if (this.has_marker(lineNo, start))
|
||||
{
|
||||
this.update_element(this.create_marker(lineNo, start), nameColor || color);
|
||||
this.set_state(lineNo, start, color, nameColor);
|
||||
return;
|
||||
}
|
||||
|
||||
var el = this.create_marker(lineNo, start);
|
||||
|
||||
this.update_element(el, nameColor || color);
|
||||
this.set_state(lineNo, start, color, nameColor || color);
|
||||
this.set_mark(lineNo, start, el);
|
||||
}
|
||||
});
|
||||
// initial runMode is performed by CodeMirror before setting our option
|
||||
// so we register the hooks right away - not a problem as our js is loaded on demand
|
||||
registerHooks();
|
||||
})();
|
||||
|
|
Loading…
Reference in New Issue
Block a user