verben-workflow-ui 0.2.2 → 0.2.4

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 (47) hide show
  1. package/esm2022/lib/components/designer/models/types.mjs +1 -1
  2. package/esm2022/lib/components/designer/services/wf-mapper.service.mjs +3 -3
  3. package/esm2022/lib/components/workflow-designer/conditions-popup/conditions-popup.component.mjs +76 -0
  4. package/esm2022/lib/components/workflow-designer/decision-popup/decision-popup.component.mjs +65 -0
  5. package/esm2022/lib/components/workflow-designer/designer-canvas/designer-canvas.component.mjs +245 -17
  6. package/esm2022/lib/components/workflow-designer/designer-toolbar/designer-toolbar.component.mjs +26 -14
  7. package/esm2022/lib/components/workflow-designer/services/connection.service.mjs +220 -0
  8. package/esm2022/lib/components/workflow-designer/services/node-management.service.mjs +288 -0
  9. package/esm2022/lib/components/workflow-designer/services/popup.service.mjs +235 -0
  10. package/esm2022/lib/components/workflow-designer/services/swimlane.service.mjs +145 -0
  11. package/esm2022/lib/components/workflow-designer/services/transformer.service.mjs +260 -0
  12. package/esm2022/lib/components/workflow-designer/stage-dialog/stage-dialog.component.mjs +67 -11
  13. package/esm2022/lib/components/workflow-designer/stage-node/stage-node.component.mjs +9 -18
  14. package/esm2022/lib/components/workflow-designer/swimlane-dialog/swimlane-dialog.component.mjs +30 -24
  15. package/esm2022/lib/components/workflow-designer/workflow-data.service.mjs +2 -2
  16. package/esm2022/lib/components/workflow-designer/workflow-designer.component.mjs +138 -96
  17. package/esm2022/lib/components/workflow-designer/workflow-designer.module.mjs +27 -3
  18. package/esm2022/lib/components/workflow-designer/workflow-designer.state.mjs +243 -205
  19. package/esm2022/lib/components/workflow-designer/workflow-designer.types.mjs +1 -1
  20. package/esm2022/lib/models/SwimLane.mjs +1 -1
  21. package/esm2022/lib/models/Workflow.mjs +1 -1
  22. package/esm2022/lib/models/WorkflowStage.mjs +1 -1
  23. package/esm2022/lib/services/http-web-request.service.mjs +2 -2
  24. package/fesm2022/verben-workflow-ui.mjs +2644 -1005
  25. package/fesm2022/verben-workflow-ui.mjs.map +1 -1
  26. package/lib/components/designer/models/types.d.ts +1 -1
  27. package/lib/components/workflow-designer/conditions-popup/conditions-popup.component.d.ts +34 -0
  28. package/lib/components/workflow-designer/decision-popup/decision-popup.component.d.ts +26 -0
  29. package/lib/components/workflow-designer/designer-canvas/designer-canvas.component.d.ts +35 -1
  30. package/lib/components/workflow-designer/designer-toolbar/designer-toolbar.component.d.ts +14 -3
  31. package/lib/components/workflow-designer/services/connection.service.d.ts +85 -0
  32. package/lib/components/workflow-designer/services/node-management.service.d.ts +52 -0
  33. package/lib/components/workflow-designer/services/popup.service.d.ts +139 -0
  34. package/lib/components/workflow-designer/services/swimlane.service.d.ts +51 -0
  35. package/lib/components/workflow-designer/services/transformer.service.d.ts +36 -0
  36. package/lib/components/workflow-designer/stage-dialog/stage-dialog.component.d.ts +17 -2
  37. package/lib/components/workflow-designer/stage-node/stage-node.component.d.ts +2 -2
  38. package/lib/components/workflow-designer/swimlane-dialog/swimlane-dialog.component.d.ts +7 -1
  39. package/lib/components/workflow-designer/workflow-designer.component.d.ts +2 -1
  40. package/lib/components/workflow-designer/workflow-designer.module.d.ts +6 -4
  41. package/lib/components/workflow-designer/workflow-designer.state.d.ts +17 -1
  42. package/lib/components/workflow-designer/workflow-designer.types.d.ts +2 -0
  43. package/lib/models/SwimLane.d.ts +1 -0
  44. package/lib/models/Workflow.d.ts +1 -0
  45. package/lib/models/WorkflowStage.d.ts +1 -1
  46. package/package.json +1 -1
  47. package/styles/styles.css +23 -0
@@ -41,22 +41,24 @@ export class WorkflowDesignerComponent {
41
41
  this.isSaving = true;
42
42
  // Transform the UI model to the API model
43
43
  const workflowModel = this.state.transformToWorkflowModel();
44
- // Save to the backend
45
- this.dataService
46
- .saveWorkflows([workflowModel])
47
- .then((response) => {
48
- console.log('Workflow saved successfully:', response);
49
- // If the API returns an updated workflow, we might want to reload it
50
- if (response?.Result?.Code) {
51
- this.workflowCode = response.Result.Code;
52
- }
53
- this.isSaving = false;
54
- })
55
- .catch((error) => {
56
- console.error('Error saving workflow:', error);
57
- this.isSaving = false;
58
- // Handle error - show a notification, etc.
59
- });
44
+ if (workflowModel) {
45
+ // Save to the backend
46
+ this.dataService
47
+ .saveWorkflows([workflowModel])
48
+ .then((response) => {
49
+ console.log('Workflow saved successfully:', response);
50
+ // If the API returns an updated workflow, we might want to reload it
51
+ if (response?.Result?.Code) {
52
+ this.workflowCode = response.Result.Code;
53
+ }
54
+ this.isSaving = false;
55
+ })
56
+ .catch((error) => {
57
+ console.error('Error saving workflow:', error);
58
+ this.isSaving = false;
59
+ // Handle error - show a notification, etc.
60
+ });
61
+ }
60
62
  }
