tin-spa 2.9.0 → 2.9.2

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.
@@ -115,11 +115,25 @@ export class DetailsDialogLite {
115
115
  }
116
116
  testDisabled(row, buttonName) {
117
117
  const button = this.getButton(buttonName);
118
- return button ? this.buttonService.testDisabled(button, row) : false;
118
+ if (!button)
119
+ return false;
120
+ // Create a copy of the button with component's detailsConfig
121
+ const buttonWithConfig = {
122
+ ...button,
123
+ detailsConfig: button.detailsConfig ? { ...button.detailsConfig } : this.detailsConfig
124
+ };
125
+ return this.buttonService.testDisabled(buttonWithConfig, row);
119
126
  }
120
127
  testVisible(row, buttonName) {
121
128
  const button = this.getButton(buttonName);
122
- return button ? this.buttonService.isButtonVisible(button, row, this.isLoadComplete) : false;
129
+ if (!button)
130
+ return false;
131
+ // Create a copy of the button with component's detailsConfig
132
+ const buttonWithConfig = {
133
+ ...button,
134
+ detailsConfig: button.detailsConfig ? { ...button.detailsConfig } : this.detailsConfig
135
+ };
136
+ return this.buttonService.testVisible(buttonWithConfig, row);
123
137
  }
124
138
  testVisibleTab(tblConfig) {
125
139
  if (!this.tableConfigService.hasTableAccess(tblConfig)) {
@@ -262,4 +276,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImpor
262
276
  type: Output
263
277
  }] } });
264
278
  ;
