@junte/ui
Version:
Quality Angular UI components kit
170 lines • 17.2 kB
JavaScript
var SelectableDirective_1;
import { __decorate, __metadata, __param } from "tslib";
import { CommonModule } from '@angular/common';
import { Directive, EventEmitter, forwardRef, HostBinding, HostListener, Inject, InjectionToken, Input, NgModule, OnInit } from '@angular/core';
import { NG_VALUE_ACCESSOR } from '@angular/forms';
import { NGXLogger } from 'ngx-logger';
import { PropertyApi } from '../decorators/api';
import { UI } from '../enums/ui';
var SelectMode;
(function (SelectMode) {
SelectMode["single"] = "single";
SelectMode["multiple"] = "multiple";
})(SelectMode || (SelectMode = {}));
function isEqual(a, b) {
return JSON.stringify(a) === JSON.stringify(b);
}
class Config {
constructor(defs = null) {
if (!!defs) {
Object.assign(this, defs);
}
}
}
const SELECTABLE_SIGNALS = new InjectionToken('selectable_signals');
const hub = new EventEmitter();
export function eventEmitterFactory() {
return hub;
}
let SelectableDirective = SelectableDirective_1 = class SelectableDirective {
constructor(signals, logger) {
this.signals = signals;
this.logger = logger;
this.config = new Config({
mode: SelectMode.single,
enabled: true,
features: []
});
this.disabled = false;
this._mode = SelectMode.single;
this.onChange = () => this.logger.error('value accessor is not registered');
this.onTouched = () => this.logger.error('value accessor is not registered');
this.registerOnChange = fn => this.onChange = fn;
this.registerOnTouched = fn => this.onTouched = fn;
this.onBlur = () => this.onTouched();
}
get selected() {
const { value } = this.config;
return this.state.findIndex(e => isEqual(e, value)) !== -1;
}
set configure(config) {
Object.assign(this.config, config);
}
ngOnInit() {
this.signals.subscribe(state => this.state = state);
}
writeValue(value) {
this.state = !!value ? Array.isArray(value) ? value : [value] : [];
}
setDisabledState(disabled) {
this.disabled = disabled;
}
select() {
const { mode, value, enabled, features } = this.config;
if (!enabled) {
return;
}
switch (mode) {
case SelectMode.single:
const current = this.state.length > 0 ? this.state[0] : null;
if (!!current) {
const same = isEqual(current, value);
if (same && !features.includes(UI.feature.allowEmpty)) {
return;
}
this.state = same ? [] : [value];
this.onChange(same ? null : value);
}
else {
this.state = [value];
this.onChange(value);
}
break;
case SelectMode.multiple:
const index = this.state.findIndex(e => isEqual(e, value));
if (index !== -1) {
this.state.splice(index, 1);
}
else {
this.state.push(value);
}
this.onChange(this.state);
break;
}
this.signals.emit(this.state);
}
};
SelectableDirective.ctorParameters = () => [
{ type: EventEmitter, decorators: [{ type: Inject, args: [SELECTABLE_SIGNALS,] }] },
{ type: NGXLogger }
];
__decorate([
HostBinding('attr.data-disabled'),
__metadata("design:type", Object)
], SelectableDirective.prototype, "disabled", void 0);
__decorate([
HostBinding('attr.data-selected'),
__metadata("design:type", Object),
__metadata("design:paramtypes", [])
], SelectableDirective.prototype, "selected", null);
__decorate([
HostBinding('attr.data-mode'),
__metadata("design:type", String)
], SelectableDirective.prototype, "_mode", void 0);
__decorate([
PropertyApi({
description: 'Selectable configuration',
type: '{mode?: SelectMode, value: any, enabled?: boolean, features?: Feature[]}',
default: '{}'
}),
Input('jntSelectable'),
__metadata("design:type", Object),
__metadata("design:paramtypes", [Object])
], SelectableDirective.prototype, "configure", null);
__decorate([
HostListener('blur'),
__metadata("design:type", Object)
], SelectableDirective.prototype, "onBlur", void 0);
__decorate([
HostListener('click'),
__metadata("design:type", Function),
__metadata("design:paramtypes", []),
__metadata("design:returntype", void 0)
], SelectableDirective.prototype, "select", null);
SelectableDirective = SelectableDirective_1 = __decorate([
Directive({
selector: '[jntSelectable]',
providers: [
{
provide: NG_VALUE_ACCESSOR,
useExisting: forwardRef(() => SelectableDirective_1),
multi: true
},
{
provide: SELECTABLE_SIGNALS,
useFactory: eventEmitterFactory
}
]
}),
__param(0, Inject(SELECTABLE_SIGNALS)),
__metadata("design:paramtypes", [EventEmitter,
NGXLogger])
], SelectableDirective);
export { SelectableDirective };
let SelectableModule = class SelectableModule {
};
SelectableModule = __decorate([
NgModule({
declarations: [
SelectableDirective
],
imports: [
CommonModule
],
exports: [
SelectableDirective
]
})
], SelectableModule);
export { SelectableModule };
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"selectable.js","sourceRoot":"ng://@junte/ui/","sources":["lib/core/directives/selectable.ts"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport {\n  Directive,\n  EventEmitter,\n  forwardRef,\n  HostBinding,\n  HostListener,\n  Inject,\n  InjectionToken,\n  Input,\n  NgModule,\n  OnInit\n} from '@angular/core';\nimport { ControlValueAccessor, NG_VALUE_ACCESSOR } from '@angular/forms';\nimport { NGXLogger } from 'ngx-logger';\nimport { PropertyApi } from '../decorators/api';\nimport { Feature } from '../enums/feature';\nimport { UI } from '../enums/ui';\n\nenum SelectMode {\n  single = 'single',\n  multiple = 'multiple'\n}\n\nfunction isEqual(a, b) {\n  return JSON.stringify(a) === JSON.stringify(b);\n}\n\nclass Config {\n  mode: SelectMode;\n  value: any;\n  enabled: true;\n  features: Feature[];\n\n  constructor(defs: any = null) {\n    if (!!defs) {\n      Object.assign(this, defs);\n    }\n  }\n}\n\nconst SELECTABLE_SIGNALS = new InjectionToken('selectable_signals');\n\nconst hub = new EventEmitter();\n\nexport function eventEmitterFactory() {\n  return hub;\n}\n\n@Directive({\n  selector: '[jntSelectable]',\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      useExisting: forwardRef(() => SelectableDirective),\n      multi: true\n    },\n    {\n      provide: SELECTABLE_SIGNALS,\n      useFactory: eventEmitterFactory\n    }]\n})\nexport class SelectableDirective implements OnInit, ControlValueAccessor {\n\n  config: Config = new Config({\n    mode: SelectMode.single,\n    enabled: true,\n    features: []\n  });\n\n  @HostBinding('attr.data-disabled')\n  disabled = false;\n\n  @HostBinding('attr.data-selected')\n  get selected() {\n    const {value} = this.config;\n    return this.state.findIndex(e => isEqual(e, value)) !== -1;\n  }\n\n  @HostBinding('attr.data-mode')\n  _mode: SelectMode = SelectMode.single;\n\n  @PropertyApi({\n    description: 'Selectable configuration',\n    type: '{mode?: SelectMode, value: any, enabled?: boolean, features?: Feature[]}',\n    default: '{}'\n  })\n  @Input('jntSelectable')\n  set configure(config: {\n    mode?: SelectMode,\n    value: any,\n    enabled?: boolean,\n    features?: Feature[]\n  }) {\n    Object.assign(this.config, config);\n  }\n\n  state: any[];\n\n  onChange: (value: any) => void = () => this.logger.error('value accessor is not registered');\n  onTouched: () => void = () => this.logger.error('value accessor is not registered');\n  registerOnChange = fn => this.onChange = fn;\n  registerOnTouched = fn => this.onTouched = fn;\n  @HostListener('blur') onBlur = () => this.onTouched();\n\n  constructor(@Inject(SELECTABLE_SIGNALS) private signals: EventEmitter<any>,\n              private logger: NGXLogger) {\n  }\n\n  ngOnInit() {\n    this.signals.subscribe(state => this.state = state);\n  }\n\n  writeValue(value: any | any[]) {\n    this.state = !!value ? Array.isArray(value) ? value : [value] : [];\n  }\n\n  setDisabledState(disabled: boolean) {\n    this.disabled = disabled;\n  }\n\n  @HostListener('click')\n  select() {\n    const {mode, value, enabled, features} = this.config;\n    if (!enabled) {\n      return;\n    }\n\n    switch (mode) {\n      case SelectMode.single:\n        const current = this.state.length > 0 ? this.state[0] : null;\n        if (!!current) {\n          const same = isEqual(current, value);\n          if (same && !features.includes(UI.feature.allowEmpty)) {\n            return;\n          }\n          this.state = same ? [] : [value];\n          this.onChange(same ? null : value);\n        } else {\n          this.state = [value];\n          this.onChange(value);\n        }\n        break;\n      case SelectMode.multiple:\n        const index = this.state.findIndex(e => isEqual(e, value));\n        if (index !== -1) {\n          this.state.splice(index, 1);\n        } else {\n          this.state.push(value);\n        }\n        this.onChange(this.state);\n        break;\n    }\n    this.signals.emit(this.state);\n  }\n\n}\n\n@NgModule({\n  declarations: [\n    SelectableDirective\n  ],\n  imports: [\n    CommonModule\n  ],\n  exports: [\n    SelectableDirective\n  ]\n})\nexport class SelectableModule {\n}\n"]}