<!DOCTYPE html>
<html>
<head>
<title>Angular 2 Material Plunker</title>
<script src="https://code.angularjs.org/tools/typescript.js"></script>
<script src="https://angular-code.firebaseapp.com/zone.js/dist/zone.js"></script>
<script src="https://angular-code.firebaseapp.com/reflect-metadata/Reflect.js"></script>
<script src="https://rawgithub.com/systemjs/systemjs/0.19.27/dist/system.js"></script>
<!-- Load Bundle from Asset Host -->
<script src="https://angular-code.firebaseapp.com/bundle.js"></script>
<!-- Configure SystemJS -->
<script src="systemjs.config.js"></script>
<style>
.lightText {
color: #fff;
}
.md-input-element {
color: inherit;
}
body {
font-family: Roboto, "Helvetica Neue", sans-serif;
}
#demoDiv {
height: 300px;
}
</style>
<script>
System
.import('app.component.ts')
.catch(console.error.bind(console));
</script>
</head>
<body>
<material-app>Loading the Angular 2 Material App...</material-app>
</body>
</html>
<!--
Copyright 2016 Google Inc. All Rights Reserved.
Use of this source code is governed by an MIT-style license that
can be found in the LICENSE file at http://angular.io/license
-->
import {Component} from '@angular/core';
import {bootstrap} from '@angular/platform-browser-dynamic';
import {MdToolbar} from '@angular2-material/toolbar';
import {MdButton} from '@angular2-material/button';
import {MdInput} from '@angular2-material/input';
import {ColorPickerDirective} from './color-picker/color-picker.directive';
@Component({
selector: 'material-app',
templateUrl: 'app.component.html',
directives: [MdToolbar, MdButton, MdInput, ColorPickerDirective]
})
export class AppComponent {
private color: string = "#127bdc";
private color2: string = "#1e8e44";
private color3: string = "#79b78e";
private color4: string = "#d200ff";
private color5: string = "rgba(78,53,47,0.86)";
private color6: string = "#cfd138";
private color7: string = "#38c4d1";
}
bootstrap(AppComponent);
/*
Copyright 2016 Google Inc. All Rights Reserved.
Use of this source code is governed by an MIT-style license that
can be found in the LICENSE file at http://angular.io/license
*/
<md-toolbar color="primary">
Angular 2/ Material Color Picker Demo
</md-toolbar>
<div style="padding: 7px">
<h3>Standard</h3>
<input [(colorPicker)]="color"
[style.background]="color"
[value]="color"
type="text"
/><br><br>
<h3>Simplified</h3>
<input [(colorPicker)]="color2"
[style.background]="color2"
[value]="color2"
[hasActions]="false"
[hasFormatters]="false"
type="text"
/><br><br>
<h3>Material Input</h3>
<md-input [(colorPicker)]="color3"
[style.color]="color3"
placeholder="Color 3:"
[value]="color3" [position]="'bottom'"
[positionOffset]="'100%'"
[positionRelativeToArrow]="false"
type="text"
></md-input><br><br>
<h3>Only Hex/RGB (Formatter Controls)</h3>
<md-input [(colorPicker)]="color4"
[style.color]="color4"
placeholder="Color 4:"
[value]="color4"
[formats]="'rgb,hex'"
[position]="'bottom'"
[positionOffset]="'15%'"
[positionRelativeToArrow]="true"
type="text"
></md-input><br><br>
<h4 [(colorPicker)]="color6"
[position]="'bottom'"
[positionOffset]="'15%'"
[positionRelativeToArrow]="true"
[style.color]="color6">Text Color can be controled?</h4><br><br>
<div id="demoDiv" [(colorPicker)]="color7"
[position]="'top'"
[positionOffset]="'15%'"
[style.background]="color7">
<p>Yeah Even a Div! Notice my text color changes relative to the tone</p>
</div>
<br><br>
</div>
<!--
Copyright 2016 Google Inc. All Rights Reserved.
Use of this source code is governed by an MIT-style license that
can be found in the LICENSE file at http://angular.io/license
-->
System.config({
transpiler: 'typescript',
typescriptOptions: {
emitDecoratorMetadata: true
},
packages: {
'.': {
defaultExtension: 'ts'
}
},
map: {
'@angular': 'node_modules/@angular',
'@angular2-material': 'node_modules/@angular2-material',
'rxjs': 'node_modules/rxjs'
}
});
var _packageConfig = {};
var _defaultPackages = [
'@angular/core', '@angular/common', '@angular/compiler', '@angular/http', '@angular/router',
'@angular/platform-browser', '@angular/platform-browser-dynamic', 'rxjs'
];
// Angular Material 2 Packages to load.
var _materialPackages = [
'core', 'toolbar', 'button', 'card', 'checkbox', 'icon', 'input', 'list', 'progress-bar',
'progress-circle', 'radio', 'sidenav'
];
_materialPackages.forEach(function(item) {
// All Material 2 components are prefixed with @angular2-material and use
// the components name as entry point.
_packageConfig['@angular2-material/' + item] = { main: item };
});
_defaultPackages.forEach(function (item) {
// Angular's Default Packages are always using `index` as an entry point.
_packageConfig[item] = { main: 'index' };
});
// Apply the new generated packages to the SystemJS configuration.
System.config({ packages: _packageConfig });
/*
Copyright 2016 Google Inc. All Rights Reserved.
Use of this source code is governed by an MIT-style license that
can be found in the LICENSE file at http://angular.io/license
*/
import {Component, DynamicComponentLoader, Directive, Input, Output, ViewContainerRef,
ElementRef, EventEmitter, OnInit, HostBinding, ViewEncapsulation} from '@angular/core';
import {NgClass} from '@angular/common';
import {MD_CARD_DIRECTIVES} from '@angular2-material/card/card';
import {MdButton} from '@angular2-material/button/button';
import {Rgba, Hsla, Hsva, SliderPosition, SliderDimension, ColorPickerUtils} from 'color-picker/color-picker.utils';
@Directive({
selector: '[colorPicker]',
host: {
'(input)': 'changeInput($event.target.value)',
'(click)': 'onClick()'
}
})
export class ColorPickerDirective implements OnInit {
//Two way bind based on the selector(colorPicker)
@Input('colorPicker') colorPicker: string;
@Output('colorPickerChange') colorPickerChange = new EventEmitter<string>();
//Add class to host if text related to this should be light(white)
@HostBinding('class.lightText') get lightText { return this.textTone == 'light'; };
//Legacy Positioning **TODO: Refactor with portal/overlay
@Input('position') position: string = 'right';
@Input('positionOffset') positionOffset: string = '0%';
@Input('positionRelativeToArrow') positionRelativeToArrow: boolean = false;
//What the output "value" string will be.
@Input('outputFormat') outputFormat: string = 'hex';
//What formats are listed in the tabs
@Input('formats') formats:string = 'rgb,hsla,hex';
private activeFormats = {
rgb:false,
hsla:false,
hex: false,
cmyk:false
}
@Input('hasFormatters') hasFormatters: boolean = true;
@Input('hasActions') hasActions: boolean = true;
//Holder for dialog component related to what the directive is bound to.
private dialog: any = null;
//Color processing functions
private colorUtils: any = new ColorPickerUtils();
//Whether or not the text should be black/white based on the colors balance
public textTone: string = 'dark';
constructor(private dcl: DynamicComponentLoader, private vcRef: ViewContainerRef, private el: ElementRef) { }
ngOnInit() {
//convert all input types to hsva to be easier to work with
let hsva = this.colorUtils.stringToHsva(this.colorPicker);
if (hsva !== null) {
this.colorPickerChange.emit(this.colorUtils.outputFormat(hsva, this.outputFormat));
this.textTone = this.colorUtils.getYiq(this.colorPicker);
}
//activate formats
if (this.formats != '') {
var formatsArray = this.formats.split(',');
formatsArray.forEach((format:string) => {
this.activeFormats[format] = true;
});
};
}
//Open function. If it doesn't exist it gets created. If it exists but is hiding it shows.
onClick() {
if (!this.dialog) {
this.dcl.loadNextToLocation(ColorPickerComponent, this.vcRef)
.then((res) => {
res.instance.setDialog(this);
this.dialog = res.instance;
});
} else {
this.dialog.setInitialColor(this.colorPicker);
this.dialog.openColorPicker();
}
}
//accessible emitters
colorChanged(value: string) {
this.colorPickerChange.emit(value)
}
changeInput(value: string) {
this.dialog.setColorFromString(value)
this.colorPickerChange.emit(value)
}
}
/*** Input Directive, not 100% what the rg thing is about yet..**/
@Directive({
selector: '[text]',
host: {
'(input)': 'changeInput($event.target.value)'
}
})
export class TextDirective {
@Output('newValue') newValue = new EventEmitter<any>();
@Input('text') text: any;
@Input('rg') rg: number;
changeInput(value: string) {
if (this.rg === undefined) {
this.newValue.emit(value);
} else {
let numeric = parseFloat(value)
if (!isNaN(numeric) && numeric >= 0 && numeric <= this.rg) {
this.newValue.emit({ v: numeric, rg: this.rg });
}
}
}
}
/*** Slider Directive **/
@Directive({
selector: '[slider]',
host: {
'(mousedown)': 'start($event)',
'(touchstart)': 'start($event)'
}
})
export class SliderDirective {
@Output('newValue') newValue = new EventEmitter<any>();
@Input('slider') slider: string;
@Input('rgX') rgX: number;
@Input('rgY') rgY: number;
private listenerMove: any;
private listenerStop: any;
//Adds class so children can activate when this is active based on css
@HostBinding('class.active') public active: boolean = false;
constructor(private el: ElementRef) {
this.listenerMove = (event: any) => { this.move(event) };
this.listenerStop = () => { this.stop() };
}
setCursor(event: any) {
let height = this.el.nativeElement.offsetHeight;
let width = this.el.nativeElement.offsetWidth;
let x = Math.max(0, Math.min(this.getX(event), width));
let y = Math.max(0, Math.min(this.getY(event), height));
if (this.rgX !== undefined && this.rgY !== undefined) {
this.newValue.emit({ s: x / width, v: (1 - y / height), rgX: this.rgX, rgY: this.rgY });
} else if (this.rgX === undefined && this.rgY !== undefined) {//ready to use vertical sliders
this.newValue.emit({ v: y / height, rg: this.rgY });
} else {
this.newValue.emit({ v: x / width, rg: this.rgX });
}
}
move(event: any) {
event.preventDefault();
this.setCursor(event);
}
start(event: any) {
this.setCursor(event);
this.active = true;
document.addEventListener('mousemove', this.listenerMove);
document.addEventListener('touchmove', this.listenerMove);
document.addEventListener('mouseup', this.listenerStop);
document.addEventListener('touchend', this.listenerStop);
}
stop() {
this.active = false;
document.removeEventListener('mousemove', this.listenerMove);
document.removeEventListener('touchmove', this.listenerMove);
document.removeEventListener('mouseup', this.listenerStop);
document.removeEventListener('touchend', this.listenerStop);
}
getX(event: any) {
return (event.pageX !== undefined ? event.pageX : event.touches[0].pageX) - this.el.nativeElement.getBoundingClientRect().left - window.pageXOffset;
}
getY(event: any) {
return (event.pageY !== undefined ? event.pageY : event.touches[0].pageY) - this.el.nativeElement.getBoundingClientRect().top - window.pageYOffset;
}
}
/*
*
*
*
*
* COMPONENT ******************/
@Component({
selector: 'color-picker',
templateUrl: 'color-picker/color-picker.html',
styleUrls: ['color-picker/color-picker.css'],
// moduleId: module.id,
directives: [NgClass, SliderDirective, TextDirective, MD_CARD_DIRECTIVES, MdButton],
encapsulation: ViewEncapsulation.None,
})
export class ColorPickerComponent implements OnInit {
private service: ColorPickerUtils;
private directiveInstance: any;
private directiveElementRef: ElementRef;
private hsva: Hsva;
private initialColor: string;
private outputColor: string;
private alphaSliderColor: string;
private hueSliderColor: string;
private textTone: string;
private rgbaText: Rgba;
private hslaText: Hsla;
private hexText: string;
private slider: SliderPosition;
private sliderDimMax: SliderDimension;
private format: number;
private show: boolean;
private activeFormats: any;
@HostBinding('class.has-formatters') hasFormatters: boolean = true;
@HostBinding('class.has-actions') hasActions: boolean = true;
private listenerMouseDown: any;
private listenerResize: any;
private top: number;
private left: number;
private position: string;
private positionOffset: number;
private outputFormat: string;
private cpHeight: number;
private dialogWidth: number = 232;
private dialogArrowSize: number = 10;
private dialogArrowOffset: number = 15;
private arrowTop: number;
constructor(private el: ElementRef) {
this.service = new ColorPickerUtils();
}
setDialog(instance: any) {
this.directiveInstance = instance;
this.initialColor = instance.colorPicker;
this.directiveElementRef = instance.el;
this.position = instance.position;
this.positionOffset = parseInt(instance.positionOffset);
this.outputFormat = instance.outputFormat;
this.activeFormats = instance.activeFormats;
this.hasActions = instance.hasActions;
this.hasFormatters = instance.hasFormatters;
if (!instance.positionRelativeToArrow) {
this.dialogArrowOffset = 0;
}
}
setInitialColor(color: any) {
this.initialColor = color;
}
ngOnInit() {
let hsva = this.service.stringToHsva(this.initialColor);
if (hsva !== null) {
this.hsva = hsva;
} else {
this.hsva = new Hsva(0, 1, 1, 1);
}
this.sliderDimMax = new SliderDimension(140, 230, 160, 140);
this.slider = new SliderPosition(0, 0, 0, 0);
if (this.outputFormat === 'rgba') {
this.format = 1;
} else if (this.outputFormat === 'hsla') {
this.format = 2;
} else {
this.format = 0;
}
this.listenerMouseDown = (event: any) => { this.onMouseDown(event) };
this.listenerResize = () => { this.onResize() };
this.update();
this.openColorPicker();
}
//Set position and show/hide functions ******************************************
openColorPicker() {
if (!this.show) {
this.setDialogPosition();
this.show = true;
document.addEventListener('mousedown', this.listenerMouseDown);
window.addEventListener('resize', this.listenerResize);
}
}
onMouseDown(event: any) {
if (!this.isDescendant(this.el.nativeElement, event.target)
&& event.target != this.directiveElementRef.nativeElement) {
this.closeColorPicker();
}
}
closeColorPicker() {
this.show = false;
document.removeEventListener('mouseup', this.listenerMouseDown);
window.removeEventListener('resize', this.listenerResize);
}
onResize() {
if (this.position === 'fixed') {
this.setDialogPosition();
}
}
setDialogPosition() {
var node = this.directiveElementRef.nativeElement, position = 'static';
let parentNode = null;
while (node !== null && node.tagName !== 'HTML') {
position = window.getComputedStyle(node).getPropertyValue("position");
if (position !== 'static' && parentNode === null) {
parentNode = node;
}
if (position === 'fixed') {
break;
}
node = node.parentNode;
}
if (position !== 'fixed') {
var boxDirective = this.createBox(this.directiveElementRef.nativeElement, true);
if (parentNode === null) { parentNode = node }
let boxParent = this.createBox(parentNode, true);
this.top = boxDirective.top - boxParent.top;
this.left = boxDirective.left - boxParent.left;
} else {
var boxDirective = this.createBox(this.directiveElementRef.nativeElement, false);
this.top = boxDirective.top;
this.left = boxDirective.left;
this.position = 'fixed';
}
if (this.position === 'left') {
this.top += boxDirective.height * this.positionOffset / 100 - this.dialogArrowOffset;
this.left -= this.dialogWidth + this.dialogArrowSize;
} else if (this.position === 'top') {
this.top -= this.cpHeight + this.dialogArrowSize;
this.left += this.positionOffset / 100 * boxDirective.width - this.dialogArrowOffset;
this.arrowTop = this.cpHeight - 1;
} else if (this.position === 'bottom') {
this.top += boxDirective.height + this.dialogArrowSize;
this.left += this.positionOffset / 100 * boxDirective.width - this.dialogArrowOffset;
} else {
this.top += boxDirective.height * this.positionOffset / 100 - this.dialogArrowOffset;
this.left += boxDirective.width + this.dialogArrowSize;
}
}
isDescendant(parent, child) {
var node = child.parentNode;
while (node !== null) {
if (node === parent) {
return true;
}
node = node.parentNode;
}
return false;
}
createBox(element, offset) {
return {
top: element.getBoundingClientRect().top + (offset ? window.pageYOffset : 0),
left: element.getBoundingClientRect().left + (offset ? window.pageXOffset : 0),
width: element.offsetWidth,
height: element.offsetHeight
};
}
//Color value setters **************************************
setSaturation(val: { v: number, rg: number }) {
let hsla = this.service.hsva2hsla(this.hsva);
hsla.s = val.v / val.rg;
this.hsva = this.service.hsla2hsva(hsla);
this.update();
}
setLightness(val: { v: number, rg: number }) {
let hsla = this.service.hsva2hsla(this.hsva);
hsla.l = val.v / val.rg;
this.hsva = this.service.hsla2hsva(hsla);
this.update();
}
setHue(val: { v: number, rg: number }) {
this.hsva.h = val.v / val.rg;
this.update();
}
setAlpha(val: { v: number, rg: number }) {
this.hsva.a = val.v / val.rg;
this.update();
}
setR(val: { v: number, rg: number }) {
let rgba = this.service.hsvaToRgba(this.hsva);
rgba.r = val.v / val.rg;
this.hsva = this.service.rgbaToHsva(rgba);
this.update();
}
setG(val: { v: number, rg: number }) {
let rgba = this.service.hsvaToRgba(this.hsva);
rgba.g = val.v / val.rg;
this.hsva = this.service.rgbaToHsva(rgba);
this.update();
}
setB(val: { v: number, rg: number }) {
let rgba = this.service.hsvaToRgba(this.hsva);
rgba.b = val.v / val.rg;
this.hsva = this.service.rgbaToHsva(rgba);
this.update();
}
setSaturationAndBrightness(val: { s: number, v: number, rgX: number, rgY: number }) {
this.hsva.s = val.s / val.rgX;
this.hsva.v = val.v / val.rgY;
this.update();
}
setColorFromString(value: string) {
let hsva = this.service.stringToHsva(value);
if (hsva !== null) {
this.hsva = hsva;
}
this.update();
}
setFormat(set?: number) {
this.format = (set != null) ? set : (this.format + 1) % 3;
if (this.format === 0 && this.hsva.a < 1) {
this.format++;
}
return this.format;
}
update() {
let hsla = this.service.hsva2hsla(this.hsva);
let rgba = this.service.denormalizeRGBA(this.service.hsvaToRgba(this.hsva));
let hueRgba = this.service.denormalizeRGBA(this.service.hsvaToRgba(new Hsva(this.hsva.h, 1, 1, 1)));
this.hslaText = new Hsla(Math.round((hsla.h) * 360), Math.round(hsla.s * 100), Math.round(hsla.l * 100), Math.round(hsla.a * 100) / 100);
this.rgbaText = new Rgba(rgba.r, rgba.g, rgba.b, Math.round(rgba.a * 100) / 100);
this.hexText = this.service.hexText(rgba);
this.textTone = this.service.getYiq(this.hexText);
this.alphaSliderColor = 'rgb(' + rgba.r + ',' + rgba.g + ',' + rgba.b + ')';
this.hueSliderColor = 'rgb(' + hueRgba.r + ',' + hueRgba.g + ',' + hueRgba.b + ')';
if (this.format === 0 && this.hsva.a < 1) {
this.format++;
}
this.outputColor = this.service.outputFormat(this.hsva, this.outputFormat);
this.slider = new SliderPosition((this.hsva.h) * this.sliderDimMax.h - 8, this.hsva.s * this.sliderDimMax.s - 8,
(1 - this.hsva.v) * this.sliderDimMax.v - 8, this.hsva.a * this.sliderDimMax.a - 8);
this.directiveInstance.textTone = this.textTone;
this.directiveInstance.colorChanged(this.outputColor);
}
//Cancel button
cancelColor() {
this.setColorFromString(this.initialColor);
this.closeColorPicker();
}
}
<md-card class="color-picker md-whiteframe-4dp" *ngIf="show" [style.top.px]="top" [style.left.px]="left" [style.position]="position">
<md-card-content class="top">
<div class="arrow arrow-{{position}}" [style.top.px]="arrowTop"></div>
<div [slider] [style.background-color]="hueSliderColor" [rgX]="1" [rgY]="1" (newValue)="setSaturationAndBrightness($event)" class="saturation-lightness">
<div [style.left.px]="slider.s" [style.top.px]="slider.v" class="cursor"></div>
</div>
<div [slider] [rgX]="1" (newValue)="setHue($event)" class="hue slider">
<div [style.left.px]="slider.h" class="cursor">
<div class="filler" [style.background-color]="hueSliderColor"></div>
</div>
</div>
<div [slider] [style.background-color]="alphaSliderColor" [rgX]="1" (newValue)="setAlpha($event)" class="alpha slider">
<div [style.left.px]="slider.a" class="cursor">
<div class=" filler selected-color-background"></div>
<div class="filler" [style.background-color]="outputColor"></div>
</div>
</div>
<div class="selected-color-background"></div>
<div [style.background-color]="outputColor" class="selected-color"></div>
</md-card-content>
<md-card-actions layout="row" layout-align="end center" class="actions" *ngIf="hasFormatters">
<button md-button flex [disabled]="(hsva.a < 1)" [ngClass]="{'selected': (format ==0)}"
*ngIf="activeFormats.hex" (click)="setFormat(0)">#</button>
<button md-button flex (click)="setFormat(1)" [ngClass]="{'selected': (format ==1)}"
*ngIf="activeFormats.rgb" >RGB</button>
<button md-button flex (click)="setFormat(2)" [ngClass]="{'selected': (format ==2)}"
*ngIf="activeFormats.hsla" >HSLA</button>
</md-card-actions>
<md-card-content class="fields" *ngIf="hasFormatters">
<div [hidden]="format!=2" class="hsla-text" *ngIf="activeFormats.hsla">
<input [text] type="number" pattern="[0-9]*" min="0" max="360" [rg]="360" (newValue)="setHue($event)" [value]="hslaText.h"/>
<input [text] type="number" pattern="[0-9]*" min="0" max="100" [rg]="100" (newValue)="setSaturation($event)" [value]="hslaText.s"/>
<input [text] type="number" pattern="[0-9]*" min="0" max="100" [rg]="100" (newValue)="setLightness($event)" [value]="hslaText.l"/>
<input [text] type="number" pattern="[0-9]+([\.,][0-9]{1,2})?" min="0" max="1" step="0.1" [rg]="1" (newValue)="setAlpha($event)" [value]="hslaText.a"/>
<div>H</div><div>S</div><div>L</div><div>A</div>
</div>
<div [hidden]="format!=1" class="rgba-text" *ngIf="activeFormats.rgb">
<input [text] type="number" pattern="[0-9]*" min="0" max="255" [rg]="255" (newValue)="setR($event)" [value]="rgbaText.r"/>
<input [text] type="number" pattern="[0-9]*" min="0" max="255" [rg]="255" (newValue)="setG($event)" [value]="rgbaText.g"/>
<input [text] type="number" pattern="[0-9]*" min="0" max="255" [rg]="255" (newValue)="setB($event)" [value]="rgbaText.b"/>
<input [text] type="number" pattern="[0-9]+([\.,][0-9]{1,2})?" min="0" max="1" step="0.1" [rg]="1" (newValue)="setAlpha($event)" [value]="rgbaText.a"/>
<div>R</div><div>G</div><div>B</div><div>A</div>
</div>
<div [hidden]="format!=0" class="hex-text" *ngIf="activeFormats.hex">
<input [text] (newValue)="setColorFromString($event)" [value]="hexText" [style.background]="outputColor" [ngClass]="{'light':(textTone == 'light')}"/>
<div>Hex</div>
</div>
</md-card-content>
<md-card-actions layout="row" layout-align="end center" class="bottom" *ngIf="hasActions">
<button md-button md-warn class="md-warn" (click)="cancelColor()">Cancel</button>
<button md-button md-primary class="md-primary" (click)="closeColorPicker()">Ok</button>
</md-card-actions>
</md-card>
//Most functions from original color-picker source.
//CMYK functions based on: https://gist.github.com/felipesabino/5066336
export class Hsva {
constructor(public h: number, public s: number, public v: number, public a: number) { }
}
export class Hsla {
constructor(public h: number, public s: number, public l: number, public a: number) { }
}
export class Rgba {
constructor(public r: number, public g: number, public b: number, public a: number) { }
}
export class Cmyk {
constructor(public c: number, public m: number, public y: number, public k: number) { };
}
export class SliderPosition {
constructor(public h: number, public s: number, public v: number, public a: number) { }
}
export class SliderDimension {
constructor(public h: number, public s: number, public v: number, public a: number) { }
}
export class ColorPickerUtils {
constructor() { }
hsla2hsva(hsla: Hsla) {
var h = Math.min(hsla.h, 1), s = Math.min(hsla.s, 1), l = Math.min(hsla.l, 1), a = Math.min(hsla.a, 1);
if (l === 0) {
return { h: h, s: 0, v: 0, a: a };
} else {
var v = l + s * (1 - Math.abs(2 * l - 1)) / 2;
return { h: h, s: 2 * (v - l) / v, v: v, a: a };
}
}
hsva2hsla(hsva: Hsva) {
var h = hsva.h, s = hsva.s, v = hsva.v, a = hsva.a;
if (v === 0) {
return new Hsla(h, 0, 0, a)
} else if (s === 0 && v === 1) {
return new Hsla(h, 1, 1, a)
} else {
var l = v * (2 - s) / 2;
return new Hsla(h, v * s / (1 - Math.abs(2 * l - 1)), l, a)
}
}
rgbaToHsva(rgba: Rgba) {
var r = Math.min(rgba.r, 1), g = Math.min(rgba.g, 1), b = Math.min(rgba.b, 1), a = Math.min(rgba.a, 1);
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;
} 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 new Hsva(h, s, v, a)
}
hsvaToRgba(hsva: Hsva) {
var h = hsva.h, s = hsva.s, v = hsva.v, a = hsva.a;
var r, g, b;
var i = Math.floor(h * 6);
var f = h * 6 - i;
var p = v * (1 - s);
var q = v * (1 - f * s);
var t = v * (1 - (1 - f) * s);
switch (i % 6) {
case 0:
r = v, g = t, b = p;
break;
case 1:
r = q, g = v, b = p;
break;
case 2:
r = p, g = v, b = t;
break;
case 3:
r = p, g = q, b = v;
break;
case 4:
r = t, g = p, b = v;
break;
case 5:
r = v, g = p, b = q;
break;
}
return new Rgba(r, g, b, a)
}
//RGB to CMYK
rgbToCmyk(RGB: Rgba) {
var result = {c:0,m:0,y:0,k:0};
var r = RGB.r / 255;
var g = RGB.g / 255;
var b = RGB.b / 255;
result.k = Math.min(1 - r, 1 - g, 1 - b);
result.c = (1 - r - result.k) / (1 - result.k);
result.m = (1 - g - result.k) / (1 - result.k);
result.y = (1 - b - result.k) / (1 - result.k);
result.c = Math.round(result.c * 100);
result.m = Math.round(result.m * 100);
result.y = Math.round(result.y * 100);
result.k = Math.round(result.k * 100);
return result;
}
cmykToRgb(CMYK:Cmyk) {
var result = new Rgba(0, 0, 0, 1);
var c = CMYK.c / 100;
var m = CMYK.m / 100;
var y = CMYK.y / 100;
var k = CMYK.k / 100;
result.r = 1 - Math.min(1, c * (1 - k) + k);
result.g = 1 - Math.min(1, m * (1 - k) + k);
result.b = 1 - Math.min(1, y * (1 - k) + k);
result.r = Math.round(result.r * 255);
result.g = Math.round(result.g * 255);
result.b = Math.round(result.b * 255);
return result;
}
//get text tone.
//yiq formula from: https://24ways.org/2010/calculating-color-contrast/
getYiq(color:string) {
var processed = this.hsvaToRgba(this.stringToHsva(color));
var yiq = ((processed.r*255 * 299) + (processed.g*255 * 587) + (processed.b*255 * 114)) / 1000;
return (yiq >= 128) ? 'dark' : 'light';
}
stringToHsva(colorString: string) {
var stringParsers = [
{
re: /(rgb)a?\(\s*(\d{1,3})\s*,\s*(\d{1,3})\s*%?,\s*(\d{1,3})\s*%?(?:,\s*(\d+(?:\.\d+)?)\s*)?\)/,
parse: function(execResult) {
return new Rgba(parseInt(execResult[2]) / 255,
parseInt(execResult[3]) / 255,
parseInt(execResult[4]) / 255,
isNaN(parseFloat(execResult[5])) ? 1 : parseFloat(execResult[5]));
}
},
{
re: /(hsl)a?\(\s*(\d{1,3})\s*,\s*(\d{1,3})%\s*,\s*(\d{1,3})%\s*(?:,\s*(\d+(?:\.\d+)?)\s*)?\)/,
parse: function(execResult) {
return new Hsla(parseInt(execResult[2]) / 360,
parseInt(execResult[3]) / 100,
parseInt(execResult[4]) / 100,
isNaN(parseFloat(execResult[5])) ? 1 : parseFloat(execResult[5]));
}
},
{
re: /#([a-fA-F0-9]{2})([a-fA-F0-9]{2})([a-fA-F0-9]{2})$/,
parse: function(execResult) {
return new Rgba(parseInt(execResult[1], 16) / 255,
parseInt(execResult[2], 16) / 255,
parseInt(execResult[3], 16) / 255,
1);
}
},
{
re: /#([a-fA-F0-9])([a-fA-F0-9])([a-fA-F0-9])$/,
parse: function(execResult) {
return new Rgba(parseInt(execResult[1] + execResult[1], 16) / 255,
parseInt(execResult[2] + execResult[2], 16) / 255,
parseInt(execResult[3] + execResult[3], 16) / 255,
1);
}
}
];
colorString = colorString.toLowerCase();
var hsva:Hsva = null;
for (var key in stringParsers) {
if (stringParsers.hasOwnProperty(key)) {
var parser = stringParsers[key];
var match = parser.re.exec(colorString), color = match && parser.parse(match);
if (color) {
if (color instanceof Rgba) {
hsva = this.rgbaToHsva(color);
} else if (color instanceof Hsla) {
hsva = this.hsla2hsva(color);
}
return hsva;
}
}
}
return hsva;
}
outputFormat(hsva: Hsva, outputFormat: string) {
if (hsva.a < 1) {
switch (outputFormat) {
case 'hsla':
let hsla = this.hsva2hsla(hsva);
let hslaText = new Hsla(Math.round((hsla.h) * 360), Math.round(hsla.s * 100), Math.round(hsla.l * 100), Math.round(hsla.a * 100) / 100);
return 'hsla(' + hslaText.h + ',' + hslaText.s + '%,' + hslaText.l + '%,' + hslaText.a + ')';
default:
let rgba = this.denormalizeRGBA(this.hsvaToRgba(hsva));
return 'rgba(' + rgba.r + ',' + rgba.g + ',' + rgba.b + ',' + Math.round(rgba.a * 100) / 100 + ')';
}
} else {
switch (outputFormat) {
case 'hsla':
let hsla = this.hsva2hsla(hsva);
let hslaText = new Hsla(Math.round((hsla.h) * 360), Math.round(hsla.s * 100), Math.round(hsla.l * 100), Math.round(hsla.a * 100) / 100);
return 'hsl(' + hslaText.h + ',' + hslaText.s + '%,' + hslaText.l + '%)';
case 'rgba':
let rgba = this.denormalizeRGBA(this.hsvaToRgba(hsva));
return 'rgb(' + rgba.r + ',' + rgba.g + ',' + rgba.b + ')';
default:
return this.hexText(this.denormalizeRGBA(this.hsvaToRgba(hsva)));
}
}
}
hexText(rgba: Rgba) {
let hexText = '#' + ((1 << 24) | (rgba.r << 16) | (rgba.g << 8) | rgba.b).toString(16).substr(1);
if (hexText[1] === hexText[2] && hexText[3] === hexText[4] && hexText[5] === hexText[6]) {
hexText = '#' + hexText[1] + hexText[3] + hexText[5];
}
return hexText;
}
denormalizeRGBA(rgba: Rgba) {
return new Rgba(Math.round(rgba.r * 255), Math.round(rgba.g * 255), Math.round(rgba.b * 255), rgba.a);
}
}
/*
* Styles for Color Picker
*
* Alberto Pujante
*
* @licence: http://opensource.org/licenses/MIT
*/
.color-picker, .color-picker *{
-webkit-box-sizing: border-box;
-moz-box-sizing: border-box;
box-sizing: border-box;
margin:0;
font-size:11px;
}
@mixin boder-radius($radius) {
-moz-border-radius: $radius;
-webkit-border-radius: $radius;
border-radius: $radius;
-khtml-border-radius: $radius;
}
button {
color: #009688;
&[md-button] {
min-width: 50px;
}
}
color-picker {
.color-picker{
height: 228px;
}
&.has-formatters {
.color-picker{
height: 248px;
}
&.has-actions {
.color-picker{
height: 362px;
}
}
}
&.has-actions {
.color-picker{
height: 255px;
}
}
}
.color-picker{
width: 230px;
border: none;
left:30px;
top:250px;
position:absolute;
z-index:1000;
background-color: #fff;
-webkit-touch-callout: none;
-webkit-user-select: none;
-khtml-user-select: none;
-moz-user-select: none;
-ms-user-select: none;
user-select: none;
//Inner card segments**********************************
.top {
height: 195px;
}
.actions {
button {
color: #939598;
border-radius: 0;
height: 35px;
&.selected {
color: #009688;
border-bottom: 2px solid #009688;
}
}
}
.bottom {
position: absolute;
bottom: 16px;
width: 100%;
margin: 0 -24px 0;
button[md-button] {
padding: 0 8px;
&[md-warn] {
margin-right: 0;
}
&[md-primary] {
margin-left: 0;
}
}
}
//Class Helper for light text
.light {
color: #ffffff;
}
i{
cursor:default;
position:relative;
}
input{
text-align: center;
font-size: 13px;
height: 26px;
&:invalid{
box-shadow: none;
}
&:-moz-submit-invalid{
box-shadow: none;
}
&:-moz-ui-invalid{
box-shadow: none;
}
&::-webkit-inner-spin-button,
&::-webkit-outer-spin-button {
-webkit-appearance: none;
margin: 0;
}
-moz-appearance: textfield;
}
//******** Legacy arrows and positioning
.arrow{
height: 0;
width: 0;
border-style: solid;
position:absolute;
z-index: 999999;
}
.arrow-right{
border-width: 5px 10px;
border-color: rgba(0,0,0,0) #777 rgba(0,0,0,0) rgba(0,0,0,0);
top: 10px;
left:-20px;
}
.arrow-left{
border-width: 5px 10px;
border-color: rgba(0,0,0,0) rgba(0,0,0,0) rgba(0,0,0,0) #777;
top: 10px;
left:231px;
}
.arrow-bottom{
border-width: 10px 5px;
border-color: rgba(0,0,0,0) rgba(0,0,0,0) #777 rgba(0,0,0,0);
top: -20px;
left:10px;
}
.arrow-top{
border-width: 10px 5px;
border-color: #777 rgba(0,0,0,0) rgba(0,0,0,0) rgba(0,0,0,0);
left:10px;
}
//** Sliders and child components
div.cursor-sv{
cursor:default;
position:relative;
@include boder-radius(50%);
width: 15px;
height: 15px;
border: #ddd solid 1px;
}
div.slider.active .cursor .filler{
display: block;
};
div.cursor{
cursor:default;
position:relative;
@include boder-radius(50%);
width: 16px;
height: 16px;
border: #222 solid 2px;
top: -5px;
.filler {
display: none;
position: absolute;
height: 100%;
width: 100%;
@include boder-radius(50%);
top: 0;
left: 0;
}
}
//** Sliders
.saturation-lightness{
width: 230px;
height: 160px;
border:none;
top:0;
left:0;
position:absolute;
background-size: cover;
background-image: url('');
}
.hue{
width: 140px;
height: 5px;
border:none;
top:175px;
left:70px;
position:absolute;
background-size: cover;
background-image: url('');
}
.alpha{
width: 140px;
height: 5px;
border:none;
top:205px;
left:70px;
position:absolute;
background-size: cover;
background-image: url('');
}
.selected-color{
width: 40px;
height: 40px;
top:173px;
left:15px;
position:absolute;
@include boder-radius(50%);
}
.selected-color-background{
width: 40px;
height: 40px;
top:173px;
left:15px;
position:absolute;
@include boder-radius(50%);
background-image: url('');
}
.type-policy{
position: absolute;
top: 256px;
left: 206px;
background-image: url('');
background-repeat: no-repeat;
background-position: center;
background-size: 8px 16px;
-moz-background-size: 8px 16px;
-webkit-background-size: 8px 16px;
-o-background-size: 8px 16px;
width:16px;
height:24px;
}
//** Text Boxes
.hsla-text, .rgba-text{
position:absolute;
top: 280px;
left: 12px;
font-size:11px;
input{
margin: 0;
float:left;
width: 40px;
margin-left:7px;
border: #d3d3d3 solid 1px;
padding: 1px;
}
div{
float:left;
width: 40px;
text-align: center;
color: #555;
margin-left:7px;
margin-top: 4px;
}
div:nth-child(5){
clear:left;
}
}
.hex-text{
position:absolute;
top: 280px;
left: 30px;
font-size: 11px;
input{
float:left;
width: 160px;
border: #d3d3d3 solid 1px;
padding: 1px;
}
div{
text-align: center;
color: #555;
float:left;
clear: left;
width: 160px;
margin-top: 4px;
}
}
}
/*
* Styles for Color Picker
*
* Alberto Pujante
*
* @licence: http://opensource.org/licenses/MIT
*/
.color-picker, .color-picker * {
box-sizing: border-box;
margin: 0;
font-size: 11px; }
button {
color: #009688; }
button[md-button] {
min-width: 50px; }
color-picker .color-picker {
height: 228px; }
color-picker.has-formatters .color-picker {
height: 248px; }
color-picker.has-formatters.has-actions .color-picker {
height: 362px; }
color-picker.has-actions .color-picker {
height: 255px; }
.color-picker md-card-actions {
display: flex;
}
.color-picker {
width: 230px;
border: none;
left: 30px;
top: 250px;
position: absolute;
z-index: 1000;
background-color: #fff;
-webkit-touch-callout: none;
-webkit-user-select: none;
-moz-user-select: none;
-ms-user-select: none;
user-select: none; }
.color-picker .top {
height: 195px; }
.color-picker .actions button {
color: #939598;
border-radius: 0;
height: 35px; }
.color-picker .actions button.selected {
color: #009688;
border-bottom: 2px solid #009688; }
.color-picker .bottom {
position: absolute;
justify-content: flex-end;
bottom: 16px;
width: 100%;
margin: 0 -24px 0; }
.color-picker .bottom button[md-button] {
padding: 0 8px; }
.color-picker .bottom button[md-button][md-warn] {
margin-right: 0; }
.color-picker .bottom button[md-button][md-primary] {
margin-left: 0; }
.color-picker .light {
color: #ffffff; }
.color-picker i {
cursor: default;
position: relative; }
.color-picker input {
text-align: center;
font-size: 13px;
height: 26px;
-moz-appearance: textfield; }
.color-picker input:invalid {
box-shadow: none; }
.color-picker input:-moz-submit-invalid {
box-shadow: none; }
.color-picker input:-moz-ui-invalid {
box-shadow: none; }
.color-picker input::-webkit-inner-spin-button, .color-picker input::-webkit-outer-spin-button {
-webkit-appearance: none;
margin: 0; }
.color-picker .arrow {
height: 0;
width: 0;
border-style: solid;
position: absolute;
z-index: 999999; }
.color-picker .arrow-right {
border-width: 5px 10px;
border-color: transparent #777 transparent transparent;
top: 10px;
left: -20px; }
.color-picker .arrow-left {
border-width: 5px 10px;
border-color: transparent transparent transparent #777;
top: 10px;
left: 231px; }
.color-picker .arrow-bottom {
border-width: 10px 5px;
border-color: transparent transparent #777 transparent;
top: -20px;
left: 10px; }
.color-picker .arrow-top {
border-width: 10px 5px;
border-color: #777 transparent transparent transparent;
left: 10px; }
.color-picker div.cursor-sv {
cursor: default;
position: relative;
border-radius: 50%;
-khtml-border-radius: 50%;
width: 15px;
height: 15px;
border: #ddd solid 1px; }
.color-picker div.slider.active .cursor .filler {
display: block; }
.color-picker div.cursor {
cursor: default;
position: relative;
border-radius: 50%;
-khtml-border-radius: 50%;
width: 16px;
height: 16px;
border: #222 solid 2px;
top: -5px; }
.color-picker div.cursor .filler {
display: none;
position: absolute;
height: 100%;
width: 100%;
border-radius: 50%;
-khtml-border-radius: 50%;
top: 0;
left: 0; }
.color-picker .saturation-lightness {
width: 230px;
height: 160px;
border: none;
top: 0;
left: 0;
position: absolute;
background-size: cover;
background-image: url(""); }
.color-picker .hue {
width: 140px;
height: 5px;
border: none;
top: 175px;
left: 70px;
position: absolute;
background-size: cover;
background-image: url(""); }
.color-picker .alpha {
width: 140px;
height: 5px;
border: none;
top: 205px;
left: 70px;
position: absolute;
background-size: cover;
background-image: url(""); }
.color-picker .selected-color {
width: 40px;
height: 40px;
top: 173px;
left: 15px;
position: absolute;
border-radius: 50%;
-khtml-border-radius: 50%; }
.color-picker .selected-color-background {
width: 40px;
height: 40px;
top: 173px;
left: 15px;
position: absolute;
border-radius: 50%;
-khtml-border-radius: 50%;
background-image: url(""); }
.color-picker .type-policy {
position: absolute;
top: 256px;
left: 206px;
background-image: url("");
background-repeat: no-repeat;
background-position: center;
background-size: 8px 16px;
-moz-background-size: 8px 16px;
-webkit-background-size: 8px 16px;
-o-background-size: 8px 16px;
width: 16px;
height: 24px; }
.color-picker .hsla-text, .color-picker .rgba-text {
position: absolute;
top: 280px;
left: 12px;
font-size: 11px; }
.color-picker .hsla-text input, .color-picker .rgba-text input {
margin: 0;
float: left;
width: 40px;
margin-left: 7px;
border: #d3d3d3 solid 1px;
padding: 1px; }
.color-picker .hsla-text div, .color-picker .rgba-text div {
float: left;
width: 40px;
text-align: center;
color: #555;
margin-left: 7px;
margin-top: 4px; }
.color-picker .hsla-text div:nth-child(5), .color-picker .rgba-text div:nth-child(5) {
clear: left; }
.color-picker .hex-text {
position: absolute;
top: 280px;
left: 30px;
font-size: 11px; }
.color-picker .hex-text input {
float: left;
width: 160px;
border: #d3d3d3 solid 1px;
padding: 1px; }
.color-picker .hex-text div {
text-align: center;
color: #555;
float: left;
clear: left;
width: 160px;
margin-top: 4px; }