265
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"detailsDialog-lite.component.js","sourceRoot":"","sources":["../../../../../../projects/tin-spa/src/lib/components/table-lite/detailsDialog-lite.component.ts","../../../../../../projects/tin-spa/src/lib/components/table-lite/detailsDialog-lite.component.html"],"names":[],"mappings":"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","sourcesContent":["\r\nimport { Component, OnInit, Inject, Output, EventEmitter } from '@angular/core';\r\nimport { MAT_DIALOG_DATA, MatDialogRef } from '@angular/material/dialog';\r\nimport { FormConfig, Field, TableConfig, Button, DetailsDialogConfig, Action, ApiResponse, StepConfig } from '../../classes/Classes';\r\nimport { MessageService } from '../../services/message.service';\r\nimport { DataServiceLib } from '../../services/datalib.service';\r\nimport { LoaderService } from '../../services/loader.service';\r\nimport { BreakpointObserver, BreakpointState } from '@angular/cdk/layout';\r\nimport { Core, DetailsDialogProcessor } from '../../classes/TinCore';\r\nimport { DialogService } from '../../services/dialog.service';\r\nimport { ButtonService } from '../../services/button.service';\r\nimport { Subject } from 'rxjs';\r\nimport { AuthService } from '../../services/auth.service';\r\nimport { TableConfigService } from '../../services/table-config.service';\r\n@Component({\r\n  selector: 'spa-detailsDialog-lite', //uniq to lite\r\n  templateUrl: './detailsDialog-lite.component.html',\r\n  styleUrls: ['./table-lite.component.css']\r\n})\r\nexport class DetailsDialogLite implements OnInit {\r\n\r\n  constructor(private breakpointObserver: BreakpointObserver,private loaderService: LoaderService,public dataService: DataServiceLib,private messageService: MessageService,\r\n    private dialogRef: MatDialogRef<DetailsDialogLite>, @Inject(MAT_DIALOG_DATA) public detailsConfig: DetailsDialogConfig, private buttonService: ButtonService,\r\n    private dialogService: DialogService, private authService: AuthService, public tableConfigService: TableConfigService)\r\n  {\r\n    // detect screen size changes\r\n    this.breakpointObserver.observe([\"(max-width: 600px)\"]).subscribe((result: BreakpointState) => {\r\n\r\n      if (result.matches) {\r\n        this.smallScreen = true\r\n      }else {\r\n        this.smallScreen = false\r\n      }\r\n\r\n    });\r\n  }\r\n\r\n  ngOnInit() {\r\n\r\n    const { tableConfig, tableConfigs, formConfig, stepConfig, buttons } = DetailsDialogProcessor.initializeConfigs(this.detailsConfig);\r\n    this.tableConfig = tableConfig;\r\n    this.tableConfigs = tableConfigs;\r\n    this.formConfig = formConfig;\r\n    this.stepConfig = stepConfig;\r\n    this.buttons = buttons;\r\n\r\n    let resp = DetailsDialogProcessor.handleFormMode(this.detailsConfig, this.formConfig,);\r\n\r\n    if (resp.loadByAction) {\r\n      this.loadByAction = true;\r\n      formConfig.loadAction = resp.action;\r\n      this.loadData(resp.action, false);\r\n    } else {\r\n      this.details = resp.details;\r\n      DetailsDialogProcessor.setHeroValue(this.detailsConfig, this.details);\r\n      this.loadByAction = false;\r\n      this.isLoadComplete = true;\r\n    }\r\n\r\n    this.fileField = DetailsDialogProcessor.setFileField(this.formConfig);\r\n\r\n    this.loaderService.isLoading.subscribe(x => this.isProcessing = x);\r\n\r\n    const { extraButtons, createButton, editButton, deleteButton } = DetailsDialogProcessor.setupButtons(this.formConfig, this.buttons);\r\n    this.extraButtons = extraButtons;\r\n    this.createButton = createButton;\r\n    this.editButton = editButton;\r\n    this.deleteButton = deleteButton;\r\n\r\n    this.setTitleAction()\r\n\r\n  }\r\n\r\n\r\n  titleAction = \"View\";\r\n  loadByAction = false;\r\n\r\n  extraButtons: Button[];\r\n  buttons: Button[];\r\n  createButton: Button;\r\n  editButton: Button;\r\n  deleteButton: Button;\r\n\r\n  // modeButton : Button\r\n  fileField: Field;\r\n  files: any[] = [];\r\n\r\n  tableReload: Subject<boolean> = new Subject();\r\n\r\n  tableConfig: TableConfig\r\n  tableConfigs : TableConfig[]\r\n  details: any\r\n  formConfig: FormConfig;\r\n  stepConfig: StepConfig;\r\n  smallScreen = false;\r\n\r\n  isLoadComplete: boolean = false;\r\n  isProcessing: boolean = false;\r\n\r\n  @Output() inputChange = new EventEmitter();\r\n\r\n\r\n  loadData(action: Action, reload) {\r\n\r\n    if (action && this.loadByAction) {\r\n\r\n      this.dataService.CallApi(action).subscribe((apiResponse: ApiResponse) => {\r\n\r\n        if (apiResponse.success) {\r\n          this.details = apiResponse.data\r\n          this.isLoadComplete = true;\r\n          // DetailsDialogProcessor.setHeroValue(this.detailsConfig, this.details);\r\n          if (reload) this.tableReload.next(true);\r\n        }\r\n        else {\r\n          this.messageService.toast(\"Error: \" + apiResponse.message);\r\n        }\r\n\r\n      });\r\n    }\r\n\r\n  }\r\n\r\n  inputChanged(event) {\r\n    this.inputChange.emit({ field: event.field, value: event.value })\r\n  }\r\n\r\n\r\n  setMode(newMode) {\r\n    this.formConfig.mode = newMode\r\n    this.setTitleAction()\r\n  }\r\n\r\n  setTitleAction() {\r\n    if (this.formConfig.fixedTitle) {\r\n      this.titleAction = \"\";\r\n      return;\r\n    }\r\n\r\n    this.titleAction = this.buttonService.getTitleAction(this.formConfig.mode, this.buttons);\r\n  }\r\n\r\n  getButton(name: string) {\r\n    return this.buttonService.getButton(this.buttons, name);\r\n  }\r\n\r\n  testDisabled(row, buttonName: string): boolean {\r\n    const button = this.getButton(buttonName);\r\n    return button ? this.buttonService.testDisabled(button, row) : false;\r\n  }\r\n\r\n\r\n  testVisible(row, buttonName: string): boolean {\r\n    const button = this.getButton(buttonName);\r\n    return button ? this.buttonService.isButtonVisible(button, row, this.isLoadComplete) : false;\r\n  }\r\n\r\n  testVisibleTab(tblConfig: TableConfig): boolean {\r\n\r\n    if (!this.tableConfigService.hasTableAccess(tblConfig)) {\r\n      return false;\r\n    }\r\n\r\n    return this.buttonService.isTabVisible(tblConfig);\r\n  }\r\n\r\n  getButtonColor(button: Button, row) {\r\n    return this.buttonService.getButtonColor(button, row);\r\n  }\r\n\r\n  getVisibleFields() {\r\n    return this.formConfig.fields?.filter(x => Core.testVisible(this.formConfig, this.details, x, this.authService.currentRoleSource.value))\r\n  }\r\n\r\n\r\n  create() {\r\n    this.handleButtonAction('create');\r\n  }\r\n\r\n  edit() {\r\n    this.handleButtonAction('edit');\r\n  }\r\n\r\n  delete() {\r\n    this.messageService.confirm(`DELETE ?`).subscribe((result) => {\r\n      if (result == \"yes\") {\r\n        this.handleButtonAction('delete');\r\n      }\r\n    });\r\n  }\r\n\r\n\r\n  custom(button: Button) {\r\n    if (button.dialog && button.detailsConfig) {\r\n      this.openNestedDetailsDialog(button);\r\n    } else {\r\n      this.handleButtonAction(button.name);\r\n    }\r\n  }\r\n\r\n  private openNestedDetailsDialog(button: Button): void {\r\n\r\n    this.dialogService.openConfiguredDetailsDialog(button, this.details, DetailsDialogLite).subscribe(result => {\r\n\r\n      if (result.action === 'inputChange') {\r\n        this.inputChanged(result.change);\r\n      } else if (result && result.message === 'success') {\r\n        this.refreshData();\r\n      }\r\n\r\n    });\r\n  }\r\n\r\n  private refreshData(): void {\r\n    if (this.formConfig.loadAction) {\r\n      this.loadData(this.formConfig.loadAction, true);\r\n    }\r\n  }\r\n\r\n\r\n  handleButtonAction(buttonName: string) {\r\n\r\n    const button = this.getButton(buttonName);\r\n    if (!button) return;\r\n\r\n    if (!button.action) {\r\n      this.dialogRef.close({ message: 'emit', data: this.details });\r\n      return;\r\n    }\r\n\r\n    if (this.validateForm()) {\r\n      this.executeAction(button, this.details);\r\n    }\r\n  }\r\n\r\n  private validateForm(): boolean {\r\n    let validationResult = Core.validateObject(this.getVisibleFields(), this.details);\r\n    if (validationResult !== '') {\r\n      this.messageService.toast(validationResult);\r\n      return false;\r\n    }\r\n\r\n    if (this.files.length < 1 && this.fileField?.required && this.formConfig.mode === 'create') {\r\n      this.messageService.toast(\"Please attach file(s)\");\r\n      return false;\r\n    }\r\n\r\n    // Validate composite fields\r\n    for (const field of this.getVisibleFields()) {\r\n      if (field.type === 'composite') {\r\n        validationResult = Core.validateObject(Core.getVisibleSubfields(this.formConfig, this.details, field, this.authService.currentRoleSource.value), this.details);\r\n        if (validationResult !== '') {\r\n          this.messageService.toast(validationResult);\r\n          return false;\r\n        }\r\n      }\r\n    }\r\n\r\n    return true;\r\n  }\r\n\r\n  private prepareActionData(button: Button, data: any): any {\r\n    if (button.action?.isFormData) {\r\n      const formData = new FormData();\r\n      for (let i = 0; i < this.files.length; i++) {\r\n        formData.append(`uploadFile${i}`, this.files[i], this.files[i].name);\r\n      }\r\n      formData.append(\"data\", JSON.stringify(data));\r\n      return formData;\r\n    }\r\n    return data;\r\n  }\r\n\r\n  private executeAction(button: Button, data: any) {\r\n    const actionData = this.prepareActionData(button, data);\r\n\r\n    if (button.confirm) {\r\n      this.messageService.confirm(button.confirm.message).subscribe((result) => {\r\n        if (result === \"yes\") {\r\n          this.performApiCall(button, actionData);\r\n        }\r\n      });\r\n    } else {\r\n      this.performApiCall(button, actionData);\r\n    }\r\n  }\r\n\r\n  private performApiCall(button: Button, data: any) {\r\n    this.dataService.CallApi(button.action, data).subscribe((apiResponse: ApiResponse) => {\r\n\r\n      if (apiResponse.success) {\r\n\r\n        if (apiResponse.message.includes(\"approval\") || apiResponse.message.includes(\"approve\") || apiResponse.message.includes(\"decline\")) {\r\n          this.messageService.toast(apiResponse.message);\r\n        } else if (button.action.successMessage) {\r\n          this.messageService.toast(button.action.successMessage);\r\n        } else if (apiResponse.message != \"success\" && apiResponse.message != \"\") {\r\n          this.messageService.toast(apiResponse.message);\r\n        } else {\r\n          this.messageService.toast(\"Updated\");\r\n        }\r\n\r\n        this.dialogRef.close({ message: 'success', data: { ...apiResponse, data: this.details } });\r\n      } else {\r\n        this.messageService.toast(\"Error: \" + apiResponse.message);\r\n      }\r\n    });\r\n  }\r\n\r\n\r\n};\r\n\r\n\r\n\r\n","<div class=\"dialog-container\">\r\n\r\n  <div class=\"dialog-content\">\r\n\r\n    <mat-progress-bar mode=\"indeterminate\" *ngIf=\"isProcessing && dataService.appConfig.progressLine\"></mat-progress-bar>\r\n\r\n    <div class=\"d-flex justify-content-between align-items-center mt-0\">\r\n\r\n      <div>\r\n        <label style=\"font-size: 20px; font-weight:500;margin-bottom: 10px;\">{{titleAction | titlecase}} {{formConfig?.title}}</label>\r\n      </div>\r\n\r\n      <div class=\"d-flex justify-content-between align-items-center\">\r\n\r\n        <div *ngIf=\"formConfig.mode=='view' && editButton && testVisible(details,editButton.name)\">\r\n          <button mat-icon-button matTooltipPosition=\"above\" matTooltip=\"Edit\" color=\"primary\" (click)=\"setMode('edit')\" [disabled]=\"testDisabled(details,editButton.name)\"><mat-icon>edit</mat-icon></button>\r\n        </div>\r\n\r\n        <button [disabled]=\"isProcessing\" *ngIf=\"loadByAction\" mat-icon-button matTooltipPosition=\"above\" matTooltip=\"Refresh\" color=\"primary\" (click)=\"loadData(formConfig.loadAction, true)\"><mat-icon class=\"refreshIcon\">cached</mat-icon></button>\r\n      </div>\r\n\r\n    </div>\r\n\r\n    <mat-dialog-content class=\"mat-typography dialog-scroll-content\">\r\n\r\n          <spa-alert [alertMessages]=\"formConfig.alertMessages\" [data]=\"details\"></spa-alert>\r\n\r\n      <div class=\"tin-input\" style=\"font-size:14px\">\r\n\r\n         <p *ngIf=\"formConfig && !details\"><em>Loading...</em></p>\r\n\r\n        <spa-form *ngIf=\"formConfig && details\" [files]=\"files\" [data]=\"details\" [config]=\"formConfig\" (inputChange)=\"inputChanged($event)\">\r\n          <ng-template #dynamicSelect let-field=\"field\" let-data=\"data\" let-testReadOnly=\"testReadOnly\" let-selectChanged=\"selectChanged\">\r\n            <spa-select-lite\r\n              [display]=\"field.alias ?? field.name | camelToWords\"\r\n              [width]=\"field.width\"\r\n              [nullable]=\"field.nullable\"\r\n              [options]=\"field.options\"\r\n              [masterOptions]=\"field.masterOptions\"\r\n              [masterField]=\"field.masterField\"\r\n              [optionDisplay]=\"field.optionDisplay ?? 'name'\"\r\n              [optionValue]=\"field.optionValue ?? 'value'\"\r\n              [(value)]=\"data[field.name]\"\r\n              [defaultFirstValue]=\"field.defaultFirstValue\"\r\n              [readonly]=\"testReadOnly(field)\"\r\n              [hint]=\"field.hint\"\r\n              [loadAction]=\"field.loadAction\"\r\n              [field]=\"field\"\r\n              [infoMessage]=\"field.infoMessage\"\r\n              [copyContent]=\"field.copyContent\"\r\n              (valueChange)=\"selectChanged(field)\"\r\n            ></spa-select-lite>\r\n          </ng-template>\r\n        </spa-form>\r\n\r\n\r\n        <!-- <mat-tab-group *ngIf=\"tableConfigs\">\r\n\r\n          <ng-container *ngFor=\"let tbl of tableConfigService.getVisibleTableConfigs(tableConfigs)\">\r\n            <mat-tab *ngIf=\"testVisibleTab(tbl)\">\r\n\r\n              <ng-template matTabLabel>\r\n                <span>{{tbl.tabTitle ?? 'Untitled'}}</span>\r\n              </ng-template>\r\n\r\n              <spa-table-internal [config]=\"tbl\" [reload]=\"tableReload\" (dataLoad)=\"loadData(formConfig.loadAction, false)\"></spa-table-internal>\r\n\r\n            </mat-tab>\r\n          </ng-container>\r\n\r\n        </mat-tab-group> -->\r\n\r\n      </div>\r\n\r\n    </mat-dialog-content>\r\n\r\n\r\n  </div>\r\n\r\n  <mat-dialog-actions>\r\n\r\n    <div>\r\n\r\n      <button mat-raised-button [disabled]=\"isProcessing\" *ngIf=\"formConfig.mode=='create' && createButton\" color=\"primary\" (click)=\"create()\" cdkFocusInitial>{{createButton.display ?? 'Submit'}}\r\n      </button>\r\n\r\n      <button mat-raised-button [disabled]=\"isProcessing\" *ngIf=\"formConfig.mode=='edit' && editButton\" color=\"primary\" (click)=\"edit()\" cdkFocusInitial>{{editButton.display ?? 'Submit'}}\r\n      </button>\r\n\r\n      <ng-container *ngFor=\"let btn of extraButtons\">\r\n        <button *ngIf=\"testVisible(details,btn.name)\" mat-stroked-button [disabled]=\"isProcessing || testDisabled(details,btn.name)\" [ngStyle]=\"{'color': getButtonColor(btn, details)}\" (click)=\"custom(btn)\" cdkFocusInitial><mat-icon [ngStyle]=\"{'color': getButtonColor(btn, details)}\">{{btn.icon.name}}</mat-icon>{{btn.display ?? btn.name | titlecase}}\r\n        </button>\r\n      </ng-container>\r\n\r\n      <button mat-stroked-button color=\"primary\" mat-dialog-close>Cancel</button>\r\n\r\n    </div>\r\n\r\n    <div class=\"col d-flex justify-content-end\" *ngIf=\"smallScreen\">\r\n      <button mat-icon-button matTooltipPosition=\"above\" matTooltip=\"Delete\" [disabled]=\"isProcessing\" style=\"color: red;\" (click)=\"delete()\" *ngIf=\"formConfig.mode!='create'  && deleteButton\"><mat-icon>delete</mat-icon></button>\r\n    </div>\r\n\r\n\r\n  </mat-dialog-actions>\r\n\r\n\r\n</div>\r\n"]}
279
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"detailsDialog-lite.component.js","sourceRoot":"","sources":["../../../../../../projects/tin-spa/src/lib/components/table-lite/detailsDialog-lite.component.ts","../../../../../../projects/tin-spa/src/lib/components/table-lite/detailsDialog-lite.component.html"],"names":[],"mappings":"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","sourcesContent":["\r\nimport { Component, OnInit, Inject, Output, EventEmitter } from '@angular/core';\r\nimport { MAT_DIALOG_DATA, MatDialogRef } from '@angular/material/dialog';\r\nimport { FormConfig, Field, TableConfig, Button, DetailsDialogConfig, Action, ApiResponse, StepConfig } from '../../classes/Classes';\r\nimport { MessageService } from '../../services/message.service';\r\nimport { DataServiceLib } from '../../services/datalib.service';\r\nimport { LoaderService } from '../../services/loader.service';\r\nimport { BreakpointObserver, BreakpointState } from '@angular/cdk/layout';\r\nimport { Core, DetailsDialogProcessor } from '../../classes/TinCore';\r\nimport { DialogService } from '../../services/dialog.service';\r\nimport { ButtonService } from '../../services/button.service';\r\nimport { Subject } from 'rxjs';\r\nimport { AuthService } from '../../services/auth.service';\r\nimport { TableConfigService } from '../../services/table-config.service';\r\n@Component({\r\n  selector: 'spa-detailsDialog-lite', //uniq to lite\r\n  templateUrl: './detailsDialog-lite.component.html',\r\n  styleUrls: ['./table-lite.component.css']\r\n})\r\nexport class DetailsDialogLite implements OnInit {\r\n\r\n  constructor(private breakpointObserver: BreakpointObserver,private loaderService: LoaderService,public dataService: DataServiceLib,private messageService: MessageService,\r\n    private dialogRef: MatDialogRef<DetailsDialogLite>, @Inject(MAT_DIALOG_DATA) public detailsConfig: DetailsDialogConfig, private buttonService: ButtonService,\r\n    private dialogService: DialogService, private authService: AuthService, public tableConfigService: TableConfigService)\r\n  {\r\n    // detect screen size changes\r\n    this.breakpointObserver.observe([\"(max-width: 600px)\"]).subscribe((result: BreakpointState) => {\r\n\r\n      if (result.matches) {\r\n        this.smallScreen = true\r\n      }else {\r\n        this.smallScreen = false\r\n      }\r\n\r\n    });\r\n  }\r\n\r\n  ngOnInit() {\r\n\r\n    const { tableConfig, tableConfigs, formConfig, stepConfig, buttons } = DetailsDialogProcessor.initializeConfigs(this.detailsConfig);\r\n    this.tableConfig = tableConfig;\r\n    this.tableConfigs = tableConfigs;\r\n    this.formConfig = formConfig;\r\n    this.stepConfig = stepConfig;\r\n    this.buttons = buttons;\r\n\r\n    let resp = DetailsDialogProcessor.handleFormMode(this.detailsConfig, this.formConfig,);\r\n\r\n    if (resp.loadByAction) {\r\n      this.loadByAction = true;\r\n      formConfig.loadAction = resp.action;\r\n      this.loadData(resp.action, false);\r\n    } else {\r\n      this.details = resp.details;\r\n      DetailsDialogProcessor.setHeroValue(this.detailsConfig, this.details);\r\n      this.loadByAction = false;\r\n      this.isLoadComplete = true;\r\n    }\r\n\r\n    this.fileField = DetailsDialogProcessor.setFileField(this.formConfig);\r\n\r\n    this.loaderService.isLoading.subscribe(x => this.isProcessing = x);\r\n\r\n    const { extraButtons, createButton, editButton, deleteButton } = DetailsDialogProcessor.setupButtons(this.formConfig, this.buttons);\r\n    this.extraButtons = extraButtons;\r\n    this.createButton = createButton;\r\n    this.editButton = editButton;\r\n    this.deleteButton = deleteButton;\r\n\r\n    this.setTitleAction()\r\n\r\n  }\r\n\r\n\r\n  titleAction = \"View\";\r\n  loadByAction = false;\r\n\r\n  extraButtons: Button[];\r\n  buttons: Button[];\r\n  createButton: Button;\r\n  editButton: Button;\r\n  deleteButton: Button;\r\n\r\n  // modeButton : Button\r\n  fileField: Field;\r\n  files: any[] = [];\r\n\r\n  tableReload: Subject<boolean> = new Subject();\r\n\r\n  tableConfig: TableConfig\r\n  tableConfigs : TableConfig[]\r\n  details: any\r\n  formConfig: FormConfig;\r\n  stepConfig: StepConfig;\r\n  smallScreen = false;\r\n\r\n  isLoadComplete: boolean = false;\r\n  isProcessing: boolean = false;\r\n\r\n  @Output() inputChange = new EventEmitter();\r\n\r\n\r\n  loadData(action: Action, reload) {\r\n\r\n    if (action && this.loadByAction) {\r\n\r\n      this.dataService.CallApi(action).subscribe((apiResponse: ApiResponse) => {\r\n\r\n        if (apiResponse.success) {\r\n          this.details = apiResponse.data\r\n          this.isLoadComplete = true;\r\n          // DetailsDialogProcessor.setHeroValue(this.detailsConfig, this.details);\r\n          if (reload) this.tableReload.next(true);\r\n        }\r\n        else {\r\n          this.messageService.toast(\"Error: \" + apiResponse.message);\r\n        }\r\n\r\n      });\r\n    }\r\n\r\n  }\r\n\r\n  inputChanged(event) {\r\n    this.inputChange.emit({ field: event.field, value: event.value })\r\n  }\r\n\r\n\r\n  setMode(newMode) {\r\n    this.formConfig.mode = newMode\r\n    this.setTitleAction()\r\n  }\r\n\r\n  setTitleAction() {\r\n    if (this.formConfig.fixedTitle) {\r\n      this.titleAction = \"\";\r\n      return;\r\n    }\r\n\r\n    this.titleAction = this.buttonService.getTitleAction(this.formConfig.mode, this.buttons);\r\n  }\r\n\r\n  getButton(name: string) {\r\n    return this.buttonService.getButton(this.buttons, name);\r\n  }\r\n\r\n  testDisabled(row, buttonName: string): boolean {\r\n    const button = this.getButton(buttonName);\r\n\r\n    if (!button) return false;\r\n\r\n    // Create a copy of the button with component's detailsConfig\r\n    const buttonWithConfig = {\r\n      ...button,\r\n      detailsConfig: button.detailsConfig ? { ...button.detailsConfig } : this.detailsConfig\r\n    };\r\n\r\n    return this.buttonService.testDisabled(buttonWithConfig, row);\r\n  }\r\n\r\n\r\n  testVisible(row, buttonName: string): boolean {\r\n    const button = this.getButton(buttonName);\r\n    if (!button) return false;\r\n\r\n    // Create a copy of the button with component's detailsConfig\r\n    const buttonWithConfig = {\r\n      ...button,\r\n      detailsConfig: button.detailsConfig ? { ...button.detailsConfig } : this.detailsConfig\r\n    };\r\n\r\n    return this.buttonService.testVisible(buttonWithConfig, row)\r\n  }\r\n\r\n  testVisibleTab(tblConfig: TableConfig): boolean {\r\n\r\n    if (!this.tableConfigService.hasTableAccess(tblConfig)) {\r\n      return false;\r\n    }\r\n\r\n    return this.buttonService.isTabVisible(tblConfig);\r\n  }\r\n\r\n  getButtonColor(button: Button, row) {\r\n    return this.buttonService.getButtonColor(button, row);\r\n  }\r\n\r\n  getVisibleFields() {\r\n    return this.formConfig.fields?.filter(x => Core.testVisible(this.formConfig, this.details, x, this.authService.currentRoleSource.value))\r\n  }\r\n\r\n\r\n  create() {\r\n    this.handleButtonAction('create');\r\n  }\r\n\r\n  edit() {\r\n    this.handleButtonAction('edit');\r\n  }\r\n\r\n  delete() {\r\n    this.messageService.confirm(`DELETE ?`).subscribe((result) => {\r\n      if (result == \"yes\") {\r\n        this.handleButtonAction('delete');\r\n      }\r\n    });\r\n  }\r\n\r\n\r\n  custom(button: Button) {\r\n    if (button.dialog && button.detailsConfig) {\r\n      this.openNestedDetailsDialog(button);\r\n    } else {\r\n      this.handleButtonAction(button.name);\r\n    }\r\n  }\r\n\r\n  private openNestedDetailsDialog(button: Button): void {\r\n\r\n    this.dialogService.openConfiguredDetailsDialog(button, this.details, DetailsDialogLite).subscribe(result => {\r\n\r\n      if (result.action === 'inputChange') {\r\n        this.inputChanged(result.change);\r\n      } else if (result && result.message === 'success') {\r\n        this.refreshData();\r\n      }\r\n\r\n    });\r\n  }\r\n\r\n  private refreshData(): void {\r\n    if (this.formConfig.loadAction) {\r\n      this.loadData(this.formConfig.loadAction, true);\r\n    }\r\n  }\r\n\r\n\r\n  handleButtonAction(buttonName: string) {\r\n\r\n    const button = this.getButton(buttonName);\r\n    if (!button) return;\r\n\r\n    if (!button.action) {\r\n      this.dialogRef.close({ message: 'emit', data: this.details });\r\n      return;\r\n    }\r\n\r\n    if (this.validateForm()) {\r\n      this.executeAction(button, this.details);\r\n    }\r\n  }\r\n\r\n  private validateForm(): boolean {\r\n    let validationResult = Core.validateObject(this.getVisibleFields(), this.details);\r\n    if (validationResult !== '') {\r\n      this.messageService.toast(validationResult);\r\n      return false;\r\n    }\r\n\r\n    if (this.files.length < 1 && this.fileField?.required && this.formConfig.mode === 'create') {\r\n      this.messageService.toast(\"Please attach file(s)\");\r\n      return false;\r\n    }\r\n\r\n    // Validate composite fields\r\n    for (const field of this.getVisibleFields()) {\r\n      if (field.type === 'composite') {\r\n        validationResult = Core.validateObject(Core.getVisibleSubfields(this.formConfig, this.details, field, this.authService.currentRoleSource.value), this.details);\r\n        if (validationResult !== '') {\r\n          this.messageService.toast(validationResult);\r\n          return false;\r\n        }\r\n      }\r\n    }\r\n\r\n    return true;\r\n  }\r\n\r\n  private prepareActionData(button: Button, data: any): any {\r\n    if (button.action?.isFormData) {\r\n      const formData = new FormData();\r\n      for (let i = 0; i < this.files.length; i++) {\r\n        formData.append(`uploadFile${i}`, this.files[i], this.files[i].name);\r\n      }\r\n      formData.append(\"data\", JSON.stringify(data));\r\n      return formData;\r\n    }\r\n    return data;\r\n  }\r\n\r\n  private executeAction(button: Button, data: any) {\r\n    const actionData = this.prepareActionData(button, data);\r\n\r\n    if (button.confirm) {\r\n      this.messageService.confirm(button.confirm.message).subscribe((result) => {\r\n        if (result === \"yes\") {\r\n          this.performApiCall(button, actionData);\r\n        }\r\n      });\r\n    } else {\r\n      this.performApiCall(button, actionData);\r\n    }\r\n  }\r\n\r\n  private performApiCall(button: Button, data: any) {\r\n    this.dataService.CallApi(button.action, data).subscribe((apiResponse: ApiResponse) => {\r\n\r\n      if (apiResponse.success) {\r\n\r\n        if (apiResponse.message.includes(\"approval\") || apiResponse.message.includes(\"approve\") || apiResponse.message.includes(\"decline\")) {\r\n          this.messageService.toast(apiResponse.message);\r\n        } else if (button.action.successMessage) {\r\n          this.messageService.toast(button.action.successMessage);\r\n        } else if (apiResponse.message != \"success\" && apiResponse.message != \"\") {\r\n          this.messageService.toast(apiResponse.message);\r\n        } else {\r\n          this.messageService.toast(\"Updated\");\r\n        }\r\n\r\n        this.dialogRef.close({ message: 'success', data: { ...apiResponse, data: this.details } });\r\n      } else {\r\n        this.messageService.toast(\"Error: \" + apiResponse.message);\r\n      }\r\n    });\r\n  }\r\n\r\n\r\n};\r\n\r\n\r\n\r\n","<div class=\"dialog-container\">\r\n\r\n  <div class=\"dialog-content\">\r\n\r\n    <mat-progress-bar mode=\"indeterminate\" *ngIf=\"isProcessing && dataService.appConfig.progressLine\"></mat-progress-bar>\r\n\r\n    <div class=\"d-flex justify-content-between align-items-center mt-0\">\r\n\r\n      <div>\r\n        <label style=\"font-size: 20px; font-weight:500;margin-bottom: 10px;\">{{titleAction | titlecase}} {{formConfig?.title}}</label>\r\n      </div>\r\n\r\n      <div class=\"d-flex justify-content-between align-items-center\">\r\n\r\n        <div *ngIf=\"formConfig.mode=='view' && editButton && testVisible(details,editButton.name)\">\r\n          <button mat-icon-button matTooltipPosition=\"above\" matTooltip=\"Edit\" color=\"primary\" (click)=\"setMode('edit')\" [disabled]=\"testDisabled(details,editButton.name)\"><mat-icon>edit</mat-icon></button>\r\n        </div>\r\n\r\n        <button [disabled]=\"isProcessing\" *ngIf=\"loadByAction\" mat-icon-button matTooltipPosition=\"above\" matTooltip=\"Refresh\" color=\"primary\" (click)=\"loadData(formConfig.loadAction, true)\"><mat-icon class=\"refreshIcon\">cached</mat-icon></button>\r\n      </div>\r\n\r\n    </div>\r\n\r\n    <mat-dialog-content class=\"mat-typography dialog-scroll-content\">\r\n\r\n          <spa-alert [alertMessages]=\"formConfig.alertMessages\" [data]=\"details\"></spa-alert>\r\n\r\n      <div class=\"tin-input\" style=\"font-size:14px\">\r\n\r\n         <p *ngIf=\"formConfig && !details\"><em>Loading...</em></p>\r\n\r\n        <spa-form *ngIf=\"formConfig && details\" [files]=\"files\" [data]=\"details\" [config]=\"formConfig\" (inputChange)=\"inputChanged($event)\">\r\n          <ng-template #dynamicSelect let-field=\"field\" let-data=\"data\" let-testReadOnly=\"testReadOnly\" let-selectChanged=\"selectChanged\">\r\n            <spa-select-lite\r\n              [display]=\"field.alias ?? field.name | camelToWords\"\r\n              [width]=\"field.width\"\r\n              [nullable]=\"field.nullable\"\r\n              [options]=\"field.options\"\r\n              [masterOptions]=\"field.masterOptions\"\r\n              [masterField]=\"field.masterField\"\r\n              [optionDisplay]=\"field.optionDisplay ?? 'name'\"\r\n              [optionValue]=\"field.optionValue ?? 'value'\"\r\n              [(value)]=\"data[field.name]\"\r\n              [defaultFirstValue]=\"field.defaultFirstValue\"\r\n              [readonly]=\"testReadOnly(field)\"\r\n              [hint]=\"field.hint\"\r\n              [loadAction]=\"field.loadAction\"\r\n              [field]=\"field\"\r\n              [infoMessage]=\"field.infoMessage\"\r\n              [copyContent]=\"field.copyContent\"\r\n              (valueChange)=\"selectChanged(field)\"\r\n            ></spa-select-lite>\r\n          </ng-template>\r\n        </spa-form>\r\n\r\n\r\n        <!-- <mat-tab-group *ngIf=\"tableConfigs\">\r\n\r\n          <ng-container *ngFor=\"let tbl of tableConfigService.getVisibleTableConfigs(tableConfigs)\">\r\n            <mat-tab *ngIf=\"testVisibleTab(tbl)\">\r\n\r\n              <ng-template matTabLabel>\r\n                <span>{{tbl.tabTitle ?? 'Untitled'}}</span>\r\n              </ng-template>\r\n\r\n              <spa-table-internal [config]=\"tbl\" [reload]=\"tableReload\" (dataLoad)=\"loadData(formConfig.loadAction, false)\"></spa-table-internal>\r\n\r\n            </mat-tab>\r\n          </ng-container>\r\n\r\n        </mat-tab-group> -->\r\n\r\n      </div>\r\n\r\n    </mat-dialog-content>\r\n\r\n\r\n  </div>\r\n\r\n  <mat-dialog-actions>\r\n\r\n    <div>\r\n\r\n      <button mat-raised-button [disabled]=\"isProcessing\" *ngIf=\"formConfig.mode=='create' && createButton\" color=\"primary\" (click)=\"create()\" cdkFocusInitial>{{createButton.display ?? 'Submit'}}\r\n      </button>\r\n\r\n      <button mat-raised-button [disabled]=\"isProcessing\" *ngIf=\"formConfig.mode=='edit' && editButton\" color=\"primary\" (click)=\"edit()\" cdkFocusInitial>{{editButton.display ?? 'Submit'}}\r\n      </button>\r\n\r\n      <ng-container *ngFor=\"let btn of extraButtons\">\r\n        <button *ngIf=\"testVisible(details,btn.name)\" mat-stroked-button [disabled]=\"isProcessing || testDisabled(details,btn.name)\" [ngStyle]=\"{'color': getButtonColor(btn, details)}\" (click)=\"custom(btn)\" cdkFocusInitial><mat-icon [ngStyle]=\"{'color': getButtonColor(btn, details)}\">{{btn.icon.name}}</mat-icon>{{btn.display ?? btn.name | titlecase}}\r\n        </button>\r\n      </ng-container>\r\n\r\n      <button mat-stroked-button color=\"primary\" mat-dialog-close>Cancel</button>\r\n\r\n    </div>\r\n\r\n    <div class=\"col d-flex justify-content-end\" *ngIf=\"smallScreen\">\r\n      <button mat-icon-button matTooltipPosition=\"above\" matTooltip=\"Delete\" [disabled]=\"isProcessing\" style=\"color: red;\" (click)=\"delete()\" *ngIf=\"formConfig.mode!='create'  && deleteButton\"><mat-icon>delete</mat-icon></button>\r\n    </div>\r\n\r\n\r\n  </mat-dialog-actions>\r\n\r\n\r\n</div>\r\n"]}
@@ -67,6 +67,7 @@ export class RolesComponent {
67
67
  this.loadRoles();
68
68
  }