61
63
  loadWorkflow(code) {
62
64
  this.isLoading = true;
@@ -64,7 +66,10 @@ export class WorkflowDesignerComponent {
64
66
  .getWorkflowWithParam(code)
65
67
  .then((response) => {
66
68
  if (response && response.Result) {
67
- this.parseWorkflowData(response.Result);
69
+ // Store the workflow ID
70
+ const wf = response.Result[0];
71
+ this.state.setWorkflowId(wf.Id);
72
+ this.parseWorkflowData(wf);
68
73
  }
69
74
  this.isLoading = false;
70
75
  })
@@ -75,64 +80,115 @@ export class WorkflowDesignerComponent {
75
80
  }
76
81
  // Method to parse workflow data from API and populate the designer
77
82
  parseWorkflowData(workflow) {
83
+ // Store the complete workflow
84
+ this.state.workflow = workflow;
78
85
  // Clear existing state
79
86
  this.state.swimlanes = [];
80
87
  this.state.connections = [];
88
+ this.state.swimlaneRecord = {};
89
+ this.state.stageRecord = {};
90
+ this.state.actionRecord = {};
81
91
  // Set workflow form if exists
82
92
  if (workflow.Form) {
83
93
  this.state.setWorkflowForm(workflow.Form, workflow.FormName || 'Workflow Form');
84
94
  }
95
+ // Store the workflow ID
96
+ this.state.setWorkflowId(workflow.Id);
85
97
  // Process swimlanes first
86
98
  if (workflow.Lanes && workflow.Lanes.length) {
87
99
  workflow.Lanes.sort((a, b) => a.Position - b.Position).forEach((lane) => {
88
- this.state.addSwimlane(lane.Name || `Lane ${lane.Position}`, lane.Tags || []);
100
+ // Store original lane in record
101
+ this.state.swimlaneRecord[lane.Id] = lane;
102
+ // Add swimlane to UI model with the original ID
103
+ const swimlane = {
104
+ order: lane.Position,
105
+ label: lane.Name || `Lane ${lane.Position}`,
106
+ tags: lane.Tags || [],
107
+ nodes: [],
108
+ id: lane.Id, // Store the original ID
109
+ };
110
+ this.state.swimlanes.push(swimlane);
111
+ // Register loaded object
112
+ this.state.registerLoadedObject(lane.Id, lane.Code);
89
113
  });
90
114
  }
91
115
  // Process stages
92
116
  if (workflow.Stages && workflow.Stages.length) {
93
117
  workflow.Stages.forEach((stage) => {
94
- // Find swimlane index
95
- const swimlaneIndex = this.findSwimlaneIndexByLaneId(stage.SwimLane);
118
+ // Store original stage in record
119
+ this.state.stageRecord[stage.Id] = stage;
120
+ // Find swimlane index by SwimLane ID
121
+ const swimlaneIndex = this.state.swimlanes.findIndex((lane) => lane.id === stage.SwimLane);
96
122
  if (swimlaneIndex !== -1) {
97
123
  const x = stage.Coordinates?.X || 100;
98
124
  const y = stage.Coordinates?.Y || 50;
99
- this.state.addNode(swimlaneIndex, 'stage', x, y, {
100
- Name: stage.Name,
101
- Description: stage.Description,
102
- Duration: stage.Duration,
103
- PassOnRule: stage.PassOnRule,
104
- ActorRule: stage.ActorRule,
105
- MinNoOfActor: stage.MinNoOfActor,
106
- Tags: stage.Tags || [],
107
- IsParallel: stage.IsParallel,
108
- IsEntryPoint: stage.IsEntryPoint,
109
- IsExitPoint: stage.IsExitPoint,
110
- Id: stage.Id,
111
- });
125
+ // Create node with original ID
126
+ const node = {
127
+ id: stage.Id, // Use original ID
128
+ type: 'stage',
129
+ x: x,
130
+ y: y - swimlaneIndex * 263 - 40, // Adjust Y for swimlane offset
131
+ width: 169,
132
+ height: 100,
133
+ isStartNode: stage.IsEntryPoint,
134
+ stageData: {
135
+ Name: stage.Name,
136
+ Description: stage.Description,
137
+ Duration: stage.Duration,
138
+ PassOnRule: stage.PassOnRule,
139
+ ActorRule: stage.ActorRule,
140
+ MinNoOfActor: stage.MinNoOfActor,
141
+ Tags: stage.Tags || [],
142
+ IsParallel: stage.IsParallel,
143
+ IsEntryPoint: stage.IsEntryPoint,
144
+ IsExitPoint: stage.IsExitPoint,
145
+ Id: stage.Id, // Keep original ID
146
+ },
147
+ connectionPoints: [], // Initialize with empty array
148
+ };
149
+ // Generate connection points
150
+ node.connectionPoints = this.state.generateConnectionPoints(node);
151
+ // Add node to swimlane
152
+ this.state.swimlanes[swimlaneIndex].nodes =
153
+ this.state.swimlanes[swimlaneIndex].nodes || [];
154
+ this.state.swimlanes[swimlaneIndex].nodes.push(node);
155
+ // If this is a start node, update state
156
+ if (stage.IsEntryPoint) {
157
+ this.state.startNodeId = stage.Id;
158
+ }
159
+ // Register loaded stage
160
+ this.state.registerLoadedObject(stage.Id, stage.Code);
112
161
  }
113
162
  });
114
163
  }
115
164
  // Process connections/actions
116
165
  if (workflow.Actions && workflow.Actions.length) {
117
166
  workflow.Actions.forEach((action) => {
167
+ // Store original action in record
168
+ this.state.actionRecord[action.Id] = action;
118
169
  // Find source and target nodes
119
170
  const sourceNodeInfo = this.state.findNodeById(action.FromStage);
120
171
  const targetNodeInfo = this.state.findNodeById(action.ToStage);
121
172
  if (sourceNodeInfo && targetNodeInfo) {
122
173
  // Find suitable connection points
123
- const sourcePoint = sourceNodeInfo.node.connectionPoints?.[0];
124
- const targetPoint = targetNodeInfo.node.connectionPoints?.[0];
174
+ const sourceNode = sourceNodeInfo.node;
175
+ const targetNode = targetNodeInfo.node;
176
+ const sourcePoint = sourceNode.connectionPoints?.find((p) => p.type === 'right' || p.type === 'bottom');
177
+ const targetPoint = targetNode.connectionPoints?.find((p) => p.type === 'left' || p.type === 'top');
125
178
  if (sourcePoint && targetPoint) {
126
179
  const connection = {
127
- id: action.Id,
180
+ id: action.Id, // Use original ID
128
181
  sourceNodeId: action.FromStage,
129
182
  targetNodeId: action.ToStage,
130
183
  sourcePointId: sourcePoint.id,
131
184
  targetPointId: targetPoint.id,
132
185
  sourceSwimlaneIndex: sourceNodeInfo.swimlaneIndex,
133
186
  targetSwimlaneIndex: targetNodeInfo.swimlaneIndex,
187
+ condition: action.PassOnRule || '',
134
188
  };
135
189
  this.state.connections.push(connection);
190
+ // Register loaded action
191
+ this.state.registerLoadedObject(action.Id, action.Code);
136
192
  }
137
193
  }
138
194
  });
@@ -140,8 +196,11 @@ export class WorkflowDesignerComponent {
140
196
  }
141
197
  // Helper method to find swimlane index by Lane ID
142
198
  findSwimlaneIndexByLaneId(laneId) {
143
- // In a real implementation, you would maintain a mapping between API lane IDs and UI swimlane indices
144
- // For now we'll just return the index based on position
199
+ // First check our mapping
200
+ if (this.state.laneIdToIndexMap.has(laneId)) {
201
+ return this.state.laneIdToIndexMap.get(laneId);
202
+ }
203
+ // Fallback to the original logic (for backward compatibility)
145
204
  const parts = laneId.split('-');
146
205
  if (parts.length > 1) {
147
206
  const index = parseInt(parts[1]);
@@ -198,18 +257,18 @@ export class WorkflowDesignerComponent {
198
257
  console.log('Opening swimlane dialog for new swimlane');
199
258
  this.editingSwimlaneIndex = null; // Set to null to indicate creating a new swimlane
200
259
  this.openSwimlaneDialog();
260
+ this.selectedTool = null;
201
261
  break;
202
262
  case 'stage':
203
263
  if (swimlaneIndex >= 0 && swimlaneIndex < this.state.swimlanes.length) {
204
264
  console.log(`Adding ${this.selectedTool} to swimlane ${swimlaneIndex}`);
205
- // Store position for later node creation
206
- this.pendingStagePosition = {
207
- swimlaneIndex,
208
- x: event.x,
209
- y: event.y,
210
- };
211
- // Show stage properties dialog
212
- this.showStageDialog = true;
265
+ // Create the stage directly with default properties
266
+ const node = this.state.addNode(swimlaneIndex, 'stage', event.x, event.y, { Name: 'New Stage' } // Add default properties
267
+ );
268
+ if (node) {
269
+ // Reset the selected tool after placing a node
270
+ this.selectedTool = null;
271
+ }
213
272
  }
214
273
  break;
215
274
  case 'decision':
@@ -255,56 +314,39 @@ export class WorkflowDesignerComponent {
255
314
  this.editingSwimlaneIndex = null;
256
315
  }
257
316
  onStageDialogSaved(stageData) {
258
- if (this.pendingStagePosition) {
259
- // Create the stage node
260
- const node = this.state.addNode(this.pendingStagePosition.swimlaneIndex, 'stage', this.pendingStagePosition.x, this.pendingStagePosition.y, stageData);
261
- console.log('Source point on dialog save:', this.state.draggingConnectionData.sourcePoint);
262
- console.log('Is creating from connection:', this.isCreatingStageFromConnection);
263
- if (node) {
264
- // If this was created from a connection, create the connection
265
- if (this.isCreatingStageFromConnection &&
266
- this.pendingConnectionSourcePoint) {
267
- console.log('Using stored source point:', this.pendingConnectionSourcePoint);
268
- // Find a suitable connection point on the new node
269
- if (node.connectionPoints && node.connectionPoints.length > 0) {
270
- const sourcePoint = this.pendingConnectionSourcePoint;
271
- // Find opposing type connection point
272
- const opposingType = {
273
- right: 'left',
274
- left: 'right',
275
- top: 'bottom',
276
- bottom: 'top',
277
- }[sourcePoint.type];
278
- const targetPoint = node.connectionPoints.find((p) => p.type === opposingType);
279
- if (targetPoint) {
280
- // Create the connection using our stored source data
281
- const connection = {
282
- id: `conn-${Date.now()}-${Math.floor(Math.random() * 1000)}`,
283
- sourceNodeId: sourcePoint.nodeId,
284
- targetNodeId: node.id,
285
- sourcePointId: sourcePoint.id,
286
- targetPointId: targetPoint.id,
287
- sourceSwimlaneIndex: this.pendingConnectionSourceSwimlaneIndex,
288
- targetSwimlaneIndex: this.pendingStagePosition.swimlaneIndex,
289
- };
290
- // Add connection to the state
291
- this.state.connections.push(connection);
292
- console.log('Created connection:', connection);
293
- }
294
- }
295
- // End the connection drag
296
- // this.state.endConnectionDrag();
317
+ if (this.canvasRef.activeStageId) {
318
+ // Editing an existing stage
319
+ const nodeInfo = this.state.findNodeById(this.canvasRef.activeStageId);
320
+ if (nodeInfo) {
321
+ // Update the stage data
322
+ nodeInfo.node.stageData = {
323
+ ...nodeInfo.node.stageData,
324
+ ...stageData,
325
+ };
326
+ console.log('Updated stage data:', nodeInfo.node.stageData);
327
+ }
328
+ this.canvasRef.activeStageId = null;
329
+ }
330
+ else if (this.pendingStagePosition) {
331
+ if (this.canvasRef.activeStageId) {
332
+ // Editing an existing stage
333
+ const nodeInfo = this.state.findNodeById(this.canvasRef.activeStageId);
334
+ if (nodeInfo) {
335
+ // Update the stage data
336
+ nodeInfo.node.stageData = {
337
+ ...nodeInfo.node.stageData,
338
+ ...stageData,
339
+ };
340
+ console.log('Updated stage data:', nodeInfo.node.stageData);
297
341
  }
298
- // Reset the selected tool after placing a node
299
- this.selectedTool = null;
342
+ this.canvasRef.activeStageId = null;
343
+ }
344
+ else if (this.pendingStagePosition) {
345
+ // Creating a new stage (keep existing logic for this case)
346
+ // ...existing code...
300
347
  }
301
- // Clear pending data
302
- this.pendingStagePosition = null;
303
- this.isCreatingStageFromConnection = false;
304
- this.pendingConnectionSourcePoint = null;
305
- this.pendingConnectionSourceSwimlaneIndex = null;
348
+ this.showStageDialog = false;
306
349
  }
307
- console.log('Connections after adding new one:', this.state.connections);
308
350
  this.showStageDialog = false;
309
351
  }
310
352
  onSubflowSelected() {
@@ -312,15 +354,15 @@ export class WorkflowDesignerComponent {
312
354
  this.selectedTool = null;
313
355
  }
314
356
  static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: WorkflowDesignerComponent, deps: [{ token: i1.WorkflowDesignerState }, { token: i2.WorkflowDataService }], target: i0.ɵɵFactoryTarget.Component });
315
- static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: WorkflowDesignerComponent, selector: "lib-workflow-designer", inputs: { workflowCode: "workflowCode" }, viewQueries: [{ propertyName: "canvasRef", first: true, predicate: ["canvasRef"], descendants: true }], ngImport: i0, template: "<div class=\"workflow-designer\">\n <div class=\"workflow-header\">\n <h1>Workflow Designer</h1>\n </div>\n\n <!-- Toolbar Component -->\n <lib-designer-toolbar\n [selectedTool]=\"selectedTool\"\n (toolSelected)=\"onToolSelected($event)\"\n >\n </lib-designer-toolbar>\n\n <!-- Show loading indicator if needed -->\n <div *ngIf=\"isLoading\" class=\"loading-overlay\">\n <div class=\"loading-spinner\"></div>\n <div class=\"loading-text\">Loading workflow...</div>\n </div>\n\n <!-- Canvas Component -->\n <lib-designer-canvas\n [selectedTool]=\"selectedTool\"\n (clickedPosition)=\"handleCanvasPositionClick($event)\"\n (subflowSelected)=\"onSubflowSelected()\"\n (showStageDialog)=\"onShowStageDialog($event)\"\n #canvasRef\n >\n </lib-designer-canvas>\n\n <lib-swimlane-dialog\n [visible]=\"showSwimlaneDialog\"\n [swimlaneData]=\"\n editingSwimlaneIndex !== null\n ? {\n name: state.swimlanes[editingSwimlaneIndex].label,\n tags: state.swimlanes[editingSwimlaneIndex].tags\n }\n : null\n \"\n (created)=\"onSwimlaneDialogFilled($event)\"\n (closed)=\"showSwimlaneDialog = false\"\n ></lib-swimlane-dialog>\n\n <lib-stage-dialog\n [visible]=\"showStageDialog\"\n [stageData]=\"pendingStageData\"\n (closed)=\"showStageDialog = false\"\n (saved)=\"onStageDialogSaved($event)\"\n ></lib-stage-dialog>\n</div>\n", styles: [".workflow-designer{display:flex;flex-direction:column;height:100vh;background-color:#f8fafc}.workflow-header{padding:1rem;background-color:#fff;border-bottom:1px solid #e2e8f0}.workflow-header h1{font-size:1.25rem;font-weight:600;color:#334155;margin:0}.loading-overlay{position:absolute;inset:0;background-color:#ffffffb3;display:flex;flex-direction:column;justify-content:center;align-items:center;z-index:1000}.loading-spinner{border:4px solid #f3f3f3;border-top:4px solid #d8b4fe;border-radius:50%;width:40px;height:40px;animation:spin 1s linear infinite}.loading-text{margin-top:1rem;font-size:1rem;color:#7e22ce}@keyframes spin{0%{transform:rotate(0)}to{transform:rotate(360deg)}}\n"], dependencies: [{ kind: "directive", type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i4.DesignerToolbarComponent, selector: "lib-designer-toolbar", inputs: ["selectedTool", "isSaving"], outputs: ["toolSelected", "saveWorkflow"] }, { kind: "component", type: i5.DesignerCanvasComponent, selector: "lib-designer-canvas", inputs: ["selectedTool"], outputs: ["clickedPosition", "subflowSelected", "showStageDialog"] }, { kind: "component", type: i6.SwimlaneDialogComponent, selector: "lib-swimlane-dialog", inputs: ["visible", "swimlaneData"], outputs: ["closed", "created"] }, { kind: "component", type: i7.StageDialogComponent, selector: "lib-stage-dialog", inputs: ["visible", "stageData"], outputs: ["closed", "saved"] }] });
357
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: WorkflowDesignerComponent, selector: "lib-workflow-designer", inputs: { workflowCode: "workflowCode" }, viewQueries: [{ propertyName: "canvasRef", first: true, predicate: ["canvasRef"], descendants: true }], ngImport: i0, template: "<div class=\"workflow-designer\">\n <div class=\"workflow-header\">\n <h1>Workflow Designer</h1>\n </div>\n\n <!-- Toolbar Component -->\n <lib-designer-toolbar\n [selectedTool]=\"selectedTool\"\n [isSaving]=\"isSaving\"\n (toolSelected)=\"onToolSelected($event)\"\n (saveWorkflow)=\"saveWorkflow()\"\n >\n </lib-designer-toolbar>\n\n <!-- Show loading indicator if needed -->\n <div *ngIf=\"isLoading\" class=\"loading-overlay\">\n <div class=\"loading-spinner\"></div>\n <div class=\"loading-text\">Loading workflow...</div>\n </div>\n\n <!-- Canvas Component -->\n <lib-designer-canvas\n [selectedTool]=\"selectedTool\"\n (clickedPosition)=\"handleCanvasPositionClick($event)\"\n (subflowSelected)=\"onSubflowSelected()\"\n (showStageDialog)=\"onShowStageDialog($event)\"\n #canvasRef\n >\n </lib-designer-canvas>\n\n <lib-swimlane-dialog\n [visible]=\"showSwimlaneDialog\"\n [swimlaneData]=\"\n editingSwimlaneIndex !== null\n ? {\n name: state.swimlanes[editingSwimlaneIndex].label,\n tags: state.swimlanes[editingSwimlaneIndex].tags\n }\n : null\n \"\n (created)=\"onSwimlaneDialogFilled($event)\"\n (closed)=\"showSwimlaneDialog = false\"\n ></lib-swimlane-dialog>\n\n <lib-stage-dialog\n [visible]=\"showStageDialog\"\n [stageData]=\"pendingStageData\"\n (closed)=\"showStageDialog = false\"\n (saved)=\"onStageDialogSaved($event)\"\n ></lib-stage-dialog>\n</div>\n", styles: [".workflow-designer{display:flex;flex-direction:column;height:100vh;background-color:#f8fafc}.workflow-header{padding:1rem;background-color:#fff;border-bottom:1px solid #e2e8f0}.workflow-header h1{font-size:1.25rem;font-weight:600;color:#334155;margin:0}.loading-overlay{position:absolute;inset:0;background-color:#ffffffb3;display:flex;flex-direction:column;justify-content:center;align-items:center;z-index:1000}.loading-spinner{border:4px solid #f3f3f3;border-top:4px solid #d8b4fe;border-radius:50%;width:40px;height:40px;animation:spin 1s linear infinite}.loading-text{margin-top:1rem;font-size:1rem;color:#7e22ce}@keyframes spin{0%{transform:rotate(0)}to{transform:rotate(360deg)}}\n"], dependencies: [{ kind: "directive", type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i4.DesignerToolbarComponent, selector: "lib-designer-toolbar", inputs: ["selectedTool", "isSaving"], outputs: ["toolSelected", "saveWorkflow"] }, { kind: "component", type: i5.DesignerCanvasComponent, selector: "lib-designer-canvas", inputs: ["selectedTool"], outputs: ["clickedPosition", "subflowSelected", "showStageDialog", "stagePropertiesUpdated"] }, { kind: "component", type: i6.SwimlaneDialogComponent, selector: "lib-swimlane-dialog", inputs: ["visible", "swimlaneData"], outputs: ["closed", "created"] }, { kind: "component", type: i7.StageDialogComponent, selector: "lib-stage-dialog", inputs: ["visible", "stageData"], outputs: ["closed", "saved"] }] });
316
358
  }
317
359
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: WorkflowDesignerComponent, decorators: [{
318
360
  type: Component,
319
- args: [{ selector: 'lib-workflow-designer', template: "<div class=\"workflow-designer\">\n <div class=\"workflow-header\">\n <h1>Workflow Designer</h1>\n </div>\n\n <!-- Toolbar Component -->\n <lib-designer-toolbar\n [selectedTool]=\"selectedTool\"\n (toolSelected)=\"onToolSelected($event)\"\n >\n </lib-designer-toolbar>\n\n <!-- Show loading indicator if needed -->\n <div *ngIf=\"isLoading\" class=\"loading-overlay\">\n <div class=\"loading-spinner\"></div>\n <div class=\"loading-text\">Loading workflow...</div>\n </div>\n\n <!-- Canvas Component -->\n <lib-designer-canvas\n [selectedTool]=\"selectedTool\"\n (clickedPosition)=\"handleCanvasPositionClick($event)\"\n (subflowSelected)=\"onSubflowSelected()\"\n (showStageDialog)=\"onShowStageDialog($event)\"\n #canvasRef\n >\n </lib-designer-canvas>\n\n <lib-swimlane-dialog\n [visible]=\"showSwimlaneDialog\"\n [swimlaneData]=\"\n editingSwimlaneIndex !== null\n ? {\n name: state.swimlanes[editingSwimlaneIndex].label,\n tags: state.swimlanes[editingSwimlaneIndex].tags\n }\n : null\n \"\n (created)=\"onSwimlaneDialogFilled($event)\"\n (closed)=\"showSwimlaneDialog = false\"\n ></lib-swimlane-dialog>\n\n <lib-stage-dialog\n [visible]=\"showStageDialog\"\n [stageData]=\"pendingStageData\"\n (closed)=\"showStageDialog = false\"\n (saved)=\"onStageDialogSaved($event)\"\n ></lib-stage-dialog>\n</div>\n", styles: [".workflow-designer{display:flex;flex-direction:column;height:100vh;background-color:#f8fafc}.workflow-header{padding:1rem;background-color:#fff;border-bottom:1px solid #e2e8f0}.workflow-header h1{font-size:1.25rem;font-weight:600;color:#334155;margin:0}.loading-overlay{position:absolute;inset:0;background-color:#ffffffb3;display:flex;flex-direction:column;justify-content:center;align-items:center;z-index:1000}.loading-spinner{border:4px solid #f3f3f3;border-top:4px solid #d8b4fe;border-radius:50%;width:40px;height:40px;animation:spin 1s linear infinite}.loading-text{margin-top:1rem;font-size:1rem;color:#7e22ce}@keyframes spin{0%{transform:rotate(0)}to{transform:rotate(360deg)}}\n"] }]
361
+ args: [{ selector: 'lib-workflow-designer', template: "<div class=\"workflow-designer\">\n <div class=\"workflow-header\">\n <h1>Workflow Designer</h1>\n </div>\n\n <!-- Toolbar Component -->\n <lib-designer-toolbar\n [selectedTool]=\"selectedTool\"\n [isSaving]=\"isSaving\"\n (toolSelected)=\"onToolSelected($event)\"\n (saveWorkflow)=\"saveWorkflow()\"\n >\n </lib-designer-toolbar>\n\n <!-- Show loading indicator if needed -->\n <div *ngIf=\"isLoading\" class=\"loading-overlay\">\n <div class=\"loading-spinner\"></div>\n <div class=\"loading-text\">Loading workflow...</div>\n </div>\n\n <!-- Canvas Component -->\n <lib-designer-canvas\n [selectedTool]=\"selectedTool\"\n (clickedPosition)=\"handleCanvasPositionClick($event)\"\n (subflowSelected)=\"onSubflowSelected()\"\n (showStageDialog)=\"onShowStageDialog($event)\"\n #canvasRef\n >\n </lib-designer-canvas>\n\n <lib-swimlane-dialog\n [visible]=\"showSwimlaneDialog\"\n [swimlaneData]=\"\n editingSwimlaneIndex !== null\n ? {\n name: state.swimlanes[editingSwimlaneIndex].label,\n tags: state.swimlanes[editingSwimlaneIndex].tags\n }\n : null\n \"\n (created)=\"onSwimlaneDialogFilled($event)\"\n (closed)=\"showSwimlaneDialog = false\"\n ></lib-swimlane-dialog>\n\n <lib-stage-dialog\n [visible]=\"showStageDialog\"\n [stageData]=\"pendingStageData\"\n (closed)=\"showStageDialog = false\"\n (saved)=\"onStageDialogSaved($event)\"\n ></lib-stage-dialog>\n</div>\n", styles: [".workflow-designer{display:flex;flex-direction:column;height:100vh;background-color:#f8fafc}.workflow-header{padding:1rem;background-color:#fff;border-bottom:1px solid #e2e8f0}.workflow-header h1{font-size:1.25rem;font-weight:600;color:#334155;margin:0}.loading-overlay{position:absolute;inset:0;background-color:#ffffffb3;display:flex;flex-direction:column;justify-content:center;align-items:center;z-index:1000}.loading-spinner{border:4px solid #f3f3f3;border-top:4px solid #d8b4fe;border-radius:50%;width:40px;height:40px;animation:spin 1s linear infinite}.loading-text{margin-top:1rem;font-size:1rem;color:#7e22ce}@keyframes spin{0%{transform:rotate(0)}to{transform:rotate(360deg)}}\n"] }]
320
362
  }], ctorParameters: () => [{ type: i1.WorkflowDesignerState }, { type: i2.WorkflowDataService }], propDecorators: { canvasRef: [{
321
363
  type: ViewChild,
322
364
  args: ['canvasRef']
323
365
  }], workflowCode: [{
324
366
  type: Input
325
367
  }] } });
326
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"workflow-designer.component.js","sourceRoot":"","sources":["../../../../../../projects/verben-workflow-ui/src/lib/components/workflow-designer/workflow-designer.component.ts","../../../../../../projects/verben-workflow-ui/src/lib/components/workflow-designer/workflow-designer.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, ViewChild } from '@angular/core';\nimport { ConnectionPoint, ToolType } from './workflow-designer.types';\nimport { WorkflowDesignerState } from './workflow-designer.state';\nimport { Tag } from '../../models/Tag';\nimport { WorkflowStage } from '../../models/WorkflowStage';\nimport { DesignerCanvasComponent } from './designer-canvas/designer-canvas.component';\nimport { WorkflowDataService } from './workflow-data.service';\n\n@Component({\n  selector: 'lib-workflow-designer',\n  templateUrl: './workflow-designer.component.html',\n  styleUrl: './workflow-designer.component.css',\n})\nexport class WorkflowDesignerComponent {\n  @ViewChild('canvasRef') canvasRef!: DesignerCanvasComponent;\n  @Input() workflowCode: string | null = null;\n\n  // Currently selected tool\n  selectedTool: ToolType = null;\n  showSwimlaneDialog: boolean = false;\n\n  // New properties for stage dialog\n  showStageDialog: boolean = false;\n  pendingStagePosition: { swimlaneIndex: number; x: number; y: number } | null =\n    null;\n  pendingStageData: Partial<WorkflowStage> = {};\n\n  editingSwimlaneIndex: number | null = null;\n  isCreatingStageFromConnection: boolean = false;\n\n  pendingConnectionSourcePoint: ConnectionPoint | null = null;\n  pendingConnectionSourceSwimlaneIndex: number | null = null;\n\n  isLoading: boolean = false;\n  isSaving: boolean = false;\n\n  constructor(\n    public state: WorkflowDesignerState,\n    private dataService: WorkflowDataService\n  ) {}\n\n  ngOnInit(): void {\n    // Load workflow data if code is provided\n    if (this.workflowCode) {\n      this.loadWorkflow(this.workflowCode);\n    }\n  }\n\n  saveWorkflow(): void {\n    if (this.isSaving) return;\n\n    this.isSaving = true;\n\n    // Transform the UI model to the API model\n    const workflowModel = this.state.transformToWorkflowModel();\n\n    // Save to the backend\n    this.dataService\n      .saveWorkflows([workflowModel])\n      .then((response) => {\n        console.log('Workflow saved successfully:', response);\n        // If the API returns an updated workflow, we might want to reload it\n        if (response?.Result?.Code) {\n          this.workflowCode = response.Result.Code;\n        }\n        this.isSaving = false;\n      })\n      .catch((error) => {\n        console.error('Error saving workflow:', error);\n        this.isSaving = false;\n        // Handle error - show a notification, etc.\n      });\n  }\n\n  loadWorkflow(code: string): void {\n    this.isLoading = true;\n    this.dataService\n      .getWorkflowWithParam(code)\n      .then((response) => {\n        if (response && response.Result) {\n          this.parseWorkflowData(response.Result);\n        }\n        this.isLoading = false;\n      })\n      .catch((error) => {\n        console.error('Error loading workflow:', error);\n        this.isLoading = false;\n      });\n  }\n\n  // Method to parse workflow data from API and populate the designer\n  parseWorkflowData(workflow: any): void {\n    // Clear existing state\n    this.state.swimlanes = [];\n    this.state.connections = [];\n\n    // Set workflow form if exists\n    if (workflow.Form) {\n      this.state.setWorkflowForm(\n        workflow.Form,\n        workflow.FormName || 'Workflow Form'\n      );\n    }\n\n    // Process swimlanes first\n    if (workflow.Lanes && workflow.Lanes.length) {\n      workflow.Lanes.sort((a: any, b: any) => a.Position - b.Position).forEach(\n        (lane: any) => {\n          this.state.addSwimlane(\n            lane.Name || `Lane ${lane.Position}`,\n            lane.Tags || []\n          );\n        }\n      );\n    }\n\n    // Process stages\n    if (workflow.Stages && workflow.Stages.length) {\n      workflow.Stages.forEach((stage: any) => {\n        // Find swimlane index\n        const swimlaneIndex = this.findSwimlaneIndexByLaneId(stage.SwimLane);\n        if (swimlaneIndex !== -1) {\n          const x = stage.Coordinates?.X || 100;\n          const y = stage.Coordinates?.Y || 50;\n\n          this.state.addNode(swimlaneIndex, 'stage', x, y, {\n            Name: stage.Name,\n            Description: stage.Description,\n            Duration: stage.Duration,\n            PassOnRule: stage.PassOnRule,\n            ActorRule: stage.ActorRule,\n            MinNoOfActor: stage.MinNoOfActor,\n            Tags: stage.Tags || [],\n            IsParallel: stage.IsParallel,\n            IsEntryPoint: stage.IsEntryPoint,\n            IsExitPoint: stage.IsExitPoint,\n            Id: stage.Id,\n          });\n        }\n      });\n    }\n\n    // Process connections/actions\n    if (workflow.Actions && workflow.Actions.length) {\n      workflow.Actions.forEach((action: any) => {\n        // Find source and target nodes\n        const sourceNodeInfo = this.state.findNodeById(action.FromStage);\n        const targetNodeInfo = this.state.findNodeById(action.ToStage);\n\n        if (sourceNodeInfo && targetNodeInfo) {\n          // Find suitable connection points\n          const sourcePoint = sourceNodeInfo.node.connectionPoints?.[0];\n          const targetPoint = targetNodeInfo.node.connectionPoints?.[0];\n\n          if (sourcePoint && targetPoint) {\n            const connection = {\n              id: action.Id,\n              sourceNodeId: action.FromStage,\n              targetNodeId: action.ToStage,\n              sourcePointId: sourcePoint.id,\n              targetPointId: targetPoint.id,\n              sourceSwimlaneIndex: sourceNodeInfo.swimlaneIndex,\n              targetSwimlaneIndex: targetNodeInfo.swimlaneIndex,\n            };\n            this.state.connections.push(connection);\n          }\n        }\n      });\n    }\n  }\n\n  // Helper method to find swimlane index by Lane ID\n  findSwimlaneIndexByLaneId(laneId: string): number {\n    // In a real implementation, you would maintain a mapping between API lane IDs and UI swimlane indices\n    // For now we'll just return the index based on position\n    const parts = laneId.split('-');\n    if (parts.length > 1) {\n      const index = parseInt(parts[1]);\n      if (!isNaN(index) && index < this.state.swimlanes.length) {\n        return index;\n      }\n    }\n    return -1;\n  }\n\n  // Handle tool selection from toolbar\n  onToolSelected(tool: ToolType): void {\n    // Toggle selection if the same tool is clicked again\n    this.selectedTool = this.selectedTool === tool ? null : tool;\n    console.log('Selected tool:', this.selectedTool);\n  }\n\n  openSwimlaneDialog() {\n    this.showSwimlaneDialog = true;\n  }\n\n  onShowStageDialog(event: { position: any; isFromConnection: boolean }) {\n    console.log('Received showStageDialog event:', event);\n    this.pendingStagePosition = event.position;\n    this.showStageDialog = true;\n\n    // If this is from a connection creation, store that info\n    this.isCreatingStageFromConnection = event.isFromConnection;\n\n    // Store the source point data before it gets lost\n    if (\n      event.isFromConnection &&\n      this.state.draggingConnectionData.sourcePoint\n    ) {\n      this.pendingConnectionSourcePoint =\n        this.state.draggingConnectionData.sourcePoint;\n      this.pendingConnectionSourceSwimlaneIndex =\n        this.state.draggingConnectionData.sourceSwimlaneIndex ?? null;\n      console.log(\n        'Stored source point for connection:',\n        this.pendingConnectionSourcePoint\n      );\n    }\n\n    console.log('Stage dialog state:', {\n      pendingPosition: this.pendingStagePosition,\n      isVisible: this.showStageDialog,\n    });\n  }\n\n  handleCanvasPositionClick(event: { x: number; y: number }) {\n    console.log(event);\n\n    // Check if this is an edit swimlane event\n    if (\n      event.x === -1 &&\n      event.y >= 0 &&\n      event.y < this.state.swimlanes.length\n    ) {\n      console.log('Editing swimlane', event.y);\n      this.editingSwimlaneIndex = event.y;\n      this.openSwimlaneDialog();\n      return;\n    }\n\n    // Normal canvas click handling\n    const swimlaneIndex = Math.floor(event.y / 263);\n\n    switch (this.selectedTool) {\n      case 'swimlane':\n        console.log('Opening swimlane dialog for new swimlane');\n        this.editingSwimlaneIndex = null; // Set to null to indicate creating a new swimlane\n        this.openSwimlaneDialog();\n        break;\n      case 'stage':\n        if (swimlaneIndex >= 0 && swimlaneIndex < this.state.swimlanes.length) {\n          console.log(\n            `Adding ${this.selectedTool} to swimlane ${swimlaneIndex}`\n          );\n\n          // Store position for later node creation\n          this.pendingStagePosition = {\n            swimlaneIndex,\n            x: event.x,\n            y: event.y,\n          };\n\n          // Show stage properties dialog\n          this.showStageDialog = true;\n        }\n        break;\n      case 'decision':\n      case 'form':\n        if (swimlaneIndex >= 0 && swimlaneIndex < this.state.swimlanes.length) {\n          console.log(\n            `Adding ${this.selectedTool} to swimlane ${swimlaneIndex}`\n          );\n          const node = this.state.addNode(\n            swimlaneIndex,\n            this.selectedTool,\n            event.x,\n            event.y\n          );\n          if (node) {\n            // Reset the selected tool after placing a node\n            this.selectedTool = null;\n          }\n        }\n        break;\n      case 'subflow':\n        if (swimlaneIndex >= 0 && swimlaneIndex < this.state.swimlanes.length) {\n          console.log(\n            `Adding ${this.selectedTool} to swimlane ${swimlaneIndex}`\n          );\n          // Show subflow workflow selection popup\n          this.canvasRef.showSubflowSelectionPopup(\n            event.x,\n            event.y,\n            swimlaneIndex\n          );\n        }\n        break;\n      default:\n        break;\n    }\n  }\n\n  onSwimlaneDialogFilled(event: { tags: Tag[]; name: string }) {\n    console.log(\n      'Swimlane dialog filled:',\n      event,\n      'Editing index:',\n      this.editingSwimlaneIndex\n    );\n\n    // Validate that name is not empty\n    if (!event.name || event.name.trim() === '') {\n      console.error('Swimlane name cannot be empty');\n      return;\n    }\n\n    if (this.editingSwimlaneIndex !== null) {\n      // Update existing swimlane\n      console.log(\n        'Updating existing swimlane at index:',\n        this.editingSwimlaneIndex\n      );\n      this.state.updateSwimlane(\n        this.editingSwimlaneIndex,\n        event.name,\n        event.tags\n      );\n    } else {\n      // Create new swimlane\n      console.log('Creating new swimlane');\n      this.state.addSwimlane(event.name, event.tags);\n    }\n\n    this.showSwimlaneDialog = false;\n    this.editingSwimlaneIndex = null;\n  }\n\n  onStageDialogSaved(stageData: Partial<WorkflowStage>) {\n    if (this.pendingStagePosition) {\n      // Create the stage node\n      const node = this.state.addNode(\n        this.pendingStagePosition.swimlaneIndex,\n        'stage',\n        this.pendingStagePosition.x,\n        this.pendingStagePosition.y,\n        stageData\n      );\n\n      console.log(\n        'Source point on dialog save:',\n        this.state.draggingConnectionData.sourcePoint\n      );\n      console.log(\n        'Is creating from connection:',\n        this.isCreatingStageFromConnection\n      );\n\n      if (node) {\n        // If this was created from a connection, create the connection\n        if (\n          this.isCreatingStageFromConnection &&\n          this.pendingConnectionSourcePoint\n        ) {\n          console.log(\n            'Using stored source point:',\n            this.pendingConnectionSourcePoint\n          );\n\n          // Find a suitable connection point on the new node\n          if (node.connectionPoints && node.connectionPoints.length > 0) {\n            const sourcePoint = this.pendingConnectionSourcePoint;\n\n            // Find opposing type connection point\n            const opposingType = {\n              right: 'left',\n              left: 'right',\n              top: 'bottom',\n              bottom: 'top',\n            }[sourcePoint.type] as 'top' | 'right' | 'bottom' | 'left';\n\n            const targetPoint = node.connectionPoints.find(\n              (p) => p.type === opposingType\n            );\n\n            if (targetPoint) {\n              // Create the connection using our stored source data\n              const connection = {\n                id: `conn-${Date.now()}-${Math.floor(Math.random() * 1000)}`,\n                sourceNodeId: sourcePoint.nodeId,\n                targetNodeId: node.id,\n                sourcePointId: sourcePoint.id,\n                targetPointId: targetPoint.id,\n                sourceSwimlaneIndex: this.pendingConnectionSourceSwimlaneIndex!,\n                targetSwimlaneIndex: this.pendingStagePosition.swimlaneIndex,\n              };\n\n              // Add connection to the state\n              this.state.connections.push(connection);\n              console.log('Created connection:', connection);\n            }\n          }\n\n          // End the connection drag\n          // this.state.endConnectionDrag();\n        }\n\n        // Reset the selected tool after placing a node\n        this.selectedTool = null;\n      }\n\n      // Clear pending data\n      this.pendingStagePosition = null;\n      this.isCreatingStageFromConnection = false;\n      this.pendingConnectionSourcePoint = null;\n      this.pendingConnectionSourceSwimlaneIndex = null;\n    }\n\n    console.log('Connections after adding new one:', this.state.connections);\n\n    this.showStageDialog = false;\n  }\n\n  onSubflowSelected() {\n    // Reset the selected tool after placing a subflow node\n    this.selectedTool = null;\n  }\n}\n","<div class=\"workflow-designer\">\n  <div class=\"workflow-header\">\n    <h1>Workflow Designer</h1>\n  </div>\n\n  <!-- Toolbar Component -->\n  <lib-designer-toolbar\n    [selectedTool]=\"selectedTool\"\n    (toolSelected)=\"onToolSelected($event)\"\n  >\n  </lib-designer-toolbar>\n\n  <!-- Show loading indicator if needed -->\n  <div *ngIf=\"isLoading\" class=\"loading-overlay\">\n    <div class=\"loading-spinner\"></div>\n    <div class=\"loading-text\">Loading workflow...</div>\n  </div>\n\n  <!-- Canvas Component -->\n  <lib-designer-canvas\n    [selectedTool]=\"selectedTool\"\n    (clickedPosition)=\"handleCanvasPositionClick($event)\"\n    (subflowSelected)=\"onSubflowSelected()\"\n    (showStageDialog)=\"onShowStageDialog($event)\"\n    #canvasRef\n  >\n  </lib-designer-canvas>\n\n  <lib-swimlane-dialog\n    [visible]=\"showSwimlaneDialog\"\n    [swimlaneData]=\"\n      editingSwimlaneIndex !== null\n        ? {\n            name: state.swimlanes[editingSwimlaneIndex].label,\n            tags: state.swimlanes[editingSwimlaneIndex].tags\n          }\n        : null\n    \"\n    (created)=\"onSwimlaneDialogFilled($event)\"\n    (closed)=\"showSwimlaneDialog = false\"\n  ></lib-swimlane-dialog>\n\n  <lib-stage-dialog\n    [visible]=\"showStageDialog\"\n    [stageData]=\"pendingStageData\"\n    (closed)=\"showStageDialog = false\"\n    (saved)=\"onStageDialogSaved($event)\"\n  ></lib-stage-dialog>\n</div>\n"]}
368
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"workflow-designer.component.js","sourceRoot":"","sources":["../../../../../../projects/verben-workflow-ui/src/lib/components/workflow-designer/workflow-designer.component.ts","../../../../../../projects/verben-workflow-ui/src/lib/components/workflow-designer/workflow-designer.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, ViewChild } from '@angular/core';\nimport { ConnectionPoint, Node, ToolType } from './workflow-designer.types';\nimport { WorkflowDesignerState } from './workflow-designer.state';\nimport { Tag } from '../../models/Tag';\nimport { WorkflowStage } from '../../models/WorkflowStage';\nimport { DesignerCanvasComponent } from './designer-canvas/designer-canvas.component';\nimport { WorkflowDataService } from './workflow-data.service';\nimport { Workflow } from '../../models/Workflow';\n\n@Component({\n  selector: 'lib-workflow-designer',\n  templateUrl: './workflow-designer.component.html',\n  styleUrl: './workflow-designer.component.css',\n})\nexport class WorkflowDesignerComponent {\n  @ViewChild('canvasRef') canvasRef!: DesignerCanvasComponent;\n  @Input() workflowCode: string | null = null;\n\n  // Currently selected tool\n  selectedTool: ToolType = null;\n  showSwimlaneDialog: boolean = false;\n\n  // New properties for stage dialog\n  showStageDialog: boolean = false;\n  pendingStagePosition: { swimlaneIndex: number; x: number; y: number } | null =\n    null;\n  pendingStageData: Partial<WorkflowStage> = {};\n\n  editingSwimlaneIndex: number | null = null;\n  isCreatingStageFromConnection: boolean = false;\n\n  pendingConnectionSourcePoint: ConnectionPoint | null = null;\n  pendingConnectionSourceSwimlaneIndex: number | null = null;\n\n  isLoading: boolean = false;\n  isSaving: boolean = false;\n\n  constructor(\n    public state: WorkflowDesignerState,\n    private dataService: WorkflowDataService\n  ) {}\n\n  ngOnInit(): void {\n    // Load workflow data if code is provided\n    if (this.workflowCode) {\n      this.loadWorkflow(this.workflowCode);\n    }\n  }\n\n  saveWorkflow(): void {\n    if (this.isSaving) return;\n\n    this.isSaving = true;\n\n    // Transform the UI model to the API model\n    const workflowModel = this.state.transformToWorkflowModel();\n\n    if (workflowModel) {\n      // Save to the backend\n      this.dataService\n        .saveWorkflows([workflowModel])\n        .then((response) => {\n          console.log('Workflow saved successfully:', response);\n          // If the API returns an updated workflow, we might want to reload it\n          if (response?.Result?.Code) {\n            this.workflowCode = response.Result.Code;\n          }\n          this.isSaving = false;\n        })\n        .catch((error) => {\n          console.error('Error saving workflow:', error);\n          this.isSaving = false;\n          // Handle error - show a notification, etc.\n        });\n    }\n  }\n\n  loadWorkflow(code: string): void {\n    this.isLoading = true;\n    this.dataService\n      .getWorkflowWithParam(code)\n      .then((response) => {\n        if (response && response.Result) {\n          // Store the workflow ID\n          const wf = response.Result[0];\n          this.state.setWorkflowId(wf.Id);\n          this.parseWorkflowData(wf);\n        }\n        this.isLoading = false;\n      })\n      .catch((error) => {\n        console.error('Error loading workflow:', error);\n        this.isLoading = false;\n      });\n  }\n\n  // Method to parse workflow data from API and populate the designer\n  parseWorkflowData(workflow: Workflow): void {\n    // Store the complete workflow\n    this.state.workflow = workflow;\n\n    // Clear existing state\n    this.state.swimlanes = [];\n    this.state.connections = [];\n    this.state.swimlaneRecord = {};\n    this.state.stageRecord = {};\n    this.state.actionRecord = {};\n\n    // Set workflow form if exists\n    if (workflow.Form) {\n      this.state.setWorkflowForm(\n        workflow.Form,\n        workflow.FormName || 'Workflow Form'\n      );\n    }\n\n    // Store the workflow ID\n    this.state.setWorkflowId(workflow.Id);\n\n    // Process swimlanes first\n    if (workflow.Lanes && workflow.Lanes.length) {\n      workflow.Lanes.sort((a, b) => a.Position - b.Position).forEach((lane) => {\n        // Store original lane in record\n        this.state.swimlaneRecord[lane.Id] = lane;\n\n        // Add swimlane to UI model with the original ID\n        const swimlane = {\n          order: lane.Position,\n          label: lane.Name || `Lane ${lane.Position}`,\n          tags: lane.Tags || [],\n          nodes: [],\n          id: lane.Id, // Store the original ID\n        };\n\n        this.state.swimlanes.push(swimlane);\n\n        // Register loaded object\n        this.state.registerLoadedObject(lane.Id, lane.Code);\n      });\n    }\n\n    // Process stages\n    if (workflow.Stages && workflow.Stages.length) {\n      workflow.Stages.forEach((stage) => {\n        // Store original stage in record\n        this.state.stageRecord[stage.Id] = stage;\n\n        // Find swimlane index by SwimLane ID\n        const swimlaneIndex = this.state.swimlanes.findIndex(\n          (lane) => lane.id === stage.SwimLane\n        );\n\n        if (swimlaneIndex !== -1) {\n          const x = stage.Coordinates?.X || 100;\n          const y = stage.Coordinates?.Y || 50;\n\n          // Create node with original ID\n          const node: Node = {\n            id: stage.Id, // Use original ID\n            type: 'stage' as const,\n            x: x,\n            y: y - swimlaneIndex * 263 - 40, // Adjust Y for swimlane offset\n            width: 169,\n            height: 100,\n            isStartNode: stage.IsEntryPoint,\n            stageData: {\n              Name: stage.Name,\n              Description: stage.Description,\n              Duration: stage.Duration,\n              PassOnRule: stage.PassOnRule,\n              ActorRule: stage.ActorRule,\n              MinNoOfActor: stage.MinNoOfActor,\n              Tags: stage.Tags || [],\n              IsParallel: stage.IsParallel,\n              IsEntryPoint: stage.IsEntryPoint,\n              IsExitPoint: stage.IsExitPoint,\n              Id: stage.Id, // Keep original ID\n            },\n            connectionPoints: [], // Initialize with empty array\n          };\n\n          // Generate connection points\n          node.connectionPoints = this.state.generateConnectionPoints(node);\n\n          // Add node to swimlane\n          this.state.swimlanes[swimlaneIndex].nodes =\n            this.state.swimlanes[swimlaneIndex].nodes || [];\n          this.state.swimlanes[swimlaneIndex].nodes.push(node);\n\n          // If this is a start node, update state\n          if (stage.IsEntryPoint) {\n            this.state.startNodeId = stage.Id;\n          }\n\n          // Register loaded stage\n          this.state.registerLoadedObject(stage.Id, stage.Code);\n        }\n      });\n    }\n\n    // Process connections/actions\n    if (workflow.Actions && workflow.Actions.length) {\n      workflow.Actions.forEach((action) => {\n        // Store original action in record\n        this.state.actionRecord[action.Id] = action;\n\n        // Find source and target nodes\n        const sourceNodeInfo = this.state.findNodeById(action.FromStage);\n        const targetNodeInfo = this.state.findNodeById(action.ToStage);\n\n        if (sourceNodeInfo && targetNodeInfo) {\n          // Find suitable connection points\n          const sourceNode = sourceNodeInfo.node;\n          const targetNode = targetNodeInfo.node;\n\n          const sourcePoint = sourceNode.connectionPoints?.find(\n            (p) => p.type === 'right' || p.type === 'bottom'\n          );\n\n          const targetPoint = targetNode.connectionPoints?.find(\n            (p) => p.type === 'left' || p.type === 'top'\n          );\n\n          if (sourcePoint && targetPoint) {\n            const connection = {\n              id: action.Id, // Use original ID\n              sourceNodeId: action.FromStage,\n              targetNodeId: action.ToStage,\n              sourcePointId: sourcePoint.id,\n              targetPointId: targetPoint.id,\n              sourceSwimlaneIndex: sourceNodeInfo.swimlaneIndex,\n              targetSwimlaneIndex: targetNodeInfo.swimlaneIndex,\n              condition: action.PassOnRule || '',\n            };\n\n            this.state.connections.push(connection);\n\n            // Register loaded action\n            this.state.registerLoadedObject(action.Id, action.Code);\n          }\n        }\n      });\n    }\n  }\n\n  // Helper method to find swimlane index by Lane ID\n  findSwimlaneIndexByLaneId(laneId: string): number {\n    // First check our mapping\n    if (this.state.laneIdToIndexMap.has(laneId)) {\n      return this.state.laneIdToIndexMap.get(laneId)!;\n    }\n\n    // Fallback to the original logic (for backward compatibility)\n    const parts = laneId.split('-');\n    if (parts.length > 1) {\n      const index = parseInt(parts[1]);\n      if (!isNaN(index) && index < this.state.swimlanes.length) {\n        return index;\n      }\n    }\n\n    return -1;\n  }\n\n  // Handle tool selection from toolbar\n  onToolSelected(tool: ToolType): void {\n    // Toggle selection if the same tool is clicked again\n    this.selectedTool = this.selectedTool === tool ? null : tool;\n    console.log('Selected tool:', this.selectedTool);\n  }\n\n  openSwimlaneDialog() {\n    this.showSwimlaneDialog = true;\n  }\n\n  onShowStageDialog(event: { position: any; isFromConnection: boolean }) {\n    console.log('Received showStageDialog event:', event);\n    this.pendingStagePosition = event.position;\n    this.showStageDialog = true;\n\n    // If this is from a connection creation, store that info\n    this.isCreatingStageFromConnection = event.isFromConnection;\n\n    // Store the source point data before it gets lost\n    if (\n      event.isFromConnection &&\n      this.state.draggingConnectionData.sourcePoint\n    ) {\n      this.pendingConnectionSourcePoint =\n        this.state.draggingConnectionData.sourcePoint;\n      this.pendingConnectionSourceSwimlaneIndex =\n        this.state.draggingConnectionData.sourceSwimlaneIndex ?? null;\n      console.log(\n        'Stored source point for connection:',\n        this.pendingConnectionSourcePoint\n      );\n    }\n\n    console.log('Stage dialog state:', {\n      pendingPosition: this.pendingStagePosition,\n      isVisible: this.showStageDialog,\n    });\n  }\n\n  handleCanvasPositionClick(event: { x: number; y: number }) {\n    console.log(event);\n\n    // Check if this is an edit swimlane event\n    if (\n      event.x === -1 &&\n      event.y >= 0 &&\n      event.y < this.state.swimlanes.length\n    ) {\n      console.log('Editing swimlane', event.y);\n      this.editingSwimlaneIndex = event.y;\n      this.openSwimlaneDialog();\n      return;\n    }\n\n    // Normal canvas click handling\n    const swimlaneIndex = Math.floor(event.y / 263);\n\n    switch (this.selectedTool) {\n      case 'swimlane':\n        console.log('Opening swimlane dialog for new swimlane');\n        this.editingSwimlaneIndex = null; // Set to null to indicate creating a new swimlane\n        this.openSwimlaneDialog();\n        this.selectedTool = null;\n        break;\n      case 'stage':\n        if (swimlaneIndex >= 0 && swimlaneIndex < this.state.swimlanes.length) {\n          console.log(\n            `Adding ${this.selectedTool} to swimlane ${swimlaneIndex}`\n          );\n\n          // Create the stage directly with default properties\n          const node = this.state.addNode(\n            swimlaneIndex,\n            'stage',\n            event.x,\n            event.y,\n            { Name: 'New Stage' } // Add default properties\n          );\n\n          if (node) {\n            // Reset the selected tool after placing a node\n            this.selectedTool = null;\n          }\n        }\n        break;\n      case 'decision':\n      case 'form':\n        if (swimlaneIndex >= 0 && swimlaneIndex < this.state.swimlanes.length) {\n          console.log(\n            `Adding ${this.selectedTool} to swimlane ${swimlaneIndex}`\n          );\n          const node = this.state.addNode(\n            swimlaneIndex,\n            this.selectedTool,\n            event.x,\n            event.y\n          );\n          if (node) {\n            // Reset the selected tool after placing a node\n            this.selectedTool = null;\n          }\n        }\n        break;\n      case 'subflow':\n        if (swimlaneIndex >= 0 && swimlaneIndex < this.state.swimlanes.length) {\n          console.log(\n            `Adding ${this.selectedTool} to swimlane ${swimlaneIndex}`\n          );\n          // Show subflow workflow selection popup\n          this.canvasRef.showSubflowSelectionPopup(\n            event.x,\n            event.y,\n            swimlaneIndex\n          );\n        }\n        break;\n      default:\n        break;\n    }\n  }\n\n  onSwimlaneDialogFilled(event: { tags: Tag[]; name: string }) {\n    console.log(\n      'Swimlane dialog filled:',\n      event,\n      'Editing index:',\n      this.editingSwimlaneIndex\n    );\n\n    // Validate that name is not empty\n    if (!event.name || event.name.trim() === '') {\n      console.error('Swimlane name cannot be empty');\n      return;\n    }\n\n    if (this.editingSwimlaneIndex !== null) {\n      // Update existing swimlane\n      console.log(\n        'Updating existing swimlane at index:',\n        this.editingSwimlaneIndex\n      );\n      this.state.updateSwimlane(\n        this.editingSwimlaneIndex,\n        event.name,\n        event.tags\n      );\n    } else {\n      // Create new swimlane\n      console.log('Creating new swimlane');\n      this.state.addSwimlane(event.name, event.tags);\n    }\n\n    this.showSwimlaneDialog = false;\n    this.editingSwimlaneIndex = null;\n  }\n\n  onStageDialogSaved(stageData: Partial<WorkflowStage>) {\n    if (this.canvasRef.activeStageId) {\n      // Editing an existing stage\n      const nodeInfo = this.state.findNodeById(this.canvasRef.activeStageId);\n      if (nodeInfo) {\n        // Update the stage data\n        nodeInfo.node.stageData = {\n          ...nodeInfo.node.stageData,\n          ...stageData,\n        };\n        console.log('Updated stage data:', nodeInfo.node.stageData);\n      }\n      this.canvasRef.activeStageId = null;\n    } else if (this.pendingStagePosition) {\n      if (this.canvasRef.activeStageId) {\n        // Editing an existing stage\n        const nodeInfo = this.state.findNodeById(this.canvasRef.activeStageId);\n        if (nodeInfo) {\n          // Update the stage data\n          nodeInfo.node.stageData = {\n            ...nodeInfo.node.stageData,\n            ...stageData,\n          };\n          console.log('Updated stage data:', nodeInfo.node.stageData);\n        }\n        this.canvasRef.activeStageId = null;\n      } else if (this.pendingStagePosition) {\n        // Creating a new stage (keep existing logic for this case)\n        // ...existing code...\n      }\n\n      this.showStageDialog = false;\n    }\n\n    this.showStageDialog = false;\n  }\n\n  onSubflowSelected() {\n    // Reset the selected tool after placing a subflow node\n    this.selectedTool = null;\n  }\n}\n","<div class=\"workflow-designer\">\n  <div class=\"workflow-header\">\n    <h1>Workflow Designer</h1>\n  </div>\n\n  <!-- Toolbar Component -->\n  <lib-designer-toolbar\n    [selectedTool]=\"selectedTool\"\n    [isSaving]=\"isSaving\"\n    (toolSelected)=\"onToolSelected($event)\"\n    (saveWorkflow)=\"saveWorkflow()\"\n  >\n  </lib-designer-toolbar>\n\n  <!-- Show loading indicator if needed -->\n  <div *ngIf=\"isLoading\" class=\"loading-overlay\">\n    <div class=\"loading-spinner\"></div>\n    <div class=\"loading-text\">Loading workflow...</div>\n  </div>\n\n  <!-- Canvas Component -->\n  <lib-designer-canvas\n    [selectedTool]=\"selectedTool\"\n    (clickedPosition)=\"handleCanvasPositionClick($event)\"\n    (subflowSelected)=\"onSubflowSelected()\"\n    (showStageDialog)=\"onShowStageDialog($event)\"\n    #canvasRef\n  >\n  </lib-designer-canvas>\n\n  <lib-swimlane-dialog\n    [visible]=\"showSwimlaneDialog\"\n    [swimlaneData]=\"\n      editingSwimlaneIndex !== null\n        ? {\n            name: state.swimlanes[editingSwimlaneIndex].label,\n            tags: state.swimlanes[editingSwimlaneIndex].tags\n          }\n        : null\n    \"\n    (created)=\"onSwimlaneDialogFilled($event)\"\n    (closed)=\"showSwimlaneDialog = false\"\n  ></lib-swimlane-dialog>\n\n  <lib-stage-dialog\n    [visible]=\"showStageDialog\"\n    [stageData]=\"pendingStageData\"\n    (closed)=\"showStageDialog = false\"\n    (saved)=\"onStageDialogSaved($event)\"\n  ></lib-stage-dialog>\n</div>\n"]}
@@ -8,6 +8,13 @@ import { VerbenDialogueModule, VerbenPopUpModule } from 'verben-ng-ui';
8
8
  import { FormsModule, ReactiveFormsModule } from '@angular/forms';
