UNPKG

@xui/components

Version:

xUI Components for Angular

159 lines 25.9 kB
import { ChangeDetectionStrategy, Component, EventEmitter, input, Output, signal } from '@angular/core'; import { ComponentPortal } from '@angular/cdk/portal'; import { animate, state, style, transition, trigger, useAnimation } from '@angular/animations'; import { bounce } from '../utils/animations'; import { delay } from '../utils'; import { XuiSnackBar } from '../snack-bar'; import { SaveResetSnackbar } from './settings-snackbar'; import * as i0 from "@angular/core"; import * as i1 from "../snack-bar"; import * as i2 from "../icon/icon"; import * as i3 from "@angular/cdk/portal"; import * as i4 from "@angular/cdk/a11y"; import * as i5 from "@ngx-translate/core"; export class XuiSettings { constructor(snackBar) { this.snackBar = snackBar; this.canExit = true; this._animationState = false; this._mouseDown = false; this._menuFocused = false; this._focusedItem = signal(null); this._defaultPage = signal(1); this._isOpened = signal(false); this._openedAnimation = signal('closed'); this.items = input(); this.afterClosed = new EventEmitter(); this.stateChanged = (canExit) => { this.canExit = canExit; if (!canExit) { if (!this.snackbarRef) { this.snackbarRef = this.snackBar.openFromComponent(SaveResetSnackbar, { panelClass: 'x-settings-snackbar-panel', duration: undefined, data: { save: async () => { try { await this.instance?.save(); await this.hideSnackbar(); } catch (e) { // TODO: show error snackbar } }, reset: async () => { await this.instance?.reset(); await this.hideSnackbar(); } } }); } } else { this.hideSnackbar(); } }; } open(page = 1) { this._defaultPage.set(page); this._isOpened.set(true); this._openedAnimation.set('opened'); this._navigate(this._defaultPage()); } async close() { if (!this.canExit) { this._animationState = true; this.snackbarRef?.instance.alert(); return; } this._openedAnimation.set('closed'); await delay(100); this._isOpened.set(false); this.afterClosed.emit(); } attached(ref) { this.instance = ref.instance; this.instance.stateChanged = this.stateChanged; } _navigate(idx) { if (this.instance) { if (!this.canExit) { return; } } const item = this.items()?.[idx]; if (item) { this._defaultPage.set(idx); this._focusedItem.set(idx); if (item.action) { item.action(); } else if (item.component) { this._portal = new ComponentPortal(item.component); } } } _focusPrev() { let cur = this._focusedItem() ?? this.items()?.length ?? 0; do { cur--; if (cur < 0) { cur = (this.items()?.length ?? 1) - 1; } } while (this.items()?.[cur].type !== 'item'); this._focusedItem.set(cur); } _focusNext() { let cur = this._focusedItem() ?? this.items()?.length ?? 0; do { cur++; if (cur >= (this.items()?.length ?? 0)) { cur = 0; } } while (this.items()?.[cur].type !== 'item'); this._focusedItem.set(cur); } async hideSnackbar() { await this.snackbarRef?.instance.close(); this.snackbarRef?.dismiss(); this.snackbarRef = undefined; this.canExit = true; } static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.0.1", ngImport: i0, type: XuiSettings, deps: [{ token: i1.XuiSnackBar }], target: i0.ɵɵFactoryTarget.Component }); } static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.0.1", type: XuiSettings, selector: "xui-settings", inputs: { items: { classPropertyName: "items", publicName: "items", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { afterClosed: "afterClosed" }, host: { listeners: { "document:keydown.escape": "close()" } }, ngImport: i0, template: "<dialog class=\"x-settings-dialog\" [open]=\"_isOpened()\" cdkTrapFocus cdkTrapFocusAutoCapture>\n <div\n class=\"x-settings\"\n [@open]=\"_openedAnimation()\"\n [@bounce]=\"_animationState\"\n (@bounce.done)=\"_animationState = false\"\n >\n <div class=\"x-settings-menu\">\n <nav\n tabindex=\"0\"\n (keydown.enter)=\"_navigate(_focusedItem() ?? 0)\"\n (keydown.space)=\"_navigate(_focusedItem() ?? 0)\"\n (keydown.arrowDown)=\"_focusNext()\"\n (keydown.arrowUp)=\"_focusPrev()\"\n (mousedown)=\"_mouseDown = true\"\n (mouseup)=\"_mouseDown = false\"\n (focusin)=\"_menuFocused = !_mouseDown\"\n (focusout)=\"_menuFocused = false\"\n >\n @for (item of items(); track item; let i = $index) {\n <div\n [class.x-settings-menu-item]=\"item.type === 'item'\"\n [class.x-settings-menu-item-active]=\"i === _defaultPage()\"\n [class.x-settings-menu-item-focus]=\"i === _focusedItem() && _menuFocused\"\n [class.x-settings-menu-category]=\"item.type === 'category'\"\n [class.x-settings-menu-divider]=\"item.type === 'divider'\"\n [class.x-settings-menu-red]=\"item.critical\"\n (click)=\"_navigate(i); _menuFocused = false\"\n >\n {{ item.name ? (item.name | translate) : '' }}\n </div>\n }\n </nav>\n </div>\n <div class=\"x-settings-right\">\n <div class=\"x-settings-content\">\n <ng-template [cdkPortalOutlet]=\"_portal\" (attached)=\"attached($event)\" />\n </div>\n\n <div class=\"x-settings-close-anchor\">\n <div class=\"x-settings-close\" (click)=\"close()\">\n <xui-icon tabindex=\"0\" (keydown.space)=\"close()\" (keydown.enter)=\"close()\" icon=\"cancel\"></xui-icon>\n ESC\n </div>\n </div>\n </div>\n </div>\n</dialog>\n", dependencies: [{ kind: "component", type: i2.XuiIcon, selector: "xui-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "icon"], exportAs: ["xuiIcon"] }, { kind: "directive", type: i3.CdkPortalOutlet, selector: "[cdkPortalOutlet]", inputs: ["cdkPortalOutlet"], outputs: ["attached"], exportAs: ["cdkPortalOutlet"] }, { kind: "directive", type: i4.CdkTrapFocus, selector: "[cdkTrapFocus]", inputs: ["cdkTrapFocus", "cdkTrapFocusAutoCapture"], exportAs: ["cdkTrapFocus"] }, { kind: "pipe", type: i5.TranslatePipe, name: "translate" }], animations: [ trigger('bounce', [transition('* => *', useAnimation(bounce))]), trigger('open', [ state('closed', style({ opacity: 0, transform: 'scale(1.05)' })), state('opened', style({ opacity: 1, transform: 'scale(1)' })), transition('* => *', animate(100)) ]) ], changeDetection: i0.ChangeDetectionStrategy.OnPush }); } } i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.0.1", ngImport: i0, type: XuiSettings, decorators: [{ type: Component, args: [{ selector: 'xui-settings', changeDetection: ChangeDetectionStrategy.OnPush, animations: [ trigger('bounce', [transition('* => *', useAnimation(bounce))]), trigger('open', [ state('closed', style({ opacity: 0, transform: 'scale(1.05)' })), state('opened', style({ opacity: 1, transform: 'scale(1)' })), transition('* => *', animate(100)) ]) ], host: { '(document:keydown.escape)': 'close()' }, template: "<dialog class=\"x-settings-dialog\" [open]=\"_isOpened()\" cdkTrapFocus cdkTrapFocusAutoCapture>\n <div\n class=\"x-settings\"\n [@open]=\"_openedAnimation()\"\n [@bounce]=\"_animationState\"\n (@bounce.done)=\"_animationState = false\"\n >\n <div class=\"x-settings-menu\">\n <nav\n tabindex=\"0\"\n (keydown.enter)=\"_navigate(_focusedItem() ?? 0)\"\n (keydown.space)=\"_navigate(_focusedItem() ?? 0)\"\n (keydown.arrowDown)=\"_focusNext()\"\n (keydown.arrowUp)=\"_focusPrev()\"\n (mousedown)=\"_mouseDown = true\"\n (mouseup)=\"_mouseDown = false\"\n (focusin)=\"_menuFocused = !_mouseDown\"\n (focusout)=\"_menuFocused = false\"\n >\n @for (item of items(); track item; let i = $index) {\n <div\n [class.x-settings-menu-item]=\"item.type === 'item'\"\n [class.x-settings-menu-item-active]=\"i === _defaultPage()\"\n [class.x-settings-menu-item-focus]=\"i === _focusedItem() && _menuFocused\"\n [class.x-settings-menu-category]=\"item.type === 'category'\"\n [class.x-settings-menu-divider]=\"item.type === 'divider'\"\n [class.x-settings-menu-red]=\"item.critical\"\n (click)=\"_navigate(i); _menuFocused = false\"\n >\n {{ item.name ? (item.name | translate) : '' }}\n </div>\n }\n </nav>\n </div>\n <div class=\"x-settings-right\">\n <div class=\"x-settings-content\">\n <ng-template [cdkPortalOutlet]=\"_portal\" (attached)=\"attached($event)\" />\n </div>\n\n <div class=\"x-settings-close-anchor\">\n <div class=\"x-settings-close\" (click)=\"close()\">\n <xui-icon tabindex=\"0\" (keydown.space)=\"close()\" (keydown.enter)=\"close()\" icon=\"cancel\"></xui-icon>\n ESC\n </div>\n </div>\n </div>\n </div>\n</dialog>\n" }] }], ctorParameters: () => [{ type: i1.XuiSnackBar }], propDecorators: { afterClosed: [{ type: Output }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"settings.js","sourceRoot":"","sources":["../../../../../libs/xui/src/settings/settings.ts","../../../../../libs/xui/src/settings/settings.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, ComponentRef, EventEmitter, input, Output, signal } from '@angular/core';\nimport { CdkPortalOutletAttachedRef, ComponentPortal } from '@angular/cdk/portal';\nimport { MenuItem, SettingsPage } from './settings.types';\nimport { animate, state, style, transition, trigger, useAnimation } from '@angular/animations';\nimport { bounce } from '../utils/animations';\nimport { delay } from '../utils';\nimport { SnackBarRef, XuiSnackBar } from '../snack-bar';\nimport { SaveResetSnackbar } from './settings-snackbar';\n\n@Component({\n  selector: 'xui-settings',\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  templateUrl: 'settings.html',\n  animations: [\n    trigger('bounce', [transition('* => *', useAnimation(bounce))]),\n    trigger('open', [\n      state(\n        'closed',\n        style({\n          opacity: 0,\n          transform: 'scale(1.05)'\n        })\n      ),\n      state(\n        'opened',\n        style({\n          opacity: 1,\n          transform: 'scale(1)'\n        })\n      ),\n      transition('* => *', animate(100))\n    ])\n  ],\n  host: {\n    '(document:keydown.escape)': 'close()'\n  }\n})\nexport class XuiSettings {\n  private instance?: SettingsPage;\n  private snackbarRef?: SnackBarRef<SaveResetSnackbar>;\n  private canExit = true;\n\n  _portal?: ComponentPortal<SettingsPage>;\n  _animationState = false;\n  _mouseDown = false;\n  _menuFocused = false;\n\n  _focusedItem = signal<number | null>(null);\n  _defaultPage = signal(1);\n  _isOpened = signal(false);\n  _openedAnimation = signal<'opened' | 'closed'>('closed');\n\n  items = input<MenuItem[]>();\n  @Output() afterClosed = new EventEmitter<void>();\n\n  constructor(private snackBar: XuiSnackBar) {}\n\n  stateChanged = (canExit: boolean) => {\n    this.canExit = canExit;\n\n    if (!canExit) {\n      if (!this.snackbarRef) {\n        this.snackbarRef = this.snackBar.openFromComponent(SaveResetSnackbar, {\n          panelClass: 'x-settings-snackbar-panel',\n          duration: undefined!,\n          data: {\n            save: async () => {\n              try {\n                await this.instance?.save();\n                await this.hideSnackbar();\n              } catch (e) {\n                // TODO: show error snackbar\n              }\n            },\n            reset: async () => {\n              await this.instance?.reset();\n              await this.hideSnackbar();\n            }\n          }\n        });\n      }\n    } else {\n      this.hideSnackbar();\n    }\n  };\n\n  open(page = 1) {\n    this._defaultPage.set(page);\n    this._isOpened.set(true);\n    this._openedAnimation.set('opened');\n    this._navigate(this._defaultPage());\n  }\n\n  async close() {\n    if (!this.canExit) {\n      this._animationState = true;\n      this.snackbarRef?.instance.alert();\n      return;\n    }\n\n    this._openedAnimation.set('closed');\n    await delay(100);\n    this._isOpened.set(false);\n    this.afterClosed.emit();\n  }\n\n  attached(ref: CdkPortalOutletAttachedRef) {\n    this.instance = (ref as ComponentRef<SettingsPage>).instance;\n    this.instance.stateChanged = this.stateChanged;\n  }\n\n  _navigate(idx: number) {\n    if (this.instance) {\n      if (!this.canExit) {\n        return;\n      }\n    }\n\n    const item = this.items()?.[idx];\n    if (item) {\n      this._defaultPage.set(idx);\n      this._focusedItem.set(idx);\n\n      if (item.action) {\n        item.action();\n      } else if (item.component) {\n        this._portal = new ComponentPortal(item.component);\n      }\n    }\n  }\n\n  _focusPrev() {\n    let cur = this._focusedItem() ?? this.items()?.length ?? 0;\n\n    do {\n      cur--;\n      if (cur < 0) {\n        cur = (this.items()?.length ?? 1) - 1;\n      }\n    } while (this.items()?.[cur].type !== 'item');\n    this._focusedItem.set(cur);\n  }\n\n  _focusNext() {\n    let cur = this._focusedItem() ?? this.items()?.length ?? 0;\n\n    do {\n      cur++;\n      if (cur >= (this.items()?.length ?? 0)) {\n        cur = 0;\n      }\n    } while (this.items()?.[cur].type !== 'item');\n    this._focusedItem.set(cur);\n  }\n\n  private async hideSnackbar() {\n    await this.snackbarRef?.instance.close();\n    this.snackbarRef?.dismiss();\n    this.snackbarRef = undefined;\n    this.canExit = true;\n  }\n}\n","<dialog class=\"x-settings-dialog\" [open]=\"_isOpened()\" cdkTrapFocus cdkTrapFocusAutoCapture>\n  <div\n    class=\"x-settings\"\n    [@open]=\"_openedAnimation()\"\n    [@bounce]=\"_animationState\"\n    (@bounce.done)=\"_animationState = false\"\n  >\n    <div class=\"x-settings-menu\">\n      <nav\n        tabindex=\"0\"\n        (keydown.enter)=\"_navigate(_focusedItem() ?? 0)\"\n        (keydown.space)=\"_navigate(_focusedItem() ?? 0)\"\n        (keydown.arrowDown)=\"_focusNext()\"\n        (keydown.arrowUp)=\"_focusPrev()\"\n        (mousedown)=\"_mouseDown = true\"\n        (mouseup)=\"_mouseDown = false\"\n        (focusin)=\"_menuFocused = !_mouseDown\"\n        (focusout)=\"_menuFocused = false\"\n      >\n        @for (item of items(); track item; let i = $index) {\n          <div\n            [class.x-settings-menu-item]=\"item.type === 'item'\"\n            [class.x-settings-menu-item-active]=\"i === _defaultPage()\"\n            [class.x-settings-menu-item-focus]=\"i === _focusedItem() && _menuFocused\"\n            [class.x-settings-menu-category]=\"item.type === 'category'\"\n            [class.x-settings-menu-divider]=\"item.type === 'divider'\"\n            [class.x-settings-menu-red]=\"item.critical\"\n            (click)=\"_navigate(i); _menuFocused = false\"\n          >\n            {{ item.name ? (item.name | translate) : '' }}\n          </div>\n        }\n      </nav>\n    </div>\n    <div class=\"x-settings-right\">\n      <div class=\"x-settings-content\">\n        <ng-template [cdkPortalOutlet]=\"_portal\" (attached)=\"attached($event)\" />\n      </div>\n\n      <div class=\"x-settings-close-anchor\">\n        <div class=\"x-settings-close\" (click)=\"close()\">\n          <xui-icon tabindex=\"0\" (keydown.space)=\"close()\" (keydown.enter)=\"close()\" icon=\"cancel\"></xui-icon>\n          ESC\n        </div>\n      </div>\n    </div>\n  </div>\n</dialog>\n"]}