69
69
  onCapItemChange(capItem, checked, role) {
70
+ return; //disabled to allow sub component access without granting whole menu access
70
71
  if (!checked && capItem.capSubItems) {
71
72
  // When parent is unchecked, set all subitems to Restricted
72
73
  capItem.capSubItems.forEach(subItem => {
@@ -74,6 +75,21 @@ export class RolesComponent {
74
75
  });
75
76
  }
76
77
  }
78
+ hasSubItemsAccess(capItem, role) {
79
+ if (!capItem.capSubItems)
80
+ return false;
81
+ // Check if any sub-item has access
82
+ return capItem.capSubItems.some(subItem => {
83
+ // Check direct sub-items
84
+ if (role[subItem.name] > 0)
85
+ return true;
86
+ // Check sub-sub-items recursively
87
+ if (subItem.capSubItems) {
88
+ return subItem.capSubItems.some(subSubItem => role[subSubItem.name] > 0);
89
+ }
90
+ return false;
91
+ });
92
+ }
77
93
  addRole() {
78
94
  const dialogRef = this.dialog.open(addRoleDialog, {
79
95
  width: "1100px",
@@ -128,9 +144,9 @@ export class RolesComponent {
128
144
  }
129
145
  }
130
146
  RolesComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: RolesComponent, deps: [{ token: i1.HttpService }, { token: i2.Router }, { token: i3.AuthService }, { token: i4.DataServiceLib }, { token: i5.DialogService }, { token: i6.MatDialog }, { token: i7.MessageService }], target: i0.ɵɵFactoryTarget.Component });
131
- RolesComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: RolesComponent, selector: "spa-roles", ngImport: i0, template: "<h4> Roles </h4>\r\n<hr />\r\n\r\n<div class=\"container-fluid mb-5\">\r\n\r\n <div class=\"d-flex justify-content-between mb-2\">\r\n\r\n <div >\r\n <button id=\"btnNewRole\" mat-raised-button color=\"primary\" (click)=\"addRole()\">New Role</button>\r\n </div>\r\n\r\n <div class=\"d-flex justify-content-end\">\r\n <button id=\"btnRefresh\" mat-mini-fab color=\"primary\" (click)=\"refresh()\" matTooltip=\"refresh data\" matTooltipPosition=\"right\"><mat-icon class=\"refreshIcon\">refresh</mat-icon></button>\r\n </div>\r\n\r\n </div>\r\n\r\n\r\n <div class=\"row mt-2 mb-1\" *ngFor=\"let role of roles\">\r\n\r\n <mat-card class=\"mat-elevation-z8\" style=\"width:100%\">\r\n\r\n <div class=\"d-flex justify-content-between align-items-center\">\r\n\r\n <label style=\"font-size: 16px;\">{{role.roleName}}</label>\r\n\r\n <button mat-icon-button color=\"primary\" matTooltip=\"Rename Role\" (click)=\"renameRole(role)\">\r\n <mat-icon>edit</mat-icon>\r\n </button>\r\n </div>\r\n\r\n <hr style=\"margin-top: 0px;\">\r\n\r\n <div class=\"tin-row\" style=\" font-size:12px;\">\r\n\r\n\r\n <div class=\"tin-row\" *ngFor=\"let capItem of appConfig.capItems\">\r\n\r\n <mat-checkbox *ngIf=\"capItem.isBool || capItem.capSubItems\"\r\n color=\"primary\" style=\"min-width: 100px;\" [(ngModel)]=\"role[capItem.name]\" (ngModelChange)=\"onCapItemChange(capItem, $event, role)\">\r\n {{capItem.display}}\r\n </mat-checkbox>\r\n\r\n <spa-select\r\n *ngIf=\"!capItem.isBool && !capItem.capSubItems\"\r\n [options]=\"roleAccessOptions\"\r\n optionDisplay=\"name\"\r\n optionValue=\"value\"\r\n [display]=\"capItem.display\"\r\n [(value)]=\"role[capItem.name]\"\r\n width=\"100px\" style=\"font-size: 12px;\">\r\n </spa-select>\r\n\r\n <ng-container *ngIf=\"capItem.capSubItems && role[capItem.name]\">\r\n\r\n <div class=\"tin-row\" *ngFor=\"let capSubItem of capItem.capSubItems\">\r\n\r\n <mat-checkbox *ngIf=\"capSubItem.isBool\"\r\n color=\"primary\" style=\"min-width: 100px;\" [(ngModel)]=\"role[capSubItem.name]\">\r\n {{capSubItem.display}}\r\n </mat-checkbox>\r\n\r\n <spa-select\r\n *ngIf=\"!capSubItem.isBool\"\r\n [options]=\"roleAccessOptions\"\r\n optionDisplay=\"name\"\r\n optionValue=\"value\"\r\n [display]=\"capSubItem.display\"\r\n [(value)]=\"role[capSubItem.name]\"\r\n width=\"100px\" style=\"font-size: 12px;\">\r\n </spa-select>\r\n\r\n\r\n </div>\r\n\r\n </ng-container>\r\n\r\n\r\n\r\n </div>\r\n\r\n </div>\r\n\r\n\r\n <mat-card-actions>\r\n\r\n <button mat-mini-fab color=\"primary\" (click)=\"updateRole(role)\" style=\"margin-right:10px;\">\r\n <mat-icon>done_all</mat-icon>\r\n </button>\r\n\r\n <button mat-mini-fab color=\"warn\" (click)=\"deleteRole(role)\" style=\"margin-right:10px\">\r\n <mat-icon>delete</mat-icon>\r\n </button>\r\n\r\n </mat-card-actions>\r\n\r\n </mat-card>\r\n\r\n </div>\r\n\r\n\r\n</div>\r\n\r\n", styles: [".mat-mini-fab{width:32px;height:32px}.mat-mini-fab mat-icon{font-size:16px;margin-top:-3px}.refreshIcon{font-size:22px!important;margin-top:-7px!important}\n"], dependencies: [{ kind: "directive", type: i8.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i9.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i9.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i8.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "component", type: i10.MatCheckbox, selector: "mat-checkbox", inputs: ["disableRipple", "color", "tabIndex"], exportAs: ["matCheckbox"] }, { kind: "component", type: i11.MatButton, selector: "button[mat-button], button[mat-raised-button], button[mat-icon-button], button[mat-fab], button[mat-mini-fab], button[mat-stroked-button], button[mat-flat-button]", inputs: ["disabled", "disableRipple", "color"], exportAs: ["matButton"] }, { kind: "component", type: i12.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "component", type: i13.MatCard, selector: "mat-card", exportAs: ["matCard"] }, { kind: "directive", type: i13.MatCardActions, selector: "mat-card-actions", inputs: ["align"], exportAs: ["matCardActions"] }, { kind: "directive", type: i14.MatTooltip, selector: "[matTooltip]", exportAs: ["matTooltip"] }, { kind: "component", type: i15.SelectComponent, selector: "spa-select", inputs: ["detailsConfig"] }] });
147
+ RolesComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: RolesComponent, selector: "spa-roles", ngImport: i0, template: "<h4> Roles </h4>\r\n<hr />\r\n\r\n<div class=\"container-fluid mb-5\">\r\n\r\n <div class=\"d-flex justify-content-between mb-2\">\r\n\r\n <div >\r\n <button id=\"btnNewRole\" mat-raised-button color=\"primary\" (click)=\"addRole()\">New Role</button>\r\n </div>\r\n\r\n <div class=\"d-flex justify-content-end\">\r\n <button id=\"btnRefresh\" mat-mini-fab color=\"primary\" (click)=\"refresh()\" matTooltip=\"refresh data\" matTooltipPosition=\"right\"><mat-icon class=\"refreshIcon\">refresh</mat-icon></button>\r\n </div>\r\n\r\n </div>\r\n\r\n\r\n <div class=\"row mt-2 mb-1\" *ngFor=\"let role of roles\">\r\n\r\n <mat-card class=\"mat-elevation-z8\" style=\"width:100%\">\r\n\r\n <div class=\"d-flex justify-content-between align-items-center\">\r\n\r\n <label style=\"font-size: 16px;\">{{role.roleName}}</label>\r\n\r\n <button mat-icon-button color=\"primary\" matTooltip=\"Rename Role\" (click)=\"renameRole(role)\">\r\n <mat-icon>edit</mat-icon>\r\n </button>\r\n </div>\r\n\r\n <hr style=\"margin-top: 0px;\">\r\n\r\n <div class=\"tin-row\" style=\" font-size:12px;\">\r\n\r\n\r\n <div class=\"tin-row\" *ngFor=\"let capItem of appConfig.capItems\">\r\n\r\n <mat-checkbox *ngIf=\"capItem.isBool || capItem.capSubItems\"\r\n color=\"primary\" style=\"min-width: 100px;\" [(ngModel)]=\"role[capItem.name]\" (ngModelChange)=\"onCapItemChange(capItem, $event, role)\">\r\n {{capItem.display}}\r\n <span *ngIf=\"!role[capItem.name] && hasSubItemsAccess(capItem, role)\" class=\"asterisk\" style=\"color: red;\">*</span>\r\n </mat-checkbox>\r\n\r\n <spa-select\r\n *ngIf=\"!capItem.isBool && !capItem.capSubItems\"\r\n [options]=\"roleAccessOptions\"\r\n optionDisplay=\"name\"\r\n optionValue=\"value\"\r\n [display]=\"capItem.display\"\r\n [(value)]=\"role[capItem.name]\"\r\n width=\"100px\" style=\"font-size: 12px;\">\r\n </spa-select>\r\n\r\n <ng-container *ngIf=\"capItem.capSubItems && role[capItem.name]\">\r\n\r\n <div class=\"tin-row\" *ngFor=\"let capSubItem of capItem.capSubItems\">\r\n\r\n <mat-checkbox *ngIf=\"capSubItem.isBool\"\r\n color=\"primary\" style=\"min-width: 100px;\" [(ngModel)]=\"role[capSubItem.name]\">\r\n {{capSubItem.display}}\r\n </mat-checkbox>\r\n\r\n <spa-select\r\n *ngIf=\"!capSubItem.isBool\"\r\n [options]=\"roleAccessOptions\"\r\n optionDisplay=\"name\"\r\n optionValue=\"value\"\r\n [display]=\"capSubItem.display\"\r\n [(value)]=\"role[capSubItem.name]\"\r\n width=\"100px\" style=\"font-size: 12px;\">\r\n </spa-select>\r\n\r\n <ng-container *ngIf=\"capSubItem.capSubItems\">\r\n\r\n <div class=\"tin-row\" *ngFor=\"let capSubSubItem of capSubItem.capSubItems\">\r\n\r\n <mat-checkbox *ngIf=\"capSubSubItem.isBool\"\r\n color=\"primary\" style=\"min-width: 100px;\" [(ngModel)]=\"role[capSubSubItem.name]\">\r\n {{capSubSubItem.display}}\r\n </mat-checkbox>\r\n\r\n <spa-select\r\n *ngIf=\"!capSubSubItem.isBool\"\r\n [options]=\"roleAccessOptions\"\r\n optionDisplay=\"name\"\r\n optionValue=\"value\"\r\n [display]=\"capSubSubItem.display\"\r\n [(value)]=\"role[capSubSubItem.name]\"\r\n width=\"100px\" style=\"font-size: 12px;\">\r\n </spa-select>\r\n\r\n </div>\r\n\r\n </ng-container>\r\n\r\n\r\n\r\n </div>\r\n\r\n </ng-container>\r\n\r\n </div>\r\n\r\n </div>\r\n\r\n\r\n <mat-card-actions>\r\n\r\n <button mat-mini-fab color=\"primary\" (click)=\"updateRole(role)\" style=\"margin-right:10px;\">\r\n <mat-icon>done_all</mat-icon>\r\n </button>\r\n\r\n <button mat-mini-fab color=\"warn\" (click)=\"deleteRole(role)\" style=\"margin-right:10px\">\r\n <mat-icon>delete</mat-icon>\r\n </button>\r\n\r\n </mat-card-actions>\r\n\r\n </mat-card>\r\n\r\n </div>\r\n\r\n\r\n</div>\r\n\r\n", styles: [".mat-mini-fab{width:32px;height:32px}.mat-mini-fab mat-icon{font-size:16px;margin-top:-3px}.refreshIcon{font-size:22px!important;margin-top:-7px!important}\n"], dependencies: [{ kind: "directive", type: i8.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i9.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i9.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i8.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "component", type: i10.MatCheckbox, selector: "mat-checkbox", inputs: ["disableRipple", "color", "tabIndex"], exportAs: ["matCheckbox"] }, { kind: "component", type: i11.MatButton, selector: "button[mat-button], button[mat-raised-button], button[mat-icon-button], button[mat-fab], button[mat-mini-fab], button[mat-stroked-button], button[mat-flat-button]", inputs: ["disabled", "disableRipple", "color"], exportAs: ["matButton"] }, { kind: "component", type: i12.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "component", type: i13.MatCard, selector: "mat-card", exportAs: ["matCard"] }, { kind: "directive", type: i13.MatCardActions, selector: "mat-card-actions", inputs: ["align"], exportAs: ["matCardActions"] }, { kind: "directive", type: i14.MatTooltip, selector: "[matTooltip]", exportAs: ["matTooltip"] }, { kind: "component", type: i15.SelectComponent, selector: "spa-select", inputs: ["detailsConfig"] }] });
132
148
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: RolesComponent, decorators: [{
133
149
  type: Component,
134
- args: [{ selector: "spa-roles", template: "<h4> Roles </h4>\r\n<hr />\r\n\r\n<div class=\"container-fluid mb-5\">\r\n\r\n <div class=\"d-flex justify-content-between mb-2\">\r\n\r\n <div >\r\n <button id=\"btnNewRole\" mat-raised-button color=\"primary\" (click)=\"addRole()\">New Role</button>\r\n </div>\r\n\r\n <div class=\"d-flex justify-content-end\">\r\n <button id=\"btnRefresh\" mat-mini-fab color=\"primary\" (click)=\"refresh()\" matTooltip=\"refresh data\" matTooltipPosition=\"right\"><mat-icon class=\"refreshIcon\">refresh</mat-icon></button>\r\n </div>\r\n\r\n </div>\r\n\r\n\r\n <div class=\"row mt-2 mb-1\" *ngFor=\"let role of roles\">\r\n\r\n <mat-card class=\"mat-elevation-z8\" style=\"width:100%\">\r\n\r\n <div class=\"d-flex justify-content-between align-items-center\">\r\n\r\n <label style=\"font-size: 16px;\">{{role.roleName}}</label>\r\n\r\n <button mat-icon-button color=\"primary\" matTooltip=\"Rename Role\" (click)=\"renameRole(role)\">\r\n <mat-icon>edit</mat-icon>\r\n </button>\r\n </div>\r\n\r\n <hr style=\"margin-top: 0px;\">\r\n\r\n <div class=\"tin-row\" style=\" font-size:12px;\">\r\n\r\n\r\n <div class=\"tin-row\" *ngFor=\"let capItem of appConfig.capItems\">\r\n\r\n <mat-checkbox *ngIf=\"capItem.isBool || capItem.capSubItems\"\r\n color=\"primary\" style=\"min-width: 100px;\" [(ngModel)]=\"role[capItem.name]\" (ngModelChange)=\"onCapItemChange(capItem, $event, role)\">\r\n {{capItem.display}}\r\n </mat-checkbox>\r\n\r\n <spa-select\r\n *ngIf=\"!capItem.isBool && !capItem.capSubItems\"\r\n [options]=\"roleAccessOptions\"\r\n optionDisplay=\"name\"\r\n optionValue=\"value\"\r\n [display]=\"capItem.display\"\r\n [(value)]=\"role[capItem.name]\"\r\n width=\"100px\" style=\"font-size: 12px;\">\r\n </spa-select>\r\n\r\n <ng-container *ngIf=\"capItem.capSubItems && role[capItem.name]\">\r\n\r\n <div class=\"tin-row\" *ngFor=\"let capSubItem of capItem.capSubItems\">\r\n\r\n <mat-checkbox *ngIf=\"capSubItem.isBool\"\r\n color=\"primary\" style=\"min-width: 100px;\" [(ngModel)]=\"role[capSubItem.name]\">\r\n {{capSubItem.display}}\r\n </mat-checkbox>\r\n\r\n <spa-select\r\n *ngIf=\"!capSubItem.isBool\"\r\n [options]=\"roleAccessOptions\"\r\n optionDisplay=\"name\"\r\n optionValue=\"value\"\r\n [display]=\"capSubItem.display\"\r\n [(value)]=\"role[capSubItem.name]\"\r\n width=\"100px\" style=\"font-size: 12px;\">\r\n </spa-select>\r\n\r\n\r\n </div>\r\n\r\n </ng-container>\r\n\r\n\r\n\r\n </div>\r\n\r\n </div>\r\n\r\n\r\n <mat-card-actions>\r\n\r\n <button mat-mini-fab color=\"primary\" (click)=\"updateRole(role)\" style=\"margin-right:10px;\">\r\n <mat-icon>done_all</mat-icon>\r\n </button>\r\n\r\n <button mat-mini-fab color=\"warn\" (click)=\"deleteRole(role)\" style=\"margin-right:10px\">\r\n <mat-icon>delete</mat-icon>\r\n </button>\r\n\r\n </mat-card-actions>\r\n\r\n </mat-card>\r\n\r\n </div>\r\n\r\n\r\n</div>\r\n\r\n", styles: [".mat-mini-fab{width:32px;height:32px}.mat-mini-fab mat-icon{font-size:16px;margin-top:-3px}.refreshIcon{font-size:22px!important;margin-top:-7px!important}\n"] }]
150
+ args: [{ selector: "spa-roles", template: "<h4> Roles </h4>\r\n<hr />\r\n\r\n<div class=\"container-fluid mb-5\">\r\n\r\n <div class=\"d-flex justify-content-between mb-2\">\r\n\r\n <div >\r\n <button id=\"btnNewRole\" mat-raised-button color=\"primary\" (click)=\"addRole()\">New Role</button>\r\n </div>\r\n\r\n <div class=\"d-flex justify-content-end\">\r\n <button id=\"btnRefresh\" mat-mini-fab color=\"primary\" (click)=\"refresh()\" matTooltip=\"refresh data\" matTooltipPosition=\"right\"><mat-icon class=\"refreshIcon\">refresh</mat-icon></button>\r\n </div>\r\n\r\n </div>\r\n\r\n\r\n <div class=\"row mt-2 mb-1\" *ngFor=\"let role of roles\">\r\n\r\n <mat-card class=\"mat-elevation-z8\" style=\"width:100%\">\r\n\r\n <div class=\"d-flex justify-content-between align-items-center\">\r\n\r\n <label style=\"font-size: 16px;\">{{role.roleName}}</label>\r\n\r\n <button mat-icon-button color=\"primary\" matTooltip=\"Rename Role\" (click)=\"renameRole(role)\">\r\n <mat-icon>edit</mat-icon>\r\n </button>\r\n </div>\r\n\r\n <hr style=\"margin-top: 0px;\">\r\n\r\n <div class=\"tin-row\" style=\" font-size:12px;\">\r\n\r\n\r\n <div class=\"tin-row\" *ngFor=\"let capItem of appConfig.capItems\">\r\n\r\n <mat-checkbox *ngIf=\"capItem.isBool || capItem.capSubItems\"\r\n color=\"primary\" style=\"min-width: 100px;\" [(ngModel)]=\"role[capItem.name]\" (ngModelChange)=\"onCapItemChange(capItem, $event, role)\">\r\n {{capItem.display}}\r\n <span *ngIf=\"!role[capItem.name] && hasSubItemsAccess(capItem, role)\" class=\"asterisk\" style=\"color: red;\">*</span>\r\n </mat-checkbox>\r\n\r\n <spa-select\r\n *ngIf=\"!capItem.isBool && !capItem.capSubItems\"\r\n [options]=\"roleAccessOptions\"\r\n optionDisplay=\"name\"\r\n optionValue=\"value\"\r\n [display]=\"capItem.display\"\r\n [(value)]=\"role[capItem.name]\"\r\n width=\"100px\" style=\"font-size: 12px;\">\r\n </spa-select>\r\n\r\n <ng-container *ngIf=\"capItem.capSubItems && role[capItem.name]\">\r\n\r\n <div class=\"tin-row\" *ngFor=\"let capSubItem of capItem.capSubItems\">\r\n\r\n <mat-checkbox *ngIf=\"capSubItem.isBool\"\r\n color=\"primary\" style=\"min-width: 100px;\" [(ngModel)]=\"role[capSubItem.name]\">\r\n {{capSubItem.display}}\r\n </mat-checkbox>\r\n\r\n <spa-select\r\n *ngIf=\"!capSubItem.isBool\"\r\n [options]=\"roleAccessOptions\"\r\n optionDisplay=\"name\"\r\n optionValue=\"value\"\r\n [display]=\"capSubItem.display\"\r\n [(value)]=\"role[capSubItem.name]\"\r\n width=\"100px\" style=\"font-size: 12px;\">\r\n </spa-select>\r\n\r\n <ng-container *ngIf=\"capSubItem.capSubItems\">\r\n\r\n <div class=\"tin-row\" *ngFor=\"let capSubSubItem of capSubItem.capSubItems\">\r\n\r\n <mat-checkbox *ngIf=\"capSubSubItem.isBool\"\r\n color=\"primary\" style=\"min-width: 100px;\" [(ngModel)]=\"role[capSubSubItem.name]\">\r\n {{capSubSubItem.display}}\r\n </mat-checkbox>\r\n\r\n <spa-select\r\n *ngIf=\"!capSubSubItem.isBool\"\r\n [options]=\"roleAccessOptions\"\r\n optionDisplay=\"name\"\r\n optionValue=\"value\"\r\n [display]=\"capSubSubItem.display\"\r\n [(value)]=\"role[capSubSubItem.name]\"\r\n width=\"100px\" style=\"font-size: 12px;\">\r\n </spa-select>\r\n\r\n </div>\r\n\r\n </ng-container>\r\n\r\n\r\n\r\n </div>\r\n\r\n </ng-container>\r\n\r\n </div>\r\n\r\n </div>\r\n\r\n\r\n <mat-card-actions>\r\n\r\n <button mat-mini-fab color=\"primary\" (click)=\"updateRole(role)\" style=\"margin-right:10px;\">\r\n <mat-icon>done_all</mat-icon>\r\n </button>\r\n\r\n <button mat-mini-fab color=\"warn\" (click)=\"deleteRole(role)\" style=\"margin-right:10px\">\r\n <mat-icon>delete</mat-icon>\r\n </button>\r\n\r\n </mat-card-actions>\r\n\r\n </mat-card>\r\n\r\n </div>\r\n\r\n\r\n</div>\r\n\r\n", styles: [".mat-mini-fab{width:32px;height:32px}.mat-mini-fab mat-icon{font-size:16px;margin-top:-3px}.refreshIcon{font-size:22px!important;margin-top:-7px!important}\n"] }]
135
151
  }], ctorParameters: function () { return [{ type: i1.HttpService }, { type: i2.Router }, { type: i3.AuthService }, { type: i4.DataServiceLib }, { type: i5.DialogService }, { type: i6.MatDialog }, { type: i7.MessageService }]; } });
