File size: 15,216 Bytes
615e9f1
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
import xml.etree.ElementTree as ET
from utils import class_dict

def rescale(scale, boxes):
    for i in range(len(boxes)):
                boxes[i] = [boxes[i][0]*scale,
                            boxes[i][1]*scale,
                            boxes[i][2]*scale,
                            boxes[i][3]*scale]
    return boxes

def create_BPMN_id(data):
    enums = {
        'end_event': 1,
        'start_event': 1,
        'task': 1,
        'sequenceFlow': 1,
        'messageFlow': 1,
        'message_event': 1,
        'exclusiveGateway': 1,
        'parallelGateway': 1,
        'dataAssociation': 1,
        'pool': 1,
        'dataObject': 1,
        'timerEvent': 1
    }

    BPMN_name = [class_dict[label] for label in data['labels']]

    for idx, Bpmn_id in enumerate(BPMN_name):
        if Bpmn_id == 'event':
            if data['links'][idx][0] is not None and data['links'][idx][1] is None:
                key = 'end_event'
            elif data['links'][idx][0] is None and data['links'][idx][1] is not None:
                key = 'start_event'
        else:
            key = {
                'task': 'task',
                'dataObject': 'dataObject',
                'sequenceFlow': 'sequenceFlow',
                'messageFlow': 'messageFlow',
                'messageEvent': 'message_event',
                'exclusiveGateway': 'exclusiveGateway',
                'parallelGateway': 'parallelGateway',
                'dataAssociation': 'dataAssociation',
                'pool': 'pool',
                'timerEvent': 'timerEvent'
            }.get(Bpmn_id, None)

        if key:
            data['BPMN_id'][idx] = f'{key}_{enums[key]}'
            enums[key] += 1

    return data



def add_diagram_elements(parent, element_id, x, y, width, height):
    """Utility to add BPMN diagram notation for elements."""
    shape = ET.SubElement(parent, 'bpmndi:BPMNShape', attrib={
        'bpmnElement': element_id,
        'id': element_id + '_di'
    })
    bounds = ET.SubElement(shape, 'dc:Bounds', attrib={
        'x': str(x),
        'y': str(y),
        'width': str(width),
        'height': str(height)
    })

def add_diagram_edge(parent, element_id, waypoints):
    """Utility to add BPMN diagram notation for sequence flows."""
    edge = ET.SubElement(parent, 'bpmndi:BPMNEdge', attrib={
        'bpmnElement': element_id,
        'id': element_id + '_di'
    })
    for x, y in waypoints:
        ET.SubElement(edge, 'di:waypoint', attrib={
            'x': str(x),
            'y': str(y)
        })


def check_status(link, keep_elements):
    if link[0] in keep_elements and link[1] in keep_elements:
        return 'middle'
    elif link[0] is None and link[1] in keep_elements:
        return 'start'
    elif link[0] in keep_elements and link[1] is None:
        return 'end'
    else:
        return 'middle'
    
def check_data_association(i, links, labels, keep_elements):
    for j, (k,l) in enumerate(links):
        if labels[j] == 14:
            if k==i:
                return 'output',j
            elif l==i:
                return 'input',j
            
    return 'no association', None

def create_data_Association(bpmn,data,size,element_id,source_id,target_id):
    waypoints = calculate_waypoints(data, size, source_id, target_id)
    add_diagram_edge(bpmn, element_id, waypoints)
        
