463 |
464 |
465 | toHEXString =
466 |
467 | toRGBString =
468 |
469 | R, G, B =
470 |
471 | H, S, V =
472 |
473 |
474 |
475 |
476 |
477 |
488 |
489 |
494 |
495 |
496 |
497 |
498 |
499 |
500 |
959 |
960 |
961 |
--------------------------------------------------------------------------------
/data/jscolor.js:
--------------------------------------------------------------------------------
1 | /**
2 | * jscolor - JavaScript Color Picker
3 | *
4 | * @link http://jscolor.com
5 | * @license For open source use: GPLv3
6 | * For commercial use: JSColor Commercial License
7 | * @author Jan Odvarko
8 | * @version 2.0.4
9 | *
10 | * See usage examples at http://jscolor.com/examples/
11 | */
12 |
13 |
14 | "use strict";
15 |
16 |
17 | if (!window.jscolor) { window.jscolor = (function () {
18 |
19 |
20 | var jsc = {
21 |
22 |
23 | register : function () {
24 | jsc.attachDOMReadyEvent(jsc.init);
25 | jsc.attachEvent(document, 'mousedown', jsc.onDocumentMouseDown);
26 | jsc.attachEvent(document, 'touchstart', jsc.onDocumentTouchStart);
27 | jsc.attachEvent(window, 'resize', jsc.onWindowResize);
28 | },
29 |
30 |
31 | init : function () {
32 | if (jsc.jscolor.lookupClass) {
33 | jsc.jscolor.installByClassName(jsc.jscolor.lookupClass);
34 | }
35 | },
36 |
37 |
38 | tryInstallOnElements : function (elms, className) {
39 | var matchClass = new RegExp('(^|\\s)(' + className + ')(\\s*(\\{[^}]*\\})|\\s|$)', 'i');
40 |
41 | for (var i = 0; i < elms.length; i += 1) {
42 | if (elms[i].type !== undefined && elms[i].type.toLowerCase() == 'color') {
43 | if (jsc.isColorAttrSupported) {
44 | // skip inputs of type 'color' if supported by the browser
45 | continue;
46 | }
47 | }
48 | var m;
49 | if (!elms[i].jscolor && elms[i].className && (m = elms[i].className.match(matchClass))) {
50 | var targetElm = elms[i];
51 | var optsStr = null;
52 |
53 | var dataOptions = jsc.getDataAttr(targetElm, 'jscolor');
54 | if (dataOptions !== null) {
55 | optsStr = dataOptions;
56 | } else if (m[4]) {
57 | optsStr = m[4];
58 | }
59 |
60 | var opts = {};
61 | if (optsStr) {
62 | try {
63 | opts = (new Function ('return (' + optsStr + ')'))();
64 | } catch(eParseError) {
65 | jsc.warn('Error parsing jscolor options: ' + eParseError + ':\n' + optsStr);
66 | }
67 | }
68 | targetElm.jscolor = new jsc.jscolor(targetElm, opts);
69 | }
70 | }
71 | },
72 |
73 |
74 | isColorAttrSupported : (function () {
75 | var elm = document.createElement('input');
76 | if (elm.setAttribute) {
77 | elm.setAttribute('type', 'color');
78 | if (elm.type.toLowerCase() == 'color') {
79 | return true;
80 | }
81 | }
82 | return false;
83 | })(),
84 |
85 |
86 | isCanvasSupported : (function () {
87 | var elm = document.createElement('canvas');
88 | return !!(elm.getContext && elm.getContext('2d'));
89 | })(),
90 |
91 |
92 | fetchElement : function (mixed) {
93 | return typeof mixed === 'string' ? document.getElementById(mixed) : mixed;
94 | },
95 |
96 |
97 | isElementType : function (elm, type) {
98 | return elm.nodeName.toLowerCase() === type.toLowerCase();
99 | },
100 |
101 |
102 | getDataAttr : function (el, name) {
103 | var attrName = 'data-' + name;
104 | var attrValue = el.getAttribute(attrName);
105 | if (attrValue !== null) {
106 | return attrValue;
107 | }
108 | return null;
109 | },
110 |
111 |
112 | attachEvent : function (el, evnt, func) {
113 | if (el.addEventListener) {
114 | el.addEventListener(evnt, func, false);
115 | } else if (el.attachEvent) {
116 | el.attachEvent('on' + evnt, func);
117 | }
118 | },
119 |
120 |
121 | detachEvent : function (el, evnt, func) {
122 | if (el.removeEventListener) {
123 | el.removeEventListener(evnt, func, false);
124 | } else if (el.detachEvent) {
125 | el.detachEvent('on' + evnt, func);
126 | }
127 | },
128 |
129 |
130 | _attachedGroupEvents : {},
131 |
132 |
133 | attachGroupEvent : function (groupName, el, evnt, func) {
134 | if (!jsc._attachedGroupEvents.hasOwnProperty(groupName)) {
135 | jsc._attachedGroupEvents[groupName] = [];
136 | }
137 | jsc._attachedGroupEvents[groupName].push([el, evnt, func]);
138 | jsc.attachEvent(el, evnt, func);
139 | },
140 |
141 |
142 | detachGroupEvents : function (groupName) {
143 | if (jsc._attachedGroupEvents.hasOwnProperty(groupName)) {
144 | for (var i = 0; i < jsc._attachedGroupEvents[groupName].length; i += 1) {
145 | var evt = jsc._attachedGroupEvents[groupName][i];
146 | jsc.detachEvent(evt[0], evt[1], evt[2]);
147 | }
148 | delete jsc._attachedGroupEvents[groupName];
149 | }
150 | },
151 |
152 |
153 | attachDOMReadyEvent : function (func) {
154 | var fired = false;
155 | var fireOnce = function () {
156 | if (!fired) {
157 | fired = true;
158 | func();
159 | }
160 | };
161 |
162 | if (document.readyState === 'complete') {
163 | setTimeout(fireOnce, 1); // async
164 | return;
165 | }
166 |
167 | if (document.addEventListener) {
168 | document.addEventListener('DOMContentLoaded', fireOnce, false);
169 |
170 | // Fallback
171 | window.addEventListener('load', fireOnce, false);
172 |
173 | } else if (document.attachEvent) {
174 | // IE
175 | document.attachEvent('onreadystatechange', function () {
176 | if (document.readyState === 'complete') {
177 | document.detachEvent('onreadystatechange', arguments.callee);
178 | fireOnce();
179 | }
180 | })
181 |
182 | // Fallback
183 | window.attachEvent('onload', fireOnce);
184 |
185 | // IE7/8
186 | if (document.documentElement.doScroll && window == window.top) {
187 | var tryScroll = function () {
188 | if (!document.body) { return; }
189 | try {
190 | document.documentElement.doScroll('left');
191 | fireOnce();
192 | } catch (e) {
193 | setTimeout(tryScroll, 1);
194 | }
195 | };
196 | tryScroll();
197 | }
198 | }
199 | },
200 |
201 |
202 | warn : function (msg) {
203 | if (window.console && window.console.warn) {
204 | window.console.warn(msg);
205 | }
206 | },
207 |
208 |
209 | preventDefault : function (e) {
210 | if (e.preventDefault) { e.preventDefault(); }
211 | e.returnValue = false;
212 | },
213 |
214 |
215 | captureTarget : function (target) {
216 | // IE
217 | if (target.setCapture) {
218 | jsc._capturedTarget = target;
219 | jsc._capturedTarget.setCapture();
220 | }
221 | },
222 |
223 |
224 | releaseTarget : function () {
225 | // IE
226 | if (jsc._capturedTarget) {
227 | jsc._capturedTarget.releaseCapture();
228 | jsc._capturedTarget = null;
229 | }
230 | },
231 |
232 |
233 | fireEvent : function (el, evnt) {
234 | if (!el) {
235 | return;
236 | }
237 | if (document.createEvent) {
238 | var ev = document.createEvent('HTMLEvents');
239 | ev.initEvent(evnt, true, true);
240 | el.dispatchEvent(ev);
241 | } else if (document.createEventObject) {
242 | var ev = document.createEventObject();
243 | el.fireEvent('on' + evnt, ev);
244 | } else if (el['on' + evnt]) { // alternatively use the traditional event model
245 | el['on' + evnt]();
246 | }
247 | },
248 |
249 |
250 | classNameToList : function (className) {
251 | return className.replace(/^\s+|\s+$/g, '').split(/\s+/);
252 | },
253 |
254 |
255 | // The className parameter (str) can only contain a single class name
256 | hasClass : function (elm, className) {
257 | if (!className) {
258 | return false;
259 | }
260 | return -1 != (' ' + elm.className.replace(/\s+/g, ' ') + ' ').indexOf(' ' + className + ' ');
261 | },
262 |
263 |
264 | // The className parameter (str) can contain multiple class names separated by whitespace
265 | setClass : function (elm, className) {
266 | var classList = jsc.classNameToList(className);
267 | for (var i = 0; i < classList.length; i += 1) {
268 | if (!jsc.hasClass(elm, classList[i])) {
269 | elm.className += (elm.className ? ' ' : '') + classList[i];
270 | }
271 | }
272 | },
273 |
274 |
275 | // The className parameter (str) can contain multiple class names separated by whitespace
276 | unsetClass : function (elm, className) {
277 | var classList = jsc.classNameToList(className);
278 | for (var i = 0; i < classList.length; i += 1) {
279 | var repl = new RegExp(
280 | '^\\s*' + classList[i] + '\\s*|' +
281 | '\\s*' + classList[i] + '\\s*$|' +
282 | '\\s+' + classList[i] + '(\\s+)',
283 | 'g'
284 | );
285 | elm.className = elm.className.replace(repl, '$1');
286 | }
287 | },
288 |
289 |
290 | getStyle : function (elm) {
291 | return window.getComputedStyle ? window.getComputedStyle(elm) : elm.currentStyle;
292 | },
293 |
294 |
295 | setStyle : (function () {
296 | var helper = document.createElement('div');
297 | var getSupportedProp = function (names) {
298 | for (var i = 0; i < names.length; i += 1) {
299 | if (names[i] in helper.style) {
300 | return names[i];
301 | }
302 | }
303 | };
304 | var props = {
305 | borderRadius: getSupportedProp(['borderRadius', 'MozBorderRadius', 'webkitBorderRadius']),
306 | boxShadow: getSupportedProp(['boxShadow', 'MozBoxShadow', 'webkitBoxShadow'])
307 | };
308 | return function (elm, prop, value) {
309 | switch (prop.toLowerCase()) {
310 | case 'opacity':
311 | var alphaOpacity = Math.round(parseFloat(value) * 100);
312 | elm.style.opacity = value;
313 | elm.style.filter = 'alpha(opacity=' + alphaOpacity + ')';
314 | break;
315 | default:
316 | elm.style[props[prop]] = value;
317 | break;
318 | }
319 | };
320 | })(),
321 |
322 |
323 | setBorderRadius : function (elm, value) {
324 | jsc.setStyle(elm, 'borderRadius', value || '0');
325 | },
326 |
327 |
328 | setBoxShadow : function (elm, value) {
329 | jsc.setStyle(elm, 'boxShadow', value || 'none');
330 | },
331 |
332 |
333 | getElementPos : function (e, relativeToViewport) {
334 | var x=0, y=0;
335 | var rect = e.getBoundingClientRect();
336 | x = rect.left;
337 | y = rect.top;
338 | if (!relativeToViewport) {
339 | var viewPos = jsc.getViewPos();
340 | x += viewPos[0];
341 | y += viewPos[1];
342 | }
343 | return [x, y];
344 | },
345 |
346 |
347 | getElementSize : function (e) {
348 | return [e.offsetWidth, e.offsetHeight];
349 | },
350 |
351 |
352 | // get pointer's X/Y coordinates relative to viewport
353 | getAbsPointerPos : function (e) {
354 | if (!e) { e = window.event; }
355 | var x = 0, y = 0;
356 | if (typeof e.changedTouches !== 'undefined' && e.changedTouches.length) {
357 | // touch devices
358 | x = e.changedTouches[0].clientX;
359 | y = e.changedTouches[0].clientY;
360 | } else if (typeof e.clientX === 'number') {
361 | x = e.clientX;
362 | y = e.clientY;
363 | }
364 | return { x: x, y: y };
365 | },
366 |
367 |
368 | // get pointer's X/Y coordinates relative to target element
369 | getRelPointerPos : function (e) {
370 | if (!e) { e = window.event; }
371 | var target = e.target || e.srcElement;
372 | var targetRect = target.getBoundingClientRect();
373 |
374 | var x = 0, y = 0;
375 |
376 | var clientX = 0, clientY = 0;
377 | if (typeof e.changedTouches !== 'undefined' && e.changedTouches.length) {
378 | // touch devices
379 | clientX = e.changedTouches[0].clientX;
380 | clientY = e.changedTouches[0].clientY;
381 | } else if (typeof e.clientX === 'number') {
382 | clientX = e.clientX;
383 | clientY = e.clientY;
384 | }
385 |
386 | x = clientX - targetRect.left;
387 | y = clientY - targetRect.top;
388 | return { x: x, y: y };
389 | },
390 |
391 |
392 | getViewPos : function () {
393 | var doc = document.documentElement;
394 | return [
395 | (window.pageXOffset || doc.scrollLeft) - (doc.clientLeft || 0),
396 | (window.pageYOffset || doc.scrollTop) - (doc.clientTop || 0)
397 | ];
398 | },
399 |
400 |
401 | getViewSize : function () {
402 | var doc = document.documentElement;
403 | return [
404 | (window.innerWidth || doc.clientWidth),
405 | (window.innerHeight || doc.clientHeight),
406 | ];
407 | },
408 |
409 |
410 | redrawPosition : function () {
411 |
412 | if (jsc.picker && jsc.picker.owner) {
413 | var thisObj = jsc.picker.owner;
414 |
415 | var tp, vp;
416 |
417 | if (thisObj.fixed) {
418 | // Fixed elements are positioned relative to viewport,
419 | // therefore we can ignore the scroll offset
420 | tp = jsc.getElementPos(thisObj.targetElement, true); // target pos
421 | vp = [0, 0]; // view pos
422 | } else {
423 | tp = jsc.getElementPos(thisObj.targetElement); // target pos
424 | vp = jsc.getViewPos(); // view pos
425 | }
426 |
427 | var ts = jsc.getElementSize(thisObj.targetElement); // target size
428 | var vs = jsc.getViewSize(); // view size
429 | var ps = jsc.getPickerOuterDims(thisObj); // picker size
430 | var a, b, c;
431 | switch (thisObj.position.toLowerCase()) {
432 | case 'left': a=1; b=0; c=-1; break;
433 | case 'right':a=1; b=0; c=1; break;
434 | case 'top': a=0; b=1; c=-1; break;
435 | default: a=0; b=1; c=1; break;
436 | }
437 | var l = (ts[b]+ps[b])/2;
438 |
439 | // compute picker position
440 | if (!thisObj.smartPosition) {
441 | var pp = [
442 | tp[a],
443 | tp[b]+ts[b]-l+l*c
444 | ];
445 | } else {
446 | var pp = [
447 | -vp[a]+tp[a]+ps[a] > vs[a] ?
448 | (-vp[a]+tp[a]+ts[a]/2 > vs[a]/2 && tp[a]+ts[a]-ps[a] >= 0 ? tp[a]+ts[a]-ps[a] : tp[a]) :
449 | tp[a],
450 | -vp[b]+tp[b]+ts[b]+ps[b]-l+l*c > vs[b] ?
451 | (-vp[b]+tp[b]+ts[b]/2 > vs[b]/2 && tp[b]+ts[b]-l-l*c >= 0 ? tp[b]+ts[b]-l-l*c : tp[b]+ts[b]-l+l*c) :
452 | (tp[b]+ts[b]-l+l*c >= 0 ? tp[b]+ts[b]-l+l*c : tp[b]+ts[b]-l-l*c)
453 | ];
454 | }
455 |
456 | var x = pp[a];
457 | var y = pp[b];
458 | var positionValue = thisObj.fixed ? 'fixed' : 'absolute';
459 | var contractShadow =
460 | (pp[0] + ps[0] > tp[0] || pp[0] < tp[0] + ts[0]) &&
461 | (pp[1] + ps[1] < tp[1] + ts[1]);
462 |
463 | jsc._drawPosition(thisObj, x, y, positionValue, contractShadow);
464 | }
465 | },
466 |
467 |
468 | _drawPosition : function (thisObj, x, y, positionValue, contractShadow) {
469 | var vShadow = contractShadow ? 0 : thisObj.shadowBlur; // px
470 |
471 | jsc.picker.wrap.style.position = positionValue;
472 | jsc.picker.wrap.style.left = x + 'px';
473 | jsc.picker.wrap.style.top = y + 'px';
474 |
475 | jsc.setBoxShadow(
476 | jsc.picker.boxS,
477 | thisObj.shadow ?
478 | new jsc.BoxShadow(0, vShadow, thisObj.shadowBlur, 0, thisObj.shadowColor) :
479 | null);
480 | },
481 |
482 |
483 | getPickerDims : function (thisObj) {
484 | var displaySlider = !!jsc.getSliderComponent(thisObj);
485 | var dims = [
486 | 2 * thisObj.insetWidth + 2 * thisObj.padding + thisObj.width +
487 | (displaySlider ? 2 * thisObj.insetWidth + jsc.getPadToSliderPadding(thisObj) + thisObj.sliderSize : 0),
488 | 2 * thisObj.insetWidth + 2 * thisObj.padding + thisObj.height +
489 | (thisObj.closable ? 2 * thisObj.insetWidth + thisObj.padding + thisObj.buttonHeight : 0)
490 | ];
491 | return dims;
492 | },
493 |
494 |
495 | getPickerOuterDims : function (thisObj) {
496 | var dims = jsc.getPickerDims(thisObj);
497 | return [
498 | dims[0] + 2 * thisObj.borderWidth,
499 | dims[1] + 2 * thisObj.borderWidth
500 | ];
501 | },
502 |
503 |
504 | getPadToSliderPadding : function (thisObj) {
505 | return Math.max(thisObj.padding, 1.5 * (2 * thisObj.pointerBorderWidth + thisObj.pointerThickness));
506 | },
507 |
508 |
509 | getPadYComponent : function (thisObj) {
510 | switch (thisObj.mode.charAt(1).toLowerCase()) {
511 | case 'v': return 'v'; break;
512 | }
513 | return 's';
514 | },
515 |
516 |
517 | getSliderComponent : function (thisObj) {
518 | if (thisObj.mode.length > 2) {
519 | switch (thisObj.mode.charAt(2).toLowerCase()) {
520 | case 's': return 's'; break;
521 | case 'v': return 'v'; break;
522 | }
523 | }
524 | return null;
525 | },
526 |
527 |
528 | onDocumentMouseDown : function (e) {
529 | if (!e) { e = window.event; }
530 | var target = e.target || e.srcElement;
531 |
532 | if (target._jscLinkedInstance) {
533 | if (target._jscLinkedInstance.showOnClick) {
534 | target._jscLinkedInstance.show();
535 | }
536 | } else if (target._jscControlName) {
537 | jsc.onControlPointerStart(e, target, target._jscControlName, 'mouse');
538 | } else {
539 | // Mouse is outside the picker controls -> hide the color picker!
540 | if (jsc.picker && jsc.picker.owner) {
541 | jsc.picker.owner.hide();
542 | }
543 | }
544 | },
545 |
546 |
547 | onDocumentTouchStart : function (e) {
548 | if (!e) { e = window.event; }
549 | var target = e.target || e.srcElement;
550 |
551 | if (target._jscLinkedInstance) {
552 | if (target._jscLinkedInstance.showOnClick) {
553 | target._jscLinkedInstance.show();
554 | }
555 | } else if (target._jscControlName) {
556 | jsc.onControlPointerStart(e, target, target._jscControlName, 'touch');
557 | } else {
558 | if (jsc.picker && jsc.picker.owner) {
559 | jsc.picker.owner.hide();
560 | }
561 | }
562 | },
563 |
564 |
565 | onWindowResize : function (e) {
566 | jsc.redrawPosition();
567 | },
568 |
569 |
570 | onParentScroll : function (e) {
571 | // hide the picker when one of the parent elements is scrolled
572 | if (jsc.picker && jsc.picker.owner) {
573 | jsc.picker.owner.hide();
574 | }
575 | },
576 |
577 |
578 | _pointerMoveEvent : {
579 | mouse: 'mousemove',
580 | touch: 'touchmove'
581 | },
582 | _pointerEndEvent : {
583 | mouse: 'mouseup',
584 | touch: 'touchend'
585 | },
586 |
587 |
588 | _pointerOrigin : null,
589 | _capturedTarget : null,
590 |
591 |
592 | onControlPointerStart : function (e, target, controlName, pointerType) {
593 | var thisObj = target._jscInstance;
594 |
595 | jsc.preventDefault(e);
596 | jsc.captureTarget(target);
597 |
598 | var registerDragEvents = function (doc, offset) {
599 | jsc.attachGroupEvent('drag', doc, jsc._pointerMoveEvent[pointerType],
600 | jsc.onDocumentPointerMove(e, target, controlName, pointerType, offset));
601 | jsc.attachGroupEvent('drag', doc, jsc._pointerEndEvent[pointerType],
602 | jsc.onDocumentPointerEnd(e, target, controlName, pointerType));
603 | };
604 |
605 | registerDragEvents(document, [0, 0]);
606 |
607 | if (window.parent && window.frameElement) {
608 | var rect = window.frameElement.getBoundingClientRect();
609 | var ofs = [-rect.left, -rect.top];
610 | registerDragEvents(window.parent.window.document, ofs);
611 | }
612 |
613 | var abs = jsc.getAbsPointerPos(e);
614 | var rel = jsc.getRelPointerPos(e);
615 | jsc._pointerOrigin = {
616 | x: abs.x - rel.x,
617 | y: abs.y - rel.y
618 | };
619 |
620 | switch (controlName) {
621 | case 'pad':
622 | // if the slider is at the bottom, move it up
623 | switch (jsc.getSliderComponent(thisObj)) {
624 | case 's': if (thisObj.hsv[1] === 0) { thisObj.fromHSV(null, 100, null); }; break;
625 | case 'v': if (thisObj.hsv[2] === 0) { thisObj.fromHSV(null, null, 100); }; break;
626 | }
627 | jsc.setPad(thisObj, e, 0, 0);
628 | break;
629 |
630 | case 'sld':
631 | jsc.setSld(thisObj, e, 0);
632 | break;
633 | }
634 |
635 | jsc.dispatchFineChange(thisObj);
636 | },
637 |
638 |
639 | onDocumentPointerMove : function (e, target, controlName, pointerType, offset) {
640 | return function (e) {
641 | var thisObj = target._jscInstance;
642 | switch (controlName) {
643 | case 'pad':
644 | if (!e) { e = window.event; }
645 | jsc.setPad(thisObj, e, offset[0], offset[1]);
646 | jsc.dispatchFineChange(thisObj);
647 | break;
648 |
649 | case 'sld':
650 | if (!e) { e = window.event; }
651 | jsc.setSld(thisObj, e, offset[1]);
652 | jsc.dispatchFineChange(thisObj);
653 | break;
654 | }
655 | }
656 | },
657 |
658 |
659 | onDocumentPointerEnd : function (e, target, controlName, pointerType) {
660 | return function (e) {
661 | var thisObj = target._jscInstance;
662 | jsc.detachGroupEvents('drag');
663 | jsc.releaseTarget();
664 | // Always dispatch changes after detaching outstanding mouse handlers,
665 | // in case some user interaction will occur in user's onchange callback
666 | // that would intrude with current mouse events
667 | jsc.dispatchChange(thisObj);
668 | };
669 | },
670 |
671 |
672 | dispatchChange : function (thisObj) {
673 | if (thisObj.valueElement) {
674 | if (jsc.isElementType(thisObj.valueElement, 'input')) {
675 | jsc.fireEvent(thisObj.valueElement, 'change');
676 | }
677 | }
678 | },
679 |
680 |
681 | dispatchFineChange : function (thisObj) {
682 | if (thisObj.onFineChange) {
683 | var callback;
684 | if (typeof thisObj.onFineChange === 'string') {
685 | callback = new Function (thisObj.onFineChange);
686 | } else {
687 | callback = thisObj.onFineChange;
688 | }
689 | callback.call(thisObj);
690 | }
691 | },
692 |
693 |
694 | setPad : function (thisObj, e, ofsX, ofsY) {
695 | var pointerAbs = jsc.getAbsPointerPos(e);
696 | var x = ofsX + pointerAbs.x - jsc._pointerOrigin.x - thisObj.padding - thisObj.insetWidth;
697 | var y = ofsY + pointerAbs.y - jsc._pointerOrigin.y - thisObj.padding - thisObj.insetWidth;
698 |
699 | var xVal = x * (360 / (thisObj.width - 1));
700 | var yVal = 100 - (y * (100 / (thisObj.height - 1)));
701 |
702 | switch (jsc.getPadYComponent(thisObj)) {
703 | case 's': thisObj.fromHSV(xVal, yVal, null, jsc.leaveSld); break;
704 | case 'v': thisObj.fromHSV(xVal, null, yVal, jsc.leaveSld); break;
705 | }
706 | },
707 |
708 |
709 | setSld : function (thisObj, e, ofsY) {
710 | var pointerAbs = jsc.getAbsPointerPos(e);
711 | var y = ofsY + pointerAbs.y - jsc._pointerOrigin.y - thisObj.padding - thisObj.insetWidth;
712 |
713 | var yVal = 100 - (y * (100 / (thisObj.height - 1)));
714 |
715 | switch (jsc.getSliderComponent(thisObj)) {
716 | case 's': thisObj.fromHSV(null, yVal, null, jsc.leavePad); break;
717 | case 'v': thisObj.fromHSV(null, null, yVal, jsc.leavePad); break;
718 | }
719 | },
720 |
721 |
722 | _vmlNS : 'jsc_vml_',
723 | _vmlCSS : 'jsc_vml_css_',
724 | _vmlReady : false,
725 |
726 |
727 | initVML : function () {
728 | if (!jsc._vmlReady) {
729 | // init VML namespace
730 | var doc = document;
731 | if (!doc.namespaces[jsc._vmlNS]) {
732 | doc.namespaces.add(jsc._vmlNS, 'urn:schemas-microsoft-com:vml');
733 | }
734 | if (!doc.styleSheets[jsc._vmlCSS]) {
735 | var tags = ['shape', 'shapetype', 'group', 'background', 'path', 'formulas', 'handles', 'fill', 'stroke', 'shadow', 'textbox', 'textpath', 'imagedata', 'line', 'polyline', 'curve', 'rect', 'roundrect', 'oval', 'arc', 'image'];
736 | var ss = doc.createStyleSheet();
737 | ss.owningElement.id = jsc._vmlCSS;
738 | for (var i = 0; i < tags.length; i += 1) {
739 | ss.addRule(jsc._vmlNS + '\\:' + tags[i], 'behavior:url(#default#VML);');
740 | }
741 | }
742 | jsc._vmlReady = true;
743 | }
744 | },
745 |
746 |
747 | createPalette : function () {
748 |
749 | var paletteObj = {
750 | elm: null,
751 | draw: null
752 | };
753 |
754 | if (jsc.isCanvasSupported) {
755 | // Canvas implementation for modern browsers
756 |
757 | var canvas = document.createElement('canvas');
758 | var ctx = canvas.getContext('2d');
759 |
760 | var drawFunc = function (width, height, type) {
761 | canvas.width = width;
762 | canvas.height = height;
763 |
764 | ctx.clearRect(0, 0, canvas.width, canvas.height);
765 |
766 | var hGrad = ctx.createLinearGradient(0, 0, canvas.width, 0);
767 | hGrad.addColorStop(0 / 6, '#F00');
768 | hGrad.addColorStop(1 / 6, '#FF0');
769 | hGrad.addColorStop(2 / 6, '#0F0');
770 | hGrad.addColorStop(3 / 6, '#0FF');
771 | hGrad.addColorStop(4 / 6, '#00F');
772 | hGrad.addColorStop(5 / 6, '#F0F');
773 | hGrad.addColorStop(6 / 6, '#F00');
774 |
775 | ctx.fillStyle = hGrad;
776 | ctx.fillRect(0, 0, canvas.width, canvas.height);
777 |
778 | var vGrad = ctx.createLinearGradient(0, 0, 0, canvas.height);
779 | switch (type.toLowerCase()) {
780 | case 's':
781 | vGrad.addColorStop(0, 'rgba(255,255,255,0)');
782 | vGrad.addColorStop(1, 'rgba(255,255,255,1)');
783 | break;
784 | case 'v':
785 | vGrad.addColorStop(0, 'rgba(0,0,0,0)');
786 | vGrad.addColorStop(1, 'rgba(0,0,0,1)');
787 | break;
788 | }
789 | ctx.fillStyle = vGrad;
790 | ctx.fillRect(0, 0, canvas.width, canvas.height);
791 | };
792 |
793 | paletteObj.elm = canvas;
794 | paletteObj.draw = drawFunc;
795 |
796 | } else {
797 | // VML fallback for IE 7 and 8
798 |
799 | jsc.initVML();
800 |
801 | var vmlContainer = document.createElement('div');
802 | vmlContainer.style.position = 'relative';
803 | vmlContainer.style.overflow = 'hidden';
804 |
805 | var hGrad = document.createElement(jsc._vmlNS + ':fill');
806 | hGrad.type = 'gradient';
807 | hGrad.method = 'linear';
808 | hGrad.angle = '90';
809 | hGrad.colors = '16.67% #F0F, 33.33% #00F, 50% #0FF, 66.67% #0F0, 83.33% #FF0'
810 |
811 | var hRect = document.createElement(jsc._vmlNS + ':rect');
812 | hRect.style.position = 'absolute';
813 | hRect.style.left = -1 + 'px';
814 | hRect.style.top = -1 + 'px';
815 | hRect.stroked = false;
816 | hRect.appendChild(hGrad);
817 | vmlContainer.appendChild(hRect);
818 |
819 | var vGrad = document.createElement(jsc._vmlNS + ':fill');
820 | vGrad.type = 'gradient';
821 | vGrad.method = 'linear';
822 | vGrad.angle = '180';
823 | vGrad.opacity = '0';
824 |
825 | var vRect = document.createElement(jsc._vmlNS + ':rect');
826 | vRect.style.position = 'absolute';
827 | vRect.style.left = -1 + 'px';
828 | vRect.style.top = -1 + 'px';
829 | vRect.stroked = false;
830 | vRect.appendChild(vGrad);
831 | vmlContainer.appendChild(vRect);
832 |
833 | var drawFunc = function (width, height, type) {
834 | vmlContainer.style.width = width + 'px';
835 | vmlContainer.style.height = height + 'px';
836 |
837 | hRect.style.width =
838 | vRect.style.width =
839 | (width + 1) + 'px';
840 | hRect.style.height =
841 | vRect.style.height =
842 | (height + 1) + 'px';
843 |
844 | // Colors must be specified during every redraw, otherwise IE won't display
845 | // a full gradient during a subsequential redraw
846 | hGrad.color = '#F00';
847 | hGrad.color2 = '#F00';
848 |
849 | switch (type.toLowerCase()) {
850 | case 's':
851 | vGrad.color = vGrad.color2 = '#FFF';
852 | break;
853 | case 'v':
854 | vGrad.color = vGrad.color2 = '#000';
855 | break;
856 | }
857 | };
858 |
859 | paletteObj.elm = vmlContainer;
860 | paletteObj.draw = drawFunc;
861 | }
862 |
863 | return paletteObj;
864 | },
865 |
866 |
867 | createSliderGradient : function () {
868 |
869 | var sliderObj = {
870 | elm: null,
871 | draw: null
872 | };
873 |
874 | if (jsc.isCanvasSupported) {
875 | // Canvas implementation for modern browsers
876 |
877 | var canvas = document.createElement('canvas');
878 | var ctx = canvas.getContext('2d');
879 |
880 | var drawFunc = function (width, height, color1, color2) {
881 | canvas.width = width;
882 | canvas.height = height;
883 |
884 | ctx.clearRect(0, 0, canvas.width, canvas.height);
885 |
886 | var grad = ctx.createLinearGradient(0, 0, 0, canvas.height);
887 | grad.addColorStop(0, color1);
888 | grad.addColorStop(1, color2);
889 |
890 | ctx.fillStyle = grad;
891 | ctx.fillRect(0, 0, canvas.width, canvas.height);
892 | };
893 |
894 | sliderObj.elm = canvas;
895 | sliderObj.draw = drawFunc;
896 |
897 | } else {
898 | // VML fallback for IE 7 and 8
899 |
900 | jsc.initVML();
901 |
902 | var vmlContainer = document.createElement('div');
903 | vmlContainer.style.position = 'relative';
904 | vmlContainer.style.overflow = 'hidden';
905 |
906 | var grad = document.createElement(jsc._vmlNS + ':fill');
907 | grad.type = 'gradient';
908 | grad.method = 'linear';
909 | grad.angle = '180';
910 |
911 | var rect = document.createElement(jsc._vmlNS + ':rect');
912 | rect.style.position = 'absolute';
913 | rect.style.left = -1 + 'px';
914 | rect.style.top = -1 + 'px';
915 | rect.stroked = false;
916 | rect.appendChild(grad);
917 | vmlContainer.appendChild(rect);
918 |
919 | var drawFunc = function (width, height, color1, color2) {
920 | vmlContainer.style.width = width + 'px';
921 | vmlContainer.style.height = height + 'px';
922 |
923 | rect.style.width = (width + 1) + 'px';
924 | rect.style.height = (height + 1) + 'px';
925 |
926 | grad.color = color1;
927 | grad.color2 = color2;
928 | };
929 |
930 | sliderObj.elm = vmlContainer;
931 | sliderObj.draw = drawFunc;
932 | }
933 |
934 | return sliderObj;
935 | },
936 |
937 |
938 | leaveValue : 1<<0,
939 | leaveStyle : 1<<1,
940 | leavePad : 1<<2,
941 | leaveSld : 1<<3,
942 |
943 |
944 | BoxShadow : (function () {
945 | var BoxShadow = function (hShadow, vShadow, blur, spread, color, inset) {
946 | this.hShadow = hShadow;
947 | this.vShadow = vShadow;
948 | this.blur = blur;
949 | this.spread = spread;
950 | this.color = color;
951 | this.inset = !!inset;
952 | };
953 |
954 | BoxShadow.prototype.toString = function () {
955 | var vals = [
956 | Math.round(this.hShadow) + 'px',
957 | Math.round(this.vShadow) + 'px',
958 | Math.round(this.blur) + 'px',
959 | Math.round(this.spread) + 'px',
960 | this.color
961 | ];
962 | if (this.inset) {
963 | vals.push('inset');
964 | }
965 | return vals.join(' ');
966 | };
967 |
968 | return BoxShadow;
969 | })(),
970 |
971 |
972 | //
973 | // Usage:
974 | // var myColor = new jscolor(