cloud-ide-layout 0.0.1 → 1.0.1
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/README.md +466 -13
- package/esm2022/lib/cloud-ide-layout.routes.mjs +24 -10
- package/esm2022/lib/layout/console/console-wrapper/console-wrapper.component.mjs +6 -6
- package/esm2022/lib/layout/console/console.service.mjs +28 -0
- package/esm2022/lib/layout/footer/footer-wrapper/footer-wrapper.component.mjs +16 -8
- package/esm2022/lib/layout/footer/footer.service.mjs +28 -0
- package/esm2022/lib/layout/header/header-wrapper/header-wrapper.component.mjs +69 -18
- package/esm2022/lib/layout/header/header.service.mjs +17 -6
- package/esm2022/lib/layout/home/home-wrapper/home-wrapper.component.mjs +194 -7
- package/esm2022/lib/layout/layout/layout-wrapper/layout-wrapper.component.mjs +18 -9
- package/esm2022/lib/layout/request/request-wrapper/request-wrapper.component.mjs +54 -6
- package/esm2022/lib/layout/request/request.service.mjs +235 -0
- package/esm2022/lib/layout/request/tab-content/tab-content.component.mjs +84 -0
- package/esm2022/lib/layout/shared/shared-wrapper/shared-wrapper.component.mjs +32 -20
- package/esm2022/lib/layout/shared/shared.service.mjs +490 -14
- package/esm2022/lib/layout/sidebar/sidebar-wrapper/sidebar-wrapper.component.mjs +480 -15
- package/esm2022/lib/layout/sidebar/sidebar.service.mjs +20 -7
- package/esm2022/lib/layout/sidedrawer/cide-lyt-drawer-base.class.mjs +40 -0
- package/esm2022/lib/layout/sidedrawer/drawer-theme/drawer-theme.component.mjs +296 -0
- package/esm2022/lib/layout/sidedrawer/sidedrawer-notes/sidedrawer-notes.component.mjs +175 -0
- package/esm2022/lib/layout/sidedrawer/sidedrawer-notes/sidedrawer-notes.service.mjs +51 -0
- package/esm2022/lib/layout/sidedrawer/sidedrawer-wrapper/sidedrawer-wrapper.component.mjs +254 -9
- package/esm2022/lib/layout/sidedrawer/sidedrawer.service.mjs +70 -0
- package/esm2022/lib/services/app-state/app-state-helper.service.mjs +222 -0
- package/esm2022/lib/services/app-state/app-state.service.mjs +256 -0
- package/esm2022/lib/services/app-state/index.mjs +3 -0
- package/esm2022/lib/services/cache-manager/cache-manager.service.mjs +112 -0
- package/esm2022/lib/services/file-manager/file-manager.service.mjs +7 -7
- package/esm2022/lib/services/tab-state.service.mjs +128 -0
- package/esm2022/lib/services/theme/theme.service.mjs +151 -0
- package/esm2022/lib/services/user-status/user-status.service.mjs +135 -0
- package/esm2022/lib/utils/custom-route-reuse-strategy.mjs +61 -0
- package/esm2022/lib/utils/index.mjs +2 -0
- package/esm2022/public-api.mjs +7 -1
- package/fesm2022/cloud-ide-layout-cloud-ide-layout-DqL3A73V.mjs +2914 -0
- package/fesm2022/cloud-ide-layout-cloud-ide-layout-DqL3A73V.mjs.map +1 -0
- package/fesm2022/cloud-ide-layout-drawer-theme.component-xd8IG3WP.mjs +443 -0
- package/fesm2022/cloud-ide-layout-drawer-theme.component-xd8IG3WP.mjs.map +1 -0
- package/fesm2022/cloud-ide-layout-home-wrapper.component-DEdsEiaq.mjs +197 -0
- package/fesm2022/cloud-ide-layout-home-wrapper.component-DEdsEiaq.mjs.map +1 -0
- package/fesm2022/cloud-ide-layout-sidedrawer-notes.component-ds12wZbh.mjs +260 -0
- package/fesm2022/cloud-ide-layout-sidedrawer-notes.component-ds12wZbh.mjs.map +1 -0
- package/fesm2022/cloud-ide-layout.mjs +1 -261
- package/fesm2022/cloud-ide-layout.mjs.map +1 -1
- package/lib/cloud-ide-layout.routes.d.ts +1 -0
- package/lib/layout/console/console-wrapper/console-wrapper.component.d.ts +3 -3
- package/lib/layout/console/console.service.d.ts +9 -0
- package/lib/layout/footer/footer-wrapper/footer-wrapper.component.d.ts +7 -3
- package/lib/layout/footer/footer.service.d.ts +9 -0
- package/lib/layout/header/header-wrapper/header-wrapper.component.d.ts +17 -9
- package/lib/layout/header/header.service.d.ts +4 -3
- package/lib/layout/home/home-wrapper/home-wrapper.component.d.ts +90 -1
- package/lib/layout/request/request-wrapper/request-wrapper.component.d.ts +10 -1
- package/lib/layout/request/request.service.d.ts +60 -0
- package/lib/layout/request/tab-content/tab-content.component.d.ts +19 -0
- package/lib/layout/shared/shared-wrapper/shared-wrapper.component.d.ts +4 -5
- package/lib/layout/shared/shared.service.d.ts +104 -6
- package/lib/layout/sidebar/sidebar-wrapper/sidebar-wrapper.component.d.ts +135 -6
- package/lib/layout/sidebar/sidebar.service.d.ts +3 -2
- package/lib/layout/sidedrawer/cide-lyt-drawer-base.class.d.ts +16 -0
- package/lib/layout/sidedrawer/drawer-theme/drawer-theme.component.d.ts +71 -0
- package/lib/layout/sidedrawer/sidedrawer-notes/sidedrawer-notes.component.d.ts +48 -0
- package/lib/layout/sidedrawer/sidedrawer-notes/sidedrawer-notes.service.d.ts +13 -0
- package/lib/layout/sidedrawer/sidedrawer-wrapper/sidedrawer-wrapper.component.d.ts +28 -3
- package/lib/layout/sidedrawer/sidedrawer.service.d.ts +30 -0
- package/lib/services/app-state/app-state-helper.service.d.ts +150 -0
- package/lib/services/app-state/app-state.service.d.ts +85 -0
- package/lib/services/app-state/index.d.ts +3 -0
- package/lib/services/cache-manager/cache-manager.service.d.ts +64 -0
- package/lib/services/file-manager/file-manager.service.d.ts +0 -2
- package/lib/services/tab-state.service.d.ts +79 -0
- package/lib/services/theme/theme.service.d.ts +71 -0
- package/lib/services/user-status/user-status.service.d.ts +57 -0
- package/lib/utils/custom-route-reuse-strategy.d.ts +15 -0
- package/lib/utils/index.d.ts +1 -0
- package/package.json +2 -2
- package/public-api.d.ts +4 -0
- package/src/lib/assets/css/cide-lyt-style.scss +182 -0
- package/src/lib/assets/css/cide-lyt-variable.scss +119 -0
- package/fesm2022/cloud-ide-layout-home-wrapper.component-BV7ptL0Y.mjs +0 -14
- package/fesm2022/cloud-ide-layout-home-wrapper.component-BV7ptL0Y.mjs.map +0 -1
|
@@ -0,0 +1,112 @@
|
|
|
1
|
+
import { Injectable, inject, computed } from '@angular/core';
|
|
2
|
+
import { CideLytSharedService } from '../../layout/shared/shared.service';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
export class CacheManagerService {
|
|
5
|
+
constructor() {
|
|
6
|
+
this.sharedService = inject(CideLytSharedService);
|
|
7
|
+
// Computed cache statistics
|
|
8
|
+
this.cacheStats = computed(() => {
|
|
9
|
+
const info = this.sharedService.cacheInfo();
|
|
10
|
+
return {
|
|
11
|
+
...info,
|
|
12
|
+
hitRatio: info.totalCached > 0 ? (info.validCached / info.totalCached) * 100 : 0,
|
|
13
|
+
memoryEstimate: this.estimateCacheMemoryUsage()
|
|
14
|
+
};
|
|
15
|
+
});
|
|
16
|
+
// Public access to cache state
|
|
17
|
+
this.isLoading = (pageCode) => this.sharedService.isLoading(pageCode);
|
|
18
|
+
this.getCachedData = (pageCode) => this.sharedService.getCachedPageData(pageCode);
|
|
19
|
+
}
|
|
20
|
+
/**
|
|
21
|
+
* Get page data with caching - this is the main method components should use
|
|
22
|
+
*/
|
|
23
|
+
getPageData(pageCode, forceRefresh = false) {
|
|
24
|
+
return this.sharedService.loadAndProcessPageData({ sypg_page_code: pageCode }, { forceRefresh, setTitle: true, setSidebarContext: true, updateLayout: true });
|
|
25
|
+
}
|
|
26
|
+
/**
|
|
27
|
+
* Preload multiple pages for better performance
|
|
28
|
+
*/
|
|
29
|
+
preloadPages(pageCodes) {
|
|
30
|
+
console.log(`🚀 PRELOADING: Starting preload for ${pageCodes.length} pages`);
|
|
31
|
+
this.sharedService.preloadPageData(pageCodes);
|
|
32
|
+
}
|
|
33
|
+
/**
|
|
34
|
+
* Invalidate specific page cache
|
|
35
|
+
*/
|
|
36
|
+
invalidatePage(pageCode) {
|
|
37
|
+
this.sharedService.invalidateCache(pageCode);
|
|
38
|
+
}
|
|
39
|
+
/**
|
|
40
|
+
* Clear all cache
|
|
41
|
+
*/
|
|
42
|
+
clearCache() {
|
|
43
|
+
this.sharedService.clearAllCache();
|
|
44
|
+
}
|
|
45
|
+
/**
|
|
46
|
+
* Get cache status for debugging
|
|
47
|
+
*/
|
|
48
|
+
getCacheStatus() {
|
|
49
|
+
const cache = this.sharedService.pageDataCache();
|
|
50
|
+
const loadingStates = this.sharedService.loadingStates();
|
|
51
|
+
const allPageCodes = new Set([
|
|
52
|
+
...Object.keys(cache),
|
|
53
|
+
...Object.keys(loadingStates)
|
|
54
|
+
]);
|
|
55
|
+
return Array.from(allPageCodes).map(pageCode => ({
|
|
56
|
+
pageCode,
|
|
57
|
+
cached: !!cache[pageCode],
|
|
58
|
+
expired: cache[pageCode] ? !this.isCacheValid(cache[pageCode]) : false,
|
|
59
|
+
loading: !!loadingStates[pageCode]
|
|
60
|
+
}));
|
|
61
|
+
}
|
|
62
|
+
/**
|
|
63
|
+
* Estimate memory usage of cache (rough estimate)
|
|
64
|
+
*/
|
|
65
|
+
estimateCacheMemoryUsage() {
|
|
66
|
+
const cache = this.sharedService.pageDataCache();
|
|
67
|
+
const jsonString = JSON.stringify(cache);
|
|
68
|
+
const bytes = new Blob([jsonString]).size;
|
|
69
|
+
if (bytes < 1024)
|
|
70
|
+
return `${bytes} B`;
|
|
71
|
+
if (bytes < 1024 * 1024)
|
|
72
|
+
return `${(bytes / 1024).toFixed(1)} KB`;
|
|
73
|
+
return `${(bytes / (1024 * 1024)).toFixed(1)} MB`;
|
|
74
|
+
}
|
|
75
|
+
/**
|
|
76
|
+
* Check if cache entry is valid
|
|
77
|
+
*/
|
|
78
|
+
isCacheValid(cached) {
|
|
79
|
+
if (!cached)
|
|
80
|
+
return false;
|
|
81
|
+
return Date.now() - cached.timestamp < cached.ttl;
|
|
82
|
+
}
|
|
83
|
+
/**
|
|
84
|
+
* Export cache data for debugging or backup
|
|
85
|
+
*/
|
|
86
|
+
exportCacheData() {
|
|
87
|
+
return JSON.stringify(this.sharedService.pageDataCache(), null, 2);
|
|
88
|
+
}
|
|
89
|
+
/**
|
|
90
|
+
* Get performance metrics
|
|
91
|
+
*/
|
|
92
|
+
getPerformanceMetrics() {
|
|
93
|
+
const stats = this.cacheStats();
|
|
94
|
+
const loadingStates = this.sharedService.loadingStates();
|
|
95
|
+
return {
|
|
96
|
+
totalPages: stats.totalCached,
|
|
97
|
+
cachedPages: stats.validCached,
|
|
98
|
+
loadingPages: Object.keys(loadingStates).length,
|
|
99
|
+
cacheHitRatio: stats.hitRatio,
|
|
100
|
+
// averageResponseTime could be implemented with timing tracking
|
|
101
|
+
};
|
|
102
|
+
}
|
|
103
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: CacheManagerService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
104
|
+
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: CacheManagerService, providedIn: 'root' }); }
|
|
105
|
+
}
|
|
106
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: CacheManagerService, decorators: [{
|
|
107
|
+
type: Injectable,
|
|
108
|
+
args: [{
|
|
109
|
+
providedIn: 'root'
|
|
110
|
+
}]
|
|
111
|
+
}] });
|
|
112
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cache-manager.service.js","sourceRoot":"","sources":["../../../../../../projects/cloud-ide-layout/src/lib/services/cache-manager/cache-manager.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, inject, computed } from '@angular/core';\r\nimport { CideLytSharedService } from '../../layout/shared/shared.service';\r\nimport { Observable } from 'rxjs';\r\nimport { designConfigControllerResponse } from 'cloud-ide-lms-model';\r\n\r\n@Injectable({\r\n  providedIn: 'root'\r\n})\r\nexport class CacheManagerService {\r\n  private sharedService = inject(CideLytSharedService);\r\n\r\n  // Computed cache statistics\r\n  public readonly cacheStats = computed(() => {\r\n    const info = this.sharedService.cacheInfo();\r\n    return {\r\n      ...info,\r\n      hitRatio: info.totalCached > 0 ? (info.validCached / info.totalCached) * 100 : 0,\r\n      memoryEstimate: this.estimateCacheMemoryUsage()\r\n    };\r\n  });\r\n\r\n  // Public access to cache state\r\n  public readonly isLoading = (pageCode: string) => this.sharedService.isLoading(pageCode);\r\n  public readonly getCachedData = (pageCode: string) => this.sharedService.getCachedPageData(pageCode);\r\n\r\n  /**\r\n   * Get page data with caching - this is the main method components should use\r\n   */\r\n  getPageData(pageCode: string, forceRefresh: boolean = false): Observable<designConfigControllerResponse> {\r\n    return this.sharedService.loadAndProcessPageData(\r\n      { sypg_page_code: pageCode },\r\n      { forceRefresh, setTitle: true, setSidebarContext: true, updateLayout: true }\r\n    );\r\n  }\r\n\r\n  /**\r\n   * Preload multiple pages for better performance\r\n   */\r\n  preloadPages(pageCodes: string[]): void {\r\n    console.log(`🚀 PRELOADING: Starting preload for ${pageCodes.length} pages`);\r\n    this.sharedService.preloadPageData(pageCodes);\r\n  }\r\n\r\n  /**\r\n   * Invalidate specific page cache\r\n   */\r\n  invalidatePage(pageCode: string): void {\r\n    this.sharedService.invalidateCache(pageCode);\r\n  }\r\n\r\n  /**\r\n   * Clear all cache\r\n   */\r\n  clearCache(): void {\r\n    this.sharedService.clearAllCache();\r\n  }\r\n\r\n  /**\r\n   * Get cache status for debugging\r\n   */\r\n  getCacheStatus(): { pageCode: string; cached: boolean; expired: boolean; loading: boolean }[] {\r\n    const cache = this.sharedService.pageDataCache();\r\n    const loadingStates = this.sharedService.loadingStates();\r\n    \r\n    const allPageCodes = new Set([\r\n      ...Object.keys(cache),\r\n      ...Object.keys(loadingStates)\r\n    ]);\r\n\r\n    return Array.from(allPageCodes).map(pageCode => ({\r\n      pageCode,\r\n      cached: !!cache[pageCode],\r\n      expired: cache[pageCode] ? !this.isCacheValid(cache[pageCode]) : false,\r\n      loading: !!loadingStates[pageCode]\r\n    }));\r\n  }\r\n\r\n  /**\r\n   * Estimate memory usage of cache (rough estimate)\r\n   */\r\n  private estimateCacheMemoryUsage(): string {\r\n    const cache = this.sharedService.pageDataCache();\r\n    const jsonString = JSON.stringify(cache);\r\n    const bytes = new Blob([jsonString]).size;\r\n    \r\n    if (bytes < 1024) return `${bytes} B`;\r\n    if (bytes < 1024 * 1024) return `${(bytes / 1024).toFixed(1)} KB`;\r\n    return `${(bytes / (1024 * 1024)).toFixed(1)} MB`;\r\n  }\r\n\r\n  /**\r\n   * Check if cache entry is valid\r\n   */\r\n  private isCacheValid(cached: { timestamp: number; ttl: number } | undefined): boolean {\r\n    if (!cached) return false;\r\n    return Date.now() - cached.timestamp < cached.ttl;\r\n  }\r\n\r\n  /**\r\n   * Export cache data for debugging or backup\r\n   */\r\n  exportCacheData(): string {\r\n    return JSON.stringify(this.sharedService.pageDataCache(), null, 2);\r\n  }\r\n\r\n  /**\r\n   * Get performance metrics\r\n   */\r\n  getPerformanceMetrics(): {\r\n    totalPages: number;\r\n    cachedPages: number;\r\n    loadingPages: number;\r\n    cacheHitRatio: number;\r\n    averageResponseTime?: number;\r\n  } {\r\n    const stats = this.cacheStats();\r\n    const loadingStates = this.sharedService.loadingStates();\r\n    \r\n    return {\r\n      totalPages: stats.totalCached,\r\n      cachedPages: stats.validCached,\r\n      loadingPages: Object.keys(loadingStates).length,\r\n      cacheHitRatio: stats.hitRatio,\r\n      // averageResponseTime could be implemented with timing tracking\r\n    };\r\n  }\r\n} "]}
|
|
@@ -1,15 +1,15 @@
|
|
|
1
|
-
import {
|
|
1
|
+
import { HttpClient } from '@angular/common/http';
|
|
2
|
+
import { Injectable, inject } from '@angular/core';
|
|
2
3
|
import { cidePath, coreRoutesUrl, hostManagerRoutesUrl } from 'cloud-ide-lms-model';
|
|
3
4
|
import * as i0 from "@angular/core";
|
|
4
|
-
import * as i1 from "@angular/common/http";
|
|
5
5
|
export class CideLytFileManagerService {
|
|
6
|
-
constructor(
|
|
7
|
-
this.http =
|
|
6
|
+
constructor() {
|
|
7
|
+
this.http = inject(HttpClient);
|
|
8
8
|
}
|
|
9
9
|
getFileDetails(body) {
|
|
10
10
|
return this.http.post(cidePath.join([hostManagerRoutesUrl.cideSuiteHost, coreRoutesUrl?.module, coreRoutesUrl?.getFileDetails]), body);
|
|
11
11
|
}
|
|
12
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: CideLytFileManagerService, deps: [
|
|
12
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: CideLytFileManagerService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
13
13
|
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: CideLytFileManagerService, providedIn: 'root' }); }
|
|
14
14
|
}
|
|
15
15
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: CideLytFileManagerService, decorators: [{
|
|
@@ -17,5 +17,5 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.7", ngImpor
|
|
|
17
17
|
args: [{
|
|
18
18
|
providedIn: 'root'
|
|
19
19
|
}]
|
|
20
|
-
}]
|
|
21
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
20
|
+
}] });
|
|
21
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZmlsZS1tYW5hZ2VyLnNlcnZpY2UuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9jbG91ZC1pZGUtbGF5b3V0L3NyYy9saWIvc2VydmljZXMvZmlsZS1tYW5hZ2VyL2ZpbGUtbWFuYWdlci5zZXJ2aWNlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxVQUFVLEVBQUUsTUFBTSxzQkFBc0IsQ0FBQztBQUNsRCxPQUFPLEVBQUUsVUFBVSxFQUFFLE1BQU0sRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUNuRCxPQUFPLEVBQUUsUUFBUSxFQUFFLGFBQWEsRUFBRSxvQkFBb0IsRUFBK0MsTUFBTSxxQkFBcUIsQ0FBQzs7QUFLakksTUFBTSxPQUFPLHlCQUF5QjtJQUh0QztRQUlVLFNBQUksR0FBRyxNQUFNLENBQUMsVUFBVSxDQUFDLENBQUM7S0FLbkM7SUFIQyxjQUFjLENBQUMsSUFBa0I7UUFDL0IsT0FBTyxJQUFJLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxRQUFRLENBQUMsSUFBSSxDQUFDLENBQUMsb0JBQW9CLENBQUMsYUFBYSxFQUFFLGFBQWEsRUFBRSxNQUFNLEVBQUUsYUFBYSxFQUFFLGNBQWMsQ0FBQyxDQUFDLEVBQUUsSUFBSSxDQUE4QyxDQUFDO0lBQ3RMLENBQUM7OEdBTFUseUJBQXlCO2tIQUF6Qix5QkFBeUIsY0FGeEIsTUFBTTs7MkZBRVAseUJBQXlCO2tCQUhyQyxVQUFVO21CQUFDO29CQUNWLFVBQVUsRUFBRSxNQUFNO2lCQUNuQiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEh0dHBDbGllbnQgfSBmcm9tICdAYW5ndWxhci9jb21tb24vaHR0cCc7XG5pbXBvcnQgeyBJbmplY3RhYmxlLCBpbmplY3QgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IGNpZGVQYXRoLCBjb3JlUm91dGVzVXJsLCBob3N0TWFuYWdlclJvdXRlc1VybCwgTUZpbGVNYW5hZ2VyLCBmaWxlTWFuYWdlckNvbnRyb2xsZXJSZXNwb25zZSB9IGZyb20gJ2Nsb3VkLWlkZS1sbXMtbW9kZWwnO1xuaW1wb3J0IHsgT2JzZXJ2YWJsZSB9IGZyb20gJ3J4anMnO1xuQEluamVjdGFibGUoe1xuICBwcm92aWRlZEluOiAncm9vdCdcbn0pXG5leHBvcnQgY2xhc3MgQ2lkZUx5dEZpbGVNYW5hZ2VyU2VydmljZSB7XG4gIHByaXZhdGUgaHR0cCA9IGluamVjdChIdHRwQ2xpZW50KTtcblxuICBnZXRGaWxlRGV0YWlscyhib2R5OiBNRmlsZU1hbmFnZXIpOiBPYnNlcnZhYmxlPGZpbGVNYW5hZ2VyQ29udHJvbGxlclJlc3BvbnNlPiB7XG4gICAgcmV0dXJuIHRoaXMuaHR0cC5wb3N0KGNpZGVQYXRoLmpvaW4oW2hvc3RNYW5hZ2VyUm91dGVzVXJsLmNpZGVTdWl0ZUhvc3QsIGNvcmVSb3V0ZXNVcmw/Lm1vZHVsZSwgY29yZVJvdXRlc1VybD8uZ2V0RmlsZURldGFpbHNdKSwgYm9keSkgYXMgT2JzZXJ2YWJsZTxmaWxlTWFuYWdlckNvbnRyb2xsZXJSZXNwb25zZT47XG4gIH1cbn1cbiJdfQ==
|
|
@@ -0,0 +1,128 @@
|
|
|
1
|
+
import { Injectable } from '@angular/core';
|
|
2
|
+
import { BehaviorSubject } from 'rxjs';
|
|
3
|
+
import { distinctUntilChanged, map } from 'rxjs/operators';
|
|
4
|
+
import { merge } from 'lodash';
|
|
5
|
+
import * as i0 from "@angular/core";
|
|
6
|
+
/**
|
|
7
|
+
* Creates the default state for a new tab.
|
|
8
|
+
*/
|
|
9
|
+
const createDefaultComponentState = () => ({
|
|
10
|
+
sideDrawer: { isOpen: false, activeComponent: null, componentState: {} },
|
|
11
|
+
footer: { someData: '' },
|
|
12
|
+
console: { history: [] },
|
|
13
|
+
});
|
|
14
|
+
export class TabStateService {
|
|
15
|
+
/**
|
|
16
|
+
* Adds a new tab to the state.
|
|
17
|
+
* @param title The title for the new tab.
|
|
18
|
+
* @param id An optional unique ID. If not provided, one will be generated.
|
|
19
|
+
* @returns The ID of the newly created tab.
|
|
20
|
+
*/
|
|
21
|
+
addTab(title, id) {
|
|
22
|
+
const newTabId = id || `tab_${Date.now()}_${Math.random()}`;
|
|
23
|
+
const newTab = {
|
|
24
|
+
id: newTabId,
|
|
25
|
+
title,
|
|
26
|
+
componentState: createDefaultComponentState(),
|
|
27
|
+
};
|
|
28
|
+
const currentTabs = this._tabsState.getValue();
|
|
29
|
+
const newTabs = new Map(currentTabs).set(newTabId, newTab);
|
|
30
|
+
this._tabsState.next(newTabs);
|
|
31
|
+
this.setActiveTab(newTabId);
|
|
32
|
+
return newTabId;
|
|
33
|
+
}
|
|
34
|
+
/**
|
|
35
|
+
* Removes a tab from the state.
|
|
36
|
+
* @param tabIdToRemove The ID of the tab to remove.
|
|
37
|
+
*/
|
|
38
|
+
removeTab(tabIdToRemove) {
|
|
39
|
+
const currentTabs = this._tabsState.getValue();
|
|
40
|
+
if (!currentTabs.has(tabIdToRemove)) {
|
|
41
|
+
return;
|
|
42
|
+
}
|
|
43
|
+
const newTabs = new Map(currentTabs);
|
|
44
|
+
newTabs.delete(tabIdToRemove);
|
|
45
|
+
this._tabsState.next(newTabs);
|
|
46
|
+
}
|
|
47
|
+
constructor() {
|
|
48
|
+
// Holds the state for all open tabs, keyed by tab ID.
|
|
49
|
+
this._tabsState = new BehaviorSubject(new Map());
|
|
50
|
+
// Holds the ID of the currently active tab.
|
|
51
|
+
this._activeTabId = new BehaviorSubject(null);
|
|
52
|
+
/** An observable of all open tabs. */
|
|
53
|
+
this.tabs$ = this._tabsState.pipe(map(tabsMap => Array.from(tabsMap.values())));
|
|
54
|
+
/** An observable of the currently active tab's ID. */
|
|
55
|
+
this.activeTabId$ = this._activeTabId.asObservable().pipe(distinctUntilChanged());
|
|
56
|
+
/** An observable of the state of the currently active tab. */
|
|
57
|
+
this.activeTabState$ = this.activeTabId$.pipe(map(activeId => (activeId ? this._tabsState.getValue().get(activeId) ?? null : null)),
|
|
58
|
+
// Use a deep-ish comparison to prevent emissions when only the object reference changes
|
|
59
|
+
distinctUntilChanged((prev, curr) => JSON.stringify(prev) === JSON.stringify(curr)));
|
|
60
|
+
/** An observable of just the component-specific state for the active tab. */
|
|
61
|
+
this.activeComponentState$ = this.activeTabState$.pipe(map(activeTab => activeTab?.componentState ?? null));
|
|
62
|
+
// You can initialize with default tabs if needed
|
|
63
|
+
// For example:
|
|
64
|
+
// this.addTab('Welcome Tab');
|
|
65
|
+
}
|
|
66
|
+
/**
|
|
67
|
+
* Sets the currently active tab.
|
|
68
|
+
* @param tabId The unique identifier of the tab to activate.
|
|
69
|
+
*/
|
|
70
|
+
setActiveTab(tabId) {
|
|
71
|
+
if (this._activeTabId.getValue() !== tabId) {
|
|
72
|
+
this._activeTabId.next(tabId);
|
|
73
|
+
}
|
|
74
|
+
}
|
|
75
|
+
/**
|
|
76
|
+
* Updates the state for the currently active tab.
|
|
77
|
+
* This performs a deep merge to avoid overwriting nested state.
|
|
78
|
+
* @param partialState The partial state to update for the active tab.
|
|
79
|
+
*/
|
|
80
|
+
updateActiveTabState(partialState) {
|
|
81
|
+
const activeId = this._activeTabId.getValue();
|
|
82
|
+
if (!activeId) {
|
|
83
|
+
console.warn('Cannot update state, no active tab.');
|
|
84
|
+
return;
|
|
85
|
+
}
|
|
86
|
+
const tabs = this._tabsState.getValue();
|
|
87
|
+
const activeTabState = tabs.get(activeId);
|
|
88
|
+
if (activeTabState) {
|
|
89
|
+
// Use lodash merge for a deep merge
|
|
90
|
+
const updatedState = merge({}, activeTabState, { componentState: partialState });
|
|
91
|
+
const newTabsState = new Map(tabs);
|
|
92
|
+
newTabsState.set(activeId, updatedState);
|
|
93
|
+
this._tabsState.next(newTabsState);
|
|
94
|
+
}
|
|
95
|
+
}
|
|
96
|
+
/**
|
|
97
|
+
* Gets an observable for a specific tab's state.
|
|
98
|
+
* @param tabId The ID of the tab to observe.
|
|
99
|
+
*/
|
|
100
|
+
getTabState(tabId) {
|
|
101
|
+
return this._tabsState.pipe(map(tabs => tabs.get(tabId)));
|
|
102
|
+
}
|
|
103
|
+
/**
|
|
104
|
+
* Determines the ID of the next tab to activate when a tab is closed.
|
|
105
|
+
* @param idToRemove The ID of the tab being removed.
|
|
106
|
+
* @returns The ID of the next tab to activate, or null if no tabs are left.
|
|
107
|
+
*/
|
|
108
|
+
getNextActiveTabId(idToRemove) {
|
|
109
|
+
const tabsMap = this._tabsState.getValue();
|
|
110
|
+
if (tabsMap.size <= 1) {
|
|
111
|
+
return null;
|
|
112
|
+
}
|
|
113
|
+
const tabIds = Array.from(tabsMap.keys());
|
|
114
|
+
const removedIndex = tabIds.indexOf(idToRemove);
|
|
115
|
+
// If it's the last tab in the list, activate the one before it. Otherwise, activate the next one.
|
|
116
|
+
const nextIndex = removedIndex === tabIds.length - 1 ? removedIndex - 1 : removedIndex + 1;
|
|
117
|
+
return tabIds[nextIndex];
|
|
118
|
+
}
|
|
119
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: TabStateService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
120
|
+
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: TabStateService, providedIn: 'root' }); }
|
|
121
|
+
}
|
|
122
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: TabStateService, decorators: [{
|
|
123
|
+
type: Injectable,
|
|
124
|
+
args: [{
|
|
125
|
+
providedIn: 'root',
|
|
126
|
+
}]
|
|
127
|
+
}], ctorParameters: () => [] });
|
|
128
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tab-state.service.js","sourceRoot":"","sources":["../../../../../projects/cloud-ide-layout/src/lib/services/tab-state.service.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,eAAe,CAAC;AAC3C,OAAO,EAAE,eAAe,EAAc,MAAM,MAAM,CAAC;AACnD,OAAO,EAAE,oBAAoB,EAAE,GAAG,EAAE,MAAM,gBAAgB,CAAC;AAC3D,OAAO,EAAE,KAAK,EAAE,MAAM,QAAQ,CAAC;;AA8B/B;;GAEG;AACH,MAAM,2BAA2B,GAAG,GAAgB,EAAE,CAAC,CAAC;IACtD,UAAU,EAAE,EAAE,MAAM,EAAE,KAAK,EAAE,eAAe,EAAE,IAAI,EAAE,cAAc,EAAE,EAAE,EAAE;IACxE,MAAM,EAAE,EAAE,QAAQ,EAAE,EAAE,EAAE;IACxB,OAAO,EAAE,EAAE,OAAO,EAAE,EAAE,EAAE;CACzB,CAAC,CAAC;AAKH,MAAM,OAAO,eAAe;IA2B1B;;;;;OAKG;IACI,MAAM,CAAC,KAAa,EAAE,EAAW;QACtC,MAAM,QAAQ,GAAG,EAAE,IAAI,OAAO,IAAI,CAAC,GAAG,EAAE,IAAI,IAAI,CAAC,MAAM,EAAE,EAAE,CAAC;QAC5D,MAAM,MAAM,GAAa;YACvB,EAAE,EAAE,QAAQ;YACZ,KAAK;YACL,cAAc,EAAE,2BAA2B,EAAE;SAC9C,CAAC;QAEF,MAAM,WAAW,GAAG,IAAI,CAAC,UAAU,CAAC,QAAQ,EAAE,CAAC;QAC/C,MAAM,OAAO,GAAG,IAAI,GAAG,CAAC,WAAW,CAAC,CAAC,GAAG,CAAC,QAAQ,EAAE,MAAM,CAAC,CAAC;QAC3D,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QAC9B,IAAI,CAAC,YAAY,CAAC,QAAQ,CAAC,CAAC;QAE5B,OAAO,QAAQ,CAAC;IAClB,CAAC;IAED;;;OAGG;IACI,SAAS,CAAC,aAAqB;QACpC,MAAM,WAAW,GAAG,IAAI,CAAC,UAAU,CAAC,QAAQ,EAAE,CAAC;QAC/C,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,aAAa,CAAC,EAAE,CAAC;YACpC,OAAO;QACT,CAAC;QAED,MAAM,OAAO,GAAG,IAAI,GAAG,CAAC,WAAW,CAAC,CAAC;QACrC,OAAO,CAAC,MAAM,CAAC,aAAa,CAAC,CAAC;QAC9B,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;IAChC,CAAC;IAED;QA/DA,sDAAsD;QACrC,eAAU,GAAG,IAAI,eAAe,CAAwB,IAAI,GAAG,EAAE,CAAC,CAAC;QAEpF,4CAA4C;QAC3B,iBAAY,GAAG,IAAI,eAAe,CAAgB,IAAI,CAAC,CAAC;QAEzE,sCAAsC;QACtB,UAAK,GAA2B,IAAI,CAAC,UAAU,CAAC,IAAI,CAClE,GAAG,CAAC,OAAO,CAAC,EAAE,CAAC,KAAK,CAAC,IAAI,CAAC,OAAO,CAAC,MAAM,EAAE,CAAC,CAAC,CAC7C,CAAC;QAEF,sDAAsD;QACtC,iBAAY,GAAG,IAAI,CAAC,YAAY,CAAC,YAAY,EAAE,CAAC,IAAI,CAAC,oBAAoB,EAAE,CAAC,CAAC;QAE7F,8DAA8D;QAC9C,oBAAe,GAAgC,IAAI,CAAC,YAAY,CAAC,IAAI,CACnF,GAAG,CAAC,QAAQ,CAAC,EAAE,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC,IAAI,CAAC,UAAU,CAAC,QAAQ,EAAE,CAAC,GAAG,CAAC,QAAQ,CAAC,IAAI,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC;QACrF,wFAAwF;QACxF,oBAAoB,CAAC,CAAC,IAAI,EAAE,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,KAAK,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,CACpF,CAAC;QAEF,6EAA6E;QAC7D,0BAAqB,GAAmC,IAAI,CAAC,eAAe,CAAC,IAAI,CAC/F,GAAG,CAAC,SAAS,CAAC,EAAE,CAAC,SAAS,EAAE,cAAc,IAAI,IAAI,CAAC,CACpD,CAAC;QAwCA,iDAAiD;QACjD,eAAe;QACf,8BAA8B;IAChC,CAAC;IAED;;;OAGG;IACI,YAAY,CAAC,KAAoB;QACtC,IAAI,IAAI,CAAC,YAAY,CAAC,QAAQ,EAAE,KAAK,KAAK,EAAE,CAAC;YAC3C,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QAChC,CAAC;IACH,CAAC;IAED;;;;OAIG;IACI,oBAAoB,CAAC,YAAkC;QAC5D,MAAM,QAAQ,GAAG,IAAI,CAAC,YAAY,CAAC,QAAQ,EAAE,CAAC;QAC9C,IAAI,CAAC,QAAQ,EAAE,CAAC;YACd,OAAO,CAAC,IAAI,CAAC,qCAAqC,CAAC,CAAC;YACpD,OAAO;QACT,CAAC;QAED,MAAM,IAAI,GAAG,IAAI,CAAC,UAAU,CAAC,QAAQ,EAAE,CAAC;QACxC,MAAM,cAAc,GAAG,IAAI,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;QAE1C,IAAI,cAAc,EAAE,CAAC;YACnB,oCAAoC;YACpC,MAAM,YAAY,GAAG,KAAK,CAAC,EAAE,EAAE,cAAc,EAAE,EAAE,cAAc,EAAE,YAAY,EAAE,CAAC,CAAC;YACjF,MAAM,YAAY,GAAG,IAAI,GAAG,CAAC,IAAI,CAAC,CAAC;YACnC,YAAY,CAAC,GAAG,CAAC,QAAQ,EAAE,YAAY,CAAC,CAAC;YACzC,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;QACrC,CAAC;IACH,CAAC;IAED;;;OAGG;IACI,WAAW,CAAC,KAAa;QAC9B,OAAO,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;IAC5D,CAAC;IAED;;;;OAIG;IACK,kBAAkB,CAAC,UAAkB;QAC3C,MAAM,OAAO,GAAG,IAAI,CAAC,UAAU,CAAC,QAAQ,EAAE,CAAC;QAC3C,IAAI,OAAO,CAAC,IAAI,IAAI,CAAC,EAAE,CAAC;YACtB,OAAO,IAAI,CAAC;QACd,CAAC;QAED,MAAM,MAAM,GAAG,KAAK,CAAC,IAAI,CAAC,OAAO,CAAC,IAAI,EAAE,CAAC,CAAC;QAC1C,MAAM,YAAY,GAAG,MAAM,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC;QAEhD,kGAAkG;QAClG,MAAM,SAAS,GAAG,YAAY,KAAK,MAAM,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,YAAY,GAAG,CAAC,CAAC,CAAC,CAAC,YAAY,GAAG,CAAC,CAAC;QAC3F,OAAO,MAAM,CAAC,SAAS,CAAC,CAAC;IAC3B,CAAC;8GAjIU,eAAe;kHAAf,eAAe,cAFd,MAAM;;2FAEP,eAAe;kBAH3B,UAAU;mBAAC;oBACV,UAAU,EAAE,MAAM;iBACnB","sourcesContent":["import { Injectable } from '@angular/core';\r\nimport { BehaviorSubject, Observable } from 'rxjs';\r\nimport { distinctUntilChanged, map } from 'rxjs/operators';\r\nimport { merge } from 'lodash';\r\n\r\n/**\r\n * Defines the shape of the state for components that need to persist\r\n * data on a per-tab basis. Using Partial<> for flexibility.\r\n */\r\nexport interface PerTabState {\r\n  sideDrawer?: {\r\n    isOpen: boolean;\r\n    activeComponent: string | null;\r\n    componentState?: { [componentKey: string]: unknown };\r\n  };\r\n  footer?: {\r\n    someData?: string;\r\n  };\r\n  console?: {\r\n    history: string[];\r\n  };\r\n}\r\n\r\n/**\r\n * Represents the entire state for a single tab, including its metadata\r\n * and the state of various components.\r\n */\r\nexport interface TabState {\r\n  id: string;\r\n  title: string;\r\n  componentState: PerTabState;\r\n}\r\n\r\n/**\r\n * Creates the default state for a new tab.\r\n */\r\nconst createDefaultComponentState = (): PerTabState => ({\r\n  sideDrawer: { isOpen: false, activeComponent: null, componentState: {} },\r\n  footer: { someData: '' },\r\n  console: { history: [] },\r\n});\r\n\r\n@Injectable({\r\n  providedIn: 'root',\r\n})\r\nexport class TabStateService {\r\n  // Holds the state for all open tabs, keyed by tab ID.\r\n  private readonly _tabsState = new BehaviorSubject<Map<string, TabState>>(new Map());\r\n\r\n  // Holds the ID of the currently active tab.\r\n  private readonly _activeTabId = new BehaviorSubject<string | null>(null);\r\n\r\n  /** An observable of all open tabs. */\r\n  public readonly tabs$: Observable<TabState[]> = this._tabsState.pipe(\r\n    map(tabsMap => Array.from(tabsMap.values()))\r\n  );\r\n\r\n  /** An observable of the currently active tab's ID. */\r\n  public readonly activeTabId$ = this._activeTabId.asObservable().pipe(distinctUntilChanged());\r\n\r\n  /** An observable of the state of the currently active tab. */\r\n  public readonly activeTabState$: Observable<TabState | null> = this.activeTabId$.pipe(\r\n    map(activeId => (activeId ? this._tabsState.getValue().get(activeId) ?? null : null)),\r\n    // Use a deep-ish comparison to prevent emissions when only the object reference changes\r\n    distinctUntilChanged((prev, curr) => JSON.stringify(prev) === JSON.stringify(curr))\r\n  );\r\n\r\n  /** An observable of just the component-specific state for the active tab. */\r\n  public readonly activeComponentState$: Observable<PerTabState | null> = this.activeTabState$.pipe(\r\n    map(activeTab => activeTab?.componentState ?? null)\r\n  );\r\n\r\n  /**\r\n   * Adds a new tab to the state.\r\n   * @param title The title for the new tab.\r\n   * @param id An optional unique ID. If not provided, one will be generated.\r\n   * @returns The ID of the newly created tab.\r\n   */\r\n  public addTab(title: string, id?: string): string {\r\n    const newTabId = id || `tab_${Date.now()}_${Math.random()}`;\r\n    const newTab: TabState = {\r\n      id: newTabId,\r\n      title,\r\n      componentState: createDefaultComponentState(),\r\n    };\r\n\r\n    const currentTabs = this._tabsState.getValue();\r\n    const newTabs = new Map(currentTabs).set(newTabId, newTab);\r\n    this._tabsState.next(newTabs);\r\n    this.setActiveTab(newTabId);\r\n\r\n    return newTabId;\r\n  }\r\n\r\n  /**\r\n   * Removes a tab from the state.\r\n   * @param tabIdToRemove The ID of the tab to remove.\r\n   */\r\n  public removeTab(tabIdToRemove: string): void {\r\n    const currentTabs = this._tabsState.getValue();\r\n    if (!currentTabs.has(tabIdToRemove)) {\r\n      return;\r\n    }\r\n\r\n    const newTabs = new Map(currentTabs);\r\n    newTabs.delete(tabIdToRemove);\r\n    this._tabsState.next(newTabs);\r\n  }\r\n\r\n  constructor() {\r\n    // You can initialize with default tabs if needed\r\n    // For example:\r\n    // this.addTab('Welcome Tab');\r\n  }\r\n\r\n  /**\r\n   * Sets the currently active tab.\r\n   * @param tabId The unique identifier of the tab to activate.\r\n   */\r\n  public setActiveTab(tabId: string | null): void {\r\n    if (this._activeTabId.getValue() !== tabId) {\r\n      this._activeTabId.next(tabId);\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Updates the state for the currently active tab.\r\n   * This performs a deep merge to avoid overwriting nested state.\r\n   * @param partialState The partial state to update for the active tab.\r\n   */\r\n  public updateActiveTabState(partialState: Partial<PerTabState>): void {\r\n    const activeId = this._activeTabId.getValue();\r\n    if (!activeId) {\r\n      console.warn('Cannot update state, no active tab.');\r\n      return;\r\n    }\r\n\r\n    const tabs = this._tabsState.getValue();\r\n    const activeTabState = tabs.get(activeId);\r\n\r\n    if (activeTabState) {\r\n      // Use lodash merge for a deep merge\r\n      const updatedState = merge({}, activeTabState, { componentState: partialState });\r\n      const newTabsState = new Map(tabs);\r\n      newTabsState.set(activeId, updatedState);\r\n      this._tabsState.next(newTabsState);\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Gets an observable for a specific tab's state.\r\n   * @param tabId The ID of the tab to observe.\r\n   */\r\n  public getTabState(tabId: string): Observable<TabState | undefined> {\r\n    return this._tabsState.pipe(map(tabs => tabs.get(tabId)));\r\n  }\r\n\r\n  /**\r\n   * Determines the ID of the next tab to activate when a tab is closed.\r\n   * @param idToRemove The ID of the tab being removed.\r\n   * @returns The ID of the next tab to activate, or null if no tabs are left.\r\n   */\r\n  private getNextActiveTabId(idToRemove: string): string | null {\r\n    const tabsMap = this._tabsState.getValue();\r\n    if (tabsMap.size <= 1) {\r\n      return null;\r\n    }\r\n\r\n    const tabIds = Array.from(tabsMap.keys());\r\n    const removedIndex = tabIds.indexOf(idToRemove);\r\n\r\n    // If it's the last tab in the list, activate the one before it. Otherwise, activate the next one.\r\n    const nextIndex = removedIndex === tabIds.length - 1 ? removedIndex - 1 : removedIndex + 1;\r\n    return tabIds[nextIndex];\r\n  }\r\n}"]}
|
|
@@ -0,0 +1,151 @@
|
|
|
1
|
+
import { Injectable, PLATFORM_ID, InjectionToken, signal, inject } from '@angular/core';
|
|
2
|
+
import { HttpClient } from '@angular/common/http';
|
|
3
|
+
import { tap } from 'rxjs/operators';
|
|
4
|
+
import { of } from 'rxjs';
|
|
5
|
+
import { cidePath, designConfigRoutesUrl, hostManagerRoutesUrl } from 'cloud-ide-lms-model';
|
|
6
|
+
import { isPlatformBrowser } from '@angular/common';
|
|
7
|
+
import * as i0 from "@angular/core";
|
|
8
|
+
// Define an InjectionToken for the environment configuration
|
|
9
|
+
// This is needed to inject environment variables into services provided in 'root'
|
|
10
|
+
export const ENVIRONMENT_CONFIG = new InjectionToken('ENVIRONMENT_CONFIG');
|
|
11
|
+
/**
|
|
12
|
+
* Utility function to convert database format keys to CSS format and handle units
|
|
13
|
+
* @param key - The database format key (e.g., 'font_size_xxxl')
|
|
14
|
+
* @param value - The value to be set
|
|
15
|
+
* @returns Object with cssVarName and processed value
|
|
16
|
+
*/
|
|
17
|
+
export function processThemeVariable(key, value) {
|
|
18
|
+
// 1. Convert the key from the DB format to the CSS format.
|
|
19
|
+
const cssVarName = `${key.replace(/_/g, '-')}`;
|
|
20
|
+
// 2. Append the font_size and line_height value to rem
|
|
21
|
+
let processedValue = value;
|
|
22
|
+
if (cssVarName.includes('font-size') || cssVarName.includes('line-height')) {
|
|
23
|
+
processedValue = parseFloat(value) + 'rem';
|
|
24
|
+
}
|
|
25
|
+
return { cssVarName, processedValue: processedValue };
|
|
26
|
+
}
|
|
27
|
+
/**
|
|
28
|
+
* Sets a single CSS variable on the document root element
|
|
29
|
+
* @param key - The database format key (e.g., 'font_size_xxxl')
|
|
30
|
+
* @param value - The value to be set
|
|
31
|
+
* @param platformId - Platform ID for browser check
|
|
32
|
+
*/
|
|
33
|
+
export function setCSSVariable(key, value, platformId) {
|
|
34
|
+
if (isPlatformBrowser(platformId)) {
|
|
35
|
+
const { cssVarName, processedValue } = processThemeVariable(key, value);
|
|
36
|
+
const root = document.documentElement;
|
|
37
|
+
root.style.setProperty(cssVarName, processedValue);
|
|
38
|
+
}
|
|
39
|
+
}
|
|
40
|
+
export class CideLytThemeService {
|
|
41
|
+
constructor() {
|
|
42
|
+
// Modern Angular signals for reactive state management
|
|
43
|
+
this.selectedThemeSignal = signal("");
|
|
44
|
+
this.selectedTheme = this.selectedThemeSignal.asReadonly();
|
|
45
|
+
// Modern dependency injection
|
|
46
|
+
this.http = inject(HttpClient);
|
|
47
|
+
this.platformId = inject(PLATFORM_ID);
|
|
48
|
+
this.environment = inject(ENVIRONMENT_CONFIG);
|
|
49
|
+
}
|
|
50
|
+
/**
|
|
51
|
+
* Fetches the theme from the API and applies it to the document.
|
|
52
|
+
* This method is intended to be called when the theme should actually be applied.
|
|
53
|
+
*/
|
|
54
|
+
fetchAndApplyTheme() {
|
|
55
|
+
console.log('Fetching and applying theme via CideLytThemeService...');
|
|
56
|
+
return this.loadTheme(); // Use the existing loadTheme logic
|
|
57
|
+
}
|
|
58
|
+
/**
|
|
59
|
+
* Fetches the list of system themes from the API.
|
|
60
|
+
*/
|
|
61
|
+
fetchSystemThemesList() {
|
|
62
|
+
return this.http.get(cidePath.join([
|
|
63
|
+
hostManagerRoutesUrl.cideSuiteHost,
|
|
64
|
+
designConfigRoutesUrl?.module,
|
|
65
|
+
designConfigRoutesUrl?.getSystemThemesList
|
|
66
|
+
]));
|
|
67
|
+
}
|
|
68
|
+
/**
|
|
69
|
+
* Internal method to fetch the theme from the API and apply it.
|
|
70
|
+
*/
|
|
71
|
+
loadTheme() {
|
|
72
|
+
return this.http.post(cidePath.join([hostManagerRoutesUrl.cideSuiteHost, designConfigRoutesUrl?.module, designConfigRoutesUrl?.getSystemTheme]), { syupth_user_id_user: "659a7ea0a1f0e08246661766" }).pipe(tap(theme => {
|
|
73
|
+
console.log("Applying theme:", theme);
|
|
74
|
+
this.applyTheme(theme?.data);
|
|
75
|
+
}));
|
|
76
|
+
}
|
|
77
|
+
/**
|
|
78
|
+
* Sets the CSS variables on the root element.
|
|
79
|
+
* @param theme - An object where keys are CSS variable names (e.g., '--color-text-body')
|
|
80
|
+
* and values are the color strings (e.g., '#ffffff').
|
|
81
|
+
*/
|
|
82
|
+
applyTheme(theme) {
|
|
83
|
+
console.log("Applying theme:", theme);
|
|
84
|
+
let merged = {};
|
|
85
|
+
if (theme?.core_system_themes?.syth_properties) {
|
|
86
|
+
merged = { ...merged, ...theme.core_system_themes.syth_properties };
|
|
87
|
+
this.selectedThemeSignal.set(theme.core_system_themes._id);
|
|
88
|
+
}
|
|
89
|
+
if (theme?.core_system_organization_themes?.syoth_overrides) {
|
|
90
|
+
merged = { ...merged, ...theme.core_system_organization_themes.syoth_overrides };
|
|
91
|
+
}
|
|
92
|
+
if (theme?.core_system_theme_user_preferences?.syupth_overrides) {
|
|
93
|
+
merged = { ...merged, ...theme.core_system_theme_user_preferences.syupth_overrides };
|
|
94
|
+
}
|
|
95
|
+
console.log("Merged theme:", merged);
|
|
96
|
+
// Check if the code is running in a browser environment.
|
|
97
|
+
if (isPlatformBrowser(this.platformId)) {
|
|
98
|
+
const root = document.documentElement;
|
|
99
|
+
for (const [key, value] of Object.entries(merged)) {
|
|
100
|
+
const { cssVarName, processedValue } = processThemeVariable(key, value);
|
|
101
|
+
// Set the CSS custom property on the root element.
|
|
102
|
+
root.style.setProperty(cssVarName, processedValue);
|
|
103
|
+
}
|
|
104
|
+
}
|
|
105
|
+
}
|
|
106
|
+
/**
|
|
107
|
+
* Saves the system theme user preferences.
|
|
108
|
+
* @param themeVars - The theme variables to save.
|
|
109
|
+
* @returns An observable of the system theme user preferences.
|
|
110
|
+
*/
|
|
111
|
+
saveSystemThemeUserPreferences(themeVars) {
|
|
112
|
+
return this.http.post(cidePath.join([hostManagerRoutesUrl.cideSuiteHost, designConfigRoutesUrl?.module, designConfigRoutesUrl?.systemThemeUserPreferences]), themeVars);
|
|
113
|
+
}
|
|
114
|
+
/**
|
|
115
|
+
* Updates a single CSS variable using the utility function
|
|
116
|
+
* @param key - The database format key (e.g., 'font_size_xxxl')
|
|
117
|
+
* @param value - The value to be set
|
|
118
|
+
*/
|
|
119
|
+
updateCSSVariable(key, value) {
|
|
120
|
+
setCSSVariable(key, value, this.platformId);
|
|
121
|
+
}
|
|
122
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: CideLytThemeService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
123
|
+
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: CideLytThemeService, providedIn: 'root' }); }
|
|
124
|
+
}
|
|
125
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: CideLytThemeService, decorators: [{
|
|
126
|
+
type: Injectable,
|
|
127
|
+
args: [{
|
|
128
|
+
providedIn: 'root'
|
|
129
|
+
}]
|
|
130
|
+
}] });
|
|
131
|
+
/**
|
|
132
|
+
* Factory function for APP_INITIALIZER.
|
|
133
|
+
* This function is used in your app.config.ts.
|
|
134
|
+
* It conditionally triggers theme loading based on the environment setting.
|
|
135
|
+
*/
|
|
136
|
+
export function themeFactory(themeService, environmentConfig) {
|
|
137
|
+
return () => {
|
|
138
|
+
if (!environmentConfig.applyThemeOnLogin) {
|
|
139
|
+
// If theme should be applied on app initialize, fetch and apply it.
|
|
140
|
+
console.log('APP_INITIALIZER: Applying theme...');
|
|
141
|
+
return themeService.fetchAndApplyTheme();
|
|
142
|
+
}
|
|
143
|
+
else {
|
|
144
|
+
// If theme should be applied on login, APP_INITIALIZER does nothing for the theme.
|
|
145
|
+
// We return an observable that immediately completes.
|
|
146
|
+
console.log('APP_INITIALIZER: Theme application deferred to after login (per environment config).');
|
|
147
|
+
return of(undefined); // Return an observable that completes immediately
|
|
148
|
+
}
|
|
149
|
+
};
|
|
150
|
+
}
|
|
151
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"theme.service.js","sourceRoot":"","sources":["../../../../../../projects/cloud-ide-layout/src/lib/services/theme/theme.service.ts"],"names":[],"mappings":"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","sourcesContent":["\nimport { Injectable, PLATFORM_ID, InjectionToken, signal, inject } from '@angular/core';\nimport { HttpClient } from '@angular/common/http';\nimport { tap } from 'rxjs/operators';\nimport { Observable, of } from 'rxjs';\nimport { cidePath, designConfigRoutesUrl, hostManagerRoutesUrl, SystemThemeControllerRespoonse, SystemThemesListControllerRespoonse, IEnvironmentConfig, SystemThemeUserPreferencesControllerRespoonse, MSystemThemeUserPreferencesRequest, SystemThemeControllerRespoonseData } from 'cloud-ide-lms-model';\nimport { isPlatformBrowser } from '@angular/common';\n\n// Define an InjectionToken for the environment configuration\n// This is needed to inject environment variables into services provided in 'root'\nexport const ENVIRONMENT_CONFIG = new InjectionToken<IEnvironmentConfig>('ENVIRONMENT_CONFIG');\n\n// Define an interface for your theme for type safety\nexport interface Theme {\n  [key: string]: string; // Allows any string key, e.g., 'color-text-body'\n}\n\n/**\n * Utility function to convert database format keys to CSS format and handle units\n * @param key - The database format key (e.g., 'font_size_xxxl')\n * @param value - The value to be set\n * @returns Object with cssVarName and processed value\n */\nexport function processThemeVariable(key: string, value: string): { cssVarName: string; processedValue: string } {\n  // 1. Convert the key from the DB format to the CSS format.\n  const cssVarName = `${key.replace(/_/g, '-')}`;\n\n  // 2. Append the font_size and line_height value to rem\n  let processedValue = value;\n  if (cssVarName.includes('font-size') || cssVarName.includes('line-height')) {\n    processedValue = parseFloat(value) + 'rem';\n  }\n\n  return { cssVarName, processedValue: processedValue };\n}\n\n/**\n * Sets a single CSS variable on the document root element\n * @param key - The database format key (e.g., 'font_size_xxxl')\n * @param value - The value to be set\n * @param platformId - Platform ID for browser check\n */\nexport function setCSSVariable(key: string, value: string, platformId: object): void {\n  if (isPlatformBrowser(platformId)) {\n    const { cssVarName, processedValue } = processThemeVariable(key, value);\n    const root = document.documentElement;\n    root.style.setProperty(cssVarName, processedValue);\n  }\n}\n\n@Injectable({\n  providedIn: 'root'\n})\nexport class CideLytThemeService {\n  // Modern Angular signals for reactive state management\n  private selectedThemeSignal = signal<string>(\"\");\n  public selectedTheme = this.selectedThemeSignal.asReadonly();\n  \n  // Modern dependency injection\n  private http = inject(HttpClient);\n  private platformId = inject(PLATFORM_ID);\n  private environment = inject(ENVIRONMENT_CONFIG);\n\n  /**\n   * Fetches the theme from the API and applies it to the document.\n   * This method is intended to be called when the theme should actually be applied.\n   */\n  public fetchAndApplyTheme(): Observable<SystemThemeControllerRespoonse> {\n    console.log('Fetching and applying theme via CideLytThemeService...');\n    return this.loadTheme(); // Use the existing loadTheme logic\n  }\n\n  /**\n   * Fetches the list of system themes from the API.\n   */\n  public fetchSystemThemesList(): Observable<SystemThemesListControllerRespoonse> {\n    return this.http.get<SystemThemesListControllerRespoonse>(\n      cidePath.join([\n        hostManagerRoutesUrl.cideSuiteHost,\n        designConfigRoutesUrl?.module,\n        designConfigRoutesUrl?.getSystemThemesList\n      ]));\n  }\n\n  /**\n   * Internal method to fetch the theme from the API and apply it.\n   */\n  loadTheme() {\n    return this.http.post<SystemThemeControllerRespoonse>(cidePath.join([hostManagerRoutesUrl.cideSuiteHost, designConfigRoutesUrl?.module, designConfigRoutesUrl?.getSystemTheme]), { syupth_user_id_user: \"659a7ea0a1f0e08246661766\" }).pipe(\n      tap(theme => {\n        console.log(\"Applying theme:\", theme);\n        this.applyTheme(theme?.data as SystemThemeControllerRespoonseData);\n      })\n    );\n  }\n\n  /**\n   * Sets the CSS variables on the root element.\n   * @param theme - An object where keys are CSS variable names (e.g., '--color-text-body')\n   * and values are the color strings (e.g., '#ffffff').\n   */\n  private applyTheme(theme: SystemThemeControllerRespoonseData): void {\n    console.log(\"Applying theme:\", theme);\n    let merged = {};\n    if (theme?.core_system_themes?.syth_properties) {\n      merged = { ...merged, ...theme.core_system_themes.syth_properties };\n      this.selectedThemeSignal.set(theme.core_system_themes._id as string);\n    }\n    if (theme?.core_system_organization_themes?.syoth_overrides) {\n      merged = { ...merged, ...theme.core_system_organization_themes.syoth_overrides };\n    }\n    if (theme?.core_system_theme_user_preferences?.syupth_overrides) {\n      merged = { ...merged, ...theme.core_system_theme_user_preferences.syupth_overrides };\n    }\n    console.log(\"Merged theme:\", merged);\n    // Check if the code is running in a browser environment.\n    if (isPlatformBrowser(this.platformId)) {\n      const root = document.documentElement;\n\n      for (const [key, value] of Object.entries(merged)) {\n        const { cssVarName, processedValue } = processThemeVariable(key, value as string);\n        // Set the CSS custom property on the root element.\n        root.style.setProperty(cssVarName, processedValue);\n      }\n    }\n  }\n\n  /**\n   * Saves the system theme user preferences.\n   * @param themeVars - The theme variables to save.\n   * @returns An observable of the system theme user preferences.\n   */\n  saveSystemThemeUserPreferences(themeVars: MSystemThemeUserPreferencesRequest): Observable<SystemThemeUserPreferencesControllerRespoonse> {\n    return this.http.post<SystemThemeUserPreferencesControllerRespoonse>(cidePath.join([hostManagerRoutesUrl.cideSuiteHost, designConfigRoutesUrl?.module, designConfigRoutesUrl?.systemThemeUserPreferences]), themeVars);\n  }\n\n  /**\n   * Updates a single CSS variable using the utility function\n   * @param key - The database format key (e.g., 'font_size_xxxl')\n   * @param value - The value to be set\n   */\n  updateCSSVariable(key: string, value: string): void {\n    setCSSVariable(key, value, this.platformId);\n  }\n}\n\n/**\n * Factory function for APP_INITIALIZER.\n * This function is used in your app.config.ts.\n * It conditionally triggers theme loading based on the environment setting.\n */\nexport function themeFactory(themeService: CideLytThemeService, environmentConfig: IEnvironmentConfig): () => Observable<SystemThemeControllerRespoonse | undefined> {\n  return () => {\n    if (!environmentConfig.applyThemeOnLogin) {\n      // If theme should be applied on app initialize, fetch and apply it.\n      console.log('APP_INITIALIZER: Applying theme...');\n      return themeService.fetchAndApplyTheme();\n    } else {\n      // If theme should be applied on login, APP_INITIALIZER does nothing for the theme.\n      // We return an observable that immediately completes.\n      console.log('APP_INITIALIZER: Theme application deferred to after login (per environment config).');\n      return of(undefined); // Return an observable that completes immediately\n    }\n  };\n}"]}
|