@acrodata/gradient-picker
Version:
A powerful and beautiful gradient picker.
103 lines • 19.9 kB
JavaScript
import { booleanAttribute, ChangeDetectionStrategy, ChangeDetectorRef, Component, forwardRef, inject, Input, ViewEncapsulation, } from '@angular/core';
import { FormsModule, NG_VALUE_ACCESSOR } from '@angular/forms';
import { ConicGradientPicker } from './conic-gradient-picker';
import { GradientFormGroup, GradientInputField } from './form-controls';
import { LinearGradientPicker } from './linear-gradient-picker';
import { RadialGradientPicker } from './radial-gradient-picker';
import { convertAngleToPercentage, parseGradient } from './utils';
import * as i0 from "@angular/core";
import * as i1 from "@angular/forms";
export class GradientPicker {
cdr = inject(ChangeDetectorRef);
disabled = false;
value = '';
types = [
{ label: 'Linear', value: 'linear' },
{ label: 'Radial', value: 'radial' },
{ label: 'Conic', value: 'conic' },
];
type = 'linear';
onChange = () => { };
onTouched = () => { };
writeValue(value) {
if (!value) {
this.type = 'linear';
}
else if (value.includes('linear')) {
this.type = 'linear';
}
else if (value.includes('radial')) {
this.type = 'radial';
}
else if (value.includes('conic')) {
this.type = 'conic';
}
this.value = value;
this.cdr.markForCheck();
}
registerOnChange(fn) {
this.onChange = fn;
}
registerOnTouched(fn) {
this.onTouched = fn;
}
setDisabledState(isDisabled) {
this.disabled = isDisabled;
this.cdr.markForCheck();
}
onValueChange() {
this.onChange(this.value);
}
onTypeChange() {
const { color, stops } = parseGradient(this.value) || {};
const props = [];
const stopsStr = convertAngleToPercentage(stops || [])
.map(s => s.color + (s.offset ? ` ${s.offset.value}${s.offset.unit}` : ''))
.join(', ') || 'transparent, #000000';
if (color && color.space) {
props.push(`in ${color.space} ${color.method || ''}`.trim());
}
props.push(stopsStr);
if (this.type === 'linear') {
this.value = `linear-gradient(${props.join(', ')})`;
}
else if (this.type === 'radial') {
this.value = `radial-gradient(${props.join(', ')})`;
}
else if (this.type === 'conic') {
this.value = `conic-gradient(${props.join(', ')})`;
}
this.onValueChange();
}
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: GradientPicker, deps: [], target: i0.ɵɵFactoryTarget.Component });
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.7", type: GradientPicker, isStandalone: true, selector: "gradient-picker", inputs: { disabled: ["disabled", "disabled", booleanAttribute] }, host: { classAttribute: "gradient-picker" }, providers: [
{
provide: NG_VALUE_ACCESSOR,
useExisting: forwardRef(() => GradientPicker),
multi: true,
},
], ngImport: i0, template: "<gradient-form-group label=\"Type\">\n <gradient-input-field>\n <select [(ngModel)]=\"type\" (ngModelChange)=\"onTypeChange()\">\n @for (t of types; track $index) {\n <option [value]=\"t.value\">{{ t.label }}</option>\n }\n </select>\n </gradient-input-field>\n</gradient-form-group>\n\n@if (type === 'linear') {\n <linear-gradient-picker [(ngModel)]=\"value\" (ngModelChange)=\"onValueChange()\" />\n} @else if (type === 'radial') {\n <radial-gradient-picker [(ngModel)]=\"value\" (ngModelChange)=\"onValueChange()\" />\n} @else if (type === 'conic') {\n <conic-gradient-picker [(ngModel)]=\"value\" (ngModelChange)=\"onValueChange()\" />\n}\n", styles: [".gradient-picker{--gp-container-text-color: light-dark(rgba(0, 0, 0, .9), #fff);--gp-container-background-color: light-dark(#fff, #2c2c2c);--gp-input-background-color: light-dark(#f5f5f5, #383838);--gp-input-outline-color: light-dark(#e6e6e6, #444);--gp-input-hover-outline-color: light-dark(#d6d6d6, #545454);--gp-input-focus-outline-color: light-dark(#0d99ff, #0c8ce9);--gp-unit-select-hover-background-color: light-dark(rgba(0, 0, 0, .12), rgba(255, 255, 255, .12));--gp-icon-button-hover-background-color: light-dark(rgba(0, 0, 0, .06), rgba(255, 255, 255, .06));--gp-icon-button-active-background-color: light-dark(rgba(0, 0, 0, .12), rgba(255, 255, 255, .12));--gp-stops-slider-bar-outline-color: light-dark(rgba(0, 0, 0, .12), rgba(255, 255, 255, .12));--gp-stops-slider-thumb-background-color: light-dark(#fff, #383838);--gp-stops-slider-thumb-toggle-outline-color: light-dark(rgba(0, 0, 0, .12), rgba(255, 255, 255, .12));--gp-stop-item-active-color: light-dark(#e5f4ff, #4a5878);display:block;width:var(--gp-container-width, 240px);padding:var(--gp-container-vertical-padding, 8px) 0;margin:var(--gp-container-margin, 0);box-shadow:var(--gp-container-elevation-shadow, inset 0 0 1px 0 rgba(255, 255, 255, .64), 0 2px 4px 0 rgba(0, 0, 0, .16), 0 8px 16px 0 rgba(0, 0, 0, .12), 0 0 1px 0 rgba(0, 0, 0, .12));background-color:var(--gp-container-background-color);border-radius:var(--gp-container-shape, 8px)}\n"], dependencies: [{ kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i1.NgSelectOption, selector: "option", inputs: ["ngValue", "value"] }, { kind: "directive", type: i1.ɵNgSelectMultipleOption, selector: "option", inputs: ["ngValue", "value"] }, { kind: "directive", type: i1.SelectControlValueAccessor, selector: "select:not([multiple])[formControlName],select:not([multiple])[formControl],select:not([multiple])[ngModel]", inputs: ["compareWith"] }, { kind: "directive", type: i1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i1.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "component", type: LinearGradientPicker, selector: "linear-gradient-picker", inputs: ["disabled"] }, { kind: "component", type: RadialGradientPicker, selector: "radial-gradient-picker", inputs: ["disabled"] }, { kind: "component", type: ConicGradientPicker, selector: "conic-gradient-picker", inputs: ["disabled"] }, { kind: "component", type: GradientFormGroup, selector: "gradient-form-group", inputs: ["label"] }, { kind: "component", type: GradientInputField, selector: "gradient-input-field" }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None });
}
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: GradientPicker, decorators: [{
type: Component,
args: [{ selector: 'gradient-picker', standalone: true, imports: [
FormsModule,
LinearGradientPicker,
RadialGradientPicker,
ConicGradientPicker,
GradientFormGroup,
GradientInputField,
], host: {
class: 'gradient-picker',
}, encapsulation: ViewEncapsulation.None, changeDetection: ChangeDetectionStrategy.OnPush, providers: [
{
provide: NG_VALUE_ACCESSOR,
useExisting: forwardRef(() => GradientPicker),
multi: true,
},
], template: "<gradient-form-group label=\"Type\">\n <gradient-input-field>\n <select [(ngModel)]=\"type\" (ngModelChange)=\"onTypeChange()\">\n @for (t of types; track $index) {\n <option [value]=\"t.value\">{{ t.label }}</option>\n }\n </select>\n </gradient-input-field>\n</gradient-form-group>\n\n@if (type === 'linear') {\n <linear-gradient-picker [(ngModel)]=\"value\" (ngModelChange)=\"onValueChange()\" />\n} @else if (type === 'radial') {\n <radial-gradient-picker [(ngModel)]=\"value\" (ngModelChange)=\"onValueChange()\" />\n} @else if (type === 'conic') {\n <conic-gradient-picker [(ngModel)]=\"value\" (ngModelChange)=\"onValueChange()\" />\n}\n", styles: [".gradient-picker{--gp-container-text-color: light-dark(rgba(0, 0, 0, .9), #fff);--gp-container-background-color: light-dark(#fff, #2c2c2c);--gp-input-background-color: light-dark(#f5f5f5, #383838);--gp-input-outline-color: light-dark(#e6e6e6, #444);--gp-input-hover-outline-color: light-dark(#d6d6d6, #545454);--gp-input-focus-outline-color: light-dark(#0d99ff, #0c8ce9);--gp-unit-select-hover-background-color: light-dark(rgba(0, 0, 0, .12), rgba(255, 255, 255, .12));--gp-icon-button-hover-background-color: light-dark(rgba(0, 0, 0, .06), rgba(255, 255, 255, .06));--gp-icon-button-active-background-color: light-dark(rgba(0, 0, 0, .12), rgba(255, 255, 255, .12));--gp-stops-slider-bar-outline-color: light-dark(rgba(0, 0, 0, .12), rgba(255, 255, 255, .12));--gp-stops-slider-thumb-background-color: light-dark(#fff, #383838);--gp-stops-slider-thumb-toggle-outline-color: light-dark(rgba(0, 0, 0, .12), rgba(255, 255, 255, .12));--gp-stop-item-active-color: light-dark(#e5f4ff, #4a5878);display:block;width:var(--gp-container-width, 240px);padding:var(--gp-container-vertical-padding, 8px) 0;margin:var(--gp-container-margin, 0);box-shadow:var(--gp-container-elevation-shadow, inset 0 0 1px 0 rgba(255, 255, 255, .64), 0 2px 4px 0 rgba(0, 0, 0, .16), 0 8px 16px 0 rgba(0, 0, 0, .12), 0 0 1px 0 rgba(0, 0, 0, .12));background-color:var(--gp-container-background-color);border-radius:var(--gp-container-shape, 8px)}\n"] }]
}], propDecorators: { disabled: [{
type: Input,
args: [{ transform: booleanAttribute }]
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"gradient-picker.js","sourceRoot":"","sources":["../../../../projects/gradient-picker/src/lib/gradient-picker.ts","../../../../projects/gradient-picker/src/lib/gradient-picker.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  booleanAttribute,\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  forwardRef,\n  inject,\n  Input,\n  ViewEncapsulation,\n} from '@angular/core';\nimport { ControlValueAccessor, FormsModule, NG_VALUE_ACCESSOR } from '@angular/forms';\nimport { ConicGradientPicker } from './conic-gradient-picker';\nimport { GradientFormGroup, GradientInputField } from './form-controls';\nimport { LinearGradientPicker } from './linear-gradient-picker';\nimport { RadialGradientPicker } from './radial-gradient-picker';\nimport { convertAngleToPercentage, parseGradient } from './utils';\n\n@Component({\n  selector: 'gradient-picker',\n  standalone: true,\n  imports: [\n    FormsModule,\n    LinearGradientPicker,\n    RadialGradientPicker,\n    ConicGradientPicker,\n    GradientFormGroup,\n    GradientInputField,\n  ],\n  templateUrl: './gradient-picker.html',\n  styleUrl: './gradient-picker.scss',\n  host: {\n    class: 'gradient-picker',\n  },\n  encapsulation: ViewEncapsulation.None,\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      useExisting: forwardRef(() => GradientPicker),\n      multi: true,\n    },\n  ],\n})\nexport class GradientPicker implements ControlValueAccessor {\n  private cdr = inject(ChangeDetectorRef);\n\n  @Input({ transform: booleanAttribute }) disabled = false;\n\n  value = '';\n\n  types = [\n    { label: 'Linear', value: 'linear' },\n    { label: 'Radial', value: 'radial' },\n    { label: 'Conic', value: 'conic' },\n  ];\n\n  type: 'linear' | 'radial' | 'conic' = 'linear';\n\n  private onChange: (value: string) => void = () => {};\n  private onTouched: () => void = () => {};\n\n  writeValue(value: string): void {\n    if (!value) {\n      this.type = 'linear';\n    } else if (value.includes('linear')) {\n      this.type = 'linear';\n    } else if (value.includes('radial')) {\n      this.type = 'radial';\n    } else if (value.includes('conic')) {\n      this.type = 'conic';\n    }\n    this.value = value;\n    this.cdr.markForCheck();\n  }\n\n  registerOnChange(fn: (value: string) => void): void {\n    this.onChange = fn;\n  }\n\n  registerOnTouched(fn: () => void): void {\n    this.onTouched = fn;\n  }\n\n  setDisabledState(isDisabled: boolean) {\n    this.disabled = isDisabled;\n    this.cdr.markForCheck();\n  }\n\n  onValueChange() {\n    this.onChange(this.value);\n  }\n\n  onTypeChange() {\n    const { color, stops } = parseGradient(this.value) || {};\n\n    const props: string[] = [];\n\n    const stopsStr =\n      convertAngleToPercentage(stops || [])\n        .map(s => s.color + (s.offset ? ` ${s.offset.value}${s.offset.unit}` : ''))\n        .join(', ') || 'transparent, #000000';\n\n    if (color && color.space) {\n      props.push(`in ${color.space} ${color.method || ''}`.trim());\n    }\n\n    props.push(stopsStr);\n\n    if (this.type === 'linear') {\n      this.value = `linear-gradient(${props.join(', ')})`;\n    } else if (this.type === 'radial') {\n      this.value = `radial-gradient(${props.join(', ')})`;\n    } else if (this.type === 'conic') {\n      this.value = `conic-gradient(${props.join(', ')})`;\n    }\n\n    this.onValueChange();\n  }\n}\n","<gradient-form-group label=\"Type\">\n  <gradient-input-field>\n    <select [(ngModel)]=\"type\" (ngModelChange)=\"onTypeChange()\">\n      @for (t of types; track $index) {\n        <option [value]=\"t.value\">{{ t.label }}</option>\n      }\n    </select>\n  </gradient-input-field>\n</gradient-form-group>\n\n@if (type === 'linear') {\n  <linear-gradient-picker [(ngModel)]=\"value\" (ngModelChange)=\"onValueChange()\" />\n} @else if (type === 'radial') {\n  <radial-gradient-picker [(ngModel)]=\"value\" (ngModelChange)=\"onValueChange()\" />\n} @else if (type === 'conic') {\n  <conic-gradient-picker [(ngModel)]=\"value\" (ngModelChange)=\"onValueChange()\" />\n}\n"]}