@xui/components
Version:
xUI Components for Angular
107 lines • 23.4 kB
JavaScript
import { booleanAttribute, ChangeDetectionStrategy, Component, computed, effect, ElementRef, input, Input, Optional, Self, signal, ViewChild } from '@angular/core';
import { FormsModule, NgControl } from '@angular/forms';
import { inNextTick } from '../utils';
import { CommonModule } from '@angular/common';
import { XuiTooltip, XuiTooltipModule } from '../tooltip';
import { DragDropModule } from '@angular/cdk/drag-drop';
import { XuiFocusModule } from '../utils/focus.service';
import * as i0 from "@angular/core";
import * as i1 from "@angular/forms";
import * as i2 from "../tooltip/tooltip.directive";
import * as i3 from "@angular/cdk/drag-drop";
export class XuiSlider {
_getColor(color) {
return `x-slider-${color}`;
}
_getPercentage(absolute) {
return ((absolute - this.min()) / (this.max() - this.min())) * 100;
}
constructor(control) {
this.control = control;
this._disabled = signal(false);
this.color = input('primary');
this.secondColor = input();
this.min = input(0, { transform: (v) => Number(v) });
this.max = input(100, { transform: (v) => Number(v) });
this.step = input(1, { transform: (v) => Number(v) });
this.marks = input();
this.value = input(undefined, { transform: (v) => Number(v) });
// range = input(false, { transform: (v: string | boolean) => convertToBoolean(v) });
this.disabled = input(undefined, {
transform: booleanAttribute
});
this.tooltipDisabled = input(false, { transform: booleanAttribute });
this._value = signal(this.min());
this._tooltip = computed(() => String(this.tipFormatter ? this.tipFormatter(this._value()) : this._value()));
this.percentage = computed(() => this._getPercentage(this._value()));
this._width = computed(() => 100 - this.percentage());
this._position = computed(() => ({ x: (this.percentage() / 100) * this.hostRect.width - 5, y: 0 }));
if (this.control) {
this.control.valueAccessor = this;
}
effect(() => this.disabled() && this._disabled.set(this.disabled()), { allowSignalWrites: true });
effect(() => this.value() && this._value.set(this.value()), { allowSignalWrites: true });
effect(() => this.onChange?.(this._value()));
}
writeValue(source) {
this._value.set(source);
}
registerOnChange(onChange) {
this.onChange = onChange;
}
registerOnTouched(onTouched) {
this._onTouched = onTouched;
}
setDisabledState(isDisabled) {
this._disabled.set(isDisabled);
}
_decreaseKey() {
this._value.set(Math.min(Math.max(this._value() - this.step(), this.min()), this.max()));
}
_increaseKey() {
this._value.set(Math.min(Math.max(this._value() + this.step(), this.min()), this.max()));
}
async _move(screenX) {
if (this._disabled()) {
return;
}
const cursorOffset = 5;
const rect = this.hostRect;
const width = rect.width - 10;
const pos = Math.min(Math.max(screenX - rect.x - cursorOffset, 0), width);
const value = (pos / width) * (this.max() - this.min()); // 0 <-> max - min
const newValue = Math.round(value / this.step()) * this.step();
this._value.set(newValue + this.min());
await inNextTick();
this.tooltipRef.show();
}
get hostRect() {
return this.trackElm.nativeElement.getBoundingClientRect();
}
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.0.1", ngImport: i0, type: XuiSlider, deps: [{ token: i1.NgControl, optional: true, self: true }], target: i0.ɵɵFactoryTarget.Component }); }
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.0.1", type: XuiSlider, isStandalone: true, selector: "xui-slider", inputs: { tipFormatter: { classPropertyName: "tipFormatter", publicName: "tipFormatter", isSignal: false, isRequired: false, transformFunction: null }, color: { classPropertyName: "color", publicName: "color", isSignal: true, isRequired: false, transformFunction: null }, secondColor: { classPropertyName: "secondColor", publicName: "secondColor", isSignal: true, isRequired: false, transformFunction: null }, min: { classPropertyName: "min", publicName: "min", isSignal: true, isRequired: false, transformFunction: null }, max: { classPropertyName: "max", publicName: "max", isSignal: true, isRequired: false, transformFunction: null }, step: { classPropertyName: "step", publicName: "step", isSignal: true, isRequired: false, transformFunction: null }, marks: { classPropertyName: "marks", publicName: "marks", isSignal: true, isRequired: false, transformFunction: null }, value: { classPropertyName: "value", publicName: "value", isSignal: true, isRequired: false, transformFunction: null }, disabled: { classPropertyName: "disabled", publicName: "disabled", isSignal: true, isRequired: false, transformFunction: null }, tooltipDisabled: { classPropertyName: "tooltipDisabled", publicName: "tooltipDisabled", isSignal: true, isRequired: false, transformFunction: null } }, host: { listeners: { "click": "_move($event.x)", "focusout": "_onTouched?.()", "keydown.arrowLeft": "_decreaseKey()", "keydown.arrowRight": "_increaseKey()" }, properties: { "class.x-slider-disabled": "_disabled()" }, classAttribute: "x-slider" }, viewQueries: [{ propertyName: "trackElm", first: true, predicate: ["track"], descendants: true, static: true }, { propertyName: "tooltipRef", first: true, predicate: ["tooltipRef"], descendants: true }], ngImport: i0, template: "<div [class]=\"'x-slider-track ' + _getColor(color())\">\n <div [style.width.%]=\"_width()\" [class]=\"'x-slider-track-half ' + _getColor(secondColor() ?? 'none')\"></div>\n</div>\n\n<div class=\"x-slider-container\" #track>\n <div\n [tabindex]=\"_disabled() ? -1 : 0\"\n cdkDrag\n [cdkDragDisabled]=\"_disabled()\"\n [cdkDragFreeDragPosition]=\"_position()\"\n (cdkDragMoved)=\"_move($event.pointerPosition.x)\"\n class=\"x-slider-thumb\"\n xuiTooltipPosition=\"top\"\n [xuiTooltip]=\"_tooltip()\"\n [xuiTooltipDisabled]=\"tooltipDisabled()\"\n #tooltipRef=\"xuiTooltip\"\n ></div>\n\n @for (mark of marks(); track mark.value) {\n <div\n [style.left.%]=\"_getPercentage(mark.value)\"\n class=\"x-slider-mark\"\n [class]=\"mark.color ? 'x-slider-mark-' + mark.color : ''\"\n >\n <div class=\"x-slider-mark-label\">{{ mark.label }}</div>\n <div class=\"x-slider-mark-dash\"></div>\n </div>\n }\n</div>\n", dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "ngmodule", type: FormsModule }, { kind: "ngmodule", type: XuiTooltipModule }, { kind: "directive", type: i2.XuiTooltip, selector: "[xuiTooltip]", inputs: ["xuiTooltip", "xuiTooltipPosition", "xuiTooltipDisabled"], exportAs: ["xuiTooltip"] }, { kind: "ngmodule", type: DragDropModule }, { kind: "directive", type: i3.CdkDrag, selector: "[cdkDrag]", inputs: ["cdkDragData", "cdkDragLockAxis", "cdkDragRootElement", "cdkDragBoundary", "cdkDragStartDelay", "cdkDragFreeDragPosition", "cdkDragDisabled", "cdkDragConstrainPosition", "cdkDragPreviewClass", "cdkDragPreviewContainer"], outputs: ["cdkDragStarted", "cdkDragReleased", "cdkDragEnded", "cdkDragEntered", "cdkDragExited", "cdkDragDropped", "cdkDragMoved"], exportAs: ["cdkDrag"] }, { kind: "ngmodule", type: XuiFocusModule }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
}
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.0.1", ngImport: i0, type: XuiSlider, decorators: [{
type: Component,
args: [{ standalone: true, imports: [CommonModule, FormsModule, XuiTooltipModule, DragDropModule, XuiFocusModule], selector: 'xui-slider', changeDetection: ChangeDetectionStrategy.OnPush, host: {
class: 'x-slider',
'[class.x-slider-disabled]': '_disabled()',
'(click)': '_move($event.x)',
'(focusout)': '_onTouched?.()',
'(keydown.arrowLeft)': '_decreaseKey()',
'(keydown.arrowRight)': '_increaseKey()'
}, template: "<div [class]=\"'x-slider-track ' + _getColor(color())\">\n <div [style.width.%]=\"_width()\" [class]=\"'x-slider-track-half ' + _getColor(secondColor() ?? 'none')\"></div>\n</div>\n\n<div class=\"x-slider-container\" #track>\n <div\n [tabindex]=\"_disabled() ? -1 : 0\"\n cdkDrag\n [cdkDragDisabled]=\"_disabled()\"\n [cdkDragFreeDragPosition]=\"_position()\"\n (cdkDragMoved)=\"_move($event.pointerPosition.x)\"\n class=\"x-slider-thumb\"\n xuiTooltipPosition=\"top\"\n [xuiTooltip]=\"_tooltip()\"\n [xuiTooltipDisabled]=\"tooltipDisabled()\"\n #tooltipRef=\"xuiTooltip\"\n ></div>\n\n @for (mark of marks(); track mark.value) {\n <div\n [style.left.%]=\"_getPercentage(mark.value)\"\n class=\"x-slider-mark\"\n [class]=\"mark.color ? 'x-slider-mark-' + mark.color : ''\"\n >\n <div class=\"x-slider-mark-label\">{{ mark.label }}</div>\n <div class=\"x-slider-mark-dash\"></div>\n </div>\n }\n</div>\n" }]
}], ctorParameters: () => [{ type: i1.NgControl, decorators: [{
type: Self
}, {
type: Optional
}] }], propDecorators: { tipFormatter: [{
type: Input
}], trackElm: [{
type: ViewChild,
args: ['track', { static: true }]
}], tooltipRef: [{
type: ViewChild,
args: ['tooltipRef']
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"slider.js","sourceRoot":"","sources":["../../../../../libs/xui/src/slider/slider.ts","../../../../../libs/xui/src/slider/slider.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  booleanAttribute,\n  ChangeDetectionStrategy,\n  Component,\n  computed,\n  effect,\n  ElementRef,\n  input,\n  Input,\n  Optional,\n  Self,\n  signal,\n  ViewChild\n} from '@angular/core';\nimport { ControlValueAccessor, FormsModule, NgControl } from '@angular/forms';\nimport { inNextTick } from '../utils';\nimport { SliderColor, SliderMark } from './slider.types';\nimport { CommonModule } from '@angular/common';\nimport { XuiTooltip, XuiTooltipModule } from '../tooltip';\nimport { DragDropModule } from '@angular/cdk/drag-drop';\nimport { XuiFocusModule } from '../utils/focus.service';\n\n@Component({\n  standalone: true,\n  imports: [CommonModule, FormsModule, XuiTooltipModule, DragDropModule, XuiFocusModule],\n  selector: 'xui-slider',\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  templateUrl: 'slider.html',\n  host: {\n    class: 'x-slider',\n    '[class.x-slider-disabled]': '_disabled()',\n    '(click)': '_move($event.x)',\n    '(focusout)': '_onTouched?.()',\n    '(keydown.arrowLeft)': '_decreaseKey()',\n    '(keydown.arrowRight)': '_increaseKey()'\n  }\n})\nexport class XuiSlider implements ControlValueAccessor {\n  private onChange?: (source: number) => void;\n  _onTouched?: () => void;\n  _disabled = signal(false);\n\n  @Input() tipFormatter?: (value: number) => string;\n  color = input<SliderColor>('primary');\n  secondColor = input<SliderColor>();\n  min = input(0, { transform: (v: string | number) => Number(v) });\n  max = input(100, { transform: (v: string | number) => Number(v) });\n  step = input(1, { transform: (v: string | number) => Number(v) });\n  marks = input<SliderMark[]>();\n  value = input<number | undefined, string | number>(undefined, { transform: (v: string | number) => Number(v) });\n  // range = input(false, { transform: (v: string | boolean) => convertToBoolean(v) });\n  disabled = input<boolean | undefined, string | boolean>(undefined, {\n    transform: booleanAttribute\n  });\n  tooltipDisabled = input(false, { transform: booleanAttribute });\n\n  @ViewChild('track', { static: true }) private trackElm!: ElementRef;\n  @ViewChild('tooltipRef') private tooltipRef!: XuiTooltip;\n\n  _value = signal(this.min());\n  _tooltip = computed(() => String(this.tipFormatter ? this.tipFormatter(this._value()) : this._value()));\n  private percentage = computed(() => this._getPercentage(this._value()));\n  _width = computed(() => 100 - this.percentage());\n  _position = computed(() => ({ x: (this.percentage() / 100) * this.hostRect.width - 5, y: 0 }));\n\n  _getColor(color: string) {\n    return `x-slider-${color}`;\n  }\n\n  _getPercentage(absolute: number) {\n    return ((absolute - this.min()) / (this.max() - this.min())) * 100;\n  }\n\n  constructor(@Self() @Optional() public control?: NgControl) {\n    if (this.control) {\n      this.control.valueAccessor = this;\n    }\n\n    effect(() => this.disabled() && this._disabled.set(this.disabled()!), { allowSignalWrites: true });\n    effect(() => this.value() && this._value.set(this.value()!), { allowSignalWrites: true });\n    effect(() => this.onChange?.(this._value()));\n  }\n\n  writeValue(source: number) {\n    this._value.set(source);\n  }\n\n  registerOnChange(onChange: (source: number) => void) {\n    this.onChange = onChange;\n  }\n\n  registerOnTouched(onTouched: () => void) {\n    this._onTouched = onTouched;\n  }\n\n  setDisabledState(isDisabled: boolean): void {\n    this._disabled.set(isDisabled);\n  }\n\n  _decreaseKey() {\n    this._value.set(Math.min(Math.max(this._value() - this.step(), this.min()), this.max()));\n  }\n\n  _increaseKey() {\n    this._value.set(Math.min(Math.max(this._value() + this.step(), this.min()), this.max()));\n  }\n\n  async _move(screenX: number) {\n    if (this._disabled()) {\n      return;\n    }\n\n    const cursorOffset = 5;\n    const rect = this.hostRect;\n    const width = rect.width - 10;\n\n    const pos = Math.min(Math.max(screenX - rect.x - cursorOffset, 0), width);\n    const value = (pos / width) * (this.max() - this.min()); // 0 <-> max - min\n    const newValue = Math.round(value / this.step()) * this.step();\n    this._value.set(newValue + this.min());\n\n    await inNextTick();\n    this.tooltipRef.show();\n  }\n\n  private get hostRect() {\n    return this.trackElm.nativeElement.getBoundingClientRect() as DOMRect;\n  }\n}\n","<div [class]=\"'x-slider-track ' + _getColor(color())\">\n  <div [style.width.%]=\"_width()\" [class]=\"'x-slider-track-half ' + _getColor(secondColor() ?? 'none')\"></div>\n</div>\n\n<div class=\"x-slider-container\" #track>\n  <div\n    [tabindex]=\"_disabled() ? -1 : 0\"\n    cdkDrag\n    [cdkDragDisabled]=\"_disabled()\"\n    [cdkDragFreeDragPosition]=\"_position()\"\n    (cdkDragMoved)=\"_move($event.pointerPosition.x)\"\n    class=\"x-slider-thumb\"\n    xuiTooltipPosition=\"top\"\n    [xuiTooltip]=\"_tooltip()\"\n    [xuiTooltipDisabled]=\"tooltipDisabled()\"\n    #tooltipRef=\"xuiTooltip\"\n  ></div>\n\n  @for (mark of marks(); track mark.value) {\n    <div\n      [style.left.%]=\"_getPercentage(mark.value)\"\n      class=\"x-slider-mark\"\n      [class]=\"mark.color ? 'x-slider-mark-' + mark.color : ''\"\n    >\n      <div class=\"x-slider-mark-label\">{{ mark.label }}</div>\n      <div class=\"x-slider-mark-dash\"></div>\n    </div>\n  }\n</div>\n"]}