136
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"roles.component.js","sourceRoot":"","sources":["../../../../../../projects/tin-spa/src/lib/pages/roles/roles.component.ts","../../../../../../projects/tin-spa/src/lib/pages/roles/roles.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ApiResponse, AppConfig, Button, CapItem, DetailsDialogConfig } from './../../classes/Classes';\r\nimport { Component, OnInit, Input } from \"@angular/core\";\r\nimport { MatDialog } from \"@angular/material/dialog\";\r\nimport { DataServiceLib } from \"../../services/datalib.service\";\r\nimport { addRoleDialog } from \"./addRoleDialog.component\";\r\nimport { Router } from \"@angular/router\";\r\nimport { MessageService } from '../../services/message.service';\r\nimport { AuthService } from '../../services/auth.service';\r\nimport { HttpService } from '../../services/http.service';\r\nimport { Role, RoleAccess } from '../../classes/LibClasses';\r\nimport { DetailsDialog } from '../../components/table/detailsDialog.component';\r\nimport { DialogService } from '../../services/dialog.service';\r\n\r\n\r\n@Component({\r\n  selector: \"spa-roles\",\r\n  templateUrl: \"./roles.component.html\",\r\n  styleUrls: [\"./roles.component.css\"],\r\n})\r\nexport class RolesComponent implements OnInit {\r\n\r\n  constructor(private httpService: HttpService, private router: Router, private authService: AuthService, private dataService: DataServiceLib, private dialogService : DialogService, public dialog: MatDialog, private messageService: MessageService) { }\r\n\r\n  ngOnInit() {\r\n    this.authService.isAuthorised(this.dataService.capRoles.name);\r\n    this.loadRoles();\r\n    this.dataService.appConfigObserv.subscribe(x => this.appConfig = x);\r\n  }\r\n\r\n  isProcessing = false;\r\n  roles: Role[];\r\n  RoleAccess = RoleAccess;\r\n\r\n  appConfig = new AppConfig();\r\n\r\n  roleAccessOptions = [\r\n    { value: RoleAccess.None, name: 'None' },\r\n    { value: RoleAccess.View, name: 'View' },\r\n    { value: RoleAccess.Edit, name: 'Edit' },\r\n    { value: RoleAccess.Create, name: 'Create' },\r\n    { value: RoleAccess.Full, name: 'Full' }\r\n  ];\r\n\r\n\r\n\r\n  loadRoles() {\r\n    this.dataService.GetRole(\"all\", \"\").subscribe((apiResponse : ApiResponse) => {\r\n      this.roles = apiResponse.data;\r\n    });\r\n  }\r\n\r\n  refresh() {\r\n    this.loadRoles();\r\n  }\r\n\r\n  onCapItemChange(capItem: CapItem, checked: boolean, role: any) {\r\n    if (!checked && capItem.capSubItems) {\r\n      // When parent is unchecked, set all subitems to Restricted\r\n      capItem.capSubItems.forEach(subItem => {\r\n        role[subItem.name] = RoleAccess.None;\r\n      });\r\n    }\r\n  }\r\n\r\n\r\n\r\n  addRole() {\r\n    const dialogRef = this.dialog.open(addRoleDialog, {\r\n      width: \"1100px\",\r\n      data: this.appConfig.capItems,\r\n    });\r\n\r\n    dialogRef.afterClosed().subscribe((result) => {\r\n      if (result == \"success\") {\r\n        this.loadRoles();\r\n      }\r\n    });\r\n  }\r\n\r\n  updateRole(role: Role) {\r\n\r\n    this.messageService.confirm(`UPDATE ${role.roleName} ?`).subscribe((result) => {\r\n\r\n      if (result == \"yes\") {\r\n\r\n        this.dataService.PostRole(role, \"edit\").subscribe((apiResponse: ApiResponse) => {\r\n\r\n          if (apiResponse.success) {\r\n            this.messageService.toast(\"Role Updated\");\r\n            this.loadRoles();\r\n          } else {\r\n            this.messageService.toast(\"Error: \" + apiResponse);\r\n          }\r\n        });\r\n\r\n      }\r\n\r\n    })\r\n\r\n\r\n  };\r\n\r\n  deleteRole(role: Role) {\r\n\r\n    this.messageService.confirm(`DELETE ${role.roleName} ?`).subscribe((result) => {\r\n\r\n      if (result == \"yes\") {\r\n\r\n        this.dataService.PostRole(role, \"delete\").subscribe((apiResponse: ApiResponse) => {\r\n          if (apiResponse.success) {\r\n            this.messageService.toast(\"Role Deleted\");\r\n            this.loadRoles();\r\n          } else {\r\n            this.messageService.toast(\"Error: \" + apiResponse);\r\n          }\r\n        });\r\n\r\n      }\r\n\r\n    })\r\n\r\n  }\r\n\r\n  renameDialogConfig: DetailsDialogConfig = {\r\n    formConfig: {\r\n      title: 'Rename Role',\r\n      fixedTitle: true,\r\n      fields: [\r\n        { name: 'roleName', type: 'text', required: true, span: true, alias: 'Role Name' }\r\n      ]\r\n    },\r\n    mode: 'edit',\r\n    buttons: [\r\n      { name: 'edit', dialog: true, action: { url: 'role?action=edit', method: 'post', successMessage: 'Role renamed successfully' } }\r\n    ]\r\n  };\r\n\r\n  renameButton : Button = {name : 'edit', dialog : true, detailsConfig : this.renameDialogConfig};\r\n\r\n  renameRole(role: Role) {\r\n    // Create role copy to avoid modifying the original during editing\r\n    const roleData = { ...role };\r\n\r\n    // Open rename dialog\r\n    const dialogRef = this.dialogService.openConfiguredDetailsDialog(\r\n      this.renameButton,\r\n      roleData,\r\n      DetailsDialog\r\n    ).subscribe(result => {\r\n      if (result.message === 'success') {\r\n        this.loadRoles();\r\n      }\r\n    });\r\n  }\r\n\r\n\r\n}\r\n\r\n\r\n","<h4> Roles </h4>\r\n<hr />\r\n\r\n<div class=\"container-fluid mb-5\">\r\n\r\n  <div class=\"d-flex justify-content-between mb-2\">\r\n\r\n    <div >\r\n      <button id=\"btnNewRole\" mat-raised-button color=\"primary\" (click)=\"addRole()\">New Role</button>\r\n    </div>\r\n\r\n    <div class=\"d-flex justify-content-end\">\r\n      <button id=\"btnRefresh\" mat-mini-fab color=\"primary\" (click)=\"refresh()\"  matTooltip=\"refresh data\" matTooltipPosition=\"right\"><mat-icon class=\"refreshIcon\">refresh</mat-icon></button>\r\n    </div>\r\n\r\n  </div>\r\n\r\n\r\n  <div class=\"row mt-2 mb-1\"  *ngFor=\"let role of roles\">\r\n\r\n    <mat-card class=\"mat-elevation-z8\" style=\"width:100%\">\r\n\r\n      <div class=\"d-flex justify-content-between align-items-center\">\r\n\r\n        <label style=\"font-size: 16px;\">{{role.roleName}}</label>\r\n\r\n        <button mat-icon-button color=\"primary\" matTooltip=\"Rename Role\" (click)=\"renameRole(role)\">\r\n          <mat-icon>edit</mat-icon>\r\n        </button>\r\n      </div>\r\n\r\n      <hr style=\"margin-top: 0px;\">\r\n\r\n      <div class=\"tin-row\" style=\" font-size:12px;\">\r\n\r\n\r\n        <div class=\"tin-row\"  *ngFor=\"let capItem of appConfig.capItems\">\r\n\r\n          <mat-checkbox  *ngIf=\"capItem.isBool || capItem.capSubItems\"\r\n            color=\"primary\" style=\"min-width: 100px;\" [(ngModel)]=\"role[capItem.name]\" (ngModelChange)=\"onCapItemChange(capItem, $event, role)\">\r\n            {{capItem.display}}\r\n          </mat-checkbox>\r\n\r\n          <spa-select\r\n           *ngIf=\"!capItem.isBool && !capItem.capSubItems\"\r\n            [options]=\"roleAccessOptions\"\r\n            optionDisplay=\"name\"\r\n            optionValue=\"value\"\r\n            [display]=\"capItem.display\"\r\n            [(value)]=\"role[capItem.name]\"\r\n            width=\"100px\" style=\"font-size: 12px;\">\r\n          </spa-select>\r\n\r\n          <ng-container *ngIf=\"capItem.capSubItems && role[capItem.name]\">\r\n\r\n            <div class=\"tin-row\" *ngFor=\"let capSubItem of capItem.capSubItems\">\r\n\r\n              <mat-checkbox *ngIf=\"capSubItem.isBool\"\r\n                color=\"primary\" style=\"min-width: 100px;\" [(ngModel)]=\"role[capSubItem.name]\">\r\n                {{capSubItem.display}}\r\n              </mat-checkbox>\r\n\r\n              <spa-select\r\n                *ngIf=\"!capSubItem.isBool\"\r\n                [options]=\"roleAccessOptions\"\r\n                optionDisplay=\"name\"\r\n                optionValue=\"value\"\r\n                [display]=\"capSubItem.display\"\r\n                [(value)]=\"role[capSubItem.name]\"\r\n                width=\"100px\" style=\"font-size: 12px;\">\r\n              </spa-select>\r\n\r\n\r\n            </div>\r\n\r\n          </ng-container>\r\n\r\n\r\n\r\n        </div>\r\n\r\n      </div>\r\n\r\n\r\n      <mat-card-actions>\r\n\r\n        <button  mat-mini-fab color=\"primary\" (click)=\"updateRole(role)\" style=\"margin-right:10px;\">\r\n           <mat-icon>done_all</mat-icon>\r\n        </button>\r\n\r\n        <button mat-mini-fab color=\"warn\" (click)=\"deleteRole(role)\" style=\"margin-right:10px\">\r\n          <mat-icon>delete</mat-icon>\r\n        </button>\r\n\r\n      </mat-card-actions>\r\n\r\n    </mat-card>\r\n\r\n  </div>\r\n\r\n\r\n</div>\r\n\r\n"]}
152
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"roles.component.js","sourceRoot":"","sources":["../../../../../../projects/tin-spa/src/lib/pages/roles/roles.component.ts","../../../../../../projects/tin-spa/src/lib/pages/roles/roles.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ApiResponse, AppConfig, Button, CapItem, DetailsDialogConfig } from './../../classes/Classes';\r\nimport { Component, OnInit, Input } from \"@angular/core\";\r\nimport { MatDialog } from \"@angular/material/dialog\";\r\nimport { DataServiceLib } from \"../../services/datalib.service\";\r\nimport { addRoleDialog } from \"./addRoleDialog.component\";\r\nimport { Router } from \"@angular/router\";\r\nimport { MessageService } from '../../services/message.service';\r\nimport { AuthService } from '../../services/auth.service';\r\nimport { HttpService } from '../../services/http.service';\r\nimport { Role, RoleAccess } from '../../classes/LibClasses';\r\nimport { DetailsDialog } from '../../components/table/detailsDialog.component';\r\nimport { DialogService } from '../../services/dialog.service';\r\n\r\n\r\n@Component({\r\n  selector: \"spa-roles\",\r\n  templateUrl: \"./roles.component.html\",\r\n  styleUrls: [\"./roles.component.css\"],\r\n})\r\nexport class RolesComponent implements OnInit {\r\n\r\n  constructor(private httpService: HttpService, private router: Router, private authService: AuthService, private dataService: DataServiceLib, private dialogService : DialogService, public dialog: MatDialog, private messageService: MessageService) { }\r\n\r\n  ngOnInit() {\r\n    this.authService.isAuthorised(this.dataService.capRoles.name);\r\n    this.loadRoles();\r\n    this.dataService.appConfigObserv.subscribe(x => this.appConfig = x);\r\n  }\r\n\r\n  isProcessing = false;\r\n  roles: Role[];\r\n  RoleAccess = RoleAccess;\r\n\r\n  appConfig = new AppConfig();\r\n\r\n  roleAccessOptions = [\r\n    { value: RoleAccess.None, name: 'None' },\r\n    { value: RoleAccess.View, name: 'View' },\r\n    { value: RoleAccess.Edit, name: 'Edit' },\r\n    { value: RoleAccess.Create, name: 'Create' },\r\n    { value: RoleAccess.Full, name: 'Full' }\r\n  ];\r\n\r\n\r\n\r\n  loadRoles() {\r\n    this.dataService.GetRole(\"all\", \"\").subscribe((apiResponse : ApiResponse) => {\r\n      this.roles = apiResponse.data;\r\n    });\r\n  }\r\n\r\n  refresh() {\r\n    this.loadRoles();\r\n  }\r\n\r\n  onCapItemChange(capItem: CapItem, checked: boolean, role: any) {\r\n\r\n    return; //disabled to allow sub component access without granting whole menu access\r\n\r\n    if (!checked && capItem.capSubItems) {\r\n      // When parent is unchecked, set all subitems to Restricted\r\n      capItem.capSubItems.forEach(subItem => {\r\n        role[subItem.name] = RoleAccess.None;\r\n      });\r\n    }\r\n  }\r\n\r\n  hasSubItemsAccess(capItem: CapItem, role: any): boolean {\r\n    if (!capItem.capSubItems) return false;\r\n\r\n    // Check if any sub-item has access\r\n    return capItem.capSubItems.some(subItem => {\r\n      // Check direct sub-items\r\n      if (role[subItem.name] > 0) return true;\r\n\r\n      // Check sub-sub-items recursively\r\n      if (subItem.capSubItems) {\r\n        return subItem.capSubItems.some(subSubItem => role[subSubItem.name] > 0);\r\n      }\r\n      return false;\r\n    });\r\n  }\r\n\r\n\r\n  addRole() {\r\n    const dialogRef = this.dialog.open(addRoleDialog, {\r\n      width: \"1100px\",\r\n      data: this.appConfig.capItems,\r\n    });\r\n\r\n    dialogRef.afterClosed().subscribe((result) => {\r\n      if (result == \"success\") {\r\n        this.loadRoles();\r\n      }\r\n    });\r\n  }\r\n\r\n  updateRole(role: Role) {\r\n\r\n    this.messageService.confirm(`UPDATE ${role.roleName} ?`).subscribe((result) => {\r\n\r\n      if (result == \"yes\") {\r\n\r\n        this.dataService.PostRole(role, \"edit\").subscribe((apiResponse: ApiResponse) => {\r\n\r\n          if (apiResponse.success) {\r\n            this.messageService.toast(\"Role Updated\");\r\n            this.loadRoles();\r\n          } else {\r\n            this.messageService.toast(\"Error: \" + apiResponse);\r\n          }\r\n        });\r\n\r\n      }\r\n\r\n    })\r\n\r\n\r\n  };\r\n\r\n  deleteRole(role: Role) {\r\n\r\n    this.messageService.confirm(`DELETE ${role.roleName} ?`).subscribe((result) => {\r\n\r\n      if (result == \"yes\") {\r\n\r\n        this.dataService.PostRole(role, \"delete\").subscribe((apiResponse: ApiResponse) => {\r\n          if (apiResponse.success) {\r\n            this.messageService.toast(\"Role Deleted\");\r\n            this.loadRoles();\r\n          } else {\r\n            this.messageService.toast(\"Error: \" + apiResponse);\r\n          }\r\n        });\r\n\r\n      }\r\n\r\n    })\r\n\r\n  }\r\n\r\n  renameDialogConfig: DetailsDialogConfig = {\r\n    formConfig: {\r\n      title: 'Rename Role',\r\n      fixedTitle: true,\r\n      fields: [\r\n        { name: 'roleName', type: 'text', required: true, span: true, alias: 'Role Name' }\r\n      ]\r\n    },\r\n    mode: 'edit',\r\n    buttons: [\r\n      { name: 'edit', dialog: true, action: { url: 'role?action=edit', method: 'post', successMessage: 'Role renamed successfully' } }\r\n    ]\r\n  };\r\n\r\n  renameButton : Button = {name : 'edit', dialog : true, detailsConfig : this.renameDialogConfig};\r\n\r\n  renameRole(role: Role) {\r\n    // Create role copy to avoid modifying the original during editing\r\n    const roleData = { ...role };\r\n\r\n    // Open rename dialog\r\n    const dialogRef = this.dialogService.openConfiguredDetailsDialog(\r\n      this.renameButton,\r\n      roleData,\r\n      DetailsDialog\r\n    ).subscribe(result => {\r\n      if (result.message === 'success') {\r\n        this.loadRoles();\r\n      }\r\n    });\r\n  }\r\n\r\n\r\n}\r\n\r\n\r\n","<h4> Roles </h4>\r\n<hr />\r\n\r\n<div class=\"container-fluid mb-5\">\r\n\r\n  <div class=\"d-flex justify-content-between mb-2\">\r\n\r\n    <div >\r\n      <button id=\"btnNewRole\" mat-raised-button color=\"primary\" (click)=\"addRole()\">New Role</button>\r\n    </div>\r\n\r\n    <div class=\"d-flex justify-content-end\">\r\n      <button id=\"btnRefresh\" mat-mini-fab color=\"primary\" (click)=\"refresh()\"  matTooltip=\"refresh data\" matTooltipPosition=\"right\"><mat-icon class=\"refreshIcon\">refresh</mat-icon></button>\r\n    </div>\r\n\r\n  </div>\r\n\r\n\r\n  <div class=\"row mt-2 mb-1\"  *ngFor=\"let role of roles\">\r\n\r\n    <mat-card class=\"mat-elevation-z8\" style=\"width:100%\">\r\n\r\n      <div class=\"d-flex justify-content-between align-items-center\">\r\n\r\n        <label style=\"font-size: 16px;\">{{role.roleName}}</label>\r\n\r\n        <button mat-icon-button color=\"primary\" matTooltip=\"Rename Role\" (click)=\"renameRole(role)\">\r\n          <mat-icon>edit</mat-icon>\r\n        </button>\r\n      </div>\r\n\r\n      <hr style=\"margin-top: 0px;\">\r\n\r\n      <div class=\"tin-row\" style=\" font-size:12px;\">\r\n\r\n\r\n        <div class=\"tin-row\"  *ngFor=\"let capItem of appConfig.capItems\">\r\n\r\n          <mat-checkbox  *ngIf=\"capItem.isBool || capItem.capSubItems\"\r\n            color=\"primary\" style=\"min-width: 100px;\" [(ngModel)]=\"role[capItem.name]\" (ngModelChange)=\"onCapItemChange(capItem, $event, role)\">\r\n            {{capItem.display}}\r\n            <span *ngIf=\"!role[capItem.name] && hasSubItemsAccess(capItem, role)\" class=\"asterisk\" style=\"color: red;\">*</span>\r\n          </mat-checkbox>\r\n\r\n          <spa-select\r\n           *ngIf=\"!capItem.isBool && !capItem.capSubItems\"\r\n            [options]=\"roleAccessOptions\"\r\n            optionDisplay=\"name\"\r\n            optionValue=\"value\"\r\n            [display]=\"capItem.display\"\r\n            [(value)]=\"role[capItem.name]\"\r\n            width=\"100px\" style=\"font-size: 12px;\">\r\n          </spa-select>\r\n\r\n          <ng-container *ngIf=\"capItem.capSubItems && role[capItem.name]\">\r\n\r\n            <div class=\"tin-row\" *ngFor=\"let capSubItem of capItem.capSubItems\">\r\n\r\n              <mat-checkbox *ngIf=\"capSubItem.isBool\"\r\n                color=\"primary\" style=\"min-width: 100px;\" [(ngModel)]=\"role[capSubItem.name]\">\r\n                {{capSubItem.display}}\r\n              </mat-checkbox>\r\n\r\n              <spa-select\r\n                *ngIf=\"!capSubItem.isBool\"\r\n                [options]=\"roleAccessOptions\"\r\n                optionDisplay=\"name\"\r\n                optionValue=\"value\"\r\n                [display]=\"capSubItem.display\"\r\n                [(value)]=\"role[capSubItem.name]\"\r\n                width=\"100px\" style=\"font-size: 12px;\">\r\n              </spa-select>\r\n\r\n              <ng-container *ngIf=\"capSubItem.capSubItems\">\r\n\r\n                <div class=\"tin-row\" *ngFor=\"let capSubSubItem of capSubItem.capSubItems\">\r\n\r\n                  <mat-checkbox *ngIf=\"capSubSubItem.isBool\"\r\n                    color=\"primary\" style=\"min-width: 100px;\" [(ngModel)]=\"role[capSubSubItem.name]\">\r\n                    {{capSubSubItem.display}}\r\n                  </mat-checkbox>\r\n\r\n                  <spa-select\r\n                    *ngIf=\"!capSubSubItem.isBool\"\r\n                    [options]=\"roleAccessOptions\"\r\n                    optionDisplay=\"name\"\r\n                    optionValue=\"value\"\r\n                    [display]=\"capSubSubItem.display\"\r\n                    [(value)]=\"role[capSubSubItem.name]\"\r\n                    width=\"100px\" style=\"font-size: 12px;\">\r\n                  </spa-select>\r\n\r\n                </div>\r\n\r\n              </ng-container>\r\n\r\n\r\n\r\n            </div>\r\n\r\n          </ng-container>\r\n\r\n        </div>\r\n\r\n      </div>\r\n\r\n\r\n      <mat-card-actions>\r\n\r\n        <button  mat-mini-fab color=\"primary\" (click)=\"updateRole(role)\" style=\"margin-right:10px;\">\r\n           <mat-icon>done_all</mat-icon>\r\n        </button>\r\n\r\n        <button mat-mini-fab color=\"warn\" (click)=\"deleteRole(role)\" style=\"margin-right:10px\">\r\n          <mat-icon>delete</mat-icon>\r\n        </button>\r\n\r\n      </mat-card-actions>\r\n\r\n    </mat-card>\r\n\r\n  </div>\r\n\r\n\r\n</div>\r\n\r\n"]}
@@ -32,11 +32,10 @@ export class ButtonService {
32
32
  };