9
9
  import { StageNodeComponent } from './stage-node/stage-node.component';
10
10
  import { StageDialogComponent } from './stage-dialog/stage-dialog.component';
11
+ import { ConditionsPopupComponent } from './conditions-popup/conditions-popup.component';
12
+ import { NodeManagementService } from './services/node-management.service';
13
+ import { ConnectionService } from './services/connection.service';
14
+ import { SwimlaneService } from './services/swimlane.service';
15
+ import { TransformerService } from './services/transformer.service';
16
+ import { PopupService } from './services/popup.service';
17
+ import { DecisionPopupComponent } from './decision-popup/decision-popup.component';
11
18
  import * as i0 from "@angular/core";
12
19
  export class WorkflowDesignerModule {
13
20
  static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: WorkflowDesignerModule, deps: [], target: i0.ɵɵFactoryTarget.NgModule });
@@ -16,12 +23,20 @@ export class WorkflowDesignerModule {
16
23
  DesignerCanvasComponent,
17
24
  SwimlaneDialogComponent,
18
25
  StageNodeComponent,
19
- StageDialogComponent], imports: [CommonModule,
26
+ StageDialogComponent,
27
+ ConditionsPopupComponent,
28
+ DecisionPopupComponent], imports: [CommonModule,
20
29
  VerbenDialogueModule,
21
30
  FormsModule,
22
31
  ReactiveFormsModule,
23
32
  VerbenPopUpModule], exports: [WorkflowDesignerComponent] });
