@acrodata/gradient-picker
Version:
A powerful and beautiful gradient picker.
99 lines • 29.7 kB
JavaScript
import { booleanAttribute, ChangeDetectionStrategy, ChangeDetectorRef, Component, forwardRef, inject, Input, ViewEncapsulation, } from '@angular/core';
import { FormsModule, NG_VALUE_ACCESSOR } from '@angular/forms';
import { GradientCheckbox, GradientFormGroup, GradientIconButton, GradientInputField, GradientRadioButton, GradientUnitInput, } from './form-controls';
import { GradientStops } from './gradient-stops';
import { parseRadialGradient, stringifyRadialGradient } from './parser';
import { hueInterpolationMethods, lengthUnits, polarColorSpaces, positionXKeywords, positionYKeywords, rectangularColorSpaces, reverseColorStops, } from './utils';
import * as i0 from "@angular/core";
import * as i1 from "@angular/forms";
export class RadialGradientPicker {
cdr = inject(ChangeDetectorRef);
disabled = false;
radialGradient = {
repeating: false,
shape: 'ellipse',
size: [],
position: {
x: { type: 'keyword', value: 'center' },
y: { type: 'keyword', value: 'center' },
},
stops: [{ color: '#000000' }],
};
value = '';
sizeKeywords = ['farthest-corner', 'farthest-side', 'closest-corner', 'closest-side'];
lengthUnits = lengthUnits;
posXOptions = positionXKeywords;
posYOptions = positionYKeywords;
colorSpaceOptgroups = [
{ label: 'Rectangular', options: rectangularColorSpaces },
{ label: 'Polar', options: polarColorSpaces },
];
hueInterpolationMethodOptions = hueInterpolationMethods;
get isPolarColorSpace() {
return polarColorSpaces.includes(this.radialGradient.color?.space || '');
}
onChange = () => { };
onTouched = () => { };
writeValue(value) {
this.value = value || 'radial-gradient(transparent, #000000)';
this.radialGradient = parseRadialGradient(this.value);
this.cdr.markForCheck();
}
registerOnChange(fn) {
this.onChange = fn;
}
registerOnTouched(fn) {
this.onTouched = fn;
}
setDisabledState(isDisabled) {
this.disabled = isDisabled;
this.cdr.markForCheck();
}
onGradientChange() {
this.value = stringifyRadialGradient(this.radialGradient);
this.onChange(this.value);
}
reverseStops() {
this.radialGradient.stops = reverseColorStops(this.radialGradient.stops);
this.onGradientChange();
}
onColorSpaceChange() {
if (!this.isPolarColorSpace && this.radialGradient.color) {
this.radialGradient.color.method = undefined;
}
this.onGradientChange();
}
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: RadialGradientPicker, deps: [], target: i0.ɵɵFactoryTarget.Component });
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.7", type: RadialGradientPicker, isStandalone: true, selector: "radial-gradient-picker", inputs: { disabled: ["disabled", "disabled", booleanAttribute] }, host: { classAttribute: "radial-gradient-picker" }, providers: [
{
provide: NG_VALUE_ACCESSOR,
useExisting: forwardRef(() => RadialGradientPicker),
multi: true,
},
], ngImport: i0, template: "<gradient-form-group label=\"Shape\">\n <gradient-input-field>\n <label gradientRadioButton>\n <input\n type=\"radio\"\n value=\"ellipse\"\n [(ngModel)]=\"radialGradient.shape\"\n (change)=\"onGradientChange()\"\n />\n <span>Ellipse</span>\n </label>\n <label gradientRadioButton>\n <input\n type=\"radio\"\n value=\"circle\"\n [(ngModel)]=\"radialGradient.shape\"\n (change)=\"onGradientChange()\"\n />\n <span>Circle</span>\n </label>\n </gradient-input-field>\n</gradient-form-group>\n\n<gradient-form-group label=\"Size\">\n @for (size of radialGradient.size; track $index) {\n @if (size.type === 'keyword') {\n <gradient-input-field>\n <select [(ngModel)]=\"size.value\" (change)=\"onGradientChange()\">\n @for (sk of sizeKeywords; track $index) {\n <option [value]=\"sk\">{{ sk }}</option>\n }\n </select>\n </gradient-input-field>\n } @else {\n <gradient-unit-input\n [units]=\"lengthUnits\"\n [(ngModel)]=\"size.value\"\n (change)=\"onGradientChange()\"\n />\n }\n }\n</gradient-form-group>\n\n<gradient-form-group label=\"Position\">\n @if (radialGradient.position.x.type === 'keyword') {\n <gradient-input-field>\n <select [(ngModel)]=\"radialGradient.position.x.value\" (change)=\"onGradientChange()\">\n @for (posX of posXOptions; track $index) {\n <option [value]=\"posX\">{{ posX }}</option>\n }\n </select>\n </gradient-input-field>\n } @else {\n <gradient-unit-input\n [units]=\"lengthUnits\"\n [(ngModel)]=\"radialGradient.position.x.value\"\n (change)=\"onGradientChange()\"\n />\n }\n @if (radialGradient.position.y.type === 'keyword') {\n <gradient-input-field>\n <select [(ngModel)]=\"radialGradient.position.y.value\" (change)=\"onGradientChange()\">\n @for (posY of posYOptions; track $index) {\n <option [value]=\"posY\">{{ posY }}</option>\n }\n </select>\n </gradient-input-field>\n } @else {\n <gradient-unit-input\n [units]=\"lengthUnits\"\n [(ngModel)]=\"radialGradient.position.y.value\"\n (change)=\"onGradientChange()\"\n />\n }\n</gradient-form-group>\n\n@if (radialGradient.color) {\n <gradient-form-group label=\"Color interpolation\">\n <gradient-input-field>\n <select [(ngModel)]=\"radialGradient.color.space\" (change)=\"onColorSpaceChange()\">\n @for (colorSpaceGroup of colorSpaceOptgroups; track $index) {\n <optgroup [label]=\"colorSpaceGroup.label\">\n @for (colorSpace of colorSpaceGroup.options; track $index) {\n <option [value]=\"colorSpace\">{{ colorSpace }}</option>\n }\n </optgroup>\n }\n </select>\n </gradient-input-field>\n @if (isPolarColorSpace) {\n <gradient-input-field>\n <select [(ngModel)]=\"radialGradient.color.method\" (change)=\"onGradientChange()\">\n @for (hueInterp of hueInterpolationMethodOptions; track $index) {\n <option [value]=\"hueInterp\">{{ hueInterp }}</option>\n }\n </select>\n </gradient-input-field>\n }\n </gradient-form-group>\n}\n\n<gradient-form-group>\n <label gradientCheckbox>\n <input type=\"checkbox\" [(ngModel)]=\"radialGradient.repeating\" (change)=\"onGradientChange()\" />\n <span>Repeat</span>\n </label>\n\n <gradient-icon-button>\n <button type=\"button\" (click)=\"reverseStops()\" title=\"Reverse stops\" aria-label=\"Reverse stops\">\n <svg viewBox=\"0 0 24 24\">\n <path\n fill=\"currentColor\"\n d=\"M8.354 6.354a.5.5 0 1 0-.708-.708l-2.5 2.5a.5.5 0 0 0 0 .708l2.5 2.5a.5.5 0 0 0 .708-.708L6.707 9H18.5a.5.5 0 0 0 0-1H6.707zm7.292 7a.5.5 0 0 1 .708-.708l2.5 2.5a.5.5 0 0 1 0 .708l-2.5 2.5a.5.5 0 0 1-.708-.708L17.293 16H5.5a.5.5 0 0 1 0-1h11.793z\"\n />\n </svg>\n </button>\n </gradient-icon-button>\n</gradient-form-group>\n\n<gradient-stops [ngModel]=\"radialGradient.stops\" (ngModelChange)=\"onGradientChange()\" />\n", styles: [""], 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.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i1.CheckboxControlValueAccessor, selector: "input[type=checkbox][formControlName],input[type=checkbox][formControl],input[type=checkbox][ngModel]" }, { kind: "directive", type: i1.SelectControlValueAccessor, selector: "select:not([multiple])[formControlName],select:not([multiple])[formControl],select:not([multiple])[ngModel]", inputs: ["compareWith"] }, { kind: "directive", type: i1.RadioControlValueAccessor, selector: "input[type=radio][formControlName],input[type=radio][formControl],input[type=radio][ngModel]", inputs: ["name", "formControlName", "value"] }, { 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: GradientStops, selector: "gradient-stops", inputs: ["disabled", "colorStops"], outputs: ["colorStopsChange"] }, { kind: "component", type: GradientInputField, selector: "gradient-input-field" }, { kind: "component", type: GradientFormGroup, selector: "gradient-form-group", inputs: ["label"] }, { kind: "component", type: GradientUnitInput, selector: "gradient-unit-input", inputs: ["disabled", "units"] }, { kind: "component", type: GradientCheckbox, selector: "[gradientCheckbox]" }, { kind: "component", type: GradientRadioButton, selector: "[gradientRadioButton]" }, { kind: "component", type: GradientIconButton, selector: "gradient-icon-button" }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None });
}
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: RadialGradientPicker, decorators: [{
type: Component,
args: [{ selector: 'radial-gradient-picker', standalone: true, imports: [
FormsModule,
GradientStops,
GradientInputField,
GradientFormGroup,
GradientUnitInput,
GradientCheckbox,
GradientRadioButton,
GradientIconButton,
], host: {
class: 'radial-gradient-picker',
}, encapsulation: ViewEncapsulation.None, changeDetection: ChangeDetectionStrategy.OnPush, providers: [
{
provide: NG_VALUE_ACCESSOR,
useExisting: forwardRef(() => RadialGradientPicker),
multi: true,
},
], template: "<gradient-form-group label=\"Shape\">\n <gradient-input-field>\n <label gradientRadioButton>\n <input\n type=\"radio\"\n value=\"ellipse\"\n [(ngModel)]=\"radialGradient.shape\"\n (change)=\"onGradientChange()\"\n />\n <span>Ellipse</span>\n </label>\n <label gradientRadioButton>\n <input\n type=\"radio\"\n value=\"circle\"\n [(ngModel)]=\"radialGradient.shape\"\n (change)=\"onGradientChange()\"\n />\n <span>Circle</span>\n </label>\n </gradient-input-field>\n</gradient-form-group>\n\n<gradient-form-group label=\"Size\">\n @for (size of radialGradient.size; track $index) {\n @if (size.type === 'keyword') {\n <gradient-input-field>\n <select [(ngModel)]=\"size.value\" (change)=\"onGradientChange()\">\n @for (sk of sizeKeywords; track $index) {\n <option [value]=\"sk\">{{ sk }}</option>\n }\n </select>\n </gradient-input-field>\n } @else {\n <gradient-unit-input\n [units]=\"lengthUnits\"\n [(ngModel)]=\"size.value\"\n (change)=\"onGradientChange()\"\n />\n }\n }\n</gradient-form-group>\n\n<gradient-form-group label=\"Position\">\n @if (radialGradient.position.x.type === 'keyword') {\n <gradient-input-field>\n <select [(ngModel)]=\"radialGradient.position.x.value\" (change)=\"onGradientChange()\">\n @for (posX of posXOptions; track $index) {\n <option [value]=\"posX\">{{ posX }}</option>\n }\n </select>\n </gradient-input-field>\n } @else {\n <gradient-unit-input\n [units]=\"lengthUnits\"\n [(ngModel)]=\"radialGradient.position.x.value\"\n (change)=\"onGradientChange()\"\n />\n }\n @if (radialGradient.position.y.type === 'keyword') {\n <gradient-input-field>\n <select [(ngModel)]=\"radialGradient.position.y.value\" (change)=\"onGradientChange()\">\n @for (posY of posYOptions; track $index) {\n <option [value]=\"posY\">{{ posY }}</option>\n }\n </select>\n </gradient-input-field>\n } @else {\n <gradient-unit-input\n [units]=\"lengthUnits\"\n [(ngModel)]=\"radialGradient.position.y.value\"\n (change)=\"onGradientChange()\"\n />\n }\n</gradient-form-group>\n\n@if (radialGradient.color) {\n <gradient-form-group label=\"Color interpolation\">\n <gradient-input-field>\n <select [(ngModel)]=\"radialGradient.color.space\" (change)=\"onColorSpaceChange()\">\n @for (colorSpaceGroup of colorSpaceOptgroups; track $index) {\n <optgroup [label]=\"colorSpaceGroup.label\">\n @for (colorSpace of colorSpaceGroup.options; track $index) {\n <option [value]=\"colorSpace\">{{ colorSpace }}</option>\n }\n </optgroup>\n }\n </select>\n </gradient-input-field>\n @if (isPolarColorSpace) {\n <gradient-input-field>\n <select [(ngModel)]=\"radialGradient.color.method\" (change)=\"onGradientChange()\">\n @for (hueInterp of hueInterpolationMethodOptions; track $index) {\n <option [value]=\"hueInterp\">{{ hueInterp }}</option>\n }\n </select>\n </gradient-input-field>\n }\n </gradient-form-group>\n}\n\n<gradient-form-group>\n <label gradientCheckbox>\n <input type=\"checkbox\" [(ngModel)]=\"radialGradient.repeating\" (change)=\"onGradientChange()\" />\n <span>Repeat</span>\n </label>\n\n <gradient-icon-button>\n <button type=\"button\" (click)=\"reverseStops()\" title=\"Reverse stops\" aria-label=\"Reverse stops\">\n <svg viewBox=\"0 0 24 24\">\n <path\n fill=\"currentColor\"\n d=\"M8.354 6.354a.5.5 0 1 0-.708-.708l-2.5 2.5a.5.5 0 0 0 0 .708l2.5 2.5a.5.5 0 0 0 .708-.708L6.707 9H18.5a.5.5 0 0 0 0-1H6.707zm7.292 7a.5.5 0 0 1 .708-.708l2.5 2.5a.5.5 0 0 1 0 .708l-2.5 2.5a.5.5 0 0 1-.708-.708L17.293 16H5.5a.5.5 0 0 1 0-1h11.793z\"\n />\n </svg>\n </button>\n </gradient-icon-button>\n</gradient-form-group>\n\n<gradient-stops [ngModel]=\"radialGradient.stops\" (ngModelChange)=\"onGradientChange()\" />\n" }]
}], propDecorators: { disabled: [{
type: Input,
args: [{ transform: booleanAttribute }]
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"radial-gradient-picker.js","sourceRoot":"","sources":["../../../../projects/gradient-picker/src/lib/radial-gradient-picker.ts","../../../../projects/gradient-picker/src/lib/radial-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 {\n  GradientCheckbox,\n  GradientFormGroup,\n  GradientIconButton,\n  GradientInputField,\n  GradientRadioButton,\n  GradientUnitInput,\n} from './form-controls';\nimport { GradientStops } from './gradient-stops';\nimport { parseRadialGradient, RadialGradientResult, stringifyRadialGradient } from './parser';\nimport {\n  hueInterpolationMethods,\n  lengthUnits,\n  polarColorSpaces,\n  positionXKeywords,\n  positionYKeywords,\n  rectangularColorSpaces,\n  reverseColorStops,\n} from './utils';\n\n@Component({\n  selector: 'radial-gradient-picker',\n  standalone: true,\n  imports: [\n    FormsModule,\n    GradientStops,\n    GradientInputField,\n    GradientFormGroup,\n    GradientUnitInput,\n    GradientCheckbox,\n    GradientRadioButton,\n    GradientIconButton,\n  ],\n  templateUrl: './radial-gradient-picker.html',\n  styleUrl: './radial-gradient-picker.scss',\n  host: {\n    class: 'radial-gradient-picker',\n  },\n  encapsulation: ViewEncapsulation.None,\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      useExisting: forwardRef(() => RadialGradientPicker),\n      multi: true,\n    },\n  ],\n})\nexport class RadialGradientPicker implements ControlValueAccessor {\n  private cdr = inject(ChangeDetectorRef);\n\n  @Input({ transform: booleanAttribute }) disabled = false;\n\n  radialGradient: RadialGradientResult = {\n    repeating: false,\n    shape: 'ellipse',\n    size: [],\n    position: {\n      x: { type: 'keyword', value: 'center' },\n      y: { type: 'keyword', value: 'center' },\n    },\n    stops: [{ color: '#000000' }],\n  };\n\n  value = '';\n\n  sizeKeywords = ['farthest-corner', 'farthest-side', 'closest-corner', 'closest-side'];\n\n  lengthUnits = lengthUnits;\n\n  posXOptions = positionXKeywords;\n\n  posYOptions = positionYKeywords;\n\n  colorSpaceOptgroups = [\n    { label: 'Rectangular', options: rectangularColorSpaces },\n    { label: 'Polar', options: polarColorSpaces },\n  ];\n\n  hueInterpolationMethodOptions = hueInterpolationMethods;\n\n  get isPolarColorSpace() {\n    return polarColorSpaces.includes(this.radialGradient.color?.space || '');\n  }\n\n  private onChange: (value: string) => void = () => {};\n  private onTouched: () => void = () => {};\n\n  writeValue(value: string): void {\n    this.value = value || 'radial-gradient(transparent, #000000)';\n    this.radialGradient = parseRadialGradient(this.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  onGradientChange() {\n    this.value = stringifyRadialGradient(this.radialGradient);\n    this.onChange(this.value);\n  }\n\n  reverseStops() {\n    this.radialGradient.stops = reverseColorStops(this.radialGradient.stops);\n    this.onGradientChange();\n  }\n\n  onColorSpaceChange() {\n    if (!this.isPolarColorSpace && this.radialGradient.color) {\n      this.radialGradient.color.method = undefined;\n    }\n    this.onGradientChange();\n  }\n}\n","<gradient-form-group label=\"Shape\">\n  <gradient-input-field>\n    <label gradientRadioButton>\n      <input\n        type=\"radio\"\n        value=\"ellipse\"\n        [(ngModel)]=\"radialGradient.shape\"\n        (change)=\"onGradientChange()\"\n      />\n      <span>Ellipse</span>\n    </label>\n    <label gradientRadioButton>\n      <input\n        type=\"radio\"\n        value=\"circle\"\n        [(ngModel)]=\"radialGradient.shape\"\n        (change)=\"onGradientChange()\"\n      />\n      <span>Circle</span>\n    </label>\n  </gradient-input-field>\n</gradient-form-group>\n\n<gradient-form-group label=\"Size\">\n  @for (size of radialGradient.size; track $index) {\n    @if (size.type === 'keyword') {\n      <gradient-input-field>\n        <select [(ngModel)]=\"size.value\" (change)=\"onGradientChange()\">\n          @for (sk of sizeKeywords; track $index) {\n            <option [value]=\"sk\">{{ sk }}</option>\n          }\n        </select>\n      </gradient-input-field>\n    } @else {\n      <gradient-unit-input\n        [units]=\"lengthUnits\"\n        [(ngModel)]=\"size.value\"\n        (change)=\"onGradientChange()\"\n      />\n    }\n  }\n</gradient-form-group>\n\n<gradient-form-group label=\"Position\">\n  @if (radialGradient.position.x.type === 'keyword') {\n    <gradient-input-field>\n      <select [(ngModel)]=\"radialGradient.position.x.value\" (change)=\"onGradientChange()\">\n        @for (posX of posXOptions; track $index) {\n          <option [value]=\"posX\">{{ posX }}</option>\n        }\n      </select>\n    </gradient-input-field>\n  } @else {\n    <gradient-unit-input\n      [units]=\"lengthUnits\"\n      [(ngModel)]=\"radialGradient.position.x.value\"\n      (change)=\"onGradientChange()\"\n    />\n  }\n  @if (radialGradient.position.y.type === 'keyword') {\n    <gradient-input-field>\n      <select [(ngModel)]=\"radialGradient.position.y.value\" (change)=\"onGradientChange()\">\n        @for (posY of posYOptions; track $index) {\n          <option [value]=\"posY\">{{ posY }}</option>\n        }\n      </select>\n    </gradient-input-field>\n  } @else {\n    <gradient-unit-input\n      [units]=\"lengthUnits\"\n      [(ngModel)]=\"radialGradient.position.y.value\"\n      (change)=\"onGradientChange()\"\n    />\n  }\n</gradient-form-group>\n\n@if (radialGradient.color) {\n  <gradient-form-group label=\"Color interpolation\">\n    <gradient-input-field>\n      <select [(ngModel)]=\"radialGradient.color.space\" (change)=\"onColorSpaceChange()\">\n        @for (colorSpaceGroup of colorSpaceOptgroups; track $index) {\n          <optgroup [label]=\"colorSpaceGroup.label\">\n            @for (colorSpace of colorSpaceGroup.options; track $index) {\n              <option [value]=\"colorSpace\">{{ colorSpace }}</option>\n            }\n          </optgroup>\n        }\n      </select>\n    </gradient-input-field>\n    @if (isPolarColorSpace) {\n      <gradient-input-field>\n        <select [(ngModel)]=\"radialGradient.color.method\" (change)=\"onGradientChange()\">\n          @for (hueInterp of hueInterpolationMethodOptions; track $index) {\n            <option [value]=\"hueInterp\">{{ hueInterp }}</option>\n          }\n        </select>\n      </gradient-input-field>\n    }\n  </gradient-form-group>\n}\n\n<gradient-form-group>\n  <label gradientCheckbox>\n    <input type=\"checkbox\" [(ngModel)]=\"radialGradient.repeating\" (change)=\"onGradientChange()\" />\n    <span>Repeat</span>\n  </label>\n\n  <gradient-icon-button>\n    <button type=\"button\" (click)=\"reverseStops()\" title=\"Reverse stops\" aria-label=\"Reverse stops\">\n      <svg viewBox=\"0 0 24 24\">\n        <path\n          fill=\"currentColor\"\n          d=\"M8.354 6.354a.5.5 0 1 0-.708-.708l-2.5 2.5a.5.5 0 0 0 0 .708l2.5 2.5a.5.5 0 0 0 .708-.708L6.707 9H18.5a.5.5 0 0 0 0-1H6.707zm7.292 7a.5.5 0 0 1 .708-.708l2.5 2.5a.5.5 0 0 1 0 .708l-2.5 2.5a.5.5 0 0 1-.708-.708L17.293 16H5.5a.5.5 0 0 1 0-1h11.793z\"\n        />\n      </svg>\n    </button>\n  </gradient-icon-button>\n</gradient-form-group>\n\n<gradient-stops [ngModel]=\"radialGradient.stops\" (ngModelChange)=\"onGradientChange()\" />\n"]}