From f06609afc76657da33a1e73191f304349c926cbb Mon Sep 17 00:00:00 2001 From: gnibeda Date: Thu, 17 Aug 2023 15:46:24 +0300 Subject: [PATCH] * added workaround for invalid filter urls with `=&` characters * added option to encode filters as base64 string when sharing dashboard * fixed issue with invisible share button when opening the dashboard with filters --- package.json | 2 +- .../components/ui/header/header.component.ts | 2 +- .../share-dashboard.component.html | 3 +- .../share-dashboard.component.scss | 15 +++++ .../share-dashboard.component.ts | 56 +++++++++++++++++-- src/app/services/filter.service.ts | 12 ++++ src/changelog.md | 5 ++ 7 files changed, 87 insertions(+), 8 deletions(-) diff --git a/package.json b/package.json index ccef316d..76e017a5 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "deep-see-web", - "version": "3.2.5", + "version": "3.2.6", "scripts": { "ng": "ng", "start": "ng serve --proxy-config=proxy.conf.json --port 4007", diff --git a/src/app/components/ui/header/header.component.ts b/src/app/components/ui/header/header.component.ts index c63321db..077276a7 100644 --- a/src/app/components/ui/header/header.component.ts +++ b/src/app/components/ui/header/header.component.ts @@ -173,7 +173,7 @@ export class HeaderComponent implements OnInit, AfterViewInit, OnDestroy { * @param path */ private initSearch(path: string[]) { - this.isSearch = !path[path.length - 1]?.endsWith('.dashboard'); + this.isSearch = !path[path.length - 1]?.split('?')[0]?.endsWith('.dashboard'); } /** diff --git a/src/app/components/ui/share-dashboard/share-dashboard/share-dashboard.component.html b/src/app/components/ui/share-dashboard/share-dashboard/share-dashboard.component.html index 179b5aea..20923f43 100644 --- a/src/app/components/ui/share-dashboard/share-dashboard/share-dashboard.component.html +++ b/src/app/components/ui/share-dashboard/share-dashboard/share-dashboard.component.html @@ -1,5 +1,6 @@

{{title}}

- + +
diff --git a/src/app/components/ui/share-dashboard/share-dashboard/share-dashboard.component.scss b/src/app/components/ui/share-dashboard/share-dashboard/share-dashboard.component.scss index 46895c0e..23f085a2 100644 --- a/src/app/components/ui/share-dashboard/share-dashboard/share-dashboard.component.scss +++ b/src/app/components/ui/share-dashboard/share-dashboard/share-dashboard.component.scss @@ -44,3 +44,18 @@ textarea { button { width: 100%; } + +input { + min-width: 16px; + width: 16px; + display: inline-block; + margin: 0; + margin-right: 4px; + vertical-align: bottom +} + +label { + display: block; + margin-bottom: 10px; + font-weight: normal; +} diff --git a/src/app/components/ui/share-dashboard/share-dashboard/share-dashboard.component.ts b/src/app/components/ui/share-dashboard/share-dashboard/share-dashboard.component.ts index 17f99a7f..9a91f66e 100644 --- a/src/app/components/ui/share-dashboard/share-dashboard/share-dashboard.component.ts +++ b/src/app/components/ui/share-dashboard/share-dashboard/share-dashboard.component.ts @@ -1,12 +1,14 @@ -import {Component, Input, OnDestroy, OnInit} from '@angular/core'; +import {ChangeDetectionStrategy, Component, Input, OnChanges, OnDestroy, OnInit, SimpleChanges} from '@angular/core'; import {UtilService} from "../../../../services/util.service"; +import {StorageService} from "../../../../services/storage.service"; @Component({ selector: 'dsw-share-dashboard', templateUrl: './share-dashboard.component.html', - styleUrls: ['./share-dashboard.component.scss'] + styleUrls: ['./share-dashboard.component.scss'], + changeDetection: ChangeDetectionStrategy.OnPush }) -export class ShareDashboardComponent implements OnDestroy { +export class ShareDashboardComponent implements OnInit, OnChanges, OnDestroy { @Input() title = 'Share dashboard'; @Input() btnTitle = 'Copy link'; @Input() shareUrl = ''; @@ -14,12 +16,26 @@ export class ShareDashboardComponent implements OnDestroy { onCopy = () => {}; isCopied = false; timeout = 0; + asBase64 = false; + url = ''; - constructor(private us: UtilService) { + constructor(private us: UtilService, + private ss: StorageService) { + this.asBase64 = this.ss.storage.getItem('dsw-share-format-base64') === '1'; + } + + ngOnInit() { + this.convertLink(); + } + + ngOnChanges(changes: SimpleChanges) { + if (changes['shareUrl'].previousValue !== changes['shareUrl'].currentValue) { + this.convertLink(); + } } onCopyClick() { - this.us.copyToClipboard(this.shareUrl); + this.us.copyToClipboard(this.url); this.isCopied = true; clearTimeout(this.timeout); this.timeout = setTimeout(() => { @@ -32,4 +48,34 @@ export class ShareDashboardComponent implements OnDestroy { ngOnDestroy() { clearTimeout(this.timeout); } + + onFormatChange() { + this.ss.storage.setItem('dsw-share-format-base64', this.asBase64 ? '1' : '0'); + this.convertLink(); + } + + private convertLink() { + this.url = this.shareUrl; + if (this.asBase64) { + const parts = this.shareUrl.split('?'); + const query = parts[1]; + if (!query) { + return; + } + let params = query.split('&'); + params = params.map(p => { + const pa = p.split('='); + if (pa[0] === 'FILTERS') { + pa[1] = btoa(pa[1]); + return pa.join('='); + } else { + return p; + } + }); + parts[1] = params.join('&'); + this.url = parts.join('?'); + } else { + this.url = this.shareUrl; + } + } } diff --git a/src/app/services/filter.service.ts b/src/app/services/filter.service.ts index 444a4d48..7ad21a68 100644 --- a/src/app/services/filter.service.ts +++ b/src/app/services/filter.service.ts @@ -164,6 +164,13 @@ export class FilterService { return encodeURIComponent(f.join(seporator)); } + private isBase64(str: string) { + try { + return btoa(atob(str)) === str; + } catch (e) { + return false; + } + } loadFiltersFromUrl() { let query = window.location.hash.split('?')[1]; @@ -171,7 +178,9 @@ export class FilterService { return; } query = query.replace(/\.&%5B/g, '.%26%5B'); + query = query.replace(/\.=&%5B/g, '.%26%5B'); query = query.replace(/\.&\[/g, '.%26%5B'); + query = query.replace(/\.=&\[/g, '.%26%5B'); const p = query.split('&'); let param = ''; p.forEach(q => { @@ -180,6 +189,9 @@ export class FilterService { }; }); + if (this.isBase64(param)) { + param = atob(param); + } //let param = this.route.snapshot.queryParamMap.get('FILTERS'); if (!param) { // Workaround for invalid escaped links where "=" char is escaped. Requested by Shvarov diff --git a/src/changelog.md b/src/changelog.md index 84f4d459..f8e026e8 100644 --- a/src/changelog.md +++ b/src/changelog.md @@ -1,3 +1,8 @@ +#### 3.2.6 +* added workaround for invalid filter urls with `=&` characters +* added option to encode filters as base64 string when sharing dashboard +* fixed issue with invisible share button when opening the dashboard with filters + #### 3.2.5 * added workaround for invalid filter urls with `&` character * fixed issue with dashboard loading without filters