ngx-vflow 1.10.1 → 1.11.0

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.
@@ -34,6 +34,8 @@ export class ResizableComponent {
34
34
  this.zoom = computed(() => this.viewportService.readableViewport().zoom ?? 0);
35
35
  this.minWidth = 0;
36
36
  this.minHeight = 0;
37
+ this.maxWidth = Infinity;
38
+ this.maxHeight = Infinity;
37
39
  // TODO: allow reszie beside the flow
38
40
  this.resizeOnGlobalMouseMove = this.rootPointer.pointerMovement$
39
41
  .pipe(filter(() => this.resizeSide !== null), filter((event) => event.movementX !== 0 || event.movementY !== 0), tap((event) => this.resize(event)), takeUntilDestroyed())
@@ -57,6 +59,8 @@ export class ResizableComponent {
57
59
  ngAfterViewInit() {
58
60
  this.minWidth = +getComputedStyle(this.hostRef.nativeElement).minWidth.replace('px', '') || 0;
59
61
  this.minHeight = +getComputedStyle(this.hostRef.nativeElement).minHeight.replace('px', '') || 0;
62
+ this.maxWidth = +getComputedStyle(this.hostRef.nativeElement).maxWidth.replace('px', '') || Infinity;
63
+ this.maxHeight = +getComputedStyle(this.hostRef.nativeElement).maxHeight.replace('px', '') || Infinity;
60
64
  }
61
65
  startResize(side, event) {
62
66
  event.stopPropagation();
@@ -68,7 +72,7 @@ export class ResizableComponent {
68
72
  return;
69
73
  const offset = calcOffset(event.movementX, event.movementY, this.zoom());
70
74
  const resized = this.applyResize(this.resizeSide, this.model, offset, this.getDistanceToEdge(event));
71
- const { x, y, width, height } = constrainRect(resized, this.model, this.resizeSide, this.minWidth, this.minHeight);
75
+ const { x, y, width, height } = constrainRect(resized, this.model, this.resizeSide, this.minWidth, this.minHeight, this.maxWidth, this.maxHeight);
72
76
  this.model.setPoint({ x, y });
73
77
  this.model.width.set(width);
74
78
  this.model.height.set(height);
@@ -201,7 +205,7 @@ function calcOffset(movementX, movementY, zoom) {
201
205
  y: round(movementY / zoom),
202
206
  };
203
207
  }
204
- function constrainRect(rect, model, side, minWidth, minHeight) {
208
+ function constrainRect(rect, model, side, minWidth, minHeight, maxWidth, maxHeight) {
205
209
  let { x, y, width, height } = rect;
206
210
  // 1. Prevent negative dimensions
207
211
  width = Math.max(width, 0);
@@ -209,9 +213,13 @@ function constrainRect(rect, model, side, minWidth, minHeight) {
209
213
  // 2. Apply minimum size constraints
210
214
  width = Math.max(minWidth, width);
211
215
  height = Math.max(minHeight, height);
216
+ width = Math.min(maxWidth, width);
217
+ height = Math.min(maxHeight, height);
212
218
  // Apply left/top constraints based on minimum size
213
219
  x = Math.min(x, model.point().x + model.width() - minWidth);
214
220
  y = Math.min(y, model.point().y + model.height() - minHeight);
221
+ x = Math.max(x, model.point().x + model.width() - maxWidth);
222
+ y = Math.max(y, model.point().y + model.height() - maxHeight);
215
223
  const parent = model.parent();
216
224
  // 3. Apply maximum size constraints based on parent size (if exists)
217
225
  if (parent) {
@@ -257,4 +265,4 @@ function constrainRect(rect, model, side, minWidth, minHeight) {
257
265
  height,
258
266
  };
259
267
  }
260
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"resizable.component.js","sourceRoot":"","sources":["../../../../../../../projects/ngx-vflow-lib/src/lib/vflow/public-components/resizable/resizable.component.ts","../../../../../../../projects/ngx-vflow-lib/src/lib/vflow/public-components/resizable/resizable.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterViewInit,\n  Component,\n  computed,\n  ElementRef,\n  inject,\n  OnInit,\n  TemplateRef,\n  input,\n  viewChild,\n  effect,\n  ChangeDetectionStrategy,\n} from '@angular/core';\nimport { RootPointerDirective } from '../../directives/root-pointer.directive';\nimport { filter, tap } from 'rxjs';\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\nimport { ViewportService } from '../../services/viewport.service';\nimport { round } from '../../utils/round';\nimport { Microtask } from '../../decorators/microtask.decorator';\nimport { getNodesBounds } from '../../utils/nodes';\nimport { NodeAccessorService } from '../../services/node-accessor.service';\nimport { NodeModel } from '../../models/node.model';\nimport { Rect } from '../../interfaces/rect';\nimport { PointerEvent } from '../../directives/root-pointer.directive';\nimport { SpacePointContextDirective } from '../../directives/space-point-context.directive';\nimport { PointerDirective } from '../../directives/pointer.directive';\nimport { FlowSettingsService } from '../../services/flow-settings.service';\nimport { Point } from '../../interfaces/point.interface';\nimport { align } from '../../utils/align-number';\n\ntype Side = 'top' | 'right' | 'bottom' | 'left' | 'top-right' | 'top-left' | 'bottom-right' | 'bottom-left';\n\ninterface DistanceToEdge {\n  left: number;\n  right: number;\n  top: number;\n  bottom: number;\n}\n\n@Component({\n  standalone: true,\n  selector: '[resizable]',\n  templateUrl: './resizable.component.html',\n  styleUrls: ['./resizable.component.scss'],\n  imports: [PointerDirective],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ResizableComponent implements OnInit, AfterViewInit {\n  private nodeAccessor = inject(NodeAccessorService);\n  private rootPointer = inject(RootPointerDirective);\n  private viewportService = inject(ViewportService);\n  private spacePointContext = inject(SpacePointContextDirective);\n  private settingsService = inject(FlowSettingsService);\n  private hostRef = inject<ElementRef<Element>>(ElementRef);\n\n  public resizable = input<boolean | ''>();\n\n  public resizerColor = input('#2e414c');\n\n  public gap = input(1.5);\n\n  private resizer = viewChild.required<TemplateRef<unknown>>('resizer');\n\n  protected get model() {\n    return this.nodeAccessor.model()!;\n  }\n\n  protected lineGap = 3;\n  protected handleSize = 6;\n\n  private resizeSide: Side | null = null;\n\n  private zoom = computed(() => this.viewportService.readableViewport().zoom ?? 0);\n\n  private minWidth = 0;\n  private minHeight = 0;\n\n  // TODO: allow reszie beside the flow\n  protected resizeOnGlobalMouseMove = this.rootPointer.pointerMovement$\n    .pipe(\n      filter(() => this.resizeSide !== null),\n      filter((event) => event.movementX !== 0 || event.movementY !== 0),\n      tap((event) => this.resize(event)),\n      takeUntilDestroyed(),\n    )\n    .subscribe();\n\n  protected endResizeOnGlobalMouseUp = this.rootPointer.documentPointerEnd$\n    .pipe(\n      tap(() => this.endResize()),\n      takeUntilDestroyed(),\n    )\n    .subscribe();\n\n  constructor() {\n    effect(\n      () => {\n        const resizable = this.resizable();\n\n        if (typeof resizable === 'boolean') {\n          this.model.resizable.set(resizable);\n        } else {\n          this.model.resizable.set(true);\n        }\n      },\n      { allowSignalWrites: true },\n    );\n  }\n\n  public ngOnInit(): void {\n    this.model.resizerTemplate.set(this.resizer());\n  }\n\n  @Microtask\n  public ngAfterViewInit() {\n    this.minWidth = +getComputedStyle(this.hostRef.nativeElement).minWidth.replace('px', '') || 0;\n    this.minHeight = +getComputedStyle(this.hostRef.nativeElement).minHeight.replace('px', '') || 0;\n  }\n\n  protected startResize(side: Side, event: Event) {\n    event.stopPropagation();\n    this.resizeSide = side;\n    this.model.resizing.set(true);\n  }\n\n  protected resize(event: PointerEvent) {\n    if (!this.resizeSide) return;\n\n    const offset = calcOffset(event.movementX, event.movementY, this.zoom());\n\n    const resized = this.applyResize(this.resizeSide, this.model, offset, this.getDistanceToEdge(event));\n\n    const { x, y, width, height } = constrainRect(resized, this.model, this.resizeSide, this.minWidth, this.minHeight);\n\n    this.model.setPoint({ x, y });\n    this.model.width.set(width);\n    this.model.height.set(height);\n  }\n\n  protected endResize() {\n    this.resizeSide = null;\n    this.model.resizing.set(false);\n  }\n\n  private getDistanceToEdge(event: PointerEvent): DistanceToEdge {\n    const flowPoint = this.spacePointContext.documentPointToFlowPoint({ x: event.x, y: event.y });\n    const { x, y } = this.model.globalPoint();\n\n    return {\n      left: flowPoint.x - x,\n      right: flowPoint.x - (x + this.model.width()),\n      top: flowPoint.y - y,\n      bottom: flowPoint.y - (y + this.model.height()),\n    };\n  }\n\n  private applyResize(side: Side, model: NodeModel, offset: Point, distanceToEdge: DistanceToEdge): Rect {\n    const { x, y } = model.point();\n    const width = model.width();\n    const height = model.height();\n    const [snapX, snapY] = this.settingsService.snapGrid();\n\n    switch (side) {\n      case 'left': {\n        const movementX = offset.x + distanceToEdge.left;\n        const newX = align(x + movementX, snapX);\n        const deltaX = newX - x;\n\n        return {\n          x: newX,\n          y,\n          width: width - deltaX,\n          height,\n        };\n      }\n      case 'right': {\n        const movementX = offset.x + distanceToEdge.right;\n        const newWidth = align(width + movementX, snapX);\n\n        return {\n          x,\n          y,\n          width: newWidth,\n          height,\n        };\n      }\n      case 'top': {\n        const movementY = offset.y + distanceToEdge.top;\n        const newY = align(y + movementY, snapY);\n        const deltaY = newY - y;\n\n        return {\n          x,\n          y: newY,\n          width,\n          height: height - deltaY,\n        };\n      }\n      case 'bottom': {\n        const movementY = offset.y + distanceToEdge.bottom;\n        const newHeight = align(height + movementY, snapY);\n\n        return {\n          x,\n          y,\n          width,\n          height: newHeight,\n        };\n      }\n      case 'top-left': {\n        const movementX = offset.x + distanceToEdge.left;\n        const movementY = offset.y + distanceToEdge.top;\n        const newX = align(x + movementX, snapX);\n        const newY = align(y + movementY, snapY);\n        const deltaX = newX - x;\n        const deltaY = newY - y;\n\n        return {\n          x: newX,\n          y: newY,\n          width: width - deltaX,\n          height: height - deltaY,\n        };\n      }\n      case 'top-right': {\n        const movementX = offset.x + distanceToEdge.right;\n        const movementY = offset.y + distanceToEdge.top;\n        const newY = align(y + movementY, snapY);\n        const deltaY = newY - y;\n\n        return {\n          x,\n          y: newY,\n          width: align(width + movementX, snapX),\n          height: height - deltaY,\n        };\n      }\n      case 'bottom-left': {\n        const movementX = offset.x + distanceToEdge.left;\n        const movementY = offset.y + distanceToEdge.bottom;\n        const newX = align(x + movementX, snapX);\n        const deltaX = newX - x;\n\n        return {\n          x: newX,\n          y,\n          width: width - deltaX,\n          height: align(height + movementY, snapY),\n        };\n      }\n      case 'bottom-right': {\n        const movementX = offset.x + distanceToEdge.right;\n        const movementY = offset.y + distanceToEdge.bottom;\n\n        return {\n          x,\n          y,\n          width: align(width + movementX, snapX),\n          height: align(height + movementY, snapY),\n        };\n      }\n    }\n  }\n}\n\nfunction calcOffset(movementX: number, movementY: number, zoom: number): Point {\n  return {\n    x: round(movementX / zoom),\n    y: round(movementY / zoom),\n  };\n}\n\nfunction constrainRect(rect: Rect, model: NodeModel, side: Side, minWidth: number, minHeight: number) {\n  let { x, y, width, height } = rect;\n\n  // 1. Prevent negative dimensions\n  width = Math.max(width, 0);\n  height = Math.max(height, 0);\n\n  // 2. Apply minimum size constraints\n  width = Math.max(minWidth, width);\n  height = Math.max(minHeight, height);\n\n  // Apply left/top constraints based on minimum size\n  x = Math.min(x, model.point().x + model.width() - minWidth);\n  y = Math.min(y, model.point().y + model.height() - minHeight);\n\n  const parent = model.parent();\n  // 3. Apply maximum size constraints based on parent size (if exists)\n  if (parent) {\n    const parentWidth = parent.width();\n    const parentHeight = parent.height();\n    const modelX = model.point().x;\n    const modelY = model.point().y;\n\n    x = Math.max(x, 0);\n    y = Math.max(y, 0);\n\n    // Stop resizing when hitting left or top boundary\n    if (side.includes('left') && x === 0) {\n      width = Math.min(width, modelX + model.width());\n    }\n    if (side.includes('top') && y === 0) {\n      height = Math.min(height, modelY + model.height());\n    }\n\n    // Allow right/bottom resizing without being blocked\n    width = Math.min(width, parentWidth - x);\n    height = Math.min(height, parentHeight - y);\n  }\n\n  const bounds = getNodesBounds(model.children());\n  // 4. Apply child node constraints (if children exist)\n  if (bounds) {\n    if (side.includes('left')) {\n      x = Math.min(x, model.point().x + model.width() - (bounds.x + bounds.width));\n      width = Math.max(width, bounds.x + bounds.width);\n    }\n\n    if (side.includes('right')) {\n      width = Math.max(width, bounds.x + bounds.width);\n    }\n\n    if (side.includes('bottom')) {\n      height = Math.max(height, bounds.y + bounds.height);\n    }\n\n    if (side.includes('top')) {\n      y = Math.min(y, model.point().y + model.height() - (bounds.y + bounds.height));\n      height = Math.max(height, bounds.y + bounds.height);\n    }\n  }\n\n  return {\n    x,\n    y,\n    width,\n    height,\n  };\n}\n","<ng-template #resizer>\n  <svg:g>\n    <!-- top line -->\n    <svg:line\n      class=\"top\"\n      stroke-width=\"2\"\n      [attr.x1]=\"lineGap\"\n      [attr.y1]=\"-gap()\"\n      [attr.x2]=\"model.size().width - lineGap\"\n      [attr.y2]=\"-gap()\"\n      [attr.stroke]=\"resizerColor()\"\n      (pointerStart)=\"startResize('top', $event)\" />\n    <!-- Left line -->\n    <svg:line\n      class=\"left\"\n      stroke-width=\"2\"\n      [attr.x1]=\"-gap()\"\n      [attr.y1]=\"lineGap\"\n      [attr.x2]=\"-gap()\"\n      [attr.y2]=\"model.size().height - lineGap\"\n      [attr.stroke]=\"resizerColor()\"\n      (pointerStart)=\"startResize('left', $event)\" />\n    <!-- Bottom line -->\n    <svg:line\n      class=\"bottom\"\n      stroke-width=\"2\"\n      [attr.x1]=\"lineGap\"\n      [attr.y1]=\"model.size().height + gap()\"\n      [attr.x2]=\"model.size().width - lineGap\"\n      [attr.y2]=\"model.size().height + gap()\"\n      [attr.stroke]=\"resizerColor()\"\n      (pointerStart)=\"startResize('bottom', $event)\" />\n    <!-- Right line -->\n    <svg:line\n      class=\"right\"\n      stroke-width=\"2\"\n      [attr.x1]=\"model.size().width + gap()\"\n      [attr.y1]=\"lineGap\"\n      [attr.x2]=\"model.size().width + gap()\"\n      [attr.y2]=\"model.size().height - lineGap\"\n      [attr.stroke]=\"resizerColor()\"\n      (pointerStart)=\"startResize('right', $event)\" />\n\n    <!-- Top Left -->\n    <svg:rect\n      class=\"top-left\"\n      [attr.x]=\"-(handleSize / 2) - gap()\"\n      [attr.y]=\"-(handleSize / 2) - gap()\"\n      [attr.width]=\"handleSize\"\n      [attr.height]=\"handleSize\"\n      [attr.fill]=\"resizerColor()\"\n      (pointerStart)=\"startResize('top-left', $event)\" />\n\n    <!-- Top right -->\n    <svg:rect\n      class=\"top-right\"\n      [attr.x]=\"model.size().width - handleSize / 2 + gap()\"\n      [attr.y]=\"-(handleSize / 2) - gap()\"\n      [attr.width]=\"handleSize\"\n      [attr.height]=\"handleSize\"\n      [attr.fill]=\"resizerColor()\"\n      (pointerStart)=\"startResize('top-right', $event)\" />\n\n    <!-- Bottom left -->\n    <svg:rect\n      class=\"bottom-left\"\n      [attr.x]=\"-(handleSize / 2) - gap()\"\n      [attr.y]=\"model.size().height - handleSize / 2 + gap()\"\n      [attr.width]=\"handleSize\"\n      [attr.height]=\"handleSize\"\n      [attr.fill]=\"resizerColor()\"\n      (pointerStart)=\"startResize('bottom-left', $event)\" />\n\n    <!-- Bottom right -->\n    <svg:rect\n      class=\"bottom-right\"\n      [attr.x]=\"model.size().width - handleSize / 2 + gap()\"\n      [attr.y]=\"model.size().height - handleSize / 2 + gap()\"\n      [attr.width]=\"handleSize\"\n      [attr.height]=\"handleSize\"\n      [attr.fill]=\"resizerColor()\"\n      (pointerStart)=\"startResize('bottom-right', $event)\" />\n  </svg:g>\n</ng-template>\n\n<ng-content />\n"]}
268
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"resizable.component.js","sourceRoot":"","sources":["../../../../../../../projects/ngx-vflow-lib/src/lib/vflow/public-components/resizable/resizable.component.ts","../../../../../../../projects/ngx-vflow-lib/src/lib/vflow/public-components/resizable/resizable.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterViewInit,\n  Component,\n  computed,\n  ElementRef,\n  inject,\n  OnInit,\n  TemplateRef,\n  input,\n  viewChild,\n  effect,\n  ChangeDetectionStrategy,\n} from '@angular/core';\nimport { RootPointerDirective } from '../../directives/root-pointer.directive';\nimport { filter, tap } from 'rxjs';\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\nimport { ViewportService } from '../../services/viewport.service';\nimport { round } from '../../utils/round';\nimport { Microtask } from '../../decorators/microtask.decorator';\nimport { getNodesBounds } from '../../utils/nodes';\nimport { NodeAccessorService } from '../../services/node-accessor.service';\nimport { NodeModel } from '../../models/node.model';\nimport { Rect } from '../../interfaces/rect';\nimport { PointerEvent } from '../../directives/root-pointer.directive';\nimport { SpacePointContextDirective } from '../../directives/space-point-context.directive';\nimport { PointerDirective } from '../../directives/pointer.directive';\nimport { FlowSettingsService } from '../../services/flow-settings.service';\nimport { Point } from '../../interfaces/point.interface';\nimport { align } from '../../utils/align-number';\n\ntype Side = 'top' | 'right' | 'bottom' | 'left' | 'top-right' | 'top-left' | 'bottom-right' | 'bottom-left';\n\ninterface DistanceToEdge {\n  left: number;\n  right: number;\n  top: number;\n  bottom: number;\n}\n\n@Component({\n  standalone: true,\n  selector: '[resizable]',\n  templateUrl: './resizable.component.html',\n  styleUrls: ['./resizable.component.scss'],\n  imports: [PointerDirective],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ResizableComponent implements OnInit, AfterViewInit {\n  private nodeAccessor = inject(NodeAccessorService);\n  private rootPointer = inject(RootPointerDirective);\n  private viewportService = inject(ViewportService);\n  private spacePointContext = inject(SpacePointContextDirective);\n  private settingsService = inject(FlowSettingsService);\n  private hostRef = inject<ElementRef<Element>>(ElementRef);\n\n  public resizable = input<boolean | ''>();\n\n  public resizerColor = input('#2e414c');\n\n  public gap = input(1.5);\n\n  private resizer = viewChild.required<TemplateRef<unknown>>('resizer');\n\n  protected get model() {\n    return this.nodeAccessor.model()!;\n  }\n\n  protected lineGap = 3;\n  protected handleSize = 6;\n\n  private resizeSide: Side | null = null;\n\n  private zoom = computed(() => this.viewportService.readableViewport().zoom ?? 0);\n\n  private minWidth = 0;\n  private minHeight = 0;\n  private maxWidth = Infinity;\n  private maxHeight = Infinity;\n\n  // TODO: allow reszie beside the flow\n  protected resizeOnGlobalMouseMove = this.rootPointer.pointerMovement$\n    .pipe(\n      filter(() => this.resizeSide !== null),\n      filter((event) => event.movementX !== 0 || event.movementY !== 0),\n      tap((event) => this.resize(event)),\n      takeUntilDestroyed(),\n    )\n    .subscribe();\n\n  protected endResizeOnGlobalMouseUp = this.rootPointer.documentPointerEnd$\n    .pipe(\n      tap(() => this.endResize()),\n      takeUntilDestroyed(),\n    )\n    .subscribe();\n\n  constructor() {\n    effect(\n      () => {\n        const resizable = this.resizable();\n\n        if (typeof resizable === 'boolean') {\n          this.model.resizable.set(resizable);\n        } else {\n          this.model.resizable.set(true);\n        }\n      },\n      { allowSignalWrites: true },\n    );\n  }\n\n  public ngOnInit(): void {\n    this.model.resizerTemplate.set(this.resizer());\n  }\n\n  @Microtask\n  public ngAfterViewInit() {\n    this.minWidth = +getComputedStyle(this.hostRef.nativeElement).minWidth.replace('px', '') || 0;\n    this.minHeight = +getComputedStyle(this.hostRef.nativeElement).minHeight.replace('px', '') || 0;\n    this.maxWidth = +getComputedStyle(this.hostRef.nativeElement).maxWidth.replace('px', '') || Infinity;\n    this.maxHeight = +getComputedStyle(this.hostRef.nativeElement).maxHeight.replace('px', '') || Infinity;\n  }\n\n  protected startResize(side: Side, event: Event) {\n    event.stopPropagation();\n    this.resizeSide = side;\n    this.model.resizing.set(true);\n  }\n\n  protected resize(event: PointerEvent) {\n    if (!this.resizeSide) return;\n\n    const offset = calcOffset(event.movementX, event.movementY, this.zoom());\n\n    const resized = this.applyResize(this.resizeSide, this.model, offset, this.getDistanceToEdge(event));\n\n    const { x, y, width, height } = constrainRect(\n      resized,\n      this.model,\n      this.resizeSide,\n      this.minWidth,\n      this.minHeight,\n      this.maxWidth,\n      this.maxHeight,\n    );\n\n    this.model.setPoint({ x, y });\n    this.model.width.set(width);\n    this.model.height.set(height);\n  }\n\n  protected endResize() {\n    this.resizeSide = null;\n    this.model.resizing.set(false);\n  }\n\n  private getDistanceToEdge(event: PointerEvent): DistanceToEdge {\n    const flowPoint = this.spacePointContext.documentPointToFlowPoint({ x: event.x, y: event.y });\n    const { x, y } = this.model.globalPoint();\n\n    return {\n      left: flowPoint.x - x,\n      right: flowPoint.x - (x + this.model.width()),\n      top: flowPoint.y - y,\n      bottom: flowPoint.y - (y + this.model.height()),\n    };\n  }\n\n  private applyResize(side: Side, model: NodeModel, offset: Point, distanceToEdge: DistanceToEdge): Rect {\n    const { x, y } = model.point();\n    const width = model.width();\n    const height = model.height();\n    const [snapX, snapY] = this.settingsService.snapGrid();\n\n    switch (side) {\n      case 'left': {\n        const movementX = offset.x + distanceToEdge.left;\n        const newX = align(x + movementX, snapX);\n        const deltaX = newX - x;\n\n        return {\n          x: newX,\n          y,\n          width: width - deltaX,\n          height,\n        };\n      }\n      case 'right': {\n        const movementX = offset.x + distanceToEdge.right;\n        const newWidth = align(width + movementX, snapX);\n\n        return {\n          x,\n          y,\n          width: newWidth,\n          height,\n        };\n      }\n      case 'top': {\n        const movementY = offset.y + distanceToEdge.top;\n        const newY = align(y + movementY, snapY);\n        const deltaY = newY - y;\n\n        return {\n          x,\n          y: newY,\n          width,\n          height: height - deltaY,\n        };\n      }\n      case 'bottom': {\n        const movementY = offset.y + distanceToEdge.bottom;\n        const newHeight = align(height + movementY, snapY);\n\n        return {\n          x,\n          y,\n          width,\n          height: newHeight,\n        };\n      }\n      case 'top-left': {\n        const movementX = offset.x + distanceToEdge.left;\n        const movementY = offset.y + distanceToEdge.top;\n        const newX = align(x + movementX, snapX);\n        const newY = align(y + movementY, snapY);\n        const deltaX = newX - x;\n        const deltaY = newY - y;\n\n        return {\n          x: newX,\n          y: newY,\n          width: width - deltaX,\n          height: height - deltaY,\n        };\n      }\n      case 'top-right': {\n        const movementX = offset.x + distanceToEdge.right;\n        const movementY = offset.y + distanceToEdge.top;\n        const newY = align(y + movementY, snapY);\n        const deltaY = newY - y;\n\n        return {\n          x,\n          y: newY,\n          width: align(width + movementX, snapX),\n          height: height - deltaY,\n        };\n      }\n      case 'bottom-left': {\n        const movementX = offset.x + distanceToEdge.left;\n        const movementY = offset.y + distanceToEdge.bottom;\n        const newX = align(x + movementX, snapX);\n        const deltaX = newX - x;\n\n        return {\n          x: newX,\n          y,\n          width: width - deltaX,\n          height: align(height + movementY, snapY),\n        };\n      }\n      case 'bottom-right': {\n        const movementX = offset.x + distanceToEdge.right;\n        const movementY = offset.y + distanceToEdge.bottom;\n\n        return {\n          x,\n          y,\n          width: align(width + movementX, snapX),\n          height: align(height + movementY, snapY),\n        };\n      }\n    }\n  }\n}\n\nfunction calcOffset(movementX: number, movementY: number, zoom: number): Point {\n  return {\n    x: round(movementX / zoom),\n    y: round(movementY / zoom),\n  };\n}\n\nfunction constrainRect(\n  rect: Rect,\n  model: NodeModel,\n  side: Side,\n  minWidth: number,\n  minHeight: number,\n  maxWidth: number,\n  maxHeight: number,\n) {\n  let { x, y, width, height } = rect;\n\n  // 1. Prevent negative dimensions\n  width = Math.max(width, 0);\n  height = Math.max(height, 0);\n\n  // 2. Apply minimum size constraints\n  width = Math.max(minWidth, width);\n  height = Math.max(minHeight, height);\n  width = Math.min(maxWidth, width);\n  height = Math.min(maxHeight, height);\n\n  // Apply left/top constraints based on minimum size\n  x = Math.min(x, model.point().x + model.width() - minWidth);\n  y = Math.min(y, model.point().y + model.height() - minHeight);\n  x = Math.max(x, model.point().x + model.width() - maxWidth);\n  y = Math.max(y, model.point().y + model.height() - maxHeight);\n\n  const parent = model.parent();\n  // 3. Apply maximum size constraints based on parent size (if exists)\n  if (parent) {\n    const parentWidth = parent.width();\n    const parentHeight = parent.height();\n    const modelX = model.point().x;\n    const modelY = model.point().y;\n\n    x = Math.max(x, 0);\n    y = Math.max(y, 0);\n\n    // Stop resizing when hitting left or top boundary\n    if (side.includes('left') && x === 0) {\n      width = Math.min(width, modelX + model.width());\n    }\n    if (side.includes('top') && y === 0) {\n      height = Math.min(height, modelY + model.height());\n    }\n\n    // Allow right/bottom resizing without being blocked\n    width = Math.min(width, parentWidth - x);\n    height = Math.min(height, parentHeight - y);\n  }\n\n  const bounds = getNodesBounds(model.children());\n  // 4. Apply child node constraints (if children exist)\n  if (bounds) {\n    if (side.includes('left')) {\n      x = Math.min(x, model.point().x + model.width() - (bounds.x + bounds.width));\n      width = Math.max(width, bounds.x + bounds.width);\n    }\n\n    if (side.includes('right')) {\n      width = Math.max(width, bounds.x + bounds.width);\n    }\n\n    if (side.includes('bottom')) {\n      height = Math.max(height, bounds.y + bounds.height);\n    }\n\n    if (side.includes('top')) {\n      y = Math.min(y, model.point().y + model.height() - (bounds.y + bounds.height));\n      height = Math.max(height, bounds.y + bounds.height);\n    }\n  }\n\n  return {\n    x,\n    y,\n    width,\n    height,\n  };\n}\n","<ng-template #resizer>\n  <svg:g>\n    <!-- top line -->\n    <svg:line\n      class=\"top\"\n      stroke-width=\"2\"\n      [attr.x1]=\"lineGap\"\n      [attr.y1]=\"-gap()\"\n      [attr.x2]=\"model.size().width - lineGap\"\n      [attr.y2]=\"-gap()\"\n      [attr.stroke]=\"resizerColor()\"\n      (pointerStart)=\"startResize('top', $event)\" />\n    <!-- Left line -->\n    <svg:line\n      class=\"left\"\n      stroke-width=\"2\"\n      [attr.x1]=\"-gap()\"\n      [attr.y1]=\"lineGap\"\n      [attr.x2]=\"-gap()\"\n      [attr.y2]=\"model.size().height - lineGap\"\n      [attr.stroke]=\"resizerColor()\"\n      (pointerStart)=\"startResize('left', $event)\" />\n    <!-- Bottom line -->\n    <svg:line\n      class=\"bottom\"\n      stroke-width=\"2\"\n      [attr.x1]=\"lineGap\"\n      [attr.y1]=\"model.size().height + gap()\"\n      [attr.x2]=\"model.size().width - lineGap\"\n      [attr.y2]=\"model.size().height + gap()\"\n      [attr.stroke]=\"resizerColor()\"\n      (pointerStart)=\"startResize('bottom', $event)\" />\n    <!-- Right line -->\n    <svg:line\n      class=\"right\"\n      stroke-width=\"2\"\n      [attr.x1]=\"model.size().width + gap()\"\n      [attr.y1]=\"lineGap\"\n      [attr.x2]=\"model.size().width + gap()\"\n      [attr.y2]=\"model.size().height - lineGap\"\n      [attr.stroke]=\"resizerColor()\"\n      (pointerStart)=\"startResize('right', $event)\" />\n\n    <!-- Top Left -->\n    <svg:rect\n      class=\"top-left\"\n      [attr.x]=\"-(handleSize / 2) - gap()\"\n      [attr.y]=\"-(handleSize / 2) - gap()\"\n      [attr.width]=\"handleSize\"\n      [attr.height]=\"handleSize\"\n      [attr.fill]=\"resizerColor()\"\n      (pointerStart)=\"startResize('top-left', $event)\" />\n\n    <!-- Top right -->\n    <svg:rect\n      class=\"top-right\"\n      [attr.x]=\"model.size().width - handleSize / 2 + gap()\"\n      [attr.y]=\"-(handleSize / 2) - gap()\"\n      [attr.width]=\"handleSize\"\n      [attr.height]=\"handleSize\"\n      [attr.fill]=\"resizerColor()\"\n      (pointerStart)=\"startResize('top-right', $event)\" />\n\n    <!-- Bottom left -->\n    <svg:rect\n      class=\"bottom-left\"\n      [attr.x]=\"-(handleSize / 2) - gap()\"\n      [attr.y]=\"model.size().height - handleSize / 2 + gap()\"\n      [attr.width]=\"handleSize\"\n      [attr.height]=\"handleSize\"\n      [attr.fill]=\"resizerColor()\"\n      (pointerStart)=\"startResize('bottom-left', $event)\" />\n\n    <!-- Bottom right -->\n    <svg:rect\n      class=\"bottom-right\"\n      [attr.x]=\"model.size().width - handleSize / 2 + gap()\"\n      [attr.y]=\"model.size().height - handleSize / 2 + gap()\"\n      [attr.width]=\"handleSize\"\n      [attr.height]=\"handleSize\"\n      [attr.fill]=\"resizerColor()\"\n      (pointerStart)=\"startResize('bottom-right', $event)\" />\n  </svg:g>\n</ng-template>\n\n<ng-content />\n"]}
@@ -2392,6 +2392,8 @@ class ResizableComponent {
2392
2392
  this.zoom = computed(() => this.viewportService.readableViewport().zoom ?? 0);
2393
2393
  this.minWidth = 0;
2394
2394
  this.minHeight = 0;
2395
+ this.maxWidth = Infinity;
2396
+ this.maxHeight = Infinity;
2395
2397
  // TODO: allow reszie beside the flow
2396
2398
  this.resizeOnGlobalMouseMove = this.rootPointer.pointerMovement$
2397
2399
  .pipe(filter(() => this.resizeSide !== null), filter((event) => event.movementX !== 0 || event.movementY !== 0), tap((event) => this.resize(event)), takeUntilDestroyed())
@@ -2415,6 +2417,8 @@ class ResizableComponent {
2415
2417
  ngAfterViewInit() {
2416
2418
  this.minWidth = +getComputedStyle(this.hostRef.nativeElement).minWidth.replace('px', '') || 0;
2417
2419
  this.minHeight = +getComputedStyle(this.hostRef.nativeElement).minHeight.replace('px', '') || 0;
2420
+ this.maxWidth = +getComputedStyle(this.hostRef.nativeElement).maxWidth.replace('px', '') || Infinity;
2421
+ this.maxHeight = +getComputedStyle(this.hostRef.nativeElement).maxHeight.replace('px', '') || Infinity;
2418
2422
  }
2419
2423
  startResize(side, event) {
2420
2424
  event.stopPropagation();
@@ -2426,7 +2430,7 @@ class ResizableComponent {
2426
2430
  return;
2427
2431
  const offset = calcOffset(event.movementX, event.movementY, this.zoom());
2428
2432
  const resized = this.applyResize(this.resizeSide, this.model, offset, this.getDistanceToEdge(event));
2429
- const { x, y, width, height } = constrainRect(resized, this.model, this.resizeSide, this.minWidth, this.minHeight);
2433
+ const { x, y, width, height } = constrainRect(resized, this.model, this.resizeSide, this.minWidth, this.minHeight, this.maxWidth, this.maxHeight);
2430
2434
  this.model.setPoint({ x, y });
2431
2435
  this.model.width.set(width);
2432
2436
  this.model.height.set(height);
@@ -2559,7 +2563,7 @@ function calcOffset(movementX, movementY, zoom) {
2559
2563
  y: round(movementY / zoom),
2560
2564
  };
2561
2565
  }
2562
- function constrainRect(rect, model, side, minWidth, minHeight) {
2566
+ function constrainRect(rect, model, side, minWidth, minHeight, maxWidth, maxHeight) {
2563
2567
  let { x, y, width, height } = rect;
2564
2568
  // 1. Prevent negative dimensions
2565
2569
  width = Math.max(width, 0);
@@ -2567,9 +2571,13 @@ function constrainRect(rect, model, side, minWidth, minHeight) {
2567
2571
  // 2. Apply minimum size constraints
2568
2572
  width = Math.max(minWidth, width);
2569
2573
  height = Math.max(minHeight, height);
2574
+ width = Math.min(maxWidth, width);
2575
+ height = Math.min(maxHeight, height);
2570
2576
  // Apply left/top constraints based on minimum size
2571
2577
  x = Math.min(x, model.point().x + model.width() - minWidth);
2572
2578
  y = Math.min(y, model.point().y + model.height() - minHeight);
2579
+ x = Math.max(x, model.point().x + model.width() - maxWidth);
2580
+ y = Math.max(y, model.point().y + model.height() - maxHeight);
2573
2581
  const parent = model.parent();
2574
2582
  // 3. Apply maximum size constraints based on parent size (if exists)
2575
2583
  if (parent) {