{"version":3,"file":"fabric.js","names":["fabric","version","exports","define","amd","document","window","HTMLDocument","Document","implementation","createHTMLDocument","jsdom","require","virtualWindow","JSDOM","decodeURIComponent","features","FetchExternalResources","resources","jsdomImplForWrapper","implForWrapper","nodeCanvas","Canvas","DOMParser","isTouchSupported","navigator","maxTouchPoints","isLikelyNode","Buffer","SHARED_ATTRIBUTES","DPI","reNum","commaWsp","rePathCommand","reNonWord","fontPaths","iMatrix","svgNS","perfLimitSizeTotal","maxCacheSideLimit","minCacheSideLimit","charWidthsCache","textureSize","disableStyleCopyPaste","enableGLFiltering","devicePixelRatio","webkitDevicePixelRatio","mozDevicePixelRatio","browserShadowBlurConstant","arcToSegmentsCache","boundsOfCurveCache","cachesBoundsOfCurve","forceGLPutImageData","initFilterBackend","isWebglSupported","console","log","maxTextureSize","WebglFilterBackend","tileSize","Canvas2dFilterBackend","_removeEventListener","eventName","handler","this","__eventListeners","eventListener","indexOf","util","array","fill","on","arguments","length","prop","push","_once","_handler","apply","off","bind","once","call","fire","options","listenersForEvent","i","len","filter","value","Observable","Collection","_objects","add","_onObjectAdded","renderOnAddRemove","requestRenderAll","insertAt","object","index","nonSplicing","objects","splice","remove","somethingRemoved","_onObjectRemoved","forEachObject","callback","context","getObjects","type","concat","o","item","isEmpty","size","contains","deep","some","obj","complexity","reduce","memo","current","CommonMethods","_setOptions","set","_initGradient","filler","property","colorStops","Gradient","_initPattern","source","Pattern","_setObject","_set","key","toggle","get","global","sqrt","Math","atan2","pow","PiBy180","PI","PiBy2","cos","angle","angleSlice","sin","sign","removeFromArray","idx","getRandomInt","min","max","floor","random","degreesToRadians","degrees","radiansToDegrees","radians","rotatePoint","point","origin","newPoint","Point","x","y","v","rotateVector","addEquals","vector","rx","ry","createVector","from","to","calcAngleBetweenVectors","a","b","acos","hypot","getHatVector","multiply","getBisector","A","B","C","AB","AC","alpha","ro","phi","projectStrokeOnPoints","points","openPath","coords","s","strokeWidth","strokeUniformScalar","strokeUniform","scaleX","scaleY","getStrokeHatVector","scalar","forEach","p","bisector","bisectorVector","miterVector","strokeLineJoin","strokeMiterLimit","subtract","SQRT2","transformPoint","t","ignoreOffset","makeBoundingBoxFromPoints","transform","xPoints","minX","maxX","width","yPoints","minY","maxY","height","left","top","invertTransform","r","toFixed","number","fractionDigits","parseFloat","Number","parseUnit","fontSize","unit","exec","Text","DEFAULT_SVG_FONT_SIZE","falseFunction","getKlass","namespace","string","camelize","charAt","toUpperCase","slice","resolveNamespace","getSvgAttributes","attributes","parts","split","loadImage","url","crossOrigin","img","createImage","onLoadCallback","onload","onerror","src","undefined","substring","loadImageInDom","div","createElement","style","position","appendChild","querySelector","parentNode","removeChild","enlivenObjects","reviver","enlivenedObjects","numLoadedObjects","numTotalObjects","onLoaded","klass","fromObject","error","enlivenObjectEnlivables","enlivenProps","Object","ENLIVEN_PROPS","map","enlivedProps","enlivenPatterns","patterns","numLoadedPatterns","numPatterns","enlivenedPatterns","pattern","groupSVGElements","elements","path","sourcePath","centerPoint","Group","populateWithProperties","destination","properties","Array","isArray","createCanvasElement","copyCanvasElement","canvas","newCanvas","getContext","willReadFrequently","drawImage","toDataURL","canvasEl","format","quality","multiplyTransformMatrices","is2x2","qrDecompose","denom","skewX","skewY","translateX","translateY","calcRotateMatrix","theta","calcDimensionsMatrix","scaleMatrix","flipX","flipY","tan","composeMatrix","matrix","resetObjectTransform","target","rotate","saveObjectTransform","isTransparent","ctx","tolerance","_isTransparent","temp","imageData","getImageData","l","data","parsePreserveAspectRatioAttribute","attribute","meetOrSlice","alignX","alignY","aspectRatioAttrs","align","pop","clearFabricFontCache","fontFamily","toLowerCase","limitDimsByArea","ar","maximumArea","roughWidth","perfLimitSizeY","capValue","findScaleToFit","findScaleToCover","matrixToSVG","NUM_FRACTION_DIGITS","join","removeTransformFromObject","inverted","finalTransform","calcOwnMatrix","applyTransformToObject","addTransformToObject","center","setPositionByOrigin","sizeAfterTransform","dimX","dimY","transformMatrix","bbox","mergeClipPaths","c1","c2","calcTransformMatrix","clipPath","hasStyleChanged","prevStyle","thisStyle","forTextSpans","stroke","fontWeight","fontStyle","textBackgroundColor","deltaY","overline","underline","linethrough","stylesToArray","styles","text","clone","textLines","charIndex","stylesArray","c","keys","styleChanged","start","end","stylesFromArray","styleIndex","stylesObject","assign","_join","prototype","commandLengths","m","h","q","repeatedCommands","M","segmentToBezier","th2","th3","cosTh","sinTh","cx1","cy1","mT","fromX","fromY","costh2","sinth2","costh3","sinth3","toX","toY","cp1X","cp1Y","cp2X","cp2Y","arcToSegments","large","sweep","rotateX","th","abs","px","py","rx2","ry2","py2","px2","pl","root","cx","cy","mTheta","calcVectorAngle","dtheta","segments","ceil","result","mDelta","ux","uy","vx","vy","ta","tb","getBoundsOfCurve","x0","y0","x1","y1","x2","y2","x3","y3","argsString","tvalues","bounds","t1","t2","b2ac","sqrtb2ac","j","jlen","mt","fromArcToBeziers","fx","fy","rot","tx","ty","segsNorm","makePathSimpler","converted","destinationPath","previous","controlX","controlY","calcLineLength","CB1","CB2","CB3","CB4","getPointOnCubicBezierIterator","p1x","p1y","p2x","p2y","p3x","p3y","p4x","p4y","pct","c3","c4","getTangentCubicIterator","invT","tangentX","tangentY","QB1","QB2","QB3","getPointOnQuadraticBezierIterator","getTangentQuadraticIterator","pathIterator","iterator","tempP","tmpLen","perc","findPercentageForDistance","segInfo","distance","nextLen","nextStep","angleFinder","lastPerc","getPathSegmentsInfo","totalLength","info","tempInfo","command","destX","destY","getPointOnPath","infos","segPercent","segment","lerp","parsePath","pathString","currentPath","parsed","re","rNumber","rNumberCommaWsp","rFlagCommaWsp","rArcSeq","regArcArgumentSequence","RegExp","match","coordsStr","coordsParsed","trim","args","isNaN","commandLength","repeatedCommand","k","klen","getSmoothPathFromPoints","correction","p1","p2","multSignX","multSignY","manyPoints","eq","midPoint","midPointFrom","transformPath","pathOffset","pathSegment","newSegment","joinPath","pathData","invoke","method","byProperty","find","value1","value2","condition","extend","Element","hasOwnProperty","replace","character","capitalize","firstLetterOnly","escapeXml","graphemeSplit","textstring","chr","graphemes","getWholeChar","str","code","charCodeAt","next","prev","emptyFunction","IS_DONTENUM_BUGGY","toString","addMethods","parent","superclass","constructor","returnValue","valueOf","Subclass","callSuper","methodName","parentMethod","_this","superClassMethod","createClass","shift","initialize","subclasses","couldUseAttachEvent","attachEvent","touchEvents","addListener","element","addEventListener","removeListener","removeEventListener","getTouchInfo","event","touchProp","changedTouches","getPointer","scroll","getScrollLeftTop","_evt","clientX","clientY","isTouchEvent","pointerType","setStyle","elementStyle","cssText","setOpacity","normalizedProperty","styleFloat","setProperty","parseEl","supportsOpacity","opacity","supportsFilters","reOpacity","es","currentStyle","hasLayout","zoom","test","_slice","getById","id","getElementById","sliceCanConvertNodelists","toArray","arrayLike","childNodes","err","arr","makeElement","tagName","el","className","htmlFor","setAttribute","addClass","wrapElement","wrapper","replaceChild","docElement","documentElement","body","scrollLeft","scrollTop","host","nodeType","getElementOffset","docElem","doc","ownerDocument","box","offset","scrollLeftTop","offsetAttributes","borderLeftWidth","borderTopWidth","paddingLeft","paddingTop","attr","parseInt","getElementStyle","getBoundingClientRect","clientLeft","clientTop","defaultView","getComputedStyle","selectProp","makeElementUnselectable","onselectstart","unselectable","makeElementSelectable","getNodeCanvas","impl","_canvas","_image","cleanUpJsdomNode","_currentSrc","_attributes","_classList","setImageSmoothing","imageSmoothingEnabled","webkitImageSmoothingEnabled","mozImageSmoothingEnabled","msImageSmoothingEnabled","oImageSmoothingEnabled","addParamToUrl","param","emptyFn","request","onComplete","xhr","XMLHttpRequest","parameters","onreadystatechange","readyState","open","setRequestHeader","send","warn","message","RUNNING_ANIMATIONS","cancelAll","animations","animation","cancel","cancelByCanvas","cancelled","cancelByTarget","findAnimationsByTarget","findAnimationIndex","cancelFunc","findAnimation","noop","defaultEasing","d","animate","removeFromRegistry","runningAnimations","currentValue","startValue","completionRate","durationRate","requestAnimFrame","timestamp","Date","duration","finish","time","onChange","abort","easing","isMany","endValue","byValue","onStart","tick","ticktime","currentTime","timePerc","_value","valuePerc","_requestAnimFrame","requestAnimationFrame","webkitRequestAnimationFrame","mozRequestAnimationFrame","oRequestAnimationFrame","msRequestAnimationFrame","setTimeout","_cancelAnimFrame","cancelAnimationFrame","clearTimeout","cancelAnimFrame","calculateColor","begin","pos","color","animateColor","fromColor","toColor","startColor","Color","getSource","endColor","originalOnComplete","originalOnChange","posValue","colorEasing","normalize","asin","elastic","opts","easeOutCubic","easeInOutCubic","easeInQuart","easeOutQuart","easeInOutQuart","easeInQuint","easeOutQuint","easeInOutQuint","easeInSine","easeOutSine","easeInOutSine","easeInExpo","easeOutExpo","easeInOutExpo","easeInCirc","easeOutCirc","easeInOutCirc","easeInElastic","easeOutElastic","easeInOutElastic","easeInBack","easeOutBack","easeInOutBack","easeInBounce","easeOutBounce","easeInOutBounce","ease","easeInQuad","easeOutQuad","easeInOutQuad","easeInCubic","svgValidTagNames","svgViewBoxElements","svgInvalidAncestors","svgValidParents","attributesMap","display","visibility","colorAttributes","fSize","cPath","svgValidTagNamesRegEx","getSvgRegex","svgViewBoxElementsRegEx","svgInvalidAncestorsRegEx","svgValidParentsRegEx","cssRules","gradientDefs","clipPaths","normalizeAttr","normalizeValue","parentAttributes","parseTransformAttribute","visible","fillIndex","strokeIndex","_setStrokeFillOpacity","setAlpha","getAlpha","toRgba","_getMultipleNodes","nodeNames","nodeName","nodeArray","nodeList","getElementsByTagName","rotateMatrix","multiplierX","multiplierY","skewMatrix","translateMatrix","scale","translate","transforms","transformList","reTransformList","reTransform","attributeValue","matrices","operation","combinedMatrix","parseStyleString","oStyle","chunk","pair","parseStyleObject","getGlobalStylesForElement","svgUid","rule","elementMatchesRule","selectors","firstMatching","parentMatching","selectorMatches","doesSomeParentMatch","selector","classNames","getAttribute","matcher","elementById","node","nodelist","parseUseDirectives","xlinkAttribute","xlink","el2","cloneNode","currentTrans","oldLength","attrs","applyViewboxTransform","el3","createElementNS","setAttributeNS","nodeValue","firstChild","removeAttribute","reViewBoxAttrValue","viewBoxAttr","viewBoxWidth","viewBoxHeight","widthAttr","heightAttr","preserveAspectRatio","missingViewBox","missingDimAttr","toBeParsed","parsedDim","widthDiff","heightDiff","hasAncestorWithNodeName","parseSVGDocument","parsingOptions","__uid","descendants","selectNodes","getGradientDefs","getCSSRules","parseElements","instances","recursivelyParseGradientsXlink","gradient","gradientsAttrs","xlinkAttr","xLink","referencedGradient","hasAttribute","children","referenceClone","reFontDeclaration","parseFontDeclaration","lineHeight","tagArray","elList","parseAttributes","parentFontSize","ownAttributes","cssAttrs","parseStyleAttribute","normalizedAttr","normalizedValue","normalizedStyle","font","mergedAttrs","ElementsParser","parse","parsePointsAttribute","parsedPoints","allRules","rules","styleContents","textContent","ruleObj","declaration","propertyValuePairs","_rule","loadSVGFromURL","xml","responseXML","results","_options","allElements","loadSVGFromString","parser","parseFromString","regexUrl","proto","numElements","createObjects","createObject","findTag","fromElement","createCallback","checkIfDone","resolveGradient","Image","_originalElement","_removeTransformMatrix","resolveClipPath","extractPropertyDefinition","storage","regex","lastIndex","gradientDef","opacityAttr","createClipPathCallback","container","_newObj","fillRule","clipRule","usingElement","objTransformInv","gTransform","clipPathTag","clipPathOwner","that","scalarAdd","scalarAddEquals","subtractEquals","scalarSubtract","scalarSubtractEquals","multiplyEquals","divide","divideEquals","lt","lte","gt","gte","distanceFrom","dx","dy","setXY","setX","setY","setFromPoint","swap","Intersection","status","appendPoint","appendPoints","intersectLineLine","a1","a2","b1","b2","uaT","ubT","uB","ua","ub","intersectLinePolygon","inter","intersectPolygonPolygon","points1","points2","intersectPolygonRectangle","r1","r2","topRight","bottomLeft","inter1","inter2","inter3","inter4","setSource","_tryParsingColor","colorNameMap","sourceFromHex","sourceFromRgb","sourceFromHsl","_rgbToHsl","g","round","_source","toRgb","toHsl","hsl","toHsla","toHex","toHexa","toGrayscale","average","currentAlpha","toBlackWhite","threshold","overlayWith","otherColor","otherAlpha","otherSource","reRGBa","reHSLa","reHex","aliceblue","antiquewhite","aqua","aquamarine","EntraID","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","fromRgb","fromSource","fromRgba","fromHsl","fromHsla","fromHex","isShortNotation","isRGBa","oColor","scaleMap","skewMap","controls","LEFT","TOP","RIGHT","BOTTOM","CENTER","opposite","bottom","right","findCornerQuadrant","fabricObject","control","cornerAngle","fireEvent","canvasOptions","scaleIsProportional","eventData","uniScaleKey","uniformIsToggled","uniformScaling","isTransformCentered","originX","originY","scalingIsForbidden","by","scaleProportionally","lockX","lockScalingX","lockY","lockScalingY","scaleCursorStyleHandler","notAllowed","n","skewCursorStyleHandler","lockSkewingY","lockSkewingX","scaleSkewCursorStyleHandler","altActionKey","scaleOrSkewActionName","isAlternative","rotationStyleHandler","lockRotation","cursorStyle","commonEventInfo","e","pointer","wrapWithFixedAnchor","actionHandler","getCenterPoint","constraint","translateToOriginPoint","actionPerformed","wrapWithFireEvent","getLocalPoint","corner","getZoom","padding","localPoint","toLocalPoint","offsetX","offsetY","targetHasOneFlip","compensateScaleForSkew","oppositeSkew","scaleToCompensate","axis","reference","newDim","_getTransformedDimensions","newValue","skewObjectX","dimNoSkew","totalSkewSize","currentSkew","newSkew","hasSkewed","dimBeforeSkewing","skewObjectY","skewHandlerX","localPointFromCenter","finalHandler","skewHandlerY","rotationWithSnapping","pivotPoint","lastAngle","ey","ex","curAngle","hasRotated","snapAngle","snapThreshold","rightAngleLocked","leftAngleLocked","scaleObject","dim","forbidScaling","signX","signY","gestureScale","lockScalingFlip","original","originalDistance","oldScaleX","oldScaleY","scaleObjectFromCorner","scaleObjectX","scaleObjectY","scalingYOrSkewingX","scalingY","scalingXOrSkewingY","scalingX","changeWidth","strokePadding","multiplier","oldWidth","newWidth","dragHandler","newLeft","newTop","moveX","moveY","scalingEqually","controlsUtils","renderCircleControl","styleOverride","xSize","sizeX","cornerSize","ySize","sizeY","transparentCorners","cornerStrokeColor","myLeft","myTop","save","fillStyle","cornerColor","strokeStyle","lineWidth","beginPath","arc","restore","renderSquareControl","xSizeBy2","ySizeBy2","strokeRect","Control","actionName","touchSizeX","touchSizeY","withConnection","mouseDownHandler","mouseUpHandler","getActionHandler","getMouseDownHandler","getMouseUpHandler","cursorStyleHandler","getActionName","getVisibility","controlKey","objectVisibility","_controlsVisibility","setVisibility","positionHandler","finalMatrix","calcCornerCoords","objectAngle","objectCornerSize","centerX","centerY","isTouch","cosHalfOffset","sinHalfOffset","cosHalfOffsetComp","sinHalfOffsetComp","controlTriangleAngle","cornerHypotenuse","newTheta","newThetaComp","tl","tr","bl","br","render","cornerStyle","getColorStop","colorAlpha","keyValuePairs","getLinearCoords","getRadialCoords","gradientTransform","gradientUnits","option","addColorStop","toObject","propertiesToInclude","toSVG","markup","commonAttributes","needsSwap","withViewport","additionalTransform","sort","reverse","minRadius","maxRadius","percentageShift","colorStop","toLive","createLinearGradient","createRadialGradient","instance","svgOptions","colorStopEls","__convertPercentUnitsToValues","propValue","finalValue","repeat","patternTransform","setOptions","isError","patternSource","patternWidth","patternHeight","patternOffsetX","patternOffsetY","patternImgSrc","complete","naturalWidth","naturalHeight","createPattern","Shadow","blur","affectStroke","includeDefaultValues","nonScaling","_parseShadow","shadow","shadowStr","offsetsAndBlur","reOffsetsAndBlur","fBoxX","fBoxY","BLUR_BOX","StaticCanvas","CANVAS_INIT_ERROR","Error","renderAndResetBound","renderAndReset","requestRenderAllBound","_initStatic","backgroundColor","backgroundImage","overlayColor","overlayImage","stateful","controlsAboveOverlay","allowTouchScrolling","viewportTransform","backgroundVpt","overlayVpt","enableRetinaScaling","vptCoords","skipOffscreen","cb","_createLowerCanvas","_initOptions","interactive","_initRetinaScaling","setOverlayImage","setBackgroundImage","setBackgroundColor","setOverlayColor","calcOffset","_isRetinaScaling","getRetinaScaling","scaleRatio","__initRetinaScaling","lowerCanvasEl","contextContainer","upperCanvasEl","contextTop","_offset","image","__setBgOverlayImage","__setBgOverlayColor","_createCanvasElement","_originalCanvasStyle","_applyCanvasStyle","getWidth","getHeight","setWidth","setDimensions","setHeight","dimensions","cssValue","cssOnly","_setBackstoreDimension","hasLostContext","backstoreOnly","_setCssDimension","_isCurrentlyDrawing","freeDrawingBrush","_setBrushStyles","cacheCanvasEl","wrapperEl","setViewportTransform","vpt","activeObject","_activeObject","backgroundObject","overlayObject","group","setCoords","calcViewportBoundaries","zoomToPoint","before","after","setZoom","absolutePan","relativePan","getElement","setupState","clearContext","clearRect","clear","_hasITextHandlers","_mouseUpITextHandler","_iTextInstances","renderAll","canvasToDrawOn","renderCanvas","isRendering","iVpt","cancelRequestedRender","_renderBackground","_renderObjects","drawControls","shouldCache","_transformDone","renderCache","forClipping","drawClipPathOnCanvas","_renderOverlay","globalCompositeOperation","zoomX","zoomY","_cacheCanvas","cacheTranslationX","cacheTranslationY","_renderBackgroundOrOverlay","needsVpt","moveTo","lineTo","closePath","getCenter","centerObjectH","_centerObject","centerObjectV","centerObject","viewportCenterObject","vpCenter","getVpCenter","viewportCenterObjectH","viewportCenterObjectV","toDatalessJSON","toDatalessObject","_toObjectMethod","_toObjects","excludeFromExport","_toObject","__serializeBgOverlay","originalValue","bgImage","bgColor","background","overlay","svgViewportTransformation","_setSVGPreamble","_setSVGHeader","clipPathId","_setSVGBgOverlayColor","_setSVGBgOverlayImage","_setSVGObjects","suppressPreamble","encoding","viewBox","createSVGFontFacesMarkup","createSVGRefElementsMarkup","createSVGClipPathMarkup","toClipPathSVG","shouldTransform","fontList","row","rowIndex","_char","_setSVGObject","finalWidth","finalHeight","shouldInvert","sendToBack","activeSelection","objs","unshift","bringToFront","sendBackwards","intersecting","newIdx","objsMoved","_findNewLowerIndex","isIntersecting","intersectsWithObject","isContainedWithinObject","bringForward","_findNewUpperIndex","dispose","classList","DataURLExporter","EMPTY_JSON","supports","setLineDash","toJSON","createPNGStream","createJPEGStream","BaseBrush","strokeLineCap","strokeDashArray","limitedToCanvasSize","lineCap","miterLimit","lineJoin","_saveAndTransform","_setShadow","shadowColor","shadowBlur","shadowOffsetX","shadowOffsetY","needsFullRender","_resetShadow","_isOutSideCanvas","PencilBrush","decimate","drawStraightLine","straightLineKey","_points","_hasStraightLine","_drawSegment","quadraticCurveTo","onMouseDown","_isMainEvent","_prepareForDrawing","_captureDrawingPath","_render","onMouseMove","oldEnd","onMouseUp","_finalizeAndAddPath","_reset","_addPoint","pointerPoint","convertPointsToSVGPath","_isEmptySVGPath","createPath","Path","decimatePoints","adjustedDistance","lastPoint","newPoints","cDistance","CircleBrush","drawDot","addPoint","dot","radius","originalRenderOnAddRemove","circles","circle","Circle","circleRadius","circleColor","SprayBrush","density","dotWidth","dotWidthVariance","randomOpacity","optimizeOverlapping","sprayChunks","addSprayChunk","sprayChunkPoints","rects","ilen","sprayChunk","rect","Rect","_getOptimizedRects","uniqueRects","uniqueRectsArray","globalAlpha","fillRect","PatternBrush","getPatternSrc","dotDistance","patternCanvas","patternCtx","getPatternSrcFunction","String","getPattern","topLeft","_getLeftTopCoords","_initInteractive","_createCacheCanvas","centeredScaling","centeredRotation","centeredKey","selection","selectionKey","altSelectionKey","selectionColor","selectionDashArray","selectionBorderColor","selectionLineWidth","selectionFullyContained","hoverCursor","moveCursor","defaultCursor","freeDrawingCursor","notAllowedCursor","containerClass","perPixelTargetFind","targetFindTolerance","skipTargetFind","isDrawingMode","preserveObjectStacking","stopContextMenu","fireRightClick","fireMiddleClick","targets","enablePointerEvents","_hoveredTarget","_hoveredTargets","_currentTransform","_groupSelector","_initWrapperElement","_createUpperCanvas","_initEventListeners","_chooseObjectsToRender","activeObjects","getActiveObjects","objsToRender","activeGroupObjects","contextTopDirty","renderTopLayer","_drawSelection","renderTop","_normalizePointer","invertedM","vptPointer","restorePointerVpt","isTargetTransparent","normalizedPointer","targetRelativeX","targetRelativeY","_cacheContext","contextCache","originalColor","selectionBackgroundColor","_isSelectionKeyPressed","selectionKeyPressed","_shouldClearSelection","evented","selectable","_shouldCenterTransform","action","altKey","centerTransform","_getOriginFromCorner","_getActionFromCorner","alreadySelected","_setupCurrentTransform","__corner","lastX","lastY","shiftKey","_beforeTransform","setCursor","cursor","viewportStart","viewportExtent","extent","strokeOffset","_setLineDash","findTarget","skipGroup","ignoreZoom","aObjects","activeTarget","activeTargetSubs","shouldLookForActive","_findTargetCorner","_searchPossibleTargets","_checkTarget","globalPointer","containsPoint","isEditing","subTarget","objToCheck","pointerToUse","subTargetCheck","_absolutePointer","_pointer","boundsWidth","boundsHeight","cssScale","retinaScaling","lowerCanvasClass","_copyCanvasStyle","getTopContext","class","fromEl","toEl","getSelectionContext","getSelectionElement","getActiveObject","active","_discardActiveObject","_fireSelectionEvents","oldObjects","somethingChanged","added","removed","oldObject","selected","deselected","setActiveObject","currentActives","_setActiveObject","onSelect","onDeselect","discardActiveObject","removeListeners","_renderControls","originalProperties","_realizeGroupTransformOnObject","_unwindGroupTransformOnObject","layoutProps","originalValues","clearContextTop","RIGHT_CLICK","MIDDLE_CLICK","LEFT_CLICK","addEventOptions","passive","checkClick","button","mainTouchId","_bindEvents","addOrRemove","_getEventPrefix","functor","eventjsFunctor","canvasElement","eventTypePrefix","_onResize","_onMouseDown","_onMouseMove","_onMouseOut","_onMouseEnter","_onMouseWheel","_onContextMenu","_onDoubleClick","_onDragOver","_onDragEnter","_onDragLeave","_onDrop","_onTouchStart","eventjs","_onGesture","_onDrag","_onOrientationChange","_onShake","_onLongPress","_onMouseUp","_onTouchEnd","eventsBound","_simpleEventHandler","self","__onTransformGesture","__onDrag","__onMouseWheel","_target","__onOrientationChange","__onShake","__onLongPress","preventDefault","_fireEnterLeaveEvents","stopPropagation","_cacheTransformEventData","_handleEvent","_resetTransformEventData","getPointerId","evt","identifier","pointerId","isPrimary","touches","__onMouseDown","__onMouseUp","_willAddMouseDown","__onMouseMove","_shouldRender","groupSelector","shouldRender","isClick","_onMouseUpInDrawingMode","_finalizeCurrentTransform","targetWasActive","_maybeGroupObjects","activeOn","isMoving","originalControl","originalMouseUpHandler","_setCursorFromEvent","eventType","subTargets","absolutePointer","currentTarget","currentSubTargets","_scaling","hasStateChanged","_fire","_onMouseDownInDrawingMode","_onMouseMoveInDrawingMode","_previousPointer","shouldGroup","_shouldGroup","_handleGrouping","saveState","_fireOverOutEvents","_transformObject","fireSyntheticInOutEvents","oldTarget","evtOut","canvasEvtOut","evtIn","canvasEvtIn","_draggedoverTarget","config","inOpt","outOpt","outFires","inFires","targetChanged","previousTarget","nextTarget","reset","_performTransformAction","getCornerCursor","_updateActiveSelection","_createActiveSelection","currentActiveObjects","removeWithUpdate","addWithUpdate","_createGroup","isActiveLower","groupObjects","exitEditing","ActiveSelection","_groupSelectedObjects","_collectObjects","aGroup","currentObject","selectionX1Y1","selectionX2Y2","allowIntersect","intersectsWithRect","isContainedWithinRect","toCanvasElement","cropping","scaledWidth","scaledHeight","originalWidth","originalHeight","newZoom","vp","originalInteractive","newVp","originalRetina","originalContextTop","loadFromJSON","json","serialized","JSON","_enlivenObjects","_setBgOverlay","enlivenedCanvasClip","__setupCanvas","loaded","cbIfLoaded","__setBgOverlay","enlivedObject","_toDataURL","_toDataURLWithMultiplier","toDataURLWithMultiplier","stringify","cloneWithoutData","backgroundImageOpacity","backgroundImageStretch","objectCaching","ALIASING_LIMIT","touchCornerSize","borderColor","borderDashArray","cornerDashArray","strokeDashOffset","borderOpacityWhenMoving","borderScaleFactor","minScaleLimit","hasControls","hasBorders","lockMovementX","lockMovementY","statefullCache","noScaleCache","dirty","paintFirst","stateProperties","cacheProperties","colorProperties","absolutePositioned","_cacheProperties","_updateCacheCanvas","_limitCacheSize","dims","limitedDims","capped","_getCacheCanvasDimensions","objectScale","getTotalObjectScaling","neededX","neededY","targetCanvas","minCacheSize","drawingWidth","drawingHeight","dimensionsChanged","cacheWidth","cacheHeight","zoomChanged","shouldRedraw","additionalWidth","additionalHeight","shouldResizeCanvas","canvasWidth","canvasHeight","sizeGrowing","sizeShrinking","getHeightOfLine","setTransform","needFullTransform","_removeDefaultValues","getObjectScaling","retina","getObjectOpacity","shouldConstrainValue","isChanged","groupNeedsUpdate","_constrainScale","isOnACache","setOnGroup","getViewportTransform","isNotVisible","isOnScreen","_setupCompositeOperation","drawSelectionBackground","_setOpacity","drawCacheOnCanvas","_removeCacheCanvas","drawObject","propertySet","isCacheDirty","hasStroke","hasFill","needsItsOwnCache","ownCaching","willDrawShadow","drawClipPathOnCache","originalFill","originalStroke","_setClippingProperties","_drawClipPath","skipCanvas","_getNonTransformedDimensions","_removeShadow","_setStrokeStyles","decl","lineDashOffset","_applyPatternForTransformedGradient","_applyPatternGradientTransform","_setFillStyles","dashArray","drawBorders","forActiveSelection","drawBordersInGroup","scaling","multX","multY","_renderPaintInOrder","_renderStroke","_renderFill","pCanvas","pCtx","_findCenterFromElement","_assignTransformMatrixProps","preserveAspectRatioOptions","cropX","cropY","offsetLeft","offsetTop","objectForm","_fromObject","cloneAsImage","utils","origParams","originalGroup","originalShadow","withoutTransform","withoutShadow","boundingRect","getBoundingRect","shadowOffset","originalCanvas","isType","includes","shouldCenterOrigin","_setOriginToCenter","_resetOrigin","centerH","viewportCenterH","centerV","viewportCenterV","viewportCenter","getLocalPointer","pClicked","objectLeftTop","createAccessors","extraParam","originXOffset","originYOffset","translateToGivenOrigin","fromOriginX","fromOriginY","toOriginX","toOriginY","translateToCenterPoint","leftTop","getPointByOrigin","adjustPosition","hypotFull","getScaledWidth","xFull","yFull","offsetFrom","offsetTo","_originalOriginX","_originalOriginY","originPoint","arrayFromCoords","multiplyMatrices","oCoords","aCoords","lineCoords","ownMatrixCache","matrixCache","_getCoords","absolute","calculate","calcACoords","calcLineCoords","getCoords","pointTL","pointBR","intersection","other","otherCoords","lines","_getImageLines","_findCrossPoints","_containsCenterOfCanvas","isPartiallyOnScreen","allPointsAreOutside","every","topline","rightline","bottomline","leftline","xi","xcount","iLine","lineKey","getScaledHeight","scaleToWidth","boundingRectFactor","scaleToHeight","cosP","sinP","cosPSinP","cosPMinusSinP","calcOCoords","_calcRotateMatrix","_calcTranslateMatrix","startMatrix","_calculateCurrentDimensions","forEachControl","w","skipCorners","_setCornerCoords","transformMatrixKey","sep","prefix","cache","tMatrix","noSkew","_finalizeDimensions","getSvgColorString","getSvgStyles","skipShadow","getSvgFilter","getSvgSpanStyles","useWhiteSpace","term","textDecoration","getSvgTextDecoration","decoration","getSvgCommons","getSvgTransform","full","svgTransform","_setSVGBg","textBgRects","_getFillAttributes","_createBaseSVGMarkup","_toSVG","_createBaseClipPathSVGMarkup","objectMarkup","commonPieces","noStyle","styleInfo","shadowInfo","withShadow","vectorEffect","absoluteClipPath","clipPathMarkup","addPaintOrder","originalSet","saveProps","props","tmpObj","_isEqual","origValue","firstPass","dashedPropertySet","forTouch","isControlVisible","touchCorner","fn","controlObject","wh","shouldStroke","setControlVisible","setControlsVisibility","FX_DURATION","fxCenterObjectH","callbacks","empty","fxCenterObjectV","fxRemove","propsToAnimate","skipCallbacks","out","_animate","propPair","propIsColor","valueProgress","timeProgress","coordProps","Line","_setWidthHeight","_getLeftToOriginX","_getTopToOriginY","makeEdgeToOriginGetter","axis1","axis2","dimension","nearest","farthest","calcLinePoints","origStrokeStyle","xMult","yMult","ATTRIBUTE_NAMES","parsedAttributes","_callback","propertyNames","originValues","startAngle","endAngle","setRadius","svgString","startX","startY","endX","endY","largeFlag","getRadiusX","getRadiusY","isValidRadius","Triangle","widthBy2","heightBy2","piBy2","Ellipse","getRx","getRy","_initRxRy","isRounded","bezierCurveTo","Polyline","exactBoundingBox","_setPositionDimensions","_projectStrokeOnPoints","calcDim","_calcDimensions","correctLeftTop","correctSize","fromSVG","diffX","diffY","commonRender","fromElementGenerator","_class","Polygon","_setPath","_renderPathCommands","subpathStartX","subpathStartY","_getOffsetTransform","digits","aX","aY","deltaX","pathUrl","elivenedObjects","useSetOnGroup","isAlreadyGrouped","_calcBounds","_updateObjectsCoords","_updateObjectsACoords","skipControls","_updateObjectCoords","objectLeft","objectTop","nested","_restoreObjectsState","_includeDefaultValues","objsToObject","originalDefaults","_obj","ownCache","groupMatrix","destroy","toActiveSelection","ungroupOnCanvas","setObjectsCoords","onlyWidthHeight","iLen","jLen","_getBounds","minXY","maxXY","toGroup","newGroup","childrenOverride","srcFromAttribute","_lastScaleX","_lastScaleY","_filterScalingX","_filterScalingY","minimumScaleTrigger","cacheKey","imageSmoothing","filters","_initElement","_element","setElement","removeTexture","_initConfig","applyFilters","resizeFilter","applyResizeFilters","backend","filterBackend","evictCachesForKey","getCrossOrigin","getOriginalSize","_stroke","filterObj","getSrc","hasCrop","imageMarkup","strokeSvg","imageRendering","getSvgSrc","origFill","filtered","setSrc","minimumScale","elementToFilter","_filteredEl","sourceWidth","sourceHeight","isNeutralState","imgElement","_needsResize","elementToDraw","elWidth","elHeight","sX","sY","sW","sH","maxDestW","maxDestH","_resetWidthHeight","CSS_CANVAS","_initFilters","pAR","rWidth","rHeight","pWidth","pHeight","_object","resizeFilters","fromURL","imgOptions","_getAngleValueForStraighten","straighten","fxStraighten","straightenObject","fxStraightenObject","testPrecision","gl","precision","fragmentSource","fragmentShader","createShader","FRAGMENT_SHADER","shaderSource","compileShader","getShaderParameter","COMPILE_STATUS","isSupported","getParameter","MAX_TEXTURE_SIZE","precisions","webGlPrecision","setupGLContext","captureGPUInfo","createWebGLCanvas","aPosition","Float32Array","chooseFastestCopyGLTo2DMethod","canMeasurePerf","performance","canUseImageData","ImageData","canUseArrayBuffer","ArrayBuffer","canUseUint8Clamped","Uint8ClampedArray","imageBuffer","copyGLTo2D","copyGLTo2DPutImageData","testContext","destinationWidth","destinationHeight","startTime","drawImageTime","putImageDataTime","now","copyGLTo2DDrawImage","glOptions","premultipliedAlpha","depth","stencil","antialias","clearColor","cachedTexture","getCachedTexture","pipelineState","sourceTexture","createTexture","targetTexture","originalTexture","passes","webgl","programCache","pass","tempFbo","createFramebuffer","bindFramebuffer","FRAMEBUFFER","applyTo","resizeCanvasIfNeeded","bindTexture","TEXTURE_2D","deleteTexture","deleteFramebuffer","clearWebGLCaches","textureCache","textureImageSource","filterType","texture","texParameteri","TEXTURE_MAG_FILTER","NEAREST","TEXTURE_MIN_FILTER","TEXTURE_WRAP_S","CLAMP_TO_EDGE","TEXTURE_WRAP_T","texImage2D","RGBA","UNSIGNED_BYTE","uniqueId","gpuInfo","renderer","vendor","ext","getExtension","UNMASKED_RENDERER_WEBGL","UNMASKED_VENDOR_WEBGL","dWidth","dHeight","glCanvas","sourceY","numBytes","u8","Uint8Array","u8Clamped","readPixels","imgData","putImageData","sourceElement","originalImageData","originalEl","BaseFilter","vertexSource","createProgram","vertexShader","VERTEX_SHADER","getShaderInfoLog","program","attachShader","linkProgram","getProgramParameter","LINK_STATUS","getProgramInfoLog","attributeLocations","getAttributeLocations","uniformLocations","getUniformLocations","uStepW","getUniformLocation","uStepH","getAttribLocation","sendAttributeData","aPositionData","attributeLocation","buffer","createBuffer","bindBuffer","ARRAY_BUFFER","enableVertexAttribArray","vertexAttribPointer","FLOAT","bufferData","STATIC_DRAW","_setupFrameBuffer","framebufferTexture2D","COLOR_ATTACHMENT0","_swapTextures","main","mainParameter","applyToWebGL","applyTo2d","retrieveShader","shader","useProgram","uniform1f","sendUniformData","viewport","drawArrays","TRIANGLE_STRIP","bindAdditionalTexture","textureUnit","activeTexture","TEXTURE0","unbindAdditionalTexture","getMainParameter","setMainParameter","createHelpLayer","helpLayer","mainP","ColorMatrix","colorsOnly","uColorMatrix","uConstants","constants","uniformMatrix4fv","uniform4fv","Brightness","brightness","uBrightness","Convolute","opaque","Convolute_3_1","Convolute_3_0","Convolute_5_1","Convolute_5_0","Convolute_7_1","Convolute_7_0","Convolute_9_1","Convolute_9_0","weights","side","halfSide","sw","sh","output","createImageData","dst","alphaFac","dstOff","scx","scy","srcOff","wt","uMatrix","uOpaque","uHalfSize","uSize","uniform1fv","Grayscale","lightness","luminosity","mode","uMode","uniform1i","Invert","invert","uInvert","Noise","noise","rand","uNoise","uSeed","Pixelate","blocksize","_i","_j","_iLen","_jLen","uBlocksize","RemoveColor","useAlpha","lowC","highC","uLow","uHigh","Brownie","Vintage","Kodachrome","Technicolor","Polaroid","Sepia","BlackWhite","BlendColor","screen","diff","lighten","darken","exclusion","tint","buildSource","tg","alpha1","uColor","BlendImage","mask","TEXTURE1","calculateMatrix","canvas1","blendData","blendImage","uTransformMatrix","uImage","uniformMatrix3fv","Resize","resizeType","lanczosLobes","uDelta","uTaps","uniform2fv","horizontal","taps","filterWindow","getFilterWindow","generateShader","tempScale","getTaps","lobeFunction","lanczosCreate","offsets","fragmentSourceTOP","dW","dH","lobes","xx","rcpScaleX","rcpScaleY","oW","oH","newData","sliceByTwo","hermiteFastResize","bilinearFiltering","lanczosResize","mult","doneW","doneH","stepW","stepH","tmpCanvas","dX","dY","process","u","weight","fX","fY","ratioX","icenter","ratioY","range2X","cacheLanc","range2Y","lanczos","rcpRatioX","rcpRatioY","srcData","destData","destImg","xDiff","yDiff","chnl","origPix","w4","pixels","destImage","destPixels","ratioW","ratioH","ratioWHalf","ratioHHalf","img2","data2","weightsAlpha","gxR","gxG","gxB","gxA","yy","w0","Contrast","contrast","contrastF","uContrast","Saturation","saturation","adjust","uSaturation","Vibrance","vibrance","avg","amt","uVibrance","Blur","aspectRatio","simpleBlur","canvas2","blurLayer1","blurLayer2","ctx1","ctx2","nSamples","percent","newImageData","delta","chooseRightDelta","blurScale","Gamma","gamma","rInv","gInv","bInv","rVals","gVals","bVals","uGamma","uniform3fv","Composed","subFilters","HueRotation","rotation","rad","aThird","aThirdSqtSin","OneMinusCos","additionalProps","_dimensionAffectingProps","_reNewline","_reSpacesAndTabs","_reSpaceAndTab","_reWords","textAlign","superscript","baseline","subscript","pathStartOffset","pathSide","pathAlign","_fontSizeFraction","_fontSizeMult","charSpacing","_measuringContext","direction","_styleProperties","__charBounds","CACHE_FONT_SIZE","MIN_TEXT_WIDTH","__skipDimension","setPathInfo","initDimensions","segmentsInfo","getMeasuringContext","_splitText","newLines","_splitTextIntoLines","_textLines","graphemeLines","_unwrappedTextLines","_unwrappedLines","_text","graphemeText","_clearCache","calcTextWidth","cursorWidth","calcTextHeight","enlargeSpaces","diffSpace","currentLineWidth","numberOfSpaces","accumulatedSpace","line","charBound","spaces","isEndOfWrapping","getLineWidth","kernedWidth","lineIndex","missingNewlineOffset","_setTextStyles","_renderTextLinesBackground","_renderTextDecoration","_renderText","_renderTextStroke","_renderTextFill","charStyle","forMeasuring","textBaseline","_getFontDeclaration","maxWidth","_renderTextLine","_renderChars","styleHas","heightOfLine","lineLeftOffset","lastColor","leftOffset","_getLeftOffset","lineTopOffset","_getTopOffset","boxStart","boxWidth","charBox","currentColor","drawStart","_getLineLeftOffset","getValueOfPropertyAt","renderLeft","getFontCache","cacheProp","_measureChar","previousChar","prevCharStyle","fontCache","fontDeclaration","previousFontDeclaration","couple","stylesAreEqual","coupleWidth","previousWidth","fontMultiplier","measureText","getHeightOfChar","measureLine","lineInfo","_measureLine","_getWidthOfCharSpacing","grapheme","prevGrapheme","graphemeInfo","numOfSpaces","lineBounds","positionInPath","startingPoint","totalPathLength","_getGraphemeBox","_setGraphemeOnPath","centerPosition","skipLeft","getCompleteStyleDeclaration","previousBox","__lineHeights","maxHeight","_renderTextCommon","lineHeights","isEmptyStyles","isJustify","actualStyle","nextStyle","charsToRender","timeToRender","shortCut","isLtr","drawingLeft","currentDirection","_renderChar","_applyPatternGradientTransformText","handleFiller","_getStyleDeclaration","fullDecl","shouldFill","fillOffsets","strokeOffsets","fillText","strokeText","setSuperscript","_setScript","setSubscript","schema","loc","get2DCursorLocation","setSelectionStyles","lineDiff","__lineWidths","_shouldClearDimensionCache","shouldClear","_forceClearCache","_size","_dy","lastDecoration","topOffset","currentDecoration","currentFill","lastFill","styleObject","family","fontIsGeneric","genericFonts","newLine","newText","allProperties","needsDims","isAddingPath","_key","parsedAnchor","textAnchor","originalStrokeWidth","textHeightScaleFactor","lineHeightDiff","scaledDiff","textHeight","offX","objectCopy","textInstance","pathInstance","p3","cleanStyle","stylesCount","letterCount","stylePropertyValue","allStyleObjectPropertiesMatch","graphemeCount","stylePropertyHasBeenSet","removeStyle","lineNum","charNum","_extendStyles","_getLineStyle","_setLineStyle","_setStyleDeclaration","selectionStart","skipWrapping","getSelectionStyles","startIndex","endIndex","selectionEnd","getStyleAtPosition","lineStyle","_deleteStyleDeclaration","_deleteLineStyle","parseDecoration","IText","editable","editingBorderColor","cursorColor","cursorDelay","cursorDuration","caching","hiddenTextareaContainer","_reSpace","_currentCursorOpacity","_selectionDirection","_abortCursorAnimation","__widthOfSpace","inCompositionMode","initBehavior","setSelectionStart","_updateAndFire","setSelectionEnd","_fireSelectionChanged","_updateTextarea","initDelayedCursor","cursorOffsetCache","renderCursorOrSelection","skipRestore","_clearTextArea","boundaries","_getCursorBoundaries","renderCursor","renderSelection","_getCursorBoundariesOffsets","cursorPosition","bound","cursorLocation","charHeight","__isMousedown","hiddenTextarea","startLine","endLine","startChar","endChar","lineOffset","realLineHeight","boxEnd","drawWidth","drawHeight","extraTop","compositionColor","getCurrentCharFontSize","cp","_getCurrentCharIndex","getCurrentCharColor","objCopy","initAddedHandler","initRemovedHandler","initCursorSelectionHandlers","initDoubleClickSimulation","mouseMoveHandler","_initCanvasHandlers","_removeCanvasHandlers","_tick","_currentTickState","_animateCursor","targetOpacity","completeMethod","tickState","isAborted","_onTickComplete","_cursorTimeout1","_currentTickCompleteState","restart","delay","abortCursorAnimation","_cursorTimeout2","selectAll","getSelectedText","findWordBoundaryLeft","startFrom","findWordBoundaryRight","findLineBoundaryLeft","findLineBoundaryRight","searchWordBoundary","selectWord","newSelectionStart","newSelectionEnd","selectLine","enterEditing","exitEditingOnOthers","initHiddenTextarea","focus","_saveEditingProps","_setEditingProps","_textBeforeEdit","initMouseMoveHandler","activeElement","getSelectionStartFromPointer","currentStart","currentEnd","__selectionStartOnMouseDown","restartCursorIfNeeded","fromStringToGraphemeSelection","smallerTextStart","graphemeStart","smallerTextEnd","graphemeEnd","fromGraphemeToStringSelection","newSelection","updateTextareaPosition","updateFromTextArea","_calcTextareaPosition","desiredPosition","compositionStart","upperCanvas","upperCanvasWidth","upperCanvasHeight","clientWidth","clientHeight","_savedProps","_restoreEditingProps","isTextChanged","_removeExtraneousStyles","removeStyleFromTo","cursorStart","cursorEnd","lineStart","charStart","lineEnd","charEnd","styleObj","shiftLineStyles","numericChar","clonedStyles","numericLine","insertNewlineStyleObject","qty","copiedStyle","currentCharStyle","newLineStyles","somethingAdded","isEndOfLine","numIndex","styleCarriedOver","insertCharStyleObject","quantity","currentLineStyles","currentLineStylesCloned","numericIndex","newStyle","insertNewStyleBlock","insertedText","cursorLoc","addedLines","linesLength","setSelectionStartEndWithShift","setSelectionInBoundaries","__lastClickTime","__lastLastClickTime","__lastPointer","__newClickTime","newPointer","isTripleClick","_stopEvent","__lastIsEditing","__lastSelected","initMousedownHandler","initMouseupHandler","initClicks","doubleClickHandler","tripleClickHandler","_mouseDownHandler","setCursorByClick","_mouseDownHandlerBefore","currentActive","mouseOffset","prevWidth","_getNewSelectionStartFromOffset","distanceBtwLastCharAndCursor","distanceBtwNextCharAndCursor","onKeyDown","onKeyUp","onInput","copy","paste","onCompositionStart","onCompositionUpdate","onCompositionEnd","_clickHandlerInitialized","onClick","keysMap","keysMapRtl","ctrlKeysMapUp","ctrlKeysMapDown","keyMap","keyCode","ctrlKey","metaKey","stopImmediatePropagation","_copyDone","fromPaste","nextText","charCount","nextCharCount","removedText","charDiff","removeFrom","removeTo","textareaSelection","backDelete","copiedText","copiedTextStyle","compositionEnd","_getClipboardData","clipboardData","_getWidthBeforeCursor","widthBeforeCursor","getDownCursorOffset","isRight","selectionProp","_getSelectionForOffset","indexOnOtherLine","_getIndexOnLine","textAfterCursor","getUpCursorOffset","textBeforeCursor","widthOfCharsOnLine","indexOnLine","charWidth","foundMatch","leftEdge","rightEdge","offsetFromLeftEdge","offsetFromRightEdge","moveCursorDown","_moveCursorUpOrDown","moveCursorUp","moveCursorWithShift","moveCursorWithoutShift","moveCursorLeft","_moveCursorLeftOrRight","_move","_moveLeft","_moveRight","moveCursorLeftWithoutShift","change","moveCursorLeftWithShift","moveCursorRight","moveCursorRightWithShift","moveCursorRightWithoutShift","changed","removeChars","insertChars","multipleSpacesRegex","_getSVGLeftTopOffsets","textAndBg","_getSVGTextAndBg","textTop","textLeft","_wrapSVGTextAndBg","lineTop","noShadow","textSpans","textTopOffset","textLeftOffset","_setSVGTextLineBg","_setSVGTextLineText","_createTextCharSpan","styleDecl","shouldUseWhitespace","styleProps","fillStyles","dySpan","_pushTextBgRect","fillColor","_getSVGLineTopOffset","lastHeight","svgStyle","Textbox","minWidth","dynamicMinWidth","__cachedLines","_wordJoiners","splitByGrapheme","_styleMap","_generateStyleMap","textInfo","realLineCount","realLineCharCount","isWrapping","nextLineIndex","nextOffset","shouldLimit","mapNextLine","_wrapText","desiredWidth","wrapped","_wrapLine","_measureWord","word","charOffset","_line","reservedSpace","words","infix","wordWidth","infixWidth","largestWordWidth","lineJustStarted","additionalSpace","getMinWidth","linesToKeep","scaleSkewStyleHandler","scaleStyleHandler","objectControls","ml","mr","mb","mtr","textBoxControls","defaultOnTouchStartHandler"],"sources":["../../../scripts/leanforms/fabric.js"],"sourcesContent":["/* build: `node build.js modules=ALL exclude=gestures,accessors,erasing requirejs minifier=uglifyjs` */\r\n/*! Fabric.js Copyright 2008-2015, Printio (Juriy Zaytsev, Maxim Chernyak) */\r\n\r\nvar fabric = fabric || { version: '5.3.0' };\r\nif (typeof exports !== 'undefined') {\r\n exports.fabric = fabric;\r\n}\r\n/* _AMD_START_ */\r\nelse if (typeof define === 'function' && define.amd) {\r\n define([], function() { return fabric; });\r\n}\r\n/* _AMD_END_ */\r\nif (typeof document !== 'undefined' && typeof window !== 'undefined') {\r\n if (document instanceof (typeof HTMLDocument !== 'undefined' ? HTMLDocument : Document)) {\r\n fabric.document = document;\r\n }\r\n else {\r\n fabric.document = document.implementation.createHTMLDocument('');\r\n }\r\n fabric.window = window;\r\n}\r\nelse {\r\n // assume we're running under node.js when document/window are not present\r\n var jsdom = require('jsdom');\r\n var virtualWindow = new jsdom.JSDOM(\r\n decodeURIComponent('%3C!DOCTYPE%20html%3E%3Chtml%3E%3Chead%3E%3C%2Fhead%3E%3Cbody%3E%3C%2Fbody%3E%3C%2Fhtml%3E'),\r\n {\r\n features: {\r\n FetchExternalResources: ['img']\r\n },\r\n resources: 'usable'\r\n }).window;\r\n fabric.document = virtualWindow.document;\r\n fabric.jsdomImplForWrapper = require('jsdom/lib/jsdom/living/generated/utils').implForWrapper;\r\n fabric.nodeCanvas = require('jsdom/lib/jsdom/utils').Canvas;\r\n fabric.window = virtualWindow;\r\n DOMParser = fabric.window.DOMParser;\r\n}\r\n\r\n/**\r\n * True when in environment that supports touch events\r\n * @type boolean\r\n */\r\nfabric.isTouchSupported = 'ontouchstart' in fabric.window || 'ontouchstart' in fabric.document ||\r\n (fabric.window && fabric.window.navigator && fabric.window.navigator.maxTouchPoints > 0);\r\n\r\n/**\r\n * True when in environment that's probably Node.js\r\n * @type boolean\r\n */\r\nfabric.isLikelyNode = typeof Buffer !== 'undefined' &&\r\n typeof window === 'undefined';\r\n\r\n/* _FROM_SVG_START_ */\r\n/**\r\n * Attributes parsed from all SVG elements\r\n * @type array\r\n */\r\nfabric.SHARED_ATTRIBUTES = [\r\n 'display',\r\n 'transform',\r\n 'fill', 'fill-opacity', 'fill-rule',\r\n 'opacity',\r\n 'stroke', 'stroke-dasharray', 'stroke-linecap', 'stroke-dashoffset',\r\n 'stroke-linejoin', 'stroke-miterlimit',\r\n 'stroke-opacity', 'stroke-width',\r\n 'id', 'paint-order', 'vector-effect',\r\n 'instantiated_by_use', 'clip-path',\r\n];\r\n/* _FROM_SVG_END_ */\r\n\r\n/**\r\n * Pixel per Inch as a default value set to 96. Can be changed for more realistic conversion.\r\n */\r\nfabric.DPI = 96;\r\nfabric.reNum = '(?:[-+]?(?:\\\\d+|\\\\d*\\\\.\\\\d+)(?:[eE][-+]?\\\\d+)?)';\r\nfabric.commaWsp = '(?:\\\\s+,?\\\\s*|,\\\\s*)';\r\nfabric.rePathCommand = /([-+]?((\\d+\\.\\d+)|((\\d+)|(\\.\\d+)))(?:[eE][-+]?\\d+)?)/ig;\r\nfabric.reNonWord = /[ \\n\\.,;!\\?\\-]/;\r\nfabric.fontPaths = { };\r\nfabric.iMatrix = [1, 0, 0, 1, 0, 0];\r\nfabric.svgNS = 'http://www.w3.org/2000/svg';\r\n\r\n/**\r\n * Pixel limit for cache canvases. 1Mpx , 4Mpx should be fine.\r\n * @since 1.7.14\r\n * @type Number\r\n * @default\r\n */\r\nfabric.perfLimitSizeTotal = 2097152;\r\n\r\n/**\r\n * Pixel limit for cache canvases width or height. IE fixes the maximum at 5000\r\n * @since 1.7.14\r\n * @type Number\r\n * @default\r\n */\r\nfabric.maxCacheSideLimit = 4096;\r\n\r\n/**\r\n * Lowest pixel limit for cache canvases, set at 256PX\r\n * @since 1.7.14\r\n * @type Number\r\n * @default\r\n */\r\nfabric.minCacheSideLimit = 256;\r\n\r\n/**\r\n * Cache Object for widths of chars in text rendering.\r\n */\r\nfabric.charWidthsCache = { };\r\n\r\n/**\r\n * if webgl is enabled and available, textureSize will determine the size\r\n * of the canvas backend\r\n * @since 2.0.0\r\n * @type Number\r\n * @default\r\n */\r\nfabric.textureSize = 2048;\r\n\r\n/**\r\n * When 'true', style information is not retained when copy/pasting text, making\r\n * pasted text use destination style.\r\n * Defaults to 'false'.\r\n * @type Boolean\r\n * @default\r\n */\r\nfabric.disableStyleCopyPaste = false;\r\n\r\n/**\r\n * Enable webgl for filtering picture is available\r\n * A filtering backend will be initialized, this will both take memory and\r\n * time since a default 2048x2048 canvas will be created for the gl context\r\n * @since 2.0.0\r\n * @type Boolean\r\n * @default\r\n */\r\nfabric.enableGLFiltering = true;\r\n\r\n/**\r\n * Device Pixel Ratio\r\n * @see https://developer.apple.com/library/safari/documentation/AudioVideo/Conceptual/HTML-canvas-guide/SettingUptheCanvas/SettingUptheCanvas.html\r\n */\r\nfabric.devicePixelRatio = fabric.window.devicePixelRatio ||\r\n fabric.window.webkitDevicePixelRatio ||\r\n fabric.window.mozDevicePixelRatio ||\r\n 1;\r\n/**\r\n * Browser-specific constant to adjust CanvasRenderingContext2D.shadowBlur value,\r\n * which is unitless and not rendered equally across browsers.\r\n *\r\n * Values that work quite well (as of October 2017) are:\r\n * - Chrome: 1.5\r\n * - Edge: 1.75\r\n * - Firefox: 0.9\r\n * - Safari: 0.95\r\n *\r\n * @since 2.0.0\r\n * @type Number\r\n * @default 1\r\n */\r\nfabric.browserShadowBlurConstant = 1;\r\n\r\n/**\r\n * This object contains the result of arc to bezier conversion for faster retrieving if the same arc needs to be converted again.\r\n * It was an internal variable, is accessible since version 2.3.4\r\n */\r\nfabric.arcToSegmentsCache = { };\r\n\r\n/**\r\n * This object keeps the results of the boundsOfCurve calculation mapped by the joined arguments necessary to calculate it.\r\n * It does speed up calculation, if you parse and add always the same paths, but in case of heavy usage of freedrawing\r\n * you do not get any speed benefit and you get a big object in memory.\r\n * The object was a private variable before, while now is appended to the lib so that you have access to it and you\r\n * can eventually clear it.\r\n * It was an internal variable, is accessible since version 2.3.4\r\n */\r\nfabric.boundsOfCurveCache = { };\r\n\r\n/**\r\n * If disabled boundsOfCurveCache is not used. For apps that make heavy usage of pencil drawing probably disabling it is better\r\n * @default true\r\n */\r\nfabric.cachesBoundsOfCurve = true;\r\n\r\n/**\r\n * Skip performance testing of setupGLContext and force the use of putImageData that seems to be the one that works best on\r\n * Chrome + old hardware. if your users are experiencing empty images after filtering you may try to force this to true\r\n * this has to be set before instantiating the filtering backend ( before filtering the first image )\r\n * @type Boolean\r\n * @default false\r\n */\r\nfabric.forceGLPutImageData = false;\r\n\r\nfabric.initFilterBackend = function() {\r\n if (fabric.enableGLFiltering && fabric.isWebglSupported && fabric.isWebglSupported(fabric.textureSize)) {\r\n console.log('max texture size: ' + fabric.maxTextureSize);\r\n return (new fabric.WebglFilterBackend({ tileSize: fabric.textureSize }));\r\n }\r\n else if (fabric.Canvas2dFilterBackend) {\r\n return (new fabric.Canvas2dFilterBackend());\r\n }\r\n};\r\n\r\n\r\nif (typeof document !== 'undefined' && typeof window !== 'undefined') {\r\n // ensure globality even if entire library were function wrapped (as in Meteor.js packaging system)\r\n window.fabric = fabric;\r\n}\r\n\r\n\r\n(function() {\r\n\r\n /**\r\n * @private\r\n * @param {String} eventName\r\n * @param {Function} handler\r\n */\r\n function _removeEventListener(eventName, handler) {\r\n if (!this.__eventListeners[eventName]) {\r\n return;\r\n }\r\n var eventListener = this.__eventListeners[eventName];\r\n if (handler) {\r\n eventListener[eventListener.indexOf(handler)] = false;\r\n }\r\n else {\r\n fabric.util.array.fill(eventListener, false);\r\n }\r\n }\r\n\r\n /**\r\n * Observes specified event\r\n * @memberOf fabric.Observable\r\n * @alias on\r\n * @param {String|Object} eventName Event name (eg. 'after:render') or object with key/value pairs (eg. {'after:render': handler, 'selection:cleared': handler})\r\n * @param {Function} handler Function that receives a notification when an event of the specified type occurs\r\n * @return {Self} thisArg\r\n * @chainable\r\n */\r\n function on(eventName, handler) {\r\n if (!this.__eventListeners) {\r\n this.__eventListeners = { };\r\n }\r\n // one object with key/value pairs was passed\r\n if (arguments.length === 1) {\r\n for (var prop in eventName) {\r\n this.on(prop, eventName[prop]);\r\n }\r\n }\r\n else {\r\n if (!this.__eventListeners[eventName]) {\r\n this.__eventListeners[eventName] = [];\r\n }\r\n this.__eventListeners[eventName].push(handler);\r\n }\r\n return this;\r\n }\r\n\r\n function _once(eventName, handler) {\r\n var _handler = function () {\r\n handler.apply(this, arguments);\r\n this.off(eventName, _handler);\r\n }.bind(this);\r\n this.on(eventName, _handler);\r\n }\r\n\r\n function once(eventName, handler) {\r\n // one object with key/value pairs was passed\r\n if (arguments.length === 1) {\r\n for (var prop in eventName) {\r\n _once.call(this, prop, eventName[prop]);\r\n }\r\n }\r\n else {\r\n _once.call(this, eventName, handler);\r\n }\r\n return this;\r\n }\r\n\r\n /**\r\n * Stops event observing for a particular event handler. Calling this method\r\n * without arguments removes all handlers for all events\r\n * @memberOf fabric.Observable\r\n * @alias off\r\n * @param {String|Object} eventName Event name (eg. 'after:render') or object with key/value pairs (eg. {'after:render': handler, 'selection:cleared': handler})\r\n * @param {Function} handler Function to be deleted from EventListeners\r\n * @return {Self} thisArg\r\n * @chainable\r\n */\r\n function off(eventName, handler) {\r\n if (!this.__eventListeners) {\r\n return this;\r\n }\r\n\r\n // remove all key/value pairs (event name -> event handler)\r\n if (arguments.length === 0) {\r\n for (eventName in this.__eventListeners) {\r\n _removeEventListener.call(this, eventName);\r\n }\r\n }\r\n // one object with key/value pairs was passed\r\n else if (arguments.length === 1 && typeof arguments[0] === 'object') {\r\n for (var prop in eventName) {\r\n _removeEventListener.call(this, prop, eventName[prop]);\r\n }\r\n }\r\n else {\r\n _removeEventListener.call(this, eventName, handler);\r\n }\r\n return this;\r\n }\r\n\r\n /**\r\n * Fires event with an optional options object\r\n * @memberOf fabric.Observable\r\n * @param {String} eventName Event name to fire\r\n * @param {Object} [options] Options object\r\n * @return {Self} thisArg\r\n * @chainable\r\n */\r\n function fire(eventName, options) {\r\n if (!this.__eventListeners) {\r\n return this;\r\n }\r\n\r\n var listenersForEvent = this.__eventListeners[eventName];\r\n if (!listenersForEvent) {\r\n return this;\r\n }\r\n\r\n for (var i = 0, len = listenersForEvent.length; i < len; i++) {\r\n listenersForEvent[i] && listenersForEvent[i].call(this, options || { });\r\n }\r\n this.__eventListeners[eventName] = listenersForEvent.filter(function(value) {\r\n return value !== false;\r\n });\r\n return this;\r\n }\r\n\r\n /**\r\n * @namespace fabric.Observable\r\n * @tutorial {@link http://fabricjs.com/fabric-intro-part-2#events}\r\n * @see {@link http://fabricjs.com/events|Events demo}\r\n */\r\n fabric.Observable = {\r\n fire: fire,\r\n on: on,\r\n once: once,\r\n off: off,\r\n };\r\n})();\r\n\r\n\r\n/**\r\n * @namespace fabric.Collection\r\n */\r\nfabric.Collection = {\r\n\r\n _objects: [],\r\n\r\n /**\r\n * Adds objects to collection, Canvas or Group, then renders canvas\r\n * (if `renderOnAddRemove` is not `false`).\r\n * in case of Group no changes to bounding box are made.\r\n * Objects should be instances of (or inherit from) fabric.Object\r\n * Use of this function is highly discouraged for groups.\r\n * you can add a bunch of objects with the add method but then you NEED\r\n * to run a addWithUpdate call for the Group class or position/bbox will be wrong.\r\n * @param {...fabric.Object} object Zero or more fabric instances\r\n * @return {Self} thisArg\r\n * @chainable\r\n */\r\n add: function () {\r\n this._objects.push.apply(this._objects, arguments);\r\n if (this._onObjectAdded) {\r\n for (var i = 0, length = arguments.length; i < length; i++) {\r\n this._onObjectAdded(arguments[i]);\r\n }\r\n }\r\n this.renderOnAddRemove && this.requestRenderAll();\r\n return this;\r\n },\r\n\r\n /**\r\n * Inserts an object into collection at specified index, then renders canvas (if `renderOnAddRemove` is not `false`)\r\n * An object should be an instance of (or inherit from) fabric.Object\r\n * Use of this function is highly discouraged for groups.\r\n * you can add a bunch of objects with the insertAt method but then you NEED\r\n * to run a addWithUpdate call for the Group class or position/bbox will be wrong.\r\n * @param {Object} object Object to insert\r\n * @param {Number} index Index to insert object at\r\n * @param {Boolean} nonSplicing When `true`, no splicing (shifting) of objects occurs\r\n * @return {Self} thisArg\r\n * @chainable\r\n */\r\n insertAt: function (object, index, nonSplicing) {\r\n var objects = this._objects;\r\n if (nonSplicing) {\r\n objects[index] = object;\r\n }\r\n else {\r\n objects.splice(index, 0, object);\r\n }\r\n this._onObjectAdded && this._onObjectAdded(object);\r\n this.renderOnAddRemove && this.requestRenderAll();\r\n return this;\r\n },\r\n\r\n /**\r\n * Removes objects from a collection, then renders canvas (if `renderOnAddRemove` is not `false`)\r\n * @param {...fabric.Object} object Zero or more fabric instances\r\n * @return {Self} thisArg\r\n * @chainable\r\n */\r\n remove: function() {\r\n var objects = this._objects,\r\n index, somethingRemoved = false;\r\n\r\n for (var i = 0, length = arguments.length; i < length; i++) {\r\n index = objects.indexOf(arguments[i]);\r\n\r\n // only call onObjectRemoved if an object was actually removed\r\n if (index !== -1) {\r\n somethingRemoved = true;\r\n objects.splice(index, 1);\r\n this._onObjectRemoved && this._onObjectRemoved(arguments[i]);\r\n }\r\n }\r\n\r\n this.renderOnAddRemove && somethingRemoved && this.requestRenderAll();\r\n return this;\r\n },\r\n\r\n /**\r\n * Executes given function for each object in this group\r\n * @param {Function} callback\r\n * Callback invoked with current object as first argument,\r\n * index - as second and an array of all objects - as third.\r\n * Callback is invoked in a context of Global Object (e.g. `window`)\r\n * when no `context` argument is given\r\n *\r\n * @param {Object} context Context (aka thisObject)\r\n * @return {Self} thisArg\r\n * @chainable\r\n */\r\n forEachObject: function(callback, context) {\r\n var objects = this.getObjects();\r\n for (var i = 0, len = objects.length; i < len; i++) {\r\n callback.call(context, objects[i], i, objects);\r\n }\r\n return this;\r\n },\r\n\r\n /**\r\n * Returns an array of children objects of this instance\r\n * Type parameter introduced in 1.3.10\r\n * since 2.3.5 this method return always a COPY of the array;\r\n * @param {String} [type] When specified, only objects of this type are returned\r\n * @return {Array}\r\n */\r\n getObjects: function(type) {\r\n if (typeof type === 'undefined') {\r\n return this._objects.concat();\r\n }\r\n return this._objects.filter(function(o) {\r\n return o.type === type;\r\n });\r\n },\r\n\r\n /**\r\n * Returns object at specified index\r\n * @param {Number} index\r\n * @return {Self} thisArg\r\n */\r\n item: function (index) {\r\n return this._objects[index];\r\n },\r\n\r\n /**\r\n * Returns true if collection contains no objects\r\n * @return {Boolean} true if collection is empty\r\n */\r\n isEmpty: function () {\r\n return this._objects.length === 0;\r\n },\r\n\r\n /**\r\n * Returns a size of a collection (i.e: length of an array containing its objects)\r\n * @return {Number} Collection size\r\n */\r\n size: function() {\r\n return this._objects.length;\r\n },\r\n\r\n /**\r\n * Returns true if collection contains an object\r\n * @param {Object} object Object to check against\r\n * @param {Boolean} [deep=false] `true` to check all descendants, `false` to check only `_objects`\r\n * @return {Boolean} `true` if collection contains an object\r\n */\r\n contains: function (object, deep) {\r\n if (this._objects.indexOf(object) > -1) {\r\n return true;\r\n }\r\n else if (deep) {\r\n return this._objects.some(function (obj) {\r\n return typeof obj.contains === 'function' && obj.contains(object, true);\r\n });\r\n }\r\n return false;\r\n },\r\n\r\n /**\r\n * Returns number representation of a collection complexity\r\n * @return {Number} complexity\r\n */\r\n complexity: function () {\r\n return this._objects.reduce(function (memo, current) {\r\n memo += current.complexity ? current.complexity() : 0;\r\n return memo;\r\n }, 0);\r\n }\r\n};\r\n\r\n\r\n/**\r\n * @namespace fabric.CommonMethods\r\n */\r\nfabric.CommonMethods = {\r\n\r\n /**\r\n * Sets object's properties from options\r\n * @param {Object} [options] Options object\r\n */\r\n _setOptions: function(options) {\r\n for (var prop in options) {\r\n this.set(prop, options[prop]);\r\n }\r\n },\r\n\r\n /**\r\n * @private\r\n * @param {Object} [filler] Options object\r\n * @param {String} [property] property to set the Gradient to\r\n */\r\n _initGradient: function(filler, property) {\r\n if (filler && filler.colorStops && !(filler instanceof fabric.Gradient)) {\r\n this.set(property, new fabric.Gradient(filler));\r\n }\r\n },\r\n\r\n /**\r\n * @private\r\n * @param {Object} [filler] Options object\r\n * @param {String} [property] property to set the Pattern to\r\n * @param {Function} [callback] callback to invoke after pattern load\r\n */\r\n _initPattern: function(filler, property, callback) {\r\n if (filler && filler.source && !(filler instanceof fabric.Pattern)) {\r\n this.set(property, new fabric.Pattern(filler, callback));\r\n }\r\n else {\r\n callback && callback();\r\n }\r\n },\r\n\r\n /**\r\n * @private\r\n */\r\n _setObject: function(obj) {\r\n for (var prop in obj) {\r\n this._set(prop, obj[prop]);\r\n }\r\n },\r\n\r\n /**\r\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()`.\r\n * @param {String|Object} key Property name or object (if object, iterate over the object properties)\r\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)\r\n * @return {fabric.Object} thisArg\r\n * @chainable\r\n */\r\n set: function(key, value) {\r\n if (typeof key === 'object') {\r\n this._setObject(key);\r\n }\r\n else {\r\n this._set(key, value);\r\n }\r\n return this;\r\n },\r\n\r\n _set: function(key, value) {\r\n this[key] = value;\r\n },\r\n\r\n /**\r\n * Toggles specified property from `true` to `false` or from `false` to `true`\r\n * @param {String} property Property to toggle\r\n * @return {fabric.Object} thisArg\r\n * @chainable\r\n */\r\n toggle: function(property) {\r\n var value = this.get(property);\r\n if (typeof value === 'boolean') {\r\n this.set(property, !value);\r\n }\r\n return this;\r\n },\r\n\r\n /**\r\n * Basic getter\r\n * @param {String} property Property name\r\n * @return {*} value of a property\r\n */\r\n get: function(property) {\r\n return this[property];\r\n }\r\n};\r\n\r\n\r\n(function(global) {\r\n\r\n var sqrt = Math.sqrt,\r\n atan2 = Math.atan2,\r\n pow = Math.pow,\r\n PiBy180 = Math.PI / 180,\r\n PiBy2 = Math.PI / 2;\r\n\r\n /**\r\n * @namespace fabric.util\r\n */\r\n fabric.util = {\r\n\r\n /**\r\n * Calculate the cos of an angle, avoiding returning floats for known results\r\n * @static\r\n * @memberOf fabric.util\r\n * @param {Number} angle the angle in radians or in degree\r\n * @return {Number}\r\n */\r\n cos: function(angle) {\r\n if (angle === 0) { return 1; }\r\n if (angle < 0) {\r\n // cos(a) = cos(-a)\r\n angle = -angle;\r\n }\r\n var angleSlice = angle / PiBy2;\r\n switch (angleSlice) {\r\n case 1: case 3: return 0;\r\n case 2: return -1;\r\n }\r\n return Math.cos(angle);\r\n },\r\n\r\n /**\r\n * Calculate the sin of an angle, avoiding returning floats for known results\r\n * @static\r\n * @memberOf fabric.util\r\n * @param {Number} angle the angle in radians or in degree\r\n * @return {Number}\r\n */\r\n sin: function(angle) {\r\n if (angle === 0) { return 0; }\r\n var angleSlice = angle / PiBy2, sign = 1;\r\n if (angle < 0) {\r\n // sin(-a) = -sin(a)\r\n sign = -1;\r\n }\r\n switch (angleSlice) {\r\n case 1: return sign;\r\n case 2: return 0;\r\n case 3: return -sign;\r\n }\r\n return Math.sin(angle);\r\n },\r\n\r\n /**\r\n * Removes value from an array.\r\n * Presence of value (and its position in an array) is determined via `Array.prototype.indexOf`\r\n * @static\r\n * @memberOf fabric.util\r\n * @param {Array} array\r\n * @param {*} value\r\n * @return {Array} original array\r\n */\r\n removeFromArray: function(array, value) {\r\n var idx = array.indexOf(value);\r\n if (idx !== -1) {\r\n array.splice(idx, 1);\r\n }\r\n return array;\r\n },\r\n\r\n /**\r\n * Returns random number between 2 specified ones.\r\n * @static\r\n * @memberOf fabric.util\r\n * @param {Number} min lower limit\r\n * @param {Number} max upper limit\r\n * @return {Number} random value (between min and max)\r\n */\r\n getRandomInt: function(min, max) {\r\n return Math.floor(Math.random() * (max - min + 1)) + min;\r\n },\r\n\r\n /**\r\n * Transforms degrees to radians.\r\n * @static\r\n * @memberOf fabric.util\r\n * @param {Number} degrees value in degrees\r\n * @return {Number} value in radians\r\n */\r\n degreesToRadians: function(degrees) {\r\n return degrees * PiBy180;\r\n },\r\n\r\n /**\r\n * Transforms radians to degrees.\r\n * @static\r\n * @memberOf fabric.util\r\n * @param {Number} radians value in radians\r\n * @return {Number} value in degrees\r\n */\r\n radiansToDegrees: function(radians) {\r\n return radians / PiBy180;\r\n },\r\n\r\n /**\r\n * Rotates `point` around `origin` with `radians`\r\n * @static\r\n * @memberOf fabric.util\r\n * @param {fabric.Point} point The point to rotate\r\n * @param {fabric.Point} origin The origin of the rotation\r\n * @param {Number} radians The radians of the angle for the rotation\r\n * @return {fabric.Point} The new rotated point\r\n */\r\n rotatePoint: function(point, origin, radians) {\r\n var newPoint = new fabric.Point(point.x - origin.x, point.y - origin.y),\r\n v = fabric.util.rotateVector(newPoint, radians);\r\n return new fabric.Point(v.x, v.y).addEquals(origin);\r\n },\r\n\r\n /**\r\n * Rotates `vector` with `radians`\r\n * @static\r\n * @memberOf fabric.util\r\n * @param {Object} vector The vector to rotate (x and y)\r\n * @param {Number} radians The radians of the angle for the rotation\r\n * @return {Object} The new rotated point\r\n */\r\n rotateVector: function(vector, radians) {\r\n var sin = fabric.util.sin(radians),\r\n cos = fabric.util.cos(radians),\r\n rx = vector.x * cos - vector.y * sin,\r\n ry = vector.x * sin + vector.y * cos;\r\n return {\r\n x: rx,\r\n y: ry\r\n };\r\n },\r\n\r\n /**\r\n * Creates a vetor from points represented as a point\r\n * @static\r\n * @memberOf fabric.util\r\n *\r\n * @typedef {Object} Point\r\n * @property {number} x\r\n * @property {number} y\r\n *\r\n * @param {Point} from\r\n * @param {Point} to\r\n * @returns {Point} vector\r\n */\r\n createVector: function (from, to) {\r\n return new fabric.Point(to.x - from.x, to.y - from.y);\r\n },\r\n\r\n /**\r\n * Calculates angle between 2 vectors using dot product\r\n * @static\r\n * @memberOf fabric.util\r\n * @param {Point} a\r\n * @param {Point} b\r\n * @returns the angle in radian between the vectors\r\n */\r\n calcAngleBetweenVectors: function (a, b) {\r\n return Math.acos((a.x * b.x + a.y * b.y) / (Math.hypot(a.x, a.y) * Math.hypot(b.x, b.y)));\r\n },\r\n\r\n /**\r\n * @static\r\n * @memberOf fabric.util\r\n * @param {Point} v\r\n * @returns {Point} vector representing the unit vector of pointing to the direction of `v`\r\n */\r\n getHatVector: function (v) {\r\n return new fabric.Point(v.x, v.y).multiply(1 / Math.hypot(v.x, v.y));\r\n },\r\n\r\n /**\r\n * @static\r\n * @memberOf fabric.util\r\n * @param {Point} A\r\n * @param {Point} B\r\n * @param {Point} C\r\n * @returns {{ vector: Point, angle: number }} vector representing the bisector of A and A's angle\r\n */\r\n getBisector: function (A, B, C) {\r\n var AB = fabric.util.createVector(A, B), AC = fabric.util.createVector(A, C);\r\n var alpha = fabric.util.calcAngleBetweenVectors(AB, AC);\r\n // check if alpha is relative to AB->BC\r\n var ro = fabric.util.calcAngleBetweenVectors(fabric.util.rotateVector(AB, alpha), AC);\r\n var phi = alpha * (ro === 0 ? 1 : -1) / 2;\r\n return {\r\n vector: fabric.util.getHatVector(fabric.util.rotateVector(AB, phi)),\r\n angle: alpha\r\n };\r\n },\r\n\r\n /**\r\n * Project stroke width on points returning 2 projections for each point as follows:\r\n * - `miter`: 2 points corresponding to the outer boundary and the inner boundary of stroke.\r\n * - `bevel`: 2 points corresponding to the bevel boundaries, tangent to the bisector.\r\n * - `round`: same as `bevel`\r\n * Used to calculate object's bounding box\r\n * @static\r\n * @memberOf fabric.util\r\n * @param {Point[]} points\r\n * @param {Object} options\r\n * @param {number} options.strokeWidth\r\n * @param {'miter'|'bevel'|'round'} options.strokeLineJoin\r\n * @param {number} options.strokeMiterLimit https://developer.mozilla.org/en-US/docs/Web/SVG/Attribute/stroke-miterlimit\r\n * @param {boolean} options.strokeUniform\r\n * @param {number} options.scaleX\r\n * @param {number} options.scaleY\r\n * @param {boolean} [openPath] whether the shape is open or not, affects the calculations of the first and last points\r\n * @returns {fabric.Point[]} array of size 2n/4n of all suspected points\r\n */\r\n projectStrokeOnPoints: function (points, options, openPath) {\r\n var coords = [], s = options.strokeWidth / 2,\r\n strokeUniformScalar = options.strokeUniform ?\r\n new fabric.Point(1 / options.scaleX, 1 / options.scaleY) : new fabric.Point(1, 1),\r\n getStrokeHatVector = function (v) {\r\n var scalar = s / (Math.hypot(v.x, v.y));\r\n return new fabric.Point(v.x * scalar * strokeUniformScalar.x, v.y * scalar * strokeUniformScalar.y);\r\n };\r\n if (points.length <= 1) {return coords;}\r\n points.forEach(function (p, index) {\r\n var A = new fabric.Point(p.x, p.y), B, C;\r\n if (index === 0) {\r\n C = points[index + 1];\r\n B = openPath ? getStrokeHatVector(fabric.util.createVector(C, A)).addEquals(A) : points[points.length - 1];\r\n }\r\n else if (index === points.length - 1) {\r\n B = points[index - 1];\r\n C = openPath ? getStrokeHatVector(fabric.util.createVector(B, A)).addEquals(A) : points[0];\r\n }\r\n else {\r\n B = points[index - 1];\r\n C = points[index + 1];\r\n }\r\n var bisector = fabric.util.getBisector(A, B, C),\r\n bisectorVector = bisector.vector,\r\n alpha = bisector.angle,\r\n scalar,\r\n miterVector;\r\n if (options.strokeLineJoin === 'miter') {\r\n scalar = -s / Math.sin(alpha / 2);\r\n miterVector = new fabric.Point(\r\n bisectorVector.x * scalar * strokeUniformScalar.x,\r\n bisectorVector.y * scalar * strokeUniformScalar.y\r\n );\r\n if (Math.hypot(miterVector.x, miterVector.y) / s <= options.strokeMiterLimit) {\r\n coords.push(A.add(miterVector));\r\n coords.push(A.subtract(miterVector));\r\n return;\r\n }\r\n }\r\n scalar = -s * Math.SQRT2;\r\n miterVector = new fabric.Point(\r\n bisectorVector.x * scalar * strokeUniformScalar.x,\r\n bisectorVector.y * scalar * strokeUniformScalar.y\r\n );\r\n coords.push(A.add(miterVector));\r\n coords.push(A.subtract(miterVector));\r\n });\r\n return coords;\r\n },\r\n\r\n /**\r\n * Apply transform t to point p\r\n * @static\r\n * @memberOf fabric.util\r\n * @param {fabric.Point} p The point to transform\r\n * @param {Array} t The transform\r\n * @param {Boolean} [ignoreOffset] Indicates that the offset should not be applied\r\n * @return {fabric.Point} The transformed point\r\n */\r\n transformPoint: function(p, t, ignoreOffset) {\r\n if (ignoreOffset) {\r\n return new fabric.Point(\r\n t[0] * p.x + t[2] * p.y,\r\n t[1] * p.x + t[3] * p.y\r\n );\r\n }\r\n return new fabric.Point(\r\n t[0] * p.x + t[2] * p.y + t[4],\r\n t[1] * p.x + t[3] * p.y + t[5]\r\n );\r\n },\r\n\r\n /**\r\n * Returns coordinates of points's bounding rectangle (left, top, width, height)\r\n * @param {Array} points 4 points array\r\n * @param {Array} [transform] an array of 6 numbers representing a 2x3 transform matrix\r\n * @return {Object} Object with left, top, width, height properties\r\n */\r\n makeBoundingBoxFromPoints: function(points, transform) {\r\n if (transform) {\r\n for (var i = 0; i < points.length; i++) {\r\n points[i] = fabric.util.transformPoint(points[i], transform);\r\n }\r\n }\r\n var xPoints = [points[0].x, points[1].x, points[2].x, points[3].x],\r\n minX = fabric.util.array.min(xPoints),\r\n maxX = fabric.util.array.max(xPoints),\r\n width = maxX - minX,\r\n yPoints = [points[0].y, points[1].y, points[2].y, points[3].y],\r\n minY = fabric.util.array.min(yPoints),\r\n maxY = fabric.util.array.max(yPoints),\r\n height = maxY - minY;\r\n\r\n return {\r\n left: minX,\r\n top: minY,\r\n width: width,\r\n height: height\r\n };\r\n },\r\n\r\n /**\r\n * Invert transformation t\r\n * @static\r\n * @memberOf fabric.util\r\n * @param {Array} t The transform\r\n * @return {Array} The inverted transform\r\n */\r\n invertTransform: function(t) {\r\n var a = 1 / (t[0] * t[3] - t[1] * t[2]),\r\n r = [a * t[3], -a * t[1], -a * t[2], a * t[0]],\r\n o = fabric.util.transformPoint({ x: t[4], y: t[5] }, r, true);\r\n r[4] = -o.x;\r\n r[5] = -o.y;\r\n return r;\r\n },\r\n\r\n /**\r\n * A wrapper around Number#toFixed, which contrary to native method returns number, not string.\r\n * @static\r\n * @memberOf fabric.util\r\n * @param {Number|String} number number to operate on\r\n * @param {Number} fractionDigits number of fraction digits to \"leave\"\r\n * @return {Number}\r\n */\r\n toFixed: function(number, fractionDigits) {\r\n return parseFloat(Number(number).toFixed(fractionDigits));\r\n },\r\n\r\n /**\r\n * Converts from attribute value to pixel value if applicable.\r\n * Returns converted pixels or original value not converted.\r\n * @param {Number|String} value number to operate on\r\n * @param {Number} fontSize\r\n * @return {Number|String}\r\n */\r\n parseUnit: function(value, fontSize) {\r\n var unit = /\\D{0,2}$/.exec(value),\r\n number = parseFloat(value);\r\n if (!fontSize) {\r\n fontSize = fabric.Text.DEFAULT_SVG_FONT_SIZE;\r\n }\r\n switch (unit[0]) {\r\n case 'mm':\r\n return number * fabric.DPI / 25.4;\r\n\r\n case 'cm':\r\n return number * fabric.DPI / 2.54;\r\n\r\n case 'in':\r\n return number * fabric.DPI;\r\n\r\n case 'pt':\r\n return number * fabric.DPI / 72; // or * 4 / 3\r\n\r\n case 'pc':\r\n return number * fabric.DPI / 72 * 12; // or * 16\r\n\r\n case 'em':\r\n return number * fontSize;\r\n\r\n default:\r\n return number;\r\n }\r\n },\r\n\r\n /**\r\n * Function which always returns `false`.\r\n * @static\r\n * @memberOf fabric.util\r\n * @return {Boolean}\r\n */\r\n falseFunction: function() {\r\n return false;\r\n },\r\n\r\n /**\r\n * Returns klass \"Class\" object of given namespace\r\n * @memberOf fabric.util\r\n * @param {String} type Type of object (eg. 'circle')\r\n * @param {String} namespace Namespace to get klass \"Class\" object from\r\n * @return {Object} klass \"Class\"\r\n */\r\n getKlass: function(type, namespace) {\r\n // capitalize first letter only\r\n type = fabric.util.string.camelize(type.charAt(0).toUpperCase() + type.slice(1));\r\n return fabric.util.resolveNamespace(namespace)[type];\r\n },\r\n\r\n /**\r\n * Returns array of attributes for given svg that fabric parses\r\n * @memberOf fabric.util\r\n * @param {String} type Type of svg element (eg. 'circle')\r\n * @return {Array} string names of supported attributes\r\n */\r\n getSvgAttributes: function(type) {\r\n var attributes = [\r\n 'instantiated_by_use',\r\n 'style',\r\n 'id',\r\n 'class'\r\n ];\r\n switch (type) {\r\n case 'linearGradient':\r\n attributes = attributes.concat(['x1', 'y1', 'x2', 'y2', 'gradientUnits', 'gradientTransform']);\r\n break;\r\n case 'radialGradient':\r\n attributes = attributes.concat(['gradientUnits', 'gradientTransform', 'cx', 'cy', 'r', 'fx', 'fy', 'fr']);\r\n break;\r\n case 'stop':\r\n attributes = attributes.concat(['offset', 'stop-color', 'stop-opacity']);\r\n break;\r\n }\r\n return attributes;\r\n },\r\n\r\n /**\r\n * Returns object of given namespace\r\n * @memberOf fabric.util\r\n * @param {String} namespace Namespace string e.g. 'fabric.Image.filter' or 'fabric'\r\n * @return {Object} Object for given namespace (default fabric)\r\n */\r\n resolveNamespace: function(namespace) {\r\n if (!namespace) {\r\n return fabric;\r\n }\r\n\r\n var parts = namespace.split('.'),\r\n len = parts.length, i,\r\n obj = global || fabric.window;\r\n\r\n for (i = 0; i < len; ++i) {\r\n obj = obj[parts[i]];\r\n }\r\n\r\n return obj;\r\n },\r\n\r\n /**\r\n * Loads image element from given url and passes it to a callback\r\n * @memberOf fabric.util\r\n * @param {String} url URL representing an image\r\n * @param {Function} callback Callback; invoked with loaded image\r\n * @param {*} [context] Context to invoke callback in\r\n * @param {Object} [crossOrigin] crossOrigin value to set image element to\r\n */\r\n loadImage: function(url, callback, context, crossOrigin) {\r\n if (!url) {\r\n callback && callback.call(context, url);\r\n return;\r\n }\r\n\r\n var img = fabric.util.createImage();\r\n\r\n /** @ignore */\r\n var onLoadCallback = function () {\r\n callback && callback.call(context, img, false);\r\n img = img.onload = img.onerror = null;\r\n };\r\n\r\n img.onload = onLoadCallback;\r\n /** @ignore */\r\n img.onerror = function() {\r\n fabric.log('Error loading ' + img.src);\r\n callback && callback.call(context, null, true);\r\n img = img.onload = img.onerror = null;\r\n };\r\n\r\n // data-urls appear to be buggy with crossOrigin\r\n // https://github.com/kangax/fabric.js/commit/d0abb90f1cd5c5ef9d2a94d3fb21a22330da3e0a#commitcomment-4513767\r\n // see https://code.google.com/p/chromium/issues/detail?id=315152\r\n // https://bugzilla.mozilla.org/show_bug.cgi?id=935069\r\n // crossOrigin null is the same as not set.\r\n if (url.indexOf('data') !== 0 &&\r\n crossOrigin !== undefined &&\r\n crossOrigin !== null) {\r\n img.crossOrigin = crossOrigin;\r\n }\r\n\r\n // IE10 / IE11-Fix: SVG contents from data: URI\r\n // will only be available if the IMG is present\r\n // in the DOM (and visible)\r\n if (url.substring(0,14) === 'data:image/svg') {\r\n img.onload = null;\r\n fabric.util.loadImageInDom(img, onLoadCallback);\r\n }\r\n\r\n img.src = url;\r\n },\r\n\r\n /**\r\n * Attaches SVG image with data: URL to the dom\r\n * @memberOf fabric.util\r\n * @param {Object} img Image object with data:image/svg src\r\n * @param {Function} callback Callback; invoked with loaded image\r\n * @return {Object} DOM element (div containing the SVG image)\r\n */\r\n loadImageInDom: function(img, onLoadCallback) {\r\n var div = fabric.document.createElement('div');\r\n div.style.width = div.style.height = '1px';\r\n div.style.left = div.style.top = '-100%';\r\n div.style.position = 'absolute';\r\n div.appendChild(img);\r\n fabric.document.querySelector('body').appendChild(div);\r\n /**\r\n * Wrap in function to:\r\n * 1. Call existing callback\r\n * 2. Cleanup DOM\r\n */\r\n img.onload = function () {\r\n onLoadCallback();\r\n div.parentNode.removeChild(div);\r\n div = null;\r\n };\r\n },\r\n\r\n /**\r\n * Creates corresponding fabric instances from their object representations\r\n * @static\r\n * @memberOf fabric.util\r\n * @param {Array} objects Objects to enliven\r\n * @param {Function} callback Callback to invoke when all objects are created\r\n * @param {String} namespace Namespace to get klass \"Class\" object from\r\n * @param {Function} reviver Method for further parsing of object elements,\r\n * called after each fabric object created.\r\n */\r\n enlivenObjects: function(objects, callback, namespace, reviver) {\r\n objects = objects || [];\r\n\r\n var enlivenedObjects = [],\r\n numLoadedObjects = 0,\r\n numTotalObjects = objects.length;\r\n\r\n function onLoaded() {\r\n if (++numLoadedObjects === numTotalObjects) {\r\n callback && callback(enlivenedObjects.filter(function(obj) {\r\n // filter out undefined objects (objects that gave error)\r\n return obj;\r\n }));\r\n }\r\n }\r\n\r\n if (!numTotalObjects) {\r\n callback && callback(enlivenedObjects);\r\n return;\r\n }\r\n\r\n objects.forEach(function (o, index) {\r\n // if sparse array\r\n if (!o || !o.type) {\r\n onLoaded();\r\n return;\r\n }\r\n var klass = fabric.util.getKlass(o.type, namespace);\r\n klass.fromObject(o, function (obj, error) {\r\n error || (enlivenedObjects[index] = obj);\r\n reviver && reviver(o, obj, error);\r\n onLoaded();\r\n });\r\n });\r\n },\r\n\r\n /**\r\n * Creates corresponding fabric instances residing in an object, e.g. `clipPath`\r\n * @see {@link fabric.Object.ENLIVEN_PROPS}\r\n * @param {Object} object\r\n * @param {Object} [context] assign enlived props to this object (pass null to skip this)\r\n * @param {(objects:fabric.Object[]) => void} callback\r\n */\r\n enlivenObjectEnlivables: function (object, context, callback) {\r\n var enlivenProps = fabric.Object.ENLIVEN_PROPS.filter(function (key) { return !!object[key]; });\r\n fabric.util.enlivenObjects(enlivenProps.map(function (key) { return object[key]; }), function (enlivedProps) {\r\n var objects = {};\r\n enlivenProps.forEach(function (key, index) {\r\n objects[key] = enlivedProps[index];\r\n context && (context[key] = enlivedProps[index]);\r\n });\r\n callback && callback(objects);\r\n });\r\n },\r\n\r\n /**\r\n * Create and wait for loading of patterns\r\n * @static\r\n * @memberOf fabric.util\r\n * @param {Array} patterns Objects to enliven\r\n * @param {Function} callback Callback to invoke when all objects are created\r\n * called after each fabric object created.\r\n */\r\n enlivenPatterns: function(patterns, callback) {\r\n patterns = patterns || [];\r\n\r\n function onLoaded() {\r\n if (++numLoadedPatterns === numPatterns) {\r\n callback && callback(enlivenedPatterns);\r\n }\r\n }\r\n\r\n var enlivenedPatterns = [],\r\n numLoadedPatterns = 0,\r\n numPatterns = patterns.length;\r\n\r\n if (!numPatterns) {\r\n callback && callback(enlivenedPatterns);\r\n return;\r\n }\r\n\r\n patterns.forEach(function (p, index) {\r\n if (p && p.source) {\r\n new fabric.Pattern(p, function(pattern) {\r\n enlivenedPatterns[index] = pattern;\r\n onLoaded();\r\n });\r\n }\r\n else {\r\n enlivenedPatterns[index] = p;\r\n onLoaded();\r\n }\r\n });\r\n },\r\n\r\n /**\r\n * Groups SVG elements (usually those retrieved from SVG document)\r\n * @static\r\n * @memberOf fabric.util\r\n * @param {Array} elements SVG elements to group\r\n * @param {Object} [options] Options object\r\n * @param {String} path Value to set sourcePath to\r\n * @return {fabric.Object|fabric.Group}\r\n */\r\n groupSVGElements: function(elements, options, path) {\r\n var object;\r\n if (elements && elements.length === 1) {\r\n if (typeof path !== 'undefined') {\r\n elements[0].sourcePath = path;\r\n }\r\n return elements[0];\r\n }\r\n if (options) {\r\n if (options.width && options.height) {\r\n options.centerPoint = {\r\n x: options.width / 2,\r\n y: options.height / 2\r\n };\r\n }\r\n else {\r\n delete options.width;\r\n delete options.height;\r\n }\r\n }\r\n object = new fabric.Group(elements, options);\r\n if (typeof path !== 'undefined') {\r\n object.sourcePath = path;\r\n }\r\n return object;\r\n },\r\n\r\n /**\r\n * Populates an object with properties of another object\r\n * @static\r\n * @memberOf fabric.util\r\n * @param {Object} source Source object\r\n * @param {Object} destination Destination object\r\n * @return {Array} properties Properties names to include\r\n */\r\n populateWithProperties: function(source, destination, properties) {\r\n if (properties && Array.isArray(properties)) {\r\n for (var i = 0, len = properties.length; i < len; i++) {\r\n if (properties[i] in source) {\r\n destination[properties[i]] = source[properties[i]];\r\n }\r\n }\r\n }\r\n },\r\n\r\n /**\r\n * Creates canvas element\r\n * @static\r\n * @memberOf fabric.util\r\n * @return {CanvasElement} initialized canvas element\r\n */\r\n createCanvasElement: function() {\r\n return fabric.document.createElement('canvas');\r\n },\r\n\r\n /**\r\n * Creates a canvas element that is a copy of another and is also painted\r\n * @param {CanvasElement} canvas to copy size and content of\r\n * @static\r\n * @memberOf fabric.util\r\n * @return {CanvasElement} initialized canvas element\r\n */\r\n copyCanvasElement: function(canvas) {\r\n var newCanvas = fabric.util.createCanvasElement();\r\n newCanvas.width = canvas.width;\r\n newCanvas.height = canvas.height;\r\n newCanvas.getContext('2d', { willReadFrequently: true }).drawImage(canvas, 0, 0);\r\n return newCanvas;\r\n },\r\n\r\n /**\r\n * since 2.6.0 moved from canvas instance to utility.\r\n * @param {CanvasElement} canvasEl to copy size and content of\r\n * @param {String} format 'jpeg' or 'png', in some browsers 'webp' is ok too\r\n * @param {Number} quality <= 1 and > 0\r\n * @static\r\n * @memberOf fabric.util\r\n * @return {String} data url\r\n */\r\n toDataURL: function(canvasEl, format, quality) {\r\n return canvasEl.toDataURL('image/' + format, quality);\r\n },\r\n\r\n /**\r\n * Creates image element (works on client and node)\r\n * @static\r\n * @memberOf fabric.util\r\n * @return {HTMLImageElement} HTML image element\r\n */\r\n createImage: function() {\r\n return fabric.document.createElement('img');\r\n },\r\n\r\n /**\r\n * Multiply matrix A by matrix B to nest transformations\r\n * @static\r\n * @memberOf fabric.util\r\n * @param {Array} a First transformMatrix\r\n * @param {Array} b Second transformMatrix\r\n * @param {Boolean} is2x2 flag to multiply matrices as 2x2 matrices\r\n * @return {Array} The product of the two transform matrices\r\n */\r\n multiplyTransformMatrices: function(a, b, is2x2) {\r\n // Matrix multiply a * b\r\n return [\r\n a[0] * b[0] + a[2] * b[1],\r\n a[1] * b[0] + a[3] * b[1],\r\n a[0] * b[2] + a[2] * b[3],\r\n a[1] * b[2] + a[3] * b[3],\r\n is2x2 ? 0 : a[0] * b[4] + a[2] * b[5] + a[4],\r\n is2x2 ? 0 : a[1] * b[4] + a[3] * b[5] + a[5]\r\n ];\r\n },\r\n\r\n /**\r\n * Decomposes standard 2x3 matrix into transform components\r\n * @static\r\n * @memberOf fabric.util\r\n * @param {Array} a transformMatrix\r\n * @return {Object} Components of transform\r\n */\r\n qrDecompose: function(a) {\r\n var angle = atan2(a[1], a[0]),\r\n denom = pow(a[0], 2) + pow(a[1], 2),\r\n scaleX = sqrt(denom),\r\n scaleY = (a[0] * a[3] - a[2] * a[1]) / scaleX,\r\n skewX = atan2(a[0] * a[2] + a[1] * a [3], denom);\r\n return {\r\n angle: angle / PiBy180,\r\n scaleX: scaleX,\r\n scaleY: scaleY,\r\n skewX: skewX / PiBy180,\r\n skewY: 0,\r\n translateX: a[4],\r\n translateY: a[5]\r\n };\r\n },\r\n\r\n /**\r\n * Returns a transform matrix starting from an object of the same kind of\r\n * the one returned from qrDecompose, useful also if you want to calculate some\r\n * transformations from an object that is not enlived yet\r\n * @static\r\n * @memberOf fabric.util\r\n * @param {Object} options\r\n * @param {Number} [options.angle] angle in degrees\r\n * @return {Number[]} transform matrix\r\n */\r\n calcRotateMatrix: function(options) {\r\n if (!options.angle) {\r\n return fabric.iMatrix.concat();\r\n }\r\n var theta = fabric.util.degreesToRadians(options.angle),\r\n cos = fabric.util.cos(theta),\r\n sin = fabric.util.sin(theta);\r\n return [cos, sin, -sin, cos, 0, 0];\r\n },\r\n\r\n /**\r\n * Returns a transform matrix starting from an object of the same kind of\r\n * the one returned from qrDecompose, useful also if you want to calculate some\r\n * transformations from an object that is not enlived yet.\r\n * is called DimensionsTransformMatrix because those properties are the one that influence\r\n * the size of the resulting box of the object.\r\n * @static\r\n * @memberOf fabric.util\r\n * @param {Object} options\r\n * @param {Number} [options.scaleX]\r\n * @param {Number} [options.scaleY]\r\n * @param {Boolean} [options.flipX]\r\n * @param {Boolean} [options.flipY]\r\n * @param {Number} [options.skewX]\r\n * @param {Number} [options.skewY]\r\n * @return {Number[]} transform matrix\r\n */\r\n calcDimensionsMatrix: function(options) {\r\n var scaleX = typeof options.scaleX === 'undefined' ? 1 : options.scaleX,\r\n scaleY = typeof options.scaleY === 'undefined' ? 1 : options.scaleY,\r\n scaleMatrix = [\r\n options.flipX ? -scaleX : scaleX,\r\n 0,\r\n 0,\r\n options.flipY ? -scaleY : scaleY,\r\n 0,\r\n 0],\r\n multiply = fabric.util.multiplyTransformMatrices,\r\n degreesToRadians = fabric.util.degreesToRadians;\r\n if (options.skewX) {\r\n scaleMatrix = multiply(\r\n scaleMatrix,\r\n [1, 0, Math.tan(degreesToRadians(options.skewX)), 1],\r\n true);\r\n }\r\n if (options.skewY) {\r\n scaleMatrix = multiply(\r\n scaleMatrix,\r\n [1, Math.tan(degreesToRadians(options.skewY)), 0, 1],\r\n true);\r\n }\r\n return scaleMatrix;\r\n },\r\n\r\n /**\r\n * Returns a transform matrix starting from an object of the same kind of\r\n * the one returned from qrDecompose, useful also if you want to calculate some\r\n * transformations from an object that is not enlived yet\r\n * @static\r\n * @memberOf fabric.util\r\n * @param {Object} options\r\n * @param {Number} [options.angle]\r\n * @param {Number} [options.scaleX]\r\n * @param {Number} [options.scaleY]\r\n * @param {Boolean} [options.flipX]\r\n * @param {Boolean} [options.flipY]\r\n * @param {Number} [options.skewX]\r\n * @param {Number} [options.skewX]\r\n * @param {Number} [options.translateX]\r\n * @param {Number} [options.translateY]\r\n * @return {Number[]} transform matrix\r\n */\r\n composeMatrix: function(options) {\r\n var matrix = [1, 0, 0, 1, options.translateX || 0, options.translateY || 0],\r\n multiply = fabric.util.multiplyTransformMatrices;\r\n if (options.angle) {\r\n matrix = multiply(matrix, fabric.util.calcRotateMatrix(options));\r\n }\r\n if (options.scaleX !== 1 || options.scaleY !== 1 ||\r\n options.skewX || options.skewY || options.flipX || options.flipY) {\r\n matrix = multiply(matrix, fabric.util.calcDimensionsMatrix(options));\r\n }\r\n return matrix;\r\n },\r\n\r\n /**\r\n * reset an object transform state to neutral. Top and left are not accounted for\r\n * @static\r\n * @memberOf fabric.util\r\n * @param {fabric.Object} target object to transform\r\n */\r\n resetObjectTransform: function (target) {\r\n target.scaleX = 1;\r\n target.scaleY = 1;\r\n target.skewX = 0;\r\n target.skewY = 0;\r\n target.flipX = false;\r\n target.flipY = false;\r\n target.rotate(0);\r\n },\r\n\r\n /**\r\n * Extract Object transform values\r\n * @static\r\n * @memberOf fabric.util\r\n * @param {fabric.Object} target object to read from\r\n * @return {Object} Components of transform\r\n */\r\n saveObjectTransform: function (target) {\r\n return {\r\n scaleX: target.scaleX,\r\n scaleY: target.scaleY,\r\n skewX: target.skewX,\r\n skewY: target.skewY,\r\n angle: target.angle,\r\n left: target.left,\r\n flipX: target.flipX,\r\n flipY: target.flipY,\r\n top: target.top\r\n };\r\n },\r\n\r\n /**\r\n * Returns true if context has transparent pixel\r\n * at specified location (taking tolerance into account)\r\n * @param {CanvasRenderingContext2D} ctx context\r\n * @param {Number} x x coordinate\r\n * @param {Number} y y coordinate\r\n * @param {Number} tolerance Tolerance\r\n */\r\n isTransparent: function(ctx, x, y, tolerance) {\r\n\r\n // If tolerance is > 0 adjust start coords to take into account.\r\n // If moves off Canvas fix to 0\r\n if (tolerance > 0) {\r\n if (x > tolerance) {\r\n x -= tolerance;\r\n }\r\n else {\r\n x = 0;\r\n }\r\n if (y > tolerance) {\r\n y -= tolerance;\r\n }\r\n else {\r\n y = 0;\r\n }\r\n }\r\n\r\n var _isTransparent = true, i, temp,\r\n imageData = ctx.getImageData(x, y, (tolerance * 2) || 1, (tolerance * 2) || 1),\r\n l = imageData.data.length;\r\n\r\n // Split image data - for tolerance > 1, pixelDataSize = 4;\r\n for (i = 3; i < l; i += 4) {\r\n temp = imageData.data[i];\r\n _isTransparent = temp <= 0;\r\n if (_isTransparent === false) {\r\n break; // Stop if colour found\r\n }\r\n }\r\n\r\n imageData = null;\r\n\r\n return _isTransparent;\r\n },\r\n\r\n /**\r\n * Parse preserveAspectRatio attribute from element\r\n * @param {string} attribute to be parsed\r\n * @return {Object} an object containing align and meetOrSlice attribute\r\n */\r\n parsePreserveAspectRatioAttribute: function(attribute) {\r\n var meetOrSlice = 'meet', alignX = 'Mid', alignY = 'Mid',\r\n aspectRatioAttrs = attribute.split(' '), align;\r\n\r\n if (aspectRatioAttrs && aspectRatioAttrs.length) {\r\n meetOrSlice = aspectRatioAttrs.pop();\r\n if (meetOrSlice !== 'meet' && meetOrSlice !== 'slice') {\r\n align = meetOrSlice;\r\n meetOrSlice = 'meet';\r\n }\r\n else if (aspectRatioAttrs.length) {\r\n align = aspectRatioAttrs.pop();\r\n }\r\n }\r\n //divide align in alignX and alignY\r\n alignX = align !== 'none' ? align.slice(1, 4) : 'none';\r\n alignY = align !== 'none' ? align.slice(5, 8) : 'none';\r\n return {\r\n meetOrSlice: meetOrSlice,\r\n alignX: alignX,\r\n alignY: alignY\r\n };\r\n },\r\n\r\n /**\r\n * Clear char widths cache for the given font family or all the cache if no\r\n * fontFamily is specified.\r\n * Use it if you know you are loading fonts in a lazy way and you are not waiting\r\n * for custom fonts to load properly when adding text objects to the canvas.\r\n * If a text object is added when its own font is not loaded yet, you will get wrong\r\n * measurement and so wrong bounding boxes.\r\n * After the font cache is cleared, either change the textObject text content or call\r\n * initDimensions() to trigger a recalculation\r\n * @memberOf fabric.util\r\n * @param {String} [fontFamily] font family to clear\r\n */\r\n clearFabricFontCache: function(fontFamily) {\r\n fontFamily = (fontFamily || '').toLowerCase();\r\n if (!fontFamily) {\r\n fabric.charWidthsCache = { };\r\n }\r\n else if (fabric.charWidthsCache[fontFamily]) {\r\n delete fabric.charWidthsCache[fontFamily];\r\n }\r\n },\r\n\r\n /**\r\n * Given current aspect ratio, determines the max width and height that can\r\n * respect the total allowed area for the cache.\r\n * @memberOf fabric.util\r\n * @param {Number} ar aspect ratio\r\n * @param {Number} maximumArea Maximum area you want to achieve\r\n * @return {Object.x} Limited dimensions by X\r\n * @return {Object.y} Limited dimensions by Y\r\n */\r\n limitDimsByArea: function(ar, maximumArea) {\r\n var roughWidth = Math.sqrt(maximumArea * ar),\r\n perfLimitSizeY = Math.floor(maximumArea / roughWidth);\r\n return { x: Math.floor(roughWidth), y: perfLimitSizeY };\r\n },\r\n\r\n capValue: function(min, value, max) {\r\n return Math.max(min, Math.min(value, max));\r\n },\r\n\r\n /**\r\n * Finds the scale for the object source to fit inside the object destination,\r\n * keeping aspect ratio intact.\r\n * respect the total allowed area for the cache.\r\n * @memberOf fabric.util\r\n * @param {Object | fabric.Object} source\r\n * @param {Number} source.height natural unscaled height of the object\r\n * @param {Number} source.width natural unscaled width of the object\r\n * @param {Object | fabric.Object} destination\r\n * @param {Number} destination.height natural unscaled height of the object\r\n * @param {Number} destination.width natural unscaled width of the object\r\n * @return {Number} scale factor to apply to source to fit into destination\r\n */\r\n findScaleToFit: function(source, destination) {\r\n return Math.min(destination.width / source.width, destination.height / source.height);\r\n },\r\n\r\n /**\r\n * Finds the scale for the object source to cover entirely the object destination,\r\n * keeping aspect ratio intact.\r\n * respect the total allowed area for the cache.\r\n * @memberOf fabric.util\r\n * @param {Object | fabric.Object} source\r\n * @param {Number} source.height natural unscaled height of the object\r\n * @param {Number} source.width natural unscaled width of the object\r\n * @param {Object | fabric.Object} destination\r\n * @param {Number} destination.height natural unscaled height of the object\r\n * @param {Number} destination.width natural unscaled width of the object\r\n * @return {Number} scale factor to apply to source to cover destination\r\n */\r\n findScaleToCover: function(source, destination) {\r\n return Math.max(destination.width / source.width, destination.height / source.height);\r\n },\r\n\r\n /**\r\n * given an array of 6 number returns something like `\"matrix(...numbers)\"`\r\n * @memberOf fabric.util\r\n * @param {Array} transform an array with 6 numbers\r\n * @return {String} transform matrix for svg\r\n * @return {Object.y} Limited dimensions by Y\r\n */\r\n matrixToSVG: function(transform) {\r\n return 'matrix(' + transform.map(function(value) {\r\n return fabric.util.toFixed(value, fabric.Object.NUM_FRACTION_DIGITS);\r\n }).join(' ') + ')';\r\n },\r\n\r\n /**\r\n * given an object and a transform, apply the inverse transform to the object,\r\n * this is equivalent to remove from that object that transformation, so that\r\n * added in a space with the removed transform, the object will be the same as before.\r\n * Removing from an object a transform that scale by 2 is like scaling it by 1/2.\r\n * Removing from an object a transfrom that rotate by 30deg is like rotating by 30deg\r\n * in the opposite direction.\r\n * This util is used to add objects inside transformed groups or nested groups.\r\n * @memberOf fabric.util\r\n * @param {fabric.Object} object the object you want to transform\r\n * @param {Array} transform the destination transform\r\n */\r\n removeTransformFromObject: function(object, transform) {\r\n var inverted = fabric.util.invertTransform(transform),\r\n finalTransform = fabric.util.multiplyTransformMatrices(inverted, object.calcOwnMatrix());\r\n fabric.util.applyTransformToObject(object, finalTransform);\r\n },\r\n\r\n /**\r\n * given an object and a transform, apply the transform to the object.\r\n * this is equivalent to change the space where the object is drawn.\r\n * Adding to an object a transform that scale by 2 is like scaling it by 2.\r\n * This is used when removing an object from an active selection for example.\r\n * @memberOf fabric.util\r\n * @param {fabric.Object} object the object you want to transform\r\n * @param {Array} transform the destination transform\r\n */\r\n addTransformToObject: function(object, transform) {\r\n fabric.util.applyTransformToObject(\r\n object,\r\n fabric.util.multiplyTransformMatrices(transform, object.calcOwnMatrix())\r\n );\r\n },\r\n\r\n /**\r\n * discard an object transform state and apply the one from the matrix.\r\n * @memberOf fabric.util\r\n * @param {fabric.Object} object the object you want to transform\r\n * @param {Array} transform the destination transform\r\n */\r\n applyTransformToObject: function(object, transform) {\r\n var options = fabric.util.qrDecompose(transform),\r\n center = new fabric.Point(options.translateX, options.translateY);\r\n object.flipX = false;\r\n object.flipY = false;\r\n object.set('scaleX', options.scaleX);\r\n object.set('scaleY', options.scaleY);\r\n object.skewX = options.skewX;\r\n object.skewY = options.skewY;\r\n object.angle = options.angle;\r\n object.setPositionByOrigin(center, 'center', 'center');\r\n },\r\n\r\n /**\r\n * given a width and height, return the size of the bounding box\r\n * that can contains the box with width/height with applied transform\r\n * described in options.\r\n * Use to calculate the boxes around objects for controls.\r\n * @memberOf fabric.util\r\n * @param {Number} width\r\n * @param {Number} height\r\n * @param {Object} options\r\n * @param {Number} options.scaleX\r\n * @param {Number} options.scaleY\r\n * @param {Number} options.skewX\r\n * @param {Number} options.skewY\r\n * @return {Object.x} width of containing\r\n * @return {Object.y} height of containing\r\n */\r\n sizeAfterTransform: function(width, height, options) {\r\n var dimX = width / 2, dimY = height / 2,\r\n points = [\r\n {\r\n x: -dimX,\r\n y: -dimY\r\n },\r\n {\r\n x: dimX,\r\n y: -dimY\r\n },\r\n {\r\n x: -dimX,\r\n y: dimY\r\n },\r\n {\r\n x: dimX,\r\n y: dimY\r\n }],\r\n transformMatrix = fabric.util.calcDimensionsMatrix(options),\r\n bbox = fabric.util.makeBoundingBoxFromPoints(points, transformMatrix);\r\n return {\r\n x: bbox.width,\r\n y: bbox.height,\r\n };\r\n },\r\n\r\n /**\r\n * Merges 2 clip paths into one visually equal clip path\r\n *\r\n * **IMPORTANT**:\\\r\n * Does **NOT** clone the arguments, clone them proir if necessary.\r\n *\r\n * Creates a wrapper (group) that contains one clip path and is clipped by the other so content is kept where both overlap.\r\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.\r\n *\r\n * In order to handle the `inverted` property we follow logic described in the following cases:\\\r\n * **(1)** both clip paths are inverted - the clip paths pass the inverted prop to the wrapper and loose it themselves.\\\r\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.\\\r\n * **(3)** both clip paths are not inverted - wrapper and clip paths remain unchanged.\r\n *\r\n * @memberOf fabric.util\r\n * @param {fabric.Object} c1\r\n * @param {fabric.Object} c2\r\n * @returns {fabric.Object} merged clip path\r\n */\r\n mergeClipPaths: function (c1, c2) {\r\n var a = c1, b = c2;\r\n if (a.inverted && !b.inverted) {\r\n // case (2)\r\n a = c2;\r\n b = c1;\r\n }\r\n // `b` becomes `a`'s clip path so we transform `b` to `a` coordinate plane\r\n fabric.util.applyTransformToObject(\r\n b,\r\n fabric.util.multiplyTransformMatrices(\r\n fabric.util.invertTransform(a.calcTransformMatrix()),\r\n b.calcTransformMatrix()\r\n )\r\n );\r\n // assign the `inverted` prop to the wrapping group\r\n var inverted = a.inverted && b.inverted;\r\n if (inverted) {\r\n // case (1)\r\n a.inverted = b.inverted = false;\r\n }\r\n return new fabric.Group([a], { clipPath: b, inverted: inverted });\r\n },\r\n\r\n /**\r\n * @memberOf fabric.util\r\n * @param {Object} prevStyle first style to compare\r\n * @param {Object} thisStyle second style to compare\r\n * @param {boolean} forTextSpans whether to check overline, underline, and line-through properties\r\n * @return {boolean} true if the style changed\r\n */\r\n hasStyleChanged: function(prevStyle, thisStyle, forTextSpans) {\r\n forTextSpans = forTextSpans || false;\r\n return (prevStyle.fill !== thisStyle.fill ||\r\n prevStyle.stroke !== thisStyle.stroke ||\r\n prevStyle.strokeWidth !== thisStyle.strokeWidth ||\r\n prevStyle.fontSize !== thisStyle.fontSize ||\r\n prevStyle.fontFamily !== thisStyle.fontFamily ||\r\n prevStyle.fontWeight !== thisStyle.fontWeight ||\r\n prevStyle.fontStyle !== thisStyle.fontStyle ||\r\n prevStyle.textBackgroundColor !== thisStyle.textBackgroundColor ||\r\n prevStyle.deltaY !== thisStyle.deltaY) ||\r\n (forTextSpans &&\r\n (prevStyle.overline !== thisStyle.overline ||\r\n prevStyle.underline !== thisStyle.underline ||\r\n prevStyle.linethrough !== thisStyle.linethrough));\r\n },\r\n\r\n /**\r\n * Returns the array form of a text object's inline styles property with styles grouped in ranges\r\n * rather than per character. This format is less verbose, and is better suited for storage\r\n * so it is used in serialization (not during runtime).\r\n * @memberOf fabric.util\r\n * @param {object} styles per character styles for a text object\r\n * @param {String} text the text string that the styles are applied to\r\n * @return {{start: number, end: number, style: object}[]}\r\n */\r\n stylesToArray: function(styles, text) {\r\n // clone style structure to prevent mutation\r\n var styles = fabric.util.object.clone(styles, true),\r\n textLines = text.split('\\n'),\r\n charIndex = -1, prevStyle = {}, stylesArray = [];\r\n //loop through each textLine\r\n for (var i = 0; i < textLines.length; i++) {\r\n if (!styles[i]) {\r\n //no styles exist for this line, so add the line's length to the charIndex total\r\n charIndex += textLines[i].length;\r\n continue;\r\n }\r\n //loop through each character of the current line\r\n for (var c = 0; c < textLines[i].length; c++) {\r\n charIndex++;\r\n var thisStyle = styles[i][c];\r\n //check if style exists for this character\r\n if (thisStyle && Object.keys(thisStyle).length > 0) {\r\n var styleChanged = fabric.util.hasStyleChanged(prevStyle, thisStyle, true);\r\n if (styleChanged) {\r\n stylesArray.push({\r\n start: charIndex,\r\n end: charIndex + 1,\r\n style: thisStyle\r\n });\r\n }\r\n else {\r\n //if style is the same as previous character, increase end index\r\n stylesArray[stylesArray.length - 1].end++;\r\n }\r\n }\r\n prevStyle = thisStyle || {};\r\n }\r\n }\r\n return stylesArray;\r\n },\r\n\r\n /**\r\n * Returns the object form of the styles property with styles that are assigned per\r\n * character rather than grouped by range. This format is more verbose, and is\r\n * only used during runtime (not for serialization/storage)\r\n * @memberOf fabric.util\r\n * @param {Array} styles the serialized form of a text object's styles\r\n * @param {String} text the text string that the styles are applied to\r\n * @return {Object}\r\n */\r\n stylesFromArray: function(styles, text) {\r\n if (!Array.isArray(styles)) {\r\n return styles;\r\n }\r\n var textLines = text.split('\\n'),\r\n charIndex = -1, styleIndex = 0, stylesObject = {};\r\n //loop through each textLine\r\n for (var i = 0; i < textLines.length; i++) {\r\n //loop through each character of the current line\r\n for (var c = 0; c < textLines[i].length; c++) {\r\n charIndex++;\r\n //check if there's a style collection that includes the current character\r\n if (styles[styleIndex]\r\n && styles[styleIndex].start <= charIndex\r\n && charIndex < styles[styleIndex].end) {\r\n //create object for line index if it doesn't exist\r\n stylesObject[i] = stylesObject[i] || {};\r\n //assign a style at this character's index\r\n stylesObject[i][c] = Object.assign({}, styles[styleIndex].style);\r\n //if character is at the end of the current style collection, move to the next\r\n if (charIndex === styles[styleIndex].end - 1) {\r\n styleIndex++;\r\n }\r\n }\r\n }\r\n }\r\n return stylesObject;\r\n }\r\n };\r\n})(typeof exports !== 'undefined' ? exports : this);\r\n\r\n\r\n(function() {\r\n var _join = Array.prototype.join,\r\n commandLengths = {\r\n m: 2,\r\n l: 2,\r\n h: 1,\r\n v: 1,\r\n c: 6,\r\n s: 4,\r\n q: 4,\r\n t: 2,\r\n a: 7\r\n },\r\n repeatedCommands = {\r\n m: 'l',\r\n M: 'L'\r\n };\r\n function segmentToBezier(th2, th3, cosTh, sinTh, rx, ry, cx1, cy1, mT, fromX, fromY) {\r\n var costh2 = fabric.util.cos(th2),\r\n sinth2 = fabric.util.sin(th2),\r\n costh3 = fabric.util.cos(th3),\r\n sinth3 = fabric.util.sin(th3),\r\n toX = cosTh * rx * costh3 - sinTh * ry * sinth3 + cx1,\r\n toY = sinTh * rx * costh3 + cosTh * ry * sinth3 + cy1,\r\n cp1X = fromX + mT * ( -cosTh * rx * sinth2 - sinTh * ry * costh2),\r\n cp1Y = fromY + mT * ( -sinTh * rx * sinth2 + cosTh * ry * costh2),\r\n cp2X = toX + mT * ( cosTh * rx * sinth3 + sinTh * ry * costh3),\r\n cp2Y = toY + mT * ( sinTh * rx * sinth3 - cosTh * ry * costh3);\r\n\r\n return ['C',\r\n cp1X, cp1Y,\r\n cp2X, cp2Y,\r\n toX, toY\r\n ];\r\n }\r\n\r\n /* Adapted from http://dxr.mozilla.org/mozilla-central/source/content/svg/content/src/nsSVGPathDataParser.cpp\r\n * by Andrea Bogazzi code is under MPL. if you don't have a copy of the license you can take it here\r\n * http://mozilla.org/MPL/2.0/\r\n */\r\n function arcToSegments(toX, toY, rx, ry, large, sweep, rotateX) {\r\n var PI = Math.PI, th = rotateX * PI / 180,\r\n sinTh = fabric.util.sin(th),\r\n cosTh = fabric.util.cos(th),\r\n fromX = 0, fromY = 0;\r\n\r\n rx = Math.abs(rx);\r\n ry = Math.abs(ry);\r\n\r\n var px = -cosTh * toX * 0.5 - sinTh * toY * 0.5,\r\n py = -cosTh * toY * 0.5 + sinTh * toX * 0.5,\r\n rx2 = rx * rx, ry2 = ry * ry, py2 = py * py, px2 = px * px,\r\n pl = rx2 * ry2 - rx2 * py2 - ry2 * px2,\r\n root = 0;\r\n\r\n if (pl < 0) {\r\n var s = Math.sqrt(1 - pl / (rx2 * ry2));\r\n rx *= s;\r\n ry *= s;\r\n }\r\n else {\r\n root = (large === sweep ? -1.0 : 1.0) *\r\n Math.sqrt( pl / (rx2 * py2 + ry2 * px2));\r\n }\r\n\r\n var cx = root * rx * py / ry,\r\n cy = -root * ry * px / rx,\r\n cx1 = cosTh * cx - sinTh * cy + toX * 0.5,\r\n cy1 = sinTh * cx + cosTh * cy + toY * 0.5,\r\n mTheta = calcVectorAngle(1, 0, (px - cx) / rx, (py - cy) / ry),\r\n dtheta = calcVectorAngle((px - cx) / rx, (py - cy) / ry, (-px - cx) / rx, (-py - cy) / ry);\r\n\r\n if (sweep === 0 && dtheta > 0) {\r\n dtheta -= 2 * PI;\r\n }\r\n else if (sweep === 1 && dtheta < 0) {\r\n dtheta += 2 * PI;\r\n }\r\n\r\n // Convert into cubic bezier segments <= 90deg\r\n var segments = Math.ceil(Math.abs(dtheta / PI * 2)),\r\n result = [], mDelta = dtheta / segments,\r\n mT = 8 / 3 * Math.sin(mDelta / 4) * Math.sin(mDelta / 4) / Math.sin(mDelta / 2),\r\n th3 = mTheta + mDelta;\r\n\r\n for (var i = 0; i < segments; i++) {\r\n result[i] = segmentToBezier(mTheta, th3, cosTh, sinTh, rx, ry, cx1, cy1, mT, fromX, fromY);\r\n fromX = result[i][5];\r\n fromY = result[i][6];\r\n mTheta = th3;\r\n th3 += mDelta;\r\n }\r\n return result;\r\n }\r\n\r\n /*\r\n * Private\r\n */\r\n function calcVectorAngle(ux, uy, vx, vy) {\r\n var ta = Math.atan2(uy, ux),\r\n tb = Math.atan2(vy, vx);\r\n if (tb >= ta) {\r\n return tb - ta;\r\n }\r\n else {\r\n return 2 * Math.PI - (ta - tb);\r\n }\r\n }\r\n\r\n /**\r\n * Calculate bounding box of a beziercurve\r\n * @param {Number} x0 starting point\r\n * @param {Number} y0\r\n * @param {Number} x1 first control point\r\n * @param {Number} y1\r\n * @param {Number} x2 secondo control point\r\n * @param {Number} y2\r\n * @param {Number} x3 end of bezier\r\n * @param {Number} y3\r\n */\r\n // taken from http://jsbin.com/ivomiq/56/edit no credits available for that.\r\n // TODO: can we normalize this with the starting points set at 0 and then translated the bbox?\r\n function getBoundsOfCurve(x0, y0, x1, y1, x2, y2, x3, y3) {\r\n var argsString;\r\n if (fabric.cachesBoundsOfCurve) {\r\n argsString = _join.call(arguments);\r\n if (fabric.boundsOfCurveCache[argsString]) {\r\n return fabric.boundsOfCurveCache[argsString];\r\n }\r\n }\r\n\r\n var sqrt = Math.sqrt,\r\n min = Math.min, max = Math.max,\r\n abs = Math.abs, tvalues = [],\r\n bounds = [[], []],\r\n a, b, c, t, t1, t2, b2ac, sqrtb2ac;\r\n\r\n b = 6 * x0 - 12 * x1 + 6 * x2;\r\n a = -3 * x0 + 9 * x1 - 9 * x2 + 3 * x3;\r\n c = 3 * x1 - 3 * x0;\r\n\r\n for (var i = 0; i < 2; ++i) {\r\n if (i > 0) {\r\n b = 6 * y0 - 12 * y1 + 6 * y2;\r\n a = -3 * y0 + 9 * y1 - 9 * y2 + 3 * y3;\r\n c = 3 * y1 - 3 * y0;\r\n }\r\n\r\n if (abs(a) < 1e-12) {\r\n if (abs(b) < 1e-12) {\r\n continue;\r\n }\r\n t = -c / b;\r\n if (0 < t && t < 1) {\r\n tvalues.push(t);\r\n }\r\n continue;\r\n }\r\n b2ac = b * b - 4 * c * a;\r\n if (b2ac < 0) {\r\n continue;\r\n }\r\n sqrtb2ac = sqrt(b2ac);\r\n t1 = (-b + sqrtb2ac) / (2 * a);\r\n if (0 < t1 && t1 < 1) {\r\n tvalues.push(t1);\r\n }\r\n t2 = (-b - sqrtb2ac) / (2 * a);\r\n if (0 < t2 && t2 < 1) {\r\n tvalues.push(t2);\r\n }\r\n }\r\n\r\n var x, y, j = tvalues.length, jlen = j, mt;\r\n while (j--) {\r\n t = tvalues[j];\r\n mt = 1 - t;\r\n x = (mt * mt * mt * x0) + (3 * mt * mt * t * x1) + (3 * mt * t * t * x2) + (t * t * t * x3);\r\n bounds[0][j] = x;\r\n\r\n y = (mt * mt * mt * y0) + (3 * mt * mt * t * y1) + (3 * mt * t * t * y2) + (t * t * t * y3);\r\n bounds[1][j] = y;\r\n }\r\n\r\n bounds[0][jlen] = x0;\r\n bounds[1][jlen] = y0;\r\n bounds[0][jlen + 1] = x3;\r\n bounds[1][jlen + 1] = y3;\r\n var result = [\r\n {\r\n x: min.apply(null, bounds[0]),\r\n y: min.apply(null, bounds[1])\r\n },\r\n {\r\n x: max.apply(null, bounds[0]),\r\n y: max.apply(null, bounds[1])\r\n }\r\n ];\r\n if (fabric.cachesBoundsOfCurve) {\r\n fabric.boundsOfCurveCache[argsString] = result;\r\n }\r\n return result;\r\n }\r\n\r\n /**\r\n * Converts arc to a bunch of bezier curves\r\n * @param {Number} fx starting point x\r\n * @param {Number} fy starting point y\r\n * @param {Array} coords Arc command\r\n */\r\n function fromArcToBeziers(fx, fy, coords) {\r\n var rx = coords[1],\r\n ry = coords[2],\r\n rot = coords[3],\r\n large = coords[4],\r\n sweep = coords[5],\r\n tx = coords[6],\r\n ty = coords[7],\r\n segsNorm = arcToSegments(tx - fx, ty - fy, rx, ry, large, sweep, rot);\r\n\r\n for (var i = 0, len = segsNorm.length; i < len; i++) {\r\n segsNorm[i][1] += fx;\r\n segsNorm[i][2] += fy;\r\n segsNorm[i][3] += fx;\r\n segsNorm[i][4] += fy;\r\n segsNorm[i][5] += fx;\r\n segsNorm[i][6] += fy;\r\n }\r\n return segsNorm;\r\n };\r\n\r\n /**\r\n * This function take a parsed SVG path and make it simpler for fabricJS logic.\r\n * simplification consist of: only UPPERCASE absolute commands ( relative converted to absolute )\r\n * S converted in C, T converted in Q, A converted in C.\r\n * @param {Array} path the array of commands of a parsed svg path for fabric.Path\r\n * @return {Array} the simplified array of commands of a parsed svg path for fabric.Path\r\n */\r\n function makePathSimpler(path) {\r\n // x and y represent the last point of the path. the previous command point.\r\n // we add them to each relative command to make it an absolute comment.\r\n // we also swap the v V h H with L, because are easier to transform.\r\n var x = 0, y = 0, len = path.length,\r\n // x1 and y1 represent the last point of the subpath. the subpath is started with\r\n // m or M command. When a z or Z command is drawn, x and y need to be resetted to\r\n // the last x1 and y1.\r\n x1 = 0, y1 = 0, current, i, converted,\r\n // previous will host the letter of the previous command, to handle S and T.\r\n // controlX and controlY will host the previous reflected control point\r\n destinationPath = [], previous, controlX, controlY;\r\n for (i = 0; i < len; ++i) {\r\n converted = false;\r\n current = path[i].slice(0);\r\n switch (current[0]) { // first letter\r\n case 'l': // lineto, relative\r\n current[0] = 'L';\r\n current[1] += x;\r\n current[2] += y;\r\n // falls through\r\n case 'L':\r\n x = current[1];\r\n y = current[2];\r\n break;\r\n case 'h': // horizontal lineto, relative\r\n current[1] += x;\r\n // falls through\r\n case 'H':\r\n current[0] = 'L';\r\n current[2] = y;\r\n x = current[1];\r\n break;\r\n case 'v': // vertical lineto, relative\r\n current[1] += y;\r\n // falls through\r\n case 'V':\r\n current[0] = 'L';\r\n y = current[1];\r\n current[1] = x;\r\n current[2] = y;\r\n break;\r\n case 'm': // moveTo, relative\r\n current[0] = 'M';\r\n current[1] += x;\r\n current[2] += y;\r\n // falls through\r\n case 'M':\r\n x = current[1];\r\n y = current[2];\r\n x1 = current[1];\r\n y1 = current[2];\r\n break;\r\n case 'c': // bezierCurveTo, relative\r\n current[0] = 'C';\r\n current[1] += x;\r\n current[2] += y;\r\n current[3] += x;\r\n current[4] += y;\r\n current[5] += x;\r\n current[6] += y;\r\n // falls through\r\n case 'C':\r\n controlX = current[3];\r\n controlY = current[4];\r\n x = current[5];\r\n y = current[6];\r\n break;\r\n case 's': // shorthand cubic bezierCurveTo, relative\r\n current[0] = 'S';\r\n current[1] += x;\r\n current[2] += y;\r\n current[3] += x;\r\n current[4] += y;\r\n // falls through\r\n case 'S':\r\n // would be sScC but since we are swapping sSc for C, we check just that.\r\n if (previous === 'C') {\r\n // calculate reflection of previous control points\r\n controlX = 2 * x - controlX;\r\n controlY = 2 * y - controlY;\r\n }\r\n else {\r\n // If there is no previous command or if the previous command was not a C, c, S, or s,\r\n // the control point is coincident with the current point\r\n controlX = x;\r\n controlY = y;\r\n }\r\n x = current[3];\r\n y = current[4];\r\n current[0] = 'C';\r\n current[5] = current[3];\r\n current[6] = current[4];\r\n current[3] = current[1];\r\n current[4] = current[2];\r\n current[1] = controlX;\r\n current[2] = controlY;\r\n // current[3] and current[4] are NOW the second control point.\r\n // we keep it for the next reflection.\r\n controlX = current[3];\r\n controlY = current[4];\r\n break;\r\n case 'q': // quadraticCurveTo, relative\r\n current[0] = 'Q';\r\n current[1] += x;\r\n current[2] += y;\r\n current[3] += x;\r\n current[4] += y;\r\n // falls through\r\n case 'Q':\r\n controlX = current[1];\r\n controlY = current[2];\r\n x = current[3];\r\n y = current[4];\r\n break;\r\n case 't': // shorthand quadraticCurveTo, relative\r\n current[0] = 'T';\r\n current[1] += x;\r\n current[2] += y;\r\n // falls through\r\n case 'T':\r\n if (previous === 'Q') {\r\n // calculate reflection of previous control point\r\n controlX = 2 * x - controlX;\r\n controlY = 2 * y - controlY;\r\n }\r\n else {\r\n // If there is no previous command or if the previous command was not a Q, q, T or t,\r\n // assume the control point is coincident with the current point\r\n controlX = x;\r\n controlY = y;\r\n }\r\n current[0] = 'Q';\r\n x = current[1];\r\n y = current[2];\r\n current[1] = controlX;\r\n current[2] = controlY;\r\n current[3] = x;\r\n current[4] = y;\r\n break;\r\n case 'a':\r\n current[0] = 'A';\r\n current[6] += x;\r\n current[7] += y;\r\n // falls through\r\n case 'A':\r\n converted = true;\r\n destinationPath = destinationPath.concat(fromArcToBeziers(x, y, current));\r\n x = current[6];\r\n y = current[7];\r\n break;\r\n case 'z':\r\n case 'Z':\r\n x = x1;\r\n y = y1;\r\n break;\r\n default:\r\n }\r\n if (!converted) {\r\n destinationPath.push(current);\r\n }\r\n previous = current[0];\r\n }\r\n return destinationPath;\r\n };\r\n\r\n /**\r\n * Calc length from point x1,y1 to x2,y2\r\n * @param {Number} x1 starting point x\r\n * @param {Number} y1 starting point y\r\n * @param {Number} x2 starting point x\r\n * @param {Number} y2 starting point y\r\n * @return {Number} length of segment\r\n */\r\n function calcLineLength(x1, y1, x2, y2) {\r\n return Math.sqrt((x2 - x1) * (x2 - x1) + (y2 - y1) * (y2 - y1));\r\n }\r\n\r\n // functions for the Cubic beizer\r\n // taken from: https://github.com/konvajs/konva/blob/7.0.5/src/shapes/Path.ts#L350\r\n function CB1(t) {\r\n return t * t * t;\r\n }\r\n function CB2(t) {\r\n return 3 * t * t * (1 - t);\r\n }\r\n function CB3(t) {\r\n return 3 * t * (1 - t) * (1 - t);\r\n }\r\n function CB4(t) {\r\n return (1 - t) * (1 - t) * (1 - t);\r\n }\r\n\r\n function getPointOnCubicBezierIterator(p1x, p1y, p2x, p2y, p3x, p3y, p4x, p4y) {\r\n return function(pct) {\r\n var c1 = CB1(pct), c2 = CB2(pct), c3 = CB3(pct), c4 = CB4(pct);\r\n return {\r\n x: p4x * c1 + p3x * c2 + p2x * c3 + p1x * c4,\r\n y: p4y * c1 + p3y * c2 + p2y * c3 + p1y * c4\r\n };\r\n };\r\n }\r\n\r\n function getTangentCubicIterator(p1x, p1y, p2x, p2y, p3x, p3y, p4x, p4y) {\r\n return function (pct) {\r\n var invT = 1 - pct,\r\n tangentX = (3 * invT * invT * (p2x - p1x)) + (6 * invT * pct * (p3x - p2x)) +\r\n (3 * pct * pct * (p4x - p3x)),\r\n tangentY = (3 * invT * invT * (p2y - p1y)) + (6 * invT * pct * (p3y - p2y)) +\r\n (3 * pct * pct * (p4y - p3y));\r\n return Math.atan2(tangentY, tangentX);\r\n };\r\n }\r\n\r\n function QB1(t) {\r\n return t * t;\r\n }\r\n\r\n function QB2(t) {\r\n return 2 * t * (1 - t);\r\n }\r\n\r\n function QB3(t) {\r\n return (1 - t) * (1 - t);\r\n }\r\n\r\n function getPointOnQuadraticBezierIterator(p1x, p1y, p2x, p2y, p3x, p3y) {\r\n return function(pct) {\r\n var c1 = QB1(pct), c2 = QB2(pct), c3 = QB3(pct);\r\n return {\r\n x: p3x * c1 + p2x * c2 + p1x * c3,\r\n y: p3y * c1 + p2y * c2 + p1y * c3\r\n };\r\n };\r\n }\r\n\r\n function getTangentQuadraticIterator(p1x, p1y, p2x, p2y, p3x, p3y) {\r\n return function (pct) {\r\n var invT = 1 - pct,\r\n tangentX = (2 * invT * (p2x - p1x)) + (2 * pct * (p3x - p2x)),\r\n tangentY = (2 * invT * (p2y - p1y)) + (2 * pct * (p3y - p2y));\r\n return Math.atan2(tangentY, tangentX);\r\n };\r\n }\r\n\r\n\r\n // this will run over a path segment ( a cubic or quadratic segment) and approximate it\r\n // with 100 segemnts. This will good enough to calculate the length of the curve\r\n function pathIterator(iterator, x1, y1) {\r\n var tempP = { x: x1, y: y1 }, p, tmpLen = 0, perc;\r\n for (perc = 1; perc <= 100; perc += 1) {\r\n p = iterator(perc / 100);\r\n tmpLen += calcLineLength(tempP.x, tempP.y, p.x, p.y);\r\n tempP = p;\r\n }\r\n return tmpLen;\r\n }\r\n\r\n /**\r\n * Given a pathInfo, and a distance in pixels, find the percentage from 0 to 1\r\n * that correspond to that pixels run over the path.\r\n * The percentage will be then used to find the correct point on the canvas for the path.\r\n * @param {Array} segInfo fabricJS collection of information on a parsed path\r\n * @param {Number} distance from starting point, in pixels.\r\n * @return {Object} info object with x and y ( the point on canvas ) and angle, the tangent on that point;\r\n */\r\n function findPercentageForDistance(segInfo, distance) {\r\n var perc = 0, tmpLen = 0, iterator = segInfo.iterator, tempP = { x: segInfo.x, y: segInfo.y },\r\n p, nextLen, nextStep = 0.01, angleFinder = segInfo.angleFinder, lastPerc;\r\n // nextStep > 0.0001 covers 0.00015625 that 1/64th of 1/100\r\n // the path\r\n while (tmpLen < distance && nextStep > 0.0001) {\r\n p = iterator(perc);\r\n lastPerc = perc;\r\n nextLen = calcLineLength(tempP.x, tempP.y, p.x, p.y);\r\n // compare tmpLen each cycle with distance, decide next perc to test.\r\n if ((nextLen + tmpLen) > distance) {\r\n // we discard this step and we make smaller steps.\r\n perc -= nextStep;\r\n nextStep /= 2;\r\n }\r\n else {\r\n tempP = p;\r\n perc += nextStep;\r\n tmpLen += nextLen;\r\n }\r\n }\r\n p.angle = angleFinder(lastPerc);\r\n return p;\r\n }\r\n\r\n /**\r\n * Run over a parsed and simplifed path and extrac some informations.\r\n * informations are length of each command and starting point\r\n * @param {Array} path fabricJS parsed path commands\r\n * @return {Array} path commands informations\r\n */\r\n function getPathSegmentsInfo(path) {\r\n var totalLength = 0, len = path.length, current,\r\n //x2 and y2 are the coords of segment start\r\n //x1 and y1 are the coords of the current point\r\n x1 = 0, y1 = 0, x2 = 0, y2 = 0, info = [], iterator, tempInfo, angleFinder;\r\n for (var i = 0; i < len; i++) {\r\n current = path[i];\r\n tempInfo = {\r\n x: x1,\r\n y: y1,\r\n command: current[0],\r\n };\r\n switch (current[0]) { //first letter\r\n case 'M':\r\n tempInfo.length = 0;\r\n x2 = x1 = current[1];\r\n y2 = y1 = current[2];\r\n break;\r\n case 'L':\r\n tempInfo.length = calcLineLength(x1, y1, current[1], current[2]);\r\n x1 = current[1];\r\n y1 = current[2];\r\n break;\r\n case 'C':\r\n iterator = getPointOnCubicBezierIterator(\r\n x1,\r\n y1,\r\n current[1],\r\n current[2],\r\n current[3],\r\n current[4],\r\n current[5],\r\n current[6]\r\n );\r\n angleFinder = getTangentCubicIterator(\r\n x1,\r\n y1,\r\n current[1],\r\n current[2],\r\n current[3],\r\n current[4],\r\n current[5],\r\n current[6]\r\n );\r\n tempInfo.iterator = iterator;\r\n tempInfo.angleFinder = angleFinder;\r\n tempInfo.length = pathIterator(iterator, x1, y1);\r\n x1 = current[5];\r\n y1 = current[6];\r\n break;\r\n case 'Q':\r\n iterator = getPointOnQuadraticBezierIterator(\r\n x1,\r\n y1,\r\n current[1],\r\n current[2],\r\n current[3],\r\n current[4]\r\n );\r\n angleFinder = getTangentQuadraticIterator(\r\n x1,\r\n y1,\r\n current[1],\r\n current[2],\r\n current[3],\r\n current[4]\r\n );\r\n tempInfo.iterator = iterator;\r\n tempInfo.angleFinder = angleFinder;\r\n tempInfo.length = pathIterator(iterator, x1, y1);\r\n x1 = current[3];\r\n y1 = current[4];\r\n break;\r\n case 'Z':\r\n case 'z':\r\n // we add those in order to ease calculations later\r\n tempInfo.destX = x2;\r\n tempInfo.destY = y2;\r\n tempInfo.length = calcLineLength(x1, y1, x2, y2);\r\n x1 = x2;\r\n y1 = y2;\r\n break;\r\n }\r\n totalLength += tempInfo.length;\r\n info.push(tempInfo);\r\n }\r\n info.push({ length: totalLength, x: x1, y: y1 });\r\n return info;\r\n }\r\n\r\n function getPointOnPath(path, distance, infos) {\r\n if (!infos) {\r\n infos = getPathSegmentsInfo(path);\r\n }\r\n var i = 0;\r\n while ((distance - infos[i].length > 0) && i < (infos.length - 2)) {\r\n distance -= infos[i].length;\r\n i++;\r\n }\r\n // var distance = infos[infos.length - 1] * perc;\r\n var segInfo = infos[i], segPercent = distance / segInfo.length,\r\n command = segInfo.command, segment = path[i], info;\r\n\r\n switch (command) {\r\n case 'M':\r\n return { x: segInfo.x, y: segInfo.y, angle: 0 };\r\n case 'Z':\r\n case 'z':\r\n info = new fabric.Point(segInfo.x, segInfo.y).lerp(\r\n new fabric.Point(segInfo.destX, segInfo.destY),\r\n segPercent\r\n );\r\n info.angle = Math.atan2(segInfo.destY - segInfo.y, segInfo.destX - segInfo.x);\r\n return info;\r\n case 'L':\r\n info = new fabric.Point(segInfo.x, segInfo.y).lerp(\r\n new fabric.Point(segment[1], segment[2]),\r\n segPercent\r\n );\r\n info.angle = Math.atan2(segment[2] - segInfo.y, segment[1] - segInfo.x);\r\n return info;\r\n case 'C':\r\n return findPercentageForDistance(segInfo, distance);\r\n case 'Q':\r\n return findPercentageForDistance(segInfo, distance);\r\n }\r\n }\r\n\r\n /**\r\n *\r\n * @param {string} pathString\r\n * @return {(string|number)[][]} An array of SVG path commands\r\n * @example