Spaces:
Sleeping
Sleeping
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) | |