33
33
  return colorMap[button.name] || '#4050B5';
34
34
  }
35
- isButtonVisible(button, row, isLoadComplete) {
36
- if (!isLoadComplete)
37
- return false;
38
- return this.testVisible(button, row);
39
- }
35
+ // isButtonVisible(button: Button, row: any, isLoadComplete: boolean): boolean {
36
+ // if (!isLoadComplete) return false;
37
+ // return this.testVisible(button,row);
38
+ // }
40
39
  getTitleAction(mode, buttons) {
41
40
  const modeMap = {
42
41
  'view': 'View',
@@ -54,11 +53,25 @@ export class ButtonService {
54
53
  }
55
54
  return this.getDefaultIcon(button.name, config.flatButtons);
56
55
  }
57
- testDisabled(button, row) {
56
+ testDisabled(button, row, config) {
57
+ // Check standard locking/approval rules
58
58
  if ((button.name === 'edit' || button.name === 'delete') && row?.locked)
59
59
  return true;
60
60
  if (row?.pendingApproval && !button.ignoreApproval && button.name != 'view')
61
61
  return true;
62
+ // Check form access if detailsConfig exists
63
+ if (button.detailsConfig?.formConfig) {
64
+ const access = Core.getFormAccess(button.detailsConfig.formConfig, this.authService.currentRoleSource.value);
65
+ if (!this.hasRequiredAccess(access, button.name))
66
+ return true;
67
+ }
68
+ // Check form access levels from table config
69
+ if (['create', 'edit', 'view', 'delete'].includes(button.name) && config?.formConfig) {
70
+ const access = Core.getFormAccess(config.formConfig, this.authService.currentRoleSource.value);
71
+ if (!this.hasRequiredAccess(access, button.name))
72
+ return true;
73
+ }
74
+ // Check button-specific disabled rules
62
75
  return button.disabled ? button.disabled(row) : false;
63
76
  }
64
77
  testVisible(button, row, config) {
@@ -125,4 +138,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImpor
125
138
  providedIn: 'root'
126
139
  }]
127
140
  }], ctorParameters: function () { return [{ type: i1.AuthService }]; } });
