| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532 | /* eslint-disable @typescript-eslint/no-explicit-any */import { intersectRayBounds, TLBounds } from '@tldraw/intersect'import Vec from '@tldraw/vec'import { action, autorun, computed, makeObservable, observable, toJS, transaction } from 'mobx'import { BINDING_DISTANCE } from '../../constants'import { TLResizeCorner, type TLBinding, type TLEventMap, type TLHandle } from '../../types'import { BoundsUtils, deepCopy, PointUtils } from '../../utils'import type { TLLineShape, TLShape, TLShapeModel } from '../shapes'import type { TLApp } from '../TLApp'export interface TLPageModel<S extends TLShape = TLShape> {  id: string  name: string  shapes: TLShapeModel<S['props']>[]  bindings: Record<string, TLBinding>  nonce?: number}export interface TLPageProps<S> {  id: string  name: string  shapes: S[]  bindings: Record<string, TLBinding>  nonce?: number}export class TLPage<S extends TLShape = TLShape, E extends TLEventMap = TLEventMap> {  constructor(app: TLApp<S, E>, props = {} as TLPageProps<S>) {    const { id, name, shapes = [], bindings = {}, nonce } = props    this.id = id    this.name = name    this.bindings = Object.assign({}, bindings) // make sure it is type of object    this.app = app    this.nonce = nonce || 0    this.addShapes(...shapes)    makeObservable(this)    autorun(() => {      const newShapesNouncesMap = Object.fromEntries(        this.shapes.map(shape => [shape.id, shape.nonce])      )      if (this.lastShapesNounces) {        const lastShapesNounces = this.lastShapesNounces        const allIds = new Set([          ...Object.keys(newShapesNouncesMap),          ...Object.keys(lastShapesNounces),        ])        const changedShapeIds = [...allIds].filter(s => {          return lastShapesNounces[s] !== newShapesNouncesMap[s]        })        requestAnimationFrame(() => {          this.cleanup(changedShapeIds)        })      }      this.lastShapesNounces = newShapesNouncesMap    })  }  lastShapesNounces = null as Record<string, number> | null  app: TLApp<S, E>  @observable id: string  @observable name: string  @observable shapes: S[] = []  @observable bindings: Record<string, TLBinding> = {}  @computed get serialized(): TLPageModel<S> {    return {      id: this.id,      name: this.name,      // @ts-expect-error maybe later      shapes: this.shapes        .map(shape => shape.serialized)        .filter(s => !!s)        .map(s => toJS(s)),      bindings: deepCopy(this.bindings),      nonce: this.nonce,    }  }  @observable nonce = 0  @action bump = () => {    this.nonce++  }  @action update(props: Partial<TLPageProps<S>>) {    Object.assign(this, props)    return this  }  @action updateBindings(bindings: Record<string, TLBinding>) {    Object.assign(this.bindings, bindings)    return this  }  @action addShapes(...shapes: S[] | TLShapeModel[]) {    if (shapes.length === 0) return    const shapeInstances =      'getBounds' in shapes[0]        ? (shapes as S[])        : (shapes as TLShapeModel[]).map(shape => {            const ShapeClass = this.app.getShapeClass(shape.type)            return new ShapeClass(shape)          })    this.shapes.push(...shapeInstances)    this.bump()    return shapeInstances  }  private parseShapesArg<S>(shapes: S[] | string[]) {    if (typeof shapes[0] === 'string') {      return this.shapes.filter(shape => (shapes as string[]).includes(shape.id))    } else {      return shapes as S[]    }  }  @action removeShapes(...shapes: S[] | string[]) {    const shapeInstances = this.parseShapesArg(shapes)    this.shapes = this.shapes.filter(shape => !shapeInstances.includes(shape))    return shapeInstances  }  @action bringForward = (shapes: S[] | string[]): this => {    const shapesToMove = this.parseShapesArg(shapes)    shapesToMove      .sort((a, b) => this.shapes.indexOf(b) - this.shapes.indexOf(a))      .map(shape => this.shapes.indexOf(shape))      .forEach(index => {        if (index === this.shapes.length - 1) return        const next = this.shapes[index + 1]        if (shapesToMove.includes(next)) return        const t = this.shapes[index]        this.shapes[index] = this.shapes[index + 1]        this.shapes[index + 1] = t      })    this.app.persist()    return this  }  @action sendBackward = (shapes: S[] | string[]): this => {    const shapesToMove = this.parseShapesArg(shapes)    shapesToMove      .sort((a, b) => this.shapes.indexOf(a) - this.shapes.indexOf(b))      .map(shape => this.shapes.indexOf(shape))      .forEach(index => {        if (index === 0) return        const next = this.shapes[index - 1]        if (shapesToMove.includes(next)) return        const t = this.shapes[index]        this.shapes[index] = this.shapes[index - 1]        this.shapes[index - 1] = t      })    this.app.persist()    return this  }  @action bringToFront = (shapes: S[] | string[]): this => {    const shapesToMove = this.parseShapesArg(shapes)    this.shapes = this.shapes.filter(shape => !shapesToMove.includes(shape)).concat(shapesToMove)    this.app.persist()    return this  }  @action sendToBack = (shapes: S[] | string[]): this => {    const shapesToMove = this.parseShapesArg(shapes)    this.shapes = shapesToMove.concat(this.shapes.filter(shape => !shapesToMove.includes(shape)))    this.app.persist()    return this  }  flip = (shapes: S[] | string[], direction: 'horizontal' | 'vertical'): this => {    const shapesToMove = this.parseShapesArg(shapes)    const commonBounds = BoundsUtils.getCommonBounds(shapesToMove.map(shape => shape.bounds))    shapesToMove.forEach(shape => {      const relativeBounds = BoundsUtils.getRelativeTransformedBoundingBox(        commonBounds,        commonBounds,        shape.bounds,        direction === 'horizontal',        direction === 'vertical'      )      if (shape.serialized) {        shape.onResize(shape.serialized, {          bounds: relativeBounds,          center: BoundsUtils.getBoundsCenter(relativeBounds),          rotation: shape.props.rotation ?? 0 * -1,          type: TLResizeCorner.TopLeft,          scale:            shape.canFlip && shape.props.scale              ? direction === 'horizontal'                ? [-shape.props.scale[0], 1]                : [1, -shape.props.scale[1]]              : [1, 1],          clip: false,          transformOrigin: [0.5, 0.5],        })      }    })    this.app.persist()    return this  }  getBindableShapes() {    return this.shapes      .filter(shape => shape.canBind)      .sort((a, b) => b.nonce - a.nonce)      .map(s => s.id)  }  getShapeById = <T extends S>(id: string): T | undefined => {    const shape = this.shapes.find(shape => shape.id === id) as T    return shape  }  /** Recalculate binding positions for changed shapes etc. Will also persist state when needed. */  @action  cleanup = (changedShapeIds: string[]) => {    // Get bindings related to the changed shapes    const bindingsToUpdate = getRelatedBindings(this.serialized, changedShapeIds)    const visitedShapes = new Set<string>()    let shapeChanged = false    let bindingChanged = false    const newBindings = deepCopy(this.bindings)    // Update all of the bindings we've just collected    bindingsToUpdate.forEach(binding => {      if (!this.bindings[binding.id]) {        return      }      const toShape = this.getShapeById(binding.toId)      const fromShape = this.getShapeById(binding.fromId)      if (!(toShape && fromShape)) {        delete newBindings[binding.id]        bindingChanged = true        return      }      if (visitedShapes.has(fromShape.id)) {        return      }      // We only need to update the binding's "from" shape (an arrow)      // @ts-expect-error ???      const fromDelta = this.updateArrowBindings(fromShape)      visitedShapes.add(fromShape.id)      if (fromDelta) {        const nextShape = {          ...fromShape.props,          ...fromDelta,        }        shapeChanged = true        this.getShapeById(nextShape.id)?.update(nextShape, false, true)      }    })    // Cleanup outdated bindings    Object.keys(newBindings).forEach(id => {      const binding = this.bindings[id]      const relatedShapes = this.shapes.filter(        shape => shape.id === binding.fromId || shape.id === binding.toId      )      if (relatedShapes.length === 0) {        delete newBindings[id]        bindingChanged = true      }    })    if (bindingChanged) {      this.update({        bindings: newBindings,      })    }    if (shapeChanged || bindingChanged) {      this.app.persist(true)    }  }  private updateArrowBindings = (lineShape: TLLineShape) => {    const result = {      start: deepCopy(lineShape.props.handles.start),      end: deepCopy(lineShape.props.handles.end),    }    type HandleInfo = {      handle: TLHandle      point: number[] // in page space    } & (      | {          isBound: false        }      | {          isBound: true          hasDecoration: boolean          binding: TLBinding          target: TLShape          bounds: TLBounds          expandedBounds: TLBounds          intersectBounds: TLBounds          center: number[]        }    )    let start: HandleInfo = {      isBound: false,      handle: lineShape.props.handles.start,      point: Vec.add(lineShape.props.handles.start.point, lineShape.props.point),    }    let end: HandleInfo = {      isBound: false,      handle: lineShape.props.handles.end,      point: Vec.add(lineShape.props.handles.end.point, lineShape.props.point),    }    if (lineShape.props.handles.start.bindingId) {      const hasDecoration = lineShape.props.decorations?.start !== undefined      const handle = lineShape.props.handles.start      const binding = this.bindings[lineShape.props.handles.start.bindingId]      // if (!binding) throw Error("Could not find a binding to match the start handle's bindingId")      const target = this.getShapeById(binding?.toId)      if (target) {        const bounds = target.getBounds()        const expandedBounds = target.getExpandedBounds()        const intersectBounds = BoundsUtils.expandBounds(          bounds,          hasDecoration ? binding.distance : 1        )        const { minX, minY, width, height } = expandedBounds        const anchorPoint = Vec.add(          [minX, minY],          Vec.mulV(            [width, height],            Vec.rotWith(binding.point, [0.5, 0.5], target.props.rotation || 0)          )        )        start = {          isBound: true,          hasDecoration,          binding,          handle,          point: anchorPoint,          target,          bounds,          expandedBounds,          intersectBounds,          center: target.getCenter(),        }      }    }    if (lineShape.props.handles.end.bindingId) {      const hasDecoration = lineShape.props.decorations?.end !== undefined      const handle = lineShape.props.handles.end      const binding = this.bindings[lineShape.props.handles.end.bindingId]      const target = this.getShapeById(binding?.toId)      if (target) {        const bounds = target.getBounds()        const expandedBounds = target.getExpandedBounds()        const intersectBounds = hasDecoration          ? BoundsUtils.expandBounds(bounds, binding.distance)          : bounds        const { minX, minY, width, height } = expandedBounds        const anchorPoint = Vec.add(          [minX, minY],          Vec.mulV(            [width, height],            Vec.rotWith(binding.point, [0.5, 0.5], target.props.rotation || 0)          )        )        end = {          isBound: true,          hasDecoration,          binding,          handle,          point: anchorPoint,          target,          bounds,          expandedBounds,          intersectBounds,          center: target.getCenter(),        }      }    }    for (const ID of ['end', 'start'] as const) {      const A = ID === 'start' ? start : end      const B = ID === 'start' ? end : start      if (A.isBound) {        if (!A.binding.distance) {          // If the binding distance is zero, then the arrow is bound to a specific point          // in the target shape. The resulting handle should be exactly at that point.          result[ID].point = Vec.sub(A.point, lineShape.props.point)        } else {          // We'll need to figure out the handle's true point based on some intersections          // between the opposite handle point and this handle point. This is different          // for each type of shape.          const direction = Vec.uni(Vec.sub(A.point, B.point))          switch (A.target.type) {            // TODO: do we need to support othershapes?            default: {              const hits = intersectRayBounds(                B.point,                direction,                A.intersectBounds,                A.target.props.rotation              )                .filter(int => int.didIntersect)                .map(int => int.points[0])                .sort((a, b) => Vec.dist(a, B.point) - Vec.dist(b, B.point))              if (!hits[0]) continue              let bHit: number[] | undefined = undefined              if (B.isBound) {                const bHits = intersectRayBounds(                  B.point,                  direction,                  B.intersectBounds,                  B.target.props.rotation                )                  .filter(int => int.didIntersect)                  .map(int => int.points[0])                  .sort((a, b) => Vec.dist(a, B.point) - Vec.dist(b, B.point))                bHit = bHits[0]              }              if (                B.isBound &&                (hits.length < 2 ||                  (bHit &&                    hits[0] &&                    Math.ceil(Vec.dist(hits[0], bHit)) < BINDING_DISTANCE * 2.5) ||                  BoundsUtils.boundsContain(A.expandedBounds, B.expandedBounds) ||                  BoundsUtils.boundsCollide(A.expandedBounds, B.expandedBounds))              ) {                // If the other handle is bound, and if we need to fallback to the short arrow method...                const shortArrowDirection = Vec.uni(Vec.sub(B.point, A.point))                const shortArrowHits = intersectRayBounds(                  A.point,                  shortArrowDirection,                  A.bounds,                  A.target.props.rotation                )                  .filter(int => int.didIntersect)                  .map(int => int.points[0])                if (!shortArrowHits[0]) continue                result[ID].point = Vec.toFixed(Vec.sub(shortArrowHits[0], lineShape.props.point))                result[ID === 'start' ? 'end' : 'start'].point = Vec.toFixed(                  Vec.add(                    Vec.sub(shortArrowHits[0], lineShape.props.point),                    Vec.mul(                      shortArrowDirection,                      Math.min(                        Vec.dist(shortArrowHits[0], B.point),                        BINDING_DISTANCE *                          2.5 *                          (BoundsUtils.boundsContain(B.bounds, A.intersectBounds) ? -1 : 1)                      )                    )                  )                )              } else if (                !B.isBound &&                ((hits[0] && Vec.dist(hits[0], B.point) < BINDING_DISTANCE * 2.5) ||                  PointUtils.pointInBounds(B.point, A.intersectBounds))              ) {                // Short arrow time!                const shortArrowDirection = Vec.uni(Vec.sub(A.center, B.point))                return lineShape.getHandlesChange?.(lineShape.props, {                  [ID]: {                    ...lineShape.props.handles[ID],                    point: Vec.toFixed(                      Vec.add(                        Vec.sub(B.point, lineShape.props.point),                        Vec.mul(shortArrowDirection, BINDING_DISTANCE * 2.5)                      )                    ),                  },                })              } else if (hits[0]) {                result[ID].point = Vec.toFixed(Vec.sub(hits[0], lineShape.props.point))              }            }          }        }      }    }    return lineShape.getHandlesChange(lineShape.props, result)  }}function getRelatedBindings(page: TLPageModel, ids: string[]): TLBinding[] {  const changedShapeIds = new Set(ids)  const bindingsArr = Object.values(page.bindings)  // Start with bindings that are directly bound to our changed shapes  const bindingsToUpdate = new Set(    bindingsArr.filter(      binding => changedShapeIds.has(binding.toId) || changedShapeIds.has(binding.fromId)    )  )  // Next, look for other bindings that effect the same shapes  let prevSize = bindingsToUpdate.size  let delta = -1  while (delta !== 0) {    bindingsToUpdate.forEach(binding => {      const fromId = binding.fromId      for (const otherBinding of bindingsArr) {        if (otherBinding.fromId === fromId) {          bindingsToUpdate.add(otherBinding)        }        if (otherBinding.toId === fromId) {          bindingsToUpdate.add(otherBinding)        }      }    })    // Continue until we stop finding new bindings to update    delta = bindingsToUpdate.size - prevSize    prevSize = bindingsToUpdate.size  }  return Array.from(bindingsToUpdate.values())}
 |