@sarasanalytics-com/design-system 0.0.116 → 0.0.117
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.
|
@@ -42,9 +42,17 @@ export class MessageBannerV2Component {
|
|
|
42
42
|
this.state.update(s => ({ ...s, description: value }));
|
|
43
43
|
}
|
|
44
44
|
ngOnChanges(changes) {
|
|
45
|
-
this
|
|
45
|
+
// Check if this is the first initialization or if messageType has changed
|
|
46
|
+
const shouldUpdateBasedOnType = changes['messageType'] &&
|
|
47
|
+
// Only update based on type if we don't have custom values set
|
|
48
|
+
!(this.state().title || this.state().description || this.state().messageIcon);
|
|
49
|
+
if (shouldUpdateBasedOnType) {
|
|
50
|
+
this.updateContentBasedOnType(this.messageType || 'default');
|
|
51
|
+
}
|
|
46
52
|
}
|
|
47
53
|
updateContentBasedOnType(type) {
|
|
54
|
+
// Only apply default content if custom content hasn't been provided
|
|
55
|
+
const currentState = this.state();
|
|
48
56
|
const content = {
|
|
49
57
|
default: {
|
|
50
58
|
title: 'Oops! Something went wrong. Please try again or try after some time',
|
|
@@ -63,7 +71,7 @@ export class MessageBannerV2Component {
|
|
|
63
71
|
},
|
|
64
72
|
accessRestricted: {
|
|
65
73
|
title: 'Access Restricted!',
|
|
66
|
-
description: `Looks like you don
|
|
74
|
+
description: `Looks like you don't have access to this module just yet. No worries—reach out to your admin to request access to this feature. If you're facing any issues, feel free to <a class="custom-contact-link" href="${this.contactUrl ?? 'https://support.default.com'}" target="_blank">Contact Us</a>.`,
|
|
67
75
|
messageIcon: '/assets/access-restricted.gif',
|
|
68
76
|
spacingBetweenImageAndDescription: '0px',
|
|
69
77
|
contentGap: '20px',
|
|
@@ -75,7 +83,15 @@ export class MessageBannerV2Component {
|
|
|
75
83
|
this.state.update(s => ({
|
|
76
84
|
...s,
|
|
77
85
|
type,
|
|
78
|
-
|
|
86
|
+
// Only apply default values if custom values haven't been provided
|
|
87
|
+
title: s.title || contentItem.title,
|
|
88
|
+
description: s.description || contentItem.description,
|
|
89
|
+
messageIcon: s.messageIcon || contentItem.messageIcon,
|
|
90
|
+
// For other properties, only apply them if they exist in the contentItem
|
|
91
|
+
...(contentItem.spacingBetweenImageAndDescription && { spacingBetweenImageAndDescription: contentItem.spacingBetweenImageAndDescription }),
|
|
92
|
+
...(contentItem.contentGap && { contentGap: contentItem.contentGap }),
|
|
93
|
+
...(contentItem.titleClass && { titleClass: contentItem.titleClass }),
|
|
94
|
+
...(contentItem.descriptionClass && { descriptionClass: contentItem.descriptionClass })
|
|
79
95
|
}));
|
|
80
96
|
}
|
|
81
97
|
sanitizeDescription(description) {
|
|
@@ -119,4 +135,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.2.4", ngImpor
|
|
|
119
135
|
}], buttonClick: [{
|
|
120
136
|
type: Output
|
|
121
137
|
}] } });
|
|
122
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"message-banner-v2.component.js","sourceRoot":"","sources":["../../../../../projects/component-library/src/lib/message-banner-v2/message-banner-v2.component.ts","../../../../../projects/component-library/src/lib/message-banner-v2/message-banner-v2.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, computed, signal, effect, EventEmitter, Output, SimpleChanges, OnChanges } from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport { IconService } from '../icon/icon.service';\nimport { ButtonComponent } from '../button/button.component';\nimport { IconName } from '../../interfaces/icon-interface';\nimport { ButtonIconPosition, ButtonSize, ButtonType, ButtonInterface, ButtonClickEvent } from '../../interfaces/button-interface';\nimport { HttpClientModule } from '@angular/common/http';\nimport { IconComponent } from '../icon/icon.component';\n\ninterface MessageBannerState {\n  type: 'default' | 'connection' | 'access' | 'accessRestricted';\n  title: string;\n  description: string;\n  messageIcon: string;\n  messageIconSize?: {\n    width?: string;\n    height?: string;\n  };\n  spacingBetweenImageAndDescription?: string;\n  contentGap?: string; \n  contactUrl?: string;\n  titleClass?: string;\n  descriptionClass?: string;\n}\n\n@Component({\n  selector: 'sa-message-banner-v2',\n  standalone: true,\n  imports: [CommonModule, IconComponent, ButtonComponent, HttpClientModule],\n  providers: [IconService],\n  templateUrl: './message-banner-v2.component.html',\n  styleUrls: ['./message-banner-v2.component.css']\n})\nexport class MessageBannerV2Component implements OnChanges {\n  // State management using signals\n  private state = signal<MessageBannerState>({\n    type: 'default',\n    title: '',\n    description: '',\n    messageIcon: '',\n    messageIconSize: {\n      height: '12.5rem'\n    }\n  });\n\n  // Computed properties\n  protected bannerContent = computed(() => {\n    const currentState = this.state();\n    return {\n      ...currentState,\n      description: this.sanitizeDescription(currentState.description)\n    };\n  });\n\n  @Input() messageType: 'default' | 'connection' | 'access' | 'accessRestricted';\n  @Input() contactUrl?: string;\n  @Input() spacingBetweenImageAndDescription?: string;\n  @Input() contentGap?: string;\n  @Input() titleClass?: string;\n  @Input() descriptionClass?: string;\n\n\n  @Input() set messageIcon(value: string) {\n    this.state.update(s => ({ ...s, messageIcon: value }));\n  }\n\n  @Input() set messageIconSize(value: { width?: string; height?: string }) {\n    this.state.update(s => ({ ...s, messageIconSize: value }));\n  }\n\n  @Input() set title(value: string) {\n    this.state.update(s => ({ ...s, title: value }));\n  }\n\n  @Input() set description(value: string) {\n    this.state.update(s => ({ ...s, description: value }));\n  }\n\n  @Input() buttons: ButtonInterface[] = [];\n  @Output() buttonClick = new EventEmitter<ButtonClickEvent>();\n\n  ngOnChanges(changes: SimpleChanges): void {\n    this.updateContentBasedOnType(this.messageType);\n  }\n\n  private updateContentBasedOnType(type: 'default' | 'connection' | 'access' | 'accessRestricted') {\n    const content = {\n      default: {\n        title: 'Oops! Something went wrong. Please try again or try after some time',\n        description: 'If the issue persists, contact our support team at <span class=\"support-email\">support@sarasanalytics.com</span>',\n        messageIcon: '/assets/error.gif'\n      },\n      connection: {\n        title: 'We\\'re having trouble connecting. Please check your internet connection and try again',\n        description: 'You can also reach out to our support team at <span class=\"support-email\">support@sarasanalytics.com</span> for assistance',\n        messageIcon: '/assets/lost-connection.gif'\n      },\n      access: {\n        title: 'This is taking a bit longer than usual. Please hold on!',\n        description: 'If the delay continues, check your connection and try refreshing the page.',\n        messageIcon: '/assets/long-loading.gif'\n      },\n      accessRestricted: {\n        title: 'Access Restricted!',\n        description: `Looks like you don’t have access to this module just yet. No worries—reach out to your admin to request access to this feature. If you're facing any issues, feel free to <a class=\"custom-contact-link\" href=\"${this.contactUrl ?? 'https://support.default.com'}\" target=\"_blank\">Contact Us</a>.`,\n        messageIcon: '/assets/access-restricted.gif',\n        spacingBetweenImageAndDescription: '0px',\n        contentGap:'20px',\n        titleClass: 'access-restricted-title',\n        descriptionClass: 'access-restricted-description',\n      }\n    };\n\n    const contentItem = content[type];\n\n    this.state.update(s => ({\n      ...s,\n      type,\n      ...contentItem\n    }));\n  }\n\n  private sanitizeDescription(description: string): string {\n    return description.replace(\n      /support@sarasanalytics\\.com/g,\n      '<span class=\"support-email\">support@sarasanalytics.com</span>'\n    );\n  }\n\n  onButtonClick(button: ButtonInterface, event: Event): void {\n    this.buttonClick.emit({\n      type: button.type as 'primary' | 'secondary',\n      button,\n      originalEvent: event\n    });\n  }\n}\n","@if (bannerContent(); as content) {\n  <div class=\"error-page message-banner-v2\">\n    <div class=\"content-container\" \n         [ngStyle]=\"{ gap: contentGap || 'var(--medium-32px, 32px)' }\">\n      <div class=\"image-container\"\n           [ngStyle]=\"{ 'margin-bottom': spacingBetweenImageAndDescription ?? 'var(--small-16px, 16px)' }\">\n        <img [src]=\"content.messageIcon\" \n             [alt]=\"content.type + ' image'\"\n             [style.--message-icon-height]=\"content.messageIconSize?.height\"\n             [style.--message-icon-width]=\"content.messageIconSize?.width\" />\n      </div>\n      <div class=\"details\">\n        <div class=\"title-description\">\n          <div class=\"title-class\" [ngClass]=\"titleClass || 'title-class'\" [innerHTML]=\"content.title\"></div>\n          <p class=\"description\" [ngClass]=\"descriptionClass || 'description'\" [innerHTML]=\"content.description\"></p>\n        </div>\n        @if(buttons){\n        <div class=\"button-group\">\n          @for (button of buttons; track button.text) {\n            <sa-button \n              [text]=\"button.text\"\n              [icon]=\"button.icon\"\n              [size]=\"button.size\"\n              [type]=\"button.type\"\n              [iconPosition]=\"button.iconPosition\"\n              [buttonIconSize]=\"button.buttonIconSize\"\n              (onClickEvent)=\"onButtonClick(button, $event)\"\n              role=\"button\"\n              [attr.aria-label]=\"button.text\">\n            </sa-button>\n          }\n        </div>\n      }\n      </div>\n    </div>\n  </div>\n}\n"]}
|
|
138
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"message-banner-v2.component.js","sourceRoot":"","sources":["../../../../../projects/component-library/src/lib/message-banner-v2/message-banner-v2.component.ts","../../../../../projects/component-library/src/lib/message-banner-v2/message-banner-v2.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, computed, signal, effect, EventEmitter, Output, SimpleChanges, OnChanges } from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport { IconService } from '../icon/icon.service';\nimport { ButtonComponent } from '../button/button.component';\nimport { IconName } from '../../interfaces/icon-interface';\nimport { ButtonIconPosition, ButtonSize, ButtonType, ButtonInterface, ButtonClickEvent } from '../../interfaces/button-interface';\nimport { HttpClientModule } from '@angular/common/http';\nimport { IconComponent } from '../icon/icon.component';\n\ninterface MessageBannerState {\n  type: 'default' | 'connection' | 'access' | 'accessRestricted';\n  title: string;\n  description: string;\n  messageIcon: string;\n  messageIconSize?: {\n    width?: string;\n    height?: string;\n  };\n  spacingBetweenImageAndDescription?: string;\n  contentGap?: string; \n  contactUrl?: string;\n  titleClass?: string;\n  descriptionClass?: string;\n}\n\n@Component({\n  selector: 'sa-message-banner-v2',\n  standalone: true,\n  imports: [CommonModule, IconComponent, ButtonComponent, HttpClientModule],\n  providers: [IconService],\n  templateUrl: './message-banner-v2.component.html',\n  styleUrls: ['./message-banner-v2.component.css']\n})\nexport class MessageBannerV2Component implements OnChanges {\n  // State management using signals\n  private state = signal<MessageBannerState>({\n    type: 'default',\n    title: '',\n    description: '',\n    messageIcon: '',\n    messageIconSize: {\n      height: '12.5rem'\n    }\n  });\n\n  // Computed properties\n  protected bannerContent = computed(() => {\n    const currentState = this.state();\n    return {\n      ...currentState,\n      description: this.sanitizeDescription(currentState.description)\n    };\n  });\n\n  @Input() messageType: 'default' | 'connection' | 'access' | 'accessRestricted';\n  @Input() contactUrl?: string;\n  @Input() spacingBetweenImageAndDescription?: string;\n  @Input() contentGap?: string;\n  @Input() titleClass?: string;\n  @Input() descriptionClass?: string;\n\n\n  @Input() set messageIcon(value: string) {\n    this.state.update(s => ({ ...s, messageIcon: value }));\n  }\n\n  @Input() set messageIconSize(value: { width?: string; height?: string }) {\n    this.state.update(s => ({ ...s, messageIconSize: value }));\n  }\n\n  @Input() set title(value: string) {\n    this.state.update(s => ({ ...s, title: value }));\n  }\n\n  @Input() set description(value: string) {\n    this.state.update(s => ({ ...s, description: value }));\n  }\n\n  @Input() buttons: ButtonInterface[] = [];\n  @Output() buttonClick = new EventEmitter<ButtonClickEvent>();\n\n  ngOnChanges(changes: SimpleChanges): void {\n    // Check if this is the first initialization or if messageType has changed\n    const shouldUpdateBasedOnType = changes['messageType'] && \n      // Only update based on type if we don't have custom values set\n      !(this.state().title || this.state().description || this.state().messageIcon);\n    \n    if (shouldUpdateBasedOnType) {\n      this.updateContentBasedOnType(this.messageType || 'default');\n    }\n  }\n\n  private updateContentBasedOnType(type: 'default' | 'connection' | 'access' | 'accessRestricted') {\n    // Define a type for the content items to ensure all properties are properly typed\n    type ContentItem = {\n      title: string;\n      description: string;\n      messageIcon: string;\n      spacingBetweenImageAndDescription?: string;\n      contentGap?: string;\n      titleClass?: string;\n      descriptionClass?: string;\n    };\n\n    // Only apply default content if custom content hasn't been provided\n    const currentState = this.state();\n    const content: Record<'default' | 'connection' | 'access' | 'accessRestricted', ContentItem> = {\n      default: {\n        title: 'Oops! Something went wrong. Please try again or try after some time',\n        description: 'If the issue persists, contact our support team at <span class=\"support-email\">support@sarasanalytics.com</span>',\n        messageIcon: '/assets/error.gif'\n      },\n      connection: {\n        title: 'We\\'re having trouble connecting. Please check your internet connection and try again',\n        description: 'You can also reach out to our support team at <span class=\"support-email\">support@sarasanalytics.com</span> for assistance',\n        messageIcon: '/assets/lost-connection.gif'\n      },\n      access: {\n        title: 'This is taking a bit longer than usual. Please hold on!',\n        description: 'If the delay continues, check your connection and try refreshing the page.',\n        messageIcon: '/assets/long-loading.gif'\n      },\n      accessRestricted: {\n        title: 'Access Restricted!',\n        description: `Looks like you don't have access to this module just yet. No worries—reach out to your admin to request access to this feature. If you're facing any issues, feel free to <a class=\"custom-contact-link\" href=\"${this.contactUrl ?? 'https://support.default.com'}\" target=\"_blank\">Contact Us</a>.`,\n        messageIcon: '/assets/access-restricted.gif',\n        spacingBetweenImageAndDescription: '0px',\n        contentGap:'20px',\n        titleClass: 'access-restricted-title',\n        descriptionClass: 'access-restricted-description',\n      }\n    };\n\n    const contentItem = content[type];\n\n    this.state.update(s => ({\n      ...s,\n      type,\n      // Only apply default values if custom values haven't been provided\n      title: s.title || contentItem.title,\n      description: s.description || contentItem.description,\n      messageIcon: s.messageIcon || contentItem.messageIcon,\n      // For other properties, only apply them if they exist in the contentItem\n      ...(contentItem.spacingBetweenImageAndDescription && { spacingBetweenImageAndDescription: contentItem.spacingBetweenImageAndDescription }),\n      ...(contentItem.contentGap && { contentGap: contentItem.contentGap }),\n      ...(contentItem.titleClass && { titleClass: contentItem.titleClass }),\n      ...(contentItem.descriptionClass && { descriptionClass: contentItem.descriptionClass })\n    }));\n  }\n\n  private sanitizeDescription(description: string): string {\n    return description.replace(\n      /support@sarasanalytics\\.com/g,\n      '<span class=\"support-email\">support@sarasanalytics.com</span>'\n    );\n  }\n\n  onButtonClick(button: ButtonInterface, event: Event): void {\n    this.buttonClick.emit({\n      type: button.type as 'primary' | 'secondary',\n      button,\n      originalEvent: event\n    });\n  }\n}\n","@if (bannerContent(); as content) {\n  <div class=\"error-page message-banner-v2\">\n    <div class=\"content-container\" \n         [ngStyle]=\"{ gap: contentGap || 'var(--medium-32px, 32px)' }\">\n      <div class=\"image-container\"\n           [ngStyle]=\"{ 'margin-bottom': spacingBetweenImageAndDescription ?? 'var(--small-16px, 16px)' }\">\n        <img [src]=\"content.messageIcon\" \n             [alt]=\"content.type + ' image'\"\n             [style.--message-icon-height]=\"content.messageIconSize?.height\"\n             [style.--message-icon-width]=\"content.messageIconSize?.width\" />\n      </div>\n      <div class=\"details\">\n        <div class=\"title-description\">\n          <div class=\"title-class\" [ngClass]=\"titleClass || 'title-class'\" [innerHTML]=\"content.title\"></div>\n          <p class=\"description\" [ngClass]=\"descriptionClass || 'description'\" [innerHTML]=\"content.description\"></p>\n        </div>\n        @if(buttons){\n        <div class=\"button-group\">\n          @for (button of buttons; track button.text) {\n            <sa-button \n              [text]=\"button.text\"\n              [icon]=\"button.icon\"\n              [size]=\"button.size\"\n              [type]=\"button.type\"\n              [iconPosition]=\"button.iconPosition\"\n              [buttonIconSize]=\"button.buttonIconSize\"\n              (onClickEvent)=\"onButtonClick(button, $event)\"\n              role=\"button\"\n              [attr.aria-label]=\"button.text\">\n            </sa-button>\n          }\n        </div>\n      }\n      </div>\n    </div>\n  </div>\n}\n"]}
|
|
@@ -1387,9 +1387,17 @@ class MessageBannerV2Component {
|
|
|
1387
1387
|
this.state.update(s => ({ ...s, description: value }));
|
|
1388
1388
|
}
|
|
1389
1389
|
ngOnChanges(changes) {
|
|
1390
|
-
this
|
|
1390
|
+
// Check if this is the first initialization or if messageType has changed
|
|
1391
|
+
const shouldUpdateBasedOnType = changes['messageType'] &&
|
|
1392
|
+
// Only update based on type if we don't have custom values set
|
|
1393
|
+
!(this.state().title || this.state().description || this.state().messageIcon);
|
|
1394
|
+
if (shouldUpdateBasedOnType) {
|
|
1395
|
+
this.updateContentBasedOnType(this.messageType || 'default');
|
|
1396
|
+
}
|
|
1391
1397
|
}
|
|
1392
1398
|
updateContentBasedOnType(type) {
|
|
1399
|
+
// Only apply default content if custom content hasn't been provided
|
|
1400
|
+
const currentState = this.state();
|
|
1393
1401
|
const content = {
|
|
1394
1402
|
default: {
|
|
1395
1403
|
title: 'Oops! Something went wrong. Please try again or try after some time',
|
|
@@ -1408,7 +1416,7 @@ class MessageBannerV2Component {
|
|
|
1408
1416
|
},
|
|
1409
1417
|
accessRestricted: {
|
|
1410
1418
|
title: 'Access Restricted!',
|
|
1411
|
-
description: `Looks like you don
|
|
1419
|
+
description: `Looks like you don't have access to this module just yet. No worries—reach out to your admin to request access to this feature. If you're facing any issues, feel free to <a class="custom-contact-link" href="${this.contactUrl ?? 'https://support.default.com'}" target="_blank">Contact Us</a>.`,
|
|
1412
1420
|
messageIcon: '/assets/access-restricted.gif',
|
|
1413
1421
|
spacingBetweenImageAndDescription: '0px',
|
|
1414
1422
|
contentGap: '20px',
|
|
@@ -1420,7 +1428,15 @@ class MessageBannerV2Component {
|
|
|
1420
1428
|
this.state.update(s => ({
|
|
1421
1429
|
...s,
|
|
1422
1430
|
type,
|
|
1423
|
-
|
|
1431
|
+
// Only apply default values if custom values haven't been provided
|
|
1432
|
+
title: s.title || contentItem.title,
|
|
1433
|
+
description: s.description || contentItem.description,
|
|
1434
|
+
messageIcon: s.messageIcon || contentItem.messageIcon,
|
|
1435
|
+
// For other properties, only apply them if they exist in the contentItem
|
|
1436
|
+
...(contentItem.spacingBetweenImageAndDescription && { spacingBetweenImageAndDescription: contentItem.spacingBetweenImageAndDescription }),
|
|
1437
|
+
...(contentItem.contentGap && { contentGap: contentItem.contentGap }),
|
|
1438
|
+
...(contentItem.titleClass && { titleClass: contentItem.titleClass }),
|
|
1439
|
+
...(contentItem.descriptionClass && { descriptionClass: contentItem.descriptionClass })
|
|
1424
1440
|
}));
|
|
1425
1441
|
}
|
|
1426
1442
|
sanitizeDescription(description) {
|