128
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"button.service.js","sourceRoot":"","sources":["../../../../../projects/tin-spa/src/lib/services/button.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\r\nimport { Button, TableConfig } from '../classes/Classes';\r\nimport { AuthService } from './auth.service';\r\nimport { Core } from '../classes/TinCore';\r\nimport { RoleAccess } from '../classes/LibClasses';\r\n\r\n@Injectable({\r\n  providedIn: 'root'\r\n})\r\nexport class ButtonService {\r\n\r\n  constructor(private authService: AuthService) { }\r\n\r\n  getButton(buttons: Button[], name: string): Button | null {\r\n    return buttons?.find(x => x.name === name) || null;\r\n  }\r\n\r\n  getButtonProperties(button: Button, row: any, config: TableConfig): {\r\n    color: string,\r\n    icon: string,\r\n    disabled: boolean,\r\n    visible: boolean,\r\n    tip: string\r\n  } {\r\n    return {\r\n      color: this.getButtonColor(button, row),\r\n      icon: this.getIcon(button, config),\r\n      disabled: this.testDisabled(button, row),\r\n      visible: this.testVisible(button, row, config),\r\n      tip: button.tip || button.name\r\n    };\r\n  }\r\n\r\n  getButtonColor(button: Button, row: any): string {\r\n    if (this.testDisabled(button, row)) return 'lightgray';\r\n\r\n    if (button?.color) return button.color;\r\n    if (button.icon?.color) return button.icon.color;\r\n\r\n    const colorMap = {\r\n      'delete': '#F44336',\r\n      'create': 'green'\r\n    };\r\n\r\n    return colorMap[button.name] || '#4050B5';\r\n  }\r\n\r\n\r\n\r\n  isButtonVisible(button: Button, row: any, isLoadComplete: boolean): boolean {\r\n    if (!isLoadComplete) return false;\r\n    return this.testVisible(button,row);\r\n  }\r\n\r\n  getTitleAction(mode: string, buttons: Button[]): string {\r\n    const modeMap = {\r\n      'view': 'View',\r\n      'edit': buttons?.find(b => b.name === 'edit')?.display || 'Edit',\r\n      'create': buttons?.find(b => b.name === 'create')?.display || 'Create'\r\n    };\r\n    return modeMap[mode] || '';\r\n  }\r\n\r\n  isTabVisible(tableConfig: TableConfig): boolean {\r\n    return tableConfig.hideTabCondition ? !tableConfig.hideTabCondition(tableConfig.parentData) : true;\r\n  }\r\n\r\n  getIcon(button: Button, config: TableConfig): string {\r\n    if (button.icon) {\r\n      return button.icon.name;\r\n    }\r\n    return this.getDefaultIcon(button.name, config.flatButtons);\r\n  }\r\n\r\n  testDisabled(button: Button, row: any): boolean {\r\n\r\n    if ((button.name === 'edit' || button.name === 'delete') && row?.locked) return true;\r\n\r\n    if (row?.pendingApproval && !button.ignoreApproval && button.name != 'view') return true;\r\n\r\n    return button.disabled ? button.disabled(row) : false;\r\n  }\r\n\r\n  testVisible(button: Button, row: any, config?: TableConfig): boolean {\r\n\r\n    if (button.visible && !button.visible(row)) {\r\n      return false;\r\n    }\r\n\r\n    const currentRole = this.authService.currentRoleSource.value;\r\n\r\n    // Check if button has detailsConfig with form restrictions\r\n    if (button.detailsConfig?.formConfig) {\r\n      const access = Core.getFormAccess(button.detailsConfig.formConfig, currentRole);\r\n      if (!this.hasRequiredAccess(access, button.name)) {\r\n        return false;\r\n      }\r\n    }\r\n\r\n    // For create/edit/view buttons, check table's formConfig\r\n    if (['create', 'edit', 'view', 'delete'].includes(button.name) && config?.formConfig) {\r\n      const access = Core.getFormAccess(config.formConfig, currentRole);\r\n      if (!this.hasRequiredAccess(access, button.name)) {\r\n        return false;\r\n      }\r\n    }\r\n\r\n    return true;\r\n  }\r\n\r\n  private hasRequiredAccess(access: RoleAccess, buttonName: string): boolean {\r\n    switch (buttonName) {\r\n      case 'view':\r\n        return access >= RoleAccess.View;\r\n      case 'edit':\r\n        return access >= RoleAccess.Edit;\r\n      case 'create':\r\n        return access >= RoleAccess.Create;\r\n      case 'delete':\r\n        return access >= RoleAccess.Full;\r\n      default:\r\n        return true; // Custom buttons don't require specific access\r\n    }\r\n  }\r\n\r\n  private getDefaultColor(buttonName: string): string {\r\n    switch (buttonName) {\r\n      case 'delete': return '#F44336';\r\n      case 'create': return 'green';\r\n      default: return '#4050B5';\r\n    }\r\n  }\r\n\r\n  private getDefaultIcon(buttonName: string, flatButtons: boolean): string {\r\n    switch (buttonName) {\r\n      case 'view':\r\n        return flatButtons ? 'open_in_new' : 'pageview';\r\n      case 'edit': return 'edit';\r\n      case 'delete': return 'delete';\r\n      case 'create': return 'add';\r\n      default: return 'more_horiz';\r\n    }\r\n  }\r\n\r\n  getCreateButton(buttons: Button[]): Button | undefined {\r\n    return buttons?.find(button => button?.name === 'create');\r\n  }\r\n\r\n\r\n\r\n}\r\n"]}
141
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"button.service.js","sourceRoot":"","sources":["../../../../../projects/tin-spa/src/lib/services/button.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\r\nimport { Button, TableConfig } from '../classes/Classes';\r\nimport { AuthService } from './auth.service';\r\nimport { Core } from '../classes/TinCore';\r\nimport { RoleAccess } from '../classes/LibClasses';\r\n\r\n@Injectable({\r\n  providedIn: 'root'\r\n})\r\nexport class ButtonService {\r\n\r\n  constructor(private authService: AuthService) { }\r\n\r\n  getButton(buttons: Button[], name: string): Button | null {\r\n    return buttons?.find(x => x.name === name) || null;\r\n  }\r\n\r\n  getButtonProperties(button: Button, row: any, config: TableConfig): {\r\n    color: string,\r\n    icon: string,\r\n    disabled: boolean,\r\n    visible: boolean,\r\n    tip: string\r\n  } {\r\n    return {\r\n      color: this.getButtonColor(button, row),\r\n      icon: this.getIcon(button, config),\r\n      disabled: this.testDisabled(button, row),\r\n      visible: this.testVisible(button, row, config),\r\n      tip: button.tip || button.name\r\n    };\r\n  }\r\n\r\n  getButtonColor(button: Button, row: any): string {\r\n    if (this.testDisabled(button, row)) return 'lightgray';\r\n\r\n    if (button?.color) return button.color;\r\n    if (button.icon?.color) return button.icon.color;\r\n\r\n    const colorMap = {\r\n      'delete': '#F44336',\r\n      'create': 'green'\r\n    };\r\n\r\n    return colorMap[button.name] || '#4050B5';\r\n  }\r\n\r\n\r\n\r\n  // isButtonVisible(button: Button, row: any, isLoadComplete: boolean): boolean {\r\n  //   if (!isLoadComplete) return false;\r\n  //   return this.testVisible(button,row);\r\n  // }\r\n\r\n  getTitleAction(mode: string, buttons: Button[]): string {\r\n    const modeMap = {\r\n      'view': 'View',\r\n      'edit': buttons?.find(b => b.name === 'edit')?.display || 'Edit',\r\n      'create': buttons?.find(b => b.name === 'create')?.display || 'Create'\r\n    };\r\n    return modeMap[mode] || '';\r\n  }\r\n\r\n  isTabVisible(tableConfig: TableConfig): boolean {\r\n    return tableConfig.hideTabCondition ? !tableConfig.hideTabCondition(tableConfig.parentData) : true;\r\n  }\r\n\r\n  getIcon(button: Button, config: TableConfig): string {\r\n    if (button.icon) {\r\n      return button.icon.name;\r\n    }\r\n    return this.getDefaultIcon(button.name, config.flatButtons);\r\n  }\r\n\r\n  testDisabled(button: Button, row: any, config?: TableConfig): boolean {\r\n\r\n    // Check standard locking/approval rules\r\n    if ((button.name === 'edit' || button.name === 'delete') && row?.locked) return true;\r\n    if (row?.pendingApproval && !button.ignoreApproval && button.name != 'view') return true;\r\n\r\n    // Check form access if detailsConfig exists\r\n    if (button.detailsConfig?.formConfig) {\r\n      const access = Core.getFormAccess(button.detailsConfig.formConfig, this.authService.currentRoleSource.value);\r\n      if (!this.hasRequiredAccess(access, button.name)) return true;\r\n    }\r\n\r\n    // Check form access levels from table config\r\n    if (['create', 'edit', 'view', 'delete'].includes(button.name) && config?.formConfig) {\r\n      const access = Core.getFormAccess(config.formConfig, this.authService.currentRoleSource.value);\r\n      if (!this.hasRequiredAccess(access, button.name)) return true;\r\n    }\r\n\r\n    // Check button-specific disabled rules\r\n    return button.disabled ? button.disabled(row) : false;\r\n  }\r\n\r\n  testVisible(button: Button, row: any, config?: TableConfig): boolean {\r\n\r\n    if (button.visible && !button.visible(row)) {\r\n      return false;\r\n    }\r\n\r\n    const currentRole = this.authService.currentRoleSource.value;\r\n\r\n    // Check if button has detailsConfig with form restrictions\r\n    if (button.detailsConfig?.formConfig) {\r\n      const access = Core.getFormAccess(button.detailsConfig.formConfig, currentRole);\r\n      if (!this.hasRequiredAccess(access, button.name)) {\r\n        return false;\r\n      }\r\n    }\r\n\r\n    // For create/edit/view buttons, check table's formConfig\r\n    if (['create', 'edit', 'view', 'delete'].includes(button.name) && config?.formConfig) {\r\n      const access = Core.getFormAccess(config.formConfig, currentRole);\r\n      if (!this.hasRequiredAccess(access, button.name)) {\r\n        return false;\r\n      }\r\n    }\r\n\r\n    return true;\r\n  }\r\n\r\n  private hasRequiredAccess(access: RoleAccess, buttonName: string): boolean {\r\n    switch (buttonName) {\r\n      case 'view':\r\n        return access >= RoleAccess.View;\r\n      case 'edit':\r\n        return access >= RoleAccess.Edit;\r\n      case 'create':\r\n        return access >= RoleAccess.Create;\r\n      case 'delete':\r\n        return access >= RoleAccess.Full;\r\n      default:\r\n        return true; // Custom buttons don't require specific access\r\n    }\r\n  }\r\n\r\n  private getDefaultColor(buttonName: string): string {\r\n    switch (buttonName) {\r\n      case 'delete': return '#F44336';\r\n      case 'create': return 'green';\r\n      default: return '#4050B5';\r\n    }\r\n  }\r\n\r\n  private getDefaultIcon(buttonName: string, flatButtons: boolean): string {\r\n    switch (buttonName) {\r\n      case 'view':\r\n        return flatButtons ? 'open_in_new' : 'pageview';\r\n      case 'edit': return 'edit';\r\n      case 'delete': return 'delete';\r\n      case 'create': return 'add';\r\n      default: return 'more_horiz';\r\n    }\r\n  }\r\n\r\n  getCreateButton(buttons: Button[]): Button | undefined {\r\n    return buttons?.find(button => button?.name === 'create');\r\n  }\r\n\r\n\r\n\r\n}\r\n"]}