UNPKG

@xui/components

Version:

xUI Components for Angular

114 lines (112 loc) 14.5 kB
import { booleanAttribute, ChangeDetectionStrategy, Component, computed, ElementRef, EventEmitter, Host, input, Input, Optional, Output, signal } from '@angular/core'; import { BUTTON_MODULE, XuiConfigService } from '../config'; import { delay } from '../utils'; import { XuiButtonGroup } from './button-group'; import * as i0 from "@angular/core"; import * as i1 from "../config"; import * as i2 from "./button-group"; export class XuiButton { constructor( // TODO: Anchor for popover; consider removing it and refactoring elementRef, configService, group) { this.elementRef = elementRef; this.configService = configService; this.group = group; this._moduleName = BUTTON_MODULE; this._state = signal(0); this.type = input(); this.size = input(); this.color = input(); this.shine = input(false, { transform: booleanAttribute }); this.disabled = input(false, { transform: booleanAttribute }); this.stateDelay = input(5000); // Used to emit event when user interacts with button with spacebar or enter // eslint-disable-next-line @angular-eslint/no-output-native this.click = new EventEmitter(); this._class = computed(() => { const config = this.configService.getConfigForComponent(BUTTON_MODULE); return (`x-button-${this.size() ?? this.group?.size() ?? config?.size ?? 'medium'} ` + `x-button-${this.type() ?? this.group?.type() ?? config?.type ?? 'normal'} ` + `x-button-${this.color() ?? this.group?.color() ?? config?.color ?? 'primary'}`); }); } async _keyboardInteraction(event) { if (this.disabled()) { return; } this.click.emit(event); await this._onAsync(event); } async _onAsync(event) { if (this.disabled()) { return; } // Do not prevent events when button is disabled event?.preventDefault(); event?.stopPropagation(); if (!this.onClick) { return; } this._state.set(1); try { this._state.set((await this.onClick()) ? 2 : 3); } catch { this._state.set(2); } await delay(5000); this._state.set(0); } static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.0.1", ngImport: i0, type: XuiButton, deps: [{ token: i0.ElementRef }, { token: i1.XuiConfigService }, { token: i2.XuiButtonGroup, host: true, optional: true }], target: i0.ɵɵFactoryTarget.Component }); } static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.0.1", type: XuiButton, selector: "xui-button", inputs: { type: { classPropertyName: "type", publicName: "type", isSignal: true, isRequired: false, transformFunction: null }, size: { classPropertyName: "size", publicName: "size", isSignal: true, isRequired: false, transformFunction: null }, color: { classPropertyName: "color", publicName: "color", isSignal: true, isRequired: false, transformFunction: null }, shine: { classPropertyName: "shine", publicName: "shine", isSignal: true, isRequired: false, transformFunction: null }, disabled: { classPropertyName: "disabled", publicName: "disabled", isSignal: true, isRequired: false, transformFunction: null }, stateDelay: { classPropertyName: "stateDelay", publicName: "stateDelay", isSignal: true, isRequired: false, transformFunction: null }, onClick: { classPropertyName: "onClick", publicName: "onClick", isSignal: false, isRequired: false, transformFunction: null } }, outputs: { click: "click" }, host: { listeners: { "click": "_onAsync($event)", "keydown.enter": "_keyboardInteraction($event)", "keydown.space": "_keyboardInteraction($event)" }, properties: { "class": "_class()", "class.x-button--non-idle": "_state() != 0", "class.x-button--loading": "_state() == 1", "class.x-button--succeeded": "_state() == 2", "class.x-button--failed": "_state() == 3", "tabindex": "disabled() ? -1 : 0", "attr.disabled": "disabled() || null" }, classAttribute: "x-button" }, ngImport: i0, template: `<div class="x-button-content"> <ng-content /> </div> <div class="x-button-state-image"></div> @if (shine() && !disabled()) { <div class="x-button-shine"> <div class="x-button-shine-inner"> <div class="x-button-shine-element"></div> </div> </div> }`, isInline: true, changeDetection: i0.ChangeDetectionStrategy.OnPush }); } } i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.0.1", ngImport: i0, type: XuiButton, decorators: [{ type: Component, args: [{ selector: 'xui-button', changeDetection: ChangeDetectionStrategy.OnPush, template: `<div class="x-button-content"> <ng-content /> </div> <div class="x-button-state-image"></div> @if (shine() && !disabled()) { <div class="x-button-shine"> <div class="x-button-shine-inner"> <div class="x-button-shine-element"></div> </div> </div> }`, host: { class: 'x-button', '[class]': '_class()', '[class.x-button--non-idle]': '_state() != 0', '[class.x-button--loading]': '_state() == 1', '[class.x-button--succeeded]': '_state() == 2', '[class.x-button--failed]': '_state() == 3', '[tabindex]': 'disabled() ? -1 : 0', '[attr.disabled]': 'disabled() || null', '(click)': '_onAsync($event)', '(keydown.enter)': '_keyboardInteraction($event)', '(keydown.space)': '_keyboardInteraction($event)' } }] }], ctorParameters: () => [{ type: i0.ElementRef }, { type: i1.XuiConfigService }, { type: i2.XuiButtonGroup, decorators: [{ type: Optional }, { type: Host }] }], propDecorators: { onClick: [{ type: Input }], click: [{ type: Output }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"button.js","sourceRoot":"","sources":["../../../../../libs/xui/src/button/button.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  booleanAttribute,\n  ChangeDetectionStrategy,\n  Component,\n  computed,\n  ElementRef,\n  EventEmitter,\n  Host,\n  input,\n  Input,\n  Optional,\n  Output,\n  signal\n} from '@angular/core';\nimport { BUTTON_MODULE, XuiConfigService } from '../config';\nimport { delay } from '../utils';\nimport { ButtonColor, ButtonSize, ButtonType } from './button.types';\nimport { XuiButtonGroup } from './button-group';\n\n@Component({\n  selector: 'xui-button',\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  template: `<div class=\"x-button-content\">\n      <ng-content />\n    </div>\n    <div class=\"x-button-state-image\"></div>\n\n    @if (shine() && !disabled()) {\n      <div class=\"x-button-shine\">\n        <div class=\"x-button-shine-inner\">\n          <div class=\"x-button-shine-element\"></div>\n        </div>\n      </div>\n    }`,\n  host: {\n    class: 'x-button',\n    '[class]': '_class()',\n    '[class.x-button--non-idle]': '_state() != 0',\n    '[class.x-button--loading]': '_state() == 1',\n    '[class.x-button--succeeded]': '_state() == 2',\n    '[class.x-button--failed]': '_state() == 3',\n    '[tabindex]': 'disabled() ? -1 : 0',\n    '[attr.disabled]': 'disabled() || null',\n    '(click)': '_onAsync($event)',\n    '(keydown.enter)': '_keyboardInteraction($event)',\n    '(keydown.space)': '_keyboardInteraction($event)'\n  }\n})\nexport class XuiButton {\n  private readonly _moduleName = BUTTON_MODULE;\n  _state = signal<0 | 1 | 2 | 3>(0);\n\n  type = input<ButtonType>();\n  size = input<ButtonSize>();\n  color = input<ButtonColor>();\n  shine = input(false, { transform: booleanAttribute });\n  disabled = input(false, { transform: booleanAttribute });\n  stateDelay = input<number>(5000);\n  @Input() onClick?: () => Promise<boolean>;\n\n  // Used to emit event when user interacts with button with spacebar or enter\n  // eslint-disable-next-line @angular-eslint/no-output-native\n  @Output() readonly click = new EventEmitter<unknown>();\n\n  _class = computed(() => {\n    const config = this.configService.getConfigForComponent(BUTTON_MODULE);\n    return (\n      `x-button-${this.size() ?? this.group?.size() ?? config?.size ?? 'medium'} ` +\n      `x-button-${this.type() ?? this.group?.type() ?? config?.type ?? 'normal'} ` +\n      `x-button-${this.color() ?? this.group?.color() ?? config?.color ?? 'primary'}`\n    );\n  });\n\n  constructor(\n    // TODO: Anchor for popover; consider removing it and refactoring\n    public elementRef: ElementRef,\n    private configService: XuiConfigService,\n    @Optional() @Host() private group?: XuiButtonGroup\n  ) {}\n\n  async _keyboardInteraction(event?: MouseEvent | KeyboardEvent) {\n    if (this.disabled()) {\n      return;\n    }\n\n    this.click.emit(event);\n    await this._onAsync(event);\n  }\n\n  async _onAsync(event?: MouseEvent | KeyboardEvent | Event) {\n    if (this.disabled()) {\n      return;\n    }\n\n    // Do not prevent events when button is disabled\n    event?.preventDefault();\n    event?.stopPropagation();\n\n    if (!this.onClick) {\n      return;\n    }\n\n    this._state.set(1);\n\n    try {\n      this._state.set((await this.onClick()) ? 2 : 3);\n    } catch {\n      this._state.set(2);\n    }\n\n    await delay(5000);\n    this._state.set(0);\n  }\n}\n"]}