cloud-ide-layout 1.0.8 → 1.0.10

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.
Files changed (83) hide show
  1. package/README.md +34 -448
  2. package/fesm2022/{cloud-ide-layout-cloud-ide-layout-D3xzVSNA.mjs → cloud-ide-layout-cloud-ide-layout-DLceZY3Y.mjs} +335 -343
  3. package/fesm2022/{cloud-ide-layout-cloud-ide-layout-D3xzVSNA.mjs.map → cloud-ide-layout-cloud-ide-layout-DLceZY3Y.mjs.map} +1 -1
  4. package/fesm2022/{cloud-ide-layout-drawer-theme.component-BSUFE_o2.mjs → cloud-ide-layout-drawer-theme.component-Bn5--C_Q.mjs} +69 -67
  5. package/fesm2022/{cloud-ide-layout-drawer-theme.component-BSUFE_o2.mjs.map → cloud-ide-layout-drawer-theme.component-Bn5--C_Q.mjs.map} +1 -1
  6. package/fesm2022/{cloud-ide-layout-home-wrapper.component-Dr4MO4MB.mjs → cloud-ide-layout-home-wrapper.component-Bo-4LNLx.mjs} +56 -51
  7. package/fesm2022/{cloud-ide-layout-home-wrapper.component-Dr4MO4MB.mjs.map → cloud-ide-layout-home-wrapper.component-Bo-4LNLx.mjs.map} +1 -1
  8. package/fesm2022/{cloud-ide-layout-sidedrawer-notes.component-BWx03JYS.mjs → cloud-ide-layout-sidedrawer-notes.component-BmadMzdl.mjs} +28 -32
  9. package/fesm2022/{cloud-ide-layout-sidedrawer-notes.component-BWx03JYS.mjs.map → cloud-ide-layout-sidedrawer-notes.component-BmadMzdl.mjs.map} +1 -1
  10. package/fesm2022/cloud-ide-layout.mjs +1 -1
  11. package/index.d.ts +579 -3
  12. package/package.json +4 -6
  13. package/esm2022/cloud-ide-layout.mjs +0 -5
  14. package/esm2022/lib/cloud-ide-layout.component.mjs +0 -16
  15. package/esm2022/lib/cloud-ide-layout.routes.mjs +0 -28
  16. package/esm2022/lib/cloud-ide-layout.service.mjs +0 -14
  17. package/esm2022/lib/layout/console/console-wrapper/console-wrapper.component.mjs +0 -11
  18. package/esm2022/lib/layout/console/console.service.mjs +0 -28
  19. package/esm2022/lib/layout/footer/footer-wrapper/footer-wrapper.component.mjs +0 -20
  20. package/esm2022/lib/layout/footer/footer.service.mjs +0 -28
  21. package/esm2022/lib/layout/header/header-wrapper/header-wrapper.component.mjs +0 -84
  22. package/esm2022/lib/layout/header/header.service.mjs +0 -28
  23. package/esm2022/lib/layout/home/home-wrapper/home-wrapper.component.mjs +0 -198
  24. package/esm2022/lib/layout/layout/layout-wrapper/layout-wrapper.component.mjs +0 -28
  25. package/esm2022/lib/layout/request/request-wrapper/request-wrapper.component.mjs +0 -60
  26. package/esm2022/lib/layout/request/request.service.mjs +0 -235
  27. package/esm2022/lib/layout/request/tab-content/tab-content.component.mjs +0 -84
  28. package/esm2022/lib/layout/shared/shared-wrapper/shared-wrapper.component.mjs +0 -49
  29. package/esm2022/lib/layout/shared/shared.service.mjs +0 -508
  30. package/esm2022/lib/layout/sidebar/sidebar-wrapper/sidebar-wrapper.component.mjs +0 -488
  31. package/esm2022/lib/layout/sidebar/sidebar.service.mjs +0 -35
  32. package/esm2022/lib/layout/sidedrawer/cide-lyt-drawer-base.class.mjs +0 -40
  33. package/esm2022/lib/layout/sidedrawer/drawer-theme/drawer-theme.component.mjs +0 -296
  34. package/esm2022/lib/layout/sidedrawer/sidedrawer-notes/sidedrawer-notes.component.mjs +0 -175
  35. package/esm2022/lib/layout/sidedrawer/sidedrawer-notes/sidedrawer-notes.service.mjs +0 -51
  36. package/esm2022/lib/layout/sidedrawer/sidedrawer-wrapper/sidedrawer-wrapper.component.mjs +0 -257
  37. package/esm2022/lib/layout/sidedrawer/sidedrawer.service.mjs +0 -70
  38. package/esm2022/lib/services/app-state/app-state-helper.service.mjs +0 -222
  39. package/esm2022/lib/services/app-state/app-state.service.mjs +0 -256
  40. package/esm2022/lib/services/app-state/index.mjs +0 -3
  41. package/esm2022/lib/services/cache-manager/cache-manager.service.mjs +0 -112
  42. package/esm2022/lib/services/file-manager/file-manager.service.mjs +0 -21
  43. package/esm2022/lib/services/tab-state.service.mjs +0 -128
  44. package/esm2022/lib/services/theme/theme.service.mjs +0 -151
  45. package/esm2022/lib/services/user-status/user-status.service.mjs +0 -135
  46. package/esm2022/lib/utils/custom-route-reuse-strategy.mjs +0 -61
  47. package/esm2022/lib/utils/index.mjs +0 -2
  48. package/esm2022/public-api.mjs +0 -18
  49. package/lib/cloud-ide-layout.component.d.ts +0 -5
  50. package/lib/cloud-ide-layout.routes.d.ts +0 -3
  51. package/lib/cloud-ide-layout.service.d.ts +0 -6
  52. package/lib/layout/console/console-wrapper/console-wrapper.component.d.ts +0 -5
  53. package/lib/layout/console/console.service.d.ts +0 -9
  54. package/lib/layout/footer/footer-wrapper/footer-wrapper.component.d.ts +0 -9
  55. package/lib/layout/footer/footer.service.d.ts +0 -9
  56. package/lib/layout/header/header-wrapper/header-wrapper.component.d.ts +0 -25
  57. package/lib/layout/header/header.service.d.ts +0 -9
  58. package/lib/layout/home/home-wrapper/home-wrapper.component.d.ts +0 -94
  59. package/lib/layout/layout/layout-wrapper/layout-wrapper.component.d.ts +0 -5
  60. package/lib/layout/request/request-wrapper/request-wrapper.component.d.ts +0 -14
  61. package/lib/layout/request/request.service.d.ts +0 -60
  62. package/lib/layout/request/tab-content/tab-content.component.d.ts +0 -19
  63. package/lib/layout/shared/shared-wrapper/shared-wrapper.component.d.ts +0 -17
  64. package/lib/layout/shared/shared.service.d.ts +0 -114
  65. package/lib/layout/sidebar/sidebar-wrapper/sidebar-wrapper.component.d.ts +0 -140
  66. package/lib/layout/sidebar/sidebar.service.d.ts +0 -12
  67. package/lib/layout/sidedrawer/cide-lyt-drawer-base.class.d.ts +0 -16
  68. package/lib/layout/sidedrawer/drawer-theme/drawer-theme.component.d.ts +0 -71
  69. package/lib/layout/sidedrawer/sidedrawer-notes/sidedrawer-notes.component.d.ts +0 -48
  70. package/lib/layout/sidedrawer/sidedrawer-notes/sidedrawer-notes.service.d.ts +0 -13
  71. package/lib/layout/sidedrawer/sidedrawer-wrapper/sidedrawer-wrapper.component.d.ts +0 -30
  72. package/lib/layout/sidedrawer/sidedrawer.service.d.ts +0 -30
  73. package/lib/services/app-state/app-state-helper.service.d.ts +0 -150
  74. package/lib/services/app-state/app-state.service.d.ts +0 -85
  75. package/lib/services/app-state/index.d.ts +0 -3
  76. package/lib/services/cache-manager/cache-manager.service.d.ts +0 -64
  77. package/lib/services/file-manager/file-manager.service.d.ts +0 -9
  78. package/lib/services/tab-state.service.d.ts +0 -79
  79. package/lib/services/theme/theme.service.d.ts +0 -71
  80. package/lib/services/user-status/user-status.service.d.ts +0 -57
  81. package/lib/utils/custom-route-reuse-strategy.d.ts +0 -15
  82. package/lib/utils/index.d.ts +0 -1
  83. package/public-api.d.ts +0 -12
