|
@@ -1,1855 +0,0 @@
|
|
-/**
|
|
|
|
- * jscolor - JavaScript Color Picker
|
|
|
|
- *
|
|
|
|
- * @link http://jscolor.com
|
|
|
|
- * @license For open source use: GPLv3
|
|
|
|
- * For commercial use: JSColor Commercial License
|
|
|
|
- * @author Jan Odvarko
|
|
|
|
- * @version 2.0.5
|
|
|
|
- *
|
|
|
|
- * See usage examples at http://jscolor.com/examples/
|
|
|
|
- */
|
|
|
|
-
|
|
|
|
-
|
|
|
|
-"use strict";
|
|
|
|
-
|
|
|
|
-
|
|
|
|
-if (!window.jscolor) { window.jscolor = (function () {
|
|
|
|
-
|
|
|
|
-
|
|
|
|
-var jsc = {
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- register : function () {
|
|
|
|
- jsc.attachDOMReadyEvent(jsc.init);
|
|
|
|
- jsc.attachEvent(document, 'mousedown', jsc.onDocumentMouseDown);
|
|
|
|
- jsc.attachEvent(document, 'touchstart', jsc.onDocumentTouchStart);
|
|
|
|
- jsc.attachEvent(window, 'resize', jsc.onWindowResize);
|
|
|
|
- },
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- init : function () {
|
|
|
|
- if (jsc.jscolor.lookupClass) {
|
|
|
|
- jsc.jscolor.installByClassName(jsc.jscolor.lookupClass);
|
|
|
|
- }
|
|
|
|
- },
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- tryInstallOnElements : function (elms, className) {
|
|
|
|
- var matchClass = new RegExp('(^|\\s)(' + className + ')(\\s*(\\{[^}]*\\})|\\s|$)', 'i');
|
|
|
|
-
|
|
|
|
- for (var i = 0; i < elms.length; i += 1) {
|
|
|
|
- if (elms[i].type !== undefined && elms[i].type.toLowerCase() == 'color') {
|
|
|
|
- if (jsc.isColorAttrSupported) {
|
|
|
|
- // skip inputs of type 'color' if supported by the browser
|
|
|
|
- continue;
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
- var m;
|
|
|
|
- if (!elms[i].jscolor && elms[i].className && (m = elms[i].className.match(matchClass))) {
|
|
|
|
- var targetElm = elms[i];
|
|
|
|
- var optsStr = null;
|
|
|
|
-
|
|
|
|
- var dataOptions = jsc.getDataAttr(targetElm, 'jscolor');
|
|
|
|
- if (dataOptions !== null) {
|
|
|
|
- optsStr = dataOptions;
|
|
|
|
- } else if (m[4]) {
|
|
|
|
- optsStr = m[4];
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- var opts = {};
|
|
|
|
- if (optsStr) {
|
|
|
|
- try {
|
|
|
|
- opts = (new Function ('return (' + optsStr + ')'))();
|
|
|
|
- } catch(eParseError) {
|
|
|
|
- jsc.warn('Error parsing jscolor options: ' + eParseError + ':\n' + optsStr);
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
- targetElm.jscolor = new jsc.jscolor(targetElm, opts);
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
- },
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- isColorAttrSupported : (function () {
|
|
|
|
- var elm = document.createElement('input');
|
|
|
|
- if (elm.setAttribute) {
|
|
|
|
- elm.setAttribute('type', 'color');
|
|
|
|
- if (elm.type.toLowerCase() == 'color') {
|
|
|
|
- return true;
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
- return false;
|
|
|
|
- })(),
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- isCanvasSupported : (function () {
|
|
|
|
- var elm = document.createElement('canvas');
|
|
|
|
- return !!(elm.getContext && elm.getContext('2d'));
|
|
|
|
- })(),
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- fetchElement : function (mixed) {
|
|
|
|
- return typeof mixed === 'string' ? document.getElementById(mixed) : mixed;
|
|
|
|
- },
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- isElementType : function (elm, type) {
|
|
|
|
- return elm.nodeName.toLowerCase() === type.toLowerCase();
|
|
|
|
- },
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- getDataAttr : function (el, name) {
|
|
|
|
- var attrName = 'data-' + name;
|
|
|
|
- var attrValue = el.getAttribute(attrName);
|
|
|
|
- if (attrValue !== null) {
|
|
|
|
- return attrValue;
|
|
|
|
- }
|
|
|
|
- return null;
|
|
|
|
- },
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- attachEvent : function (el, evnt, func) {
|
|
|
|
- if (el.addEventListener) {
|
|
|
|
- el.addEventListener(evnt, func, false);
|
|
|
|
- } else if (el.attachEvent) {
|
|
|
|
- el.attachEvent('on' + evnt, func);
|
|
|
|
- }
|
|
|
|
- },
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- detachEvent : function (el, evnt, func) {
|
|
|
|
- if (el.removeEventListener) {
|
|
|
|
- el.removeEventListener(evnt, func, false);
|
|
|
|
- } else if (el.detachEvent) {
|
|
|
|
- el.detachEvent('on' + evnt, func);
|
|
|
|
- }
|
|
|
|
- },
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- _attachedGroupEvents : {},
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- attachGroupEvent : function (groupName, el, evnt, func) {
|
|
|
|
- if (!jsc._attachedGroupEvents.hasOwnProperty(groupName)) {
|
|
|
|
- jsc._attachedGroupEvents[groupName] = [];
|
|
|
|
- }
|
|
|
|
- jsc._attachedGroupEvents[groupName].push([el, evnt, func]);
|
|
|
|
- jsc.attachEvent(el, evnt, func);
|
|
|
|
- },
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- detachGroupEvents : function (groupName) {
|
|
|
|
- if (jsc._attachedGroupEvents.hasOwnProperty(groupName)) {
|
|
|
|
- for (var i = 0; i < jsc._attachedGroupEvents[groupName].length; i += 1) {
|
|
|
|
- var evt = jsc._attachedGroupEvents[groupName][i];
|
|
|
|
- jsc.detachEvent(evt[0], evt[1], evt[2]);
|
|
|
|
- }
|
|
|
|
- delete jsc._attachedGroupEvents[groupName];
|
|
|
|
- }
|
|
|
|
- },
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- attachDOMReadyEvent : function (func) {
|
|
|
|
- var fired = false;
|
|
|
|
- var fireOnce = function () {
|
|
|
|
- if (!fired) {
|
|
|
|
- fired = true;
|
|
|
|
- func();
|
|
|
|
- }
|
|
|
|
- };
|
|
|
|
-
|
|
|
|
- if (document.readyState === 'complete') {
|
|
|
|
- setTimeout(fireOnce, 1); // async
|
|
|
|
- return;
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- if (document.addEventListener) {
|
|
|
|
- document.addEventListener('DOMContentLoaded', fireOnce, false);
|
|
|
|
-
|
|
|
|
- // Fallback
|
|
|
|
- window.addEventListener('load', fireOnce, false);
|
|
|
|
-
|
|
|
|
- } else if (document.attachEvent) {
|
|
|
|
- // IE
|
|
|
|
- document.attachEvent('onreadystatechange', function () {
|
|
|
|
- if (document.readyState === 'complete') {
|
|
|
|
- document.detachEvent('onreadystatechange', arguments.callee);
|
|
|
|
- fireOnce();
|
|
|
|
- }
|
|
|
|
- })
|
|
|
|
-
|
|
|
|
- // Fallback
|
|
|
|
- window.attachEvent('onload', fireOnce);
|
|
|
|
-
|
|
|
|
- // IE7/8
|
|
|
|
- if (document.documentElement.doScroll && window == window.top) {
|
|
|
|
- var tryScroll = function () {
|
|
|
|
- if (!document.body) { return; }
|
|
|
|
- try {
|
|
|
|
- document.documentElement.doScroll('left');
|
|
|
|
- fireOnce();
|
|
|
|
- } catch (e) {
|
|
|
|
- setTimeout(tryScroll, 1);
|
|
|
|
- }
|
|
|
|
- };
|
|
|
|
- tryScroll();
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
- },
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- warn : function (msg) {
|
|
|
|
- if (window.console && window.console.warn) {
|
|
|
|
- window.console.warn(msg);
|
|
|
|
- }
|
|
|
|
- },
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- preventDefault : function (e) {
|
|
|
|
- if (e.preventDefault) { e.preventDefault(); }
|
|
|
|
- e.returnValue = false;
|
|
|
|
- },
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- captureTarget : function (target) {
|
|
|
|
- // IE
|
|
|
|
- if (target.setCapture) {
|
|
|
|
- jsc._capturedTarget = target;
|
|
|
|
- jsc._capturedTarget.setCapture();
|
|
|
|
- }
|
|
|
|
- },
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- releaseTarget : function () {
|
|
|
|
- // IE
|
|
|
|
- if (jsc._capturedTarget) {
|
|
|
|
- jsc._capturedTarget.releaseCapture();
|
|
|
|
- jsc._capturedTarget = null;
|
|
|
|
- }
|
|
|
|
- },
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- fireEvent : function (el, evnt) {
|
|
|
|
- if (!el) {
|
|
|
|
- return;
|
|
|
|
- }
|
|
|
|
- if (document.createEvent) {
|
|
|
|
- var ev = document.createEvent('HTMLEvents');
|
|
|
|
- ev.initEvent(evnt, true, true);
|
|
|
|
- el.dispatchEvent(ev);
|
|
|
|
- } else if (document.createEventObject) {
|
|
|
|
- var ev = document.createEventObject();
|
|
|
|
- el.fireEvent('on' + evnt, ev);
|
|
|
|
- } else if (el['on' + evnt]) { // alternatively use the traditional event model
|
|
|
|
- el['on' + evnt]();
|
|
|
|
- }
|
|
|
|
- },
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- classNameToList : function (className) {
|
|
|
|
- return className.replace(/^\s+|\s+$/g, '').split(/\s+/);
|
|
|
|
- },
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- // The className parameter (str) can only contain a single class name
|
|
|
|
- hasClass : function (elm, className) {
|
|
|
|
- if (!className) {
|
|
|
|
- return false;
|
|
|
|
- }
|
|
|
|
- return -1 != (' ' + elm.className.replace(/\s+/g, ' ') + ' ').indexOf(' ' + className + ' ');
|
|
|
|
- },
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- // The className parameter (str) can contain multiple class names separated by whitespace
|
|
|
|
- setClass : function (elm, className) {
|
|
|
|
- var classList = jsc.classNameToList(className);
|
|
|
|
- for (var i = 0; i < classList.length; i += 1) {
|
|
|
|
- if (!jsc.hasClass(elm, classList[i])) {
|
|
|
|
- elm.className += (elm.className ? ' ' : '') + classList[i];
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
- },
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- // The className parameter (str) can contain multiple class names separated by whitespace
|
|
|
|
- unsetClass : function (elm, className) {
|
|
|
|
- var classList = jsc.classNameToList(className);
|
|
|
|
- for (var i = 0; i < classList.length; i += 1) {
|
|
|
|
- var repl = new RegExp(
|
|
|
|
- '^\\s*' + classList[i] + '\\s*|' +
|
|
|
|
- '\\s*' + classList[i] + '\\s*$|' +
|
|
|
|
- '\\s+' + classList[i] + '(\\s+)',
|
|
|
|
- 'g'
|
|
|
|
- );
|
|
|
|
- elm.className = elm.className.replace(repl, '$1');
|
|
|
|
- }
|
|
|
|
- },
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- getStyle : function (elm) {
|
|
|
|
- return window.getComputedStyle ? window.getComputedStyle(elm) : elm.currentStyle;
|
|
|
|
- },
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- setStyle : (function () {
|
|
|
|
- var helper = document.createElement('div');
|
|
|
|
- var getSupportedProp = function (names) {
|
|
|
|
- for (var i = 0; i < names.length; i += 1) {
|
|
|
|
- if (names[i] in helper.style) {
|
|
|
|
- return names[i];
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
- };
|
|
|
|
- var props = {
|
|
|
|
- borderRadius: getSupportedProp(['borderRadius', 'MozBorderRadius', 'webkitBorderRadius']),
|
|
|
|
- boxShadow: getSupportedProp(['boxShadow', 'MozBoxShadow', 'webkitBoxShadow'])
|
|
|
|
- };
|
|
|
|
- return function (elm, prop, value) {
|
|
|
|
- switch (prop.toLowerCase()) {
|
|
|
|
- case 'opacity':
|
|
|
|
- var alphaOpacity = Math.round(parseFloat(value) * 100);
|
|
|
|
- elm.style.opacity = value;
|
|
|
|
- elm.style.filter = 'alpha(opacity=' + alphaOpacity + ')';
|
|
|
|
- break;
|
|
|
|
- default:
|
|
|
|
- elm.style[props[prop]] = value;
|
|
|
|
- break;
|
|
|
|
- }
|
|
|
|
- };
|
|
|
|
- })(),
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- setBorderRadius : function (elm, value) {
|
|
|
|
- jsc.setStyle(elm, 'borderRadius', value || '0');
|
|
|
|
- },
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- setBoxShadow : function (elm, value) {
|
|
|
|
- jsc.setStyle(elm, 'boxShadow', value || 'none');
|
|
|
|
- },
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- getElementPos : function (e, relativeToViewport) {
|
|
|
|
- var x=0, y=0;
|
|
|
|
- var rect = e.getBoundingClientRect();
|
|
|
|
- x = rect.left;
|
|
|
|
- y = rect.top;
|
|
|
|
- if (!relativeToViewport) {
|
|
|
|
- var viewPos = jsc.getViewPos();
|
|
|
|
- x += viewPos[0];
|
|
|
|
- y += viewPos[1];
|
|
|
|
- }
|
|
|
|
- return [x, y];
|
|
|
|
- },
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- getElementSize : function (e) {
|
|
|
|
- return [e.offsetWidth, e.offsetHeight];
|
|
|
|
- },
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- // get pointer's X/Y coordinates relative to viewport
|
|
|
|
- getAbsPointerPos : function (e) {
|
|
|
|
- if (!e) { e = window.event; }
|
|
|
|
- var x = 0, y = 0;
|
|
|
|
- if (typeof e.changedTouches !== 'undefined' && e.changedTouches.length) {
|
|
|
|
- // touch devices
|
|
|
|
- x = e.changedTouches[0].clientX;
|
|
|
|
- y = e.changedTouches[0].clientY;
|
|
|
|
- } else if (typeof e.clientX === 'number') {
|
|
|
|
- x = e.clientX;
|
|
|
|
- y = e.clientY;
|
|
|
|
- }
|
|
|
|
- return { x: x, y: y };
|
|
|
|
- },
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- // get pointer's X/Y coordinates relative to target element
|
|
|
|
- getRelPointerPos : function (e) {
|
|
|
|
- if (!e) { e = window.event; }
|
|
|
|
- var target = e.target || e.srcElement;
|
|
|
|
- var targetRect = target.getBoundingClientRect();
|
|
|
|
-
|
|
|
|
- var x = 0, y = 0;
|
|
|
|
-
|
|
|
|
- var clientX = 0, clientY = 0;
|
|
|
|
- if (typeof e.changedTouches !== 'undefined' && e.changedTouches.length) {
|
|
|
|
- // touch devices
|
|
|
|
- clientX = e.changedTouches[0].clientX;
|
|
|
|
- clientY = e.changedTouches[0].clientY;
|
|
|
|
- } else if (typeof e.clientX === 'number') {
|
|
|
|
- clientX = e.clientX;
|
|
|
|
- clientY = e.clientY;
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- x = clientX - targetRect.left;
|
|
|
|
- y = clientY - targetRect.top;
|
|
|
|
- return { x: x, y: y };
|
|
|
|
- },
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- getViewPos : function () {
|
|
|
|
- var doc = document.documentElement;
|
|
|
|
- return [
|
|
|
|
- (window.pageXOffset || doc.scrollLeft) - (doc.clientLeft || 0),
|
|
|
|
- (window.pageYOffset || doc.scrollTop) - (doc.clientTop || 0)
|
|
|
|
- ];
|
|
|
|
- },
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- getViewSize : function () {
|
|
|
|
- var doc = document.documentElement;
|
|
|
|
- return [
|
|
|
|
- (window.innerWidth || doc.clientWidth),
|
|
|
|
- (window.innerHeight || doc.clientHeight),
|
|
|
|
- ];
|
|
|
|
- },
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- redrawPosition : function () {
|
|
|
|
-
|
|
|
|
- if (jsc.picker && jsc.picker.owner) {
|
|
|
|
- var thisObj = jsc.picker.owner;
|
|
|
|
-
|
|
|
|
- var tp, vp;
|
|
|
|
-
|
|
|
|
- if (thisObj.fixed) {
|
|
|
|
- // Fixed elements are positioned relative to viewport,
|
|
|
|
- // therefore we can ignore the scroll offset
|
|
|
|
- tp = jsc.getElementPos(thisObj.targetElement, true); // target pos
|
|
|
|
- vp = [0, 0]; // view pos
|
|
|
|
- } else {
|
|
|
|
- tp = jsc.getElementPos(thisObj.targetElement); // target pos
|
|
|
|
- vp = jsc.getViewPos(); // view pos
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- var ts = jsc.getElementSize(thisObj.targetElement); // target size
|
|
|
|
- var vs = jsc.getViewSize(); // view size
|
|
|
|
- var ps = jsc.getPickerOuterDims(thisObj); // picker size
|
|
|
|
- var a, b, c;
|
|
|
|
- switch (thisObj.position.toLowerCase()) {
|
|
|
|
- case 'left': a=1; b=0; c=-1; break;
|
|
|
|
- case 'right':a=1; b=0; c=1; break;
|
|
|
|
- case 'top': a=0; b=1; c=-1; break;
|
|
|
|
- default: a=0; b=1; c=1; break;
|
|
|
|
- }
|
|
|
|
- var l = (ts[b]+ps[b])/2;
|
|
|
|
-
|
|
|
|
- // compute picker position
|
|
|
|
- if (!thisObj.smartPosition) {
|
|
|
|
- var pp = [
|
|
|
|
- tp[a],
|
|
|
|
- tp[b]+ts[b]-l+l*c
|
|
|
|
- ];
|
|
|
|
- } else {
|
|
|
|
- var pp = [
|
|
|
|
- -vp[a]+tp[a]+ps[a] > vs[a] ?
|
|
|
|
- (-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]) :
|
|
|
|
- tp[a],
|
|
|
|
- -vp[b]+tp[b]+ts[b]+ps[b]-l+l*c > vs[b] ?
|
|
|
|
- (-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) :
|
|
|
|
- (tp[b]+ts[b]-l+l*c >= 0 ? tp[b]+ts[b]-l+l*c : tp[b]+ts[b]-l-l*c)
|
|
|
|
- ];
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- var x = pp[a];
|
|
|
|
- var y = pp[b];
|
|
|
|
- var positionValue = thisObj.fixed ? 'fixed' : 'absolute';
|
|
|
|
- var contractShadow =
|
|
|
|
- (pp[0] + ps[0] > tp[0] || pp[0] < tp[0] + ts[0]) &&
|
|
|
|
- (pp[1] + ps[1] < tp[1] + ts[1]);
|
|
|
|
-
|
|
|
|
- jsc._drawPosition(thisObj, x, y, positionValue, contractShadow);
|
|
|
|
- }
|
|
|
|
- },
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- _drawPosition : function (thisObj, x, y, positionValue, contractShadow) {
|
|
|
|
- var vShadow = contractShadow ? 0 : thisObj.shadowBlur; // px
|
|
|
|
-
|
|
|
|
- jsc.picker.wrap.style.position = positionValue;
|
|
|
|
- jsc.picker.wrap.style.left = x + 'px';
|
|
|
|
- jsc.picker.wrap.style.top = y + 'px';
|
|
|
|
-
|
|
|
|
- jsc.setBoxShadow(
|
|
|
|
- jsc.picker.boxS,
|
|
|
|
- thisObj.shadow ?
|
|
|
|
- new jsc.BoxShadow(0, vShadow, thisObj.shadowBlur, 0, thisObj.shadowColor) :
|
|
|
|
- null);
|
|
|
|
- },
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- getPickerDims : function (thisObj) {
|
|
|
|
- var displaySlider = !!jsc.getSliderComponent(thisObj);
|
|
|
|
- var dims = [
|
|
|
|
- 2 * thisObj.insetWidth + 2 * thisObj.padding + thisObj.width +
|
|
|
|
- (displaySlider ? 2 * thisObj.insetWidth + jsc.getPadToSliderPadding(thisObj) + thisObj.sliderSize : 0),
|
|
|
|
- 2 * thisObj.insetWidth + 2 * thisObj.padding + thisObj.height +
|
|
|
|
- (thisObj.closable ? 2 * thisObj.insetWidth + thisObj.padding + thisObj.buttonHeight : 0)
|
|
|
|
- ];
|
|
|
|
- return dims;
|
|
|
|
- },
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- getPickerOuterDims : function (thisObj) {
|
|
|
|
- var dims = jsc.getPickerDims(thisObj);
|
|
|
|
- return [
|
|
|
|
- dims[0] + 2 * thisObj.borderWidth,
|
|
|
|
- dims[1] + 2 * thisObj.borderWidth
|
|
|
|
- ];
|
|
|
|
- },
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- getPadToSliderPadding : function (thisObj) {
|
|
|
|
- return Math.max(thisObj.padding, 1.5 * (2 * thisObj.pointerBorderWidth + thisObj.pointerThickness));
|
|
|
|
- },
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- getPadYComponent : function (thisObj) {
|
|
|
|
- switch (thisObj.mode.charAt(1).toLowerCase()) {
|
|
|
|
- case 'v': return 'v'; break;
|
|
|
|
- }
|
|
|
|
- return 's';
|
|
|
|
- },
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- getSliderComponent : function (thisObj) {
|
|
|
|
- if (thisObj.mode.length > 2) {
|
|
|
|
- switch (thisObj.mode.charAt(2).toLowerCase()) {
|
|
|
|
- case 's': return 's'; break;
|
|
|
|
- case 'v': return 'v'; break;
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
- return null;
|
|
|
|
- },
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- onDocumentMouseDown : function (e) {
|
|
|
|
- if (!e) { e = window.event; }
|
|
|
|
- var target = e.target || e.srcElement;
|
|
|
|
-
|
|
|
|
- if (target._jscLinkedInstance) {
|
|
|
|
- if (target._jscLinkedInstance.showOnClick) {
|
|
|
|
- target._jscLinkedInstance.show();
|
|
|
|
- }
|
|
|
|
- } else if (target._jscControlName) {
|
|
|
|
- jsc.onControlPointerStart(e, target, target._jscControlName, 'mouse');
|
|
|
|
- } else {
|
|
|
|
- // Mouse is outside the picker controls -> hide the color picker!
|
|
|
|
- if (jsc.picker && jsc.picker.owner) {
|
|
|
|
- jsc.picker.owner.hide();
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
- },
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- onDocumentTouchStart : function (e) {
|
|
|
|
- if (!e) { e = window.event; }
|
|
|
|
- var target = e.target || e.srcElement;
|
|
|
|
-
|
|
|
|
- if (target._jscLinkedInstance) {
|
|
|
|
- if (target._jscLinkedInstance.showOnClick) {
|
|
|
|
- target._jscLinkedInstance.show();
|
|
|
|
- }
|
|
|
|
- } else if (target._jscControlName) {
|
|
|
|
- jsc.onControlPointerStart(e, target, target._jscControlName, 'touch');
|
|
|
|
- } else {
|
|
|
|
- if (jsc.picker && jsc.picker.owner) {
|
|
|
|
- jsc.picker.owner.hide();
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
- },
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- onWindowResize : function (e) {
|
|
|
|
- jsc.redrawPosition();
|
|
|
|
- },
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- onParentScroll : function (e) {
|
|
|
|
- // hide the picker when one of the parent elements is scrolled
|
|
|
|
- if (jsc.picker && jsc.picker.owner) {
|
|
|
|
- jsc.picker.owner.hide();
|
|
|
|
- }
|
|
|
|
- },
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- _pointerMoveEvent : {
|
|
|
|
- mouse: 'mousemove',
|
|
|
|
- touch: 'touchmove'
|
|
|
|
- },
|
|
|
|
- _pointerEndEvent : {
|
|
|
|
- mouse: 'mouseup',
|
|
|
|
- touch: 'touchend'
|
|
|
|
- },
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- _pointerOrigin : null,
|
|
|
|
- _capturedTarget : null,
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- onControlPointerStart : function (e, target, controlName, pointerType) {
|
|
|
|
- var thisObj = target._jscInstance;
|
|
|
|
-
|
|
|
|
- jsc.preventDefault(e);
|
|
|
|
- jsc.captureTarget(target);
|
|
|
|
-
|
|
|
|
- var registerDragEvents = function (doc, offset) {
|
|
|
|
- jsc.attachGroupEvent('drag', doc, jsc._pointerMoveEvent[pointerType],
|
|
|
|
- jsc.onDocumentPointerMove(e, target, controlName, pointerType, offset));
|
|
|
|
- jsc.attachGroupEvent('drag', doc, jsc._pointerEndEvent[pointerType],
|
|
|
|
- jsc.onDocumentPointerEnd(e, target, controlName, pointerType));
|
|
|
|
- };
|
|
|
|
-
|
|
|
|
- registerDragEvents(document, [0, 0]);
|
|
|
|
-
|
|
|
|
- if (window.parent && window.frameElement) {
|
|
|
|
- var rect = window.frameElement.getBoundingClientRect();
|
|
|
|
- var ofs = [-rect.left, -rect.top];
|
|
|
|
- registerDragEvents(window.parent.window.document, ofs);
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- var abs = jsc.getAbsPointerPos(e);
|
|
|
|
- var rel = jsc.getRelPointerPos(e);
|
|
|
|
- jsc._pointerOrigin = {
|
|
|
|
- x: abs.x - rel.x,
|
|
|
|
- y: abs.y - rel.y
|
|
|
|
- };
|
|
|
|
-
|
|
|
|
- switch (controlName) {
|
|
|
|
- case 'pad':
|
|
|
|
- // if the slider is at the bottom, move it up
|
|
|
|
- switch (jsc.getSliderComponent(thisObj)) {
|
|
|
|
- case 's': if (thisObj.hsv[1] === 0) { thisObj.fromHSV(null, 100, null); }; break;
|
|
|
|
- case 'v': if (thisObj.hsv[2] === 0) { thisObj.fromHSV(null, null, 100); }; break;
|
|
|
|
- }
|
|
|
|
- jsc.setPad(thisObj, e, 0, 0);
|
|
|
|
- break;
|
|
|
|
-
|
|
|
|
- case 'sld':
|
|
|
|
- jsc.setSld(thisObj, e, 0);
|
|
|
|
- break;
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- jsc.dispatchFineChange(thisObj);
|
|
|
|
- },
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- onDocumentPointerMove : function (e, target, controlName, pointerType, offset) {
|
|
|
|
- return function (e) {
|
|
|
|
- var thisObj = target._jscInstance;
|
|
|
|
- switch (controlName) {
|
|
|
|
- case 'pad':
|
|
|
|
- if (!e) { e = window.event; }
|
|
|
|
- jsc.setPad(thisObj, e, offset[0], offset[1]);
|
|
|
|
- jsc.dispatchFineChange(thisObj);
|
|
|
|
- break;
|
|
|
|
-
|
|
|
|
- case 'sld':
|
|
|
|
- if (!e) { e = window.event; }
|
|
|
|
- jsc.setSld(thisObj, e, offset[1]);
|
|
|
|
- jsc.dispatchFineChange(thisObj);
|
|
|
|
- break;
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
- },
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- onDocumentPointerEnd : function (e, target, controlName, pointerType) {
|
|
|
|
- return function (e) {
|
|
|
|
- var thisObj = target._jscInstance;
|
|
|
|
- jsc.detachGroupEvents('drag');
|
|
|
|
- jsc.releaseTarget();
|
|
|
|
- // Always dispatch changes after detaching outstanding mouse handlers,
|
|
|
|
- // in case some user interaction will occur in user's onchange callback
|
|
|
|
- // that would intrude with current mouse events
|
|
|
|
- jsc.dispatchChange(thisObj);
|
|
|
|
- };
|
|
|
|
- },
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- dispatchChange : function (thisObj) {
|
|
|
|
- if (thisObj.valueElement) {
|
|
|
|
- if (jsc.isElementType(thisObj.valueElement, 'input')) {
|
|
|
|
- jsc.fireEvent(thisObj.valueElement, 'change');
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
- },
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- dispatchFineChange : function (thisObj) {
|
|
|
|
- if (thisObj.onFineChange) {
|
|
|
|
- var callback;
|
|
|
|
- if (typeof thisObj.onFineChange === 'string') {
|
|
|
|
- callback = new Function (thisObj.onFineChange);
|
|
|
|
- } else {
|
|
|
|
- callback = thisObj.onFineChange;
|
|
|
|
- }
|
|
|
|
- callback.call(thisObj);
|
|
|
|
- }
|
|
|
|
- },
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- setPad : function (thisObj, e, ofsX, ofsY) {
|
|
|
|
- var pointerAbs = jsc.getAbsPointerPos(e);
|
|
|
|
- var x = ofsX + pointerAbs.x - jsc._pointerOrigin.x - thisObj.padding - thisObj.insetWidth;
|
|
|
|
- var y = ofsY + pointerAbs.y - jsc._pointerOrigin.y - thisObj.padding - thisObj.insetWidth;
|
|
|
|
-
|
|
|
|
- var xVal = x * (360 / (thisObj.width - 1));
|
|
|
|
- var yVal = 100 - (y * (100 / (thisObj.height - 1)));
|
|
|
|
-
|
|
|
|
- switch (jsc.getPadYComponent(thisObj)) {
|
|
|
|
- case 's': thisObj.fromHSV(xVal, yVal, null, jsc.leaveSld); break;
|
|
|
|
- case 'v': thisObj.fromHSV(xVal, null, yVal, jsc.leaveSld); break;
|
|
|
|
- }
|
|
|
|
- },
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- setSld : function (thisObj, e, ofsY) {
|
|
|
|
- var pointerAbs = jsc.getAbsPointerPos(e);
|
|
|
|
- var y = ofsY + pointerAbs.y - jsc._pointerOrigin.y - thisObj.padding - thisObj.insetWidth;
|
|
|
|
-
|
|
|
|
- var yVal = 100 - (y * (100 / (thisObj.height - 1)));
|
|
|
|
-
|
|
|
|
- switch (jsc.getSliderComponent(thisObj)) {
|
|
|
|
- case 's': thisObj.fromHSV(null, yVal, null, jsc.leavePad); break;
|
|
|
|
- case 'v': thisObj.fromHSV(null, null, yVal, jsc.leavePad); break;
|
|
|
|
- }
|
|
|
|
- },
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- _vmlNS : 'jsc_vml_',
|
|
|
|
- _vmlCSS : 'jsc_vml_css_',
|
|
|
|
- _vmlReady : false,
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- initVML : function () {
|
|
|
|
- if (!jsc._vmlReady) {
|
|
|
|
- // init VML namespace
|
|
|
|
- var doc = document;
|
|
|
|
- if (!doc.namespaces[jsc._vmlNS]) {
|
|
|
|
- doc.namespaces.add(jsc._vmlNS, 'urn:schemas-microsoft-com:vml');
|
|
|
|
- }
|
|
|
|
- if (!doc.styleSheets[jsc._vmlCSS]) {
|
|
|
|
- var tags = ['shape', 'shapetype', 'group', 'background', 'path', 'formulas', 'handles', 'fill', 'stroke', 'shadow', 'textbox', 'textpath', 'imagedata', 'line', 'polyline', 'curve', 'rect', 'roundrect', 'oval', 'arc', 'image'];
|
|
|
|
- var ss = doc.createStyleSheet();
|
|
|
|
- ss.owningElement.id = jsc._vmlCSS;
|
|
|
|
- for (var i = 0; i < tags.length; i += 1) {
|
|
|
|
- ss.addRule(jsc._vmlNS + '\\:' + tags[i], 'behavior:url(#default#VML);');
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
- jsc._vmlReady = true;
|
|
|
|
- }
|
|
|
|
- },
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- createPalette : function () {
|
|
|
|
-
|
|
|
|
- var paletteObj = {
|
|
|
|
- elm: null,
|
|
|
|
- draw: null
|
|
|
|
- };
|
|
|
|
-
|
|
|
|
- if (jsc.isCanvasSupported) {
|
|
|
|
- // Canvas implementation for modern browsers
|
|
|
|
-
|
|
|
|
- var canvas = document.createElement('canvas');
|
|
|
|
- var ctx = canvas.getContext('2d');
|
|
|
|
-
|
|
|
|
- var drawFunc = function (width, height, type) {
|
|
|
|
- canvas.width = width;
|
|
|
|
- canvas.height = height;
|
|
|
|
-
|
|
|
|
- ctx.clearRect(0, 0, canvas.width, canvas.height);
|
|
|
|
-
|
|
|
|
- var hGrad = ctx.createLinearGradient(0, 0, canvas.width, 0);
|
|
|
|
- hGrad.addColorStop(0 / 6, '#F00');
|
|
|
|
- hGrad.addColorStop(1 / 6, '#FF0');
|
|
|
|
- hGrad.addColorStop(2 / 6, '#0F0');
|
|
|
|
- hGrad.addColorStop(3 / 6, '#0FF');
|
|
|
|
- hGrad.addColorStop(4 / 6, '#00F');
|
|
|
|
- hGrad.addColorStop(5 / 6, '#F0F');
|
|
|
|
- hGrad.addColorStop(6 / 6, '#F00');
|
|
|
|
-
|
|
|
|
- ctx.fillStyle = hGrad;
|
|
|
|
- ctx.fillRect(0, 0, canvas.width, canvas.height);
|
|
|
|
-
|
|
|
|
- var vGrad = ctx.createLinearGradient(0, 0, 0, canvas.height);
|
|
|
|
- switch (type.toLowerCase()) {
|
|
|
|
- case 's':
|
|
|
|
- vGrad.addColorStop(0, 'rgba(255,255,255,0)');
|
|
|
|
- vGrad.addColorStop(1, 'rgba(255,255,255,1)');
|
|
|
|
- break;
|
|
|
|
- case 'v':
|
|
|
|
- vGrad.addColorStop(0, 'rgba(0,0,0,0)');
|
|
|
|
- vGrad.addColorStop(1, 'rgba(0,0,0,1)');
|
|
|
|
- break;
|
|
|
|
- }
|
|
|
|
- ctx.fillStyle = vGrad;
|
|
|
|
- ctx.fillRect(0, 0, canvas.width, canvas.height);
|
|
|
|
- };
|
|
|
|
-
|
|
|
|
- paletteObj.elm = canvas;
|
|
|
|
- paletteObj.draw = drawFunc;
|
|
|
|
-
|
|
|
|
- } else {
|
|
|
|
- // VML fallback for IE 7 and 8
|
|
|
|
-
|
|
|
|
- jsc.initVML();
|
|
|
|
-
|
|
|
|
- var vmlContainer = document.createElement('div');
|
|
|
|
- vmlContainer.style.position = 'relative';
|
|
|
|
- vmlContainer.style.overflow = 'hidden';
|
|
|
|
-
|
|
|
|
- var hGrad = document.createElement(jsc._vmlNS + ':fill');
|
|
|
|
- hGrad.type = 'gradient';
|
|
|
|
- hGrad.method = 'linear';
|
|
|
|
- hGrad.angle = '90';
|
|
|
|
- hGrad.colors = '16.67% #F0F, 33.33% #00F, 50% #0FF, 66.67% #0F0, 83.33% #FF0'
|
|
|
|
-
|
|
|
|
- var hRect = document.createElement(jsc._vmlNS + ':rect');
|
|
|
|
- hRect.style.position = 'absolute';
|
|
|
|
- hRect.style.left = -1 + 'px';
|
|
|
|
- hRect.style.top = -1 + 'px';
|
|
|
|
- hRect.stroked = false;
|
|
|
|
- hRect.appendChild(hGrad);
|
|
|
|
- vmlContainer.appendChild(hRect);
|
|
|
|
-
|
|
|
|
- var vGrad = document.createElement(jsc._vmlNS + ':fill');
|
|
|
|
- vGrad.type = 'gradient';
|
|
|
|
- vGrad.method = 'linear';
|
|
|
|
- vGrad.angle = '180';
|
|
|
|
- vGrad.opacity = '0';
|
|
|
|
-
|
|
|
|
- var vRect = document.createElement(jsc._vmlNS + ':rect');
|
|
|
|
- vRect.style.position = 'absolute';
|
|
|
|
- vRect.style.left = -1 + 'px';
|
|
|
|
- vRect.style.top = -1 + 'px';
|
|
|
|
- vRect.stroked = false;
|
|
|
|
- vRect.appendChild(vGrad);
|
|
|
|
- vmlContainer.appendChild(vRect);
|
|
|
|
-
|
|
|
|
- var drawFunc = function (width, height, type) {
|
|
|
|
- vmlContainer.style.width = width + 'px';
|
|
|
|
- vmlContainer.style.height = height + 'px';
|
|
|
|
-
|
|
|
|
- hRect.style.width =
|
|
|
|
- vRect.style.width =
|
|
|
|
- (width + 1) + 'px';
|
|
|
|
- hRect.style.height =
|
|
|
|
- vRect.style.height =
|
|
|
|
- (height + 1) + 'px';
|
|
|
|
-
|
|
|
|
- // Colors must be specified during every redraw, otherwise IE won't display
|
|
|
|
- // a full gradient during a subsequential redraw
|
|
|
|
- hGrad.color = '#F00';
|
|
|
|
- hGrad.color2 = '#F00';
|
|
|
|
-
|
|
|
|
- switch (type.toLowerCase()) {
|
|
|
|
- case 's':
|
|
|
|
- vGrad.color = vGrad.color2 = '#FFF';
|
|
|
|
- break;
|
|
|
|
- case 'v':
|
|
|
|
- vGrad.color = vGrad.color2 = '#000';
|
|
|
|
- break;
|
|
|
|
- }
|
|
|
|
- };
|
|
|
|
-
|
|
|
|
- paletteObj.elm = vmlContainer;
|
|
|
|
- paletteObj.draw = drawFunc;
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- return paletteObj;
|
|
|
|
- },
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- createSliderGradient : function () {
|
|
|
|
-
|
|
|
|
- var sliderObj = {
|
|
|
|
- elm: null,
|
|
|
|
- draw: null
|
|
|
|
- };
|
|
|
|
-
|
|
|
|
- if (jsc.isCanvasSupported) {
|
|
|
|
- // Canvas implementation for modern browsers
|
|
|
|
-
|
|
|
|
- var canvas = document.createElement('canvas');
|
|
|
|
- var ctx = canvas.getContext('2d');
|
|
|
|
-
|
|
|
|
- var drawFunc = function (width, height, color1, color2) {
|
|
|
|
- canvas.width = width;
|
|
|
|
- canvas.height = height;
|
|
|
|
-
|
|
|
|
- ctx.clearRect(0, 0, canvas.width, canvas.height);
|
|
|
|
-
|
|
|
|
- var grad = ctx.createLinearGradient(0, 0, 0, canvas.height);
|
|
|
|
- grad.addColorStop(0, color1);
|
|
|
|
- grad.addColorStop(1, color2);
|
|
|
|
-
|
|
|
|
- ctx.fillStyle = grad;
|
|
|
|
- ctx.fillRect(0, 0, canvas.width, canvas.height);
|
|
|
|
- };
|
|
|
|
-
|
|
|
|
- sliderObj.elm = canvas;
|
|
|
|
- sliderObj.draw = drawFunc;
|
|
|
|
-
|
|
|
|
- } else {
|
|
|
|
- // VML fallback for IE 7 and 8
|
|
|
|
-
|
|
|
|
- jsc.initVML();
|
|
|
|
-
|
|
|
|
- var vmlContainer = document.createElement('div');
|
|
|
|
- vmlContainer.style.position = 'relative';
|
|
|
|
- vmlContainer.style.overflow = 'hidden';
|
|
|
|
-
|
|
|
|
- var grad = document.createElement(jsc._vmlNS + ':fill');
|
|
|
|
- grad.type = 'gradient';
|
|
|
|
- grad.method = 'linear';
|
|
|
|
- grad.angle = '180';
|
|
|
|
-
|
|
|
|
- var rect = document.createElement(jsc._vmlNS + ':rect');
|
|
|
|
- rect.style.position = 'absolute';
|
|
|
|
- rect.style.left = -1 + 'px';
|
|
|
|
- rect.style.top = -1 + 'px';
|
|
|
|
- rect.stroked = false;
|
|
|
|
- rect.appendChild(grad);
|
|
|
|
- vmlContainer.appendChild(rect);
|
|
|
|
-
|
|
|
|
- var drawFunc = function (width, height, color1, color2) {
|
|
|
|
- vmlContainer.style.width = width + 'px';
|
|
|
|
- vmlContainer.style.height = height + 'px';
|
|
|
|
-
|
|
|
|
- rect.style.width = (width + 1) + 'px';
|
|
|
|
- rect.style.height = (height + 1) + 'px';
|
|
|
|
-
|
|
|
|
- grad.color = color1;
|
|
|
|
- grad.color2 = color2;
|
|
|
|
- };
|
|
|
|
-
|
|
|
|
- sliderObj.elm = vmlContainer;
|
|
|
|
- sliderObj.draw = drawFunc;
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- return sliderObj;
|
|
|
|
- },
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- leaveValue : 1<<0,
|
|
|
|
- leaveStyle : 1<<1,
|
|
|
|
- leavePad : 1<<2,
|
|
|
|
- leaveSld : 1<<3,
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- BoxShadow : (function () {
|
|
|
|
- var BoxShadow = function (hShadow, vShadow, blur, spread, color, inset) {
|
|
|
|
- this.hShadow = hShadow;
|
|
|
|
- this.vShadow = vShadow;
|
|
|
|
- this.blur = blur;
|
|
|
|
- this.spread = spread;
|
|
|
|
- this.color = color;
|
|
|
|
- this.inset = !!inset;
|
|
|
|
- };
|
|
|
|
-
|
|
|
|
- BoxShadow.prototype.toString = function () {
|
|
|
|
- var vals = [
|
|
|
|
- Math.round(this.hShadow) + 'px',
|
|
|
|
- Math.round(this.vShadow) + 'px',
|
|
|
|
- Math.round(this.blur) + 'px',
|
|
|
|
- Math.round(this.spread) + 'px',
|
|
|
|
- this.color
|
|
|
|
- ];
|
|
|
|
- if (this.inset) {
|
|
|
|
- vals.push('inset');
|
|
|
|
- }
|
|
|
|
- return vals.join(' ');
|
|
|
|
- };
|
|
|
|
-
|
|
|
|
- return BoxShadow;
|
|
|
|
- })(),
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- //
|
|
|
|
- // Usage:
|
|
|
|
- // var myColor = new jscolor(<targetElement> [, <options>])
|
|
|
|
- //
|
|
|
|
-
|
|
|
|
- jscolor : function (targetElement, options) {
|
|
|
|
-
|
|
|
|
- // General options
|
|
|
|
- //
|
|
|
|
- this.value = null; // initial HEX color. To change it later, use methods fromString(), fromHSV() and fromRGB()
|
|
|
|
- this.valueElement = targetElement; // element that will be used to display and input the color code
|
|
|
|
- this.styleElement = targetElement; // element that will preview the picked color using CSS backgroundColor
|
|
|
|
- this.required = true; // whether the associated text <input> can be left empty
|
|
|
|
- this.refine = true; // whether to refine the entered color code (e.g. uppercase it and remove whitespace)
|
|
|
|
- this.hash = false; // whether to prefix the HEX color code with # symbol
|
|
|
|
- this.uppercase = true; // whether to show the color code in upper case
|
|
|
|
- this.onFineChange = null; // called instantly every time the color changes (value can be either a function or a string with javascript code)
|
|
|
|
- this.activeClass = 'jscolor-active'; // class to be set to the target element when a picker window is open on it
|
|
|
|
- this.overwriteImportant = false; // whether to overwrite colors of styleElement using !important
|
|
|
|
- this.minS = 0; // min allowed saturation (0 - 100)
|
|
|
|
- this.maxS = 100; // max allowed saturation (0 - 100)
|
|
|
|
- this.minV = 0; // min allowed value (brightness) (0 - 100)
|
|
|
|
- this.maxV = 100; // max allowed value (brightness) (0 - 100)
|
|
|
|
-
|
|
|
|
- // Accessing the picked color
|
|
|
|
- //
|
|
|
|
- this.hsv = [0, 0, 100]; // read-only [0-360, 0-100, 0-100]
|
|
|
|
- this.rgb = [255, 255, 255]; // read-only [0-255, 0-255, 0-255]
|
|
|
|
-
|
|
|
|
- // Color Picker options
|
|
|
|
- //
|
|
|
|
- this.width = 181; // width of color palette (in px)
|
|
|
|
- this.height = 101; // height of color palette (in px)
|
|
|
|
- this.showOnClick = true; // whether to display the color picker when user clicks on its target element
|
|
|
|
- this.mode = 'HSV'; // HSV | HVS | HS | HV - layout of the color picker controls
|
|
|
|
- this.position = 'bottom'; // left | right | top | bottom - position relative to the target element
|
|
|
|
- this.smartPosition = true; // automatically change picker position when there is not enough space for it
|
|
|
|
- this.sliderSize = 16; // px
|
|
|
|
- this.crossSize = 8; // px
|
|
|
|
- this.closable = false; // whether to display the Close button
|
|
|
|
- this.closeText = 'Close';
|
|
|
|
- this.buttonColor = '#000000'; // CSS color
|
|
|
|
- this.buttonHeight = 18; // px
|
|
|
|
- this.padding = 12; // px
|
|
|
|
- this.backgroundColor = '#FFFFFF'; // CSS color
|
|
|
|
- this.borderWidth = 1; // px
|
|
|
|
- this.borderColor = '#BBBBBB'; // CSS color
|
|
|
|
- this.borderRadius = 8; // px
|
|
|
|
- this.insetWidth = 1; // px
|
|
|
|
- this.insetColor = '#BBBBBB'; // CSS color
|
|
|
|
- this.shadow = true; // whether to display shadow
|
|
|
|
- this.shadowBlur = 15; // px
|
|
|
|
- this.shadowColor = 'rgba(0,0,0,0.2)'; // CSS color
|
|
|
|
- this.pointerColor = '#4C4C4C'; // px
|
|
|
|
- this.pointerBorderColor = '#FFFFFF'; // px
|
|
|
|
- this.pointerBorderWidth = 1; // px
|
|
|
|
- this.pointerThickness = 2; // px
|
|
|
|
- this.zIndex = 1000;
|
|
|
|
- this.container = null; // where to append the color picker (BODY element by default)
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- for (var opt in options) {
|
|
|
|
- if (options.hasOwnProperty(opt)) {
|
|
|
|
- this[opt] = options[opt];
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- this.hide = function () {
|
|
|
|
- if (isPickerOwner()) {
|
|
|
|
- detachPicker();
|
|
|
|
- }
|
|
|
|
- };
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- this.show = function () {
|
|
|
|
- drawPicker();
|
|
|
|
- };
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- this.redraw = function () {
|
|
|
|
- if (isPickerOwner()) {
|
|
|
|
- drawPicker();
|
|
|
|
- }
|
|
|
|
- };
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- this.importColor = function () {
|
|
|
|
- if (!this.valueElement) {
|
|
|
|
- this.exportColor();
|
|
|
|
- } else {
|
|
|
|
- if (jsc.isElementType(this.valueElement, 'input')) {
|
|
|
|
- if (!this.refine) {
|
|
|
|
- if (!this.fromString(this.valueElement.value, jsc.leaveValue)) {
|
|
|
|
- if (this.styleElement) {
|
|
|
|
- this.styleElement.style.backgroundImage = this.styleElement._jscOrigStyle.backgroundImage;
|
|
|
|
- this.styleElement.style.backgroundColor = this.styleElement._jscOrigStyle.backgroundColor;
|
|
|
|
- this.styleElement.style.color = this.styleElement._jscOrigStyle.color;
|
|
|
|
- }
|
|
|
|
- this.exportColor(jsc.leaveValue | jsc.leaveStyle);
|
|
|
|
- }
|
|
|
|
- } else if (!this.required && /^\s*$/.test(this.valueElement.value)) {
|
|
|
|
- this.valueElement.value = '';
|
|
|
|
- if (this.styleElement) {
|
|
|
|
- this.styleElement.style.backgroundImage = this.styleElement._jscOrigStyle.backgroundImage;
|
|
|
|
- this.styleElement.style.backgroundColor = this.styleElement._jscOrigStyle.backgroundColor;
|
|
|
|
- this.styleElement.style.color = this.styleElement._jscOrigStyle.color;
|
|
|
|
- }
|
|
|
|
- this.exportColor(jsc.leaveValue | jsc.leaveStyle);
|
|
|
|
-
|
|
|
|
- } else if (this.fromString(this.valueElement.value)) {
|
|
|
|
- // managed to import color successfully from the value -> OK, don't do anything
|
|
|
|
- } else {
|
|
|
|
- this.exportColor();
|
|
|
|
- }
|
|
|
|
- } else {
|
|
|
|
- // not an input element -> doesn't have any value
|
|
|
|
- this.exportColor();
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
- };
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- this.exportColor = function (flags) {
|
|
|
|
- if (!(flags & jsc.leaveValue) && this.valueElement) {
|
|
|
|
- var value = this.toString();
|
|
|
|
- if (this.uppercase) { value = value.toUpperCase(); }
|
|
|
|
- if (this.hash) { value = '#' + value; }
|
|
|
|
-
|
|
|
|
- if (jsc.isElementType(this.valueElement, 'input')) {
|
|
|
|
- this.valueElement.value = value;
|
|
|
|
- } else {
|
|
|
|
- this.valueElement.innerHTML = value;
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
- if (!(flags & jsc.leaveStyle)) {
|
|
|
|
- if (this.styleElement) {
|
|
|
|
- var bgColor = '#' + this.toString();
|
|
|
|
- var fgColor = this.isLight() ? '#000' : '#FFF';
|
|
|
|
-
|
|
|
|
- this.styleElement.style.backgroundImage = 'none';
|
|
|
|
- this.styleElement.style.backgroundColor = bgColor;
|
|
|
|
- this.styleElement.style.color = fgColor;
|
|
|
|
-
|
|
|
|
- if (this.overwriteImportant) {
|
|
|
|
- this.styleElement.setAttribute('style',
|
|
|
|
- 'background: ' + bgColor + ' !important; ' +
|
|
|
|
- 'color: ' + fgColor + ' !important;'
|
|
|
|
- );
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
- if (!(flags & jsc.leavePad) && isPickerOwner()) {
|
|
|
|
- redrawPad();
|
|
|
|
- }
|
|
|
|
- if (!(flags & jsc.leaveSld) && isPickerOwner()) {
|
|
|
|
- redrawSld();
|
|
|
|
- }
|
|
|
|
- };
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- // h: 0-360
|
|
|
|
- // s: 0-100
|
|
|
|
- // v: 0-100
|
|
|
|
- //
|
|
|
|
- this.fromHSV = function (h, s, v, flags) { // null = don't change
|
|
|
|
- if (h !== null) {
|
|
|
|
- if (isNaN(h)) { return false; }
|
|
|
|
- h = Math.max(0, Math.min(360, h));
|
|
|
|
- }
|
|
|
|
- if (s !== null) {
|
|
|
|
- if (isNaN(s)) { return false; }
|
|
|
|
- s = Math.max(0, Math.min(100, this.maxS, s), this.minS);
|
|
|
|
- }
|
|
|
|
- if (v !== null) {
|
|
|
|
- if (isNaN(v)) { return false; }
|
|
|
|
- v = Math.max(0, Math.min(100, this.maxV, v), this.minV);
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- this.rgb = HSV_RGB(
|
|
|
|
- h===null ? this.hsv[0] : (this.hsv[0]=h),
|
|
|
|
- s===null ? this.hsv[1] : (this.hsv[1]=s),
|
|
|
|
- v===null ? this.hsv[2] : (this.hsv[2]=v)
|
|
|
|
- );
|
|
|
|
-
|
|
|
|
- this.exportColor(flags);
|
|
|
|
- };
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- // r: 0-255
|
|
|
|
- // g: 0-255
|
|
|
|
- // b: 0-255
|
|
|
|
- //
|
|
|
|
- this.fromRGB = function (r, g, b, flags) { // null = don't change
|
|
|
|
- if (r !== null) {
|
|
|
|
- if (isNaN(r)) { return false; }
|
|
|
|
- r = Math.max(0, Math.min(255, r));
|
|
|
|
- }
|
|
|
|
- if (g !== null) {
|
|
|
|
- if (isNaN(g)) { return false; }
|
|
|
|
- g = Math.max(0, Math.min(255, g));
|
|
|
|
- }
|
|
|
|
- if (b !== null) {
|
|
|
|
- if (isNaN(b)) { return false; }
|
|
|
|
- b = Math.max(0, Math.min(255, b));
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- var hsv = RGB_HSV(
|
|
|
|
- r===null ? this.rgb[0] : r,
|
|
|
|
- g===null ? this.rgb[1] : g,
|
|
|
|
- b===null ? this.rgb[2] : b
|
|
|
|
- );
|
|
|
|
- if (hsv[0] !== null) {
|
|
|
|
- this.hsv[0] = Math.max(0, Math.min(360, hsv[0]));
|
|
|
|
- }
|
|
|
|
- if (hsv[2] !== 0) {
|
|
|
|
- this.hsv[1] = hsv[1]===null ? null : Math.max(0, this.minS, Math.min(100, this.maxS, hsv[1]));
|
|
|
|
- }
|
|
|
|
- this.hsv[2] = hsv[2]===null ? null : Math.max(0, this.minV, Math.min(100, this.maxV, hsv[2]));
|
|
|
|
-
|
|
|
|
- // update RGB according to final HSV, as some values might be trimmed
|
|
|
|
- var rgb = HSV_RGB(this.hsv[0], this.hsv[1], this.hsv[2]);
|
|
|
|
- this.rgb[0] = rgb[0];
|
|
|
|
- this.rgb[1] = rgb[1];
|
|
|
|
- this.rgb[2] = rgb[2];
|
|
|
|
-
|
|
|
|
- this.exportColor(flags);
|
|
|
|
- };
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- this.fromString = function (str, flags) {
|
|
|
|
- var m;
|
|
|
|
- if (m = str.match(/^\W*([0-9A-F]{3}([0-9A-F]{3})?)\W*$/i)) {
|
|
|
|
- // HEX notation
|
|
|
|
- //
|
|
|
|
-
|
|
|
|
- if (m[1].length === 6) {
|
|
|
|
- // 6-char notation
|
|
|
|
- this.fromRGB(
|
|
|
|
- parseInt(m[1].substr(0,2),16),
|
|
|
|
- parseInt(m[1].substr(2,2),16),
|
|
|
|
- parseInt(m[1].substr(4,2),16),
|
|
|
|
- flags
|
|
|
|
- );
|
|
|
|
- } else {
|
|
|
|
- // 3-char notation
|
|
|
|
- this.fromRGB(
|
|
|
|
- parseInt(m[1].charAt(0) + m[1].charAt(0),16),
|
|
|
|
- parseInt(m[1].charAt(1) + m[1].charAt(1),16),
|
|
|
|
- parseInt(m[1].charAt(2) + m[1].charAt(2),16),
|
|
|
|
- flags
|
|
|
|
- );
|
|
|
|
- }
|
|
|
|
- return true;
|
|
|
|
-
|
|
|
|
- } else if (m = str.match(/^\W*rgba?\(([^)]*)\)\W*$/i)) {
|
|
|
|
- var params = m[1].split(',');
|
|
|
|
- var re = /^\s*(\d*)(\.\d+)?\s*$/;
|
|
|
|
- var mR, mG, mB;
|
|
|
|
- if (
|
|
|
|
- params.length >= 3 &&
|
|
|
|
- (mR = params[0].match(re)) &&
|
|
|
|
- (mG = params[1].match(re)) &&
|
|
|
|
- (mB = params[2].match(re))
|
|
|
|
- ) {
|
|
|
|
- var r = parseFloat((mR[1] || '0') + (mR[2] || ''));
|
|
|
|
- var g = parseFloat((mG[1] || '0') + (mG[2] || ''));
|
|
|
|
- var b = parseFloat((mB[1] || '0') + (mB[2] || ''));
|
|
|
|
- this.fromRGB(r, g, b, flags);
|
|
|
|
- return true;
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
- return false;
|
|
|
|
- };
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- this.toString = function () {
|
|
|
|
- return (
|
|
|
|
- (0x100 | Math.round(this.rgb[0])).toString(16).substr(1) +
|
|
|
|
- (0x100 | Math.round(this.rgb[1])).toString(16).substr(1) +
|
|
|
|
- (0x100 | Math.round(this.rgb[2])).toString(16).substr(1)
|
|
|
|
- );
|
|
|
|
- };
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- this.toHEXString = function () {
|
|
|
|
- return '#' + this.toString().toUpperCase();
|
|
|
|
- };
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- this.toRGBString = function () {
|
|
|
|
- return ('rgb(' +
|
|
|
|
- Math.round(this.rgb[0]) + ',' +
|
|
|
|
- Math.round(this.rgb[1]) + ',' +
|
|
|
|
- Math.round(this.rgb[2]) + ')'
|
|
|
|
- );
|
|
|
|
- };
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- this.isLight = function () {
|
|
|
|
- return (
|
|
|
|
- 0.213 * this.rgb[0] +
|
|
|
|
- 0.715 * this.rgb[1] +
|
|
|
|
- 0.072 * this.rgb[2] >
|
|
|
|
- 255 / 2
|
|
|
|
- );
|
|
|
|
- };
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- this._processParentElementsInDOM = function () {
|
|
|
|
- if (this._linkedElementsProcessed) { return; }
|
|
|
|
- this._linkedElementsProcessed = true;
|
|
|
|
-
|
|
|
|
- var elm = this.targetElement;
|
|
|
|
- do {
|
|
|
|
- // If the target element or one of its parent nodes has fixed position,
|
|
|
|
- // then use fixed positioning instead
|
|
|
|
- //
|
|
|
|
- // Note: In Firefox, getComputedStyle returns null in a hidden iframe,
|
|
|
|
- // that's why we need to check if the returned style object is non-empty
|
|
|
|
- var currStyle = jsc.getStyle(elm);
|
|
|
|
- if (currStyle && currStyle.position.toLowerCase() === 'fixed') {
|
|
|
|
- this.fixed = true;
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- if (elm !== this.targetElement) {
|
|
|
|
- // Ensure to attach onParentScroll only once to each parent element
|
|
|
|
- // (multiple targetElements can share the same parent nodes)
|
|
|
|
- //
|
|
|
|
- // Note: It's not just offsetParents that can be scrollable,
|
|
|
|
- // that's why we loop through all parent nodes
|
|
|
|
- if (!elm._jscEventsAttached) {
|
|
|
|
- jsc.attachEvent(elm, 'scroll', jsc.onParentScroll);
|
|
|
|
- elm._jscEventsAttached = true;
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
- } while ((elm = elm.parentNode) && !jsc.isElementType(elm, 'body'));
|
|
|
|
- };
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- // r: 0-255
|
|
|
|
- // g: 0-255
|
|
|
|
- // b: 0-255
|
|
|
|
- //
|
|
|
|
- // returns: [ 0-360, 0-100, 0-100 ]
|
|
|
|
- //
|
|
|
|
- function RGB_HSV (r, g, b) {
|
|
|
|
- r /= 255;
|
|
|
|
- g /= 255;
|
|
|
|
- b /= 255;
|
|
|
|
- var n = Math.min(Math.min(r,g),b);
|
|
|
|
- var v = Math.max(Math.max(r,g),b);
|
|
|
|
- var m = v - n;
|
|
|
|
- if (m === 0) { return [ null, 0, 100 * v ]; }
|
|
|
|
- var h = r===n ? 3+(b-g)/m : (g===n ? 5+(r-b)/m : 1+(g-r)/m);
|
|
|
|
- return [
|
|
|
|
- 60 * (h===6?0:h),
|
|
|
|
- 100 * (m/v),
|
|
|
|
- 100 * v
|
|
|
|
- ];
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- // h: 0-360
|
|
|
|
- // s: 0-100
|
|
|
|
- // v: 0-100
|
|
|
|
- //
|
|
|
|
- // returns: [ 0-255, 0-255, 0-255 ]
|
|
|
|
- //
|
|
|
|
- function HSV_RGB (h, s, v) {
|
|
|
|
- var u = 255 * (v / 100);
|
|
|
|
-
|
|
|
|
- if (h === null) {
|
|
|
|
- return [ u, u, u ];
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- h /= 60;
|
|
|
|
- s /= 100;
|
|
|
|
-
|
|
|
|
- var i = Math.floor(h);
|
|
|
|
- var f = i%2 ? h-i : 1-(h-i);
|
|
|
|
- var m = u * (1 - s);
|
|
|
|
- var n = u * (1 - s * f);
|
|
|
|
- switch (i) {
|
|
|
|
- case 6:
|
|
|
|
- case 0: return [u,n,m];
|
|
|
|
- case 1: return [n,u,m];
|
|
|
|
- case 2: return [m,u,n];
|
|
|
|
- case 3: return [m,n,u];
|
|
|
|
- case 4: return [n,m,u];
|
|
|
|
- case 5: return [u,m,n];
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- function detachPicker () {
|
|
|
|
- jsc.unsetClass(THIS.targetElement, THIS.activeClass);
|
|
|
|
- jsc.picker.wrap.parentNode.removeChild(jsc.picker.wrap);
|
|
|
|
- delete jsc.picker.owner;
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- function drawPicker () {
|
|
|
|
-
|
|
|
|
- // At this point, when drawing the picker, we know what the parent elements are
|
|
|
|
- // and we can do all related DOM operations, such as registering events on them
|
|
|
|
- // or checking their positioning
|
|
|
|
- THIS._processParentElementsInDOM();
|
|
|
|
-
|
|
|
|
- if (!jsc.picker) {
|
|
|
|
- jsc.picker = {
|
|
|
|
- owner: null,
|
|
|
|
- wrap : document.createElement('div'),
|
|
|
|
- box : document.createElement('div'),
|
|
|
|
- boxS : document.createElement('div'), // shadow area
|
|
|
|
- boxB : document.createElement('div'), // border
|
|
|
|
- pad : document.createElement('div'),
|
|
|
|
- padB : document.createElement('div'), // border
|
|
|
|
- padM : document.createElement('div'), // mouse/touch area
|
|
|
|
- padPal : jsc.createPalette(),
|
|
|
|
- cross : document.createElement('div'),
|
|
|
|
- crossBY : document.createElement('div'), // border Y
|
|
|
|
- crossBX : document.createElement('div'), // border X
|
|
|
|
- crossLY : document.createElement('div'), // line Y
|
|
|
|
- crossLX : document.createElement('div'), // line X
|
|
|
|
- sld : document.createElement('div'),
|
|
|
|
- sldB : document.createElement('div'), // border
|
|
|
|
- sldM : document.createElement('div'), // mouse/touch area
|
|
|
|
- sldGrad : jsc.createSliderGradient(),
|
|
|
|
- sldPtrS : document.createElement('div'), // slider pointer spacer
|
|
|
|
- sldPtrIB : document.createElement('div'), // slider pointer inner border
|
|
|
|
- sldPtrMB : document.createElement('div'), // slider pointer middle border
|
|
|
|
- sldPtrOB : document.createElement('div'), // slider pointer outer border
|
|
|
|
- btn : document.createElement('div'),
|
|
|
|
- btnT : document.createElement('span') // text
|
|
|
|
- };
|
|
|
|
-
|
|
|
|
- jsc.picker.pad.appendChild(jsc.picker.padPal.elm);
|
|
|
|
- jsc.picker.padB.appendChild(jsc.picker.pad);
|
|
|
|
- jsc.picker.cross.appendChild(jsc.picker.crossBY);
|
|
|
|
- jsc.picker.cross.appendChild(jsc.picker.crossBX);
|
|
|
|
- jsc.picker.cross.appendChild(jsc.picker.crossLY);
|
|
|
|
- jsc.picker.cross.appendChild(jsc.picker.crossLX);
|
|
|
|
- jsc.picker.padB.appendChild(jsc.picker.cross);
|
|
|
|
- jsc.picker.box.appendChild(jsc.picker.padB);
|
|
|
|
- jsc.picker.box.appendChild(jsc.picker.padM);
|
|
|
|
-
|
|
|
|
- jsc.picker.sld.appendChild(jsc.picker.sldGrad.elm);
|
|
|
|
- jsc.picker.sldB.appendChild(jsc.picker.sld);
|
|
|
|
- jsc.picker.sldB.appendChild(jsc.picker.sldPtrOB);
|
|
|
|
- jsc.picker.sldPtrOB.appendChild(jsc.picker.sldPtrMB);
|
|
|
|
- jsc.picker.sldPtrMB.appendChild(jsc.picker.sldPtrIB);
|
|
|
|
- jsc.picker.sldPtrIB.appendChild(jsc.picker.sldPtrS);
|
|
|
|
- jsc.picker.box.appendChild(jsc.picker.sldB);
|
|
|
|
- jsc.picker.box.appendChild(jsc.picker.sldM);
|
|
|
|
-
|
|
|
|
- jsc.picker.btn.appendChild(jsc.picker.btnT);
|
|
|
|
- jsc.picker.box.appendChild(jsc.picker.btn);
|
|
|
|
-
|
|
|
|
- jsc.picker.boxB.appendChild(jsc.picker.box);
|
|
|
|
- jsc.picker.wrap.appendChild(jsc.picker.boxS);
|
|
|
|
- jsc.picker.wrap.appendChild(jsc.picker.boxB);
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- var p = jsc.picker;
|
|
|
|
-
|
|
|
|
- var displaySlider = !!jsc.getSliderComponent(THIS);
|
|
|
|
- var dims = jsc.getPickerDims(THIS);
|
|
|
|
- var crossOuterSize = (2 * THIS.pointerBorderWidth + THIS.pointerThickness + 2 * THIS.crossSize);
|
|
|
|
- var padToSliderPadding = jsc.getPadToSliderPadding(THIS);
|
|
|
|
- var borderRadius = Math.min(
|
|
|
|
- THIS.borderRadius,
|
|
|
|
- Math.round(THIS.padding * Math.PI)); // px
|
|
|
|
- var padCursor = 'crosshair';
|
|
|
|
-
|
|
|
|
- // wrap
|
|
|
|
- p.wrap.style.clear = 'both';
|
|
|
|
- p.wrap.style.width = (dims[0] + 2 * THIS.borderWidth) + 'px';
|
|
|
|
- p.wrap.style.height = (dims[1] + 2 * THIS.borderWidth) + 'px';
|
|
|
|
- p.wrap.style.zIndex = THIS.zIndex;
|
|
|
|
-
|
|
|
|
- // picker
|
|
|
|
- p.box.style.width = dims[0] + 'px';
|
|
|
|
- p.box.style.height = dims[1] + 'px';
|
|
|
|
-
|
|
|
|
- p.boxS.style.position = 'absolute';
|
|
|
|
- p.boxS.style.left = '0';
|
|
|
|
- p.boxS.style.top = '0';
|
|
|
|
- p.boxS.style.width = '100%';
|
|
|
|
- p.boxS.style.height = '100%';
|
|
|
|
- jsc.setBorderRadius(p.boxS, borderRadius + 'px');
|
|
|
|
-
|
|
|
|
- // picker border
|
|
|
|
- p.boxB.style.position = 'relative';
|
|
|
|
- p.boxB.style.border = THIS.borderWidth + 'px solid';
|
|
|
|
- p.boxB.style.borderColor = THIS.borderColor;
|
|
|
|
- p.boxB.style.background = THIS.backgroundColor;
|
|
|
|
- jsc.setBorderRadius(p.boxB, borderRadius + 'px');
|
|
|
|
-
|
|
|
|
- // IE hack:
|
|
|
|
- // If the element is transparent, IE will trigger the event on the elements under it,
|
|
|
|
- // e.g. on Canvas or on elements with border
|
|
|
|
- p.padM.style.background =
|
|
|
|
- p.sldM.style.background =
|
|
|
|
- '#FFF';
|
|
|
|
- jsc.setStyle(p.padM, 'opacity', '0');
|
|
|
|
- jsc.setStyle(p.sldM, 'opacity', '0');
|
|
|
|
-
|
|
|
|
- // pad
|
|
|
|
- p.pad.style.position = 'relative';
|
|
|
|
- p.pad.style.width = THIS.width + 'px';
|
|
|
|
- p.pad.style.height = THIS.height + 'px';
|
|
|
|
-
|
|
|
|
- // pad palettes (HSV and HVS)
|
|
|
|
- p.padPal.draw(THIS.width, THIS.height, jsc.getPadYComponent(THIS));
|
|
|
|
-
|
|
|
|
- // pad border
|
|
|
|
- p.padB.style.position = 'absolute';
|
|
|
|
- p.padB.style.left = THIS.padding + 'px';
|
|
|
|
- p.padB.style.top = THIS.padding + 'px';
|
|
|
|
- p.padB.style.border = THIS.insetWidth + 'px solid';
|
|
|
|
- p.padB.style.borderColor = THIS.insetColor;
|
|
|
|
-
|
|
|
|
- // pad mouse area
|
|
|
|
- p.padM._jscInstance = THIS;
|
|
|
|
- p.padM._jscControlName = 'pad';
|
|
|
|
- p.padM.style.position = 'absolute';
|
|
|
|
- p.padM.style.left = '0';
|
|
|
|
- p.padM.style.top = '0';
|
|
|
|
- p.padM.style.width = (THIS.padding + 2 * THIS.insetWidth + THIS.width + padToSliderPadding / 2) + 'px';
|
|
|
|
- p.padM.style.height = dims[1] + 'px';
|
|
|
|
- p.padM.style.cursor = padCursor;
|
|
|
|
-
|
|
|
|
- // pad cross
|
|
|
|
- p.cross.style.position = 'absolute';
|
|
|
|
- p.cross.style.left =
|
|
|
|
- p.cross.style.top =
|
|
|
|
- '0';
|
|
|
|
- p.cross.style.width =
|
|
|
|
- p.cross.style.height =
|
|
|
|
- crossOuterSize + 'px';
|
|
|
|
-
|
|
|
|
- // pad cross border Y and X
|
|
|
|
- p.crossBY.style.position =
|
|
|
|
- p.crossBX.style.position =
|
|
|
|
- 'absolute';
|
|
|
|
- p.crossBY.style.background =
|
|
|
|
- p.crossBX.style.background =
|
|
|
|
- THIS.pointerBorderColor;
|
|
|
|
- p.crossBY.style.width =
|
|
|
|
- p.crossBX.style.height =
|
|
|
|
- (2 * THIS.pointerBorderWidth + THIS.pointerThickness) + 'px';
|
|
|
|
- p.crossBY.style.height =
|
|
|
|
- p.crossBX.style.width =
|
|
|
|
- crossOuterSize + 'px';
|
|
|
|
- p.crossBY.style.left =
|
|
|
|
- p.crossBX.style.top =
|
|
|
|
- (Math.floor(crossOuterSize / 2) - Math.floor(THIS.pointerThickness / 2) - THIS.pointerBorderWidth) + 'px';
|
|
|
|
- p.crossBY.style.top =
|
|
|
|
- p.crossBX.style.left =
|
|
|
|
- '0';
|
|
|
|
-
|
|
|
|
- // pad cross line Y and X
|
|
|
|
- p.crossLY.style.position =
|
|
|
|
- p.crossLX.style.position =
|
|
|
|
- 'absolute';
|
|
|
|
- p.crossLY.style.background =
|
|
|
|
- p.crossLX.style.background =
|
|
|
|
- THIS.pointerColor;
|
|
|
|
- p.crossLY.style.height =
|
|
|
|
- p.crossLX.style.width =
|
|
|
|
- (crossOuterSize - 2 * THIS.pointerBorderWidth) + 'px';
|
|
|
|
- p.crossLY.style.width =
|
|
|
|
- p.crossLX.style.height =
|
|
|
|
- THIS.pointerThickness + 'px';
|
|
|
|
- p.crossLY.style.left =
|
|
|
|
- p.crossLX.style.top =
|
|
|
|
- (Math.floor(crossOuterSize / 2) - Math.floor(THIS.pointerThickness / 2)) + 'px';
|
|
|
|
- p.crossLY.style.top =
|
|
|
|
- p.crossLX.style.left =
|
|
|
|
- THIS.pointerBorderWidth + 'px';
|
|
|
|
-
|
|
|
|
- // slider
|
|
|
|
- p.sld.style.overflow = 'hidden';
|
|
|
|
- p.sld.style.width = THIS.sliderSize + 'px';
|
|
|
|
- p.sld.style.height = THIS.height + 'px';
|
|
|
|
-
|
|
|
|
- // slider gradient
|
|
|
|
- p.sldGrad.draw(THIS.sliderSize, THIS.height, '#000', '#000');
|
|
|
|
-
|
|
|
|
- // slider border
|
|
|
|
- p.sldB.style.display = displaySlider ? 'block' : 'none';
|
|
|
|
- p.sldB.style.position = 'absolute';
|
|
|
|
- p.sldB.style.right = THIS.padding + 'px';
|
|
|
|
- p.sldB.style.top = THIS.padding + 'px';
|
|
|
|
- p.sldB.style.border = THIS.insetWidth + 'px solid';
|
|
|
|
- p.sldB.style.borderColor = THIS.insetColor;
|
|
|
|
-
|
|
|
|
- // slider mouse area
|
|
|
|
- p.sldM._jscInstance = THIS;
|
|
|
|
- p.sldM._jscControlName = 'sld';
|
|
|
|
- p.sldM.style.display = displaySlider ? 'block' : 'none';
|
|
|
|
- p.sldM.style.position = 'absolute';
|
|
|
|
- p.sldM.style.right = '0';
|
|
|
|
- p.sldM.style.top = '0';
|
|
|
|
- p.sldM.style.width = (THIS.sliderSize + padToSliderPadding / 2 + THIS.padding + 2 * THIS.insetWidth) + 'px';
|
|
|
|
- p.sldM.style.height = dims[1] + 'px';
|
|
|
|
- p.sldM.style.cursor = 'default';
|
|
|
|
-
|
|
|
|
- // slider pointer inner and outer border
|
|
|
|
- p.sldPtrIB.style.border =
|
|
|
|
- p.sldPtrOB.style.border =
|
|
|
|
- THIS.pointerBorderWidth + 'px solid ' + THIS.pointerBorderColor;
|
|
|
|
-
|
|
|
|
- // slider pointer outer border
|
|
|
|
- p.sldPtrOB.style.position = 'absolute';
|
|
|
|
- p.sldPtrOB.style.left = -(2 * THIS.pointerBorderWidth + THIS.pointerThickness) + 'px';
|
|
|
|
- p.sldPtrOB.style.top = '0';
|
|
|
|
-
|
|
|
|
- // slider pointer middle border
|
|
|
|
- p.sldPtrMB.style.border = THIS.pointerThickness + 'px solid ' + THIS.pointerColor;
|
|
|
|
-
|
|
|
|
- // slider pointer spacer
|
|
|
|
- p.sldPtrS.style.width = THIS.sliderSize + 'px';
|
|
|
|
- p.sldPtrS.style.height = sliderPtrSpace + 'px';
|
|
|
|
-
|
|
|
|
- // the Close button
|
|
|
|
- function setBtnBorder () {
|
|
|
|
- var insetColors = THIS.insetColor.split(/\s+/);
|
|
|
|
- var outsetColor = insetColors.length < 2 ? insetColors[0] : insetColors[1] + ' ' + insetColors[0] + ' ' + insetColors[0] + ' ' + insetColors[1];
|
|
|
|
- p.btn.style.borderColor = outsetColor;
|
|
|
|
- }
|
|
|
|
- p.btn.style.display = THIS.closable ? 'block' : 'none';
|
|
|
|
- p.btn.style.position = 'absolute';
|
|
|
|
- p.btn.style.left = THIS.padding + 'px';
|
|
|
|
- p.btn.style.bottom = THIS.padding + 'px';
|
|
|
|
- p.btn.style.padding = '0 15px';
|
|
|
|
- p.btn.style.height = THIS.buttonHeight + 'px';
|
|
|
|
- p.btn.style.border = THIS.insetWidth + 'px solid';
|
|
|
|
- setBtnBorder();
|
|
|
|
- p.btn.style.color = THIS.buttonColor;
|
|
|
|
- p.btn.style.font = '12px sans-serif';
|
|
|
|
- p.btn.style.textAlign = 'center';
|
|
|
|
- try {
|
|
|
|
- p.btn.style.cursor = 'pointer';
|
|
|
|
- } catch(eOldIE) {
|
|
|
|
- p.btn.style.cursor = 'hand';
|
|
|
|
- }
|
|
|
|
- p.btn.onmousedown = function () {
|
|
|
|
- THIS.hide();
|
|
|
|
- };
|
|
|
|
- p.btnT.style.lineHeight = THIS.buttonHeight + 'px';
|
|
|
|
- p.btnT.innerHTML = '';
|
|
|
|
- p.btnT.appendChild(document.createTextNode(THIS.closeText));
|
|
|
|
-
|
|
|
|
- // place pointers
|
|
|
|
- redrawPad();
|
|
|
|
- redrawSld();
|
|
|
|
-
|
|
|
|
- // If we are changing the owner without first closing the picker,
|
|
|
|
- // make sure to first deal with the old owner
|
|
|
|
- if (jsc.picker.owner && jsc.picker.owner !== THIS) {
|
|
|
|
- jsc.unsetClass(jsc.picker.owner.targetElement, THIS.activeClass);
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- // Set the new picker owner
|
|
|
|
- jsc.picker.owner = THIS;
|
|
|
|
-
|
|
|
|
- // The redrawPosition() method needs picker.owner to be set, that's why we call it here,
|
|
|
|
- // after setting the owner
|
|
|
|
- if (jsc.isElementType(container, 'body')) {
|
|
|
|
- jsc.redrawPosition();
|
|
|
|
- } else {
|
|
|
|
- jsc._drawPosition(THIS, 0, 0, 'relative', false);
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- if (p.wrap.parentNode != container) {
|
|
|
|
- container.appendChild(p.wrap);
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- jsc.setClass(THIS.targetElement, THIS.activeClass);
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- function redrawPad () {
|
|
|
|
- // redraw the pad pointer
|
|
|
|
- switch (jsc.getPadYComponent(THIS)) {
|
|
|
|
- case 's': var yComponent = 1; break;
|
|
|
|
- case 'v': var yComponent = 2; break;
|
|
|
|
- }
|
|
|
|
- var x = Math.round((THIS.hsv[0] / 360) * (THIS.width - 1));
|
|
|
|
- var y = Math.round((1 - THIS.hsv[yComponent] / 100) * (THIS.height - 1));
|
|
|
|
- var crossOuterSize = (2 * THIS.pointerBorderWidth + THIS.pointerThickness + 2 * THIS.crossSize);
|
|
|
|
- var ofs = -Math.floor(crossOuterSize / 2);
|
|
|
|
- jsc.picker.cross.style.left = (x + ofs) + 'px';
|
|
|
|
- jsc.picker.cross.style.top = (y + ofs) + 'px';
|
|
|
|
-
|
|
|
|
- // redraw the slider
|
|
|
|
- switch (jsc.getSliderComponent(THIS)) {
|
|
|
|
- case 's':
|
|
|
|
- var rgb1 = HSV_RGB(THIS.hsv[0], 100, THIS.hsv[2]);
|
|
|
|
- var rgb2 = HSV_RGB(THIS.hsv[0], 0, THIS.hsv[2]);
|
|
|
|
- var color1 = 'rgb(' +
|
|
|
|
- Math.round(rgb1[0]) + ',' +
|
|
|
|
- Math.round(rgb1[1]) + ',' +
|
|
|
|
- Math.round(rgb1[2]) + ')';
|
|
|
|
- var color2 = 'rgb(' +
|
|
|
|
- Math.round(rgb2[0]) + ',' +
|
|
|
|
- Math.round(rgb2[1]) + ',' +
|
|
|
|
- Math.round(rgb2[2]) + ')';
|
|
|
|
- jsc.picker.sldGrad.draw(THIS.sliderSize, THIS.height, color1, color2);
|
|
|
|
- break;
|
|
|
|
- case 'v':
|
|
|
|
- var rgb = HSV_RGB(THIS.hsv[0], THIS.hsv[1], 100);
|
|
|
|
- var color1 = 'rgb(' +
|
|
|
|
- Math.round(rgb[0]) + ',' +
|
|
|
|
- Math.round(rgb[1]) + ',' +
|
|
|
|
- Math.round(rgb[2]) + ')';
|
|
|
|
- var color2 = '#000';
|
|
|
|
- jsc.picker.sldGrad.draw(THIS.sliderSize, THIS.height, color1, color2);
|
|
|
|
- break;
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- function redrawSld () {
|
|
|
|
- var sldComponent = jsc.getSliderComponent(THIS);
|
|
|
|
- if (sldComponent) {
|
|
|
|
- // redraw the slider pointer
|
|
|
|
- switch (sldComponent) {
|
|
|
|
- case 's': var yComponent = 1; break;
|
|
|
|
- case 'v': var yComponent = 2; break;
|
|
|
|
- }
|
|
|
|
- var y = Math.round((1 - THIS.hsv[yComponent] / 100) * (THIS.height - 1));
|
|
|
|
- jsc.picker.sldPtrOB.style.top = (y - (2 * THIS.pointerBorderWidth + THIS.pointerThickness) - Math.floor(sliderPtrSpace / 2)) + 'px';
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- function isPickerOwner () {
|
|
|
|
- return jsc.picker && jsc.picker.owner === THIS;
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- function blurValue () {
|
|
|
|
- THIS.importColor();
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- // Find the target element
|
|
|
|
- if (typeof targetElement === 'string') {
|
|
|
|
- var id = targetElement;
|
|
|
|
- var elm = document.getElementById(id);
|
|
|
|
- if (elm) {
|
|
|
|
- this.targetElement = elm;
|
|
|
|
- } else {
|
|
|
|
- jsc.warn('Could not find target element with ID \'' + id + '\'');
|
|
|
|
- }
|
|
|
|
- } else if (targetElement) {
|
|
|
|
- this.targetElement = targetElement;
|
|
|
|
- } else {
|
|
|
|
- jsc.warn('Invalid target element: \'' + targetElement + '\'');
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- if (this.targetElement._jscLinkedInstance) {
|
|
|
|
- jsc.warn('Cannot link jscolor twice to the same element. Skipping.');
|
|
|
|
- return;
|
|
|
|
- }
|
|
|
|
- this.targetElement._jscLinkedInstance = this;
|
|
|
|
-
|
|
|
|
- // Find the value element
|
|
|
|
- this.valueElement = jsc.fetchElement(this.valueElement);
|
|
|
|
- // Find the style element
|
|
|
|
- this.styleElement = jsc.fetchElement(this.styleElement);
|
|
|
|
-
|
|
|
|
- var THIS = this;
|
|
|
|
- var container =
|
|
|
|
- this.container ?
|
|
|
|
- jsc.fetchElement(this.container) :
|
|
|
|
- document.getElementsByTagName('body')[0];
|
|
|
|
- var sliderPtrSpace = 3; // px
|
|
|
|
-
|
|
|
|
- // For BUTTON elements it's important to stop them from sending the form when clicked
|
|
|
|
- // (e.g. in Safari)
|
|
|
|
- if (jsc.isElementType(this.targetElement, 'button')) {
|
|
|
|
- if (this.targetElement.onclick) {
|
|
|
|
- var origCallback = this.targetElement.onclick;
|
|
|
|
- this.targetElement.onclick = function (evt) {
|
|
|
|
- origCallback.call(this, evt);
|
|
|
|
- return false;
|
|
|
|
- };
|
|
|
|
- } else {
|
|
|
|
- this.targetElement.onclick = function () { return false; };
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- /*
|
|
|
|
- var elm = this.targetElement;
|
|
|
|
- do {
|
|
|
|
- // If the target element or one of its offsetParents has fixed position,
|
|
|
|
- // then use fixed positioning instead
|
|
|
|
- //
|
|
|
|
- // Note: In Firefox, getComputedStyle returns null in a hidden iframe,
|
|
|
|
- // that's why we need to check if the returned style object is non-empty
|
|
|
|
- var currStyle = jsc.getStyle(elm);
|
|
|
|
- if (currStyle && currStyle.position.toLowerCase() === 'fixed') {
|
|
|
|
- this.fixed = true;
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- if (elm !== this.targetElement) {
|
|
|
|
- // attach onParentScroll so that we can recompute the picker position
|
|
|
|
- // when one of the offsetParents is scrolled
|
|
|
|
- if (!elm._jscEventsAttached) {
|
|
|
|
- jsc.attachEvent(elm, 'scroll', jsc.onParentScroll);
|
|
|
|
- elm._jscEventsAttached = true;
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
- } while ((elm = elm.offsetParent) && !jsc.isElementType(elm, 'body'));
|
|
|
|
- */
|
|
|
|
-
|
|
|
|
- // valueElement
|
|
|
|
- if (this.valueElement) {
|
|
|
|
- if (jsc.isElementType(this.valueElement, 'input')) {
|
|
|
|
- var updateField = function () {
|
|
|
|
- THIS.fromString(THIS.valueElement.value, jsc.leaveValue);
|
|
|
|
- jsc.dispatchFineChange(THIS);
|
|
|
|
- };
|
|
|
|
- jsc.attachEvent(this.valueElement, 'keyup', updateField);
|
|
|
|
- jsc.attachEvent(this.valueElement, 'input', updateField);
|
|
|
|
- jsc.attachEvent(this.valueElement, 'blur', blurValue);
|
|
|
|
- this.valueElement.setAttribute('autocomplete', 'off');
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- // styleElement
|
|
|
|
- if (this.styleElement) {
|
|
|
|
- this.styleElement._jscOrigStyle = {
|
|
|
|
- backgroundImage : this.styleElement.style.backgroundImage,
|
|
|
|
- backgroundColor : this.styleElement.style.backgroundColor,
|
|
|
|
- color : this.styleElement.style.color
|
|
|
|
- };
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- if (this.value) {
|
|
|
|
- // Try to set the color from the .value option and if unsuccessful,
|
|
|
|
- // export the current color
|
|
|
|
- this.fromString(this.value) || this.exportColor();
|
|
|
|
- } else {
|
|
|
|
- this.importColor();
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
-};
|
|
|
|
-
|
|
|
|
-
|
|
|
|
-//================================
|
|
|
|
-// Public properties and methods
|
|
|
|
-//================================
|
|
|
|
-
|
|
|
|
-
|
|
|
|
-// By default, search for all elements with class="jscolor" and install a color picker on them.
|
|
|
|
-//
|
|
|
|
-// You can change what class name will be looked for by setting the property jscolor.lookupClass
|
|
|
|
-// anywhere in your HTML document. To completely disable the automatic lookup, set it to null.
|
|
|
|
-//
|
|
|
|
-jsc.jscolor.lookupClass = 'jscolor';
|
|
|
|
-
|
|
|
|
-
|
|
|
|
-jsc.jscolor.installByClassName = function (className) {
|
|
|
|
- var inputElms = document.getElementsByTagName('input');
|
|
|
|
- var buttonElms = document.getElementsByTagName('button');
|
|
|
|
-
|
|
|
|
- jsc.tryInstallOnElements(inputElms, className);
|
|
|
|
- jsc.tryInstallOnElements(buttonElms, className);
|
|
|
|
-};
|
|
|
|
-
|
|
|
|
-
|
|
|
|
-jsc.register();
|
|
|
|
-
|
|
|
|
-
|
|
|
|
-return jsc.jscolor;
|
|
|
|
-
|
|
|
|
-
|
|
|
|
-})(); }
|
|
|