@hmcts/rpx-xui-common-lib 2.1.8 → 2.1.9
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2022/lib/components/service-messages/service-messages.component.mjs +6 -2
- package/esm2022/lib/services/feature-toggle/feature-toggle.guard.mjs +2 -2
- package/esm2022/lib/services/feature-toggle/launch-darkly.service.mjs +6 -1
- package/fesm2022/hmcts-rpx-xui-common-lib.mjs +11 -2
- package/fesm2022/hmcts-rpx-xui-common-lib.mjs.map +1 -1
- package/package.json +1 -1
|
@@ -78,10 +78,14 @@ export class ServiceMessagesComponent {
|
|
|
78
78
|
}
|
|
79
79
|
createFilteredMessages(messages) {
|
|
80
80
|
this.hiddenBanners = this.getSessionCookie(this.serviceMessageCookie) || [];
|
|
81
|
+
const messageCount = messages.length;
|
|
81
82
|
this.filteredMessages = messages.filter((message) => {
|
|
82
83
|
const { roles, message_en } = message;
|
|
83
84
|
const regEx = new RegExp(roles);
|
|
84
|
-
|
|
85
|
+
// If userRoles is empty or undefined, allow the message through
|
|
86
|
+
const isRoleMatch = ((!this.userRoles || this.userRoles.length === 0) && messageCount === 1 && roles === '.+')
|
|
87
|
+
|| (this.userRoles && this.userRoles.some((role) => regEx.test(role)));
|
|
88
|
+
return isRoleMatch &&
|
|
85
89
|
this.compareDates(message) &&
|
|
86
90
|
!this.hiddenBanners.includes(message_en);
|
|
87
91
|
});
|
|
@@ -160,4 +164,4 @@ export class ServiceMessagesComponent {
|
|
|
160
164
|
type: Input
|
|
161
165
|
}] }); })();
|
|
162
166
|
(() => { (typeof ngDevMode === "undefined" || ngDevMode) && i0.ɵsetClassDebugInfo(ServiceMessagesComponent, { className: "ServiceMessagesComponent", filePath: "lib/components/service-messages/service-messages.component.ts", lineNumber: 10 }); })();
|
|
163
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"service-messages.component.js","sourceRoot":"","sources":["../../../../../../projects/exui-common-lib/src/lib/components/service-messages/service-messages.component.ts","../../../../../../projects/exui-common-lib/src/lib/components/service-messages/service-messages.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, OnInit } from '@angular/core';\nimport { FeatureToggleService } from '../../services/feature-toggle/feature-toggle.service';\nimport { ServiceMessages } from '../../models/service-message.model';\n\n\n@Component({\n  selector: 'xuilib-service-messages',\n  templateUrl: './service-messages.component.html'\n})\nexport class ServiceMessagesComponent implements OnInit {\n  public hiddenBanners: string[];\n  public filteredMessages: ServiceMessages[] = [];\n  public isBannerError: boolean;\n  public bannerErrorMsgs: { message: string, index: number }[] = [];\n  public originalMessages: ServiceMessages[] = [];\n\n  @Input() public userRoles: string[];\n  @Input() public featureToggleKey: string;\n  @Input() public serviceMessageCookie: string;\n\n  constructor(private readonly featureToggleService: FeatureToggleService) { }\n\n  public ngOnInit() {\n    this.getServiceMessages();\n  }\n\n  public getServiceMessages(): void {\n    this.featureToggleService.getValue<ServiceMessages[]>(this.featureToggleKey, null)\n      .subscribe(messages => {\n        this.originalMessages = messages;\n        if (!!messages) {\n          this.createFilteredMessages(messages);\n        }\n      });\n  }\n\n  private createFilteredMessages(messages: ServiceMessages[]): void {\n    this.hiddenBanners = this.getSessionCookie(this.serviceMessageCookie) || [];\n    this.filteredMessages = messages.filter((message) => {\n      const { roles, message_en } = message;\n      const regEx = new RegExp(roles);\n      return this.userRoles.some((role) => regEx.test(role)) &&\n        this.compareDates(message) &&\n        !this.hiddenBanners.includes(message_en);\n    });\n  }\n\n  private compareDates(msg: ServiceMessages): boolean {\n    let currentDateTime = new Date();\n    let beginDate: Date | null = null;\n    let endDate: Date | null = null;\n    this.bannerErrorMsgs = [];\n\n    // Parse beginDate if msg.begin is present and valid\n    if (msg.begin && !isNaN(Date.parse(msg.begin))) {\n      beginDate = new Date(msg.begin);\n    }\n\n    if (msg.end && !isNaN(Date.parse(msg.end))) {\n      endDate = new Date(msg.end);\n    }\n    this.originalMessages.forEach(msg => {\n      // Only check for errors if both beginDate and endDate are present and valid\n      if ((msg.begin && !isNaN(Date.parse(msg.end))) && (msg.end && !isNaN(Date.parse(msg.end))) && new Date(msg.begin) > new Date(msg.end)) {\n        this.isBannerError = true;\n        this.bannerErrorMsgs.push({\n          message: `The start date is greater than the end date. Message index: ${msg.index}`,\n          index: msg.index\n        });\n      }\n      \n      // Check for invalid beginDate or endDate separately, if they are present\n      if (msg.begin && isNaN(Date.parse(msg.begin))) {\n        this.isBannerError = true;\n        this.bannerErrorMsgs.push({\n          message: `Invalid start date. Message index: ${msg.index}`,\n          index: msg.index\n        });\n      }\n      if (msg.end && isNaN(Date.parse(msg.end))) {\n        this.isBannerError = true;\n        this.bannerErrorMsgs.push({\n          message: `Invalid end date. Message index: ${msg.index}`,\n          index: msg.index\n        });\n      }\n    });\n\n    const beginDateOK = !msg.begin || (beginDate && beginDate < currentDateTime);\n    const endDateOK = !msg.end || (endDate && endDate >= currentDateTime);\n\n    return beginDateOK && endDateOK;\n  }\n\n  private setSessionCookie(name: string, value: any): void {\n    document.cookie = `${name}=${encodeURIComponent(JSON.stringify(value))}; path=/; SameSite=Lax`;\n  }\n\n  public hideMessage(msg: ServiceMessages): void {\n    this.filteredMessages = this.filteredMessages.filter((f) => f.index !== msg.index);\n    this.hiddenBanners.push(msg.message_en);\n    this.setSessionCookie(this.serviceMessageCookie, this.hiddenBanners);\n  }\n\n  private getSessionCookie(name: string): any {\n    const cookies = document.cookie.split('; ');\n    const cookie = cookies.find((row) => row.startsWith(name + '='));\n    return cookie ? JSON.parse(decodeURIComponent(cookie.split('=')[1])) : [];\n  }\n}\n","<div *ngIf=\"isBannerError\" class=\"govuk-width-container govuk-!-margin-top-6\">\n    <div class=\"hmcts-banner hmcts-banner--warning\" *ngFor=\"let errMsg of bannerErrorMsgs\">\n        <svg class=\"hmcts-banner__icon\" fill=\"currentColor\" role=\"presentation\" focusable=\"false\"\n        xmlns=\"http://www.w3.org/2000/svg\" viewBox=\"0 0 25 25\" height=\"25\" width=\"25\">\n        <path d=\"M13.6,15.4h-2.3v-4.5h2.3V15.4z M13.6,19.8h-2.3v-2.2h2.3V19.8z M0,23.2h25L12.5,2L0,23.2z\" />\n    </svg>\n        <div class=\"hmcts-banner__message\">\n            <span class=\"hmcts-banner__assistive\">{{'Warning' | rpxTranslate}}</span>\n            <h2 class=\"govuk-heading-s\">Error:</h2>\n            <p [innerHTML]=\"errMsg.message\"></p>\n        </div>\n    </div>\n</div>\n<div *ngIf=\"filteredMessages?.length > 0\" class=\"govuk-width-container govuk-!-margin-top-6\">\n    <xuilib-service-message *ngFor=\"let message of filteredMessages\" [key]=\"message\"\n        [message_en]=\"message?.message_en | rpxTranslate\" \n        [message_cy]=\"message?.message_cy | rpxTranslate\"\n        (hideMessage)=\"hideMessage($event)\">\n    </xuilib-service-message>\n</div>"]}
|
|
167
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"service-messages.component.js","sourceRoot":"","sources":["../../../../../../projects/exui-common-lib/src/lib/components/service-messages/service-messages.component.ts","../../../../../../projects/exui-common-lib/src/lib/components/service-messages/service-messages.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, OnInit } from '@angular/core';\nimport { FeatureToggleService } from '../../services/feature-toggle/feature-toggle.service';\nimport { ServiceMessages } from '../../models/service-message.model';\n\n\n@Component({\n  selector: 'xuilib-service-messages',\n  templateUrl: './service-messages.component.html'\n})\nexport class ServiceMessagesComponent implements OnInit {\n  public hiddenBanners: string[];\n  public filteredMessages: ServiceMessages[] = [];\n  public isBannerError: boolean;\n  public bannerErrorMsgs: { message: string, index: number }[] = [];\n  public originalMessages: ServiceMessages[] = [];\n\n  @Input() public userRoles: string[];\n  @Input() public featureToggleKey: string;\n  @Input() public serviceMessageCookie: string;\n\n  constructor(private readonly featureToggleService: FeatureToggleService) { }\n\n  public ngOnInit() {\n    this.getServiceMessages();\n  }\n\n  public getServiceMessages(): void {\n    this.featureToggleService.getValue<ServiceMessages[]>(this.featureToggleKey, null)\n      .subscribe(messages => {\n        this.originalMessages = messages;\n        if (!!messages) {\n          this.createFilteredMessages(messages);\n        }\n      });\n  }\n\n  private createFilteredMessages(messages: ServiceMessages[]): void {\n    this.hiddenBanners = this.getSessionCookie(this.serviceMessageCookie) || [];\n    const messageCount = messages.length;\n    this.filteredMessages = messages.filter((message) => {\n      const { roles, message_en } = message;\n      const regEx = new RegExp(roles);\n      // If userRoles is empty or undefined, allow the message through\n      const isRoleMatch =\n        ((!this.userRoles || this.userRoles.length === 0) && messageCount === 1 && roles === '.+')\n        || (this.userRoles && this.userRoles.some((role) => regEx.test(role)));\n\n      return isRoleMatch &&\n        this.compareDates(message) &&\n        !this.hiddenBanners.includes(message_en);\n    });\n  }\n\n  private compareDates(msg: ServiceMessages): boolean {\n    let currentDateTime = new Date();\n    let beginDate: Date | null = null;\n    let endDate: Date | null = null;\n    this.bannerErrorMsgs = [];\n\n    // Parse beginDate if msg.begin is present and valid\n    if (msg.begin && !isNaN(Date.parse(msg.begin))) {\n      beginDate = new Date(msg.begin);\n    }\n\n    if (msg.end && !isNaN(Date.parse(msg.end))) {\n      endDate = new Date(msg.end);\n    }\n    this.originalMessages.forEach(msg => {\n      // Only check for errors if both beginDate and endDate are present and valid\n      if ((msg.begin && !isNaN(Date.parse(msg.end))) && (msg.end && !isNaN(Date.parse(msg.end))) && new Date(msg.begin) > new Date(msg.end)) {\n        this.isBannerError = true;\n        this.bannerErrorMsgs.push({\n          message: `The start date is greater than the end date. Message index: ${msg.index}`,\n          index: msg.index\n        });\n      }\n      \n      // Check for invalid beginDate or endDate separately, if they are present\n      if (msg.begin && isNaN(Date.parse(msg.begin))) {\n        this.isBannerError = true;\n        this.bannerErrorMsgs.push({\n          message: `Invalid start date. Message index: ${msg.index}`,\n          index: msg.index\n        });\n      }\n      if (msg.end && isNaN(Date.parse(msg.end))) {\n        this.isBannerError = true;\n        this.bannerErrorMsgs.push({\n          message: `Invalid end date. Message index: ${msg.index}`,\n          index: msg.index\n        });\n      }\n    });\n\n    const beginDateOK = !msg.begin || (beginDate && beginDate < currentDateTime);\n    const endDateOK = !msg.end || (endDate && endDate >= currentDateTime);\n\n    return beginDateOK && endDateOK;\n  }\n\n  private setSessionCookie(name: string, value: any): void {\n    document.cookie = `${name}=${encodeURIComponent(JSON.stringify(value))}; path=/; SameSite=Lax`;\n  }\n\n  public hideMessage(msg: ServiceMessages): void {\n    this.filteredMessages = this.filteredMessages.filter((f) => f.index !== msg.index);\n    this.hiddenBanners.push(msg.message_en);\n    this.setSessionCookie(this.serviceMessageCookie, this.hiddenBanners);\n  }\n\n  private getSessionCookie(name: string): any {\n    const cookies = document.cookie.split('; ');\n    const cookie = cookies.find((row) => row.startsWith(name + '='));\n    return cookie ? JSON.parse(decodeURIComponent(cookie.split('=')[1])) : [];\n  }\n}\n","<div *ngIf=\"isBannerError\" class=\"govuk-width-container govuk-!-margin-top-6\">\n    <div class=\"hmcts-banner hmcts-banner--warning\" *ngFor=\"let errMsg of bannerErrorMsgs\">\n        <svg class=\"hmcts-banner__icon\" fill=\"currentColor\" role=\"presentation\" focusable=\"false\"\n        xmlns=\"http://www.w3.org/2000/svg\" viewBox=\"0 0 25 25\" height=\"25\" width=\"25\">\n        <path d=\"M13.6,15.4h-2.3v-4.5h2.3V15.4z M13.6,19.8h-2.3v-2.2h2.3V19.8z M0,23.2h25L12.5,2L0,23.2z\" />\n    </svg>\n        <div class=\"hmcts-banner__message\">\n            <span class=\"hmcts-banner__assistive\">{{'Warning' | rpxTranslate}}</span>\n            <h2 class=\"govuk-heading-s\">Error:</h2>\n            <p [innerHTML]=\"errMsg.message\"></p>\n        </div>\n    </div>\n</div>\n<div *ngIf=\"filteredMessages?.length > 0\" class=\"govuk-width-container govuk-!-margin-top-6\">\n    <xuilib-service-message *ngFor=\"let message of filteredMessages\" [key]=\"message\"\n        [message_en]=\"message?.message_en | rpxTranslate\" \n        [message_cy]=\"message?.message_cy | rpxTranslate\"\n        (hideMessage)=\"hideMessage($event)\">\n    </xuilib-service-message>\n</div>"]}
|
|
@@ -23,7 +23,7 @@ export class FeatureToggleGuard {
|
|
|
23
23
|
* the redirecting navigation will now use replaceUrl if the initial navigation was also using the replaceUrl option.
|
|
24
24
|
*/
|
|
25
25
|
canActivate(route) {
|
|
26
|
-
return combineLatest([...route.data.needsFeaturesEnabled.map(feature => this.featureToggleService.getValueOnce(feature,
|
|
26
|
+
return combineLatest([...route.data.needsFeaturesEnabled.map(feature => this.featureToggleService.getValueOnce(feature, true))]).pipe(map(featureStatuses => featureStatuses.every(status => status)), map(status => (route.data.expectFeatureEnabled !== false && status) || (route.data.expectFeatureEnabled === false && !status) || this.router.parseUrl(route.data.featureDisabledRedirect)));
|
|
27
27
|
}
|
|
28
28
|
static { this.ɵfac = function FeatureToggleGuard_Factory(__ngFactoryType__) { return new (__ngFactoryType__ || FeatureToggleGuard)(i0.ɵɵinject(i1.FeatureToggleService), i0.ɵɵinject(i2.Router)); }; }
|
|
29
29
|
static { this.ɵprov = /*@__PURE__*/ i0.ɵɵdefineInjectable({ token: FeatureToggleGuard, factory: FeatureToggleGuard.ɵfac, providedIn: 'root' }); }
|
|
@@ -34,4 +34,4 @@ export class FeatureToggleGuard {
|
|
|
34
34
|
providedIn: 'root'
|
|
35
35
|
}]
|
|
36
36
|
}], () => [{ type: i1.FeatureToggleService }, { type: i2.Router }], null); })();
|
|
37
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
37
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZmVhdHVyZS10b2dnbGUuZ3VhcmQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9leHVpLWNvbW1vbi1saWIvc3JjL2xpYi9zZXJ2aWNlcy9mZWF0dXJlLXRvZ2dsZS9mZWF0dXJlLXRvZ2dsZS5ndWFyZC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsVUFBVSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQzNDLE9BQU8sRUFBMEIsTUFBTSxFQUFXLE1BQU0saUJBQWlCLENBQUM7QUFDMUUsT0FBTyxFQUFFLGFBQWEsRUFBYyxNQUFNLE1BQU0sQ0FBQztBQUNqRCxPQUFPLEVBQUUsR0FBRyxFQUFFLE1BQU0sZ0JBQWdCLENBQUM7QUFDckMsT0FBTyxFQUFFLG9CQUFvQixFQUFFLE1BQU0sMEJBQTBCLENBQUM7Ozs7QUFLaEUsTUFBTSxPQUFPLGtCQUFrQjtJQUUzQixZQUNxQixvQkFBMEMsRUFDMUMsTUFBYztRQURkLHlCQUFvQixHQUFwQixvQkFBb0IsQ0FBc0I7UUFDMUMsV0FBTSxHQUFOLE1BQU0sQ0FBUTtJQUNoQyxDQUFDO0lBRUosWUFBWTtJQUNaOzs7Ozs7Ozs7T0FTRztJQUNJLFdBQVcsQ0FBQyxLQUE2QjtRQUM1QyxPQUFPLGFBQWEsQ0FBQyxDQUFDLEdBQUksS0FBSyxDQUFDLElBQUksQ0FBQyxvQkFBaUMsQ0FBQyxHQUFHLENBQ2xFLE9BQU8sQ0FBQyxFQUFFLENBQUMsSUFBSSxDQUFDLG9CQUFvQixDQUFDLFlBQVksQ0FBVSxPQUFPLEVBQUUsSUFBSSxDQUFDLENBQ2hGLENBQUMsQ0FBQyxDQUFDLElBQUksQ0FDSixHQUFHLENBQUMsZUFBZSxDQUFDLEVBQUUsQ0FBQyxlQUFlLENBQUMsS0FBSyxDQUFDLE1BQU0sQ0FBQyxFQUFFLENBQUMsTUFBTSxDQUFDLENBQUMsRUFDL0QsR0FBRyxDQUFDLE1BQU0sQ0FBQyxFQUFFLENBQUMsQ0FBQyxLQUFLLENBQUMsSUFBSSxDQUFDLG9CQUFvQixLQUFLLEtBQUssSUFBSSxNQUFNLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxJQUFJLENBQUMsb0JBQW9CLEtBQUssS0FBSyxJQUFJLENBQUMsTUFBTSxDQUFDLElBQUksSUFBSSxDQUFDLE1BQU0sQ0FBQyxRQUFRLENBQUMsS0FBSyxDQUFDLElBQUksQ0FBQyx1QkFBaUMsQ0FBQyxDQUFDLENBQ3ZNLENBQUM7SUFDTixDQUFDO21IQXpCUSxrQkFBa0I7dUVBQWxCLGtCQUFrQixXQUFsQixrQkFBa0IsbUJBRmYsTUFBTTs7aUZBRVQsa0JBQWtCO2NBSDlCLFVBQVU7ZUFBQztnQkFDUixVQUFVLEVBQUUsTUFBTTthQUNyQiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEluamVjdGFibGUgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IEFjdGl2YXRlZFJvdXRlU25hcHNob3QsIFJvdXRlciwgVXJsVHJlZSB9IGZyb20gJ0Bhbmd1bGFyL3JvdXRlcic7XG5pbXBvcnQgeyBjb21iaW5lTGF0ZXN0LCBPYnNlcnZhYmxlIH0gZnJvbSAncnhqcyc7XG5pbXBvcnQgeyBtYXAgfSBmcm9tICdyeGpzL29wZXJhdG9ycyc7XG5pbXBvcnQgeyBGZWF0dXJlVG9nZ2xlU2VydmljZSB9IGZyb20gJy4vZmVhdHVyZS10b2dnbGUuc2VydmljZSc7XG5cbkBJbmplY3RhYmxlKHtcbiAgICBwcm92aWRlZEluOiAncm9vdCdcbn0pXG5leHBvcnQgY2xhc3MgRmVhdHVyZVRvZ2dsZUd1YXJkICB7XG5cbiAgICBwdWJsaWMgY29uc3RydWN0b3IoXG4gICAgICAgIHByaXZhdGUgcmVhZG9ubHkgZmVhdHVyZVRvZ2dsZVNlcnZpY2U6IEZlYXR1cmVUb2dnbGVTZXJ2aWNlLFxuICAgICAgICBwcml2YXRlIHJlYWRvbmx5IHJvdXRlcjogUm91dGVyXG4gICAgKSB7fVxuXG4gICAgLy8gVEVTVC1UT0RPXG4gICAgLyoqXG4gICAgICogVXNhZ2U6IEFkZCB0aGUgRmVhdHVyZVRvZ2dsZUd1YXJkIGFzIHRoZSBHdWFyZCBmb3IgYSByb3V0ZSwgYW5kIHByb3ZpZGUgdGhlIGZvbGxvd2luZ1xuICAgICAqIGluIHRoZSBkYXRhIGFycmF5IGZvciB0aGUgcm91dGU6XG4gICAgICogLSBuZWVkc0ZlYXR1cmVzRW5hYmxlZDogQW4gYXJyYXkgb2YgZmVhdHVyZSBrZXlzIHRoYXQgbmVlZCB0byBiZSBlbmFibGVkIGZvciB0aGlzIHJvdXRlXG4gICAgICogLSBmZWF0dXJlRGlzYWJsZWRSZWRpcmVjdDogdGhlIFVSTCB0byByZWRpcmVjdCB0byB3aGVuIHRoZSB0aGlzIHJvdXRlIGlzIG5vdCBhY2Nlc3NpYmxlIGR1ZSB0byBkaXNhYmxlZCBmZWF0dXJlc1xuICAgICAqIC0gZXhwZWN0RmVhdHVyZUVuYWJsZWQ6IFNldHMgd2hldGhlciBhIHJvdXRlIHNob3VsZCBiZSBlbmFibGVkL2Rpc2FibGVkIGJhc2VkIG9uIHdoZXRoZXIgZmVhdHVyZSBpcyBwcmVzZW50XG4gICAgICogQHBhcmFtIHJvdXRlIEF1dG9tYXRpY2FsbHkgcHJvdmlkZWQgYnkgQW5ndWxhclxuICAgICAqIE5vdGU6IFBlciBBbmd1bGFyIDE4LCB3aGVuIGEgZ3VhcmQgcmV0dXJucyBhIFVybFRyZWUgYXMgYSByZWRpcmVjdCxcbiAgICAgKiB0aGUgcmVkaXJlY3RpbmcgbmF2aWdhdGlvbiB3aWxsIG5vdyB1c2UgcmVwbGFjZVVybCBpZiB0aGUgaW5pdGlhbCBuYXZpZ2F0aW9uIHdhcyBhbHNvIHVzaW5nIHRoZSByZXBsYWNlVXJsIG9wdGlvbi5cbiAgICAgKi9cbiAgICBwdWJsaWMgY2FuQWN0aXZhdGUocm91dGU6IEFjdGl2YXRlZFJvdXRlU25hcHNob3QpOiBPYnNlcnZhYmxlPGJvb2xlYW4gfCBVcmxUcmVlPiB7XG4gICAgICAgIHJldHVybiBjb21iaW5lTGF0ZXN0KFsuLi4ocm91dGUuZGF0YS5uZWVkc0ZlYXR1cmVzRW5hYmxlZCBhcyBzdHJpbmdbXSkubWFwKFxuICAgICAgICAgICAgICAgIGZlYXR1cmUgPT4gdGhpcy5mZWF0dXJlVG9nZ2xlU2VydmljZS5nZXRWYWx1ZU9uY2U8Ym9vbGVhbj4oZmVhdHVyZSwgdHJ1ZSlcbiAgICAgICAgKV0pLnBpcGUoXG4gICAgICAgICAgICBtYXAoZmVhdHVyZVN0YXR1c2VzID0+IGZlYXR1cmVTdGF0dXNlcy5ldmVyeShzdGF0dXMgPT4gc3RhdHVzKSksXG4gICAgICAgICAgICBtYXAoc3RhdHVzID0+IChyb3V0ZS5kYXRhLmV4cGVjdEZlYXR1cmVFbmFibGVkICE9PSBmYWxzZSAmJiBzdGF0dXMpIHx8IChyb3V0ZS5kYXRhLmV4cGVjdEZlYXR1cmVFbmFibGVkID09PSBmYWxzZSAmJiAhc3RhdHVzKSB8fCB0aGlzLnJvdXRlci5wYXJzZVVybChyb3V0ZS5kYXRhLmZlYXR1cmVEaXNhYmxlZFJlZGlyZWN0IGFzIHN0cmluZykpXG4gICAgICAgICk7XG4gICAgfVxufVxuIl19
|
|
@@ -18,6 +18,11 @@ export class LaunchDarklyService {
|
|
|
18
18
|
this.client.on('ready', () => {
|
|
19
19
|
this.client.identify(this.context).then(() => this.ready.next(true));
|
|
20
20
|
});
|
|
21
|
+
this.client.on('error', (err) => {
|
|
22
|
+
console.error('LaunchDarkly client network error:', err);
|
|
23
|
+
// Allow the app to proceed by emitting true, so subscribers are not blocked
|
|
24
|
+
this.ready.next(true);
|
|
25
|
+
});
|
|
21
26
|
}
|
|
22
27
|
isEnabled(feature, defaultValue = false) {
|
|
23
28
|
return this.getValue(feature, defaultValue);
|
|
@@ -62,4 +67,4 @@ export class LaunchDarklyService {
|
|
|
62
67
|
providedIn: 'root'
|
|
63
68
|
}]
|
|
64
69
|
}], null, null); })();
|
|
65
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
70
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"launch-darkly.service.js","sourceRoot":"","sources":["../../../../../../projects/exui-common-lib/src/lib/services/feature-toggle/launch-darkly.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { initialize, LDClient, LDContext } from 'launchdarkly-js-client-sdk';\nimport { BehaviorSubject, Observable } from 'rxjs';\nimport { distinctUntilChanged, filter, map } from 'rxjs/operators';\nimport { FeatureUser } from '../../models/feature-user';\nimport { FeatureToggleService } from './feature-toggle.service';\n\n@Injectable({\n  providedIn: 'root'\n})\nexport class LaunchDarklyService implements FeatureToggleService {\n  private client: LDClient;\n  private readonly ready: BehaviorSubject<boolean> = new BehaviorSubject<boolean>(false);\n  private readonly features: Record<string, BehaviorSubject<any>> = {};\n  private context: LDContext = { kind: 'user', anonymous: true };\n  private clientId: string = '';\n\n  public initialize(user: FeatureUser = { anonymous: true }, clientId: string): void {\n    this.ready.next(false);\n    this.context = { kind: 'user', ...user };\n    this.clientId = clientId;\n    this.client = initialize(this.clientId, this.context, { useReport: true });\n    this.client.on('ready', () => {\n      this.client.identify(this.context).then(() => this.ready.next(true));\n    });\n\n    this.client.on('error', (err: any) => {\n        console.error('LaunchDarkly client network error:', err);\n        // Allow the app to proceed by emitting true, so subscribers are not blocked\n        this.ready.next(true);\n      });\n  }\n\n  public isEnabled(feature: string, defaultValue: boolean = false): Observable<boolean> {\n    return this.getValue<boolean>(feature, defaultValue);\n  }\n\n  public getArray<R = any>(feature: string): Observable<R[]> {\n    return this.getValue<R[]>(feature, []);\n  }\n\n  // Note that this function always emits its default value first, which can lead to unexpected results\n  public getValue<R>(feature: string, defaultValue: R): Observable<R> {\n    if (!this.features.hasOwnProperty(feature)) {\n      this.features[feature] = new BehaviorSubject<R>(defaultValue);\n      this.ready.pipe(\n        filter((ready) => ready),\n        map(() => this.client.variation(feature, defaultValue))\n      ).subscribe((value) => {\n        this.features[feature].next(value);\n        this.client.on(`change:${feature}`, (val: R) => {\n          this.features[feature].next(val);\n        });\n      });\n    }\n    return this.features[feature].pipe(\n      distinctUntilChanged()\n    );\n  }\n\n  /**\n     * This method returns an observable that will only get the state of the feature toggle\n     * once. It calls the LD SDK directly, and should only be used in circumstances where\n     * only one value should be emitted, that value coming directly from LD. This will likely\n     * only apply for Guards, and should be used only when absolutely necessary.\n     * @see getValue for regular usage.\n     * @param feature string\n     * @param defaultValue R\n     */\n  public getValueOnce<R>(feature: string, defaultValue: R): Observable<R> {\n    return this.ready.pipe(\n      filter((ready) => ready),\n      map(() => this.client.variation(feature, defaultValue))\n    );\n  }\n\n  public getValueSync<R>(feature: string, defaultValue: R): R {\n    return this.client.variation(feature, defaultValue);\n  }\n}\n"]}
|
|
@@ -1394,7 +1394,7 @@ class FeatureToggleGuard {
|
|
|
1394
1394
|
* the redirecting navigation will now use replaceUrl if the initial navigation was also using the replaceUrl option.
|
|
1395
1395
|
*/
|
|
1396
1396
|
canActivate(route) {
|
|
1397
|
-
return combineLatest([...route.data.needsFeaturesEnabled.map(feature => this.featureToggleService.getValueOnce(feature,
|
|
1397
|
+
return combineLatest([...route.data.needsFeaturesEnabled.map(feature => this.featureToggleService.getValueOnce(feature, true))]).pipe(map(featureStatuses => featureStatuses.every(status => status)), map(status => (route.data.expectFeatureEnabled !== false && status) || (route.data.expectFeatureEnabled === false && !status) || this.router.parseUrl(route.data.featureDisabledRedirect)));
|
|
1398
1398
|
}
|
|
1399
1399
|
static { this.ɵfac = function FeatureToggleGuard_Factory(__ngFactoryType__) { return new (__ngFactoryType__ || FeatureToggleGuard)(i0.ɵɵinject(FeatureToggleService), i0.ɵɵinject(i2$1.Router)); }; }
|
|
1400
1400
|
static { this.ɵprov = /*@__PURE__*/ i0.ɵɵdefineInjectable({ token: FeatureToggleGuard, factory: FeatureToggleGuard.ɵfac, providedIn: 'root' }); }
|
|
@@ -1421,6 +1421,11 @@ class LaunchDarklyService {
|
|
|
1421
1421
|
this.client.on('ready', () => {
|
|
1422
1422
|
this.client.identify(this.context).then(() => this.ready.next(true));
|
|
1423
1423
|
});
|
|
1424
|
+
this.client.on('error', (err) => {
|
|
1425
|
+
console.error('LaunchDarkly client network error:', err);
|
|
1426
|
+
// Allow the app to proceed by emitting true, so subscribers are not blocked
|
|
1427
|
+
this.ready.next(true);
|
|
1428
|
+
});
|
|
1424
1429
|
}
|
|
1425
1430
|
isEnabled(feature, defaultValue = false) {
|
|
1426
1431
|
return this.getValue(feature, defaultValue);
|
|
@@ -6257,10 +6262,14 @@ class ServiceMessagesComponent {
|
|
|
6257
6262
|
}
|
|
6258
6263
|
createFilteredMessages(messages) {
|
|
6259
6264
|
this.hiddenBanners = this.getSessionCookie(this.serviceMessageCookie) || [];
|
|
6265
|
+
const messageCount = messages.length;
|
|
6260
6266
|
this.filteredMessages = messages.filter((message) => {
|
|
6261
6267
|
const { roles, message_en } = message;
|
|
6262
6268
|
const regEx = new RegExp(roles);
|
|
6263
|
-
|
|
6269
|
+
// If userRoles is empty or undefined, allow the message through
|
|
6270
|
+
const isRoleMatch = ((!this.userRoles || this.userRoles.length === 0) && messageCount === 1 && roles === '.+')
|
|
6271
|
+
|| (this.userRoles && this.userRoles.some((role) => regEx.test(role)));
|
|
6272
|
+
return isRoleMatch &&
|
|
6264
6273
|
this.compareDates(message) &&
|
|
6265
6274
|
!this.hiddenBanners.includes(message_en);
|
|
6266
6275
|
});
|