@xui/components
Version:
xUI Components for Angular
114 lines (112 loc) • 14.5 kB
JavaScript
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>
(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>
(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"]}