@thednp/color-picker 0.0.1 → 0.0.2-alpha1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +1 -0
- package/dist/css/color-picker.css +1 -1
- package/dist/css/color-picker.min.css +1 -1
- package/dist/css/color-picker.rtl.css +1 -1
- package/dist/css/color-picker.rtl.min.css +1 -1
- package/dist/js/color-esm.js +1178 -0
- package/dist/js/color-esm.min.js +2 -0
- package/dist/js/color-palette-esm.js +1252 -0
- package/dist/js/color-palette-esm.min.js +2 -0
- package/dist/js/color-palette.js +1260 -0
- package/dist/js/color-palette.min.js +2 -0
- package/dist/js/color-picker-element-esm.js +287 -319
- package/dist/js/color-picker-element-esm.min.js +2 -2
- package/dist/js/color-picker-element.js +289 -321
- package/dist/js/color-picker-element.min.js +2 -2
- package/dist/js/color-picker-esm.js +520 -552
- package/dist/js/color-picker-esm.min.js +2 -2
- package/dist/js/color-picker.js +522 -554
- package/dist/js/color-picker.min.js +2 -2
- package/dist/js/color.js +1186 -0
- package/dist/js/color.min.js +2 -0
- package/package.json +19 -3
- package/src/js/color-palette.js +10 -3
- package/src/js/color-picker-element.js +1 -1
- package/src/js/color-picker.js +7 -120
- package/src/js/color.js +88 -91
- package/src/js/util/getColorMenu.js +12 -7
- package/src/js/util/setMarkup.js +122 -0
- package/src/js/util/version.js +6 -0
- package/types/cp.d.ts +47 -17
- package/src/js/util/templates.js +0 -10
@@ -1,13 +1,13 @@
|
|
1
1
|
/*!
|
2
|
-
* ColorPickerElement v0.0.
|
2
|
+
* ColorPickerElement v0.0.2alpha1 (http://thednp.github.io/color-picker)
|
3
3
|
* Copyright 2022 © thednp
|
4
4
|
* Licensed under MIT (https://github.com/thednp/color-picker/blob/master/LICENSE)
|
5
5
|
*/
|
6
6
|
(function (global, factory) {
|
7
7
|
typeof exports === 'object' && typeof module !== 'undefined' ? module.exports = factory() :
|
8
8
|
typeof define === 'function' && define.amd ? define(factory) :
|
9
|
-
(global = global || self, global.ColorPickerElement = factory());
|
10
|
-
}(this, (function () { 'use strict';
|
9
|
+
(global = typeof globalThis !== 'undefined' ? globalThis : global || self, global.ColorPickerElement = factory());
|
10
|
+
})(this, (function () { 'use strict';
|
11
11
|
|
12
12
|
/**
|
13
13
|
* Returns the `document` or the `#document` element.
|
@@ -88,14 +88,9 @@
|
|
88
88
|
const getAttribute = (element, attribute) => element.getAttribute(attribute);
|
89
89
|
|
90
90
|
/**
|
91
|
-
*
|
92
|
-
*
|
93
|
-
* @param {(Node | HTMLElement | Element | globalThis)=} node
|
94
|
-
* @returns {HTMLElement | HTMLHeadElement}
|
91
|
+
* A global namespace for `document.head`.
|
95
92
|
*/
|
96
|
-
|
97
|
-
return getDocument(node).head;
|
98
|
-
}
|
93
|
+
const { head: documentHead } = document;
|
99
94
|
|
100
95
|
/**
|
101
96
|
* Shortcut for `window.getComputedStyle(element).propertyName`
|
@@ -116,20 +111,21 @@
|
|
116
111
|
return property in computedStyle ? computedStyle[property] : '';
|
117
112
|
}
|
118
113
|
|
119
|
-
/**
|
120
|
-
* Shortcut for `Object.keys()` static method.
|
121
|
-
* @param {Record<string, any>} obj a target object
|
122
|
-
* @returns {string[]}
|
123
|
-
*/
|
124
|
-
const ObjectKeys = (obj) => Object.keys(obj);
|
125
|
-
|
126
114
|
/**
|
127
115
|
* Shortcut for multiple uses of `HTMLElement.style.propertyName` method.
|
128
116
|
* @param {HTMLElement | Element} element target element
|
129
117
|
* @param {Partial<CSSStyleDeclaration>} styles attribute value
|
130
118
|
*/
|
131
119
|
// @ts-ignore
|
132
|
-
const setElementStyle = (element, styles) => ObjectAssign(element.style, styles);
|
120
|
+
const setElementStyle = (element, styles) => { ObjectAssign(element.style, styles); };
|
121
|
+
|
122
|
+
/**
|
123
|
+
* Shortcut for `String.toLowerCase()`.
|
124
|
+
*
|
125
|
+
* @param {string} source input string
|
126
|
+
* @returns {string} lowercase output string
|
127
|
+
*/
|
128
|
+
const toLowerCase = (source) => source.toLowerCase();
|
133
129
|
|
134
130
|
/**
|
135
131
|
* A list of explicit default non-color values.
|
@@ -147,7 +143,7 @@
|
|
147
143
|
}
|
148
144
|
|
149
145
|
// Color supported formats
|
150
|
-
const COLOR_FORMAT = ['rgb', 'hex', 'hsl', '
|
146
|
+
const COLOR_FORMAT = ['rgb', 'hex', 'hsl', 'hsv', 'hwb'];
|
151
147
|
|
152
148
|
// Hue angles
|
153
149
|
const ANGLES = 'deg|rad|grad|turn';
|
@@ -169,10 +165,17 @@
|
|
169
165
|
// Add angles to the mix
|
170
166
|
const CSS_UNIT2 = `(?:${CSS_UNIT})|(?:${CSS_ANGLE})`;
|
171
167
|
|
168
|
+
// Start & end
|
169
|
+
const START_MATCH = '(?:[\\s|\\(\\s|\\s\\(\\s]+)?';
|
170
|
+
const END_MATCH = '(?:[\\s|\\)\\s]+)?';
|
171
|
+
// Components separation
|
172
|
+
const SEP = '(?:[,|\\s]+)';
|
173
|
+
const SEP2 = '(?:[,|\\/\\s]*)?';
|
174
|
+
|
172
175
|
// Actual matching.
|
173
176
|
// Parentheses and commas are optional, but not required.
|
174
177
|
// Whitespace can take the place of commas or opening paren
|
175
|
-
const PERMISSIVE_MATCH =
|
178
|
+
const PERMISSIVE_MATCH = `${START_MATCH}(${CSS_UNIT2})${SEP}(${CSS_UNIT})${SEP}(${CSS_UNIT})${SEP2}(${CSS_UNIT})?${END_MATCH}`;
|
176
179
|
|
177
180
|
const matchers = {
|
178
181
|
CSS_UNIT: new RegExp(CSS_UNIT2),
|
@@ -205,23 +208,22 @@
|
|
205
208
|
return `${n}`.includes('%');
|
206
209
|
}
|
207
210
|
|
208
|
-
/**
|
209
|
-
* Check to see if string passed in is an angle
|
210
|
-
* @param {string} n testing string
|
211
|
-
* @returns {boolean} the query result
|
212
|
-
*/
|
213
|
-
function isAngle(n) {
|
214
|
-
return ANGLES.split('|').some((a) => `${n}`.includes(a));
|
215
|
-
}
|
216
|
-
|
217
211
|
/**
|
218
212
|
* Check to see if string passed is a web safe colour.
|
213
|
+
* @see https://stackoverflow.com/a/16994164
|
219
214
|
* @param {string} color a colour name, EG: *red*
|
220
215
|
* @returns {boolean} the query result
|
221
216
|
*/
|
222
217
|
function isColorName(color) {
|
223
|
-
|
224
|
-
|
218
|
+
if (nonColors.includes(color)
|
219
|
+
|| ['#', ...COLOR_FORMAT].some((f) => color.includes(f))) return false;
|
220
|
+
|
221
|
+
return ['rgb(255, 255, 255)', 'rgb(0, 0, 0)'].every((c) => {
|
222
|
+
setElementStyle(documentHead, { color });
|
223
|
+
const computedColor = getElementStyle(documentHead, 'color');
|
224
|
+
setElementStyle(documentHead, { color: '' });
|
225
|
+
return computedColor !== c;
|
226
|
+
});
|
225
227
|
}
|
226
228
|
|
227
229
|
/**
|
@@ -242,15 +244,15 @@
|
|
242
244
|
*/
|
243
245
|
function bound01(N, max) {
|
244
246
|
let n = N;
|
245
|
-
if (isOnePointZero(
|
247
|
+
if (isOnePointZero(N)) n = '100%';
|
246
248
|
|
247
|
-
|
248
|
-
|
249
|
-
|
250
|
-
|
249
|
+
const processPercent = isPercentage(n);
|
250
|
+
n = max === 360
|
251
|
+
? parseFloat(n)
|
252
|
+
: Math.min(max, Math.max(0, parseFloat(n)));
|
251
253
|
|
252
254
|
// Automatically convert percentage into number
|
253
|
-
if (
|
255
|
+
if (processPercent) n = (n * max) / 100;
|
254
256
|
|
255
257
|
// Handle floating point rounding errors
|
256
258
|
if (Math.abs(n - max) < 0.000001) {
|
@@ -261,11 +263,11 @@
|
|
261
263
|
// If n is a hue given in degrees,
|
262
264
|
// wrap around out-of-range values into [0, 360] range
|
263
265
|
// then convert into [0, 1].
|
264
|
-
n = (n < 0 ? (n % max) + max : n % max) /
|
266
|
+
n = (n < 0 ? (n % max) + max : n % max) / max;
|
265
267
|
} else {
|
266
268
|
// If n not a hue given in degrees
|
267
269
|
// Convert into [0, 1] range if it isn't already.
|
268
|
-
n = (n % max) /
|
270
|
+
n = (n % max) / max;
|
269
271
|
}
|
270
272
|
return n;
|
271
273
|
}
|
@@ -300,7 +302,6 @@
|
|
300
302
|
* @returns {string}
|
301
303
|
*/
|
302
304
|
function getRGBFromName(name) {
|
303
|
-
const documentHead = getDocumentHead();
|
304
305
|
setElementStyle(documentHead, { color: name });
|
305
306
|
const colorName = getElementStyle(documentHead, 'color');
|
306
307
|
setElementStyle(documentHead, { color: '' });
|
@@ -399,6 +400,36 @@
|
|
399
400
|
return p;
|
400
401
|
}
|
401
402
|
|
403
|
+
/**
|
404
|
+
* Converts an HSL colour value to RGB.
|
405
|
+
*
|
406
|
+
* @param {number} h Hue Angle [0, 1]
|
407
|
+
* @param {number} s Saturation [0, 1]
|
408
|
+
* @param {number} l Lightness Angle [0, 1]
|
409
|
+
* @returns {CP.RGB} {r,g,b} object with [0, 255] ranged values
|
410
|
+
*/
|
411
|
+
function hslToRgb(h, s, l) {
|
412
|
+
let r = 0;
|
413
|
+
let g = 0;
|
414
|
+
let b = 0;
|
415
|
+
|
416
|
+
if (s === 0) {
|
417
|
+
// achromatic
|
418
|
+
g = l;
|
419
|
+
b = l;
|
420
|
+
r = l;
|
421
|
+
} else {
|
422
|
+
const q = l < 0.5 ? l * (1 + s) : l + s - l * s;
|
423
|
+
const p = 2 * l - q;
|
424
|
+
r = hueToRgb(p, q, h + 1 / 3);
|
425
|
+
g = hueToRgb(p, q, h);
|
426
|
+
b = hueToRgb(p, q, h - 1 / 3);
|
427
|
+
}
|
428
|
+
[r, g, b] = [r, g, b].map((x) => x * 255);
|
429
|
+
|
430
|
+
return { r, g, b };
|
431
|
+
}
|
432
|
+
|
402
433
|
/**
|
403
434
|
* Returns an HWB colour object from an RGB colour object.
|
404
435
|
* @link https://www.w3.org/TR/css-color-4/#hwb-to-rgb
|
@@ -461,36 +492,6 @@
|
|
461
492
|
return { r, g, b };
|
462
493
|
}
|
463
494
|
|
464
|
-
/**
|
465
|
-
* Converts an HSL colour value to RGB.
|
466
|
-
*
|
467
|
-
* @param {number} h Hue Angle [0, 1]
|
468
|
-
* @param {number} s Saturation [0, 1]
|
469
|
-
* @param {number} l Lightness Angle [0, 1]
|
470
|
-
* @returns {CP.RGB} {r,g,b} object with [0, 255] ranged values
|
471
|
-
*/
|
472
|
-
function hslToRgb(h, s, l) {
|
473
|
-
let r = 0;
|
474
|
-
let g = 0;
|
475
|
-
let b = 0;
|
476
|
-
|
477
|
-
if (s === 0) {
|
478
|
-
// achromatic
|
479
|
-
g = l;
|
480
|
-
b = l;
|
481
|
-
r = l;
|
482
|
-
} else {
|
483
|
-
const q = l < 0.5 ? l * (1 + s) : l + s - l * s;
|
484
|
-
const p = 2 * l - q;
|
485
|
-
r = hueToRgb(p, q, h + 1 / 3);
|
486
|
-
g = hueToRgb(p, q, h);
|
487
|
-
b = hueToRgb(p, q, h - 1 / 3);
|
488
|
-
}
|
489
|
-
[r, g, b] = [r, g, b].map((x) => x * 255);
|
490
|
-
|
491
|
-
return { r, g, b };
|
492
|
-
}
|
493
|
-
|
494
495
|
/**
|
495
496
|
* Converts an RGB colour value to HSV.
|
496
497
|
*
|
@@ -546,10 +547,11 @@
|
|
546
547
|
const q = v * (1 - f * s);
|
547
548
|
const t = v * (1 - (1 - f) * s);
|
548
549
|
const mod = i % 6;
|
549
|
-
|
550
|
-
|
551
|
-
|
552
|
-
|
550
|
+
let r = [v, q, p, p, t, v][mod];
|
551
|
+
let g = [t, v, v, q, p, p][mod];
|
552
|
+
let b = [p, p, t, v, v, q][mod];
|
553
|
+
[r, g, b] = [r, g, b].map((n) => n * 255);
|
554
|
+
return { r, g, b };
|
553
555
|
}
|
554
556
|
|
555
557
|
/**
|
@@ -573,7 +575,7 @@
|
|
573
575
|
// Return a 3 character hex if possible
|
574
576
|
if (allow3Char && hex[0].charAt(0) === hex[0].charAt(1)
|
575
577
|
&& hex[1].charAt(0) === hex[1].charAt(1)
|
576
|
-
|
578
|
+
&& hex[2].charAt(0) === hex[2].charAt(1)) {
|
577
579
|
return hex[0].charAt(0) + hex[1].charAt(0) + hex[2].charAt(0);
|
578
580
|
}
|
579
581
|
|
@@ -601,39 +603,24 @@
|
|
601
603
|
// Return a 4 character hex if possible
|
602
604
|
if (allow4Char && hex[0].charAt(0) === hex[0].charAt(1)
|
603
605
|
&& hex[1].charAt(0) === hex[1].charAt(1)
|
604
|
-
|
605
|
-
|
606
|
+
&& hex[2].charAt(0) === hex[2].charAt(1)
|
607
|
+
&& hex[3].charAt(0) === hex[3].charAt(1)) {
|
606
608
|
return hex[0].charAt(0) + hex[1].charAt(0) + hex[2].charAt(0) + hex[3].charAt(0);
|
607
609
|
}
|
608
610
|
return hex.join('');
|
609
611
|
}
|
610
612
|
|
611
|
-
/**
|
612
|
-
* Returns a colour object corresponding to a given number.
|
613
|
-
* @param {number} color input number
|
614
|
-
* @returns {CP.RGB} {r,g,b} object with [0, 255] ranged values
|
615
|
-
*/
|
616
|
-
function numberInputToObject(color) {
|
617
|
-
/* eslint-disable no-bitwise */
|
618
|
-
return {
|
619
|
-
r: color >> 16,
|
620
|
-
g: (color & 0xff00) >> 8,
|
621
|
-
b: color & 0xff,
|
622
|
-
};
|
623
|
-
/* eslint-enable no-bitwise */
|
624
|
-
}
|
625
|
-
|
626
613
|
/**
|
627
614
|
* Permissive string parsing. Take in a number of formats, and output an object
|
628
615
|
* based on detected format. Returns {r,g,b} or {h,s,l} or {h,s,v}
|
629
616
|
* @param {string} input colour value in any format
|
630
|
-
* @returns {Record<string, (number | string)> | false} an object matching the RegExp
|
617
|
+
* @returns {Record<string, (number | string | boolean)> | false} an object matching the RegExp
|
631
618
|
*/
|
632
619
|
function stringInputToObject(input) {
|
633
|
-
let color = input.trim()
|
620
|
+
let color = toLowerCase(input.trim());
|
634
621
|
if (color.length === 0) {
|
635
622
|
return {
|
636
|
-
r: 0, g: 0, b: 0, a:
|
623
|
+
r: 0, g: 0, b: 0, a: 1,
|
637
624
|
};
|
638
625
|
}
|
639
626
|
let named = false;
|
@@ -641,11 +628,9 @@
|
|
641
628
|
color = getRGBFromName(color);
|
642
629
|
named = true;
|
643
630
|
} else if (nonColors.includes(color)) {
|
644
|
-
const
|
645
|
-
const rgb = isTransparent ? 0 : 255;
|
646
|
-
const a = isTransparent ? 0 : 1;
|
631
|
+
const a = color === 'transparent' ? 0 : 1;
|
647
632
|
return {
|
648
|
-
r:
|
633
|
+
r: 0, g: 0, b: 0, a, format: 'rgb', ok: true,
|
649
634
|
};
|
650
635
|
}
|
651
636
|
|
@@ -685,7 +670,6 @@
|
|
685
670
|
g: parseIntFromHex(m2),
|
686
671
|
b: parseIntFromHex(m3),
|
687
672
|
a: convertHexToDecimal(m4),
|
688
|
-
// format: named ? 'rgb' : 'hex8',
|
689
673
|
format: named ? 'rgb' : 'hex',
|
690
674
|
};
|
691
675
|
}
|
@@ -749,6 +733,7 @@
|
|
749
733
|
function inputToRGB(input) {
|
750
734
|
let rgb = { r: 0, g: 0, b: 0 };
|
751
735
|
let color = input;
|
736
|
+
/** @type {string | number} */
|
752
737
|
let a = 1;
|
753
738
|
let s = null;
|
754
739
|
let v = null;
|
@@ -759,7 +744,8 @@
|
|
759
744
|
let r = null;
|
760
745
|
let g = null;
|
761
746
|
let ok = false;
|
762
|
-
|
747
|
+
const inputFormat = typeof color === 'object' && color.format;
|
748
|
+
let format = inputFormat && COLOR_FORMAT.includes(inputFormat) ? inputFormat : 'rgb';
|
763
749
|
|
764
750
|
if (typeof input === 'string') {
|
765
751
|
// @ts-ignore -- this now is converted to object
|
@@ -800,14 +786,17 @@
|
|
800
786
|
format = 'hwb';
|
801
787
|
}
|
802
788
|
if (isValidCSSUnit(color.a)) {
|
803
|
-
a = color.a;
|
804
|
-
a = isPercentage(`${a}`) ? bound01(a, 100) : a;
|
789
|
+
a = color.a; // @ts-ignore -- `parseFloat` works with numbers too
|
790
|
+
a = isPercentage(`${a}`) || parseFloat(a) > 1 ? bound01(a, 100) : a;
|
805
791
|
}
|
806
792
|
}
|
793
|
+
if (typeof color === 'undefined') {
|
794
|
+
ok = true;
|
795
|
+
}
|
807
796
|
|
808
797
|
return {
|
809
|
-
ok,
|
810
|
-
format
|
798
|
+
ok,
|
799
|
+
format,
|
811
800
|
r: Math.min(255, Math.max(rgb.r, 0)),
|
812
801
|
g: Math.min(255, Math.max(rgb.g, 0)),
|
813
802
|
b: Math.min(255, Math.max(rgb.b, 0)),
|
@@ -836,7 +825,8 @@
|
|
836
825
|
color = inputToRGB(color);
|
837
826
|
}
|
838
827
|
if (typeof color === 'number') {
|
839
|
-
|
828
|
+
const len = `${color}`.length;
|
829
|
+
color = `#${(len === 2 ? '0' : '00')}${color}`;
|
840
830
|
}
|
841
831
|
const {
|
842
832
|
r, g, b, a, ok, format,
|
@@ -846,7 +836,7 @@
|
|
846
836
|
const self = this;
|
847
837
|
|
848
838
|
/** @type {CP.ColorInput} */
|
849
|
-
self.originalInput =
|
839
|
+
self.originalInput = input;
|
850
840
|
/** @type {number} */
|
851
841
|
self.r = r;
|
852
842
|
/** @type {number} */
|
@@ -1236,6 +1226,7 @@
|
|
1236
1226
|
isOnePointZero,
|
1237
1227
|
isPercentage,
|
1238
1228
|
isValidCSSUnit,
|
1229
|
+
isColorName,
|
1239
1230
|
pad2,
|
1240
1231
|
clamp01,
|
1241
1232
|
bound01,
|
@@ -1253,10 +1244,11 @@
|
|
1253
1244
|
hueToRgb,
|
1254
1245
|
hwbToRgb,
|
1255
1246
|
parseIntFromHex,
|
1256
|
-
numberInputToObject,
|
1257
1247
|
stringInputToObject,
|
1258
1248
|
inputToRGB,
|
1259
1249
|
roundPart,
|
1250
|
+
getElementStyle,
|
1251
|
+
setElementStyle,
|
1260
1252
|
ObjectAssign,
|
1261
1253
|
});
|
1262
1254
|
|
@@ -1386,24 +1378,6 @@
|
|
1386
1378
|
*/
|
1387
1379
|
const ariaValueNow = 'aria-valuenow';
|
1388
1380
|
|
1389
|
-
/**
|
1390
|
-
* A global namespace for aria-haspopup.
|
1391
|
-
* @type {string}
|
1392
|
-
*/
|
1393
|
-
const ariaHasPopup = 'aria-haspopup';
|
1394
|
-
|
1395
|
-
/**
|
1396
|
-
* A global namespace for aria-hidden.
|
1397
|
-
* @type {string}
|
1398
|
-
*/
|
1399
|
-
const ariaHidden = 'aria-hidden';
|
1400
|
-
|
1401
|
-
/**
|
1402
|
-
* A global namespace for aria-labelledby.
|
1403
|
-
* @type {string}
|
1404
|
-
*/
|
1405
|
-
const ariaLabelledBy = 'aria-labelledby';
|
1406
|
-
|
1407
1381
|
/**
|
1408
1382
|
* A global namespace for `ArrowDown` key.
|
1409
1383
|
* @type {string} e.which = 40 equivalent
|
@@ -1530,37 +1504,6 @@
|
|
1530
1504
|
*/
|
1531
1505
|
const focusoutEvent = 'focusout';
|
1532
1506
|
|
1533
|
-
// @ts-ignore
|
1534
|
-
const { userAgentData: uaDATA } = navigator;
|
1535
|
-
|
1536
|
-
/**
|
1537
|
-
* A global namespace for `userAgentData` object.
|
1538
|
-
*/
|
1539
|
-
const userAgentData = uaDATA;
|
1540
|
-
|
1541
|
-
const { userAgent: userAgentString } = navigator;
|
1542
|
-
|
1543
|
-
/**
|
1544
|
-
* A global namespace for `navigator.userAgent` string.
|
1545
|
-
*/
|
1546
|
-
const userAgent = userAgentString;
|
1547
|
-
|
1548
|
-
const mobileBrands = /iPhone|iPad|iPod|Android/i;
|
1549
|
-
let isMobileCheck = false;
|
1550
|
-
|
1551
|
-
if (userAgentData) {
|
1552
|
-
isMobileCheck = userAgentData.brands
|
1553
|
-
.some((/** @type {Record<String, any>} */x) => mobileBrands.test(x.brand));
|
1554
|
-
} else {
|
1555
|
-
isMobileCheck = mobileBrands.test(userAgent);
|
1556
|
-
}
|
1557
|
-
|
1558
|
-
/**
|
1559
|
-
* A global `boolean` for mobile detection.
|
1560
|
-
* @type {boolean}
|
1561
|
-
*/
|
1562
|
-
const isMobile = isMobileCheck;
|
1563
|
-
|
1564
1507
|
/**
|
1565
1508
|
* Returns the `document.documentElement` or the `<html>` element.
|
1566
1509
|
*
|
@@ -1745,30 +1688,6 @@
|
|
1745
1688
|
return lookUp.getElementsByClassName(selector);
|
1746
1689
|
}
|
1747
1690
|
|
1748
|
-
/**
|
1749
|
-
* This is a shortie for `document.createElementNS` method
|
1750
|
-
* which allows you to create a new `HTMLElement` for a given `tagName`
|
1751
|
-
* or based on an object with specific non-readonly attributes:
|
1752
|
-
* `id`, `className`, `textContent`, `style`, etc.
|
1753
|
-
* @see https://developer.mozilla.org/en-US/docs/Web/API/Document/createElementNS
|
1754
|
-
*
|
1755
|
-
* @param {string} namespace `namespaceURI` to associate with the new `HTMLElement`
|
1756
|
-
* @param {Record<string, string> | string} param `tagName` or object
|
1757
|
-
* @return {HTMLElement | Element} a new `HTMLElement` or `Element`
|
1758
|
-
*/
|
1759
|
-
function createElementNS(namespace, param) {
|
1760
|
-
if (typeof param === 'string') {
|
1761
|
-
return getDocument().createElementNS(namespace, param);
|
1762
|
-
}
|
1763
|
-
|
1764
|
-
const { tagName } = param;
|
1765
|
-
const attr = { ...param };
|
1766
|
-
const newElement = createElementNS(namespace, tagName);
|
1767
|
-
delete attr.tagName;
|
1768
|
-
ObjectAssign(newElement, attr);
|
1769
|
-
return newElement;
|
1770
|
-
}
|
1771
|
-
|
1772
1691
|
/**
|
1773
1692
|
* Shortcut for the `Element.dispatchEvent(Event)` method.
|
1774
1693
|
*
|
@@ -1886,12 +1805,11 @@
|
|
1886
1805
|
}
|
1887
1806
|
|
1888
1807
|
/**
|
1889
|
-
* Shortcut for `
|
1890
|
-
*
|
1891
|
-
* @
|
1892
|
-
* @returns {string} lowercase output string
|
1808
|
+
* Shortcut for `Object.keys()` static method.
|
1809
|
+
* @param {Record<string, any>} obj a target object
|
1810
|
+
* @returns {string[]}
|
1893
1811
|
*/
|
1894
|
-
const
|
1812
|
+
const ObjectKeys = (obj) => Object.keys(obj);
|
1895
1813
|
|
1896
1814
|
/**
|
1897
1815
|
* Utility to normalize component options.
|
@@ -1996,6 +1914,80 @@
|
|
1996
1914
|
*/
|
1997
1915
|
const removeAttribute = (element, attribute) => element.removeAttribute(attribute);
|
1998
1916
|
|
1917
|
+
/**
|
1918
|
+
* @class
|
1919
|
+
* Returns a color palette with a given set of parameters.
|
1920
|
+
* @example
|
1921
|
+
* new ColorPalette(0, 12, 10);
|
1922
|
+
* // => { hue: 0, hueSteps: 12, lightSteps: 10, colors: Array<Color> }
|
1923
|
+
*/
|
1924
|
+
class ColorPalette {
|
1925
|
+
/**
|
1926
|
+
* The `hue` parameter is optional, which would be set to 0.
|
1927
|
+
* @param {number[]} args represeinting hue, hueSteps, lightSteps
|
1928
|
+
* * `args.hue` the starting Hue [0, 360]
|
1929
|
+
* * `args.hueSteps` Hue Steps Count [5, 24]
|
1930
|
+
* * `args.lightSteps` Lightness Steps Count [5, 12]
|
1931
|
+
*/
|
1932
|
+
constructor(...args) {
|
1933
|
+
let hue = 0;
|
1934
|
+
let hueSteps = 12;
|
1935
|
+
let lightSteps = 10;
|
1936
|
+
let lightnessArray = [0.5];
|
1937
|
+
|
1938
|
+
if (args.length === 3) {
|
1939
|
+
[hue, hueSteps, lightSteps] = args;
|
1940
|
+
} else if (args.length === 2) {
|
1941
|
+
[hueSteps, lightSteps] = args;
|
1942
|
+
if ([hueSteps, lightSteps].some((n) => n < 1)) {
|
1943
|
+
throw TypeError('ColorPalette: when 2 arguments used, both must be larger than 0.');
|
1944
|
+
}
|
1945
|
+
} else {
|
1946
|
+
throw TypeError('ColorPalette requires minimum 2 arguments');
|
1947
|
+
}
|
1948
|
+
|
1949
|
+
/** @type {Color[]} */
|
1950
|
+
const colors = [];
|
1951
|
+
|
1952
|
+
const hueStep = 360 / hueSteps;
|
1953
|
+
const half = roundPart((lightSteps - (lightSteps % 2 ? 1 : 0)) / 2);
|
1954
|
+
const estimatedStep = 100 / (lightSteps + (lightSteps % 2 ? 0 : 1)) / 100;
|
1955
|
+
|
1956
|
+
let lightStep = 0.25;
|
1957
|
+
lightStep = [4, 5].includes(lightSteps) ? 0.2 : lightStep;
|
1958
|
+
lightStep = [6, 7].includes(lightSteps) ? 0.15 : lightStep;
|
1959
|
+
lightStep = [8, 9].includes(lightSteps) ? 0.11 : lightStep;
|
1960
|
+
lightStep = [10, 11].includes(lightSteps) ? 0.09 : lightStep;
|
1961
|
+
lightStep = [12, 13].includes(lightSteps) ? 0.075 : lightStep;
|
1962
|
+
lightStep = lightSteps > 13 ? estimatedStep : lightStep;
|
1963
|
+
|
1964
|
+
// light tints
|
1965
|
+
for (let i = 1; i < half + 1; i += 1) {
|
1966
|
+
lightnessArray = [...lightnessArray, (0.5 + lightStep * (i))];
|
1967
|
+
}
|
1968
|
+
|
1969
|
+
// dark tints
|
1970
|
+
for (let i = 1; i < lightSteps - half; i += 1) {
|
1971
|
+
lightnessArray = [(0.5 - lightStep * (i)), ...lightnessArray];
|
1972
|
+
}
|
1973
|
+
|
1974
|
+
// feed `colors` Array
|
1975
|
+
for (let i = 0; i < hueSteps; i += 1) {
|
1976
|
+
const currentHue = ((hue + i * hueStep) % 360) / 360;
|
1977
|
+
lightnessArray.forEach((l) => {
|
1978
|
+
colors.push(new Color({ h: currentHue, s: 1, l }));
|
1979
|
+
});
|
1980
|
+
}
|
1981
|
+
|
1982
|
+
this.hue = hue;
|
1983
|
+
this.hueSteps = hueSteps;
|
1984
|
+
this.lightSteps = lightSteps;
|
1985
|
+
this.colors = colors;
|
1986
|
+
}
|
1987
|
+
}
|
1988
|
+
|
1989
|
+
ObjectAssign(ColorPalette, { Color });
|
1990
|
+
|
1999
1991
|
/** @type {Record<string, string>} */
|
2000
1992
|
const colorPickerLabels = {
|
2001
1993
|
pickerLabel: 'Colour Picker',
|
@@ -2023,6 +2015,22 @@
|
|
2023
2015
|
*/
|
2024
2016
|
const colorNames = ['white', 'black', 'grey', 'red', 'orange', 'brown', 'gold', 'olive', 'yellow', 'lime', 'green', 'teal', 'cyan', 'blue', 'violet', 'magenta', 'pink'];
|
2025
2017
|
|
2018
|
+
const tabIndex = 'tabindex';
|
2019
|
+
|
2020
|
+
/**
|
2021
|
+
* Check if a string is valid JSON string.
|
2022
|
+
* @param {string} str the string input
|
2023
|
+
* @returns {boolean} the query result
|
2024
|
+
*/
|
2025
|
+
function isValidJSON(str) {
|
2026
|
+
try {
|
2027
|
+
JSON.parse(str);
|
2028
|
+
} catch (e) {
|
2029
|
+
return false;
|
2030
|
+
}
|
2031
|
+
return true;
|
2032
|
+
}
|
2033
|
+
|
2026
2034
|
/**
|
2027
2035
|
* Shortcut for `String.toUpperCase()`.
|
2028
2036
|
*
|
@@ -2031,6 +2039,48 @@
|
|
2031
2039
|
*/
|
2032
2040
|
const toUpperCase = (source) => source.toUpperCase();
|
2033
2041
|
|
2042
|
+
/**
|
2043
|
+
* A global namespace for aria-haspopup.
|
2044
|
+
* @type {string}
|
2045
|
+
*/
|
2046
|
+
const ariaHasPopup = 'aria-haspopup';
|
2047
|
+
|
2048
|
+
/**
|
2049
|
+
* A global namespace for aria-hidden.
|
2050
|
+
* @type {string}
|
2051
|
+
*/
|
2052
|
+
const ariaHidden = 'aria-hidden';
|
2053
|
+
|
2054
|
+
/**
|
2055
|
+
* A global namespace for aria-labelledby.
|
2056
|
+
* @type {string}
|
2057
|
+
*/
|
2058
|
+
const ariaLabelledBy = 'aria-labelledby';
|
2059
|
+
|
2060
|
+
/**
|
2061
|
+
* This is a shortie for `document.createElementNS` method
|
2062
|
+
* which allows you to create a new `HTMLElement` for a given `tagName`
|
2063
|
+
* or based on an object with specific non-readonly attributes:
|
2064
|
+
* `id`, `className`, `textContent`, `style`, etc.
|
2065
|
+
* @see https://developer.mozilla.org/en-US/docs/Web/API/Document/createElementNS
|
2066
|
+
*
|
2067
|
+
* @param {string} namespace `namespaceURI` to associate with the new `HTMLElement`
|
2068
|
+
* @param {Record<string, string> | string} param `tagName` or object
|
2069
|
+
* @return {HTMLElement | Element} a new `HTMLElement` or `Element`
|
2070
|
+
*/
|
2071
|
+
function createElementNS(namespace, param) {
|
2072
|
+
if (typeof param === 'string') {
|
2073
|
+
return getDocument().createElementNS(namespace, param);
|
2074
|
+
}
|
2075
|
+
|
2076
|
+
const { tagName } = param;
|
2077
|
+
const attr = { ...param };
|
2078
|
+
const newElement = createElementNS(namespace, tagName);
|
2079
|
+
delete attr.tagName;
|
2080
|
+
ObjectAssign(newElement, attr);
|
2081
|
+
return newElement;
|
2082
|
+
}
|
2083
|
+
|
2034
2084
|
const vHidden = 'v-hidden';
|
2035
2085
|
|
2036
2086
|
/**
|
@@ -2110,8 +2160,6 @@
|
|
2110
2160
|
*/
|
2111
2161
|
const ariaValueMax = 'aria-valuemax';
|
2112
2162
|
|
2113
|
-
const tabIndex = 'tabindex';
|
2114
|
-
|
2115
2163
|
/**
|
2116
2164
|
* Returns all color controls for `ColorPicker`.
|
2117
2165
|
*
|
@@ -2219,75 +2267,6 @@
|
|
2219
2267
|
});
|
2220
2268
|
}
|
2221
2269
|
|
2222
|
-
/**
|
2223
|
-
* @class
|
2224
|
-
* Returns a color palette with a given set of parameters.
|
2225
|
-
* @example
|
2226
|
-
* new ColorPalette(0, 12, 10);
|
2227
|
-
* // => { hue: 0, hueSteps: 12, lightSteps: 10, colors: array }
|
2228
|
-
*/
|
2229
|
-
class ColorPalette {
|
2230
|
-
/**
|
2231
|
-
* The `hue` parameter is optional, which would be set to 0.
|
2232
|
-
* @param {number[]} args represeinting hue, hueSteps, lightSteps
|
2233
|
-
* * `args.hue` the starting Hue [0, 360]
|
2234
|
-
* * `args.hueSteps` Hue Steps Count [5, 24]
|
2235
|
-
* * `args.lightSteps` Lightness Steps Count [5, 12]
|
2236
|
-
*/
|
2237
|
-
constructor(...args) {
|
2238
|
-
let hue = 0;
|
2239
|
-
let hueSteps = 12;
|
2240
|
-
let lightSteps = 10;
|
2241
|
-
let lightnessArray = [0.5];
|
2242
|
-
|
2243
|
-
if (args.length === 3) {
|
2244
|
-
[hue, hueSteps, lightSteps] = args;
|
2245
|
-
} else if (args.length === 2) {
|
2246
|
-
[hueSteps, lightSteps] = args;
|
2247
|
-
} else {
|
2248
|
-
throw TypeError('ColorPalette requires minimum 2 arguments');
|
2249
|
-
}
|
2250
|
-
|
2251
|
-
/** @type {string[]} */
|
2252
|
-
const colors = [];
|
2253
|
-
|
2254
|
-
const hueStep = 360 / hueSteps;
|
2255
|
-
const half = roundPart((lightSteps - (lightSteps % 2 ? 1 : 0)) / 2);
|
2256
|
-
const estimatedStep = 100 / (lightSteps + (lightSteps % 2 ? 0 : 1)) / 100;
|
2257
|
-
|
2258
|
-
let lightStep = 0.25;
|
2259
|
-
lightStep = [4, 5].includes(lightSteps) ? 0.2 : lightStep;
|
2260
|
-
lightStep = [6, 7].includes(lightSteps) ? 0.15 : lightStep;
|
2261
|
-
lightStep = [8, 9].includes(lightSteps) ? 0.11 : lightStep;
|
2262
|
-
lightStep = [10, 11].includes(lightSteps) ? 0.09 : lightStep;
|
2263
|
-
lightStep = [12, 13].includes(lightSteps) ? 0.075 : lightStep;
|
2264
|
-
lightStep = lightSteps > 13 ? estimatedStep : lightStep;
|
2265
|
-
|
2266
|
-
// light tints
|
2267
|
-
for (let i = 1; i < half + 1; i += 1) {
|
2268
|
-
lightnessArray = [...lightnessArray, (0.5 + lightStep * (i))];
|
2269
|
-
}
|
2270
|
-
|
2271
|
-
// dark tints
|
2272
|
-
for (let i = 1; i < lightSteps - half; i += 1) {
|
2273
|
-
lightnessArray = [(0.5 - lightStep * (i)), ...lightnessArray];
|
2274
|
-
}
|
2275
|
-
|
2276
|
-
// feed `colors` Array
|
2277
|
-
for (let i = 0; i < hueSteps; i += 1) {
|
2278
|
-
const currentHue = ((hue + i * hueStep) % 360) / 360;
|
2279
|
-
lightnessArray.forEach((l) => {
|
2280
|
-
colors.push(new Color({ h: currentHue, s: 1, l }).toHexString());
|
2281
|
-
});
|
2282
|
-
}
|
2283
|
-
|
2284
|
-
this.hue = hue;
|
2285
|
-
this.hueSteps = hueSteps;
|
2286
|
-
this.lightSteps = lightSteps;
|
2287
|
-
this.colors = colors;
|
2288
|
-
}
|
2289
|
-
}
|
2290
|
-
|
2291
2270
|
/**
|
2292
2271
|
* Returns a color-defaults with given values and class.
|
2293
2272
|
* @param {CP.ColorPicker} self
|
@@ -2321,7 +2300,8 @@
|
|
2321
2300
|
optionSize = fit > 5 && isMultiLine ? 1.5 : optionSize;
|
2322
2301
|
const menuHeight = `${(rowCount || 1) * optionSize}rem`;
|
2323
2302
|
const menuHeightHover = `calc(${rowCountHover} * ${optionSize}rem + ${rowCountHover - 1} * ${gap})`;
|
2324
|
-
|
2303
|
+
/** @type {HTMLUListElement} */
|
2304
|
+
// @ts-ignore -- <UL> is an `HTMLElement`
|
2325
2305
|
const menu = createElement({
|
2326
2306
|
tagName: 'ul',
|
2327
2307
|
className: finalClass,
|
@@ -2329,7 +2309,7 @@
|
|
2329
2309
|
setAttribute(menu, 'role', 'listbox');
|
2330
2310
|
setAttribute(menu, ariaLabel, menuLabel);
|
2331
2311
|
|
2332
|
-
if (isScrollable) {
|
2312
|
+
if (isScrollable) {
|
2333
2313
|
setCSSProperties(menu, {
|
2334
2314
|
'--grid-item-size': `${optionSize}rem`,
|
2335
2315
|
'--grid-fit': fit,
|
@@ -2340,15 +2320,19 @@
|
|
2340
2320
|
}
|
2341
2321
|
|
2342
2322
|
colorsArray.forEach((x) => {
|
2343
|
-
|
2344
|
-
|
2345
|
-
|
2323
|
+
let [value, label] = typeof x === 'string' ? x.trim().split(':') : [];
|
2324
|
+
if (x instanceof Color) {
|
2325
|
+
value = x.toHexString();
|
2326
|
+
label = value;
|
2327
|
+
}
|
2328
|
+
const color = new Color(x instanceof Color ? x : value, format);
|
2329
|
+
const isActive = color.toString() === getAttribute(input, 'value');
|
2346
2330
|
const active = isActive ? ' active' : '';
|
2347
2331
|
|
2348
2332
|
const option = createElement({
|
2349
2333
|
tagName: 'li',
|
2350
2334
|
className: `color-option${active}`,
|
2351
|
-
innerText: `${label ||
|
2335
|
+
innerText: `${label || value}`,
|
2352
2336
|
});
|
2353
2337
|
|
2354
2338
|
setAttribute(option, tabIndex, '0');
|
@@ -2357,7 +2341,7 @@
|
|
2357
2341
|
setAttribute(option, ariaSelected, isActive ? 'true' : 'false');
|
2358
2342
|
|
2359
2343
|
if (isOptionsMenu) {
|
2360
|
-
setElementStyle(option, { backgroundColor:
|
2344
|
+
setElementStyle(option, { backgroundColor: value });
|
2361
2345
|
}
|
2362
2346
|
|
2363
2347
|
menu.append(option);
|
@@ -2366,55 +2350,10 @@
|
|
2366
2350
|
}
|
2367
2351
|
|
2368
2352
|
/**
|
2369
|
-
|
2370
|
-
|
2371
|
-
|
2372
|
-
|
2373
|
-
function isValidJSON(str) {
|
2374
|
-
try {
|
2375
|
-
JSON.parse(str);
|
2376
|
-
} catch (e) {
|
2377
|
-
return false;
|
2378
|
-
}
|
2379
|
-
return true;
|
2380
|
-
}
|
2381
|
-
|
2382
|
-
var version = "0.0.1";
|
2383
|
-
|
2384
|
-
// @ts-ignore
|
2385
|
-
|
2386
|
-
const Version = version;
|
2387
|
-
|
2388
|
-
// ColorPicker GC
|
2389
|
-
// ==============
|
2390
|
-
const colorPickerString = 'color-picker';
|
2391
|
-
const colorPickerSelector = `[data-function="${colorPickerString}"]`;
|
2392
|
-
const colorPickerParentSelector = `.${colorPickerString},${colorPickerString}`;
|
2393
|
-
const colorPickerDefaults = {
|
2394
|
-
componentLabels: colorPickerLabels,
|
2395
|
-
colorLabels: colorNames,
|
2396
|
-
format: 'rgb',
|
2397
|
-
colorPresets: false,
|
2398
|
-
colorKeywords: false,
|
2399
|
-
};
|
2400
|
-
|
2401
|
-
// ColorPicker Static Methods
|
2402
|
-
// ==========================
|
2403
|
-
|
2404
|
-
/** @type {CP.GetInstance<ColorPicker>} */
|
2405
|
-
const getColorPickerInstance = (element) => getInstance(element, colorPickerString);
|
2406
|
-
|
2407
|
-
/** @type {CP.InitCallback<ColorPicker>} */
|
2408
|
-
const initColorPicker = (element) => new ColorPicker(element);
|
2409
|
-
|
2410
|
-
// ColorPicker Private Methods
|
2411
|
-
// ===========================
|
2412
|
-
|
2413
|
-
/**
|
2414
|
-
* Generate HTML markup and update instance properties.
|
2415
|
-
* @param {ColorPicker} self
|
2416
|
-
*/
|
2417
|
-
function initCallback(self) {
|
2353
|
+
* Generate HTML markup and update instance properties.
|
2354
|
+
* @param {CP.ColorPicker} self
|
2355
|
+
*/
|
2356
|
+
function setMarkup(self) {
|
2418
2357
|
const {
|
2419
2358
|
input, parent, format, id, componentLabels, colorKeywords, colorPresets,
|
2420
2359
|
} = self;
|
@@ -2429,9 +2368,7 @@
|
|
2429
2368
|
self.color = new Color(color, format);
|
2430
2369
|
|
2431
2370
|
// set initial controls dimensions
|
2432
|
-
|
2433
|
-
const dropClass = isMobile ? ' mobile' : '';
|
2434
|
-
const formatString = format === 'hex' ? hexLabel : format.toUpperCase();
|
2371
|
+
const formatString = format === 'hex' ? hexLabel : toUpperCase(format);
|
2435
2372
|
|
2436
2373
|
const pickerBtn = createElement({
|
2437
2374
|
id: `picker-btn-${id}`,
|
@@ -2448,7 +2385,7 @@
|
|
2448
2385
|
|
2449
2386
|
const pickerDropdown = createElement({
|
2450
2387
|
tagName: 'div',
|
2451
|
-
className:
|
2388
|
+
className: 'color-dropdown picker',
|
2452
2389
|
});
|
2453
2390
|
setAttribute(pickerDropdown, ariaLabelledBy, `picker-btn-${id}`);
|
2454
2391
|
setAttribute(pickerDropdown, 'role', 'group');
|
@@ -2464,7 +2401,7 @@
|
|
2464
2401
|
if (colorKeywords || colorPresets) {
|
2465
2402
|
const presetsDropdown = createElement({
|
2466
2403
|
tagName: 'div',
|
2467
|
-
className:
|
2404
|
+
className: 'color-dropdown scrollable menu',
|
2468
2405
|
});
|
2469
2406
|
|
2470
2407
|
// color presets
|
@@ -2514,6 +2451,37 @@
|
|
2514
2451
|
setAttribute(input, tabIndex, '-1');
|
2515
2452
|
}
|
2516
2453
|
|
2454
|
+
var version = "0.0.2alpha1";
|
2455
|
+
|
2456
|
+
// @ts-ignore
|
2457
|
+
|
2458
|
+
const Version = version;
|
2459
|
+
|
2460
|
+
// ColorPicker GC
|
2461
|
+
// ==============
|
2462
|
+
const colorPickerString = 'color-picker';
|
2463
|
+
const colorPickerSelector = `[data-function="${colorPickerString}"]`;
|
2464
|
+
const colorPickerParentSelector = `.${colorPickerString},${colorPickerString}`;
|
2465
|
+
const colorPickerDefaults = {
|
2466
|
+
componentLabels: colorPickerLabels,
|
2467
|
+
colorLabels: colorNames,
|
2468
|
+
format: 'rgb',
|
2469
|
+
colorPresets: false,
|
2470
|
+
colorKeywords: false,
|
2471
|
+
};
|
2472
|
+
|
2473
|
+
// ColorPicker Static Methods
|
2474
|
+
// ==========================
|
2475
|
+
|
2476
|
+
/** @type {CP.GetInstance<ColorPicker>} */
|
2477
|
+
const getColorPickerInstance = (element) => getInstance(element, colorPickerString);
|
2478
|
+
|
2479
|
+
/** @type {CP.InitCallback<ColorPicker>} */
|
2480
|
+
const initColorPicker = (element) => new ColorPicker(element);
|
2481
|
+
|
2482
|
+
// ColorPicker Private Methods
|
2483
|
+
// ===========================
|
2484
|
+
|
2517
2485
|
/**
|
2518
2486
|
* Add / remove `ColorPicker` main event listeners.
|
2519
2487
|
* @param {ColorPicker} self
|
@@ -2747,7 +2715,7 @@
|
|
2747
2715
|
self.handleKnobs = self.handleKnobs.bind(self);
|
2748
2716
|
|
2749
2717
|
// generate markup
|
2750
|
-
|
2718
|
+
setMarkup(self);
|
2751
2719
|
|
2752
2720
|
const [colorPicker, colorMenu] = getElementsByClassName('color-dropdown', parent);
|
2753
2721
|
// set main elements
|
@@ -2943,7 +2911,7 @@
|
|
2943
2911
|
const self = this;
|
2944
2912
|
const { activeElement } = getDocument(self.input);
|
2945
2913
|
|
2946
|
-
if ((
|
2914
|
+
if ((e.type === touchmoveEvent && self.dragElement)
|
2947
2915
|
|| (activeElement && self.controlKnobs.includes(activeElement))) {
|
2948
2916
|
e.stopPropagation();
|
2949
2917
|
e.preventDefault();
|
@@ -3856,4 +3824,4 @@
|
|
3856
3824
|
|
3857
3825
|
return ColorPickerElement;
|
3858
3826
|
|
3859
|
-
}))
|
3827
|
+
}));
|