@@ -1,235 +0,0 @@
1
- import { Injectable, inject, signal, computed } from '@angular/core';
2
- import { Router, RouteReuseStrategy } from '@angular/router';
3
- import { CustomRouteReuseStrategy } from '../../utils/custom-route-reuse-strategy';
4
- import { TabStateService } from '../../services/tab-state.service';
5
- import { CideLytSidedrawerService } from '../sidedrawer/sidedrawer.service';
6
- import { CideLytSharedService } from '../shared/shared.service';
7
- import * as i0 from "@angular/core";
8
- export class CideLytRequestService {
9
- constructor() {
10
- this.requestVisible = false;
11
- // Modern Angular v20+ pattern: Use Signals instead of BehaviorSubject
12
- this.tabsSignal = signal([]);
13
- this.activeTabIdSignal = signal(null);
14
- // Computed signals for derived state and public readonly access
15
- this.tabs = computed(() => this.tabsSignal());
16
- this.activeTabId = computed(() => this.activeTabIdSignal());
17
- this.activeTab = computed(() => this.tabsSignal().find(tab => tab.active));
18
- // Use inject() for all dependencies
19
- this.router = inject(Router);
20
- this.routeReuseStrategy = inject(RouteReuseStrategy);
21
- this.tabStateService = inject(TabStateService);
22
- this.sidedrawerService = inject(CideLytSidedrawerService);
23
- this.sharedService = inject(CideLytSharedService);
24
- // Initialize router
25
- this.router = inject(Router);
26
- this.router.routeReuseStrategy = new CustomRouteReuseStrategy();
27
- // Register tab management callback with shared service to avoid circular dependency
28
- this.sharedService.registerTabManagement(this.handleRouteBasedTabManagement.bind(this));
29
- // Register request visibility callback
30
- this.sharedService.registerRequestVisibility(this.handleRequestVisibility.bind(this));
31
- }
32
- /**
33
- * Handle request visibility changes from shared service
34
- */
35
- handleRequestVisibility(show) {
36
- if (show) {
37
- this.showRequest();
38
- }
39
- else {
40
- this.hideRequest();
41
- }
42
- }
43
- /**
44
- * Handle sidebar sync when tab becomes active
45
- */
46
- handleSidebarSync(tabInfo) {
47
- // The sidebar sync will be handled automatically by the shared service
48
- // when page data is loaded, so we don't need additional logic here
49
- console.log('🔗 REQUEST SERVICE: Tab activated, sidebar sync handled by shared service:', tabInfo.title);
50
- }
51
- /**
52
- * Handle route-based tab management - called by shared service
53
- * This method checks for existing tabs and creates/activates as needed
54
- */
55
- handleRouteBasedTabManagement(currentRoutePath, queryParams, layout, pageData, pageCode) {
56
- console.log('handleRouteBasedTabManagement', currentRoutePath, queryParams, layout, pageData, pageCode);
57
- const currentTabs = this.tabsSignal();
58
- const tabAlreadyExists = currentTabs.find((t) => {
59
- const tRoutePath = t.route.startsWith('/') ? t.route : '/' + t.route;
60
- const tParamsMatch = JSON.stringify(t.params || {}) === JSON.stringify(queryParams || {});
61
- return tRoutePath === currentRoutePath && tParamsMatch;
62
- });
63
- console.log('tabAlreadyExists', tabAlreadyExists);
64
- if (!tabAlreadyExists) {
65
- const title = pageData.data?.page?.sypg_title || pageCode.toString() || 'Untitled Tab';
66
- console.log(`🆕 Adding new tab: ${title}`);
67
- this.addTab(title, currentRoutePath, queryParams, layout);
68
- }
69
- else if (tabAlreadyExists && !tabAlreadyExists.active) {
70
- console.log(`🔄 Activating existing tab: ${tabAlreadyExists.title}`);
71
- this.activateTab(tabAlreadyExists.id);
72
- }
73
- else if (tabAlreadyExists && tabAlreadyExists.active) {
74
- console.log(`🔄 Activating existing tab: ${tabAlreadyExists.title}`);
75
- this.activateTab(tabAlreadyExists.id);
76
- }
77
- }
78
- addTab(title, route, params, layout) {
79
- console.log('addTab', title, route, params, layout);
80
- const id = this.generateId();
81
- const currentTabs = this.tabsSignal();
82
- console.log('currentTabs', currentTabs, route);
83
- // Check if a tab with the same route and params already exists
84
- const existingTab = currentTabs.find((tab) => {
85
- const paramsMatch = JSON.stringify(tab.params || {}) === JSON.stringify(params || {});
86
- return tab.route === route && paramsMatch;
87
- });
88
- console.log('existingTab', existingTab);
89
- if (existingTab) {
90
- this.activateTab(existingTab.id);
91
- return existingTab.id;
92
- }
93
- // Create new tab
94
- const newTab = {
95
- id,
96
- title,
97
- route,
98
- params,
99
- active: true,
100
- layout,
101
- sytm_page_id_sypg: '',
102
- themeId: ''
103
- };
104
- // Deactivate all other tabs
105
- currentTabs.forEach((tab) => tab.active = false);
106
- console.log('currentTabs after deactivate', currentTabs, 'activateTab', id);
107
- // Add new tab
108
- this.tabsSignal.set([...currentTabs, newTab]);
109
- this.activeTabIdSignal.set(id);
110
- // Sync with TabStateService
111
- this.tabStateService.addTab(title, id);
112
- // Request visibility is now controlled by layout configuration in setPageData
113
- // No need to automatically show request wrapper here
114
- // Navigate to the new route
115
- if (params) {
116
- this.router.navigate([route], { queryParams: params });
117
- }
118
- else {
119
- this.router.navigate([route]);
120
- }
121
- return id;
122
- }
123
- activateTab(tabId) {
124
- console.log('🔍 REQUEST SERVICE: Activating tab:', tabId);
125
- const tabs = this.tabsSignal();
126
- const tabToActivate = tabs.find(tab => tab.id === tabId);
127
- if (!tabToActivate) {
128
- console.warn('⚠️ Tab not found:', tabId);
129
- return;
130
- }
131
- // Update tabs: deactivate all, then activate the target
132
- const updatedTabs = tabs.map(tab => ({
133
- ...tab,
134
- active: tab.id === tabId
135
- }));
136
- this.tabsSignal.set(updatedTabs);
137
- this.activeTabIdSignal.set(tabId);
138
- console.log('✅ REQUEST SERVICE: Tab activated:', tabToActivate.title);
139
- // Navigate to the tab's route
140
- this.router.navigate([tabToActivate.route], {
141
- queryParams: tabToActivate.params || {},
142
- replaceUrl: false
143
- });
144
- // Trigger sidebar sync for the activated tab by calling shared service
145
- // The shared service will handle the sidebar sync when the route loads
146
- console.log('🔗 REQUEST SERVICE: Sidebar sync will be handled by route change');
147
- }
148
- closeTab(id) {
149
- const currentTabs = this.tabsSignal();
150
- const tabIndex = currentTabs.findIndex((tab) => tab.id === id);
151
- if (tabIndex === -1)
152
- return;
153
- const tabToClose = currentTabs[tabIndex];
154
- const wasActive = tabToClose.active;
155
- // Sync with TabStateService by removing the tab state
156
- this.tabStateService.removeTab(id);
157
- // Clear stored route from strategy
158
- if (this.routeReuseStrategy instanceof CustomRouteReuseStrategy) {
159
- let pathKey = tabToClose.route.startsWith('/') ? tabToClose.route.substring(1) : tabToClose.route;
160
- if (tabToClose.params && Object.keys(tabToClose.params).length > 0) {
161
- const queryParamsString = Object.keys(tabToClose.params)
162
- .sort()
163
- .map(key => `${key}=${tabToClose.params[key]}`)
164
- .join('&');
165
- pathKey += '?' + queryParamsString;
166
- }
167
- this.routeReuseStrategy.clearStoredRoute(pathKey);
168
- }
169
- // Remove the tab from this service's list
170
- const newTabs = currentTabs.filter((tab) => tab.id !== id);
171
- // If we're closing the active tab, activate the next available tab
172
- if (wasActive && newTabs.length > 0) {
173
- const newActiveIndex = Math.max(0, Math.min(tabIndex, newTabs.length - 1));
174
- const newActiveTab = newTabs[newActiveIndex];
175
- // Update sidedrawer for the new active tab
176
- this.sidedrawerService.updateDrawerItems(newActiveTab.layout);
177
- newTabs[newActiveIndex].active = true;
178
- this.activeTabIdSignal.set(newActiveTab.id);
179
- this.tabStateService.setActiveTab(newActiveTab.id);
180
- // Navigate to the new active tab's route
181
- if (newActiveTab.params) {
182
- this.router.navigate([newActiveTab.route], { queryParams: newActiveTab.params });
183
- }
184
- else {
185
- this.router.navigate([newActiveTab.route]);
186
- }
187
- }
188
- else if (newTabs.length === 0) {
189
- this.activeTabIdSignal.set(null);
190
- this.tabStateService.setActiveTab(null);
191
- // Clear sidedrawer as no tabs are open
192
- this.sidedrawerService.updateDrawerItems(undefined);
193
- }
194
- this.tabsSignal.set(newTabs);
195
- // Request wrapper visibility is now controlled by layout configuration
196
- // The wrapper will be hidden/shown based on sytm_layout_request.status in setPageData
197
- }
198
- // Hide Request
199
- hideRequest() {
200
- console.log('🚫 REQUEST SERVICE - Hiding request wrapper');
201
- this.requestVisible = false;
202
- document.querySelector(`#cide-lyt-request-wrapper`)?.classList.add('cide-lyt-request-wrapper-hide');
203
- document.querySelector(`body`)?.classList.remove('cide-lyt-request-exist');
204
- }
205
- // Show Request
206
- showRequest() {
207
- console.log('✅ REQUEST SERVICE - Showing request wrapper');
208
- this.requestVisible = true;
209
- document.querySelector(`#cide-lyt-request-wrapper`)?.classList.remove('cide-lyt-request-wrapper-hide');
210
- document.querySelector(`body`)?.classList.add('cide-lyt-request-exist');
211
- }
212
- generateId() {
213
- return 'tab-' + Math.random().toString(36).substr(2, 9);
214
- }
215
- updateTabScrollPosition(tabId, scrollTop, scrollLeft) {
216
- const currentTabs = this.tabsSignal();
217
- const tabIndex = currentTabs.findIndex((t) => t.id === tabId);
218
- if (tabIndex !== -1) {
219
- const tabToUpdate = currentTabs[tabIndex];
220
- tabToUpdate.scrollTop = scrollTop;
221
- tabToUpdate.scrollLeft = scrollLeft;
222
- // Create a new array reference to trigger change detection
223
- this.tabsSignal.set([...currentTabs]);
224
- }
225
- }
226
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: CideLytRequestService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
227
- static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: CideLytRequestService, providedIn: 'root' }); }
228
- }
229
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: CideLytRequestService, decorators: [{
230
- type: Injectable,
231
- args: [{
232
- providedIn: 'root'
233
- }]
234
- }], ctorParameters: () => [] });
235
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"request.service.js","sourceRoot":"","sources":["../../../../../../projects/cloud-ide-layout/src/lib/layout/request/request.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, inject, signal, WritableSignal, Signal, computed } from '@angular/core';\nimport { Router, RouteReuseStrategy } from '@angular/router';\nimport { CustomRouteReuseStrategy } from '../../utils/custom-route-reuse-strategy';\nimport { TabStateService } from '../../services/tab-state.service';\nimport { CideLytSidedrawerService } from '../sidedrawer/sidedrawer.service';\nimport { CideLytSharedService } from '../shared/shared.service';\nimport { ISytmLayout } from 'cloud-ide-lms-model';\n\nexport interface TabInfo {\n  id: string;\n  title: string;\n  route: string;\n  params?: Record<string, unknown>;\n  active: boolean;\n  sytm_page_id_sypg: string;\n  themeId: string;\n  loadedComponent?: unknown;\n  scrollTop?: number;\n  scrollLeft?: number;\n  layout?: ISytmLayout;\n}\n\n// Interface for page data passed from shared service\ninterface PageDataResponse {\n  data?: {\n    page?: {\n      sypg_title?: string;\n    };\n  };\n}\n\n@Injectable({\n  providedIn: 'root'\n})\nexport class CideLytRequestService {\n  public requestVisible: boolean = false;\n  // Modern Angular v20+ pattern: Use Signals instead of BehaviorSubject\n  public tabsSignal: WritableSignal<TabInfo[]> = signal([]);\n  public activeTabIdSignal: WritableSignal<string | null> = signal(null);\n  \n  // Computed signals for derived state and public readonly access\n  public tabs: Signal<TabInfo[]> = computed(() => this.tabsSignal());\n  public activeTabId: Signal<string | null> = computed(() => this.activeTabIdSignal());\n  public activeTab: Signal<TabInfo | undefined> = computed(() => \n    this.tabsSignal().find(tab => tab.active)\n  );\n\n  // Use inject() for all dependencies\n  private router = inject(Router);\n  private routeReuseStrategy = inject(RouteReuseStrategy);\n  private tabStateService = inject(TabStateService);\n  private sidedrawerService = inject(CideLytSidedrawerService);\n  private sharedService = inject(CideLytSharedService);\n\n  constructor() {\n    // Initialize router\n    this.router = inject(Router);\n    this.router.routeReuseStrategy = new CustomRouteReuseStrategy();\n    \n    // Register tab management callback with shared service to avoid circular dependency\n    this.sharedService.registerTabManagement(this.handleRouteBasedTabManagement.bind(this));\n    \n    // Register request visibility callback\n    this.sharedService.registerRequestVisibility(this.handleRequestVisibility.bind(this));\n  }\n\n  /**\n   * Handle request visibility changes from shared service\n   */\n  private handleRequestVisibility(show: boolean): void {\n    if (show) {\n      this.showRequest();\n    } else {\n      this.hideRequest();\n    }\n  }\n\n  /**\n   * Handle sidebar sync when tab becomes active\n   */\n  private handleSidebarSync(tabInfo: TabInfo): void {\n    // The sidebar sync will be handled automatically by the shared service\n    // when page data is loaded, so we don't need additional logic here\n    console.log('🔗 REQUEST SERVICE: Tab activated, sidebar sync handled by shared service:', tabInfo.title);\n  }\n\n  /**\n   * Handle route-based tab management - called by shared service\n   * This method checks for existing tabs and creates/activates as needed\n   */\n  handleRouteBasedTabManagement(\n    currentRoutePath: string,\n    queryParams: Record<string, unknown>,\n    layout: ISytmLayout,\n    pageData: PageDataResponse,\n    pageCode: string\n  ) {\n    console.log('handleRouteBasedTabManagement', currentRoutePath, queryParams, layout, pageData, pageCode);\n    const currentTabs = this.tabsSignal();\n    const tabAlreadyExists = currentTabs.find((t: TabInfo) => {\n      const tRoutePath = t.route.startsWith('/') ? t.route : '/' + t.route;\n      const tParamsMatch = JSON.stringify(t.params || {}) === JSON.stringify(queryParams || {});\n      return tRoutePath === currentRoutePath && tParamsMatch;\n    });\n    console.log('tabAlreadyExists', tabAlreadyExists);\n    if (!tabAlreadyExists) {\n      const title = pageData.data?.page?.sypg_title || pageCode.toString() || 'Untitled Tab';\n      console.log(`🆕 Adding new tab: ${title}`);\n      this.addTab(title, currentRoutePath, queryParams, layout);\n    } else if (tabAlreadyExists && !tabAlreadyExists.active) {\n      console.log(`🔄 Activating existing tab: ${tabAlreadyExists.title}`);\n      this.activateTab(tabAlreadyExists.id);\n    }else if (tabAlreadyExists && tabAlreadyExists.active) {\n      console.log(`🔄 Activating existing tab: ${tabAlreadyExists.title}`);\n      this.activateTab(tabAlreadyExists.id);\n    }\n  }\n\n  addTab(title: string, route: string, params?: Record<string, unknown>, layout?: ISytmLayout): string {\n    console.log('addTab', title, route, params, layout);\n    const id = this.generateId();\n    const currentTabs = this.tabsSignal();\n    console.log('currentTabs', currentTabs, route);\n    // Check if a tab with the same route and params already exists\n    const existingTab = currentTabs.find((tab: TabInfo) => {\n      const paramsMatch = JSON.stringify(tab.params || {}) === JSON.stringify(params || {});\n      return tab.route === route && paramsMatch;\n    });\n    console.log('existingTab', existingTab);\n    if (existingTab) {\n      this.activateTab(existingTab.id);\n      return existingTab.id;\n    }\n    \n    // Create new tab\n    const newTab: TabInfo = {\n      id,\n      title,\n      route,\n      params,\n      active: true,\n      layout,\n      sytm_page_id_sypg: '',\n      themeId: ''\n    };\n\n    // Deactivate all other tabs\n    currentTabs.forEach((tab: TabInfo) => tab.active = false);\n    console.log('currentTabs after deactivate', currentTabs, 'activateTab', id);\n    // Add new tab\n    this.tabsSignal.set([...currentTabs, newTab]);\n    this.activeTabIdSignal.set(id);\n\n    // Sync with TabStateService\n    this.tabStateService.addTab(title, id);\n    \n    // Request visibility is now controlled by layout configuration in setPageData\n    // No need to automatically show request wrapper here\n    \n    // Navigate to the new route\n    if (params) {\n      this.router.navigate([route], { queryParams: params });\n    } else {\n      this.router.navigate([route]);\n    }\n\n    return id;\n  }\n\n  activateTab(tabId: string): void {\n    console.log('🔍 REQUEST SERVICE: Activating tab:', tabId);\n    const tabs = this.tabsSignal();\n    const tabToActivate = tabs.find(tab => tab.id === tabId);\n    \n    if (!tabToActivate) {\n      console.warn('⚠️ Tab not found:', tabId);\n      return;\n    }\n\n    // Update tabs: deactivate all, then activate the target\n    const updatedTabs = tabs.map(tab => ({\n      ...tab,\n      active: tab.id === tabId\n    }));\n\n    this.tabsSignal.set(updatedTabs);\n    this.activeTabIdSignal.set(tabId);\n    \n    console.log('✅ REQUEST SERVICE: Tab activated:', tabToActivate.title);\n\n    // Navigate to the tab's route\n    this.router.navigate([tabToActivate.route], { \n      queryParams: tabToActivate.params || {},\n      replaceUrl: false \n    });\n    \n    // Trigger sidebar sync for the activated tab by calling shared service\n    // The shared service will handle the sidebar sync when the route loads\n    console.log('🔗 REQUEST SERVICE: Sidebar sync will be handled by route change');\n  }\n\n  closeTab(id: string): void {\n    const currentTabs = this.tabsSignal();\n    const tabIndex = currentTabs.findIndex((tab: TabInfo) => tab.id === id);\n\n    if (tabIndex === -1) return;\n\n    const tabToClose = currentTabs[tabIndex];\n    const wasActive = tabToClose.active;\n\n    // Sync with TabStateService by removing the tab state\n    this.tabStateService.removeTab(id);\n\n    // Clear stored route from strategy\n    if (this.routeReuseStrategy instanceof CustomRouteReuseStrategy) {\n      let pathKey = tabToClose.route.startsWith('/') ? tabToClose.route.substring(1) : tabToClose.route;\n\n      if (tabToClose.params && Object.keys(tabToClose.params).length > 0) {\n        const queryParamsString = Object.keys(tabToClose.params)\n          .sort()\n          .map(key => `${key}=${tabToClose.params![key]}`)\n          .join('&');\n        pathKey += '?' + queryParamsString;\n      }\n\n      (this.routeReuseStrategy as CustomRouteReuseStrategy).clearStoredRoute(pathKey);\n    }\n\n    // Remove the tab from this service's list\n    const newTabs = currentTabs.filter((tab: TabInfo) => tab.id !== id);\n\n    // If we're closing the active tab, activate the next available tab\n    if (wasActive && newTabs.length > 0) {\n      const newActiveIndex = Math.max(0, Math.min(tabIndex, newTabs.length - 1));\n      const newActiveTab = newTabs[newActiveIndex];\n\n      // Update sidedrawer for the new active tab\n      this.sidedrawerService.updateDrawerItems(newActiveTab.layout);\n\n      newTabs[newActiveIndex].active = true;\n      this.activeTabIdSignal.set(newActiveTab.id);\n      this.tabStateService.setActiveTab(newActiveTab.id);\n\n      // Navigate to the new active tab's route\n      if (newActiveTab.params) {\n        this.router.navigate([newActiveTab.route], { queryParams: newActiveTab.params });\n      } else {\n        this.router.navigate([newActiveTab.route]);\n      }\n    } else if (newTabs.length === 0) {\n      this.activeTabIdSignal.set(null);\n      this.tabStateService.setActiveTab(null);\n      // Clear sidedrawer as no tabs are open\n      this.sidedrawerService.updateDrawerItems(undefined);\n    }\n    this.tabsSignal.set(newTabs);\n    // Request wrapper visibility is now controlled by layout configuration\n    // The wrapper will be hidden/shown based on sytm_layout_request.status in setPageData\n  }\n\n  // Hide Request\n  hideRequest() {\n    console.log('🚫 REQUEST SERVICE - Hiding request wrapper');\n    this.requestVisible = false;\n    document.querySelector(`#cide-lyt-request-wrapper`)?.classList.add('cide-lyt-request-wrapper-hide');\n    document.querySelector(`body`)?.classList.remove('cide-lyt-request-exist');\n  }\n\n  // Show Request\n  showRequest() {\n    console.log('✅ REQUEST SERVICE - Showing request wrapper');\n    this.requestVisible = true;\n    document.querySelector(`#cide-lyt-request-wrapper`)?.classList.remove('cide-lyt-request-wrapper-hide');\n    document.querySelector(`body`)?.classList.add('cide-lyt-request-exist');\n  }\n\n  private generateId(): string {\n    return 'tab-' + Math.random().toString(36).substr(2, 9);\n  }\n\n  updateTabScrollPosition(tabId: string, scrollTop: number, scrollLeft: number): void {\n    const currentTabs = this.tabsSignal();\n    const tabIndex = currentTabs.findIndex((t: TabInfo) => t.id === tabId);\n    if (tabIndex !== -1) {\n      const tabToUpdate = currentTabs[tabIndex];\n      tabToUpdate.scrollTop = scrollTop;\n      tabToUpdate.scrollLeft = scrollLeft;\n      // Create a new array reference to trigger change detection\n      this.tabsSignal.set([...currentTabs]);\n    }\n  }\n}\n"]}
@@ -1,84 +0,0 @@
1
- import { Component, ViewChild, effect } from '@angular/core';
2
- import { CommonModule } from '@angular/common';
3
- import { NavigationEnd, RouterModule } from '@angular/router';
4
- import { filter } from 'rxjs/operators';
5
- import * as i0 from "@angular/core";
6
- import * as i1 from "../request.service";
7
- import * as i2 from "@angular/router";
8
- export class TabContentComponent {
9
- constructor(requestService, router, // Inject Router to listen for NavigationEnd
10
- cdr) {
11
- this.requestService = requestService;
12
- this.router = router;
13
- this.cdr = cdr;
14
- this.currentActiveTabId = null;
15
- this.isRestoringScroll = false; // Flag to prevent saving scroll during restoration attempts
16
- // Modern Angular signal pattern: Use effect to react to active tab changes
17
- effect(() => {
18
- const activeTab = this.requestService.activeTab();
19
- if (activeTab && activeTab.id !== this.currentActiveTabId) {
20
- this.currentActiveTabId = activeTab.id;
21
- // Delay scroll restoration to ensure DOM is ready
22
- const delay = (ms) => new Promise(resolve => setTimeout(resolve, ms));
23
- delay(100).then(() => this.restoreScrollPosition(activeTab));
24
- }
25
- });
26
- // Listen to route changes for scroll restoration
27
- this.router.events.pipe(filter(event => event instanceof NavigationEnd)).subscribe({
28
- next: () => {
29
- // Use signal to get active tab after navigation
30
- const activeTab = this.requestService.activeTab();
31
- if (activeTab && activeTab.id !== this.currentActiveTabId) {
32
- this.currentActiveTabId = activeTab.id;
33
- const delay = (ms) => new Promise(resolve => setTimeout(resolve, ms));
34
- delay(150).then(() => this.restoreScrollPosition(activeTab));
35
- }
36
- }
37
- });
38
- }
39
- ngAfterViewInit() {
40
- // Initial scroll restoration for the active tab
41
- const activeTab = this.requestService.activeTab();
42
- if (activeTab) {
43
- this.currentActiveTabId = activeTab.id;
44
- const delay = (ms) => new Promise(resolve => setTimeout(resolve, ms));
45
- delay(50).then(() => this.restoreScrollPosition(activeTab));
46
- }
47
- }
48
- onScroll() {
49
- if (this.isRestoringScroll || !this.scrollContainerRef || !this.currentActiveTabId) {
50
- return;
51
- }
52
- const { scrollTop, scrollLeft } = this.scrollContainerRef.nativeElement;
53
- this.requestService.updateTabScrollPosition(this.currentActiveTabId, scrollTop, scrollLeft);
54
- }
55
- restoreScrollPosition(tab) {
56
- if (tab && this.scrollContainerRef?.nativeElement) {
57
- this.isRestoringScroll = true;
58
- const container = this.scrollContainerRef.nativeElement;
59
- container.scrollTop = tab.scrollTop ?? 0;
60
- container.scrollLeft = tab.scrollLeft ?? 0;
61
- // Modern ES2022+ pattern: Use Promise-based delay
62
- const delay = (ms) => new Promise(resolve => setTimeout(resolve, ms));
63
- delay(150).then(() => { this.isRestoringScroll = false; }); // Allow scroll event to settle
64
- }
65
- }
66
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: TabContentComponent, deps: [{ token: i1.CideLytRequestService }, { token: i2.Router }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component }); }
67
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.7", type: TabContentComponent, isStandalone: true, selector: "cide-lyt-tab-content", viewQueries: [{ propertyName: "scrollContainerRef", first: true, predicate: ["scrollContainer"], descendants: true }], ngImport: i0, template: `
68
- <div class="tab-content-container" #scrollContainer (scroll)="onScroll()">
69
- <router-outlet></router-outlet>
70
- </div>
71
- `, isInline: true, styles: [":host{height:100%;display:flex;flex-direction:column}.tab-content-container{height:100%;width:100%;position:relative;overflow:auto;display:flex;flex-direction:column;flex:1}::ng-deep router-outlet+*{height:100%;flex:1;display:flex;flex-direction:column}::ng-deep cide-lyt-home-wrapper,::ng-deep cide-adm-home-wrapper,::ng-deep cide-adm-entity,::ng-deep cide-adm-entity-list{height:100%!important;flex:1!important;display:flex!important;flex-direction:column!important}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "ngmodule", type: RouterModule }, { kind: "directive", type: i2.RouterOutlet, selector: "router-outlet", inputs: ["name"], outputs: ["activate", "deactivate", "attach", "detach"], exportAs: ["outlet"] }] }); }
72
- }
73
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: TabContentComponent, decorators: [{
74
- type: Component,
75
- args: [{ selector: 'cide-lyt-tab-content', standalone: true, imports: [CommonModule, RouterModule], template: `
76
- <div class="tab-content-container" #scrollContainer (scroll)="onScroll()">
77
- <router-outlet></router-outlet>
78
- </div>
79
- `, styles: [":host{height:100%;display:flex;flex-direction:column}.tab-content-container{height:100%;width:100%;position:relative;overflow:auto;display:flex;flex-direction:column;flex:1}::ng-deep router-outlet+*{height:100%;flex:1;display:flex;flex-direction:column}::ng-deep cide-lyt-home-wrapper,::ng-deep cide-adm-home-wrapper,::ng-deep cide-adm-entity,::ng-deep cide-adm-entity-list{height:100%!important;flex:1!important;display:flex!important;flex-direction:column!important}\n"] }]
80
- }], ctorParameters: () => [{ type: i1.CideLytRequestService }, { type: i2.Router }, { type: i0.ChangeDetectorRef }], propDecorators: { scrollContainerRef: [{
81
- type: ViewChild,
82
- args: ['scrollContainer']
83
- }] } });
84
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tab-content.component.js","sourceRoot":"","sources":["../../../../../../../projects/cloud-ide-layout/src/lib/layout/request/tab-content/tab-content.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { Component, ElementRef, ViewChild, AfterViewInit, ChangeDetectorRef, effect } from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport { NavigationEnd, Router, RouterModule } from '@angular/router';\nimport { CideLytRequestService, TabInfo } from '../request.service';\nimport { filter } from 'rxjs/operators';\n\n@Component({\n  selector: 'cide-lyt-tab-content',\n  standalone: true,\n  imports: [CommonModule, RouterModule],\n  template: `\n    <div class=\"tab-content-container\" #scrollContainer (scroll)=\"onScroll()\">\n      <router-outlet></router-outlet>\n    </div>\n  `,\n  styles: [`\n    :host {\n      height: 100%;\n      display: flex;\n      flex-direction: column;\n    }\n    \n    .tab-content-container {\n      height: 100%;\n      width: 100%;\n      position: relative;\n      overflow: auto;\n      display: flex;\n      flex-direction: column;\n      flex: 1;\n    }\n    \n    /* Ensure router outlet components take full height */\n    ::ng-deep router-outlet + * {\n      height: 100%;\n      flex: 1;\n      display: flex;\n      flex-direction: column;\n    }\n    \n    /* Specific height fixes for common component selectors */\n    ::ng-deep cide-lyt-home-wrapper,\n    ::ng-deep cide-adm-home-wrapper,\n    ::ng-deep cide-adm-entity,\n    ::ng-deep cide-adm-entity-list {\n      height: 100% !important;\n      flex: 1 !important;\n      display: flex !important;\n      flex-direction: column !important;\n    }\n  `]\n})\nexport class TabContentComponent implements AfterViewInit {\n  @ViewChild('scrollContainer') scrollContainerRef!: ElementRef<HTMLDivElement>;\n\n  private currentActiveTabId: string | null = null;\n  private isRestoringScroll = false; // Flag to prevent saving scroll during restoration attempts\n\n  constructor(\n    private requestService: CideLytRequestService,\n    private router: Router, // Inject Router to listen for NavigationEnd\n    private cdr: ChangeDetectorRef\n  ) {\n    // Modern Angular signal pattern: Use effect to react to active tab changes\n    effect(() => {\n      const activeTab = this.requestService.activeTab();\n      if (activeTab && activeTab.id !== this.currentActiveTabId) {\n        this.currentActiveTabId = activeTab.id;\n        // Delay scroll restoration to ensure DOM is ready\n        const delay = (ms: number) => new Promise(resolve => setTimeout(resolve, ms));\n        delay(100).then(() => this.restoreScrollPosition(activeTab));\n      }\n    });\n\n    // Listen to route changes for scroll restoration\n    this.router.events.pipe(\n      filter(event => event instanceof NavigationEnd)\n    ).subscribe({\n      next: () => {\n        // Use signal to get active tab after navigation\n        const activeTab = this.requestService.activeTab();\n        if (activeTab && activeTab.id !== this.currentActiveTabId) {\n          this.currentActiveTabId = activeTab.id;\n          const delay = (ms: number) => new Promise(resolve => setTimeout(resolve, ms));\n          delay(150).then(() => this.restoreScrollPosition(activeTab));\n        }\n      }\n    });\n  }\n\n  ngAfterViewInit(): void {\n    // Initial scroll restoration for the active tab\n    const activeTab = this.requestService.activeTab();\n    if (activeTab) {\n      this.currentActiveTabId = activeTab.id;\n      const delay = (ms: number) => new Promise(resolve => setTimeout(resolve, ms));\n      delay(50).then(() => this.restoreScrollPosition(activeTab));\n    }\n  }\n\n  onScroll(): void {\n    if (this.isRestoringScroll || !this.scrollContainerRef || !this.currentActiveTabId) {\n      return;\n    }\n    const { scrollTop, scrollLeft } = this.scrollContainerRef.nativeElement;\n    this.requestService.updateTabScrollPosition(this.currentActiveTabId, scrollTop, scrollLeft);\n  }\n\n  private restoreScrollPosition(tab: TabInfo | undefined): void {\n    if (tab && this.scrollContainerRef?.nativeElement) {\n      this.isRestoringScroll = true;\n      const container = this.scrollContainerRef.nativeElement;\n      container.scrollTop = tab.scrollTop ?? 0;\n      container.scrollLeft = tab.scrollLeft ?? 0;\n      // Modern ES2022+ pattern: Use Promise-based delay\n      const delay = (ms: number) => new Promise(resolve => setTimeout(resolve, ms));\n      delay(150).then(() => { this.isRestoringScroll = false; }); // Allow scroll event to settle\n    }\n  }\n\n\n}\n"]}
@@ -1,49 +0,0 @@
1
- import { Component, inject } from '@angular/core';
2
- import { CideLytSharedService } from '../shared.service';
3
- import { CideLytSidedrawerService } from '../../sidedrawer/sidedrawer.service';
4
- import { AppStateHelperService } from '../../../services/app-state/app-state-helper.service';
5
- import * as i0 from "@angular/core";
6
- export class CideLytSharedWrapperComponent {
7
- constructor() {
8
- this.shared_wrapper_setup_param = {};
9
- this.page_data = {};
10
- this.sharedService = inject(CideLytSharedService);
11
- this.sidedrawerService = inject(CideLytSidedrawerService);
12
- this.appState = inject(AppStateHelperService);
13
- }
14
- ngOnInit() {
15
- // Load and process page data using modern approach
16
- if (this.shared_wrapper_setup_param?.sypg_page_code) {
17
- const entityId = this.appState.getActiveEntityId();
18
- const page_data_payload = {
19
- sypg_page_code: this.shared_wrapper_setup_param?.sypg_page_code,
20
- sytm_entity_id_syen: entityId || undefined
21
- };
22
- console.log('🔍 Page data payload:', page_data_payload);
23
- this.sharedService.loadAndProcessPageData(page_data_payload, {
24
- setTitle: true,
25
- setSidebarContext: true,
26
- updateLayout: true
27
- }).subscribe({
28
- next: (page_data_response) => {
29
- console.log('✅ SHARED WRAPPER: Page data loaded and processed for:', page_data_payload.sypg_page_code);
30
- this.page_data = page_data_response?.data || {};
31
- // Sidebar context is automatically set by loadAndProcessPageData when setSidebarContext: true
32
- const pageId = page_data_response?.data?.page?._id || '';
33
- const themeId = page_data_response?.data?.theme?._id || '';
34
- this.sidedrawerService.setContext(pageId, themeId);
35
- },
36
- error: (err) => {
37
- console.error('❌ SHARED WRAPPER: Error loading page data:', err);
38
- }
39
- });
40
- }
41
- }
42
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: CideLytSharedWrapperComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
43
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.7", type: CideLytSharedWrapperComponent, isStandalone: true, selector: "cide-lyt-shared-wrapper", ngImport: i0, template: "<p>shared-wrapper works!</p>\r\n", styles: [""] }); }
44
- }
45
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: CideLytSharedWrapperComponent, decorators: [{
46
- type: Component,
47
- args: [{ selector: 'cide-lyt-shared-wrapper', standalone: true, imports: [], template: "<p>shared-wrapper works!</p>\r\n" }]
48
- }] });
49
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2hhcmVkLXdyYXBwZXIuY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvY2xvdWQtaWRlLWxheW91dC9zcmMvbGliL2xheW91dC9zaGFyZWQvc2hhcmVkLXdyYXBwZXIvc2hhcmVkLXdyYXBwZXIuY29tcG9uZW50LnRzIiwiLi4vLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvY2xvdWQtaWRlLWxheW91dC9zcmMvbGliL2xheW91dC9zaGFyZWQvc2hhcmVkLXdyYXBwZXIvc2hhcmVkLXdyYXBwZXIuY29tcG9uZW50Lmh0bWwiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFNBQVMsRUFBVSxNQUFNLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDMUQsT0FBTyxFQUFFLG9CQUFvQixFQUFFLE1BQU0sbUJBQW1CLENBQUM7QUFFekQsT0FBTyxFQUFFLHdCQUF3QixFQUFFLE1BQU0scUNBQXFDLENBQUM7QUFDL0UsT0FBTyxFQUFFLHFCQUFxQixFQUFFLE1BQU0sc0RBQXNELENBQUM7O0FBUzdGLE1BQU0sT0FBTyw2QkFBNkI7SUFQMUM7UUFRUywrQkFBMEIsR0FBNEMsRUFBRSxDQUFDO1FBQ3pFLGNBQVMsR0FBNkIsRUFBRSxDQUFDO1FBQ3hDLGtCQUFhLEdBQUcsTUFBTSxDQUFDLG9CQUFvQixDQUFDLENBQUM7UUFDN0Msc0JBQWlCLEdBQUcsTUFBTSxDQUFDLHdCQUF3QixDQUFDLENBQUM7UUFDbkQsYUFBUSxHQUFHLE1BQU0sQ0FBQyxxQkFBcUIsQ0FBQyxDQUFDO0tBa0NwRDtJQWhDQyxRQUFRO1FBQ04sbURBQW1EO1FBQ25ELElBQUksSUFBSSxDQUFDLDBCQUEwQixFQUFFLGNBQWMsRUFBRSxDQUFDO1lBQ3BELE1BQU0sUUFBUSxHQUFHLElBQUksQ0FBQyxRQUFRLENBQUMsaUJBQWlCLEVBQUUsQ0FBQztZQUNuRCxNQUFNLGlCQUFpQixHQUFrQjtnQkFDdkMsY0FBYyxFQUFFLElBQUksQ0FBQywwQkFBMEIsRUFBRSxjQUFjO2dCQUMvRCxtQkFBbUIsRUFBRSxRQUFRLElBQUksU0FBUzthQUMzQyxDQUFDO1lBQ0YsT0FBTyxDQUFDLEdBQUcsQ0FBQyx1QkFBdUIsRUFBRSxpQkFBaUIsQ0FBQyxDQUFDO1lBRXhELElBQUksQ0FBQyxhQUFhLENBQUMsc0JBQXNCLENBQUMsaUJBQWlCLEVBQUU7Z0JBQzNELFFBQVEsRUFBRSxJQUFJO2dCQUNkLGlCQUFpQixFQUFFLElBQUk7Z0JBQ3ZCLFlBQVksRUFBRSxJQUFJO2FBQ25CLENBQUMsQ0FBQyxTQUFTLENBQUM7Z0JBQ1gsSUFBSSxFQUFFLENBQUMsa0JBQWtELEVBQUUsRUFBRTtvQkFDM0QsT0FBTyxDQUFDLEdBQUcsQ0FBQyx1REFBdUQsRUFBRSxpQkFBaUIsQ0FBQyxjQUFjLENBQUMsQ0FBQztvQkFDdkcsSUFBSSxDQUFDLFNBQVMsR0FBRyxrQkFBa0IsRUFBRSxJQUFJLElBQUksRUFBRSxDQUFDO29CQUVoRCw4RkFBOEY7b0JBQzlGLE1BQU0sTUFBTSxHQUFHLGtCQUFrQixFQUFFLElBQUksRUFBRSxJQUFJLEVBQUUsR0FBRyxJQUFJLEVBQUUsQ0FBQztvQkFDekQsTUFBTSxPQUFPLEdBQUcsa0JBQWtCLEVBQUUsSUFBSSxFQUFFLEtBQUssRUFBRSxHQUFHLElBQUksRUFBRSxDQUFDO29CQUMzRCxJQUFJLENBQUMsaUJBQWlCLENBQUMsVUFBVSxDQUFDLE1BQU0sRUFBRSxPQUFPLENBQUMsQ0FBQztnQkFDckQsQ0FBQztnQkFDRCxLQUFLLEVBQUUsQ0FBQyxHQUFHLEVBQUUsRUFBRTtvQkFDYixPQUFPLENBQUMsS0FBSyxDQUFDLDRDQUE0QyxFQUFFLEdBQUcsQ0FBQyxDQUFDO2dCQUNuRSxDQUFDO2FBQ0YsQ0FBQyxDQUFDO1FBQ0wsQ0FBQztJQUNILENBQUM7OEdBcENVLDZCQUE2QjtrR0FBN0IsNkJBQTZCLG1GQ2IxQyxrQ0FDQTs7MkZEWWEsNkJBQTZCO2tCQVB6QyxTQUFTOytCQUNFLHlCQUF5QixjQUN2QixJQUFJLFdBQ1AsRUFBRSIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENvbXBvbmVudCwgT25Jbml0LCBpbmplY3QgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IENpZGVMeXRTaGFyZWRTZXJ2aWNlIH0gZnJvbSAnLi4vc2hhcmVkLnNlcnZpY2UnO1xuaW1wb3J0IHsgZGVzaWduQ29uZmlnQ29udHJvbGxlclJlc3BvbnNlLCBkZXNpZ25Db25maWdSZXNwb25zZURhdGEsIE1EZXNpZ25Db25maWcgfSBmcm9tICdjbG91ZC1pZGUtbG1zLW1vZGVsJztcbmltcG9ydCB7IENpZGVMeXRTaWRlZHJhd2VyU2VydmljZSB9IGZyb20gJy4uLy4uL3NpZGVkcmF3ZXIvc2lkZWRyYXdlci5zZXJ2aWNlJztcbmltcG9ydCB7IEFwcFN0YXRlSGVscGVyU2VydmljZSB9IGZyb20gJy4uLy4uLy4uL3NlcnZpY2VzL2FwcC1zdGF0ZS9hcHAtc3RhdGUtaGVscGVyLnNlcnZpY2UnO1xuXG5AQ29tcG9uZW50KHtcbiAgc2VsZWN0b3I6ICdjaWRlLWx5dC1zaGFyZWQtd3JhcHBlcicsXG4gIHN0YW5kYWxvbmU6IHRydWUsXG4gIGltcG9ydHM6IFtdLFxuICB0ZW1wbGF0ZVVybDogJy4vc2hhcmVkLXdyYXBwZXIuY29tcG9uZW50Lmh0bWwnLFxuICBzdHlsZVVybDogJy4vc2hhcmVkLXdyYXBwZXIuY29tcG9uZW50LmNzcydcbn0pXG5leHBvcnQgY2xhc3MgQ2lkZUx5dFNoYXJlZFdyYXBwZXJDb21wb25lbnQgaW1wbGVtZW50cyBPbkluaXQge1xuICBwdWJsaWMgc2hhcmVkX3dyYXBwZXJfc2V0dXBfcGFyYW06IFBhcnRpYWw8Q2lkZUx5dFNoYXJlZFdyYXBwZXJTZXR1cFBhcmFtPiA9IHt9O1xuICBwdWJsaWMgcGFnZV9kYXRhOiBkZXNpZ25Db25maWdSZXNwb25zZURhdGEgPSB7fTtcbiAgcHJpdmF0ZSBzaGFyZWRTZXJ2aWNlID0gaW5qZWN0KENpZGVMeXRTaGFyZWRTZXJ2aWNlKTtcbiAgcHJpdmF0ZSBzaWRlZHJhd2VyU2VydmljZSA9IGluamVjdChDaWRlTHl0U2lkZWRyYXdlclNlcnZpY2UpO1xuICBwcm90ZWN0ZWQgYXBwU3RhdGUgPSBpbmplY3QoQXBwU3RhdGVIZWxwZXJTZXJ2aWNlKTtcblxuICBuZ09uSW5pdCgpIHtcbiAgICAvLyBMb2FkIGFuZCBwcm9jZXNzIHBhZ2UgZGF0YSB1c2luZyBtb2Rlcm4gYXBwcm9hY2hcbiAgICBpZiAodGhpcy5zaGFyZWRfd3JhcHBlcl9zZXR1cF9wYXJhbT8uc3lwZ19wYWdlX2NvZGUpIHtcbiAgICAgIGNvbnN0IGVudGl0eUlkID0gdGhpcy5hcHBTdGF0ZS5nZXRBY3RpdmVFbnRpdHlJZCgpO1xuICAgICAgY29uc3QgcGFnZV9kYXRhX3BheWxvYWQ6IE1EZXNpZ25Db25maWcgPSB7IFxuICAgICAgICBzeXBnX3BhZ2VfY29kZTogdGhpcy5zaGFyZWRfd3JhcHBlcl9zZXR1cF9wYXJhbT8uc3lwZ19wYWdlX2NvZGUsIFxuICAgICAgICBzeXRtX2VudGl0eV9pZF9zeWVuOiBlbnRpdHlJZCB8fCB1bmRlZmluZWQgXG4gICAgICB9O1xuICAgICAgY29uc29sZS5sb2coJ/CflI0gUGFnZSBkYXRhIHBheWxvYWQ6JywgcGFnZV9kYXRhX3BheWxvYWQpO1xuICAgICAgXG4gICAgICB0aGlzLnNoYXJlZFNlcnZpY2UubG9hZEFuZFByb2Nlc3NQYWdlRGF0YShwYWdlX2RhdGFfcGF5bG9hZCwge1xuICAgICAgICBzZXRUaXRsZTogdHJ1ZSxcbiAgICAgICAgc2V0U2lkZWJhckNvbnRleHQ6IHRydWUsXG4gICAgICAgIHVwZGF0ZUxheW91dDogdHJ1ZVxuICAgICAgfSkuc3Vic2NyaWJlKHtcbiAgICAgICAgbmV4dDogKHBhZ2VfZGF0YV9yZXNwb25zZTogZGVzaWduQ29uZmlnQ29udHJvbGxlclJlc3BvbnNlKSA9PiB7XG4gICAgICAgICAgY29uc29sZS5sb2coJ+KchSBTSEFSRUQgV1JBUFBFUjogUGFnZSBkYXRhIGxvYWRlZCBhbmQgcHJvY2Vzc2VkIGZvcjonLCBwYWdlX2RhdGFfcGF5bG9hZC5zeXBnX3BhZ2VfY29kZSk7XG4gICAgICAgICAgdGhpcy5wYWdlX2RhdGEgPSBwYWdlX2RhdGFfcmVzcG9uc2U/LmRhdGEgfHwge307XG4gICAgICAgICAgXG4gICAgICAgICAgLy8gU2lkZWJhciBjb250ZXh0IGlzIGF1dG9tYXRpY2FsbHkgc2V0IGJ5IGxvYWRBbmRQcm9jZXNzUGFnZURhdGEgd2hlbiBzZXRTaWRlYmFyQ29udGV4dDogdHJ1ZVxuICAgICAgICAgIGNvbnN0IHBhZ2VJZCA9IHBhZ2VfZGF0YV9yZXNwb25zZT8uZGF0YT8ucGFnZT8uX2lkIHx8ICcnO1xuICAgICAgICAgIGNvbnN0IHRoZW1lSWQgPSBwYWdlX2RhdGFfcmVzcG9uc2U/LmRhdGE/LnRoZW1lPy5faWQgfHwgJyc7XG4gICAgICAgICAgdGhpcy5zaWRlZHJhd2VyU2VydmljZS5zZXRDb250ZXh0KHBhZ2VJZCwgdGhlbWVJZCk7XG4gICAgICAgIH0sXG4gICAgICAgIGVycm9yOiAoZXJyKSA9PiB7XG4gICAgICAgICAgY29uc29sZS5lcnJvcign4p2MIFNIQVJFRCBXUkFQUEVSOiBFcnJvciBsb2FkaW5nIHBhZ2UgZGF0YTonLCBlcnIpO1xuICAgICAgICB9XG4gICAgICB9KTtcbiAgICB9XG4gIH1cblxuXG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgQ2lkZUx5dFNoYXJlZFdyYXBwZXJTZXR1cFBhcmFtIHtcbiAgc3lwZ19wYWdlX2NvZGU6IHN0cmluZ1xufSIsIjxwPnNoYXJlZC13cmFwcGVyIHdvcmtzITwvcD5cclxuIl19