24
- static ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: WorkflowDesignerModule, imports: [CommonModule,
33
+ static ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: WorkflowDesignerModule, providers: [
34
+ NodeManagementService,
35
+ ConnectionService,
36
+ SwimlaneService,
37
+ TransformerService,
38
+ PopupService,
39
+ ], imports: [CommonModule,
25
40
  VerbenDialogueModule,
26
41
  FormsModule,
27
42
  ReactiveFormsModule,
@@ -37,6 +52,8 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
37
52
  SwimlaneDialogComponent,
38
53
  StageNodeComponent,
39
54
  StageDialogComponent,
55
+ ConditionsPopupComponent,
56
+ DecisionPopupComponent,
40
57
  ],
41
58
  imports: [
42
59
  CommonModule,
@@ -45,7 +62,14 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
45
62
  ReactiveFormsModule,
46
63
  VerbenPopUpModule,
47
64
  ],
65
+ providers: [
66
+ NodeManagementService,
67
+ ConnectionService,
68
+ SwimlaneService,
69
+ TransformerService,
70
+ PopupService,
71
+ ],
48
72
  exports: [WorkflowDesignerComponent],
49
73
  }]
50
74
  }] });
51
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoid29ya2Zsb3ctZGVzaWduZXIubW9kdWxlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvdmVyYmVuLXdvcmtmbG93LXVpL3NyYy9saWIvY29tcG9uZW50cy93b3JrZmxvdy1kZXNpZ25lci93b3JrZmxvdy1kZXNpZ25lci5tb2R1bGUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFFBQVEsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUN6QyxPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0saUJBQWlCLENBQUM7QUFDL0MsT0FBTyxFQUFFLHlCQUF5QixFQUFFLE1BQU0sK0JBQStCLENBQUM7QUFDMUUsT0FBTyxFQUFFLHdCQUF3QixFQUFFLE1BQU0sK0NBQStDLENBQUM7QUFDekYsT0FBTyxFQUFFLHVCQUF1QixFQUFFLE1BQU0sNkNBQTZDLENBQUM7QUFDdEYsT0FBTyxFQUFFLHVCQUF1QixFQUFFLE1BQU0sNkNBQTZDLENBQUM7QUFDdEYsT0FBTyxFQUFFLG9CQUFvQixFQUFFLGlCQUFpQixFQUFFLE1BQU0sY0FBYyxDQUFDO0FBQ3ZFLE9BQU8sRUFBRSxXQUFXLEVBQUUsbUJBQW1CLEVBQUUsTUFBTSxnQkFBZ0IsQ0FBQztBQUNsRSxPQUFPLEVBQUUsa0JBQWtCLEVBQUUsTUFBTSxtQ0FBbUMsQ0FBQztBQUN2RSxPQUFPLEVBQUUsb0JBQW9CLEVBQUUsTUFBTSx1Q0FBdUMsQ0FBQzs7QUFvQjdFLE1BQU0sT0FBTyxzQkFBc0I7d0dBQXRCLHNCQUFzQjt5R0FBdEIsc0JBQXNCLGlCQWhCL0IseUJBQXlCO1lBQ3pCLHdCQUF3QjtZQUN4Qix1QkFBdUI7WUFDdkIsdUJBQXVCO1lBQ3ZCLGtCQUFrQjtZQUNsQixvQkFBb0IsYUFHcEIsWUFBWTtZQUNaLG9CQUFvQjtZQUNwQixXQUFXO1lBQ1gsbUJBQW1CO1lBQ25CLGlCQUFpQixhQUVULHlCQUF5Qjt5R0FFeEIsc0JBQXNCLFlBUi9CLFlBQVk7WUFDWixvQkFBb0I7WUFDcEIsV0FBVztZQUNYLG1CQUFtQjtZQUNuQixpQkFBaUI7OzRGQUlSLHNCQUFzQjtrQkFsQmxDLFFBQVE7bUJBQUM7b0JBQ1IsWUFBWSxFQUFFO3dCQUNaLHlCQUF5Qjt3QkFDekIsd0JBQXdCO3dCQUN4Qix1QkFBdUI7d0JBQ3ZCLHVCQUF1Qjt3QkFDdkIsa0JBQWtCO3dCQUNsQixvQkFBb0I7cUJBQ3JCO29CQUNELE9BQU8sRUFBRTt3QkFDUCxZQUFZO3dCQUNaLG9CQUFvQjt3QkFDcEIsV0FBVzt3QkFDWCxtQkFBbUI7d0JBQ25CLGlCQUFpQjtxQkFDbEI7b0JBQ0QsT0FBTyxFQUFFLENBQUMseUJBQXlCLENBQUM7aUJBQ3JDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgTmdNb2R1bGUgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IENvbW1vbk1vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvbW1vbic7XG5pbXBvcnQgeyBXb3JrZmxvd0Rlc2lnbmVyQ29tcG9uZW50IH0gZnJvbSAnLi93b3JrZmxvdy1kZXNpZ25lci5jb21wb25lbnQnO1xuaW1wb3J0IHsgRGVzaWduZXJUb29sYmFyQ29tcG9uZW50IH0gZnJvbSAnLi9kZXNpZ25lci10b29sYmFyL2Rlc2lnbmVyLXRvb2xiYXIuY29tcG9uZW50JztcbmltcG9ydCB7IERlc2lnbmVyQ2FudmFzQ29tcG9uZW50IH0gZnJvbSAnLi9kZXNpZ25lci1jYW52YXMvZGVzaWduZXItY2FudmFzLmNvbXBvbmVudCc7XG5pbXBvcnQgeyBTd2ltbGFuZURpYWxvZ0NvbXBvbmVudCB9IGZyb20gJy4vc3dpbWxhbmUtZGlhbG9nL3N3aW1sYW5lLWRpYWxvZy5jb21wb25lbnQnO1xuaW1wb3J0IHsgVmVyYmVuRGlhbG9ndWVNb2R1bGUsIFZlcmJlblBvcFVwTW9kdWxlIH0gZnJvbSAndmVyYmVuLW5nLXVpJztcbmltcG9ydCB7IEZvcm1zTW9kdWxlLCBSZWFjdGl2ZUZvcm1zTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvZm9ybXMnO1xuaW1wb3J0IHsgU3RhZ2VOb2RlQ29tcG9uZW50IH0gZnJvbSAnLi9zdGFnZS1ub2RlL3N0YWdlLW5vZGUuY29tcG9uZW50JztcbmltcG9ydCB7IFN0YWdlRGlhbG9nQ29tcG9uZW50IH0gZnJvbSAnLi9zdGFnZS1kaWFsb2cvc3RhZ2UtZGlhbG9nLmNvbXBvbmVudCc7XG5cbkBOZ01vZHVsZSh7XG4gIGRlY2xhcmF0aW9uczogW1xuICAgIFdvcmtmbG93RGVzaWduZXJDb21wb25lbnQsXG4gICAgRGVzaWduZXJUb29sYmFyQ29tcG9uZW50LFxuICAgIERlc2lnbmVyQ2FudmFzQ29tcG9uZW50LFxuICAgIFN3aW1sYW5lRGlhbG9nQ29tcG9uZW50LFxuICAgIFN0YWdlTm9kZUNvbXBvbmVudCxcbiAgICBTdGFnZURpYWxvZ0NvbXBvbmVudCxcbiAgXSxcbiAgaW1wb3J0czogW1xuICAgIENvbW1vbk1vZHVsZSxcbiAgICBWZXJiZW5EaWFsb2d1ZU1vZHVsZSxcbiAgICBGb3Jtc01vZHVsZSxcbiAgICBSZWFjdGl2ZUZvcm1zTW9kdWxlLFxuICAgIFZlcmJlblBvcFVwTW9kdWxlLFxuICBdLFxuICBleHBvcnRzOiBbV29ya2Zsb3dEZXNpZ25lckNvbXBvbmVudF0sXG59KVxuZXhwb3J0IGNsYXNzIFdvcmtmbG93RGVzaWduZXJNb2R1bGUge31cbiJdfQ==
75
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoid29ya2Zsb3ctZGVzaWduZXIubW9kdWxlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvdmVyYmVuLXdvcmtmbG93LXVpL3NyYy9saWIvY29tcG9uZW50cy93b3JrZmxvdy1kZXNpZ25lci93b3JrZmxvdy1kZXNpZ25lci5tb2R1bGUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFFBQVEsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUN6QyxPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0saUJBQWlCLENBQUM7QUFDL0MsT0FBTyxFQUFFLHlCQUF5QixFQUFFLE1BQU0sK0JBQStCLENBQUM7QUFDMUUsT0FBTyxFQUFFLHdCQUF3QixFQUFFLE1BQU0sK0NBQStDLENBQUM7QUFDekYsT0FBTyxFQUFFLHVCQUF1QixFQUFFLE1BQU0sNkNBQTZDLENBQUM7QUFDdEYsT0FBTyxFQUFFLHVCQUF1QixFQUFFLE1BQU0sNkNBQTZDLENBQUM7QUFDdEYsT0FBTyxFQUFFLG9CQUFvQixFQUFFLGlCQUFpQixFQUFFLE1BQU0sY0FBYyxDQUFDO0FBQ3ZFLE9BQU8sRUFBRSxXQUFXLEVBQUUsbUJBQW1CLEVBQUUsTUFBTSxnQkFBZ0IsQ0FBQztBQUNsRSxPQUFPLEVBQUUsa0JBQWtCLEVBQUUsTUFBTSxtQ0FBbUMsQ0FBQztBQUN2RSxPQUFPLEVBQUUsb0JBQW9CLEVBQUUsTUFBTSx1Q0FBdUMsQ0FBQztBQUM3RSxPQUFPLEVBQUUsd0JBQXdCLEVBQUUsTUFBTSwrQ0FBK0MsQ0FBQztBQUN6RixPQUFPLEVBQUUscUJBQXFCLEVBQUUsTUFBTSxvQ0FBb0MsQ0FBQztBQUMzRSxPQUFPLEVBQUUsaUJBQWlCLEVBQUUsTUFBTSwrQkFBK0IsQ0FBQztBQUNsRSxPQUFPLEVBQUUsZUFBZSxFQUFFLE1BQU0sNkJBQTZCLENBQUM7QUFDOUQsT0FBTyxFQUFFLGtCQUFrQixFQUFFLE1BQU0sZ0NBQWdDLENBQUM7QUFDcEUsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLDBCQUEwQixDQUFDO0FBQ3hELE9BQU8sRUFBRSxzQkFBc0IsRUFBRSxNQUFNLDJDQUEyQyxDQUFDOztBQTZCbkYsTUFBTSxPQUFPLHNCQUFzQjt3R0FBdEIsc0JBQXNCO3lHQUF0QixzQkFBc0IsaUJBekIvQix5QkFBeUI7WUFDekIsd0JBQXdCO1lBQ3hCLHVCQUF1QjtZQUN2Qix1QkFBdUI7WUFDdkIsa0JBQWtCO1lBQ2xCLG9CQUFvQjtZQUNwQix3QkFBd0I7WUFDeEIsc0JBQXNCLGFBR3RCLFlBQVk7WUFDWixvQkFBb0I7WUFDcEIsV0FBVztZQUNYLG1CQUFtQjtZQUNuQixpQkFBaUIsYUFTVCx5QkFBeUI7eUdBRXhCLHNCQUFzQixhQVR0QjtZQUNULHFCQUFxQjtZQUNyQixpQkFBaUI7WUFDakIsZUFBZTtZQUNmLGtCQUFrQjtZQUNsQixZQUFZO1NBQ2IsWUFaQyxZQUFZO1lBQ1osb0JBQW9CO1lBQ3BCLFdBQVc7WUFDWCxtQkFBbUI7WUFDbkIsaUJBQWlCOzs0RkFXUixzQkFBc0I7a0JBM0JsQyxRQUFRO21CQUFDO29CQUNSLFlBQVksRUFBRTt3QkFDWix5QkFBeUI7d0JBQ3pCLHdCQUF3Qjt3QkFDeEIsdUJBQXVCO3dCQUN2Qix1QkFBdUI7d0JBQ3ZCLGtCQUFrQjt3QkFDbEIsb0JBQW9CO3dCQUNwQix3QkFBd0I7d0JBQ3hCLHNCQUFzQjtxQkFDdkI7b0JBQ0QsT0FBTyxFQUFFO3dCQUNQLFlBQVk7d0JBQ1osb0JBQW9CO3dCQUNwQixXQUFXO3dCQUNYLG1CQUFtQjt3QkFDbkIsaUJBQWlCO3FCQUNsQjtvQkFDRCxTQUFTLEVBQUU7d0JBQ1QscUJBQXFCO3dCQUNyQixpQkFBaUI7d0JBQ2pCLGVBQWU7d0JBQ2Ysa0JBQWtCO3dCQUNsQixZQUFZO3FCQUNiO29CQUNELE9BQU8sRUFBRSxDQUFDLHlCQUF5QixDQUFDO2lCQUNyQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IE5nTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBDb21tb25Nb2R1bGUgfSBmcm9tICdAYW5ndWxhci9jb21tb24nO1xuaW1wb3J0IHsgV29ya2Zsb3dEZXNpZ25lckNvbXBvbmVudCB9IGZyb20gJy4vd29ya2Zsb3ctZGVzaWduZXIuY29tcG9uZW50JztcbmltcG9ydCB7IERlc2lnbmVyVG9vbGJhckNvbXBvbmVudCB9IGZyb20gJy4vZGVzaWduZXItdG9vbGJhci9kZXNpZ25lci10b29sYmFyLmNvbXBvbmVudCc7XG5pbXBvcnQgeyBEZXNpZ25lckNhbnZhc0NvbXBvbmVudCB9IGZyb20gJy4vZGVzaWduZXItY2FudmFzL2Rlc2lnbmVyLWNhbnZhcy5jb21wb25lbnQnO1xuaW1wb3J0IHsgU3dpbWxhbmVEaWFsb2dDb21wb25lbnQgfSBmcm9tICcuL3N3aW1sYW5lLWRpYWxvZy9zd2ltbGFuZS1kaWFsb2cuY29tcG9uZW50JztcbmltcG9ydCB7IFZlcmJlbkRpYWxvZ3VlTW9kdWxlLCBWZXJiZW5Qb3BVcE1vZHVsZSB9IGZyb20gJ3ZlcmJlbi1uZy11aSc7XG5pbXBvcnQgeyBGb3Jtc01vZHVsZSwgUmVhY3RpdmVGb3Jtc01vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL2Zvcm1zJztcbmltcG9ydCB7IFN0YWdlTm9kZUNvbXBvbmVudCB9IGZyb20gJy4vc3RhZ2Utbm9kZS9zdGFnZS1ub2RlLmNvbXBvbmVudCc7XG5pbXBvcnQgeyBTdGFnZURpYWxvZ0NvbXBvbmVudCB9IGZyb20gJy4vc3RhZ2UtZGlhbG9nL3N0YWdlLWRpYWxvZy5jb21wb25lbnQnO1xuaW1wb3J0IHsgQ29uZGl0aW9uc1BvcHVwQ29tcG9uZW50IH0gZnJvbSAnLi9jb25kaXRpb25zLXBvcHVwL2NvbmRpdGlvbnMtcG9wdXAuY29tcG9uZW50JztcbmltcG9ydCB7IE5vZGVNYW5hZ2VtZW50U2VydmljZSB9IGZyb20gJy4vc2VydmljZXMvbm9kZS1tYW5hZ2VtZW50LnNlcnZpY2UnO1xuaW1wb3J0IHsgQ29ubmVjdGlvblNlcnZpY2UgfSBmcm9tICcuL3NlcnZpY2VzL2Nvbm5lY3Rpb24uc2VydmljZSc7XG5pbXBvcnQgeyBTd2ltbGFuZVNlcnZpY2UgfSBmcm9tICcuL3NlcnZpY2VzL3N3aW1sYW5lLnNlcnZpY2UnO1xuaW1wb3J0IHsgVHJhbnNmb3JtZXJTZXJ2aWNlIH0gZnJvbSAnLi9zZXJ2aWNlcy90cmFuc2Zvcm1lci5zZXJ2aWNlJztcbmltcG9ydCB7IFBvcHVwU2VydmljZSB9IGZyb20gJy4vc2VydmljZXMvcG9wdXAuc2VydmljZSc7XG5pbXBvcnQgeyBEZWNpc2lvblBvcHVwQ29tcG9uZW50IH0gZnJvbSAnLi9kZWNpc2lvbi1wb3B1cC9kZWNpc2lvbi1wb3B1cC5jb21wb25lbnQnO1xuXG5ATmdNb2R1bGUoe1xuICBkZWNsYXJhdGlvbnM6IFtcbiAgICBXb3JrZmxvd0Rlc2lnbmVyQ29tcG9uZW50LFxuICAgIERlc2lnbmVyVG9vbGJhckNvbXBvbmVudCxcbiAgICBEZXNpZ25lckNhbnZhc0NvbXBvbmVudCxcbiAgICBTd2ltbGFuZURpYWxvZ0NvbXBvbmVudCxcbiAgICBTdGFnZU5vZGVDb21wb25lbnQsXG4gICAgU3RhZ2VEaWFsb2dDb21wb25lbnQsXG4gICAgQ29uZGl0aW9uc1BvcHVwQ29tcG9uZW50LFxuICAgIERlY2lzaW9uUG9wdXBDb21wb25lbnQsXG4gIF0sXG4gIGltcG9ydHM6IFtcbiAgICBDb21tb25Nb2R1bGUsXG4gICAgVmVyYmVuRGlhbG9ndWVNb2R1bGUsXG4gICAgRm9ybXNNb2R1bGUsXG4gICAgUmVhY3RpdmVGb3Jtc01vZHVsZSxcbiAgICBWZXJiZW5Qb3BVcE1vZHVsZSxcbiAgXSxcbiAgcHJvdmlkZXJzOiBbXG4gICAgTm9kZU1hbmFnZW1lbnRTZXJ2aWNlLFxuICAgIENvbm5lY3Rpb25TZXJ2aWNlLFxuICAgIFN3aW1sYW5lU2VydmljZSxcbiAgICBUcmFuc2Zvcm1lclNlcnZpY2UsXG4gICAgUG9wdXBTZXJ2aWNlLFxuICBdLFxuICBleHBvcnRzOiBbV29ya2Zsb3dEZXNpZ25lckNvbXBvbmVudF0sXG59KVxuZXhwb3J0IGNsYXNzIFdvcmtmbG93RGVzaWduZXJNb2R1bGUge31cbiJdfQ==