diff --git a/packages/bpmn-editor/src/diagram/nodes/NodeIcons.tsx b/packages/bpmn-editor/src/diagram/nodes/NodeIcons.tsx
index db6772074bc..d32bf47c998 100644
--- a/packages/bpmn-editor/src/diagram/nodes/NodeIcons.tsx
+++ b/packages/bpmn-editor/src/diagram/nodes/NodeIcons.tsx
@@ -52,11 +52,11 @@ export function NodeIcon({ nodeType }: { nodeType: BpmnNodeType }) {
export function EventDefitnitionIcon({
variant,
- fill,
+ filled,
stroke,
}: {
variant?: EventVariant;
- fill: boolean;
+ filled: boolean;
stroke: string;
}) {
const cx = nodeSvgProps.x + nodeSvgProps.width / 2;
@@ -69,14 +69,14 @@ export function EventDefitnitionIcon({
);
diff --git a/packages/bpmn-editor/src/diagram/nodes/NodeSvgs.tsx b/packages/bpmn-editor/src/diagram/nodes/NodeSvgs.tsx
index ef94d23379f..253d9735a81 100644
--- a/packages/bpmn-editor/src/diagram/nodes/NodeSvgs.tsx
+++ b/packages/bpmn-editor/src/diagram/nodes/NodeSvgs.tsx
@@ -159,14 +159,15 @@ export function StartEventNodeSvg(__props: NodeSvgProps & { variant: EventVarian
/>
>
);
@@ -185,8 +186,8 @@ export function IntermediateCatchEventNodeSvg(
const { rimWidth, variant, ...props } = { ..._props };
- const outerCirculeRadius = width / 2;
- const innerCircleRadius = outerCirculeRadius - (rimWidth ?? 5);
+ const outerCircleRadius = width / 2;
+ const innerCircleRadius = outerCircleRadius - (rimWidth ?? 5);
const cx = x + width / 2;
const cy = y + height / 2;
@@ -202,7 +203,7 @@ export function IntermediateCatchEventNodeSvg(
fill={NODE_COLORS.intermediateCatchEvent.background}
stroke={NODE_COLORS.intermediateCatchEvent.foreground}
strokeLinejoin={"round"}
- r={outerCirculeRadius}
+ r={outerCircleRadius}
{...props}
/>
>
);
@@ -245,8 +247,8 @@ export function IntermediateThrowEventNodeSvg(
const { rimWidth, variant, ...props } = { ..._props };
- const outerCirculeRadius = width / 2;
- const innerCircleRadius = outerCirculeRadius - (rimWidth ?? 5);
+ const outerCircleRadius = width / 2;
+ const innerCircleRadius = outerCircleRadius - (rimWidth ?? 5);
const cx = x + width / 2;
const cy = y + height / 2;
@@ -262,7 +264,7 @@ export function IntermediateThrowEventNodeSvg(
fill={NODE_COLORS.intermediateThrowEvent.background}
stroke={NODE_COLORS.intermediateThrowEvent.foreground}
strokeLinejoin={"round"}
- r={outerCirculeRadius}
+ r={outerCircleRadius}
{...props}
/>
>
);
@@ -324,14 +327,15 @@ export function EndEventNodeSvg(__props: NodeSvgProps & { variant: EventVariant
/>
>
);
@@ -749,8 +753,9 @@ export function EventVariantSymbolSvg({
x,
y,
innerCircleRadius,
- outerCirculeRadius,
+ outerCircleRadius,
fill,
+ filled,
}: {
variant: EventVariant | "none";
stroke: string;
@@ -760,118 +765,81 @@ export function EventVariantSymbolSvg({
x: number;
y: number;
innerCircleRadius: number;
- outerCirculeRadius: number;
- fill: boolean;
+ outerCircleRadius: number;
+ fill?: string;
+ filled: boolean;
}) {
return (
<>
- {/* FIXME: Tiago: tmp icon */}
{variant === "messageEventDefinition" && (
- <>
-
- ✉️
-
- >
+
)}
- {/* FIXME: Tiago: tmp icon */}
{variant === "timerEventDefinition" && (
- <>
-
- 🕑
-
- >
+
)}
- {/* FIXME: Tiago: tmp icon */}
{variant === "errorEventDefinition" && (
- <>
-
- ⚡️
-
- >
+
)}
- {/* FIXME: Tiago: tmp icon */}
{variant === "escalationEventDefinition" && (
- <>
-
- ☝️
-
- >
+
)}
- {/* FIXME: Tiago: tmp icon */}
{variant === "cancelEventDefinition" && (
- <>
-
- ❌
-
- >
+
)}
- {/* FIXME: Tiago: tmp icon */}
{variant === "compensateEventDefinition" && (
- <>
-
- ⏪
-
- >
+
)}
- {/* FIXME: Tiago: tmp icon */}
{variant === "conditionalEventDefinition" && (
- <>
-
- ≣
-
- >
+
)}
- {/* FIXME: Tiago: tmp icon */}
{variant === "linkEventDefinition" && (
- <>
-
- ⇨
-
- >
+
)}
{variant === "signalEventDefinition" && (
)}
@@ -892,7 +860,7 @@ export function EventVariantSymbolSvg({
fill={NODE_COLORS.endEvent.foreground}
stroke={NODE_COLORS.endEvent.foreground}
strokeLinejoin={"round"}
- r={outerCirculeRadius / 2}
+ r={outerCircleRadius / 2}
/>
>
)}
@@ -902,6 +870,429 @@ export function EventVariantSymbolSvg({
);
}
+export function MessageEventSymbolSvg({
+ stroke,
+ cx,
+ cy,
+ innerCircleRadius,
+ fill,
+ filled,
+}: {
+ stroke: string;
+ cx: number;
+ cy: number;
+ innerCircleRadius: number;
+ fill: string;
+ filled: boolean;
+}) {
+ const bodyWidth = innerCircleRadius * 1.2;
+ const bodyHeight = innerCircleRadius * 0.8;
+
+ const envelopeBody = {
+ topLeft: { x: cx - bodyWidth / 2, y: cy - bodyHeight / 2 },
+ bottomRight: { x: cx + bodyWidth / 2, y: cy + bodyHeight / 2 },
+ };
+
+ const flapHeight = bodyHeight * 0.5;
+ const envelopeFlap = [
+ { x: envelopeBody.topLeft.x, y: envelopeBody.topLeft.y },
+ { x: envelopeBody.bottomRight.x, y: envelopeBody.topLeft.y },
+ { x: cx, y: envelopeBody.topLeft.y + flapHeight },
+ ];
+
+ return (
+ <>
+
+
+ `${point.x},${point.y}`).join(" ")}
+ strokeWidth={1.5}
+ strokeLinejoin={"round"}
+ fill={filled ? stroke : fill}
+ stroke={stroke}
+ />
+
+ {filled && (
+ <>
+
+
+ >
+ )}
+ >
+ );
+}
+
+export function TimerEventSymbolSvg({
+ stroke,
+ cx,
+ cy,
+ innerCircleRadius,
+ outerCircleRadius,
+ filled,
+}: {
+ stroke: string;
+ cx: number;
+ cy: number;
+ innerCircleRadius: number;
+ outerCircleRadius: number;
+ filled: boolean;
+}) {
+ const padding = 1.2 * (outerCircleRadius - innerCircleRadius);
+ const clockRadius = innerCircleRadius - padding;
+
+ const shortHandLength = clockRadius * 0.5;
+ const longHandLength = clockRadius * 0.9;
+
+ const hourHandAngle = Math.PI / 12;
+ const minuteHandAngle = (-5 * Math.PI) / 12;
+
+ const hourHand = {
+ x: cx + shortHandLength * Math.cos(hourHandAngle),
+ y: cy + shortHandLength * Math.sin(hourHandAngle),
+ };
+
+ const minuteHand = {
+ x: cx + longHandLength * Math.cos(minuteHandAngle),
+ y: cy + longHandLength * Math.sin(minuteHandAngle),
+ };
+
+ return (
+ <>
+
+
+
+
+
+
+ {Array.from({ length: 12 }, (_, index) => {
+ const angle = (index / 12) * 2 * Math.PI;
+ const x1 = cx + clockRadius * Math.cos(angle);
+ const y1 = cy + clockRadius * Math.sin(angle);
+ const x2 = cx + (clockRadius - padding * 0.5) * Math.cos(angle);
+ const y2 = cy + (clockRadius - padding * 0.5) * Math.sin(angle);
+
+ return ;
+ })}
+ >
+ );
+}
+
+export function ErrorEventSymbolSvg({
+ stroke,
+ cx,
+ cy,
+ innerCircleRadius,
+ outerCircleRadius,
+ filled,
+}: {
+ stroke: string;
+ cx: number;
+ cy: number;
+ innerCircleRadius: number;
+ outerCircleRadius: number;
+ filled: boolean;
+}) {
+ const padding = 1.5 * (outerCircleRadius - innerCircleRadius);
+ const hx = innerCircleRadius - padding;
+ const hy = innerCircleRadius - padding;
+
+ const scaleFactor = 20;
+ const shiftLeft = 3.5;
+ const shiftUp = 1;
+
+ const points = [
+ { x: cx - shiftLeft, y: cy },
+ { x: cx - hx * 0.037 * scaleFactor - shiftLeft, y: cy + hy * 0.052 * scaleFactor - shiftUp },
+ { x: cx - hx * 0.003 * scaleFactor - shiftLeft, y: cy - hy * 0.05 * scaleFactor - shiftUp },
+ { x: cx + hx * 0.027 * scaleFactor - shiftLeft, y: cy + hy * 0.016 * scaleFactor - shiftUp },
+ { x: cx + hx * 0.058 * scaleFactor - shiftLeft, y: cy - hy * 0.046 * scaleFactor - shiftUp },
+ { x: cx + hx * 0.029 * scaleFactor - shiftLeft, y: cy + hy * 0.066 * scaleFactor - shiftUp },
+ { x: cx - shiftLeft, y: cy },
+ ];
+
+ return (
+ <>
+ `${point.x},${point.y}`).join(" ")}
+ strokeWidth={1.5}
+ strokeLinejoin={"round"}
+ fill={filled ? stroke : "transparent"}
+ stroke={stroke}
+ />
+ >
+ );
+}
+
+export function EscalationEventSymbolSvg({
+ stroke,
+ cx,
+ cy,
+ innerCircleRadius,
+ filled,
+}: {
+ stroke: string;
+ cx: number;
+ cy: number;
+ innerCircleRadius: number;
+ filled: boolean;
+}) {
+ const arrowHeight = innerCircleRadius * 1.2;
+ const arrowBaseWidth = innerCircleRadius * 1;
+
+ const arrow = [
+ { x: cx - arrowBaseWidth / 2, y: cy + arrowHeight / 2 }, // left
+ { x: cx, y: cy - arrowHeight / 2 }, // top center
+ { x: cx + arrowBaseWidth / 2, y: cy + arrowHeight / 2 }, // right
+ { x: cx, y: cy + arrowHeight / 20 }, // mid center
+ ] as const;
+
+ return (
+ <>
+ `${point.x},${point.y}`).join(" ")}
+ strokeWidth={1.5}
+ strokeLinejoin={"round"}
+ fill={filled ? stroke : "transparent"}
+ stroke={stroke}
+ />
+ >
+ );
+}
+
+export function CancelEventSymbolSvg({
+ stroke,
+ cx,
+ cy,
+ innerCircleRadius,
+ filled,
+}: {
+ stroke: string;
+ cx: number;
+ cy: number;
+ innerCircleRadius: number;
+ filled: boolean;
+}) {
+ const farXPoint = 1.3;
+ const closeXPoint = 1.7;
+ const lowYPoint = 9;
+ const highYPoint = 5;
+
+ const cross = [
+ { x: cx - innerCircleRadius / farXPoint, y: cy - innerCircleRadius + lowYPoint }, // upper left point 1
+ { x: cx - innerCircleRadius / closeXPoint, y: cy - innerCircleRadius + highYPoint }, // upper left point 2
+ { x: cx, y: cy - innerCircleRadius / highYPoint }, // upper joiner
+ { x: cx + innerCircleRadius / closeXPoint, y: cy - innerCircleRadius + highYPoint }, // upper right point 2
+ { x: cx + innerCircleRadius / farXPoint, y: cy - innerCircleRadius + lowYPoint }, // upper right point 1
+ { x: cx + innerCircleRadius / highYPoint, y: cy }, // right joiner
+ { x: cx + innerCircleRadius / farXPoint, y: cy + innerCircleRadius - lowYPoint }, // lower right point 2
+ { x: cx + innerCircleRadius / closeXPoint, y: cy + innerCircleRadius - highYPoint }, // lower right point 1
+ { x: cx, y: cy + innerCircleRadius / highYPoint }, // lower joiner
+ { x: cx - innerCircleRadius / closeXPoint, y: cy + innerCircleRadius - highYPoint }, // lower left point 1
+ { x: cx - innerCircleRadius / farXPoint, y: cy + innerCircleRadius - lowYPoint }, // lower left point 2
+ { x: cx - innerCircleRadius / highYPoint, y: cy }, // left joiner
+ ] as const;
+
+ return (
+ <>
+ `${point.x},${point.y}`).join(" ")}
+ strokeWidth={1.5}
+ strokeLinejoin={"round"}
+ fill={filled ? stroke : "transparent"}
+ stroke={stroke}
+ />
+ >
+ );
+}
+
+export function CompensationEventSymbolSvg({
+ stroke,
+ cx,
+ cy,
+ x,
+ y,
+ innerCircleRadius,
+ outerCircleRadius,
+ filled,
+}: {
+ stroke: string;
+ cx: number;
+ cy: number;
+ x: number;
+ y: number;
+ innerCircleRadius: number;
+ outerCircleRadius: number;
+ filled: boolean;
+}) {
+ const padding = 1.5 * (outerCircleRadius - innerCircleRadius);
+
+ const hx = x + innerCircleRadius - padding * 0.6;
+ const hy = y + innerCircleRadius - padding * 0.2;
+
+ const rightOffset = 0.15 * innerCircleRadius;
+
+ const firstTriangle = [
+ { x: cx + hx - rightOffset, y: cy - hy + (outerCircleRadius - innerCircleRadius) },
+ { x: cx - rightOffset, y: cy },
+ { x: cx + hx - rightOffset, y: cy + hy - (outerCircleRadius - innerCircleRadius) },
+ ] as const;
+
+ const secondTriangle = [
+ { x: cx - rightOffset, y: cy - hy + (outerCircleRadius - innerCircleRadius) },
+ { x: cx - hx - rightOffset, y: cy },
+ { x: cx - rightOffset, y: cy + hy - (outerCircleRadius - innerCircleRadius) },
+ ] as const;
+
+ return (
+ <>
+ `${point.x},${point.y}`).join(" ")}
+ strokeWidth={1.5}
+ strokeLinejoin={"round"}
+ fill={filled ? stroke : "transparent"}
+ stroke={stroke}
+ />
+ `${point.x},${point.y}`).join(" ")}
+ strokeWidth={1.5}
+ strokeLinejoin={"round"}
+ fill={filled ? stroke : "transparent"}
+ stroke={stroke}
+ />
+ >
+ );
+}
+
+export function ConditionalEventSymbolSvg({
+ stroke,
+ cx,
+ cy,
+ outerCircleRadius,
+ filled,
+}: {
+ stroke: string;
+ cx: number;
+ cy: number;
+ outerCircleRadius: number;
+ filled: boolean;
+}) {
+ const squareSize = outerCircleRadius * 1.1;
+ const lineSpacing = squareSize / 5;
+ const lineThickness = 2;
+
+ return (
+ <>
+
+
+ {[...Array(4)].map((_, index) => (
+
+ ))}
+ >
+ );
+}
+
+export function LinkEventSymbolSvg({
+ stroke,
+ cx,
+ cy,
+ innerCircleRadius,
+ filled,
+}: {
+ stroke: string;
+ cx: number;
+ cy: number;
+ innerCircleRadius: number;
+ filled: boolean;
+}) {
+ const arrowHeight = innerCircleRadius * 1.2;
+ const arrowBaseWidth = innerCircleRadius * 1;
+ const shiftLeft = 7;
+ const rectangleHeight = 5;
+ const arrowPadding = 2;
+
+ const arrow = [
+ { x: cx - arrowBaseWidth / 2 - shiftLeft, y: cy + arrowHeight / 2 - rectangleHeight }, // bottom left rectangle
+ { x: cx - arrowBaseWidth / 2 - shiftLeft, y: cy - arrowHeight / 2 + rectangleHeight }, // top left rectangle
+ { x: cx + arrowBaseWidth / 2, y: cy - arrowHeight / 2 + rectangleHeight }, // top right rectangle
+ { x: cx + arrowBaseWidth / 2, y: cy - arrowHeight / 2 - arrowPadding }, // upper arrow start
+ { x: cx + arrowBaseWidth / 2 + 8, y: cy }, // arrow point
+ { x: cx + arrowBaseWidth / 2, y: cy + arrowHeight / 2 + arrowPadding }, // lower arrow start
+ { x: cx + arrowBaseWidth / 2, y: cy + arrowHeight / 2 - rectangleHeight }, // bottom right rectangle
+ ] as const;
+
+ return (
+ <>
+ `${point.x},${point.y}`).join(" ")}
+ strokeWidth={1.5}
+ strokeLinejoin={"round"}
+ fill={filled ? stroke : "transparent"}
+ stroke={stroke}
+ />
+ >
+ );
+}
+
export function SignalEventSymbolSvg({
stroke,
strokeWidth,
@@ -910,8 +1301,8 @@ export function SignalEventSymbolSvg({
x,
y,
innerCircleRadius,
- outerCirculeRadius,
- fill,
+ outerCircleRadius,
+ filled,
}: {
stroke: string;
strokeWidth?: number;
@@ -920,10 +1311,10 @@ export function SignalEventSymbolSvg({
x: number;
y: number;
innerCircleRadius: number;
- outerCirculeRadius: number;
- fill: boolean;
+ outerCircleRadius: number;
+ filled: boolean;
}) {
- const padding = 1.5 * (outerCirculeRadius - innerCircleRadius);
+ const padding = 1.5 * (outerCircleRadius - innerCircleRadius);
const hx = x + innerCircleRadius - padding;
const hy = y + innerCircleRadius - padding;
const triangle = [
@@ -935,10 +1326,10 @@ export function SignalEventSymbolSvg({
return (
<>
`${point.x},${point.y}`).join(" ")}
+ strokeWidth={strokeWidth ?? 1.5}
strokeLinejoin={"round"}
- fill={fill ? stroke : "transparent"}
+ fill={filled ? stroke : "transparent"}
stroke={stroke}
/>
>
diff --git a/packages/bpmn-editor/src/diagram/nodes/morphing/useEventNodeMorphingActions.tsx b/packages/bpmn-editor/src/diagram/nodes/morphing/useEventNodeMorphingActions.tsx
index c3f35494286..bf5ba008eb6 100644
--- a/packages/bpmn-editor/src/diagram/nodes/morphing/useEventNodeMorphingActions.tsx
+++ b/packages/bpmn-editor/src/diagram/nodes/morphing/useEventNodeMorphingActions.tsx
@@ -41,7 +41,7 @@ export function useEventNodeMorphingActions(event: Event) {
const bpmnEditorStoreApi = useBpmnEditorStoreApi();
const foregroundColor = NODE_COLORS[event.__$$element].foreground;
- const fill = event.__$$element === "intermediateThrowEvent" || event.__$$element === "endEvent";
+ const filled = event.__$$element === "intermediateThrowEvent" || event.__$$element === "endEvent";
const morphEvent = useCallback(
(eventDefinitionElement: undefined | Unpacked["__$$element"]) => {
@@ -165,63 +165,63 @@ export function useEventNodeMorphingActions(event: Event) {
action: () => morphEvent(undefined),
} as const, // none
{
- icon: ,
+ icon: ,
key: "2",
title: "Message",
id: "messageEventDefinition",
action: () => morphEvent("messageEventDefinition"),
} as const,
{
- icon: ,
+ icon: ,
key: "3",
title: "Timer",
id: "timerEventDefinition",
action: () => morphEvent("timerEventDefinition"),
} as const,
{
- icon: ,
+ icon: ,
key: "4",
title: "Error",
id: "errorEventDefinition",
action: () => morphEvent("errorEventDefinition"),
} as const,
{
- icon: ,
+ icon: ,
key: "5",
title: "Escalation",
id: "escalationEventDefinition",
action: () => morphEvent("escalationEventDefinition"),
} as const,
{
- icon: ,
+ icon: ,
key: "6",
title: "Cancelation",
id: "cancelEventDefinition",
action: () => morphEvent("cancelEventDefinition"),
} as const,
{
- icon: ,
+ icon: ,
key: "7",
title: "Compensation",
id: "compensateEventDefinition",
action: () => morphEvent("compensateEventDefinition"),
} as const,
{
- icon: ,
+ icon: ,
key: "8",
title: "Conditional",
id: "conditionalEventDefinition",
action: () => morphEvent("conditionalEventDefinition"),
} as const,
{
- icon: ,
+ icon: ,
key: "9",
title: "Link",
id: "linkEventDefinition",
action: () => morphEvent("linkEventDefinition"),
} as const,
{
- icon: ,
+ icon: ,
key: "0",
title: "Signal",
id: "signalEventDefinition",
@@ -235,7 +235,7 @@ export function useEventNodeMorphingActions(event: Event) {
action: () => morphEvent("terminateEventDefinition"),
} as const,
];
- }, [event.__$$element, foregroundColor, fill, morphEvent]);
+ }, [event.__$$element, foregroundColor, filled, morphEvent]);
return morphingActions;
}