# Function to dynamically create and layout BPMN elements
def create_bpmn_object(process, bpmnplane, text_mapping, definitions, size, data, keep_elements):
    elements = data['BPMN_id']
    positions = data['boxes']
    links = data['links']

    for i in keep_elements:
        element_id = elements[i]
        if element_id is None:
            continue
        
        element_type = element_id.split('_')[0]
        x, y = positions[i][:2]

        # Start Event
        if element_type == 'start':
            element = ET.SubElement(process, 'bpmn:startEvent', id=element_id, name=text_mapping[element_id])
            add_diagram_elements(bpmnplane, element_id, x, y, size['start'][0], size['start'][1])

        # Task
        elif element_type == 'task':
            element = ET.SubElement(process, 'bpmn:task', id=element_id, name=text_mapping[element_id])
            status, dataAssociation_idx = check_data_association(i, data['links'], data['labels'], keep_elements)

            # Handle Data Input Association
            if status == 'input':
                dataObject_idx = links[dataAssociation_idx][0]
                dataObject_name = elements[dataObject_idx]
                dataObject_ref = f'DataObjectReference_{dataObject_name.split("_")[1]}'
                sub_element = ET.SubElement(element, 'bpmn:dataInputAssociation', id=f'dataInputAssociation_{dataObject_ref.split("_")[1]}')
                ET.SubElement(sub_element, 'bpmn:sourceRef').text = dataObject_ref
                create_data_Association(bpmnplane, data, size, sub_element.attrib['id'], dataObject_name, element_id)

            # Handle Data Output Association
            elif status == 'output':
                dataObject_idx = links[dataAssociation_idx][1]
                dataObject_name = elements[dataObject_idx]
                dataObject_ref = f'DataObjectReference_{dataObject_name.split("_")[1]}'
                sub_element = ET.SubElement(element, 'bpmn:dataOutputAssociation', id=f'dataOutputAssociation_{dataObject_ref.split("_")[1]}')
                ET.SubElement(sub_element, 'bpmn:targetRef').text = dataObject_ref
                create_data_Association(bpmnplane, data, size, sub_element.attrib['id'], element_id, dataObject_name)

            add_diagram_elements(bpmnplane, element_id, x, y, size['task'][0], size['task'][1])

        # Message Events (Start, Intermediate, End)
        elif element_type == 'message':
            status = check_status(links[i], keep_elements)
            if status == 'start':
                element = ET.SubElement(process, 'bpmn:startEvent', id=element_id, name=text_mapping[element_id])
            elif status == 'middle':
                element = ET.SubElement(process, 'bpmn:intermediateCatchEvent', id=element_id, name=text_mapping[element_id])
            elif status == 'end':
                element = ET.SubElement(process, 'bpmn:endEvent', id=element_id, name=text_mapping[element_id])
            ET.SubElement(element, 'bpmn:messageEventDefinition', id=f'MessageEventDefinition_{i+1}')
            add_diagram_elements(bpmnplane, element_id, x, y, size['message'][0], size['message'][1])

        # End Event
        elif element_type == 'end':
            element = ET.SubElement(process, 'bpmn:endEvent', id=element_id, name=text_mapping[element_id])
            add_diagram_elements(bpmnplane, element_id, x, y, size['end'][0], size['end'][1])

        # Gateways (Exclusive, Parallel)
        elif element_type in ['exclusiveGateway', 'parallelGateway']:
            gateway_type = 'exclusiveGateway' if element_type == 'exclusiveGateway' else 'parallelGateway'
            element = ET.SubElement(process, f'bpmn:{gateway_type}', id=element_id)
            add_diagram_elements(bpmnplane, element_id, x, y, size[element_type][0], size[element_type][1])

        # Data Object
        elif element_type == 'dataObject':
            dataObject_idx = element_id.split('_')[1]
            dataObject_ref = f'DataObjectReference_{dataObject_idx}'
            element = ET.SubElement(process, 'bpmn:dataObjectReference', id=dataObject_ref, dataObjectRef=element_id, name=text_mapping[element_id])
            ET.SubElement(process, 'bpmn:dataObject', id=element_id)
            add_diagram_elements(bpmnplane, dataObject_ref, x, y, size['dataObject'][0], size['dataObject'][1])

        # Timer Event
        elif element_type == 'timerEvent':
            element = ET.SubElement(process, 'bpmn:intermediateCatchEvent', id=element_id, name=text_mapping[element_id])
            ET.SubElement(element, 'bpmn:timerEventDefinition', id=f'TimerEventDefinition_{i+1}')
            add_diagram_elements(bpmnplane, element_id, x, y, size['timerEvent'][0], size['timerEvent'][1])



# Calculate simple waypoints between two elements (this function assumes direct horizontal links for simplicity)
def calculate_waypoints(data, size, source_id, target_id):
    source_idx = data['BPMN_id'].index(source_id)
    target_idx = data['BPMN_id'].index(target_id)
    name_source = source_id.split('_')[0]
    name_target = target_id.split('_')[0]

    #Get the position of the source and target
    source_x, source_y = data['boxes'][source_idx][:2]
    target_x, target_y = data['boxes'][target_idx][:2]

    # Calculate relative position between source and target from their centers
    relative_x = (target_x+size[name_target][0])/2 - (source_x+size[name_source][0])/2
    relative_y = (target_y+size[name_target][1])/2 - (source_y+size[name_source][1])/2

    # Get the size of the elements
    size_x_source = size[name_source][0]
    size_y_source = size[name_source][1]
    size_x_target = size[name_target][0]
    size_y_target = size[name_target][1]

    #if it going to right
    if relative_x >= size[name_source][0]:
        source_x += size_x_source
        source_y += size_y_source / 2
        target_x  = target_x
        target_y += size_y_target / 2
        #if the source is going up
        if relative_y < -size[name_source][1]:
            source_x -= size_x_source / 2
            source_y -= size_y_source / 2
        #if the source is going down
        elif relative_y > size[name_source][1]:
            source_x -= size_x_source / 2
            source_y += size_y_source / 2
    #if it going to left
    elif relative_x < -size[name_source][0]:
        source_x = source_x
        source_y += size_y_source / 2
        target_x += size_x_target
        target_y += size_y_target / 2
        #if the source is going up
        if relative_y < -size[name_source][1]:
            source_x += size_x_source / 2
            source_y -= size_y_source / 2
        #if the source is going down
        elif relative_y > size[name_source][1]:
            source_x += size_x_source / 2
            source_y += size_y_source / 2    
    #if it going up and down
    elif -size[name_source][0] < relative_x < size[name_source][0]:
        source_x += size_x_source / 2
        target_x += size_x_target / 2
        #if it's going down
        if relative_y >= size[name_source][1]/2:
            source_y += size_y_source
        #if it's going up
        elif relative_y < -size[name_source][1]/2:
            source_y = source_y
            target_y += size_y_target
        else:
            if relative_x >= 0:
                source_x += size_x_source/2
                source_y += size_y_source/2
                target_x -= size_x_target/2
                target_y += size_y_target/2
            else:
                source_x -= size_x_source/2
                source_y += size_y_source/2
                target_x += size_x_target/2
                target_y += size_y_target/2

    return [(source_x, source_y), (target_x, target_y)]
   

