{"version":3,"file":"index.min.mjs","sources":["../src/config.ts","../src/util/internals/console.ts","../src/filters/GLProbes/GLProbe.ts","../src/filters/GLProbes/WebGLProbe.ts","../src/env/browser.ts","../src/env/index.ts","../src/cache.ts","../src/constants.ts","../src/ClassRegistry.ts","../src/util/animation/AnimationRegistry.ts","../src/Observable.ts","../src/util/internals/findRight.ts","../src/util/internals/getRandomInt.ts","../src/util/internals/ifNaN.ts","../src/util/internals/removeFromArray.ts","../src/util/misc/cos.ts","../src/util/misc/sin.ts","../src/Point.ts","../src/Collection.ts","../src/CommonMethods.ts","../src/util/animation/AnimationFrameProvider.ts","../src/util/internals/uid.ts","../src/util/misc/dom.ts","../src/util/misc/radiansDegreesConversion.ts","../src/util/misc/matrix.ts","../src/util/misc/objectEnlive.ts","../src/util/misc/pick.ts","../src/color/color_map.ts","../src/color/util.ts","../src/color/Color.ts","../src/color/constants.ts","../src/util/misc/toFixed.ts","../src/util/misc/svgParsing.ts","../src/util/typeAssertions.ts","../src/util/dom_misc.ts","../src/util/dom_style.ts","../src/canvas/DOMManagers/util.ts","../src/canvas/DOMManagers/StaticCanvasDOMManager.ts","../src/canvas/StaticCanvasOptions.ts","../src/canvas/StaticCanvas.ts","../src/util/dom_event.ts","../src/util/misc/boundingBoxFromPoints.ts","../src/util/misc/objectTransforms.ts","../src/util/misc/planeChange.ts","../src/controls/fireEvent.ts","../src/util/misc/resolveOrigin.ts","../src/controls/util.ts","../src/controls/drag.ts","../src/shapes/Object/FabricObjectSVGExportMixin.ts","../src/util/animation/easing.ts","../src/util/animation/AnimationBase.ts","../src/util/animation/ValueAnimation.ts","../src/util/animation/ArrayAnimation.ts","../src/util/misc/capValue.ts","../src/util/animation/ColorAnimation.ts","../src/util/animation/animate.ts","../src/util/misc/vectors.ts","../src/Intersection.ts","../src/shapes/Object/ObjectOrigin.ts","../src/shapes/Object/ObjectGeometry.ts","../src/shapes/Object/StackedObject.ts","../src/shapes/Object/AnimatableObject.ts","../src/parser/getSvgRegex.ts","../src/parser/constants.ts","../src/Shadow.ts","../src/util/internals/cloneDeep.ts","../src/shapes/Object/defaultValues.ts","../src/shapes/Object/Object.ts","../src/controls/wrapWithFireEvent.ts","../src/controls/wrapWithFixedAnchor.ts","../src/controls/changeWidth.ts","../src/controls/controlRendering.ts","../src/controls/Control.ts","../src/controls/scale.ts","../src/controls/skew.ts","../src/controls/scaleSkew.ts","../src/controls/commonControls.ts","../src/shapes/Object/InteractiveObject.ts","../src/util/applyMixins.ts","../src/shapes/Object/FabricObject.ts","../src/util/misc/isTransparent.ts","../src/util/misc/projectStroke/StrokeProjectionsBase.ts","../src/util/misc/projectStroke/StrokeLineJoinProjections.ts","../src/util/misc/projectStroke/StrokeLineCapProjections.ts","../src/util/misc/projectStroke/index.ts","../src/util/lang_string.ts","../src/util/misc/textStyles.ts","../src/parser/attributes.ts","../src/parser/selectorMatches.ts","../src/parser/elementMatchesRule.ts","../src/parser/doesSomeParentMatch.ts","../src/parser/normalizeAttr.ts","../src/util/internals/cleanupSvgAttribute.ts","../src/parser/parseTransformAttribute.ts","../src/parser/normalizeValue.ts","../src/parser/parseFontDeclaration.ts","../src/parser/parseStyleAttribute.ts","../src/parser/parseStyleString.ts","../src/parser/parseStyleObject.ts","../src/parser/setStrokeFillOpacity.ts","../src/parser/parseAttributes.ts","../src/parser/getGlobalStylesForElement.ts","../src/shapes/Rect.ts","../src/LayoutManager/constants.ts","../src/LayoutManager/LayoutStrategies/utils.ts","../src/LayoutManager/LayoutStrategies/LayoutStrategy.ts","../src/LayoutManager/LayoutStrategies/FitContentLayout.ts","../src/LayoutManager/LayoutManager.ts","../src/shapes/Group.ts","../src/util/misc/groupSVGElements.ts","../src/util/misc/findScaleTo.ts","../src/util/path/regex.ts","../src/util/path/index.ts","../src/util/internals/dom_request.ts","../src/util/transform_matrix_removal.ts","../src/util/misc/mergeClipPaths.ts","../src/util/misc/rotatePoint.ts","../src/canvas/DOMManagers/CanvasDOMManager.ts","../src/canvas/SelectableCanvas.ts","../src/canvas/CanvasOptions.ts","../src/canvas/TextEditingManager.ts","../src/canvas/Canvas.ts","../src/gradient/constants.ts","../src/parser/percent.ts","../src/gradient/parser/parseColorStops.ts","../src/gradient/parser/misc.ts","../src/gradient/parser/parseCoords.ts","../src/gradient/Gradient.ts","../src/Pattern/Pattern.ts","../src/brushes/BaseBrush.ts","../src/shapes/Path.ts","../src/brushes/PencilBrush.ts","../src/shapes/Circle.ts","../src/brushes/CircleBrush.ts","../src/brushes/SprayBrush.ts","../src/brushes/PatternBrush.ts","../src/shapes/Line.ts","../src/shapes/Triangle.ts","../src/shapes/Ellipse.ts","../src/parser/parsePointsAttribute.ts","../src/shapes/Polyline.ts","../src/shapes/Polygon.ts","../src/shapes/Text/constants.ts","../src/shapes/Text/StyledText.ts","../src/shapes/Text/TextSVGExportMixin.ts","../src/shapes/Text/Text.ts","../src/shapes/IText/DraggableTextDelegate.ts","../src/shapes/IText/ITextBehavior.ts","../src/shapes/IText/ITextKeyBehavior.ts","../src/shapes/IText/ITextClickBehavior.ts","../src/shapes/IText/constants.ts","../src/shapes/IText/IText.ts","../src/LayoutManager/LayoutStrategies/ClipPathLayout.ts","../src/LayoutManager/LayoutStrategies/FixedLayout.ts","../src/LayoutManager/ActiveSelectionLayoutManager.ts","../src/shapes/ActiveSelection.ts","../src/filters/Canvas2dFilterBackend.ts","../src/filters/WebGLFilterBackend.ts","../src/filters/FilterBackend.ts","../src/shapes/Image.ts","../src/parser/applyViewboxTransform.ts","../src/parser/getTagName.ts","../src/parser/hasInvalidAncestor.ts","../src/parser/getMultipleNodes.ts","../src/parser/recursivelyParseGradientsXlink.ts","../src/parser/getGradientDefs.ts","../src/parser/getCSSRules.ts","../src/parser/elements_parser.ts","../src/parser/parseSVGDocument.ts","../src/parser/parseUseDirectives.ts","../src/parser/loadSVGFromString.ts","../src/parser/loadSVGFromURL.ts","../src/controls/polyControl.ts","../src/controls/rotate.ts","../src/filters/utils.ts","../src/filters/shaders/baseFilter.ts","../src/filters/BaseFilter.ts","../src/filters/shaders/blendColor.ts","../src/filters/BlendColor.ts","../src/filters/shaders/blendImage.ts","../src/filters/BlendImage.ts","../src/filters/Blur.ts","../src/filters/shaders/blur.ts","../src/filters/Brightness.ts","../src/filters/shaders/brightness.ts","../src/filters/shaders/colorMatrix.ts","../src/filters/ColorMatrix.ts","../src/filters/ColorMatrixFilters.ts","../src/filters/Composed.ts","../src/filters/Contrast.ts","../src/filters/shaders/constrast.ts","../src/filters/shaders/convolute.ts","../src/filters/Convolute.ts","../src/filters/shaders/gamma.ts","../src/filters/Gamma.ts","../src/filters/shaders/grayscale.ts","../src/filters/Grayscale.ts","../src/filters/HueRotation.ts","../src/filters/Invert.ts","../src/filters/shaders/invert.ts","../src/filters/Noise.ts","../src/filters/shaders/noise.ts","../src/filters/Pixelate.ts","../src/filters/shaders/pixelate.ts","../src/filters/RemoveColor.ts","../src/filters/shaders/removeColor.ts","../src/filters/Resize.ts","../src/filters/Saturation.ts","../src/filters/shaders/saturation.ts","../src/filters/Vibrance.ts","../src/filters/shaders/vibrance.ts","../src/controls/X_commonControls.ts","../src/shapes/canvasx/types.ts","../src/shapes/Textbox.ts","../src/shapes/canvasx/type/widget.entity.textbox.ts","../src/shapes/canvasx/XTextbase.ts","../src/shapes/canvasx/type/widget.entity.circlenote.ts","../src/shapes/canvasx/XCircleNotes.ts","../src/shapes/canvasx/type/widget.entity.rectnote.ts","../src/shapes/canvasx/XRectNotes.ts","../src/shapes/canvasx/XShapeNotes/types.ts","../src/shapes/canvasx/type/widget.entity.shapenote.ts","../src/shapes/canvasx/XShapeNotes/XShapeNotes.ts","../src/shapes/canvasx/XTextbox.ts","../src/shapes/canvasx/type/widget.entity.image.ts","../src/shapes/canvasx/XImage.ts","../src/controls/pathControl.ts","../src/shapes/canvasx/type/widget.entity.connector.ts","../src/shapes/canvasx/XConnector.ts","../src/shapes/canvasx/type/widget.entity.file.ts","../src/shapes/canvasx/XFile/XFile.ts","../src/shapes/canvasx/XFile/XFileAudio.ts","../src/shapes/canvasx/XFile/XFileVideo.ts","../src/shapes/canvasx/XFile/XFileExcel.ts","../src/shapes/canvasx/XFile/XFileWord.ts","../src/shapes/canvasx/XFile/XFileZip.ts","../src/shapes/canvasx/XFile/XFilePDF.ts","../src/shapes/canvasx/XFile/XFilePPT.ts","../src/shapes/canvasx/type/widget.entity.group.ts","../src/shapes/canvasx/XGroup.ts","../src/shapes/canvasx/type/widget.entity.url.ts","../src/shapes/canvasx/XURL/XURL.ts","../src/shapes/canvasx/type/widget.entity.markdown.ts","../src/shapes/canvasx/XMarkdown.ts","../src/shapes/canvasx/type/widget.entity.chart.ts","../src/shapes/canvasx/XChart.ts","../src/shapes/canvasx/type/widget.entity.frame.ts","../src/shapes/canvasx/XFrame.ts","../src/shapes/canvasx/type/widget.entity.path.ts","../src/shapes/canvasx/XPath.ts","../src/canvas/canvasx/bx-canvas.ts"],"sourcesContent":["export type TConfiguration = Partial<BaseConfiguration>;\n\nclass BaseConfiguration {\n  /**\n   * Browser-specific constant to adjust CanvasRenderingContext2D.shadowBlur value,\n   * which is unitless and not rendered equally across browsers.\n   *\n   * Values that work quite well (as of October 2017) are:\n   * - Chrome: 1.5\n   * - Edge: 1.75\n   * - Firefox: 0.9\n   * - Safari: 0.95\n   *\n   * @since 2.0.0\n   * @type Number\n   * @default 1\n   */\n  browserShadowBlurConstant = 1;\n\n  /**\n   * Pixel per Inch as a default value set to 96. Can be changed for more realistic conversion.\n   */\n  DPI = 96;\n\n  /**\n   * Device Pixel Ratio\n   * @see https://developer.apple.com/library/safari/documentation/AudioVideo/Conceptual/HTML-canvas-guide/SettingUptheCanvas/SettingUptheCanvas.html\n   */\n  devicePixelRatio =\n    typeof window !== 'undefined' ? window.devicePixelRatio : 1; // eslint-disable-line no-restricted-globals\n\n  /**\n   * Pixel limit for cache canvases. 1Mpx , 4Mpx should be fine.\n   * @since 1.7.14\n   * @type Number\n   * @default\n   */\n  perfLimitSizeTotal = 2097152;\n\n  /**\n   * Pixel limit for cache canvases width or height. IE fixes the maximum at 5000\n   * @since 1.7.14\n   * @type Number\n   * @default\n   */\n  maxCacheSideLimit = 4096;\n\n  /**\n   * Lowest pixel limit for cache canvases, set at 256PX\n   * @since 1.7.14\n   * @type Number\n   * @default\n   */\n  minCacheSideLimit = 256;\n\n  /**\n   * When 'true', style information is not retained when copy/pasting text, making\n   * pasted text use destination style.\n   * Defaults to 'false'.\n   * @type Boolean\n   * @default\n   * @deprecated\n   */\n  disableStyleCopyPaste = false;\n\n  /**\n   * Enable webgl for filtering picture is available\n   * A filtering backend will be initialized, this will both take memory and\n   * time since a default 2048x2048 canvas will be created for the gl context\n   * @since 2.0.0\n   * @type Boolean\n   * @default\n   */\n  enableGLFiltering = true;\n\n  /**\n   * if webgl is enabled and available, textureSize will determine the size\n   * of the canvas backend\n   *\n   * In order to support old hardware set to `2048` to avoid OOM\n   *\n   * @since 2.0.0\n   * @type Number\n   * @default\n   */\n  textureSize = 4096;\n\n  /**\n   * Skip performance testing of setupGLContext and force the use of putImageData that seems to be the one that works best on\n   * Chrome + old hardware. if your users are experiencing empty images after filtering you may try to force this to true\n   * this has to be set before instantiating the filtering backend ( before filtering the first image )\n   * @type Boolean\n   * @default false\n   */\n  forceGLPutImageData = false;\n\n  /**\n   * If disabled boundsOfCurveCache is not used. For apps that make heavy usage of pencil drawing probably disabling it is better\n   * @default true\n   */\n  cachesBoundsOfCurve = true;\n\n  /**\n   * Map of font files\n   * Map<fontFamily, pathToFile> of font files\n   */\n  fontPaths: Record</** fontFamily */ string, /** pathToFile */ string> = {};\n\n  /**\n   * Defines the number of fraction digits to use when serializing object values.\n   * Used in exporting methods (`toObject`, `toJSON`, `toSVG`)\n   * You can use it to increase/decrease precision of such values like left, top, scaleX, scaleY, etc.\n   */\n  NUM_FRACTION_DIGITS = 4;\n}\n\nexport class Configuration extends BaseConfiguration {\n  constructor(config?: TConfiguration) {\n    super();\n    this.configure(config);\n  }\n\n  configure(config: TConfiguration = {}) {\n    Object.assign(this, config);\n  }\n\n  /**\n   * Map<fontFamily, pathToFile> of font files\n   */\n  addFonts(\n    paths: Record</** fontFamily */ string, /** pathToFile */ string> = {}\n  ) {\n    this.fontPaths = {\n      ...this.fontPaths,\n      ...paths,\n    };\n  }\n\n  removeFonts(fontFamilys: string[] = []) {\n    fontFamilys.forEach((fontFamily) => {\n      delete this.fontPaths[fontFamily];\n    });\n  }\n\n  clearFonts() {\n    this.fontPaths = {};\n  }\n\n  restoreDefaults<T extends BaseConfiguration>(keys?: (keyof T)[]) {\n    const defaults = new BaseConfiguration() as T;\n    const config =\n      keys?.reduce((acc, key) => {\n        acc[key] = defaults[key];\n        return acc;\n      }, {} as T) || defaults;\n    this.configure(config);\n  }\n}\n\nexport const config = new Configuration();\n","export const log = (\n  severity: 'log' | 'warn' | 'error',\n  ...optionalParams: any[]\n) =>\n  // eslint-disable-next-line no-restricted-syntax\n  console[severity]('fabric', ...optionalParams);\n\nexport class FabricError extends Error {\n  constructor(message?: string, options?: ErrorOptions) {\n    super(`fabric: ${message}`, options);\n  }\n}\n\nexport class SignalAbortedError extends FabricError {\n  constructor(context: string) {\n    super(`${context} 'options.signal' is in 'aborted' state`);\n  }\n}\n","export type GLPrecision = 'lowp' | 'mediump' | 'highp';\n\nexport abstract class GLProbe {\n  declare GLPrecision: GLPrecision | undefined;\n  abstract queryWebGL(canvas: HTMLCanvasElement): void;\n  abstract isSupported(textureSize: number): boolean;\n}\n","import { log } from '../../util/internals/console';\nimport { GLProbe } from './GLProbe';\nimport type { GLPrecision } from './GLProbe';\n\n/**\n * Lazy initialize WebGL constants\n */\nexport class WebGLProbe extends GLProbe {\n  declare maxTextureSize?: number;\n\n  /**\n   * Tests if webgl supports certain precision\n   * @param {WebGL} Canvas WebGL context to test on\n   * @param {GLPrecision} Precision to test can be any of following\n   * @returns {Boolean} Whether the user's browser WebGL supports given precision.\n   */\n  private testPrecision(\n    gl: WebGLRenderingContext,\n    precision: GLPrecision\n  ): boolean {\n    const fragmentSource = `precision ${precision} float;\\nvoid main(){}`;\n    const fragmentShader = gl.createShader(gl.FRAGMENT_SHADER);\n    if (!fragmentShader) {\n      return false;\n    }\n    gl.shaderSource(fragmentShader, fragmentSource);\n    gl.compileShader(fragmentShader);\n    return !!gl.getShaderParameter(fragmentShader, gl.COMPILE_STATUS);\n  }\n\n  /**\n   * query browser for WebGL\n   */\n  queryWebGL(canvas: HTMLCanvasElement) {\n    const gl = canvas.getContext('webgl');\n    if (gl) {\n      this.maxTextureSize = gl.getParameter(gl.MAX_TEXTURE_SIZE);\n      this.GLPrecision = (['highp', 'mediump', 'lowp'] as const).find(\n        (precision) => this.testPrecision(gl, precision)\n      );\n      gl.getExtension('WEBGL_lose_context')!.loseContext();\n      log('log', `WebGL: max texture size ${this.maxTextureSize}`);\n    }\n  }\n\n  isSupported(textureSize: number) {\n    return !!this.maxTextureSize && this.maxTextureSize >= textureSize;\n  }\n}\n","/* eslint-disable no-restricted-globals */\nimport { WebGLProbe } from '../filters/GLProbes/WebGLProbe';\nimport type { TCopyPasteData, TFabricEnv } from './types';\n\nconst copyPasteData: TCopyPasteData = {};\n\nexport const getEnv = (): TFabricEnv => {\n  return {\n    document,\n    window,\n    isTouchSupported:\n      'ontouchstart' in window ||\n      'ontouchstart' in document ||\n      (window && window.navigator && window.navigator.maxTouchPoints > 0),\n    WebGLProbe: new WebGLProbe(),\n    dispose() {\n      // noop\n    },\n    copyPasteData,\n  };\n};\n","/**\n * This file is consumed by fabric.\n * The `./node` and `./browser` files define the env variable that is used by this module.\n * The `./browser` module is defined to be the default env and doesn't set the env at all.\n * This is done in order to support isomorphic usage for browser and node applications\n * since window and document aren't defined at time of import in SSR, we can't set env so we avoid it by deferring to the default env.\n */\n\nimport { config } from '../config';\nimport { getEnv as getBrowserEnv } from './browser';\nimport type { TFabricEnv } from './types';\nimport type { DOMWindow } from 'jsdom';\n\nlet env: TFabricEnv;\n\n/**\n * Sets the environment variables used by fabric.\\\n * This is exposed for special cases, such as configuring a test environment, and should be used with care.\n *\n * **CAUTION**: Must be called before using the package.\n *\n * @example\n * <caption>Passing `window` and `document` objects to fabric (in case they are mocked or something)</caption>\n * import { getEnv, setEnv } from 'fabric';\n * // we want fabric to use the `window` and `document` objects exposed by the environment we are running in.\n * setEnv({ ...getEnv(), window, document });\n * // done with setup, using fabric is now safe\n */\nexport const setEnv = (value: TFabricEnv) => {\n  env = value;\n};\n\n/**\n * In order to support SSR we **MUST** access the browser env only after the window has loaded\n */\nexport const getEnv = () => env || (env = getBrowserEnv());\n\nexport const getFabricDocument = (): Document => getEnv().document;\n\nexport const getFabricWindow = (): (Window & typeof globalThis) | DOMWindow =>\n  getEnv().window;\n\n/**\n * @returns the config value if defined, fallbacks to the environment value\n */\nexport const getDevicePixelRatio = () =>\n  Math.max(config.devicePixelRatio ?? getFabricWindow().devicePixelRatio, 1);\n","import { config } from './config';\nimport type { TRectBounds } from './typedefs';\n\nexport class Cache {\n  /**\n   * Cache of widths of chars in text rendering.\n   */\n  charWidthsCache: Record<\n    /** fontFamily */ string,\n    Record<\n      /** fontStyleCacheKey */ string,\n      Record</** char */ string, /** width */ number>\n    >\n  > = {};\n\n  /**\n   * @return {Object} reference to cache\n   */\n  getFontCache({\n    fontFamily,\n    fontStyle,\n    fontWeight,\n  }: {\n    fontFamily: string;\n    fontStyle: string;\n    fontWeight: string | number;\n  }) {\n    fontFamily = fontFamily.toLowerCase();\n    if (!this.charWidthsCache[fontFamily]) {\n      this.charWidthsCache[fontFamily] = {};\n    }\n    const fontCache = this.charWidthsCache[fontFamily];\n    const cacheKey = `${fontStyle.toLowerCase()}_${(\n      fontWeight + ''\n    ).toLowerCase()}`;\n    if (!fontCache[cacheKey]) {\n      fontCache[cacheKey] = {};\n    }\n    return fontCache[cacheKey];\n  }\n\n  /**\n   * Clear char widths cache for the given font family or all the cache if no\n   * fontFamily is specified.\n   * Use it if you know you are loading fonts in a lazy way and you are not waiting\n   * for custom fonts to load properly when adding text objects to the canvas.\n   * If a text object is added when its own font is not loaded yet, you will get wrong\n   * measurement and so wrong bounding boxes.\n   * After the font cache is cleared, either change the textObject text content or call\n   * initDimensions() to trigger a recalculation\n   * @param {String} [fontFamily] font family to clear\n   */\n  clearFontCache(fontFamily?: string) {\n    fontFamily = (fontFamily || '').toLowerCase();\n    if (!fontFamily) {\n      this.charWidthsCache = {};\n    } else if (this.charWidthsCache[fontFamily]) {\n      delete this.charWidthsCache[fontFamily];\n    }\n  }\n\n  /**\n   * Given current aspect ratio, determines the max width and height that can\n   * respect the total allowed area for the cache.\n   * @param {number} ar aspect ratio\n   * @return {number[]} Limited dimensions X and Y\n   */\n  limitDimsByArea(ar: number) {\n    const { perfLimitSizeTotal } = config;\n    const roughWidth = Math.sqrt(perfLimitSizeTotal * ar);\n    // we are not returning a point on purpose, to avoid circular dependencies\n    // this is an internal utility\n    return [\n      Math.floor(roughWidth),\n      Math.floor(perfLimitSizeTotal / roughWidth),\n    ];\n  }\n\n  /**\n   * This object keeps the results of the boundsOfCurve calculation mapped by the joined arguments necessary to calculate it.\n   * It does speed up calculation, if you parse and add always the same paths, but in case of heavy usage of freedrawing\n   * you do not get any speed benefit and you get a big object in memory.\n   * The object was a private variable before, while now is appended to the lib so that you have access to it and you\n   * can eventually clear it.\n   * It was an internal variable, is accessible since version 2.3.4\n   */\n  boundsOfCurveCache: Record<string, TRectBounds> = {};\n}\n\nexport const cache = new Cache();\n","import type { TMat2D } from './typedefs';\n// use this syntax so babel plugin see this import here\nimport { version } from '../package.json';\n\nexport const VERSION = version;\n// eslint-disable-next-line @typescript-eslint/no-empty-function\nexport function noop() {}\n\nexport const halfPI = Math.PI / 2;\nexport const twoMathPi = Math.PI * 2;\nexport const PiBy180 = Math.PI / 180;\n\nexport const iMatrix = Object.freeze([1, 0, 0, 1, 0, 0]) as TMat2D;\nexport const DEFAULT_SVG_FONT_SIZE = 16;\nexport const ALIASING_LIMIT = 2;\n\n/* \"magic number\" for bezier approximations of arcs (http://itc.ktu.lt/itc354/Riskus354.pdf) */\nexport const kRect = 1 - 0.5522847498;\n\nexport const CENTER = 'center';\nexport const LEFT = 'left';\nexport const TOP = 'top';\nexport const BOTTOM = 'bottom';\nexport const RIGHT = 'right';\nexport const NONE = 'none';\n\nexport const MOVING = 'moving';\nexport const SCALING = 'scaling';\nexport const ROTATING = 'rotating';\nexport const ROTATE = 'rotate';\nexport const SKEWING = 'skewing';\nexport const RESIZING = 'resizing';\nexport const MODIFY_POLY = 'modifyPoly';\nexport const MODIFY_PATH = 'modifyPath';\nexport const CHANGED = 'changed';\nexport const SCALE = 'scale';\nexport const SCALE_X = 'scaleX';\nexport const SCALE_Y = 'scaleY';\nexport const SKEW_X = 'skewX';\nexport const SKEW_Y = 'skewY';\nexport const FILL = 'fill';\nexport const STROKE = 'stroke';\nexport const MODIFIED = 'modified';\n\nexport const reNewline = /\\r?\\n/;\n","import { FabricError } from './util/internals/console';\n\n/*\n * This Map connects the objects type value with their\n * class implementation. It used from any object to understand which are\n * the classes to enlive when requesting a object.type = 'path' for example.\n * Objects uses it for clipPath, Canvas uses it for everything.\n * This is necessary for generic code to run and enlive instances from serialized representation.\n * You can customize which classes get enlived from SVG parsing using this classRegistry.\n * The Registry start empty and gets filled in depending which files you import.\n * If you want to be able to parse arbitrary SVGs or JSON representation of canvases, coming from\n * different sources you will need to import all fabric because you may need all classes.\n */\n\nexport const JSON = 'json';\nexport const SVG = 'svg';\n\nexport class ClassRegistry {\n  declare [JSON]: Map<string, any>;\n  declare [SVG]: Map<string, any>;\n\n  constructor() {\n    this[JSON] = new Map();\n    this[SVG] = new Map();\n  }\n\n  getClass<T>(classType: string): T {\n    const constructor = this[JSON].get(classType);\n    if (!constructor) {\n      throw new FabricError(`No class registered for ${classType}`);\n    }\n    return constructor;\n  }\n\n  setClass(classConstructor: any, classType?: string) {\n    if (classType) {\n      this[JSON].set(classType, classConstructor);\n    } else {\n      this[JSON].set(classConstructor.type, classConstructor);\n      // legacy\n      // @TODO: needs to be removed in fabric 7 or 8\n      this[JSON].set(classConstructor.type.toLowerCase(), classConstructor);\n    }\n  }\n\n  getSVGClass(SVGTagName: string): any {\n    return this[SVG].get(SVGTagName);\n  }\n\n  setSVGClass(classConstructor: any, SVGTagName?: string) {\n    this[SVG].set(\n      SVGTagName ?? classConstructor.type.toLowerCase(),\n      classConstructor\n    );\n  }\n}\n\nexport const classRegistry = new ClassRegistry();\n","import type { StaticCanvas } from '../../canvas/StaticCanvas';\nimport type { FabricObject } from '../../shapes/Object/FabricObject';\nimport type { AnimationBase } from './AnimationBase';\n\n/**\n * Array holding all running animations\n */\nclass AnimationRegistry extends Array<AnimationBase> {\n  /**\n   * Remove a single animation using an animation context\n   * @param {AnimationBase} context\n   */\n  remove(context: AnimationBase) {\n    const index = this.indexOf(context);\n    index > -1 && this.splice(index, 1);\n  }\n\n  /**\n   * Cancel all running animations on the next frame\n   */\n  cancelAll() {\n    const animations = this.splice(0);\n    animations.forEach((animation) => animation.abort());\n    return animations;\n  }\n\n  /**\n   * Cancel all running animations attached to a canvas on the next frame\n   * @param {StaticCanvas} canvas\n   */\n  cancelByCanvas(canvas: StaticCanvas) {\n    if (!canvas) {\n      return [];\n    }\n    const animations = this.filter(\n      (animation) =>\n        animation.target === canvas ||\n        (typeof animation.target === 'object' &&\n          (animation.target as FabricObject)?.canvas === canvas)\n    );\n    animations.forEach((animation) => animation.abort());\n    return animations;\n  }\n\n  /**\n   * Cancel all running animations for target on the next frame\n   * @param target\n   */\n  cancelByTarget(target: AnimationBase['target']) {\n    if (!target) {\n      return [];\n    }\n    const animations = this.filter((animation) => animation.target === target);\n    animations.forEach((animation) => animation.abort());\n    return animations;\n  }\n}\n\nexport const runningAnimations = new AnimationRegistry();\n","export type TEventCallback<T = any> = (options: T) => any;\n\ntype EventRegistryObject<E> = {\n  [K in keyof E]?: TEventCallback<E[K]>;\n};\n\n/**\n * @tutorial {@link http://fabricjs.com/fabric-intro-part-2#events}\n * @see {@link http://fabricjs.com/events|Events demo}\n */\nexport class Observable<EventSpec> {\n  private __eventListeners: Record<keyof EventSpec, TEventCallback[]> =\n    {} as Record<keyof EventSpec, TEventCallback[]>;\n\n  /**\n   * Observes specified event\n   * @alias on\n   * @param {string} eventName Event name (eg. 'after:render')\n   * @param {EventRegistryObject} handlers key/value pairs (eg. {'after:render': handler, 'selection:cleared': handler})\n   * @param {Function} handler Function that receives a notification when an event of the specified type occurs\n   * @return {Function} disposer\n   */\n  on<K extends keyof EventSpec, E extends EventSpec[K]>(\n    eventName: K,\n    handler: TEventCallback<E>\n  ): VoidFunction;\n  on(handlers: EventRegistryObject<EventSpec>): VoidFunction;\n  on<K extends keyof EventSpec, E extends EventSpec[K]>(\n    arg0: K | EventRegistryObject<EventSpec>,\n    handler?: TEventCallback<E>\n  ): VoidFunction {\n    if (!this.__eventListeners) {\n      this.__eventListeners = {} as Record<keyof EventSpec, TEventCallback[]>;\n    }\n    if (typeof arg0 === 'object') {\n      // one object with key/value pairs was passed\n      Object.entries(arg0).forEach(([eventName, handler]) => {\n        this.on(eventName as K, handler as TEventCallback);\n      });\n      return () => this.off(arg0);\n    } else if (handler) {\n      const eventName = arg0;\n      if (!this.__eventListeners[eventName]) {\n        this.__eventListeners[eventName] = [];\n      }\n      this.__eventListeners[eventName].push(handler);\n      return () => this.off(eventName, handler);\n    } else {\n      // noop\n      return () => false;\n    }\n  }\n\n  /**\n   * Observes specified event **once**\n   * @alias once\n   * @param {string} eventName Event name (eg. 'after:render')\n   * @param {EventRegistryObject} handlers key/value pairs (eg. {'after:render': handler, 'selection:cleared': handler})\n   * @param {Function} handler Function that receives a notification when an event of the specified type occurs\n   * @return {Function} disposer\n   */\n  once<K extends keyof EventSpec, E extends EventSpec[K]>(\n    eventName: K,\n    handler: TEventCallback<E>\n  ): VoidFunction;\n  once(handlers: EventRegistryObject<EventSpec>): VoidFunction;\n  once<K extends keyof EventSpec, E extends EventSpec[K]>(\n    arg0: K | EventRegistryObject<EventSpec>,\n    handler?: TEventCallback<E>\n  ): VoidFunction {\n    if (typeof arg0 === 'object') {\n      // one object with key/value pairs was passed\n      const disposers: VoidFunction[] = [];\n      Object.entries(arg0).forEach(([eventName, handler]) => {\n        disposers.push(this.once(eventName as K, handler as TEventCallback));\n      });\n      return () => disposers.forEach((d) => d());\n    } else if (handler) {\n      const disposer = this.on<K, E>(\n        arg0,\n        function onceHandler(this: Observable<EventSpec>, ...args) {\n          handler.call(this, ...args);\n          disposer();\n        }\n      );\n      return disposer;\n    } else {\n      // noop\n      return () => false;\n    }\n  }\n\n  /**\n   * @private\n   * @param {string} eventName\n   * @param {Function} [handler]\n   */\n  private _removeEventListener<K extends keyof EventSpec>(\n    eventName: K,\n    handler?: TEventCallback\n  ) {\n    if (!this.__eventListeners[eventName]) {\n      return;\n    }\n\n    if (handler) {\n      const eventListener = this.__eventListeners[eventName];\n      const index = eventListener.indexOf(handler);\n      index > -1 && eventListener.splice(index, 1);\n    } else {\n      this.__eventListeners[eventName] = [];\n    }\n  }\n\n  /**\n   * unsubscribe an event listener\n   * @param {string} eventName event name (eg. 'after:render')\n   * @param {TEventCallback} handler event listener to unsubscribe\n   */\n  off<K extends keyof EventSpec>(eventName: K, handler: TEventCallback): void;\n  /**\n   * unsubscribe event listeners\n   * @param handlers handlers key/value pairs (eg. {'after:render': handler, 'selection:cleared': handler})\n   */\n  off(handlers: EventRegistryObject<EventSpec>): void;\n  /**\n   * unsubscribe all event listeners\n   */\n  off(): void;\n  off<K extends keyof EventSpec>(\n    arg0?: K | EventRegistryObject<EventSpec>,\n    handler?: TEventCallback\n  ) {\n    if (!this.__eventListeners) {\n      return;\n    }\n\n    // remove all key/value pairs (event name -> event handler)\n    if (typeof arg0 === 'undefined') {\n      for (const eventName in this.__eventListeners) {\n        this._removeEventListener(eventName);\n      }\n    }\n    // one object with key/value pairs was passed\n    else if (typeof arg0 === 'object') {\n      Object.entries(arg0).forEach(([eventName, handler]) => {\n        this._removeEventListener(eventName as K, handler as TEventCallback);\n      });\n    } else {\n      this._removeEventListener(arg0, handler);\n    }\n  }\n\n  /**\n   * Fires event with an optional options object\n   * @param {String} eventName Event name to fire\n   * @param {Object} [options] Options object\n   */\n  fire<K extends keyof EventSpec>(eventName: K, options?: EventSpec[K]) {\n    if (!this.__eventListeners) {\n      return;\n    }\n\n    const listenersForEvent = this.__eventListeners[eventName]?.concat();\n    if (listenersForEvent) {\n      for (let i = 0; i < listenersForEvent.length; i++) {\n        listenersForEvent[i].call(this, options || {});\n      }\n    }\n  }\n}\n","export const findIndexRight = <T>(\n  array: T[],\n  predicate: (value: T, index: number, array: T[]) => boolean\n) => {\n  for (let index = array.length - 1; index >= 0; index--) {\n    if (predicate(array[index], index, array)) {\n      return index;\n    }\n  }\n  return -1;\n};\n","/**\n * Returns random number between 2 specified ones.\n * @param {Number} min lower limit\n * @param {Number} max upper limit\n * @return {Number} random value (between min and max)\n */\nexport const getRandomInt = (min: number, max: number): number =>\n  Math.floor(Math.random() * (max - min + 1)) + min;\n","/**\n *\n * @param value value to check if NaN\n * @param [valueIfNaN]\n * @returns `fallback` is `value is NaN\n */\nexport const ifNaN = (value: number, valueIfNaN?: number) => {\n  return isNaN(value) && typeof valueIfNaN === 'number' ? valueIfNaN : value;\n};\n","/**\n * Removes value from an array.\n * Presence of value (and its position in an array) is determined via `Array.prototype.indexOf`\n * @param {Array} array\n * @param {*} value\n * @return {Array} original array\n */\nexport const removeFromArray = <T>(array: T[], value: T): T[] => {\n  const idx = array.indexOf(value);\n  if (idx !== -1) {\n    array.splice(idx, 1);\n  }\n  return array;\n};\n","import type { TRadian } from '../../typedefs';\nimport { halfPI } from '../../constants';\n\n/**\n * Calculate the cos of an angle, avoiding returning floats for known results\n * This function is here just to avoid getting 0.999999999999999 when dealing\n * with numbers that are really 1 or 0.\n * @param {TRadian} angle the angle\n * @return {Number} the cosin value for angle.\n */\nexport const cos = (angle: TRadian): number => {\n  if (angle === 0) {\n    return 1;\n  }\n  const angleSlice = Math.abs(angle) / halfPI;\n  switch (angleSlice) {\n    case 1:\n    case 3:\n      return 0;\n    case 2:\n      return -1;\n  }\n  return Math.cos(angle);\n};\n","import type { TRadian } from '../../typedefs';\nimport { halfPI } from '../../constants';\n\n/**\n * Calculate the cos of an angle, avoiding returning floats for known results\n * This function is here just to avoid getting 0.999999999999999 when dealing\n * with numbers that are really 1 or 0.\n * @param {TRadian} angle the angle\n * @return {Number} the sin value for angle.\n */\nexport const sin = (angle: TRadian): number => {\n  if (angle === 0) {\n    return 0;\n  }\n  const angleSlice = angle / halfPI;\n  const value = Math.sign(angle);\n  switch (angleSlice) {\n    case 1:\n      return value;\n    case 2:\n      return 0;\n    case 3:\n      return -value;\n  }\n  return Math.sin(angle);\n};\n","import type { TMat2D, TRadian } from './typedefs';\nimport { cos } from './util/misc/cos';\nimport { sin } from './util/misc/sin';\n\nexport interface XY {\n  x: number;\n  y: number;\n}\n\n/**\n * Adaptation of work of Kevin Lindsey(kevin@kevlindev.com)\n */\nexport class Point implements XY {\n  declare x: number;\n\n  declare y: number;\n\n  constructor();\n  constructor(x: number, y: number);\n  constructor(point?: XY);\n  constructor(arg0: number | XY = 0, y = 0) {\n    if (typeof arg0 === 'object') {\n      this.x = arg0.x;\n      this.y = arg0.y;\n    } else {\n      this.x = arg0;\n      this.y = y;\n    }\n  }\n\n  /**\n   * Adds another point to this one and returns another one\n   * @param {XY} that\n   * @return {Point} new Point instance with added values\n   */\n  add(that: XY): Point {\n    return new Point(this.x + that.x, this.y + that.y);\n  }\n\n  /**\n   * Adds another point to this one\n   * @param {XY} that\n   * @return {Point} thisArg\n   * @chainable\n   * @deprecated\n   */\n  addEquals(that: XY): Point {\n    this.x += that.x;\n    this.y += that.y;\n    return this;\n  }\n\n  /**\n   * Adds value to this point and returns a new one\n   * @param {Number} scalar\n   * @return {Point} new Point with added value\n   */\n  scalarAdd(scalar: number): Point {\n    return new Point(this.x + scalar, this.y + scalar);\n  }\n\n  /**\n   * Adds value to this point\n   * @param {Number} scalar\n   * @return {Point} thisArg\n   * @chainable\n   * @deprecated\n   */\n  scalarAddEquals(scalar: number): Point {\n    this.x += scalar;\n    this.y += scalar;\n    return this;\n  }\n\n  /**\n   * Subtracts another point from this point and returns a new one\n   * @param {XY} that\n   * @return {Point} new Point object with subtracted values\n   */\n  subtract(that: XY): Point {\n    return new Point(this.x - that.x, this.y - that.y);\n  }\n\n  /**\n   * Subtracts another point from this point\n   * @param {XY} that\n   * @return {Point} thisArg\n   * @chainable\n   * @deprecated\n   */\n  subtractEquals(that: XY): Point {\n    this.x -= that.x;\n    this.y -= that.y;\n    return this;\n  }\n\n  /**\n   * Subtracts value from this point and returns a new one\n   * @param {Number} scalar\n   * @return {Point}\n   */\n  scalarSubtract(scalar: number): Point {\n    return new Point(this.x - scalar, this.y - scalar);\n  }\n\n  /**\n   * Subtracts value from this point\n   * @param {Number} scalar\n   * @return {Point} thisArg\n   * @chainable\n   * @deprecated\n   */\n  scalarSubtractEquals(scalar: number): Point {\n    this.x -= scalar;\n    this.y -= scalar;\n    return this;\n  }\n\n  /**\n   * Multiplies this point by another value and returns a new one\n   * @param {XY} that\n   * @return {Point}\n   */\n  multiply(that: XY): Point {\n    return new Point(this.x * that.x, this.y * that.y);\n  }\n\n  /**\n   * Multiplies this point by a value and returns a new one\n   * @param {Number} scalar\n   * @return {Point}\n   */\n  scalarMultiply(scalar: number): Point {\n    return new Point(this.x * scalar, this.y * scalar);\n  }\n\n  /**\n   * Multiplies this point by a value\n   * @param {Number} scalar\n   * @return {Point} thisArg\n   * @chainable\n   * @deprecated\n   */\n  scalarMultiplyEquals(scalar: number): Point {\n    this.x *= scalar;\n    this.y *= scalar;\n    return this;\n  }\n\n  /**\n   * Divides this point by another and returns a new one\n   * @param {XY} that\n   * @return {Point}\n   */\n  divide(that: XY): Point {\n    return new Point(this.x / that.x, this.y / that.y);\n  }\n\n  /**\n   * Divides this point by a value and returns a new one\n   * @param {Number} scalar\n   * @return {Point}\n   */\n  scalarDivide(scalar: number): Point {\n    return new Point(this.x / scalar, this.y / scalar);\n  }\n\n  /**\n   * Divides this point by a value\n   * @param {Number} scalar\n   * @return {Point} thisArg\n   * @chainable\n   * @deprecated\n   */\n  scalarDivideEquals(scalar: number): Point {\n    this.x /= scalar;\n    this.y /= scalar;\n    return this;\n  }\n\n  /**\n   * Returns true if this point is equal to another one\n   * @param {XY} that\n   * @return {Boolean}\n   */\n  eq(that: XY): boolean {\n    return this.x === that.x && this.y === that.y;\n  }\n\n  /**\n   * Returns true if this point is less than another one\n   * @param {XY} that\n   * @return {Boolean}\n   */\n  lt(that: XY): boolean {\n    return this.x < that.x && this.y < that.y;\n  }\n\n  /**\n   * Returns true if this point is less than or equal to another one\n   * @param {XY} that\n   * @return {Boolean}\n   */\n  lte(that: XY): boolean {\n    return this.x <= that.x && this.y <= that.y;\n  }\n\n  /**\n\n   * Returns true if this point is greater another one\n   * @param {XY} that\n   * @return {Boolean}\n   */\n  gt(that: XY): boolean {\n    return this.x > that.x && this.y > that.y;\n  }\n\n  /**\n   * Returns true if this point is greater than or equal to another one\n   * @param {XY} that\n   * @return {Boolean}\n   */\n  gte(that: XY): boolean {\n    return this.x >= that.x && this.y >= that.y;\n  }\n\n  /**\n   * Returns new point which is the result of linear interpolation with this one and another one\n   * @param {XY} that\n   * @param {Number} t , position of interpolation, between 0 and 1 default 0.5\n   * @return {Point}\n   */\n  lerp(that: XY, t = 0.5): Point {\n    t = Math.max(Math.min(1, t), 0);\n    return new Point(\n      this.x + (that.x - this.x) * t,\n      this.y + (that.y - this.y) * t\n    );\n  }\n\n  /**\n   * Returns distance from this point and another one\n   * @param {XY} that\n   * @return {Number}\n   */\n  distanceFrom(that: XY): number {\n    const dx = this.x - that.x,\n      dy = this.y - that.y;\n    return Math.sqrt(dx * dx + dy * dy);\n  }\n\n  /**\n   * Returns the point between this point and another one\n   * @param {XY} that\n   * @return {Point}\n   */\n  midPointFrom(that: XY): Point {\n    return this.lerp(that);\n  }\n\n  /**\n   * Returns a new point which is the min of this and another one\n   * @param {XY} that\n   * @return {Point}\n   */\n  min(that: XY): Point {\n    return new Point(Math.min(this.x, that.x), Math.min(this.y, that.y));\n  }\n\n  /**\n   * Returns a new point which is the max of this and another one\n   * @param {XY} that\n   * @return {Point}\n   */\n  max(that: XY): Point {\n    return new Point(Math.max(this.x, that.x), Math.max(this.y, that.y));\n  }\n\n  /**\n   * Returns string representation of this point\n   * @return {String}\n   */\n  toString(): string {\n    return `${this.x},${this.y}`;\n  }\n\n  /**\n   * Sets x/y of this point\n   * @param {Number} x\n   * @param {Number} y\n   * @chainable\n   */\n  setXY(x: number, y: number) {\n    this.x = x;\n    this.y = y;\n    return this;\n  }\n\n  /**\n   * Sets x of this point\n   * @param {Number} x\n   * @chainable\n   */\n  setX(x: number) {\n    this.x = x;\n    return this;\n  }\n\n  /**\n   * Sets y of this point\n   * @param {Number} y\n   * @chainable\n   */\n  setY(y: number) {\n    this.y = y;\n    return this;\n  }\n\n  /**\n   * Sets x/y of this point from another point\n   * @param {XY} that\n   * @chainable\n   */\n  setFromPoint(that: XY) {\n    this.x = that.x;\n    this.y = that.y;\n    return this;\n  }\n\n  /**\n   * Swaps x/y of this point and another point\n   * @param {XY} that\n   */\n  swap(that: XY) {\n    const x = this.x,\n      y = this.y;\n    this.x = that.x;\n    this.y = that.y;\n    that.x = x;\n    that.y = y;\n  }\n\n  /**\n   * return a cloned instance of the point\n   * @return {Point}\n   */\n  clone(): Point {\n    return new Point(this.x, this.y);\n  }\n\n  /**\n   * Rotates `point` around `origin` with `radians`\n   * @static\n   * @memberOf fabric.util\n   * @param {XY} origin The origin of the rotation\n   * @param {TRadian} radians The radians of the angle for the rotation\n   * @return {Point} The new rotated point\n   */\n  rotate(radians: TRadian, origin: XY = ZERO): Point {\n    // TODO benchmark and verify the add and subtract how much cost\n    // and then in case early return if no origin is passed\n    const sinus = sin(radians),\n      cosinus = cos(radians);\n    const p = this.subtract(origin);\n    const rotated = new Point(\n      p.x * cosinus - p.y * sinus,\n      p.x * sinus + p.y * cosinus\n    );\n    return rotated.add(origin);\n  }\n\n  /**\n   * Apply transform t to point p\n   * @static\n   * @memberOf fabric.util\n   * @param  {TMat2D} t The transform\n   * @param  {Boolean} [ignoreOffset] Indicates that the offset should not be applied\n   * @return {Point} The transformed point\n   */\n  transform(t: TMat2D, ignoreOffset = false): Point {\n    return new Point(\n      t[0] * this.x + t[2] * this.y + (ignoreOffset ? 0 : t[4]),\n      t[1] * this.x + t[3] * this.y + (ignoreOffset ? 0 : t[5])\n    );\n  }\n}\n\nexport const ZERO = new Point(0, 0);\n","import type { Constructor, TBBox } from './typedefs';\nimport { removeFromArray } from './util/internals';\nimport { Point } from './Point';\nimport type { ActiveSelection } from './shapes/ActiveSelection';\nimport type { Group } from './shapes/Group';\nimport type { InteractiveFabricObject } from './shapes/Object/InteractiveObject';\nimport type { FabricObject } from './shapes/Object/FabricObject';\n\nexport const isCollection = (\n  fabricObject?: FabricObject\n): fabricObject is Group | ActiveSelection => {\n  return !!fabricObject && Array.isArray((fabricObject as Group)._objects);\n};\n\nexport function createCollectionMixin<TBase extends Constructor>(Base: TBase) {\n  class Collection extends Base {\n    /**\n     * @type {FabricObject[]}\n     * @TODO needs to end up in the constructor too\n     */\n    _objects: FabricObject[] = [];\n\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    _onObjectAdded(object: FabricObject) {\n      // subclasses should override this method\n    }\n\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    _onObjectRemoved(object: FabricObject) {\n      // subclasses should override this method\n    }\n\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    _onStackOrderChanged(object: FabricObject) {\n      // subclasses should override this method\n    }\n\n    /**\n     * Adds objects to collection\n     * Objects should be instances of (or inherit from) FabricObject\n     * @param {...FabricObject[]} objects to add\n     * @returns {number} new array length\n     */\n    add(...objects: FabricObject[]): number {\n      const size = this._objects.push(...objects);\n      objects.forEach((object) => this._onObjectAdded(object));\n      return size;\n    }\n\n    /**\n     * Inserts an object into collection at specified index\n     * @param {number} index Index to insert object at\n     * @param {...FabricObject[]} objects Object(s) to insert\n     * @returns {number} new array length\n     */\n    insertAt(index: number, ...objects: FabricObject[]) {\n      this._objects.splice(index, 0, ...objects);\n      objects.forEach((object) => this._onObjectAdded(object));\n      return this._objects.length;\n    }\n\n    /**\n     * Removes objects from a collection, then renders canvas (if `renderOnAddRemove` is not `false`)\n     * @private\n     * @param {...FabricObject[]} objects objects to remove\n     * @returns {FabricObject[]} removed objects\n     */\n    remove(...objects: FabricObject[]) {\n      const array = this._objects,\n        removed: FabricObject[] = [];\n      objects.forEach((object) => {\n        const index = array.indexOf(object);\n        // only call onObjectRemoved if an object was actually removed\n        if (index !== -1) {\n          array.splice(index, 1);\n          removed.push(object);\n          this._onObjectRemoved(object);\n        }\n      });\n      return removed;\n    }\n\n    /**\n     * Executes given function for each object in this group\n     * A simple shortcut for getObjects().forEach, before es6 was more complicated,\n     * now is just a shortcut.\n     * @param {Function} callback\n     *                   Callback invoked with current object as first argument,\n     *                   index - as second and an array of all objects - as third.\n     */\n    forEachObject(\n      callback: (\n        object: FabricObject,\n        index: number,\n        array: FabricObject[]\n      ) => any\n    ) {\n      this.getObjects().forEach((object, index, objects) =>\n        callback(object, index, objects)\n      );\n    }\n\n    /**\n     * Returns an array of children objects of this instance\n     * @param {...String} [types] When specified, only objects of these types are returned\n     * @return {Array}\n     */\n    getObjects(...types: string[]) {\n      if (types.length === 0) {\n        return [...this._objects];\n      }\n      return this._objects.filter((o) => o.isType(...types));\n    }\n\n    /**\n     * Returns object at specified index\n     * @param {Number} index\n     * @return {Object} object at index\n     */\n    item(index: number) {\n      return this._objects[index];\n    }\n\n    /**\n     * Returns true if collection contains no objects\n     * @return {Boolean} true if collection is empty\n     */\n    isEmpty() {\n      return this._objects.length === 0;\n    }\n\n    /**\n     * Returns a size of a collection (i.e: length of an array containing its objects)\n     * @return {Number} Collection size\n     */\n    size() {\n      return this._objects.length;\n    }\n\n    /**\n     * Returns true if collection contains an object.\\\n     * **Prefer using {@link FabricObject#isDescendantOf} for performance reasons**\n     * instead of `a.contains(b)` use `b.isDescendantOf(a)`\n     * @param {Object} object Object to check against\n     * @param {Boolean} [deep=false] `true` to check all descendants, `false` to check only `_objects`\n     * @return {Boolean} `true` if collection contains an object\n     */\n    contains(object: FabricObject, deep?: boolean): boolean {\n      if (this._objects.includes(object)) {\n        return true;\n      } else if (deep) {\n        return this._objects.some(\n          (obj) =>\n            obj instanceof Collection &&\n            (obj as unknown as Collection).contains(object, true)\n        );\n      }\n      return false;\n    }\n\n    /**\n     * Returns number representation of a collection complexity\n     * @return {Number} complexity\n     */\n    complexity() {\n      return this._objects.reduce((memo, current) => {\n        memo += current.complexity ? current.complexity() : 0;\n        return memo;\n      }, 0);\n    }\n\n    /**\n     * Moves an object or the objects of a multiple selection\n     * to the bottom of the stack of drawn objects\n     * @param {fabric.Object} object Object to send to back\n     * @returns {boolean} true if change occurred\n     */\n    sendObjectToBack(object: FabricObject) {\n      if (!object || object === this._objects[0]) {\n        return false;\n      }\n      removeFromArray(this._objects, object);\n      this._objects.unshift(object);\n      this._onStackOrderChanged(object);\n      return true;\n    }\n\n    /**\n     * Moves an object or the objects of a multiple selection\n     * to the top of the stack of drawn objects\n     * @param {fabric.Object} object Object to send\n     * @returns {boolean} true if change occurred\n     */\n    bringObjectToFront(object: FabricObject) {\n      if (!object || object === this._objects[this._objects.length - 1]) {\n        return false;\n      }\n      removeFromArray(this._objects, object);\n      this._objects.push(object);\n      this._onStackOrderChanged(object);\n      return true;\n    }\n\n    /**\n     * Moves an object or a selection down in stack of drawn objects\n     * An optional parameter, `intersecting` allows to move the object in behind\n     * the first intersecting object. Where intersection is calculated with\n     * bounding box. If no intersection is found, there will not be change in the\n     * stack.\n     * @param {fabric.Object} object Object to send\n     * @param {boolean} [intersecting] If `true`, send object behind next lower intersecting object\n     * @returns {boolean} true if change occurred\n     */\n    sendObjectBackwards(object: FabricObject, intersecting?: boolean) {\n      if (!object) {\n        return false;\n      }\n      const idx = this._objects.indexOf(object);\n      if (idx !== 0) {\n        // if object is not on the bottom of stack\n        const newIdx = this.findNewLowerIndex(object, idx, intersecting);\n        removeFromArray(this._objects, object);\n        this._objects.splice(newIdx, 0, object);\n        this._onStackOrderChanged(object);\n        return true;\n      }\n      return false;\n    }\n\n    /**\n     * Moves an object or a selection up in stack of drawn objects\n     * An optional parameter, intersecting allows to move the object in front\n     * of the first intersecting object. Where intersection is calculated with\n     * bounding box. If no intersection is found, there will not be change in the\n     * stack.\n     * @param {fabric.Object} object Object to send\n     * @param {boolean} [intersecting] If `true`, send object in front of next upper intersecting object\n     * @returns {boolean} true if change occurred\n     */\n    bringObjectForward(object: FabricObject, intersecting?: boolean) {\n      if (!object) {\n        return false;\n      }\n      const idx = this._objects.indexOf(object);\n      if (idx !== this._objects.length - 1) {\n        // if object is not on top of stack (last item in an array)\n        const newIdx = this.findNewUpperIndex(object, idx, intersecting);\n        removeFromArray(this._objects, object);\n        this._objects.splice(newIdx, 0, object);\n        this._onStackOrderChanged(object);\n        return true;\n      }\n      return false;\n    }\n\n    /**\n     * Moves an object to specified level in stack of drawn objects\n     * @param {fabric.Object} object Object to send\n     * @param {number} index Position to move to\n     * @returns {boolean} true if change occurred\n     */\n    moveObjectTo(object: FabricObject, index: number) {\n      if (object === this._objects[index]) {\n        return false;\n      }\n      removeFromArray(this._objects, object);\n      this._objects.splice(index, 0, object);\n      this._onStackOrderChanged(object);\n      return true;\n    }\n\n    findNewLowerIndex(\n      object: FabricObject,\n      idx: number,\n      intersecting?: boolean\n    ) {\n      let newIdx;\n\n      if (intersecting) {\n        newIdx = idx;\n        // traverse down the stack looking for the nearest intersecting object\n        for (let i = idx - 1; i >= 0; --i) {\n          if (object.isOverlapping(this._objects[i])) {\n            newIdx = i;\n            break;\n          }\n        }\n      } else {\n        newIdx = idx - 1;\n      }\n\n      return newIdx;\n    }\n\n    findNewUpperIndex(\n      object: FabricObject,\n      idx: number,\n      intersecting?: boolean\n    ) {\n      let newIdx;\n\n      if (intersecting) {\n        newIdx = idx;\n        // traverse up the stack looking for the nearest intersecting object\n        for (let i = idx + 1; i < this._objects.length; ++i) {\n          if (object.isOverlapping(this._objects[i])) {\n            newIdx = i;\n            break;\n          }\n        }\n      } else {\n        newIdx = idx + 1;\n      }\n\n      return newIdx;\n    }\n\n    /**\n     * Given a bounding box, return all the objects of the collection that are contained in the bounding box.\n     * If `includeIntersecting` is true, return also the objects that intersect the bounding box as well.\n     * This is meant to work with selection. Is not a generic method.\n     * @param {TBBox} bbox a bounding box in scene coordinates\n     * @param {{ includeIntersecting?: boolean }} options an object with includeIntersecting\n     * @returns array of objects contained in the bounding box, ordered from top to bottom stacking wise\n     */\n    collectObjects(\n      { left, top, width, height }: TBBox,\n      { includeIntersecting = true }: { includeIntersecting?: boolean } = {}\n    ) {\n      const objects: InteractiveFabricObject[] = [],\n        tl = new Point(left, top),\n        br = tl.add(new Point(width, height));\n\n      // we iterate reverse order to collect top first in case of click.\n      for (let i = this._objects.length - 1; i >= 0; i--) {\n        const object = this._objects[i] as unknown as InteractiveFabricObject;\n        if (\n          object.selectable &&\n          object.visible &&\n          ((includeIntersecting && object.intersectsWithRect(tl, br)) ||\n            object.isContainedWithinRect(tl, br) ||\n            (includeIntersecting && object.containsPoint(tl)) ||\n            (includeIntersecting && object.containsPoint(br)))\n        ) {\n          objects.push(object);\n        }\n      }\n\n      return objects;\n    }\n  }\n\n  // https://github.com/microsoft/TypeScript/issues/32080\n  return Collection as typeof Collection & TBase;\n}\n","import { Observable } from './Observable';\n\nexport class CommonMethods<EventSpec> extends Observable<EventSpec> {\n  /**\n   * Sets object's properties from options, for initialization only\n   * @protected\n   * @param {Object} [options] Options object\n   */\n  protected _setOptions(options: any = {}) {\n    for (const prop in options) {\n      this.set(prop, options[prop]);\n    }\n  }\n\n  /**\n   * @private\n   */\n  _setObject(obj: Record<string, any>) {\n    for (const prop in obj) {\n      this._set(prop, obj[prop]);\n    }\n  }\n\n  /**\n   * Sets property to a given value. When changing position/dimension -related properties (left, top, scale, angle, etc.) `set` does not update position of object's borders/controls. If you need to update those, call `setCoords()`.\n   * @param {String|Object} key Property name or object (if object, iterate over the object properties)\n   * @param {Object|Function} value Property value (if function, the value is passed into it and its return value is used as a new one)\n   */\n  set(key: string | Record<string, any>, value?: any) {\n    if (typeof key === 'object') {\n      this._setObject(key);\n    } else {\n      this._set(key, value);\n    }\n    return this;\n  }\n\n  _set(key: string, value: any) {\n    this[key as keyof this] = value;\n  }\n\n  /**\n   * Toggles specified property from `true` to `false` or from `false` to `true`\n   * @param {String} property Property to toggle\n   */\n  toggle(property: string) {\n    const value = this.get(property);\n    if (typeof value === 'boolean') {\n      this.set(property, !value);\n    }\n    return this;\n  }\n\n  /**\n   * Basic getter\n   * @param {String} property Property name\n   * @return {*} value of a property\n   */\n  get(property: string): any {\n    return this[property as keyof this];\n  }\n}\n","import { getFabricWindow } from '../../env';\n\nexport function requestAnimFrame(callback: FrameRequestCallback): number {\n  return getFabricWindow().requestAnimationFrame(callback);\n}\n\nexport function cancelAnimFrame(handle: number): void {\n  return getFabricWindow().cancelAnimationFrame(handle);\n}\n","let id = 0;\n\nexport const uid = () => id++;\n","import { getFabricDocument } from '../../env';\nimport type { ImageFormat } from '../../typedefs';\nimport { FabricError } from '../internals/console';\n/**\n * Creates canvas element\n * @return {CanvasElement} initialized canvas element\n */\nexport const createCanvasElement = (): HTMLCanvasElement => {\n  const element = getFabricDocument().createElement('canvas');\n  if (!element || typeof element.getContext === 'undefined') {\n    throw new FabricError('Failed to create `canvas` element');\n  }\n  return element;\n};\n\n/**\n * Creates image element (works on client and node)\n * @return {HTMLImageElement} HTML image element\n */\nexport const createImage = (): HTMLImageElement =>\n  getFabricDocument().createElement('img');\n\n/**\n * Creates a canvas element that is a copy of another and is also painted\n * @param {CanvasElement} canvas to copy size and content of\n * @return {CanvasElement} initialized canvas element\n */\nexport const copyCanvasElement = (\n  canvas: HTMLCanvasElement\n): HTMLCanvasElement => {\n  const newCanvas = createCanvasElement();\n  newCanvas.width = canvas.width;\n  newCanvas.height = canvas.height;\n  newCanvas.getContext('2d')?.drawImage(canvas, 0, 0);\n  return newCanvas;\n};\n\n/**\n * since 2.6.0 moved from canvas instance to utility.\n * possibly useless\n * @param {CanvasElement} canvasEl to copy size and content of\n * @param {String} format 'jpeg' or 'png', in some browsers 'webp' is ok too\n * @param {Number} quality <= 1 and > 0\n * @return {String} data url\n */\nexport const toDataURL = (\n  canvasEl: HTMLCanvasElement,\n  format: ImageFormat,\n  quality: number\n) => canvasEl.toDataURL(`image/${format}`, quality);\n\nexport const isHTMLCanvas = (\n  canvas?: HTMLCanvasElement | string\n): canvas is HTMLCanvasElement => {\n  return !!canvas && (canvas as HTMLCanvasElement).getContext !== undefined;\n};\n","import type { TRadian, TDegree } from '../../typedefs';\nimport { PiBy180 } from '../../constants';\n\n/**\n * Transforms degrees to radians.\n * @param {TDegree} degrees value in degrees\n * @return {TRadian} value in radians\n */\nexport const degreesToRadians = (degrees: TDegree): TRadian =>\n  (degrees * PiBy180) as TRadian;\n\n/**\n * Transforms radians to degrees.\n * @param {TRadian} radians value in radians\n * @return {TDegree} value in degrees\n */\nexport const radiansToDegrees = (radians: TRadian): TDegree =>\n  (radians / PiBy180) as TDegree;\n","import { iMatrix } from '../../constants';\nimport type { XY } from '../../Point';\nimport { Point } from '../../Point';\nimport type { TDegree, TRadian, TMat2D } from '../../typedefs';\nimport { cos } from './cos';\nimport { degreesToRadians, radiansToDegrees } from './radiansDegreesConversion';\nimport { sin } from './sin';\n\nexport type TRotateMatrixArgs = {\n  angle?: TDegree;\n};\n\nexport type TTranslateMatrixArgs = {\n  translateX?: number;\n  translateY?: number;\n};\n\nexport type TScaleMatrixArgs = {\n  scaleX?: number;\n  scaleY?: number;\n  flipX?: boolean;\n  flipY?: boolean;\n  skewX?: TDegree;\n  skewY?: TDegree;\n};\n\nexport type TComposeMatrixArgs = TTranslateMatrixArgs &\n  TRotateMatrixArgs &\n  TScaleMatrixArgs;\n\nexport type TQrDecomposeOut = Required<\n  Omit<TComposeMatrixArgs, 'flipX' | 'flipY'>\n>;\n\nexport const isIdentityMatrix = (mat: TMat2D) =>\n  mat.every((value, index) => value === iMatrix[index]);\n\n/**\n * Apply transform t to point p\n * @deprecated use {@link Point#transform}\n * @param  {Point | XY} p The point to transform\n * @param  {Array} t The transform\n * @param  {Boolean} [ignoreOffset] Indicates that the offset should not be applied\n * @return {Point} The transformed point\n */\nexport const transformPoint = (\n  p: XY,\n  t: TMat2D,\n  ignoreOffset?: boolean\n): Point => new Point(p).transform(t, ignoreOffset);\n\n/**\n * Invert transformation t\n * @param {Array} t The transform\n * @return {Array} The inverted transform\n */\nexport const invertTransform = (t: TMat2D): TMat2D => {\n  const a = 1 / (t[0] * t[3] - t[1] * t[2]),\n    r = [a * t[3], -a * t[1], -a * t[2], a * t[0], 0, 0] as TMat2D,\n    { x, y } = new Point(t[4], t[5]).transform(r, true);\n  r[4] = -x;\n  r[5] = -y;\n  return r;\n};\n\n/**\n * Multiply matrix A by matrix B to nest transformations\n * @param  {TMat2D} a First transformMatrix\n * @param  {TMat2D} b Second transformMatrix\n * @param  {Boolean} is2x2 flag to multiply matrices as 2x2 matrices\n * @return {TMat2D} The product of the two transform matrices\n */\nexport const multiplyTransformMatrices = (\n  a: TMat2D,\n  b: TMat2D,\n  is2x2?: boolean\n): TMat2D =>\n  [\n    a[0] * b[0] + a[2] * b[1],\n    a[1] * b[0] + a[3] * b[1],\n    a[0] * b[2] + a[2] * b[3],\n    a[1] * b[2] + a[3] * b[3],\n    is2x2 ? 0 : a[0] * b[4] + a[2] * b[5] + a[4],\n    is2x2 ? 0 : a[1] * b[4] + a[3] * b[5] + a[5],\n  ] as TMat2D;\n\n/**\n * Multiplies {@link matrices} such that a matrix defines the plane for the rest of the matrices **after** it\n *\n * `multiplyTransformMatrixArray([A, B, C, D])` is equivalent to `A(B(C(D)))`\n *\n * @param matrices an array of matrices\n * @param [is2x2] flag to multiply matrices as 2x2 matrices\n * @returns the multiplication product\n */\nexport const multiplyTransformMatrixArray = (\n  matrices: (TMat2D | undefined | null | false)[],\n  is2x2?: boolean\n) =>\n  matrices.reduceRight(\n    (product: TMat2D, curr) =>\n      curr && product\n        ? multiplyTransformMatrices(curr, product, is2x2)\n        : curr || product,\n    undefined as unknown as TMat2D\n  ) || iMatrix.concat();\n\nexport const calcPlaneRotation = ([a, b]: TMat2D) =>\n  Math.atan2(b, a) as TRadian;\n\n/**\n * Decomposes standard 2x3 matrix into transform components\n * @param  {TMat2D} a transformMatrix\n * @return {Object} Components of transform\n */\nexport const qrDecompose = (a: TMat2D): TQrDecomposeOut => {\n  const angle = calcPlaneRotation(a),\n    denom = Math.pow(a[0], 2) + Math.pow(a[1], 2),\n    scaleX = Math.sqrt(denom),\n    scaleY = (a[0] * a[3] - a[2] * a[1]) / scaleX,\n    skewX = Math.atan2(a[0] * a[2] + a[1] * a[3], denom);\n  return {\n    angle: radiansToDegrees(angle),\n    scaleX,\n    scaleY,\n    skewX: radiansToDegrees(skewX),\n    skewY: 0 as TDegree,\n    translateX: a[4] || 0,\n    translateY: a[5] || 0,\n  };\n};\n\n/**\n * Generate a translation matrix\n *\n * A translation matrix in the form of\n * [ 1 0 x ]\n * [ 0 1 y ]\n * [ 0 0 1 ]\n *\n * See @link https://developer.mozilla.org/en-US/docs/Web/SVG/Attribute/transform#translate for more details\n *\n * @param {number} x translation on X axis\n * @param {number} [y] translation on Y axis\n * @returns {TMat2D} matrix\n */\nexport const createTranslateMatrix = (x: number, y = 0): TMat2D => [\n  1,\n  0,\n  0,\n  1,\n  x,\n  y,\n];\n\n/**\n * Generate a rotation matrix around around a point (x,y), defaulting to (0,0)\n *\n * A matrix in the form of\n * [cos(a) -sin(a) -x*cos(a)+y*sin(a)+x]\n * [sin(a)  cos(a) -x*sin(a)-y*cos(a)+y]\n * [0       0      1                 ]\n *\n *\n * @param {TDegree} angle rotation in degrees\n * @param {XY} [pivotPoint] pivot point to rotate around\n * @returns {TMat2D} matrix\n */\nexport function createRotateMatrix(\n  { angle = 0 }: TRotateMatrixArgs = {},\n  { x = 0, y = 0 }: Partial<XY> = {}\n): TMat2D {\n  const angleRadiant = degreesToRadians(angle),\n    cosValue = cos(angleRadiant),\n    sinValue = sin(angleRadiant);\n  return [\n    cosValue,\n    sinValue,\n    -sinValue,\n    cosValue,\n    x ? x - (cosValue * x - sinValue * y) : 0,\n    y ? y - (sinValue * x + cosValue * y) : 0,\n  ];\n}\n\n/**\n * Generate a scale matrix around the point (0,0)\n *\n * A matrix in the form of\n * [x 0 0]\n * [0 y 0]\n * [0 0 1]\n *\n * @link https://developer.mozilla.org/en-US/docs/Web/SVG/Attribute/transform#scale\n *\n * @param {number} x scale on X axis\n * @param {number} [y] scale on Y axis\n * @returns {TMat2D} matrix\n */\nexport const createScaleMatrix = (x: number, y: number = x): TMat2D => [\n  x,\n  0,\n  0,\n  y,\n  0,\n  0,\n];\n\nexport const angleToSkew = (angle: TDegree) =>\n  Math.tan(degreesToRadians(angle));\n\nexport const skewToAngle = (value: TRadian) =>\n  radiansToDegrees(Math.atan(value));\n\n/**\n * Generate a skew matrix for the X axis\n *\n * A matrix in the form of\n * [1 x 0]\n * [0 1 0]\n * [0 0 1]\n *\n * @link https://developer.mozilla.org/en-US/docs/Web/SVG/Attribute/transform#skewx\n *\n * @param {TDegree} skewValue translation on X axis\n * @returns {TMat2D} matrix\n */\nexport const createSkewXMatrix = (skewValue: TDegree): TMat2D => [\n  1,\n  0,\n  angleToSkew(skewValue),\n  1,\n  0,\n  0,\n];\n\n/**\n * Generate a skew matrix for the Y axis\n *\n * A matrix in the form of\n * [1 0 0]\n * [y 1 0]\n * [0 0 1]\n *\n * @link https://developer.mozilla.org/en-US/docs/Web/SVG/Attribute/transform#skewy\n *\n * @param {TDegree} skewValue translation on Y axis\n * @returns {TMat2D} matrix\n */\nexport const createSkewYMatrix = (skewValue: TDegree): TMat2D => [\n  1,\n  angleToSkew(skewValue),\n  0,\n  1,\n  0,\n  0,\n];\n\n/**\n * Returns a transform matrix starting from an object of the same kind of\n * the one returned from qrDecompose, useful also if you want to calculate some\n * transformations from an object that is not enlived yet.\n * is called DimensionsTransformMatrix because those properties are the one that influence\n * the size of the resulting box of the object.\n * @param  {Object} options\n * @param  {Number} [options.scaleX]\n * @param  {Number} [options.scaleY]\n * @param  {Boolean} [options.flipX]\n * @param  {Boolean} [options.flipY]\n * @param  {Number} [options.skewX]\n * @param  {Number} [options.skewY]\n * @return {Number[]} transform matrix\n */\nexport const calcDimensionsMatrix = ({\n  scaleX = 1,\n  scaleY = 1,\n  flipX = false,\n  flipY = false,\n  skewX = 0 as TDegree,\n  skewY = 0 as TDegree,\n}: TScaleMatrixArgs) => {\n  let matrix = createScaleMatrix(\n    flipX ? -scaleX : scaleX,\n    flipY ? -scaleY : scaleY\n  );\n  if (skewX) {\n    matrix = multiplyTransformMatrices(matrix, createSkewXMatrix(skewX), true);\n  }\n  if (skewY) {\n    matrix = multiplyTransformMatrices(matrix, createSkewYMatrix(skewY), true);\n  }\n  return matrix;\n};\n\n/**\n * Returns a transform matrix starting from an object of the same kind of\n * the one returned from qrDecompose, useful also if you want to calculate some\n * transformations from an object that is not enlived yet\n * Before changing this function look at: src/benchmarks/calcTransformMatrix.mjs\n * @param  {Object} options\n * @param  {Number} [options.angle]\n * @param  {Number} [options.scaleX]\n * @param  {Number} [options.scaleY]\n * @param  {Boolean} [options.flipX]\n * @param  {Boolean} [options.flipY]\n * @param  {Number} [options.skewX]\n * @param  {Number} [options.skewY]\n * @param  {Number} [options.translateX]\n * @param  {Number} [options.translateY]\n * @return {Number[]} transform matrix\n */\nexport const composeMatrix = (options: TComposeMatrixArgs): TMat2D => {\n  const { translateX = 0, translateY = 0, angle = 0 as TDegree } = options;\n  let matrix = createTranslateMatrix(translateX, translateY);\n  if (angle) {\n    matrix = multiplyTransformMatrices(matrix, createRotateMatrix({ angle }));\n  }\n  const scaleMatrix = calcDimensionsMatrix(options);\n  if (!isIdentityMatrix(scaleMatrix)) {\n    matrix = multiplyTransformMatrices(matrix, scaleMatrix);\n  }\n  return matrix;\n};\n","import { noop } from '../../constants';\nimport type { Pattern } from '../../Pattern';\nimport type { FabricObject } from '../../shapes/Object/FabricObject';\nimport type {\n  Abortable,\n  Constructor,\n  TCrossOrigin,\n  TFiller,\n} from '../../typedefs';\nimport { createImage } from './dom';\nimport { classRegistry } from '../../ClassRegistry';\nimport type { BaseFilter } from '../../filters/BaseFilter';\nimport type { FabricObject as BaseFabricObject } from '../../shapes/Object/Object';\nimport { FabricError, SignalAbortedError } from '../internals/console';\nimport type { Shadow } from '../../Shadow';\n\nexport type LoadImageOptions = Abortable & {\n  /**\n   * cors value for the image loading, default to anonymous\n   */\n  crossOrigin?: TCrossOrigin;\n};\n\n/**\n * Loads image element from given url and resolve it, or catch.\n * @param {String} url URL representing an image\n * @param {LoadImageOptions} [options] image loading options\n * @returns {Promise<HTMLImageElement>} the loaded image.\n */\nexport const loadImage = (\n  url: string,\n  { signal, crossOrigin = null }: LoadImageOptions = {}\n) =>\n  new Promise<HTMLImageElement>(function (resolve, reject) {\n    if (signal && signal.aborted) {\n      return reject(new SignalAbortedError('loadImage'));\n    }\n    const img = createImage();\n    let abort: EventListenerOrEventListenerObject;\n    if (signal) {\n      abort = function (err: Event) {\n        img.src = '';\n        reject(err);\n      };\n      signal.addEventListener('abort', abort, { once: true });\n    }\n    const done = function () {\n      img.onload = img.onerror = null;\n      abort && signal?.removeEventListener('abort', abort);\n      resolve(img);\n    };\n    if (!url) {\n      done();\n      return;\n    }\n    img.onload = done;\n    img.onerror = function () {\n      abort && signal?.removeEventListener('abort', abort);\n      reject(new FabricError(`Error loading ${img.src}`));\n    };\n    crossOrigin && (img.crossOrigin = crossOrigin);\n    img.src = url;\n  });\n\nexport type EnlivenObjectOptions = Abortable & {\n  /**\n   * Method for further parsing of object elements,\n   * called after each fabric object created.\n   */\n  reviver?: <\n    T extends BaseFabricObject | FabricObject | BaseFilter | Shadow | TFiller\n  >(\n    serializedObj: Record<string, any>,\n    instance: T\n  ) => void;\n};\n\n/**\n * Creates corresponding fabric instances from their object representations\n * @param {Object[]} objects Objects to enliven\n * @param {EnlivenObjectOptions} [options]\n * @param {(serializedObj: object, instance: FabricObject) => any} [options.reviver] Method for further parsing of object elements,\n * called after each fabric object created.\n * @param {AbortSignal} [options.signal] handle aborting, see https://developer.mozilla.org/en-US/docs/Web/API/AbortController/signal\n * @returns {Promise<FabricObject[]>}\n */\nexport const enlivenObjects = <\n  T extends BaseFabricObject | FabricObject | BaseFilter | Shadow | TFiller\n>(\n  objects: any[],\n  { signal, reviver = noop }: EnlivenObjectOptions = {}\n) =>\n  new Promise<T[]>((resolve, reject) => {\n    const instances: T[] = [];\n    signal && signal.addEventListener('abort', reject, { once: true });\n    Promise.all(\n      objects.map((obj) =>\n        classRegistry\n          .getClass<\n            Constructor<T> & {\n              fromObject(options: any, context: Abortable): Promise<T>;\n            }\n          >(obj.type)\n          .fromObject(obj, { signal })\n          .then((fabricInstance) => {\n            reviver(obj, fabricInstance);\n            instances.push(fabricInstance);\n            return fabricInstance;\n          })\n      )\n    )\n      .then(resolve)\n      .catch((error) => {\n        // cleanup\n        instances.forEach((instance) => {\n          (instance as FabricObject).dispose &&\n            (instance as FabricObject).dispose();\n        });\n        reject(error);\n      })\n      .finally(() => {\n        signal && signal.removeEventListener('abort', reject);\n      });\n  });\n\n/**\n * Creates corresponding fabric instances residing in an object, e.g. `clipPath`\n * @param {Object} object with properties to enlive ( fill, stroke, clipPath, path )\n * @param {object} [options]\n * @param {AbortSignal} [options.signal] handle aborting, see https://developer.mozilla.org/en-US/docs/Web/API/AbortController/signal\n * @returns {Promise<Record<string, FabricObject | TFiller | null>>} the input object with enlived values\n */\nexport const enlivenObjectEnlivables = <\n  R = Record<string, FabricObject | TFiller | null>\n>(\n  serializedObject: any,\n  { signal }: Abortable = {}\n) =>\n  new Promise<R>((resolve, reject) => {\n    const instances: (FabricObject | TFiller | Shadow)[] = [];\n    signal && signal.addEventListener('abort', reject, { once: true });\n    // enlive every possible property\n    const promises = Object.values(serializedObject).map((value: any) => {\n      if (!value) {\n        return value;\n      }\n      // clipPath or shadow or gradient\n      if (value.type) {\n        return enlivenObjects<FabricObject | Shadow | TFiller>([value], {\n          signal,\n        }).then(([enlived]) => {\n          instances.push(enlived);\n          return enlived;\n        });\n      }\n      // pattern\n      if (value.source) {\n        return classRegistry\n          .getClass<typeof Pattern>('pattern')\n          .fromObject(value, { signal })\n          .then((pattern: Pattern) => {\n            instances.push(pattern);\n            return pattern;\n          });\n      }\n      return value;\n    });\n    const keys = Object.keys(serializedObject);\n    Promise.all(promises)\n      .then((enlived) => {\n        return enlived.reduce((acc, instance, index) => {\n          acc[keys[index]] = instance;\n          return acc;\n        }, {});\n      })\n      .then(resolve)\n      .catch((error) => {\n        // cleanup\n        instances.forEach((instance: any) => {\n          instance.dispose && instance.dispose();\n        });\n        reject(error);\n      })\n      .finally(() => {\n        signal && signal.removeEventListener('abort', reject);\n      });\n  });\n","/**\n * Populates an object with properties of another object\n * @param {Object} source Source object\n * @param {string[]} properties Properties names to include\n * @returns object populated with the picked keys\n */\nexport const pick = <T extends Record<string, any>>(\n  source: T,\n  keys: (keyof T)[] = []\n) => {\n  return keys.reduce((o, key) => {\n    if (key in source) {\n      o[key] = source[key];\n    }\n    return o;\n  }, {} as Partial<T>);\n};\n\nexport const pickBy = <T extends Record<string, any>>(\n  source: T,\n  predicate: <K extends keyof T>(value: T[K], key: K, collection: T) => boolean\n) => {\n  return (Object.keys(source) as (keyof T)[]).reduce((o, key) => {\n    if (predicate(source[key], key, source)) {\n      o[key] = source[key];\n    }\n    return o;\n  }, {} as Partial<T>);\n};\n","/**\n * Map of the 148 color names with HEX code\n * @see: https://www.w3.org/TR/css3-color/#svg-color\n */\nexport const ColorNameMap = {\n  aliceblue: '#F0F8FF',\n  antiquewhite: '#FAEBD7',\n  aqua: '#0FF',\n  aquamarine: '#7FFFD4',\n  azure: '#F0FFFF',\n  beige: '#F5F5DC',\n  bisque: '#FFE4C4',\n  black: '#000',\n  blanchedalmond: '#FFEBCD',\n  blue: '#00F',\n  blueviolet: '#8A2BE2',\n  brown: '#A52A2A',\n  burlywood: '#DEB887',\n  cadetblue: '#5F9EA0',\n  chartreuse: '#7FFF00',\n  chocolate: '#D2691E',\n  coral: '#FF7F50',\n  cornflowerblue: '#6495ED',\n  cornsilk: '#FFF8DC',\n  crimson: '#DC143C',\n  cyan: '#0FF',\n  darkblue: '#00008B',\n  darkcyan: '#008B8B',\n  darkgoldenrod: '#B8860B',\n  darkgray: '#A9A9A9',\n  darkgrey: '#A9A9A9',\n  darkgreen: '#006400',\n  darkkhaki: '#BDB76B',\n  darkmagenta: '#8B008B',\n  darkolivegreen: '#556B2F',\n  darkorange: '#FF8C00',\n  darkorchid: '#9932CC',\n  darkred: '#8B0000',\n  darksalmon: '#E9967A',\n  darkseagreen: '#8FBC8F',\n  darkslateblue: '#483D8B',\n  darkslategray: '#2F4F4F',\n  darkslategrey: '#2F4F4F',\n  darkturquoise: '#00CED1',\n  darkviolet: '#9400D3',\n  deeppink: '#FF1493',\n  deepskyblue: '#00BFFF',\n  dimgray: '#696969',\n  dimgrey: '#696969',\n  dodgerblue: '#1E90FF',\n  firebrick: '#B22222',\n  floralwhite: '#FFFAF0',\n  forestgreen: '#228B22',\n  fuchsia: '#F0F',\n  gainsboro: '#DCDCDC',\n  ghostwhite: '#F8F8FF',\n  gold: '#FFD700',\n  goldenrod: '#DAA520',\n  gray: '#808080',\n  grey: '#808080',\n  green: '#008000',\n  greenyellow: '#ADFF2F',\n  honeydew: '#F0FFF0',\n  hotpink: '#FF69B4',\n  indianred: '#CD5C5C',\n  indigo: '#4B0082',\n  ivory: '#FFFFF0',\n  khaki: '#F0E68C',\n  lavender: '#E6E6FA',\n  lavenderblush: '#FFF0F5',\n  lawngreen: '#7CFC00',\n  lemonchiffon: '#FFFACD',\n  lightblue: '#ADD8E6',\n  lightcoral: '#F08080',\n  lightcyan: '#E0FFFF',\n  lightgoldenrodyellow: '#FAFAD2',\n  lightgray: '#D3D3D3',\n  lightgrey: '#D3D3D3',\n  lightgreen: '#90EE90',\n  lightpink: '#FFB6C1',\n  lightsalmon: '#FFA07A',\n  lightseagreen: '#20B2AA',\n  lightskyblue: '#87CEFA',\n  lightslategray: '#789',\n  lightslategrey: '#789',\n  lightsteelblue: '#B0C4DE',\n  lightyellow: '#FFFFE0',\n  lime: '#0F0',\n  limegreen: '#32CD32',\n  linen: '#FAF0E6',\n  magenta: '#F0F',\n  maroon: '#800000',\n  mediumaquamarine: '#66CDAA',\n  mediumblue: '#0000CD',\n  mediumorchid: '#BA55D3',\n  mediumpurple: '#9370DB',\n  mediumseagreen: '#3CB371',\n  mediumslateblue: '#7B68EE',\n  mediumspringgreen: '#00FA9A',\n  mediumturquoise: '#48D1CC',\n  mediumvioletred: '#C71585',\n  midnightblue: '#191970',\n  mintcream: '#F5FFFA',\n  mistyrose: '#FFE4E1',\n  moccasin: '#FFE4B5',\n  navajowhite: '#FFDEAD',\n  navy: '#000080',\n  oldlace: '#FDF5E6',\n  olive: '#808000',\n  olivedrab: '#6B8E23',\n  orange: '#FFA500',\n  orangered: '#FF4500',\n  orchid: '#DA70D6',\n  palegoldenrod: '#EEE8AA',\n  palegreen: '#98FB98',\n  paleturquoise: '#AFEEEE',\n  palevioletred: '#DB7093',\n  papayawhip: '#FFEFD5',\n  peachpuff: '#FFDAB9',\n  peru: '#CD853F',\n  pink: '#FFC0CB',\n  plum: '#DDA0DD',\n  powderblue: '#B0E0E6',\n  purple: '#800080',\n  rebeccapurple: '#639',\n  red: '#F00',\n  rosybrown: '#BC8F8F',\n  royalblue: '#4169E1',\n  saddlebrown: '#8B4513',\n  salmon: '#FA8072',\n  sandybrown: '#F4A460',\n  seagreen: '#2E8B57',\n  seashell: '#FFF5EE',\n  sienna: '#A0522D',\n  silver: '#C0C0C0',\n  skyblue: '#87CEEB',\n  slateblue: '#6A5ACD',\n  slategray: '#708090',\n  slategrey: '#708090',\n  snow: '#FFFAFA',\n  springgreen: '#00FF7F',\n  steelblue: '#4682B4',\n  tan: '#D2B48C',\n  teal: '#008080',\n  thistle: '#D8BFD8',\n  tomato: '#FF6347',\n  turquoise: '#40E0D0',\n  violet: '#EE82EE',\n  wheat: '#F5DEB3',\n  white: '#FFF',\n  whitesmoke: '#F5F5F5',\n  yellow: '#FF0',\n  yellowgreen: '#9ACD32',\n};\n","import type { TRGBAColorSource } from './typedefs';\n\n/**\n * @param {Number} p\n * @param {Number} q\n * @param {Number} t\n * @return {Number}\n */\nexport const hue2rgb = (p: number, q: number, t: number): number => {\n  if (t < 0) {\n    t += 1;\n  }\n  if (t > 1) {\n    t -= 1;\n  }\n  if (t < 1 / 6) {\n    return p + (q - p) * 6 * t;\n  }\n  if (t < 1 / 2) {\n    return q;\n  }\n  if (t < 2 / 3) {\n    return p + (q - p) * (2 / 3 - t) * 6;\n  }\n  return p;\n};\n\n/**\n * Adapted from {@link https://gist.github.com/mjackson/5311256 https://gist.github.com/mjackson}\n * @param {Number} r Red color value\n * @param {Number} g Green color value\n * @param {Number} b Blue color value\n * @param {Number} a Alpha color value pass through\n * @return {TRGBColorSource} Hsl color\n */\nexport const rgb2Hsl = (\n  r: number,\n  g: number,\n  b: number,\n  a: number\n): TRGBAColorSource => {\n  r /= 255;\n  g /= 255;\n  b /= 255;\n  const maxValue = Math.max(r, g, b),\n    minValue = Math.min(r, g, b);\n\n  let h!: number, s: number;\n  const l = (maxValue + minValue) / 2;\n\n  if (maxValue === minValue) {\n    h = s = 0; // achromatic\n  } else {\n    const d = maxValue - minValue;\n    s = l > 0.5 ? d / (2 - maxValue - minValue) : d / (maxValue + minValue);\n    switch (maxValue) {\n      case r:\n        h = (g - b) / d + (g < b ? 6 : 0);\n        break;\n      case g:\n        h = (b - r) / d + 2;\n        break;\n      case b:\n        h = (r - g) / d + 4;\n        break;\n    }\n    h /= 6;\n  }\n\n  return [Math.round(h * 360), Math.round(s * 100), Math.round(l * 100), a];\n};\n\nexport const fromAlphaToFloat = (value = '1') =>\n  parseFloat(value) / (value.endsWith('%') ? 100 : 1);\n\n/**\n * Convert a value in the inclusive range [0, 255] to hex\n */\nexport const hexify = (value: number) =>\n  Math.min(Math.round(value), 255).toString(16).toUpperCase().padStart(2, '0');\n\n/**\n * Calculate the grey average value for rgb and pass through alpha\n */\nexport const greyAverage = ([\n  r,\n  g,\n  b,\n  a = 1,\n]: TRGBAColorSource): TRGBAColorSource => {\n  const avg = Math.round(r * 0.3 + g * 0.59 + b * 0.11);\n  return [avg, avg, avg, a];\n};\n","import { ColorNameMap } from './color_map';\nimport { reHSLa, reHex, reRGBa } from './constants';\nimport type { TRGBAColorSource, TColorArg } from './typedefs';\nimport {\n  hue2rgb,\n  hexify,\n  rgb2Hsl,\n  fromAlphaToFloat,\n  greyAverage,\n} from './util';\n\n/**\n * @class Color common color operations\n * @tutorial {@link http://fabricjs.com/fabric-intro-part-2/#colors colors}\n */\nexport class Color {\n  private declare _source: TRGBAColorSource;\n\n  /**\n   *\n   * @param {string} [color] optional in hex or rgb(a) or hsl format or from known color list\n   */\n  constructor(color?: TColorArg) {\n    if (!color) {\n      // we default to black as canvas does\n      this.setSource([0, 0, 0, 1]);\n    } else if (color instanceof Color) {\n      this.setSource([...color._source]);\n    } else if (Array.isArray(color)) {\n      const [r, g, b, a = 1] = color;\n      this.setSource([r, g, b, a]);\n    } else {\n      this.setSource(this._tryParsingColor(color));\n    }\n  }\n\n  /**\n   * @private\n   * @param {string} [color] Color value to parse\n   * @returns {TRGBAColorSource}\n   */\n  protected _tryParsingColor(color: string) {\n    if (color in ColorNameMap) {\n      color = ColorNameMap[color as keyof typeof ColorNameMap];\n    }\n    return color === 'transparent'\n      ? ([255, 255, 255, 0] as TRGBAColorSource)\n      : Color.sourceFromHex(color) ||\n          Color.sourceFromRgb(color) ||\n          Color.sourceFromHsl(color) ||\n          // color is not recognized\n          // we default to black as canvas does\n          ([0, 0, 0, 1] as TRGBAColorSource);\n  }\n\n  /**\n   * Returns source of this color (where source is an array representation; ex: [200, 200, 100, 1])\n   * @return {TRGBAColorSource}\n   */\n  getSource() {\n    return this._source;\n  }\n\n  /**\n   * Sets source of this color (where source is an array representation; ex: [200, 200, 100, 1])\n   * @param {TRGBAColorSource} source\n   */\n  setSource(source: TRGBAColorSource) {\n    this._source = source;\n  }\n\n  /**\n   * Returns color representation in RGB format\n   * @return {String} ex: rgb(0-255,0-255,0-255)\n   */\n  toRgb() {\n    const [r, g, b] = this.getSource();\n    return `rgb(${r},${g},${b})`;\n  }\n\n  /**\n   * Returns color representation in RGBA format\n   * @return {String} ex: rgba(0-255,0-255,0-255,0-1)\n   */\n  toRgba() {\n    return `rgba(${this.getSource().join(',')})`;\n  }\n\n  /**\n   * Returns color representation in HSL format\n   * @return {String} ex: hsl(0-360,0%-100%,0%-100%)\n   */\n  toHsl() {\n    const [h, s, l] = rgb2Hsl(...this.getSource());\n    return `hsl(${h},${s}%,${l}%)`;\n  }\n\n  /**\n   * Returns color representation in HSLA format\n   * @return {String} ex: hsla(0-360,0%-100%,0%-100%,0-1)\n   */\n  toHsla() {\n    const [h, s, l, a] = rgb2Hsl(...this.getSource());\n    return `hsla(${h},${s}%,${l}%,${a})`;\n  }\n\n  /**\n   * Returns color representation in HEX format\n   * @return {String} ex: FF5555\n   */\n  toHex() {\n    const fullHex = this.toHexa();\n    return fullHex.slice(0, 6);\n  }\n\n  /**\n   * Returns color representation in HEXA format\n   * @return {String} ex: FF5555CC\n   */\n  toHexa() {\n    const [r, g, b, a] = this.getSource();\n    return `${hexify(r)}${hexify(g)}${hexify(b)}${hexify(Math.round(a * 255))}`;\n  }\n\n  /**\n   * Gets value of alpha channel for this color\n   * @return {Number} 0-1\n   */\n  getAlpha() {\n    return this.getSource()[3];\n  }\n\n  /**\n   * Sets value of alpha channel for this color\n   * @param {Number} alpha Alpha value 0-1\n   * @return {Color} thisArg\n   */\n  setAlpha(alpha: number) {\n    this._source[3] = alpha;\n    return this;\n  }\n\n  /**\n   * Transforms color to its grayscale representation\n   * @return {Color} thisArg\n   */\n  toGrayscale() {\n    this.setSource(greyAverage(this.getSource()));\n    return this;\n  }\n\n  /**\n   * Transforms color to its black and white representation\n   * @param {Number} threshold\n   * @return {Color} thisArg\n   */\n  toBlackWhite(threshold: number) {\n    const [average, , , a] = greyAverage(this.getSource()),\n      bOrW = average < (threshold || 127) ? 0 : 255;\n    this.setSource([bOrW, bOrW, bOrW, a]);\n    return this;\n  }\n\n  /**\n   * Overlays color with another color\n   * @param {String|Color} otherColor\n   * @return {Color} thisArg\n   */\n  overlayWith(otherColor: string | Color) {\n    if (!(otherColor instanceof Color)) {\n      otherColor = new Color(otherColor);\n    }\n\n    const source = this.getSource(),\n      otherAlpha = 0.5,\n      otherSource = otherColor.getSource(),\n      [R, G, B] = source.map((value, index) =>\n        Math.round(value * (1 - otherAlpha) + otherSource[index] * otherAlpha)\n      );\n\n    this.setSource([R, G, B, source[3]]);\n    return this;\n  }\n\n  /**\n   * Returns new color object, when given a color in RGB format\n   * @memberOf Color\n   * @param {String} color Color value ex: rgb(0-255,0-255,0-255)\n   * @return {Color}\n   */\n  static fromRgb(color: string): Color {\n    return Color.fromRgba(color);\n  }\n\n  /**\n   * Returns new color object, when given a color in RGBA format\n   * @static\n   * @function\n   * @memberOf Color\n   * @param {String} color\n   * @return {Color}\n   */\n  static fromRgba(color: string): Color {\n    return new Color(Color.sourceFromRgb(color));\n  }\n\n  /**\n   * Returns array representation (ex: [100, 100, 200, 1]) of a color that's in RGB or RGBA format\n   * @memberOf Color\n   * @param {String} color Color value ex: rgb(0-255,0-255,0-255), rgb(0%-100%,0%-100%,0%-100%)\n   * @return {TRGBAColorSource | undefined} source\n   */\n  static sourceFromRgb(color: string): TRGBAColorSource | undefined {\n    const match = color.match(reRGBa());\n    if (match) {\n      const [r, g, b] = match.slice(1, 4).map((value) => {\n        const parsedValue = parseFloat(value);\n        return value.endsWith('%')\n          ? Math.round(parsedValue * 2.55)\n          : parsedValue;\n      });\n      return [r, g, b, fromAlphaToFloat(match[4])];\n    }\n  }\n\n  /**\n   * Returns new color object, when given a color in HSL format\n   * @param {String} color Color value ex: hsl(0-260,0%-100%,0%-100%)\n   * @memberOf Color\n   * @return {Color}\n   */\n  static fromHsl(color: string): Color {\n    return Color.fromHsla(color);\n  }\n\n  /**\n   * Returns new color object, when given a color in HSLA format\n   * @static\n   * @function\n   * @memberOf Color\n   * @param {String} color\n   * @return {Color}\n   */\n  static fromHsla(color: string): Color {\n    return new Color(Color.sourceFromHsl(color));\n  }\n\n  /**\n   * Returns array representation (ex: [100, 100, 200, 1]) of a color that's in HSL or HSLA format.\n   * Adapted from <a href=\"https://rawgithub.com/mjijackson/mjijackson.github.com/master/2008/02/rgb-to-hsl-and-rgb-to-hsv-color-model-conversion-algorithms-in-javascript.html\">https://github.com/mjijackson</a>\n   * @memberOf Color\n   * @param {String} color Color value ex: hsl(0-360,0%-100%,0%-100%) or hsla(0-360,0%-100%,0%-100%, 0-1)\n   * @return {TRGBAColorSource | undefined} source\n   * @see http://http://www.w3.org/TR/css3-color/#hsl-color\n   */\n  static sourceFromHsl(color: string): TRGBAColorSource | undefined {\n    const match = color.match(reHSLa());\n    if (!match) {\n      return;\n    }\n\n    const h = (((parseFloat(match[1]) % 360) + 360) % 360) / 360,\n      s = parseFloat(match[2]) / 100,\n      l = parseFloat(match[3]) / 100;\n    let r: number, g: number, b: number;\n\n    if (s === 0) {\n      r = g = b = l;\n    } else {\n      const q = l <= 0.5 ? l * (s + 1) : l + s - l * s,\n        p = l * 2 - q;\n\n      r = hue2rgb(p, q, h + 1 / 3);\n      g = hue2rgb(p, q, h);\n      b = hue2rgb(p, q, h - 1 / 3);\n    }\n\n    return [\n      Math.round(r * 255),\n      Math.round(g * 255),\n      Math.round(b * 255),\n      fromAlphaToFloat(match[4]),\n    ];\n  }\n\n  /**\n   * Returns new color object, when given a color in HEX format\n   * @static\n   * @memberOf Color\n   * @param {String} color Color value ex: FF5555\n   * @return {Color}\n   */\n  static fromHex(color: string): Color {\n    return new Color(Color.sourceFromHex(color));\n  }\n\n  /**\n   * Returns array representation (ex: [100, 100, 200, 1]) of a color that's in HEX format\n   * @static\n   * @memberOf Color\n   * @param {String} color ex: FF5555 or FF5544CC (RGBa)\n   * @return {TRGBAColorSource | undefined} source\n   */\n  static sourceFromHex(color: string): TRGBAColorSource | undefined {\n    if (color.match(reHex())) {\n      const value = color.slice(color.indexOf('#') + 1),\n        isShortNotation = value.length <= 4;\n      let expandedValue: string[];\n      if (isShortNotation) {\n        expandedValue = value.split('').map((hex) => hex + hex);\n      } else {\n        expandedValue = value.match(/.{2}/g)!;\n      }\n      const [r, g, b, a = 255] = expandedValue.map((hexCouple) =>\n        parseInt(hexCouple, 16)\n      );\n      return [r, g, b, a / 255];\n    }\n  }\n}\n","/**\n * Regex matching color in RGB or RGBA formats (ex: `rgb(0, 0, 0)`, `rgba(255, 100, 10, 0.5)`, `rgba( 255 , 100 , 10 , 0.5 )`, `rgb(1,1,1)`, `rgba(100%, 60%, 10%, 0.5)`)\n * Also matching rgba(r g b / a) as per new specs\n * https://developer.mozilla.org/en-US/docs/Web/CSS/color_value/rgb\n * Formal syntax at the time of writing:\n * <rgb()> =\n *  rgb( [ <percentage> | none ]{3} [ / [ <alpha-value> | none ] ]? )  |\n *  rgb( [ <number> | none ]{3} [ / [ <alpha-value> | none ] ]? )\n * <alpha-value> = <number> | <percentage>\n *\n * For learners this is how you can read this regex\n * Regular expression for matching an rgba or rgb CSS color value\n *\n * /^          # Beginning of the string\n * rgba?       # \"rgb\" or \"rgba\"\n * \\(\\s*       # Opening parenthesis and optional whitespace\n * (\\d{0,3}    # 0 to three digits R channel\n *  (?:\\.\\d+)? # Optional decimal with one or more digits\n * )           # End of capturing group for the first color component\n * %?          # Optional percent sign after the first color component\n * \\s*         # Optional whitespace\n * [\\s|,]      # Separator between color components can be a space or comma\n * \\s*         # Optional whitespace\n * (\\d{0,3}    # 0 to three digits G channel\n *  (?:\\.\\d+)? # Optional decimal with one or more digits\n * )           # End of capturing group for the second color component\n * %?          # Optional percent sign after the second color component\n * \\s*         # Optional whitespace\n * [\\s|,]      # Separator between color components can be a space or comma\n * \\s*         # Optional whitespace\n * (\\d{0,3}    # 0 to three digits B channel\n *  (?:\\.\\d+)? # Optional decimal with one or more digits\n * )           # End of capturing group for the third color component\n * %?          # Optional percent sign after the third color component\n * \\s*         # Optional whitespace\n * (?:         # Beginning of non-capturing group for alpha value\n *  \\s*        # Optional whitespace\n *  [,/]       # Comma or slash separator for alpha value\n *  \\s*        # Optional whitespace\n *  (\\d{0,3}   # Zero to three digits\n *    (?:\\.\\d+)? # Optional decimal with one or more digits\n *  )          # End of capturing group for alpha value\n *  %?         # Optional percent sign after alpha value\n *  \\s*        # Optional whitespace\n * )?          # End of non-capturing group for alpha value (optional)\n * \\)          # Closing parenthesis\n * $           # End of the string\n *\n * The alpha channel can be in the format 0.4 .7 or 1 or 73%\n *\n * WARNING this regex doesn't fail on off spec colors. it matches everything that could be a color.\n * So the spec does not allow for `rgba(30 , 45%  35, 49%)` but this will work anyways for us\n */\nexport const reRGBa = () =>\n  /^rgba?\\(\\s*(\\d{0,3}(?:\\.\\d+)?%?)\\s*[\\s|,]\\s*(\\d{0,3}(?:\\.\\d+)?%?)\\s*[\\s|,]\\s*(\\d{0,3}(?:\\.\\d+)?%?)\\s*(?:\\s*[,/]\\s*(\\d{0,3}(?:\\.\\d+)?%?)\\s*)?\\)$/i;\n\n/**\n * Regex matching color in HSL or HSLA formats (ex: hsl(0, 0, 0), rgba(255, 100, 10, 0.5), rgba( 255 , 100 , 10 , 0.5 ), rgb(1,1,1), rgba(100%, 60%, 10%, 0.5))\n * Also matching rgba(r g b / a) as per new specs\n * https://developer.mozilla.org/en-US/docs/Web/CSS/color_value/hsl\n * Formal syntax at the time of writing:\n * <hsl()> =\n *   hsl( [ <hue> | none ] [ <percentage> | none ] [ <percentage> | none ] [ / [ <alpha-value> | none ] ]? )\n *\n * <hue> =\n *   <number>  |\n *   <angle>\n *\n * <alpha-value> =\n *   <number>      |\n *   <percentage>\n *\n * For learners this is how you can read this regex\n * Regular expression for matching an hsla or hsl CSS color value\n *\n * /^hsla?\\(         // Matches the beginning of the string and the opening parenthesis of \"hsl\" or \"hsla\"\n * \\s*               // Matches any whitespace characters (space, tab, etc.) zero or more times\n * (\\d{1,3})         // Hue: Matches one to three digits and captures it in a group\n * \\s*               // Matches any whitespace characters zero or more times\n * [\\s|,]            // Matches a space, tab or comma\n * \\s*               // Matches any whitespace characters zero or more times\n * (\\d{1,3}%)        // Saturation: Matches one to three digits followed by a percentage sign and captures it in a group\n * \\s*               // Matches any whitespace characters zero or more times\n * [\\s|,]            // Matches a space, tab or comma\n * \\s*               // Matches any whitespace characters zero or more times\n * (\\d{1,3}%)        // Lightness: Matches one to three digits followed by a percentage sign and captures it in a group\n * \\s*               // Matches any whitespace characters zero or more times\n * (?:               // Alpha: Begins a non-capturing group for the alpha value\n *   \\s*             // Matches any whitespace characters zero or more times\n *   [,/]            // Matches a comma or forward slash\n *   \\s*             // Matches any whitespace characters zero or more times\n *   (\\d*(?:\\.\\d+)?%?) // Matches zero or more digits, optionally followed by a decimal point and one or more digits, followed by an optional percentage sign and captures it in a group\n *   \\s*             // Matches any whitespace characters zero or more times\n * )?                // Makes the alpha value group optional\n * \\)                // Matches the closing parenthesis\n * $/i               // Matches the end of the string and sets the regular expression to case-insensitive mode\n *\n * WARNING this regex doesn't fail on off spec colors. It matches everything that could be a color.\n * So the spec does not allow `hsl(30 , 45%  35, 49%)` but this will work anyways for us.\n */\nexport const reHSLa = () =>\n  /^hsla?\\(\\s*([+-]?\\d{1,3})\\s*[\\s|,]\\s*(\\d{1,3}%)\\s*[\\s|,]\\s*(\\d{1,3}%)\\s*(?:\\s*[,/]\\s*(\\d*(?:\\.\\d+)?%?)\\s*)?\\)$/i;\n\n/**\n * Regex matching color in HEX format (ex: #FF5544CC, #FF5555, 010155, aff)\n */\nexport const reHex = () => /^#?(([0-9a-f]){3,4}|([0-9a-f]{2}){3,4})$/i;\n","/**\n * A wrapper around Number#toFixed, which contrary to native method returns number, not string.\n * @param {number|string} number number to operate on\n * @param {number} fractionDigits number of fraction digits to \"leave\"\n * @return {number}\n */\nexport const toFixed = (number: number | string, fractionDigits: number) =>\n  parseFloat(Number(number).toFixed(fractionDigits));\n","import { Color } from '../../color/Color';\nimport { config } from '../../config';\nimport { DEFAULT_SVG_FONT_SIZE, NONE } from '../../constants';\nimport type {\n  TBBox,\n  TMat2D,\n  SVGElementName,\n  SupportedSVGUnit,\n} from '../../typedefs';\nimport { toFixed } from './toFixed';\n\n/**\n * Returns array of attributes for given svg that fabric parses\n * @param {SVGElementName} type Type of svg element (eg. 'circle')\n * @return {Array} string names of supported attributes\n */\nexport const getSvgAttributes = (type: SVGElementName) => {\n  const commonAttributes = ['instantiated_by_use', 'style', 'id', 'class'];\n  switch (type) {\n    case 'linearGradient':\n      return commonAttributes.concat([\n        'x1',\n        'y1',\n        'x2',\n        'y2',\n        'gradientUnits',\n        'gradientTransform',\n      ]);\n    case 'radialGradient':\n      return commonAttributes.concat([\n        'gradientUnits',\n        'gradientTransform',\n        'cx',\n        'cy',\n        'r',\n        'fx',\n        'fy',\n        'fr',\n      ]);\n    case 'stop':\n      return commonAttributes.concat(['offset', 'stop-color', 'stop-opacity']);\n  }\n  return commonAttributes;\n};\n\n/**\n * Converts from attribute value to pixel value if applicable.\n * Returns converted pixels or original value not converted.\n * @param {string} value number to operate on\n * @param {number} fontSize\n * @return {number}\n */\nexport const parseUnit = (value: string, fontSize = DEFAULT_SVG_FONT_SIZE) => {\n  const unit = /\\D{0,2}$/.exec(value),\n    number = parseFloat(value);\n  const dpi = config.DPI;\n  switch (unit?.[0] as SupportedSVGUnit) {\n    case 'mm':\n      return (number * dpi) / 25.4;\n\n    case 'cm':\n      return (number * dpi) / 2.54;\n\n    case 'in':\n      return number * dpi;\n\n    case 'pt':\n      return (number * dpi) / 72; // or * 4 / 3\n\n    case 'pc':\n      return ((number * dpi) / 72) * 12; // or * 16\n\n    case 'em':\n      return number * fontSize;\n\n    default:\n      return number;\n  }\n};\n\nexport type MeetOrSlice = 'meet' | 'slice';\n\nexport type MinMidMax = 'Min' | 'Mid' | 'Max' | 'none';\n\nexport type TPreserveArParsed = {\n  meetOrSlice: MeetOrSlice;\n  alignX: MinMidMax;\n  alignY: MinMidMax;\n};\n\n// align can be either none or undefined or a combination of mid/max\nconst parseAlign = (align: string): MinMidMax[] => {\n  //divide align in alignX and alignY\n  if (align && align !== NONE) {\n    return [align.slice(1, 4) as MinMidMax, align.slice(5, 8) as MinMidMax];\n  } else if (align === NONE) {\n    return [align, align];\n  }\n  return ['Mid', 'Mid'];\n};\n\n/**\n * Parse preserveAspectRatio attribute from element\n * https://developer.mozilla.org/en-US/docs/Web/SVG/Attribute/preserveAspectRatio\n * @param {string} attribute to be parsed\n * @return {Object} an object containing align and meetOrSlice attribute\n */\nexport const parsePreserveAspectRatioAttribute = (\n  attribute: string\n): TPreserveArParsed => {\n  const [firstPart, secondPart] = attribute.trim().split(' ') as [\n    MinMidMax,\n    MeetOrSlice | undefined\n  ];\n  const [alignX, alignY] = parseAlign(firstPart);\n  return {\n    meetOrSlice: secondPart || 'meet',\n    alignX,\n    alignY,\n  };\n};\n\n/**\n * given an array of 6 number returns something like `\"matrix(...numbers)\"`\n * @param {TMat2D} transform an array with 6 numbers\n * @return {String} transform matrix for svg\n */\nexport const matrixToSVG = (transform: TMat2D) =>\n  'matrix(' +\n  transform\n    .map((value) => toFixed(value, config.NUM_FRACTION_DIGITS))\n    .join(' ') +\n  ')';\n\n/**\n * Adobe Illustrator (at least CS5) is unable to render rgba()-based fill values\n * we work around it by \"moving\" alpha channel into opacity attribute and setting fill's alpha to 1\n * @param prop\n * @param value\n * @param {boolean} inlineStyle The default is inline style, the separator used is \":\", The other is \"=\"\n * @returns\n */\nexport const colorPropToSVG = (\n  prop: string,\n  value?: any,\n  inlineStyle = true\n) => {\n  let colorValue;\n  let opacityValue;\n  if (!value) {\n    colorValue = 'none';\n  } else if (value.toLive) {\n    colorValue = `url(#SVGID_${value.id})`;\n  } else {\n    const color = new Color(value),\n      opacity = color.getAlpha();\n\n    colorValue = color.toRgb();\n    if (opacity !== 1) {\n      opacityValue = opacity.toString();\n    }\n  }\n  if (inlineStyle) {\n    return `${prop}: ${colorValue}; ${\n      opacityValue ? `${prop}-opacity: ${opacityValue}; ` : ''\n    }`;\n  } else {\n    return `${prop}=\"${colorValue}\" ${\n      opacityValue ? `${prop}-opacity=\"${opacityValue}\" ` : ''\n    }`;\n  }\n};\n\nexport const createSVGRect = (\n  color: string,\n  { left, top, width, height }: TBBox,\n  precision = config.NUM_FRACTION_DIGITS\n) => {\n  const svgColor = colorPropToSVG('fill', color, false);\n  const [x, y, w, h] = [left, top, width, height].map((value) =>\n    toFixed(value, precision)\n  );\n  return `<rect ${svgColor} x=\"${x}\" y=\"${y}\" width=\"${w}\" height=\"${h}\"></rect>`;\n};\n","import type { FabricObject } from '../shapes/Object/Object';\nimport type { TFiller } from '../typedefs';\nimport type { FabricText } from '../shapes/Text/Text';\nimport type { Pattern } from '../Pattern';\nimport type { Path } from '../shapes/Path';\nimport type { ActiveSelection } from '../shapes/ActiveSelection';\n\nexport const isFiller = (\n  filler: TFiller | string | null\n): filler is TFiller => {\n  return !!filler && (filler as TFiller).toLive !== undefined;\n};\n\nexport const isSerializableFiller = (\n  filler: TFiller | string | null\n): filler is TFiller => {\n  return !!filler && typeof (filler as TFiller).toObject === 'function';\n};\n\nexport const isPattern = (filler: TFiller): filler is Pattern => {\n  return (\n    !!filler && (filler as Pattern).offsetX !== undefined && 'source' in filler\n  );\n};\n\nexport const isTextObject = (\n  fabricObject?: FabricObject\n): fabricObject is FabricText => {\n  return (\n    !!fabricObject &&\n    typeof (fabricObject as FabricText)._renderText === 'function'\n  );\n};\n\nexport const isPath = (fabricObject?: FabricObject): fabricObject is Path => {\n  // we could use instanceof but that would mean pulling in Text code for a simple check\n  // @todo discuss what to do and how to do\n  return (\n    !!fabricObject &&\n    typeof (fabricObject as Path)._renderPathCommands === 'function'\n  );\n};\n\nexport const isActiveSelection = (\n  fabricObject?: FabricObject\n): fabricObject is ActiveSelection =>\n  !!fabricObject && 'multiSelectionStacking' in fabricObject;\n","/**\n * Returns element scroll offsets\n * @param {HTMLElement} element Element to operate on\n * @return {Object} Object with left/top values\n */\nexport function getScrollLeftTop(element: HTMLElement | null) {\n  const doc = element && getDocumentFromElement(element);\n  let left = 0,\n    top = 0;\n  if (!element || !doc) {\n    return { left, top };\n  }\n\n  const docElement = doc.documentElement,\n    body = doc.body || {\n      scrollLeft: 0,\n      scrollTop: 0,\n    };\n  // While loop checks (and then sets element to) .parentNode OR .host\n  //  to account for ShadowDOM. We still want to traverse up out of ShadowDOM,\n  //  but the .parentNode of a root ShadowDOM node will always be null, instead\n  //  it should be accessed through .host. See http://stackoverflow.com/a/24765528/4383938\n  // @ts-expect-error Set element to element parent, or 'host' in case of ShadowDOM\n  while (element && (element.parentNode || element.host)) {\n    // @ts-expect-error Set element to element parent, or 'host' in case of ShadowDOM\n    element = element.parentNode || element.host;\n    // @ts-expect-error because element is typed as HTMLElement but it can go up to document\n    if (element === doc) {\n      left = body.scrollLeft || docElement.scrollLeft || 0;\n      top = body.scrollTop || docElement.scrollTop || 0;\n    } else {\n      left += element!.scrollLeft || 0;\n      top += element!.scrollTop || 0;\n    }\n\n    if (element!.nodeType === 1 && element!.style.position === 'fixed') {\n      break;\n    }\n  }\n\n  return { left, top };\n}\n\nexport const getDocumentFromElement = (el: HTMLElement) =>\n  el.ownerDocument || null;\n\nexport const getWindowFromElement = (el: HTMLElement) =>\n  el.ownerDocument?.defaultView || null;\n","// TODO this file needs to go away, cross browser style support is not fabricjs domain.\n\n/**\n * wrapper for setting element's style\n * @param {HTMLElement} element\n * @param {Object | string} styles\n */\nexport function setStyle(\n  element: HTMLElement,\n  styles: string | Record<string, string>\n) {\n  const elementStyle = element.style;\n  if (!elementStyle) {\n    return;\n  } else if (typeof styles === 'string') {\n    element.style.cssText += ';' + styles;\n  } else {\n    Object.entries(styles).forEach(([property, value]) =>\n      elementStyle.setProperty(property, value)\n    );\n  }\n}\n","import { LEFT, NONE, TOP } from '../../constants';\nimport type { TSize } from '../../typedefs';\nimport {\n  getDocumentFromElement,\n  getWindowFromElement,\n  getScrollLeftTop,\n} from '../../util/dom_misc';\nimport { setStyle } from '../../util/dom_style';\n\nexport const setCanvasDimensions = (\n  el: HTMLCanvasElement,\n  ctx: CanvasRenderingContext2D,\n  { width, height }: TSize,\n  retinaScaling = 1\n) => {\n  el.width = width;\n  el.height = height;\n  if (retinaScaling > 1) {\n    el.setAttribute('width', (width * retinaScaling).toString());\n    el.setAttribute('height', (height * retinaScaling).toString());\n    ctx.scale(retinaScaling, retinaScaling);\n  }\n};\n\nexport function allowTouchScrolling(element: HTMLElement, allow: boolean) {\n  const touchAction = allow ? 'manipulation' : NONE;\n  setStyle(element, {\n    'touch-action': touchAction,\n    '-ms-touch-action': touchAction,\n  });\n}\n\nexport type CSSDimensions = {\n  width: number | string;\n  height: number | string;\n};\n\nexport const setCSSDimensions = (\n  el: HTMLElement,\n  { width, height }: Partial<CSSDimensions>\n) => {\n  width && (el.style.width = typeof width === 'number' ? `${width}px` : width);\n  height &&\n    (el.style.height = typeof height === 'number' ? `${height}px` : height);\n};\n\n/**\n * Returns offset for a given element\n * @param {HTMLElement} element Element to get offset for\n * @return {Object} Object with \"left\" and \"top\" properties\n */\nexport function getElementOffset(element: HTMLElement) {\n  let box = { left: 0, top: 0 };\n  const doc = element && getDocumentFromElement(element),\n    offset = { left: 0, top: 0 },\n    offsetAttributes = {\n      borderLeftWidth: LEFT,\n      borderTopWidth: TOP,\n      paddingLeft: LEFT,\n      paddingTop: TOP,\n    } as const;\n\n  if (!doc) {\n    return offset;\n  }\n  const elemStyle =\n    getWindowFromElement(element)?.getComputedStyle(element, null) || {};\n  for (const attr in offsetAttributes) {\n    // @ts-expect-error TS learn to iterate!\n    offset[offsetAttributes[attr]] += parseInt(elemStyle[attr], 10) || 0;\n  }\n\n  const docElem = doc.documentElement;\n  if (typeof element.getBoundingClientRect !== 'undefined') {\n    box = element.getBoundingClientRect();\n  }\n\n  const scrollLeftTop = getScrollLeftTop(element);\n\n  return {\n    left:\n      box.left + scrollLeftTop.left - (docElem.clientLeft || 0) + offset.left,\n    top: box.top + scrollLeftTop.top - (docElem.clientTop || 0) + offset.top,\n  };\n}\n\n/**\n * Makes element unselectable\n * @param {HTMLElement} element Element to make unselectable\n * @return {HTMLElement} Element that was passed in\n */\nexport function makeElementUnselectable(element: HTMLElement) {\n  if (typeof element.onselectstart !== 'undefined') {\n    element.onselectstart = () => false;\n  }\n  element.style.userSelect = NONE;\n  return element;\n}\n","import { getEnv, getFabricDocument } from '../../env';\nimport type { TSize } from '../../typedefs';\nimport type { CSSDimensions } from './util';\nimport { setCSSDimensions, getElementOffset } from './util';\nimport { createCanvasElement, isHTMLCanvas } from '../../util/misc/dom';\nimport { setCanvasDimensions } from './util';\nimport { FabricError } from '../../util/internals/console';\n\nexport type CanvasItem = {\n  el: HTMLCanvasElement;\n  ctx: CanvasRenderingContext2D;\n};\n\nexport class StaticCanvasDOMManager {\n  /**\n   * Keeps a copy of the canvas style before setting retina scaling and other potions\n   * in order to return it to original state on dispose\n   * @type string\n   */\n  private _originalCanvasStyle?: string;\n\n  lower: CanvasItem;\n\n  constructor(arg0?: string | HTMLCanvasElement) {\n    const el = this.createLowerCanvas(arg0);\n    this.lower = { el, ctx: el.getContext('2d')! };\n  }\n\n  protected createLowerCanvas(arg0?: HTMLCanvasElement | string) {\n    // canvasEl === 'HTMLCanvasElement' does not work on jsdom/node\n    const el = isHTMLCanvas(arg0)\n      ? arg0\n      : (arg0 &&\n          (getFabricDocument().getElementById(arg0) as HTMLCanvasElement)) ||\n        createCanvasElement();\n    if (el.hasAttribute('data-fabric')) {\n      throw new FabricError(\n        'Trying to initialize a canvas that has already been initialized. Did you forget to dispose the canvas?'\n      );\n    }\n    this._originalCanvasStyle = el.style.cssText;\n    el.setAttribute('data-fabric', 'main');\n    el.classList.add('lower-canvas');\n    return el;\n  }\n\n  cleanupDOM({ width, height }: TSize) {\n    const { el } = this.lower;\n    // restore canvas style and attributes\n    el.classList.remove('lower-canvas');\n    el.removeAttribute('data-fabric');\n    // restore canvas size to original size in case retina scaling was applied\n    el.setAttribute('width', `${width}`);\n    el.setAttribute('height', `${height}`);\n    el.style.cssText = this._originalCanvasStyle || '';\n    this._originalCanvasStyle = undefined;\n  }\n\n  setDimensions(size: TSize, retinaScaling: number) {\n    const { el, ctx } = this.lower;\n    setCanvasDimensions(el, ctx, size, retinaScaling);\n  }\n\n  setCSSDimensions(size: Partial<CSSDimensions>) {\n    setCSSDimensions(this.lower.el, size);\n  }\n\n  /**\n   * Calculates canvas element offset relative to the document\n   */\n  calcOffset() {\n    return getElementOffset(this.lower.el);\n  }\n\n  dispose() {\n    getEnv().dispose(this.lower.el);\n    // @ts-expect-error disposing\n    delete this.lower;\n  }\n}\n","import { iMatrix } from '../constants';\nimport type { FabricObject } from '../shapes/Object/FabricObject';\nimport type { TFiller, TMat2D, TOptions } from '../typedefs';\nimport type { StaticCanvas } from './StaticCanvas';\n\ninterface CanvasDrawableOptions {\n  /**\n   * if set to false background image is not affected by viewport transform\n   * @since 1.6.3\n   * @type Boolean\n   * @todo we should really find a different way to do this\n   * @default\n   */\n  backgroundVpt: boolean;\n\n  /**\n   * Background color of canvas instance.\n   * @type {(String|TFiller)}\n   * @default\n   */\n  backgroundColor: TFiller | string;\n\n  /**\n   * Background image of canvas instance.\n   * since 2.4.0 image caching is active, please when putting an image as background, add to the\n   * canvas property a reference to the canvas it is on. Otherwise the image cannot detect the zoom\n   * vale. As an alternative you can disable image objectCaching\n   * @type FabricObject\n   * @default\n   */\n  backgroundImage?: FabricObject;\n\n  /**\n   * if set to false overlay image is not affected by viewport transform\n   * @since 1.6.3\n   * @type Boolean\n   * @todo we should really find a different way to do this\n   * @default\n   */\n  overlayVpt: boolean;\n\n  /**\n   * Overlay color of canvas instance.\n   * @since 1.3.9\n   * @type {(String|TFiller)}\n   * @default\n   */\n  overlayColor: TFiller | string;\n\n  /**\n   * Overlay image of canvas instance.\n   * since 2.4.0 image caching is active, please when putting an image as overlay, add to the\n   * canvas property a reference to the canvas it is on. Otherwise the image cannot detect the zoom\n   * vale. As an alternative you can disable image objectCaching\n   * @type FabricObject\n   * @default\n   */\n  overlayImage?: FabricObject;\n}\n\ninterface CanvasRenderingOptions {\n  /**\n   * Indicates whether {@link StaticCanvas#add}, {@link StaticCanvas#insertAt} and {@link StaticCanvas#remove},\n   * {@link StaticCanvas#moveTo}, {@link StaticCanvas#clear} and many more, should also re-render canvas.\n   * Disabling this option will not give a performance boost when adding/removing a lot of objects to/from canvas at once\n   * since the renders are queued and executed one per frame.\n   * Disabling is suggested anyway and managing the renders of the app manually is not a big effort ( canvas.requestRenderAll() )\n   * Left default to true to do not break documentation and old app, fiddles.\n   * @type Boolean\n   * @default\n   */\n  renderOnAddRemove: boolean;\n\n  /**\n   * Based on vptCoords and object.aCoords, skip rendering of objects that\n   * are not included in current viewport.\n   * May greatly help in applications with crowded canvas and use of zoom/pan\n   * If One of the corner of the bounding box of the object is on the canvas\n   * the objects get rendered.\n   * @type Boolean\n   * @default true\n   */\n  skipOffscreen: boolean;\n\n  /**\n   * When true, canvas is scaled by devicePixelRatio for better rendering on retina screens\n   * @type Boolean\n   * @default\n   */\n  enableRetinaScaling: boolean;\n\n  /**\n   * Indicates whether this canvas will use image smoothing, this is on by default in browsers\n   * @type Boolean\n   * @default\n   */\n  imageSmoothingEnabled: boolean;\n\n  /**\n   * a fabricObject that, without stroke define a clipping area with their shape. filled in black\n   * the clipPath object gets used when the canvas has rendered, and the context is placed in the\n   * top left corner of the canvas.\n   * clipPath will clip away controls, if you do not want this to happen use controlsAboveOverlay = true\n   * @type FabricObject\n   */\n  clipPath?: FabricObject;\n}\n\nexport interface CanvasExportOptions {\n  /**\n   * Indicates whether toObject/toDatalessObject should include default values\n   * if set to false, takes precedence over the object value.\n   * @type Boolean\n   * @default\n   */\n  includeDefaultValues: boolean;\n\n  /**\n   * When true, getSvgTransform() will apply the StaticCanvas.viewportTransform to the SVG transformation. When true,\n   * a zoomed canvas will then produce zoomed SVG output.\n   * @type Boolean\n   * @default\n   */\n  svgViewportTransformation: boolean;\n}\n\nexport interface StaticCanvasOptions\n  extends CanvasDrawableOptions,\n    CanvasRenderingOptions,\n    CanvasExportOptions {\n  /**\n   * Width in virtual/logical pixels of the canvas.\n   * The canvas can be larger than width if retina scaling is active\n   * @type number\n   */\n  width: number;\n\n  /**\n   * Height in virtual/logical pixels of the canvas.\n   * The canvas can be taller than width if retina scaling is active\n   * @type height\n   */\n  height: number;\n\n  /**\n   * Indicates whether object controls (borders/controls) are rendered above overlay image\n   * @type Boolean\n   * @default\n   *\n   * @todo move to Canvas\n   */\n  controlsAboveOverlay: boolean;\n\n  /**\n   * Indicates whether the browser can be scrolled when using a touchscreen and dragging on the canvas\n   * @type Boolean\n   * @default\n   *\n   * @todo move to Canvas\n   */\n  allowTouchScrolling: boolean;\n\n  /**\n   * The transformation (a Canvas 2D API transform matrix) which focuses the viewport\n   * @type Array\n   * @example <caption>Default transform</caption>\n   * canvas.viewportTransform = [1, 0, 0, 1, 0, 0];\n   * @example <caption>Scale by 70% and translate toward bottom-right by 50, without skewing</caption>\n   * canvas.viewportTransform = [0.7, 0, 0, 0.7, 50, 50];\n   * @default\n   */\n  viewportTransform: TMat2D;\n}\n\nexport const staticCanvasDefaults: TOptions<StaticCanvasOptions> = {\n  backgroundVpt: true,\n  backgroundColor: '',\n  overlayVpt: true,\n  overlayColor: '',\n\n  includeDefaultValues: true,\n  svgViewportTransformation: true,\n\n  renderOnAddRemove: true,\n  skipOffscreen: true,\n  enableRetinaScaling: true,\n  imageSmoothingEnabled: true,\n\n  /**\n   * @todo move to Canvas\n   */\n  controlsAboveOverlay: false,\n  /**\n   * @todo move to Canvas\n   */\n  allowTouchScrolling: false,\n\n  viewportTransform: [...iMatrix],\n};\n","import { config } from '../config';\nimport { CENTER, VERSION } from '../constants';\nimport type { CanvasEvents, StaticCanvasEvents } from '../EventTypeDefs';\nimport type { Gradient } from '../gradient/Gradient';\nimport { createCollectionMixin, isCollection } from '../Collection';\nimport { CommonMethods } from '../CommonMethods';\nimport type { Pattern } from '../Pattern';\nimport { Point } from '../Point';\nimport type { TCachedFabricObject } from '../shapes/Object/Object';\nimport type {\n  Abortable,\n  Constructor,\n  TCornerPoint,\n  TDataUrlOptions,\n  TFiller,\n  TMat2D,\n  TSize,\n  TSVGReviver,\n  TToCanvasElementOptions,\n  TValidToObjectMethod,\n  TOptions,\n} from '../typedefs';\nimport {\n  cancelAnimFrame,\n  requestAnimFrame,\n} from '../util/animation/AnimationFrameProvider';\nimport { runningAnimations } from '../util/animation/AnimationRegistry';\nimport { uid } from '../util/internals/uid';\nimport { createCanvasElement, toDataURL } from '../util/misc/dom';\nimport { invertTransform, transformPoint } from '../util/misc/matrix';\nimport type { EnlivenObjectOptions } from '../util/misc/objectEnlive';\nimport {\n  enlivenObjectEnlivables,\n  enlivenObjects,\n} from '../util/misc/objectEnlive';\nimport { pick } from '../util/misc/pick';\nimport { matrixToSVG } from '../util/misc/svgParsing';\nimport { toFixed } from '../util/misc/toFixed';\nimport { isFiller, isPattern, isTextObject } from '../util/typeAssertions';\nimport { StaticCanvasDOMManager } from './DOMManagers/StaticCanvasDOMManager';\nimport type { CSSDimensions } from './DOMManagers/util';\nimport type { FabricObject } from '../shapes/Object/FabricObject';\nimport type { StaticCanvasOptions } from './StaticCanvasOptions';\nimport { staticCanvasDefaults } from './StaticCanvasOptions';\nimport { log, FabricError } from '../util/internals/console';\nimport { getDevicePixelRatio } from '../env';\n\n/**\n * Having both options in TCanvasSizeOptions set to true transform the call in a calcOffset\n * Better try to restrict with types to avoid confusion.\n */\nexport type TCanvasSizeOptions =\n  | {\n      backstoreOnly?: true;\n      cssOnly?: false;\n    }\n  | {\n      backstoreOnly?: false;\n      cssOnly?: true;\n    };\n\nexport type TSVGExportOptions = {\n  suppressPreamble?: boolean;\n  viewBox?: {\n    x: number;\n    y: number;\n    width: number;\n    height: number;\n  };\n  encoding?: 'UTF-8'; // test Encoding type and see what happens\n  width?: string;\n  height?: string;\n  reviver?: TSVGReviver;\n};\n\n/**\n * Static canvas class\n * @see {@link http://fabricjs.com/static_canvas|StaticCanvas demo}\n * @fires before:render\n * @fires after:render\n * @fires canvas:cleared\n * @fires object:added\n * @fires object:removed\n */\n// TODO: fix `EventSpec` inheritance https://github.com/microsoft/TypeScript/issues/26154#issuecomment-1366616260\nexport class StaticCanvas<\n    EventSpec extends StaticCanvasEvents = StaticCanvasEvents\n  >\n  extends createCollectionMixin(CommonMethods<CanvasEvents>)\n  implements StaticCanvasOptions\n{\n  declare width: number;\n  declare height: number;\n\n  // background\n  declare backgroundVpt: boolean;\n  declare backgroundColor: TFiller | string;\n  declare backgroundImage?: FabricObject;\n  // overlay\n  declare overlayVpt: boolean;\n  declare overlayColor: TFiller | string;\n  declare overlayImage?: FabricObject;\n\n  declare clipPath?: FabricObject;\n\n  declare includeDefaultValues: boolean;\n\n  // rendering config\n  declare renderOnAddRemove: boolean;\n  declare skipOffscreen: boolean;\n  declare enableRetinaScaling: boolean;\n  declare imageSmoothingEnabled: boolean;\n\n  /**\n   * @todo move to Canvas\n   */\n  declare controlsAboveOverlay: boolean;\n\n  /**\n   * @todo move to Canvas\n   */\n  declare allowTouchScrolling: boolean;\n\n  declare viewportTransform: TMat2D;\n\n  /**\n   * The viewport bounding box in scene plane coordinates, see {@link calcViewportBoundaries}\n   */\n  declare vptCoords: TCornerPoint;\n\n  /**\n   * A reference to the canvas actual HTMLCanvasElement.\n   * Can be use to read the raw pixels, but never write or manipulate\n   * @type HTMLCanvasElement\n   */\n  get lowerCanvasEl() {\n    return this.elements.lower?.el;\n  }\n\n  get contextContainer() {\n    return this.elements.lower?.ctx;\n  }\n\n  /**\n   * If true the Canvas is in the process or has been disposed/destroyed.\n   * No more rendering operation will be executed on this canvas.\n   * @type boolean\n   */\n  declare destroyed?: boolean;\n\n  /**\n   * Started the process of disposing but not done yet.\n   * WIll likely complete the render cycle already scheduled but stopping adding more.\n   * @type boolean\n   */\n  declare disposed?: boolean;\n\n  declare _offset: { left: number; top: number };\n  protected declare hasLostContext: boolean;\n  protected declare nextRenderHandle: number;\n\n  declare elements: StaticCanvasDOMManager;\n\n  static ownDefaults = staticCanvasDefaults;\n\n  // reference to\n  protected declare __cleanupTask?: {\n    (): void;\n    kill: (reason?: any) => void;\n  };\n\n  static getDefaults(): Record<string, any> {\n    return StaticCanvas.ownDefaults;\n  }\n\n  constructor(\n    el?: string | HTMLCanvasElement,\n    options: TOptions<StaticCanvasOptions> = {}\n  ) {\n    super();\n    Object.assign(\n      this,\n      (this.constructor as typeof StaticCanvas).getDefaults()\n    );\n    this.set(options);\n    this.initElements(el);\n    this._setDimensionsImpl({\n      width: this.width || this.elements.lower.el.width || 0,\n      height: this.height || this.elements.lower.el.height || 0,\n    });\n    this.viewportTransform = [...this.viewportTransform];\n    this.calcViewportBoundaries();\n  }\n\n  protected initElements(el?: string | HTMLCanvasElement) {\n    this.elements = new StaticCanvasDOMManager(el);\n  }\n\n  add(...objects: FabricObject[]) {\n    const size = super.add(...objects);\n    objects.length > 0 && this.renderOnAddRemove && this.requestRenderAll();\n    return size;\n  }\n\n  insertAt(index: number, ...objects: FabricObject[]) {\n    const size = super.insertAt(index, ...objects);\n    objects.length > 0 && this.renderOnAddRemove && this.requestRenderAll();\n    return size;\n  }\n\n  remove(...objects: FabricObject[]) {\n    const removed = super.remove(...objects);\n    removed.length > 0 && this.renderOnAddRemove && this.requestRenderAll();\n    return removed;\n  }\n\n  _onObjectAdded(obj: FabricObject) {\n    if (obj.canvas && (obj.canvas as StaticCanvas) !== this) {\n      log(\n        'warn',\n        'Canvas is trying to add an object that belongs to a different canvas.\\n' +\n          'Resulting to default behavior: removing object from previous canvas and adding to new canvas'\n      );\n      obj.canvas.remove(obj);\n    }\n    obj._set('canvas', this);\n    obj.setCoords();\n    this.fire('object:added', { target: obj });\n    obj.fire('added', { target: this });\n  }\n\n  _onObjectRemoved(obj: FabricObject) {\n    obj._set('canvas', undefined);\n    this.fire('object:removed', { target: obj });\n    obj.fire('removed', { target: this });\n  }\n\n  _onStackOrderChanged() {\n    this.renderOnAddRemove && this.requestRenderAll();\n  }\n\n  /**\n   * @private\n   * @see https://developer.apple.com/library/safari/documentation/AudioVideo/Conceptual/HTML-canvas-guide/SettingUptheCanvas/SettingUptheCanvas.html\n   * @return {Number} retinaScaling if applied, otherwise 1;\n   */\n  getRetinaScaling() {\n    return this.enableRetinaScaling ? getDevicePixelRatio() : 1;\n  }\n\n  /**\n   * Calculates canvas element offset relative to the document\n   * This method is also attached as \"resize\" event handler of window\n   */\n  calcOffset() {\n    return (this._offset = this.elements.calcOffset());\n  }\n\n  /**\n   * Returns canvas width (in px)\n   * @return {Number}\n   */\n  getWidth(): number {\n    return this.width;\n  }\n\n  /**\n   * Returns canvas height (in px)\n   * @return {Number}\n   */\n  getHeight(): number {\n    return this.height;\n  }\n\n  /**\n   * Sets width of this canvas instance\n   * @param {Number|String} value                         Value to set width to\n   * @param {Object}        [options]                     Options object\n   * @param {Boolean}       [options.backstoreOnly=false] Set the given dimensions only as canvas backstore dimensions\n   * @param {Boolean}       [options.cssOnly=false]       Set the given dimensions only as css dimensions\n   * @deprecated will be removed in 7.0\n   */\n  setWidth(\n    value: TSize['width'],\n    options?: { backstoreOnly?: true; cssOnly?: false }\n  ): void;\n  setWidth(\n    value: CSSDimensions['width'],\n    options?: { cssOnly?: true; backstoreOnly?: false }\n  ): void;\n  setWidth(value: number, options?: never) {\n    return this.setDimensions({ width: value }, options);\n  }\n\n  /**s\n   * Sets height of this canvas instance\n   * @param {Number|String} value                         Value to set height to\n   * @param {Object}        [options]                     Options object\n   * @param {Boolean}       [options.backstoreOnly=false] Set the given dimensions only as canvas backstore dimensions\n   * @param {Boolean}       [options.cssOnly=false]       Set the given dimensions only as css dimensions\n   * @deprecated will be removed in 7.0\n   */\n  setHeight(\n    value: TSize['height'],\n    options?: { backstoreOnly?: true; cssOnly?: false }\n  ): void;\n  setHeight(\n    value: CSSDimensions['height'],\n    options?: { cssOnly?: true; backstoreOnly?: false }\n  ): void;\n  setHeight(value: CSSDimensions['height'], options?: never) {\n    return this.setDimensions({ height: value }, options);\n  }\n\n  /**\n   * Internal use only\n   * @protected\n   */\n  protected _setDimensionsImpl(\n    dimensions: Partial<TSize | CSSDimensions>,\n    { cssOnly = false, backstoreOnly = false }: TCanvasSizeOptions = {}\n  ) {\n    if (!cssOnly) {\n      const size = {\n        width: this.width,\n        height: this.height,\n        ...(dimensions as Partial<TSize>),\n      };\n      this.elements.setDimensions(size, this.getRetinaScaling());\n      this.hasLostContext = true;\n      this.width = size.width;\n      this.height = size.height;\n    }\n    if (!backstoreOnly) {\n      this.elements.setCSSDimensions(dimensions);\n    }\n\n    this.calcOffset();\n  }\n\n  /**\n   * Sets dimensions (width, height) of this canvas instance. when options.cssOnly flag active you should also supply the unit of measure (px/%/em)\n   * @param {Object}        dimensions                    Object with width/height properties\n   * @param {Number|String} [dimensions.width]            Width of canvas element\n   * @param {Number|String} [dimensions.height]           Height of canvas element\n   * @param {Object}        [options]                     Options object\n   * @param {Boolean}       [options.backstoreOnly=false] Set the given dimensions only as canvas backstore dimensions\n   * @param {Boolean}       [options.cssOnly=false]       Set the given dimensions only as css dimensions\n   */\n  setDimensions(\n    dimensions: Partial<CSSDimensions>,\n    options?: { cssOnly?: true; backstoreOnly?: false }\n  ): void;\n  setDimensions(\n    dimensions: Partial<TSize>,\n    options?: { backstoreOnly?: true; cssOnly?: false }\n  ): void;\n  setDimensions(dimensions: Partial<TSize>, options?: never): void;\n  setDimensions(\n    dimensions: Partial<TSize | CSSDimensions>,\n    options?: TCanvasSizeOptions\n  ) {\n    this._setDimensionsImpl(dimensions, options);\n    if (!options || !options.cssOnly) {\n      this.requestRenderAll();\n    }\n  }\n\n  /**\n   * Returns canvas zoom level\n   * @return {Number}\n   */\n  getZoom() {\n    return this.viewportTransform[0];\n  }\n\n  /**\n   * Sets viewport transformation of this canvas instance\n   * @param {Array} vpt a Canvas 2D API transform matrix\n   */\n  setViewportTransform(vpt: TMat2D) {\n    const backgroundObject = this.backgroundImage,\n      overlayObject = this.overlayImage,\n      len = this._objects.length;\n\n    this.viewportTransform = vpt;\n    for (let i = 0; i < len; i++) {\n      const object = this._objects[i];\n      object.group || object.setCoords();\n    }\n    if (backgroundObject) {\n      backgroundObject.setCoords();\n    }\n    if (overlayObject) {\n      overlayObject.setCoords();\n    }\n    this.calcViewportBoundaries();\n    this.renderOnAddRemove && this.requestRenderAll();\n  }\n\n  /**\n   * Sets zoom level of this canvas instance, the zoom centered around point\n   * meaning that following zoom to point with the same point will have the visual\n   * effect of the zoom originating from that point. The point won't move.\n   * It has nothing to do with canvas center or visual center of the viewport.\n   * @param {Point} point to zoom with respect to\n   * @param {Number} value to set zoom to, less than 1 zooms out\n   */\n  zoomToPoint(point: Point, value: number) {\n    // TODO: just change the scale, preserve other transformations\n    const before = point,\n      vpt: TMat2D = [...this.viewportTransform];\n    const newPoint = transformPoint(point, invertTransform(vpt));\n    vpt[0] = value;\n    vpt[3] = value;\n    const after = transformPoint(newPoint, vpt);\n    vpt[4] += before.x - after.x;\n    vpt[5] += before.y - after.y;\n    this.setViewportTransform(vpt);\n  }\n\n  /**\n   * Sets zoom level of this canvas instance\n   * @param {Number} value to set zoom to, less than 1 zooms out\n   */\n  setZoom(value: number) {\n    this.zoomToPoint(new Point(0, 0), value);\n  }\n\n  /**\n   * Pan viewport so as to place point at top left corner of canvas\n   * @param {Point} point to move to\n   */\n  absolutePan(point: Point) {\n    const vpt: TMat2D = [...this.viewportTransform];\n    vpt[4] = -point.x;\n    vpt[5] = -point.y;\n    return this.setViewportTransform(vpt);\n  }\n\n  /**\n   * Pans viewpoint relatively\n   * @param {Point} point (position vector) to move by\n   */\n  relativePan(point: Point) {\n    return this.absolutePan(\n      new Point(\n        -point.x - this.viewportTransform[4],\n        -point.y - this.viewportTransform[5]\n      )\n    );\n  }\n\n  /**\n   * Returns &lt;canvas> element corresponding to this instance\n   * @return {HTMLCanvasElement}\n   */\n  getElement(): HTMLCanvasElement {\n    return this.elements.lower.el;\n  }\n\n  /**\n   * Clears specified context of canvas element\n   * @param {CanvasRenderingContext2D} ctx Context to clear\n   */\n  clearContext(ctx: CanvasRenderingContext2D) {\n    ctx.clearRect(0, 0, this.width, this.height);\n  }\n\n  /**\n   * Returns context of canvas where objects are drawn\n   * @return {CanvasRenderingContext2D}\n   */\n  getContext(): CanvasRenderingContext2D {\n    return this.elements.lower.ctx;\n  }\n\n  /**\n   * Clears all contexts (background, main, top) of an instance\n   */\n  clear() {\n    this.remove(...this.getObjects());\n    this.backgroundImage = undefined;\n    this.overlayImage = undefined;\n    this.backgroundColor = '';\n    this.overlayColor = '';\n    this.clearContext(this.getContext());\n    this.fire('canvas:cleared');\n    this.renderOnAddRemove && this.requestRenderAll();\n  }\n\n  /**\n   * Renders the canvas\n   */\n  renderAll() {\n    this.cancelRequestedRender();\n    if (this.destroyed) {\n      return;\n    }\n    this.renderCanvas(this.getContext(), this._objects);\n  }\n\n  /**\n   * Function created to be instance bound at initialization\n   * used in requestAnimationFrame rendering\n   * Let the fabricJS call it. If you call it manually you could have more\n   * animationFrame stacking on to of each other\n   * for an imperative rendering, use canvas.renderAll\n   * @private\n   */\n  renderAndReset() {\n    this.nextRenderHandle = 0;\n    this.renderAll();\n  }\n\n  /**\n   * Append a renderAll request to next animation frame.\n   * unless one is already in progress, in that case nothing is done\n   * a boolean flag will avoid appending more.\n   */\n  requestRenderAll() {\n    if (!this.nextRenderHandle && !this.disposed && !this.destroyed) {\n      this.nextRenderHandle = requestAnimFrame(() => this.renderAndReset());\n    }\n  }\n\n  /**\n   * Calculate the position of the 4 corner of canvas with current viewportTransform.\n   * helps to determinate when an object is in the current rendering viewport\n   */\n  calcViewportBoundaries(): TCornerPoint {\n    const width = this.width,\n      height = this.height,\n      iVpt = invertTransform(this.viewportTransform),\n      a = transformPoint({ x: 0, y: 0 }, iVpt),\n      b = transformPoint({ x: width, y: height }, iVpt),\n      // we don't support vpt flipping\n      // but the code is robust enough to mostly work with flipping\n      min = a.min(b),\n      max = a.max(b);\n    return (this.vptCoords = {\n      tl: min,\n      tr: new Point(max.x, min.y),\n      bl: new Point(min.x, max.y),\n      br: max,\n    });\n  }\n\n  cancelRequestedRender() {\n    if (this.nextRenderHandle) {\n      cancelAnimFrame(this.nextRenderHandle);\n      this.nextRenderHandle = 0;\n    }\n  }\n\n  drawControls(ctx: CanvasRenderingContext2D) {\n    // Static canvas has no controls\n  }\n\n  /**\n   * Renders background, objects, overlay and controls.\n   * @param {CanvasRenderingContext2D} ctx\n   * @param {Array} objects to render\n   */\n  renderCanvas(ctx: CanvasRenderingContext2D, objects: FabricObject[]) {\n    if (this.destroyed) {\n      return;\n    }\n\n    const v = this.viewportTransform,\n      path = this.clipPath;\n    this.calcViewportBoundaries();\n    this.clearContext(ctx);\n    ctx.imageSmoothingEnabled = this.imageSmoothingEnabled;\n    // @ts-expect-error node-canvas stuff\n    ctx.patternQuality = 'best';\n    this.fire('before:render', { ctx });\n    this._renderBackground(ctx);\n\n    ctx.save();\n    //apply viewport transform once for all rendering process\n    ctx.transform(v[0], v[1], v[2], v[3], v[4], v[5]);\n    this._renderObjects(ctx, objects);\n    ctx.restore();\n    if (!this.controlsAboveOverlay) {\n      this.drawControls(ctx);\n    }\n    if (path) {\n      path._set('canvas', this);\n      // needed to setup a couple of variables\n      path.shouldCache();\n      path._transformDone = true;\n      path.renderCache({ forClipping: true });\n      this.drawClipPathOnCanvas(ctx, path as TCachedFabricObject);\n    }\n    this._renderOverlay(ctx);\n    if (this.controlsAboveOverlay) {\n      this.drawControls(ctx);\n    }\n    this.fire('after:render', { ctx });\n\n    if (this.__cleanupTask) {\n      this.__cleanupTask();\n      this.__cleanupTask = undefined;\n    }\n  }\n\n  /**\n   * Paint the cached clipPath on the lowerCanvasEl\n   * @param {CanvasRenderingContext2D} ctx Context to render on\n   */\n  drawClipPathOnCanvas(\n    ctx: CanvasRenderingContext2D,\n    clipPath: TCachedFabricObject\n  ) {\n    const v = this.viewportTransform;\n    ctx.save();\n    ctx.transform(...v);\n    // DEBUG: uncomment this line, comment the following\n    // ctx.globalAlpha = 0.4;\n    ctx.globalCompositeOperation = 'destination-in';\n    clipPath.transform(ctx);\n    ctx.scale(1 / clipPath.zoomX, 1 / clipPath.zoomY);\n    ctx.drawImage(\n      clipPath._cacheCanvas,\n      -clipPath.cacheTranslationX,\n      -clipPath.cacheTranslationY\n    );\n    ctx.restore();\n  }\n\n  /**\n   * @private\n   * @param {CanvasRenderingContext2D} ctx Context to render on\n   * @param {Array} objects to render\n   */\n  _renderObjects(ctx: CanvasRenderingContext2D, objects: FabricObject[]) {\n    for (let i = 0, len = objects.length; i < len; ++i) {\n      objects[i] && objects[i].render(ctx);\n    }\n  }\n\n  /**\n   * @private\n   * @param {CanvasRenderingContext2D} ctx Context to render on\n   * @param {string} property 'background' or 'overlay'\n   */\n  _renderBackgroundOrOverlay(\n    ctx: CanvasRenderingContext2D,\n    property: 'background' | 'overlay'\n  ) {\n    const fill = this[`${property}Color`],\n      object = this[`${property}Image`],\n      v = this.viewportTransform,\n      needsVpt = this[`${property}Vpt`];\n    if (!fill && !object) {\n      return;\n    }\n    const isAFiller = isFiller(fill);\n    if (fill) {\n      ctx.save();\n      ctx.beginPath();\n      ctx.moveTo(0, 0);\n      ctx.lineTo(this.width, 0);\n      ctx.lineTo(this.width, this.height);\n      ctx.lineTo(0, this.height);\n      ctx.closePath();\n      ctx.fillStyle = isAFiller ? fill.toLive(ctx /* this */)! : fill;\n      if (needsVpt) {\n        ctx.transform(...v);\n      }\n      if (isAFiller) {\n        ctx.transform(1, 0, 0, 1, fill.offsetX || 0, fill.offsetY || 0);\n        const m = ((fill as Gradient<'linear'>).gradientTransform ||\n          (fill as Pattern).patternTransform) as TMat2D;\n        m && ctx.transform(...m);\n      }\n      ctx.fill();\n      ctx.restore();\n    }\n    if (object) {\n      ctx.save();\n      const { skipOffscreen } = this;\n      // if the object doesn't move with the viewport,\n      // the offscreen concept does not apply;\n      this.skipOffscreen = needsVpt;\n      if (needsVpt) {\n        ctx.transform(...v);\n      }\n      object.render(ctx);\n      this.skipOffscreen = skipOffscreen;\n      ctx.restore();\n    }\n  }\n\n  /**\n   * @private\n   * @param {CanvasRenderingContext2D} ctx Context to render on\n   */\n  _renderBackground(ctx: CanvasRenderingContext2D) {\n    this._renderBackgroundOrOverlay(ctx, 'background');\n  }\n\n  /**\n   * @private\n   * @param {CanvasRenderingContext2D} ctx Context to render on\n   */\n  _renderOverlay(ctx: CanvasRenderingContext2D) {\n    this._renderBackgroundOrOverlay(ctx, 'overlay');\n  }\n\n  /**\n   * Returns coordinates of a center of canvas.\n   * Returned value is an object with top and left properties\n   * @return {Object} object with \"top\" and \"left\" number values\n   * @deprecated migrate to `getCenterPoint`\n   */\n  getCenter() {\n    return {\n      top: this.height / 2,\n      left: this.width / 2,\n    };\n  }\n\n  /**\n   * Returns coordinates of a center of canvas.\n   * @return {Point}\n   */\n  getCenterPoint() {\n    return new Point(this.width / 2, this.height / 2);\n  }\n\n  /**\n   * Centers object horizontally in the canvas\n   */\n  centerObjectH(object: FabricObject) {\n    return this._centerObject(\n      object,\n      new Point(this.getCenterPoint().x, object.getCenterPoint().y)\n    );\n  }\n\n  /**\n   * Centers object vertically in the canvas\n   * @param {FabricObject} object Object to center vertically\n   */\n  centerObjectV(object: FabricObject) {\n    return this._centerObject(\n      object,\n      new Point(object.getCenterPoint().x, this.getCenterPoint().y)\n    );\n  }\n\n  /**\n   * Centers object vertically and horizontally in the canvas\n   * @param {FabricObject} object Object to center vertically and horizontally\n   */\n  centerObject(object: FabricObject) {\n    return this._centerObject(object, this.getCenterPoint());\n  }\n\n  /**\n   * Centers object vertically and horizontally in the viewport\n   * @param {FabricObject} object Object to center vertically and horizontally\n   */\n  viewportCenterObject(object: FabricObject) {\n    return this._centerObject(object, this.getVpCenter());\n  }\n\n  /**\n   * Centers object horizontally in the viewport, object.top is unchanged\n   * @param {FabricObject} object Object to center vertically and horizontally\n   */\n  viewportCenterObjectH(object: FabricObject) {\n    return this._centerObject(\n      object,\n      new Point(this.getVpCenter().x, object.getCenterPoint().y)\n    );\n  }\n\n  /**\n   * Centers object Vertically in the viewport, object.top is unchanged\n   * @param {FabricObject} object Object to center vertically and horizontally\n   */\n  viewportCenterObjectV(object: FabricObject) {\n    return this._centerObject(\n      object,\n      new Point(object.getCenterPoint().x, this.getVpCenter().y)\n    );\n  }\n\n  /**\n   * Calculate the point in canvas that correspond to the center of actual viewport.\n   * @return {Point} vpCenter, viewport center\n   */\n  getVpCenter(): Point {\n    return transformPoint(\n      this.getCenterPoint(),\n      invertTransform(this.viewportTransform)\n    );\n  }\n\n  /**\n   * @private\n   * @param {FabricObject} object Object to center\n   * @param {Point} center Center point\n   */\n  _centerObject(object: FabricObject, center: Point) {\n    object.setXY(center, CENTER, CENTER);\n    object.setCoords();\n    this.renderOnAddRemove && this.requestRenderAll();\n  }\n\n  /**\n   * Returns dataless JSON representation of canvas\n   * @param {Array} [propertiesToInclude] Any properties that you might want to additionally include in the output\n   * @return {String} json string\n   */\n  toDatalessJSON(propertiesToInclude?: string[]) {\n    return this.toDatalessObject(propertiesToInclude);\n  }\n\n  /**\n   * Returns object representation of canvas\n   * @param {Array} [propertiesToInclude] Any properties that you might want to additionally include in the output\n   * @return {Object} object representation of an instance\n   */\n  toObject(propertiesToInclude?: string[]) {\n    return this._toObjectMethod('toObject', propertiesToInclude);\n  }\n\n  /**\n   * Returns Object representation of canvas\n   * this alias is provided because if you call JSON.stringify on an instance,\n   * the toJSON object will be invoked if it exists.\n   * Having a toJSON method means you can do JSON.stringify(myCanvas)\n   * @return {Object} JSON compatible object\n   * @tutorial {@link http://fabricjs.com/fabric-intro-part-3#serialization}\n   * @see {@link http://jsfiddle.net/fabricjs/pec86/|jsFiddle demo}\n   * @example <caption>JSON without additional properties</caption>\n   * var json = canvas.toJSON();\n   * @example <caption>JSON with additional properties included</caption>\n   * var json = canvas.toJSON(['lockMovementX', 'lockMovementY', 'lockRotation', 'lockScalingX', 'lockScalingY']);\n   * @example <caption>JSON without default values</caption>\n   * var json = canvas.toJSON();\n   */\n  toJSON() {\n    return this.toObject();\n  }\n\n  /**\n   * Returns dataless object representation of canvas\n   * @param {Array} [propertiesToInclude] Any properties that you might want to additionally include in the output\n   * @return {Object} object representation of an instance\n   */\n  toDatalessObject(propertiesToInclude?: string[]) {\n    return this._toObjectMethod('toDatalessObject', propertiesToInclude);\n  }\n\n  /**\n   * @private\n   */\n  _toObjectMethod(\n    methodName: TValidToObjectMethod,\n    propertiesToInclude?: string[]\n  ) {\n    const clipPath = this.clipPath;\n    const clipPathData =\n      clipPath && !clipPath.excludeFromExport\n        ? this._toObject(clipPath, methodName, propertiesToInclude)\n        : null;\n    return {\n      version: VERSION,\n      ...pick(this, propertiesToInclude as (keyof this)[]),\n      objects: this._objects\n        .filter((object) => !object.excludeFromExport)\n        .map((instance) =>\n          this._toObject(instance, methodName, propertiesToInclude)\n        ),\n      ...this.__serializeBgOverlay(methodName, propertiesToInclude),\n      ...(clipPathData ? { clipPath: clipPathData } : null),\n    };\n  }\n\n  /**\n   * @private\n   */\n  protected _toObject(\n    instance: FabricObject,\n    methodName: TValidToObjectMethod,\n    propertiesToInclude?: string[]\n  ) {\n    let originalValue;\n\n    if (!this.includeDefaultValues) {\n      originalValue = instance.includeDefaultValues;\n      instance.includeDefaultValues = false;\n    }\n\n    const object = instance[methodName](propertiesToInclude);\n    if (!this.includeDefaultValues) {\n      instance.includeDefaultValues = !!originalValue;\n    }\n    return object;\n  }\n\n  /**\n   * @private\n   */\n  __serializeBgOverlay(\n    methodName: TValidToObjectMethod,\n    propertiesToInclude?: string[]\n  ) {\n    const data: any = {},\n      bgImage = this.backgroundImage,\n      overlayImage = this.overlayImage,\n      bgColor = this.backgroundColor,\n      overlayColor = this.overlayColor;\n\n    if (isFiller(bgColor)) {\n      if (!bgColor.excludeFromExport) {\n        data.background = bgColor.toObject(propertiesToInclude);\n      }\n    } else if (bgColor) {\n      data.background = bgColor;\n    }\n\n    if (isFiller(overlayColor)) {\n      if (!overlayColor.excludeFromExport) {\n        data.overlay = overlayColor.toObject(propertiesToInclude);\n      }\n    } else if (overlayColor) {\n      data.overlay = overlayColor;\n    }\n\n    if (bgImage && !bgImage.excludeFromExport) {\n      data.backgroundImage = this._toObject(\n        bgImage,\n        methodName,\n        propertiesToInclude\n      );\n    }\n    if (overlayImage && !overlayImage.excludeFromExport) {\n      data.overlayImage = this._toObject(\n        overlayImage,\n        methodName,\n        propertiesToInclude\n      );\n    }\n\n    return data;\n  }\n\n  /* _TO_SVG_START_ */\n\n  declare svgViewportTransformation: boolean;\n\n  /**\n   * Returns SVG representation of canvas\n   * @function\n   * @param {Object} [options] Options object for SVG output\n   * @param {Boolean} [options.suppressPreamble=false] If true xml tag is not included\n   * @param {Object} [options.viewBox] SVG viewbox object\n   * @param {Number} [options.viewBox.x] x-coordinate of viewbox\n   * @param {Number} [options.viewBox.y] y-coordinate of viewbox\n   * @param {Number} [options.viewBox.width] Width of viewbox\n   * @param {Number} [options.viewBox.height] Height of viewbox\n   * @param {String} [options.encoding=UTF-8] Encoding of SVG output\n   * @param {String} [options.width] desired width of svg with or without units\n   * @param {String} [options.height] desired height of svg with or without units\n   * @param {Function} [reviver] Method for further parsing of svg elements, called after each fabric object converted into svg representation.\n   * @return {String} SVG string\n   * @tutorial {@link http://fabricjs.com/fabric-intro-part-3#serialization}\n   * @see {@link http://jsfiddle.net/fabricjs/jQ3ZZ/|jsFiddle demo}\n   * @example <caption>Normal SVG output</caption>\n   * var svg = canvas.toSVG();\n   * @example <caption>SVG output without preamble (without &lt;?xml ../>)</caption>\n   * var svg = canvas.toSVG({suppressPreamble: true});\n   * @example <caption>SVG output with viewBox attribute</caption>\n   * var svg = canvas.toSVG({\n   *   viewBox: {\n   *     x: 100,\n   *     y: 100,\n   *     width: 200,\n   *     height: 300\n   *   }\n   * });\n   * @example <caption>SVG output with different encoding (default: UTF-8)</caption>\n   * var svg = canvas.toSVG({encoding: 'ISO-8859-1'});\n   * @example <caption>Modify SVG output with reviver function</caption>\n   * var svg = canvas.toSVG(null, function(svg) {\n   *   return svg.replace('stroke-dasharray: ; stroke-linecap: butt; stroke-linejoin: miter; stroke-miterlimit: 10; ', '');\n   * });\n   */\n  toSVG(options: TSVGExportOptions = {}, reviver: TSVGReviver) {\n    options.reviver = reviver;\n    const markup: string[] = [];\n\n    this._setSVGPreamble(markup, options);\n    this._setSVGHeader(markup, options);\n    if (this.clipPath) {\n      markup.push(`<g clip-path=\"url(#${this.clipPath.clipPathId})\" >\\n`);\n    }\n    this._setSVGBgOverlayColor(markup, 'background');\n    this._setSVGBgOverlayImage(markup, 'backgroundImage', reviver);\n    this._setSVGObjects(markup, reviver);\n    if (this.clipPath) {\n      markup.push('</g>\\n');\n    }\n    this._setSVGBgOverlayColor(markup, 'overlay');\n    this._setSVGBgOverlayImage(markup, 'overlayImage', reviver);\n\n    markup.push('</svg>');\n\n    return markup.join('');\n  }\n\n  /**\n   * @private\n   */\n  _setSVGPreamble(markup: string[], options: TSVGExportOptions): void {\n    if (options.suppressPreamble) {\n      return;\n    }\n    markup.push(\n      '<?xml version=\"1.0\" encoding=\"',\n      options.encoding || 'UTF-8',\n      '\" standalone=\"no\" ?>\\n',\n      '<!DOCTYPE svg PUBLIC \"-//W3C//DTD SVG 1.1//EN\" ',\n      '\"http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd\">\\n'\n    );\n  }\n\n  /**\n   * @private\n   */\n  _setSVGHeader(markup: string[], options: TSVGExportOptions): void {\n    const width = options.width || `${this.width}`,\n      height = options.height || `${this.height}`,\n      NUM_FRACTION_DIGITS = config.NUM_FRACTION_DIGITS,\n      optViewBox = options.viewBox;\n    let viewBox: string;\n    if (optViewBox) {\n      viewBox = `viewBox=\"${optViewBox.x} ${optViewBox.y} ${optViewBox.width} ${optViewBox.height}\" `;\n    } else if (this.svgViewportTransformation) {\n      const vpt = this.viewportTransform;\n      viewBox = `viewBox=\"${toFixed(\n        -vpt[4] / vpt[0],\n        NUM_FRACTION_DIGITS\n      )} ${toFixed(-vpt[5] / vpt[3], NUM_FRACTION_DIGITS)} ${toFixed(\n        this.width / vpt[0],\n        NUM_FRACTION_DIGITS\n      )} ${toFixed(this.height / vpt[3], NUM_FRACTION_DIGITS)}\" `;\n    } else {\n      viewBox = `viewBox=\"0 0 ${this.width} ${this.height}\" `;\n    }\n\n    markup.push(\n      '<svg ',\n      'xmlns=\"http://www.w3.org/2000/svg\" ',\n      'xmlns:xlink=\"http://www.w3.org/1999/xlink\" ',\n      'version=\"1.1\" ',\n      'width=\"',\n      width,\n      '\" ',\n      'height=\"',\n      height,\n      '\" ',\n      viewBox,\n      'xml:space=\"preserve\">\\n',\n      '<desc>Created with Fabric.js ',\n      VERSION,\n      '</desc>\\n',\n      '<defs>\\n',\n      this.createSVGFontFacesMarkup(),\n      this.createSVGRefElementsMarkup(),\n      this.createSVGClipPathMarkup(options),\n      '</defs>\\n'\n    );\n  }\n\n  createSVGClipPathMarkup(options: TSVGExportOptions): string {\n    const clipPath = this.clipPath;\n    if (clipPath) {\n      clipPath.clipPathId = `CLIPPATH_${uid()}`;\n      return `<clipPath id=\"${clipPath.clipPathId}\" >\\n${clipPath.toClipPathSVG(\n        options.reviver\n      )}</clipPath>\\n`;\n    }\n    return '';\n  }\n\n  /**\n   * Creates markup containing SVG referenced elements like patterns, gradients etc.\n   * @return {String}\n   */\n  createSVGRefElementsMarkup(): string {\n    return (['background', 'overlay'] as const)\n      .map((prop) => {\n        const fill = this[`${prop}Color`];\n        if (isFiller(fill)) {\n          const shouldTransform = this[`${prop}Vpt`],\n            vpt = this.viewportTransform,\n            object = {\n              // otherwise circular dependency\n              isType: () => false,\n              width: this.width / (shouldTransform ? vpt[0] : 1),\n              height: this.height / (shouldTransform ? vpt[3] : 1),\n            };\n          return fill.toSVG(object as FabricObject, {\n            additionalTransform: shouldTransform ? matrixToSVG(vpt) : '',\n          });\n        }\n      })\n      .join('');\n  }\n\n  /**\n   * Creates markup containing SVG font faces,\n   * font URLs for font faces must be collected by developers\n   * and are not extracted from the DOM by fabricjs\n   * @param {Array} objects Array of fabric objects\n   * @return {String}\n   */\n  createSVGFontFacesMarkup(): string {\n    const objects: FabricObject[] = [],\n      fontList: Record<string, boolean> = {},\n      fontPaths = config.fontPaths;\n\n    this._objects.forEach(function add(object) {\n      objects.push(object);\n      if (isCollection(object)) {\n        object._objects.forEach(add);\n      }\n    });\n\n    objects.forEach((obj) => {\n      if (!isTextObject(obj)) {\n        return;\n      }\n      const { styles, fontFamily } = obj;\n      if (fontList[fontFamily] || !fontPaths[fontFamily]) {\n        return;\n      }\n      fontList[fontFamily] = true;\n      if (!styles) {\n        return;\n      }\n      Object.values(styles).forEach((styleRow) => {\n        Object.values(styleRow).forEach(({ fontFamily = '' }) => {\n          if (!fontList[fontFamily] && fontPaths[fontFamily]) {\n            fontList[fontFamily] = true;\n          }\n        });\n      });\n    });\n\n    const fontListMarkup = Object.keys(fontList)\n      .map(\n        (fontFamily) =>\n          `\\t\\t@font-face {\\n\\t\\t\\tfont-family: '${fontFamily}';\\n\\t\\t\\tsrc: url('${fontPaths[fontFamily]}');\\n\\t\\t}\\n`\n      )\n      .join('');\n\n    if (fontListMarkup) {\n      return `\\t<style type=\"text/css\"><![CDATA[\\n${fontListMarkup}]]></style>\\n`;\n    }\n    return '';\n  }\n\n  /**\n   * @private\n   */\n  _setSVGObjects(markup: string[], reviver: TSVGReviver) {\n    this.forEachObject((fabricObject) => {\n      if (fabricObject.excludeFromExport) {\n        return;\n      }\n      this._setSVGObject(markup, fabricObject, reviver);\n    });\n  }\n\n  /**\n   * This is its own function because the Canvas ( non static ) requires extra code here\n   * @private\n   */\n  _setSVGObject(\n    markup: string[],\n    instance: FabricObject,\n    reviver: TSVGReviver\n  ) {\n    markup.push(instance.toSVG(reviver));\n  }\n\n  /**\n   * @private\n   */\n  _setSVGBgOverlayImage(\n    markup: string[],\n    property: 'overlayImage' | 'backgroundImage',\n    reviver: TSVGReviver\n  ) {\n    const bgOrOverlay = this[property];\n    if (bgOrOverlay && !bgOrOverlay.excludeFromExport && bgOrOverlay.toSVG) {\n      markup.push(bgOrOverlay.toSVG(reviver));\n    }\n  }\n\n  /**\n   * @TODO this seems to handle patterns but fail at gradients.\n   * @private\n   */\n  _setSVGBgOverlayColor(markup: string[], property: 'background' | 'overlay') {\n    const filler = this[`${property}Color`];\n    if (!filler) {\n      return;\n    }\n    if (isFiller(filler)) {\n      const repeat = (filler as Pattern).repeat || '',\n        finalWidth = this.width,\n        finalHeight = this.height,\n        shouldInvert = this[`${property}Vpt`],\n        additionalTransform = shouldInvert\n          ? matrixToSVG(invertTransform(this.viewportTransform))\n          : '';\n      markup.push(\n        `<rect transform=\"${additionalTransform} translate(${finalWidth / 2},${\n          finalHeight / 2\n        })\" x=\"${filler.offsetX - finalWidth / 2}\" y=\"${\n          filler.offsetY - finalHeight / 2\n        }\" width=\"${\n          (repeat === 'repeat-y' || repeat === 'no-repeat') && isPattern(filler)\n            ? (filler.source as HTMLImageElement).width\n            : finalWidth\n        }\" height=\"${\n          (repeat === 'repeat-x' || repeat === 'no-repeat') && isPattern(filler)\n            ? (filler.source as HTMLImageElement).height\n            : finalHeight\n        }\" fill=\"url(#SVGID_${filler.id})\"></rect>\\n`\n      );\n    } else {\n      markup.push(\n        '<rect x=\"0\" y=\"0\" width=\"100%\" height=\"100%\" ',\n        'fill=\"',\n        filler,\n        '\"',\n        '></rect>\\n'\n      );\n    }\n  }\n  /* _TO_SVG_END_ */\n\n  /**\n   * Populates canvas with data from the specified JSON.\n   * JSON format must conform to the one of {@link fabric.Canvas#toJSON}\n   *\n   * **IMPORTANT**: It is recommended to abort loading tasks before calling this method to prevent race conditions and unnecessary networking\n   *\n   * @param {String|Object} json JSON string or object\n   * @param {Function} [reviver] Method for further parsing of JSON elements, called after each fabric object created.\n   * @param {Object} [options] options\n   * @param {AbortSignal} [options.signal] see https://developer.mozilla.org/en-US/docs/Web/API/AbortController/signal\n   * @return {Promise<Canvas | StaticCanvas>} instance\n   * @tutorial {@link http://fabricjs.com/fabric-intro-part-3#deserialization}\n   * @see {@link http://jsfiddle.net/fabricjs/fmgXt/|jsFiddle demo}\n   * @example <caption>loadFromJSON</caption>\n   * canvas.loadFromJSON(json).then((canvas) => canvas.requestRenderAll());\n   * @example <caption>loadFromJSON with reviver</caption>\n   * canvas.loadFromJSON(json, function(o, object) {\n   *   // `o` = json object\n   *   // `object` = fabric.Object instance\n   *   // ... do some stuff ...\n   * }).then((canvas) => {\n   *   ... canvas is restored, add your code.\n   * });\n   *\n   */\n  loadFromJSON(\n    json: string | Record<string, any>,\n    reviver?: EnlivenObjectOptions['reviver'],\n    { signal }: Abortable = {}\n  ): Promise<this> {\n    if (!json) {\n      return Promise.reject(new FabricError('`json` is undefined'));\n    }\n\n    // parse json if it wasn't already\n    const serialized = typeof json === 'string' ? JSON.parse(json) : json;\n    const {\n      objects = [],\n      backgroundImage,\n      background,\n      overlayImage,\n      overlay,\n      clipPath,\n    } = serialized;\n    const renderOnAddRemove = this.renderOnAddRemove;\n    this.renderOnAddRemove = false;\n\n    return Promise.all([\n      enlivenObjects<FabricObject>(objects, {\n        reviver,\n        signal,\n      }),\n      enlivenObjectEnlivables(\n        {\n          backgroundImage,\n          backgroundColor: background,\n          overlayImage,\n          overlayColor: overlay,\n          clipPath,\n        },\n        { signal }\n      ),\n    ]).then(([enlived, enlivedMap]) => {\n      this.clear();\n      this.add(...enlived);\n      this.set(serialized);\n      this.set(enlivedMap);\n      this.renderOnAddRemove = renderOnAddRemove;\n      return this;\n    });\n  }\n\n  /**\n   * Clones canvas instance\n   * @param {string[]} [properties] Array of properties to include in the cloned canvas and children\n   */\n  clone(properties: string[]) {\n    const data = this.toObject(properties);\n    const canvas = this.cloneWithoutData();\n    return canvas.loadFromJSON(data);\n  }\n\n  /**\n   * Clones canvas instance without cloning existing data.\n   * This essentially copies canvas dimensions since loadFromJSON does not affect canvas size.\n   */\n  cloneWithoutData() {\n    const el = createCanvasElement();\n    el.width = this.width;\n    el.height = this.height;\n    return new (this.constructor as Constructor<this>)(el);\n  }\n\n  /**\n   * Exports canvas element to a dataurl image. Note that when multiplier is used, cropping is scaled appropriately\n   * @param {Object} [options] Options object\n   * @param {String} [options.format=png] The format of the output image. Either \"jpeg\" or \"png\"\n   * @param {Number} [options.quality=1] Quality level (0..1). Only used for jpeg.\n   * @param {Number} [options.multiplier=1] Multiplier to scale by, to have consistent\n   * @param {Number} [options.left] Cropping left offset. Introduced in v1.2.14\n   * @param {Number} [options.top] Cropping top offset. Introduced in v1.2.14\n   * @param {Number} [options.width] Cropping width. Introduced in v1.2.14\n   * @param {Number} [options.height] Cropping height. Introduced in v1.2.14\n   * @param {Boolean} [options.enableRetinaScaling] Enable retina scaling for clone image. Introduce in 2.0.0\n   * @param {(object: fabric.Object) => boolean} [options.filter] Function to filter objects.\n   * @return {String} Returns a data: URL containing a representation of the object in the format specified by options.format\n   * @see {@link https://jsfiddle.net/xsjua1rd/ demo}\n   * @example <caption>Generate jpeg dataURL with lower quality</caption>\n   * var dataURL = canvas.toDataURL({\n   *   format: 'jpeg',\n   *   quality: 0.8\n   * });\n   * @example <caption>Generate cropped png dataURL (clipping of canvas)</caption>\n   * var dataURL = canvas.toDataURL({\n   *   format: 'png',\n   *   left: 100,\n   *   top: 100,\n   *   width: 200,\n   *   height: 200\n   * });\n   * @example <caption>Generate double scaled png dataURL</caption>\n   * var dataURL = canvas.toDataURL({\n   *   format: 'png',\n   *   multiplier: 2\n   * });\n   * @example <caption>Generate dataURL with objects that overlap a specified object</caption>\n   * var myObject;\n   * var dataURL = canvas.toDataURL({\n   *   filter: (object) => object.isContainedWithinObject(myObject) || object.intersectsWithObject(myObject)\n   * });\n   */\n  toDataURL(options = {} as TDataUrlOptions): string {\n    const {\n      format = 'png',\n      quality = 1,\n      multiplier = 1,\n      enableRetinaScaling = false,\n    } = options;\n    const finalMultiplier =\n      multiplier * (enableRetinaScaling ? this.getRetinaScaling() : 1);\n\n    return toDataURL(\n      this.toCanvasElement(finalMultiplier, options),\n      format,\n      quality\n    );\n  }\n\n  /**\n   * Create a new HTMLCanvas element painted with the current canvas content.\n   * No need to resize the actual one or repaint it.\n   * Will transfer object ownership to a new canvas, paint it, and set everything back.\n   * This is an intermediary step used to get to a dataUrl but also it is useful to\n   * create quick image copies of a canvas without passing for the dataUrl string\n   * @param {Number} [multiplier] a zoom factor.\n   * @param {Object} [options] Cropping informations\n   * @param {Number} [options.left] Cropping left offset.\n   * @param {Number} [options.top] Cropping top offset.\n   * @param {Number} [options.width] Cropping width.\n   * @param {Number} [options.height] Cropping height.\n   * @param {(object: fabric.Object) => boolean} [options.filter] Function to filter objects.\n   */\n  toCanvasElement(\n    multiplier = 1,\n    { width, height, left, top, filter } = {} as TToCanvasElementOptions\n  ): HTMLCanvasElement {\n    const scaledWidth = (width || this.width) * multiplier,\n      scaledHeight = (height || this.height) * multiplier,\n      zoom = this.getZoom(),\n      originalWidth = this.width,\n      originalHeight = this.height,\n      newZoom = zoom * multiplier,\n      vp = this.viewportTransform,\n      translateX = (vp[4] - (left || 0)) * multiplier,\n      translateY = (vp[5] - (top || 0)) * multiplier,\n      newVp = [newZoom, 0, 0, newZoom, translateX, translateY] as TMat2D,\n      originalRetina = this.enableRetinaScaling,\n      canvasEl = createCanvasElement(),\n      objectsToRender = filter\n        ? this._objects.filter((obj) => filter(obj))\n        : this._objects;\n    canvasEl.width = scaledWidth;\n    canvasEl.height = scaledHeight;\n    this.enableRetinaScaling = false;\n    this.viewportTransform = newVp;\n    this.width = scaledWidth;\n    this.height = scaledHeight;\n    this.calcViewportBoundaries();\n    this.renderCanvas(canvasEl.getContext('2d')!, objectsToRender);\n    this.viewportTransform = vp;\n    this.width = originalWidth;\n    this.height = originalHeight;\n    this.calcViewportBoundaries();\n    this.enableRetinaScaling = originalRetina;\n    return canvasEl;\n  }\n\n  /**\n   * Waits until rendering has settled to destroy the canvas\n   * @returns {Promise<boolean>} a promise resolving to `true` once the canvas has been destroyed or to `false` if the canvas has was already destroyed\n   * @throws if aborted by a consequent call\n   */\n  dispose() {\n    !this.disposed &&\n      this.elements.cleanupDOM({ width: this.width, height: this.height });\n    runningAnimations.cancelByCanvas(this);\n    this.disposed = true;\n    return new Promise<boolean>((resolve, reject) => {\n      const task = () => {\n        this.destroy();\n        resolve(true);\n      };\n      task.kill = reject;\n      if (this.__cleanupTask) {\n        this.__cleanupTask.kill('aborted');\n      }\n\n      if (this.destroyed) {\n        resolve(false);\n      } else if (this.nextRenderHandle) {\n        this.__cleanupTask = task;\n      } else {\n        task();\n      }\n    });\n  }\n\n  /**\n   * Clears the canvas element, disposes objects and frees resources.\n   *\n   * Invoked as part of the **async** operation of {@link dispose}.\n   *\n   * **CAUTION**:\n   *\n   * This method is **UNSAFE**.\n   * You may encounter a race condition using it if there's a requested render.\n   * Call this method only if you are sure rendering has settled.\n   * Consider using {@link dispose} as it is **SAFE**\n   *\n   * @private\n   */\n  destroy() {\n    this.destroyed = true;\n    this.cancelRequestedRender();\n    this.forEachObject((object) => object.dispose());\n    this._objects = [];\n    if (this.backgroundImage) {\n      this.backgroundImage.dispose();\n    }\n    this.backgroundImage = undefined;\n    if (this.overlayImage) {\n      this.overlayImage.dispose();\n    }\n    this.overlayImage = undefined;\n    this.elements.dispose();\n  }\n\n  /**\n   * Returns a string representation of an instance\n   * @return {String} string representation of an instance\n   */\n  toString() {\n    return `#<Canvas (${this.complexity()}): { objects: ${\n      this._objects.length\n    } }>`;\n  }\n}\n","import type { TPointerEvent } from '../EventTypeDefs';\nimport { Point } from '../Point';\nimport { getScrollLeftTop } from './dom_misc';\n\nconst touchEvents = ['touchstart', 'touchmove', 'touchend'];\n\nfunction getTouchInfo(event: TouchEvent | MouseEvent): MouseEvent | Touch {\n  const touchProp = (event as TouchEvent).changedTouches;\n  if (touchProp && touchProp[0]) {\n    return touchProp[0];\n  }\n  return event as MouseEvent;\n}\n\nexport const getPointer = (event: TPointerEvent): Point => {\n  const element = event.target as HTMLElement,\n    scroll = getScrollLeftTop(element),\n    _evt = getTouchInfo(event);\n  return new Point(_evt.clientX + scroll.left, _evt.clientY + scroll.top);\n};\n\nexport const isTouchEvent = (event: TPointerEvent) =>\n  touchEvents.includes(event.type) ||\n  (event as PointerEvent).pointerType === 'touch';\n\nexport const stopEvent = (e: Event) => {\n  e.preventDefault();\n  e.stopPropagation();\n};\n","import type { XY } from '../../Point';\nimport { Point } from '../../Point';\nimport type { TBBox } from '../../typedefs';\n\n/**\n * Calculates bounding box (left, top, width, height) from given `points`\n * @param {XY[]} points\n * @return {Object} Object with left, top, width, height properties\n */\nexport const makeBoundingBoxFromPoints = (points: XY[]): TBBox => {\n  if (points.length === 0) {\n    return {\n      left: 0,\n      top: 0,\n      width: 0,\n      height: 0,\n    };\n  }\n\n  const { min, max } = points.reduce(\n    ({ min, max }, curr) => {\n      return {\n        min: min.min(curr),\n        max: max.max(curr),\n      };\n    },\n    { min: new Point(points[0]), max: new Point(points[0]) }\n  );\n\n  const size = max.subtract(min);\n\n  return {\n    left: min.x,\n    top: min.y,\n    width: size.x,\n    height: size.y,\n  };\n};\n","import { Point } from '../../Point';\nimport { CENTER } from '../../constants';\nimport type { FabricObject } from '../../shapes/Object/Object';\nimport type { TMat2D } from '../../typedefs';\nimport { makeBoundingBoxFromPoints } from './boundingBoxFromPoints';\nimport {\n  invertTransform,\n  multiplyTransformMatrices,\n  qrDecompose,\n} from './matrix';\n\n/**\n * given an object and a transform, apply the inverse transform to the object,\n * this is equivalent to remove from that object that transformation, so that\n * added in a space with the removed transform, the object will be the same as before.\n * Removing from an object a transform that scale by 2 is like scaling it by 1/2.\n * Removing from an object a transform that rotate by 30deg is like rotating by 30deg\n * in the opposite direction.\n * This util is used to add objects inside transformed groups or nested groups.\n * @param {FabricObject} object the object you want to transform\n * @param {TMat2D} transform the destination transform\n */\nexport const removeTransformFromObject = (\n  object: FabricObject,\n  transform: TMat2D\n) => {\n  const inverted = invertTransform(transform),\n    finalTransform = multiplyTransformMatrices(\n      inverted,\n      object.calcOwnMatrix()\n    );\n  applyTransformToObject(object, finalTransform);\n};\n\n/**\n * given an object and a transform, apply the transform to the object.\n * this is equivalent to change the space where the object is drawn.\n * Adding to an object a transform that scale by 2 is like scaling it by 2.\n * This is used when removing an object from an active selection for example.\n * @param {FabricObject} object the object you want to transform\n * @param {Array} transform the destination transform\n */\nexport const addTransformToObject = (object: FabricObject, transform: TMat2D) =>\n  applyTransformToObject(\n    object,\n    multiplyTransformMatrices(transform, object.calcOwnMatrix())\n  );\n\n/**\n * discard an object transform state and apply the one from the matrix.\n * @param {FabricObject} object the object you want to transform\n * @param {Array} transform the destination transform\n */\nexport const applyTransformToObject = (\n  object: FabricObject,\n  transform: TMat2D\n) => {\n  const { translateX, translateY, scaleX, scaleY, ...otherOptions } =\n      qrDecompose(transform),\n    center = new Point(translateX, translateY);\n  object.flipX = false;\n  object.flipY = false;\n  Object.assign(object, otherOptions);\n  object.set({ scaleX, scaleY });\n  object.setPositionByOrigin(center, CENTER, CENTER);\n};\n/**\n * reset an object transform state to neutral. Top and left are not accounted for\n * @param  {FabricObject} target object to transform\n */\nexport const resetObjectTransform = (target: FabricObject) => {\n  target.scaleX = 1;\n  target.scaleY = 1;\n  target.skewX = 0;\n  target.skewY = 0;\n  target.flipX = false;\n  target.flipY = false;\n  target.rotate(0);\n};\n\n/**\n * Extract Object transform values\n * @param  {FabricObject} target object to read from\n * @return {Object} Components of transform\n */\nexport const saveObjectTransform = (target: FabricObject) => ({\n  scaleX: target.scaleX,\n  scaleY: target.scaleY,\n  skewX: target.skewX,\n  skewY: target.skewY,\n  angle: target.angle,\n  left: target.left,\n  flipX: target.flipX,\n  flipY: target.flipY,\n  top: target.top,\n});\n\n/**\n * given a width and height, return the size of the bounding box\n * that can contains the box with width/height with applied transform.\n * Use to calculate the boxes around objects for controls.\n * @param {Number} width\n * @param {Number} height\n * @param {TMat2D} t\n * @returns {Point} size\n */\nexport const sizeAfterTransform = (\n  width: number,\n  height: number,\n  t: TMat2D\n) => {\n  const dimX = width / 2,\n    dimY = height / 2,\n    points = [\n      new Point(-dimX, -dimY),\n      new Point(dimX, -dimY),\n      new Point(-dimX, dimY),\n      new Point(dimX, dimY),\n    ].map((p) => p.transform(t)),\n    bbox = makeBoundingBoxFromPoints(points);\n  return new Point(bbox.width, bbox.height);\n};\n","import { iMatrix } from '../../constants';\nimport type { Point } from '../../Point';\nimport type { FabricObject } from '../../shapes/Object/Object';\nimport type { TMat2D } from '../../typedefs';\nimport { invertTransform, multiplyTransformMatrices } from './matrix';\nimport { applyTransformToObject } from './objectTransforms';\n\n/**\n * We are actually looking for the transformation from the destination plane to the source plane (change of basis matrix)\\\n * The object will exist on the destination plane and we want it to seem unchanged by it so we invert the destination matrix (`to`) and then apply the source matrix (`from`)\n * @param [from]\n * @param [to]\n * @returns\n */\nexport const calcPlaneChangeMatrix = (\n  from: TMat2D = iMatrix,\n  to: TMat2D = iMatrix\n) => multiplyTransformMatrices(invertTransform(to), from);\n\n/**\n * Sends a point from the source coordinate plane to the destination coordinate plane.\\\n * From the canvas/viewer's perspective the point remains unchanged.\n *\n * @example <caption>Send point from canvas plane to group plane</caption>\n * var obj = new Rect({ left: 20, top: 20, width: 60, height: 60, strokeWidth: 0 });\n * var group = new Group([obj], { strokeWidth: 0 });\n * var sentPoint1 = sendPointToPlane(new Point(50, 50), undefined, group.calcTransformMatrix());\n * var sentPoint2 = sendPointToPlane(new Point(50, 50), iMatrix, group.calcTransformMatrix());\n * console.log(sentPoint1, sentPoint2) //  both points print (0,0) which is the center of group\n *\n * @param {Point} point\n * @param {TMat2D} [from] plane matrix containing object. Passing `undefined` is equivalent to passing the identity matrix, which means `point` exists in the canvas coordinate plane.\n * @param {TMat2D} [to] destination plane matrix to contain object. Passing `undefined` means `point` should be sent to the canvas coordinate plane.\n * @returns {Point} transformed point\n */\nexport const sendPointToPlane = (\n  point: Point,\n  from: TMat2D = iMatrix,\n  to: TMat2D = iMatrix\n): Point => point.transform(calcPlaneChangeMatrix(from, to));\n\n/**\n * See {@link sendPointToPlane}\n */\nexport const sendVectorToPlane = (\n  point: Point,\n  from: TMat2D = iMatrix,\n  to: TMat2D = iMatrix\n): Point => point.transform(calcPlaneChangeMatrix(from, to), true);\n\n/**\n *\n * A util that abstracts applying transform to objects.\\\n * Sends `object` to the destination coordinate plane by applying the relevant transformations.\\\n * Changes the space/plane where `object` is drawn.\\\n * From the canvas/viewer's perspective `object` remains unchanged.\n *\n * @example <caption>Move clip path from one object to another while preserving it's appearance as viewed by canvas/viewer</caption>\n * let obj, obj2;\n * let clipPath = new Circle({ radius: 50 });\n * obj.clipPath = clipPath;\n * // render\n * sendObjectToPlane(clipPath, obj.calcTransformMatrix(), obj2.calcTransformMatrix());\n * obj.clipPath = undefined;\n * obj2.clipPath = clipPath;\n * // render, clipPath now clips obj2 but seems unchanged from the eyes of the viewer\n *\n * @example <caption>Clip an object's clip path with an existing object</caption>\n * let obj, existingObj;\n * let clipPath = new Circle({ radius: 50 });\n * obj.clipPath = clipPath;\n * let transformTo = multiplyTransformMatrices(obj.calcTransformMatrix(), clipPath.calcTransformMatrix());\n * sendObjectToPlane(existingObj, existingObj.group?.calcTransformMatrix(), transformTo);\n * clipPath.clipPath = existingObj;\n *\n * @param {FabricObject} object\n * @param {Matrix} [from] plane matrix containing object. Passing `undefined` is equivalent to passing the identity matrix, which means `object` is a direct child of canvas.\n * @param {Matrix} [to] destination plane matrix to contain object. Passing `undefined` means `object` should be sent to the canvas coordinate plane.\n * @returns {Matrix} the transform matrix that was applied to `object`\n */\nexport const sendObjectToPlane = (\n  object: FabricObject,\n  from?: TMat2D,\n  to?: TMat2D\n): TMat2D => {\n  const t = calcPlaneChangeMatrix(from, to);\n  applyTransformToObject(\n    object,\n    multiplyTransformMatrices(t, object.calcOwnMatrix())\n  );\n  return t;\n};\n","import type {\n  BasicTransformEvent,\n  TModificationEvents,\n} from '../EventTypeDefs';\n\nexport const fireEvent = (\n  eventName: TModificationEvents,\n  options: BasicTransformEvent\n) => {\n  const {\n    transform: { target },\n  } = options;\n  target.canvas?.fire(`object:${eventName}`, {\n    ...options,\n    target,\n  });\n  target.fire(eventName, options);\n};\n","import type { TOriginX, TOriginY } from '../../typedefs';\n\nconst originOffset = {\n  left: -0.5,\n  top: -0.5,\n  center: 0,\n  bottom: 0.5,\n  right: 0.5,\n};\n/**\n * Resolves origin value relative to center\n * @private\n * @param {TOriginX | TOriginY} originValue originX / originY\n * @returns number\n */\n\nexport const resolveOrigin = (\n  originValue: TOriginX | TOriginY | number\n): number =>\n  typeof originValue === 'string'\n    ? originOffset[originValue]\n    : originValue - 0.5;\n","import type {\n  TPointerEvent,\n  Transform,\n  TransformAction,\n  BasicTransformEvent,\n} from '../EventTypeDefs';\nimport { resolveOrigin } from '../util/misc/resolveOrigin';\nimport { Point } from '../Point';\nimport type { FabricObject } from '../shapes/Object/FabricObject';\nimport type { TOriginX, TOriginY } from '../typedefs';\nimport {\n  degreesToRadians,\n  radiansToDegrees,\n} from '../util/misc/radiansDegreesConversion';\nimport type { Control } from './Control';\nimport { CENTER } from '../constants';\n\nexport const NOT_ALLOWED_CURSOR = 'not-allowed';\n\n/**\n * @param {Boolean} alreadySelected true if target is already selected\n * @param {String} corner a string representing the corner ml, mr, tl ...\n * @param {Event} e Event object\n * @param {FabricObject} [target] inserted back to help overriding. Unused\n */\nexport const getActionFromCorner = (\n  alreadySelected: boolean,\n  corner: string | undefined,\n  e: TPointerEvent,\n  target: FabricObject\n) => {\n  if (!corner || !alreadySelected) {\n    return 'drag';\n  }\n  const control = target.controls[corner];\n  return control.getActionName(e, control, target);\n};\n\n/**\n * Checks if transform is centered\n * @param {Object} transform transform data\n * @return {Boolean} true if transform is centered\n */\nexport function isTransformCentered(transform: Transform) {\n  return transform.originX === CENTER && transform.originY === CENTER;\n}\n\nexport function invertOrigin(origin: TOriginX | TOriginY) {\n  return -resolveOrigin(origin) + 0.5;\n}\n\nexport const isLocked = (\n  target: FabricObject,\n  lockingKey:\n    | 'lockMovementX'\n    | 'lockMovementY'\n    | 'lockRotation'\n    | 'lockScalingX'\n    | 'lockScalingY'\n    | 'lockSkewingX'\n    | 'lockSkewingY'\n    | 'lockScalingFlip'\n) => target[lockingKey];\n\nexport const commonEventInfo: TransformAction<\n  Transform,\n  BasicTransformEvent\n> = (eventData, transform, x, y) => {\n  return {\n    e: eventData,\n    transform,\n    pointer: new Point(x, y),\n  };\n};\n\n/**\n * Combine control position and object angle to find the control direction compared\n * to the object center.\n * @param {FabricObject} fabricObject the fabric object for which we are rendering controls\n * @param {Control} control the control class\n * @return {Number} 0 - 7 a quadrant number\n */\nexport function findCornerQuadrant(\n  fabricObject: FabricObject,\n  control: Control\n): number {\n  //  angle is relative to canvas plane\n  const angle = fabricObject.getTotalAngle(),\n    cornerAngle =\n      angle + radiansToDegrees(Math.atan2(control.y, control.x)) + 360;\n  return Math.round((cornerAngle % 360) / 45);\n}\n\n/**\n * @returns the normalized point (rotated relative to center) in local coordinates\n */\nfunction normalizePoint(\n  target: FabricObject,\n  point: Point,\n  originX: TOriginX,\n  originY: TOriginY\n): Point {\n  const center = target.getRelativeCenterPoint(),\n    p =\n      typeof originX !== 'undefined' && typeof originY !== 'undefined'\n        ? target.translateToGivenOrigin(\n            center,\n            CENTER,\n            CENTER,\n            originX,\n            originY\n          )\n        : new Point(target.left, target.top),\n    p2 = target.angle\n      ? point.rotate(-degreesToRadians(target.angle), center)\n      : point;\n  return p2.subtract(p);\n}\n\n/**\n * Transforms a point to the offset from the given origin\n * @param {Object} transform\n * @param {String} originX\n * @param {String} originY\n * @param {number} x\n * @param {number} y\n * @return {Fabric.Point} the normalized point\n */\nexport function getLocalPoint(\n  { target, corner }: Transform,\n  originX: TOriginX,\n  originY: TOriginY,\n  x: number,\n  y: number\n) {\n  const control = target.controls[corner],\n    zoom = target.canvas?.getZoom() || 1,\n    padding = target.padding / zoom,\n    localPoint = normalizePoint(target, new Point(x, y), originX, originY);\n  if (localPoint.x >= padding) {\n    localPoint.x -= padding;\n  }\n  if (localPoint.x <= -padding) {\n    localPoint.x += padding;\n  }\n  if (localPoint.y >= padding) {\n    localPoint.y -= padding;\n  }\n  if (localPoint.y <= padding) {\n    localPoint.y += padding;\n  }\n  localPoint.x -= control.offsetX;\n  localPoint.y -= control.offsetY;\n  return localPoint;\n}\n","import type { TransformActionHandler } from '../EventTypeDefs';\nimport { LEFT, TOP } from '../constants';\nimport { fireEvent } from './fireEvent';\nimport { commonEventInfo, isLocked } from './util';\n\n/**\n * Action handler\n * @private\n * @param {Event} eventData javascript event that is doing the transform\n * @param {Object} transform javascript object containing a series of information around the current transform\n * @param {number} x current mouse x position, canvas normalized\n * @param {number} y current mouse y position, canvas normalized\n * @return {Boolean} true if the translation occurred\n */\nexport const dragHandler: TransformActionHandler = (\n  eventData,\n  transform,\n  x,\n  y\n) => {\n  const { target, offsetX, offsetY } = transform,\n    newLeft = x - offsetX,\n    newTop = y - offsetY,\n    moveX = !isLocked(target, 'lockMovementX') && target.left !== newLeft,\n    moveY = !isLocked(target, 'lockMovementY') && target.top !== newTop;\n  moveX && target.set(LEFT, newLeft);\n  moveY && target.set(TOP, newTop);\n  if (moveX || moveY) {\n    fireEvent('moving', commonEventInfo(eventData, transform, x, y));\n  }\n  return moveX || moveY;\n};\n","import type { TSVGReviver } from '../../typedefs';\nimport { uid } from '../../util/internals/uid';\nimport { colorPropToSVG, matrixToSVG } from '../../util/misc/svgParsing';\nimport { NONE } from '../../constants';\nimport type { FabricObject } from './FabricObject';\nimport { isFiller } from '../../util/typeAssertions';\n\nexport class FabricObjectSVGExportMixin {\n  /**\n   * When an object is being exported as SVG as a clippath, a reference inside the SVG is needed.\n   * This reference is a UID in the fabric namespace and is temporary stored here.\n   * @type {String}\n   */\n  declare clipPathId?: string;\n\n  /**\n   * Returns styles-string for svg-export\n   * @param {Boolean} skipShadow a boolean to skip shadow filter output\n   * @return {String}\n   */\n  getSvgStyles(\n    this: FabricObjectSVGExportMixin & FabricObject,\n    skipShadow?: boolean\n  ) {\n    const fillRule = this.fillRule ? this.fillRule : 'nonzero',\n      strokeWidth = this.strokeWidth ? this.strokeWidth : '0',\n      strokeDashArray = this.strokeDashArray\n        ? this.strokeDashArray.join(' ')\n        : NONE,\n      strokeDashOffset = this.strokeDashOffset ? this.strokeDashOffset : '0',\n      strokeLineCap = this.strokeLineCap ? this.strokeLineCap : 'butt',\n      strokeLineJoin = this.strokeLineJoin ? this.strokeLineJoin : 'miter',\n      strokeMiterLimit = this.strokeMiterLimit ? this.strokeMiterLimit : '4',\n      opacity = typeof this.opacity !== 'undefined' ? this.opacity : '1',\n      visibility = this.visible ? '' : ' visibility: hidden;',\n      filter = skipShadow ? '' : this.getSvgFilter(),\n      fill = colorPropToSVG('fill', this.fill),\n      stroke = colorPropToSVG('stroke', this.stroke);\n\n    return [\n      stroke,\n      'stroke-width: ',\n      strokeWidth,\n      '; ',\n      'stroke-dasharray: ',\n      strokeDashArray,\n      '; ',\n      'stroke-linecap: ',\n      strokeLineCap,\n      '; ',\n      'stroke-dashoffset: ',\n      strokeDashOffset,\n      '; ',\n      'stroke-linejoin: ',\n      strokeLineJoin,\n      '; ',\n      'stroke-miterlimit: ',\n      strokeMiterLimit,\n      '; ',\n      fill,\n      'fill-rule: ',\n      fillRule,\n      '; ',\n      'opacity: ',\n      opacity,\n      ';',\n      filter,\n      visibility,\n    ].join('');\n  }\n\n  /**\n   * Returns filter for svg shadow\n   * @return {String}\n   */\n  getSvgFilter(this: FabricObjectSVGExportMixin & FabricObject) {\n    return this.shadow ? `filter: url(#SVGID_${this.shadow.id});` : '';\n  }\n\n  /**\n   * Returns id attribute for svg output\n   * @return {String}\n   */\n  getSvgCommons(\n    this: FabricObjectSVGExportMixin & FabricObject & { id?: string }\n  ) {\n    return [\n      this.id ? `id=\"${this.id}\" ` : '',\n      this.clipPath\n        ? `clip-path=\"url(#${\n            (this.clipPath as FabricObjectSVGExportMixin & FabricObject)\n              .clipPathId\n          })\" `\n        : '',\n    ].join('');\n  }\n\n  /**\n   * Returns transform-string for svg-export\n   * @param {Boolean} use the full transform or the single object one.\n   * @return {String}\n   */\n  getSvgTransform(\n    this: FabricObjectSVGExportMixin & FabricObject,\n    full?: boolean,\n    additionalTransform = ''\n  ) {\n    const transform = full ? this.calcTransformMatrix() : this.calcOwnMatrix(),\n      svgTransform = `transform=\"${matrixToSVG(transform)}`;\n    return `${svgTransform}${additionalTransform}\" `;\n  }\n\n  /**\n   * Returns svg representation of an instance\n   * This function is implemented in each subclass\n   * This is just because typescript otherwise cryies all the time\n   * @return {Array} an array of strings with the specific svg representation\n   * of the instance\n   */\n  _toSVG(reviver?: TSVGReviver): string[] {\n    return [''];\n  }\n\n  /**\n   * Returns svg representation of an instance\n   * @param {TSVGReviver} [reviver] Method for further parsing of svg representation.\n   * @return {String} svg representation of an instance\n   */\n  toSVG(\n    this: FabricObjectSVGExportMixin & FabricObject,\n    reviver?: TSVGReviver\n  ) {\n    return this._createBaseSVGMarkup(this._toSVG(reviver), {\n      reviver,\n    });\n  }\n\n  /**\n   * Returns svg clipPath representation of an instance\n   * @param {TSVGReviver} [reviver] Method for further parsing of svg representation.\n   * @return {String} svg representation of an instance\n   */\n  toClipPathSVG(\n    this: FabricObjectSVGExportMixin & FabricObject,\n    reviver?: TSVGReviver\n  ) {\n    return (\n      '\\t' +\n      this._createBaseClipPathSVGMarkup(this._toSVG(reviver), {\n        reviver,\n      })\n    );\n  }\n\n  /**\n   * @private\n   */\n  _createBaseClipPathSVGMarkup(\n    this: FabricObjectSVGExportMixin & FabricObject,\n    objectMarkup: string[],\n    {\n      reviver,\n      additionalTransform = '',\n    }: { reviver?: TSVGReviver; additionalTransform?: string } = {}\n  ) {\n    const commonPieces = [\n        this.getSvgTransform(true, additionalTransform),\n        this.getSvgCommons(),\n      ].join(''),\n      // insert commons in the markup, style and svgCommons\n      index = objectMarkup.indexOf('COMMON_PARTS');\n    objectMarkup[index] = commonPieces;\n    return reviver ? reviver(objectMarkup.join('')) : objectMarkup.join('');\n  }\n\n  /**\n   * @private\n   */\n  _createBaseSVGMarkup(\n    this: FabricObjectSVGExportMixin & FabricObject,\n    objectMarkup: string[],\n    {\n      noStyle,\n      reviver,\n      withShadow,\n      additionalTransform,\n    }: {\n      noStyle?: boolean;\n      reviver?: TSVGReviver;\n      withShadow?: boolean;\n      additionalTransform?: string;\n    } = {}\n  ): string {\n    const styleInfo = noStyle ? '' : `style=\"${this.getSvgStyles()}\" `,\n      shadowInfo = withShadow ? `style=\"${this.getSvgFilter()}\" ` : '',\n      clipPath = this.clipPath as FabricObjectSVGExportMixin & FabricObject,\n      vectorEffect = this.strokeUniform\n        ? 'vector-effect=\"non-scaling-stroke\" '\n        : '',\n      absoluteClipPath = clipPath && clipPath.absolutePositioned,\n      stroke = this.stroke,\n      fill = this.fill,\n      shadow = this.shadow,\n      markup = [],\n      // insert commons in the markup, style and svgCommons\n      index = objectMarkup.indexOf('COMMON_PARTS');\n    let clipPathMarkup;\n    if (clipPath) {\n      clipPath.clipPathId = `CLIPPATH_${uid()}`;\n      clipPathMarkup = `<clipPath id=\"${\n        clipPath.clipPathId\n      }\" >\\n${clipPath.toClipPathSVG(reviver)}</clipPath>\\n`;\n    }\n    if (absoluteClipPath) {\n      markup.push('<g ', shadowInfo, this.getSvgCommons(), ' >\\n');\n    }\n    markup.push(\n      '<g ',\n      this.getSvgTransform(false),\n      !absoluteClipPath ? shadowInfo + this.getSvgCommons() : '',\n      ' >\\n'\n    );\n    const commonPieces = [\n      styleInfo,\n      vectorEffect,\n      noStyle ? '' : this.addPaintOrder(),\n      ' ',\n      additionalTransform ? `transform=\"${additionalTransform}\" ` : '',\n    ].join('');\n    objectMarkup[index] = commonPieces;\n    if (isFiller(fill)) {\n      markup.push(fill.toSVG(this));\n    }\n    if (isFiller(stroke)) {\n      markup.push(stroke.toSVG(this));\n    }\n    if (shadow) {\n      markup.push(shadow.toSVG(this));\n    }\n    if (clipPath) {\n      markup.push(clipPathMarkup);\n    }\n    markup.push(objectMarkup.join(''));\n    markup.push('</g>\\n');\n    absoluteClipPath && markup.push('</g>\\n');\n    return reviver ? reviver(markup.join('')) : markup.join('');\n  }\n\n  addPaintOrder(this: FabricObjectSVGExportMixin & FabricObject) {\n    return this.paintFirst !== 'fill'\n      ? ` paint-order=\"${this.paintFirst}\" `\n      : '';\n  }\n}\n","/**\n * Easing functions\n * @see {@link http://gizma.com/easing/ Easing Equations by Robert Penner}\n */\n\nimport { twoMathPi, halfPI } from '../../constants';\nimport type { TEasingFunction } from './types';\n\nconst normalize = (a: number, c: number, p: number, s: number) => {\n  if (a < Math.abs(c)) {\n    a = c;\n    s = p / 4;\n  } else {\n    //handle the 0/0 case:\n    if (c === 0 && a === 0) {\n      s = (p / twoMathPi) * Math.asin(1);\n    } else {\n      s = (p / twoMathPi) * Math.asin(c / a);\n    }\n  }\n  return { a, c, p, s };\n};\n\nconst elastic = (\n  a: number,\n  s: number,\n  p: number,\n  t: number,\n  d: number\n): number =>\n  a * Math.pow(2, 10 * (t -= 1)) * Math.sin(((t * d - s) * twoMathPi) / p);\n\n/**\n * Default sinusoidal easing\n */\nexport const defaultEasing: TEasingFunction = (t, b, c, d) =>\n  -c * Math.cos((t / d) * halfPI) + c + b;\n\n/**\n * Cubic easing in\n */\nexport const easeInCubic: TEasingFunction = (t, b, c, d) =>\n  c * (t / d) ** 3 + b;\n\n/**\n * Cubic easing out\n */\nexport const easeOutCubic: TEasingFunction = (t, b, c, d) =>\n  c * ((t / d - 1) ** 3 + 1) + b;\n\n/**\n * Cubic easing in and out\n */\nexport const easeInOutCubic: TEasingFunction = (t, b, c, d) => {\n  t /= d / 2;\n  if (t < 1) {\n    return (c / 2) * t ** 3 + b;\n  }\n  return (c / 2) * ((t - 2) ** 3 + 2) + b;\n};\n\n/**\n * Quartic easing in\n */\nexport const easeInQuart: TEasingFunction = (t, b, c, d) =>\n  c * (t /= d) * t ** 3 + b;\n\n/**\n * Quartic easing out\n */\nexport const easeOutQuart: TEasingFunction = (t, b, c, d) =>\n  -c * ((t = t / d - 1) * t ** 3 - 1) + b;\n\n/**\n * Quartic easing in and out\n */\nexport const easeInOutQuart: TEasingFunction = (t, b, c, d) => {\n  t /= d / 2;\n  if (t < 1) {\n    return (c / 2) * t ** 4 + b;\n  }\n  return (-c / 2) * ((t -= 2) * t ** 3 - 2) + b;\n};\n\n/**\n * Quintic easing in\n */\nexport const easeInQuint: TEasingFunction = (t, b, c, d) =>\n  c * (t / d) ** 5 + b;\n\n/**\n * Quintic easing out\n */\nexport const easeOutQuint: TEasingFunction = (t, b, c, d) =>\n  c * ((t / d - 1) ** 5 + 1) + b;\n\n/**\n * Quintic easing in and out\n */\nexport const easeInOutQuint: TEasingFunction = (t, b, c, d) => {\n  t /= d / 2;\n  if (t < 1) {\n    return (c / 2) * t ** 5 + b;\n  }\n  return (c / 2) * ((t - 2) ** 5 + 2) + b;\n};\n\n/**\n * Sinusoidal easing in\n */\nexport const easeInSine: TEasingFunction = (t, b, c, d) =>\n  -c * Math.cos((t / d) * halfPI) + c + b;\n\n/**\n * Sinusoidal easing out\n */\nexport const easeOutSine: TEasingFunction = (t, b, c, d) =>\n  c * Math.sin((t / d) * halfPI) + b;\n\n/**\n * Sinusoidal easing in and out\n */\nexport const easeInOutSine: TEasingFunction = (t, b, c, d) =>\n  (-c / 2) * (Math.cos((Math.PI * t) / d) - 1) + b;\n\n/**\n * Exponential easing in\n */\nexport const easeInExpo: TEasingFunction = (t, b, c, d) =>\n  t === 0 ? b : c * 2 ** (10 * (t / d - 1)) + b;\n\n/**\n * Exponential easing out\n */\nexport const easeOutExpo: TEasingFunction = (t, b, c, d) =>\n  t === d ? b + c : c * -(2 ** ((-10 * t) / d) + 1) + b;\n\n/**\n * Exponential easing in and out\n */\nexport const easeInOutExpo: TEasingFunction = (t, b, c, d) => {\n  if (t === 0) {\n    return b;\n  }\n  if (t === d) {\n    return b + c;\n  }\n  t /= d / 2;\n  if (t < 1) {\n    return (c / 2) * 2 ** (10 * (t - 1)) + b;\n  }\n  return (c / 2) * -(2 ** (-10 * --t) + 2) + b;\n};\n\n/**\n * Circular easing in\n */\nexport const easeInCirc: TEasingFunction = (t, b, c, d) =>\n  -c * (Math.sqrt(1 - (t /= d) * t) - 1) + b;\n\n/**\n * Circular easing out\n */\nexport const easeOutCirc: TEasingFunction = (t, b, c, d) =>\n  c * Math.sqrt(1 - (t = t / d - 1) * t) + b;\n\n/**\n * Circular easing in and out\n */\nexport const easeInOutCirc: TEasingFunction = (t, b, c, d) => {\n  t /= d / 2;\n  if (t < 1) {\n    return (-c / 2) * (Math.sqrt(1 - t ** 2) - 1) + b;\n  }\n  return (c / 2) * (Math.sqrt(1 - (t -= 2) * t) + 1) + b;\n};\n\n/**\n * Elastic easing in\n */\nexport const easeInElastic: TEasingFunction = (t, b, c, d) => {\n  const s = 1.70158,\n    a = c;\n  let p = 0;\n  if (t === 0) {\n    return b;\n  }\n  t /= d;\n  if (t === 1) {\n    return b + c;\n  }\n  if (!p) {\n    p = d * 0.3;\n  }\n  const { a: normA, s: normS, p: normP } = normalize(a, c, p, s);\n  return -elastic(normA, normS, normP, t, d) + b;\n};\n\n/**\n * Elastic easing out\n */\nexport const easeOutElastic: TEasingFunction = (t, b, c, d) => {\n  const s = 1.70158,\n    a = c;\n  let p = 0;\n  if (t === 0) {\n    return b;\n  }\n  t /= d;\n  if (t === 1) {\n    return b + c;\n  }\n  if (!p) {\n    p = d * 0.3;\n  }\n  const { a: normA, s: normS, p: normP, c: normC } = normalize(a, c, p, s);\n  return (\n    normA * 2 ** (-10 * t) * Math.sin(((t * d - normS) * twoMathPi) / normP) +\n    normC +\n    b\n  );\n};\n\n/**\n * Elastic easing in and out\n */\nexport const easeInOutElastic: TEasingFunction = (t, b, c, d) => {\n  const s = 1.70158,\n    a = c;\n  let p = 0;\n  if (t === 0) {\n    return b;\n  }\n  t /= d / 2;\n  if (t === 2) {\n    return b + c;\n  }\n  if (!p) {\n    p = d * (0.3 * 1.5);\n  }\n  const { a: normA, s: normS, p: normP, c: normC } = normalize(a, c, p, s);\n  if (t < 1) {\n    return -0.5 * elastic(normA, normS, normP, t, d) + b;\n  }\n  return (\n    normA *\n      Math.pow(2, -10 * (t -= 1)) *\n      Math.sin(((t * d - normS) * twoMathPi) / normP) *\n      0.5 +\n    normC +\n    b\n  );\n};\n\n/**\n * Backwards easing in\n */\nexport const easeInBack: TEasingFunction = (t, b, c, d, s = 1.70158) =>\n  c * (t /= d) * t * ((s + 1) * t - s) + b;\n\n/**\n * Backwards easing out\n */\nexport const easeOutBack: TEasingFunction = (t, b, c, d, s = 1.70158) =>\n  c * ((t = t / d - 1) * t * ((s + 1) * t + s) + 1) + b;\n\n/**\n * Backwards easing in and out\n */\nexport const easeInOutBack: TEasingFunction = (t, b, c, d, s = 1.70158) => {\n  t /= d / 2;\n  if (t < 1) {\n    return (c / 2) * (t * t * (((s *= 1.525) + 1) * t - s)) + b;\n  }\n  return (c / 2) * ((t -= 2) * t * (((s *= 1.525) + 1) * t + s) + 2) + b;\n};\n\n/**\n * Bouncing easing out\n */\nexport const easeOutBounce: TEasingFunction = (t, b, c, d) => {\n  if ((t /= d) < 1 / 2.75) {\n    return c * (7.5625 * t * t) + b;\n  } else if (t < 2 / 2.75) {\n    return c * (7.5625 * (t -= 1.5 / 2.75) * t + 0.75) + b;\n  } else if (t < 2.5 / 2.75) {\n    return c * (7.5625 * (t -= 2.25 / 2.75) * t + 0.9375) + b;\n  } else {\n    return c * (7.5625 * (t -= 2.625 / 2.75) * t + 0.984375) + b;\n  }\n};\n\n/**\n * Bouncing easing in\n */\nexport const easeInBounce: TEasingFunction = (t, b, c, d) =>\n  c - easeOutBounce(d - t, 0, c, d) + b;\n\n/**\n * Bouncing easing in and out\n */\nexport const easeInOutBounce: TEasingFunction = (t, b, c, d) =>\n  t < d / 2\n    ? easeInBounce(t * 2, 0, c, d) * 0.5 + b\n    : easeOutBounce(t * 2 - d, 0, c, d) * 0.5 + c * 0.5 + b;\n\n/**\n * Quadratic easing in\n */\nexport const easeInQuad: TEasingFunction = (t, b, c, d) => c * (t /= d) * t + b;\n\n/**\n * Quadratic easing out\n */\nexport const easeOutQuad: TEasingFunction = (t, b, c, d) =>\n  -c * (t /= d) * (t - 2) + b;\n\n/**\n * Quadratic easing in and out\n */\nexport const easeInOutQuad: TEasingFunction = (t, b, c, d) => {\n  t /= d / 2;\n  if (t < 1) {\n    return (c / 2) * t ** 2 + b;\n  }\n  return (-c / 2) * (--t * (t - 2) - 1) + b;\n};\n","import { noop } from '../../constants';\nimport { requestAnimFrame } from './AnimationFrameProvider';\nimport { runningAnimations } from './AnimationRegistry';\nimport { defaultEasing } from './easing';\nimport type {\n  AnimationState,\n  TAbortCallback,\n  TBaseAnimationOptions,\n  TEasingFunction,\n  TOnAnimationChangeCallback,\n} from './types';\n\nconst defaultAbort = () => false;\n\nexport abstract class AnimationBase<\n  T extends number | number[] = number | number[]\n> {\n  declare readonly startValue: T;\n  declare readonly endValue: T;\n  declare readonly duration: number;\n  declare readonly delay: number;\n\n  protected declare readonly byValue: T;\n  protected declare readonly easing: TEasingFunction<T>;\n\n  private declare readonly _onStart: VoidFunction;\n  private declare readonly _onChange: TOnAnimationChangeCallback<T, void>;\n  private declare readonly _onComplete: TOnAnimationChangeCallback<T, void>;\n  private declare readonly _abort: TAbortCallback<T>;\n\n  /**\n   * Used to register the animation to a target object\n   * so that it can be cancelled within the object context\n   */\n  declare readonly target?: unknown;\n\n  private _state: AnimationState = 'pending';\n  /**\n   * Time %, or the ratio of `timeElapsed / duration`\n   * @see tick\n   */\n  durationProgress = 0;\n  /**\n   * Value %, or the ratio of `(currentValue - startValue) / (endValue - startValue)`\n   */\n  valueProgress = 0;\n  /**\n   * Current value\n   */\n  declare value: T;\n  /**\n   * Animation start time ms\n   */\n  private declare startTime: number;\n\n  constructor({\n    startValue,\n    byValue,\n    duration = 500,\n    delay = 0,\n    easing = defaultEasing,\n    onStart = noop,\n    onChange = noop,\n    onComplete = noop,\n    abort = defaultAbort,\n    target,\n  }: TBaseAnimationOptions<T>) {\n    this.tick = this.tick.bind(this);\n\n    this.duration = duration;\n    this.delay = delay;\n    this.easing = easing;\n    this._onStart = onStart;\n    this._onChange = onChange;\n    this._onComplete = onComplete;\n    this._abort = abort;\n    this.target = target;\n\n    this.startValue = startValue;\n    this.byValue = byValue;\n    this.value = this.startValue;\n    this.endValue = Object.freeze(this.calculate(this.duration).value);\n  }\n\n  get state() {\n    return this._state;\n  }\n\n  isDone() {\n    return this._state === 'aborted' || this._state === 'completed';\n  }\n\n  /**\n   * Calculate the current value based on the easing parameters\n   * @param timeElapsed in ms\n   * @protected\n   */\n  protected abstract calculate(timeElapsed: number): {\n    value: T;\n    valueProgress: number;\n  };\n\n  start() {\n    const firstTick: FrameRequestCallback = (timestamp) => {\n      if (this._state !== 'pending') return;\n      this.startTime = timestamp || +new Date();\n      this._state = 'running';\n      this._onStart();\n      this.tick(this.startTime);\n    };\n\n    this.register();\n\n    // setTimeout(cb, 0) will run cb on the next frame, causing a delay\n    // we don't want that\n    if (this.delay > 0) {\n      setTimeout(() => requestAnimFrame(firstTick), this.delay);\n    } else {\n      requestAnimFrame(firstTick);\n    }\n  }\n\n  private tick(t: number) {\n    const durationMs = (t || +new Date()) - this.startTime;\n    const boundDurationMs = Math.min(durationMs, this.duration);\n    this.durationProgress = boundDurationMs / this.duration;\n    const { value, valueProgress } = this.calculate(boundDurationMs);\n    this.value = Object.freeze(value);\n    this.valueProgress = valueProgress;\n\n    if (this._state === 'aborted') {\n      return;\n    } else if (\n      this._abort(this.value, this.valueProgress, this.durationProgress)\n    ) {\n      this._state = 'aborted';\n      this.unregister();\n    } else if (durationMs >= this.duration) {\n      this.durationProgress = this.valueProgress = 1;\n      this._onChange(this.endValue, this.valueProgress, this.durationProgress);\n      this._state = 'completed';\n      this._onComplete(\n        this.endValue,\n        this.valueProgress,\n        this.durationProgress\n      );\n      this.unregister();\n    } else {\n      this._onChange(this.value, this.valueProgress, this.durationProgress);\n      requestAnimFrame(this.tick);\n    }\n  }\n\n  private register() {\n    runningAnimations.push(this as unknown as AnimationBase);\n  }\n\n  private unregister() {\n    runningAnimations.remove(this as unknown as AnimationBase);\n  }\n\n  abort() {\n    this._state = 'aborted';\n    this.unregister();\n  }\n}\n","import { AnimationBase } from './AnimationBase';\nimport type { ValueAnimationOptions } from './types';\n\nexport class ValueAnimation extends AnimationBase<number> {\n  constructor({\n    startValue = 0,\n    endValue = 100,\n    ...otherOptions\n  }: ValueAnimationOptions) {\n    super({\n      ...otherOptions,\n      startValue,\n      byValue: endValue - startValue,\n    });\n  }\n\n  protected calculate(timeElapsed: number) {\n    const value = this.easing(\n      timeElapsed,\n      this.startValue,\n      this.byValue,\n      this.duration\n    );\n    return {\n      value,\n      valueProgress: Math.abs((value - this.startValue) / this.byValue),\n    };\n  }\n}\n","import { AnimationBase } from './AnimationBase';\nimport type { ArrayAnimationOptions } from './types';\n\nexport class ArrayAnimation extends AnimationBase<number[]> {\n  constructor({\n    startValue = [0],\n    endValue = [100],\n    ...options\n  }: ArrayAnimationOptions) {\n    super({\n      ...options,\n      startValue,\n      byValue: endValue.map((value, i) => value - startValue[i]),\n    });\n  }\n  protected calculate(timeElapsed: number) {\n    const values = this.startValue.map((value, i) =>\n      this.easing(timeElapsed, value, this.byValue[i], this.duration, i)\n    );\n    return {\n      value: values,\n      valueProgress: Math.abs(\n        (values[0] - this.startValue[0]) / this.byValue[0]\n      ),\n    };\n  }\n}\n","export const capValue = (min: number, value: number, max: number) =>\n  Math.max(min, Math.min(value, max));\n","import { Color } from '../../color/Color';\nimport type { TRGBAColorSource } from '../../color/typedefs';\nimport { halfPI } from '../../constants';\nimport { capValue } from '../misc/capValue';\nimport { AnimationBase } from './AnimationBase';\nimport type {\n  ColorAnimationOptions,\n  TEasingFunction,\n  TOnAnimationChangeCallback,\n} from './types';\n\nconst defaultColorEasing: TEasingFunction = (\n  timeElapsed,\n  startValue,\n  byValue,\n  duration\n) => {\n  const durationProgress = 1 - Math.cos((timeElapsed / duration) * halfPI);\n  return startValue + byValue * durationProgress;\n};\n\nconst wrapColorCallback = <R>(\n  callback?: TOnAnimationChangeCallback<string, R>\n) =>\n  callback &&\n  ((rgba: TRGBAColorSource, valueProgress: number, durationProgress: number) =>\n    callback(new Color(rgba).toRgba(), valueProgress, durationProgress));\n\nexport class ColorAnimation extends AnimationBase<TRGBAColorSource> {\n  constructor({\n    startValue,\n    endValue,\n    easing = defaultColorEasing,\n    onChange,\n    onComplete,\n    abort,\n    ...options\n  }: ColorAnimationOptions) {\n    const startColor = new Color(startValue).getSource();\n    const endColor = new Color(endValue).getSource();\n    super({\n      ...options,\n      startValue: startColor,\n      byValue: endColor.map(\n        (value, i) => value - startColor[i]\n      ) as TRGBAColorSource,\n      easing,\n      onChange: wrapColorCallback(onChange),\n      onComplete: wrapColorCallback(onComplete),\n      abort: wrapColorCallback(abort),\n    });\n  }\n  protected calculate(timeElapsed: number) {\n    const [r, g, b, a] = this.startValue.map((value, i) =>\n      this.easing(timeElapsed, value, this.byValue[i], this.duration, i)\n    ) as TRGBAColorSource;\n    const value = [\n      ...[r, g, b].map(Math.round),\n      capValue(0, a, 1),\n    ] as TRGBAColorSource;\n    return {\n      value,\n      valueProgress:\n        // to correctly calculate the change ratio we must find a changed value\n        value\n          .map((p, i) =>\n            this.byValue[i] !== 0\n              ? Math.abs((p - this.startValue[i]) / this.byValue[i])\n              : 0\n          )\n          .find((p) => p !== 0) || 0,\n    };\n  }\n}\n","import { ValueAnimation } from './ValueAnimation';\nimport { ArrayAnimation } from './ArrayAnimation';\nimport { ColorAnimation } from './ColorAnimation';\nimport type {\n  ValueAnimationOptions,\n  ArrayAnimationOptions,\n  ColorAnimationOptions,\n} from './types';\nimport type { TColorArg } from '../../color/typedefs';\n\nexport type TAnimation<T extends number | number[] | TColorArg> =\n  T extends TColorArg\n    ? ColorAnimation\n    : T extends number[]\n    ? ArrayAnimation\n    : ValueAnimation;\n\nconst isArrayAnimation = (\n  options: ArrayAnimationOptions | ValueAnimationOptions\n): options is ArrayAnimationOptions => {\n  return Array.isArray(options.startValue) || Array.isArray(options.endValue);\n};\n\n/**\n * Changes value(s) from startValue to endValue within a certain period of time,\n * invoking callbacks as the value(s) change.\n *\n * @example\n * animate({\n *   startValue: 1,\n *   endValue: 0,\n *   onChange: (v) => {\n *     obj.set('opacity', v);\n *     // since we are running in a requested frame we should call `renderAll` and not `requestRenderAll`\n *     canvas.renderAll();\n *   }\n * });\n *\n * @example Using lists:\n * animate({\n *   startValue: [1, 2, 3],\n *   endValue: [2, 4, 6],\n *   onChange: ([x, y, zoom]) => {\n *     canvas.zoomToPoint(new Point(x, y), zoom);\n *     canvas.renderAll();\n *   }\n * });\n *\n */\nexport function animate(options: ArrayAnimationOptions): ArrayAnimation;\nexport function animate(options: ValueAnimationOptions): ValueAnimation;\nexport function animate<\n  T extends ValueAnimationOptions | ArrayAnimationOptions\n>(\n  options: T\n): T extends ArrayAnimationOptions ? ArrayAnimation : ValueAnimation;\nexport function animate<\n  T extends ValueAnimationOptions | ArrayAnimationOptions,\n  R extends T extends ArrayAnimationOptions ? ArrayAnimation : ValueAnimation\n>(options: T): R {\n  const animation = (\n    isArrayAnimation(options)\n      ? new ArrayAnimation(options)\n      : new ValueAnimation(options)\n  ) as R;\n  animation.start();\n  return animation;\n}\n\nexport function animateColor(options: ColorAnimationOptions) {\n  const animation = new ColorAnimation(options);\n  animation.start();\n  return animation;\n}\n","import type { XY } from '../../Point';\nimport { Point } from '../../Point';\nimport type { TRadian } from '../../typedefs';\n\nconst unitVectorX = new Point(1, 0);\nconst zero = new Point();\n\n/**\n * Rotates `vector` with `radians`\n * @param {Point} vector The vector to rotate (x and y)\n * @param {Number} radians The radians of the angle for the rotation\n * @return {Point} The new rotated point\n */\nexport const rotateVector = (vector: Point, radians: TRadian) =>\n  vector.rotate(radians);\n\n/**\n * Creates a vector from points represented as a point\n *\n * @param {Point} from\n * @param {Point} to\n * @returns {Point} vector\n */\nexport const createVector = (from: XY, to: XY): Point =>\n  new Point(to).subtract(from);\n\n/**\n * return the magnitude of a vector\n * @return {number}\n */\nexport const magnitude = (point: Point) => point.distanceFrom(zero);\n\n/**\n * Calculates the angle between 2 vectors\n * @param {Point} a\n * @param {Point} b\n * @returns the angle in radians from `a` to `b`\n */\nexport const calcAngleBetweenVectors = (a: Point, b: Point): TRadian =>\n  Math.atan2(crossProduct(a, b), dotProduct(a, b)) as TRadian;\n\n/**\n * Calculates the angle between the x axis and the vector\n * @param {Point} v\n * @returns the angle in radians of `v`\n */\nexport const calcVectorRotation = (v: Point) =>\n  calcAngleBetweenVectors(unitVectorX, v);\n\n/**\n * @param {Point} v\n * @returns {Point} vector representing the unit vector pointing to the direction of `v`\n */\nexport const getUnitVector = (v: Point): Point =>\n  v.eq(zero) ? v : v.scalarDivide(magnitude(v));\n\n/**\n * @param {Point} v\n * @param {Boolean} [counterClockwise] the direction of the orthogonal vector, defaults to `true`\n * @returns {Point} the unit orthogonal vector\n */\nexport const getOrthonormalVector = (\n  v: Point,\n  counterClockwise = true\n): Point =>\n  getUnitVector(new Point(-v.y, v.x).scalarMultiply(counterClockwise ? 1 : -1));\n\n/**\n * Cross product of two vectors in 2D\n * @param {Point} a\n * @param {Point} b\n * @returns {number} the magnitude of Z vector\n */\nexport const crossProduct = (a: Point, b: Point): number =>\n  a.x * b.y - a.y * b.x;\n\n/**\n * Dot product of two vectors in 2D\n * @param {Point} a\n * @param {Point} b\n * @returns {number}\n */\nexport const dotProduct = (a: Point, b: Point): number => a.x * b.x + a.y * b.y;\n\n/**\n * Checks if the vector is between two others. It is considered\n * to be inside when the vector to be tested is between the\n * initial vector and the final vector (included) in a counterclockwise direction.\n * @param {Point} t vector to be tested\n * @param {Point} a initial vector\n * @param {Point} b final vector\n * @returns {boolean} true if the vector is among the others\n */\nexport const isBetweenVectors = (t: Point, a: Point, b: Point): boolean => {\n  if (t.eq(a) || t.eq(b)) return true;\n  const AxB = crossProduct(a, b),\n    AxT = crossProduct(a, t),\n    BxT = crossProduct(b, t);\n  return AxB >= 0 ? AxT >= 0 && BxT <= 0 : !(AxT <= 0 && BxT >= 0);\n};\n","import { Point } from './Point';\nimport { createVector } from './util/misc/vectors';\n\n/* Adaptation of work of Kevin Lindsey (kevin@kevlindev.com) */\n\nexport type IntersectionType = 'Intersection' | 'Coincident' | 'Parallel';\n\nexport class Intersection {\n  declare points: Point[];\n\n  declare status?: IntersectionType;\n\n  constructor(status?: IntersectionType) {\n    this.status = status;\n    this.points = [];\n  }\n\n  /**\n   * Used to verify if a point is alredy in the collection\n   * @param {Point} point\n   * @returns {boolean}\n   */\n  private includes(point: Point): boolean {\n    return this.points.some((p) => p.eq(point));\n  }\n\n  /**\n   * Appends points of intersection\n   * @param {...Point[]} points\n   * @return {Intersection} thisArg\n   * @chainable\n   */\n  private append(...points: Point[]): Intersection {\n    this.points = this.points.concat(\n      points.filter((point) => {\n        return !this.includes(point);\n      })\n    );\n    return this;\n  }\n\n  /**\n   * check if point T is on the segment or line defined between A and B\n   *\n   * @param {Point} T the point we are checking for\n   * @param {Point} A one extremity of the segment\n   * @param {Point} B the other extremity of the segment\n   * @param [infinite] if true checks if `T` is on the line defined by `A` and `B`\n   * @returns true if `T` is contained\n   */\n  static isPointContained(T: Point, A: Point, B: Point, infinite = false) {\n    if (A.eq(B)) {\n      // Edge case: the segment is a point, we check for coincidence,\n      // infinite param has no meaning because there are infinite lines to consider\n      return T.eq(A);\n    } else if (A.x === B.x) {\n      // Edge case: horizontal line.\n      // we first check if T.x has the same value, and then if T.y is contained between A.y and B.y\n      return (\n        T.x === A.x &&\n        (infinite || (T.y >= Math.min(A.y, B.y) && T.y <= Math.max(A.y, B.y)))\n      );\n    } else if (A.y === B.y) {\n      // Edge case: vertical line.\n      // we first check if T.y has the same value, and then if T.x is contained between A.x and B.x\n      return (\n        T.y === A.y &&\n        (infinite || (T.x >= Math.min(A.x, B.x) && T.x <= Math.max(A.x, B.x)))\n      );\n    } else {\n      // Generic case: sloped line.\n      // we check that AT has the same slope as AB\n      // for the segment case we need both the vectors to have the same direction and for AT to be lte AB in size\n      // for the infinite case we check the absolute value of the slope, since direction is meaningless\n      const AB = createVector(A, B);\n      const AT = createVector(A, T);\n      const s = AT.divide(AB);\n      return infinite\n        ? Math.abs(s.x) === Math.abs(s.y)\n        : s.x === s.y && s.x >= 0 && s.x <= 1;\n    }\n  }\n\n  /**\n   * Use the ray casting algorithm to determine if {@link point} is in the polygon defined by {@link points}\n   * @see https://en.wikipedia.org/wiki/Point_in_polygon\n   * @param point\n   * @param points polygon points\n   * @returns\n   */\n  static isPointInPolygon(point: Point, points: Point[]) {\n    const other = new Point(point).setX(\n      Math.min(point.x - 1, ...points.map((p) => p.x))\n    );\n    let hits = 0;\n    for (let index = 0; index < points.length; index++) {\n      const inter = this.intersectSegmentSegment(\n        // polygon side\n        points[index],\n        points[(index + 1) % points.length],\n        // ray\n        point,\n        other\n      );\n      if (inter.includes(point)) {\n        // point is on the polygon side\n        return true;\n      }\n      hits += Number(inter.status === 'Intersection');\n    }\n    return hits % 2 === 1;\n  }\n\n  /**\n   * Checks if a line intersects another\n   * @see {@link https://en.wikipedia.org/wiki/Line%E2%80%93line_intersection line intersection}\n   * @see {@link https://en.wikipedia.org/wiki/Cramer%27s_rule Cramer's rule}\n   * @static\n   * @param {Point} a1\n   * @param {Point} a2\n   * @param {Point} b1\n   * @param {Point} b2\n   * @param {boolean} [aInfinite=true] check segment intersection by passing `false`\n   * @param {boolean} [bInfinite=true] check segment intersection by passing `false`\n   * @return {Intersection}\n   */\n  static intersectLineLine(\n    a1: Point,\n    a2: Point,\n    b1: Point,\n    b2: Point,\n    aInfinite = true,\n    bInfinite = true\n  ): Intersection {\n    const a2xa1x = a2.x - a1.x,\n      a2ya1y = a2.y - a1.y,\n      b2xb1x = b2.x - b1.x,\n      b2yb1y = b2.y - b1.y,\n      a1xb1x = a1.x - b1.x,\n      a1yb1y = a1.y - b1.y,\n      uaT = b2xb1x * a1yb1y - b2yb1y * a1xb1x,\n      ubT = a2xa1x * a1yb1y - a2ya1y * a1xb1x,\n      uB = b2yb1y * a2xa1x - b2xb1x * a2ya1y;\n    if (uB !== 0) {\n      const ua = uaT / uB,\n        ub = ubT / uB;\n      if (\n        (aInfinite || (0 <= ua && ua <= 1)) &&\n        (bInfinite || (0 <= ub && ub <= 1))\n      ) {\n        return new Intersection('Intersection').append(\n          new Point(a1.x + ua * a2xa1x, a1.y + ua * a2ya1y)\n        );\n      } else {\n        return new Intersection();\n      }\n    } else {\n      if (uaT === 0 || ubT === 0) {\n        const segmentsCoincide =\n          aInfinite ||\n          bInfinite ||\n          Intersection.isPointContained(a1, b1, b2) ||\n          Intersection.isPointContained(a2, b1, b2) ||\n          Intersection.isPointContained(b1, a1, a2) ||\n          Intersection.isPointContained(b2, a1, a2);\n        return new Intersection(segmentsCoincide ? 'Coincident' : undefined);\n      } else {\n        return new Intersection('Parallel');\n      }\n    }\n  }\n\n  /**\n   * Checks if a segment intersects a line\n   * @see {@link intersectLineLine} for line intersection\n   * @static\n   * @param {Point} s1 boundary point of segment\n   * @param {Point} s2 other boundary point of segment\n   * @param {Point} l1 point on line\n   * @param {Point} l2 other point on line\n   * @return {Intersection}\n   */\n  static intersectSegmentLine(\n    s1: Point,\n    s2: Point,\n    l1: Point,\n    l2: Point\n  ): Intersection {\n    return Intersection.intersectLineLine(s1, s2, l1, l2, false, true);\n  }\n\n  /**\n   * Checks if a segment intersects another\n   * @see {@link intersectLineLine} for line intersection\n   * @static\n   * @param {Point} a1 boundary point of segment\n   * @param {Point} a2 other boundary point of segment\n   * @param {Point} b1 boundary point of segment\n   * @param {Point} b2 other boundary point of segment\n   * @return {Intersection}\n   */\n  static intersectSegmentSegment(\n    a1: Point,\n    a2: Point,\n    b1: Point,\n    b2: Point\n  ): Intersection {\n    return Intersection.intersectLineLine(a1, a2, b1, b2, false, false);\n  }\n\n  /**\n   * Checks if line intersects polygon\n   *\n   * @todo account for stroke\n   *\n   * @static\n   * @see {@link intersectSegmentPolygon} for segment intersection\n   * @param {Point} a1 point on line\n   * @param {Point} a2 other point on line\n   * @param {Point[]} points polygon points\n   * @param {boolean} [infinite=true] check segment intersection by passing `false`\n   * @return {Intersection}\n   */\n  static intersectLinePolygon(\n    a1: Point,\n    a2: Point,\n    points: Point[],\n    infinite = true\n  ): Intersection {\n    const result = new Intersection();\n    const length = points.length;\n\n    for (let i = 0, b1, b2, inter; i < length; i++) {\n      b1 = points[i];\n      b2 = points[(i + 1) % length];\n      inter = Intersection.intersectLineLine(a1, a2, b1, b2, infinite, false);\n      if (inter.status === 'Coincident') {\n        return inter;\n      }\n      result.append(...inter.points);\n    }\n\n    if (result.points.length > 0) {\n      result.status = 'Intersection';\n    }\n\n    return result;\n  }\n\n  /**\n   * Checks if segment intersects polygon\n   * @static\n   * @see {@link intersectLinePolygon} for line intersection\n   * @param {Point} a1 boundary point of segment\n   * @param {Point} a2 other boundary point of segment\n   * @param {Point[]} points polygon points\n   * @return {Intersection}\n   */\n  static intersectSegmentPolygon(\n    a1: Point,\n    a2: Point,\n    points: Point[]\n  ): Intersection {\n    return Intersection.intersectLinePolygon(a1, a2, points, false);\n  }\n\n  /**\n   * Checks if polygon intersects another polygon\n   *\n   * @todo account for stroke\n   *\n   * @static\n   * @param {Point[]} points1\n   * @param {Point[]} points2\n   * @return {Intersection}\n   */\n  static intersectPolygonPolygon(\n    points1: Point[],\n    points2: Point[]\n  ): Intersection {\n    const result = new Intersection(),\n      length = points1.length;\n    const coincidences: Intersection[] = [];\n\n    for (let i = 0; i < length; i++) {\n      const a1 = points1[i],\n        a2 = points1[(i + 1) % length],\n        inter = Intersection.intersectSegmentPolygon(a1, a2, points2);\n      if (inter.status === 'Coincident') {\n        coincidences.push(inter);\n        result.append(a1, a2);\n      } else {\n        result.append(...inter.points);\n      }\n    }\n\n    if (coincidences.length > 0 && coincidences.length === points1.length) {\n      return new Intersection('Coincident');\n    } else if (result.points.length > 0) {\n      result.status = 'Intersection';\n    }\n\n    return result;\n  }\n\n  /**\n   * Checks if polygon intersects rectangle\n   * @static\n   * @see {@link intersectPolygonPolygon} for polygon intersection\n   * @param {Point[]} points polygon points\n   * @param {Point} r1 top left point of rect\n   * @param {Point} r2 bottom right point of rect\n   * @return {Intersection}\n   */\n  static intersectPolygonRectangle(\n    points: Point[],\n    r1: Point,\n    r2: Point\n  ): Intersection {\n    const min = r1.min(r2),\n      max = r1.max(r2),\n      topRight = new Point(max.x, min.y),\n      bottomLeft = new Point(min.x, max.y);\n\n    return Intersection.intersectPolygonPolygon(points, [\n      min,\n      topRight,\n      max,\n      bottomLeft,\n    ]);\n  }\n}\n","import { Point } from '../../Point';\nimport type { Group } from '../Group';\nimport type { TDegree, TOriginX, TOriginY } from '../../typedefs';\nimport { calcDimensionsMatrix, transformPoint } from '../../util/misc/matrix';\nimport { sizeAfterTransform } from '../../util/misc/objectTransforms';\nimport { degreesToRadians } from '../../util/misc/radiansDegreesConversion';\nimport { CommonMethods } from '../../CommonMethods';\nimport { resolveOrigin } from '../../util/misc/resolveOrigin';\nimport type { BaseProps } from './types/BaseProps';\nimport type { FillStrokeProps } from './types/FillStrokeProps';\nimport { CENTER, LEFT, TOP } from '../../constants';\n\nexport class ObjectOrigin<EventSpec>\n  extends CommonMethods<EventSpec>\n  implements BaseProps, Pick<FillStrokeProps, 'strokeWidth' | 'strokeUniform'>\n{\n  declare top: number;\n  declare left: number;\n  declare width: number;\n  declare height: number;\n  declare flipX: boolean;\n  declare flipY: boolean;\n  declare scaleX: number;\n  declare scaleY: number;\n  declare skewX: number;\n  declare skewY: number;\n  declare originX: TOriginX;\n  declare originY: TOriginY;\n  declare angle: TDegree;\n  declare strokeWidth: number;\n  declare strokeUniform: boolean;\n\n  /**\n   * Object containing this object.\n   * can influence its size and position\n   */\n  declare group?: Group;\n\n  /**\n   * Calculate object bounding box dimensions from its properties scale, skew.\n   * This bounding box is aligned with object angle and not with canvas axis or screen.\n   * @param {Object} [options]\n   * @param {Number} [options.scaleX]\n   * @param {Number} [options.scaleY]\n   * @param {Number} [options.skewX]\n   * @param {Number} [options.skewY]\n   * @private\n   * @returns {Point} dimensions\n   */\n  _getTransformedDimensions(options: any = {}): Point {\n    const dimOptions = {\n      // if scaleX or scaleY are negative numbers,\n      // this will return dimensions that are negative.\n      // and this will break assumptions around the codebase\n      scaleX: this.scaleX,\n      scaleY: this.scaleY,\n      skewX: this.skewX,\n      skewY: this.skewY,\n      width: this.width,\n      height: this.height,\n      strokeWidth: this.strokeWidth,\n      ...options,\n    };\n    // stroke is applied before/after transformations are applied according to `strokeUniform`\n    const strokeWidth = dimOptions.strokeWidth;\n    let preScalingStrokeValue = strokeWidth,\n      postScalingStrokeValue = 0;\n\n    if (this.strokeUniform) {\n      preScalingStrokeValue = 0;\n      postScalingStrokeValue = strokeWidth;\n    }\n    const dimX = dimOptions.width + preScalingStrokeValue,\n      dimY = dimOptions.height + preScalingStrokeValue,\n      noSkew = dimOptions.skewX === 0 && dimOptions.skewY === 0;\n    let finalDimensions;\n    if (noSkew) {\n      finalDimensions = new Point(\n        dimX * dimOptions.scaleX,\n        dimY * dimOptions.scaleY\n      );\n    } else {\n      finalDimensions = sizeAfterTransform(\n        dimX,\n        dimY,\n        calcDimensionsMatrix(dimOptions)\n      );\n    }\n\n    return finalDimensions.scalarAdd(postScalingStrokeValue);\n  }\n\n  /**\n   * Translates the coordinates from a set of origin to another (based on the object's dimensions)\n   * @param {Point} point The point which corresponds to the originX and originY params\n   * @param {TOriginX} fromOriginX Horizontal origin: 'left', 'center' or 'right'\n   * @param {TOriginY} fromOriginY Vertical origin: 'top', 'center' or 'bottom'\n   * @param {TOriginX} toOriginX Horizontal origin: 'left', 'center' or 'right'\n   * @param {TOriginY} toOriginY Vertical origin: 'top', 'center' or 'bottom'\n   * @return {Point}\n   */\n  translateToGivenOrigin(\n    point: Point,\n    fromOriginX: TOriginX,\n    fromOriginY: TOriginY,\n    toOriginX: TOriginX,\n    toOriginY: TOriginY\n  ): Point {\n    let x = point.x,\n      y = point.y;\n    const offsetX = resolveOrigin(toOriginX) - resolveOrigin(fromOriginX),\n      offsetY = resolveOrigin(toOriginY) - resolveOrigin(fromOriginY);\n\n    if (offsetX || offsetY) {\n      const dim = this._getTransformedDimensions();\n      x += offsetX * dim.x;\n      y += offsetY * dim.y;\n    }\n\n    return new Point(x, y);\n  }\n\n  /**\n   * Translates the coordinates from origin to center coordinates (based on the object's dimensions)\n   * @param {Point} point The point which corresponds to the originX and originY params\n   * @param {TOriginX} originX Horizontal origin: 'left', 'center' or 'right'\n   * @param {TOriginY} originY Vertical origin: 'top', 'center' or 'bottom'\n   * @return {Point}\n   */\n  translateToCenterPoint(\n    point: Point,\n    originX: TOriginX,\n    originY: TOriginY\n  ): Point {\n    const p = this.translateToGivenOrigin(\n      point,\n      originX,\n      originY,\n      CENTER,\n      CENTER\n    );\n    if (this.angle) {\n      return p.rotate(degreesToRadians(this.angle), point);\n    }\n    return p;\n  }\n\n  /**\n   * Translates the coordinates from center to origin coordinates (based on the object's dimensions)\n   * @param {Point} center The point which corresponds to center of the object\n   * @param {OriginX} originX Horizontal origin: 'left', 'center' or 'right'\n   * @param {OriginY} originY Vertical origin: 'top', 'center' or 'bottom'\n   * @return {Point}\n   */\n  translateToOriginPoint(\n    center: Point,\n    originX: TOriginX,\n    originY: TOriginY\n  ): Point {\n    const p = this.translateToGivenOrigin(\n      center,\n      CENTER,\n      CENTER,\n      originX,\n      originY\n    );\n    if (this.angle) {\n      return p.rotate(degreesToRadians(this.angle), center);\n    }\n    return p;\n  }\n\n  /**\n   * Returns the center coordinates of the object relative to canvas\n   * @return {Point}\n   */\n  getCenterPoint(): Point {\n    const relCenter = this.getRelativeCenterPoint();\n    return this.group\n      ? transformPoint(relCenter, this.group.calcTransformMatrix())\n      : relCenter;\n  }\n\n  /**\n   * Returns the center coordinates of the object relative to it's parent\n   * @return {Point}\n   */\n  getRelativeCenterPoint(): Point {\n    return this.translateToCenterPoint(\n      new Point(this.left, this.top),\n      this.originX,\n      this.originY\n    );\n  }\n\n  /**\n   * Returns the coordinates of the object as if it has a different origin\n   * @param {TOriginX} originX Horizontal origin: 'left', 'center' or 'right'\n   * @param {TOriginY} originY Vertical origin: 'top', 'center' or 'bottom'\n   * @return {Point}\n   */\n  getPointByOrigin(originX: TOriginX, originY: TOriginY): Point {\n    return this.translateToOriginPoint(\n      this.getRelativeCenterPoint(),\n      originX,\n      originY\n    );\n  }\n\n  /**\n   * Sets the position of the object taking into consideration the object's origin\n   * @param {Point} pos The new position of the object\n   * @param {TOriginX} originX Horizontal origin: 'left', 'center' or 'right'\n   * @param {TOriginY} originY Vertical origin: 'top', 'center' or 'bottom'\n   * @return {void}\n   */\n  setPositionByOrigin(pos: Point, originX: TOriginX, originY: TOriginY) {\n    const center = this.translateToCenterPoint(pos, originX, originY),\n      position = this.translateToOriginPoint(\n        center,\n        this.originX,\n        this.originY\n      );\n    this.set({ left: position.x, top: position.y });\n  }\n\n  /**\n   * @private\n   */\n  _getLeftTopCoords() {\n    return this.translateToOriginPoint(\n      this.getRelativeCenterPoint(),\n      LEFT,\n      TOP\n    );\n  }\n}\n","import type {\n  TBBox,\n  TCornerPoint,\n  TDegree,\n  TMat2D,\n  TOriginX,\n  TOriginY,\n} from '../../typedefs';\nimport { iMatrix } from '../../constants';\nimport { Intersection } from '../../Intersection';\nimport { Point } from '../../Point';\nimport { makeBoundingBoxFromPoints } from '../../util/misc/boundingBoxFromPoints';\nimport {\n  createRotateMatrix,\n  createTranslateMatrix,\n  composeMatrix,\n  invertTransform,\n  multiplyTransformMatrices,\n  transformPoint,\n  calcPlaneRotation,\n} from '../../util/misc/matrix';\nimport { radiansToDegrees } from '../../util/misc/radiansDegreesConversion';\nimport type { Canvas } from '../../canvas/Canvas';\nimport type { StaticCanvas } from '../../canvas/StaticCanvas';\nimport { ObjectOrigin } from './ObjectOrigin';\nimport type { ObjectEvents } from '../../EventTypeDefs';\nimport type { ControlProps } from './types/ControlProps';\nimport { resolveOrigin } from '../../util/misc/resolveOrigin';\n\ntype TMatrixCache = {\n  key: number[];\n  value: TMat2D;\n};\n\ntype TACoords = TCornerPoint;\n\nexport class ObjectGeometry<EventSpec extends ObjectEvents = ObjectEvents>\n  extends ObjectOrigin<EventSpec>\n  implements Pick<ControlProps, 'padding'>\n{\n  declare padding: number;\n\n  /**\n   * Describe object's corner position in scene coordinates.\n   * The coordinates are derived from the following:\n   * left, top, width, height, scaleX, scaleY, skewX, skewY, angle, strokeWidth.\n   * The coordinates do not depend on viewport changes.\n   * The coordinates get updated with {@link setCoords}.\n   * You can calculate them without updating with {@link calcACoords()}\n   */\n  declare aCoords: TACoords;\n\n  /**\n   * storage cache for object transform matrix\n   */\n  declare ownMatrixCache?: TMatrixCache;\n\n  /**\n   * storage cache for object full transform matrix\n   */\n  declare matrixCache?: TMatrixCache;\n\n  /**\n   * A Reference of the Canvas where the object is actually added\n   * @type StaticCanvas | Canvas;\n   * @default undefined\n   * @private\n   */\n  declare canvas?: StaticCanvas | Canvas;\n\n  /**\n   * @returns {number} x position according to object's {@link originX} property in canvas coordinate plane\n   */\n  getX(): number {\n    return this.getXY().x;\n  }\n\n  /**\n   * @param {number} value x position according to object's {@link originX} property in canvas coordinate plane\n   */\n  setX(value: number) {\n    this.setXY(this.getXY().setX(value));\n  }\n\n  /**\n   * @returns {number} y position according to object's {@link originY} property in canvas coordinate plane\n   */\n  getY(): number {\n    return this.getXY().y;\n  }\n\n  /**\n   * @param {number} value y position according to object's {@link originY} property in canvas coordinate plane\n   */\n  setY(value: number) {\n    this.setXY(this.getXY().setY(value));\n  }\n\n  /**\n   * @returns {number} x position according to object's {@link originX} property in parent's coordinate plane\\\n   * if parent is canvas then this property is identical to {@link getX}\n   */\n  getRelativeX(): number {\n    return this.left;\n  }\n\n  /**\n   * @param {number} value x position according to object's {@link originX} property in parent's coordinate plane\\\n   * if parent is canvas then this method is identical to {@link setX}\n   */\n  setRelativeX(value: number) {\n    this.left = value;\n  }\n\n  /**\n   * @returns {number} y position according to object's {@link originY} property in parent's coordinate plane\\\n   * if parent is canvas then this property is identical to {@link getY}\n   */\n  getRelativeY(): number {\n    return this.top;\n  }\n\n  /**\n   * @param {number} value y position according to object's {@link originY} property in parent's coordinate plane\\\n   * if parent is canvas then this property is identical to {@link setY}\n   */\n  setRelativeY(value: number) {\n    this.top = value;\n  }\n\n  /**\n   * @returns {Point} x position according to object's {@link originX} {@link originY} properties in canvas coordinate plane\n   */\n  getXY(): Point {\n    const relativePosition = this.getRelativeXY();\n    return this.group\n      ? transformPoint(relativePosition, this.group.calcTransformMatrix())\n      : relativePosition;\n  }\n\n  /**\n   * Set an object position to a particular point, the point is intended in absolute ( canvas ) coordinate.\n   * You can specify {@link originX} and {@link originY} values,\n   * that otherwise are the object's current values.\n   * @example <caption>Set object's bottom left corner to point (5,5) on canvas</caption>\n   * object.setXY(new Point(5, 5), 'left', 'bottom').\n   * @param {Point} point position in canvas coordinate plane\n   * @param {TOriginX} [originX] Horizontal origin: 'left', 'center' or 'right'\n   * @param {TOriginY} [originY] Vertical origin: 'top', 'center' or 'bottom'\n   */\n  setXY(point: Point, originX?: TOriginX, originY?: TOriginY) {\n    if (this.group) {\n      point = transformPoint(\n        point,\n        invertTransform(this.group.calcTransformMatrix())\n      );\n    }\n    this.setRelativeXY(point, originX, originY);\n  }\n\n  /**\n   * @returns {Point} x,y position according to object's {@link originX} {@link originY} properties in parent's coordinate plane\n   */\n  getRelativeXY(): Point {\n    return new Point(this.left, this.top);\n  }\n\n  /**\n   * As {@link setXY}, but in current parent's coordinate plane (the current group if any or the canvas)\n   * @param {Point} point position according to object's {@link originX} {@link originY} properties in parent's coordinate plane\n   * @param {TOriginX} [originX] Horizontal origin: 'left', 'center' or 'right'\n   * @param {TOriginY} [originY] Vertical origin: 'top', 'center' or 'bottom'\n   */\n  setRelativeXY(\n    point: Point,\n    originX: TOriginX = this.originX,\n    originY: TOriginY = this.originY\n  ) {\n    this.setPositionByOrigin(point, originX, originY);\n  }\n\n  /**\n   * @deprecated intermidiate method to be removed, do not use\n   */\n  protected isStrokeAccountedForInDimensions() {\n    return false;\n  }\n\n  /**\n   * @return {Point[]} [tl, tr, br, bl] in the scene plane\n   */\n  getCoords(): Point[] {\n    const { tl, tr, br, bl } =\n      this.aCoords || (this.aCoords = this.calcACoords());\n    const coords = [tl, tr, br, bl];\n    if (this.group) {\n      const t = this.group.calcTransformMatrix();\n      return coords.map((p) => transformPoint(p, t));\n    }\n    return coords;\n  }\n\n  /**\n   * Checks if object intersects with the scene rect formed by {@link tl} and {@link br}\n   */\n  intersectsWithRect(tl: Point, br: Point): boolean {\n    const intersection = Intersection.intersectPolygonRectangle(\n      this.getCoords(),\n      tl,\n      br\n    );\n    return intersection.status === 'Intersection';\n  }\n\n  /**\n   * Checks if object intersects with another object\n   * @param {Object} other Object to test\n   * @return {Boolean} true if object intersects with another object\n   */\n  intersectsWithObject(other: ObjectGeometry): boolean {\n    const intersection = Intersection.intersectPolygonPolygon(\n      this.getCoords(),\n      other.getCoords()\n    );\n\n    return (\n      intersection.status === 'Intersection' ||\n      intersection.status === 'Coincident' ||\n      other.isContainedWithinObject(this) ||\n      this.isContainedWithinObject(other)\n    );\n  }\n\n  /**\n   * Checks if object is fully contained within area of another object\n   * @param {Object} other Object to test\n   * @return {Boolean} true if object is fully contained within area of another object\n   */\n  isContainedWithinObject(other: ObjectGeometry): boolean {\n    const points = this.getCoords();\n    return points.every((point) => other.containsPoint(point));\n  }\n\n  /**\n   * Checks if object is fully contained within the scene rect formed by {@link tl} and {@link br}\n   */\n  isContainedWithinRect(tl: Point, br: Point): boolean {\n    const { left, top, width, height } = this.getBoundingRect();\n    return (\n      left >= tl.x &&\n      left + width <= br.x &&\n      top >= tl.y &&\n      top + height <= br.y\n    );\n  }\n\n  isOverlapping<T extends ObjectGeometry>(other: T): boolean {\n    return (\n      this.intersectsWithObject(other) ||\n      this.isContainedWithinObject(other) ||\n      other.isContainedWithinObject(this)\n    );\n  }\n\n  /**\n   * Checks if point is inside the object\n   * @param {Point} point Point to check against\n   * @return {Boolean} true if point is inside the object\n   */\n  containsPoint(point: Point): boolean {\n    return Intersection.isPointInPolygon(point, this.getCoords());\n  }\n\n  /**\n   * Checks if object is contained within the canvas with current viewportTransform\n   * the check is done stopping at first point that appears on screen\n   * @return {Boolean} true if object is fully or partially contained within canvas\n   */\n  isOnScreen(): boolean {\n    if (!this.canvas) {\n      return false;\n    }\n    const { tl, br } = this.canvas.vptCoords;\n    const points = this.getCoords();\n    // if some point is on screen, the object is on screen.\n    if (\n      points.some(\n        (point) =>\n          point.x <= br.x &&\n          point.x >= tl.x &&\n          point.y <= br.y &&\n          point.y >= tl.y\n      )\n    ) {\n      return true;\n    }\n    // no points on screen, check intersection with absolute coordinates\n    if (this.intersectsWithRect(tl, br)) {\n      return true;\n    }\n    // check if the object is so big that it contains the entire viewport\n    return this.containsPoint(tl.midPointFrom(br));\n  }\n\n  /**\n   * Checks if object is partially contained within the canvas with current viewportTransform\n   * @return {Boolean} true if object is partially contained within canvas\n   */\n  isPartiallyOnScreen(): boolean {\n    if (!this.canvas) {\n      return false;\n    }\n    const { tl, br } = this.canvas.vptCoords;\n    if (this.intersectsWithRect(tl, br)) {\n      return true;\n    }\n    const allPointsAreOutside = this.getCoords().every(\n      (point) =>\n        (point.x >= br.x || point.x <= tl.x) &&\n        (point.y >= br.y || point.y <= tl.y)\n    );\n    // check if the object is so big that it contains the entire viewport\n    return allPointsAreOutside && this.containsPoint(tl.midPointFrom(br));\n  }\n\n  /**\n   * Returns coordinates of object's bounding rectangle (left, top, width, height)\n   * the box is intended as aligned to axis of canvas.\n   * @return {Object} Object with left, top, width, height properties\n   */\n  getBoundingRect(): TBBox {\n    return makeBoundingBoxFromPoints(this.getCoords());\n  }\n\n  /**\n   * Returns width of an object's bounding box counting transformations\n   * @todo shouldn't this account for group transform and return the actual size in canvas coordinate plane?\n   * @return {Number} width value\n   */\n  getScaledWidth(): number {\n    console.log('getScaledWidth');\n    return this._getTransformedDimensions().x;\n  }\n\n  /**\n   * Returns height of an object bounding box counting transformations\n   * @todo shouldn't this account for group transform and return the actual size in canvas coordinate plane?\n   * @return {Number} height value\n   */\n  getScaledHeight(): number {\n    console.log('getScaledHeight');\n    return this._getTransformedDimensions().y;\n  }\n\n  /**\n   * Scales an object (equally by x and y)\n   * @param {Number} value Scale factor\n   * @return {void}\n   */\n  scale(value: number): void {\n    this._set('scaleX', value);\n    this._set('scaleY', value);\n    this.setCoords();\n  }\n\n  /**\n   * Scales an object to a given width, with respect to bounding box (scaling by x/y equally)\n   * @param {Number} value New width value\n   * @return {void}\n   */\n  scaleToWidth(value: number) {\n    // adjust to bounding rect factor so that rotated shapes would fit as well\n    const boundingRectFactor =\n      this.getBoundingRect().width / this.getScaledWidth();\n    return this.scale(value / this.width / boundingRectFactor);\n  }\n\n  /**\n   * Scales an object to a given height, with respect to bounding box (scaling by x/y equally)\n   * @param {Number} value New height value\n   * @return {void}\n   */\n  scaleToHeight(value: number) {\n    // adjust to bounding rect factor so that rotated shapes would fit as well\n    const boundingRectFactor =\n      this.getBoundingRect().height / this.getScaledHeight();\n    return this.scale(value / this.height / boundingRectFactor);\n  }\n\n  getCanvasRetinaScaling() {\n    return this.canvas?.getRetinaScaling() || 1;\n  }\n\n  /**\n   * Returns the object angle relative to canvas counting also the group property\n   * @returns {TDegree}\n   */\n  getTotalAngle(): TDegree {\n    return this.group\n      ? radiansToDegrees(calcPlaneRotation(this.calcTransformMatrix()))\n      : this.angle;\n  }\n\n  /**\n   * Retrieves viewportTransform from Object's canvas if available\n   * @return {TMat2D}\n   */\n  getViewportTransform(): TMat2D {\n    return this.canvas?.viewportTransform || (iMatrix.concat() as TMat2D);\n  }\n\n  /**\n   * Calculates the coordinates of the 4 corner of the bbox, in absolute coordinates.\n   * those never change with zoom or viewport changes.\n   * @return {TCornerPoint}\n   */\n  calcACoords(): TCornerPoint {\n    const rotateMatrix = createRotateMatrix({ angle: this.angle }),\n      { x, y } = this.getRelativeCenterPoint(),\n      tMatrix = createTranslateMatrix(x, y),\n      finalMatrix = multiplyTransformMatrices(tMatrix, rotateMatrix),\n      dim = this._getTransformedDimensions(),\n      w = dim.x / 2,\n      h = dim.y / 2;\n    return {\n      // corners\n      tl: transformPoint({ x: -w, y: -h }, finalMatrix),\n      tr: transformPoint({ x: w, y: -h }, finalMatrix),\n      bl: transformPoint({ x: -w, y: h }, finalMatrix),\n      br: transformPoint({ x: w, y: h }, finalMatrix),\n    };\n  }\n\n  /**\n   * Sets corner and controls position coordinates based on current angle, width and height, left and top.\n   * aCoords are used to quickly find an object on the canvas.\n   * See {@link https://github.com/fabricjs/fabric.js/wiki/When-to-call-setCoords} and {@link http://fabricjs.com/fabric-gotchas}\n   */\n  setCoords(): void {\n    this.aCoords = this.calcACoords();\n  }\n\n  transformMatrixKey(skipGroup = false): number[] {\n    let prefix: number[] = [];\n    if (!skipGroup && this.group) {\n      prefix = this.group.transformMatrixKey(skipGroup);\n    }\n    prefix.push(\n      this.top,\n      this.left,\n      this.width,\n      this.height,\n      this.scaleX,\n      this.scaleY,\n      this.angle,\n      this.strokeWidth,\n      this.skewX,\n      this.skewY,\n      +this.flipX,\n      +this.flipY,\n      resolveOrigin(this.originX),\n      resolveOrigin(this.originY)\n    );\n\n    return prefix;\n  }\n\n  /**\n   * calculate transform matrix that represents the current transformations from the\n   * object's properties.\n   * @param {Boolean} [skipGroup] return transform matrix for object not counting parent transformations\n   * There are some situation in which this is useful to avoid the fake rotation.\n   * @return {TMat2D} transform matrix for the object\n   */\n  calcTransformMatrix(skipGroup = false): TMat2D {\n    let matrix = this.calcOwnMatrix();\n    if (skipGroup || !this.group) {\n      return matrix;\n    }\n    const key = this.transformMatrixKey(skipGroup),\n      cache = this.matrixCache;\n    if (cache && cache.key.every((x, i) => x === key[i])) {\n      return cache.value;\n    }\n    if (this.group) {\n      matrix = multiplyTransformMatrices(\n        this.group.calcTransformMatrix(false),\n        matrix\n      );\n    }\n    this.matrixCache = {\n      key,\n      value: matrix,\n    };\n    return matrix;\n  }\n\n  /**\n   * calculate transform matrix that represents the current transformations from the\n   * object's properties, this matrix does not include the group transformation\n   * @return {TMat2D} transform matrix for the object\n   */\n  calcOwnMatrix(): TMat2D {\n    const key = this.transformMatrixKey(true),\n      cache = this.ownMatrixCache;\n    if (cache && cache.key === key) {\n      return cache.value;\n    }\n    const center = this.getRelativeCenterPoint(),\n      options = {\n        angle: this.angle,\n        translateX: center.x,\n        translateY: center.y,\n        scaleX: this.scaleX,\n        scaleY: this.scaleY,\n        skewX: this.skewX,\n        skewY: this.skewY,\n        flipX: this.flipX,\n        flipY: this.flipY,\n      },\n      value = composeMatrix(options);\n    this.ownMatrixCache = {\n      key,\n      value,\n    };\n    return value;\n  }\n\n  /**\n   * Calculate object dimensions from its properties\n   * @private\n   * @returns {Point} dimensions\n   */\n  _getNonTransformedDimensions(): Point {\n    return new Point(this.width, this.height).scalarAdd(this.strokeWidth);\n  }\n\n  /**\n   * Calculate object dimensions for controls box, including padding and canvas zoom.\n   * and active selection\n   * @private\n   * @param {object} [options] transform options\n   * @returns {Point} dimensions\n   */\n  _calculateCurrentDimensions(options?: any): Point {\n    return this._getTransformedDimensions(options)\n      .transform(this.getViewportTransform(), true)\n      .scalarAdd(2 * this.padding);\n  }\n}\n","import type { ObjectEvents } from '../../EventTypeDefs';\nimport type { Group } from '../Group';\nimport type { Canvas } from '../../canvas/Canvas';\nimport type { StaticCanvas } from '../../canvas/StaticCanvas';\nimport { ObjectGeometry } from './ObjectGeometry';\n\ntype TAncestor = StackedObject | Canvas | StaticCanvas;\ntype TCollection = Group | Canvas | StaticCanvas;\n\n/**\n * Strict: only ancestors that are objects (without canvas)\n */\nexport type Ancestors<Strict> = Strict extends true\n  ? [StackedObject | Group] | [StackedObject | Group, ...Group[]] | Group[]\n  :\n      | [StackedObject | Group | Canvas | StaticCanvas]\n      | [StackedObject | Group, Canvas | StaticCanvas]\n      | [StackedObject, ...Group[]]\n      | Group[]\n      | [StackedObject | Group, ...Group[], Canvas | StaticCanvas];\n\nexport type AncestryComparison<Strict> = {\n  /**\n   * common ancestors of `this` and`other`(may include`this` | `other`)\n   */\n  common: Ancestors<Strict>;\n  /**\n   * ancestors that are of `this` only\n   */\n  fork: Ancestors<Strict>;\n  /**\n   * ancestors that are of `other` only\n   */\n  otherFork: Ancestors<Strict>;\n};\n\nexport class StackedObject<\n  EventSpec extends ObjectEvents = ObjectEvents\n> extends ObjectGeometry<EventSpec> {\n  /**\n   * A reference to the parent of the object\n   * Used to keep the original parent ref when the object has been added to an ActiveSelection, hence loosing the `group` ref\n   */\n  declare parent?: Group;\n\n  /**\n   * Checks if object is descendant of target\n   * Should be used instead of {@link Group.contains} or {@link StaticCanvas.contains} for performance reasons\n   * @param {TAncestor} target\n   * @returns {boolean}\n   */\n  isDescendantOf(target: TAncestor): boolean {\n    const { parent, group } = this;\n    return (\n      parent === target ||\n      group === target ||\n      this.canvas === target ||\n      // walk up\n      (!!parent && parent.isDescendantOf(target)) ||\n      (!!group && group !== parent && group.isDescendantOf(target))\n    );\n  }\n\n  /**\n   *\n   * @param {boolean} [strict] returns only ancestors that are objects (without canvas)\n   * @returns {Ancestors} ancestors (excluding `ActiveSelection`) from bottom to top\n   */\n  getAncestors<T extends boolean>(strict?: T): Ancestors<T> {\n    const ancestors: TAncestor[] = [];\n    // eslint-disable-next-line @typescript-eslint/no-this-alias\n    let parent: TAncestor | undefined = this;\n    do {\n      parent =\n        parent instanceof StackedObject\n          ? parent.parent ?? (!strict ? parent.canvas : undefined)\n          : undefined;\n      parent && ancestors.push(parent);\n    } while (parent);\n    return ancestors as Ancestors<T>;\n  }\n\n  /**\n   * Compare ancestors\n   *\n   * @param {StackedObject} other\n   * @param {boolean} [strict] finds only ancestors that are objects (without canvas)\n   * @returns {AncestryComparison} an object that represent the ancestry situation.\n   */\n  findCommonAncestors<T extends this, S extends boolean>(\n    other: T,\n    strict?: S\n  ): AncestryComparison<S> {\n    if (this === other) {\n      return {\n        fork: [],\n        otherFork: [],\n        common: [this, ...this.getAncestors(strict)],\n      } as AncestryComparison<S>;\n    }\n    const ancestors = this.getAncestors(strict);\n    const otherAncestors = other.getAncestors(strict);\n    //  if `this` has no ancestors and `this` is top ancestor of `other` we must handle the following case\n    if (\n      ancestors.length === 0 &&\n      otherAncestors.length > 0 &&\n      this === otherAncestors[otherAncestors.length - 1]\n    ) {\n      return {\n        fork: [],\n        otherFork: [\n          other,\n          ...otherAncestors.slice(0, otherAncestors.length - 1),\n        ],\n        common: [this],\n      } as AncestryComparison<S>;\n    }\n    //  compare ancestors\n    for (let i = 0, ancestor; i < ancestors.length; i++) {\n      ancestor = ancestors[i];\n      if (ancestor === other) {\n        return {\n          fork: [this, ...ancestors.slice(0, i)],\n          otherFork: [],\n          common: ancestors.slice(i),\n        } as AncestryComparison<S>;\n      }\n      for (let j = 0; j < otherAncestors.length; j++) {\n        if (this === otherAncestors[j]) {\n          return {\n            fork: [],\n            otherFork: [other, ...otherAncestors.slice(0, j)],\n            common: [this, ...ancestors],\n          } as AncestryComparison<S>;\n        }\n        if (ancestor === otherAncestors[j]) {\n          return {\n            fork: [this, ...ancestors.slice(0, i)],\n            otherFork: [other, ...otherAncestors.slice(0, j)],\n            common: ancestors.slice(i),\n          } as AncestryComparison<S>;\n        }\n      }\n    }\n    // nothing shared\n    return {\n      fork: [this, ...ancestors],\n      otherFork: [other, ...otherAncestors],\n      common: [],\n    } as AncestryComparison<S>;\n  }\n\n  /**\n   *\n   * @param {StackedObject} other\n   * @param {boolean} [strict] checks only ancestors that are objects (without canvas)\n   * @returns {boolean}\n   */\n  hasCommonAncestors<T extends this>(other: T, strict?: boolean): boolean {\n    const commonAncestors = this.findCommonAncestors(other, strict);\n    return commonAncestors && !!commonAncestors.common.length;\n  }\n\n  /**\n   *\n   * @param {FabricObject} other object to compare against\n   * @returns {boolean | undefined} if objects do not share a common ancestor or they are strictly equal it is impossible to determine which is in front of the other; in such cases the function returns `undefined`\n   */\n  isInFrontOf<T extends this>(other: T): boolean | undefined {\n    if (this === other) {\n      return undefined;\n    }\n    const ancestorData = this.findCommonAncestors(other);\n    if (!ancestorData) {\n      return undefined;\n    }\n    if (ancestorData.fork.includes(other as any)) {\n      return true;\n    }\n    if (ancestorData.otherFork.includes(this as any)) {\n      return false;\n    }\n    const firstCommonAncestor = ancestorData.common[0];\n    if (!firstCommonAncestor) {\n      return undefined;\n    }\n    const headOfFork = ancestorData.fork.pop(),\n      headOfOtherFork = ancestorData.otherFork.pop(),\n      thisIndex = (firstCommonAncestor as TCollection)._objects.indexOf(\n        headOfFork as any\n      ),\n      otherIndex = (firstCommonAncestor as TCollection)._objects.indexOf(\n        headOfOtherFork as any\n      );\n    return thisIndex > -1 && thisIndex > otherIndex;\n  }\n}\n","import type { TColorArg } from '../../color/typedefs';\nimport type { ObjectEvents } from '../../EventTypeDefs';\nimport type { TAnimation } from '../../util/animation/animate';\nimport { animate, animateColor } from '../../util/animation/animate';\nimport type {\n  AnimationOptions,\n  ArrayAnimationOptions,\n  ColorAnimationOptions,\n  ValueAnimationOptions,\n} from '../../util/animation/types';\nimport { StackedObject } from './StackedObject';\n\nexport abstract class AnimatableObject<\n  EventSpec extends ObjectEvents = ObjectEvents\n> extends StackedObject<EventSpec> {\n  /**\n   * List of properties to consider for animating colors.\n   * @type String[]\n   */\n  static colorProperties: string[] = ['fill', 'stroke', 'backgroundColor'];\n\n  /**\n   * Animates object's properties\n   * @param {Record<string, number | number[] | TColorArg>} animatable map of keys and end values\n   * @param {Partial<AnimationOptions<T>>} options\n   * @tutorial {@link http://fabricjs.com/fabric-intro-part-2#animation}\n   * @return {Record<string, TAnimation<T>>} map of animation contexts\n   *\n   * As object — multiple properties\n   *\n   * object.animate({ left: ..., top: ... });\n   * object.animate({ left: ..., top: ... }, { duration: ... });\n   */\n  animate<T extends number | number[] | TColorArg>(\n    animatable: Record<string, T>,\n    options?: Partial<AnimationOptions<T>>\n  ): Record<string, TAnimation<T>> {\n    return Object.entries(animatable).reduce((acc, [key, endValue]) => {\n      acc[key] = this._animate(key, endValue, options);\n      return acc;\n    }, {} as Record<string, TAnimation<T>>);\n  }\n\n  /**\n   * @private\n   * @param {String} key Property to animate\n   * @param {String} to Value to animate to\n   * @param {Object} [options] Options object\n   */\n  _animate<T extends number | number[] | TColorArg>(\n    key: string,\n    endValue: T,\n    options: Partial<AnimationOptions<T>> = {}\n  ): TAnimation<T> {\n    const path = key.split('.');\n    const propIsColor = (\n      this.constructor as typeof AnimatableObject\n    ).colorProperties.includes(path[path.length - 1]);\n    const { abort, startValue, onChange, onComplete } = options;\n    const animationOptions = {\n      ...options,\n      target: this,\n      // path.reduce... is the current value in case start value isn't provided\n      startValue:\n        startValue ?? path.reduce((deep: any, key) => deep[key], this),\n      endValue,\n      abort: abort?.bind(this),\n      onChange: (\n        value: number | number[] | string,\n        valueProgress: number,\n        durationProgress: number\n      ) => {\n        path.reduce((deep: Record<string, any>, key, index) => {\n          if (index === path.length - 1) {\n            deep[key] = value;\n          }\n          return deep[key];\n        }, this);\n        onChange &&\n          // @ts-expect-error generic callback arg0 is wrong\n          onChange(value, valueProgress, durationProgress);\n      },\n      onComplete: (\n        value: number | number[] | string,\n        valueProgress: number,\n        durationProgress: number\n      ) => {\n        this.setCoords();\n        onComplete &&\n          // @ts-expect-error generic callback arg0 is wrong\n          onComplete(value, valueProgress, durationProgress);\n      },\n    } as AnimationOptions<T>;\n\n    return (\n      propIsColor\n        ? animateColor(animationOptions as ColorAnimationOptions)\n        : animate(\n            animationOptions as ValueAnimationOptions | ArrayAnimationOptions\n          )\n    ) as TAnimation<T>;\n  }\n}\n","export function getSvgRegex(arr: string[]) {\n  return new RegExp('^(' + arr.join('|') + ')\\\\b', 'i');\n}\n","import { getSvgRegex } from './getSvgRegex';\nimport { LEFT, TOP } from '../constants';\n\nexport const reNum = String.raw`(?:[-+]?(?:\\d*\\.\\d+|\\d+\\.?)(?:[eE][-+]?\\d+)?)`;\n\nexport const svgNS = 'http://www.w3.org/2000/svg';\n\nexport const commaWsp = String.raw`(?:\\s+,?\\s*|,\\s*|$)`;\n\nexport const reFontDeclaration = new RegExp(\n  '(normal|italic)?\\\\s*(normal|small-caps)?\\\\s*' +\n    '(normal|bold|bolder|lighter|100|200|300|400|500|600|700|800|900)?\\\\s*(' +\n    reNum +\n    '(?:px|cm|mm|em|pt|pc|in)*)(?:\\\\/(normal|' +\n    reNum +\n    '))?\\\\s+(.*)'\n);\n\nexport const svgValidTagNames = [\n    'path',\n    'circle',\n    'polygon',\n    'polyline',\n    'ellipse',\n    'rect',\n    'line',\n    'image',\n    'text',\n  ],\n  svgViewBoxElements = ['symbol', 'image', 'marker', 'pattern', 'view', 'svg'],\n  svgInvalidAncestors = [\n    'pattern',\n    'defs',\n    'symbol',\n    'metadata',\n    'clipPath',\n    'mask',\n    'desc',\n  ],\n  svgValidParents = ['symbol', 'g', 'a', 'svg', 'clipPath', 'defs'],\n  attributesMap = {\n    cx: LEFT,\n    x: LEFT,\n    r: 'radius',\n    cy: TOP,\n    y: TOP,\n    display: 'visible',\n    visibility: 'visible',\n    transform: 'transformMatrix',\n    'fill-opacity': 'fillOpacity',\n    'fill-rule': 'fillRule',\n    'font-family': 'fontFamily',\n    'font-size': 'fontSize',\n    'font-style': 'fontStyle',\n    'font-weight': 'fontWeight',\n    'letter-spacing': 'charSpacing',\n    'paint-order': 'paintFirst',\n    'stroke-dasharray': 'strokeDashArray',\n    'stroke-dashoffset': 'strokeDashOffset',\n    'stroke-linecap': 'strokeLineCap',\n    'stroke-linejoin': 'strokeLineJoin',\n    'stroke-miterlimit': 'strokeMiterLimit',\n    'stroke-opacity': 'strokeOpacity',\n    'stroke-width': 'strokeWidth',\n    'text-decoration': 'textDecoration',\n    'text-anchor': 'textAnchor',\n    opacity: 'opacity',\n    'clip-path': 'clipPath',\n    'clip-rule': 'clipRule',\n    'vector-effect': 'strokeUniform',\n    'image-rendering': 'imageSmoothing',\n  },\n  fSize = 'font-size',\n  cPath = 'clip-path';\n\nexport const svgValidTagNamesRegEx = getSvgRegex(svgValidTagNames);\n\nexport const svgViewBoxElementsRegEx = getSvgRegex(svgViewBoxElements);\n\nexport const svgValidParentsRegEx = getSvgRegex(svgValidParents);\n\n// http://www.w3.org/TR/SVG/coords.html#ViewBoxAttribute\n// matches, e.g.: +14.56e-12, etc.\nexport const reViewBoxAttrValue = new RegExp(\n  '^' +\n    '\\\\s*(' +\n    reNum +\n    '+)\\\\s*,?' +\n    '\\\\s*(' +\n    reNum +\n    '+)\\\\s*,?' +\n    '\\\\s*(' +\n    reNum +\n    '+)\\\\s*,?' +\n    '\\\\s*(' +\n    reNum +\n    '+)\\\\s*' +\n    '$'\n);\n","import { classRegistry } from './ClassRegistry';\nimport { Color } from './color/Color';\nimport { config } from './config';\nimport { reNum } from './parser/constants';\nimport { Point } from './Point';\nimport type { FabricObject } from './shapes/Object/FabricObject';\nimport type { TClassProperties } from './typedefs';\nimport { uid } from './util/internals/uid';\nimport { pickBy } from './util/misc/pick';\nimport { degreesToRadians } from './util/misc/radiansDegreesConversion';\nimport { toFixed } from './util/misc/toFixed';\nimport { rotateVector } from './util/misc/vectors';\n\n/**\n   * Regex matching shadow offsetX, offsetY and blur (ex: \"2px 2px 10px rgba(0,0,0,0.2)\", \"rgb(0,255,0) 2px 2px\")\n   * - (?:\\s|^): This part captures either a whitespace character (\\s) or the beginning of a line (^). It's non-capturing (due to (?:...)), meaning it doesn't create a capturing group.\n   * - (-?\\d+(?:\\.\\d*)?(?:px)?(?:\\s?|$))?: This captures the first component of the shadow, which is the horizontal offset. Breaking it down:\n   *   - (-?\\d+): Captures an optional minus sign followed by one or more digits (integer part of the number).\n   *   - (?:\\.\\d*)?: Optionally captures a decimal point followed by zero or more digits (decimal part of the number).\n   *   - (?:px)?: Optionally captures the \"px\" unit.\n   *   - (?:\\s?|$): Captures either an optional whitespace or the end of the line. This whole part is wrapped in a non-capturing group and marked as optional with ?.\n   * - (-?\\d+(?:\\.\\d*)?(?:px)?(?:\\s?|$))?: Similar to the previous step, this captures the vertical offset.\n\n(\\d+(?:\\.\\d*)?(?:px)?)?: This captures the blur radius. It's similar to the horizontal offset but without the optional minus sign.\n\n(?:\\s+(-?\\d+(?:\\.\\d*)?(?:px)?(?:\\s?|$))?){0,1}: This captures an optional part for the color. It allows for whitespace followed by a component with an optional minus sign, digits, decimal point, and \"px\" unit.\n\n(?:$|\\s): This captures either the end of the line or a whitespace character. It ensures that the match ends either at the end of the string or with a whitespace character.\n   */\n// eslint-disable-next-line max-len\n\nconst shadowOffsetRegex = '(-?\\\\d+(?:\\\\.\\\\d*)?(?:px)?(?:\\\\s?|$))?';\n\nconst reOffsetsAndBlur = () =>\n  new RegExp(\n    '(?:\\\\s|^)' +\n      shadowOffsetRegex +\n      shadowOffsetRegex +\n      '(' +\n      reNum +\n      '?(?:px)?)?(?:\\\\s?|$)(?:$|\\\\s)'\n  );\n\nexport const shadowDefaultValues: Partial<TClassProperties<Shadow>> = {\n  color: 'rgb(0,0,0)',\n  blur: 0,\n  offsetX: 0,\n  offsetY: 0,\n  affectStroke: false,\n  includeDefaultValues: true,\n  nonScaling: false,\n};\n\nexport type SerializedShadowOptions = {\n  color: string;\n  blur: number;\n  offsetX: number;\n  offsetY: number;\n  affectStroke: boolean;\n  nonScaling: boolean;\n  type: string;\n};\n\nexport class Shadow {\n  /**\n   * Shadow color\n   * @type String\n   * @default\n   */\n  declare color: string;\n\n  /**\n   * Shadow blur\n   * @type Number\n   */\n  declare blur: number;\n\n  /**\n   * Shadow horizontal offset\n   * @type Number\n   * @default\n   */\n  declare offsetX: number;\n\n  /**\n   * Shadow vertical offset\n   * @type Number\n   * @default\n   */\n  declare offsetY: number;\n\n  /**\n   * Whether the shadow should affect stroke operations\n   * @type Boolean\n   * @default\n   */\n  declare affectStroke: boolean;\n\n  /**\n   * Indicates whether toObject should include default values\n   * @type Boolean\n   * @default\n   */\n  declare includeDefaultValues: boolean;\n\n  /**\n   * When `false`, the shadow will scale with the object.\n   * When `true`, the shadow's offsetX, offsetY, and blur will not be affected by the object's scale.\n   * default to false\n   * @type Boolean\n   * @default\n   */\n  declare nonScaling: boolean;\n\n  declare id: number;\n\n  static ownDefaults = shadowDefaultValues;\n\n  static type = 'shadow';\n\n  /**\n   * @see {@link http://fabricjs.com/shadows|Shadow demo}\n   * @param {Object|String} [options] Options object with any of color, blur, offsetX, offsetY properties or string (e.g. \"rgba(0,0,0,0.2) 2px 2px 10px\")\n   */\n  constructor(options: Partial<TClassProperties<Shadow>>);\n  constructor(svgAttribute: string);\n  constructor(arg0: string | Partial<TClassProperties<Shadow>>) {\n    const options: Partial<TClassProperties<Shadow>> =\n      typeof arg0 === 'string' ? Shadow.parseShadow(arg0) : arg0;\n    Object.assign(this, (this.constructor as typeof Shadow).ownDefaults);\n    for (const prop in options) {\n      // @ts-expect-error for loops are so messy in TS\n      this[prop] = options[prop];\n    }\n\n    this.id = uid();\n  }\n\n  /**\n   * @param {String} value Shadow value to parse\n   * @return {Object} Shadow object with color, offsetX, offsetY and blur\n   */\n  static parseShadow(value: string) {\n    const shadowStr = value.trim(),\n      regex = reOffsetsAndBlur(),\n      [, offsetX = 0, offsetY = 0, blur = 0] = (\n        regex.exec(shadowStr) || []\n      ).map((value) => parseFloat(value) || 0),\n      color = (shadowStr.replace(regex, '') || 'rgb(0,0,0)').trim();\n\n    return {\n      color,\n      offsetX,\n      offsetY,\n      blur,\n    };\n  }\n\n  /**\n   * Returns a string representation of an instance\n   * @see http://www.w3.org/TR/css-text-decor-3/#text-shadow\n   * @return {String} Returns CSS3 text-shadow declaration\n   */\n  toString() {\n    return [this.offsetX, this.offsetY, this.blur, this.color].join('px ');\n  }\n\n  /**\n   * Returns SVG representation of a shadow\n   * @param {FabricObject} object\n   * @return {String} SVG representation of a shadow\n   */\n  toSVG(object: FabricObject) {\n    const offset = rotateVector(\n        new Point(this.offsetX, this.offsetY),\n        degreesToRadians(-object.angle)\n      ),\n      BLUR_BOX = 20,\n      color = new Color(this.color);\n    let fBoxX = 40,\n      fBoxY = 40;\n\n    if (object.width && object.height) {\n      //http://www.w3.org/TR/SVG/filters.html#FilterEffectsRegion\n      // we add some extra space to filter box to contain the blur ( 20 )\n      fBoxX =\n        toFixed(\n          (Math.abs(offset.x) + this.blur) / object.width,\n          config.NUM_FRACTION_DIGITS\n        ) *\n          100 +\n        BLUR_BOX;\n      fBoxY =\n        toFixed(\n          (Math.abs(offset.y) + this.blur) / object.height,\n          config.NUM_FRACTION_DIGITS\n        ) *\n          100 +\n        BLUR_BOX;\n    }\n    if (object.flipX) {\n      offset.x *= -1;\n    }\n    if (object.flipY) {\n      offset.y *= -1;\n    }\n\n    return `<filter id=\"SVGID_${this.id}\" y=\"-${fBoxY}%\" height=\"${\n      100 + 2 * fBoxY\n    }%\" x=\"-${fBoxX}%\" width=\"${\n      100 + 2 * fBoxX\n    }%\" >\\n\\t<feGaussianBlur in=\"SourceAlpha\" stdDeviation=\"${toFixed(\n      this.blur ? this.blur / 2 : 0,\n      config.NUM_FRACTION_DIGITS\n    )}\"></feGaussianBlur>\\n\\t<feOffset dx=\"${toFixed(\n      offset.x,\n      config.NUM_FRACTION_DIGITS\n    )}\" dy=\"${toFixed(\n      offset.y,\n      config.NUM_FRACTION_DIGITS\n    )}\" result=\"oBlur\" ></feOffset>\\n\\t<feFlood flood-color=\"${color.toRgb()}\" flood-opacity=\"${color.getAlpha()}\"/>\\n\\t<feComposite in2=\"oBlur\" operator=\"in\" />\\n\\t<feMerge>\\n\\t\\t<feMergeNode></feMergeNode>\\n\\t\\t<feMergeNode in=\"SourceGraphic\"></feMergeNode>\\n\\t</feMerge>\\n</filter>\\n`;\n  }\n\n  /**\n   * Returns object representation of a shadow\n   * @return {Object} Object representation of a shadow instance\n   */\n  toObject() {\n    const data: SerializedShadowOptions = {\n      color: this.color,\n      blur: this.blur,\n      offsetX: this.offsetX,\n      offsetY: this.offsetY,\n      affectStroke: this.affectStroke,\n      nonScaling: this.nonScaling,\n      type: (this.constructor as typeof Shadow).type,\n    };\n    const defaults = Shadow.ownDefaults as SerializedShadowOptions;\n    return !this.includeDefaultValues\n      ? pickBy(data, (value, key) => value !== defaults[key])\n      : data;\n  }\n\n  static async fromObject(options: Partial<TClassProperties<Shadow>>) {\n    return new this(options);\n  }\n}\n\nclassRegistry.setClass(Shadow, 'shadow');\n","export const cloneDeep = <T extends object>(object: T): T =>\n  JSON.parse(JSON.stringify(object));\n","import { TOP, LEFT } from '../../constants';\nimport type { TClassProperties } from '../../typedefs';\nimport type { InteractiveFabricObject } from './InteractiveObject';\nimport type { FabricObject } from './Object';\n\nexport const stateProperties = [\n  TOP,\n  LEFT,\n  'scaleX',\n  'scaleY',\n  'flipX',\n  'flipY',\n  'originX',\n  'originY',\n  'angle',\n  'opacity',\n  'globalCompositeOperation',\n  'shadow',\n  'visible',\n  'skewX',\n  'skewY',\n];\n\nexport const cacheProperties = [\n  'fill',\n  'stroke',\n  'strokeWidth',\n  'strokeDashArray',\n  'width',\n  'height',\n  'paintFirst',\n  'strokeUniform',\n  'strokeLineCap',\n  'strokeDashOffset',\n  'strokeLineJoin',\n  'strokeMiterLimit',\n  'backgroundColor',\n  'clipPath',\n];\n\nexport const fabricObjectDefaultValues: Partial<\n  TClassProperties<FabricObject>\n> = {\n  // see composeMatrix() to see order of transforms. First defaults listed based on this\n  top: 0,\n  left: 0,\n  width: 0,\n  height: 0,\n  angle: 0,\n  flipX: false,\n  flipY: false,\n  scaleX: 1,\n  scaleY: 1,\n  minScaleLimit: 0,\n  skewX: 0,\n  skewY: 0,\n  originX: LEFT,\n  originY: TOP,\n  strokeWidth: 1,\n  strokeUniform: false,\n  padding: 0,\n  opacity: 1,\n  paintFirst: 'fill',\n  fill: 'rgb(0,0,0)',\n  fillRule: 'nonzero',\n  stroke: null,\n  strokeDashArray: null,\n  strokeDashOffset: 0,\n  strokeLineCap: 'butt',\n  strokeLineJoin: 'miter',\n  strokeMiterLimit: 4,\n  globalCompositeOperation: 'source-over',\n  backgroundColor: '',\n  shadow: null,\n  visible: true,\n  includeDefaultValues: true,\n  excludeFromExport: false,\n  objectCaching: true,\n  clipPath: undefined,\n  inverted: false,\n  absolutePositioned: false,\n  centeredRotation: true,\n  centeredScaling: false,\n  dirty: true,\n} as const;\n\nexport const interactiveObjectDefaultValues: Partial<\n  TClassProperties<InteractiveFabricObject>\n> = {\n  noScaleCache: true,\n  lockMovementX: false,\n  lockMovementY: false,\n  lockRotation: false,\n  lockScalingX: false,\n  lockScalingY: false,\n  lockSkewingX: false,\n  lockSkewingY: false,\n  lockScalingFlip: false,\n  cornerSize: 13,\n  touchCornerSize: 24,\n  transparentCorners: true,\n  cornerColor: 'rgb(178,204,255)',\n  cornerStrokeColor: '',\n  cornerStyle: 'rect',\n  cornerDashArray: null,\n  hasControls: true,\n  borderColor: 'rgb(178,204,255)',\n  borderDashArray: null,\n  borderOpacityWhenMoving: 0.4,\n  borderScaleFactor: 1,\n  hasBorders: true,\n  selectionBackgroundColor: '',\n  selectable: true,\n  evented: true,\n  perPixelTargetFind: false,\n  activeOn: 'down',\n  hoverCursor: null,\n  moveCursor: null,\n};\n","import { cache } from '../../cache';\nimport { config } from '../../config';\nimport {\n  ALIASING_LIMIT,\n  CENTER,\n  iMatrix,\n  LEFT,\n  TOP,\n  VERSION,\n} from '../../constants';\nimport type { ObjectEvents } from '../../EventTypeDefs';\nimport { AnimatableObject } from './AnimatableObject';\nimport { Point, XY } from '../../Point';\nimport { Shadow } from '../../Shadow';\nimport type {\n  TDegree,\n  TFiller,\n  TSize,\n  TCacheCanvasDimensions,\n  Abortable,\n  TOptions,\n  ImageFormat,\n} from '../../typedefs';\nimport { classRegistry } from '../../ClassRegistry';\nimport { runningAnimations } from '../../util/animation/AnimationRegistry';\nimport { cloneDeep } from '../../util/internals/cloneDeep';\nimport { capValue } from '../../util/misc/capValue';\nimport { createCanvasElement, toDataURL } from '../../util/misc/dom';\nimport {\n  invertTransform,\n  multiplyTransformMatrices,\n  qrDecompose,\n} from '../../util/misc/matrix';\nimport { enlivenObjectEnlivables } from '../../util/misc/objectEnlive';\nimport {\n  resetObjectTransform,\n  saveObjectTransform,\n} from '../../util/misc/objectTransforms';\nimport { sendObjectToPlane } from '../../util/misc/planeChange';\nimport { pick, pickBy } from '../../util/misc/pick';\nimport { toFixed } from '../../util/misc/toFixed';\nimport type { Group } from '../Group';\nimport { StaticCanvas } from '../../canvas/StaticCanvas';\nimport {\n  isFiller,\n  isSerializableFiller,\n  isTextObject,\n} from '../../util/typeAssertions';\nimport type { FabricImage } from '../Image';\nimport {\n  cacheProperties,\n  fabricObjectDefaultValues,\n  stateProperties,\n} from './defaultValues';\nimport type { Gradient } from '../../gradient/Gradient';\nimport type { Pattern } from '../../Pattern';\nimport type { Canvas } from '../../canvas/Canvas';\nimport type { SerializedObjectProps } from './types/SerializedObjectProps';\nimport type { ObjectProps } from './types/ObjectProps';\nimport { getDevicePixelRatio, getEnv } from '../../env';\nimport { log } from '../../util/internals/console';\n\nexport type TCachedFabricObject<T extends FabricObject = FabricObject> = T &\n  Required<\n    Pick<\n      T,\n      | 'zoomX'\n      | 'zoomY'\n      | '_cacheCanvas'\n      | '_cacheContext'\n      | 'cacheTranslationX'\n      | 'cacheTranslationY'\n    >\n  > & {\n    _cacheContext: CanvasRenderingContext2D;\n  };\n\nexport type ObjectToCanvasElementOptions = {\n  format?: ImageFormat;\n  /** Multiplier to scale by */\n  multiplier?: number;\n  /** Cropping left offset. Introduced in v1.2.14 */\n  left?: number;\n  /** Cropping top offset. Introduced in v1.2.14 */\n  top?: number;\n  /** Cropping width. Introduced in v1.2.14 */\n  width?: number;\n  /** Cropping height. Introduced in v1.2.14 */\n  height?: number;\n  /** Enable retina scaling for clone image. Introduce in 1.6.4 */\n  enableRetinaScaling?: boolean;\n  /** Remove current object transform ( no scale , no angle, no flip, no skew ). Introduced in 2.3.4 */\n  withoutTransform?: boolean;\n  /** Remove current object shadow. Introduced in 2.4.2 */\n  withoutShadow?: boolean;\n  /** Account for canvas viewport transform */\n  viewportTransform?: boolean;\n\n  id?: string;\n  /** Function to create the output canvas to export onto */\n  canvasProvider?: <T extends StaticCanvas>(el?: HTMLCanvasElement) => T;\n};\n\ntype toDataURLOptions = ObjectToCanvasElementOptions & {\n  quality?: number;\n};\n\n/**\n * Root object class from which all 2d shape classes inherit from\n * @tutorial {@link http://fabricjs.com/fabric-intro-part-1#objects}\n *\n * @fires added\n * @fires removed\n *\n * @fires selected\n * @fires deselected\n *\n * @fires rotating\n * @fires scaling\n * @fires moving\n * @fires skewing\n * @fires modified\n *\n * @fires mousedown\n * @fires mouseup\n * @fires mouseover\n * @fires mouseout\n * @fires mousewheel\n * @fires mousedblclick\n *\n * @fires dragover\n * @fires dragenter\n * @fires dragleave\n * @fires drop\n */\nexport class FabricObject<\n    Props extends TOptions<ObjectProps> = Partial<ObjectProps>,\n    SProps extends SerializedObjectProps = SerializedObjectProps,\n    EventSpec extends ObjectEvents = ObjectEvents\n  >\n  extends AnimatableObject<EventSpec>\n  implements ObjectProps\n{\n  declare minScaleLimit: number;\n\n  declare opacity: number;\n\n  declare paintFirst: 'fill' | 'stroke';\n  declare fill: string | TFiller | null;\n  declare fillRule: CanvasFillRule;\n  declare stroke: string | TFiller | null;\n  declare strokeDashArray: number[] | null;\n  declare strokeDashOffset: number;\n  declare strokeLineCap: CanvasLineCap;\n  declare strokeLineJoin: CanvasLineJoin;\n  declare strokeMiterLimit: number;\n\n  declare globalCompositeOperation: GlobalCompositeOperation;\n  declare backgroundColor: string;\n\n  declare shadow: Shadow | null;\n\n  declare visible: boolean;\n\n  declare includeDefaultValues: boolean;\n  declare excludeFromExport: boolean;\n\n  declare objectCaching: boolean;\n\n  declare clipPath?: FabricObject;\n  declare inverted: boolean;\n  declare absolutePositioned: boolean;\n  declare centeredRotation: boolean;\n  declare centeredScaling: boolean;\n\n  //canvasx specific\n  declare id: string;\n  declare locked: boolean;\n\n  /**\n   * This list of properties is used to check if the state of an object is changed.\n   * This state change now is only used for children of groups to understand if a group\n   * needs its cache regenerated during a .set call\n   * @type Array\n   */\n  static stateProperties: string[] = stateProperties;\n\n  /**\n   * List of properties to consider when checking if cache needs refresh\n   * Those properties are checked by\n   * calls to Object.set(key, value). If the key is in this list, the object is marked as dirty\n   * and refreshed at the next render\n   * @type Array\n   */\n  static cacheProperties: string[] = cacheProperties;\n\n  /**\n   * When set to `true`, object's cache will be rerendered next render call.\n   * since 1.7.0\n   * @type Boolean\n   * @default true\n   */\n  declare dirty: boolean;\n\n  /**\n   * Quick access for the _cacheCanvas rendering context\n   * This is part of the objectCaching feature\n   * since 1.7.0\n   * @type boolean\n   * @default undefined\n   * @private\n   */\n  _cacheContext: CanvasRenderingContext2D | null = null;\n\n  /**\n   * A reference to the HTMLCanvasElement that is used to contain the cache of the object\n   * this canvas element is resized and cleared as needed\n   * Is marked private, you can read it, don't use it since it is handled by fabric\n   * since 1.7.0\n   * @type HTMLCanvasElement\n   * @default undefined\n   * @private\n   */\n  declare _cacheCanvas?: HTMLCanvasElement;\n\n  /**\n   * Size of the cache canvas, width\n   * since 1.7.0\n   * @type number\n   * @default undefined\n   * @private\n   */\n  declare cacheWidth?: number;\n\n  /**\n   * Size of the cache canvas, height\n   * since 1.7.0\n   * @type number\n   * @default undefined\n   * @private\n   */\n  declare cacheHeight?: number;\n\n  /**\n   * zoom level used on the cacheCanvas to draw the cache, X axe\n   * since 1.7.0\n   * @type number\n   * @default undefined\n   * @private\n   */\n  declare zoomX?: number;\n\n  /**\n   * zoom level used on the cacheCanvas to draw the cache, Y axe\n   * since 1.7.0\n   * @type number\n   * @default undefined\n   * @private\n   */\n  declare zoomY?: number;\n\n  /**\n   * zoom level used on the cacheCanvas to draw the cache, Y axe\n   * since 1.7.0\n   * @type number\n   * @default undefined\n   * @private\n   */\n  declare cacheTranslationX?: number;\n\n  /**\n   * translation of the cacheCanvas away from the center, for subpixel accuracy and crispness\n   * since 1.7.0\n   * @type number\n   * @default undefined\n   * @private\n   */\n  declare cacheTranslationY?: number;\n\n  /**\n   * A reference to the parent of the object, usually a Group\n   * @type number\n   * @default undefined\n   * @private\n   */\n  declare group?: Group;\n\n  /**\n   * Indicate if the object is sitting on a cache dedicated to it\n   * or is part of a larger cache for many object ( a group for example)\n   * @type number\n   * @default undefined\n   * @private\n   */\n  declare ownCaching?: boolean;\n\n  /**\n   * Private. indicates if the object inside a group is on a transformed context or not\n   * or is part of a larger cache for many object ( a group for example)\n   * @type boolean\n   * @default undefined\n   * @private\n   */\n  declare _transformDone?: boolean;\n\n  static ownDefaults = fabricObjectDefaultValues;\n\n  static getDefaults(): Record<string, any> {\n    return { ...FabricObject.ownDefaults };\n  }\n\n  /**\n   * The class type. Used to identify which class this is.\n   * This is used for serialization purposes and internally it can be used\n   * to identify classes. As a developer you could use `instance of Class`\n   * but to avoid importing all the code and blocking tree shaking we try\n   * to avoid doing that.\n   */\n  static type = 'FabricObject';\n\n  /**\n   * Legacy identifier of the class. Prefer using utils like isType or instanceOf\n   * Will be removed in fabric 7 or 8.\n   * The setter exists because is very hard to catch all the ways in which a type value\n   * could be set in the instance\n   * @TODO add sustainable warning message\n   * @type string\n   * @deprecated\n   */\n  get type() {\n    const name = (this.constructor as typeof FabricObject).type;\n    if (name === 'FabricObject') {\n      return 'object';\n    }\n    return name.toLowerCase();\n  }\n\n  set type(value) {\n    log('warn', 'Setting type has no effect', value);\n  }\n\n  /**\n   * Constructor\n   * @param {Object} [options] Options object\n   */\n  constructor(options: Props = {} as Props) {\n    super();\n    Object.assign(\n      this,\n      (this.constructor as typeof FabricObject).getDefaults()\n    );\n    this.setOptions(options);\n  }\n\n  /**\n   * Create a the canvas used to keep the cached copy of the object\n   * @private\n   */\n  _createCacheCanvas() {\n    this._cacheCanvas = createCanvasElement();\n    this._cacheContext = this._cacheCanvas.getContext('2d');\n    this._updateCacheCanvas();\n    // if canvas gets created, is empty, so dirty.\n    this.dirty = true;\n  }\n\n  /**\n   * Limit the cache dimensions so that X * Y do not cross config.perfLimitSizeTotal\n   * and each side do not cross fabric.cacheSideLimit\n   * those numbers are configurable so that you can get as much detail as you want\n   * making bargain with performances.\n   * @param {Object} dims\n   * @param {Object} dims.width width of canvas\n   * @param {Object} dims.height height of canvas\n   * @param {Object} dims.zoomX zoomX zoom value to unscale the canvas before drawing cache\n   * @param {Object} dims.zoomY zoomY zoom value to unscale the canvas before drawing cache\n   * @return {Object}.width width of canvas\n   * @return {Object}.height height of canvas\n   * @return {Object}.zoomX zoomX zoom value to unscale the canvas before drawing cache\n   * @return {Object}.zoomY zoomY zoom value to unscale the canvas before drawing cache\n   */\n  _limitCacheSize(\n    dims: TSize & { zoomX: number; zoomY: number; capped: boolean } & any\n  ) {\n    const width = dims.width,\n      height = dims.height,\n      max = config.maxCacheSideLimit,\n      min = config.minCacheSideLimit;\n    if (\n      width <= max &&\n      height <= max &&\n      width * height <= config.perfLimitSizeTotal\n    ) {\n      if (width < min) {\n        dims.width = min;\n      }\n      if (height < min) {\n        dims.height = min;\n      }\n      return dims;\n    }\n    const ar = width / height,\n      [limX, limY] = cache.limitDimsByArea(ar),\n      x = capValue(min, limX, max),\n      y = capValue(min, limY, max);\n    if (width > x) {\n      dims.zoomX /= width / x;\n      dims.width = x;\n      dims.capped = true;\n    }\n    if (height > y) {\n      dims.zoomY /= height / y;\n      dims.height = y;\n      dims.capped = true;\n    }\n    return dims;\n  }\n\n  /**\n   * Return the dimension and the zoom level needed to create a cache canvas\n   * big enough to host the object to be cached.\n   * @private\n   * @return {Object}.x width of object to be cached\n   * @return {Object}.y height of object to be cached\n   * @return {Object}.width width of canvas\n   * @return {Object}.height height of canvas\n   * @return {Object}.zoomX zoomX zoom value to unscale the canvas before drawing cache\n   * @return {Object}.zoomY zoomY zoom value to unscale the canvas before drawing cache\n   */\n  _getCacheCanvasDimensions(): TCacheCanvasDimensions {\n    const objectScale = this.getTotalObjectScaling(),\n      // calculate dimensions without skewing\n      dim = this._getTransformedDimensions({ skewX: 0, skewY: 0 }),\n      neededX = (dim.x * objectScale.x) / this.scaleX,\n      neededY = (dim.y * objectScale.y) / this.scaleY;\n    return {\n      // for sure this ALIASING_LIMIT is slightly creating problem\n      // in situation in which the cache canvas gets an upper limit\n      // also objectScale contains already scaleX and scaleY\n      width: neededX + ALIASING_LIMIT,\n      height: neededY + ALIASING_LIMIT,\n      zoomX: objectScale.x,\n      zoomY: objectScale.y,\n      x: neededX,\n      y: neededY,\n    };\n  }\n\n  /**\n   * Update width and height of the canvas for cache\n   * returns true or false if canvas needed resize.\n   * @private\n   * @return {Boolean} true if the canvas has been resized\n   */\n  _updateCacheCanvas() {\n    const canvas = this._cacheCanvas,\n      context = this._cacheContext,\n      dims = this._limitCacheSize(this._getCacheCanvasDimensions()),\n      minCacheSize = config.minCacheSideLimit,\n      width = dims.width,\n      height = dims.height,\n      zoomX = dims.zoomX,\n      zoomY = dims.zoomY,\n      dimensionsChanged =\n        width !== this.cacheWidth || height !== this.cacheHeight,\n      zoomChanged = this.zoomX !== zoomX || this.zoomY !== zoomY;\n\n    if (!canvas || !context) {\n      return false;\n    }\n\n    let drawingWidth,\n      drawingHeight,\n      shouldRedraw = dimensionsChanged || zoomChanged,\n      additionalWidth = 0,\n      additionalHeight = 0,\n      shouldResizeCanvas = false;\n\n    if (dimensionsChanged) {\n      const canvasWidth = (this._cacheCanvas as HTMLCanvasElement).width,\n        canvasHeight = (this._cacheCanvas as HTMLCanvasElement).height,\n        sizeGrowing = width > canvasWidth || height > canvasHeight,\n        sizeShrinking =\n          (width < canvasWidth * 0.9 || height < canvasHeight * 0.9) &&\n          canvasWidth > minCacheSize &&\n          canvasHeight > minCacheSize;\n      shouldResizeCanvas = sizeGrowing || sizeShrinking;\n      if (\n        sizeGrowing &&\n        !dims.capped &&\n        (width > minCacheSize || height > minCacheSize)\n      ) {\n        additionalWidth = width * 0.1;\n        additionalHeight = height * 0.1;\n      }\n    }\n    if (isTextObject(this) && this.path) {\n      shouldRedraw = true;\n      shouldResizeCanvas = true;\n      // IMHO in those lines we are using zoomX and zoomY not the this version.\n      additionalWidth += this.getHeightOfLine(0) * this.zoomX!;\n      additionalHeight += this.getHeightOfLine(0) * this.zoomY!;\n    }\n    if (shouldRedraw) {\n      if (shouldResizeCanvas) {\n        canvas.width = Math.ceil(width + additionalWidth);\n        canvas.height = Math.ceil(height + additionalHeight);\n      } else {\n        context.setTransform(1, 0, 0, 1, 0, 0);\n        context.clearRect(0, 0, canvas.width, canvas.height);\n      }\n      drawingWidth = dims.x / 2;\n      drawingHeight = dims.y / 2;\n      this.cacheTranslationX =\n        Math.round(canvas.width / 2 - drawingWidth) + drawingWidth;\n      this.cacheTranslationY =\n        Math.round(canvas.height / 2 - drawingHeight) + drawingHeight;\n      this.cacheWidth = width;\n      this.cacheHeight = height;\n      context.translate(this.cacheTranslationX, this.cacheTranslationY);\n      context.scale(zoomX, zoomY);\n      this.zoomX = zoomX;\n      this.zoomY = zoomY;\n      return true;\n    }\n    return false;\n  }\n\n  /**\n   * Sets object's properties from options, for class constructor only.\n   * Needs to be overridden for different defaults.\n   * @protected\n   * @param {Object} [options] Options object\n   */\n  protected setOptions(options: Record<string, any> = {}) {\n    this._setOptions(options);\n  }\n\n  /**\n   * Transforms context when rendering an object\n   * @param {CanvasRenderingContext2D} ctx Context\n   */\n  transform(ctx: CanvasRenderingContext2D) {\n    const needFullTransform =\n      (this.group && !this.group._transformDone) ||\n      (this.group && this.canvas && ctx === (this.canvas as Canvas).contextTop);\n    const m = this.calcTransformMatrix(!needFullTransform);\n    ctx.transform(m[0], m[1], m[2], m[3], m[4], m[5]);\n  }\n\n  /**\n   * Returns an object representation of an instance\n   * @param {string[]} [propertiesToInclude] Any properties that you might want to additionally include in the output\n   * @return {Object} Object representation of an instance\n   */\n  toObject(propertiesToInclude: any[] = []): any {\n    const NUM_FRACTION_DIGITS = config.NUM_FRACTION_DIGITS,\n      clipPathData =\n        this.clipPath && !this.clipPath.excludeFromExport\n          ? {\n              ...this.clipPath.toObject(propertiesToInclude),\n              inverted: this.clipPath.inverted,\n              absolutePositioned: this.clipPath.absolutePositioned,\n            }\n          : null,\n      object = {\n        ...pick(this, propertiesToInclude as (keyof this)[]),\n        type: (this.constructor as typeof FabricObject).type,\n        version: VERSION,\n        originX: this.originX,\n        originY: this.originY,\n        left: toFixed(this.left, NUM_FRACTION_DIGITS),\n        top: toFixed(this.top, NUM_FRACTION_DIGITS),\n        width: toFixed(this.width, NUM_FRACTION_DIGITS),\n        height: toFixed(this.height, NUM_FRACTION_DIGITS),\n        fill: isSerializableFiller(this.fill)\n          ? this.fill.toObject()\n          : this.fill,\n        stroke: isSerializableFiller(this.stroke)\n          ? this.stroke.toObject()\n          : this.stroke,\n        strokeWidth: toFixed(this.strokeWidth, NUM_FRACTION_DIGITS),\n        strokeDashArray: this.strokeDashArray\n          ? this.strokeDashArray.concat()\n          : this.strokeDashArray,\n        strokeLineCap: this.strokeLineCap,\n        strokeDashOffset: this.strokeDashOffset,\n        strokeLineJoin: this.strokeLineJoin,\n        strokeUniform: this.strokeUniform,\n        strokeMiterLimit: toFixed(this.strokeMiterLimit, NUM_FRACTION_DIGITS),\n        scaleX: toFixed(this.scaleX, NUM_FRACTION_DIGITS),\n        scaleY: toFixed(this.scaleY, NUM_FRACTION_DIGITS),\n        angle: toFixed(this.angle, NUM_FRACTION_DIGITS),\n        flipX: this.flipX,\n        flipY: this.flipY,\n        opacity: toFixed(this.opacity, NUM_FRACTION_DIGITS),\n        shadow:\n          this.shadow && this.shadow.toObject\n            ? this.shadow.toObject()\n            : this.shadow,\n        visible: this.visible,\n        backgroundColor: this.backgroundColor,\n        fillRule: this.fillRule,\n        paintFirst: this.paintFirst,\n        globalCompositeOperation: this.globalCompositeOperation,\n        skewX: toFixed(this.skewX, NUM_FRACTION_DIGITS),\n        skewY: toFixed(this.skewY, NUM_FRACTION_DIGITS),\n        ...(clipPathData ? { clipPath: clipPathData } : null),\n      };\n\n    return !this.includeDefaultValues\n      ? this._removeDefaultValues(object)\n      : object;\n  }\n\n  /**\n   * Returns (dataless) object representation of an instance\n   * @param {Array} [propertiesToInclude] Any properties that you might want to additionally include in the output\n   * @return {Object} Object representation of an instance\n   */\n  toDatalessObject(propertiesToInclude?: any[]): any {\n    // will be overwritten by subclasses\n    return this.toObject(propertiesToInclude);\n  }\n\n  /**\n   * @private\n   * @param {Object} object\n   */\n  _removeDefaultValues<T extends object>(object: T): Partial<T> {\n    // getDefaults() ( get from static ownDefaults ) should win over prototype since anyway they get assigned to instance\n    // ownDefault vs prototype is swappable only if you change all the fabric objects consistently.\n    const defaults = (this.constructor as typeof FabricObject).getDefaults();\n    const hasStaticDefaultValues = Object.keys(defaults).length > 0;\n    const baseValues = hasStaticDefaultValues\n      ? defaults\n      : Object.getPrototypeOf(this);\n\n    return pickBy(object, (value, key) => {\n      if (key === LEFT || key === TOP || key === 'type') {\n        return true;\n      }\n      const baseValue = baseValues[key];\n      return (\n        value !== baseValue &&\n        // basically a check for [] === []\n        !(\n          Array.isArray(value) &&\n          Array.isArray(baseValue) &&\n          value.length === 0 &&\n          baseValue.length === 0\n        )\n      );\n    });\n  }\n\n  /**\n   * Returns a string representation of an instance\n   * @return {String}\n   */\n  toString() {\n    return `#<${(this.constructor as typeof FabricObject).type}>`;\n  }\n\n  /**\n   * Return the object scale factor counting also the group scaling\n   * @return {Point}\n   */\n  getObjectScaling() {\n    // if the object is a top level one, on the canvas, we go for simple aritmetic\n    // otherwise the complex method with angles will return approximations and decimals\n    // and will likely kill the cache when not needed\n    // https://github.com/fabricjs/fabric.js/issues/7157\n    if (!this.group) {\n      return new Point(Math.abs(this.scaleX), Math.abs(this.scaleY));\n    }\n    // if we are inside a group total zoom calculation is complex, we defer to generic matrices\n    const options = qrDecompose(this.calcTransformMatrix());\n    return new Point(Math.abs(options.scaleX), Math.abs(options.scaleY));\n  }\n\n  /**\n   * Return the object scale factor counting also the group scaling, zoom and retina\n   * @return {Object} object with scaleX and scaleY properties\n   */\n  getTotalObjectScaling() {\n    const scale = this.getObjectScaling();\n    if (this.canvas) {\n      const zoom = this.canvas.getZoom();\n      const retina = this.getCanvasRetinaScaling();\n      return scale.scalarMultiply(zoom * retina);\n    }\n    return scale;\n  }\n\n  /**\n   * Return the object opacity counting also the group property\n   * @return {Number}\n   */\n  getObjectOpacity() {\n    let opacity = this.opacity;\n    if (this.group) {\n      opacity *= this.group.getObjectOpacity();\n    }\n    return opacity;\n  }\n\n  /**\n   * Makes sure the scale is valid and modifies it if necessary\n   * @todo: this is a control action issue, not a geometry one\n   * @private\n   * @param {Number} value, unconstrained\n   * @return {Number} constrained value;\n   */\n  _constrainScale(value: number): number {\n    if (Math.abs(value) < this.minScaleLimit) {\n      if (value < 0) {\n        return -this.minScaleLimit;\n      } else {\n        return this.minScaleLimit;\n      }\n    } else if (value === 0) {\n      return 0.0001;\n    }\n    return value;\n  }\n\n  /**\n   * Handles setting values on the instance and handling internal side effects\n   * @protected\n   * @param {String} key\n   * @param {*} value\n   */\n  _set(key: string, value: any) {\n    if (key === 'scaleX' || key === 'scaleY') {\n      value = this._constrainScale(value);\n    }\n    if (key === 'scaleX' && value < 0) {\n      this.flipX = !this.flipX;\n      value *= -1;\n    } else if (key === 'scaleY' && value < 0) {\n      this.flipY = !this.flipY;\n      value *= -1;\n      // i don't like this automatic initialization here\n    } else if (key === 'shadow' && value && !(value instanceof Shadow)) {\n      value = new Shadow(value);\n    }\n\n    const isChanged = this[key as keyof this] !== value;\n    this[key as keyof this] = value;\n\n    // invalidate caches\n    if (\n      isChanged &&\n      (this.constructor as typeof FabricObject).cacheProperties.includes(key)\n    ) {\n      this.dirty = true;\n    }\n    // a dirty child makes the parent dirty.\n    // but a non dirty child does not make the parent not dirty.\n    // the parent could be dirty for some other reason.\n    this.parent &&\n      (this.dirty ||\n        (isChanged &&\n          (this.constructor as typeof FabricObject).stateProperties.includes(\n            key\n          ))) &&\n      this.parent._set('dirty', true);\n\n    return this;\n  }\n\n  /*\n   * @private\n   * return if the object would be visible in rendering\n   * @memberOf FabricObject.prototype\n   * @return {Boolean}\n   */\n  isNotVisible() {\n    return (\n      this.opacity === 0 ||\n      (!this.width && !this.height && this.strokeWidth === 0) ||\n      !this.visible\n    );\n  }\n\n  /**\n   * Renders an object on a specified context\n   * @param {CanvasRenderingContext2D} ctx Context to render on\n   */\n  render(ctx: CanvasRenderingContext2D) {\n    // do not render if width/height are zeros or object is not visible\n    if (this.isNotVisible()) {\n      return;\n    }\n    if (\n      this.canvas &&\n      this.canvas.skipOffscreen &&\n      !this.group &&\n      !this.isOnScreen()\n    ) {\n      return;\n    }\n    ctx.save();\n    this._setupCompositeOperation(ctx);\n    this.drawSelectionBackground(ctx);\n    this.transform(ctx);\n    this._setOpacity(ctx);\n    this._setShadow(ctx);\n    if (this.shouldCache()) {\n      this.renderCache();\n      (this as TCachedFabricObject).drawCacheOnCanvas(ctx);\n    } else {\n      this._removeCacheCanvas();\n      this.drawObject(ctx);\n      this.dirty = false;\n    }\n    ctx.restore();\n  }\n\n  drawSelectionBackground(ctx: CanvasRenderingContext2D) {\n    /* no op */\n  }\n\n  renderCache(options?: any) {\n    options = options || {};\n    if (!this._cacheCanvas || !this._cacheContext) {\n      this._createCacheCanvas();\n    }\n    if (this.isCacheDirty() && this._cacheContext) {\n      this.drawObject(this._cacheContext, options.forClipping);\n      this.dirty = false;\n    }\n  }\n\n  /**\n   * Remove cacheCanvas and its dimensions from the objects\n   */\n  _removeCacheCanvas() {\n    this._cacheCanvas = undefined;\n    this._cacheContext = null;\n    this.cacheWidth = 0;\n    this.cacheHeight = 0;\n  }\n\n  /**\n   * return true if the object will draw a stroke\n   * Does not consider text styles. This is just a shortcut used at rendering time\n   * We want it to be an approximation and be fast.\n   * wrote to avoid extra caching, it has to return true when stroke happens,\n   * can guess when it will not happen at 100% chance, does not matter if it misses\n   * some use case where the stroke is invisible.\n   * @since 3.0.0\n   * @returns Boolean\n   */\n  hasStroke() {\n    return (\n      this.stroke && this.stroke !== 'transparent' && this.strokeWidth !== 0\n    );\n  }\n\n  /**\n   * return true if the object will draw a fill\n   * Does not consider text styles. This is just a shortcut used at rendering time\n   * We want it to be an approximation and be fast.\n   * wrote to avoid extra caching, it has to return true when fill happens,\n   * can guess when it will not happen at 100% chance, does not matter if it misses\n   * some use case where the fill is invisible.\n   * @since 3.0.0\n   * @returns Boolean\n   */\n  hasFill() {\n    return this.fill && this.fill !== 'transparent';\n  }\n\n  /**\n   * When set to `true`, force the object to have its own cache, even if it is inside a group\n   * it may be needed when your object behave in a particular way on the cache and always needs\n   * its own isolated canvas to render correctly.\n   * Created to be overridden\n   * since 1.7.12\n   * @returns Boolean\n   */\n  needsItsOwnCache() {\n    if (\n      this.paintFirst === 'stroke' &&\n      this.hasFill() &&\n      this.hasStroke() &&\n      !!this.shadow\n    ) {\n      return true;\n    }\n    if (this.clipPath) {\n      return true;\n    }\n    return false;\n  }\n\n  /**\n   * Decide if the object should cache or not. Create its own cache level\n   * objectCaching is a global flag, wins over everything\n   * needsItsOwnCache should be used when the object drawing method requires\n   * a cache step. None of the fabric classes requires it.\n   * Generally you do not cache objects in groups because the group outside is cached.\n   * Read as: cache if is needed, or if the feature is enabled but we are not already caching.\n   * @return {Boolean}\n   */\n  shouldCache() {\n    this.ownCaching =\n      this.needsItsOwnCache() ||\n      (this.objectCaching && (!this.parent || !this.parent.isOnACache()));\n    return this.ownCaching;\n  }\n\n  /**\n   * Check if this object will cast a shadow with an offset.\n   * used by Group.shouldCache to know if child has a shadow recursively\n   * @return {Boolean}\n   * @deprecated\n   */\n  willDrawShadow() {\n    return (\n      !!this.shadow && (this.shadow.offsetX !== 0 || this.shadow.offsetY !== 0)\n    );\n  }\n\n  /**\n   * Execute the drawing operation for an object clipPath\n   * @param {CanvasRenderingContext2D} ctx Context to render on\n   * @param {FabricObject} clipPath\n   */\n  drawClipPathOnCache(\n    ctx: CanvasRenderingContext2D,\n    clipPath: TCachedFabricObject\n  ) {\n    ctx.save();\n    // DEBUG: uncomment this line, comment the following\n    // ctx.globalAlpha = 0.4\n    if (clipPath.inverted) {\n      ctx.globalCompositeOperation = 'destination-out';\n    } else {\n      ctx.globalCompositeOperation = 'destination-in';\n    }\n    //ctx.scale(1 / 2, 1 / 2);\n    if (clipPath.absolutePositioned) {\n      const m = invertTransform(this.calcTransformMatrix());\n      ctx.transform(m[0], m[1], m[2], m[3], m[4], m[5]);\n    }\n    clipPath.transform(ctx);\n    ctx.scale(1 / clipPath.zoomX, 1 / clipPath.zoomY);\n    ctx.drawImage(\n      clipPath._cacheCanvas,\n      -clipPath.cacheTranslationX,\n      -clipPath.cacheTranslationY\n    );\n    ctx.restore();\n  }\n\n  /**\n   * Execute the drawing operation for an object on a specified context\n   * @param {CanvasRenderingContext2D} ctx Context to render on\n   * @param {boolean} forClipping apply clipping styles\n   */\n  drawObject(ctx: CanvasRenderingContext2D, forClipping?: boolean) {\n    const originalFill = this.fill,\n      originalStroke = this.stroke;\n    if (forClipping) {\n      this.fill = 'black';\n      this.stroke = '';\n      this._setClippingProperties(ctx);\n    } else {\n      this._renderBackground(ctx);\n    }\n    this._render(ctx);\n    this._drawClipPath(ctx, this.clipPath);\n    this.fill = originalFill;\n    this.stroke = originalStroke;\n  }\n\n  /**\n   * Prepare clipPath state and cache and draw it on instance's cache\n   * @param {CanvasRenderingContext2D} ctx\n   * @param {FabricObject} clipPath\n   */\n  _drawClipPath(ctx: CanvasRenderingContext2D, clipPath?: FabricObject) {\n    if (!clipPath) {\n      return;\n    }\n    // needed to setup a couple of variables\n    // path canvas gets overridden with this one.\n    // TODO find a better solution?\n    clipPath._set('canvas', this.canvas);\n    clipPath.shouldCache();\n    clipPath._transformDone = true;\n    clipPath.renderCache({ forClipping: true });\n    this.drawClipPathOnCache(ctx, clipPath as TCachedFabricObject);\n  }\n\n  /**\n   * Paint the cached copy of the object on the target context.\n   * @param {CanvasRenderingContext2D} ctx Context to render on\n   */\n  drawCacheOnCanvas(this: TCachedFabricObject, ctx: CanvasRenderingContext2D) {\n    ctx.scale(1 / this.zoomX, 1 / this.zoomY);\n    ctx.drawImage(\n      this._cacheCanvas,\n      -this.cacheTranslationX,\n      -this.cacheTranslationY\n    );\n  }\n\n  /**\n   * Check if cache is dirty\n   * @param {Boolean} skipCanvas skip canvas checks because this object is painted\n   * on parent canvas.\n   */\n  isCacheDirty(skipCanvas = false) {\n    if (this.isNotVisible()) {\n      return false;\n    }\n    if (\n      this._cacheCanvas &&\n      this._cacheContext &&\n      !skipCanvas &&\n      this._updateCacheCanvas()\n    ) {\n      // in this case the context is already cleared.\n      return true;\n    } else {\n      if (this.dirty || (this.clipPath && this.clipPath.absolutePositioned)) {\n        if (this._cacheCanvas && this._cacheContext && !skipCanvas) {\n          const width = this.cacheWidth! / this.zoomX!;\n          const height = this.cacheHeight! / this.zoomY!;\n          this._cacheContext.clearRect(-width / 2, -height / 2, width, height);\n        }\n        return true;\n      }\n    }\n    return false;\n  }\n\n  /**\n   * Draws a background for the object big as its untransformed dimensions\n   * @private\n   * @param {CanvasRenderingContext2D} ctx Context to render on\n   */\n  _renderBackground(ctx: CanvasRenderingContext2D) {\n    if (!this.backgroundColor) {\n      return;\n    }\n    const dim = this._getNonTransformedDimensions();\n    ctx.fillStyle = this.backgroundColor;\n\n    ctx.fillRect(-dim.x / 2, -dim.y / 2, dim.x, dim.y);\n    // if there is background color no other shadows\n    // should be casted\n    this._removeShadow(ctx);\n  }\n\n  /**\n   * @private\n   * @param {CanvasRenderingContext2D} ctx Context to render on\n   */\n  _setOpacity(ctx: CanvasRenderingContext2D) {\n    if (this.group && !this.group._transformDone) {\n      ctx.globalAlpha = this.getObjectOpacity();\n    } else {\n      ctx.globalAlpha *= this.opacity;\n    }\n  }\n\n  _setStrokeStyles(\n    ctx: CanvasRenderingContext2D,\n    decl: Pick<\n      this,\n      | 'stroke'\n      | 'strokeWidth'\n      | 'strokeLineCap'\n      | 'strokeDashOffset'\n      | 'strokeLineJoin'\n      | 'strokeMiterLimit'\n    >\n  ) {\n    const stroke = decl.stroke;\n    if (stroke) {\n      ctx.lineWidth = decl.strokeWidth;\n      ctx.lineCap = decl.strokeLineCap;\n      ctx.lineDashOffset = decl.strokeDashOffset;\n      ctx.lineJoin = decl.strokeLineJoin;\n      ctx.miterLimit = decl.strokeMiterLimit;\n      if (isFiller(stroke)) {\n        if (\n          (stroke as Gradient<'linear'>).gradientUnits === 'percentage' ||\n          (stroke as Gradient<'linear'>).gradientTransform ||\n          (stroke as Pattern).patternTransform\n        ) {\n          // need to transform gradient in a pattern.\n          // this is a slow process. If you are hitting this codepath, and the object\n          // is not using caching, you should consider switching it on.\n          // we need a canvas as big as the current object caching canvas.\n          this._applyPatternForTransformedGradient(ctx, stroke);\n        } else {\n          // is a simple gradient or pattern\n          ctx.strokeStyle = stroke.toLive(ctx)!;\n          this._applyPatternGradientTransform(ctx, stroke);\n        }\n      } else {\n        // is a color\n        ctx.strokeStyle = decl.stroke as string;\n      }\n    }\n  }\n\n  _setFillStyles(ctx: CanvasRenderingContext2D, { fill }: Pick<this, 'fill'>) {\n    if (fill) {\n      if (isFiller(fill)) {\n        ctx.fillStyle = fill.toLive(ctx)!;\n        this._applyPatternGradientTransform(ctx, fill);\n      } else {\n        ctx.fillStyle = fill;\n      }\n    }\n  }\n\n  _setClippingProperties(ctx: CanvasRenderingContext2D) {\n    ctx.globalAlpha = 1;\n    ctx.strokeStyle = 'transparent';\n    ctx.fillStyle = '#000000';\n  }\n\n  /**\n   * @private\n   * Sets line dash\n   * @param {CanvasRenderingContext2D} ctx Context to set the dash line on\n   * @param {Array} dashArray array representing dashes\n   */\n  _setLineDash(ctx: CanvasRenderingContext2D, dashArray?: number[] | null) {\n    if (!dashArray || dashArray.length === 0) {\n      return;\n    }\n    // Spec requires the concatenation of two copies of the dash array when the number of elements is odd\n    if (1 & dashArray.length) {\n      dashArray.push(...dashArray);\n    }\n    ctx.setLineDash(dashArray);\n  }\n\n  /**\n   * @private\n   * @param {CanvasRenderingContext2D} ctx Context to render on\n   */\n  _setShadow(ctx: CanvasRenderingContext2D) {\n    if (!this.shadow) {\n      return;\n    }\n\n    const shadow = this.shadow,\n      canvas = this.canvas,\n      retinaScaling = this.getCanvasRetinaScaling(),\n      [sx, , , sy] = canvas?.viewportTransform || iMatrix,\n      multX = sx * retinaScaling,\n      multY = sy * retinaScaling,\n      scaling = shadow.nonScaling ? new Point(1, 1) : this.getObjectScaling();\n    ctx.shadowColor = shadow.color;\n    ctx.shadowBlur =\n      (shadow.blur *\n        config.browserShadowBlurConstant *\n        (multX + multY) *\n        (scaling.x + scaling.y)) /\n      4;\n    ctx.shadowOffsetX = shadow.offsetX * multX * scaling.x;\n    ctx.shadowOffsetY = shadow.offsetY * multY * scaling.y;\n  }\n\n  /**\n   * @private\n   * @param {CanvasRenderingContext2D} ctx Context to render on\n   */\n  _removeShadow(ctx: CanvasRenderingContext2D) {\n    if (!this.shadow) {\n      return;\n    }\n\n    ctx.shadowColor = '';\n    ctx.shadowBlur = ctx.shadowOffsetX = ctx.shadowOffsetY = 0;\n  }\n\n  /**\n   * @private\n   * @param {CanvasRenderingContext2D} ctx Context to render on\n   * @param {TFiller} filler {@link Pattern} or {@link Gradient}\n   */\n  _applyPatternGradientTransform(\n    ctx: CanvasRenderingContext2D,\n    filler: TFiller\n  ) {\n    if (!isFiller(filler)) {\n      return { offsetX: 0, offsetY: 0 };\n    }\n    const t =\n      (filler as Gradient<'linear'>).gradientTransform ||\n      (filler as Pattern).patternTransform;\n    const offsetX = -this.width / 2 + filler.offsetX || 0,\n      offsetY = -this.height / 2 + filler.offsetY || 0;\n\n    if ((filler as Gradient<'linear'>).gradientUnits === 'percentage') {\n      ctx.transform(this.width, 0, 0, this.height, offsetX, offsetY);\n    } else {\n      ctx.transform(1, 0, 0, 1, offsetX, offsetY);\n    }\n    if (t) {\n      ctx.transform(t[0], t[1], t[2], t[3], t[4], t[5]);\n    }\n    return { offsetX: offsetX, offsetY: offsetY };\n  }\n\n  /**\n   * @private\n   * @param {CanvasRenderingContext2D} ctx Context to render on\n   */\n  _renderPaintInOrder(ctx: CanvasRenderingContext2D) {\n    if (this.paintFirst === 'stroke') {\n      this._renderStroke(ctx);\n      this._renderFill(ctx);\n    } else {\n      this._renderFill(ctx);\n      this._renderStroke(ctx);\n    }\n  }\n\n  /**\n   * @private\n   * function that actually render something on the context.\n   * empty here to allow Obects to work on tests to benchmark fabric functionalites\n   * not related to rendering\n   * @param {CanvasRenderingContext2D} ctx Context to render on\n   */\n  _render(ctx: CanvasRenderingContext2D) {\n    // placeholder to be overridden\n  }\n\n  /**\n   * @private\n   * @param {CanvasRenderingContext2D} ctx Context to render on\n   */\n  _renderFill(ctx: CanvasRenderingContext2D) {\n    if (!this.fill) {\n      return;\n    }\n\n    ctx.save();\n    this._setFillStyles(ctx, this);\n    if (this.fillRule === 'evenodd') {\n      ctx.fill('evenodd');\n    } else {\n      ctx.fill();\n    }\n    ctx.restore();\n  }\n\n  /**\n   * @private\n   * @param {CanvasRenderingContext2D} ctx Context to render on\n   */\n  _renderStroke(ctx: CanvasRenderingContext2D) {\n    if (!this.stroke || this.strokeWidth === 0) {\n      return;\n    }\n\n    if (this.shadow && !this.shadow.affectStroke) {\n      this._removeShadow(ctx);\n    }\n\n    ctx.save();\n    if (this.strokeUniform) {\n      const scaling = this.getObjectScaling();\n      ctx.scale(1 / scaling.x, 1 / scaling.y);\n    }\n    this._setLineDash(ctx, this.strokeDashArray);\n    this._setStrokeStyles(ctx, this);\n    ctx.stroke();\n    ctx.restore();\n  }\n\n  /**\n   * This function try to patch the missing gradientTransform on canvas gradients.\n   * transforming a context to transform the gradient, is going to transform the stroke too.\n   * we want to transform the gradient but not the stroke operation, so we create\n   * a transformed gradient on a pattern and then we use the pattern instead of the gradient.\n   * this method has drawbacks: is slow, is in low resolution, needs a patch for when the size\n   * is limited.\n   * @private\n   * @param {CanvasRenderingContext2D} ctx Context to render on\n   * @param {Gradient} filler\n   */\n  _applyPatternForTransformedGradient(\n    ctx: CanvasRenderingContext2D,\n    filler: TFiller\n  ) {\n    const dims = this._limitCacheSize(this._getCacheCanvasDimensions()),\n      pCanvas = createCanvasElement(),\n      retinaScaling = this.getCanvasRetinaScaling(),\n      width = dims.x / this.scaleX / retinaScaling,\n      height = dims.y / this.scaleY / retinaScaling;\n    // in case width and height are less than 1px, we have to round up.\n    // since the pattern is no-repeat, this is fine\n    pCanvas.width = Math.ceil(width);\n    pCanvas.height = Math.ceil(height);\n    const pCtx = pCanvas.getContext('2d');\n    if (!pCtx) {\n      return;\n    }\n    pCtx.beginPath();\n    pCtx.moveTo(0, 0);\n    pCtx.lineTo(width, 0);\n    pCtx.lineTo(width, height);\n    pCtx.lineTo(0, height);\n    pCtx.closePath();\n    pCtx.translate(width / 2, height / 2);\n    pCtx.scale(\n      dims.zoomX / this.scaleX / retinaScaling,\n      dims.zoomY / this.scaleY / retinaScaling\n    );\n    this._applyPatternGradientTransform(pCtx, filler);\n    pCtx.fillStyle = filler.toLive(ctx)!;\n    pCtx.fill();\n    ctx.translate(\n      -this.width / 2 - this.strokeWidth / 2,\n      -this.height / 2 - this.strokeWidth / 2\n    );\n    ctx.scale(\n      (retinaScaling * this.scaleX) / dims.zoomX,\n      (retinaScaling * this.scaleY) / dims.zoomY\n    );\n    ctx.strokeStyle = pCtx.createPattern(pCanvas, 'no-repeat') ?? '';\n  }\n\n  /**\n   * This function is an helper for svg import. it returns the center of the object in the svg\n   * untransformed coordinates\n   * @private\n   * @return {Point} center point from element coordinates\n   */\n  _findCenterFromElement() {\n    return new Point(this.left + this.width / 2, this.top + this.height / 2);\n  }\n\n  /**\n   * Clones an instance.\n   * @param {Array} [propertiesToInclude] Any properties that you might want to additionally include in the output\n   * @returns {Promise<FabricObject>}\n   */\n  clone(propertiesToInclude?: string[]): Promise<this> {\n    const objectForm = this.toObject(propertiesToInclude);\n    return (this.constructor as typeof FabricObject).fromObject(\n      objectForm\n    ) as unknown as Promise<this>;\n  }\n\n  /**\n   * Creates an instance of Image out of an object\n   * makes use of toCanvasElement.\n   * Once this method was based on toDataUrl and loadImage, so it also had a quality\n   * and format option. toCanvasElement is faster and produce no loss of quality.\n   * If you need to get a real Jpeg or Png from an object, using toDataURL is the right way to do it.\n   * toCanvasElement and then toBlob from the obtained canvas is also a good option.\n   * @todo fix the export type, it could not be Image but the type that getClass return for 'image'.\n   * @param {ObjectToCanvasElementOptions} [options] for clone as image, passed to toDataURL\n   * @param {Number} [options.multiplier=1] Multiplier to scale by\n   * @param {Number} [options.left] Cropping left offset. Introduced in v1.2.14\n   * @param {Number} [options.top] Cropping top offset. Introduced in v1.2.14\n   * @param {Number} [options.width] Cropping width. Introduced in v1.2.14\n   * @param {Number} [options.height] Cropping height. Introduced in v1.2.14\n   * @param {Boolean} [options.enableRetinaScaling] Enable retina scaling for clone image. Introduce in 1.6.4\n   * @param {Boolean} [options.withoutTransform] Remove current object transform ( no scale , no angle, no flip, no skew ). Introduced in 2.3.4\n   * @param {Boolean} [options.withoutShadow] Remove current object shadow. Introduced in 2.4.2\n   * @return {FabricImage} Object cloned as image.\n   */\n  cloneAsImage(options: ObjectToCanvasElementOptions): FabricImage {\n    const canvasEl = this.toCanvasElement(options);\n    // TODO: how to import Image w/o an import cycle?\n    const ImageClass = classRegistry.getClass<typeof FabricImage>('image');\n    return new ImageClass(canvasEl);\n  }\n\n  /**\n   * Converts an object into a HTMLCanvas element\n   * @param {ObjectToCanvasElementOptions} options Options object\n   * @param {Number} [options.multiplier=1] Multiplier to scale by\n   * @param {Number} [options.left] Cropping left offset. Introduced in v1.2.14\n   * @param {Number} [options.top] Cropping top offset. Introduced in v1.2.14\n   * @param {Number} [options.width] Cropping width. Introduced in v1.2.14\n   * @param {Number} [options.height] Cropping height. Introduced in v1.2.14\n   * @param {Boolean} [options.enableRetinaScaling] Enable retina scaling for clone image. Introduce in 1.6.4\n   * @param {Boolean} [options.withoutTransform] Remove current object transform ( no scale , no angle, no flip, no skew ). Introduced in 2.3.4\n   * @param {Boolean} [options.withoutShadow] Remove current object shadow. Introduced in 2.4.2\n   * @param {Boolean} [options.viewportTransform] Account for canvas viewport transform\n   * @param {(el?: HTMLCanvasElement) => StaticCanvas} [options.canvasProvider] Create the output canvas\n   * @return {HTMLCanvasElement} Returns DOM element <canvas> with the FabricObject\n   */\n  toCanvasElement(options: ObjectToCanvasElementOptions = {}) {\n    const origParams = saveObjectTransform(this),\n      originalGroup = this.group,\n      originalShadow = this.shadow,\n      abs = Math.abs,\n      retinaScaling = options.enableRetinaScaling ? getDevicePixelRatio() : 1,\n      multiplier = (options.multiplier || 1) * retinaScaling,\n      canvasProvider: (el: HTMLCanvasElement) => StaticCanvas =\n        options.canvasProvider ||\n        ((el: HTMLCanvasElement) =>\n          new StaticCanvas(el, {\n            enableRetinaScaling: false,\n            renderOnAddRemove: false,\n            skipOffscreen: false,\n          }));\n    delete this.group;\n    if (options.withoutTransform) {\n      resetObjectTransform(this);\n    }\n    if (options.withoutShadow) {\n      this.shadow = null;\n    }\n    if (options.viewportTransform) {\n      sendObjectToPlane(this, this.getViewportTransform());\n    }\n\n    this.setCoords();\n    const el = createCanvasElement(),\n      boundingRect = this.getBoundingRect(),\n      shadow = this.shadow,\n      shadowOffset = new Point();\n\n    if (shadow) {\n      const shadowBlur = shadow.blur;\n      const scaling = shadow.nonScaling\n        ? new Point(1, 1)\n        : this.getObjectScaling();\n      // consider non scaling shadow.\n      shadowOffset.x =\n        2 * Math.round(abs(shadow.offsetX) + shadowBlur) * abs(scaling.x);\n      shadowOffset.y =\n        2 * Math.round(abs(shadow.offsetY) + shadowBlur) * abs(scaling.y);\n    }\n    const width = boundingRect.width + shadowOffset.x,\n      height = boundingRect.height + shadowOffset.y;\n    // if the current width/height is not an integer\n    // we need to make it so.\n    el.width = Math.ceil(width);\n    el.height = Math.ceil(height);\n    const canvas = canvasProvider(el);\n    if (options.format === 'jpeg') {\n      canvas.backgroundColor = '#fff';\n    }\n    this.setPositionByOrigin(\n      new Point(canvas.width / 2, canvas.height / 2),\n      CENTER,\n      CENTER\n    );\n    const originalCanvas = this.canvas;\n    // static canvas and canvas have both an array of InteractiveObjects\n    // @ts-expect-error this needs to be fixed somehow, or ignored globally\n    canvas._objects = [this];\n    this.set('canvas', canvas);\n    this.setCoords();\n    const canvasEl = canvas.toCanvasElement(multiplier || 1, options);\n    this.set('canvas', originalCanvas);\n    this.shadow = originalShadow;\n    if (originalGroup) {\n      this.group = originalGroup;\n    }\n    this.set(origParams);\n    this.setCoords();\n    // canvas.dispose will call image.dispose that will nullify the elements\n    // since this canvas is a simple element for the process, we remove references\n    // to objects in this way in order to avoid object trashing.\n    canvas._objects = [];\n    // since render has settled it is safe to destroy canvas\n    canvas.destroy();\n    return canvasEl;\n  }\n\n  /**\n   * Converts an object into a data-url-like string\n   * @param {Object} options Options object\n   * @param {String} [options.format=png] The format of the output image. Either \"jpeg\" or \"png\"\n   * @param {Number} [options.quality=1] Quality level (0..1). Only used for jpeg.\n   * @param {Number} [options.multiplier=1] Multiplier to scale by\n   * @param {Number} [options.left] Cropping left offset. Introduced in v1.2.14\n   * @param {Number} [options.top] Cropping top offset. Introduced in v1.2.14\n   * @param {Number} [options.width] Cropping width. Introduced in v1.2.14\n   * @param {Number} [options.height] Cropping height. Introduced in v1.2.14\n   * @param {Boolean} [options.enableRetinaScaling] Enable retina scaling for clone image. Introduce in 1.6.4\n   * @param {Boolean} [options.withoutTransform] Remove current object transform ( no scale , no angle, no flip, no skew ). Introduced in 2.3.4\n   * @param {Boolean} [options.withoutShadow] Remove current object shadow. Introduced in 2.4.2\n   * @return {String} Returns a data: URL containing a representation of the object in the format specified by options.format\n   */\n  toDataURL(options: toDataURLOptions = {}) {\n    return toDataURL(\n      this.toCanvasElement(options),\n      options.format || 'png',\n      options.quality || 1\n    );\n  }\n\n  /**\n   * Returns true if any of the specified types is identical to the type of an instance\n   * @param {String} type Type to check against\n   * @return {Boolean}\n   */\n  isType(...types: string[]) {\n    return (\n      types.includes((this.constructor as typeof FabricObject).type) ||\n      types.includes(this.type)\n    );\n  }\n\n  /**\n   * Returns complexity of an instance\n   * @return {Number} complexity of this instance (is 1 unless subclassed)\n   */\n  complexity() {\n    return 1;\n  }\n\n  /**\n   * Returns a JSON representation of an instance\n   * @return {Object} JSON\n   */\n  toJSON() {\n    // delegate, not alias\n    return this.toObject();\n  }\n\n  /**\n   * Sets \"angle\" of an instance with centered rotation\n   * @param {TDegree} angle Angle value (in degrees)\n   */\n  rotate(angle: TDegree) {\n    const { centeredRotation, originX, originY } = this;\n\n    if (centeredRotation) {\n      const { x, y } = this.getRelativeCenterPoint();\n      this.originX = CENTER;\n      this.originY = CENTER;\n      this.left = x;\n      this.top = y;\n    }\n\n    this.set('angle', angle);\n\n    if (centeredRotation) {\n      const { x, y } = this.translateToOriginPoint(\n        this.getRelativeCenterPoint(),\n        originX,\n        originY\n      );\n      this.left = x;\n      this.top = y;\n      this.originX = originX;\n      this.originY = originY;\n    }\n  }\n\n  /**\n   * This callback function is called by the parent group of an object every\n   * time a non-delegated property changes on the group. It is passed the key\n   * and value as parameters. Not adding in this function's signature to avoid\n   * Travis build error about unused variables.\n   */\n  setOnGroup() {\n    // implemented by sub-classes, as needed.\n  }\n\n  /**\n   * Sets canvas globalCompositeOperation for specific object\n   * custom composition operation for the particular object can be specified using globalCompositeOperation property\n   * @param {CanvasRenderingContext2D} ctx Rendering canvas context\n   */\n  _setupCompositeOperation(ctx: CanvasRenderingContext2D) {\n    if (this.globalCompositeOperation) {\n      ctx.globalCompositeOperation = this.globalCompositeOperation;\n    }\n  }\n\n  /**\n   * cancel instance's running animations\n   * override if necessary to dispose artifacts such as `clipPath`\n   */\n  dispose() {\n    runningAnimations.cancelByTarget(this);\n    this.off();\n    this._set('canvas', undefined);\n    // clear caches\n    this._cacheCanvas && getEnv().dispose(this._cacheCanvas);\n    this._cacheCanvas = undefined;\n    this._cacheContext = null;\n  }\n\n  /**\n   *\n   * @param {Function} klass\n   * @param {object} object\n   * @param {object} [options]\n   * @param {string} [options.extraParam] property to pass as first argument to the constructor\n   * @param {AbortSignal} [options.signal] handle aborting, see https://developer.mozilla.org/en-US/docs/Web/API/AbortController/signal\n   * @returns {Promise<FabricObject>}\n   */\n  static _fromObject<S extends FabricObject>(\n    { type, ...object }: Record<string, unknown>,\n    { extraParam, ...options }: Abortable & { extraParam?: string } = {}\n  ): Promise<S> {\n    return enlivenObjectEnlivables<any>(cloneDeep(object), options).then(\n      (enlivedMap) => {\n        const allOptions = { ...options, ...enlivedMap };\n        // from the resulting enlived options, extract options.extraParam to arg0\n        // to avoid accidental overrides later\n        if (extraParam) {\n          const { [extraParam]: arg0, ...rest } = allOptions;\n          // @ts-expect-error different signature\n          return new this(arg0, rest);\n        } else {\n          return new this(allOptions);\n        }\n      }\n    ) as Promise<S>;\n  }\n\n  /**\n   *\n   * @param {object} object\n   * @param {object} [options]\n   * @param {AbortSignal} [options.signal] handle aborting, see https://developer.mozilla.org/en-US/docs/Web/API/AbortController/signal\n   * @returns {Promise<FabricObject>}\n   */\n  static fromObject<T extends TOptions<SerializedObjectProps>>(\n    object: T,\n    options?: Abortable\n  ) {\n    return this._fromObject(object, options);\n  }\n\n  getText() {\n    //@ts-ignore\n    return this.text & this.text?.join('\\n');\n  }\n\n  saveData(type: string, fields: string[]): boolean {\n    return true;\n  }\n  transformPointToCanvas(point: XY) {\n    const self = this;\n    const toTransformPoint = new Point(point);\n    const transformedPoint = toTransformPoint.transform(\n      self.calcTransformMatrix()\n    );\n    return transformedPoint;\n  }\n  transformPointFromCanvas(point: XY) {\n    const self = this;\n    const toTransformPoint = new Point(point);\n    const transformedPoint = toTransformPoint.transform(\n      invertTransform(self.calcTransformMatrix())\n    );\n    return transformedPoint;\n  }\n\n  transformPointToViewport(point: XY) {\n    const self = this;\n    const toTransformPoint = new Point(point);\n\n    const mCanvas = self.canvas?.viewportTransform;\n    const mObject = self.calcTransformMatrix();\n    const matrix = mCanvas\n      ? multiplyTransformMatrices(mCanvas, mObject)\n      : mObject;\n\n    const transformedPoint = toTransformPoint.transform(matrix);\n    return transformedPoint;\n  }\n}\n\n// import '../canvasx/X_Object';\n\n// FabricObject.prototype.helloWorld = function () {\n//   console.log('Hello World');\n// };\n\n// Object.assign(FabricObject.prototype, FabricObject2.prototype);\n\nclassRegistry.setClass(FabricObject);\nclassRegistry.setClass(FabricObject, 'object');\n","import type {\n  TModificationEvents,\n  Transform,\n  TransformActionHandler,\n} from '../EventTypeDefs';\nimport { fireEvent } from './fireEvent';\nimport { commonEventInfo } from './util';\n\n/**\n * Wrap an action handler with firing an event if the action is performed\n * @param {TModificationEvents} eventName the event we want to fire\n * @param {TransformActionHandler<T>} actionHandler the function to wrap\n * @param {object} extraEventInfo extra information to pas to the event handler\n * @return {TransformActionHandler<T>} a function with an action handler signature\n */\nexport const wrapWithFireEvent = <\n  T extends Transform,\n  P extends object = Record<string, never>\n>(\n  eventName: TModificationEvents,\n  actionHandler: TransformActionHandler<T>,\n  extraEventInfo?: P\n) => {\n  return ((eventData, transform, x, y) => {\n    const actionPerformed = actionHandler(eventData, transform, x, y);\n    if (actionPerformed) {\n      fireEvent(eventName, {\n        ...commonEventInfo(eventData, transform, x, y),\n        ...extraEventInfo,\n      });\n    }\n    return actionPerformed;\n  }) as TransformActionHandler<T>;\n};\n","import type { Transform, TransformActionHandler } from '../EventTypeDefs';\n\n/**\n * Wrap an action handler with saving/restoring object position on the transform.\n * this is the code that permits to objects to keep their position while transforming.\n * @param {Function} actionHandler the function to wrap\n * @return {Function} a function with an action handler signature\n */\nexport function wrapWithFixedAnchor<T extends Transform>(\n  actionHandler: TransformActionHandler<T>\n) {\n  return ((eventData, transform, x, y) => {\n    const { target, originX, originY } = transform,\n      centerPoint = target.getRelativeCenterPoint(),\n      constraint = target.translateToOriginPoint(centerPoint, originX, originY),\n      actionPerformed = actionHandler(eventData, transform, x, y);\n    // flipping requires to change the transform origin, so we read from the mutated transform\n    // instead of leveraging the one destructured before\n    target.setPositionByOrigin(\n      constraint,\n      transform.originX,\n      transform.originY\n    );\n    return actionPerformed;\n  }) as TransformActionHandler<T>;\n}\n","import type { TransformActionHandler } from '../EventTypeDefs';\nimport { CENTER, LEFT, RIGHT } from '../constants';\nimport { getLocalPoint, isTransformCentered } from './util';\nimport { wrapWithFireEvent } from './wrapWithFireEvent';\nimport { wrapWithFixedAnchor } from './wrapWithFixedAnchor';\n\n/**\n * Action handler to change object's width\n * Needs to be wrapped with `wrapWithFixedAnchor` to be effective\n * @param {Event} eventData javascript event that is doing the transform\n * @param {Object} transform javascript object containing a series of information around the current transform\n * @param {number} x current mouse x position, canvas normalized\n * @param {number} y current mouse y position, canvas normalized\n * @return {Boolean} true if some change happened\n */\nexport const changeObjectWidth: TransformActionHandler = (\n  eventData,\n  transform,\n  x,\n  y\n) => {\n  const localPoint = getLocalPoint(\n    transform,\n    transform.originX,\n    transform.originY,\n    x,\n    y\n  );\n  //  make sure the control changes width ONLY from it's side of target\n  if (\n    transform.originX === CENTER ||\n    (transform.originX === RIGHT && localPoint.x < 0) ||\n    (transform.originX === LEFT && localPoint.x > 0)\n  ) {\n    const { target } = transform,\n      strokePadding =\n        target.strokeWidth / (target.strokeUniform ? target.scaleX : 1),\n      multiplier = isTransformCentered(transform) ? 2 : 1,\n      oldWidth = target.width,\n      newWidth = Math.ceil(\n        Math.abs((localPoint.x * multiplier) / target.scaleX) - strokePadding\n      );\n    target.set('width', Math.max(newWidth, 0));\n    //  check against actual target width in case `newWidth` was rejected\n    return oldWidth !== target.width;\n  }\n  return false;\n};\n\nexport const changeWidth = wrapWithFireEvent(\n  'resizing',\n  wrapWithFixedAnchor(changeObjectWidth)\n);\n","import { twoMathPi } from '../constants';\nimport type { InteractiveFabricObject } from '../shapes/Object/InteractiveObject';\nimport { degreesToRadians } from '../util/misc/radiansDegreesConversion';\nimport type { Control } from './Control';\n\nexport type ControlRenderingStyleOverride = Partial<\n  Pick<\n    InteractiveFabricObject,\n    | 'cornerStyle'\n    | 'cornerSize'\n    | 'cornerColor'\n    | 'cornerStrokeColor'\n    | 'cornerDashArray'\n    | 'transparentCorners'\n  >\n>;\n\nexport type ControlRenderer<\n  O extends InteractiveFabricObject = InteractiveFabricObject\n> = (\n  ctx: CanvasRenderingContext2D,\n  left: number,\n  top: number,\n  styleOverride: ControlRenderingStyleOverride,\n  fabricObject: O\n) => void;\n\n/**\n * Render a round control, as per fabric features.\n * This function is written to respect object properties like transparentCorners, cornerSize\n * cornerColor, cornerStrokeColor\n * plus the addition of offsetY and offsetX.\n * @param {CanvasRenderingContext2D} ctx context to render on\n * @param {Number} left x coordinate where the control center should be\n * @param {Number} top y coordinate where the control center should be\n * @param {Object} styleOverride override for FabricObject controls style\n * @param {FabricObject} fabricObject the fabric object for which we are rendering controls\n */\nexport function renderCircleControl(\n  this: Control,\n  ctx: CanvasRenderingContext2D,\n  left: number,\n  top: number,\n  styleOverride: ControlRenderingStyleOverride,\n  fabricObject: InteractiveFabricObject\n) {\n  styleOverride = styleOverride || {};\n  const xSize =\n      this.sizeX || styleOverride.cornerSize || fabricObject.cornerSize,\n    ySize = this.sizeY || styleOverride.cornerSize || fabricObject.cornerSize,\n    transparentCorners =\n      typeof styleOverride.transparentCorners !== 'undefined'\n        ? styleOverride.transparentCorners\n        : fabricObject.transparentCorners,\n    methodName = transparentCorners ? 'stroke' : 'fill',\n    stroke =\n      !transparentCorners &&\n      (styleOverride.cornerStrokeColor || fabricObject.cornerStrokeColor);\n  let myLeft = left,\n    myTop = top,\n    size;\n  ctx.save();\n  ctx.fillStyle = styleOverride.cornerColor || fabricObject.cornerColor || '';\n  ctx.strokeStyle =\n    styleOverride.cornerStrokeColor || fabricObject.cornerStrokeColor || '';\n  // TODO: use proper ellipse code.\n  if (xSize > ySize) {\n    size = xSize;\n    ctx.scale(1.0, ySize / xSize);\n    myTop = (top * xSize) / ySize;\n  } else if (ySize > xSize) {\n    size = ySize;\n    ctx.scale(xSize / ySize, 1.0);\n    myLeft = (left * ySize) / xSize;\n  } else {\n    size = xSize;\n  }\n  // this is still wrong\n  ctx.lineWidth = 1;\n  ctx.beginPath();\n  ctx.arc(myLeft, myTop, size / 2, 0, twoMathPi, false);\n  ctx[methodName]();\n  if (stroke) {\n    ctx.stroke();\n  }\n  ctx.restore();\n}\n\n/**\n * Render a square control, as per fabric features.\n * This function is written to respect object properties like transparentCorners, cornerSize\n * cornerColor, cornerStrokeColor\n * plus the addition of offsetY and offsetX.\n * @param {CanvasRenderingContext2D} ctx context to render on\n * @param {Number} left x coordinate where the control center should be\n * @param {Number} top y coordinate where the control center should be\n * @param {Object} styleOverride override for FabricObject controls style\n * @param {FabricObject} fabricObject the fabric object for which we are rendering controls\n */\nexport function renderSquareControl(\n  this: Control,\n  ctx: CanvasRenderingContext2D,\n  left: number,\n  top: number,\n  styleOverride: ControlRenderingStyleOverride,\n  fabricObject: InteractiveFabricObject\n) {\n  styleOverride = styleOverride || {};\n  const xSize =\n      this.sizeX || styleOverride.cornerSize || fabricObject.cornerSize,\n    ySize = this.sizeY || styleOverride.cornerSize || fabricObject.cornerSize,\n    transparentCorners =\n      typeof styleOverride.transparentCorners !== 'undefined'\n        ? styleOverride.transparentCorners\n        : fabricObject.transparentCorners,\n    methodName = transparentCorners ? 'stroke' : 'fill',\n    stroke =\n      !transparentCorners &&\n      (styleOverride.cornerStrokeColor || fabricObject.cornerStrokeColor),\n    xSizeBy2 = xSize / 2,\n    ySizeBy2 = ySize / 2;\n  ctx.save();\n  ctx.fillStyle = styleOverride.cornerColor || fabricObject.cornerColor || '';\n  ctx.strokeStyle =\n    styleOverride.cornerStrokeColor || fabricObject.cornerStrokeColor || '';\n  // this is still wrong\n  ctx.lineWidth = 1;\n  ctx.translate(left, top);\n  //  angle is relative to canvas plane\n  const angle = fabricObject.getTotalAngle();\n  ctx.rotate(degreesToRadians(angle));\n  // this does not work, and fixed with ( && ) does not make sense.\n  // to have real transparent corners we need the controls on upperCanvas\n  // transparentCorners || ctx.clearRect(-xSizeBy2, -ySizeBy2, xSize, ySize);\n  ctx[`${methodName}Rect`](-xSizeBy2, -ySizeBy2, xSize, ySize);\n  if (stroke) {\n    ctx.strokeRect(-xSizeBy2, -ySizeBy2, xSize, ySize);\n  }\n  ctx.restore();\n}\n","/* eslint-disable @typescript-eslint/no-unused-vars */\nimport type {\n  ControlActionHandler,\n  TPointerEvent,\n  TransformActionHandler,\n} from '../EventTypeDefs';\nimport { Intersection } from '../Intersection';\nimport { Point } from '../Point';\nimport type { InteractiveFabricObject } from '../shapes/Object/InteractiveObject';\nimport type { TCornerPoint, TDegree, TMat2D } from '../typedefs';\nimport {\n  createRotateMatrix,\n  createScaleMatrix,\n  createTranslateMatrix,\n  multiplyTransformMatrixArray,\n} from '../util/misc/matrix';\nimport type { ControlRenderingStyleOverride } from './controlRendering';\nimport { renderCircleControl, renderSquareControl } from './controlRendering';\n\nexport class Control {\n  /**\n   * keep track of control visibility.\n   * mainly for backward compatibility.\n   * if you do not want to see a control, you can remove it\n   * from the control set.\n   * @type {Boolean}\n   * @default true\n   */\n  visible = true;\n\n  /**\n   * Name of the action that the control will likely execute.\n   * This is optional. FabricJS uses to identify what the user is doing for some\n   * extra optimizations. If you are writing a custom control and you want to know\n   * somewhere else in the code what is going on, you can use this string here.\n   * you can also provide a custom getActionName if your control run multiple actions\n   * depending on some external state.\n   * default to scale since is the most common, used on 4 corners by default\n   * @type {String}\n   * @default 'scale'\n   */\n  actionName = 'scale';\n\n  /**\n   * Drawing angle of the control.\n   * NOT used for now, but name marked as needed for internal logic\n   * example: to reuse the same drawing function for different rotated controls\n   * @type {Number}\n   * @default 0\n   */\n  angle = 0;\n\n  /**\n   * Relative position of the control. X\n   * 0,0 is the center of the Object, while -0.5 (left) or 0.5 (right) are the extremities\n   * of the bounding box.\n   * @type {Number}\n   * @default 0\n   */\n  x = 0;\n\n  /**\n   * Relative position of the control. Y\n   * 0,0 is the center of the Object, while -0.5 (top) or 0.5 (bottom) are the extremities\n   * of the bounding box.\n   * @type {Number}\n   * @default 0\n   */\n  y = 0;\n\n  /**\n   * Horizontal offset of the control from the defined position. In pixels\n   * Positive offset moves the control to the right, negative to the left.\n   * It used when you want to have position of control that does not scale with\n   * the bounding box. Example: rotation control is placed at x:0, y: 0.5 on\n   * the boundind box, with an offset of 30 pixels vertically. Those 30 pixels will\n   * stay 30 pixels no matter how the object is big. Another example is having 2\n   * controls in the corner, that stay in the same position when the object scale.\n   * of the bounding box.\n   * @type {Number}\n   * @default 0\n   */\n  offsetX = 0;\n\n  /**\n   * Vertical offset of the control from the defined position. In pixels\n   * Positive offset moves the control to the bottom, negative to the top.\n   * @type {Number}\n   * @default 0\n   */\n  offsetY = 0;\n\n  /**\n   * Sets the length of the control. If null, defaults to object's cornerSize.\n   * Expects both sizeX and sizeY to be set when set.\n   * @type {?Number}\n   * @default null\n   */\n  sizeX = 0;\n\n  /**\n   * Sets the height of the control. If null, defaults to object's cornerSize.\n   * Expects both sizeX and sizeY to be set when set.\n   * @type {?Number}\n   * @default null\n   */\n  sizeY = 0;\n\n  /**\n   * Sets the length of the touch area of the control. If null, defaults to object's touchCornerSize.\n   * Expects both touchSizeX and touchSizeY to be set when set.\n   * @type {?Number}\n   * @default null\n   */\n  touchSizeX = 0;\n\n  /**\n   * Sets the height of the touch area of the control. If null, defaults to object's touchCornerSize.\n   * Expects both touchSizeX and touchSizeY to be set when set.\n   * @type {?Number}\n   * @default null\n   */\n  touchSizeY = 0;\n\n  /**\n   * Css cursor style to display when the control is hovered.\n   * if the method `cursorStyleHandler` is provided, this property is ignored.\n   * @type {String}\n   * @default 'crosshair'\n   */\n  cursorStyle = 'crosshair';\n\n  /**\n   * If controls has an offsetY or offsetX, draw a line that connects\n   * the control to the bounding box\n   * @type {Boolean}\n   * @default false\n   */\n  withConnection = false;\n\n  constructor(options?: Partial<Control>) {\n    Object.assign(this, options);\n  }\n\n  /**\n   * The control actionHandler, provide one to handle action ( control being moved )\n   * @param {Event} eventData the native mouse event\n   * @param {Transform} transformData properties of the current transform\n   * @param {Number} x x position of the cursor\n   * @param {Number} y y position of the cursor\n   * @return {Boolean} true if the action/event modified the object\n   */\n  declare actionHandler: TransformActionHandler;\n\n  /**\n   * The control handler for mouse down, provide one to handle mouse down on control\n   * @param {Event} eventData the native mouse event\n   * @param {Transform} transformData properties of the current transform\n   * @param {Number} x x position of the cursor\n   * @param {Number} y y position of the cursor\n   * @return {Boolean} true if the action/event modified the object\n   */\n  declare mouseDownHandler?: ControlActionHandler;\n\n  /**\n   * The control mouseUpHandler, provide one to handle an effect on mouse up.\n   * @param {Event} eventData the native mouse event\n   * @param {Transform} transformData properties of the current transform\n   * @param {Number} x x position of the cursor\n   * @param {Number} y y position of the cursor\n   * @return {Boolean} true if the action/event modified the object\n   */\n  declare mouseUpHandler?: ControlActionHandler;\n\n  shouldActivate(\n    controlKey: string,\n    fabricObject: InteractiveFabricObject,\n    pointer: Point,\n    { tl, tr, br, bl }: TCornerPoint\n  ) {\n    // TODO: locking logic can be handled here instead of in the control handler logic\n    return (\n      fabricObject.canvas?.getActiveObject() === fabricObject &&\n      fabricObject.isControlVisible(controlKey) &&\n      Intersection.isPointInPolygon(pointer, [tl, tr, br, bl])\n    );\n  }\n\n  /**\n   * Returns control actionHandler\n   * @param {Event} eventData the native mouse event\n   * @param {FabricObject} fabricObject on which the control is displayed\n   * @param {Control} control control for which the action handler is being asked\n   * @return {Function} the action handler\n   */\n  getActionHandler(\n    eventData: TPointerEvent,\n    fabricObject: InteractiveFabricObject,\n    control: Control\n  ): TransformActionHandler | undefined {\n    return this.actionHandler;\n  }\n\n  /**\n   * Returns control mouseDown handler\n   * @param {Event} eventData the native mouse event\n   * @param {FabricObject} fabricObject on which the control is displayed\n   * @param {Control} control control for which the action handler is being asked\n   * @return {Function} the action handler\n   */\n  getMouseDownHandler(\n    eventData: TPointerEvent,\n    fabricObject: InteractiveFabricObject,\n    control: Control\n  ): ControlActionHandler | undefined {\n    return this.mouseDownHandler;\n  }\n\n  /**\n   * Returns control mouseUp handler.\n   * During actions the fabricObject or the control can be of different obj\n   * @param {Event} eventData the native mouse event\n   * @param {FabricObject} fabricObject on which the control is displayed\n   * @param {Control} control control for which the action handler is being asked\n   * @return {Function} the action handler\n   */\n  getMouseUpHandler(\n    eventData: TPointerEvent,\n    fabricObject: InteractiveFabricObject,\n    control: Control\n  ): ControlActionHandler | undefined {\n    return this.mouseUpHandler;\n  }\n\n  /**\n   * Returns control cursorStyle for css using cursorStyle. If you need a more elaborate\n   * function you can pass one in the constructor\n   * the cursorStyle property\n   * @param {Event} eventData the native mouse event\n   * @param {Control} control the current control ( likely this)\n   * @param {FabricObject} object on which the control is displayed\n   * @return {String}\n   */\n  cursorStyleHandler(\n    eventData: TPointerEvent,\n    control: Control,\n    fabricObject: InteractiveFabricObject\n  ) {\n    return control.cursorStyle;\n  }\n\n  /**\n   * Returns the action name. The basic implementation just return the actionName property.\n   * @param {Event} eventData the native mouse event\n   * @param {Control} control the current control ( likely this)\n   * @param {FabricObject} object on which the control is displayed\n   * @return {String}\n   */\n  getActionName(\n    eventData: TPointerEvent,\n    control: Control,\n    fabricObject: InteractiveFabricObject\n  ) {\n    return control.actionName;\n  }\n\n  /**\n   * Returns controls visibility\n   * @param {FabricObject} object on which the control is displayed\n   * @param {String} controlKey key where the control is memorized on the\n   * @return {Boolean}\n   */\n  getVisibility(fabricObject: InteractiveFabricObject, controlKey: string) {\n    return fabricObject._controlsVisibility?.[controlKey] ?? this.visible;\n  }\n\n  /**\n   * Sets controls visibility\n   * @param {Boolean} visibility for the object\n   * @return {Void}\n   */\n  setVisibility(\n    visibility: boolean,\n    name: string,\n    fabricObject: InteractiveFabricObject\n  ) {\n    this.visible = visibility;\n  }\n\n  positionHandler(\n    dim: Point,\n    finalMatrix: TMat2D,\n    fabricObject: InteractiveFabricObject,\n    currentControl: Control\n  ) {\n    return new Point(\n      this.x * dim.x + this.offsetX,\n      this.y * dim.y + this.offsetY\n    ).transform(finalMatrix);\n  }\n\n  /**\n   * Returns the coords for this control based on object values.\n   * @param {Number} objectAngle angle from the fabric object holding the control\n   * @param {Number} objectCornerSize cornerSize from the fabric object holding the control (or touchCornerSize if\n   *   isTouch is true)\n   * @param {Number} centerX x coordinate where the control center should be\n   * @param {Number} centerY y coordinate where the control center should be\n   * @param {boolean} isTouch true if touch corner, false if normal corner\n   */\n  calcCornerCoords(\n    angle: TDegree,\n    objectCornerSize: number,\n    centerX: number,\n    centerY: number,\n    isTouch: boolean,\n    fabricObject: InteractiveFabricObject\n  ) {\n    const t = multiplyTransformMatrixArray([\n      createTranslateMatrix(centerX, centerY),\n      createRotateMatrix({ angle }),\n      createScaleMatrix(\n        (isTouch ? this.touchSizeX : this.sizeX) || objectCornerSize,\n        (isTouch ? this.touchSizeY : this.sizeY) || objectCornerSize\n      ),\n    ]);\n    return {\n      tl: new Point(-0.5, -0.5).transform(t),\n      tr: new Point(0.5, -0.5).transform(t),\n      br: new Point(0.5, 0.5).transform(t),\n      bl: new Point(-0.5, 0.5).transform(t),\n    };\n  }\n\n  /**\n   * Render function for the control.\n   * When this function runs the context is unscaled. unrotate. Just retina scaled.\n   * all the functions will have to translate to the point left,top before starting Drawing\n   * if they want to draw a control where the position is detected.\n   * left and top are the result of the positionHandler function\n   * @param {RenderingContext2D} ctx the context where the control will be drawn\n   * @param {Number} left position of the canvas where we are about to render the control.\n   * @param {Number} top position of the canvas where we are about to render the control.\n   * @param {Object} styleOverride\n   * @param {FabricObject} fabricObject the object where the control is about to be rendered\n   */\n  render(\n    ctx: CanvasRenderingContext2D,\n    left: number,\n    top: number,\n    styleOverride: ControlRenderingStyleOverride | undefined,\n    fabricObject: InteractiveFabricObject\n  ) {\n    styleOverride = styleOverride || {};\n    switch (styleOverride.cornerStyle || fabricObject.cornerStyle) {\n      case 'circle':\n        renderCircleControl.call(\n          this,\n          ctx,\n          left,\n          top,\n          styleOverride,\n          fabricObject\n        );\n        break;\n      default:\n        renderSquareControl.call(\n          this,\n          ctx,\n          left,\n          top,\n          styleOverride,\n          fabricObject\n        );\n    }\n  }\n}\n","import type {\n  ControlCursorCallback,\n  TPointerEvent,\n  Transform,\n  TransformActionHandler,\n} from '../EventTypeDefs';\nimport type { FabricObject } from '../shapes/Object/FabricObject';\nimport type { TAxis } from '../typedefs';\nimport type { Canvas } from '../canvas/Canvas';\nimport {\n  findCornerQuadrant,\n  getLocalPoint,\n  invertOrigin,\n  isLocked,\n  isTransformCentered,\n  NOT_ALLOWED_CURSOR,\n} from './util';\nimport { wrapWithFireEvent } from './wrapWithFireEvent';\nimport { wrapWithFixedAnchor } from './wrapWithFixedAnchor';\n\ntype ScaleTransform = Transform & {\n  gestureScale?: number;\n  signX?: number;\n  signY?: number;\n};\n\ntype ScaleBy = TAxis | 'equally' | '' | undefined;\n\n/**\n * Inspect event and fabricObject properties to understand if the scaling action\n * @param {Event} eventData from the user action\n * @param {FabricObject} fabricObject the fabric object about to scale\n * @return {Boolean} true if scale is proportional\n */\nexport function scaleIsProportional(\n  eventData: TPointerEvent,\n  fabricObject: FabricObject\n): boolean {\n  const canvas = fabricObject.canvas as Canvas,\n    uniformIsToggled = eventData[canvas.uniScaleKey!];\n  return (\n    (canvas.uniformScaling && !uniformIsToggled) ||\n    (!canvas.uniformScaling && uniformIsToggled)\n  );\n}\n\n/**\n * Inspect fabricObject to understand if the current scaling action is allowed\n * @param {FabricObject} fabricObject the fabric object about to scale\n * @param {String} by 'x' or 'y' or ''\n * @param {Boolean} scaleProportionally true if we are trying to scale proportionally\n * @return {Boolean} true if scaling is not allowed at current conditions\n */\nexport function scalingIsForbidden(\n  fabricObject: FabricObject,\n  by: ScaleBy,\n  scaleProportionally: boolean\n) {\n  const lockX = isLocked(fabricObject, 'lockScalingX'),\n    lockY = isLocked(fabricObject, 'lockScalingY');\n  if (lockX && lockY) {\n    return true;\n  }\n  if (!by && (lockX || lockY) && scaleProportionally) {\n    return true;\n  }\n  if (lockX && by === 'x') {\n    return true;\n  }\n  if (lockY && by === 'y') {\n    return true;\n  }\n  // code crashes because of a division by 0 if a 0 sized object is scaled\n  // forbid to prevent scaling to happen. ISSUE-9475\n  const { width, height, strokeWidth } = fabricObject;\n  if (width === 0 && strokeWidth === 0 && by !== 'y') {\n    return true;\n  }\n  if (height === 0 && strokeWidth === 0 && by !== 'x') {\n    return true;\n  }\n  return false;\n}\n\nconst scaleMap = ['e', 'se', 's', 'sw', 'w', 'nw', 'n', 'ne', 'e'];\n\n/**\n * return the correct cursor style for the scale action\n * @param {Event} eventData the javascript event that is causing the scale\n * @param {Control} control the control that is interested in the action\n * @param {FabricObject} fabricObject the fabric object that is interested in the action\n * @return {String} a valid css string for the cursor\n */\nexport const scaleCursorStyleHandler: ControlCursorCallback = (\n  eventData,\n  control,\n  fabricObject\n) => {\n  const scaleProportionally = scaleIsProportional(eventData, fabricObject),\n    by =\n      control.x !== 0 && control.y === 0\n        ? 'x'\n        : control.x === 0 && control.y !== 0\n        ? 'y'\n        : '';\n  if (scalingIsForbidden(fabricObject, by, scaleProportionally)) {\n    return NOT_ALLOWED_CURSOR;\n  }\n  const n = findCornerQuadrant(fabricObject, control);\n  return `${scaleMap[n]}-resize`;\n};\n\n/**\n * Basic scaling logic, reused with different constrain for scaling X,Y, freely or equally.\n * Needs to be wrapped with `wrapWithFixedAnchor` to be effective\n * @param {Event} eventData javascript event that is doing the transform\n * @param {Object} transform javascript object containing a series of information around the current transform\n * @param {number} x current mouse x position, canvas normalized\n * @param {number} y current mouse y position, canvas normalized\n * @param {Object} options additional information for scaling\n * @param {String} options.by 'x', 'y', 'equally' or '' to indicate type of scaling\n * @return {Boolean} true if some change happened\n * @private\n */\nfunction scaleObject(\n  eventData: TPointerEvent,\n  transform: ScaleTransform,\n  x: number,\n  y: number,\n  options: { by?: ScaleBy } = {}\n) {\n  const target = transform.target,\n    by = options.by,\n    scaleProportionally = scaleIsProportional(eventData, target),\n    forbidScaling = scalingIsForbidden(target, by, scaleProportionally);\n  let newPoint, scaleX, scaleY, dim, signX, signY;\n\n  if (forbidScaling) {\n    return false;\n  }\n  if (transform.gestureScale) {\n    scaleX = transform.scaleX * transform.gestureScale;\n    scaleY = transform.scaleY * transform.gestureScale;\n  } else {\n    newPoint = getLocalPoint(\n      transform,\n      transform.originX,\n      transform.originY,\n      x,\n      y\n    );\n    // use of sign: We use sign to detect change of direction of an action. sign usually change when\n    // we cross the origin point with the mouse. So a scale flip for example. There is an issue when scaling\n    // by center and scaling using one middle control ( default: mr, mt, ml, mb), the mouse movement can easily\n    // cross many time the origin point and flip the object. so we need a way to filter out the noise.\n    // This ternary here should be ok to filter out X scaling when we want Y only and vice versa.\n    signX = by !== 'y' ? Math.sign(newPoint.x || transform.signX || 1) : 1;\n    signY = by !== 'x' ? Math.sign(newPoint.y || transform.signY || 1) : 1;\n    if (!transform.signX) {\n      transform.signX = signX;\n    }\n    if (!transform.signY) {\n      transform.signY = signY;\n    }\n\n    if (\n      isLocked(target, 'lockScalingFlip') &&\n      (transform.signX !== signX || transform.signY !== signY)\n    ) {\n      return false;\n    }\n\n    dim = target._getTransformedDimensions();\n    // missing detection of flip and logic to switch the origin\n    if (scaleProportionally && !by) {\n      // uniform scaling\n      const distance = Math.abs(newPoint.x) + Math.abs(newPoint.y),\n        { original } = transform,\n        originalDistance =\n          Math.abs((dim.x * original.scaleX) / target.scaleX) +\n          Math.abs((dim.y * original.scaleY) / target.scaleY),\n        scale = distance / originalDistance;\n      scaleX = original.scaleX * scale;\n      scaleY = original.scaleY * scale;\n    } else {\n      scaleX = Math.abs((newPoint.x * target.scaleX) / dim.x);\n      scaleY = Math.abs((newPoint.y * target.scaleY) / dim.y);\n    }\n    // if we are scaling by center, we need to double the scale\n    if (isTransformCentered(transform)) {\n      scaleX *= 2;\n      scaleY *= 2;\n    }\n    if (transform.signX !== signX && by !== 'y') {\n      transform.originX = invertOrigin(transform.originX);\n      scaleX *= -1;\n      transform.signX = signX;\n    }\n    if (transform.signY !== signY && by !== 'x') {\n      transform.originY = invertOrigin(transform.originY);\n      scaleY *= -1;\n      transform.signY = signY;\n    }\n  }\n  // minScale is taken care of in the setter.\n  const oldScaleX = target.scaleX,\n    oldScaleY = target.scaleY;\n  if (!by) {\n    !isLocked(target, 'lockScalingX') && target.set('scaleX', scaleX);\n    !isLocked(target, 'lockScalingY') && target.set('scaleY', scaleY);\n  } else {\n    // forbidden cases already handled on top here.\n    by === 'x' && target.set('scaleX', scaleX);\n    by === 'y' && target.set('scaleY', scaleY);\n  }\n  return oldScaleX !== target.scaleX || oldScaleY !== target.scaleY;\n}\n\n/**\n * Generic scaling logic, to scale from corners either equally or freely.\n * Needs to be wrapped with `wrapWithFixedAnchor` to be effective\n * @param {Event} eventData javascript event that is doing the transform\n * @param {Object} transform javascript object containing a series of information around the current transform\n * @param {number} x current mouse x position, canvas normalized\n * @param {number} y current mouse y position, canvas normalized\n * @return {Boolean} true if some change happened\n */\nexport const scaleObjectFromCorner: TransformActionHandler<ScaleTransform> = (\n  eventData,\n  transform,\n  x,\n  y\n) => {\n  return scaleObject(eventData, transform, x, y);\n};\n\n/**\n * Scaling logic for the X axis.\n * Needs to be wrapped with `wrapWithFixedAnchor` to be effective\n * @param {Event} eventData javascript event that is doing the transform\n * @param {Object} transform javascript object containing a series of information around the current transform\n * @param {number} x current mouse x position, canvas normalized\n * @param {number} y current mouse y position, canvas normalized\n * @return {Boolean} true if some change happened\n */\nconst scaleObjectX: TransformActionHandler<ScaleTransform> = (\n  eventData,\n  transform,\n  x,\n  y\n) => {\n  return scaleObject(eventData, transform, x, y, { by: 'x' });\n};\n\n/**\n * Scaling logic for the Y axis.\n * Needs to be wrapped with `wrapWithFixedAnchor` to be effective\n * @param {Event} eventData javascript event that is doing the transform\n * @param {Object} transform javascript object containing a series of information around the current transform\n * @param {number} x current mouse x position, canvas normalized\n * @param {number} y current mouse y position, canvas normalized\n * @return {Boolean} true if some change happened\n */\nconst scaleObjectY: TransformActionHandler<ScaleTransform> = (\n  eventData,\n  transform,\n  x,\n  y\n) => {\n  return scaleObject(eventData, transform, x, y, { by: 'y' });\n};\n\nexport const scalingEqually = wrapWithFireEvent(\n  'scaling',\n  wrapWithFixedAnchor(scaleObjectFromCorner)\n);\n\nexport const scalingX = wrapWithFireEvent(\n  'scaling',\n  wrapWithFixedAnchor(scaleObjectX)\n);\n\nexport const scalingY = wrapWithFireEvent(\n  'scaling',\n  wrapWithFixedAnchor(scaleObjectY)\n);\n","import type {\n  ControlCursorCallback,\n  TPointerEvent,\n  Transform,\n  TransformActionHandler,\n} from '../EventTypeDefs';\nimport { resolveOrigin } from '../util/misc/resolveOrigin';\nimport { Point } from '../Point';\nimport type { TAxis, TAxisKey } from '../typedefs';\nimport {\n  degreesToRadians,\n  radiansToDegrees,\n} from '../util/misc/radiansDegreesConversion';\nimport {\n  findCornerQuadrant,\n  getLocalPoint,\n  isLocked,\n  NOT_ALLOWED_CURSOR,\n} from './util';\nimport { wrapWithFireEvent } from './wrapWithFireEvent';\nimport { wrapWithFixedAnchor } from './wrapWithFixedAnchor';\nimport { CENTER } from '../constants';\n\nexport type SkewTransform = Transform & { skewingSide: -1 | 1 };\n\nconst AXIS_KEYS: Record<\n  TAxis,\n  {\n    counterAxis: TAxis;\n    scale: TAxisKey<'scale'>;\n    skew: TAxisKey<'skew'>;\n    lockSkewing: TAxisKey<'lockSkewing'>;\n    origin: TAxisKey<'origin'>;\n    flip: TAxisKey<'flip'>;\n  }\n> = {\n  x: {\n    counterAxis: 'y',\n    scale: 'scaleX',\n    skew: 'skewX',\n    lockSkewing: 'lockSkewingX',\n    origin: 'originX',\n    flip: 'flipX',\n  },\n  y: {\n    counterAxis: 'x',\n    scale: 'scaleY',\n    skew: 'skewY',\n    lockSkewing: 'lockSkewingY',\n    origin: 'originY',\n    flip: 'flipY',\n  },\n};\n\nconst skewMap = ['ns', 'nesw', 'ew', 'nwse'];\n\n/**\n * return the correct cursor style for the skew action\n * @param {Event} eventData the javascript event that is causing the scale\n * @param {Control} control the control that is interested in the action\n * @param {FabricObject} fabricObject the fabric object that is interested in the action\n * @return {String} a valid css string for the cursor\n */\nexport const skewCursorStyleHandler: ControlCursorCallback = (\n  eventData,\n  control,\n  fabricObject\n) => {\n  if (control.x !== 0 && isLocked(fabricObject, 'lockSkewingY')) {\n    return NOT_ALLOWED_CURSOR;\n  }\n  if (control.y !== 0 && isLocked(fabricObject, 'lockSkewingX')) {\n    return NOT_ALLOWED_CURSOR;\n  }\n  const n = findCornerQuadrant(fabricObject, control) % 4;\n  return `${skewMap[n]}-resize`;\n};\n\n/**\n * Since skewing is applied before scaling, calculations are done in a scaleless plane\n * @see https://github.com/fabricjs/fabric.js/pull/8380\n */\nfunction skewObject(\n  axis: TAxis,\n  { target, ex, ey, skewingSide, ...transform }: SkewTransform,\n  pointer: Point\n) {\n  const { skew: skewKey } = AXIS_KEYS[axis],\n    offset = pointer\n      .subtract(new Point(ex, ey))\n      .divide(new Point(target.scaleX, target.scaleY))[axis],\n    skewingBefore = target[skewKey],\n    skewingStart = transform[skewKey],\n    shearingStart = Math.tan(degreesToRadians(skewingStart)),\n    // let a, b be the size of target\n    // let a' be the value of a after applying skewing\n    // then:\n    // a' = a + b * skewA => skewA = (a' - a) / b\n    // the value b is tricky since skewY is applied before skewX\n    b =\n      axis === 'y'\n        ? target._getTransformedDimensions({\n            scaleX: 1,\n            scaleY: 1,\n            // since skewY is applied before skewX, b (=width) is not affected by skewX\n            skewX: 0,\n          }).x\n        : target._getTransformedDimensions({\n            scaleX: 1,\n            scaleY: 1,\n          }).y;\n\n  const shearing =\n    (2 * offset * skewingSide) /\n      // we max out fractions to safeguard from asymptotic behavior\n      Math.max(b, 1) +\n    // add starting state\n    shearingStart;\n\n  const skewing = radiansToDegrees(Math.atan(shearing));\n\n  target.set(skewKey, skewing);\n  const changed = skewingBefore !== target[skewKey];\n\n  if (changed && axis === 'y') {\n    // we don't want skewing to affect scaleX\n    // so we factor it by the inverse skewing diff to make it seem unchanged to the viewer\n    const { skewX, scaleX } = target,\n      dimBefore = target._getTransformedDimensions({ skewY: skewingBefore }),\n      dimAfter = target._getTransformedDimensions(),\n      compensationFactor = skewX !== 0 ? dimBefore.x / dimAfter.x : 1;\n    compensationFactor !== 1 &&\n      target.set('scaleX', compensationFactor * scaleX);\n  }\n\n  return changed;\n}\n\n/**\n * Wrapped Action handler for skewing on a given axis, takes care of the\n * skew direction and determines the correct transform origin for the anchor point\n * @param {Event} eventData javascript event that is doing the transform\n * @param {Object} transform javascript object containing a series of information around the current transform\n * @param {number} x current mouse x position, canvas normalized\n * @param {number} y current mouse y position, canvas normalized\n * @return {Boolean} true if some change happened\n */\nfunction skewHandler(\n  axis: TAxis,\n  eventData: TPointerEvent,\n  transform: Transform,\n  x: number,\n  y: number\n) {\n  const { target } = transform,\n    {\n      counterAxis,\n      origin: originKey,\n      lockSkewing: lockSkewingKey,\n      skew: skewKey,\n      flip: flipKey,\n    } = AXIS_KEYS[axis];\n  if (isLocked(target, lockSkewingKey)) {\n    return false;\n  }\n\n  const { origin: counterOriginKey, flip: counterFlipKey } =\n      AXIS_KEYS[counterAxis],\n    counterOriginFactor =\n      resolveOrigin(transform[counterOriginKey]) *\n      (target[counterFlipKey] ? -1 : 1),\n    // if the counter origin is top/left (= -0.5) then we are skewing x/y values on the bottom/right side of target respectively.\n    // if the counter origin is bottom/right (= 0.5) then we are skewing x/y values on the top/left side of target respectively.\n    // skewing direction on the top/left side of target is OPPOSITE to the direction of the movement of the pointer,\n    // so we factor skewing direction by this value.\n    skewingSide = (-Math.sign(counterOriginFactor) *\n      (target[flipKey] ? -1 : 1)) as 1 | -1,\n    skewingDirection =\n      ((target[skewKey] === 0 &&\n        // in case skewing equals 0 we use the pointer offset from target center to determine the direction of skewing\n        getLocalPoint(transform, CENTER, CENTER, x, y)[axis] > 0) ||\n      // in case target has skewing we use that as the direction\n      target[skewKey] > 0\n        ? 1\n        : -1) * skewingSide,\n    // anchor to the opposite side of the skewing direction\n    // normalize value from [-1, 1] to origin value [0, 1]\n    origin = -skewingDirection * 0.5 + 0.5;\n\n  const finalHandler = wrapWithFireEvent<SkewTransform>(\n    'skewing',\n    wrapWithFixedAnchor((eventData, transform, x, y) =>\n      skewObject(axis, transform, new Point(x, y))\n    )\n  );\n\n  return finalHandler(\n    eventData,\n    {\n      ...transform,\n      [originKey]: origin,\n      skewingSide,\n    },\n    x,\n    y\n  );\n}\n\n/**\n * Wrapped Action handler for skewing on the X axis, takes care of the\n * skew direction and determines the correct transform origin for the anchor point\n * @param {Event} eventData javascript event that is doing the transform\n * @param {Object} transform javascript object containing a series of information around the current transform\n * @param {number} x current mouse x position, canvas normalized\n * @param {number} y current mouse y position, canvas normalized\n * @return {Boolean} true if some change happened\n */\nexport const skewHandlerX: TransformActionHandler = (\n  eventData,\n  transform,\n  x,\n  y\n) => {\n  return skewHandler('x', eventData, transform, x, y);\n};\n\n/**\n * Wrapped Action handler for skewing on the Y axis, takes care of the\n * skew direction and determines the correct transform origin for the anchor point\n * @param {Event} eventData javascript event that is doing the transform\n * @param {Object} transform javascript object containing a series of information around the current transform\n * @param {number} x current mouse x position, canvas normalized\n * @param {number} y current mouse y position, canvas normalized\n * @return {Boolean} true if some change happened\n */\nexport const skewHandlerY: TransformActionHandler = (\n  eventData,\n  transform,\n  x,\n  y\n) => {\n  return skewHandler('y', eventData, transform, x, y);\n};\n","import type {\n  ControlCallback,\n  ControlCursorCallback,\n  TPointerEvent,\n  TransformActionHandler,\n} from '../EventTypeDefs';\nimport type { FabricObject } from '../shapes/Object/FabricObject';\nimport type { TAxisKey } from '../typedefs';\nimport { scaleCursorStyleHandler, scalingX, scalingY } from './scale';\nimport { skewCursorStyleHandler, skewHandlerX, skewHandlerY } from './skew';\n\nfunction isAltAction(eventData: TPointerEvent, target: FabricObject) {\n  return eventData[target.canvas!.altActionKey!];\n}\n\n/**\n * Inspect event, control and fabricObject to return the correct action name\n * @param {Event} eventData the javascript event that is causing the scale\n * @param {Control} control the control that is interested in the action\n * @param {FabricObject} fabricObject the fabric object that is interested in the action\n * @return {String} an action name\n */\nexport const scaleOrSkewActionName: ControlCallback<\n  TAxisKey<'skew' | 'scale'> | ''\n> = (eventData, control, fabricObject) => {\n  const isAlternative = isAltAction(eventData, fabricObject);\n  if (control.x === 0) {\n    // then is scaleY or skewX\n    return isAlternative ? 'skewX' : 'scaleY';\n  }\n  if (control.y === 0) {\n    // then is scaleY or skewX\n    return isAlternative ? 'skewY' : 'scaleX';\n  }\n  return '';\n};\n\n/**\n * Combine skew and scale style handlers to cover fabric standard use case\n * @param {Event} eventData the javascript event that is causing the scale\n * @param {Control} control the control that is interested in the action\n * @param {FabricObject} fabricObject the fabric object that is interested in the action\n * @return {String} a valid css string for the cursor\n */\nexport const scaleSkewCursorStyleHandler: ControlCursorCallback = (\n  eventData,\n  control,\n  fabricObject\n) => {\n  return isAltAction(eventData, fabricObject)\n    ? skewCursorStyleHandler(eventData, control, fabricObject)\n    : scaleCursorStyleHandler(eventData, control, fabricObject);\n};\n/**\n * Composed action handler to either scale X or skew Y\n * Needs to be wrapped with `wrapWithFixedAnchor` to be effective\n * @param {Event} eventData javascript event that is doing the transform\n * @param {Object} transform javascript object containing a series of information around the current transform\n * @param {number} x current mouse x position, canvas normalized\n * @param {number} y current mouse y position, canvas normalized\n * @return {Boolean} true if some change happened\n */\nexport const scalingXOrSkewingY: TransformActionHandler = (\n  eventData,\n  transform,\n  x,\n  y\n) => {\n  return isAltAction(eventData, transform.target)\n    ? skewHandlerY(eventData, transform, x, y)\n    : scalingX(eventData, transform, x, y);\n};\n\n/**\n * Composed action handler to either scale Y or skew X\n * Needs to be wrapped with `wrapWithFixedAnchor` to be effective\n * @param {Event} eventData javascript event that is doing the transform\n * @param {Object} transform javascript object containing a series of information around the current transform\n * @param {number} x current mouse x position, canvas normalized\n * @param {number} y current mouse y position, canvas normalized\n * @return {Boolean} true if some change happened\n */\nexport const scalingYOrSkewingX: TransformActionHandler = (\n  eventData,\n  transform,\n  x,\n  y\n) => {\n  return isAltAction(eventData, transform.target)\n    ? skewHandlerX(eventData, transform, x, y)\n    : scalingY(eventData, transform, x, y);\n};\n","//@ts-nocheck\nimport { changeWidth } from './changeWidth';\nimport { Control } from './Control';\nimport { scaleCursorStyleHandler, scalingEqually } from './scale';\nimport {\n  scaleOrSkewActionName,\n  scaleSkewCursorStyleHandler,\n  scalingYOrSkewingX,\n} from './scaleSkew';\nimport { renderCircleControl } from './controlRendering';\n\n// use this function if you want to generate new controls for every instance\nexport const createObjectDefaultControls = () => ({\n  tl: new Control({\n    x: -0.5,\n    y: -0.5,\n    cursorStyleHandler: scaleCursorStyleHandler,\n    actionHandler: scalingEqually,\n  }),\n\n  tr: new Control({\n    x: 0.5,\n    y: -0.5,\n    cursorStyleHandler: scaleCursorStyleHandler,\n    actionHandler: scalingEqually,\n  }),\n\n  bl: new Control({\n    x: -0.5,\n    y: 0.5,\n    cursorStyleHandler: scaleCursorStyleHandler,\n    actionHandler: scalingEqually,\n  }),\n\n  br: new Control({\n    x: 0.5,\n    y: 0.5,\n    cursorStyleHandler: scaleCursorStyleHandler,\n    actionHandler: scalingEqually,\n  }),\n\n  // mtr: new Control({\n  //   x: 0,\n  //   y: -0.5,\n  //   actionHandler: rotationWithSnapping,\n  //   cursorStyleHandler: rotationStyleHandler,\n  //   offsetY: -40,\n  //   withConnection: true,\n  //   actionName: 'rotate',\n  // }),\n});\n\nexport const createObjectDefaultNoRotateControls = () => ({\n  tl: new Control({\n    x: -0.5,\n    y: -0.5,\n    cursorStyleHandler: scaleCursorStyleHandler,\n    actionHandler: scalingEqually,\n  }),\n\n  tr: new Control({\n    x: 0.5,\n    y: -0.5,\n    cursorStyleHandler: scaleCursorStyleHandler,\n    actionHandler: scalingEqually,\n  }),\n\n  bl: new Control({\n    x: -0.5,\n    y: 0.5,\n    cursorStyleHandler: scaleCursorStyleHandler,\n    actionHandler: scalingEqually,\n  }),\n\n  br: new Control({\n    x: 0.5,\n    y: 0.5,\n    cursorStyleHandler: scaleCursorStyleHandler,\n    actionHandler: scalingEqually,\n  }),\n});\n\nexport const createObjectImageControls = () => ({\n  tl: new Control({\n    x: -0.5,\n    y: -0.5,\n    cursorStyleHandler: scaleCursorStyleHandler,\n    actionHandler: scalingEqually,\n  }),\n\n  tr: new Control({\n    x: 0.5,\n    y: -0.5,\n    cursorStyleHandler: scaleCursorStyleHandler,\n    actionHandler: scalingEqually,\n  }),\n\n  bl: new Control({\n    x: -0.5,\n    y: 0.5,\n    cursorStyleHandler: scaleCursorStyleHandler,\n    actionHandler: scalingEqually,\n  }),\n\n  br: new Control({\n    x: 0.5,\n    y: 0.5,\n    cursorStyleHandler: scaleCursorStyleHandler,\n    actionHandler: scalingEqually,\n  }),\n});\nexport const createObjectFileControls = () => ({\n  tl: new Control({\n    x: -0.5,\n    y: -0.5,\n    cursorStyleHandler: scaleCursorStyleHandler,\n    actionHandler: scalingEqually,\n  }),\n  br: new Control({\n    x: 0.5,\n    y: 0.5,\n    cursorStyleHandler: scaleCursorStyleHandler,\n    actionHandler: scalingEqually,\n  }),\n});\nexport const createObjectArrowControls = () => ({\n  tl: new Control({\n    x: -0.5,\n    y: -0.5,\n    cursorStyle: 'crosshair',\n    cursorStyleHandler: scaleCursorStyleHandler,\n    actionHandler: scalingEqually,\n  }),\n  br: new Control({\n    x: 0.5,\n    y: 0.5,\n    cursorStyleHandler: scaleCursorStyleHandler,\n    actionHandler: scalingEqually,\n  }),\n});\n\nexport const createObjectConnectorControls = () => {\n  function renderCustomControl(ctx, left, top, fabricObject) {\n    const styleOverride1 = {\n      cornerSize: 10,\n      lineWidth: 2,\n    };\n    renderCircleControl.call(\n      fabricObject,\n      ctx,\n      left,\n      top,\n      styleOverride1,\n      fabricObject\n    );\n  }\n  return {\n    mtaStart: new Control({\n      x: 0,\n      y: -0.5,\n      offsetX: 0,\n      offsetY: -20,\n      render: renderCustomControl,\n      mouseDownHandler: (eventData, transformData) => {\n        // this.controlMousedownProcess(transformData, 0.0, -0.5);\n        return true;\n      },\n      name: 'mtaStart',\n    }),\n    mbaStart: new Control({\n      x: 0,\n      y: 0.5,\n      offsetX: 0,\n      offsetY: 20,\n      render: renderCustomControl,\n      mouseDownHandler: (eventData, transformData) => {\n        // this.controlMousedownProcess(transformData, 0.0, 0.5);\n        return true;\n      },\n      name: 'mbaStart',\n    }),\n    mlaStart: new Control({\n      x: -0.5,\n      y: 0,\n      offsetX: -20,\n      offsetY: 0,\n      render: renderCustomControl,\n      mouseDownHandler: (eventData, transformData) => {\n        // this.controlMousedownProcess(transformData, -0.5, 0.0);\n        return true;\n      },\n      name: 'mlaStart',\n    }),\n    mraStart: new Control({\n      x: 0.5,\n      y: 0,\n      offsetX: 20,\n      offsetY: 0,\n      render: renderCustomControl,\n      mouseDownHandler: (eventData, transformData) => {\n        // this.controlMousedownProcess(transformData, 0.5, 0.0);\n        return true;\n      },\n      name: 'mraStart',\n    }),\n  };\n};\n\nexport const createResizeControls = () => ({\n  mr: new Control({\n    x: 0.5,\n    y: 0,\n    offsetX: 20,\n    offsetY: 0,\n    actionHandler: changeWidth,\n    cursorStyleHandler: scaleSkewCursorStyleHandler,\n    actionName: 'resizing',\n  }),\n  ml: new Control({\n    x: -0.5,\n    y: 0,\n    offsetX: -20,\n    offsetY: 0,\n    actionHandler: changeWidth,\n    cursorStyleHandler: scaleSkewCursorStyleHandler,\n    actionName: 'resizing',\n  }),\n  mb: new Control({\n    x: 0,\n    y: 0.5,\n    offsetX: 0,\n    offsetY: 20,\n    cursorStyleHandler: scaleSkewCursorStyleHandler,\n    actionHandler: scalingYOrSkewingX,\n    getActionName: scaleOrSkewActionName,\n  }),\n\n  mt: new Control({\n    x: 0,\n    y: -0.5,\n    offsetX: 0,\n    offsetY: -20,\n    cursorStyleHandler: scaleSkewCursorStyleHandler,\n    actionHandler: scalingYOrSkewingX,\n    getActionName: scaleOrSkewActionName,\n  }),\n});\n\nexport const createResizeControlsForText = () => ({\n  mr: new Control({\n    x: 0.5,\n    y: 0,\n    actionHandler: changeWidth,\n    cursorStyleHandler: scaleSkewCursorStyleHandler,\n    actionName: 'resizing',\n  }),\n  ml: new Control({\n    x: -0.5,\n    y: 0,\n    actionHandler: changeWidth,\n    cursorStyleHandler: scaleSkewCursorStyleHandler,\n    actionName: 'resizing',\n  }),\n});\n\nexport const createTextboxDefaultControls = () => ({\n  //...createObjectDefaultControls(),\n  ...createResizeControlsForText(),\n});\nexport const createRectNotesDefaultControls = () => ({\n  ...createObjectDefaultNoRotateControls(),\n  ...createObjectConnectorControls(),\n});\nexport const createShapeNotesDefaultControls = () => ({\n  ...createObjectDefaultNoRotateControls(),\n  ...createResizeControls(),\n  ...createObjectConnectorControls,\n});\nexport const createPathDefaultControls = () => ({\n  ...createObjectDefaultNoRotateControls(),\n});\nexport const createImageDefaultControls = () => ({\n  ...createObjectImageControls(),\n});\nexport const createFileDefaultControls = () => ({\n  ...createObjectFileControls(),\n});\n","import { Point, ZERO } from '../../Point';\nimport type { TCornerPoint, TDegree } from '../../typedefs';\nimport { FabricObject } from './Object';\nimport { degreesToRadians } from '../../util/misc/radiansDegreesConversion';\nimport type { TQrDecomposeOut } from '../../util/misc/matrix';\nimport {\n  calcDimensionsMatrix,\n  createRotateMatrix,\n  createTranslateMatrix,\n  multiplyTransformMatrices,\n  qrDecompose,\n} from '../../util/misc/matrix';\nimport type { Control } from '../../controls/Control';\nimport { sizeAfterTransform } from '../../util/misc/objectTransforms';\nimport type { ObjectEvents, TPointerEvent } from '../../EventTypeDefs';\nimport type { Canvas } from '../../canvas/Canvas';\nimport type { ControlRenderingStyleOverride } from '../../controls/controlRendering';\nimport type { FabricObjectProps } from './types/FabricObjectProps';\nimport type { TFabricObjectProps, SerializedObjectProps } from './types';\nimport { createObjectDefaultControls } from '../../controls/commonControls';\nimport { interactiveObjectDefaultValues } from './defaultValues';\n\nexport type TOCoord = Point & {\n  corner: TCornerPoint;\n  touchCorner: TCornerPoint;\n};\n\nexport type TControlSet = Record<string, Control>;\n\nexport type TBorderRenderingStyleOverride = Partial<\n  Pick<InteractiveFabricObject, 'borderColor' | 'borderDashArray'>\n>;\n\nexport type TStyleOverride = ControlRenderingStyleOverride &\n  TBorderRenderingStyleOverride &\n  Partial<\n    Pick<InteractiveFabricObject, 'hasBorders' | 'hasControls'> & {\n      forActiveSelection: boolean;\n    }\n  >;\n\nexport class InteractiveFabricObject<\n  Props extends TFabricObjectProps = Partial<FabricObjectProps>,\n  SProps extends SerializedObjectProps = SerializedObjectProps,\n  EventSpec extends ObjectEvents = ObjectEvents\n>\n  extends FabricObject<Props, SProps, EventSpec>\n  implements FabricObjectProps {\n\n\n\n  declare noScaleCache: boolean;\n\n  declare snapAngle?: TDegree;\n  declare snapThreshold?: TDegree;\n\n  declare lockMovementX: boolean;\n  declare lockMovementY: boolean;\n  declare lockRotation: boolean;\n  declare lockScalingX: boolean;\n  declare lockScalingY: boolean;\n  declare lockSkewingX: boolean;\n  declare lockSkewingY: boolean;\n  declare lockScalingFlip: boolean;\n\n  declare cornerSize: number;\n  declare touchCornerSize: number;\n  declare transparentCorners: boolean;\n  declare cornerColor: string;\n  declare cornerStrokeColor: string;\n  declare cornerStyle: 'rect' | 'circle';\n  declare cornerDashArray: number[] | null;\n  declare hasControls: boolean;\n\n  declare borderColor: string;\n  declare borderDashArray: number[] | null;\n  declare borderOpacityWhenMoving: number;\n  declare borderScaleFactor: number;\n  declare hasBorders: boolean;\n  declare selectionBackgroundColor: string;\n\n  declare selectable: boolean;\n  declare evented: boolean;\n  declare perPixelTargetFind: boolean;\n  declare activeOn: 'down' | 'up';\n\n  declare hoverCursor: CSSStyleDeclaration['cursor'] | null;\n  declare moveCursor: CSSStyleDeclaration['cursor'] | null;\n\n  /**\n   * The object's controls' position in viewport coordinates\n   * Calculated by {@link Control#positionHandler} and {@link Control#calcCornerCoords}, depending on {@link padding}.\n   * `corner/touchCorner` describe the 4 points forming the interactive area of the corner.\n   * Used to draw and locate controls.\n   */\n  declare oCoords: Record<string, TOCoord>;\n\n  /**\n   * keeps the value of the last hovered corner during mouse move.\n   * 0 is no corner, or 'mt', 'ml', 'mtr' etc..\n   * It should be private, but there is no harm in using it as\n   * a read-only property.\n   * this isn't cleaned automatically. Non selected objects may have wrong values\n   * @type [string]\n   */\n  declare __corner?: string;\n\n  /**\n   * a map of control visibility for this object.\n   * this was left when controls were introduced to not break the api too much\n   * this takes priority over the generic control visibility\n   */\n  declare _controlsVisibility: Record<string, boolean>;\n\n  /**\n   * holds the controls for the object.\n   * controls are added by default_controls.js\n   */\n  declare controls: TControlSet;\n\n  /**\n   * internal boolean to signal the code that the object is\n   * part of the move action.\n   */\n  declare isMoving?: boolean;\n\n  /**\n   * A boolean used from the gesture module to keep tracking of a scaling\n   * action when there is no scaling transform in place.\n   * This is an edge case and is used twice in all codebase.\n   * Probably added to keep track of some performance issues\n   * @TODO use git blame to investigate why it was added\n   * DON'T USE IT. WE WILL TRY TO REMOVE IT\n   */\n  declare _scaling?: boolean;\n\n  declare canvas?: Canvas;\n\n  static ownDefaults = interactiveObjectDefaultValues;\n\n  static getDefaults(): Record<string, any> {\n    return {\n      ...super.getDefaults(),\n      controls: createObjectDefaultControls(),\n      ...InteractiveFabricObject.ownDefaults,\n    };\n  }\n\n  /**\n   * Update width and height of the canvas for cache\n   * returns true or false if canvas needed resize.\n   * @private\n   * @return {Boolean} true if the canvas has been resized\n   */\n  _updateCacheCanvas() {\n    const targetCanvas = this.canvas;\n    if (this.noScaleCache && targetCanvas && targetCanvas._currentTransform) {\n      const transform = targetCanvas._currentTransform,\n        target = transform.target,\n        action = transform.action;\n      if (\n        this === (target as unknown as this) &&\n        action &&\n        action.startsWith('scale')\n      ) {\n        return false;\n      }\n    }\n    return super._updateCacheCanvas();\n  }\n\n  getActiveControl() {\n    const key = this.__corner;\n    return key\n      ? {\n        key,\n        control: this.controls[key],\n        coord: this.oCoords[key],\n      }\n      : undefined;\n  }\n\n  /**\n   * Determines which corner is under the mouse cursor, represented by `pointer`.\n   * This function is return a corner only if the object is the active one.\n   * This is done to avoid selecting corner of non active object and activating transformations\n   * rather than drag action. The default behavior of fabricJS is that if you want to transform\n   * an object, first you select it to show the control set\n   * @private\n   * @param {Object} pointer The pointer indicating the mouse position\n   * @param {boolean} forTouch indicates if we are looking for interaction area with a touch action\n   * @return {String|Boolean} corner code (tl, tr, bl, br, etc.), or 0 if nothing is found.\n   */\n  findControl(\n    pointer: Point,\n    forTouch = false\n  ): { key: string; control: Control; coord: TOCoord } | undefined {\n    if (!this.hasControls || !this.canvas) {\n      return undefined;\n    }\n\n    this.__corner = undefined;\n    const cornerEntries = Object.entries(this.oCoords);\n    for (let i = cornerEntries.length - 1; i >= 0; i--) {\n      const [key, corner] = cornerEntries[i];\n      const control = this.controls[key];\n\n      if (\n        control.shouldActivate(\n          key,\n          this,\n          pointer,\n          forTouch ? corner.touchCorner : corner.corner\n        )\n      ) {\n        // this.canvas.contextTop.fillRect(pointer.x - 1, pointer.y - 1, 2, 2);\n        this.__corner = key;\n\n        return { key, control, coord: this.oCoords[key] };\n      }\n    }\n\n    return undefined;\n  }\n\n  /**\n   * Calculates the coordinates of the center of each control plus the corners of the control itself\n   * This basically just delegates to each control positionHandler\n   * WARNING: changing what is passed to positionHandler is a breaking change, since position handler\n   * is a public api and should be done just if extremely necessary\n   * @return {Record<string, TOCoord>}\n   */\n  calcOCoords(): Record<string, TOCoord> {\n    const vpt = this.getViewportTransform(),\n      center = this.getCenterPoint(),\n      tMatrix = createTranslateMatrix(center.x, center.y),\n      rMatrix = createRotateMatrix({\n        angle: this.getTotalAngle() - (!!this.group && this.flipX ? 180 : 0),\n      }),\n      positionMatrix = multiplyTransformMatrices(tMatrix, rMatrix),\n      startMatrix = multiplyTransformMatrices(vpt, positionMatrix),\n      finalMatrix = multiplyTransformMatrices(startMatrix, [\n        1 / vpt[0],\n        0,\n        0,\n        1 / vpt[3],\n        0,\n        0,\n      ]),\n      transformOptions = this.group\n        ? qrDecompose(this.calcTransformMatrix())\n        : undefined;\n    // decomposing could bring negative scaling and `_calculateCurrentDimensions` can't take it\n    if (transformOptions) {\n      transformOptions.scaleX = Math.abs(transformOptions.scaleX);\n      transformOptions.scaleY = Math.abs(transformOptions.scaleY);\n    }\n    const dim = this._calculateCurrentDimensions(transformOptions),\n      coords: Record<string, TOCoord> = {};\n\n    this.forEachControl((control, key) => {\n      const position = control.positionHandler(dim, finalMatrix, this, control);\n      // coords[key] are sometimes used as points. Those are points to which we add\n      // the property corner and touchCorner from `_calcCornerCoords`.\n      // don't remove this assign for an object spread.\n      coords[key] = Object.assign(\n        position,\n        this._calcCornerCoords(control, position)\n      );\n    });\n\n    // debug code\n    /*\n      const canvas = this.canvas;\n      setTimeout(function () {\n      if (!canvas) return;\n        canvas.contextTop.clearRect(0, 0, 700, 700);\n        canvas.contextTop.fillStyle = 'green';\n        Object.keys(coords).forEach(function(key) {\n          const control = coords[key];\n          canvas.contextTop.fillRect(control.x, control.y, 3, 3);\n        });\n      } 50);\n    */\n    return coords;\n  }\n\n  /**\n   * Sets the coordinates that determine the interaction area of each control\n   * note: if we would switch to ROUND corner area, all of this would disappear.\n   * everything would resolve to a single point and a pythagorean theorem for the distance\n   * @todo evaluate simplification of code switching to circle interaction area at runtime\n   * @private\n   */\n  private _calcCornerCoords(control: Control, position: Point) {\n    const angle = this.getTotalAngle();\n    const corner = control.calcCornerCoords(\n      angle,\n      this.cornerSize,\n      position.x,\n      position.y,\n      false,\n      this\n    );\n    const touchCorner = control.calcCornerCoords(\n      angle,\n      this.touchCornerSize,\n      position.x,\n      position.y,\n      true,\n      this\n    );\n    return { corner, touchCorner };\n  }\n\n  /**\n   * @override set controls' coordinates as well\n   * See {@link https://github.com/fabricjs/fabric.js/wiki/When-to-call-setCoords} and {@link http://fabricjs.com/fabric-gotchas}\n   * @return {void}\n   */\n  setCoords(): void {\n    super.setCoords();\n    this.canvas && (this.oCoords = this.calcOCoords());\n  }\n\n  /**\n   * Calls a function for each control. The function gets called,\n   * with the control, the control's key and the object that is calling the iterator\n   * @param {Function} fn function to iterate over the controls over\n   */\n  forEachControl(\n    fn: (\n      control: Control,\n      key: string,\n      fabricObject: InteractiveFabricObject\n    ) => any\n  ) {\n    for (const i in this.controls) {\n      fn(this.controls[i], i, this);\n    }\n  }\n\n  /**\n   * Draws a colored layer behind the object, inside its selection borders.\n   * Requires public options: padding, selectionBackgroundColor\n   * this function is called when the context is transformed\n   * has checks to be skipped when the object is on a staticCanvas\n   * @todo evaluate if make this disappear in favor of a pre-render hook for objects\n   * this was added by Andrea Bogazzi to make possible some feature for work reasons\n   * it seemed a good option, now is an edge case\n   * @param {CanvasRenderingContext2D} ctx Context to draw on\n   */\n  drawSelectionBackground(ctx: CanvasRenderingContext2D): void {\n    if (\n      !this.selectionBackgroundColor ||\n      (this.canvas && (this.canvas._activeObject as unknown as this) !== this)\n    ) {\n      return;\n    }\n    ctx.save();\n    const center = this.getRelativeCenterPoint(),\n      wh = this._calculateCurrentDimensions(),\n      vpt = this.getViewportTransform();\n    ctx.translate(center.x, center.y);\n    ctx.scale(1 / vpt[0], 1 / vpt[3]);\n    ctx.rotate(degreesToRadians(this.angle));\n    ctx.fillStyle = this.selectionBackgroundColor;\n    ctx.fillRect(-wh.x / 2, -wh.y / 2, wh.x, wh.y);\n    ctx.restore();\n  }\n\n  /**\n   * @public override this function in order to customize the drawing of the control box, e.g. rounded corners, different border style.\n   * @param {CanvasRenderingContext2D} ctx ctx is rotated and translated so that (0,0) is at object's center\n   * @param {Point} size the control box size used\n   */\n  strokeBorders(ctx: CanvasRenderingContext2D, size: Point): void {\n    ctx.strokeRect(-size.x / 2, -size.y / 2, size.x, size.y);\n  }\n\n  /**\n   * @private\n   * @param {CanvasRenderingContext2D} ctx Context to draw on\n   * @param {Point} size\n   * @param {TStyleOverride} styleOverride object to override the object style\n   */\n  _drawBorders(\n    ctx: CanvasRenderingContext2D,\n    size: Point,\n    styleOverride: TStyleOverride = {}\n  ): void {\n    const options = {\n      hasControls: this.hasControls,\n      borderColor: this.borderColor,\n      borderDashArray: this.borderDashArray,\n      ...styleOverride,\n    };\n    ctx.save();\n    ctx.strokeStyle = options.borderColor;\n    this._setLineDash(ctx, options.borderDashArray);\n    this.strokeBorders(ctx, size);\n    options.hasControls && this.drawControlsConnectingLines(ctx, size);\n    ctx.restore();\n  }\n\n  /**\n   * Renders controls and borders for the object\n   * the context here is not transformed\n   * @todo move to interactivity\n   * @param {CanvasRenderingContext2D} ctx Context to render on\n   * @param {TStyleOverride} [styleOverride] properties to override the object style\n   */\n  _renderControls(\n    ctx: CanvasRenderingContext2D,\n    styleOverride: TStyleOverride = {}\n  ) {\n    const { hasBorders, hasControls } = this;\n    const styleOptions = {\n      hasBorders,\n      hasControls,\n      ...styleOverride,\n    };\n    const vpt = this.getViewportTransform(),\n      shouldDrawBorders = styleOptions.hasBorders,\n      shouldDrawControls = styleOptions.hasControls;\n    const matrix = multiplyTransformMatrices(vpt, this.calcTransformMatrix());\n    const options = qrDecompose(matrix);\n    ctx.save();\n    ctx.translate(options.translateX, options.translateY);\n    ctx.lineWidth = 1 * this.borderScaleFactor;\n    // since interactive groups have been introduced, an object could be inside a group and needing controls\n    // the following equality check `this.group === this.parent` covers:\n    // object without a group ( undefined === undefined )\n    // object inside a group\n    // excludes object inside a group but multi selected since group and parent will differ in value\n    if (this.group === this.parent) {\n      ctx.globalAlpha = this.isMoving ? this.borderOpacityWhenMoving : 1;\n    }\n    if (this.flipX) {\n      options.angle -= 180;\n    }\n    ctx.rotate(degreesToRadians(this.group ? options.angle : this.angle));\n    shouldDrawBorders && this.drawBorders(ctx, options, styleOverride);\n    shouldDrawControls && this.drawControls(ctx, styleOverride);\n    ctx.restore();\n  }\n\n  /**\n   * Draws borders of an object's bounding box.\n   * Requires public properties: width, height\n   * Requires public options: padding, borderColor\n   * @param {CanvasRenderingContext2D} ctx Context to draw on\n   * @param {object} options object representing current object parameters\n   * @param {TStyleOverride} [styleOverride] object to override the object style\n   */\n  drawBorders(\n    ctx: CanvasRenderingContext2D,\n    options: TQrDecomposeOut,\n    styleOverride: TStyleOverride\n  ): void {\n    let size;\n    if ((styleOverride && styleOverride.forActiveSelection) || this.group) {\n      const bbox = sizeAfterTransform(\n        this.width,\n        this.height,\n        calcDimensionsMatrix(options)\n      ),\n        stroke = !this.isStrokeAccountedForInDimensions()\n          ? (this.strokeUniform\n            ? new Point().scalarAdd(this.canvas ? this.canvas.getZoom() : 1)\n            : // this is extremely confusing. options comes from the upper function\n            // and is the qrDecompose of a matrix that takes in account zoom too\n            new Point(options.scaleX, options.scaleY)\n          ).scalarMultiply(this.strokeWidth)\n          : ZERO;\n      size = bbox\n        .add(stroke)\n        .scalarAdd(this.borderScaleFactor)\n        .scalarAdd(this.padding * 2);\n    } else {\n      size = this._calculateCurrentDimensions().scalarAdd(\n        this.borderScaleFactor\n      );\n    }\n    this._drawBorders(ctx, size, styleOverride);\n  }\n\n  /**\n   * Draws lines from a borders of an object's bounding box to controls that have `withConnection` property set.\n   * Requires public properties: width, height\n   * Requires public options: padding, borderColor\n   * @param {CanvasRenderingContext2D} ctx Context to draw on\n   * @param {Point} size object size x = width, y = height\n   */\n  drawControlsConnectingLines(\n    ctx: CanvasRenderingContext2D,\n    size: Point\n  ): void {\n    let shouldStroke = false;\n\n    ctx.beginPath();\n    this.forEachControl((control, key) => {\n      // in this moment, the ctx is centered on the object.\n      // width and height of the above function are the size of the bbox.\n      if (control.withConnection && control.getVisibility(this, key)) {\n        // reset movement for each control\n        shouldStroke = true;\n        ctx.moveTo(control.x * size.x, control.y * size.y);\n        ctx.lineTo(\n          control.x * size.x + control.offsetX,\n          control.y * size.y + control.offsetY\n        );\n      }\n    });\n    shouldStroke && ctx.stroke();\n  }\n\n  /**\n   * Draws corners of an object's bounding box.\n   * Requires public properties: width, height\n   * Requires public options: cornerSize, padding\n   * @param {CanvasRenderingContext2D} ctx Context to draw on\n   * @param {ControlRenderingStyleOverride} styleOverride object to override the object style\n   */\n  drawControls(\n    ctx: CanvasRenderingContext2D,\n    styleOverride: ControlRenderingStyleOverride = {}\n  ) {\n    ctx.save();\n    const retinaScaling = this.getCanvasRetinaScaling();\n    const { cornerStrokeColor, cornerDashArray, cornerColor } = this;\n    const options = {\n      cornerStrokeColor,\n      cornerDashArray,\n      cornerColor,\n      ...styleOverride,\n    };\n    ctx.setTransform(retinaScaling, 0, 0, retinaScaling, 0, 0);\n    ctx.strokeStyle = ctx.fillStyle = options.cornerColor;\n    if (!this.transparentCorners) {\n      ctx.strokeStyle = options.cornerStrokeColor;\n    }\n    this._setLineDash(ctx, options.cornerDashArray);\n    this.setCoords();\n    this.forEachControl((control, key) => {\n      if (control.getVisibility(this, key)) {\n        const p = this.oCoords[key];\n        control.render(ctx, p.x, p.y, options, this);\n      }\n    });\n    ctx.restore();\n  }\n\n  /**\n   * Returns true if the specified control is visible, false otherwise.\n   * @param {string} controlKey The key of the control. Possible values are usually 'tl', 'tr', 'br', 'bl', 'ml', 'mt', 'mr', 'mb', 'mtr',\n   * but since the control api allow for any control name, can be any string.\n   * @returns {boolean} true if the specified control is visible, false otherwise\n   */\n  isControlVisible(controlKey: string): boolean {\n    return (\n      this.controls[controlKey] &&\n      this.controls[controlKey].getVisibility(this, controlKey)\n    );\n  }\n\n  /**\n   * Sets the visibility of the specified control.\n   * please do not use.\n   * @param {String} controlKey The key of the control. Possible values are 'tl', 'tr', 'br', 'bl', 'ml', 'mt', 'mr', 'mb', 'mtr'.\n   * but since the control api allow for any control name, can be any string.\n   * @param {Boolean} visible true to set the specified control visible, false otherwise\n   * @todo discuss this overlap of priority here with the team. Andrea Bogazzi for details\n   */\n  setControlVisible(controlKey: string, visible: boolean) {\n    if (!this._controlsVisibility) {\n      this._controlsVisibility = {};\n    }\n    this._controlsVisibility[controlKey] = visible;\n  }\n\n  /**\n   * Sets the visibility state of object controls, this is just a bulk option for setControlVisible;\n   * @param {Record<string, boolean>} [options] with an optional key per control\n   * example: {Boolean} [options.bl] true to enable the bottom-left control, false to disable it\n   */\n  setControlsVisibility(options: Record<string, boolean> = {}) {\n    Object.entries(options).forEach(([controlKey, visibility]) =>\n      this.setControlVisible(controlKey, visibility)\n    );\n  }\n\n  /**\n   * Clears the canvas.contextTop in a specific area that corresponds to the object's bounding box\n   * that is in the canvas.contextContainer.\n   * This function is used to clear pieces of contextTop where we render ephemeral effects on top of the object.\n   * Example: blinking cursor text selection, drag effects.\n   * @todo discuss swapping restoreManually with a renderCallback, but think of async issues\n   * @param {Boolean} [restoreManually] When true won't restore the context after clear, in order to draw something else.\n   * @return {CanvasRenderingContext2D|undefined} canvas.contextTop that is either still transformed\n   * with the object transformMatrix, or restored to neutral transform\n   */\n  clearContextTop(\n    restoreManually?: boolean\n  ): CanvasRenderingContext2D | undefined {\n    if (!this.canvas) {\n      return;\n    }\n    const ctx = this.canvas.contextTop;\n    if (!ctx) {\n      return;\n    }\n    const v = this.canvas.viewportTransform;\n    ctx.save();\n    ctx.transform(v[0], v[1], v[2], v[3], v[4], v[5]);\n    this.transform(ctx);\n    // we add 4 pixel, to be sure to do not leave any pixel out\n    const width = this.width + 4,\n      height = this.height + 4;\n    ctx.clearRect(-width / 2, -height / 2, width, height);\n\n    restoreManually || ctx.restore();\n    return ctx;\n  }\n\n  /**\n   * This callback function is called every time _discardActiveObject or _setActiveObject\n   * try to to deselect this object. If the function returns true, the process is cancelled\n   * @param {Object} [options] options sent from the upper functions\n   * @param {TPointerEvent} [options.e] event if the process is generated by an event\n   * @param {FabricObject} [options.object] next object we are setting as active, and reason why\n   * this is being deselected\n   */\n  onDeselect(options?: {\n    e?: TPointerEvent;\n    object?: InteractiveFabricObject;\n  }): boolean {\n    // implemented by sub-classes, as needed.\n    return false;\n  }\n\n  /**\n   * This callback function is called every time _discardActiveObject or _setActiveObject\n   * try to to select this object. If the function returns true, the process is cancelled\n   * @param {Object} [options] options sent from the upper functions\n   * @param {Event} [options.e] event if the process is generated by an event\n   */\n  onSelect(options?: { e?: TPointerEvent }): boolean {\n    // implemented by sub-classes, as needed.\n    return false;\n  }\n\n  /**\n   * Override to customize Drag behavior\n   * Fired from {@link Canvas#_onMouseMove}\n   * @returns true in order for the window to start a drag session\n   */\n  shouldStartDragging(e: TPointerEvent) {\n    return false;\n  }\n\n  /**\n   * Override to customize Drag behavior\\\n   * Fired once a drag session has started\n   * @returns true to handle the drag event\n   */\n  onDragStart(e: DragEvent) {\n    return false;\n  }\n\n  /**\n   * Override to customize drag and drop behavior\n   * @public\n   * @param {DragEvent} e\n   * @returns {boolean} true if the object currently dragged can be dropped on the target\n   */\n  canDrop(e: DragEvent): boolean {\n    return false;\n  }\n\n  /**\n   * Override to customize drag and drop behavior\n   * render a specific effect when an object is the source of a drag event\n   * example: render the selection status for the part of text that is being dragged from a text object\n   * @public\n   * @param {DragEvent} e\n   */\n  renderDragSourceEffect(e: DragEvent) {\n    // for subclasses\n  }\n\n  /**\n   * Override to customize drag and drop behavior\n   * render a specific effect when an object is the target of a drag event\n   * used to show that the underly object can receive a drop, or to show how the\n   * object will change when dropping. example: show the cursor where the text is about to be dropped\n   * @public\n   * @param {DragEvent} e\n   */\n  renderDropTargetEffect(e: DragEvent) {\n    // for subclasses\n  }\n}\n","import type { Constructor } from '../typedefs';\n\n/***\n * https://www.typescriptlang.org/docs/handbook/mixins.html#alternative-pattern\n */\nexport function applyMixins<T extends Constructor, S extends Constructor>(\n  derivedCtor: T,\n  constructors: S[]\n) {\n  constructors.forEach((baseCtor) => {\n    Object.getOwnPropertyNames(baseCtor.prototype).forEach((name) => {\n      name !== 'constructor' &&\n        Object.defineProperty(\n          derivedCtor.prototype,\n          name,\n          Object.getOwnPropertyDescriptor(baseCtor.prototype, name) ||\n            Object.create(null)\n        );\n    });\n  });\n  return derivedCtor as T & { prototype: InstanceType<T & S> };\n}\n","import type { ObjectEvents } from '../../EventTypeDefs';\nimport { FabricObjectSVGExportMixin } from './FabricObjectSVGExportMixin';\nimport { InteractiveFabricObject } from './InteractiveObject';\nimport { applyMixins } from '../../util/applyMixins';\nimport type { FabricObjectProps } from './types/FabricObjectProps';\nimport type { TFabricObjectProps, SerializedObjectProps } from './types';\nimport { classRegistry } from '../../ClassRegistry';\n// TODO somehow we have to make a tree-shakeable import\n// eslint-disable-next-line @typescript-eslint/no-empty-interface, @typescript-eslint/no-unused-vars\nexport interface FabricObject<\n  Props extends TFabricObjectProps = Partial<FabricObjectProps>,\n  SProps extends SerializedObjectProps = SerializedObjectProps,\n  EventSpec extends ObjectEvents = ObjectEvents\n> extends FabricObjectSVGExportMixin {}\n\nexport class FabricObject<\n  Props extends TFabricObjectProps = Partial<FabricObjectProps>,\n  SProps extends SerializedObjectProps = SerializedObjectProps,\n  EventSpec extends ObjectEvents = ObjectEvents\n> extends InteractiveFabricObject<Props, SProps, EventSpec> {}\n\napplyMixins(FabricObject, [FabricObjectSVGExportMixin]);\nclassRegistry.setClass(FabricObject);\nclassRegistry.setClass(FabricObject, 'object');\n\nexport { cacheProperties } from './defaultValues';\n","/**\n * Returns true if context has transparent pixel\n * at specified location (taking tolerance into account)\n * @param {CanvasRenderingContext2D} ctx context\n * @param {Number} x x coordinate in canvasElementCoordinate, not fabric space. integer\n * @param {Number} y y coordinate in canvasElementCoordinate, not fabric space. integer\n * @param {Number} tolerance Tolerance pixels around the point, not alpha tolerance, integer\n * @return {boolean} true if transparent\n */\nexport const isTransparent = (\n  ctx: CanvasRenderingContext2D,\n  x: number,\n  y: number,\n  tolerance: number\n): boolean => {\n  tolerance = Math.round(tolerance);\n  const size = tolerance * 2 + 1;\n  const { data } = ctx.getImageData(x - tolerance, y - tolerance, size, size);\n\n  // Split image data - for tolerance > 1, pixelDataSize = 4;\n  for (let i = 3; i < data.length; i += 4) {\n    const alphaChannel = data[i];\n    if (alphaChannel > 0) {\n      return false;\n    }\n  }\n  return true;\n};\n","import type { XY } from '../../../Point';\nimport { Point } from '../../../Point';\nimport { degreesToRadians } from '../radiansDegreesConversion';\nimport { createVector } from '../vectors';\nimport type { TProjectStrokeOnPointsOptions, TProjection } from './types';\n\n/**\n * @see https://github.com/fabricjs/fabric.js/pull/8344\n * @todo consider removing skewing from points before calculating stroke projection,\n * see https://github.com/fabricjs/fabric.js/commit/494a10ee2f8c2278ae9a55b20bf50cf6ee25b064#commitcomment-94751537\n */\nexport abstract class StrokeProjectionsBase {\n  declare options: TProjectStrokeOnPointsOptions;\n  declare scale: Point;\n  declare strokeUniformScalar: Point;\n  declare strokeProjectionMagnitude: number;\n\n  constructor(options: TProjectStrokeOnPointsOptions) {\n    this.options = options;\n    this.strokeProjectionMagnitude = this.options.strokeWidth / 2;\n    this.scale = new Point(this.options.scaleX, this.options.scaleY);\n    this.strokeUniformScalar = this.options.strokeUniform\n      ? new Point(1 / this.options.scaleX, 1 / this.options.scaleY)\n      : new Point(1, 1);\n  }\n\n  /**\n   * When the stroke is uniform, scaling affects the arrangement of points. So we must take it into account.\n   */\n  protected createSideVector(from: XY, to: XY) {\n    const v = createVector(from, to);\n    return this.options.strokeUniform ? v.multiply(this.scale) : v;\n  }\n\n  protected abstract calcOrthogonalProjection(\n    from: Point,\n    to: Point,\n    magnitude?: number\n  ): Point;\n\n  protected projectOrthogonally(from: Point, to: Point, magnitude?: number) {\n    return this.applySkew(\n      from.add(this.calcOrthogonalProjection(from, to, magnitude))\n    );\n  }\n\n  protected isSkewed() {\n    return this.options.skewX !== 0 || this.options.skewY !== 0;\n  }\n\n  protected applySkew(point: Point) {\n    const p = new Point(point);\n    // skewY must be applied before skewX as this distortion affects skewX calculation\n    p.y += p.x * Math.tan(degreesToRadians(this.options.skewY));\n    p.x += p.y * Math.tan(degreesToRadians(this.options.skewX));\n    return p;\n  }\n\n  protected scaleUnitVector(unitVector: Point, scalar: number) {\n    return unitVector.multiply(this.strokeUniformScalar).scalarMultiply(scalar);\n  }\n\n  protected abstract projectPoints(): Point[];\n\n  public abstract project(): TProjection[];\n}\n","import type { XY } from '../../../Point';\nimport { Point } from '../../../Point';\nimport { halfPI, twoMathPi } from '../../../constants';\nimport type { TRadian } from '../../../typedefs';\nimport { degreesToRadians } from '../radiansDegreesConversion';\nimport {\n  calcAngleBetweenVectors,\n  calcVectorRotation,\n  crossProduct,\n  getOrthonormalVector,\n  getUnitVector,\n  isBetweenVectors,\n  magnitude,\n  rotateVector,\n} from '../vectors';\nimport { StrokeProjectionsBase } from './StrokeProjectionsBase';\nimport type { TProjection, TProjectStrokeOnPointsOptions } from './types';\n\nconst zeroVector = new Point();\n\n/**\n * class in charge of finding projections for each type of line join\n * @see {@link [Closed path projections at #8344](https://github.com/fabricjs/fabric.js/pull/8344#2-closed-path)}\n *\n * - MDN:\n *   - https://developer.mozilla.org/en-US/docs/Web/API/CanvasRenderingContext2D/lineJoin\n *   - https://developer.mozilla.org/en-US/docs/Web/SVG/Attribute/stroke-linejoin\n * - Spec: https://svgwg.org/svg2-draft/painting.html#StrokeLinejoinProperty\n * - Playground to understand how the line joins works: https://hypertolosana.github.io/efficient-webgl-stroking/index.html\n * - View the calculated projections for each of the control points: https://codesandbox.io/s/project-stroke-points-with-context-to-trace-b8jc4j?file=/src/index.js\n *\n */\nexport class StrokeLineJoinProjections extends StrokeProjectionsBase {\n  /**\n   * The point being projected (the angle ∠BAC)\n   */\n  declare A: Point;\n  /**\n   * The point before A\n   */\n  declare B: Point;\n  /**\n   * The point after A\n   */\n  declare C: Point;\n  /**\n   * The AB vector\n   */\n  AB: Point;\n  /**\n   * The AC vector\n   */\n  AC: Point;\n  /**\n   * The angle of A (∠BAC)\n   */\n  alpha: TRadian;\n  /**\n   * The bisector of A (∠BAC)\n   */\n  bisector: Point;\n\n  static getOrthogonalRotationFactor(vector1: Point, vector2?: Point) {\n    const angle = vector2\n      ? calcAngleBetweenVectors(vector1, vector2)\n      : calcVectorRotation(vector1);\n    return Math.abs(angle) < halfPI ? -1 : 1;\n  }\n\n  constructor(A: XY, B: XY, C: XY, options: TProjectStrokeOnPointsOptions) {\n    super(options);\n    this.A = new Point(A);\n    this.B = new Point(B);\n    this.C = new Point(C);\n    this.AB = this.createSideVector(this.A, this.B);\n    this.AC = this.createSideVector(this.A, this.C);\n    this.alpha = calcAngleBetweenVectors(this.AB, this.AC);\n    this.bisector = getUnitVector(\n      // if AC is also the zero vector nothing will be projected\n      // in that case the next point will handle the projection\n      rotateVector(this.AB.eq(zeroVector) ? this.AC : this.AB, this.alpha / 2)\n    );\n  }\n\n  calcOrthogonalProjection(\n    from: Point,\n    to: Point,\n    magnitude: number = this.strokeProjectionMagnitude\n  ) {\n    const vector = this.createSideVector(from, to);\n    const orthogonalProjection = getOrthonormalVector(vector);\n    const correctSide = StrokeLineJoinProjections.getOrthogonalRotationFactor(\n      orthogonalProjection,\n      this.bisector\n    );\n    return this.scaleUnitVector(orthogonalProjection, magnitude * correctSide);\n  }\n\n  /**\n   * BEVEL\n   * Calculation: the projection points are formed by the vector orthogonal to the vertex.\n   *\n   * @see https://github.com/fabricjs/fabric.js/pull/8344#2-2-bevel\n   */\n  projectBevel() {\n    const projections: Point[] = [];\n    // if `alpha` equals 0 or 2*PI, the projections are the same for `B` and `C`\n    (this.alpha % twoMathPi === 0 ? [this.B] : [this.B, this.C]).forEach(\n      (to) => {\n        projections.push(this.projectOrthogonally(this.A, to));\n        projections.push(\n          this.projectOrthogonally(this.A, to, -this.strokeProjectionMagnitude)\n        );\n      }\n    );\n    return projections;\n  }\n\n  /**\n   * MITER\n   * Calculation: the corner is formed by extending the outer edges of the stroke\n   * at the tangents of the path segments until they intersect.\n   *\n   * @see https://github.com/fabricjs/fabric.js/pull/8344#2-1-miter\n   */\n  projectMiter() {\n    const projections: Point[] = [],\n      alpha = Math.abs(this.alpha),\n      hypotUnitScalar = 1 / Math.sin(alpha / 2),\n      miterVector = this.scaleUnitVector(\n        this.bisector,\n        -this.strokeProjectionMagnitude * hypotUnitScalar\n      );\n\n    // When two line segments meet at a sharp angle, it is possible for the join to extend,\n    // far beyond the thickness of the line stroking the path. The stroke-miterlimit imposes\n    // a limit on the extent of the line join.\n    // MDN: https://developer.mozilla.org/en-US/docs/Web/SVG/Attribute/stroke-miterlimit\n    // When the stroke is uniform, scaling changes the arrangement of points, this changes the miter-limit\n    const strokeMiterLimit = this.options.strokeUniform\n      ? magnitude(\n          this.scaleUnitVector(this.bisector, this.options.strokeMiterLimit)\n        )\n      : this.options.strokeMiterLimit;\n\n    if (\n      magnitude(miterVector) / this.strokeProjectionMagnitude <=\n      strokeMiterLimit\n    ) {\n      projections.push(this.applySkew(this.A.add(miterVector)));\n    }\n    /* when the miter-limit is reached, the stroke line join becomes of type bevel.\n      We always need two orthogonal projections which are basically bevel-type projections,\n      so regardless of whether the miter-limit was reached or not, we include these projections.\n    */\n    projections.push(...this.projectBevel());\n\n    return projections;\n  }\n\n  /**\n   * ROUND (without skew)\n   * Calculation: the projections are the two vectors parallel to X and Y axes\n   *\n   * @see https://github.com/fabricjs/fabric.js/pull/8344#2-3-1-round-without-skew\n   */\n  private projectRoundNoSkew(startCircle: Point, endCircle: Point) {\n    const projections: Point[] = [],\n      // correctSide is used to only consider projecting for the outer side\n      correctSide = new Point(\n        StrokeLineJoinProjections.getOrthogonalRotationFactor(this.bisector),\n        StrokeLineJoinProjections.getOrthogonalRotationFactor(\n          new Point(this.bisector.y, this.bisector.x)\n        )\n      ),\n      radiusOnAxisX = new Point(1, 0)\n        .scalarMultiply(this.strokeProjectionMagnitude)\n        .multiply(this.strokeUniformScalar)\n        .multiply(correctSide),\n      radiusOnAxisY = new Point(0, 1)\n        .scalarMultiply(this.strokeProjectionMagnitude)\n        .multiply(this.strokeUniformScalar)\n        .multiply(correctSide);\n\n    [radiusOnAxisX, radiusOnAxisY].forEach((vector) => {\n      if (isBetweenVectors(vector, startCircle, endCircle)) {\n        projections.push(this.A.add(vector));\n      }\n    });\n    return projections;\n  }\n\n  /**\n   * ROUND (with skew)\n   * Calculation: the projections are the points furthest from the vertex in\n   * the direction of the X and Y axes after distortion.\n   *\n   * @see https://github.com/fabricjs/fabric.js/pull/8344#2-3-2-round-skew\n   */\n  private projectRoundWithSkew(startCircle: Point, endCircle: Point) {\n    const projections: Point[] = [];\n\n    const { skewX, skewY, scaleX, scaleY, strokeUniform } = this.options,\n      shearing = new Point(\n        Math.tan(degreesToRadians(skewX)),\n        Math.tan(degreesToRadians(skewY))\n      );\n    // The points furthest from the vertex in the direction of the X and Y axes after distortion\n    const circleRadius = this.strokeProjectionMagnitude,\n      newY = strokeUniform\n        ? circleRadius /\n          scaleY /\n          Math.sqrt(1 / scaleY ** 2 + (1 / scaleX ** 2) * shearing.y ** 2)\n        : circleRadius / Math.sqrt(1 + shearing.y ** 2),\n      furthestY = new Point(\n        // Safe guard due to floating point precision. In some situations the square root\n        // was returning NaN because of a negative number close to zero.\n        Math.sqrt(Math.max(circleRadius ** 2 - newY ** 2, 0)),\n        newY\n      ),\n      newX = strokeUniform\n        ? circleRadius /\n          Math.sqrt(\n            1 +\n              (shearing.x ** 2 * (1 / scaleY) ** 2) /\n                (1 / scaleX + (1 / scaleX) * shearing.x * shearing.y) ** 2\n          )\n        : circleRadius /\n          Math.sqrt(1 + shearing.x ** 2 / (1 + shearing.x * shearing.y) ** 2),\n      furthestX = new Point(\n        newX,\n        Math.sqrt(Math.max(circleRadius ** 2 - newX ** 2, 0))\n      );\n\n    [\n      furthestX,\n      furthestX.scalarMultiply(-1),\n      furthestY,\n      furthestY.scalarMultiply(-1),\n    ]\n      // We need to skew the vector here as this information is used to check if\n      // it is between the start and end of the circle segment\n      .map((vector) =>\n        this.applySkew(\n          strokeUniform ? vector.multiply(this.strokeUniformScalar) : vector\n        )\n      )\n      .forEach((vector) => {\n        if (isBetweenVectors(vector, startCircle, endCircle)) {\n          projections.push(this.applySkew(this.A).add(vector));\n        }\n      });\n\n    return projections;\n  }\n\n  projectRound() {\n    const projections: Point[] = [];\n    /* Include the start and end points of the circle segment, so that only\n      the projections contained within it are included */\n    // add the orthogonal projections (start and end points of circle segment)\n    projections.push(...this.projectBevel());\n    // let's determines which one of the orthogonal projection is the beginning and end of the circle segment.\n    // when `alpha` equals 0 or 2*PI, we have a straight line, so the way to find the start/end is different.\n    const isStraightLine = this.alpha % twoMathPi === 0,\n      // change the origin of the projections to point A\n      // so that the cross product calculation is correct\n      newOrigin = this.applySkew(this.A),\n      proj0 = projections[isStraightLine ? 0 : 2].subtract(newOrigin),\n      proj1 = projections[isStraightLine ? 1 : 0].subtract(newOrigin),\n      // when `isStraightLine` === true, we compare with the vector opposite AB, otherwise we compare with the bisector.\n      comparisonVector = isStraightLine\n        ? this.applySkew(this.AB.scalarMultiply(-1))\n        : this.applySkew(\n            this.bisector.multiply(this.strokeUniformScalar).scalarMultiply(-1)\n          ),\n      // the beginning of the circle segment is always to the right of the comparison vector (cross product > 0)\n      isProj0Start = crossProduct(proj0, comparisonVector) > 0,\n      startCircle = isProj0Start ? proj0 : proj1,\n      endCircle = isProj0Start ? proj1 : proj0;\n    if (!this.isSkewed()) {\n      projections.push(...this.projectRoundNoSkew(startCircle, endCircle));\n    } else {\n      projections.push(...this.projectRoundWithSkew(startCircle, endCircle));\n    }\n    return projections;\n  }\n\n  /**\n   * Project stroke width on points returning projections for each point as follows:\n   * - `miter`: 1 point corresponding to the outer boundary. If the miter limit is exceeded, it will be 2 points (becomes bevel)\n   * - `bevel`: 2 points corresponding to the bevel possible boundaries, orthogonal to the stroke.\n   * - `round`: same as `bevel` when it has no skew, with skew are 4 points.\n   */\n  protected projectPoints() {\n    switch (this.options.strokeLineJoin) {\n      case 'miter':\n        return this.projectMiter();\n      case 'round':\n        return this.projectRound();\n      default:\n        return this.projectBevel();\n    }\n  }\n\n  public project(): TProjection[] {\n    return this.projectPoints().map((point) => ({\n      originPoint: this.A,\n      projectedPoint: point,\n      angle: this.alpha,\n      bisector: this.bisector,\n    }));\n  }\n}\n","import type { XY } from '../../../Point';\nimport { Point } from '../../../Point';\nimport { getOrthonormalVector, getUnitVector } from '../vectors';\nimport { StrokeLineJoinProjections } from './StrokeLineJoinProjections';\nimport { StrokeProjectionsBase } from './StrokeProjectionsBase';\nimport type { TProjection, TProjectStrokeOnPointsOptions } from './types';\n\n/**\n * class in charge of finding projections for each type of line cap for start/end of an open path\n * @see {@link [Open path projections at #8344](https://github.com/fabricjs/fabric.js/pull/8344#1-open-path)}\n *\n * Reference:\n * - MDN:\n *   - https://developer.mozilla.org/en-US/docs/Web/API/CanvasRenderingContext2D/lineCap\n *   - https://developer.mozilla.org/en-US/docs/Web/SVG/Attribute/stroke-linecap\n * - Spec: https://html.spec.whatwg.org/multipage/canvas.html#dom-context-2d-linecap-dev\n * - Playground to understand how the line joins works: https://hypertolosana.github.io/efficient-webgl-stroking/index.html\n * - View the calculated projections for each of the control points: https://codesandbox.io/s/project-stroke-points-with-context-to-trace-b8jc4j?file=/src/index.js\n */\nexport class StrokeLineCapProjections extends StrokeProjectionsBase {\n  /**\n   * edge point\n   */\n  declare A: Point;\n  /**\n   * point next to edge point\n   */\n  declare T: Point;\n\n  constructor(A: XY, T: XY, options: TProjectStrokeOnPointsOptions) {\n    super(options);\n    this.A = new Point(A);\n    this.T = new Point(T);\n  }\n\n  calcOrthogonalProjection(\n    from: Point,\n    to: Point,\n    magnitude: number = this.strokeProjectionMagnitude\n  ) {\n    const vector = this.createSideVector(from, to);\n    return this.scaleUnitVector(getOrthonormalVector(vector), magnitude);\n  }\n\n  /**\n   * OPEN PATH START/END - Line cap: Butt\n   * Calculation: to find the projections, just find the points orthogonal to the stroke\n   *\n   * @see https://github.com/fabricjs/fabric.js/pull/8344#1-1-butt\n   */\n  projectButt() {\n    return [\n      this.projectOrthogonally(this.A, this.T, this.strokeProjectionMagnitude),\n      this.projectOrthogonally(this.A, this.T, -this.strokeProjectionMagnitude),\n    ];\n  }\n\n  /**\n   * OPEN PATH START/END - Line cap: Round\n   * Calculation: same as stroke line join `round`\n   *\n   * @see https://github.com/fabricjs/fabric.js/pull/8344#1-2-round\n   */\n  projectRound() {\n    const projections: Point[] = [];\n\n    if (!this.isSkewed() && this.A.eq(this.T)) {\n      /* 1 point case without `skew`\n        When `strokeUniform` is true, scaling has no effect.\n        So we divide by scale, to remove its effect.\n      */\n      const projection = new Point(1, 1)\n        .scalarMultiply(this.strokeProjectionMagnitude)\n        .multiply(this.strokeUniformScalar);\n      projections.push(\n        this.applySkew(this.A.add(projection)),\n        this.applySkew(this.A.subtract(projection))\n      );\n    } else {\n      projections.push(\n        ...new StrokeLineJoinProjections(\n          this.A,\n          this.T,\n          this.T,\n          this.options\n        ).projectRound()\n      );\n    }\n\n    return projections;\n  }\n\n  /**\n   * OPEN PATH START/END - Line cap: Square\n   * Calculation: project a rectangle of points on the stroke in the opposite direction of the vector `AT`\n   *\n   * @see https://github.com/fabricjs/fabric.js/pull/8344#1-3-square\n   */\n  projectSquare() {\n    const projections: Point[] = [];\n\n    if (this.A.eq(this.T)) {\n      /* 1 point case without `skew`\n        When `strokeUniform` is true, scaling has no effect.\n        So we divide by scale, to remove its effect.\n      */\n      const projection = new Point(1, 1)\n        .scalarMultiply(this.strokeProjectionMagnitude)\n        .multiply(this.strokeUniformScalar);\n      projections.push(this.A.add(projection), this.A.subtract(projection));\n    } else {\n      const orthogonalProjection = this.calcOrthogonalProjection(\n        this.A,\n        this.T,\n        this.strokeProjectionMagnitude\n      );\n      const strokePointingOut = this.scaleUnitVector(\n        getUnitVector(this.createSideVector(this.A, this.T)),\n        -this.strokeProjectionMagnitude\n      );\n      const projectedA = this.A.add(strokePointingOut);\n      projections.push(\n        projectedA.add(orthogonalProjection),\n        projectedA.subtract(orthogonalProjection)\n      );\n    }\n\n    return projections.map((p) => this.applySkew(p));\n  }\n\n  protected projectPoints() {\n    switch (this.options.strokeLineCap) {\n      case 'round':\n        return this.projectRound();\n      case 'square':\n        return this.projectSquare();\n      default:\n        return this.projectButt();\n    }\n  }\n\n  public project(): TProjection[] {\n    return this.projectPoints().map((point) => ({\n      originPoint: this.A,\n      projectedPoint: point,\n    }));\n  }\n}\n","import { Point, type XY } from '../../../Point';\nimport { findIndexRight } from '../../internals';\nimport { StrokeLineCapProjections } from './StrokeLineCapProjections';\nimport { StrokeLineJoinProjections } from './StrokeLineJoinProjections';\nimport type { TProjection, TProjectStrokeOnPointsOptions } from './types';\n\nexport * from './types';\n\n/**\n *\n * Used to calculate object's bounding box\n *\n * @see https://github.com/fabricjs/fabric.js/pull/8344\n *\n */\nexport const projectStrokeOnPoints = (\n  points: XY[],\n  options: TProjectStrokeOnPointsOptions,\n  openPath = false\n): TProjection[] => {\n  const projections: TProjection[] = [];\n\n  if (points.length === 0) {\n    return projections;\n  }\n\n  // first we remove duplicate neighboring points\n  const reduced = points.reduce(\n    (reduced, point) => {\n      if (!reduced[reduced.length - 1].eq(point)) {\n        reduced.push(new Point(point));\n      }\n      return reduced;\n    },\n    [new Point(points[0])]\n  );\n\n  if (reduced.length === 1) {\n    openPath = true;\n  } else if (!openPath) {\n    // remove points from end in case they equal the first point\n    // in order to correctly project the first point\n    const start = reduced[0];\n    const index = findIndexRight(reduced, (point) => !point.eq(start));\n    reduced.splice(index + 1);\n  }\n\n  reduced.forEach((A, index, points) => {\n    let B: XY, C: XY;\n    if (index === 0) {\n      C = points[1];\n      B = openPath ? A : points[points.length - 1];\n    } else if (index === points.length - 1) {\n      B = points[index - 1];\n      C = openPath ? A : points[0];\n    } else {\n      B = points[index - 1];\n      C = points[index + 1];\n    }\n\n    if (openPath && points.length === 1) {\n      projections.push(\n        ...new StrokeLineCapProjections(A, A, options).project()\n      );\n    } else if (openPath && (index === 0 || index === points.length - 1)) {\n      projections.push(\n        ...new StrokeLineCapProjections(\n          A,\n          index === 0 ? C : B,\n          options\n        ).project()\n      );\n    } else {\n      projections.push(\n        ...new StrokeLineJoinProjections(A, B, C, options).project()\n      );\n    }\n  });\n\n  return projections;\n};\n","/**\n * Capitalizes a string\n * @param {String} string String to capitalize\n * @param {Boolean} [firstLetterOnly] If true only first letter is capitalized\n * and other letters stay untouched, if false first letter is capitalized\n * and other letters are converted to lowercase.\n * @return {String} Capitalized version of a string\n */\nexport const capitalize = (string: string, firstLetterOnly = false): string =>\n  `${string.charAt(0).toUpperCase()}${\n    firstLetterOnly ? string.slice(1) : string.slice(1).toLowerCase()\n  }`;\n\n/**\n * Escapes XML in a string\n * @param {String} string String to escape\n * @return {String} Escaped version of a string\n */\nexport const escapeXml = (string: string): string =>\n  string\n    .replace(/&/g, '&amp;')\n    .replace(/\"/g, '&quot;')\n    .replace(/'/g, '&apos;')\n    .replace(/</g, '&lt;')\n    .replace(/>/g, '&gt;');\n\n/**\n * Divide a string in the user perceived single units\n * @param {String} textstring String to escape\n * @return {Array} array containing the graphemes\n */\nexport const graphemeSplit = (textstring: string): string[] => {\n  const graphemes = [];\n  for (let i = 0, chr; i < textstring.length; i++) {\n    if ((chr = getWholeChar(textstring, i)) === false) {\n      continue;\n    }\n    graphemes.push(chr as string);\n  }\n  return graphemes;\n};\n\n// taken from mdn in the charAt doc page.\nconst getWholeChar = (str: string, i: number): string | boolean => {\n  const code = str.charCodeAt(i);\n  if (isNaN(code)) {\n    return ''; // Position not found\n  }\n  if (code < 0xd800 || code > 0xdfff) {\n    return str.charAt(i);\n  }\n\n  // High surrogate (could change last hex to 0xDB7F to treat high private\n  // surrogates as single characters)\n  if (0xd800 <= code && code <= 0xdbff) {\n    if (str.length <= i + 1) {\n      throw 'High surrogate without following low surrogate';\n    }\n    const next = str.charCodeAt(i + 1);\n    if (0xdc00 > next || next > 0xdfff) {\n      throw 'High surrogate without following low surrogate';\n    }\n    return str.charAt(i) + str.charAt(i + 1);\n  }\n  // Low surrogate (0xDC00 <= code && code <= 0xDFFF)\n  if (i === 0) {\n    throw 'Low surrogate without preceding high surrogate';\n  }\n  const prev = str.charCodeAt(i - 1);\n\n  // (could change last hex to 0xDB7F to treat high private\n  // surrogates as single characters)\n  if (0xd800 > prev || prev > 0xdbff) {\n    throw 'Low surrogate without preceding high surrogate';\n  }\n  // We can pass over low surrogates now as the second component\n  // in a pair which we have already processed\n  return false;\n};\n","import { reNewline } from '../../constants';\nimport type {\n  TextStyle,\n  TextStyleDeclaration,\n} from '../../shapes/Text/StyledText';\nimport { cloneDeep } from '../internals/cloneDeep';\nimport { graphemeSplit } from '../lang_string';\n\nexport type TextStyleArray = {\n  start: number;\n  end: number;\n  style: TextStyleDeclaration;\n}[];\n\n/**\n * @param {Object} prevStyle first style to compare\n * @param {Object} thisStyle second style to compare\n * @param {boolean} forTextSpans whether to check overline, underline, and line-through properties\n * @return {boolean} true if the style changed\n */\nexport const hasStyleChanged = (\n  prevStyle: TextStyleDeclaration,\n  thisStyle: TextStyleDeclaration,\n  forTextSpans = false\n) =>\n  prevStyle.fill !== thisStyle.fill ||\n  prevStyle.stroke !== thisStyle.stroke ||\n  prevStyle.strokeWidth !== thisStyle.strokeWidth ||\n  prevStyle.fontSize !== thisStyle.fontSize ||\n  prevStyle.fontFamily !== thisStyle.fontFamily ||\n  prevStyle.fontWeight !== thisStyle.fontWeight ||\n  prevStyle.fontStyle !== thisStyle.fontStyle ||\n  prevStyle.textBackgroundColor !== thisStyle.textBackgroundColor ||\n  prevStyle.deltaY !== thisStyle.deltaY ||\n  (forTextSpans &&\n    (prevStyle.overline !== thisStyle.overline ||\n      prevStyle.underline !== thisStyle.underline ||\n      prevStyle.linethrough !== thisStyle.linethrough));\n\n/**\n * Returns the array form of a text object's inline styles property with styles grouped in ranges\n * rather than per character. This format is less verbose, and is better suited for storage\n * so it is used in serialization (not during runtime).\n * @param {object} styles per character styles for a text object\n * @param {String} text the text string that the styles are applied to\n * @return {{start: number, end: number, style: object}[]}\n */\nexport const stylesToArray = (\n  styles: TextStyle,\n  text: string\n): TextStyleArray => {\n  const textLines = text.split('\\n'),\n    stylesArray = [];\n  let charIndex = -1,\n    prevStyle = {};\n  // clone style structure to prevent mutation\n  styles = cloneDeep(styles);\n\n  //loop through each textLine\n  for (let i = 0; i < textLines.length; i++) {\n    const chars = graphemeSplit(textLines[i]);\n    if (!styles[i]) {\n      //no styles exist for this line, so add the line's length to the charIndex total and reset prevStyle\n      charIndex += chars.length;\n      prevStyle = {};\n      continue;\n    }\n    //loop through each character of the current line\n    for (let c = 0; c < chars.length; c++) {\n      charIndex++;\n      const thisStyle = styles[i][c];\n      //check if style exists for this character\n      if (thisStyle && Object.keys(thisStyle).length > 0) {\n        if (hasStyleChanged(prevStyle, thisStyle, true)) {\n          stylesArray.push({\n            start: charIndex,\n            end: charIndex + 1,\n            style: thisStyle,\n          });\n        } else {\n          //if style is the same as previous character, increase end index\n          stylesArray[stylesArray.length - 1].end++;\n        }\n      }\n      prevStyle = thisStyle || {};\n    }\n  }\n  return stylesArray;\n};\n\n/**\n * Returns the object form of the styles property with styles that are assigned per\n * character rather than grouped by range. This format is more verbose, and is\n * only used during runtime (not for serialization/storage)\n * @param {Array} styles the serialized form of a text object's styles\n * @param {String} text the text string that the styles are applied to\n * @return {Object}\n */\nexport const stylesFromArray = (\n  styles: TextStyleArray | TextStyle,\n  text: string\n): TextStyle => {\n  if (!Array.isArray(styles)) {\n    // clone to prevent mutation\n    return cloneDeep(styles);\n  }\n  const textLines = text.split(reNewline),\n    stylesObject: TextStyle = {};\n  let charIndex = -1,\n    styleIndex = 0;\n  //loop through each textLine\n  for (let i = 0; i < textLines.length; i++) {\n    const chars = graphemeSplit(textLines[i]);\n\n    //loop through each character of the current line\n    for (let c = 0; c < chars.length; c++) {\n      charIndex++;\n      //check if there's a style collection that includes the current character\n      if (\n        styles[styleIndex] &&\n        styles[styleIndex].start <= charIndex &&\n        charIndex < styles[styleIndex].end\n      ) {\n        //create object for line index if it doesn't exist\n        stylesObject[i] = stylesObject[i] || {};\n        //assign a style at this character's index\n        stylesObject[i][c] = { ...styles[styleIndex].style };\n        //if character is at the end of the current style collection, move to the next\n        if (charIndex === styles[styleIndex].end - 1) {\n          styleIndex++;\n        }\n      }\n    }\n  }\n  return stylesObject;\n};\n","/**\n * Attributes parsed from all SVG elements\n * @type array\n */\nexport const SHARED_ATTRIBUTES = [\n  'display',\n  'transform',\n  'fill',\n  'fill-opacity',\n  'fill-rule',\n  'opacity',\n  'stroke',\n  'stroke-dasharray',\n  'stroke-linecap',\n  'stroke-dashoffset',\n  'stroke-linejoin',\n  'stroke-miterlimit',\n  'stroke-opacity',\n  'stroke-width',\n  'id',\n  'paint-order',\n  'vector-effect',\n  'instantiated_by_use',\n  'clip-path',\n];\n","export function selectorMatches(element: HTMLElement, selector: string) {\n  const nodeName = element.nodeName;\n  const classNames = element.getAttribute('class');\n  const id = element.getAttribute('id');\n  const azAz = '(?![a-zA-Z\\\\-]+)';\n  let matcher;\n  // i check if a selector matches slicing away part from it.\n  // if i get empty string i should match\n  matcher = new RegExp('^' + nodeName, 'i');\n  selector = selector.replace(matcher, '');\n  if (id && selector.length) {\n    matcher = new RegExp('#' + id + azAz, 'i');\n    selector = selector.replace(matcher, '');\n  }\n  if (classNames && selector.length) {\n    const splitClassNames = classNames.split(' ');\n    for (let i = splitClassNames.length; i--; ) {\n      matcher = new RegExp('\\\\.' + splitClassNames[i] + azAz, 'i');\n      selector = selector.replace(matcher, '');\n    }\n  }\n  return selector.length === 0;\n}\n","import { selectorMatches } from './selectorMatches';\nimport { doesSomeParentMatch } from './doesSomeParentMatch';\n\n/**\n * @private\n */\n\nexport function elementMatchesRule(element: HTMLElement, selectors: string[]) {\n  let parentMatching = true;\n  // start from rightmost selector.\n  const firstMatching = selectorMatches(element, selectors.pop()!);\n  if (firstMatching && selectors.length) {\n    parentMatching = doesSomeParentMatch(element, selectors);\n  }\n  return firstMatching && parentMatching && selectors.length === 0;\n}\n","import { selectorMatches } from './selectorMatches';\n\nexport function doesSomeParentMatch(element: HTMLElement, selectors: string[]) {\n  let selector: string,\n    parentMatching = true;\n  while (\n    element.parentElement &&\n    element.parentElement.nodeType === 1 &&\n    selectors.length\n  ) {\n    if (parentMatching) {\n      selector = selectors.pop()!;\n    }\n    element = element.parentElement;\n    parentMatching = selectorMatches(element, selector!);\n  }\n  return selectors.length === 0;\n}\n","import { attributesMap } from './constants';\n\nexport const normalizeAttr = (\n  attr: keyof typeof attributesMap | string\n): string => attributesMap[attr as keyof typeof attributesMap] ?? attr;\n","import { reNum } from '../../parser/constants';\n\nexport const cleanupSvgAttribute = (attributeValue: string) =>\n  attributeValue\n    .replace(new RegExp(`(${reNum})`, 'gi'), ' $1 ')\n    // replace annoying commas and arbitrary whitespace with single spaces\n    .replace(/,/gi, ' ')\n    .replace(/\\s+/gi, ' ');\n","import { iMatrix } from '../constants';\nimport { reNum } from './constants';\nimport type { TMat2D } from '../typedefs';\nimport { cleanupSvgAttribute } from '../util/internals/cleanupSvgAttribute';\nimport {\n  createRotateMatrix,\n  createScaleMatrix,\n  createSkewXMatrix,\n  createSkewYMatrix,\n  createTranslateMatrix,\n  multiplyTransformMatrixArray,\n} from '../util/misc/matrix';\n\n// == begin transform regexp\nconst p = `(${reNum})`;\nconst skewX = String.raw`(skewX)\\(${p}\\)`;\nconst skewY = String.raw`(skewY)\\(${p}\\)`;\nconst rotate = String.raw`(rotate)\\(${p}(?: ${p} ${p})?\\)`;\nconst scale = String.raw`(scale)\\(${p}(?: ${p})?\\)`;\nconst translate = String.raw`(translate)\\(${p}(?: ${p})?\\)`;\nconst matrix = String.raw`(matrix)\\(${p} ${p} ${p} ${p} ${p} ${p}\\)`;\nconst transform = `(?:${matrix}|${translate}|${rotate}|${scale}|${skewX}|${skewY})`;\nconst transforms = `(?:${transform}*)`;\nconst transformList = String.raw`^\\s*(?:${transforms}?)\\s*$`;\n// http://www.w3.org/TR/SVG/coords.html#TransformAttribute\nconst reTransformList = new RegExp(transformList);\n// == end transform regexp\nconst reTransform = new RegExp(transform, 'g');\n\n/**\n * Parses \"transform\" attribute, returning an array of values\n * @static\n * @function\n * @memberOf fabric\n * @param {String} attributeValue String containing attribute value\n * @return {TTransformMatrix} Array of 6 elements representing transformation matrix\n */\nexport function parseTransformAttribute(attributeValue: string): TMat2D {\n  // first we clean the string\n  attributeValue = cleanupSvgAttribute(attributeValue)\n    // remove spaces around front parentheses\n    .replace(/\\s*([()])\\s*/gi, '$1');\n\n  // start with identity matrix\n  const matrices: TMat2D[] = [];\n\n  // return if no argument was given or\n  // an argument does not match transform attribute regexp\n  if (\n    !attributeValue ||\n    (attributeValue && !reTransformList.test(attributeValue))\n  ) {\n    return [...iMatrix];\n  }\n\n  for (const match of attributeValue.matchAll(reTransform)) {\n    const transformMatch = new RegExp(transform).exec(match[0]);\n    if (!transformMatch) {\n      continue;\n    }\n    let matrix: TMat2D = iMatrix;\n    const matchedParams = transformMatch.filter((m) => !!m);\n    const [, operation, ...rawArgs] = matchedParams;\n    const [arg0, arg1, arg2, arg3, arg4, arg5] = rawArgs.map((arg) =>\n      parseFloat(arg)\n    );\n\n    switch (operation) {\n      case 'translate':\n        matrix = createTranslateMatrix(arg0, arg1);\n        break;\n      case 'rotate':\n        matrix = createRotateMatrix({ angle: arg0 }, { x: arg1, y: arg2 });\n        break;\n      case 'scale':\n        matrix = createScaleMatrix(arg0, arg1);\n        break;\n      case 'skewX':\n        matrix = createSkewXMatrix(arg0);\n        break;\n      case 'skewY':\n        matrix = createSkewYMatrix(arg0);\n        break;\n      case 'matrix':\n        matrix = [arg0, arg1, arg2, arg3, arg4, arg5];\n        break;\n    }\n\n    // snapshot current matrix into matrices array\n    matrices.push(matrix);\n  }\n\n  return multiplyTransformMatrixArray(matrices);\n}\n","import { multiplyTransformMatrices } from '../util/misc/matrix';\nimport { parseUnit } from '../util/misc/svgParsing';\nimport { parseTransformAttribute } from './parseTransformAttribute';\nimport { CENTER, LEFT, RIGHT, NONE } from '../constants';\n\nexport function normalizeValue(\n  attr: string,\n  value: any,\n  parentAttributes: Record<string, any>,\n  fontSize: number\n): string | null | boolean | number[] | number {\n  const isArray = Array.isArray(value);\n  let parsed: number | number[];\n  let ouputValue: string | null | boolean | number[] | number = value;\n  if ((attr === 'fill' || attr === 'stroke') && value === NONE) {\n    ouputValue = '';\n  } else if (attr === 'strokeUniform') {\n    return value === 'non-scaling-stroke';\n  } else if (attr === 'strokeDashArray') {\n    if (value === NONE) {\n      ouputValue = null;\n    } else {\n      ouputValue = value.replace(/,/g, ' ').split(/\\s+/).map(parseFloat);\n    }\n  } else if (attr === 'transformMatrix') {\n    if (parentAttributes && parentAttributes.transformMatrix) {\n      ouputValue = multiplyTransformMatrices(\n        parentAttributes.transformMatrix,\n        parseTransformAttribute(value)\n      );\n    } else {\n      ouputValue = parseTransformAttribute(value);\n    }\n  } else if (attr === 'visible') {\n    ouputValue = value !== NONE && value !== 'hidden';\n    // display=none on parent element always takes precedence over child element\n    if (parentAttributes && parentAttributes.visible === false) {\n      ouputValue = false;\n    }\n  } else if (attr === 'opacity') {\n    ouputValue = parseFloat(value);\n    if (parentAttributes && typeof parentAttributes.opacity !== 'undefined') {\n      ouputValue *= parentAttributes.opacity as number;\n    }\n  } else if (attr === 'textAnchor' /* text-anchor */) {\n    ouputValue = value === 'start' ? LEFT : value === 'end' ? RIGHT : CENTER;\n  } else if (attr === 'charSpacing') {\n    // parseUnit returns px and we convert it to em\n    parsed = (parseUnit(value, fontSize) / fontSize) * 1000;\n  } else if (attr === 'paintFirst') {\n    const fillIndex = value.indexOf('fill');\n    const strokeIndex = value.indexOf('stroke');\n    ouputValue = 'fill';\n    if (fillIndex > -1 && strokeIndex > -1 && strokeIndex < fillIndex) {\n      ouputValue = 'stroke';\n    } else if (fillIndex === -1 && strokeIndex > -1) {\n      ouputValue = 'stroke';\n    }\n  } else if (attr === 'href' || attr === 'xlink:href' || attr === 'font') {\n    return value;\n  } else if (attr === 'imageSmoothing') {\n    return value === 'optimizeQuality';\n  } else {\n    parsed = isArray\n      ? (value as string[]).map(parseUnit)\n      : parseUnit(value, fontSize);\n  }\n\n  return !isArray && isNaN(parsed! as number) ? ouputValue : parsed!;\n}\n","import { parseUnit } from '../util/misc/svgParsing';\nimport { reFontDeclaration } from './constants';\n\n/**\n * Parses a short font declaration, building adding its properties to a style object\n * @static\n * @function\n * @memberOf fabric\n * @param {String} value font declaration\n * @param {Object} oStyle definition\n */\nexport function parseFontDeclaration(\n  value: string,\n  oStyle: Record<string, any>\n): void {\n  const match = value.match(reFontDeclaration);\n\n  if (!match) {\n    return;\n  }\n  const fontStyle = match[1],\n    // font variant is not used\n    // fontVariant = match[2],\n    fontWeight = match[3],\n    fontSize = match[4],\n    lineHeight = match[5],\n    fontFamily = match[6];\n\n  if (fontStyle) {\n    oStyle.fontStyle = fontStyle;\n  }\n  if (fontWeight) {\n    oStyle.fontWeight = isNaN(parseFloat(fontWeight))\n      ? fontWeight\n      : parseFloat(fontWeight);\n  }\n  if (fontSize) {\n    oStyle.fontSize = parseUnit(fontSize);\n  }\n  if (fontFamily) {\n    oStyle.fontFamily = fontFamily;\n  }\n  if (lineHeight) {\n    oStyle.lineHeight = lineHeight === 'normal' ? 1 : lineHeight;\n  }\n}\n","import { parseStyleObject } from './parseStyleObject';\nimport { parseStyleString } from './parseStyleString';\n\n/**\n * Parses \"style\" attribute, retuning an object with values\n * @static\n * @memberOf fabric\n * @param {SVGElement} element Element to parse\n * @return {Object} Objects with values parsed from style attribute of an element\n */\nexport function parseStyleAttribute(element: HTMLElement): Record<string, any> {\n  const oStyle: Record<string, any> = {},\n    style = element.getAttribute('style');\n\n  if (!style) {\n    return oStyle;\n  }\n\n  if (typeof style === 'string') {\n    parseStyleString(style, oStyle);\n  } else {\n    parseStyleObject(style, oStyle);\n  }\n\n  return oStyle;\n}\n","/**\n * Takes a style string and parses it in one that has only defined values\n * and lowercases properties\n * @param style\n * @param oStyle\n */\nexport function parseStyleString(\n  style: string,\n  oStyle: Record<string, any>\n): void {\n  style\n    .replace(/;\\s*$/, '')\n    .split(';')\n    .forEach((chunk) => {\n      const [attr, value] = chunk.split(':');\n      oStyle[attr.trim().toLowerCase()] = value.trim();\n    });\n}\n","/**\n * Takes a style object and parses it in one that has only defined values\n * and lowercases properties\n * @param style\n * @param oStyle\n */\nexport function parseStyleObject(\n  style: Record<string, any>,\n  oStyle: Record<string, any>\n): void {\n  Object.entries(style).forEach(([prop, value]) => {\n    if (value === undefined) {\n      return;\n    }\n    oStyle[prop.toLowerCase()] = value;\n  });\n}\n","import { Color } from '../color/Color';\nimport { toFixed } from '../util/misc/toFixed';\nimport { FabricObject } from '../shapes/Object/FabricObject';\n\nconst colorAttributesMap = {\n  stroke: 'strokeOpacity',\n  fill: 'fillOpacity',\n};\n\n/**\n * @private\n * @param {Object} attributes Array of attributes to parse\n */\n\nexport function setStrokeFillOpacity(\n  attributes: Record<string, any>\n): Record<string, any> {\n  const defaults = FabricObject.getDefaults();\n  Object.entries(colorAttributesMap).forEach(([attr, colorAttr]) => {\n    if (\n      typeof attributes[colorAttr] === 'undefined' ||\n      attributes[attr] === ''\n    ) {\n      return;\n    }\n    if (typeof attributes[attr] === 'undefined') {\n      if (!defaults[attr]) {\n        return;\n      }\n      attributes[attr] = defaults[attr];\n    }\n    if (attributes[attr].indexOf('url(') === 0) {\n      return;\n    }\n    const color = new Color(attributes[attr]);\n    attributes[attr] = color\n      .setAlpha(toFixed(color.getAlpha() * attributes[colorAttr], 2))\n      .toRgba();\n  });\n  return attributes;\n}\n","import { DEFAULT_SVG_FONT_SIZE } from '../constants';\nimport { parseUnit } from '../util/misc/svgParsing';\nimport { cPath, fSize, svgValidParentsRegEx } from './constants';\nimport { getGlobalStylesForElement } from './getGlobalStylesForElement';\nimport { normalizeAttr } from './normalizeAttr';\nimport { normalizeValue } from './normalizeValue';\nimport { parseFontDeclaration } from './parseFontDeclaration';\nimport { parseStyleAttribute } from './parseStyleAttribute';\nimport { setStrokeFillOpacity } from './setStrokeFillOpacity';\nimport type { CSSRules } from './typedefs';\n\n/**\n * Returns an object of attributes' name/value, given element and an array of attribute names;\n * Parses parent \"g\" nodes recursively upwards.\n * @param {SVGElement | HTMLElement} element Element to parse\n * @param {Array} attributes Array of attributes to parse\n * @return {Object} object containing parsed attributes' names/values\n */\nexport function parseAttributes(\n  element: HTMLElement | null,\n  attributes: string[],\n  cssRules?: CSSRules\n): Record<string, any> {\n  if (!element) {\n    return {};\n  }\n\n  let parentAttributes: Record<string, string> = {},\n    fontSize: number,\n    parentFontSize = DEFAULT_SVG_FONT_SIZE;\n\n  // if there's a parent container (`g` or `a` or `symbol` node), parse its attributes recursively upwards\n  if (\n    element.parentNode &&\n    svgValidParentsRegEx.test(element.parentNode.nodeName)\n  ) {\n    parentAttributes = parseAttributes(\n      element.parentElement,\n      attributes,\n      cssRules\n    );\n    if (parentAttributes.fontSize) {\n      fontSize = parentFontSize = parseUnit(parentAttributes.fontSize);\n    }\n  }\n\n  const ownAttributes: Record<string, string> = {\n    ...attributes.reduce<Record<string, string>>((memo, attr) => {\n      const value = element.getAttribute(attr);\n      if (value) {\n        memo[attr] = value;\n      }\n      return memo;\n    }, {}),\n    // add values parsed from style, which take precedence over attributes\n    // (see: http://www.w3.org/TR/SVG/styling.html#UsingPresentationAttributes)\n    ...getGlobalStylesForElement(element, cssRules),\n    ...parseStyleAttribute(element),\n  };\n\n  if (ownAttributes[cPath]) {\n    element.setAttribute(cPath, ownAttributes[cPath]);\n  }\n  if (ownAttributes[fSize]) {\n    // looks like the minimum should be 9px when dealing with ems. this is what looks like in browsers.\n    fontSize = parseUnit(ownAttributes[fSize], parentFontSize);\n    ownAttributes[fSize] = `${fontSize}`;\n  }\n\n  // this should have its own complex type\n  const normalizedStyle: Record<\n    string,\n    string | boolean | number | number[] | null\n  > = {};\n  for (const attr in ownAttributes) {\n    const normalizedAttr = normalizeAttr(attr);\n    const normalizedValue = normalizeValue(\n      normalizedAttr,\n      ownAttributes[attr],\n      parentAttributes,\n      fontSize!\n    );\n    normalizedStyle[normalizedAttr] = normalizedValue;\n  }\n  if (normalizedStyle && normalizedStyle.font) {\n    parseFontDeclaration(normalizedStyle.font as string, normalizedStyle);\n  }\n  const mergedAttrs = { ...parentAttributes, ...normalizedStyle };\n  return svgValidParentsRegEx.test(element.nodeName)\n    ? mergedAttrs\n    : setStrokeFillOpacity(mergedAttrs);\n}\n","import { elementMatchesRule } from './elementMatchesRule';\nimport type { CSSRules } from './typedefs';\n\n/**\n * @private\n */\n\nexport function getGlobalStylesForElement(\n  element: HTMLElement,\n  cssRules: CSSRules = {}\n) {\n  let styles: Record<string, string> = {};\n  for (const rule in cssRules) {\n    if (elementMatchesRule(element, rule.split(' '))) {\n      styles = {\n        ...styles,\n        ...cssRules[rule],\n      };\n    }\n  }\n  return styles;\n}\n","import { kRect } from '../constants';\nimport { SHARED_ATTRIBUTES } from '../parser/attributes';\nimport { parseAttributes } from '../parser/parseAttributes';\nimport type { Abortable, TClassProperties, TOptions } from '../typedefs';\nimport { classRegistry } from '../ClassRegistry';\nimport { FabricObject, cacheProperties } from './Object/FabricObject';\nimport type { FabricObjectProps, SerializedObjectProps } from './Object/types';\nimport type { ObjectEvents } from '../EventTypeDefs';\nimport type { CSSRules } from '../parser/typedefs';\n\nexport const rectDefaultValues: Partial<TClassProperties<Rect>> = {\n  rx: 0,\n  ry: 0,\n};\n\ninterface UniqueRectProps {\n  rx: number;\n  ry: number;\n}\n\nexport interface SerializedRectProps\n  extends SerializedObjectProps,\n    UniqueRectProps {}\n\nexport interface RectProps extends FabricObjectProps, UniqueRectProps {}\n\nconst RECT_PROPS = ['rx', 'ry'] as const;\n\nexport class Rect<\n    Props extends TOptions<RectProps> = Partial<RectProps>,\n    SProps extends SerializedRectProps = SerializedRectProps,\n    EventSpec extends ObjectEvents = ObjectEvents\n  >\n  extends FabricObject<Props, SProps, EventSpec>\n  implements RectProps\n{\n  /**\n   * Horizontal border radius\n   * @type Number\n   * @default\n   */\n  declare rx: number;\n\n  /**\n   * Vertical border radius\n   * @type Number\n   * @default\n   */\n  declare ry: number;\n\n  static type = 'Rect';\n\n  static cacheProperties = [...cacheProperties, ...RECT_PROPS];\n\n  static ownDefaults = rectDefaultValues;\n\n  static getDefaults(): Record<string, any> {\n    return {\n      ...super.getDefaults(),\n      ...Rect.ownDefaults,\n    };\n  }\n\n  /**\n   * Constructor\n   * @param {Object} [options] Options object\n   * @return {Object} thisArg\n   */\n  constructor(options: Props) {\n    super(options);\n    this._initRxRy();\n  }\n\n  /**\n   * Initializes rx/ry attributes\n   * @private\n   */\n  _initRxRy() {\n    const { rx, ry } = this;\n    if (rx && !ry) {\n      this.ry = rx;\n    } else if (ry && !rx) {\n      this.rx = ry;\n    }\n  }\n\n  /**\n   * @private\n   * @param {CanvasRenderingContext2D} ctx Context to render on\n   */\n  _render(ctx: CanvasRenderingContext2D) {\n    const { width: w, height: h } = this;\n    const x = -w / 2;\n    const y = -h / 2;\n    const rx = this.rx ? Math.min(this.rx, w / 2) : 0;\n    const ry = this.ry ? Math.min(this.ry, h / 2) : 0;\n    const isRounded = rx !== 0 || ry !== 0;\n\n    ctx.beginPath();\n\n    ctx.moveTo(x + rx, y);\n\n    ctx.lineTo(x + w - rx, y);\n    isRounded &&\n      ctx.bezierCurveTo(\n        x + w - kRect * rx,\n        y,\n        x + w,\n        y + kRect * ry,\n        x + w,\n        y + ry\n      );\n\n    ctx.lineTo(x + w, y + h - ry);\n    isRounded &&\n      ctx.bezierCurveTo(\n        x + w,\n        y + h - kRect * ry,\n        x + w - kRect * rx,\n        y + h,\n        x + w - rx,\n        y + h\n      );\n\n    ctx.lineTo(x + rx, y + h);\n    isRounded &&\n      ctx.bezierCurveTo(\n        x + kRect * rx,\n        y + h,\n        x,\n        y + h - kRect * ry,\n        x,\n        y + h - ry\n      );\n\n    ctx.lineTo(x, y + ry);\n    isRounded &&\n      ctx.bezierCurveTo(x, y + kRect * ry, x + kRect * rx, y, x + rx, y);\n\n    ctx.closePath();\n\n    this._renderPaintInOrder(ctx);\n  }\n\n  /**\n   * Returns object representation of an instance\n   * @param {Array} [propertiesToInclude] Any properties that you might want to additionally include in the output\n   * @return {Object} object representation of an instance\n   */\n  toObject<\n    T extends Omit<Props & TClassProperties<this>, keyof SProps>,\n    K extends keyof T = never\n  >(propertiesToInclude: K[] = []): Pick<T, K> & SProps {\n    return super.toObject([...RECT_PROPS, ...propertiesToInclude]);\n  }\n\n  /**\n   * Returns svg representation of an instance\n   * @return {Array} an array of strings with the specific svg representation\n   * of the instance\n   */\n  _toSVG() {\n    const { width, height, rx, ry } = this;\n    return [\n      '<rect ',\n      'COMMON_PARTS',\n      `x=\"${-width / 2}\" y=\"${\n        -height / 2\n      }\" rx=\"${rx}\" ry=\"${ry}\" width=\"${width}\" height=\"${height}\" />\\n`,\n    ];\n  }\n\n  /**\n   * List of attribute names to account for when parsing SVG element (used by `Rect.fromElement`)\n   * @static\n   * @memberOf Rect\n   * @see: http://www.w3.org/TR/SVG/shapes.html#RectElement\n   */\n  static ATTRIBUTE_NAMES = [\n    ...SHARED_ATTRIBUTES,\n    'x',\n    'y',\n    'rx',\n    'ry',\n    'width',\n    'height',\n  ];\n\n  /* _FROM_SVG_START_ */\n\n  /**\n   * Returns {@link Rect} instance from an SVG element\n   * @static\n   * @memberOf Rect\n   * @param {HTMLElement} element Element to parse\n   * @param {Object} [options] Options object\n   */\n  static async fromElement(\n    element: HTMLElement,\n    options: Abortable,\n    cssRules?: CSSRules\n  ) {\n    const {\n      left = 0,\n      top = 0,\n      width = 0,\n      height = 0,\n      visible = true,\n      ...restOfparsedAttributes\n    } = parseAttributes(element, this.ATTRIBUTE_NAMES, cssRules);\n\n    return new this({\n      ...options,\n      ...restOfparsedAttributes,\n      left,\n      top,\n      width,\n      height,\n      visible: Boolean(visible && width && height),\n    });\n  }\n\n  /* _FROM_SVG_END_ */\n}\n\nclassRegistry.setClass(Rect);\nclassRegistry.setSVGClass(Rect);\n","export const LAYOUT_TYPE_INITIALIZATION = 'initialization';\nexport const LAYOUT_TYPE_ADDED = 'added';\nexport const LAYOUT_TYPE_REMOVED = 'removed';\nexport const LAYOUT_TYPE_IMPERATIVE = 'imperative';\nexport const LAYOUT_TYPE_OBJECT_MODIFIED = 'object_modified';\nexport const LAYOUT_TYPE_OBJECT_MODIFYING = 'object_modifying';\n","import { Point, ZERO } from '../../Point';\nimport type { Group } from '../../shapes/Group';\nimport type { FabricObject } from '../../shapes/Object/FabricObject';\nimport { multiplyTransformMatrixArray } from '../../util/misc/matrix';\nimport { sizeAfterTransform } from '../../util/misc/objectTransforms';\nimport {\n  calcPlaneChangeMatrix,\n  sendVectorToPlane,\n} from '../../util/misc/planeChange';\n\n/**\n * @returns 2 points, the tl and br corners of the non rotated bounding box of an object\n * in the {@link group} plane, taking into account objects that {@link group} is their parent\n * but also belong to the active selection.\n */\nexport const getObjectBounds = (\n  destinationGroup: Group,\n  object: FabricObject\n): Point[] => {\n  const {\n    strokeUniform,\n    strokeWidth,\n    width,\n    height,\n    group: currentGroup,\n  } = object;\n  const t =\n    currentGroup && currentGroup !== destinationGroup\n      ? calcPlaneChangeMatrix(\n          currentGroup.calcTransformMatrix(),\n          destinationGroup.calcTransformMatrix()\n        )\n      : null;\n  const objectCenter = t\n    ? object.getRelativeCenterPoint().transform(t)\n    : object.getRelativeCenterPoint();\n  const accountForStroke = !object['isStrokeAccountedForInDimensions']();\n  const strokeUniformVector =\n    strokeUniform && accountForStroke\n      ? sendVectorToPlane(\n          new Point(strokeWidth, strokeWidth),\n          undefined,\n          destinationGroup.calcTransformMatrix()\n        )\n      : ZERO;\n  const scalingStrokeWidth =\n    !strokeUniform && accountForStroke ? strokeWidth : 0;\n  const sizeVector = sizeAfterTransform(\n    width + scalingStrokeWidth,\n    height + scalingStrokeWidth,\n    multiplyTransformMatrixArray([t, object.calcOwnMatrix()], true)\n  )\n    .add(strokeUniformVector)\n    .scalarDivide(2);\n  return [objectCenter.subtract(sizeVector), objectCenter.add(sizeVector)];\n};\n","import { Point } from '../../Point';\nimport type { FabricObject } from '../../shapes/Object/FabricObject';\nimport { makeBoundingBoxFromPoints } from '../../util/misc/boundingBoxFromPoints';\nimport {\n  LAYOUT_TYPE_INITIALIZATION,\n  LAYOUT_TYPE_IMPERATIVE,\n} from '../constants';\nimport type {\n  InitializationLayoutContext,\n  LayoutStrategyResult,\n  StrictLayoutContext,\n} from '../types';\nimport { getObjectBounds } from './utils';\n\n/**\n * Exposes a main public method {@link calcLayoutResult} that is used by the `LayoutManager` to perform layout.\n * Returning `undefined` signals the `LayoutManager` to skip the layout.\n *\n * In charge of calculating the bounding box of the passed objects.\n */\nexport abstract class LayoutStrategy {\n  /**\n   * override by subclass for persistence (TS does not support `static abstract`)\n   */\n  static type = 'strategy';\n\n  /**\n   * Used by the `LayoutManager` to perform layout\n   * @TODO/fix: if this method is calcResult, should calc unconditionally.\n   * the condition to not calc should be evaluated by the layoutManager.\n   * @returns layout result **OR** `undefined` to skip layout\n   */\n  public calcLayoutResult(\n    context: StrictLayoutContext,\n    objects: FabricObject[]\n  ): LayoutStrategyResult | undefined {\n    if (this.shouldPerformLayout(context)) {\n      return this.calcBoundingBox(objects, context);\n    }\n  }\n\n  shouldPerformLayout(context: StrictLayoutContext) {\n    return (\n      context.type === LAYOUT_TYPE_INITIALIZATION ||\n      context.type === LAYOUT_TYPE_IMPERATIVE ||\n      (!!context.prevStrategy && context.strategy !== context.prevStrategy)\n    );\n  }\n\n  shouldLayoutClipPath({ type, target: { clipPath } }: StrictLayoutContext) {\n    return (\n      type !== LAYOUT_TYPE_INITIALIZATION &&\n      clipPath &&\n      !clipPath.absolutePositioned\n    );\n  }\n\n  getInitialSize(\n    context: StrictLayoutContext & InitializationLayoutContext,\n    result: Pick<LayoutStrategyResult, 'center' | 'size'>\n  ) {\n    return result.size;\n  }\n\n  /**\n   * Override this method to customize layout.\n   */\n  calcBoundingBox(\n    objects: FabricObject[],\n    context: StrictLayoutContext\n  ): LayoutStrategyResult | undefined {\n    if (context.type === LAYOUT_TYPE_IMPERATIVE && context.overrides) {\n      return context.overrides;\n    }\n    if (objects.length === 0) {\n      return;\n    }\n    const { target } = context;\n    const { left, top, width, height } = makeBoundingBoxFromPoints(\n      objects\n        .map((object) => getObjectBounds(target, object))\n        .reduce<Point[]>((coords, curr) => coords.concat(curr), [])\n    );\n    const bboxSize = new Point(width, height);\n    const bboxLeftTop = new Point(left, top);\n    const bboxCenter = bboxLeftTop.add(bboxSize.scalarDivide(2));\n\n    if (context.type === LAYOUT_TYPE_INITIALIZATION) {\n      const actualSize = this.getInitialSize(context, {\n        size: bboxSize,\n        center: bboxCenter,\n      });\n      return {\n        // in `initialization` we do not account for target's transformation matrix\n        center: bboxCenter,\n        // TODO: investigate if this is still necessary\n        relativeCorrection: new Point(0, 0),\n        size: actualSize,\n      };\n    } else {\n      //  we send `relativeCenter` up to group's containing plane\n      const center = bboxCenter.transform(target.calcOwnMatrix());\n      return {\n        center,\n        size: bboxSize,\n      };\n    }\n  }\n}\n","import type { StrictLayoutContext } from '../types';\nimport { LayoutStrategy } from './LayoutStrategy';\nimport { classRegistry } from '../../ClassRegistry';\n\n/**\n * Layout will adjust the bounding box to fit target's objects.\n */\nexport class FitContentLayout extends LayoutStrategy {\n  static readonly type = 'fit-content';\n\n  /**\n   * @override layout on all triggers\n   * Override at will\n   */\n  // eslint-disable-next-line @typescript-eslint/no-unused-vars\n  shouldPerformLayout(context: StrictLayoutContext) {\n    return true;\n  }\n}\n\nclassRegistry.setClass(FitContentLayout);\n","import { Point } from '../Point';\nimport { CENTER, iMatrix } from '../constants';\nimport type { Group } from '../shapes/Group';\nimport type { FabricObject } from '../shapes/Object/FabricObject';\nimport { invertTransform } from '../util/misc/matrix';\nimport { resolveOrigin } from '../util/misc/resolveOrigin';\nimport { FitContentLayout } from './LayoutStrategies/FitContentLayout';\nimport type { LayoutStrategy } from './LayoutStrategies/LayoutStrategy';\nimport {\n  LAYOUT_TYPE_INITIALIZATION,\n  LAYOUT_TYPE_ADDED,\n  LAYOUT_TYPE_REMOVED,\n  LAYOUT_TYPE_IMPERATIVE,\n  LAYOUT_TYPE_OBJECT_MODIFIED,\n  LAYOUT_TYPE_OBJECT_MODIFYING,\n} from './constants';\nimport type {\n  LayoutContext,\n  LayoutResult,\n  RegistrationContext,\n  StrictLayoutContext,\n} from './types';\nimport { classRegistry } from '../ClassRegistry';\nimport type { TModificationEvents } from '../EventTypeDefs';\n\nconst LAYOUT_MANAGER = 'layoutManager';\n\nexport type SerializedLayoutManager = {\n  type: string;\n  strategy: string;\n};\n\nexport class LayoutManager {\n  private declare _prevLayoutStrategy?: LayoutStrategy;\n  protected declare _subscriptions: Map<FabricObject, VoidFunction[]>;\n\n  strategy: LayoutStrategy;\n\n  constructor(strategy: LayoutStrategy = new FitContentLayout()) {\n    this.strategy = strategy;\n    this._subscriptions = new Map();\n  }\n\n  public performLayout(context: LayoutContext) {\n    const strictContext: StrictLayoutContext = {\n      bubbles: true,\n      strategy: this.strategy,\n      ...context,\n      prevStrategy: this._prevLayoutStrategy,\n      stopPropagation() {\n        this.bubbles = false;\n      },\n    };\n\n    this.onBeforeLayout(strictContext);\n\n    const layoutResult = this.getLayoutResult(strictContext);\n    if (layoutResult) {\n      this.commitLayout(strictContext, layoutResult);\n    }\n\n    this.onAfterLayout(strictContext, layoutResult);\n    this._prevLayoutStrategy = strictContext.strategy;\n  }\n\n  /**\n   * Attach handlers for events that we know will invalidate the layout when\n   * performed on child objects ( general transforms ).\n   * Returns the disposers for later unsubscribing and cleanup\n   * @param {FabricObject} object\n   * @param {RegistrationContext & Partial<StrictLayoutContext>} context\n   * @returns {VoidFunction[]} disposers remove the handlers\n   */\n  protected attachHandlers(\n    object: FabricObject,\n    context: RegistrationContext & Partial<StrictLayoutContext>\n  ): VoidFunction[] {\n    const { target } = context;\n    return (\n      [\n        'modified',\n        'moving',\n        'resizing',\n        'rotating',\n        'scaling',\n        'skewing',\n        'changed',\n        'modifyPoly',\n      ] as (TModificationEvents & 'modified')[]\n    ).map((key) =>\n      object.on(key, (e) =>\n        this.performLayout(\n          key === 'modified'\n            ? {\n                type: LAYOUT_TYPE_OBJECT_MODIFIED,\n                trigger: key,\n                e,\n                target,\n              }\n            : {\n                type: LAYOUT_TYPE_OBJECT_MODIFYING,\n                trigger: key,\n                e,\n                target,\n              }\n        )\n      )\n    );\n  }\n\n  /**\n   * Subscribe an object to transform events that will trigger a layout change on the parent\n   * This is important only for interactive groups.\n   * @param object\n   * @param context\n   */\n  protected subscribe(\n    object: FabricObject,\n    context: RegistrationContext & Partial<StrictLayoutContext>\n  ) {\n    this.unsubscribe(object, context);\n    const disposers = this.attachHandlers(object, context);\n    this._subscriptions.set(object, disposers);\n  }\n\n  /**\n   * unsubscribe object layout triggers\n   */\n  protected unsubscribe(\n    object: FabricObject,\n    context?: RegistrationContext & Partial<StrictLayoutContext>\n  ) {\n    (this._subscriptions.get(object) || []).forEach((d) => d());\n    this._subscriptions.delete(object);\n  }\n\n  unsubscribeTargets(\n    context: RegistrationContext & Partial<StrictLayoutContext>\n  ) {\n    context.targets.forEach((object) => this.unsubscribe(object, context));\n  }\n\n  subscribeTargets(\n    context: RegistrationContext & Partial<StrictLayoutContext>\n  ) {\n    context.targets.forEach((object) => this.subscribe(object, context));\n  }\n\n  protected onBeforeLayout(context: StrictLayoutContext) {\n    const { target, type } = context;\n    const { canvas } = target;\n    // handle layout triggers subscription\n    // @TODO: gate the registration when the group is interactive\n    if (type === LAYOUT_TYPE_INITIALIZATION || type === LAYOUT_TYPE_ADDED) {\n      this.subscribeTargets(context);\n    } else if (type === LAYOUT_TYPE_REMOVED) {\n      this.unsubscribeTargets(context);\n    }\n    // fire layout event (event will fire only for layouts after initialization layout)\n    target.fire('layout:before', {\n      context,\n    });\n    canvas &&\n      canvas.fire('object:layout:before', {\n        target,\n        context,\n      });\n\n    if (type === LAYOUT_TYPE_IMPERATIVE && context.deep) {\n      const { strategy: _, ...tricklingContext } = context;\n      // traverse the tree\n      target.forEachObject(\n        (object) =>\n          (object as Group).layoutManager &&\n          (object as Group).layoutManager.performLayout({\n            ...tricklingContext,\n            bubbles: false,\n            target: object as Group,\n          })\n      );\n    }\n  }\n\n  protected getLayoutResult(\n    context: StrictLayoutContext\n  ): Required<LayoutResult> | undefined {\n    const { target } = context;\n\n    const result = context.strategy.calcLayoutResult(\n      context,\n      target.getObjects()\n    );\n\n    if (!result) {\n      return;\n    }\n\n    const prevCenter =\n      context.type === LAYOUT_TYPE_INITIALIZATION\n        ? new Point()\n        : target.getRelativeCenterPoint();\n\n    const {\n      center: nextCenter,\n      correction = new Point(),\n      relativeCorrection = new Point(),\n    } = result;\n    const offset = prevCenter\n      .subtract(nextCenter)\n      .add(correction)\n      .transform(\n        // in `initialization` we do not account for target's transformation matrix\n        context.type === LAYOUT_TYPE_INITIALIZATION\n          ? iMatrix\n          : invertTransform(target.calcOwnMatrix()),\n        true\n      )\n      .add(relativeCorrection);\n\n    return {\n      result,\n      prevCenter,\n      nextCenter,\n      offset,\n    };\n  }\n\n  protected commitLayout(\n    context: StrictLayoutContext,\n    layoutResult: Required<LayoutResult>\n  ) {\n    const { target } = context;\n    const {\n      result: { size },\n      nextCenter,\n    } = layoutResult;\n    // set dimensions\n    target.set({ width: size.x, height: size.y });\n    // layout descendants\n    this.layoutObjects(context, layoutResult);\n    //  set position\n    // in `initialization` we do not account for target's transformation matrix\n    if (context.type === LAYOUT_TYPE_INITIALIZATION) {\n      // TODO: what about strokeWidth?\n      target.set({\n        left:\n          context.x ?? nextCenter.x + size.x * resolveOrigin(target.originX),\n        top: context.y ?? nextCenter.y + size.y * resolveOrigin(target.originY),\n      });\n    } else {\n      target.setPositionByOrigin(nextCenter, CENTER, CENTER);\n      // invalidate\n      target.setCoords();\n      target.set('dirty', true);\n    }\n  }\n\n  protected layoutObjects(\n    context: StrictLayoutContext,\n    layoutResult: Required<LayoutResult>\n  ) {\n    const { target } = context;\n    //  adjust objects to account for new center\n    target.forEachObject((object) => {\n      object.group === target &&\n        this.layoutObject(context, layoutResult, object);\n    });\n    // adjust clip path to account for new center\n    context.strategy.shouldLayoutClipPath(context) &&\n      this.layoutObject(context, layoutResult, target.clipPath as FabricObject);\n  }\n\n  /**\n   * @param {FabricObject} object\n   * @param {Point} offset\n   */\n  protected layoutObject(\n    context: StrictLayoutContext,\n    { offset }: Required<LayoutResult>,\n    object: FabricObject\n  ) {\n    // TODO: this is here for cache invalidation.\n    // verify if this is necessary since we have explicit\n    // cache invalidation at the end of commitLayout\n    object.set({\n      left: object.left + offset.x,\n      top: object.top + offset.y,\n    });\n  }\n\n  protected onAfterLayout(\n    context: StrictLayoutContext,\n    layoutResult?: LayoutResult\n  ) {\n    const {\n      target,\n      strategy,\n      bubbles,\n      prevStrategy: _,\n      ...bubblingContext\n    } = context;\n    const { canvas } = target;\n\n    //  fire layout event (event will fire only for layouts after initialization layout)\n    target.fire('layout:after', {\n      context,\n      result: layoutResult,\n    });\n    canvas &&\n      canvas.fire('object:layout:after', {\n        context,\n        result: layoutResult,\n        target,\n      });\n\n    //  bubble\n    const parent = target.parent;\n    if (bubbles && parent?.layoutManager) {\n      //  add target to context#path\n      (bubblingContext.path || (bubblingContext.path = [])).push(target);\n      //  all parents should invalidate their layout\n      parent.layoutManager.performLayout({\n        ...bubblingContext,\n        target: parent,\n      });\n    }\n    target.set('dirty', true);\n  }\n\n  dispose() {\n    this._subscriptions.forEach((disposers) => disposers.forEach((d) => d()));\n    this._subscriptions.clear();\n  }\n\n  toObject() {\n    return {\n      type: LAYOUT_MANAGER,\n      strategy: (this.strategy.constructor as typeof LayoutStrategy).type,\n    };\n  }\n\n  toJSON() {\n    return this.toObject();\n  }\n}\n\nclassRegistry.setClass(LayoutManager, LAYOUT_MANAGER);\n","import type { CollectionEvents, ObjectEvents } from '../EventTypeDefs';\nimport { createCollectionMixin } from '../Collection';\nimport type {\n  TClassProperties,\n  TSVGReviver,\n  TOptions,\n  Abortable,\n} from '../typedefs';\nimport {\n  invertTransform,\n  multiplyTransformMatrices,\n} from '../util/misc/matrix';\nimport {\n  enlivenObjectEnlivables,\n  enlivenObjects,\n} from '../util/misc/objectEnlive';\nimport { applyTransformToObject } from '../util/misc/objectTransforms';\nimport { FabricObject } from './Object/FabricObject';\nimport { Rect } from './Rect';\nimport { classRegistry } from '../ClassRegistry';\nimport type { FabricObjectProps, SerializedObjectProps } from './Object/types';\nimport { log } from '../util/internals/console';\nimport type {\n  ImperativeLayoutOptions,\n  LayoutBeforeEvent,\n  LayoutAfterEvent,\n} from '../LayoutManager/types';\nimport { LayoutManager } from '../LayoutManager/LayoutManager';\nimport {\n  LAYOUT_TYPE_ADDED,\n  LAYOUT_TYPE_IMPERATIVE,\n  LAYOUT_TYPE_INITIALIZATION,\n  LAYOUT_TYPE_REMOVED,\n} from '../LayoutManager/constants';\nimport type { SerializedLayoutManager } from '../LayoutManager/LayoutManager';\nimport type { FitContentLayout } from '../LayoutManager';\n\n/**\n * This class handles the specific case of creating a group using {@link Group#fromObject} and is not meant to be used in any other case.\n * We could have used a boolean in the constructor, as we did previously, but we think the boolean\n * would stay in the group's constructor interface and create confusion, therefore it was removed.\n * This layout manager doesn't do anything and therefore keeps the exact layout the group had when {@link Group#toObject} was called.\n */\nclass NoopLayoutManager extends LayoutManager {\n  // eslint-disable-next-line @typescript-eslint/no-empty-function\n  performLayout() { }\n}\n\nexport interface GroupEvents extends ObjectEvents, CollectionEvents {\n  'layout:before': LayoutBeforeEvent;\n  'layout:after': LayoutAfterEvent;\n}\n\nexport interface GroupOwnProps {\n  subTargetCheck: boolean;\n  interactive: boolean;\n}\n\nexport interface SerializedGroupProps\n  extends SerializedObjectProps,\n  GroupOwnProps {\n  objects: SerializedObjectProps[];\n  layoutManager: SerializedLayoutManager;\n}\n\nexport interface GroupProps extends FabricObjectProps, GroupOwnProps {\n  layoutManager: LayoutManager;\n}\n\nexport const groupDefaultValues: Partial<TClassProperties<Group>> = {\n  strokeWidth: 0,\n  subTargetCheck: false,\n  interactive: false,\n};\n\n/**\n * @fires object:added\n * @fires object:removed\n * @fires layout:before\n * @fires layout:after\n */\nexport class Group\n  extends createCollectionMixin(\n    FabricObject<GroupProps, SerializedGroupProps, GroupEvents>\n  )\n  implements GroupProps {\n  /**\n   * Used to optimize performance\n   * set to `false` if you don't need contained objects to be targets of events\n   * @default\n   * @type boolean\n   */\n  declare subTargetCheck: boolean;\n\n  /**\n   * Used to allow targeting of object inside groups.\n   * set to true if you want to select an object inside a group.\\\n   * **REQUIRES** `subTargetCheck` set to true\n   * This will be not removed but slowly replaced with a method setInteractive\n   * that will take care of enabling subTargetCheck and necessary object events.\n   * There is too much attached to group interactivity to just be evaluated by a\n   * boolean in the code\n   * @default\n   * @deprecated\n   * @type boolean\n   */\n  declare interactive: boolean;\n\n  declare layoutManager: LayoutManager;\n\n  /**\n   * Used internally to optimize performance\n   * Once an object is selected, instance is rendered without the selected object.\n   * This way instance is cached only once for the entire interaction with the selected object.\n   * @private\n   */\n  protected _activeObjects: FabricObject[] = [];\n\n  static type = 'Group';\n\n  static ownDefaults: Record<string, any> = groupDefaultValues;\n  private __objectSelectionTracker: (ev: ObjectEvents['selected']) => void;\n  private __objectSelectionDisposer: (ev: ObjectEvents['deselected']) => void;\n\n  static getDefaults(): Record<string, any> {\n    return {\n      ...super.getDefaults(),\n      ...Group.ownDefaults,\n    };\n  }\n\n  /**\n   * Constructor\n   *\n   * @param {FabricObject[]} [objects] instance objects\n   * @param {Object} [options] Options object\n   */\n  constructor(objects: FabricObject[] = [], options: Partial<GroupProps> = {}) {\n    // @ts-expect-error options error\n    super(options);\n    this._objects = [...objects]; // Avoid unwanted mutations of Collection to affect the caller\n\n    this.__objectSelectionTracker = this.__objectSelectionMonitor.bind(\n      this,\n      true\n    );\n    this.__objectSelectionDisposer = this.__objectSelectionMonitor.bind(\n      this,\n      false\n    );\n\n    this.forEachObject((object) => {\n      this.enterGroup(object, false);\n    });\n\n    // perform initial layout\n    this.layoutManager = options.layoutManager || new LayoutManager();\n    this.layoutManager.performLayout({\n      type: LAYOUT_TYPE_INITIALIZATION,\n      target: this,\n      targets: [...objects],\n      // @TODO remove this concept from the layout manager.\n      // Layout manager will calculate the correct position,\n      // group options can override it later.\n      x: options.left,\n      y: options.top,\n    });\n  }\n\n  /**\n   * Checks if object can enter group and logs relevant warnings\n   * @private\n   * @param {FabricObject} object\n   * @returns\n   */\n  canEnterGroup(object: FabricObject) {\n    if (object === this || this.isDescendantOf(object)) {\n      //  prevent circular object tree\n      log(\n        'error',\n        'Group: circular object trees are not supported, this call has no effect'\n      );\n      return false;\n    } else if (this._objects.indexOf(object) !== -1) {\n      // is already in the objects array\n      log(\n        'error',\n        'Group: duplicate objects are not supported inside group, this call has no effect'\n      );\n      return false;\n    }\n    return true;\n  }\n\n  /**\n   * Override this method to enhance performance (for groups with a lot of objects).\n   * If Overriding, be sure not pass illegal objects to group - it will break your app.\n   * @private\n   */\n  protected _filterObjectsBeforeEnteringGroup(objects: FabricObject[]) {\n    return objects.filter((object, index, array) => {\n      // can enter AND is the first occurrence of the object in the passed args (to prevent adding duplicates)\n      return this.canEnterGroup(object) && array.indexOf(object) === index;\n    });\n  }\n\n  /**\n   * Add objects\n   * @param {...FabricObject[]} objects\n   */\n  add(...objects: FabricObject[]) {\n    const allowedObjects = this._filterObjectsBeforeEnteringGroup(objects);\n    const size = super.add(...allowedObjects);\n    this._onAfterObjectsChange(LAYOUT_TYPE_ADDED, allowedObjects);\n    return size;\n  }\n\n  /**\n   * Inserts an object into collection at specified index\n   * @param {FabricObject[]} objects Object to insert\n   * @param {Number} index Index to insert object at\n   */\n  insertAt(index: number, ...objects: FabricObject[]) {\n    const allowedObjects = this._filterObjectsBeforeEnteringGroup(objects);\n    const size = super.insertAt(index, ...allowedObjects);\n    this._onAfterObjectsChange(LAYOUT_TYPE_ADDED, allowedObjects);\n    return size;\n  }\n\n  /**\n   * Remove objects\n   * @param {...FabricObject[]} objects\n   * @returns {FabricObject[]} removed objects\n   */\n  remove(...objects: FabricObject[]) {\n    const removed = super.remove(...objects);\n    this._onAfterObjectsChange(LAYOUT_TYPE_REMOVED, removed);\n    return removed;\n  }\n\n  _onObjectAdded(object: FabricObject) {\n    this.enterGroup(object, true);\n    this.fire('object:added', { target: object });\n    object.fire('added', { target: this });\n  }\n\n  /**\n   * @private\n   * @param {FabricObject} object\n   * @param {boolean} [removeParentTransform] true if object should exit group without applying group's transform to it\n   */\n  _onObjectRemoved(object: FabricObject, removeParentTransform?: boolean) {\n    this.exitGroup(object, removeParentTransform);\n    this.fire('object:removed', { target: object });\n    object.fire('removed', { target: this });\n  }\n\n  /**\n   * @private\n   * @param {'added'|'removed'} type\n   * @param {FabricObject[]} targets\n   */\n  _onAfterObjectsChange(type: 'added' | 'removed', targets: FabricObject[]) {\n    this.layoutManager.performLayout({\n      type,\n      targets,\n      target: this,\n    });\n  }\n\n  _onStackOrderChanged() {\n    this._set('dirty', true);\n  }\n\n  /**\n   * @private\n   * @param {string} key\n   * @param {*} value\n   */\n  _set(key: string, value: any) {\n    const prev = this[key as keyof this];\n    super._set(key, value);\n    if (key === 'canvas' && prev !== value) {\n      (this._objects || []).forEach((object) => {\n        object._set(key, value);\n      });\n    }\n    return this;\n  }\n\n  /**\n   * @private\n   */\n  _shouldSetNestedCoords() {\n    return this.subTargetCheck;\n  }\n\n  /**\n   * Remove all objects\n   * @returns {FabricObject[]} removed objects\n   */\n  removeAll() {\n    this._activeObjects = [];\n    return this.remove(...this._objects);\n  }\n\n  /**\n   * keeps track of the selected objects\n   * @private\n   */\n  __objectSelectionMonitor<T extends boolean>(\n    selected: T,\n    { target: object }: ObjectEvents[T extends true ? 'selected' : 'deselected']\n  ) {\n    const activeObjects = this._activeObjects;\n    if (selected) {\n      activeObjects.push(object);\n      this._set('dirty', true);\n    } else if (activeObjects.length > 0) {\n      const index = activeObjects.indexOf(object);\n      if (index > -1) {\n        activeObjects.splice(index, 1);\n        this._set('dirty', true);\n      }\n    }\n  }\n\n  /**\n   * @private\n   * @param {boolean} watch\n   * @param {FabricObject} object\n   */\n  _watchObject(watch: boolean, object: FabricObject) {\n    //  make sure we listen only once\n    watch && this._watchObject(false, object);\n    if (watch) {\n      object.on('selected', this.__objectSelectionTracker);\n      object.on('deselected', this.__objectSelectionDisposer);\n    } else {\n      object.off('selected', this.__objectSelectionTracker);\n      object.off('deselected', this.__objectSelectionDisposer);\n    }\n  }\n\n  /**\n   * @private\n   * @param {FabricObject} object\n   * @param {boolean} [removeParentTransform] true if object is in canvas coordinate plane\n   */\n  enterGroup(object: FabricObject, removeParentTransform?: boolean) {\n    object.group && object.group.remove(object);\n    object._set('parent', this);\n    this._enterGroup(object, removeParentTransform);\n  }\n\n  /**\n   * @private\n   * @param {FabricObject} object\n   * @param {boolean} [removeParentTransform] true if object is in canvas coordinate plane\n   */\n  _enterGroup(object: FabricObject, removeParentTransform?: boolean) {\n    if (removeParentTransform) {\n      // can this be converted to utils (sendObjectToPlane)?\n      applyTransformToObject(\n        object,\n        multiplyTransformMatrices(\n          invertTransform(this.calcTransformMatrix()),\n          object.calcTransformMatrix()\n        )\n      );\n    }\n    this._shouldSetNestedCoords() && object.setCoords();\n    object._set('group', this);\n    object._set('canvas', this.canvas);\n    this._watchObject(true, object);\n    const activeObject =\n      this.canvas &&\n      this.canvas.getActiveObject &&\n      this.canvas.getActiveObject();\n    // if we are adding the activeObject in a group\n    if (\n      activeObject &&\n      (activeObject === object || object.isDescendantOf(activeObject))\n    ) {\n      this._activeObjects.push(object);\n    }\n  }\n\n  /**\n   * @private\n   * @param {FabricObject} object\n   * @param {boolean} [removeParentTransform] true if object should exit group without applying group's transform to it\n   */\n  exitGroup(object: FabricObject, removeParentTransform?: boolean) {\n    this._exitGroup(object, removeParentTransform);\n    object._set('parent', undefined);\n    object._set('canvas', undefined);\n  }\n\n  /**\n   * Executes the inner fabric logic of exiting a group.\n   * - Stop watching the object\n   * - Remove the object from the optimization map this._activeObjects\n   * - unset the group property of the object\n   * @protected\n   * @param {FabricObject} object\n   * @param {boolean} [removeParentTransform] true if object should exit group without applying group's transform to it\n   */\n  _exitGroup(object: FabricObject, removeParentTransform?: boolean) {\n    object._set('group', undefined);\n    if (!removeParentTransform) {\n      applyTransformToObject(\n        object,\n        multiplyTransformMatrices(\n          this.calcTransformMatrix(),\n          object.calcTransformMatrix()\n        )\n      );\n      object.setCoords();\n    }\n    this._watchObject(false, object);\n    const index =\n      this._activeObjects.length > 0 ? this._activeObjects.indexOf(object) : -1;\n    if (index > -1) {\n      this._activeObjects.splice(index, 1);\n    }\n  }\n\n  /**\n   * Decide if the object should cache or not. Create its own cache level\n   * needsItsOwnCache should be used when the object drawing method requires\n   * a cache step. None of the fabric classes requires it.\n   * Generally you do not cache objects in groups because the group is already cached.\n   * @return {Boolean}\n   */\n  shouldCache() {\n    const ownCache = FabricObject.prototype.shouldCache.call(this);\n    if (ownCache) {\n      for (let i = 0; i < this._objects.length; i++) {\n        if (this._objects[i].willDrawShadow()) {\n          this.ownCaching = false;\n          return false;\n        }\n      }\n    }\n    return ownCache;\n  }\n\n  /**\n   * Check if this object or a child object will cast a shadow\n   * @return {Boolean}\n   */\n  willDrawShadow() {\n    if (super.willDrawShadow()) {\n      return true;\n    }\n    for (let i = 0; i < this._objects.length; i++) {\n      if (this._objects[i].willDrawShadow()) {\n        return true;\n      }\n    }\n    return false;\n  }\n\n  /**\n   * Check if instance or its group are caching, recursively up\n   * @return {Boolean}\n   */\n  isOnACache(): boolean {\n    return this.ownCaching || (!!this.parent && this.parent.isOnACache());\n  }\n\n  /**\n   * Execute the drawing operation for an object on a specified context\n   * @param {CanvasRenderingContext2D} ctx Context to render on\n   */\n  drawObject(ctx: CanvasRenderingContext2D) {\n    this._renderBackground(ctx);\n    for (let i = 0; i < this._objects.length; i++) {\n      // TODO: handle rendering edge case somehow\n      if (\n        this.canvas?.preserveObjectStacking &&\n        this._objects[i].group !== this\n      ) {\n        ctx.save();\n        ctx.transform(...invertTransform(this.calcTransformMatrix()));\n        this._objects[i].render(ctx);\n        ctx.restore();\n      } else if (this._objects[i].group === this) {\n        this._objects[i].render(ctx);\n      }\n    }\n    this._drawClipPath(ctx, this.clipPath);\n  }\n\n  /**\n   * @override\n   * @return {Boolean}\n   */\n  setCoords() {\n    super.setCoords();\n    this._shouldSetNestedCoords() &&\n      this.forEachObject((object) => object.setCoords());\n  }\n\n  triggerLayout(options: ImperativeLayoutOptions = {}) {\n    this.layoutManager.performLayout({\n      target: this,\n      type: LAYOUT_TYPE_IMPERATIVE,\n      ...options,\n    });\n  }\n\n  /**\n   * Renders instance on a given context\n   * @param {CanvasRenderingContext2D} ctx context to render instance on\n   */\n  render(ctx: CanvasRenderingContext2D) {\n    this._transformDone = true;\n    super.render(ctx);\n    this._transformDone = false;\n  }\n\n  /**\n   *\n   * @private\n   * @param {'toObject'|'toDatalessObject'} [method]\n   * @param {string[]} [propertiesToInclude] Any properties that you might want to additionally include in the output\n   * @returns {FabricObject[]} serialized objects\n   */\n  __serializeObjects(\n    method: 'toObject' | 'toDatalessObject',\n    propertiesToInclude?: string[]\n  ) {\n    const _includeDefaultValues = this.includeDefaultValues;\n    return this._objects\n      .filter(function (obj) {\n        return !obj.excludeFromExport;\n      })\n      .map(function (obj) {\n        const originalDefaults = obj.includeDefaultValues;\n        obj.includeDefaultValues = _includeDefaultValues;\n        const data = obj[method || 'toObject'](propertiesToInclude);\n        obj.includeDefaultValues = originalDefaults;\n        // delete data.version;\n        return data;\n      });\n  }\n\n  /**\n   * Returns object representation of an instance\n   * @param {string[]} [propertiesToInclude] Any properties that you might want to additionally include in the output\n   * @return {Object} object representation of an instance\n   */\n  toObject<\n    T extends Omit<\n      GroupProps & TClassProperties<this>,\n      keyof SerializedGroupProps\n    >,\n    K extends keyof T = never\n  >(propertiesToInclude: K[] = []): Pick<T, K> & SerializedGroupProps {\n    const layoutManager = this.layoutManager.toObject();\n\n    return {\n      ...super.toObject([\n        'subTargetCheck',\n        'interactive',\n        ...propertiesToInclude,\n      ]),\n      ...(layoutManager.strategy !== 'fit-content' || this.includeDefaultValues\n        ? { layoutManager }\n        : {}),\n      objects: this.__serializeObjects(\n        'toObject',\n        propertiesToInclude as string[]\n      ),\n    };\n  }\n\n  toString() {\n    return `#<Group: (${this.complexity()})>`;\n  }\n\n  dispose() {\n    this.layoutManager.unsubscribeTargets({\n      targets: this.getObjects(),\n      target: this,\n    });\n    this._activeObjects = [];\n    this.forEachObject((object) => {\n      this._watchObject(false, object);\n      object.dispose();\n    });\n    super.dispose();\n  }\n\n  /**\n   * @private\n   */\n  _createSVGBgRect(reviver?: TSVGReviver) {\n    if (!this.backgroundColor) {\n      return '';\n    }\n    const fillStroke = Rect.prototype._toSVG.call(this);\n    const commons = fillStroke.indexOf('COMMON_PARTS');\n    fillStroke[commons] = 'for=\"group\" ';\n    const markup = fillStroke.join('');\n    return reviver ? reviver(markup) : markup;\n  }\n\n  /**\n   * Returns svg representation of an instance\n   * @param {TSVGReviver} [reviver] Method for further parsing of svg representation.\n   * @return {String} svg representation of an instance\n   */\n  _toSVG(reviver?: TSVGReviver) {\n    const svgString = ['<g ', 'COMMON_PARTS', ' >\\n'];\n    const bg = this._createSVGBgRect(reviver);\n    bg && svgString.push('\\t\\t', bg);\n    for (let i = 0; i < this._objects.length; i++) {\n      svgString.push('\\t\\t', this._objects[i].toSVG(reviver));\n    }\n    svgString.push('</g>\\n');\n    return svgString;\n  }\n\n  /**\n   * Returns styles-string for svg-export, specific version for group\n   * @return {String}\n   */\n  getSvgStyles(): string {\n    const opacity =\n      typeof this.opacity !== 'undefined' && this.opacity !== 1\n        ? `opacity: ${this.opacity};`\n        : '',\n      visibility = this.visible ? '' : ' visibility: hidden;';\n    return [opacity, this.getSvgFilter(), visibility].join('');\n  }\n\n  /**\n   * Returns svg clipPath representation of an instance\n   * @param {Function} [reviver] Method for further parsing of svg representation.\n   * @return {String} svg representation of an instance\n   */\n  toClipPathSVG(reviver?: TSVGReviver): string {\n    const svgString = [];\n    const bg = this._createSVGBgRect(reviver);\n    bg && svgString.push('\\t', bg);\n    for (let i = 0; i < this._objects.length; i++) {\n      svgString.push('\\t', this._objects[i].toClipPathSVG(reviver));\n    }\n    return this._createBaseClipPathSVGMarkup(svgString, {\n      reviver,\n    });\n  }\n\n  /**\n   * @todo support loading from svg\n   * @private\n   * @static\n   * @memberOf Group\n   * @param {Object} object Object to create a group from\n   * @returns {Promise<Group>}\n   */\n  static fromObject<T extends TOptions<SerializedGroupProps>>(\n    { type, objects = [], layoutManager, ...options }: T,\n    abortable?: Abortable\n  ) {\n    return Promise.all([\n      enlivenObjects<FabricObject>(objects, abortable),\n      enlivenObjectEnlivables(options, abortable),\n    ]).then(([objects, hydratedOptions]) => {\n      const group = new this(objects, {\n        ...options,\n        ...hydratedOptions,\n        layoutManager: new NoopLayoutManager(),\n      });\n      if (layoutManager) {\n        const layoutClass = classRegistry.getClass<typeof LayoutManager>(\n          layoutManager.type\n        );\n        const strategyClass = classRegistry.getClass<typeof FitContentLayout>(\n          layoutManager.strategy\n        );\n        group.layoutManager = new layoutClass(new strategyClass());\n      } else {\n        group.layoutManager = new LayoutManager();\n      }\n      group.layoutManager.subscribeTargets({\n        type: LAYOUT_TYPE_INITIALIZATION,\n        target: group,\n        targets: group.getObjects(),\n      });\n      group.setCoords();\n      return group;\n    });\n  }\n}\n\nclassRegistry.setClass(Group);\n","import type { GroupProps } from '../../shapes/Group';\nimport { Group } from '../../shapes/Group';\nimport type { FabricObject } from '../../shapes/Object/FabricObject';\n\n/**\n * TODO experiment with different layout manager and svg results ( fixed fit content )\n * Groups SVG elements (usually those retrieved from SVG document)\n * @static\n * @param {FabricObject[]} elements FabricObject(s) parsed from svg, to group\n * @return {FabricObject | Group}\n */\nexport const groupSVGElements = (\n  elements: FabricObject[],\n  options?: Partial<GroupProps>\n) => {\n  if (elements && elements.length === 1) {\n    return elements[0];\n  }\n  return new Group(elements, options);\n};\n","import type { TSize } from '../../typedefs';\n\n/**\n * Finds the scale for the object source to fit inside the object destination,\n * keeping aspect ratio intact.\n * respect the total allowed area for the cache.\n * @param {TSize} source natural unscaled size of the object\n * @param {TSize} destination natural unscaled size of the object\n * @return {Number} scale factor to apply to source to fit into destination\n */\nexport const findScaleToFit = (source: TSize, destination: TSize) =>\n  Math.min(\n    destination.width / source.width,\n    destination.height / source.height\n  );\n\n/**\n * Finds the scale for the object source to cover entirely the object destination,\n * keeping aspect ratio intact.\n * respect the total allowed area for the cache.\n * @param {TSize} source natural unscaled size of the object\n * @param {TSize} destination natural unscaled size of the object\n * @return {Number} scale factor to apply to source to cover destination\n */\nexport const findScaleToCover = (source: TSize, destination: TSize) =>\n  Math.max(\n    destination.width / source.width,\n    destination.height / source.height\n  );\n","import { reNum } from '../../parser/constants';\n\n/**\n * p for param\n * using \"bad naming\" here because it makes the regex much easier to read\n */\nconst p = `(${reNum})`;\n\nconst reMoveToCommand = `(M) (?:${p} ${p} ?)+`;\n\nconst reLineCommand = `(L) (?:${p} ${p} ?)+`;\n\nconst reHorizontalLineCommand = `(H) (?:${p} ?)+`;\n\nconst reVerticalLineCommand = `(V) (?:${p} ?)+`;\n\nconst reClosePathCommand = String.raw`(Z)\\s*`;\n\nconst reCubicCurveCommand = `(C) (?:${p} ${p} ${p} ${p} ${p} ${p} ?)+`;\n\nconst reCubicCurveShortcutCommand = `(S) (?:${p} ${p} ${p} ${p} ?)+`;\n\nconst reQuadraticCurveCommand = `(Q) (?:${p} ${p} ${p} ${p} ?)+`;\n\nconst reQuadraticCurveShortcutCommand = `(T) (?:${p} ${p} ?)+`;\n\nconst reArcCommand = `(A) (?:${p} ${p} ${p} ([01]) ?([01]) ${p} ${p} ?)+`;\n\nexport const rePathCommand =\n  `(?:(?:${reMoveToCommand})` +\n  `|(?:${reLineCommand})` +\n  `|(?:${reHorizontalLineCommand})` +\n  `|(?:${reVerticalLineCommand})` +\n  `|(?:${reClosePathCommand})` +\n  `|(?:${reCubicCurveCommand})` +\n  `|(?:${reCubicCurveShortcutCommand})` +\n  `|(?:${reQuadraticCurveCommand})` +\n  `|(?:${reQuadraticCurveShortcutCommand})` +\n  `|(?:${reArcCommand}))`;\n","import { cache } from '../../cache';\nimport { config } from '../../config';\nimport { halfPI, PiBy180 } from '../../constants';\nimport type { TMat2D, TRadian, TRectBounds } from '../../typedefs';\nimport { cos } from '../misc/cos';\nimport { multiplyTransformMatrices, transformPoint } from '../misc/matrix';\nimport { sin } from '../misc/sin';\nimport { toFixed } from '../misc/toFixed';\nimport type {\n  TCurveInfo,\n  TComplexPathData,\n  TParsedAbsoluteCubicCurveCommand,\n  TParsedCubicCurveCommand,\n  TPathSegmentInfo,\n  TPointAngle,\n  TSimpleParsedCommand,\n  TSimplePathData,\n  TPathSegmentCommandInfo,\n  TComplexParsedCommand,\n  TPathSegmentInfoCommon,\n  TEndPathInfo,\n  TParsedArcCommand,\n} from './typedefs';\nimport type { XY } from '../../Point';\nimport { Point } from '../../Point';\nimport { rePathCommand } from './regex';\nimport { cleanupSvgAttribute } from '../internals/cleanupSvgAttribute';\n\n/**\n * Commands that may be repeated\n */\nconst repeatedCommands: Record<string, string | undefined> = {\n  m: 'l',\n  M: 'L',\n};\n\n/**\n * Convert an arc of a rotated ellipse to a Bezier Curve\n * @param {TRadian} theta1 start of the arc\n * @param {TRadian} theta2 end of the arc\n * @param cosTh cosine of the angle of rotation\n * @param sinTh sine of the angle of rotation\n * @param rx x-axis radius (before rotation)\n * @param ry y-axis radius (before rotation)\n * @param cx1 center x of the ellipse\n * @param cy1 center y of the ellipse\n * @param mT\n * @param fromX starting point of arc x\n * @param fromY starting point of arc y\n */\nconst segmentToBezier = (\n  theta1: TRadian,\n  theta2: TRadian,\n  cosTh: number,\n  sinTh: number,\n  rx: number,\n  ry: number,\n  cx1: number,\n  cy1: number,\n  mT: number,\n  fromX: number,\n  fromY: number\n): TParsedCubicCurveCommand => {\n  const costh1 = cos(theta1),\n    sinth1 = sin(theta1),\n    costh2 = cos(theta2),\n    sinth2 = sin(theta2),\n    toX = cosTh * rx * costh2 - sinTh * ry * sinth2 + cx1,\n    toY = sinTh * rx * costh2 + cosTh * ry * sinth2 + cy1,\n    cp1X = fromX + mT * (-cosTh * rx * sinth1 - sinTh * ry * costh1),\n    cp1Y = fromY + mT * (-sinTh * rx * sinth1 + cosTh * ry * costh1),\n    cp2X = toX + mT * (cosTh * rx * sinth2 + sinTh * ry * costh2),\n    cp2Y = toY + mT * (sinTh * rx * sinth2 - cosTh * ry * costh2);\n\n  return ['C', cp1X, cp1Y, cp2X, cp2Y, toX, toY];\n};\n\n/**\n * Adapted from {@link http://dxr.mozilla.org/mozilla-central/source/dom/svg/SVGPathDataParser.cpp}\n * by Andrea Bogazzi code is under MPL. if you don't have a copy of the license you can take it here\n * http://mozilla.org/MPL/2.0/\n * @param toX\n * @param toY\n * @param rx\n * @param ry\n * @param {number} large 0 or 1 flag\n * @param {number} sweep 0 or 1 flag\n * @param rotateX\n */\nconst arcToSegments = (\n  toX: number,\n  toY: number,\n  rx: number,\n  ry: number,\n  large: number,\n  sweep: number,\n  rotateX: TRadian\n): TParsedAbsoluteCubicCurveCommand[] => {\n  if (rx === 0 || ry === 0) {\n    return [];\n  }\n  let fromX = 0,\n    fromY = 0,\n    root = 0;\n  const PI = Math.PI,\n    theta = rotateX * PiBy180,\n    sinTheta = sin(theta),\n    cosTh = cos(theta),\n    px = 0.5 * (-cosTh * toX - sinTheta * toY),\n    py = 0.5 * (-cosTh * toY + sinTheta * toX),\n    rx2 = rx ** 2,\n    ry2 = ry ** 2,\n    py2 = py ** 2,\n    px2 = px ** 2,\n    pl = rx2 * ry2 - rx2 * py2 - ry2 * px2;\n  let _rx = Math.abs(rx);\n  let _ry = Math.abs(ry);\n\n  if (pl < 0) {\n    const s = Math.sqrt(1 - pl / (rx2 * ry2));\n    _rx *= s;\n    _ry *= s;\n  } else {\n    root =\n      (large === sweep ? -1.0 : 1.0) * Math.sqrt(pl / (rx2 * py2 + ry2 * px2));\n  }\n\n  const cx = (root * _rx * py) / _ry,\n    cy = (-root * _ry * px) / _rx,\n    cx1 = cosTh * cx - sinTheta * cy + toX * 0.5,\n    cy1 = sinTheta * cx + cosTh * cy + toY * 0.5;\n  let mTheta = calcVectorAngle(1, 0, (px - cx) / _rx, (py - cy) / _ry);\n  let dtheta = calcVectorAngle(\n    (px - cx) / _rx,\n    (py - cy) / _ry,\n    (-px - cx) / _rx,\n    (-py - cy) / _ry\n  );\n\n  if (sweep === 0 && dtheta > 0) {\n    dtheta -= 2 * PI;\n  } else if (sweep === 1 && dtheta < 0) {\n    dtheta += 2 * PI;\n  }\n\n  // Convert into cubic bezier segments <= 90deg\n  const segments = Math.ceil(Math.abs((dtheta / PI) * 2)),\n    result = new Array(segments),\n    mDelta = dtheta / segments,\n    mT =\n      ((8 / 3) * Math.sin(mDelta / 4) * Math.sin(mDelta / 4)) /\n      Math.sin(mDelta / 2);\n  let th3 = mTheta + mDelta;\n\n  for (let i = 0; i < segments; i++) {\n    result[i] = segmentToBezier(\n      mTheta,\n      th3,\n      cosTh,\n      sinTheta,\n      _rx,\n      _ry,\n      cx1,\n      cy1,\n      mT,\n      fromX,\n      fromY\n    );\n    fromX = result[i][5];\n    fromY = result[i][6];\n    mTheta = th3;\n    th3 += mDelta;\n  }\n  return result;\n};\n\n/**\n * @private\n * Calculate the angle between two vectors\n * @param ux u endpoint x\n * @param uy u endpoint y\n * @param vx v endpoint x\n * @param vy v endpoint y\n */\nconst calcVectorAngle = (\n  ux: number,\n  uy: number,\n  vx: number,\n  vy: number\n): TRadian => {\n  const ta = Math.atan2(uy, ux),\n    tb = Math.atan2(vy, vx);\n  if (tb >= ta) {\n    return tb - ta;\n  } else {\n    return 2 * Math.PI - (ta - tb);\n  }\n};\n\n// functions for the Cubic beizer\n// taken from: https://github.com/konvajs/konva/blob/7.0.5/src/shapes/Path.ts#L350\nconst CB1 = (t: number) => t ** 3;\nconst CB2 = (t: number) => 3 * t ** 2 * (1 - t);\nconst CB3 = (t: number) => 3 * t * (1 - t) ** 2;\nconst CB4 = (t: number) => (1 - t) ** 3;\n\n/**\n * Calculate bounding box of a cubic Bezier curve\n * Taken from http://jsbin.com/ivomiq/56/edit (no credits available)\n * TODO: can we normalize this with the starting points set at 0 and then translated the bbox?\n * @param {number} begx starting point\n * @param {number} begy\n * @param {number} cp1x first control point\n * @param {number} cp1y\n * @param {number} cp2x second control point\n * @param {number} cp2y\n * @param {number} endx end of bezier\n * @param {number} endy\n * @return {TRectBounds} the rectangular bounds\n */\nexport function getBoundsOfCurve(\n  begx: number,\n  begy: number,\n  cp1x: number,\n  cp1y: number,\n  cp2x: number,\n  cp2y: number,\n  endx: number,\n  endy: number\n): TRectBounds {\n  let argsString: string;\n  if (config.cachesBoundsOfCurve) {\n    // eslint-disable-next-line\n    argsString = [...arguments].join();\n    if (cache.boundsOfCurveCache[argsString]) {\n      return cache.boundsOfCurveCache[argsString];\n    }\n  }\n\n  const sqrt = Math.sqrt,\n    abs = Math.abs,\n    tvalues = [],\n    bounds: [[x: number, y: number], [x: number, y: number]] = [\n      [0, 0],\n      [0, 0],\n    ];\n\n  let b = 6 * begx - 12 * cp1x + 6 * cp2x;\n  let a = -3 * begx + 9 * cp1x - 9 * cp2x + 3 * endx;\n  let c = 3 * cp1x - 3 * begx;\n\n  for (let i = 0; i < 2; ++i) {\n    if (i > 0) {\n      b = 6 * begy - 12 * cp1y + 6 * cp2y;\n      a = -3 * begy + 9 * cp1y - 9 * cp2y + 3 * endy;\n      c = 3 * cp1y - 3 * begy;\n    }\n\n    if (abs(a) < 1e-12) {\n      if (abs(b) < 1e-12) {\n        continue;\n      }\n      const t = -c / b;\n      if (0 < t && t < 1) {\n        tvalues.push(t);\n      }\n      continue;\n    }\n    const b2ac = b * b - 4 * c * a;\n    if (b2ac < 0) {\n      continue;\n    }\n    const sqrtb2ac = sqrt(b2ac);\n    const t1 = (-b + sqrtb2ac) / (2 * a);\n    if (0 < t1 && t1 < 1) {\n      tvalues.push(t1);\n    }\n    const t2 = (-b - sqrtb2ac) / (2 * a);\n    if (0 < t2 && t2 < 1) {\n      tvalues.push(t2);\n    }\n  }\n\n  let j = tvalues.length;\n  const jlen = j;\n  const iterator = getPointOnCubicBezierIterator(\n    begx,\n    begy,\n    cp1x,\n    cp1y,\n    cp2x,\n    cp2y,\n    endx,\n    endy\n  );\n  while (j--) {\n    const { x, y } = iterator(tvalues[j]);\n    bounds[0][j] = x;\n    bounds[1][j] = y;\n  }\n\n  bounds[0][jlen] = begx;\n  bounds[1][jlen] = begy;\n  bounds[0][jlen + 1] = endx;\n  bounds[1][jlen + 1] = endy;\n  const result: TRectBounds = [\n    new Point(Math.min(...bounds[0]), Math.min(...bounds[1])),\n    new Point(Math.max(...bounds[0]), Math.max(...bounds[1])),\n  ];\n  if (config.cachesBoundsOfCurve) {\n    cache.boundsOfCurveCache[argsString!] = result;\n  }\n  return result;\n}\n\n/**\n * Converts arc to a bunch of cubic Bezier curves\n * @param {number} fx starting point x\n * @param {number} fy starting point y\n * @param {TParsedArcCommand} coords Arc command\n */\nexport const fromArcToBeziers = (\n  fx: number,\n  fy: number,\n  [_, rx, ry, rot, large, sweep, tx, ty]: TParsedArcCommand\n): TParsedAbsoluteCubicCurveCommand[] => {\n  const segsNorm = arcToSegments(tx - fx, ty - fy, rx, ry, large, sweep, rot);\n\n  for (let i = 0, len = segsNorm.length; i < len; i++) {\n    segsNorm[i][1] += fx;\n    segsNorm[i][2] += fy;\n    segsNorm[i][3] += fx;\n    segsNorm[i][4] += fy;\n    segsNorm[i][5] += fx;\n    segsNorm[i][6] += fy;\n  }\n  return segsNorm;\n};\n\n/**\n * This function takes a parsed SVG path and makes it simpler for fabricJS logic.\n * Simplification consist of:\n * - All commands converted to absolute (lowercase to uppercase)\n * - S converted to C\n * - T converted to Q\n * - A converted to C\n * @param {TComplexPathData} path the array of commands of a parsed SVG path for `Path`\n * @return {TSimplePathData} the simplified array of commands of a parsed SVG path for `Path`\n * TODO: figure out how to remove the type assertions in a nice way\n */\nexport const makePathSimpler = (path: TComplexPathData): TSimplePathData => {\n  // x and y represent the last point of the path, AKA the previous command point.\n  // we add them to each relative command to make it an absolute comment.\n  // we also swap the v V h H with L, because are easier to transform.\n  let x = 0,\n    y = 0;\n  // x1 and y1 represent the last point of the subpath. the subpath is started with\n  // m or M command. When a z or Z command is drawn, x and y need to be resetted to\n  // the last x1 and y1.\n  let x1 = 0,\n    y1 = 0;\n  // previous will host the letter of the previous command, to handle S and T.\n  // controlX and controlY will host the previous reflected control point\n  const destinationPath: TSimplePathData = [];\n  let previous,\n    // placeholders\n    controlX = 0,\n    controlY = 0;\n  for (const parsedCommand of path) {\n    const current: TComplexParsedCommand = [...parsedCommand];\n    let converted: TSimpleParsedCommand | undefined;\n    switch (\n      current[0] // first letter\n    ) {\n      case 'l': // lineto, relative\n        current[1] += x;\n        current[2] += y;\n      // falls through\n      case 'L':\n        x = current[1];\n        y = current[2];\n        converted = ['L', x, y];\n        break;\n      case 'h': // horizontal lineto, relative\n        current[1] += x;\n      // falls through\n      case 'H':\n        x = current[1];\n        converted = ['L', x, y];\n        break;\n      case 'v': // vertical lineto, relative\n        current[1] += y;\n      // falls through\n      case 'V':\n        y = current[1];\n        converted = ['L', x, y];\n        break;\n      case 'm': // moveTo, relative\n        current[1] += x;\n        current[2] += y;\n      // falls through\n      case 'M':\n        x = current[1];\n        y = current[2];\n        x1 = current[1];\n        y1 = current[2];\n        converted = ['M', x, y];\n        break;\n      case 'c': // bezierCurveTo, relative\n        current[1] += x;\n        current[2] += y;\n        current[3] += x;\n        current[4] += y;\n        current[5] += x;\n        current[6] += y;\n      // falls through\n      case 'C':\n        controlX = current[3];\n        controlY = current[4];\n        x = current[5];\n        y = current[6];\n        converted = ['C', current[1], current[2], controlX, controlY, x, y];\n        break;\n      case 's': // shorthand cubic bezierCurveTo, relative\n        current[1] += x;\n        current[2] += y;\n        current[3] += x;\n        current[4] += y;\n      // falls through\n      case 'S':\n        // would be sScC but since we are swapping sSc for C, we check just that.\n        if (previous === 'C') {\n          // calculate reflection of previous control points\n          controlX = 2 * x - controlX;\n          controlY = 2 * y - controlY;\n        } else {\n          // If there is no previous command or if the previous command was not a C, c, S, or s,\n          // the control point is coincident with the current point\n          controlX = x;\n          controlY = y;\n        }\n        x = current[3];\n        y = current[4];\n        converted = ['C', controlX, controlY, current[1], current[2], x, y];\n        // converted[3] and converted[4] are NOW the second control point.\n        // we keep it for the next reflection.\n        controlX = converted[3];\n        controlY = converted[4];\n        break;\n      case 'q': // quadraticCurveTo, relative\n        current[1] += x;\n        current[2] += y;\n        current[3] += x;\n        current[4] += y;\n      // falls through\n      case 'Q':\n        controlX = current[1];\n        controlY = current[2];\n        x = current[3];\n        y = current[4];\n        converted = ['Q', controlX, controlY, x, y];\n        break;\n      case 't': // shorthand quadraticCurveTo, relative\n        current[1] += x;\n        current[2] += y;\n      // falls through\n      case 'T':\n        if (previous === 'Q') {\n          // calculate reflection of previous control point\n          controlX = 2 * x - controlX;\n          controlY = 2 * y - controlY;\n        } else {\n          // If there is no previous command or if the previous command was not a Q, q, T or t,\n          // assume the control point is coincident with the current point\n          controlX = x;\n          controlY = y;\n        }\n        x = current[1];\n        y = current[2];\n        converted = ['Q', controlX, controlY, x, y];\n        break;\n      case 'a':\n        current[6] += x;\n        current[7] += y;\n      // falls through\n      case 'A':\n        fromArcToBeziers(x, y, current).forEach((b) => destinationPath.push(b));\n        x = current[6];\n        y = current[7];\n        break;\n      case 'z':\n      case 'Z':\n        x = x1;\n        y = y1;\n        converted = ['Z'];\n        break;\n      default:\n    }\n    if (converted) {\n      destinationPath.push(converted);\n      previous = converted[0];\n    } else {\n      previous = '';\n    }\n  }\n  return destinationPath;\n};\n\n// todo verify if we can just use the point class here\n/**\n * Calc length from point x1,y1 to x2,y2\n * @param {number} x1 starting point x\n * @param {number} y1 starting point y\n * @param {number} x2 starting point x\n * @param {number} y2 starting point y\n * @return {number} length of segment\n */\nconst calcLineLength = (\n  x1: number,\n  y1: number,\n  x2: number,\n  y2: number\n): number => Math.sqrt((x2 - x1) ** 2 + (y2 - y1) ** 2);\n\n/**\n * Get an iterator that takes a percentage and returns a point\n * @param {number} begx\n * @param {number} begy\n * @param {number} cp1x\n * @param {number} cp1y\n * @param {number} cp2x\n * @param {number} cp2y\n * @param {number} endx\n * @param {number} endy\n */\nconst getPointOnCubicBezierIterator =\n  (\n    begx: number,\n    begy: number,\n    cp1x: number,\n    cp1y: number,\n    cp2x: number,\n    cp2y: number,\n    endx: number,\n    endy: number\n  ) =>\n  (pct: number) => {\n    const c1 = CB1(pct),\n      c2 = CB2(pct),\n      c3 = CB3(pct),\n      c4 = CB4(pct);\n    return new Point(\n      endx * c1 + cp2x * c2 + cp1x * c3 + begx * c4,\n      endy * c1 + cp2y * c2 + cp1y * c3 + begy * c4\n    );\n  };\n\nconst QB1 = (t: number) => t ** 2;\nconst QB2 = (t: number) => 2 * t * (1 - t);\nconst QB3 = (t: number) => (1 - t) ** 2;\n\nconst getTangentCubicIterator =\n  (\n    p1x: number,\n    p1y: number,\n    p2x: number,\n    p2y: number,\n    p3x: number,\n    p3y: number,\n    p4x: number,\n    p4y: number\n  ) =>\n  (pct: number) => {\n    const qb1 = QB1(pct),\n      qb2 = QB2(pct),\n      qb3 = QB3(pct),\n      tangentX =\n        3 * (qb3 * (p2x - p1x) + qb2 * (p3x - p2x) + qb1 * (p4x - p3x)),\n      tangentY =\n        3 * (qb3 * (p2y - p1y) + qb2 * (p3y - p2y) + qb1 * (p4y - p3y));\n    return Math.atan2(tangentY, tangentX);\n  };\n\nconst getPointOnQuadraticBezierIterator =\n  (\n    p1x: number,\n    p1y: number,\n    p2x: number,\n    p2y: number,\n    p3x: number,\n    p3y: number\n  ) =>\n  (pct: number) => {\n    const c1 = QB1(pct),\n      c2 = QB2(pct),\n      c3 = QB3(pct);\n    return new Point(\n      p3x * c1 + p2x * c2 + p1x * c3,\n      p3y * c1 + p2y * c2 + p1y * c3\n    );\n  };\n\nconst getTangentQuadraticIterator =\n  (\n    p1x: number,\n    p1y: number,\n    p2x: number,\n    p2y: number,\n    p3x: number,\n    p3y: number\n  ) =>\n  (pct: number) => {\n    const invT = 1 - pct,\n      tangentX = 2 * (invT * (p2x - p1x) + pct * (p3x - p2x)),\n      tangentY = 2 * (invT * (p2y - p1y) + pct * (p3y - p2y));\n    return Math.atan2(tangentY, tangentX);\n  };\n\n// this will run over a path segment (a cubic or quadratic segment) and approximate it\n// with 100 segments. This will good enough to calculate the length of the curve\nconst pathIterator = (\n  iterator: (pct: number) => Point,\n  x1: number,\n  y1: number\n) => {\n  let tempP = new Point(x1, y1),\n    tmpLen = 0;\n  for (let perc = 1; perc <= 100; perc += 1) {\n    const p = iterator(perc / 100);\n    tmpLen += calcLineLength(tempP.x, tempP.y, p.x, p.y);\n    tempP = p;\n  }\n  return tmpLen;\n};\n\n/**\n * Given a pathInfo, and a distance in pixels, find the percentage from 0 to 1\n * that correspond to that pixels run over the path.\n * The percentage will be then used to find the correct point on the canvas for the path.\n * @param {Array} segInfo fabricJS collection of information on a parsed path\n * @param {number} distance from starting point, in pixels.\n * @return {TPointAngle} info object with x and y ( the point on canvas ) and angle, the tangent on that point;\n */\nconst findPercentageForDistance = (\n  segInfo: TCurveInfo<'Q' | 'C'>,\n  distance: number\n): TPointAngle => {\n  let perc = 0,\n    tmpLen = 0,\n    tempP: XY = { x: segInfo.x, y: segInfo.y },\n    p: XY = { ...tempP },\n    nextLen: number,\n    nextStep = 0.01,\n    lastPerc = 0;\n  // nextStep > 0.0001 covers 0.00015625 that 1/64th of 1/100\n  // the path\n  const iterator = segInfo.iterator,\n    angleFinder = segInfo.angleFinder;\n  while (tmpLen < distance && nextStep > 0.0001) {\n    p = iterator(perc);\n    lastPerc = perc;\n    nextLen = calcLineLength(tempP.x, tempP.y, p.x, p.y);\n    // compare tmpLen each cycle with distance, decide next perc to test.\n    if (nextLen + tmpLen > distance) {\n      // we discard this step and we make smaller steps.\n      perc -= nextStep;\n      nextStep /= 2;\n    } else {\n      tempP = p;\n      perc += nextStep;\n      tmpLen += nextLen;\n    }\n  }\n  return { ...p, angle: angleFinder(lastPerc) };\n};\n\n/**\n * Run over a parsed and simplified path and extract some information (length of each command and starting point)\n * @param {TSimplePathData} path parsed path commands\n * @return {TPathSegmentInfo[]} path commands information\n */\nexport const getPathSegmentsInfo = (\n  path: TSimplePathData\n): TPathSegmentInfo[] => {\n  let totalLength = 0,\n    //x2 and y2 are the coords of segment start\n    //x1 and y1 are the coords of the current point\n    x1 = 0,\n    y1 = 0,\n    x2 = 0,\n    y2 = 0,\n    iterator,\n    tempInfo: TPathSegmentInfo;\n  const info: TPathSegmentInfo[] = [];\n  for (const current of path) {\n    const basicInfo: TPathSegmentInfoCommon<keyof TPathSegmentCommandInfo> = {\n      x: x1,\n      y: y1,\n      command: current[0],\n      length: 0,\n    };\n    switch (\n      current[0] //first letter\n    ) {\n      case 'M':\n        tempInfo = <TPathSegmentInfoCommon<'M'>>basicInfo;\n        tempInfo.x = x2 = x1 = current[1];\n        tempInfo.y = y2 = y1 = current[2];\n        break;\n      case 'L':\n        tempInfo = <TPathSegmentInfoCommon<'L'>>basicInfo;\n        tempInfo.length = calcLineLength(x1, y1, current[1], current[2]);\n        x1 = current[1];\n        y1 = current[2];\n        break;\n      case 'C':\n        iterator = getPointOnCubicBezierIterator(\n          x1,\n          y1,\n          current[1],\n          current[2],\n          current[3],\n          current[4],\n          current[5],\n          current[6]\n        );\n        tempInfo = <TCurveInfo<'C'>>basicInfo;\n        tempInfo.iterator = iterator;\n        tempInfo.angleFinder = getTangentCubicIterator(\n          x1,\n          y1,\n          current[1],\n          current[2],\n          current[3],\n          current[4],\n          current[5],\n          current[6]\n        );\n        tempInfo.length = pathIterator(iterator, x1, y1);\n\n        x1 = current[5];\n        y1 = current[6];\n        break;\n      case 'Q':\n        iterator = getPointOnQuadraticBezierIterator(\n          x1,\n          y1,\n          current[1],\n          current[2],\n          current[3],\n          current[4]\n        );\n        tempInfo = <TCurveInfo<'Q'>>basicInfo;\n        tempInfo.iterator = iterator;\n        tempInfo.angleFinder = getTangentQuadraticIterator(\n          x1,\n          y1,\n          current[1],\n          current[2],\n          current[3],\n          current[4]\n        );\n        tempInfo.length = pathIterator(iterator, x1, y1);\n        x1 = current[3];\n        y1 = current[4];\n        break;\n      case 'Z':\n        // we add those in order to ease calculations later\n        tempInfo = <TEndPathInfo>basicInfo;\n        tempInfo.destX = x2;\n        tempInfo.destY = y2;\n        tempInfo.length = calcLineLength(x1, y1, x2, y2);\n        x1 = x2;\n        y1 = y2;\n        break;\n    }\n    totalLength += tempInfo.length;\n    info.push(tempInfo);\n  }\n  info.push({ length: totalLength, x: x1, y: y1 });\n  return info;\n};\n\n/**\n * Get the point on the path that is distance along the path\n * @param path\n * @param distance\n * @param infos\n */\nexport const getPointOnPath = (\n  path: TSimplePathData,\n  distance: number,\n  infos: TPathSegmentInfo[] = getPathSegmentsInfo(path)\n): TPointAngle | undefined => {\n  let i = 0;\n  while (distance - infos[i].length > 0 && i < infos.length - 2) {\n    distance -= infos[i].length;\n    i++;\n  }\n  const segInfo = infos[i],\n    segPercent = distance / segInfo.length,\n    segment = path[i];\n\n  switch (segInfo.command) {\n    case 'M':\n      return { x: segInfo.x, y: segInfo.y, angle: 0 };\n    case 'Z':\n      return {\n        ...new Point(segInfo.x, segInfo.y).lerp(\n          new Point(segInfo.destX, segInfo.destY),\n          segPercent\n        ),\n        angle: Math.atan2(segInfo.destY - segInfo.y, segInfo.destX - segInfo.x),\n      };\n    case 'L':\n      return {\n        ...new Point(segInfo.x, segInfo.y).lerp(\n          new Point(segment[1]!, segment[2]!),\n          segPercent\n        ),\n        angle: Math.atan2(segment[2]! - segInfo.y, segment[1]! - segInfo.x),\n      };\n    case 'C':\n      return findPercentageForDistance(segInfo, distance);\n    case 'Q':\n      return findPercentageForDistance(segInfo, distance);\n    default:\n    // throw Error('Invalid command');\n  }\n};\n\n/**\n *\n * @param {string} pathString\n * @return {TComplexPathData} An array of SVG path commands\n * @example <caption>Usage</caption>\n * parsePath('M 3 4 Q 3 5 2 1 4 0 Q 9 12 2 1 4 0') === [\n *   ['M', 3, 4],\n *   ['Q', 3, 5, 2, 1, 4, 0],\n *   ['Q', 9, 12, 2, 1, 4, 0],\n * ];\n *\n */\nexport const parsePath = (pathString: string): TComplexPathData => {\n  // clean the string\n  // add spaces around the numbers\n  pathString = cleanupSvgAttribute(pathString);\n\n  const res: TComplexPathData = [];\n  for (const match of pathString.matchAll(new RegExp(rePathCommand, 'gi'))) {\n    let matchStr = match[0];\n    const chain: TComplexPathData = [];\n    let paramArr: RegExpExecArray | null;\n    do {\n      paramArr = new RegExp(rePathCommand, 'i').exec(matchStr);\n      if (!paramArr) {\n        break;\n      }\n      // ignore undefined match groups\n      const filteredGroups = paramArr.filter((g) => g);\n      // remove the first element from the match array since it's just the whole command\n      filteredGroups.shift();\n      // if we can't parse the number, just interpret it as a string\n      // (since it's probably the path command)\n      const command = filteredGroups.map((g) => {\n        const numParse = Number.parseFloat(g);\n        if (Number.isNaN(numParse)) {\n          return g;\n        } else {\n          return numParse;\n        }\n      });\n      chain.push(command as any);\n      // stop now if it's a z command\n      if (filteredGroups.length <= 1) {\n        break;\n      }\n      // remove the last part of the chained command\n      filteredGroups.shift();\n      // ` ?` is to support commands with optional spaces between flags\n      matchStr = matchStr.replace(\n        new RegExp(`${filteredGroups.join(' ?')} ?$`),\n        ''\n      );\n    } while (paramArr);\n    // add the chain, convert multiple m's to l's in the process\n    chain.reverse().forEach((c, idx) => {\n      const transformed = repeatedCommands[c[0]];\n      if (idx > 0 && (transformed == 'l' || transformed == 'L')) {\n        c[0] = transformed;\n      }\n      res.push(c);\n    });\n  }\n  return res;\n};\n\n/**\n *\n * Converts points to a smooth SVG path\n * @param {XY[]} points Array of points\n * @param {number} [correction] Apply a correction to the path (usually we use `width / 1000`). If value is undefined 0 is used as the correction value.\n * @return {(string|number)[][]} An array of SVG path commands\n */\nexport const getSmoothPathFromPoints = (\n  points: Point[],\n  correction = 0\n): TSimplePathData => {\n  let p1 = new Point(points[0]),\n    p2 = new Point(points[1]),\n    multSignX = 1,\n    multSignY = 0;\n  const path: TSimplePathData = [],\n    len = points.length,\n    manyPoints = len > 2;\n\n  if (manyPoints) {\n    multSignX = points[2].x < p2.x ? -1 : points[2].x === p2.x ? 0 : 1;\n    multSignY = points[2].y < p2.y ? -1 : points[2].y === p2.y ? 0 : 1;\n  }\n  path.push([\n    'M',\n    p1.x - multSignX * correction,\n    p1.y - multSignY * correction,\n  ]);\n  let i;\n  for (i = 1; i < len; i++) {\n    if (!p1.eq(p2)) {\n      const midPoint = p1.midPointFrom(p2);\n      // p1 is our bezier control point\n      // midpoint is our endpoint\n      // start point is p(i-1) value.\n      path.push(['Q', p1.x, p1.y, midPoint.x, midPoint.y]);\n    }\n    p1 = points[i];\n    if (i + 1 < points.length) {\n      p2 = points[i + 1];\n    }\n  }\n  if (manyPoints) {\n    multSignX = p1.x > points[i - 2].x ? 1 : p1.x === points[i - 2].x ? 0 : -1;\n    multSignY = p1.y > points[i - 2].y ? 1 : p1.y === points[i - 2].y ? 0 : -1;\n  }\n  path.push([\n    'L',\n    p1.x + multSignX * correction,\n    p1.y + multSignY * correction,\n  ]);\n  return path;\n};\n\n/**\n * Transform a path by transforming each segment.\n * it has to be a simplified path or it won't work.\n * WARNING: this depends from pathOffset for correct operation\n * @param {TSimplePathData} path fabricJS parsed and simplified path commands\n * @param {TMat2D} transform matrix that represent the transformation\n * @param {Point} [pathOffset] `Path.pathOffset`\n * @returns {TSimplePathData} the transformed path\n */\nexport const transformPath = (\n  path: TSimplePathData,\n  transform: TMat2D,\n  pathOffset: Point\n): TSimplePathData => {\n  if (pathOffset) {\n    transform = multiplyTransformMatrices(transform, [\n      1,\n      0,\n      0,\n      1,\n      -pathOffset.x,\n      -pathOffset.y,\n    ]);\n  }\n  return path.map((pathSegment) => {\n    const newSegment: TSimpleParsedCommand = [...pathSegment];\n    for (let i = 1; i < pathSegment.length - 1; i += 2) {\n      // TODO: is there a way to get around casting to any?\n      const { x, y } = transformPoint(\n        {\n          x: pathSegment[i] as number,\n          y: pathSegment[i + 1] as number,\n        },\n        transform\n      );\n      newSegment[i] = x;\n      newSegment[i + 1] = y;\n    }\n    return newSegment;\n  });\n};\n\n/**\n * Returns an array of path commands to create a regular polygon\n * @param {number} numVertexes\n * @param {number} radius\n * @returns {TSimplePathData} An array of SVG path commands\n */\nexport const getRegularPolygonPath = (\n  numVertexes: number,\n  radius: number\n): TSimplePathData => {\n  const interiorAngle = (Math.PI * 2) / numVertexes;\n  // rotationAdjustment rotates the path by 1/2 the interior angle so that the polygon always has a flat side on the bottom\n  // This isn't strictly necessary, but it's how we tend to think of and expect polygons to be drawn\n  let rotationAdjustment = -halfPI;\n  if (numVertexes % 2 === 0) {\n    rotationAdjustment += interiorAngle / 2;\n  }\n  const d = new Array(numVertexes + 1);\n  for (let i = 0; i < numVertexes; i++) {\n    const rad = i * interiorAngle + rotationAdjustment;\n    const { x, y } = new Point(cos(rad), sin(rad)).scalarMultiply(radius);\n    d[i] = [i === 0 ? 'M' : 'L', x, y];\n  }\n  d[numVertexes] = ['Z'];\n  return d;\n};\n\n/**\n * Join path commands to go back to svg format\n * @param {TSimplePathData} pathData fabricJS parsed path commands\n * @param {number} fractionDigits number of fraction digits to \"leave\"\n * @return {String} joined path 'M 0 0 L 20 30'\n */\nexport const joinPath = (pathData: TSimplePathData, fractionDigits?: number) =>\n  pathData\n    .map((segment) => {\n      return segment\n        .map((arg, i) => {\n          if (i === 0) return arg;\n          return fractionDigits === undefined\n            ? arg\n            : toFixed(arg, fractionDigits);\n        })\n        .join(' ');\n    })\n    .join(' ');\n","import { getFabricWindow } from '../../env';\nimport { noop } from '../../constants';\nimport type { Abortable } from '../../typedefs';\nimport { SignalAbortedError } from './console';\n\ntype requestOptions = Abortable & {\n  onComplete?: (xhr: XMLHttpRequest) => void;\n};\n\n/**\n * Cross-browser abstraction for sending XMLHttpRequest\n * @deprecated this has to go away, we can use a modern browser method to do the same.\n * @param {String} url URL to send XMLHttpRequest to\n * @param {Object} [options] Options object\n * @param {AbortSignal} [options.signal] handle aborting, see https://developer.mozilla.org/en-US/docs/Web/API/AbortController/signal\n * @param {Function} options.onComplete Callback to invoke when request is completed\n * @return {XMLHttpRequest} request\n */\n\nexport function request(url: string, options: requestOptions = {}) {\n  const onComplete = options.onComplete || noop,\n    xhr = new (getFabricWindow().XMLHttpRequest)(),\n    signal = options.signal,\n    abort = function () {\n      xhr.abort();\n    },\n    removeListener = function () {\n      signal && signal.removeEventListener('abort', abort);\n      xhr.onerror = xhr.ontimeout = noop;\n    };\n\n  if (signal && signal.aborted) {\n    throw new SignalAbortedError('request');\n  } else if (signal) {\n    signal.addEventListener('abort', abort, { once: true });\n  }\n\n  /** @ignore */\n  xhr.onreadystatechange = function () {\n    if (xhr.readyState === 4) {\n      removeListener();\n      onComplete(xhr);\n      xhr.onreadystatechange = noop;\n    }\n  };\n\n  xhr.onerror = xhr.ontimeout = removeListener;\n\n  xhr.open('get', url, true);\n\n  xhr.send();\n  return xhr;\n}\n","import { CENTER } from '../constants';\nimport type { FabricImage } from '../shapes/Image';\nimport type { FabricObject } from '../shapes/Object/FabricObject';\nimport type { TMat2D } from '../typedefs';\nimport { qrDecompose } from './misc/matrix';\n\ntype FabricObjectWithTransformMatrix = FabricObject & {\n  transformMatrix?: TMat2D;\n};\n\n/**\n * This function is an helper for svg import. it decompose the transformMatrix\n * and assign properties to object.\n * untransformed coordinates\n * @private\n */\nconst _assignTransformMatrixProps = (\n  object: FabricObjectWithTransformMatrix\n) => {\n  if (object.transformMatrix) {\n    const { scaleX, scaleY, angle, skewX } = qrDecompose(\n      object.transformMatrix\n    );\n    object.flipX = false;\n    object.flipY = false;\n    object.set('scaleX', scaleX);\n    object.set('scaleY', scaleY);\n    object.angle = angle;\n    object.skewX = skewX;\n    object.skewY = 0;\n  }\n};\n\n/**\n * This function is an helper for svg import. it removes the transform matrix\n * and set to object properties that fabricjs can handle\n * @private\n * @param {Object} preserveAspectRatioOptions\n */\nexport const removeTransformMatrixForSvgParsing = (\n  object: FabricObjectWithTransformMatrix,\n  preserveAspectRatioOptions?: any\n) => {\n  let center = object._findCenterFromElement();\n  if (object.transformMatrix) {\n    _assignTransformMatrixProps(object);\n    center = center.transform(object.transformMatrix);\n  }\n  delete object.transformMatrix;\n  if (preserveAspectRatioOptions) {\n    object.scaleX *= preserveAspectRatioOptions.scaleX;\n    object.scaleY *= preserveAspectRatioOptions.scaleY;\n    (object as FabricImage).cropX = preserveAspectRatioOptions.cropX;\n    (object as FabricImage).cropY = preserveAspectRatioOptions.cropY;\n    center.x += preserveAspectRatioOptions.offsetLeft;\n    center.y += preserveAspectRatioOptions.offsetTop;\n    object.width = preserveAspectRatioOptions.width;\n    object.height = preserveAspectRatioOptions.height;\n  }\n  object.setPositionByOrigin(center, CENTER, CENTER);\n};\n","import type { FabricObject } from '../../shapes/Object/FabricObject';\nimport { sendObjectToPlane } from './planeChange';\nimport { Group } from '../../shapes/Group';\n/**\n * Merges 2 clip paths into one visually equal clip path\n *\n * **IMPORTANT**:\\\n * Does **NOT** clone the arguments, clone them proir if necessary.\n *\n * Creates a wrapper (group) that contains one clip path and is clipped by the other so content is kept where both overlap.\n * Use this method if both the clip paths may have nested clip paths of their own, so assigning one to the other's clip path property is not possible.\n *\n * In order to handle the `inverted` property we follow logic described in the following cases:\\\n * **(1)** both clip paths are inverted - the clip paths pass the inverted prop to the wrapper and loose it themselves.\\\n * **(2)** one is inverted and the other isn't - the wrapper shouldn't become inverted and the inverted clip path must clip the non inverted one to produce an identical visual effect.\\\n * **(3)** both clip paths are not inverted - wrapper and clip paths remain unchanged.\n *\n * @memberOf fabric.util\n * @param {fabric.Object} c1\n * @param {fabric.Object} c2\n * @returns {fabric.Object} merged clip path\n */\nexport const mergeClipPaths = (c1: FabricObject, c2: FabricObject) => {\n  let a = c1,\n    b = c2;\n  if (a.inverted && !b.inverted) {\n    //  case (2)\n    a = c2;\n    b = c1;\n  }\n  //  `b` becomes `a`'s clip path so we transform `b` to `a` coordinate plane\n  sendObjectToPlane(b, b.group?.calcTransformMatrix(), a.calcTransformMatrix());\n  //  assign the `inverted` prop to the wrapping group\n  const inverted = a.inverted && b.inverted;\n  if (inverted) {\n    //  case (1)\n    a.inverted = b.inverted = false;\n  }\n  return new Group([a], { clipPath: b, inverted });\n};\n","import type { Point } from '../../Point';\nimport type { TRadian } from '../../typedefs';\n/**\n * Rotates `point` around `origin` with `radians`\n * @deprecated use the Point.rotate\n * @param {Point} origin The origin of the rotation\n * @param {Point} origin The origin of the rotation\n * @param {TRadian} radians The radians of the angle for the rotation\n * @return {Point} The new rotated point\n */\nexport const rotatePoint = (\n  point: Point,\n  origin: Point,\n  radians: TRadian\n): Point => point.rotate(radians, origin);\n","import { getEnv, getFabricDocument } from '../../env';\nimport type { TSize } from '../../typedefs';\nimport { createCanvasElement, setStyle } from '../../util';\nimport type { CSSDimensions } from './util';\nimport {\n  allowTouchScrolling,\n  makeElementUnselectable,\n  setCSSDimensions,\n} from './util';\nimport type { CanvasItem } from './StaticCanvasDOMManager';\nimport { StaticCanvasDOMManager } from './StaticCanvasDOMManager';\nimport { setCanvasDimensions } from './util';\n\nexport class CanvasDOMManager extends StaticCanvasDOMManager {\n  upper: CanvasItem;\n  container: HTMLDivElement;\n\n  constructor(\n    arg0?: string | HTMLCanvasElement,\n    {\n      allowTouchScrolling = false,\n      containerClass = '',\n    }: {\n      allowTouchScrolling?: boolean;\n      /**\n       * @deprecated here only for backward compatibility\n       */\n      containerClass?: string;\n    } = {}\n  ) {\n    super(arg0);\n    const { el: lowerCanvasEl } = this.lower;\n    const upperCanvasEl = this.createUpperCanvas();\n    this.upper = { el: upperCanvasEl, ctx: upperCanvasEl.getContext('2d')! };\n    this.applyCanvasStyle(lowerCanvasEl, {\n      allowTouchScrolling,\n    });\n    this.applyCanvasStyle(upperCanvasEl, {\n      allowTouchScrolling,\n    });\n    const container = this.createContainerElement();\n    container.classList.add(containerClass);\n    if (lowerCanvasEl.parentNode) {\n      lowerCanvasEl.parentNode.replaceChild(container, lowerCanvasEl);\n    }\n    container.append(lowerCanvasEl, upperCanvasEl);\n    this.container = container;\n  }\n\n  protected createUpperCanvas() {\n    const { el: lowerCanvasEl } = this.lower;\n    const el = createCanvasElement();\n    // we assign the same classname of the lowerCanvas\n    el.className = lowerCanvasEl.className;\n    // but then we remove the lower-canvas specific className\n    el.classList.remove('lower-canvas');\n    // we add the specific upper-canvas class\n    el.classList.add('upper-canvas');\n    el.setAttribute('data-fabric', 'top');\n    el.style.cssText = lowerCanvasEl.style.cssText;\n    el.setAttribute('draggable', 'true');\n    return el;\n  }\n\n  protected createContainerElement() {\n    const container = getFabricDocument().createElement('div');\n    container.setAttribute('data-fabric', 'wrapper');\n    setStyle(container, {\n      position: 'relative',\n    });\n    makeElementUnselectable(container);\n    return container;\n  }\n\n  /**\n   * @private\n   * @param {HTMLCanvasElement} element canvas element to apply styles on\n   */\n  protected applyCanvasStyle(\n    element: HTMLCanvasElement,\n    { allowTouchScrolling: allow }: { allowTouchScrolling: boolean }\n  ) {\n    setStyle(element, {\n      position: 'absolute',\n      left: '0',\n      top: '0',\n    });\n    allowTouchScrolling(element, allow);\n    makeElementUnselectable(element);\n  }\n\n  setDimensions(size: TSize, retinaScaling: number) {\n    super.setDimensions(size, retinaScaling);\n    const { el, ctx } = this.upper;\n    setCanvasDimensions(el, ctx, size, retinaScaling);\n  }\n\n  setCSSDimensions(size: Partial<CSSDimensions>): void {\n    super.setCSSDimensions(size);\n    setCSSDimensions(this.upper.el, size);\n    setCSSDimensions(this.container, size);\n  }\n\n  cleanupDOM(size: TSize) {\n    const container = this.container,\n      { el: lowerCanvasEl } = this.lower,\n      { el: upperCanvasEl } = this.upper;\n    super.cleanupDOM(size);\n    container.removeChild(upperCanvasEl);\n    container.removeChild(lowerCanvasEl);\n    if (container.parentNode) {\n      container.parentNode.replaceChild(lowerCanvasEl, container);\n    }\n  }\n\n  dispose() {\n    super.dispose();\n    getEnv().dispose(this.upper.el);\n    // @ts-expect-error disposing\n    delete this.upper;\n    // @ts-expect-error disposing\n    delete this.container;\n  }\n}\n","import { dragHandler } from '../controls/drag';\nimport { getActionFromCorner } from '../controls/util';\nimport { Point } from '../Point';\nimport { FabricObject } from '../shapes/Object/FabricObject';\nimport type {\n  CanvasEvents,\n  ModifierKey,\n  TOptionalModifierKey,\n  TPointerEvent,\n  Transform,\n} from '../EventTypeDefs';\nimport {\n  addTransformToObject,\n  saveObjectTransform,\n} from '../util/misc/objectTransforms';\nimport type { TCanvasSizeOptions } from './StaticCanvas';\nimport { StaticCanvas } from './StaticCanvas';\nimport { isCollection } from '../Collection';\nimport { isTransparent } from '../util/misc/isTransparent';\nimport type {\n  TMat2D,\n  TOriginX,\n  TOriginY,\n  TSize,\n  TSVGReviver,\n} from '../typedefs';\nimport { degreesToRadians } from '../util/misc/radiansDegreesConversion';\nimport { getPointer, isTouchEvent } from '../util/dom_event';\nimport type { IText } from '../shapes/IText/IText';\nimport type { BaseBrush } from '../brushes/BaseBrush';\nimport { pick } from '../util/misc/pick';\nimport { sendPointToPlane } from '../util/misc/planeChange';\nimport { cos, createCanvasElement, sin } from '../util';\nimport { CanvasDOMManager } from './DOMManagers/CanvasDOMManager';\nimport { BOTTOM, CENTER, LEFT, RIGHT, TOP } from '../constants';\nimport type { CanvasOptions } from './CanvasOptions';\nimport { canvasDefaults } from './CanvasOptions';\nimport { Intersection } from '../Intersection';\nimport { isActiveSelection } from '../util/typeAssertions';\n\n/**\n * Canvas class\n * @class Canvas\n * @extends StaticCanvas\n * @tutorial {@link http://fabricjs.com/fabric-intro-part-1#canvas}\n *\n * @fires object:modified at the end of a transform\n * @fires object:rotating while an object is being rotated from the control\n * @fires object:scaling while an object is being scaled by controls\n * @fires object:moving while an object is being dragged\n * @fires object:skewing while an object is being skewed from the controls\n *\n * @fires before:transform before a transform is is started\n * @fires before:selection:cleared\n * @fires selection:cleared\n * @fires selection:updated\n * @fires selection:created\n *\n * @fires path:created after a drawing operation ends and the path is added\n * @fires mouse:down\n * @fires mouse:move\n * @fires mouse:up\n * @fires mouse:down:before  on mouse down, before the inner fabric logic runs\n * @fires mouse:move:before on mouse move, before the inner fabric logic runs\n * @fires mouse:up:before on mouse up, before the inner fabric logic runs\n * @fires mouse:over\n * @fires mouse:out\n * @fires mouse:dblclick whenever a native dbl click event fires on the canvas.\n *\n * @fires dragover\n * @fires dragenter\n * @fires dragleave\n * @fires drag:enter object drag enter\n * @fires drag:leave object drag leave\n * @fires drop:before before drop event. Prepare for the drop event (same native event).\n * @fires drop\n * @fires drop:after after drop event. Run logic on canvas after event has been accepted/declined (same native event).\n * @example\n * let a: fabric.Object, b: fabric.Object;\n * let flag = false;\n * canvas.add(a, b);\n * a.on('drop:before', opt => {\n *  //  we want a to accept the drop even though it's below b in the stack\n *  flag = this.canDrop(opt.e);\n * });\n * b.canDrop = function(e) {\n *  !flag && this.draggableTextDelegate.canDrop(e);\n * }\n * b.on('dragover', opt => b.set('fill', opt.dropTarget === b ? 'pink' : 'black'));\n * a.on('drop', opt => {\n *  opt.e.defaultPrevented  //  drop occurred\n *  opt.didDrop             //  drop occurred on canvas\n *  opt.target              //  drop target\n *  opt.target !== a && a.set('text', 'I lost');\n * });\n * canvas.on('drop:after', opt => {\n *  //  inform user who won\n *  if(!opt.e.defaultPrevented) {\n *    // no winners\n *  }\n *  else if(!opt.didDrop) {\n *    //  my objects didn't win, some other lucky object\n *  }\n *  else {\n *    //  we have a winner it's opt.target!!\n *  }\n * })\n *\n * @fires after:render at the end of the render process, receives the context in the callback\n * @fires before:render at start the render process, receives the context in the callback\n *\n * @fires contextmenu:before\n * @fires contextmenu\n * @example\n * let handler;\n * targets.forEach(target => {\n *   target.on('contextmenu:before', opt => {\n *     //  decide which target should handle the event before canvas hijacks it\n *     if (someCaseHappens && opt.targets.includes(target)) {\n *       handler = target;\n *     }\n *   });\n *   target.on('contextmenu', opt => {\n *     //  do something fantastic\n *   });\n * });\n * canvas.on('contextmenu', opt => {\n *   if (!handler) {\n *     //  no one takes responsibility, it's always left to me\n *     //  let's show them how it's done!\n *   }\n * });\n *\n */\nexport class SelectableCanvas<EventSpec extends CanvasEvents = CanvasEvents>\n  extends StaticCanvas<EventSpec>\n  implements Omit<CanvasOptions, 'enablePointerEvents'>\n{\n  declare _objects: FabricObject[];\n\n  // transform config\n  declare uniformScaling: boolean;\n  declare uniScaleKey: TOptionalModifierKey;\n  declare centeredScaling: boolean;\n  declare centeredRotation: boolean;\n  declare centeredKey: TOptionalModifierKey;\n  declare altActionKey: TOptionalModifierKey;\n\n  // selection config\n  declare selection: boolean;\n  declare selectionKey: TOptionalModifierKey | ModifierKey[];\n  declare altSelectionKey: TOptionalModifierKey;\n  declare selectionColor: string;\n  declare selectionDashArray: number[];\n  declare selectionBorderColor: string;\n  declare selectionLineWidth: number;\n  declare selectionFullyContained: boolean;\n\n  // cursors\n  declare hoverCursor: CSSStyleDeclaration['cursor'];\n  declare moveCursor: CSSStyleDeclaration['cursor'];\n  declare defaultCursor: CSSStyleDeclaration['cursor'];\n  declare freeDrawingCursor: CSSStyleDeclaration['cursor'];\n  declare notAllowedCursor: CSSStyleDeclaration['cursor'];\n\n  declare containerClass: string;\n\n  // target find config\n  declare perPixelTargetFind: boolean;\n  declare targetFindTolerance: number;\n  declare skipTargetFind: boolean;\n\n  /**\n   * When true, mouse events on canvas (mousedown/mousemove/mouseup) result in free drawing.\n   * After mousedown, mousemove creates a shape,\n   * and then mouseup finalizes it and adds an instance of `fabric.Path` onto canvas.\n   * @tutorial {@link http://fabricjs.com/fabric-intro-part-4#free_drawing}\n   * @type Boolean\n   * @default\n   */\n  declare isDrawingMode: boolean;\n\n  declare preserveObjectStacking: boolean;\n\n  // event config\n  declare stopContextMenu: boolean;\n  declare fireRightClick: boolean;\n  declare fireMiddleClick: boolean;\n\n  /**\n   * Keep track of the subTargets for Mouse Events, ordered bottom up from innermost nested subTarget\n   * @type FabricObject[]\n   */\n  targets: FabricObject[] = [];\n\n  /**\n   * Keep track of the hovered target\n   * @type FabricObject | null\n   * @private\n   */\n  declare _hoveredTarget?: FabricObject;\n\n  /**\n   * hold the list of nested targets hovered\n   * @type FabricObject[]\n   * @private\n   */\n  _hoveredTargets: FabricObject[] = [];\n\n  /**\n   * hold the list of objects to render\n   * @type FabricObject[]\n   * @private\n   */\n  _objectsToRender?: FabricObject[];\n\n  /**\n   * hold a reference to a data structure that contains information\n   * on the current on going transform\n   * @type\n   * @private\n   */\n  _currentTransform: Transform | null = null;\n\n  /**\n   * hold a reference to a data structure used to track the selection\n   * box on canvas drag\n   * on the current on going transform\n   * x, y, deltaX and deltaY are in scene plane\n   * @type\n   * @private\n   */\n  protected _groupSelector: {\n    x: number;\n    y: number;\n    deltaX: number;\n    deltaY: number;\n  } | null = null;\n\n  /**\n   * internal flag used to understand if the context top requires a cleanup\n   * in case this is true, the contextTop will be cleared at the next render\n   * @type boolean\n   * @private\n   */\n  contextTopDirty = false;\n\n  /**\n   * During a mouse event we may need the pointer multiple times in multiple functions.\n   * _absolutePointer holds a reference to the pointer in fabricCanvas/design coordinates that is valid for the event\n   * lifespan. Every fabricJS mouse event create and delete the cache every time\n   * We do this because there are some HTML DOM inspection functions to get the actual pointer coordinates\n   * @type {Point}\n   */\n  protected declare _absolutePointer?: Point;\n\n  /**\n   * During a mouse event we may need the pointer multiple times in multiple functions.\n   * _pointer holds a reference to the pointer in html coordinates that is valid for the event\n   * lifespan. Every fabricJS mouse event create and delete the cache every time\n   * We do this because there are some HTML DOM inspection functions to get the actual pointer coordinates\n   * @type {Point}\n   */\n  protected declare _pointer?: Point;\n\n  /**\n   * During a mouse event we may need the target multiple times in multiple functions.\n   * _target holds a reference to the target that is valid for the event\n   * lifespan. Every fabricJS mouse event create and delete the cache every time\n   * @type {FabricObject}\n   */\n  protected declare _target?: FabricObject;\n\n  static ownDefaults = canvasDefaults;\n\n  static getDefaults(): Record<string, any> {\n    return { ...super.getDefaults(), ...SelectableCanvas.ownDefaults };\n  }\n\n  declare elements: CanvasDOMManager;\n  get upperCanvasEl() {\n    return this.elements.upper?.el;\n  }\n  get contextTop() {\n    return this.elements.upper?.ctx;\n  }\n  get wrapperEl() {\n    return this.elements.container;\n  }\n  private declare pixelFindCanvasEl: HTMLCanvasElement;\n  private declare pixelFindContext: CanvasRenderingContext2D;\n\n  protected declare _isCurrentlyDrawing: boolean;\n  declare freeDrawingBrush?: BaseBrush;\n  declare _activeObject?: FabricObject;\n\n  protected initElements(el?: string | HTMLCanvasElement) {\n    this.elements = new CanvasDOMManager(el, {\n      allowTouchScrolling: this.allowTouchScrolling,\n      containerClass: this.containerClass,\n    });\n    this._createCacheCanvas();\n  }\n\n  /**\n   * @private\n   * @param {FabricObject} obj Object that was added\n   */\n  _onObjectAdded(obj: FabricObject) {\n    this._objectsToRender = undefined;\n    super._onObjectAdded(obj);\n  }\n\n  /**\n   * @private\n   * @param {FabricObject} obj Object that was removed\n   */\n  _onObjectRemoved(obj: FabricObject) {\n    this._objectsToRender = undefined;\n    // removing active object should fire \"selection:cleared\" events\n    if (obj === this._activeObject) {\n      this.fire('before:selection:cleared', { deselected: [obj] });\n      this._discardActiveObject();\n      this.fire('selection:cleared', { deselected: [obj] });\n      obj.fire('deselected', {\n        target: obj,\n      });\n    }\n    if (obj === this._hoveredTarget) {\n      this._hoveredTarget = undefined;\n      this._hoveredTargets = [];\n    }\n    super._onObjectRemoved(obj);\n  }\n\n  _onStackOrderChanged() {\n    this._objectsToRender = undefined;\n    super._onStackOrderChanged();\n  }\n\n  /**\n   * Divides objects in two groups, one to render immediately\n   * and one to render as activeGroup.\n   * @return {Array} objects to render immediately and pushes the other in the activeGroup.\n   */\n  _chooseObjectsToRender(): FabricObject[] {\n    const activeObject = this._activeObject;\n    return !this.preserveObjectStacking && activeObject\n      ? this._objects\n          .filter((object) => !object.group && object !== activeObject)\n          .concat(activeObject)\n      : this._objects;\n  }\n\n  /**\n   * Renders both the top canvas and the secondary container canvas.\n   */\n  renderAll() {\n    this.cancelRequestedRender();\n    if (this.destroyed) {\n      return;\n    }\n    if (this.contextTopDirty && !this._groupSelector && !this.isDrawingMode) {\n      this.clearContext(this.contextTop);\n      this.contextTopDirty = false;\n    }\n    if (this.hasLostContext) {\n      this.renderTopLayer(this.contextTop);\n      this.hasLostContext = false;\n    }\n    !this._objectsToRender &&\n      (this._objectsToRender = this._chooseObjectsToRender());\n    this.renderCanvas(this.getContext(), this._objectsToRender);\n  }\n\n  /**\n   * text selection is rendered by the active text instance during the rendering cycle\n   */\n  renderTopLayer(ctx: CanvasRenderingContext2D): void {\n    ctx.save();\n    if (this.isDrawingMode && this._isCurrentlyDrawing) {\n      this.freeDrawingBrush && this.freeDrawingBrush._render();\n      this.contextTopDirty = true;\n    }\n    // we render the top context - last object\n    if (this.selection && this._groupSelector) {\n      this._drawSelection(ctx);\n      this.contextTopDirty = true;\n    }\n    ctx.restore();\n  }\n\n  /**\n   * Method to render only the top canvas.\n   * Also used to render the group selection box.\n   * Does not render text selection.\n   */\n  renderTop() {\n    const ctx = this.contextTop;\n    this.clearContext(ctx);\n    this.renderTopLayer(ctx);\n    // todo: how do i know if the after:render is for the top or normal contex?\n    this.fire('after:render', { ctx });\n  }\n\n  /**\n   * Set the canvas tolerance value for pixel taret find.\n   * Use only integer numbers.\n   * @private\n   */\n  setTargetFindTolerance(value: number) {\n    value = Math.round(value);\n    this.targetFindTolerance = value;\n    const retina = this.getRetinaScaling();\n    const size = Math.ceil((value * 2 + 1) * retina);\n    this.pixelFindCanvasEl.width = this.pixelFindCanvasEl.height = size;\n    this.pixelFindContext.scale(retina, retina);\n  }\n\n  /**\n   * Returns true if object is transparent at a certain location\n   * Clarification: this is `is target transparent at location X or are controls there`\n   * @TODO this seems dumb that we treat controls with transparency. we can find controls\n   * programmatically without painting them, the cache canvas optimization is always valid\n   * @param {FabricObject} target Object to check\n   * @param {Number} x Left coordinate in viewport space\n   * @param {Number} y Top coordinate in viewport space\n   * @return {Boolean}\n   */\n  isTargetTransparent(target: FabricObject, x: number, y: number): boolean {\n    const tolerance = this.targetFindTolerance;\n    const ctx = this.pixelFindContext;\n    this.clearContext(ctx);\n    ctx.save();\n    ctx.translate(-x + tolerance, -y + tolerance);\n    ctx.transform(...this.viewportTransform);\n    const selectionBgc = target.selectionBackgroundColor;\n    target.selectionBackgroundColor = '';\n    target.render(ctx);\n    target.selectionBackgroundColor = selectionBgc;\n    ctx.restore();\n    // our canvas is square, and made around tolerance.\n    // so tolerance in this case also represent the center of the canvas.\n    const enhancedTolerance = Math.round(tolerance * this.getRetinaScaling());\n    return isTransparent(\n      ctx,\n      enhancedTolerance,\n      enhancedTolerance,\n      enhancedTolerance\n    );\n  }\n\n  /**\n   * takes an event and determines if selection key has been pressed\n   * @private\n   * @param {TPointerEvent} e Event object\n   */\n  _isSelectionKeyPressed(e: TPointerEvent): boolean {\n    const sKey = this.selectionKey;\n    if (!sKey) {\n      return false;\n    }\n    if (Array.isArray(sKey)) {\n      return !!sKey.find((key) => !!key && e[key] === true);\n    } else {\n      return e[sKey];\n    }\n  }\n\n  /**\n   * @private\n   * @param {TPointerEvent} e Event object\n   * @param {FabricObject} target\n   */\n  _shouldClearSelection(\n    e: TPointerEvent,\n    target?: FabricObject\n  ): target is undefined {\n    const activeObjects = this.getActiveObjects(),\n      activeObject = this._activeObject;\n\n    return !!(\n      !target ||\n      (target &&\n        activeObject &&\n        activeObjects.length > 1 &&\n        activeObjects.indexOf(target) === -1 &&\n        activeObject !== target &&\n        !this._isSelectionKeyPressed(e)) ||\n      (target && !target.evented) ||\n      (target && !target.selectable && activeObject && activeObject !== target)\n    );\n  }\n\n  /**\n   * This method will take in consideration a modifier key pressed and the control we are\n   * about to drag, and try to guess the anchor point ( origin ) of the transormation.\n   * This should be really in the realm of controls, and we should remove specific code for legacy\n   * embedded actions.\n   * @TODO this probably deserve discussion/rediscovery and change/refactor\n   * @private\n   * @deprecated\n   * @param {FabricObject} target\n   * @param {string} action\n   * @param {boolean} altKey\n   * @returns {boolean} true if the transformation should be centered\n   */\n  private _shouldCenterTransform(\n    target: FabricObject,\n    action: string,\n    modifierKeyPressed: boolean\n  ) {\n    if (!target) {\n      return;\n    }\n\n    let centerTransform;\n\n    if (\n      action === 'scale' ||\n      action === 'scaleX' ||\n      action === 'scaleY' ||\n      action === 'resizing'\n    ) {\n      centerTransform = this.centeredScaling || target.centeredScaling;\n    } else if (action === 'rotate') {\n      centerTransform = this.centeredRotation || target.centeredRotation;\n    }\n\n    return centerTransform ? !modifierKeyPressed : modifierKeyPressed;\n  }\n\n  /**\n   * Given the control clicked, determine the origin of the transform.\n   * This is bad because controls can totally have custom names\n   * should disappear before release 4.0\n   * @private\n   * @deprecated\n   */\n  _getOriginFromCorner(\n    target: FabricObject,\n    controlName: string\n  ): { x: TOriginX; y: TOriginY } {\n    const origin = {\n      x: target.originX,\n      y: target.originY,\n    };\n\n    if (!controlName) {\n      return origin;\n    }\n\n    // is a left control ?\n    if (['ml', 'tl', 'bl'].includes(controlName)) {\n      origin.x = RIGHT;\n      // is a right control ?\n    } else if (['mr', 'tr', 'br'].includes(controlName)) {\n      origin.x = LEFT;\n    }\n    // is a top control ?\n    if (['tl', 'mt', 'tr'].includes(controlName)) {\n      origin.y = BOTTOM;\n      // is a bottom control ?\n    } else if (['bl', 'mb', 'br'].includes(controlName)) {\n      origin.y = TOP;\n    }\n    return origin;\n  }\n\n  /**\n   * @private\n   * @param {Event} e Event object\n   * @param {FabricObject} target\n   * @param {boolean} [alreadySelected] pass true to setup the active control\n   */\n  _setupCurrentTransform(\n    e: TPointerEvent,\n    target: FabricObject,\n    alreadySelected: boolean\n  ): void {\n    const pointer = target.group\n      ? // transform pointer to target's containing coordinate plane\n        sendPointToPlane(\n          this.getScenePoint(e),\n          undefined,\n          target.group.calcTransformMatrix()\n        )\n      : this.getScenePoint(e);\n    const { key: corner = '', control } = target.getActiveControl() || {},\n      actionHandler =\n        alreadySelected && control\n          ? control.getActionHandler(e, target, control)?.bind(control)\n          : dragHandler,\n      action = getActionFromCorner(alreadySelected, corner, e, target),\n      altKey = e[this.centeredKey as ModifierKey],\n      origin = this._shouldCenterTransform(target, action, altKey)\n        ? ({ x: CENTER, y: CENTER } as const)\n        : this._getOriginFromCorner(target, corner),\n      /**\n       * relative to target's containing coordinate plane\n       * both agree on every point\n       **/\n      transform: Transform = {\n        target: target,\n        action,\n        actionHandler,\n        actionPerformed: false,\n        corner,\n        scaleX: target.scaleX,\n        scaleY: target.scaleY,\n        skewX: target.skewX,\n        skewY: target.skewY,\n        offsetX: pointer.x - target.left,\n        offsetY: pointer.y - target.top,\n        originX: origin.x,\n        originY: origin.y,\n        ex: pointer.x,\n        ey: pointer.y,\n        lastX: pointer.x,\n        lastY: pointer.y,\n        theta: degreesToRadians(target.angle),\n        width: target.width,\n        height: target.height,\n        shiftKey: e.shiftKey,\n        altKey,\n        original: {\n          ...saveObjectTransform(target),\n          originX: origin.x,\n          originY: origin.y,\n        },\n      };\n\n    this._currentTransform = transform;\n\n    this.fire('before:transform', {\n      e,\n      transform,\n    });\n  }\n\n  /**\n   * Set the cursor type of the canvas element\n   * @param {String} value Cursor type of the canvas element.\n   * @see http://www.w3.org/TR/css3-ui/#cursor\n   */\n  setCursor(value: CSSStyleDeclaration['cursor']): void {\n    this.upperCanvasEl.style.cursor = value;\n  }\n\n  /**\n   * @private\n   * @param {CanvasRenderingContext2D} ctx to draw the selection on\n   */\n  _drawSelection(ctx: CanvasRenderingContext2D): void {\n    const { x, y, deltaX, deltaY } = this._groupSelector!,\n      start = new Point(x, y).transform(this.viewportTransform),\n      extent = new Point(x + deltaX, y + deltaY).transform(\n        this.viewportTransform\n      ),\n      strokeOffset = this.selectionLineWidth / 2;\n    let minX = Math.min(start.x, extent.x),\n      minY = Math.min(start.y, extent.y),\n      maxX = Math.max(start.x, extent.x),\n      maxY = Math.max(start.y, extent.y);\n\n    if (this.selectionColor) {\n      ctx.fillStyle = this.selectionColor;\n      ctx.fillRect(minX, minY, maxX - minX, maxY - minY);\n    }\n\n    if (!this.selectionLineWidth || !this.selectionBorderColor) {\n      return;\n    }\n    ctx.lineWidth = this.selectionLineWidth;\n    ctx.strokeStyle = this.selectionBorderColor;\n\n    minX += strokeOffset;\n    minY += strokeOffset;\n    maxX -= strokeOffset;\n    maxY -= strokeOffset;\n    // selection border\n    // @TODO: is _setLineDash still necessary on modern canvas?\n    FabricObject.prototype._setLineDash.call(\n      this,\n      ctx,\n      this.selectionDashArray\n    );\n    ctx.strokeRect(minX, minY, maxX - minX, maxY - minY);\n  }\n\n  /**\n   * Method that determines what object we are clicking on\n   * 11/09/2018 TODO: would be cool if findTarget could discern between being a full target\n   * or the outside part of the corner.\n   * @param {Event} e mouse event\n   * @return {FabricObject | null} the target found\n   */\n  findTarget(e: TPointerEvent): FabricObject | undefined {\n    if (this.skipTargetFind) {\n      return undefined;\n    }\n\n    const pointer = this.getViewportPoint(e),\n      activeObject = this._activeObject,\n      aObjects = this.getActiveObjects();\n\n    this.targets = [];\n\n    if (activeObject && aObjects.length >= 1) {\n      if (activeObject.findControl(pointer, isTouchEvent(e))) {\n        // if we hit the corner of the active object, let's return that.\n        return activeObject;\n      } else if (\n        aObjects.length > 1 &&\n        // check pointer is over active selection and possibly perform `subTargetCheck`\n        this.searchPossibleTargets([activeObject], pointer)\n      ) {\n        // active selection does not select sub targets like normal groups\n        return activeObject;\n      } else if (\n        activeObject === this.searchPossibleTargets([activeObject], pointer)\n      ) {\n        // active object is not an active selection\n        if (!this.preserveObjectStacking) {\n          return activeObject;\n        } else {\n          const subTargets = this.targets;\n          this.targets = [];\n          const target = this.searchPossibleTargets(this._objects, pointer);\n          if (\n            e[this.altSelectionKey as ModifierKey] &&\n            target &&\n            target !== activeObject\n          ) {\n            // alt selection: select active object even though it is not the top most target\n            // restore targets\n            this.targets = subTargets;\n            return activeObject;\n          }\n          return target;\n        }\n      }\n    }\n\n    return this.searchPossibleTargets(this._objects, pointer);\n  }\n\n  /**\n   * Checks if the point is inside the object selection area including padding\n   * @param {FabricObject} obj Object to test against\n   * @param {Object} [pointer] point in scene coordinates\n   * @return {Boolean} true if point is contained within an area of given object\n   * @private\n   */\n  private _pointIsInObjectSelectionArea(obj: FabricObject, point: Point) {\n    // getCoords will already take care of group de-nesting\n    let coords = obj.getCoords();\n    const viewportZoom = this.getZoom();\n    const padding = obj.padding / viewportZoom;\n    if (padding) {\n      const [tl, tr, br, bl] = coords;\n      // what is the angle of the object?\n      // we could use getTotalAngle, but is way easier to look at it\n      // from how coords are oriented, since if something went wrong\n      // at least we are consistent.\n      const angleRadians = Math.atan2(tr.y - tl.y, tr.x - tl.x),\n        cosP = cos(angleRadians) * padding,\n        sinP = sin(angleRadians) * padding,\n        cosPSinP = cosP + sinP,\n        cosPMinusSinP = cosP - sinP;\n\n      coords = [\n        new Point(tl.x - cosPMinusSinP, tl.y - cosPSinP),\n        new Point(tr.x + cosPSinP, tr.y - cosPMinusSinP),\n        new Point(br.x + cosPMinusSinP, br.y + cosPSinP),\n        new Point(bl.x - cosPSinP, bl.y + cosPMinusSinP),\n      ];\n      // in case of padding we calculate the new coords on the fly.\n      // otherwise we have to maintain 2 sets of coordinates for everything.\n      // we can reiterate on storing them.\n      // if this is slow, for now the semplification is large and doesn't impact\n      // rendering.\n      // the idea behind this is that outside target check we don't need ot know\n      // where those coords are\n    }\n    return Intersection.isPointInPolygon(point, coords);\n  }\n\n  /**\n   * Checks point is inside the object selection condition. Either area with padding\n   * or over pixels if perPixelTargetFind is enabled\n   * @param {FabricObject} obj Object to test against\n   * @param {Object} [pointer] point from viewport.\n   * @return {Boolean} true if point is contained within an area of given object\n   * @private\n   */\n  _checkTarget(obj: FabricObject, pointer: Point): boolean {\n    if (\n      obj &&\n      obj.visible &&\n      obj.evented &&\n      this._pointIsInObjectSelectionArea(\n        obj,\n        sendPointToPlane(pointer, undefined, this.viewportTransform)\n      )\n    ) {\n      if (\n        (this.perPixelTargetFind || obj.perPixelTargetFind) &&\n        !(obj as unknown as IText).isEditing\n      ) {\n        if (!this.isTargetTransparent(obj, pointer.x, pointer.y)) {\n          return true;\n        }\n      } else {\n        return true;\n      }\n    }\n    return false;\n  }\n\n  /**\n   * Internal Function used to search inside objects an object that contains pointer in bounding box or that contains pointerOnCanvas when painted\n   * @param {Array} [objects] objects array to look into\n   * @param {Object} [pointer] x,y object of point coordinates we want to check.\n   * @return {FabricObject} **top most object from given `objects`** that contains pointer\n   * @private\n   */\n  _searchPossibleTargets(\n    objects: FabricObject[],\n    pointer: Point\n  ): FabricObject | undefined {\n    // Cache all targets where their bounding box contains point.\n    let i = objects.length;\n    // Do not check for currently grouped objects, since we check the parent group itself.\n    // until we call this function specifically to search inside the activeGroup\n    while (i--) {\n      const target = objects[i];\n      if (this._checkTarget(target, pointer)) {\n        if (isCollection(target) && target.subTargetCheck) {\n          const subTarget = this._searchPossibleTargets(\n            target._objects as FabricObject[],\n            pointer\n          );\n          subTarget && this.targets.push(subTarget);\n        }\n        return target;\n      }\n    }\n  }\n\n  /**\n   * Function used to search inside objects an object that contains pointer in bounding box or that contains pointerOnCanvas when painted\n   * @see {@link _searchPossibleTargets}\n   * @param {FabricObject[]} [objects] objects array to look into\n   * @param {Point} [pointer] coordinates from viewport to check.\n   * @return {FabricObject} **top most object on screen** that contains pointer\n   */\n  searchPossibleTargets(\n    objects: FabricObject[],\n    pointer: Point\n  ): FabricObject | undefined {\n    const target = this._searchPossibleTargets(objects, pointer);\n\n    // if we found something in this.targets, and the group is interactive, return the innermost subTarget\n    // that is still interactive\n    // TODO: reverify why interactive. the target should be returned always, but selected only\n    // if interactive.\n    if (\n      target &&\n      isCollection(target) &&\n      target.interactive &&\n      this.targets[0]\n    ) {\n      /** targets[0] is the innermost nested target, but it could be inside non interactive groups and so not a selection target */\n      const targets = this.targets;\n      for (let i = targets.length - 1; i > 0; i--) {\n        const t = targets[i];\n        if (!(isCollection(t) && t.interactive)) {\n          // one of the subtargets was not interactive. that is the last subtarget we can return.\n          // we can't dig more deep;\n          return t;\n        }\n      }\n      return targets[0];\n    }\n\n    return target;\n  }\n\n  /**\n   * @returns point existing in the same plane as the {@link HTMLCanvasElement},\n   * `(0, 0)` being the top left corner of the {@link HTMLCanvasElement}.\n   * This means that changes to the {@link viewportTransform} do not change the values of the point\n   * and it remains unchanged from the viewer's perspective.\n   *\n   * @example\n   * const scenePoint = sendPointToPlane(\n   *  this.getViewportPoint(e),\n   *  undefined,\n   *  canvas.viewportTransform\n   * );\n   *\n   */\n  getViewportPoint(e: TPointerEvent) {\n    if (this._pointer) {\n      return this._pointer;\n    }\n    return this.getPointer(e, true);\n  }\n\n  /**\n   * @returns point existing in the scene (the same plane as the plane {@link FabricObject#getCenterPoint} exists in).\n   * This means that changes to the {@link viewportTransform} do not change the values of the point,\n   * however, from the viewer's perspective, the point is changed.\n   *\n   * @example\n   * const viewportPoint = sendPointToPlane(\n   *  this.getScenePoint(e),\n   *  canvas.viewportTransform\n   * );\n   *\n   */\n  getScenePoint(e: TPointerEvent) {\n    if (this._absolutePointer) {\n      return this._absolutePointer;\n    }\n    return this.getPointer(e);\n  }\n\n  /**\n   * Returns pointer relative to canvas.\n   *\n   * @deprecated This method is deprecated since v6 to protect you from misuse.\n   * Use {@link getViewportPoint} or {@link getScenePoint} instead.\n   *\n   * @param {Event} e\n   * @param {Boolean} [fromViewport] whether to return the point from the viewport or in the scene\n   * @return {Point}\n   */\n  getPointer(e: TPointerEvent, fromViewport = false): Point {\n    const upperCanvasEl = this.upperCanvasEl,\n      bounds = upperCanvasEl.getBoundingClientRect();\n    let pointer = getPointer(e),\n      boundsWidth = bounds.width || 0,\n      boundsHeight = bounds.height || 0;\n\n    if (!boundsWidth || !boundsHeight) {\n      if (TOP in bounds && BOTTOM in bounds) {\n        boundsHeight = Math.abs(bounds.top - bounds.bottom);\n      }\n      if (RIGHT in bounds && LEFT in bounds) {\n        boundsWidth = Math.abs(bounds.right - bounds.left);\n      }\n    }\n\n    this.calcOffset();\n    pointer.x = pointer.x - this._offset.left;\n    pointer.y = pointer.y - this._offset.top;\n    if (!fromViewport) {\n      pointer = sendPointToPlane(pointer, undefined, this.viewportTransform);\n    }\n\n    const retinaScaling = this.getRetinaScaling();\n    if (retinaScaling !== 1) {\n      pointer.x /= retinaScaling;\n      pointer.y /= retinaScaling;\n    }\n\n    // If bounds are not available (i.e. not visible), do not apply scale.\n    const cssScale =\n      boundsWidth === 0 || boundsHeight === 0\n        ? new Point(1, 1)\n        : new Point(\n            upperCanvasEl.width / boundsWidth,\n            upperCanvasEl.height / boundsHeight\n          );\n\n    return pointer.multiply(cssScale);\n  }\n\n  /**\n   * Internal use only\n   * @protected\n   */\n  protected _setDimensionsImpl(\n    dimensions: TSize,\n    options?: TCanvasSizeOptions\n  ) {\n    // @ts-expect-error this method exists in the subclass - should be moved or declared as abstract\n    this._resetTransformEventData();\n    super._setDimensionsImpl(dimensions, options);\n    if (this._isCurrentlyDrawing) {\n      this.freeDrawingBrush &&\n        this.freeDrawingBrush._setBrushStyles(this.contextTop);\n    }\n  }\n\n  protected _createCacheCanvas() {\n    this.pixelFindCanvasEl = createCanvasElement();\n    this.pixelFindContext = this.pixelFindCanvasEl.getContext('2d', {\n      willReadFrequently: true,\n    })!;\n    this.setTargetFindTolerance(this.targetFindTolerance);\n  }\n\n  /**\n   * Returns context of top canvas where interactions are drawn\n   * @returns {CanvasRenderingContext2D}\n   */\n  getTopContext(): CanvasRenderingContext2D {\n    return this.elements.upper.ctx;\n  }\n\n  /**\n   * Returns context of canvas where object selection is drawn\n   * @alias\n   * @return {CanvasRenderingContext2D}\n   */\n  getSelectionContext(): CanvasRenderingContext2D {\n    return this.elements.upper.ctx;\n  }\n\n  /**\n   * Returns &lt;canvas> element on which object selection is drawn\n   * @return {HTMLCanvasElement}\n   */\n  getSelectionElement(): HTMLCanvasElement {\n    return this.elements.upper.el;\n  }\n\n  /**\n   * Returns currently active object\n   * @return {FabricObject | null} active object\n   */\n  getActiveObject(): FabricObject | undefined {\n    return this._activeObject;\n  }\n\n  /**\n   * Returns an array with the current selected objects\n   * @return {FabricObject[]} active objects array\n   */\n  getActiveObjects(): FabricObject[] {\n    const active = this._activeObject;\n    return isActiveSelection(active)\n      ? active.getObjects()\n      : active\n      ? [active]\n      : [];\n  }\n\n  /**\n   * @private\n   * Compares the old activeObject with the current one and fires correct events\n   * @param {FabricObject[]} oldObjects old activeObject\n   * @param {TPointerEvent} e mouse event triggering the selection events\n   */\n  _fireSelectionEvents(oldObjects: FabricObject[], e?: TPointerEvent) {\n    let somethingChanged = false,\n      invalidate = false;\n    const objects = this.getActiveObjects(),\n      added: FabricObject[] = [],\n      removed: FabricObject[] = [];\n\n    oldObjects.forEach((target) => {\n      if (!objects.includes(target)) {\n        somethingChanged = true;\n        target.fire('deselected', {\n          e,\n          target,\n        });\n        removed.push(target);\n      }\n    });\n\n    objects.forEach((target) => {\n      if (!oldObjects.includes(target)) {\n        somethingChanged = true;\n        target.fire('selected', {\n          e,\n          target,\n        });\n        added.push(target);\n      }\n    });\n\n    if (oldObjects.length > 0 && objects.length > 0) {\n      invalidate = true;\n      somethingChanged &&\n        this.fire('selection:updated', {\n          e,\n          selected: added,\n          deselected: removed,\n        });\n    } else if (objects.length > 0) {\n      invalidate = true;\n      this.fire('selection:created', {\n        e,\n        selected: added,\n      });\n    } else if (oldObjects.length > 0) {\n      invalidate = true;\n      this.fire('selection:cleared', {\n        e,\n        deselected: removed,\n      });\n    }\n    invalidate && (this._objectsToRender = undefined);\n  }\n\n  /**\n   * Sets given object as the only active object on canvas\n   * @param {FabricObject} object Object to set as an active one\n   * @param {TPointerEvent} [e] Event (passed along when firing \"object:selected\")\n   * @return {Boolean} true if the object has been selected\n   */\n  setActiveObject(object: FabricObject, e?: TPointerEvent) {\n    // we can't inline this, since _setActiveObject will change what getActiveObjects returns\n    const currentActives = this.getActiveObjects();\n    const selected = this._setActiveObject(object, e);\n    this._fireSelectionEvents(currentActives, e);\n    return selected;\n  }\n\n  /**\n   * This is supposed to be equivalent to setActiveObject but without firing\n   * any event. There is commitment to have this stay this way.\n   * This is the functional part of setActiveObject.\n   * @param {Object} object to set as active\n   * @param {Event} [e] Event (passed along when firing \"object:selected\")\n   * @return {Boolean} true if the object has been selected\n   */\n  _setActiveObject(object: FabricObject, e?: TPointerEvent) {\n    const prevActiveObject = this._activeObject;\n    if (prevActiveObject === object) {\n      return false;\n    }\n    // after calling this._discardActiveObject, this,_activeObject could be undefined\n    if (!this._discardActiveObject(e, object) && this._activeObject) {\n      // refused to deselect\n      return false;\n    }\n    if (object.onSelect({ e })) {\n      return false;\n    }\n\n    this._activeObject = object;\n\n    if (isActiveSelection(object) && prevActiveObject !== object) {\n      object.set('canvas', this);\n      object.setCoords();\n    }\n\n    return true;\n  }\n\n  /**\n   * This is supposed to be equivalent to discardActiveObject but without firing\n   * any selection events ( can still fire object transformation events ). There is commitment to have this stay this way.\n   * This is the functional part of discardActiveObject.\n   * @param {Event} [e] Event (passed along when firing \"object:deselected\")\n   * @param {Object} object the next object to set as active, reason why we are discarding this\n   * @return {Boolean} true if the active object has been discarded\n   */\n  _discardActiveObject(\n    e?: TPointerEvent,\n    object?: FabricObject\n  ): this is { _activeObject: undefined } {\n    const obj = this._activeObject;\n    if (obj) {\n      // onDeselect return TRUE to cancel selection;\n      if (obj.onDeselect({ e, object })) {\n        return false;\n      }\n      if (this._currentTransform && this._currentTransform.target === obj) {\n        this.endCurrentTransform(e);\n      }\n      this._activeObject = undefined;\n      return true;\n    }\n    return false;\n  }\n\n  /**\n   * Discards currently active object and fire events. If the function is called by fabric\n   * as a consequence of a mouse event, the event is passed as a parameter and\n   * sent to the fire function for the custom events. When used as a method the\n   * e param does not have any application.\n   * @param {event} e\n   * @return {Boolean} true if the active object has been discarded\n   */\n  discardActiveObject(e?: TPointerEvent): this is { _activeObject: undefined } {\n    const currentActives = this.getActiveObjects(),\n      activeObject = this.getActiveObject();\n    if (currentActives.length) {\n      this.fire('before:selection:cleared', {\n        e,\n        deselected: [activeObject!],\n      });\n    }\n    const discarded = this._discardActiveObject(e);\n    this._fireSelectionEvents(currentActives, e);\n    return discarded;\n  }\n\n  /**\n   * End the current transform.\n   * You don't usually need to call this method unless you are interrupting a user initiated transform\n   * because of some other event ( a press of key combination, or something that block the user UX )\n   * @param {Event} [e] send the mouse event that generate the finalize down, so it can be used in the event\n   */\n  endCurrentTransform(e?: TPointerEvent) {\n    const transform = this._currentTransform;\n    this._finalizeCurrentTransform(e);\n    if (transform && transform.target) {\n      // this could probably go inside _finalizeCurrentTransform\n      transform.target.isMoving = false;\n    }\n    this._currentTransform = null;\n  }\n\n  /**\n   * @private\n   * @param {Event} e send the mouse event that generate the finalize down, so it can be used in the event\n   */\n  _finalizeCurrentTransform(e?: TPointerEvent) {\n    const transform = this._currentTransform!,\n      target = transform.target,\n      options = {\n        e,\n        target,\n        transform,\n        action: transform.action,\n      };\n\n    if (target._scaling) {\n      target._scaling = false;\n    }\n\n    target.setCoords();\n\n    if (transform.actionPerformed) {\n      this.fire('object:modified', options);\n      target.fire('modified', options);\n    }\n  }\n\n  /**\n   * Sets viewport transformation of this canvas instance\n   * @param {Array} vpt a Canvas 2D API transform matrix\n   */\n  setViewportTransform(vpt: TMat2D) {\n    super.setViewportTransform(vpt);\n    const activeObject = this._activeObject;\n    if (activeObject) {\n      activeObject.setCoords();\n    }\n  }\n\n  /**\n   * @override clears active selection ref and interactive canvas elements and contexts\n   */\n  destroy() {\n    // dispose of active selection\n    const activeObject = this._activeObject;\n    if (isActiveSelection(activeObject)) {\n      activeObject.removeAll();\n      activeObject.dispose();\n    }\n\n    delete this._activeObject;\n\n    super.destroy();\n\n    // free resources\n\n    // pixel find canvas\n    // @ts-expect-error disposing\n    this.pixelFindContext = null;\n    // @ts-expect-error disposing\n    this.pixelFindCanvasEl = undefined;\n  }\n\n  /**\n   * Clears all contexts (background, main, top) of an instance\n   */\n  clear() {\n    // discard active object and fire events\n    this.discardActiveObject();\n    // make sure we clear the active object in case it refused to be discarded\n    this._activeObject = undefined;\n    this.clearContext(this.contextTop);\n    super.clear();\n  }\n\n  /**\n   * Draws objects' controls (borders/controls)\n   * @param {CanvasRenderingContext2D} ctx Context to render controls on\n   */\n  drawControls(ctx: CanvasRenderingContext2D) {\n    const activeObject = this._activeObject;\n\n    if (activeObject) {\n      activeObject._renderControls(ctx);\n    }\n  }\n\n  /**\n   * @private\n   */\n  protected _toObject(\n    instance: FabricObject,\n    methodName: 'toObject' | 'toDatalessObject',\n    propertiesToInclude: string[]\n  ): Record<string, any> {\n    // If the object is part of the current selection group, it should\n    // be transformed appropriately\n    // i.e. it should be serialised as it would appear if the selection group\n    // were to be destroyed.\n    const originalProperties = this._realizeGroupTransformOnObject(instance),\n      object = super._toObject(instance, methodName, propertiesToInclude);\n    //Undo the damage we did by changing all of its properties\n    instance.set(originalProperties);\n    return object;\n  }\n\n  /**\n   * Realizes an object's group transformation on it\n   * @private\n   * @param {FabricObject} [instance] the object to transform (gets mutated)\n   * @returns the original values of instance which were changed\n   */\n  private _realizeGroupTransformOnObject(\n    instance: FabricObject\n  ): Partial<typeof instance> {\n    const { group } = instance;\n    if (group && isActiveSelection(group) && this._activeObject === group) {\n      const layoutProps = [\n        'angle',\n        'flipX',\n        'flipY',\n        LEFT,\n        'scaleX',\n        'scaleY',\n        'skewX',\n        'skewY',\n        TOP,\n      ] as (keyof typeof instance)[];\n      const originalValues = pick<typeof instance>(instance, layoutProps);\n      addTransformToObject(instance, group.calcOwnMatrix());\n      return originalValues;\n    } else {\n      return {};\n    }\n  }\n\n  /**\n   * @private\n   */\n  _setSVGObject(\n    markup: string[],\n    instance: FabricObject,\n    reviver: TSVGReviver\n  ) {\n    // If the object is in a selection group, simulate what would happen to that\n    // object when the group is deselected\n    const originalProperties = this._realizeGroupTransformOnObject(instance);\n    super._setSVGObject(markup, instance, reviver);\n    instance.set(originalProperties);\n  }\n}\n","import type { ModifierKey, TOptionalModifierKey } from '../EventTypeDefs';\nimport type { TOptions } from '../typedefs';\nimport type { StaticCanvasOptions } from './StaticCanvasOptions';\n\nexport interface CanvasTransformOptions {\n  /**\n   * When true, objects can be transformed by one side (unproportionately)\n   * when dragged on the corners that normally would not do that.\n   * @type Boolean\n   * @default\n   * @since fabric 4.0 // changed name and default value\n   */\n  uniformScaling: boolean;\n\n  /**\n   * Indicates which key switches uniform scaling.\n   * values: 'altKey', 'shiftKey', 'ctrlKey'.\n   * If `null` or 'none' or any other string that is not a modifier key\n   * feature is disabled.\n   * totally wrong named. this sounds like `uniform scaling`\n   * if Canvas.uniformScaling is true, pressing this will set it to false\n   * and viceversa.\n   * @since 1.6.2\n   * @type ModifierKey\n   * @default\n   */\n  uniScaleKey: TOptionalModifierKey;\n\n  /**\n   * When true, objects use center point as the origin of scale transformation.\n   * <b>Backwards incompatibility note:</b> This property replaces \"centerTransform\" (Boolean).\n   * @since 1.3.4\n   * @type Boolean\n   * @default\n   */\n  centeredScaling: boolean;\n\n  /**\n   * When true, objects use center point as the origin of rotate transformation.\n   * <b>Backwards incompatibility note:</b> This property replaces \"centerTransform\" (Boolean).\n   * @since 1.3.4\n   * @type Boolean\n   * @default\n   */\n  centeredRotation: boolean;\n\n  /**\n   * Indicates which key enable centered Transform\n   * values: 'altKey', 'shiftKey', 'ctrlKey'.\n   * If `null` or 'none' or any other string that is not a modifier key\n   * feature is disabled feature disabled.\n   * @since 1.6.2\n   * @type ModifierKey\n   * @default\n   */\n  centeredKey: TOptionalModifierKey;\n\n  /**\n   * Indicates which key enable alternate action on corner\n   * values: 'altKey', 'shiftKey', 'ctrlKey'.\n   * If `null` or 'none' or any other string that is not a modifier key\n   * feature is disabled feature disabled.\n   * @since 1.6.2\n   * @type ModifierKey\n   * @default\n   */\n  altActionKey: TOptionalModifierKey;\n}\n\nexport interface CanvasSelectionOptions {\n  /**\n   * Indicates whether group selection should be enabled\n   * @type Boolean\n   * @default\n   */\n  selection: boolean;\n\n  /**\n   * Indicates which key or keys enable multiple click selection\n   * Pass value as a string or array of strings\n   * values: 'altKey', 'shiftKey', 'ctrlKey'.\n   * If `null` or empty or containing any other string that is not a modifier key\n   * feature is disabled.\n   * @since 1.6.2\n   * @type ModifierKey|ModifierKey[]\n   * @default\n   */\n  selectionKey: TOptionalModifierKey | ModifierKey[];\n\n  /**\n   * Indicates which key enable alternative selection\n   * in case of target overlapping with active object\n   * values: 'altKey', 'shiftKey', 'ctrlKey'.\n   * For a series of reason that come from the general expectations on how\n   * things should work, this feature works only for preserveObjectStacking true.\n   * If `null` or 'none' or any other string that is not a modifier key\n   * feature is disabled.\n   * @since 1.6.5\n   * @type null|ModifierKey\n   * @default\n   */\n  altSelectionKey: TOptionalModifierKey;\n\n  /**\n   * Color of selection\n   * @type String\n   * @default\n   */\n  selectionColor: string;\n\n  /**\n   * Default dash array pattern\n   * If not empty the selection border is dashed\n   * @type Array\n   */\n  selectionDashArray: number[];\n\n  /**\n   * Color of the border of selection (usually slightly darker than color of selection itself)\n   * @type String\n   * @default\n   */\n  selectionBorderColor: string;\n\n  /**\n   * Width of a line used in object/group selection\n   * @type Number\n   * @default\n   */\n  selectionLineWidth: number;\n\n  /**\n   * Select only shapes that are fully contained in the dragged selection rectangle.\n   * @type Boolean\n   * @default\n   */\n  selectionFullyContained: boolean;\n}\n\nexport interface CanvasCursorOptions {\n  /**\n   * Default cursor value used when hovering over an object on canvas\n   * @type CSSStyleDeclaration['cursor']\n   * @default move\n   */\n  hoverCursor: CSSStyleDeclaration['cursor'];\n\n  /**\n   * Default cursor value used when moving an object on canvas\n   * @type CSSStyleDeclaration['cursor']\n   * @default move\n   */\n  moveCursor: CSSStyleDeclaration['cursor'];\n\n  /**\n   * Default cursor value used for the entire canvas\n   * @type String\n   * @default default\n   */\n  defaultCursor: CSSStyleDeclaration['cursor'];\n\n  /**\n   * Cursor value used during free drawing\n   * @type String\n   * @default crosshair\n   */\n  freeDrawingCursor: CSSStyleDeclaration['cursor'];\n\n  /**\n   * Cursor value used for disabled elements ( corners with disabled action )\n   * @type String\n   * @since 2.0.0\n   * @default not-allowed\n   */\n  notAllowedCursor: CSSStyleDeclaration['cursor'];\n}\n\nexport interface TargetFindOptions {\n  /**\n   * When true, object detection happens on per-pixel basis rather than on per-bounding-box\n   * @type Boolean\n   * @default\n   */\n  perPixelTargetFind: boolean;\n\n  /**\n   * Number of pixels around target pixel to tolerate (consider active) during object detection\n   * @type Number\n   * @default\n   */\n  targetFindTolerance: number;\n\n  /**\n   * When true, target detection is skipped. Target detection will return always undefined.\n   * click selection won't work anymore, events will fire with no targets.\n   * if something is selected before setting it to true, it will be deselected at the first click.\n   * area selection will still work. check the `selection` property too.\n   * if you deactivate both, you should look into staticCanvas.\n   * @type Boolean\n   * @default\n   */\n  skipTargetFind: boolean;\n}\n\nexport interface CanvasEventsOptions {\n  /**\n   * Indicates if the right click on canvas can output the context menu or not\n   * @type Boolean\n   * @since 1.6.5\n   * @default\n   */\n  stopContextMenu: boolean;\n\n  /**\n   * Indicates if the canvas can fire right click events\n   * @type Boolean\n   * @since 1.6.5\n   * @default\n   */\n  fireRightClick: boolean;\n\n  /**\n   * Indicates if the canvas can fire middle click events\n   * @type Boolean\n   * @since 1.7.8\n   * @default\n   */\n  fireMiddleClick: boolean;\n\n  /**\n   * When the option is enabled, PointerEvent is used instead of TPointerEvent.\n   * @type Boolean\n   * @default\n   */\n  enablePointerEvents: boolean;\n}\n\nexport interface CanvasOptions\n  extends StaticCanvasOptions,\n    CanvasTransformOptions,\n    CanvasSelectionOptions,\n    CanvasCursorOptions,\n    TargetFindOptions,\n    CanvasEventsOptions {\n  /**\n   * Default element class that's given to wrapper (div) element of canvas\n   * @type String\n   * @default\n   * @deprecated customize {@link CanvasDOMManager} instead or access {@link elements} directly\n   */\n  containerClass: string;\n\n  /**\n   * Indicates whether objects should remain in current stack position when selected.\n   * When false objects are brought to top and rendered as part of the selection group\n   * @type Boolean\n   * @default\n   */\n  preserveObjectStacking: boolean;\n}\n\nexport type TCanvasOptions = TOptions<CanvasOptions>;\n\nexport const canvasDefaults: TOptions<CanvasOptions> = {\n  uniformScaling: true,\n  uniScaleKey: 'shiftKey',\n  centeredScaling: false,\n  centeredRotation: false,\n  centeredKey: 'altKey',\n  altActionKey: 'shiftKey',\n\n  selection: true,\n  selectionKey: 'shiftKey',\n  selectionColor: 'rgba(100, 100, 255, 0.3)',\n  selectionDashArray: [],\n  selectionBorderColor: 'rgba(255, 255, 255, 0.3)',\n  selectionLineWidth: 1,\n  selectionFullyContained: false,\n\n  hoverCursor: 'move',\n  moveCursor: 'move',\n  defaultCursor: 'default',\n  freeDrawingCursor: 'crosshair',\n  notAllowedCursor: 'not-allowed',\n\n  perPixelTargetFind: false,\n  targetFindTolerance: 0,\n  skipTargetFind: false,\n\n  stopContextMenu: false,\n  fireRightClick: false,\n  fireMiddleClick: false,\n  enablePointerEvents: false,\n\n  containerClass: 'canvas-container',\n\n  preserveObjectStacking: false,\n};\n","import type { TPointerEvent } from '../EventTypeDefs';\nimport type { ITextBehavior } from '../shapes/IText/ITextBehavior';\nimport { removeFromArray } from '../util/internals';\nimport type { Canvas } from './Canvas';\n\n/**\n * In charge of synchronizing all interactive text instances of a canvas\n */\nexport class TextEditingManager {\n  private targets: ITextBehavior[] = [];\n  private declare target?: ITextBehavior;\n  private __disposer: VoidFunction;\n\n  constructor(canvas: Canvas) {\n    const cb = () => {\n      const { hiddenTextarea } =\n        (canvas.getActiveObject() as ITextBehavior) || {};\n      hiddenTextarea && hiddenTextarea.focus();\n    };\n    const el = canvas.upperCanvasEl;\n    el.addEventListener('click', cb);\n    this.__disposer = () => el.removeEventListener('click', cb);\n  }\n\n  exitTextEditing() {\n    this.target = undefined;\n    this.targets.forEach((target) => {\n      if (target.isEditing) {\n        target.exitEditing();\n      }\n    });\n  }\n\n  add(target: ITextBehavior) {\n    this.targets.push(target);\n  }\n\n  remove(target: ITextBehavior) {\n    this.unregister(target);\n    removeFromArray(this.targets, target);\n  }\n\n  register(target: ITextBehavior) {\n    this.target = target;\n  }\n\n  unregister(target: ITextBehavior) {\n    if (target === this.target) {\n      this.target = undefined;\n    }\n  }\n\n  onMouseMove(e: TPointerEvent) {\n    this.target?.isEditing && this.target.updateSelectionOnMouseMove(e);\n  }\n\n  clear() {\n    this.targets = [];\n    this.target = undefined;\n  }\n\n  dispose() {\n    this.clear();\n    this.__disposer();\n    // @ts-expect-error disposing\n    delete this.__disposer;\n  }\n}\n","import { classRegistry } from '../ClassRegistry';\nimport { NONE } from '../constants';\nimport type {\n  CanvasEvents,\n  DragEventData,\n  ObjectEvents,\n  TPointerEvent,\n  TPointerEventNames,\n  Transform,\n} from '../EventTypeDefs';\nimport { Point } from '../Point';\nimport type { ActiveSelection } from '../shapes/ActiveSelection';\nimport type { Group } from '../shapes/Group';\nimport type { IText } from '../shapes/IText/IText';\nimport type { FabricObject } from '../shapes/Object/FabricObject';\nimport { isTouchEvent, stopEvent } from '../util/dom_event';\nimport { getDocumentFromElement, getWindowFromElement } from '../util/dom_misc';\nimport { sendPointToPlane } from '../util/misc/planeChange';\nimport { isActiveSelection } from '../util/typeAssertions';\nimport type { CanvasOptions, TCanvasOptions } from './CanvasOptions';\nimport { SelectableCanvas } from './SelectableCanvas';\nimport { TextEditingManager } from './TextEditingManager';\n\nconst addEventOptions = { passive: false } as EventListenerOptions;\n\nconst getEventPoints = (canvas: Canvas, e: TPointerEvent) => {\n  const viewportPoint = canvas.getViewportPoint(e);\n  const scenePoint = canvas.getScenePoint(e);\n  return {\n    viewportPoint,\n    scenePoint,\n    pointer: viewportPoint,\n    absolutePointer: scenePoint,\n  };\n};\n\n// just to be clear, the utils are now deprecated and those are here exactly as minifier helpers\n// because el.addEventListener can't me be minified while a const yes and we use it 47 times in this file.\n// few bytes but why give it away.\nconst addListener = (\n  el: HTMLElement | Document,\n  ...args: Parameters<HTMLElement['addEventListener']>\n) => el.addEventListener(...args);\nconst removeListener = (\n  el: HTMLElement | Document,\n  ...args: Parameters<HTMLElement['removeEventListener']>\n) => el.removeEventListener(...args);\n\nconst syntheticEventConfig = {\n  mouse: {\n    in: 'over',\n    out: 'out',\n    targetIn: 'mouseover',\n    targetOut: 'mouseout',\n    canvasIn: 'mouse:over',\n    canvasOut: 'mouse:out',\n  },\n  drag: {\n    in: 'enter',\n    out: 'leave',\n    targetIn: 'dragenter',\n    targetOut: 'dragleave',\n    canvasIn: 'drag:enter',\n    canvasOut: 'drag:leave',\n  },\n} as const;\n\ntype TSyntheticEventContext = {\n  mouse: { e: TPointerEvent };\n  drag: DragEventData;\n};\n\nexport class Canvas extends SelectableCanvas implements CanvasOptions {\n  /**\n   * Contains the id of the touch event that owns the fabric transform\n   * @type Number\n   * @private\n   */\n  declare mainTouchId?: number;\n\n  declare enablePointerEvents: boolean;\n\n  /**\n   * Holds a reference to a setTimeout timer for event synchronization\n   * @type number\n   * @private\n   */\n  private declare _willAddMouseDown: number;\n\n  /**\n   * Holds a reference to an object on the canvas that is receiving the drag over event.\n   * @type FabricObject\n   * @private\n   */\n  private declare _draggedoverTarget?: FabricObject;\n\n  /**\n   * Holds a reference to an object on the canvas from where the drag operation started\n   * @type FabricObject\n   * @private\n   */\n  private declare _dragSource?: FabricObject;\n\n  /**\n   * Holds a reference to an object on the canvas that is the current drop target\n   * May differ from {@link _draggedoverTarget}\n   * @todo inspect whether {@link _draggedoverTarget} and {@link _dropTarget} should be merged somehow\n   * @type FabricObject\n   * @private\n   */\n  private declare _dropTarget: FabricObject<ObjectEvents> | undefined;\n\n  private _isClick: boolean;\n\n  textEditingManager = new TextEditingManager(this);\n\n  constructor(el?: string | HTMLCanvasElement, options: TCanvasOptions = {}) {\n    super(el, options);\n    // bind event handlers\n    (\n      [\n        '_onMouseDown',\n        '_onTouchStart',\n        '_onMouseMove',\n        '_onMouseUp',\n        '_onTouchEnd',\n        '_onResize',\n        // '_onGesture',\n        // '_onDrag',\n        // '_onShake',\n        // '_onLongPress',\n        // '_onOrientationChange',\n        '_onMouseWheel',\n        '_onMouseOut',\n        '_onMouseEnter',\n        '_onContextMenu',\n        '_onDoubleClick',\n        '_onDragStart',\n        '_onDragEnd',\n        '_onDragProgress',\n        '_onDragOver',\n        '_onDragEnter',\n        '_onDragLeave',\n        '_onDrop',\n      ] as (keyof this)[]\n    ).forEach((eventHandler) => {\n      this[eventHandler] = (this[eventHandler] as Function).bind(this);\n    });\n    // register event handlers\n    this.addOrRemove(addListener, 'add');\n  }\n\n  /**\n   * return an event prefix pointer or mouse.\n   * @private\n   */\n  private _getEventPrefix() {\n    return this.enablePointerEvents ? 'pointer' : 'mouse';\n  }\n\n  addOrRemove(functor: any, eventjsFunctor: 'add' | 'remove') {\n    const canvasElement = this.upperCanvasEl,\n      eventTypePrefix = this._getEventPrefix();\n    functor(getWindowFromElement(canvasElement), 'resize', this._onResize);\n    functor(canvasElement, eventTypePrefix + 'down', this._onMouseDown);\n    functor(\n      canvasElement,\n      `${eventTypePrefix}move`,\n      this._onMouseMove,\n      addEventOptions\n    );\n    functor(canvasElement, `${eventTypePrefix}out`, this._onMouseOut);\n    functor(canvasElement, `${eventTypePrefix}enter`, this._onMouseEnter);\n    functor(canvasElement, 'wheel', this._onMouseWheel);\n    functor(canvasElement, 'contextmenu', this._onContextMenu);\n    functor(canvasElement, 'dblclick', this._onDoubleClick);\n    functor(canvasElement, 'dragstart', this._onDragStart);\n    functor(canvasElement, 'dragend', this._onDragEnd);\n    functor(canvasElement, 'dragover', this._onDragOver);\n    functor(canvasElement, 'dragenter', this._onDragEnter);\n    functor(canvasElement, 'dragleave', this._onDragLeave);\n    functor(canvasElement, 'drop', this._onDrop);\n    if (!this.enablePointerEvents) {\n      functor(canvasElement, 'touchstart', this._onTouchStart, addEventOptions);\n    }\n    // if (typeof eventjs !== 'undefined' && eventjsFunctor in eventjs) {\n    //   eventjs[eventjsFunctor](canvasElement, 'gesture', this._onGesture);\n    //   eventjs[eventjsFunctor](canvasElement, 'drag', this._onDrag);\n    //   eventjs[eventjsFunctor](\n    //     canvasElement,\n    //     'orientation',\n    //     this._onOrientationChange\n    //   );\n    //   eventjs[eventjsFunctor](canvasElement, 'shake', this._onShake);\n    //   eventjs[eventjsFunctor](canvasElement, 'longpress', this._onLongPress);\n    // }\n  }\n\n  /**\n   * Removes all event listeners\n   */\n  removeListeners() {\n    this.addOrRemove(removeListener, 'remove');\n    // if you dispose on a mouseDown, before mouse up, you need to clean document to...\n    const eventTypePrefix = this._getEventPrefix();\n    const doc = getDocumentFromElement(this.upperCanvasEl);\n    removeListener(\n      doc,\n      `${eventTypePrefix}up`,\n      this._onMouseUp as EventListener\n    );\n    removeListener(\n      doc,\n      'touchend',\n      this._onTouchEnd as EventListener,\n      addEventOptions\n    );\n    removeListener(\n      doc,\n      `${eventTypePrefix}move`,\n      this._onMouseMove as EventListener,\n      addEventOptions\n    );\n    removeListener(\n      doc,\n      'touchmove',\n      this._onMouseMove as EventListener,\n      addEventOptions\n    );\n  }\n\n  /**\n   * @private\n   * @param {Event} [e] Event object fired on wheel event\n   */\n  private _onMouseWheel(e: MouseEvent) {\n    this.__onMouseWheel(e);\n  }\n\n  /**\n   * @private\n   * @param {Event} e Event object fired on mousedown\n   */\n  private _onMouseOut(e: TPointerEvent) {\n    const target = this._hoveredTarget;\n    const shared = {\n      e,\n      ...getEventPoints(this, e),\n    };\n    this.fire('mouse:out', { ...shared, target });\n    this._hoveredTarget = undefined;\n    target && target.fire('mouseout', { ...shared });\n    this._hoveredTargets.forEach((nestedTarget) => {\n      this.fire('mouse:out', { ...shared, target: nestedTarget });\n      nestedTarget && nestedTarget.fire('mouseout', { ...shared });\n    });\n    this._hoveredTargets = [];\n  }\n\n  /**\n   * @private\n   * @param {Event} e Event object fired on mouseenter\n   */\n  private _onMouseEnter(e: TPointerEvent) {\n    // This find target and consequent 'mouse:over' is used to\n    // clear old instances on hovered target.\n    // calling findTarget has the side effect of killing target.__corner.\n    // as a short term fix we are not firing this if we are currently transforming.\n    // as a long term fix we need to separate the action of finding a target with the\n    // side effects we added to it.\n    if (!this._currentTransform && !this.findTarget(e)) {\n      this.fire('mouse:over', {\n        e,\n        ...getEventPoints(this, e),\n      });\n      this._hoveredTarget = undefined;\n      this._hoveredTargets = [];\n    }\n  }\n\n  /**\n   * supports native like text dragging\n   * @private\n   * @param {DragEvent} e\n   */\n  private _onDragStart(e: DragEvent) {\n    this._isClick = false;\n    const activeObject = this.getActiveObject();\n    if (activeObject && activeObject.onDragStart(e)) {\n      this._dragSource = activeObject;\n      const options = { e, target: activeObject };\n      this.fire('dragstart', options);\n      activeObject.fire('dragstart', options);\n      addListener(\n        this.upperCanvasEl,\n        'drag',\n        this._onDragProgress as EventListener\n      );\n      return;\n    }\n    stopEvent(e);\n  }\n\n  /**\n   * First we clear top context where the effects are being rendered.\n   * Then we render the effects.\n   * Doing so will render the correct effect for all cases including an overlap between `source` and `target`.\n   * @private\n   */\n  private _renderDragEffects(\n    e: DragEvent,\n    source?: FabricObject,\n    target?: FabricObject\n  ) {\n    let dirty = false;\n    // clear top context\n    const dropTarget = this._dropTarget;\n    if (dropTarget && dropTarget !== source && dropTarget !== target) {\n      dropTarget.clearContextTop();\n      dirty = true;\n    }\n    source?.clearContextTop();\n    target !== source && target?.clearContextTop();\n    // render effects\n    const ctx = this.contextTop;\n    ctx.save();\n    ctx.transform(...this.viewportTransform);\n    if (source) {\n      ctx.save();\n      source.transform(ctx);\n      source.renderDragSourceEffect(e);\n      ctx.restore();\n      dirty = true;\n    }\n    if (target) {\n      ctx.save();\n      target.transform(ctx);\n      target.renderDropTargetEffect(e);\n      ctx.restore();\n      dirty = true;\n    }\n    ctx.restore();\n    dirty && (this.contextTopDirty = true);\n  }\n\n  /**\n   * supports native like text dragging\n   * https://developer.mozilla.org/en-US/docs/Web/API/HTML_Drag_and_Drop_API/Drag_operations#finishing_a_drag\n   * @private\n   * @param {DragEvent} e\n   */\n  private _onDragEnd(e: DragEvent) {\n    const didDrop = !!e.dataTransfer && e.dataTransfer.dropEffect !== NONE,\n      dropTarget = didDrop ? this._activeObject : undefined,\n      options = {\n        e,\n        target: this._dragSource as FabricObject,\n        subTargets: this.targets,\n        dragSource: this._dragSource as FabricObject,\n        didDrop,\n        dropTarget: dropTarget as FabricObject,\n      };\n    removeListener(\n      this.upperCanvasEl,\n      'drag',\n      this._onDragProgress as EventListener\n    );\n    this.fire('dragend', options);\n    this._dragSource && this._dragSource.fire('dragend', options);\n    delete this._dragSource;\n    // we need to call mouse up synthetically because the browser won't\n    this._onMouseUp(e);\n  }\n\n  /**\n   * fire `drag` event on canvas and drag source\n   * @private\n   * @param {DragEvent} e\n   */\n  private _onDragProgress(e: DragEvent) {\n    const options = {\n      e,\n      target: this._dragSource as FabricObject | undefined,\n      dragSource: this._dragSource as FabricObject | undefined,\n      dropTarget: this._draggedoverTarget as FabricObject,\n    };\n    this.fire('drag', options);\n    this._dragSource && this._dragSource.fire('drag', options);\n  }\n\n  /**\n   * As opposed to {@link findTarget} we want the top most object to be returned w/o the active object cutting in line.\n   * Override at will\n   */\n  protected findDragTargets(e: DragEvent) {\n    this.targets = [];\n    const target = this._searchPossibleTargets(\n      this._objects,\n      this.getViewportPoint(e)\n    );\n    return {\n      target,\n      targets: [...this.targets],\n    };\n  }\n\n  /**\n   * prevent default to allow drop event to be fired\n   * https://developer.mozilla.org/en-US/docs/Web/API/HTML_Drag_and_Drop_API/Drag_operations#specifying_drop_targets\n   * @private\n   * @param {DragEvent} [e] Event object fired on Event.js shake\n   */\n  private _onDragOver(e: DragEvent) {\n    const eventType = 'dragover';\n    const { target, targets } = this.findDragTargets(e);\n    const dragSource = this._dragSource as FabricObject;\n    const options = {\n      e,\n      target,\n      subTargets: targets,\n      dragSource,\n      canDrop: false,\n      dropTarget: undefined,\n    };\n    let dropTarget;\n    //  fire on canvas\n    this.fire(eventType, options);\n    //  make sure we fire dragenter events before dragover\n    //  if dragleave is needed, object will not fire dragover so we don't need to trouble ourselves with it\n    this._fireEnterLeaveEvents(target, options);\n    if (target) {\n      if (target.canDrop(e)) {\n        dropTarget = target;\n      }\n      target.fire(eventType, options);\n    }\n    //  propagate the event to subtargets\n    for (let i = 0; i < targets.length; i++) {\n      const subTarget = targets[i];\n      // accept event only if previous targets didn't (the accepting target calls `preventDefault` to inform that the event is taken)\n      // TODO: verify if those should loop in inverse order then?\n      // what is the order of subtargets?\n      if (subTarget.canDrop(e)) {\n        dropTarget = subTarget;\n      }\n      subTarget.fire(eventType, options);\n    }\n    //  render drag effects now that relations between source and target is clear\n    this._renderDragEffects(e, dragSource, dropTarget);\n    this._dropTarget = dropTarget;\n  }\n\n  /**\n   * fire `dragleave` on `dragover` targets\n   * @private\n   * @param {Event} [e] Event object fired on Event.js shake\n   */\n  private _onDragEnter(e: DragEvent) {\n    const { target, targets } = this.findDragTargets(e);\n    const options = {\n      e,\n      target,\n      subTargets: targets,\n      dragSource: this._dragSource,\n    };\n    this.fire('dragenter', options);\n    //  fire dragenter on targets\n    this._fireEnterLeaveEvents(target, options);\n  }\n\n  /**\n   * fire `dragleave` on `dragover` targets\n   * @private\n   * @param {Event} [e] Event object fired on Event.js shake\n   */\n  private _onDragLeave(e: DragEvent) {\n    const options = {\n      e,\n      target: this._draggedoverTarget,\n      subTargets: this.targets,\n      dragSource: this._dragSource,\n    };\n    this.fire('dragleave', options);\n\n    //  fire dragleave on targets\n    this._fireEnterLeaveEvents(undefined, options);\n    this._renderDragEffects(e, this._dragSource);\n    this._dropTarget = undefined;\n    //  clear targets\n    this.targets = [];\n    this._hoveredTargets = [];\n  }\n\n  /**\n   * `drop:before` is a an event that allows you to schedule logic\n   * before the `drop` event. Prefer `drop` event always, but if you need\n   * to run some drop-disabling logic on an event, since there is no way\n   * to handle event handlers ordering, use `drop:before`\n   * @private\n   * @param {Event} e\n   */\n  private _onDrop(e: DragEvent) {\n    const { target, targets } = this.findDragTargets(e);\n    const options = this._basicEventHandler('drop:before', {\n      e,\n      target,\n      subTargets: targets,\n      dragSource: this._dragSource,\n      ...getEventPoints(this, e),\n    });\n    //  will be set by the drop target\n    options.didDrop = false;\n    //  will be set by the drop target, used in case options.target refuses the drop\n    options.dropTarget = undefined;\n    //  fire `drop`\n    this._basicEventHandler('drop', options);\n    //  inform canvas of the drop\n    //  we do this because canvas was unaware of what happened at the time the `drop` event was fired on it\n    //  use for side effects\n    this.fire('drop:after', options);\n  }\n\n  /**\n   * @private\n   * @param {Event} e Event object fired on mousedown\n   */\n  private _onContextMenu(e: TPointerEvent): false {\n    const target = this.findTarget(e),\n      subTargets = this.targets || [];\n    const options = this._basicEventHandler('contextmenu:before', {\n      e,\n      target,\n      subTargets,\n    });\n    // TODO: this line is silly because the dev can subscribe to the event and prevent it themselves\n    this.stopContextMenu && stopEvent(e);\n    this._basicEventHandler('contextmenu', options);\n    return false;\n  }\n\n  /**\n   * @private\n   * @param {Event} e Event object fired on mousedown\n   */\n  private _onDoubleClick(e: TPointerEvent) {\n    this._cacheTransformEventData(e);\n    this._handleEvent(e, 'dblclick');\n    this._resetTransformEventData();\n  }\n\n  /**\n   * Return a the id of an event.\n   * returns either the pointerId or the identifier or 0 for the mouse event\n   * @private\n   * @param {Event} evt Event object\n   */\n  getPointerId(evt: TouchEvent | PointerEvent): number {\n    const changedTouches = (evt as TouchEvent).changedTouches;\n\n    if (changedTouches) {\n      return changedTouches[0] && changedTouches[0].identifier;\n    }\n\n    if (this.enablePointerEvents) {\n      return (evt as PointerEvent).pointerId;\n    }\n\n    return -1;\n  }\n\n  /**\n   * Determines if an event has the id of the event that is considered main\n   * @private\n   * @param {evt} event Event object\n   */\n  _isMainEvent(evt: TPointerEvent): boolean {\n    if ((evt as PointerEvent).isPrimary === true) {\n      return true;\n    }\n    if ((evt as PointerEvent).isPrimary === false) {\n      return false;\n    }\n    if (evt.type === 'touchend' && (evt as TouchEvent).touches.length === 0) {\n      return true;\n    }\n    if ((evt as TouchEvent).changedTouches) {\n      return (\n        (evt as TouchEvent).changedTouches[0].identifier === this.mainTouchId\n      );\n    }\n    return true;\n  }\n\n  /**\n   * @private\n   * @param {Event} e Event object fired on mousedown\n   */\n  _onTouchStart(e: TouchEvent) {\n    e.preventDefault();\n    if (this.mainTouchId === undefined) {\n      this.mainTouchId = this.getPointerId(e);\n    }\n    this.__onMouseDown(e);\n    this._resetTransformEventData();\n    const canvasElement = this.upperCanvasEl,\n      eventTypePrefix = this._getEventPrefix();\n    const doc = getDocumentFromElement(canvasElement);\n    addListener(\n      doc,\n      'touchend',\n      this._onTouchEnd as EventListener,\n      addEventOptions\n    );\n    addListener(\n      doc,\n      'touchmove',\n      this._onMouseMove as EventListener,\n      addEventOptions\n    );\n    // Unbind mousedown to prevent double triggers from touch devices\n    removeListener(\n      canvasElement,\n      `${eventTypePrefix}down`,\n      this._onMouseDown as EventListener\n    );\n  }\n\n  /**\n   * @private\n   * @param {Event} e Event object fired on mousedown\n   */\n  _onMouseDown(e: TPointerEvent) {\n    this.__onMouseDown(e);\n    this._resetTransformEventData();\n    const canvasElement = this.upperCanvasEl,\n      eventTypePrefix = this._getEventPrefix();\n    removeListener(\n      canvasElement,\n      `${eventTypePrefix}move`,\n      this._onMouseMove as EventListener,\n      addEventOptions\n    );\n    const doc = getDocumentFromElement(canvasElement);\n    addListener(doc, `${eventTypePrefix}up`, this._onMouseUp as EventListener);\n    addListener(\n      doc,\n      `${eventTypePrefix}move`,\n      this._onMouseMove as EventListener,\n      addEventOptions\n    );\n  }\n\n  /**\n   * @private\n   * @param {Event} e Event object fired on mousedown\n   */\n  _onTouchEnd(e: TouchEvent) {\n    if (e.touches.length > 0) {\n      // if there are still touches stop here\n      return;\n    }\n    this.__onMouseUp(e);\n    this._resetTransformEventData();\n    delete this.mainTouchId;\n    const eventTypePrefix = this._getEventPrefix();\n    const doc = getDocumentFromElement(this.upperCanvasEl);\n    removeListener(\n      doc,\n      'touchend',\n      this._onTouchEnd as EventListener,\n      addEventOptions\n    );\n    removeListener(\n      doc,\n      'touchmove',\n      this._onMouseMove as EventListener,\n      addEventOptions\n    );\n    if (this._willAddMouseDown) {\n      clearTimeout(this._willAddMouseDown);\n    }\n    this._willAddMouseDown = setTimeout(() => {\n      // Wait 400ms before rebinding mousedown to prevent double triggers\n      // from touch devices\n      addListener(\n        this.upperCanvasEl,\n        `${eventTypePrefix}down`,\n        this._onMouseDown as EventListener\n      );\n      this._willAddMouseDown = 0;\n    }, 400) as unknown as number;\n  }\n\n  /**\n   * @private\n   * @param {Event} e Event object fired on mouseup\n   */\n  _onMouseUp(e: TPointerEvent) {\n    this.__onMouseUp(e);\n    this._resetTransformEventData();\n    const canvasElement = this.upperCanvasEl,\n      eventTypePrefix = this._getEventPrefix();\n    if (this._isMainEvent(e)) {\n      const doc = getDocumentFromElement(this.upperCanvasEl);\n      removeListener(\n        doc,\n        `${eventTypePrefix}up`,\n        this._onMouseUp as EventListener\n      );\n      removeListener(\n        doc,\n        `${eventTypePrefix}move`,\n        this._onMouseMove as EventListener,\n        addEventOptions\n      );\n      addListener(\n        canvasElement,\n        `${eventTypePrefix}move`,\n        this._onMouseMove as EventListener,\n        addEventOptions\n      );\n    }\n  }\n\n  /**\n   * @private\n   * @param {Event} e Event object fired on mousemove\n   */\n  _onMouseMove(e: TPointerEvent) {\n    const activeObject = this.getActiveObject();\n    !this.allowTouchScrolling &&\n      (!activeObject ||\n        // a drag event sequence is started by the active object flagging itself on mousedown / mousedown:before\n        // we must not prevent the event's default behavior in order for the window to start dragging\n        !activeObject.shouldStartDragging(e)) &&\n      e.preventDefault &&\n      e.preventDefault();\n    this.__onMouseMove(e);\n  }\n\n  /**\n   * @private\n   */\n  _onResize() {\n    this.calcOffset();\n    this._resetTransformEventData();\n  }\n\n  /**\n   * Decides whether the canvas should be redrawn in mouseup and mousedown events.\n   * @private\n   * @param {Object} target\n   */\n  _shouldRender(target: FabricObject | undefined) {\n    const activeObject = this.getActiveObject();\n    // if just one of them is available or if they are both but are different objects\n    // this covers: switch of target, from target to no target, selection of target\n    // multiSelection with key and mouse\n    return (\n      !!activeObject !== !!target ||\n      (activeObject && target && activeObject !== target)\n    );\n  }\n\n  /**\n   * Method that defines the actions when mouse is released on canvas.\n   * The method resets the currentTransform parameters, store the image corner\n   * position in the image object and render the canvas on top.\n   * @private\n   * @param {Event} e Event object fired on mouseup\n   */\n  __onMouseUp(e: TPointerEvent) {\n    this._cacheTransformEventData(e);\n    this._handleEvent(e, 'up:before');\n\n    const transform = this._currentTransform;\n    const isClick = this._isClick;\n    const target = this._target;\n\n    // if right/middle click just fire events and return\n    // target undefined will make the _handleEvent search the target\n    const { button } = e as MouseEvent;\n    if (button) {\n      ((this.fireMiddleClick && button === 1) ||\n        (this.fireRightClick && button === 2)) &&\n        this._handleEvent(e, 'up');\n      this._resetTransformEventData();\n      return;\n    }\n\n    if (this.isDrawingMode && this._isCurrentlyDrawing) {\n      this._onMouseUpInDrawingMode(e);\n      return;\n    }\n\n    if (!this._isMainEvent(e)) {\n      return;\n    }\n    let shouldRender = false;\n    if (transform) {\n      this._finalizeCurrentTransform(e);\n      shouldRender = transform.actionPerformed;\n    }\n    if (!isClick) {\n      const targetWasActive = target === this._activeObject;\n      this.handleSelection(e);\n      if (!shouldRender) {\n        shouldRender =\n          this._shouldRender(target) ||\n          (!targetWasActive && target === this._activeObject);\n      }\n    }\n    let pointer, corner;\n    if (target) {\n      const found = target.findControl(\n        this.getViewportPoint(e),\n        isTouchEvent(e)\n      );\n      const { key, control } = found || {};\n      corner = key;\n      if (\n        target.selectable &&\n        target !== this._activeObject &&\n        target.activeOn === 'up'\n      ) {\n        this.setActiveObject(target, e);\n        shouldRender = true;\n      } else if (control) {\n        const mouseUpHandler = control.getMouseUpHandler(e, target, control);\n        if (mouseUpHandler) {\n          pointer = this.getScenePoint(e);\n          mouseUpHandler.call(control, e, transform!, pointer.x, pointer.y);\n        }\n      }\n      target.isMoving = false;\n    }\n    // if we are ending up a transform on a different control or a new object\n    // fire the original mouse up from the corner that started the transform\n    if (\n      transform &&\n      (transform.target !== target || transform.corner !== corner)\n    ) {\n      const originalControl =\n          transform.target && transform.target.controls[transform.corner],\n        originalMouseUpHandler =\n          originalControl &&\n          originalControl.getMouseUpHandler(\n            e,\n            transform.target,\n            originalControl\n          );\n      pointer = pointer || this.getScenePoint(e);\n      originalMouseUpHandler &&\n        originalMouseUpHandler.call(\n          originalControl,\n          e,\n          transform,\n          pointer.x,\n          pointer.y\n        );\n    }\n    this._setCursorFromEvent(e, target);\n    this._handleEvent(e, 'up');\n    this._groupSelector = null;\n    this._currentTransform = null;\n    // reset the target information about which corner is selected\n    target && (target.__corner = undefined);\n    if (shouldRender) {\n      this.requestRenderAll();\n    } else if (!isClick && !(this._activeObject as IText)?.isEditing) {\n      this.renderTop();\n    }\n  }\n\n  _basicEventHandler<T extends keyof (CanvasEvents | ObjectEvents)>(\n    eventType: T,\n    options: (CanvasEvents & ObjectEvents)[T]\n  ) {\n    const { target, subTargets = [] } = options as {\n      target?: FabricObject;\n      subTargets: FabricObject[];\n    };\n    this.fire(eventType, options);\n    target && target.fire(eventType, options);\n    for (let i = 0; i < subTargets.length; i++) {\n      subTargets[i] !== target && subTargets[i].fire(eventType, options);\n    }\n    return options;\n  }\n\n  /**\n   * @private\n   * Handle event firing for target and subtargets\n   * @param {TPointerEvent} e event from mouse\n   * @param {TPointerEventNames} eventType\n   */\n  _handleEvent<T extends TPointerEventNames>(e: TPointerEvent, eventType: T) {\n    const target = this._target,\n      targets = this.targets || [],\n      options: CanvasEvents[`mouse:${T}`] = {\n        e,\n        target,\n        subTargets: targets,\n        ...getEventPoints(this, e),\n        transform: this._currentTransform,\n        ...(eventType === 'up:before' || eventType === 'up'\n          ? {\n              isClick: this._isClick,\n              currentTarget: this.findTarget(e),\n              // set by the preceding `findTarget` call\n              currentSubTargets: this.targets,\n            }\n          : {}),\n      } as CanvasEvents[`mouse:${T}`];\n    this.fire(`mouse:${eventType}`, options);\n    // this may be a little be more complicated of what we want to handle\n    target && target.fire(`mouse${eventType}`, options);\n    for (let i = 0; i < targets.length; i++) {\n      targets[i] !== target && targets[i].fire(`mouse${eventType}`, options);\n    }\n  }\n\n  /**\n   * @private\n   * @param {Event} e Event object fired on mousedown\n   */\n  _onMouseDownInDrawingMode(e: TPointerEvent) {\n    this._isCurrentlyDrawing = true;\n    if (this.getActiveObject()) {\n      this.discardActiveObject(e);\n      this.requestRenderAll();\n    }\n    // TODO: this is a scene point so it should be renamed\n    const pointer = this.getScenePoint(e);\n    this.freeDrawingBrush &&\n      this.freeDrawingBrush.onMouseDown(pointer, { e, pointer });\n    this._handleEvent(e, 'down');\n  }\n\n  /**\n   * @private\n   * @param {Event} e Event object fired on mousemove\n   */\n  _onMouseMoveInDrawingMode(e: TPointerEvent) {\n    if (this._isCurrentlyDrawing) {\n      const pointer = this.getScenePoint(e);\n      this.freeDrawingBrush &&\n        this.freeDrawingBrush.onMouseMove(pointer, {\n          e,\n          // this is an absolute pointer, the naming is wrong\n          pointer,\n        });\n    }\n    this.setCursor(this.freeDrawingCursor);\n    this._handleEvent(e, 'move');\n  }\n\n  /**\n   * @private\n   * @param {Event} e Event object fired on mouseup\n   */\n  _onMouseUpInDrawingMode(e: TPointerEvent) {\n    const pointer = this.getScenePoint(e);\n    if (this.freeDrawingBrush) {\n      this._isCurrentlyDrawing = !!this.freeDrawingBrush.onMouseUp({\n        e: e,\n        // this is an absolute pointer, the naming is wrong\n        pointer,\n      });\n    } else {\n      this._isCurrentlyDrawing = false;\n    }\n    this._handleEvent(e, 'up');\n  }\n\n  /**\n   * Method that defines the actions when mouse is clicked on canvas.\n   * The method inits the currentTransform parameters and renders all the\n   * canvas so the current image can be placed on the top canvas and the rest\n   * in on the container one.\n   * @private\n   * @param {Event} e Event object fired on mousedown\n   */\n  __onMouseDown(e: TPointerEvent) {\n    this._isClick = true;\n    this._cacheTransformEventData(e);\n    this._handleEvent(e, 'down:before');\n\n    let target: FabricObject | undefined = this._target;\n\n    // if right/middle click just fire events\n    const { button } = e as MouseEvent;\n    if (button) {\n      ((this.fireMiddleClick && button === 1) ||\n        (this.fireRightClick && button === 2)) &&\n        this._handleEvent(e, 'down');\n      this._resetTransformEventData();\n      return;\n    }\n\n    if (this.isDrawingMode) {\n      this._onMouseDownInDrawingMode(e);\n      return;\n    }\n\n    if (!this._isMainEvent(e)) {\n      return;\n    }\n\n    // ignore if some object is being transformed at this moment\n    if (this._currentTransform) {\n      return;\n    }\n\n    let shouldRender = this._shouldRender(target);\n    let grouped = false;\n    if (this.handleMultiSelection(e, target)) {\n      // active object might have changed while grouping\n      target = this._activeObject;\n      grouped = true;\n      shouldRender = true;\n    } else if (this._shouldClearSelection(e, target)) {\n      this.discardActiveObject(e);\n    }\n    // we start a group selector rectangle if\n    // selection is enabled\n    // and there is no target, or the following 3 conditions are satisfied:\n    // target is not selectable ( otherwise we selected it )\n    // target is not editing\n    // target is not already selected ( otherwise we drag )\n    if (\n      this.selection &&\n      (!target ||\n        (!target.selectable &&\n          !(target as IText).isEditing &&\n          target !== this._activeObject))\n    ) {\n      const p = this.getScenePoint(e);\n      this._groupSelector = {\n        x: p.x,\n        y: p.y,\n        deltaY: 0,\n        deltaX: 0,\n      };\n    }\n\n    if (target) {\n      const alreadySelected = target === this._activeObject;\n      if (target.selectable && target.activeOn === 'down') {\n        this.setActiveObject(target, e);\n      }\n      const handle = target.findControl(\n        this.getViewportPoint(e),\n        isTouchEvent(e)\n      );\n      if (target === this._activeObject && (handle || !grouped)) {\n        this._setupCurrentTransform(e, target, alreadySelected);\n        const control = handle ? handle.control : undefined,\n          pointer = this.getScenePoint(e),\n          mouseDownHandler =\n            control && control.getMouseDownHandler(e, target, control);\n        mouseDownHandler &&\n          mouseDownHandler.call(\n            control,\n            e,\n            this._currentTransform!,\n            pointer.x,\n            pointer.y\n          );\n      }\n    }\n    //  we clear `_objectsToRender` in case of a change in order to repopulate it at rendering\n    //  run before firing the `down` event to give the dev a chance to populate it themselves\n    shouldRender && (this._objectsToRender = undefined);\n    this._handleEvent(e, 'down');\n    // we must renderAll so that we update the visuals\n    shouldRender && this.requestRenderAll();\n  }\n\n  /**\n   * reset cache form common information needed during event processing\n   * @private\n   */\n  _resetTransformEventData() {\n    this._target = undefined;\n    this._pointer = undefined;\n    this._absolutePointer = undefined;\n  }\n\n  /**\n   * Cache common information needed during event processing\n   * @private\n   * @param {Event} e Event object fired on event\n   */\n  _cacheTransformEventData(e: TPointerEvent) {\n    // reset in order to avoid stale caching\n    this._resetTransformEventData();\n    this._pointer = this.getViewportPoint(e);\n    this._absolutePointer = sendPointToPlane(\n      this._pointer,\n      undefined,\n      this.viewportTransform\n    );\n    this._target = this._currentTransform\n      ? this._currentTransform.target\n      : this.findTarget(e);\n  }\n\n  /**\n   * Method that defines the actions when mouse is hovering the canvas.\n   * The currentTransform parameter will define whether the user is rotating/scaling/translating\n   * an image or neither of them (only hovering). A group selection is also possible and would cancel\n   * all any other type of action.\n   * In case of an image transformation only the top canvas will be rendered.\n   * @private\n   * @param {Event} e Event object fired on mousemove\n   */\n  __onMouseMove(e: TPointerEvent) {\n    this._isClick = false;\n    this._cacheTransformEventData(e);\n    this._handleEvent(e, 'move:before');\n\n    if (this.isDrawingMode) {\n      this._onMouseMoveInDrawingMode(e);\n      return;\n    }\n\n    if (!this._isMainEvent(e)) {\n      return;\n    }\n\n    const groupSelector = this._groupSelector;\n\n    // We initially clicked in an empty area, so we draw a box for multiple selection\n    if (groupSelector) {\n      const pointer = this.getScenePoint(e);\n\n      groupSelector.deltaX = pointer.x - groupSelector.x;\n      groupSelector.deltaY = pointer.y - groupSelector.y;\n\n      this.renderTop();\n    } else if (!this._currentTransform) {\n      const target = this.findTarget(e);\n      this._setCursorFromEvent(e, target);\n      this._fireOverOutEvents(e, target);\n    } else {\n      this._transformObject(e);\n    }\n    this.textEditingManager.onMouseMove(e);\n    this._handleEvent(e, 'move');\n    this._resetTransformEventData();\n  }\n\n  /**\n   * Manage the mouseout, mouseover events for the fabric object on the canvas\n   * @param {Fabric.Object} target the target where the target from the mousemove event\n   * @param {Event} e Event object fired on mousemove\n   * @private\n   */\n  _fireOverOutEvents(e: TPointerEvent, target?: FabricObject) {\n    const _hoveredTarget = this._hoveredTarget,\n      _hoveredTargets = this._hoveredTargets,\n      targets = this.targets,\n      length = Math.max(_hoveredTargets.length, targets.length);\n\n    this.fireSyntheticInOutEvents('mouse', {\n      e,\n      target,\n      oldTarget: _hoveredTarget,\n      fireCanvas: true,\n    });\n    for (let i = 0; i < length; i++) {\n      this.fireSyntheticInOutEvents('mouse', {\n        e,\n        target: targets[i],\n        oldTarget: _hoveredTargets[i],\n      });\n    }\n    this._hoveredTarget = target;\n    this._hoveredTargets = this.targets.concat();\n  }\n\n  /**\n   * Manage the dragEnter, dragLeave events for the fabric objects on the canvas\n   * @param {Fabric.Object} target the target where the target from the onDrag event\n   * @param {Object} data Event object fired on dragover\n   * @private\n   */\n  _fireEnterLeaveEvents(target: FabricObject | undefined, data: DragEventData) {\n    const draggedoverTarget = this._draggedoverTarget,\n      _hoveredTargets = this._hoveredTargets,\n      targets = this.targets,\n      length = Math.max(_hoveredTargets.length, targets.length);\n\n    this.fireSyntheticInOutEvents('drag', {\n      ...data,\n      target,\n      oldTarget: draggedoverTarget,\n      fireCanvas: true,\n    });\n    for (let i = 0; i < length; i++) {\n      this.fireSyntheticInOutEvents('drag', {\n        ...data,\n        target: targets[i],\n        oldTarget: _hoveredTargets[i],\n      });\n    }\n    this._draggedoverTarget = target;\n  }\n\n  /**\n   * Manage the synthetic in/out events for the fabric objects on the canvas\n   * @param {Fabric.Object} target the target where the target from the supported events\n   * @param {Object} data Event object fired\n   * @param {Object} config configuration for the function to work\n   * @param {String} config.targetName property on the canvas where the old target is stored\n   * @param {String} [config.canvasEvtOut] name of the event to fire at canvas level for out\n   * @param {String} config.evtOut name of the event to fire for out\n   * @param {String} [config.canvasEvtIn] name of the event to fire at canvas level for in\n   * @param {String} config.evtIn name of the event to fire for in\n   * @private\n   */\n  fireSyntheticInOutEvents<T extends keyof TSyntheticEventContext>(\n    type: T,\n    {\n      target,\n      oldTarget,\n      fireCanvas,\n      e,\n      ...data\n    }: TSyntheticEventContext[T] & {\n      target?: FabricObject;\n      oldTarget?: FabricObject;\n      fireCanvas?: boolean;\n    }\n  ) {\n    const { targetIn, targetOut, canvasIn, canvasOut } =\n      syntheticEventConfig[type];\n    const targetChanged = oldTarget !== target;\n\n    if (oldTarget && targetChanged) {\n      const outOpt: CanvasEvents[typeof canvasOut] = {\n        ...data,\n        e,\n        target: oldTarget,\n        nextTarget: target,\n        ...getEventPoints(this, e),\n      };\n      fireCanvas && this.fire(canvasOut, outOpt);\n      oldTarget.fire(targetOut, outOpt);\n    }\n    if (target && targetChanged) {\n      const inOpt: CanvasEvents[typeof canvasIn] = {\n        ...data,\n        e,\n        target,\n        previousTarget: oldTarget,\n        ...getEventPoints(this, e),\n      };\n      fireCanvas && this.fire(canvasIn, inOpt);\n      target.fire(targetIn, inOpt);\n    }\n  }\n\n  /**\n   * Method that defines actions when an Event Mouse Wheel\n   * @param {Event} e Event object fired on mouseup\n   */\n  __onMouseWheel(e: TPointerEvent) {\n    this._cacheTransformEventData(e);\n    this._handleEvent(e, 'wheel');\n    this._resetTransformEventData();\n  }\n\n  /**\n   * @private\n   * @param {Event} e Event fired on mousemove\n   */\n  _transformObject(e: TPointerEvent) {\n    const scenePoint = this.getScenePoint(e),\n      transform = this._currentTransform!,\n      target = transform.target,\n      //  transform pointer to target's containing coordinate plane\n      //  both pointer and object should agree on every point\n      localPointer = target.group\n        ? sendPointToPlane(\n            scenePoint,\n            undefined,\n            target.group.calcTransformMatrix()\n          )\n        : scenePoint;\n    transform.shiftKey = e.shiftKey;\n    transform.altKey = !!this.centeredKey && e[this.centeredKey];\n\n    this._performTransformAction(e, transform, localPointer);\n    transform.actionPerformed && this.requestRenderAll();\n  }\n\n  /**\n   * @private\n   */\n  _performTransformAction(\n    e: TPointerEvent,\n    transform: Transform,\n    pointer: Point\n  ) {\n    const x = pointer.x,\n      y = pointer.y,\n      action = transform.action,\n      actionHandler = transform.actionHandler;\n    let actionPerformed = false;\n    // this object could be created from the function in the control handlers\n\n    if (actionHandler) {\n      actionPerformed = actionHandler(e, transform, x, y);\n    }\n    if (action === 'drag' && actionPerformed) {\n      transform.target.isMoving = true;\n      this.setCursor(transform.target.moveCursor || this.moveCursor);\n    }\n    transform.actionPerformed = transform.actionPerformed || actionPerformed;\n  }\n\n  /**\n   * Sets the cursor depending on where the canvas is being hovered.\n   * Note: very buggy in Opera\n   * @param {Event} e Event object\n   * @param {Object} target Object that the mouse is hovering, if so.\n   */\n  _setCursorFromEvent(e: TPointerEvent, target?: FabricObject) {\n    if (!target) {\n      this.setCursor(this.defaultCursor);\n      return;\n    }\n    let hoverCursor = target.hoverCursor || this.hoverCursor;\n    const activeSelection = isActiveSelection(this._activeObject)\n        ? this._activeObject\n        : null,\n      // only show proper corner when group selection is not active\n      corner =\n        (!activeSelection || target.group !== activeSelection) &&\n        // here we call findTargetCorner always with undefined for the touch parameter.\n        // we assume that if you are using a cursor you do not need to interact with\n        // the bigger touch area.\n        target.findControl(this.getViewportPoint(e));\n\n    if (!corner) {\n      if ((target as Group).subTargetCheck) {\n        // hoverCursor should come from top-most subTarget,\n        // so we walk the array backwards\n        this.targets\n          .concat()\n          .reverse()\n          .map((_target) => {\n            hoverCursor = _target.hoverCursor || hoverCursor;\n          });\n      }\n      this.setCursor(hoverCursor);\n    } else {\n      const control = corner.control;\n      this.setCursor(control.cursorStyleHandler(e, control, target));\n    }\n  }\n\n  /**\n   * ## Handles multiple selection\n   * - toggles `target` selection (selects/deselects `target` if it isn't/is selected respectively)\n   * - sets the active object in case it is not set or in case there is a single active object left under active selection.\n   * ---\n   * - If the active object is the active selection we add/remove `target` from it\n   * - If not, add the active object and `target` to the active selection and make it the active object.\n   * @private\n   * @param {TPointerEvent} e Event object\n   * @param {FabricObject} target target of event to select/deselect\n   * @returns true if grouping occurred\n   */\n  protected handleMultiSelection(e: TPointerEvent, target?: FabricObject) {\n    const activeObject = this._activeObject;\n    const isAS = isActiveSelection(activeObject);\n    if (\n      // check if an active object exists on canvas and if the user is pressing the `selectionKey` while canvas supports multi selection.\n      !!activeObject &&\n      this._isSelectionKeyPressed(e) &&\n      this.selection &&\n      // on top of that the user also has to hit a target that is selectable.\n      !!target &&\n      target.selectable &&\n      // group target and active object only if they are different objects\n      // else we try to find a subtarget of `ActiveSelection`\n      (activeObject !== target || isAS) &&\n      //  make sure `activeObject` and `target` aren't ancestors of each other in case `activeObject` is not `ActiveSelection`\n      // if it is then we want to remove `target` from it\n      (isAS ||\n        (!target.isDescendantOf(activeObject) &&\n          !activeObject.isDescendantOf(target))) &&\n      //  target accepts selection\n      !target.onSelect({ e }) &&\n      // make sure we are not on top of a control\n      !activeObject.getActiveControl()\n    ) {\n      if (isAS) {\n        const prevActiveObjects = activeObject.getObjects();\n        if (target === activeObject) {\n          const pointer = this.getViewportPoint(e);\n          target =\n            // first search active objects for a target to remove\n            this.searchPossibleTargets(prevActiveObjects, pointer) ||\n            //  if not found, search under active selection for a target to add\n            // `prevActiveObjects` will be searched but we already know they will not be found\n            this.searchPossibleTargets(this._objects, pointer);\n          // if nothing is found bail out\n          if (!target || !target.selectable) {\n            return false;\n          }\n        }\n        if (target.group === activeObject) {\n          // `target` is part of active selection => remove it\n          activeObject.remove(target);\n          this._hoveredTarget = target;\n          this._hoveredTargets = [...this.targets];\n          // if after removing an object we are left with one only...\n          if (activeObject.size() === 1) {\n            // activate last remaining object\n            // deselecting the active selection will remove the remaining object from it\n            this._setActiveObject(activeObject.item(0), e);\n          }\n        } else {\n          // `target` isn't part of active selection => add it\n          activeObject.multiSelectAdd(target);\n          this._hoveredTarget = activeObject;\n          this._hoveredTargets = [...this.targets];\n        }\n        this._fireSelectionEvents(prevActiveObjects, e);\n      } else {\n        (activeObject as IText).exitEditing &&\n          (activeObject as IText).exitEditing();\n        // add the active object and the target to the active selection and set it as the active object\n        const klass =\n          classRegistry.getClass<typeof ActiveSelection>('ActiveSelection');\n        const newActiveSelection = new klass([], {\n          /**\n           * it is crucial to pass the canvas ref before calling {@link ActiveSelection#multiSelectAdd}\n           * since it uses {@link FabricObject#isInFrontOf} which relies on the canvas ref\n           */\n          canvas: this,\n        });\n        newActiveSelection.multiSelectAdd(activeObject, target);\n        this._hoveredTarget = newActiveSelection;\n        // ISSUE 4115: should we consider subTargets here?\n        // this._hoveredTargets = [];\n        // this._hoveredTargets = this.targets.concat();\n        this._setActiveObject(newActiveSelection, e);\n        this._fireSelectionEvents([activeObject], e);\n      }\n      return true;\n    }\n    return false;\n  }\n\n  /**\n   * ## Handles selection\n   * - selects objects that are contained in (and possibly intersecting) the selection bounding box\n   * - sets the active object\n   * ---\n   * runs on mouse up after a mouse move\n   */\n  protected handleSelection(e: TPointerEvent) {\n    if (!this.selection || !this._groupSelector) {\n      return false;\n    }\n    const { x, y, deltaX, deltaY } = this._groupSelector,\n      point1 = new Point(x, y),\n      point2 = point1.add(new Point(deltaX, deltaY)),\n      tl = point1.min(point2),\n      br = point1.max(point2),\n      size = br.subtract(tl);\n\n    const collectedObjects = this.collectObjects(\n      {\n        left: tl.x,\n        top: tl.y,\n        width: size.x,\n        height: size.y,\n      },\n      { includeIntersecting: !this.selectionFullyContained }\n    ) as FabricObject[];\n\n    const objects =\n      // though this method runs only after mouse move the pointer could do a mouse up on the same position as mouse down\n      // should it be handled as is?\n      point1.eq(point2)\n        ? collectedObjects[0]\n          ? [collectedObjects[0]]\n          : []\n        : collectedObjects.length > 1\n        ? collectedObjects.filter((object) => !object.onSelect({ e })).reverse()\n        : // `setActiveObject` will call `onSelect(collectedObjects[0])` in this case\n          collectedObjects;\n\n    // set active object\n    if (objects.length === 1) {\n      // set as active object\n      this.setActiveObject(objects[0], e);\n    } else if (objects.length > 1) {\n      // add to active selection and make it the active object\n      const klass =\n        classRegistry.getClass<typeof ActiveSelection>('ActiveSelection');\n      this.setActiveObject(new klass(objects, { canvas: this }), e);\n    }\n\n    // cleanup\n    this._groupSelector = null;\n    return true;\n  }\n\n  /**\n   * @override clear {@link textEditingManager}\n   */\n  clear() {\n    this.textEditingManager.clear();\n    super.clear();\n  }\n\n  /**\n   * @override clear {@link textEditingManager}\n   */\n  destroy() {\n    this.removeListeners();\n    this.textEditingManager.dispose();\n    super.destroy();\n  }\n}\n","export const linearDefaultCoords = {\n  x1: 0,\n  y1: 0,\n  x2: 0,\n  y2: 0,\n};\n\nexport const radialDefaultCoords = {\n  ...linearDefaultCoords,\n  r1: 0,\n  r2: 0,\n};\n","import { ifNaN } from '../util/internals';\nimport { capValue } from '../util/misc/capValue';\n\nconst RE_PERCENT = /^(\\d+\\.\\d+)%|(\\d+)%$/;\n\nexport function isPercent(value: string | null) {\n  return value && RE_PERCENT.test(value);\n}\n\n/**\n *\n * @param value\n * @param valueIfNaN\n * @returns ∈ [0, 1]\n */\nexport function parsePercent(\n  value: string | number | null | undefined,\n  valueIfNaN?: number\n) {\n  const parsed =\n    typeof value === 'number'\n      ? value\n      : typeof value === 'string'\n      ? parseFloat(value) / (isPercent(value) ? 100 : 1)\n      : NaN;\n  return capValue(0, ifNaN(parsed, valueIfNaN), 1);\n}\n","import { Color } from '../../color/Color';\nimport { parsePercent } from '../../parser/percent';\nimport { ifNaN } from '../../util/internals';\nimport type { ColorStop } from '../typedefs';\n\nconst RE_KEY_VALUE_PAIRS = /\\s*;\\s*/;\nconst RE_KEY_VALUE = /\\s*:\\s*/;\n\nfunction parseColorStop(el: SVGStopElement, multiplier: number) {\n  let colorValue, opacity;\n  const style = el.getAttribute('style');\n  if (style) {\n    const keyValuePairs = style.split(RE_KEY_VALUE_PAIRS);\n\n    if (keyValuePairs[keyValuePairs.length - 1] === '') {\n      keyValuePairs.pop();\n    }\n\n    for (let i = keyValuePairs.length; i--; ) {\n      const [key, value] = keyValuePairs[i]\n        .split(RE_KEY_VALUE)\n        .map((s) => s.trim());\n      if (key === 'stop-color') {\n        colorValue = value;\n      } else if (key === 'stop-opacity') {\n        opacity = value;\n      }\n    }\n  }\n\n  const color = new Color(\n    colorValue || el.getAttribute('stop-color') || 'rgb(0,0,0)'\n  );\n\n  return {\n    offset: parsePercent(el.getAttribute('offset'), 0),\n    color: color.toRgb(),\n    opacity:\n      ifNaN(parseFloat(opacity || el.getAttribute('stop-opacity') || ''), 1) *\n      color.getAlpha() *\n      multiplier,\n  };\n}\n\nexport function parseColorStops(\n  el: SVGGradientElement,\n  opacityAttr: string | null\n) {\n  const colorStops: ColorStop[] = [],\n    colorStopEls = el.getElementsByTagName('stop'),\n    multiplier = parsePercent(opacityAttr, 1);\n  for (let i = colorStopEls.length; i--; ) {\n    colorStops.push(parseColorStop(colorStopEls[i], multiplier));\n  }\n  return colorStops;\n}\n","import type { GradientType, GradientUnits } from '../typedefs';\n\nexport function parseType(el: SVGGradientElement): GradientType {\n  return el.nodeName === 'linearGradient' || el.nodeName === 'LINEARGRADIENT'\n    ? 'linear'\n    : 'radial';\n}\n\nexport function parseGradientUnits(el: SVGGradientElement): GradientUnits {\n  return el.getAttribute('gradientUnits') === 'userSpaceOnUse'\n    ? 'pixels'\n    : 'percentage';\n}\n","import { isPercent } from '../../parser/percent';\nimport type { TSize } from '../../typedefs';\nimport type { GradientCoords, GradientType, GradientUnits } from '../typedefs';\nimport { parseGradientUnits, parseType } from './misc';\n\nfunction convertPercentUnitsToValues<\n  T extends GradientType,\n  K extends keyof GradientCoords<T>\n>(\n  valuesToConvert: Record<K, string | number>,\n  { width, height, gradientUnits }: TSize & { gradientUnits: GradientUnits }\n) {\n  let finalValue;\n  return (Object.keys(valuesToConvert) as K[]).reduce((acc, prop) => {\n    const propValue = valuesToConvert[prop];\n    if (propValue === 'Infinity') {\n      finalValue = 1;\n    } else if (propValue === '-Infinity') {\n      finalValue = 0;\n    } else {\n      finalValue =\n        typeof propValue === 'string' ? parseFloat(propValue) : propValue;\n      if (typeof propValue === 'string' && isPercent(propValue)) {\n        finalValue *= 0.01;\n        if (gradientUnits === 'pixels') {\n          // then we need to fix those percentages here in svg parsing\n          if (prop === 'x1' || prop === 'x2' || prop === 'r2') {\n            finalValue *= width;\n          }\n          if (prop === 'y1' || prop === 'y2') {\n            finalValue *= height;\n          }\n        }\n      }\n    }\n    acc[prop] = finalValue;\n    return acc;\n  }, {} as Record<K, number>);\n}\n\nfunction getValue(el: SVGGradientElement, key: string) {\n  return el.getAttribute(key);\n}\n\nexport function parseLinearCoords(el: SVGGradientElement) {\n  return {\n    x1: getValue(el, 'x1') || 0,\n    y1: getValue(el, 'y1') || 0,\n    x2: getValue(el, 'x2') || '100%',\n    y2: getValue(el, 'y2') || 0,\n  };\n}\n\nexport function parseRadialCoords(el: SVGGradientElement) {\n  return {\n    x1: getValue(el, 'fx') || getValue(el, 'cx') || '50%',\n    y1: getValue(el, 'fy') || getValue(el, 'cy') || '50%',\n    r1: 0,\n    x2: getValue(el, 'cx') || '50%',\n    y2: getValue(el, 'cy') || '50%',\n    r2: getValue(el, 'r') || '50%',\n  };\n}\n\nexport function parseCoords(el: SVGGradientElement, size: TSize) {\n  return convertPercentUnitsToValues(\n    parseType(el) === 'linear' ? parseLinearCoords(el) : parseRadialCoords(el),\n    {\n      ...size,\n      gradientUnits: parseGradientUnits(el),\n    }\n  );\n}\n","import { Color } from '../color/Color';\nimport { iMatrix } from '../constants';\nimport { parseTransformAttribute } from '../parser/parseTransformAttribute';\nimport type { FabricObject } from '../shapes/Object/FabricObject';\nimport type { TMat2D } from '../typedefs';\nimport { uid } from '../util/internals/uid';\nimport { pick } from '../util/misc/pick';\nimport { matrixToSVG } from '../util/misc/svgParsing';\nimport { linearDefaultCoords, radialDefaultCoords } from './constants';\nimport { parseColorStops } from './parser/parseColorStops';\nimport { parseCoords } from './parser/parseCoords';\nimport { parseType, parseGradientUnits } from './parser/misc';\nimport type {\n  ColorStop,\n  GradientCoords,\n  GradientOptions,\n  GradientType,\n  GradientUnits,\n  SVGOptions,\n} from './typedefs';\nimport { classRegistry } from '../ClassRegistry';\nimport { isPath } from '../util/typeAssertions';\n\n/**\n * Gradient class\n * @class Gradient\n * @tutorial {@link http://fabricjs.com/fabric-intro-part-2#gradients}\n */\nexport class Gradient<\n  S,\n  T extends GradientType = S extends GradientType ? S : 'linear'\n> {\n  /**\n   * Horizontal offset for aligning gradients coming from SVG when outside pathgroups\n   * @type Number\n   * @default 0\n   */\n  declare offsetX: number;\n\n  /**\n   * Vertical offset for aligning gradients coming from SVG when outside pathgroups\n   * @type Number\n   * @default 0\n   */\n  declare offsetY: number;\n\n  /**\n   * A transform matrix to apply to the gradient before painting.\n   * Imported from svg gradients, is not applied with the current transform in the center.\n   * Before this transform is applied, the origin point is at the top left corner of the object\n   * plus the addition of offsetY and offsetX.\n   * @type Number[]\n   * @default null\n   */\n  declare gradientTransform?: TMat2D;\n\n  /**\n   * coordinates units for coords.\n   * If `pixels`, the number of coords are in the same unit of width / height.\n   * If set as `percentage` the coords are still a number, but 1 means 100% of width\n   * for the X and 100% of the height for the y. It can be bigger than 1 and negative.\n   * allowed values pixels or percentage.\n   * @type GradientUnits\n   * @default 'pixels'\n   */\n  declare gradientUnits: GradientUnits;\n\n  /**\n   * Gradient type linear or radial\n   * @type GradientType\n   * @default 'linear'\n   */\n  declare type: T;\n\n  /**\n   * Defines how the gradient is located in space and spread\n   * @type GradientCoords\n   */\n  declare coords: GradientCoords<T>;\n\n  /**\n   * Defines how many colors a gradient has and how they are located on the axis\n   * defined by coords\n   * @type GradientCoords\n   */\n  declare colorStops: ColorStop[];\n\n  /**\n   * If true, this object will not be exported during the serialization of a canvas\n   * @type boolean\n   */\n  declare excludeFromExport?: boolean;\n\n  /**\n   * ID used for SVG export functionalities\n   * @type number | string\n   */\n  declare readonly id: string | number;\n\n  static type = 'Gradient';\n\n  constructor({\n    type = 'linear' as T,\n    gradientUnits = 'pixels',\n    coords = {},\n    colorStops = [],\n    offsetX = 0,\n    offsetY = 0,\n    gradientTransform,\n    id,\n  }: GradientOptions<T>) {\n    this.id = id ? `${id}_${uid()}` : uid();\n    this.type = type;\n    this.gradientUnits = gradientUnits;\n    this.gradientTransform = gradientTransform;\n    this.offsetX = offsetX;\n    this.offsetY = offsetY;\n    this.coords = {\n      ...(this.type === 'radial' ? radialDefaultCoords : linearDefaultCoords),\n      ...coords,\n    } as GradientCoords<T>;\n    this.colorStops = colorStops.slice();\n  }\n\n  // isType<S extends GradientType>(type: S): this is Gradient<S> {\n  //   return (this.type as GradientType) === type;\n  // }\n\n  /**\n   * Adds another colorStop\n   * @param {Record<string, string>} colorStop Object with offset and color\n   * @return {Gradient} thisArg\n   */\n  addColorStop(colorStops: Record<string, string>) {\n    for (const position in colorStops) {\n      const color = new Color(colorStops[position]);\n      this.colorStops.push({\n        offset: parseFloat(position),\n        color: color.toRgb(),\n        opacity: color.getAlpha(),\n      });\n    }\n    return this;\n  }\n\n  /**\n   * Returns object representation of a gradient\n   * @param {string[]} [propertiesToInclude] Any properties that you might want to additionally include in the output\n   * @return {object}\n   */\n  toObject(propertiesToInclude?: (keyof this | string)[]) {\n    return {\n      ...pick(this, propertiesToInclude as (keyof this)[]),\n      type: this.type,\n      coords: this.coords,\n      colorStops: this.colorStops,\n      offsetX: this.offsetX,\n      offsetY: this.offsetY,\n      gradientUnits: this.gradientUnits,\n      gradientTransform: this.gradientTransform\n        ? [...this.gradientTransform]\n        : undefined,\n    };\n  }\n\n  /* _TO_SVG_START_ */\n  /**\n   * Returns SVG representation of an gradient\n   * @param {FabricObject} object Object to create a gradient for\n   * @return {String} SVG representation of an gradient (linear/radial)\n   */\n  toSVG(\n    object: FabricObject,\n    { additionalTransform: preTransform }: { additionalTransform?: string } = {}\n  ) {\n    const markup = [],\n      transform = (\n        this.gradientTransform\n          ? this.gradientTransform.concat()\n          : iMatrix.concat()\n      ) as TMat2D,\n      gradientUnits =\n        this.gradientUnits === 'pixels'\n          ? 'userSpaceOnUse'\n          : 'objectBoundingBox';\n    // colorStops must be sorted ascending, and guarded against deep mutations\n    const colorStops = this.colorStops\n      .map((colorStop) => ({ ...colorStop }))\n      .sort((a, b) => {\n        return a.offset - b.offset;\n      });\n\n    let offsetX = -this.offsetX,\n      offsetY = -this.offsetY;\n    if (gradientUnits === 'objectBoundingBox') {\n      offsetX /= object.width;\n      offsetY /= object.height;\n    } else {\n      offsetX += object.width / 2;\n      offsetY += object.height / 2;\n    }\n    // todo what about polygon/polyline?\n    if (isPath(object) && this.gradientUnits !== 'percentage') {\n      offsetX -= object.pathOffset.x;\n      offsetY -= object.pathOffset.y;\n    }\n    transform[4] -= offsetX;\n    transform[5] -= offsetY;\n\n    const commonAttributes = [\n      `id=\"SVGID_${this.id}\"`,\n      `gradientUnits=\"${gradientUnits}\"`,\n      `gradientTransform=\"${\n        preTransform ? preTransform + ' ' : ''\n      }${matrixToSVG(transform)}\"`,\n      '',\n    ].join(' ');\n\n    if (this.type === 'linear') {\n      const { x1, y1, x2, y2 } = this.coords;\n      markup.push(\n        '<linearGradient ',\n        commonAttributes,\n        ' x1=\"',\n        x1,\n        '\" y1=\"',\n        y1,\n        '\" x2=\"',\n        x2,\n        '\" y2=\"',\n        y2,\n        '\">\\n'\n      );\n    } else if (this.type === 'radial') {\n      const { x1, y1, x2, y2, r1, r2 } = this\n        .coords as GradientCoords<'radial'>;\n      const needsSwap = r1 > r2;\n      // svg radial gradient has just 1 radius. the biggest.\n      markup.push(\n        '<radialGradient ',\n        commonAttributes,\n        ' cx=\"',\n        needsSwap ? x1 : x2,\n        '\" cy=\"',\n        needsSwap ? y1 : y2,\n        '\" r=\"',\n        needsSwap ? r1 : r2,\n        '\" fx=\"',\n        needsSwap ? x2 : x1,\n        '\" fy=\"',\n        needsSwap ? y2 : y1,\n        '\">\\n'\n      );\n      if (needsSwap) {\n        // svg goes from internal to external radius. if radius are inverted, swap color stops.\n        colorStops.reverse(); //  mutates array\n        colorStops.forEach((colorStop) => {\n          colorStop.offset = 1 - colorStop.offset;\n        });\n      }\n      const minRadius = Math.min(r1, r2);\n      if (minRadius > 0) {\n        // i have to shift all colorStops and add new one in 0.\n        const maxRadius = Math.max(r1, r2),\n          percentageShift = minRadius / maxRadius;\n        colorStops.forEach((colorStop) => {\n          colorStop.offset += percentageShift * (1 - colorStop.offset);\n        });\n      }\n    }\n\n    colorStops.forEach(({ color, offset, opacity }) => {\n      markup.push(\n        '<stop ',\n        'offset=\"',\n        offset * 100 + '%',\n        '\" style=\"stop-color:',\n        color,\n        typeof opacity !== 'undefined' ? ';stop-opacity: ' + opacity : ';',\n        '\"/>\\n'\n      );\n    });\n\n    markup.push(\n      this.type === 'linear' ? '</linearGradient>' : '</radialGradient>',\n      '\\n'\n    );\n\n    return markup.join('');\n  }\n  /* _TO_SVG_END_ */\n\n  /**\n   * Returns an instance of CanvasGradient\n   * @param {CanvasRenderingContext2D} ctx Context to render on\n   * @return {CanvasGradient}\n   */\n  toLive(ctx: CanvasRenderingContext2D): CanvasGradient {\n    const coords = this.coords as GradientCoords<'radial'>;\n    const gradient =\n      this.type === 'linear'\n        ? ctx.createLinearGradient(coords.x1, coords.y1, coords.x2, coords.y2)\n        : ctx.createRadialGradient(\n            coords.x1,\n            coords.y1,\n            coords.r1,\n            coords.x2,\n            coords.y2,\n            coords.r2\n          );\n\n    this.colorStops.forEach(({ color, opacity, offset }) => {\n      gradient.addColorStop(\n        offset,\n        typeof opacity !== 'undefined'\n          ? new Color(color).setAlpha(opacity).toRgba()\n          : color\n      );\n    });\n\n    return gradient;\n  }\n\n  static async fromObject(\n    options: GradientOptions<'linear'>\n  ): Promise<Gradient<'radial'>>;\n  static async fromObject(\n    options: GradientOptions<'radial'>\n  ): Promise<Gradient<'radial'>>;\n  static async fromObject(\n    options: GradientOptions<'linear'> | GradientOptions<'radial'>\n  ) {\n    return new this(options);\n  }\n\n  /* _FROM_SVG_START_ */\n  /**\n   * Returns {@link Gradient} instance from an SVG element\n   * @static\n   * @memberOf Gradient\n   * @param {SVGGradientElement} el SVG gradient element\n   * @param {FabricObject} instance\n   * @param {String} opacity A fill-opacity or stroke-opacity attribute to multiply to each stop's opacity.\n   * @param {SVGOptions} svgOptions an object containing the size of the SVG in order to parse correctly gradients\n   * that uses gradientUnits as 'userSpaceOnUse' and percentages.\n   * @return {Gradient} Gradient instance\n   * @see http://www.w3.org/TR/SVG/pservers.html#LinearGradientElement\n   * @see http://www.w3.org/TR/SVG/pservers.html#RadialGradientElement\n   *\n   *  @example\n   *\n   *  <linearGradient id=\"linearGrad1\">\n   *    <stop offset=\"0%\" stop-color=\"white\"/>\n   *    <stop offset=\"100%\" stop-color=\"black\"/>\n   *  </linearGradient>\n   *\n   *  OR\n   *\n   *  <linearGradient id=\"linearGrad2\">\n   *    <stop offset=\"0\" style=\"stop-color:rgb(255,255,255)\"/>\n   *    <stop offset=\"1\" style=\"stop-color:rgb(0,0,0)\"/>\n   *  </linearGradient>\n   *\n   *  OR\n   *\n   *  <radialGradient id=\"radialGrad1\">\n   *    <stop offset=\"0%\" stop-color=\"white\" stop-opacity=\"1\" />\n   *    <stop offset=\"50%\" stop-color=\"black\" stop-opacity=\"0.5\" />\n   *    <stop offset=\"100%\" stop-color=\"white\" stop-opacity=\"1\" />\n   *  </radialGradient>\n   *\n   *  OR\n   *\n   *  <radialGradient id=\"radialGrad2\">\n   *    <stop offset=\"0\" stop-color=\"rgb(255,255,255)\" />\n   *    <stop offset=\"0.5\" stop-color=\"rgb(0,0,0)\" />\n   *    <stop offset=\"1\" stop-color=\"rgb(255,255,255)\" />\n   *  </radialGradient>\n   *\n   */\n  static fromElement(\n    el: SVGGradientElement,\n    instance: FabricObject,\n    svgOptions: SVGOptions\n  ): Gradient<GradientType> {\n    const gradientUnits = parseGradientUnits(el);\n    const center = instance._findCenterFromElement();\n    return new this({\n      id: el.getAttribute('id') || undefined,\n      type: parseType(el),\n      coords: parseCoords(el, {\n        width: svgOptions.viewBoxWidth || svgOptions.width,\n        height: svgOptions.viewBoxHeight || svgOptions.height,\n      }),\n      colorStops: parseColorStops(el, svgOptions.opacity),\n      gradientUnits,\n      gradientTransform: parseTransformAttribute(\n        el.getAttribute('gradientTransform') || ''\n      ),\n      ...(gradientUnits === 'pixels'\n        ? {\n            offsetX: instance.width / 2 - center.x,\n            offsetY: instance.height / 2 - center.y,\n          }\n        : {\n            offsetX: 0,\n            offsetY: 0,\n          }),\n    });\n  }\n  /* _FROM_SVG_END_ */\n}\n\nclassRegistry.setClass(Gradient, 'gradient');\nclassRegistry.setClass(Gradient, 'linear');\nclassRegistry.setClass(Gradient, 'radial');\n","import { config } from '../config';\nimport type { Abortable, TCrossOrigin, TMat2D, TSize } from '../typedefs';\nimport { ifNaN } from '../util/internals';\nimport { uid } from '../util/internals/uid';\nimport { loadImage } from '../util/misc/objectEnlive';\nimport { pick } from '../util/misc/pick';\nimport { toFixed } from '../util/misc/toFixed';\nimport { classRegistry } from '../ClassRegistry';\nimport type {\n  PatternRepeat,\n  PatternOptions,\n  SerializedPatternOptions,\n} from './types';\nimport { log } from '../util/internals/console';\n\n/**\n * @see {@link http://fabricjs.com/patterns demo}\n * @see {@link http://fabricjs.com/dynamic-patterns demo}\n */\nexport class Pattern {\n  static type = 'Pattern';\n\n  /**\n   * Legacy identifier of the class. Prefer using this.constructor.type 'Pattern'\n   * or utils like isPattern, or instance of to indentify a pattern in your code.\n   * Will be removed in future versiones\n   * @TODO add sustainable warning message\n   * @type string\n   * @deprecated\n   */\n  get type() {\n    return 'pattern';\n  }\n\n  set type(value) {\n    log('warn', 'Setting type has no effect', value);\n  }\n\n  /**\n   * @type PatternRepeat\n   * @defaults\n   */\n  repeat: PatternRepeat = 'repeat';\n\n  /**\n   * Pattern horizontal offset from object's left/top corner\n   * @type Number\n   * @default\n   */\n  offsetX = 0;\n\n  /**\n   * Pattern vertical offset from object's left/top corner\n   * @type Number\n   * @default\n   */\n  offsetY = 0;\n\n  /**\n   * @type TCrossOrigin\n   * @default\n   */\n  crossOrigin: TCrossOrigin = '';\n\n  /**\n   * transform matrix to change the pattern, imported from svgs.\n   * @todo verify if using the identity matrix as default makes the rest of the code more easy\n   * @type Array\n   * @default\n   */\n  patternTransform: TMat2D | null = null;\n\n  /**\n   * The actual pixel source of the pattern\n   */\n  declare source: CanvasImageSource;\n\n  /**\n   * If true, this object will not be exported during the serialization of a canvas\n   * @type boolean\n   */\n  declare excludeFromExport?: boolean;\n\n  /**\n   * ID used for SVG export functionalities\n   * @type number\n   */\n  declare readonly id: number;\n\n  /**\n   * Constructor\n   * @param {Object} [options] Options object\n   * @param {option.source} [source] the pattern source, eventually empty or a drawable\n   */\n  constructor(options: PatternOptions) {\n    this.id = uid();\n    Object.assign(this, options);\n  }\n\n  /**\n   * @returns true if {@link source} is an <img> element\n   */\n  isImageSource(): this is { source: HTMLImageElement } {\n    return (\n      !!this.source && typeof (this.source as HTMLImageElement).src === 'string'\n    );\n  }\n\n  /**\n   * @returns true if {@link source} is a <canvas> element\n   */\n  isCanvasSource(): this is { source: HTMLCanvasElement } {\n    return !!this.source && !!(this.source as HTMLCanvasElement).toDataURL;\n  }\n\n  sourceToString(): string {\n    return this.isImageSource()\n      ? this.source.src\n      : this.isCanvasSource()\n      ? this.source.toDataURL()\n      : '';\n  }\n\n  /**\n   * Returns an instance of CanvasPattern\n   * @param {CanvasRenderingContext2D} ctx Context to create pattern\n   * @return {CanvasPattern}\n   */\n  toLive(ctx: CanvasRenderingContext2D): CanvasPattern | null {\n    if (\n      // if the image failed to load, return, and allow rest to continue loading\n      !this.source ||\n      // if an image\n      (this.isImageSource() &&\n        (!this.source.complete ||\n          this.source.naturalWidth === 0 ||\n          this.source.naturalHeight === 0))\n    ) {\n      return null;\n    }\n\n    return ctx.createPattern(this.source, this.repeat)!;\n  }\n\n  /**\n   * Returns object representation of a pattern\n   * @param {Array} [propertiesToInclude] Any properties that you might want to additionally include in the output\n   * @return {object} Object representation of a pattern instance\n   */\n  toObject(propertiesToInclude: string[] = []): Record<string, any> {\n    const { repeat, crossOrigin } = this;\n    return {\n      ...pick(this, propertiesToInclude as (keyof this)[]),\n      type: 'pattern',\n      source: this.sourceToString(),\n      repeat,\n      crossOrigin,\n      offsetX: toFixed(this.offsetX, config.NUM_FRACTION_DIGITS),\n      offsetY: toFixed(this.offsetY, config.NUM_FRACTION_DIGITS),\n      patternTransform: this.patternTransform\n        ? [...this.patternTransform]\n        : null,\n    };\n  }\n\n  /* _TO_SVG_START_ */\n  /**\n   * Returns SVG representation of a pattern\n   */\n  toSVG({ width, height }: TSize): string {\n    const { source: patternSource, repeat, id } = this,\n      patternOffsetX = ifNaN(this.offsetX / width, 0),\n      patternOffsetY = ifNaN(this.offsetY / height, 0),\n      patternWidth =\n        repeat === 'repeat-y' || repeat === 'no-repeat'\n          ? 1 + Math.abs(patternOffsetX || 0)\n          : ifNaN(\n              ((patternSource as HTMLImageElement).width as number) / width,\n              0\n            ),\n      patternHeight =\n        repeat === 'repeat-x' || repeat === 'no-repeat'\n          ? 1 + Math.abs(patternOffsetY || 0)\n          : ifNaN(\n              ((patternSource as HTMLImageElement).height as number) / height,\n              0\n            );\n\n    return [\n      `<pattern id=\"SVGID_${id}\" x=\"${patternOffsetX}\" y=\"${patternOffsetY}\" width=\"${patternWidth}\" height=\"${patternHeight}\">`,\n      `<image x=\"0\" y=\"0\" width=\"${\n        (patternSource as HTMLImageElement).width\n      }\" height=\"${\n        (patternSource as HTMLImageElement).height\n      }\" xlink:href=\"${this.sourceToString()}\"></image>`,\n      `</pattern>`,\n      '',\n    ].join('\\n');\n  }\n  /* _TO_SVG_END_ */\n\n  static async fromObject(\n    { type, source, ...serialized }: SerializedPatternOptions,\n    options: Abortable\n  ): Promise<Pattern> {\n    const img = await loadImage(source, {\n      ...options,\n      crossOrigin: serialized.crossOrigin,\n    });\n    return new this({ ...serialized, source: img });\n  }\n}\n\nclassRegistry.setClass(Pattern);\n// kept for compatibility reason\nclassRegistry.setClass(Pattern, 'pattern');\n","import { Color } from '../color/Color';\nimport type { Point } from '../Point';\nimport type { Shadow } from '../Shadow';\nimport type { Canvas } from '../canvas/Canvas';\nimport type { TBrushEventData } from './typedefs';\n\n/**\n * @see {@link http://fabricjs.com/freedrawing|Freedrawing demo}\n */\nexport abstract class BaseBrush {\n  /**\n   * Color of a brush\n   * @type String\n   * @default\n   */\n  color = 'rgb(0, 0, 0)';\n\n  /**\n   * Width of a brush, has to be a Number, no string literals\n   * @type Number\n   * @default\n   */\n  width = 1;\n\n  /**\n   * Shadow object representing shadow of this shape.\n   * <b>Backwards incompatibility note:</b> This property replaces \"shadowColor\" (String), \"shadowOffsetX\" (Number),\n   * \"shadowOffsetY\" (Number) and \"shadowBlur\" (Number) since v1.2.12\n   * @type Shadow\n   * @default\n   */\n  shadow: Shadow | null = null;\n\n  /**\n   * Line endings style of a brush (one of \"butt\", \"round\", \"square\")\n   * @type String\n   * @default\n   */\n  strokeLineCap: CanvasLineCap = 'round';\n\n  /**\n   * Corner style of a brush (one of \"bevel\", \"round\", \"miter\")\n   * @type String\n   * @default\n   */\n  strokeLineJoin: CanvasLineJoin = 'round';\n\n  /**\n   * Maximum miter length (used for strokeLineJoin = \"miter\") of a brush's\n   * @type Number\n   * @default\n   */\n  strokeMiterLimit = 10;\n\n  /**\n   * Stroke Dash Array.\n   * @type Array\n   * @default\n   */\n  strokeDashArray: number[] | null = null;\n\n  /**\n   * When `true`, the free drawing is limited to the whiteboard size. Default to false.\n   * @type Boolean\n   * @default false\n   */\n\n  limitedToCanvasSize = false;\n\n  /**\n   * @todo add type\n   */\n  declare canvas: Canvas;\n\n  constructor(canvas: Canvas) {\n    this.canvas = canvas;\n  }\n\n  abstract _render(): void;\n  abstract onMouseDown(pointer: Point, ev: TBrushEventData): void;\n  abstract onMouseMove(pointer: Point, ev: TBrushEventData): void;\n  /**\n   * @returns true if brush should continue blocking interaction\n   */\n  abstract onMouseUp(ev: TBrushEventData): boolean | void;\n\n  /**\n   * Sets brush styles\n   * @private\n   * @param {CanvasRenderingContext2D} ctx\n   */\n  _setBrushStyles(ctx: CanvasRenderingContext2D) {\n    ctx.strokeStyle = this.color;\n    ctx.lineWidth = this.width;\n    ctx.lineCap = this.strokeLineCap;\n    ctx.miterLimit = this.strokeMiterLimit;\n    ctx.lineJoin = this.strokeLineJoin;\n    ctx.setLineDash(this.strokeDashArray || []);\n  }\n\n  /**\n   * Sets the transformation on given context\n   * @param {CanvasRenderingContext2D} ctx context to render on\n   * @private\n   */\n  protected _saveAndTransform(ctx: CanvasRenderingContext2D) {\n    const v = this.canvas.viewportTransform;\n    ctx.save();\n    ctx.transform(v[0], v[1], v[2], v[3], v[4], v[5]);\n  }\n\n  protected needsFullRender() {\n    const color = new Color(this.color);\n    return color.getAlpha() < 1 || !!this.shadow;\n  }\n\n  /**\n   * Sets brush shadow styles\n   * @private\n   */\n  protected _setShadow() {\n    if (!this.shadow || !this.canvas) {\n      return;\n    }\n\n    const canvas = this.canvas,\n      shadow = this.shadow,\n      ctx = canvas.contextTop,\n      zoom = canvas.getZoom() * canvas.getRetinaScaling();\n\n    ctx.shadowColor = shadow.color;\n    ctx.shadowBlur = shadow.blur * zoom;\n    ctx.shadowOffsetX = shadow.offsetX * zoom;\n    ctx.shadowOffsetY = shadow.offsetY * zoom;\n  }\n\n  /**\n   * Removes brush shadow styles\n   * @private\n   */\n  protected _resetShadow() {\n    const ctx = this.canvas.contextTop;\n\n    ctx.shadowColor = '';\n    ctx.shadowBlur = ctx.shadowOffsetX = ctx.shadowOffsetY = 0;\n  }\n\n  /**\n   * Check is pointer is outside canvas boundaries\n   * @param {Object} pointer\n   * @private\n   */\n  protected _isOutSideCanvas(pointer: Point) {\n    return (\n      pointer.x < 0 ||\n      pointer.x > this.canvas.getWidth() ||\n      pointer.y < 0 ||\n      pointer.y > this.canvas.getHeight()\n    );\n  }\n}\n","import { config } from '../config';\nimport { SHARED_ATTRIBUTES } from '../parser/attributes';\nimport { parseAttributes } from '../parser/parseAttributes';\nimport type { XY } from '../Point';\nimport { Point } from '../Point';\nimport { makeBoundingBoxFromPoints } from '../util/misc/boundingBoxFromPoints';\nimport { toFixed } from '../util/misc/toFixed';\nimport {\n  getBoundsOfCurve,\n  joinPath,\n  makePathSimpler,\n  parsePath,\n} from '../util/path';\nimport { classRegistry } from '../ClassRegistry';\nimport { FabricObject, cacheProperties } from './Object/FabricObject';\nimport type {\n  TComplexPathData,\n  TPathSegmentInfo,\n  TSimplePathData,\n} from '../util/path/typedefs';\nimport type { FabricObjectProps, SerializedObjectProps } from './Object/types';\nimport type { ObjectEvents } from '../EventTypeDefs';\nimport type {\n  TBBox,\n  TClassProperties,\n  TSVGReviver,\n  TOptions,\n} from '../typedefs';\nimport { CENTER, LEFT, TOP } from '../constants';\nimport type { CSSRules } from '../parser/typedefs';\n\ninterface UniquePathProps {\n  sourcePath?: string;\n  path?: TSimplePathData;\n}\n\nexport interface SerializedPathProps\n  extends SerializedObjectProps,\n    UniquePathProps {}\n\nexport interface PathProps extends FabricObjectProps, UniquePathProps {}\n\nexport interface IPathBBox extends TBBox {\n  left: number;\n  top: number;\n  pathOffset: Point;\n}\n\nexport class Path<\n  Props extends TOptions<PathProps> = Partial<PathProps>,\n  SProps extends SerializedPathProps = SerializedPathProps,\n  EventSpec extends ObjectEvents = ObjectEvents\n> extends FabricObject<Props, SProps, EventSpec> {\n  /**\n   * Array of path points\n   * @type Array\n   * @default\n   */\n  declare path: TSimplePathData;\n\n  declare pathOffset: Point;\n\n  declare sourcePath?: string;\n\n  declare segmentsInfo?: TPathSegmentInfo[];\n\n  static type = 'Path';\n  fromObjectId: string;\n  toObjectId: string;\n\n  static cacheProperties = [...cacheProperties, 'path', 'fillRule'];\n\n  /**\n   * Constructor\n   * @param {TComplexPathData} path Path data (sequence of coordinates and corresponding \"command\" tokens)\n   * @param {Partial<PathProps>} [options] Options object\n   * @return {Path} thisArg\n   */\n  constructor(\n    path: TComplexPathData | string,\n    { path: _, left, top, ...options }: Partial<Props> = {}\n  ) {\n    super(options as Props);\n    this._setPath(path || [], true);\n    typeof left === 'number' && this.set(LEFT, left);\n    typeof top === 'number' && this.set(TOP, top);\n  }\n\n  /**\n   * @private\n   * @param {TComplexPathData | string} path Path data (sequence of coordinates and corresponding \"command\" tokens)\n   * @param {boolean} [adjustPosition] pass true to reposition the object according to the bounding box\n   * @returns {Point} top left position of the bounding box, useful for complementary positioning\n   */\n  _setPath(path: TComplexPathData | string, adjustPosition?: boolean) {\n    this.path = makePathSimpler(Array.isArray(path) ? path : parsePath(path));\n    this.setBoundingBox(adjustPosition);\n  }\n\n  /**\n   * This function is an helper for svg import. it returns the center of the object in the svg\n   * untransformed coordinates, by look at the polyline/polygon points.\n   * @private\n   * @return {Point} center point from element coordinates\n   */\n  _findCenterFromElement(): Point {\n    const bbox = this._calcBoundsFromPath();\n    return new Point(bbox.left + bbox.width / 2, bbox.top + bbox.height / 2);\n  }\n\n  /**\n   * @private\n   * @param {CanvasRenderingContext2D} ctx context to render path on\n   */\n  _renderPathCommands(ctx: CanvasRenderingContext2D) {\n    const l = -this.pathOffset.x,\n      t = -this.pathOffset.y;\n\n    ctx.beginPath();\n\n    for (const command of this.path) {\n      switch (\n        command[0] // first letter\n      ) {\n        case 'L': // lineto, absolute\n          ctx.lineTo(command[1] + l, command[2] + t);\n          break;\n\n        case 'M': // moveTo, absolute\n          ctx.moveTo(command[1] + l, command[2] + t);\n          break;\n\n        case 'C': // bezierCurveTo, absolute\n          ctx.bezierCurveTo(\n            command[1] + l,\n            command[2] + t,\n            command[3] + l,\n            command[4] + t,\n            command[5] + l,\n            command[6] + t\n          );\n          break;\n\n        case 'Q': // quadraticCurveTo, absolute\n          ctx.quadraticCurveTo(\n            command[1] + l,\n            command[2] + t,\n            command[3] + l,\n            command[4] + t\n          );\n          break;\n\n        case 'Z':\n          ctx.closePath();\n          break;\n      }\n    }\n  }\n\n  /**\n   * @private\n   * @param {CanvasRenderingContext2D} ctx context to render path on\n   */\n  _render(ctx: CanvasRenderingContext2D) {\n    this._renderPathCommands(ctx);\n    this._renderPaintInOrder(ctx);\n  }\n\n  /**\n   * Returns string representation of an instance\n   * @return {string} string representation of an instance\n   */\n  toString() {\n    return `#<Path (${this.complexity()}): { \"top\": ${this.top}, \"left\": ${\n      this.left\n    } }>`;\n  }\n\n  /**\n   * Returns object representation of an instance\n   * @param {Array} [propertiesToInclude] Any properties that you might want to additionally include in the output\n   * @return {Object} object representation of an instance\n   */\n  toObject<\n    T extends Omit<Props & TClassProperties<this>, keyof SProps>,\n    K extends keyof T = never\n  >(propertiesToInclude: K[] = []): Pick<T, K> & SProps {\n    return {\n      ...super.toObject(propertiesToInclude),\n      path: this.path.map((pathCmd) => pathCmd.slice()),\n    };\n  }\n\n  /**\n   * Returns dataless object representation of an instance\n   * @param {Array} [propertiesToInclude] Any properties that you might want to additionally include in the output\n   * @return {Object} object representation of an instance\n   */\n  toDatalessObject<\n    T extends Omit<Props & TClassProperties<this>, keyof SProps>,\n    K extends keyof T = never\n  >(propertiesToInclude: K[] = []): Pick<T, K> & SProps {\n    const o = this.toObject<T, K>(propertiesToInclude);\n    if (this.sourcePath) {\n      delete o.path;\n      o.sourcePath = this.sourcePath;\n    }\n    return o;\n  }\n\n  /**\n   * Returns svg representation of an instance\n   * @return {Array} an array of strings with the specific svg representation\n   * of the instance\n   */\n  _toSVG() {\n    const path = joinPath(this.path, config.NUM_FRACTION_DIGITS);\n    return [\n      '<path ',\n      'COMMON_PARTS',\n      `d=\"${path}\" stroke-linecap=\"round\" />\\n`,\n    ];\n  }\n\n  /**\n   * @private\n   * @return the path command's translate transform attribute\n   */\n  _getOffsetTransform() {\n    const digits = config.NUM_FRACTION_DIGITS;\n    return ` translate(${toFixed(-this.pathOffset.x, digits)}, ${toFixed(\n      -this.pathOffset.y,\n      digits\n    )})`;\n  }\n\n  /**\n   * Returns svg clipPath representation of an instance\n   * @param {Function} [reviver] Method for further parsing of svg representation.\n   * @return {string} svg representation of an instance\n   */\n  toClipPathSVG(reviver: TSVGReviver): string {\n    const additionalTransform = this._getOffsetTransform();\n    return (\n      '\\t' +\n      this._createBaseClipPathSVGMarkup(this._toSVG(), {\n        reviver: reviver,\n        additionalTransform: additionalTransform,\n      })\n    );\n  }\n\n  /**\n   * Returns svg representation of an instance\n   * @param {Function} [reviver] Method for further parsing of svg representation.\n   * @return {string} svg representation of an instance\n   */\n  toSVG(reviver: TSVGReviver): string {\n    const additionalTransform = this._getOffsetTransform();\n    return this._createBaseSVGMarkup(this._toSVG(), {\n      reviver: reviver,\n      additionalTransform: additionalTransform,\n    });\n  }\n\n  /**\n   * Returns number representation of an instance complexity\n   * @return {number} complexity of this instance\n   */\n  complexity() {\n    return this.path.length;\n  }\n\n  setDimensions() {\n    this.setBoundingBox();\n  }\n\n  setBoundingBox(adjustPosition?: boolean) {\n    const { width, height, pathOffset } = this._calcDimensions();\n    this.set({ width, height, pathOffset });\n    // using pathOffset because it match the use case.\n    // if pathOffset change here we need to use left + width/2 , top + height/2\n    adjustPosition && this.setPositionByOrigin(pathOffset, CENTER, CENTER);\n  }\n\n  _calcBoundsFromPath(): TBBox {\n    const bounds: XY[] = [];\n    let subpathStartX = 0,\n      subpathStartY = 0,\n      x = 0, // current x\n      y = 0; // current y\n\n    for (const command of this.path) {\n      // current instruction\n      switch (\n        command[0] // first letter\n      ) {\n        case 'L': // lineto, absolute\n          x = command[1];\n          y = command[2];\n          bounds.push(new Point(subpathStartX, subpathStartY), new Point(x, y));\n          break;\n\n        case 'M': // moveTo, absolute\n          x = command[1];\n          y = command[2];\n          subpathStartX = x;\n          subpathStartY = y;\n          break;\n\n        case 'C': // bezierCurveTo, absolute\n          bounds.push(\n            ...getBoundsOfCurve(\n              x,\n              y,\n              command[1],\n              command[2],\n              command[3],\n              command[4],\n              command[5],\n              command[6]\n            )\n          );\n          x = command[5];\n          y = command[6];\n          break;\n\n        case 'Q': // quadraticCurveTo, absolute\n          bounds.push(\n            ...getBoundsOfCurve(\n              x,\n              y,\n              command[1],\n              command[2],\n              command[1],\n              command[2],\n              command[3],\n              command[4]\n            )\n          );\n          x = command[3];\n          y = command[4];\n          break;\n\n        case 'Z':\n          x = subpathStartX;\n          y = subpathStartY;\n          break;\n      }\n    }\n    return makeBoundingBoxFromPoints(bounds);\n  }\n\n  /**\n   * @private\n   */\n  _calcDimensions(): IPathBBox {\n    const bbox = this._calcBoundsFromPath();\n\n    return {\n      ...bbox,\n      pathOffset: new Point(\n        bbox.left + bbox.width / 2,\n        bbox.top + bbox.height / 2\n      ),\n    };\n  }\n\n  /**\n   * List of attribute names to account for when parsing SVG element (used by `Path.fromElement`)\n   * @static\n   * @memberOf Path\n   * @see http://www.w3.org/TR/SVG/paths.html#PathElement\n   */\n  static ATTRIBUTE_NAMES = [...SHARED_ATTRIBUTES, 'd'];\n\n  /**\n   * Creates an instance of Path from an object\n   * @static\n   * @memberOf Path\n   * @param {Object} object\n   * @returns {Promise<Path>}\n   */\n  static fromObject<T extends TOptions<SerializedPathProps>>(object: T) {\n    return this._fromObject<Path>(object, {\n      extraParam: 'path',\n    });\n  }\n\n  /**\n   * Creates an instance of Path from an SVG <path> element\n   * @static\n   * @memberOf Path\n   * @param {HTMLElement} element to parse\n   * @param {Partial<PathProps>} [options] Options object\n   */\n  static async fromElement(\n    element: HTMLElement,\n    options: Partial<PathProps>,\n    cssRules?: CSSRules\n  ) {\n    const { d, ...parsedAttributes } = parseAttributes(\n      element,\n      this.ATTRIBUTE_NAMES,\n      cssRules\n    );\n    return new this(d, {\n      ...parsedAttributes,\n      ...options,\n      // we pass undefined to instruct the constructor to position the object using the bbox\n      left: undefined,\n      top: undefined,\n    });\n  }\n}\n\nclassRegistry.setClass(Path);\nclassRegistry.setSVGClass(Path);\n\n/* _FROM_SVG_START_ */\n","import type { ModifierKey, TEvent } from '../EventTypeDefs';\nimport type { Point } from '../Point';\nimport { Shadow } from '../Shadow';\nimport { Path } from '../shapes/Path';\nimport { getSmoothPathFromPoints, joinPath } from '../util/path';\nimport type { Canvas } from '../canvas/Canvas';\nimport { BaseBrush } from './BaseBrush';\nimport type { TSimplePathData } from '../util/path/typedefs';\n\n/**\n * @private\n * @param {TSimplePathData} pathData SVG path commands\n * @returns {boolean}\n */\nfunction isEmptySVGPath(pathData: TSimplePathData): boolean {\n  return joinPath(pathData) === 'M 0 0 Q 0 0 0 0 L 0 0';\n}\n\nexport class PencilBrush extends BaseBrush {\n  /**\n   * Discard points that are less than `decimate` pixel distant from each other\n   * @type Number\n   * @default 0.4\n   */\n  decimate = 0.4;\n\n  /**\n   * Draws a straight line between last recorded point to current pointer\n   * Used for `shift` functionality\n   *\n   * @type boolean\n   * @default false\n   */\n  drawStraightLine = false;\n\n  /**\n   * The event modifier key that makes the brush draw a straight line.\n   * If `null` or 'none' or any other string that is not a modifier key the feature is disabled.\n   * @type {ModifierKey | undefined | null}\n   */\n  straightLineKey: ModifierKey | undefined | null = 'shiftKey';\n\n  private declare _points: Point[];\n  private declare _hasStraightLine: boolean;\n  private declare oldEnd?: Point;\n\n  constructor(canvas: Canvas) {\n    super(canvas);\n    this._points = [];\n    this._hasStraightLine = false;\n  }\n\n  needsFullRender() {\n    return super.needsFullRender() || this._hasStraightLine;\n  }\n\n  static drawSegment(ctx: CanvasRenderingContext2D, p1: Point, p2: Point) {\n    const midPoint = p1.midPointFrom(p2);\n    ctx.quadraticCurveTo(p1.x, p1.y, midPoint.x, midPoint.y);\n    return midPoint;\n  }\n\n  /**\n   * Invoked on mouse down\n   * @param {Point} pointer\n   */\n  onMouseDown(pointer: Point, { e }: TEvent) {\n    if (!this.canvas._isMainEvent(e)) {\n      return;\n    }\n    this.drawStraightLine = !!this.straightLineKey && e[this.straightLineKey];\n    this._prepareForDrawing(pointer);\n    // capture coordinates immediately\n    // this allows to draw dots (when movement never occurs)\n    this._addPoint(pointer);\n    this._render();\n  }\n\n  /**\n   * Invoked on mouse move\n   * @param {Point} pointer\n   */\n  onMouseMove(pointer: Point, { e }: TEvent) {\n    if (!this.canvas._isMainEvent(e)) {\n      return;\n    }\n    this.drawStraightLine = !!this.straightLineKey && e[this.straightLineKey];\n    if (this.limitedToCanvasSize === true && this._isOutSideCanvas(pointer)) {\n      return;\n    }\n    if (this._addPoint(pointer) && this._points.length > 1) {\n      if (this.needsFullRender()) {\n        // redraw curve\n        // clear top canvas\n        this.canvas.clearContext(this.canvas.contextTop);\n        this._render();\n      } else {\n        const points = this._points,\n          length = points.length,\n          ctx = this.canvas.contextTop;\n        // draw the curve update\n        this._saveAndTransform(ctx);\n        if (this.oldEnd) {\n          ctx.beginPath();\n          ctx.moveTo(this.oldEnd.x, this.oldEnd.y);\n        }\n        this.oldEnd = PencilBrush.drawSegment(\n          ctx,\n          points[length - 2],\n          points[length - 1]\n        );\n        ctx.stroke();\n        ctx.restore();\n      }\n    }\n  }\n\n  /**\n   * Invoked on mouse up\n   */\n  onMouseUp({ e }: TEvent) {\n    if (!this.canvas._isMainEvent(e)) {\n      return true;\n    }\n    this.drawStraightLine = false;\n    this.oldEnd = undefined;\n    this._finalizeAndAddPath();\n    return false;\n  }\n\n  /**\n   * @private\n   * @param {Point} pointer Actual mouse position related to the canvas.\n   */\n  _prepareForDrawing(pointer: Point) {\n    this._reset();\n    this._addPoint(pointer);\n    this.canvas.contextTop.moveTo(pointer.x, pointer.y);\n  }\n\n  /**\n   * @private\n   * @param {Point} point Point to be added to points array\n   */\n  _addPoint(point: Point) {\n    if (\n      this._points.length > 1 &&\n      point.eq(this._points[this._points.length - 1])\n    ) {\n      return false;\n    }\n    if (this.drawStraightLine && this._points.length > 1) {\n      this._hasStraightLine = true;\n      this._points.pop();\n    }\n    this._points.push(point);\n    return true;\n  }\n\n  /**\n   * Clear points array and set contextTop canvas style.\n   * @private\n   */\n  _reset() {\n    this._points = [];\n    this._setBrushStyles(this.canvas.contextTop);\n    this._setShadow();\n    this._hasStraightLine = false;\n  }\n\n  /**\n   * Draw a smooth path on the topCanvas using quadraticCurveTo\n   * @private\n   * @param {CanvasRenderingContext2D} [ctx]\n   */\n  _render(ctx: CanvasRenderingContext2D = this.canvas.contextTop) {\n    let p1 = this._points[0],\n      p2 = this._points[1];\n    this._saveAndTransform(ctx);\n    ctx.beginPath();\n    //if we only have 2 points in the path and they are the same\n    //it means that the user only clicked the canvas without moving the mouse\n    //then we should be drawing a dot. A path isn't drawn between two identical dots\n    //that's why we set them apart a bit\n    if (this._points.length === 2 && p1.x === p2.x && p1.y === p2.y) {\n      const width = this.width / 1000;\n      p1.x -= width;\n      p2.x += width;\n    }\n    ctx.moveTo(p1.x, p1.y);\n\n    for (let i = 1; i < this._points.length; i++) {\n      // we pick the point between pi + 1 & pi + 2 as the\n      // end point and p1 as our control point.\n      PencilBrush.drawSegment(ctx, p1, p2);\n      p1 = this._points[i];\n      p2 = this._points[i + 1];\n    }\n    // Draw last line as a straight line while\n    // we wait for the next point to be able to calculate\n    // the bezier control point\n    ctx.lineTo(p1.x, p1.y);\n    ctx.stroke();\n    ctx.restore();\n  }\n\n  /**\n   * Converts points to SVG path\n   * @param {Point[]} points Array of points\n   * @return {TSimplePathData} SVG path commands\n   */\n  convertPointsToSVGPath(points: Point[]): TSimplePathData {\n    const correction = this.width / 1000;\n    return getSmoothPathFromPoints(points, correction);\n  }\n\n  /**\n   * Creates a Path object to add on canvas\n   * @param {TSimplePathData} pathData Path data\n   * @return {Path} Path to add on canvas\n   */\n  createPath(pathData: TSimplePathData): Path {\n    const path = new Path(pathData, {\n      fill: null,\n      stroke: this.color,\n      strokeWidth: this.width,\n      strokeLineCap: this.strokeLineCap,\n      strokeMiterLimit: this.strokeMiterLimit,\n      strokeLineJoin: this.strokeLineJoin,\n      strokeDashArray: this.strokeDashArray,\n    });\n    if (this.shadow) {\n      this.shadow.affectStroke = true;\n      path.shadow = new Shadow(this.shadow);\n    }\n\n    return path;\n  }\n\n  /**\n   * Decimate points array with the decimate value\n   */\n  decimatePoints(points: Point[], distance: number) {\n    if (points.length <= 2) {\n      return points;\n    }\n    let lastPoint = points[0],\n      cDistance;\n    const zoom = this.canvas.getZoom(),\n      adjustedDistance = Math.pow(distance / zoom, 2),\n      l = points.length - 1,\n      newPoints = [lastPoint];\n    for (let i = 1; i < l - 1; i++) {\n      cDistance =\n        Math.pow(lastPoint.x - points[i].x, 2) +\n        Math.pow(lastPoint.y - points[i].y, 2);\n      if (cDistance >= adjustedDistance) {\n        lastPoint = points[i];\n        newPoints.push(lastPoint);\n      }\n    }\n    // Add the last point from the original line to the end of the array.\n    // This ensures decimate doesn't delete the last point on the line, and ensures the line is > 1 point.\n    newPoints.push(points[l]);\n    return newPoints;\n  }\n\n  /**\n   * On mouseup after drawing the path on contextTop canvas\n   * we use the points captured to create an new Path object\n   * and add it to the canvas.\n   */\n  _finalizeAndAddPath() {\n    const ctx = this.canvas.contextTop;\n    ctx.closePath();\n    if (this.decimate) {\n      this._points = this.decimatePoints(this._points, this.decimate);\n    }\n    const pathData = this.convertPointsToSVGPath(this._points);\n    if (isEmptySVGPath(pathData)) {\n      // do not create 0 width/height paths, as they are\n      // rendered inconsistently across browsers\n      // Firefox 4, for example, renders a dot,\n      // whereas Chrome 10 renders nothing\n      this.canvas.requestRenderAll();\n      return;\n    }\n\n    const path = this.createPath(pathData);\n    this.canvas.clearContext(this.canvas.contextTop);\n    this.canvas.fire('before:path:created', { path: path });\n    this.canvas.add(path);\n    this.canvas.requestRenderAll();\n    path.setCoords();\n    this._resetShadow();\n\n    // fire event 'path' created\n    this.canvas.fire('path:created', { path: path });\n  }\n}\n","import type { ObjectEvents } from '../EventTypeDefs';\nimport { SHARED_ATTRIBUTES } from '../parser/attributes';\nimport { parseAttributes } from '../parser/parseAttributes';\nimport { cos } from '../util/misc/cos';\nimport { degreesToRadians } from '../util/misc/radiansDegreesConversion';\nimport { sin } from '../util/misc/sin';\nimport { classRegistry } from '../ClassRegistry';\nimport { FabricObject, cacheProperties } from './Object/FabricObject';\nimport type { Abortable, TClassProperties, TOptions } from '../typedefs';\nimport type { FabricObjectProps, SerializedObjectProps } from './Object/types';\nimport type { CSSRules } from '../parser/typedefs';\n\ninterface UniqueCircleProps {\n  /**\n   * Radius of this circle\n   * @type Number\n   * @default 0\n   */\n  radius: number;\n\n  /**\n   * Angle for the start of the circle, in degrees.\n   * @type TDegree 0 - 359\n   * @default 0\n   */\n  startAngle: number;\n\n  /**\n   * Angle for the end of the circle, in degrees\n   * @type TDegree 1 - 360\n   * @default 360\n   */\n  endAngle: number;\n\n  /**\n   * Orientation for the direction of the circle.\n   * Setting to true will switch the arc of the circle to traverse from startAngle to endAngle in a counter-clockwise direction.\n   * Note: this will only change how the circle is drawn, and does not affect rotational transformation.\n   * @default false\n   */\n  counterClockwise: boolean;\n}\n\nexport interface SerializedCircleProps\n  extends SerializedObjectProps,\n    UniqueCircleProps {}\n\nexport interface CircleProps extends FabricObjectProps, UniqueCircleProps {}\n\nconst CIRCLE_PROPS = [\n  'radius',\n  'startAngle',\n  'endAngle',\n  'counterClockwise',\n] as const;\n\nexport const circleDefaultValues: Partial<TClassProperties<Circle>> = {\n  radius: 0,\n  startAngle: 0,\n  endAngle: 360,\n  counterClockwise: false,\n};\n\nexport class Circle<\n    Props extends TOptions<CircleProps> = Partial<CircleProps>,\n    SProps extends SerializedCircleProps = SerializedCircleProps,\n    EventSpec extends ObjectEvents = ObjectEvents\n  >\n  extends FabricObject<Props, SProps, EventSpec>\n  implements UniqueCircleProps\n{\n  declare radius: number;\n  declare startAngle: number;\n  declare endAngle: number;\n  declare counterClockwise: boolean;\n\n  static type = 'Circle';\n\n  static cacheProperties = [...cacheProperties, ...CIRCLE_PROPS];\n\n  static ownDefaults = circleDefaultValues;\n\n  static getDefaults(): Record<string, any> {\n    return {\n      ...super.getDefaults(),\n      ...Circle.ownDefaults,\n    };\n  }\n\n  /**\n   * @private\n   * @param {String} key\n   * @param {*} value\n   */\n  _set(key: string, value: any) {\n    super._set(key, value);\n\n    if (key === 'radius') {\n      this.setRadius(value);\n    }\n\n    return this;\n  }\n\n  /**\n   * @private\n   * @param {CanvasRenderingContext2D} ctx context to render on\n   */\n  _render(ctx: CanvasRenderingContext2D) {\n    ctx.beginPath();\n    ctx.arc(\n      0,\n      0,\n      this.radius,\n      degreesToRadians(this.startAngle),\n      degreesToRadians(this.endAngle),\n      this.counterClockwise\n    );\n    this._renderPaintInOrder(ctx);\n  }\n\n  /**\n   * Returns horizontal radius of an object (according to how an object is scaled)\n   * @return {Number}\n   */\n  getRadiusX(): number {\n    return this.get('radius') * this.get('scaleX');\n  }\n\n  /**\n   * Returns vertical radius of an object (according to how an object is scaled)\n   * @return {Number}\n   */\n  getRadiusY(): number {\n    return this.get('radius') * this.get('scaleY');\n  }\n\n  /**\n   * Sets radius of an object (and updates width accordingly)\n   */\n  setRadius(value: number) {\n    this.radius = value;\n    this.set({ width: value * 2, height: value * 2 });\n  }\n\n  /**\n   * Returns object representation of an instance\n   * @param {Array} [propertiesToInclude] Any properties that you might want to additionally include in the output\n   * @return {Object} object representation of an instance\n   */\n  toObject<\n    T extends Omit<Props & TClassProperties<this>, keyof SProps>,\n    K extends keyof T = never\n  >(propertiesToInclude: K[] = []): Pick<T, K> & SProps {\n    return super.toObject([...CIRCLE_PROPS, ...propertiesToInclude]);\n  }\n\n  /* _TO_SVG_START_ */\n\n  /**\n   * Returns svg representation of an instance\n   * @return {Array} an array of strings with the specific svg representation\n   * of the instance\n   */\n  _toSVG(): string[] {\n    const angle = (this.endAngle - this.startAngle) % 360;\n\n    if (angle === 0) {\n      return [\n        '<circle ',\n        'COMMON_PARTS',\n        'cx=\"0\" cy=\"0\" ',\n        'r=\"',\n        `${this.radius}`,\n        '\" />\\n',\n      ];\n    } else {\n      const { radius } = this;\n      const start = degreesToRadians(this.startAngle),\n        end = degreesToRadians(this.endAngle),\n        startX = cos(start) * radius,\n        startY = sin(start) * radius,\n        endX = cos(end) * radius,\n        endY = sin(end) * radius,\n        largeFlag = angle > 180 ? 1 : 0,\n        sweepFlag = this.counterClockwise ? 0 : 1;\n      return [\n        `<path d=\"M ${startX} ${startY} A ${radius} ${radius} 0 ${largeFlag} ${sweepFlag} ${endX} ${endY}\" `,\n        'COMMON_PARTS',\n        ' />\\n',\n      ];\n    }\n  }\n  /* _TO_SVG_END_ */\n\n  /* _FROM_SVG_START_ */\n  /**\n   * List of attribute names to account for when parsing SVG element (used by {@link Circle.fromElement})\n   * @static\n   * @memberOf Circle\n   * @see: http://www.w3.org/TR/SVG/shapes.html#CircleElement\n   */\n  static ATTRIBUTE_NAMES = ['cx', 'cy', 'r', ...SHARED_ATTRIBUTES];\n\n  /**\n   * Returns {@link Circle} instance from an SVG element\n   * @static\n   * @memberOf Circle\n   * @param {HTMLElement} element Element to parse\n   * @param {Object} [options] Partial Circle object to default missing properties on the element.\n   * @throws {Error} If value of `r` attribute is missing or invalid\n   */\n  static async fromElement(\n    element: HTMLElement,\n    options: Abortable,\n    cssRules?: CSSRules\n  ): Promise<Circle> {\n    const {\n      left = 0,\n      top = 0,\n      radius = 0,\n      ...otherParsedAttributes\n    } = parseAttributes(\n      element,\n      this.ATTRIBUTE_NAMES,\n      cssRules\n    ) as Partial<CircleProps>;\n\n    // this probably requires to be fixed for default origins not being top/left.\n\n    return new this({\n      ...otherParsedAttributes,\n      radius,\n      left: left - radius,\n      top: top - radius,\n    });\n  }\n\n  /* _FROM_SVG_END_ */\n\n  /**\n   * @todo how do we declare this??\n   */\n  static fromObject<T extends TOptions<SerializedCircleProps>>(object: T) {\n    return super._fromObject<Circle>(object);\n  }\n}\n\nclassRegistry.setClass(Circle);\nclassRegistry.setSVGClass(Circle);\n","import { Color } from '../color/Color';\nimport type { Point } from '../Point';\nimport { Shadow } from '../Shadow';\nimport { Circle } from '../shapes/Circle';\nimport { Group } from '../shapes/Group';\nimport { getRandomInt } from '../util/internals';\nimport type { Canvas } from '../canvas/Canvas';\nimport { BaseBrush } from './BaseBrush';\nimport type { CircleBrushPoint } from './typedefs';\nimport { CENTER } from '../constants';\n\nexport class CircleBrush extends BaseBrush {\n  /**\n   * Width of a brush\n   * @type Number\n   * @default\n   */\n  width = 10;\n\n  declare points: CircleBrushPoint[];\n\n  constructor(canvas: Canvas) {\n    super(canvas);\n    this.points = [];\n  }\n\n  /**\n   * Invoked inside on mouse down and mouse move\n   * @param {Point} pointer\n   */\n  drawDot(pointer: Point) {\n    const point = this.addPoint(pointer),\n      ctx = this.canvas.contextTop;\n    this._saveAndTransform(ctx);\n    this.dot(ctx, point);\n    ctx.restore();\n  }\n\n  dot(ctx: CanvasRenderingContext2D, point: CircleBrushPoint) {\n    ctx.fillStyle = point.fill;\n    ctx.beginPath();\n    ctx.arc(point.x, point.y, point.radius, 0, Math.PI * 2, false);\n    ctx.closePath();\n    ctx.fill();\n  }\n\n  /**\n   * Invoked on mouse down\n   */\n  onMouseDown(pointer: Point) {\n    this.points = [];\n    this.canvas.clearContext(this.canvas.contextTop);\n    this._setShadow();\n    this.drawDot(pointer);\n  }\n\n  /**\n   * Render the full state of the brush\n   * @private\n   */\n  _render() {\n    const ctx = this.canvas.contextTop,\n      points = this.points;\n    this._saveAndTransform(ctx);\n    for (let i = 0; i < points.length; i++) {\n      this.dot(ctx, points[i]);\n    }\n    ctx.restore();\n  }\n\n  /**\n   * Invoked on mouse move\n   * @param {Point} pointer\n   */\n  onMouseMove(pointer: Point) {\n    if (this.limitedToCanvasSize === true && this._isOutSideCanvas(pointer)) {\n      return;\n    }\n    if (this.needsFullRender()) {\n      this.canvas.clearContext(this.canvas.contextTop);\n      this.addPoint(pointer);\n      this._render();\n    } else {\n      this.drawDot(pointer);\n    }\n  }\n\n  /**\n   * Invoked on mouse up\n   */\n  onMouseUp() {\n    const originalRenderOnAddRemove = this.canvas.renderOnAddRemove;\n    this.canvas.renderOnAddRemove = false;\n\n    const circles: Circle[] = [];\n\n    for (let i = 0; i < this.points.length; i++) {\n      const point = this.points[i],\n        circle = new Circle({\n          radius: point.radius,\n          left: point.x,\n          top: point.y,\n          originX: CENTER,\n          originY: CENTER,\n          fill: point.fill,\n        });\n\n      this.shadow && (circle.shadow = new Shadow(this.shadow));\n\n      circles.push(circle);\n    }\n    const group = new Group(circles, { canvas: this.canvas });\n\n    this.canvas.fire('before:path:created', { path: group });\n    this.canvas.add(group);\n    this.canvas.fire('path:created', { path: group });\n\n    this.canvas.clearContext(this.canvas.contextTop);\n    this._resetShadow();\n    this.canvas.renderOnAddRemove = originalRenderOnAddRemove;\n    this.canvas.requestRenderAll();\n  }\n\n  /**\n   * @param {Object} pointer\n   * @return {Point} Just added pointer point\n   */\n  addPoint({ x, y }: Point) {\n    const pointerPoint: CircleBrushPoint = {\n      x,\n      y,\n      radius: getRandomInt(Math.max(0, this.width - 20), this.width + 20) / 2,\n      fill: new Color(this.color).setAlpha(getRandomInt(0, 100) / 100).toRgba(),\n    };\n\n    this.points.push(pointerPoint);\n\n    return pointerPoint;\n  }\n}\n","import type { Point } from '../Point';\nimport { Group } from '../shapes/Group';\nimport { Shadow } from '../Shadow';\nimport { Rect } from '../shapes/Rect';\nimport { getRandomInt } from '../util/internals';\nimport type { Canvas } from '../canvas/Canvas';\nimport { BaseBrush } from './BaseBrush';\nimport type { SprayBrushPoint } from './typedefs';\nimport { CENTER } from '../constants';\n\n/**\n *\n * @param rects\n * @returns\n */\nfunction getUniqueRects(rects: Rect[]) {\n  const uniqueRects: Record<string, boolean> = {};\n  const uniqueRectsArray: Rect[] = [];\n\n  for (let i = 0, key: string; i < rects.length; i++) {\n    key = `${rects[i].left}${rects[i].top}`;\n    if (!uniqueRects[key]) {\n      uniqueRects[key] = true;\n      uniqueRectsArray.push(rects[i]);\n    }\n  }\n\n  return uniqueRectsArray;\n}\n\nexport class SprayBrush extends BaseBrush {\n  /**\n   * Width of a spray\n   * @type Number\n   * @default\n   */\n  width = 10;\n\n  /**\n   * Density of a spray (number of dots per chunk)\n   * @type Number\n   * @default\n   */\n  density = 20;\n\n  /**\n   * Width of spray dots\n   * @type Number\n   * @default\n   */\n  dotWidth = 1;\n\n  /**\n   * Width variance of spray dots\n   * @type Number\n   * @default\n   */\n  dotWidthVariance = 1;\n\n  /**\n   * Whether opacity of a dot should be random\n   * @type Boolean\n   * @default\n   */\n  randomOpacity = false;\n\n  /**\n   * Whether overlapping dots (rectangles) should be removed (for performance reasons)\n   * @type Boolean\n   * @default\n   */\n  optimizeOverlapping = true;\n\n  private declare sprayChunks: SprayBrushPoint[][];\n\n  private declare sprayChunk: SprayBrushPoint[];\n\n  /**\n   * Constructor\n   * @param {Canvas} canvas\n   * @return {SprayBrush} Instance of a spray brush\n   */\n  constructor(canvas: Canvas) {\n    super(canvas);\n    this.sprayChunks = [];\n    this.sprayChunk = [];\n  }\n\n  /**\n   * Invoked on mouse down\n   * @param {Point} pointer\n   */\n  onMouseDown(pointer: Point) {\n    this.sprayChunks = [];\n    this.canvas.clearContext(this.canvas.contextTop);\n    this._setShadow();\n\n    this.addSprayChunk(pointer);\n    this.renderChunck(this.sprayChunk);\n  }\n\n  /**\n   * Invoked on mouse move\n   * @param {Point} pointer\n   */\n  onMouseMove(pointer: Point) {\n    if (this.limitedToCanvasSize === true && this._isOutSideCanvas(pointer)) {\n      return;\n    }\n    this.addSprayChunk(pointer);\n    this.renderChunck(this.sprayChunk);\n  }\n\n  /**\n   * Invoked on mouse up\n   */\n  onMouseUp() {\n    const originalRenderOnAddRemove = this.canvas.renderOnAddRemove;\n    this.canvas.renderOnAddRemove = false;\n\n    const rects: Rect[] = [];\n\n    for (let i = 0; i < this.sprayChunks.length; i++) {\n      const sprayChunk = this.sprayChunks[i];\n      for (let j = 0; j < sprayChunk.length; j++) {\n        const chunck = sprayChunk[j];\n        const rect = new Rect({\n          width: chunck.width,\n          height: chunck.width,\n          left: chunck.x + 1,\n          top: chunck.y + 1,\n          originX: CENTER,\n          originY: CENTER,\n          fill: this.color,\n        });\n        rects.push(rect);\n      }\n    }\n\n    const group = new Group(\n      this.optimizeOverlapping ? getUniqueRects(rects) : rects,\n      {\n        objectCaching: true,\n        subTargetCheck: false,\n        interactive: false,\n      }\n    );\n    this.shadow && group.set('shadow', new Shadow(this.shadow));\n    this.canvas.fire('before:path:created', { path: group });\n    this.canvas.add(group);\n    this.canvas.fire('path:created', { path: group });\n\n    this.canvas.clearContext(this.canvas.contextTop);\n    this._resetShadow();\n    this.canvas.renderOnAddRemove = originalRenderOnAddRemove;\n    this.canvas.requestRenderAll();\n  }\n\n  renderChunck(sprayChunck: SprayBrushPoint[]) {\n    const ctx = this.canvas.contextTop;\n    ctx.fillStyle = this.color;\n\n    this._saveAndTransform(ctx);\n\n    for (let i = 0; i < sprayChunck.length; i++) {\n      const point = sprayChunck[i];\n      ctx.globalAlpha = point.opacity;\n      ctx.fillRect(point.x, point.y, point.width, point.width);\n    }\n\n    ctx.restore();\n  }\n\n  /**\n   * Render all spray chunks\n   */\n  _render() {\n    const ctx = this.canvas.contextTop;\n    ctx.fillStyle = this.color;\n\n    this._saveAndTransform(ctx);\n\n    for (let i = 0; i < this.sprayChunks.length; i++) {\n      this.renderChunck(this.sprayChunks[i]);\n    }\n    ctx.restore();\n  }\n\n  /**\n   * @param {Point} pointer\n   */\n  addSprayChunk(pointer: Point) {\n    this.sprayChunk = [];\n    const radius = this.width / 2;\n\n    for (let i = 0; i < this.density; i++) {\n      this.sprayChunk.push({\n        x: getRandomInt(pointer.x - radius, pointer.x + radius),\n        y: getRandomInt(pointer.y - radius, pointer.y + radius),\n        width: this.dotWidthVariance\n          ? getRandomInt(\n              // bottom clamp width to 1\n              Math.max(1, this.dotWidth - this.dotWidthVariance),\n              this.dotWidth + this.dotWidthVariance\n            )\n          : this.dotWidth,\n        opacity: this.randomOpacity ? getRandomInt(0, 100) / 100 : 1,\n      });\n    }\n\n    this.sprayChunks.push(this.sprayChunk);\n  }\n}\n","import { Pattern } from '../Pattern';\nimport { createCanvasElement } from '../util/misc/dom';\nimport type { Canvas } from '../canvas/Canvas';\nimport { PencilBrush } from './PencilBrush';\nimport type { TSimplePathData } from '../util/path/typedefs';\n\nexport class PatternBrush extends PencilBrush {\n  declare source?: CanvasImageSource;\n\n  constructor(canvas: Canvas) {\n    super(canvas);\n  }\n\n  getPatternSrc() {\n    const dotWidth = 20,\n      dotDistance = 5,\n      patternCanvas = createCanvasElement(),\n      patternCtx = patternCanvas.getContext('2d');\n\n    patternCanvas.width = patternCanvas.height = dotWidth + dotDistance;\n    if (patternCtx) {\n      patternCtx.fillStyle = this.color;\n      patternCtx.beginPath();\n      patternCtx.arc(\n        dotWidth / 2,\n        dotWidth / 2,\n        dotWidth / 2,\n        0,\n        Math.PI * 2,\n        false\n      );\n      patternCtx.closePath();\n      patternCtx.fill();\n    }\n    return patternCanvas;\n  }\n\n  /**\n   * Creates \"pattern\" instance property\n   * @param {CanvasRenderingContext2D} ctx\n   */\n  getPattern(ctx: CanvasRenderingContext2D) {\n    return ctx.createPattern(this.source || this.getPatternSrc(), 'repeat');\n  }\n\n  /**\n   * Sets brush styles\n   * @param {CanvasRenderingContext2D} ctx\n   */\n  _setBrushStyles(ctx: CanvasRenderingContext2D) {\n    super._setBrushStyles(ctx);\n    const pattern = this.getPattern(ctx);\n    pattern && (ctx.strokeStyle = pattern);\n  }\n\n  /**\n   * Creates path\n   */\n  createPath(pathData: TSimplePathData) {\n    const path = super.createPath(pathData),\n      topLeft = path._getLeftTopCoords().scalarAdd(path.strokeWidth / 2);\n\n    path.stroke = new Pattern({\n      source: this.source || this.getPatternSrc(),\n      offsetX: -topLeft.x,\n      offsetY: -topLeft.y,\n    });\n    return path;\n  }\n}\n","import { SHARED_ATTRIBUTES } from '../parser/attributes';\nimport { parseAttributes } from '../parser/parseAttributes';\nimport type { Abortable, TClassProperties, TOptions } from '../typedefs';\nimport { classRegistry } from '../ClassRegistry';\nimport { FabricObject, cacheProperties } from './Object/FabricObject';\nimport { Point } from '../Point';\nimport { isFiller } from '../util/typeAssertions';\nimport type { FabricObjectProps, SerializedObjectProps } from './Object/types';\nimport type { ObjectEvents } from '../EventTypeDefs';\nimport { makeBoundingBoxFromPoints } from '../util';\nimport { CENTER, LEFT, TOP } from '../constants';\nimport type { CSSRules } from '../parser/typedefs';\n\n// @TODO this code is terrible and Line should be a special case of polyline.\n\nconst coordProps = ['x1', 'x2', 'y1', 'y2'] as const;\n\ninterface UniqueLineProps {\n  x1: number;\n  x2: number;\n  y1: number;\n  y2: number;\n}\n\nexport interface SerializedLineProps\n  extends SerializedObjectProps,\n    UniqueLineProps {}\n\nexport class Line<\n    Props extends TOptions<FabricObjectProps> = Partial<FabricObjectProps>,\n    SProps extends SerializedLineProps = SerializedLineProps,\n    EventSpec extends ObjectEvents = ObjectEvents\n  >\n  extends FabricObject<Props, SProps, EventSpec>\n  implements UniqueLineProps\n{\n  /**\n   * x value or first line edge\n   * @type number\n   * @default\n   */\n  declare x1: number;\n\n  /**\n   * y value or first line edge\n   * @type number\n   * @default\n   */\n  declare y1: number;\n\n  /**\n   * x value or second line edge\n   * @type number\n   * @default\n   */\n  declare x2: number;\n\n  /**\n   * y value or second line edge\n   * @type number\n   * @default\n   */\n  declare y2: number;\n\n  static type = 'Line';\n\n  static cacheProperties = [...cacheProperties, ...coordProps];\n  /**\n   * Constructor\n   * @param {Array} [points] Array of points\n   * @param {Object} [options] Options object\n   * @return {Line} thisArg\n   */\n  constructor([x1, y1, x2, y2] = [0, 0, 0, 0], options: Props = {} as Props) {\n    super({ ...options, x1, y1, x2, y2 });\n    this._setWidthHeight();\n    const { left, top } = options;\n    typeof left === 'number' && this.set(LEFT, left);\n    typeof top === 'number' && this.set(TOP, top);\n  }\n\n  /**\n   * @private\n   * @param {Object} [options] Options\n   */\n  _setWidthHeight() {\n    const { x1, y1, x2, y2 } = this;\n    this.width = Math.abs(x2 - x1);\n    this.height = Math.abs(y2 - y1);\n    const { left, top, width, height } = makeBoundingBoxFromPoints([\n      { x: x1, y: y1 },\n      { x: x2, y: y2 },\n    ]);\n    const position = new Point(left + width / 2, top + height / 2);\n    this.setPositionByOrigin(position, CENTER, CENTER);\n  }\n\n  /**\n   * @private\n   * @param {String} key\n   * @param {*} value\n   */\n  _set(key: string, value: any) {\n    super._set(key, value);\n    if (coordProps.includes(key as keyof UniqueLineProps)) {\n      // this doesn't make sense very much, since setting x1 when top or left\n      // are already set, is just going to show a strange result since the\n      // line will move way more than the developer expect.\n      // in fabric5 it worked only when the line didn't have extra transformations,\n      // in fabric6 too. With extra transform they behave bad in different ways.\n      // This needs probably a good rework or a tutorial if you have to create a dynamic line\n      this._setWidthHeight();\n    }\n    return this;\n  }\n\n  /**\n   * @private\n   * @param {CanvasRenderingContext2D} ctx Context to render on\n   */\n  _render(ctx: CanvasRenderingContext2D) {\n    ctx.beginPath();\n\n    const p = this.calcLinePoints();\n    ctx.moveTo(p.x1, p.y1);\n    ctx.lineTo(p.x2, p.y2);\n\n    ctx.lineWidth = this.strokeWidth;\n\n    // TODO: test this\n    // make sure setting \"fill\" changes color of a line\n    // (by copying fillStyle to strokeStyle, since line is stroked, not filled)\n    const origStrokeStyle = ctx.strokeStyle;\n    if (isFiller(this.stroke)) {\n      ctx.strokeStyle = this.stroke.toLive(ctx)!;\n    } else {\n      ctx.strokeStyle = this.stroke ?? ctx.fillStyle;\n    }\n    this.stroke && this._renderStroke(ctx);\n    ctx.strokeStyle = origStrokeStyle;\n  }\n\n  /**\n   * This function is an helper for svg import. it returns the center of the object in the svg\n   * untransformed coordinates\n   * @private\n   * @return {Point} center point from element coordinates\n   */\n  _findCenterFromElement(): Point {\n    return new Point((this.x1 + this.x2) / 2, (this.y1 + this.y2) / 2);\n  }\n\n  /**\n   * Returns object representation of an instance\n   * @method toObject\n   * @param {Array} [propertiesToInclude] Any properties that you might want to additionally include in the output\n   * @return {Object} object representation of an instance\n   */\n  toObject<\n    T extends Omit<Props & TClassProperties<this>, keyof SProps>,\n    K extends keyof T = never\n  >(propertiesToInclude: K[] = []): Pick<T, K> & SProps {\n    return {\n      ...super.toObject(propertiesToInclude),\n      ...this.calcLinePoints(),\n    };\n  }\n\n  /*\n   * Calculate object dimensions from its properties\n   * @private\n   */\n  _getNonTransformedDimensions(): Point {\n    const dim = super._getNonTransformedDimensions();\n    if (this.strokeLineCap === 'butt') {\n      if (this.width === 0) {\n        dim.y -= this.strokeWidth;\n      }\n      if (this.height === 0) {\n        dim.x -= this.strokeWidth;\n      }\n    }\n    return dim;\n  }\n\n  /**\n   * Recalculates line points given width and height\n   * Those points are simply placed around the center,\n   * This is not useful outside internal render functions and svg output\n   * Is not meant to be for the developer.\n   * @private\n   */\n  calcLinePoints(): UniqueLineProps {\n    const { x1: _x1, x2: _x2, y1: _y1, y2: _y2, width, height } = this;\n    const xMult = _x1 <= _x2 ? -1 : 1,\n      yMult = _y1 <= _y2 ? -1 : 1,\n      x1 = (xMult * width) / 2,\n      y1 = (yMult * height) / 2,\n      x2 = (xMult * -width) / 2,\n      y2 = (yMult * -height) / 2;\n\n    return {\n      x1,\n      x2,\n      y1,\n      y2,\n    };\n  }\n\n  /* _FROM_SVG_START_ */\n\n  /**\n   * Returns svg representation of an instance\n   * @return {Array} an array of strings with the specific svg representation\n   * of the instance\n   */\n  _toSVG() {\n    const { x1, x2, y1, y2 } = this.calcLinePoints();\n    return [\n      '<line ',\n      'COMMON_PARTS',\n      `x1=\"${x1}\" y1=\"${y1}\" x2=\"${x2}\" y2=\"${y2}\" />\\n`,\n    ];\n  }\n\n  /**\n   * List of attribute names to account for when parsing SVG element (used by {@link Line.fromElement})\n   * @static\n   * @memberOf Line\n   * @see http://www.w3.org/TR/SVG/shapes.html#LineElement\n   */\n  static ATTRIBUTE_NAMES = SHARED_ATTRIBUTES.concat(coordProps);\n\n  /**\n   * Returns Line instance from an SVG element\n   * @static\n   * @memberOf Line\n   * @param {HTMLElement} element Element to parse\n   * @param {Object} [options] Options object\n   * @param {Function} [callback] callback function invoked after parsing\n   */\n  static async fromElement(\n    element: HTMLElement,\n    options: Abortable,\n    cssRules?: CSSRules\n  ) {\n    const {\n      x1 = 0,\n      y1 = 0,\n      x2 = 0,\n      y2 = 0,\n      ...parsedAttributes\n    } = parseAttributes(element, this.ATTRIBUTE_NAMES, cssRules);\n    return new this([x1, y1, x2, y2], parsedAttributes);\n  }\n\n  /* _FROM_SVG_END_ */\n\n  /**\n   * Returns Line instance from an object representation\n   * @static\n   * @memberOf Line\n   * @param {Object} object Object to create an instance from\n   * @returns {Promise<Line>}\n   */\n  static fromObject<T extends TOptions<SerializedLineProps>>({\n    x1,\n    y1,\n    x2,\n    y2,\n    ...object\n  }: T) {\n    return this._fromObject<Line>(\n      {\n        ...object,\n        points: [x1, y1, x2, y2],\n      },\n      {\n        extraParam: 'points',\n      }\n    );\n  }\n}\n\nclassRegistry.setClass(Line);\nclassRegistry.setSVGClass(Line);\n","import { classRegistry } from '../ClassRegistry';\nimport { FabricObject } from './Object/FabricObject';\nimport type { FabricObjectProps, SerializedObjectProps } from './Object/types';\nimport type { TClassProperties, TOptions } from '../typedefs';\nimport type { ObjectEvents } from '../EventTypeDefs';\n\nexport const triangleDefaultValues: Partial<TClassProperties<Triangle>> = {\n  width: 100,\n  height: 100,\n};\n\nexport class Triangle<\n    Props extends TOptions<FabricObjectProps> = Partial<FabricObjectProps>,\n    SProps extends SerializedObjectProps = SerializedObjectProps,\n    EventSpec extends ObjectEvents = ObjectEvents\n  >\n  extends FabricObject<Props, SProps, EventSpec>\n  implements FabricObjectProps\n{\n  static type = 'Triangle';\n\n  static ownDefaults = triangleDefaultValues;\n\n  static getDefaults(): Record<string, any> {\n    return { ...super.getDefaults(), ...Triangle.ownDefaults };\n  }\n\n  /**\n   * @private\n   * @param {CanvasRenderingContext2D} ctx Context to render on\n   */\n  _render(ctx: CanvasRenderingContext2D) {\n    const widthBy2 = this.width / 2,\n      heightBy2 = this.height / 2;\n\n    ctx.beginPath();\n    ctx.moveTo(-widthBy2, heightBy2);\n    ctx.lineTo(0, -heightBy2);\n    ctx.lineTo(widthBy2, heightBy2);\n    ctx.closePath();\n\n    this._renderPaintInOrder(ctx);\n  }\n\n  /**\n   * Returns svg representation of an instance\n   * @return {Array} an array of strings with the specific svg representation\n   * of the instance\n   */\n  _toSVG() {\n    const widthBy2 = this.width / 2,\n      heightBy2 = this.height / 2,\n      points = `${-widthBy2} ${heightBy2},0 ${-heightBy2},${widthBy2} ${heightBy2}`;\n    return ['<polygon ', 'COMMON_PARTS', 'points=\"', points, '\" />'];\n  }\n}\n\nclassRegistry.setClass(Triangle);\nclassRegistry.setSVGClass(Triangle);\n","import { twoMathPi } from '../constants';\nimport { SHARED_ATTRIBUTES } from '../parser/attributes';\nimport { parseAttributes } from '../parser/parseAttributes';\nimport type { Abortable, TClassProperties, TOptions } from '../typedefs';\nimport { classRegistry } from '../ClassRegistry';\nimport { FabricObject, cacheProperties } from './Object/FabricObject';\nimport type { FabricObjectProps, SerializedObjectProps } from './Object/types';\nimport type { ObjectEvents } from '../EventTypeDefs';\nimport type { CSSRules } from '../parser/typedefs';\n\nexport const ellipseDefaultValues: Partial<TClassProperties<Ellipse>> = {\n  rx: 0,\n  ry: 0,\n};\n\ninterface UniqueEllipseProps {\n  rx: number;\n  ry: number;\n}\n\nexport interface SerializedEllipseProps\n  extends SerializedObjectProps,\n    UniqueEllipseProps {}\n\nexport interface EllipseProps extends FabricObjectProps, UniqueEllipseProps {}\n\nconst ELLIPSE_PROPS = ['rx', 'ry'] as const;\n\nexport class Ellipse<\n    Props extends TOptions<EllipseProps> = Partial<EllipseProps>,\n    SProps extends SerializedEllipseProps = SerializedEllipseProps,\n    EventSpec extends ObjectEvents = ObjectEvents\n  >\n  extends FabricObject<Props, SProps, EventSpec>\n  implements EllipseProps\n{\n  /**\n   * Horizontal radius\n   * @type Number\n   * @default\n   */\n  declare rx: number;\n\n  /**\n   * Vertical radius\n   * @type Number\n   * @default\n   */\n  declare ry: number;\n\n  static type = 'Ellipse';\n\n  static cacheProperties = [...cacheProperties, ...ELLIPSE_PROPS];\n\n  static ownDefaults = ellipseDefaultValues;\n\n  static getDefaults(): Record<string, any> {\n    return {\n      ...super.getDefaults(),\n      ...Ellipse.ownDefaults,\n    };\n  }\n\n  /**\n   * @private\n   * @param {String} key\n   * @param {*} value\n   * @return {Ellipse} thisArg\n   */\n  _set(key: string, value: any) {\n    super._set(key, value);\n    switch (key) {\n      case 'rx':\n        this.rx = value;\n        this.set('width', value * 2);\n        break;\n\n      case 'ry':\n        this.ry = value;\n        this.set('height', value * 2);\n        break;\n    }\n    return this;\n  }\n\n  /**\n   * Returns horizontal radius of an object (according to how an object is scaled)\n   * @return {Number}\n   */\n  getRx() {\n    return this.get('rx') * this.get('scaleX');\n  }\n\n  /**\n   * Returns Vertical radius of an object (according to how an object is scaled)\n   * @return {Number}\n   */\n  getRy() {\n    return this.get('ry') * this.get('scaleY');\n  }\n\n  /**\n   * Returns object representation of an instance\n   * @param {Array} [propertiesToInclude] Any properties that you might want to additionally include in the output\n   * @return {Object} object representation of an instance\n   */\n  toObject<\n    T extends Omit<Props & TClassProperties<this>, keyof SProps>,\n    K extends keyof T = never\n  >(propertiesToInclude: K[] = []): Pick<T, K> & SProps {\n    return super.toObject([...ELLIPSE_PROPS, ...propertiesToInclude]);\n  }\n\n  /**\n   * Returns svg representation of an instance\n   * @return {Array} an array of strings with the specific svg representation\n   * of the instance\n   */\n  _toSVG(): string[] {\n    return [\n      '<ellipse ',\n      'COMMON_PARTS',\n      `cx=\"0\" cy=\"0\" rx=\"${this.rx}\" ry=\"${this.ry}\" />\\n`,\n    ];\n  }\n\n  /**\n   * @private\n   * @param {CanvasRenderingContext2D} ctx context to render on\n   */\n  _render(ctx: CanvasRenderingContext2D) {\n    ctx.beginPath();\n    ctx.save();\n    ctx.transform(1, 0, 0, this.ry / this.rx, 0, 0);\n    ctx.arc(0, 0, this.rx, 0, twoMathPi, false);\n    ctx.restore();\n    this._renderPaintInOrder(ctx);\n  }\n\n  /* _FROM_SVG_START_ */\n\n  /**\n   * List of attribute names to account for when parsing SVG element (used by {@link Ellipse.fromElement})\n   * @static\n   * @memberOf Ellipse\n   * @see http://www.w3.org/TR/SVG/shapes.html#EllipseElement\n   */\n  static ATTRIBUTE_NAMES = [...SHARED_ATTRIBUTES, 'cx', 'cy', 'rx', 'ry'];\n\n  /**\n   * Returns {@link Ellipse} instance from an SVG element\n   * @static\n   * @memberOf Ellipse\n   * @param {HTMLElement} element Element to parse\n   * @return {Ellipse}\n   */\n  static async fromElement(\n    element: HTMLElement,\n    options: Abortable,\n    cssRules?: CSSRules\n  ) {\n    const parsedAttributes = parseAttributes(\n      element,\n      this.ATTRIBUTE_NAMES,\n      cssRules\n    );\n\n    parsedAttributes.left = (parsedAttributes.left || 0) - parsedAttributes.rx;\n    parsedAttributes.top = (parsedAttributes.top || 0) - parsedAttributes.ry;\n    return new this(parsedAttributes);\n  }\n\n  /* _FROM_SVG_END_ */\n}\n\nclassRegistry.setClass(Ellipse);\nclassRegistry.setSVGClass(Ellipse);\n","import type { XY } from '../Point';\n\n/**\n * Parses \"points\" attribute, returning an array of values\n * @static\n * @memberOf fabric\n * @param {String} points points attribute string\n * @return {Array} array of points\n */\nexport function parsePointsAttribute(points: string | null): XY[] {\n  // points attribute is required and must not be empty\n  if (!points) {\n    return [];\n  }\n\n  // replace commas with whitespace and remove bookending whitespace\n  const pointsSplit: string[] = points.replace(/,/g, ' ').trim().split(/\\s+/);\n\n  const parsedPoints = [];\n\n  for (let i = 0; i < pointsSplit.length; i += 2) {\n    parsedPoints.push({\n      x: parseFloat(pointsSplit[i]),\n      y: parseFloat(pointsSplit[i + 1]),\n    });\n  }\n\n  // odd number of points is an error\n  // if (parsedPoints.length % 2 !== 0) {\n  //   return null;\n  // }\n  return parsedPoints;\n}\n","import { config } from '../config';\nimport { SHARED_ATTRIBUTES } from '../parser/attributes';\nimport { parseAttributes } from '../parser/parseAttributes';\nimport { parsePointsAttribute } from '../parser/parsePointsAttribute';\nimport type { XY } from '../Point';\nimport { Point } from '../Point';\nimport type { Abortable, TClassProperties, TOptions } from '../typedefs';\nimport { classRegistry } from '../ClassRegistry';\nimport { makeBoundingBoxFromPoints } from '../util/misc/boundingBoxFromPoints';\nimport { calcDimensionsMatrix, transformPoint } from '../util/misc/matrix';\nimport { projectStrokeOnPoints } from '../util/misc/projectStroke';\nimport type { TProjectStrokeOnPointsOptions } from '../util/misc/projectStroke/types';\nimport { degreesToRadians } from '../util/misc/radiansDegreesConversion';\nimport { toFixed } from '../util/misc/toFixed';\nimport { FabricObject, cacheProperties } from './Object/FabricObject';\nimport type { FabricObjectProps, SerializedObjectProps } from './Object/types';\nimport type { ObjectEvents } from '../EventTypeDefs';\nimport { cloneDeep } from '../util/internals/cloneDeep';\nimport { CENTER, LEFT, TOP } from '../constants';\nimport type { CSSRules } from '../parser/typedefs';\n\nexport const polylineDefaultValues: Partial<TClassProperties<Polyline>> = {\n  /**\n   * @deprecated transient option soon to be removed in favor of a different design\n   */\n  exactBoundingBox: false,\n};\n\nexport interface SerializedPolylineProps extends SerializedObjectProps {\n  points: XY[];\n}\n\nexport class Polyline<\n  Props extends TOptions<FabricObjectProps> = Partial<FabricObjectProps>,\n  SProps extends SerializedPolylineProps = SerializedPolylineProps,\n  EventSpec extends ObjectEvents = ObjectEvents\n> extends FabricObject<Props, SProps, EventSpec> {\n  /**\n   * Points array\n   * @type Array\n   * @default\n   */\n  declare points: XY[];\n\n  /**\n   * WARNING: Feature in progress\n   * Calculate the exact bounding box taking in account strokeWidth on acute angles\n   * this will be turned to true by default on fabric 6.0\n   * maybe will be left in as an optimization since calculations may be slow\n   * @deprecated transient option soon to be removed in favor of a different design\n   * @type Boolean\n   * @default false\n   */\n  declare exactBoundingBox: boolean;\n\n  private declare initialized: true | undefined;\n\n  static ownDefaults = polylineDefaultValues;\n\n  static type = 'Polyline';\n\n  static getDefaults(): Record<string, any> {\n    return {\n      ...super.getDefaults(),\n      ...Polyline.ownDefaults,\n    };\n  }\n  /**\n   * A list of properties that if changed trigger a recalculation of dimensions\n   * @todo check if you really need to recalculate for all cases\n   */\n  static layoutProperties: (keyof Polyline)[] = [\n    'skewX',\n    'skewY',\n    'strokeLineCap',\n    'strokeLineJoin',\n    'strokeMiterLimit',\n    'strokeWidth',\n    'strokeUniform',\n    'points',\n  ];\n\n  declare pathOffset: Point;\n\n  declare strokeOffset: Point;\n\n  static cacheProperties = [...cacheProperties, 'points'];\n\n  strokeDiff: Point;\n\n  /**\n   * Constructor\n   * @param {Array} points Array of points (where each point is an object with x and y)\n   * @param {Object} [options] Options object\n   * @return {Polyline} thisArg\n   * @example\n   * var poly = new Polyline([\n   *     { x: 10, y: 10 },\n   *     { x: 50, y: 30 },\n   *     { x: 40, y: 70 },\n   *     { x: 60, y: 50 },\n   *     { x: 100, y: 150 },\n   *     { x: 40, y: 100 }\n   *   ], {\n   *   stroke: 'red',\n   *   left: 100,\n   *   top: 100\n   * });\n   */\n  constructor(points: XY[] = [], options: Props = {} as Props) {\n    super({ ...options, points });\n    const { left, top } = options;\n    this.initialized = true;\n    this.setBoundingBox(true);\n    typeof left === 'number' && this.set(LEFT, left);\n    typeof top === 'number' && this.set(TOP, top);\n  }\n\n  protected isOpen() {\n    return true;\n  }\n\n  private _projectStrokeOnPoints(options: TProjectStrokeOnPointsOptions) {\n    return projectStrokeOnPoints(this.points, options, this.isOpen());\n  }\n\n  /**\n   * Calculate the polygon bounding box\n   * @private\n   */\n  _calcDimensions(options?: Partial<TProjectStrokeOnPointsOptions>) {\n    options = {\n      scaleX: this.scaleX,\n      scaleY: this.scaleY,\n      skewX: this.skewX,\n      skewY: this.skewY,\n      strokeLineCap: this.strokeLineCap,\n      strokeLineJoin: this.strokeLineJoin,\n      strokeMiterLimit: this.strokeMiterLimit,\n      strokeUniform: this.strokeUniform,\n      strokeWidth: this.strokeWidth,\n      ...(options || {}),\n    };\n    const points = this.exactBoundingBox\n      ? this._projectStrokeOnPoints(\n          options as TProjectStrokeOnPointsOptions\n        ).map((projection) => projection.projectedPoint)\n      : this.points;\n    if (points.length === 0) {\n      return {\n        left: 0,\n        top: 0,\n        width: 0,\n        height: 0,\n        pathOffset: new Point(),\n        strokeOffset: new Point(),\n        strokeDiff: new Point(),\n      };\n    }\n    const bbox = makeBoundingBoxFromPoints(points),\n      // Remove scale effect, since it's applied after\n      matrix = calcDimensionsMatrix({ ...options, scaleX: 1, scaleY: 1 }),\n      bboxNoStroke = makeBoundingBoxFromPoints(\n        this.points.map((p) => transformPoint(p, matrix, true))\n      ),\n      scale = new Point(this.scaleX, this.scaleY);\n    let offsetX = bbox.left + bbox.width / 2,\n      offsetY = bbox.top + bbox.height / 2;\n    if (this.exactBoundingBox) {\n      offsetX = offsetX - offsetY * Math.tan(degreesToRadians(this.skewX));\n      // Order of those assignments is important.\n      // offsetY relies on offsetX being already changed by the line above\n      offsetY = offsetY - offsetX * Math.tan(degreesToRadians(this.skewY));\n    }\n\n    return {\n      ...bbox,\n      pathOffset: new Point(offsetX, offsetY),\n      strokeOffset: new Point(bboxNoStroke.left, bboxNoStroke.top)\n        .subtract(new Point(bbox.left, bbox.top))\n        .multiply(scale),\n      strokeDiff: new Point(bbox.width, bbox.height)\n        .subtract(new Point(bboxNoStroke.width, bboxNoStroke.height))\n        .multiply(scale),\n    };\n  }\n\n  /**\n   * This function is an helper for svg import. it returns the center of the object in the svg\n   * untransformed coordinates, by look at the polyline/polygon points.\n   * @private\n   * @return {Point} center point from element coordinates\n   */\n  _findCenterFromElement(): Point {\n    const bbox = makeBoundingBoxFromPoints(this.points);\n    return new Point(bbox.left + bbox.width / 2, bbox.top + bbox.height / 2);\n  }\n\n  setDimensions() {\n    this.setBoundingBox();\n  }\n\n  setBoundingBox(adjustPosition?: boolean) {\n    const { left, top, width, height, pathOffset, strokeOffset, strokeDiff } =\n      this._calcDimensions();\n    this.set({ width, height, pathOffset, strokeOffset, strokeDiff });\n    adjustPosition &&\n      this.setPositionByOrigin(\n        new Point(left + width / 2, top + height / 2),\n        CENTER,\n        CENTER\n      );\n  }\n\n  /**\n   * @deprecated intermidiate method to be removed, do not use\n   */\n  protected isStrokeAccountedForInDimensions() {\n    return this.exactBoundingBox;\n  }\n\n  /**\n   * @override stroke is taken in account in size\n   */\n  _getNonTransformedDimensions() {\n    return this.exactBoundingBox\n      ? // TODO: fix this\n        new Point(this.width, this.height)\n      : super._getNonTransformedDimensions();\n  }\n\n  /**\n   * @override stroke and skewing are taken into account when projecting stroke on points,\n   * therefore we don't want the default calculation to account for skewing as well.\n   * Though it is possible to pass `width` and `height` in `options`, doing so is very strange, use with discretion.\n   *\n   * @private\n   */\n  _getTransformedDimensions(options: any = {}) {\n    if (this.exactBoundingBox) {\n      let size: Point;\n      /* When `strokeUniform = true`, any changes to the properties require recalculating the `width` and `height` because\n        the stroke projections are affected.\n        When `strokeUniform = false`, we don't need to recalculate for scale transformations, as the effect of scale on\n        projections follows a linear function (e.g. scaleX of 2 just multiply width by 2)*/\n      if (\n        Object.keys(options).some(\n          (key) =>\n            this.strokeUniform ||\n            (this.constructor as typeof Polyline).layoutProperties.includes(\n              key as keyof TProjectStrokeOnPointsOptions\n            )\n        )\n      ) {\n        const { width, height } = this._calcDimensions(options);\n        size = new Point(options.width ?? width, options.height ?? height);\n      } else {\n        size = new Point(\n          options.width ?? this.width,\n          options.height ?? this.height\n        );\n      }\n      return size.multiply(\n        new Point(options.scaleX || this.scaleX, options.scaleY || this.scaleY)\n      );\n    } else {\n      return super._getTransformedDimensions(options);\n    }\n  }\n\n  /**\n   * Recalculates dimensions when changing skew and scale\n   * @private\n   */\n  _set(key: string, value: any) {\n    const changed = this.initialized && this[key as keyof this] !== value;\n    const output = super._set(key, value);\n    if (\n      this.exactBoundingBox &&\n      changed &&\n      (((key === 'scaleX' || key === 'scaleY') &&\n        this.strokeUniform &&\n        (this.constructor as typeof Polyline).layoutProperties.includes(\n          'strokeUniform'\n        )) ||\n        (this.constructor as typeof Polyline).layoutProperties.includes(\n          key as keyof Polyline\n        ))\n    ) {\n      this.setDimensions();\n    }\n    return output;\n  }\n\n  /**\n   * Returns object representation of an instance\n   * @param {Array} [propertiesToInclude] Any properties that you might want to additionally include in the output\n   * @return {Object} Object representation of an instance\n   */\n  toObject<\n    T extends Omit<Props & TClassProperties<this>, keyof SProps>,\n    K extends keyof T = never\n  >(propertiesToInclude: K[] = []): Pick<T, K> & SProps {\n    return {\n      ...super.toObject(propertiesToInclude),\n      points: cloneDeep(this.points),\n    };\n  }\n\n  /**\n   * Returns svg representation of an instance\n   * @return {Array} an array of strings with the specific svg representation\n   * of the instance\n   */\n  _toSVG() {\n    const points = [],\n      diffX = this.pathOffset.x,\n      diffY = this.pathOffset.y,\n      NUM_FRACTION_DIGITS = config.NUM_FRACTION_DIGITS;\n\n    for (let i = 0, len = this.points.length; i < len; i++) {\n      points.push(\n        toFixed(this.points[i].x - diffX, NUM_FRACTION_DIGITS),\n        ',',\n        toFixed(this.points[i].y - diffY, NUM_FRACTION_DIGITS),\n        ' '\n      );\n    }\n    return [\n      `<${\n        (this.constructor as typeof Polyline).type.toLowerCase() as\n          | 'polyline'\n          | 'polygon'\n      } `,\n      'COMMON_PARTS',\n      `points=\"${points.join('')}\" />\\n`,\n    ];\n  }\n\n  /**\n   * @private\n   * @param {CanvasRenderingContext2D} ctx Context to render on\n   */\n  _render(ctx: CanvasRenderingContext2D) {\n    const len = this.points.length,\n      x = this.pathOffset.x,\n      y = this.pathOffset.y;\n\n    if (!len || isNaN(this.points[len - 1].y)) {\n      // do not draw if no points or odd points\n      // NaN comes from parseFloat of a empty string in parser\n      return;\n    }\n    ctx.beginPath();\n    ctx.moveTo(this.points[0].x - x, this.points[0].y - y);\n    for (let i = 0; i < len; i++) {\n      const point = this.points[i];\n      ctx.lineTo(point.x - x, point.y - y);\n    }\n    !this.isOpen() && ctx.closePath();\n    this._renderPaintInOrder(ctx);\n  }\n\n  /**\n   * Returns complexity of an instance\n   * @return {Number} complexity of this instance\n   */\n  complexity(): number {\n    return this.points.length;\n  }\n\n  /* _FROM_SVG_START_ */\n\n  /**\n   * List of attribute names to account for when parsing SVG element (used by {@link Polyline.fromElement})\n   * @static\n   * @memberOf Polyline\n   * @see: http://www.w3.org/TR/SVG/shapes.html#PolylineElement\n   */\n  static ATTRIBUTE_NAMES = [...SHARED_ATTRIBUTES];\n\n  /**\n   * Returns Polyline instance from an SVG element\n   * @static\n   * @memberOf Polyline\n   * @param {HTMLElement} element Element to parser\n   * @param {Object} [options] Options object\n   */\n  static async fromElement(\n    element: HTMLElement,\n    options: Abortable,\n    cssRules?: CSSRules\n  ) {\n    const points = parsePointsAttribute(element.getAttribute('points')),\n      // we omit left and top to instruct the constructor to position the object using the bbox\n      // eslint-disable-next-line @typescript-eslint/no-unused-vars\n      { left, top, ...parsedAttributes } = parseAttributes(\n        element,\n        this.ATTRIBUTE_NAMES,\n        cssRules\n      );\n    return new this(points, {\n      ...parsedAttributes,\n      ...options,\n    });\n  }\n\n  /* _FROM_SVG_END_ */\n\n  /**\n   * Returns Polyline instance from an object representation\n   * @static\n   * @memberOf Polyline\n   * @param {Object} object Object to create an instance from\n   * @returns {Promise<Polyline>}\n   */\n  static fromObject<T extends TOptions<SerializedPolylineProps>>(object: T) {\n    return this._fromObject<Polyline>(object, {\n      extraParam: 'points',\n    });\n  }\n}\n\nclassRegistry.setClass(Polyline);\nclassRegistry.setSVGClass(Polyline);\n","import { classRegistry } from '../ClassRegistry';\nimport { Polyline, polylineDefaultValues } from './Polyline';\n\nexport class Polygon extends Polyline {\n  static ownDefaults = polylineDefaultValues;\n\n  static type = 'Polygon';\n\n  static getDefaults(): Record<string, any> {\n    return {\n      ...super.getDefaults(),\n      ...Polyline.ownDefaults,\n    };\n  }\n\n  protected isOpen() {\n    return false;\n  }\n}\n\nclassRegistry.setClass(Polygon);\nclassRegistry.setSVGClass(Polygon);\n","import { LEFT, reNewline } from '../../constants';\nimport type { TClassProperties } from '../../typedefs';\nimport type { FabricText } from './Text';\n\nconst fontProperties = [\n  'fontSize',\n  'fontWeight',\n  'fontFamily',\n  'fontStyle',\n] as const;\n\nexport const textDecorationProperties = [\n  'underline',\n  'overline',\n  'linethrough',\n] as const;\n\nexport const textLayoutProperties: string[] = [\n  ...fontProperties,\n  'lineHeight',\n  'text',\n  'charSpacing',\n  'textAlign',\n  'styles',\n  'path',\n  'pathStartOffset',\n  'pathSide',\n  'pathAlign',\n];\n\nexport const additionalProps = [\n  ...textLayoutProperties,\n  ...textDecorationProperties,\n  'textBackgroundColor',\n  'direction',\n] as const;\n\nexport type StylePropertiesType =\n  | 'fill'\n  | 'stroke'\n  | 'strokeWidth'\n  | 'fontSize'\n  | 'fontFamily'\n  | 'fontWeight'\n  | 'fontStyle'\n  | 'textBackgroundColor'\n  | 'deltaY'\n  | 'overline'\n  | 'underline'\n  | 'linethrough';\n\nexport const styleProperties: Readonly<StylePropertiesType[]> = [\n  ...fontProperties,\n  ...textDecorationProperties,\n  'stroke',\n  'strokeWidth',\n  'fill',\n  'deltaY',\n  'textBackgroundColor',\n] as const;\n\n// @TODO: Many things here are configuration related and shouldn't be on the class nor prototype\n// regexes, list of properties that are not suppose to change by instances, magic consts.\n// this will be a separated effort\nexport const textDefaultValues: Partial<TClassProperties<FabricText>> = {\n  _reNewline: reNewline,\n  _reSpacesAndTabs: /[ \\t\\r]/g,\n  _reSpaceAndTab: /[ \\t\\r]/,\n  _reWords: /\\S+/g,\n  fontSize: 40,\n  fontWeight: 'normal',\n  fontFamily: 'Times New Roman',\n  underline: false,\n  overline: false,\n  linethrough: false,\n  textAlign: LEFT,\n  fontStyle: 'normal',\n  lineHeight: 1.16,\n  superscript: {\n    size: 0.6, // fontSize factor\n    baseline: -0.35, // baseline-shift factor (upwards)\n  },\n  subscript: {\n    size: 0.6, // fontSize factor\n    baseline: 0.11, // baseline-shift factor (downwards)\n  },\n  textBackgroundColor: '',\n  stroke: null,\n  shadow: null,\n  path: undefined,\n  pathStartOffset: 0,\n  pathSide: LEFT,\n  pathAlign: 'baseline',\n  _fontSizeFraction: 0.222,\n  offsets: {\n    underline: 0.1,\n    linethrough: -0.315,\n    overline: -0.88,\n  },\n  _fontSizeMult: 1.13,\n  charSpacing: 0,\n  deltaY: 0,\n  direction: 'ltr',\n  CACHE_FONT_SIZE: 400,\n  MIN_TEXT_WIDTH: 2,\n};\n\nexport const JUSTIFY = 'justify';\nexport const JUSTIFY_LEFT = 'justify-left';\nexport const JUSTIFY_RIGHT = 'justify-right';\nexport const JUSTIFY_CENTER = 'justify-center';\n","import type { ObjectEvents } from '../../EventTypeDefs';\nimport type { FabricObjectProps, SerializedObjectProps } from '../Object/types';\nimport type { TOptions } from '../../typedefs';\nimport { FabricObject } from '../Object/FabricObject';\nimport { styleProperties } from './constants';\nimport type { StylePropertiesType } from './constants';\nimport type { FabricText } from './Text';\nimport { pick } from '../../util';\nimport { pickBy } from '../../util/misc/pick';\n\nexport type CompleteTextStyleDeclaration = Pick<\n  FabricText,\n  StylePropertiesType\n>;\n\nexport type TextStyleDeclaration = Partial<CompleteTextStyleDeclaration>;\n\nexport type TextStyle = {\n  [line: number | string]: { [char: number | string]: TextStyleDeclaration };\n};\n\nexport abstract class StyledText<\n  Props extends TOptions<FabricObjectProps> = Partial<FabricObjectProps>,\n  SProps extends SerializedObjectProps = SerializedObjectProps,\n  EventSpec extends ObjectEvents = ObjectEvents\n> extends FabricObject<Props, SProps, EventSpec> {\n  declare abstract styles: TextStyle;\n  protected declare abstract _textLines: string[][];\n  protected declare _forceClearCache: boolean;\n  static _styleProperties: Readonly<StylePropertiesType[]> = styleProperties;\n  abstract get2DCursorLocation(\n    selectionStart: number,\n    skipWrapping?: boolean\n  ): { charIndex: number; lineIndex: number };\n\n  /**\n   * Returns true if object has no styling or no styling in a line\n   * @param {Number} lineIndex , lineIndex is on wrapped lines.\n   * @return {Boolean}\n   */\n  isEmptyStyles(lineIndex?: number): boolean {\n    if (!this.styles) {\n      return true;\n    }\n    if (typeof lineIndex !== 'undefined' && !this.styles[lineIndex]) {\n      return true;\n    }\n    const obj =\n      typeof lineIndex === 'undefined'\n        ? this.styles\n        : { line: this.styles[lineIndex] };\n    for (const p1 in obj) {\n      for (const p2 in obj[p1]) {\n        // eslint-disable-next-line no-unused-vars\n        for (const p3 in obj[p1][p2]) {\n          return false;\n        }\n      }\n    }\n    return true;\n  }\n\n  /**\n   * Returns true if object has a style property or has it ina specified line\n   * This function is used to detect if a text will use a particular property or not.\n   * @param {String} property to check for\n   * @param {Number} lineIndex to check the style on\n   * @return {Boolean}\n   */\n  styleHas(property: keyof TextStyleDeclaration, lineIndex?: number): boolean {\n    if (!this.styles) {\n      return false;\n    }\n    if (typeof lineIndex !== 'undefined' && !this.styles[lineIndex]) {\n      return false;\n    }\n    const obj =\n      typeof lineIndex === 'undefined'\n        ? this.styles\n        : { 0: this.styles[lineIndex] };\n    // eslint-disable-next-line\n    for (const p1 in obj) {\n      // eslint-disable-next-line\n      for (const p2 in obj[p1]) {\n        if (typeof obj[p1][p2][property] !== 'undefined') {\n          return true;\n        }\n      }\n    }\n    return false;\n  }\n\n  /**\n   * Check if characters in a text have a value for a property\n   * whose value matches the textbox's value for that property.  If so,\n   * the character-level property is deleted.  If the character\n   * has no other properties, then it is also deleted.  Finally,\n   * if the line containing that character has no other characters\n   * then it also is deleted.\n   *\n   * @param {string} property The property to compare between characters and text.\n   */\n  cleanStyle(property: keyof TextStyleDeclaration) {\n    if (!this.styles) {\n      return false;\n    }\n    const obj = this.styles;\n    let stylesCount = 0,\n      letterCount,\n      stylePropertyValue,\n      allStyleObjectPropertiesMatch = true,\n      graphemeCount = 0;\n    for (const p1 in obj) {\n      letterCount = 0;\n      for (const p2 in obj[p1]) {\n        const styleObject = obj[p1][p2] || {},\n          stylePropertyHasBeenSet = styleObject[property] !== undefined;\n\n        stylesCount++;\n\n        if (stylePropertyHasBeenSet) {\n          if (!stylePropertyValue) {\n            stylePropertyValue = styleObject[property];\n          } else if (styleObject[property] !== stylePropertyValue) {\n            allStyleObjectPropertiesMatch = false;\n          }\n\n          if (styleObject[property] === this[property as keyof this]) {\n            delete styleObject[property];\n          }\n        } else {\n          allStyleObjectPropertiesMatch = false;\n        }\n\n        if (Object.keys(styleObject).length !== 0) {\n          letterCount++;\n        } else {\n          delete obj[p1][p2];\n        }\n      }\n\n      if (letterCount === 0) {\n        delete obj[p1];\n      }\n    }\n    // if every grapheme has the same style set then\n    // delete those styles and set it on the parent\n    for (let i = 0; i < this._textLines.length; i++) {\n      graphemeCount += this._textLines[i].length;\n    }\n    if (allStyleObjectPropertiesMatch && stylesCount === graphemeCount) {\n      // @ts-expect-error conspiracy theory of TS\n      this[property as keyof this] = stylePropertyValue;\n      this.removeStyle(property);\n    }\n  }\n\n  /**\n   * Remove a style property or properties from all individual character styles\n   * in a text object.  Deletes the character style object if it contains no other style\n   * props.  Deletes a line style object if it contains no other character styles.\n   *\n   * @param {String} props The property to remove from character styles.\n   */\n  removeStyle(property: keyof TextStyleDeclaration) {\n    if (!this.styles) {\n      return;\n    }\n    const obj = this.styles;\n    let line, lineNum, charNum;\n    for (lineNum in obj) {\n      line = obj[lineNum];\n      for (charNum in line) {\n        delete line[charNum][property];\n        if (Object.keys(line[charNum]).length === 0) {\n          delete line[charNum];\n        }\n      }\n      if (Object.keys(line).length === 0) {\n        delete obj[lineNum];\n      }\n    }\n  }\n\n  private _extendStyles(index: number, style: TextStyleDeclaration): void {\n    const { lineIndex, charIndex } = this.get2DCursorLocation(index);\n\n    if (!this._getLineStyle(lineIndex)) {\n      this._setLineStyle(lineIndex);\n    }\n\n    const newStyle = pickBy(\n      {\n        // first create a new object that is a merge of existing and new\n        ...this._getStyleDeclaration(lineIndex, charIndex),\n        ...style,\n        // use the predicate to discard undefined values\n      },\n      (value) => value !== undefined\n    );\n\n    // finally assign to the old position the new style\n    this._setStyleDeclaration(lineIndex, charIndex, newStyle);\n  }\n\n  /**\n   * Gets style of a current selection/cursor (at the start position)\n   * @param {Number} startIndex Start index to get styles at\n   * @param {Number} endIndex End index to get styles at, if not specified startIndex + 1\n   * @param {Boolean} [complete] get full style or not\n   * @return {Array} styles an array with one, zero or more Style objects\n   */\n  getSelectionStyles(\n    startIndex: number,\n    endIndex?: number,\n    complete?: boolean\n  ): TextStyleDeclaration[] {\n    const styles: TextStyleDeclaration[] = [];\n    for (let i = startIndex; i < (endIndex || startIndex); i++) {\n      styles.push(this.getStyleAtPosition(i, complete));\n    }\n    return styles;\n  }\n\n  /**\n   * Gets style of a current selection/cursor position\n   * @param {Number} position  to get styles at\n   * @param {Boolean} [complete] full style if true\n   * @return {Object} style Style object at a specified index\n   * @private\n   */\n  getStyleAtPosition(position: number, complete?: boolean) {\n    const { lineIndex, charIndex } = this.get2DCursorLocation(position);\n    return complete\n      ? this.getCompleteStyleDeclaration(lineIndex, charIndex)\n      : this._getStyleDeclaration(lineIndex, charIndex);\n  }\n\n  /**\n   * Sets style of a current selection, if no selection exist, do not set anything.\n   * @param {Object} styles Styles object\n   * @param {Number} startIndex Start index to get styles at\n   * @param {Number} [endIndex] End index to get styles at, if not specified startIndex + 1\n   */\n  setSelectionStyles(styles: object, startIndex: number, endIndex?: number) {\n    for (let i = startIndex; i < (endIndex || startIndex); i++) {\n      this._extendStyles(i, styles);\n    }\n    /* not included in _extendStyles to avoid clearing cache more than once */\n    this._forceClearCache = true;\n  }\n\n  /**\n   * Get a reference, not a clone, to the style object for a given character,\n   * if no style is set for a line or char, return a new empty object.\n   * This is tricky and confusing because when you get an empty object you can't\n   * determine if it is a reference or a new one.\n   * @TODO this should always return a reference or always a clone or undefined when necessary.\n   * @protected\n   * @param {Number} lineIndex\n   * @param {Number} charIndex\n   * @return {TextStyleDeclaration} a style object reference to the existing one or a new empty object when undefined\n   */\n  _getStyleDeclaration(\n    lineIndex: number,\n    charIndex: number\n  ): TextStyleDeclaration {\n    const lineStyle = this.styles && this.styles[lineIndex];\n    return lineStyle ? lineStyle[charIndex] ?? {} : {};\n  }\n\n  /**\n   * return a new object that contains all the style property for a character\n   * the object returned is newly created\n   * @param {Number} lineIndex of the line where the character is\n   * @param {Number} charIndex position of the character on the line\n   * @return {Object} style object\n   */\n  getCompleteStyleDeclaration(\n    lineIndex: number,\n    charIndex: number\n  ): CompleteTextStyleDeclaration {\n    return {\n      // @ts-expect-error readonly\n      ...pick(this, (this.constructor as typeof StyledText)._styleProperties),\n      ...this._getStyleDeclaration(lineIndex, charIndex),\n    } as CompleteTextStyleDeclaration;\n  }\n\n  /**\n   * @param {Number} lineIndex\n   * @param {Number} charIndex\n   * @param {Object} style\n   * @private\n   */\n  protected _setStyleDeclaration(\n    lineIndex: number,\n    charIndex: number,\n    style: object\n  ) {\n    this.styles[lineIndex][charIndex] = style;\n  }\n\n  /**\n   *\n   * @param {Number} lineIndex\n   * @param {Number} charIndex\n   * @private\n   */\n  protected _deleteStyleDeclaration(lineIndex: number, charIndex: number) {\n    delete this.styles[lineIndex][charIndex];\n  }\n\n  /**\n   * @param {Number} lineIndex\n   * @return {Boolean} if the line exists or not\n   * @private\n   */\n  protected _getLineStyle(lineIndex: number): boolean {\n    return !!this.styles[lineIndex];\n  }\n\n  /**\n   * Set the line style to an empty object so that is initialized\n   * @param {Number} lineIndex\n   * @private\n   */\n  protected _setLineStyle(lineIndex: number) {\n    this.styles[lineIndex] = {};\n  }\n\n  protected _deleteLineStyle(lineIndex: number) {\n    delete this.styles[lineIndex];\n  }\n}\n","import { config } from '../../config';\nimport type { TSVGReviver } from '../../typedefs';\nimport { escapeXml } from '../../util/lang_string';\nimport { colorPropToSVG, createSVGRect } from '../../util/misc/svgParsing';\nimport { hasStyleChanged } from '../../util/misc/textStyles';\nimport { toFixed } from '../../util/misc/toFixed';\nimport { FabricObjectSVGExportMixin } from '../Object/FabricObjectSVGExportMixin';\nimport { type TextStyleDeclaration } from './StyledText';\nimport { JUSTIFY } from '../Text/constants';\nimport type { FabricText } from './Text';\n\nconst multipleSpacesRegex = /  +/g;\nconst dblQuoteRegex = /\"/g;\n\nfunction createSVGInlineRect(\n  color: string,\n  left: number,\n  top: number,\n  width: number,\n  height: number\n) {\n  return `\\t\\t${createSVGRect(color, { left, top, width, height })}\\n`;\n}\n\nexport class TextSVGExportMixin extends FabricObjectSVGExportMixin {\n  _toSVG(this: TextSVGExportMixin & FabricText): string[] {\n    const offsets = this._getSVGLeftTopOffsets(),\n      textAndBg = this._getSVGTextAndBg(offsets.textTop, offsets.textLeft);\n    return this._wrapSVGTextAndBg(textAndBg);\n  }\n\n  toSVG(this: TextSVGExportMixin & FabricText, reviver: TSVGReviver): string {\n    return this._createBaseSVGMarkup(this._toSVG(), {\n      reviver,\n      noStyle: true,\n      withShadow: true,\n    });\n  }\n\n  private _getSVGLeftTopOffsets(this: TextSVGExportMixin & FabricText) {\n    return {\n      textLeft: -this.width / 2,\n      textTop: -this.height / 2,\n      lineTop: this.getHeightOfLine(0),\n    };\n  }\n\n  private _wrapSVGTextAndBg(\n    this: TextSVGExportMixin & FabricText,\n    {\n      textBgRects,\n      textSpans,\n    }: {\n      textSpans: string[];\n      textBgRects: string[];\n    }\n  ) {\n    const noShadow = true,\n      textDecoration = this.getSvgTextDecoration(this);\n    return [\n      textBgRects.join(''),\n      '\\t\\t<text xml:space=\"preserve\" ',\n      this.fontFamily\n        ? `font-family=\"${this.fontFamily.replace(dblQuoteRegex, \"'\")}\" `\n        : '',\n      this.fontSize ? `font-size=\"${this.fontSize}\" ` : '',\n      this.fontStyle ? `font-style=\"${this.fontStyle}\" ` : '',\n      this.fontWeight ? `font-weight=\"${this.fontWeight}\" ` : '',\n      textDecoration ? `text-decoration=\"${textDecoration}\" ` : '',\n      this.direction === 'rtl' ? `direction=\"${this.direction}\" ` : '',\n      'style=\"',\n      this.getSvgStyles(noShadow),\n      '\"',\n      this.addPaintOrder(),\n      ' >',\n      textSpans.join(''),\n      '</text>\\n',\n    ];\n  }\n\n  /**\n   * @private\n   * @param {Number} textTopOffset Text top offset\n   * @param {Number} textLeftOffset Text left offset\n   * @return {Object}\n   */\n  private _getSVGTextAndBg(\n    this: TextSVGExportMixin & FabricText,\n    textTopOffset: number,\n    textLeftOffset: number\n  ) {\n    const textSpans: string[] = [],\n      textBgRects: string[] = [];\n    let height = textTopOffset,\n      lineOffset;\n\n    // bounding-box background\n    this.backgroundColor &&\n      textBgRects.push(\n        ...createSVGInlineRect(\n          this.backgroundColor,\n          -this.width / 2,\n          -this.height / 2,\n          this.width,\n          this.height\n        )\n      );\n\n    // text and text-background\n    for (let i = 0, len = this._textLines.length; i < len; i++) {\n      lineOffset = this._getLineLeftOffset(i);\n      if (this.direction === 'rtl') {\n        lineOffset += this.width;\n      }\n      if (this.textBackgroundColor || this.styleHas('textBackgroundColor', i)) {\n        this._setSVGTextLineBg(\n          textBgRects,\n          i,\n          textLeftOffset + lineOffset,\n          height\n        );\n      }\n      this._setSVGTextLineText(\n        textSpans,\n        i,\n        textLeftOffset + lineOffset,\n        height\n      );\n      height += this.getHeightOfLine(i);\n    }\n\n    return {\n      textSpans,\n      textBgRects,\n    };\n  }\n\n  private _createTextCharSpan(\n    this: TextSVGExportMixin & FabricText,\n    char: string,\n    styleDecl: TextStyleDeclaration,\n    left: number,\n    top: number\n  ) {\n    const styleProps = this.getSvgSpanStyles(\n        styleDecl,\n        char !== char.trim() || !!char.match(multipleSpacesRegex)\n      ),\n      fillStyles = styleProps ? `style=\"${styleProps}\"` : '',\n      dy = styleDecl.deltaY,\n      dySpan = dy ? ` dy=\"${toFixed(dy, config.NUM_FRACTION_DIGITS)}\" ` : '';\n\n    return `<tspan x=\"${toFixed(\n      left,\n      config.NUM_FRACTION_DIGITS\n    )}\" y=\"${toFixed(\n      top,\n      config.NUM_FRACTION_DIGITS\n    )}\" ${dySpan}${fillStyles}>${escapeXml(char)}</tspan>`;\n  }\n\n  private _setSVGTextLineText(\n    this: TextSVGExportMixin & FabricText,\n    textSpans: string[],\n    lineIndex: number,\n    textLeftOffset: number,\n    textTopOffset: number\n  ) {\n    const lineHeight = this.getHeightOfLine(lineIndex),\n      isJustify = this.textAlign.includes(JUSTIFY),\n      line = this._textLines[lineIndex];\n    let actualStyle,\n      nextStyle,\n      charsToRender = '',\n      charBox,\n      style,\n      boxWidth = 0,\n      timeToRender;\n\n    textTopOffset +=\n      (lineHeight * (1 - this._fontSizeFraction)) / this.lineHeight;\n    for (let i = 0, len = line.length - 1; i <= len; i++) {\n      timeToRender = i === len || this.charSpacing;\n      charsToRender += line[i];\n      charBox = this.__charBounds[lineIndex][i];\n      if (boxWidth === 0) {\n        textLeftOffset += charBox.kernedWidth - charBox.width;\n        boxWidth += charBox.width;\n      } else {\n        boxWidth += charBox.kernedWidth;\n      }\n      if (isJustify && !timeToRender) {\n        if (this._reSpaceAndTab.test(line[i])) {\n          timeToRender = true;\n        }\n      }\n      if (!timeToRender) {\n        // if we have charSpacing, we render char by char\n        actualStyle =\n          actualStyle || this.getCompleteStyleDeclaration(lineIndex, i);\n        nextStyle = this.getCompleteStyleDeclaration(lineIndex, i + 1);\n        timeToRender = hasStyleChanged(actualStyle, nextStyle, true);\n      }\n      if (timeToRender) {\n        style = this._getStyleDeclaration(lineIndex, i);\n        textSpans.push(\n          this._createTextCharSpan(\n            charsToRender,\n            style,\n            textLeftOffset,\n            textTopOffset\n          )\n        );\n        charsToRender = '';\n        actualStyle = nextStyle;\n        if (this.direction === 'rtl') {\n          textLeftOffset -= boxWidth;\n        } else {\n          textLeftOffset += boxWidth;\n        }\n        boxWidth = 0;\n      }\n    }\n  }\n\n  private _setSVGTextLineBg(\n    this: TextSVGExportMixin & FabricText,\n    textBgRects: (string | number)[],\n    i: number,\n    leftOffset: number,\n    textTopOffset: number\n  ) {\n    const line = this._textLines[i],\n      heightOfLine = this.getHeightOfLine(i) / this.lineHeight;\n    let boxWidth = 0,\n      boxStart = 0,\n      currentColor,\n      lastColor = this.getValueOfPropertyAt(i, 0, 'textBackgroundColor');\n    for (let j = 0; j < line.length; j++) {\n      const { left, width, kernedWidth } = this.__charBounds[i][j];\n      currentColor = this.getValueOfPropertyAt(i, j, 'textBackgroundColor');\n      if (currentColor !== lastColor) {\n        lastColor &&\n          textBgRects.push(\n            ...createSVGInlineRect(\n              lastColor,\n              leftOffset + boxStart,\n              textTopOffset,\n              boxWidth,\n              heightOfLine\n            )\n          );\n        boxStart = left;\n        boxWidth = width;\n        lastColor = currentColor;\n      } else {\n        boxWidth += kernedWidth;\n      }\n    }\n    currentColor &&\n      textBgRects.push(\n        ...createSVGInlineRect(\n          lastColor,\n          leftOffset + boxStart,\n          textTopOffset,\n          boxWidth,\n          heightOfLine\n        )\n      );\n  }\n\n  /**\n   * @deprecated unused\n   */\n  _getSVGLineTopOffset(\n    this: TextSVGExportMixin & FabricText,\n    lineIndex: number\n  ) {\n    let lineTopOffset = 0,\n      j;\n    for (j = 0; j < lineIndex; j++) {\n      lineTopOffset += this.getHeightOfLine(j);\n    }\n    const lastHeight = this.getHeightOfLine(j);\n    return {\n      lineTop: lineTopOffset,\n      offset:\n        ((this._fontSizeMult - this._fontSizeFraction) * lastHeight) /\n        (this.lineHeight * this._fontSizeMult),\n    };\n  }\n\n  /**\n   * Returns styles-string for svg-export\n   * @param {Boolean} skipShadow a boolean to skip shadow filter output\n   * @return {String}\n   */\n  getSvgStyles(this: TextSVGExportMixin & FabricText, skipShadow?: boolean) {\n    // cant use ts-expect-error because of ts 5.3 cross check\n    // @ts-ignore TS doesn't respect this type casting\n    return `${super.getSvgStyles(skipShadow)} white-space: pre;`;\n  }\n\n  /**\n   * Returns styles-string for svg-export\n   * @param {Object} style the object from which to retrieve style properties\n   * @param {Boolean} useWhiteSpace a boolean to include an additional attribute in the style.\n   * @return {String}\n   */\n  getSvgSpanStyles(\n    this: TextSVGExportMixin & FabricText,\n    style: TextStyleDeclaration,\n    useWhiteSpace?: boolean\n  ) {\n    const {\n      fontFamily,\n      strokeWidth,\n      stroke,\n      fill,\n      fontSize,\n      fontStyle,\n      fontWeight,\n      deltaY,\n    } = style;\n\n    const textDecoration = this.getSvgTextDecoration(style);\n\n    return [\n      stroke ? colorPropToSVG('stroke', stroke) : '',\n      strokeWidth ? `stroke-width: ${strokeWidth}; ` : '',\n      fontFamily\n        ? `font-family: ${\n            !fontFamily.includes(\"'\") && !fontFamily.includes('\"')\n              ? `'${fontFamily}'`\n              : fontFamily\n          }; `\n        : '',\n      fontSize ? `font-size: ${fontSize}px; ` : '',\n      fontStyle ? `font-style: ${fontStyle}; ` : '',\n      fontWeight ? `font-weight: ${fontWeight}; ` : '',\n      textDecoration ? `text-decoration: ${textDecoration}; ` : textDecoration,\n      fill ? colorPropToSVG('fill', fill) : '',\n      deltaY ? `baseline-shift: ${-deltaY}; ` : '',\n      useWhiteSpace ? 'white-space: pre; ' : '',\n    ].join('');\n  }\n\n  /**\n   * Returns text-decoration property for svg-export\n   * @param {Object} style the object from which to retrieve style properties\n   * @return {String}\n   */\n  getSvgTextDecoration(\n    this: TextSVGExportMixin & FabricText,\n    style: TextStyleDeclaration\n  ) {\n    return (['overline', 'underline', 'line-through'] as const)\n      .filter(\n        (decoration) =>\n          style[\n            decoration.replace('-', '') as\n              | 'overline'\n              | 'underline'\n              | 'linethrough'\n          ]\n      )\n      .join(' ');\n  }\n}\n","import { cache } from '../../cache';\nimport { DEFAULT_SVG_FONT_SIZE } from '../../constants';\nimport type { ObjectEvents } from '../../EventTypeDefs';\nimport type {\n  CompleteTextStyleDeclaration,\n  TextStyle,\n  TextStyleDeclaration,\n} from './StyledText';\nimport { StyledText } from './StyledText';\nimport { SHARED_ATTRIBUTES } from '../../parser/attributes';\nimport { parseAttributes } from '../../parser/parseAttributes';\nimport type {\n  Abortable,\n  TCacheCanvasDimensions,\n  TClassProperties,\n  TFiller,\n  TOptions,\n} from '../../typedefs';\nimport { classRegistry } from '../../ClassRegistry';\nimport { graphemeSplit } from '../../util/lang_string';\nimport { createCanvasElement } from '../../util/misc/dom';\nimport type { TextStyleArray } from '../../util/misc/textStyles';\nimport {\n  hasStyleChanged,\n  stylesFromArray,\n  stylesToArray,\n} from '../../util/misc/textStyles';\nimport { getPathSegmentsInfo, getPointOnPath } from '../../util/path';\nimport { cacheProperties } from '../Object/FabricObject';\nimport type { Path } from '../Path';\nimport { TextSVGExportMixin } from './TextSVGExportMixin';\nimport { applyMixins } from '../../util/applyMixins';\nimport type { FabricObjectProps, SerializedObjectProps } from '../Object/types';\nimport type { StylePropertiesType } from './constants';\nimport {\n  additionalProps,\n  textDefaultValues,\n  textLayoutProperties,\n  JUSTIFY,\n  JUSTIFY_CENTER,\n  JUSTIFY_LEFT,\n  JUSTIFY_RIGHT,\n} from './constants';\nimport { CENTER, LEFT, RIGHT, TOP, BOTTOM } from '../../constants';\nimport { isFiller } from '../../util/typeAssertions';\nimport type { Gradient } from '../../gradient/Gradient';\nimport type { Pattern } from '../../Pattern';\nimport type { CSSRules } from '../../parser/typedefs';\n\nlet measuringContext: CanvasRenderingContext2D | null;\n\n/**\n * Return a context for measurement of text string.\n * if created it gets stored for reuse\n */\nfunction getMeasuringContext() {\n  if (!measuringContext) {\n    const canvas = createCanvasElement();\n    canvas.width = canvas.height = 0;\n    measuringContext = canvas.getContext('2d');\n  }\n  return measuringContext;\n}\n\nexport type TPathSide = 'left' | 'right';\n\nexport type TPathAlign = 'baseline' | 'center' | 'ascender' | 'descender';\n\nexport type TextLinesInfo = {\n  lines: string[];\n  graphemeLines: string[][];\n  graphemeText: string[];\n  _unwrappedLines: string[][];\n};\n\n/**\n * Measure and return the info of a single grapheme.\n * needs the the info of previous graphemes already filled\n * Override to customize measuring\n */\nexport type GraphemeBBox = {\n  width: number;\n  height: number;\n  kernedWidth: number;\n  left: number;\n  deltaY: number;\n  renderLeft?: number;\n  renderTop?: number;\n  angle?: number;\n};\n\n// @TODO this is not complete\ninterface UniqueTextProps {\n  charSpacing: number;\n  lineHeight: number;\n  fontSize: number;\n  fontWeight: string;\n  fontFamily: string;\n  fontStyle: string;\n  pathSide: TPathSide;\n  pathAlign: TPathAlign;\n  underline: boolean;\n  overline: boolean;\n  linethrough: boolean;\n  textAlign: string;\n  direction: CanvasDirection;\n  path?: Path;\n}\n\nexport interface SerializedTextProps\n  extends SerializedObjectProps,\n    UniqueTextProps {\n  styles: TextStyleArray | TextStyle;\n}\n\nexport interface TextProps extends FabricObjectProps, UniqueTextProps {\n  styles: TextStyle;\n}\n\n/**\n * Text class\n * @tutorial {@link http://fabricjs.com/fabric-intro-part-2#text}\n */\nexport class FabricText<\n    Props extends TOptions<TextProps> = Partial<TextProps>,\n    SProps extends SerializedTextProps = SerializedTextProps,\n    EventSpec extends ObjectEvents = ObjectEvents\n  >\n  extends StyledText<Props, SProps, EventSpec>\n  implements UniqueTextProps\n{\n  /**\n   * Properties that requires a text layout recalculation when changed\n   * @type string[]\n   * @protected\n   */\n  static textLayoutProperties: string[] = textLayoutProperties;\n\n  /**\n   * @private\n   */\n  declare _reNewline: RegExp;\n\n  /**\n   * Use this regular expression to filter for whitespaces that is not a new line.\n   * Mostly used when text is 'justify' aligned.\n   * @private\n   */\n  declare _reSpacesAndTabs: RegExp;\n\n  /**\n   * Use this regular expression to filter for whitespace that is not a new line.\n   * Mostly used when text is 'justify' aligned.\n   * @private\n   */\n  declare _reSpaceAndTab: RegExp;\n\n  /**\n   * Use this regular expression to filter consecutive groups of non spaces.\n   * Mostly used when text is 'justify' aligned.\n   * @private\n   */\n  declare _reWords: RegExp;\n\n  declare text: string;\n\n  /**\n   * Font size (in pixels)\n   * @type Number\n   * @default\n   */\n  declare fontSize: number;\n\n  /**\n   * Font weight (e.g. bold, normal, 400, 600, 800)\n   * @type {(Number|String)}\n   * @default\n   */\n  declare fontWeight: string;\n\n  /**\n   * Font family\n   * @type String\n   * @default\n   */\n  declare fontFamily: string;\n\n  /**\n   * Text decoration underline.\n   * @type Boolean\n   * @default\n   */\n  declare underline: boolean;\n\n  /**\n   * Text decoration overline.\n   * @type Boolean\n   * @default\n   */\n  declare overline: boolean;\n\n  /**\n   * Text decoration linethrough.\n   * @type Boolean\n   * @default\n   */\n  declare linethrough: boolean;\n\n  /**\n   * Text alignment. Possible values: \"left\", \"center\", \"right\", \"justify\",\n   * \"justify-left\", \"justify-center\" or \"justify-right\".\n   * @type String\n   * @default\n   */\n  declare textAlign: string;\n\n  /**\n   * Font style . Possible values: \"\", \"normal\", \"italic\" or \"oblique\".\n   * @type String\n   * @default\n   */\n  declare fontStyle: string;\n\n  /**\n   * Line height\n   * @type Number\n   * @default\n   */\n  declare lineHeight: number;\n\n  /**\n   * Superscript schema object (minimum overlap)\n   */\n  declare superscript: {\n    /**\n     * fontSize factor\n     * @default 0.6\n     */\n    size: number;\n    /**\n     * baseline-shift factor (upwards)\n     * @default -0.35\n     */\n    baseline: number;\n  };\n\n  /**\n   * Subscript schema object (minimum overlap)\n   */\n  declare subscript: {\n    /**\n     * fontSize factor\n     * @default 0.6\n     */\n    size: number;\n    /**\n     * baseline-shift factor (downwards)\n     * @default 0.11\n     */\n    baseline: number;\n  };\n\n  /**\n   * Background color of text lines\n   * @type String\n   * @default\n   */\n  declare textBackgroundColor: string;\n\n  declare styles: TextStyle;\n\n  /**\n   * Path that the text should follow.\n   * since 4.6.0 the path will be drawn automatically.\n   * if you want to make the path visible, give it a stroke and strokeWidth or fill value\n   * if you want it to be hidden, assign visible = false to the path.\n   * This feature is in BETA, and SVG import/export is not yet supported.\n   * @type Path\n   * @example\n   * const textPath = new Text('Text on a path', {\n   *     top: 150,\n   *     left: 150,\n   *     textAlign: 'center',\n   *     charSpacing: -50,\n   *     path: new Path('M 0 0 C 50 -100 150 -100 200 0', {\n   *         strokeWidth: 1,\n   *         visible: false\n   *     }),\n   *     pathSide: 'left',\n   *     pathStartOffset: 0\n   * });\n   * @default\n   */\n  declare path?: Path;\n\n  /**\n   * Offset amount for text path starting position\n   * Only used when text has a path\n   * @type Number\n   * @default\n   */\n  declare pathStartOffset: number;\n\n  /**\n   * Which side of the path the text should be drawn on.\n   * Only used when text has a path\n   * @type {TPathSide} 'left|right'\n   * @default\n   */\n  declare pathSide: TPathSide;\n\n  /**\n   * How text is aligned to the path. This property determines\n   * the perpendicular position of each character relative to the path.\n   * (one of \"baseline\", \"center\", \"ascender\", \"descender\")\n   * This feature is in BETA, and its behavior may change\n   * @type TPathAlign\n   * @default\n   */\n  declare pathAlign: TPathAlign;\n\n  /**\n   * @private\n   */\n  declare _fontSizeFraction: number;\n\n  /**\n   * @private\n   */\n  declare offsets: { underline: number; linethrough: number; overline: number };\n\n  /**\n   * Text Line proportion to font Size (in pixels)\n   * @type Number\n   * @default\n   */\n  declare _fontSizeMult: number;\n\n  /**\n   * additional space between characters\n   * expressed in thousands of em unit\n   * @type Number\n   * @default\n   */\n  declare charSpacing: number;\n\n  /**\n   * Baseline shift, styles only, keep at 0 for the main text object\n   * @type {Number}\n   * @default\n   */\n  declare deltaY: number;\n\n  /**\n   * WARNING: EXPERIMENTAL. NOT SUPPORTED YET\n   * determine the direction of the text.\n   * This has to be set manually together with textAlign and originX for proper\n   * experience.\n   * some interesting link for the future\n   * https://www.w3.org/International/questions/qa-bidi-unicode-controls\n   * @since 4.5.0\n   * @type {CanvasDirection} 'ltr|rtl'\n   * @default\n   */\n  declare direction: CanvasDirection;\n\n  /**\n   * contains characters bounding boxes\n   * This variable is considered to be protected.\n   * But for how mixins are implemented right now, we can't leave it private\n   * @protected\n   */\n  __charBounds: GraphemeBBox[][] = [];\n\n  /**\n   * use this size when measuring text. To avoid IE11 rounding errors\n   * @type {Number}\n   * @default\n   * @readonly\n   * @private\n   */\n  declare CACHE_FONT_SIZE: number;\n\n  /**\n   * contains the min text width to avoid getting 0\n   * @type {Number}\n   * @default\n   */\n  declare MIN_TEXT_WIDTH: number;\n\n  /**\n   * contains the the text of the object, divided in lines as they are displayed\n   * on screen. Wrapping will divide the text independently of line breaks\n   * @type {string[]}\n   * @default\n   */\n  declare textLines: string[];\n\n  /**\n   * same as textlines, but each line is an array of graphemes as split by splitByGrapheme\n   * @type {string[]}\n   * @default\n   */\n  declare _textLines: string[][];\n\n  declare _unwrappedTextLines: string[][];\n  declare _text: string[];\n  declare cursorWidth: number;\n  declare __lineHeights: number[];\n  declare __lineWidths: number[];\n  declare initialized?: true;\n\n  static cacheProperties = [...cacheProperties, ...additionalProps];\n\n  static ownDefaults = textDefaultValues;\n\n  static type = 'Text';\n\n  static getDefaults(): Record<string, any> {\n    return { ...super.getDefaults(), ...FabricText.ownDefaults };\n  }\n\n  constructor(text: string, options: Props = {} as Props) {\n    super({ ...options, text, styles: options?.styles || {} });\n    this.initialized = true;\n    if (this.path) {\n      this.setPathInfo();\n    }\n    this.initDimensions();\n    this.setCoords();\n  }\n\n  /**\n   * If text has a path, it will add the extra information needed\n   * for path and text calculations\n   */\n  setPathInfo() {\n    const path = this.path;\n    if (path) {\n      path.segmentsInfo = getPathSegmentsInfo(path.path);\n    }\n  }\n\n  /**\n   * @private\n   * Divides text into lines of text and lines of graphemes.\n   */\n  _splitText(): TextLinesInfo {\n    const newLines = this._splitTextIntoLines(this.text);\n    this.textLines = newLines.lines;\n    this._textLines = newLines.graphemeLines;\n    this._unwrappedTextLines = newLines._unwrappedLines;\n    this._text = newLines.graphemeText;\n    return newLines;\n  }\n\n  /**\n   * Initialize or update text dimensions.\n   * Updates this.width and this.height with the proper values.\n   * Does not return dimensions.\n   */\n  initDimensions() {\n    this._splitText();\n    this._clearCache();\n    this.dirty = true;\n    if (this.path) {\n      this.width = this.path.width;\n      this.height = this.path.height;\n    } else {\n      this.width =\n        this.calcTextWidth() || this.cursorWidth || this.MIN_TEXT_WIDTH;\n      this.height = this.calcTextHeight();\n    }\n    if (this.textAlign.includes(JUSTIFY)) {\n      // once text is measured we need to make space fatter to make justified text.\n      this.enlargeSpaces();\n    }\n  }\n\n  /**\n   * Enlarge space boxes and shift the others\n   */\n  enlargeSpaces() {\n    let diffSpace,\n      currentLineWidth,\n      numberOfSpaces,\n      accumulatedSpace,\n      line,\n      charBound,\n      spaces;\n    for (let i = 0, len = this._textLines.length; i < len; i++) {\n      if (\n        this.textAlign !== JUSTIFY &&\n        (i === len - 1 || this.isEndOfWrapping(i))\n      ) {\n        continue;\n      }\n      accumulatedSpace = 0;\n      line = this._textLines[i];\n      currentLineWidth = this.getLineWidth(i);\n      if (\n        currentLineWidth < this.width &&\n        (spaces = this.textLines[i].match(this._reSpacesAndTabs))\n      ) {\n        numberOfSpaces = spaces.length;\n        diffSpace = (this.width - currentLineWidth) / numberOfSpaces;\n        for (let j = 0; j <= line.length; j++) {\n          charBound = this.__charBounds[i][j];\n          if (this._reSpaceAndTab.test(line[j])) {\n            charBound.width += diffSpace;\n            charBound.kernedWidth += diffSpace;\n            charBound.left += accumulatedSpace;\n            accumulatedSpace += diffSpace;\n          } else {\n            charBound.left += accumulatedSpace;\n          }\n        }\n      }\n    }\n  }\n\n  /**\n   * Detect if the text line is ended with an hard break\n   * text and itext do not have wrapping, return false\n   * @return {Boolean}\n   */\n  isEndOfWrapping(lineIndex: number): boolean {\n    return lineIndex === this._textLines.length - 1;\n  }\n\n  /**\n   * Detect if a line has a linebreak and so we need to account for it when moving\n   * and counting style.\n   * It return always 1 for text and Itext. Textbox has its own implementation\n   * @return Number\n   */\n  missingNewlineOffset(lineIndex: number, skipWrapping?: boolean): 0 | 1;\n  missingNewlineOffset(lineIndex: number): 1 {\n    return 1;\n  }\n\n  /**\n   * Returns 2d representation (lineIndex and charIndex) of cursor\n   * @param {Number} selectionStart\n   * @param {Boolean} [skipWrapping] consider the location for unwrapped lines. useful to manage styles.\n   */\n  get2DCursorLocation(selectionStart: number, skipWrapping?: boolean) {\n    const lines = skipWrapping ? this._unwrappedTextLines : this._textLines;\n    let i: number;\n    for (i = 0; i < lines.length; i++) {\n      if (selectionStart <= lines[i].length) {\n        return {\n          lineIndex: i,\n          charIndex: selectionStart,\n        };\n      }\n      selectionStart -=\n        lines[i].length + this.missingNewlineOffset(i, skipWrapping);\n    }\n    return {\n      lineIndex: i - 1,\n      charIndex:\n        lines[i - 1].length < selectionStart\n          ? lines[i - 1].length\n          : selectionStart,\n    };\n  }\n\n  /**\n   * Returns string representation of an instance\n   * @return {String} String representation of text object\n   */\n  toString(): string {\n    return `#<Text (${this.complexity()}): { \"text\": \"${\n      this.text\n    }\", \"fontFamily\": \"${this.fontFamily}\" }>`;\n  }\n\n  /**\n   * Return the dimension and the zoom level needed to create a cache canvas\n   * big enough to host the object to be cached.\n   * @private\n   * @param {Object} dim.x width of object to be cached\n   * @param {Object} dim.y height of object to be cached\n   * @return {Object}.width width of canvas\n   * @return {Object}.height height of canvas\n   * @return {Object}.zoomX zoomX zoom value to unscale the canvas before drawing cache\n   * @return {Object}.zoomY zoomY zoom value to unscale the canvas before drawing cache\n   */\n  _getCacheCanvasDimensions(): TCacheCanvasDimensions {\n    const dims = super._getCacheCanvasDimensions();\n    const fontSize = this.fontSize;\n    dims.width += fontSize * dims.zoomX;\n    dims.height += fontSize * dims.zoomY;\n    return dims;\n  }\n\n  /**\n   * @private\n   * @param {CanvasRenderingContext2D} ctx Context to render on\n   */\n  _render(ctx: CanvasRenderingContext2D) {\n    const path = this.path;\n    path && !path.isNotVisible() && path._render(ctx);\n    this._setTextStyles(ctx);\n    this._renderTextLinesBackground(ctx);\n    this._renderTextDecoration(ctx, 'underline');\n    this._renderText(ctx);\n    this._renderTextDecoration(ctx, 'overline');\n    this._renderTextDecoration(ctx, 'linethrough');\n  }\n\n  /**\n   * @private\n   * @param {CanvasRenderingContext2D} ctx Context to render on\n   */\n  _renderText(ctx: CanvasRenderingContext2D) {\n    if (this.paintFirst === 'stroke') {\n      this._renderTextStroke(ctx);\n      this._renderTextFill(ctx);\n    } else {\n      this._renderTextFill(ctx);\n      this._renderTextStroke(ctx);\n    }\n  }\n\n  /**\n   * Set the font parameter of the context with the object properties or with charStyle\n   * @private\n   * @param {CanvasRenderingContext2D} ctx Context to render on\n   * @param {Object} [charStyle] object with font style properties\n   * @param {String} [charStyle.fontFamily] Font Family\n   * @param {Number} [charStyle.fontSize] Font size in pixels. ( without px suffix )\n   * @param {String} [charStyle.fontWeight] Font weight\n   * @param {String} [charStyle.fontStyle] Font style (italic|normal)\n   */\n  _setTextStyles(\n    ctx: CanvasRenderingContext2D,\n    charStyle?: any,\n    forMeasuring?: boolean\n  ) {\n    ctx.textBaseline = 'alphabetic';\n    if (this.path) {\n      switch (this.pathAlign) {\n        case CENTER:\n          ctx.textBaseline = 'middle';\n          break;\n        case 'ascender':\n          ctx.textBaseline = TOP;\n          break;\n        case 'descender':\n          ctx.textBaseline = BOTTOM;\n          break;\n      }\n    }\n    ctx.font = this._getFontDeclaration(charStyle, forMeasuring);\n  }\n\n  /**\n   * calculate and return the text Width measuring each line.\n   * @private\n   * @param {CanvasRenderingContext2D} ctx Context to render on\n   * @return {Number} Maximum width of Text object\n   */\n  calcTextWidth(): number {\n    let maxWidth = this.getLineWidth(0);\n\n    for (let i = 1, len = this._textLines.length; i < len; i++) {\n      const currentLineWidth = this.getLineWidth(i);\n      if (currentLineWidth > maxWidth) {\n        maxWidth = currentLineWidth;\n      }\n    }\n    return maxWidth;\n  }\n\n  /**\n   * @private\n   * @param {String} method Method name (\"fillText\" or \"strokeText\")\n   * @param {CanvasRenderingContext2D} ctx Context to render on\n   * @param {String} line Text to render\n   * @param {Number} left Left position of text\n   * @param {Number} top Top position of text\n   * @param {Number} lineIndex Index of a line in a text\n   */\n  _renderTextLine(\n    method: 'fillText' | 'strokeText',\n    ctx: CanvasRenderingContext2D,\n    line: string[],\n    left: number,\n    top: number,\n    lineIndex: number\n  ) {\n    this._renderChars(method, ctx, line, left, top, lineIndex);\n  }\n\n  /**\n   * Renders the text background for lines, taking care of style\n   * @private\n   * @param {CanvasRenderingContext2D} ctx Context to render on\n   */\n  _renderTextLinesBackground(ctx: CanvasRenderingContext2D) {\n    if (!this.textBackgroundColor && !this.styleHas('textBackgroundColor')) {\n      return;\n    }\n    const originalFill = ctx.fillStyle,\n      leftOffset = this._getLeftOffset();\n    let lineTopOffset = this._getTopOffset();\n\n    for (let i = 0, len = this._textLines.length; i < len; i++) {\n      const heightOfLine = this.getHeightOfLine(i);\n      if (\n        !this.textBackgroundColor &&\n        !this.styleHas('textBackgroundColor', i)\n      ) {\n        lineTopOffset += heightOfLine;\n        continue;\n      }\n      const jlen = this._textLines[i].length;\n      const lineLeftOffset = this._getLineLeftOffset(i);\n      let boxWidth = 0;\n      let boxStart = 0;\n      let drawStart;\n      let currentColor;\n      let lastColor = this.getValueOfPropertyAt(i, 0, 'textBackgroundColor');\n      for (let j = 0; j < jlen; j++) {\n        // at this point charbox are either standard or full with pathInfo if there is a path.\n        const charBox = this.__charBounds[i][j] as Required<GraphemeBBox>;\n        currentColor = this.getValueOfPropertyAt(i, j, 'textBackgroundColor');\n        if (this.path) {\n          ctx.save();\n          ctx.translate(charBox.renderLeft, charBox.renderTop);\n          ctx.rotate(charBox.angle);\n          ctx.fillStyle = currentColor;\n          currentColor &&\n            ctx.fillRect(\n              -charBox.width / 2,\n              (-heightOfLine / this.lineHeight) * (1 - this._fontSizeFraction),\n              charBox.width,\n              heightOfLine / this.lineHeight\n            );\n          ctx.restore();\n        } else if (currentColor !== lastColor) {\n          drawStart = leftOffset + lineLeftOffset + boxStart;\n          if (this.direction === 'rtl') {\n            drawStart = this.width - drawStart - boxWidth;\n          }\n          ctx.fillStyle = lastColor;\n          lastColor &&\n            ctx.fillRect(\n              drawStart,\n              lineTopOffset,\n              boxWidth,\n              heightOfLine / this.lineHeight\n            );\n          boxStart = charBox.left;\n          boxWidth = charBox.width;\n          lastColor = currentColor;\n        } else {\n          boxWidth += charBox.kernedWidth;\n        }\n      }\n      if (currentColor && !this.path) {\n        drawStart = leftOffset + lineLeftOffset + boxStart;\n        if (this.direction === 'rtl') {\n          drawStart = this.width - drawStart - boxWidth;\n        }\n        ctx.fillStyle = currentColor;\n        ctx.fillRect(\n          drawStart,\n          lineTopOffset,\n          boxWidth,\n          heightOfLine / this.lineHeight\n        );\n      }\n      lineTopOffset += heightOfLine;\n    }\n    ctx.fillStyle = originalFill;\n    // if there is text background color no\n    // other shadows should be casted\n    this._removeShadow(ctx);\n  }\n\n  /**\n   * measure and return the width of a single character.\n   * possibly overridden to accommodate different measure logic or\n   * to hook some external lib for character measurement\n   * @private\n   * @param {String} _char, char to be measured\n   * @param {Object} charStyle style of char to be measured\n   * @param {String} [previousChar] previous char\n   * @param {Object} [prevCharStyle] style of previous char\n   */\n  _measureChar(\n    _char: string,\n    charStyle: CompleteTextStyleDeclaration,\n    previousChar: string | undefined,\n    prevCharStyle: CompleteTextStyleDeclaration | Record<string, never>\n  ) {\n    const fontCache = cache.getFontCache(charStyle),\n      fontDeclaration = this._getFontDeclaration(charStyle),\n      couple = previousChar + _char,\n      stylesAreEqual =\n        previousChar &&\n        fontDeclaration === this._getFontDeclaration(prevCharStyle),\n      fontMultiplier = charStyle.fontSize / this.CACHE_FONT_SIZE;\n    let width: number | undefined,\n      coupleWidth: number | undefined,\n      previousWidth: number | undefined,\n      kernedWidth: number | undefined;\n\n    if (previousChar && fontCache[previousChar] !== undefined) {\n      previousWidth = fontCache[previousChar];\n    }\n    if (fontCache[_char] !== undefined) {\n      kernedWidth = width = fontCache[_char];\n    }\n    if (stylesAreEqual && fontCache[couple] !== undefined) {\n      coupleWidth = fontCache[couple];\n      kernedWidth = coupleWidth - previousWidth!;\n    }\n    if (\n      width === undefined ||\n      previousWidth === undefined ||\n      coupleWidth === undefined\n    ) {\n      const ctx = getMeasuringContext()!;\n      // send a TRUE to specify measuring font size CACHE_FONT_SIZE\n      this._setTextStyles(ctx, charStyle, true);\n      if (width === undefined) {\n        kernedWidth = width = ctx.measureText(_char).width;\n        fontCache[_char] = width;\n      }\n      if (previousWidth === undefined && stylesAreEqual && previousChar) {\n        previousWidth = ctx.measureText(previousChar).width;\n        fontCache[previousChar] = previousWidth;\n      }\n      if (stylesAreEqual && coupleWidth === undefined) {\n        // we can measure the kerning couple and subtract the width of the previous character\n        coupleWidth = ctx.measureText(couple).width;\n        fontCache[couple] = coupleWidth;\n        // safe to use the non-null since if undefined we defined it before.\n        kernedWidth = coupleWidth - previousWidth!;\n      }\n    }\n    return {\n      width: width * fontMultiplier,\n      kernedWidth: kernedWidth! * fontMultiplier,\n    };\n  }\n\n  /**\n   * Computes height of character at given position\n   * @param {Number} line the line index number\n   * @param {Number} _char the character index number\n   * @return {Number} fontSize of the character\n   */\n  getHeightOfChar(line: number, _char: number): number {\n    return this.getValueOfPropertyAt(line, _char, 'fontSize');\n  }\n\n  /**\n   * measure a text line measuring all characters.\n   * @param {Number} lineIndex line number\n   */\n  measureLine(lineIndex: number) {\n    const lineInfo = this._measureLine(lineIndex);\n    if (this.charSpacing !== 0) {\n      lineInfo.width -= this._getWidthOfCharSpacing();\n    }\n    if (lineInfo.width < 0) {\n      lineInfo.width = 0;\n    }\n    return lineInfo;\n  }\n\n  /**\n   * measure every grapheme of a line, populating __charBounds\n   * @param {Number} lineIndex\n   * @return {Object} object.width total width of characters\n   * @return {Object} object.numOfSpaces length of chars that match this._reSpacesAndTabs\n   */\n  _measureLine(lineIndex: number) {\n    let width = 0,\n      prevGrapheme: string | undefined,\n      graphemeInfo: GraphemeBBox | undefined;\n\n    const reverse = this.pathSide === RIGHT,\n      path = this.path,\n      line = this._textLines[lineIndex],\n      llength = line.length,\n      lineBounds = new Array<GraphemeBBox>(llength);\n\n    this.__charBounds[lineIndex] = lineBounds;\n    for (let i = 0; i < llength; i++) {\n      const grapheme = line[i];\n      graphemeInfo = this._getGraphemeBox(grapheme, lineIndex, i, prevGrapheme);\n      lineBounds[i] = graphemeInfo;\n      width += graphemeInfo.kernedWidth;\n      prevGrapheme = grapheme;\n    }\n    // this latest bound box represent the last character of the line\n    // to simplify cursor handling in interactive mode.\n    lineBounds[llength] = {\n      left: graphemeInfo ? graphemeInfo.left + graphemeInfo.width : 0,\n      width: 0,\n      kernedWidth: 0,\n      height: this.fontSize,\n      deltaY: 0,\n    } as GraphemeBBox;\n    if (path && path.segmentsInfo) {\n      let positionInPath = 0;\n      const totalPathLength =\n        path.segmentsInfo[path.segmentsInfo.length - 1].length;\n      switch (this.textAlign) {\n        case LEFT:\n          positionInPath = reverse ? totalPathLength - width : 0;\n          break;\n        case CENTER:\n          positionInPath = (totalPathLength - width) / 2;\n          break;\n        case RIGHT:\n          positionInPath = reverse ? 0 : totalPathLength - width;\n          break;\n        //todo - add support for justify\n      }\n      positionInPath += this.pathStartOffset * (reverse ? -1 : 1);\n      for (\n        let i = reverse ? llength - 1 : 0;\n        reverse ? i >= 0 : i < llength;\n        reverse ? i-- : i++\n      ) {\n        graphemeInfo = lineBounds[i];\n        if (positionInPath > totalPathLength) {\n          positionInPath %= totalPathLength;\n        } else if (positionInPath < 0) {\n          positionInPath += totalPathLength;\n        }\n        // it would probably much faster to send all the grapheme position for a line\n        // and calculate path position/angle at once.\n        this._setGraphemeOnPath(positionInPath, graphemeInfo);\n        positionInPath += graphemeInfo.kernedWidth;\n      }\n    }\n    return { width: width, numOfSpaces: 0 };\n  }\n\n  /**\n   * Calculate the angle  and the left,top position of the char that follow a path.\n   * It appends it to graphemeInfo to be reused later at rendering\n   * @private\n   * @param {Number} positionInPath to be measured\n   * @param {GraphemeBBox} graphemeInfo current grapheme box information\n   * @param {Object} startingPoint position of the point\n   */\n  _setGraphemeOnPath(positionInPath: number, graphemeInfo: GraphemeBBox) {\n    const centerPosition = positionInPath + graphemeInfo.kernedWidth / 2,\n      path = this.path!;\n\n    // we are at currentPositionOnPath. we want to know what point on the path is.\n    const info = getPointOnPath(path.path, centerPosition, path.segmentsInfo)!;\n    graphemeInfo.renderLeft = info.x - path.pathOffset.x;\n    graphemeInfo.renderTop = info.y - path.pathOffset.y;\n    graphemeInfo.angle = info.angle + (this.pathSide === RIGHT ? Math.PI : 0);\n  }\n\n  /**\n   *\n   * @param {String} grapheme to be measured\n   * @param {Number} lineIndex index of the line where the char is\n   * @param {Number} charIndex position in the line\n   * @param {String} [prevGrapheme] character preceding the one to be measured\n   * @returns {GraphemeBBox} grapheme bbox\n   */\n  _getGraphemeBox(\n    grapheme: string,\n    lineIndex: number,\n    charIndex: number,\n    prevGrapheme?: string,\n    skipLeft?: boolean\n  ): GraphemeBBox {\n    const style = this.getCompleteStyleDeclaration(lineIndex, charIndex),\n      prevStyle = prevGrapheme\n        ? this.getCompleteStyleDeclaration(lineIndex, charIndex - 1)\n        : {},\n      info = this._measureChar(grapheme, style, prevGrapheme, prevStyle);\n    let kernedWidth = info.kernedWidth,\n      width = info.width,\n      charSpacing;\n\n    if (this.charSpacing !== 0) {\n      charSpacing = this._getWidthOfCharSpacing();\n      width += charSpacing;\n      kernedWidth += charSpacing;\n    }\n\n    const box: GraphemeBBox = {\n      width,\n      left: 0,\n      height: style.fontSize,\n      kernedWidth,\n      deltaY: style.deltaY,\n    };\n    if (charIndex > 0 && !skipLeft) {\n      const previousBox = this.__charBounds[lineIndex][charIndex - 1];\n      box.left =\n        previousBox.left + previousBox.width + info.kernedWidth - info.width;\n    }\n    return box;\n  }\n\n  /**\n   * Calculate height of line at 'lineIndex'\n   * @param {Number} lineIndex index of line to calculate\n   * @return {Number}\n   */\n  getHeightOfLine(lineIndex: number): number {\n    if (this.__lineHeights[lineIndex]) {\n      return this.__lineHeights[lineIndex];\n    }\n\n    // char 0 is measured before the line cycle because it needs to char\n    // emptylines\n    let maxHeight = this.getHeightOfChar(lineIndex, 0);\n    for (let i = 1, len = this._textLines[lineIndex].length; i < len; i++) {\n      maxHeight = Math.max(this.getHeightOfChar(lineIndex, i), maxHeight);\n    }\n\n    return (this.__lineHeights[lineIndex] =\n      maxHeight * this.lineHeight * this._fontSizeMult);\n  }\n\n  /**\n   * Calculate text box height\n   */\n  calcTextHeight() {\n    let lineHeight,\n      height = 0;\n    for (let i = 0, len = this._textLines.length; i < len; i++) {\n      lineHeight = this.getHeightOfLine(i);\n      height += i === len - 1 ? lineHeight / this.lineHeight : lineHeight;\n    }\n    return height;\n  }\n\n  /**\n   * @private\n   * @return {Number} Left offset\n   */\n  _getLeftOffset(): number {\n    return this.direction === 'ltr' ? -this.width / 2 : this.width / 2;\n  }\n\n  /**\n   * @private\n   * @return {Number} Top offset\n   */\n  _getTopOffset(): number {\n    return -this.height / 2;\n  }\n\n  /**\n   * @private\n   * @param {CanvasRenderingContext2D} ctx Context to render on\n   * @param {String} method Method name (\"fillText\" or \"strokeText\")\n   */\n  _renderTextCommon(\n    ctx: CanvasRenderingContext2D,\n    method: 'fillText' | 'strokeText'\n  ) {\n    ctx.save();\n    let lineHeights = 0;\n    const left = this._getLeftOffset(),\n      top = this._getTopOffset();\n    for (let i = 0, len = this._textLines.length; i < len; i++) {\n      const heightOfLine = this.getHeightOfLine(i),\n        maxHeight = heightOfLine / this.lineHeight,\n        leftOffset = this._getLineLeftOffset(i);\n      this._renderTextLine(\n        method,\n        ctx,\n        this._textLines[i],\n        left + leftOffset,\n        top + lineHeights + maxHeight,\n        i\n      );\n      lineHeights += heightOfLine;\n    }\n    ctx.restore();\n  }\n\n  /**\n   * @private\n   * @param {CanvasRenderingContext2D} ctx Context to render on\n   */\n  _renderTextFill(ctx: CanvasRenderingContext2D) {\n    if (!this.fill && !this.styleHas('fill')) {\n      return;\n    }\n\n    this._renderTextCommon(ctx, 'fillText');\n  }\n\n  /**\n   * @private\n   * @param {CanvasRenderingContext2D} ctx Context to render on\n   */\n  _renderTextStroke(ctx: CanvasRenderingContext2D) {\n    if ((!this.stroke || this.strokeWidth === 0) && this.isEmptyStyles()) {\n      return;\n    }\n\n    if (this.shadow && !this.shadow.affectStroke) {\n      this._removeShadow(ctx);\n    }\n\n    ctx.save();\n    this._setLineDash(ctx, this.strokeDashArray);\n    ctx.beginPath();\n    this._renderTextCommon(ctx, 'strokeText');\n    ctx.closePath();\n    ctx.restore();\n  }\n\n  /**\n   * @private\n   * @param {String} method fillText or strokeText.\n   * @param {CanvasRenderingContext2D} ctx Context to render on\n   * @param {Array} line Content of the line, splitted in an array by grapheme\n   * @param {Number} left\n   * @param {Number} top\n   * @param {Number} lineIndex\n   */\n  _renderChars(\n    method: 'fillText' | 'strokeText',\n    ctx: CanvasRenderingContext2D,\n    line: Array<any>,\n    left: number,\n    top: number,\n    lineIndex: number\n  ) {\n    const lineHeight = this.getHeightOfLine(lineIndex),\n      isJustify = this.textAlign.includes(JUSTIFY),\n      path = this.path,\n      shortCut =\n        !isJustify &&\n        this.charSpacing === 0 &&\n        this.isEmptyStyles(lineIndex) &&\n        !path,\n      isLtr = this.direction === 'ltr',\n      sign = this.direction === 'ltr' ? 1 : -1,\n      // this was changed in the PR #7674\n      // currentDirection = ctx.canvas.getAttribute('dir');\n      currentDirection = ctx.direction;\n\n    let actualStyle,\n      nextStyle,\n      charsToRender = '',\n      charBox,\n      boxWidth = 0,\n      timeToRender,\n      drawingLeft;\n\n    ctx.save();\n    if (currentDirection !== this.direction) {\n      ctx.canvas.setAttribute('dir', isLtr ? 'ltr' : 'rtl');\n      ctx.direction = isLtr ? 'ltr' : 'rtl';\n      ctx.textAlign = isLtr ? LEFT : RIGHT;\n    }\n    top -= (lineHeight * this._fontSizeFraction) / this.lineHeight;\n    if (shortCut) {\n      // render all the line in one pass without checking\n      // drawingLeft = isLtr ? left : left - this.getLineWidth(lineIndex);\n      this._renderChar(method, ctx, lineIndex, 0, line.join(''), left, top);\n      ctx.restore();\n      return;\n    }\n    for (let i = 0, len = line.length - 1; i <= len; i++) {\n      timeToRender = i === len || this.charSpacing || path;\n      charsToRender += line[i];\n      charBox = this.__charBounds[lineIndex][i] as Required<GraphemeBBox>;\n      if (boxWidth === 0) {\n        left += sign * (charBox.kernedWidth - charBox.width);\n        boxWidth += charBox.width;\n      } else {\n        boxWidth += charBox.kernedWidth;\n      }\n      if (isJustify && !timeToRender) {\n        if (this._reSpaceAndTab.test(line[i])) {\n          timeToRender = true;\n        }\n      }\n      if (!timeToRender) {\n        // if we have charSpacing, we render char by char\n        actualStyle =\n          actualStyle || this.getCompleteStyleDeclaration(lineIndex, i);\n        nextStyle = this.getCompleteStyleDeclaration(lineIndex, i + 1);\n        timeToRender = hasStyleChanged(actualStyle, nextStyle, false);\n      }\n      if (timeToRender) {\n        if (path) {\n          ctx.save();\n          ctx.translate(charBox.renderLeft, charBox.renderTop);\n          ctx.rotate(charBox.angle);\n          this._renderChar(\n            method,\n            ctx,\n            lineIndex,\n            i,\n            charsToRender,\n            -boxWidth / 2,\n            0\n          );\n          ctx.restore();\n        } else {\n          drawingLeft = left;\n          this._renderChar(\n            method,\n            ctx,\n            lineIndex,\n            i,\n            charsToRender,\n            drawingLeft,\n            top\n          );\n        }\n        charsToRender = '';\n        actualStyle = nextStyle;\n        left += sign * boxWidth;\n        boxWidth = 0;\n      }\n    }\n    ctx.restore();\n  }\n\n  /**\n   * This function try to patch the missing gradientTransform on canvas gradients.\n   * transforming a context to transform the gradient, is going to transform the stroke too.\n   * we want to transform the gradient but not the stroke operation, so we create\n   * a transformed gradient on a pattern and then we use the pattern instead of the gradient.\n   * this method has drawbacks: is slow, is in low resolution, needs a patch for when the size\n   * is limited.\n   * @private\n   * @param {TFiller} filler a fabric gradient instance\n   * @return {CanvasPattern} a pattern to use as fill/stroke style\n   */\n  _applyPatternGradientTransformText(filler: TFiller) {\n    const pCanvas = createCanvasElement(),\n      // TODO: verify compatibility with strokeUniform\n      width = this.width + this.strokeWidth,\n      height = this.height + this.strokeWidth,\n      pCtx = pCanvas.getContext('2d')!;\n    pCanvas.width = width;\n    pCanvas.height = height;\n    pCtx.beginPath();\n    pCtx.moveTo(0, 0);\n    pCtx.lineTo(width, 0);\n    pCtx.lineTo(width, height);\n    pCtx.lineTo(0, height);\n    pCtx.closePath();\n    pCtx.translate(width / 2, height / 2);\n    pCtx.fillStyle = filler.toLive(pCtx)!;\n    this._applyPatternGradientTransform(pCtx, filler);\n    pCtx.fill();\n    return pCtx.createPattern(pCanvas, 'no-repeat')!;\n  }\n\n  handleFiller<T extends 'fill' | 'stroke'>(\n    ctx: CanvasRenderingContext2D,\n    property: `${T}Style`,\n    filler: TFiller | string\n  ): { offsetX: number; offsetY: number } {\n    let offsetX: number, offsetY: number;\n    if (isFiller(filler)) {\n      if (\n        (filler as Gradient<'linear'>).gradientUnits === 'percentage' ||\n        (filler as Gradient<'linear'>).gradientTransform ||\n        (filler as Pattern).patternTransform\n      ) {\n        // need to transform gradient in a pattern.\n        // this is a slow process. If you are hitting this codepath, and the object\n        // is not using caching, you should consider switching it on.\n        // we need a canvas as big as the current object caching canvas.\n        offsetX = -this.width / 2;\n        offsetY = -this.height / 2;\n        ctx.translate(offsetX, offsetY);\n        ctx[property] = this._applyPatternGradientTransformText(filler);\n        return { offsetX, offsetY };\n      } else {\n        // is a simple gradient or pattern\n        ctx[property] = filler.toLive(ctx)!;\n        return this._applyPatternGradientTransform(ctx, filler);\n      }\n    } else {\n      // is a color\n      ctx[property] = filler;\n    }\n    return { offsetX: 0, offsetY: 0 };\n  }\n\n  /**\n   * This function prepare the canvas for a stroke style, and stroke and strokeWidth\n   * need to be sent in as defined\n   * @param {CanvasRenderingContext2D} ctx\n   * @param {CompleteTextStyleDeclaration} style with stroke and strokeWidth defined\n   * @returns\n   */\n  _setStrokeStyles(\n    ctx: CanvasRenderingContext2D,\n    {\n      stroke,\n      strokeWidth,\n    }: Pick<CompleteTextStyleDeclaration, 'stroke' | 'strokeWidth'>\n  ) {\n    ctx.lineWidth = strokeWidth;\n    ctx.lineCap = this.strokeLineCap;\n    ctx.lineDashOffset = this.strokeDashOffset;\n    ctx.lineJoin = this.strokeLineJoin;\n    ctx.miterLimit = this.strokeMiterLimit;\n    return this.handleFiller(ctx, 'strokeStyle', stroke!);\n  }\n\n  /**\n   * This function prepare the canvas for a ill style, and fill\n   * need to be sent in as defined\n   * @param {CanvasRenderingContext2D} ctx\n   * @param {CompleteTextStyleDeclaration} style with ill defined\n   * @returns\n   */\n  _setFillStyles(ctx: CanvasRenderingContext2D, { fill }: Pick<this, 'fill'>) {\n    return this.handleFiller(ctx, 'fillStyle', fill!);\n  }\n\n  /**\n   * @private\n   * @param {String} method\n   * @param {CanvasRenderingContext2D} ctx Context to render on\n   * @param {Number} lineIndex\n   * @param {Number} charIndex\n   * @param {String} _char\n   * @param {Number} left Left coordinate\n   * @param {Number} top Top coordinate\n   * @param {Number} lineHeight Height of the line\n   */\n  _renderChar(\n    method: 'fillText' | 'strokeText',\n    ctx: CanvasRenderingContext2D,\n    lineIndex: number,\n    charIndex: number,\n    _char: string,\n    left: number,\n    top: number\n  ) {\n    const decl = this._getStyleDeclaration(lineIndex, charIndex),\n      fullDecl = this.getCompleteStyleDeclaration(lineIndex, charIndex),\n      shouldFill = method === 'fillText' && fullDecl.fill,\n      shouldStroke =\n        method === 'strokeText' && fullDecl.stroke && fullDecl.strokeWidth;\n\n    if (!shouldStroke && !shouldFill) {\n      return;\n    }\n    ctx.save();\n\n    ctx.font = this._getFontDeclaration(fullDecl);\n\n    if (decl.textBackgroundColor) {\n      this._removeShadow(ctx);\n    }\n    if (decl.deltaY) {\n      top += decl.deltaY;\n    }\n\n    if (shouldFill) {\n      const fillOffsets = this._setFillStyles(ctx, fullDecl);\n      ctx.fillText(\n        _char,\n        left - fillOffsets.offsetX,\n        top - fillOffsets.offsetY\n      );\n    }\n\n    if (shouldStroke) {\n      const strokeOffsets = this._setStrokeStyles(ctx, fullDecl);\n      ctx.strokeText(\n        _char,\n        left - strokeOffsets.offsetX,\n        top - strokeOffsets.offsetY\n      );\n    }\n\n    ctx.restore();\n  }\n\n  /**\n   * Turns the character into a 'superior figure' (i.e. 'superscript')\n   * @param {Number} start selection start\n   * @param {Number} end selection end\n   */\n  setSuperscript(start: number, end: number) {\n    this._setScript(start, end, this.superscript);\n  }\n\n  /**\n   * Turns the character into an 'inferior figure' (i.e. 'subscript')\n   * @param {Number} start selection start\n   * @param {Number} end selection end\n   */\n  setSubscript(start: number, end: number) {\n    this._setScript(start, end, this.subscript);\n  }\n\n  /**\n   * Applies 'schema' at given position\n   * @private\n   * @param {Number} start selection start\n   * @param {Number} end selection end\n   * @param {Number} schema\n   */\n  protected _setScript(\n    start: number,\n    end: number,\n    schema: {\n      size: number;\n      baseline: number;\n    }\n  ) {\n    const loc = this.get2DCursorLocation(start, true),\n      fontSize = this.getValueOfPropertyAt(\n        loc.lineIndex,\n        loc.charIndex,\n        'fontSize'\n      ),\n      dy = this.getValueOfPropertyAt(loc.lineIndex, loc.charIndex, 'deltaY'),\n      style = {\n        fontSize: fontSize * schema.size,\n        deltaY: dy + fontSize * schema.baseline,\n      };\n    this.setSelectionStyles(style, start, end);\n  }\n\n  /**\n   * @private\n   * @param {Number} lineIndex index text line\n   * @return {Number} Line left offset\n   */\n  _getLineLeftOffset(lineIndex: number): number {\n    const lineWidth = this.getLineWidth(lineIndex),\n      lineDiff = this.width - lineWidth,\n      textAlign = this.textAlign,\n      direction = this.direction,\n      isEndOfWrapping = this.isEndOfWrapping(lineIndex);\n    let leftOffset = 0;\n    if (\n      textAlign === JUSTIFY ||\n      (textAlign === JUSTIFY_CENTER && !isEndOfWrapping) ||\n      (textAlign === JUSTIFY_RIGHT && !isEndOfWrapping) ||\n      (textAlign === JUSTIFY_LEFT && !isEndOfWrapping)\n    ) {\n      return 0;\n    }\n    if (textAlign === CENTER) {\n      leftOffset = lineDiff / 2;\n    }\n    if (textAlign === RIGHT) {\n      leftOffset = lineDiff;\n    }\n    if (textAlign === JUSTIFY_CENTER) {\n      leftOffset = lineDiff / 2;\n    }\n    if (textAlign === JUSTIFY_RIGHT) {\n      leftOffset = lineDiff;\n    }\n    if (direction === 'rtl') {\n      if (\n        textAlign === RIGHT ||\n        textAlign === JUSTIFY ||\n        textAlign === JUSTIFY_RIGHT\n      ) {\n        leftOffset = 0;\n      } else if (textAlign === LEFT || textAlign === JUSTIFY_LEFT) {\n        leftOffset = -lineDiff;\n      } else if (textAlign === CENTER || textAlign === JUSTIFY_CENTER) {\n        leftOffset = -lineDiff / 2;\n      }\n    }\n    return leftOffset;\n  }\n\n  /**\n   * @private\n   */\n  _clearCache() {\n    this._forceClearCache = false;\n    this.__lineWidths = [];\n    this.__lineHeights = [];\n    this.__charBounds = [];\n  }\n\n  /**\n   * Measure a single line given its index. Used to calculate the initial\n   * text bounding box. The values are calculated and stored in __lineWidths cache.\n   * @private\n   * @param {Number} lineIndex line number\n   * @return {Number} Line width\n   */\n  getLineWidth(lineIndex: number): number {\n    if (this.__lineWidths[lineIndex] !== undefined) {\n      return this.__lineWidths[lineIndex];\n    }\n\n    const { width } = this.measureLine(lineIndex);\n    this.__lineWidths[lineIndex] = width;\n    return width;\n  }\n\n  _getWidthOfCharSpacing() {\n    if (this.charSpacing !== 0) {\n      return (this.fontSize * this.charSpacing) / 1000;\n    }\n    return 0;\n  }\n\n  /**\n   * Retrieves the value of property at given character position\n   * @param {Number} lineIndex the line number\n   * @param {Number} charIndex the character number\n   * @param {String} property the property name\n   * @returns the value of 'property'\n   */\n  getValueOfPropertyAt<T extends StylePropertiesType>(\n    lineIndex: number,\n    charIndex: number,\n    property: T\n  ): this[T] {\n    const charStyle = this._getStyleDeclaration(lineIndex, charIndex);\n    return (charStyle[property] ?? this[property]) as this[T];\n  }\n\n  /**\n   * @private\n   * @param {CanvasRenderingContext2D} ctx Context to render on\n   */\n  _renderTextDecoration(\n    ctx: CanvasRenderingContext2D,\n    type: 'underline' | 'linethrough' | 'overline'\n  ) {\n    if (!this[type] && !this.styleHas(type)) {\n      return;\n    }\n    let topOffset = this._getTopOffset();\n    const leftOffset = this._getLeftOffset(),\n      path = this.path,\n      charSpacing = this._getWidthOfCharSpacing(),\n      offsetY = this.offsets[type];\n\n    for (let i = 0, len = this._textLines.length; i < len; i++) {\n      const heightOfLine = this.getHeightOfLine(i);\n      if (!this[type] && !this.styleHas(type, i)) {\n        topOffset += heightOfLine;\n        continue;\n      }\n      const line = this._textLines[i];\n      const maxHeight = heightOfLine / this.lineHeight;\n      const lineLeftOffset = this._getLineLeftOffset(i);\n      let boxStart = 0;\n      let boxWidth = 0;\n      let lastDecoration = this.getValueOfPropertyAt(i, 0, type);\n      let lastFill = this.getValueOfPropertyAt(i, 0, 'fill');\n      let currentDecoration;\n      let currentFill;\n      const top = topOffset + maxHeight * (1 - this._fontSizeFraction);\n      let size = this.getHeightOfChar(i, 0);\n      let dy = this.getValueOfPropertyAt(i, 0, 'deltaY');\n      for (let j = 0, jlen = line.length; j < jlen; j++) {\n        const charBox = this.__charBounds[i][j] as Required<GraphemeBBox>;\n        currentDecoration = this.getValueOfPropertyAt(i, j, type);\n        currentFill = this.getValueOfPropertyAt(i, j, 'fill');\n        const currentSize = this.getHeightOfChar(i, j);\n        const currentDy = this.getValueOfPropertyAt(i, j, 'deltaY');\n        if (path && currentDecoration && currentFill) {\n          ctx.save();\n          // bug? verify lastFill is a valid fill here.\n          ctx.fillStyle = lastFill as string;\n          ctx.translate(charBox.renderLeft, charBox.renderTop);\n          ctx.rotate(charBox.angle);\n          ctx.fillRect(\n            -charBox.kernedWidth / 2,\n            offsetY * currentSize + currentDy,\n            charBox.kernedWidth,\n            this.fontSize / 15\n          );\n          ctx.restore();\n        } else if (\n          (currentDecoration !== lastDecoration ||\n            currentFill !== lastFill ||\n            currentSize !== size ||\n            currentDy !== dy) &&\n          boxWidth > 0\n        ) {\n          let drawStart = leftOffset + lineLeftOffset + boxStart;\n          if (this.direction === 'rtl') {\n            drawStart = this.width - drawStart - boxWidth;\n          }\n          if (lastDecoration && lastFill) {\n            // bug? verify lastFill is a valid fill here.\n            ctx.fillStyle = lastFill as string;\n            ctx.fillRect(\n              drawStart,\n              top + offsetY * size + dy,\n              boxWidth,\n              this.fontSize / 15\n            );\n          }\n          boxStart = charBox.left;\n          boxWidth = charBox.width;\n          lastDecoration = currentDecoration;\n          lastFill = currentFill;\n          size = currentSize;\n          dy = currentDy;\n        } else {\n          boxWidth += charBox.kernedWidth;\n        }\n      }\n      let drawStart = leftOffset + lineLeftOffset + boxStart;\n      if (this.direction === 'rtl') {\n        drawStart = this.width - drawStart - boxWidth;\n      }\n      ctx.fillStyle = currentFill as string;\n      currentDecoration &&\n        currentFill &&\n        ctx.fillRect(\n          drawStart,\n          top + offsetY * size + dy,\n          boxWidth - charSpacing,\n          this.fontSize / 15\n        );\n      topOffset += heightOfLine;\n    }\n    // if there is text background color no\n    // other shadows should be casted\n    this._removeShadow(ctx);\n  }\n\n  /**\n   * return font declaration string for canvas context\n   * @param {Object} [styleObject] object\n   * @returns {String} font declaration formatted for canvas context.\n   */\n  _getFontDeclaration(\n    {\n      fontFamily = this.fontFamily,\n      fontStyle = this.fontStyle,\n      fontWeight = this.fontWeight,\n      fontSize = this.fontSize,\n    }: Partial<\n      Pick<\n        TextStyleDeclaration,\n        'fontFamily' | 'fontStyle' | 'fontWeight' | 'fontSize'\n      >\n    > = {},\n    forMeasuring?: boolean\n  ): string {\n    const parsedFontFamily =\n      fontFamily.includes(\"'\") ||\n      fontFamily.includes('\"') ||\n      fontFamily.includes(',') ||\n      FabricText.genericFonts.includes(fontFamily.toLowerCase())\n        ? fontFamily\n        : `\"${fontFamily}\"`;\n    return [\n      fontStyle,\n      fontWeight,\n      `${forMeasuring ? this.CACHE_FONT_SIZE : fontSize}px`,\n      parsedFontFamily,\n    ].join(' ');\n  }\n\n  /**\n   * Renders text instance on a specified context\n   * @param {CanvasRenderingContext2D} ctx Context to render on\n   */\n  render(ctx: CanvasRenderingContext2D) {\n    if (!this.visible) {\n      return;\n    }\n    if (\n      this.canvas &&\n      this.canvas.skipOffscreen &&\n      !this.group &&\n      !this.isOnScreen()\n    ) {\n      return;\n    }\n    if (this._forceClearCache) {\n      this.initDimensions();\n    }\n    super.render(ctx);\n  }\n\n  /**\n   * Override this method to customize grapheme splitting\n   * @todo the util `graphemeSplit` needs to be injectable in some way.\n   * is more comfortable to inject the correct util rather than having to override text\n   * in the middle of the prototype chain\n   * @param {string} value\n   * @returns {string[]} array of graphemes\n   */\n  graphemeSplit(value: string): string[] {\n    return graphemeSplit(value);\n  }\n\n  /**\n   * Returns the text as an array of lines.\n   * @param {String} text text to split\n   * @returns  Lines in the text\n   */\n  _splitTextIntoLines(text: string): TextLinesInfo {\n    const lines = text.split(this._reNewline),\n      newLines = new Array<string[]>(lines.length),\n      newLine = ['\\n'];\n    let newText: string[] = [];\n    for (let i = 0; i < lines.length; i++) {\n      newLines[i] = this.graphemeSplit(lines[i]);\n      newText = newText.concat(newLines[i], newLine);\n    }\n    newText.pop();\n    return {\n      _unwrappedLines: newLines,\n      lines: lines,\n      graphemeText: newText,\n      graphemeLines: newLines,\n    };\n  }\n\n  /**\n   * Returns object representation of an instance\n   * @param {Array} [propertiesToInclude] Any properties that you might want to additionally include in the output\n   * @return {Object} Object representation of an instance\n   */\n  toObject<\n    T extends Omit<Props & TClassProperties<this>, keyof SProps>,\n    K extends keyof T = never\n  >(propertiesToInclude: K[] = []): Pick<T, K> & SProps {\n    return {\n      ...super.toObject([...additionalProps, ...propertiesToInclude] as K[]),\n      styles: stylesToArray(this.styles, this.text),\n      ...(this.path ? { path: this.path.toObject() } : {}),\n    };\n  }\n\n  set(key: string | any, value?: any) {\n    const { textLayoutProperties } = this.constructor as typeof FabricText;\n    super.set(key, value);\n    let needsDims = false;\n    let isAddingPath = false;\n    if (typeof key === 'object') {\n      for (const _key in key) {\n        if (_key === 'path') {\n          this.setPathInfo();\n        }\n        needsDims = needsDims || textLayoutProperties.includes(_key);\n        isAddingPath = isAddingPath || _key === 'path';\n      }\n    } else {\n      needsDims = textLayoutProperties.includes(key);\n      isAddingPath = key === 'path';\n    }\n    if (isAddingPath) {\n      this.setPathInfo();\n    }\n    if (needsDims && this.initialized) {\n      this.initDimensions();\n      this.setCoords();\n    }\n    return this;\n  }\n\n  /**\n   * Returns complexity of an instance\n   * @return {Number} complexity\n   */\n  complexity(): number {\n    return 1;\n  }\n\n  static genericFonts = [\n    'sans-serif',\n    'serif',\n    'cursive',\n    'fantasy',\n    'monospace',\n  ];\n\n  /* _FROM_SVG_START_ */\n\n  /**\n   * List of attribute names to account for when parsing SVG element (used by {@link FabricText.fromElement})\n   * @static\n   * @memberOf Text\n   * @see: http://www.w3.org/TR/SVG/text.html#TextElement\n   */\n  static ATTRIBUTE_NAMES = SHARED_ATTRIBUTES.concat(\n    'x',\n    'y',\n    'dx',\n    'dy',\n    'font-family',\n    'font-style',\n    'font-weight',\n    'font-size',\n    'letter-spacing',\n    'text-decoration',\n    'text-anchor'\n  );\n\n  /**\n   * Returns FabricText instance from an SVG element (<b>not yet implemented</b>)\n   * @static\n   * @memberOf Text\n   * @param {HTMLElement} element Element to parse\n   * @param {Object} [options] Options object\n   */\n  static async fromElement(\n    element: HTMLElement,\n    options: Abortable,\n    cssRules?: CSSRules\n  ) {\n    const parsedAttributes = parseAttributes(\n      element,\n      FabricText.ATTRIBUTE_NAMES,\n      cssRules\n    );\n\n    const {\n      textAnchor = LEFT as typeof LEFT | typeof CENTER | typeof RIGHT,\n      textDecoration = '',\n      dx = 0,\n      dy = 0,\n      top = 0,\n      left = 0,\n      fontSize = DEFAULT_SVG_FONT_SIZE,\n      strokeWidth = 1,\n      ...restOfOptions\n    } = { ...options, ...parsedAttributes };\n\n    const textContent = (element.textContent || '')\n      .replace(/^\\s+|\\s+$|\\n+/g, '')\n      .replace(/\\s+/g, ' ');\n\n    // this code here is probably the usual issue for SVG center find\n    // this can later looked at again and probably removed.\n\n    const text = new this(textContent, {\n        left: left + dx,\n        top: top + dy,\n        underline: textDecoration.includes('underline'),\n        overline: textDecoration.includes('overline'),\n        linethrough: textDecoration.includes('line-through'),\n        // we initialize this as 0\n        strokeWidth: 0,\n        fontSize,\n        ...restOfOptions,\n      }),\n      textHeightScaleFactor = text.getScaledHeight() / text.height,\n      lineHeightDiff =\n        (text.height + text.strokeWidth) * text.lineHeight - text.height,\n      scaledDiff = lineHeightDiff * textHeightScaleFactor,\n      textHeight = text.getScaledHeight() + scaledDiff;\n\n    let offX = 0;\n    /*\n      Adjust positioning:\n        x/y attributes in SVG correspond to the bottom-left corner of text bounding box\n        fabric output by default at top, left.\n    */\n    if (textAnchor === CENTER) {\n      offX = text.getScaledWidth() / 2;\n    }\n    if (textAnchor === RIGHT) {\n      offX = text.getScaledWidth();\n    }\n    text.set({\n      left: text.left - offX,\n      top:\n        text.top -\n        (textHeight - text.fontSize * (0.07 + text._fontSizeFraction)) /\n          text.lineHeight,\n      strokeWidth,\n    });\n    return text;\n  }\n\n  /* _FROM_SVG_END_ */\n\n  /**\n   * Returns FabricText instance from an object representation\n   * @param {Object} object plain js Object to create an instance from\n   * @returns {Promise<FabricText>}\n   */\n  static fromObject<\n    T extends TOptions<SerializedTextProps>,\n    S extends FabricText\n  >(object: T) {\n    return this._fromObject<S>(\n      {\n        ...object,\n        styles: stylesFromArray(object.styles || {}, object.text),\n      },\n      {\n        extraParam: 'text',\n      }\n    );\n  }\n}\n\napplyMixins(FabricText, [TextSVGExportMixin]);\nclassRegistry.setClass(FabricText);\nclassRegistry.setSVGClass(FabricText);\n","import type { Canvas } from '../../canvas/Canvas';\nimport type {\n  DragEventData,\n  DropEventData,\n  TPointerEvent,\n} from '../../EventTypeDefs';\nimport { Point } from '../../Point';\nimport type { IText } from './IText';\nimport { setStyle } from '../../util/dom_style';\nimport { cloneDeep } from '../../util/internals/cloneDeep';\nimport type { TextStyleDeclaration } from '../Text/StyledText';\nimport { getDocumentFromElement } from '../../util/dom_misc';\nimport { NONE } from '../../constants';\n\n/**\n * #### Dragging IText/Textbox Lifecycle\n * - {@link start} is called from `mousedown` {@link IText#_mouseDownHandler} and determines if dragging should start by testing {@link isPointerOverSelection}\n * - if true `mousedown` {@link IText#_mouseDownHandler} is blocked to keep selection\n * - if the pointer moves, canvas fires numerous mousemove {@link Canvas#_onMouseMove} that we make sure **aren't** prevented ({@link IText#shouldStartDragging}) in order for the window to start a drag session\n * - once/if the session starts canvas calls {@link onDragStart} on the active object to determine if dragging should occur\n * - canvas fires relevant drag events that are handled by the handlers defined in this scope\n * - {@link end} is called from `mouseup` {@link IText#mouseUpHandler}, blocking IText default click behavior\n * - in case the drag session didn't occur, {@link end} handles a click, since logic to do so was blocked during `mousedown`\n */\nexport class DraggableTextDelegate {\n  readonly target: IText;\n  private __mouseDownInPlace = false;\n  private __dragStartFired = false;\n  private __isDraggingOver = false;\n  private __dragStartSelection?: {\n    selectionStart: number;\n    selectionEnd: number;\n  };\n  private __dragImageDisposer?: VoidFunction;\n  private _dispose?: () => void;\n\n  constructor(target: IText) {\n    this.target = target;\n    const disposers = [\n      this.target.on('dragenter', this.dragEnterHandler.bind(this)),\n      this.target.on('dragover', this.dragOverHandler.bind(this)),\n      this.target.on('dragleave', this.dragLeaveHandler.bind(this)),\n      this.target.on('dragend', this.dragEndHandler.bind(this)),\n      this.target.on('drop', this.dropHandler.bind(this)),\n    ];\n    this._dispose = () => {\n      disposers.forEach((d) => d());\n      this._dispose = undefined;\n    };\n  }\n\n  isPointerOverSelection(e: TPointerEvent) {\n    const target = this.target;\n    const newSelection = target.getSelectionStartFromPointer(e);\n    return (\n      target.isEditing &&\n      newSelection >= target.selectionStart &&\n      newSelection <= target.selectionEnd &&\n      target.selectionStart < target.selectionEnd\n    );\n  }\n\n  /**\n   * @public override this method to disable dragging and default to mousedown logic\n   */\n  start(e: TPointerEvent) {\n    return (this.__mouseDownInPlace = this.isPointerOverSelection(e));\n  }\n\n  /**\n   * @public override this method to disable dragging without discarding selection\n   */\n  isActive() {\n    return this.__mouseDownInPlace;\n  }\n\n  /**\n   * Ends interaction and sets cursor in case of a click\n   * @returns true if was active\n   */\n  end(e: TPointerEvent) {\n    const active = this.isActive();\n    if (active && !this.__dragStartFired) {\n      // mousedown has been blocked since `active` is true => cursor has not been set.\n      // `__dragStartFired` is false => dragging didn't occur, pointer didn't move and is over selection.\n      // meaning this is actually a click, `active` is a false positive.\n      this.target.setCursorByClick(e);\n      this.target.initDelayedCursor(true);\n    }\n    this.__mouseDownInPlace = false;\n    this.__dragStartFired = false;\n    this.__isDraggingOver = false;\n    return active;\n  }\n\n  getDragStartSelection() {\n    return this.__dragStartSelection;\n  }\n\n  /**\n   * Override to customize the drag image\n   * https://developer.mozilla.org/en-US/docs/Web/API/DataTransfer/setDragImage\n   */\n  setDragImage(\n    e: DragEvent,\n    {\n      selectionStart,\n      selectionEnd,\n    }: {\n      selectionStart: number;\n      selectionEnd: number;\n    }\n  ) {\n    const target = this.target;\n    const canvas = target.canvas!;\n    const flipFactor = new Point(target.flipX ? -1 : 1, target.flipY ? -1 : 1);\n    const boundaries = target._getCursorBoundaries(selectionStart);\n    const selectionPosition = new Point(\n      boundaries.left + boundaries.leftOffset,\n      boundaries.top + boundaries.topOffset\n    ).multiply(flipFactor);\n    const pos = selectionPosition.transform(target.calcTransformMatrix());\n    const pointer = canvas.getScenePoint(e);\n    const diff = pointer.subtract(pos);\n    const retinaScaling = target.getCanvasRetinaScaling();\n    const bbox = target.getBoundingRect();\n    const correction = pos.subtract(new Point(bbox.left, bbox.top));\n    const vpt = canvas.viewportTransform;\n    const offset = correction.add(diff).transform(vpt, true);\n    //  prepare instance for drag image snapshot by making all non selected text invisible\n    const bgc = target.backgroundColor;\n    const styles = cloneDeep(target.styles);\n    target.backgroundColor = '';\n    const styleOverride = {\n      stroke: 'transparent',\n      fill: 'transparent',\n      textBackgroundColor: 'transparent',\n    };\n    target.setSelectionStyles(styleOverride, 0, selectionStart);\n    target.setSelectionStyles(styleOverride, selectionEnd, target.text.length);\n    target.dirty = true;\n    const dragImage = target.toCanvasElement({\n      enableRetinaScaling: canvas.enableRetinaScaling,\n      viewportTransform: true,\n    });\n    // restore values\n    target.backgroundColor = bgc;\n    target.styles = styles;\n    target.dirty = true;\n    //  position drag image offscreen\n    setStyle(dragImage, {\n      position: 'fixed',\n      left: `${-dragImage.width}px`,\n      border: NONE,\n      width: `${dragImage.width / retinaScaling}px`,\n      height: `${dragImage.height / retinaScaling}px`,\n    });\n    this.__dragImageDisposer && this.__dragImageDisposer();\n    this.__dragImageDisposer = () => {\n      dragImage.remove();\n    };\n    getDocumentFromElement(\n      (e.target || this.target.hiddenTextarea)! as HTMLElement\n    ).body.appendChild(dragImage);\n    e.dataTransfer?.setDragImage(dragImage, offset.x, offset.y);\n  }\n\n  /**\n   * @returns {boolean} determines whether {@link target} should/shouldn't become a drag source\n   */\n  onDragStart(e: DragEvent): boolean {\n    this.__dragStartFired = true;\n    const target = this.target;\n    const active = this.isActive();\n    if (active && e.dataTransfer) {\n      const selection = (this.__dragStartSelection = {\n        selectionStart: target.selectionStart,\n        selectionEnd: target.selectionEnd,\n      });\n      const value = target._text\n        .slice(selection.selectionStart, selection.selectionEnd)\n        .join('');\n      const data = { text: target.text, value, ...selection };\n      e.dataTransfer.setData('text/plain', value);\n      e.dataTransfer.setData(\n        'application/fabric',\n        JSON.stringify({\n          value: value,\n          styles: target.getSelectionStyles(\n            selection.selectionStart,\n            selection.selectionEnd,\n            true\n          ),\n        })\n      );\n      e.dataTransfer.effectAllowed = 'copyMove';\n      this.setDragImage(e, data);\n    }\n    target.abortCursorAnimation();\n    return active;\n  }\n\n  /**\n   * use {@link targetCanDrop} to respect overriding\n   * @returns {boolean} determines whether {@link target} should/shouldn't become a drop target\n   */\n  canDrop(e: DragEvent): boolean {\n    if (\n      this.target.editable &&\n      !this.target.getActiveControl() &&\n      !e.defaultPrevented\n    ) {\n      if (this.isActive() && this.__dragStartSelection) {\n        //  drag source trying to drop over itself\n        //  allow dropping only outside of drag start selection\n        const index = this.target.getSelectionStartFromPointer(e);\n        const dragStartSelection = this.__dragStartSelection;\n        return (\n          index < dragStartSelection.selectionStart ||\n          index > dragStartSelection.selectionEnd\n        );\n      }\n      return true;\n    }\n    return false;\n  }\n\n  /**\n   * in order to respect overriding {@link IText#canDrop} we call that instead of calling {@link canDrop} directly\n   */\n  protected targetCanDrop(e: DragEvent) {\n    return this.target.canDrop(e);\n  }\n\n  dragEnterHandler({ e }: DragEventData) {\n    const canDrop = this.targetCanDrop(e);\n    if (!this.__isDraggingOver && canDrop) {\n      this.__isDraggingOver = true;\n    }\n  }\n\n  dragOverHandler(ev: DragEventData) {\n    const { e } = ev;\n    const canDrop = this.targetCanDrop(e);\n    if (!this.__isDraggingOver && canDrop) {\n      this.__isDraggingOver = true;\n    } else if (this.__isDraggingOver && !canDrop) {\n      //  drop state has changed\n      this.__isDraggingOver = false;\n    }\n    if (this.__isDraggingOver) {\n      //  can be dropped, inform browser\n      e.preventDefault();\n      //  inform event subscribers\n      ev.canDrop = true;\n      ev.dropTarget = this.target;\n    }\n  }\n\n  dragLeaveHandler() {\n    if (this.__isDraggingOver || this.isActive()) {\n      this.__isDraggingOver = false;\n    }\n  }\n\n  /**\n   * Override the `text/plain | application/fabric` types of {@link DragEvent#dataTransfer}\n   * in order to change the drop value or to customize styling respectively, by listening to the `drop:before` event\n   * https://developer.mozilla.org/en-US/docs/Web/API/HTML_Drag_and_Drop_API/Drag_operations#performing_a_drop\n   */\n  dropHandler(ev: DropEventData) {\n    const { e } = ev;\n    const didDrop = e.defaultPrevented;\n    this.__isDraggingOver = false;\n    // inform browser that the drop has been accepted\n    e.preventDefault();\n    let insert = e.dataTransfer?.getData('text/plain');\n    if (insert && !didDrop) {\n      const target = this.target;\n      const canvas = target.canvas!;\n      let insertAt = target.getSelectionStartFromPointer(e);\n      const { styles } = (\n        e.dataTransfer!.types.includes('application/fabric')\n          ? JSON.parse(e.dataTransfer!.getData('application/fabric'))\n          : {}\n      ) as { styles: TextStyleDeclaration[] };\n      const trailing = insert[Math.max(0, insert.length - 1)];\n      const selectionStartOffset = 0;\n      //  drag and drop in same instance\n      if (this.__dragStartSelection) {\n        const selectionStart = this.__dragStartSelection.selectionStart;\n        const selectionEnd = this.__dragStartSelection.selectionEnd;\n        if (insertAt > selectionStart && insertAt <= selectionEnd) {\n          insertAt = selectionStart;\n        } else if (insertAt > selectionEnd) {\n          insertAt -= selectionEnd - selectionStart;\n        }\n        target.removeChars(selectionStart, selectionEnd);\n        // prevent `dragend` from handling event\n        delete this.__dragStartSelection;\n      }\n      //  remove redundant line break\n      if (\n        target._reNewline.test(trailing) &&\n        (target._reNewline.test(target._text[insertAt]) ||\n          insertAt === target._text.length)\n      ) {\n        insert = insert.trimEnd();\n      }\n      //  inform subscribers\n      ev.didDrop = true;\n      ev.dropTarget = target;\n      //  finalize\n      target.insertChars(insert, styles, insertAt);\n      // can this part be moved in an outside event? andrea to check.\n      canvas.setActiveObject(target);\n      target.enterEditing(e);\n      target.selectionStart = Math.min(\n        insertAt + selectionStartOffset,\n        target._text.length\n      );\n      target.selectionEnd = Math.min(\n        target.selectionStart + insert.length,\n        target._text.length\n      );\n      target.hiddenTextarea!.value = target.text;\n      target._updateTextarea();\n      target.hiddenTextarea!.focus();\n      target.fire('changed', {\n        index: insertAt + selectionStartOffset,\n        action: 'drop',\n      });\n      canvas.fire('text:changed', { target });\n      canvas.contextTopDirty = true;\n      canvas.requestRenderAll();\n    }\n  }\n\n  /**\n   * fired only on the drag source after drop (if occurred)\n   * handle changes to the drag source in case of a drop on another object or a cancellation\n   * https://developer.mozilla.org/en-US/docs/Web/API/HTML_Drag_and_Drop_API/Drag_operations#finishing_a_drag\n   */\n  dragEndHandler({ e }: DragEventData) {\n    if (this.isActive() && this.__dragStartFired) {\n      //  once the drop event finishes we check if we need to change the drag source\n      //  if the drag source received the drop we bail out since the drop handler has already handled logic\n      if (this.__dragStartSelection) {\n        const target = this.target;\n        const canvas = this.target.canvas!;\n        const { selectionStart, selectionEnd } = this.__dragStartSelection;\n        const dropEffect = e.dataTransfer?.dropEffect || NONE;\n        if (dropEffect === NONE) {\n          // pointer is back over selection\n          target.selectionStart = selectionStart;\n          target.selectionEnd = selectionEnd;\n          target._updateTextarea();\n          target.hiddenTextarea!.focus();\n        } else {\n          target.clearContextTop();\n          if (dropEffect === 'move') {\n            target.removeChars(selectionStart, selectionEnd);\n            target.selectionStart = target.selectionEnd = selectionStart;\n            target.hiddenTextarea &&\n              (target.hiddenTextarea.value = target.text);\n            target._updateTextarea();\n            target.fire('changed', {\n              index: selectionStart,\n              action: 'dragend',\n            });\n            canvas.fire('text:changed', { target });\n            canvas.requestRenderAll();\n          }\n          target.exitEditing();\n        }\n      }\n    }\n\n    this.__dragImageDisposer && this.__dragImageDisposer();\n    delete this.__dragImageDisposer;\n    delete this.__dragStartSelection;\n    this.__isDraggingOver = false;\n  }\n\n  dispose() {\n    this._dispose && this._dispose();\n  }\n}\n","import type {\n  ObjectEvents,\n  TPointerEvent,\n  TPointerEventInfo,\n} from '../../EventTypeDefs';\nimport { Point } from '../../Point';\nimport type { FabricObject } from '../Object/FabricObject';\nimport { FabricText } from '../Text/Text';\nimport { animate } from '../../util/animation/animate';\nimport type { TOnAnimationChangeCallback } from '../../util/animation/types';\nimport type { ValueAnimation } from '../../util/animation/ValueAnimation';\nimport type { TextStyleDeclaration } from '../Text/StyledText';\nimport type { SerializedTextProps, TextProps } from '../Text/Text';\nimport type { TOptions } from '../../typedefs';\nimport { getDocumentFromElement } from '../../util/dom_misc';\nimport { LEFT, RIGHT, reNewline } from '../../constants';\nimport type { IText } from './IText';\n\n/**\n *  extend this regex to support non english languages\n *\n *  - ` `      Matches a SPACE character (char code 32).\n *  - `\\n`     Matches a LINE FEED character (char code 10).\n *  - `\\.`     Matches a \".\" character (char code 46).\n *  - `,`      Matches a \",\" character (char code 44).\n *  - `;`      Matches a \";\" character (char code 59).\n *  - `!`      Matches a \"!\" character (char code 33).\n *  - `\\?`     Matches a \"?\" character (char code 63).\n *  - `\\-`     Matches a \"-\" character (char code 45).\n */\n// eslint-disable-next-line no-useless-escape\nconst reNonWord = /[ \\n\\.,;!\\?\\-]/;\n\nexport type ITextEvents = ObjectEvents & {\n  'selection:changed': never;\n  changed: never | { index: number; action: string };\n  tripleclick: TPointerEventInfo;\n  'editing:entered': never | { e: TPointerEvent };\n  'editing:exited': never;\n};\n\nexport abstract class ITextBehavior<\n  Props extends TOptions<TextProps> = Partial<TextProps>,\n  SProps extends SerializedTextProps = SerializedTextProps,\n  EventSpec extends ITextEvents = ITextEvents\n> extends FabricText<Props, SProps, EventSpec> {\n  declare abstract isEditing: boolean;\n  declare abstract cursorDelay: number;\n  declare abstract selectionStart: number;\n  declare abstract selectionEnd: number;\n  declare abstract cursorDuration: number;\n  declare abstract editable: boolean;\n  declare abstract editingBorderColor: string;\n\n  declare abstract compositionStart: number;\n  declare abstract compositionEnd: number;\n\n  declare abstract hiddenTextarea: HTMLTextAreaElement | null;\n\n  /**\n   * Helps determining when the text is in composition, so that the cursor\n   * rendering is altered.\n   */\n  protected declare inCompositionMode: boolean;\n\n  protected declare _reSpace: RegExp;\n  private declare _currentTickState?: ValueAnimation;\n  private declare _currentTickCompleteState?: ValueAnimation;\n  protected _currentCursorOpacity = 1;\n  private declare _textBeforeEdit: string;\n  protected declare __selectionStartOnMouseDown: number;\n\n  protected declare selected: boolean;\n  protected declare cursorOffsetCache: { left?: number; top?: number };\n  protected declare _savedProps?: {\n    hasControls: boolean;\n    borderColor: string;\n    lockMovementX: boolean;\n    lockMovementY: boolean;\n    selectable: boolean;\n    hoverCursor: CSSStyleDeclaration['cursor'] | null;\n    defaultCursor?: CSSStyleDeclaration['cursor'];\n    moveCursor?: CSSStyleDeclaration['cursor'];\n  };\n  protected declare _selectionDirection: 'left' | 'right' | null;\n\n  abstract initHiddenTextarea(): void;\n  abstract _fireSelectionChanged(): void;\n  abstract renderCursorOrSelection(): void;\n  abstract getSelectionStartFromPointer(e: TPointerEvent): number;\n  abstract _getCursorBoundaries(\n    index: number,\n    skipCaching?: boolean\n  ): {\n    left: number;\n    top: number;\n    leftOffset: number;\n    topOffset: number;\n  };\n\n  /**\n   * Initializes all the interactive behavior of IText\n   */\n  initBehavior() {\n    this._tick = this._tick.bind(this);\n    this._onTickComplete = this._onTickComplete.bind(this);\n    this.updateSelectionOnMouseMove =\n      this.updateSelectionOnMouseMove.bind(this);\n  }\n\n  onDeselect(options?: { e?: TPointerEvent; object?: FabricObject }) {\n    this.isEditing && this.exitEditing();\n    this.selected = false;\n    return super.onDeselect(options);\n  }\n\n  /**\n   * @private\n   */\n  _animateCursor({\n    toValue,\n    duration,\n    delay,\n    onComplete,\n  }: {\n    toValue: number;\n    duration: number;\n    delay?: number;\n    onComplete?: TOnAnimationChangeCallback<number, void>;\n  }) {\n    return animate({\n      startValue: this._currentCursorOpacity,\n      endValue: toValue,\n      duration,\n      delay,\n      onComplete,\n      abort: () =>\n        !this.canvas ||\n        // we do not want to animate a selection, only cursor\n        this.selectionStart !== this.selectionEnd,\n      onChange: (value) => {\n        this._currentCursorOpacity = value;\n        this.renderCursorOrSelection();\n      },\n    });\n  }\n\n  /**\n   * changes the cursor from visible to invisible\n   */\n  private _tick(delay?: number) {\n    this._currentTickState = this._animateCursor({\n      toValue: 0,\n      duration: this.cursorDuration / 2,\n      delay: Math.max(delay || 0, 100),\n      onComplete: this._onTickComplete,\n    });\n  }\n\n  /**\n   * Changes the cursor from invisible to visible\n   */\n  private _onTickComplete() {\n    this._currentTickCompleteState?.abort();\n    this._currentTickCompleteState = this._animateCursor({\n      toValue: 1,\n      duration: this.cursorDuration,\n      onComplete: this._tick,\n    });\n  }\n\n  /**\n   * Initializes delayed cursor\n   */\n  initDelayedCursor(restart?: boolean) {\n    this.abortCursorAnimation();\n    this._tick(restart ? 0 : this.cursorDelay);\n  }\n\n  /**\n   * Aborts cursor animation, clears all timeouts and clear textarea context if necessary\n   */\n  abortCursorAnimation() {\n    let shouldClear = false;\n    [this._currentTickState, this._currentTickCompleteState].forEach(\n      (cursorAnimation) => {\n        if (cursorAnimation && !cursorAnimation.isDone()) {\n          shouldClear = true;\n          cursorAnimation.abort();\n        }\n      }\n    );\n\n    this._currentCursorOpacity = 1;\n\n    //  make sure we clear context even if instance is not editing\n    if (shouldClear) {\n      this.clearContextTop();\n    }\n  }\n\n  /**\n   * Restart tue cursor animation if either is in complete state ( between animations )\n   * or if it never started before\n   */\n  restartCursorIfNeeded() {\n    if (\n      [this._currentTickState, this._currentTickCompleteState].some(\n        (cursorAnimation) => !cursorAnimation || cursorAnimation.isDone()\n      )\n    ) {\n      this.initDelayedCursor();\n    }\n  }\n\n  /**\n   * Selects entire text\n   */\n  selectAll() {\n    this.selectionStart = 0;\n    this.selectionEnd = this._text.length;\n    this._fireSelectionChanged();\n    this._updateTextarea();\n    return this;\n  }\n\n  /**\n   * Returns selected text\n   * @return {String}\n   */\n  getSelectedText(): string {\n    return this._text.slice(this.selectionStart, this.selectionEnd).join('');\n  }\n\n  /**\n   * Find new selection index representing start of current word according to current selection index\n   * @param {Number} startFrom Current selection index\n   * @return {Number} New selection index\n   */\n  findWordBoundaryLeft(startFrom: number): number {\n    let offset = 0,\n      index = startFrom - 1;\n\n    // remove space before cursor first\n    if (this._reSpace.test(this._text[index])) {\n      while (this._reSpace.test(this._text[index])) {\n        offset++;\n        index--;\n      }\n    }\n    while (/\\S/.test(this._text[index]) && index > -1) {\n      offset++;\n      index--;\n    }\n\n    return startFrom - offset;\n  }\n\n  /**\n   * Find new selection index representing end of current word according to current selection index\n   * @param {Number} startFrom Current selection index\n   * @return {Number} New selection index\n   */\n  findWordBoundaryRight(startFrom: number): number {\n    let offset = 0,\n      index = startFrom;\n\n    // remove space after cursor first\n    if (this._reSpace.test(this._text[index])) {\n      while (this._reSpace.test(this._text[index])) {\n        offset++;\n        index++;\n      }\n    }\n    while (/\\S/.test(this._text[index]) && index < this._text.length) {\n      offset++;\n      index++;\n    }\n\n    return startFrom + offset;\n  }\n\n  /**\n   * Find new selection index representing start of current line according to current selection index\n   * @param {Number} startFrom Current selection index\n   * @return {Number} New selection index\n   */\n  findLineBoundaryLeft(startFrom: number): number {\n    let offset = 0,\n      index = startFrom - 1;\n\n    while (!/\\n/.test(this._text[index]) && index > -1) {\n      offset++;\n      index--;\n    }\n\n    return startFrom - offset;\n  }\n\n  /**\n   * Find new selection index representing end of current line according to current selection index\n   * @param {Number} startFrom Current selection index\n   * @return {Number} New selection index\n   */\n  findLineBoundaryRight(startFrom: number): number {\n    let offset = 0,\n      index = startFrom;\n\n    while (!/\\n/.test(this._text[index]) && index < this._text.length) {\n      offset++;\n      index++;\n    }\n\n    return startFrom + offset;\n  }\n\n  /**\n   * Finds index corresponding to beginning or end of a word\n   * @param {Number} selectionStart Index of a character\n   * @param {Number} direction 1 or -1\n   * @return {Number} Index of the beginning or end of a word\n   */\n  searchWordBoundary(selectionStart: number, direction: 1 | -1): number {\n    const text = this._text;\n    // if we land on a space we move the cursor backwards\n    // if we are searching boundary end we move the cursor backwards ONLY if we don't land on a line break\n    let index =\n        selectionStart > 0 &&\n        this._reSpace.test(text[selectionStart]) &&\n        (direction === -1 || !reNewline.test(text[selectionStart - 1]))\n          ? selectionStart - 1\n          : selectionStart,\n      _char = text[index];\n    while (index > 0 && index < text.length && !reNonWord.test(_char)) {\n      index += direction;\n      _char = text[index];\n    }\n    if (direction === -1 && reNonWord.test(_char)) {\n      index++;\n    }\n    return index;\n  }\n\n  /**\n   * TODO fix: selectionStart set as 0 will be ignored?\n   * Selects a word based on the index\n   * @param {Number} selectionStart Index of a character\n   */\n  selectWord(selectionStart?: number) {\n    selectionStart = selectionStart || this.selectionStart;\n    // search backwards\n    const newSelectionStart = this.searchWordBoundary(selectionStart, -1),\n      // search forward\n      newSelectionEnd = Math.max(\n        newSelectionStart,\n        this.searchWordBoundary(selectionStart, 1)\n      );\n\n    this.selectionStart = newSelectionStart;\n    this.selectionEnd = newSelectionEnd;\n    this._fireSelectionChanged();\n    this._updateTextarea();\n    this.renderCursorOrSelection();\n  }\n\n  /**\n   * TODO fix: selectionStart set as 0 will be ignored?\n   * Selects a line based on the index\n   * @param {Number} selectionStart Index of a character\n   */\n  selectLine(selectionStart?: number) {\n    selectionStart = selectionStart || this.selectionStart;\n    const newSelectionStart = this.findLineBoundaryLeft(selectionStart),\n      newSelectionEnd = this.findLineBoundaryRight(selectionStart);\n\n    this.selectionStart = newSelectionStart;\n    this.selectionEnd = newSelectionEnd;\n    this._fireSelectionChanged();\n    this._updateTextarea();\n    return this;\n  }\n\n  /**\n   * Enters editing state\n   */\n  enterEditing(e?: TPointerEvent) {\n    if (this.isEditing || !this.editable) {\n      return;\n    }\n    if (this.canvas) {\n      this.canvas.calcOffset();\n      this.canvas.textEditingManager.exitTextEditing();\n    }\n\n    this.isEditing = true;\n\n    this.initHiddenTextarea();\n    this.hiddenTextarea!.focus();\n    this.hiddenTextarea!.value = this.text;\n    this._updateTextarea();\n    this._saveEditingProps();\n    this._setEditingProps();\n    this._textBeforeEdit = this.text;\n\n    this._tick();\n    this.fire('editing:entered', e ? { e } : undefined);\n    this._fireSelectionChanged();\n    if (this.canvas) {\n      // @ts-expect-error in reality it is an IText instance\n      this.canvas.fire('text:editing:entered', { target: this, e });\n      this.canvas.requestRenderAll();\n    }\n  }\n\n  /**\n   * called by {@link Canvas#textEditingManager}\n   */\n  updateSelectionOnMouseMove(e: TPointerEvent) {\n    if (this.getActiveControl()) {\n      return;\n    }\n\n    const el = this.hiddenTextarea!;\n    // regain focus\n    getDocumentFromElement(el).activeElement !== el && el.focus();\n\n    const newSelectionStart = this.getSelectionStartFromPointer(e),\n      currentStart = this.selectionStart,\n      currentEnd = this.selectionEnd;\n    if (\n      (newSelectionStart !== this.__selectionStartOnMouseDown ||\n        currentStart === currentEnd) &&\n      (currentStart === newSelectionStart || currentEnd === newSelectionStart)\n    ) {\n      return;\n    }\n    if (newSelectionStart > this.__selectionStartOnMouseDown) {\n      this.selectionStart = this.__selectionStartOnMouseDown;\n      this.selectionEnd = newSelectionStart;\n    } else {\n      this.selectionStart = newSelectionStart;\n      this.selectionEnd = this.__selectionStartOnMouseDown;\n    }\n    if (\n      this.selectionStart !== currentStart ||\n      this.selectionEnd !== currentEnd\n    ) {\n      this._fireSelectionChanged();\n      this._updateTextarea();\n      this.renderCursorOrSelection();\n    }\n  }\n\n  /**\n   * @private\n   */\n  _setEditingProps() {\n    this.hoverCursor = 'text';\n\n    if (this.canvas) {\n      this.canvas.defaultCursor = this.canvas.moveCursor = 'text';\n    }\n\n    this.borderColor = this.editingBorderColor;\n    this.hasControls = this.selectable = false;\n    this.lockMovementX = this.lockMovementY = true;\n  }\n\n  /**\n   * convert from textarea to grapheme indexes\n   */\n  fromStringToGraphemeSelection(start: number, end: number, text: string) {\n    const smallerTextStart = text.slice(0, start),\n      graphemeStart = this.graphemeSplit(smallerTextStart).length;\n    if (start === end) {\n      return { selectionStart: graphemeStart, selectionEnd: graphemeStart };\n    }\n    const smallerTextEnd = text.slice(start, end),\n      graphemeEnd = this.graphemeSplit(smallerTextEnd).length;\n    return {\n      selectionStart: graphemeStart,\n      selectionEnd: graphemeStart + graphemeEnd,\n    };\n  }\n\n  /**\n   * convert from fabric to textarea values\n   */\n  fromGraphemeToStringSelection(\n    start: number,\n    end: number,\n    graphemes: string[]\n  ) {\n    const smallerTextStart = graphemes.slice(0, start),\n      graphemeStart = smallerTextStart.join('').length;\n    if (start === end) {\n      return { selectionStart: graphemeStart, selectionEnd: graphemeStart };\n    }\n    const smallerTextEnd = graphemes.slice(start, end),\n      graphemeEnd = smallerTextEnd.join('').length;\n    return {\n      selectionStart: graphemeStart,\n      selectionEnd: graphemeStart + graphemeEnd,\n    };\n  }\n\n  /**\n   * @private\n   */\n  _updateTextarea() {\n    this.cursorOffsetCache = {};\n    if (!this.hiddenTextarea) {\n      return;\n    }\n    if (!this.inCompositionMode) {\n      const newSelection = this.fromGraphemeToStringSelection(\n        this.selectionStart,\n        this.selectionEnd,\n        this._text\n      );\n      this.hiddenTextarea.selectionStart = newSelection.selectionStart;\n      this.hiddenTextarea.selectionEnd = newSelection.selectionEnd;\n    }\n    this.updateTextareaPosition();\n  }\n\n  /**\n   * @private\n   */\n  updateFromTextArea() {\n    if (!this.hiddenTextarea) {\n      return;\n    }\n    this.cursorOffsetCache = {};\n    const textarea = this.hiddenTextarea;\n    this.text = textarea.value;\n    this.set('dirty', true);\n    this.initDimensions();\n    this.setCoords();\n    const newSelection = this.fromStringToGraphemeSelection(\n      textarea.selectionStart,\n      textarea.selectionEnd,\n      textarea.value\n    );\n    this.selectionEnd = this.selectionStart = newSelection.selectionEnd;\n    if (!this.inCompositionMode) {\n      this.selectionStart = newSelection.selectionStart;\n    }\n    this.updateTextareaPosition();\n  }\n\n  /**\n   * @private\n   */\n  updateTextareaPosition() {\n    if (this.selectionStart === this.selectionEnd) {\n      const style = this._calcTextareaPosition();\n      this.hiddenTextarea!.style.left = style.left;\n      this.hiddenTextarea!.style.top = style.top;\n    }\n  }\n\n  /**\n   * @private\n   * @return {Object} style contains style for hiddenTextarea\n   */\n  _calcTextareaPosition() {\n    if (!this.canvas) {\n      return { left: '1px', top: '1px' };\n    }\n    const desiredPosition = this.inCompositionMode\n        ? this.compositionStart\n        : this.selectionStart,\n      boundaries = this._getCursorBoundaries(desiredPosition),\n      cursorLocation = this.get2DCursorLocation(desiredPosition),\n      lineIndex = cursorLocation.lineIndex,\n      charIndex = cursorLocation.charIndex,\n      charHeight =\n        this.getValueOfPropertyAt(lineIndex, charIndex, 'fontSize') *\n        this.lineHeight,\n      leftOffset = boundaries.leftOffset,\n      retinaScaling = this.getCanvasRetinaScaling(),\n      upperCanvas = this.canvas.upperCanvasEl,\n      upperCanvasWidth = upperCanvas.width / retinaScaling,\n      upperCanvasHeight = upperCanvas.height / retinaScaling,\n      maxWidth = upperCanvasWidth - charHeight,\n      maxHeight = upperCanvasHeight - charHeight;\n\n    const p = new Point(\n      boundaries.left + leftOffset,\n      boundaries.top + boundaries.topOffset + charHeight\n    )\n      .transform(this.calcTransformMatrix())\n      .transform(this.canvas.viewportTransform)\n      .multiply(\n        new Point(\n          upperCanvas.clientWidth / upperCanvasWidth,\n          upperCanvas.clientHeight / upperCanvasHeight\n        )\n      );\n\n    if (p.x < 0) {\n      p.x = 0;\n    }\n    if (p.x > maxWidth) {\n      p.x = maxWidth;\n    }\n    if (p.y < 0) {\n      p.y = 0;\n    }\n    if (p.y > maxHeight) {\n      p.y = maxHeight;\n    }\n\n    // add canvas offset on document\n    p.x += this.canvas._offset.left;\n    p.y += this.canvas._offset.top;\n\n    return {\n      left: `${p.x}px`,\n      top: `${p.y}px`,\n      fontSize: `${charHeight}px`,\n      charHeight: charHeight,\n    };\n  }\n\n  /**\n   * @private\n   */\n  _saveEditingProps() {\n    this._savedProps = {\n      hasControls: this.hasControls,\n      borderColor: this.borderColor,\n      lockMovementX: this.lockMovementX,\n      lockMovementY: this.lockMovementY,\n      hoverCursor: this.hoverCursor,\n      selectable: this.selectable,\n      defaultCursor: this.canvas && this.canvas.defaultCursor,\n      moveCursor: this.canvas && this.canvas.moveCursor,\n    };\n  }\n\n  /**\n   * @private\n   */\n  _restoreEditingProps() {\n    if (!this._savedProps) {\n      return;\n    }\n\n    this.hoverCursor = this._savedProps.hoverCursor;\n    this.hasControls = this._savedProps.hasControls;\n    this.borderColor = this._savedProps.borderColor;\n    this.selectable = this._savedProps.selectable;\n    this.lockMovementX = this._savedProps.lockMovementX;\n    this.lockMovementY = this._savedProps.lockMovementY;\n\n    if (this.canvas) {\n      this.canvas.defaultCursor =\n        this._savedProps.defaultCursor || this.canvas.defaultCursor;\n      this.canvas.moveCursor =\n        this._savedProps.moveCursor || this.canvas.moveCursor;\n    }\n\n    delete this._savedProps;\n  }\n\n  /**\n   * runs the actual logic that exits from editing state, see {@link exitEditing}\n   */\n  protected _exitEditing() {\n    const hiddenTextarea = this.hiddenTextarea;\n    this.selected = false;\n    this.isEditing = false;\n\n    if (hiddenTextarea) {\n      hiddenTextarea.blur && hiddenTextarea.blur();\n      hiddenTextarea.parentNode &&\n        hiddenTextarea.parentNode.removeChild(hiddenTextarea);\n    }\n    this.hiddenTextarea = null;\n    this.abortCursorAnimation();\n    this.selectionStart !== this.selectionEnd && this.clearContextTop();\n  }\n\n  /**\n   * Exits from editing state and fires relevant events\n   */\n  exitEditing() {\n    const isTextChanged = this._textBeforeEdit !== this.text;\n    this._exitEditing();\n    this.selectionEnd = this.selectionStart;\n    this._restoreEditingProps();\n    if (this._forceClearCache) {\n      this.initDimensions();\n      this.setCoords();\n    }\n    this.fire('editing:exited');\n    isTextChanged && this.fire('modified');\n    if (this.canvas) {\n      this.canvas.fire('text:editing:exited', {\n        target: this as unknown as IText,\n      });\n      isTextChanged && this.canvas.fire('object:modified', { target: this });\n    }\n    return this;\n  }\n\n  /**\n   * @private\n   */\n  _removeExtraneousStyles() {\n    for (const prop in this.styles) {\n      if (!this._textLines[prop as unknown as number]) {\n        delete this.styles[prop];\n      }\n    }\n  }\n\n  /**\n   * remove and reflow a style block from start to end.\n   * @param {Number} start linear start position for removal (included in removal)\n   * @param {Number} end linear end position for removal ( excluded from removal )\n   */\n  removeStyleFromTo(start: number, end: number) {\n    const { lineIndex: lineStart, charIndex: charStart } =\n        this.get2DCursorLocation(start, true),\n      { lineIndex: lineEnd, charIndex: charEnd } = this.get2DCursorLocation(\n        end,\n        true\n      );\n    if (lineStart !== lineEnd) {\n      // step1 remove the trailing of lineStart\n      if (this.styles[lineStart]) {\n        for (\n          let i = charStart;\n          i < this._unwrappedTextLines[lineStart].length;\n          i++\n        ) {\n          delete this.styles[lineStart][i];\n        }\n      }\n      // step2 move the trailing of lineEnd to lineStart if needed\n      if (this.styles[lineEnd]) {\n        for (\n          let i = charEnd;\n          i < this._unwrappedTextLines[lineEnd].length;\n          i++\n        ) {\n          const styleObj = this.styles[lineEnd][i];\n          if (styleObj) {\n            this.styles[lineStart] || (this.styles[lineStart] = {});\n            this.styles[lineStart][charStart + i - charEnd] = styleObj;\n          }\n        }\n      }\n      // step3 detects lines will be completely removed.\n      for (let i = lineStart + 1; i <= lineEnd; i++) {\n        delete this.styles[i];\n      }\n      // step4 shift remaining lines.\n      this.shiftLineStyles(lineEnd, lineStart - lineEnd);\n    } else {\n      // remove and shift left on the same line\n      if (this.styles[lineStart]) {\n        const styleObj = this.styles[lineStart];\n        const diff = charEnd - charStart;\n        for (let i = charStart; i < charEnd; i++) {\n          delete styleObj[i];\n        }\n        for (const char in this.styles[lineStart]) {\n          const numericChar = parseInt(char, 10);\n          if (numericChar >= charEnd) {\n            styleObj[numericChar - diff] = styleObj[char];\n            delete styleObj[char];\n          }\n        }\n      }\n    }\n  }\n\n  /**\n   * Shifts line styles up or down\n   * @param {Number} lineIndex Index of a line\n   * @param {Number} offset Can any number?\n   */\n  shiftLineStyles(lineIndex: number, offset: number) {\n    const clonedStyles = Object.assign({}, this.styles);\n    for (const line in this.styles) {\n      const numericLine = parseInt(line, 10);\n      if (numericLine > lineIndex) {\n        this.styles[numericLine + offset] = clonedStyles[numericLine];\n        if (!clonedStyles[numericLine - offset]) {\n          delete this.styles[numericLine];\n        }\n      }\n    }\n  }\n\n  /**\n   * Handle insertion of more consecutive style lines for when one or more\n   * newlines gets added to the text. Since current style needs to be shifted\n   * first we shift the current style of the number lines needed, then we add\n   * new lines from the last to the first.\n   * @param {Number} lineIndex Index of a line\n   * @param {Number} charIndex Index of a char\n   * @param {Number} qty number of lines to add\n   * @param {Array} copiedStyle Array of objects styles\n   */\n  insertNewlineStyleObject(\n    lineIndex: number,\n    charIndex: number,\n    qty: number,\n    copiedStyle?: { [index: number]: TextStyleDeclaration }\n  ) {\n    const newLineStyles: { [index: number]: TextStyleDeclaration } = {};\n    const originalLineLength = this._unwrappedTextLines[lineIndex].length;\n    const isEndOfLine = originalLineLength === charIndex;\n\n    let someStyleIsCarryingOver = false;\n    qty || (qty = 1);\n    this.shiftLineStyles(lineIndex, qty);\n    const currentCharStyle = this.styles[lineIndex]\n      ? this.styles[lineIndex][charIndex === 0 ? charIndex : charIndex - 1]\n      : undefined;\n\n    // we clone styles of all chars\n    // after cursor onto the current line\n    for (const index in this.styles[lineIndex]) {\n      const numIndex = parseInt(index, 10);\n      if (numIndex >= charIndex) {\n        someStyleIsCarryingOver = true;\n        newLineStyles[numIndex - charIndex] = this.styles[lineIndex][index];\n        // remove lines from the previous line since they're on a new line now\n        if (!(isEndOfLine && charIndex === 0)) {\n          delete this.styles[lineIndex][index];\n        }\n      }\n    }\n    let styleCarriedOver = false;\n    if (someStyleIsCarryingOver && !isEndOfLine) {\n      // if is end of line, the extra style we copied\n      // is probably not something we want\n      this.styles[lineIndex + qty] = newLineStyles;\n      styleCarriedOver = true;\n    }\n    if (styleCarriedOver || originalLineLength > charIndex) {\n      // skip the last line of since we already prepared it.\n      // or contains text without style that we don't want to style\n      // just because it changed lines\n      qty--;\n    }\n    // for the all the lines or all the other lines\n    // we clone current char style onto the next (otherwise empty) line\n    while (qty > 0) {\n      if (copiedStyle && copiedStyle[qty - 1]) {\n        this.styles[lineIndex + qty] = {\n          0: { ...copiedStyle[qty - 1] },\n        };\n      } else if (currentCharStyle) {\n        this.styles[lineIndex + qty] = {\n          0: { ...currentCharStyle },\n        };\n      } else {\n        delete this.styles[lineIndex + qty];\n      }\n      qty--;\n    }\n    this._forceClearCache = true;\n  }\n\n  /**\n   * Inserts style object for a given line/char index\n   * @param {Number} lineIndex Index of a line\n   * @param {Number} charIndex Index of a char\n   * @param {Number} quantity number Style object to insert, if given\n   * @param {Array} copiedStyle array of style objects\n   */\n  insertCharStyleObject(\n    lineIndex: number,\n    charIndex: number,\n    quantity: number,\n    copiedStyle?: TextStyleDeclaration[]\n  ) {\n    if (!this.styles) {\n      this.styles = {};\n    }\n    const currentLineStyles = this.styles[lineIndex],\n      currentLineStylesCloned = currentLineStyles\n        ? { ...currentLineStyles }\n        : {};\n\n    quantity || (quantity = 1);\n    // shift all char styles by quantity forward\n    // 0,1,2,3 -> (charIndex=2) -> 0,1,3,4 -> (insert 2) -> 0,1,2,3,4\n    for (const index in currentLineStylesCloned) {\n      const numericIndex = parseInt(index, 10);\n      if (numericIndex >= charIndex) {\n        currentLineStyles[numericIndex + quantity] =\n          currentLineStylesCloned[numericIndex];\n        // only delete the style if there was nothing moved there\n        if (!currentLineStylesCloned[numericIndex - quantity]) {\n          delete currentLineStyles[numericIndex];\n        }\n      }\n    }\n    this._forceClearCache = true;\n    if (copiedStyle) {\n      while (quantity--) {\n        if (!Object.keys(copiedStyle[quantity]).length) {\n          continue;\n        }\n        if (!this.styles[lineIndex]) {\n          this.styles[lineIndex] = {};\n        }\n        this.styles[lineIndex][charIndex + quantity] = {\n          ...copiedStyle[quantity],\n        };\n      }\n      return;\n    }\n    if (!currentLineStyles) {\n      return;\n    }\n    const newStyle = currentLineStyles[charIndex ? charIndex - 1 : 1];\n    while (newStyle && quantity--) {\n      this.styles[lineIndex][charIndex + quantity] = { ...newStyle };\n    }\n  }\n\n  /**\n   * Inserts style object(s)\n   * @param {Array} insertedText Characters at the location where style is inserted\n   * @param {Number} start cursor index for inserting style\n   * @param {Array} [copiedStyle] array of style objects to insert.\n   */\n  insertNewStyleBlock(\n    insertedText: string[],\n    start: number,\n    copiedStyle?: TextStyleDeclaration[]\n  ) {\n    const cursorLoc = this.get2DCursorLocation(start, true),\n      addedLines = [0];\n    let linesLength = 0;\n    // get an array of how many char per lines are being added.\n    for (let i = 0; i < insertedText.length; i++) {\n      if (insertedText[i] === '\\n') {\n        linesLength++;\n        addedLines[linesLength] = 0;\n      } else {\n        addedLines[linesLength]++;\n      }\n    }\n    // for the first line copy the style from the current char position.\n    if (addedLines[0] > 0) {\n      this.insertCharStyleObject(\n        cursorLoc.lineIndex,\n        cursorLoc.charIndex,\n        addedLines[0],\n        copiedStyle\n      );\n      copiedStyle = copiedStyle && copiedStyle.slice(addedLines[0] + 1);\n    }\n    linesLength &&\n      this.insertNewlineStyleObject(\n        cursorLoc.lineIndex,\n        cursorLoc.charIndex + addedLines[0],\n        linesLength\n      );\n    let i;\n    for (i = 1; i < linesLength; i++) {\n      if (addedLines[i] > 0) {\n        this.insertCharStyleObject(\n          cursorLoc.lineIndex + i,\n          0,\n          addedLines[i],\n          copiedStyle\n        );\n      } else if (copiedStyle) {\n        // this test is required in order to close #6841\n        // when a pasted buffer begins with a newline then\n        // this.styles[cursorLoc.lineIndex + i] and copiedStyle[0]\n        // may be undefined for some reason\n        if (this.styles[cursorLoc.lineIndex + i] && copiedStyle[0]) {\n          this.styles[cursorLoc.lineIndex + i][0] = copiedStyle[0];\n        }\n      }\n      copiedStyle = copiedStyle && copiedStyle.slice(addedLines[i] + 1);\n    }\n    if (addedLines[i] > 0) {\n      this.insertCharStyleObject(\n        cursorLoc.lineIndex + i,\n        0,\n        addedLines[i],\n        copiedStyle\n      );\n    }\n  }\n\n  /**\n   * Removes characters from start/end\n   * start/end ar per grapheme position in _text array.\n   *\n   * @param {Number} start\n   * @param {Number} end default to start + 1\n   */\n  removeChars(start: number, end: number = start + 1) {\n    this.removeStyleFromTo(start, end);\n    this._text.splice(start, end - start);\n    this.text = this._text.join('');\n    this.set('dirty', true);\n    this.initDimensions();\n    this.setCoords();\n    this._removeExtraneousStyles();\n  }\n\n  /**\n   * insert characters at start position, before start position.\n   * start  equal 1 it means the text get inserted between actual grapheme 0 and 1\n   * if style array is provided, it must be as the same length of text in graphemes\n   * if end is provided and is bigger than start, old text is replaced.\n   * start/end ar per grapheme position in _text array.\n   *\n   * @param {String} text text to insert\n   * @param {Array} style array of style objects\n   * @param {Number} start\n   * @param {Number} end default to start + 1\n   */\n  insertChars(\n    text: string,\n    style: TextStyleDeclaration[] | undefined,\n    start: number,\n    end: number = start\n  ) {\n    if (end > start) {\n      this.removeStyleFromTo(start, end);\n    }\n    const graphemes = this.graphemeSplit(text);\n    this.insertNewStyleBlock(graphemes, start, style);\n    this._text = [\n      ...this._text.slice(0, start),\n      ...graphemes,\n      ...this._text.slice(end),\n    ];\n    this.text = this._text.join('');\n    this.set('dirty', true);\n    this.initDimensions();\n    this.setCoords();\n    this._removeExtraneousStyles();\n  }\n\n  /**\n   * Set the selectionStart and selectionEnd according to the new position of cursor\n   * mimic the key - mouse navigation when shift is pressed.\n   */\n  setSelectionStartEndWithShift(\n    start: number,\n    end: number,\n    newSelection: number\n  ) {\n    if (newSelection <= start) {\n      if (end === start) {\n        this._selectionDirection = LEFT;\n      } else if (this._selectionDirection === RIGHT) {\n        this._selectionDirection = LEFT;\n        this.selectionEnd = start;\n      }\n      this.selectionStart = newSelection;\n    } else if (newSelection > start && newSelection < end) {\n      if (this._selectionDirection === RIGHT) {\n        this.selectionEnd = newSelection;\n      } else {\n        this.selectionStart = newSelection;\n      }\n    } else {\n      // newSelection is > selection start and end\n      if (end === start) {\n        this._selectionDirection = RIGHT;\n      } else if (this._selectionDirection === LEFT) {\n        this._selectionDirection = RIGHT;\n        this.selectionStart = end;\n      }\n      this.selectionEnd = newSelection;\n    }\n  }\n}\n","import { config } from '../../config';\nimport { getFabricDocument, getEnv } from '../../env';\nimport { capValue } from '../../util/misc/capValue';\nimport type { ITextEvents } from './ITextBehavior';\nimport { ITextBehavior } from './ITextBehavior';\nimport type { TKeyMapIText } from './constants';\nimport type { TOptions } from '../../typedefs';\nimport type { TextProps, SerializedTextProps } from '../Text/Text';\nimport { getDocumentFromElement } from '../../util/dom_misc';\nimport { LEFT, RIGHT } from '../../constants';\nimport type { IText } from './IText';\nimport type { TextStyleDeclaration } from '../Text/StyledText';\n\nexport abstract class ITextKeyBehavior<\n  Props extends TOptions<TextProps> = Partial<TextProps>,\n  SProps extends SerializedTextProps = SerializedTextProps,\n  EventSpec extends ITextEvents = ITextEvents\n> extends ITextBehavior<Props, SProps, EventSpec> {\n  /**\n   * For functionalities on keyDown\n   * Map a special key to a function of the instance/prototype\n   * If you need different behavior for ESC or TAB or arrows, you have to change\n   * this map setting the name of a function that you build on the IText or\n   * your prototype.\n   * the map change will affect all Instances unless you need for only some text Instances\n   * in that case you have to clone this object and assign your Instance.\n   * this.keysMap = Object.assign({}, this.keysMap);\n   * The function must be in IText.prototype.myFunction And will receive event as args[0]\n   */\n  declare keysMap: TKeyMapIText;\n\n  declare keysMapRtl: TKeyMapIText;\n\n  /**\n   * For functionalities on keyUp + ctrl || cmd\n   */\n  declare ctrlKeysMapUp: TKeyMapIText;\n\n  /**\n   * For functionalities on keyDown + ctrl || cmd\n   */\n  declare ctrlKeysMapDown: TKeyMapIText;\n\n  declare hiddenTextarea: HTMLTextAreaElement | null;\n\n  /**\n   * DOM container to append the hiddenTextarea.\n   * An alternative to attaching to the document.body.\n   * Useful to reduce laggish redraw of the full document.body tree and\n   * also with modals event capturing that won't let the textarea take focus.\n   * @type HTMLElement\n   * @default\n   */\n  declare hiddenTextareaContainer?: HTMLElement | null;\n\n  private declare _clickHandlerInitialized: boolean;\n  private declare _copyDone: boolean;\n  private declare fromPaste: boolean;\n\n  /**\n   * Initializes hidden textarea (needed to bring up keyboard in iOS)\n   */\n  initHiddenTextarea() {\n    const doc =\n      (this.canvas && getDocumentFromElement(this.canvas.getElement())) ||\n      getFabricDocument();\n    const textarea = doc.createElement('textarea');\n    Object.entries({\n      autocapitalize: 'off',\n      autocorrect: 'off',\n      autocomplete: 'off',\n      spellcheck: 'false',\n      'data-fabric': 'textarea',\n      wrap: 'off',\n    }).map(([attribute, value]) => textarea.setAttribute(attribute, value));\n    const { top, left, fontSize } = this._calcTextareaPosition();\n    // line-height: 1px; was removed from the style to fix this:\n    // https://bugs.chromium.org/p/chromium/issues/detail?id=870966\n    textarea.style.cssText = `position: absolute; top: ${top}; left: ${left}; z-index: -999; opacity: 0; width: 1px; height: 1px; font-size: 1px; padding-top: ${fontSize};`;\n\n    (this.hiddenTextareaContainer || doc.body).appendChild(textarea);\n\n    Object.entries({\n      blur: 'blur',\n      keydown: 'onKeyDown',\n      keyup: 'onKeyUp',\n      input: 'onInput',\n      copy: 'copy',\n      cut: 'copy',\n      paste: 'paste',\n      compositionstart: 'onCompositionStart',\n      compositionupdate: 'onCompositionUpdate',\n      compositionend: 'onCompositionEnd',\n    } as Record<string, keyof this>).map(([eventName, handler]) =>\n      textarea.addEventListener(\n        eventName,\n        (this[handler] as Function).bind(this)\n      )\n    );\n    this.hiddenTextarea = textarea;\n  }\n\n  /**\n   * Override this method to customize cursor behavior on textbox blur\n   */\n  blur() {\n    this.abortCursorAnimation();\n  }\n\n  /**\n   * Handles keydown event\n   * only used for arrows and combination of modifier keys.\n   * @param {KeyboardEvent} e Event object\n   */\n  onKeyDown(e: KeyboardEvent) {\n    if (!this.isEditing) {\n      return;\n    }\n    const keyMap = this.direction === 'rtl' ? this.keysMapRtl : this.keysMap;\n    if (e.keyCode in keyMap) {\n      // @ts-expect-error legacy method calling pattern\n      this[keyMap[e.keyCode]](e);\n    } else if (e.keyCode in this.ctrlKeysMapDown && (e.ctrlKey || e.metaKey)) {\n      // @ts-expect-error legacy method calling pattern\n      this[this.ctrlKeysMapDown[e.keyCode]](e);\n    } else {\n      return;\n    }\n    e.stopImmediatePropagation();\n    e.preventDefault();\n    if (e.keyCode >= 33 && e.keyCode <= 40) {\n      // if i press an arrow key just update selection\n      this.inCompositionMode = false;\n      this.clearContextTop();\n      this.renderCursorOrSelection();\n    } else {\n      this.canvas && this.canvas.requestRenderAll();\n    }\n  }\n\n  /**\n   * Handles keyup event\n   * We handle KeyUp because ie11 and edge have difficulties copy/pasting\n   * if a copy/cut event fired, keyup is dismissed\n   * @param {KeyboardEvent} e Event object\n   */\n  onKeyUp(e: KeyboardEvent) {\n    if (!this.isEditing || this._copyDone || this.inCompositionMode) {\n      this._copyDone = false;\n      return;\n    }\n    if (e.keyCode in this.ctrlKeysMapUp && (e.ctrlKey || e.metaKey)) {\n      // @ts-expect-error legacy method calling pattern\n      this[this.ctrlKeysMapUp[e.keyCode]](e);\n    } else {\n      return;\n    }\n    e.stopImmediatePropagation();\n    e.preventDefault();\n    this.canvas && this.canvas.requestRenderAll();\n  }\n\n  /**\n   * Handles onInput event\n   * @param {Event} e Event object\n   */\n  onInput(this: this & { hiddenTextarea: HTMLTextAreaElement }, e: Event) {\n    const fromPaste = this.fromPaste;\n    this.fromPaste = false;\n    e && e.stopPropagation();\n    if (!this.isEditing) {\n      return;\n    }\n    const updateAndFire = () => {\n      this.updateFromTextArea();\n      this.fire('changed');\n      if (this.canvas) {\n        this.canvas.fire('text:changed', { target: this as unknown as IText });\n        this.canvas.requestRenderAll();\n      }\n    };\n    if (this.hiddenTextarea.value === '') {\n      this.styles = {};\n      updateAndFire();\n      return;\n    }\n    // decisions about style changes.\n    const nextText = this._splitTextIntoLines(\n        this.hiddenTextarea.value\n      ).graphemeText,\n      charCount = this._text.length,\n      nextCharCount = nextText.length,\n      selectionStart = this.selectionStart,\n      selectionEnd = this.selectionEnd,\n      selection = selectionStart !== selectionEnd;\n    let copiedStyle: TextStyleDeclaration[] | undefined,\n      removedText,\n      charDiff = nextCharCount - charCount,\n      removeFrom,\n      removeTo;\n\n    const textareaSelection = this.fromStringToGraphemeSelection(\n      this.hiddenTextarea.selectionStart,\n      this.hiddenTextarea.selectionEnd,\n      this.hiddenTextarea.value\n    );\n    const backDelete = selectionStart > textareaSelection.selectionStart;\n\n    if (selection) {\n      removedText = this._text.slice(selectionStart, selectionEnd);\n      charDiff += selectionEnd - selectionStart;\n    } else if (nextCharCount < charCount) {\n      if (backDelete) {\n        removedText = this._text.slice(selectionEnd + charDiff, selectionEnd);\n      } else {\n        removedText = this._text.slice(\n          selectionStart,\n          selectionStart - charDiff\n        );\n      }\n    }\n    const insertedText = nextText.slice(\n      textareaSelection.selectionEnd - charDiff,\n      textareaSelection.selectionEnd\n    );\n    if (removedText && removedText.length) {\n      if (insertedText.length) {\n        // let's copy some style before deleting.\n        // we want to copy the style before the cursor OR the style at the cursor if selection\n        // is bigger than 0.\n        copiedStyle = this.getSelectionStyles(\n          selectionStart,\n          selectionStart + 1,\n          false\n        );\n        // now duplicate the style one for each inserted text.\n        copiedStyle = insertedText.map(\n          () =>\n            // this return an array of references, but that is fine since we are\n            // copying the style later.\n            copiedStyle![0]\n        );\n      }\n      if (selection) {\n        removeFrom = selectionStart;\n        removeTo = selectionEnd;\n      } else if (backDelete) {\n        // detect differences between forwardDelete and backDelete\n        removeFrom = selectionEnd - removedText.length;\n        removeTo = selectionEnd;\n      } else {\n        removeFrom = selectionEnd;\n        removeTo = selectionEnd + removedText.length;\n      }\n      this.removeStyleFromTo(removeFrom, removeTo);\n    }\n    if (insertedText.length) {\n      const { copyPasteData } = getEnv();\n      if (\n        fromPaste &&\n        insertedText.join('') === copyPasteData.copiedText &&\n        !config.disableStyleCopyPaste\n      ) {\n        copiedStyle = copyPasteData.copiedTextStyle;\n      }\n      this.insertNewStyleBlock(insertedText, selectionStart, copiedStyle);\n    }\n    updateAndFire();\n  }\n\n  /**\n   * Composition start\n   */\n  onCompositionStart() {\n    this.inCompositionMode = true;\n  }\n\n  /**\n   * Composition end\n   */\n  onCompositionEnd() {\n    this.inCompositionMode = false;\n  }\n\n  onCompositionUpdate({ target }: CompositionEvent) {\n    const { selectionStart, selectionEnd } = target as HTMLTextAreaElement;\n    this.compositionStart = selectionStart;\n    this.compositionEnd = selectionEnd;\n    this.updateTextareaPosition();\n  }\n\n  /**\n   * Copies selected text\n   */\n  copy() {\n    if (this.selectionStart === this.selectionEnd) {\n      //do not cut-copy if no selection\n      return;\n    }\n    const { copyPasteData } = getEnv();\n    copyPasteData.copiedText = this.getSelectedText();\n    if (!config.disableStyleCopyPaste) {\n      copyPasteData.copiedTextStyle = this.getSelectionStyles(\n        this.selectionStart,\n        this.selectionEnd,\n        true\n      );\n    } else {\n      copyPasteData.copiedTextStyle = undefined;\n    }\n    this._copyDone = true;\n  }\n\n  /**\n   * Pastes text\n   */\n  paste() {\n    this.fromPaste = true;\n  }\n\n  /**\n   * Finds the width in pixels before the cursor on the same line\n   * @private\n   * @param {Number} lineIndex\n   * @param {Number} charIndex\n   * @return {Number} widthBeforeCursor width before cursor\n   */\n  _getWidthBeforeCursor(lineIndex: number, charIndex: number): number {\n    let widthBeforeCursor = this._getLineLeftOffset(lineIndex),\n      bound;\n\n    if (charIndex > 0) {\n      bound = this.__charBounds[lineIndex][charIndex - 1];\n      widthBeforeCursor += bound.left + bound.width;\n    }\n    return widthBeforeCursor;\n  }\n\n  /**\n   * Gets start offset of a selection\n   * @param {KeyboardEvent} e Event object\n   * @param {Boolean} isRight\n   * @return {Number}\n   */\n  getDownCursorOffset(e: KeyboardEvent, isRight: boolean): number {\n    const selectionProp = this._getSelectionForOffset(e, isRight),\n      cursorLocation = this.get2DCursorLocation(selectionProp),\n      lineIndex = cursorLocation.lineIndex;\n    // if on last line, down cursor goes to end of line\n    if (\n      lineIndex === this._textLines.length - 1 ||\n      e.metaKey ||\n      e.keyCode === 34\n    ) {\n      // move to the end of a text\n      return this._text.length - selectionProp;\n    }\n    const charIndex = cursorLocation.charIndex,\n      widthBeforeCursor = this._getWidthBeforeCursor(lineIndex, charIndex),\n      indexOnOtherLine = this._getIndexOnLine(lineIndex + 1, widthBeforeCursor),\n      textAfterCursor = this._textLines[lineIndex].slice(charIndex);\n    return (\n      textAfterCursor.length +\n      indexOnOtherLine +\n      1 +\n      this.missingNewlineOffset(lineIndex)\n    );\n  }\n\n  /**\n   * private\n   * Helps finding if the offset should be counted from Start or End\n   * @param {KeyboardEvent} e Event object\n   * @param {Boolean} isRight\n   * @return {Number}\n   */\n  _getSelectionForOffset(e: KeyboardEvent, isRight: boolean): number {\n    if (e.shiftKey && this.selectionStart !== this.selectionEnd && isRight) {\n      return this.selectionEnd;\n    } else {\n      return this.selectionStart;\n    }\n  }\n\n  /**\n   * @param {KeyboardEvent} e Event object\n   * @param {Boolean} isRight\n   * @return {Number}\n   */\n  getUpCursorOffset(e: KeyboardEvent, isRight: boolean): number {\n    const selectionProp = this._getSelectionForOffset(e, isRight),\n      cursorLocation = this.get2DCursorLocation(selectionProp),\n      lineIndex = cursorLocation.lineIndex;\n    if (lineIndex === 0 || e.metaKey || e.keyCode === 33) {\n      // if on first line, up cursor goes to start of line\n      return -selectionProp;\n    }\n    const charIndex = cursorLocation.charIndex,\n      widthBeforeCursor = this._getWidthBeforeCursor(lineIndex, charIndex),\n      indexOnOtherLine = this._getIndexOnLine(lineIndex - 1, widthBeforeCursor),\n      textBeforeCursor = this._textLines[lineIndex].slice(0, charIndex),\n      missingNewlineOffset = this.missingNewlineOffset(lineIndex - 1);\n    // return a negative offset\n    return (\n      -this._textLines[lineIndex - 1].length +\n      indexOnOtherLine -\n      textBeforeCursor.length +\n      (1 - missingNewlineOffset)\n    );\n  }\n\n  /**\n   * for a given width it founds the matching character.\n   * @private\n   */\n  _getIndexOnLine(lineIndex: number, width: number) {\n    const line = this._textLines[lineIndex],\n      lineLeftOffset = this._getLineLeftOffset(lineIndex);\n    let widthOfCharsOnLine = lineLeftOffset,\n      indexOnLine = 0,\n      charWidth,\n      foundMatch;\n\n    for (let j = 0, jlen = line.length; j < jlen; j++) {\n      charWidth = this.__charBounds[lineIndex][j].width;\n      widthOfCharsOnLine += charWidth;\n      if (widthOfCharsOnLine > width) {\n        foundMatch = true;\n        const leftEdge = widthOfCharsOnLine - charWidth,\n          rightEdge = widthOfCharsOnLine,\n          offsetFromLeftEdge = Math.abs(leftEdge - width),\n          offsetFromRightEdge = Math.abs(rightEdge - width);\n\n        indexOnLine = offsetFromRightEdge < offsetFromLeftEdge ? j : j - 1;\n        break;\n      }\n    }\n\n    // reached end\n    if (!foundMatch) {\n      indexOnLine = line.length - 1;\n    }\n\n    return indexOnLine;\n  }\n\n  /**\n   * Moves cursor down\n   * @param {KeyboardEvent} e Event object\n   */\n  moveCursorDown(e: KeyboardEvent) {\n    if (\n      this.selectionStart >= this._text.length &&\n      this.selectionEnd >= this._text.length\n    ) {\n      return;\n    }\n    this._moveCursorUpOrDown('Down', e);\n  }\n\n  /**\n   * Moves cursor up\n   * @param {KeyboardEvent} e Event object\n   */\n  moveCursorUp(e: KeyboardEvent) {\n    if (this.selectionStart === 0 && this.selectionEnd === 0) {\n      return;\n    }\n    this._moveCursorUpOrDown('Up', e);\n  }\n\n  /**\n   * Moves cursor up or down, fires the events\n   * @param {String} direction 'Up' or 'Down'\n   * @param {KeyboardEvent} e Event object\n   */\n  _moveCursorUpOrDown(direction: 'Up' | 'Down', e: KeyboardEvent) {\n    const offset = this[`get${direction}CursorOffset`](\n      e,\n      this._selectionDirection === RIGHT\n    );\n    if (e.shiftKey) {\n      this.moveCursorWithShift(offset);\n    } else {\n      this.moveCursorWithoutShift(offset);\n    }\n    if (offset !== 0) {\n      const max = this.text.length;\n      this.selectionStart = capValue(0, this.selectionStart, max);\n      this.selectionEnd = capValue(0, this.selectionEnd, max);\n      // TODO fix: abort and init should be an alternative depending\n      // on selectionStart/End being equal or different\n      this.abortCursorAnimation();\n      this.initDelayedCursor();\n      this._fireSelectionChanged();\n      this._updateTextarea();\n    }\n  }\n\n  /**\n   * Moves cursor with shift\n   * @param {Number} offset\n   */\n  moveCursorWithShift(offset: number) {\n    const newSelection =\n      this._selectionDirection === LEFT\n        ? this.selectionStart + offset\n        : this.selectionEnd + offset;\n    this.setSelectionStartEndWithShift(\n      this.selectionStart,\n      this.selectionEnd,\n      newSelection\n    );\n    return offset !== 0;\n  }\n\n  /**\n   * Moves cursor up without shift\n   * @param {Number} offset\n   */\n  moveCursorWithoutShift(offset: number) {\n    if (offset < 0) {\n      this.selectionStart += offset;\n      this.selectionEnd = this.selectionStart;\n    } else {\n      this.selectionEnd += offset;\n      this.selectionStart = this.selectionEnd;\n    }\n    return offset !== 0;\n  }\n\n  /**\n   * Moves cursor left\n   * @param {KeyboardEvent} e Event object\n   */\n  moveCursorLeft(e: KeyboardEvent) {\n    if (this.selectionStart === 0 && this.selectionEnd === 0) {\n      return;\n    }\n    this._moveCursorLeftOrRight('Left', e);\n  }\n\n  /**\n   * @private\n   * @return {Boolean} true if a change happened\n   *\n   * @todo refactor not to use method name composition\n   */\n  _move(\n    e: KeyboardEvent,\n    prop: 'selectionStart' | 'selectionEnd',\n    direction: 'Left' | 'Right'\n  ): boolean {\n    let newValue: number | undefined;\n    if (e.altKey) {\n      newValue = this[`findWordBoundary${direction}`](this[prop]);\n    } else if (e.metaKey || e.keyCode === 35 || e.keyCode === 36) {\n      newValue = this[`findLineBoundary${direction}`](this[prop]);\n    } else {\n      this[prop] += direction === 'Left' ? -1 : 1;\n      return true;\n    }\n    if (typeof newValue !== 'undefined' && this[prop] !== newValue) {\n      this[prop] = newValue;\n      return true;\n    }\n    return false;\n  }\n\n  /**\n   * @private\n   */\n  _moveLeft(e: KeyboardEvent, prop: 'selectionStart' | 'selectionEnd') {\n    return this._move(e, prop, 'Left');\n  }\n\n  /**\n   * @private\n   */\n  _moveRight(e: KeyboardEvent, prop: 'selectionStart' | 'selectionEnd') {\n    return this._move(e, prop, 'Right');\n  }\n\n  /**\n   * Moves cursor left without keeping selection\n   * @param {KeyboardEvent} e\n   */\n  moveCursorLeftWithoutShift(e: KeyboardEvent) {\n    let change = true;\n    this._selectionDirection = LEFT;\n\n    // only move cursor when there is no selection,\n    // otherwise we discard it, and leave cursor on same place\n    if (\n      this.selectionEnd === this.selectionStart &&\n      this.selectionStart !== 0\n    ) {\n      change = this._moveLeft(e, 'selectionStart');\n    }\n    this.selectionEnd = this.selectionStart;\n    return change;\n  }\n\n  /**\n   * Moves cursor left while keeping selection\n   * @param {KeyboardEvent} e\n   */\n  moveCursorLeftWithShift(e: KeyboardEvent) {\n    if (\n      this._selectionDirection === RIGHT &&\n      this.selectionStart !== this.selectionEnd\n    ) {\n      return this._moveLeft(e, 'selectionEnd');\n    } else if (this.selectionStart !== 0) {\n      this._selectionDirection = LEFT;\n      return this._moveLeft(e, 'selectionStart');\n    }\n  }\n\n  /**\n   * Moves cursor right\n   * @param {KeyboardEvent} e Event object\n   */\n  moveCursorRight(e: KeyboardEvent) {\n    if (\n      this.selectionStart >= this._text.length &&\n      this.selectionEnd >= this._text.length\n    ) {\n      return;\n    }\n    this._moveCursorLeftOrRight('Right', e);\n  }\n\n  /**\n   * Moves cursor right or Left, fires event\n   * @param {String} direction 'Left', 'Right'\n   * @param {KeyboardEvent} e Event object\n   */\n  _moveCursorLeftOrRight(direction: 'Left' | 'Right', e: KeyboardEvent) {\n    const actionName = `moveCursor${direction}${\n      e.shiftKey ? 'WithShift' : 'WithoutShift'\n    }` as const;\n    this._currentCursorOpacity = 1;\n    if (this[actionName](e)) {\n      // TODO fix: abort and init should be an alternative depending\n      // on selectionStart/End being equal or different\n      this.abortCursorAnimation();\n      this.initDelayedCursor();\n      this._fireSelectionChanged();\n      this._updateTextarea();\n    }\n  }\n\n  /**\n   * Moves cursor right while keeping selection\n   * @param {KeyboardEvent} e\n   */\n  moveCursorRightWithShift(e: KeyboardEvent) {\n    if (\n      this._selectionDirection === LEFT &&\n      this.selectionStart !== this.selectionEnd\n    ) {\n      return this._moveRight(e, 'selectionStart');\n    } else if (this.selectionEnd !== this._text.length) {\n      this._selectionDirection = RIGHT;\n      return this._moveRight(e, 'selectionEnd');\n    }\n  }\n\n  /**\n   * Moves cursor right without keeping selection\n   * @param {KeyboardEvent} e Event object\n   */\n  moveCursorRightWithoutShift(e: KeyboardEvent) {\n    let changed = true;\n    this._selectionDirection = RIGHT;\n\n    if (this.selectionStart === this.selectionEnd) {\n      changed = this._moveRight(e, 'selectionStart');\n      this.selectionEnd = this.selectionStart;\n    } else {\n      this.selectionStart = this.selectionEnd;\n    }\n    return changed;\n  }\n}\n","import type { TPointerEvent, TPointerEventInfo } from '../../EventTypeDefs';\nimport type { XY } from '../../Point';\nimport { Point } from '../../Point';\nimport { stopEvent } from '../../util/dom_event';\nimport { invertTransform } from '../../util/misc/matrix';\nimport { DraggableTextDelegate } from './DraggableTextDelegate';\nimport type { ITextEvents } from './ITextBehavior';\nimport { ITextKeyBehavior } from './ITextKeyBehavior';\nimport type { TOptions } from '../../typedefs';\nimport type { TextProps, SerializedTextProps } from '../Text/Text';\n\n/**\n * `LEFT_CLICK === 0`\n */\nconst notALeftClick = (e: Event) => !!(e as MouseEvent).button;\n\nexport abstract class ITextClickBehavior<\n  Props extends TOptions<TextProps> = Partial<TextProps>,\n  SProps extends SerializedTextProps = SerializedTextProps,\n  EventSpec extends ITextEvents = ITextEvents\n> extends ITextKeyBehavior<Props, SProps, EventSpec> {\n  private declare __lastSelected: boolean;\n  private declare __lastClickTime: number;\n  private declare __lastLastClickTime: number;\n  private declare __lastPointer: XY | Record<string, never>;\n  private declare __newClickTime: number;\n\n  protected draggableTextDelegate: DraggableTextDelegate;\n\n  initBehavior() {\n    // Initializes event handlers related to cursor or selection\n    this.on('mousedown', this._mouseDownHandler);\n    this.on('mousedown:before', this._mouseDownHandlerBefore);\n    this.on('mouseup', this.mouseUpHandler);\n    this.on('mousedblclick', this.doubleClickHandler);\n    this.on('tripleclick', this.tripleClickHandler);\n\n    // Initializes \"dbclick\" event handler\n    this.__lastClickTime = +new Date();\n    // for triple click\n    this.__lastLastClickTime = +new Date();\n    this.__lastPointer = {};\n    this.on('mousedown', this.onMouseDown);\n\n    // @ts-expect-error in reality it is an IText instance\n    this.draggableTextDelegate = new DraggableTextDelegate(this);\n\n    super.initBehavior();\n  }\n\n  /**\n   * If this method returns true a mouse move operation over a text selection\n   * will not prevent the native mouse event allowing the browser to start a drag operation.\n   * shouldStartDragging can be read 'do not prevent default for mouse move event'\n   * To prevent drag and drop between objects both shouldStartDragging and onDragStart should return false\n   * @returns\n   */\n  shouldStartDragging() {\n    return this.draggableTextDelegate.isActive();\n  }\n\n  /**\n   * @public override this method to control whether instance should/shouldn't become a drag source,\n   * @see also {@link DraggableTextDelegate#isActive}\n   * To prevent drag and drop between objects both shouldStartDragging and onDragStart should return false\n   * @returns {boolean} should handle event\n   */\n  onDragStart(e: DragEvent) {\n    return this.draggableTextDelegate.onDragStart(e);\n  }\n\n  /**\n   * @public override this method to control whether instance should/shouldn't become a drop target\n   */\n  canDrop(e: DragEvent) {\n    return this.draggableTextDelegate.canDrop(e);\n  }\n\n  /**\n   * Default event handler to simulate triple click\n   * @private\n   */\n  onMouseDown(options: TPointerEventInfo) {\n    if (!this.canvas) {\n      return;\n    }\n    this.__newClickTime = +new Date();\n    const newPointer = options.pointer;\n    if (this.isTripleClick(newPointer)) {\n      this.fire('tripleclick', options);\n      stopEvent(options.e);\n    }\n    this.__lastLastClickTime = this.__lastClickTime;\n    this.__lastClickTime = this.__newClickTime;\n    this.__lastPointer = newPointer;\n    this.__lastSelected = this.selected && !this.getActiveControl();\n  }\n\n  isTripleClick(newPointer: XY) {\n    return (\n      this.__newClickTime - this.__lastClickTime < 500 &&\n      this.__lastClickTime - this.__lastLastClickTime < 500 &&\n      this.__lastPointer.x === newPointer.x &&\n      this.__lastPointer.y === newPointer.y\n    );\n  }\n\n  /**\n   * Default handler for double click, select a word\n   */\n  doubleClickHandler(options: TPointerEventInfo) {\n    if (!this.isEditing) {\n      return;\n    }\n    this.selectWord(this.getSelectionStartFromPointer(options.e));\n  }\n\n  /**\n   * Default handler for triple click, select a line\n   */\n  tripleClickHandler(options: TPointerEventInfo) {\n    if (!this.isEditing) {\n      return;\n    }\n    this.selectLine(this.getSelectionStartFromPointer(options.e));\n  }\n\n  /**\n   * Default event handler for the basic functionalities needed on _mouseDown\n   * can be overridden to do something different.\n   * Scope of this implementation is: find the click position, set selectionStart\n   * find selectionEnd, initialize the drawing of either cursor or selection area\n   * initializing a mousedDown on a text area will cancel fabricjs knowledge of\n   * current compositionMode. It will be set to false.\n   */\n  _mouseDownHandler({ e }: TPointerEventInfo) {\n    if (\n      !this.canvas ||\n      !this.editable ||\n      notALeftClick(e) ||\n      this.getActiveControl()\n    ) {\n      return;\n    }\n\n    if (this.draggableTextDelegate.start(e)) {\n      return;\n    }\n\n    this.canvas.textEditingManager.register(this);\n\n    if (this.selected) {\n      this.inCompositionMode = false;\n      this.setCursorByClick(e);\n    }\n\n    if (this.isEditing) {\n      this.__selectionStartOnMouseDown = this.selectionStart;\n      if (this.selectionStart === this.selectionEnd) {\n        this.abortCursorAnimation();\n      }\n      this.renderCursorOrSelection();\n    }\n  }\n\n  /**\n   * Default event handler for the basic functionalities needed on mousedown:before\n   * can be overridden to do something different.\n   * Scope of this implementation is: verify the object is already selected when mousing down\n   */\n  _mouseDownHandlerBefore({ e }: TPointerEventInfo) {\n    if (!this.canvas || !this.editable || notALeftClick(e)) {\n      return;\n    }\n    // we want to avoid that an object that was selected and then becomes unselectable,\n    // may trigger editing mode in some way.\n    this.selected = this === this.canvas._activeObject;\n  }\n\n  /**\n   * standard handler for mouse up, overridable\n   * @private\n   */\n  mouseUpHandler({ e, transform }: TPointerEventInfo) {\n    const didDrag = this.draggableTextDelegate.end(e);\n    if (this.canvas) {\n      this.canvas.textEditingManager.unregister(this);\n\n      const activeObject = this.canvas._activeObject;\n      if (activeObject && activeObject !== this) {\n        // avoid running this logic when there is an active object\n        // this because is possible with shift click and fast clicks,\n        // to rapidly deselect and reselect this object and trigger an enterEdit\n        return;\n      }\n    }\n    if (\n      !this.editable ||\n      (this.group && !this.group.interactive) ||\n      (transform && transform.actionPerformed) ||\n      notALeftClick(e) ||\n      didDrag\n    ) {\n      return;\n    }\n\n    if (this.__lastSelected && !this.getActiveControl()) {\n      this.selected = false;\n      this.__lastSelected = false;\n      this.enterEditing(e);\n      if (this.selectionStart === this.selectionEnd) {\n        this.initDelayedCursor(true);\n      } else {\n        this.renderCursorOrSelection();\n      }\n    } else {\n      this.selected = true;\n    }\n  }\n\n  /**\n   * Changes cursor location in a text depending on passed pointer (x/y) object\n   * @param {TPointerEvent} e Event object\n   */\n  setCursorByClick(e: TPointerEvent) {\n    const newSelection = this.getSelectionStartFromPointer(e),\n      start = this.selectionStart,\n      end = this.selectionEnd;\n    if (e.shiftKey) {\n      this.setSelectionStartEndWithShift(start, end, newSelection);\n    } else {\n      this.selectionStart = newSelection;\n      this.selectionEnd = newSelection;\n    }\n    if (this.isEditing) {\n      this._fireSelectionChanged();\n      this._updateTextarea();\n    }\n  }\n\n  /**\n   * Returns index of a character corresponding to where an object was clicked\n   * @param {TPointerEvent} e Event object\n   * @return {Number} Index of a character\n   */\n  getSelectionStartFromPointer(e: TPointerEvent): number {\n    const mouseOffset = this.canvas!.getScenePoint(e)\n      .transform(invertTransform(this.calcTransformMatrix()))\n      .add(new Point(-this._getLeftOffset(), -this._getTopOffset()));\n    let height = 0,\n      charIndex = 0,\n      lineIndex = 0;\n\n    for (let i = 0; i < this._textLines.length; i++) {\n      if (height <= mouseOffset.y) {\n        height += this.getHeightOfLine(i);\n        lineIndex = i;\n        if (i > 0) {\n          charIndex +=\n            this._textLines[i - 1].length + this.missingNewlineOffset(i - 1);\n        }\n      } else {\n        break;\n      }\n    }\n    const lineLeftOffset = Math.abs(this._getLineLeftOffset(lineIndex));\n    let width = lineLeftOffset;\n    const charLength = this._textLines[lineIndex].length;\n    const chars = this.__charBounds[lineIndex];\n    for (let j = 0; j < charLength; j++) {\n      // i removed something about flipX here, check.\n      const charWidth = chars[j].kernedWidth;\n      const widthAfter = width + charWidth;\n      if (mouseOffset.x <= widthAfter) {\n        // if the pointer is closer to the end of the char we increment charIndex\n        // in order to position the cursor after the char\n        if (\n          Math.abs(mouseOffset.x - widthAfter) <=\n          Math.abs(mouseOffset.x - width)\n        ) {\n          charIndex++;\n        }\n        break;\n      }\n      width = widthAfter;\n      charIndex++;\n    }\n\n    return Math.min(\n      // if object is horizontally flipped, mirror cursor location from the end\n      this.flipX ? charLength - charIndex : charIndex,\n      this._text.length\n    );\n  }\n}\n","export type TKeyMapIText = Record<\n  KeyboardEvent['keyCode'],\n  CursorHandlingMethods\n>;\n\nexport type CursorHandlingMethods =\n  | 'moveCursorUp'\n  | 'moveCursorDown'\n  | 'moveCursorLeft'\n  | 'moveCursorRight'\n  | 'exitEditing'\n  | 'copy'\n  | 'cut'\n  | 'selectAll';\n\nconst MOVE_CURSOR_UP: CursorHandlingMethods = 'moveCursorUp';\nconst MOVE_CURSOR_DOWN: CursorHandlingMethods = 'moveCursorDown';\nconst MOVE_CURSOR_LEFT: CursorHandlingMethods = 'moveCursorLeft';\nconst MOVE_CURSOR_RIGHT: CursorHandlingMethods = 'moveCursorRight';\nconst EXIT_EDITING: CursorHandlingMethods = 'exitEditing';\n\n// @TODO look into import { Key } from 'ts-key-enum';\n// and transition from keyCode to Key\n// also reduce string duplication\nexport const keysMap: TKeyMapIText = {\n  9: EXIT_EDITING,\n  27: EXIT_EDITING,\n  33: MOVE_CURSOR_UP,\n  34: MOVE_CURSOR_DOWN,\n  35: MOVE_CURSOR_RIGHT,\n  36: MOVE_CURSOR_LEFT,\n  37: MOVE_CURSOR_LEFT,\n  38: MOVE_CURSOR_UP,\n  39: MOVE_CURSOR_RIGHT,\n  40: MOVE_CURSOR_DOWN,\n};\n\nexport const keysMapRtl: TKeyMapIText = {\n  9: EXIT_EDITING,\n  27: EXIT_EDITING,\n  33: MOVE_CURSOR_UP,\n  34: MOVE_CURSOR_DOWN,\n  35: MOVE_CURSOR_LEFT,\n  36: MOVE_CURSOR_RIGHT,\n  37: MOVE_CURSOR_RIGHT,\n  38: MOVE_CURSOR_UP,\n  39: MOVE_CURSOR_LEFT,\n  40: MOVE_CURSOR_DOWN,\n};\n\n/**\n * For functionalities on keyUp + ctrl || cmd\n */\nexport const ctrlKeysMapUp: TKeyMapIText = {\n  67: 'copy',\n  // there was a reason this wasn't deleted. for now leave it here\n  88: 'cut',\n};\n\n/**\n * For functionalities on keyDown + ctrl || cmd\n */\nexport const ctrlKeysMapDown: TKeyMapIText = {\n  65: 'selectAll',\n};\n","import { Canvas } from '../../canvas/Canvas';\nimport type { ITextEvents } from './ITextBehavior';\nimport { ITextClickBehavior } from './ITextClickBehavior';\nimport {\n  ctrlKeysMapDown,\n  ctrlKeysMapUp,\n  keysMap,\n  keysMapRtl,\n} from './constants';\nimport type { TClassProperties, TFiller, TOptions } from '../../typedefs';\nimport { classRegistry } from '../../ClassRegistry';\nimport type { SerializedTextProps, TextProps } from '../Text/Text';\nimport {\n  JUSTIFY,\n  JUSTIFY_CENTER,\n  JUSTIFY_LEFT,\n  JUSTIFY_RIGHT,\n} from '../Text/constants';\nimport { CENTER, LEFT, RIGHT } from '../../constants';\nimport type { ObjectToCanvasElementOptions } from '../Object/Object';\n\ntype CursorBoundaries = {\n  left: number;\n  top: number;\n  leftOffset: number;\n  topOffset: number;\n};\n\n// Declare IText protected properties to workaround TS\nconst protectedDefaultValues = {\n  _selectionDirection: null,\n  _reSpace: /\\s|\\r?\\n/,\n  inCompositionMode: false,\n};\n\nexport const iTextDefaultValues: Partial<TClassProperties<IText>> = {\n  selectionStart: 0,\n  selectionEnd: 0,\n  selectionColor: 'rgba(17,119,255,0.3)',\n  isEditing: false,\n  editable: true,\n  editingBorderColor: 'rgba(102,153,255,0.25)',\n  cursorWidth: 2,\n  cursorColor: '',\n  cursorDelay: 1000,\n  cursorDuration: 600,\n  caching: true,\n  hiddenTextareaContainer: null,\n  keysMap,\n  keysMapRtl,\n  ctrlKeysMapDown,\n  ctrlKeysMapUp,\n  ...protectedDefaultValues,\n};\n\n// @TODO this is not complete\ninterface UniqueITextProps {\n  selectionStart: number;\n  selectionEnd: number;\n}\n\nexport interface SerializedITextProps\n  extends SerializedTextProps,\n    UniqueITextProps {}\n\nexport interface ITextProps extends TextProps, UniqueITextProps {}\n\n/**\n * @fires changed\n * @fires selection:changed\n * @fires editing:entered\n * @fires editing:exited\n * @fires dragstart\n * @fires drag drag event firing on the drag source\n * @fires dragend\n * @fires copy\n * @fires cut\n * @fires paste\n *\n * #### Supported key combinations\n * ```\n *   Move cursor:                    left, right, up, down\n *   Select character:               shift + left, shift + right\n *   Select text vertically:         shift + up, shift + down\n *   Move cursor by word:            alt + left, alt + right\n *   Select words:                   shift + alt + left, shift + alt + right\n *   Move cursor to line start/end:  cmd + left, cmd + right or home, end\n *   Select till start/end of line:  cmd + shift + left, cmd + shift + right or shift + home, shift + end\n *   Jump to start/end of text:      cmd + up, cmd + down\n *   Select till start/end of text:  cmd + shift + up, cmd + shift + down or shift + pgUp, shift + pgDown\n *   Delete character:               backspace\n *   Delete word:                    alt + backspace\n *   Delete line:                    cmd + backspace\n *   Forward delete:                 delete\n *   Copy text:                      ctrl/cmd + c\n *   Paste text:                     ctrl/cmd + v\n *   Cut text:                       ctrl/cmd + x\n *   Select entire text:             ctrl/cmd + a\n *   Quit editing                    tab or esc\n * ```\n *\n * #### Supported mouse/touch combination\n * ```\n *   Position cursor:                click/touch\n *   Create selection:               click/touch & drag\n *   Create selection:               click & shift + click\n *   Select word:                    double click\n *   Select line:                    triple click\n * ```\n */\nexport class IText<\n    Props extends TOptions<ITextProps> = Partial<ITextProps>,\n    SProps extends SerializedITextProps = SerializedITextProps,\n    EventSpec extends ITextEvents = ITextEvents\n  >\n  extends ITextClickBehavior<Props, SProps, EventSpec>\n  implements UniqueITextProps\n{\n  /**\n   * Index where text selection starts (or where cursor is when there is no selection)\n   * @type Number\n   * @default\n   */\n  declare selectionStart: number;\n\n  /**\n   * Index where text selection ends\n   * @type Number\n   * @default\n   */\n  declare selectionEnd: number;\n\n  declare compositionStart: number;\n\n  declare compositionEnd: number;\n\n  /**\n   * Color of text selection\n   * @type String\n   * @default\n   */\n  declare selectionColor: string;\n\n  /**\n   * Indicates whether text is in editing mode\n   * @type Boolean\n   * @default\n   */\n  declare isEditing: boolean;\n\n  /**\n   * Indicates whether a text can be edited\n   * @type Boolean\n   * @default\n   */\n  declare editable: boolean;\n\n  /**\n   * Border color of text object while it's in editing mode\n   * @type String\n   * @default\n   */\n  declare editingBorderColor: string;\n\n  /**\n   * Width of cursor (in px)\n   * @type Number\n   * @default\n   */\n  declare cursorWidth: number;\n\n  /**\n   * Color of text cursor color in editing mode.\n   * if not set (default) will take color from the text.\n   * if set to a color value that fabric can understand, it will\n   * be used instead of the color of the text at the current position.\n   * @type String\n   * @default\n   */\n  declare cursorColor: string;\n\n  /**\n   * Delay between cursor blink (in ms)\n   * @type Number\n   * @default\n   */\n  declare cursorDelay: number;\n\n  /**\n   * Duration of cursor fade in (in ms)\n   * @type Number\n   * @default\n   */\n  declare cursorDuration: number;\n\n  declare compositionColor: string;\n\n  /**\n   * Indicates whether internal text char widths can be cached\n   * @type Boolean\n   * @default\n   */\n  declare caching: boolean;\n\n  static ownDefaults = iTextDefaultValues;\n\n  static getDefaults(): Record<string, any> {\n    return { ...super.getDefaults(), ...IText.ownDefaults };\n  }\n\n  static type = 'IText';\n\n  get type() {\n    const type = super.type;\n    // backward compatibility\n    return type === 'itext' ? 'i-text' : type;\n  }\n\n  /**\n\n   * Constructor\n   * @param {String} text Text string\n   * @param {Object} [options] Options object\n   */\n  constructor(text: string, options?: Props) {\n    super(text, options);\n    this.initBehavior();\n  }\n\n  /**\n   * While editing handle differently\n   * @private\n   * @param {string} key\n   * @param {*} value\n   */\n  _set(key: string, value: any) {\n    if (this.isEditing && this._savedProps && key in this._savedProps) {\n      // @ts-expect-error irritating TS\n      this._savedProps[key] = value;\n      return this;\n    }\n    if (key === 'canvas') {\n      this.canvas instanceof Canvas &&\n        this.canvas.textEditingManager.remove(this);\n      value instanceof Canvas && value.textEditingManager.add(this);\n    }\n    return super._set(key, value);\n  }\n\n  /**\n   * Sets selection start (left boundary of a selection)\n   * @param {Number} index Index to set selection start to\n   */\n  setSelectionStart(index: number) {\n    index = Math.max(index, 0);\n    this._updateAndFire('selectionStart', index);\n  }\n\n  /**\n   * Sets selection end (right boundary of a selection)\n   * @param {Number} index Index to set selection end to\n   */\n  setSelectionEnd(index: number) {\n    index = Math.min(index, this.text.length);\n    this._updateAndFire('selectionEnd', index);\n  }\n\n  /**\n   * @private\n   * @param {String} property 'selectionStart' or 'selectionEnd'\n   * @param {Number} index new position of property\n   */\n  protected _updateAndFire(\n    property: 'selectionStart' | 'selectionEnd',\n    index: number\n  ) {\n    if (this[property] !== index) {\n      this._fireSelectionChanged();\n      this[property] = index;\n    }\n    this._updateTextarea();\n  }\n\n  /**\n   * Fires the even of selection changed\n   * @private\n   */\n  _fireSelectionChanged() {\n    this.fire('selection:changed');\n    this.canvas && this.canvas.fire('text:selection:changed', { target: this });\n  }\n\n  /**\n   * Initialize text dimensions. Render all text on given context\n   * or on a offscreen canvas to get the text width with measureText.\n   * Updates this.width and this.height with the proper values.\n   * Does not return dimensions.\n   * @private\n   */\n  initDimensions() {\n    this.isEditing && this.initDelayedCursor();\n    super.initDimensions();\n  }\n\n  /**\n   * Gets style of a current selection/cursor (at the start position)\n   * if startIndex or endIndex are not provided, selectionStart or selectionEnd will be used.\n   * @param {Number} startIndex Start index to get styles at\n   * @param {Number} endIndex End index to get styles at, if not specified selectionEnd or startIndex + 1\n   * @param {Boolean} [complete] get full style or not\n   * @return {Array} styles an array with one, zero or more Style objects\n   */\n  getSelectionStyles(\n    startIndex: number = this.selectionStart || 0,\n    endIndex: number = this.selectionEnd,\n    complete?: boolean\n  ) {\n    return super.getSelectionStyles(startIndex, endIndex, complete);\n  }\n\n  /**\n   * Sets style of a current selection, if no selection exist, do not set anything.\n   * @param {Object} [styles] Styles object\n   * @param {Number} [startIndex] Start index to get styles at\n   * @param {Number} [endIndex] End index to get styles at, if not specified selectionEnd or startIndex + 1\n   */\n  setSelectionStyles(\n    styles: object,\n    startIndex: number = this.selectionStart || 0,\n    endIndex: number = this.selectionEnd\n  ) {\n    return super.setSelectionStyles(styles, startIndex, endIndex);\n  }\n\n  /**\n   * Returns 2d representation (lineIndex and charIndex) of cursor (or selection start)\n   * @param {Number} [selectionStart] Optional index. When not given, current selectionStart is used.\n   * @param {Boolean} [skipWrapping] consider the location for unwrapped lines. useful to manage styles.\n   */\n  get2DCursorLocation(\n    selectionStart = this.selectionStart,\n    skipWrapping?: boolean\n  ) {\n    return super.get2DCursorLocation(selectionStart, skipWrapping);\n  }\n\n  /**\n   * @private\n   * @param {CanvasRenderingContext2D} ctx Context to render on\n   */\n  render(ctx: CanvasRenderingContext2D) {\n    super.render(ctx);\n    // clear the cursorOffsetCache, so we ensure to calculate once per renderCursor\n    // the correct position but not at every cursor animation.\n    this.cursorOffsetCache = {};\n    this.renderCursorOrSelection();\n  }\n\n  /**\n   * @override block cursor/selection logic while rendering the exported canvas\n   * @todo this workaround should be replaced with a more robust solution\n   */\n  toCanvasElement(options?: ObjectToCanvasElementOptions): HTMLCanvasElement {\n    const isEditing = this.isEditing;\n    this.isEditing = false;\n    const canvas = super.toCanvasElement(options);\n    this.isEditing = isEditing;\n    return canvas;\n  }\n\n  /**\n   * Renders cursor or selection (depending on what exists)\n   * it does on the contextTop. If contextTop is not available, do nothing.\n   */\n  renderCursorOrSelection() {\n    if (!this.isEditing) {\n      return;\n    }\n    const ctx = this.clearContextTop(true);\n    if (!ctx) {\n      return;\n    }\n    const boundaries = this._getCursorBoundaries();\n    if (this.selectionStart === this.selectionEnd) {\n      this.renderCursor(ctx, boundaries);\n    } else {\n      this.renderSelection(ctx, boundaries);\n    }\n    this.canvas!.contextTopDirty = true;\n    ctx.restore();\n  }\n\n  /**\n   * Returns cursor boundaries (left, top, leftOffset, topOffset)\n   * left/top are left/top of entire text box\n   * leftOffset/topOffset are offset from that left/top point of a text box\n   * @private\n   * @param {number} [index] index from start\n   * @param {boolean} [skipCaching]\n   */\n  _getCursorBoundaries(\n    index: number = this.selectionStart,\n    skipCaching?: boolean\n  ): CursorBoundaries {\n    const left = this._getLeftOffset(),\n      top = this._getTopOffset(),\n      offsets = this._getCursorBoundariesOffsets(index, skipCaching);\n    return {\n      left: left,\n      top: top,\n      leftOffset: offsets.left,\n      topOffset: offsets.top,\n    };\n  }\n\n  /**\n   * Caches and returns cursor left/top offset relative to instance's center point\n   * @private\n   * @param {number} index index from start\n   * @param {boolean} [skipCaching]\n   */\n  _getCursorBoundariesOffsets(\n    index: number,\n    skipCaching?: boolean\n  ): { left: number; top: number } {\n    if (skipCaching) {\n      return this.__getCursorBoundariesOffsets(index);\n    }\n    if (this.cursorOffsetCache && 'top' in this.cursorOffsetCache) {\n      return this.cursorOffsetCache as { left: number; top: number };\n    }\n    return (this.cursorOffsetCache = this.__getCursorBoundariesOffsets(index));\n  }\n\n  /**\n   * Calculates cursor left/top offset relative to instance's center point\n   * @private\n   * @param {number} index index from start\n   */\n  __getCursorBoundariesOffsets(index: number) {\n    let topOffset = 0,\n      leftOffset = 0;\n    const { charIndex, lineIndex } = this.get2DCursorLocation(index);\n\n    for (let i = 0; i < lineIndex; i++) {\n      topOffset += this.getHeightOfLine(i);\n    }\n    const lineLeftOffset = this._getLineLeftOffset(lineIndex);\n    const bound = this.__charBounds[lineIndex][charIndex];\n    bound && (leftOffset = bound.left);\n    if (\n      this.charSpacing !== 0 &&\n      charIndex === this._textLines[lineIndex].length\n    ) {\n      leftOffset -= this._getWidthOfCharSpacing();\n    }\n    const boundaries = {\n      top: topOffset,\n      left: lineLeftOffset + (leftOffset > 0 ? leftOffset : 0),\n    };\n    if (this.direction === 'rtl') {\n      if (\n        this.textAlign === RIGHT ||\n        this.textAlign === JUSTIFY ||\n        this.textAlign === JUSTIFY_RIGHT\n      ) {\n        boundaries.left *= -1;\n      } else if (this.textAlign === LEFT || this.textAlign === JUSTIFY_LEFT) {\n        boundaries.left = lineLeftOffset - (leftOffset > 0 ? leftOffset : 0);\n      } else if (\n        this.textAlign === CENTER ||\n        this.textAlign === JUSTIFY_CENTER\n      ) {\n        boundaries.left = lineLeftOffset - (leftOffset > 0 ? leftOffset : 0);\n      }\n    }\n    return boundaries;\n  }\n\n  /**\n   * Renders cursor on context Top, outside the animation cycle, on request\n   * Used for the drag/drop effect.\n   * If contextTop is not available, do nothing.\n   */\n  renderCursorAt(selectionStart: number) {\n    const boundaries = this._getCursorBoundaries(selectionStart, true);\n    this._renderCursor(this.canvas!.contextTop, boundaries, selectionStart);\n  }\n\n  /**\n   * Renders cursor\n   * @param {Object} boundaries\n   * @param {CanvasRenderingContext2D} ctx transformed context to draw on\n   */\n  renderCursor(ctx: CanvasRenderingContext2D, boundaries: CursorBoundaries) {\n    this._renderCursor(ctx, boundaries, this.selectionStart);\n  }\n\n  _renderCursor(\n    ctx: CanvasRenderingContext2D,\n    boundaries: CursorBoundaries,\n    selectionStart: number\n  ) {\n    const cursorLocation = this.get2DCursorLocation(selectionStart),\n      lineIndex = cursorLocation.lineIndex,\n      charIndex =\n        cursorLocation.charIndex > 0 ? cursorLocation.charIndex - 1 : 0,\n      charHeight = this.getValueOfPropertyAt(lineIndex, charIndex, 'fontSize'),\n      multiplier = this.getObjectScaling().x * this.canvas!.getZoom(),\n      cursorWidth = this.cursorWidth / multiplier,\n      dy = this.getValueOfPropertyAt(lineIndex, charIndex, 'deltaY'),\n      topOffset =\n        boundaries.topOffset +\n        ((1 - this._fontSizeFraction) * this.getHeightOfLine(lineIndex)) /\n          this.lineHeight -\n        charHeight * (1 - this._fontSizeFraction);\n\n    if (this.inCompositionMode) {\n      // TODO: investigate why there isn't a return inside the if,\n      // and why can't happen at the top of the function\n      this.renderSelection(ctx, boundaries);\n    }\n    ctx.fillStyle =\n      this.cursorColor ||\n      (this.getValueOfPropertyAt(lineIndex, charIndex, 'fill') as string);\n    ctx.globalAlpha = this._currentCursorOpacity;\n    ctx.fillRect(\n      boundaries.left + boundaries.leftOffset - cursorWidth / 2,\n      topOffset + boundaries.top + dy,\n      cursorWidth,\n      charHeight\n    );\n  }\n\n  /**\n   * Renders text selection\n   * @param {Object} boundaries Object with left/top/leftOffset/topOffset\n   * @param {CanvasRenderingContext2D} ctx transformed context to draw on\n   */\n  renderSelection(ctx: CanvasRenderingContext2D, boundaries: CursorBoundaries) {\n    const selection = {\n      selectionStart: this.inCompositionMode\n        ? this.hiddenTextarea!.selectionStart\n        : this.selectionStart,\n      selectionEnd: this.inCompositionMode\n        ? this.hiddenTextarea!.selectionEnd\n        : this.selectionEnd,\n    };\n    this._renderSelection(ctx, selection, boundaries);\n  }\n\n  /**\n   * Renders drag start text selection\n   */\n  renderDragSourceEffect() {\n    const dragStartSelection =\n      this.draggableTextDelegate.getDragStartSelection()!;\n    this._renderSelection(\n      this.canvas!.contextTop,\n      dragStartSelection,\n      this._getCursorBoundaries(dragStartSelection.selectionStart, true)\n    );\n  }\n\n  renderDropTargetEffect(e: DragEvent) {\n    const dragSelection = this.getSelectionStartFromPointer(e);\n    this.renderCursorAt(dragSelection);\n  }\n\n  /**\n   * Renders text selection\n   * @private\n   * @param {{ selectionStart: number, selectionEnd: number }} selection\n   * @param {Object} boundaries Object with left/top/leftOffset/topOffset\n   * @param {CanvasRenderingContext2D} ctx transformed context to draw on\n   */\n  _renderSelection(\n    ctx: CanvasRenderingContext2D,\n    selection: { selectionStart: number; selectionEnd: number },\n    boundaries: CursorBoundaries\n  ) {\n    const selectionStart = selection.selectionStart,\n      selectionEnd = selection.selectionEnd,\n      isJustify = this.textAlign.includes(JUSTIFY),\n      start = this.get2DCursorLocation(selectionStart),\n      end = this.get2DCursorLocation(selectionEnd),\n      startLine = start.lineIndex,\n      endLine = end.lineIndex,\n      startChar = start.charIndex < 0 ? 0 : start.charIndex,\n      endChar = end.charIndex < 0 ? 0 : end.charIndex;\n\n    for (let i = startLine; i <= endLine; i++) {\n      const lineOffset = this._getLineLeftOffset(i) || 0;\n      let lineHeight = this.getHeightOfLine(i),\n        realLineHeight = 0,\n        boxStart = 0,\n        boxEnd = 0;\n\n      if (i === startLine) {\n        boxStart = this.__charBounds[startLine][startChar].left;\n      }\n      if (i >= startLine && i < endLine) {\n        boxEnd =\n          isJustify && !this.isEndOfWrapping(i)\n            ? this.width\n            : this.getLineWidth(i) || 5; // WTF is this 5?\n      } else if (i === endLine) {\n        if (endChar === 0) {\n          boxEnd = this.__charBounds[endLine][endChar].left;\n        } else {\n          const charSpacing = this._getWidthOfCharSpacing();\n          boxEnd =\n            this.__charBounds[endLine][endChar - 1].left +\n            this.__charBounds[endLine][endChar - 1].width -\n            charSpacing;\n        }\n      }\n      realLineHeight = lineHeight;\n      if (this.lineHeight < 1 || (i === endLine && this.lineHeight > 1)) {\n        lineHeight /= this.lineHeight;\n      }\n      let drawStart = boundaries.left + lineOffset + boxStart,\n        drawHeight = lineHeight,\n        extraTop = 0;\n      const drawWidth = boxEnd - boxStart;\n      if (this.inCompositionMode) {\n        ctx.fillStyle = this.compositionColor || 'black';\n        drawHeight = 1;\n        extraTop = lineHeight;\n      } else {\n        ctx.fillStyle = this.selectionColor;\n      }\n      if (this.direction === 'rtl') {\n        if (\n          this.textAlign === RIGHT ||\n          this.textAlign === JUSTIFY ||\n          this.textAlign === JUSTIFY_RIGHT\n        ) {\n          drawStart = this.width - drawStart - drawWidth;\n        } else if (this.textAlign === LEFT || this.textAlign === JUSTIFY_LEFT) {\n          drawStart = boundaries.left + lineOffset - boxEnd;\n        } else if (\n          this.textAlign === CENTER ||\n          this.textAlign === JUSTIFY_CENTER\n        ) {\n          drawStart = boundaries.left + lineOffset - boxEnd;\n        }\n      }\n      ctx.fillRect(\n        drawStart,\n        boundaries.top + boundaries.topOffset + extraTop,\n        drawWidth,\n        drawHeight\n      );\n      boundaries.topOffset += realLineHeight;\n    }\n  }\n\n  /**\n   * High level function to know the height of the cursor.\n   * the currentChar is the one that precedes the cursor\n   * Returns fontSize of char at the current cursor\n   * Unused from the library, is for the end user\n   * @return {Number} Character font size\n   */\n  getCurrentCharFontSize(): number {\n    const cp = this._getCurrentCharIndex();\n    return this.getValueOfPropertyAt(cp.l, cp.c, 'fontSize');\n  }\n\n  /**\n   * High level function to know the color of the cursor.\n   * the currentChar is the one that precedes the cursor\n   * Returns color (fill) of char at the current cursor\n   * if the text object has a pattern or gradient for filler, it will return that.\n   * Unused by the library, is for the end user\n   * @return {String | TFiller} Character color (fill)\n   */\n  getCurrentCharColor(): string | TFiller | null {\n    const cp = this._getCurrentCharIndex();\n    return this.getValueOfPropertyAt(cp.l, cp.c, 'fill');\n  }\n\n  /**\n   * Returns the cursor position for the getCurrent.. functions\n   * @private\n   */\n  _getCurrentCharIndex() {\n    const cursorPosition = this.get2DCursorLocation(this.selectionStart, true),\n      charIndex =\n        cursorPosition.charIndex > 0 ? cursorPosition.charIndex - 1 : 0;\n    return { l: cursorPosition.lineIndex, c: charIndex };\n  }\n\n  dispose() {\n    this._exitEditing();\n    this.draggableTextDelegate.dispose();\n    super.dispose();\n  }\n}\n\nclassRegistry.setClass(IText);\n// legacy\nclassRegistry.setClass(IText, 'i-text');\n","import { Point } from '../../Point';\nimport type { FabricObject } from '../../shapes/Object/FabricObject';\nimport { makeBoundingBoxFromPoints } from '../../util/misc/boundingBoxFromPoints';\nimport { sendPointToPlane } from '../../util/misc/planeChange';\nimport type { LayoutStrategyResult, StrictLayoutContext } from '../types';\nimport { LayoutStrategy } from './LayoutStrategy';\nimport { getObjectBounds } from './utils';\nimport { classRegistry } from '../../ClassRegistry';\n\n/**\n * Layout will adjust the bounding box to match the clip path bounding box.\n */\nexport class ClipPathLayout extends LayoutStrategy {\n  static readonly type = 'clip-path';\n\n  shouldPerformLayout(context: StrictLayoutContext): boolean {\n    return !!context.target.clipPath && super.shouldPerformLayout(context);\n  }\n\n  shouldLayoutClipPath() {\n    return false;\n  }\n\n  calcLayoutResult(\n    context: StrictLayoutContext,\n    objects: FabricObject[]\n  ): LayoutStrategyResult | undefined {\n    const { target } = context;\n    const { clipPath } = target;\n    if (!clipPath || !this.shouldPerformLayout(context)) {\n      return;\n    }\n    // TODO: remove stroke calculation from this case\n    const { width, height } = makeBoundingBoxFromPoints(\n      getObjectBounds(target, clipPath as FabricObject)\n    );\n    const size = new Point(width, height);\n    if (clipPath.absolutePositioned) {\n      //  we want the center point to exist in group's containing plane\n      const clipPathCenter = sendPointToPlane(\n        clipPath.getRelativeCenterPoint(),\n        undefined,\n        target.group?.calcTransformMatrix()\n      );\n      return {\n        center: clipPathCenter,\n        size,\n      };\n    } else {\n      //  we want the center point to exist in group's containing plane, so we send it upwards\n      const clipPathCenter = clipPath\n        .getRelativeCenterPoint()\n        .transform(target.calcOwnMatrix(), true);\n      if (this.shouldPerformLayout(context)) {\n        // the clip path is positioned relative to the group's center which is affected by the bbox\n        // so we first calculate the bbox\n        const { center = new Point(), correction = new Point() } =\n          this.calcBoundingBox(objects, context) || {};\n        return {\n          center: center.add(clipPathCenter),\n          correction: correction.subtract(clipPathCenter),\n          size,\n        };\n      } else {\n        return {\n          center: target.getRelativeCenterPoint().add(clipPathCenter),\n          size,\n        };\n      }\n    }\n  }\n}\n\nclassRegistry.setClass(ClipPathLayout);\n","import { Point } from '../../Point';\nimport type {\n  InitializationLayoutContext,\n  LayoutStrategyResult,\n  StrictLayoutContext,\n} from '../types';\nimport { LayoutStrategy } from './LayoutStrategy';\nimport { classRegistry } from '../../ClassRegistry';\n\n/**\n * Layout will keep target's initial size.\n */\nexport class FixedLayout extends LayoutStrategy {\n  static readonly type = 'fixed';\n\n  /**\n   * @override respect target's initial size\n   */\n  getInitialSize(\n    { target }: StrictLayoutContext & InitializationLayoutContext,\n    { size }: Pick<LayoutStrategyResult, 'center' | 'size'>\n  ): Point {\n    return new Point(target.width || size.x, target.height || size.y);\n  }\n}\n\nclassRegistry.setClass(FixedLayout);\n","import { LayoutManager } from './LayoutManager';\nimport type { RegistrationContext, StrictLayoutContext } from './types';\nimport type { Group } from '../shapes/Group';\n\n/**\n * Today the LayoutManager class also takes care of subscribing event handlers\n * to update the group layout when the group is interactive and a transform is applied\n * to a child object.\n * The ActiveSelection is never interactive, but it could contain objects from\n * groups that are.\n * The standard LayoutManager would subscribe the children of the activeSelection to\n * perform layout changes to the active selection itself, what we need instead is that\n * the transformation applied to the active selection will trigger changes to the\n * original group of the children ( the one referenced under the parent property )\n * This subclass of the LayoutManager has a single duty to fill the gap of this difference.`\n */\nexport class ActiveSelectionLayoutManager extends LayoutManager {\n  subscribeTargets(\n    context: RegistrationContext & Partial<StrictLayoutContext>\n  ): void {\n    const activeSelection = context.target;\n    const parents = context.targets.reduce((parents, target) => {\n      target.parent && parents.add(target.parent);\n      return parents;\n    }, new Set<Group>());\n    parents.forEach((parent) => {\n      parent.layoutManager.subscribeTargets({\n        target: parent,\n        targets: [activeSelection],\n      });\n    });\n  }\n\n  /**\n   * unsubscribe from parent only if all its children were deselected\n   */\n  unsubscribeTargets(\n    context: RegistrationContext & Partial<StrictLayoutContext>\n  ): void {\n    const activeSelection = context.target;\n    const selectedObjects = activeSelection.getObjects();\n    const parents = context.targets.reduce((parents, target) => {\n      target.parent && parents.add(target.parent);\n      return parents;\n    }, new Set<Group>());\n    parents.forEach((parent) => {\n      !selectedObjects.some((object) => object.parent === parent) &&\n        parent.layoutManager.unsubscribeTargets({\n          target: parent,\n          targets: [activeSelection],\n        });\n    });\n  }\n}\n","import type { ControlRenderingStyleOverride } from '../controls/controlRendering';\nimport { classRegistry } from '../ClassRegistry';\nimport type { GroupProps } from './Group';\nimport { Group } from './Group';\nimport type { FabricObject } from './Object/FabricObject';\nimport {\n  LAYOUT_TYPE_ADDED,\n  LAYOUT_TYPE_REMOVED,\n} from '../LayoutManager/constants';\nimport type { TClassProperties } from '../typedefs';\nimport { log } from '../util/internals/console';\nimport { ActiveSelectionLayoutManager } from '../LayoutManager/ActiveSelectionLayoutManager';\n\nexport type MultiSelectionStacking = 'canvas-stacking' | 'selection-order';\n\nexport interface ActiveSelectionOptions extends GroupProps {\n  multiSelectionStacking: MultiSelectionStacking;\n}\n\nconst activeSelectionDefaultValues: Partial<TClassProperties<ActiveSelection>> =\n  {\n    multiSelectionStacking: 'canvas-stacking',\n    cornerStyle: 'circle',\n    cornerColor: 'white',\n    cornerStrokeColor: 'gray',\n    transparentCorners: false,\n  };\n\n/**\n * Used by Canvas to manage selection.\n *\n * @example\n * class MyActiveSelection extends ActiveSelection {\n *   ...\n * }\n *\n * // override the default `ActiveSelection` class\n * classRegistry.setClass(MyActiveSelection)\n */\nexport class ActiveSelection extends Group {\n  static type = 'ActiveSelection';\n\n  static ownDefaults: Record<string, any> = activeSelectionDefaultValues;\n\n  static getDefaults(): Record<string, any> {\n    return { ...super.getDefaults(), ...ActiveSelection.ownDefaults };\n  }\n\n  /**\n   * The ActiveSelection needs to use the ActiveSelectionLayoutManager\n   * or selections on interactive groups may be broken\n   */\n  declare layoutManager: ActiveSelectionLayoutManager;\n\n  /**\n   * controls how selected objects are added during a multiselection event\n   * - `canvas-stacking` adds the selected object to the active selection while respecting canvas object stacking order\n   * - `selection-order` adds the selected object to the top of the stack,\n   * meaning that the stack is ordered by the order in which objects were selected\n   * @default `canvas-stacking`\n   */\n  declare multiSelectionStacking: MultiSelectionStacking;\n\n  constructor(\n    objects: FabricObject[] = [],\n    options: Partial<ActiveSelectionOptions> = {}\n  ) {\n    super(objects, {\n      ...options,\n      layoutManager:\n        options.layoutManager ?? new ActiveSelectionLayoutManager(),\n    });\n  }\n\n  /**\n   * @private\n   */\n  _shouldSetNestedCoords() {\n    return true;\n  }\n\n  /**\n   * @private\n   * @override we don't want the selection monitor to be active\n   */\n  __objectSelectionMonitor() {\n    //  noop\n  }\n\n  /**\n   * Adds objects with respect to {@link multiSelectionStacking}\n   * @param targets object to add to selection\n   */\n  multiSelectAdd(...targets: FabricObject[]) {\n    if (this.multiSelectionStacking === 'selection-order') {\n      this.add(...targets);\n    } else {\n      //  respect object stacking as it is on canvas\n      //  perf enhancement for large ActiveSelection: consider a binary search of `isInFrontOf`\n      targets.forEach((target) => {\n        const index = this._objects.findIndex((obj) => obj.isInFrontOf(target));\n        const insertAt =\n          index === -1\n            ? //  `target` is in front of all other objects\n              this.size()\n            : index;\n        this.insertAt(insertAt, target);\n      });\n    }\n  }\n\n  /**\n   * @override block ancestors/descendants of selected objects from being selected to prevent a circular object tree\n   */\n  canEnterGroup(object: FabricObject) {\n    if (\n      this.getObjects().some(\n        (o) => o.isDescendantOf(object) || object.isDescendantOf(o)\n      )\n    ) {\n      //  prevent circular object tree\n      log(\n        'error',\n        'ActiveSelection: circular object trees are not supported, this call has no effect'\n      );\n      return false;\n    }\n\n    return super.canEnterGroup(object);\n  }\n\n  /**\n   * Change an object so that it can be part of an active selection.\n   * this method is called by multiselectAdd from canvas code.\n   * @private\n   * @param {FabricObject} object\n   * @param {boolean} [removeParentTransform] true if object is in canvas coordinate plane\n   */\n  enterGroup(object: FabricObject, removeParentTransform?: boolean) {\n    console.log('enterGroup');\n    // This condition check that the object has currently a group, and the group\n    // is also its parent, meaning that is not in an active selection, but is\n    // in a normal group.\n    if (object.parent && object.parent === object.group) {\n      // Disconnect the object from the group functionalities, but keep the ref parent intact\n      // for later re-enter\n      object.parent._exitGroup(object);\n      // in this case the object is probably inside an active selection.\n    } else if (object.group && object.parent !== object.group) {\n      // in this case group.remove will also clear the old parent reference.\n      object.group.remove(object);\n    }\n    // enter the active selection from a render perspective\n    // the object will be in the objects array of both the ActiveSelection and the Group\n    // but referenced in the group's _activeObjects so that it won't be rendered twice.\n    this._enterGroup(object, removeParentTransform);\n  }\n\n  /**\n   * we want objects to retain their canvas ref when exiting instance\n   * @private\n   * @param {FabricObject} object\n   * @param {boolean} [removeParentTransform] true if object should exit group without applying group's transform to it\n   */\n  exitGroup(object: FabricObject, removeParentTransform?: boolean) {\n    this._exitGroup(object, removeParentTransform);\n    // return to parent\n    object.parent && object.parent._enterGroup(object, true);\n  }\n\n  /**\n   * @private\n   * @param {'added'|'removed'} type\n   * @param {FabricObject[]} targets\n   */\n  _onAfterObjectsChange(type: 'added' | 'removed', targets: FabricObject[]) {\n    super._onAfterObjectsChange(type, targets);\n    const groups = new Set<Group>();\n    targets.forEach((object) => {\n      const { parent } = object;\n      parent && groups.add(parent);\n    });\n    if (type === LAYOUT_TYPE_REMOVED) {\n      //  invalidate groups' layout and mark as dirty\n      groups.forEach((group) => {\n        group._onAfterObjectsChange(LAYOUT_TYPE_ADDED, targets);\n      });\n    } else {\n      //  mark groups as dirty\n      groups.forEach((group) => {\n        group._set('dirty', true);\n      });\n    }\n  }\n\n  /**\n   * @override remove all objects\n   */\n  onDeselect() {\n    this.removeAll();\n    return false;\n  }\n\n  /**\n   * Returns string representation of a group\n   * @return {String}\n   */\n  toString() {\n    return `#<ActiveSelection: (${this.complexity()})>`;\n  }\n\n  /**\n   * Decide if the object should cache or not. Create its own cache level\n   * objectCaching is a global flag, wins over everything\n   * needsItsOwnCache should be used when the object drawing method requires\n   * a cache step. None of the fabric classes requires it.\n   * Generally you do not cache objects in groups because the group outside is cached.\n   * @return {Boolean}\n   */\n  shouldCache() {\n    return false;\n  }\n\n  /**\n   * Check if this group or its parent group are caching, recursively up\n   * @return {Boolean}\n   */\n  isOnACache() {\n    return false;\n  }\n\n  /**\n   * Renders controls and borders for the object\n   * @param {CanvasRenderingContext2D} ctx Context to render on\n   * @param {Object} [styleOverride] properties to override the object style\n   * @param {Object} [childrenOverride] properties to override the children overrides\n   */\n  _renderControls(\n    ctx: CanvasRenderingContext2D,\n    styleOverride?: ControlRenderingStyleOverride,\n    childrenOverride?: ControlRenderingStyleOverride\n  ) {\n    ctx.save();\n    ctx.globalAlpha = this.isMoving ? this.borderOpacityWhenMoving : 1;\n    super._renderControls(ctx, styleOverride);\n    const options = {\n      hasControls: false,\n      ...childrenOverride,\n      forActiveSelection: true,\n    };\n    for (let i = 0; i < this._objects.length; i++) {\n      this._objects[i]._renderControls(ctx, options);\n    }\n    ctx.restore();\n  }\n\n  //  override the default behavior of `getText` to return a concatenated string of all text objects\\\n  // canvasX custoom method\n  getText(): any {\n    if (this.getObjects().length > 1) {\n      const textsArray = this.getObjects().map((item) => item.getText());\n      return textsArray.join('/n').trim();\n    } else {\n      return '';\n    }\n  }\n}\n\nclassRegistry.setClass(ActiveSelection);\nclassRegistry.setClass(ActiveSelection, 'activeSelection');\n","/**\n * Canvas 2D filter backend.\n */\nimport type { BaseFilter } from './BaseFilter';\nimport type { T2DPipelineState, TPipelineResources } from './typedefs';\n\nexport class Canvas2dFilterBackend {\n  /**\n   * Experimental. This object is a sort of repository of help layers used to avoid\n   * of recreating them during frequent filtering. If you are previewing a filter with\n   * a slider you probably do not want to create help layers every filter step.\n   * in this object there will be appended some canvases, created once, resized sometimes\n   * cleared never. Clearing is left to the developer.\n   **/\n  resources: TPipelineResources = {};\n\n  /**\n   * Apply a set of filters against a source image and draw the filtered output\n   * to the provided destination canvas.\n   *\n   * @param {EnhancedFilter} filters The filter to apply.\n   * @param {HTMLImageElement|HTMLCanvasElement} sourceElement The source to be filtered.\n   * @param {Number} sourceWidth The width of the source input.\n   * @param {Number} sourceHeight The height of the source input.\n   * @param {HTMLCanvasElement} targetCanvas The destination for filtered output to be drawn.\n   */\n  applyFilters(\n    filters: BaseFilter[],\n    sourceElement: CanvasImageSource,\n    sourceWidth: number,\n    sourceHeight: number,\n    targetCanvas: HTMLCanvasElement\n  ): T2DPipelineState | void {\n    const ctx = targetCanvas.getContext('2d');\n    if (!ctx) {\n      return;\n    }\n    ctx.drawImage(sourceElement, 0, 0, sourceWidth, sourceHeight);\n    const imageData = ctx.getImageData(0, 0, sourceWidth, sourceHeight);\n    const originalImageData = ctx.getImageData(0, 0, sourceWidth, sourceHeight);\n    const pipelineState: T2DPipelineState = {\n      sourceWidth,\n      sourceHeight,\n      imageData,\n      originalEl: sourceElement,\n      originalImageData,\n      canvasEl: targetCanvas,\n      ctx,\n      filterBackend: this,\n    };\n    filters.forEach((filter) => {\n      filter.applyTo(pipelineState);\n    });\n    const { imageData: imageDataPostFilter } = pipelineState;\n    if (\n      imageDataPostFilter.width !== sourceWidth ||\n      imageDataPostFilter.height !== sourceHeight\n    ) {\n      targetCanvas.width = imageDataPostFilter.width;\n      targetCanvas.height = imageDataPostFilter.height;\n    }\n    ctx.putImageData(imageDataPostFilter, 0, 0);\n    return pipelineState;\n  }\n}\n","import { config } from '../config';\nimport { createCanvasElement } from '../util/misc/dom';\nimport type {\n  TWebGLPipelineState,\n  TProgramCache,\n  TTextureCache,\n  TPipelineResources,\n} from './typedefs';\nimport type { BaseFilter } from './BaseFilter';\n\nexport class WebGLFilterBackend {\n  declare tileSize: number;\n\n  /**\n   * Define ...\n   **/\n  aPosition: Float32Array = new Float32Array([0, 0, 0, 1, 1, 0, 1, 1]);\n\n  /**\n   * If GLPut data is the fastest operation, or if forced, this buffer will be used\n   * to transfer the data back in the 2d logic\n   **/\n  declare imageBuffer?: ArrayBuffer;\n\n  declare canvas: HTMLCanvasElement;\n\n  /**\n   * The Webgl context that will execute the operations for filtering\n   **/\n  declare gl: WebGLRenderingContext;\n\n  /**\n   * Keyed map for shader cache\n   **/\n  declare programCache: TProgramCache;\n\n  /**\n   * Keyed map for texture cache\n   **/\n  declare textureCache: TTextureCache;\n\n  /**\n   * Contains GPU info for debug\n   **/\n  declare gpuInfo: any;\n\n  /**\n   * Experimental. This object is a sort of repository of help layers used to avoid\n   * of recreating them during frequent filtering. If you are previewing a filter with\n   * a slider you probably do not want to create help layers every filter step.\n   * in this object there will be appended some canvases, created once, resized sometimes\n   * cleared never. Clearing is left to the developer.\n   **/\n  resources: TPipelineResources = {};\n\n  constructor({ tileSize = config.textureSize } = {}) {\n    this.tileSize = tileSize;\n    this.setupGLContext(tileSize, tileSize);\n    this.captureGPUInfo();\n  }\n\n  /**\n   * Setup a WebGL context suitable for filtering, and bind any needed event handlers.\n   */\n  setupGLContext(width: number, height: number): void {\n    this.dispose();\n    this.createWebGLCanvas(width, height);\n  }\n\n  /**\n   * Create a canvas element and associated WebGL context and attaches them as\n   * class properties to the GLFilterBackend class.\n   */\n  createWebGLCanvas(width: number, height: number): void {\n    const canvas = createCanvasElement();\n    canvas.width = width;\n    canvas.height = height;\n    const glOptions = {\n        alpha: true,\n        premultipliedAlpha: false,\n        depth: false,\n        stencil: false,\n        antialias: false,\n      },\n      gl = canvas.getContext('webgl', glOptions) as WebGLRenderingContext;\n\n    if (!gl) {\n      return;\n    }\n    gl.clearColor(0, 0, 0, 0);\n    // this canvas can fire webglcontextlost and webglcontextrestored\n    this.canvas = canvas;\n    this.gl = gl;\n  }\n\n  /**\n   * Attempts to apply the requested filters to the source provided, drawing the filtered output\n   * to the provided target canvas.\n   *\n   * @param {Array} filters The filters to apply.\n   * @param {TexImageSource} source The source to be filtered.\n   * @param {Number} width The width of the source input.\n   * @param {Number} height The height of the source input.\n   * @param {HTMLCanvasElement} targetCanvas The destination for filtered output to be drawn.\n   * @param {String|undefined} cacheKey A key used to cache resources related to the source. If\n   * omitted, caching will be skipped.\n   */\n  applyFilters(\n    filters: BaseFilter[],\n    source: TexImageSource,\n    width: number,\n    height: number,\n    targetCanvas: HTMLCanvasElement,\n    cacheKey?: string\n  ): TWebGLPipelineState | undefined {\n    const gl = this.gl;\n    const ctx = targetCanvas.getContext('2d');\n    if (!gl || !ctx) {\n      return;\n    }\n    let cachedTexture;\n    if (cacheKey) {\n      cachedTexture = this.getCachedTexture(cacheKey, source);\n    }\n    const pipelineState: TWebGLPipelineState = {\n      originalWidth:\n        (source as HTMLImageElement).width ||\n        // @ts-expect-error is this a bug? should this be naturalWidth? or is this the pipeline state?\n        (source as HTMLImageElement).originalWidth ||\n        0,\n      originalHeight:\n        (source as HTMLImageElement).height ||\n        // @ts-expect-error is this a bug? should this be naturalHeight? or is this the pipeline state?\n        (source as HTMLImageElement).originalHeight ||\n        0,\n      sourceWidth: width,\n      sourceHeight: height,\n      destinationWidth: width,\n      destinationHeight: height,\n      context: gl,\n      sourceTexture: this.createTexture(\n        gl,\n        width,\n        height,\n        !cachedTexture ? source : undefined\n      ),\n      targetTexture: this.createTexture(gl, width, height),\n      originalTexture:\n        cachedTexture ||\n        this.createTexture(\n          gl,\n          width,\n          height,\n          !cachedTexture ? source : undefined\n        )!,\n      passes: filters.length,\n      webgl: true,\n      aPosition: this.aPosition,\n      programCache: this.programCache,\n      pass: 0,\n      filterBackend: this,\n      targetCanvas: targetCanvas,\n    };\n    const tempFbo = gl.createFramebuffer();\n    gl.bindFramebuffer(gl.FRAMEBUFFER, tempFbo);\n    filters.forEach((filter: any) => {\n      filter && filter.applyTo(pipelineState);\n    });\n    resizeCanvasIfNeeded(pipelineState);\n    this.copyGLTo2D(gl, pipelineState);\n    gl.bindTexture(gl.TEXTURE_2D, null);\n    gl.deleteTexture(pipelineState.sourceTexture);\n    gl.deleteTexture(pipelineState.targetTexture);\n    gl.deleteFramebuffer(tempFbo);\n    ctx.setTransform(1, 0, 0, 1, 0, 0);\n    return pipelineState;\n  }\n\n  /**\n   * Detach event listeners, remove references, and clean up caches.\n   */\n  dispose() {\n    if (this.canvas) {\n      // we are disposing, we don't care about the fact\n      // that the canvas shouldn't be null.\n      // @ts-expect-error disposing\n      this.canvas = null;\n      // @ts-expect-error disposing\n      this.gl = null;\n    }\n    this.clearWebGLCaches();\n  }\n\n  /**\n   * Wipe out WebGL-related caches.\n   */\n  clearWebGLCaches() {\n    this.programCache = {};\n    this.textureCache = {};\n  }\n\n  /**\n   * Create a WebGL texture object.\n   *\n   * Accepts specific dimensions to initialize the texture to or a source image.\n   *\n   * @param {WebGLRenderingContext} gl The GL context to use for creating the texture.\n   * @param {number} width The width to initialize the texture at.\n   * @param {number} height The height to initialize the texture.\n   * @param {TexImageSource} textureImageSource A source for the texture data.\n   * @param {number} filter gl.NEAREST default or gl.LINEAR filters for the texture.\n   * This filter is very useful for LUTs filters. If you need interpolation use gl.LINEAR\n   * @returns {WebGLTexture}\n   */\n  createTexture(\n    gl: WebGLRenderingContext,\n    width: number,\n    height: number,\n    textureImageSource?: TexImageSource,\n    filter?:\n      | WebGLRenderingContextBase['NEAREST']\n      | WebGLRenderingContextBase['LINEAR']\n  ) {\n    const {\n      NEAREST,\n      TEXTURE_2D,\n      RGBA,\n      UNSIGNED_BYTE,\n      CLAMP_TO_EDGE,\n      TEXTURE_MAG_FILTER,\n      TEXTURE_MIN_FILTER,\n      TEXTURE_WRAP_S,\n      TEXTURE_WRAP_T,\n    } = gl;\n    const texture = gl.createTexture();\n    gl.bindTexture(TEXTURE_2D, texture);\n    gl.texParameteri(TEXTURE_2D, TEXTURE_MAG_FILTER, filter || NEAREST);\n    gl.texParameteri(TEXTURE_2D, TEXTURE_MIN_FILTER, filter || NEAREST);\n    gl.texParameteri(TEXTURE_2D, TEXTURE_WRAP_S, CLAMP_TO_EDGE);\n    gl.texParameteri(TEXTURE_2D, TEXTURE_WRAP_T, CLAMP_TO_EDGE);\n    if (textureImageSource) {\n      gl.texImage2D(\n        TEXTURE_2D,\n        0,\n        RGBA,\n        RGBA,\n        UNSIGNED_BYTE,\n        textureImageSource\n      );\n    } else {\n      gl.texImage2D(\n        TEXTURE_2D,\n        0,\n        RGBA,\n        width,\n        height,\n        0,\n        RGBA,\n        UNSIGNED_BYTE,\n        null\n      );\n    }\n    return texture;\n  }\n\n  /**\n   * Can be optionally used to get a texture from the cache array\n   *\n   * If an existing texture is not found, a new texture is created and cached.\n   *\n   * @param {String} uniqueId A cache key to use to find an existing texture.\n   * @param {HTMLImageElement|HTMLCanvasElement} textureImageSource A source to use to create the\n   * texture cache entry if one does not already exist.\n   */\n  getCachedTexture(\n    uniqueId: string,\n    textureImageSource: TexImageSource,\n    filter?:\n      | WebGLRenderingContextBase['NEAREST']\n      | WebGLRenderingContextBase['LINEAR']\n  ): WebGLTexture | null {\n    const { textureCache } = this;\n    if (textureCache[uniqueId]) {\n      return textureCache[uniqueId];\n    } else {\n      const texture = this.createTexture(\n        this.gl,\n        (textureImageSource as HTMLImageElement).width,\n        (textureImageSource as HTMLImageElement).height,\n        textureImageSource,\n        filter\n      );\n      if (texture) {\n        textureCache[uniqueId] = texture;\n      }\n      return texture;\n    }\n  }\n\n  /**\n   * Clear out cached resources related to a source image that has been\n   * filtered previously.\n   *\n   * @param {String} cacheKey The cache key provided when the source image was filtered.\n   */\n  evictCachesForKey(cacheKey: string) {\n    if (this.textureCache[cacheKey]) {\n      this.gl.deleteTexture(this.textureCache[cacheKey]);\n      delete this.textureCache[cacheKey];\n    }\n  }\n\n  /**\n   * Copy an input WebGL canvas on to an output 2D canvas.\n   *\n   * The WebGL canvas is assumed to be upside down, with the top-left pixel of the\n   * desired output image appearing in the bottom-left corner of the WebGL canvas.\n   *\n   * @param {WebGLRenderingContext} sourceContext The WebGL context to copy from.\n   * @param {Object} pipelineState The 2D target canvas to copy on to.\n   */\n  copyGLTo2D(gl: WebGLRenderingContext, pipelineState: TWebGLPipelineState) {\n    const glCanvas = gl.canvas,\n      targetCanvas = pipelineState.targetCanvas,\n      ctx = targetCanvas.getContext('2d');\n    if (!ctx) {\n      return;\n    }\n    ctx.translate(0, targetCanvas.height); // move it down again\n    ctx.scale(1, -1); // vertical flip\n    // where is my image on the big glcanvas?\n    const sourceY = glCanvas.height - targetCanvas.height;\n    ctx.drawImage(\n      glCanvas,\n      0,\n      sourceY,\n      targetCanvas.width,\n      targetCanvas.height,\n      0,\n      0,\n      targetCanvas.width,\n      targetCanvas.height\n    );\n  }\n\n  /**\n   * Copy an input WebGL canvas on to an output 2D canvas using 2d canvas' putImageData\n   * API. Measurably faster than using ctx.drawImage in Firefox (version 54 on OSX Sierra).\n   *\n   * @param {WebGLRenderingContext} sourceContext The WebGL context to copy from.\n   * @param {HTMLCanvasElement} targetCanvas The 2D target canvas to copy on to.\n   * @param {Object} pipelineState The 2D target canvas to copy on to.\n   */\n  copyGLTo2DPutImageData(\n    this: Required<WebGLFilterBackend>,\n    gl: WebGLRenderingContext,\n    pipelineState: TWebGLPipelineState\n  ) {\n    const targetCanvas = pipelineState.targetCanvas,\n      ctx = targetCanvas.getContext('2d'),\n      dWidth = pipelineState.destinationWidth,\n      dHeight = pipelineState.destinationHeight,\n      numBytes = dWidth * dHeight * 4;\n    if (!ctx) {\n      return;\n    }\n    const u8 = new Uint8Array(this.imageBuffer, 0, numBytes);\n    const u8Clamped = new Uint8ClampedArray(this.imageBuffer, 0, numBytes);\n\n    gl.readPixels(0, 0, dWidth, dHeight, gl.RGBA, gl.UNSIGNED_BYTE, u8);\n    const imgData = new ImageData(u8Clamped, dWidth, dHeight);\n    ctx.putImageData(imgData, 0, 0);\n  }\n\n  /**\n   * Attempt to extract GPU information strings from a WebGL context.\n   *\n   * Useful information when debugging or blacklisting specific GPUs.\n   *\n   * @returns {Object} A GPU info object with renderer and vendor strings.\n   */\n  captureGPUInfo() {\n    if (this.gpuInfo) {\n      return this.gpuInfo;\n    }\n    const gl = this.gl,\n      gpuInfo = { renderer: '', vendor: '' };\n    if (!gl) {\n      return gpuInfo;\n    }\n    const ext = gl.getExtension('WEBGL_debug_renderer_info');\n    if (ext) {\n      const renderer = gl.getParameter(ext.UNMASKED_RENDERER_WEBGL);\n      const vendor = gl.getParameter(ext.UNMASKED_VENDOR_WEBGL);\n      if (renderer) {\n        gpuInfo.renderer = renderer.toLowerCase();\n      }\n      if (vendor) {\n        gpuInfo.vendor = vendor.toLowerCase();\n      }\n    }\n    this.gpuInfo = gpuInfo;\n    return gpuInfo;\n  }\n}\n\nfunction resizeCanvasIfNeeded(pipelineState: TWebGLPipelineState): void {\n  const targetCanvas = pipelineState.targetCanvas,\n    width = targetCanvas.width,\n    height = targetCanvas.height,\n    dWidth = pipelineState.destinationWidth,\n    dHeight = pipelineState.destinationHeight;\n\n  if (width !== dWidth || height !== dHeight) {\n    targetCanvas.width = dWidth;\n    targetCanvas.height = dHeight;\n  }\n}\n","import { config } from '../config';\nimport { getEnv } from '../env';\nimport { createCanvasElement } from '../util/misc/dom';\nimport { Canvas2dFilterBackend } from './Canvas2dFilterBackend';\nimport { WebGLFilterBackend } from './WebGLFilterBackend';\n\nexport type FilterBackend = WebGLFilterBackend | Canvas2dFilterBackend;\n\nlet filterBackend: FilterBackend;\n\n/**\n * Verifies if it is possible to initialize webgl or fallback on a canvas2d filtering backend\n */\nexport function initFilterBackend(): FilterBackend {\n  const { WebGLProbe } = getEnv();\n  WebGLProbe.queryWebGL(createCanvasElement());\n  if (config.enableGLFiltering && WebGLProbe.isSupported(config.textureSize)) {\n    return new WebGLFilterBackend({ tileSize: config.textureSize });\n  } else {\n    return new Canvas2dFilterBackend();\n  }\n}\n\n/**\n * Get the current fabricJS filter backend  or initialize one if not available yet\n * @param [strict] pass `true` to create the backend if it wasn't created yet (default behavior),\n * pass `false` to get the backend ref without mutating it\n */\nexport function getFilterBackend(strict = true): FilterBackend {\n  if (!filterBackend && strict) {\n    filterBackend = initFilterBackend();\n  }\n  return filterBackend;\n}\n\nexport function setFilterBackend(backend: FilterBackend) {\n  filterBackend = backend;\n}\n","import { getFabricDocument, getEnv } from '../env';\nimport type { BaseFilter } from '../filters/BaseFilter';\nimport { getFilterBackend } from '../filters/FilterBackend';\nimport { SHARED_ATTRIBUTES } from '../parser/attributes';\nimport { parseAttributes } from '../parser/parseAttributes';\nimport type {\n  TClassProperties,\n  TCrossOrigin,\n  TSize,\n  Abortable,\n  TOptions,\n} from '../typedefs';\nimport { uid } from '../util/internals/uid';\nimport { createCanvasElement } from '../util/misc/dom';\nimport { findScaleToCover, findScaleToFit } from '../util/misc/findScaleTo';\nimport type { LoadImageOptions } from '../util/misc/objectEnlive';\nimport {\n  enlivenObjectEnlivables,\n  enlivenObjects,\n  loadImage,\n} from '../util/misc/objectEnlive';\nimport { parsePreserveAspectRatioAttribute } from '../util/misc/svgParsing';\nimport { classRegistry } from '../ClassRegistry';\nimport { FabricObject, cacheProperties } from './Object/FabricObject';\nimport type { FabricObjectProps, SerializedObjectProps } from './Object/types';\nimport type { ObjectEvents } from '../EventTypeDefs';\nimport { WebGLFilterBackend } from '../filters/WebGLFilterBackend';\nimport { NONE } from '../constants';\nimport { getDocumentFromElement } from '../util/dom_misc';\nimport type { CSSRules } from '../parser/typedefs';\nimport type { Resize } from '../filters/Resize';\nimport type { TCachedFabricObject } from './Object/Object';\nimport { log } from '../util/internals/console';\n\n// @todo Would be nice to have filtering code not imported directly.\n\nexport type ImageSource =\n  | HTMLImageElement\n  | HTMLVideoElement\n  | HTMLCanvasElement;\n\ninterface UniqueImageProps {\n  srcFromAttribute: boolean;\n  minimumScaleTrigger: number;\n  cropX: number;\n  cropY: number;\n  imageSmoothing: boolean;\n  filters: BaseFilter[];\n  resizeFilter?: Resize;\n}\n\nexport const imageDefaultValues: Partial<TClassProperties<FabricImage>> = {\n  strokeWidth: 0,\n  srcFromAttribute: false,\n  minimumScaleTrigger: 0.5,\n  cropX: 0,\n  cropY: 0,\n  imageSmoothing: true,\n};\n\nexport interface SerializedImageProps extends SerializedObjectProps {\n  src: string;\n  crossOrigin: TCrossOrigin;\n  filters: any[];\n  resizeFilter?: any;\n  cropX: number;\n  cropY: number;\n}\n\nexport interface ImageProps extends FabricObjectProps, UniqueImageProps {}\n\nconst IMAGE_PROPS = ['cropX', 'cropY'] as const;\n\n/**\n * @tutorial {@link http://fabricjs.com/fabric-intro-part-1#images}\n */\nexport class FabricImage<\n    Props extends TOptions<ImageProps> = Partial<ImageProps>,\n    SProps extends SerializedImageProps = SerializedImageProps,\n    EventSpec extends ObjectEvents = ObjectEvents\n  >\n  extends FabricObject<Props, SProps, EventSpec>\n  implements ImageProps\n{\n  /**\n   * When calling {@link FabricImage.getSrc}, return value from element src with `element.getAttribute('src')`.\n   * This allows for relative urls as image src.\n   * @since 2.7.0\n   * @type Boolean\n   * @default false\n   */\n  declare srcFromAttribute: boolean;\n\n  /**\n   * private\n   * contains last value of scaleX to detect\n   * if the Image got resized after the last Render\n   * @type Number\n   */\n  protected _lastScaleX = 1;\n\n  /**\n   * private\n   * contains last value of scaleY to detect\n   * if the Image got resized after the last Render\n   * @type Number\n   */\n  protected _lastScaleY = 1;\n\n  /**\n   * private\n   * contains last value of scaling applied by the apply filter chain\n   * @type Number\n   */\n  protected _filterScalingX = 1;\n\n  /**\n   * private\n   * contains last value of scaling applied by the apply filter chain\n   * @type Number\n   */\n  protected _filterScalingY = 1;\n\n  /**\n   * minimum scale factor under which any resizeFilter is triggered to resize the image\n   * 0 will disable the automatic resize. 1 will trigger automatically always.\n   * number bigger than 1 are not implemented yet.\n   * @type Number\n   */\n  declare minimumScaleTrigger: number;\n\n  /**\n   * key used to retrieve the texture representing this image\n   * @since 2.0.0\n   * @type String\n   * @default\n   */\n  declare cacheKey: string;\n\n  /**\n   * Image crop in pixels from original image size.\n   * @since 2.0.0\n   * @type Number\n   * @default\n   */\n  declare cropX: number;\n\n  /**\n   * Image crop in pixels from original image size.\n   * @since 2.0.0\n   * @type Number\n   * @default\n   */\n  declare cropY: number;\n\n  /**\n   * Indicates whether this canvas will use image smoothing when painting this image.\n   * Also influence if the cacheCanvas for this image uses imageSmoothing\n   * @since 4.0.0-beta.11\n   * @type Boolean\n   * @default\n   */\n  declare imageSmoothing: boolean;\n\n  declare preserveAspectRatio: string;\n\n  protected declare src: string;\n\n  declare filters: BaseFilter[];\n  declare resizeFilter: Resize;\n\n  declare _element: ImageSource;\n  declare _filteredEl?: HTMLCanvasElement;\n  declare _originalElement: ImageSource;\n\n  static type = 'Image';\n\n  static cacheProperties = [...cacheProperties, ...IMAGE_PROPS];\n\n  static ownDefaults = imageDefaultValues;\n\n  static getDefaults(): Record<string, any> {\n    return {\n      ...super.getDefaults(),\n      ...FabricImage.ownDefaults,\n    };\n  }\n  /**\n   * Constructor\n   * Image can be initialized with any canvas drawable or a string.\n   * The string should be a url and will be loaded as an image.\n   * Canvas and Image element work out of the box, while videos require extra code to work.\n   * Please check video element events for seeking.\n   * @param {ImageSource | string} element Image element\n   * @param {Object} [options] Options object\n   */\n  constructor(elementId: string, options?: Props);\n  constructor(element: ImageSource, options?: Props);\n  constructor(arg0: ImageSource | string, options: Props = {} as Props) {\n    super({ filters: [], ...options });\n    this.cacheKey = `texture${uid()}`;\n    this.setElement(\n      typeof arg0 === 'string'\n        ? ((\n            (this.canvas && getDocumentFromElement(this.canvas.getElement())) ||\n            getFabricDocument()\n          ).getElementById(arg0) as ImageSource)\n        : arg0,\n      options\n    );\n  }\n\n  /**\n   * Returns image element which this instance if based on\n   */\n  getElement() {\n    return this._element;\n  }\n\n  /**\n   * Sets image element for this instance to a specified one.\n   * If filters defined they are applied to new image.\n   * You might need to call `canvas.renderAll` and `object.setCoords` after replacing, to render new image and update controls area.\n   * @param {HTMLImageElement} element\n   * @param {Partial<TSize>} [size] Options object\n   */\n  setElement(element: ImageSource, size: Partial<TSize> = {}) {\n    this.removeTexture(this.cacheKey);\n    this.removeTexture(`${this.cacheKey}_filtered`);\n    this._element = element;\n    this._originalElement = element;\n    this._setWidthHeight(size);\n    // element.classList.add(FabricImage.CSS_CANVAS);\n    if (this.filters.length !== 0) {\n      this.applyFilters();\n    }\n    // resizeFilters work on the already filtered copy.\n    // we need to apply resizeFilters AFTER normal filters.\n    // applyResizeFilters is run more often than normal filters\n    // and is triggered by user interactions rather than dev code\n    if (this.resizeFilter) {\n      this.applyResizeFilters();\n    }\n  }\n\n  /**\n   * Delete a single texture if in webgl mode\n   */\n  removeTexture(key: string) {\n    const backend = getFilterBackend(false);\n    if (backend instanceof WebGLFilterBackend) {\n      backend.evictCachesForKey(key);\n    }\n  }\n\n  /**\n   * Delete textures, reference to elements and eventually JSDOM cleanup\n   */\n  dispose() {\n    super.dispose();\n    this.removeTexture(this.cacheKey);\n    this.removeTexture(`${this.cacheKey}_filtered`);\n    this._cacheContext = null;\n    (\n      ['_originalElement', '_element', '_filteredEl', '_cacheCanvas'] as const\n    ).forEach((elementKey) => {\n      const el = this[elementKey];\n      el && getEnv().dispose(el);\n      // @ts-expect-error disposing\n      this[elementKey] = undefined;\n    });\n  }\n\n  /**\n   * Get the crossOrigin value (of the corresponding image element)\n   */\n  getCrossOrigin(): string | null {\n    return (\n      this._originalElement &&\n      ((this._originalElement as any).crossOrigin || null)\n    );\n  }\n\n  /**\n   * Returns original size of an image\n   */\n  getOriginalSize() {\n    const element = this.getElement() as any;\n    if (!element) {\n      return {\n        width: 0,\n        height: 0,\n      };\n    }\n    return {\n      width: element.naturalWidth || element.width,\n      height: element.naturalHeight || element.height,\n    };\n  }\n\n  /**\n   * @private\n   * @param {CanvasRenderingContext2D} ctx Context to render on\n   */\n  _stroke(ctx: CanvasRenderingContext2D) {\n    if (!this.stroke || this.strokeWidth === 0) {\n      return;\n    }\n    const w = this.width / 2,\n      h = this.height / 2;\n    ctx.beginPath();\n    ctx.moveTo(-w, -h);\n    ctx.lineTo(w, -h);\n    ctx.lineTo(w, h);\n    ctx.lineTo(-w, h);\n    ctx.lineTo(-w, -h);\n    ctx.closePath();\n  }\n\n  /**\n   * Returns object representation of an instance\n   * @param {Array} [propertiesToInclude] Any properties that you might want to additionally include in the output\n   * @return {Object} Object representation of an instance\n   */\n  toObject<\n    T extends Omit<Props & TClassProperties<this>, keyof SProps>,\n    K extends keyof T = never\n  >(propertiesToInclude: K[] = []): Pick<T, K> & SProps {\n    const filters: Record<string, any>[] = [];\n    this.filters.forEach((filterObj) => {\n      filterObj && filters.push(filterObj.toObject());\n    });\n    return {\n      ...super.toObject([...IMAGE_PROPS, ...propertiesToInclude]),\n      src: this.getSrc(),\n      crossOrigin: this.getCrossOrigin(),\n      filters,\n      ...(this.resizeFilter\n        ? { resizeFilter: this.resizeFilter.toObject() }\n        : {}),\n    };\n  }\n\n  /**\n   * Returns true if an image has crop applied, inspecting values of cropX,cropY,width,height.\n   * @return {Boolean}\n   */\n  hasCrop() {\n    return (\n      !!this.cropX ||\n      !!this.cropY ||\n      this.width < this._element.width ||\n      this.height < this._element.height\n    );\n  }\n\n  /**\n   * Returns svg representation of an instance\n   * @return {string[]} an array of strings with the specific svg representation\n   * of the instance\n   */\n  _toSVG() {\n    const imageMarkup: string[] = [],\n      element = this._element,\n      x = -this.width / 2,\n      y = -this.height / 2;\n    let svgString: string[] = [],\n      strokeSvg: string[] = [],\n      clipPath = '',\n      imageRendering = '';\n    if (!element) {\n      return [];\n    }\n    if (this.hasCrop()) {\n      const clipPathId = uid();\n      svgString.push(\n        '<clipPath id=\"imageCrop_' + clipPathId + '\">\\n',\n        '\\t<rect x=\"' +\n          x +\n          '\" y=\"' +\n          y +\n          '\" width=\"' +\n          this.width +\n          '\" height=\"' +\n          this.height +\n          '\" />\\n',\n        '</clipPath>\\n'\n      );\n      clipPath = ' clip-path=\"url(#imageCrop_' + clipPathId + ')\" ';\n    }\n    if (!this.imageSmoothing) {\n      imageRendering = ' image-rendering=\"optimizeSpeed\"';\n    }\n    imageMarkup.push(\n      '\\t<image ',\n      'COMMON_PARTS',\n      `xlink:href=\"${this.getSvgSrc(true)}\" x=\"${x - this.cropX}\" y=\"${\n        y - this.cropY\n        // we're essentially moving origin of transformation from top/left corner to the center of the shape\n        // by wrapping it in container <g> element with actual transformation, then offsetting object to the top/left\n        // so that object's center aligns with container's left/top\n      }\" width=\"${\n        element.width || (element as HTMLImageElement).naturalWidth\n      }\" height=\"${\n        element.height || (element as HTMLImageElement).naturalHeight\n      }\"${imageRendering}${clipPath}></image>\\n`\n    );\n\n    if (this.stroke || this.strokeDashArray) {\n      const origFill = this.fill;\n      this.fill = null;\n      strokeSvg = [\n        `\\t<rect x=\"${x}\" y=\"${y}\" width=\"${this.width}\" height=\"${\n          this.height\n        }\" style=\"${this.getSvgStyles()}\" />\\n`,\n      ];\n      this.fill = origFill;\n    }\n    if (this.paintFirst !== 'fill') {\n      svgString = svgString.concat(strokeSvg, imageMarkup);\n    } else {\n      svgString = svgString.concat(imageMarkup, strokeSvg);\n    }\n    return svgString;\n  }\n\n  /**\n   * Returns source of an image\n   * @param {Boolean} filtered indicates if the src is needed for svg\n   * @return {String} Source of an image\n   */\n  getSrc(filtered?: boolean): string {\n    const element = filtered ? this._element : this._originalElement;\n    if (element) {\n      if ((element as HTMLCanvasElement).toDataURL) {\n        return (element as HTMLCanvasElement).toDataURL();\n      }\n\n      if (this.srcFromAttribute) {\n        return element.getAttribute('src') || '';\n      } else {\n        return (element as HTMLImageElement).src;\n      }\n    } else {\n      return this.src || '';\n    }\n  }\n\n  /**\n   * Alias for getSrc\n   * @param filtered\n   * @deprecated\n   */\n  getSvgSrc(filtered?: boolean) {\n    return this.getSrc(filtered);\n  }\n\n  /**\n   * Loads and sets source of an image\\\n   * **IMPORTANT**: It is recommended to abort loading tasks before calling this method to prevent race conditions and unnecessary networking\n   * @param {String} src Source string (URL)\n   * @param {LoadImageOptions} [options] Options object\n   */\n  setSrc(src: string, { crossOrigin, signal }: LoadImageOptions = {}) {\n    return loadImage(src, { crossOrigin, signal }).then((img) => {\n      typeof crossOrigin !== 'undefined' && this.set({ crossOrigin });\n      this.setElement(img);\n    });\n  }\n\n  /**\n   * Returns string representation of an instance\n   * @return {String} String representation of an instance\n   */\n  toString() {\n    return `#<Image: { src: \"${this.getSrc()}\" }>`;\n  }\n\n  applyResizeFilters() {\n    const filter = this.resizeFilter,\n      minimumScale = this.minimumScaleTrigger,\n      objectScale = this.getTotalObjectScaling(),\n      scaleX = objectScale.x,\n      scaleY = objectScale.y,\n      elementToFilter = this._filteredEl || this._originalElement;\n    if (this.group) {\n      this.set('dirty', true);\n    }\n    if (!filter || (scaleX > minimumScale && scaleY > minimumScale)) {\n      this._element = elementToFilter;\n      this._filterScalingX = 1;\n      this._filterScalingY = 1;\n      this._lastScaleX = scaleX;\n      this._lastScaleY = scaleY;\n      return;\n    }\n    const canvasEl = createCanvasElement(),\n      sourceWidth = elementToFilter.width,\n      sourceHeight = elementToFilter.height;\n    canvasEl.width = sourceWidth;\n    canvasEl.height = sourceHeight;\n    this._element = canvasEl;\n    this._lastScaleX = filter.scaleX = scaleX;\n    this._lastScaleY = filter.scaleY = scaleY;\n    getFilterBackend().applyFilters(\n      [filter as BaseFilter],\n      elementToFilter,\n      sourceWidth,\n      sourceHeight,\n      this._element\n    );\n    this._filterScalingX = canvasEl.width / this._originalElement.width;\n    this._filterScalingY = canvasEl.height / this._originalElement.height;\n  }\n\n  /**\n   * Applies filters assigned to this image (from \"filters\" array) or from filter param\n   * @method applyFilters\n   * @param {Array} filters to be applied\n   * @param {Boolean} forResizing specify if the filter operation is a resize operation\n   */\n  applyFilters(filters: BaseFilter[] = this.filters || []) {\n    filters = filters.filter((filter) => filter && !filter.isNeutralState());\n    this.set('dirty', true);\n\n    // needs to clear out or WEBGL will not resize correctly\n    this.removeTexture(`${this.cacheKey}_filtered`);\n\n    if (filters.length === 0) {\n      this._element = this._originalElement;\n      // this is unsafe and needs to be rethinkend\n      this._filteredEl = undefined;\n      this._filterScalingX = 1;\n      this._filterScalingY = 1;\n      return;\n    }\n\n    const imgElement = this._originalElement,\n      sourceWidth =\n        (imgElement as HTMLImageElement).naturalWidth || imgElement.width,\n      sourceHeight =\n        (imgElement as HTMLImageElement).naturalHeight || imgElement.height;\n\n    if (this._element === this._originalElement) {\n      // if the _element a reference to _originalElement\n      // we need to create a new element to host the filtered pixels\n      const canvasEl = createCanvasElement();\n      canvasEl.width = sourceWidth;\n      canvasEl.height = sourceHeight;\n      this._element = canvasEl;\n      this._filteredEl = canvasEl;\n    } else if (this._filteredEl) {\n      // if the _element is it own element,\n      // and we also have a _filteredEl, then we clean up _filteredEl\n      // and we assign it to _element.\n      // in this way we invalidate the eventual old resize filtered element\n      this._element = this._filteredEl;\n      this._filteredEl\n        .getContext('2d')!\n        .clearRect(0, 0, sourceWidth, sourceHeight);\n      // we also need to resize again at next renderAll, so remove saved _lastScaleX/Y\n      this._lastScaleX = 1;\n      this._lastScaleY = 1;\n    }\n    getFilterBackend().applyFilters(\n      filters,\n      this._originalElement,\n      sourceWidth,\n      sourceHeight,\n      this._element as HTMLCanvasElement\n    );\n    if (\n      this._originalElement.width !== this._element.width ||\n      this._originalElement.height !== this._element.height\n    ) {\n      this._filterScalingX = this._element.width / this._originalElement.width;\n      this._filterScalingY =\n        this._element.height / this._originalElement.height;\n    }\n  }\n\n  /**\n   * @private\n   * @param {CanvasRenderingContext2D} ctx Context to render on\n   */\n  _render(ctx: CanvasRenderingContext2D) {\n    ctx.imageSmoothingEnabled = this.imageSmoothing;\n    if (this.isMoving !== true && this.resizeFilter && this._needsResize()) {\n      this.applyResizeFilters();\n    }\n    this._stroke(ctx);\n    this._renderPaintInOrder(ctx);\n  }\n\n  /**\n   * Paint the cached copy of the object on the target context.\n   * it will set the imageSmoothing for the draw operation\n   * @param {CanvasRenderingContext2D} ctx Context to render on\n   */\n  drawCacheOnCanvas(\n    this: TCachedFabricObject<FabricImage>,\n    ctx: CanvasRenderingContext2D\n  ) {\n    ctx.imageSmoothingEnabled = this.imageSmoothing;\n    // cant use ts-expect-error because of ts 5.3 cross check\n    // @ts-ignore TS doesn't respect this type casting\n    super.drawCacheOnCanvas(ctx);\n  }\n\n  /**\n   * Decide if the object should cache or not. Create its own cache level\n   * needsItsOwnCache should be used when the object drawing method requires\n   * a cache step. None of the fabric classes requires it.\n   * Generally you do not cache objects in groups because the group outside is cached.\n   * This is the special image version where we would like to avoid caching where possible.\n   * Essentially images do not benefit from caching. They may require caching, and in that\n   * case we do it. Also caching an image usually ends in a loss of details.\n   * A full performance audit should be done.\n   * @return {Boolean}\n   */\n  shouldCache() {\n    return this.needsItsOwnCache();\n  }\n\n  _renderFill(ctx: CanvasRenderingContext2D) {\n    const elementToDraw = this._element;\n    if (!elementToDraw) {\n      return;\n    }\n    const scaleX = this._filterScalingX,\n      scaleY = this._filterScalingY,\n      w = this.width,\n      h = this.height,\n      // crop values cannot be lesser than 0.\n      cropX = Math.max(this.cropX, 0),\n      cropY = Math.max(this.cropY, 0),\n      elWidth =\n        (elementToDraw as HTMLImageElement).naturalWidth || elementToDraw.width,\n      elHeight =\n        (elementToDraw as HTMLImageElement).naturalHeight ||\n        elementToDraw.height,\n      sX = cropX * scaleX,\n      sY = cropY * scaleY,\n      // the width height cannot exceed element width/height, starting from the crop offset.\n      sW = Math.min(w * scaleX, elWidth - sX),\n      sH = Math.min(h * scaleY, elHeight - sY),\n      x = -w / 2,\n      y = -h / 2,\n      maxDestW = Math.min(w, elWidth / scaleX - cropX),\n      maxDestH = Math.min(h, elHeight / scaleY - cropY);\n\n    elementToDraw &&\n      ctx.drawImage(elementToDraw, sX, sY, sW, sH, x, y, maxDestW, maxDestH);\n  }\n\n  /**\n   * needed to check if image needs resize\n   * @private\n   */\n  _needsResize() {\n    const scale = this.getTotalObjectScaling();\n    return scale.x !== this._lastScaleX || scale.y !== this._lastScaleY;\n  }\n\n  /**\n   * @private\n   * @deprecated unused\n   */\n  _resetWidthHeight() {\n    this.set(this.getOriginalSize());\n  }\n\n  /**\n   * @private\n   * Set the width and the height of the image object, using the element or the\n   * options.\n   */\n  _setWidthHeight({ width, height }: Partial<TSize> = {}) {\n    const size = this.getOriginalSize();\n    this.width = width || size.width;\n    this.height = height || size.height;\n  }\n\n  /**\n   * Calculate offset for center and scale factor for the image in order to respect\n   * the preserveAspectRatio attribute\n   * @private\n   */\n  parsePreserveAspectRatioAttribute() {\n    const pAR = parsePreserveAspectRatioAttribute(\n        this.preserveAspectRatio || ''\n      ),\n      pWidth = this.width,\n      pHeight = this.height,\n      parsedAttributes = { width: pWidth, height: pHeight };\n    let rWidth = this._element.width,\n      rHeight = this._element.height,\n      scaleX = 1,\n      scaleY = 1,\n      offsetLeft = 0,\n      offsetTop = 0,\n      cropX = 0,\n      cropY = 0,\n      offset;\n\n    if (pAR && (pAR.alignX !== NONE || pAR.alignY !== NONE)) {\n      if (pAR.meetOrSlice === 'meet') {\n        scaleX = scaleY = findScaleToFit(this._element, parsedAttributes);\n        offset = (pWidth - rWidth * scaleX) / 2;\n        if (pAR.alignX === 'Min') {\n          offsetLeft = -offset;\n        }\n        if (pAR.alignX === 'Max') {\n          offsetLeft = offset;\n        }\n        offset = (pHeight - rHeight * scaleY) / 2;\n        if (pAR.alignY === 'Min') {\n          offsetTop = -offset;\n        }\n        if (pAR.alignY === 'Max') {\n          offsetTop = offset;\n        }\n      }\n      if (pAR.meetOrSlice === 'slice') {\n        scaleX = scaleY = findScaleToCover(this._element, parsedAttributes);\n        offset = rWidth - pWidth / scaleX;\n        if (pAR.alignX === 'Mid') {\n          cropX = offset / 2;\n        }\n        if (pAR.alignX === 'Max') {\n          cropX = offset;\n        }\n        offset = rHeight - pHeight / scaleY;\n        if (pAR.alignY === 'Mid') {\n          cropY = offset / 2;\n        }\n        if (pAR.alignY === 'Max') {\n          cropY = offset;\n        }\n        rWidth = pWidth / scaleX;\n        rHeight = pHeight / scaleY;\n      }\n    } else {\n      scaleX = pWidth / rWidth;\n      scaleY = pHeight / rHeight;\n    }\n    return {\n      width: rWidth,\n      height: rHeight,\n      scaleX,\n      scaleY,\n      offsetLeft,\n      offsetTop,\n      cropX,\n      cropY,\n    };\n  }\n\n  /**\n   * Default CSS class name for canvas\n   * @static\n   * @type String\n   * @default\n   */\n  static CSS_CANVAS = 'canvas-img';\n\n  /**\n   * List of attribute names to account for when parsing SVG element (used by {@link FabricImage.fromElement})\n   * @static\n   * @see {@link http://www.w3.org/TR/SVG/struct.html#ImageElement}\n   */\n  static ATTRIBUTE_NAMES = [\n    ...SHARED_ATTRIBUTES,\n    'x',\n    'y',\n    'width',\n    'height',\n    'preserveAspectRatio',\n    'xlink:href',\n    'crossOrigin',\n    'image-rendering',\n  ];\n\n  /**\n   * Creates an instance of FabricImage from its object representation\n   * @static\n   * @param {Object} object Object to create an instance from\n   * @param {object} [options] Options object\n   * @param {AbortSignal} [options.signal] handle aborting, see https://developer.mozilla.org/en-US/docs/Web/API/AbortController/signal\n   * @returns {Promise<FabricImage>}\n   */\n  static fromObject<T extends TOptions<SerializedImageProps>>(\n    { filters: f, resizeFilter: rf, src, crossOrigin, type, ...object }: T,\n    options?: Abortable\n  ) {\n    return Promise.all([\n      loadImage(src!, { ...options, crossOrigin }),\n      f && enlivenObjects<BaseFilter>(f, options),\n      // TODO: redundant - handled by enlivenObjectEnlivables\n      rf && enlivenObjects<BaseFilter>([rf], options),\n      enlivenObjectEnlivables(object, options),\n    ]).then(([el, filters = [], [resizeFilter] = [], hydratedProps = {}]) => {\n      return new this(el, {\n        ...object,\n        // TODO: this creates a difference between image creation and restoring from JSON\n        src,\n        filters,\n        resizeFilter,\n        ...hydratedProps,\n      });\n    });\n  }\n\n  /**\n   * Creates an instance of Image from an URL string\n   * @static\n   * @param {String} url URL to create an image from\n   * @param {LoadImageOptions} [options] Options object\n   * @returns {Promise<FabricImage>}\n   */\n  static fromURL<T extends TOptions<ImageProps>>(\n    url: string,\n    { crossOrigin = null, signal }: LoadImageOptions = {},\n    imageOptions?: T\n  ): Promise<FabricImage> {\n    return loadImage(url, { crossOrigin, signal }).then(\n      (img) => new this(img, imageOptions)\n    );\n  }\n\n  /**\n   * Returns {@link FabricImage} instance from an SVG element\n   * @static\n   * @param {HTMLElement} element Element to parse\n   * @param {Object} [options] Options object\n   * @param {AbortSignal} [options.signal] handle aborting, see https://developer.mozilla.org/en-US/docs/Web/API/AbortController/signal\n   * @param {Function} callback Callback to execute when Image object is created\n   */\n  static async fromElement(\n    element: HTMLElement,\n    options: Abortable = {},\n    cssRules?: CSSRules\n  ) {\n    const parsedAttributes = parseAttributes(\n      element,\n      this.ATTRIBUTE_NAMES,\n      cssRules\n    );\n    return this.fromURL(\n      parsedAttributes['xlink:href'],\n      options,\n      parsedAttributes\n    ).catch((err) => {\n      log('log', 'Unable to parse Image', err);\n      return null;\n    });\n  }\n}\n\nclassRegistry.setClass(FabricImage);\nclassRegistry.setSVGClass(FabricImage);\n","import { svgNS } from './constants';\nimport {\n  parsePreserveAspectRatioAttribute,\n  parseUnit,\n} from '../util/misc/svgParsing';\nimport { svgViewBoxElementsRegEx, reViewBoxAttrValue } from './constants';\nimport { NONE } from '../constants';\n\nexport type ParsedViewboxTransform = Partial<{\n  width: number;\n  height: number;\n  minX: number;\n  minY: number;\n  viewBoxWidth: number;\n  viewBoxHeight: number;\n}>;\n\n/**\n * Add a <g> element that envelop all child elements and makes the viewbox transformMatrix descend on all elements\n */\nexport function applyViewboxTransform(\n  element: Element\n): ParsedViewboxTransform {\n  if (!svgViewBoxElementsRegEx.test(element.nodeName)) {\n    return {};\n  }\n  const viewBoxAttr: string | null = element.getAttribute('viewBox');\n  let scaleX = 1;\n  let scaleY = 1;\n  let minX = 0;\n  let minY = 0;\n  let matrix;\n  let el;\n  const widthAttr = element.getAttribute('width');\n  const heightAttr = element.getAttribute('height');\n  const x = element.getAttribute('x') || 0;\n  const y = element.getAttribute('y') || 0;\n  const goodViewbox = viewBoxAttr && reViewBoxAttrValue.test(viewBoxAttr);\n  const missingViewBox = !goodViewbox;\n  const missingDimAttr =\n    !widthAttr || !heightAttr || widthAttr === '100%' || heightAttr === '100%';\n\n  let translateMatrix = '';\n  let widthDiff = 0;\n  let heightDiff = 0;\n\n  if (missingViewBox) {\n    if (\n      (x || y) &&\n      element.parentNode &&\n      element.parentNode.nodeName !== '#document'\n    ) {\n      translateMatrix =\n        ' translate(' + parseUnit(x || '0') + ' ' + parseUnit(y || '0') + ') ';\n      matrix = (element.getAttribute('transform') || '') + translateMatrix;\n      element.setAttribute('transform', matrix);\n      element.removeAttribute('x');\n      element.removeAttribute('y');\n    }\n  }\n\n  if (missingViewBox && missingDimAttr) {\n    return {\n      width: 0,\n      height: 0,\n    };\n  }\n\n  const parsedDim: ParsedViewboxTransform = {\n    width: 0,\n    height: 0,\n  };\n\n  if (missingViewBox) {\n    parsedDim.width = parseUnit(widthAttr!);\n    parsedDim.height = parseUnit(heightAttr!);\n    // set a transform for elements that have x y and are inner(only) SVGs\n    return parsedDim;\n  }\n\n  const pasedViewBox = viewBoxAttr.match(reViewBoxAttrValue)!;\n  minX = -parseFloat(pasedViewBox[1]);\n  minY = -parseFloat(pasedViewBox[2]);\n  const viewBoxWidth = parseFloat(pasedViewBox[3]);\n  const viewBoxHeight = parseFloat(pasedViewBox[4]);\n  parsedDim.minX = minX;\n  parsedDim.minY = minY;\n  parsedDim.viewBoxWidth = viewBoxWidth;\n  parsedDim.viewBoxHeight = viewBoxHeight;\n  if (!missingDimAttr) {\n    parsedDim.width = parseUnit(widthAttr);\n    parsedDim.height = parseUnit(heightAttr);\n    scaleX = parsedDim.width / viewBoxWidth;\n    scaleY = parsedDim.height / viewBoxHeight;\n  } else {\n    parsedDim.width = viewBoxWidth;\n    parsedDim.height = viewBoxHeight;\n  }\n\n  // default is to preserve aspect ratio\n  const preserveAspectRatio = parsePreserveAspectRatioAttribute(\n    element.getAttribute('preserveAspectRatio') || ''\n  );\n  if (preserveAspectRatio.alignX !== NONE) {\n    //translate all container for the effect of Mid, Min, Max\n    if (preserveAspectRatio.meetOrSlice === 'meet') {\n      scaleY = scaleX = scaleX > scaleY ? scaleY : scaleX;\n      // calculate additional translation to move the viewbox\n    }\n    if (preserveAspectRatio.meetOrSlice === 'slice') {\n      scaleY = scaleX = scaleX > scaleY ? scaleX : scaleY;\n      // calculate additional translation to move the viewbox\n    }\n    widthDiff = parsedDim.width - viewBoxWidth * scaleX;\n    heightDiff = parsedDim.height - viewBoxHeight * scaleX;\n    if (preserveAspectRatio.alignX === 'Mid') {\n      widthDiff /= 2;\n    }\n    if (preserveAspectRatio.alignY === 'Mid') {\n      heightDiff /= 2;\n    }\n    if (preserveAspectRatio.alignX === 'Min') {\n      widthDiff = 0;\n    }\n    if (preserveAspectRatio.alignY === 'Min') {\n      heightDiff = 0;\n    }\n  }\n\n  if (\n    scaleX === 1 &&\n    scaleY === 1 &&\n    minX === 0 &&\n    minY === 0 &&\n    x === 0 &&\n    y === 0\n  ) {\n    return parsedDim;\n  }\n  if ((x || y) && element.parentNode!.nodeName !== '#document') {\n    translateMatrix =\n      ' translate(' + parseUnit(x || '0') + ' ' + parseUnit(y || '0') + ') ';\n  }\n\n  matrix =\n    translateMatrix +\n    ' matrix(' +\n    scaleX +\n    ' 0' +\n    ' 0 ' +\n    scaleY +\n    ' ' +\n    (minX * scaleX + widthDiff) +\n    ' ' +\n    (minY * scaleY + heightDiff) +\n    ') ';\n  // seems unused.\n  // parsedDim.viewboxTransform = parseTransformAttribute(matrix);\n  if (element.nodeName === 'svg') {\n    el = element.ownerDocument.createElementNS(svgNS, 'g');\n    // element.firstChild != null\n    while (element.firstChild) {\n      el.appendChild(element.firstChild);\n    }\n    element.appendChild(el);\n  } else {\n    el = element;\n    el.removeAttribute('x');\n    el.removeAttribute('y');\n    matrix = el.getAttribute('transform') + matrix;\n  }\n  el.setAttribute('transform', matrix);\n  return parsedDim;\n}\n","export const getTagName = (node: Element) => node.tagName.replace('svg:', '');\n","import { svgInvalidAncestors } from './constants';\nimport { getSvgRegex } from './getSvgRegex';\nimport { getTagName } from './getTagName';\n\nconst svgInvalidAncestorsRegEx = getSvgRegex(svgInvalidAncestors);\n\nexport function hasInvalidAncestor(element: Element) {\n  let _element: Element | null = element;\n  while (_element && (_element = _element.parentElement)) {\n    if (\n      _element &&\n      _element.nodeName &&\n      svgInvalidAncestorsRegEx.test(getTagName(_element)) &&\n      !_element.getAttribute('instantiated_by_use')\n    ) {\n      return true;\n    }\n  }\n  return false;\n}\n","export function getMultipleNodes(\n  doc: Document,\n  nodeNames: string[]\n): Element[] {\n  let nodeName,\n    nodeArray: Element[] = [],\n    nodeList,\n    i,\n    len;\n  for (i = 0, len = nodeNames.length; i < len; i++) {\n    nodeName = nodeNames[i];\n    nodeList = doc.getElementsByTagNameNS(\n      'http://www.w3.org/2000/svg',\n      nodeName\n    );\n    nodeArray = nodeArray.concat(Array.from(nodeList));\n  }\n  return nodeArray;\n}\n","const gradientsAttrs = [\n  'gradientTransform',\n  'x1',\n  'x2',\n  'y1',\n  'y2',\n  'gradientUnits',\n  'cx',\n  'cy',\n  'r',\n  'fx',\n  'fy',\n];\nconst xlinkAttr = 'xlink:href';\n\nexport function recursivelyParseGradientsXlink(\n  doc: Document,\n  gradient: Element\n) {\n  const xLink = gradient.getAttribute(xlinkAttr)?.slice(1) || '',\n    referencedGradient = doc.getElementById(xLink);\n  if (referencedGradient && referencedGradient.getAttribute(xlinkAttr)) {\n    recursivelyParseGradientsXlink(doc, referencedGradient as Element);\n  }\n  if (referencedGradient) {\n    gradientsAttrs.forEach((attr) => {\n      const value = referencedGradient.getAttribute(attr);\n      if (!gradient.hasAttribute(attr) && value) {\n        gradient.setAttribute(attr, value);\n      }\n    });\n    if (!gradient.children.length) {\n      const referenceClone = referencedGradient.cloneNode(true);\n      while (referenceClone.firstChild) {\n        gradient.appendChild(referenceClone.firstChild);\n      }\n    }\n  }\n  gradient.removeAttribute(xlinkAttr);\n}\n","import { getMultipleNodes } from './getMultipleNodes';\nimport { recursivelyParseGradientsXlink } from './recursivelyParseGradientsXlink';\n\nconst tagArray = [\n  'linearGradient',\n  'radialGradient',\n  'svg:linearGradient',\n  'svg:radialGradient',\n];\n\n/**\n * Parses an SVG document, returning all of the gradient declarations found in it\n * @param {SVGDocument} doc SVG document to parse\n * @return {Object} Gradient definitions; key corresponds to element id, value -- to gradient definition element\n */\nexport function getGradientDefs(\n  doc: Document\n): Record<string, SVGGradientElement> {\n  const elList = getMultipleNodes(doc, tagArray);\n  const gradientDefs: Record<string, SVGGradientElement> = {};\n  let j = elList.length;\n  while (j--) {\n    const el = elList[j];\n    if (el.getAttribute('xlink:href')) {\n      recursivelyParseGradientsXlink(doc, el);\n    }\n    const id = el.getAttribute('id');\n    if (id) {\n      gradientDefs[id] = el as SVGGradientElement;\n    }\n  }\n  return gradientDefs;\n}\n","import type { CSSRules } from './typedefs';\n\n/**\n * Returns CSS rules for a given SVG document\n * @param {HTMLElement} doc SVG document to parse\n * @return {Object} CSS rules of this document\n */\nexport function getCSSRules(doc: Document) {\n  const styles = doc.getElementsByTagName('style');\n  let i;\n  let len;\n  const allRules: CSSRules = {};\n\n  // very crude parsing of style contents\n  for (i = 0, len = styles.length; i < len; i++) {\n    const styleContents = (styles[i].textContent || '').replace(\n      // remove comments\n      /\\/\\*[\\s\\S]*?\\*\\//g,\n      ''\n    );\n\n    if (styleContents.trim() === '') {\n      continue;\n    }\n    // recovers all the rule in this form `body { style code... }`\n    // rules = styleContents.match(/[^{]*\\{[\\s\\S]*?\\}/g);\n    styleContents\n      .split('}')\n      // remove empty rules and remove everything if we didn't split in at least 2 pieces\n      .filter((rule, index, array) => array.length > 1 && rule.trim())\n      // at this point we have hopefully an array of rules `body { style code... `\n      .forEach((rule) => {\n        // if there is more than one opening bracket and the rule starts with '@', it is likely\n        // a nested at-rule like @media, @supports, @scope, etc. Ignore these as the code below\n        // can not handle it.\n        if (\n          (rule.match(/{/g) || []).length > 1 &&\n          rule.trim().startsWith('@')\n        ) {\n          return;\n        }\n\n        const match = rule.split('{'),\n          ruleObj: Record<string, string> = {},\n          declaration = match[1].trim(),\n          propertyValuePairs = declaration.split(';').filter(function (pair) {\n            return pair.trim();\n          });\n\n        for (i = 0, len = propertyValuePairs.length; i < len; i++) {\n          const pair = propertyValuePairs[i].split(':'),\n            property = pair[0].trim(),\n            value = pair[1].trim();\n          ruleObj[property] = value;\n        }\n        rule = match[0].trim();\n        rule.split(',').forEach((_rule) => {\n          _rule = _rule.replace(/^svg/i, '').trim();\n          if (_rule === '') {\n            return;\n          }\n          allRules[_rule] = {\n            ...(allRules[_rule] || {}),\n            ...ruleObj,\n          };\n        });\n      });\n  }\n  return allRules;\n}\n","import { Gradient } from '../gradient/Gradient';\nimport { Group } from '../shapes/Group';\nimport { FabricImage } from '../shapes/Image';\nimport { classRegistry } from '../ClassRegistry';\nimport {\n  invertTransform,\n  multiplyTransformMatrices,\n  qrDecompose,\n} from '../util/misc/matrix';\nimport { removeTransformMatrixForSvgParsing } from '../util/transform_matrix_removal';\nimport type { FabricObject } from '../shapes/Object/FabricObject';\nimport { Point } from '../Point';\nimport { CENTER } from '../constants';\nimport { getGradientDefs } from './getGradientDefs';\nimport { getCSSRules } from './getCSSRules';\nimport type { LoadImageOptions } from '../util';\nimport type { CSSRules, TSvgReviverCallback } from './typedefs';\nimport type { ParsedViewboxTransform } from './applyViewboxTransform';\nimport type { SVGOptions } from '../gradient';\nimport { getTagName } from './getTagName';\nimport { parseTransformAttribute } from './parseTransformAttribute';\n\nconst findTag = (el: Element) =>\n  classRegistry.getSVGClass(getTagName(el).toLowerCase());\n\ntype StorageType = {\n  fill: SVGGradientElement;\n  stroke: SVGGradientElement;\n  clipPath: Element[];\n};\n\ntype NotParsedFabricObject = FabricObject & {\n  fill: string;\n  stroke: string;\n  clipPath?: string;\n  clipRule?: CanvasFillRule;\n};\n\nexport class ElementsParser {\n  declare elements: Element[];\n  declare options: LoadImageOptions & ParsedViewboxTransform;\n  declare reviver: TSvgReviverCallback | undefined;\n  declare regexUrl: RegExp;\n  declare doc: Document;\n  declare clipPaths: Record<string, Element[]>;\n  declare gradientDefs: Record<string, SVGGradientElement>;\n  declare cssRules: CSSRules;\n\n  constructor(\n    elements: Element[],\n    options: LoadImageOptions & ParsedViewboxTransform,\n    reviver: TSvgReviverCallback | undefined,\n    doc: Document,\n    clipPaths: Record<string, Element[]>\n  ) {\n    this.elements = elements;\n    this.options = options;\n    this.reviver = reviver;\n    this.regexUrl = /^url\\(['\"]?#([^'\"]+)['\"]?\\)/g;\n    this.doc = doc;\n    this.clipPaths = clipPaths;\n    this.gradientDefs = getGradientDefs(doc);\n    this.cssRules = getCSSRules(doc);\n  }\n\n  parse(): Promise<Array<FabricObject | null>> {\n    return Promise.all(\n      this.elements.map((element) => this.createObject(element))\n    );\n  }\n\n  async createObject(el: Element): Promise<FabricObject | null> {\n    const klass = findTag(el);\n    if (klass) {\n      const obj: NotParsedFabricObject = await klass.fromElement(\n        el,\n        this.options,\n        this.cssRules\n      );\n      this.resolveGradient(obj, el, 'fill');\n      this.resolveGradient(obj, el, 'stroke');\n      if (obj instanceof FabricImage && obj._originalElement) {\n        removeTransformMatrixForSvgParsing(\n          obj,\n          obj.parsePreserveAspectRatioAttribute()\n        );\n      } else {\n        removeTransformMatrixForSvgParsing(obj);\n      }\n      await this.resolveClipPath(obj, el);\n      this.reviver && this.reviver(el, obj);\n      return obj;\n    }\n    return null;\n  }\n\n  extractPropertyDefinition(\n    obj: NotParsedFabricObject,\n    property: 'fill' | 'stroke' | 'clipPath',\n    storage: Record<string, StorageType[typeof property]>\n  ): StorageType[typeof property] | undefined {\n    const value = obj[property]!,\n      regex = this.regexUrl;\n    if (!regex.test(value)) {\n      return undefined;\n    }\n    // verify: can we remove the 'g' flag? and remove lastIndex changes?\n    regex.lastIndex = 0;\n    // we passed the regex test, so we know is not null;\n    const id = regex.exec(value)![1];\n    regex.lastIndex = 0;\n    // @todo fix this\n    return storage[id];\n  }\n\n  resolveGradient(\n    obj: NotParsedFabricObject,\n    el: Element,\n    property: 'fill' | 'stroke'\n  ) {\n    const gradientDef = this.extractPropertyDefinition(\n      obj,\n      property,\n      this.gradientDefs\n    ) as SVGGradientElement;\n    if (gradientDef) {\n      const opacityAttr = el.getAttribute(property + '-opacity');\n      const gradient = Gradient.fromElement(gradientDef, obj, {\n        ...this.options,\n        opacity: opacityAttr,\n      } as SVGOptions);\n      obj.set(property, gradient);\n    }\n  }\n\n  // TODO: resolveClipPath could be run once per clippath with minor work per object.\n  // is a refactor that i m not sure is worth on this code\n  async resolveClipPath(obj: NotParsedFabricObject, usingElement: Element) {\n    const clipPathElements = this.extractPropertyDefinition(\n      obj,\n      'clipPath',\n      this.clipPaths\n    ) as Element[];\n    if (clipPathElements) {\n      const objTransformInv = invertTransform(obj.calcTransformMatrix());\n      const clipPathTag = clipPathElements[0].parentElement!;\n      let clipPathOwner = usingElement;\n      while (\n        clipPathOwner.parentElement &&\n        clipPathOwner.getAttribute('clip-path') !== obj.clipPath\n      ) {\n        clipPathOwner = clipPathOwner.parentElement;\n      }\n      // move the clipPath tag as sibling to the real element that is using it\n      clipPathOwner.parentElement!.appendChild(clipPathTag!);\n\n      // this multiplication order could be opposite.\n      // but i don't have an svg to test it\n      // at the first SVG that has a transform on both places and is misplaced\n      // try to invert this multiplication order\n      const finalTransform = parseTransformAttribute(\n        `${clipPathOwner.getAttribute('transform') || ''} ${\n          clipPathTag.getAttribute('originalTransform') || ''\n        }`\n      );\n\n      clipPathTag.setAttribute(\n        'transform',\n        `matrix(${finalTransform.join(',')})`\n      );\n\n      const container = await Promise.all(\n        clipPathElements.map((clipPathElement) => {\n          return findTag(clipPathElement)\n            .fromElement(clipPathElement, this.options, this.cssRules)\n            .then((enlivedClippath: NotParsedFabricObject) => {\n              removeTransformMatrixForSvgParsing(enlivedClippath);\n              enlivedClippath.fillRule = enlivedClippath.clipRule!;\n              delete enlivedClippath.clipRule;\n              return enlivedClippath;\n            });\n        })\n      );\n      const clipPath =\n        container.length === 1 ? container[0] : new Group(container);\n      const gTransform = multiplyTransformMatrices(\n        objTransformInv,\n        clipPath.calcTransformMatrix()\n      );\n      if (clipPath.clipPath) {\n        await this.resolveClipPath(clipPath, clipPathOwner);\n      }\n      const { scaleX, scaleY, angle, skewX, translateX, translateY } =\n        qrDecompose(gTransform);\n      clipPath.set({\n        flipX: false,\n        flipY: false,\n      });\n      clipPath.set({\n        scaleX,\n        scaleY,\n        angle,\n        skewX,\n        skewY: 0,\n      });\n      clipPath.setPositionByOrigin(\n        new Point(translateX, translateY),\n        CENTER,\n        CENTER\n      );\n      obj.clipPath = clipPath;\n    } else {\n      // if clip-path does not resolve to any element, delete the property.\n      delete obj.clipPath;\n      return;\n    }\n  }\n}\n","import { applyViewboxTransform } from './applyViewboxTransform';\nimport { svgValidTagNamesRegEx } from './constants';\nimport { hasInvalidAncestor } from './hasInvalidAncestor';\nimport { parseUseDirectives } from './parseUseDirectives';\nimport type { SVGParsingOutput, TSvgReviverCallback } from './typedefs';\nimport type { LoadImageOptions } from '../util/misc/objectEnlive';\nimport { ElementsParser } from './elements_parser';\nimport { log, SignalAbortedError } from '../util/internals/console';\nimport { getTagName } from './getTagName';\n\nconst isValidSvgTag = (el: Element) =>\n  svgValidTagNamesRegEx.test(getTagName(el));\n\nexport const createEmptyResponse = (): SVGParsingOutput => ({\n  objects: [],\n  elements: [],\n  options: {},\n  allElements: [],\n});\n\n/**\n * Parses an SVG document, converts it to an array of corresponding fabric.* instances and passes them to a callback\n * @static\n * @function\n * @memberOf fabric\n * @param {HTMLElement} doc SVG document to parse\n * @param {TSvgParsedCallback} callback Invoked when the parsing is done, with null if parsing wasn't possible with the list of svg nodes.\n * @param {TSvgReviverCallback} [reviver] Extra callback for further parsing of SVG elements, called after each fabric object has been created.\n * Takes as input the original svg element and the generated `FabricObject` as arguments. Used to inspect extra properties not parsed by fabric,\n * or extra custom manipulation\n * @param {Object} [options] Object containing options for parsing\n * @param {String} [options.crossOrigin] crossOrigin setting to use for external resources\n * @param {AbortSignal} [options.signal] handle aborting, see https://developer.mozilla.org/en-US/docs/Web/API/AbortController/signal\n * @return {SVGParsingOutput}\n * {@link SVGParsingOutput} also receives `allElements` array as the last argument. This is the full list of svg nodes available in the document.\n * You may want to use it if you are trying to regroup the objects as they were originally grouped in the SVG. ( This was the reason why it was added )\n */\nexport async function parseSVGDocument(\n  doc: Document,\n  reviver?: TSvgReviverCallback,\n  { crossOrigin, signal }: LoadImageOptions = {}\n): Promise<SVGParsingOutput> {\n  if (signal && signal.aborted) {\n    log('log', new SignalAbortedError('parseSVGDocument'));\n    // this is an unhappy path, we dont care about speed\n    return createEmptyResponse();\n  }\n  const documentElement = doc.documentElement;\n  parseUseDirectives(doc);\n\n  const descendants = Array.from(documentElement.getElementsByTagName('*')),\n    options = {\n      ...applyViewboxTransform(documentElement),\n      crossOrigin,\n      signal,\n    };\n\n  const elements = descendants.filter((el) => {\n    applyViewboxTransform(el);\n    return isValidSvgTag(el) && !hasInvalidAncestor(el); // http://www.w3.org/TR/SVG/struct.html#DefsElement\n  });\n  if (!elements || (elements && !elements.length)) {\n    return {\n      ...createEmptyResponse(),\n      options,\n      allElements: descendants,\n    };\n  }\n  const localClipPaths: Record<string, Element[]> = {};\n  descendants\n    .filter((el) => getTagName(el) === 'clipPath')\n    .forEach((el) => {\n      el.setAttribute('originalTransform', el.getAttribute('transform') || '');\n      const id = el.getAttribute('id')!;\n      localClipPaths[id] = Array.from(el.getElementsByTagName('*')).filter(\n        (el) => isValidSvgTag(el)\n      );\n    });\n\n  // Precedence of rules:   style > class > attribute\n  const elementParser = new ElementsParser(\n    elements,\n    options,\n    reviver,\n    doc,\n    localClipPaths\n  );\n\n  const instances = await elementParser.parse();\n\n  return {\n    objects: instances,\n    elements,\n    options,\n    allElements: descendants,\n  };\n}\n","import { svgNS } from './constants';\nimport { getMultipleNodes } from './getMultipleNodes';\nimport { applyViewboxTransform } from './applyViewboxTransform';\n\nexport function parseUseDirectives(doc: Document) {\n  const nodelist = getMultipleNodes(doc, ['use', 'svg:use']);\n  let i = 0;\n  while (nodelist.length && i < nodelist.length) {\n    const el = nodelist[i],\n      xlinkAttribute = el.getAttribute('xlink:href') || el.getAttribute('href');\n\n    if (xlinkAttribute === null) {\n      return;\n    }\n\n    const xlink = xlinkAttribute.slice(1);\n    const x = el.getAttribute('x') || 0;\n    const y = el.getAttribute('y') || 0;\n    const el2Orig = doc.getElementById(xlink);\n    if (el2Orig === null) {\n      // if we can't find the target of the xlink, consider this use tag bad, similar to no xlink\n      return;\n    }\n    let el2 = el2Orig.cloneNode(true) as Element;\n    let currentTrans =\n      (el2.getAttribute('transform') || '') +\n      ' translate(' +\n      x +\n      ', ' +\n      y +\n      ')';\n    const oldLength = nodelist.length;\n    const namespace = svgNS;\n\n    applyViewboxTransform(el2);\n    if (/^svg$/i.test(el2.nodeName)) {\n      const el3 = el2.ownerDocument.createElementNS(namespace, 'g');\n      for (\n        let j = 0, attrs = el2.attributes, len = attrs.length;\n        j < len;\n        j++\n      ) {\n        const attr: Attr | null = attrs.item(j);\n        attr && el3.setAttributeNS(namespace, attr.nodeName, attr.nodeValue!);\n      }\n      // el2.firstChild != null\n      while (el2.firstChild) {\n        el3.appendChild(el2.firstChild);\n      }\n      el2 = el3;\n    }\n\n    for (let j = 0, attrs = el.attributes, len = attrs.length; j < len; j++) {\n      const attr = attrs.item(j);\n      if (!attr) {\n        continue;\n      }\n      const { nodeName, nodeValue } = attr;\n      if (\n        nodeName === 'x' ||\n        nodeName === 'y' ||\n        nodeName === 'xlink:href' ||\n        nodeName === 'href'\n      ) {\n        continue;\n      }\n\n      if (nodeName === 'transform') {\n        currentTrans = nodeValue + ' ' + currentTrans;\n      } else {\n        el2.setAttribute(nodeName, nodeValue!);\n      }\n    }\n\n    el2.setAttribute('transform', currentTrans);\n    el2.setAttribute('instantiated_by_use', '1');\n    el2.removeAttribute('id');\n    const parentNode = el.parentNode;\n    parentNode!.replaceChild(el2, el);\n    // some browsers do not shorten nodelist after replaceChild (IE8)\n    if (nodelist.length === oldLength) {\n      i++;\n    }\n  }\n}\n","import { getFabricWindow } from '../env';\nimport type { LoadImageOptions } from '../util/misc/objectEnlive';\nimport { parseSVGDocument } from './parseSVGDocument';\nimport type { SVGParsingOutput, TSvgReviverCallback } from './typedefs';\n\n/**\n * Takes string corresponding to an SVG document, and parses it into a set of fabric objects\n * @memberOf fabric\n * @param {String} string representing the svg\n * @param {TSvgParsedCallback} callback Invoked when the parsing is done, with null if parsing wasn't possible with the list of svg nodes.\n * {@link TSvgParsedCallback} also receives `allElements` array as the last argument. This is the full list of svg nodes available in the document.\n * You may want to use it if you are trying to regroup the objects as they were originally grouped in the SVG. ( This was the reason why it was added )\n * @param {TSvgReviverCallback} [reviver] Extra callback for further parsing of SVG elements, called after each fabric object has been created.\n * Takes as input the original svg element and the generated `FabricObject` as arguments. Used to inspect extra properties not parsed by fabric,\n * or extra custom manipulation\n * @param {Object} [options] Object containing options for parsing\n * @param {String} [options.crossOrigin] crossOrigin setting to use for external resources\n * @param {AbortSignal} [options.signal] handle aborting, see https://developer.mozilla.org/en-US/docs/Web/API/AbortController/signal\n */\nexport function loadSVGFromString(\n  string: string,\n  reviver?: TSvgReviverCallback,\n  options?: LoadImageOptions\n): Promise<SVGParsingOutput> {\n  const parser = new (getFabricWindow().DOMParser)(),\n    // should we use `image/svg+xml` here?\n    doc = parser.parseFromString(string.trim(), 'text/xml');\n  return parseSVGDocument(doc, reviver, options);\n}\n","import { request } from '../util/internals/dom_request';\nimport { parseSVGDocument, createEmptyResponse } from './parseSVGDocument';\nimport type { SVGParsingOutput, TSvgReviverCallback } from './typedefs';\nimport type { LoadImageOptions } from '../util/misc/objectEnlive';\n\n/**\n * Takes url corresponding to an SVG document, and parses it into a set of fabric objects.\n * Note that SVG is fetched via XMLHttpRequest, so it needs to conform to SOP (Same Origin Policy)\n * @memberOf fabric\n * @param {string} url where the SVG is\n * @param {TSvgParsedCallback} callback Invoked when the parsing is done, with null if parsing wasn't possible with the list of svg nodes.\n * {@link TSvgParsedCallback} also receives `allElements` array as the last argument. This is the full list of svg nodes available in the document.\n * You may want to use it if you are trying to regroup the objects as they were originally grouped in the SVG. ( This was the reason why it was added )\n * @param {TSvgReviverCallback} [reviver] Extra callback for further parsing of SVG elements, called after each fabric object has been created.\n * Takes as input the original svg element and the generated `FabricObject` as arguments. Used to inspect extra properties not parsed by fabric,\n * or extra custom manipulation\n * @param {Object} [options] Object containing options for parsing\n * @param {String} [options.crossOrigin] crossOrigin setting to use for external resources\n * @param {AbortSignal} [options.signal] handle aborting, see https://developer.mozilla.org/en-US/docs/Web/API/AbortController/signal\n */\nexport function loadSVGFromURL(\n  url: string,\n  reviver?: TSvgReviverCallback,\n  options: LoadImageOptions = {}\n): Promise<SVGParsingOutput> {\n  // need to handle error properly\n  return new Promise<Document>((resolve, reject) => {\n    const onComplete = (r: XMLHttpRequest) => {\n      const xml = r.responseXML;\n      if (xml) {\n        resolve(xml);\n      }\n      reject();\n    };\n\n    request(url.replace(/^\\n\\s*/, '').trim(), {\n      onComplete,\n      signal: options.signal,\n    });\n  })\n    .then((parsedDoc) => parseSVGDocument(parsedDoc, reviver, options))\n    .catch(() => {\n      // this is an unhappy path, we dont care about speed\n      return createEmptyResponse();\n    });\n}\n","import { Point } from '../Point';\nimport { Control } from './Control';\nimport type { TMat2D } from '../typedefs';\nimport type { Polyline } from '../shapes/Polyline';\nimport { multiplyTransformMatrices } from '../util/misc/matrix';\nimport type {\n  TModificationEvents,\n  TPointerEvent,\n  Transform,\n  TransformActionHandler,\n} from '../EventTypeDefs';\nimport { wrapWithFireEvent } from './wrapWithFireEvent';\nimport { sendPointToPlane } from '../util';\n\nconst ACTION_NAME: TModificationEvents = 'modifyPoly';\n\ntype TTransformAnchor = Transform & { pointIndex: number };\n\n/**\n * This function locates the controls.\n * It'll be used both for drawing and for interaction.\n */\nexport const createPolyPositionHandler = (pointIndex: number) => {\n  return function (dim: Point, finalMatrix: TMat2D, polyObject: Polyline) {\n    const { points, pathOffset } = polyObject;\n    return new Point(points[pointIndex])\n      .subtract(pathOffset)\n      .transform(\n        multiplyTransformMatrices(\n          polyObject.getViewportTransform(),\n          polyObject.calcTransformMatrix()\n        )\n      );\n  };\n};\n\n/**\n * This function defines what the control does.\n * It'll be called on every mouse move after a control has been clicked and is being dragged.\n * The function receives as argument the mouse event, the current transform object\n * and the current position in canvas coordinate `transform.target` is a reference to the\n * current object being transformed.\n */\nexport const polyActionHandler = (\n  eventData: TPointerEvent,\n  transform: TTransformAnchor,\n  x: number,\n  y: number\n) => {\n  const { target, pointIndex } = transform;\n  const poly = target as Polyline;\n  const mouseLocalPosition = sendPointToPlane(\n    new Point(x, y),\n    undefined,\n    poly.calcOwnMatrix()\n  );\n\n  poly.points[pointIndex] = mouseLocalPosition.add(poly.pathOffset);\n  poly.setDimensions();\n\n  return true;\n};\n\n/**\n * Keep the polygon in the same position when we change its `width`/`height`/`top`/`left`.\n */\nexport const factoryPolyActionHandler = (\n  pointIndex: number,\n  fn: TransformActionHandler<TTransformAnchor>\n) => {\n  return function (\n    eventData: TPointerEvent,\n    transform: Transform,\n    x: number,\n    y: number\n  ) {\n    const poly = transform.target as Polyline,\n      anchorPoint = new Point(\n        poly.points[(pointIndex > 0 ? pointIndex : poly.points.length) - 1]\n      ),\n      anchorPointInParentPlane = anchorPoint\n        .subtract(poly.pathOffset)\n        .transform(poly.calcOwnMatrix()),\n      actionPerformed = fn(eventData, { ...transform, pointIndex }, x, y);\n\n    const newAnchorPointInParentPlane = anchorPoint\n      .subtract(poly.pathOffset)\n      .transform(poly.calcOwnMatrix());\n\n    const diff = newAnchorPointInParentPlane.subtract(anchorPointInParentPlane);\n    poly.left -= diff.x;\n    poly.top -= diff.y;\n\n    return actionPerformed;\n  };\n};\n\nexport const createPolyActionHandler = (pointIndex: number) =>\n  wrapWithFireEvent(\n    ACTION_NAME,\n    factoryPolyActionHandler(pointIndex, polyActionHandler)\n  );\n\nexport function createPolyControls(\n  poly: Polyline,\n  options?: Partial<Control>\n): Record<string, Control>;\nexport function createPolyControls(\n  numOfControls: number,\n  options?: Partial<Control>\n): Record<string, Control>;\nexport function createPolyControls(\n  arg0: number | Polyline,\n  options: Partial<Control> = {}\n) {\n  const controls = {} as Record<string, Control>;\n  for (\n    let idx = 0;\n    idx < (typeof arg0 === 'number' ? arg0 : arg0.points.length);\n    idx++\n  ) {\n    controls[`p${idx}`] = new Control({\n      actionName: ACTION_NAME,\n      positionHandler: createPolyPositionHandler(idx),\n      actionHandler: createPolyActionHandler(idx),\n      ...options,\n    });\n  }\n  return controls;\n}\n","import type {\n  ControlCursorCallback,\n  TransformActionHandler,\n} from '../EventTypeDefs';\nimport { radiansToDegrees } from '../util/misc/radiansDegreesConversion';\nimport { isLocked, NOT_ALLOWED_CURSOR } from './util';\nimport { wrapWithFireEvent } from './wrapWithFireEvent';\nimport { wrapWithFixedAnchor } from './wrapWithFixedAnchor';\n\n/**\n * Find the correct style for the control that is used for rotation.\n * this function is very simple and it just take care of not-allowed or standard cursor\n * @param {Event} eventData the javascript event that is causing the scale\n * @param {Control} control the control that is interested in the action\n * @param {FabricObject} fabricObject the fabric object that is interested in the action\n * @return {String} a valid css string for the cursor\n */\nexport const rotationStyleHandler: ControlCursorCallback = (\n  eventData,\n  control,\n  fabricObject\n) => {\n  if (fabricObject.lockRotation) {\n    return NOT_ALLOWED_CURSOR;\n  }\n  return control.cursorStyle;\n};\n\n/**\n * Action handler for rotation and snapping, without anchor point.\n * Needs to be wrapped with `wrapWithFixedAnchor` to be effective\n * @param {Event} eventData javascript event that is doing the transform\n * @param {Object} transform javascript object containing a series of information around the current transform\n * @param {number} x current mouse x position, canvas normalized\n * @param {number} y current mouse y position, canvas normalized\n * @return {Boolean} true if some change happened\n * @private\n */\nconst rotateObjectWithSnapping: TransformActionHandler = (\n  eventData,\n  { target, ex, ey, theta, originX, originY },\n  x,\n  y\n) => {\n  const pivotPoint = target.translateToOriginPoint(\n    target.getRelativeCenterPoint(),\n    originX,\n    originY\n  );\n\n  if (isLocked(target, 'lockRotation')) {\n    return false;\n  }\n\n  const lastAngle = Math.atan2(ey - pivotPoint.y, ex - pivotPoint.x),\n    curAngle = Math.atan2(y - pivotPoint.y, x - pivotPoint.x);\n  let angle = radiansToDegrees(curAngle - lastAngle + theta);\n\n  if (target.snapAngle && target.snapAngle > 0) {\n    const snapAngle = target.snapAngle,\n      snapThreshold = target.snapThreshold || snapAngle,\n      rightAngleLocked = Math.ceil(angle / snapAngle) * snapAngle,\n      leftAngleLocked = Math.floor(angle / snapAngle) * snapAngle;\n\n    if (Math.abs(angle - leftAngleLocked) < snapThreshold) {\n      angle = leftAngleLocked;\n    } else if (Math.abs(angle - rightAngleLocked) < snapThreshold) {\n      angle = rightAngleLocked;\n    }\n  }\n\n  // normalize angle to positive value\n  if (angle < 0) {\n    angle = 360 + angle;\n  }\n  angle %= 360;\n\n  const hasRotated = target.angle !== angle;\n  // TODO: why aren't we using set?\n  target.angle = angle;\n  return hasRotated;\n};\n\nexport const rotationWithSnapping = wrapWithFireEvent(\n  'rotating',\n  wrapWithFixedAnchor(rotateObjectWithSnapping)\n);\n","import { getFabricWindow } from '../env';\nimport { createCanvasElement } from '../util/misc/dom';\nimport { WebGLFilterBackend } from './WebGLFilterBackend';\nimport type { TWebGLPipelineState, T2DPipelineState } from './typedefs';\n\nexport const isWebGLPipelineState = (\n  options: TWebGLPipelineState | T2DPipelineState\n): options is TWebGLPipelineState => {\n  return (options as TWebGLPipelineState).webgl !== undefined;\n};\n\n/**\n * Pick a method to copy data from GL context to 2d canvas.  In some browsers using\n * drawImage should be faster, but is also bugged for a small combination of old hardware\n * and drivers.\n * putImageData is faster than drawImage for that specific operation.\n */\nexport const isPutImageFaster = (width: number, height: number): boolean => {\n  const targetCanvas = createCanvasElement();\n  const sourceCanvas = createCanvasElement();\n  const gl = sourceCanvas.getContext('webgl')!;\n  // eslint-disable-next-line no-undef\n  const imageBuffer = new ArrayBuffer(width * height * 4);\n\n  const testContext = {\n    imageBuffer: imageBuffer,\n  } as unknown as Required<WebGLFilterBackend>;\n  const testPipelineState = {\n    destinationWidth: width,\n    destinationHeight: height,\n    targetCanvas: targetCanvas,\n  } as unknown as TWebGLPipelineState;\n  let startTime;\n  targetCanvas.width = width;\n  targetCanvas.height = height;\n\n  startTime = getFabricWindow().performance.now();\n  WebGLFilterBackend.prototype.copyGLTo2D.call(\n    testContext,\n    gl,\n    testPipelineState\n  );\n  const drawImageTime = getFabricWindow().performance.now() - startTime;\n\n  startTime = getFabricWindow().performance.now();\n  WebGLFilterBackend.prototype.copyGLTo2DPutImageData.call(\n    testContext,\n    gl,\n    testPipelineState\n  );\n  const putImageDataTime = getFabricWindow().performance.now() - startTime;\n\n  return drawImageTime > putImageDataTime;\n};\n","export const highPsourceCode = `precision highp float`;\n\nexport const identityFragmentShader = `\n    ${highPsourceCode};\n    varying vec2 vTexCoord;\n    uniform sampler2D uTexture;\n    void main() {\n      gl_FragColor = texture2D(uTexture, vTexCoord);\n    }`;\n\nexport const vertexSource = `\n    attribute vec2 aPosition;\n    varying vec2 vTexCoord;\n    void main() {\n      vTexCoord = aPosition;\n      gl_Position = vec4(aPosition * 2.0 - 1.0, 0.0, 1.0);\n    }`;\n","import { getEnv } from '../env';\nimport { createCanvasElement } from '../util/misc/dom';\nimport type {\n  T2DPipelineState,\n  TWebGLAttributeLocationMap,\n  TWebGLPipelineState,\n  TWebGLProgramCacheItem,\n  TWebGLUniformLocationMap,\n} from './typedefs';\nimport { isWebGLPipelineState } from './utils';\nimport {\n  highPsourceCode,\n  identityFragmentShader,\n  vertexSource,\n} from './shaders/baseFilter';\nimport type { Abortable } from '../typedefs';\nimport { FabricError } from '../util/internals/console';\n\nexport class BaseFilter {\n  /**\n   * Filter type\n   * @param {String} type\n   * @default\n   */\n  get type(): string {\n    return (this.constructor as typeof BaseFilter).type;\n  }\n\n  /**\n   * The class type. Used to identify which class this is.\n   * This is used for serialization purposes and internally it can be used\n   * to identify classes. As a developer you could use `instance of Class`\n   * but to avoid importing all the code and blocking tree shaking we try\n   * to avoid doing that.\n   */\n  static type = 'BaseFilter';\n\n  declare static defaults: Record<string, any>;\n\n  /**\n   * Array of attributes to send with buffers. do not modify\n   * @private\n   */\n  vertexSource = vertexSource;\n\n  /**\n   * Name of the parameter that can be changed in the filter.\n   * Some filters have more than one parameter and there is no\n   * mainParameter\n   * @private\n   */\n  declare mainParameter?: keyof this | undefined;\n\n  /**\n   * Constructor\n   * @param {Object} [options] Options object\n   */\n  constructor({ type, ...options }: Record<string, any> = {}) {\n    Object.assign(\n      this,\n      (this.constructor as typeof BaseFilter).defaults,\n      options\n    );\n  }\n\n  protected getFragmentSource(): string {\n    return identityFragmentShader;\n  }\n\n  /**\n   * Compile this filter's shader program.\n   *\n   * @param {WebGLRenderingContext} gl The GL canvas context to use for shader compilation.\n   * @param {String} fragmentSource fragmentShader source for compilation\n   * @param {String} vertexSource vertexShader source for compilation\n   */\n  createProgram(\n    gl: WebGLRenderingContext,\n    fragmentSource: string = this.getFragmentSource(),\n    vertexSource: string = this.vertexSource\n  ) {\n    const {\n      WebGLProbe: { GLPrecision = 'highp' },\n    } = getEnv();\n    if (GLPrecision !== 'highp') {\n      fragmentSource = fragmentSource.replace(\n        new RegExp(highPsourceCode, 'g'),\n        highPsourceCode.replace('highp', GLPrecision)\n      );\n    }\n    const vertexShader = gl.createShader(gl.VERTEX_SHADER);\n    const fragmentShader = gl.createShader(gl.FRAGMENT_SHADER);\n    const program = gl.createProgram();\n\n    if (!vertexShader || !fragmentShader || !program) {\n      throw new FabricError(\n        'Vertex, fragment shader or program creation error'\n      );\n    }\n    gl.shaderSource(vertexShader, vertexSource);\n    gl.compileShader(vertexShader);\n    if (!gl.getShaderParameter(vertexShader, gl.COMPILE_STATUS)) {\n      throw new FabricError(\n        `Vertex shader compile error for ${this.type}: ${gl.getShaderInfoLog(\n          vertexShader\n        )}`\n      );\n    }\n\n    gl.shaderSource(fragmentShader, fragmentSource);\n    gl.compileShader(fragmentShader);\n    if (!gl.getShaderParameter(fragmentShader, gl.COMPILE_STATUS)) {\n      throw new FabricError(\n        `Fragment shader compile error for ${this.type}: ${gl.getShaderInfoLog(\n          fragmentShader\n        )}`\n      );\n    }\n\n    gl.attachShader(program, vertexShader);\n    gl.attachShader(program, fragmentShader);\n    gl.linkProgram(program);\n    if (!gl.getProgramParameter(program, gl.LINK_STATUS)) {\n      throw new FabricError(\n        `Shader link error for \"${this.type}\" ${gl.getProgramInfoLog(program)}`\n      );\n    }\n\n    const uniformLocations = this.getUniformLocations(gl, program) || {};\n    uniformLocations.uStepW = gl.getUniformLocation(program, 'uStepW');\n    uniformLocations.uStepH = gl.getUniformLocation(program, 'uStepH');\n    return {\n      program,\n      attributeLocations: this.getAttributeLocations(gl, program),\n      uniformLocations,\n    };\n  }\n\n  /**\n   * Return a map of attribute names to WebGLAttributeLocation objects.\n   *\n   * @param {WebGLRenderingContext} gl The canvas context used to compile the shader program.\n   * @param {WebGLShaderProgram} program The shader program from which to take attribute locations.\n   * @returns {Object} A map of attribute names to attribute locations.\n   */\n  getAttributeLocations(\n    gl: WebGLRenderingContext,\n    program: WebGLProgram\n  ): TWebGLAttributeLocationMap {\n    return {\n      aPosition: gl.getAttribLocation(program, 'aPosition'),\n    };\n  }\n\n  /**\n   * Return a map of uniform names to WebGLUniformLocation objects.\n   *\n   * Intended to be overridden by subclasses.\n   *\n   * @param {WebGLRenderingContext} gl The canvas context used to compile the shader program.\n   * @param {WebGLShaderProgram} program The shader program from which to take uniform locations.\n   * @returns {Object} A map of uniform names to uniform locations.\n   */\n  getUniformLocations(\n    gl: WebGLRenderingContext,\n    program: WebGLProgram\n  ): TWebGLUniformLocationMap {\n    return {};\n  }\n\n  /**\n   * Send attribute data from this filter to its shader program on the GPU.\n   *\n   * @param {WebGLRenderingContext} gl The canvas context used to compile the shader program.\n   * @param {Object} attributeLocations A map of shader attribute names to their locations.\n   */\n  sendAttributeData(\n    gl: WebGLRenderingContext,\n    attributeLocations: Record<string, number>,\n    aPositionData: Float32Array\n  ) {\n    const attributeLocation = attributeLocations.aPosition;\n    const buffer = gl.createBuffer();\n    gl.bindBuffer(gl.ARRAY_BUFFER, buffer);\n    gl.enableVertexAttribArray(attributeLocation);\n    gl.vertexAttribPointer(attributeLocation, 2, gl.FLOAT, false, 0, 0);\n    gl.bufferData(gl.ARRAY_BUFFER, aPositionData, gl.STATIC_DRAW);\n  }\n\n  _setupFrameBuffer(options: TWebGLPipelineState) {\n    const gl = options.context;\n    if (options.passes > 1) {\n      const width = options.destinationWidth;\n      const height = options.destinationHeight;\n      if (options.sourceWidth !== width || options.sourceHeight !== height) {\n        gl.deleteTexture(options.targetTexture);\n        options.targetTexture = options.filterBackend.createTexture(\n          gl,\n          width,\n          height\n        );\n      }\n      gl.framebufferTexture2D(\n        gl.FRAMEBUFFER,\n        gl.COLOR_ATTACHMENT0,\n        gl.TEXTURE_2D,\n        options.targetTexture,\n        0\n      );\n    } else {\n      // draw last filter on canvas and not to framebuffer.\n      gl.bindFramebuffer(gl.FRAMEBUFFER, null);\n      gl.finish();\n    }\n  }\n\n  _swapTextures(options: TWebGLPipelineState) {\n    options.passes--;\n    options.pass++;\n    const temp = options.targetTexture;\n    options.targetTexture = options.sourceTexture;\n    options.sourceTexture = temp;\n  }\n\n  /**\n   * Generic isNeutral implementation for one parameter based filters.\n   * Used only in image applyFilters to discard filters that will not have an effect\n   * on the image\n   * Other filters may need their own version ( ColorMatrix, HueRotation, gamma, ComposedFilter )\n   * @param {Object} options\n   **/\n  // eslint-disable-next-line @typescript-eslint/no-unused-vars\n  isNeutralState(options?: any): boolean {\n    const main = this.mainParameter,\n      defaultValue = (this.constructor as typeof BaseFilter).defaults[\n        main as string\n      ];\n    if (main) {\n      const thisValue = this[main];\n      if (Array.isArray(defaultValue) && Array.isArray(thisValue)) {\n        return defaultValue.every(\n          (value: any, i: number) => value === thisValue[i]\n        );\n      } else {\n        return defaultValue === thisValue;\n      }\n    } else {\n      return false;\n    }\n  }\n\n  /**\n   * Apply this filter to the input image data provided.\n   *\n   * Determines whether to use WebGL or Canvas2D based on the options.webgl flag.\n   *\n   * @param {Object} options\n   * @param {Number} options.passes The number of filters remaining to be executed\n   * @param {Boolean} options.webgl Whether to use webgl to render the filter.\n   * @param {WebGLTexture} options.sourceTexture The texture setup as the source to be filtered.\n   * @param {WebGLTexture} options.targetTexture The texture where filtered output should be drawn.\n   * @param {WebGLRenderingContext} options.context The GL context used for rendering.\n   * @param {Object} options.programCache A map of compiled shader programs, keyed by filter type.\n   */\n  applyTo(options: TWebGLPipelineState | T2DPipelineState) {\n    if (isWebGLPipelineState(options)) {\n      this._setupFrameBuffer(options);\n      this.applyToWebGL(options);\n      this._swapTextures(options);\n    } else {\n      this.applyTo2d(options);\n    }\n  }\n\n  applyTo2d(options: T2DPipelineState): void {\n    // override by subclass\n  }\n\n  /**\n   * Returns a string that represent the current selected shader code for the filter.\n   * Used to force recompilation when parameters change or to retrieve the shader from cache\n   * @type string\n   **/\n  getCacheKey() {\n    return this.type;\n  }\n\n  /**\n   * Retrieves the cached shader.\n   * @param {Object} options\n   * @param {WebGLRenderingContext} options.context The GL context used for rendering.\n   * @param {Object} options.programCache A map of compiled shader programs, keyed by filter type.\n   * @return {WebGLProgram} the compiled program shader\n   */\n  retrieveShader(options: TWebGLPipelineState): TWebGLProgramCacheItem {\n    const key = this.getCacheKey();\n    if (!options.programCache[key]) {\n      options.programCache[key] = this.createProgram(options.context);\n    }\n    return options.programCache[key];\n  }\n\n  /**\n   * Apply this filter using webgl.\n   *\n   * @param {Object} options\n   * @param {Number} options.passes The number of filters remaining to be executed\n   * @param {Boolean} options.webgl Whether to use webgl to render the filter.\n   * @param {WebGLTexture} options.originalTexture The texture of the original input image.\n   * @param {WebGLTexture} options.sourceTexture The texture setup as the source to be filtered.\n   * @param {WebGLTexture} options.targetTexture The texture where filtered output should be drawn.\n   * @param {WebGLRenderingContext} options.context The GL context used for rendering.\n   * @param {Object} options.programCache A map of compiled shader programs, keyed by filter type.\n   */\n  applyToWebGL(options: TWebGLPipelineState) {\n    const gl = options.context;\n    const shader = this.retrieveShader(options);\n    if (options.pass === 0 && options.originalTexture) {\n      gl.bindTexture(gl.TEXTURE_2D, options.originalTexture);\n    } else {\n      gl.bindTexture(gl.TEXTURE_2D, options.sourceTexture);\n    }\n    gl.useProgram(shader.program);\n    this.sendAttributeData(gl, shader.attributeLocations, options.aPosition);\n\n    gl.uniform1f(shader.uniformLocations.uStepW, 1 / options.sourceWidth);\n    gl.uniform1f(shader.uniformLocations.uStepH, 1 / options.sourceHeight);\n\n    this.sendUniformData(gl, shader.uniformLocations);\n    gl.viewport(0, 0, options.destinationWidth, options.destinationHeight);\n    gl.drawArrays(gl.TRIANGLE_STRIP, 0, 4);\n  }\n\n  bindAdditionalTexture(\n    gl: WebGLRenderingContext,\n    texture: WebGLTexture,\n    textureUnit: number\n  ) {\n    gl.activeTexture(textureUnit);\n    gl.bindTexture(gl.TEXTURE_2D, texture);\n    // reset active texture to 0 as usual\n    gl.activeTexture(gl.TEXTURE0);\n  }\n\n  unbindAdditionalTexture(gl: WebGLRenderingContext, textureUnit: number) {\n    gl.activeTexture(textureUnit);\n    gl.bindTexture(gl.TEXTURE_2D, null);\n    gl.activeTexture(gl.TEXTURE0);\n  }\n\n  getMainParameter() {\n    return this.mainParameter ? this[this.mainParameter] : undefined;\n  }\n\n  setMainParameter(value: any) {\n    if (this.mainParameter) {\n      this[this.mainParameter] = value;\n    }\n  }\n\n  /**\n   * Send uniform data from this filter to its shader program on the GPU.\n   *\n   * Intended to be overridden by subclasses.\n   *\n   * @param {WebGLRenderingContext} gl The canvas context used to compile the shader program.\n   * @param {Object} uniformLocations A map of shader uniform names to their locations.\n   */\n  sendUniformData(\n    gl: WebGLRenderingContext,\n    uniformLocations: TWebGLUniformLocationMap\n  ): void {\n    // override by subclass\n  }\n\n  /**\n   * If needed by a 2d filter, this functions can create an helper canvas to be used\n   * remember that options.targetCanvas is available for use till end of chain.\n   */\n  createHelpLayer(options: T2DPipelineState) {\n    if (!options.helpLayer) {\n      const helpLayer = createCanvasElement();\n      helpLayer.width = options.sourceWidth;\n      helpLayer.height = options.sourceHeight;\n      options.helpLayer = helpLayer;\n    }\n  }\n\n  /**\n   * Returns object representation of an instance\n   * @return {Object} Object representation of an instance\n   */\n  toObject() {\n    const mainP = this.mainParameter;\n    return {\n      type: this.type,\n      ...(mainP ? { [mainP]: this[mainP] } : {}),\n    };\n  }\n\n  /**\n   * Returns a JSON representation of an instance\n   * @return {Object} JSON\n   */\n  toJSON() {\n    // delegate, not alias\n    return this.toObject();\n  }\n\n  static async fromObject(\n    { type, ...filterOptions }: Record<string, any>,\n    options: Abortable\n  ) {\n    return new this(filterOptions);\n  }\n}\n","export const blendColorFragmentSource = {\n  multiply: 'gl_FragColor.rgb *= uColor.rgb;\\n',\n  screen:\n    'gl_FragColor.rgb = 1.0 - (1.0 - gl_FragColor.rgb) * (1.0 - uColor.rgb);\\n',\n  add: 'gl_FragColor.rgb += uColor.rgb;\\n',\n  difference: 'gl_FragColor.rgb = abs(gl_FragColor.rgb - uColor.rgb);\\n',\n  subtract: 'gl_FragColor.rgb -= uColor.rgb;\\n',\n  lighten: 'gl_FragColor.rgb = max(gl_FragColor.rgb, uColor.rgb);\\n',\n  darken: 'gl_FragColor.rgb = min(gl_FragColor.rgb, uColor.rgb);\\n',\n  exclusion:\n    'gl_FragColor.rgb += uColor.rgb - 2.0 * (uColor.rgb * gl_FragColor.rgb);\\n',\n  overlay: `\n    if (uColor.r < 0.5) {\n      gl_FragColor.r *= 2.0 * uColor.r;\n    } else {\n      gl_FragColor.r = 1.0 - 2.0 * (1.0 - gl_FragColor.r) * (1.0 - uColor.r);\n    }\n    if (uColor.g < 0.5) {\n      gl_FragColor.g *= 2.0 * uColor.g;\n    } else {\n      gl_FragColor.g = 1.0 - 2.0 * (1.0 - gl_FragColor.g) * (1.0 - uColor.g);\n    }\n    if (uColor.b < 0.5) {\n      gl_FragColor.b *= 2.0 * uColor.b;\n    } else {\n      gl_FragColor.b = 1.0 - 2.0 * (1.0 - gl_FragColor.b) * (1.0 - uColor.b);\n    }\n    `,\n  tint: `\n    gl_FragColor.rgb *= (1.0 - uColor.a);\n    gl_FragColor.rgb += uColor.rgb;\n    `,\n} as const;\n","import { Color } from '../color/Color';\nimport type { TClassProperties } from '../typedefs';\nimport { BaseFilter } from './BaseFilter';\nimport type { T2DPipelineState, TWebGLUniformLocationMap } from './typedefs';\nimport { classRegistry } from '../ClassRegistry';\nimport { blendColorFragmentSource } from './shaders/blendColor';\n\nexport type TBlendMode =\n  | 'multiply'\n  | 'add'\n  | 'difference'\n  | 'screen'\n  | 'subtract'\n  | 'darken'\n  | 'lighten'\n  | 'overlay'\n  | 'exclusion'\n  | 'tint';\n\nexport const blendColorDefaultValues: Partial<TClassProperties<BlendColor>> = {\n  color: '#F95C63',\n  mode: 'multiply',\n  alpha: 1,\n};\n\n/**\n * Color Blend filter class\n * @example\n * const filter = new BlendColor({\n *  color: '#000',\n *  mode: 'multiply'\n * });\n *\n * const filter = new BlendImage({\n *  image: fabricImageObject,\n *  mode: 'multiply'\n * });\n * object.filters.push(filter);\n * object.applyFilters();\n * canvas.renderAll();\n */\nexport class BlendColor extends BaseFilter {\n  /**\n   * Color to make the blend operation with. default to a reddish color since black or white\n   * gives always strong result.\n   * @type String\n   * @default\n   **/\n  declare color: string;\n\n  /**\n   * Blend mode for the filter: one of multiply, add, difference, screen, subtract,\n   * darken, lighten, overlay, exclusion, tint.\n   * @type String\n   * @default\n   **/\n  declare mode: TBlendMode;\n\n  /**\n   * alpha value. represent the strength of the blend color operation.\n   * @type Number\n   * @default\n   **/\n  declare alpha: number;\n\n  static defaults = blendColorDefaultValues;\n\n  static type = 'BlendColor';\n\n  getCacheKey() {\n    return `${this.type}_${this.mode}`;\n  }\n\n  protected getFragmentSource(): string {\n    return `\n      precision highp float;\n      uniform sampler2D uTexture;\n      uniform vec4 uColor;\n      varying vec2 vTexCoord;\n      void main() {\n        vec4 color = texture2D(uTexture, vTexCoord);\n        gl_FragColor = color;\n        if (color.a > 0.0) {\n          ${blendColorFragmentSource[this.mode]}\n        }\n      }\n      `;\n  }\n\n  /**\n   * Apply the Blend operation to a Uint8ClampedArray representing the pixels of an image.\n   *\n   * @param {Object} options\n   * @param {ImageData} options.imageData The Uint8ClampedArray to be filtered.\n   */\n  applyTo2d({ imageData: { data } }: T2DPipelineState) {\n    const source = new Color(this.color).getSource();\n    const tr = source[0] * this.alpha;\n    const tg = source[1] * this.alpha;\n    const tb = source[2] * this.alpha;\n    const alpha1 = 1 - this.alpha;\n\n    for (let i = 0; i < data.length; i += 4) {\n      const r = data[i];\n      const g = data[i + 1];\n      const b = data[i + 2];\n\n      switch (this.mode) {\n        case 'multiply':\n          data[i] = (r * tr) / 255;\n          data[i + 1] = (g * tg) / 255;\n          data[i + 2] = (b * tb) / 255;\n          break;\n        case 'screen':\n          data[i] = 255 - ((255 - r) * (255 - tr)) / 255;\n          data[i + 1] = 255 - ((255 - g) * (255 - tg)) / 255;\n          data[i + 2] = 255 - ((255 - b) * (255 - tb)) / 255;\n          break;\n        case 'add':\n          data[i] = r + tr;\n          data[i + 1] = g + tg;\n          data[i + 2] = b + tb;\n          break;\n        case 'difference':\n          data[i] = Math.abs(r - tr);\n          data[i + 1] = Math.abs(g - tg);\n          data[i + 2] = Math.abs(b - tb);\n          break;\n        case 'subtract':\n          data[i] = r - tr;\n          data[i + 1] = g - tg;\n          data[i + 2] = b - tb;\n          break;\n        case 'darken':\n          data[i] = Math.min(r, tr);\n          data[i + 1] = Math.min(g, tg);\n          data[i + 2] = Math.min(b, tb);\n          break;\n        case 'lighten':\n          data[i] = Math.max(r, tr);\n          data[i + 1] = Math.max(g, tg);\n          data[i + 2] = Math.max(b, tb);\n          break;\n        case 'overlay':\n          data[i] =\n            tr < 128\n              ? (2 * r * tr) / 255\n              : 255 - (2 * (255 - r) * (255 - tr)) / 255;\n          data[i + 1] =\n            tg < 128\n              ? (2 * g * tg) / 255\n              : 255 - (2 * (255 - g) * (255 - tg)) / 255;\n          data[i + 2] =\n            tb < 128\n              ? (2 * b * tb) / 255\n              : 255 - (2 * (255 - b) * (255 - tb)) / 255;\n          break;\n        case 'exclusion':\n          data[i] = tr + r - (2 * tr * r) / 255;\n          data[i + 1] = tg + g - (2 * tg * g) / 255;\n          data[i + 2] = tb + b - (2 * tb * b) / 255;\n          break;\n        case 'tint':\n          data[i] = tr + r * alpha1;\n          data[i + 1] = tg + g * alpha1;\n          data[i + 2] = tb + b * alpha1;\n      }\n    }\n  }\n\n  /**\n   * Return WebGL uniform locations for this filter's shader.\n   *\n   * @param {WebGLRenderingContext} gl The GL canvas context used to compile this filter's shader.\n   * @param {WebGLShaderProgram} program This filter's compiled shader program.\n   */\n  getUniformLocations(\n    gl: WebGLRenderingContext,\n    program: WebGLProgram\n  ): TWebGLUniformLocationMap {\n    return {\n      uColor: gl.getUniformLocation(program, 'uColor'),\n    };\n  }\n\n  /**\n   * Send data from this filter to its shader program's uniforms.\n   *\n   * @param {WebGLRenderingContext} gl The GL canvas context used to compile this filter's shader.\n   * @param {Object} uniformLocations A map of string uniform names to WebGLUniformLocation objects\n   */\n  sendUniformData(\n    gl: WebGLRenderingContext,\n    uniformLocations: TWebGLUniformLocationMap\n  ) {\n    const source = new Color(this.color).getSource();\n    source[0] = (this.alpha * source[0]) / 255;\n    source[1] = (this.alpha * source[1]) / 255;\n    source[2] = (this.alpha * source[2]) / 255;\n    source[3] = this.alpha;\n    gl.uniform4fv(uniformLocations.uColor, source);\n  }\n\n  /**\n   * Returns object representation of an instance\n   * @return {Object} Object representation of an instance\n   */\n  toObject() {\n    return {\n      type: this.type,\n      color: this.color,\n      mode: this.mode,\n      alpha: this.alpha,\n    };\n  }\n}\n\nclassRegistry.setClass(BlendColor);\n","import type { TBlendImageMode } from '../BlendImage';\n\nexport const fragmentSource: Record<TBlendImageMode, string> = {\n  multiply: `\n    precision highp float;\n    uniform sampler2D uTexture;\n    uniform sampler2D uImage;\n    uniform vec4 uColor;\n    varying vec2 vTexCoord;\n    varying vec2 vTexCoord2;\n    void main() {\n      vec4 color = texture2D(uTexture, vTexCoord);\n      vec4 color2 = texture2D(uImage, vTexCoord2);\n      color.rgba *= color2.rgba;\n      gl_FragColor = color;\n    }\n    `,\n  mask: `\n    precision highp float;\n    uniform sampler2D uTexture;\n    uniform sampler2D uImage;\n    uniform vec4 uColor;\n    varying vec2 vTexCoord;\n    varying vec2 vTexCoord2;\n    void main() {\n      vec4 color = texture2D(uTexture, vTexCoord);\n      vec4 color2 = texture2D(uImage, vTexCoord2);\n      color.a = color2.a;\n      gl_FragColor = color;\n    }\n    `,\n} as const;\n","import { FabricImage } from '../shapes/Image';\nimport type { TClassProperties } from '../typedefs';\nimport { createCanvasElement } from '../util/misc/dom';\nimport { BaseFilter } from './BaseFilter';\nimport type {\n  T2DPipelineState,\n  TWebGLPipelineState,\n  TWebGLUniformLocationMap,\n} from './typedefs';\nimport type { WebGLFilterBackend } from './WebGLFilterBackend';\nimport { classRegistry } from '../ClassRegistry';\nimport { fragmentSource } from './shaders/blendImage';\n\nexport type TBlendImageMode = 'multiply' | 'mask';\n\nexport const blendImageDefaultValues: Partial<TClassProperties<BlendImage>> = {\n  mode: 'multiply',\n  alpha: 1,\n  vertexSource: `\n    attribute vec2 aPosition;\n    varying vec2 vTexCoord;\n    varying vec2 vTexCoord2;\n    uniform mat3 uTransformMatrix;\n    void main() {\n      vTexCoord = aPosition;\n      vTexCoord2 = (uTransformMatrix * vec3(aPosition, 1.0)).xy;\n      gl_Position = vec4(aPosition * 2.0 - 1.0, 0.0, 1.0);\n    }\n    `,\n};\n\n/**\n * Image Blend filter class\n * @example\n * const filter = new filters.BlendColor({\n *  color: '#000',\n *  mode: 'multiply'\n * });\n *\n * const filter = new BlendImage({\n *  image: fabricImageObject,\n *  mode: 'multiply'\n * });\n * object.filters.push(filter);\n * object.applyFilters();\n * canvas.renderAll();\n */\nexport class BlendImage extends BaseFilter {\n  /**\n   * Image to make the blend operation with.\n   **/\n  declare image: FabricImage;\n\n  /**\n   * Blend mode for the filter: either 'multiply' or 'mask'. 'multiply' will\n   * multiply the values of each channel (R, G, B, and A) of the filter image by\n   * their corresponding values in the base image. 'mask' will only look at the\n   * alpha channel of the filter image, and apply those values to the base\n   * image's alpha channel.\n   * @type String\n   * @default\n   **/\n  declare mode: TBlendImageMode;\n\n  /**\n   * alpha value. represent the strength of the blend image operation.\n   * not implemented.\n   **/\n  declare alpha: number;\n\n  static type = 'BlendImage';\n\n  static defaults = blendImageDefaultValues;\n\n  getCacheKey() {\n    return `${this.type}_${this.mode}`;\n  }\n\n  getFragmentSource(): string {\n    return fragmentSource[this.mode];\n  }\n\n  applyToWebGL(options: TWebGLPipelineState) {\n    const gl = options.context,\n      texture = this.createTexture(options.filterBackend, this.image);\n    this.bindAdditionalTexture(gl, texture!, gl.TEXTURE1);\n    super.applyToWebGL(options);\n    this.unbindAdditionalTexture(gl, gl.TEXTURE1);\n  }\n\n  createTexture(backend: WebGLFilterBackend, image: FabricImage) {\n    return backend.getCachedTexture(image.cacheKey, image.getElement());\n  }\n\n  /**\n   * Calculate a transformMatrix to adapt the image to blend over\n   * @param {Object} options\n   * @param {WebGLRenderingContext} options.context The GL context used for rendering.\n   * @param {Object} options.programCache A map of compiled shader programs, keyed by filter type.\n   */\n  calculateMatrix() {\n    const image = this.image,\n      { width, height } = image.getElement();\n    return [\n      1 / image.scaleX,\n      0,\n      0,\n      0,\n      1 / image.scaleY,\n      0,\n      -image.left / width,\n      -image.top / height,\n      1,\n    ];\n  }\n\n  /**\n   * Apply the Blend operation to a Uint8ClampedArray representing the pixels of an image.\n   *\n   * @param {Object} options\n   * @param {ImageData} options.imageData The Uint8ClampedArray to be filtered.\n   */\n  applyTo2d({\n    imageData: { data, width, height },\n    filterBackend: { resources },\n  }: T2DPipelineState) {\n    const image = this.image;\n    if (!resources.blendImage) {\n      resources.blendImage = createCanvasElement();\n    }\n    const canvas1 = resources.blendImage;\n    const context = canvas1.getContext('2d')!;\n    if (canvas1.width !== width || canvas1.height !== height) {\n      canvas1.width = width;\n      canvas1.height = height;\n    } else {\n      context.clearRect(0, 0, width, height);\n    }\n    context.setTransform(\n      image.scaleX,\n      0,\n      0,\n      image.scaleY,\n      image.left,\n      image.top\n    );\n    context.drawImage(image.getElement(), 0, 0, width, height);\n    const blendData = context.getImageData(0, 0, width, height).data;\n    for (let i = 0; i < data.length; i += 4) {\n      const r = data[i];\n      const g = data[i + 1];\n      const b = data[i + 2];\n      const a = data[i + 3];\n\n      const tr = blendData[i];\n      const tg = blendData[i + 1];\n      const tb = blendData[i + 2];\n      const ta = blendData[i + 3];\n\n      switch (this.mode) {\n        case 'multiply':\n          data[i] = (r * tr) / 255;\n          data[i + 1] = (g * tg) / 255;\n          data[i + 2] = (b * tb) / 255;\n          data[i + 3] = (a * ta) / 255;\n          break;\n        case 'mask':\n          data[i + 3] = ta;\n          break;\n      }\n    }\n  }\n\n  /**\n   * Return WebGL uniform locations for this filter's shader.\n   *\n   * @param {WebGLRenderingContext} gl The GL canvas context used to compile this filter's shader.\n   * @param {WebGLShaderProgram} program This filter's compiled shader program.\n   */\n  getUniformLocations(\n    gl: WebGLRenderingContext,\n    program: WebGLProgram\n  ): TWebGLUniformLocationMap {\n    return {\n      uTransformMatrix: gl.getUniformLocation(program, 'uTransformMatrix'),\n      uImage: gl.getUniformLocation(program, 'uImage'),\n    };\n  }\n\n  /**\n   * Send data from this filter to its shader program's uniforms.\n   *\n   * @param {WebGLRenderingContext} gl The GL canvas context used to compile this filter's shader.\n   * @param {Object} uniformLocations A map of string uniform names to WebGLUniformLocation objects\n   */\n  sendUniformData(\n    gl: WebGLRenderingContext,\n    uniformLocations: TWebGLUniformLocationMap\n  ) {\n    const matrix = this.calculateMatrix();\n    gl.uniform1i(uniformLocations.uImage, 1); // texture unit 1.\n    gl.uniformMatrix3fv(uniformLocations.uTransformMatrix, false, matrix);\n  }\n\n  /**\n   * Returns object representation of an instance\n   * @return {Object} Object representation of an instance\n   */\n  toObject() {\n    return {\n      type: this.type,\n      image: this.image && this.image.toObject(),\n      mode: this.mode,\n      alpha: this.alpha,\n    };\n  }\n\n  /**\n   * Create filter instance from an object representation\n   * @static\n   * @param {object} object Object to create an instance from\n   * @param {object} [options]\n   * @param {AbortSignal} [options.signal] handle aborting image loading, see https://developer.mozilla.org/en-US/docs/Web/API/AbortController/signal\n   * @returns {Promise<BlendImage>}\n   */\n  static fromObject(\n    { type, image, ...filterOptions }: Record<string, any>,\n    options: { signal: AbortSignal }\n  ) {\n    return FabricImage.fromObject(image, options).then(\n      (enlivedImage) =>\n        new this({ ...filterOptions, image: enlivedImage }) as BaseFilter\n    );\n  }\n}\n\nclassRegistry.setClass(BlendImage);\n","import type { TClassProperties } from '../typedefs';\nimport { createCanvasElement } from '../util/misc/dom';\nimport { BaseFilter } from './BaseFilter';\nimport type {\n  TWebGLPipelineState,\n  T2DPipelineState,\n  TWebGLUniformLocationMap,\n} from './typedefs';\nimport { isWebGLPipelineState } from './utils';\nimport { classRegistry } from '../ClassRegistry';\nimport { fragmentSource } from './shaders/blur';\n\nexport const blurDefaultValues: Partial<TClassProperties<Blur>> = {\n  blur: 0,\n  mainParameter: 'blur',\n};\n\n/**\n * Blur filter class\n * @example\n * const filter = new Blur({\n *   blur: 0.5\n * });\n * object.filters.push(filter);\n * object.applyFilters();\n * canvas.renderAll();\n */\nexport class Blur extends BaseFilter {\n  /**\n   * blur value, in percentage of image dimensions.\n   * specific to keep the image blur constant at different resolutions\n   * range between 0 and 1.\n   * @type Number\n   * @default\n   */\n  declare blur: number;\n\n  declare horizontal: boolean;\n  declare aspectRatio: number;\n\n  static type = 'Blur';\n\n  static defaults = blurDefaultValues;\n\n  getFragmentSource(): string {\n    return fragmentSource;\n  }\n\n  applyTo(options: TWebGLPipelineState | T2DPipelineState) {\n    if (isWebGLPipelineState(options)) {\n      // this aspectRatio is used to give the same blur to vertical and horizontal\n      this.aspectRatio = options.sourceWidth / options.sourceHeight;\n      options.passes++;\n      this._setupFrameBuffer(options);\n      this.horizontal = true;\n      this.applyToWebGL(options);\n      this._swapTextures(options);\n      this._setupFrameBuffer(options);\n      this.horizontal = false;\n      this.applyToWebGL(options);\n      this._swapTextures(options);\n    } else {\n      this.applyTo2d(options);\n    }\n  }\n\n  applyTo2d(options: T2DPipelineState) {\n    options.imageData = this.simpleBlur(options);\n  }\n\n  simpleBlur({\n    ctx,\n    imageData,\n    filterBackend: { resources },\n  }: T2DPipelineState) {\n    const { width, height } = imageData;\n    if (!resources.blurLayer1) {\n      resources.blurLayer1 = createCanvasElement();\n      resources.blurLayer2 = createCanvasElement();\n    }\n    const canvas1 = resources.blurLayer1!;\n    const canvas2 = resources.blurLayer2!;\n    if (canvas1.width !== width || canvas1.height !== height) {\n      canvas2.width = canvas1.width = width;\n      canvas2.height = canvas1.height = height;\n    }\n    const ctx1 = canvas1.getContext('2d')!,\n      ctx2 = canvas2.getContext('2d')!,\n      nSamples = 15,\n      blur = this.blur * 0.06 * 0.5;\n    let random, percent, j, i;\n\n    // load first canvas\n    ctx1.putImageData(imageData, 0, 0);\n    ctx2.clearRect(0, 0, width, height);\n\n    for (i = -nSamples; i <= nSamples; i++) {\n      random = (Math.random() - 0.5) / 4;\n      percent = i / nSamples;\n      j = blur * percent * width + random;\n      ctx2.globalAlpha = 1 - Math.abs(percent);\n      ctx2.drawImage(canvas1, j, random);\n      ctx1.drawImage(canvas2, 0, 0);\n      ctx2.globalAlpha = 1;\n      ctx2.clearRect(0, 0, canvas2.width, canvas2.height);\n    }\n    for (i = -nSamples; i <= nSamples; i++) {\n      random = (Math.random() - 0.5) / 4;\n      percent = i / nSamples;\n      j = blur * percent * height + random;\n      ctx2.globalAlpha = 1 - Math.abs(percent);\n      ctx2.drawImage(canvas1, random, j);\n      ctx1.drawImage(canvas2, 0, 0);\n      ctx2.globalAlpha = 1;\n      ctx2.clearRect(0, 0, canvas2.width, canvas2.height);\n    }\n    ctx.drawImage(canvas1, 0, 0);\n    const newImageData = ctx.getImageData(0, 0, canvas1.width, canvas1.height);\n    ctx1.globalAlpha = 1;\n    ctx1.clearRect(0, 0, canvas1.width, canvas1.height);\n    return newImageData;\n  }\n\n  /**\n   * Return WebGL uniform locations for this filter's shader.\n   *\n   * @param {WebGLRenderingContext} gl The GL canvas context used to compile this filter's shader.\n   * @param {WebGLShaderProgram} program This filter's compiled shader program.\n   */\n  getUniformLocations(\n    gl: WebGLRenderingContext,\n    program: WebGLProgram\n  ): TWebGLUniformLocationMap {\n    return {\n      delta: gl.getUniformLocation(program, 'uDelta'),\n    };\n  }\n\n  /**\n   * Send data from this filter to its shader program's uniforms.\n   *\n   * @param {WebGLRenderingContext} gl The GL canvas context used to compile this filter's shader.\n   * @param {Object} uniformLocations A map of string uniform names to WebGLUniformLocation objects\n   */\n  sendUniformData(\n    gl: WebGLRenderingContext,\n    uniformLocations: TWebGLUniformLocationMap\n  ) {\n    const delta = this.chooseRightDelta();\n    gl.uniform2fv(uniformLocations.delta, delta);\n  }\n\n  /**\n   * choose right value of image percentage to blur with\n   * @returns {Array} a numeric array with delta values\n   */\n  chooseRightDelta() {\n    let blurScale = 1;\n    const delta = [0, 0];\n    if (this.horizontal) {\n      if (this.aspectRatio > 1) {\n        // image is wide, i want to shrink radius horizontal\n        blurScale = 1 / this.aspectRatio;\n      }\n    } else {\n      if (this.aspectRatio < 1) {\n        // image is tall, i want to shrink radius vertical\n        blurScale = this.aspectRatio;\n      }\n    }\n    const blur = blurScale * this.blur * 0.12;\n    if (this.horizontal) {\n      delta[0] = blur;\n    } else {\n      delta[1] = blur;\n    }\n    return delta;\n  }\n}\n\nclassRegistry.setClass(Blur);\n","export const fragmentSource = `\n    precision highp float;\n    uniform sampler2D uTexture;\n    uniform vec2 uDelta;\n    varying vec2 vTexCoord;\n    const float nSamples = 15.0;\n    vec3 v3offset = vec3(12.9898, 78.233, 151.7182);\n    float random(vec3 scale) {\n      /* use the fragment position for a different seed per-pixel */\n      return fract(sin(dot(gl_FragCoord.xyz, scale)) * 43758.5453);\n    }\n    void main() {\n      vec4 color = vec4(0.0);\n      float total = 0.0;\n      float offset = random(v3offset);\n      for (float t = -nSamples; t <= nSamples; t++) {\n        float percent = (t + offset - 0.5) / nSamples;\n        float weight = 1.0 - abs(percent);\n        color += texture2D(uTexture, vTexCoord + uDelta * percent) * weight;\n        total += weight;\n      }\n      gl_FragColor = color / total;\n    }\n  ` as const;\n","import type { TClassProperties } from '../typedefs';\nimport { BaseFilter } from './BaseFilter';\nimport type { T2DPipelineState, TWebGLUniformLocationMap } from './typedefs';\nimport { classRegistry } from '../ClassRegistry';\nimport { fragmentSource } from './shaders/brightness';\nexport const brightnessDefaultValues: Partial<TClassProperties<Brightness>> = {\n  brightness: 0,\n  mainParameter: 'brightness',\n};\n\n/**\n * Brightness filter class\n * @example\n * const filter = new Brightness({\n *   brightness: 0.05\n * });\n * object.filters.push(filter);\n * object.applyFilters();\n */\nexport class Brightness extends BaseFilter {\n  /**\n   * Brightness value, from -1 to 1.\n   * translated to -255 to 255 for 2d\n   * 0.0039215686 is the part of 1 that get translated to 1 in 2d\n   * @param {Number} brightness\n   * @default\n   */\n  declare brightness: number;\n\n  static type = 'Brightness';\n\n  static defaults = brightnessDefaultValues;\n\n  getFragmentSource() {\n    return fragmentSource;\n  }\n\n  /**\n   * Apply the Brightness operation to a Uint8ClampedArray representing the pixels of an image.\n   *\n   * @param {Object} options\n   * @param {ImageData} options.imageData The Uint8ClampedArray to be filtered.\n   */\n  applyTo2d({ imageData: { data } }: T2DPipelineState) {\n    if (this.brightness === 0) {\n      return;\n    }\n    const brightness = Math.round(this.brightness * 255);\n    for (let i = 0; i < data.length; i += 4) {\n      data[i] = data[i] + brightness;\n      data[i + 1] = data[i + 1] + brightness;\n      data[i + 2] = data[i + 2] + brightness;\n    }\n  }\n\n  /**\n   * Return WebGL uniform locations for this filter's shader.\n   *\n   * @param {WebGLRenderingContext} gl The GL canvas context used to compile this filter's shader.\n   * @param {WebGLShaderProgram} program This filter's compiled shader program.\n   */\n  getUniformLocations(\n    gl: WebGLRenderingContext,\n    program: WebGLProgram\n  ): TWebGLUniformLocationMap {\n    return {\n      uBrightness: gl.getUniformLocation(program, 'uBrightness'),\n    };\n  }\n\n  /**\n   * Send data from this filter to its shader program's uniforms.\n   *\n   * @param {WebGLRenderingContext} gl The GL canvas context used to compile this filter's shader.\n   * @param {Object} uniformLocations A map of string uniform names to WebGLUniformLocation objects\n   */\n  sendUniformData(\n    gl: WebGLRenderingContext,\n    uniformLocations: TWebGLUniformLocationMap\n  ) {\n    gl.uniform1f(uniformLocations.uBrightness, this.brightness);\n  }\n}\n\nclassRegistry.setClass(Brightness);\n","export const fragmentSource = `\n  precision highp float;\n  uniform sampler2D uTexture;\n  uniform float uBrightness;\n  varying vec2 vTexCoord;\n  void main() {\n    vec4 color = texture2D(uTexture, vTexCoord);\n    color.rgb += uBrightness;\n    gl_FragColor = color;\n  }\n`;\n","export const fragmentSource = `\n  precision highp float;\n  uniform sampler2D uTexture;\n  varying vec2 vTexCoord;\n  uniform mat4 uColorMatrix;\n  uniform vec4 uConstants;\n  void main() {\n    vec4 color = texture2D(uTexture, vTexCoord);\n    color *= uColorMatrix;\n    color += uConstants;\n    gl_FragColor = color;\n  }`;\n","import type { TClassProperties } from '../typedefs';\nimport { BaseFilter } from './BaseFilter';\nimport type { T2DPipelineState, TWebGLUniformLocationMap } from './typedefs';\nimport { classRegistry } from '../ClassRegistry';\nimport { fragmentSource } from './shaders/colorMatrix';\nexport const colorMatrixDefaultValues: Partial<TClassProperties<ColorMatrix>> =\n  {\n    matrix: [1, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 1, 0],\n    mainParameter: 'matrix',\n    colorsOnly: true,\n  };\n\n/**\n   * Color Matrix filter class\n   * @see {@link http://fabricjs.com/image-filters|ImageFilters demo}\n   * @see {@Link http://phoboslab.org/log/2013/11/fast-image-filters-with-webgl demo}\n   * @example <caption>Kodachrome filter</caption>\n   * const filter = new ColorMatrix({\n   *  matrix: [\n       1.1285582396593525, -0.3967382283601348, -0.03992559172921793, 0, 63.72958762196502,\n       -0.16404339962244616, 1.0835251566291304, -0.05498805115633132, 0, 24.732407896706203,\n       -0.16786010706155763, -0.5603416277695248, 1.6014850761964943, 0, 35.62982807460946,\n       0, 0, 0, 1, 0\n      ]\n   * });\n   * object.filters.push(filter);\n   * object.applyFilters();\n   */\nexport class ColorMatrix extends BaseFilter {\n  /**\n   * Colormatrix for pixels.\n   * array of 20 floats. Numbers in positions 4, 9, 14, 19 loose meaning\n   * outside the -1, 1 range.\n   * 0.0039215686 is the part of 1 that get translated to 1 in 2d\n   * @param {Array} matrix array of 20 numbers.\n   * @default\n   */\n  declare matrix: number[];\n\n  /**\n   * Lock the colormatrix on the color part, skipping alpha, mainly for non webgl scenario\n   * to save some calculation\n   * @type Boolean\n   * @default true\n   */\n  declare colorsOnly: boolean;\n\n  static type = 'ColorMatrix';\n\n  static defaults = colorMatrixDefaultValues;\n\n  setOptions({ matrix, ...options }: Record<string, any>) {\n    if (matrix) {\n      // safeguard against mutation\n      this.matrix = [...matrix];\n    }\n    Object.assign(this, options);\n  }\n\n  getFragmentSource(): string {\n    return fragmentSource;\n  }\n\n  /**\n   * Apply the ColorMatrix operation to a Uint8Array representing the pixels of an image.\n   *\n   * @param {Object} options\n   * @param {ImageData} options.imageData The Uint8Array to be filtered.\n   */\n  applyTo2d(options: T2DPipelineState) {\n    const imageData = options.imageData,\n      data = imageData.data,\n      m = this.matrix,\n      colorsOnly = this.colorsOnly;\n\n    for (let i = 0; i < data.length; i += 4) {\n      const r = data[i];\n      const g = data[i + 1];\n      const b = data[i + 2];\n      if (colorsOnly) {\n        data[i] = r * m[0] + g * m[1] + b * m[2] + m[4] * 255;\n        data[i + 1] = r * m[5] + g * m[6] + b * m[7] + m[9] * 255;\n        data[i + 2] = r * m[10] + g * m[11] + b * m[12] + m[14] * 255;\n      } else {\n        const a = data[i + 3];\n        data[i] = r * m[0] + g * m[1] + b * m[2] + a * m[3] + m[4] * 255;\n        data[i + 1] = r * m[5] + g * m[6] + b * m[7] + a * m[8] + m[9] * 255;\n        data[i + 2] =\n          r * m[10] + g * m[11] + b * m[12] + a * m[13] + m[14] * 255;\n        data[i + 3] =\n          r * m[15] + g * m[16] + b * m[17] + a * m[18] + m[19] * 255;\n      }\n    }\n  }\n\n  /**\n   * Return WebGL uniform locations for this filter's shader.\n   *\n   * @param {WebGLRenderingContext} gl The GL canvas context used to compile this filter's shader.\n   * @param {WebGLShaderProgram} program This filter's compiled shader program.\n   */\n  getUniformLocations(\n    gl: WebGLRenderingContext,\n    program: WebGLProgram\n  ): TWebGLUniformLocationMap {\n    return {\n      uColorMatrix: gl.getUniformLocation(program, 'uColorMatrix'),\n      uConstants: gl.getUniformLocation(program, 'uConstants'),\n    };\n  }\n\n  /**\n   * Send data from this filter to its shader program's uniforms.\n   *\n   * @param {WebGLRenderingContext} gl The GL canvas context used to compile this filter's shader.\n   * @param {Object} uniformLocations A map of string uniform names to WebGLUniformLocation objects\n   */\n  sendUniformData(\n    gl: WebGLRenderingContext,\n    uniformLocations: TWebGLUniformLocationMap\n  ) {\n    const m = this.matrix,\n      matrix = [\n        m[0],\n        m[1],\n        m[2],\n        m[3],\n        m[5],\n        m[6],\n        m[7],\n        m[8],\n        m[10],\n        m[11],\n        m[12],\n        m[13],\n        m[15],\n        m[16],\n        m[17],\n        m[18],\n      ],\n      constants = [m[4], m[9], m[14], m[19]];\n    gl.uniformMatrix4fv(uniformLocations.uColorMatrix, false, matrix);\n    gl.uniform4fv(uniformLocations.uConstants, constants);\n  }\n}\n\nclassRegistry.setClass(ColorMatrix);\n","import { ColorMatrix, colorMatrixDefaultValues } from './ColorMatrix';\nimport { classRegistry } from '../ClassRegistry';\n\nexport function createColorMatrixFilter(key: string, matrix: number[]) {\n  const newClass = class extends ColorMatrix {\n    static type = key;\n\n    static defaults = {\n      ...colorMatrixDefaultValues,\n      /**\n       * Lock the matrix export for this kind of static, parameter less filters.\n       */\n      mainParameter: undefined,\n      matrix,\n    };\n  };\n  classRegistry.setClass(newClass, key);\n  return newClass;\n}\n\nexport const Brownie = createColorMatrixFilter(\n  'Brownie',\n  [\n    0.5997, 0.34553, -0.27082, 0, 0.186, -0.0377, 0.86095, 0.15059, 0, -0.1449,\n    0.24113, -0.07441, 0.44972, 0, -0.02965, 0, 0, 0, 1, 0,\n  ]\n);\n\nexport const Vintage = createColorMatrixFilter(\n  'Vintage',\n  [\n    0.62793, 0.32021, -0.03965, 0, 0.03784, 0.02578, 0.64411, 0.03259, 0,\n    0.02926, 0.0466, -0.08512, 0.52416, 0, 0.02023, 0, 0, 0, 1, 0,\n  ]\n);\n\nexport const Kodachrome = createColorMatrixFilter(\n  'Kodachrome',\n  [\n    1.12855, -0.39673, -0.03992, 0, 0.24991, -0.16404, 1.08352, -0.05498, 0,\n    0.09698, -0.16786, -0.56034, 1.60148, 0, 0.13972, 0, 0, 0, 1, 0,\n  ]\n);\n\nexport const Technicolor = createColorMatrixFilter(\n  'Technicolor',\n  [\n    1.91252, -0.85453, -0.09155, 0, 0.04624, -0.30878, 1.76589, -0.10601, 0,\n    -0.27589, -0.2311, -0.75018, 1.84759, 0, 0.12137, 0, 0, 0, 1, 0,\n  ]\n);\n\nexport const Polaroid = createColorMatrixFilter(\n  'Polaroid',\n  [\n    1.438, -0.062, -0.062, 0, 0, -0.122, 1.378, -0.122, 0, 0, -0.016, -0.016,\n    1.483, 0, 0, 0, 0, 0, 1, 0,\n  ]\n);\n\nexport const Sepia = createColorMatrixFilter(\n  'Sepia',\n  [\n    0.393, 0.769, 0.189, 0, 0, 0.349, 0.686, 0.168, 0, 0, 0.272, 0.534, 0.131,\n    0, 0, 0, 0, 0, 1, 0,\n  ]\n);\n\nexport const BlackWhite = createColorMatrixFilter(\n  'BlackWhite',\n  [\n    1.5, 1.5, 1.5, 0, -1, 1.5, 1.5, 1.5, 0, -1, 1.5, 1.5, 1.5, 0, -1, 0, 0, 0,\n    1, 0,\n  ]\n);\n","import { BaseFilter } from './BaseFilter';\nimport type { T2DPipelineState, TWebGLPipelineState } from './typedefs';\nimport { isWebGLPipelineState } from './utils';\nimport { classRegistry } from '../ClassRegistry';\n\n/**\n * A container class that knows how to apply a sequence of filters to an input image.\n */\nexport class Composed extends BaseFilter {\n  /**\n   * A non sparse array of filters to apply\n   */\n  declare subFilters: BaseFilter[];\n\n  static type = 'Composed';\n\n  constructor({\n    subFilters = [],\n    ...options\n  }: { subFilters?: BaseFilter[] } & Record<string, any> = {}) {\n    super(options);\n    this.subFilters = subFilters;\n  }\n\n  /**\n   * Apply this container's filters to the input image provided.\n   *\n   * @param {Object} options\n   * @param {Number} options.passes The number of filters remaining to be applied.\n   */\n  applyTo(options: TWebGLPipelineState | T2DPipelineState) {\n    if (isWebGLPipelineState(options)) {\n      options.passes += this.subFilters.length - 1;\n    }\n    this.subFilters.forEach((filter) => {\n      filter.applyTo(options);\n    });\n  }\n\n  /**\n   * Serialize this filter into JSON.\n   *\n   * @returns {Object} A JSON representation of this filter.\n   */\n  toObject() {\n    return {\n      ...super.toObject(),\n      subFilters: this.subFilters.map((filter) => filter.toObject()),\n    };\n  }\n\n  isNeutralState() {\n    return !this.subFilters.some((filter) => !filter.isNeutralState());\n  }\n\n  /**\n   * Deserialize a JSON definition of a ComposedFilter into a concrete instance.\n   * @static\n   * @param {oject} object Object to create an instance from\n   * @param {object} [options]\n   * @param {AbortSignal} [options.signal] handle aborting `BlendImage` filter loading, see https://developer.mozilla.org/en-US/docs/Web/API/AbortController/signal\n   * @returns {Promise<Composed>}\n   */\n  static fromObject(\n    object: Record<string, any>,\n    options: { signal: AbortSignal }\n  ) {\n    return Promise.all(\n      ((object.subFilters || []) as BaseFilter[]).map((filter) =>\n        classRegistry\n          .getClass<typeof BaseFilter>(filter.type)\n          .fromObject(filter, options)\n      )\n    ).then(\n      (enlivedFilters) => new this({ subFilters: enlivedFilters }) as BaseFilter\n    );\n  }\n}\n\nclassRegistry.setClass(Composed);\n","import type { TClassProperties } from '../typedefs';\nimport { BaseFilter } from './BaseFilter';\nimport type { T2DPipelineState, TWebGLUniformLocationMap } from './typedefs';\nimport { classRegistry } from '../ClassRegistry';\nimport { fragmentSource } from './shaders/constrast';\nexport const contrastDefaultValues: Partial<TClassProperties<Contrast>> = {\n  contrast: 0,\n  mainParameter: 'contrast',\n};\n\n/**\n * Contrast filter class\n * @example\n * const filter = new Contrast({\n *   contrast: 0.25\n * });\n * object.filters.push(filter);\n * object.applyFilters();\n */\nexport class Contrast extends BaseFilter {\n  /**\n   * contrast value, range from -1 to 1.\n   * @param {Number} contrast\n   * @default 0\n   */\n  declare contrast: number;\n\n  static type = 'Contrast';\n\n  static defaults = contrastDefaultValues;\n\n  getFragmentSource() {\n    return fragmentSource;\n  }\n  /**\n   * Apply the Contrast operation to a Uint8Array representing the pixels of an image.\n   *\n   * @param {Object} options\n   * @param {ImageData} options.imageData The Uint8Array to be filtered.\n   */\n  applyTo2d({ imageData: { data } }: T2DPipelineState) {\n    if (this.contrast === 0) {\n      return;\n    }\n    const contrast = Math.floor(this.contrast * 255),\n      contrastF = (259 * (contrast + 255)) / (255 * (259 - contrast));\n\n    for (let i = 0; i < data.length; i += 4) {\n      data[i] = contrastF * (data[i] - 128) + 128;\n      data[i + 1] = contrastF * (data[i + 1] - 128) + 128;\n      data[i + 2] = contrastF * (data[i + 2] - 128) + 128;\n    }\n  }\n\n  /**\n   * Return WebGL uniform locations for this filter's shader.\n   *\n   * @param {WebGLRenderingContext} gl The GL canvas context used to compile this filter's shader.\n   * @param {WebGLShaderProgram} program This filter's compiled shader program.\n   */\n  getUniformLocations(\n    gl: WebGLRenderingContext,\n    program: WebGLProgram\n  ): TWebGLUniformLocationMap {\n    return {\n      uContrast: gl.getUniformLocation(program, 'uContrast'),\n    };\n  }\n\n  /**\n   * Send data from this filter to its shader program's uniforms.\n   *\n   * @param {WebGLRenderingContext} gl The GL canvas context used to compile this filter's shader.\n   * @param {Object} uniformLocations A map of string uniform names to WebGLUniformLocation objects\n   */\n  sendUniformData(\n    gl: WebGLRenderingContext,\n    uniformLocations: TWebGLUniformLocationMap\n  ) {\n    gl.uniform1f(uniformLocations.uContrast, this.contrast);\n  }\n}\n\nclassRegistry.setClass(Contrast);\n","export const fragmentSource = `\n  precision highp float;\n  uniform sampler2D uTexture;\n  uniform float uContrast;\n  varying vec2 vTexCoord;\n  void main() {\n    vec4 color = texture2D(uTexture, vTexCoord);\n    float contrastF = 1.015 * (uContrast + 1.0) / (1.0 * (1.015 - uContrast));\n    color.rgb = contrastF * (color.rgb - 0.5) + 0.5;\n    gl_FragColor = color;\n  }`;\n","export const fragmentSource = {\n  Convolute_3_1: `\n    precision highp float;\n    uniform sampler2D uTexture;\n    uniform float uMatrix[9];\n    uniform float uStepW;\n    uniform float uStepH;\n    varying vec2 vTexCoord;\n    void main() {\n      vec4 color = vec4(0, 0, 0, 0);\n      for (float h = 0.0; h < 3.0; h+=1.0) {\n        for (float w = 0.0; w < 3.0; w+=1.0) {\n          vec2 matrixPos = vec2(uStepW * (w - 1), uStepH * (h - 1));\n          color += texture2D(uTexture, vTexCoord + matrixPos) * uMatrix[int(h * 3.0 + w)];\n        }\n      }\n      gl_FragColor = color;\n    }\n    `,\n  Convolute_3_0: `\n    precision highp float;\n    uniform sampler2D uTexture;\n    uniform float uMatrix[9];\n    uniform float uStepW;\n    uniform float uStepH;\n    varying vec2 vTexCoord;\n    void main() {\n      vec4 color = vec4(0, 0, 0, 1);\n      for (float h = 0.0; h < 3.0; h+=1.0) {\n        for (float w = 0.0; w < 3.0; w+=1.0) {\n          vec2 matrixPos = vec2(uStepW * (w - 1.0), uStepH * (h - 1.0));\n          color.rgb += texture2D(uTexture, vTexCoord + matrixPos).rgb * uMatrix[int(h * 3.0 + w)];\n        }\n      }\n      float alpha = texture2D(uTexture, vTexCoord).a;\n      gl_FragColor = color;\n      gl_FragColor.a = alpha;\n    }\n    `,\n  Convolute_5_1: `\n    precision highp float;\n    uniform sampler2D uTexture;\n    uniform float uMatrix[25];\n    uniform float uStepW;\n    uniform float uStepH;\n    varying vec2 vTexCoord;\n    void main() {\n      vec4 color = vec4(0, 0, 0, 0);\n      for (float h = 0.0; h < 5.0; h+=1.0) {\n        for (float w = 0.0; w < 5.0; w+=1.0) {\n          vec2 matrixPos = vec2(uStepW * (w - 2.0), uStepH * (h - 2.0));\n          color += texture2D(uTexture, vTexCoord + matrixPos) * uMatrix[int(h * 5.0 + w)];\n        }\n      }\n      gl_FragColor = color;\n    }\n    `,\n  Convolute_5_0: `\n    precision highp float;\n    uniform sampler2D uTexture;\n    uniform float uMatrix[25];\n    uniform float uStepW;\n    uniform float uStepH;\n    varying vec2 vTexCoord;\n    void main() {\n      vec4 color = vec4(0, 0, 0, 1);\n      for (float h = 0.0; h < 5.0; h+=1.0) {\n        for (float w = 0.0; w < 5.0; w+=1.0) {\n          vec2 matrixPos = vec2(uStepW * (w - 2.0), uStepH * (h - 2.0));\n          color.rgb += texture2D(uTexture, vTexCoord + matrixPos).rgb * uMatrix[int(h * 5.0 + w)];\n        }\n      }\n      float alpha = texture2D(uTexture, vTexCoord).a;\n      gl_FragColor = color;\n      gl_FragColor.a = alpha;\n    }\n    `,\n  Convolute_7_1: `\n    precision highp float;\n    uniform sampler2D uTexture;\n    uniform float uMatrix[49];\n    uniform float uStepW;\n    uniform float uStepH;\n    varying vec2 vTexCoord;\n    void main() {\n      vec4 color = vec4(0, 0, 0, 0);\n      for (float h = 0.0; h < 7.0; h+=1.0) {\n        for (float w = 0.0; w < 7.0; w+=1.0) {\n          vec2 matrixPos = vec2(uStepW * (w - 3.0), uStepH * (h - 3.0));\n          color += texture2D(uTexture, vTexCoord + matrixPos) * uMatrix[int(h * 7.0 + w)];\n        }\n      }\n      gl_FragColor = color;\n    }\n    `,\n  Convolute_7_0: `\n    precision highp float;\n    uniform sampler2D uTexture;\n    uniform float uMatrix[49];\n    uniform float uStepW;\n    uniform float uStepH;\n    varying vec2 vTexCoord;\n    void main() {\n      vec4 color = vec4(0, 0, 0, 1);\n      for (float h = 0.0; h < 7.0; h+=1.0) {\n        for (float w = 0.0; w < 7.0; w+=1.0) {\n          vec2 matrixPos = vec2(uStepW * (w - 3.0), uStepH * (h - 3.0));\n          color.rgb += texture2D(uTexture, vTexCoord + matrixPos).rgb * uMatrix[int(h * 7.0 + w)];\n        }\n      }\n      float alpha = texture2D(uTexture, vTexCoord).a;\n      gl_FragColor = color;\n      gl_FragColor.a = alpha;\n    }\n    `,\n  Convolute_9_1: `\n    precision highp float;\n    uniform sampler2D uTexture;\n    uniform float uMatrix[81];\n    uniform float uStepW;\n    uniform float uStepH;\n    varying vec2 vTexCoord;\n    void main() {\n      vec4 color = vec4(0, 0, 0, 0);\n      for (float h = 0.0; h < 9.0; h+=1.0) {\n        for (float w = 0.0; w < 9.0; w+=1.0) {\n          vec2 matrixPos = vec2(uStepW * (w - 4.0), uStepH * (h - 4.0));\n          color += texture2D(uTexture, vTexCoord + matrixPos) * uMatrix[int(h * 9.0 + w)];\n        }\n      }\n      gl_FragColor = color;\n    }\n    `,\n  Convolute_9_0: `\n    precision highp float;\n    uniform sampler2D uTexture;\n    uniform float uMatrix[81];\n    uniform float uStepW;\n    uniform float uStepH;\n    varying vec2 vTexCoord;\n    void main() {\n      vec4 color = vec4(0, 0, 0, 1);\n      for (float h = 0.0; h < 9.0; h+=1.0) {\n        for (float w = 0.0; w < 9.0; w+=1.0) {\n          vec2 matrixPos = vec2(uStepW * (w - 4.0), uStepH * (h - 4.0));\n          color.rgb += texture2D(uTexture, vTexCoord + matrixPos).rgb * uMatrix[int(h * 9.0 + w)];\n        }\n      }\n      float alpha = texture2D(uTexture, vTexCoord).a;\n      gl_FragColor = color;\n      gl_FragColor.a = alpha;\n    }\n    `,\n};\n","import type { TClassProperties } from '../typedefs';\nimport { BaseFilter } from './BaseFilter';\nimport type { T2DPipelineState, TWebGLUniformLocationMap } from './typedefs';\nimport { classRegistry } from '../ClassRegistry';\nimport { fragmentSource } from './shaders/convolute';\n\nexport const convoluteDefaultValues: Partial<TClassProperties<Convolute>> = {\n  opaque: false,\n  matrix: [0, 0, 0, 0, 1, 0, 0, 0, 0],\n};\n\n/**\n * Adapted from <a href=\"http://www.html5rocks.com/en/tutorials/canvas/imagefilters/\">html5rocks article</a>\n * @example <caption>Sharpen filter</caption>\n * const filter = new Convolute({\n *   matrix: [ 0, -1,  0,\n *            -1,  5, -1,\n *             0, -1,  0 ]\n * });\n * object.filters.push(filter);\n * object.applyFilters();\n * canvas.renderAll();\n * @example <caption>Blur filter</caption>\n * const filter = new Convolute({\n *   matrix: [ 1/9, 1/9, 1/9,\n *             1/9, 1/9, 1/9,\n *             1/9, 1/9, 1/9 ]\n * });\n * object.filters.push(filter);\n * object.applyFilters();\n * canvas.renderAll();\n * @example <caption>Emboss filter</caption>\n * const filter = new Convolute({\n *   matrix: [ 1,   1,  1,\n *             1, 0.7, -1,\n *            -1,  -1, -1 ]\n * });\n * object.filters.push(filter);\n * object.applyFilters();\n * canvas.renderAll();\n * @example <caption>Emboss filter with opaqueness</caption>\n * const filter = new Convolute({\n *   opaque: true,\n *   matrix: [ 1,   1,  1,\n *             1, 0.7, -1,\n *            -1,  -1, -1 ]\n * });\n * object.filters.push(filter);\n * object.applyFilters();\n * canvas.renderAll();\n */\nexport class Convolute extends BaseFilter {\n  /*\n   * Opaque value (true/false)\n   */\n  declare opaque: boolean;\n\n  /*\n   * matrix for the filter, max 9x9\n   */\n  declare matrix: number[];\n\n  static type = 'Convolute';\n\n  static defaults = convoluteDefaultValues;\n\n  getCacheKey() {\n    return `${this.type}_${Math.sqrt(this.matrix.length)}_${\n      this.opaque ? 1 : 0\n    }` as keyof typeof fragmentSource;\n  }\n\n  getFragmentSource() {\n    return fragmentSource[this.getCacheKey()];\n  }\n\n  /**\n   * Apply the Brightness operation to a Uint8ClampedArray representing the pixels of an image.\n   *\n   * @param {Object} options\n   * @param {ImageData} options.imageData The Uint8ClampedArray to be filtered.\n   */\n  applyTo2d(options: T2DPipelineState) {\n    const imageData = options.imageData,\n      data = imageData.data,\n      weights = this.matrix,\n      side = Math.round(Math.sqrt(weights.length)),\n      halfSide = Math.floor(side / 2),\n      sw = imageData.width,\n      sh = imageData.height,\n      output = options.ctx.createImageData(sw, sh),\n      dst = output.data,\n      // go through the destination image pixels\n      alphaFac = this.opaque ? 1 : 0;\n    let r, g, b, a, dstOff, scx, scy, srcOff, wt, x, y, cx, cy;\n\n    for (y = 0; y < sh; y++) {\n      for (x = 0; x < sw; x++) {\n        dstOff = (y * sw + x) * 4;\n        // calculate the weighed sum of the source image pixels that\n        // fall under the convolution matrix\n        r = 0;\n        g = 0;\n        b = 0;\n        a = 0;\n\n        for (cy = 0; cy < side; cy++) {\n          for (cx = 0; cx < side; cx++) {\n            scy = y + cy - halfSide;\n            scx = x + cx - halfSide;\n\n            // eslint-disable-next-line max-depth\n            if (scy < 0 || scy >= sh || scx < 0 || scx >= sw) {\n              continue;\n            }\n\n            srcOff = (scy * sw + scx) * 4;\n            wt = weights[cy * side + cx];\n\n            r += data[srcOff] * wt;\n            g += data[srcOff + 1] * wt;\n            b += data[srcOff + 2] * wt;\n            // eslint-disable-next-line max-depth\n            if (!alphaFac) {\n              a += data[srcOff + 3] * wt;\n            }\n          }\n        }\n        dst[dstOff] = r;\n        dst[dstOff + 1] = g;\n        dst[dstOff + 2] = b;\n        if (!alphaFac) {\n          dst[dstOff + 3] = a;\n        } else {\n          dst[dstOff + 3] = data[dstOff + 3];\n        }\n      }\n    }\n    options.imageData = output;\n  }\n\n  /**\n   * Return WebGL uniform locations for this filter's shader.\n   *\n   * @param {WebGLRenderingContext} gl The GL canvas context used to compile this filter's shader.\n   * @param {WebGLShaderProgram} program This filter's compiled shader program.\n   */\n  getUniformLocations(\n    gl: WebGLRenderingContext,\n    program: WebGLProgram\n  ): TWebGLUniformLocationMap {\n    return {\n      uMatrix: gl.getUniformLocation(program, 'uMatrix'),\n      uOpaque: gl.getUniformLocation(program, 'uOpaque'),\n      uHalfSize: gl.getUniformLocation(program, 'uHalfSize'),\n      uSize: gl.getUniformLocation(program, 'uSize'),\n    };\n  }\n\n  /**\n   * Send data from this filter to its shader program's uniforms.\n   *\n   * @param {WebGLRenderingContext} gl The GL canvas context used to compile this filter's shader.\n   * @param {Object} uniformLocations A map of string uniform names to WebGLUniformLocation objects\n   */\n  sendUniformData(\n    gl: WebGLRenderingContext,\n    uniformLocations: TWebGLUniformLocationMap\n  ) {\n    gl.uniform1fv(uniformLocations.uMatrix, this.matrix);\n  }\n\n  /**\n   * Returns object representation of an instance\n   * @return {Object} Object representation of an instance\n   */\n  toObject() {\n    return {\n      ...super.toObject(),\n      opaque: this.opaque,\n      matrix: [...this.matrix],\n    };\n  }\n}\n\nclassRegistry.setClass(Convolute);\n","export const fragmentSource = `\n  precision highp float;\n  uniform sampler2D uTexture;\n  uniform vec3 uGamma;\n  varying vec2 vTexCoord;\n  void main() {\n    vec4 color = texture2D(uTexture, vTexCoord);\n    vec3 correction = (1.0 / uGamma);\n    color.r = pow(color.r, correction.r);\n    color.g = pow(color.g, correction.g);\n    color.b = pow(color.b, correction.b);\n    gl_FragColor = color;\n    gl_FragColor.rgb *= color.a;\n  }\n`;\n","import type { TClassProperties } from '../typedefs';\nimport { BaseFilter } from './BaseFilter';\nimport type { T2DPipelineState, TWebGLUniformLocationMap } from './typedefs';\nimport { classRegistry } from '../ClassRegistry';\nimport { fragmentSource } from './shaders/gamma';\nexport type GammaInput = [number, number, number];\n\nexport const gammaDefaultValues: Partial<TClassProperties<Gamma>> = {\n  mainParameter: 'gamma',\n  gamma: [1, 1, 1],\n};\n\n/**\n * Gamma filter class\n * @example\n * const filter = new Gamma({\n *   gamma: [1, 0.5, 2.1]\n * });\n * object.filters.push(filter);\n * object.applyFilters();\n */\nexport class Gamma extends BaseFilter {\n  /**\n   * Gamma array value, from 0.01 to 2.2.\n   * @param {Array} gamma\n   * @default\n   */\n  declare gamma: GammaInput;\n  declare rgbValues?: {\n    r: Uint8Array;\n    g: Uint8Array;\n    b: Uint8Array;\n  };\n\n  static type = 'Gamma';\n\n  static defaults = gammaDefaultValues;\n\n  getFragmentSource() {\n    return fragmentSource;\n  }\n\n  constructor({ gamma = [1, 1, 1], ...options }: { gamma?: GammaInput } = {}) {\n    super(options);\n    this.gamma = gamma;\n  }\n\n  /**\n   * Apply the Gamma operation to a Uint8Array representing the pixels of an image.\n   *\n   * @param {Object} options\n   * @param {ImageData} options.imageData The Uint8Array to be filtered.\n   */\n  applyTo2d({ imageData: { data } }: T2DPipelineState) {\n    const gamma = this.gamma,\n      rInv = 1 / gamma[0],\n      gInv = 1 / gamma[1],\n      bInv = 1 / gamma[2];\n\n    if (!this.rgbValues) {\n      this.rgbValues = {\n        r: new Uint8Array(256),\n        g: new Uint8Array(256),\n        b: new Uint8Array(256),\n      };\n    }\n\n    // This is an optimization - pre-compute a look-up table for each color channel\n    // instead of performing these pow calls for each pixel in the image.\n    const rgb = this.rgbValues;\n    for (let i = 0; i < 256; i++) {\n      rgb.r[i] = Math.pow(i / 255, rInv) * 255;\n      rgb.g[i] = Math.pow(i / 255, gInv) * 255;\n      rgb.b[i] = Math.pow(i / 255, bInv) * 255;\n    }\n    for (let i = 0; i < data.length; i += 4) {\n      data[i] = rgb.r[data[i]];\n      data[i + 1] = rgb.g[data[i + 1]];\n      data[i + 2] = rgb.b[data[i + 2]];\n    }\n  }\n\n  /**\n   * Return WebGL uniform locations for this filter's shader.\n   *\n   * @param {WebGLRenderingContext} gl The GL canvas context used to compile this filter's shader.\n   * @param {WebGLShaderProgram} program This filter's compiled shader program.\n   */\n  getUniformLocations(\n    gl: WebGLRenderingContext,\n    program: WebGLProgram\n  ): TWebGLUniformLocationMap {\n    return {\n      uGamma: gl.getUniformLocation(program, 'uGamma'),\n    };\n  }\n\n  /**\n   * Send data from this filter to its shader program's uniforms.\n   *\n   * @param {WebGLRenderingContext} gl The GL canvas context used to compile this filter's shader.\n   * @param {Object} uniformLocations A map of string uniform names to WebGLUniformLocation objects\n   */\n  sendUniformData(\n    gl: WebGLRenderingContext,\n    uniformLocations: TWebGLUniformLocationMap\n  ) {\n    gl.uniform3fv(uniformLocations.uGamma, this.gamma);\n  }\n}\n\nclassRegistry.setClass(Gamma);\n","import type { TGrayscaleMode } from '../Grayscale';\n\nexport const fragmentSource: Record<TGrayscaleMode, string> = {\n  average: `\n    precision highp float;\n    uniform sampler2D uTexture;\n    varying vec2 vTexCoord;\n    void main() {\n      vec4 color = texture2D(uTexture, vTexCoord);\n      float average = (color.r + color.b + color.g) / 3.0;\n      gl_FragColor = vec4(average, average, average, color.a);\n    }\n    `,\n  lightness: `\n    precision highp float;\n    uniform sampler2D uTexture;\n    uniform int uMode;\n    varying vec2 vTexCoord;\n    void main() {\n      vec4 col = texture2D(uTexture, vTexCoord);\n      float average = (max(max(col.r, col.g),col.b) + min(min(col.r, col.g),col.b)) / 2.0;\n      gl_FragColor = vec4(average, average, average, col.a);\n    }\n    `,\n  luminosity: `\n    precision highp float;\n    uniform sampler2D uTexture;\n    uniform int uMode;\n    varying vec2 vTexCoord;\n    void main() {\n      vec4 col = texture2D(uTexture, vTexCoord);\n      float average = 0.21 * col.r + 0.72 * col.g + 0.07 * col.b;\n      gl_FragColor = vec4(average, average, average, col.a);\n    }\n    `,\n};\n","import type { TClassProperties } from '../typedefs';\nimport { BaseFilter } from './BaseFilter';\nimport type { T2DPipelineState, TWebGLUniformLocationMap } from './typedefs';\nimport { classRegistry } from '../ClassRegistry';\nimport { fragmentSource } from './shaders/grayscale';\n\nexport type TGrayscaleMode = 'average' | 'lightness' | 'luminosity';\n\nexport const grayscaleDefaultValues: Partial<TClassProperties<Grayscale>> = {\n  mode: 'average',\n  mainParameter: 'mode',\n};\n\n/**\n * Grayscale image filter class\n * @example\n * const filter = new Grayscale();\n * object.filters.push(filter);\n * object.applyFilters();\n */\nexport class Grayscale extends BaseFilter {\n  declare mode: TGrayscaleMode;\n\n  static type = 'Grayscale';\n\n  static defaults = grayscaleDefaultValues;\n\n  /**\n   * Apply the Grayscale operation to a Uint8Array representing the pixels of an image.\n   *\n   * @param {Object} options\n   * @param {ImageData} options.imageData The Uint8Array to be filtered.\n   */\n  applyTo2d({ imageData: { data } }: T2DPipelineState) {\n    for (let i = 0, value: number; i < data.length; i += 4) {\n      switch (this.mode) {\n        case 'average':\n          value = (data[i] + data[i + 1] + data[i + 2]) / 3;\n          break;\n        case 'lightness':\n          value =\n            (Math.min(data[i], data[i + 1], data[i + 2]) +\n              Math.max(data[i], data[i + 1], data[i + 2])) /\n            2;\n          break;\n        case 'luminosity':\n          value = 0.21 * data[i] + 0.72 * data[i + 1] + 0.07 * data[i + 2];\n          break;\n      }\n\n      data[i] = value;\n      data[i + 1] = value;\n      data[i + 2] = value;\n    }\n  }\n\n  getCacheKey() {\n    return `${this.type}_${this.mode}`;\n  }\n\n  getFragmentSource() {\n    return fragmentSource[this.mode];\n  }\n\n  /**\n   * Return WebGL uniform locations for this filter's shader.\n   *\n   * @param {WebGLRenderingContext} gl The GL canvas context used to compile this filter's shader.\n   * @param {WebGLShaderProgram} program This filter's compiled shader program.\n   */\n  getUniformLocations(\n    gl: WebGLRenderingContext,\n    program: WebGLProgram\n  ): TWebGLUniformLocationMap {\n    return {\n      uMode: gl.getUniformLocation(program, 'uMode'),\n    };\n  }\n\n  /**\n   * Send data from this filter to its shader program's uniforms.\n   *\n   * @param {WebGLRenderingContext} gl The GL canvas context used to compile this filter's shader.\n   * @param {Object} uniformLocations A map of string uniform names to WebGLUniformLocation objects\n   */\n  sendUniformData(\n    gl: WebGLRenderingContext,\n    uniformLocations: TWebGLUniformLocationMap\n  ) {\n    const mode = 1;\n    gl.uniform1i(uniformLocations.uMode, mode);\n  }\n\n  /**\n   * Grayscale filter isNeutralState implementation\n   * The filter is never neutral\n   * on the image\n   **/\n  isNeutralState() {\n    return false;\n  }\n}\n\nclassRegistry.setClass(Grayscale);\n","import type { TClassProperties } from '../typedefs';\nimport { cos } from '../util/misc/cos';\nimport { sin } from '../util/misc/sin';\nimport { ColorMatrix } from './ColorMatrix';\nimport type { TWebGLPipelineState, T2DPipelineState } from './typedefs';\nimport { classRegistry } from '../ClassRegistry';\n\nexport const hueRotationDefaultValues: Partial<TClassProperties<HueRotation>> =\n  {\n    rotation: 0,\n    mainParameter: 'rotation',\n  };\n\n/**\n * HueRotation filter class\n * @example\n * const filter = new HueRotation({\n *   rotation: -0.5\n * });\n * object.filters.push(filter);\n * object.applyFilters();\n */\n// @ts-expect-error some babbling about mainParameter\nexport class HueRotation extends ColorMatrix {\n  /**\n   * HueRotation value, from -1 to 1.\n   */\n  declare rotation: number;\n\n  static type = 'HueRotation';\n\n  static defaults = hueRotationDefaultValues;\n\n  calculateMatrix() {\n    const rad = this.rotation * Math.PI,\n      cosine = cos(rad),\n      sine = sin(rad),\n      aThird = 1 / 3,\n      aThirdSqtSin = Math.sqrt(aThird) * sine,\n      OneMinusCos = 1 - cosine;\n    this.matrix = [1, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 1, 0];\n    this.matrix[0] = cosine + OneMinusCos / 3;\n    this.matrix[1] = aThird * OneMinusCos - aThirdSqtSin;\n    this.matrix[2] = aThird * OneMinusCos + aThirdSqtSin;\n    this.matrix[5] = aThird * OneMinusCos + aThirdSqtSin;\n    this.matrix[6] = cosine + aThird * OneMinusCos;\n    this.matrix[7] = aThird * OneMinusCos - aThirdSqtSin;\n    this.matrix[10] = aThird * OneMinusCos - aThirdSqtSin;\n    this.matrix[11] = aThird * OneMinusCos + aThirdSqtSin;\n    this.matrix[12] = cosine + aThird * OneMinusCos;\n  }\n\n  isNeutralState() {\n    this.calculateMatrix();\n    return super.isNeutralState();\n  }\n\n  applyTo(options: TWebGLPipelineState | T2DPipelineState) {\n    this.calculateMatrix();\n    super.applyTo(options);\n  }\n}\n\nclassRegistry.setClass(HueRotation);\n","import type { TClassProperties } from '../typedefs';\nimport { BaseFilter } from './BaseFilter';\nimport type { T2DPipelineState, TWebGLUniformLocationMap } from './typedefs';\nimport { classRegistry } from '../ClassRegistry';\nimport { fragmentSource } from './shaders/invert';\n\nexport const invertDefaultValues: Partial<TClassProperties<Invert>> = {\n  alpha: false,\n  invert: true,\n  mainParameter: 'invert',\n};\n\n/**\n * @example\n * const filter = new Invert();\n * object.filters.push(filter);\n * object.applyFilters(canvas.renderAll.bind(canvas));\n */\nexport class Invert extends BaseFilter {\n  /**\n   * Invert also alpha.\n   * @param {Boolean} alpha\n   * @default\n   **/\n  declare alpha: boolean;\n\n  /**\n   * Filter invert. if false, does nothing\n   * @param {Boolean} invert\n   * @default\n   */\n  declare invert: boolean;\n\n  static type = 'Invert';\n\n  static defaults = invertDefaultValues;\n\n  /**\n   * Apply the Invert operation to a Uint8Array representing the pixels of an image.\n   *\n   * @param {Object} options\n   * @param {ImageData} options.imageData The Uint8Array to be filtered.\n   */\n  applyTo2d({ imageData: { data } }: T2DPipelineState) {\n    for (let i = 0; i < data.length; i += 4) {\n      data[i] = 255 - data[i];\n      data[i + 1] = 255 - data[i + 1];\n      data[i + 2] = 255 - data[i + 2];\n\n      if (this.alpha) {\n        data[i + 3] = 255 - data[i + 3];\n      }\n    }\n  }\n\n  protected getFragmentSource(): string {\n    return fragmentSource;\n  }\n\n  /**\n   * Invert filter isNeutralState implementation\n   * Used only in image applyFilters to discard filters that will not have an effect\n   * on the image\n   * @param {Object} options\n   **/\n  isNeutralState() {\n    return !this.invert;\n  }\n\n  /**\n   * Return WebGL uniform locations for this filter's shader.\n   *\n   * @param {WebGLRenderingContext} gl The GL canvas context used to compile this filter's shader.\n   * @param {WebGLShaderProgram} program This filter's compiled shader program.\n   */\n  getUniformLocations(\n    gl: WebGLRenderingContext,\n    program: WebGLProgram\n  ): TWebGLUniformLocationMap {\n    return {\n      uInvert: gl.getUniformLocation(program, 'uInvert'),\n      uAlpha: gl.getUniformLocation(program, 'uAlpha'),\n    };\n  }\n\n  /**\n   * Send data from this filter to its shader program's uniforms.\n   *\n   * @param {WebGLRenderingContext} gl The GL canvas context used to compile this filter's shader.\n   * @param {Object} uniformLocations A map of string uniform names to WebGLUniformLocation objects\n   */\n  sendUniformData(\n    gl: WebGLRenderingContext,\n    uniformLocations: TWebGLUniformLocationMap\n  ) {\n    gl.uniform1i(uniformLocations.uInvert, Number(this.invert));\n    gl.uniform1i(uniformLocations.uAlpha, Number(this.alpha));\n  }\n}\n\nclassRegistry.setClass(Invert);\n","export const fragmentSource = `\n  precision highp float;\n  uniform sampler2D uTexture;\n  uniform int uInvert;\n  uniform int uAlpha;\n  varying vec2 vTexCoord;\n  void main() {\n    vec4 color = texture2D(uTexture, vTexCoord);\n    if (uInvert == 1) {\n      if (uAlpha == 1) {\n        gl_FragColor = vec4(1.0 - color.r,1.0 -color.g,1.0 -color.b,1.0 -color.a);\n      } else {\n        gl_FragColor = vec4(1.0 - color.r,1.0 -color.g,1.0 -color.b,color.a);\n      }\n    } else {\n      gl_FragColor = color;\n    }\n  }\n`;\n","import type { TClassProperties } from '../typedefs';\nimport { BaseFilter } from './BaseFilter';\nimport type { T2DPipelineState, TWebGLUniformLocationMap } from './typedefs';\nimport { classRegistry } from '../ClassRegistry';\nimport { fragmentSource } from './shaders/noise';\n\nexport const noiseDefaultValues: Partial<TClassProperties<Noise>> = {\n  mainParameter: 'noise',\n  noise: 0,\n};\n\n/**\n * Noise filter class\n * @example\n * const filter = new Noise({\n *   noise: 700\n * });\n * object.filters.push(filter);\n * object.applyFilters();\n * canvas.renderAll();\n */\nexport class Noise extends BaseFilter {\n  /**\n   * Noise value, from\n   * @param {Number} noise\n   * @default\n   */\n  declare noise: number;\n\n  static type = 'Noise';\n\n  static defaults = noiseDefaultValues;\n\n  getFragmentSource() {\n    return fragmentSource;\n  }\n\n  /**\n   * Apply the Brightness operation to a Uint8ClampedArray representing the pixels of an image.\n   *\n   * @param {Object} options\n   * @param {ImageData} options.imageData The Uint8ClampedArray to be filtered.\n   */\n  applyTo2d({ imageData: { data } }: T2DPipelineState) {\n    if (this.noise === 0) {\n      return;\n    }\n    const noise = this.noise;\n    for (let i = 0; i < data.length; i += 4) {\n      const rand = (0.5 - Math.random()) * noise;\n      data[i] += rand;\n      data[i + 1] += rand;\n      data[i + 2] += rand;\n    }\n  }\n\n  /**\n   * Return WebGL uniform locations for this filter's shader.\n   *\n   * @param {WebGLRenderingContext} gl The GL canvas context used to compile this filter's shader.\n   * @param {WebGLShaderProgram} program This filter's compiled shader program.\n   */\n  getUniformLocations(\n    gl: WebGLRenderingContext,\n    program: WebGLProgram\n  ): TWebGLUniformLocationMap {\n    return {\n      uNoise: gl.getUniformLocation(program, 'uNoise'),\n      uSeed: gl.getUniformLocation(program, 'uSeed'),\n    };\n  }\n\n  /**\n   * Send data from this filter to its shader program's uniforms.\n   *\n   * @param {WebGLRenderingContext} gl The GL canvas context used to compile this filter's shader.\n   * @param {Object} uniformLocations A map of string uniform names to WebGLUniformLocation objects\n   */\n  sendUniformData(\n    gl: WebGLRenderingContext,\n    uniformLocations: TWebGLUniformLocationMap\n  ) {\n    gl.uniform1f(uniformLocations.uNoise, this.noise / 255);\n    gl.uniform1f(uniformLocations.uSeed, Math.random());\n  }\n\n  /**\n   * Returns object representation of an instance\n   * @return {Object} Object representation of an instance\n   */\n  toObject() {\n    return { ...super.toObject(), noise: this.noise };\n  }\n}\n\nclassRegistry.setClass(Noise);\n","export const fragmentSource = `\n  precision highp float;\n  uniform sampler2D uTexture;\n  uniform float uStepH;\n  uniform float uNoise;\n  uniform float uSeed;\n  varying vec2 vTexCoord;\n  float rand(vec2 co, float seed, float vScale) {\n    return fract(sin(dot(co.xy * vScale ,vec2(12.9898 , 78.233))) * 43758.5453 * (seed + 0.01) / 2.0);\n  }\n  void main() {\n    vec4 color = texture2D(uTexture, vTexCoord);\n    color.rgb += (0.5 - rand(vTexCoord, uSeed, 0.1 / uStepH)) * uNoise;\n    gl_FragColor = color;\n  }\n`;\n","import type { TClassProperties } from '../typedefs';\nimport { BaseFilter } from './BaseFilter';\nimport type { T2DPipelineState, TWebGLUniformLocationMap } from './typedefs';\nimport { classRegistry } from '../ClassRegistry';\nimport { fragmentSource } from './shaders/pixelate';\n\nexport const pixelateDefaultValues: Partial<TClassProperties<Pixelate>> = {\n  blocksize: 4,\n  mainParameter: 'blocksize',\n};\n\n/**\n * Pixelate filter class\n * @example\n * const filter = new Pixelate({\n *   blocksize: 8\n * });\n * object.filters.push(filter);\n * object.applyFilters();\n */\nexport class Pixelate extends BaseFilter {\n  declare blocksize: number;\n\n  static type = 'Pixelate';\n\n  static defaults = pixelateDefaultValues;\n\n  /**\n   * Apply the Pixelate operation to a Uint8ClampedArray representing the pixels of an image.\n   *\n   * @param {Object} options\n   * @param {ImageData} options.imageData The Uint8ClampedArray to be filtered.\n   */\n  applyTo2d({ imageData: { data, width, height } }: T2DPipelineState) {\n    for (let i = 0; i < height; i += this.blocksize) {\n      for (let j = 0; j < width; j += this.blocksize) {\n        const index = i * 4 * width + j * 4;\n        const r = data[index];\n        const g = data[index + 1];\n        const b = data[index + 2];\n        const a = data[index + 3];\n\n        for (let _i = i; _i < Math.min(i + this.blocksize, height); _i++) {\n          for (let _j = j; _j < Math.min(j + this.blocksize, width); _j++) {\n            const index = _i * 4 * width + _j * 4;\n            data[index] = r;\n            data[index + 1] = g;\n            data[index + 2] = b;\n            data[index + 3] = a;\n          }\n        }\n      }\n    }\n  }\n\n  /**\n   * Indicate when the filter is not gonna apply changes to the image\n   **/\n  isNeutralState() {\n    return this.blocksize === 1;\n  }\n\n  protected getFragmentSource(): string {\n    return fragmentSource;\n  }\n\n  /**\n   * Return WebGL uniform locations for this filter's shader.\n   *\n   * @param {WebGLRenderingContext} gl The GL canvas context used to compile this filter's shader.\n   * @param {WebGLShaderProgram} program This filter's compiled shader program.\n   */\n  getUniformLocations(\n    gl: WebGLRenderingContext,\n    program: WebGLProgram\n  ): TWebGLUniformLocationMap {\n    return {\n      uBlocksize: gl.getUniformLocation(program, 'uBlocksize'),\n      uStepW: gl.getUniformLocation(program, 'uStepW'),\n      uStepH: gl.getUniformLocation(program, 'uStepH'),\n    };\n  }\n\n  /**\n   * Send data from this filter to its shader program's uniforms.\n   *\n   * @param {WebGLRenderingContext} gl The GL canvas context used to compile this filter's shader.\n   * @param {Object} uniformLocations A map of string uniform names to WebGLUniformLocation objects\n   */\n  sendUniformData(\n    gl: WebGLRenderingContext,\n    uniformLocations: TWebGLUniformLocationMap\n  ) {\n    gl.uniform1f(uniformLocations.uBlocksize, this.blocksize);\n  }\n}\n\nclassRegistry.setClass(Pixelate);\n","export const fragmentSource = `\n  precision highp float;\n  uniform sampler2D uTexture;\n  uniform float uBlocksize;\n  uniform float uStepW;\n  uniform float uStepH;\n  varying vec2 vTexCoord;\n  void main() {\n    float blockW = uBlocksize * uStepW;\n    float blockH = uBlocksize * uStepW;\n    int posX = int(vTexCoord.x / blockW);\n    int posY = int(vTexCoord.y / blockH);\n    float fposX = float(posX);\n    float fposY = float(posY);\n    vec2 squareCoords = vec2(fposX * blockW, fposY * blockH);\n    vec4 color = texture2D(uTexture, squareCoords);\n    gl_FragColor = color;\n  }\n`;\n","import { Color } from '../color/Color';\nimport type { TClassProperties } from '../typedefs';\nimport { BaseFilter } from './BaseFilter';\nimport type { T2DPipelineState, TWebGLUniformLocationMap } from './typedefs';\nimport { classRegistry } from '../ClassRegistry';\nimport { fragmentShader } from './shaders/removeColor';\nexport const removeColorDefaultValues: Partial<TClassProperties<RemoveColor>> =\n  {\n    color: '#FFFFFF',\n    distance: 0.02,\n    useAlpha: false,\n  };\n\n/**\n * Remove white filter class\n * @example\n * const filter = new RemoveColor({\n *   threshold: 0.2,\n * });\n * object.filters.push(filter);\n * object.applyFilters();\n * canvas.renderAll();\n */\nexport class RemoveColor extends BaseFilter {\n  /**\n   * Color to remove, in any format understood by {@link Color}.\n   * @param {String} type\n   * @default\n   */\n  declare color: string;\n\n  /**\n   * distance to actual color, as value up or down from each r,g,b\n   * between 0 and 1\n   **/\n  declare distance: number;\n\n  /**\n   * For color to remove inside distance, use alpha channel for a smoother deletion\n   * NOT IMPLEMENTED YET\n   **/\n  declare useAlpha: boolean;\n\n  static type = 'RemoveColor';\n\n  static defaults = removeColorDefaultValues;\n\n  getFragmentShader() {\n    return fragmentShader;\n  }\n\n  /**\n   * Applies filter to canvas element\n   * @param {Object} canvasEl Canvas element to apply filter to\n   */\n  applyTo2d({ imageData: { data } }: T2DPipelineState) {\n    const distance = this.distance * 255,\n      source = new Color(this.color).getSource(),\n      lowC = [source[0] - distance, source[1] - distance, source[2] - distance],\n      highC = [\n        source[0] + distance,\n        source[1] + distance,\n        source[2] + distance,\n      ];\n\n    for (let i = 0; i < data.length; i += 4) {\n      const r = data[i];\n      const g = data[i + 1];\n      const b = data[i + 2];\n\n      if (\n        r > lowC[0] &&\n        g > lowC[1] &&\n        b > lowC[2] &&\n        r < highC[0] &&\n        g < highC[1] &&\n        b < highC[2]\n      ) {\n        data[i + 3] = 0;\n      }\n    }\n  }\n\n  /**\n   * Return WebGL uniform locations for this filter's shader.\n   *\n   * @param {WebGLRenderingContext} gl The GL canvas context used to compile this filter's shader.\n   * @param {WebGLShaderProgram} program This filter's compiled shader program.\n   */\n  getUniformLocations(\n    gl: WebGLRenderingContext,\n    program: WebGLProgram\n  ): TWebGLUniformLocationMap {\n    return {\n      uLow: gl.getUniformLocation(program, 'uLow'),\n      uHigh: gl.getUniformLocation(program, 'uHigh'),\n    };\n  }\n\n  /**\n   * Send data from this filter to its shader program's uniforms.\n   *\n   * @param {WebGLRenderingContext} gl The GL canvas context used to compile this filter's shader.\n   * @param {Object} uniformLocations A map of string uniform names to WebGLUniformLocation objects\n   */\n  sendUniformData(\n    gl: WebGLRenderingContext,\n    uniformLocations: TWebGLUniformLocationMap\n  ) {\n    const source = new Color(this.color).getSource(),\n      distance = this.distance,\n      lowC = [\n        0 + source[0] / 255 - distance,\n        0 + source[1] / 255 - distance,\n        0 + source[2] / 255 - distance,\n        1,\n      ],\n      highC = [\n        source[0] / 255 + distance,\n        source[1] / 255 + distance,\n        source[2] / 255 + distance,\n        1,\n      ];\n    gl.uniform4fv(uniformLocations.uLow, lowC);\n    gl.uniform4fv(uniformLocations.uHigh, highC);\n  }\n\n  /**\n   * Returns object representation of an instance\n   * @return {Object} Object representation of an instance\n   */\n  toObject() {\n    return { ...super.toObject(), color: this.color, distance: this.distance };\n  }\n}\n\nclassRegistry.setClass(RemoveColor);\n","export const fragmentShader = `\nprecision highp float;\nuniform sampler2D uTexture;\nuniform vec4 uLow;\nuniform vec4 uHigh;\nvarying vec2 vTexCoord;\nvoid main() {\n  gl_FragColor = texture2D(uTexture, vTexCoord);\n  if(all(greaterThan(gl_FragColor.rgb,uLow.rgb)) && all(greaterThan(uHigh.rgb,gl_FragColor.rgb))) {\n    gl_FragColor.a = 0.0;\n  }\n}\n`;\n","import type { TClassProperties } from '../typedefs';\nimport { BaseFilter } from './BaseFilter';\nimport type {\n  T2DPipelineState,\n  TWebGLPipelineState,\n  TWebGLUniformLocationMap,\n} from './typedefs';\nimport { isWebGLPipelineState } from './utils';\nimport { classRegistry } from '../ClassRegistry';\nimport { createCanvasElement } from '../util/misc/dom';\nimport type { XY } from '../Point';\n\nexport const resizeDefaultValues: Partial<TClassProperties<Resize>> = {\n  resizeType: 'hermite',\n  scaleX: 1,\n  scaleY: 1,\n  lanczosLobes: 3,\n  fragmentSourceTOP: `\n    precision highp float;\n    uniform sampler2D uTexture;\n    uniform vec2 uDelta;\n    varying vec2 vTexCoord;\n  `,\n};\n\nexport type TResizeType = 'bilinear' | 'hermite' | 'sliceHack' | 'lanczos';\n\ntype ResizeDuring2DResize = Resize & {\n  rcpScaleX: number;\n  rcpScaleY: number;\n};\n\ntype ResizeDuringWEBGLResize = Resize & {\n  rcpScaleX: number;\n  rcpScaleY: number;\n  horizontal: boolean;\n  width: number;\n  height: number;\n  taps: number[];\n  tempScale: number;\n  dH: number;\n  dW: number;\n};\n\n/**\n * Resize image filter class\n * @example\n * const filter = new Resize();\n * object.filters.push(filter);\n * object.applyFilters(canvas.renderAll.bind(canvas));\n */\nexport class Resize extends BaseFilter {\n  /**\n   * Resize type\n   * for webgl resizeType is just lanczos, for canvas2d can be:\n   * bilinear, hermite, sliceHack, lanczos.\n   * @default\n   */\n  declare resizeType: TResizeType;\n\n  /**\n   * Scale factor for resizing, x axis\n   * @param {Number} scaleX\n   * @default\n   */\n  declare scaleX: number;\n\n  /**\n   * Scale factor for resizing, y axis\n   * @param {Number} scaleY\n   * @default\n   */\n  declare scaleY: number;\n\n  /**\n   * LanczosLobes parameter for lanczos filter, valid for resizeType lanczos\n   * @param {Number} lanczosLobes\n   * @default\n   */\n  declare lanczosLobes: number;\n\n  declare fragmentSourceTOP: string;\n\n  static type = 'Resize';\n\n  static defaults = resizeDefaultValues;\n\n  /**\n   * Return WebGL uniform locations for this filter's shader.\n   *\n   * @param {WebGLRenderingContext} gl The GL canvas context used to compile this filter's shader.\n   * @param {WebGLShaderProgram} program This filter's compiled shader program.\n   */\n  getUniformLocations(gl: WebGLRenderingContext, program: WebGLProgram) {\n    return {\n      uDelta: gl.getUniformLocation(program, 'uDelta'),\n      uTaps: gl.getUniformLocation(program, 'uTaps'),\n    };\n  }\n\n  /**\n   * Send data from this filter to its shader program's uniforms.\n   *\n   * @param {WebGLRenderingContext} gl The GL canvas context used to compile this filter's shader.\n   * @param {Object} uniformLocations A map of string uniform names to WebGLUniformLocation objects\n   */\n  sendUniformData(\n    this: ResizeDuringWEBGLResize,\n    gl: WebGLRenderingContext,\n    uniformLocations: TWebGLUniformLocationMap\n  ) {\n    gl.uniform2fv(\n      uniformLocations.uDelta,\n      this.horizontal ? [1 / this.width, 0] : [0, 1 / this.height]\n    );\n    gl.uniform1fv(uniformLocations.uTaps, this.taps);\n  }\n\n  getFilterWindow(this: ResizeDuringWEBGLResize) {\n    const scale = this.tempScale;\n    return Math.ceil(this.lanczosLobes / scale);\n  }\n\n  getCacheKey(this: ResizeDuringWEBGLResize): string {\n    const filterWindow = this.getFilterWindow();\n    return `${this.type}_${filterWindow}`;\n  }\n\n  getFragmentSource(this: ResizeDuringWEBGLResize): string {\n    const filterWindow = this.getFilterWindow();\n    return this.generateShader(filterWindow);\n  }\n\n  getTaps(this: ResizeDuringWEBGLResize) {\n    const lobeFunction = this.lanczosCreate(this.lanczosLobes),\n      scale = this.tempScale,\n      filterWindow = this.getFilterWindow(),\n      taps = new Array(filterWindow);\n    for (let i = 1; i <= filterWindow; i++) {\n      taps[i - 1] = lobeFunction(i * scale);\n    }\n    return taps;\n  }\n\n  /**\n   * Generate vertex and shader sources from the necessary steps numbers\n   * @param {Number} filterWindow\n   */\n  generateShader(filterWindow: number) {\n    const offsets = new Array(filterWindow);\n    for (let i = 1; i <= filterWindow; i++) {\n      offsets[i - 1] = `${i}.0 * uDelta`;\n    }\n    return `\n      ${this.fragmentSourceTOP}\n      uniform float uTaps[${filterWindow}];\n      void main() {\n        vec4 color = texture2D(uTexture, vTexCoord);\n        float sum = 1.0;\n        ${offsets\n          .map(\n            (offset, i) => `\n              color += texture2D(uTexture, vTexCoord + ${offset}) * uTaps[${i}] + texture2D(uTexture, vTexCoord - ${offset}) * uTaps[${i}];\n              sum += 2.0 * uTaps[${i}];\n            `\n          )\n          .join('\\n')}\n        gl_FragColor = color / sum;\n      }\n    `;\n  }\n\n  applyToForWebgl(this: ResizeDuringWEBGLResize, options: TWebGLPipelineState) {\n    options.passes++;\n    this.width = options.sourceWidth;\n    this.horizontal = true;\n    this.dW = Math.round(this.width * this.scaleX);\n    this.dH = options.sourceHeight;\n    this.tempScale = this.dW / this.width;\n    this.taps = this.getTaps();\n    options.destinationWidth = this.dW;\n    super.applyTo(options);\n    options.sourceWidth = options.destinationWidth;\n\n    this.height = options.sourceHeight;\n    this.horizontal = false;\n    this.dH = Math.round(this.height * this.scaleY);\n    this.tempScale = this.dH / this.height;\n    this.taps = this.getTaps();\n    options.destinationHeight = this.dH;\n    super.applyTo(options);\n    options.sourceHeight = options.destinationHeight;\n  }\n\n  /**\n   * Apply the resize filter to the image\n   * Determines whether to use WebGL or Canvas2D based on the options.webgl flag.\n   *\n   * @param {Object} options\n   * @param {Number} options.passes The number of filters remaining to be executed\n   * @param {Boolean} options.webgl Whether to use webgl to render the filter.\n   * @param {WebGLTexture} options.sourceTexture The texture setup as the source to be filtered.\n   * @param {WebGLTexture} options.targetTexture The texture where filtered output should be drawn.\n   * @param {WebGLRenderingContext} options.context The GL context used for rendering.\n   * @param {Object} options.programCache A map of compiled shader programs, keyed by filter type.\n   */\n  applyTo(options: TWebGLPipelineState | T2DPipelineState) {\n    if (isWebGLPipelineState(options)) {\n      (this as unknown as ResizeDuringWEBGLResize).applyToForWebgl(options);\n    } else {\n      (this as unknown as ResizeDuring2DResize).applyTo2d(options);\n    }\n  }\n\n  isNeutralState() {\n    return this.scaleX === 1 && this.scaleY === 1;\n  }\n\n  lanczosCreate(lobes: number) {\n    return (x: number) => {\n      if (x >= lobes || x <= -lobes) {\n        return 0.0;\n      }\n      if (x < 1.1920929e-7 && x > -1.1920929e-7) {\n        return 1.0;\n      }\n      x *= Math.PI;\n      const xx = x / lobes;\n      return ((Math.sin(x) / x) * Math.sin(xx)) / xx;\n    };\n  }\n\n  applyTo2d(this: ResizeDuring2DResize, options: T2DPipelineState) {\n    const imageData = options.imageData,\n      scaleX = this.scaleX,\n      scaleY = this.scaleY;\n\n    this.rcpScaleX = 1 / scaleX;\n    this.rcpScaleY = 1 / scaleY;\n\n    const oW = imageData.width;\n    const oH = imageData.height;\n    const dW = Math.round(oW * scaleX);\n    const dH = Math.round(oH * scaleY);\n    let newData: ImageData;\n\n    if (this.resizeType === 'sliceHack') {\n      newData = this.sliceByTwo(options, oW, oH, dW, dH);\n    } else if (this.resizeType === 'hermite') {\n      newData = this.hermiteFastResize(options, oW, oH, dW, dH);\n    } else if (this.resizeType === 'bilinear') {\n      newData = this.bilinearFiltering(options, oW, oH, dW, dH);\n    } else if (this.resizeType === 'lanczos') {\n      newData = this.lanczosResize(options, oW, oH, dW, dH);\n    } else {\n      // this should never trigger, is here just for safety net.\n      newData = new ImageData(dW, dH);\n    }\n    options.imageData = newData;\n  }\n\n  /**\n   * Filter sliceByTwo\n   * @param {Object} canvasEl Canvas element to apply filter to\n   * @param {Number} oW Original Width\n   * @param {Number} oH Original Height\n   * @param {Number} dW Destination Width\n   * @param {Number} dH Destination Height\n   * @returns {ImageData}\n   */\n  sliceByTwo(\n    options: T2DPipelineState,\n    oW: number,\n    oH: number,\n    dW: number,\n    dH: number\n  ) {\n    const imageData = options.imageData;\n    const mult = 0.5;\n    let doneW = false;\n    let doneH = false;\n    let stepW = oW * mult;\n    let stepH = oH * mult;\n    const resources = options.filterBackend.resources;\n    let sX = 0;\n    let sY = 0;\n    const dX = oW;\n    let dY = 0;\n    if (!resources.sliceByTwo) {\n      resources.sliceByTwo = createCanvasElement();\n    }\n    const tmpCanvas = resources.sliceByTwo;\n    if (tmpCanvas.width < oW * 1.5 || tmpCanvas.height < oH) {\n      tmpCanvas.width = oW * 1.5;\n      tmpCanvas.height = oH;\n    }\n    const ctx = tmpCanvas.getContext('2d')!;\n    ctx.clearRect(0, 0, oW * 1.5, oH);\n    ctx.putImageData(imageData, 0, 0);\n\n    dW = Math.floor(dW);\n    dH = Math.floor(dH);\n\n    while (!doneW || !doneH) {\n      oW = stepW;\n      oH = stepH;\n      if (dW < Math.floor(stepW * mult)) {\n        stepW = Math.floor(stepW * mult);\n      } else {\n        stepW = dW;\n        doneW = true;\n      }\n      if (dH < Math.floor(stepH * mult)) {\n        stepH = Math.floor(stepH * mult);\n      } else {\n        stepH = dH;\n        doneH = true;\n      }\n      ctx.drawImage(tmpCanvas, sX, sY, oW, oH, dX, dY, stepW, stepH);\n      sX = dX;\n      sY = dY;\n      dY += stepH;\n    }\n    return ctx.getImageData(sX, sY, dW, dH);\n  }\n\n  /**\n   * Filter lanczosResize\n   * @param {Object} canvasEl Canvas element to apply filter to\n   * @param {Number} oW Original Width\n   * @param {Number} oH Original Height\n   * @param {Number} dW Destination Width\n   * @param {Number} dH Destination Height\n   * @returns {ImageData}\n   */\n  lanczosResize(\n    this: ResizeDuring2DResize,\n    options: T2DPipelineState,\n    oW: number,\n    oH: number,\n    dW: number,\n    dH: number\n  ): ImageData {\n    function process(u: number): ImageData {\n      let v, i, weight, idx, a, red, green, blue, alpha, fX, fY;\n      center.x = (u + 0.5) * ratioX;\n      icenter.x = Math.floor(center.x);\n      for (v = 0; v < dH; v++) {\n        center.y = (v + 0.5) * ratioY;\n        icenter.y = Math.floor(center.y);\n        a = 0;\n        red = 0;\n        green = 0;\n        blue = 0;\n        alpha = 0;\n        for (i = icenter.x - range2X; i <= icenter.x + range2X; i++) {\n          if (i < 0 || i >= oW) {\n            continue;\n          }\n          fX = Math.floor(1000 * Math.abs(i - center.x));\n          if (!cacheLanc[fX]) {\n            cacheLanc[fX] = {};\n          }\n          for (let j = icenter.y - range2Y; j <= icenter.y + range2Y; j++) {\n            if (j < 0 || j >= oH) {\n              continue;\n            }\n            fY = Math.floor(1000 * Math.abs(j - center.y));\n            if (!cacheLanc[fX][fY]) {\n              cacheLanc[fX][fY] = lanczos(\n                Math.sqrt(\n                  Math.pow(fX * rcpRatioX, 2) + Math.pow(fY * rcpRatioY, 2)\n                ) / 1000\n              );\n            }\n            weight = cacheLanc[fX][fY];\n            if (weight > 0) {\n              idx = (j * oW + i) * 4;\n              a += weight;\n              red += weight * srcData[idx];\n              green += weight * srcData[idx + 1];\n              blue += weight * srcData[idx + 2];\n              alpha += weight * srcData[idx + 3];\n            }\n          }\n        }\n        idx = (v * dW + u) * 4;\n        destData[idx] = red / a;\n        destData[idx + 1] = green / a;\n        destData[idx + 2] = blue / a;\n        destData[idx + 3] = alpha / a;\n      }\n\n      if (++u < dW) {\n        return process(u);\n      } else {\n        return destImg;\n      }\n    }\n\n    const srcData = options.imageData.data,\n      destImg = options.ctx.createImageData(dW, dH),\n      destData = destImg.data,\n      lanczos = this.lanczosCreate(this.lanczosLobes),\n      ratioX = this.rcpScaleX,\n      ratioY = this.rcpScaleY,\n      rcpRatioX = 2 / this.rcpScaleX,\n      rcpRatioY = 2 / this.rcpScaleY,\n      range2X = Math.ceil((ratioX * this.lanczosLobes) / 2),\n      range2Y = Math.ceil((ratioY * this.lanczosLobes) / 2),\n      cacheLanc: Record<number, Record<number, number>> = {},\n      center: XY = { x: 0, y: 0 },\n      icenter: XY = { x: 0, y: 0 };\n\n    return process(0);\n  }\n\n  /**\n   * bilinearFiltering\n   * @param {Object} canvasEl Canvas element to apply filter to\n   * @param {Number} oW Original Width\n   * @param {Number} oH Original Height\n   * @param {Number} dW Destination Width\n   * @param {Number} dH Destination Height\n   * @returns {ImageData}\n   */\n  bilinearFiltering(\n    this: ResizeDuring2DResize,\n    options: T2DPipelineState,\n    oW: number,\n    oH: number,\n    dW: number,\n    dH: number\n  ) {\n    let a;\n    let b;\n    let c;\n    let d;\n    let x;\n    let y;\n    let i;\n    let j;\n    let xDiff;\n    let yDiff;\n    let chnl;\n    let color;\n    let offset = 0;\n    let origPix;\n    const ratioX = this.rcpScaleX;\n    const ratioY = this.rcpScaleY;\n    const w4 = 4 * (oW - 1);\n    const img = options.imageData;\n    const pixels = img.data;\n    const destImage = options.ctx.createImageData(dW, dH);\n    const destPixels = destImage.data;\n    for (i = 0; i < dH; i++) {\n      for (j = 0; j < dW; j++) {\n        x = Math.floor(ratioX * j);\n        y = Math.floor(ratioY * i);\n        xDiff = ratioX * j - x;\n        yDiff = ratioY * i - y;\n        origPix = 4 * (y * oW + x);\n\n        for (chnl = 0; chnl < 4; chnl++) {\n          a = pixels[origPix + chnl];\n          b = pixels[origPix + 4 + chnl];\n          c = pixels[origPix + w4 + chnl];\n          d = pixels[origPix + w4 + 4 + chnl];\n          color =\n            a * (1 - xDiff) * (1 - yDiff) +\n            b * xDiff * (1 - yDiff) +\n            c * yDiff * (1 - xDiff) +\n            d * xDiff * yDiff;\n          destPixels[offset++] = color;\n        }\n      }\n    }\n    return destImage;\n  }\n\n  /**\n   * hermiteFastResize\n   * @param {Object} canvasEl Canvas element to apply filter to\n   * @param {Number} oW Original Width\n   * @param {Number} oH Original Height\n   * @param {Number} dW Destination Width\n   * @param {Number} dH Destination Height\n   * @returns {ImageData}\n   */\n  hermiteFastResize(\n    this: ResizeDuring2DResize,\n    options: T2DPipelineState,\n    oW: number,\n    oH: number,\n    dW: number,\n    dH: number\n  ) {\n    const ratioW = this.rcpScaleX,\n      ratioH = this.rcpScaleY,\n      ratioWHalf = Math.ceil(ratioW / 2),\n      ratioHHalf = Math.ceil(ratioH / 2),\n      img = options.imageData,\n      data = img.data,\n      img2 = options.ctx.createImageData(dW, dH),\n      data2 = img2.data;\n    for (let j = 0; j < dH; j++) {\n      for (let i = 0; i < dW; i++) {\n        const x2 = (i + j * dW) * 4;\n        let weight = 0;\n        let weights = 0;\n        let weightsAlpha = 0;\n        let gxR = 0;\n        let gxG = 0;\n        let gxB = 0;\n        let gxA = 0;\n        const centerY = (j + 0.5) * ratioH;\n        for (let yy = Math.floor(j * ratioH); yy < (j + 1) * ratioH; yy++) {\n          const dy = Math.abs(centerY - (yy + 0.5)) / ratioHHalf,\n            centerX = (i + 0.5) * ratioW,\n            w0 = dy * dy;\n          for (let xx = Math.floor(i * ratioW); xx < (i + 1) * ratioW; xx++) {\n            let dx = Math.abs(centerX - (xx + 0.5)) / ratioWHalf;\n            const w = Math.sqrt(w0 + dx * dx);\n            /* eslint-disable max-depth */\n            if (w > 1 && w < -1) {\n              continue;\n            }\n            //hermite filter\n            weight = 2 * w * w * w - 3 * w * w + 1;\n            if (weight > 0) {\n              dx = 4 * (xx + yy * oW);\n              //alpha\n              gxA += weight * data[dx + 3];\n              weightsAlpha += weight;\n              //colors\n              if (data[dx + 3] < 255) {\n                weight = (weight * data[dx + 3]) / 250;\n              }\n              gxR += weight * data[dx];\n              gxG += weight * data[dx + 1];\n              gxB += weight * data[dx + 2];\n              weights += weight;\n            }\n            /* eslint-enable max-depth */\n          }\n        }\n        data2[x2] = gxR / weights;\n        data2[x2 + 1] = gxG / weights;\n        data2[x2 + 2] = gxB / weights;\n        data2[x2 + 3] = gxA / weightsAlpha;\n      }\n    }\n    return img2;\n  }\n\n  /**\n   * Returns object representation of an instance\n   * @return {Object} Object representation of an instance\n   */\n  toObject() {\n    return {\n      type: this.type,\n      scaleX: this.scaleX,\n      scaleY: this.scaleY,\n      resizeType: this.resizeType,\n      lanczosLobes: this.lanczosLobes,\n    };\n  }\n}\n\nclassRegistry.setClass(Resize);\n","import type { TClassProperties } from '../typedefs';\nimport { BaseFilter } from './BaseFilter';\nimport type { T2DPipelineState, TWebGLUniformLocationMap } from './typedefs';\nimport { classRegistry } from '../ClassRegistry';\nimport { fragmentSource } from './shaders/saturation';\n\n/**\n * Saturate filter class\n * @example\n * const filter = new Saturation({\n *   saturation: 1\n * });\n * object.filters.push(filter);\n * object.applyFilters();\n */\n\nexport const saturationDefaultValues: Partial<TClassProperties<Saturation>> = {\n  saturation: 0,\n  mainParameter: 'saturation',\n};\n\nexport class Saturation extends BaseFilter {\n  /**\n   * Saturation value, from -1 to 1.\n   * Increases/decreases the color saturation.\n   * A value of 0 has no effect.\n   *\n   * @param {Number} saturation\n   * @default\n   */\n  declare saturation: number;\n\n  static type = 'Saturation';\n\n  static defaults = saturationDefaultValues;\n\n  getFragmentSource() {\n    return fragmentSource;\n  }\n\n  /**\n   * Apply the Saturation operation to a Uint8ClampedArray representing the pixels of an image.\n   *\n   * @param {Object} options\n   * @param {ImageData} options.imageData The Uint8ClampedArray to be filtered.\n   */\n  applyTo2d({ imageData: { data } }: T2DPipelineState) {\n    if (this.saturation === 0) {\n      return;\n    }\n    const adjust = -this.saturation;\n    for (let i = 0; i < data.length; i += 4) {\n      const max = Math.max(data[i], data[i + 1], data[i + 2]);\n      data[i] += max !== data[i] ? (max - data[i]) * adjust : 0;\n      data[i + 1] += max !== data[i + 1] ? (max - data[i + 1]) * adjust : 0;\n      data[i + 2] += max !== data[i + 2] ? (max - data[i + 2]) * adjust : 0;\n    }\n  }\n\n  /**\n   * Return WebGL uniform locations for this filter's shader.\n   *\n   * @param {WebGLRenderingContext} gl The GL canvas context used to compile this filter's shader.\n   * @param {WebGLShaderProgram} program This filter's compiled shader program.\n   */\n  getUniformLocations(\n    gl: WebGLRenderingContext,\n    program: WebGLProgram\n  ): TWebGLUniformLocationMap {\n    return {\n      uSaturation: gl.getUniformLocation(program, 'uSaturation'),\n    };\n  }\n\n  /**\n   * Send data from this filter to its shader program's uniforms.\n   *\n   * @param {WebGLRenderingContext} gl The GL canvas context used to compile this filter's shader.\n   * @param {Object} uniformLocations A map of string uniform names to WebGLUniformLocation objects\n   */\n  sendUniformData(\n    gl: WebGLRenderingContext,\n    uniformLocations: TWebGLUniformLocationMap\n  ) {\n    gl.uniform1f(uniformLocations.uSaturation, -this.saturation);\n  }\n}\n\nclassRegistry.setClass(Saturation);\n","export const fragmentSource = `\n  precision highp float;\n  uniform sampler2D uTexture;\n  uniform float uSaturation;\n  varying vec2 vTexCoord;\n  void main() {\n    vec4 color = texture2D(uTexture, vTexCoord);\n    float rgMax = max(color.r, color.g);\n    float rgbMax = max(rgMax, color.b);\n    color.r += rgbMax != color.r ? (rgbMax - color.r) * uSaturation : 0.00;\n    color.g += rgbMax != color.g ? (rgbMax - color.g) * uSaturation : 0.00;\n    color.b += rgbMax != color.b ? (rgbMax - color.b) * uSaturation : 0.00;\n    gl_FragColor = color;\n  }\n`;\n","import type { TClassProperties } from '../typedefs';\nimport { BaseFilter } from './BaseFilter';\nimport type { T2DPipelineState, TWebGLUniformLocationMap } from './typedefs';\nimport { classRegistry } from '../ClassRegistry';\nimport { fragmentSource } from './shaders/vibrance';\n\nexport const vibranceDefaultValues: Partial<TClassProperties<Vibrance>> = {\n  vibrance: 0,\n  mainParameter: 'vibrance',\n};\n\n/**\n * Vibrance filter class\n * @example\n * const filter = new Vibrance({\n *   vibrance: 1\n * });\n * object.filters.push(filter);\n * object.applyFilters();\n */\nexport class Vibrance extends BaseFilter {\n  /**\n   * Vibrance value, from -1 to 1.\n   * Increases/decreases the saturation of more muted colors with less effect on saturated colors.\n   * A value of 0 has no effect.\n   *\n   * @param {Number} vibrance\n   * @default\n   */\n  declare vibrance: number;\n\n  static type = 'Vibrance';\n\n  static defaults = vibranceDefaultValues;\n\n  getFragmentSource() {\n    return fragmentSource;\n  }\n\n  /**\n   * Apply the Vibrance operation to a Uint8ClampedArray representing the pixels of an image.\n   *\n   * @param {Object} options\n   * @param {ImageData} options.imageData The Uint8ClampedArray to be filtered.\n   */\n  applyTo2d({ imageData: { data } }: T2DPipelineState) {\n    if (this.vibrance === 0) {\n      return;\n    }\n    const adjust = -this.vibrance;\n    for (let i = 0; i < data.length; i += 4) {\n      const max = Math.max(data[i], data[i + 1], data[i + 2]);\n      const avg = (data[i] + data[i + 1] + data[i + 2]) / 3;\n      const amt = ((Math.abs(max - avg) * 2) / 255) * adjust;\n      data[i] += max !== data[i] ? (max - data[i]) * amt : 0;\n      data[i + 1] += max !== data[i + 1] ? (max - data[i + 1]) * amt : 0;\n      data[i + 2] += max !== data[i + 2] ? (max - data[i + 2]) * amt : 0;\n    }\n  }\n\n  /**\n   * Return WebGL uniform locations for this filter's shader.\n   *\n   * @param {WebGLRenderingContext} gl The GL canvas context used to compile this filter's shader.\n   * @param {WebGLShaderProgram} program This filter's compiled shader program.\n   */\n  getUniformLocations(\n    gl: WebGLRenderingContext,\n    program: WebGLProgram\n  ): TWebGLUniformLocationMap {\n    return {\n      uVibrance: gl.getUniformLocation(program, 'uVibrance'),\n    };\n  }\n\n  /**\n   * Send data from this filter to its shader program's uniforms.\n   *\n   * @param {WebGLRenderingContext} gl The GL canvas context used to compile this filter's shader.\n   * @param {Object} uniformLocations A map of string uniform names to WebGLUniformLocation objects\n   */\n  sendUniformData(\n    gl: WebGLRenderingContext,\n    uniformLocations: TWebGLUniformLocationMap\n  ) {\n    gl.uniform1f(uniformLocations.uVibrance, -this.vibrance);\n  }\n}\n\nclassRegistry.setClass(Vibrance);\n","export const fragmentSource = `\n  precision highp float;\n  uniform sampler2D uTexture;\n  uniform float uVibrance;\n  varying vec2 vTexCoord;\n  void main() {\n    vec4 color = texture2D(uTexture, vTexCoord);\n    float max = max(color.r, max(color.g, color.b));\n    float avg = (color.r + color.g + color.b) / 3.0;\n    float amt = (abs(max - avg) * 2.0) * uVibrance;\n    color.r += max != color.r ? (max - color.r) * amt : 0.00;\n    color.g += max != color.g ? (max - color.g) * amt : 0.00;\n    color.b += max != color.b ? (max - color.b) * amt : 0.00;\n    gl_FragColor = color;\n  }\n`;\n","//@ts-nocheck\nimport { changeWidth } from './changeWidth';\nimport { Control } from './Control';\nimport { scaleCursorStyleHandler, scalingEqually } from './scale';\nimport {\n  scaleOrSkewActionName,\n  scaleSkewCursorStyleHandler,\n  scalingYOrSkewingX,\n} from './scaleSkew';\nimport { renderCircleControl } from './controlRendering';\n\n// use this function if you want to generate new controls for every instance\nexport const createObjectDefaultControls = () => ({\n  tl: new Control({\n    x: -0.5,\n    y: -0.5,\n    cursorStyleHandler: scaleCursorStyleHandler,\n    actionHandler: scalingEqually,\n  }),\n\n  tr: new Control({\n    x: 0.5,\n    y: -0.5,\n    cursorStyleHandler: scaleCursorStyleHandler,\n    actionHandler: scalingEqually,\n  }),\n\n  bl: new Control({\n    x: -0.5,\n    y: 0.5,\n    cursorStyleHandler: scaleCursorStyleHandler,\n    actionHandler: scalingEqually,\n  }),\n\n  br: new Control({\n    x: 0.5,\n    y: 0.5,\n    cursorStyleHandler: scaleCursorStyleHandler,\n    actionHandler: scalingEqually,\n  }),\n\n  // mtr: new Control({\n  //   x: 0,\n  //   y: -0.5,\n  //   actionHandler: rotationWithSnapping,\n  //   cursorStyleHandler: rotationStyleHandler,\n  //   offsetY: -40,\n  //   withConnection: true,\n  //   actionName: 'rotate',\n  // }),\n});\n\nexport const createObjectDefaultNoRotateControls = () => ({\n  tl: new Control({\n    x: -0.5,\n    y: -0.5,\n    cursorStyleHandler: scaleCursorStyleHandler,\n    actionHandler: scalingEqually,\n  }),\n\n  tr: new Control({\n    x: 0.5,\n    y: -0.5,\n    cursorStyleHandler: scaleCursorStyleHandler,\n    actionHandler: scalingEqually,\n  }),\n\n  bl: new Control({\n    x: -0.5,\n    y: 0.5,\n    cursorStyleHandler: scaleCursorStyleHandler,\n    actionHandler: scalingEqually,\n  }),\n\n  br: new Control({\n    x: 0.5,\n    y: 0.5,\n    cursorStyleHandler: scaleCursorStyleHandler,\n    actionHandler: scalingEqually,\n  }),\n});\n\nexport const createObjectImageControls = () => ({\n  tl: new Control({\n    x: -0.5,\n    y: -0.5,\n    cursorStyleHandler: scaleCursorStyleHandler,\n    actionHandler: scalingEqually,\n  }),\n\n  tr: new Control({\n    x: 0.5,\n    y: -0.5,\n    cursorStyleHandler: scaleCursorStyleHandler,\n    actionHandler: scalingEqually,\n  }),\n\n  bl: new Control({\n    x: -0.5,\n    y: 0.5,\n    cursorStyleHandler: scaleCursorStyleHandler,\n    actionHandler: scalingEqually,\n  }),\n\n  br: new Control({\n    x: 0.5,\n    y: 0.5,\n    cursorStyleHandler: scaleCursorStyleHandler,\n    actionHandler: scalingEqually,\n  }),\n});\nexport const createObjectFileControls = () => ({\n  tl: new Control({\n    x: -0.5,\n    y: -0.5,\n    cursorStyleHandler: scaleCursorStyleHandler,\n    actionHandler: scalingEqually,\n  }),\n  br: new Control({\n    x: 0.5,\n    y: 0.5,\n    cursorStyleHandler: scaleCursorStyleHandler,\n    actionHandler: scalingEqually,\n  }),\n});\nexport const createObjectArrowControls = () => ({\n  tl: new Control({\n    x: -0.5,\n    y: -0.5,\n    cursorStyle: 'crosshair',\n    cursorStyleHandler: scaleCursorStyleHandler,\n    actionHandler: scalingEqually,\n  }),\n  br: new Control({\n    x: 0.5,\n    y: 0.5,\n    cursorStyleHandler: scaleCursorStyleHandler,\n    actionHandler: scalingEqually,\n  }),\n});\n\nexport const createObjectConnectorControls = function (targetObject: any) {\n  function renderCustomControl(control, ctx, left, top, fabricObject) {\n    let cornerSize, cornerColor;\n\n    // if (control.targetObject.hoveringControl === control.controlName) {\n    //   cornerSize = 15;\n    //   cornerColor = 'red';\n    // } else {\n    cornerSize = 10;\n    cornerColor = 'white';\n    // }\n\n    const styleOverride1 = {\n      cornerSize: cornerSize,\n      cornerColor,\n      lineWidth: 2,\n    };\n\n    renderCircleControl.call(\n      fabricObject,\n      ctx,\n      left,\n      top,\n      styleOverride1,\n      fabricObject\n    );\n  }\n  return {\n    mtaStart: new Control({\n      x: 0,\n      y: -0.5,\n      offsetX: 0,\n      offsetY: -20,\n      render: renderCustomControl.bind(this, {\n        controlName: 'mtaStart',\n        targetObject,\n      }),\n      mouseDownHandler: (eventData, transformData) => {\n        // this.controlMousedownProcess(transformData, 0.0, -0.5);\n\n        return true;\n      },\n      name: 'mtaStart',\n    }),\n    mbaStart: new Control({\n      x: 0,\n      y: 0.5,\n      offsetX: 0,\n      offsetY: 20,\n      render: renderCustomControl.bind(this, {\n        controlName: 'mbaStart',\n        targetObject,\n      }),\n      mouseDownHandler: (eventData, transformData) => {\n        // this.controlMousedownProcess(transformData, 0.0, 0.5);\n        return true;\n      },\n      name: 'mbaStart',\n    }),\n    mlaStart: new Control({\n      x: -0.5,\n      y: 0,\n      offsetX: -20,\n      offsetY: 0,\n      render: renderCustomControl.bind(this, {\n        controlName: 'mlaStart',\n        targetObject,\n      }),\n      mouseDownHandler: (eventData, transformData) => {\n        // this.controlMousedownProcess(transformData, -0.5, 0.0);\n        return true;\n      },\n      name: 'mlaStart',\n    }),\n    mraStart: new Control({\n      x: 0.5,\n      y: 0,\n      offsetX: 20,\n      offsetY: 0,\n      render: renderCustomControl.bind(this, {\n        controlName: 'mraStart',\n        targetObject,\n      }),\n      mouseDownHandler: (eventData, transformData) => {\n        // this.controlMousedownProcess(transformData, 0.5, 0.0);\n        return true;\n      },\n      name: 'mraStart',\n    }),\n  };\n};\n\nexport const createResizeControls = () => ({\n  mr: new Control({\n    x: 0.5,\n    y: 0,\n    offsetX: 20,\n    offsetY: 0,\n    actionHandler: changeWidth,\n    cursorStyleHandler: scaleSkewCursorStyleHandler,\n    actionName: 'resizing',\n  }),\n  ml: new Control({\n    x: -0.5,\n    y: 0,\n    offsetX: -20,\n    offsetY: 0,\n    actionHandler: changeWidth,\n    cursorStyleHandler: scaleSkewCursorStyleHandler,\n    actionName: 'resizing',\n  }),\n  mb: new Control({\n    x: 0,\n    y: 0.5,\n    offsetX: 0,\n    offsetY: 20,\n    cursorStyleHandler: scaleSkewCursorStyleHandler,\n    actionHandler: scalingYOrSkewingX,\n    getActionName: scaleOrSkewActionName,\n  }),\n\n  mt: new Control({\n    x: 0,\n    y: -0.5,\n    offsetX: 0,\n    offsetY: -20,\n    cursorStyleHandler: scaleSkewCursorStyleHandler,\n    actionHandler: scalingYOrSkewingX,\n    getActionName: scaleOrSkewActionName,\n  }),\n});\n\nexport const createResizeControlsForText = () => ({\n  mr: new Control({\n    x: 0.5,\n    y: 0,\n    actionHandler: changeWidth,\n    cursorStyleHandler: scaleSkewCursorStyleHandler,\n    actionName: 'resizing',\n  }),\n  ml: new Control({\n    x: -0.5,\n    y: 0,\n    actionHandler: changeWidth,\n    cursorStyleHandler: scaleSkewCursorStyleHandler,\n    actionName: 'resizing',\n  }),\n});\n\nexport const createTextboxDefaultControls = () => ({\n  //...createObjectDefaultControls(),\n  ...createResizeControlsForText(),\n});\nexport const createRectNotesDefaultControls = (targetObject: any) => ({\n  ...createObjectDefaultNoRotateControls(),\n  ...createObjectConnectorControls(targetObject),\n});\nexport const createShapeNotesDefaultControls = (targetObject: any) => ({\n  ...createObjectDefaultNoRotateControls(),\n  // ...createResizeControls(),\n  ...createObjectConnectorControls(targetObject),\n});\nexport const createPathDefaultControls = () => ({\n  ...createObjectDefaultNoRotateControls(),\n});\nexport const createImageDefaultControls = () => ({\n  ...createObjectImageControls(),\n});\nexport const createFileDefaultControls = () => ({\n  ...createObjectFileControls(),\n});\n","export enum EventName {\n  MODIFIED = 'modified',\n  CHANGED = 'changed',\n  TRIPLECLICK = 'tripleclick',\n  DBLCLICK = 'dblclick',\n  MOUSEOUT = 'mouseout',\n  EDITINGENTERED = 'editing:entered',\n  MOUSEOVER = 'mouseover',\n  EDITINGEXITED = 'editing:exited',\n  INITIALIZE = 'initialize',\n  INSERTNEWSTYLEBLOCK = 'insertNewStyleBlock',\n  RENDER = '_render',\n  SET = 'set',\n  OBJECTMOVING = 'object:moving',\n  OBJECTMOVED = 'object:moved',\n  SELECTIONCREATED = 'selection:created',\n  SELECTIONCLEARED = 'selection:cleared',\n  SELECTED = 'selected',\n  DESELECTED = 'deselected',\n  MOUSEDBLCLICK = 'mousedblclick',\n  MOUSEUP = 'mouseup',\n  MOUSE_UP = 'mouse:up',\n  TOUCHSTART = 'touchstart',\n  SETSRC = 'setSrc',\n  REMOVED = 'removed',\n  MOUSEDOWN = 'mousedown',\n  MOVING = 'moving',\n  SELECTABLE = 'selectable',\n  EVENTED = 'evented',\n  FILL = 'fill',\n  ROTATING = 'rotating',\n  SCALING = 'scaling',\n  ADDED = 'added',\n  OBJECTSCALING = 'object:scaling',\n  OBJECTSCALED = 'object:scaled',\n  OBJECTROTATING = 'object:rotating',\n  OBJECTROTATED = 'object:rotated',\n  SELECTIONUPDATED = 'selection:updated',\n  DRAGENTER = 'dragenter',\n  DRAGLEAVE = 'dragleave',\n\n  CANVAS_MOUSE_DOWN = 'canvas:mousedown',\n  CANVAS_MOUSE_DOWN_BEFORE = 'canvas:mousedownbefore',\n  CANVAS_MOUSE_MOVE = 'canvas:mousemove',\n  CANVAS_MOUSE_UP = 'canvas:mouseup',\n  CANVAS_MOUSE_OUT = 'canvas:mouseout',\n  CANVAS_SCROLL = 'canvas:scroll',\n  DOCUMENT_KEY_UP = 'document:keyup',\n  DOCUMENT_KEY_DOWN = 'document:keydown',\n  DOCUMENT_MOUSE_DOWN = 'mousedown',\n  DOCUMENT_MOUSE_UP = 'mouseup',\n  DOCUMENT_MOUSE_MOVE = 'mousemove',\n  DOCUMENT_MOUSE_WHEEL = 'wheel',\n  DOCUMENT_DRAG_OVER = 'dragover',\n  DOCUMENT_PAUSE = 'pause',\n  DOCUMENT_RESUME = 'resume',\n  DOCUMENT_VISIBILITY_CHANGE = 'visibilitychange',\n  DOCUMENT_PASTE = 'paste',\n  WINDOW_BEFORE_UNLOAD = 'beforeunload',\n  WINDOW_MOUSE_MOVE = 'mousemove',\n  WINDOW_GESTURE_START = 'gesturestart',\n  WINDOW_GESTURE_CHANGE = 'gesturechange',\n  A_CTRL_KEY_DOWN = 'a_ctrl:keydown',\n  B_CTRL_KEY_DOWN = 'b_ctrl:keydown',\n  D_CTRL_KEY_DOWN = 'd_ctrl:keydown',\n  G_CTRL_SHIFT_KEY_DOWN = 'g_ctrl_shift:keydown',\n  G_CTRL_KEY_DOWN = 'g_ctrl:keydown',\n  T_KEY_DOWN = 't:keydown',\n  P_KEY_DOWN = 'p:keydown',\n  S_KEY_UP = 's:keyup',\n  S_KEY_DOWN = 's:keydown',\n  O_KEY_UP = 'o:keyup',\n  O_KEY_DOWN = 'o:keydown',\n  R_KEY_UP = 'r:keyup',\n  R_KEY_DOWN = 'r:keydown',\n  F_KEY_UP = 'f:keyup',\n  F_KEY_DOWN = 'f:keydown',\n  L_KEY_UP = 'l:keyup',\n  L_KEY_DOWN = 'l:keydown',\n\n  Z_CTRL_KEY_UP = 'z_ctrl:keyup',\n\n  Z_CTRL_KEY_DOWN = 'z_ctrl:keydown',\n\n  Z_CTRL_SHIFT_KEY_UP = 'z_ctrl_shift:keyup',\n\n  Z_CTRL_SHIFT_KEY_DOWN = 'z_ctrl_shift:keydown',\n\n  SPACE_KEY_UP = 'space:keyup',\n\n  SPACE_KEY_DOWN = 'space:keydown',\n\n  SHIFT_KEY_UP = 'shift:keyup',\n\n  SHIFT_KEY_DOWN = 'shift:keydown',\n\n  SLASH_SHIFT_KEY_DOWN = 'slash_shift:keydown',\n\n  ESC_KEY_DOWN = 'esc:keydown',\n\n  DELETE_KEY_DOWN = 'delete:keydown',\n\n  CTRL_SHIFT_FIVE_KEY_DOWN = 'ctrl_shift_five:keydown',\n\n  CTRL_PLUS_KEY_DOWN = 'ctrl_plus:keydown',\n\n  CTRL_MINUS_KEY_DOWN = 'ctrl_minus:keydown',\n\n  CTRL_ZERO_KEY_DOWN = 'ctrl_zero:keydown',\n\n  UP_DOWN_LEFT_RIGHT_KEY_DOWN = 'up_down_left_right:keydown',\n\n  CANVAS_BEFORE_SELECTION_CLEARED = 'before:selection:cleared',\n\n  CANVAS_SELECTION_UPDATED = 'selection:updated',\n\n  CANVAS_SELECTION_CREATED = 'selection:created',\n\n  SELECTION_CLEARED = 'selection:cleared',\n\n  TEXT_CHANGED = 'text:changed',\n\n  BEFORE_RENDER = 'before:render',\n\n  AFTER_RENDER = 'after:render',\n\n  MOUSE_DOWN_BEFORE = 'mouse:down:before',\n  MOUSE_DOWN = 'mouse:down',\n\n  TEXT_EDITING_EXISTED = 'text:editing:exited',\n\n  MOUSE_DBCLICK = 'mouse:dblclick',\n\n  HAMMER_TAP1 = 'tap1',\n\n  HAMMER_DOUBLE_TAP = 'doubletap',\n\n  HAMMER_PAN1_START = 'pan1start',\n\n  HAMMER_PAN1_MOVE = 'pan1move',\n\n  HAMMER_PAN1_END = 'pan1end',\n\n  HAMMER_PAN3 = 'pan3',\n\n  HAMMER_PINCH_START = 'pinchstart',\n\n  HAMMER_PINCH_MOVE = 'pinchmove',\n\n  RETURN_DEFAULT_ZOOMM = 'returndefaultzoom',\n\n  MOUSE_MOVE = 'mouse:move',\n  RESIZING = 'resizing',\n}\n\nexport enum TextAlign {\n  LEFT = 'left',\n  CENTER = 'center',\n  RIGHT = 'right',\n  JUSTIFY = 'justify',\n  JUSTIFYLEFT = 'justify-left',\n  JUSTIFYCENTER = 'justify-center',\n  JUSTIFYRIGHT = 'justify-right',\n}\nexport enum WidgetType {\n  XActiveSelection = 'activeSelection',\n  XConnector = 'XConnector',\n  XCircleNotes = 'XCircleNotes',\n  XFile = 'XFile',\n  XGroup = 'XGroup',\n  XImage = 'XImage',\n  XIText = 'XIText',\n  XObject = 'XObject',\n  XPath = 'XPath',\n  XPolyline = 'XPolyline',\n  XLine = 'XLine',\n  XRdRectPanel = 'XRdRectPanel',\n  XRect = 'XRect',\n  XRectNotes = 'XRectNotes',\n  XRectPanel = 'XRectPanel',\n  XShape = 'XShape',\n  XShapeNotes = 'XShapeNotes',\n  XTriangle = 'XTriangle',\n  XURL = 'XURL',\n  XText = 'XText',\n  XTextbox = 'XTextbox',\n  Common = 'common',\n}\n\nexport enum NoteType {\n  RECT = 'rect',\n  SQUARE = 'square',\n  CIRCLE = 'circle',\n  TEXT = 'text',\n}\n\nexport enum Origin {\n  Left = 'left',\n  Right = 'right',\n  Center = 'center',\n}\nexport enum ModifiedField {\n  Width = 'width',\n  Height = 'height',\n  Left = 'left',\n  Top = 'top',\n  ShapeScalex = 'shapeScaleX',\n  ScaleX = 'scaleX',\n  ScaleY = 'scaleY',\n  FlipX = 'flipX',\n  FLipY = 'flipY',\n  MaxHeight = 'maxHeight',\n  FixedScaleChange = 'fixedScaleChange',\n  Lines = 'lines',\n  PanelObj = 'panelObj',\n  Relationship = 'relationship',\n  ZIndex = 'zIndex',\n  x1 = 'x1',\n  x2 = 'x2',\n  y1 = 'y1',\n  y2 = 'y2',\n  ConnectorStart = 'connectorStart',\n  ConnectorEnd = 'connectorEnd',\n  Fill = 'fill',\n  Text = 'text',\n}\nexport enum SaveAction {\n  MOVED = 'MOVED',\n  SCALED = 'SCALED',\n  ADDED = 'ADDED',\n  PASTE = 'PASTE',\n  REMOVED = 'REMOVED',\n  MODIFIED = 'MODIFIED',\n  UNBIND = 'UNBIND',\n  BIND = 'BIND',\n  ROTATED = 'ROTATED',\n}\n","import type { TClassProperties, TOptions } from '../typedefs';\nimport { IText } from './IText/IText';\nimport { classRegistry } from '../ClassRegistry';\nimport { createTextboxDefaultControls } from '../controls/commonControls';\nimport { JUSTIFY } from './Text/constants';\nimport type { TextStyleDeclaration } from './Text/StyledText';\nimport type { SerializedITextProps, ITextProps } from './IText/IText';\nimport type { ITextEvents } from './IText/ITextBehavior';\nimport type { TextLinesInfo } from './Text/Text';\n\n// @TODO: Many things here are configuration related and shouldn't be on the class nor prototype\n// regexes, list of properties that are not suppose to change by instances, magic consts.\n// this will be a separated effort\nexport const textboxDefaultValues: Partial<TClassProperties<Textbox>> = {\n  minWidth: 20,\n  dynamicMinWidth: 2,\n  lockScalingFlip: true,\n  noScaleCache: false,\n  _wordJoiners: /[ \\t\\r]/,\n  splitByGrapheme: false,\n};\n\nexport type GraphemeData = {\n  wordsData: {\n    word: string[];\n    width: number;\n  }[][];\n  largestWordWidth: number;\n};\n\nexport type StyleMap = Record<string, { line: number; offset: number }>;\n\n// @TODO this is not complete\ninterface UniqueTextboxProps {\n  minWidth: number;\n  splitByGrapheme: boolean;\n  dynamicMinWidth: number;\n  _wordJoiners: RegExp;\n}\n\nexport interface SerializedTextboxProps\n  extends SerializedITextProps,\n  Pick<UniqueTextboxProps, 'minWidth' | 'splitByGrapheme'> { }\n\nexport interface TextboxProps extends ITextProps, UniqueTextboxProps { }\n\n/**\n * Textbox class, based on IText, allows the user to resize the text rectangle\n * and wraps lines automatically. Textboxes have their Y scaling locked, the\n * user can only change width. Height is adjusted automatically based on the\n * wrapping of lines.\n */\nexport class Textbox<\n  Props extends TOptions<TextboxProps> = Partial<TextboxProps>,\n  SProps extends SerializedTextboxProps = SerializedTextboxProps,\n  EventSpec extends ITextEvents = ITextEvents\n>\n  extends IText<Props, SProps, EventSpec>\n  implements UniqueTextboxProps {\n  /**\n   * Minimum width of textbox, in pixels.\n   * @type Number\n   * @default\n   */\n  declare minWidth: number;\n\n  /**\n   * Minimum calculated width of a textbox, in pixels.\n   * fixed to 2 so that an empty textbox cannot go to 0\n   * and is still selectable without text.\n   * @type Number\n   * @default\n   */\n  declare dynamicMinWidth: number;\n\n  /**\n   * Use this boolean property in order to split strings that have no white space concept.\n   * this is a cheap way to help with chinese/japanese\n   * @type Boolean\n   * @since 2.6.0\n   */\n  declare splitByGrapheme: boolean;\n\n  declare _wordJoiners: RegExp;\n\n  declare _styleMap: StyleMap;\n\n  declare isWrapping: boolean;\n\n  static type = 'Textbox';\n\n  static textLayoutProperties = [...IText.textLayoutProperties, 'width'];\n\n  static ownDefaults = textboxDefaultValues;\n\n  static getDefaults(): Record<string, any> {\n    return {\n      ...super.getDefaults(),\n      controls: createTextboxDefaultControls(),\n      ...Textbox.ownDefaults,\n    };\n  }\n\n  /**\n   * Unlike superclass's version of this function, Textbox does not update\n   * its width.\n   * @private\n   * @override\n   */\n  initDimensions() {\n    if (!this.initialized) {\n      return;\n    }\n    this.isEditing && this.initDelayedCursor();\n    this._clearCache();\n    // clear dynamicMinWidth as it will be different after we re-wrap line\n    this.dynamicMinWidth = 0;\n    // wrap lines\n    this._styleMap = this._generateStyleMap(this._splitText());\n    // if after wrapping, the width is smaller than dynamicMinWidth, change the width and re-wrap\n    if (this.dynamicMinWidth > this.width) {\n      this._set('width', this.dynamicMinWidth);\n    }\n    if (this.textAlign.includes(JUSTIFY)) {\n      // once text is measured we need to make space fatter to make justified text.\n      this.enlargeSpaces();\n    }\n    // clear cache and re-calculate height\n    this.height = this.calcTextHeight();\n  }\n\n  /**\n   * Generate an object that translates the style object so that it is\n   * broken up by visual lines (new lines and automatic wrapping).\n   * The original text styles object is broken up by actual lines (new lines only),\n   * which is only sufficient for Text / IText\n   * @private\n   */\n  _generateStyleMap(textInfo: TextLinesInfo): StyleMap {\n    let realLineCount = 0,\n      realLineCharCount = 0,\n      charCount = 0;\n    const map: StyleMap = {};\n\n    for (let i = 0; i < textInfo.graphemeLines.length; i++) {\n      if (textInfo.graphemeText[charCount] === '\\n' && i > 0) {\n        realLineCharCount = 0;\n        charCount++;\n        realLineCount++;\n      } else if (\n        !this.splitByGrapheme &&\n        this._reSpaceAndTab.test(textInfo.graphemeText[charCount]) &&\n        i > 0\n      ) {\n        // this case deals with space's that are removed from end of lines when wrapping\n        realLineCharCount++;\n        charCount++;\n      }\n\n      map[i] = { line: realLineCount, offset: realLineCharCount };\n\n      charCount += textInfo.graphemeLines[i].length;\n      realLineCharCount += textInfo.graphemeLines[i].length;\n    }\n\n    return map;\n  }\n\n  /**\n   * Returns true if object has a style property or has it on a specified line\n   * @param {Number} lineIndex\n   * @return {Boolean}\n   */\n  styleHas(property: keyof TextStyleDeclaration, lineIndex: number): boolean {\n    if (this._styleMap && !this.isWrapping) {\n      const map = this._styleMap[lineIndex];\n      if (map) {\n        lineIndex = map.line;\n      }\n    }\n    return super.styleHas(property, lineIndex);\n  }\n\n  /**\n   * Returns true if object has no styling or no styling in a line\n   * @param {Number} lineIndex , lineIndex is on wrapped lines.\n   * @return {Boolean}\n   */\n  isEmptyStyles(lineIndex: number): boolean {\n    if (!this.styles) {\n      return true;\n    }\n    let offset = 0,\n      nextLineIndex = lineIndex + 1,\n      nextOffset: number,\n      shouldLimit = false;\n    const map = this._styleMap[lineIndex],\n      mapNextLine = this._styleMap[lineIndex + 1];\n    if (map) {\n      lineIndex = map.line;\n      offset = map.offset;\n    }\n    if (mapNextLine) {\n      nextLineIndex = mapNextLine.line;\n      shouldLimit = nextLineIndex === lineIndex;\n      nextOffset = mapNextLine.offset;\n    }\n    const obj =\n      typeof lineIndex === 'undefined'\n        ? this.styles\n        : { line: this.styles[lineIndex] };\n    for (const p1 in obj) {\n      for (const p2 in obj[p1]) {\n        const p2Number = parseInt(p2, 10);\n        if (p2Number >= offset && (!shouldLimit || p2Number < nextOffset!)) {\n          // eslint-disable-next-line no-unused-vars\n          for (const p3 in obj[p1][p2]) {\n            return false;\n          }\n        }\n      }\n    }\n    return true;\n  }\n\n  /**\n   * @protected\n   * @param {Number} lineIndex\n   * @param {Number} charIndex\n   * @return {TextStyleDeclaration} a style object reference to the existing one or a new empty object when undefined\n   */\n  _getStyleDeclaration(\n    lineIndex: number,\n    charIndex: number\n  ): TextStyleDeclaration {\n    if (this._styleMap && !this.isWrapping) {\n      const map = this._styleMap[lineIndex];\n      if (!map) {\n        return {};\n      }\n      lineIndex = map.line;\n      charIndex = map.offset + charIndex;\n    }\n    return super._getStyleDeclaration(lineIndex, charIndex);\n  }\n\n  /**\n   * @param {Number} lineIndex\n   * @param {Number} charIndex\n   * @param {Object} style\n   * @private\n   */\n  protected _setStyleDeclaration(\n    lineIndex: number,\n    charIndex: number,\n    style: object\n  ) {\n    const map = this._styleMap[lineIndex];\n    super._setStyleDeclaration(map.line, map.offset + charIndex, style);\n  }\n\n  /**\n   * @param {Number} lineIndex\n   * @param {Number} charIndex\n   * @private\n   */\n  protected _deleteStyleDeclaration(lineIndex: number, charIndex: number) {\n    const map = this._styleMap[lineIndex];\n    super._deleteStyleDeclaration(map.line, map.offset + charIndex);\n  }\n\n  /**\n   * probably broken need a fix\n   * Returns the real style line that correspond to the wrapped lineIndex line\n   * Used just to verify if the line does exist or not.\n   * @param {Number} lineIndex\n   * @returns {Boolean} if the line exists or not\n   * @private\n   */\n  protected _getLineStyle(lineIndex: number): boolean {\n    const map = this._styleMap[lineIndex];\n    return !!this.styles[map.line];\n  }\n\n  /**\n   * Set the line style to an empty object so that is initialized\n   * @param {Number} lineIndex\n   * @param {Object} style\n   * @private\n   */\n  protected _setLineStyle(lineIndex: number) {\n    const map = this._styleMap[lineIndex];\n    super._setLineStyle(map.line);\n  }\n\n  /**\n   * Wraps text using the 'width' property of Textbox. First this function\n   * splits text on newlines, so we preserve newlines entered by the user.\n   * Then it wraps each line using the width of the Textbox by calling\n   * _wrapLine().\n   * @param {Array} lines The string array of text that is split into lines\n   * @param {Number} desiredWidth width you want to wrap to\n   * @returns {Array} Array of lines\n   */\n  _wrapText(lines: string[], desiredWidth: number): string[][] {\n    this.isWrapping = true;\n    // extract all thewords and the widths to optimally wrap lines.\n    const data = this.getGraphemeDataForRender(lines);\n    const wrapped: string[][] = [];\n    for (let i = 0; i < data.wordsData.length; i++) {\n      wrapped.push(...this._wrapLine(i, desiredWidth, data));\n    }\n    this.isWrapping = false;\n    return wrapped;\n  }\n\n  /**\n   * For each line of text terminated by an hard line stop,\n   * measure each word width and extract the largest word from all.\n   * The returned words here are the one that at the end will be rendered.\n   * @param {string[]} lines the lines we need to measure\n   *\n   */\n  getGraphemeDataForRender(lines: string[]): GraphemeData {\n    const splitByGrapheme = this.splitByGrapheme,\n      infix = splitByGrapheme ? '' : ' ';\n\n    let largestWordWidth = 0;\n\n    const data = lines.map((line, lineIndex) => {\n      let offset = 0;\n      const wordsOrGraphemes = splitByGrapheme\n        ? this.graphemeSplit(line)\n        : this.wordSplit(line);\n\n      if (wordsOrGraphemes.length === 0) {\n        return [{ word: [], width: 0 }];\n      }\n\n      return wordsOrGraphemes.map((word: string) => {\n        // if using splitByGrapheme words are already in graphemes.\n        const graphemeArray = splitByGrapheme\n          ? [word]\n          : this.graphemeSplit(word);\n        const width = this._measureWord(graphemeArray, lineIndex, offset);\n        largestWordWidth = Math.max(width, largestWordWidth);\n        offset += graphemeArray.length + infix.length;\n        return { word: graphemeArray, width };\n      });\n    });\n\n    return {\n      wordsData: data,\n      largestWordWidth,\n    };\n  }\n\n  /**\n   * Helper function to measure a string of text, given its lineIndex and charIndex offset\n   * It gets called when charBounds are not available yet.\n   * Override if necessary\n   * Use with {@link Textbox#wordSplit}\n   *\n   * @param {CanvasRenderingContext2D} ctx\n   * @param {String} text\n   * @param {number} lineIndex\n   * @param {number} charOffset\n   * @returns {number}\n   */\n  _measureWord(word: string[], lineIndex: number, charOffset = 0): number {\n    let width = 0,\n      prevGrapheme;\n    const skipLeft = true;\n    for (let i = 0, len = word.length; i < len; i++) {\n      const box = this._getGraphemeBox(\n        word[i],\n        lineIndex,\n        i + charOffset,\n        prevGrapheme,\n        skipLeft\n      );\n      width += box.kernedWidth;\n      prevGrapheme = word[i];\n    }\n    return width;\n  }\n\n  /**\n   * Override this method to customize word splitting\n   * Use with {@link Textbox#_measureWord}\n   * @param {string} value\n   * @returns {string[]} array of words\n   */\n  wordSplit(value: string): string[] {\n    return value.split(this._wordJoiners);\n  }\n\n  /**\n   * Wraps a line of text using the width of the Textbox as desiredWidth\n   * and leveraging the known width o words from GraphemeData\n   * @private\n   * @param {Number} lineIndex\n   * @param {Number} desiredWidth width you want to wrap the line to\n   * @param {GraphemeData} graphemeData an object containing all the lines' words width.\n   * @param {Number} reservedSpace space to remove from wrapping for custom functionalities\n   * @returns {Array} Array of line(s) into which the given text is wrapped\n   * to.\n   */\n  _wrapLine(\n    lineIndex: number,\n    desiredWidth: number,\n    { largestWordWidth, wordsData }: GraphemeData,\n    reservedSpace = 0\n  ): string[][] {\n    const additionalSpace = this._getWidthOfCharSpacing(),\n      splitByGrapheme = this.splitByGrapheme,\n      graphemeLines = [],\n      infix = splitByGrapheme ? '' : ' ';\n\n    let lineWidth = 0,\n      line: string[] = [],\n      // spaces in different languages?\n      offset = 0,\n      infixWidth = 0,\n      lineJustStarted = true;\n\n    desiredWidth -= reservedSpace;\n\n    const maxWidth = Math.max(\n      desiredWidth,\n      largestWordWidth,\n      this.dynamicMinWidth\n    );\n    // layout words\n    const data = wordsData[lineIndex];\n    offset = 0;\n    let i;\n    for (i = 0; i < data.length; i++) {\n      const { word, width: wordWidth } = data[i];\n      offset += word.length;\n\n      lineWidth += infixWidth + wordWidth - additionalSpace;\n      if (lineWidth > maxWidth && !lineJustStarted) {\n        graphemeLines.push(line);\n        line = [];\n        lineWidth = wordWidth;\n        lineJustStarted = true;\n      } else {\n        lineWidth += additionalSpace;\n      }\n\n      if (!lineJustStarted && !splitByGrapheme) {\n        line.push(infix);\n      }\n      line = line.concat(word);\n\n      infixWidth = splitByGrapheme\n        ? 0\n        : this._measureWord([infix], lineIndex, offset);\n      offset++;\n      lineJustStarted = false;\n    }\n\n    i && graphemeLines.push(line);\n\n    // TODO: this code is probably not necessary anymore.\n    // it can be moved out of this function since largestWordWidth is now\n    // known in advance\n    if (largestWordWidth + reservedSpace > this.dynamicMinWidth) {\n      this.dynamicMinWidth = largestWordWidth - additionalSpace + reservedSpace;\n    }\n    return graphemeLines;\n  }\n\n  /**\n   * Detect if the text line is ended with an hard break\n   * text and itext do not have wrapping, return false\n   * @param {Number} lineIndex text to split\n   * @return {Boolean}\n   */\n  isEndOfWrapping(lineIndex: number): boolean {\n    if (!this._styleMap[lineIndex + 1]) {\n      // is last line, return true;\n      return true;\n    }\n    if (this._styleMap[lineIndex + 1].line !== this._styleMap[lineIndex].line) {\n      // this is last line before a line break, return true;\n      return true;\n    }\n    return false;\n  }\n\n  /**\n   * Detect if a line has a linebreak and so we need to account for it when moving\n   * and counting style.\n   * This is important only for splitByGrapheme at the end of wrapping.\n   * If we are not wrapping the offset is always 1\n   * @return Number\n   */\n  missingNewlineOffset(lineIndex: number, skipWrapping?: boolean): 0 | 1 {\n    if (this.splitByGrapheme && !skipWrapping) {\n      return this.isEndOfWrapping(lineIndex) ? 1 : 0;\n    }\n    return 1;\n  }\n\n  /**\n   * Gets lines of text to render in the Textbox. This function calculates\n   * text wrapping on the fly every time it is called.\n   * @param {String} text text to split\n   * @returns {Array} Array of lines in the Textbox.\n   * @override\n   */\n  _splitTextIntoLines(text: string) {\n    const newText = super._splitTextIntoLines(text),\n      graphemeLines = this._wrapText(newText.lines, this.width),\n      lines = new Array(graphemeLines.length);\n    for (let i = 0; i < graphemeLines.length; i++) {\n      lines[i] = graphemeLines[i].join('');\n    }\n    newText.lines = lines;\n    newText.graphemeLines = graphemeLines;\n    return newText;\n  }\n\n  getMinWidth() {\n    return Math.max(this.minWidth, this.dynamicMinWidth);\n  }\n\n  _removeExtraneousStyles() {\n    const linesToKeep = new Map();\n    for (const prop in this._styleMap) {\n      const propNumber = parseInt(prop, 10);\n      if (this._textLines[propNumber]) {\n        const lineIndex = this._styleMap[prop].line;\n        linesToKeep.set(`${lineIndex}`, true);\n      }\n    }\n    for (const prop in this.styles) {\n      if (!linesToKeep.has(prop)) {\n        delete this.styles[prop];\n      }\n    }\n  }\n\n  /**\n   * Returns object representation of an instance\n   * @method toObject\n   * @param {Array} [propertiesToInclude] Any properties that you might want to additionally include in the output\n   * @return {Object} object representation of an instance\n   */\n  // cant use ts-expect-error because of ts 5.3 cross check\n  // @ts-ignore TS this typing limitations\n  toObject<\n    T extends Omit<Props & TClassProperties<this>, keyof SProps>,\n    K extends keyof T = never\n  >(propertiesToInclude: K[] = []): Pick<T, K> & SProps {\n    return super.toObject<T, K>([\n      'minWidth',\n      'splitByGrapheme',\n      ...propertiesToInclude,\n    ] as K[]) as Pick<T, K> & SProps;\n  }\n}\n\nclassRegistry.setClass(Textbox);\n","import { WidgetBaseInterface, TOriginX, TOriginY } from './widget.entity.base';\nimport { xy } from './widget.entity.connector';\nimport { WidgetType } from './widget.type';\n\nexport type Connector = {\n  connectorId: string;\n  connectorType: string;\n  point: xy;\n}\n\n\n\nexport interface WidgetTextboxInterface extends WidgetBaseInterface {\n  fontFamily: string;\n  fill: string | any | null;\n  fontSize: number;\n  fontWeight: string;\n  lineHeight: number;\n  text: string;\n  textAlign: string;\n  editable: boolean;\n  fixedScaleChange: boolean;\n  connectors: Connector[]; // You can replace 'any' with the appropriate type for connectors\n}\n\nexport class WidgetTextboxClass implements WidgetTextboxInterface {\n  updatedBy: string = \"\";\n  updatedByName: string = \"\";\n  fill: string = '#eeeeee';\n  createdByName: string = \"\";\n  fontFamily: string = 'Inter';\n  fontSize: number = 16;\n  fontWeight: string = '400';\n  lineHeight: number = 1.2;\n  text: string = '';\n  textAlign: string = 'left';\n  editable: boolean = true;\n  maxHeight: number = 100;\n  fixedScaleChange: boolean = false;\n  connectors: Connector[] = [];\n  id: string = '';\n  boardId: string = '';\n  backgroundColor: string = '#FFFFFF';\n  width: number = 100;\n  height: number = 50;\n  left: number = 0;\n  locked: boolean = false;\n  objType: WidgetType = \"XTextbox\";\n  originX: TOriginX = 'center';\n  originY: TOriginY = 'center';\n  scaleX: number = 1;\n  scaleY: number = 1;\n  selectable: boolean = true;\n  top: number = 0;\n  zIndex: number = 0;\n  version: string = '1.0';\n  updatedAt: number = Date.now();\n\n  createdAt: number = Date.now();\n  createdBy: string = '';\n  visible: boolean = true;\n}\n\nexport const EntityKeys = Object.keys(new WidgetTextboxClass()) as (keyof WidgetTextboxInterface)[];","import { TClassProperties, TOriginX, TOriginY } from '../../typedefs';\nimport { IText } from '../IText/IText';\nimport { classRegistry } from '../../ClassRegistry';\nimport { createTextboxDefaultControls } from '../../controls/X_commonControls';\nimport { EventName, Origin } from './types';\nimport { Textbox } from '../Textbox';\n\nimport { WidgetType } from './type/widget.type';\nimport { WidgetTextboxInterface, EntityKeys } from './type/widget.entity.textbox';\n\n\nimport { Point } from '../../Point';\nimport { XConnector } from './XConnector';\nimport { FabricObject } from '../Object/Object';\n\n\n// @TODO: Many things here are configuration related and shouldn't be on the class nor prototype\n// regexes, list of properties that are not suppose to change by instances, magic consts.\n// this will be a separated effort\nexport const textboxDefaultValues: Partial<TClassProperties<XTextbase>> = {\n  minWidth: 20,\n  dynamicMinWidth: 2,\n  // _wordJoiners: /[ \\t\\r]/,\n  splitByGrapheme: false,\n  cornerColor: 'white',\n  cornerSize: 10,\n  cornerStyle: 'circle',\n  transparentCorners: false,\n  cornerStrokeColor: 'gray',\n  connectors: [],\n};\n\nclass Connector {\n  connectorId: string;\n  connectorType: string;\n  point: Point;\n}\n\nexport const XTextbaseProps: Partial<TClassProperties<XTextbase>> = {};\n\n/**\n * Textbox class, based on IText, allows the user to resize the text rectangle\n * and wraps lines automatically. Textboxes have their Y scaling locked, the\n * user can only change width. Height is adjusted automatically based on the\n * wrapping of lines.\n */\nexport class XTextbase\n  extends Textbox\n  implements WidgetTextboxInterface {\n  static type: WidgetType = 'XTextbase';\n  static objType: WidgetType = 'XTextbase';\n  /**\n   * Minimum width of textbox, in pixels.\n   * @type Number\n   * @default\n   */\n  declare minWidth: number;\n\n  declare tempTop: number;\n\n  declare hasNoText: boolean;\n\n  /**\n   * Minimum calculated width of a textbox, in pixels.\n   * fixed to 2 so that an empty textbox cannot go to 0\n   * and is still selectable without text.\n   * @type Number\n   * @default\n   */\n  declare dynamicMinWidth: number;\n\n  declare oneLine: boolean;\n\n  declare fromCopy: boolean;\n  declare originX: TOriginX;\n  declare originY: TOriginY;\n  declare connectors: Connector[];\n  /**\n   * Use this boolean property in order to split strings that have no white space concept.\n   * this is a cheap way to help with chinese/japanese\n   * @type Boolean\n   * @since 2.6.0\n   */\n  declare splitByGrapheme: boolean;\n\n  static textLayoutProperties = [...IText.textLayoutProperties, 'width'];\n\n  static ownDefaults: Record<string, any> = textboxDefaultValues;\n\n\n\n  static getDefaults() {\n    return {\n      ...super.getDefaults(),\n      controls: createTextboxDefaultControls(),\n      ...XTextbase.ownDefaults,\n    };\n  }\n\n\n\n  constructor(text: string, options: any) {\n\n    options.oneLine = options.oneLine || true;\n    options.connectors = options.connectors || [];\n    options.originX = options.originX || 'center';\n    options.originY = options.originY || 'center';\n    options.minWidth = options.minWidth || 20;\n    options.dynamicMinWidth = options.dynamicMinWidth || 2;\n    options.splitByGrapheme = options.splitByGrapheme || false;\n    options.fixedScaleChange = options.fixedScaleChange || false;\n    options.boardId = options.boardId || '';\n\n    options.userId = options.userId || '';\n    options.zIndex = options.zIndex || 0;\n    options.version = options.version || '';\n    options.updatedAt = options.updatedAt || Date.now();\n    options.lastEditedByName = options.lastEditedByName || '';\n    options.createdByName = options.createdByName || '';\n    options.objType = 'XTextbase';\n\n\n    super(text, options);\n    this.initializeEvent();\n    delete options.height;\n    Object.assign(this, options);\n\n\n    // this.resetResizeControls();\n  }\n  updatedBy: string;\n  updatedByName: string;\n\n  createdByName: string;\n  maxHeight: number;\n  fixedScaleChange: boolean;\n  boardId: string;\n  objType: WidgetType;\n  userId: string;\n  zIndex: number;\n  version: string;\n  updatedAt: number;\n\n  createdAt: number;\n  createdBy: string;\n  /* boardx extend function */\n\n  updateConnector(point: any, connector: XConnector, type: string) {\n    const controlPoint = this.calculateControlPoint(\n      new Point(point.x, point.y)\n    );\n\n    //recalculate the startpoint or endpoint of the connector, and also the ControlPoint\n    if (type === 'from') {\n      connector.update({\n        fromPoint: point,\n        control1: controlPoint,\n      });\n    }\n    if (type === 'to') {\n      connector.update({\n        toPoint: point,\n        control2: controlPoint,\n      });\n    }\n  }\n\n  moveOrScaleHandler(e: any) {\n    //if there is a connector, move the connector\n    if (this.connectors?.length === 0) return;\n    this.connectors?.forEach((connector: any) => {\n      const pointConnector = connector.point;\n\n      //get canvas point of the connector point\n      const point = new Point(pointConnector.x, pointConnector.y);\n      //@ts-ignore\n      const transformedPoint = this.transformPointToCanvas(point);\n\n      //use the connectorId to find the connector and then update the connector\n      //@ts-ignore\n      const connectorObj = this.canvas?.findById(connector.connectorId);\n\n      if (!connectorObj) return;\n\n      if (\n        this.id === connectorObj.fromObjectId &&\n        connector.connectorType === 'from'\n      ) {\n        this.updateConnector(transformedPoint, connectorObj, 'from');\n      }\n\n      if (\n        this.id === connectorObj.toObjectId &&\n        connector.connectorType === 'to'\n      ) {\n        this.updateConnector(transformedPoint, connectorObj, 'to');\n      }\n    });\n  }\n\n  calculateControlPoint(connectingPoint: Point): Point {\n    const boundingBox = this.getBoundingRect();\n    let left = boundingBox.left;\n    let top = boundingBox.top;\n\n    const width = boundingBox.width;\n    const height = boundingBox.height;\n\n    const right = left + width;\n    const bottom = top + height;\n\n    const connectingX = connectingPoint.x;\n    const connectingY = connectingPoint.y;\n\n    let controlX: number = 0;\n    let controlY: number = 0;\n\n    // Find the nearest border and calculate the control point outside the bounding box\n    const distances = [\n      { side: 'left', distance: Math.abs(connectingX - left) },\n      { side: 'right', distance: Math.abs(connectingX - right) },\n      { side: 'top', distance: Math.abs(connectingY - top) },\n      { side: 'bottom', distance: Math.abs(connectingY - bottom) },\n    ];\n\n    const nearestBorder = distances.reduce((min, current) =>\n      current.distance < min.distance ? current : min\n    );\n\n    switch (nearestBorder.side) {\n      case 'left':\n        controlX = left - 220 * this.scaleX;\n        controlY = connectingY;\n        break;\n      case 'right':\n        controlX = right + 220 * this.scaleX;\n        controlY = connectingY;\n        break;\n      case 'top':\n        controlX = connectingX;\n        controlY = top - 220 * this.scaleY;\n        break;\n      case 'bottom':\n        controlX = connectingX;\n        controlY = bottom + 220 * this.scaleY;\n        break;\n    }\n\n    return new Point(controlX, controlY);\n  }\n\n\n\n  /**\n   * Generate an object that translates the style object so that it is\n   * broken up by visual lines (new lines and automatic wrapping).\n   * The original text styles object is broken up by actual lines (new lines only),\n   * which is only sufficient for Text / IText\n   * @private\n   */\n  _generateStyleMap(textInfo: any) {\n    let realLineCount = 0,\n      realLineCharCount = 0,\n      charCount = 0;\n    const map: any = {};\n\n    for (let i = 0; i < textInfo.graphemeLines.length; i++) {\n      if (textInfo.graphemeText[charCount] === '\\n' && i > 0) {\n        realLineCharCount = 0;\n        charCount++;\n        realLineCount++;\n      } else if (\n        !this.splitByGrapheme &&\n        this._reSpaceAndTab.test(textInfo.graphemeText[charCount]) &&\n        i > 0\n      ) {\n        // this case deals with space's that are removed from end of lines when wrapping\n        realLineCharCount++;\n        charCount++;\n      }\n\n      map[i] = { line: realLineCount, offset: realLineCharCount };\n\n      charCount += textInfo.graphemeLines[i].length;\n      realLineCharCount += textInfo.graphemeLines[i].length;\n    }\n\n    return map;\n  }\n\n  /**\n   * Returns true if object has a style property or has it on a specified line\n   * @param {Number} lineIndex\n   * @return {Boolean}\n   */\n  styleHas(property: any, lineIndex: number): boolean {\n    if (this._styleMap && !this.isWrapping) {\n      const map = this._styleMap[lineIndex];\n      if (map) {\n        lineIndex = map.line;\n      }\n    }\n    return super.styleHas(property, lineIndex);\n  }\n\n  /**\n   * Returns true if object has no styling or no styling in a line\n   * @param {Number} lineIndex , lineIndex is on wrapped lines.\n   * @return {Boolean}\n   */\n  isEmptyStyles(lineIndex: number): boolean {\n    if (!this.styles) {\n      return true;\n    }\n    let offset: number = 0,\n      nextLineIndex = lineIndex + 1,\n      nextOffset: any,\n      shouldLimit = false;\n    const map = this._styleMap[lineIndex],\n      mapNextLine = this._styleMap[lineIndex + 1];\n    if (map) {\n      lineIndex = map.line;\n      offset = map.offset;\n    }\n    if (mapNextLine) {\n      nextLineIndex = mapNextLine.line;\n      shouldLimit = nextLineIndex === lineIndex;\n      nextOffset = mapNextLine.offset;\n    }\n    const obj =\n      typeof lineIndex === 'undefined'\n        ? this.styles\n        : { line: this.styles[lineIndex] };\n    for (const p1 in obj as any) {\n      for (const p2 in obj[p1] as any) {\n        if (Number(p2) >= offset && (!shouldLimit || Number(p2) < nextOffset)) {\n          // eslint-disable-next-line no-unused-vars\n          for (const p3 in obj[p1][p2]) {\n            return false;\n          }\n        }\n      }\n    }\n    return true;\n  }\n\n  /**\n   * Helper function to measure a string of text, given its lineIndex and charIndex offset\n   * It gets called when charBounds are not available yet.\n   * Override if necessary\n   * Use with {@link Textbox#wordSplit}\n   *\n   * @param {CanvasRenderingContext2D} ctx\n   * @param {String} text\n   * @param {number} lineIndex\n   * @param {number} charOffset\n   * @returns {number}\n   */\n  _measureWord(word: any, lineIndex: number, charOffset = 0): number {\n    let width = 0,\n      prevGrapheme;\n    const skipLeft = true;\n    for (let i = 0, len = word.length; i < len; i++) {\n      const box = this._getGraphemeBox(\n        word[i],\n        lineIndex,\n        i + charOffset,\n        prevGrapheme,\n        skipLeft\n      );\n      width += box.kernedWidth;\n      prevGrapheme = word[i];\n    }\n    return width;\n  }\n\n  /**\n   * Override this method to customize word splitting\n   * Use with {@link Textbox#_measureWord}\n   * @param {string} value\n   * @returns {string[]} array of words\n   */\n  wordSplit(value: string): string[] {\n    return value.split(this._wordJoiners);\n  }\n\n\n  /**\n   * Detect if the text line is ended with an hard break\n   * text and itext do not have wrapping, return false\n   * @param {Number} lineIndex text to split\n   * @return {Boolean}\n   */\n  isEndOfWrapping(lineIndex: number): boolean {\n    if (!this._styleMap[lineIndex + 1]) {\n      // is last line, return true;\n      return true;\n    }\n    if (this._styleMap[lineIndex + 1].line !== this._styleMap[lineIndex].line) {\n      // this is last line before a line break, return true;\n      return true;\n    }\n    return false;\n  }\n\n  /**\n   * Detect if a line has a linebreak and so we need to account for it when moving\n   * and counting style.\n   * @return Number\n   */\n  missingNewlineOffset(lineIndex: number) {\n    if (this.splitByGrapheme) {\n      return this.isEndOfWrapping(lineIndex) ? 1 : 0;\n    }\n    return 1;\n  }\n\n  /**\n   * Gets lines of text to render in the Textbox. This function calculates\n   * text wrapping on the fly every time it is called.\n   * @param {String} text text to split\n   * @returns {Array} Array of lines in the Textbox.\n   * @override\n   */\n  _splitTextIntoLines(text: string) {\n    const newText = super._splitTextIntoLines(text);\n    // Check if text contains Chinese characters\n    if (/[\\u3400-\\u9FBF]/.test(this.text)) {\n      this.splitByGrapheme = true;\n    }\n\n    if (!this.fromCopy) {\n      if (\n        (this.objType === 'XText' || this.objType === 'XTextbase') &&\n        this.textLines &&\n        this.textLines.length > 1 &&\n        this.isEditing\n      ) {\n        this.oneLine = false;\n      } else {\n        this.oneLine = true;\n      }\n    } else {\n      this.oneLine = false;\n    }\n    if (\n      (this.objType === 'XText' || this.objType === 'XTextbase') &&\n      newText &&\n      newText.lines &&\n      this.oneLine &&\n      this.isEditing\n    ) {\n      if (newText.lines[0].length > 1) {\n        this.width =\n          this._measureWord(newText.lines[0], 0, 0) > this.width\n            ? this._measureWord(newText.lines[0], 0, 0) + 10\n            : this.width;\n      }\n    }\n    const graphemeLines = this._wrapText(newText.lines, this.width);\n    const lines = new Array(graphemeLines.length);\n    for (let i = 0; i < graphemeLines.length; i++) {\n      lines[i] = graphemeLines[i].join('');\n    }\n    newText.lines = lines;\n    newText.graphemeLines = graphemeLines;\n    return newText;\n  }\n\n  getMinWidth() {\n    return Math.max(this.minWidth, this.dynamicMinWidth);\n  }\n\n  controlMousedownProcess(transformData: any, rx: any, ry: any) {\n    return;\n  }\n\n  getObject() {\n    const entityKeys: string[] = EntityKeys;\n    const result: Record<string, any> = {};\n\n    entityKeys.forEach((key) => {\n      if (key in this) {\n        result[key] = (this as any)[key];\n      }\n    });\n\n    return result;\n  }\n\n\n  // toObject(propertiesToInclude: Array<any>): object {\n  //   return super.toObject(\n  //     ['minWidth', 'splitByGrapheme'].concat(propertiesToInclude)\n  //   );\n  // }\n  /**extend function for fronted */\n  checkTextboxChange() { }\n  initializeEvent() {\n    const self = this;\n    const canvas = this.canvas;\n\n    self.on(EventName.EDITINGENTERED, () => {\n      // Set originX based on textAlign\n      // self.originX = self.textAlign as TOriginX;\n\n      // Adjust left position to maintain visual consistency\n      // if (self.textAlign === TextAlign.LEFT) {\n      //   self.left -= (self.width * self.scaleX) / 2;\n      // }\n\n      // if (self.textAlign === TextAlign.RIGHT) {\n      //   // Corrected adjustment: Subtract instead of Add\n      //   self.left -= (self.width * self.scaleX) / 2;\n      // }\n\n      if (self.objType === 'XTextbase') {\n        self.originY = 'top';\n\n        self.top -= (self.height * self.scaleY) / 2;\n\n        self.tempTop = self.top;\n\n        if (self.text === 'Type here...') {\n          self.selectAll();\n\n          self.text = '';\n\n          self.dirty = true;\n\n          self.fill = 'rgb(0, 0, 0)';\n\n          canvas?.requestRenderAll();\n        }\n      }\n    });\n\n    self.on(EventName.EDITINGEXITED, () => {\n      // Revert originX and originY to center\n      self.originX = Origin.Center;\n      self.originY = Origin.Center;\n\n      // // Reverse the left adjustment based on textAlign\n      // if (self.textAlign === TextAlign.LEFT) {\n      //   self.left += (self.width * self.scaleX) / 2;\n      // }\n\n      // if (self.textAlign === TextAlign.RIGHT) {\n      //   // Reverse the corrected adjustment: Add back\n      //   self.left += (self.width * self.scaleX) / 2;\n      // }\n\n      if (self.objType === 'XTextbase') {\n        self.top = self.tempTop + (self.height * self.scaleY) / 2;\n        self.tempTop = self.top;\n      }\n    });\n\n\n    this.on('moving', (e: any) => {\n      this.moveOrScaleHandler(e);\n    });\n\n    this.on('scaling', (e: any) => {\n      this.moveOrScaleHandler(e);\n    });\n    self.on(EventName.MODIFIED, () => {\n      self.checkTextboxChange();\n\n      // canvas.requestRenderAll();\n    });\n    self.on(EventName.CHANGED, () => {\n      if (self.styles[0]) {\n        self.styles = {};\n\n        // self.canvas.requestRenderAll();\n      }\n    });\n\n\n    // Other event listeners remain unchanged...\n  }\n\n\n\n\n\n  drawObject(ctx: CanvasRenderingContext2D) {\n    super.drawObject(ctx);\n    // console.log('!@@ drawObject', this.canvas?.dockingWidget, this);\n    //@ts-ignore\n    if (this == this.canvas?.dockingWidget) {\n      this.renderDockingControls(ctx);\n    }\n  }\n\n  renderDockingControls(ctx: CanvasRenderingContext2D) {\n    console.log('!!@renderDockingControls');\n    const self = this;\n    const canvas = self.canvas;\n    const controls = self.controls;\n\n    let cornerColor = 'white';\n\n    if (!canvas) return;\n\n    for (const controlKey in controls) {\n      const control = controls[controlKey];\n      if (\n        !(\n          controlKey === 'mbaStart' ||\n          controlKey === 'mlaStart' ||\n          controlKey === 'mraStart' ||\n          controlKey === 'mtaStart'\n        )\n      )\n        continue;\n\n      if (\n        //@ts-ignore\n        this.canvas!.hoveringControl &&\n        //@ts-ignore\n        this.canvas!.hoveringControl === controlKey\n      ) {\n        cornerColor = '#F21D6B';\n      } else {\n        cornerColor = 'white';\n      }\n\n      //render 4 controls, mbaStart, mlaStart, mraStart, mtaStart\n\n      this._renderControl(\n        ctx,\n        control.x * self.width,\n        control.y * self.height,\n        { cornerStyle: 'circle', cornerColor },\n        self\n      );\n    }\n  }\n\n  _renderControl(\n    ctx: any,\n    left: number,\n    top: number,\n    styleOverride: any,\n    fabricObject: FabricObject\n  ) {\n    console.log('!!@  _renderControl', left, top);\n    let color = styleOverride.cornerColor || 'white';\n\n    ctx.save();\n    ctx.fillStyle = color;\n    ctx.strokeStyle = 'gray';\n    ctx.beginPath();\n    ctx.arc(left, top, 6, 0, Math.PI * 2, false);\n    ctx.closePath();\n    ctx.fill();\n    ctx.stroke();\n    ctx.restore();\n  }\n\n\n\n  resetResizeControls() {\n    const self = this;\n    const textAlign = self.textAlign;\n\n    if (\n      self.objType === 'XTextbase' &&\n      (textAlign === 'left' || textAlign === 'center')\n    ) {\n      self.setControlVisible('ml', false);\n      self.setControlVisible('mr', true);\n    }\n\n    if (self.objType === 'XText' && textAlign === 'right') {\n      self.setControlVisible('ml', true);\n      self.setControlVisible('mr', false);\n    }\n    if (self.canvas) self.canvas.requestRenderAll();\n  }\n}\n\nclassRegistry.setClass(XTextbase);\n// classRegistry.getSVGClass(Textbox);\n","import { TOriginX, TOriginY } from './widget.entity.base';\nimport   { WidgetTextboxInterface,Connector } from './widget.entity.textbox';\nimport { WidgetType } from './widget.type';\n \nexport   interface WidgetCircleNotesInterface extends WidgetTextboxInterface {}\n\nexport class WidgetCircleNotesClass implements WidgetCircleNotesInterface {\n    fill: string= '#eeeeee';\n    updatedBy: string = \"\";\n    updatedByName: string = \"\";\n    createdByName: string=\"\";\n    fontFamily: string = 'Inter';\n    fontSize: number = 12;\n    fontWeight: string = '400';\n    lineHeight: number = 1.2;\n    text: string = '';\n    textAlign: string = 'center';\n    editable: boolean = true;\n    maxHeight: number = 138;\n    fixedScaleChange: boolean = false;\n    connectors: Connector[] = [];\n    id: string = '';\n    boardId: string = '';\n    backgroundColor: string = 'rgba(252,236,138,1)';\n    width: number = 138;\n    height: number = 138;\n    left: number = 0;\n    locked: boolean = false;\n    objType: WidgetType = 'XCircleNotes';\n    originX: TOriginX = 'center';\n    originY: TOriginY = 'center';\n    scaleX: number = 1;\n    scaleY: number = 1;\n    selectable: boolean = true;\n    top: number = 0;\n    zIndex: number = Date.now()*100;\n    version: string = '1.0';\n    updatedAt: number = Date.now();\n    createdAt: number = Date.now();\n    createdBy: string = '';\n    visible: boolean = true;\n}\n\n\nexport const EntityKeys = Object.keys(new WidgetCircleNotesClass()) as (keyof WidgetCircleNotesInterface)[];","import { TClassProperties } from '../../typedefs';\nimport { classRegistry } from '../../ClassRegistry';\nimport { XTextbase } from './XTextbase';\nimport { createRectNotesDefaultControls } from '../../controls/X_commonControls';\n\nimport { EntityKeys, WidgetCircleNotesInterface } from './type/widget.entity.circlenote';\nimport { WidgetType } from './type/widget.type';\n\n// @TODO: Many things here are configuration related and shouldn't be on the class nor prototype\n// regexes, list of properties that are not suppose to change by instances, magic consts.\n// this will be a separated effort\nexport const circleNotesDefaultValues: Partial<TClassProperties<XCircleNotes>> =\n{\n  minWidth: 20,\n  dynamicMinWidth: 2,\n  verticalAlign: 'middle',\n  lockScalingFlip: true,\n  noScaleCache: false,\n  _wordJoiners: /[ \\t\\r]/,\n  splitByGrapheme: true,\n  objType: 'XCircleNotes',\n  height: 138,\n  maxHeight: 138,\n  width: 138,\n  noteType: 'circle',\n  radius: 138,\n  cornerStrokeColor: 'gray',\n  cornerStyle: 'circle',\n  cornerColor: 'white',\n  transparentCorners: false,\n};\n\nexport interface CircleNotesProps {\n  id: string;\n  originX: string;\n  originY: string;\n  top: number;\n  left: number;\n  textAlign: string;\n  width: number;\n  height: number;\n  backgroundColor: string;\n}\n\n/**\n * Textbox class, based on IText, allows the user to resize the text rectangle\n * and wraps lines automatically. Textboxes have their Y scaling locked, the\n * user can only change width. Height is adjusted automatically based on the\n * wrapping of lines.\n */\nexport class XCircleNotes extends XTextbase implements WidgetCircleNotesInterface {\n  /**selectable\n   * Minimum width of textbox, in pixels.\n   * @type Number\n   * @default\n   */\n  declare minWidth: number;\n  declare maxHeight: number;\n  declare noteType: string;\n  declare radius: number;\n\n  static type: WidgetType = 'XCircleNotes';\n  static objType: WidgetType = 'XCircleNotes';\n\n  /* boardx cusotm function */\n  declare id: string;\n\n  declare locked: boolean;\n\n  declare boardId: string;\n\n  declare userId: string;\n\n  declare timestamp: Date;\n\n  declare verticalAlign: string;\n\n  declare zIndex: number;\n\n  declare lines: object[];\n\n  declare relationship: object[];\n\n  declare emoj: object[];\n\n  declare userEmoji: object[];\n\n  /**\n   * Minimum calculated width of a textbox, in pixels.\n   * fixed to 2 so that an empty textbox cannot go to 0\n   * and is still selectable without text.\n   * @type Number\n   * @default\n   */\n  declare dynamicMinWidth: number;\n\n  /**\n   * Use this boolean property in order to split strings that have no white space concept.\n   * this is a cheap way to help with chinese/japanese\n   * @type Boolean\n   * @since 2.6.0\n   */\n  declare splitByGrapheme: boolean;\n\n  static textLayoutProperties = [...XTextbase.textLayoutProperties, 'width'];\n\n  static ownDefaults: Record<string, any> = circleNotesDefaultValues;\n\n  static getDefaults() {\n    return {\n      ...super.getDefaults(),\n\n      ...XCircleNotes.ownDefaults,\n    };\n  }\n\n  constructor(text: string, options: any) {\n\n    super(text, options);\n    Object.assign(this, options);\n    Object.assign(this, {\n      controls: { ...createRectNotesDefaultControls(this) },\n    });\n    this.objType = 'XCircleNotes';\n  }\n\n\n  getObject() {\n    const entityKeys: string[] = EntityKeys;\n    const result: Record<string, any> = {};\n\n    entityKeys.forEach((key) => {\n      if (key in this) {\n        result[key] = (this as any)[key];\n      }\n    });\n\n    return result;\n  }\n\n\n\n  /**\n   * Unlike superclass's version of this function, Textbox does not update\n   * its width.\n   * @private\n   * @override\n   */\n  initDimensions() {\n    if (!this.initialized) {\n      return;\n    }\n    this.isEditing && this.initDelayedCursor();\n    this._clearCache();\n    // clear dynamicMinWidth as it will be different after we re-wrap line\n    this.dynamicMinWidth = 0;\n    // wrap lines\n    this._styleMap = this._generateStyleMap(this._splitText());\n    // if after wrapping, the width is smaller than dynamicMinWidth, change the width and re-wrap\n    if (this.dynamicMinWidth > this.width) {\n      this.set('fontSize', this.fontSize - 2);\n      this._splitTextIntoLines(this.text);\n      return;\n    }\n    if (this.textAlign.indexOf('justify') !== -1) {\n      // once text is measured we need to make space fatter to make justified text.\n      this.enlargeSpaces();\n    }\n    // clear cache and re-calculate height\n    const height = this.calcTextHeight();\n    if (height > 76 && this.fontSize > 2) {\n      this.set('fontSize', this.fontSize - 2);\n      this._splitTextIntoLines(this.text);\n      return;\n    }\n\n    this.height = this.maxHeight;\n    return this.height;\n  }\n\n  /**\n   * Generate an object that translates the style object so that it is\n   * broken up by visual lines (new lines and automatic wrapping).\n   * The original text styles object is broken up by actual lines (new lines only),\n   * which is only sufficient for Text / IText\n   * @private\n   */\n  _generateStyleMap(textInfo: any) {\n    let realLineCount = 0,\n      realLineCharCount = 0,\n      charCount = 0;\n    const map: any = {};\n\n    for (let i = 0; i < textInfo.graphemeLines.length; i++) {\n      if (textInfo.graphemeText[charCount] === '\\n' && i > 0) {\n        realLineCharCount = 0;\n        charCount++;\n        realLineCount++;\n      } else if (\n        !this.splitByGrapheme &&\n        this._reSpaceAndTab.test(textInfo.graphemeText[charCount]) &&\n        i > 0\n      ) {\n        // this case deals with space's that are removed from end of lines when wrapping\n        realLineCharCount++;\n        charCount++;\n      }\n\n      map[i] = { line: realLineCount, offset: realLineCharCount };\n\n      charCount += textInfo.graphemeLines[i].length;\n      realLineCharCount += textInfo.graphemeLines[i].length;\n    }\n\n    return map;\n  }\n\n  // /**\n  //  * Returns true if object has a style property or has it on a specified line\n  //  * @param {Number} lineIndex\n  //  * @return {Boolean}\n  //  */\n  // styleHas(property, lineIndex: number): boolean {\n  //   if (this._styleMap && !this.isWrapping) {\n  //     const map = this._styleMap[lineIndex];\n  //     if (map) {\n  //       lineIndex = map.line;\n  //     }\n  //   }\n  //   return super.styleHas(property, lineIndex);\n  // }\n\n  // /**\n  //  * Returns true if object has no styling or no styling in a line\n  //  * @param {Number} lineIndex , lineIndex is on wrapped lines.\n  //  * @return {Boolean}\n  //  */\n  // isEmptyStyles(lineIndex: number): boolean {\n  //   if (!this.styles) {\n  //     return true;\n  //   }\n  //   let offset = 0,\n  //     nextLineIndex = lineIndex + 1,\n  //     nextOffset,\n  //     shouldLimit = false;\n  //   const map = this._styleMap[lineIndex],\n  //     mapNextLine = this._styleMap[lineIndex + 1];\n  //   if (map) {\n  //     lineIndex = map.line;\n  //     offset = map.offset;\n  //   }\n  //   if (mapNextLine) {\n  //     nextLineIndex = mapNextLine.line;\n  //     shouldLimit = nextLineIndex === lineIndex;\n  //     nextOffset = mapNextLine.offset;\n  //   }\n  //   const obj =\n  //     typeof lineIndex === 'undefined'\n  //       ? this.styles\n  //       : { line: this.styles[lineIndex] };\n  //   for (const p1 in obj) {\n  //     for (const p2 in obj[p1]) {\n  //       if (p2 >= offset && (!shouldLimit || p2 < nextOffset)) {\n  //         // eslint-disable-next-line no-unused-vars\n  //         for (const p3 in obj[p1][p2]) {\n  //           return false;\n  //         }\n  //       }\n  //     }\n  //   }\n  //   return true;\n  // }\n\n  // /**\n  //  * @param {Number} lineIndex\n  //  * @param {Number} charIndex\n  //  * @private\n  //  */\n  // _getStyleDeclaration(lineIndex: number, charIndex: number) {\n  //   if (this._styleMap && !this.isWrapping) {\n  //     const map = this._styleMap[lineIndex];\n  //     if (!map) {\n  //       return null;\n  //     }\n  //     lineIndex = map.line;\n  //     charIndex = map.offset + charIndex;\n  //   }\n  //   return super._getStyleDeclaration(lineIndex, charIndex);\n  // }\n\n  /**\n   * @param {Number} lineIndex\n   * @param {Number} charIndex\n   * @param {Object} style\n   * @private\n   */\n  _setStyleDeclaration(lineIndex: number, charIndex: number, style: object) {\n    const map = this._styleMap[lineIndex];\n    lineIndex = map.line;\n    charIndex = map.offset + charIndex;\n\n    this.styles[lineIndex][charIndex] = style;\n  }\n\n  /**\n   * @param {Number} lineIndex\n   * @param {Number} charIndex\n   * @private\n   */\n  _deleteStyleDeclaration(lineIndex: number, charIndex: number) {\n    const map = this._styleMap[lineIndex];\n    lineIndex = map.line;\n    charIndex = map.offset + charIndex;\n    delete this.styles[lineIndex][charIndex];\n  }\n\n  /**\n   * probably broken need a fix\n   * Returns the real style line that correspond to the wrapped lineIndex line\n   * Used just to verify if the line does exist or not.\n   * @param {Number} lineIndex\n   * @returns {Boolean} if the line exists or not\n   * @private\n   */\n  _getLineStyle(lineIndex: number): boolean {\n    const map = this._styleMap[lineIndex];\n    return !!this.styles[map.line];\n  }\n\n  /**\n   * Set the line style to an empty object so that is initialized\n   * @param {Number} lineIndex\n   * @param {Object} style\n   * @private\n   */\n  _setLineStyle(lineIndex: number) {\n    const map = this._styleMap[lineIndex];\n    this.styles[map.line] = {};\n  }\n\n  // /**\n  //  * Wraps text using the 'width' property of Textbox. First this function\n  //  * splits text on newlines, so we preserve newlines entered by the user.\n  //  * Then it wraps each line using the width of the Textbox by calling\n  //  * _wrapLine().\n  //  * @param {Array} lines The string array of text that is split into lines\n  //  * @param {Number} desiredWidth width you want to wrap to\n  //  * @returns {Array} Array of lines\n  //  */\n  // _wrapText(lines: Array<any>, desiredWidth: number): Array<any> {\n  //   const wrapped = [];\n  //   this.isWrapping = true;\n  //   for (let i = 0; i < lines.length; i++) {\n  //     wrapped.push(...this._wrapLine(lines[i], i, desiredWidth));\n  //   }\n  //   this.isWrapping = false;\n  //   return wrapped;\n  // }\n\n  /**\n   * Helper function to measure a string of text, given its lineIndex and charIndex offset\n   * It gets called when charBounds are not available yet.\n   * Override if necessary\n   * Use with {@link Textbox#wordSplit}\n   *\n   * @param {CanvasRenderingContext2D} ctx\n   * @param {String} text\n   * @param {number} lineIndex\n   * @param {number} charOffset\n   * @returns {number}\n   */\n  _measureWord(word: any, lineIndex: number, charOffset = 0): number {\n    let width = 0,\n      prevGrapheme;\n    const skipLeft = true;\n    for (let i = 0, len = word.length; i < len; i++) {\n      const box = this._getGraphemeBox(\n        word[i],\n        lineIndex,\n        i + charOffset,\n        prevGrapheme,\n        skipLeft\n      );\n      width += box.kernedWidth;\n      prevGrapheme = word[i];\n    }\n    return width;\n  }\n\n  /**\n   * Override this method to customize word splitting\n   * Use with {@link Textbox#_measureWord}\n   * @param {string} value\n   * @returns {string[]} array of words\n   */\n  wordSplit(value: string): string[] {\n    return value.split(this._wordJoiners);\n  }\n\n  /**\n   * Wraps a line of text using the width of the Textbox and a context.\n   * @param {Array} line The grapheme array that represent the line\n   * @param {Number} lineIndex\n   * @param {Number} desiredWidth width you want to wrap the line to\n   * @param {Number} reservedSpace space to remove from wrapping for custom functionalities\n   * @returns {Array} Array of line(s) into which the given text is wrapped\n   * to.\n   */\n  graphemeSplitForRectNotes(textstring: string): string[] {\n    const graphemes = [];\n    const words = textstring.split(/\\b/);\n    for (let i = 0; i < words.length; i++) {\n      // 检查单词是否全为拉丁字母，长度不大于13，且没有四个或更多的连续相同的字母\n      if (\n        /^[a-zA-Z]+$/.test(words[i]) &&\n        words[i].length <= 13 &&\n        !/(\\w)\\1{3,}/.test(words[i])\n      ) {\n        graphemes.push(words[i]);\n      } else {\n        for (let j = 0; j < words[i].length; j++) {\n          graphemes.push(words[i][j]);\n        }\n      }\n    }\n    return graphemes;\n  }\n\n  // _wrapLine(\n  //   _line: any,\n  //   lineIndex: number,\n  //   desiredWidth: number,\n  //   reservedSpace = 0\n  // ): Array<any> {\n  //   const additionalSpace = this._getWidthOfCharSpacing(),\n  //     splitByGrapheme = this.splitByGrapheme,\n  //     graphemeLines = [],\n  //     words = splitByGrapheme\n  //       ? this.graphemeSplitForRectNotes(_line)\n  //       : this.wordSplit(_line),\n  //     infix = splitByGrapheme ? '' : ' ';\n\n  //   let lineWidth = 0,\n  //     line = [],\n  //     // spaces in different languages?\n  //     offset = 0,\n  //     infixWidth = 0,\n  //     largestWordWidth = 0,\n  //     lineJustStarted = true;\n  //   // fix a difference between split and graphemeSplit\n  //   if (words.length === 0) {\n  //     words.push([]);\n  //   }\n  //   desiredWidth -= reservedSpace;\n  //   // measure words\n  //   const data = words.map((word) => {\n  //     // if using splitByGrapheme words are already in graphemes.\n  //     word = splitByGrapheme ? word : this.graphemeSplitForRectNotes(word);\n  //     const width = this._measureWord(word, lineIndex, offset);\n  //     largestWordWidth = Math.max(width, largestWordWidth);\n  //     offset += word.length + 1;\n  //     return { word: word, width: width };\n  //   });\n  //   const maxWidth = Math.max(\n  //     desiredWidth,\n  //     largestWordWidth,\n  //     this.dynamicMinWidth\n  //   );\n  //   // layout words\n  //   offset = 0;\n  //   let i;\n  //   for (i = 0; i < words.length; i++) {\n  //     const word = data[i].word;\n  //     const wordWidth = data[i].width;\n  //     offset += word.length;\n\n  //     lineWidth += infixWidth + wordWidth - additionalSpace;\n  //     if (lineWidth > maxWidth && !lineJustStarted) {\n  //       graphemeLines.push(line);\n  //       line = [];\n  //       lineWidth = wordWidth;\n  //       lineJustStarted = true;\n  //     } else {\n  //       lineWidth += additionalSpace;\n  //     }\n\n  //     if (!lineJustStarted && !splitByGrapheme) {\n  //       line.push(infix);\n  //     }\n  //     if (word.length > 1) {\n  //       line = line.concat(word.split(''));\n  //     } else {\n  //       line = line.concat(word);\n  //     }\n\n  //     infixWidth = splitByGrapheme\n  //       ? 0\n  //       : this._measureWord([infix], lineIndex, offset);\n  //     offset++;\n  //     lineJustStarted = false;\n  //   }\n\n  //   i && graphemeLines.push(line);\n\n  //   if (largestWordWidth + reservedSpace > this.dynamicMinWidth) {\n  //     this.dynamicMinWidth = largestWordWidth - additionalSpace + reservedSpace;\n  //   }\n  //   return graphemeLines;\n  // }\n\n  /**\n   * Detect if the text line is ended with an hard break\n   * text and itext do not have wrapping, return false\n   * @param {Number} lineIndex text to split\n   * @return {Boolean}\n   */\n  isEndOfWrapping(lineIndex: number): boolean {\n    if (!this._styleMap[lineIndex + 1]) {\n      // is last line, return true;\n      return true;\n    }\n    if (this._styleMap[lineIndex + 1].line !== this._styleMap[lineIndex].line) {\n      // this is last line before a line break, return true;\n      return true;\n    }\n    return false;\n  }\n\n  // /**\n  //  * Detect if a line has a linebreak and so we need to account for it when moving\n  //  * and counting style.\n  //  * @return Number\n  //  */\n  // missingNewlineOffset(lineIndex) {\n  //   if (this.splitByGrapheme) {\n  //     return this.isEndOfWrapping(lineIndex) ? 1 : 0;\n  //   }\n  //   return 1;\n  // }\n\n  /**\n   * Gets lines of text to render in the Textbox. This function calculates\n   * text wrapping on the fly every time it is called.\n   * @param {String} text text to split\n   * @returns {Array} Array of lines in the Textbox.\n   * @override\n   */\n  _splitTextIntoLines(text: string) {\n    const newText = super._splitTextIntoLines(text),\n      graphemeLines = this._wrapText(newText.lines, this.width),\n      lines = new Array(graphemeLines.length);\n    for (let i = 0; i < graphemeLines.length; i++) {\n      lines[i] = graphemeLines[i].join('');\n    }\n    newText.lines = lines;\n    newText.graphemeLines = graphemeLines;\n    return newText;\n  }\n\n  getMinWidth() {\n    return Math.max(this.minWidth, this.dynamicMinWidth);\n  }\n\n\n\n  /* caculate cusor positon in the middle of the textbox */\n  getCenteredTop(rectHeight: any) {\n    const textHeight = this.height;\n    return (rectHeight - textHeight) / 2;\n  }\n\n  _getTopOffset() {\n    switch (this.verticalAlign) {\n      case 'middle':\n        return -this._getTotalLineHeights() / 2;\n      case 'bottom':\n        return this.height / 2 - this._getTotalLineHeights();\n      default:\n        return -this.height / 2;\n    }\n  }\n\n  _getTotalLineHeight() {\n    return this._textLines.reduce(\n      (total, _line, index) => total + this.getHeightOfLine(index),\n      0\n    );\n  }\n\n  _getTotalLineHeights() {\n    return this._textLines.reduce(\n      (total, line, index) => total + this.getHeightOfLine(index),\n      0\n    );\n  }\n\n  _render(ctx: any) {\n    const path: any = this.path;\n\n    path && !path.isNotVisible() && path._render(ctx);\n    this._setTextStyles(ctx);\n    this._renderTextLinesBackground(ctx);\n    this._renderTextDecoration(ctx, 'underline');\n    this._renderText(ctx);\n    this._renderTextDecoration(ctx, 'overline');\n    this._renderTextDecoration(ctx, 'linethrough');\n\n    // const isEmojiExist = !(\n    //   this.emoji === undefined || this.emoji.join() === '0,0,0,0,0'\n    // );\n    // if (isEmojiExist) {\n    //   this.renderEmoji(ctx);\n    // }\n  }\n\n  // renderEmoji(ctx) {\n  //   if (this.emoji === undefined) {\n  //     return;\n  //   }\n\n  //   let width = 0;\n  //   const imageList = [\n  //     this.canvas.emoji_thumb,\n  //     this.canvas.emoji_love,\n  //     this.canvas.emoji_smile,\n  //     this.canvas.emoji_shock,\n  //     this.canvas.emoji_question,\n  //   ];\n  //   const imageListArray = [];\n  //   const emojiList = [];\n  //   for (let i = 0; i < 5; i++) {\n  //     if (this.emoji[i] !== 0) {\n  //       imageListArray.push(imageList[i]);\n  //       emojiList.push(this.emoji[i]);\n  //       width += 26.6;\n  //     }\n  //   }\n\n  //   if (emojiList.length === 0) return;\n\n  //   const x = this.width / 2 - width + this.padding / 2;\n  //   const y = this.height / 2 - 18 + this.padding / 2;\n  //   ctx.font = '10px Inter ';\n  //   ctx.lineJoin = 'round';\n  //   ctx.save();\n  //   ctx.translate(x - 10, y);\n  //   this.drawRoundRectPath(ctx, width, 15, 2);\n  //   ctx.fillStyle = 'rgba(255, 255, 255, 1)';\n  //   ctx.fill();\n  //   ctx.restore();\n\n  //   //ctx.strokeRect(x - 10, y, width, 16);\n  //   //ctx.fillRect(x - 10 + 10 / 2, y + 10 / 2, width - 10, 16 - 10);\n  //   ctx.fillStyle = '#000';\n  //   const isEmojiThumbExist = !(this.canvas.emoji_thumb === undefined);\n  //   if (isEmojiThumbExist) {\n  //     let modifier = 0;\n  //     for (let i = 0; i < imageListArray.length; i++) {\n  //       const imageX = this.width / 2 - 33.6 + modifier + 2 + this.padding / 2;\n  //       const imageY = this.height / 2 - 15 + this.padding / 2;\n  //       const imageW = 10;\n  //       const imageH = 10;\n  //       ctx.drawImage(imageListArray[i], imageX, imageY, imageW, imageH);\n  //       ctx.fillText(\n  //         emojiList[i].toString(),\n  //         this.width / 2 - 20.6 + modifier + 1 + this.padding / 2,\n  //         y + 12\n  //       );\n  //       modifier -= 23.6;\n  //     }\n  //   }\n  // }\n  _renderBackground(ctx: any) {\n    if (!this.backgroundColor) {\n      return;\n    }\n    const dim = this._getNonTransformedDimensions();\n    ctx.fillStyle = this.backgroundColor;\n    ctx.beginPath(); // start new path\n    const radius =\n      dim.x / 2 + this.padding / this.scaleX / (this.canvas?.getZoom() ?? 1);\n    ctx.arc(0, 0, radius, 0, 2 * Math.PI); // draw circle path\n    ctx.closePath(); // close path\n    ctx.strokeStyle = this.backgroundColor;\n    ctx.fillStyle = this.backgroundColor;\n    ctx.stroke();\n    ctx.fill();\n  }\n  _renderText(ctx: any) {\n    ctx.shadowOffsetX = ctx.shadowOffsetY = ctx.shadowBlur = 0;\n    ctx.shadowColor = '';\n\n    if (this.paintFirst === 'stroke') {\n      this._renderTextStroke(ctx);\n      this._renderTextFill(ctx);\n    } else {\n      this._renderTextFill(ctx);\n      this._renderTextStroke(ctx);\n    }\n  }\n  _renderTextCommon(ctx: any, method: any) {\n    ctx.save();\n    let lineHeights = 0;\n    const left = this._getLeftOffset();\n    const top = this._getTopOffset();\n\n    const offsets = this._applyPatternGradientTransform(\n      ctx,\n      //@ts-ignore\n      method === 'fillText' ? this.fill : this.stroke\n    );\n\n    for (let i = 0, len = this._textLines.length; i < len; i++) {\n      const heightOfLine = this.getHeightOfLine(i);\n      const maxHeight = heightOfLine / this.lineHeight;\n      const leftOffset = this._getLineLeftOffset(i);\n      this._renderTextLine(\n        method,\n        ctx,\n        this._textLines[i],\n        left + leftOffset - offsets.offsetX,\n        top + lineHeights + maxHeight - offsets.offsetY,\n        i\n      );\n      lineHeights += heightOfLine;\n    }\n    ctx.restore();\n  }\n\n  _getSVGLeftTopOffsets() {\n    return {\n      textLeft: -this.width / 2,\n      textTop: this._getTopOffset(),\n      lineTop: this.getHeightOfLine(0),\n    };\n  }\n\n  drawRoundRectPath(cxt: any, width: any, height: any, radius: any) {\n    cxt.beginPath(0);\n    //从右下角顺时针绘制，弧度从0到1/2PI\n    cxt.arc(width - radius, height - radius, radius, 0, Math.PI / 2);\n\n    //矩形下边线\n    cxt.lineTo(radius, height);\n\n    //左下角圆弧，弧度从1/2PI到PI\n    cxt.arc(radius, height - radius, radius, Math.PI / 2, Math.PI);\n\n    //矩形左边线\n    cxt.lineTo(0, radius);\n\n    //左上角圆弧，弧度从PI到3/2PI\n    cxt.arc(radius, radius, radius, Math.PI, (Math.PI * 3) / 2);\n\n    //上边线\n    cxt.lineTo(width - radius, 0);\n\n    //右上角圆弧\n    cxt.arc(width - radius, radius, radius, (Math.PI * 3) / 2, Math.PI * 2);\n\n    //右边线\n    cxt.lineTo(width, height - radius);\n    cxt.closePath();\n  }\n}\n\nclassRegistry.setClass(XCircleNotes);\nclassRegistry.setSVGClass(XCircleNotes, 'XCircleNotes');\n","import { TOriginX, TOriginY } from './widget.entity.base';\nimport   {WidgetTextboxInterface, Connector } from './widget.entity.textbox';\nimport { WidgetType } from './widget.type';\n\nexport   interface WidgetRectNotesInterface extends WidgetTextboxInterface {}\n\nexport class WidgetRectNotesClass implements WidgetRectNotesInterface {\n    updatedBy: string = \"\";\n    updatedByName: string = \"\";\n    createdByName: string=\"\";\n    fontFamily: string = 'Inter';\n    fontSize: number = 16;\n    fontWeight: string = '400';\n    lineHeight: number = 1.2;\n    text: string = '';\n    textAlign: string = 'center';\n    editable: boolean = true;\n    fixedScaleChange: boolean = false;\n    connectors: Connector[] = [];\n    id: string = '';\n    boardId: string = '';\n    backgroundColor: string = 'rgba(252,236,138,1)';\n    width: number = 230;\n    height: number = 138;\n    left: number = 0;\n    locked: boolean = false;\n    objType: WidgetType = \"XRectNotes\";\n    originX: TOriginX = 'center';\n    originY: TOriginY = 'center';\n    scaleX: number = 1;\n    scaleY: number = 1;\n    selectable: boolean = true;\n    top: number = 0;\n    zIndex: number = Date.now()*100;\n    version: string = '1.0';\n    updatedAt: number = Date.now();\n    fill: string = 'rgba(33,33,33,1)';\n    createdAt: number = Date.now();\n    createdBy: string = '';\n    visible: boolean = true;\n}\n\nexport const EntityKeys = Object.keys(new WidgetRectNotesClass()) as (keyof WidgetRectNotesInterface)[];","import { classRegistry } from '../../ClassRegistry';\nimport { XTextbase } from './XTextbase';\nimport type { TClassProperties, TOriginX, TOriginY } from '../../typedefs';\nimport { createRectNotesDefaultControls } from '../../controls/X_commonControls';\n\n\nimport { WidgetRectNotesInterface, EntityKeys } from './type/widget.entity.rectnote';\n// this will be a separated effort\nexport const rectNotesDefaultValues: Partial<TClassProperties<XRectNotes>> = {\n  minWidth: 20,\n  dynamicMinWidth: 2,\n  splitByGrapheme: true,\n  height: 138,\n  maxHeight: 138,\n  width: 230,\n  cornerStrokeColor: 'gray',\n  cornerStyle: 'circle',\n  cornerColor: 'white',\n  transparentCorners: false,\n};\n\n/**\n * Textbox class, based on IText, allows the user to resize the text rectangle\n * and wraps lines automatically. Textboxes have their Y scaling locked, the\n * user can only change width. Height is adjusted automatically based on the\n * wrapping of lines.\n */\n///@ts-ignore\nexport class XRectNotes extends XTextbase implements WidgetRectNotesInterface {\n  /**selectable\n   * Minimum width of textbox, in pixels.\n   * @type Number\n   * @default\n   */\n  declare minWidth: number;\n  static type = 'XRectNotes';\n  static objType = 'XRectNotes';\n  declare locked: boolean;\n  declare cornerStyle: any;\n  declare verticalAlign: string;\n  declare originX: TOriginX;\n  declare originY: TOriginY;\n  declare width: number;\n  declare cornerStrokeColor: string;\n\n  declare cornerColor: string;\n  declare transparentCorners: boolean;\n  declare zIndex: number;\n  declare height: number;\n  declare maxHeight: number;\n\n  declare id: string;\n  declare boardId: string;\n\n\n  /**\n   * Minimum calculated width of a textbox, in pixels.\n   * fixed to 2 so that an empty textbox cannot go to 0\n   * and is still selectable without text.\n   * @type Number\n   * @default\n   */\n  declare dynamicMinWidth: number;\n\n  /**\n   * Use this boolean property in order to split strings that have no white space concept.\n   * this is a cheap way to help with chinese/japanese\n   * @type Boolean\n   * @since 2.6.0\n   */\n  declare splitByGrapheme: boolean;\n\n  static textLayoutProperties = [...XTextbase.textLayoutProperties, 'width'];\n\n  static ownDefaults: Record<string, any> = rectNotesDefaultValues;\n\n  static getDefaults() {\n    return {\n      ...super.getDefaults(),\n\n      ...XRectNotes.ownDefaults,\n    };\n  }\n  constructor(\n    text: string,\n    options: Partial<TClassProperties<XRectNotes>> = {}\n  ) {\n\n    options.createdByName = options.createdByName ?? '';\n    options.fontFamily = options.fontFamily ?? 'Inter';\n    options.fontSize = options.fontSize ?? 12;\n    options.fontWeight = options.fontWeight ?? \"400\";\n    options.lineHeight = 1.2;\n    options.text = options.text ?? '';\n    options.textAlign = options.textAlign ?? 'center';\n    options.editable = options.editable ?? true;\n    options.fixedScaleChange = options.fixedScaleChange ?? false;\n    options.connectors = options.connectors ?? [];\n    options.id = options.id ?? '';\n    options.boardId = options.boardId ?? '';\n    options.backgroundColor = options.backgroundColor ?? '#FCEC8A';\n    options.left = options.left ?? 0;\n    options.locked = options.locked ?? false;\n    options.objType = options.objType ?? 'XRectNotes';\n    options.originX = options.originX ?? 'center';\n    options.originY = options.originY ?? 'center';\n    options.scaleX = options.scaleX ?? 1;\n    options.scaleY = options.scaleY ?? 1;\n    options.selectable = options.selectable ?? true;\n    options.top = options.top ?? 0;\n    options.userId = options.userId ?? '';\n    options.zIndex = options.zIndex ?? Date.now() * 100;\n    options.version = options.version ?? '1.0';\n    options.updatedAt = options.updatedAt ?? Date.now();\n    options.updatedBy = options.updatedBy ?? '';\n    options.updatedByName = options.updatedByName ?? '';\n    options.createdAt = options.createdAt ?? Date.now();\n    options.createdBy = options.createdBy ?? '';\n    options.visible = options.visible ?? true;\n    options.splitByGrapheme = true;\n\n    //fixed default value\n    options.perPixelTargetFind = false;\n    options.height = 138;\n    options.oneLine = false;\n\n    super(text, options);\n    this.maxHeight = 138;\n    Object.assign(this, {\n      controls: {\n        ...createRectNotesDefaultControls(this),\n        // mr: { /* add your desired value here */ },\n      },\n    });\n    // Object.assign(this, options);\n\n    this.splitByGrapheme = true;\n    this.dirty = true;\n    this.objType = 'XRectNotes';\n    // this.initializeEvent();\n  }\n\n  /**\n   * Unlike superclass's version of this function, Textbox does not update\n   * its width.\n   * @private\n   * @override\n   */\n  initDimensions() {\n    if (!this.initialized) {\n      return;\n    }\n    this.isEditing && this.initDelayedCursor();\n    this._clearCache();\n    // clear dynamicMinWidth as it will be different after we re-wrap line\n    this.dynamicMinWidth = 0;\n    // wrap lines\n    this._styleMap = this._generateStyleMap(this._splitText());\n    // if after wrapping, the width is smaller than dynamicMinWidth, change the width and re-wrap\n    if (this.dynamicMinWidth > this.width) {\n      this.set('fontSize', this.fontSize - 2);\n      this._splitTextIntoLines(this.text);\n      return;\n    }\n    if (this.textAlign.indexOf('justify') !== -1) {\n      // once text is measured we need to make space fatter to make justified text.\n      this.enlargeSpaces();\n    }\n    // clear cache and re-calculate height\n    const height = this.calcTextHeight();\n    if (height > this.maxHeight && this.fontSize > 6) {\n      this.set('fontSize', this.fontSize - 2);\n      this._splitTextIntoLines(this.text);\n      return;\n    }\n    if (height > 130 && this.fontSize === 6) {\n      const prenum = 125 / height;\n      const newText = this.text.substring(0, this.text.length * prenum - 5);\n      this.set('text', newText + '...');\n    }\n    this.height = this.maxHeight;\n    return this.height;\n  }\n\n  // /**\n  //  * Generate an object that translates the style object so that it is\n  //  * broken up by visual lines (new lines and automatic wrapping).\n  //  * The original text styles object is broken up by actual lines (new lines only),\n  //  * which is only sufficient for Text / IText\n  //  * @private\n  //  */\n  // _generateStyleMap(textInfo: any) {\n  //   let realLineCount = 0;\n  //   let realLineCharCount = 0;\n  //   let charCount = 0;\n  //   const map: any = {};\n\n  //   for (let i = 0; i < textInfo.graphemeLines.length; i++) {\n  //     if (textInfo.graphemeText[charCount] === '\\n' && i > 0) {\n  //       realLineCharCount = 0;\n  //       charCount++;\n  //       realLineCount++;\n  //     } else if (\n  //       !this.splitByGrapheme &&\n  //       this._reSpaceAndTab.test(textInfo.graphemeText[charCount]) &&\n  //       i > 0\n  //     ) {\n  //       // this case deals with space's that are removed from end of lines when wrapping\n  //       realLineCharCount++;\n  //       charCount++;\n  //     }\n\n  //     map[i] = {\n  //       line: realLineCount,\n  //       offset: realLineCharCount,\n  //     };\n\n  //     charCount += textInfo.graphemeLines[i].length;\n  //     realLineCharCount += textInfo.graphemeLines[i].length;\n  //   }\n\n  //   return map;\n  // }\n\n\n  getObject() {\n    const entityKeys: string[] = EntityKeys;\n    const result: Record<string, any> = {};\n\n    entityKeys.forEach((key) => {\n      if (key in this) {\n        result[key] = (this as any)[key];\n      }\n    });\n\n    return result;\n  }\n\n\n\n\n  /**\n   * Returns true if object has a style property or has it on a specified line\n   * @param {Number} lineIndex\n   * @return {Boolean}\n   */\n  styleHas(property: any, lineIndex: number): boolean {\n    if (this._styleMap && !this.isWrapping) {\n      const map = this._styleMap[lineIndex];\n      if (map) {\n        lineIndex = map.line;\n      }\n    }\n    return super.styleHas(property, lineIndex);\n  }\n\n  /**\n   * probably broken need a fix\n   * Returns the real style line that correspond to the wrapped lineIndex line\n   * Used just to verify if the line does exist or not.\n   * @param {Number} lineIndex\n   * @returns {Boolean} if the line exists or not\n   * @private\n   */\n  _getLineStyle(lineIndex: number): boolean {\n    const map = this._styleMap[lineIndex];\n    return !!this.styles[map.line];\n  }\n\n  /**\n   * Set the line style to an empty object so that is initialized\n   * @param {Number} lineIndex\n   * @param {Object} style\n   * @private\n   */\n  _setLineStyle(lineIndex: number) {\n    const map = this._styleMap[lineIndex];\n    this.styles[map.line] = {};\n  }\n  /*\n   * Override this method to customize word splitting\n   * Use with {@link Textbox#_measureWord}\n   * @param {string} value\n   * @returns {string[]} array of words\n   */\n  wordSplit(value: string): string[] {\n    return value.split(this._wordJoiners);\n  }\n\n  // /**\n  //  * Wraps a line of text using the width of the Textbox and a context.\n  //  * @param {Array} line The grapheme array that represent the line\n  //  * @param {Number} lineIndex\n  //  * @param {Number} desiredWidth width you want to wrap the line to\n  //  * @param {Number} reservedSpace space to remove from wrapping for custom functionalities\n  //  * @returns {Array} Array of line(s) into which the given text is wrapped\n  //  * to.\n  //  */\n\n  // graphemeSplitForRectNotes(textstring: string): string[] {\n  //   const graphemes = [];\n  //   const words = textstring.split(/\\b/);\n  //   for (let i = 0; i < words.length; i++) {\n  //     // 检查单词是否全为拉丁字母，长度不大于16\n  //     if (/^[a-zA-Z]{1,16}$/.test(words[i])) {\n  //       graphemes.push(words[i]);\n  //     } else {\n  //       for (let j = 0; j < words[i].length; j++) {\n  //         graphemes.push(words[i][j]);\n  //       }\n  //     }\n  //   }\n  //   return graphemes;\n  // }\n\n  /**\n   * Detect if the text line is ended with an hard break\n   * text and itext do not have wrapping, return false\n   * @param {Number} lineIndex text to split\n   * @return {Boolean}\n   */\n  isEndOfWrapping(lineIndex: number): boolean {\n    if (!this._styleMap[lineIndex + 1]) {\n      // is last line, return true;\n      return true;\n    }\n    if (this._styleMap[lineIndex + 1].line !== this._styleMap[lineIndex].line) {\n      // this is last line before a line break, return true;\n      return true;\n    }\n    return false;\n  }\n\n  /**\n   * Detect if a line has a linebreak and so we need to account for it when moving\n   * and counting style.\n   * @return Number\n   */\n  missingNewlineOffset(lineIndex: number) {\n    if (this.splitByGrapheme) {\n      return this.isEndOfWrapping(lineIndex) ? 1 : 0;\n    }\n    return 1;\n  }\n\n  /**\n   * Gets lines of text to render in the Textbox. This function calculates\n   * text wrapping on the fly every time it is called.\n   * @param {String} text text to split\n   * @returns {Array} Array of lines in the Textbox.\n   * @override\n   */\n  _splitTextIntoLines(text: string) {\n    const newText = super._splitTextIntoLines(text),\n      graphemeLines = this._wrapText(newText.lines, this.width),\n      lines = new Array(graphemeLines.length);\n    for (let i = 0; i < graphemeLines.length; i++) {\n      lines[i] = graphemeLines[i].join('');\n    }\n    newText.lines = lines;\n    newText.graphemeLines = graphemeLines;\n    return newText;\n  }\n\n  getMinWidth() {\n    return Math.max(this.minWidth, this.dynamicMinWidth);\n  }\n\n  /* caculate cusor positon in the middle of the textbox */\n  getCenteredTop(rectHeight: number) {\n    const textHeight = this.height;\n    return (rectHeight - textHeight) / 2;\n  }\n\n  _render(ctx: CanvasRenderingContext2D) {\n    const path: any = this.path;\n\n    path && !path.isNotVisible() && path._render(ctx);\n    this._setTextStyles(ctx);\n    this._renderTextLinesBackground(ctx);\n    this._renderTextDecoration(ctx, 'underline');\n    this._renderText(ctx);\n    this._renderTextDecoration(ctx, 'overline');\n    this._renderTextDecoration(ctx, 'linethrough');\n  }\n\n  _renderBackground(ctx: CanvasRenderingContext2D) {\n    if (!this.backgroundColor) {\n      return;\n    }\n    const dim = this._getNonTransformedDimensions();\n    ctx.fillStyle = this.backgroundColor;\n\n    // ctx.shadowBlur = 20;\n    // ctx.shadowOffsetX = 2 * this.scaleX * canvas.getZoom();\n    // ctx.shadowOffsetY = 6 * this.scaleY * canvas.getZoom();\n    // ctx.shadowColor = 'rgba(0,0,0,0.1)';\n    // ctx.shadowColor = 'rgba(0,0,0,1)';\n\n    ctx.fillRect(-dim.x / 2, -dim.y / 2, dim.x, dim.y);\n\n    // if there is background color no other shadows\n    // should be casted\n    // this._removeShadow(ctx);\n  }\n  _getTopOffset() {\n    return -this._getTotalLineHeights() / 2;\n  }\n  _getTotalLineHeights() {\n    return this._textLines.reduce(\n      (total, line, index) => total + this.getHeightOfLine(index),\n      0\n    );\n  }\n\n  _getTotalLineHeight() {\n    return this._textLines.reduce(\n      (total, _line, index) => total + this.getHeightOfLine(index),\n      0\n    );\n  }\n\n  _renderText(ctx: CanvasRenderingContext2D) {\n    ctx.shadowOffsetX = ctx.shadowOffsetY = ctx.shadowBlur = 0;\n    ctx.shadowColor = '';\n\n    if (this.paintFirst === 'stroke') {\n      this._renderTextStroke(ctx);\n      this._renderTextFill(ctx);\n    } else {\n      this._renderTextFill(ctx);\n      this._renderTextStroke(ctx);\n    }\n  }\n  drawRoundRectPath(\n    cxt: CanvasRenderingContext2D,\n    width: number,\n    height: number,\n    radius: number\n  ) {\n    cxt.beginPath();\n    //从右下角顺时针绘制，弧度从0到1/2PI\n    cxt.arc(width - radius, height - radius, radius, 0, Math.PI / 2);\n\n    //矩形下边线\n    cxt.lineTo(radius, height);\n\n    //左下角圆弧，弧度从1/2PI到PI\n    cxt.arc(radius, height - radius, radius, Math.PI / 2, Math.PI);\n\n    //矩形左边线\n    cxt.lineTo(0, radius);\n\n    //左上角圆弧，弧度从PI到3/2PI\n    cxt.arc(radius, radius, radius, Math.PI, (Math.PI * 3) / 2);\n\n    //上边线\n    cxt.lineTo(width - radius, 0);\n\n    //右上角圆弧\n    cxt.arc(width - radius, radius, radius, (Math.PI * 3) / 2, Math.PI * 2);\n\n    //右边线\n    cxt.lineTo(width, height - radius);\n    cxt.closePath();\n  }\n}\n\nclassRegistry.setClass(XRectNotes);\nclassRegistry.setSVGClass(XRectNotes, 'XRectNotes');\n","export const shapeList = [\n  {\n    name: 'rect',\n    path: 'M-50,-50L50,-50 50,50 -50,50z',\n    offsetX: 0,\n    offsetY: 0,\n    verticalAlign: 'middle',\n    textAlign: 'center',\n    textMaxHeight: 50,\n  },\n  {\n    name: 'diamond',\n    path: 'm-50,0 l50,-50 50,50 -50,50 -50,-50z',\n    offsetX: 0,\n    offsetY: 0,\n    verticalAlign: 'middle',\n    textAlign: 'center',\n  },\n  {\n    name: 'roundedRect',\n    path: 'M-50,-35 Q-50,-50 -35,-50 L35,-50 Q50,-50 50,-35 L50,35 Q50,50 35,50 L-35,50 Q-50,50 -50,35 Z',\n    offsetX: 0,\n    offsetY: 0,\n    verticalAlign: 'middle',\n    textAlign: 'center',\n  },\n  {\n    name: 'circle',\n    path: 'M-50,0a50,50 0 1,0 100,0a50,50 0 1,0 -100,0',\n    offsetX: 0,\n    offsetY: 0,\n    verticalAlign: 'middle',\n    textAlign: 'center',\n  },\n  {\n    name: 'hexagon',\n    path: 'm-43.476 -25.4636 l 43.476 -24.5364 l 43.7551 25.2641 l 0 50.5157 l -43.7551 24.2202 l -43.7551 -25.2654 l 0 -50.5157 z',\n    offsetX: 0,\n    offsetY: 0,\n    verticalAlign: 'middle',\n    textAlign: 'center',\n  },\n  {\n    name: 'triangle',\n    path: 'm-50,50l50,-100l50,100l-100,0z',\n    offsetX: 0,\n    offsetY: 0,\n    verticalAlign: 'bottom',\n    textAlign: 'center',\n  },\n  {\n    name: 'parallelogramIcon',\n    path: 'm-50,50l20,-100l80,0l-20,100l-80,0z',\n    offsetX: 0,\n    offsetY: 0,\n    verticalAlign: 'middle',\n    textAlign: 'center',\n  },\n  {\n    name: 'star',\n    path: 'm-50,-10l38,0l12,-38l12,38l38,0l-30,23l12,38l-30,-24l-30,24l12,-38l-30,-23z',\n    offsetX: 0,\n    offsetY: 0,\n    verticalAlign: 'middle',\n    textAlign: 'center',\n  },\n  {\n    name: 'cross',\n    path: 'm-50,-15l33,0l0,-33l34,0l0,33l33,0l0,34l-33,0l0,33l-34,0l0,-33l-33,0l0,-34z',\n    offsetX: 0,\n    offsetY: 0,\n    verticalAlign: 'middle',\n    textAlign: 'center',\n  },\n  {\n    name: 'leftsideRightTriangle',\n    path: 'm50,50l-100,0l0,-100l100,100z',\n    offsetX: 0,\n    offsetY: 0,\n    verticalAlign: 'bottom',\n    textAlign: 'left',\n    textMaxHeight: 50,\n  },\n  {\n    name: 'rightsideRightTriangle',\n    path: 'm-50,50l100,0l0,-100l-100,100z',\n    offsetX: 0,\n    offsetY: 0,\n    verticalAlign: 'bottom',\n    textAlign: 'right',\n    textMaxHeight: 50,\n  },\n  {\n    name: 'topsideSemicircleCircle',\n    path: 'm50,25l-100,-0.00205c3.5,-27.5 25,-48 50,-48c25,0 46.5,20.5 50,48z',\n    offsetX: 0,\n    offsetY: 0,\n    verticalAlign: 'middle',\n    textAlign: 'center',\n  },\n  {\n    name: 'topLeftQuarterCircle',\n    path: 'm-50,50c7,-55 47,-96 97,-97l0,65l0,32l-97,-0.00392z',\n    offsetX: 0,\n    offsetY: 0,\n    verticalAlign: 'middle',\n    textAlign: 'center',\n  },\n  {\n    name: 'constellationRect',\n    path: 'm20,0l25,23l0,24l-100,0l0,-100l100,0l0,23l-25,24l-0.15,0.14l0.15,0.14l-0.00001,0z',\n    offsetX: 0,\n    offsetY: 0,\n    verticalAlign: 'middle',\n    textAlign: 'center',\n  },\n  {\n    name: 'constellationRound',\n    path: 'm20,1c0.002,0.002 0.004,0.004 0.01,0.001c0.007,0.01 0.015,0.02 0.02,0.03l0,0l0.001,0.0004l0.005,0.003l0.04,0.03l0.3,0.2l2.5,2l20,15c-7,15 -23,26 -42,26c-25,0 -45,-20 -45,-45c0,-25 20,-45 45,-45c19,0 35,10 42,25c-11,9 -17,14 -20,16c-1.5,1 -2,1.5 -2.5,2c-0.2,0.15 -0.3,0.2 -0.35,0.3c-0.02,0.02 -0.04,0.04 -0.05,0.06c-0.006,0.01 -0.02,0.03 -0.03,0.06c-0.005,0.015 -0.02,0.04 -0.03,0.07c-0.005,0.015 -0.01,0.04 -0.007,0.07c0.002,0.02 0.007,0.04 0.015,0.06c0.003,0.006 0.006,0.012 0.009,0.018c0.003,0.004 0.006,0.009 0.009,0.014c0.003,0.005 0.009,0.01 0.01,0.015c0.004,0.005 0.01,0.015 0.012,0.017z',\n    offsetX: 0,\n    offsetY: 0,\n    verticalAlign: 'middle',\n    textAlign: 'center',\n  },\n];\nexport type shapeType =\n  | 'rect'\n  | 'diamond'\n  | 'roundedRect'\n  | 'circle'\n  | 'hexagon'\n  | 'triangle'\n  | 'parallelogramIcon'\n  | 'star'\n  | 'cross'\n  | 'leftsideRightTriangle'\n  | 'rightsideRightTriangle'\n  | 'topsideSemicircleCircle'\n  | 'topLeftQuarterCircle'\n  | 'constellationRect'\n  | 'constellationRound';\n","import { TOriginX, TOriginY } from './widget.entity.base';\nimport   { WidgetTextboxInterface,Connector } from './widget.entity.textbox';\nimport { WidgetType } from './widget.type';\n\nexport   interface WidgetShapeNotesInterface extends WidgetTextboxInterface {\n  shapeName: shapeType;\n}\n\nexport type shapeType =\n  | 'rect'\n  | 'diamond'\n  | 'roundedRect'\n  | 'circle'\n  | 'hexagon'\n  | 'triangle'\n  | 'parallelogramIcon'\n  | 'star'\n  | 'cross'\n  | 'leftsideRightTriangle'\n  | 'rightsideRightTriangle'\n  | 'topsideSemicircleCircle'\n  | 'topLeftQuarterCircle'\n  | 'constellationRect'\n  | 'constellationRound';\n\nexport class WidgetShapeNotesClass implements WidgetShapeNotesInterface {\n  fill: string=\"#eeeeee\";\n  updatedBy: string = \"\";\n  updatedByName: string = \"\";\n  createdByName: string=\"\";\n  shapeName: shapeType = 'rect';\n  fontFamily: string = 'Inter';\n  fontSize: number = 16;\n  fontWeight: string = '400';\n  lineHeight: number = 1.2;\n  text: string = '';\n  textAlign: string = 'center';\n  editable: boolean = true;\n  maxHeight: number = 100;\n  fixedScaleChange: boolean = false;\n  connectors: Connector[] = [];\n  id: string = '';\n  boardId: string = '';\n  backgroundColor: string = '#eeeeee';\n  width: number = 200;\n  height: number = 200;\n  left: number = 0;\n  locked: boolean = false;\n  objType: WidgetType = 'XShapeNotes';\n  originX: TOriginX = 'center';\n  originY: TOriginY = 'center';\n  scaleX: number = 1;\n  scaleY: number = 1;\n  selectable: boolean = true;\n  top: number = 0;\n  zIndex: number = Date.now() * 100;\n  version: string = '1.0';\n  updatedAt: number = Date.now();\n  createdAt: number = Date.now();\n  createdBy: string = '';\n  visible: boolean = true;\n}\n\nexport const EntityKeys = Object.keys(new WidgetShapeNotesClass()) as (keyof WidgetShapeNotesInterface)[];\n","import { classRegistry } from '../../../ClassRegistry';\nimport { getFabricDocument } from '../../../env';\nimport { TClassProperties, TFiller } from '../../../typedefs';\nimport { XTextbase } from '../XTextbase';\n\nimport { shapeList } from './types';\nimport { shapeType } from './types';\n\nimport { WidgetShapeNotesInterface, EntityKeys } from '../type/widget.entity.shapenote';\nimport { createShapeNotesDefaultControls } from '../../../controls/X_commonControls';\nimport { WidgetType } from '../type/widget.type';\n\n\n\nexport type shapeInfo = {\n  name: shapeType;\n  path: string;\n  offsetX: number;\n  offsetY: number;\n  verticalAlign: 'middle' | 'top' | 'bottom';\n  textAlign: 'center' | 'left' | 'right';\n  textMaxHeight?: number;\n  textMaxWidth?: number;\n};\n\nfunction getShapeInfo(shape: string): shapeInfo | null {\n  const shapeObj = shapeList.find((item) => item.name === shape);\n\n  if (!shapeObj) {\n    return shapeList[0] as shapeInfo;\n  } else {\n    return shapeObj as shapeInfo; // Update the type of shapeObj to shapeInfo\n  }\n}\n\nexport const XShapeNotesDefaultValues: Partial<TClassProperties<XShapeNotes>> =\n{\n\n};\n\n\nexport class XShapeNotes extends XTextbase implements WidgetShapeNotesInterface {\n  static type: WidgetType = 'XShapeNotes';\n  static objType: WidgetType = 'XShapeNotes';\n\n  bgShape: shapeInfo | null;\n  verticalAlign: string;\n\n  minHeight: number;\n  shapeName: shapeType;\n\n\n  constructor(text: string, options: Partial<WidgetShapeNotesInterface> & { type: string }) {\n\n\n    super(text, options);\n    this.bgShape = options.shapeName ? getShapeInfo(options.shapeName) : null;\n    // this.width = (options.width || 200) * (options.scaleX || 1);\n    // this.height = (options.height || 200) * (options.scaleY || 1);\n    this.scaleX = 1;\n    this.scaleY = 1;\n    this.id = options.id || '';\n    this.verticalAlign = this.bgShape?.verticalAlign || 'middle';\n    this.textAlign = this.bgShape?.textAlign || 'center';\n    this.shapeName = options.shapeName || 'rect';\n    this.fontSize = options.fontSize || 14;\n    this.fontFamily = options.fontFamily || 'Inter';\n    this.fontWeight = options.fontWeight || 'normal';\n    this.lineHeight = 1.2;\n    this.text = text;\n    this.minWidth = 20;\n    this.minHeight = 20;\n    this.dynamicMinWidth = 2;\n    this.lockScalingFlip = true;\n    this.noScaleCache = false;\n    this._wordJoiners = /[ \\t\\r]/;\n    this.splitByGrapheme = true;\n    this.objType = 'XShapeNotes';\n    this.textAlign = 'center';\n    this.centeredScaling = false;\n    this.cornerColor = 'white';\n    this.cornerStrokeColor = 'gray';\n    this.cornerSize = 10;\n    this.cornerStyle = 'circle';\n    this.transparentCorners = false;\n    this.verticalAlign = 'middle';\n\n\n\n    this.resetSplitByGrapheme();\n    Object.assign(this, {\n      controls: { ...createShapeNotesDefaultControls(this) },\n    });\n\n    this.on('scaling', this.handleScaling);\n    this.on('modified', this.handleModified);\n    this.on('changed', this.handleModified);\n  }\n\n\n\n  getObject() {\n    const entityKeys: string[] = EntityKeys;\n    const result: Record<string, any> = {};\n\n    entityKeys.forEach((key) => {\n      if (key in this) {\n        result[key] = (this as any)[key];\n      }\n    });\n\n    return result;\n  }\n\n\n  handleModified() {\n    this.canvas!.uniformScaling = false;\n    this.resetSplitByGrapheme();\n    this.canvas!.renderAll();\n  }\n\n  resetSplitByGrapheme() {\n    const chineseRegex = /[\\u4e00-\\u9fa5]/; // Regex to match Chinese characters\n    const text = this.text;\n    const includesChinese = chineseRegex.test(text);\n    this.splitByGrapheme = includesChinese;\n  }\n\n  static ownDefaults: Record<string, any> = XShapeNotesDefaultValues;\n\n  static getDefaults() {\n    return {\n      ...super.getDefaults(),\n      ...XShapeNotes.ownDefaults,\n    };\n  }\n\n  handleScaling(event: any) {\n    this.canvas!.uniformScaling = false;\n\n    const newScaleX = this.get('scaleX');\n    const newScaleY = this.get('scaleY');\n\n    this.width *= newScaleX;\n    this.height *= newScaleY;\n\n    if (this.width < this.minWidth) {\n      this.width = this.minWidth;\n    }\n    if (this.height < this.minHeight) {\n      this.height = this.minHeight;\n    }\n\n    this.set({\n      scaleX: 1,\n      scaleY: 1,\n    });\n\n    this.initDimensions();\n    this.setCoords();\n    this.dirty = true;\n    this.canvas?.renderAll();\n  }\n\n  _renderBackground(ctx: any) {\n    if (!this.backgroundColor) {\n      return;\n    }\n    const dim = this._getNonTransformedDimensions();\n    ctx.fillStyle = this.backgroundColor;\n\n    ctx.save();\n\n    const svgPath = new Path2D(this.bgShape?.path || '');\n    const m = getFabricDocument()\n      .createElementNS('http://www.w3.org/2000/svg', 'svg')\n      .createSVGMatrix();\n    m.a = this.width / 100;\n    m.b = 0;\n    m.c = 0;\n    m.d = this.height / 100;\n    m.e = 0;\n    m.f = 0;\n    const path = new Path2D();\n    path.addPath(svgPath, m);\n    ctx.strokeStyle = this.stroke;\n    ctx.stroke(path);\n    ctx.fillStyle = this.backgroundColor;\n    ctx.fill(path);\n    ctx.restore();\n  }\n\n  _getTopOffset() {\n    switch (this.verticalAlign) {\n      case 'middle':\n        return -this._getTotalLineHeight() / 2;\n      case 'bottom':\n        return this.height / 2 - this._getTotalLineHeight();\n      default:\n        return -this.height / 2;\n    }\n  }\n\n  _getTotalLineHeight() {\n    return this._textLines.reduce(\n      (total, _line, index) => total + this.getHeightOfLine(index),\n      0\n    );\n  }\n\n  _renderTextCommon(ctx: any, method: any) {\n    ctx.save();\n    let lineHeights = 0;\n    const left = this._getLeftOffset() + this.bgShape?.offsetX!;\n    const top = this._getTopOffset() + this.bgShape?.offsetY!;\n    const offsets = this._applyPatternGradientTransform(\n      ctx,\n      ((method === 'fillText' ? this.fill : this.stroke) as TFiller) ||\n      this.fill!\n    );\n\n    for (let i = 0, len = this._textLines.length; i < len; i++) {\n      const heightOfLine = this.getHeightOfLine(i);\n      const maxHeight = heightOfLine / this.lineHeight;\n      const leftOffset = this._getLineLeftOffset(i);\n      this._renderTextLine(\n        method,\n        ctx,\n        this._textLines[i],\n        left + leftOffset - offsets.offsetX,\n        top + lineHeights + maxHeight - offsets.offsetY,\n        i\n      );\n      lineHeights += heightOfLine;\n    }\n    ctx.restore();\n  }\n\n  initDimensions() {\n    if (!this.initialized) {\n      return;\n    }\n    this.isEditing && this.initDelayedCursor();\n    this._clearCache();\n    this.dynamicMinWidth = 0;\n    this._styleMap = this._generateStyleMap(this._splitText());\n    if (this.dynamicMinWidth > this.width) {\n      this.set('fontSize', this.fontSize - 2);\n      this._splitTextIntoLines(this.text);\n      return;\n    }\n    if (this.textAlign.indexOf('justify') !== -1) {\n      this.enlargeSpaces();\n    }\n    const height = this.calcTextHeight();\n    if (height > this.height && this.fontSize > 6) {\n      this.set('fontSize', this.fontSize - 2);\n      this._splitTextIntoLines(this.text);\n      return;\n    }\n\n    return this.height;\n  }\n\n  _getTotalLineHeights() {\n    return this._textLines.reduce(\n      (total, _line, index) => total + this.getHeightOfLine(index),\n      0\n    );\n  }\n\n  graphemeSplitForRectNotes(textstring: string): string[] {\n    const graphemes = [];\n    const words = textstring.split(/\\b/);\n    for (let i = 0; i < words.length; i++) {\n      if (/^[a-zA-Z]{1,16}$/.test(words[i])) {\n        graphemes.push(words[i]);\n      } else {\n        for (let j = 0; j < words[i].length; j++) {\n          graphemes.push(words[i][j]);\n        }\n      }\n    }\n    return graphemes;\n  }\n}\n\nclassRegistry.setClass(XShapeNotes);\n","import { TClassProperties, TOriginX, TOriginY } from '../../typedefs';\nimport { IText } from '../IText/IText';\nimport { classRegistry } from '../../ClassRegistry';\nimport { createTextboxDefaultControls } from '../../controls/X_commonControls';\nimport { XTextbase } from './XTextbase';\nimport { isTransformCentered, getLocalPoint } from '../../controls/util';\nimport { EntityKeys } from './type/widget.entity.textbox';\nimport { WidgetType } from './type/widget.type';\nimport { Point } from '../../Point';\nimport { WidgetTextInterface } from './type/widget.entity.text';\n\n// Default values for the textbox\nexport const textboxDefaultValues: Partial<TClassProperties<XTextbase>> = {\n    minWidth: 20,\n    dynamicMinWidth: 2,\n    splitByGrapheme: false,\n    cornerColor: 'white',\n    cornerSize: 10,\n    cornerStyle: 'circle',\n    transparentCorners: false,\n    cornerStrokeColor: 'gray',\n    connectors: [],\n};\n\n// Connector class definition\nclass Connector {\n    connectorId: string;\n    connectorType: string;\n    point: Point;\n}\n\nexport const XTextboxProps: Partial<TClassProperties<XTextbase>> = {};\n\n/**\n * Textbox class, based on IText, allows the user to resize the text rectangle\n * and wraps lines automatically. Textboxes have their Y scaling locked, the\n * user can only change width. Height is adjusted automatically based on the\n * wrapping of lines.\n */\nexport class XTextbox extends XTextbase implements WidgetTextInterface {\n    // Property declarations\n    declare minWidth: number;\n    declare tempTop: number;\n    declare hasNoText: boolean;\n    static objType: WidgetType = 'XTextbox';\n    static type: WidgetType = 'XTextbox';\n\n    declare dynamicMinWidth: number;\n    declare oneLine: boolean;\n    declare fromCopy: boolean;\n    declare originX: TOriginX;\n    declare originY: TOriginY;\n    declare connectors: Connector[];\n\n    /**\n     * Use this boolean property in order to split strings that have no white space concept.\n     * This is a cheap way to help with Chinese/Japanese.\n     * @type Boolean\n     * @since 2.6.0\n     */\n    declare splitByGrapheme: boolean;\n\n    static textLayoutProperties = [...IText.textLayoutProperties, 'width'];\n\n    static ownDefaults: Record<string, any> = textboxDefaultValues;\n\n    /**\n     * Override the getDefaults method to set default origin to center\n     */\n    static getDefaults() {\n        return {\n            ...super.getDefaults(),\n            controls: createTextboxDefaultControls(),\n            originX: 'center', // Default originX\n            originY: 'center', // Default originY\n            ...XTextbox.ownDefaults,\n        };\n    }\n\n\n    /**\n     * Constructor to initialize the textbox with default origin\n     * @param text - The initial text content\n     * @param options - Configuration options\n     */\n    constructor(text: string, options: any) {\n        super(text, options);\n\n        // Initialize dimensions\n        this.initDimensions();\n\n        // Remove height from options to manage it dynamically\n        delete options.height;\n        Object.assign(this, options);\n        this.objType = 'XTextbox';\n\n        this.dirty = true;\n\n        // Lock scaling flip and rotation to maintain aspect\n        this.lockScalingFlip = true;\n        this.lockRotation = true;\n        // Setup custom resize controls\n        this.setupCustomResizeControls();\n        // Bind event listeners for editing\n        this.on('editing:entered', this.onEditingEntered.bind(this));\n        this.on('editing:exited', this.onEditingExited.bind(this));\n    }\n    /**\n     * Set up custom resize controls that use our implementation\n     */\n    setupCustomResizeControls() {\n        // Override the action handler for 'mr' (middle right) control\n        if (this.controls && this.controls.mr) {\n            this.controls.mr.actionHandler = this.handleWidthChange.bind(this);\n        }\n\n        // Override the action handler for 'ml' (middle left) control\n        if (this.controls && this.controls.ml) {\n            this.controls.ml.actionHandler = this.handleWidthChange.bind(this);\n        }\n\n\n    }\n\n    // reserve for external interface\n    handleWidthChange2(eventData: any, transform: any, x: number, y: number) {\n\n    }\n    /**\n     * Event handler for entering edit mode\n     */\n    private onEditingEntered() {\n        const canvas = this.canvas;\n        if (!canvas) return;\n\n        // Calculate the current top-left corner in canvas coordinates\n        const topLeft = this.getBoundingRect();\n\n        // Store the current center position\n        const center = this.getCenterPoint();\n\n        // Change origin to 'left' and 'top'\n        this.originX = 'left';\n        this.originY = 'top';\n\n        // Update the position to keep the top-left corner fixed\n        this.set({\n            left: topLeft.left,\n            top: topLeft.top,\n        });\n\n        this.setCoords();\n    }\n\n    /**\n     * Event handler for exiting edit mode\n     */\n    private onEditingExited() {\n        const canvas = this.canvas;\n        if (!canvas) return;\n\n        // Calculate the current top-left corner in canvas coordinates\n        const topLeft = this.getBoundingRect();\n\n        // Change origin back to 'center'\n        this.originX = 'center';\n        this.originY = 'center';\n\n        // Calculate the new center based on top-left to keep position fixed\n        this.set({\n            left: topLeft.left + this.width / 2,\n            top: topLeft.top + this.height / 2,\n        });\n\n        this.setCoords();\n    }\n\n    /**\n     * Retrieves the object's properties\n     * @returns A record of the object's properties\n     */\n    getObject() {\n        const entityKeys: string[] = EntityKeys;\n        const result: Record<string, any> = {};\n\n        entityKeys.forEach((key) => {\n            if (key in this) {\n                result[key] = (this as any)[key];\n            }\n        });\n\n        return result;\n    }\n\n    /**\n     * Override the initDimensions method to ensure height increases downward\n     */\n    initDimensions() {\n        if (!this.initialized) {\n            return;\n        }\n        if (this.isEditing) {\n            this.initDelayedCursor();\n        }\n        this._clearCache();\n        // Clear dynamicMinWidth as it will be different after we re-wrap line\n        this.dynamicMinWidth = 0;\n\n        // Check if text contains Chinese characters\n        if (/[\\u3400-\\u9FBF]/.test(this.text)) {\n            this.splitByGrapheme = true;\n        }\n\n        // Wrap lines\n        this._styleMap = this._generateStyleMap(this._splitText());\n        // If after wrapping, the width is smaller than dynamicMinWidth, change the width and re-wrap\n        if (this.dynamicMinWidth > this.width) {\n            this.set('width', this.dynamicMinWidth);\n        }\n        if (this.textAlign.indexOf('justify') !== -1) {\n            // Once text is measured we need to make space fatter to make justified text.\n            this.enlargeSpaces();\n        }\n        // Calculate height based on wrapped text\n        const newHeight = this.calcTextHeight();\n        if (newHeight !== this.height) {\n            this.set('height', newHeight);\n            this.setCoords(); // Update coordinates after height change\n        }\n    }\n\n\n    /**\n     * Handler for width changes that maintains top-left position\n     */\n    handleWidthChange(eventData: any, transform: any, x: number, y: number) {\n        // Store original top-left position\n        const oldBoundingRect = this.getBoundingRect();\n\n        // Calculate new width based on mouse position\n        const localPoint = getLocalPoint(\n            transform,\n            transform.originX,\n            transform.originY,\n            x,\n            y\n        );\n\n        const strokePadding =\n            this.strokeWidth / (this.strokeUniform ? this.scaleX : 1);\n        const multiplier = isTransformCentered(transform) ? 2 : 1;\n        const oldWidth = this.width;\n\n        // Calculate and set new width\n        const newWidth =\n            Math.abs((localPoint.x * multiplier) / this.scaleX) - strokePadding;\n\n        // Record original left/top before width change\n        const originalLeft = this.left;\n        const originalTop = this.top;\n\n        // Set new width and update dimensions\n        this.set('width', Math.max(newWidth, this.getMinWidth()));\n\n        // Recalculate text dimensions and height\n        this.initDimensions();\n        this.set('dirty', true);\n\n        // After dimensions update, get new bounding rectangle\n        const newBoundingRect = this.getBoundingRect();\n\n        // Calculate position adjustment to keep top-left fixed\n        const dx = newBoundingRect.left - oldBoundingRect.left;\n        const dy = newBoundingRect.top - oldBoundingRect.top;\n\n        // Apply position adjustment\n        this.set({\n            left: originalLeft - dx,\n            top: originalTop - dy\n        });\n\n        // Update coordinates\n        this.setCoords();\n        this.handleWidthChange2(eventData, transform, x, y);\n\n        return oldWidth !== this.width;\n    }\n}\n\n// Register the XTextbox class with the class registry\nclassRegistry.setClass(XTextbox);\n// classRegistry.getSVGClass(Textbox); // Uncomment if SVG class registration is needed\n","import { FileObject } from './file';\nimport   {WidgetBaseInterface, TOriginX, TOriginY } from './widget.entity.base';\nimport { WidgetType } from './widget.type';\n\nexport   interface WidgetImageInterface extends WidgetBaseInterface {\n  cropX: number;\n  cropY: number;\n  cropWidth: number;\n  cropHeight: number;\n  previewImage: FileObject|null;\n  imageSrc: FileObject|null;\n}\n\nexport class WidgetImageClass implements WidgetImageInterface {\n  updatedBy: string = \"\";\n  updatedByName: string = \"\";\n   createdByName: string=\"\";\n  markdownText: string = '';\n  cropX: number = 0;\n  cropY: number = 0;\n  cropWidth: number = 0;\n  cropHeight: number = 0;\n  previewImage: FileObject |null =null;\n  imageSrc: FileObject  |null =null;\n  id: string = '';\n  boardId: string = '';\n  backgroundColor: string = 'transparent';\n  width: number = 0;\n  height: number = 0;\n  left: number = 0;\n  locked: boolean = false;\n  objType: WidgetType = \"XImage\"; // Assuming WidgetType.Default is a valid enum value\n  originX: TOriginX = 'center'; // Assuming 'left' is a valid TOriginX value\n  originY: TOriginY = 'center'; // Assuming 'top' is a valid TOriginY value\n  scaleX: number = 1;\n  scaleY: number = 1;\n  selectable: boolean = true;\n  top: number = 0;\n  userId: string = '';\n  zIndex: number = Date.now() *100;\n  version: string = '';\n  updatedAt: number = Date.now();\n   createdAt: number = Date.now();\n  createdBy: string = '';\n  visible: boolean = true;\n}\n\nexport const EntityKeys = Object.keys(new WidgetImageClass()) as (keyof WidgetImageInterface)[];","import type { BaseFilter } from '../../filters/BaseFilter';\nimport { SHARED_ATTRIBUTES } from '../../parser/attributes';\nimport { TSize } from '../../typedefs';\nimport { findScaleToCover, findScaleToFit } from '../../util/misc/findScaleTo';\nimport { parsePreserveAspectRatioAttribute } from '../../util/misc/svgParsing';\nimport { classRegistry } from '../../ClassRegistry';\nimport { TOptions } from '../../typedefs';\nimport type { FabricObjectProps, SerializedObjectProps } from '../Object/types';\nimport type { ObjectEvents } from '../../EventTypeDefs';\n// @todo Would be nice to have filtering code not imported directly.\n\nimport { WidgetImageInterface, EntityKeys } from './type/widget.entity.image';\nimport { FabricImage } from '../Image';\nimport { WidgetType } from './type/widget.type';\nimport { FileObject } from './type/file';\n\n\nexport type ImageSource =\n  | HTMLImageElement\n  | HTMLVideoElement\n  | HTMLCanvasElement;\n\ninterface UniqueImageProps {\n  srcFromAttribute: boolean;\n  minimumScaleTrigger: number;\n  cropX: number;\n  cropY: number;\n  imageSmoothing: boolean;\n  crossOrigin: string | null;\n  filters: BaseFilter[];\n  resizeFilter?: BaseFilter;\n}\n\nexport const imageDefaultValues: Partial<UniqueImageProps> &\n  Partial<FabricObjectProps> = {\n  strokeWidth: 0,\n  srcFromAttribute: false,\n  minimumScaleTrigger: 0.5,\n  cropX: 0,\n  cropY: 0,\n  imageSmoothing: true,\n  crossOrigin: 'anonymous',\n  originX: 'center',\n  originY: 'center',\n};\n\nexport interface SerializedImageProps extends SerializedObjectProps {\n  src: string;\n  crossOrigin: string | null;\n  filters: any[];\n  resizeFilter?: any;\n  cropX: number;\n  cropY: number;\n}\n\nexport interface ImageProps extends FabricObjectProps, UniqueImageProps { }\n\nconst IMAGE_PROPS = ['cropX', 'cropY'] as const;\n\n/**\n * @tutorial {@link http://fabricjs.com/fabric-intro-part-1#images}\n */\nexport class XImage<\n  Props extends TOptions<ImageProps> = Partial<ImageProps>,\n  SProps extends SerializedImageProps = SerializedImageProps,\n  EventSpec extends ObjectEvents = ObjectEvents\n> extends FabricImage implements WidgetImageInterface {\n\n  cropWidth: number;\n  cropHeight: number;\n  previewImage: FileObject;\n  imageSrc: FileObject;\n  version: string;\n  updatedAt: number;\n\n  createdAt: number;\n  createdBy: string;\n  boardId: string;\n  objType: WidgetType;\n  userId: string;\n  zIndex: number;\n  /* boardx cusotm function */\n\n  static type: WidgetType = 'XImage';\n  static objType: WidgetType = 'XImage';\n\n  constructor(image: any, options: any) {\n\n    options.objType = 'XImage';\n    options.cornerColor = 'white';\n    options.cornerSize = 10;\n    options.cornerStyle = 'circle';\n    options.transparentCorners = false;\n    options.cornerStrokeColor = 'gray';\n    super(image, options);\n    Object.assign(this, options);\n\n  }\n  updatedBy: string;\n  updatedByName: string;\n  markdownText: string;\n\n  createdByName: string;\n\n\n  getObject() {\n    const entityKeys: string[] = EntityKeys;\n    const result: Record<string, any> = {};\n\n    entityKeys.forEach((key) => {\n      if (key in this) {\n        result[key] = (this as any)[key];\n      }\n    });\n\n    return result;\n  }\n\n\n  /**\n   * Renders the image on the canvas context, correctly handling scaling and cropping.\n   * @private\n   * @param {CanvasRenderingContext2D} ctx Context to render on\n   */\n  _renderFill(ctx: CanvasRenderingContext2D) {\n    const elementToDraw = this._element;\n    if (!elementToDraw) {\n      return;\n    }\n\n    // Get the object's dimensions\n    const w = this.width,\n      h = this.height,\n      // Crop values cannot be less than 0\n      cropX = Math.max(this.cropX, 0),\n      cropY = Math.max(this.cropY, 0),\n      // Get the natural dimensions of the image element\n      elWidth =\n        (elementToDraw as HTMLImageElement).naturalWidth || elementToDraw.width,\n      elHeight =\n        (elementToDraw as HTMLImageElement).naturalHeight || elementToDraw.height,\n      // Calculate source width and height, ensuring we don't exceed image bounds\n      sX = cropX,\n      sY = cropY,\n      sW = Math.min(w, elWidth - cropX),\n      sH = Math.min(h, elHeight - cropY),\n      // Destination coordinates (centered)\n      x = -w / 2,\n      y = -h / 2,\n      destW = w,\n      destH = h;\n\n    // Draw the image onto the canvas context\n    ctx.drawImage(\n      elementToDraw,\n      sX,\n      sY,\n      sW,\n      sH,\n      x,\n      y,\n      destW,\n      destH\n    );\n  }\n\n\n  /**\n   * Decide if the object should cache or not. Create its own cache level\n   * needsItsOwnCache should be used when the object drawing method requires\n   * a cache step. None of the fabric classes requires it.\n   * Generally you do not cache objects in groups because the group outside is cached.\n   * This is the special image version where we would like to avoid caching where possible.\n   * Essentially images do not benefit from caching. They may require caching, and in that\n   * case we do it. Also caching an image usually ends in a loss of details.\n   * A full performance audit should be done.\n   * @return {Boolean}\n   */\n  shouldCache() {\n    return this.needsItsOwnCache();\n  }\n\n  /**\n   * needed to check if image needs resize\n   * @private\n   */\n  _needsResize() {\n    const scale = this.getTotalObjectScaling();\n    return scale.x !== this._lastScaleX || scale.y !== this._lastScaleY;\n  }\n\n  /**\n   * @private\n   * @deprecated unused\n   */\n  _resetWidthHeight() {\n    this.set(this.getOriginalSize());\n  }\n\n  /**\n   * @private\n   * Set the width and the height of the image object, using the element or the\n   * options.\n   */\n  _setWidthHeight({ width, height }: Partial<TSize> = {}) {\n    const size = this.getOriginalSize();\n    this.width = width || size.width;\n    this.height = height || size.height;\n  }\n\n  /**\n   * Calculate offset for center and scale factor for the image in order to respect\n   * the preserveAspectRatio attribute\n   * @private\n   */\n  parsePreserveAspectRatioAttribute() {\n    const pAR = parsePreserveAspectRatioAttribute(\n      this.preserveAspectRatio || ''\n    ),\n      pWidth = this.width,\n      pHeight = this.height,\n      parsedAttributes = { width: pWidth, height: pHeight };\n    let rWidth = this._element.width,\n      rHeight = this._element.height,\n      scaleX = 1,\n      scaleY = 1,\n      offsetLeft = 0,\n      offsetTop = 0,\n      cropX = 0,\n      cropY = 0,\n      offset;\n\n    if (pAR && (pAR.alignX !== 'none' || pAR.alignY !== 'none')) {\n      if (pAR.meetOrSlice === 'meet') {\n        scaleX = scaleY = findScaleToFit(this._element, parsedAttributes);\n        offset = (pWidth - rWidth * scaleX) / 2;\n        if (pAR.alignX === 'Min') {\n          offsetLeft = -offset;\n        }\n        if (pAR.alignX === 'Max') {\n          offsetLeft = offset;\n        }\n        offset = (pHeight - rHeight * scaleY) / 2;\n        if (pAR.alignY === 'Min') {\n          offsetTop = -offset;\n        }\n        if (pAR.alignY === 'Max') {\n          offsetTop = offset;\n        }\n      }\n      if (pAR.meetOrSlice === 'slice') {\n        scaleX = scaleY = findScaleToCover(this._element, parsedAttributes);\n        offset = rWidth - pWidth / scaleX;\n        if (pAR.alignX === 'Mid') {\n          cropX = offset / 2;\n        }\n        if (pAR.alignX === 'Max') {\n          cropX = offset;\n        }\n        offset = rHeight - pHeight / scaleY;\n        if (pAR.alignY === 'Mid') {\n          cropY = offset / 2;\n        }\n        if (pAR.alignY === 'Max') {\n          cropY = offset;\n        }\n        rWidth = pWidth / scaleX;\n        rHeight = pHeight / scaleY;\n      }\n    } else {\n      scaleX = pWidth / rWidth;\n      scaleY = pHeight / rHeight;\n    }\n    return {\n      width: rWidth,\n      height: rHeight,\n      scaleX,\n      scaleY,\n      offsetLeft,\n      offsetTop,\n      cropX,\n      cropY,\n    };\n  }\n\n  /**\n   * Default CSS class name for canvas\n   * @static\n   * @type String\n   * @default\n   */\n  static CSS_CANVAS = 'canvas-img';\n\n  /**\n   * List of attribute names to account for when parsing SVG element (used by {@link Image.fromElement})\n   * @static\n   * @see {@link http://www.w3.org/TR/SVG/struct.html#ImageElement}\n   */\n  static ATTRIBUTE_NAMES = [\n    ...SHARED_ATTRIBUTES,\n    'x',\n    'y',\n    'width',\n    'height',\n    'preserveAspectRatio',\n    'xlink:href',\n    'crossOrigin',\n    'image-rendering',\n  ];\n\n  _stopEvent(e: any) {\n    if (e.preventDefault) e.preventDefault();\n    if (e.stopPropagation) e.stopPropagation();\n  }\n\n  cloneWidget() {\n    return this.toObject();\n  }\n}\n\nclassRegistry.setClass(XImage);\nclassRegistry.setSVGClass(XImage);\n","//@ts-nocheck\nimport { Point } from '../Point';\nimport { Control } from './Control';\nimport type { TMat2D } from '../typedefs';\nimport type { Path } from '../shapes/Path';\nimport { multiplyTransformMatrices } from '../util/misc/matrix';\nimport type {\n  TModificationEvents,\n  TPointerEvent,\n  Transform,\n} from '../EventTypeDefs';\nimport { sendPointToPlane } from '../util/misc/planeChange';\nimport type { TSimpleParseCommandType } from '../util/path/typedefs';\nimport type { ControlRenderingStyleOverride } from './controlRendering';\nimport { fireEvent } from './fireEvent';\nimport { commonEventInfo } from './util';\n\nconst ACTION_NAME: TModificationEvents = 'modifyPath' as const;\n\ntype TTransformAnchor = Transform;\n\nexport type PathPointControlStyle = {\n  controlFill?: string;\n  controlStroke?: string;\n  connectionDashArray?: number[];\n};\n\nexport const calcPathPointPosition = (\n  pathObject: Path,\n  commandIndex: number,\n  pointIndex: number\n) => {\n  const { path, pathOffset } = pathObject;\n  const command = path[commandIndex];\n  return new Point(\n    (command[pointIndex] as number) - pathOffset.x,\n    (command[pointIndex + 1] as number) - pathOffset.y\n  ).transform(\n    multiplyTransformMatrices(\n      pathObject.getViewportTransform(),\n      pathObject.calcTransformMatrix()\n    )\n  );\n};\n\nexport const movePathPoint = (\n  pathObject: Path,\n  x: number,\n  y: number,\n  commandIndex: number,\n  pointIndex: number\n) => {\n  const { path, pathOffset } = pathObject;\n\n  const anchorCommand =\n    path[(commandIndex > 0 ? commandIndex : path.length) - 1];\n  const anchorPoint = new Point(\n    anchorCommand[pointIndex] as number,\n    anchorCommand[pointIndex + 1] as number\n  );\n\n  const anchorPointInParentPlane = anchorPoint\n    .subtract(pathOffset)\n    .transform(pathObject.calcOwnMatrix());\n\n  const mouseLocalPosition = sendPointToPlane(\n    new Point(x, y),\n    undefined,\n    pathObject.calcOwnMatrix()\n  );\n\n  path[commandIndex][pointIndex] = mouseLocalPosition.x + pathOffset.x;\n  path[commandIndex][pointIndex + 1] = mouseLocalPosition.y + pathOffset.y;\n  pathObject.setDimensions();\n\n  const newAnchorPointInParentPlane = anchorPoint\n    .subtract(pathObject.pathOffset)\n    .transform(pathObject.calcOwnMatrix());\n\n  const diff = newAnchorPointInParentPlane.subtract(anchorPointInParentPlane);\n  pathObject.left -= diff.x;\n  pathObject.top -= diff.y;\n\n  return true;\n};\n\n/**\n * This function locates the controls.\n * It'll be used both for drawing and for interaction.\n */\nfunction pathPositionHandler(\n  this: PathPointControl,\n  dim: Point,\n  finalMatrix: TMat2D,\n  pathObject: Path\n) {\n  const { commandIndex, pointIndex } = this;\n  return calcPathPointPosition(pathObject, commandIndex, pointIndex);\n}\n\n/**\n * This function defines what the control does.\n * It'll be called on every mouse move after a control has been clicked and is being dragged.\n * The function receives as argument the mouse event, the current transform object\n * and the current position in canvas coordinate `transform.target` is a reference to the\n * current object being transformed.\n */\nexport function pathActionHandler(\n  this: PathPointControl,\n  eventData: TPointerEvent,\n  transform: TTransformAnchor,\n  x: number,\n  y: number\n) {\n  const { target } = transform;\n  const { commandIndex, pointIndex } = this;\n  const actionPerformed = movePathPoint(\n    target as Path,\n    x,\n    y,\n    commandIndex,\n    pointIndex\n  );\n  if (actionPerformed) {\n    fireEvent(this.actionName as TModificationEvents, {\n      ...commonEventInfo(eventData, transform, x, y),\n      commandIndex,\n      pointIndex,\n    });\n  }\n  return actionPerformed;\n}\n\nconst indexFromPrevCommand = (previousCommandType: TSimpleParseCommandType) =>\n  previousCommandType === 'C' ? 5 : previousCommandType === 'Q' ? 3 : 1;\n\nclass PathPointControl extends Control {\n  declare commandIndex: number;\n  declare pointIndex: number;\n  declare controlFill: string;\n  declare controlStroke: string;\n  constructor(options?: Partial<PathPointControl>) {\n    super(options);\n  }\n\n  render(\n    ctx: CanvasRenderingContext2D,\n    left: number,\n    top: number,\n    styleOverride: ControlRenderingStyleOverride | undefined,\n    fabricObject: Path\n  ) {\n    const overrides: ControlRenderingStyleOverride = {\n      ...styleOverride,\n      cornerColor: this.controlFill,\n      cornerStrokeColor: this.controlStroke,\n      transparentCorners: !this.controlFill,\n    };\n    super.render(ctx, left, top, overrides, fabricObject);\n  }\n}\n\nclass PathControlPointControl extends PathPointControl {\n  declare connectionDashArray?: number[];\n  declare connectToCommandIndex: number;\n  declare connectToPointIndex: number;\n  constructor(options?: Partial<PathControlPointControl>) {\n    super(options);\n  }\n\n  render(\n    this: PathControlPointControl,\n    ctx: CanvasRenderingContext2D,\n    left: number,\n    top: number,\n    styleOverride: ControlRenderingStyleOverride | undefined,\n    fabricObject: Path\n  ) {\n    const { path } = fabricObject;\n    const {\n      commandIndex,\n      pointIndex,\n      connectToCommandIndex,\n      connectToPointIndex,\n    } = this;\n    ctx.save();\n    ctx.strokeStyle = this.controlStroke;\n    if (this.connectionDashArray) {\n      ctx.setLineDash(this.connectionDashArray);\n    }\n    const [commandType] = path[commandIndex];\n    const point = calcPathPointPosition(\n      fabricObject,\n      connectToCommandIndex,\n      connectToPointIndex\n    );\n\n    if (commandType === 'Q') {\n      // one control point connects to 2 points\n      const point2 = calcPathPointPosition(\n        fabricObject,\n        commandIndex,\n        pointIndex + 2\n      );\n      ctx.moveTo(point2.x, point2.y);\n      ctx.lineTo(left, top);\n    } else {\n      ctx.moveTo(left, top);\n    }\n    ctx.lineTo(point.x, point.y);\n    ctx.stroke();\n    ctx.restore();\n\n    super.render(ctx, left, top, styleOverride, fabricObject);\n  }\n}\n\nconst createControl = (\n  commandIndexPos: number,\n  pointIndexPos: number,\n  isControlPoint: boolean,\n  options: Partial<Control> & {\n    controlPointStyle?: PathPointControlStyle;\n    pointStyle?: PathPointControlStyle;\n  },\n  connectToCommandIndex?: number,\n  connectToPointIndex?: number\n) =>\n  new (isControlPoint ? PathControlPointControl : PathPointControl)({\n    commandIndex: commandIndexPos,\n    pointIndex: pointIndexPos,\n    actionName: ACTION_NAME,\n    positionHandler: pathPositionHandler,\n    actionHandler: pathActionHandler,\n    connectToCommandIndex,\n    connectToPointIndex,\n    ...options,\n    ...(isControlPoint ? options.controlPointStyle : options.pointStyle),\n  } as Partial<PathControlPointControl>);\n\nexport function createPathControls(\n  path: Path,\n  options: Partial<Control> & {\n    controlPointStyle?: PathPointControlStyle;\n    pointStyle?: PathPointControlStyle;\n  } = {}\n): Record<string, Control> {\n  const controls = {} as Record<string, Control>;\n  let previousCommandType: TSimpleParseCommandType = 'M';\n  path.path.forEach((command, commandIndex) => {\n    const commandType = command[0];\n\n    if (commandType !== 'Z') {\n      controls[`c_${commandIndex}_${commandType}`] = createControl(\n        commandIndex,\n        command.length - 2,\n        false,\n        options\n      );\n    }\n    switch (commandType) {\n      case 'C':\n        controls[`c_${commandIndex}_C_CP_1`] = createControl(\n          commandIndex,\n          1,\n          true,\n          options,\n          commandIndex - 1,\n          indexFromPrevCommand(previousCommandType)\n        );\n        controls[`c_${commandIndex}_C_CP_2`] = createControl(\n          commandIndex,\n          3,\n          true,\n          options,\n          commandIndex,\n          5\n        );\n        break;\n      case 'Q':\n        controls[`c_${commandIndex}_Q_CP_1`] = createControl(\n          commandIndex,\n          1,\n          true,\n          options,\n          commandIndex,\n          3\n        );\n        break;\n    }\n    previousCommandType = commandType;\n  });\n  return controls;\n}\n","import { WidgetBaseInterface, TOriginX, TOriginY } from './widget.entity.base';\nimport { WidgetType } from './widget.type';\n\nexport type pathType = 'curvePath' | 'straightPath';\nexport type pathArrowTip = 'none' | 'start' | 'end' | 'both';\nexport type xy = { x: number; y: number };\n\nexport interface WidgetConnectorInterface extends WidgetBaseInterface {\n  path: any[];\n  fromObjectId: string;\n  toObjectId: string;\n  pathType: pathType;\n  pathArrowTip: pathArrowTip;\n  fromPoint: xy;\n  toPoint: xy;\n  control1: xy;\n  control2: xy;\n  style: any;\n\n  fill: string | null | any;\n  stroke: string | null | any;\n  strokeWidth: number;\n  strokeLineCap: string;\n  strokeDashOffset: number;\n  strokeLineJoin: string;\n  strokeUniform: boolean;\n  strokeMiterLimit: number;\n  fillRule: string;\n}\n\nexport class WidgetConnectorClass implements WidgetConnectorInterface {\n  updatedBy: string = \"\";\n  updatedByName: string = \"\";\n  path: any[] = [''];\n  fill: any = 'transparent';\n  stroke: any = '#000000';\n  strokeWidth: number = 1;\n  strokeLineCap: string = 'butt';\n  strokeDashOffset: number = 0;\n  strokeLineJoin: string = 'miter';\n  strokeUniform: boolean = false;\n  strokeMiterLimit: number = 10;\n  fillRule: string = 'nonzero';\n   createdByName: string = \"\";\n  fromObjectId: string = '';\n  toObjectId: string = '';\n  pathType: pathType = \"curvePath\";\n  pathArrowTip: pathArrowTip = 'none';\n  fromPoint: xy = { x: 0, y: 0 };\n  toPoint: xy = { x: 0, y: 0 };\n  control1: xy = { x: 0, y: 0 };\n  control2: xy = { x: 0, y: 0 };\n  style: any = {};\n  id: string = '';\n  boardId: string = '';\n  backgroundColor: string = 'transparent';\n  width: number = 0;\n  height: number = 0;\n  left: number = 0;\n  locked: boolean = false;\n  objType: WidgetType = \"XConnector\"; // Replace with an actual default value\n  originX: TOriginX = 'center'; // Replace with an actual default value\n  originY: TOriginY = 'center'; // Replace with an actual default value\n  scaleX: number = 1;\n  scaleY: number = 1;\n  selectable: boolean = true;\n  top: number = 0;\n  zIndex: number = Date.now() * 100;\n  version: string = '';\n  updatedAt: number = Date.now();\n   createdAt: number = Date.now();\n  createdBy: string = '';\n  visible: boolean = true;\n}\n\nexport const EntityKeys = Object.keys(new WidgetConnectorClass()) as (keyof WidgetConnectorInterface)[];\n","import { Path } from '../Path';\nimport { sendPointToPlane, TSimpleParsedCommand } from '../../util';\nimport { Point, XY } from '../../Point';\nimport { classRegistry } from '../../ClassRegistry';\nimport { iMatrix } from '../../constants';\nimport { createPathControls } from '../../controls/pathControl';\nimport { XCanvas } from '../../canvas/canvasx/bx-canvas';\nimport { Transform } from '../../EventTypeDefs';\n\nimport { EntityKeys, WidgetConnectorInterface } from './type/widget.entity.connector';\nimport { WidgetType } from './type/widget.type';\n\nconst getPath = (\n  fromPoint: XY,\n  toPoint: XY,\n  control1: XY,\n  control2: XY,\n  pathType: 'curvePath' | 'straightPath' = 'curvePath'\n) => {\n  if (pathType === 'curvePath') {\n    return `M ${fromPoint.x} ${fromPoint.y} C ${control1.x}, ${control1.y}, ${control2.x} ${control2.y}, ${toPoint.x} ${toPoint.y}`;\n  } else {\n    return `M ${fromPoint.x} ${fromPoint.y} L ${toPoint.x} ${toPoint.y}`;\n  }\n};\n\nclass XConnector extends Path implements WidgetConnectorInterface {\n  static type: WidgetType = 'XConnector';\n  static objType: WidgetType = 'XConnector';\n  style: any;\n  declare fromObjectId: string;\n  declare toObjectId: string;\n  declare pathType: 'curvePath' | 'straightPath';\n  declare pathArrowTip: 'none' | 'start' | 'end' | 'both';\n  declare fromPoint: XY;\n  declare toPoint: XY;\n  declare control1: XY;\n  declare control2: XY;\n\n  /**\n   * Contains the path to draw the arrow tip start\n   */\n  declare pathStart: TSimpleParsedCommand[];\n\n  /**\n   * Contains the path to draw the arrow tip end\n   */\n  declare pathEnd: TSimpleParsedCommand[];\n\n\n  constructor(\n    fromPoint: XY,\n    toPoint: XY,\n    control1: XY,\n    control2: XY,\n    options: any = {}\n  ) {\n    const path = getPath(fromPoint, toPoint, control1, control2, options.pathType);\n\n\n\n\n    super(path, options);\n    // Object.assign(this, options);\n\n    //default values\n    this.perPixelTargetFind = true;\n    this.cornerColor = 'white';\n    this.cornerStyle = 'circle';\n    this.type = 'XConnector';\n    this.objType = 'XConnector';\n    this.transparentCorners = false;\n    this.cornerStrokeColor = 'gray';\n    this.hasBorders = false;\n    this.objectCaching = false;\n\n\n    this.pathType = options.pathType || 'curvePath';\n    this.pathArrowTip = options.pathArrowTip || 'both';\n    this.fromObjectId = options.fromObjectId;\n    this.toObjectId = options.toObjectId;\n    this.fromPoint = fromPoint;\n    this.toPoint = toPoint;\n    this.control1 = control1;\n    this.control2 = control2;\n    this.createdByName = options.createdByName;\n    this.createdBy = options.createdBy;\n    this.createdAt = options.createdAt;\n    this.boardId = options.boardId;\n    this.style = options.style;\n    this.fill = options.fill || 'transparent';\n    this.stroke = options.stroke || '#000000';\n    this.createdAt = options.createdAt;\n    this.updatedBy = options.updatedBy;\n    this.updatedByName = options.updatedByName;\n    this.version = options.version;\n    this.zIndex = options.zIndex;\n\n    // Object.assign(this, options);\n    this._setMovementLock();\n    this.calcStartEndPath();\n    this.controls = {\n      ...createPathControls(this, {\n        mouseDownHandler: this._mouseDownControl.bind(this),\n        mouseUpHandler: this._mouseUpControl.bind(this),\n        cursorStyle: 'crosshair',\n        pointStyle: {\n          controlFill: 'white',\n          controlStroke: 'gray',\n        },\n        controlPointStyle: {\n          controlFill: 'white',\n          connectionDashArray: [5, 5],\n          controlStroke: 'gray',\n        },\n      }),\n    };\n    this.on('modifyPath', function (this: XConnector, evtOpt) {\n      this.calcStartEndPath();\n      const { commandIndex, pointIndex } = evtOpt;\n      // commandIndex === 0 is always start,\n      // all the commandIndex === 1 are control points apart the 5\n      if (commandIndex === 1 && pointIndex !== 5) {\n        return;\n      }\n      this.dragActionEventHandler(evtOpt.commandIndex, evtOpt.pointIndex);\n    });\n  }\n  updatedBy: string;\n  updatedByName: string;\n\n  createdByName: string;\n  boardId: string;\n  objType: WidgetType;\n  userId: string;\n  zIndex: number;\n  version: string;\n  updatedAt: number;\n\n  createdAt: number;\n  createdBy: string;\n\n  getFromPoint() {\n    const command = this.path[0];\n    return new Point(command[1]!, command[2]!);\n  }\n  getObject() {\n    const entityKeys: string[] = EntityKeys;\n    const result: Record<string, any> = {};\n\n    entityKeys.forEach((key) => {\n      if (key in this) {\n        result[key] = (this as any)[key];\n      }\n    });\n\n    return result;\n  }\n\n\n  getToPoint() {\n    const lastCommand = this.path[this.path.length - 1];\n    return lastCommand[0] === 'L'\n      ? new Point(lastCommand[1]!, lastCommand[2]!)\n      : new Point(lastCommand[5]!, lastCommand[6]!);\n  }\n\n  /**\n   * calculate the drawing commands for the connector tips\n   */\n  calcStartEndPath() {\n    this.pathStart = [];\n    this.pathEnd = [];\n\n    const { pathType, pathArrowTip } = this;\n    const firstCommand = this.path[0];\n    const lastCommand = this.path[this.path.length - 1];\n\n    const fromPoint = new Point(firstCommand[1]!, firstCommand[2]!);\n    const toPoint =\n      lastCommand[0] === 'L'\n        ? new Point(lastCommand[1]!, lastCommand[2]!)\n        : new Point(lastCommand[5]!, lastCommand[6]!);\n\n    /* Calculate Path START */\n    if (pathArrowTip === 'start' || pathArrowTip === 'both') {\n      const startArrowSize = 20 + this.strokeWidth; // Adjust the size of the start arrow tip\n\n      const startAngle =\n        pathType === 'straightPath'\n          ? Math.atan2(fromPoint.y - toPoint.y, fromPoint.x - toPoint.x) +\n          Math.PI\n          : Math.atan2(\n            lastCommand[2]! - fromPoint.y,\n            lastCommand[1]! - fromPoint.x\n          );\n\n      const startArrow1X =\n        fromPoint.x + startArrowSize * Math.cos(startAngle + Math.PI / 6);\n      const startArrow1Y =\n        fromPoint.y + startArrowSize * Math.sin(startAngle + Math.PI / 6);\n      const startArrow2X =\n        fromPoint.x + startArrowSize * Math.cos(startAngle - Math.PI / 6);\n      const startArrow2Y =\n        fromPoint.y + startArrowSize * Math.sin(startAngle - Math.PI / 6);\n\n      this.pathStart = [\n        ['M', startArrow1X, startArrow1Y],\n        ['L', fromPoint.x, fromPoint.y],\n        ['L', startArrow2X, startArrow2Y],\n      ];\n    }\n    /* Calculate Path End */\n    if (pathArrowTip === 'end' || pathArrowTip === 'both') {\n      const endArrowSize = 20 + this.strokeWidth; // Adjust the size of the end arrow tip\n      const endAngle =\n        pathType === 'straightPath'\n          ? Math.atan2(toPoint.y - fromPoint.y, toPoint.x - fromPoint.x)\n          : Math.atan2(\n            toPoint.y - lastCommand[4]!,\n            toPoint.x - lastCommand[3]!\n          );\n\n      const endArrow1X =\n        toPoint.x - endArrowSize * Math.cos(endAngle - Math.PI / 6);\n      const endArrow1Y =\n        toPoint.y - endArrowSize * Math.sin(endAngle - Math.PI / 6);\n      const endArrow2X =\n        toPoint.x - endArrowSize * Math.cos(endAngle + Math.PI / 6);\n      const endArrow2Y =\n        toPoint.y - endArrowSize * Math.sin(endAngle + Math.PI / 6);\n\n      this.pathEnd = [\n        ['M', endArrow1X, endArrow1Y],\n        ['L', toPoint.x, toPoint.y],\n        ['L', endArrow2X, endArrow2Y],\n      ];\n    }\n  }\n\n  calculateControlPoint(controlPointType: 'from' | 'to', point: any) {\n    let controlPoint;\n    if (controlPointType === 'from') {\n      //@ts-ignore\n      const fromObject = this.canvas?.findById(this.fromObjectId);\n      if (fromObject && fromObject.calculateControlPoint) {\n        controlPoint = fromObject.calculateControlPoint(point);\n      }\n    }\n\n    if (controlPointType === 'to') {\n      //@ts-ignore\n      const toObject = this.canvas?.findById(this.toObjectId);\n      if (toObject && toObject.calculateControlPoint) {\n        controlPoint = toObject.calculateControlPoint(point);\n      }\n    }\n\n    if (controlPoint) {\n      return controlPoint;\n    } else {\n      return point;\n    }\n  }\n\n  /**\n   * Given the points in scene coordinates, updates the path\n   * This function is called by other objects that are moving or changing properties\n   */\n  update({ fromPoint, toPoint, control1, control2, style }: any = {}) {\n    const finalCommand = this.path[this.path.length - 1];\n\n    if (!fromPoint) {\n      fromPoint = TransformPointFromPathToCanvas(\n        this,\n        new Point(this.path[0][1]!, this.path[0][2]!)\n      );\n      this.fromPoint = fromPoint;\n    }\n\n    if (!toPoint) {\n      toPoint = TransformPointFromPathToCanvas(\n        this,\n        finalCommand[0] === 'L'\n          ? new Point(finalCommand[1]!, finalCommand[2]!)\n          : new Point(finalCommand[5]!, finalCommand[6]!)\n      );\n      finalCommand[0] === 'L' ? 1 : 5;\n      this.toPoint = toPoint;\n    }\n    let controlPoint1: Point, controlPoint2: Point;\n    if (finalCommand[0] === 'L') {\n      controlPoint1 = this.calculateControlPoint('from', fromPoint);\n      controlPoint2 = this.calculateControlPoint('to', toPoint);\n    }\n    if (!control1) {\n      control1 = TransformPointFromPathToCanvas(\n        this,\n        finalCommand[0] === 'L'\n          ? controlPoint1!\n          : new Point(finalCommand[1]!, finalCommand[2]!)\n      );\n      this.control1 = control1;\n    }\n\n    if (!control2) {\n      control2 = TransformPointFromPathToCanvas(\n        this,\n        finalCommand[0] === 'L'\n          ? controlPoint2!\n          : new Point(finalCommand[3]!, finalCommand[4]!)\n      );\n      this.control2 = control2;\n    }\n\n    if (style) {\n      this.style = style;\n    }\n\n    const path = getPath(fromPoint, toPoint, control1, control2, this.pathType);\n\n    const { path: newPath } = new Path(path);\n    this.path = newPath;\n    this.setBoundingBox(true);\n\n    this.calcStartEndPath();\n    this.dirty = true;\n  }\n\n  _mouseDownControl(\n    eventData: any,\n    transform: Transform,\n    x: number,\n    y: number\n  ) {\n    const target = transform.target;\n    target.objectCaching = false;\n\n    this.mouseDownHandler(eventData, transform, x, y);\n  }\n\n  mouseDownHandler(eventData: any, transform: Transform, x: number, y: number) {\n    //reserve for subclass\n  }\n\n  mouseUpHandler(eventData: any, transform: Transform, x: number, y: number) {\n    //reserve for subclass\n  }\n  /**\n   * Compared to Path, it will render the official Path + the arrow tips.\n   * @param ctx\n   */\n  _renderPathCommands(ctx: CanvasRenderingContext2D) {\n    const path = this.path;\n    this.path = [...this.pathStart, ...this.path, ...this.pathEnd];\n    super._renderPathCommands(ctx);\n    this.path = path;\n  }\n\n  _mouseUpControl(eventData: any, transform: Transform, x: number, y: number) {\n    const { target } = transform;\n    if (!target.canvas) {\n      return;\n    }\n    (target.canvas as XCanvas).dockingWidget = null;\n    target.dirty = true;\n    target.setCoords();\n    transform.target.canvas?.requestRenderAll();\n    this.mouseUpHandler(eventData, transform, x, y);\n    this._setMovementLock();\n  }\n\n  _setMovementLock() {\n    if (this.fromObjectId || this.toObjectId) {\n      this.lockMovementX = true;\n      this.lockMovementY = true;\n    } else {\n      this.lockMovementX = false;\n      this.lockMovementY = false;\n    }\n  }\n\n  getControlPointOnCanvas(obj: any, controlName: string) {\n    const controlInfo = obj.controls[controlName];\n    const x = controlInfo.x * obj.width;\n    const y = controlInfo.y * obj.height;\n    const point = new Point(x, y);\n\n    const transformedPoint = obj.transformPointToCanvas(point);\n\n    return transformedPoint;\n  }\n\n  /**\n   * Responds to the path points being moved calculating the docking.\n   * @param commandIndex The command index in the path we are dragging\n   * @param pointIndex  the index of the X coordinate of the point we are moving in .path[commandIndex]\n   * @returns\n   */\n  dragActionEventHandler(commandIndex: number, pointIndex: number) {\n    const target = this;\n    // const relevantPoint = getLocalPoint(transform, 'center', 'top', x, y);\n    //@ts-ignore\n    const currentDockingObject = target.canvas?.dockingWidget;\n\n    const property = commandIndex === 0 ? 'fromObjectId' : 'toObjectId';\n    const existingConnectionId = target[property];\n    let connectedObject: any = null;\n    if (existingConnectionId) {\n      connectedObject = (target.canvas as XCanvas).findById(\n        existingConnectionId\n      ) as any;\n    }\n\n    // Andrea, followup: currentDockingObject.hoveringControl relies on a mousemove event that gets added\n    // and removed when we start the connector drag.\n    // this logic should be resolved inside the connector.\n    const connectorType = commandIndex === 0 ? 'from' : 'to';\n    if (\n      currentDockingObject &&\n      currentDockingObject.controls[\n      currentDockingObject.canvas.hoveringControl\n      ] &&\n      currentDockingObject.calculateControlPoint\n    ) {\n      const hoverPoint = this.getControlPointOnCanvas(\n        currentDockingObject,\n        currentDockingObject.canvas.hoveringControl\n      );\n\n      const targetX = hoverPoint.x;\n      const targetY = hoverPoint.y;\n\n      if (existingConnectionId) {\n        if (connectedObject) {\n          connectedObject.connectors = connectedObject.connectors?.filter(\n            (connector: any) =>\n              !(\n                connector.connectorId === target.id &&\n                connector.connectorType === connectorType\n              )\n          );\n          if (connectedObject.calculateControlPoint) {\n            const controlPoint =\n              connectedObject.calculateControlPoint(hoverPoint);\n            if (commandIndex === 0) {\n              this.update({\n                fromPoint: { x: targetX, y: targetY },\n                control1: controlPoint,\n              });\n            } else {\n              this.update({\n                toPoint: { x: targetX, y: targetY },\n                control2: controlPoint,\n              });\n            }\n          }\n        }\n      }\n\n      target[commandIndex === 0 ? 'fromObjectId' : 'toObjectId'] =\n        currentDockingObject.id;\n\n      if (!currentDockingObject.connectors) {\n        currentDockingObject.connectors = [];\n      }\n\n      currentDockingObject.connectors.push({\n        connectorId: target.id,\n        connectorType: connectorType,\n        point: currentDockingObject.transformPointFromCanvas({\n          x: targetX,\n          y: targetY,\n        }),\n      });\n    }\n\n    if (!currentDockingObject && connectedObject) {\n      //if it is not attached to object, remove the connector from the existing connected object and clear the from/to object id\n      target[property] = '';\n      connectedObject.connectors = connectedObject.connectors?.filter(\n        (connector: any) =>\n          !(\n            connector.connectorId === target.id &&\n            connector.connectorType === connectorType\n          )\n      );\n    }\n  }\n}\n\nexport { XConnector };\n\nexport const TransformPointFromPathToCanvas = (\n  object: XConnector,\n  point: Point\n) =>\n  sendPointToPlane(\n    point.subtract(object.pathOffset),\n    object.calcOwnMatrix(),\n    iMatrix\n  );\n\nclassRegistry.setClass(XConnector);\n","import { WidgetBaseInterface, TOriginX, TOriginY } from './widget.entity.base';\nimport { WidgetType } from './widget.type';\nimport { FileObject } from './file';\n\nexport enum FileDocument {\n  DOC = 'Word Document',\n  DOCX = 'Word Document',\n  XLS = 'Excel Document',\n  XLSX = 'Excel Document',\n  PPT = 'PPT Document',\n  PPTX = 'PPT Document',\n  PDF = 'PDF Document',\n  ZIP = 'ZIP File',\n  MP4 = 'Video Document',\n  WEBM = 'Video Document',\n  MP3 = 'Audio Document',\n  M4A = 'Audio Document',\n  WAV = 'Audio Document',\n  AAC = 'Audio Document',\n  FLAC = 'Audio Document',\n  OGG = 'Audio Document',\n  AIFF = 'Audio Document',\n  WMA = 'Audio Document',\n  APE = 'Audio Document',\n}\n\nexport type FileObjectType = keyof typeof FileEnum;\n\nexport enum FileEnum {\n  DOC = 'doc',\n  DOCX = 'docx',\n  XLS = 'xls',\n  XLSX = 'xlsx',\n  PPT = 'ppt',\n  PPTX = 'pptx',\n  PDF = 'pdf',\n  ZIP = 'zip',\n  MP4 = 'mp4',\n  WEBM = 'webm',\n\n  MP3 = 'mp3',\n  M4A = 'm4a',\n  WAV = 'wav',\n  AAC = 'aac',\n  FLAC = 'flac',\n  OGG = 'ogg',\n  AIFF = 'aiff',\n  WMA = 'wma',\n  APE = 'ape',\n  UNKNOWN = 'unknown',\n}\n\n\nexport interface WidgetFileInterface extends WidgetBaseInterface {\n  fileName: string;\n  fileSrc: FileObject | null;\n  vectorSrc: FileObject | null;\n  transcription: string;\n  fileObjectType: FileObjectType;\n  previewImage: FileObject | null;\n}\n\n\nexport class WidgetFileClass implements WidgetFileInterface {\n  updatedBy: string = \"\";\n  updatedByName: string = \"\";\n  createdByName: string = \"\";\n  fileName: string = '';\n  fileSrc: FileObject | null = null;\n  vectorSrc: FileObject | null = null;\n  transcription: string = '';\n  fileObjectType: FileObjectType = 'UNKNOWN';\n  previewImage: FileObject | null = null;\n  id: string = '';\n  boardId: string = '';\n  backgroundColor: string = 'transparent';\n  width: number = 300;\n  height: number = 400;\n  left: number = 0;\n  locked: boolean = false;\n  objType: WidgetType = \"XFile\";\n  originX: TOriginX = 'center';\n  originY: TOriginY = 'center';\n  scaleX: number = 1;\n  scaleY: number = 1;\n  selectable: boolean = true;\n  top: number = 0;\n   zIndex: number = Date.now() * 100;\n  version: string = '';\n  updatedAt: number = Date.now();\n  createdAt: number = Date.now();\n  createdBy: string = '';\n  visible: boolean = true;\n}\n\nexport const EntityKeys = Object.keys(new WidgetFileClass()) as (keyof WidgetFileInterface)[];","import { TOriginX, TOriginY } from 'fabric';\nimport { classRegistry } from '../../../ClassRegistry';\nimport { Shadow } from '../../../Shadow';\nimport { getFabricWindow } from '../../../env';\nimport { loadImage } from '../../../util/misc/objectEnlive';\n\nimport { ImageProps } from '../../Image';\nimport { FabricObject } from '../../Object/FabricObject';\nimport { Rect } from '../../Rect';\n\nimport {\n  WidgetFileInterface,\n  WidgetFileClass,\n  FileObjectType,\n  FileEnum\n} from '../type/widget.entity.file';\nimport { WidgetType, WidgetFileType } from '../type/widget.type';\nimport { EntityKeys, } from \"../type/widget.entity.file\";\nimport { FileObject } from \"../type/file\";\n\n\n\nexport type XFileProps = ImageProps & WidgetFileClass;\nconst FILE_ICON_PATHS: Record<WidgetFileType, string> = {\n  XFileWord: '/boardxstatic/fileIcons/word.png',\n  XFileExcel: '/boardxstatic/fileIcons/excel.png',\n  XFilePPT: '/boardxstatic/fileIcons/ppt.png',\n  XFilePDF: '/boardxstatic/fileIcons/pdf.svg',\n  XFileZip: '/boardxstatic/fileIcons/zip.png',\n  XFileVideo: '/boardxstatic/fileIcons/mp4.png',\n  XFileAudio: '/boardxstatic/fileIcons/audio.png',\n  XFile: '/boardxstatic/fileIcons/file.png',\n};\n\nexport const FILE_TYPE_NAMES: Record<FileObjectType, string> = {\n  DOC: 'Word Document',\n  DOCX: 'Word Document',\n  XLS: 'Excel Document',\n  XLSX: 'Excel Document',\n  PPT: 'PPT Document',\n  PPTX: 'PPT Document',\n  PDF: 'PDF Document',\n  ZIP: 'ZIP File',\n  MP4: 'Video Document',\n  WEBM: 'Video Document',\n  MP3: 'Audio Document',\n  M4A: 'Audio Document',\n  WAV: 'Audio Document',\n  AAC: 'Audio Document',\n  FLAC: 'Audio Document',\n  OGG: 'Audio Document',\n  AIFF: 'Audio Document',\n  WMA: 'Audio Document',\n  APE: 'Audio Document',\n  UNKNOWN: 'Other Document',\n};\n\nexport function getWidgetFileType(fileName: string): WidgetFileType {\n  const extension = fileName.split('.').pop()?.toUpperCase() as FileObjectType;\n  switch (extension) {\n    case 'DOC':\n    case 'DOCX':\n      return 'XFileWord';\n    case 'XLS':\n    case 'XLSX':\n      return 'XFileExcel';\n    case 'PPT':\n    case 'PPTX':\n      return 'XFilePPT';\n    case 'PDF':\n      return 'XFilePDF';\n    case 'ZIP':\n      return 'XFileZip';\n    case 'MP4':\n    case 'WEBM':\n      return 'XFileVideo';\n    case 'MP3':\n    case 'M4A':\n    case 'WAV':\n    case 'AAC':\n    case 'FLAC':\n    case 'OGG':\n    case 'AIFF':\n    case 'WMA':\n    case 'APE':\n      return 'XFileAudio';\n    default:\n      return 'XFile';\n  }\n}\n\nconst VIDEO_FILE_EXTENSIONS = new Set([FileEnum.MP4, FileEnum.WEBM]);\n\nexport class XFile extends FabricObject implements WidgetFileInterface {\n  static objType: WidgetFileType = 'XFile';\n  static type: WidgetFileType = 'XFile';\n\n  // WidgetFile properties\n  id: string = '';\n  boardId: string = '';\n  backgroundColor: string = 'rgba(0,0,0,0)';\n  fill: string = 'rgba(0,0,0,0)';\n  width: number = 230;\n  height: number = 248;\n  left: number = 0;\n  locked: boolean = false;\n  objType: WidgetType = 'XFile';\n  originX: TOriginX = 'center';\n  originY: TOriginY = 'center';\n  scaleX: number = 1;\n  scaleY: number = 1;\n  selectable: boolean = true;\n  top: number = 0;\n  userId: string = '';\n  zIndex: number = 0;\n  version: string = '';\n  updatedAt: number = Date.now();\n  lastEditedBy: string = '';\n  createdAt: number = Date.now();\n  createdBy: string = '';\n  visible: boolean = true;\n\n  // WidgetFile specific properties\n  fileName: string = '';\n  fileSrc: FileObject = { tmpPath: '', id: '', path: '' };\n  vectorSrc: FileObject = { tmpPath: '', id: '', path: '' };\n  transcription: string = '';\n\n  previewImage: FileObject = { tmpPath: '', id: '', path: '' };\n\n  private _previewImage: HTMLImageElement | null = null;\n\n\n\n  constructor(options: Partial<XFileProps & { type: string }> = {}) {\n    super(options);\n    this.objType = 'XFile';\n    this.initializeVisuals();\n    Object.assign(this, options);\n    this.fileObjectType = XFile.getFileType(options.fileName || '');\n    this.fill = options.backgroundColor || this.backgroundColor;\n    this.loadPreviewImage(\n      this.getFileIconURL(options.objType as WidgetFileType),\n      options.fileName!\n    );\n    this.on(\"mousedblclick\", this.onDoubleClick.bind(this)); // Attach event listener\n  }\n\n  updatedBy: string;\n  updatedByName: string;\n\n  createdByName: string;\n  fileObjectType: FileObjectType;\n\n\n\n  private initializeVisuals() {\n    this.cornerColor = 'white';\n    this.cornerStrokeColor = 'gray';\n    this.cornerSize = 10;\n    this.cornerStyle = 'circle';\n    this.transparentCorners = false;\n    this.shadow = new Shadow({\n      color: 'rgba(217, 161, 177, 0.54)',\n      offsetX: 1,\n      offsetY: 2,\n      blur: 4,\n      id: 310,\n    });\n\n    this.clipPath = new Rect({\n      left: 0,\n      top: 0,\n      rx: 8,\n      ry: 8,\n      width: this.width,\n      height: this.height,\n      fill: '#000000',\n    });\n  }\n\n  toObject(propertiesToInclude: string[] = []): any {\n    return super.toObject([...EntityKeys, ...propertiesToInclude]);\n  }\n\n  getObject() {\n    const entityKeys: string[] = EntityKeys;\n    const result: Record<string, any> = {};\n\n    entityKeys.forEach((key) => {\n      if (key in this) {\n        result[key] = (this as any)[key];\n      }\n    });\n\n    return result;\n  }\n\n  onDoubleClick(): void {\n    getFabricWindow().open(this.fileSrc?.tmpPath, '_blank');\n  }\n\n  drawObject(ctx: CanvasRenderingContext2D): void {\n    this.drawBorder(ctx);\n    this.drawPreviewImage(ctx);\n    this.renderTitle(ctx, this.fileName);\n    this._renderStroke(ctx);\n  }\n\n  private drawBorder(ctx: CanvasRenderingContext2D): void {\n    ctx.beginPath();\n    ctx.fillStyle = 'rgba(255,255,255,0)';\n    ctx.fillRect(-this.width / 2, -this.height / 2, this.width, this.height);\n    ctx.lineWidth = 0;\n    ctx.strokeStyle = '#ffffff';\n    ctx.moveTo(-this.width / 2, -this.height / 2);\n    ctx.stroke();\n  }\n\n\n  private drawPreviewImage(ctx: CanvasRenderingContext2D): void {\n    if (this._previewImage) {\n      // Calculate the available height for the image to avoid overlapping the title\n      const titleHeight = 90; // Total height reserved for the title and background\n      const availableHeight = this.height - titleHeight;\n\n      // Calculate image dimensions while maintaining aspect ratio\n      const imageAspectRatio = this._previewImage.width / this._previewImage.height;\n      const drawWidth = this.width;\n      const drawHeight = drawWidth / imageAspectRatio;\n\n      // Adjust if the calculated height exceeds the available height\n      const finalDrawHeight = Math.min(drawHeight, availableHeight);\n\n      ctx.drawImage(\n        this._previewImage,\n        -this.width / 2,\n        -this.height / 2,\n        drawWidth,\n        finalDrawHeight\n      );\n    }\n  }\n\n  static getFileTypeName(fileName: string = ''): string {\n    const extension = fileName\n      .split('.')\n      .pop()\n      ?.toUpperCase() as FileObjectType;\n    return FILE_TYPE_NAMES[extension] || 'Other Document';\n  }\n\n  static getFileType(fileName: string = ''): FileObjectType {\n    const extension: FileObjectType = fileName\n      .split('.')\n      .pop()\n      ?.toUpperCase() as FileObjectType;\n\n    return extension;\n  }\n\n  isFileVideo(fileName: string): boolean {\n    const extension = fileName.split('.').pop()?.toUpperCase();\n    return VIDEO_FILE_EXTENSIONS.has(extension as FileEnum);\n  }\n\n  renderTitle(ctx: CanvasRenderingContext2D, title: string): void {\n    const maxWidth = this.width;\n    const x = -this.width / 2;\n    const y = this.height / 2 - 60;\n\n    ctx.font = '16px Inter';\n    ctx.fillStyle = 'rgba(255, 255, 255, 1)';\n    ctx.fillRect(x, y - 29, maxWidth, 90);\n    ctx.fillStyle = '#190FA1';\n\n    const sanitizedTitle = this.sanitizeTitle(title);\n    this.wrapText(ctx, sanitizedTitle, x + 15, y - 5, maxWidth - 20, 23);\n\n    // const newUrl = this.getShortenedUrl();\n    // ctx.font = '12px Inter';\n    // ctx.fillStyle = 'rgba(35, 41, 48, 0.65)';\n    // this.wrapText(ctx, newUrl, x + 15, y + 45, maxWidth - 20, 25);\n  }\n\n  private sanitizeTitle(title: string): string {\n    const unicodeTitle = this.toUnicode(title);\n    if (!title || unicodeTitle.includes('\\\\ufffd')) {\n      const parts = this.fileSrc.tmpPath.split('.');\n      return parts.length > 2 ? parts[1] : 'Untitled';\n    }\n    return title;\n  }\n\n  private toUnicode(str: string): string {\n    return escape(str).toLowerCase().replace(/%u/gi, '\\\\u');\n  }\n\n  private getShortenedUrl(): string {\n    if (!this.fileSrc.tmpPath) return '';\n    const parts = this.fileSrc.tmpPath.split('/');\n    return parts.length >= 3\n      ? `${parts[0]}/${parts[1]}/${parts[2]}`\n      : this.fileSrc.tmpPath;\n  }\n\n  private wrapText(\n    ctx: CanvasRenderingContext2D,\n    text: string,\n    x: number,\n    y: number,\n    maxWidth: number,\n    lineHeight: number\n  ): void {\n    const words = text.includes(' ') ? text.split(' ') : text.split('');\n    let line = '';\n    let lineCount = 1;\n    let tempLine = '';\n    let currentY = y;\n\n    for (let n = 0; n < words.length; n++) {\n      if (lineCount === 3) return;\n\n      const testLine = line + (words[n] + (words.length > 1 ? ' ' : ''));\n      const metrics = ctx.measureText(testLine);\n      const testWidth = metrics.width;\n\n      if (testWidth > maxWidth && n > 0) {\n        if (lineCount === 2) {\n          line = `${line.slice(0, -3)}...`;\n        }\n        ctx.fillText(line, x, currentY);\n        line = words[n] + ' ';\n        currentY += lineHeight;\n        lineCount++;\n      } else {\n        line = testLine;\n      }\n    }\n\n    if (lineCount < 3) {\n      ctx.fillText(line, x, currentY);\n    }\n  }\n\n  getFileIconURL(objType: WidgetFileType): string {\n    return FILE_ICON_PATHS[objType] || FILE_ICON_PATHS['XFile'];\n  }\n\n  async loadPreviewImage(\n    previewImage: string,\n    fileName: string\n  ): Promise<void> {\n    const url = previewImage || this.getFileIconURL(XFile.objType);\n    try {\n      this._previewImage = await loadImage(url, { crossOrigin: 'anonymous' });\n      this.dirty = true;\n      this.canvas?.requestRenderAll();\n    } catch (error) {\n      console.error('Failed to load preview image:', error);\n    }\n  }\n}\n\nclassRegistry.setClass(XFile);\n","import { XFile } from './XFile';\nimport { XFileProps } from './XFile';\nimport { classRegistry } from '../../../ClassRegistry';\n\nexport class XFileAudio extends XFile {\n  constructor(options: Partial<XFileProps>) {\n    super(options);\n    this.type = 'XFileAudio';\n    this.objType = 'XFileAudio';\n  }\n}\nclassRegistry.setClass(XFileAudio);\n","import { XFile } from './XFile';\nimport { XFileProps } from './XFile';\nimport { classRegistry } from '../../../ClassRegistry';\n\nexport class XFileVideo extends XFile {\n  constructor(options: Partial<XFileProps>) {\n    super(options);\n    this.type = 'XFileVideo';\n    this.objType = 'XFileVideo';\n  }\n}\n\nclassRegistry.setClass(XFileVideo);\n","import { XFile } from './XFile';\nimport { XFileProps } from './XFile';\nimport { classRegistry } from '../../../ClassRegistry';\n\nexport class XFileExcel extends XFile {\n  constructor(options: Partial<XFileProps>) {\n    super(options);\n    this.type = 'XFileExcel';\n    this.objType = 'XFileExcel';\n  }\n}\nclassRegistry.setClass(XFileExcel);\n","import { XFile } from './XFile';\nimport { XFileProps } from './XFile';\nimport { classRegistry } from '../../../ClassRegistry';\n\nexport class XFileWord extends XFile {\n  constructor(options: Partial<XFileProps>) {\n    super(options);\n    this.type = 'XFileWord';\n    this.objType = 'XFileWord';\n  }\n}\n\nclassRegistry.setClass(XFileWord);\n","import { XFile } from './XFile';\nimport { XFileProps } from './XFile';\nimport { classRegistry } from '../../../ClassRegistry';\n\nexport class XFileZip extends XFile {\n  constructor(options: Partial<XFileProps>) {\n    super(options);\n    this.type = 'XFileZip';\n    this.objType = 'XFileZip';\n  }\n}\n\nclassRegistry.setClass(XFileZip);\n","import { XFile } from './XFile';\nimport { XFileProps } from './XFile';\nimport { classRegistry } from '../../../ClassRegistry';\n\nexport class XFilePDF extends XFile {\n  constructor(options: Partial<XFileProps>) {\n    super(options);\n    this.type = 'XFilePDF';\n    this.objType = 'XFilePDF';\n  }\n}\nclassRegistry.setClass(XFilePDF);\n","import { XFile } from './XFile';\nimport { XFileProps } from './XFile';\nimport { classRegistry } from '../../../ClassRegistry';\n\nexport class XFilePPT extends XFile {\n  constructor(options: Partial<XFileProps>) {\n    super(options);\n    this.type = 'XFilePPT';\n    this.objType = 'XFilePPT';\n  }\n}\n\nclassRegistry.setClass(XFilePPT);\n","import  {WidgetBaseInterface, TOriginX, TOriginY } from './widget.entity.base';\nimport { WidgetType } from './widget.type';\n\nexport   interface WidgetGroupInterface extends WidgetBaseInterface {\n  objectArr: any[];\n}\n\nexport class WidgetGroupClass implements WidgetGroupInterface {\n  updatedBy: string = \"\";\n  updatedByName: string = \"\";\n   createdByName: string=\"\";\n  objectArr: any[] = [];\n  id: string = '';\n  boardId: string = '';\n  backgroundColor: string = 'transparent';\n  width: number = 0;\n  height: number = 0;\n  left: number = 0;\n  locked: boolean = false;\n  objType: WidgetType = \"XGroup\"; // Assuming WidgetType.Default is a valid enum value\n  originX: TOriginX = 'center'; // Assuming 'left' is a valid TOriginX value\n  originY: TOriginY = 'center'; // Assuming 'top' is a valid TOriginY value\n  scaleX: number = 1;\n  scaleY: number = 1;\n  selectable: boolean = true;\n  top: number = 0;\n  zIndex: number = Date.now() *100;\n  version: string = '1.0';\n  updatedAt: number = Date.now();\n   createdAt: number = Date.now();\n  createdBy: string = '';\n  visible: boolean = true;\n}\n\nexport const EntityKeys = Object.keys(new WidgetGroupClass()) as (keyof WidgetGroupInterface)[];\n","import { classRegistry } from '../../ClassRegistry';\nimport { Group } from '../Group';\n\nimport { WidgetType } from './type/widget.type';\nimport { EntityKeys } from './type/widget.entity.group';\nimport { WidgetGroupInterface } from './type/widget.entity.group';\n\n\nexport class XGroup extends Group implements WidgetGroupInterface {\n  static type: WidgetType = 'XGroup';\n  static objType: WidgetType = 'XGroup';\n\n\n  constructor(objects: any, options: any) {\n\n\n    super(objects, options);\n    Object.assign(this, options);\n    this.objType = 'XGroup';\n    this.cornerColor = 'white';\n    this.cornerSize = 10;\n    this.cornerStyle = 'circle';\n    this.transparentCorners = false;\n    this.cornerStrokeColor = 'gray';\n  }\n  updatedBy: string;\n  updatedByName: string;\n\n  createdByName: string;\n  objectArr: any[];\n  boardId: string;\n  objType: WidgetType;\n  userId: string;\n  zIndex: number;\n  version: string;\n  updatedAt: number;\n\n  createdAt: number;\n  createdBy: string;\n\n\n  getObject() {\n    const entityKeys: string[] = EntityKeys;\n    const result: Record<string, any> = {};\n\n    entityKeys.forEach((key) => {\n      if (key in this) {\n        result[key] = (this as any)[key];\n      }\n    });\n\n    return result;\n  }\n\n  //  override the default behavior of `getText` to return a concatenated string of all text objects\\\n  // canvasX custoom method\n  getText(): any {\n    if (this.getObjects().length > 1) {\n      const textsArray = this.getObjects().map((item) => item.getText());\n      return textsArray.join('/n').trim();\n    } else {\n      return '';\n    }\n  }\n}\nclassRegistry.setClass(XGroup);\n","import { WidgetBaseInterface, TOriginX, TOriginY } from './widget.entity.base';\nimport { FileObject } from './file';\nimport { WidgetType } from './widget.type';\nexport interface WidgetURLInterface extends WidgetBaseInterface {\n  transcription: string;\n  vectorSrc: FileObject | null;\n  url: string;\n  fileName: string;\n  previewImage: FileObject | null;\n}\n\n\nexport class WidgetURLClass implements WidgetURLInterface {\n  updatedBy: string = \"\";\n  updatedByName: string = \"\";\n  createdByName: string = \"\";\n  transcription: string = '';\n  vectorSrc: FileObject | null = null;\n  url: string  = \"https://www.boardx.us\";\n  fileName: string = '';\n  previewImage: FileObject | null = null;\n  id: string = '';\n  boardId: string = '';\n  backgroundColor: string = '#FFFFFF';\n  width: number = 0;\n  height: number = 0;\n  left: number = 0;\n  locked: boolean = false;\n  objType: WidgetType = \"XURL\";\n  originX: TOriginX = 'center';\n  originY: TOriginY = 'center';\n  scaleX: number = 1;\n  scaleY: number = 1;\n  selectable: boolean = true;\n  top: number = 0;\n  zIndex: number = Date.now() * 100;\n  version: string = '1.0';\n  updatedAt: number = Date.now();\n  createdAt: number = Date.now();\n  createdBy: string = '';\n  visible: boolean = true;\n}\n\n\nexport const EntityKeys = Object.keys(new WidgetURLClass()) as string[];","import { classRegistry } from '../../../ClassRegistry';\nimport { Shadow } from '../../../Shadow';\nimport { getFabricWindow } from '../../../env';\nimport { loadImage } from '../../../util/misc/objectEnlive';\n\nimport { ImageProps } from '../../Image';\nimport { FabricObject } from '../../Object/FabricObject';\nimport { Rect } from '../../Rect';\nimport { FileObject } from '../type/file';\n\nimport { WidgetURLInterface, EntityKeys } from '../type/widget.entity.url';\nimport { WidgetType } from '../type/widget.type';\n\nexport type XURLProps = ImageProps & WidgetURLInterface;\n\nexport const XURLDefaultValues: Partial<XURLProps> = {\n  originX: 'center',\n  originY: 'center',\n  cornerColor: 'white',\n  cornerStrokeColor: 'gray',\n  cornerSize: 10,\n  cornerStyle: 'circle',\n  transparentCorners: false,\n};\n\nexport class XURL extends FabricObject implements WidgetURLInterface {\n  static objType = 'XURL';\n  static type = 'XURL';\n  transcription: string;\n\n  _previewImage: HTMLImageElement | null = null;\n\n  public extendedProperties = [\n    'id',\n    'objType',\n    'fileName',\n    'transcription',\n    'vectorSrc',\n    'fileSrc',\n    'previewImage',\n    'description',\n    'userId',\n    'clientId',\n    'zIndex',\n    'locked',\n    'boardId',\n  ];\n  constructor(url: any, options: Partial<XURLProps>) {\n    super(options);\n\n    const previewImageURL = options.previewImage?.tmpPath\n      ? options.previewImage?.tmpPath\n      : '/boardxstatic/fileIcons/weblink.png';\n\n    Object.assign(this, options);\n\n    this.on('mousedblclick', this.onDoubleClick.bind(this));\n    this.objType = 'XURL';\n\n    (this.cornerColor = 'white'),\n      (this.cornerStrokeColor = 'gray'),\n      (this.cornerSize = 15),\n      (this.cornerStyle = 'circle'),\n      (this.transparentCorners = false),\n      (this.shadow = new Shadow({\n        color: 'rgba(217, 161, 177, 0.54)',\n        offsetX: 1,\n        offsetY: 2,\n        blur: 4,\n        id: 310,\n      }));\n    this.clipPath = new Rect({\n      left: 0,\n      top: 0,\n      rx: 8,\n      ry: 8,\n      width: 230,\n      height: 248,\n      fill: '#000000',\n    });\n    this.width = 230;\n    this.height = 248;\n    this.loadPreviewImage(previewImageURL!);\n  }\n  url: string;\n  updatedBy: string;\n  updatedByName: string;\n\n  createdByName: string;\n  vectorSrc: FileObject;\n  fileSrc: FileObject;\n  fileName: string;\n  previewImage: FileObject;\n  boardId: string;\n  objType: WidgetType;\n  userId: string;\n  zIndex: number;\n  version: string;\n  updatedAt: number;\n\n  createdAt: number;\n  createdBy: string;\n\n  getObject() {\n    const entityKeys: string[] = EntityKeys;\n    const result: Record<string, any> = {};\n\n    entityKeys.forEach((key) => {\n      if (key in this) {\n        result[key] = (this as any)[key];\n      }\n    });\n\n    return result;\n  }\n\n  toObject(propertiesToInclude: Array<any>): any {\n    return super.toObject([...this.extendedProperties, ...propertiesToInclude]);\n  }\n  onDoubleClick() {\n    getFabricWindow().open(this.url, '_blank');\n  }\n\n  drawObject(ctx: CanvasRenderingContext2D) {\n    // Draw solid background to eliminate transparency\n    ctx.beginPath();\n    ctx.fillStyle = '#FFFFFF'; // Set to white or any preferred color\n    ctx.fillRect(-this.width / 2, -this.height / 2, this.width, this.height);\n    ctx.lineWidth = 0;\n    ctx.strokeStyle = '#ffffff';\n    ctx.moveTo(-this.width / 2, -this.height / 2);\n    ctx.stroke();\n\n    const imgWidth = 230;\n    const imgHeight = 160;\n\n    if (this._previewImage) {\n      this.drawPreviewImage(ctx, imgWidth, imgHeight);\n    }\n\n    this.renderTitle(ctx, this.fileName);\n    this._renderStroke(ctx);\n  }\n\n  private drawPreviewImage(\n    ctx: CanvasRenderingContext2D,\n    imgWidth: number,\n    imgHeight: number\n  ) {\n    const previewImage = this._previewImage!;\n    const imageWidth = previewImage.width;\n    const imageHeight = previewImage.height;\n\n    // Calculate aspect ratios\n    const imageAspect = imageWidth / imageHeight;\n    const canvasAspect = imgWidth / imgHeight;\n\n    let drawWidth, drawHeight, offsetX, offsetY;\n\n    if (imageAspect > canvasAspect) {\n      // Image is wider than canvas aspect ratio\n      drawHeight = imgHeight;\n      drawWidth = imageAspect * imgHeight;\n      offsetX = (imgWidth - drawWidth) / 2;\n      offsetY = 0;\n    } else {\n      // Image is taller than canvas aspect ratio\n      drawWidth = imgWidth;\n      drawHeight = imgWidth / imageAspect;\n      offsetX = 0;\n      offsetY = (imgHeight - drawHeight) / 2;\n    }\n\n    ctx.save();\n\n    // Set clipping region to the image area\n    ctx.beginPath();\n    ctx.rect(-this.width / 2, -this.height / 2, imgWidth, imgHeight);\n    ctx.clip();\n\n    // Draw the image within the clipping region\n    ctx.drawImage(\n      previewImage,\n      -this.width / 2 + offsetX,\n      -this.height / 2 + offsetY,\n      drawWidth,\n      drawHeight\n    );\n\n    ctx.restore();\n  }\n\n  renderTitle(ctx: any, title: string) {\n    const maxWidth = this.width;\n    const x = -this.width / 2;\n    const y = this.height / 2 - 60;\n\n    // Set font styles\n    ctx.font = '16px Arial';\n    ctx.fillStyle = 'rgba(255, 255, 255, 1)';\n\n    // White background behind the title\n    ctx.fillRect(x, y - 29, maxWidth, 90);\n    ctx.fillStyle = '#190FA1';\n\n    // Handle null or empty title\n    if (!title && this.fileSrc?.tmpPath) {\n      const firstChar = this.fileSrc?.tmpPath.indexOf('.');\n      const lastChar = this.fileSrc?.tmpPath.indexOf('.', firstChar + 1);\n      title = this.fileSrc?.tmpPath.substring(firstChar + 1, lastChar);\n    }\n\n    // Title rendering\n    this.wrapText(ctx, title, x + 15, y - 5, maxWidth - 20, 23);\n\n    // URL rendering\n    const newurl = this.fileSrc && this.fileSrc?.tmpPath\n      ? `${this.fileSrc?.tmpPath.split('/')[0]}/${this.fileSrc.tmpPath.split('/')[1]\n      }/${this.fileSrc?.tmpPath.split('/')[2]}`\n      : '';\n    ctx.font = '12px Inter';\n    ctx.fillStyle = 'rgba(35, 41, 48, 0.65)';\n    this.wrapText(ctx, newurl, x + 15, y + 45, maxWidth - 20, 25);\n  }\n\n  wrapText(\n    context: any,\n    text: string,\n    x: number,\n    y: number,\n    maxWidth: number,\n    lineHeight: number\n  ) {\n    let line = '';\n    let lineCount = 0;\n    const chars = text.split('');\n    for (let n = 0; n < chars.length; n++) {\n      const testLine = line + chars[n];\n      const metrics = context.measureText(testLine);\n      const testWidth = metrics.width;\n\n      if (testWidth > maxWidth && n > 0) {\n        if (lineCount === 2) {\n          line = line.substring(0, line.length - 3) + '...';\n          context.fillText(line, x, y);\n          return;\n        }\n        context.fillText(line, x, y);\n        line = chars[n];\n        y += lineHeight;\n        lineCount++;\n      } else {\n        line = testLine;\n      }\n    }\n    context.fillText(line, x, y);\n  }\n\n  async loadPreviewImage(previewImage: string) {\n    const url = previewImage;\n\n    const loadedImg = await loadImage(url, {\n      crossOrigin: 'anonymous',\n    });\n    this._previewImage = loadedImg;\n    this.dirty = true;\n    this.canvas?.requestRenderAll();\n  }\n}\n\nclassRegistry.setClass(XURL);\n","import   { WidgetBaseInterface, TOriginX, TOriginY } from './widget.entity.base';\nimport { WidgetType } from './widget.type';\n\nexport   interface WidgetMarkdownInterface extends WidgetBaseInterface {\n  markdownText: string;\n}\n\nexport class WidgetMarkdownClass implements WidgetBaseInterface {\n  updatedBy: string = \"\";\n  updatedByName: string = \"\";\n   createdByName: string=\"\";\n  id: string = '';\n  boardId: string = '';\n  backgroundColor: string = '#FFFFFF';\n  width: number = 100;\n  height: number = 100;\n  left: number = 0;\n  locked: boolean = false;\n  objType: WidgetType = \"XMarkdown\";\n  originX: TOriginX = 'center';\n  originY: TOriginY = 'center';\n  scaleX: number = 1;\n  scaleY: number = 1;\n  selectable: boolean = true;\n  top: number = 0;\n  zIndex: number = Date.now() *100;\n  version: string = '1.0';\n  updatedAt: number = Date.now();\n   createdAt: number = Date.now();\n  createdBy: string = '';\n  visible: boolean = true;\n}\n\n\nexport const EntityKeys = Object.keys(new WidgetMarkdownClass()) as (keyof WidgetMarkdownInterface)[];","import { classRegistry } from '../../ClassRegistry';\nimport { XTextbase } from '../canvasx/XTextbase';\nimport html2canvas from 'html2canvas';\nimport { FabricImage } from '../../shapes/Image';\nimport hljs from 'highlight.js'; // https://highlightjs.org\n\n//@ts-ignore\nimport markdownit from 'markdown-it';\n//@ts-ignore\nimport javascript from 'highlight.js/lib/languages/javascript';\nimport { WidgetMarkdownInterface, EntityKeys } from './type/widget.entity.markdown';\nimport { WidgetType } from './type/widget.type';\n\nhljs.registerLanguage('javascript', javascript);\n\n\n\nclass XMarkdown extends XTextbase implements WidgetMarkdownInterface {\n  public markdownText: string;\n  isEditing: boolean = false;\n  private renderedImage: FabricImage | null = null;\n  private md: any;\n\n  static type: WidgetType = 'XMarkdown';\n  static objType: WidgetType = 'XMarkdown';\n  constructor(text: string, options: WidgetMarkdownInterface) {\n\n    super(text, options);\n    this.markdownText = options?.markdownText || text;\n    this.objType = 'XMarkdown';\n    // full options list (defaults)\n    this.md = markdownit({\n      // Enable HTML tags in source\n      html: true,\n\n      // Use '/' to close single tags (<br />).\n      // This is only for full CommonMark compatibility.\n      xhtmlOut: false,\n\n      // Convert '\\n' in paragraphs into <br>\n      breaks: false,\n\n      // CSS language prefix for fenced blocks. Can be\n      // useful for external highlighters.\n      langPrefix: 'language-',\n\n      // Autoconvert URL-like text to links\n      linkify: false,\n\n      // Enable some language-neutral replacement + quotes beautification\n      // For the full list of replacements, see https://github.com/markdown-it/markdown-it/blob/master/lib/rules_core/replacements.mjs\n      typographer: true,\n\n      // Double + single quotes replacement pairs, when typographer enabled,\n      // and smartquotes on. Could be either a String or an Array.\n      //\n      // For example, you can use '«»„“' for Russian, '„“‚‘' for German,\n      // and ['«\\xA0', '\\xA0»', '‹\\xA0', '\\xA0›'] for French (including nbsp).\n      quotes: '“”‘’',\n\n      // Highlighter function. Should return escaped HTML,\n      // or '' if the source string is not changed and should be escaped externally.\n      // If result starts with <pre... internal wrapper is skipped.\n      highlight: function (str: string, lang: string) {\n        if (lang && hljs.getLanguage(lang)) {\n          try {\n            return hljs.highlight(str, { language: lang }).value;\n          } catch (__) { }\n        }\n\n        return ''; // use external default escaping\n      },\n    });\n    Object.assign(this, options);\n\n    this.on('editing:entered', this.onEditingEntered.bind(this));\n    this.on('editing:exited', this.onEditingExited.bind(this));\n    this.on('scaling', this.onScaling.bind(this));\n    this.on('resizing', this.onScaled.bind(this));\n    this.renderMarkdown();\n  }\n\n\n\n  private onEditingEntered() {\n    this.isEditing = true;\n    this.text = this.markdownText;\n    this.dirty = true;\n    this.canvas?.renderAll();\n  }\n\n  private onEditingExited() {\n    this.isEditing = false;\n    this.markdownText = this.text;\n    this.renderMarkdown();\n  }\n\n  private onScaling() {\n    this.renderMarkdown();\n    // Optional: You can add any logic needed during scaling\n  }\n\n  private onScaled() {\n    this.renderMarkdown();\n  }\n\n  async renderMarkdown() {\n    if (this.isEditing) return;\n\n    const html = this.md.render(this.markdownText);\n    this.parseHtmlToImage(html).then((img) => {\n      this.renderedImage = img;\n      this.dirty = true;\n      this.canvas?.renderAll();\n    });\n  }\n\n\n  getObject() {\n    const entityKeys: string[] = EntityKeys;\n    const result: Record<string, any> = {};\n\n    entityKeys.forEach((key) => {\n      if (key in this) {\n        result[key] = (this as any)[key];\n      }\n    });\n\n    return result;\n  }\n\n  private parseHtmlToImage(html: string): Promise<FabricImage> {\n    return new Promise((resolve) => {\n      const blob = new Blob([html], { type: 'text/html' });\n      const url = URL.createObjectURL(blob);\n\n      const iframe = document.createElement('iframe');\n      iframe.style.width = `${this.width}px`;\n      iframe.style.height = `${this.height}px`;\n      iframe.style.visibility = 'hidden';\n      iframe.style.position = 'absolute';\n      iframe.style.left = '-9999px';\n      document.body.appendChild(iframe);\n\n      iframe.onload = async () => {\n        if (iframe.contentDocument) {\n          const style = iframe.contentDocument.createElement('style');\n          style.textContent = `\n            body { \n              font-family: Arial, sans-serif; \n              font-size: 14px; \n              line-height: 1.6; \n              color: #333; \n              padding: 20px; \n              margin: 0;\n              width: ${this.width}px;\n              height: ${this.height}px;\n            }\n            h1, h2, h3, h4, h5, h6 { margin-top: 0; }\n            ul, ol { padding-left: 20px; }\n          /* Code block styling */\ncode {\n  background-color: #f5f5f5; /* Slightly lighter background for better contrast */\n  padding: 3px 5px; /* More padding for better spacing */\n  border-radius: 5px; /* Smoother border radius */\n  font-family: 'Courier New', Courier, monospace; /* Monospace font for code */\n}\n\npre {\n  background-color: #f5f5f5; /* Matching background color with inline code */\n  padding: 12px; /* Slightly more padding for comfort */\n  border-radius: 5px; /* Smoother border radius */\n  overflow-x: auto; /* Ensure horizontal scroll for long code lines */\n  font-family: 'Courier New', Courier, monospace; /* Monospace font for code */\n}\n\npre code {\n  display: block;\n}\n\n/* Table styling */\ntable {\n  border-collapse: collapse;\n \n  margin-bottom: 1em; /* Add some margin below tables */\n  font-family: Arial, sans-serif; /* Better font for readability */\n}\n\ntable, th, td {\n  border: 1px solid #ddd; /* Lighter border color for a cleaner look */\n}\n\nth, td {\n  padding: 10px; /* More padding for better spacing */\n  text-align: left;\n}\n\nth {\n  background-color: #f2f2f2; /* Slight background color for headers */\n  font-weight: bold; /* Bold headers */\n}\n\ntd {\n  background-color: #fff; /* Ensure a white background for table cells */\n}\n\n/* Additional table row hover effect */\ntr:hover {\n  background-color: #f1f1f1; /* Highlight row on hover */\n}\n          `;\n          iframe.contentDocument.head.appendChild(style);\n        }\n\n        iframe.contentDocument!.body.innerHTML = html;\n\n        await new Promise((resolve) => setTimeout(resolve, 100)); // Small delay to ensure styles are applied\n\n        const canvas = await html2canvas(\n          iframe.contentDocument?.body as HTMLElement\n        );\n        const img = await FabricImage.fromURL(canvas.toDataURL());\n        img.set({\n          left: this.left,\n          top: this.top,\n          width: this.width,\n          height: this.height,\n          scaleX: 1,\n          scaleY: 1,\n        });\n        resolve(img);\n\n        document.body.removeChild(iframe);\n        URL.revokeObjectURL(url);\n      };\n\n      iframe.src = url;\n    });\n  }\n\n  setMarkdown(newMarkdownText: string) {\n    this.markdownText = newMarkdownText;\n    if (this.isEditing) {\n      this.text = this.markdownText;\n      this.dirty = true;\n    } else {\n      this.renderMarkdown();\n    }\n    this.canvas?.renderAll();\n  }\n\n  _render(ctx: CanvasRenderingContext2D) {\n    if (this.isEditing) {\n      super._render(ctx);\n    } else if (this.renderedImage) {\n      this.renderedImage._render(ctx);\n    }\n  }\n}\n\nexport { XMarkdown };\n\nclassRegistry.setClass(XMarkdown);\n","import { WidgetBaseInterface, TOriginX, TOriginY } from './widget.entity.base';\nimport { WidgetType } from './widget.type';\n\nexport interface WidgetChartInterface extends WidgetBaseInterface {\n  chartOptions: any;\n\n}\n\nclass WidgetChartClass implements WidgetChartInterface {\n  updatedBy: string = \"\";\n  updatedByName: string = \"\";\n  createdByName: string = \"\";\n  id: string = '';\n  boardId: string = '';\n  backgroundColor: string = '';\n  width: number = 400;\n  height: number =500;\n  left: number = 0;\n  locked: boolean = false;\n  objType: WidgetType = \"XChart\"; // Replace with an appropriate default value\n  originX: TOriginX = 'center'; // Replace with an appropriate default value\n  originY: TOriginY = 'center'; // Replace with an appropriate default value\n  scaleX: number = 1;\n  scaleY: number = 1;\n  selectable: boolean = true;\n  top: number = 0;\n  zIndex: number = Date.now() *100;\n  version: string = '';\n  updatedAt: number = Date.now();\n  createdAt: number = Date.now();\n  createdBy: string = '';\n  visible: boolean = true;\n  chartOptions: any = {};\n}\n\nexport const EntityKeys = Object.keys(new WidgetChartClass()) as (keyof WidgetChartInterface)[];","import { ObjectProps } from '../Object/types/ObjectProps';\nimport { FabricObject } from '../Object/FabricObject';\nimport { Chart, ChartConfiguration, registerables } from 'chart.js';\nimport { classRegistry } from '../../ClassRegistry';\nimport { createCanvasElement } from '../../util/misc/dom';\nimport { WidgetChartInterface, EntityKeys } from './type/widget.entity.chart';\nimport { WidgetType } from './type/widget.type';\n\nChart.register(...registerables);\n\ninterface ChartObjectOptions extends ObjectProps {\n  chartConfig: ChartConfiguration;\n  width: number;\n  height: number;\n}\n\nclass XChart extends FabricObject implements WidgetChartInterface {\n  private chartConfig: ChartConfiguration;\n  private chartInstance: Chart | null = null;\n  private canvasElement: HTMLCanvasElement | null = null;\n  private needsUpdate: boolean = true; // Flag to track if update is needed\n\n  static type: WidgetType = 'XChart';\n  static objType: WidgetType = 'XChart';\n\n  constructor(options: any) {\n\n    super(options);\n    this.chartConfig = options.chartConfig!;\n    this.width = options.width!;\n    this.height = options.height!;\n    this.createCanvasElement();\n    this.addDoubleClickEventListener();\n    Object.assign(this, options);\n    this.objType = 'XChart';\n  }\n  updatedBy: string;\n  updatedByName: string;\n\n  createdByName: string;\n  chartOptions: any;\n  boardId: string;\n  objType: WidgetType;\n  userId: string;\n  zIndex: number;\n  version: string;\n  updatedAt: number;\n\n  createdAt: number;\n  createdBy: string;\n\n  private createCanvasElement() {\n    if (!this.canvasElement) {\n      this.canvasElement = createCanvasElement();\n      this.canvasElement.width = this.width!;\n      this.canvasElement.height = this.height!;\n      this.canvasElement.style.width = `${this.width}px`;\n      this.canvasElement.style.height = `${this.height}px`;\n      this.canvasElement.style.position = 'absolute';\n      this.canvasElement.style.top = `${-1000000}px`;\n      this.canvasElement.style.left = `${-1000000}px`;\n\n      document.body.appendChild(this.canvasElement); // Ensure canvas is in the DOM\n    }\n  }\n\n  private async createOrUpdateChart() {\n    if (!this.canvasElement) {\n      this.createCanvasElement();\n    }\n\n    // Ensure the canvas element is part of the DOM\n    if (!document.body.contains(this.canvasElement!)) {\n      document.body.appendChild(this.canvasElement!);\n    }\n\n    if (this.chartInstance) {\n      this.chartInstance.destroy();\n    }\n\n    return new Promise<void>((resolve) => {\n      this.canvasElement!.width = this.width!;\n      this.canvasElement!.height = this.height!;\n      this.canvasElement!.style.width = `${this.width}px`;\n      this.canvasElement!.style.height = `${this.height}px`;\n\n      this.chartInstance = new Chart(\n        this.canvasElement!.getContext('2d')!,\n        this.chartConfig\n      );\n\n      // Wait for the next animation frame to ensure the chart has rendered\n      requestAnimationFrame(() => {\n        this.chartInstance!.resize(); // Force a resize to ensure proper rendering\n        resolve();\n      });\n    });\n  }\n  getObject() {\n    const entityKeys: string[] = EntityKeys;\n    const result: Record<string, any> = {};\n\n    entityKeys.forEach((key) => {\n      if (key in this) {\n        result[key] = (this as any)[key];\n      }\n    });\n\n    return result;\n  }\n\n\n  async _render(ctx: CanvasRenderingContext2D) {\n    if (this.needsUpdate) {\n      await this.createOrUpdateChart();\n      this.needsUpdate = false; // Reset the flag after updating\n    }\n\n    if (\n      this.canvasElement &&\n      this.canvasElement.width > 0 &&\n      this.canvasElement.height > 0\n    ) {\n      ctx.drawImage(\n        this.canvasElement,\n        -this.width! / 2,\n        -this.height! / 2,\n        this.width!,\n        this.height!\n      );\n    }\n  }\n\n  updateChart(newConfig: ChartConfiguration) {\n    this.chartConfig = newConfig;\n    this.needsUpdate = true; // Set the flag to indicate an update is needed\n    this.dirty = true;\n    this.canvas?.requestRenderAll();\n  }\n\n  private addDoubleClickEventListener() {\n    this.on('mousedblclick', () => {\n      this.openEditModal();\n    });\n  }\n\n  private openEditModal() {\n    // Implement modal opening logic here\n    const modal = document.createElement('div');\n    modal.style.position = 'fixed';\n    modal.style.top = '50%';\n    modal.style.left = '50%';\n    modal.style.width = '400px';\n    modal.style.transform = 'translate(-50%, -50%)';\n    modal.style.padding = '20px';\n    modal.style.backgroundColor = 'white';\n    modal.style.boxShadow = '0 0 10px rgba(0, 0, 0, 0.5)';\n    modal.innerHTML = `\n      <h2>Edit Chart Data</h2>\n      <textarea id=\"chartDataInput\" rows=\"10\" cols=\"40\">${JSON.stringify(\n      this.chartConfig.data,\n      null,\n      2\n    )}</textarea>\n      <br />\n      <button id=\"saveChartData\">Save</button>\n      <button id=\"cancelEdit\">Cancel</button>\n    `;\n    document.body.appendChild(modal);\n\n    const saveButton = modal.querySelector('#saveChartData')!;\n    const cancelButton = modal.querySelector('#cancelEdit')!;\n    const chartDataInput = modal.querySelector(\n      '#chartDataInput'\n    ) as HTMLTextAreaElement;\n\n    saveButton.addEventListener('click', () => {\n      const newChartData = JSON.parse(chartDataInput.value);\n      this.chartConfig.data = newChartData;\n      this.updateChart(this.chartConfig);\n      document.body.removeChild(modal);\n    });\n\n    cancelButton.addEventListener('click', () => {\n      document.body.removeChild(modal);\n    });\n  }\n}\n\nexport { XChart };\n\nclassRegistry.setClass(XChart);\n","import   {WidgetBaseInterface, TOriginX, TOriginY } from './widget.entity.base';\nimport { WidgetType } from './widget.type';\n\nexport   interface WidgetFrameInterface extends WidgetBaseInterface {\n  titleText: string;\n}\n\nexport class WidgetFrameClass implements WidgetFrameInterface {\n  updatedBy: string = \"\";\n  updatedByName: string = \"\";\n   createdByName: string=\"\";\n  titleText: string = '';\n  id: string = '';\n  boardId: string = '';\n  backgroundColor: string = 'white';\n  width: number = 500;\n  height: number = 500;\n  left: number = 0;\n  locked: boolean = false;\n  objType: WidgetType = \"XFrame\"; // Assuming WidgetType.Default is a valid enum value\n  originX: TOriginX = 'center'; // Assuming 'left' is a valid TOriginX value\n  originY: TOriginY = 'center'; // Assuming 'top' is a valid TOriginY value\n  scaleX: number = 1;\n  scaleY: number = 1;\n  selectable: boolean = true;\n  top: number = 0;\n  zIndex: number = Date.now() *100;\n  version: string = '1.0';\n  updatedAt: number = Date.now();\n   createdAt: number = Date.now();\n  createdBy: string = '';\n  visible: boolean = true;\n}\n\nexport const EntityKeys = Object.keys(new WidgetFrameClass()) as (keyof WidgetFrameInterface)[];\n","import { FabricObject } from '../Object/FabricObject';\nimport { Textbox } from '../Textbox';\nimport { Rect } from '../Rect';\nimport { Group } from '../Group';\nimport { LayoutManager } from '../../LayoutManager';\nimport { FitContentLayout } from '../../LayoutManager/LayoutStrategies/FitContentLayout';\nimport { classRegistry } from '../../ClassRegistry';\nimport { EntityKeys, WidgetFrameInterface } from './type/widget.entity.frame';\nimport { WidgetType } from './type/widget.type';\n\nclass XFrame extends FabricObject implements WidgetFrameInterface {\n  title: Textbox;\n  titleText: string;\n  body: Rect;\n  objects: Group;\n  layoutManager: LayoutManager;\n  static type: WidgetType = 'XFrame';\n  static objType: WidgetType = 'XFrame';\n\n  constructor(options: Partial<WidgetFrameInterface> = {}) {\n\n    super(options);\n    Object.assign(this, options);\n    this.titleText = 'Frame Title',\n      this.left = 100,\n      this.top = 100,\n      this.width = 400,\n      this.height = 300\n    this.title = new Textbox(this.titleText, {\n      left: this.left,\n      top: this.top,\n      fontSize: 18,\n      fontWeight: 'bold',\n      selectable: true,\n      editable: true,\n    });\n    this.titleText = this.titleText;\n    this.objType = 'XFrame';\n    this.body = new Rect({\n      left: this.left,\n      top: this.top + 30,\n      width: this.width,\n      height: this.height,\n      fill: 'rgba(0,0,0,0.1)',\n      selectable: false,\n      evented: false,\n    });\n\n    this.objects = new Group([this.body, this.title], {\n      left: this.left,\n      top: this.top,\n      selectable: true,\n    });\n\n    // Create a layout manager for the frame\n    this.layoutManager = new LayoutManager(new FitContentLayout());\n\n    this.canvas!.add(this.objects);\n\n    this.title.on('mousedblclick', () => {\n      this.title.enterEditing();\n    });\n\n    this.objects.on('scaling', (event) => {\n      const scaleX = this.objects.scaleX;\n      const scaleY = this.objects.scaleY;\n\n      this.objects.getObjects().forEach((obj) => {\n        if (obj !== this.title) {\n          obj.scaleX = scaleX;\n          obj.scaleY = scaleY;\n          obj.setCoords();\n        }\n      });\n\n      this.objects.scaleX = 1;\n      this.objects.scaleY = 1;\n      this.objects.setCoords();\n\n      this.canvas!.renderAll();\n    });\n\n    this.objects.on('mousedown', (event) => {\n      if (event.target === this.title) {\n        this.objects.set('selectable', true);\n        this.canvas!.setActiveObject(this.objects);\n      }\n    });\n\n    this.objects.on('moving', (event) => {\n      //@ts-ignore\n      const deltaX = event.movementX;\n      //@ts-ignore\n      const deltaY = event.movementY;\n\n      this.objects.getObjects().forEach((obj) => {\n        if (obj !== this.body && obj !== this.title) {\n          obj.set('left', obj.left + deltaX);\n          obj.set('top', obj.top + deltaY);\n        }\n      });\n\n      this.canvas!.renderAll();\n    });\n\n    this.canvas!.on('object:moving', (e) => {\n      if (e.target !== this.objects && !this.objects.contains(e.target)) {\n        this.checkObjectInFrame(e.target);\n      }\n    });\n\n    this.canvas!.on('mouse:up', (e) => {\n      if (e.target && this.objects.contains(e.target)) {\n        this.addObject(e.target);\n      } else if (e.target && !this.objects.contains(e.target)) {\n        this.removeObject(e.target);\n      }\n    });\n\n    this.canvas!.renderAll();\n  }\n  updatedBy: string;\n  updatedByName: string;\n\n  createdByName: string;\n  boardId: string;\n  objType: WidgetType;\n  userId: string;\n  zIndex: number;\n  version: string;\n  updatedAt: number;\n\n  createdAt: number;\n  createdBy: string;\n\n  checkObjectInFrame(obj: any) {\n    const objBound = obj.getBoundingRect();\n    const frameBound = this.body.getBoundingRect();\n\n    if (\n      objBound.left >= frameBound.left &&\n      objBound.top >= frameBound.top &&\n      objBound.left + objBound.width <= frameBound.left + frameBound.width &&\n      objBound.top + objBound.height <= frameBound.top + frameBound.height\n    ) {\n      this.addObject(obj);\n    } else {\n      this.removeObject(obj);\n    }\n  }\n\n  getObject() {\n    const entityKeys: string[] = EntityKeys;\n    const result: Record<string, any> = {};\n\n    entityKeys.forEach((key) => {\n      if (key in this) {\n        result[key] = (this as any)[key];\n      }\n    });\n\n    return result;\n  }\n  addObject(obj: any) {\n    if (!this.objects.contains(obj)) {\n      this.objects.add(obj);\n      obj.set({\n        left: obj.left - this.objects.left,\n        top: obj.top - this.objects.top,\n        selectable: true,\n      });\n      this.canvas!.remove(obj);\n      this.canvas!.renderAll();\n    }\n  }\n\n  removeObject(obj: any) {\n    if (this.objects.contains(obj)) {\n      this.objects.remove(obj);\n      obj.set({\n        left: obj.left + this.objects.left,\n        top: obj.top + this.objects.top,\n        selectable: true,\n      });\n      this.canvas!.add(obj);\n      this.canvas!.renderAll();\n    }\n  }\n}\n\nclassRegistry.setClass(XFrame);\n\nexport { XFrame };\n","import { WidgetBaseInterface, TOriginX, TOriginY } from './widget.entity.base';\nimport { WidgetType } from './widget.type';\n\nexport interface WidgetPathInterface extends WidgetBaseInterface {\n  path: any[];\n  fill: string | null | any;\n  stroke: string | null | any;\n  strokeWidth: number;\n  strokeLineCap: string;\n  strokeDashOffset: number;\n  strokeLineJoin: string;\n  strokeUniform: boolean;\n  strokeMiterLimit: number;\n  fillRule: string;\n}\n\nexport class WidgetPathClass implements WidgetPathInterface {\n  updatedBy: string = \"\";\n  updatedByName: string = \"\";\n  fill: string = 'transparent';\n \n  createdByName: string = \"\";\n  path: any[] = [''];\n  stroke: string = '#000000';\n  strokeWidth: number = 1;\n  strokeLineCap: string = 'butt';\n  strokeDashOffset: number = 0;\n  strokeLineJoin: string = 'miter';\n  strokeUniform: boolean = false;\n  strokeMiterLimit: number = 10;\n  fillRule: string = 'nonzero';\n  id: string = '';\n  boardId: string = '';\n  backgroundColor: string = 'transparent';\n  width: number = 0;\n  height: number = 0;\n  left: number = 0;\n  locked: boolean = false;\n  objType: WidgetType = \"XPath\";\n  originX: TOriginX = 'center';\n  originY: TOriginY = 'center';\n  scaleX: number = 1;\n  scaleY: number = 1;\n  selectable: boolean = true;\n  top: number = 0;\n  zIndex: number = Date.now() * 100;\n  version: string = '1.0';\n  updatedAt: number = Date.now();\n  createdAt: number = Date.now();\n  createdBy: string = '';\n  visible: boolean = true;\n}\n\nexport const EntityKeys = Object.keys(new WidgetPathClass()) as string[];","import { config } from '../../config';\nimport { SHARED_ATTRIBUTES } from '../../parser/attributes';\nimport { Point, XY } from '../../Point';\nimport { makeBoundingBoxFromPoints } from '../../util/misc/boundingBoxFromPoints';\nimport { toFixed } from '../../util/misc/toFixed';\nimport {\n  getBoundsOfCurve,\n  joinPath,\n  makePathSimpler,\n  parsePath,\n} from '../../util/path';\nimport { classRegistry } from '../../ClassRegistry';\nimport { FabricObject, cacheProperties } from '../Object/FabricObject';\nimport {\n  TComplexPathData,\n  TPathSegmentInfo,\n  TSimplePathData,\n} from '../../util/path/typedefs';\nimport type { FabricObjectProps, SerializedObjectProps } from '../Object/types';\nimport type { TOptions } from '../../typedefs';\nimport type { ObjectEvents } from '../../EventTypeDefs';\nimport { TBBox, TClassProperties, TSVGReviver } from '../../typedefs';\nimport { cloneDeep } from '../../util/internals/cloneDeep';\nimport { createPathDefaultControls } from '../../controls/commonControls';\nimport { WidgetPathInterface, EntityKeys } from './type/widget.entity.path';\nimport { WidgetType } from './type/widget.type';\n\ninterface UniquePathProps {\n  sourcePath?: string;\n  path?: TSimplePathData;\n  objType: 'XPath';\n}\n\nexport interface SerializedPathProps\n  extends SerializedObjectProps,\n  UniquePathProps { }\n\nexport interface PathProps extends FabricObjectProps {\n  objType: 'XPath';\n}\n\nexport interface IPathBBox extends TBBox {\n  left: number;\n  top: number;\n  pathOffset: Point;\n}\n\nexport class XPath<\n  Props extends TOptions<PathProps> = Partial<PathProps>,\n  SProps extends SerializedPathProps = SerializedPathProps,\n  EventSpec extends ObjectEvents = ObjectEvents\n> extends FabricObject<Props, SProps, EventSpec> implements WidgetPathInterface {\n  /**\n   * Array of path points\n   * @type Array\n   * @default\n   */\n  declare path: TSimplePathData;\n\n  declare pathOffset: Point;\n\n  declare sourcePath?: string;\n\n  declare segmentsInfo?: TPathSegmentInfo[];\n\n  static cacheProperties = [...cacheProperties, 'path', 'fillRule'];\n\n  declare locked: boolean;\n\n  declare boardId: string;\n\n  declare userId: string;\n\n  declare timestamp: Date;\n\n  declare zIndex: number;\n\n  declare lines: object[];\n\n  declare relationship: object[];\n\n  declare id: string;\n\n  declare userNo: string;\n\n  declare version: string;\n\n  declare lineWidth: any;\n\n  declare radius: any;\n\n  static objType: WidgetType = 'XPath';\n  static type: WidgetType = 'XPath';\n\n\n  /**\n   * Constructor\n   * @param {TComplexPathData} path Path data (sequence of coordinates and corresponding \"command\" tokens)\n   * @param {Partial<PathProps>} [options] Options object\n   * @return {Path} thisArg\n   */\n  constructor(\n    path: TComplexPathData | string,\n    { path: _, left, top, ...options }: Partial<any> = {}\n  ) {\n\n    //fixed default value\n    options.perPixelTargetFind = true;\n\n    super(options as Props);\n    this._setPath(path || [], true);\n    typeof left === 'number' && this.set('left', left);\n    typeof top === 'number' && this.set('top', top);\n\n\n    Object.assign(this, options);\n    this.objType = 'XPath';\n  }\n  updatedBy: string;\n  updatedByName: string;\n\n  createdByName: string;\n  objType: WidgetType;\n  updatedAt: number;\n\n  createdAt: number;\n  createdBy: string;\n\n  static getDefaults() {\n    return {\n      ...super.getDefaults(),\n      controls: createPathDefaultControls(),\n    };\n  }\n\n  /**\n   * @private\n   * @param {TComplexPathData | string} path Path data (sequence of coordinates and corresponding \"command\" tokens)\n   * @param {boolean} [adjustPosition] pass true to reposition the object according to the bounding box\n   * @returns {Point} top left position of the bounding box, useful for complementary positioning\n   */\n  _setPath(path: TComplexPathData | string, adjustPosition?: boolean) {\n    this.path = makePathSimpler(Array.isArray(path) ? path : parsePath(path));\n    this.setBoundingBox(adjustPosition);\n  }\n\n  /**\n   * This function is an helper for svg import. it returns the center of the object in the svg\n   * untransformed coordinates, by look at the polyline/polygon points.\n   * @private\n   * @return {Point} center point from element coordinates\n   */\n  _findCenterFromElement(): Point {\n    const bbox = this._calcBoundsFromPath();\n    return new Point(bbox.left + bbox.width / 2, bbox.top + bbox.height / 2);\n  }\n\n  /**\n   * @private\n   * @param {CanvasRenderingContext2D} ctx context to render path on\n   */\n  _renderPathCommands(ctx: CanvasRenderingContext2D) {\n    let subpathStartX = 0,\n      subpathStartY = 0,\n      x = 0, // current x\n      y = 0, // current y\n      controlX = 0, // current control point x\n      controlY = 0; // current control point y\n    const l = -this.pathOffset.x,\n      t = -this.pathOffset.y;\n\n    ctx.beginPath();\n\n    for (const command of this.path) {\n      switch (\n      command[0] // first letter\n      ) {\n        case 'L': // lineto, absolute\n          x = command[1];\n          y = command[2];\n          ctx.lineTo(x + l, y + t);\n          break;\n\n        case 'M': // moveTo, absolute\n          x = command[1];\n          y = command[2];\n          subpathStartX = x;\n          subpathStartY = y;\n          ctx.moveTo(x + l, y + t);\n          break;\n\n        case 'C': // bezierCurveTo, absolute\n          x = command[5];\n          y = command[6];\n          controlX = command[3];\n          controlY = command[4];\n          ctx.bezierCurveTo(\n            command[1] + l,\n            command[2] + t,\n            controlX + l,\n            controlY + t,\n            x + l,\n            y + t\n          );\n          break;\n\n        case 'Q': // quadraticCurveTo, absolute\n          ctx.quadraticCurveTo(\n            command[1] + l,\n            command[2] + t,\n            command[3] + l,\n            command[4] + t\n          );\n          x = command[3];\n          y = command[4];\n          controlX = command[1];\n          controlY = command[2];\n          break;\n\n        case 'Z':\n          x = subpathStartX;\n          y = subpathStartY;\n          ctx.closePath();\n          break;\n      }\n    }\n  }\n\n  /**\n   * @private\n   * @param {CanvasRenderingContext2D} ctx context to render path on\n   */\n  _render(ctx: CanvasRenderingContext2D) {\n    this._renderPathCommands(ctx);\n    this._renderPaintInOrder(ctx);\n  }\n\n  /**\n   * Returns string representation of an instance\n   * @return {string} string representation of an instance\n   */\n  toString() {\n    return `#<Path (${this.complexity()}): { \"top\": ${this.top}, \"left\": ${this.left\n      } }>`;\n  }\n\n\n  getObject() {\n    const entityKeys: string[] = EntityKeys;\n    const result: Record<string, any> = {};\n\n    entityKeys.forEach((key) => {\n      if (key in this) {\n        result[key] = (this as any)[key];\n      }\n    });\n\n    return result;\n  }\n\n\n  /**\n   * Returns object representation of an instance\n   * @param {Array} [propertiesToInclude] Any properties that you might want to additionally include in the output\n   * @return {Object} object representation of an instance\n   */\n  toObject<\n    T extends Omit<Props & TClassProperties<this>, keyof SProps>,\n    K extends keyof T = never\n  >(propertiesToInclude: K[] = []): Pick<T, K> & SProps {\n    return {\n      ...super.toObject([...propertiesToInclude]),\n      path: cloneDeep(this.path),\n    };\n  }\n\n  /**\n   * Returns dataless object representation of an instance\n   * @param {Array} [propertiesToInclude] Any properties that you might want to additionally include in the output\n   * @return {Object} object representation of an instance\n   */\n  toDatalessObject<\n    T extends Omit<Props & TClassProperties<this>, keyof SProps>,\n    K extends keyof T = never\n  >(propertiesToInclude: K[] = []): Pick<T, K> & SProps {\n    const o = this.toObject<T, K>(propertiesToInclude);\n    if (this.sourcePath) {\n      delete o.path;\n      o.sourcePath = this.sourcePath;\n    }\n    return o;\n  }\n\n  /**\n   * Returns svg representation of an instance\n   * @return {Array} an array of strings with the specific svg representation\n   * of the instance\n   */\n  _toSVG() {\n    const path = joinPath(this.path, config.NUM_FRACTION_DIGITS);\n    return [\n      '<path ',\n      'COMMON_PARTS',\n      `d=\"${path}\" stroke-linecap=\"round\" />\\n`,\n    ];\n  }\n\n  /**\n   * @private\n   * @return the path command's translate transform attribute\n   */\n  _getOffsetTransform() {\n    const digits = config.NUM_FRACTION_DIGITS;\n    return ` translate(${toFixed(-this.pathOffset.x, digits)}, ${toFixed(\n      -this.pathOffset.y,\n      digits\n    )})`;\n  }\n\n  /**\n   * Returns svg clipPath representation of an instance\n   * @param {Function} [reviver] Method for further parsing of svg representation.\n   * @return {string} svg representation of an instance\n   */\n  toClipPathSVG(reviver: TSVGReviver): any {\n    const additionalTransform = this._getOffsetTransform();\n    return (\n      '\\t' +\n      this._createBaseClipPathSVGMarkup(this._toSVG(), {\n        reviver: reviver,\n        additionalTransform: additionalTransform,\n      })\n    );\n  }\n\n  /**\n   * Returns svg representation of an instance\n   * @param {Function} [reviver] Method for further parsing of svg representation.\n   * @return {string} svg representation of an instance\n   */\n  toSVG(reviver: TSVGReviver): any {\n    const additionalTransform = this._getOffsetTransform();\n    return this._createBaseSVGMarkup(this._toSVG(), {\n      reviver: reviver,\n      additionalTransform: additionalTransform,\n    });\n  }\n\n  /**\n   * Returns number representation of an instance complexity\n   * @return {number} complexity of this instance\n   */\n  complexity() {\n    return this.path.length;\n  }\n\n  setDimensions() {\n    this.setBoundingBox();\n  }\n\n  setBoundingBox(adjustPosition?: boolean) {\n    const { width, height, pathOffset } = this._calcDimensions();\n    this.set({ width, height, pathOffset });\n    // using pathOffset because it match the use case.\n    // if pathOffset change here we need to use left + width/2 , top + height/2\n    adjustPosition && this.setPositionByOrigin(pathOffset, 'center', 'center');\n  }\n\n  _calcBoundsFromPath(): TBBox {\n    const bounds: XY[] = [];\n    let subpathStartX = 0,\n      subpathStartY = 0,\n      x = 0, // current x\n      y = 0; // current y\n\n    for (const command of this.path) {\n      // current instruction\n      switch (\n      command[0] // first letter\n      ) {\n        case 'L': // lineto, absolute\n          x = command[1];\n          y = command[2];\n          bounds.push(new Point(subpathStartX, subpathStartY), new Point(x, y));\n          break;\n\n        case 'M': // moveTo, absolute\n          x = command[1];\n          y = command[2];\n          subpathStartX = x;\n          subpathStartY = y;\n          break;\n\n        case 'C': // bezierCurveTo, absolute\n          bounds.push(\n            ...getBoundsOfCurve(\n              x,\n              y,\n              command[1],\n              command[2],\n              command[3],\n              command[4],\n              command[5],\n              command[6]\n            )\n          );\n          x = command[5];\n          y = command[6];\n          break;\n\n        case 'Q': // quadraticCurveTo, absolute\n          bounds.push(\n            ...getBoundsOfCurve(\n              x,\n              y,\n              command[1],\n              command[2],\n              command[1],\n              command[2],\n              command[3],\n              command[4]\n            )\n          );\n          x = command[3];\n          y = command[4];\n          break;\n\n        case 'Z':\n          x = subpathStartX;\n          y = subpathStartY;\n          break;\n      }\n    }\n    return makeBoundingBoxFromPoints(bounds);\n  }\n\n  /**\n   * @private\n   */\n  _calcDimensions(): IPathBBox {\n    const bbox = this._calcBoundsFromPath();\n\n    return {\n      ...bbox,\n      pathOffset: new Point(\n        bbox.left + bbox.width / 2,\n        bbox.top + bbox.height / 2\n      ),\n    };\n  }\n\n  /**\n   * List of attribute names to account for when parsing SVG element (used by `Path.fromElement`)\n   * @static\n   * @memberOf Path\n   * @see http://www.w3.org/TR/SVG/paths.html#PathElement\n   */\n  static ATTRIBUTE_NAMES = [...SHARED_ATTRIBUTES, 'd'];\n\n  /**\n   * Creates an instance of Path from an object\n   * @static\n   * @memberOf Path\n   * @param {Object} object\n   * @returns {Promise<Path>}\n   */\n  static fromObject<T extends TOptions<SerializedPathProps>>(object: T) {\n    return this._fromObject<XPath>(object, {\n      extraParam: 'path',\n    });\n  }\n}\n\nclassRegistry.setClass(XPath);\nclassRegistry.setSVGClass(XPath);\n\n/* _FROM_SVG_START_ */\n","import { FabricObject } from '../../shapes/Object/FabricObject';\nimport { FabricObjectProps } from '../../shapes/Object/types';\nimport { ObjectEvents } from '../../EventTypeDefs';\nimport { SerializedObjectProps } from '../../shapes/Object/types';\n\nimport { Canvas } from '../Canvas';\nimport { Point } from '../../Point';\n\ninterface ConnectorControlPoints {\n  mtaStart: Point | null;\n  mbaStart: Point | null;\n  mlaStart: Point | null;\n  mraStart: Point | null;\n}\n\nconst DEFAULT_SELECTION_COLOR = 'rgba(179, 205, 253, 0.5)';\nconst DEFAULT_SELECTION_BORDER_COLOR = '#31A4F5';\nconst DEFAULT_MOVE_CURSOR = 'default';\nconst TARGET_FIND_TOLERANCE = 8;\n\nexport class XCanvas extends Canvas {\n  uniformScaling = true;\n  interactionMode = 'mouse';\n  isEnablePanMoving = false;\n  selectionFullyContained = false;\n  skipOffscreen = true;\n  preserveObjectStacking = true;\n  targetFindTolerance = TARGET_FIND_TOLERANCE;\n  stopAnimateToRectStatus = false;\n  stopAnimateObjectToPositionStatus = false;\n  moveCursor = DEFAULT_MOVE_CURSOR;\n  selectionColor = DEFAULT_SELECTION_COLOR;\n  selectionBorderColor = DEFAULT_SELECTION_BORDER_COLOR;\n  selectionLineWidth = 1;\n  fireMiddleClick = true;\n  showBackgroundDots = true;\n  whiteboardWidth = 1920 * 5;\n  whiteboardHeight = 1080 * 6;\n  isEnableTouchMoving = false;\n  conextMenuObject: Record<string, any> = {};\n  notesDrawCanvas: HTMLCanvasElement | null = null;\n  widgetPadding = 5;\n  connectorStart: Point | null = null;\n  connectorArrow: any = null;\n  vAlignLineTimer: NodeJS.Timeout | null = null;\n  hAlignLineTimer: NodeJS.Timeout | null = null;\n  isDrawingMode = false;\n  isErasingMode = false;\n  group_zIndex: number | null = null;\n  defaultNote: Record<string, any> = {};\n  boundHandlerMouseMove: ((e: any) => void) | null = null;\n  dockingWidget: FabricObject | null = null;\n  instanceOfConnector: any = null;\n  startPointOfConnector: Point | null = null;\n  endPointOfConnector: Point | null = null;\n  inConnectingMode = false;\n  toUpdateNewObjectRemote: any[] = [];\n  toUpdateRemovedObjectRemote: any[] = [];\n  anyChanges = false;\n  thumbnail = '';\n  toUpdateObjectRemote: any[] = [];\n  lastMouseData: any;\n  _numOfColumns = 0;\n  hoveringControl = '';\n\n  findById(\n    id: string\n  ): FabricObject<\n    Partial<FabricObjectProps>,\n    SerializedObjectProps,\n    ObjectEvents\n  > | null {\n    return (\n      this.getObjects().find((widget: FabricObject) => widget.id === id) || null\n    );\n  }\n\n  getAbsoluteCoords(object: any): { left: number; top: number } {\n    return {\n      left: object.left + this._offset.left,\n      top: object.top + this._offset.top,\n    };\n  }\n\n  clearData(): void {\n    while (this._objects.length > 0) {\n      this.remove(this._objects.pop()!);\n    }\n  }\n\n  translateWidget(language: string): Promise<void> {\n    throw new Error('Method not implemented.');\n  }\n  zoomToViewAllObjects(): number {\n    return 1;\n  }\n}\n"],"names":["BaseConfiguration","constructor","_defineProperty","this","window","devicePixelRatio","config","super","configure","arguments","length","undefined","Object","assign","addFonts","paths","fontPaths","_objectSpread","removeFonts","forEach","fontFamily","clearFonts","restoreDefaults","keys","defaults","reduce","acc","key","log","severity","_len","optionalParams","Array","_key","console","FabricError","Error","message","options","concat","SignalAbortedError","context","GLProbe","WebGLProbe","testPrecision","gl","precision","fragmentSource","fragmentShader","createShader","FRAGMENT_SHADER","shaderSource","compileShader","getShaderParameter","COMPILE_STATUS","queryWebGL","canvas","getContext","maxTextureSize","getParameter","MAX_TEXTURE_SIZE","GLPrecision","find","getExtension","loseContext","isSupported","textureSize","copyPasteData","env","setEnv","value","getEnv","document","isTouchSupported","navigator","maxTouchPoints","dispose","getFabricDocument","getFabricWindow","getDevicePixelRatio","_config$devicePixelRa","Math","max","cache","getFontCache","_ref","fontStyle","fontWeight","toLowerCase","charWidthsCache","fontCache","cacheKey","clearFontCache","limitDimsByArea","ar","perfLimitSizeTotal","roughWidth","sqrt","floor","VERSION","noop","halfPI","PI","twoMathPi","PiBy180","iMatrix","freeze","DEFAULT_SVG_FONT_SIZE","kRect","CENTER","LEFT","TOP","BOTTOM","RIGHT","NONE","reNewline","JSON","SVG","classRegistry","Map","getClass","classType","get","setClass","classConstructor","set","type","getSVGClass","SVGTagName","setSVGClass","runningAnimations","remove","index","indexOf","splice","cancelAll","animations","animation","abort","cancelByCanvas","filter","_animation$target","target","cancelByTarget","Observable","on","arg0","handler","__eventListeners","entries","eventName","off","push","once","disposers","_ref2","d","disposer","args","call","_removeEventListener","eventListener","_ref3","fire","_this$__eventListener","listenersForEvent","i","getRandomInt","min","random","ifNaN","valueIfNaN","isNaN","removeFromArray","array","idx","cos","angle","abs","sin","angleSlice","sign","Point","y","x","add","that","addEquals","scalarAdd","scalar","scalarAddEquals","subtract","subtractEquals","scalarSubtract","scalarSubtractEquals","multiply","scalarMultiply","scalarMultiplyEquals","divide","scalarDivide","scalarDivideEquals","eq","lt","lte","gt","gte","lerp","t","distanceFrom","dx","dy","midPointFrom","toString","setXY","setX","setY","setFromPoint","swap","clone","rotate","radians","origin","ZERO","sinus","cosinus","p","transform","ignoreOffset","isCollection","fabricObject","isArray","_objects","createCollectionMixin","Base","Collection","_onObjectAdded","object","_onObjectRemoved","_onStackOrderChanged","objects","size","insertAt","_len2","_key2","removed","_len3","_key3","forEachObject","callback","getObjects","_len4","types","_key4","o","isType","item","isEmpty","contains","deep","includes","some","obj","complexity","memo","current","sendObjectToBack","unshift","bringObjectToFront","sendObjectBackwards","intersecting","newIdx","findNewLowerIndex","bringObjectForward","findNewUpperIndex","moveObjectTo","isOverlapping","collectObjects","left","top","width","height","includeIntersecting","tl","br","selectable","visible","intersectsWithRect","isContainedWithinRect","containsPoint","CommonMethods","_setOptions","prop","_setObject","_set","toggle","property","requestAnimFrame","requestAnimationFrame","cancelAnimFrame","handle","cancelAnimationFrame","id","uid","createCanvasElement","element","createElement","createImage","toDataURL","canvasEl","format","quality","degreesToRadians","degrees","radiansToDegrees","isIdentityMatrix","mat","every","transformPoint","invertTransform","a","r","multiplyTransformMatrices","b","is2x2","multiplyTransformMatrixArray","matrices","reduceRight","product","curr","calcPlaneRotation","atan2","qrDecompose","denom","pow","scaleX","scaleY","skewX","skewY","translateX","translateY","createTranslateMatrix","createRotateMatrix","angleRadiant","cosValue","sinValue","createScaleMatrix","angleToSkew","tan","createSkewXMatrix","skewValue","createSkewYMatrix","calcDimensionsMatrix","flipX","flipY","matrix","composeMatrix","scaleMatrix","loadImage","url","signal","crossOrigin","Promise","resolve","reject","aborted","img","err","src","addEventListener","done","onload","onerror","removeEventListener","enlivenObjects","reviver","instances","all","map","fromObject","then","fabricInstance","catch","error","instance","finally","enlivenObjectEnlivables","serializedObject","promises","values","enlived","source","pattern","pick","pickBy","predicate","ColorNameMap","aliceblue","antiquewhite","aqua","aquamarine","azure","beige","bisque","black","blanchedalmond","blue","blueviolet","brown","burlywood","cadetblue","chartreuse","chocolate","coral","cornflowerblue","cornsilk","crimson","cyan","darkblue","darkcyan","darkgoldenrod","darkgray","darkgrey","darkgreen","darkkhaki","darkmagenta","darkolivegreen","darkorange","darkorchid","darkred","darksalmon","darkseagreen","darkslateblue","darkslategray","darkslategrey","darkturquoise","darkviolet","deeppink","deepskyblue","dimgray","dimgrey","dodgerblue","firebrick","floralwhite","forestgreen","fuchsia","gainsboro","ghostwhite","gold","goldenrod","gray","grey","green","greenyellow","honeydew","hotpink","indianred","indigo","ivory","khaki","lavender","lavenderblush","lawngreen","lemonchiffon","lightblue","lightcoral","lightcyan","lightgoldenrodyellow","lightgray","lightgrey","lightgreen","lightpink","lightsalmon","lightseagreen","lightskyblue","lightslategray","lightslategrey","lightsteelblue","lightyellow","lime","limegreen","linen","magenta","maroon","mediumaquamarine","mediumblue","mediumorchid","mediumpurple","mediumseagreen","mediumslateblue","mediumspringgreen","mediumturquoise","mediumvioletred","midnightblue","mintcream","mistyrose","moccasin","navajowhite","navy","oldlace","olive","olivedrab","orange","orangered","orchid","palegoldenrod","palegreen","paleturquoise","palevioletred","papayawhip","peachpuff","peru","pink","plum","powderblue","purple","rebeccapurple","red","rosybrown","royalblue","saddlebrown","salmon","sandybrown","seagreen","seashell","sienna","silver","skyblue","slateblue","slategray","slategrey","snow","springgreen","steelblue","teal","thistle","tomato","turquoise","violet","wheat","white","whitesmoke","yellow","yellowgreen","hue2rgb","q","rgb2Hsl","g","maxValue","minValue","h","s","l","round","fromAlphaToFloat","parseFloat","endsWith","hexify","toUpperCase","padStart","greyAverage","avg","Color","color","setSource","_source","_tryParsingColor","sourceFromHex","sourceFromRgb","sourceFromHsl","getSource","toRgb","toRgba","join","toHsl","toHsla","toHex","toHexa","slice","getAlpha","setAlpha","alpha","toGrayscale","toBlackWhite","threshold","average","bOrW","overlayWith","otherColor","otherSource","R","G","B","fromRgb","fromRgba","match","parsedValue","fromHsl","fromHsla","fromHex","expandedValue","split","hex","hexCouple","parseInt","toFixed","number","fractionDigits","Number","parseUnit","fontSize","unit","exec","dpi","DPI","parsePreserveAspectRatioAttribute","attribute","firstPart","secondPart","trim","alignX","alignY","align","meetOrSlice","matrixToSVG","NUM_FRACTION_DIGITS","colorPropToSVG","colorValue","opacityValue","inlineStyle","toLive","opacity","isFiller","filler","isSerializableFiller","toObject","isPattern","offsetX","isTextObject","_renderText","isActiveSelection","getScrollLeftTop","doc","getDocumentFromElement","docElement","documentElement","body","scrollLeft","scrollTop","parentNode","host","nodeType","style","position","el","ownerDocument","getWindowFromElement","_el$ownerDocument","defaultView","setStyle","styles","elementStyle","cssText","setProperty","setCanvasDimensions","ctx","retinaScaling","setAttribute","scale","setCSSDimensions","makeElementUnselectable","onselectstart","userSelect","StaticCanvasDOMManager","createLowerCanvas","lower","getElementById","hasAttribute","_originalCanvasStyle","classList","cleanupDOM","removeAttribute","setDimensions","calcOffset","_getWindowFromElement","box","offset","offsetAttributes","borderLeftWidth","borderTopWidth","paddingLeft","paddingTop","elemStyle","getComputedStyle","attr","docElem","getBoundingClientRect","scrollLeftTop","clientLeft","clientTop","getElementOffset","staticCanvasDefaults","backgroundVpt","backgroundColor","overlayVpt","overlayColor","includeDefaultValues","svgViewportTransformation","renderOnAddRemove","skipOffscreen","enableRetinaScaling","imageSmoothingEnabled","controlsAboveOverlay","allowTouchScrolling","viewportTransform","StaticCanvas","lowerCanvasEl","_this$elements$lower","elements","contextContainer","_this$elements$lower2","getDefaults","ownDefaults","initElements","_setDimensionsImpl","calcViewportBoundaries","requestRenderAll","setCoords","getRetinaScaling","_offset","getWidth","getHeight","setWidth","setHeight","dimensions","cssOnly","backstoreOnly","hasLostContext","getZoom","setViewportTransform","vpt","backgroundObject","backgroundImage","overlayObject","overlayImage","len","group","zoomToPoint","point","before","newPoint","after","setZoom","absolutePan","relativePan","getElement","clearContext","clearRect","clear","renderAll","cancelRequestedRender","destroyed","renderCanvas","renderAndReset","nextRenderHandle","disposed","iVpt","vptCoords","tr","bl","drawControls","v","path","clipPath","patternQuality","_renderBackground","save","_renderObjects","restore","shouldCache","_transformDone","renderCache","forClipping","drawClipPathOnCanvas","_renderOverlay","__cleanupTask","globalCompositeOperation","zoomX","zoomY","drawImage","_cacheCanvas","cacheTranslationX","cacheTranslationY","render","_renderBackgroundOrOverlay","fill","needsVpt","isAFiller","beginPath","moveTo","lineTo","closePath","fillStyle","offsetY","m","gradientTransform","patternTransform","getCenter","getCenterPoint","centerObjectH","_centerObject","centerObjectV","centerObject","viewportCenterObject","getVpCenter","viewportCenterObjectH","viewportCenterObjectV","center","toDatalessJSON","propertiesToInclude","toDatalessObject","_toObjectMethod","toJSON","methodName","clipPathData","excludeFromExport","_toObject","version","__serializeBgOverlay","originalValue","data","bgImage","bgColor","background","overlay","toSVG","markup","_setSVGPreamble","_setSVGHeader","clipPathId","_setSVGBgOverlayColor","_setSVGBgOverlayImage","_setSVGObjects","suppressPreamble","encoding","optViewBox","viewBox","createSVGFontFacesMarkup","createSVGRefElementsMarkup","createSVGClipPathMarkup","toClipPathSVG","shouldTransform","additionalTransform","fontList","styleRow","fontListMarkup","_setSVGObject","bgOrOverlay","repeat","finalWidth","finalHeight","loadFromJSON","json","serialized","parse","enlivedMap","properties","cloneWithoutData","multiplier","finalMultiplier","toCanvasElement","scaledWidth","scaledHeight","zoom","originalWidth","originalHeight","newZoom","vp","newVp","originalRetina","objectsToRender","task","destroy","kill","touchEvents","getPointer","event","scroll","_evt","touchProp","changedTouches","getTouchInfo","clientX","clientY","isTouchEvent","pointerType","stopEvent","e","preventDefault","stopPropagation","makeBoundingBoxFromPoints","points","addTransformToObject","applyTransformToObject","calcOwnMatrix","_qrDecompose","otherOptions","_objectWithoutProperties","_excluded","setPositionByOrigin","resetObjectTransform","saveObjectTransform","sizeAfterTransform","dimX","dimY","bbox","calcPlaneChangeMatrix","from","sendPointToPlane","to","sendVectorToPlane","sendObjectToPlane","fireEvent","_target$canvas","originOffset","bottom","right","resolveOrigin","originValue","NOT_ALLOWED_CURSOR","isTransformCentered","originX","originY","invertOrigin","isLocked","lockingKey","commonEventInfo","eventData","pointer","findCornerQuadrant","control","cornerAngle","getTotalAngle","getLocalPoint","corner","controls","padding","localPoint","getRelativeCenterPoint","translateToGivenOrigin","normalizePoint","dragHandler","newLeft","newTop","moveX","moveY","FabricObjectSVGExportMixin","getSvgStyles","skipShadow","fillRule","strokeWidth","strokeDashArray","strokeDashOffset","strokeLineCap","strokeLineJoin","strokeMiterLimit","visibility","getSvgFilter","stroke","shadow","getSvgCommons","getSvgTransform","full","calcTransformMatrix","svgTransform","_toSVG","_createBaseSVGMarkup","_createBaseClipPathSVGMarkup","objectMarkup","commonPieces","noStyle","withShadow","styleInfo","shadowInfo","vectorEffect","strokeUniform","absoluteClipPath","absolutePositioned","clipPathMarkup","addPaintOrder","paintFirst","normalize","c","asin","elastic","defaultEasing","easeOutBounce","easeInBounce","easeInCirc","easeInCubic","easeInElastic","normA","normS","normP","easeInExpo","easeInOutBounce","easeInOutCirc","easeInOutCubic","easeInOutElastic","normC","easeInOutExpo","easeInOutQuad","easeInOutQuart","easeInOutQuint","easeInOutSine","easeInQuad","easeInQuart","easeInQuint","easeInSine","easeOutCirc","easeOutCubic","easeOutElastic","easeOutExpo","easeOutQuad","easeOutQuart","easeOutQuint","easeOutSine","defaultAbort","AnimationBase","startValue","byValue","duration","delay","easing","onStart","onChange","onComplete","tick","bind","_onStart","_onChange","_onComplete","_abort","endValue","calculate","state","_state","isDone","start","firstTick","timestamp","startTime","Date","register","setTimeout","durationMs","boundDurationMs","durationProgress","valueProgress","unregister","ValueAnimation","timeElapsed","ArrayAnimation","capValue","defaultColorEasing","wrapColorCallback","rgba","ColorAnimation","startColor","endColor","animate","isArrayAnimation","animateColor","unitVectorX","zero","rotateVector","vector","createVector","magnitude","calcAngleBetweenVectors","crossProduct","dotProduct","calcVectorRotation","getUnitVector","getOrthonormalVector","counterClockwise","isBetweenVectors","AxB","AxT","BxT","Intersection","status","append","isPointContained","T","A","infinite","AB","isPointInPolygon","other","hits","inter","intersectSegmentSegment","intersectLineLine","a1","a2","b1","b2","aInfinite","bInfinite","a2xa1x","a2ya1y","b2xb1x","b2yb1y","a1xb1x","a1yb1y","uaT","ubT","uB","ua","ub","segmentsCoincide","intersectSegmentLine","s1","s2","l1","l2","intersectLinePolygon","result","intersectSegmentPolygon","intersectPolygonPolygon","points1","points2","coincidences","intersectPolygonRectangle","r1","r2","topRight","bottomLeft","ObjectOrigin","_getTransformedDimensions","dimOptions","preScalingStrokeValue","postScalingStrokeValue","finalDimensions","fromOriginX","fromOriginY","toOriginX","toOriginY","dim","translateToCenterPoint","translateToOriginPoint","relCenter","getPointByOrigin","pos","_getLeftTopCoords","ObjectGeometry","getX","getXY","getY","getRelativeX","setRelativeX","getRelativeY","setRelativeY","relativePosition","getRelativeXY","setRelativeXY","isStrokeAccountedForInDimensions","getCoords","aCoords","calcACoords","coords","intersectsWithObject","intersection","isContainedWithinObject","getBoundingRect","isOnScreen","isPartiallyOnScreen","getScaledWidth","getScaledHeight","scaleToWidth","boundingRectFactor","scaleToHeight","getCanvasRetinaScaling","_this$canvas","getViewportTransform","_this$canvas2","rotateMatrix","tMatrix","finalMatrix","w","transformMatrixKey","skipGroup","prefix","matrixCache","ownMatrixCache","_getNonTransformedDimensions","_calculateCurrentDimensions","StackedObject","isDescendantOf","parent","getAncestors","strict","ancestors","_parent$parent","findCommonAncestors","fork","otherFork","common","otherAncestors","ancestor","j","hasCommonAncestors","commonAncestors","isInFrontOf","ancestorData","firstCommonAncestor","headOfFork","pop","headOfOtherFork","thisIndex","otherIndex","AnimatableObject","animatable","_animate","propIsColor","colorProperties","animationOptions","getSvgRegex","arr","RegExp","reNum","String","raw","_templateObject","_taggedTemplateLiteral","svgNS","_templateObject2","reFontDeclaration","attributesMap","cx","cy","display","fSize","cPath","svgValidTagNamesRegEx","svgViewBoxElementsRegEx","svgValidParentsRegEx","reViewBoxAttrValue","shadowOffsetRegex","Shadow","parseShadow","shadowStr","regex","blur","replace","fBoxX","fBoxY","affectStroke","nonScaling","cloneDeep","stringify","stateProperties","cacheProperties","fabricObjectDefaultValues","minScaleLimit","objectCaching","inverted","centeredRotation","centeredScaling","dirty","FabricObject","name","setOptions","_createCacheCanvas","_cacheContext","_updateCacheCanvas","_limitCacheSize","dims","maxCacheSideLimit","minCacheSideLimit","limX","limY","capped","_getCacheCanvasDimensions","objectScale","getTotalObjectScaling","neededX","neededY","minCacheSize","dimensionsChanged","cacheWidth","cacheHeight","zoomChanged","drawingWidth","drawingHeight","shouldRedraw","additionalWidth","additionalHeight","shouldResizeCanvas","canvasWidth","canvasHeight","sizeGrowing","getHeightOfLine","ceil","setTransform","translate","needFullTransform","contextTop","_removeDefaultValues","baseValues","getPrototypeOf","baseValue","getObjectScaling","retina","getObjectOpacity","_constrainScale","isChanged","isNotVisible","_setupCompositeOperation","drawSelectionBackground","_setOpacity","_setShadow","drawCacheOnCanvas","_removeCacheCanvas","drawObject","isCacheDirty","hasStroke","hasFill","needsItsOwnCache","ownCaching","isOnACache","willDrawShadow","drawClipPathOnCache","originalFill","originalStroke","_setClippingProperties","_render","_drawClipPath","skipCanvas","fillRect","_removeShadow","globalAlpha","_setStrokeStyles","decl","lineWidth","lineCap","lineDashOffset","lineJoin","miterLimit","gradientUnits","_applyPatternForTransformedGradient","strokeStyle","_applyPatternGradientTransform","_setFillStyles","_setLineDash","dashArray","setLineDash","sx","sy","multX","multY","scaling","shadowColor","shadowBlur","browserShadowBlurConstant","shadowOffsetX","shadowOffsetY","_renderPaintInOrder","_renderStroke","_renderFill","_pCtx$createPattern","pCanvas","pCtx","createPattern","_findCenterFromElement","objectForm","cloneAsImage","origParams","originalGroup","originalShadow","canvasProvider","withoutTransform","withoutShadow","boundingRect","shadowOffset","originalCanvas","setOnGroup","_fromObject","extraParam","_excluded2","allOptions","_toPropertyKey","getText","_this$text","text","saveData","fields","transformPointToCanvas","transformPointFromCanvas","transformPointToViewport","_self$canvas","toTransformPoint","mCanvas","mObject","wrapWithFireEvent","actionHandler","extraEventInfo","actionPerformed","wrapWithFixedAnchor","centerPoint","constraint","changeWidth","changeObjectWidth","strokePadding","oldWidth","newWidth","renderCircleControl","styleOverride","xSize","sizeX","cornerSize","ySize","sizeY","transparentCorners","cornerStrokeColor","myLeft","myTop","cornerColor","arc","renderSquareControl","xSizeBy2","ySizeBy2","strokeRect","Control","shouldActivate","controlKey","_fabricObject$canvas","getActiveObject","isControlVisible","getActionHandler","getMouseDownHandler","mouseDownHandler","getMouseUpHandler","mouseUpHandler","cursorStyleHandler","cursorStyle","getActionName","actionName","getVisibility","_fabricObject$_contro","_fabricObject$_contro2","_controlsVisibility","setVisibility","positionHandler","currentControl","calcCornerCoords","objectCornerSize","centerX","centerY","isTouch","touchSizeX","touchSizeY","cornerStyle","scaleIsProportional","uniformIsToggled","uniScaleKey","uniformScaling","scalingIsForbidden","by","scaleProportionally","lockX","lockY","scaleMap","scaleCursorStyleHandler","n","scaleObject","signX","signY","gestureScale","distance","original","oldScaleX","oldScaleY","scalingEqually","scaleObjectFromCorner","scalingX","scaleObjectX","scalingY","scaleObjectY","AXIS_KEYS","counterAxis","skew","lockSkewing","flip","skewMap","skewCursorStyleHandler","skewHandler","axis","originKey","lockSkewingKey","skewKey","flipKey","counterOriginKey","counterFlipKey","counterOriginFactor","skewingSide","finalHandler","ex","ey","skewingBefore","skewingStart","shearingStart","shearing","skewing","atan","changed","dimBefore","dimAfter","compensationFactor","skewObject","skewHandlerX","skewHandlerY","isAltAction","altActionKey","scaleOrSkewActionName","isAlternative","scaleSkewCursorStyleHandler","scalingYOrSkewingX","createObjectDefaultControls","createObjectDefaultNoRotateControls","createObjectImageControls","createObjectFileControls","createObjectConnectorControls","renderCustomControl","mtaStart","transformData","mbaStart","mlaStart","mraStart","createResizeControls","mr","ml","mb","mt","createResizeControlsForText","createTextboxDefaultControls","createPathDefaultControls","InteractiveFabricObject","targetCanvas","noScaleCache","_currentTransform","action","startsWith","getActiveControl","__corner","coord","oCoords","findControl","forTouch","hasControls","cornerEntries","touchCorner","calcOCoords","rMatrix","positionMatrix","startMatrix","transformOptions","forEachControl","_calcCornerCoords","touchCornerSize","fn","selectionBackgroundColor","_activeObject","wh","strokeBorders","_drawBorders","borderColor","borderDashArray","drawControlsConnectingLines","_renderControls","hasBorders","styleOptions","shouldDrawBorders","shouldDrawControls","borderScaleFactor","isMoving","borderOpacityWhenMoving","drawBorders","forActiveSelection","shouldStroke","withConnection","cornerDashArray","setControlVisible","setControlsVisibility","clearContextTop","restoreManually","onDeselect","onSelect","shouldStartDragging","onDragStart","canDrop","renderDragSourceEffect","renderDropTargetEffect","applyMixins","derivedCtor","constructors","baseCtor","getOwnPropertyNames","prototype","defineProperty","getOwnPropertyDescriptor","create","lockMovementX","lockMovementY","lockRotation","lockScalingX","lockScalingY","lockSkewingX","lockSkewingY","lockScalingFlip","evented","perPixelTargetFind","activeOn","hoverCursor","moveCursor","isTransparent","tolerance","getImageData","StrokeProjectionsBase","strokeProjectionMagnitude","strokeUniformScalar","createSideVector","projectOrthogonally","applySkew","calcOrthogonalProjection","isSkewed","scaleUnitVector","unitVector","zeroVector","StrokeLineJoinProjections","getOrthogonalRotationFactor","vector1","vector2","C","AC","bisector","orthogonalProjection","correctSide","projectBevel","projections","projectMiter","hypotUnitScalar","miterVector","projectRoundNoSkew","startCircle","endCircle","projectRoundWithSkew","circleRadius","newY","furthestY","newX","furthestX","projectRound","isStraightLine","newOrigin","proj0","proj1","comparisonVector","isProj0Start","projectPoints","project","originPoint","projectedPoint","StrokeLineCapProjections","projectButt","projection","projectSquare","strokePointingOut","projectedA","projectStrokeOnPoints","openPath","reduced","findIndexRight","escapeXml","string","graphemeSplit","textstring","graphemes","chr","getWholeChar","str","code","charCodeAt","charAt","next","prev","firstLetterOnly","hasStyleChanged","prevStyle","thisStyle","forTextSpans","textBackgroundColor","deltaY","overline","underline","linethrough","stylesToArray","textLines","stylesArray","charIndex","chars","end","stylesFromArray","stylesObject","styleIndex","SHARED_ATTRIBUTES","selectorMatches","selector","nodeName","classNames","getAttribute","azAz","matcher","splitClassNames","elementMatchesRule","selectors","parentMatching","firstMatching","parentElement","doesSomeParentMatch","normalizeAttr","_attributesMap","cleanupSvgAttribute","attributeValue","_templateObject3","_templateObject4","_templateObject5","_templateObject6","transforms","transformList","_templateObject7","reTransformList","reTransform","parseTransformAttribute","test","matchAll","transformMatch","matchedParams","operation","rawArgs","arg1","arg2","arg3","arg4","arg5","arg","normalizeValue","parentAttributes","parsed","ouputValue","transformMatrix","fillIndex","strokeIndex","parseFontDeclaration","oStyle","lineHeight","parseStyleAttribute","chunk","parseStyleString","parseStyleObject","colorAttributesMap","parseAttributes","attributes","cssRules","parentFontSize","ownAttributes","rule","getGlobalStylesForElement","normalizedStyle","normalizedAttr","normalizedValue","font","mergedAttrs","colorAttr","setStrokeFillOpacity","RECT_PROPS","Rect","_initRxRy","rx","ry","isRounded","bezierCurveTo","fromElement","_parseAttributes","ATTRIBUTE_NAMES","restOfparsedAttributes","Boolean","LAYOUT_TYPE_INITIALIZATION","LAYOUT_TYPE_ADDED","LAYOUT_TYPE_REMOVED","LAYOUT_TYPE_IMPERATIVE","getObjectBounds","destinationGroup","currentGroup","objectCenter","accountForStroke","strokeUniformVector","scalingStrokeWidth","sizeVector","LayoutStrategy","calcLayoutResult","shouldPerformLayout","calcBoundingBox","prevStrategy","strategy","shouldLayoutClipPath","getInitialSize","overrides","bboxSize","bboxCenter","actualSize","relativeCorrection","FitContentLayout","LAYOUT_MANAGER","LayoutManager","_subscriptions","performLayout","strictContext","bubbles","_prevLayoutStrategy","onBeforeLayout","layoutResult","getLayoutResult","commitLayout","onAfterLayout","attachHandlers","trigger","subscribe","unsubscribe","delete","unsubscribeTargets","targets","subscribeTargets","tricklingContext","layoutManager","prevCenter","nextCenter","correction","_context$x","_context$y","layoutObjects","layoutObject","_","bubblingContext","NoopLayoutManager","Group","__objectSelectionTracker","__objectSelectionMonitor","__objectSelectionDisposer","enterGroup","canEnterGroup","_filterObjectsBeforeEnteringGroup","allowedObjects","_onAfterObjectsChange","removeParentTransform","exitGroup","_shouldSetNestedCoords","subTargetCheck","removeAll","_activeObjects","selected","activeObjects","_watchObject","watch","_enterGroup","activeObject","_exitGroup","ownCache","preserveObjectStacking","triggerLayout","__serializeObjects","method","_includeDefaultValues","originalDefaults","_createSVGBgRect","fillStroke","commons","svgString","bg","abortable","hydratedOptions","layoutClass","strategyClass","interactive","findScaleToFit","destination","findScaleToCover","reMoveToCommand","reLineCommand","reHorizontalLineCommand","reVerticalLineCommand","reClosePathCommand","reCubicCurveCommand","reCubicCurveShortcutCommand","reQuadraticCurveCommand","reQuadraticCurveShortcutCommand","reArcCommand","rePathCommand","repeatedCommands","M","segmentToBezier","theta1","theta2","cosTh","sinTh","cx1","cy1","mT","fromX","fromY","costh1","sinth1","costh2","sinth2","toX","toY","calcVectorAngle","ux","uy","vx","vy","ta","tb","getBoundsOfCurve","begx","begy","cp1x","cp1y","cp2x","cp2y","endx","endy","argsString","cachesBoundsOfCurve","boundsOfCurveCache","tvalues","bounds","b2ac","sqrtb2ac","t1","t2","jlen","iterator","getPointOnCubicBezierIterator","fromArcToBeziers","fx","fy","rot","large","sweep","tx","ty","segsNorm","arcToSegments","rotateX","root","theta","sinTheta","px","py","rx2","ry2","py2","px2","pl","_rx","_ry","mTheta","dtheta","segments","mDelta","th3","makePathSimpler","x1","y1","destinationPath","previous","controlX","controlY","parsedCommand","converted","calcLineLength","x2","y2","pct","c1","c2","CB2","c3","CB3","c4","CB4","QB1","QB2","QB3","getTangentCubicIterator","p1x","p1y","p2x","p2y","p3x","p3y","p4x","p4y","qb1","qb2","qb3","tangentX","tangentY","getPointOnQuadraticBezierIterator","getTangentQuadraticIterator","invT","pathIterator","tempP","tmpLen","perc","findPercentageForDistance","segInfo","nextLen","nextStep","lastPerc","angleFinder","getPathSegmentsInfo","tempInfo","totalLength","info","basicInfo","command","destX","destY","getPointOnPath","infos","segPercent","segment","parsePath","pathString","res","matchStr","chain","paramArr","filteredGroups","shift","numParse","reverse","transformed","getSmoothPathFromPoints","p1","p2","multSignX","multSignY","manyPoints","midPoint","joinPath","pathData","request","xhr","removeListener","ontimeout","onreadystatechange","readyState","open","send","removeTransformMatrixForSvgParsing","preserveAspectRatioOptions","_assignTransformMatrixProps","cropX","cropY","offsetLeft","offsetTop","_newCanvas$getContext","newCanvas","getRegularPolygonPath","numVertexes","radius","interiorAngle","rotationAdjustment","rad","commonAttributes","groupSVGElements","mergeClipPaths","_b$group","removeTransformFromObject","finalTransform","rotatePoint","transformPath","pathOffset","pathSegment","newSegment","CanvasDOMManager","containerClass","upperCanvasEl","createUpperCanvas","upper","applyCanvasStyle","container","createContainerElement","replaceChild","className","allow","touchAction","removeChild","SelectableCanvas","_this$elements$upper","_this$elements$upper2","wrapperEl","_objectsToRender","deselected","_discardActiveObject","_hoveredTarget","_hoveredTargets","_chooseObjectsToRender","contextTopDirty","_groupSelector","isDrawingMode","renderTopLayer","_isCurrentlyDrawing","freeDrawingBrush","selection","_drawSelection","renderTop","setTargetFindTolerance","targetFindTolerance","pixelFindCanvasEl","pixelFindContext","isTargetTransparent","selectionBgc","enhancedTolerance","_isSelectionKeyPressed","sKey","selectionKey","_shouldClearSelection","getActiveObjects","_shouldCenterTransform","modifierKeyPressed","centerTransform","_getOriginFromCorner","controlName","_setupCurrentTransform","alreadySelected","_control$getActionHan","getScenePoint","getActionFromCorner","altKey","centeredKey","lastX","lastY","shiftKey","setCursor","cursor","deltaX","extent","strokeOffset","selectionLineWidth","minX","minY","maxX","maxY","selectionColor","selectionBorderColor","selectionDashArray","findTarget","skipTargetFind","getViewportPoint","aObjects","searchPossibleTargets","subTargets","altSelectionKey","_pointIsInObjectSelectionArea","viewportZoom","angleRadians","cosP","sinP","cosPSinP","cosPMinusSinP","_checkTarget","isEditing","_searchPossibleTargets","subTarget","_pointer","_absolutePointer","fromViewport","boundsWidth","boundsHeight","cssScale","_resetTransformEventData","_setBrushStyles","willReadFrequently","getTopContext","getSelectionContext","getSelectionElement","active","_fireSelectionEvents","oldObjects","somethingChanged","invalidate","added","setActiveObject","currentActives","_setActiveObject","prevActiveObject","endCurrentTransform","discardActiveObject","discarded","_finalizeCurrentTransform","_scaling","originalProperties","_realizeGroupTransformOnObject","originalValues","selectionFullyContained","defaultCursor","freeDrawingCursor","notAllowedCursor","stopContextMenu","fireRightClick","fireMiddleClick","enablePointerEvents","TextEditingManager","cb","hiddenTextarea","focus","__disposer","exitTextEditing","exitEditing","onMouseMove","_this$target","updateSelectionOnMouseMove","addEventOptions","passive","getEventPoints","viewportPoint","scenePoint","absolutePointer","addListener","syntheticEventConfig","mouse","in","out","targetIn","targetOut","canvasIn","canvasOut","drag","Canvas","eventHandler","addOrRemove","_getEventPrefix","functor","eventjsFunctor","canvasElement","eventTypePrefix","_onResize","_onMouseDown","_onMouseMove","_onMouseOut","_onMouseEnter","_onMouseWheel","_onContextMenu","_onDoubleClick","_onDragStart","_onDragEnd","_onDragOver","_onDragEnter","_onDragLeave","_onDrop","_onTouchStart","removeListeners","_onMouseUp","_onTouchEnd","__onMouseWheel","shared","nestedTarget","_isClick","_dragSource","_onDragProgress","_renderDragEffects","dropTarget","_dropTarget","didDrop","dataTransfer","dropEffect","dragSource","_draggedoverTarget","findDragTargets","eventType","_fireEnterLeaveEvents","_basicEventHandler","_cacheTransformEventData","_handleEvent","getPointerId","evt","identifier","pointerId","_isMainEvent","isPrimary","touches","mainTouchId","__onMouseDown","__onMouseUp","_willAddMouseDown","clearTimeout","__onMouseMove","_shouldRender","_this$_activeObject","isClick","_target","button","_onMouseUpInDrawingMode","shouldRender","targetWasActive","handleSelection","found","originalControl","originalMouseUpHandler","_setCursorFromEvent","currentTarget","currentSubTargets","_onMouseDownInDrawingMode","onMouseDown","_onMouseMoveInDrawingMode","onMouseUp","grouped","handleMultiSelection","groupSelector","_transformObject","_fireOverOutEvents","textEditingManager","fireSyntheticInOutEvents","oldTarget","fireCanvas","draggedoverTarget","targetChanged","outOpt","nextTarget","inOpt","previousTarget","localPointer","_performTransformAction","activeSelection","isAS","prevActiveObjects","multiSelectAdd","newActiveSelection","point1","point2","collectedObjects","klass","linearDefaultCoords","radialDefaultCoords","RE_PERCENT","isPercent","parsePercent","NaN","RE_KEY_VALUE_PAIRS","RE_KEY_VALUE","parseColorStop","keyValuePairs","parseColorStops","opacityAttr","colorStops","colorStopEls","getElementsByTagName","parseType","parseGradientUnits","getValue","parseCoords","valuesToConvert","finalValue","propValue","convertPercentUnitsToValues","parseLinearCoords","parseRadialCoords","Gradient","addColorStop","preTransform","colorStop","sort","_renderPathCommands","needsSwap","minRadius","percentageShift","gradient","createLinearGradient","createRadialGradient","svgOptions","viewBoxWidth","viewBoxHeight","Pattern","isImageSource","isCanvasSource","sourceToString","complete","naturalWidth","naturalHeight","patternSource","patternOffsetX","patternOffsetY","patternWidth","patternHeight","BaseBrush","_saveAndTransform","needsFullRender","_resetShadow","_isOutSideCanvas","Path","_setPath","adjustPosition","setBoundingBox","_calcBoundsFromPath","quadraticCurveTo","pathCmd","sourcePath","_getOffsetTransform","digits","_calcDimensions","subpathStartX","subpathStartY","parsedAttributes","PencilBrush","_points","_hasStraightLine","drawSegment","drawStraightLine","straightLineKey","_prepareForDrawing","_addPoint","limitedToCanvasSize","oldEnd","_finalizeAndAddPath","_reset","convertPointsToSVGPath","createPath","decimatePoints","cDistance","lastPoint","adjustedDistance","newPoints","decimate","isEmptySVGPath","CIRCLE_PROPS","Circle","setRadius","startAngle","endAngle","getRadiusX","getRadiusY","startX","startY","endX","endY","largeFlag","sweepFlag","CircleBrush","drawDot","addPoint","dot","originalRenderOnAddRemove","circles","circle","pointerPoint","SprayBrush","sprayChunks","sprayChunk","addSprayChunk","renderChunck","rects","chunck","rect","optimizeOverlapping","uniqueRects","uniqueRectsArray","getUniqueRects","sprayChunck","density","dotWidthVariance","dotWidth","randomOpacity","PatternBrush","getPatternSrc","patternCanvas","patternCtx","getPattern","topLeft","coordProps","Line","_setWidthHeight","calcLinePoints","origStrokeStyle","_this$stroke","_x1","_x2","_y1","_y2","xMult","yMult","Triangle","widthBy2","heightBy2","ELLIPSE_PROPS","Ellipse","getRx","getRy","parsePointsAttribute","pointsSplit","parsedPoints","polylineDefaultValues","exactBoundingBox","Polyline","initialized","isOpen","_projectStrokeOnPoints","strokeDiff","bboxNoStroke","layoutProperties","_options$width","_options$height","_options$width2","_options$height2","output","diffX","diffY","Polygon","fontProperties","textDecorationProperties","textLayoutProperties","additionalProps","styleProperties","textDefaultValues","_reNewline","_reSpacesAndTabs","_reSpaceAndTab","_reWords","textAlign","superscript","baseline","subscript","pathStartOffset","pathSide","pathAlign","_fontSizeFraction","offsets","_fontSizeMult","charSpacing","direction","CACHE_FONT_SIZE","MIN_TEXT_WIDTH","JUSTIFY","JUSTIFY_LEFT","JUSTIFY_RIGHT","JUSTIFY_CENTER","StyledText","isEmptyStyles","lineIndex","line","p3","styleHas","cleanStyle","letterCount","stylePropertyValue","stylesCount","allStyleObjectPropertiesMatch","graphemeCount","styleObject","_textLines","removeStyle","lineNum","charNum","_extendStyles","get2DCursorLocation","_getLineStyle","_setLineStyle","newStyle","_getStyleDeclaration","_setStyleDeclaration","getSelectionStyles","startIndex","endIndex","getStyleAtPosition","getCompleteStyleDeclaration","setSelectionStyles","_forceClearCache","_lineStyle$charIndex","lineStyle","_styleProperties","_deleteStyleDeclaration","_deleteLineStyle","multipleSpacesRegex","dblQuoteRegex","createSVGInlineRect","svgColor","createSVGRect","measuringContext","FabricText","setPathInfo","initDimensions","segmentsInfo","_splitText","newLines","_splitTextIntoLines","lines","graphemeLines","_unwrappedTextLines","_unwrappedLines","_text","graphemeText","_clearCache","calcTextWidth","cursorWidth","calcTextHeight","enlargeSpaces","diffSpace","currentLineWidth","numberOfSpaces","accumulatedSpace","charBound","spaces","isEndOfWrapping","getLineWidth","__charBounds","kernedWidth","missingNewlineOffset","selectionStart","skipWrapping","_setTextStyles","_renderTextLinesBackground","_renderTextDecoration","_renderTextStroke","_renderTextFill","charStyle","forMeasuring","textBaseline","_getFontDeclaration","maxWidth","_renderTextLine","_renderChars","leftOffset","_getLeftOffset","lineTopOffset","_getTopOffset","heightOfLine","lineLeftOffset","_getLineLeftOffset","drawStart","currentColor","boxWidth","boxStart","lastColor","getValueOfPropertyAt","charBox","renderLeft","_measureChar","_char","previousChar","prevCharStyle","fontDeclaration","couple","stylesAreEqual","fontMultiplier","coupleWidth","previousWidth","getMeasuringContext","measureText","getHeightOfChar","measureLine","lineInfo","_measureLine","_getWidthOfCharSpacing","prevGrapheme","graphemeInfo","llength","lineBounds","grapheme","_getGraphemeBox","positionInPath","totalPathLength","_setGraphemeOnPath","numOfSpaces","centerPosition","skipLeft","previousBox","__lineHeights","maxHeight","_renderTextCommon","lineHeights","isJustify","shortCut","isLtr","currentDirection","actualStyle","nextStyle","timeToRender","drawingLeft","charsToRender","_renderChar","_applyPatternGradientTransformText","handleFiller","fullDecl","shouldFill","fillOffsets","fillText","strokeOffsets","strokeText","setSuperscript","_setScript","setSubscript","schema","loc","lineDiff","__lineWidths","_charStyle$property","topOffset","currentDecoration","currentFill","lastDecoration","lastFill","currentSize","currentDy","parsedFontFamily","genericFonts","newLine","newText","needsDims","isAddingPath","_options$parsedAttrib","textAnchor","textDecoration","restOfOptions","textContent","textHeightScaleFactor","scaledDiff","textHeight","offX","_getSVGLeftTopOffsets","textAndBg","_getSVGTextAndBg","textTop","textLeft","_wrapSVGTextAndBg","lineTop","textBgRects","textSpans","getSvgTextDecoration","textTopOffset","textLeftOffset","lineOffset","_setSVGTextLineBg","_setSVGTextLineText","_createTextCharSpan","char","styleDecl","styleProps","getSvgSpanStyles","fillStyles","dySpan","_getSVGLineTopOffset","lastHeight","useWhiteSpace","decoration","DraggableTextDelegate","dragEnterHandler","dragOverHandler","dragLeaveHandler","dragEndHandler","dropHandler","_dispose","isPointerOverSelection","newSelection","getSelectionStartFromPointer","selectionEnd","__mouseDownInPlace","isActive","__dragStartFired","setCursorByClick","initDelayedCursor","__isDraggingOver","getDragStartSelection","__dragStartSelection","setDragImage","_e$dataTransfer","flipFactor","boundaries","_getCursorBoundaries","diff","bgc","dragImage","border","__dragImageDisposer","appendChild","setData","effectAllowed","abortCursorAnimation","editable","defaultPrevented","dragStartSelection","targetCanDrop","ev","_e$dataTransfer2","insert","getData","trailing","selectionStartOffset","removeChars","trimEnd","insertChars","enterEditing","_updateTextarea","_e$dataTransfer3","reNonWord","ITextBehavior","initBehavior","_tick","_onTickComplete","_animateCursor","toValue","_currentCursorOpacity","renderCursorOrSelection","_currentTickState","cursorDuration","_this$_currentTickCom","_currentTickCompleteState","restart","cursorDelay","shouldClear","cursorAnimation","restartCursorIfNeeded","selectAll","_fireSelectionChanged","getSelectedText","findWordBoundaryLeft","startFrom","_reSpace","findWordBoundaryRight","findLineBoundaryLeft","findLineBoundaryRight","searchWordBoundary","selectWord","newSelectionStart","newSelectionEnd","selectLine","initHiddenTextarea","_saveEditingProps","_setEditingProps","_textBeforeEdit","activeElement","currentStart","currentEnd","__selectionStartOnMouseDown","editingBorderColor","fromStringToGraphemeSelection","smallerTextStart","graphemeStart","smallerTextEnd","fromGraphemeToStringSelection","cursorOffsetCache","inCompositionMode","updateTextareaPosition","updateFromTextArea","textarea","_calcTextareaPosition","desiredPosition","compositionStart","cursorLocation","charHeight","upperCanvas","upperCanvasWidth","upperCanvasHeight","clientWidth","clientHeight","_savedProps","_restoreEditingProps","_exitEditing","isTextChanged","_removeExtraneousStyles","removeStyleFromTo","lineStart","charStart","lineEnd","charEnd","styleObj","shiftLineStyles","numericChar","clonedStyles","numericLine","insertNewlineStyleObject","qty","copiedStyle","newLineStyles","originalLineLength","isEndOfLine","someStyleIsCarryingOver","currentCharStyle","numIndex","styleCarriedOver","insertCharStyleObject","quantity","currentLineStyles","currentLineStylesCloned","numericIndex","insertNewStyleBlock","insertedText","cursorLoc","addedLines","linesLength","setSelectionStartEndWithShift","_selectionDirection","ITextKeyBehavior","autocapitalize","autocorrect","autocomplete","spellcheck","wrap","hiddenTextareaContainer","keydown","keyup","input","copy","cut","paste","compositionstart","compositionupdate","compositionend","onKeyDown","keyMap","keysMapRtl","keysMap","keyCode","ctrlKeysMapDown","ctrlKey","metaKey","stopImmediatePropagation","onKeyUp","_copyDone","ctrlKeysMapUp","onInput","fromPaste","updateAndFire","nextText","charCount","nextCharCount","removedText","removeFrom","removeTo","charDiff","textareaSelection","backDelete","copiedText","disableStyleCopyPaste","copiedTextStyle","onCompositionStart","onCompositionEnd","onCompositionUpdate","compositionEnd","_getWidthBeforeCursor","bound","widthBeforeCursor","getDownCursorOffset","isRight","selectionProp","_getSelectionForOffset","indexOnOtherLine","_getIndexOnLine","getUpCursorOffset","textBeforeCursor","charWidth","foundMatch","widthOfCharsOnLine","indexOnLine","leftEdge","rightEdge","offsetFromLeftEdge","moveCursorDown","_moveCursorUpOrDown","moveCursorUp","moveCursorWithShift","moveCursorWithoutShift","moveCursorLeft","_moveCursorLeftOrRight","_move","newValue","_moveLeft","_moveRight","moveCursorLeftWithoutShift","change","moveCursorLeftWithShift","moveCursorRight","moveCursorRightWithShift","moveCursorRightWithoutShift","notALeftClick","ITextClickBehavior","_mouseDownHandler","_mouseDownHandlerBefore","doubleClickHandler","tripleClickHandler","__lastClickTime","__lastLastClickTime","__lastPointer","draggableTextDelegate","__newClickTime","newPointer","isTripleClick","__lastSelected","didDrag","mouseOffset","charLength","widthAfter","MOVE_CURSOR_UP","MOVE_CURSOR_DOWN","MOVE_CURSOR_LEFT","MOVE_CURSOR_RIGHT","EXIT_EDITING","iTextDefaultValues","cursorColor","caching","IText","setSelectionStart","_updateAndFire","setSelectionEnd","renderCursor","renderSelection","skipCaching","_getCursorBoundariesOffsets","__getCursorBoundariesOffsets","renderCursorAt","_renderCursor","_renderSelection","dragSelection","startLine","endLine","startChar","endChar","realLineHeight","boxEnd","drawHeight","extraTop","drawWidth","compositionColor","getCurrentCharFontSize","cp","_getCurrentCharIndex","getCurrentCharColor","cursorPosition","ClipPathLayout","_target$group","clipPathCenter","FixedLayout","ActiveSelectionLayoutManager","parents","Set","selectedObjects","ActiveSelection","_options$layoutManage","multiSelectionStacking","findIndex","groups","childrenOverride","Canvas2dFilterBackend","applyFilters","filters","sourceElement","sourceWidth","sourceHeight","pipelineState","imageData","originalEl","originalImageData","filterBackend","applyTo","imageDataPostFilter","putImageData","WebGLFilterBackend","tileSize","Float32Array","setupGLContext","captureGPUInfo","createWebGLCanvas","premultipliedAlpha","depth","stencil","antialias","clearColor","cachedTexture","getCachedTexture","destinationWidth","destinationHeight","sourceTexture","createTexture","targetTexture","originalTexture","passes","webgl","aPosition","programCache","pass","tempFbo","createFramebuffer","bindFramebuffer","FRAMEBUFFER","dWidth","dHeight","resizeCanvasIfNeeded","copyGLTo2D","bindTexture","TEXTURE_2D","deleteTexture","deleteFramebuffer","clearWebGLCaches","textureCache","textureImageSource","NEAREST","RGBA","UNSIGNED_BYTE","CLAMP_TO_EDGE","TEXTURE_MAG_FILTER","TEXTURE_MIN_FILTER","TEXTURE_WRAP_S","TEXTURE_WRAP_T","texture","texParameteri","texImage2D","uniqueId","evictCachesForKey","glCanvas","sourceY","copyGLTo2DPutImageData","numBytes","u8","Uint8Array","imageBuffer","u8Clamped","Uint8ClampedArray","readPixels","imgData","ImageData","gpuInfo","renderer","vendor","ext","UNMASKED_RENDERER_WEBGL","UNMASKED_VENDOR_WEBGL","initFilterBackend","enableGLFiltering","getFilterBackend","setFilterBackend","backend","IMAGE_PROPS","FabricImage","setElement","_element","removeTexture","_originalElement","resizeFilter","applyResizeFilters","elementKey","getCrossOrigin","getOriginalSize","_stroke","filterObj","getSrc","hasCrop","imageMarkup","strokeSvg","imageRendering","imageSmoothing","getSvgSrc","origFill","filtered","srcFromAttribute","setSrc","minimumScale","minimumScaleTrigger","elementToFilter","_filteredEl","_filterScalingX","_filterScalingY","_lastScaleX","_lastScaleY","isNeutralState","imgElement","_needsResize","elementToDraw","elWidth","elHeight","sX","sY","sW","sH","maxDestW","maxDestH","_resetWidthHeight","pAR","preserveAspectRatio","pWidth","pHeight","rWidth","rHeight","f","rf","hydratedProps","fromURL","imageOptions","applyViewboxTransform","viewBoxAttr","widthAttr","heightAttr","missingViewBox","missingDimAttr","translateMatrix","widthDiff","heightDiff","parsedDim","pasedViewBox","createElementNS","firstChild","getTagName","node","tagName","svgInvalidAncestorsRegEx","getMultipleNodes","nodeNames","nodeList","nodeArray","getElementsByTagNameNS","gradientsAttrs","xlinkAttr","recursivelyParseGradientsXlink","_gradient$getAttribut","xLink","referencedGradient","children","referenceClone","cloneNode","tagArray","getCSSRules","allRules","styleContents","ruleObj","propertyValuePairs","pair","_rule","findTag","ElementsParser","clipPaths","regexUrl","gradientDefs","elList","getGradientDefs","createObject","resolveGradient","resolveClipPath","extractPropertyDefinition","storage","lastIndex","gradientDef","usingElement","clipPathElements","objTransformInv","clipPathTag","clipPathOwner","clipPathElement","enlivedClippath","clipRule","gTransform","isValidSvgTag","async","parseSVGDocument","allElements","nodelist","xlinkAttribute","xlink","el2Orig","el2","currentTrans","oldLength","namespace","el3","attrs","setAttributeNS","nodeValue","parseUseDirectives","descendants","hasInvalidAncestor","createEmptyResponse","localClipPaths","elementParser","loadSVGFromString","parseFromString","loadSVGFromURL","xml","responseXML","parsedDoc","ACTION_NAME","createPolyPositionHandler","pointIndex","polyObject","polyActionHandler","poly","mouseLocalPosition","factoryPolyActionHandler","anchorPoint","anchorPointInParentPlane","createPolyActionHandler","rotationWithSnapping","rotateObjectWithSnapping","pivotPoint","lastAngle","curAngle","snapAngle","snapThreshold","rightAngleLocked","leftAngleLocked","hasRotated","createFileDefaultControls","createImageDefaultControls","createObjectArrowControls","createRectNotesDefaultControls","createShapeNotesDefaultControls","rotationStyleHandler","scalingXOrSkewingY","isWebGLPipelineState","isPutImageFaster","testContext","ArrayBuffer","testPipelineState","performance","now","drawImageTime","highPsourceCode","identityFragmentShader","BaseFilter","getFragmentSource","createProgram","vertexSource","vertexShader","VERTEX_SHADER","program","getShaderInfoLog","attachShader","linkProgram","getProgramParameter","LINK_STATUS","getProgramInfoLog","uniformLocations","getUniformLocations","uStepW","getUniformLocation","uStepH","attributeLocations","getAttributeLocations","getAttribLocation","sendAttributeData","aPositionData","attributeLocation","buffer","createBuffer","bindBuffer","ARRAY_BUFFER","enableVertexAttribArray","vertexAttribPointer","FLOAT","bufferData","STATIC_DRAW","_setupFrameBuffer","framebufferTexture2D","COLOR_ATTACHMENT0","finish","_swapTextures","temp","main","mainParameter","defaultValue","thisValue","applyToWebGL","applyTo2d","getCacheKey","retrieveShader","shader","useProgram","uniform1f","sendUniformData","viewport","drawArrays","TRIANGLE_STRIP","bindAdditionalTexture","textureUnit","activeTexture","TEXTURE0","unbindAdditionalTexture","getMainParameter","setMainParameter","createHelpLayer","helpLayer","mainP","blendColorFragmentSource","screen","difference","lighten","darken","exclusion","tint","BlendColor","mode","tg","alpha1","uColor","uniform4fv","mask","BlendImage","image","TEXTURE1","calculateMatrix","resources","blendImage","canvas1","blendData","uTransformMatrix","uImage","uniform1i","uniformMatrix3fv","filterOptions","enlivedImage","Blur","aspectRatio","horizontal","simpleBlur","blurLayer1","blurLayer2","canvas2","ctx1","ctx2","nSamples","percent","newImageData","delta","chooseRightDelta","uniform2fv","blurScale","Brightness","brightness","uBrightness","colorMatrixDefaultValues","colorsOnly","ColorMatrix","uColorMatrix","uConstants","constants","uniformMatrix4fv","createColorMatrixFilter","_Class","newClass","Brownie","Vintage","Kodachrome","Technicolor","Polaroid","Sepia","BlackWhite","Composed","subFilters","enlivedFilters","Contrast","contrast","contrastF","uContrast","Convolute_3_1","Convolute_3_0","Convolute_5_1","Convolute_5_0","Convolute_7_1","Convolute_7_0","Convolute_9_1","Convolute_9_0","Convolute","opaque","weights","side","halfSide","sw","sh","createImageData","dst","alphaFac","dstOff","scx","scy","srcOff","wt","uMatrix","uOpaque","uHalfSize","uSize","uniform1fv","Gamma","gamma","rInv","gInv","bInv","rgbValues","rgb","uGamma","uniform3fv","lightness","luminosity","Grayscale","uMode","HueRotation","rotation","cosine","sine","aThird","aThirdSqtSin","OneMinusCos","Invert","invert","uInvert","uAlpha","Noise","noise","rand","uNoise","uSeed","Pixelate","blocksize","_i","_j","uBlocksize","RemoveColor","getFragmentShader","lowC","highC","uLow","uHigh","useAlpha","Resize","uDelta","uTaps","taps","getFilterWindow","tempScale","lanczosLobes","filterWindow","generateShader","getTaps","lobeFunction","lanczosCreate","fragmentSourceTOP","applyToForWebgl","dW","dH","lobes","xx","rcpScaleX","rcpScaleY","oW","oH","newData","resizeType","sliceByTwo","hermiteFastResize","bilinearFiltering","lanczosResize","mult","doneW","doneH","stepW","stepH","dX","dY","tmpCanvas","srcData","destImg","destData","lanczos","ratioX","ratioY","rcpRatioX","rcpRatioY","range2X","range2Y","cacheLanc","icenter","process","u","weight","fX","fY","xDiff","yDiff","chnl","origPix","w4","pixels","destImage","destPixels","ratioW","ratioH","ratioWHalf","ratioHHalf","img2","data2","weightsAlpha","gxR","gxG","gxB","gxA","yy","w0","Saturation","saturation","adjust","uSaturation","Vibrance","vibrance","amt","uVibrance","targetObject","styleOverride1","EventName","Origin","Textbox","dynamicMinWidth","_styleMap","_generateStyleMap","textInfo","realLineCount","realLineCharCount","splitByGrapheme","isWrapping","nextOffset","nextLineIndex","shouldLimit","mapNextLine","p2Number","_wrapText","desiredWidth","getGraphemeDataForRender","wrapped","wordsData","_wrapLine","infix","largestWordWidth","wordsOrGraphemes","wordSplit","word","graphemeArray","_measureWord","charOffset","_wordJoiners","reservedSpace","additionalSpace","infixWidth","lineJustStarted","wordWidth","getMinWidth","minWidth","linesToKeep","propNumber","has","EntityKeys","XTextbase","oneLine","connectors","fixedScaleChange","boardId","userId","zIndex","updatedAt","lastEditedByName","createdByName","objType","initializeEvent","updateConnector","connector","controlPoint","calculateControlPoint","update","fromPoint","control1","toPoint","control2","moveOrScaleHandler","_this$connectors","_this$connectors2","pointConnector","transformedPoint","connectorObj","findById","connectorId","fromObjectId","connectorType","toObjectId","connectingPoint","boundingBox","connectingX","connectingY","fromCopy","controlMousedownProcess","getObject","checkTextboxChange","self","EDITINGENTERED","tempTop","EDITINGEXITED","Center","MODIFIED","CHANGED","dockingWidget","renderDockingControls","hoveringControl","_renderControl","resetResizeControls","XCircleNotes","graphemeSplitForRectNotes","words","getCenteredTop","rectHeight","verticalAlign","_getTotalLineHeights","_getTotalLineHeight","total","_line","_this$canvas$getZoom","drawRoundRectPath","cxt","noteType","XRectNotes","_options$createdByNam","_options$fontFamily","_options$fontSize","_options$fontWeight","_options$text","_options$textAlign","_options$editable","_options$fixedScaleCh","_options$connectors","_options$id","_options$boardId","_options$backgroundCo","_options$left","_options$locked","_options$objType","_options$originX","_options$originY","_options$scaleX","_options$scaleY","_options$selectable","_options$top","_options$userId","_options$zIndex","_options$version","_options$updatedAt","_options$updatedBy","_options$updatedByNam","_options$createdAt","_options$createdBy","_options$visible","locked","updatedBy","updatedByName","createdAt","createdBy","prenum","substring","shapeList","textMaxHeight","XShapeNotes","_this$bgShape","_this$bgShape2","shape","bgShape","shapeName","minHeight","resetSplitByGrapheme","handleScaling","handleModified","includesChinese","newScaleX","newScaleY","_this$bgShape3","svgPath","Path2D","createSVGMatrix","addPath","_this$bgShape4","_this$bgShape5","XTextbox","setupCustomResizeControls","onEditingEntered","onEditingExited","handleWidthChange","handleWidthChange2","newHeight","oldBoundingRect","originalLeft","originalTop","newBoundingRect","XImage","destW","destH","_stopEvent","cloneWidget","calcPathPointPosition","pathObject","commandIndex","pathPositionHandler","pathActionHandler","movePathPoint","anchorCommand","PathPointControl","controlFill","controlStroke","PathControlPointControl","connectToCommandIndex","connectToPointIndex","connectionDashArray","commandType","createControl","commandIndexPos","pointIndexPos","isControlPoint","controlPointStyle","pointStyle","getPath","XConnector","pathType","pathArrowTip","_setMovementLock","calcStartEndPath","previousCommandType","indexFromPrevCommand","createPathControls","_mouseDownControl","_mouseUpControl","evtOpt","dragActionEventHandler","getFromPoint","getToPoint","lastCommand","pathStart","pathEnd","firstCommand","startArrowSize","startArrow1X","startArrow1Y","startArrow2X","startArrow2Y","endArrowSize","endArrow1X","endArrow1Y","endArrow2X","endArrow2Y","controlPointType","finalCommand","controlPoint1","controlPoint2","TransformPointFromPathToCanvas","newPath","_transform$target$can","getControlPointOnCanvas","controlInfo","currentDockingObject","existingConnectionId","connectedObject","hoverPoint","targetX","targetY","_connectedObject$conn","_connectedObject$conn2","FileEnum","FILE_ICON_PATHS","XFileWord","XFileExcel","XFilePPT","XFilePDF","XFileZip","XFileVideo","XFileAudio","XFile","FILE_TYPE_NAMES","DOC","DOCX","XLS","XLSX","PPT","PPTX","PDF","ZIP","MP4","WEBM","MP3","M4A","WAV","AAC","FLAC","OGG","AIFF","WMA","APE","UNKNOWN","VIDEO_FILE_EXTENSIONS","tmpPath","initializeVisuals","fileObjectType","getFileType","fileName","loadPreviewImage","getFileIconURL","onDoubleClick","_this$fileSrc","fileSrc","drawBorder","drawPreviewImage","renderTitle","_previewImage","titleHeight","availableHeight","imageAspectRatio","finalDrawHeight","getFileTypeName","_fileName$split$pop2","extension","_fileName$split$pop3","isFileVideo","_fileName$split$pop4","title","sanitizedTitle","sanitizeTitle","wrapText","unicodeTitle","toUnicode","parts","escape","getShortenedUrl","lineCount","currentY","testLine","previewImage","XGroup","XURL","_options$previewImage","_options$previewImage2","previewImageURL","extendedProperties","imgWidth","imgHeight","imageAspect","clip","_this$fileSrc5","_this$fileSrc6","_this$fileSrc7","_this$fileSrc2","_this$fileSrc3","_this$fileSrc4","firstChar","lastChar","newurl","loadedImg","hljs","registerLanguage","javascript","XMarkdown","markdownText","md","markdownit","html","xhtmlOut","breaks","langPrefix","linkify","typographer","quotes","highlight","lang","getLanguage","language","__","onScaling","onScaled","renderMarkdown","parseHtmlToImage","renderedImage","blob","Blob","URL","createObjectURL","iframe","_iframe$contentDocume","contentDocument","head","innerHTML","html2canvas","revokeObjectURL","setMarkdown","newMarkdownText","_this$canvas3","Chart","registerables","XChart","chartConfig","addDoubleClickEventListener","createOrUpdateChart","chartInstance","resize","needsUpdate","updateChart","newConfig","openEditModal","modal","boxShadow","saveButton","querySelector","cancelButton","chartDataInput","newChartData","XFrame","titleText","movementX","movementY","checkObjectInFrame","addObject","removeObject","objBound","frameBound","XPath","XCanvas","widget","getAbsoluteCoords","clearData","translateWidget","zoomToViewAllObjects"],"mappings":"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"}