'use strict';
(function(jQuery, angular)
{
// h, s, v [0,1]
function hsvToRgb(h,s,v)
{
if( s === 0 ){
return {r: v, g: v, b: v};
}
var i = Math.floor(h*6);
var f = 6*h - i; // fractional part
var p = v * (1 - s);
var q = v * (1 - s*f);
var t = v * (1 - s*(1-f));
switch(i%6){
case 0:
return {r:v,g:t,b:p};
case 1:
return {r:q,g:v,b:p};
case 2:
return {r:p,g:v,b:t};
case 3:
return {r:p,g:q,b:v};
case 4:
return {r:t,g:p,b:v};
default:
return {r:v,g:p,b:q};
}
}
function rgbToHsv(r,g,b)
{
var max = Math.max(r, g, b), min = Math.min(r, g, b);
var h, s, v = max;
var d = max - min;
s = max === 0 ? 0 : d / max;
if(max == min) {
h = 0; // achromatic
}
else {
switch(max) {
case r: h = (g - b) / d + (g < b ? 6 : 0); break;
case g: h = (b - r) / d + 2; break;
case b: h = (r - g) / d + 4; break;
}
h /= 6;
}
return { h: h, s: s, v: v };
}
function expandColor(val, partial)
{
if(!val) val = {};
if(val.a === undefined) val.a = 1.0;
if(partial && partial.a !== undefined) val.a = partial.a;
if(partial && (partial.h!==undefined || partial.s!==undefined || partial.v!==undefined))
{
if(partial.h !== undefined) val.h = partial.h;
if(partial.s !== undefined) val.s = partial.s;
if(partial.v !== undefined) val.v = partial.v;
var rgb = hsvToRgb(val.h, val.s, val.v);
val.r = Math.max(rgb.r, 0);
val.g = Math.max(rgb.g, 0);
val.b = Math.max(rgb.b, 0);
val.hex =
('00'+Math.round(val.r*255).toString(16)).slice(-2)
+('00'+Math.round(val.g*255).toString(16)).slice(-2)
+('00'+Math.round(val.b*255).toString(16)).slice(-2)
}
else if(partial && (partial.r!==undefined || partial.g!==undefined || partial.b!==undefined))
{
if(partial.r !== undefined) val.r = partial.r;
if(partial.g !== undefined) val.g = partial.g;
if(partial.b !== undefined) val.b = partial.b;
var hsv = rgbToHsv(val.r, val.g, val.b);
val.h = hsv.h;
val.s = hsv.s;
val.v = hsv.v;
val.hex =
('00'+Math.round(val.r*255).toString(16)).slice(-2)
+('00'+Math.round(val.g*255).toString(16)).slice(-2)
+('00'+Math.round(val.b*255).toString(16)).slice(-2)
}
else if( /^[0-9A-Fa-f]{6}$/.test(partial) )
{
val.hex = partial;
partial = parseInt(partial, 16);
val.r = ((partial & 0xff0000) >> 16) / 255;
val.g = ((partial & 0x00ff00) >> 8) / 255;
val.b = (partial & 0x0000ff) / 255;
var hsv = rgbToHsv(val.r, val.g, val.b);
val.h = hsv.h;
val.s = hsv.s;
val.v = hsv.v;
}
var rgba = [Math.round(val.r*255), Math.round(val.g*255), Math.round(val.b*255), val.a];
val.css = 'rgba('+rgba.join(',')+')';
val.background = 'linear-gradient('+val.css+','+val.css+'), url('+transparencyBgUrl+')';
return val;
}
var template = '
\n\n
\n #\n \n \n \n \n
\n
\n
\n';
var vertShaderSrc = 'precision lowp float;\nattribute vec3 vertPosition;\nvarying vec2 windowPosition;\nuniform vec2 windowDimensions;\n\nvoid main(void)\n{\n mat3 xform = mat3(0.5*windowDimensions.x, 0.0, 0.0, 0.0, 0.5*windowDimensions.y, 0.0, windowDimensions.x/2.0, windowDimensions.y/2.0, 1.0);\n windowPosition = (xform * vec3(vertPosition.xy, 1.0)).xy;\n gl_Position = vec4(vertPosition,1);\n}\n\n';
var fragShaderSrc = 'precision lowp float;\n#define M_PI 3.141592653589\n\nvarying vec2 windowPosition;\nuniform vec4 selectedColor;\nuniform float swatchWidth;\nuniform float marginWidth;\nuniform vec2 windowDimensions;\nuniform bool radial;\nuniform bool useAlpha;\nuniform sampler2D tex;\n\nvec3 hsv2rgb(vec3 c){\n vec4 K = vec4(1.0, 2.0 / 3.0, 1.0 / 3.0, 3.0);\n vec3 p = abs(fract(c.xxx + K.xyz) * 6.0 - K.www);\n return c.z * mix(K.xxx, clamp(p - K.xxx, 0.0, 1.0), c.y);\n}\n\nvec4 getSelectionColor(vec4 baseColor){\n return vec4( vec3(1.0)-baseColor.rgb, baseColor.a );\n}\n\nvoid main(void){\n\n float taLeft = useAlpha ? swatchWidth + marginWidth : 0.0;\n float taWidth = windowDimensions.x - swatchWidth - marginWidth - taLeft;\n vec2 center = vec2(taWidth/2.0+taLeft, windowDimensions.y/2.0);\n float aspect = taWidth / windowDimensions.y;\n vec4 color;\n vec2 selectionPosition;\n\n if( windowPosition.x >= taLeft && windowPosition.x <= taLeft+taWidth )\n {\n if(radial)\n {\n vec2 radialVec = (windowPosition - center)*vec2(2.0/taWidth, 2.0/windowDimensions.y);\n radialVec = mat2(max(1.0,aspect), 0.0, 0.0, max(1.0,1.0/aspect)) * radialVec;\n if(length(radialVec) > 1.0) discard;\n float hue = atan(radialVec.y,radialVec.x)/(2.0*M_PI) + 0.5;\n color = vec4(hue, length(radialVec), selectedColor.z, 1.0);\n\n float angle = (selectedColor.x-0.5)*2.0*M_PI;\n selectionPosition = min(center.x-taLeft, center.y) * selectedColor.y * vec2(cos(angle), sin(angle)) + center;\n }\n else {\n color = vec4((windowPosition.x-taLeft)/taWidth, windowPosition.y/windowDimensions.y, selectedColor.z, 1.0);\n selectionPosition = vec2(selectedColor.x*taWidth+taLeft, selectedColor.y*windowDimensions.y);\n }\n\n vec2 difference = selectionPosition - windowPosition;\n float radius = length(difference);\n\n if( radius > 4.5 && radius < 6.0 )\n gl_FragColor = getSelectionColor(vec4(hsv2rgb(color.xyz), 1.0 ));\n else\n gl_FragColor = vec4( hsv2rgb(color.xyz), 1.0);\n }\n\n else if(useAlpha && windowPosition.x < swatchWidth)\n {\n vec4 color = vec4(mix( texture2D(tex, windowPosition/16.0).xyz, hsv2rgb(selectedColor.xyz), windowPosition.y/windowDimensions.y), 1.0);\n\n if( windowDimensions.y * abs(windowPosition.y/windowDimensions.y-selectedColor.w) < 1.0 )\n gl_FragColor = getSelectionColor(color);\n else\n gl_FragColor = color;\n }\n\n else if(windowPosition.x > windowDimensions.x-swatchWidth)\n {\n vec4 color = vec4( selectedColor.x, selectedColor.y, windowPosition.y/windowDimensions.y, 1.0);\n\n if( windowDimensions.y * abs(windowPosition.y/windowDimensions.y-selectedColor.z) < 1.0 )\n gl_FragColor = getSelectionColor(vec4(hsv2rgb(color.xyz), 1.0 ));\n else\n gl_FragColor = vec4( hsv2rgb(color.xyz), 1.0);\n }\n else\n discard;\n}\n\n';
var transparencyBgUrl = '';
var paletteInitialized = false;
var PalettePopup = {
"colorCallback": null,
"colorSelectCallback": null,
"elem": document.createElement('div'),
"initialize": function()
{
var self = this;
if(document.getElementById('htmlPalettePopup'))
return;
// insert template into dom
this.elem = document.createElement('div');
this.elem.id = 'htmlPalettePopup';
this.elem.innerHTML = template;
this.elem.onclick = function(e){ e.stopPropagation(); }
this.elem.style.display = 'none';
document.body.appendChild(this.elem);
document.addEventListener('click', function(evt){
self.hide();
});
this.canvas = this.elem.children[0].children[0];
// get gl context
this.gl = this.canvas.getContext('webgl');
if(!this.gl) this.gl = this.canvas.getContext('experimental-webgl');
var gl = this.gl;
if(!gl){
console.error('Your web browser does not support WebGL, so cannot make use of the HtmlPalette color picker. Sorry :(');
return;
}
/********************************
* Initialize the webgl canvas
*********************************/
// set up vert shader
var vert = gl.createShader(gl.VERTEX_SHADER);
gl.shaderSource(vert, vertShaderSrc);
gl.compileShader(vert);
if( !gl.getShaderParameter(vert, gl.COMPILE_STATUS) ){
console.error('Vert shader error:', gl.getShaderInfoLog(vert));
return;
}
// set up frag shader
var frag = gl.createShader(gl.FRAGMENT_SHADER);
gl.shaderSource(frag, fragShaderSrc);
gl.compileShader(frag);
if( !gl.getShaderParameter(frag, gl.COMPILE_STATUS) ){
console.error('Frag shader error:', gl.getShaderInfoLog(frag));
return;
}
// link shaders
var program = this._program = gl.createProgram();
gl.attachShader(program, vert);
gl.attachShader(program, frag);
gl.linkProgram(program);
if( !gl.getProgramParameter(program, gl.LINK_STATUS) ){
console.error('Unable to link program');
return;
}
else {
gl.useProgram(program);
this._program = program;
}
var vertPositionAttrib = gl.getAttribLocation(program, 'vertPosition');
gl.enableVertexAttribArray(vertPositionAttrib);
var buffer = gl.createBuffer();
gl.bindBuffer(gl.ARRAY_BUFFER, buffer);
gl.bufferData(gl.ARRAY_BUFFER, new Float32Array([
-1.0, -1.0, 0.0,
-1.0, 1.0, 0.0,
1.0, -1.0, 0.0,
1.0, 1.0, 0.0
]), gl.STATIC_DRAW);
gl.vertexAttribPointer(vertPositionAttrib, 3, gl.FLOAT, false, 0, 0);
// bind non-configurable uniforms
gl.uniform1f(gl.getUniformLocation(program, 'swatchWidth'), 20);
gl.uniform1f(gl.getUniformLocation(program, 'marginWidth'), 10);
// bind transparency texture
var gltex = gl.createTexture();
var img = new Image();
img.onload = function()
{
gl.bindTexture(gl.TEXTURE_2D, gltex);
gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, gl.RGBA, gl.UNSIGNED_BYTE, img);
gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.NEAREST);
gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.NEAREST);
};
img.src = transparencyBgUrl;
gl.clearColor(0.0, 0.0, 0.0, 0.0);
/******************************************
* Bind event listeners
******************************************/
var twoaxis = this.elem.querySelector('.twoaxis'),
oneaxis = this.elem.querySelector('.oneaxis'),
alpha = this.elem.querySelector('.alpha');
twoaxis.ondragstart = function(evt){
if(evt.dataTransfer.setDragImage)
evt.dataTransfer.setDragImage(document.createElement('div'),0,0);
if(evt.dataTransfer.dropEffect)
evt.dataTransfer.dropEffect = 'move';
}
twoaxis.ondrag = twoaxis.onmousedown = function(evt){
var paletteWidth = self.canvas.width - (self.lastOpts && self.lastOpts.useAlpha ? 60 : 30);
var h = self.canvas.height;
if( self.lastOpts.radial ){
var center = [paletteWidth/2, h/2];
var clickDiff = [evt.offsetX-1 - center[0], -(evt.offsetY-1) + center[1]];
var hue = Math.atan2(clickDiff[1], clickDiff[0])/(2*Math.PI) + 0.5;
var sat = Math.sqrt(clickDiff[0]*clickDiff[0] + clickDiff[1]*clickDiff[1]) / Math.min(center[0], center[1]);
if( evt.screenX !== 0 || evt.screenY !== 0 )
self.color({h: hue, s: Math.min(sat, 1.0)});
}
else if( evt.screenX !== 0 || evt.screenY !== 0 )
self.color({h: Math.min(1, Math.max(0, (evt.offsetX-1)/paletteWidth)), s: Math.min(1, Math.max(0, (h-evt.offsetY+1)/(h-1)))});
}
oneaxis.ondragstart = function(evt){
evt.dataTransfer.setDragImage(document.createElement('div'),0,0);
}
oneaxis.ondrag = oneaxis.onmousedown = function(evt){
var h = self.canvas.height;
if( evt.screenX !== 0 || evt.screenY !== 0 )
self.color({v: Math.max(0, Math.min(1, (h-evt.offsetY-1)/(h-1)))});
}
alpha.ondragstart = function(evt){
evt.dataTransfer.setDragImage(document.createElement('div'),0,0);
}
alpha.ondrag = alpha.onmousedown = function(evt){
var h = self.canvas.height;
if( evt.screenX !== 0 || evt.screenY !== 0 )
self.color({a: Math.max(0, Math.min(1, (h-evt.offsetY-1)/(h-1)))});
}
twoaxis.ondragend = twoaxis.onmouseup = oneaxis.ondragend = oneaxis.onmouseup = alpha.ondragend = alpha.onmouseup =
function(evt){
if(self.colorSelectCallback)
self.colorSelectCallback(self.selection);
};
var initialValue, initialMouse;
function bindRGBElement(e, channel)
{
e.ondragstart = function(evt){
initialValue = Math.round(self.selection[channel]*255);
initialMouse = evt.offsetY;
evt.dataTransfer.effectAllowed = 'none';
evt.dataTransfer.setDragImage(document.createElement('div'),0,0);
}
e.ondrag = function(evt){
//evt.preventDefault();
var newVal = initialValue + initialMouse-evt.offsetY;
if(evt.screenX !== 0 || evt.screenY !== 0){
var color = {};
color[channel] = Math.max(0, Math.min(1, newVal/255));
self.color(color);
}
}
e.ondragend = function(evt){
if(self.colorSelectCallback)
self.colorSelectCallback(self.selection);
};
}
bindRGBElement(this.elem.querySelector('.r'), 'r');
bindRGBElement(this.elem.querySelector('.g'), 'g');
bindRGBElement(this.elem.querySelector('.b'), 'b');
bindRGBElement(this.elem.querySelector('.a'), 'a');
},
// bind configurable uniforms
"rebind": function(useAlpha, radial)
{
var style = window.getComputedStyle(this.elem.querySelector('.palette'));
var w = parseInt(style.width), h = parseInt(style.height)-4;
this.gl.uniform2f(this.gl.getUniformLocation(this._program, 'windowDimensions'), w,h);
this.gl.uniform1i(this.gl.getUniformLocation(this._program, 'useAlpha'), !!useAlpha);
this.gl.uniform1i(this.gl.getUniformLocation(this._program, 'radial'), !!radial);
},
"placePopup": function(evt, popupEdge)
{
evt.stopPropagation();
var popupStyle = window.getComputedStyle(this.elem);
var popupWidth = parseInt(popupStyle.width) || 0;
var popupHeight = parseInt(popupStyle.height) || 0;
var box = evt.target.getBoundingClientRect();
var origin = {
x: (box.left+box.right)/2,
y: (box.top+box.bottom)/2
};
var offset = 20;
// fall back on sw on missing or invalid option
if( !/^[ns]?[we]?$/.test(popupEdge) || !popupEdge )
popupEdge = 'se';
// set position vertically
if(/^n/.test(popupEdge))
this.elem.style.top = origin.y - popupHeight - offset + 'px';
else if(/^s/.test(popupEdge))
this.elem.style.top = origin.y + offset + 'px';
else
this.elem.style.top = origin.y - popupHeight/2 + 'px';
// set position horizontally
if(/e$/.test(popupEdge))
this.elem.style.left = origin.x + offset + 'px';
else if(/w$/.test(popupEdge))
this.elem.style.left = origin.x - popupWidth - offset + 'px';
else
this.elem.style.left = origin.x - popupWidth/2 + 'px';
this.elem.style.display = '';
},
// manually position the various ui elements
"sizePopup": function(useAlpha, classes)
{
this.elem.setAttribute('class', classes || '');
var popupStyle = window.getComputedStyle(this.elem);
this._popupWidth = parseInt(popupStyle.width) || 0;
this._popupHeight = parseInt(popupStyle.height) || 0;
// size the canvas
var style = window.getComputedStyle(this.elem.querySelector('.palette'));
var w = parseInt(style.width), h = parseInt(style.height)-4;
this.canvas.width = w;
this.canvas.height = h;
this.gl.viewport(0, 0, w, h);
// size the overlay
var paletteWidth = useAlpha ? w-60 : w-30;
var twoaxis = this.elem.querySelector('.twoaxis');
var oneaxis = this.elem.querySelector('.oneaxis')
var alpha = this.elem.querySelector('.alpha')
twoaxis.style.width = paletteWidth + 'px';
twoaxis.style.height = h + 'px';
oneaxis.style.height = h + 'px';
alpha.style.height = h + 'px';
if(!useAlpha){
alpha.style.display = 'none';
this.elem.querySelector('.rgbInput .a').style.display = 'none';
twoaxis.style.left = '0px';
}
else {
alpha.style.display = '';
this.elem.querySelector('.rgbInput .a').style.display = '';
twoaxis.style.left = '';
}
},
"color": function(val)
{
this.selection = expandColor( this.selection, val );
if(val)
{
this.redraw();
if(this.colorCallback)
this.colorCallback(this.selection);
}
},
"redraw": function()
{
var selectionUniform = this.gl.getUniformLocation(this._program, 'selectedColor');
this.gl.uniform4f(selectionUniform, this.selection.h, this.selection.s, this.selection.v, this.selection.a);
this.gl.clear(this.gl.COLOR_BUFFER_BIT);
this.gl.drawArrays(this.gl.TRIANGLE_STRIP, 0, 4);
this.elem.querySelector('.rgbInput .r').innerHTML = this.selection.hex.slice(0,2);
this.elem.querySelector('.rgbInput .g').innerHTML = this.selection.hex.slice(2,4);
this.elem.querySelector('.rgbInput .b').innerHTML = this.selection.hex.slice(4,6);
this.elem.querySelector('.rgbInput .a').innerHTML = ('00'+Math.round(this.selection.a*255).toString(16)).slice(-2);
this.elem.querySelector('.colorswatch').style['background'] = this.selection.background;
},
"show": function(opts, evt)
{
if(this.gl){
this.lastOpts = opts;
opts = opts || {};
this.placePopup(evt, opts.popupEdge);
this.sizePopup(opts.useAlpha, opts.css);
this.rebind(opts.useAlpha, opts.radial);
this.selection = opts.color;
this.redraw();
}
},
"hide": function(){
this.elem.style.display = 'none';
this.colorCallback = null;
this.colorSelectCallback = null;
this.triggerElem = null;
}
};
function Trigger(triggerElem, opts)
{
this.triggerElem = triggerElem;
this.colorCallback = opts.colorCallback || null;
this.colorSelectCallback = opts.colorSelectCallback || null;
this.popupEdge = opts.popupEdge || 'se';
this.css = opts.css || '';
this.radial = opts.radial || false;
this.useAlpha = opts.useAlpha || false;
this.updateTriggerBg = opts.updateTriggerBg !== undefined ? opts.updateTriggerBg : true;
this.disabled = opts.disabled || false;
this.selection = {};
this.color(opts.initialColor || 'aaaaaa');
PalettePopup.initialize();
var self = this;
triggerElem.addEventListener('click', this._clickHandler = function(event)
{
if(PalettePopup.triggerElem !== self.triggerElem && !self.disabled)
{
PalettePopup.triggerElem = self.triggerElem;
PalettePopup.colorCallback = self.color.bind(self);
PalettePopup.colorSelectCallback = function(color){
if(self.colorSelectCallback)
self.colorSelectCallback(JSON.parse(JSON.stringify(color)));
};
PalettePopup.show({
radial: self.radial,
useAlpha: self.useAlpha,
popupEdge: self.popupEdge,
css: self.css,
color: self.selection
}, event);
}
});
}
Trigger.prototype.color = function(val, isAngular)
{
if(val === undefined)
return this.selection;
else
{
/*var flag = false;
for(var i in val){
if(this.selection[i] !== val[i])
flag = true;
}
if(!flag) return;*/
if( this.selection !== val )
this.selection = expandColor(this.selection, val);
if(this.updateTriggerBg){
this.triggerElem.style['background'] = this.selection.background;
}
if(this.colorCallback){
this.colorCallback(JSON.parse(JSON.stringify(this.selection)), isAngular);
}
}
}
Trigger.prototype.destroy = function()
{
this.triggerElem.removeEventListener('click', this._clickHandler);
if(PalettePopup.triggerElem === this.triggerElem){
PalettePopup.hide();
}
}
window.HtmlPalette = Trigger;
window.HtmlPalette.PalettePopup = PalettePopup;
if(jQuery)
{
jQuery.fn.extend({
'HtmlPalette': function(cmd)
{
var args = Array.prototype.slice.call(arguments, 1);
var palette = this.data('HtmlPalette');
// when in doubt, return the palette object
if(!cmd)
return palette;
// initialize
else if(cmd instanceof Object)
{
if(palette)
palette.destroy();
palette = this.data('HtmlPalette', new HtmlPalette(this[0], cmd));
if(palette.colorCallback)
palette.colorCallback = palette.colorCallback.bind(this);
}
// error on command without initialization
else if(!palette) {
throw new Error('HtmlPalette is uninitialized on this element');
}
// evaluate command
else switch(cmd)
{
case 'color':
if(args.length)
palette.color(args[0]);
else
return palette.color();
break;
case 'colorCallback':
if(args.length)
palette.colorCallback = args[0].bind(this);
else
return palette.colorCallback;
break;
case 'colorSelectCallback':
if(args.length)
palette.colorSelectCallback = args[0].bind(this);
else
return palette.colorSelectCallback;
break;
case 'popupEdge':
if(args.length)
palette.popupEdge = args[0];
else
return palette.popupEdge;
break;
case 'radial':
if(args.length){
palette.radial = args[0];
}
else
return palette.radial;
break;
case 'updateTriggerBg':
if(args.length)
palette.updateTriggerBg = args[0];
else
return palette.updateTriggerBg;
break;
case 'disabled':
if(args.length)
palette.disabled = args[0];
else
return palette.disabled;
break;
case 'destroy':
palette.destroy();
this.data('htmlPalette', null);
break;
}
}
});
}
if(angular)
{
var app = angular.module('html-palette', []);
app.directive('htmlPalette', ['$timeout', function($timeout)
{
return {
restrict: 'AE',
scope: {
color: '=',
radial: '=?',
alpha: '=?',
disabled: '=?',
onColorSelect: '&?'
},
link: function($scope, elem, attrs)
{
var throttleTimeout = null;
attrs.initialColor = $scope.color;
var palette = new Trigger(elem[0], attrs);
$scope.$watch('radial', function(newval){
palette.radial = !!newval;
});
$scope.$watch('alpha', function(newval){
palette.alpha = !!newval;
});
$scope.$watch('disabled', function(newval){
palette.disabled = !!newval;
});
elem.bind('$destroy', function(){
palette.destroy();
});
palette.colorSelectCallback = function(color){
$scope.onColorSelect && $scope.onColorSelect({color: color});
};
palette.colorCallback = function(color, isAngular)
{
if($scope.color)
{
if(attrs.colorProfile === 'hex')
$scope.color.hex = color.hex;
else if(/^[rgbhsva]+$/.test(attrs.colorProfile)){
for(var i=0; i