def calculate_pool_bounds(data, keep_elements, size):
    min_x = min_y = float('10000')
    max_x = max_y = float('0')
    
    for i in keep_elements:
        if i >= len(data['BPMN_id']):
            print("Problem with the index")
            continue
        element = data['BPMN_id'][i]
        if element is None or data['labels'][i] == 13 or data['labels'][i] == 14 or data['labels'][i] == 15 or data['labels'][i] == 7 or data['labels'][i] == 15: 
            continue
        
        element_type = element.split('_')[0]
        x, y = data['boxes'][i][:2]
        element_width, element_height = size[element_type]
        
        min_x = min(min_x, x)
        min_y = min(min_y, y)
        max_x = max(max_x, x + element_width)
        max_y = max(max_y, y + element_height)
    
    return min_x, min_y, max_x, max_y

    
def calculate_pool_waypoints(idx, data, size, source_idx, target_idx, source_element, target_element):
    # Get the bounding boxes of the source and target elements
    source_box = data['boxes'][source_idx]
    target_box = data['boxes'][target_idx]

    # Get the midpoints of the source element
    source_mid_x = (source_box[0] + source_box[2]) / 2
    source_mid_y = (source_box[1] + source_box[3]) / 2

    # Check if the connection involves a pool
    if source_element == 'pool':
        pool_box = source_box
        element_box = (target_box[0], target_box[1], target_box[0]+size[target_element][0], target_box[1]+size[target_element][1])
        element_mid_x = (element_box[0] + element_box[2]) / 2
        element_mid_y = (element_box[1] + element_box[3]) / 2
        # Connect the pool's bottom or top side to the target element's top or bottom center
        if pool_box[3] < element_box[1]:  # Pool is above the target element
            waypoints = [(element_mid_x, pool_box[3]-50), (element_mid_x, element_box[1])]
        else:  # Pool is below the target element
            waypoints = [(element_mid_x, element_box[3]), (element_mid_x, pool_box[1]-50)]
    else:
        pool_box = target_box
        element_box = (source_box[0], source_box[1], source_box[0]+size[source_element][0], source_box[1]+size[source_element][1])
        element_mid_x = (element_box[0] + element_box[2]) / 2
        element_mid_y = (element_box[1] + element_box[3]) / 2

        # Connect the element's bottom or top center to the pool's top or bottom side
        if pool_box[3] < element_box[1]:  # Pool is above the target element
            waypoints = [(element_mid_x, element_box[1]), (element_mid_x, pool_box[3]-50)]
        else:  # Pool is below the target element
            waypoints = [(element_mid_x, element_box[3]), (element_mid_x, pool_box[1]-50)]

    return waypoints



def create_flow_element(bpmn, text_mapping, idx, size, data, parent, message=False):
    source_idx, target_idx = data['links'][idx]
    source_id, target_id = data['BPMN_id'][source_idx], data['BPMN_id'][target_idx]
    if message:
        element_id = f'messageflow_{source_id}_{target_id}'
    else:
        element_id = f'sequenceflow_{source_id}_{target_id}'

    if source_id.split('_')[0] == 'pool' or target_id.split('_')[0] == 'pool':
        waypoints = calculate_pool_waypoints(idx, data, size, source_idx, target_idx, source_id.split('_')[0], target_id.split('_')[0])
        #waypoints = data['best_points'][idx]
        if source_id.split('_')[0] == 'pool':
            source_id = f"participant_{source_id.split('_')[1]}"
        if target_id.split('_')[0] == 'pool':
            target_id = f"participant_{target_id.split('_')[1]}"
    else:
        waypoints = calculate_waypoints(data, size, source_id, target_id)
        #waypoints = data['best_points'][idx]

    #waypoints = data['best_points'][idx]
    if message:
        element = ET.SubElement(parent, 'bpmn:messageFlow', id=element_id, sourceRef=source_id, targetRef=target_id, name=text_mapping[data['BPMN_id'][idx]])
    else:
        element = ET.SubElement(parent, 'bpmn:sequenceFlow', id=element_id, sourceRef=source_id, targetRef=target_id, name=text_mapping[data['BPMN_id'][idx]])
    add_diagram_edge(bpmn, element_id, waypoints)