ng-prime-tools 1.0.62 → 1.0.64
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2022/lib/models/text-style.model.mjs +1 -1
- package/esm2022/lib/pt-form-builder/pt-form-builder.component.mjs +12 -5
- package/esm2022/lib/pt-metric-card/pt-metric-card.component.mjs +59 -39
- package/esm2022/lib/pt-side-bar-menu/pt-side-bar-menu.component.mjs +2 -2
- package/fesm2022/ng-prime-tools.mjs +71 -44
- package/fesm2022/ng-prime-tools.mjs.map +1 -1
- package/lib/models/text-style.model.d.ts +1 -0
- package/lib/models/text-style.model.d.ts.map +1 -1
- package/lib/pt-form-builder/pt-form-builder.component.d.ts +8 -1
- package/lib/pt-form-builder/pt-form-builder.component.d.ts.map +1 -1
- package/lib/pt-group/pt-group.component.d.ts +1 -1
- package/lib/pt-metric-card/pt-metric-card.component.d.ts +17 -9
- package/lib/pt-metric-card/pt-metric-card.component.d.ts.map +1 -1
- package/package.json +1 -1
|
@@ -1,2 +1,2 @@
|
|
|
1
1
|
export {};
|
|
2
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGV4dC1zdHlsZS5tb2RlbC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL25nLXByaW1lLXRvb2xzL3NyYy9saWIvbW9kZWxzL3RleHQtc3R5bGUubW9kZWwudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCBpbnRlcmZhY2UgVGV4dFN0eWxlIHtcbiAgdGV4dDogc3RyaW5nO1xuICBjb2xvcj86IHN0cmluZztcbiAgZm9udFNpemU/OiBzdHJpbmc7XG4gIGZvbnRXZWlnaHQ/
|
|
2
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGV4dC1zdHlsZS5tb2RlbC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL25nLXByaW1lLXRvb2xzL3NyYy9saWIvbW9kZWxzL3RleHQtc3R5bGUubW9kZWwudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCBpbnRlcmZhY2UgVGV4dFN0eWxlIHtcbiAgdGV4dDogc3RyaW5nO1xuICBjb2xvcj86IHN0cmluZztcbiAgZm9udFNpemU/OiBzdHJpbmc7XG4gIGZvbnRXZWlnaHQ/OiBzdHJpbmc7XG4gIHBvc2l0aW9uPzogJ2xlZnQnIHwgJ2NlbnRlcicgfCAncmlnaHQnO1xufVxuIl19
|
|
@@ -11,11 +11,17 @@ export class PTFormBuilderComponent {
|
|
|
11
11
|
this.fb = fb;
|
|
12
12
|
this.mainGroup = { fields: [], groups: [] };
|
|
13
13
|
this.buttons = [];
|
|
14
|
+
/**
|
|
15
|
+
* Largeur des inputs/champs
|
|
16
|
+
*/
|
|
14
17
|
this.inputWidth = '100%';
|
|
18
|
+
/**
|
|
19
|
+
* Nouvelle largeur globale du composant pt-form-builder
|
|
20
|
+
*/
|
|
21
|
+
this.formWidth = '100%';
|
|
15
22
|
this.language = 'en';
|
|
16
23
|
this.formSubmit = new EventEmitter();
|
|
17
24
|
this.formReady = new EventEmitter();
|
|
18
|
-
// ✅ NEW
|
|
19
25
|
this.formChange = new EventEmitter();
|
|
20
26
|
this.FormInputTypeEnum = FormInputTypeEnum;
|
|
21
27
|
this.errorMessages = {
|
|
@@ -52,7 +58,6 @@ export class PTFormBuilderComponent {
|
|
|
52
58
|
this.form = this.fb.group({});
|
|
53
59
|
this.buildFormGroup(this.mainGroup);
|
|
54
60
|
this.formReady.emit(this.form);
|
|
55
|
-
// ✅ NEW: emit on every form change
|
|
56
61
|
this.formValueChangesSub = this.form.valueChanges.subscribe((value) => {
|
|
57
62
|
this.formChange.emit(value);
|
|
58
63
|
});
|
|
@@ -133,11 +138,11 @@ export class PTFormBuilderComponent {
|
|
|
133
138
|
this.formChange.emit(this.form.getRawValue());
|
|
134
139
|
}
|
|
135
140
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.11", ngImport: i0, type: PTFormBuilderComponent, deps: [{ token: i1.FormBuilder }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
136
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.11", type: PTFormBuilderComponent, selector: "pt-form-builder", inputs: { mainGroup: "mainGroup", buttons: "buttons", title: "title", titleStyle: "titleStyle", inputWidth: "inputWidth", language: "language" }, outputs: { formSubmit: "formSubmit", formReady: "formReady", formChange: "formChange" }, usesOnChanges: true, ngImport: i0, template: "<div *ngIf=\"title\" [ngStyle]=\"titleStyle\" class=\"form-title\">{{ title }}</div>\n<form [formGroup]=\"form\" (ngSubmit)=\"onSubmit()\">\n
|
|
141
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.11", type: PTFormBuilderComponent, selector: "pt-form-builder", inputs: { mainGroup: "mainGroup", buttons: "buttons", title: "title", titleStyle: "titleStyle", inputWidth: "inputWidth", formWidth: "formWidth", language: "language" }, outputs: { formSubmit: "formSubmit", formReady: "formReady", formChange: "formChange" }, usesOnChanges: true, ngImport: i0, template: "<div\n class=\"pt-form-builder-wrapper\"\n [ngStyle]=\"{ width: formWidth, maxWidth: formWidth }\"\n>\n <div *ngIf=\"title\" [ngStyle]=\"titleStyle\" class=\"form-title\">{{ title }}</div>\n\n <form [formGroup]=\"form\" (ngSubmit)=\"onSubmit()\">\n <ng-container *ngFor=\"let field of mainGroup.fields\">\n <pt-dynamic-form-field\n [field]=\"field\"\n [form]=\"form\"\n [inputWidth]=\"inputWidth\"\n ></pt-dynamic-form-field>\n </ng-container>\n\n <ng-container *ngFor=\"let group of mainGroup.groups\">\n <div\n class=\"form-field-group\"\n [ngStyle]=\"{ width: group.width || '100%' }\"\n >\n <ng-container *ngFor=\"let field of group.fields\">\n <pt-dynamic-form-field\n [field]=\"field\"\n [form]=\"form\"\n [inputWidth]=\"inputWidth\"\n class=\"flex-item\"\n ></pt-dynamic-form-field>\n </ng-container>\n </div>\n </ng-container>\n\n <div class=\"button-group\">\n <button\n *ngFor=\"let button of buttons\"\n type=\"button\"\n pButton\n [label]=\"button.text\"\n [icon]=\"button.icon || ''\"\n [class]=\"button.color\"\n (click)=\"\n button.isSubmit\n ? onSubmit()\n : button.isClear\n ? onClear()\n : button.action\n ? button.action()\n : null\n \"\n ></button>\n </div>\n </form>\n</div>\n", styles: [".pt-form-builder-wrapper{width:100%;margin:0 auto}.form-title{text-align:center;margin-bottom:1rem;font-size:1.5rem;font-weight:700}.form-field{margin-bottom:1rem}.form-field label{display:block;margin-bottom:.5rem;font-weight:700}::ng-deep .p-inputtext,::ng-deep .p-inputtextarea,::ng-deep .p-calendar,::ng-deep .p-inputnumber,::ng-deep .p-multiselect,::ng-deep .p-dropdown{width:100%!important}::ng-deep .p-inputnumber,::ng-deep p-inputnumber{display:flex!important}.button-group{display:flex;gap:1rem;margin-top:1rem;justify-content:space-between}.button-group button{flex:1;display:flex;align-items:center;justify-content:center}.button-group button i{margin-right:.5rem}.button-group button.p-button-primary{background-color:#007bff;color:#fff}.button-group button.p-button-secondary{background-color:#6c757d;color:#fff}.button-group button.p-button-success{background-color:#28a745;color:#fff}.button-group button:hover{opacity:.9}.form-field-group{display:flex;gap:1rem}.form-field-group .flex-item{flex:1}\n"], dependencies: [{ kind: "directive", type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i2.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { kind: "directive", type: i1.ɵNgNoValidate, selector: "form:not([ngNoForm]):not([ngNativeValidate])" }, { kind: "directive", type: i1.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i1.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "directive", type: i3.ButtonDirective, selector: "[pButton]", inputs: ["iconPos", "loadingIcon", "label", "icon", "loading", "severity", "raised", "rounded", "text", "outlined", "size", "plain"] }, { kind: "component", type: i4.PTDynamicFormFieldComponent, selector: "pt-dynamic-form-field", inputs: ["field", "form", "inputWidth"] }] }); }
|
|
137
142
|
}
|
|
138
143
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.11", ngImport: i0, type: PTFormBuilderComponent, decorators: [{
|
|
139
144
|
type: Component,
|
|
140
|
-
args: [{ selector: 'pt-form-builder', template: "<div *ngIf=\"title\" [ngStyle]=\"titleStyle\" class=\"form-title\">{{ title }}</div>\n<form [formGroup]=\"form\" (ngSubmit)=\"onSubmit()\">\n
|
|
145
|
+
args: [{ selector: 'pt-form-builder', template: "<div\n class=\"pt-form-builder-wrapper\"\n [ngStyle]=\"{ width: formWidth, maxWidth: formWidth }\"\n>\n <div *ngIf=\"title\" [ngStyle]=\"titleStyle\" class=\"form-title\">{{ title }}</div>\n\n <form [formGroup]=\"form\" (ngSubmit)=\"onSubmit()\">\n <ng-container *ngFor=\"let field of mainGroup.fields\">\n <pt-dynamic-form-field\n [field]=\"field\"\n [form]=\"form\"\n [inputWidth]=\"inputWidth\"\n ></pt-dynamic-form-field>\n </ng-container>\n\n <ng-container *ngFor=\"let group of mainGroup.groups\">\n <div\n class=\"form-field-group\"\n [ngStyle]=\"{ width: group.width || '100%' }\"\n >\n <ng-container *ngFor=\"let field of group.fields\">\n <pt-dynamic-form-field\n [field]=\"field\"\n [form]=\"form\"\n [inputWidth]=\"inputWidth\"\n class=\"flex-item\"\n ></pt-dynamic-form-field>\n </ng-container>\n </div>\n </ng-container>\n\n <div class=\"button-group\">\n <button\n *ngFor=\"let button of buttons\"\n type=\"button\"\n pButton\n [label]=\"button.text\"\n [icon]=\"button.icon || ''\"\n [class]=\"button.color\"\n (click)=\"\n button.isSubmit\n ? onSubmit()\n : button.isClear\n ? onClear()\n : button.action\n ? button.action()\n : null\n \"\n ></button>\n </div>\n </form>\n</div>\n", styles: [".pt-form-builder-wrapper{width:100%;margin:0 auto}.form-title{text-align:center;margin-bottom:1rem;font-size:1.5rem;font-weight:700}.form-field{margin-bottom:1rem}.form-field label{display:block;margin-bottom:.5rem;font-weight:700}::ng-deep .p-inputtext,::ng-deep .p-inputtextarea,::ng-deep .p-calendar,::ng-deep .p-inputnumber,::ng-deep .p-multiselect,::ng-deep .p-dropdown{width:100%!important}::ng-deep .p-inputnumber,::ng-deep p-inputnumber{display:flex!important}.button-group{display:flex;gap:1rem;margin-top:1rem;justify-content:space-between}.button-group button{flex:1;display:flex;align-items:center;justify-content:center}.button-group button i{margin-right:.5rem}.button-group button.p-button-primary{background-color:#007bff;color:#fff}.button-group button.p-button-secondary{background-color:#6c757d;color:#fff}.button-group button.p-button-success{background-color:#28a745;color:#fff}.button-group button:hover{opacity:.9}.form-field-group{display:flex;gap:1rem}.form-field-group .flex-item{flex:1}\n"] }]
|
|
141
146
|
}], ctorParameters: () => [{ type: i1.FormBuilder }], propDecorators: { mainGroup: [{
|
|
142
147
|
type: Input
|
|
143
148
|
}], buttons: [{
|
|
@@ -148,6 +153,8 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.11", ngImpo
|
|
|
148
153
|
type: Input
|
|
149
154
|
}], inputWidth: [{
|
|
150
155
|
type: Input
|
|
156
|
+
}], formWidth: [{
|
|
157
|
+
type: Input
|
|
151
158
|
}], language: [{
|
|
152
159
|
type: Input
|
|
153
160
|
}], formSubmit: [{
|
|
@@ -157,4 +164,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.11", ngImpo
|
|
|
157
164
|
}], formChange: [{
|
|
158
165
|
type: Output
|
|
159
166
|
}] } });
|
|
160
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"pt-form-builder.component.js","sourceRoot":"","sources":["../../../../../projects/ng-prime-tools/src/lib/pt-form-builder/pt-form-builder.component.ts","../../../../../projects/ng-prime-tools/src/lib/pt-form-builder/pt-form-builder.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  EventEmitter,\n  Input,\n  OnChanges,\n  OnDestroy,\n  OnInit,\n  Output,\n  SimpleChanges,\n} from '@angular/core';\nimport {\n  AbstractControl,\n  FormBuilder,\n  FormGroup,\n  ValidatorFn,\n  Validators,\n} from '@angular/forms';\nimport { Subscription } from 'rxjs';\n\nimport { FormButton } from '../models/form-button.model';\nimport { FormField } from '../models/form-field.model';\nimport { FormFieldGroup } from '../models/form-field-group.model';\nimport { FormInputTypeEnum } from '../enums/form-input-type.enum';\nimport { InputValidationEnum } from '../enums/input-validation.enum';\n\n@Component({\n  selector: 'pt-form-builder',\n  templateUrl: './pt-form-builder.component.html',\n  styleUrls: ['./pt-form-builder.component.css'],\n})\nexport class PTFormBuilderComponent implements OnInit, OnChanges, OnDestroy {\n  @Input() mainGroup: FormFieldGroup = { fields: [], groups: [] };\n  @Input() buttons: FormButton[] = [];\n  @Input() title?: string;\n  @Input() titleStyle?: { [key: string]: string };\n  @Input() inputWidth: string = '100%';\n  @Input() language: 'en' | 'fr' = 'en';\n\n  @Output() formSubmit = new EventEmitter<{ [key: string]: any }>();\n  @Output() formReady = new EventEmitter<FormGroup>();\n\n  // ✅ NEW\n  @Output() formChange = new EventEmitter<{ [key: string]: any }>();\n\n  form: FormGroup;\n\n  FormInputTypeEnum = FormInputTypeEnum;\n\n  private formValueChangesSub?: Subscription;\n\n  private errorMessages = {\n    en: {\n      required: 'is required',\n      email: 'is not a valid email address',\n      pattern: 'is invalid',\n      minlength: 'must be at least',\n      maxlength: 'must be at most',\n    },\n    fr: {\n      required: 'est requis',\n      email: \"n'est pas une adresse e-mail valide\",\n      pattern: 'est invalide',\n      minlength: 'doit contenir au moins',\n      maxlength: 'doit contenir au plus',\n    },\n  };\n\n  constructor(private fb: FormBuilder) {\n    this.form = this.fb.group({});\n  }\n\n  ngOnInit(): void {\n    this.rebuildForm();\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['mainGroup'] && !changes['mainGroup'].firstChange) {\n      this.rebuildForm();\n    }\n  }\n\n  ngOnDestroy(): void {\n    this.formValueChangesSub?.unsubscribe();\n  }\n\n  private rebuildForm(): void {\n    this.formValueChangesSub?.unsubscribe();\n\n    this.form = this.fb.group({});\n    this.buildFormGroup(this.mainGroup);\n\n    this.formReady.emit(this.form);\n\n    // ✅ NEW: emit on every form change\n    this.formValueChangesSub = this.form.valueChanges.subscribe((value) => {\n      this.formChange.emit(value);\n    });\n  }\n\n  private buildFormGroup(group: FormFieldGroup): void {\n    (group?.fields ?? []).forEach((field) => {\n      const validators = this.buildValidators(field);\n\n      this.form.addControl(\n        field.name,\n        this.fb.control(\n          { value: field.value ?? null, disabled: !!(field as any).disabled },\n          validators,\n        ),\n      );\n    });\n\n    (group?.groups ?? []).forEach((subGroup) => this.buildFormGroup(subGroup));\n  }\n\n  private buildValidators(field: FormField): ValidatorFn[] {\n    const validators: ValidatorFn[] = [];\n    const anyField = field as any;\n\n    if (field.required) {\n      validators.push(Validators.required);\n    }\n\n    if (anyField.minLength != null) {\n      validators.push(Validators.minLength(Number(anyField.minLength)));\n    }\n\n    if (anyField.maxLength != null) {\n      validators.push(Validators.maxLength(Number(anyField.maxLength)));\n    }\n\n    if (anyField.inputValidation) {\n      validators.push(\n        Validators.pattern(\n          new RegExp(anyField.inputValidation as InputValidationEnum),\n        ),\n      );\n    } else if (anyField.pattern) {\n      validators.push(Validators.pattern(anyField.pattern));\n    }\n\n    switch (field.type) {\n      case FormInputTypeEnum.EMAIL:\n        validators.push(Validators.email);\n        break;\n      default:\n        break;\n    }\n\n    return validators;\n  }\n\n  isInvalid(field: FormField): boolean {\n    const c = this.form.get(field.name);\n    return !!c && c.invalid && (c.touched || c.dirty);\n  }\n\n  getErrorMessage(field: FormField): string {\n    const control: AbstractControl | null = this.form.get(field.name);\n    if (!control) return '';\n\n    const lang = this.language;\n\n    if (control.hasError('required')) {\n      return (\n        (field as any).errorText ||\n        `${field.label} ${this.errorMessages[lang].required}`\n      );\n    }\n\n    if (control.hasError('email')) {\n      return (\n        (field as any).errorText ||\n        `${field.label} ${this.errorMessages[lang].email}`\n      );\n    }\n\n    if (control.hasError('pattern')) {\n      return (\n        (field as any).errorText ||\n        `${field.label} ${this.errorMessages[lang].pattern}`\n      );\n    }\n\n    if (control.hasError('minlength')) {\n      const requiredLen = control.getError('minlength')?.requiredLength;\n      return `${field.label} ${this.errorMessages[lang].minlength} ${requiredLen} ${\n        lang === 'fr' ? 'caractères' : 'characters'\n      }`;\n    }\n\n    if (control.hasError('maxlength')) {\n      const requiredLen = control.getError('maxlength')?.requiredLength;\n      return `${field.label} ${this.errorMessages[lang].maxlength} ${requiredLen} ${\n        lang === 'fr' ? 'caractères' : 'characters'\n      }`;\n    }\n\n    return '';\n  }\n\n  onSubmit(): void {\n    this.form.markAllAsTouched();\n    if (this.form.valid) {\n      this.formSubmit.emit(this.form.getRawValue());\n    }\n  }\n\n  onClear(): void {\n    this.form.reset();\n    this.formChange.emit(this.form.getRawValue());\n  }\n}\n","<div *ngIf=\"title\" [ngStyle]=\"titleStyle\" class=\"form-title\">{{ title }}</div>\n<form [formGroup]=\"form\" (ngSubmit)=\"onSubmit()\">\n  <ng-container *ngFor=\"let field of mainGroup.fields\">\n    <pt-dynamic-form-field\n      [field]=\"field\"\n      [form]=\"form\"\n      [inputWidth]=\"inputWidth\"\n    ></pt-dynamic-form-field>\n  </ng-container>\n\n  <ng-container *ngFor=\"let group of mainGroup.groups\">\n    <div class=\"form-field-group\" [ngStyle]=\"{ width: group.width || '100%' }\">\n      <ng-container *ngFor=\"let field of group.fields\">\n        <pt-dynamic-form-field\n          [field]=\"field\"\n          [form]=\"form\"\n          class=\"flex-item\"\n        ></pt-dynamic-form-field>\n      </ng-container>\n    </div>\n  </ng-container>\n\n  <div class=\"button-group\">\n    <button\n      *ngFor=\"let button of buttons\"\n      type=\"button\"\n      pButton\n      [label]=\"button.text\"\n      [icon]=\"button.icon || ''\"\n      [class]=\"button.color\"\n      (click)=\"\n        button.isSubmit\n          ? onSubmit()\n          : button.isClear\n          ? onClear()\n          : button.action\n          ? button.action()\n          : null\n      \"\n    ></button>\n  </div>\n</form>\n"]}
|
|
167
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"pt-form-builder.component.js","sourceRoot":"","sources":["../../../../../projects/ng-prime-tools/src/lib/pt-form-builder/pt-form-builder.component.ts","../../../../../projects/ng-prime-tools/src/lib/pt-form-builder/pt-form-builder.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  EventEmitter,\n  Input,\n  OnChanges,\n  OnDestroy,\n  OnInit,\n  Output,\n  SimpleChanges,\n} from '@angular/core';\nimport {\n  AbstractControl,\n  FormBuilder,\n  FormGroup,\n  ValidatorFn,\n  Validators,\n} from '@angular/forms';\nimport { Subscription } from 'rxjs';\n\nimport { FormButton } from '../models/form-button.model';\nimport { FormField } from '../models/form-field.model';\nimport { FormFieldGroup } from '../models/form-field-group.model';\nimport { FormInputTypeEnum } from '../enums/form-input-type.enum';\nimport { InputValidationEnum } from '../enums/input-validation.enum';\n\n@Component({\n  selector: 'pt-form-builder',\n  templateUrl: './pt-form-builder.component.html',\n  styleUrls: ['./pt-form-builder.component.css'],\n})\nexport class PTFormBuilderComponent implements OnInit, OnChanges, OnDestroy {\n  @Input() mainGroup: FormFieldGroup = { fields: [], groups: [] };\n  @Input() buttons: FormButton[] = [];\n  @Input() title?: string;\n  @Input() titleStyle?: { [key: string]: string };\n\n  /**\n   * Largeur des inputs/champs\n   */\n  @Input() inputWidth: string = '100%';\n\n  /**\n   * Nouvelle largeur globale du composant pt-form-builder\n   */\n  @Input() formWidth: string = '100%';\n\n  @Input() language: 'en' | 'fr' = 'en';\n\n  @Output() formSubmit = new EventEmitter<{ [key: string]: any }>();\n  @Output() formReady = new EventEmitter<FormGroup>();\n  @Output() formChange = new EventEmitter<{ [key: string]: any }>();\n\n  form: FormGroup;\n\n  FormInputTypeEnum = FormInputTypeEnum;\n\n  private formValueChangesSub?: Subscription;\n\n  private errorMessages = {\n    en: {\n      required: 'is required',\n      email: 'is not a valid email address',\n      pattern: 'is invalid',\n      minlength: 'must be at least',\n      maxlength: 'must be at most',\n    },\n    fr: {\n      required: 'est requis',\n      email: \"n'est pas une adresse e-mail valide\",\n      pattern: 'est invalide',\n      minlength: 'doit contenir au moins',\n      maxlength: 'doit contenir au plus',\n    },\n  };\n\n  constructor(private fb: FormBuilder) {\n    this.form = this.fb.group({});\n  }\n\n  ngOnInit(): void {\n    this.rebuildForm();\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['mainGroup'] && !changes['mainGroup'].firstChange) {\n      this.rebuildForm();\n    }\n  }\n\n  ngOnDestroy(): void {\n    this.formValueChangesSub?.unsubscribe();\n  }\n\n  private rebuildForm(): void {\n    this.formValueChangesSub?.unsubscribe();\n\n    this.form = this.fb.group({});\n    this.buildFormGroup(this.mainGroup);\n\n    this.formReady.emit(this.form);\n\n    this.formValueChangesSub = this.form.valueChanges.subscribe((value) => {\n      this.formChange.emit(value);\n    });\n  }\n\n  private buildFormGroup(group: FormFieldGroup): void {\n    (group?.fields ?? []).forEach((field) => {\n      const validators = this.buildValidators(field);\n\n      this.form.addControl(\n        field.name,\n        this.fb.control(\n          { value: field.value ?? null, disabled: !!(field as any).disabled },\n          validators,\n        ),\n      );\n    });\n\n    (group?.groups ?? []).forEach((subGroup) => this.buildFormGroup(subGroup));\n  }\n\n  private buildValidators(field: FormField): ValidatorFn[] {\n    const validators: ValidatorFn[] = [];\n    const anyField = field as any;\n\n    if (field.required) {\n      validators.push(Validators.required);\n    }\n\n    if (anyField.minLength != null) {\n      validators.push(Validators.minLength(Number(anyField.minLength)));\n    }\n\n    if (anyField.maxLength != null) {\n      validators.push(Validators.maxLength(Number(anyField.maxLength)));\n    }\n\n    if (anyField.inputValidation) {\n      validators.push(\n        Validators.pattern(\n          new RegExp(anyField.inputValidation as InputValidationEnum),\n        ),\n      );\n    } else if (anyField.pattern) {\n      validators.push(Validators.pattern(anyField.pattern));\n    }\n\n    switch (field.type) {\n      case FormInputTypeEnum.EMAIL:\n        validators.push(Validators.email);\n        break;\n      default:\n        break;\n    }\n\n    return validators;\n  }\n\n  isInvalid(field: FormField): boolean {\n    const c = this.form.get(field.name);\n    return !!c && c.invalid && (c.touched || c.dirty);\n  }\n\n  getErrorMessage(field: FormField): string {\n    const control: AbstractControl | null = this.form.get(field.name);\n    if (!control) return '';\n\n    const lang = this.language;\n\n    if (control.hasError('required')) {\n      return (\n        (field as any).errorText ||\n        `${field.label} ${this.errorMessages[lang].required}`\n      );\n    }\n\n    if (control.hasError('email')) {\n      return (\n        (field as any).errorText ||\n        `${field.label} ${this.errorMessages[lang].email}`\n      );\n    }\n\n    if (control.hasError('pattern')) {\n      return (\n        (field as any).errorText ||\n        `${field.label} ${this.errorMessages[lang].pattern}`\n      );\n    }\n\n    if (control.hasError('minlength')) {\n      const requiredLen = control.getError('minlength')?.requiredLength;\n      return `${field.label} ${this.errorMessages[lang].minlength} ${requiredLen} ${\n        lang === 'fr' ? 'caractères' : 'characters'\n      }`;\n    }\n\n    if (control.hasError('maxlength')) {\n      const requiredLen = control.getError('maxlength')?.requiredLength;\n      return `${field.label} ${this.errorMessages[lang].maxlength} ${requiredLen} ${\n        lang === 'fr' ? 'caractères' : 'characters'\n      }`;\n    }\n\n    return '';\n  }\n\n  onSubmit(): void {\n    this.form.markAllAsTouched();\n    if (this.form.valid) {\n      this.formSubmit.emit(this.form.getRawValue());\n    }\n  }\n\n  onClear(): void {\n    this.form.reset();\n    this.formChange.emit(this.form.getRawValue());\n  }\n}\n","<div\n  class=\"pt-form-builder-wrapper\"\n  [ngStyle]=\"{ width: formWidth, maxWidth: formWidth }\"\n>\n  <div *ngIf=\"title\" [ngStyle]=\"titleStyle\" class=\"form-title\">{{ title }}</div>\n\n  <form [formGroup]=\"form\" (ngSubmit)=\"onSubmit()\">\n    <ng-container *ngFor=\"let field of mainGroup.fields\">\n      <pt-dynamic-form-field\n        [field]=\"field\"\n        [form]=\"form\"\n        [inputWidth]=\"inputWidth\"\n      ></pt-dynamic-form-field>\n    </ng-container>\n\n    <ng-container *ngFor=\"let group of mainGroup.groups\">\n      <div\n        class=\"form-field-group\"\n        [ngStyle]=\"{ width: group.width || '100%' }\"\n      >\n        <ng-container *ngFor=\"let field of group.fields\">\n          <pt-dynamic-form-field\n            [field]=\"field\"\n            [form]=\"form\"\n            [inputWidth]=\"inputWidth\"\n            class=\"flex-item\"\n          ></pt-dynamic-form-field>\n        </ng-container>\n      </div>\n    </ng-container>\n\n    <div class=\"button-group\">\n      <button\n        *ngFor=\"let button of buttons\"\n        type=\"button\"\n        pButton\n        [label]=\"button.text\"\n        [icon]=\"button.icon || ''\"\n        [class]=\"button.color\"\n        (click)=\"\n          button.isSubmit\n            ? onSubmit()\n            : button.isClear\n              ? onClear()\n              : button.action\n                ? button.action()\n                : null\n        \"\n      ></button>\n    </div>\n  </form>\n</div>\n"]}
|
|
@@ -14,19 +14,20 @@ export class PTMetricCardComponent {
|
|
|
14
14
|
static { this.DEFAULT_ADDITIONAL_INFO_SIZE = '14px'; }
|
|
15
15
|
static { this.DEFAULT_BACKGROUND_COLOR = '#fff'; }
|
|
16
16
|
isIconObject() {
|
|
17
|
-
return typeof this.cardData.icon === 'object';
|
|
17
|
+
return (typeof this.cardData.icon === 'object' && this.cardData.icon !== null);
|
|
18
18
|
}
|
|
19
19
|
isTitleObject() {
|
|
20
|
-
return typeof this.cardData.title === 'object';
|
|
20
|
+
return (typeof this.cardData.title === 'object' && this.cardData.title !== null);
|
|
21
21
|
}
|
|
22
22
|
isValueObject() {
|
|
23
|
-
return typeof this.cardData.value === 'object';
|
|
23
|
+
return (typeof this.cardData.value === 'object' && this.cardData.value !== null);
|
|
24
24
|
}
|
|
25
25
|
isLabelObject(label) {
|
|
26
|
-
return typeof label === 'object';
|
|
26
|
+
return typeof label === 'object' && label !== null;
|
|
27
27
|
}
|
|
28
28
|
isAdditionalInfoObject() {
|
|
29
|
-
return typeof this.cardData.additionalInfo === 'object'
|
|
29
|
+
return (typeof this.cardData.additionalInfo === 'object' &&
|
|
30
|
+
this.cardData.additionalInfo !== null);
|
|
30
31
|
}
|
|
31
32
|
getIconText() {
|
|
32
33
|
return this.isIconObject()
|
|
@@ -36,35 +37,34 @@ export class PTMetricCardComponent {
|
|
|
36
37
|
getTitleText() {
|
|
37
38
|
return this.isTitleObject()
|
|
38
39
|
? this.cardData.title.text
|
|
39
|
-
: this.cardData.title;
|
|
40
|
+
: this.cardData.title || '';
|
|
40
41
|
}
|
|
41
42
|
getValueText() {
|
|
42
43
|
return this.isValueObject()
|
|
43
44
|
? this.cardData.value.text
|
|
44
|
-
: this.cardData.value;
|
|
45
|
+
: this.cardData.value || '';
|
|
45
46
|
}
|
|
46
47
|
getLabelText() {
|
|
47
48
|
const label = this.cardData.value?.label;
|
|
48
49
|
return this.isLabelObject(label)
|
|
49
50
|
? label.text
|
|
50
|
-
: label;
|
|
51
|
+
: label || '';
|
|
51
52
|
}
|
|
52
53
|
getAdditionalInfoText() {
|
|
53
54
|
return this.isAdditionalInfoObject()
|
|
54
55
|
? this.cardData.additionalInfo.text
|
|
55
|
-
: this.cardData.additionalInfo;
|
|
56
|
+
: this.cardData.additionalInfo || '';
|
|
56
57
|
}
|
|
57
58
|
getIconStyles() {
|
|
58
59
|
if (this.isIconObject()) {
|
|
59
60
|
const icon = this.cardData.icon;
|
|
60
|
-
const shape = icon.shape || 'circular';
|
|
61
|
-
// Determine border radius based on shape
|
|
61
|
+
const shape = icon.shape || 'circular';
|
|
62
62
|
const borderRadius = shape === 'circular' ? '50%' : '8px';
|
|
63
63
|
return {
|
|
64
64
|
color: icon.color || PTMetricCardComponent.DEFAULT_ICON_COLOR,
|
|
65
65
|
fontSize: icon.fontSize || PTMetricCardComponent.DEFAULT_ICON_SIZE,
|
|
66
|
-
|
|
67
|
-
|
|
66
|
+
backgroundColor: icon.backgroundColor || 'transparent',
|
|
67
|
+
borderRadius,
|
|
68
68
|
padding: '6px',
|
|
69
69
|
display: 'inline-block',
|
|
70
70
|
};
|
|
@@ -85,19 +85,43 @@ export class PTMetricCardComponent {
|
|
|
85
85
|
getValueStyles() {
|
|
86
86
|
if (this.isValueObject()) {
|
|
87
87
|
const value = this.cardData.value;
|
|
88
|
+
const position = value.position || 'left';
|
|
88
89
|
return {
|
|
89
90
|
color: value.color || PTMetricCardComponent.DEFAULT_VALUE_COLOR,
|
|
90
91
|
fontSize: value.fontSize || PTMetricCardComponent.DEFAULT_VALUE_SIZE,
|
|
91
92
|
fontWeight: value.fontWeight || 'bold',
|
|
93
|
+
textAlign: position,
|
|
94
|
+
width: '100%',
|
|
95
|
+
display: 'flex',
|
|
96
|
+
justifyContent: position === 'center'
|
|
97
|
+
? 'center'
|
|
98
|
+
: position === 'right'
|
|
99
|
+
? 'flex-end'
|
|
100
|
+
: 'flex-start',
|
|
101
|
+
alignItems: 'center',
|
|
92
102
|
};
|
|
93
103
|
}
|
|
94
|
-
|
|
104
|
+
return {
|
|
105
|
+
color: PTMetricCardComponent.DEFAULT_VALUE_COLOR,
|
|
106
|
+
fontSize: PTMetricCardComponent.DEFAULT_VALUE_SIZE,
|
|
107
|
+
fontWeight: 'bold',
|
|
108
|
+
textAlign: 'left',
|
|
109
|
+
width: '100%',
|
|
110
|
+
display: 'flex',
|
|
111
|
+
justifyContent: 'flex-start',
|
|
112
|
+
alignItems: 'center',
|
|
113
|
+
};
|
|
114
|
+
}
|
|
115
|
+
getValueTextStyles() {
|
|
116
|
+
if (this.isValueObject()) {
|
|
117
|
+
const value = this.cardData.value;
|
|
95
118
|
return {
|
|
96
|
-
|
|
97
|
-
fontSize: PTMetricCardComponent.DEFAULT_VALUE_SIZE,
|
|
98
|
-
fontWeight: 'bold',
|
|
119
|
+
textAlign: value.position || 'left',
|
|
99
120
|
};
|
|
100
121
|
}
|
|
122
|
+
return {
|
|
123
|
+
textAlign: 'left',
|
|
124
|
+
};
|
|
101
125
|
}
|
|
102
126
|
getLabelStyles(position) {
|
|
103
127
|
const label = this.cardData.value?.label;
|
|
@@ -111,13 +135,11 @@ export class PTMetricCardComponent {
|
|
|
111
135
|
fontWeight: label.fontWeight || 'normal',
|
|
112
136
|
};
|
|
113
137
|
}
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
|
|
119
|
-
};
|
|
120
|
-
}
|
|
138
|
+
return {
|
|
139
|
+
color: PTMetricCardComponent.DEFAULT_LABEL_COLOR,
|
|
140
|
+
fontSize: PTMetricCardComponent.DEFAULT_LABEL_SIZE,
|
|
141
|
+
fontWeight: 'normal',
|
|
142
|
+
};
|
|
121
143
|
}
|
|
122
144
|
getAdditionalInfoStyles() {
|
|
123
145
|
if (this.isAdditionalInfoObject()) {
|
|
@@ -130,13 +152,11 @@ export class PTMetricCardComponent {
|
|
|
130
152
|
fontWeight: additionalInfo.fontWeight || 'normal',
|
|
131
153
|
};
|
|
132
154
|
}
|
|
133
|
-
|
|
134
|
-
|
|
135
|
-
|
|
136
|
-
|
|
137
|
-
|
|
138
|
-
};
|
|
139
|
-
}
|
|
155
|
+
return {
|
|
156
|
+
color: PTMetricCardComponent.DEFAULT_ADDITIONAL_INFO_COLOR,
|
|
157
|
+
fontSize: PTMetricCardComponent.DEFAULT_ADDITIONAL_INFO_SIZE,
|
|
158
|
+
fontWeight: 'normal',
|
|
159
|
+
};
|
|
140
160
|
}
|
|
141
161
|
isLabelOnLeft() {
|
|
142
162
|
const label = this.cardData.value?.label;
|
|
@@ -148,27 +168,27 @@ export class PTMetricCardComponent {
|
|
|
148
168
|
}
|
|
149
169
|
getMetricCardContainerStyle() {
|
|
150
170
|
return {
|
|
151
|
-
|
|
152
|
-
|
|
171
|
+
minWidth: this.cardData.size?.minWidth || '100%',
|
|
172
|
+
maxWidth: this.cardData.size?.maxWidth || 'calc(25% - 16px)',
|
|
153
173
|
width: this.cardData.size?.fixedWidth || 'auto',
|
|
154
174
|
};
|
|
155
175
|
}
|
|
156
176
|
getMetricCardStyle() {
|
|
157
177
|
return {
|
|
158
|
-
|
|
178
|
+
backgroundColor: this.cardData.appearance?.backgroundColor ||
|
|
159
179
|
PTMetricCardComponent.DEFAULT_BACKGROUND_COLOR,
|
|
160
|
-
|
|
161
|
-
|
|
180
|
+
minWidth: this.cardData.size?.minWidth || '100%',
|
|
181
|
+
maxWidth: this.cardData.size?.maxWidth || 'calc(25% - 16px)',
|
|
162
182
|
width: this.cardData.size?.fixedWidth || 'auto',
|
|
163
183
|
};
|
|
164
184
|
}
|
|
165
185
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.11", ngImport: i0, type: PTMetricCardComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
166
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.11", type: PTMetricCardComponent, selector: "pt-metric-card", inputs: { cardData: "cardData" }, ngImport: i0, template: "<div class=\"metric-card-container\" [ngStyle]=\"getMetricCardContainerStyle()\">\n <div\n *ngIf=\"cardData.verticalLine?.show\"\n class=\"vertical-line\"\n [ngStyle]=\"{\n 'background-color': cardData.verticalLine?.color || '#5a67d8'
|
|
186
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.11", type: PTMetricCardComponent, selector: "pt-metric-card", inputs: { cardData: "cardData" }, ngImport: i0, template: "<div class=\"metric-card-container\" [ngStyle]=\"getMetricCardContainerStyle()\">\n <div\n *ngIf=\"cardData.verticalLine?.show\"\n class=\"vertical-line\"\n [ngStyle]=\"{\n 'background-color': cardData.verticalLine?.color || '#5a67d8',\n }\"\n ></div>\n\n <div class=\"metric-card\" [ngStyle]=\"getMetricCardStyle()\">\n <div class=\"metric-card-header\">\n <i\n *ngIf=\"isIconObject()\"\n [ngClass]=\"getIconText()\"\n [ngStyle]=\"getIconStyles()\"\n ></i>\n <i *ngIf=\"!isIconObject()\" [ngClass]=\"cardData.icon\"></i>\n\n <span\n *ngIf=\"isTitleObject()\"\n class=\"metric-card-header-text\"\n [ngStyle]=\"getTitleStyles()\"\n >\n {{ getTitleText() }}\n </span>\n <span *ngIf=\"!isTitleObject()\" class=\"metric-card-header-text\">\n {{ cardData.title }}\n </span>\n </div>\n\n <div class=\"metric-card-content\">\n <div class=\"metric-value\" [ngStyle]=\"getValueStyles()\">\n <span\n *ngIf=\"isLabelOnLeft()\"\n class=\"value-label\"\n [ngStyle]=\"getLabelStyles('left')\"\n >\n {{ getLabelText() }}\n </span>\n\n <span class=\"metric-value-text\" [ngStyle]=\"getValueTextStyles()\">\n {{ getValueText() }}\n </span>\n\n <span\n *ngIf=\"isLabelOnRight()\"\n class=\"value-label\"\n [ngStyle]=\"getLabelStyles('right')\"\n >\n {{ getLabelText() }}\n </span>\n </div>\n\n <div\n *ngIf=\"cardData.divider?.show\"\n class=\"metric-divider\"\n [ngStyle]=\"{ backgroundColor: cardData.divider?.color || '#e2e8f0' }\"\n ></div>\n\n <div\n *ngIf=\"isAdditionalInfoObject()\"\n class=\"metric-additional-info\"\n [ngStyle]=\"getAdditionalInfoStyles()\"\n >\n {{ getAdditionalInfoText() }}\n </div>\n <div\n *ngIf=\"!isAdditionalInfoObject()\"\n class=\"metric-additional-info\"\n [ngStyle]=\"getAdditionalInfoStyles()\"\n >\n {{ cardData.additionalInfo }}\n </div>\n </div>\n </div>\n</div>\n", styles: [".metric-card-container{display:flex;align-items:stretch;box-sizing:border-box}.vertical-line{width:4px;border-radius:8px 0 0 8px;margin-right:-4px;z-index:2}.metric-card{background-color:#fff;border-radius:5px;box-shadow:0 1px 3px #0000001a;padding:16px;display:flex;flex-direction:column;box-sizing:border-box;margin-left:0;width:100%}.metric-card-header{display:flex;align-items:center;margin-bottom:16px}.metric-card-header i.icon-text{margin-right:8px}.metric-card-header-text{margin-left:6px}.metric-card-content{display:flex;flex-direction:column;justify-content:space-between;width:100%;flex:1}.metric-value{width:100%;display:flex;align-items:center;gap:6px;box-sizing:border-box}.metric-value-text{display:inline-block;width:100%;line-height:1.2}.metric-number{font-weight:700}.metric-label,.value-label{margin-left:8px;white-space:nowrap}.metric-divider{width:100%;height:1px;margin:16px 0}.metric-additional-info{margin-top:8px}\n"], dependencies: [{ kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }] }); }
|
|
167
187
|
}
|
|
168
188
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.11", ngImport: i0, type: PTMetricCardComponent, decorators: [{
|
|
169
189
|
type: Component,
|
|
170
|
-
args: [{ selector: 'pt-metric-card', template: "<div class=\"metric-card-container\" [ngStyle]=\"getMetricCardContainerStyle()\">\n <div\n *ngIf=\"cardData.verticalLine?.show\"\n class=\"vertical-line\"\n [ngStyle]=\"{\n 'background-color': cardData.verticalLine?.color || '#5a67d8'
|
|
190
|
+
args: [{ selector: 'pt-metric-card', template: "<div class=\"metric-card-container\" [ngStyle]=\"getMetricCardContainerStyle()\">\n <div\n *ngIf=\"cardData.verticalLine?.show\"\n class=\"vertical-line\"\n [ngStyle]=\"{\n 'background-color': cardData.verticalLine?.color || '#5a67d8',\n }\"\n ></div>\n\n <div class=\"metric-card\" [ngStyle]=\"getMetricCardStyle()\">\n <div class=\"metric-card-header\">\n <i\n *ngIf=\"isIconObject()\"\n [ngClass]=\"getIconText()\"\n [ngStyle]=\"getIconStyles()\"\n ></i>\n <i *ngIf=\"!isIconObject()\" [ngClass]=\"cardData.icon\"></i>\n\n <span\n *ngIf=\"isTitleObject()\"\n class=\"metric-card-header-text\"\n [ngStyle]=\"getTitleStyles()\"\n >\n {{ getTitleText() }}\n </span>\n <span *ngIf=\"!isTitleObject()\" class=\"metric-card-header-text\">\n {{ cardData.title }}\n </span>\n </div>\n\n <div class=\"metric-card-content\">\n <div class=\"metric-value\" [ngStyle]=\"getValueStyles()\">\n <span\n *ngIf=\"isLabelOnLeft()\"\n class=\"value-label\"\n [ngStyle]=\"getLabelStyles('left')\"\n >\n {{ getLabelText() }}\n </span>\n\n <span class=\"metric-value-text\" [ngStyle]=\"getValueTextStyles()\">\n {{ getValueText() }}\n </span>\n\n <span\n *ngIf=\"isLabelOnRight()\"\n class=\"value-label\"\n [ngStyle]=\"getLabelStyles('right')\"\n >\n {{ getLabelText() }}\n </span>\n </div>\n\n <div\n *ngIf=\"cardData.divider?.show\"\n class=\"metric-divider\"\n [ngStyle]=\"{ backgroundColor: cardData.divider?.color || '#e2e8f0' }\"\n ></div>\n\n <div\n *ngIf=\"isAdditionalInfoObject()\"\n class=\"metric-additional-info\"\n [ngStyle]=\"getAdditionalInfoStyles()\"\n >\n {{ getAdditionalInfoText() }}\n </div>\n <div\n *ngIf=\"!isAdditionalInfoObject()\"\n class=\"metric-additional-info\"\n [ngStyle]=\"getAdditionalInfoStyles()\"\n >\n {{ cardData.additionalInfo }}\n </div>\n </div>\n </div>\n</div>\n", styles: [".metric-card-container{display:flex;align-items:stretch;box-sizing:border-box}.vertical-line{width:4px;border-radius:8px 0 0 8px;margin-right:-4px;z-index:2}.metric-card{background-color:#fff;border-radius:5px;box-shadow:0 1px 3px #0000001a;padding:16px;display:flex;flex-direction:column;box-sizing:border-box;margin-left:0;width:100%}.metric-card-header{display:flex;align-items:center;margin-bottom:16px}.metric-card-header i.icon-text{margin-right:8px}.metric-card-header-text{margin-left:6px}.metric-card-content{display:flex;flex-direction:column;justify-content:space-between;width:100%;flex:1}.metric-value{width:100%;display:flex;align-items:center;gap:6px;box-sizing:border-box}.metric-value-text{display:inline-block;width:100%;line-height:1.2}.metric-number{font-weight:700}.metric-label,.value-label{margin-left:8px;white-space:nowrap}.metric-divider{width:100%;height:1px;margin:16px 0}.metric-additional-info{margin-top:8px}\n"] }]
|
|
171
191
|
}], propDecorators: { cardData: [{
|
|
172
192
|
type: Input
|
|
173
193
|
}] } });
|
|
174
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"pt-metric-card.component.js","sourceRoot":"","sources":["../../../../../projects/ng-prime-tools/src/lib/pt-metric-card/pt-metric-card.component.ts","../../../../../projects/ng-prime-tools/src/lib/pt-metric-card/pt-metric-card.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input } from '@angular/core';\nimport {\n  LabelSetting,\n  MetricCardData,\n  ValueSetting,\n  IconStyle,\n  TextStyle,\n} from '../models';\n\n@Component({\n  selector: 'pt-metric-card',\n  templateUrl: './pt-metric-card.component.html',\n  styleUrls: ['./pt-metric-card.component.css'],\n})\nexport class PTMetricCardComponent {\n  private static readonly DEFAULT_ICON_COLOR = '#5a67d8';\n  private static readonly DEFAULT_ICON_SIZE = '24px';\n  private static readonly DEFAULT_TITLE_COLOR = '#5a67d8';\n  private static readonly DEFAULT_TITLE_SIZE = '16px';\n  private static readonly DEFAULT_VALUE_COLOR = '#2d3748';\n  private static readonly DEFAULT_VALUE_SIZE = '32px';\n  private static readonly DEFAULT_LABEL_COLOR = '#718096';\n  private static readonly DEFAULT_LABEL_SIZE = '14px';\n  private static readonly DEFAULT_ADDITIONAL_INFO_COLOR = 'gray';\n  private static readonly DEFAULT_ADDITIONAL_INFO_SIZE = '14px';\n  private static readonly DEFAULT_BACKGROUND_COLOR = '#fff';\n\n  @Input() cardData!: MetricCardData;\n\n  isIconObject(): boolean {\n    return typeof this.cardData.icon === 'object';\n  }\n\n  isTitleObject(): boolean {\n    return typeof this.cardData.title === 'object';\n  }\n\n  isValueObject(): boolean {\n    return typeof this.cardData.value === 'object';\n  }\n\n  isLabelObject(label: LabelSetting | string | undefined): boolean {\n    return typeof label === 'object';\n  }\n\n  isAdditionalInfoObject(): boolean {\n    return typeof this.cardData.additionalInfo === 'object';\n  }\n\n  getIconText(): string {\n    return this.isIconObject()\n      ? (this.cardData.icon as IconStyle).code\n      : (this.cardData.icon as string);\n  }\n\n  getTitleText(): string {\n    return this.isTitleObject()\n      ? (this.cardData.title as TextStyle).text\n      : (this.cardData.title as string);\n  }\n\n  getValueText(): string {\n    return this.isValueObject()\n      ? (this.cardData.value as ValueSetting).text\n      : (this.cardData.value as string);\n  }\n\n  getLabelText(): string {\n    const label = (this.cardData.value as ValueSetting)?.label;\n    return this.isLabelObject(label)\n      ? (label as LabelSetting).text\n      : (label as string);\n  }\n\n  getAdditionalInfoText(): string {\n    return this.isAdditionalInfoObject()\n      ? (this.cardData.additionalInfo as TextStyle).text\n      : (this.cardData.additionalInfo as string);\n  }\n\n  getIconStyles() {\n    if (this.isIconObject()) {\n      const icon = this.cardData.icon as IconStyle;\n      const shape = icon.shape || 'circular'; // Default to 'circular' if not specified\n\n      // Determine border radius based on shape\n      const borderRadius = shape === 'circular' ? '50%' : '8px';\n\n      return {\n        color: icon.color || PTMetricCardComponent.DEFAULT_ICON_COLOR,\n        fontSize: icon.fontSize || PTMetricCardComponent.DEFAULT_ICON_SIZE,\n        'background-color': icon.backgroundColor || 'transparent',\n        'border-radius': borderRadius,\n        padding: '6px',\n        display: 'inline-block',\n      };\n    }\n    return {};\n  }\n\n  getTitleStyles() {\n    if (this.isTitleObject()) {\n      const title = this.cardData.title as TextStyle;\n      return {\n        color: title.color || PTMetricCardComponent.DEFAULT_TITLE_COLOR,\n        fontSize: title.fontSize || PTMetricCardComponent.DEFAULT_TITLE_SIZE,\n        fontWeight: title.fontWeight || 'normal',\n      };\n    }\n    return {};\n  }\n\n  getValueStyles() {\n    if (this.isValueObject()) {\n      const value = this.cardData.value as ValueSetting;\n      return {\n        color: value.color || PTMetricCardComponent.DEFAULT_VALUE_COLOR,\n        fontSize: value.fontSize || PTMetricCardComponent.DEFAULT_VALUE_SIZE,\n        fontWeight: value.fontWeight || 'bold',\n      };\n    } else {\n      return {\n        color: PTMetricCardComponent.DEFAULT_VALUE_COLOR,\n        fontSize: PTMetricCardComponent.DEFAULT_VALUE_SIZE,\n        fontWeight: 'bold',\n      };\n    }\n  }\n\n  getLabelStyles(position: 'left' | 'right') {\n    const label = (this.cardData.value as ValueSetting)?.label;\n    if (\n      this.isLabelObject(label) &&\n      (label as LabelSetting).position === position\n    ) {\n      return {\n        color:\n          (label as LabelSetting).color ||\n          PTMetricCardComponent.DEFAULT_LABEL_COLOR,\n        fontSize:\n          (label as LabelSetting).fontSize ||\n          PTMetricCardComponent.DEFAULT_LABEL_SIZE,\n        fontWeight: (label as LabelSetting).fontWeight || 'normal',\n      };\n    } else {\n      return {\n        color: PTMetricCardComponent.DEFAULT_LABEL_COLOR,\n        fontSize: PTMetricCardComponent.DEFAULT_LABEL_SIZE,\n        fontWeight: 'normal',\n      };\n    }\n  }\n\n  getAdditionalInfoStyles() {\n    if (this.isAdditionalInfoObject()) {\n      const additionalInfo = this.cardData.additionalInfo as TextStyle;\n      return {\n        color:\n          additionalInfo.color ||\n          PTMetricCardComponent.DEFAULT_ADDITIONAL_INFO_COLOR,\n        fontSize:\n          additionalInfo.fontSize ||\n          PTMetricCardComponent.DEFAULT_ADDITIONAL_INFO_SIZE,\n        fontWeight: additionalInfo.fontWeight || 'normal',\n      };\n    } else {\n      return {\n        color: PTMetricCardComponent.DEFAULT_ADDITIONAL_INFO_COLOR,\n        fontSize: PTMetricCardComponent.DEFAULT_ADDITIONAL_INFO_SIZE,\n        fontWeight: 'normal',\n      };\n    }\n  }\n\n  isLabelOnLeft(): boolean {\n    const label = (this.cardData.value as ValueSetting)?.label;\n    return (\n      this.isLabelObject(label) && (label as LabelSetting).position === 'left'\n    );\n  }\n\n  isLabelOnRight(): boolean {\n    const label = (this.cardData.value as ValueSetting)?.label;\n    return (\n      this.isLabelObject(label) && (label as LabelSetting).position === 'right'\n    );\n  }\n\n  getMetricCardContainerStyle() {\n    return {\n      'min-width': this.cardData.size?.minWidth || '100%',\n      'max-width': this.cardData.size?.maxWidth || 'calc(25% - 16px)',\n      width: this.cardData.size?.fixedWidth || 'auto',\n    };\n  }\n\n  getMetricCardStyle() {\n    return {\n      'background-color':\n        this.cardData.appearance?.backgroundColor ||\n        PTMetricCardComponent.DEFAULT_BACKGROUND_COLOR,\n      'min-width': this.cardData.size?.minWidth || '100%',\n      'max-width': this.cardData.size?.maxWidth || 'calc(25% - 16px)',\n      width: this.cardData.size?.fixedWidth || 'auto',\n    };\n  }\n}\n","<div class=\"metric-card-container\" [ngStyle]=\"getMetricCardContainerStyle()\">\n  <div\n    *ngIf=\"cardData.verticalLine?.show\"\n    class=\"vertical-line\"\n    [ngStyle]=\"{\n      'background-color': cardData.verticalLine?.color || '#5a67d8'\n    }\"\n  ></div>\n\n  <div class=\"metric-card\" [ngStyle]=\"getMetricCardStyle()\">\n    <div class=\"metric-card-header\">\n      <i\n        *ngIf=\"isIconObject()\"\n        [ngClass]=\"getIconText()\"\n        [ngStyle]=\"getIconStyles()\"\n      ></i>\n      <i *ngIf=\"!isIconObject()\" [ngClass]=\"cardData.icon\"></i>\n\n      <span\n        *ngIf=\"isTitleObject()\"\n        class=\"metric-card-header-text\"\n        [ngStyle]=\"getTitleStyles()\"\n      >\n        {{ getTitleText() }}\n      </span>\n      <span *ngIf=\"!isTitleObject()\" class=\"metric-card-header-text\">\n        {{ cardData.title }}\n      </span>\n    </div>\n\n    <div class=\"metric-card-content\">\n      <div class=\"metric-value\" [ngStyle]=\"getValueStyles()\">\n        <span\n          *ngIf=\"isLabelOnLeft()\"\n          class=\"value-label\"\n          [ngStyle]=\"getLabelStyles('left')\"\n        >\n          {{ getLabelText() }}\n        </span>\n\n        <span [ngStyle]=\"getValueStyles()\">{{ getValueText() }}</span>\n\n        <span\n          *ngIf=\"isLabelOnRight()\"\n          class=\"value-label\"\n          [ngStyle]=\"getLabelStyles('right')\"\n        >\n          {{ getLabelText() }}\n        </span>\n      </div>\n\n      <div\n        *ngIf=\"cardData.divider?.show\"\n        class=\"metric-divider\"\n        [ngStyle]=\"{ backgroundColor: cardData.divider?.color || '#e2e8f0' }\"\n      ></div>\n\n      <div\n        *ngIf=\"isAdditionalInfoObject()\"\n        class=\"metric-additional-info\"\n        [ngStyle]=\"getAdditionalInfoStyles()\"\n      >\n        {{ getAdditionalInfoText() }}\n      </div>\n      <div\n        *ngIf=\"!isAdditionalInfoObject()\"\n        class=\"metric-additional-info\"\n        [ngStyle]=\"getAdditionalInfoStyles()\"\n      >\n        {{ cardData.additionalInfo }}\n      </div>\n    </div>\n  </div>\n</div>\n"]}
|
|
194
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"pt-metric-card.component.js","sourceRoot":"","sources":["../../../../../projects/ng-prime-tools/src/lib/pt-metric-card/pt-metric-card.component.ts","../../../../../projects/ng-prime-tools/src/lib/pt-metric-card/pt-metric-card.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input } from '@angular/core';\nimport {\n  LabelSetting,\n  MetricCardData,\n  ValueSetting,\n  IconStyle,\n  TextStyle,\n} from '../models';\n\n@Component({\n  selector: 'pt-metric-card',\n  templateUrl: './pt-metric-card.component.html',\n  styleUrls: ['./pt-metric-card.component.css'],\n})\nexport class PTMetricCardComponent {\n  private static readonly DEFAULT_ICON_COLOR = '#5a67d8';\n  private static readonly DEFAULT_ICON_SIZE = '24px';\n  private static readonly DEFAULT_TITLE_COLOR = '#5a67d8';\n  private static readonly DEFAULT_TITLE_SIZE = '16px';\n  private static readonly DEFAULT_VALUE_COLOR = '#2d3748';\n  private static readonly DEFAULT_VALUE_SIZE = '32px';\n  private static readonly DEFAULT_LABEL_COLOR = '#718096';\n  private static readonly DEFAULT_LABEL_SIZE = '14px';\n  private static readonly DEFAULT_ADDITIONAL_INFO_COLOR = 'gray';\n  private static readonly DEFAULT_ADDITIONAL_INFO_SIZE = '14px';\n  private static readonly DEFAULT_BACKGROUND_COLOR = '#fff';\n\n  @Input() cardData!: MetricCardData;\n\n  isIconObject(): boolean {\n    return (\n      typeof this.cardData.icon === 'object' && this.cardData.icon !== null\n    );\n  }\n\n  isTitleObject(): boolean {\n    return (\n      typeof this.cardData.title === 'object' && this.cardData.title !== null\n    );\n  }\n\n  isValueObject(): boolean {\n    return (\n      typeof this.cardData.value === 'object' && this.cardData.value !== null\n    );\n  }\n\n  isLabelObject(label: LabelSetting | string | undefined): boolean {\n    return typeof label === 'object' && label !== null;\n  }\n\n  isAdditionalInfoObject(): boolean {\n    return (\n      typeof this.cardData.additionalInfo === 'object' &&\n      this.cardData.additionalInfo !== null\n    );\n  }\n\n  getIconText(): string {\n    return this.isIconObject()\n      ? (this.cardData.icon as IconStyle).code\n      : (this.cardData.icon as string);\n  }\n\n  getTitleText(): string {\n    return this.isTitleObject()\n      ? (this.cardData.title as TextStyle).text\n      : (this.cardData.title as string) || '';\n  }\n\n  getValueText(): string {\n    return this.isValueObject()\n      ? (this.cardData.value as ValueSetting).text\n      : (this.cardData.value as string) || '';\n  }\n\n  getLabelText(): string {\n    const label = (this.cardData.value as ValueSetting)?.label;\n    return this.isLabelObject(label)\n      ? (label as LabelSetting).text\n      : (label as string) || '';\n  }\n\n  getAdditionalInfoText(): string {\n    return this.isAdditionalInfoObject()\n      ? (this.cardData.additionalInfo as TextStyle).text\n      : (this.cardData.additionalInfo as string) || '';\n  }\n\n  getIconStyles() {\n    if (this.isIconObject()) {\n      const icon = this.cardData.icon as IconStyle;\n      const shape = icon.shape || 'circular';\n      const borderRadius = shape === 'circular' ? '50%' : '8px';\n\n      return {\n        color: icon.color || PTMetricCardComponent.DEFAULT_ICON_COLOR,\n        fontSize: icon.fontSize || PTMetricCardComponent.DEFAULT_ICON_SIZE,\n        backgroundColor: icon.backgroundColor || 'transparent',\n        borderRadius,\n        padding: '6px',\n        display: 'inline-block',\n      };\n    }\n\n    return {};\n  }\n\n  getTitleStyles() {\n    if (this.isTitleObject()) {\n      const title = this.cardData.title as TextStyle;\n\n      return {\n        color: title.color || PTMetricCardComponent.DEFAULT_TITLE_COLOR,\n        fontSize: title.fontSize || PTMetricCardComponent.DEFAULT_TITLE_SIZE,\n        fontWeight: title.fontWeight || 'normal',\n      };\n    }\n\n    return {};\n  }\n\n  getValueStyles() {\n    if (this.isValueObject()) {\n      const value = this.cardData.value as ValueSetting;\n      const position = value.position || 'left';\n\n      return {\n        color: value.color || PTMetricCardComponent.DEFAULT_VALUE_COLOR,\n        fontSize: value.fontSize || PTMetricCardComponent.DEFAULT_VALUE_SIZE,\n        fontWeight: value.fontWeight || 'bold',\n        textAlign: position,\n        width: '100%',\n        display: 'flex',\n        justifyContent:\n          position === 'center'\n            ? 'center'\n            : position === 'right'\n              ? 'flex-end'\n              : 'flex-start',\n        alignItems: 'center',\n      };\n    }\n\n    return {\n      color: PTMetricCardComponent.DEFAULT_VALUE_COLOR,\n      fontSize: PTMetricCardComponent.DEFAULT_VALUE_SIZE,\n      fontWeight: 'bold',\n      textAlign: 'left',\n      width: '100%',\n      display: 'flex',\n      justifyContent: 'flex-start',\n      alignItems: 'center',\n    };\n  }\n\n  getValueTextStyles() {\n    if (this.isValueObject()) {\n      const value = this.cardData.value as ValueSetting;\n\n      return {\n        textAlign: value.position || 'left',\n      };\n    }\n\n    return {\n      textAlign: 'left',\n    };\n  }\n\n  getLabelStyles(position: 'left' | 'right') {\n    const label = (this.cardData.value as ValueSetting)?.label;\n\n    if (\n      this.isLabelObject(label) &&\n      (label as LabelSetting).position === position\n    ) {\n      return {\n        color:\n          (label as LabelSetting).color ||\n          PTMetricCardComponent.DEFAULT_LABEL_COLOR,\n        fontSize:\n          (label as LabelSetting).fontSize ||\n          PTMetricCardComponent.DEFAULT_LABEL_SIZE,\n        fontWeight: (label as LabelSetting).fontWeight || 'normal',\n      };\n    }\n\n    return {\n      color: PTMetricCardComponent.DEFAULT_LABEL_COLOR,\n      fontSize: PTMetricCardComponent.DEFAULT_LABEL_SIZE,\n      fontWeight: 'normal',\n    };\n  }\n\n  getAdditionalInfoStyles() {\n    if (this.isAdditionalInfoObject()) {\n      const additionalInfo = this.cardData.additionalInfo as TextStyle;\n\n      return {\n        color:\n          additionalInfo.color ||\n          PTMetricCardComponent.DEFAULT_ADDITIONAL_INFO_COLOR,\n        fontSize:\n          additionalInfo.fontSize ||\n          PTMetricCardComponent.DEFAULT_ADDITIONAL_INFO_SIZE,\n        fontWeight: additionalInfo.fontWeight || 'normal',\n      };\n    }\n\n    return {\n      color: PTMetricCardComponent.DEFAULT_ADDITIONAL_INFO_COLOR,\n      fontSize: PTMetricCardComponent.DEFAULT_ADDITIONAL_INFO_SIZE,\n      fontWeight: 'normal',\n    };\n  }\n\n  isLabelOnLeft(): boolean {\n    const label = (this.cardData.value as ValueSetting)?.label;\n\n    return (\n      this.isLabelObject(label) && (label as LabelSetting).position === 'left'\n    );\n  }\n\n  isLabelOnRight(): boolean {\n    const label = (this.cardData.value as ValueSetting)?.label;\n\n    return (\n      this.isLabelObject(label) && (label as LabelSetting).position === 'right'\n    );\n  }\n\n  getMetricCardContainerStyle() {\n    return {\n      minWidth: this.cardData.size?.minWidth || '100%',\n      maxWidth: this.cardData.size?.maxWidth || 'calc(25% - 16px)',\n      width: this.cardData.size?.fixedWidth || 'auto',\n    };\n  }\n\n  getMetricCardStyle() {\n    return {\n      backgroundColor:\n        this.cardData.appearance?.backgroundColor ||\n        PTMetricCardComponent.DEFAULT_BACKGROUND_COLOR,\n      minWidth: this.cardData.size?.minWidth || '100%',\n      maxWidth: this.cardData.size?.maxWidth || 'calc(25% - 16px)',\n      width: this.cardData.size?.fixedWidth || 'auto',\n    };\n  }\n}\n","<div class=\"metric-card-container\" [ngStyle]=\"getMetricCardContainerStyle()\">\n  <div\n    *ngIf=\"cardData.verticalLine?.show\"\n    class=\"vertical-line\"\n    [ngStyle]=\"{\n      'background-color': cardData.verticalLine?.color || '#5a67d8',\n    }\"\n  ></div>\n\n  <div class=\"metric-card\" [ngStyle]=\"getMetricCardStyle()\">\n    <div class=\"metric-card-header\">\n      <i\n        *ngIf=\"isIconObject()\"\n        [ngClass]=\"getIconText()\"\n        [ngStyle]=\"getIconStyles()\"\n      ></i>\n      <i *ngIf=\"!isIconObject()\" [ngClass]=\"cardData.icon\"></i>\n\n      <span\n        *ngIf=\"isTitleObject()\"\n        class=\"metric-card-header-text\"\n        [ngStyle]=\"getTitleStyles()\"\n      >\n        {{ getTitleText() }}\n      </span>\n      <span *ngIf=\"!isTitleObject()\" class=\"metric-card-header-text\">\n        {{ cardData.title }}\n      </span>\n    </div>\n\n    <div class=\"metric-card-content\">\n      <div class=\"metric-value\" [ngStyle]=\"getValueStyles()\">\n        <span\n          *ngIf=\"isLabelOnLeft()\"\n          class=\"value-label\"\n          [ngStyle]=\"getLabelStyles('left')\"\n        >\n          {{ getLabelText() }}\n        </span>\n\n        <span class=\"metric-value-text\" [ngStyle]=\"getValueTextStyles()\">\n          {{ getValueText() }}\n        </span>\n\n        <span\n          *ngIf=\"isLabelOnRight()\"\n          class=\"value-label\"\n          [ngStyle]=\"getLabelStyles('right')\"\n        >\n          {{ getLabelText() }}\n        </span>\n      </div>\n\n      <div\n        *ngIf=\"cardData.divider?.show\"\n        class=\"metric-divider\"\n        [ngStyle]=\"{ backgroundColor: cardData.divider?.color || '#e2e8f0' }\"\n      ></div>\n\n      <div\n        *ngIf=\"isAdditionalInfoObject()\"\n        class=\"metric-additional-info\"\n        [ngStyle]=\"getAdditionalInfoStyles()\"\n      >\n        {{ getAdditionalInfoText() }}\n      </div>\n      <div\n        *ngIf=\"!isAdditionalInfoObject()\"\n        class=\"metric-additional-info\"\n        [ngStyle]=\"getAdditionalInfoStyles()\"\n      >\n        {{ cardData.additionalInfo }}\n      </div>\n    </div>\n  </div>\n</div>\n"]}
|
|
@@ -203,11 +203,11 @@ export class PTSideBarMenuComponent {
|
|
|
203
203
|
}));
|
|
204
204
|
}
|
|
205
205
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.11", ngImport: i0, type: PTSideBarMenuComponent, deps: [{ token: i0.Renderer2 }, { token: i0.ElementRef }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
206
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.11", type: PTSideBarMenuComponent, selector: "pt-side-bar-menu", inputs: { menuConfig: "menuConfig" }, ngImport: i0, template: "<div class=\"pt-side-bar-menu\">\n <pt-card\n *ngIf=\"menuConfig.searchable\"\n [config]=\"searchCardConfig\"\n [ngClass]=\"{ 'sidebar-hidden': !menuConfig.isVisible }\"\n >\n <div class=\"search-input\">\n <pt-text-input [formGroup]=\"formGroup\" [formField]=\"searchField\">\n </pt-text-input>\n </div>\n </pt-card>\n\n <pt-card\n [config]=\"cardConfig\"\n [ngClass]=\"{ 'sidebar-hidden': !menuConfig.isVisible }\"\n >\n <ul class=\"menu-list\">\n <ng-container\n *ngTemplateOutlet=\"\n recursiveMenu;\n context: { $implicit: filteredMenus, level: 0 }\n \"\n ></ng-container>\n </ul>\n </pt-card>\n</div>\n\n<ng-template #recursiveMenu let-menus let-level=\"level\">\n <ng-container *ngFor=\"let item of menus\">\n <li\n class=\"menu-item\"\n [ngClass]=\"{\n 'menu-item-expanded': item.isExpanded,\n 'submenu-level-item': level > 0,\n }\"\n [attr.data-level]=\"level\"\n [ngStyle]=\"getMenuItemStyles()\"\n >\n <!-- item without children -->\n <a\n *ngIf=\"!hasChildren(item)\"\n [routerLink]=\"item.url\"\n [class.menu-link]=\"level === 0\"\n [class.submenu-link]=\"level > 0\"\n [ngStyle]=\"\n level === 0 ? getMenuLinkStyles() : getSubMenuLinkStyles(level)\n \"\n >\n <i [ngClass]=\"item.icon\"></i>\n <span [class.submenu-title]=\"level > 0\">{{ item.label }}</span>\n <span\n *ngIf=\"item.badge\"\n [ngStyle]=\"getBadgeStyles(item.badge)\"\n class=\"badge\"\n >\n {{ item.badge.count }}\n </span>\n </a>\n\n <!-- item with children -->\n <a\n *ngIf=\"hasChildren(item)\"\n href=\"#\"\n [class.menu-link]=\"level === 0\"\n [class.submenu-link]=\"level > 0\"\n (click)=\"toggleMenu(item, $event)\"\n [ngStyle]=\"\n level === 0 ? getMenuLinkStyles() : getSubMenuLinkStyles(level)\n \"\n >\n <i [ngClass]=\"item.icon\"></i>\n <span [class.submenu-title]=\"level > 0\">{{ item.label }}</span>\n <span\n *ngIf=\"item.badge\"\n [ngStyle]=\"getBadgeStyles(item.badge)\"\n class=\"badge\"\n >\n {{ item.badge.count }}\n </span>\n <i [ngClass]=\"getChevronClass(item)\" class=\"chevron\"></i>\n </a>\n\n <div\n class=\"submenu-wrapper\"\n [class.expanded]=\"item.isExpanded\"\n *ngIf=\"hasChildren(item)\"\n >\n <ul\n class=\"submenu-list\"\n [ngClass]=\"{\n 'submenu-list-root': level === 0,\n 'submenu-list-nested': level > 0,\n }\"\n >\n <ng-container\n *ngTemplateOutlet=\"\n recursiveMenu;\n context: { $implicit: item.children, level: level + 1 }\n \"\n ></ng-container>\n </ul>\n </div>\n </li>\n </ng-container>\n</ng-template>\n", styles: [".pt-side-bar-menu .badge{background-color:#38bdf8;border-radius:5px;color:#fff;font-size:12px;margin-left:auto;padding:2px 8px}.pt-side-bar-menu .chevron{color:#8c8c8c;cursor:pointer;margin-left:auto}.pt-side-bar-menu .menu-item{position:relative}.pt-side-bar-menu .menu-link{align-items:center;border-radius:8px;color:#333;cursor:pointer;display:flex;font-size:16px;margin-bottom:
|
|
206
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.11", type: PTSideBarMenuComponent, selector: "pt-side-bar-menu", inputs: { menuConfig: "menuConfig" }, ngImport: i0, template: "<div class=\"pt-side-bar-menu\">\n <pt-card\n *ngIf=\"menuConfig.searchable\"\n [config]=\"searchCardConfig\"\n [ngClass]=\"{ 'sidebar-hidden': !menuConfig.isVisible }\"\n >\n <div class=\"search-input\">\n <pt-text-input [formGroup]=\"formGroup\" [formField]=\"searchField\">\n </pt-text-input>\n </div>\n </pt-card>\n\n <pt-card\n [config]=\"cardConfig\"\n [ngClass]=\"{ 'sidebar-hidden': !menuConfig.isVisible }\"\n >\n <ul class=\"menu-list\">\n <ng-container\n *ngTemplateOutlet=\"\n recursiveMenu;\n context: { $implicit: filteredMenus, level: 0 }\n \"\n ></ng-container>\n </ul>\n </pt-card>\n</div>\n\n<ng-template #recursiveMenu let-menus let-level=\"level\">\n <ng-container *ngFor=\"let item of menus\">\n <li\n class=\"menu-item\"\n [ngClass]=\"{\n 'menu-item-expanded': item.isExpanded,\n 'submenu-level-item': level > 0,\n }\"\n [attr.data-level]=\"level\"\n [ngStyle]=\"getMenuItemStyles()\"\n >\n <!-- item without children -->\n <a\n *ngIf=\"!hasChildren(item)\"\n [routerLink]=\"item.url\"\n [class.menu-link]=\"level === 0\"\n [class.submenu-link]=\"level > 0\"\n [ngStyle]=\"\n level === 0 ? getMenuLinkStyles() : getSubMenuLinkStyles(level)\n \"\n >\n <i [ngClass]=\"item.icon\"></i>\n <span [class.submenu-title]=\"level > 0\">{{ item.label }}</span>\n <span\n *ngIf=\"item.badge\"\n [ngStyle]=\"getBadgeStyles(item.badge)\"\n class=\"badge\"\n >\n {{ item.badge.count }}\n </span>\n </a>\n\n <!-- item with children -->\n <a\n *ngIf=\"hasChildren(item)\"\n href=\"#\"\n [class.menu-link]=\"level === 0\"\n [class.submenu-link]=\"level > 0\"\n (click)=\"toggleMenu(item, $event)\"\n [ngStyle]=\"\n level === 0 ? getMenuLinkStyles() : getSubMenuLinkStyles(level)\n \"\n >\n <i [ngClass]=\"item.icon\"></i>\n <span [class.submenu-title]=\"level > 0\">{{ item.label }}</span>\n <span\n *ngIf=\"item.badge\"\n [ngStyle]=\"getBadgeStyles(item.badge)\"\n class=\"badge\"\n >\n {{ item.badge.count }}\n </span>\n <i [ngClass]=\"getChevronClass(item)\" class=\"chevron\"></i>\n </a>\n\n <div\n class=\"submenu-wrapper\"\n [class.expanded]=\"item.isExpanded\"\n *ngIf=\"hasChildren(item)\"\n >\n <ul\n class=\"submenu-list\"\n [ngClass]=\"{\n 'submenu-list-root': level === 0,\n 'submenu-list-nested': level > 0,\n }\"\n >\n <ng-container\n *ngTemplateOutlet=\"\n recursiveMenu;\n context: { $implicit: item.children, level: level + 1 }\n \"\n ></ng-container>\n </ul>\n </div>\n </li>\n </ng-container>\n</ng-template>\n", styles: [".pt-side-bar-menu .badge{background-color:#38bdf8;border-radius:5px;color:#fff;font-size:12px;margin-left:auto;padding:2px 8px}.pt-side-bar-menu .chevron{color:#8c8c8c;cursor:pointer;margin-left:auto}.pt-side-bar-menu .menu-item{position:relative}.pt-side-bar-menu .menu-link{align-items:center;border-radius:8px;color:#333;cursor:pointer;display:flex;font-size:16px;margin-bottom:6px;padding:10px 15px;text-decoration:none;transition:background-color .2s,color .2s}.pt-side-bar-menu .menu-link:hover{background-color:#f1f1f1;color:#111}.pt-side-bar-menu .menu-link i{margin-right:10px}.pt-side-bar-menu .menu-list{list-style-type:none;padding:0;margin:0}.pt-side-bar-menu pt-card{transition:transform .3s ease-in-out,opacity .3s ease-in-out;display:block}.pt-side-bar-menu .sidebar-hidden{transform:translate(-100%);opacity:0}.pt-side-bar-menu .sidebar-menu{margin:0;padding:0;transition:transform .3s ease-in-out;width:250px}.pt-side-bar-menu .submenu-wrapper{display:grid;grid-template-rows:0fr;transition:grid-template-rows .3s ease;min-height:0}.pt-side-bar-menu .submenu-wrapper.expanded{grid-template-rows:1fr}.pt-side-bar-menu .submenu-wrapper:not(.expanded){margin:0!important;padding:0!important}.pt-side-bar-menu .submenu-list{overflow:hidden;min-height:0;list-style-type:none;margin-top:5px;margin-bottom:0;position:relative}.pt-side-bar-menu .submenu-wrapper:not(.expanded) .submenu-list{margin:0!important;padding-top:0!important;padding-bottom:0!important;border-left:none!important}.pt-side-bar-menu .submenu-list-root{border-left:1px solid #e5e7eb;margin-left:23px;padding-left:25px;padding-top:5px;padding-bottom:5px}.pt-side-bar-menu .submenu-list-root:after{background-color:#e5e7eb;bottom:0;content:\"\";left:-1px;position:absolute;top:0;width:1px}.pt-side-bar-menu .submenu-list-nested{border-left:1px solid #2f3b46;margin-left:18px;padding-left:20px;padding-top:4px;padding-bottom:4px}.pt-side-bar-menu .submenu-item,.pt-side-bar-menu .submenu-level-item{padding:5px 0;position:relative}.pt-side-bar-menu .submenu-level-item:before{background-color:#e5e7eb;content:\"\";height:1px;left:-25px;position:absolute;top:22px;transform:translateY(-50%);width:25px}.pt-side-bar-menu .submenu-link{align-items:center;border-radius:5px;color:#666;display:flex;font-size:14px;padding:5px 10px;text-decoration:none;transition:background-color .2s,color .2s}.pt-side-bar-menu .submenu-link:hover{background-color:#f1f1f1;color:#000}.pt-side-bar-menu .submenu-title{margin-left:5px}\n"], dependencies: [{ kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "directive", type: i1.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { kind: "directive", type: i2.RouterLink, selector: "[routerLink]", inputs: ["target", "queryParams", "fragment", "queryParamsHandling", "state", "info", "relativeTo", "preserveFragment", "skipLocationChange", "replaceUrl", "routerLink"] }, { kind: "component", type: i3.PTCardComponent, selector: "pt-card", inputs: ["config"] }, { kind: "component", type: i4.PTTextInputComponent, selector: "pt-text-input", inputs: ["formGroup", "formField"] }] }); }
|
|
207
207
|
}
|
|
208
208
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.11", ngImport: i0, type: PTSideBarMenuComponent, decorators: [{
|
|
209
209
|
type: Component,
|
|
210
|
-
args: [{ selector: 'pt-side-bar-menu', template: "<div class=\"pt-side-bar-menu\">\n <pt-card\n *ngIf=\"menuConfig.searchable\"\n [config]=\"searchCardConfig\"\n [ngClass]=\"{ 'sidebar-hidden': !menuConfig.isVisible }\"\n >\n <div class=\"search-input\">\n <pt-text-input [formGroup]=\"formGroup\" [formField]=\"searchField\">\n </pt-text-input>\n </div>\n </pt-card>\n\n <pt-card\n [config]=\"cardConfig\"\n [ngClass]=\"{ 'sidebar-hidden': !menuConfig.isVisible }\"\n >\n <ul class=\"menu-list\">\n <ng-container\n *ngTemplateOutlet=\"\n recursiveMenu;\n context: { $implicit: filteredMenus, level: 0 }\n \"\n ></ng-container>\n </ul>\n </pt-card>\n</div>\n\n<ng-template #recursiveMenu let-menus let-level=\"level\">\n <ng-container *ngFor=\"let item of menus\">\n <li\n class=\"menu-item\"\n [ngClass]=\"{\n 'menu-item-expanded': item.isExpanded,\n 'submenu-level-item': level > 0,\n }\"\n [attr.data-level]=\"level\"\n [ngStyle]=\"getMenuItemStyles()\"\n >\n <!-- item without children -->\n <a\n *ngIf=\"!hasChildren(item)\"\n [routerLink]=\"item.url\"\n [class.menu-link]=\"level === 0\"\n [class.submenu-link]=\"level > 0\"\n [ngStyle]=\"\n level === 0 ? getMenuLinkStyles() : getSubMenuLinkStyles(level)\n \"\n >\n <i [ngClass]=\"item.icon\"></i>\n <span [class.submenu-title]=\"level > 0\">{{ item.label }}</span>\n <span\n *ngIf=\"item.badge\"\n [ngStyle]=\"getBadgeStyles(item.badge)\"\n class=\"badge\"\n >\n {{ item.badge.count }}\n </span>\n </a>\n\n <!-- item with children -->\n <a\n *ngIf=\"hasChildren(item)\"\n href=\"#\"\n [class.menu-link]=\"level === 0\"\n [class.submenu-link]=\"level > 0\"\n (click)=\"toggleMenu(item, $event)\"\n [ngStyle]=\"\n level === 0 ? getMenuLinkStyles() : getSubMenuLinkStyles(level)\n \"\n >\n <i [ngClass]=\"item.icon\"></i>\n <span [class.submenu-title]=\"level > 0\">{{ item.label }}</span>\n <span\n *ngIf=\"item.badge\"\n [ngStyle]=\"getBadgeStyles(item.badge)\"\n class=\"badge\"\n >\n {{ item.badge.count }}\n </span>\n <i [ngClass]=\"getChevronClass(item)\" class=\"chevron\"></i>\n </a>\n\n <div\n class=\"submenu-wrapper\"\n [class.expanded]=\"item.isExpanded\"\n *ngIf=\"hasChildren(item)\"\n >\n <ul\n class=\"submenu-list\"\n [ngClass]=\"{\n 'submenu-list-root': level === 0,\n 'submenu-list-nested': level > 0,\n }\"\n >\n <ng-container\n *ngTemplateOutlet=\"\n recursiveMenu;\n context: { $implicit: item.children, level: level + 1 }\n \"\n ></ng-container>\n </ul>\n </div>\n </li>\n </ng-container>\n</ng-template>\n", styles: [".pt-side-bar-menu .badge{background-color:#38bdf8;border-radius:5px;color:#fff;font-size:12px;margin-left:auto;padding:2px 8px}.pt-side-bar-menu .chevron{color:#8c8c8c;cursor:pointer;margin-left:auto}.pt-side-bar-menu .menu-item{position:relative}.pt-side-bar-menu .menu-link{align-items:center;border-radius:8px;color:#333;cursor:pointer;display:flex;font-size:16px;margin-bottom:
|
|
210
|
+
args: [{ selector: 'pt-side-bar-menu', template: "<div class=\"pt-side-bar-menu\">\n <pt-card\n *ngIf=\"menuConfig.searchable\"\n [config]=\"searchCardConfig\"\n [ngClass]=\"{ 'sidebar-hidden': !menuConfig.isVisible }\"\n >\n <div class=\"search-input\">\n <pt-text-input [formGroup]=\"formGroup\" [formField]=\"searchField\">\n </pt-text-input>\n </div>\n </pt-card>\n\n <pt-card\n [config]=\"cardConfig\"\n [ngClass]=\"{ 'sidebar-hidden': !menuConfig.isVisible }\"\n >\n <ul class=\"menu-list\">\n <ng-container\n *ngTemplateOutlet=\"\n recursiveMenu;\n context: { $implicit: filteredMenus, level: 0 }\n \"\n ></ng-container>\n </ul>\n </pt-card>\n</div>\n\n<ng-template #recursiveMenu let-menus let-level=\"level\">\n <ng-container *ngFor=\"let item of menus\">\n <li\n class=\"menu-item\"\n [ngClass]=\"{\n 'menu-item-expanded': item.isExpanded,\n 'submenu-level-item': level > 0,\n }\"\n [attr.data-level]=\"level\"\n [ngStyle]=\"getMenuItemStyles()\"\n >\n <!-- item without children -->\n <a\n *ngIf=\"!hasChildren(item)\"\n [routerLink]=\"item.url\"\n [class.menu-link]=\"level === 0\"\n [class.submenu-link]=\"level > 0\"\n [ngStyle]=\"\n level === 0 ? getMenuLinkStyles() : getSubMenuLinkStyles(level)\n \"\n >\n <i [ngClass]=\"item.icon\"></i>\n <span [class.submenu-title]=\"level > 0\">{{ item.label }}</span>\n <span\n *ngIf=\"item.badge\"\n [ngStyle]=\"getBadgeStyles(item.badge)\"\n class=\"badge\"\n >\n {{ item.badge.count }}\n </span>\n </a>\n\n <!-- item with children -->\n <a\n *ngIf=\"hasChildren(item)\"\n href=\"#\"\n [class.menu-link]=\"level === 0\"\n [class.submenu-link]=\"level > 0\"\n (click)=\"toggleMenu(item, $event)\"\n [ngStyle]=\"\n level === 0 ? getMenuLinkStyles() : getSubMenuLinkStyles(level)\n \"\n >\n <i [ngClass]=\"item.icon\"></i>\n <span [class.submenu-title]=\"level > 0\">{{ item.label }}</span>\n <span\n *ngIf=\"item.badge\"\n [ngStyle]=\"getBadgeStyles(item.badge)\"\n class=\"badge\"\n >\n {{ item.badge.count }}\n </span>\n <i [ngClass]=\"getChevronClass(item)\" class=\"chevron\"></i>\n </a>\n\n <div\n class=\"submenu-wrapper\"\n [class.expanded]=\"item.isExpanded\"\n *ngIf=\"hasChildren(item)\"\n >\n <ul\n class=\"submenu-list\"\n [ngClass]=\"{\n 'submenu-list-root': level === 0,\n 'submenu-list-nested': level > 0,\n }\"\n >\n <ng-container\n *ngTemplateOutlet=\"\n recursiveMenu;\n context: { $implicit: item.children, level: level + 1 }\n \"\n ></ng-container>\n </ul>\n </div>\n </li>\n </ng-container>\n</ng-template>\n", styles: [".pt-side-bar-menu .badge{background-color:#38bdf8;border-radius:5px;color:#fff;font-size:12px;margin-left:auto;padding:2px 8px}.pt-side-bar-menu .chevron{color:#8c8c8c;cursor:pointer;margin-left:auto}.pt-side-bar-menu .menu-item{position:relative}.pt-side-bar-menu .menu-link{align-items:center;border-radius:8px;color:#333;cursor:pointer;display:flex;font-size:16px;margin-bottom:6px;padding:10px 15px;text-decoration:none;transition:background-color .2s,color .2s}.pt-side-bar-menu .menu-link:hover{background-color:#f1f1f1;color:#111}.pt-side-bar-menu .menu-link i{margin-right:10px}.pt-side-bar-menu .menu-list{list-style-type:none;padding:0;margin:0}.pt-side-bar-menu pt-card{transition:transform .3s ease-in-out,opacity .3s ease-in-out;display:block}.pt-side-bar-menu .sidebar-hidden{transform:translate(-100%);opacity:0}.pt-side-bar-menu .sidebar-menu{margin:0;padding:0;transition:transform .3s ease-in-out;width:250px}.pt-side-bar-menu .submenu-wrapper{display:grid;grid-template-rows:0fr;transition:grid-template-rows .3s ease;min-height:0}.pt-side-bar-menu .submenu-wrapper.expanded{grid-template-rows:1fr}.pt-side-bar-menu .submenu-wrapper:not(.expanded){margin:0!important;padding:0!important}.pt-side-bar-menu .submenu-list{overflow:hidden;min-height:0;list-style-type:none;margin-top:5px;margin-bottom:0;position:relative}.pt-side-bar-menu .submenu-wrapper:not(.expanded) .submenu-list{margin:0!important;padding-top:0!important;padding-bottom:0!important;border-left:none!important}.pt-side-bar-menu .submenu-list-root{border-left:1px solid #e5e7eb;margin-left:23px;padding-left:25px;padding-top:5px;padding-bottom:5px}.pt-side-bar-menu .submenu-list-root:after{background-color:#e5e7eb;bottom:0;content:\"\";left:-1px;position:absolute;top:0;width:1px}.pt-side-bar-menu .submenu-list-nested{border-left:1px solid #2f3b46;margin-left:18px;padding-left:20px;padding-top:4px;padding-bottom:4px}.pt-side-bar-menu .submenu-item,.pt-side-bar-menu .submenu-level-item{padding:5px 0;position:relative}.pt-side-bar-menu .submenu-level-item:before{background-color:#e5e7eb;content:\"\";height:1px;left:-25px;position:absolute;top:22px;transform:translateY(-50%);width:25px}.pt-side-bar-menu .submenu-link{align-items:center;border-radius:5px;color:#666;display:flex;font-size:14px;padding:5px 10px;text-decoration:none;transition:background-color .2s,color .2s}.pt-side-bar-menu .submenu-link:hover{background-color:#f1f1f1;color:#000}.pt-side-bar-menu .submenu-title{margin-left:5px}\n"] }]
|
|
211
211
|
}], ctorParameters: () => [{ type: i0.Renderer2 }, { type: i0.ElementRef }], propDecorators: { menuConfig: [{
|
|
212
212
|
type: Input
|
|
213
213
|
}] } });
|