svgedit/dist/svgcanvas-iife.js

21635 lines
674 KiB
JavaScript
Raw Normal View History

var SvgCanvas = (function () {
'use strict';
function _typeof(obj) {
"@babel/helpers - typeof";
if (typeof Symbol === "function" && typeof Symbol.iterator === "symbol") {
_typeof = function (obj) {
return typeof obj;
};
} else {
_typeof = function (obj) {
return obj && typeof Symbol === "function" && obj.constructor === Symbol && obj !== Symbol.prototype ? "symbol" : typeof obj;
};
}
return _typeof(obj);
}
function asyncGeneratorStep(gen, resolve, reject, _next, _throw, key, arg) {
try {
var info = gen[key](arg);
var value = info.value;
} catch (error) {
reject(error);
return;
}
if (info.done) {
resolve(value);
} else {
Promise.resolve(value).then(_next, _throw);
}
}
function _asyncToGenerator(fn) {
return function () {
var self = this,
args = arguments;
return new Promise(function (resolve, reject) {
var gen = fn.apply(self, args);
function _next(value) {
asyncGeneratorStep(gen, resolve, reject, _next, _throw, "next", value);
}
function _throw(err) {
asyncGeneratorStep(gen, resolve, reject, _next, _throw, "throw", err);
}
_next(undefined);
});
};
}
function _classCallCheck(instance, Constructor) {
if (!(instance instanceof Constructor)) {
throw new TypeError("Cannot call a class as a function");
}
}
function _defineProperties(target, props) {
for (var i = 0; i < props.length; i++) {
var descriptor = props[i];
descriptor.enumerable = descriptor.enumerable || false;
descriptor.configurable = true;
if ("value" in descriptor) descriptor.writable = true;
Object.defineProperty(target, descriptor.key, descriptor);
}
}
function _createClass(Constructor, protoProps, staticProps) {
if (protoProps) _defineProperties(Constructor.prototype, protoProps);
if (staticProps) _defineProperties(Constructor, staticProps);
return Constructor;
}
function _inherits(subClass, superClass) {
if (typeof superClass !== "function" && superClass !== null) {
throw new TypeError("Super expression must either be null or a function");
}
subClass.prototype = Object.create(superClass && superClass.prototype, {
constructor: {
value: subClass,
writable: true,
configurable: true
}
});
if (superClass) _setPrototypeOf(subClass, superClass);
}
function _getPrototypeOf(o) {
_getPrototypeOf = Object.setPrototypeOf ? Object.getPrototypeOf : function _getPrototypeOf(o) {
return o.__proto__ || Object.getPrototypeOf(o);
};
return _getPrototypeOf(o);
}
function _setPrototypeOf(o, p) {
_setPrototypeOf = Object.setPrototypeOf || function _setPrototypeOf(o, p) {
o.__proto__ = p;
return o;
};
return _setPrototypeOf(o, p);
}
function _assertThisInitialized(self) {
if (self === void 0) {
throw new ReferenceError("this hasn't been initialised - super() hasn't been called");
}
return self;
}
function _possibleConstructorReturn(self, call) {
if (call && (typeof call === "object" || typeof call === "function")) {
return call;
}
return _assertThisInitialized(self);
}
function _slicedToArray(arr, i) {
return _arrayWithHoles(arr) || _iterableToArrayLimit(arr, i) || _nonIterableRest();
}
function _toConsumableArray(arr) {
return _arrayWithoutHoles(arr) || _iterableToArray(arr) || _nonIterableSpread();
}
function _arrayWithoutHoles(arr) {
if (Array.isArray(arr)) {
for (var i = 0, arr2 = new Array(arr.length); i < arr.length; i++) arr2[i] = arr[i];
return arr2;
}
}
function _arrayWithHoles(arr) {
if (Array.isArray(arr)) return arr;
}
function _iterableToArray(iter) {
if (Symbol.iterator in Object(iter) || Object.prototype.toString.call(iter) === "[object Arguments]") return Array.from(iter);
}
function _iterableToArrayLimit(arr, i) {
if (!(Symbol.iterator in Object(arr) || Object.prototype.toString.call(arr) === "[object Arguments]")) {
return;
}
var _arr = [];
var _n = true;
var _d = false;
var _e = undefined;
try {
for (var _i = arr[Symbol.iterator](), _s; !(_n = (_s = _i.next()).done); _n = true) {
_arr.push(_s.value);
if (i && _arr.length === i) break;
}
} catch (err) {
_d = true;
_e = err;
} finally {
try {
if (!_n && _i["return"] != null) _i["return"]();
} finally {
if (_d) throw _e;
}
}
return _arr;
}
function _nonIterableSpread() {
throw new TypeError("Invalid attempt to spread non-iterable instance");
}
function _nonIterableRest() {
throw new TypeError("Invalid attempt to destructure non-iterable instance");
}
/* globals SVGPathSeg, SVGPathSegMovetoRel, SVGPathSegMovetoAbs,
SVGPathSegMovetoRel, SVGPathSegLinetoRel, SVGPathSegLinetoAbs,
SVGPathSegLinetoHorizontalRel, SVGPathSegLinetoHorizontalAbs,
SVGPathSegLinetoVerticalRel, SVGPathSegLinetoVerticalAbs,
SVGPathSegClosePath, SVGPathSegCurvetoCubicRel,
SVGPathSegCurvetoCubicAbs, SVGPathSegCurvetoCubicSmoothRel,
SVGPathSegCurvetoCubicSmoothAbs, SVGPathSegCurvetoQuadraticRel,
SVGPathSegCurvetoQuadraticAbs, SVGPathSegCurvetoQuadraticSmoothRel,
SVGPathSegCurvetoQuadraticSmoothAbs, SVGPathSegArcRel, SVGPathSegArcAbs */
/**
* SVGPathSeg API polyfill
* https://github.com/progers/pathseg
*
* This is a drop-in replacement for the `SVGPathSeg` and `SVGPathSegList` APIs
* that were removed from SVG2 ({@link https://lists.w3.org/Archives/Public/www-svg/2015Jun/0044.html}),
* including the latest spec changes which were implemented in Firefox 43 and
* Chrome 46.
*/
/* eslint-disable no-shadow, class-methods-use-this, jsdoc/require-jsdoc */
// Linting: We avoid `no-shadow` as ESLint thinks these are still available globals
// Linting: We avoid `class-methods-use-this` as this is a polyfill that must
// follow the conventions
(function () {
if (!('SVGPathSeg' in window)) {
// Spec: https://www.w3.org/TR/SVG11/single-page.html#paths-InterfaceSVGPathSeg
var _SVGPathSeg = /*#__PURE__*/function () {
function _SVGPathSeg(type, typeAsLetter, owningPathSegList) {
_classCallCheck(this, _SVGPathSeg);
this.pathSegType = type;
this.pathSegTypeAsLetter = typeAsLetter;
this._owningPathSegList = owningPathSegList;
} // Notify owning PathSegList on any changes so they can be synchronized back to the path element.
_createClass(_SVGPathSeg, [{
key: "_segmentChanged",
value: function _segmentChanged() {
if (this._owningPathSegList) {
this._owningPathSegList.segmentChanged(this);
}
}
}]);
return _SVGPathSeg;
}();
_SVGPathSeg.prototype.classname = 'SVGPathSeg';
_SVGPathSeg.PATHSEG_UNKNOWN = 0;
_SVGPathSeg.PATHSEG_CLOSEPATH = 1;
_SVGPathSeg.PATHSEG_MOVETO_ABS = 2;
_SVGPathSeg.PATHSEG_MOVETO_REL = 3;
_SVGPathSeg.PATHSEG_LINETO_ABS = 4;
_SVGPathSeg.PATHSEG_LINETO_REL = 5;
_SVGPathSeg.PATHSEG_CURVETO_CUBIC_ABS = 6;
_SVGPathSeg.PATHSEG_CURVETO_CUBIC_REL = 7;
_SVGPathSeg.PATHSEG_CURVETO_QUADRATIC_ABS = 8;
_SVGPathSeg.PATHSEG_CURVETO_QUADRATIC_REL = 9;
_SVGPathSeg.PATHSEG_ARC_ABS = 10;
_SVGPathSeg.PATHSEG_ARC_REL = 11;
_SVGPathSeg.PATHSEG_LINETO_HORIZONTAL_ABS = 12;
_SVGPathSeg.PATHSEG_LINETO_HORIZONTAL_REL = 13;
_SVGPathSeg.PATHSEG_LINETO_VERTICAL_ABS = 14;
_SVGPathSeg.PATHSEG_LINETO_VERTICAL_REL = 15;
_SVGPathSeg.PATHSEG_CURVETO_CUBIC_SMOOTH_ABS = 16;
_SVGPathSeg.PATHSEG_CURVETO_CUBIC_SMOOTH_REL = 17;
_SVGPathSeg.PATHSEG_CURVETO_QUADRATIC_SMOOTH_ABS = 18;
_SVGPathSeg.PATHSEG_CURVETO_QUADRATIC_SMOOTH_REL = 19;
var _SVGPathSegClosePath = /*#__PURE__*/function (_SVGPathSeg2) {
_inherits(_SVGPathSegClosePath, _SVGPathSeg2);
function _SVGPathSegClosePath(owningPathSegList) {
_classCallCheck(this, _SVGPathSegClosePath);
return _possibleConstructorReturn(this, _getPrototypeOf(_SVGPathSegClosePath).call(this, _SVGPathSeg.PATHSEG_CLOSEPATH, 'z', owningPathSegList));
}
_createClass(_SVGPathSegClosePath, [{
key: "toString",
value: function toString() {
return '[object SVGPathSegClosePath]';
}
}, {
key: "_asPathString",
value: function _asPathString() {
return this.pathSegTypeAsLetter;
}
}, {
key: "clone",
value: function clone() {
return new _SVGPathSegClosePath(undefined);
}
}]);
return _SVGPathSegClosePath;
}(_SVGPathSeg);
var _SVGPathSegMovetoAbs = /*#__PURE__*/function (_SVGPathSeg3) {
_inherits(_SVGPathSegMovetoAbs, _SVGPathSeg3);
function _SVGPathSegMovetoAbs(owningPathSegList, x, y) {
var _this;
_classCallCheck(this, _SVGPathSegMovetoAbs);
_this = _possibleConstructorReturn(this, _getPrototypeOf(_SVGPathSegMovetoAbs).call(this, _SVGPathSeg.PATHSEG_MOVETO_ABS, 'M', owningPathSegList));
_this._x = x;
_this._y = y;
return _this;
}
_createClass(_SVGPathSegMovetoAbs, [{
key: "toString",
value: function toString() {
return '[object SVGPathSegMovetoAbs]';
}
}, {
key: "_asPathString",
value: function _asPathString() {
return this.pathSegTypeAsLetter + ' ' + this._x + ' ' + this._y;
}
}, {
key: "clone",
value: function clone() {
return new _SVGPathSegMovetoAbs(undefined, this._x, this._y);
}
}]);
return _SVGPathSegMovetoAbs;
}(_SVGPathSeg);
Object.defineProperties(_SVGPathSegMovetoAbs.prototype, {
x: {
get: function get() {
return this._x;
},
set: function set(x) {
this._x = x;
this._segmentChanged();
},
enumerable: true
},
y: {
get: function get() {
return this._y;
},
set: function set(y) {
this._y = y;
this._segmentChanged();
},
enumerable: true
}
});
var _SVGPathSegMovetoRel = /*#__PURE__*/function (_SVGPathSeg4) {
_inherits(_SVGPathSegMovetoRel, _SVGPathSeg4);
function _SVGPathSegMovetoRel(owningPathSegList, x, y) {
var _this2;
_classCallCheck(this, _SVGPathSegMovetoRel);
_this2 = _possibleConstructorReturn(this, _getPrototypeOf(_SVGPathSegMovetoRel).call(this, _SVGPathSeg.PATHSEG_MOVETO_REL, 'm', owningPathSegList));
_this2._x = x;
_this2._y = y;
return _this2;
}
_createClass(_SVGPathSegMovetoRel, [{
key: "toString",
value: function toString() {
return '[object SVGPathSegMovetoRel]';
}
}, {
key: "_asPathString",
value: function _asPathString() {
return this.pathSegTypeAsLetter + ' ' + this._x + ' ' + this._y;
}
}, {
key: "clone",
value: function clone() {
return new _SVGPathSegMovetoRel(undefined, this._x, this._y);
}
}]);
return _SVGPathSegMovetoRel;
}(_SVGPathSeg);
Object.defineProperties(_SVGPathSegMovetoRel.prototype, {
x: {
get: function get() {
return this._x;
},
set: function set(x) {
this._x = x;
this._segmentChanged();
},
enumerable: true
},
y: {
get: function get() {
return this._y;
},
set: function set(y) {
this._y = y;
this._segmentChanged();
},
enumerable: true
}
});
var _SVGPathSegLinetoAbs = /*#__PURE__*/function (_SVGPathSeg5) {
_inherits(_SVGPathSegLinetoAbs, _SVGPathSeg5);
function _SVGPathSegLinetoAbs(owningPathSegList, x, y) {
var _this3;
_classCallCheck(this, _SVGPathSegLinetoAbs);
_this3 = _possibleConstructorReturn(this, _getPrototypeOf(_SVGPathSegLinetoAbs).call(this, _SVGPathSeg.PATHSEG_LINETO_ABS, 'L', owningPathSegList));
_this3._x = x;
_this3._y = y;
return _this3;
}
_createClass(_SVGPathSegLinetoAbs, [{
key: "toString",
value: function toString() {
return '[object SVGPathSegLinetoAbs]';
}
}, {
key: "_asPathString",
value: function _asPathString() {
return this.pathSegTypeAsLetter + ' ' + this._x + ' ' + this._y;
}
}, {
key: "clone",
value: function clone() {
return new _SVGPathSegLinetoAbs(undefined, this._x, this._y);
}
}]);
return _SVGPathSegLinetoAbs;
}(_SVGPathSeg);
Object.defineProperties(_SVGPathSegLinetoAbs.prototype, {
x: {
get: function get() {
return this._x;
},
set: function set(x) {
this._x = x;
this._segmentChanged();
},
enumerable: true
},
y: {
get: function get() {
return this._y;
},
set: function set(y) {
this._y = y;
this._segmentChanged();
},
enumerable: true
}
});
var _SVGPathSegLinetoRel = /*#__PURE__*/function (_SVGPathSeg6) {
_inherits(_SVGPathSegLinetoRel, _SVGPathSeg6);
function _SVGPathSegLinetoRel(owningPathSegList, x, y) {
var _this4;
_classCallCheck(this, _SVGPathSegLinetoRel);
_this4 = _possibleConstructorReturn(this, _getPrototypeOf(_SVGPathSegLinetoRel).call(this, _SVGPathSeg.PATHSEG_LINETO_REL, 'l', owningPathSegList));
_this4._x = x;
_this4._y = y;
return _this4;
}
_createClass(_SVGPathSegLinetoRel, [{
key: "toString",
value: function toString() {
return '[object SVGPathSegLinetoRel]';
}
}, {
key: "_asPathString",
value: function _asPathString() {
return this.pathSegTypeAsLetter + ' ' + this._x + ' ' + this._y;
}
}, {
key: "clone",
value: function clone() {
return new _SVGPathSegLinetoRel(undefined, this._x, this._y);
}
}]);
return _SVGPathSegLinetoRel;
}(_SVGPathSeg);
Object.defineProperties(_SVGPathSegLinetoRel.prototype, {
x: {
get: function get() {
return this._x;
},
set: function set(x) {
this._x = x;
this._segmentChanged();
},
enumerable: true
},
y: {
get: function get() {
return this._y;
},
set: function set(y) {
this._y = y;
this._segmentChanged();
},
enumerable: true
}
});
var _SVGPathSegCurvetoCubicAbs = /*#__PURE__*/function (_SVGPathSeg7) {
_inherits(_SVGPathSegCurvetoCubicAbs, _SVGPathSeg7);
function _SVGPathSegCurvetoCubicAbs(owningPathSegList, x, y, x1, y1, x2, y2) {
var _this5;
_classCallCheck(this, _SVGPathSegCurvetoCubicAbs);
_this5 = _possibleConstructorReturn(this, _getPrototypeOf(_SVGPathSegCurvetoCubicAbs).call(this, _SVGPathSeg.PATHSEG_CURVETO_CUBIC_ABS, 'C', owningPathSegList));
_this5._x = x;
_this5._y = y;
_this5._x1 = x1;
_this5._y1 = y1;
_this5._x2 = x2;
_this5._y2 = y2;
return _this5;
}
_createClass(_SVGPathSegCurvetoCubicAbs, [{
key: "toString",
value: function toString() {
return '[object SVGPathSegCurvetoCubicAbs]';
}
}, {
key: "_asPathString",
value: function _asPathString() {
return this.pathSegTypeAsLetter + ' ' + this._x1 + ' ' + this._y1 + ' ' + this._x2 + ' ' + this._y2 + ' ' + this._x + ' ' + this._y;
}
}, {
key: "clone",
value: function clone() {
return new _SVGPathSegCurvetoCubicAbs(undefined, this._x, this._y, this._x1, this._y1, this._x2, this._y2);
}
}]);
return _SVGPathSegCurvetoCubicAbs;
}(_SVGPathSeg);
Object.defineProperties(_SVGPathSegCurvetoCubicAbs.prototype, {
x: {
get: function get() {
return this._x;
},
set: function set(x) {
this._x = x;
this._segmentChanged();
},
enumerable: true
},
y: {
get: function get() {
return this._y;
},
set: function set(y) {
this._y = y;
this._segmentChanged();
},
enumerable: true
},
x1: {
get: function get() {
return this._x1;
},
set: function set(x1) {
this._x1 = x1;
this._segmentChanged();
},
enumerable: true
},
y1: {
get: function get() {
return this._y1;
},
set: function set(y1) {
this._y1 = y1;
this._segmentChanged();
},
enumerable: true
},
x2: {
get: function get() {
return this._x2;
},
set: function set(x2) {
this._x2 = x2;
this._segmentChanged();
},
enumerable: true
},
y2: {
get: function get() {
return this._y2;
},
set: function set(y2) {
this._y2 = y2;
this._segmentChanged();
},
enumerable: true
}
});
var _SVGPathSegCurvetoCubicRel = /*#__PURE__*/function (_SVGPathSeg8) {
_inherits(_SVGPathSegCurvetoCubicRel, _SVGPathSeg8);
function _SVGPathSegCurvetoCubicRel(owningPathSegList, x, y, x1, y1, x2, y2) {
var _this6;
_classCallCheck(this, _SVGPathSegCurvetoCubicRel);
_this6 = _possibleConstructorReturn(this, _getPrototypeOf(_SVGPathSegCurvetoCubicRel).call(this, _SVGPathSeg.PATHSEG_CURVETO_CUBIC_REL, 'c', owningPathSegList));
_this6._x = x;
_this6._y = y;
_this6._x1 = x1;
_this6._y1 = y1;
_this6._x2 = x2;
_this6._y2 = y2;
return _this6;
}
_createClass(_SVGPathSegCurvetoCubicRel, [{
key: "toString",
value: function toString() {
return '[object SVGPathSegCurvetoCubicRel]';
}
}, {
key: "_asPathString",
value: function _asPathString() {
return this.pathSegTypeAsLetter + ' ' + this._x1 + ' ' + this._y1 + ' ' + this._x2 + ' ' + this._y2 + ' ' + this._x + ' ' + this._y;
}
}, {
key: "clone",
value: function clone() {
return new _SVGPathSegCurvetoCubicRel(undefined, this._x, this._y, this._x1, this._y1, this._x2, this._y2);
}
}]);
return _SVGPathSegCurvetoCubicRel;
}(_SVGPathSeg);
Object.defineProperties(_SVGPathSegCurvetoCubicRel.prototype, {
x: {
get: function get() {
return this._x;
},
set: function set(x) {
this._x = x;
this._segmentChanged();
},
enumerable: true
},
y: {
get: function get() {
return this._y;
},
set: function set(y) {
this._y = y;
this._segmentChanged();
},
enumerable: true
},
x1: {
get: function get() {
return this._x1;
},
set: function set(x1) {
this._x1 = x1;
this._segmentChanged();
},
enumerable: true
},
y1: {
get: function get() {
return this._y1;
},
set: function set(y1) {
this._y1 = y1;
this._segmentChanged();
},
enumerable: true
},
x2: {
get: function get() {
return this._x2;
},
set: function set(x2) {
this._x2 = x2;
this._segmentChanged();
},
enumerable: true
},
y2: {
get: function get() {
return this._y2;
},
set: function set(y2) {
this._y2 = y2;
this._segmentChanged();
},
enumerable: true
}
});
var _SVGPathSegCurvetoQuadraticAbs = /*#__PURE__*/function (_SVGPathSeg9) {
_inherits(_SVGPathSegCurvetoQuadraticAbs, _SVGPathSeg9);
function _SVGPathSegCurvetoQuadraticAbs(owningPathSegList, x, y, x1, y1) {
var _this7;
_classCallCheck(this, _SVGPathSegCurvetoQuadraticAbs);
_this7 = _possibleConstructorReturn(this, _getPrototypeOf(_SVGPathSegCurvetoQuadraticAbs).call(this, _SVGPathSeg.PATHSEG_CURVETO_QUADRATIC_ABS, 'Q', owningPathSegList));
_this7._x = x;
_this7._y = y;
_this7._x1 = x1;
_this7._y1 = y1;
return _this7;
}
_createClass(_SVGPathSegCurvetoQuadraticAbs, [{
key: "toString",
value: function toString() {
return '[object SVGPathSegCurvetoQuadraticAbs]';
}
}, {
key: "_asPathString",
value: function _asPathString() {
return this.pathSegTypeAsLetter + ' ' + this._x1 + ' ' + this._y1 + ' ' + this._x + ' ' + this._y;
}
}, {
key: "clone",
value: function clone() {
return new _SVGPathSegCurvetoQuadraticAbs(undefined, this._x, this._y, this._x1, this._y1);
}
}]);
return _SVGPathSegCurvetoQuadraticAbs;
}(_SVGPathSeg);
Object.defineProperties(_SVGPathSegCurvetoQuadraticAbs.prototype, {
x: {
get: function get() {
return this._x;
},
set: function set(x) {
this._x = x;
this._segmentChanged();
},
enumerable: true
},
y: {
get: function get() {
return this._y;
},
set: function set(y) {
this._y = y;
this._segmentChanged();
},
enumerable: true
},
x1: {
get: function get() {
return this._x1;
},
set: function set(x1) {
this._x1 = x1;
this._segmentChanged();
},
enumerable: true
},
y1: {
get: function get() {
return this._y1;
},
set: function set(y1) {
this._y1 = y1;
this._segmentChanged();
},
enumerable: true
}
});
var _SVGPathSegCurvetoQuadraticRel = /*#__PURE__*/function (_SVGPathSeg10) {
_inherits(_SVGPathSegCurvetoQuadraticRel, _SVGPathSeg10);
function _SVGPathSegCurvetoQuadraticRel(owningPathSegList, x, y, x1, y1) {
var _this8;
_classCallCheck(this, _SVGPathSegCurvetoQuadraticRel);
_this8 = _possibleConstructorReturn(this, _getPrototypeOf(_SVGPathSegCurvetoQuadraticRel).call(this, _SVGPathSeg.PATHSEG_CURVETO_QUADRATIC_REL, 'q', owningPathSegList));
_this8._x = x;
_this8._y = y;
_this8._x1 = x1;
_this8._y1 = y1;
return _this8;
}
_createClass(_SVGPathSegCurvetoQuadraticRel, [{
key: "toString",
value: function toString() {
return '[object SVGPathSegCurvetoQuadraticRel]';
}
}, {
key: "_asPathString",
value: function _asPathString() {
return this.pathSegTypeAsLetter + ' ' + this._x1 + ' ' + this._y1 + ' ' + this._x + ' ' + this._y;
}
}, {
key: "clone",
value: function clone() {
return new _SVGPathSegCurvetoQuadraticRel(undefined, this._x, this._y, this._x1, this._y1);
}
}]);
return _SVGPathSegCurvetoQuadraticRel;
}(_SVGPathSeg);
Object.defineProperties(_SVGPathSegCurvetoQuadraticRel.prototype, {
x: {
get: function get() {
return this._x;
},
set: function set(x) {
this._x = x;
this._segmentChanged();
},
enumerable: true
},
y: {
get: function get() {
return this._y;
},
set: function set(y) {
this._y = y;
this._segmentChanged();
},
enumerable: true
},
x1: {
get: function get() {
return this._x1;
},
set: function set(x1) {
this._x1 = x1;
this._segmentChanged();
},
enumerable: true
},
y1: {
get: function get() {
return this._y1;
},
set: function set(y1) {
this._y1 = y1;
this._segmentChanged();
},
enumerable: true
}
});
var _SVGPathSegArcAbs = /*#__PURE__*/function (_SVGPathSeg11) {
_inherits(_SVGPathSegArcAbs, _SVGPathSeg11);
function _SVGPathSegArcAbs(owningPathSegList, x, y, r1, r2, angle, largeArcFlag, sweepFlag) {
var _this9;
_classCallCheck(this, _SVGPathSegArcAbs);
_this9 = _possibleConstructorReturn(this, _getPrototypeOf(_SVGPathSegArcAbs).call(this, _SVGPathSeg.PATHSEG_ARC_ABS, 'A', owningPathSegList));
_this9._x = x;
_this9._y = y;
_this9._r1 = r1;
_this9._r2 = r2;
_this9._angle = angle;
_this9._largeArcFlag = largeArcFlag;
_this9._sweepFlag = sweepFlag;
return _this9;
}
_createClass(_SVGPathSegArcAbs, [{
key: "toString",
value: function toString() {
return '[object SVGPathSegArcAbs]';
}
}, {
key: "_asPathString",
value: function _asPathString() {
return this.pathSegTypeAsLetter + ' ' + this._r1 + ' ' + this._r2 + ' ' + this._angle + ' ' + (this._largeArcFlag ? '1' : '0') + ' ' + (this._sweepFlag ? '1' : '0') + ' ' + this._x + ' ' + this._y;
}
}, {
key: "clone",
value: function clone() {
return new _SVGPathSegArcAbs(undefined, this._x, this._y, this._r1, this._r2, this._angle, this._largeArcFlag, this._sweepFlag);
}
}]);
return _SVGPathSegArcAbs;
}(_SVGPathSeg);
Object.defineProperties(_SVGPathSegArcAbs.prototype, {
x: {
get: function get() {
return this._x;
},
set: function set(x) {
this._x = x;
this._segmentChanged();
},
enumerable: true
},
y: {
get: function get() {
return this._y;
},
set: function set(y) {
this._y = y;
this._segmentChanged();
},
enumerable: true
},
r1: {
get: function get() {
return this._r1;
},
set: function set(r1) {
this._r1 = r1;
this._segmentChanged();
},
enumerable: true
},
r2: {
get: function get() {
return this._r2;
},
set: function set(r2) {
this._r2 = r2;
this._segmentChanged();
},
enumerable: true
},
angle: {
get: function get() {
return this._angle;
},
set: function set(angle) {
this._angle = angle;
this._segmentChanged();
},
enumerable: true
},
largeArcFlag: {
get: function get() {
return this._largeArcFlag;
},
set: function set(largeArcFlag) {
this._largeArcFlag = largeArcFlag;
this._segmentChanged();
},
enumerable: true
},
sweepFlag: {
get: function get() {
return this._sweepFlag;
},
set: function set(sweepFlag) {
this._sweepFlag = sweepFlag;
this._segmentChanged();
},
enumerable: true
}
});
var _SVGPathSegArcRel = /*#__PURE__*/function (_SVGPathSeg12) {
_inherits(_SVGPathSegArcRel, _SVGPathSeg12);
function _SVGPathSegArcRel(owningPathSegList, x, y, r1, r2, angle, largeArcFlag, sweepFlag) {
var _this10;
_classCallCheck(this, _SVGPathSegArcRel);
_this10 = _possibleConstructorReturn(this, _getPrototypeOf(_SVGPathSegArcRel).call(this, _SVGPathSeg.PATHSEG_ARC_REL, 'a', owningPathSegList));
_this10._x = x;
_this10._y = y;
_this10._r1 = r1;
_this10._r2 = r2;
_this10._angle = angle;
_this10._largeArcFlag = largeArcFlag;
_this10._sweepFlag = sweepFlag;
return _this10;
}
_createClass(_SVGPathSegArcRel, [{
key: "toString",
value: function toString() {
return '[object SVGPathSegArcRel]';
}
}, {
key: "_asPathString",
value: function _asPathString() {
return this.pathSegTypeAsLetter + ' ' + this._r1 + ' ' + this._r2 + ' ' + this._angle + ' ' + (this._largeArcFlag ? '1' : '0') + ' ' + (this._sweepFlag ? '1' : '0') + ' ' + this._x + ' ' + this._y;
}
}, {
key: "clone",
value: function clone() {
return new _SVGPathSegArcRel(undefined, this._x, this._y, this._r1, this._r2, this._angle, this._largeArcFlag, this._sweepFlag);
}
}]);
return _SVGPathSegArcRel;
}(_SVGPathSeg);
Object.defineProperties(_SVGPathSegArcRel.prototype, {
x: {
get: function get() {
return this._x;
},
set: function set(x) {
this._x = x;
this._segmentChanged();
},
enumerable: true
},
y: {
get: function get() {
return this._y;
},
set: function set(y) {
this._y = y;
this._segmentChanged();
},
enumerable: true
},
r1: {
get: function get() {
return this._r1;
},
set: function set(r1) {
this._r1 = r1;
this._segmentChanged();
},
enumerable: true
},
r2: {
get: function get() {
return this._r2;
},
set: function set(r2) {
this._r2 = r2;
this._segmentChanged();
},
enumerable: true
},
angle: {
get: function get() {
return this._angle;
},
set: function set(angle) {
this._angle = angle;
this._segmentChanged();
},
enumerable: true
},
largeArcFlag: {
get: function get() {
return this._largeArcFlag;
},
set: function set(largeArcFlag) {
this._largeArcFlag = largeArcFlag;
this._segmentChanged();
},
enumerable: true
},
sweepFlag: {
get: function get() {
return this._sweepFlag;
},
set: function set(sweepFlag) {
this._sweepFlag = sweepFlag;
this._segmentChanged();
},
enumerable: true
}
});
var _SVGPathSegLinetoHorizontalAbs = /*#__PURE__*/function (_SVGPathSeg13) {
_inherits(_SVGPathSegLinetoHorizontalAbs, _SVGPathSeg13);
function _SVGPathSegLinetoHorizontalAbs(owningPathSegList, x) {
var _this11;
_classCallCheck(this, _SVGPathSegLinetoHorizontalAbs);
_this11 = _possibleConstructorReturn(this, _getPrototypeOf(_SVGPathSegLinetoHorizontalAbs).call(this, _SVGPathSeg.PATHSEG_LINETO_HORIZONTAL_ABS, 'H', owningPathSegList));
_this11._x = x;
return _this11;
}
_createClass(_SVGPathSegLinetoHorizontalAbs, [{
key: "toString",
value: function toString() {
return '[object SVGPathSegLinetoHorizontalAbs]';
}
}, {
key: "_asPathString",
value: function _asPathString() {
return this.pathSegTypeAsLetter + ' ' + this._x;
}
}, {
key: "clone",
value: function clone() {
return new _SVGPathSegLinetoHorizontalAbs(undefined, this._x);
}
}]);
return _SVGPathSegLinetoHorizontalAbs;
}(_SVGPathSeg);
Object.defineProperty(_SVGPathSegLinetoHorizontalAbs.prototype, 'x', {
get: function get() {
return this._x;
},
set: function set(x) {
this._x = x;
this._segmentChanged();
},
enumerable: true
});
var _SVGPathSegLinetoHorizontalRel = /*#__PURE__*/function (_SVGPathSeg14) {
_inherits(_SVGPathSegLinetoHorizontalRel, _SVGPathSeg14);
function _SVGPathSegLinetoHorizontalRel(owningPathSegList, x) {
var _this12;
_classCallCheck(this, _SVGPathSegLinetoHorizontalRel);
_this12 = _possibleConstructorReturn(this, _getPrototypeOf(_SVGPathSegLinetoHorizontalRel).call(this, _SVGPathSeg.PATHSEG_LINETO_HORIZONTAL_REL, 'h', owningPathSegList));
_this12._x = x;
return _this12;
}
_createClass(_SVGPathSegLinetoHorizontalRel, [{
key: "toString",
value: function toString() {
return '[object SVGPathSegLinetoHorizontalRel]';
}
}, {
key: "_asPathString",
value: function _asPathString() {
return this.pathSegTypeAsLetter + ' ' + this._x;
}
}, {
key: "clone",
value: function clone() {
return new _SVGPathSegLinetoHorizontalRel(undefined, this._x);
}
}]);
return _SVGPathSegLinetoHorizontalRel;
}(_SVGPathSeg);
Object.defineProperty(_SVGPathSegLinetoHorizontalRel.prototype, 'x', {
get: function get() {
return this._x;
},
set: function set(x) {
this._x = x;
this._segmentChanged();
},
enumerable: true
});
var _SVGPathSegLinetoVerticalAbs = /*#__PURE__*/function (_SVGPathSeg15) {
_inherits(_SVGPathSegLinetoVerticalAbs, _SVGPathSeg15);
function _SVGPathSegLinetoVerticalAbs(owningPathSegList, y) {
var _this13;
_classCallCheck(this, _SVGPathSegLinetoVerticalAbs);
_this13 = _possibleConstructorReturn(this, _getPrototypeOf(_SVGPathSegLinetoVerticalAbs).call(this, _SVGPathSeg.PATHSEG_LINETO_VERTICAL_ABS, 'V', owningPathSegList));
_this13._y = y;
return _this13;
}
_createClass(_SVGPathSegLinetoVerticalAbs, [{
key: "toString",
value: function toString() {
return '[object SVGPathSegLinetoVerticalAbs]';
}
}, {
key: "_asPathString",
value: function _asPathString() {
return this.pathSegTypeAsLetter + ' ' + this._y;
}
}, {
key: "clone",
value: function clone() {
return new _SVGPathSegLinetoVerticalAbs(undefined, this._y);
}
}]);
return _SVGPathSegLinetoVerticalAbs;
}(_SVGPathSeg);
Object.defineProperty(_SVGPathSegLinetoVerticalAbs.prototype, 'y', {
get: function get() {
return this._y;
},
set: function set(y) {
this._y = y;
this._segmentChanged();
},
enumerable: true
});
var _SVGPathSegLinetoVerticalRel = /*#__PURE__*/function (_SVGPathSeg16) {
_inherits(_SVGPathSegLinetoVerticalRel, _SVGPathSeg16);
function _SVGPathSegLinetoVerticalRel(owningPathSegList, y) {
var _this14;
_classCallCheck(this, _SVGPathSegLinetoVerticalRel);
_this14 = _possibleConstructorReturn(this, _getPrototypeOf(_SVGPathSegLinetoVerticalRel).call(this, _SVGPathSeg.PATHSEG_LINETO_VERTICAL_REL, 'v', owningPathSegList));
_this14._y = y;
return _this14;
}
_createClass(_SVGPathSegLinetoVerticalRel, [{
key: "toString",
value: function toString() {
return '[object SVGPathSegLinetoVerticalRel]';
}
}, {
key: "_asPathString",
value: function _asPathString() {
return this.pathSegTypeAsLetter + ' ' + this._y;
}
}, {
key: "clone",
value: function clone() {
return new _SVGPathSegLinetoVerticalRel(undefined, this._y);
}
}]);
return _SVGPathSegLinetoVerticalRel;
}(_SVGPathSeg);
Object.defineProperty(_SVGPathSegLinetoVerticalRel.prototype, 'y', {
get: function get() {
return this._y;
},
set: function set(y) {
this._y = y;
this._segmentChanged();
},
enumerable: true
});
var _SVGPathSegCurvetoCubicSmoothAbs = /*#__PURE__*/function (_SVGPathSeg17) {
_inherits(_SVGPathSegCurvetoCubicSmoothAbs, _SVGPathSeg17);
function _SVGPathSegCurvetoCubicSmoothAbs(owningPathSegList, x, y, x2, y2) {
var _this15;
_classCallCheck(this, _SVGPathSegCurvetoCubicSmoothAbs);
_this15 = _possibleConstructorReturn(this, _getPrototypeOf(_SVGPathSegCurvetoCubicSmoothAbs).call(this, _SVGPathSeg.PATHSEG_CURVETO_CUBIC_SMOOTH_ABS, 'S', owningPathSegList));
_this15._x = x;
_this15._y = y;
_this15._x2 = x2;
_this15._y2 = y2;
return _this15;
}
_createClass(_SVGPathSegCurvetoCubicSmoothAbs, [{
key: "toString",
value: function toString() {
return '[object SVGPathSegCurvetoCubicSmoothAbs]';
}
}, {
key: "_asPathString",
value: function _asPathString() {
return this.pathSegTypeAsLetter + ' ' + this._x2 + ' ' + this._y2 + ' ' + this._x + ' ' + this._y;
}
}, {
key: "clone",
value: function clone() {
return new _SVGPathSegCurvetoCubicSmoothAbs(undefined, this._x, this._y, this._x2, this._y2);
}
}]);
return _SVGPathSegCurvetoCubicSmoothAbs;
}(_SVGPathSeg);
Object.defineProperties(_SVGPathSegCurvetoCubicSmoothAbs.prototype, {
x: {
get: function get() {
return this._x;
},
set: function set(x) {
this._x = x;
this._segmentChanged();
},
enumerable: true
},
y: {
get: function get() {
return this._y;
},
set: function set(y) {
this._y = y;
this._segmentChanged();
},
enumerable: true
},
x2: {
get: function get() {
return this._x2;
},
set: function set(x2) {
this._x2 = x2;
this._segmentChanged();
},
enumerable: true
},
y2: {
get: function get() {
return this._y2;
},
set: function set(y2) {
this._y2 = y2;
this._segmentChanged();
},
enumerable: true
}
});
var _SVGPathSegCurvetoCubicSmoothRel = /*#__PURE__*/function (_SVGPathSeg18) {
_inherits(_SVGPathSegCurvetoCubicSmoothRel, _SVGPathSeg18);
function _SVGPathSegCurvetoCubicSmoothRel(owningPathSegList, x, y, x2, y2) {
var _this16;
_classCallCheck(this, _SVGPathSegCurvetoCubicSmoothRel);
_this16 = _possibleConstructorReturn(this, _getPrototypeOf(_SVGPathSegCurvetoCubicSmoothRel).call(this, _SVGPathSeg.PATHSEG_CURVETO_CUBIC_SMOOTH_REL, 's', owningPathSegList));
_this16._x = x;
_this16._y = y;
_this16._x2 = x2;
_this16._y2 = y2;
return _this16;
}
_createClass(_SVGPathSegCurvetoCubicSmoothRel, [{
key: "toString",
value: function toString() {
return '[object SVGPathSegCurvetoCubicSmoothRel]';
}
}, {
key: "_asPathString",
value: function _asPathString() {
return this.pathSegTypeAsLetter + ' ' + this._x2 + ' ' + this._y2 + ' ' + this._x + ' ' + this._y;
}
}, {
key: "clone",
value: function clone() {
return new _SVGPathSegCurvetoCubicSmoothRel(undefined, this._x, this._y, this._x2, this._y2);
}
}]);
return _SVGPathSegCurvetoCubicSmoothRel;
}(_SVGPathSeg);
Object.defineProperties(_SVGPathSegCurvetoCubicSmoothRel.prototype, {
x: {
get: function get() {
return this._x;
},
set: function set(x) {
this._x = x;
this._segmentChanged();
},
enumerable: true
},
y: {
get: function get() {
return this._y;
},
set: function set(y) {
this._y = y;
this._segmentChanged();
},
enumerable: true
},
x2: {
get: function get() {
return this._x2;
},
set: function set(x2) {
this._x2 = x2;
this._segmentChanged();
},
enumerable: true
},
y2: {
get: function get() {
return this._y2;
},
set: function set(y2) {
this._y2 = y2;
this._segmentChanged();
},
enumerable: true
}
});
var _SVGPathSegCurvetoQuadraticSmoothAbs = /*#__PURE__*/function (_SVGPathSeg19) {
_inherits(_SVGPathSegCurvetoQuadraticSmoothAbs, _SVGPathSeg19);
function _SVGPathSegCurvetoQuadraticSmoothAbs(owningPathSegList, x, y) {
var _this17;
_classCallCheck(this, _SVGPathSegCurvetoQuadraticSmoothAbs);
_this17 = _possibleConstructorReturn(this, _getPrototypeOf(_SVGPathSegCurvetoQuadraticSmoothAbs).call(this, _SVGPathSeg.PATHSEG_CURVETO_QUADRATIC_SMOOTH_ABS, 'T', owningPathSegList));
_this17._x = x;
_this17._y = y;
return _this17;
}
_createClass(_SVGPathSegCurvetoQuadraticSmoothAbs, [{
key: "toString",
value: function toString() {
return '[object SVGPathSegCurvetoQuadraticSmoothAbs]';
}
}, {
key: "_asPathString",
value: function _asPathString() {
return this.pathSegTypeAsLetter + ' ' + this._x + ' ' + this._y;
}
}, {
key: "clone",
value: function clone() {
return new _SVGPathSegCurvetoQuadraticSmoothAbs(undefined, this._x, this._y);
}
}]);
return _SVGPathSegCurvetoQuadraticSmoothAbs;
}(_SVGPathSeg);
Object.defineProperties(_SVGPathSegCurvetoQuadraticSmoothAbs.prototype, {
x: {
get: function get() {
return this._x;
},
set: function set(x) {
this._x = x;
this._segmentChanged();
},
enumerable: true
},
y: {
get: function get() {
return this._y;
},
set: function set(y) {
this._y = y;
this._segmentChanged();
},
enumerable: true
}
});
var _SVGPathSegCurvetoQuadraticSmoothRel = /*#__PURE__*/function (_SVGPathSeg20) {
_inherits(_SVGPathSegCurvetoQuadraticSmoothRel, _SVGPathSeg20);
function _SVGPathSegCurvetoQuadraticSmoothRel(owningPathSegList, x, y) {
var _this18;
_classCallCheck(this, _SVGPathSegCurvetoQuadraticSmoothRel);
_this18 = _possibleConstructorReturn(this, _getPrototypeOf(_SVGPathSegCurvetoQuadraticSmoothRel).call(this, _SVGPathSeg.PATHSEG_CURVETO_QUADRATIC_SMOOTH_REL, 't', owningPathSegList));
_this18._x = x;
_this18._y = y;
return _this18;
}
_createClass(_SVGPathSegCurvetoQuadraticSmoothRel, [{
key: "toString",
value: function toString() {
return '[object SVGPathSegCurvetoQuadraticSmoothRel]';
}
}, {
key: "_asPathString",
value: function _asPathString() {
return this.pathSegTypeAsLetter + ' ' + this._x + ' ' + this._y;
}
}, {
key: "clone",
value: function clone() {
return new _SVGPathSegCurvetoQuadraticSmoothRel(undefined, this._x, this._y);
}
}]);
return _SVGPathSegCurvetoQuadraticSmoothRel;
}(_SVGPathSeg);
Object.defineProperties(_SVGPathSegCurvetoQuadraticSmoothRel.prototype, {
x: {
get: function get() {
return this._x;
},
set: function set(x) {
this._x = x;
this._segmentChanged();
},
enumerable: true
},
y: {
get: function get() {
return this._y;
},
set: function set(y) {
this._y = y;
this._segmentChanged();
},
enumerable: true
}
}); // Add createSVGPathSeg* functions to SVGPathElement.
// Spec: https://www.w3.org/TR/SVG11/single-page.html#paths-InterfaceSVGPathElement.
SVGPathElement.prototype.createSVGPathSegClosePath = function () {
return new _SVGPathSegClosePath(undefined);
};
SVGPathElement.prototype.createSVGPathSegMovetoAbs = function (x, y) {
return new _SVGPathSegMovetoAbs(undefined, x, y);
};
SVGPathElement.prototype.createSVGPathSegMovetoRel = function (x, y) {
return new _SVGPathSegMovetoRel(undefined, x, y);
};
SVGPathElement.prototype.createSVGPathSegLinetoAbs = function (x, y) {
return new _SVGPathSegLinetoAbs(undefined, x, y);
};
SVGPathElement.prototype.createSVGPathSegLinetoRel = function (x, y) {
return new _SVGPathSegLinetoRel(undefined, x, y);
};
SVGPathElement.prototype.createSVGPathSegCurvetoCubicAbs = function (x, y, x1, y1, x2, y2) {
return new _SVGPathSegCurvetoCubicAbs(undefined, x, y, x1, y1, x2, y2);
};
SVGPathElement.prototype.createSVGPathSegCurvetoCubicRel = function (x, y, x1, y1, x2, y2) {
return new _SVGPathSegCurvetoCubicRel(undefined, x, y, x1, y1, x2, y2);
};
SVGPathElement.prototype.createSVGPathSegCurvetoQuadraticAbs = function (x, y, x1, y1) {
return new _SVGPathSegCurvetoQuadraticAbs(undefined, x, y, x1, y1);
};
SVGPathElement.prototype.createSVGPathSegCurvetoQuadraticRel = function (x, y, x1, y1) {
return new _SVGPathSegCurvetoQuadraticRel(undefined, x, y, x1, y1);
};
SVGPathElement.prototype.createSVGPathSegArcAbs = function (x, y, r1, r2, angle, largeArcFlag, sweepFlag) {
return new _SVGPathSegArcAbs(undefined, x, y, r1, r2, angle, largeArcFlag, sweepFlag);
};
SVGPathElement.prototype.createSVGPathSegArcRel = function (x, y, r1, r2, angle, largeArcFlag, sweepFlag) {
return new _SVGPathSegArcRel(undefined, x, y, r1, r2, angle, largeArcFlag, sweepFlag);
};
SVGPathElement.prototype.createSVGPathSegLinetoHorizontalAbs = function (x) {
return new _SVGPathSegLinetoHorizontalAbs(undefined, x);
};
SVGPathElement.prototype.createSVGPathSegLinetoHorizontalRel = function (x) {
return new _SVGPathSegLinetoHorizontalRel(undefined, x);
};
SVGPathElement.prototype.createSVGPathSegLinetoVerticalAbs = function (y) {
return new _SVGPathSegLinetoVerticalAbs(undefined, y);
};
SVGPathElement.prototype.createSVGPathSegLinetoVerticalRel = function (y) {
return new _SVGPathSegLinetoVerticalRel(undefined, y);
};
SVGPathElement.prototype.createSVGPathSegCurvetoCubicSmoothAbs = function (x, y, x2, y2) {
return new _SVGPathSegCurvetoCubicSmoothAbs(undefined, x, y, x2, y2);
};
SVGPathElement.prototype.createSVGPathSegCurvetoCubicSmoothRel = function (x, y, x2, y2) {
return new _SVGPathSegCurvetoCubicSmoothRel(undefined, x, y, x2, y2);
};
SVGPathElement.prototype.createSVGPathSegCurvetoQuadraticSmoothAbs = function (x, y) {
return new _SVGPathSegCurvetoQuadraticSmoothAbs(undefined, x, y);
};
SVGPathElement.prototype.createSVGPathSegCurvetoQuadraticSmoothRel = function (x, y) {
return new _SVGPathSegCurvetoQuadraticSmoothRel(undefined, x, y);
};
if (!('getPathSegAtLength' in SVGPathElement.prototype)) {
// Add getPathSegAtLength to SVGPathElement.
// Spec: https://www.w3.org/TR/SVG11/single-page.html#paths-__svg__SVGPathElement__getPathSegAtLength
// This polyfill requires SVGPathElement.getTotalLength to implement the distance-along-a-path algorithm.
SVGPathElement.prototype.getPathSegAtLength = function (distance) {
if (distance === undefined || !isFinite(distance)) {
throw new Error('Invalid arguments.');
}
var measurementElement = document.createElementNS('http://www.w3.org/2000/svg', 'path');
measurementElement.setAttribute('d', this.getAttribute('d'));
var lastPathSegment = measurementElement.pathSegList.numberOfItems - 1; // If the path is empty, return 0.
if (lastPathSegment <= 0) {
return 0;
}
do {
measurementElement.pathSegList.removeItem(lastPathSegment);
if (distance > measurementElement.getTotalLength()) {
break;
}
lastPathSegment--;
} while (lastPathSegment > 0);
return lastPathSegment;
};
}
window.SVGPathSeg = _SVGPathSeg;
window.SVGPathSegClosePath = _SVGPathSegClosePath;
window.SVGPathSegMovetoAbs = _SVGPathSegMovetoAbs;
window.SVGPathSegMovetoRel = _SVGPathSegMovetoRel;
window.SVGPathSegLinetoAbs = _SVGPathSegLinetoAbs;
window.SVGPathSegLinetoRel = _SVGPathSegLinetoRel;
window.SVGPathSegCurvetoCubicAbs = _SVGPathSegCurvetoCubicAbs;
window.SVGPathSegCurvetoCubicRel = _SVGPathSegCurvetoCubicRel;
window.SVGPathSegCurvetoQuadraticAbs = _SVGPathSegCurvetoQuadraticAbs;
window.SVGPathSegCurvetoQuadraticRel = _SVGPathSegCurvetoQuadraticRel;
window.SVGPathSegArcAbs = _SVGPathSegArcAbs;
window.SVGPathSegArcRel = _SVGPathSegArcRel;
window.SVGPathSegLinetoHorizontalAbs = _SVGPathSegLinetoHorizontalAbs;
window.SVGPathSegLinetoHorizontalRel = _SVGPathSegLinetoHorizontalRel;
window.SVGPathSegLinetoVerticalAbs = _SVGPathSegLinetoVerticalAbs;
window.SVGPathSegLinetoVerticalRel = _SVGPathSegLinetoVerticalRel;
window.SVGPathSegCurvetoCubicSmoothAbs = _SVGPathSegCurvetoCubicSmoothAbs;
window.SVGPathSegCurvetoCubicSmoothRel = _SVGPathSegCurvetoCubicSmoothRel;
window.SVGPathSegCurvetoQuadraticSmoothAbs = _SVGPathSegCurvetoQuadraticSmoothAbs;
window.SVGPathSegCurvetoQuadraticSmoothRel = _SVGPathSegCurvetoQuadraticSmoothRel;
} // Checking for SVGPathSegList in window checks for the case of an implementation without the
// SVGPathSegList API.
// The second check for appendItem is specific to Firefox 59+ which removed only parts of the
// SVGPathSegList API (e.g., appendItem). In this case we need to re-implement the entire API
// so the polyfill data (i.e., _list) is used throughout.
if (!('SVGPathSegList' in window) || !('appendItem' in window.SVGPathSegList.prototype)) {
// Spec: https://www.w3.org/TR/SVG11/single-page.html#paths-InterfaceSVGPathSegList
var SVGPathSegList = /*#__PURE__*/function () {
function SVGPathSegList(pathElement) {
_classCallCheck(this, SVGPathSegList);
this._pathElement = pathElement;
this._list = this._parsePath(this._pathElement.getAttribute('d')); // Use a MutationObserver to catch changes to the path's "d" attribute.
this._mutationObserverConfig = {
attributes: true,
attributeFilter: ['d']
};
this._pathElementMutationObserver = new MutationObserver(this._updateListFromPathMutations.bind(this));
this._pathElementMutationObserver.observe(this._pathElement, this._mutationObserverConfig);
} // Process any pending mutations to the path element and update the list as needed.
// This should be the first call of all public functions and is needed because
// MutationObservers are not synchronous so we can have pending asynchronous mutations.
_createClass(SVGPathSegList, [{
key: "_checkPathSynchronizedToList",
value: function _checkPathSynchronizedToList() {
this._updateListFromPathMutations(this._pathElementMutationObserver.takeRecords());
}
}, {
key: "_updateListFromPathMutations",
value: function _updateListFromPathMutations(mutationRecords) {
if (!this._pathElement) {
return;
}
var hasPathMutations = false;
mutationRecords.forEach(function (record) {
if (record.attributeName === 'd') {
hasPathMutations = true;
}
});
if (hasPathMutations) {
this._list = this._parsePath(this._pathElement.getAttribute('d'));
}
} // Serialize the list and update the path's 'd' attribute.
}, {
key: "_writeListToPath",
value: function _writeListToPath() {
this._pathElementMutationObserver.disconnect();
this._pathElement.setAttribute('d', SVGPathSegList._pathSegArrayAsString(this._list));
this._pathElementMutationObserver.observe(this._pathElement, this._mutationObserverConfig);
} // When a path segment changes the list needs to be synchronized back to the path element.
}, {
key: "segmentChanged",
value: function segmentChanged(pathSeg) {
this._writeListToPath();
}
}, {
key: "clear",
value: function clear() {
this._checkPathSynchronizedToList();
this._list.forEach(function (pathSeg) {
pathSeg._owningPathSegList = null;
});
this._list = [];
this._writeListToPath();
}
}, {
key: "initialize",
value: function initialize(newItem) {
this._checkPathSynchronizedToList();
this._list = [newItem];
newItem._owningPathSegList = this;
this._writeListToPath();
return newItem;
}
}, {
key: "_checkValidIndex",
value: function _checkValidIndex(index) {
if (isNaN(index) || index < 0 || index >= this.numberOfItems) {
throw new Error('INDEX_SIZE_ERR');
}
}
}, {
key: "getItem",
value: function getItem(index) {
this._checkPathSynchronizedToList();
this._checkValidIndex(index);
return this._list[index];
}
}, {
key: "insertItemBefore",
value: function insertItemBefore(newItem, index) {
this._checkPathSynchronizedToList(); // Spec: If the index is greater than or equal to numberOfItems, then the new item is appended to the end of the list.
if (index > this.numberOfItems) {
index = this.numberOfItems;
}
if (newItem._owningPathSegList) {
// SVG2 spec says to make a copy.
newItem = newItem.clone();
}
this._list.splice(index, 0, newItem);
newItem._owningPathSegList = this;
this._writeListToPath();
return newItem;
}
}, {
key: "replaceItem",
value: function replaceItem(newItem, index) {
this._checkPathSynchronizedToList();
if (newItem._owningPathSegList) {
// SVG2 spec says to make a copy.
newItem = newItem.clone();
}
this._checkValidIndex(index);
this._list[index] = newItem;
newItem._owningPathSegList = this;
this._writeListToPath();
return newItem;
}
}, {
key: "removeItem",
value: function removeItem(index) {
this._checkPathSynchronizedToList();
this._checkValidIndex(index);
var item = this._list[index];
this._list.splice(index, 1);
this._writeListToPath();
return item;
}
}, {
key: "appendItem",
value: function appendItem(newItem) {
this._checkPathSynchronizedToList();
if (newItem._owningPathSegList) {
// SVG2 spec says to make a copy.
newItem = newItem.clone();
}
this._list.push(newItem);
newItem._owningPathSegList = this; // TODO: Optimize this to just append to the existing attribute.
this._writeListToPath();
return newItem;
} // This closely follows SVGPathParser::parsePath from Source/core/svg/SVGPathParser.cpp.
}, {
key: "_parsePath",
value: function _parsePath(string) {
if (!string || !string.length) {
return [];
}
var owningPathSegList = this; // eslint-disable-line consistent-this
var Builder = /*#__PURE__*/function () {
function Builder() {
_classCallCheck(this, Builder);
this.pathSegList = [];
}
_createClass(Builder, [{
key: "appendSegment",
value: function appendSegment(pathSeg) {
this.pathSegList.push(pathSeg);
}
}]);
return Builder;
}();
var Source = /*#__PURE__*/function () {
function Source(string) {
_classCallCheck(this, Source);
this._string = string;
this._currentIndex = 0;
this._endIndex = this._string.length;
this._previousCommand = SVGPathSeg.PATHSEG_UNKNOWN;
this._skipOptionalSpaces();
}
_createClass(Source, [{
key: "_isCurrentSpace",
value: function _isCurrentSpace() {
var character = this._string[this._currentIndex];
return character <= ' ' && (character === ' ' || character === '\n' || character === '\t' || character === '\r' || character === '\f');
}
}, {
key: "_skipOptionalSpaces",
value: function _skipOptionalSpaces() {
while (this._currentIndex < this._endIndex && this._isCurrentSpace()) {
this._currentIndex++;
}
return this._currentIndex < this._endIndex;
}
}, {
key: "_skipOptionalSpacesOrDelimiter",
value: function _skipOptionalSpacesOrDelimiter() {
if (this._currentIndex < this._endIndex && !this._isCurrentSpace() && this._string.charAt(this._currentIndex) !== ',') {
return false;
}
if (this._skipOptionalSpaces()) {
if (this._currentIndex < this._endIndex && this._string.charAt(this._currentIndex) === ',') {
this._currentIndex++;
this._skipOptionalSpaces();
}
}
return this._currentIndex < this._endIndex;
}
}, {
key: "hasMoreData",
value: function hasMoreData() {
return this._currentIndex < this._endIndex;
}
}, {
key: "peekSegmentType",
value: function peekSegmentType() {
var lookahead = this._string[this._currentIndex];
return this._pathSegTypeFromChar(lookahead);
}
}, {
key: "_pathSegTypeFromChar",
value: function _pathSegTypeFromChar(lookahead) {
switch (lookahead) {
case 'Z':
case 'z':
return SVGPathSeg.PATHSEG_CLOSEPATH;
case 'M':
return SVGPathSeg.PATHSEG_MOVETO_ABS;
case 'm':
return SVGPathSeg.PATHSEG_MOVETO_REL;
case 'L':
return SVGPathSeg.PATHSEG_LINETO_ABS;
case 'l':
return SVGPathSeg.PATHSEG_LINETO_REL;
case 'C':
return SVGPathSeg.PATHSEG_CURVETO_CUBIC_ABS;
case 'c':
return SVGPathSeg.PATHSEG_CURVETO_CUBIC_REL;
case 'Q':
return SVGPathSeg.PATHSEG_CURVETO_QUADRATIC_ABS;
case 'q':
return SVGPathSeg.PATHSEG_CURVETO_QUADRATIC_REL;
case 'A':
return SVGPathSeg.PATHSEG_ARC_ABS;
case 'a':
return SVGPathSeg.PATHSEG_ARC_REL;
case 'H':
return SVGPathSeg.PATHSEG_LINETO_HORIZONTAL_ABS;
case 'h':
return SVGPathSeg.PATHSEG_LINETO_HORIZONTAL_REL;
case 'V':
return SVGPathSeg.PATHSEG_LINETO_VERTICAL_ABS;
case 'v':
return SVGPathSeg.PATHSEG_LINETO_VERTICAL_REL;
case 'S':
return SVGPathSeg.PATHSEG_CURVETO_CUBIC_SMOOTH_ABS;
case 's':
return SVGPathSeg.PATHSEG_CURVETO_CUBIC_SMOOTH_REL;
case 'T':
return SVGPathSeg.PATHSEG_CURVETO_QUADRATIC_SMOOTH_ABS;
case 't':
return SVGPathSeg.PATHSEG_CURVETO_QUADRATIC_SMOOTH_REL;
default:
return SVGPathSeg.PATHSEG_UNKNOWN;
}
}
}, {
key: "_nextCommandHelper",
value: function _nextCommandHelper(lookahead, previousCommand) {
// Check for remaining coordinates in the current command.
if ((lookahead === '+' || lookahead === '-' || lookahead === '.' || lookahead >= '0' && lookahead <= '9') && previousCommand !== SVGPathSeg.PATHSEG_CLOSEPATH) {
if (previousCommand === SVGPathSeg.PATHSEG_MOVETO_ABS) {
return SVGPathSeg.PATHSEG_LINETO_ABS;
}
if (previousCommand === SVGPathSeg.PATHSEG_MOVETO_REL) {
return SVGPathSeg.PATHSEG_LINETO_REL;
}
return previousCommand;
}
return SVGPathSeg.PATHSEG_UNKNOWN;
}
}, {
key: "initialCommandIsMoveTo",
value: function initialCommandIsMoveTo() {
// If the path is empty it is still valid, so return true.
if (!this.hasMoreData()) {
return true;
}
var command = this.peekSegmentType(); // Path must start with moveTo.
return command === SVGPathSeg.PATHSEG_MOVETO_ABS || command === SVGPathSeg.PATHSEG_MOVETO_REL;
} // Parse a number from an SVG path. This very closely follows genericParseNumber(...) from Source/core/svg/SVGParserUtilities.cpp.
// Spec: https://www.w3.org/TR/SVG11/single-page.html#paths-PathDataBNF
}, {
key: "_parseNumber",
value: function _parseNumber() {
var exponent = 0;
var integer = 0;
var frac = 1;
var decimal = 0;
var sign = 1;
var expsign = 1;
var startIndex = this._currentIndex;
this._skipOptionalSpaces(); // Read the sign.
if (this._currentIndex < this._endIndex && this._string.charAt(this._currentIndex) === '+') {
this._currentIndex++;
} else if (this._currentIndex < this._endIndex && this._string.charAt(this._currentIndex) === '-') {
this._currentIndex++;
sign = -1;
}
if (this._currentIndex === this._endIndex || (this._string.charAt(this._currentIndex) < '0' || this._string.charAt(this._currentIndex) > '9') && this._string.charAt(this._currentIndex) !== '.') {
// The first character of a number must be one of [0-9+-.].
return undefined;
} // Read the integer part, build right-to-left.
var startIntPartIndex = this._currentIndex;
while (this._currentIndex < this._endIndex && this._string.charAt(this._currentIndex) >= '0' && this._string.charAt(this._currentIndex) <= '9') {
this._currentIndex++; // Advance to first non-digit.
}
if (this._currentIndex !== startIntPartIndex) {
var scanIntPartIndex = this._currentIndex - 1;
var multiplier = 1;
while (scanIntPartIndex >= startIntPartIndex) {
integer += multiplier * (this._string.charAt(scanIntPartIndex--) - '0');
multiplier *= 10;
}
} // Read the decimals.
if (this._currentIndex < this._endIndex && this._string.charAt(this._currentIndex) === '.') {
this._currentIndex++; // There must be a least one digit following the .
if (this._currentIndex >= this._endIndex || this._string.charAt(this._currentIndex) < '0' || this._string.charAt(this._currentIndex) > '9') {
return undefined;
}
while (this._currentIndex < this._endIndex && this._string.charAt(this._currentIndex) >= '0' && this._string.charAt(this._currentIndex) <= '9') {
frac *= 10;
decimal += (this._string.charAt(this._currentIndex) - '0') / frac;
this._currentIndex += 1;
}
} // Read the exponent part.
if (this._currentIndex !== startIndex && this._currentIndex + 1 < this._endIndex && (this._string.charAt(this._currentIndex) === 'e' || this._string.charAt(this._currentIndex) === 'E') && this._string.charAt(this._currentIndex + 1) !== 'x' && this._string.charAt(this._currentIndex + 1) !== 'm') {
this._currentIndex++; // Read the sign of the exponent.
if (this._string.charAt(this._currentIndex) === '+') {
this._currentIndex++;
} else if (this._string.charAt(this._currentIndex) === '-') {
this._currentIndex++;
expsign = -1;
} // There must be an exponent.
if (this._currentIndex >= this._endIndex || this._string.charAt(this._currentIndex) < '0' || this._string.charAt(this._currentIndex) > '9') {
return undefined;
}
while (this._currentIndex < this._endIndex && this._string.charAt(this._currentIndex) >= '0' && this._string.charAt(this._currentIndex) <= '9') {
exponent *= 10;
exponent += this._string.charAt(this._currentIndex) - '0';
this._currentIndex++;
}
}
var number = integer + decimal;
number *= sign;
if (exponent) {
number *= Math.pow(10, expsign * exponent);
}
if (startIndex === this._currentIndex) {
return undefined;
}
this._skipOptionalSpacesOrDelimiter();
return number;
}
}, {
key: "_parseArcFlag",
value: function _parseArcFlag() {
if (this._currentIndex >= this._endIndex) {
return undefined;
}
var flag = false;
var flagChar = this._string.charAt(this._currentIndex++);
if (flagChar === '0') {
flag = false;
} else if (flagChar === '1') {
flag = true;
} else {
return undefined;
}
this._skipOptionalSpacesOrDelimiter();
return flag;
}
}, {
key: "parseSegment",
value: function parseSegment() {
var lookahead = this._string[this._currentIndex];
var command = this._pathSegTypeFromChar(lookahead);
if (command === SVGPathSeg.PATHSEG_UNKNOWN) {
// Possibly an implicit command. Not allowed if this is the first command.
if (this._previousCommand === SVGPathSeg.PATHSEG_UNKNOWN) {
return null;
}
command = this._nextCommandHelper(lookahead, this._previousCommand);
if (command === SVGPathSeg.PATHSEG_UNKNOWN) {
return null;
}
} else {
this._currentIndex++;
}
this._previousCommand = command;
switch (command) {
case SVGPathSeg.PATHSEG_MOVETO_REL:
return new SVGPathSegMovetoRel(owningPathSegList, this._parseNumber(), this._parseNumber());
case SVGPathSeg.PATHSEG_MOVETO_ABS:
return new SVGPathSegMovetoAbs(owningPathSegList, this._parseNumber(), this._parseNumber());
case SVGPathSeg.PATHSEG_LINETO_REL:
return new SVGPathSegLinetoRel(owningPathSegList, this._parseNumber(), this._parseNumber());
case SVGPathSeg.PATHSEG_LINETO_ABS:
return new SVGPathSegLinetoAbs(owningPathSegList, this._parseNumber(), this._parseNumber());
case SVGPathSeg.PATHSEG_LINETO_HORIZONTAL_REL:
return new SVGPathSegLinetoHorizontalRel(owningPathSegList, this._parseNumber());
case SVGPathSeg.PATHSEG_LINETO_HORIZONTAL_ABS:
return new SVGPathSegLinetoHorizontalAbs(owningPathSegList, this._parseNumber());
case SVGPathSeg.PATHSEG_LINETO_VERTICAL_REL:
return new SVGPathSegLinetoVerticalRel(owningPathSegList, this._parseNumber());
case SVGPathSeg.PATHSEG_LINETO_VERTICAL_ABS:
return new SVGPathSegLinetoVerticalAbs(owningPathSegList, this._parseNumber());
case SVGPathSeg.PATHSEG_CLOSEPATH:
this._skipOptionalSpaces();
return new SVGPathSegClosePath(owningPathSegList);
case SVGPathSeg.PATHSEG_CURVETO_CUBIC_REL:
{
var points = {
x1: this._parseNumber(),
y1: this._parseNumber(),
x2: this._parseNumber(),
y2: this._parseNumber(),
x: this._parseNumber(),
y: this._parseNumber()
};
return new SVGPathSegCurvetoCubicRel(owningPathSegList, points.x, points.y, points.x1, points.y1, points.x2, points.y2);
}
case SVGPathSeg.PATHSEG_CURVETO_CUBIC_ABS:
{
var _points = {
x1: this._parseNumber(),
y1: this._parseNumber(),
x2: this._parseNumber(),
y2: this._parseNumber(),
x: this._parseNumber(),
y: this._parseNumber()
};
return new SVGPathSegCurvetoCubicAbs(owningPathSegList, _points.x, _points.y, _points.x1, _points.y1, _points.x2, _points.y2);
}
case SVGPathSeg.PATHSEG_CURVETO_CUBIC_SMOOTH_REL:
{
var _points2 = {
x2: this._parseNumber(),
y2: this._parseNumber(),
x: this._parseNumber(),
y: this._parseNumber()
};
return new SVGPathSegCurvetoCubicSmoothRel(owningPathSegList, _points2.x, _points2.y, _points2.x2, _points2.y2);
}
case SVGPathSeg.PATHSEG_CURVETO_CUBIC_SMOOTH_ABS:
{
var _points3 = {
x2: this._parseNumber(),
y2: this._parseNumber(),
x: this._parseNumber(),
y: this._parseNumber()
};
return new SVGPathSegCurvetoCubicSmoothAbs(owningPathSegList, _points3.x, _points3.y, _points3.x2, _points3.y2);
}
case SVGPathSeg.PATHSEG_CURVETO_QUADRATIC_REL:
{
var _points4 = {
x1: this._parseNumber(),
y1: this._parseNumber(),
x: this._parseNumber(),
y: this._parseNumber()
};
return new SVGPathSegCurvetoQuadraticRel(owningPathSegList, _points4.x, _points4.y, _points4.x1, _points4.y1);
}
case SVGPathSeg.PATHSEG_CURVETO_QUADRATIC_ABS:
{
var _points5 = {
x1: this._parseNumber(),
y1: this._parseNumber(),
x: this._parseNumber(),
y: this._parseNumber()
};
return new SVGPathSegCurvetoQuadraticAbs(owningPathSegList, _points5.x, _points5.y, _points5.x1, _points5.y1);
}
case SVGPathSeg.PATHSEG_CURVETO_QUADRATIC_SMOOTH_REL:
return new SVGPathSegCurvetoQuadraticSmoothRel(owningPathSegList, this._parseNumber(), this._parseNumber());
case SVGPathSeg.PATHSEG_CURVETO_QUADRATIC_SMOOTH_ABS:
return new SVGPathSegCurvetoQuadraticSmoothAbs(owningPathSegList, this._parseNumber(), this._parseNumber());
case SVGPathSeg.PATHSEG_ARC_REL:
{
var _points6 = {
x1: this._parseNumber(),
y1: this._parseNumber(),
arcAngle: this._parseNumber(),
arcLarge: this._parseArcFlag(),
arcSweep: this._parseArcFlag(),
x: this._parseNumber(),
y: this._parseNumber()
};
return new SVGPathSegArcRel(owningPathSegList, _points6.x, _points6.y, _points6.x1, _points6.y1, _points6.arcAngle, _points6.arcLarge, _points6.arcSweep);
}
case SVGPathSeg.PATHSEG_ARC_ABS:
{
var _points7 = {
x1: this._parseNumber(),
y1: this._parseNumber(),
arcAngle: this._parseNumber(),
arcLarge: this._parseArcFlag(),
arcSweep: this._parseArcFlag(),
x: this._parseNumber(),
y: this._parseNumber()
};
return new SVGPathSegArcAbs(owningPathSegList, _points7.x, _points7.y, _points7.x1, _points7.y1, _points7.arcAngle, _points7.arcLarge, _points7.arcSweep);
}
default:
throw new Error('Unknown path seg type.');
}
}
}]);
return Source;
}();
var builder = new Builder();
var source = new Source(string);
if (!source.initialCommandIsMoveTo()) {
return [];
}
while (source.hasMoreData()) {
var pathSeg = source.parseSegment();
if (!pathSeg) {
return [];
}
builder.appendSegment(pathSeg);
}
return builder.pathSegList;
} // STATIC
}], [{
key: "_pathSegArrayAsString",
value: function _pathSegArrayAsString(pathSegArray) {
var string = '';
var first = true;
pathSegArray.forEach(function (pathSeg) {
if (first) {
first = false;
string += pathSeg._asPathString();
} else {
string += ' ' + pathSeg._asPathString();
}
});
return string;
}
}]);
return SVGPathSegList;
}();
SVGPathSegList.prototype.classname = 'SVGPathSegList';
Object.defineProperty(SVGPathSegList.prototype, 'numberOfItems', {
get: function get() {
this._checkPathSynchronizedToList();
return this._list.length;
},
enumerable: true
}); // Add the pathSegList accessors to SVGPathElement.
// Spec: https://www.w3.org/TR/SVG11/single-page.html#paths-InterfaceSVGAnimatedPathData
Object.defineProperties(SVGPathElement.prototype, {
pathSegList: {
get: function get() {
if (!this._pathSegList) {
this._pathSegList = new SVGPathSegList(this);
}
return this._pathSegList;
},
enumerable: true
},
// TODO: The following are not implemented and simply return SVGPathElement.pathSegList.
normalizedPathSegList: {
get: function get() {
return this.pathSegList;
},
enumerable: true
},
animatedPathSegList: {
get: function get() {
return this.pathSegList;
},
enumerable: true
},
animatedNormalizedPathSegList: {
get: function get() {
return this.pathSegList;
},
enumerable: true
}
});
window.SVGPathSegList = SVGPathSegList;
}
})();
/**
* A jQuery module to work with SVG attributes.
* @module jQueryAttr
* @license MIT
*/
/**
* This fixes `$(...).attr()` to work as expected with SVG elements.
* Does not currently use `*AttributeNS()` since we rarely need that.
* Adds {@link external:jQuery.fn.attr}.
* See {@link https://api.jquery.com/attr/} for basic documentation of `.attr()`.
*
* Additional functionality:
* - When getting attributes, a string that's a number is returned as type number.
* - If an array is supplied as the first parameter, multiple values are returned
* as an object with values for each given attribute.
* @function module:jQueryAttr.jQueryAttr
* @param {external:jQuery} $ The jQuery object to which to add the plug-in
* @returns {external:jQuery}
*/
function jQueryPluginSVG($) {
var proxied = $.fn.attr,
svgns = 'http://www.w3.org/2000/svg';
/**
* @typedef {PlainObject<string, string|Float>} module:jQueryAttr.Attributes
*/
/**
* @function external:jQuery.fn.attr
* @param {string|string[]|PlainObject<string, string>} key
* @param {string} value
* @returns {external:jQuery|module:jQueryAttr.Attributes}
*/
$.fn.attr = function (key, value) {
var len = this.length;
if (!len) {
return proxied.call(this, key, value);
}
for (var i = 0; i < len; ++i) {
var elem = this[i]; // set/get SVG attribute
if (elem.namespaceURI === svgns) {
// Setting attribute
if (value !== undefined) {
elem.setAttribute(key, value);
} else if (Array.isArray(key)) {
// Getting attributes from array
var obj = {};
var j = key.length;
while (j--) {
var aname = key[j];
var attr = elem.getAttribute(aname); // This returns a number when appropriate
if (attr || attr === '0') {
attr = isNaN(attr) ? attr : attr - 0;
}
obj[aname] = attr;
}
return obj;
}
if (_typeof(key) === 'object') {
// Setting attributes from object
for (var _i = 0, _Object$entries = Object.entries(key); _i < _Object$entries.length; _i++) {
var _Object$entries$_i = _slicedToArray(_Object$entries[_i], 2),
name = _Object$entries$_i[0],
val = _Object$entries$_i[1];
elem.setAttribute(name, val);
} // Getting attribute
} else {
var _attr = elem.getAttribute(key);
if (_attr || _attr === '0') {
_attr = isNaN(_attr) ? _attr : _attr - 0;
}
return _attr;
}
} else {
return proxied.call(this, key, value);
}
}
return this;
};
return $;
}
/**
* @module jQueryPluginDBox
*/
/**
* @param {external:jQuery} $
* @param {PlainObject} [strings]
* @param {PlainObject} [strings.ok]
* @param {PlainObject} [strings.cancel]
* @returns {external:jQuery}
*/
function jQueryPluginDBox($) {
var strings = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {
ok: 'Ok',
cancel: 'Cancel'
};
// This sets up alternative dialog boxes. They mostly work the same way as
// their UI counterparts, expect instead of returning the result, a callback
// needs to be included that returns the result as its first parameter.
// In the future we may want to add additional types of dialog boxes, since
// they should be easy to handle this way.
$('#dialog_container').draggable({
cancel: '#dialog_content, #dialog_buttons *',
containment: 'window'
}).css('position', 'absolute');
var box = $('#dialog_box'),
btnHolder = $('#dialog_buttons'),
dialogContent = $('#dialog_content');
/**
* @typedef {PlainObject} module:jQueryPluginDBox.PromiseResultObject
* @property {string|true} response
* @property {boolean} checked
*/
/**
* Resolves to `false` (if cancelled), for prompts and selects
* without checkboxes, it resolves to the value of the form control. For other
* types without checkboxes, it resolves to `true`. For checkboxes, it resolves
* to an object with the `response` key containing the same value as the previous
* mentioned (string or `true`) and a `checked` (boolean) property.
* @typedef {Promise<boolean|string|module:jQueryPluginDBox.PromiseResultObject>} module:jQueryPluginDBox.ResultPromise
*/
/**
* @typedef {PlainObject} module:jQueryPluginDBox.SelectOption
* @property {string} text
* @property {string} value
*/
/**
* @typedef {PlainObject} module:jQueryPluginDBox.CheckboxInfo
* @property {string} label Label for the checkbox
* @property {string} value Value of the checkbox
* @property {string} tooltip Tooltip on the checkbox label
* @property {boolean} checked Whether the checkbox is checked by default
*/
/**
* Triggered upon a change of value for the select pull-down.
* @callback module:jQueryPluginDBox.SelectChangeListener
* @returns {void}
*/
/**
* Creates a dialog of the specified type with a given message
* and any defaults and type-specific metadata. Returns a `Promise`
* which resolves differently depending on whether the dialog
* was cancelled or okayed (with the response and any checked state).
* @param {"alert"|"prompt"|"select"|"process"} type
* @param {string} msg
* @param {string} [defaultVal]
* @param {module:jQueryPluginDBox.SelectOption[]} [opts]
* @param {module:jQueryPluginDBox.SelectChangeListener} [changeListener]
* @param {module:jQueryPluginDBox.CheckboxInfo} [checkbox]
* @returns {jQueryPluginDBox.ResultPromise}
*/
function dbox(type, msg, defaultVal, opts, changeListener, checkbox) {
dialogContent.html('<p>' + msg.replace(/\n/g, '</p><p>') + '</p>').toggleClass('prompt', type === 'prompt');
btnHolder.empty();
var ok = $('<input type="button" data-ok="" value="' + strings.ok + '">').appendTo(btnHolder);
return new Promise(function (resolve, reject) {
// eslint-disable-line promise/avoid-new
if (type !== 'alert') {
$('<input type="button" value="' + strings.cancel + '">').appendTo(btnHolder).click(function () {
box.hide();
resolve(false);
});
}
var ctrl, chkbx;
if (type === 'prompt') {
ctrl = $('<input type="text">').prependTo(btnHolder);
ctrl.val(defaultVal || '');
ctrl.bind('keydown', 'return', function () {
ok.click();
});
} else if (type === 'select') {
var div = $('<div style="text-align:center;">');
ctrl = $("<select aria-label=\"".concat(msg, "\">")).appendTo(div);
if (checkbox) {
var label = $('<label>').text(checkbox.label);
chkbx = $('<input type="checkbox">').appendTo(label);
chkbx.val(checkbox.value);
if (checkbox.tooltip) {
label.attr('title', checkbox.tooltip);
}
chkbx.prop('checked', Boolean(checkbox.checked));
div.append($('<div>').append(label));
}
$.each(opts || [], function (opt, val) {
if (_typeof(val) === 'object') {
ctrl.append($('<option>').val(val.value).html(val.text));
} else {
ctrl.append($('<option>').html(val));
}
});
dialogContent.append(div);
if (defaultVal) {
ctrl.val(defaultVal);
}
if (changeListener) {
ctrl.bind('change', 'return', changeListener);
}
ctrl.bind('keydown', 'return', function () {
ok.click();
});
} else if (type === 'process') {
ok.hide();
}
box.show();
ok.click(function () {
box.hide();
var response = type === 'prompt' || type === 'select' ? ctrl.val() : true;
if (chkbx) {
resolve({
response: response,
checked: chkbx.prop('checked')
});
return;
}
resolve(response);
}).focus();
if (type === 'prompt' || type === 'select') {
ctrl.focus();
}
});
}
/**
* @param {string} msg Message to alert
* @returns {jQueryPluginDBox.ResultPromise}
*/
$.alert = function (msg) {
return dbox('alert', msg);
};
/**
* @param {string} msg Message for which to ask confirmation
* @returns {jQueryPluginDBox.ResultPromise}
*/
$.confirm = function (msg) {
return dbox('confirm', msg);
};
/**
* @param {string} msg Message to indicate upon cancelable indicator
* @returns {jQueryPluginDBox.ResultPromise}
*/
$.process_cancel = function (msg) {
return dbox('process', msg);
};
/**
* @param {string} msg Message to accompany the prompt
* @param {string} [defaultText=''] The default text to show for the prompt
* @returns {jQueryPluginDBox.ResultPromise}
*/
$.prompt = function (msg) {
var defaultText = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : '';
return dbox('prompt', msg, defaultText);
};
$.select = function (msg, opts, changeListener, txt, checkbox) {
return dbox('select', msg, txt, opts, changeListener, checkbox);
};
return $;
}
/**
* Namespaces or tools therefor.
* @module namespaces
* @license MIT
*/
/**
* Common namepaces constants in alpha order.
* @enum {string}
* @type {PlainObject}
* @memberof module:namespaces
*/
var NS = {
HTML: 'http://www.w3.org/1999/xhtml',
MATH: 'http://www.w3.org/1998/Math/MathML',
SE: 'http://svg-edit.googlecode.com',
SVG: 'http://www.w3.org/2000/svg',
XLINK: 'http://www.w3.org/1999/xlink',
XML: 'http://www.w3.org/XML/1998/namespace',
XMLNS: 'http://www.w3.org/2000/xmlns/' // see http://www.w3.org/TR/REC-xml-names/#xmlReserved
};
/**
* @function module:namespaces.getReverseNS
* @returns {string} The NS with key values switched and lowercase
*/
var getReverseNS = function getReverseNS() {
var reverseNS = {};
Object.entries(NS).forEach(function (_ref) {
var _ref2 = _slicedToArray(_ref, 2),
name = _ref2[0],
URI = _ref2[1];
reverseNS[URI] = name.toLowerCase();
});
return reverseNS;
};
var $ = jQuery;
var supportsSVG_ = function () {
return Boolean(document.createElementNS && document.createElementNS(NS.SVG, 'svg').createSVGRect);
}();
var _navigator = navigator,
userAgent = _navigator.userAgent;
var svg = document.createElementNS(NS.SVG, 'svg'); // Note: Browser sniffing should only be used if no other detection method is possible
var isOpera_ = Boolean(window.opera);
var isWebkit_ = userAgent.includes('AppleWebKit');
var isGecko_ = userAgent.includes('Gecko/');
var isIE_ = userAgent.includes('MSIE');
var isChrome_ = userAgent.includes('Chrome/');
var isWindows_ = userAgent.includes('Windows');
var isMac_ = userAgent.includes('Macintosh');
var isTouch_ = 'ontouchstart' in window;
var supportsSelectors_ = function () {
return Boolean(svg.querySelector);
}();
var supportsXpath_ = function () {
return Boolean(document.evaluate);
}(); // segList functions (for FF1.5 and 2.0)
var supportsPathReplaceItem_ = function () {
var path = document.createElementNS(NS.SVG, 'path');
path.setAttribute('d', 'M0,0 10,10');
var seglist = path.pathSegList;
var seg = path.createSVGPathSegLinetoAbs(5, 5);
try {
seglist.replaceItem(seg, 1);
return true;
} catch (err) {}
return false;
}();
var supportsPathInsertItemBefore_ = function () {
var path = document.createElementNS(NS.SVG, 'path');
path.setAttribute('d', 'M0,0 10,10');
var seglist = path.pathSegList;
var seg = path.createSVGPathSegLinetoAbs(5, 5);
try {
seglist.insertItemBefore(seg, 1);
return true;
} catch (err) {}
return false;
}(); // text character positioning (for IE9 and now Chrome)
var supportsGoodTextCharPos_ = function () {
var svgroot = document.createElementNS(NS.SVG, 'svg');
var svgcontent = document.createElementNS(NS.SVG, 'svg');
document.documentElement.append(svgroot);
svgcontent.setAttribute('x', 5);
svgroot.append(svgcontent);
var text = document.createElementNS(NS.SVG, 'text');
text.textContent = 'a';
svgcontent.append(text);
try {
// Chrome now fails here
var pos = text.getStartPositionOfChar(0).x;
return pos === 0;
} catch (err) {
return false;
} finally {
svgroot.remove();
}
}();
var supportsPathBBox_ = function () {
var svgcontent = document.createElementNS(NS.SVG, 'svg');
document.documentElement.append(svgcontent);
var path = document.createElementNS(NS.SVG, 'path');
path.setAttribute('d', 'M0,0 C0,0 10,10 10,0');
svgcontent.append(path);
var bbox = path.getBBox();
svgcontent.remove();
return bbox.height > 4 && bbox.height < 5;
}(); // Support for correct bbox sizing on groups with horizontal/vertical lines
var supportsHVLineContainerBBox_ = function () {
var svgcontent = document.createElementNS(NS.SVG, 'svg');
document.documentElement.append(svgcontent);
var path = document.createElementNS(NS.SVG, 'path');
path.setAttribute('d', 'M0,0 10,0');
var path2 = document.createElementNS(NS.SVG, 'path');
path2.setAttribute('d', 'M5,0 15,0');
var g = document.createElementNS(NS.SVG, 'g');
g.append(path, path2);
svgcontent.append(g);
var bbox = g.getBBox();
svgcontent.remove(); // Webkit gives 0, FF gives 10, Opera (correctly) gives 15
return bbox.width === 15;
}();
var supportsGoodDecimals_ = function () {
// Correct decimals on clone attributes (Opera < 10.5/win/non-en)
var rect = document.createElementNS(NS.SVG, 'rect');
rect.setAttribute('x', 0.1);
var crect = rect.cloneNode(false);
var retValue = !crect.getAttribute('x').includes(',');
if (!retValue) {
// Todo: i18nize or remove
$.alert('NOTE: This version of Opera is known to contain bugs in SVG-edit.\n' + 'Please upgrade to the <a href="http://opera.com">latest version</a> in which the problems have been fixed.');
}
return retValue;
}();
var supportsNonScalingStroke_ = function () {
var rect = document.createElementNS(NS.SVG, 'rect');
rect.setAttribute('style', 'vector-effect:non-scaling-stroke');
return rect.style.vectorEffect === 'non-scaling-stroke';
}();
var supportsNativeSVGTransformLists_ = function () {
var rect = document.createElementNS(NS.SVG, 'rect');
var rxform = rect.transform.baseVal;
var t1 = svg.createSVGTransform();
rxform.appendItem(t1);
var r1 = rxform.getItem(0);
var isSVGTransform = function isSVGTransform(o) {
// https://developer.mozilla.org/en-US/docs/Web/API/SVGTransform
return o && _typeof(o) === 'object' && typeof o.setMatrix === 'function' && 'angle' in o;
};
return isSVGTransform(r1) && isSVGTransform(t1) && r1.type === t1.type && r1.angle === t1.angle && r1.matrix.a === t1.matrix.a && r1.matrix.b === t1.matrix.b && r1.matrix.c === t1.matrix.c && r1.matrix.d === t1.matrix.d && r1.matrix.e === t1.matrix.e && r1.matrix.f === t1.matrix.f;
}(); // Public API
/**
* @function module:browser.isOpera
* @returns {boolean}
*/
var isOpera = function isOpera() {
return isOpera_;
};
/**
* @function module:browser.isWebkit
* @returns {boolean}
*/
var isWebkit = function isWebkit() {
return isWebkit_;
};
/**
* @function module:browser.isGecko
* @returns {boolean}
*/
var isGecko = function isGecko() {
return isGecko_;
};
/**
* @function module:browser.isIE
* @returns {boolean}
*/
var isIE = function isIE() {
return isIE_;
};
/**
* @function module:browser.isChrome
* @returns {boolean}
*/
var isChrome = function isChrome() {
return isChrome_;
};
/**
* @function module:browser.isTouch
* @returns {boolean}
*/
var isTouch = function isTouch() {
return isTouch_;
};
/**
* @function module:browser.supportsSelectors
* @returns {boolean}
*/
var supportsSelectors = function supportsSelectors() {
return supportsSelectors_;
};
/**
* @function module:browser.supportsXpath
* @returns {boolean}
*/
var supportsXpath = function supportsXpath() {
return supportsXpath_;
};
/**
* @function module:browser.supportsPathReplaceItem
* @returns {boolean}
*/
var supportsPathReplaceItem = function supportsPathReplaceItem() {
return supportsPathReplaceItem_;
};
/**
* @function module:browser.supportsPathInsertItemBefore
* @returns {boolean}
*/
var supportsPathInsertItemBefore = function supportsPathInsertItemBefore() {
return supportsPathInsertItemBefore_;
};
/**
* @function module:browser.supportsPathBBox
* @returns {boolean}
*/
var supportsPathBBox = function supportsPathBBox() {
return supportsPathBBox_;
};
/**
* @function module:browser.supportsHVLineContainerBBox
* @returns {boolean}
*/
var supportsHVLineContainerBBox = function supportsHVLineContainerBBox() {
return supportsHVLineContainerBBox_;
};
/**
* @function module:browser.supportsGoodTextCharPos
* @returns {boolean}
*/
var supportsGoodTextCharPos = function supportsGoodTextCharPos() {
return supportsGoodTextCharPos_;
};
/**
* @function module:browser.supportsNonScalingStroke
* @returns {boolean}
*/
var supportsNonScalingStroke = function supportsNonScalingStroke() {
return supportsNonScalingStroke_;
};
/**
* @function module:browser.supportsNativeTransformLists
* @returns {boolean}
*/
var supportsNativeTransformLists = function supportsNativeTransformLists() {
return supportsNativeSVGTransformLists_;
};
var svgroot = document.createElementNS(NS.SVG, 'svg');
/**
* Helper function to convert `SVGTransform` to a string.
* @param {SVGTransform} xform
* @returns {string}
*/
function transformToString(xform) {
var m = xform.matrix;
var text = '';
switch (xform.type) {
case 1:
// MATRIX
text = 'matrix(' + [m.a, m.b, m.c, m.d, m.e, m.f].join(',') + ')';
break;
case 2:
// TRANSLATE
text = 'translate(' + m.e + ',' + m.f + ')';
break;
case 3:
// SCALE
if (m.a === m.d) {
text = 'scale(' + m.a + ')';
} else {
text = 'scale(' + m.a + ',' + m.d + ')';
}
break;
case 4:
{
// ROTATE
var cx = 0;
var cy = 0; // this prevents divide by zero
if (xform.angle !== 0) {
var K = 1 - m.a;
cy = (K * m.f + m.b * m.e) / (K * K + m.b * m.b);
cx = (m.e - m.b * cy) / K;
}
text = 'rotate(' + xform.angle + ' ' + cx + ',' + cy + ')';
break;
}
}
return text;
}
/**
* Map of SVGTransformList objects.
*/
var listMap_ = {};
/**
* @interface module:SVGTransformList.SVGEditTransformList
* @property {Integer} numberOfItems unsigned long
*/
/**
* @function module:SVGTransformList.SVGEditTransformList#clear
* @returns {void}
*/
/**
* @function module:SVGTransformList.SVGEditTransformList#initialize
* @param {SVGTransform} newItem
* @returns {SVGTransform}
*/
/**
* DOES NOT THROW DOMException, INDEX_SIZE_ERR.
* @function module:SVGTransformList.SVGEditTransformList#getItem
* @param {Integer} index unsigned long
* @returns {SVGTransform}
*/
/**
* DOES NOT THROW DOMException, INDEX_SIZE_ERR.
* @function module:SVGTransformList.SVGEditTransformList#insertItemBefore
* @param {SVGTransform} newItem
* @param {Integer} index unsigned long
* @returns {SVGTransform}
*/
/**
* DOES NOT THROW DOMException, INDEX_SIZE_ERR.
* @function module:SVGTransformList.SVGEditTransformList#replaceItem
* @param {SVGTransform} newItem
* @param {Integer} index unsigned long
* @returns {SVGTransform}
*/
/**
* DOES NOT THROW DOMException, INDEX_SIZE_ERR.
* @function module:SVGTransformList.SVGEditTransformList#removeItem
* @param {Integer} index unsigned long
* @returns {SVGTransform}
*/
/**
* @function module:SVGTransformList.SVGEditTransformList#appendItem
* @param {SVGTransform} newItem
* @returns {SVGTransform}
*/
/**
* NOT IMPLEMENTED.
* @ignore
* @function module:SVGTransformList.SVGEditTransformList#createSVGTransformFromMatrix
* @param {SVGMatrix} matrix
* @returns {SVGTransform}
*/
/**
* NOT IMPLEMENTED.
* @ignore
* @function module:SVGTransformList.SVGEditTransformList#consolidate
* @returns {SVGTransform}
*/
/**
* SVGTransformList implementation for Webkit.
* These methods do not currently raise any exceptions.
* These methods also do not check that transforms are being inserted. This is basically
* implementing as much of SVGTransformList that we need to get the job done.
* @implements {module:SVGTransformList.SVGEditTransformList}
*/
var SVGTransformList = /*#__PURE__*/function () {
// eslint-disable-line no-shadow
/**
* @param {Element} elem
* @returns {SVGTransformList}
*/
function SVGTransformList(elem) {
_classCallCheck(this, SVGTransformList);
this._elem = elem || null;
this._xforms = []; // TODO: how do we capture the undo-ability in the changed transform list?
this._update = function () {
var tstr = ''; // /* const concatMatrix = */ svgroot.createSVGMatrix();
for (var i = 0; i < this.numberOfItems; ++i) {
var xform = this._list.getItem(i);
tstr += transformToString(xform) + ' ';
}
this._elem.setAttribute('transform', tstr);
};
this._list = this;
this._init = function () {
var _this = this;
// Transform attribute parser
var str = this._elem.getAttribute('transform');
if (!str) {
return;
} // TODO: Add skew support in future
var re = /\s*((scale|matrix|rotate|translate)\s*\(.*?\))\s*,?\s*/; // const re = /\s*(?<xform>(?:scale|matrix|rotate|translate)\s*\(.*?\))\s*,?\s*/;
var m = true;
while (m) {
m = str.match(re);
str = str.replace(re, '');
if (m && m[1]) {
(function () {
var x = m[1];
var bits = x.split(/\s*\(/);
var name = bits[0];
var valBits = bits[1].match(/\s*(.*?)\s*\)/);
valBits[1] = valBits[1].replace(/(\d)-/g, '$1 -');
var valArr = valBits[1].split(/[, ]+/);
var letters = 'abcdef'.split('');
/*
if (m && m.groups.xform) {
const x = m.groups.xform;
const [name, bits] = x.split(/\s*\(/);
const valBits = bits.match(/\s*(?<nonWhitespace>.*?)\s*\)/);
valBits.groups.nonWhitespace = valBits.groups.nonWhitespace.replace(
/(?<digit>\d)-/g, '$<digit> -'
);
const valArr = valBits.groups.nonWhitespace.split(/[, ]+/);
const letters = [...'abcdef'];
*/
var mtx = svgroot.createSVGMatrix();
Object.values(valArr).forEach(function (item, i) {
valArr[i] = parseFloat(item);
if (name === 'matrix') {
mtx[letters[i]] = valArr[i];
}
});
var xform = svgroot.createSVGTransform();
var fname = 'set' + name.charAt(0).toUpperCase() + name.slice(1);
var values = name === 'matrix' ? [mtx] : valArr;
if (name === 'scale' && values.length === 1) {
values.push(values[0]);
} else if (name === 'translate' && values.length === 1) {
values.push(0);
} else if (name === 'rotate' && values.length === 1) {
values.push(0, 0);
}
xform[fname].apply(xform, _toConsumableArray(values));
_this._list.appendItem(xform);
})();
}
}
};
this._removeFromOtherLists = function (item) {
if (item) {
// Check if this transform is already in a transformlist, and
// remove it if so.
Object.values(listMap_).some(function (tl) {
for (var i = 0, len = tl._xforms.length; i < len; ++i) {
if (tl._xforms[i] === item) {
tl.removeItem(i);
return true;
}
}
return false;
});
}
};
this.numberOfItems = 0;
}
/**
* @returns {void}
*/
_createClass(SVGTransformList, [{
key: "clear",
value: function clear() {
this.numberOfItems = 0;
this._xforms = [];
}
/**
* @param {SVGTransform} newItem
* @returns {void}
*/
}, {
key: "initialize",
value: function initialize(newItem) {
this.numberOfItems = 1;
this._removeFromOtherLists(newItem);
this._xforms = [newItem];
}
/**
* @param {Integer} index unsigned long
* @throws {Error}
* @returns {SVGTransform}
*/
}, {
key: "getItem",
value: function getItem(index) {
if (index < this.numberOfItems && index >= 0) {
return this._xforms[index];
}
var err = new Error('DOMException with code=INDEX_SIZE_ERR');
err.code = 1;
throw err;
}
/**
* @param {SVGTransform} newItem
* @param {Integer} index unsigned long
* @returns {SVGTransform}
*/
}, {
key: "insertItemBefore",
value: function insertItemBefore(newItem, index) {
var retValue = null;
if (index >= 0) {
if (index < this.numberOfItems) {
this._removeFromOtherLists(newItem);
var newxforms = new Array(this.numberOfItems + 1); // TODO: use array copying and slicing
var i;
for (i = 0; i < index; ++i) {
newxforms[i] = this._xforms[i];
}
newxforms[i] = newItem;
for (var j = i + 1; i < this.numberOfItems; ++j, ++i) {
newxforms[j] = this._xforms[i];
}
this.numberOfItems++;
this._xforms = newxforms;
retValue = newItem;
this._list._update();
} else {
retValue = this._list.appendItem(newItem);
}
}
return retValue;
}
/**
* @param {SVGTransform} newItem
* @param {Integer} index unsigned long
* @returns {SVGTransform}
*/
}, {
key: "replaceItem",
value: function replaceItem(newItem, index) {
var retValue = null;
if (index < this.numberOfItems && index >= 0) {
this._removeFromOtherLists(newItem);
this._xforms[index] = newItem;
retValue = newItem;
this._list._update();
}
return retValue;
}
/**
* @param {Integer} index unsigned long
* @throws {Error}
* @returns {SVGTransform}
*/
}, {
key: "removeItem",
value: function removeItem(index) {
if (index < this.numberOfItems && index >= 0) {
var retValue = this._xforms[index];
var newxforms = new Array(this.numberOfItems - 1);
var i;
for (i = 0; i < index; ++i) {
newxforms[i] = this._xforms[i];
}
for (var j = i; j < this.numberOfItems - 1; ++j, ++i) {
newxforms[j] = this._xforms[i + 1];
}
this.numberOfItems--;
this._xforms = newxforms;
this._list._update();
return retValue;
}
var err = new Error('DOMException with code=INDEX_SIZE_ERR');
err.code = 1;
throw err;
}
/**
* @param {SVGTransform} newItem
* @returns {SVGTransform}
*/
}, {
key: "appendItem",
value: function appendItem(newItem) {
this._removeFromOtherLists(newItem);
this._xforms.push(newItem);
this.numberOfItems++;
this._list._update();
return newItem;
}
}]);
return SVGTransformList;
}();
/**
* @function module:SVGTransformList.resetListMap
* @returns {void}
*/
var resetListMap = function resetListMap() {
listMap_ = {};
};
/**
* Removes transforms of the given element from the map.
* @function module:SVGTransformList.removeElementFromListMap
* @param {Element} elem - a DOM Element
* @returns {void}
*/
var removeElementFromListMap = function removeElementFromListMap(elem) {
// eslint-disable-line import/no-mutable-exports
if (elem.id && listMap_[elem.id]) {
delete listMap_[elem.id];
}
};
/**
* Returns an object that behaves like a `SVGTransformList` for the given DOM element.
* @function module:SVGTransformList.getTransformList
* @param {Element} elem - DOM element to get a transformlist from
* @todo The polyfill should have `SVGAnimatedTransformList` and this should use it
* @returns {SVGAnimatedTransformList|SVGTransformList}
*/
var getTransformList = function getTransformList(elem) {
if (!supportsNativeTransformLists()) {
var id = elem.id || 'temp';
var t = listMap_[id];
if (!t || id === 'temp') {
listMap_[id] = new SVGTransformList(elem);
listMap_[id]._init();
t = listMap_[id];
}
return t;
}
if (elem.transform) {
return elem.transform.baseVal;
}
if (elem.gradientTransform) {
return elem.gradientTransform.baseVal;
}
if (elem.patternTransform) {
return elem.patternTransform.baseVal;
}
return null;
};
/**
* Tools for working with units.
* @module units
* @license MIT
*
* @copyright 2010 Alexis Deveria, 2010 Jeff Schiller
*/
var wAttrs = ['x', 'x1', 'cx', 'rx', 'width'];
var hAttrs = ['y', 'y1', 'cy', 'ry', 'height'];
/*
const unitNumMap = {
'%': 2,
em: 3,
ex: 4,
px: 5,
cm: 6,
mm: 7,
in: 8,
pt: 9,
pc: 10
};
*/
// Container of elements.
var elementContainer_; // Stores mapping of unit type to user coordinates.
var typeMap_ = {};
/**
* @interface module:units.ElementContainer
*/
/**
* @function module:units.ElementContainer#getBaseUnit
* @returns {string} The base unit type of the container ('em')
*/
/**
* @function module:units.ElementContainer#getElement
* @returns {?Element} An element in the container given an id
*/
/**
* @function module:units.ElementContainer#getHeight
* @returns {Float} The container's height
*/
/**
* @function module:units.ElementContainer#getWidth
* @returns {Float} The container's width
*/
/**
* @function module:units.ElementContainer#getRoundDigits
* @returns {Integer} The number of digits number should be rounded to
*/
/**
* @typedef {PlainObject} module:units.TypeMap
* @property {Float} em
* @property {Float} ex
* @property {Float} in
* @property {Float} cm
* @property {Float} mm
* @property {Float} pt
* @property {Float} pc
* @property {Integer} px
* @property {0} %
*/
/**
* Initializes this module.
*
* @function module:units.init
* @param {module:units.ElementContainer} elementContainer - An object implementing the ElementContainer interface.
* @returns {void}
*/
var init = function init(elementContainer) {
elementContainer_ = elementContainer; // Get correct em/ex values by creating a temporary SVG.
var svg = document.createElementNS(NS.SVG, 'svg');
document.body.append(svg);
var rect = document.createElementNS(NS.SVG, 'rect');
rect.setAttribute('width', '1em');
rect.setAttribute('height', '1ex');
rect.setAttribute('x', '1in');
svg.append(rect);
var bb = rect.getBBox();
svg.remove();
var inch = bb.x;
typeMap_ = {
em: bb.width,
ex: bb.height,
"in": inch,
cm: inch / 2.54,
mm: inch / 25.4,
pt: inch / 72,
pc: inch / 6,
px: 1,
'%': 0
};
};
/**
* Group: Unit conversion functions.
*/
/**
* @function module:units.getTypeMap
* @returns {module:units.TypeMap} The unit object with values for each unit
*/
var getTypeMap = function getTypeMap() {
return typeMap_;
};
/**
* @typedef {GenericArray} module:units.CompareNumbers
* @property {Integer} length 2
* @property {Float} 0
* @property {Float} 1
*/
/**
* Rounds a given value to a float with number of digits defined in
* `round_digits` of `saveOptions`
*
* @function module:units.shortFloat
* @param {string|Float|module:units.CompareNumbers} val - The value (or Array of two numbers) to be rounded
* @returns {Float|string} If a string/number was given, returns a Float. If an array, return a string
* with comma-separated floats
*/
var shortFloat = function shortFloat(val) {
var digits = elementContainer_.getRoundDigits();
if (!isNaN(val)) {
return Number(Number(val).toFixed(digits));
}
if (Array.isArray(val)) {
return shortFloat(val[0]) + ',' + shortFloat(val[1]);
}
return parseFloat(val).toFixed(digits) - 0;
};
/**
* Converts the number to given unit or baseUnit.
* @function module:units.convertUnit
* @param {string|Float} val
* @param {"em"|"ex"|"in"|"cm"|"mm"|"pt"|"pc"|"px"|"%"} [unit]
* @returns {Float}
*/
var convertUnit = function convertUnit(val, unit) {
unit = unit || elementContainer_.getBaseUnit(); // baseVal.convertToSpecifiedUnits(unitNumMap[unit]);
// const val = baseVal.valueInSpecifiedUnits;
// baseVal.convertToSpecifiedUnits(1);
return shortFloat(val / typeMap_[unit]);
};
/**
* Sets an element's attribute based on the unit in its current value.
*
* @function module:units.setUnitAttr
* @param {Element} elem - DOM element to be changed
* @param {string} attr - Name of the attribute associated with the value
* @param {string} val - Attribute value to convert
* @returns {void}
*/
var setUnitAttr = function setUnitAttr(elem, attr, val) {
// if (!isNaN(val)) {
// New value is a number, so check currently used unit
// const oldVal = elem.getAttribute(attr);
// Enable this for alternate mode
// if (oldVal !== null && (isNaN(oldVal) || elementContainer_.getBaseUnit() !== 'px')) {
// // Old value was a number, so get unit, then convert
// let unit;
// if (oldVal.substr(-1) === '%') {
// const res = getResolution();
// unit = '%';
// val *= 100;
// if (wAttrs.includes(attr)) {
// val = val / res.w;
// } else if (hAttrs.includes(attr)) {
// val = val / res.h;
// } else {
// return val / Math.sqrt((res.w*res.w) + (res.h*res.h))/Math.sqrt(2);
// }
// } else {
// if (elementContainer_.getBaseUnit() !== 'px') {
// unit = elementContainer_.getBaseUnit();
// } else {
// unit = oldVal.substr(-2);
// }
// val = val / typeMap_[unit];
// }
//
// val += unit;
// }
// }
elem.setAttribute(attr, val);
};
/**
* Converts given values to numbers. Attributes must be supplied in
* case a percentage is given.
*
* @function module:units.convertToNum
* @param {string} attr - Name of the attribute associated with the value
* @param {string} val - Attribute value to convert
* @returns {Float} The converted number
*/
var convertToNum = function convertToNum(attr, val) {
// Return a number if that's what it already is
if (!isNaN(val)) {
return val - 0;
}
if (val.substr(-1) === '%') {
// Deal with percentage, depends on attribute
var _num = val.substr(0, val.length - 1) / 100;
var width = elementContainer_.getWidth();
var height = elementContainer_.getHeight();
if (wAttrs.includes(attr)) {
return _num * width;
}
if (hAttrs.includes(attr)) {
return _num * height;
}
return _num * Math.sqrt(width * width + height * height) / Math.sqrt(2);
}
var unit = val.substr(-2);
var num = val.substr(0, val.length - 2); // Note that this multiplication turns the string into a number
return num * typeMap_[unit];
};
/**
* Group: Undo/Redo history management.
*/
var HistoryEventTypes = {
BEFORE_APPLY: 'before_apply',
AFTER_APPLY: 'after_apply',
BEFORE_UNAPPLY: 'before_unapply',
AFTER_UNAPPLY: 'after_unapply'
}; // const removedElements = {};
/**
* Base class for commands.
*/
var Command = /*#__PURE__*/function () {
function Command() {
_classCallCheck(this, Command);
}
_createClass(Command, [{
key: "getText",
/**
* @returns {string}
*/
value: function getText() {
return this.text;
}
}]);
return Command;
}(); // Todo: Figure out why the interface members aren't showing
// up (with or without modules applied), despite our apparently following
// http://usejsdoc.org/tags-interface.html#virtual-comments
/**
* An interface that all command objects must implement.
* @interface module:history.HistoryCommand
*/
/**
* Applies.
*
* @function module:history.HistoryCommand#apply
2019-12-31 05:42:39 +00:00
* @param {module:history.HistoryEventHandler} handler
* @fires module:history~Command#event:history
* @returns {void|true}
*/
/**
*
* Unapplies.
* @function module:history.HistoryCommand#unapply
2019-12-31 05:42:39 +00:00
* @param {module:history.HistoryEventHandler} handler
* @fires module:history~Command#event:history
* @returns {void|true}
*/
/**
* Returns the elements.
* @function module:history.HistoryCommand#elements
* @returns {Element[]}
*/
/**
* Gets the text.
* @function module:history.HistoryCommand#getText
* @returns {string}
*/
/**
* Gives the type.
* @function module:history.HistoryCommand.type
* @returns {string}
*/
/**
* Gives the type.
* @function module:history.HistoryCommand#type
* @returns {string}
*/
/**
* @event module:history~Command#event:history
* @type {module:history.HistoryCommand}
*/
/**
* An interface for objects that will handle history events.
* @interface module:history.HistoryEventHandler
*/
/**
*
* @function module:history.HistoryEventHandler#handleHistoryEvent
* @param {string} eventType One of the HistoryEvent types
* @param {module:history~Command#event:history} command
* @listens module:history~Command#event:history
* @returns {void}
*
*/
/**
* History command for an element that had its DOM position changed.
* @implements {module:history.HistoryCommand}
*/
var MoveElementCommand = /*#__PURE__*/function (_Command) {
_inherits(MoveElementCommand, _Command);
/**
* @param {Element} elem - The DOM element that was moved
* @param {Element} oldNextSibling - The element's next sibling before it was moved
* @param {Element} oldParent - The element's parent before it was moved
* @param {string} [text] - An optional string visible to user related to this change
*/
function MoveElementCommand(elem, oldNextSibling, oldParent, text) {
var _this;
_classCallCheck(this, MoveElementCommand);
_this = _possibleConstructorReturn(this, _getPrototypeOf(MoveElementCommand).call(this));
_this.elem = elem;
_this.text = text ? 'Move ' + elem.tagName + ' to ' + text : 'Move ' + elem.tagName;
_this.oldNextSibling = oldNextSibling;
_this.oldParent = oldParent;
_this.newNextSibling = elem.nextSibling;
_this.newParent = elem.parentNode;
return _this;
}
/**
* @returns {"svgedit.history.MoveElementCommand"}
*/
_createClass(MoveElementCommand, [{
key: "type",
value: function type() {
// eslint-disable-line class-methods-use-this
return 'svgedit.history.MoveElementCommand';
}
/**
* Re-positions the element.
* @param {module:history.HistoryEventHandler} handler
* @fires module:history~Command#event:history
* @returns {void}
*/
}, {
key: "apply",
value: function apply(handler) {
// TODO(codedread): Refactor this common event code into a base HistoryCommand class.
if (handler) {
handler.handleHistoryEvent(HistoryEventTypes.BEFORE_APPLY, this);
}
this.elem = this.newParent.insertBefore(this.elem, this.newNextSibling);
if (handler) {
handler.handleHistoryEvent(HistoryEventTypes.AFTER_APPLY, this);
}
}
/**
* Positions the element back to its original location.
* @param {module:history.HistoryEventHandler} handler
* @fires module:history~Command#event:history
* @returns {void}
*/
}, {
key: "unapply",
value: function unapply(handler) {
if (handler) {
handler.handleHistoryEvent(HistoryEventTypes.BEFORE_UNAPPLY, this);
}
this.elem = this.oldParent.insertBefore(this.elem, this.oldNextSibling);
if (handler) {
handler.handleHistoryEvent(HistoryEventTypes.AFTER_UNAPPLY, this);
}
}
/**
* @returns {Element[]} Array with element associated with this command
*/
}, {
key: "elements",
value: function elements() {
return [this.elem];
}
}]);
return MoveElementCommand;
}(Command);
MoveElementCommand.type = MoveElementCommand.prototype.type;
/**
* History command for an element that was added to the DOM.
* @implements {module:history.HistoryCommand}
*/
var InsertElementCommand = /*#__PURE__*/function (_Command2) {
_inherits(InsertElementCommand, _Command2);
/**
* @param {Element} elem - The newly added DOM element
* @param {string} text - An optional string visible to user related to this change
*/
function InsertElementCommand(elem, text) {
var _this2;
_classCallCheck(this, InsertElementCommand);
_this2 = _possibleConstructorReturn(this, _getPrototypeOf(InsertElementCommand).call(this));
_this2.elem = elem;
_this2.text = text || 'Create ' + elem.tagName;
_this2.parent = elem.parentNode;
_this2.nextSibling = _this2.elem.nextSibling;
return _this2;
}
/**
* @returns {"svgedit.history.InsertElementCommand"}
*/
_createClass(InsertElementCommand, [{
key: "type",
value: function type() {
// eslint-disable-line class-methods-use-this
return 'svgedit.history.InsertElementCommand';
}
/**
* Re-inserts the new element.
* @param {module:history.HistoryEventHandler} handler
* @fires module:history~Command#event:history
* @returns {void}
*/
}, {
key: "apply",
value: function apply(handler) {
if (handler) {
handler.handleHistoryEvent(HistoryEventTypes.BEFORE_APPLY, this);
}
this.elem = this.parent.insertBefore(this.elem, this.nextSibling);
if (handler) {
handler.handleHistoryEvent(HistoryEventTypes.AFTER_APPLY, this);
}
}
/**
* Removes the element.
* @param {module:history.HistoryEventHandler} handler
* @fires module:history~Command#event:history
* @returns {void}
*/
}, {
key: "unapply",
value: function unapply(handler) {
if (handler) {
handler.handleHistoryEvent(HistoryEventTypes.BEFORE_UNAPPLY, this);
}
this.parent = this.elem.parentNode;
this.elem = this.elem.parentNode.removeChild(this.elem);
if (handler) {
handler.handleHistoryEvent(HistoryEventTypes.AFTER_UNAPPLY, this);
}
}
/**
* @returns {Element[]} Array with element associated with this command
*/
}, {
key: "elements",
value: function elements() {
return [this.elem];
}
}]);
return InsertElementCommand;
}(Command);
InsertElementCommand.type = InsertElementCommand.prototype.type;
/**
* History command for an element removed from the DOM.
* @implements {module:history.HistoryCommand}
*/
var RemoveElementCommand = /*#__PURE__*/function (_Command3) {
_inherits(RemoveElementCommand, _Command3);
/**
* @param {Element} elem - The removed DOM element
* @param {Node} oldNextSibling - The DOM element's nextSibling when it was in the DOM
* @param {Element} oldParent - The DOM element's parent
* @param {string} [text] - An optional string visible to user related to this change
*/
function RemoveElementCommand(elem, oldNextSibling, oldParent, text) {
var _this3;
_classCallCheck(this, RemoveElementCommand);
_this3 = _possibleConstructorReturn(this, _getPrototypeOf(RemoveElementCommand).call(this));
_this3.elem = elem;
_this3.text = text || 'Delete ' + elem.tagName;
_this3.nextSibling = oldNextSibling;
_this3.parent = oldParent; // special hack for webkit: remove this element's entry in the svgTransformLists map
removeElementFromListMap(elem);
return _this3;
}
/**
* @returns {"svgedit.history.RemoveElementCommand"}
*/
_createClass(RemoveElementCommand, [{
key: "type",
value: function type() {
// eslint-disable-line class-methods-use-this
return 'svgedit.history.RemoveElementCommand';
}
/**
* Re-removes the new element.
* @param {module:history.HistoryEventHandler} handler
* @fires module:history~Command#event:history
* @returns {void}
*/
}, {
key: "apply",
value: function apply(handler) {
if (handler) {
handler.handleHistoryEvent(HistoryEventTypes.BEFORE_APPLY, this);
}
removeElementFromListMap(this.elem);
this.parent = this.elem.parentNode;
this.elem = this.parent.removeChild(this.elem);
if (handler) {
handler.handleHistoryEvent(HistoryEventTypes.AFTER_APPLY, this);
}
}
/**
* Re-adds the new element.
* @param {module:history.HistoryEventHandler} handler
* @fires module:history~Command#event:history
* @returns {void}
*/
}, {
key: "unapply",
value: function unapply(handler) {
if (handler) {
handler.handleHistoryEvent(HistoryEventTypes.BEFORE_UNAPPLY, this);
}
removeElementFromListMap(this.elem);
if (isNullish(this.nextSibling)) {
if (window.console) {
console.log('Error: reference element was lost'); // eslint-disable-line no-console
}
}
this.parent.insertBefore(this.elem, this.nextSibling); // Don't use `before` or `prepend` as `this.nextSibling` may be `null`
if (handler) {
handler.handleHistoryEvent(HistoryEventTypes.AFTER_UNAPPLY, this);
}
}
/**
* @returns {Element[]} Array with element associated with this command
*/
}, {
key: "elements",
value: function elements() {
return [this.elem];
}
}]);
return RemoveElementCommand;
}(Command);
RemoveElementCommand.type = RemoveElementCommand.prototype.type;
/**
* @typedef {"#text"|"#href"|string} module:history.CommandAttributeName
*/
/**
* @typedef {PlainObject<module:history.CommandAttributeName, string>} module:history.CommandAttributes
*/
/**
* History command to make a change to an element.
* Usually an attribute change, but can also be textcontent.
* @implements {module:history.HistoryCommand}
*/
var ChangeElementCommand = /*#__PURE__*/function (_Command4) {
_inherits(ChangeElementCommand, _Command4);
/**
* @param {Element} elem - The DOM element that was changed
* @param {module:history.CommandAttributes} attrs - Attributes to be changed with the values they had *before* the change
* @param {string} text - An optional string visible to user related to this change
*/
function ChangeElementCommand(elem, attrs, text) {
var _this4;
_classCallCheck(this, ChangeElementCommand);
_this4 = _possibleConstructorReturn(this, _getPrototypeOf(ChangeElementCommand).call(this));
_this4.elem = elem;
_this4.text = text ? 'Change ' + elem.tagName + ' ' + text : 'Change ' + elem.tagName;
_this4.newValues = {};
_this4.oldValues = attrs;
for (var attr in attrs) {
if (attr === '#text') {
_this4.newValues[attr] = elem.textContent;
} else if (attr === '#href') {
_this4.newValues[attr] = getHref(elem);
} else {
_this4.newValues[attr] = elem.getAttribute(attr);
}
}
return _this4;
}
/**
* @returns {"svgedit.history.ChangeElementCommand"}
*/
_createClass(ChangeElementCommand, [{
key: "type",
value: function type() {
// eslint-disable-line class-methods-use-this
return 'svgedit.history.ChangeElementCommand';
}
/**
* Performs the stored change action.
* @param {module:history.HistoryEventHandler} handler
* @fires module:history~Command#event:history
* @returns {true}
*/
}, {
key: "apply",
value: function apply(handler) {
var _this5 = this;
if (handler) {
handler.handleHistoryEvent(HistoryEventTypes.BEFORE_APPLY, this);
}
var bChangedTransform = false;
Object.entries(this.newValues).forEach(function (_ref) {
var _ref2 = _slicedToArray(_ref, 2),
attr = _ref2[0],
value = _ref2[1];
if (value) {
if (attr === '#text') {
_this5.elem.textContent = value;
} else if (attr === '#href') {
setHref(_this5.elem, value);
} else {
_this5.elem.setAttribute(attr, value);
}
} else if (attr === '#text') {
_this5.elem.textContent = '';
} else {
_this5.elem.setAttribute(attr, '');
_this5.elem.removeAttribute(attr);
}
if (attr === 'transform') {
bChangedTransform = true;
}
}); // relocate rotational transform, if necessary
if (!bChangedTransform) {
var angle = getRotationAngle(this.elem);
if (angle) {
var bbox = this.elem.getBBox();
var cx = bbox.x + bbox.width / 2,
cy = bbox.y + bbox.height / 2;
var rotate = ['rotate(', angle, ' ', cx, ',', cy, ')'].join('');
if (rotate !== this.elem.getAttribute('transform')) {
this.elem.setAttribute('transform', rotate);
}
}
}
if (handler) {
handler.handleHistoryEvent(HistoryEventTypes.AFTER_APPLY, this);
}
return true;
}
/**
* Reverses the stored change action.
* @param {module:history.HistoryEventHandler} handler
* @fires module:history~Command#event:history
* @returns {true}
*/
}, {
key: "unapply",
value: function unapply(handler) {
var _this6 = this;
if (handler) {
handler.handleHistoryEvent(HistoryEventTypes.BEFORE_UNAPPLY, this);
}
var bChangedTransform = false;
Object.entries(this.oldValues).forEach(function (_ref3) {
var _ref4 = _slicedToArray(_ref3, 2),
attr = _ref4[0],
value = _ref4[1];
if (value) {
if (attr === '#text') {
_this6.elem.textContent = value;
} else if (attr === '#href') {
setHref(_this6.elem, value);
} else {
_this6.elem.setAttribute(attr, value);
}
} else if (attr === '#text') {
_this6.elem.textContent = '';
} else {
_this6.elem.removeAttribute(attr);
}
if (attr === 'transform') {
bChangedTransform = true;
}
}); // relocate rotational transform, if necessary
if (!bChangedTransform) {
var angle = getRotationAngle(this.elem);
if (angle) {
var bbox = this.elem.getBBox();
var cx = bbox.x + bbox.width / 2,
cy = bbox.y + bbox.height / 2;
var rotate = ['rotate(', angle, ' ', cx, ',', cy, ')'].join('');
if (rotate !== this.elem.getAttribute('transform')) {
this.elem.setAttribute('transform', rotate);
}
}
} // Remove transformlist to prevent confusion that causes bugs like 575.
removeElementFromListMap(this.elem);
if (handler) {
handler.handleHistoryEvent(HistoryEventTypes.AFTER_UNAPPLY, this);
}
return true;
}
/**
* @returns {Element[]} Array with element associated with this command
*/
}, {
key: "elements",
value: function elements() {
return [this.elem];
}
}]);
return ChangeElementCommand;
}(Command);
ChangeElementCommand.type = ChangeElementCommand.prototype.type; // TODO: create a 'typing' command object that tracks changes in text
// if a new Typing command is created and the top command on the stack is also a Typing
// and they both affect the same element, then collapse the two commands into one
/**
* History command that can contain/execute multiple other commands.
* @implements {module:history.HistoryCommand}
*/
var BatchCommand = /*#__PURE__*/function (_Command5) {
_inherits(BatchCommand, _Command5);
/**
* @param {string} [text] - An optional string visible to user related to this change
*/
function BatchCommand(text) {
var _this7;
_classCallCheck(this, BatchCommand);
_this7 = _possibleConstructorReturn(this, _getPrototypeOf(BatchCommand).call(this));
_this7.text = text || 'Batch Command';
_this7.stack = [];
return _this7;
}
/**
* @returns {"svgedit.history.BatchCommand"}
*/
_createClass(BatchCommand, [{
key: "type",
value: function type() {
// eslint-disable-line class-methods-use-this
return 'svgedit.history.BatchCommand';
}
/**
* Runs "apply" on all subcommands.
* @param {module:history.HistoryEventHandler} handler
* @fires module:history~Command#event:history
* @returns {void}
*/
}, {
key: "apply",
value: function apply(handler) {
if (handler) {
handler.handleHistoryEvent(HistoryEventTypes.BEFORE_APPLY, this);
}
var len = this.stack.length;
for (var i = 0; i < len; ++i) {
this.stack[i].apply(handler);
}
if (handler) {
handler.handleHistoryEvent(HistoryEventTypes.AFTER_APPLY, this);
}
}
/**
* Runs "unapply" on all subcommands.
* @param {module:history.HistoryEventHandler} handler
* @fires module:history~Command#event:history
* @returns {void}
*/
}, {
key: "unapply",
value: function unapply(handler) {
if (handler) {
handler.handleHistoryEvent(HistoryEventTypes.BEFORE_UNAPPLY, this);
}
for (var i = this.stack.length - 1; i >= 0; i--) {
this.stack[i].unapply(handler);
}
if (handler) {
handler.handleHistoryEvent(HistoryEventTypes.AFTER_UNAPPLY, this);
}
}
/**
* Iterate through all our subcommands.
* @returns {Element[]} All the elements we are changing
*/
}, {
key: "elements",
value: function elements() {
var elems = [];
var cmd = this.stack.length;
while (cmd--) {
var thisElems = this.stack[cmd].elements();
var elem = thisElems.length;
while (elem--) {
if (!elems.includes(thisElems[elem])) {
elems.push(thisElems[elem]);
}
}
}
return elems;
}
/**
* Adds a given command to the history stack.
* @param {Command} cmd - The undo command object to add
* @returns {void}
*/
}, {
key: "addSubCommand",
value: function addSubCommand(cmd) {
this.stack.push(cmd);
}
/**
* @returns {boolean} Indicates whether or not the batch command is empty
*/
}, {
key: "isEmpty",
value: function isEmpty() {
return !this.stack.length;
}
}]);
return BatchCommand;
}(Command);
BatchCommand.type = BatchCommand.prototype.type;
/**
*
*/
var UndoManager = /*#__PURE__*/function () {
/**
* @param {module:history.HistoryEventHandler} historyEventHandler
*/
function UndoManager(historyEventHandler) {
_classCallCheck(this, UndoManager);
this.handler_ = historyEventHandler || null;
this.undoStackPointer = 0;
this.undoStack = []; // this is the stack that stores the original values, the elements and
// the attribute name for begin/finish
this.undoChangeStackPointer = -1;
this.undoableChangeStack = [];
}
/**
* Resets the undo stack, effectively clearing the undo/redo history.
* @returns {void}
*/
_createClass(UndoManager, [{
key: "resetUndoStack",
value: function resetUndoStack() {
this.undoStack = [];
this.undoStackPointer = 0;
}
/**
* @returns {Integer} Current size of the undo history stack
*/
}, {
key: "getUndoStackSize",
value: function getUndoStackSize() {
return this.undoStackPointer;
}
/**
* @returns {Integer} Current size of the redo history stack
*/
}, {
key: "getRedoStackSize",
value: function getRedoStackSize() {
return this.undoStack.length - this.undoStackPointer;
}
/**
* @returns {string} String associated with the next undo command
*/
}, {
key: "getNextUndoCommandText",
value: function getNextUndoCommandText() {
return this.undoStackPointer > 0 ? this.undoStack[this.undoStackPointer - 1].getText() : '';
}
/**
* @returns {string} String associated with the next redo command
*/
}, {
key: "getNextRedoCommandText",
value: function getNextRedoCommandText() {
return this.undoStackPointer < this.undoStack.length ? this.undoStack[this.undoStackPointer].getText() : '';
}
/**
* Performs an undo step.
* @returns {void}
*/
}, {
key: "undo",
value: function undo() {
if (this.undoStackPointer > 0) {
var cmd = this.undoStack[--this.undoStackPointer];
cmd.unapply(this.handler_);
}
}
/**
* Performs a redo step.
* @returns {void}
*/
}, {
key: "redo",
value: function redo() {
if (this.undoStackPointer < this.undoStack.length && this.undoStack.length > 0) {
var cmd = this.undoStack[this.undoStackPointer++];
cmd.apply(this.handler_);
}
}
/**
* Adds a command object to the undo history stack.
* @param {Command} cmd - The command object to add
* @returns {void}
*/
}, {
key: "addCommandToHistory",
value: function addCommandToHistory(cmd) {
// TODO: we MUST compress consecutive text changes to the same element
// (right now each keystroke is saved as a separate command that includes the
// entire text contents of the text element)
// TODO: consider limiting the history that we store here (need to do some slicing)
// if our stack pointer is not at the end, then we have to remove
// all commands after the pointer and insert the new command
if (this.undoStackPointer < this.undoStack.length && this.undoStack.length > 0) {
this.undoStack = this.undoStack.splice(0, this.undoStackPointer);
}
this.undoStack.push(cmd);
this.undoStackPointer = this.undoStack.length;
}
/**
* This function tells the canvas to remember the old values of the
* `attrName` attribute for each element sent in. The elements and values
* are stored on a stack, so the next call to `finishUndoableChange()` will
* pop the elements and old values off the stack, gets the current values
* from the DOM and uses all of these to construct the undo-able command.
* @param {string} attrName - The name of the attribute being changed
* @param {Element[]} elems - Array of DOM elements being changed
* @returns {void}
*/
}, {
key: "beginUndoableChange",
value: function beginUndoableChange(attrName, elems) {
var p = ++this.undoChangeStackPointer;
var i = elems.length;
var oldValues = new Array(i),
elements = new Array(i);
while (i--) {
var elem = elems[i];
if (isNullish(elem)) {
continue;
}
elements[i] = elem;
oldValues[i] = elem.getAttribute(attrName);
}
this.undoableChangeStack[p] = {
attrName: attrName,
oldValues: oldValues,
elements: elements
};
}
/**
* This function returns a `BatchCommand` object which summarizes the
* change since `beginUndoableChange` was called. The command can then
* be added to the command history.
* @returns {BatchCommand} Batch command object with resulting changes
*/
}, {
key: "finishUndoableChange",
value: function finishUndoableChange() {
var p = this.undoChangeStackPointer--;
var changeset = this.undoableChangeStack[p];
var attrName = changeset.attrName;
var batchCmd = new BatchCommand('Change ' + attrName);
var i = changeset.elements.length;
while (i--) {
var elem = changeset.elements[i];
if (isNullish(elem)) {
continue;
}
var changes = {};
changes[attrName] = changeset.oldValues[i];
if (changes[attrName] !== elem.getAttribute(attrName)) {
batchCmd.addSubCommand(new ChangeElementCommand(elem, changes, attrName));
}
}
this.undoableChangeStack[p] = null;
return batchCmd;
}
}]);
return UndoManager;
}();
var hstry = /*#__PURE__*/Object.freeze({
__proto__: null,
HistoryEventTypes: HistoryEventTypes,
MoveElementCommand: MoveElementCommand,
InsertElementCommand: InsertElementCommand,
RemoveElementCommand: RemoveElementCommand,
ChangeElementCommand: ChangeElementCommand,
BatchCommand: BatchCommand,
UndoManager: UndoManager
});
/**
* Mathematical utilities.
* @module math
* @license MIT
*
* @copyright 2010 Alexis Deveria, 2010 Jeff Schiller
*/
var NEAR_ZERO = 1e-14; // Throw away SVGSVGElement used for creating matrices/transforms.
var svg$1 = document.createElementNS(NS.SVG, 'svg');
/**
* A (hopefully) quicker function to transform a point by a matrix
* (this function avoids any DOM calls and just does the math).
* @function module:math.transformPoint
* @param {Float} x - Float representing the x coordinate
* @param {Float} y - Float representing the y coordinate
* @param {SVGMatrix} m - Matrix object to transform the point with
* @returns {module:math.XYObject} An x, y object representing the transformed point
*/
var transformPoint = function transformPoint(x, y, m) {
return {
x: m.a * x + m.c * y + m.e,
y: m.b * x + m.d * y + m.f
};
};
/**
* Helper function to check if the matrix performs no actual transform
* (i.e. exists for identity purposes).
* @function module:math.isIdentity
* @param {SVGMatrix} m - The matrix object to check
* @returns {boolean} Indicates whether or not the matrix is 1,0,0,1,0,0
*/
var isIdentity = function isIdentity(m) {
return m.a === 1 && m.b === 0 && m.c === 0 && m.d === 1 && m.e === 0 && m.f === 0;
};
/**
* This function tries to return a `SVGMatrix` that is the multiplication `m1 * m2`.
* We also round to zero when it's near zero.
* @function module:math.matrixMultiply
* @param {...SVGMatrix} args - Matrix objects to multiply
* @returns {SVGMatrix} The matrix object resulting from the calculation
*/
var matrixMultiply = function matrixMultiply() {
for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) {
args[_key] = arguments[_key];
}
var m = args.reduceRight(function (prev, m1) {
return m1.multiply(prev);
});
if (Math.abs(m.a) < NEAR_ZERO) {
m.a = 0;
}
if (Math.abs(m.b) < NEAR_ZERO) {
m.b = 0;
}
if (Math.abs(m.c) < NEAR_ZERO) {
m.c = 0;
}
if (Math.abs(m.d) < NEAR_ZERO) {
m.d = 0;
}
if (Math.abs(m.e) < NEAR_ZERO) {
m.e = 0;
}
if (Math.abs(m.f) < NEAR_ZERO) {
m.f = 0;
}
return m;
};
var hasMatrixTransform = function hasMatrixTransform(tlist) {
if (!tlist) {
return false;
}
var num = tlist.numberOfItems;
while (num--) {
var xform = tlist.getItem(num);
if (xform.type === 1 && !isIdentity(xform.matrix)) {
return true;
}
}
return false;
};
/**
* @typedef {PlainObject} module:math.TransformedBox An object with the following values
* @property {module:math.XYObject} tl - The top left coordinate
* @property {module:math.XYObject} tr - The top right coordinate
* @property {module:math.XYObject} bl - The bottom left coordinate
* @property {module:math.XYObject} br - The bottom right coordinate
* @property {PlainObject} aabox - Object with the following values:
* @property {Float} aabox.x - Float with the axis-aligned x coordinate
* @property {Float} aabox.y - Float with the axis-aligned y coordinate
* @property {Float} aabox.width - Float with the axis-aligned width coordinate
* @property {Float} aabox.height - Float with the axis-aligned height coordinate
*/
/**
* Transforms a rectangle based on the given matrix.
* @function module:math.transformBox
* @param {Float} l - Float with the box's left coordinate
* @param {Float} t - Float with the box's top coordinate
* @param {Float} w - Float with the box width
* @param {Float} h - Float with the box height
* @param {SVGMatrix} m - Matrix object to transform the box by
* @returns {module:math.TransformedBox}
*/
var transformBox = function transformBox(l, t, w, h, m) {
var tl = transformPoint(l, t, m),
tr = transformPoint(l + w, t, m),
bl = transformPoint(l, t + h, m),
br = transformPoint(l + w, t + h, m),
minx = Math.min(tl.x, tr.x, bl.x, br.x),
maxx = Math.max(tl.x, tr.x, bl.x, br.x),
miny = Math.min(tl.y, tr.y, bl.y, br.y),
maxy = Math.max(tl.y, tr.y, bl.y, br.y);
return {
tl: tl,
tr: tr,
bl: bl,
br: br,
aabox: {
x: minx,
y: miny,
width: maxx - minx,
height: maxy - miny
}
};
};
/**
* This returns a single matrix Transform for a given Transform List
* (this is the equivalent of `SVGTransformList.consolidate()` but unlike
* that method, this one does not modify the actual `SVGTransformList`).
* This function is very liberal with its `min`, `max` arguments.
* @function module:math.transformListToTransform
* @param {SVGTransformList} tlist - The transformlist object
* @param {Integer} [min=0] - Optional integer indicating start transform position
* @param {Integer} [max] - Optional integer indicating end transform position;
* defaults to one less than the tlist's `numberOfItems`
* @returns {SVGTransform} A single matrix transform object
*/
var transformListToTransform = function transformListToTransform(tlist, min, max) {
if (isNullish(tlist)) {
// Or should tlist = null have been prevented before this?
return svg$1.createSVGTransformFromMatrix(svg$1.createSVGMatrix());
}
min = min || 0;
max = max || tlist.numberOfItems - 1;
min = parseInt(min);
max = parseInt(max);
if (min > max) {
var temp = max;
max = min;
min = temp;
}
var m = svg$1.createSVGMatrix();
for (var i = min; i <= max; ++i) {
// if our indices are out of range, just use a harmless identity matrix
var mtom = i >= 0 && i < tlist.numberOfItems ? tlist.getItem(i).matrix : svg$1.createSVGMatrix();
m = matrixMultiply(m, mtom);
}
return svg$1.createSVGTransformFromMatrix(m);
};
/**
* Get the matrix object for a given element.
* @function module:math.getMatrix
* @param {Element} elem - The DOM element to check
* @returns {SVGMatrix} The matrix object associated with the element's transformlist
*/
var getMatrix = function getMatrix(elem) {
var tlist = getTransformList(elem);
return transformListToTransform(tlist).matrix;
};
/**
* Returns a 45 degree angle coordinate associated with the two given
* coordinates.
* @function module:math.snapToAngle
* @param {Integer} x1 - First coordinate's x value
* @param {Integer} y1 - First coordinate's y value
* @param {Integer} x2 - Second coordinate's x value
* @param {Integer} y2 - Second coordinate's y value
* @returns {module:math.AngleCoord45}
*/
var snapToAngle = function snapToAngle(x1, y1, x2, y2) {
var snap = Math.PI / 4; // 45 degrees
var dx = x2 - x1;
var dy = y2 - y1;
var angle = Math.atan2(dy, dx);
var dist = Math.sqrt(dx * dx + dy * dy);
var snapangle = Math.round(angle / snap) * snap;
return {
x: x1 + dist * Math.cos(snapangle),
y: y1 + dist * Math.sin(snapangle),
a: snapangle
};
};
/**
* Check if two rectangles (BBoxes objects) intersect each other.
* @function module:math.rectsIntersect
* @param {SVGRect} r1 - The first BBox-like object
* @param {SVGRect} r2 - The second BBox-like object
* @returns {boolean} True if rectangles intersect
*/
var rectsIntersect = function rectsIntersect(r1, r2) {
return r2.x < r1.x + r1.width && r2.x + r2.width > r1.x && r2.y < r1.y + r1.height && r2.y + r2.height > r1.y;
};
var $$1 = jQuery;
var segData = {
2: ['x', 'y'],
// PATHSEG_MOVETO_ABS
4: ['x', 'y'],
// PATHSEG_LINETO_ABS
6: ['x', 'y', 'x1', 'y1', 'x2', 'y2'],
// PATHSEG_CURVETO_CUBIC_ABS
8: ['x', 'y', 'x1', 'y1'],
// PATHSEG_CURVETO_QUADRATIC_ABS
10: ['x', 'y', 'r1', 'r2', 'angle', 'largeArcFlag', 'sweepFlag'],
// PATHSEG_ARC_ABS
12: ['x'],
// PATHSEG_LINETO_HORIZONTAL_ABS
14: ['y'],
// PATHSEG_LINETO_VERTICAL_ABS
16: ['x', 'y', 'x2', 'y2'],
// PATHSEG_CURVETO_CUBIC_SMOOTH_ABS
18: ['x', 'y'] // PATHSEG_CURVETO_QUADRATIC_SMOOTH_ABS
};
/**
* @tutorial LocaleDocs
* @typedef {module:locale.LocaleStrings|PlainObject} module:path.uiStrings
* @property {PlainObject<string, string>} ui
*/
var uiStrings = {};
/**
* @function module:path.setUiStrings
* @param {module:path.uiStrings} strs
* @returns {void}
*/
var setUiStrings = function setUiStrings(strs) {
Object.assign(uiStrings, strs.ui);
};
var pathFuncs = [];
var linkControlPts = true; // Stores references to paths via IDs.
// TODO: Make this cross-document happy.
var pathData = {};
/**
* @function module:path.setLinkControlPoints
* @param {boolean} lcp
* @returns {void}
*/
var setLinkControlPoints = function setLinkControlPoints(lcp) {
linkControlPts = lcp;
};
/**
* @name module:path.path
* @type {null|module:path.Path}
* @memberof module:path
*/
var path = null; // eslint-disable-line import/no-mutable-exports
var editorContext_ = null;
/**
* @external MouseEvent
*/
/**
* Object with the following keys/values.
* @typedef {PlainObject} module:path.SVGElementJSON
* @property {string} element - Tag name of the SVG element to create
* @property {PlainObject<string, string>} attr - Has key-value attributes to assign to the new element. An `id` should be set so that {@link module:utilities.EditorContext#addSVGElementFromJson} can later re-identify the element for modification or replacement.
* @property {boolean} [curStyles=false] - Indicates whether current style attributes should be applied first
* @property {module:path.SVGElementJSON[]} [children] - Data objects to be added recursively as children
* @property {string} [namespace="http://www.w3.org/2000/svg"] - Indicate a (non-SVG) namespace
*/
/**
* @interface module:path.EditorContext
* @property {module:select.SelectorManager} selectorManager
* @property {module:svgcanvas.SvgCanvas} canvas
*/
/**
* @function module:path.EditorContext#call
* @param {"selected"|"changed"} ev - String with the event name
* @param {module:svgcanvas.SvgCanvas#event:selected|module:svgcanvas.SvgCanvas#event:changed} arg - Argument to pass through to the callback function. If the event is "changed", an array of `Element`s is passed; if "selected", a single-item array of `Element` is passed.
* @returns {void}
*/
/**
* @function module:path.EditorContext#resetD
* @param {SVGPathElement} p
* @returns {void}
*/
/**
* Note: This doesn't round to an integer necessarily.
* @function module:path.EditorContext#round
* @param {Float} val
* @returns {Float} Rounded value to nearest value based on `currentZoom`
*/
/**
* @function module:path.EditorContext#clearSelection
* @param {boolean} [noCall] - When `true`, does not call the "selected" handler
* @returns {void}
*/
/**
* @function module:path.EditorContext#addToSelection
* @param {Element[]} elemsToAdd - An array of DOM elements to add to the selection
* @param {boolean} showGrips - Indicates whether the resize grips should be shown
* @returns {void}
*/
/**
* @function module:path.EditorContext#addCommandToHistory
* @param {Command} cmd
* @returns {void}
*/
/**
* @function module:path.EditorContext#remapElement
* @param {Element} selected - DOM element to be changed
* @param {PlainObject<string, string>} changes - Object with changes to be remapped
* @param {SVGMatrix} m - Matrix object to use for remapping coordinates
* @returns {void}
*/
/**
* @function module:path.EditorContext#addSVGElementFromJson
* @param {module:path.SVGElementJSON} data
* @returns {Element} The new element
*/
/**
* @function module:path.EditorContext#getGridSnapping
* @returns {boolean}
*/
/**
* @function module:path.EditorContext#getOpacity
* @returns {Float}
*/
/**
* @function module:path.EditorContext#getSelectedElements
* @returns {Element[]} the array with selected DOM elements
*/
/**
* @function module:path.EditorContext#getContainer
* @returns {Element}
*/
/**
* @function module:path.EditorContext#setStarted
* @param {boolean} s
* @returns {void}
*/
/**
* @function module:path.EditorContext#getRubberBox
* @returns {SVGRectElement}
*/
/**
* @function module:path.EditorContext#setRubberBox
* @param {SVGRectElement} rb
* @returns {SVGRectElement} Same as parameter passed in
*/
/**
* @function module:path.EditorContext#addPtsToSelection
* @param {PlainObject} cfg
* @param {boolean} cfg.closedSubpath
* @param {SVGCircleElement[]} cfg.grips
* @returns {void}
*/
/**
* @function module:path.EditorContext#endChanges
* @param {PlainObject} cfg
* @param {string} cfg.cmd
* @param {Element} cfg.elem
* @returns {void}
*/
/**
* @function module:path.EditorContext#getCurrentZoom
* @returns {Float} The current zoom level
*/
/**
* Returns the last created DOM element ID string.
* @function module:path.EditorContext#getId
* @returns {string}
*/
/**
* Creates and returns a unique ID string for a DOM element.
* @function module:path.EditorContext#getNextId
* @returns {string}
*/
/**
* Gets the desired element from a mouse event.
* @function module:path.EditorContext#getMouseTarget
* @param {external:MouseEvent} evt - Event object from the mouse event
* @returns {Element} DOM element we want
*/
/**
* @function module:path.EditorContext#getCurrentMode
* @returns {string}
*/
/**
* @function module:path.EditorContext#setCurrentMode
* @param {string} cm The mode
* @returns {string} The same mode as passed in
*/
/**
* @function module:path.EditorContext#getDrawnPath
* @returns {SVGPathElement|null}
*/
/**
* @function module:path.EditorContext#setDrawnPath
* @param {SVGPathElement|null} dp
* @returns {SVGPathElement|null} The same value as passed in
*/
/**
* @function module:path.EditorContext#getSVGRoot
* @returns {SVGSVGElement}
*/
/**
* @function module:path.init
* @param {module:path.EditorContext} editorContext
* @returns {void}
*/
var init$1 = function init(editorContext) {
editorContext_ = editorContext;
pathFuncs = [0, 'ClosePath'];
var pathFuncsStrs = ['Moveto', 'Lineto', 'CurvetoCubic', 'CurvetoQuadratic', 'Arc', 'LinetoHorizontal', 'LinetoVertical', 'CurvetoCubicSmooth', 'CurvetoQuadraticSmooth'];
$$1.each(pathFuncsStrs, function (i, s) {
pathFuncs.push(s + 'Abs');
pathFuncs.push(s + 'Rel');
});
};
/**
* @function module:path.insertItemBefore
* @param {Element} elem
* @param {Segment} newseg
* @param {Integer} index
* @returns {void}
*/
var insertItemBefore = function insertItemBefore(elem, newseg, index) {
// Support insertItemBefore on paths for FF2
var list = elem.pathSegList;
if (supportsPathInsertItemBefore()) {
list.insertItemBefore(newseg, index);
return;
}
var len = list.numberOfItems;
var arr = [];
for (var i = 0; i < len; i++) {
var curSeg = list.getItem(i);
arr.push(curSeg);
}
list.clear();
for (var _i = 0; _i < len; _i++) {
if (_i === index) {
// index + 1
list.appendItem(newseg);
}
list.appendItem(arr[_i]);
}
};
/**
* @function module:path.ptObjToArr
* @todo See if this should just live in `replacePathSeg`
* @param {string} type
* @param {SVGPathSegMovetoAbs|SVGPathSegLinetoAbs|SVGPathSegCurvetoCubicAbs|SVGPathSegCurvetoQuadraticAbs|SVGPathSegArcAbs|SVGPathSegLinetoHorizontalAbs|SVGPathSegLinetoVerticalAbs|SVGPathSegCurvetoCubicSmoothAbs|SVGPathSegCurvetoQuadraticSmoothAbs} segItem
* @returns {ArgumentsArray}
*/
var ptObjToArr = function ptObjToArr(type, segItem) {
var props = segData[type];
return props.map(function (prop) {
return segItem[prop];
});
};
/**
* @function module:path.getGripPt
* @param {Segment} seg
* @param {module:math.XYObject} altPt
* @returns {module:math.XYObject}
*/
var getGripPt = function getGripPt(seg, altPt) {
var pth = seg.path;
var out = {
x: altPt ? altPt.x : seg.item.x,
y: altPt ? altPt.y : seg.item.y
};
if (pth.matrix) {
var pt = transformPoint(out.x, out.y, pth.matrix);
out = pt;
}
var currentZoom = editorContext_.getCurrentZoom();
out.x *= currentZoom;
out.y *= currentZoom;
return out;
};
/**
* @function module:path.getPointFromGrip
* @param {module:math.XYObject} pt
* @param {module:path.Path} pth
* @returns {module:math.XYObject}
*/
var getPointFromGrip = function getPointFromGrip(pt, pth) {
var out = {
x: pt.x,
y: pt.y
};
if (pth.matrix) {
pt = transformPoint(out.x, out.y, pth.imatrix);
out.x = pt.x;
out.y = pt.y;
}
var currentZoom = editorContext_.getCurrentZoom();
out.x /= currentZoom;
out.y /= currentZoom;
return out;
};
/**
* Requires prior call to `setUiStrings` if `xlink:title`
* to be set on the grip.
* @function module:path.addPointGrip
* @param {Integer} index
* @param {Integer} x
* @param {Integer} y
* @returns {SVGCircleElement}
*/
var addPointGrip = function addPointGrip(index, x, y) {
// create the container of all the point grips
var pointGripContainer = getGripContainer();
var pointGrip = getElem('pathpointgrip_' + index); // create it
if (!pointGrip) {
pointGrip = document.createElementNS(NS.SVG, 'circle');
var atts = {
id: 'pathpointgrip_' + index,
display: 'none',
r: 4,
fill: '#0FF',
stroke: '#00F',
'stroke-width': 2,
cursor: 'move',
style: 'pointer-events:all'
};
if ('pathNodeTooltip' in uiStrings) {
// May be empty if running path.js without svg-editor
atts['xlink:title'] = uiStrings.pathNodeTooltip;
}
assignAttributes(pointGrip, atts);
pointGrip = pointGripContainer.appendChild(pointGrip);
var grip = $$1('#pathpointgrip_' + index);
grip.dblclick(function () {
if (path) {
path.setSegType();
}
});
}
if (x && y) {
// set up the point grip element and display it
assignAttributes(pointGrip, {
cx: x,
cy: y,
display: 'inline'
});
}
return pointGrip;
};
/**
* @function module:path.getGripContainer
* @returns {Element}
*/
var getGripContainer = function getGripContainer() {
var c = getElem('pathpointgrip_container');
if (!c) {
var parentElement = getElem('selectorParentGroup');
c = parentElement.appendChild(document.createElementNS(NS.SVG, 'g'));
c.id = 'pathpointgrip_container';
}
return c;
};
/**
* Requires prior call to `setUiStrings` if `xlink:title`
* to be set on the grip.
* @function module:path.addCtrlGrip
* @param {string} id
* @returns {SVGCircleElement}
*/
var addCtrlGrip = function addCtrlGrip(id) {
var pointGrip = getElem('ctrlpointgrip_' + id);
if (pointGrip) {
return pointGrip;
}
pointGrip = document.createElementNS(NS.SVG, 'circle');
var atts = {
id: 'ctrlpointgrip_' + id,
display: 'none',
r: 4,
fill: '#0FF',
stroke: '#55F',
'stroke-width': 1,
cursor: 'move',
style: 'pointer-events:all'
};
if ('pathCtrlPtTooltip' in uiStrings) {
// May be empty if running path.js without svg-editor
atts['xlink:title'] = uiStrings.pathCtrlPtTooltip;
}
assignAttributes(pointGrip, atts);
getGripContainer().append(pointGrip);
return pointGrip;
};
/**
* @function module:path.getCtrlLine
* @param {string} id
* @returns {SVGLineElement}
*/
var getCtrlLine = function getCtrlLine(id) {
var ctrlLine = getElem('ctrlLine_' + id);
if (ctrlLine) {
return ctrlLine;
}
ctrlLine = document.createElementNS(NS.SVG, 'line');
assignAttributes(ctrlLine, {
id: 'ctrlLine_' + id,
stroke: '#555',
'stroke-width': 1,
style: 'pointer-events:none'
});
getGripContainer().append(ctrlLine);
return ctrlLine;
};
/**
* @function module:path.getPointGrip
* @param {Segment} seg
* @param {boolean} update
* @returns {SVGCircleElement}
*/
var getPointGrip = function getPointGrip(seg, update) {
var index = seg.index;
var pointGrip = addPointGrip(index);
if (update) {
var pt = getGripPt(seg);
assignAttributes(pointGrip, {
cx: pt.x,
cy: pt.y,
display: 'inline'
});
}
return pointGrip;
};
/**
* @function module:path.getControlPoints
* @param {Segment} seg
* @returns {PlainObject<string, SVGLineElement|SVGCircleElement>}
*/
var getControlPoints = function getControlPoints(seg) {
var item = seg.item,
index = seg.index;
if (!('x1' in item) || !('x2' in item)) {
return null;
}
var cpt = {};
/* const pointGripContainer = */
getGripContainer(); // Note that this is intentionally not seg.prev.item
var prev = path.segs[index - 1].item;
var segItems = [prev, item];
for (var i = 1; i < 3; i++) {
var id = index + 'c' + i;
var ctrlLine = cpt['c' + i + '_line'] = getCtrlLine(id);
var pt = getGripPt(seg, {
x: item['x' + i],
y: item['y' + i]
});
var gpt = getGripPt(seg, {
x: segItems[i - 1].x,
y: segItems[i - 1].y
});
assignAttributes(ctrlLine, {
x1: pt.x,
y1: pt.y,
x2: gpt.x,
y2: gpt.y,
display: 'inline'
});
cpt['c' + i + '_line'] = ctrlLine; // create it
var pointGrip = cpt['c' + i] = addCtrlGrip(id);
assignAttributes(pointGrip, {
cx: pt.x,
cy: pt.y,
display: 'inline'
});
cpt['c' + i] = pointGrip;
}
return cpt;
};
/**
* This replaces the segment at the given index. Type is given as number.
* @function module:path.replacePathSeg
* @param {Integer} type Possible values set during {@link module:path.init}
* @param {Integer} index
* @param {ArgumentsArray} pts
* @param {SVGPathElement} elem
* @returns {void}
*/
var replacePathSeg = function replacePathSeg(type, index, pts, elem) {
var pth = elem || path.elem;
var func = 'createSVGPathSeg' + pathFuncs[type];
var seg = pth[func].apply(pth, _toConsumableArray(pts));
if (supportsPathReplaceItem()) {
pth.pathSegList.replaceItem(seg, index);
} else {
var segList = pth.pathSegList;
var len = segList.numberOfItems;
var arr = [];
for (var i = 0; i < len; i++) {
var curSeg = segList.getItem(i);
arr.push(curSeg);
}
segList.clear();
for (var _i2 = 0; _i2 < len; _i2++) {
if (_i2 === index) {
segList.appendItem(seg);
} else {
segList.appendItem(arr[_i2]);
}
}
}
};
/**
* @function module:path.getSegSelector
* @param {Segment} seg
* @param {boolean} update
* @returns {SVGPathElement}
*/
var getSegSelector = function getSegSelector(seg, update) {
var index = seg.index;
var segLine = getElem('segline_' + index);
if (!segLine) {
var pointGripContainer = getGripContainer(); // create segline
segLine = document.createElementNS(NS.SVG, 'path');
assignAttributes(segLine, {
id: 'segline_' + index,
display: 'none',
fill: 'none',
stroke: '#0FF',
'stroke-width': 2,
style: 'pointer-events:none',
d: 'M0,0 0,0'
});
pointGripContainer.append(segLine);
}
if (update) {
var prev = seg.prev;
if (!prev) {
segLine.setAttribute('display', 'none');
return segLine;
}
var pt = getGripPt(prev); // Set start point
replacePathSeg(2, 0, [pt.x, pt.y], segLine);
var pts = ptObjToArr(seg.type, seg.item); // , true);
for (var i = 0; i < pts.length; i += 2) {
var point = getGripPt(seg, {
x: pts[i],
y: pts[i + 1]
});
pts[i] = point.x;
pts[i + 1] = point.y;
}
replacePathSeg(seg.type, 1, pts, segLine);
}
return segLine;
};
/**
* @typedef {PlainObject} Point
* @property {Integer} x The x value
* @property {Integer} y The y value
*/
/**
* Takes three points and creates a smoother line based on them.
* @function module:path.smoothControlPoints
* @param {Point} ct1 - Object with x and y values (first control point)
* @param {Point} ct2 - Object with x and y values (second control point)
* @param {Point} pt - Object with x and y values (third point)
* @returns {Point[]} Array of two "smoothed" point objects
*/
var smoothControlPoints = function smoothControlPoints(ct1, ct2, pt) {
// each point must not be the origin
var x1 = ct1.x - pt.x,
y1 = ct1.y - pt.y,
x2 = ct2.x - pt.x,
y2 = ct2.y - pt.y;
if ((x1 !== 0 || y1 !== 0) && (x2 !== 0 || y2 !== 0)) {
var r1 = Math.sqrt(x1 * x1 + y1 * y1),
r2 = Math.sqrt(x2 * x2 + y2 * y2),
nct1 = editorContext_.getSVGRoot().createSVGPoint(),
nct2 = editorContext_.getSVGRoot().createSVGPoint();
var anglea = Math.atan2(y1, x1),
angleb = Math.atan2(y2, x2);
if (anglea < 0) {
anglea += 2 * Math.PI;
}
if (angleb < 0) {
angleb += 2 * Math.PI;
}
var angleBetween = Math.abs(anglea - angleb),
angleDiff = Math.abs(Math.PI - angleBetween) / 2;
var newAnglea, newAngleb;
if (anglea - angleb > 0) {
newAnglea = angleBetween < Math.PI ? anglea + angleDiff : anglea - angleDiff;
newAngleb = angleBetween < Math.PI ? angleb - angleDiff : angleb + angleDiff;
} else {
newAnglea = angleBetween < Math.PI ? anglea - angleDiff : anglea + angleDiff;
newAngleb = angleBetween < Math.PI ? angleb + angleDiff : angleb - angleDiff;
} // rotate the points
nct1.x = r1 * Math.cos(newAnglea) + pt.x;
nct1.y = r1 * Math.sin(newAnglea) + pt.y;
nct2.x = r2 * Math.cos(newAngleb) + pt.x;
nct2.y = r2 * Math.sin(newAngleb) + pt.y;
return [nct1, nct2];
}
return undefined;
};
/**
*
*/
var Segment = /*#__PURE__*/function () {
/**
* @param {Integer} index
* @param {SVGPathSeg} item
* @todo Is `item` be more constrained here?
*/
function Segment(index, item) {
_classCallCheck(this, Segment);
this.selected = false;
this.index = index;
this.item = item;
this.type = item.pathSegType;
this.ctrlpts = [];
this.ptgrip = null;
this.segsel = null;
}
/**
* @param {boolean} y
* @returns {void}
*/
_createClass(Segment, [{
key: "showCtrlPts",
value: function showCtrlPts(y) {
for (var i in this.ctrlpts) {
if ({}.hasOwnProperty.call(this.ctrlpts, i)) {
this.ctrlpts[i].setAttribute('display', y ? 'inline' : 'none');
}
}
}
/**
* @param {boolean} y
* @returns {void}
*/
}, {
key: "selectCtrls",
value: function selectCtrls(y) {
$$1('#ctrlpointgrip_' + this.index + 'c1, #ctrlpointgrip_' + this.index + 'c2').attr('fill', y ? '#0FF' : '#EEE');
}
/**
* @param {boolean} y
* @returns {void}
*/
}, {
key: "show",
value: function show(y) {
if (this.ptgrip) {
this.ptgrip.setAttribute('display', y ? 'inline' : 'none');
this.segsel.setAttribute('display', y ? 'inline' : 'none'); // Show/hide all control points if available
this.showCtrlPts(y);
}
}
/**
* @param {boolean} y
* @returns {void}
*/
}, {
key: "select",
value: function select(y) {
if (this.ptgrip) {
this.ptgrip.setAttribute('stroke', y ? '#0FF' : '#00F');
this.segsel.setAttribute('display', y ? 'inline' : 'none');
if (this.ctrlpts) {
this.selectCtrls(y);
}
this.selected = y;
}
}
/**
* @returns {void}
*/
}, {
key: "addGrip",
value: function addGrip() {
this.ptgrip = getPointGrip(this, true);
this.ctrlpts = getControlPoints(this); // , true);
this.segsel = getSegSelector(this, true);
}
/**
* @param {boolean} full
* @returns {void}
*/
}, {
key: "update",
value: function update(full) {
if (this.ptgrip) {
var pt = getGripPt(this);
assignAttributes(this.ptgrip, {
cx: pt.x,
cy: pt.y
});
getSegSelector(this, true);
if (this.ctrlpts) {
if (full) {
this.item = path.elem.pathSegList.getItem(this.index);
this.type = this.item.pathSegType;
}
getControlPoints(this);
} // this.segsel.setAttribute('display', y ? 'inline' : 'none');
}
}
/**
* @param {Integer} dx
* @param {Integer} dy
* @returns {void}
*/
}, {
key: "move",
value: function move(dx, dy) {
var item = this.item;
var curPts = this.ctrlpts ? [item.x += dx, item.y += dy, item.x1, item.y1, item.x2 += dx, item.y2 += dy] : [item.x += dx, item.y += dy];
replacePathSeg(this.type, this.index, // type 10 means ARC
this.type === 10 ? ptObjToArr(this.type, item) : curPts);
if (this.next && this.next.ctrlpts) {
var next = this.next.item;
var nextPts = [next.x, next.y, next.x1 += dx, next.y1 += dy, next.x2, next.y2];
replacePathSeg(this.next.type, this.next.index, nextPts);
}
if (this.mate) {
// The last point of a closed subpath has a 'mate',
// which is the 'M' segment of the subpath
var itm = this.mate.item;
var pts = [itm.x += dx, itm.y += dy];
replacePathSeg(this.mate.type, this.mate.index, pts); // Has no grip, so does not need 'updating'?
}
this.update(true);
if (this.next) {
this.next.update(true);
}
}
/**
* @param {Integer} num
* @returns {void}
*/
}, {
key: "setLinked",
value: function setLinked(num) {
var seg, anum, pt;
if (num === 2) {
anum = 1;
seg = this.next;
if (!seg) {
return;
}
pt = this.item;
} else {
anum = 2;
seg = this.prev;
if (!seg) {
return;
}
pt = seg.item;
}
var _seg = seg,
item = _seg.item;
item['x' + anum] = pt.x + (pt.x - this.item['x' + num]);
item['y' + anum] = pt.y + (pt.y - this.item['y' + num]);
var pts = [item.x, item.y, item.x1, item.y1, item.x2, item.y2];
replacePathSeg(seg.type, seg.index, pts);
seg.update(true);
}
/**
* @param {Integer} num
* @param {Integer} dx
* @param {Integer} dy
* @returns {void}
*/
}, {
key: "moveCtrl",
value: function moveCtrl(num, dx, dy) {
var item = this.item;
item['x' + num] += dx;
item['y' + num] += dy;
var pts = [item.x, item.y, item.x1, item.y1, item.x2, item.y2];
replacePathSeg(this.type, this.index, pts);
this.update(true);
}
/**
* @param {Integer} newType Possible values set during {@link module:path.init}
* @param {ArgumentsArray} pts
* @returns {void}
*/
}, {
key: "setType",
value: function setType(newType, pts) {
replacePathSeg(newType, this.index, pts);
this.type = newType;
this.item = path.elem.pathSegList.getItem(this.index);
this.showCtrlPts(newType === 6);
this.ctrlpts = getControlPoints(this);
this.update(true);
}
}]);
return Segment;
}();
/**
*
*/
var Path = /*#__PURE__*/function () {
/**
* @param {SVGPathElement} elem
* @throws {Error} If constructed without a path element
*/
function Path(elem) {
_classCallCheck(this, Path);
if (!elem || elem.tagName !== 'path') {
throw new Error('svgedit.path.Path constructed without a <path> element');
}
this.elem = elem;
this.segs = [];
this.selected_pts = [];
path = this; // eslint-disable-line consistent-this
this.init();
}
/**
* Reset path data.
* @returns {module:path.Path}
*/
_createClass(Path, [{
key: "init",
value: function init() {
// Hide all grips, etc
// fixed, needed to work on all found elements, not just first
$$1(getGripContainer()).find('*').each(function () {
$$1(this).attr('display', 'none');
});
var segList = this.elem.pathSegList;
var len = segList.numberOfItems;
this.segs = [];
this.selected_pts = [];
this.first_seg = null; // Set up segs array
for (var i = 0; i < len; i++) {
var item = segList.getItem(i);
var segment = new Segment(i, item);
segment.path = this;
this.segs.push(segment);
}
var segs = this.segs;
var startI = null;
for (var _i3 = 0; _i3 < len; _i3++) {
var seg = segs[_i3];
var nextSeg = _i3 + 1 >= len ? null : segs[_i3 + 1];
var prevSeg = _i3 - 1 < 0 ? null : segs[_i3 - 1];
if (seg.type === 2) {
if (prevSeg && prevSeg.type !== 1) {
// New sub-path, last one is open,
// so add a grip to last sub-path's first point
var startSeg = segs[startI];
startSeg.next = segs[startI + 1];
startSeg.next.prev = startSeg;
startSeg.addGrip();
} // Remember that this is a starter seg
startI = _i3;
} else if (nextSeg && nextSeg.type === 1) {
// This is the last real segment of a closed sub-path
// Next is first seg after "M"
seg.next = segs[startI + 1]; // First seg after "M"'s prev is this
seg.next.prev = seg;
seg.mate = segs[startI];
seg.addGrip();
if (isNullish(this.first_seg)) {
this.first_seg = seg;
}
} else if (!nextSeg) {
if (seg.type !== 1) {
// Last seg, doesn't close so add a grip
// to last sub-path's first point
var _startSeg = segs[startI];
_startSeg.next = segs[startI + 1];
_startSeg.next.prev = _startSeg;
_startSeg.addGrip();
seg.addGrip();
if (!this.first_seg) {
// Open path, so set first as real first and add grip
this.first_seg = segs[startI];
}
}
} else if (seg.type !== 1) {
// Regular segment, so add grip and its "next"
seg.addGrip(); // Don't set its "next" if it's an "M"
if (nextSeg && nextSeg.type !== 2) {
seg.next = nextSeg;
seg.next.prev = seg;
}
}
}
return this;
}
/**
* @callback module:path.PathEachSegCallback
* @this module:path.Segment
* @param {Integer} i The index of the seg being iterated
* @returns {boolean|void} Will stop execution of `eachSeg` if returns `false`
*/
/**
* @param {module:path.PathEachSegCallback} fn
* @returns {void}
*/
}, {
key: "eachSeg",
value: function eachSeg(fn) {
var len = this.segs.length;
for (var i = 0; i < len; i++) {
var ret = fn.call(this.segs[i], i);
if (ret === false) {
break;
}
}
}
/**
* @param {Integer} index
* @returns {void}
*/
}, {
key: "addSeg",
value: function addSeg(index) {
// Adds a new segment
var seg = this.segs[index];
if (!seg.prev) {
return;
}
var prev = seg.prev;
var newseg, newX, newY;
switch (seg.item.pathSegType) {
case 4:
{
newX = (seg.item.x + prev.item.x) / 2;
newY = (seg.item.y + prev.item.y) / 2;
newseg = this.elem.createSVGPathSegLinetoAbs(newX, newY);
break;
}
case 6:
{
// make it a curved segment to preserve the shape (WRS)
// https://en.wikipedia.org/wiki/De_Casteljau%27s_algorithm#Geometric_interpretation
var p0x = (prev.item.x + seg.item.x1) / 2;
var p1x = (seg.item.x1 + seg.item.x2) / 2;
var p2x = (seg.item.x2 + seg.item.x) / 2;
var p01x = (p0x + p1x) / 2;
var p12x = (p1x + p2x) / 2;
newX = (p01x + p12x) / 2;
var p0y = (prev.item.y + seg.item.y1) / 2;
var p1y = (seg.item.y1 + seg.item.y2) / 2;
var p2y = (seg.item.y2 + seg.item.y) / 2;
var p01y = (p0y + p1y) / 2;
var p12y = (p1y + p2y) / 2;
newY = (p01y + p12y) / 2;
newseg = this.elem.createSVGPathSegCurvetoCubicAbs(newX, newY, p0x, p0y, p01x, p01y);
var pts = [seg.item.x, seg.item.y, p12x, p12y, p2x, p2y];
replacePathSeg(seg.type, index, pts);
break;
}
}
insertItemBefore(this.elem, newseg, index);
}
/**
* @param {Integer} index
* @returns {void}
*/
}, {
key: "deleteSeg",
value: function deleteSeg(index) {
var seg = this.segs[index];
var list = this.elem.pathSegList;
seg.show(false);
var next = seg.next;
if (seg.mate) {
// Make the next point be the "M" point
var pt = [next.item.x, next.item.y];
replacePathSeg(2, next.index, pt); // Reposition last node
replacePathSeg(4, seg.index, pt);
list.removeItem(seg.mate.index);
} else if (!seg.prev) {
// First node of open path, make next point the M
// const {item} = seg;
var _pt = [next.item.x, next.item.y];
replacePathSeg(2, seg.next.index, _pt);
list.removeItem(index);
} else {
list.removeItem(index);
}
}
/**
* @param {Integer} index
* @returns {void}
*/
}, {
key: "removePtFromSelection",
value: function removePtFromSelection(index) {
var pos = this.selected_pts.indexOf(index);
if (pos === -1) {
return;
}
this.segs[index].select(false);
this.selected_pts.splice(pos, 1);
}
/**
* @returns {void}
*/
}, {
key: "clearSelection",
value: function clearSelection() {
this.eachSeg(function () {
// 'this' is the segment here
this.select(false);
});
this.selected_pts = [];
}
/**
* @returns {void}
*/
}, {
key: "storeD",
value: function storeD() {
this.last_d = this.elem.getAttribute('d');
}
/**
* @param {Integer} y
* @returns {Path}
*/
}, {
key: "show",
value: function show(y) {
// Shows this path's segment grips
this.eachSeg(function () {
// 'this' is the segment here
this.show(y);
});
if (y) {
this.selectPt(this.first_seg.index);
}
return this;
}
/**
* Move selected points.
* @param {Integer} dx
* @param {Integer} dy
* @returns {void}
*/
}, {
key: "movePts",
value: function movePts(dx, dy) {
var i = this.selected_pts.length;
while (i--) {
var seg = this.segs[this.selected_pts[i]];
seg.move(dx, dy);
}
}
/**
* @param {Integer} dx
* @param {Integer} dy
* @returns {void}
*/
}, {
key: "moveCtrl",
value: function moveCtrl(dx, dy) {
var seg = this.segs[this.selected_pts[0]];
seg.moveCtrl(this.dragctrl, dx, dy);
if (linkControlPts) {
seg.setLinked(this.dragctrl);
}
}
/**
* @param {?Integer} newType See {@link https://www.w3.org/TR/SVG/single-page.html#paths-InterfaceSVGPathSeg}
* @returns {void}
*/
}, {
key: "setSegType",
value: function setSegType(newType) {
this.storeD();
var i = this.selected_pts.length;
var text;
while (i--) {
var selPt = this.selected_pts[i]; // Selected seg
var cur = this.segs[selPt];
var prev = cur.prev;
if (!prev) {
continue;
}
if (!newType) {
// double-click, so just toggle
text = 'Toggle Path Segment Type'; // Toggle segment to curve/straight line
var oldType = cur.type;
newType = oldType === 6 ? 4 : 6;
}
newType = Number(newType);
var curX = cur.item.x;
var curY = cur.item.y;
var prevX = prev.item.x;
var prevY = prev.item.y;
var points = void 0;
switch (newType) {
case 6:
{
if (cur.olditem) {
var old = cur.olditem;
points = [curX, curY, old.x1, old.y1, old.x2, old.y2];
} else {
var diffX = curX - prevX;
var diffY = curY - prevY; // get control points from straight line segment
/*
const ct1x = (prevX + (diffY/2));
const ct1y = (prevY - (diffX/2));
const ct2x = (curX + (diffY/2));
const ct2y = (curY - (diffX/2));
*/
// create control points on the line to preserve the shape (WRS)
var ct1x = prevX + diffX / 3;
var ct1y = prevY + diffY / 3;
var ct2x = curX - diffX / 3;
var ct2y = curY - diffY / 3;
points = [curX, curY, ct1x, ct1y, ct2x, ct2y];
}
break;
}
case 4:
{
points = [curX, curY]; // Store original prevve segment nums
cur.olditem = cur.item;
break;
}
}
cur.setType(newType, points);
}
path.endChanges(text);
}
/**
* @param {Integer} pt
* @param {Integer} ctrlNum
* @returns {void}
*/
}, {
key: "selectPt",
value: function selectPt(pt, ctrlNum) {
this.clearSelection();
if (isNullish(pt)) {
this.eachSeg(function (i) {
// 'this' is the segment here.
if (this.prev) {
pt = i;
}
});
}
this.addPtsToSelection(pt);
if (ctrlNum) {
this.dragctrl = ctrlNum;
if (linkControlPts) {
this.segs[pt].setLinked(ctrlNum);
}
}
}
/**
* Update position of all points.
* @returns {Path}
*/
}, {
key: "update",
value: function update() {
var elem = this.elem;
if (getRotationAngle(elem)) {
this.matrix = getMatrix(elem);
this.imatrix = this.matrix.inverse();
} else {
this.matrix = null;
this.imatrix = null;
}
this.eachSeg(function (i) {
this.item = elem.pathSegList.getItem(i);
this.update();
});
return this;
}
/**
* @param {string} text
* @returns {void}
*/
}, {
key: "endChanges",
value: function endChanges(text) {
if (isWebkit()) {
editorContext_.resetD(this.elem);
}
var cmd = new ChangeElementCommand(this.elem, {
d: this.last_d
}, text);
editorContext_.endChanges({
cmd: cmd,
elem: this.elem
});
}
/**
* @param {Integer|Integer[]} indexes
* @returns {void}
*/
}, {
key: "addPtsToSelection",
value: function addPtsToSelection(indexes) {
var _this = this;
if (!Array.isArray(indexes)) {
indexes = [indexes];
}
indexes.forEach(function (index) {
var seg = _this.segs[index];
if (seg.ptgrip) {
if (!_this.selected_pts.includes(index) && index >= 0) {
_this.selected_pts.push(index);
}
}
});
this.selected_pts.sort();
var i = this.selected_pts.length;
var grips = [];
grips.length = i; // Loop through points to be selected and highlight each
while (i--) {
var pt = this.selected_pts[i];
var seg = this.segs[pt];
seg.select(true);
grips[i] = seg.ptgrip;
}
var closedSubpath = Path.subpathIsClosed(this.selected_pts[0]);
editorContext_.addPtsToSelection({
grips: grips,
closedSubpath: closedSubpath
});
} // STATIC
/**
* @param {Integer} index
* @returns {boolean}
*/
}], [{
key: "subpathIsClosed",
value: function subpathIsClosed(index) {
var clsd = false; // Check if subpath is already open
path.eachSeg(function (i) {
if (i <= index) {
return true;
}
if (this.type === 2) {
// Found M first, so open
return false;
}
if (this.type === 1) {
// Found Z first, so closed
clsd = true;
return false;
}
return true;
});
return clsd;
}
}]);
return Path;
}();
/**
* @function module:path.getPath_
* @param {SVGPathElement} elem
* @returns {module:path.Path}
*/
var getPath_ = function getPath_(elem) {
var p = pathData[elem.id];
if (!p) {
p = pathData[elem.id] = new Path(elem);
}
return p;
};
/**
* @function module:path.removePath_
* @param {string} id
* @returns {void}
*/
var removePath_ = function removePath_(id) {
if (id in pathData) {
delete pathData[id];
}
};
var newcx, newcy, oldcx, oldcy, angle;
var getRotVals = function getRotVals(x, y) {
var dx = x - oldcx;
var dy = y - oldcy; // rotate the point around the old center
var r = Math.sqrt(dx * dx + dy * dy);
var theta = Math.atan2(dy, dx) + angle;
dx = r * Math.cos(theta) + oldcx;
dy = r * Math.sin(theta) + oldcy; // dx,dy should now hold the actual coordinates of each
// point after being rotated
// now we want to rotate them around the new center in the reverse direction
dx -= newcx;
dy -= newcy;
r = Math.sqrt(dx * dx + dy * dy);
theta = Math.atan2(dy, dx) - angle;
return {
x: r * Math.cos(theta) + newcx,
y: r * Math.sin(theta) + newcy
};
}; // If the path was rotated, we must now pay the piper:
// Every path point must be rotated into the rotated coordinate system of
// its old center, then determine the new center, then rotate it back
// This is because we want the path to remember its rotation
/**
* @function module:path.recalcRotatedPath
* @todo This is still using ye olde transform methods, can probably
* be optimized or even taken care of by `recalculateDimensions`
* @returns {void}
*/
var recalcRotatedPath = function recalcRotatedPath() {
var currentPath = path.elem;
angle = getRotationAngle(currentPath, true);
if (!angle) {
return;
} // selectedBBoxes[0] = path.oldbbox;
var oldbox = path.oldbbox; // selectedBBoxes[0],
oldcx = oldbox.x + oldbox.width / 2;
oldcy = oldbox.y + oldbox.height / 2;
var box = getBBox(currentPath);
newcx = box.x + box.width / 2;
newcy = box.y + box.height / 2; // un-rotate the new center to the proper position
var dx = newcx - oldcx,
dy = newcy - oldcy,
r = Math.sqrt(dx * dx + dy * dy),
theta = Math.atan2(dy, dx) + angle;
newcx = r * Math.cos(theta) + oldcx;
newcy = r * Math.sin(theta) + oldcy;
var list = currentPath.pathSegList;
var i = list.numberOfItems;
while (i) {
i -= 1;
var seg = list.getItem(i),
type = seg.pathSegType;
if (type === 1) {
continue;
}
var rvals = getRotVals(seg.x, seg.y),
points = [rvals.x, rvals.y];
if (!isNullish(seg.x1) && !isNullish(seg.x2)) {
var cVals1 = getRotVals(seg.x1, seg.y1);
var cVals2 = getRotVals(seg.x2, seg.y2);
points.splice(points.length, 0, cVals1.x, cVals1.y, cVals2.x, cVals2.y);
}
replacePathSeg(type, i, points);
} // loop for each point
/* box = */
getBBox(currentPath); // selectedBBoxes[0].x = box.x; selectedBBoxes[0].y = box.y;
// selectedBBoxes[0].width = box.width; selectedBBoxes[0].height = box.height;
// now we must set the new transform to be rotated around the new center
var Rnc = editorContext_.getSVGRoot().createSVGTransform(),
tlist = getTransformList(currentPath);
Rnc.setRotate(angle * 180.0 / Math.PI, newcx, newcy);
tlist.replaceItem(Rnc, 0);
}; // ====================================
// Public API starts here
/**
* @function module:path.clearData
* @returns {void}
*/
var clearData = function clearData() {
pathData = {};
}; // Making public for mocking
/**
* @function module:path.reorientGrads
* @param {Element} elem
* @param {SVGMatrix} m
* @returns {void}
*/
var reorientGrads = function reorientGrads(elem, m) {
var bb = getBBox(elem);
for (var i = 0; i < 2; i++) {
var type = i === 0 ? 'fill' : 'stroke';
var attrVal = elem.getAttribute(type);
if (attrVal && attrVal.startsWith('url(')) {
var grad = getRefElem(attrVal);
if (grad.tagName === 'linearGradient') {
var x1 = grad.getAttribute('x1') || 0;
var y1 = grad.getAttribute('y1') || 0;
var x2 = grad.getAttribute('x2') || 1;
var y2 = grad.getAttribute('y2') || 0; // Convert to USOU points
x1 = bb.width * x1 + bb.x;
y1 = bb.height * y1 + bb.y;
x2 = bb.width * x2 + bb.x;
y2 = bb.height * y2 + bb.y; // Transform those points
var pt1 = transformPoint(x1, y1, m);
var pt2 = transformPoint(x2, y2, m); // Convert back to BB points
var gCoords = {
x1: (pt1.x - bb.x) / bb.width,
y1: (pt1.y - bb.y) / bb.height,
x2: (pt2.x - bb.x) / bb.width,
y2: (pt2.y - bb.y) / bb.height
};
var newgrad = grad.cloneNode(true);
$$1(newgrad).attr(gCoords);
newgrad.id = editorContext_.getNextId();
findDefs().append(newgrad);
elem.setAttribute(type, 'url(#' + newgrad.id + ')');
}
}
}
};
/**
* This is how we map paths to our preferred relative segment types.
* @name module:path.pathMap
* @type {GenericArray}
*/
var pathMap = [0, 'z', 'M', 'm', 'L', 'l', 'C', 'c', 'Q', 'q', 'A', 'a', 'H', 'h', 'V', 'v', 'S', 's', 'T', 't'];
/**
* Convert a path to one with only absolute or relative values.
* @todo move to pathActions.js
* @function module:path.convertPath
* @param {SVGPathElement} pth - the path to convert
* @param {boolean} toRel - true of convert to relative
* @returns {string}
*/
var convertPath = function convertPath(pth, toRel) {
var pathSegList = pth.pathSegList;
var len = pathSegList.numberOfItems;
var curx = 0,
cury = 0;
var d = '';
var lastM = null;
for (var i = 0; i < len; ++i) {
var seg = pathSegList.getItem(i); // if these properties are not in the segment, set them to zero
var x = seg.x || 0,
y = seg.y || 0,
x1 = seg.x1 || 0,
y1 = seg.y1 || 0,
x2 = seg.x2 || 0,
y2 = seg.y2 || 0;
var type = seg.pathSegType;
2020-01-25 02:32:24 +00:00
var letter = pathMap[type][toRel ? 'toLowerCase' : 'toUpperCase']();
switch (type) {
case 1:
// z,Z closepath (Z/z)
d += 'z';
if (lastM && !toRel) {
curx = lastM[0];
cury = lastM[1];
}
break;
case 12:
// absolute horizontal line (H)
x -= curx;
// Fallthrough
case 13:
// relative horizontal line (h)
if (toRel) {
2020-01-25 02:32:24 +00:00
y = 0;
curx += x;
letter = 'l';
} else {
2020-01-25 02:32:24 +00:00
y = cury;
x += curx;
curx = x;
letter = 'L';
} // Convert to "line" for easier editing
2020-01-25 02:32:24 +00:00
d += pathDSegment(letter, [[x, y]]);
break;
case 14:
// absolute vertical line (V)
y -= cury;
// Fallthrough
case 15:
// relative vertical line (v)
if (toRel) {
2020-01-25 02:32:24 +00:00
x = 0;
cury += y;
letter = 'l';
} else {
2020-01-25 02:32:24 +00:00
x = curx;
y += cury;
cury = y;
letter = 'L';
} // Convert to "line" for easier editing
2020-01-25 02:32:24 +00:00
d += pathDSegment(letter, [[x, y]]);
break;
case 2: // absolute move (M)
case 4: // absolute line (L)
case 18:
// absolute smooth quad (T)
x -= curx;
y -= cury;
// Fallthrough
case 5: // relative line (l)
case 3: // relative move (m)
case 19:
// relative smooth quad (t)
if (toRel) {
curx += x;
cury += y;
} else {
x += curx;
y += cury;
curx = x;
cury = y;
}
if (type === 2 || type === 3) {
lastM = [curx, cury];
}
d += pathDSegment(letter, [[x, y]]);
break;
case 6:
// absolute cubic (C)
x -= curx;
x1 -= curx;
x2 -= curx;
y -= cury;
y1 -= cury;
y2 -= cury;
// Fallthrough
case 7:
// relative cubic (c)
if (toRel) {
curx += x;
cury += y;
} else {
x += curx;
x1 += curx;
x2 += curx;
y += cury;
y1 += cury;
y2 += cury;
curx = x;
cury = y;
}
d += pathDSegment(letter, [[x1, y1], [x2, y2], [x, y]]);
break;
case 8:
// absolute quad (Q)
x -= curx;
x1 -= curx;
y -= cury;
y1 -= cury;
// Fallthrough
case 9:
// relative quad (q)
if (toRel) {
curx += x;
cury += y;
} else {
x += curx;
x1 += curx;
y += cury;
y1 += cury;
curx = x;
cury = y;
}
d += pathDSegment(letter, [[x1, y1], [x, y]]);
break;
// eslint-disable-next-line sonarjs/no-duplicated-branches
case 10:
// absolute elliptical arc (A)
x -= curx;
y -= cury;
// Fallthrough
case 11:
// relative elliptical arc (a)
if (toRel) {
curx += x;
cury += y;
} else {
x += curx;
y += cury;
curx = x;
cury = y;
}
d += pathDSegment(letter, [[seg.r1, seg.r2]], [seg.angle, seg.largeArcFlag ? 1 : 0, seg.sweepFlag ? 1 : 0], [x, y]);
break;
case 16:
// absolute smooth cubic (S)
x -= curx;
x2 -= curx;
y -= cury;
y2 -= cury;
// Fallthrough
case 17:
// relative smooth cubic (s)
if (toRel) {
curx += x;
cury += y;
} else {
x += curx;
x2 += curx;
y += cury;
y2 += cury;
curx = x;
cury = y;
}
d += pathDSegment(letter, [[x2, y2], [x, y]]);
break;
} // switch on path segment type
} // for each segment
return d;
};
/**
* TODO: refactor callers in `convertPath` to use `getPathDFromSegments` instead of this function.
* Legacy code refactored from `svgcanvas.pathActions.convertPath`.
* @param {string} letter - path segment command (letter in potentially either case from {@link module:path.pathMap}; see [SVGPathSeg#pathSegTypeAsLetter]{@link https://www.w3.org/TR/SVG/single-page.html#paths-__svg__SVGPathSeg__pathSegTypeAsLetter})
* @param {GenericArray<GenericArray<Integer>>} points - x,y points
* @param {GenericArray<GenericArray<Integer>>} [morePoints] - x,y points
* @param {Integer[]} [lastPoint] - x,y point
* @returns {string}
*/
function pathDSegment(letter, points, morePoints, lastPoint) {
$$1.each(points, function (i, pnt) {
points[i] = shortFloat(pnt);
});
var segment = letter + points.join(' ');
if (morePoints) {
segment += ' ' + morePoints.join(' ');
}
if (lastPoint) {
segment += ' ' + shortFloat(lastPoint);
}
return segment;
}
/* eslint-disable jsdoc/require-property */
/**
* Group: Path edit functions.
* Functions relating to editing path elements.
* @namespace {PlainObject} pathActions
* @memberof module:path
*/
var pathActions = function () {
/* eslint-enable jsdoc/require-property */
var subpath = false;
var newPoint, firstCtrl;
var currentPath = null;
var hasMoved = false; // No `editorContext_` yet but should be ok as is `null` by default
// editorContext_.setDrawnPath(null);
/**
* This function converts a polyline (created by the fh_path tool) into
* a path element and coverts every three line segments into a single bezier
* curve in an attempt to smooth out the free-hand.
* @function smoothPolylineIntoPath
* @param {Element} element
* @returns {Element}
*/
var smoothPolylineIntoPath = function smoothPolylineIntoPath(element) {
var i;
var _element = element,
points = _element.points;
var N = points.numberOfItems;
if (N >= 4) {
// loop through every 3 points and convert to a cubic bezier curve segment
//
// NOTE: this is cheating, it means that every 3 points has the potential to
// be a corner instead of treating each point in an equal manner. In general,
// this technique does not look that good.
//
// I am open to better ideas!
//
// Reading:
// - http://www.efg2.com/Lab/Graphics/Jean-YvesQueinecBezierCurves.htm
// - https://www.codeproject.com/KB/graphics/BezierSpline.aspx?msg=2956963
// - https://www.ian-ko.com/ET_GeoWizards/UserGuide/smooth.htm
// - https://www.cs.mtu.edu/~shene/COURSES/cs3621/NOTES/spline/Bezier/bezier-der.html
var curpos = points.getItem(0),
prevCtlPt = null;
var d = [];
d.push(['M', curpos.x, ',', curpos.y, ' C'].join(''));
for (i = 1; i <= N - 4; i += 3) {
var ct1 = points.getItem(i);
var ct2 = points.getItem(i + 1);
var end = points.getItem(i + 2); // if the previous segment had a control point, we want to smooth out
// the control points on both sides
if (prevCtlPt) {
var newpts = smoothControlPoints(prevCtlPt, ct1, curpos);
if (newpts && newpts.length === 2) {
var prevArr = d[d.length - 1].split(',');
prevArr[2] = newpts[0].x;
prevArr[3] = newpts[0].y;
d[d.length - 1] = prevArr.join(',');
ct1 = newpts[1];
}
}
d.push([ct1.x, ct1.y, ct2.x, ct2.y, end.x, end.y].join(','));
curpos = end;
prevCtlPt = ct2;
} // handle remaining line segments
d.push('L');
while (i < N) {
var pt = points.getItem(i);
d.push([pt.x, pt.y].join(','));
i++;
}
d = d.join(' '); // create new path element
element = editorContext_.addSVGElementFromJson({
element: 'path',
curStyles: true,
attr: {
id: editorContext_.getId(),
d: d,
fill: 'none'
}
}); // No need to call "changed", as this is already done under mouseUp
}
return element;
};
return (
/** @lends module:path.pathActions */
{
/**
* @param {MouseEvent} evt
* @param {Element} mouseTarget
* @param {Float} startX
* @param {Float} startY
* @returns {boolean|void}
*/
mouseDown: function mouseDown(evt, mouseTarget, startX, startY) {
var id;
if (editorContext_.getCurrentMode() === 'path') {
var mouseX = startX; // Was this meant to work with the other `mouseX`? (was defined globally so adding `let` to at least avoid a global)
var mouseY = startY; // Was this meant to work with the other `mouseY`? (was defined globally so adding `let` to at least avoid a global)
var currentZoom = editorContext_.getCurrentZoom();
var x = mouseX / currentZoom,
y = mouseY / currentZoom,
stretchy = getElem('path_stretch_line');
newPoint = [x, y];
if (editorContext_.getGridSnapping()) {
x = snapToGrid(x);
y = snapToGrid(y);
mouseX = snapToGrid(mouseX);
mouseY = snapToGrid(mouseY);
}
if (!stretchy) {
stretchy = document.createElementNS(NS.SVG, 'path');
assignAttributes(stretchy, {
id: 'path_stretch_line',
stroke: '#22C',
'stroke-width': '0.5',
fill: 'none'
});
stretchy = getElem('selectorParentGroup').appendChild(stretchy);
}
stretchy.setAttribute('display', 'inline');
var keep = null;
var index; // if pts array is empty, create path element with M at current point
var drawnPath = editorContext_.getDrawnPath();
if (!drawnPath) {
var dAttr = 'M' + x + ',' + y + ' '; // Was this meant to work with the other `dAttr`? (was defined globally so adding `var` to at least avoid a global)
/* drawnPath = */
editorContext_.setDrawnPath(editorContext_.addSVGElementFromJson({
element: 'path',
curStyles: true,
attr: {
d: dAttr,
id: editorContext_.getNextId(),
opacity: editorContext_.getOpacity() / 2
}
})); // set stretchy line to first point
stretchy.setAttribute('d', ['M', mouseX, mouseY, mouseX, mouseY].join(' '));
index = subpath ? path.segs.length : 0;
addPointGrip(index, mouseX, mouseY);
} else {
// determine if we clicked on an existing point
var seglist = drawnPath.pathSegList;
var i = seglist.numberOfItems;
var FUZZ = 6 / currentZoom;
var clickOnPoint = false;
while (i) {
i--;
var item = seglist.getItem(i);
var px = item.x,
py = item.y; // found a matching point
if (x >= px - FUZZ && x <= px + FUZZ && y >= py - FUZZ && y <= py + FUZZ) {
clickOnPoint = true;
break;
}
} // get path element that we are in the process of creating
id = editorContext_.getId(); // Remove previous path object if previously created
removePath_(id);
var newpath = getElem(id);
var newseg;
var sSeg;
var len = seglist.numberOfItems; // if we clicked on an existing point, then we are done this path, commit it
// (i, i+1) are the x,y that were clicked on
if (clickOnPoint) {
// if clicked on any other point but the first OR
// the first point was clicked on and there are less than 3 points
// then leave the path open
// otherwise, close the path
if (i <= 1 && len >= 2) {
// Create end segment
var absX = seglist.getItem(0).x;
var absY = seglist.getItem(0).y;
sSeg = stretchy.pathSegList.getItem(1);
if (sSeg.pathSegType === 4) {
newseg = drawnPath.createSVGPathSegLinetoAbs(absX, absY);
} else {
newseg = drawnPath.createSVGPathSegCurvetoCubicAbs(absX, absY, sSeg.x1 / currentZoom, sSeg.y1 / currentZoom, absX, absY);
}
var endseg = drawnPath.createSVGPathSegClosePath();
seglist.appendItem(newseg);
seglist.appendItem(endseg);
} else if (len < 3) {
keep = false;
return keep;
}
$$1(stretchy).remove(); // This will signal to commit the path
// const element = newpath; // Other event handlers define own `element`, so this was probably not meant to interact with them or one which shares state (as there were none); I therefore adding a missing `var` to avoid a global
/* drawnPath = */
editorContext_.setDrawnPath(null);
editorContext_.setStarted(false);
if (subpath) {
if (path.matrix) {
editorContext_.remapElement(newpath, {}, path.matrix.inverse());
}
var newD = newpath.getAttribute('d');
var origD = $$1(path.elem).attr('d');
$$1(path.elem).attr('d', origD + newD);
$$1(newpath).remove();
if (path.matrix) {
recalcRotatedPath();
}
init$1();
pathActions.toEditMode(path.elem);
path.selectPt();
return false;
} // else, create a new point, update path element
} else {
// Checks if current target or parents are #svgcontent
if (!$$1.contains(editorContext_.getContainer(), editorContext_.getMouseTarget(evt))) {
// Clicked outside canvas, so don't make point
// console.log('Clicked outside canvas');
return false;
}
var num = drawnPath.pathSegList.numberOfItems;
var last = drawnPath.pathSegList.getItem(num - 1);
var lastx = last.x,
lasty = last.y;
if (evt.shiftKey) {
var xya = snapToAngle(lastx, lasty, x, y);
x = xya.x;
y = xya.y;
} // Use the segment defined by stretchy
sSeg = stretchy.pathSegList.getItem(1);
if (sSeg.pathSegType === 4) {
newseg = drawnPath.createSVGPathSegLinetoAbs(editorContext_.round(x), editorContext_.round(y));
} else {
newseg = drawnPath.createSVGPathSegCurvetoCubicAbs(editorContext_.round(x), editorContext_.round(y), sSeg.x1 / currentZoom, sSeg.y1 / currentZoom, sSeg.x2 / currentZoom, sSeg.y2 / currentZoom);
}
drawnPath.pathSegList.appendItem(newseg);
x *= currentZoom;
y *= currentZoom; // set stretchy line to latest point
stretchy.setAttribute('d', ['M', x, y, x, y].join(' '));
index = num;
if (subpath) {
index += path.segs.length;
}
addPointGrip(index, x, y);
} // keep = true;
}
return undefined;
} // TODO: Make sure currentPath isn't null at this point
if (!path) {
return undefined;
}
path.storeD();
id = evt.target.id;
var curPt;
if (id.substr(0, 14) === 'pathpointgrip_') {
// Select this point
curPt = path.cur_pt = parseInt(id.substr(14));
path.dragging = [startX, startY];
var seg = path.segs[curPt]; // only clear selection if shift is not pressed (otherwise, add
// node to selection)
if (!evt.shiftKey) {
if (path.selected_pts.length <= 1 || !seg.selected) {
path.clearSelection();
}
path.addPtsToSelection(curPt);
} else if (seg.selected) {
path.removePtFromSelection(curPt);
} else {
path.addPtsToSelection(curPt);
}
} else if (id.startsWith('ctrlpointgrip_')) {
path.dragging = [startX, startY];
var parts = id.split('_')[1].split('c');
curPt = Number(parts[0]);
var ctrlNum = Number(parts[1]);
path.selectPt(curPt, ctrlNum);
} // Start selection box
if (!path.dragging) {
var rubberBox = editorContext_.getRubberBox();
if (isNullish(rubberBox)) {
rubberBox = editorContext_.setRubberBox(editorContext_.selectorManager.getRubberBandBox());
}
2020-03-22 16:03:13 +00:00
var currentZoom = editorContext_.getCurrentZoom();
assignAttributes(rubberBox, {
2020-03-22 16:03:13 +00:00
x: startX * currentZoom,
y: startY * currentZoom,
width: 0,
height: 0,
display: 'inline'
});
}
return undefined;
},
/**
* @param {Float} mouseX
* @param {Float} mouseY
* @returns {void}
*/
mouseMove: function mouseMove(mouseX, mouseY) {
var currentZoom = editorContext_.getCurrentZoom();
hasMoved = true;
var drawnPath = editorContext_.getDrawnPath();
if (editorContext_.getCurrentMode() === 'path') {
if (!drawnPath) {
return;
}
var seglist = drawnPath.pathSegList;
var index = seglist.numberOfItems - 1;
if (newPoint) {
// First point
// if (!index) { return; }
// Set control points
var pointGrip1 = addCtrlGrip('1c1');
var pointGrip2 = addCtrlGrip('0c2'); // dragging pointGrip1
pointGrip1.setAttribute('cx', mouseX);
pointGrip1.setAttribute('cy', mouseY);
pointGrip1.setAttribute('display', 'inline');
var ptX = newPoint[0];
var ptY = newPoint[1]; // set curve
// const seg = seglist.getItem(index);
var curX = mouseX / currentZoom;
var curY = mouseY / currentZoom;
var altX = ptX + (ptX - curX);
var altY = ptY + (ptY - curY);
pointGrip2.setAttribute('cx', altX * currentZoom);
pointGrip2.setAttribute('cy', altY * currentZoom);
pointGrip2.setAttribute('display', 'inline');
var ctrlLine = getCtrlLine(1);
assignAttributes(ctrlLine, {
x1: mouseX,
y1: mouseY,
x2: altX * currentZoom,
y2: altY * currentZoom,
display: 'inline'
});
if (index === 0) {
firstCtrl = [mouseX, mouseY];
} else {
var last = seglist.getItem(index - 1);
var lastX = last.x;
var lastY = last.y;
if (last.pathSegType === 6) {
lastX += lastX - last.x2;
lastY += lastY - last.y2;
} else if (firstCtrl) {
lastX = firstCtrl[0] / currentZoom;
lastY = firstCtrl[1] / currentZoom;
}
replacePathSeg(6, index, [ptX, ptY, lastX, lastY, altX, altY], drawnPath);
}
} else {
var stretchy = getElem('path_stretch_line');
if (stretchy) {
var prev = seglist.getItem(index);
if (prev.pathSegType === 6) {
var prevX = prev.x + (prev.x - prev.x2);
var prevY = prev.y + (prev.y - prev.y2);
replacePathSeg(6, 1, [mouseX, mouseY, prevX * currentZoom, prevY * currentZoom, mouseX, mouseY], stretchy);
} else if (firstCtrl) {
replacePathSeg(6, 1, [mouseX, mouseY, firstCtrl[0], firstCtrl[1], mouseX, mouseY], stretchy);
} else {
replacePathSeg(4, 1, [mouseX, mouseY], stretchy);
}
}
}
return;
} // if we are dragging a point, let's move it
if (path.dragging) {
var pt = getPointFromGrip({
x: path.dragging[0],
y: path.dragging[1]
}, path);
var mpt = getPointFromGrip({
x: mouseX,
y: mouseY
}, path);
var diffX = mpt.x - pt.x;
var diffY = mpt.y - pt.y;
path.dragging = [mouseX, mouseY];
if (path.dragctrl) {
path.moveCtrl(diffX, diffY);
} else {
path.movePts(diffX, diffY);
}
} else {
path.selected_pts = [];
path.eachSeg(function (i) {
var seg = this; // eslint-disable-line consistent-this
if (!seg.next && !seg.prev) {
return;
} // const {item} = seg;
var rubberBox = editorContext_.getRubberBox();
var rbb = rubberBox.getBBox();
var pt = getGripPt(seg);
var ptBb = {
x: pt.x,
y: pt.y,
width: 0,
height: 0
};
var sel = rectsIntersect(rbb, ptBb);
this.select(sel); // Note that addPtsToSelection is not being run
if (sel) {
path.selected_pts.push(seg.index);
}
});
}
},
/**
* @typedef module:path.keepElement
* @type {PlainObject}
* @property {boolean} keep
* @property {Element} element
*/
/**
* @param {Event} evt
* @param {Element} element
* @param {Float} mouseX
* @param {Float} mouseY
* @returns {module:path.keepElement|void}
*/
mouseUp: function mouseUp(evt, element, mouseX, mouseY) {
var drawnPath = editorContext_.getDrawnPath(); // Create mode
if (editorContext_.getCurrentMode() === 'path') {
newPoint = null;
if (!drawnPath) {
element = getElem(editorContext_.getId());
editorContext_.setStarted(false);
firstCtrl = null;
}
return {
keep: true,
element: element
};
} // Edit mode
var rubberBox = editorContext_.getRubberBox();
if (path.dragging) {
var lastPt = path.cur_pt;
path.dragging = false;
path.dragctrl = false;
path.update();
if (hasMoved) {
path.endChanges('Move path point(s)');
}
if (!evt.shiftKey && !hasMoved) {
path.selectPt(lastPt);
}
} else if (rubberBox && rubberBox.getAttribute('display') !== 'none') {
// Done with multi-node-select
rubberBox.setAttribute('display', 'none');
if (rubberBox.getAttribute('width') <= 2 && rubberBox.getAttribute('height') <= 2) {
pathActions.toSelectMode(evt.target);
} // else, move back to select mode
} else {
pathActions.toSelectMode(evt.target);
}
hasMoved = false;
return undefined;
},
/**
* @param {Element} element
* @returns {void}
*/
toEditMode: function toEditMode(element) {
path = getPath_(element);
editorContext_.setCurrentMode('pathedit');
editorContext_.clearSelection();
path.show(true).update();
path.oldbbox = getBBox(path.elem);
subpath = false;
},
/**
* @param {Element} elem
* @fires module:svgcanvas.SvgCanvas#event:selected
* @returns {void}
*/
toSelectMode: function toSelectMode(elem) {
var selPath = elem === path.elem;
editorContext_.setCurrentMode('select');
path.show(false);
currentPath = false;
editorContext_.clearSelection();
if (path.matrix) {
// Rotated, so may need to re-calculate the center
recalcRotatedPath();
}
if (selPath) {
editorContext_.call('selected', [elem]);
editorContext_.addToSelection([elem], true);
}
},
/**
* @param {boolean} on
* @returns {void}
*/
addSubPath: function addSubPath(on) {
if (on) {
// Internally we go into "path" mode, but in the UI it will
// still appear as if in "pathedit" mode.
editorContext_.setCurrentMode('path');
subpath = true;
} else {
pathActions.clear(true);
pathActions.toEditMode(path.elem);
}
},
/**
* @param {Element} target
* @returns {void}
*/
select: function select(target) {
if (currentPath === target) {
pathActions.toEditMode(target);
editorContext_.setCurrentMode('pathedit'); // going into pathedit mode
} else {
currentPath = target;
}
},
/**
* @fires module:svgcanvas.SvgCanvas#event:changed
* @returns {void}
*/
reorient: function reorient() {
var elem = editorContext_.getSelectedElements()[0];
if (!elem) {
return;
}
var angl = getRotationAngle(elem);
if (angl === 0) {
return;
}
var batchCmd = new BatchCommand('Reorient path');
var changes = {
d: elem.getAttribute('d'),
transform: elem.getAttribute('transform')
};
batchCmd.addSubCommand(new ChangeElementCommand(elem, changes));
editorContext_.clearSelection();
this.resetOrientation(elem);
editorContext_.addCommandToHistory(batchCmd); // Set matrix to null
getPath_(elem).show(false).matrix = null;
this.clear();
editorContext_.addToSelection([elem], true);
editorContext_.call('changed', editorContext_.getSelectedElements());
},
/**
* @param {boolean} remove Not in use
* @returns {void}
*/
clear: function clear(remove) {
var drawnPath = editorContext_.getDrawnPath();
currentPath = null;
if (drawnPath) {
var elem = getElem(editorContext_.getId());
$$1(getElem('path_stretch_line')).remove();
$$1(elem).remove();
$$1(getElem('pathpointgrip_container')).find('*').attr('display', 'none');
firstCtrl = null;
editorContext_.setDrawnPath(null);
editorContext_.setStarted(false);
} else if (editorContext_.getCurrentMode() === 'pathedit') {
this.toSelectMode();
}
if (path) {
path.init().show(false);
}
},
/**
* @param {?(Element|SVGPathElement)} pth
* @returns {false|void}
*/
resetOrientation: function resetOrientation(pth) {
if (isNullish(pth) || pth.nodeName !== 'path') {
return false;
}
var tlist = getTransformList(pth);
var m = transformListToTransform(tlist).matrix;
tlist.clear();
pth.removeAttribute('transform');
var segList = pth.pathSegList; // Opera/win/non-EN throws an error here.
// TODO: Find out why!
// Presumed fixed in Opera 10.5, so commented out for now
// try {
var len = segList.numberOfItems; // } catch(err) {
// const fixed_d = pathActions.convertPath(pth);
// pth.setAttribute('d', fixed_d);
// segList = pth.pathSegList;
// const len = segList.numberOfItems;
// }
// let lastX, lastY;
var _loop = function _loop(i) {
var seg = segList.getItem(i);
var type = seg.pathSegType;
if (type === 1) {
return "continue";
}
var pts = [];
$$1.each(['', 1, 2], function (j, n) {
var x = seg['x' + n],
y = seg['y' + n];
if (x !== undefined && y !== undefined) {
var pt = transformPoint(x, y, m);
pts.splice(pts.length, 0, pt.x, pt.y);
}
});
replacePathSeg(type, i, pts, pth);
};
for (var i = 0; i < len; ++i) {
var _ret = _loop(i);
if (_ret === "continue") continue;
}
reorientGrads(pth, m);
return undefined;
},
/**
* @returns {void}
*/
zoomChange: function zoomChange() {
if (editorContext_.getCurrentMode() === 'pathedit') {
path.update();
}
},
/**
* @typedef {PlainObject} module:path.NodePoint
* @property {Float} x
* @property {Float} y
* @property {Integer} type
*/
/**
* @returns {module:path.NodePoint}
*/
getNodePoint: function getNodePoint() {
var selPt = path.selected_pts.length ? path.selected_pts[0] : 1;
var seg = path.segs[selPt];
return {
x: seg.item.x,
y: seg.item.y,
type: seg.type
};
},
/**
* @param {boolean} linkPoints
* @returns {void}
*/
linkControlPoints: function linkControlPoints(linkPoints) {
setLinkControlPoints(linkPoints);
},
/**
* @returns {void}
*/
clonePathNode: function clonePathNode() {
path.storeD();
var selPts = path.selected_pts; // const {segs} = path;
var i = selPts.length;
var nums = [];
while (i--) {
var pt = selPts[i];
path.addSeg(pt);
nums.push(pt + i);
nums.push(pt + i + 1);
}
path.init().addPtsToSelection(nums);
path.endChanges('Clone path node(s)');
},
/**
* @returns {void}
*/
opencloseSubPath: function opencloseSubPath() {
var selPts = path.selected_pts; // Only allow one selected node for now
if (selPts.length !== 1) {
return;
}
var _path = path,
elem = _path.elem;
var list = elem.pathSegList; // const len = list.numberOfItems;
var index = selPts[0];
var openPt = null;
var startItem = null; // Check if subpath is already open
path.eachSeg(function (i) {
if (this.type === 2 && i <= index) {
startItem = this.item;
}
if (i <= index) {
return true;
}
if (this.type === 2) {
// Found M first, so open
openPt = i;
return false;
}
if (this.type === 1) {
// Found Z first, so closed
openPt = false;
return false;
}
return true;
});
if (isNullish(openPt)) {
// Single path, so close last seg
openPt = path.segs.length - 1;
}
if (openPt !== false) {
// Close this path
// Create a line going to the previous "M"
var newseg = elem.createSVGPathSegLinetoAbs(startItem.x, startItem.y);
var closer = elem.createSVGPathSegClosePath();
if (openPt === path.segs.length - 1) {
list.appendItem(newseg);
list.appendItem(closer);
} else {
insertItemBefore(elem, closer, openPt);
insertItemBefore(elem, newseg, openPt);
}
path.init().selectPt(openPt + 1);
return;
} // M 1,1 L 2,2 L 3,3 L 1,1 z // open at 2,2
// M 2,2 L 3,3 L 1,1
// M 1,1 L 2,2 L 1,1 z M 4,4 L 5,5 L6,6 L 5,5 z
// M 1,1 L 2,2 L 1,1 z [M 4,4] L 5,5 L(M)6,6 L 5,5 z
var seg = path.segs[index];
if (seg.mate) {
list.removeItem(index); // Removes last "L"
list.removeItem(index); // Removes the "Z"
path.init().selectPt(index - 1);
return;
}
var lastM, zSeg; // Find this sub-path's closing point and remove
for (var i = 0; i < list.numberOfItems; i++) {
var item = list.getItem(i);
if (item.pathSegType === 2) {
// Find the preceding M
lastM = i;
} else if (i === index) {
// Remove it
list.removeItem(lastM); // index--;
} else if (item.pathSegType === 1 && index < i) {
// Remove the closing seg of this subpath
zSeg = i - 1;
list.removeItem(i);
break;
}
}
var num = index - lastM - 1;
while (num--) {
insertItemBefore(elem, list.getItem(lastM), zSeg);
}
var pt = list.getItem(lastM); // Make this point the new "M"
replacePathSeg(2, lastM, [pt.x, pt.y]); // i = index; // i is local here, so has no effect; what was the intent for this?
path.init().selectPt(0);
},
/**
* @returns {void}
*/
deletePathNode: function deletePathNode() {
if (!pathActions.canDeleteNodes) {
return;
}
path.storeD();
var selPts = path.selected_pts;
var i = selPts.length;
while (i--) {
var pt = selPts[i];
path.deleteSeg(pt);
} // Cleanup
var cleanup = function cleanup() {
var segList = path.elem.pathSegList;
var len = segList.numberOfItems;
var remItems = function remItems(pos, count) {
while (count--) {
segList.removeItem(pos);
}
};
if (len <= 1) {
return true;
}
while (len--) {
var item = segList.getItem(len);
if (item.pathSegType === 1) {
var prev = segList.getItem(len - 1);
var nprev = segList.getItem(len - 2);
if (prev.pathSegType === 2) {
remItems(len - 1, 2);
cleanup();
break;
} else if (nprev.pathSegType === 2) {
remItems(len - 2, 3);
cleanup();
break;
}
} else if (item.pathSegType === 2) {
if (len > 0) {
var prevType = segList.getItem(len - 1).pathSegType; // Path has M M
if (prevType === 2) {
remItems(len - 1, 1);
cleanup();
break; // Entire path ends with Z M
} else if (prevType === 1 && segList.numberOfItems - 1 === len) {
remItems(len, 1);
cleanup();
break;
}
}
}
}
return false;
};
cleanup(); // Completely delete a path with 1 or 0 segments
if (path.elem.pathSegList.numberOfItems <= 1) {
pathActions.toSelectMode(path.elem);
editorContext_.canvas.deleteSelectedElements();
return;
}
path.init();
path.clearSelection(); // TODO: Find right way to select point now
// path.selectPt(selPt);
if (window.opera) {
// Opera repaints incorrectly
var cp = $$1(path.elem);
cp.attr('d', cp.attr('d'));
}
path.endChanges('Delete path node(s)');
},
/* eslint-disable jsdoc/require-returns */
// Can't seem to use `@borrows` here, so using `@see`
/**
* Smooth polyline into path.
* @function module:path.pathActions.smoothPolylineIntoPath
* @see module:path~smoothPolylineIntoPath
*/
smoothPolylineIntoPath: smoothPolylineIntoPath,
/* eslint-enable jsdoc/require-returns */
/**
* @param {?Integer} v See {@link https://www.w3.org/TR/SVG/single-page.html#paths-InterfaceSVGPathSeg}
* @returns {void}
*/
setSegType: function setSegType(v) {
path.setSegType(v);
},
/**
* @param {string} attr
* @param {Float} newValue
* @returns {void}
*/
moveNode: function moveNode(attr, newValue) {
var selPts = path.selected_pts;
if (!selPts.length) {
return;
}
path.storeD(); // Get first selected point
var seg = path.segs[selPts[0]];
var diff = {
x: 0,
y: 0
};
diff[attr] = newValue - seg.item[attr];
seg.move(diff.x, diff.y);
path.endChanges('Move path point');
},
/**
* @param {Element} elem
* @returns {void}
*/
fixEnd: function fixEnd(elem) {
// Adds an extra segment if the last seg before a Z doesn't end
// at its M point
// M0,0 L0,100 L100,100 z
var segList = elem.pathSegList;
var len = segList.numberOfItems;
var lastM;
for (var i = 0; i < len; ++i) {
var item = segList.getItem(i);
if (item.pathSegType === 2) {
lastM = item;
}
if (item.pathSegType === 1) {
var prev = segList.getItem(i - 1);
if (prev.x !== lastM.x || prev.y !== lastM.y) {
// Add an L segment here
var newseg = elem.createSVGPathSegLinetoAbs(lastM.x, lastM.y);
insertItemBefore(elem, newseg, i); // Can this be done better?
pathActions.fixEnd(elem);
break;
}
}
}
if (isWebkit()) {
editorContext_.resetD(elem);
}
},
/* eslint-disable jsdoc/require-returns */
// Can't seem to use `@borrows` here, so using `@see`
/**
* Convert a path to one with only absolute or relative values.
* @function module:path.pathActions.convertPath
* @see module:path.convertPath
*/
convertPath: convertPath
/* eslint-enable jsdoc/require-returns */
}
);
}(); // end pathActions
var $$2 = jQueryPluginSVG(jQuery); // String used to encode base64.
var KEYSTR = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/='; // Much faster than running getBBox() every time
var visElems = 'a,circle,ellipse,foreignObject,g,image,line,path,polygon,polyline,rect,svg,text,tspan,use';
var visElemsArr = visElems.split(','); // const hidElems = 'clipPath,defs,desc,feGaussianBlur,filter,linearGradient,marker,mask,metadata,pattern,radialGradient,stop,switch,symbol,title,textPath';
var editorContext_$1 = null;
var domdoc_ = null;
var domcontainer_ = null;
var svgroot_ = null;
/**
* Object with the following keys/values.
* @typedef {PlainObject} module:utilities.SVGElementJSON
* @property {string} element - Tag name of the SVG element to create
* @property {PlainObject<string, string>} attr - Has key-value attributes to assign to the new element. An `id` should be set so that {@link module:utilities.EditorContext#addSVGElementFromJson} can later re-identify the element for modification or replacement.
* @property {boolean} [curStyles=false] - Indicates whether current style attributes should be applied first
* @property {module:utilities.SVGElementJSON[]} [children] - Data objects to be added recursively as children
* @property {string} [namespace="http://www.w3.org/2000/svg"] - Indicate a (non-SVG) namespace
*/
/**
* An object that creates SVG elements for the canvas.
*
* @interface module:utilities.EditorContext
* @property {module:path.pathActions} pathActions
*/
/**
* @function module:utilities.EditorContext#getSVGContent
* @returns {SVGSVGElement}
*/
/**
* Create a new SVG element based on the given object keys/values and add it
* to the current layer.
* The element will be run through `cleanupElement` before being returned.
* @function module:utilities.EditorContext#addSVGElementFromJson
* @param {module:utilities.SVGElementJSON} data
* @returns {Element} The new element
*/
/**
* @function module:utilities.EditorContext#getSelectedElements
* @returns {Element[]} the array with selected DOM elements
*/
/**
* @function module:utilities.EditorContext#getDOMDocument
* @returns {HTMLDocument}
*/
/**
* @function module:utilities.EditorContext#getDOMContainer
* @returns {HTMLElement}
*/
/**
* @function module:utilities.EditorContext#getSVGRoot
* @returns {SVGSVGElement}
*/
/**
* @function module:utilities.EditorContext#getBaseUnit
* @returns {string}
*/
/**
* @function module:utilities.EditorContext#getSnappingStep
* @returns {Float|string}
*/
/**
* @function module:utilities.init
* @param {module:utilities.EditorContext} editorContext
* @returns {void}
*/
var init$2 = function init(editorContext) {
editorContext_$1 = editorContext;
domdoc_ = editorContext.getDOMDocument();
domcontainer_ = editorContext.getDOMContainer();
svgroot_ = editorContext.getSVGRoot();
};
/**
* Used to prevent the [Billion laughs attack]{@link https://en.wikipedia.org/wiki/Billion_laughs_attack}.
* @function module:utilities.dropXMLInternalSubset
* @param {string} str String to be processed
* @returns {string} The string with entity declarations in the internal subset removed
* @todo This might be needed in other places `parseFromString` is used even without LGTM flagging
*/
var dropXMLInternalSubset = function dropXMLInternalSubset(str) {
return str.replace(/(<!DOCTYPE\s+\w*\s*\[).*(\?]>)/, '$1$2'); // return str.replace(/(?<doctypeOpen><!DOCTYPE\s+\w*\s*\[).*(?<doctypeClose>\?\]>)/, '$<doctypeOpen>$<doctypeClose>');
};
/**
* Converts characters in a string to XML-friendly entities.
* @function module:utilities.toXml
* @example `&` becomes `&amp;`
* @param {string} str - The string to be converted
* @returns {string} The converted string
*/
var toXml = function toXml(str) {
// &apos; is ok in XML, but not HTML
// &gt; does not normally need escaping, though it can if within a CDATA expression (and preceded by "]]")
return str.replace(/&/g, '&amp;').replace(/</g, '&lt;').replace(/>/g, '&gt;').replace(/"/g, '&quot;').replace(/'/g, '&#x27;'); // Note: `&apos;` is XML only
};
// public domain. It would be nice if you left this header intact.
// Base64 code from Tyler Akins -- http://rumkin.com
// schiller: Removed string concatenation in favour of Array.join() optimization,
// also precalculate the size of the array needed.
/**
* Converts a string to base64.
* @function module:utilities.encode64
* @param {string} input
* @returns {string} Base64 output
*/
function encode64(input) {
// base64 strings are 4/3 larger than the original string
input = encodeUTF8(input); // convert non-ASCII characters
// input = convertToXMLReferences(input);
if (window.btoa) {
return window.btoa(input); // Use native if available
}
var output = new Array(Math.floor((input.length + 2) / 3) * 4);
var i = 0,
p = 0;
do {
var chr1 = input.charCodeAt(i++);
var chr2 = input.charCodeAt(i++);
var chr3 = input.charCodeAt(i++);
/* eslint-disable no-bitwise */
var enc1 = chr1 >> 2;
var enc2 = (chr1 & 3) << 4 | chr2 >> 4;
var enc3 = (chr2 & 15) << 2 | chr3 >> 6;
var enc4 = chr3 & 63;
/* eslint-enable no-bitwise */
if (isNaN(chr2)) {
enc3 = 64;
enc4 = 64;
} else if (isNaN(chr3)) {
enc4 = 64;
}
output[p++] = KEYSTR.charAt(enc1);
output[p++] = KEYSTR.charAt(enc2);
output[p++] = KEYSTR.charAt(enc3);
output[p++] = KEYSTR.charAt(enc4);
} while (i < input.length);
return output.join('');
}
/**
* Converts a string from base64.
* @function module:utilities.decode64
* @param {string} input Base64-encoded input
* @returns {string} Decoded output
*/
function decode64(input) {
if (window.atob) {
return decodeUTF8(window.atob(input));
} // remove all characters that are not A-Z, a-z, 0-9, +, /, or =
input = input.replace(/[^A-Za-z\d+/=]/g, '');
var output = '';
var i = 0;
do {
var enc1 = KEYSTR.indexOf(input.charAt(i++));
var enc2 = KEYSTR.indexOf(input.charAt(i++));
var enc3 = KEYSTR.indexOf(input.charAt(i++));
var enc4 = KEYSTR.indexOf(input.charAt(i++));
/* eslint-disable no-bitwise */
var chr1 = enc1 << 2 | enc2 >> 4;
var chr2 = (enc2 & 15) << 4 | enc3 >> 2;
var chr3 = (enc3 & 3) << 6 | enc4;
/* eslint-enable no-bitwise */
output += String.fromCharCode(chr1);
if (enc3 !== 64) {
output += String.fromCharCode(chr2);
}
if (enc4 !== 64) {
output += String.fromCharCode(chr3);
}
} while (i < input.length);
return decodeUTF8(output);
}
/**
* @function module:utilities.decodeUTF8
* @param {string} argString
* @returns {string}
*/
function decodeUTF8(argString) {
return decodeURIComponent(escape(argString));
} // codedread:does not seem to work with webkit-based browsers on OSX // Brettz9: please test again as function upgraded
/**
* @function module:utilities.encodeUTF8
* @param {string} argString
* @returns {string}
*/
var encodeUTF8 = function encodeUTF8(argString) {
return unescape(encodeURIComponent(argString));
};
/**
* Convert dataURL to object URL.
* @function module:utilities.dataURLToObjectURL
* @param {string} dataurl
* @returns {string} object URL or empty string
*/
var dataURLToObjectURL = function dataURLToObjectURL(dataurl) {
if (typeof Uint8Array === 'undefined' || typeof Blob === 'undefined' || typeof URL === 'undefined' || !URL.createObjectURL) {
return '';
}
var arr = dataurl.split(','),
mime = arr[0].match(/:(.*?);/)[1],
bstr = atob(arr[1]);
/*
const [prefix, suffix] = dataurl.split(','),
{groups: {mime}} = prefix.match(/:(?<mime>.*?);/),
bstr = atob(suffix);
*/
var n = bstr.length;
var u8arr = new Uint8Array(n);
while (n--) {
u8arr[n] = bstr.charCodeAt(n);
}
var blob = new Blob([u8arr], {
type: mime
});
return URL.createObjectURL(blob);
};
/**
* Get object URL for a blob object.
* @function module:utilities.createObjectURL
* @param {Blob} blob A Blob object or File object
* @returns {string} object URL or empty string
*/
var createObjectURL = function createObjectURL(blob) {
if (!blob || typeof URL === 'undefined' || !URL.createObjectURL) {
return '';
}
return URL.createObjectURL(blob);
};
/**
* @property {string} blankPageObjectURL
*/
var blankPageObjectURL = function () {
if (typeof Blob === 'undefined') {
return '';
}
var blob = new Blob(['<html><head><title>SVG-edit</title></head><body>&nbsp;</body></html>'], {
type: 'text/html'
});
return createObjectURL(blob);
}();
/**
* Cross-browser compatible method of converting a string to an XML tree.
* Found this function [here]{@link http://groups.google.com/group/jquery-dev/browse_thread/thread/c6d11387c580a77f}.
* @function module:utilities.text2xml
* @param {string} sXML
* @throws {Error}
* @returns {XMLDocument}
*/
var text2xml = function text2xml(sXML) {
if (sXML.includes('<svg:svg')) {
sXML = sXML.replace(/<(\/?)svg:/g, '<$1').replace('xmlns:svg', 'xmlns');
}
var out, dXML;
try {
dXML = window.DOMParser ? new DOMParser() : new window.ActiveXObject('Microsoft.XMLDOM');
dXML.async = false;
} catch (e) {
throw new Error('XML Parser could not be instantiated');
}
try {
if (dXML.loadXML) {
out = dXML.loadXML(sXML) ? dXML : false;
} else {
out = dXML.parseFromString(sXML, 'text/xml');
}
} catch (e2) {
throw new Error('Error parsing XML string');
}
return out;
};
/**
* @typedef {PlainObject} module:utilities.BBoxObject (like `DOMRect`)
* @property {Float} x
* @property {Float} y
* @property {Float} width
* @property {Float} height
*/
/**
* Converts a `SVGRect` into an object.
* @function module:utilities.bboxToObj
* @param {SVGRect} bbox - a SVGRect
* @returns {module:utilities.BBoxObject} An object with properties names x, y, width, height.
*/
var bboxToObj = function bboxToObj(_ref) {
var x = _ref.x,
y = _ref.y,
width = _ref.width,
height = _ref.height;
return {
x: x,
y: y,
width: width,
height: height
};
};
/**
* @callback module:utilities.TreeWalker
* @param {Element} elem - DOM element being traversed
* @returns {void}
*/
/**
* Walks the tree and executes the callback on each element in a top-down fashion.
* @function module:utilities.walkTree
* @param {Element} elem - DOM element to traverse
* @param {module:utilities.TreeWalker} cbFn - Callback function to run on each element
* @returns {void}
*/
var walkTree = function walkTree(elem, cbFn) {
if (elem && elem.nodeType === 1) {
cbFn(elem);
var i = elem.childNodes.length;
while (i--) {
walkTree(elem.childNodes.item(i), cbFn);
}
}
};
/**
* Walks the tree and executes the callback on each element in a depth-first fashion.
* @function module:utilities.walkTreePost
* @todo Shouldn't this be calling walkTreePost?
* @param {Element} elem - DOM element to traverse
* @param {module:utilities.TreeWalker} cbFn - Callback function to run on each element
* @returns {void}
*/
var walkTreePost = function walkTreePost(elem, cbFn) {
if (elem && elem.nodeType === 1) {
var i = elem.childNodes.length;
while (i--) {
walkTree(elem.childNodes.item(i), cbFn);
}
cbFn(elem);
}
};
/**
* Extracts the URL from the `url(...)` syntax of some attributes.
* Three variants:
* - `<circle fill="url(someFile.svg#foo)" />`
* - `<circle fill="url('someFile.svg#foo')" />`
* - `<circle fill='url("someFile.svg#foo")' />`
* @function module:utilities.getUrlFromAttr
* @param {string} attrVal The attribute value as a string
* @returns {string} String with just the URL, like "someFile.svg#foo"
*/
var getUrlFromAttr = function getUrlFromAttr(attrVal) {
if (attrVal) {
// url('#somegrad')
if (attrVal.startsWith('url("')) {
return attrVal.substring(5, attrVal.indexOf('"', 6));
} // url('#somegrad')
if (attrVal.startsWith("url('")) {
return attrVal.substring(5, attrVal.indexOf("'", 6));
}
if (attrVal.startsWith('url(')) {
return attrVal.substring(4, attrVal.indexOf(')'));
}
}
return null;
};
/**
* @function module:utilities.getHref
* @param {Element} elem
* @returns {string} The given element's `xlink:href` value
*/
var getHref = function getHref(elem) {
// eslint-disable-line import/no-mutable-exports
return elem.getAttributeNS(NS.XLINK, 'href');
};
/**
* Sets the given element's `xlink:href` value.
* @function module:utilities.setHref
* @param {Element} elem
* @param {string} val
* @returns {void}
*/
var setHref = function setHref(elem, val) {
// eslint-disable-line import/no-mutable-exports
elem.setAttributeNS(NS.XLINK, 'xlink:href', val);
};
/**
* @function module:utilities.findDefs
* @returns {SVGDefsElement} The document's `<defs>` element, creating it first if necessary
*/
var findDefs = function findDefs() {
var svgElement = editorContext_$1.getSVGContent();
var defs = svgElement.getElementsByTagNameNS(NS.SVG, 'defs');
if (defs.length > 0) {
defs = defs[0];
} else {
defs = svgElement.ownerDocument.createElementNS(NS.SVG, 'defs');
if (svgElement.firstChild) {
// first child is a comment, so call nextSibling
svgElement.insertBefore(defs, svgElement.firstChild.nextSibling); // svgElement.firstChild.nextSibling.before(defs); // Not safe
} else {
svgElement.append(defs);
}
}
return defs;
}; // TODO(codedread): Consider moving the next to functions to bbox.js
/**
* Get correct BBox for a path in Webkit.
* Converted from code found [here]{@link http://blog.hackers-cafe.net/2009/06/how-to-calculate-bezier-curves-bounding.html}.
* @function module:utilities.getPathBBox
* @param {SVGPathElement} path - The path DOM element to get the BBox for
* @returns {module:utilities.BBoxObject} A BBox-like object
*/
var getPathBBox = function getPathBBox(path) {
var seglist = path.pathSegList;
var tot = seglist.numberOfItems;
var bounds = [[], []];
var start = seglist.getItem(0);
var P0 = [start.x, start.y];
var getCalc = function getCalc(j, P1, P2, P3) {
return function (t) {
return 1 - Math.pow(t, 3) * P0[j] + 3 * 1 - Math.pow(t, 2) * t * P1[j] + 3 * (1 - t) * Math.pow(t, 2) * P2[j] + Math.pow(t, 3) * P3[j];
};
};
for (var i = 0; i < tot; i++) {
var seg = seglist.getItem(i);
if (seg.x === undefined) {
continue;
} // Add actual points to limits
bounds[0].push(P0[0]);
bounds[1].push(P0[1]);
if (seg.x1) {
var P1 = [seg.x1, seg.y1],
P2 = [seg.x2, seg.y2],
P3 = [seg.x, seg.y];
for (var j = 0; j < 2; j++) {
var calc = getCalc(j, P1, P2, P3);
var b = 6 * P0[j] - 12 * P1[j] + 6 * P2[j];
var a = -3 * P0[j] + 9 * P1[j] - 9 * P2[j] + 3 * P3[j];
var c = 3 * P1[j] - 3 * P0[j];
if (a === 0) {
if (b === 0) {
continue;
}
var t = -c / b;
if (t > 0 && t < 1) {
bounds[j].push(calc(t));
}
continue;
}
var b2ac = Math.pow(b, 2) - 4 * c * a;
if (b2ac < 0) {
continue;
}
var t1 = (-b + Math.sqrt(b2ac)) / (2 * a);
if (t1 > 0 && t1 < 1) {
bounds[j].push(calc(t1));
}
var t2 = (-b - Math.sqrt(b2ac)) / (2 * a);
if (t2 > 0 && t2 < 1) {
bounds[j].push(calc(t2));
}
}
P0 = P3;
} else {
bounds[0].push(seg.x);
bounds[1].push(seg.y);
}
}
var x = Math.min.apply(null, bounds[0]);
var w = Math.max.apply(null, bounds[0]) - x;
var y = Math.min.apply(null, bounds[1]);
var h = Math.max.apply(null, bounds[1]) - y;
return {
x: x,
y: y,
width: w,
height: h
};
};
/**
* Get the given/selected element's bounding box object, checking for
* horizontal/vertical lines (see issue 717)
* Note that performance is currently terrible, so some way to improve would
* be great.
* @param {Element} selected - Container or `<use>` DOM element
* @returns {DOMRect} Bounding box object
*/
function groupBBFix(selected) {
if (supportsHVLineContainerBBox()) {
try {
return selected.getBBox();
} catch (e) {}
}
var ref = $$2.data(selected, 'ref');
var matched = null;
var ret, copy;
if (ref) {
copy = $$2(ref).children().clone().attr('visibility', 'hidden');
$$2(svgroot_).append(copy);
matched = copy.filter('line, path');
} else {
matched = $$2(selected).find('line, path');
}
var issue = false;
if (matched.length) {
matched.each(function () {
var bb = this.getBBox();
if (!bb.width || !bb.height) {
issue = true;
}
});
if (issue) {
var elems = ref ? copy : $$2(selected).children();
ret = getStrokedBBox(elems);
} else {
ret = selected.getBBox();
}
} else {
ret = selected.getBBox();
}
if (ref) {
copy.remove();
}
return ret;
}
/**
* Get the given/selected element's bounding box object, convert it to be more
* usable when necessary.
* @function module:utilities.getBBox
* @param {Element} elem - Optional DOM element to get the BBox for
* @returns {module:utilities.BBoxObject} Bounding box object
*/
var getBBox = function getBBox(elem) {
var selected = elem || editorContext_$1.geSelectedElements()[0];
if (elem.nodeType !== 1) {
return null;
}
var elname = selected.nodeName;
var ret = null;
switch (elname) {
case 'text':
if (selected.textContent === '') {
selected.textContent = 'a'; // Some character needed for the selector to use.
ret = selected.getBBox();
selected.textContent = '';
} else if (selected.getBBox) {
ret = selected.getBBox();
}
break;
case 'path':
if (!supportsPathBBox()) {
ret = getPathBBox(selected);
} else if (selected.getBBox) {
ret = selected.getBBox();
}
break;
case 'g':
case 'a':
ret = groupBBFix(selected);
break;
default:
if (elname === 'use') {
ret = groupBBFix(selected); // , true);
}
if (elname === 'use' || elname === 'foreignObject' && isWebkit()) {
if (!ret) {
ret = selected.getBBox();
} // This is resolved in later versions of webkit, perhaps we should
// have a featured detection for correct 'use' behavior?
// ——————————
if (!isWebkit()) {
var _ret = ret,
x = _ret.x,
y = _ret.y,
width = _ret.width,
height = _ret.height;
var bb = {
width: width,
height: height,
x: x + parseFloat(selected.getAttribute('x') || 0),
y: y + parseFloat(selected.getAttribute('y') || 0)
};
ret = bb;
}
} else if (visElemsArr.includes(elname)) {
if (selected) {
try {
ret = selected.getBBox();
} catch (err) {
// tspan (and textPath apparently) have no `getBBox` in Firefox: https://bugzilla.mozilla.org/show_bug.cgi?id=937268
// Re: Chrome returning bbox for containing text element, see: https://bugs.chromium.org/p/chromium/issues/detail?id=349835
var extent = selected.getExtentOfChar(0); // pos+dimensions of the first glyph
var _width = selected.getComputedTextLength(); // width of the tspan
ret = {
x: extent.x,
y: extent.y,
width: _width,
height: extent.height
};
}
} else {
// Check if element is child of a foreignObject
var fo = $$2(selected).closest('foreignObject');
if (fo.length) {
if (fo[0].getBBox) {
ret = fo[0].getBBox();
}
}
}
}
}
if (ret) {
ret = bboxToObj(ret);
} // get the bounding box from the DOM (which is in that element's coordinate system)
return ret;
};
/**
* @typedef {GenericArray} module:utilities.PathSegmentArray
* @property {Integer} length 2
* @property {"M"|"L"|"C"|"Z"} 0
* @property {Float[]} 1
*/
/**
* Create a path 'd' attribute from path segments.
* Each segment is an array of the form: `[singleChar, [x,y, x,y, ...]]`
* @function module:utilities.getPathDFromSegments
* @param {module:utilities.PathSegmentArray[]} pathSegments - An array of path segments to be converted
* @returns {string} The converted path d attribute.
*/
var getPathDFromSegments = function getPathDFromSegments(pathSegments) {
var d = '';
$$2.each(pathSegments, function (j, _ref2) {
var _ref3 = _slicedToArray(_ref2, 2),
singleChar = _ref3[0],
pts = _ref3[1];
d += singleChar;
for (var i = 0; i < pts.length; i += 2) {
d += pts[i] + ',' + pts[i + 1] + ' ';
}
});
return d;
};
/**
* Make a path 'd' attribute from a simple SVG element shape.
* @function module:utilities.getPathDFromElement
* @param {Element} elem - The element to be converted
* @returns {string} The path d attribute or `undefined` if the element type is unknown.
*/
var getPathDFromElement = function getPathDFromElement(elem) {
// Possibly the cubed root of 6, but 1.81 works best
var num = 1.81;
var d, a, rx, ry;
switch (elem.tagName) {
case 'ellipse':
case 'circle':
{
a = $$2(elem).attr(['rx', 'ry', 'cx', 'cy']);
var _a = a,
cx = _a.cx,
cy = _a.cy;
var _a2 = a;
rx = _a2.rx;
ry = _a2.ry;
if (elem.tagName === 'circle') {
ry = $$2(elem).attr('r');
rx = ry;
}
d = getPathDFromSegments([['M', [cx - rx, cy]], ['C', [cx - rx, cy - ry / num, cx - rx / num, cy - ry, cx, cy - ry]], ['C', [cx + rx / num, cy - ry, cx + rx, cy - ry / num, cx + rx, cy]], ['C', [cx + rx, cy + ry / num, cx + rx / num, cy + ry, cx, cy + ry]], ['C', [cx - rx / num, cy + ry, cx - rx, cy + ry / num, cx - rx, cy]], ['Z', []]]);
break;
}
case 'path':
d = elem.getAttribute('d');
break;
case 'line':
a = $$2(elem).attr(['x1', 'y1', 'x2', 'y2']);
d = 'M' + a.x1 + ',' + a.y1 + 'L' + a.x2 + ',' + a.y2;
break;
case 'polyline':
d = 'M' + elem.getAttribute('points');
break;
case 'polygon':
d = 'M' + elem.getAttribute('points') + ' Z';
break;
case 'rect':
{
var r = $$2(elem).attr(['rx', 'ry']);
rx = r.rx;
ry = r.ry;
var b = elem.getBBox();
var x = b.x,
y = b.y,
w = b.width,
h = b.height;
num = 4 - num; // Why? Because!
if (!rx && !ry) {
// Regular rect
d = getPathDFromSegments([['M', [x, y]], ['L', [x + w, y]], ['L', [x + w, y + h]], ['L', [x, y + h]], ['L', [x, y]], ['Z', []]]);
} else {
d = getPathDFromSegments([['M', [x, y + ry]], ['C', [x, y + ry / num, x + rx / num, y, x + rx, y]], ['L', [x + w - rx, y]], ['C', [x + w - rx / num, y, x + w, y + ry / num, x + w, y + ry]], ['L', [x + w, y + h - ry]], ['C', [x + w, y + h - ry / num, x + w - rx / num, y + h, x + w - rx, y + h]], ['L', [x + rx, y + h]], ['C', [x + rx / num, y + h, x, y + h - ry / num, x, y + h - ry]], ['L', [x, y + ry]], ['Z', []]]);
}
break;
}
}
return d;
};
/**
* Get a set of attributes from an element that is useful for convertToPath.
* @function module:utilities.getExtraAttributesForConvertToPath
* @param {Element} elem - The element to be probed
* @returns {PlainObject<"marker-start"|"marker-end"|"marker-mid"|"filter"|"clip-path", string>} An object with attributes.
*/
var getExtraAttributesForConvertToPath = function getExtraAttributesForConvertToPath(elem) {
var attrs = {}; // TODO: make this list global so that we can properly maintain it
// TODO: what about @transform, @clip-rule, @fill-rule, etc?
$$2.each(['marker-start', 'marker-end', 'marker-mid', 'filter', 'clip-path'], function () {
var a = elem.getAttribute(this);
if (a) {
attrs[this] = a;
}
});
return attrs;
};
/**
* Get the BBox of an element-as-path.
* @function module:utilities.getBBoxOfElementAsPath
* @param {Element} elem - The DOM element to be probed
* @param {module:utilities.EditorContext#addSVGElementFromJson} addSVGElementFromJson - Function to add the path element to the current layer. See canvas.addSVGElementFromJson
* @param {module:path.pathActions} pathActions - If a transform exists, `pathActions.resetOrientation()` is used. See: canvas.pathActions.
* @returns {DOMRect|false} The resulting path's bounding box object.
*/
var getBBoxOfElementAsPath = function getBBoxOfElementAsPath(elem, addSVGElementFromJson, pathActions) {
var path = addSVGElementFromJson({
element: 'path',
attr: getExtraAttributesForConvertToPath(elem)
});
var eltrans = elem.getAttribute('transform');
if (eltrans) {
path.setAttribute('transform', eltrans);
}
var parentNode = elem.parentNode;
if (elem.nextSibling) {
elem.before(path);
} else {
parentNode.append(path);
}
var d = getPathDFromElement(elem);
if (d) {
path.setAttribute('d', d);
} else {
path.remove();
} // Get the correct BBox of the new path, then discard it
pathActions.resetOrientation(path);
var bb = false;
try {
bb = path.getBBox();
} catch (e) {// Firefox fails
}
path.remove();
return bb;
};
/**
* Convert selected element to a path.
* @function module:utilities.convertToPath
* @param {Element} elem - The DOM element to be converted
* @param {module:utilities.SVGElementJSON} attrs - Apply attributes to new path. see canvas.convertToPath
* @param {module:utilities.EditorContext#addSVGElementFromJson} addSVGElementFromJson - Function to add the path element to the current layer. See canvas.addSVGElementFromJson
* @param {module:path.pathActions} pathActions - If a transform exists, pathActions.resetOrientation() is used. See: canvas.pathActions.
* @param {module:draw.DrawCanvasInit#clearSelection|module:path.EditorContext#clearSelection} clearSelection - see [canvas.clearSelection]{@link module:svgcanvas.SvgCanvas#clearSelection}
* @param {module:path.EditorContext#addToSelection} addToSelection - see [canvas.addToSelection]{@link module:svgcanvas.SvgCanvas#addToSelection}
* @param {module:history} hstry - see history module
* @param {module:path.EditorContext#addCommandToHistory|module:draw.DrawCanvasInit#addCommandToHistory} addCommandToHistory - see [canvas.addCommandToHistory]{@link module:svgcanvas~addCommandToHistory}
* @returns {SVGPathElement|null} The converted path element or null if the DOM element was not recognized.
*/
var convertToPath = function convertToPath(elem, attrs, addSVGElementFromJson, pathActions, clearSelection, addToSelection, hstry, addCommandToHistory) {
var batchCmd = new hstry.BatchCommand('Convert element to Path'); // Any attribute on the element not covered by the passed-in attributes
attrs = $$2.extend({}, attrs, getExtraAttributesForConvertToPath(elem));
var path = addSVGElementFromJson({
element: 'path',
attr: attrs
});
var eltrans = elem.getAttribute('transform');
if (eltrans) {
path.setAttribute('transform', eltrans);
}
var id = elem.id;
var parentNode = elem.parentNode;
if (elem.nextSibling) {
elem.before(path);
} else {
parentNode.append(path);
}
var d = getPathDFromElement(elem);
if (d) {
path.setAttribute('d', d); // Replace the current element with the converted one
// Reorient if it has a matrix
if (eltrans) {
var tlist = getTransformList(path);
if (hasMatrixTransform(tlist)) {
pathActions.resetOrientation(path);
}
}
var nextSibling = elem.nextSibling;
batchCmd.addSubCommand(new hstry.RemoveElementCommand(elem, nextSibling, parent));
batchCmd.addSubCommand(new hstry.InsertElementCommand(path));
clearSelection();
elem.remove();
path.setAttribute('id', id);
path.removeAttribute('visibility');
addToSelection([path], true);
addCommandToHistory(batchCmd);
return path;
} // the elem.tagName was not recognized, so no "d" attribute. Remove it, so we've haven't changed anything.
path.remove();
return null;
};
/**
* Can the bbox be optimized over the native getBBox? The optimized bbox is the same as the native getBBox when
* the rotation angle is a multiple of 90 degrees and there are no complex transforms.
* Getting an optimized bbox can be dramatically slower, so we want to make sure it's worth it.
*
* The best example for this is a circle rotate 45 degrees. The circle doesn't get wider or taller when rotated
* about it's center.
*
* The standard, unoptimized technique gets the native bbox of the circle, rotates the box 45 degrees, uses
* that width and height, and applies any transforms to get the final bbox. This means the calculated bbox
* is much wider than the original circle. If the angle had been 0, 90, 180, etc. both techniques render the
* same bbox.
*
* The optimization is not needed if the rotation is a multiple 90 degrees. The default technique is to call
* getBBox then apply the angle and any transforms.
*
* @param {Float} angle - The rotation angle in degrees
* @param {boolean} hasAMatrixTransform - True if there is a matrix transform
* @returns {boolean} True if the bbox can be optimized.
*/
function bBoxCanBeOptimizedOverNativeGetBBox(angle, hasAMatrixTransform) {
var angleModulo90 = angle % 90;
var closeTo90 = angleModulo90 < -89.99 || angleModulo90 > 89.99;
var closeTo0 = angleModulo90 > -0.001 && angleModulo90 < 0.001;
return hasAMatrixTransform || !(closeTo0 || closeTo90);
}
/**
* Get bounding box that includes any transforms.
* @function module:utilities.getBBoxWithTransform
* @param {Element} elem - The DOM element to be converted
* @param {module:utilities.EditorContext#addSVGElementFromJson} addSVGElementFromJson - Function to add the path element to the current layer. See canvas.addSVGElementFromJson
* @param {module:path.pathActions} pathActions - If a transform exists, pathActions.resetOrientation() is used. See: canvas.pathActions.
* @returns {module:utilities.BBoxObject|module:math.TransformedBox|DOMRect} A single bounding box object
*/
var getBBoxWithTransform = function getBBoxWithTransform(elem, addSVGElementFromJson, pathActions) {
// TODO: Fix issue with rotated groups. Currently they work
// fine in FF, but not in other browsers (same problem mentioned
// in Issue 339 comment #2).
var bb = getBBox(elem);
if (!bb) {
return null;
}
var tlist = getTransformList(elem);
var angle = getRotationAngleFromTransformList(tlist);
var hasMatrixXForm = hasMatrixTransform(tlist);
if (angle || hasMatrixXForm) {
var goodBb = false;
if (bBoxCanBeOptimizedOverNativeGetBBox(angle, hasMatrixXForm)) {
// Get the BBox from the raw path for these elements
// TODO: why ellipse and not circle
var elemNames = ['ellipse', 'path', 'line', 'polyline', 'polygon'];
if (elemNames.includes(elem.tagName)) {
goodBb = getBBoxOfElementAsPath(elem, addSVGElementFromJson, pathActions);
bb = goodBb;
} else if (elem.tagName === 'rect') {
// Look for radius
var rx = elem.getAttribute('rx');
var ry = elem.getAttribute('ry');
if (rx || ry) {
goodBb = getBBoxOfElementAsPath(elem, addSVGElementFromJson, pathActions);
bb = goodBb;
}
}
}
if (!goodBb) {
var _transformListToTrans = transformListToTransform(tlist),
matrix = _transformListToTrans.matrix;
bb = transformBox(bb.x, bb.y, bb.width, bb.height, matrix).aabox; // Old technique that was exceedingly slow with large documents.
//
// Accurate way to get BBox of rotated element in Firefox:
// Put element in group and get its BBox
//
// Must use clone else FF freaks out
// const clone = elem.cloneNode(true);
// const g = document.createElementNS(NS.SVG, 'g');
// const parent = elem.parentNode;
// parent.append(g);
// g.append(clone);
// const bb2 = bboxToObj(g.getBBox());
// g.remove();
}
}
return bb;
};
/**
* @param {Element} elem
* @returns {Float}
* @todo This is problematic with large stroke-width and, for example, a single
* horizontal line. The calculated BBox extends way beyond left and right sides.
*/
function getStrokeOffsetForBBox(elem) {
var sw = elem.getAttribute('stroke-width');
return !isNaN(sw) && elem.getAttribute('stroke') !== 'none' ? sw / 2 : 0;
}
/**
* @typedef {PlainObject} BBox
* @property {Integer} x The x value
* @property {Integer} y The y value
* @property {Float} width
* @property {Float} height
*/
/**
* Get the bounding box for one or more stroked and/or transformed elements.
* @function module:utilities.getStrokedBBox
* @param {Element[]} elems - Array with DOM elements to check
* @param {module:utilities.EditorContext#addSVGElementFromJson} addSVGElementFromJson - Function to add the path element to the current layer. See canvas.addSVGElementFromJson
* @param {module:path.pathActions} pathActions - If a transform exists, pathActions.resetOrientation() is used. See: canvas.pathActions.
* @returns {module:utilities.BBoxObject|module:math.TransformedBox|DOMRect} A single bounding box object
*/
var getStrokedBBox = function getStrokedBBox(elems, addSVGElementFromJson, pathActions) {
if (!elems || !elems.length) {
return false;
}
var fullBb;
$$2.each(elems, function () {
if (fullBb) {
return;
}
if (!this.parentNode) {
return;
}
fullBb = getBBoxWithTransform(this, addSVGElementFromJson, pathActions);
}); // This shouldn't ever happen...
if (fullBb === undefined) {
return null;
} // fullBb doesn't include the stoke, so this does no good!
// if (elems.length == 1) return fullBb;
var maxX = fullBb.x + fullBb.width;
var maxY = fullBb.y + fullBb.height;
var minX = fullBb.x;
var minY = fullBb.y; // If only one elem, don't call the potentially slow getBBoxWithTransform method again.
if (elems.length === 1) {
var offset = getStrokeOffsetForBBox(elems[0]);
minX -= offset;
minY -= offset;
maxX += offset;
maxY += offset;
} else {
$$2.each(elems, function (i, elem) {
var curBb = getBBoxWithTransform(elem, addSVGElementFromJson, pathActions);
if (curBb) {
var _offset = getStrokeOffsetForBBox(elem);
minX = Math.min(minX, curBb.x - _offset);
minY = Math.min(minY, curBb.y - _offset); // TODO: The old code had this test for max, but not min. I suspect this test should be for both min and max
if (elem.nodeType === 1) {
maxX = Math.max(maxX, curBb.x + curBb.width + _offset);
maxY = Math.max(maxY, curBb.y + curBb.height + _offset);
}
}
});
}
fullBb.x = minX;
fullBb.y = minY;
fullBb.width = maxX - minX;
fullBb.height = maxY - minY;
return fullBb;
};
/**
* Get all elements that have a BBox (excludes `<defs>`, `<title>`, etc).
* Note that 0-opacity, off-screen etc elements are still considered "visible"
* for this function.
* @function module:utilities.getVisibleElements
* @param {Element} parentElement - The parent DOM element to search within
* @returns {Element[]} All "visible" elements.
*/
var getVisibleElements = function getVisibleElements(parentElement) {
if (!parentElement) {
parentElement = $$2(editorContext_$1.getSVGContent()).children(); // Prevent layers from being included
}
var contentElems = [];
$$2(parentElement).children().each(function (i, elem) {
if (elem.getBBox) {
contentElems.push(elem);
}
});
return contentElems.reverse();
};
/**
* Get the bounding box for one or more stroked and/or transformed elements.
* @function module:utilities.getStrokedBBoxDefaultVisible
* @param {Element[]} elems - Array with DOM elements to check
* @returns {module:utilities.BBoxObject} A single bounding box object
*/
var getStrokedBBoxDefaultVisible = function getStrokedBBoxDefaultVisible(elems) {
if (!elems) {
elems = getVisibleElements();
}
return getStrokedBBox(elems, editorContext_$1.addSVGElementFromJson, editorContext_$1.pathActions);
};
/**
* Get the rotation angle of the given transform list.
* @function module:utilities.getRotationAngleFromTransformList
* @param {SVGTransformList} tlist - List of transforms
* @param {boolean} toRad - When true returns the value in radians rather than degrees
* @returns {Float} The angle in degrees or radians
*/
var getRotationAngleFromTransformList = function getRotationAngleFromTransformList(tlist, toRad) {
if (!tlist) {
return 0;
} // <svg> elements have no tlist
var N = tlist.numberOfItems;
for (var i = 0; i < N; ++i) {
var xform = tlist.getItem(i);
if (xform.type === 4) {
return toRad ? xform.angle * Math.PI / 180.0 : xform.angle;
}
}
return 0.0;
};
/**
* Get the rotation angle of the given/selected DOM element.
* @function module:utilities.getRotationAngle
* @param {Element} [elem] - DOM element to get the angle for. Default to first of selected elements.
* @param {boolean} [toRad=false] - When true returns the value in radians rather than degrees
* @returns {Float} The angle in degrees or radians
*/
var getRotationAngle = function getRotationAngle(elem, toRad) {
// eslint-disable-line import/no-mutable-exports
var selected = elem || editorContext_$1.getSelectedElements()[0]; // find the rotation transform (if any) and set it
var tlist = getTransformList(selected);
return getRotationAngleFromTransformList(tlist, toRad);
};
/**
* Get the reference element associated with the given attribute value.
* @function module:utilities.getRefElem
* @param {string} attrVal - The attribute value as a string
* @returns {Element} Reference element
*/
var getRefElem = function getRefElem(attrVal) {
return getElem(getUrlFromAttr(attrVal).substr(1));
};
/**
* Get a DOM element by ID within the SVG root element.
* @function module:utilities.getElem
* @param {string} id - String with the element's new ID
* @returns {?Element}
*/
var getElem = supportsSelectors() ? function (id) {
// querySelector lookup
return svgroot_.querySelector('#' + id);
} : supportsXpath() ? function (id) {
// xpath lookup
return domdoc_.evaluate('svg:svg[@id="svgroot"]//svg:*[@id="' + id + '"]', domcontainer_, function () {
return NS.SVG;
}, 9, null).singleNodeValue;
} : function (id) {
// jQuery lookup: twice as slow as xpath in FF
return $$2(svgroot_).find('[id=' + id + ']')[0];
};
/**
* Assigns multiple attributes to an element.
* @function module:utilities.assignAttributes
* @param {Element} elem - DOM element to apply new attribute values to
* @param {PlainObject<string, string>} attrs - Object with attribute keys/values
* @param {Integer} [suspendLength] - Milliseconds to suspend redraw
* @param {boolean} [unitCheck=false] - Boolean to indicate the need to use units.setUnitAttr
* @returns {void}
*/
var assignAttributes = function assignAttributes(elem, attrs, suspendLength, unitCheck) {
for (var _i = 0, _Object$entries = Object.entries(attrs); _i < _Object$entries.length; _i++) {
var _Object$entries$_i = _slicedToArray(_Object$entries[_i], 2),
key = _Object$entries$_i[0],
value = _Object$entries$_i[1];
var ns = key.substr(0, 4) === 'xml:' ? NS.XML : key.substr(0, 6) === 'xlink:' ? NS.XLINK : null;
2020-01-25 02:32:24 +00:00
if (isNullish(value)) {
if (ns) {
elem.removeAttributeNS(ns, key);
} else {
elem.removeAttribute(key);
}
continue;
}
if (ns) {
elem.setAttributeNS(ns, key, value);
} else if (!unitCheck) {
elem.setAttribute(key, value);
} else {
setUnitAttr(elem, key, value);
}
}
};
/**
* Remove unneeded (default) attributes, making resulting SVG smaller.
* @function module:utilities.cleanupElement
* @param {Element} element - DOM element to clean up
* @returns {void}
*/
var cleanupElement = function cleanupElement(element) {
var defaults = {
'fill-opacity': 1,
'stop-opacity': 1,
opacity: 1,
stroke: 'none',
'stroke-dasharray': 'none',
'stroke-linejoin': 'miter',
'stroke-linecap': 'butt',
'stroke-opacity': 1,
'stroke-width': 1,
rx: 0,
ry: 0
};
if (element.nodeName === 'ellipse') {
// Ellipse elements require rx and ry attributes
delete defaults.rx;
delete defaults.ry;
}
Object.entries(defaults).forEach(function (_ref4) {
var _ref5 = _slicedToArray(_ref4, 2),
attr = _ref5[0],
val = _ref5[1];
if (element.getAttribute(attr) === String(val)) {
element.removeAttribute(attr);
}
});
};
/**
* Round value to for snapping.
* @function module:utilities.snapToGrid
* @param {Float} value
* @returns {Integer}
*/
var snapToGrid = function snapToGrid(value) {
var unit = editorContext_$1.getBaseUnit();
var stepSize = editorContext_$1.getSnappingStep();
if (unit !== 'px') {
stepSize *= getTypeMap()[unit];
}
value = Math.round(value / stepSize) * stepSize;
return value;
};
/**
* Prevents default browser click behaviour on the given element.
* @function module:utilities.preventClickDefault
* @param {Element} img - The DOM element to prevent the click on
* @returns {void}
*/
var preventClickDefault = function preventClickDefault(img) {
$$2(img).click(function (e) {
e.preventDefault();
});
};
/**
* @callback module:utilities.GetNextID
* @returns {string} The ID
*/
/**
* Create a clone of an element, updating its ID and its children's IDs when needed.
* @function module:utilities.copyElem
* @param {Element} el - DOM element to clone
* @param {module:utilities.GetNextID} getNextId - The getter of the next unique ID.
* @returns {Element} The cloned element
*/
var copyElem = function copyElem(el, getNextId) {
// manually create a copy of the element
var newEl = document.createElementNS(el.namespaceURI, el.nodeName);
$$2.each(el.attributes, function (i, attr) {
if (attr.localName !== '-moz-math-font-style') {
newEl.setAttributeNS(attr.namespaceURI, attr.nodeName, attr.value);
}
}); // set the copied element's new id
newEl.removeAttribute('id');
newEl.id = getNextId(); // Opera's "d" value needs to be reset for Opera/Win/non-EN
// Also needed for webkit (else does not keep curved segments on clone)
if (isWebkit() && el.nodeName === 'path') {
var fixedD = convertPath(el);
newEl.setAttribute('d', fixedD);
} // now create copies of all children
$$2.each(el.childNodes, function (i, child) {
switch (child.nodeType) {
case 1:
// element node
newEl.append(copyElem(child, getNextId));
break;
case 3:
// text node
newEl.textContent = child.nodeValue;
break;
}
});
if ($$2(el).data('gsvg')) {
$$2(newEl).data('gsvg', newEl.firstChild);
} else if ($$2(el).data('symbol')) {
var ref = $$2(el).data('symbol');
$$2(newEl).data('ref', ref).data('symbol', ref);
} else if (newEl.tagName === 'image') {
preventClickDefault(newEl);
}
return newEl;
};
/**
* Whether a value is `null` or `undefined`.
* @param {any} val
* @returns {boolean}
*/
var isNullish = function isNullish(val) {
return val === null || val === undefined;
};
var $$3 = jQuery;
/**
* This class encapsulates the concept of a layer in the drawing. It can be constructed with
* an existing group element or, with three parameters, will create a new layer group element.
*
* @example
* const l1 = new Layer('name', group); // Use the existing group for this layer.
* const l2 = new Layer('name', group, svgElem); // Create a new group and add it to the DOM after group.
* const l3 = new Layer('name', null, svgElem); // Create a new group and add it to the DOM as the last layer.
* @memberof module:layer
*/
var Layer = /*#__PURE__*/function () {
/**
* @param {string} name - Layer name
* @param {SVGGElement|null} group - An existing SVG group element or null.
* If group and no svgElem, use group for this layer.
* If group and svgElem, create a new group element and insert it in the DOM after group.
* If no group and svgElem, create a new group element and insert it in the DOM as the last layer.
* @param {SVGGElement} [svgElem] - The SVG DOM element. If defined, use this to add
* a new layer to the document.
*/
function Layer(name, group, svgElem) {
_classCallCheck(this, Layer);
this.name_ = name;
this.group_ = svgElem ? null : group;
if (svgElem) {
// Create a group element with title and add it to the DOM.
var svgdoc = svgElem.ownerDocument;
this.group_ = svgdoc.createElementNS(NS.SVG, 'g');
var layerTitle = svgdoc.createElementNS(NS.SVG, 'title');
layerTitle.textContent = name;
this.group_.append(layerTitle);
if (group) {
$$3(group).after(this.group_);
} else {
svgElem.append(this.group_);
}
}
addLayerClass(this.group_);
walkTree(this.group_, function (e) {
e.setAttribute('style', 'pointer-events:inherit');
});
this.group_.setAttribute('style', svgElem ? 'pointer-events:all' : 'pointer-events:none');
}
/**
* Get the layer's name.
* @returns {string} The layer name
*/
_createClass(Layer, [{
key: "getName",
value: function getName() {
return this.name_;
}
/**
* Get the group element for this layer.
* @returns {SVGGElement} The layer SVG group
*/
}, {
key: "getGroup",
value: function getGroup() {
return this.group_;
}
/**
* Active this layer so it takes pointer events.
* @returns {void}
*/
}, {
key: "activate",
value: function activate() {
this.group_.setAttribute('style', 'pointer-events:all');
}
/**
* Deactive this layer so it does NOT take pointer events.
* @returns {void}
*/
}, {
key: "deactivate",
value: function deactivate() {
this.group_.setAttribute('style', 'pointer-events:none');
}
/**
* Set this layer visible or hidden based on 'visible' parameter.
* @param {boolean} visible - If true, make visible; otherwise, hide it.
* @returns {void}
*/
}, {
key: "setVisible",
value: function setVisible(visible) {
var expected = visible === undefined || visible ? 'inline' : 'none';
var oldDisplay = this.group_.getAttribute('display');
if (oldDisplay !== expected) {
this.group_.setAttribute('display', expected);
}
}
/**
* Is this layer visible?
* @returns {boolean} True if visible.
*/
}, {
key: "isVisible",
value: function isVisible() {
return this.group_.getAttribute('display') !== 'none';
}
/**
* Get layer opacity.
* @returns {Float} Opacity value.
*/
}, {
key: "getOpacity",
value: function getOpacity() {
var opacity = this.group_.getAttribute('opacity');
if (isNullish(opacity)) {
return 1;
}
return parseFloat(opacity);
}
/**
* Sets the opacity of this layer. If opacity is not a value between 0.0 and 1.0,
* nothing happens.
* @param {Float} opacity - A float value in the range 0.0-1.0
* @returns {void}
*/
}, {
key: "setOpacity",
value: function setOpacity(opacity) {
if (typeof opacity === 'number' && opacity >= 0.0 && opacity <= 1.0) {
this.group_.setAttribute('opacity', opacity);
}
}
/**
* Append children to this layer.
* @param {SVGGElement} children - The children to append to this layer.
* @returns {void}
*/
}, {
key: "appendChildren",
value: function appendChildren(children) {
var _iteratorNormalCompletion = true;
var _didIteratorError = false;
var _iteratorError = undefined;
try {
for (var _iterator = children[Symbol.iterator](), _step; !(_iteratorNormalCompletion = (_step = _iterator.next()).done); _iteratorNormalCompletion = true) {
var child = _step.value;
this.group_.append(child);
}
} catch (err) {
_didIteratorError = true;
_iteratorError = err;
} finally {
try {
if (!_iteratorNormalCompletion && _iterator["return"] != null) {
_iterator["return"]();
}
} finally {
if (_didIteratorError) {
throw _iteratorError;
}
}
}
}
/**
* @returns {SVGTitleElement|null}
*/
}, {
key: "getTitleElement",
value: function getTitleElement() {
var len = this.group_.childNodes.length;
for (var i = 0; i < len; ++i) {
var child = this.group_.childNodes.item(i);
if (child && child.tagName === 'title') {
return child;
}
}
return null;
}
/**
* Set the name of this layer.
* @param {string} name - The new name.
* @param {module:history.HistoryRecordingService} hrService - History recording service
* @returns {string|null} The new name if changed; otherwise, null.
*/
}, {
key: "setName",
value: function setName(name, hrService) {
var previousName = this.name_;
name = toXml(name); // now change the underlying title element contents
var title = this.getTitleElement();
if (title) {
$$3(title).empty();
title.textContent = name;
this.name_ = name;
if (hrService) {
hrService.changeElement(title, {
'#text': previousName
});
}
return this.name_;
}
return null;
}
/**
* Remove this layer's group from the DOM. No more functions on group can be called after this.
* @returns {SVGGElement} The layer SVG group that was just removed.
*/
}, {
key: "removeGroup",
value: function removeGroup() {
var parent = this.group_.parentNode;
var group = parent.removeChild(this.group_);
this.group_ = undefined;
return group;
}
}]);
return Layer;
}();
/**
* @property {string} CLASS_NAME - class attribute assigned to all layer groups.
*/
Layer.CLASS_NAME = 'layer';
/**
* @property {RegExp} CLASS_REGEX - Used to test presence of class Layer.CLASS_NAME
*/
Layer.CLASS_REGEX = new RegExp('(\\s|^)' + Layer.CLASS_NAME + '(\\s|$)');
/**
* Add class `Layer.CLASS_NAME` to the element (usually `class='layer'`).
*
* @param {SVGGElement} elem - The SVG element to update
* @returns {void}
*/
function addLayerClass(elem) {
var classes = elem.getAttribute('class');
if (isNullish(classes) || !classes.length) {
elem.setAttribute('class', Layer.CLASS_NAME);
} else if (!Layer.CLASS_REGEX.test(classes)) {
elem.setAttribute('class', classes + ' ' + Layer.CLASS_NAME);
}
}
/**
* History recording service.
*
* A self-contained service interface for recording history. Once injected, no other dependencies
* or globals are required (example: UndoManager, command types, etc.). Easy to mock for unit tests.
* Built on top of history classes in history.js.
*
* There is a simple start/end interface for batch commands.
*
* HistoryRecordingService.NO_HISTORY is a singleton that can be passed in to functions
* that record history. This helps when the caller requires that no history be recorded.
*
* The following will record history: insert, batch, insert.
* @example
* hrService = new history.HistoryRecordingService(this.undoMgr);
* hrService.insertElement(elem, text); // add simple command to history.
* hrService.startBatchCommand('create two elements');
* hrService.changeElement(elem, attrs, text); // add to batchCommand
* hrService.changeElement(elem, attrs2, text); // add to batchCommand
* hrService.endBatchCommand(); // add batch command with two change commands to history.
* hrService.insertElement(elem, text); // add simple command to history.
*
* @example
* // Note that all functions return this, so commands can be chained, like so:
* hrService
* .startBatchCommand('create two elements')
* .insertElement(elem, text)
* .changeElement(elem, attrs, text)
* .endBatchCommand();
*
* @memberof module:history
*/
var HistoryRecordingService = /*#__PURE__*/function () {
/**
* @param {history.UndoManager|null} undoManager - The undo manager.
* A value of `null` is valid for cases where no history recording is required.
* See singleton: {@link module:history.HistoryRecordingService.HistoryRecordingService.NO_HISTORY}
*/
function HistoryRecordingService(undoManager) {
_classCallCheck(this, HistoryRecordingService);
this.undoManager_ = undoManager;
this.currentBatchCommand_ = null;
this.batchCommandStack_ = [];
}
/**
* Start a batch command so multiple commands can recorded as a single history command.
* Requires a corresponding call to endBatchCommand. Start and end commands can be nested.
*
* @param {string} text - Optional string describing the batch command.
* @returns {module:history.HistoryRecordingService}
*/
_createClass(HistoryRecordingService, [{
key: "startBatchCommand",
value: function startBatchCommand(text) {
if (!this.undoManager_) {
return this;
}
this.currentBatchCommand_ = new BatchCommand(text);
this.batchCommandStack_.push(this.currentBatchCommand_);
return this;
}
/**
* End a batch command and add it to the history or a parent batch command.
* @returns {module:history.HistoryRecordingService}
*/
}, {
key: "endBatchCommand",
value: function endBatchCommand() {
if (!this.undoManager_) {
return this;
}
if (this.currentBatchCommand_) {
var batchCommand = this.currentBatchCommand_;
this.batchCommandStack_.pop();
var len = this.batchCommandStack_.length;
this.currentBatchCommand_ = len ? this.batchCommandStack_[len - 1] : null;
this.addCommand_(batchCommand);
}
return this;
}
/**
* Add a `MoveElementCommand` to the history or current batch command.
* @param {Element} elem - The DOM element that was moved
* @param {Element} oldNextSibling - The element's next sibling before it was moved
* @param {Element} oldParent - The element's parent before it was moved
* @param {string} [text] - An optional string visible to user related to this change
* @returns {module:history.HistoryRecordingService}
*/
}, {
key: "moveElement",
value: function moveElement(elem, oldNextSibling, oldParent, text) {
if (!this.undoManager_) {
return this;
}
this.addCommand_(new MoveElementCommand(elem, oldNextSibling, oldParent, text));
return this;
}
/**
* Add an `InsertElementCommand` to the history or current batch command.
* @param {Element} elem - The DOM element that was added
* @param {string} [text] - An optional string visible to user related to this change
* @returns {module:history.HistoryRecordingService}
*/
}, {
key: "insertElement",
value: function insertElement(elem, text) {
if (!this.undoManager_) {
return this;
}
this.addCommand_(new InsertElementCommand(elem, text));
return this;
}
/**
* Add a `RemoveElementCommand` to the history or current batch command.
* @param {Element} elem - The DOM element that was removed
* @param {Element} oldNextSibling - The element's next sibling before it was removed
* @param {Element} oldParent - The element's parent before it was removed
* @param {string} [text] - An optional string visible to user related to this change
* @returns {module:history.HistoryRecordingService}
*/
}, {
key: "removeElement",
value: function removeElement(elem, oldNextSibling, oldParent, text) {
if (!this.undoManager_) {
return this;
}
this.addCommand_(new RemoveElementCommand(elem, oldNextSibling, oldParent, text));
return this;
}
/**
* Add a `ChangeElementCommand` to the history or current batch command.
* @param {Element} elem - The DOM element that was changed
* @param {module:history.CommandAttributes} attrs - An object with the attributes to be changed and the values they had *before* the change
* @param {string} [text] - An optional string visible to user related to this change
* @returns {module:history.HistoryRecordingService}
*/
}, {
key: "changeElement",
value: function changeElement(elem, attrs, text) {
if (!this.undoManager_) {
return this;
}
this.addCommand_(new ChangeElementCommand(elem, attrs, text));
return this;
}
/**
* Private function to add a command to the history or current batch command.
* @private
* @param {Command} cmd
* @returns {module:history.HistoryRecordingService|void}
*/
}, {
key: "addCommand_",
value: function addCommand_(cmd) {
if (!this.undoManager_) {
return this;
}
if (this.currentBatchCommand_) {
this.currentBatchCommand_.addSubCommand(cmd);
} else {
this.undoManager_.addCommandToHistory(cmd);
}
return undefined;
}
}]);
return HistoryRecordingService;
}();
/**
* @memberof module:history.HistoryRecordingService
* @property {module:history.HistoryRecordingService} NO_HISTORY - Singleton that can be passed to functions that record history, but the caller requires that no history be recorded.
*/
HistoryRecordingService.NO_HISTORY = new HistoryRecordingService();
var $$4 = jQuery;
var visElems$1 = 'a,circle,ellipse,foreignObject,g,image,line,path,polygon,polyline,rect,svg,text,tspan,use'.split(',');
var RandomizeModes = {
LET_DOCUMENT_DECIDE: 0,
ALWAYS_RANDOMIZE: 1,
NEVER_RANDOMIZE: 2
};
var randIds = RandomizeModes.LET_DOCUMENT_DECIDE; // Array with current disabled elements (for in-group editing)
var disabledElems = [];
/**
* Get a HistoryRecordingService.
* @param {module:history.HistoryRecordingService} [hrService] - if exists, return it instead of creating a new service.
* @returns {module:history.HistoryRecordingService}
*/
function historyRecordingService(hrService) {
return hrService || new HistoryRecordingService(canvas_.undoMgr);
}
/**
* Find the layer name in a group element.
* @param {Element} group The group element to search in.
* @returns {string} The layer name or empty string.
*/
function findLayerNameInGroup(group) {
return $$4('title', group).text() || (isOpera() && group.querySelectorAll // Hack for Opera 10.60
? $$4(group.querySelectorAll('title')).text() : '');
}
/**
* Given a set of names, return a new unique name.
* @param {string[]} existingLayerNames - Existing layer names.
* @returns {string} - The new name.
*/
function getNewLayerName(existingLayerNames) {
var i = 1; // TODO(codedread): What about internationalization of "Layer"?
while (existingLayerNames.includes('Layer ' + i)) {
i++;
}
return 'Layer ' + i;
}
/**
* This class encapsulates the concept of a SVG-edit drawing.
*/
var Drawing = /*#__PURE__*/function () {
/**
* @param {SVGSVGElement} svgElem - The SVG DOM Element that this JS object
* encapsulates. If the svgElem has a se:nonce attribute on it, then
* IDs will use the nonce as they are generated.
* @param {string} [optIdPrefix=svg_] - The ID prefix to use.
* @throws {Error} If not initialized with an SVG element
*/
function Drawing(svgElem, optIdPrefix) {
_classCallCheck(this, Drawing);
if (!svgElem || !svgElem.tagName || !svgElem.namespaceURI || svgElem.tagName !== 'svg' || svgElem.namespaceURI !== NS.SVG) {
throw new Error('Error: svgedit.draw.Drawing instance initialized without a <svg> element');
}
/**
* The SVG DOM Element that represents this drawing.
* @type {SVGSVGElement}
*/
this.svgElem_ = svgElem;
/**
* The latest object number used in this drawing.
* @type {Integer}
*/
this.obj_num = 0;
/**
* The prefix to prepend to each element id in the drawing.
* @type {string}
*/
this.idPrefix = optIdPrefix || 'svg_';
/**
* An array of released element ids to immediately reuse.
* @type {Integer[]}
*/
this.releasedNums = [];
/**
* The z-ordered array of Layer objects. Each layer has a name
* and group element.
* The first layer is the one at the bottom of the rendering.
* @type {Layer[]}
*/
this.all_layers = [];
/**
* Map of all_layers by name.
*
* Note: Layers are ordered, but referenced externally by name; so, we need both container
* types depending on which function is called (i.e. all_layers and layer_map).
*
* @type {PlainObject<string, Layer>}
*/
this.layer_map = {};
/**
* The current layer being used.
* @type {Layer}
*/
this.current_layer = null;
/**
* The nonce to use to uniquely identify elements across drawings.
* @type {!string}
*/
this.nonce_ = '';
var n = this.svgElem_.getAttributeNS(NS.SE, 'nonce'); // If already set in the DOM, use the nonce throughout the document
// else, if randomizeIds(true) has been called, create and set the nonce.
if (n && randIds !== RandomizeModes.NEVER_RANDOMIZE) {
this.nonce_ = n;
} else if (randIds === RandomizeModes.ALWAYS_RANDOMIZE) {
this.setNonce(Math.floor(Math.random() * 100001));
}
}
/**
* @param {string} id Element ID to retrieve
* @returns {Element} SVG element within the root SVGSVGElement
*/
_createClass(Drawing, [{
key: "getElem_",
value: function getElem_(id) {
if (this.svgElem_.querySelector) {
// querySelector lookup
return this.svgElem_.querySelector('#' + id);
} // jQuery lookup: twice as slow as xpath in FF
return $$4(this.svgElem_).find('[id=' + id + ']')[0];
}
/**
* @returns {SVGSVGElement}
*/
}, {
key: "getSvgElem",
value: function getSvgElem() {
return this.svgElem_;
}
/**
* @returns {!(string|Integer)} The previously set nonce
*/
}, {
key: "getNonce",
value: function getNonce() {
return this.nonce_;
}
/**
* @param {!(string|Integer)} n The nonce to set
* @returns {void}
*/
}, {
key: "setNonce",
value: function setNonce(n) {
this.svgElem_.setAttributeNS(NS.XMLNS, 'xmlns:se', NS.SE);
this.svgElem_.setAttributeNS(NS.SE, 'se:nonce', n);
this.nonce_ = n;
}
/**
* Clears any previously set nonce.
* @returns {void}
*/
}, {
key: "clearNonce",
value: function clearNonce() {
// We deliberately leave any se:nonce attributes alone,
// we just don't use it to randomize ids.
this.nonce_ = '';
}
/**
* Returns the latest object id as a string.
* @returns {string} The latest object Id.
*/
}, {
key: "getId",
value: function getId() {
return this.nonce_ ? this.idPrefix + this.nonce_ + '_' + this.obj_num : this.idPrefix + this.obj_num;
}
/**
* Returns the next object Id as a string.
* @returns {string} The next object Id to use.
*/
}, {
key: "getNextId",
value: function getNextId() {
var oldObjNum = this.obj_num;
var restoreOldObjNum = false; // If there are any released numbers in the release stack,
// use the last one instead of the next obj_num.
// We need to temporarily use obj_num as that is what getId() depends on.
if (this.releasedNums.length > 0) {
this.obj_num = this.releasedNums.pop();
restoreOldObjNum = true;
} else {
// If we are not using a released id, then increment the obj_num.
this.obj_num++;
} // Ensure the ID does not exist.
var id = this.getId();
while (this.getElem_(id)) {
if (restoreOldObjNum) {
this.obj_num = oldObjNum;
restoreOldObjNum = false;
}
this.obj_num++;
id = this.getId();
} // Restore the old object number if required.
if (restoreOldObjNum) {
this.obj_num = oldObjNum;
}
return id;
}
/**
* Releases the object Id, letting it be used as the next id in getNextId().
* This method DOES NOT remove any elements from the DOM, it is expected
* that client code will do this.
* @param {string} id - The id to release.
* @returns {boolean} True if the id was valid to be released, false otherwise.
*/
}, {
key: "releaseId",
value: function releaseId(id) {
// confirm if this is a valid id for this Document, else return false
var front = this.idPrefix + (this.nonce_ ? this.nonce_ + '_' : '');
if (typeof id !== 'string' || !id.startsWith(front)) {
return false;
} // extract the obj_num of this id
var num = parseInt(id.substr(front.length)); // if we didn't get a positive number or we already released this number
// then return false.
if (typeof num !== 'number' || num <= 0 || this.releasedNums.includes(num)) {
return false;
} // push the released number into the released queue
this.releasedNums.push(num);
return true;
}
/**
* Returns the number of layers in the current drawing.
* @returns {Integer} The number of layers in the current drawing.
*/
}, {
key: "getNumLayers",
value: function getNumLayers() {
return this.all_layers.length;
}
/**
* Check if layer with given name already exists.
* @param {string} name - The layer name to check
* @returns {boolean}
*/
}, {
key: "hasLayer",
value: function hasLayer(name) {
return this.layer_map[name] !== undefined;
}
/**
* Returns the name of the ith layer. If the index is out of range, an empty string is returned.
* @param {Integer} i - The zero-based index of the layer you are querying.
* @returns {string} The name of the ith layer (or the empty string if none found)
*/
}, {
key: "getLayerName",
value: function getLayerName(i) {
return i >= 0 && i < this.getNumLayers() ? this.all_layers[i].getName() : '';
}
/**
* @returns {SVGGElement|null} The SVGGElement representing the current layer.
*/
}, {
key: "getCurrentLayer",
value: function getCurrentLayer() {
return this.current_layer ? this.current_layer.getGroup() : null;
}
/**
* Get a layer by name.
* @param {string} name
* @returns {SVGGElement} The SVGGElement representing the named layer or null.
*/
}, {
key: "getLayerByName",
value: function getLayerByName(name) {
var layer = this.layer_map[name];
return layer ? layer.getGroup() : null;
}
/**
* Returns the name of the currently selected layer. If an error occurs, an empty string
* is returned.
* @returns {string} The name of the currently active layer (or the empty string if none found).
*/
}, {
key: "getCurrentLayerName",
value: function getCurrentLayerName() {
return this.current_layer ? this.current_layer.getName() : '';
}
/**
* Set the current layer's name.
* @param {string} name - The new name.
* @param {module:history.HistoryRecordingService} hrService - History recording service
* @returns {string|null} The new name if changed; otherwise, null.
*/
}, {
key: "setCurrentLayerName",
value: function setCurrentLayerName(name, hrService) {
var finalName = null;
if (this.current_layer) {
var oldName = this.current_layer.getName();
finalName = this.current_layer.setName(name, hrService);
if (finalName) {
delete this.layer_map[oldName];
this.layer_map[finalName] = this.current_layer;
}
}
return finalName;
}
/**
* Set the current layer's position.
* @param {Integer} newpos - The zero-based index of the new position of the layer. Range should be 0 to layers-1
* @returns {{title: SVGGElement, previousName: string}|null} If the name was changed, returns {title:SVGGElement, previousName:string}; otherwise null.
*/
}, {
key: "setCurrentLayerPosition",
value: function setCurrentLayerPosition(newpos) {
var layerCount = this.getNumLayers();
if (!this.current_layer || newpos < 0 || newpos >= layerCount) {
return null;
}
var oldpos;
for (oldpos = 0; oldpos < layerCount; ++oldpos) {
if (this.all_layers[oldpos] === this.current_layer) {
break;
}
} // some unknown error condition (current_layer not in all_layers)
if (oldpos === layerCount) {
return null;
}
if (oldpos !== newpos) {
// if our new position is below us, we need to insert before the node after newpos
var currentGroup = this.current_layer.getGroup();
var oldNextSibling = currentGroup.nextSibling;
var refGroup = null;
if (newpos > oldpos) {
if (newpos < layerCount - 1) {
refGroup = this.all_layers[newpos + 1].getGroup();
} // if our new position is above us, we need to insert before the node at newpos
} else {
refGroup = this.all_layers[newpos].getGroup();
}
this.svgElem_.insertBefore(currentGroup, refGroup); // Ok to replace with `refGroup.before(currentGroup);`?
this.identifyLayers();
this.setCurrentLayer(this.getLayerName(newpos));
return {
currentGroup: currentGroup,
oldNextSibling: oldNextSibling
};
}
return null;
}
/**
* @param {module:history.HistoryRecordingService} hrService
* @returns {void}
*/
}, {
key: "mergeLayer",
value: function mergeLayer(hrService) {
var currentGroup = this.current_layer.getGroup();
var prevGroup = $$4(currentGroup).prev()[0];
if (!prevGroup) {
return;
}
hrService.startBatchCommand('Merge Layer');
var layerNextSibling = currentGroup.nextSibling;
hrService.removeElement(currentGroup, layerNextSibling, this.svgElem_);
while (currentGroup.firstChild) {
var child = currentGroup.firstChild;
if (child.localName === 'title') {
hrService.removeElement(child, child.nextSibling, currentGroup);
child.remove();
continue;
}
var oldNextSibling = child.nextSibling;
prevGroup.append(child);
hrService.moveElement(child, oldNextSibling, currentGroup);
} // Remove current layer's group
this.current_layer.removeGroup(); // Remove the current layer and set the previous layer as the new current layer
var index = this.all_layers.indexOf(this.current_layer);
if (index > 0) {
var _name = this.current_layer.getName();
this.current_layer = this.all_layers[index - 1];
this.all_layers.splice(index, 1);
delete this.layer_map[_name];
}
hrService.endBatchCommand();
}
/**
* @param {module:history.HistoryRecordingService} hrService
* @returns {void}
*/
}, {
key: "mergeAllLayers",
value: function mergeAllLayers(hrService) {
// Set the current layer to the last layer.
this.current_layer = this.all_layers[this.all_layers.length - 1];
hrService.startBatchCommand('Merge all Layers');
while (this.all_layers.length > 1) {
this.mergeLayer(hrService);
}
hrService.endBatchCommand();
}
/**
* Sets the current layer. If the name is not a valid layer name, then this
* function returns `false`. Otherwise it returns `true`. This is not an
* undo-able action.
* @param {string} name - The name of the layer you want to switch to.
* @returns {boolean} `true` if the current layer was switched, otherwise `false`
*/
}, {
key: "setCurrentLayer",
value: function setCurrentLayer(name) {
var layer = this.layer_map[name];
if (layer) {
if (this.current_layer) {
this.current_layer.deactivate();
}
this.current_layer = layer;
this.current_layer.activate();
return true;
}
return false;
}
/**
* Deletes the current layer from the drawing and then clears the selection.
* This function then calls the 'changed' handler. This is an undoable action.
* @todo Does this actually call the 'changed' handler?
* @returns {SVGGElement} The SVGGElement of the layer removed or null.
*/
}, {
key: "deleteCurrentLayer",
value: function deleteCurrentLayer() {
if (this.current_layer && this.getNumLayers() > 1) {
var oldLayerGroup = this.current_layer.removeGroup();
this.identifyLayers();
return oldLayerGroup;
}
return null;
}
/**
* Updates layer system and sets the current layer to the
* top-most layer (last `<g>` child of this drawing).
* @returns {void}
*/
}, {
key: "identifyLayers",
value: function identifyLayers() {
this.all_layers = [];
this.layer_map = {};
var numchildren = this.svgElem_.childNodes.length; // loop through all children of SVG element
var orphans = [],
layernames = [];
var layer = null;
var childgroups = false;
for (var i = 0; i < numchildren; ++i) {
var child = this.svgElem_.childNodes.item(i); // for each g, find its layer name
if (child && child.nodeType === 1) {
if (child.tagName === 'g') {
childgroups = true;
var _name2 = findLayerNameInGroup(child);
if (_name2) {
layernames.push(_name2);
layer = new Layer(_name2, child);
this.all_layers.push(layer);
this.layer_map[_name2] = layer;
} else {
// if group did not have a name, it is an orphan
orphans.push(child);
}
} else if (visElems$1.includes(child.nodeName)) {
// Child is "visible" (i.e. not a <title> or <defs> element), so it is an orphan
orphans.push(child);
}
}
} // If orphans or no layers found, create a new layer and add all the orphans to it
if (orphans.length > 0 || !childgroups) {
layer = new Layer(getNewLayerName(layernames), null, this.svgElem_);
layer.appendChildren(orphans);
this.all_layers.push(layer);
this.layer_map[name] = layer;
} else {
layer.activate();
}
this.current_layer = layer;
}
/**
* Creates a new top-level layer in the drawing with the given name and
* makes it the current layer.
* @param {string} name - The given name. If the layer name exists, a new name will be generated.
* @param {module:history.HistoryRecordingService} hrService - History recording service
* @returns {SVGGElement} The SVGGElement of the new layer, which is
* also the current layer of this drawing.
*/
}, {
key: "createLayer",
value: function createLayer(name, hrService) {
if (this.current_layer) {
this.current_layer.deactivate();
} // Check for duplicate name.
if (name === undefined || name === null || name === '' || this.layer_map[name]) {
name = getNewLayerName(Object.keys(this.layer_map));
} // Crate new layer and add to DOM as last layer
var layer = new Layer(name, null, this.svgElem_); // Like to assume hrService exists, but this is backwards compatible with old version of createLayer.
if (hrService) {
hrService.startBatchCommand('Create Layer');
hrService.insertElement(layer.getGroup());
hrService.endBatchCommand();
}
this.all_layers.push(layer);
this.layer_map[name] = layer;
this.current_layer = layer;
return layer.getGroup();
}
/**
* Creates a copy of the current layer with the given name and makes it the current layer.
* @param {string} name - The given name. If the layer name exists, a new name will be generated.
* @param {module:history.HistoryRecordingService} hrService - History recording service
* @returns {SVGGElement} The SVGGElement of the new layer, which is
* also the current layer of this drawing.
*/
}, {
key: "cloneLayer",
value: function cloneLayer(name, hrService) {
var _this = this;
if (!this.current_layer) {
return null;
}
this.current_layer.deactivate(); // Check for duplicate name.
if (name === undefined || name === null || name === '' || this.layer_map[name]) {
name = getNewLayerName(Object.keys(this.layer_map));
} // Create new group and add to DOM just after current_layer
var currentGroup = this.current_layer.getGroup();
var layer = new Layer(name, currentGroup, this.svgElem_);
var group = layer.getGroup(); // Clone children
var children = _toConsumableArray(currentGroup.childNodes);
children.forEach(function (child) {
if (child.localName === 'title') {
return;
}
group.append(_this.copyElem(child));
});
if (hrService) {
hrService.startBatchCommand('Duplicate Layer');
hrService.insertElement(group);
hrService.endBatchCommand();
} // Update layer containers and current_layer.
var index = this.all_layers.indexOf(this.current_layer);
if (index >= 0) {
this.all_layers.splice(index + 1, 0, layer);
} else {
this.all_layers.push(layer);
}
this.layer_map[name] = layer;
this.current_layer = layer;
return group;
}
/**
* Returns whether the layer is visible. If the layer name is not valid,
* then this function returns `false`.
* @param {string} layerName - The name of the layer which you want to query.
* @returns {boolean} The visibility state of the layer, or `false` if the layer name was invalid.
*/
}, {
key: "getLayerVisibility",
value: function getLayerVisibility(layerName) {
var layer = this.layer_map[layerName];
return layer ? layer.isVisible() : false;
}
/**
* Sets the visibility of the layer. If the layer name is not valid, this
* function returns `null`, otherwise it returns the `SVGElement` representing
* the layer. This is an undo-able action.
* @param {string} layerName - The name of the layer to change the visibility
* @param {boolean} bVisible - Whether the layer should be visible
* @returns {?SVGGElement} The SVGGElement representing the layer if the
* `layerName` was valid, otherwise `null`.
*/
}, {
key: "setLayerVisibility",
value: function setLayerVisibility(layerName, bVisible) {
if (typeof bVisible !== 'boolean') {
return null;
}
var layer = this.layer_map[layerName];
if (!layer) {
return null;
}
layer.setVisible(bVisible);
return layer.getGroup();
}
/**
* Returns the opacity of the given layer. If the input name is not a layer, `null` is returned.
* @param {string} layerName - name of the layer on which to get the opacity
* @returns {?Float} The opacity value of the given layer. This will be a value between 0.0 and 1.0, or `null`
* if `layerName` is not a valid layer
*/
}, {
key: "getLayerOpacity",
value: function getLayerOpacity(layerName) {
var layer = this.layer_map[layerName];
if (!layer) {
return null;
}
return layer.getOpacity();
}
/**
* Sets the opacity of the given layer. If the input name is not a layer,
* nothing happens. If opacity is not a value between 0.0 and 1.0, then
* nothing happens.
* NOTE: this function exists solely to apply a highlighting/de-emphasis
* effect to a layer. When it is possible for a user to affect the opacity
* of a layer, we will need to allow this function to produce an undo-able
* action.
* @param {string} layerName - Name of the layer on which to set the opacity
* @param {Float} opacity - A float value in the range 0.0-1.0
* @returns {void}
*/
}, {
key: "setLayerOpacity",
value: function setLayerOpacity(layerName, opacity) {
if (typeof opacity !== 'number' || opacity < 0.0 || opacity > 1.0) {
return;
}
var layer = this.layer_map[layerName];
if (layer) {
layer.setOpacity(opacity);
}
}
/**
* Create a clone of an element, updating its ID and its children's IDs when needed.
* @param {Element} el - DOM element to clone
* @returns {Element}
*/
}, {
key: "copyElem",
value: function copyElem$1(el) {
var that = this;
var getNextIdClosure = function getNextIdClosure() {
return that.getNextId();
};
return copyElem(el, getNextIdClosure);
}
}]);
return Drawing;
}();
/**
* Called to ensure that drawings will or will not have randomized ids.
* The currentDrawing will have its nonce set if it doesn't already.
* @function module:draw.randomizeIds
* @param {boolean} enableRandomization - flag indicating if documents should have randomized ids
* @param {draw.Drawing} currentDrawing
* @returns {void}
*/
var randomizeIds = function randomizeIds(enableRandomization, currentDrawing) {
randIds = enableRandomization === false ? RandomizeModes.NEVER_RANDOMIZE : RandomizeModes.ALWAYS_RANDOMIZE;
if (randIds === RandomizeModes.ALWAYS_RANDOMIZE && !currentDrawing.getNonce()) {
currentDrawing.setNonce(Math.floor(Math.random() * 100001));
} else if (randIds === RandomizeModes.NEVER_RANDOMIZE && currentDrawing.getNonce()) {
currentDrawing.clearNonce();
}
}; // Layer API Functions
/**
* Group: Layers.
*/
/**
* @see {@link https://api.jquery.com/jQuery.data/}
* @name external:jQuery.data
*/
/**
* @interface module:draw.DrawCanvasInit
* @property {module:path.pathActions} pathActions
* @property {external:jQuery.data} elData
* @property {module:history.UndoManager} undoMgr
*/
/**
* @function module:draw.DrawCanvasInit#getCurrentGroup
* @returns {Element}
*/
/**
* @function module:draw.DrawCanvasInit#setCurrentGroup
* @param {Element} cg
* @returns {void}
*/
/**
* @function module:draw.DrawCanvasInit#getSelectedElements
* @returns {Element[]} the array with selected DOM elements
*/
/**
* @function module:draw.DrawCanvasInit#getSVGContent
* @returns {SVGSVGElement}
*/
/**
* @function module:draw.DrawCanvasInit#getCurrentDrawing
* @returns {module:draw.Drawing}
*/
/**
* @function module:draw.DrawCanvasInit#clearSelection
* @param {boolean} [noCall] - When `true`, does not call the "selected" handler
* @returns {void}
*/
/**
* Run the callback function associated with the given event.
* @function module:draw.DrawCanvasInit#call
* @param {"changed"|"contextset"} ev - String with the event name
* @param {module:svgcanvas.SvgCanvas#event:changed|module:svgcanvas.SvgCanvas#event:contextset} arg - Argument to pass through to the callback
* function. If the event is "changed", a (single-item) array of `Element`s is
* passed. If the event is "contextset", the arg is `null` or `Element`.
* @returns {void}
*/
/**
* @function module:draw.DrawCanvasInit#addCommandToHistory
* @param {Command} cmd
* @returns {void}
*/
/**
* @function module:draw.DrawCanvasInit#changeSVGContent
* @returns {void}
*/
var canvas_;
/**
* @function module:draw.init
* @param {module:draw.DrawCanvasInit} canvas
* @returns {void}
*/
var init$3 = function init(canvas) {
canvas_ = canvas;
};
/**
* Updates layer system.
* @function module:draw.identifyLayers
* @returns {void}
*/
var identifyLayers = function identifyLayers() {
leaveContext();
canvas_.getCurrentDrawing().identifyLayers();
};
/**
* Creates a new top-level layer in the drawing with the given name, sets the current layer
* to it, and then clears the selection. This function then calls the 'changed' handler.
* This is an undoable action.
* @function module:draw.createLayer
* @param {string} name - The given name
* @param {module:history.HistoryRecordingService} hrService
* @fires module:svgcanvas.SvgCanvas#event:changed
* @returns {void}
*/
var createLayer = function createLayer(name, hrService) {
var newLayer = canvas_.getCurrentDrawing().createLayer(name, historyRecordingService(hrService));
canvas_.clearSelection();
canvas_.call('changed', [newLayer]);
};
/**
* Creates a new top-level layer in the drawing with the given name, copies all the current layer's contents
* to it, and then clears the selection. This function then calls the 'changed' handler.
* This is an undoable action.
* @function module:draw.cloneLayer
* @param {string} name - The given name. If the layer name exists, a new name will be generated.
* @param {module:history.HistoryRecordingService} hrService - History recording service
* @fires module:svgcanvas.SvgCanvas#event:changed
* @returns {void}
*/
var cloneLayer = function cloneLayer(name, hrService) {
// Clone the current layer and make the cloned layer the new current layer
var newLayer = canvas_.getCurrentDrawing().cloneLayer(name, historyRecordingService(hrService));
canvas_.clearSelection();
leaveContext();
canvas_.call('changed', [newLayer]);
};
/**
* Deletes the current layer from the drawing and then clears the selection. This function
* then calls the 'changed' handler. This is an undoable action.
* @function module:draw.deleteCurrentLayer
* @fires module:svgcanvas.SvgCanvas#event:changed
* @returns {boolean} `true` if an old layer group was found to delete
*/
var deleteCurrentLayer = function deleteCurrentLayer() {
var currentLayer = canvas_.getCurrentDrawing().getCurrentLayer();
var _currentLayer = currentLayer,
nextSibling = _currentLayer.nextSibling;
var parent = currentLayer.parentNode;
currentLayer = canvas_.getCurrentDrawing().deleteCurrentLayer();
if (currentLayer) {
var batchCmd = new BatchCommand('Delete Layer'); // store in our Undo History
batchCmd.addSubCommand(new RemoveElementCommand(currentLayer, nextSibling, parent));
canvas_.addCommandToHistory(batchCmd);
canvas_.clearSelection();
canvas_.call('changed', [parent]);
return true;
}
return false;
};
/**
* Sets the current layer. If the name is not a valid layer name, then this function returns
* false. Otherwise it returns true. This is not an undo-able action.
* @function module:draw.setCurrentLayer
* @param {string} name - The name of the layer you want to switch to.
* @returns {boolean} true if the current layer was switched, otherwise false
*/
var setCurrentLayer = function setCurrentLayer(name) {
var result = canvas_.getCurrentDrawing().setCurrentLayer(toXml(name));
if (result) {
canvas_.clearSelection();
}
return result;
};
/**
* Renames the current layer. If the layer name is not valid (i.e. unique), then this function
* does nothing and returns `false`, otherwise it returns `true`. This is an undo-able action.
* @function module:draw.renameCurrentLayer
* @param {string} newName - the new name you want to give the current layer. This name must
* be unique among all layer names.
* @fires module:svgcanvas.SvgCanvas#event:changed
* @returns {boolean} Whether the rename succeeded
*/
var renameCurrentLayer = function renameCurrentLayer(newName) {
var drawing = canvas_.getCurrentDrawing();
var layer = drawing.getCurrentLayer();
if (layer) {
var result = drawing.setCurrentLayerName(newName, historyRecordingService());
if (result) {
canvas_.call('changed', [layer]);
return true;
}
}
return false;
};
/**
* Changes the position of the current layer to the new value. If the new index is not valid,
* this function does nothing and returns false, otherwise it returns true. This is an
* undo-able action.
* @function module:draw.setCurrentLayerPosition
* @param {Integer} newPos - The zero-based index of the new position of the layer. This should be between
* 0 and (number of layers - 1)
* @returns {boolean} `true` if the current layer position was changed, `false` otherwise.
*/
var setCurrentLayerPosition = function setCurrentLayerPosition(newPos) {
var drawing = canvas_.getCurrentDrawing();
var result = drawing.setCurrentLayerPosition(newPos);
if (result) {
canvas_.addCommandToHistory(new MoveElementCommand(result.currentGroup, result.oldNextSibling, canvas_.getSVGContent()));
return true;
}
return false;
};
/**
* Sets the visibility of the layer. If the layer name is not valid, this function return
* `false`, otherwise it returns `true`. This is an undo-able action.
* @function module:draw.setLayerVisibility
* @param {string} layerName - The name of the layer to change the visibility
* @param {boolean} bVisible - Whether the layer should be visible
* @returns {boolean} true if the layer's visibility was set, false otherwise
*/
var setLayerVisibility = function setLayerVisibility(layerName, bVisible) {
var drawing = canvas_.getCurrentDrawing();
var prevVisibility = drawing.getLayerVisibility(layerName);
var layer = drawing.setLayerVisibility(layerName, bVisible);
if (layer) {
var oldDisplay = prevVisibility ? 'inline' : 'none';
canvas_.addCommandToHistory(new ChangeElementCommand(layer, {
display: oldDisplay
}, 'Layer Visibility'));
} else {
return false;
}
if (layer === drawing.getCurrentLayer()) {
canvas_.clearSelection();
canvas_.pathActions.clear();
} // call('changed', [selected]);
return true;
};
/**
* Moves the selected elements to layerName. If the name is not a valid layer name, then `false`
* is returned. Otherwise it returns `true`. This is an undo-able action.
* @function module:draw.moveSelectedToLayer
* @param {string} layerName - The name of the layer you want to which you want to move the selected elements
* @returns {boolean} Whether the selected elements were moved to the layer.
*/
var moveSelectedToLayer = function moveSelectedToLayer(layerName) {
// find the layer
var drawing = canvas_.getCurrentDrawing();
var layer = drawing.getLayerByName(layerName);
if (!layer) {
return false;
}
var batchCmd = new BatchCommand('Move Elements to Layer'); // loop for each selected element and move it
var selElems = canvas_.getSelectedElements();
var i = selElems.length;
while (i--) {
var elem = selElems[i];
if (!elem) {
continue;
}
var oldNextSibling = elem.nextSibling; // TODO: this is pretty brittle!
var oldLayer = elem.parentNode;
layer.append(elem);
batchCmd.addSubCommand(new MoveElementCommand(elem, oldNextSibling, oldLayer));
}
canvas_.addCommandToHistory(batchCmd);
return true;
};
/**
* @function module:draw.mergeLayer
* @param {module:history.HistoryRecordingService} hrService
* @returns {void}
*/
var mergeLayer = function mergeLayer(hrService) {
canvas_.getCurrentDrawing().mergeLayer(historyRecordingService(hrService));
canvas_.clearSelection();
leaveContext();
canvas_.changeSVGContent();
};
/**
* @function module:draw.mergeAllLayers
* @param {module:history.HistoryRecordingService} hrService
* @returns {void}
*/
var mergeAllLayers = function mergeAllLayers(hrService) {
canvas_.getCurrentDrawing().mergeAllLayers(historyRecordingService(hrService));
canvas_.clearSelection();
leaveContext();
canvas_.changeSVGContent();
};
/**
* Return from a group context to the regular kind, make any previously
* disabled elements enabled again.
* @function module:draw.leaveContext
* @fires module:svgcanvas.SvgCanvas#event:contextset
* @returns {void}
*/
var leaveContext = function leaveContext() {
var len = disabledElems.length;
if (len) {
for (var i = 0; i < len; i++) {
var elem = disabledElems[i];
var orig = canvas_.elData(elem, 'orig_opac');
if (orig !== 1) {
elem.setAttribute('opacity', orig);
} else {
elem.removeAttribute('opacity');
}
elem.setAttribute('style', 'pointer-events: inherit');
}
disabledElems = [];
canvas_.clearSelection(true);
canvas_.call('contextset', null);
}
canvas_.setCurrentGroup(null);
};
/**
* Set the current context (for in-group editing).
* @function module:draw.setContext
* @param {Element} elem
* @fires module:svgcanvas.SvgCanvas#event:contextset
* @returns {void}
*/
var setContext = function setContext(elem) {
leaveContext();
if (typeof elem === 'string') {
elem = getElem(elem);
} // Edit inside this group
canvas_.setCurrentGroup(elem); // Disable other elements
$$4(elem).parentsUntil('#svgcontent').andSelf().siblings().each(function () {
var opac = this.getAttribute('opacity') || 1; // Store the original's opacity
canvas_.elData(this, 'orig_opac', opac);
this.setAttribute('opacity', opac * 0.33);
this.setAttribute('style', 'pointer-events: none');
disabledElems.push(this);
});
canvas_.clearSelection();
canvas_.call('contextset', canvas_.getCurrentGroup());
};
var REVERSE_NS = getReverseNS(); // Todo: Split out into core attributes, presentation attributes, etc. so consistent
/**
* This defines which elements and attributes that we support (or at least
* don't remove).
* @type {PlainObject}
*/
var svgWhiteList_ = {
// SVG Elements
a: ['class', 'clip-path', 'clip-rule', 'fill', 'fill-opacity', 'fill-rule', 'filter', 'id', 'mask', 'opacity', 'stroke', 'stroke-dasharray', 'stroke-dashoffset', 'stroke-linecap', 'stroke-linejoin', 'stroke-miterlimit', 'stroke-opacity', 'stroke-width', 'style', 'systemLanguage', 'transform', 'xlink:href', 'xlink:title'],
circle: ['class', 'clip-path', 'clip-rule', 'cx', 'cy', 'fill', 'fill-opacity', 'fill-rule', 'filter', 'id', 'mask', 'opacity', 'r', 'requiredFeatures', 'stroke', 'stroke-dasharray', 'stroke-dashoffset', 'stroke-linecap', 'stroke-linejoin', 'stroke-miterlimit', 'stroke-opacity', 'stroke-width', 'style', 'systemLanguage', 'transform'],
clipPath: ['class', 'clipPathUnits', 'id'],
defs: [],
style: ['type'],
desc: [],
ellipse: ['class', 'clip-path', 'clip-rule', 'cx', 'cy', 'fill', 'fill-opacity', 'fill-rule', 'filter', 'id', 'mask', 'opacity', 'requiredFeatures', 'rx', 'ry', 'stroke', 'stroke-dasharray', 'stroke-dashoffset', 'stroke-linecap', 'stroke-linejoin', 'stroke-miterlimit', 'stroke-opacity', 'stroke-width', 'style', 'systemLanguage', 'transform'],
feGaussianBlur: ['class', 'color-interpolation-filters', 'id', 'requiredFeatures', 'stdDeviation'],
feMorphology: ['class', 'in', 'operator', 'radius'],
filter: ['class', 'color-interpolation-filters', 'filterRes', 'filterUnits', 'height', 'id', 'primitiveUnits', 'requiredFeatures', 'width', 'x', 'xlink:href', 'y'],
foreignObject: ['class', 'font-size', 'height', 'id', 'opacity', 'requiredFeatures', 'style', 'transform', 'width', 'x', 'y'],
g: ['class', 'clip-path', 'clip-rule', 'id', 'display', 'fill', 'fill-opacity', 'fill-rule', 'filter', 'mask', 'opacity', 'requiredFeatures', 'stroke', 'stroke-dasharray', 'stroke-dashoffset', 'stroke-linecap', 'stroke-linejoin', 'stroke-miterlimit', 'stroke-opacity', 'stroke-width', 'style', 'systemLanguage', 'transform', 'font-family', 'font-size', 'font-style', 'font-weight', 'text-anchor'],
image: ['class', 'clip-path', 'clip-rule', 'filter', 'height', 'id', 'mask', 'opacity', 'requiredFeatures', 'style', 'systemLanguage', 'transform', 'width', 'x', 'xlink:href', 'xlink:title', 'y'],
line: ['class', 'clip-path', 'clip-rule', 'fill', 'fill-opacity', 'fill-rule', 'filter', 'id', 'marker-end', 'marker-mid', 'marker-start', 'mask', 'opacity', 'requiredFeatures', 'stroke', 'stroke-dasharray', 'stroke-dashoffset', 'stroke-linecap', 'stroke-linejoin', 'stroke-miterlimit', 'stroke-opacity', 'stroke-width', 'style', 'systemLanguage', 'transform', 'x1', 'x2', 'y1', 'y2'],
linearGradient: ['class', 'id', 'gradientTransform', 'gradientUnits', 'requiredFeatures', 'spreadMethod', 'systemLanguage', 'x1', 'x2', 'xlink:href', 'y1', 'y2'],
marker: ['id', 'class', 'markerHeight', 'markerUnits', 'markerWidth', 'orient', 'preserveAspectRatio', 'refX', 'refY', 'systemLanguage', 'viewBox'],
mask: ['class', 'height', 'id', 'maskContentUnits', 'maskUnits', 'width', 'x', 'y'],
metadata: ['class', 'id'],
path: ['class', 'clip-path', 'clip-rule', 'd', 'fill', 'fill-opacity', 'fill-rule', 'filter', 'id', 'marker-end', 'marker-mid', 'marker-start', 'mask', 'opacity', 'requiredFeatures', 'stroke', 'stroke-dasharray', 'stroke-dashoffset', 'stroke-linecap', 'stroke-linejoin', 'stroke-miterlimit', 'stroke-opacity', 'stroke-width', 'style', 'systemLanguage', 'transform'],
pattern: ['class', 'height', 'id', 'patternContentUnits', 'patternTransform', 'patternUnits', 'requiredFeatures', 'style', 'systemLanguage', 'viewBox', 'width', 'x', 'xlink:href', 'y'],
polygon: ['class', 'clip-path', 'clip-rule', 'id', 'fill', 'fill-opacity', 'fill-rule', 'filter', 'id', 'class', 'marker-end', 'marker-mid', 'marker-start', 'mask', 'opacity', 'points', 'requiredFeatures', 'stroke', 'stroke-dasharray', 'stroke-dashoffset', 'stroke-linecap', 'stroke-linejoin', 'stroke-miterlimit', 'stroke-opacity', 'stroke-width', 'style', 'systemLanguage', 'transform'],
polyline: ['class', 'clip-path', 'clip-rule', 'id', 'fill', 'fill-opacity', 'fill-rule', 'filter', 'marker-end', 'marker-mid', 'marker-start', 'mask', 'opacity', 'points', 'requiredFeatures', 'stroke', 'stroke-dasharray', 'stroke-dashoffset', 'stroke-linecap', 'stroke-linejoin', 'stroke-miterlimit', 'stroke-opacity', 'stroke-width', 'style', 'systemLanguage', 'transform'],
radialGradient: ['class', 'cx', 'cy', 'fx', 'fy', 'gradientTransform', 'gradientUnits', 'id', 'r', 'requiredFeatures', 'spreadMethod', 'systemLanguage', 'xlink:href'],
rect: ['class', 'clip-path', 'clip-rule', 'fill', 'fill-opacity', 'fill-rule', 'filter', 'height', 'id', 'mask', 'opacity', 'requiredFeatures', 'rx', 'ry', 'stroke', 'stroke-dasharray', 'stroke-dashoffset', 'stroke-linecap', 'stroke-linejoin', 'stroke-miterlimit', 'stroke-opacity', 'stroke-width', 'style', 'systemLanguage', 'transform', 'width', 'x', 'y'],
stop: ['class', 'id', 'offset', 'requiredFeatures', 'stop-color', 'stop-opacity', 'style', 'systemLanguage'],
svg: ['class', 'clip-path', 'clip-rule', 'filter', 'id', 'height', 'mask', 'preserveAspectRatio', 'requiredFeatures', 'style', 'systemLanguage', 'viewBox', 'width', 'x', 'xmlns', 'xmlns:se', 'xmlns:xlink', 'y'],
"switch": ['class', 'id', 'requiredFeatures', 'systemLanguage'],
symbol: ['class', 'fill', 'fill-opacity', 'fill-rule', 'filter', 'font-family', 'font-size', 'font-style', 'font-weight', 'id', 'opacity', 'preserveAspectRatio', 'requiredFeatures', 'stroke', 'stroke-dasharray', 'stroke-dashoffset', 'stroke-linecap', 'stroke-linejoin', 'stroke-miterlimit', 'stroke-opacity', 'stroke-width', 'style', 'systemLanguage', 'transform', 'viewBox'],
text: ['class', 'clip-path', 'clip-rule', 'fill', 'fill-opacity', 'fill-rule', 'filter', 'font-family', 'font-size', 'font-style', 'font-weight', 'id', 'mask', 'opacity', 'requiredFeatures', 'stroke', 'stroke-dasharray', 'stroke-dashoffset', 'stroke-linecap', 'stroke-linejoin', 'stroke-miterlimit', 'stroke-opacity', 'stroke-width', 'style', 'systemLanguage', 'text-anchor', 'transform', 'x', 'xml:space', 'y'],
textPath: ['class', 'id', 'method', 'requiredFeatures', 'spacing', 'startOffset', 'style', 'systemLanguage', 'transform', 'xlink:href'],
title: [],
tspan: ['class', 'clip-path', 'clip-rule', 'dx', 'dy', 'fill', 'fill-opacity', 'fill-rule', 'filter', 'font-family', 'font-size', 'font-style', 'font-weight', 'id', 'mask', 'opacity', 'requiredFeatures', 'rotate', 'stroke', 'stroke-dasharray', 'stroke-dashoffset', 'stroke-linecap', 'stroke-linejoin', 'stroke-miterlimit', 'stroke-opacity', 'stroke-width', 'style', 'systemLanguage', 'text-anchor', 'textLength', 'transform', 'x', 'xml:space', 'y'],
use: ['class', 'clip-path', 'clip-rule', 'fill', 'fill-opacity', 'fill-rule', 'filter', 'height', 'id', 'mask', 'stroke', 'stroke-dasharray', 'stroke-dashoffset', 'stroke-linecap', 'stroke-linejoin', 'stroke-miterlimit', 'stroke-opacity', 'stroke-width', 'style', 'transform', 'width', 'x', 'xlink:href', 'y'],
// MathML Elements
annotation: ['encoding'],
'annotation-xml': ['encoding'],
maction: ['actiontype', 'other', 'selection'],
math: ['class', 'id', 'display', 'xmlns'],
menclose: ['notation'],
merror: [],
mfrac: ['linethickness'],
mi: ['mathvariant'],
mmultiscripts: [],
mn: [],
mo: ['fence', 'lspace', 'maxsize', 'minsize', 'rspace', 'stretchy'],
mover: [],
mpadded: ['lspace', 'width', 'height', 'depth', 'voffset'],
mphantom: [],
mprescripts: [],
mroot: [],
mrow: ['xlink:href', 'xlink:type', 'xmlns:xlink'],
mspace: ['depth', 'height', 'width'],
msqrt: [],
mstyle: ['displaystyle', 'mathbackground', 'mathcolor', 'mathvariant', 'scriptlevel'],
msub: [],
msubsup: [],
msup: [],
mtable: ['align', 'columnalign', 'columnlines', 'columnspacing', 'displaystyle', 'equalcolumns', 'equalrows', 'frame', 'rowalign', 'rowlines', 'rowspacing', 'width'],
mtd: ['columnalign', 'columnspan', 'rowalign', 'rowspan'],
mtext: [],
mtr: ['columnalign', 'rowalign'],
munder: [],
munderover: [],
none: [],
semantics: []
}; // Produce a Namespace-aware version of svgWhitelist
var svgWhiteListNS_ = {};
Object.entries(svgWhiteList_).forEach(function (_ref) {
var _ref2 = _slicedToArray(_ref, 2),
elt = _ref2[0],
atts = _ref2[1];
var attNS = {};
Object.entries(atts).forEach(function (_ref3) {
var _ref4 = _slicedToArray(_ref3, 2),
i = _ref4[0],
att = _ref4[1];
if (att.includes(':')) {
var v = att.split(':');
attNS[v[1]] = NS[v[0].toUpperCase()];
} else {
attNS[att] = att === 'xmlns' ? NS.XMLNS : null;
}
});
svgWhiteListNS_[elt] = attNS;
});
/**
* Sanitizes the input node and its children.
* It only keeps what is allowed from our whitelist defined above.
* @function module:sanitize.sanitizeSvg
* @param {Text|Element} node - The DOM element to be checked (we'll also check its children) or text node to be cleaned up
* @returns {void}
*/
var sanitizeSvg = function sanitizeSvg(node) {
// Cleanup text nodes
if (node.nodeType === 3) {
// 3 === TEXT_NODE
// Trim whitespace
node.nodeValue = node.nodeValue.replace(/^\s+|\s+$/g, ''); // Remove if empty
if (!node.nodeValue.length) {
node.remove();
}
} // We only care about element nodes.
// Automatically return for all non-element nodes, such as comments, etc.
if (node.nodeType !== 1) {
// 1 == ELEMENT_NODE
return;
}
var doc = node.ownerDocument;
var parent = node.parentNode; // can parent ever be null here? I think the root node's parent is the document...
if (!doc || !parent) {
return;
}
var allowedAttrs = svgWhiteList_[node.nodeName];
var allowedAttrsNS = svgWhiteListNS_[node.nodeName]; // if this element is supported, sanitize it
if (typeof allowedAttrs !== 'undefined') {
var seAttrs = [];
var i = node.attributes.length;
while (i--) {
// if the attribute is not in our whitelist, then remove it
// could use jQuery's inArray(), but I don't know if that's any better
var attr = node.attributes.item(i);
var attrName = attr.nodeName;
var attrLocalName = attr.localName;
var attrNsURI = attr.namespaceURI; // Check that an attribute with the correct localName in the correct namespace is on
// our whitelist or is a namespace declaration for one of our allowed namespaces
if (!({}.hasOwnProperty.call(allowedAttrsNS, attrLocalName) && attrNsURI === allowedAttrsNS[attrLocalName] && attrNsURI !== NS.XMLNS) && !(attrNsURI === NS.XMLNS && REVERSE_NS[attr.value])) {
// TODO(codedread): Programmatically add the se: attributes to the NS-aware whitelist.
// Bypassing the whitelist to allow se: prefixes.
// Is there a more appropriate way to do this?
if (attrName.startsWith('se:') || attrName.startsWith('data-')) {
seAttrs.push([attrName, attr.value]);
}
node.removeAttributeNS(attrNsURI, attrLocalName);
} // Add spaces before negative signs where necessary
if (isGecko()) {
switch (attrName) {
case 'transform':
case 'gradientTransform':
case 'patternTransform':
{
var val = attr.value.replace(/(\d)-/g, '$1 -'); // const val = attr.value.replace(/(?<digit>\d)-/g, '$<digit> -');
node.setAttribute(attrName, val);
break;
}
}
} // For the style attribute, rewrite it in terms of XML presentational attributes
if (attrName === 'style') {
var props = attr.value.split(';');
var p = props.length;
while (p--) {
var _props$p$split = props[p].split(':'),
_props$p$split2 = _slicedToArray(_props$p$split, 2),
name = _props$p$split2[0],
_val = _props$p$split2[1];
var styleAttrName = (name || '').trim();
var styleAttrVal = (_val || '').trim(); // Now check that this attribute is supported
if (allowedAttrs.includes(styleAttrName)) {
node.setAttribute(styleAttrName, styleAttrVal);
}
}
node.removeAttribute('style');
}
}
Object.values(seAttrs).forEach(function (_ref5) {
var _ref6 = _slicedToArray(_ref5, 2),
att = _ref6[0],
val = _ref6[1];
node.setAttributeNS(NS.SE, att, val);
}); // for some elements that have a xlink:href, ensure the URI refers to a local element
// (but not for links)
var href = getHref(node);
if (href && ['filter', 'linearGradient', 'pattern', 'radialGradient', 'textPath', 'use'].includes(node.nodeName)) {
// TODO: we simply check if the first character is a #, is this bullet-proof?
if (href[0] !== '#') {
// remove the attribute (but keep the element)
setHref(node, '');
node.removeAttributeNS(NS.XLINK, 'href');
}
} // Safari crashes on a <use> without a xlink:href, so we just remove the node here
if (node.nodeName === 'use' && !getHref(node)) {
node.remove();
return;
} // if the element has attributes pointing to a non-local reference,
// need to remove the attribute
Object.values(['clip-path', 'fill', 'filter', 'marker-end', 'marker-mid', 'marker-start', 'mask', 'stroke'], function (attr) {
var val = node.getAttribute(attr);
if (val) {
val = getUrlFromAttr(val); // simply check for first character being a '#'
if (val && val[0] !== '#') {
node.setAttribute(attr, '');
node.removeAttribute(attr);
}
}
}); // recurse to children
i = node.childNodes.length;
while (i--) {
sanitizeSvg(node.childNodes.item(i));
} // else (element not supported), remove it
} else {
// remove all children from this node and insert them before this node
// TODO: in the case of animation elements this will hardly ever be correct
var children = [];
while (node.hasChildNodes()) {
children.push(parent.insertBefore(node.firstChild, node));
} // remove this node from the document altogether
node.remove(); // call sanitizeSvg on each of those children
var _i = children.length;
while (_i--) {
sanitizeSvg(children[_i]);
}
}
};
/* eslint-disable jsdoc/require-file-overview */
/**
* Adapted from {@link https://github.com/uupaa/dynamic-import-polyfill/blob/master/importModule.js}.
* @module importModule
* @license MIT
*/
/**
* Converts a possible relative URL into an absolute one.
* @param {string} url
* @returns {string}
*/
function toAbsoluteURL(url) {
var a = document.createElement('a');
a.setAttribute('href', url); // <a href="hoge.html">
return a.cloneNode(false).href; // -> "http://example.com/hoge.html"
}
/**
* Add any of the whitelisted attributes to the script tag.
* @param {HTMLScriptElement} script
* @param {PlainObject<string, string>} atts
* @returns {void}
*/
function addScriptAtts(script, atts) {
['id', 'class', 'type'].forEach(function (prop) {
if (prop in atts) {
script[prop] = atts[prop];
}
});
} // Additions by Brett
/**
* @function module:importModule.importSetGlobal
* @param {string|string[]} url
* @param {module:importModule.ImportConfig} config
* @returns {Promise<ArbitraryModule>} The promise resolves to either an `ArbitraryModule` or
* any other value depends on the export of the targeted module.
*/
function importSetGlobal(_x, _x2) {
return _importSetGlobal.apply(this, arguments);
}
/**
*
* @author Brett Zamir (other items are from `dynamic-import-polyfill`)
* @param {string|string[]} url
* @param {PlainObject} [atts={}]
* @returns {Promise<void|Error>} Resolves to `undefined` or rejects with an `Error` upon a
* script loading error
*/
function _importSetGlobal() {
_importSetGlobal = _asyncToGenerator( /*#__PURE__*/regeneratorRuntime.mark(function _callee(url, _ref2) {
var glob, returnDefault, modularVersion;
return regeneratorRuntime.wrap(function _callee$(_context) {
while (1) {
switch (_context.prev = _context.next) {
case 0:
glob = _ref2.global, returnDefault = _ref2.returnDefault;
// Todo: Replace calls to this function with `import()` when supported
modularVersion = !('svgEditor' in window) || !window.svgEditor || window.svgEditor.modules !== false;
if (!modularVersion) {
_context.next = 4;
break;
}
return _context.abrupt("return", importModule(url, undefined, {
returnDefault: returnDefault
}));
case 4:
_context.next = 6;
return importScript(url);
case 6:
return _context.abrupt("return", window[glob]);
case 7:
case "end":
return _context.stop();
}
}
}, _callee);
}));
return _importSetGlobal.apply(this, arguments);
}
function importScript(url) {
var atts = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {};
if (Array.isArray(url)) {
return Promise.all(url.map(function (u) {
return importScript(u, atts);
}));
}
return new Promise(function (resolve, reject) {
// eslint-disable-line promise/avoid-new
var script = document.createElement('script');
/**
*
* @returns {void}
*/
function scriptOnError() {
reject(new Error("Failed to import: ".concat(url)));
destructor();
}
/**
*
* @returns {void}
*/
function scriptOnLoad() {
resolve();
destructor();
}
var destructor = function destructor() {
script.removeEventListener('error', scriptOnError);
script.removeEventListener('load', scriptOnLoad);
script.remove();
script.src = '';
};
script.defer = 'defer';
addScriptAtts(script, atts);
script.addEventListener('error', scriptOnError);
script.addEventListener('load', scriptOnLoad);
script.src = url;
document.head.append(script);
});
}
/**
*
* @param {string|string[]} url
* @param {PlainObject} [atts={}]
* @param {PlainObject} opts
* @param {boolean} [opts.returnDefault=false} = {}]
* @returns {Promise<any>} Resolves to value of loading module or rejects with
* `Error` upon a script loading error.
*/
function importModule(url) {
var atts = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {};
var _ref = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : {},
_ref$returnDefault = _ref.returnDefault,
returnDefault = _ref$returnDefault === void 0 ? false : _ref$returnDefault;
if (Array.isArray(url)) {
return Promise.all(url.map(function (u) {
return importModule(u, atts);
}));
}
return new Promise(function (resolve, reject) {
// eslint-disable-line promise/avoid-new
var vector = '$importModule$' + Math.random().toString(32).slice(2);
var script = document.createElement('script');
/**
*
* @returns {void}
*/
function scriptOnError() {
reject(new Error("Failed to import: ".concat(url)));
destructor();
}
/**
*
* @returns {void}
*/
function scriptOnLoad() {
resolve(window[vector]);
destructor();
}
var destructor = function destructor() {
delete window[vector];
script.removeEventListener('error', scriptOnError);
script.removeEventListener('load', scriptOnLoad);
script.remove();
URL.revokeObjectURL(script.src);
script.src = '';
};
addScriptAtts(script, atts);
script.defer = 'defer';
script.type = 'module';
script.addEventListener('error', scriptOnError);
script.addEventListener('load', scriptOnLoad);
var absURL = toAbsoluteURL(url);
var loader = "import * as m from '".concat(absURL.replace(/'/g, "\\'"), "'; window.").concat(vector, " = ").concat(returnDefault ? 'm.default || ' : '', "m;"); // export Module
var blob = new Blob([loader], {
type: 'text/javascript'
});
script.src = URL.createObjectURL(blob);
document.head.append(script);
});
}
var $$5 = jQuery; // this is how we map paths to our preferred relative segment types
var pathMap$1 = [0, 'z', 'M', 'm', 'L', 'l', 'C', 'c', 'Q', 'q', 'A', 'a', 'H', 'h', 'V', 'v', 'S', 's', 'T', 't'];
/**
* @interface module:coords.EditorContext
*/
/**
* @function module:coords.EditorContext#getGridSnapping
* @returns {boolean}
*/
/**
* @function module:coords.EditorContext#getDrawing
* @returns {module:draw.Drawing}
*/
/**
* @function module:coords.EditorContext#getSVGRoot
* @returns {SVGSVGElement}
*/
var editorContext_$2 = null;
/**
* @function module:coords.init
* @param {module:svgcanvas.SvgCanvas#event:pointsAdded} editorContext
* @returns {void}
*/
var init$4 = function init(editorContext) {
editorContext_$2 = editorContext;
};
/**
* Applies coordinate changes to an element based on the given matrix.
* @name module:coords.remapElement
* @type {module:path.EditorContext#remapElement}
*/
var remapElement = function remapElement(selected, changes, m) {
var remap = function remap(x, y) {
return transformPoint(x, y, m);
},
scalew = function scalew(w) {
return m.a * w;
},
scaleh = function scaleh(h) {
return m.d * h;
},
doSnapping = editorContext_$2.getGridSnapping() && selected.parentNode.parentNode.localName === 'svg',
finishUp = function finishUp() {
if (doSnapping) {
Object.entries(changes).forEach(function (_ref) {
var _ref2 = _slicedToArray(_ref, 2),
o = _ref2[0],
value = _ref2[1];
changes[o] = snapToGrid(value);
});
}
assignAttributes(selected, changes, 1000, true);
},
box = getBBox(selected);
for (var i = 0; i < 2; i++) {
var type = i === 0 ? 'fill' : 'stroke';
var attrVal = selected.getAttribute(type);
if (attrVal && attrVal.startsWith('url(')) {
if (m.a < 0 || m.d < 0) {
var grad = getRefElem(attrVal);
var newgrad = grad.cloneNode(true);
if (m.a < 0) {
// flip x
var x1 = newgrad.getAttribute('x1');
var x2 = newgrad.getAttribute('x2');
newgrad.setAttribute('x1', -(x1 - 1));
newgrad.setAttribute('x2', -(x2 - 1));
}
if (m.d < 0) {
// flip y
var y1 = newgrad.getAttribute('y1');
var y2 = newgrad.getAttribute('y2');
newgrad.setAttribute('y1', -(y1 - 1));
newgrad.setAttribute('y2', -(y2 - 1));
}
newgrad.id = editorContext_$2.getDrawing().getNextId();
findDefs().append(newgrad);
selected.setAttribute(type, 'url(#' + newgrad.id + ')');
} // Not really working :(
// if (selected.tagName === 'path') {
// reorientGrads(selected, m);
// }
}
}
var elName = selected.tagName;
if (elName === 'g' || elName === 'text' || elName === 'tspan' || elName === 'use') {
// if it was a translate, then just update x,y
if (m.a === 1 && m.b === 0 && m.c === 0 && m.d === 1 && (m.e !== 0 || m.f !== 0)) {
// [T][M] = [M][T']
// therefore [T'] = [M_inv][T][M]
var existing = transformListToTransform(selected).matrix,
tNew = matrixMultiply(existing.inverse(), m, existing);
changes.x = parseFloat(changes.x) + tNew.e;
changes.y = parseFloat(changes.y) + tNew.f;
} else {
// we just absorb all matrices into the element and don't do any remapping
var chlist = getTransformList(selected);
var mt = editorContext_$2.getSVGRoot().createSVGTransform();
mt.setMatrix(matrixMultiply(transformListToTransform(chlist).matrix, m));
chlist.clear();
chlist.appendItem(mt);
}
} // now we have a set of changes and an applied reduced transform list
// we apply the changes directly to the DOM
switch (elName) {
case 'foreignObject':
case 'rect':
case 'image':
{
// Allow images to be inverted (give them matrix when flipped)
if (elName === 'image' && (m.a < 0 || m.d < 0)) {
// Convert to matrix
var _chlist = getTransformList(selected);
var _mt = editorContext_$2.getSVGRoot().createSVGTransform();
_mt.setMatrix(matrixMultiply(transformListToTransform(_chlist).matrix, m));
_chlist.clear();
_chlist.appendItem(_mt);
} else {
var pt1 = remap(changes.x, changes.y);
changes.width = scalew(changes.width);
changes.height = scaleh(changes.height);
changes.x = pt1.x + Math.min(0, changes.width);
changes.y = pt1.y + Math.min(0, changes.height);
changes.width = Math.abs(changes.width);
changes.height = Math.abs(changes.height);
}
finishUp();
break;
}
case 'ellipse':
{
var c = remap(changes.cx, changes.cy);
changes.cx = c.x;
changes.cy = c.y;
changes.rx = scalew(changes.rx);
changes.ry = scaleh(changes.ry);
changes.rx = Math.abs(changes.rx);
changes.ry = Math.abs(changes.ry);
finishUp();
break;
}
case 'circle':
{
var _c = remap(changes.cx, changes.cy);
changes.cx = _c.x;
changes.cy = _c.y; // take the minimum of the new selected box's dimensions for the new circle radius
var tbox = transformBox(box.x, box.y, box.width, box.height, m);
var w = tbox.tr.x - tbox.tl.x,
h = tbox.bl.y - tbox.tl.y;
changes.r = Math.min(w / 2, h / 2);
if (changes.r) {
changes.r = Math.abs(changes.r);
}
finishUp();
break;
}
case 'line':
{
var _pt = remap(changes.x1, changes.y1);
var pt2 = remap(changes.x2, changes.y2);
changes.x1 = _pt.x;
changes.y1 = _pt.y;
changes.x2 = pt2.x;
changes.y2 = pt2.y;
}
// Fallthrough
case 'text':
case 'tspan':
case 'use':
{
finishUp();
break;
}
case 'g':
{
var gsvg = $$5(selected).data('gsvg');
if (gsvg) {
assignAttributes(gsvg, changes, 1000, true);
}
break;
}
case 'polyline':
case 'polygon':
{
var len = changes.points.length;
for (var _i = 0; _i < len; ++_i) {
var pt = changes.points[_i];
var _remap = remap(pt.x, pt.y),
x = _remap.x,
y = _remap.y;
changes.points[_i].x = x;
changes.points[_i].y = y;
} // const len = changes.points.length;
var pstr = '';
for (var _i2 = 0; _i2 < len; ++_i2) {
var _pt2 = changes.points[_i2];
pstr += _pt2.x + ',' + _pt2.y + ' ';
}
selected.setAttribute('points', pstr);
break;
}
case 'path':
{
var segList = selected.pathSegList;
var _len = segList.numberOfItems;
changes.d = [];
for (var _i3 = 0; _i3 < _len; ++_i3) {
var seg = segList.getItem(_i3);
changes.d[_i3] = {
type: seg.pathSegType,
x: seg.x,
y: seg.y,
x1: seg.x1,
y1: seg.y1,
x2: seg.x2,
y2: seg.y2,
r1: seg.r1,
r2: seg.r2,
angle: seg.angle,
largeArcFlag: seg.largeArcFlag,
sweepFlag: seg.sweepFlag
};
}
_len = changes.d.length;
var firstseg = changes.d[0],
currentpt = remap(firstseg.x, firstseg.y);
changes.d[0].x = currentpt.x;
changes.d[0].y = currentpt.y;
for (var _i4 = 1; _i4 < _len; ++_i4) {
var _seg = changes.d[_i4];
var _type = _seg.type; // if absolute or first segment, we want to remap x, y, x1, y1, x2, y2
// if relative, we want to scalew, scaleh
if (_type % 2 === 0) {
// absolute
var thisx = _seg.x !== undefined ? _seg.x : currentpt.x,
// for V commands
thisy = _seg.y !== undefined ? _seg.y : currentpt.y; // for H commands
var _pt3 = remap(thisx, thisy);
var _pt4 = remap(_seg.x1, _seg.y1);
var _pt5 = remap(_seg.x2, _seg.y2);
_seg.x = _pt3.x;
_seg.y = _pt3.y;
_seg.x1 = _pt4.x;
_seg.y1 = _pt4.y;
_seg.x2 = _pt5.x;
_seg.y2 = _pt5.y;
_seg.r1 = scalew(_seg.r1);
_seg.r2 = scaleh(_seg.r2);
} else {
// relative
_seg.x = scalew(_seg.x);
_seg.y = scaleh(_seg.y);
_seg.x1 = scalew(_seg.x1);
_seg.y1 = scaleh(_seg.y1);
_seg.x2 = scalew(_seg.x2);
_seg.y2 = scaleh(_seg.y2);
_seg.r1 = scalew(_seg.r1);
_seg.r2 = scaleh(_seg.r2);
}
} // for each segment
var dstr = '';
_len = changes.d.length;
for (var _i5 = 0; _i5 < _len; ++_i5) {
var _seg2 = changes.d[_i5];
var _type2 = _seg2.type;
dstr += pathMap$1[_type2];
switch (_type2) {
case 13: // relative horizontal line (h)
case 12:
// absolute horizontal line (H)
dstr += _seg2.x + ' ';
break;
case 15: // relative vertical line (v)
case 14:
// absolute vertical line (V)
dstr += _seg2.y + ' ';
break;
case 3: // relative move (m)
case 5: // relative line (l)
case 19: // relative smooth quad (t)
case 2: // absolute move (M)
case 4: // absolute line (L)
case 18:
// absolute smooth quad (T)
dstr += _seg2.x + ',' + _seg2.y + ' ';
break;
case 7: // relative cubic (c)
case 6:
// absolute cubic (C)
dstr += _seg2.x1 + ',' + _seg2.y1 + ' ' + _seg2.x2 + ',' + _seg2.y2 + ' ' + _seg2.x + ',' + _seg2.y + ' ';
break;
case 9: // relative quad (q)
case 8:
// absolute quad (Q)
dstr += _seg2.x1 + ',' + _seg2.y1 + ' ' + _seg2.x + ',' + _seg2.y + ' ';
break;
case 11: // relative elliptical arc (a)
case 10:
// absolute elliptical arc (A)
dstr += _seg2.r1 + ',' + _seg2.r2 + ' ' + _seg2.angle + ' ' + Number(_seg2.largeArcFlag) + ' ' + Number(_seg2.sweepFlag) + ' ' + _seg2.x + ',' + _seg2.y + ' ';
break;
case 17: // relative smooth cubic (s)
case 16:
// absolute smooth cubic (S)
dstr += _seg2.x2 + ',' + _seg2.y2 + ' ' + _seg2.x + ',' + _seg2.y + ' ';
break;
}
}
selected.setAttribute('d', dstr);
break;
}
}
};
/* globals jQuery */
var $$6 = jQueryPluginSVG(jQuery);
var context_;
/**
* @interface module:recalculate.EditorContext
*/
/**
* @function module:recalculate.EditorContext#getSVGRoot
* @returns {SVGSVGElement} The root DOM element
*/
/**
* @function module:recalculate.EditorContext#getStartTransform
* @returns {string}
*/
/**
* @function module:recalculate.EditorContext#setStartTransform
* @param {string} transform
* @returns {void}
*/
/**
* @function module:recalculate.init
* @param {module:recalculate.EditorContext} editorContext
* @returns {void}
*/
var init$5 = function init(editorContext) {
context_ = editorContext;
};
/**
* Updates a `<clipPath>`s values based on the given translation of an element.
* @function module:recalculate.updateClipPath
* @param {string} attr - The clip-path attribute value with the clipPath's ID
* @param {Float} tx - The translation's x value
* @param {Float} ty - The translation's y value
* @returns {void}
*/
var updateClipPath = function updateClipPath(attr, tx, ty) {
var path = getRefElem(attr).firstChild;
var cpXform = getTransformList(path);
var newxlate = context_.getSVGRoot().createSVGTransform();
newxlate.setTranslate(tx, ty);
cpXform.appendItem(newxlate); // Update clipPath's dimensions
recalculateDimensions(path);
};
/**
* Decides the course of action based on the element's transform list.
* @function module:recalculate.recalculateDimensions
* @param {Element} selected - The DOM element to recalculate
* @returns {Command} Undo command object with the resulting change
*/
var recalculateDimensions = function recalculateDimensions(selected) {
if (isNullish(selected)) {
return null;
} // Firefox Issue - 1081
if (selected.nodeName === 'svg' && navigator.userAgent.includes('Firefox/20')) {
return null;
}
var svgroot = context_.getSVGRoot();
var tlist = getTransformList(selected); // remove any unnecessary transforms
if (tlist && tlist.numberOfItems > 0) {
var k = tlist.numberOfItems;
var noi = k;
while (k--) {
var xform = tlist.getItem(k);
if (xform.type === 0) {
tlist.removeItem(k); // remove identity matrices
} else if (xform.type === 1) {
if (isIdentity(xform.matrix)) {
if (noi === 1) {
// Overcome Chrome bug (though only when noi is 1) with
// `removeItem` preventing `removeAttribute` from
// subsequently working
// See https://bugs.chromium.org/p/chromium/issues/detail?id=843901
selected.removeAttribute('transform');
return null;
}
tlist.removeItem(k);
} // remove zero-degree rotations
} else if (xform.type === 4) {
if (xform.angle === 0) {
tlist.removeItem(k);
}
}
} // End here if all it has is a rotation
if (tlist.numberOfItems === 1 && getRotationAngle(selected)) {
return null;
}
} // if this element had no transforms, we are done
if (!tlist || tlist.numberOfItems === 0) {
// Chrome apparently had a bug that requires clearing the attribute first.
selected.setAttribute('transform', ''); // However, this still next line currently doesn't work at all in Chrome
selected.removeAttribute('transform'); // selected.transform.baseVal.clear(); // Didn't help for Chrome bug
return null;
} // TODO: Make this work for more than 2
if (tlist) {
var mxs = [];
var _k = tlist.numberOfItems;
while (_k--) {
var _xform = tlist.getItem(_k);
if (_xform.type === 1) {
mxs.push([_xform.matrix, _k]);
} else if (mxs.length) {
mxs = [];
}
}
if (mxs.length === 2) {
var mNew = svgroot.createSVGTransformFromMatrix(matrixMultiply(mxs[1][0], mxs[0][0]));
tlist.removeItem(mxs[0][1]);
tlist.removeItem(mxs[1][1]);
tlist.insertItemBefore(mNew, mxs[1][1]);
} // combine matrix + translate
_k = tlist.numberOfItems;
if (_k >= 2 && tlist.getItem(_k - 2).type === 1 && tlist.getItem(_k - 1).type === 2) {
var mt = svgroot.createSVGTransform();
var m = matrixMultiply(tlist.getItem(_k - 2).matrix, tlist.getItem(_k - 1).matrix);
mt.setMatrix(m);
tlist.removeItem(_k - 2);
tlist.removeItem(_k - 2);
tlist.appendItem(mt);
}
} // If it still has a single [M] or [R][M], return null too (prevents BatchCommand from being returned).
switch (selected.tagName) {
// Ignore these elements, as they can absorb the [M]
case 'line':
case 'polyline':
case 'polygon':
case 'path':
break;
default:
if (tlist.numberOfItems === 1 && tlist.getItem(0).type === 1 || tlist.numberOfItems === 2 && tlist.getItem(0).type === 1 && tlist.getItem(0).type === 4) {
return null;
}
} // Grouped SVG element
var gsvg = $$6(selected).data('gsvg'); // we know we have some transforms, so set up return variable
var batchCmd = new BatchCommand('Transform'); // store initial values that will be affected by reducing the transform list
var changes = {};
var initial = null;
var attrs = [];
switch (selected.tagName) {
case 'line':
attrs = ['x1', 'y1', 'x2', 'y2'];
break;
case 'circle':
attrs = ['cx', 'cy', 'r'];
break;
case 'ellipse':
attrs = ['cx', 'cy', 'rx', 'ry'];
break;
case 'foreignObject':
case 'rect':
case 'image':
attrs = ['width', 'height', 'x', 'y'];
break;
case 'use':
case 'text':
case 'tspan':
attrs = ['x', 'y'];
break;
case 'polygon':
case 'polyline':
{
initial = {};
initial.points = selected.getAttribute('points');
var list = selected.points;
var len = list.numberOfItems;
changes.points = new Array(len);
for (var i = 0; i < len; ++i) {
var pt = list.getItem(i);
changes.points[i] = {
x: pt.x,
y: pt.y
};
}
break;
}
case 'path':
initial = {};
initial.d = selected.getAttribute('d');
changes.d = selected.getAttribute('d');
break;
} // switch on element type to get initial values
if (attrs.length) {
changes = $$6(selected).attr(attrs);
$$6.each(changes, function (attr, val) {
changes[attr] = convertToNum(attr, val);
});
} else if (gsvg) {
// GSVG exception
changes = {
x: $$6(gsvg).attr('x') || 0,
y: $$6(gsvg).attr('y') || 0
};
} // if we haven't created an initial array in polygon/polyline/path, then
// make a copy of initial values and include the transform
if (isNullish(initial)) {
initial = $$6.extend(true, {}, changes);
$$6.each(initial, function (attr, val) {
initial[attr] = convertToNum(attr, val);
});
} // save the start transform value too
initial.transform = context_.getStartTransform() || '';
var oldcenter, newcenter; // if it's a regular group, we have special processing to flatten transforms
if (selected.tagName === 'g' && !gsvg || selected.tagName === 'a') {
var box = getBBox(selected);
oldcenter = {
x: box.x + box.width / 2,
y: box.y + box.height / 2
};
newcenter = transformPoint(box.x + box.width / 2, box.y + box.height / 2, transformListToTransform(tlist).matrix); // let m = svgroot.createSVGMatrix();
// temporarily strip off the rotate and save the old center
var gangle = getRotationAngle(selected);
if (gangle) {
var a = gangle * Math.PI / 180;
var s;
if (Math.abs(a) > 1.0e-10) {
s = Math.sin(a) / (1 - Math.cos(a));
} else {
// TODO: This blows up if the angle is exactly 0!
s = 2 / a;
}
for (var _i = 0; _i < tlist.numberOfItems; ++_i) {
var _xform2 = tlist.getItem(_i);
if (_xform2.type === 4) {
// extract old center through mystical arts
var rm = _xform2.matrix;
oldcenter.y = (s * rm.e + rm.f) / 2;
oldcenter.x = (rm.e - s * rm.f) / 2;
tlist.removeItem(_i);
break;
}
}
}
var N = tlist.numberOfItems;
var tx = 0,
ty = 0,
operation = 0;
var firstM;
if (N) {
firstM = tlist.getItem(0).matrix;
}
var oldStartTransform; // first, if it was a scale then the second-last transform will be it
if (N >= 3 && tlist.getItem(N - 2).type === 3 && tlist.getItem(N - 3).type === 2 && tlist.getItem(N - 1).type === 2) {
operation = 3; // scale
// if the children are unrotated, pass the scale down directly
// otherwise pass the equivalent matrix() down directly
var tm = tlist.getItem(N - 3).matrix,
sm = tlist.getItem(N - 2).matrix,
tmn = tlist.getItem(N - 1).matrix;
var children = selected.childNodes;
var c = children.length;
while (c--) {
var child = children.item(c);
tx = 0;
ty = 0;
if (child.nodeType === 1) {
var childTlist = getTransformList(child); // some children might not have a transform (<metadata>, <defs>, etc)
if (!childTlist) {
continue;
}
var _m = transformListToTransform(childTlist).matrix; // Convert a matrix to a scale if applicable
// if (hasMatrixTransform(childTlist) && childTlist.numberOfItems == 1) {
// if (m.b==0 && m.c==0 && m.e==0 && m.f==0) {
// childTlist.removeItem(0);
// const translateOrigin = svgroot.createSVGTransform(),
// scale = svgroot.createSVGTransform(),
// translateBack = svgroot.createSVGTransform();
// translateOrigin.setTranslate(0, 0);
// scale.setScale(m.a, m.d);
// translateBack.setTranslate(0, 0);
// childTlist.appendItem(translateBack);
// childTlist.appendItem(scale);
// childTlist.appendItem(translateOrigin);
// }
// }
var angle = getRotationAngle(child);
oldStartTransform = context_.getStartTransform(); // const childxforms = [];
context_.setStartTransform(child.getAttribute('transform'));
if (angle || hasMatrixTransform(childTlist)) {
var e2t = svgroot.createSVGTransform();
e2t.setMatrix(matrixMultiply(tm, sm, tmn, _m));
childTlist.clear();
childTlist.appendItem(e2t); // childxforms.push(e2t);
// if not rotated or skewed, push the [T][S][-T] down to the child
} else {
// update the transform list with translate,scale,translate
// slide the [T][S][-T] from the front to the back
// [T][S][-T][M] = [M][T2][S2][-T2]
// (only bringing [-T] to the right of [M])
// [T][S][-T][M] = [T][S][M][-T2]
// [-T2] = [M_inv][-T][M]
var t2n = matrixMultiply(_m.inverse(), tmn, _m); // [T2] is always negative translation of [-T2]
var t2 = svgroot.createSVGMatrix();
t2.e = -t2n.e;
t2.f = -t2n.f; // [T][S][-T][M] = [M][T2][S2][-T2]
// [S2] = [T2_inv][M_inv][T][S][-T][M][-T2_inv]
var s2 = matrixMultiply(t2.inverse(), _m.inverse(), tm, sm, tmn, _m, t2n.inverse());
var translateOrigin = svgroot.createSVGTransform(),
scale = svgroot.createSVGTransform(),
translateBack = svgroot.createSVGTransform();
translateOrigin.setTranslate(t2n.e, t2n.f);
scale.setScale(s2.a, s2.d);
translateBack.setTranslate(t2.e, t2.f);
childTlist.appendItem(translateBack);
childTlist.appendItem(scale);
childTlist.appendItem(translateOrigin); // childxforms.push(translateBack);
// childxforms.push(scale);
// childxforms.push(translateOrigin);
// logMatrix(translateBack.matrix);
// logMatrix(scale.matrix);
} // not rotated
batchCmd.addSubCommand(recalculateDimensions(child)); // TODO: If any <use> have this group as a parent and are
// referencing this child, then we need to impose a reverse
// scale on it so that when it won't get double-translated
// const uses = selected.getElementsByTagNameNS(NS.SVG, 'use');
// const href = '#' + child.id;
// let u = uses.length;
// while (u--) {
// const useElem = uses.item(u);
// if (href == getHref(useElem)) {
// const usexlate = svgroot.createSVGTransform();
// usexlate.setTranslate(-tx,-ty);
// getTransformList(useElem).insertItemBefore(usexlate,0);
// batchCmd.addSubCommand( recalculateDimensions(useElem) );
// }
// }
context_.setStartTransform(oldStartTransform);
} // element
} // for each child
// Remove these transforms from group
tlist.removeItem(N - 1);
tlist.removeItem(N - 2);
tlist.removeItem(N - 3);
} else if (N >= 3 && tlist.getItem(N - 1).type === 1) {
operation = 3; // scale
var _m2 = transformListToTransform(tlist).matrix;
var _e2t = svgroot.createSVGTransform();
_e2t.setMatrix(_m2);
tlist.clear();
tlist.appendItem(_e2t); // next, check if the first transform was a translate
// if we had [ T1 ] [ M ] we want to transform this into [ M ] [ T2 ]
// therefore [ T2 ] = [ M_inv ] [ T1 ] [ M ]
} else if ((N === 1 || N > 1 && tlist.getItem(1).type !== 3) && tlist.getItem(0).type === 2) {
operation = 2; // translate
var T_M = transformListToTransform(tlist).matrix;
tlist.removeItem(0);
var mInv = transformListToTransform(tlist).matrix.inverse();
var M2 = matrixMultiply(mInv, T_M);
tx = M2.e;
ty = M2.f;
if (tx !== 0 || ty !== 0) {
// we pass the translates down to the individual children
var _children = selected.childNodes;
var _c = _children.length;
var clipPathsDone = [];
while (_c--) {
var _child = _children.item(_c);
if (_child.nodeType === 1) {
// Check if child has clip-path
if (_child.getAttribute('clip-path')) {
// tx, ty
var attr = _child.getAttribute('clip-path');
if (!clipPathsDone.includes(attr)) {
updateClipPath(attr, tx, ty);
clipPathsDone.push(attr);
}
}
oldStartTransform = context_.getStartTransform();
context_.setStartTransform(_child.getAttribute('transform'));
var _childTlist = getTransformList(_child); // some children might not have a transform (<metadata>, <defs>, etc)
if (_childTlist) {
var newxlate = svgroot.createSVGTransform();
newxlate.setTranslate(tx, ty);
if (_childTlist.numberOfItems) {
_childTlist.insertItemBefore(newxlate, 0);
} else {
_childTlist.appendItem(newxlate);
}
batchCmd.addSubCommand(recalculateDimensions(_child)); // If any <use> have this group as a parent and are
// referencing this child, then impose a reverse translate on it
// so that when it won't get double-translated
var uses = selected.getElementsByTagNameNS(NS.SVG, 'use');
var href = '#' + _child.id;
var u = uses.length;
while (u--) {
var useElem = uses.item(u);
if (href === getHref(useElem)) {
var usexlate = svgroot.createSVGTransform();
usexlate.setTranslate(-tx, -ty);
getTransformList(useElem).insertItemBefore(usexlate, 0);
batchCmd.addSubCommand(recalculateDimensions(useElem));
}
}
context_.setStartTransform(oldStartTransform);
}
}
}
context_.setStartTransform(oldStartTransform);
} // else, a matrix imposition from a parent group
// keep pushing it down to the children
} else if (N === 1 && tlist.getItem(0).type === 1 && !gangle) {
operation = 1;
var _m3 = tlist.getItem(0).matrix,
_children2 = selected.childNodes;
var _c2 = _children2.length;
while (_c2--) {
var _child2 = _children2.item(_c2);
if (_child2.nodeType === 1) {
oldStartTransform = context_.getStartTransform();
context_.setStartTransform(_child2.getAttribute('transform'));
var _childTlist2 = getTransformList(_child2);
if (!_childTlist2) {
continue;
}
var em = matrixMultiply(_m3, transformListToTransform(_childTlist2).matrix);
var e2m = svgroot.createSVGTransform();
e2m.setMatrix(em);
_childTlist2.clear();
_childTlist2.appendItem(e2m, 0);
batchCmd.addSubCommand(recalculateDimensions(_child2));
context_.setStartTransform(oldStartTransform); // Convert stroke
// TODO: Find out if this should actually happen somewhere else
var sw = _child2.getAttribute('stroke-width');
if (_child2.getAttribute('stroke') !== 'none' && !isNaN(sw)) {
var avg = (Math.abs(em.a) + Math.abs(em.d)) / 2;
_child2.setAttribute('stroke-width', sw * avg);
}
}
}
tlist.clear(); // else it was just a rotate
} else {
if (gangle) {
var newRot = svgroot.createSVGTransform();
newRot.setRotate(gangle, newcenter.x, newcenter.y);
if (tlist.numberOfItems) {
tlist.insertItemBefore(newRot, 0);
} else {
tlist.appendItem(newRot);
}
}
if (tlist.numberOfItems === 0) {
selected.removeAttribute('transform');
}
return null;
} // if it was a translate, put back the rotate at the new center
if (operation === 2) {
if (gangle) {
newcenter = {
x: oldcenter.x + firstM.e,
y: oldcenter.y + firstM.f
};
var _newRot = svgroot.createSVGTransform();
_newRot.setRotate(gangle, newcenter.x, newcenter.y);
if (tlist.numberOfItems) {
tlist.insertItemBefore(_newRot, 0);
} else {
tlist.appendItem(_newRot);
}
} // if it was a resize
} else if (operation === 3) {
var _m4 = transformListToTransform(tlist).matrix;
var roldt = svgroot.createSVGTransform();
roldt.setRotate(gangle, oldcenter.x, oldcenter.y);
var rold = roldt.matrix;
var rnew = svgroot.createSVGTransform();
rnew.setRotate(gangle, newcenter.x, newcenter.y);
var rnewInv = rnew.matrix.inverse(),
_mInv = _m4.inverse(),
extrat = matrixMultiply(_mInv, rnewInv, rold, _m4);
tx = extrat.e;
ty = extrat.f;
if (tx !== 0 || ty !== 0) {
// now push this transform down to the children
// we pass the translates down to the individual children
var _children3 = selected.childNodes;
var _c3 = _children3.length;
while (_c3--) {
var _child3 = _children3.item(_c3);
if (_child3.nodeType === 1) {
oldStartTransform = context_.getStartTransform();
context_.setStartTransform(_child3.getAttribute('transform'));
var _childTlist3 = getTransformList(_child3);
var _newxlate = svgroot.createSVGTransform();
_newxlate.setTranslate(tx, ty);
if (_childTlist3.numberOfItems) {
_childTlist3.insertItemBefore(_newxlate, 0);
} else {
_childTlist3.appendItem(_newxlate);
}
batchCmd.addSubCommand(recalculateDimensions(_child3));
context_.setStartTransform(oldStartTransform);
}
}
}
if (gangle) {
if (tlist.numberOfItems) {
tlist.insertItemBefore(rnew, 0);
} else {
tlist.appendItem(rnew);
}
}
} // else, it's a non-group
} else {
// TODO: box might be null for some elements (<metadata> etc), need to handle this
var _box = getBBox(selected); // Paths (and possbly other shapes) will have no BBox while still in <defs>,
// but we still may need to recalculate them (see issue 595).
// TODO: Figure out how to get BBox from these elements in case they
// have a rotation transform
if (!_box && selected.tagName !== 'path') return null;
var _m5; // = svgroot.createSVGMatrix();
// temporarily strip off the rotate and save the old center
var _angle = getRotationAngle(selected);
if (_angle) {
oldcenter = {
x: _box.x + _box.width / 2,
y: _box.y + _box.height / 2
};
newcenter = transformPoint(_box.x + _box.width / 2, _box.y + _box.height / 2, transformListToTransform(tlist).matrix);
var _a = _angle * Math.PI / 180;
var _s = Math.abs(_a) > 1.0e-10 ? Math.sin(_a) / (1 - Math.cos(_a)) // TODO: This blows up if the angle is exactly 0!
: 2 / _a;
for (var _i2 = 0; _i2 < tlist.numberOfItems; ++_i2) {
var _xform3 = tlist.getItem(_i2);
if (_xform3.type === 4) {
// extract old center through mystical arts
var _rm = _xform3.matrix;
oldcenter.y = (_s * _rm.e + _rm.f) / 2;
oldcenter.x = (_rm.e - _s * _rm.f) / 2;
tlist.removeItem(_i2);
break;
}
}
} // 2 = translate, 3 = scale, 4 = rotate, 1 = matrix imposition
var _operation = 0;
var _N = tlist.numberOfItems; // Check if it has a gradient with userSpaceOnUse, in which case
// adjust it by recalculating the matrix transform.
// TODO: Make this work in Webkit using transformlist.SVGTransformList
if (!isWebkit()) {
var fill = selected.getAttribute('fill');
if (fill && fill.startsWith('url(')) {
var paint = getRefElem(fill);
var type = 'pattern';
if (paint.tagName !== type) type = 'gradient';
var attrVal = paint.getAttribute(type + 'Units');
if (attrVal === 'userSpaceOnUse') {
// Update the userSpaceOnUse element
_m5 = transformListToTransform(tlist).matrix;
var gtlist = getTransformList(paint);
var gmatrix = transformListToTransform(gtlist).matrix;
_m5 = matrixMultiply(_m5, gmatrix);
var mStr = 'matrix(' + [_m5.a, _m5.b, _m5.c, _m5.d, _m5.e, _m5.f].join(',') + ')';
paint.setAttribute(type + 'Transform', mStr);
}
}
} // first, if it was a scale of a non-skewed element, then the second-last
// transform will be the [S]
// if we had [M][T][S][T] we want to extract the matrix equivalent of
// [T][S][T] and push it down to the element
if (_N >= 3 && tlist.getItem(_N - 2).type === 3 && tlist.getItem(_N - 3).type === 2 && tlist.getItem(_N - 1).type === 2) {
// Removed this so a <use> with a given [T][S][T] would convert to a matrix.
// Is that bad?
// && selected.nodeName != 'use'
_operation = 3; // scale
_m5 = transformListToTransform(tlist, _N - 3, _N - 1).matrix;
tlist.removeItem(_N - 1);
tlist.removeItem(_N - 2);
tlist.removeItem(_N - 3); // if we had [T][S][-T][M], then this was a skewed element being resized
// Thus, we simply combine it all into one matrix
} else if (_N === 4 && tlist.getItem(_N - 1).type === 1) {
_operation = 3; // scale
_m5 = transformListToTransform(tlist).matrix;
var _e2t2 = svgroot.createSVGTransform();
_e2t2.setMatrix(_m5);
tlist.clear();
tlist.appendItem(_e2t2); // reset the matrix so that the element is not re-mapped
_m5 = svgroot.createSVGMatrix(); // if we had [R][T][S][-T][M], then this was a rotated matrix-element
// if we had [T1][M] we want to transform this into [M][T2]
// therefore [ T2 ] = [ M_inv ] [ T1 ] [ M ] and we can push [T2]
// down to the element
} else if ((_N === 1 || _N > 1 && tlist.getItem(1).type !== 3) && tlist.getItem(0).type === 2) {
_operation = 2; // translate
var oldxlate = tlist.getItem(0).matrix,
meq = transformListToTransform(tlist, 1).matrix,
meqInv = meq.inverse();
_m5 = matrixMultiply(meqInv, oldxlate, meq);
tlist.removeItem(0); // else if this child now has a matrix imposition (from a parent group)
// we might be able to simplify
} else if (_N === 1 && tlist.getItem(0).type === 1 && !_angle) {
// Remap all point-based elements
_m5 = transformListToTransform(tlist).matrix;
switch (selected.tagName) {
case 'line':
changes = $$6(selected).attr(['x1', 'y1', 'x2', 'y2']);
// Fallthrough
case 'polyline':
case 'polygon':
changes.points = selected.getAttribute('points');
if (changes.points) {
var _list = selected.points;
var _len = _list.numberOfItems;
changes.points = new Array(_len);
for (var _i3 = 0; _i3 < _len; ++_i3) {
var _pt = _list.getItem(_i3);
changes.points[_i3] = {
x: _pt.x,
y: _pt.y
};
}
}
// Fallthrough
case 'path':
changes.d = selected.getAttribute('d');
_operation = 1;
tlist.clear();
break;
} // if it was a rotation, put the rotate back and return without a command
// (this function has zero work to do for a rotate())
} else {
// operation = 4; // rotation
if (_angle) {
var _newRot2 = svgroot.createSVGTransform();
_newRot2.setRotate(_angle, newcenter.x, newcenter.y);
if (tlist.numberOfItems) {
tlist.insertItemBefore(_newRot2, 0);
} else {
tlist.appendItem(_newRot2);
}
}
if (tlist.numberOfItems === 0) {
selected.removeAttribute('transform');
}
return null;
} // if it was a translate or resize, we need to remap the element and absorb the xform
if (_operation === 1 || _operation === 2 || _operation === 3) {
remapElement(selected, changes, _m5);
} // if we are remapping
// if it was a translate, put back the rotate at the new center
if (_operation === 2) {
if (_angle) {
if (!hasMatrixTransform(tlist)) {
newcenter = {
x: oldcenter.x + _m5.e,
y: oldcenter.y + _m5.f
};
}
var _newRot3 = svgroot.createSVGTransform();
_newRot3.setRotate(_angle, newcenter.x, newcenter.y);
if (tlist.numberOfItems) {
tlist.insertItemBefore(_newRot3, 0);
} else {
tlist.appendItem(_newRot3);
}
} // We have special processing for tspans: Tspans are not transformable
// but they can have x,y coordinates (sigh). Thus, if this was a translate,
// on a text element, also translate any tspan children.
if (selected.tagName === 'text') {
var _children4 = selected.childNodes;
var _c4 = _children4.length;
while (_c4--) {
var _child4 = _children4.item(_c4);
if (_child4.tagName === 'tspan') {
var tspanChanges = {
x: $$6(_child4).attr('x') || 0,
y: $$6(_child4).attr('y') || 0
};
remapElement(_child4, tspanChanges, _m5);
}
}
} // [Rold][M][T][S][-T] became [Rold][M]
// we want it to be [Rnew][M][Tr] where Tr is the
// translation required to re-center it
// Therefore, [Tr] = [M_inv][Rnew_inv][Rold][M]
} else if (_operation === 3 && _angle) {
var _transformListToTrans = transformListToTransform(tlist),
matrix = _transformListToTrans.matrix;
var _roldt = svgroot.createSVGTransform();
_roldt.setRotate(_angle, oldcenter.x, oldcenter.y);
var _rold = _roldt.matrix;
var _rnew = svgroot.createSVGTransform();
_rnew.setRotate(_angle, newcenter.x, newcenter.y);
var _rnewInv = _rnew.matrix.inverse();
var _mInv2 = matrix.inverse();
var _extrat = matrixMultiply(_mInv2, _rnewInv, _rold, matrix);
remapElement(selected, changes, _extrat);
if (_angle) {
if (tlist.numberOfItems) {
tlist.insertItemBefore(_rnew, 0);
} else {
tlist.appendItem(_rnew);
}
}
}
} // a non-group
// if the transform list has been emptied, remove it
if (tlist.numberOfItems === 0) {
selected.removeAttribute('transform');
}
batchCmd.addSubCommand(new ChangeElementCommand(selected, initial));
return batchCmd;
};
var $$7 = jQuery;
var svgFactory_;
var config_;
var selectorManager_; // A Singleton
var gripRadius = isTouch() ? 10 : 4;
/**
* Private class for DOM element selection boxes.
*/
var Selector = /*#__PURE__*/function () {
/**
* @param {Integer} id - Internally identify the selector
* @param {Element} elem - DOM element associated with this selector
* @param {module:utilities.BBoxObject} [bbox] - Optional bbox to use for initialization (prevents duplicate `getBBox` call).
*/
function Selector(id, elem, bbox) {
_classCallCheck(this, Selector);
// this is the selector's unique number
this.id = id; // this holds a reference to the element for which this selector is being used
this.selectedElement = elem; // this is a flag used internally to track whether the selector is being used or not
this.locked = true; // this holds a reference to the <g> element that holds all visual elements of the selector
this.selectorGroup = svgFactory_.createSVGElement({
element: 'g',
attr: {
id: 'selectorGroup' + this.id
}
}); // this holds a reference to the path rect
this.selectorRect = this.selectorGroup.appendChild(svgFactory_.createSVGElement({
element: 'path',
attr: {
id: 'selectedBox' + this.id,
fill: 'none',
stroke: '#22C',
'stroke-width': '1',
'stroke-dasharray': '5,5',
// need to specify this so that the rect is not selectable
style: 'pointer-events:none'
}
})); // this holds a reference to the grip coordinates for this selector
this.gripCoords = {
nw: null,
n: null,
ne: null,
e: null,
se: null,
s: null,
sw: null,
w: null
};
this.reset(this.selectedElement, bbox);
}
/**
* Used to reset the id and element that the selector is attached to.
* @param {Element} e - DOM element associated with this selector
* @param {module:utilities.BBoxObject} bbox - Optional bbox to use for reset (prevents duplicate getBBox call).
* @returns {void}
*/
_createClass(Selector, [{
key: "reset",
value: function reset(e, bbox) {
this.locked = true;
this.selectedElement = e;
this.resize(bbox);
this.selectorGroup.setAttribute('display', 'inline');
}
/**
* Show the resize grips of this selector.
* @param {boolean} show - Indicates whether grips should be shown or not
* @returns {void}
*/
}, {
key: "showGrips",
value: function showGrips(show) {
var bShow = show ? 'inline' : 'none';
selectorManager_.selectorGripsGroup.setAttribute('display', bShow);
var elem = this.selectedElement;
this.hasGrips = show;
if (elem && show) {
this.selectorGroup.append(selectorManager_.selectorGripsGroup);
Selector.updateGripCursors(getRotationAngle(elem));
}
}
/**
* Updates the selector to match the element's size.
* @param {module:utilities.BBoxObject} [bbox] - BBox to use for resize (prevents duplicate getBBox call).
* @returns {void}
*/
}, {
key: "resize",
value: function resize(bbox) {
var selectedBox = this.selectorRect,
mgr = selectorManager_,
selectedGrips = mgr.selectorGrips,
selected = this.selectedElement,
sw = selected.getAttribute('stroke-width'),
currentZoom = svgFactory_.getCurrentZoom();
var offset = 1 / currentZoom;
if (selected.getAttribute('stroke') !== 'none' && !isNaN(sw)) {
offset += sw / 2;
}
var tagName = selected.tagName;
if (tagName === 'text') {
offset += 2 / currentZoom;
} // loop and transform our bounding box until we reach our first rotation
var tlist = getTransformList(selected);
var m = transformListToTransform(tlist).matrix; // This should probably be handled somewhere else, but for now
// it keeps the selection box correctly positioned when zoomed
m.e *= currentZoom;
m.f *= currentZoom;
if (!bbox) {
bbox = getBBox(selected);
} // TODO: getBBox (previous line) already knows to call getStrokedBBox when tagName === 'g'. Remove this?
// TODO: getBBox doesn't exclude 'gsvg' and calls getStrokedBBox for any 'g'. Should getBBox be updated?
if (tagName === 'g' && !$$7.data(selected, 'gsvg')) {
// The bbox for a group does not include stroke vals, so we
// get the bbox based on its children.
var strokedBbox = getStrokedBBox([selected.childNodes]);
if (strokedBbox) {
bbox = strokedBbox;
}
} // apply the transforms
var l = bbox.x,
t = bbox.y,
w = bbox.width,
h = bbox.height; // bbox = {x: l, y: t, width: w, height: h}; // Not in use
// we need to handle temporary transforms too
// if skewed, get its transformed box, then find its axis-aligned bbox
// *
offset *= currentZoom;
var nbox = transformBox(l * currentZoom, t * currentZoom, w * currentZoom, h * currentZoom, m),
aabox = nbox.aabox;
var nbax = aabox.x - offset,
nbay = aabox.y - offset,
nbaw = aabox.width + offset * 2,
nbah = aabox.height + offset * 2; // now if the shape is rotated, un-rotate it
var cx = nbax + nbaw / 2,
cy = nbay + nbah / 2;
var angle = getRotationAngle(selected);
if (angle) {
var rot = svgFactory_.svgRoot().createSVGTransform();
rot.setRotate(-angle, cx, cy);
var rotm = rot.matrix;
nbox.tl = transformPoint(nbox.tl.x, nbox.tl.y, rotm);
nbox.tr = transformPoint(nbox.tr.x, nbox.tr.y, rotm);
nbox.bl = transformPoint(nbox.bl.x, nbox.bl.y, rotm);
nbox.br = transformPoint(nbox.br.x, nbox.br.y, rotm); // calculate the axis-aligned bbox
var tl = nbox.tl;
var minx = tl.x,
miny = tl.y,
maxx = tl.x,
maxy = tl.y;
var min = Math.min,
max = Math.max;
minx = min(minx, min(nbox.tr.x, min(nbox.bl.x, nbox.br.x))) - offset;
miny = min(miny, min(nbox.tr.y, min(nbox.bl.y, nbox.br.y))) - offset;
maxx = max(maxx, max(nbox.tr.x, max(nbox.bl.x, nbox.br.x))) + offset;
maxy = max(maxy, max(nbox.tr.y, max(nbox.bl.y, nbox.br.y))) + offset;
nbax = minx;
nbay = miny;
nbaw = maxx - minx;
nbah = maxy - miny;
}
var dstr = 'M' + nbax + ',' + nbay + ' L' + (nbax + nbaw) + ',' + nbay + ' ' + (nbax + nbaw) + ',' + (nbay + nbah) + ' ' + nbax + ',' + (nbay + nbah) + 'z';
selectedBox.setAttribute('d', dstr);
var xform = angle ? 'rotate(' + [angle, cx, cy].join(',') + ')' : '';
this.selectorGroup.setAttribute('transform', xform); // TODO(codedread): Is this needed?
// if (selected === selectedElements[0]) {
this.gripCoords = {
nw: [nbax, nbay],
ne: [nbax + nbaw, nbay],
sw: [nbax, nbay + nbah],
se: [nbax + nbaw, nbay + nbah],
n: [nbax + nbaw / 2, nbay],
w: [nbax, nbay + nbah / 2],
e: [nbax + nbaw, nbay + nbah / 2],
s: [nbax + nbaw / 2, nbay + nbah]
};
Object.entries(this.gripCoords).forEach(function (_ref) {
var _ref2 = _slicedToArray(_ref, 2),
dir = _ref2[0],
coords = _ref2[1];
selectedGrips[dir].setAttribute('cx', coords[0]);
selectedGrips[dir].setAttribute('cy', coords[1]);
}); // we want to go 20 pixels in the negative transformed y direction, ignoring scale
mgr.rotateGripConnector.setAttribute('x1', nbax + nbaw / 2);
mgr.rotateGripConnector.setAttribute('y1', nbay);
mgr.rotateGripConnector.setAttribute('x2', nbax + nbaw / 2);
mgr.rotateGripConnector.setAttribute('y2', nbay - gripRadius * 5);
mgr.rotateGrip.setAttribute('cx', nbax + nbaw / 2);
mgr.rotateGrip.setAttribute('cy', nbay - gripRadius * 5); // }
} // STATIC methods
/**
* Updates cursors for corner grips on rotation so arrows point the right way.
* @param {Float} angle - Current rotation angle in degrees
* @returns {void}
*/
}], [{
key: "updateGripCursors",
value: function updateGripCursors(angle) {
var dirArr = Object.keys(selectorManager_.selectorGrips);
var steps = Math.round(angle / 45);
if (steps < 0) {
steps += 8;
}
while (steps > 0) {
dirArr.push(dirArr.shift());
steps--;
}
Object.values(selectorManager_.selectorGrips).forEach(function (gripElement, i) {
gripElement.setAttribute('style', 'cursor:' + dirArr[i] + '-resize');
});
}
}]);
return Selector;
}();
/**
* Manage all selector objects (selection boxes).
*/
var SelectorManager = /*#__PURE__*/function () {
/**
* Sets up properties and calls `initGroup`.
*/
function SelectorManager() {
_classCallCheck(this, SelectorManager);
// this will hold the <g> element that contains all selector rects/grips
this.selectorParentGroup = null; // this is a special rect that is used for multi-select
this.rubberBandBox = null; // this will hold objects of type Selector (see above)
this.selectors = []; // this holds a map of SVG elements to their Selector object
this.selectorMap = {}; // this holds a reference to the grip elements
this.selectorGrips = {
nw: null,
n: null,
ne: null,
e: null,
se: null,
s: null,
sw: null,
w: null
};
this.selectorGripsGroup = null;
this.rotateGripConnector = null;
this.rotateGrip = null;
this.initGroup();
}
/**
* Resets the parent selector group element.
* @returns {void}
*/
_createClass(SelectorManager, [{
key: "initGroup",
value: function initGroup() {
var _this = this;
// remove old selector parent group if it existed
if (this.selectorParentGroup && this.selectorParentGroup.parentNode) {
this.selectorParentGroup.remove();
} // create parent selector group and add it to svgroot
this.selectorParentGroup = svgFactory_.createSVGElement({
element: 'g',
attr: {
id: 'selectorParentGroup'
}
});
this.selectorGripsGroup = svgFactory_.createSVGElement({
element: 'g',
attr: {
display: 'none'
}
});
this.selectorParentGroup.append(this.selectorGripsGroup);
svgFactory_.svgRoot().append(this.selectorParentGroup);
this.selectorMap = {};
this.selectors = [];
this.rubberBandBox = null; // add the corner grips
Object.keys(this.selectorGrips).forEach(function (dir) {
var grip = svgFactory_.createSVGElement({
element: 'circle',
attr: {
id: 'selectorGrip_resize_' + dir,
fill: '#22C',
r: gripRadius,
style: 'cursor:' + dir + '-resize',
// This expands the mouse-able area of the grips making them
// easier to grab with the mouse.
// This works in Opera and WebKit, but does not work in Firefox
// see https://bugzilla.mozilla.org/show_bug.cgi?id=500174
'stroke-width': 2,
'pointer-events': 'all'
}
});
$$7.data(grip, 'dir', dir);
$$7.data(grip, 'type', 'resize');
_this.selectorGrips[dir] = _this.selectorGripsGroup.appendChild(grip);
}); // add rotator elems
this.rotateGripConnector = this.selectorGripsGroup.appendChild(svgFactory_.createSVGElement({
element: 'line',
attr: {
id: 'selectorGrip_rotateconnector',
stroke: '#22C',
'stroke-width': '1'
}
}));
this.rotateGrip = this.selectorGripsGroup.appendChild(svgFactory_.createSVGElement({
element: 'circle',
attr: {
id: 'selectorGrip_rotate',
fill: 'lime',
r: gripRadius,
stroke: '#22C',
'stroke-width': 2,
style: 'cursor:url(' + config_.imgPath + 'rotate.png) 12 12, auto;'
}
}));
$$7.data(this.rotateGrip, 'type', 'rotate');
if ($$7('#canvasBackground').length) {
return;
}
var _config_$dimensions = _slicedToArray(config_.dimensions, 2),
width = _config_$dimensions[0],
height = _config_$dimensions[1];
var canvasbg = svgFactory_.createSVGElement({
element: 'svg',
attr: {
id: 'canvasBackground',
width: width,
height: height,
x: 0,
y: 0,
overflow: isWebkit() ? 'none' : 'visible',
// Chrome 7 has a problem with this when zooming out
style: 'pointer-events:none'
}
});
var rect = svgFactory_.createSVGElement({
element: 'rect',
attr: {
width: '100%',
height: '100%',
x: 0,
y: 0,
'stroke-width': 1,
stroke: '#000',
fill: '#FFF',
style: 'pointer-events:none'
}
}); // Both Firefox and WebKit are too slow with this filter region (especially at higher
// zoom levels) and Opera has at least one bug
// if (!isOpera()) rect.setAttribute('filter', 'url(#canvashadow)');
canvasbg.append(rect);
svgFactory_.svgRoot().insertBefore(canvasbg, svgFactory_.svgContent()); // Ok to replace above with `svgFactory_.svgContent().before(canvasbg);`?
}
/**
*
* @param {Element} elem - DOM element to get the selector for
* @param {module:utilities.BBoxObject} [bbox] - Optional bbox to use for reset (prevents duplicate getBBox call).
* @returns {Selector} The selector based on the given element
*/
}, {
key: "requestSelector",
value: function requestSelector(elem, bbox) {
if (isNullish(elem)) {
return null;
}
var N = this.selectors.length; // If we've already acquired one for this element, return it.
if (_typeof(this.selectorMap[elem.id]) === 'object') {
this.selectorMap[elem.id].locked = true;
return this.selectorMap[elem.id];
}
for (var i = 0; i < N; ++i) {
if (this.selectors[i] && !this.selectors[i].locked) {
this.selectors[i].locked = true;
this.selectors[i].reset(elem, bbox);
this.selectorMap[elem.id] = this.selectors[i];
return this.selectors[i];
}
} // if we reached here, no available selectors were found, we create one
this.selectors[N] = new Selector(N, elem, bbox);
this.selectorParentGroup.append(this.selectors[N].selectorGroup);
this.selectorMap[elem.id] = this.selectors[N];
return this.selectors[N];
}
/**
* Removes the selector of the given element (hides selection box).
*
* @param {Element} elem - DOM element to remove the selector for
* @returns {void}
*/
}, {
key: "releaseSelector",
value: function releaseSelector(elem) {
if (isNullish(elem)) {
return;
}
var N = this.selectors.length,
sel = this.selectorMap[elem.id];
if (!sel.locked) {
// TODO(codedread): Ensure this exists in this module.
console.log('WARNING! selector was released but was already unlocked'); // eslint-disable-line no-console
}
for (var i = 0; i < N; ++i) {
if (this.selectors[i] && this.selectors[i] === sel) {
delete this.selectorMap[elem.id];
sel.locked = false;
sel.selectedElement = null;
sel.showGrips(false); // remove from DOM and store reference in JS but only if it exists in the DOM
try {
sel.selectorGroup.setAttribute('display', 'none');
} catch (e) {}
break;
}
}
}
/**
* @returns {SVGRectElement} The rubberBandBox DOM element. This is the rectangle drawn by
* the user for selecting/zooming
*/
}, {
key: "getRubberBandBox",
value: function getRubberBandBox() {
if (!this.rubberBandBox) {
this.rubberBandBox = this.selectorParentGroup.appendChild(svgFactory_.createSVGElement({
element: 'rect',
attr: {
id: 'selectorRubberBand',
fill: '#22C',
'fill-opacity': 0.15,
stroke: '#22C',
'stroke-width': 0.5,
display: 'none',
style: 'pointer-events:none'
}
}));
}
return this.rubberBandBox;
}
}]);
return SelectorManager;
}();
/**
* An object that creates SVG elements for the canvas.
*
* @interface module:select.SVGFactory
*/
/**
* @function module:select.SVGFactory#createSVGElement
* @param {module:utilities.EditorContext#addSVGElementFromJson} jsonMap
* @returns {SVGElement}
*/
/**
* @function module:select.SVGFactory#svgRoot
* @returns {SVGSVGElement}
*/
/**
* @function module:select.SVGFactory#svgContent
* @returns {SVGSVGElement}
*/
/**
* @function module:select.SVGFactory#getCurrentZoom
* @returns {Float} The current zoom level
*/
/**
* @typedef {GenericArray} module:select.Dimensions
* @property {Integer} length 2
* @property {Float} 0 Width
* @property {Float} 1 Height
*/
/**
* @typedef {PlainObject} module:select.Config
* @property {string} imgPath
* @property {module:select.Dimensions} dimensions
*/
/**
* Initializes this module.
* @function module:select.init
* @param {module:select.Config} config - An object containing configurable parameters (imgPath)
* @param {module:select.SVGFactory} svgFactory - An object implementing the SVGFactory interface.
* @returns {void}
*/
var init$6 = function init(config, svgFactory) {
config_ = config;
svgFactory_ = svgFactory;
selectorManager_ = new SelectorManager();
};
/**
* @function module:select.getSelectorManager
* @returns {module:select.SelectorManager} The SelectorManager instance.
*/
var getSelectorManager = function getSelectorManager() {
return selectorManager_;
};
var $$8 = jQueryPluginSVG(jQuery);
var MoveElementCommand$1 = MoveElementCommand,
InsertElementCommand$1 = InsertElementCommand,
RemoveElementCommand$1 = RemoveElementCommand,
ChangeElementCommand$1 = ChangeElementCommand,
BatchCommand$1 = BatchCommand,
UndoManager$1 = UndoManager,
HistoryEventTypes$1 = HistoryEventTypes;
if (!window.console) {
window.console = {};
window.console.log = function (str) {
/* */
};
window.console.dir = function (str) {
/* */
};
}
if (window.opera) {
window.console.log = function (str) {
window.opera.postError(str);
};
window.console.dir = function (str) {
/* */
};
} // Reenable after fixing eslint-plugin-jsdoc to handle
/**
* The main SvgCanvas class that manages all SVG-related functions.
* @memberof module:svgcanvas
*
* @borrows module:coords.remapElement as #remapElement
* @borrows module:recalculate.recalculateDimensions as #recalculateDimensions
*
* @borrows module:utilities.cleanupElement as #cleanupElement
* @borrows module:utilities.getStrokedBBoxDefaultVisible as #getStrokedBBox
* @borrows module:utilities.getVisibleElements as #getVisibleElements
* @borrows module:utilities.findDefs as #findDefs
* @borrows module:utilities.getUrlFromAttr as #getUrlFromAttr
* @borrows module:utilities.getHref as #getHref
* @borrows module:utilities.setHref as #setHref
* @borrows module:utilities.getRotationAngle as #getRotationAngle
* @borrows module:utilities.getBBox as #getBBox
* @borrows module:utilities.getElem as #getElem
* @borrows module:utilities.getRefElem as #getRefElem
* @borrows module:utilities.assignAttributes as #assignAttributes
*
* @borrows module:SVGTransformList.getTransformList as #getTransformList
* @borrows module:math.matrixMultiply as #matrixMultiply
* @borrows module:math.hasMatrixTransform as #hasMatrixTransform
* @borrows module:math.transformListToTransform as #transformListToTransform
* @borrows module:units.convertToNum as #convertToNum
* @borrows module:sanitize.sanitizeSvg as #sanitizeSvg
* @borrows module:path.pathActions.linkControlPoints as #linkControlPoints
*/
var SvgCanvas =
/**
* @param {HTMLElement} container - The container HTML element that should hold the SVG root element
* @param {module:SVGEditor.curConfig} config - An object that contains configuration data
*/
function SvgCanvas(container, config) {
_classCallCheck(this, SvgCanvas);
// Alias Namespace constants
// Default configuration options
var curConfig = {
show_outside_canvas: true,
selectNew: true,
dimensions: [640, 480]
}; // Update config with new one if given
if (config) {
$$8.extend(curConfig, config);
} // Array with width/height of canvas
var dimensions = curConfig.dimensions;
var canvas = this; // eslint-disable-line consistent-this
// "document" element associated with the container (same as window.document using default svg-editor.js)
// NOTE: This is not actually a SVG document, but an HTML document.
var svgdoc = container.ownerDocument; // This is a container for the document being edited, not the document itself.
/**
* @name module:svgcanvas~svgroot
* @type {SVGSVGElement}
*/
var svgroot = svgdoc.importNode(text2xml('<svg id="svgroot" xmlns="' + NS.SVG + '" xlinkns="' + NS.XLINK + '" ' + 'width="' + dimensions[0] + '" height="' + dimensions[1] + '" x="' + dimensions[0] + '" y="' + dimensions[1] + '" overflow="visible">' + '<defs>' + '<filter id="canvashadow" filterUnits="objectBoundingBox">' + '<feGaussianBlur in="SourceAlpha" stdDeviation="4" result="blur"/>' + '<feOffset in="blur" dx="5" dy="5" result="offsetBlur"/>' + '<feMerge>' + '<feMergeNode in="offsetBlur"/>' + '<feMergeNode in="SourceGraphic"/>' + '</feMerge>' + '</filter>' + '</defs>' + '</svg>').documentElement, true);
container.append(svgroot);
/**
* The actual element that represents the final output SVG element.
* @name module:svgcanvas~svgcontent
* @type {SVGSVGElement}
*/
var svgcontent = svgdoc.createElementNS(NS.SVG, 'svg');
/**
* This function resets the svgcontent element while keeping it in the DOM.
* @function module:svgcanvas.SvgCanvas#clearSvgContentElement
* @returns {void}
*/
var clearSvgContentElement = canvas.clearSvgContentElement = function () {
$$8(svgcontent).empty(); // TODO: Clear out all other attributes first?
$$8(svgcontent).attr({
id: 'svgcontent',
width: dimensions[0],
height: dimensions[1],
x: dimensions[0],
y: dimensions[1],
overflow: curConfig.show_outside_canvas ? 'visible' : 'hidden',
xmlns: NS.SVG,
'xmlns:se': NS.SE,
'xmlns:xlink': NS.XLINK
}).appendTo(svgroot); // TODO: make this string optional and set by the client
var comment = svgdoc.createComment(' Created with SVG-edit - https://github.com/SVG-Edit/svgedit');
svgcontent.append(comment);
};
clearSvgContentElement(); // Prefix string for element IDs
var idprefix = 'svg_';
/**
* Changes the ID prefix to the given value.
* @function module:svgcanvas.SvgCanvas#setIdPrefix
* @param {string} p - String with the new prefix
* @returns {void}
*/
canvas.setIdPrefix = function (p) {
idprefix = p;
};
/**
* Current `draw.Drawing` object.
* @type {module:draw.Drawing}
* @name module:svgcanvas.SvgCanvas#current_drawing_
*/
canvas.current_drawing_ = new Drawing(svgcontent, idprefix);
/**
* Returns the current Drawing.
* @name module:svgcanvas.SvgCanvas#getCurrentDrawing
* @type {module:draw.DrawCanvasInit#getCurrentDrawing}
*/
var getCurrentDrawing = canvas.getCurrentDrawing = function () {
return canvas.current_drawing_;
};
/**
* Float displaying the current zoom level (1 = 100%, .5 = 50%, etc.).
* @type {Float}
*/
var currentZoom = 1; // pointer to current group (for in-group editing)
var currentGroup = null; // Object containing data for the currently selected styles
var allProperties = {
shape: {
fill: (curConfig.initFill.color === 'none' ? '' : '#') + curConfig.initFill.color,
fill_paint: null,
fill_opacity: curConfig.initFill.opacity,
stroke: '#' + curConfig.initStroke.color,
stroke_paint: null,
stroke_opacity: curConfig.initStroke.opacity,
stroke_width: curConfig.initStroke.width,
stroke_dasharray: 'none',
stroke_linejoin: 'miter',
stroke_linecap: 'butt',
opacity: curConfig.initOpacity
}
};
allProperties.text = $$8.extend(true, {}, allProperties.shape);
$$8.extend(allProperties.text, {
fill: '#000000',
stroke_width: curConfig.text && curConfig.text.stroke_width,
font_size: curConfig.text && curConfig.text.font_size,
font_family: curConfig.text && curConfig.text.font_family
}); // Current shape style properties
var curShape = allProperties.shape; // Array with all the currently selected elements
// default size of 1 until it needs to grow bigger
var selectedElements = [];
/**
* @typedef {PlainObject} module:svgcanvas.SVGAsJSON
* @property {string} element
* @property {PlainObject<string, string>} attr
* @property {module:svgcanvas.SVGAsJSON[]} children
*/
/**
* @function module:svgcanvas.SvgCanvas#getContentElem
* @param {Text|Element} data
* @returns {module:svgcanvas.SVGAsJSON}
*/
var getJsonFromSvgElement = this.getJsonFromSvgElement = function (data) {
// Text node
if (data.nodeType === 3) return data.nodeValue;
var retval = {
element: data.tagName,
// namespace: nsMap[data.namespaceURI],
attr: {},
children: []
}; // Iterate attributes
for (var i = 0, attr; attr = data.attributes[i]; i++) {
retval.attr[attr.name] = attr.value;
} // Iterate children
for (var _i = 0, node; node = data.childNodes[_i]; _i++) {
retval.children[_i] = getJsonFromSvgElement(node);
}
return retval;
};
/**
* This should really be an intersection implementing all rather than a union.
* @name module:svgcanvas.SvgCanvas#addSVGElementFromJson
* @type {module:utilities.EditorContext#addSVGElementFromJson|module:path.EditorContext#addSVGElementFromJson}
*/
var addSVGElementFromJson = this.addSVGElementFromJson = function (data) {
if (typeof data === 'string') return svgdoc.createTextNode(data);
var shape = getElem(data.attr.id); // if shape is a path but we need to create a rect/ellipse, then remove the path
var currentLayer = getCurrentDrawing().getCurrentLayer();
if (shape && data.element !== shape.tagName) {
shape.remove();
shape = null;
}
if (!shape) {
var ns = data.namespace || NS.SVG;
shape = svgdoc.createElementNS(ns, data.element);
if (currentLayer) {
(currentGroup || currentLayer).append(shape);
}
}
if (data.curStyles) {
assignAttributes(shape, {
fill: curShape.fill,
stroke: curShape.stroke,
'stroke-width': curShape.stroke_width,
'stroke-dasharray': curShape.stroke_dasharray,
'stroke-linejoin': curShape.stroke_linejoin,
'stroke-linecap': curShape.stroke_linecap,
'stroke-opacity': curShape.stroke_opacity,
'fill-opacity': curShape.fill_opacity,
opacity: curShape.opacity / 2,
style: 'pointer-events:inherit'
});
}
assignAttributes(shape, data.attr);
cleanupElement(shape); // Children
if (data.children) {
data.children.forEach(function (child) {
shape.append(addSVGElementFromJson(child));
});
}
return shape;
};
canvas.getTransformList = getTransformList;
canvas.matrixMultiply = matrixMultiply;
canvas.hasMatrixTransform = hasMatrixTransform;
canvas.transformListToTransform = transformListToTransform;
/**
* @type {module:utilities.EditorContext#getBaseUnit}
*/
var getBaseUnit = function getBaseUnit() {
return curConfig.baseUnit;
};
/**
* Initialize from units.js.
* Send in an object implementing the ElementContainer interface (see units.js).
*/
init(
/**
* @implements {module:units.ElementContainer}
*/
{
getBaseUnit: getBaseUnit,
getElement: getElem,
getHeight: function getHeight() {
return svgcontent.getAttribute('height') / currentZoom;
},
getWidth: function getWidth() {
return svgcontent.getAttribute('width') / currentZoom;
},
getRoundDigits: function getRoundDigits() {
return saveOptions.round_digits;
}
});
canvas.convertToNum = convertToNum;
/**
* This should really be an intersection implementing all rather than a union.
* @type {module:draw.DrawCanvasInit#getSVGContent|module:utilities.EditorContext#getSVGContent}
*/
var getSVGContent = function getSVGContent() {
return svgcontent;
};
/**
* Should really be an intersection with all needing to apply rather than a union.
* @name module:svgcanvas.SvgCanvas#getSelectedElements
* @type {module:utilities.EditorContext#getSelectedElements|module:draw.DrawCanvasInit#getSelectedElements|module:path.EditorContext#getSelectedElements}
*/
var getSelectedElements = this.getSelectedElems = function () {
return selectedElements;
};
var pathActions$1 = pathActions;
/**
* This should actually be an intersection as all interfaces should be met.
* @type {module:utilities.EditorContext#getSVGRoot|module:recalculate.EditorContext#getSVGRoot|module:coords.EditorContext#getSVGRoot|module:path.EditorContext#getSVGRoot}
*/
var getSVGRoot = function getSVGRoot() {
return svgroot;
};
init$2(
/**
* @implements {module:utilities.EditorContext}
*/
{
pathActions: pathActions$1,
// Ok since not modifying
getSVGContent: getSVGContent,
addSVGElementFromJson: addSVGElementFromJson,
getSelectedElements: getSelectedElements,
getDOMDocument: function getDOMDocument() {
return svgdoc;
},
getDOMContainer: function getDOMContainer() {
return container;
},
getSVGRoot: getSVGRoot,
// TODO: replace this mostly with a way to get the current drawing.
getBaseUnit: getBaseUnit,
getSnappingStep: function getSnappingStep() {
return curConfig.snappingStep;
}
});
canvas.findDefs = findDefs;
canvas.getUrlFromAttr = getUrlFromAttr;
canvas.getHref = getHref;
canvas.setHref = setHref;
/* const getBBox = */
canvas.getBBox = getBBox;
canvas.getRotationAngle = getRotationAngle;
canvas.getElem = getElem;
canvas.getRefElem = getRefElem;
canvas.assignAttributes = assignAttributes;
this.cleanupElement = cleanupElement;
/**
* This should actually be an intersection not a union as all should apply.
* @type {module:coords.EditorContext#getGridSnapping|module:path.EditorContext#getGridSnapping}
*/
var getGridSnapping = function getGridSnapping() {
return curConfig.gridSnapping;
};
init$4(
/**
* @implements {module:coords.EditorContext}
*/
{
getDrawing: function getDrawing() {
return getCurrentDrawing();
},
getSVGRoot: getSVGRoot,
getGridSnapping: getGridSnapping
});
this.remapElement = remapElement;
init$5(
/**
* @implements {module:recalculate.EditorContext}
*/
{
getSVGRoot: getSVGRoot,
getStartTransform: function getStartTransform() {
return startTransform;
},
setStartTransform: function setStartTransform(transform) {
startTransform = transform;
}
});
this.recalculateDimensions = recalculateDimensions; // import from sanitize.js
var nsMap = getReverseNS();
canvas.sanitizeSvg = sanitizeSvg;
/**
* @name undoMgr
* @memberof module:svgcanvas.SvgCanvas#
* @type {module:history.HistoryEventHandler}
*/
var undoMgr = canvas.undoMgr = new UndoManager$1({
/**
* @param {string} eventType One of the HistoryEvent types
* @param {module:history.HistoryCommand} cmd Fulfills the HistoryCommand interface
* @fires module:svgcanvas.SvgCanvas#event:changed
* @returns {void}
*/
handleHistoryEvent: function handleHistoryEvent(eventType, cmd) {
var EventTypes = HistoryEventTypes$1; // TODO: handle setBlurOffsets.
if (eventType === EventTypes.BEFORE_UNAPPLY || eventType === EventTypes.BEFORE_APPLY) {
canvas.clearSelection();
} else if (eventType === EventTypes.AFTER_APPLY || eventType === EventTypes.AFTER_UNAPPLY) {
var elems = cmd.elements();
canvas.pathActions.clear();
call('changed', elems);
var cmdType = cmd.type();
var isApply = eventType === EventTypes.AFTER_APPLY;
if (cmdType === MoveElementCommand$1.type()) {
var parent = isApply ? cmd.newParent : cmd.oldParent;
if (parent === svgcontent) {
identifyLayers();
}
} else if (cmdType === InsertElementCommand$1.type() || cmdType === RemoveElementCommand$1.type()) {
if (cmd.parent === svgcontent) {
identifyLayers();
}
if (cmdType === InsertElementCommand$1.type()) {
if (isApply) {
restoreRefElems(cmd.elem);
}
} else if (!isApply) {
restoreRefElems(cmd.elem);
}
if (cmd.elem.tagName === 'use') {
setUseData(cmd.elem);
}
} else if (cmdType === ChangeElementCommand$1.type()) {
// if we are changing layer names, re-identify all layers
if (cmd.elem.tagName === 'title' && cmd.elem.parentNode.parentNode === svgcontent) {
identifyLayers();
}
var values = isApply ? cmd.newValues : cmd.oldValues; // If stdDeviation was changed, update the blur.
if (values.stdDeviation) {
canvas.setBlurOffsets(cmd.elem.parentNode, values.stdDeviation);
} // This is resolved in later versions of webkit, perhaps we should
// have a featured detection for correct 'use' behavior?
// ——————————
// Remove & Re-add hack for Webkit (issue 775)
// if (cmd.elem.tagName === 'use' && isWebkit()) {
// const {elem} = cmd;
// if (!elem.getAttribute('x') && !elem.getAttribute('y')) {
// const parent = elem.parentNode;
// const sib = elem.nextSibling;
// elem.remove();
// parent.insertBefore(elem, sib);
// // Ok to replace above with this? `sib.before(elem);`
// }
// }
}
}
}
});
/**
* This should really be an intersection applying to all types rather than a union.
* @name module:svgcanvas~addCommandToHistory
* @type {module:path.EditorContext#addCommandToHistory|module:draw.DrawCanvasInit#addCommandToHistory}
*/
var addCommandToHistory = function addCommandToHistory(cmd) {
canvas.undoMgr.addCommandToHistory(cmd);
};
/**
* This should really be an intersection applying to all types rather than a union.
* @name module:svgcanvas.SvgCanvas#getZoom
* @type {module:path.EditorContext#getCurrentZoom|module:select.SVGFactory#getCurrentZoom}
*/
var getCurrentZoom = this.getZoom = function () {
return currentZoom;
};
/**
* This method rounds the incoming value to the nearest value based on the `currentZoom`
* @name module:svgcanvas.SvgCanvas#round
* @type {module:path.EditorContext#round}
*/
var round = this.round = function (val) {
return parseInt(val * currentZoom) / currentZoom;
};
init$6(curConfig,
/**
* Export to select.js.
* @implements {module:select.SVGFactory}
*/
{
createSVGElement: function createSVGElement(jsonMap) {
return canvas.addSVGElementFromJson(jsonMap);
},
svgRoot: function svgRoot() {
return svgroot;
},
svgContent: function svgContent() {
return svgcontent;
},
getCurrentZoom: getCurrentZoom
});
/**
* This object manages selectors for us.
* @name module:svgcanvas.SvgCanvas#selectorManager
* @type {module:select.SelectorManager}
*/
var selectorManager = this.selectorManager = getSelectorManager();
/**
* @name module:svgcanvas.SvgCanvas#getNextId
* @type {module:path.EditorContext#getNextId}
*/
var getNextId = canvas.getNextId = function () {
return getCurrentDrawing().getNextId();
};
/**
* @name module:svgcanvas.SvgCanvas#getId
* @type {module:path.EditorContext#getId}
*/
var getId = canvas.getId = function () {
return getCurrentDrawing().getId();
};
/**
* The "implements" should really be an intersection applying to all types rather than a union.
* @name module:svgcanvas.SvgCanvas#call
* @type {module:draw.DrawCanvasInit#call|module:path.EditorContext#call}
*/
var call = function call(ev, arg) {
if (events[ev]) {
return events[ev](window, arg);
}
return undefined;
};
/**
* Clears the selection. The 'selected' handler is then optionally called.
* This should really be an intersection applying to all types rather than a union.
* @name module:svgcanvas.SvgCanvas#clearSelection
* @type {module:draw.DrawCanvasInit#clearSelection|module:path.EditorContext#clearSelection}
* @fires module:svgcanvas.SvgCanvas#event:selected
*/
var clearSelection = this.clearSelection = function (noCall) {
selectedElements.forEach(function (elem) {
if (isNullish(elem)) {
return;
}
selectorManager.releaseSelector(elem);
});
selectedElements = [];
if (!noCall) {
call('selected', selectedElements);
}
};
/**
* Adds a list of elements to the selection. The 'selected' handler is then called.
* @name module:svgcanvas.SvgCanvas#addToSelection
* @type {module:path.EditorContext#addToSelection}
* @fires module:svgcanvas.SvgCanvas#event:selected
*/
var addToSelection = this.addToSelection = function (elemsToAdd, showGrips) {
if (!elemsToAdd.length) {
return;
} // find the first null in our selectedElements array
var j = 0;
while (j < selectedElements.length) {
if (isNullish(selectedElements[j])) {
break;
}
++j;
} // now add each element consecutively
var i = elemsToAdd.length;
while (i--) {
var elem = elemsToAdd[i];
if (!elem) {
continue;
}
var bbox = getBBox(elem);
if (!bbox) {
continue;
}
if (elem.tagName === 'a' && elem.childNodes.length === 1) {
// Make "a" element's child be the selected element
elem = elem.firstChild;
} // if it's not already there, add it
if (!selectedElements.includes(elem)) {
selectedElements[j] = elem; // only the first selectedBBoxes element is ever used in the codebase these days
// if (j === 0) selectedBBoxes[0] = utilsGetBBox(elem);
j++;
var sel = selectorManager.requestSelector(elem, bbox);
if (selectedElements.length > 1) {
sel.showGrips(false);
}
}
}
2020-01-25 02:32:24 +00:00
if (!selectedElements.length) {
return;
}
call('selected', selectedElements);
if (showGrips || selectedElements.length === 1) {
selectorManager.requestSelector(selectedElements[0]).showGrips(true);
} else {
selectorManager.requestSelector(selectedElements[0]).showGrips(false);
} // make sure the elements are in the correct order
// See: https://www.w3.org/TR/DOM-Level-3-Core/core.html#Node3-compareDocumentPosition
selectedElements.sort(function (a, b) {
if (a && b && a.compareDocumentPosition) {
return 3 - (b.compareDocumentPosition(a) & 6); // eslint-disable-line no-bitwise
}
if (isNullish(a)) {
return 1;
}
return 0;
}); // Make sure first elements are not null
while (isNullish(selectedElements[0])) {
selectedElements.shift(0);
}
};
/**
* @type {module:path.EditorContext#getOpacity}
*/
var getOpacity = function getOpacity() {
return curShape.opacity;
};
/**
* @name module:svgcanvas.SvgCanvas#getMouseTarget
* @type {module:path.EditorContext#getMouseTarget}
*/
var getMouseTarget = this.getMouseTarget = function (evt) {
if (isNullish(evt)) {
return null;
}
var mouseTarget = evt.target; // if it was a <use>, Opera and WebKit return the SVGElementInstance
if (mouseTarget.correspondingUseElement) {
mouseTarget = mouseTarget.correspondingUseElement;
} // for foreign content, go up until we find the foreignObject
// WebKit browsers set the mouse target to the svgcanvas div
if ([NS.MATH, NS.HTML].includes(mouseTarget.namespaceURI) && mouseTarget.id !== 'svgcanvas') {
while (mouseTarget.nodeName !== 'foreignObject') {
mouseTarget = mouseTarget.parentNode;
if (!mouseTarget) {
return svgroot;
}
}
} // Get the desired mouseTarget with jQuery selector-fu
// If it's root-like, select the root
var currentLayer = getCurrentDrawing().getCurrentLayer();
if ([svgroot, container, svgcontent, currentLayer].includes(mouseTarget)) {
return svgroot;
}
var $target = $$8(mouseTarget); // If it's a selection grip, return the grip parent
if ($target.closest('#selectorParentGroup').length) {
// While we could instead have just returned mouseTarget,
// this makes it easier to indentify as being a selector grip
return selectorManager.selectorParentGroup;
}
while (mouseTarget.parentNode !== (currentGroup || currentLayer)) {
mouseTarget = mouseTarget.parentNode;
} //
// // go up until we hit a child of a layer
// while (mouseTarget.parentNode.parentNode.tagName == 'g') {
// mouseTarget = mouseTarget.parentNode;
// }
// Webkit bubbles the mouse event all the way up to the div, so we
// set the mouseTarget to the svgroot like the other browsers
// if (mouseTarget.nodeName.toLowerCase() == 'div') {
// mouseTarget = svgroot;
// }
return mouseTarget;
};
/**
* @namespace {module:path.pathActions} pathActions
* @memberof module:svgcanvas.SvgCanvas#
* @see module:path.pathActions
*/
canvas.pathActions = pathActions$1;
/**
* @type {module:path.EditorContext#resetD}
*/
function resetD(p) {
p.setAttribute('d', pathActions$1.convertPath(p));
}
init$1(
/**
* @implements {module:path.EditorContext}
*/
{
selectorManager: selectorManager,
// Ok since not changing
canvas: canvas,
// Ok since not changing
call: call,
resetD: resetD,
round: round,
clearSelection: clearSelection,
addToSelection: addToSelection,
addCommandToHistory: addCommandToHistory,
remapElement: remapElement,
addSVGElementFromJson: addSVGElementFromJson,
getGridSnapping: getGridSnapping,
getOpacity: getOpacity,
getSelectedElements: getSelectedElements,
getContainer: function getContainer() {
return container;
},
setStarted: function setStarted(s) {
started = s;
},
getRubberBox: function getRubberBox() {
return rubberBox;
},
setRubberBox: function setRubberBox(rb) {
rubberBox = rb;
return rubberBox;
},
/**
* @param {PlainObject} ptsInfo
* @param {boolean} ptsInfo.closedSubpath
* @param {SVGCircleElement[]} ptsInfo.grips
* @fires module:svgcanvas.SvgCanvas#event:pointsAdded
* @fires module:svgcanvas.SvgCanvas#event:selected
* @returns {void}
*/
addPtsToSelection: function addPtsToSelection(_ref) {
var closedSubpath = _ref.closedSubpath,
grips = _ref.grips;
// TODO: Correct this:
pathActions$1.canDeleteNodes = true;
pathActions$1.closed_subpath = closedSubpath;
call('pointsAdded', {
closedSubpath: closedSubpath,
grips: grips
});
call('selected', grips);
},
/**
* @param {PlainObject} changes
* @param {ChangeElementCommand} changes.cmd
* @param {SVGPathElement} changes.elem
* @fires module:svgcanvas.SvgCanvas#event:changed
* @returns {void}
*/
endChanges: function endChanges(_ref2) {
var cmd = _ref2.cmd,
elem = _ref2.elem;
addCommandToHistory(cmd);
call('changed', [elem]);
},
getCurrentZoom: getCurrentZoom,
getId: getId,
getNextId: getNextId,
getMouseTarget: getMouseTarget,
getCurrentMode: function getCurrentMode() {
return currentMode;
},
setCurrentMode: function setCurrentMode(cm) {
currentMode = cm;
return currentMode;
},
getDrawnPath: function getDrawnPath() {
return drawnPath;
},
setDrawnPath: function setDrawnPath(dp) {
drawnPath = dp;
return drawnPath;
},
getSVGRoot: getSVGRoot
}); // Interface strings, usually for title elements
var uiStrings = {};
var visElems = 'a,circle,ellipse,foreignObject,g,image,line,path,polygon,polyline,rect,svg,text,tspan,use';
var refAttrs = ['clip-path', 'fill', 'filter', 'marker-end', 'marker-mid', 'marker-start', 'mask', 'stroke'];
var elData = $$8.data; // Animation element to change the opacity of any newly created element
var opacAni = document.createElementNS(NS.SVG, 'animate');
$$8(opacAni).attr({
attributeName: 'opacity',
begin: 'indefinite',
dur: 1,
fill: 'freeze'
}).appendTo(svgroot);
var restoreRefElems = function restoreRefElems(elem) {
// Look for missing reference elements, restore any found
var attrs = $$8(elem).attr(refAttrs);
Object.values(attrs).forEach(function (val) {
if (val && val.startsWith('url(')) {
var id = getUrlFromAttr(val).substr(1);
var ref = getElem(id);
if (!ref) {
findDefs().append(removedElements[id]);
delete removedElements[id];
}
}
});
var childs = elem.getElementsByTagName('*');
if (childs.length) {
for (var i = 0, l = childs.length; i < l; i++) {
restoreRefElems(childs[i]);
}
}
}; // (function () {
// TODO For Issue 208: this is a start on a thumbnail
// const svgthumb = svgdoc.createElementNS(NS.SVG, 'use');
// svgthumb.setAttribute('width', '100');
// svgthumb.setAttribute('height', '100');
// setHref(svgthumb, '#svgcontent');
// svgroot.append(svgthumb);
// }());
/**
* @typedef {PlainObject} module:svgcanvas.SaveOptions
* @property {boolean} apply
* @property {"embed"} [image]
* @property {Integer} round_digits
*/
// Object to contain image data for raster images that were found encodable
var encodableImages = {},
// Object with save options
/**
* @type {module:svgcanvas.SaveOptions}
*/
saveOptions = {
round_digits: 5
},
// Object with IDs for imported files, to see if one was already added
importIds = {},
// Current text style properties
curText = allProperties.text,
// Object to contain all included extensions
extensions = {},
// Map of deleted reference elements
removedElements = {};
var // String with image URL of last loadable image
lastGoodImgUrl = curConfig.imgPath + 'logo.png',
// Boolean indicating whether or not a draw action has been started
started = false,
// String with an element's initial transform attribute value
startTransform = null,
// String indicating the current editor mode
currentMode = 'select',
// String with the current direction in which an element is being resized
currentResizeMode = 'none',
// Current general properties
curProperties = curShape,
// Array with selected elements' Bounding box object
// selectedBBoxes = new Array(1),
// The DOM element that was just selected
justSelected = null,
// DOM element for selection rectangle drawn by the user
rubberBox = null,
// Array of current BBoxes, used in getIntersectionList().
curBBoxes = [],
// Canvas point for the most recent right click
lastClickPoint = null;
this.runExtension = function (name, action, vars) {
return this.runExtensions(action, vars, false, function (n) {
return n === name;
});
};
/**
* @typedef {module:svgcanvas.ExtensionMouseDownStatus|module:svgcanvas.ExtensionMouseUpStatus|module:svgcanvas.ExtensionIDsUpdatedStatus|module:locale.ExtensionLocaleData[]|void} module:svgcanvas.ExtensionStatus
* @tutorial ExtensionDocs
*/
/**
* @callback module:svgcanvas.ExtensionVarBuilder
* @param {string} name The name of the extension
* @returns {module:svgcanvas.SvgCanvas#event:ext_addLangData}
*/
/**
* @callback module:svgcanvas.ExtensionNameFilter
* @param {string} name
* @returns {boolean}
*/
/**
* @todo Consider: Should this return an array by default, so extension results aren't overwritten?
* @todo Would be easier to document if passing in object with key of action and vars as value; could then define an interface which tied both together
* @function module:svgcanvas.SvgCanvas#runExtensions
* @param {"mouseDown"|"mouseMove"|"mouseUp"|"zoomChanged"|"IDsUpdated"|"canvasUpdated"|"toolButtonStateUpdate"|"selectedChanged"|"elementTransition"|"elementChanged"|"langReady"|"langChanged"|"addLangData"|"onNewDocument"|"workareaResized"} action
* @param {module:svgcanvas.SvgCanvas#event:ext_mouseDown|module:svgcanvas.SvgCanvas#event:ext_mouseMove|module:svgcanvas.SvgCanvas#event:ext_mouseUp|module:svgcanvas.SvgCanvas#event:ext_zoomChanged|module:svgcanvas.SvgCanvas#event:ext_IDsUpdated|module:svgcanvas.SvgCanvas#event:ext_canvasUpdated|module:svgcanvas.SvgCanvas#event:ext_toolButtonStateUpdate|module:svgcanvas.SvgCanvas#event:ext_selectedChanged|module:svgcanvas.SvgCanvas#event:ext_elementTransition|module:svgcanvas.SvgCanvas#event:ext_elementChanged|module:svgcanvas.SvgCanvas#event:ext_langReady|module:svgcanvas.SvgCanvas#event:ext_langChanged|module:svgcanvas.SvgCanvas#event:ext_addLangData|module:svgcanvas.SvgCanvas#event:ext_onNewDocument|module:svgcanvas.SvgCanvas#event:ext_workareaResized|module:svgcanvas.ExtensionVarBuilder} [vars]
* @param {boolean} [returnArray]
* @param {module:svgcanvas.ExtensionNameFilter} nameFilter
* @returns {GenericArray<module:svgcanvas.ExtensionStatus>|module:svgcanvas.ExtensionStatus|false} See {@tutorial ExtensionDocs} on the ExtensionStatus.
*/
var runExtensions = this.runExtensions = function (action, vars, returnArray, nameFilter) {
var result = returnArray ? [] : false;
$$8.each(extensions, function (name, ext) {
if (nameFilter && !nameFilter(name)) {
return;
}
if (ext && action in ext) {
if (typeof vars === 'function') {
vars = vars(name); // ext, action
}
if (returnArray) {
result.push(ext[action](vars));
} else {
result = ext[action](vars);
}
}
});
return result;
};
/**
* @typedef {PlainObject} module:svgcanvas.ExtensionMouseDownStatus
* @property {boolean} started Indicates that creating/editing has started
*/
/**
* @typedef {PlainObject} module:svgcanvas.ExtensionMouseUpStatus
* @property {boolean} keep Indicates if the current element should be kept
* @property {boolean} started Indicates if editing should still be considered as "started"
* @property {Element} element The element being affected
*/
/**
* @typedef {PlainObject} module:svgcanvas.ExtensionIDsUpdatedStatus
* @property {string[]} remove Contains string IDs (used by `ext-connector.js`)
*/
/**
* @interface module:svgcanvas.ExtensionInitResponse
* @property {module:SVGEditor.ContextTool[]|PlainObject<string, module:SVGEditor.ContextTool>} [context_tools]
* @property {module:SVGEditor.Button[]|PlainObject<Integer, module:SVGEditor.Button>} [buttons]
* @property {string} [svgicons] The location of a local SVG or SVGz file
*/
/**
* @function module:svgcanvas.ExtensionInitResponse#mouseDown
* @param {module:svgcanvas.SvgCanvas#event:ext_mouseDown} arg
* @returns {void|module:svgcanvas.ExtensionMouseDownStatus}
*/
/**
* @function module:svgcanvas.ExtensionInitResponse#mouseMove
* @param {module:svgcanvas.SvgCanvas#event:ext_mouseMove} arg
* @returns {void}
*/
/**
* @function module:svgcanvas.ExtensionInitResponse#mouseUp
* @param {module:svgcanvas.SvgCanvas#event:ext_mouseUp} arg
* @returns {module:svgcanvas.ExtensionMouseUpStatus}
*/
/**
* @function module:svgcanvas.ExtensionInitResponse#zoomChanged
* @param {module:svgcanvas.SvgCanvas#event:ext_zoomChanged} arg
* @returns {void}
*/
/**
* @function module:svgcanvas.ExtensionInitResponse#IDsUpdated
* @param {module:svgcanvas.SvgCanvas#event:ext_IDsUpdated} arg
* @returns {module:svgcanvas.ExtensionIDsUpdatedStatus}
*/
/**
* @function module:svgcanvas.ExtensionInitResponse#canvasUpdated
* @param {module:svgcanvas.SvgCanvas#event:ext_canvasUpdated} arg
* @returns {void}
*/
/**
* @function module:svgcanvas.ExtensionInitResponse#toolButtonStateUpdate
* @param {module:svgcanvas.SvgCanvas#event:ext_toolButtonStateUpdate} arg
* @returns {void}
*/
/**
* @function module:svgcanvas.ExtensionInitResponse#selectedChanged
* @param {module:svgcanvas.SvgCanvas#event:ext_selectedChanged} arg
* @returns {void}
*/
/**
* @function module:svgcanvas.ExtensionInitResponse#elementTransition
* @param {module:svgcanvas.SvgCanvas#event:ext_elementTransition} arg
* @returns {void}
*/
/**
* @function module:svgcanvas.ExtensionInitResponse#elementChanged
* @param {module:svgcanvas.SvgCanvas#event:ext_elementChanged} arg
* @returns {void}
*/
/**
* @function module:svgcanvas.ExtensionInitResponse#langReady
* @param {module:svgcanvas.SvgCanvas#event:ext_langReady} arg
* @returns {void}
*/
/**
* @function module:svgcanvas.ExtensionInitResponse#langChanged
* @param {module:svgcanvas.SvgCanvas#event:ext_langChanged} arg
* @returns {void}
*/
/**
* @function module:svgcanvas.ExtensionInitResponse#addLangData
* @param {module:svgcanvas.SvgCanvas#event:ext_addLangData} arg
* @returns {Promise<module:locale.ExtensionLocaleData>} Resolves to {@link module:locale.ExtensionLocaleData}
*/
/**
* @function module:svgcanvas.ExtensionInitResponse#onNewDocument
* @param {module:svgcanvas.SvgCanvas#event:ext_onNewDocument} arg
* @returns {void}
*/
/**
* @function module:svgcanvas.ExtensionInitResponse#workareaResized
* @param {module:svgcanvas.SvgCanvas#event:ext_workareaResized} arg
* @returns {void}
*/
/**
* @function module:svgcanvas.ExtensionInitResponse#callback
* @this module:SVGEditor
* @param {module:svgcanvas.SvgCanvas#event:ext_callback} arg
* @returns {void}
*/
/**
* @callback module:svgcanvas.ExtensionInitCallback
* @this module:SVGEditor
* @param {module:svgcanvas.ExtensionArgumentObject} arg
* @returns {Promise<module:svgcanvas.ExtensionInitResponse|void>} Resolves to [ExtensionInitResponse]{@link module:svgcanvas.ExtensionInitResponse} or `undefined`
*/
/**
* @typedef {PlainObject} module:svgcanvas.ExtensionInitArgs
* @property {external:jQuery} $
* @property {module:SVGEditor~ImportLocale} importLocale
*/
/**
* Add an extension to the editor.
* @function module:svgcanvas.SvgCanvas#addExtension
* @param {string} name - String with the ID of the extension. Used internally; no need for i18n.
* @param {module:svgcanvas.ExtensionInitCallback} [extInitFunc] - Function supplied by the extension with its data
* @param {module:svgcanvas.ExtensionInitArgs} initArgs
* @fires module:svgcanvas.SvgCanvas#event:extension_added
* @throws {TypeError|Error} `TypeError` if `extInitFunc` is not a function, `Error`
* if extension of supplied name already exists
* @returns {Promise<void>} Resolves to `undefined`
*/
this.addExtension = /*#__PURE__*/function () {
var _ref3 = _asyncToGenerator( /*#__PURE__*/regeneratorRuntime.mark(function _callee(name, extInitFunc, _ref4) {
var jq, importLocale, argObj, extObj;
return regeneratorRuntime.wrap(function _callee$(_context) {
while (1) {
switch (_context.prev = _context.next) {
case 0:
jq = _ref4.$, importLocale = _ref4.importLocale;
if (!(typeof extInitFunc !== 'function')) {
_context.next = 3;
break;
}
throw new TypeError('Function argument expected for `svgcanvas.addExtension`');
case 3:
if (!(name in extensions)) {
_context.next = 5;
break;
}
throw new Error('Cannot add extension "' + name + '", an extension by that name already exists.');
case 5:
// Provide private vars/funcs here. Is there a better way to do this?
/**
* @typedef {module:svgcanvas.PrivateMethods} module:svgcanvas.ExtensionArgumentObject
* @property {SVGSVGElement} svgroot See {@link module:svgcanvas~svgroot}
* @property {SVGSVGElement} svgcontent See {@link module:svgcanvas~svgcontent}
* @property {!(string|Integer)} nonce See {@link module:draw.Drawing#getNonce}
* @property {module:select.SelectorManager} selectorManager
* @property {module:SVGEditor~ImportLocale} importLocale
*/
/**
* @type {module:svgcanvas.ExtensionArgumentObject}
* @see {@link module:svgcanvas.PrivateMethods} source for the other methods/properties
*/
argObj = $$8.extend(canvas.getPrivateMethods(), {
$: jq,
importLocale: importLocale,
svgroot: svgroot,
svgcontent: svgcontent,
nonce: getCurrentDrawing().getNonce(),
selectorManager: selectorManager
});
_context.next = 8;
return extInitFunc(argObj);
case 8:
extObj = _context.sent;
if (extObj) {
extObj.name = name;
}
extensions[name] = extObj;
return _context.abrupt("return", call('extension_added', extObj));
case 12:
case "end":
return _context.stop();
}
}
}, _callee);
}));
return function (_x, _x2, _x3) {
return _ref3.apply(this, arguments);
};
}();
/**
* This method sends back an array or a NodeList full of elements that
* intersect the multi-select rubber-band-box on the currentLayer only.
*
* We brute-force `getIntersectionList` for browsers that do not support it (Firefox).
*
* Reference:
* Firefox does not implement `getIntersectionList()`, see {@link https://bugzilla.mozilla.org/show_bug.cgi?id=501421}.
* @function module:svgcanvas.SvgCanvas#getIntersectionList
* @param {SVGRect} rect
* @returns {Element[]|NodeList} Bbox elements
*/
var getIntersectionList = this.getIntersectionList = function (rect) {
if (isNullish(rubberBox)) {
return null;
}
var parent = currentGroup || getCurrentDrawing().getCurrentLayer();
var rubberBBox;
if (!rect) {
rubberBBox = rubberBox.getBBox();
var bb = svgcontent.createSVGRect();
['x', 'y', 'width', 'height', 'top', 'right', 'bottom', 'left'].forEach(function (o) {
bb[o] = rubberBBox[o] / currentZoom;
});
rubberBBox = bb;
} else {
rubberBBox = svgcontent.createSVGRect();
rubberBBox.x = rect.x;
rubberBBox.y = rect.y;
rubberBBox.width = rect.width;
rubberBBox.height = rect.height;
}
var resultList = null;
if (!isIE()) {
if (typeof svgroot.getIntersectionList === 'function') {
// Offset the bbox of the rubber box by the offset of the svgcontent element.
rubberBBox.x += parseInt(svgcontent.getAttribute('x'));
rubberBBox.y += parseInt(svgcontent.getAttribute('y'));
resultList = svgroot.getIntersectionList(rubberBBox, parent);
}
}
if (isNullish(resultList) || typeof resultList.item !== 'function') {
resultList = [];
if (!curBBoxes.length) {
// Cache all bboxes
curBBoxes = getVisibleElementsAndBBoxes(parent);
}
var i = curBBoxes.length;
while (i--) {
if (!rubberBBox.width) {
continue;
}
if (rectsIntersect(rubberBBox, curBBoxes[i].bbox)) {
resultList.push(curBBoxes[i].elem);
}
}
} // addToSelection expects an array, but it's ok to pass a NodeList
// because using square-bracket notation is allowed:
// https://www.w3.org/TR/DOM-Level-2-Core/ecma-script-binding.html
return resultList;
};
this.getStrokedBBox = getStrokedBBoxDefaultVisible;
this.getVisibleElements = getVisibleElements;
/**
* @typedef {PlainObject} ElementAndBBox
* @property {Element} elem - The element
* @property {module:utilities.BBoxObject} bbox - The element's BBox as retrieved from `getStrokedBBoxDefaultVisible`
*/
/**
* Get all elements that have a BBox (excludes `<defs>`, `<title>`, etc).
* Note that 0-opacity, off-screen etc elements are still considered "visible"
* for this function.
* @function module:svgcanvas.SvgCanvas#getVisibleElementsAndBBoxes
* @param {Element} parent - The parent DOM element to search within
* @returns {ElementAndBBox[]} An array with objects that include:
*/
var getVisibleElementsAndBBoxes = this.getVisibleElementsAndBBoxes = function (parent) {
if (!parent) {
parent = $$8(svgcontent).children(); // Prevent layers from being included
}
var contentElems = [];
$$8(parent).children().each(function (i, elem) {
if (elem.getBBox) {
contentElems.push({
elem: elem,
bbox: getStrokedBBoxDefaultVisible([elem])
});
}
});
return contentElems.reverse();
};
/**
* Wrap an SVG element into a group element, mark the group as 'gsvg'.
* @function module:svgcanvas.SvgCanvas#groupSvgElem
* @param {Element} elem - SVG element to wrap
* @returns {void}
*/
var groupSvgElem = this.groupSvgElem = function (elem) {
var g = document.createElementNS(NS.SVG, 'g');
elem.replaceWith(g);
$$8(g).append(elem).data('gsvg', elem)[0].id = getNextId();
}; // Set scope for these functions
// Object to contain editor event names and callback functions
var events = {};
canvas.call = call;
/**
* Array of what was changed (elements, layers).
* @event module:svgcanvas.SvgCanvas#event:changed
* @type {Element[]}
*/
/**
* Array of selected elements.
* @event module:svgcanvas.SvgCanvas#event:selected
* @type {Element[]}
*/
/**
* Array of selected elements.
* @event module:svgcanvas.SvgCanvas#event:transition
* @type {Element[]}
*/
/**
* The Element is always `SVGGElement`?
* If not `null`, will be the set current group element.
* @event module:svgcanvas.SvgCanvas#event:contextset
* @type {null|Element}
*/
/**
* @event module:svgcanvas.SvgCanvas#event:pointsAdded
* @type {PlainObject}
* @property {boolean} closedSubpath
* @property {SVGCircleElement[]} grips Grips elements
*/
/**
* @event module:svgcanvas.SvgCanvas#event:zoomed
* @type {PlainObject}
* @property {Float} x
* @property {Float} y
* @property {Float} width
* @property {Float} height
* @property {0.5|2} factor
* @see module:SVGEditor.BBoxObjectWithFactor
*/
/**
* @event module:svgcanvas.SvgCanvas#event:updateCanvas
* @type {PlainObject}
* @property {false} center
* @property {module:math.XYObject} newCtr
*/
/**
* @typedef {PlainObject} module:svgcanvas.ExtensionInitResponsePlusName
* @implements {module:svgcanvas.ExtensionInitResponse}
* @property {string} name The extension's resolved ID (whether explicit or based on file name)
*/
/**
* Generalized extension object response of
* [`init()`]{@link module:svgcanvas.ExtensionInitCallback}
* along with the name of the extension.
* @event module:svgcanvas.SvgCanvas#event:extension_added
* @type {module:svgcanvas.ExtensionInitResponsePlusName|void}
*/
/**
* @event module:svgcanvas.SvgCanvas#event:extensions_added
* @type {void}
*/
/**
* @typedef {PlainObject} module:svgcanvas.Message
* @property {any} data The data
* @property {string} origin The origin
*/
/**
* @event module:svgcanvas.SvgCanvas#event:message
* @type {module:svgcanvas.Message}
*/
/**
* SVG canvas converted to string.
* @event module:svgcanvas.SvgCanvas#event:saved
* @type {string}
*/
/**
* @event module:svgcanvas.SvgCanvas#event:setnonce
* @type {!(string|Integer)}
*/
/**
* @event module:svgcanvas.SvgCanvas#event:unsetnonce
* @type {void}
*/
/**
* @event module:svgcanvas.SvgCanvas#event:zoomDone
* @type {void}
*/
/**
* @event module:svgcanvas.SvgCanvas#event:cleared
* @type {void}
*/
/**
* @event module:svgcanvas.SvgCanvas#event:exported
* @type {module:svgcanvas.ImageExportedResults}
*/
/**
* @event module:svgcanvas.SvgCanvas#event:exportedPDF
* @type {module:svgcanvas.PDFExportedResults}
*/
/**
* Creating a cover-all class until {@link https://github.com/jsdoc3/jsdoc/issues/1545} may be supported.
* `undefined` may be returned by {@link module:svgcanvas.SvgCanvas#event:extension_added} if the extension's `init` returns `undefined` It is also the type for the following events "zoomDone", "unsetnonce", "cleared", and "extensions_added".
* @event module:svgcanvas.SvgCanvas#event:GenericCanvasEvent
* @type {module:svgcanvas.SvgCanvas#event:selected|module:svgcanvas.SvgCanvas#event:changed|module:svgcanvas.SvgCanvas#event:contextset|module:svgcanvas.SvgCanvas#event:pointsAdded|module:svgcanvas.SvgCanvas#event:extension_added|module:svgcanvas.SvgCanvas#event:extensions_added|module:svgcanvas.SvgCanvas#event:message|module:svgcanvas.SvgCanvas#event:transition|module:svgcanvas.SvgCanvas#event:zoomed|module:svgcanvas.SvgCanvas#event:updateCanvas|module:svgcanvas.SvgCanvas#event:saved|module:svgcanvas.SvgCanvas#event:exported|module:svgcanvas.SvgCanvas#event:exportedPDF|module:svgcanvas.SvgCanvas#event:setnonce|module:svgcanvas.SvgCanvas#event:unsetnonce|void}
*/
/**
* The promise return, if present, resolves to `undefined`
* (`extension_added`, `exported`, `saved`).
* @typedef {Promise<void>|void} module:svgcanvas.EventHandlerReturn
*/
/**
* @callback module:svgcanvas.EventHandler
* @param {external:Window} win
* @param {module:svgcanvas.SvgCanvas#event:GenericCanvasEvent} arg
* @listens module:svgcanvas.SvgCanvas#event:GenericCanvasEvent
* @returns {module:svgcanvas.EventHandlerReturn}
*/
/**
* Attaches a callback function to an event.
* @function module:svgcanvas.SvgCanvas#bind
* @param {"changed"|"contextset"|"selected"|"pointsAdded"|"extension_added"|"extensions_added"|"message"|"transition"|"zoomed"|"updateCanvas"|"zoomDone"|"saved"|"exported"|"exportedPDF"|"setnonce"|"unsetnonce"|"cleared"} ev - String indicating the name of the event
* @param {module:svgcanvas.EventHandler} f - The callback function to bind to the event
* @returns {module:svgcanvas.EventHandler} The previous event
*/
canvas.bind = function (ev, f) {
var old = events[ev];
events[ev] = f;
return old;
};
/**
* Runs the SVG Document through the sanitizer and then updates its paths.
* @function module:svgcanvas.SvgCanvas#prepareSvg
* @param {XMLDocument} newDoc - The SVG DOM document
* @returns {void}
*/
this.prepareSvg = function (newDoc) {
this.sanitizeSvg(newDoc.documentElement); // convert paths into absolute commands
var paths = _toConsumableArray(newDoc.getElementsByTagNameNS(NS.SVG, 'path'));
paths.forEach(function (path) {
path.setAttribute('d', pathActions$1.convertPath(path));
pathActions$1.fixEnd(path);
});
};
/**
* Hack for Firefox bugs where text element features aren't updated or get
* messed up. See issue 136 and issue 137.
* This function clones the element and re-selects it.
* @function module:svgcanvas~ffClone
* @todo Test for this bug on load and add it to "support" object instead of
* browser sniffing
* @param {Element} elem - The (text) DOM element to clone
* @returns {Element} Cloned element
*/
var ffClone = function ffClone(elem) {
if (!isGecko()) {
return elem;
}
var clone = elem.cloneNode(true);
elem.before(clone);
elem.remove();
selectorManager.releaseSelector(elem);
selectedElements[0] = clone;
selectorManager.requestSelector(clone).showGrips(true);
return clone;
}; // `this.each` is deprecated, if any extension used this it can be recreated by doing this:
// * @example $(canvas.getRootElem()).children().each(...)
// * @function module:svgcanvas.SvgCanvas#each
// this.each = function (cb) {
// $(svgroot).children().each(cb);
// };
/**
* Removes any old rotations if present, prepends a new rotation at the
* transformed center.
* @function module:svgcanvas.SvgCanvas#setRotationAngle
* @param {string|Float} val - The new rotation angle in degrees
* @param {boolean} preventUndo - Indicates whether the action should be undoable or not
* @fires module:svgcanvas.SvgCanvas#event:changed
* @returns {void}
*/
this.setRotationAngle = function (val, preventUndo) {
// ensure val is the proper type
val = parseFloat(val);
var elem = selectedElements[0];
var oldTransform = elem.getAttribute('transform');
var bbox = getBBox(elem);
var cx = bbox.x + bbox.width / 2,
cy = bbox.y + bbox.height / 2;
var tlist = getTransformList(elem); // only remove the real rotational transform if present (i.e. at index=0)
if (tlist.numberOfItems > 0) {
var xform = tlist.getItem(0);
if (xform.type === 4) {
tlist.removeItem(0);
}
} // find Rnc and insert it
if (val !== 0) {
var center = transformPoint(cx, cy, transformListToTransform(tlist).matrix);
var Rnc = svgroot.createSVGTransform();
Rnc.setRotate(val, center.x, center.y);
if (tlist.numberOfItems) {
tlist.insertItemBefore(Rnc, 0);
} else {
tlist.appendItem(Rnc);
}
} else if (tlist.numberOfItems === 0) {
elem.removeAttribute('transform');
}
if (!preventUndo) {
// we need to undo it, then redo it so it can be undo-able! :)
// TODO: figure out how to make changes to transform list undo-able cross-browser?
var newTransform = elem.getAttribute('transform');
elem.setAttribute('transform', oldTransform);
changeSelectedAttribute('transform', newTransform, selectedElements);
call('changed', selectedElements);
} // const pointGripContainer = getElem('pathpointgrip_container');
// if (elem.nodeName === 'path' && pointGripContainer) {
// pathActions.setPointContainerTransform(elem.getAttribute('transform'));
// }
var selector = selectorManager.requestSelector(selectedElements[0]);
selector.resize();
Selector.updateGripCursors(val);
};
/**
* Runs `recalculateDimensions` on the selected elements,
* adding the changes to a single batch command.
* @function module:svgcanvas.SvgCanvas#recalculateAllSelectedDimensions
* @fires module:svgcanvas.SvgCanvas#event:changed
* @returns {void}
*/
var recalculateAllSelectedDimensions = this.recalculateAllSelectedDimensions = function () {
var text = currentResizeMode === 'none' ? 'position' : 'size';
var batchCmd = new BatchCommand$1(text);
var i = selectedElements.length;
while (i--) {
var elem = selectedElements[i]; // if (getRotationAngle(elem) && !hasMatrixTransform(getTransformList(elem))) { continue; }
var cmd = recalculateDimensions(elem);
if (cmd) {
batchCmd.addSubCommand(cmd);
}
}
if (!batchCmd.isEmpty()) {
addCommandToHistory(batchCmd);
call('changed', selectedElements);
}
};
/**
* Debug tool to easily see the current matrix in the browser's console.
* @function module:svgcanvas~logMatrix
* @param {SVGMatrix} m The matrix
* @returns {void}
*/
var logMatrix = function logMatrix(m) {
console.log([m.a, m.b, m.c, m.d, m.e, m.f]); // eslint-disable-line no-console
}; // Root Current Transformation Matrix in user units
var rootSctm = null;
/**
* Group: Selection.
*/
// TODO: do we need to worry about selectedBBoxes here?
/**
* Selects only the given elements, shortcut for `clearSelection(); addToSelection()`.
* @function module:svgcanvas.SvgCanvas#selectOnly
* @param {Element[]} elems - an array of DOM elements to be selected
* @param {boolean} showGrips - Indicates whether the resize grips should be shown
* @returns {void}
*/
var selectOnly = this.selectOnly = function (elems, showGrips) {
clearSelection(true);
addToSelection(elems, showGrips);
}; // TODO: could use slice here to make this faster?
// TODO: should the 'selected' handler
/**
* Removes elements from the selection.
* @function module:svgcanvas.SvgCanvas#removeFromSelection
* @param {Element[]} elemsToRemove - An array of elements to remove from selection
* @returns {void}
*/
/* const removeFromSelection = */
this.removeFromSelection = function (elemsToRemove) {
if (isNullish(selectedElements[0])) {
return;
}
if (!elemsToRemove.length) {
return;
} // find every element and remove it from our array copy
var newSelectedItems = [],
len = selectedElements.length;
for (var i = 0; i < len; ++i) {
var elem = selectedElements[i];
if (elem) {
// keep the item
if (!elemsToRemove.includes(elem)) {
newSelectedItems.push(elem);
} else {
// remove the item and its selector
selectorManager.releaseSelector(elem);
}
}
} // the copy becomes the master now
selectedElements = newSelectedItems;
};
/**
* Clears the selection, then adds all elements in the current layer to the selection.
* @function module:svgcanvas.SvgCanvas#selectAllInCurrentLayer
* @returns {void}
*/
this.selectAllInCurrentLayer = function () {
var currentLayer = getCurrentDrawing().getCurrentLayer();
if (currentLayer) {
currentMode = 'select';
selectOnly($$8(currentGroup || currentLayer).children());
}
};
var drawnPath = null; // Mouse events
(function () {
var freehand = {
minx: null,
miny: null,
maxx: null,
maxy: null
};
var THRESHOLD_DIST = 0.8,
STEP_COUNT = 10;
var dAttr = null,
startX = null,
startY = null,
rStartX = null,
rStartY = null,
initBbox = {},
sumDistance = 0,
controllPoint2 = {
x: 0,
y: 0
},
controllPoint1 = {
x: 0,
y: 0
},
start = {
x: 0,
y: 0
},
end = {
x: 0,
y: 0
},
bSpline = {
x: 0,
y: 0
},
nextPos = {
x: 0,
y: 0
},
parameter,
nextParameter;
var getBsplinePoint = function getBsplinePoint(t) {
var spline = {
x: 0,
y: 0
},
p0 = controllPoint2,
p1 = controllPoint1,
p2 = start,
p3 = end,
S = 1.0 / 6.0,
t2 = t * t,
t3 = t2 * t;
var m = [[-1, 3, -3, 1], [3, -6, 3, 0], [-3, 0, 3, 0], [1, 4, 1, 0]];
spline.x = S * ((p0.x * m[0][0] + p1.x * m[0][1] + p2.x * m[0][2] + p3.x * m[0][3]) * t3 + (p0.x * m[1][0] + p1.x * m[1][1] + p2.x * m[1][2] + p3.x * m[1][3]) * t2 + (p0.x * m[2][0] + p1.x * m[2][1] + p2.x * m[2][2] + p3.x * m[2][3]) * t + (p0.x * m[3][0] + p1.x * m[3][1] + p2.x * m[3][2] + p3.x * m[3][3]));
spline.y = S * ((p0.y * m[0][0] + p1.y * m[0][1] + p2.y * m[0][2] + p3.y * m[0][3]) * t3 + (p0.y * m[1][0] + p1.y * m[1][1] + p2.y * m[1][2] + p3.y * m[1][3]) * t2 + (p0.y * m[2][0] + p1.y * m[2][1] + p2.y * m[2][2] + p3.y * m[2][3]) * t + (p0.y * m[3][0] + p1.y * m[3][1] + p2.y * m[3][2] + p3.y * m[3][3]));
return {
x: spline.x,
y: spline.y
};
};
/**
* Follows these conditions:
* - When we are in a create mode, the element is added to the canvas but the
* action is not recorded until mousing up.
* - When we are in select mode, select the element, remember the position
* and do nothing else.
* @param {MouseEvent} evt
* @fires module:svgcanvas.SvgCanvas#event:ext_mouseDown
* @returns {void}
*/
var mouseDown = function mouseDown(evt) {
if (canvas.spaceKey || evt.button === 1) {
return;
}
var rightClick = evt.button === 2;
if (evt.altKey) {
// duplicate when dragging
canvas.cloneSelectedElements(0, 0);
}
rootSctm = $$8('#svgcontent g')[0].getScreenCTM().inverse();
var pt = transformPoint(evt.pageX, evt.pageY, rootSctm),
mouseX = pt.x * currentZoom,
mouseY = pt.y * currentZoom;
evt.preventDefault();
if (rightClick) {
currentMode = 'select';
lastClickPoint = pt;
} // This would seem to be unnecessary...
// if (!['select', 'resize'].includes(currentMode)) {
// setGradient();
// }
var x = mouseX / currentZoom,
y = mouseY / currentZoom;
var mouseTarget = getMouseTarget(evt);
if (mouseTarget.tagName === 'a' && mouseTarget.childNodes.length === 1) {
mouseTarget = mouseTarget.firstChild;
} // realX/y ignores grid-snap value
var realX = x;
rStartX = startX = x;
var realY = y;
rStartY = startY = y;
if (curConfig.gridSnapping) {
x = snapToGrid(x);
y = snapToGrid(y);
startX = snapToGrid(startX);
startY = snapToGrid(startY);
} // if it is a selector grip, then it must be a single element selected,
// set the mouseTarget to that and update the mode to rotate/resize
if (mouseTarget === selectorManager.selectorParentGroup && !isNullish(selectedElements[0])) {
var grip = evt.target;
var griptype = elData(grip, 'type'); // rotating
if (griptype === 'rotate') {
currentMode = 'rotate'; // resizing
} else if (griptype === 'resize') {
currentMode = 'resize';
currentResizeMode = elData(grip, 'dir');
}
mouseTarget = selectedElements[0];
}
startTransform = mouseTarget.getAttribute('transform');
var tlist = getTransformList(mouseTarget);
switch (currentMode) {
case 'select':
started = true;
currentResizeMode = 'none';
if (rightClick) {
started = false;
}
if (mouseTarget !== svgroot) {
// if this element is not yet selected, clear selection and select it
if (!selectedElements.includes(mouseTarget)) {
// only clear selection if shift is not pressed (otherwise, add
// element to selection)
if (!evt.shiftKey) {
// No need to do the call here as it will be done on addToSelection
clearSelection(true);
}
addToSelection([mouseTarget]);
justSelected = mouseTarget;
pathActions$1.clear();
} // else if it's a path, go into pathedit mode in mouseup
if (!rightClick) {
// insert a dummy transform so if the element(s) are moved it will have
// a transform to use for its translate
var _iteratorNormalCompletion = true;
var _didIteratorError = false;
var _iteratorError = undefined;
try {
for (var _iterator = selectedElements[Symbol.iterator](), _step; !(_iteratorNormalCompletion = (_step = _iterator.next()).done); _iteratorNormalCompletion = true) {
var selectedElement = _step.value;
if (isNullish(selectedElement)) {
continue;
}
var slist = getTransformList(selectedElement);
if (slist.numberOfItems) {
slist.insertItemBefore(svgroot.createSVGTransform(), 0);
} else {
slist.appendItem(svgroot.createSVGTransform());
}
}
} catch (err) {
_didIteratorError = true;
_iteratorError = err;
} finally {
try {
if (!_iteratorNormalCompletion && _iterator["return"] != null) {
_iterator["return"]();
}
} finally {
if (_didIteratorError) {
throw _iteratorError;
}
}
}
}
} else if (!rightClick) {
clearSelection();
currentMode = 'multiselect';
if (isNullish(rubberBox)) {
rubberBox = selectorManager.getRubberBandBox();
}
rStartX *= currentZoom;
rStartY *= currentZoom; // console.log('p',[evt.pageX, evt.pageY]);
// console.log('c',[evt.clientX, evt.clientY]);
// console.log('o',[evt.offsetX, evt.offsetY]);
// console.log('s',[startX, startY]);
assignAttributes(rubberBox, {
x: rStartX,
y: rStartY,
width: 0,
height: 0,
display: 'inline'
});
}
break;
case 'zoom':
started = true;
if (isNullish(rubberBox)) {
rubberBox = selectorManager.getRubberBandBox();
}
assignAttributes(rubberBox, {
x: realX * currentZoom,
y: realX * currentZoom,
width: 0,
height: 0,
display: 'inline'
});
break;
case 'resize':
{
started = true;
startX = x;
startY = y; // Getting the BBox from the selection box, since we know we
// want to orient around it
initBbox = getBBox($$8('#selectedBox0')[0]);
var bb = {};
$$8.each(initBbox, function (key, val) {
bb[key] = val / currentZoom;
});
initBbox = bb; // append three dummy transforms to the tlist so that
// we can translate,scale,translate in mousemove
var pos = getRotationAngle(mouseTarget) ? 1 : 0;
if (hasMatrixTransform(tlist)) {
tlist.insertItemBefore(svgroot.createSVGTransform(), pos);
tlist.insertItemBefore(svgroot.createSVGTransform(), pos);
tlist.insertItemBefore(svgroot.createSVGTransform(), pos);
} else {
tlist.appendItem(svgroot.createSVGTransform());
tlist.appendItem(svgroot.createSVGTransform());
tlist.appendItem(svgroot.createSVGTransform());
if (supportsNonScalingStroke()) {
// Handle crash for newer Chrome and Safari 6 (Mobile and Desktop):
// https://code.google.com/p/svg-edit/issues/detail?id=904
// Chromium issue: https://code.google.com/p/chromium/issues/detail?id=114625
// TODO: Remove this workaround once vendor fixes the issue
var iswebkit = isWebkit();
var delayedStroke;
if (iswebkit) {
delayedStroke = function delayedStroke(ele) {
var stroke_ = ele.getAttribute('stroke');
ele.removeAttribute('stroke'); // Re-apply stroke after delay. Anything higher than 1 seems to cause flicker
if (stroke_ !== null) setTimeout(function () {
ele.setAttribute('stroke', stroke_);
}, 0);
};
}
mouseTarget.style.vectorEffect = 'non-scaling-stroke';
if (iswebkit) {
delayedStroke(mouseTarget);
}
var all = mouseTarget.getElementsByTagName('*'),
len = all.length;
for (var i = 0; i < len; i++) {
if (!all[i].style) {
// mathML
continue;
}
all[i].style.vectorEffect = 'non-scaling-stroke';
if (iswebkit) {
delayedStroke(all[i]);
}
}
}
}
break;
}
case 'fhellipse':
case 'fhrect':
case 'fhpath':
start.x = realX;
start.y = realY;
2020-01-08 18:27:56 +00:00
controllPoint1 = {
x: 0,
y: 0
};
controllPoint2 = {
x: 0,
y: 0
};
started = true;
dAttr = realX + ',' + realY + ' '; // Commented out as doing nothing now:
// strokeW = parseFloat(curShape.stroke_width) === 0 ? 1 : curShape.stroke_width;
addSVGElementFromJson({
element: 'polyline',
curStyles: true,
attr: {
points: dAttr,
id: getNextId(),
fill: 'none',
opacity: curShape.opacity / 2,
'stroke-linecap': 'round',
style: 'pointer-events:none'
}
});
freehand.minx = realX;
freehand.maxx = realX;
freehand.miny = realY;
freehand.maxy = realY;
break;
case 'image':
{
started = true;
var newImage = addSVGElementFromJson({
element: 'image',
attr: {
x: x,
y: y,
width: 0,
height: 0,
id: getNextId(),
opacity: curShape.opacity / 2,
style: 'pointer-events:inherit'
}
});
setHref(newImage, lastGoodImgUrl);
preventClickDefault(newImage);
break;
}
case 'square': // TODO: once we create the rect, we lose information that this was a square
// (for resizing purposes this could be important)
// Fallthrough
case 'rect':
started = true;
startX = x;
startY = y;
addSVGElementFromJson({
element: 'rect',
curStyles: true,
attr: {
x: x,
y: y,
width: 0,
height: 0,
id: getNextId(),
opacity: curShape.opacity / 2
}
});
break;
case 'line':
{
started = true;
var strokeW = Number(curShape.stroke_width) === 0 ? 1 : curShape.stroke_width;
addSVGElementFromJson({
element: 'line',
curStyles: true,
attr: {
x1: x,
y1: y,
x2: x,
y2: y,
id: getNextId(),
stroke: curShape.stroke,
'stroke-width': strokeW,
'stroke-dasharray': curShape.stroke_dasharray,
'stroke-linejoin': curShape.stroke_linejoin,
'stroke-linecap': curShape.stroke_linecap,
'stroke-opacity': curShape.stroke_opacity,
fill: 'none',
opacity: curShape.opacity / 2,
style: 'pointer-events:none'
}
});
break;
}
case 'circle':
started = true;
addSVGElementFromJson({
element: 'circle',
curStyles: true,
attr: {
cx: x,
cy: y,
r: 0,
id: getNextId(),
opacity: curShape.opacity / 2
}
});
break;
case 'ellipse':
started = true;
addSVGElementFromJson({
element: 'ellipse',
curStyles: true,
attr: {
cx: x,
cy: y,
rx: 0,
ry: 0,
id: getNextId(),
opacity: curShape.opacity / 2
}
});
break;
case 'text':
started = true;
/* const newText = */
addSVGElementFromJson({
element: 'text',
curStyles: true,
attr: {
x: x,
y: y,
id: getNextId(),
fill: curText.fill,
'stroke-width': curText.stroke_width,
'font-size': curText.font_size,
'font-family': curText.font_family,
'text-anchor': 'middle',
'xml:space': 'preserve',
opacity: curShape.opacity
}
}); // newText.textContent = 'text';
break;
case 'path': // Fall through
case 'pathedit':
startX *= currentZoom;
startY *= currentZoom;
pathActions$1.mouseDown(evt, mouseTarget, startX, startY);
started = true;
break;
case 'textedit':
startX *= currentZoom;
startY *= currentZoom;
textActions.mouseDown(evt, mouseTarget, startX, startY);
started = true;
break;
case 'rotate':
started = true; // we are starting an undoable change (a drag-rotation)
canvas.undoMgr.beginUndoableChange('transform', selectedElements);
break;
}
/**
* The main (left) mouse button is held down on the canvas area.
* @event module:svgcanvas.SvgCanvas#event:ext_mouseDown
* @type {PlainObject}
* @property {MouseEvent} event The event object
* @property {Float} start_x x coordinate on canvas
* @property {Float} start_y y coordinate on canvas
* @property {Element[]} selectedElements An array of the selected Elements
*/
var extResult = runExtensions('mouseDown',
/** @type {module:svgcanvas.SvgCanvas#event:ext_mouseDown} */
{
event: evt,
start_x: startX,
start_y: startY,
selectedElements: selectedElements
}, true);
$$8.each(extResult, function (i, r) {
if (r && r.started) {
started = true;
}
});
}; // in this function we do not record any state changes yet (but we do update
// any elements that are still being created, moved or resized on the canvas)
/**
*
* @param {MouseEvent} evt
* @fires module:svgcanvas.SvgCanvas#event:transition
* @fires module:svgcanvas.SvgCanvas#event:ext_mouseMove
* @returns {void}
*/
var mouseMove = function mouseMove(evt) {
if (!started) {
return;
}
if (evt.button === 1 || canvas.spaceKey) {
return;
}
var i,
xya,
c,
cx,
cy,
dx,
dy,
len,
angle,
box,
selected = selectedElements[0];
var pt = transformPoint(evt.pageX, evt.pageY, rootSctm),
mouseX = pt.x * currentZoom,
mouseY = pt.y * currentZoom,
shape = getElem(getId());
var realX = mouseX / currentZoom;
var x = realX;
var realY = mouseY / currentZoom;
var y = realY;
if (curConfig.gridSnapping) {
x = snapToGrid(x);
y = snapToGrid(y);
}
evt.preventDefault();
var tlist;
switch (currentMode) {
case 'select':
{
// we temporarily use a translate on the element(s) being dragged
// this transform is removed upon mousing up and the element is
// relocated to the new location
if (selectedElements[0] !== null) {
dx = x - startX;
dy = y - startY;
if (curConfig.gridSnapping) {
dx = snapToGrid(dx);
dy = snapToGrid(dy);
}
/*
// Commenting out as currently has no effect
if (evt.shiftKey) {
xya = snapToAngle(startX, startY, x, y);
({x, y} = xya);
}
*/
if (dx !== 0 || dy !== 0) {
len = selectedElements.length;
for (i = 0; i < len; ++i) {
selected = selectedElements[i];
if (isNullish(selected)) {
break;
} // if (i === 0) {
// const box = utilsGetBBox(selected);
// selectedBBoxes[i].x = box.x + dx;
// selectedBBoxes[i].y = box.y + dy;
// }
// update the dummy transform in our transform list
// to be a translate
var xform = svgroot.createSVGTransform();
tlist = getTransformList(selected); // Note that if Webkit and there's no ID for this
// element, the dummy transform may have gotten lost.
// This results in unexpected behaviour
xform.setTranslate(dx, dy);
if (tlist.numberOfItems) {
tlist.replaceItem(xform, 0);
} else {
tlist.appendItem(xform);
} // update our internal bbox that we're tracking while dragging
selectorManager.requestSelector(selected).resize();
}
call('transition', selectedElements);
}
}
break;
}
case 'multiselect':
{
realX *= currentZoom;
realY *= currentZoom;
assignAttributes(rubberBox, {
x: Math.min(rStartX, realX),
y: Math.min(rStartY, realY),
width: Math.abs(realX - rStartX),
height: Math.abs(realY - rStartY)
}); // for each selected:
// - if newList contains selected, do nothing
// - if newList doesn't contain selected, remove it from selected
// - for any newList that was not in selectedElements, add it to selected
var elemsToRemove = selectedElements.slice(),
elemsToAdd = [],
newList = getIntersectionList(); // For every element in the intersection, add if not present in selectedElements.
len = newList.length;
for (i = 0; i < len; ++i) {
var intElem = newList[i]; // Found an element that was not selected before, so we should add it.
if (!selectedElements.includes(intElem)) {
elemsToAdd.push(intElem);
} // Found an element that was already selected, so we shouldn't remove it.
var foundInd = elemsToRemove.indexOf(intElem);
if (foundInd !== -1) {
elemsToRemove.splice(foundInd, 1);
}
}
if (elemsToRemove.length > 0) {
canvas.removeFromSelection(elemsToRemove);
}
if (elemsToAdd.length > 0) {
canvas.addToSelection(elemsToAdd);
}
break;
}
case 'resize':
{
// we track the resize bounding box and translate/scale the selected element
// while the mouse is down, when mouse goes up, we use this to recalculate
// the shape's coordinates
tlist = getTransformList(selected);
var hasMatrix = hasMatrixTransform(tlist);
box = hasMatrix ? initBbox : getBBox(selected);
var left = box.x,
top = box.y,
_box = box,
width = _box.width,
height = _box.height;
dx = x - startX;
dy = y - startY;
if (curConfig.gridSnapping) {
dx = snapToGrid(dx);
dy = snapToGrid(dy);
height = snapToGrid(height);
width = snapToGrid(width);
} // if rotated, adjust the dx,dy values
angle = getRotationAngle(selected);
if (angle) {
var r = Math.sqrt(dx * dx + dy * dy),
theta = Math.atan2(dy, dx) - angle * Math.PI / 180.0;
dx = r * Math.cos(theta);
dy = r * Math.sin(theta);
} // if not stretching in y direction, set dy to 0
// if not stretching in x direction, set dx to 0
if (!currentResizeMode.includes('n') && !currentResizeMode.includes('s')) {
dy = 0;
}
if (!currentResizeMode.includes('e') && !currentResizeMode.includes('w')) {
dx = 0;
}
var // ts = null,
tx = 0,
ty = 0,
sy = height ? (height + dy) / height : 1,
sx = width ? (width + dx) / width : 1; // if we are dragging on the north side, then adjust the scale factor and ty
if (currentResizeMode.includes('n')) {
sy = height ? (height - dy) / height : 1;
ty = height;
} // if we dragging on the east side, then adjust the scale factor and tx
if (currentResizeMode.includes('w')) {
sx = width ? (width - dx) / width : 1;
tx = width;
} // update the transform list with translate,scale,translate
var translateOrigin = svgroot.createSVGTransform(),
scale = svgroot.createSVGTransform(),
translateBack = svgroot.createSVGTransform();
if (curConfig.gridSnapping) {
left = snapToGrid(left);
tx = snapToGrid(tx);
top = snapToGrid(top);
ty = snapToGrid(ty);
}
translateOrigin.setTranslate(-(left + tx), -(top + ty));
if (evt.shiftKey) {
if (sx === 1) {
sx = sy;
} else {
sy = sx;
}
}
scale.setScale(sx, sy);
translateBack.setTranslate(left + tx, top + ty);
if (hasMatrix) {
var diff = angle ? 1 : 0;
tlist.replaceItem(translateOrigin, 2 + diff);
tlist.replaceItem(scale, 1 + diff);
tlist.replaceItem(translateBack, Number(diff));
} else {
var N = tlist.numberOfItems;
tlist.replaceItem(translateBack, N - 3);
tlist.replaceItem(scale, N - 2);
tlist.replaceItem(translateOrigin, N - 1);
}
selectorManager.requestSelector(selected).resize();
call('transition', selectedElements);
break;
}
case 'zoom':
{
realX *= currentZoom;
realY *= currentZoom;
assignAttributes(rubberBox, {
x: Math.min(rStartX * currentZoom, realX),
y: Math.min(rStartY * currentZoom, realY),
width: Math.abs(realX - rStartX * currentZoom),
height: Math.abs(realY - rStartY * currentZoom)
});
break;
}
case 'text':
{
assignAttributes(shape, {
x: x,
y: y
});
break;
}
case 'line':
{
if (curConfig.gridSnapping) {
x = snapToGrid(x);
y = snapToGrid(y);
}
var x2 = x;
var y2 = y;
if (evt.shiftKey) {
xya = snapToAngle(startX, startY, x2, y2);
x2 = xya.x;
y2 = xya.y;
}
shape.setAttribute('x2', x2);
shape.setAttribute('y2', y2);
break;
}
case 'foreignObject': // fall through
case 'square': // fall through
case 'rect': // fall through
case 'image':
{
var square = currentMode === 'square' || evt.shiftKey;
var w = Math.abs(x - startX),
h = Math.abs(y - startY);
var newX, newY;
if (square) {
w = h = Math.max(w, h);
newX = startX < x ? startX : startX - w;
newY = startY < y ? startY : startY - h;
} else {
newX = Math.min(startX, x);
newY = Math.min(startY, y);
}
if (curConfig.gridSnapping) {
w = snapToGrid(w);
h = snapToGrid(h);
newX = snapToGrid(newX);
newY = snapToGrid(newY);
}
assignAttributes(shape, {
width: w,
height: h,
x: newX,
y: newY
});
break;
}
case 'circle':
{
c = $$8(shape).attr(['cx', 'cy']);
var _c = c;
cx = _c.cx;
cy = _c.cy;
var rad = Math.sqrt((x - cx) * (x - cx) + (y - cy) * (y - cy));
if (curConfig.gridSnapping) {
rad = snapToGrid(rad);
}
shape.setAttribute('r', rad);
break;
}
case 'ellipse':
{
c = $$8(shape).attr(['cx', 'cy']);
var _c2 = c;
cx = _c2.cx;
cy = _c2.cy;
if (curConfig.gridSnapping) {
x = snapToGrid(x);
cx = snapToGrid(cx);
y = snapToGrid(y);
cy = snapToGrid(cy);
}
shape.setAttribute('rx', Math.abs(x - cx));
var ry = Math.abs(evt.shiftKey ? x - cx : y - cy);
shape.setAttribute('ry', ry);
break;
}
case 'fhellipse':
case 'fhrect':
{
freehand.minx = Math.min(realX, freehand.minx);
freehand.maxx = Math.max(realX, freehand.maxx);
freehand.miny = Math.min(realY, freehand.miny);
freehand.maxy = Math.max(realY, freehand.maxy);
}
// Fallthrough
case 'fhpath':
{
// dAttr += + realX + ',' + realY + ' ';
// shape.setAttribute('points', dAttr);
end.x = realX;
end.y = realY;
if (controllPoint2.x && controllPoint2.y) {
for (i = 0; i < STEP_COUNT - 1; i++) {
parameter = i / STEP_COUNT;
nextParameter = (i + 1) / STEP_COUNT;
bSpline = getBsplinePoint(nextParameter);
nextPos = bSpline;
bSpline = getBsplinePoint(parameter);
sumDistance += Math.sqrt((nextPos.x - bSpline.x) * (nextPos.x - bSpline.x) + (nextPos.y - bSpline.y) * (nextPos.y - bSpline.y));
if (sumDistance > THRESHOLD_DIST) {
sumDistance -= THRESHOLD_DIST; // Faster than completely re-writing the points attribute.
var point = svgcontent.createSVGPoint();
point.x = bSpline.x;
point.y = bSpline.y;
shape.points.appendItem(point);
}
}
}
controllPoint2 = {
x: controllPoint1.x,
y: controllPoint1.y
};
controllPoint1 = {
x: start.x,
y: start.y
};
start = {
x: end.x,
y: end.y
};
break; // update path stretch line coordinates
}
case 'path': // fall through
case 'pathedit':
{
x *= currentZoom;
y *= currentZoom;
if (curConfig.gridSnapping) {
x = snapToGrid(x);
y = snapToGrid(y);
startX = snapToGrid(startX);
startY = snapToGrid(startY);
}
if (evt.shiftKey) {
var path$1 = path;
var x1, y1;
if (path$1) {
x1 = path$1.dragging ? path$1.dragging[0] : startX;
y1 = path$1.dragging ? path$1.dragging[1] : startY;
} else {
x1 = startX;
y1 = startY;
}
xya = snapToAngle(x1, y1, x, y);
var _xya = xya;
x = _xya.x;
y = _xya.y;
}
if (rubberBox && rubberBox.getAttribute('display') !== 'none') {
realX *= currentZoom;
realY *= currentZoom;
assignAttributes(rubberBox, {
x: Math.min(rStartX * currentZoom, realX),
y: Math.min(rStartY * currentZoom, realY),
width: Math.abs(realX - rStartX * currentZoom),
height: Math.abs(realY - rStartY * currentZoom)
});
}
pathActions$1.mouseMove(x, y);
break;
}
case 'textedit':
{
x *= currentZoom;
y *= currentZoom; // if (rubberBox && rubberBox.getAttribute('display') !== 'none') {
// assignAttributes(rubberBox, {
// x: Math.min(startX, x),
// y: Math.min(startY, y),
// width: Math.abs(x - startX),
// height: Math.abs(y - startY)
// }, 100);
// }
textActions.mouseMove(mouseX, mouseY);
break;
}
case 'rotate':
{
box = getBBox(selected);
cx = box.x + box.width / 2;
cy = box.y + box.height / 2;
var m = getMatrix(selected),
center = transformPoint(cx, cy, m);
cx = center.x;
cy = center.y;
angle = (Math.atan2(cy - y, cx - x) * (180 / Math.PI) - 90) % 360;
if (curConfig.gridSnapping) {
angle = snapToGrid(angle);
}
if (evt.shiftKey) {
// restrict rotations to nice angles (WRS)
var snap = 45;
angle = Math.round(angle / snap) * snap;
}
canvas.setRotationAngle(angle < -180 ? 360 + angle : angle, true);
call('transition', selectedElements);
break;
}
}
/**
* The mouse has moved on the canvas area.
* @event module:svgcanvas.SvgCanvas#event:ext_mouseMove
* @type {PlainObject}
* @property {MouseEvent} event The event object
* @property {Float} mouse_x x coordinate on canvas
* @property {Float} mouse_y y coordinate on canvas
* @property {Element} selected Refers to the first selected element
*/
runExtensions('mouseMove',
/** @type {module:svgcanvas.SvgCanvas#event:ext_mouseMove} */
{
event: evt,
mouse_x: mouseX,
mouse_y: mouseY,
selected: selected
});
}; // mouseMove()
// - in create mode, the element's opacity is set properly, we create an InsertElementCommand
// and store it on the Undo stack
// - in move/resize mode, the element's attributes which were affected by the move/resize are
// identified, a ChangeElementCommand is created and stored on the stack for those attrs
// this is done in when we recalculate the selected dimensions()
/**
*
* @param {MouseEvent} evt
* @fires module:svgcanvas.SvgCanvas#event:zoomed
* @fires module:svgcanvas.SvgCanvas#event:changed
* @fires module:svgcanvas.SvgCanvas#event:ext_mouseUp
* @returns {void}
*/
var mouseUp = function mouseUp(evt) {
if (evt.button === 2) {
return;
}
var tempJustSelected = justSelected;
justSelected = null;
if (!started) {
return;
}
var pt = transformPoint(evt.pageX, evt.pageY, rootSctm),
mouseX = pt.x * currentZoom,
mouseY = pt.y * currentZoom,
x = mouseX / currentZoom,
y = mouseY / currentZoom;
var element = getElem(getId());
var keep = false;
var realX = x;
var realY = y; // TODO: Make true when in multi-unit mode
started = false;
var attrs, t;
switch (currentMode) {
// intentionally fall-through to select here
case 'resize':
case 'multiselect':
if (!isNullish(rubberBox)) {
rubberBox.setAttribute('display', 'none');
curBBoxes = [];
}
currentMode = 'select';
// Fallthrough
case 'select':
if (!isNullish(selectedElements[0])) {
// if we only have one selected element
if (isNullish(selectedElements[1])) {
// set our current stroke/fill properties to the element's
var selected = selectedElements[0];
switch (selected.tagName) {
case 'g':
case 'use':
case 'image':
case 'foreignObject':
break;
default:
curProperties.fill = selected.getAttribute('fill');
curProperties.fill_opacity = selected.getAttribute('fill-opacity');
curProperties.stroke = selected.getAttribute('stroke');
curProperties.stroke_opacity = selected.getAttribute('stroke-opacity');
curProperties.stroke_width = selected.getAttribute('stroke-width');
curProperties.stroke_dasharray = selected.getAttribute('stroke-dasharray');
curProperties.stroke_linejoin = selected.getAttribute('stroke-linejoin');
curProperties.stroke_linecap = selected.getAttribute('stroke-linecap');
}
if (selected.tagName === 'text') {
curText.font_size = selected.getAttribute('font-size');
curText.font_family = selected.getAttribute('font-family');
}
selectorManager.requestSelector(selected).showGrips(true); // This shouldn't be necessary as it was done on mouseDown...
// call('selected', [selected]);
} // always recalculate dimensions to strip off stray identity transforms
recalculateAllSelectedDimensions(); // if it was being dragged/resized
if (realX !== rStartX || realY !== rStartY) {
var len = selectedElements.length;
for (var i = 0; i < len; ++i) {
if (isNullish(selectedElements[i])) {
break;
}
if (!selectedElements[i].firstChild) {
// Not needed for groups (incorrectly resizes elems), possibly not needed at all?
selectorManager.requestSelector(selectedElements[i]).resize();
}
} // no change in position/size, so maybe we should move to pathedit
} else {
t = evt.target;
if (selectedElements[0].nodeName === 'path' && isNullish(selectedElements[1])) {
pathActions$1.select(selectedElements[0]); // if it was a path
// else, if it was selected and this is a shift-click, remove it from selection
} else if (evt.shiftKey) {
if (tempJustSelected !== t) {
canvas.removeFromSelection([t]);
}
}
} // no change in mouse position
// Remove non-scaling stroke
if (supportsNonScalingStroke()) {
var elem = selectedElements[0];
if (elem) {
elem.removeAttribute('style');
walkTree(elem, function (el) {
el.removeAttribute('style');
});
}
}
}
return;
case 'zoom':
{
if (!isNullish(rubberBox)) {
rubberBox.setAttribute('display', 'none');
}
var factor = evt.shiftKey ? 0.5 : 2;
call('zoomed', {
x: Math.min(rStartX, realX),
y: Math.min(rStartY, realY),
width: Math.abs(realX - rStartX),
height: Math.abs(realY - rStartY),
factor: factor
});
return;
}
case 'fhpath':
{
// Check that the path contains at least 2 points; a degenerate one-point path
// causes problems.
// Webkit ignores how we set the points attribute with commas and uses space
// to separate all coordinates, see https://bugs.webkit.org/show_bug.cgi?id=29870
sumDistance = 0;
controllPoint2 = {
x: 0,
y: 0
};
controllPoint1 = {
x: 0,
y: 0
};
start = {
x: 0,
y: 0
};
end = {
x: 0,
y: 0
};
var coords = element.getAttribute('points');
var commaIndex = coords.indexOf(',');
if (commaIndex >= 0) {
keep = coords.includes(',', commaIndex + 1);
} else {
keep = coords.includes(' ', coords.indexOf(' ') + 1);
}
if (keep) {
element = pathActions$1.smoothPolylineIntoPath(element);
}
break;
}
case 'line':
attrs = $$8(element).attr(['x1', 'x2', 'y1', 'y2']);
keep = attrs.x1 !== attrs.x2 || attrs.y1 !== attrs.y2;
break;
case 'foreignObject':
case 'square':
case 'rect':
case 'image':
attrs = $$8(element).attr(['width', 'height']); // Image should be kept regardless of size (use inherit dimensions later)
keep = attrs.width || attrs.height || currentMode === 'image';
break;
case 'circle':
keep = element.getAttribute('r') !== '0';
break;
case 'ellipse':
attrs = $$8(element).attr(['rx', 'ry']);
keep = attrs.rx || attrs.ry;
break;
case 'fhellipse':
if (freehand.maxx - freehand.minx > 0 && freehand.maxy - freehand.miny > 0) {
element = addSVGElementFromJson({
element: 'ellipse',
curStyles: true,
attr: {
cx: (freehand.minx + freehand.maxx) / 2,
cy: (freehand.miny + freehand.maxy) / 2,
rx: (freehand.maxx - freehand.minx) / 2,
ry: (freehand.maxy - freehand.miny) / 2,
id: getId()
}
});
call('changed', [element]);
keep = true;
}
break;
case 'fhrect':
if (freehand.maxx - freehand.minx > 0 && freehand.maxy - freehand.miny > 0) {
element = addSVGElementFromJson({
element: 'rect',
curStyles: true,
attr: {
x: freehand.minx,
y: freehand.miny,
width: freehand.maxx - freehand.minx,
height: freehand.maxy - freehand.miny,
id: getId()
}
});
call('changed', [element]);
keep = true;
}
break;
case 'text':
keep = true;
selectOnly([element]);
textActions.start(element);
break;
case 'path':
{
// set element to null here so that it is not removed nor finalized
element = null; // continue to be set to true so that mouseMove happens
started = true;
var res = pathActions$1.mouseUp(evt, element, mouseX, mouseY);
element = res.element;
keep = res.keep;
break;
}
case 'pathedit':
keep = true;
element = null;
pathActions$1.mouseUp(evt);
break;
case 'textedit':
keep = false;
element = null;
textActions.mouseUp(evt, mouseX, mouseY);
break;
case 'rotate':
{
keep = true;
element = null;
currentMode = 'select';
var batchCmd = canvas.undoMgr.finishUndoableChange();
if (!batchCmd.isEmpty()) {
addCommandToHistory(batchCmd);
} // perform recalculation to weed out any stray identity transforms that might get stuck
recalculateAllSelectedDimensions();
call('changed', selectedElements);
break;
}
}
/**
* The main (left) mouse button is released (anywhere).
* @event module:svgcanvas.SvgCanvas#event:ext_mouseUp
* @type {PlainObject}
* @property {MouseEvent} event The event object
* @property {Float} mouse_x x coordinate on canvas
* @property {Float} mouse_y y coordinate on canvas
*/
var extResult = runExtensions('mouseUp',
/** @type {module:svgcanvas.SvgCanvas#event:ext_mouseUp} */
{
event: evt,
mouse_x: mouseX,
mouse_y: mouseY
}, true);
$$8.each(extResult, function (i, r) {
if (r) {
keep = r.keep || keep;
element = r.element;
started = r.started || started;
}
});
if (!keep && !isNullish(element)) {
getCurrentDrawing().releaseId(getId());
element.remove();
element = null;
t = evt.target; // if this element is in a group, go up until we reach the top-level group
// just below the layer groups
// TODO: once we implement links, we also would have to check for <a> elements
while (t && t.parentNode && t.parentNode.parentNode && t.parentNode.parentNode.tagName === 'g') {
t = t.parentNode;
} // if we are not in the middle of creating a path, and we've clicked on some shape,
// then go to Select mode.
// WebKit returns <div> when the canvas is clicked, Firefox/Opera return <svg>
if ((currentMode !== 'path' || !drawnPath) && t && t.parentNode && t.parentNode.id !== 'selectorParentGroup' && t.id !== 'svgcanvas' && t.id !== 'svgroot') {
// switch into "select" mode if we've clicked on an element
canvas.setMode('select');
selectOnly([t], true);
}
} else if (!isNullish(element)) {
/**
* @name module:svgcanvas.SvgCanvas#addedNew
* @type {boolean}
*/
canvas.addedNew = true;
var aniDur = 0.2;
var cAni;
if (opacAni.beginElement && parseFloat(element.getAttribute('opacity')) !== curShape.opacity) {
cAni = $$8(opacAni).clone().attr({
to: curShape.opacity,
dur: aniDur
}).appendTo(element);
try {
// Fails in FF4 on foreignObject
cAni[0].beginElement();
} catch (e) {}
} else {
aniDur = 0;
} // Ideally this would be done on the endEvent of the animation,
// but that doesn't seem to be supported in Webkit
setTimeout(function () {
if (cAni) {
cAni.remove();
}
element.setAttribute('opacity', curShape.opacity);
element.setAttribute('style', 'pointer-events:inherit');
cleanupElement(element);
if (currentMode === 'path') {
pathActions$1.toEditMode(element);
} else if (curConfig.selectNew) {
selectOnly([element], true);
} // we create the insert command that is stored on the stack
// undo means to call cmd.unapply(), redo means to call cmd.apply()
addCommandToHistory(new InsertElementCommand$1(element));
call('changed', [element]);
}, aniDur * 1000);
}
startTransform = null;
};
var dblClick = function dblClick(evt) {
var evtTarget = evt.target;
var parent = evtTarget.parentNode; // Do nothing if already in current group
if (parent === currentGroup) {
return;
}
var mouseTarget = getMouseTarget(evt);
var _mouseTarget = mouseTarget,
tagName = _mouseTarget.tagName;
if (tagName === 'text' && currentMode !== 'textedit') {
var pt = transformPoint(evt.pageX, evt.pageY, rootSctm);
textActions.select(mouseTarget, pt.x, pt.y);
}
if ((tagName === 'g' || tagName === 'a') && getRotationAngle(mouseTarget)) {
// TODO: Allow method of in-group editing without having to do
// this (similar to editing rotated paths)
// Ungroup and regroup
pushGroupProperties(mouseTarget);
mouseTarget = selectedElements[0];
clearSelection(true);
} // Reset context
if (currentGroup) {
leaveContext();
}
if (parent.tagName !== 'g' && parent.tagName !== 'a' || parent === getCurrentDrawing().getCurrentLayer() || mouseTarget === selectorManager.selectorParentGroup) {
// Escape from in-group edit
return;
}
setContext(mouseTarget);
}; // prevent links from being followed in the canvas
var handleLinkInCanvas = function handleLinkInCanvas(e) {
e.preventDefault();
return false;
}; // Added mouseup to the container here.
// TODO(codedread): Figure out why after the Closure compiler, the window mouseup is ignored.
$$8(container).mousedown(mouseDown).mousemove(mouseMove).click(handleLinkInCanvas).dblclick(dblClick).mouseup(mouseUp); // $(window).mouseup(mouseUp);
// TODO(rafaelcastrocouto): User preference for shift key and zoom factor
$$8(container).bind('mousewheel DOMMouseScroll',
/**
* @param {Event} e
* @fires module:svgcanvas.SvgCanvas#event:updateCanvas
* @fires module:svgcanvas.SvgCanvas#event:zoomDone
* @returns {void}
*/
function (e) {
if (!e.shiftKey) {
return;
}
e.preventDefault();
var evt = e.originalEvent;
rootSctm = $$8('#svgcontent g')[0].getScreenCTM().inverse();
var workarea = $$8('#workarea');
var scrbar = 15;
var rulerwidth = curConfig.showRulers ? 16 : 0; // mouse relative to content area in content pixels
var pt = transformPoint(evt.pageX, evt.pageY, rootSctm); // full work area width in screen pixels
var editorFullW = workarea.width();
var editorFullH = workarea.height(); // work area width minus scroll and ruler in screen pixels
var editorW = editorFullW - scrbar - rulerwidth;
var editorH = editorFullH - scrbar - rulerwidth; // work area width in content pixels
var workareaViewW = editorW * rootSctm.a;
var workareaViewH = editorH * rootSctm.d; // content offset from canvas in screen pixels
var wOffset = workarea.offset();
var wOffsetLeft = wOffset.left + rulerwidth;
var wOffsetTop = wOffset.top + rulerwidth;
var delta = evt.wheelDelta ? evt.wheelDelta : evt.detail ? -evt.detail : 0;
if (!delta) {
return;
}
var factor = Math.max(3 / 4, Math.min(4 / 3, delta));
var wZoom, hZoom;
if (factor > 1) {
wZoom = Math.ceil(editorW / workareaViewW * factor * 100) / 100;
hZoom = Math.ceil(editorH / workareaViewH * factor * 100) / 100;
} else {
wZoom = Math.floor(editorW / workareaViewW * factor * 100) / 100;
hZoom = Math.floor(editorH / workareaViewH * factor * 100) / 100;
}
var zoomlevel = Math.min(wZoom, hZoom);
zoomlevel = Math.min(10, Math.max(0.01, zoomlevel));
if (zoomlevel === currentZoom) {
return;
}
factor = zoomlevel / currentZoom; // top left of workarea in content pixels before zoom
var topLeftOld = transformPoint(wOffsetLeft, wOffsetTop, rootSctm); // top left of workarea in content pixels after zoom
var topLeftNew = {
x: pt.x - (pt.x - topLeftOld.x) / factor,
y: pt.y - (pt.y - topLeftOld.y) / factor
}; // top left of workarea in canvas pixels relative to content after zoom
var topLeftNewCanvas = {
x: topLeftNew.x * zoomlevel,
y: topLeftNew.y * zoomlevel
}; // new center in canvas pixels
var newCtr = {
x: topLeftNewCanvas.x - rulerwidth + editorFullW / 2,
y: topLeftNewCanvas.y - rulerwidth + editorFullH / 2
};
canvas.setZoom(zoomlevel);
$$8('#zoom').val((zoomlevel * 100).toFixed(1));
call('updateCanvas', {
center: false,
newCtr: newCtr
});
call('zoomDone');
});
})();
/* eslint-disable jsdoc/require-property */
/**
* Group: Text edit functions
* Functions relating to editing text elements.
* @namespace {PlainObject} textActions
* @memberof module:svgcanvas.SvgCanvas#
*/
var textActions = canvas.textActions = function () {
/* eslint-enable jsdoc/require-property */
var curtext;
var textinput;
var cursor;
var selblock;
var blinker;
var chardata = [];
var textbb; // , transbb;
var matrix;
var lastX, lastY;
var allowDbl;
/**
*
* @param {Integer} index
* @returns {void}
*/
function setCursor(index) {
var empty = textinput.value === '';
$$8(textinput).focus();
if (!arguments.length) {
if (empty) {
index = 0;
} else {
if (textinput.selectionEnd !== textinput.selectionStart) {
return;
}
index = textinput.selectionEnd;
}
}
var charbb = chardata[index];
if (!empty) {
textinput.setSelectionRange(index, index);
}
cursor = getElem('text_cursor');
if (!cursor) {
cursor = document.createElementNS(NS.SVG, 'line');
assignAttributes(cursor, {
id: 'text_cursor',
stroke: '#333',
'stroke-width': 1
});
cursor = getElem('selectorParentGroup').appendChild(cursor);
}
if (!blinker) {
blinker = setInterval(function () {
var show = cursor.getAttribute('display') === 'none';
cursor.setAttribute('display', show ? 'inline' : 'none');
}, 600);
}
var startPt = ptToScreen(charbb.x, textbb.y);
var endPt = ptToScreen(charbb.x, textbb.y + textbb.height);
assignAttributes(cursor, {
x1: startPt.x,
y1: startPt.y,
x2: endPt.x,
y2: endPt.y,
visibility: 'visible',
display: 'inline'
});
if (selblock) {
selblock.setAttribute('d', '');
}
}
/**
*
* @param {Integer} start
* @param {Integer} end
* @param {boolean} skipInput
* @returns {void}
*/
function setSelection(start, end, skipInput) {
if (start === end) {
setCursor(end);
return;
}
if (!skipInput) {
textinput.setSelectionRange(start, end);
}
selblock = getElem('text_selectblock');
if (!selblock) {
selblock = document.createElementNS(NS.SVG, 'path');
assignAttributes(selblock, {
id: 'text_selectblock',
fill: 'green',
opacity: 0.5,
style: 'pointer-events:none'
});
getElem('selectorParentGroup').append(selblock);
}
var startbb = chardata[start];
var endbb = chardata[end];
cursor.setAttribute('visibility', 'hidden');
var tl = ptToScreen(startbb.x, textbb.y),
tr = ptToScreen(startbb.x + (endbb.x - startbb.x), textbb.y),
bl = ptToScreen(startbb.x, textbb.y + textbb.height),
br = ptToScreen(startbb.x + (endbb.x - startbb.x), textbb.y + textbb.height);
var dstr = 'M' + tl.x + ',' + tl.y + ' L' + tr.x + ',' + tr.y + ' ' + br.x + ',' + br.y + ' ' + bl.x + ',' + bl.y + 'z';
assignAttributes(selblock, {
d: dstr,
display: 'inline'
});
}
/**
*
* @param {Float} mouseX
* @param {Float} mouseY
* @returns {Integer}
*/
function getIndexFromPoint(mouseX, mouseY) {
// Position cursor here
var pt = svgroot.createSVGPoint();
pt.x = mouseX;
pt.y = mouseY; // No content, so return 0
if (chardata.length === 1) {
return 0;
} // Determine if cursor should be on left or right of character
var charpos = curtext.getCharNumAtPosition(pt);
if (charpos < 0) {
// Out of text range, look at mouse coords
charpos = chardata.length - 2;
if (mouseX <= chardata[0].x) {
charpos = 0;
}
} else if (charpos >= chardata.length - 2) {
charpos = chardata.length - 2;
}
var charbb = chardata[charpos];
var mid = charbb.x + charbb.width / 2;
if (mouseX > mid) {
charpos++;
}
return charpos;
}
/**
*
* @param {Float} mouseX
* @param {Float} mouseY
* @returns {void}
*/
function setCursorFromPoint(mouseX, mouseY) {
setCursor(getIndexFromPoint(mouseX, mouseY));
}
/**
*
* @param {Float} x
* @param {Float} y
* @param {boolean} apply
* @returns {void}
*/
function setEndSelectionFromPoint(x, y, apply) {
var i1 = textinput.selectionStart;
var i2 = getIndexFromPoint(x, y);
var start = Math.min(i1, i2);
var end = Math.max(i1, i2);
setSelection(start, end, !apply);
}
/**
*
* @param {Float} xIn
* @param {Float} yIn
* @returns {module:math.XYObject}
*/
function screenToPt(xIn, yIn) {
var out = {
x: xIn,
y: yIn
};
out.x /= currentZoom;
out.y /= currentZoom;
if (matrix) {
var pt = transformPoint(out.x, out.y, matrix.inverse());
out.x = pt.x;
out.y = pt.y;
}
return out;
}
/**
*
* @param {Float} xIn
* @param {Float} yIn
* @returns {module:math.XYObject}
*/
function ptToScreen(xIn, yIn) {
var out = {
x: xIn,
y: yIn
};
if (matrix) {
var pt = transformPoint(out.x, out.y, matrix);
out.x = pt.x;
out.y = pt.y;
}
out.x *= currentZoom;
out.y *= currentZoom;
return out;
}
/*
// Not currently in use
function hideCursor () {
if (cursor) {
cursor.setAttribute('visibility', 'hidden');
}
}
*/
/**
*
* @param {Event} evt
* @returns {void}
*/
function selectAll(evt) {
setSelection(0, curtext.textContent.length);
$$8(this).unbind(evt);
}
/**
*
* @param {Event} evt
* @returns {void}
*/
function selectWord(evt) {
if (!allowDbl || !curtext) {
return;
}
var ept = transformPoint(evt.pageX, evt.pageY, rootSctm),
mouseX = ept.x * currentZoom,
mouseY = ept.y * currentZoom;
var pt = screenToPt(mouseX, mouseY);
var index = getIndexFromPoint(pt.x, pt.y);
var str = curtext.textContent;
var first = str.substr(0, index).replace(/[a-z\d]+$/i, '').length;
var m = str.substr(index).match(/^[a-z\d]+/i);
var last = (m ? m[0].length : 0) + index;
setSelection(first, last); // Set tripleclick
$$8(evt.target).click(selectAll);
setTimeout(function () {
$$8(evt.target).unbind('click', selectAll);
}, 300);
}
return (
/** @lends module:svgcanvas.SvgCanvas#textActions */
{
/**
* @param {Element} target
* @param {Float} x
* @param {Float} y
* @returns {void}
*/
select: function select(target, x, y) {
curtext = target;
textActions.toEditMode(x, y);
},
/**
* @param {Element} elem
* @returns {void}
*/
start: function start(elem) {
curtext = elem;
textActions.toEditMode();
},
/**
* @param {external:MouseEvent} evt
* @param {Element} mouseTarget
* @param {Float} startX
* @param {Float} startY
* @returns {void}
*/
mouseDown: function mouseDown(evt, mouseTarget, startX, startY) {
var pt = screenToPt(startX, startY);
textinput.focus();
setCursorFromPoint(pt.x, pt.y);
lastX = startX;
lastY = startY; // TODO: Find way to block native selection
},
/**
* @param {Float} mouseX
* @param {Float} mouseY
* @returns {void}
*/
mouseMove: function mouseMove(mouseX, mouseY) {
var pt = screenToPt(mouseX, mouseY);
setEndSelectionFromPoint(pt.x, pt.y);
},
/**
* @param {external:MouseEvent} evt
* @param {Float} mouseX
* @param {Float} mouseY
* @returns {void}
*/
mouseUp: function mouseUp(evt, mouseX, mouseY) {
var pt = screenToPt(mouseX, mouseY);
setEndSelectionFromPoint(pt.x, pt.y, true); // TODO: Find a way to make this work: Use transformed BBox instead of evt.target
// if (lastX === mouseX && lastY === mouseY
// && !rectsIntersect(transbb, {x: pt.x, y: pt.y, width: 0, height: 0})) {
// textActions.toSelectMode(true);
// }
if (evt.target !== curtext && mouseX < lastX + 2 && mouseX > lastX - 2 && mouseY < lastY + 2 && mouseY > lastY - 2) {
textActions.toSelectMode(true);
}
},
/**
* @function
* @param {Integer} index
* @returns {void}
*/
setCursor: setCursor,
/**
* @param {Float} x
* @param {Float} y
* @returns {void}
*/
toEditMode: function toEditMode(x, y) {
allowDbl = false;
currentMode = 'textedit';
selectorManager.requestSelector(curtext).showGrips(false); // Make selector group accept clicks
/* const selector = */
selectorManager.requestSelector(curtext); // Do we need this? Has side effect of setting lock, so keeping for now, but next line wasn't being used
// const sel = selector.selectorRect;
textActions.init();
$$8(curtext).css('cursor', 'text'); // if (supportsEditableText()) {
// curtext.setAttribute('editable', 'simple');
// return;
// }
if (!arguments.length) {
setCursor();
} else {
var pt = screenToPt(x, y);
setCursorFromPoint(pt.x, pt.y);
}
setTimeout(function () {
allowDbl = true;
}, 300);
},
/**
* @param {boolean|Element} selectElem
* @fires module:svgcanvas.SvgCanvas#event:selected
* @returns {void}
*/
toSelectMode: function toSelectMode(selectElem) {
currentMode = 'select';
clearInterval(blinker);
blinker = null;
if (selblock) {
$$8(selblock).attr('display', 'none');
}
if (cursor) {
$$8(cursor).attr('visibility', 'hidden');
}
$$8(curtext).css('cursor', 'move');
if (selectElem) {
clearSelection();
$$8(curtext).css('cursor', 'move');
call('selected', [curtext]);
addToSelection([curtext], true);
}
if (curtext && !curtext.textContent.length) {
// No content, so delete
canvas.deleteSelectedElements();
}
$$8(textinput).blur();
curtext = false; // if (supportsEditableText()) {
// curtext.removeAttribute('editable');
// }
},
/**
* @param {Element} elem
* @returns {void}
*/
setInputElem: function setInputElem(elem) {
textinput = elem; // $(textinput).blur(hideCursor);
},
/**
* @returns {void}
*/
clear: function clear() {
if (currentMode === 'textedit') {
textActions.toSelectMode();
}
},
/**
* @param {Element} inputElem Not in use
* @returns {void}
*/
init: function init(inputElem) {
if (!curtext) {
return;
}
var i, end; // if (supportsEditableText()) {
// curtext.select();
// return;
// }
if (!curtext.parentNode) {
// Result of the ffClone, need to get correct element
curtext = selectedElements[0];
selectorManager.requestSelector(curtext).showGrips(false);
}
var str = curtext.textContent;
var len = str.length;
var xform = curtext.getAttribute('transform');
textbb = getBBox(curtext);
matrix = xform ? getMatrix(curtext) : null;
chardata = [];
chardata.length = len;
textinput.focus();
$$8(curtext).unbind('dblclick', selectWord).dblclick(selectWord);
if (!len) {
end = {
x: textbb.x + textbb.width / 2,
width: 0
};
}
for (i = 0; i < len; i++) {
var start = curtext.getStartPositionOfChar(i);
end = curtext.getEndPositionOfChar(i);
if (!supportsGoodTextCharPos()) {
var offset = canvas.contentW * currentZoom;
start.x -= offset;
end.x -= offset;
start.x /= currentZoom;
end.x /= currentZoom;
} // Get a "bbox" equivalent for each character. Uses the
// bbox data of the actual text for y, height purposes
// TODO: Decide if y, width and height are actually necessary
chardata[i] = {
x: start.x,
y: textbb.y,
// start.y?
width: end.x - start.x,
height: textbb.height
};
} // Add a last bbox for cursor at end of text
chardata.push({
x: end.x,
width: 0
});
setSelection(textinput.selectionStart, textinput.selectionEnd, true);
}
}
);
}();
/**
* Group: Serialization.
*/
/**
* Looks at DOM elements inside the `<defs>` to see if they are referred to,
* removes them from the DOM if they are not.
* @function module:svgcanvas.SvgCanvas#removeUnusedDefElems
* @returns {Integer} The number of elements that were removed
*/
var removeUnusedDefElems = this.removeUnusedDefElems = function () {
var defs = svgcontent.getElementsByTagNameNS(NS.SVG, 'defs');
if (!defs || !defs.length) {
return 0;
} // if (!defs.firstChild) { return; }
var defelemUses = [];
var numRemoved = 0;
var attrs = ['fill', 'stroke', 'filter', 'marker-start', 'marker-mid', 'marker-end'];
var alen = attrs.length;
var allEls = svgcontent.getElementsByTagNameNS(NS.SVG, '*');
var allLen = allEls.length;
var i, j;
for (i = 0; i < allLen; i++) {
var el = allEls[i];
for (j = 0; j < alen; j++) {
var ref = getUrlFromAttr(el.getAttribute(attrs[j]));
if (ref) {
defelemUses.push(ref.substr(1));
}
} // gradients can refer to other gradients
var href = getHref(el);
if (href && href.startsWith('#')) {
defelemUses.push(href.substr(1));
}
}
var defelems = $$8(defs).find('linearGradient, radialGradient, filter, marker, svg, symbol');
i = defelems.length;
while (i--) {
var defelem = defelems[i];
var id = defelem.id;
if (!defelemUses.includes(id)) {
// Not found, so remove (but remember)
removedElements[id] = defelem;
defelem.remove();
numRemoved++;
}
}
return numRemoved;
};
/**
* Main function to set up the SVG content for output.
* @function module:svgcanvas.SvgCanvas#svgCanvasToString
* @returns {string} The SVG image for output
*/
this.svgCanvasToString = function () {
// keep calling it until there are none to remove
while (removeUnusedDefElems() > 0) {} // eslint-disable-line no-empty
pathActions$1.clear(true); // Keep SVG-Edit comment on top
$$8.each(svgcontent.childNodes, function (i, node) {
if (i && node.nodeType === 8 && node.data.includes('Created with')) {
svgcontent.firstChild.before(node);
}
}); // Move out of in-group editing mode
if (currentGroup) {
leaveContext();
selectOnly([currentGroup]);
}
var nakedSvgs = []; // Unwrap gsvg if it has no special attributes (only id and style)
$$8(svgcontent).find('g:data(gsvg)').each(function () {
var attrs = this.attributes;
var len = attrs.length;
for (var i = 0; i < len; i++) {
if (attrs[i].nodeName === 'id' || attrs[i].nodeName === 'style') {
len--;
}
} // No significant attributes, so ungroup
if (len <= 0) {
var svg = this.firstChild;
nakedSvgs.push(svg);
$$8(this).replaceWith(svg);
}
});
var output = this.svgToString(svgcontent, 0); // Rewrap gsvg
if (nakedSvgs.length) {
$$8(nakedSvgs).each(function () {
groupSvgElem(this);
});
}
return output;
};
/**
* Sub function ran on each SVG element to convert it to a string as desired.
* @function module:svgcanvas.SvgCanvas#svgToString
* @param {Element} elem - The SVG element to convert
* @param {Integer} indent - Number of spaces to indent this tag
* @returns {string} The given element as an SVG tag
*/
this.svgToString = function (elem, indent) {
var out = [];
var unit = curConfig.baseUnit;
var unitRe = new RegExp('^-?[\\d\\.]+' + unit + '$');
if (elem) {
cleanupElement(elem);
var attrs = _toConsumableArray(elem.attributes);
var childs = elem.childNodes;
attrs.sort(function (a, b) {
return a.name > b.name ? -1 : 1;
});
for (var i = 0; i < indent; i++) {
out.push(' ');
}
out.push('<');
out.push(elem.nodeName);
if (elem.id === 'svgcontent') {
// Process root element separately
var res = getResolution();
var vb = ''; // TODO: Allow this by dividing all values by current baseVal
// Note that this also means we should properly deal with this on import
// if (curConfig.baseUnit !== 'px') {
// const unit = curConfig.baseUnit;
// const unitM = getTypeMap()[unit];
// res.w = shortFloat(res.w / unitM);
// res.h = shortFloat(res.h / unitM);
// vb = ' viewBox="' + [0, 0, res.w, res.h].join(' ') + '"';
// res.w += unit;
// res.h += unit;
// }
if (unit !== 'px') {
res.w = convertUnit(res.w, unit) + unit;
res.h = convertUnit(res.h, unit) + unit;
}
out.push(' width="' + res.w + '" height="' + res.h + '"' + vb + ' xmlns="' + NS.SVG + '"');
var nsuris = {}; // Check elements for namespaces, add if found
$$8(elem).find('*').andSelf().each(function () {
// const el = this;
// for some elements have no attribute
var uri = this.namespaceURI;
if (uri && !nsuris[uri] && nsMap[uri] && nsMap[uri] !== 'xmlns' && nsMap[uri] !== 'xml') {
nsuris[uri] = true;
out.push(' xmlns:' + nsMap[uri] + '="' + uri + '"');
}
$$8.each(this.attributes, function (i, attr) {
var u = attr.namespaceURI;
if (u && !nsuris[u] && nsMap[u] !== 'xmlns' && nsMap[u] !== 'xml') {
nsuris[u] = true;
out.push(' xmlns:' + nsMap[u] + '="' + u + '"');
}
});
});
var _i2 = attrs.length;
var attrNames = ['width', 'height', 'xmlns', 'x', 'y', 'viewBox', 'id', 'overflow'];
while (_i2--) {
var attr = attrs[_i2];
var attrVal = toXml(attr.value); // Namespaces have already been dealt with, so skip
if (attr.nodeName.startsWith('xmlns:')) {
continue;
} // only serialize attributes we don't use internally
if (attrVal !== '' && !attrNames.includes(attr.localName)) {
if (!attr.namespaceURI || nsMap[attr.namespaceURI]) {
out.push(' ');
out.push(attr.nodeName);
out.push('="');
out.push(attrVal);
out.push('"');
}
}
}
} else {
// Skip empty defs
if (elem.nodeName === 'defs' && !elem.firstChild) {
return '';
}
var mozAttrs = ['-moz-math-font-style', '_moz-math-font-style'];
for (var _i3 = attrs.length - 1; _i3 >= 0; _i3--) {
var _attr = attrs[_i3];
var _attrVal = toXml(_attr.value); // remove bogus attributes added by Gecko
if (mozAttrs.includes(_attr.localName)) {
continue;
}
2020-01-25 02:32:24 +00:00
if (_attrVal === 'null') {
var styleName = _attr.localName.replace(/-[a-z]/g, function (s) {
return s[1].toUpperCase();
});
if (Object.prototype.hasOwnProperty.call(elem.style, styleName)) {
continue;
}
}
if (_attrVal !== '') {
if (_attrVal.startsWith('pointer-events')) {
continue;
}
if (_attr.localName === 'class' && _attrVal.startsWith('se_')) {
continue;
}
out.push(' ');
if (_attr.localName === 'd') {
_attrVal = pathActions$1.convertPath(elem, true);
}
if (!isNaN(_attrVal)) {
_attrVal = shortFloat(_attrVal);
} else if (unitRe.test(_attrVal)) {
_attrVal = shortFloat(_attrVal) + unit;
} // Embed images when saving
if (saveOptions.apply && elem.nodeName === 'image' && _attr.localName === 'href' && saveOptions.images && saveOptions.images === 'embed') {
var img = encodableImages[_attrVal];
if (img) {
_attrVal = img;
}
} // map various namespaces to our fixed namespace prefixes
// (the default xmlns attribute itself does not get a prefix)
if (!_attr.namespaceURI || _attr.namespaceURI === NS.SVG || nsMap[_attr.namespaceURI]) {
out.push(_attr.nodeName);
out.push('="');
out.push(_attrVal);
out.push('"');
}
}
}
}
if (elem.hasChildNodes()) {
out.push('>');
indent++;
var bOneLine = false;
for (var _i4 = 0; _i4 < childs.length; _i4++) {
var child = childs.item(_i4);
switch (child.nodeType) {
case 1:
// element node
out.push('\n');
2020-01-25 02:32:24 +00:00
out.push(this.svgToString(child, indent));
break;
case 3:
{
// text node
var str = child.nodeValue.replace(/^\s+|\s+$/g, '');
if (str !== '') {
bOneLine = true;
out.push(String(toXml(str)));
}
break;
}
case 4:
// cdata node
out.push('\n');
out.push(new Array(indent + 1).join(' '));
out.push('<![CDATA[');
out.push(child.nodeValue);
out.push(']]>');
break;
case 8:
// comment
out.push('\n');
out.push(new Array(indent + 1).join(' '));
out.push('<!--');
out.push(child.data);
out.push('-->');
break;
} // switch on node type
}
indent--;
if (!bOneLine) {
out.push('\n');
for (var _i5 = 0; _i5 < indent; _i5++) {
out.push(' ');
}
}
out.push('</');
out.push(elem.nodeName);
out.push('>');
} else {
out.push('/>');
}
}
return out.join('');
}; // end svgToString()
/**
* Function to run when image data is found.
* @callback module:svgcanvas.ImageEmbeddedCallback
* @param {string|false} result Data URL
* @returns {void}
*/
/**
* Converts a given image file to a data URL when possible, then runs a given callback.
* @function module:svgcanvas.SvgCanvas#embedImage
* @param {string} src - The path/URL of the image
* @returns {Promise<string|false>} Resolves to a Data URL (string|false)
*/
this.embedImage = function (src) {
// Todo: Remove this Promise in favor of making an async/await `Image.load` utility
return new Promise(function (resolve, reject) {
// eslint-disable-line promise/avoid-new
// load in the image and once it's loaded, get the dimensions
$$8(new Image()).load(function (response, status, xhr) {
if (status === 'error') {
reject(new Error('Error loading image: ' + xhr.status + ' ' + xhr.statusText));
return;
} // create a canvas the same size as the raster image
var cvs = document.createElement('canvas');
cvs.width = this.width;
cvs.height = this.height; // load the raster image into the canvas
cvs.getContext('2d').drawImage(this, 0, 0); // retrieve the data: URL
try {
var urldata = ';svgedit_url=' + encodeURIComponent(src);
urldata = cvs.toDataURL().replace(';base64', urldata + ';base64');
encodableImages[src] = urldata;
} catch (e) {
encodableImages[src] = false;
}
lastGoodImgUrl = src;
resolve(encodableImages[src]);
}).attr('src', src);
});
};
/**
* Sets a given URL to be a "last good image" URL.
* @function module:svgcanvas.SvgCanvas#setGoodImage
* @param {string} val
* @returns {void}
*/
this.setGoodImage = function (val) {
lastGoodImgUrl = val;
};
/**
* Does nothing by default, handled by optional widget/extension.
* @function module:svgcanvas.SvgCanvas#open
* @returns {void}
*/
this.open = function () {
/* */
};
/**
* Serializes the current drawing into SVG XML text and passes it to the 'saved' handler.
* This function also includes the XML prolog. Clients of the `SvgCanvas` bind their save
* function to the 'saved' event.
* @function module:svgcanvas.SvgCanvas#save
* @param {module:svgcanvas.SaveOptions} opts
* @fires module:svgcanvas.SvgCanvas#event:saved
* @returns {void}
*/
this.save = function (opts) {
// remove the selected outline before serializing
clearSelection(); // Update save options if provided
if (opts) {
$$8.extend(saveOptions, opts);
}
saveOptions.apply = true; // no need for doctype, see https://jwatt.org/svg/authoring/#doctype-declaration
var str = this.svgCanvasToString();
call('saved', str);
};
/**
* @typedef {PlainObject} module:svgcanvas.IssuesAndCodes
* @property {string[]} issueCodes The locale-independent code names
* @property {string[]} issues The localized descriptions
*/
/**
* Codes only is useful for locale-independent detection.
* @returns {module:svgcanvas.IssuesAndCodes}
*/
function getIssues() {
// remove the selected outline before serializing
clearSelection(); // Check for known CanVG issues
var issues = [];
var issueCodes = []; // Selector and notice
var issueList = {
feGaussianBlur: uiStrings.exportNoBlur,
foreignObject: uiStrings.exportNoforeignObject,
'[stroke-dasharray]': uiStrings.exportNoDashArray
};
var content = $$8(svgcontent); // Add font/text check if Canvas Text API is not implemented
if (!('font' in $$8('<canvas>')[0].getContext('2d'))) {
issueList.text = uiStrings.exportNoText;
}
$$8.each(issueList, function (sel, descr) {
if (content.find(sel).length) {
issueCodes.push(sel);
issues.push(descr);
}
});
return {
issues: issues,
issueCodes: issueCodes
};
}
var canvg;
/**
* @typedef {"feGaussianBlur"|"foreignObject"|"[stroke-dasharray]"|"text"} module:svgcanvas.IssueCode
*/
/**
* @typedef {PlainObject} module:svgcanvas.ImageExportedResults
* @property {string} datauri Contents as a Data URL
* @property {string} bloburl May be the empty string
* @property {string} svg The SVG contents as a string
* @property {string[]} issues The localization messages of `issueCodes`
* @property {module:svgcanvas.IssueCode[]} issueCodes CanVG issues found with the SVG
* @property {"PNG"|"JPEG"|"BMP"|"WEBP"|"ICO"} type The chosen image type
* @property {"image/png"|"image/jpeg"|"image/bmp"|"image/webp"} mimeType The image MIME type
* @property {Float} quality A decimal between 0 and 1 (for use with JPEG or WEBP)
* @property {string} exportWindowName A convenience for passing along a `window.name` to target a window on which the export could be added
*/
/**
* Generates a PNG (or JPG, BMP, WEBP) Data URL based on the current image,
* then calls "exported" with an object including the string, image
* information, and any issues found.
* @function module:svgcanvas.SvgCanvas#rasterExport
* @param {"PNG"|"JPEG"|"BMP"|"WEBP"|"ICO"} [imgType="PNG"]
* @param {Float} [quality] Between 0 and 1
* @param {string} [exportWindowName]
* @param {PlainObject} [opts]
* @param {boolean} [opts.avoidEvent]
* @fires module:svgcanvas.SvgCanvas#event:exported
* @todo Confirm/fix ICO type
* @returns {Promise<module:svgcanvas.ImageExportedResults>} Resolves to {@link module:svgcanvas.ImageExportedResults}
*/
this.rasterExport = /*#__PURE__*/function () {
var _ref5 = _asyncToGenerator( /*#__PURE__*/regeneratorRuntime.mark(function _callee2(imgType, quality, exportWindowName) {
var opts,
type,
mimeType,
_getIssues,
issues,
issueCodes,
svg,
_ref6,
c,
_args2 = arguments;
return regeneratorRuntime.wrap(function _callee2$(_context2) {
while (1) {
switch (_context2.prev = _context2.next) {
case 0:
opts = _args2.length > 3 && _args2[3] !== undefined ? _args2[3] : {};
type = imgType === 'ICO' ? 'BMP' : imgType || 'PNG';
mimeType = 'image/' + type.toLowerCase();
_getIssues = getIssues(), issues = _getIssues.issues, issueCodes = _getIssues.issueCodes;
svg = this.svgCanvasToString();
if (canvg) {
_context2.next = 10;
break;
}
_context2.next = 8;
return importSetGlobal(curConfig.canvgPath + 'canvg.js', {
global: 'canvg'
});
case 8:
_ref6 = _context2.sent;
canvg = _ref6.canvg;
case 10:
if (!$$8('#export_canvas').length) {
$$8('<canvas>', {
id: 'export_canvas'
}).hide().appendTo('body');
}
c = $$8('#export_canvas')[0];
c.width = canvas.contentW;
c.height = canvas.contentH;
_context2.next = 16;
return canvg(c, svg);
case 16:
return _context2.abrupt("return", new Promise(function (resolve, reject) {
// eslint-disable-line promise/avoid-new
var dataURLType = type.toLowerCase();
var datauri = quality ? c.toDataURL('image/' + dataURLType, quality) : c.toDataURL('image/' + dataURLType);
var bloburl;
/**
* Called when `bloburl` is available for export.
* @returns {void}
*/
function done() {
var obj = {
datauri: datauri,
bloburl: bloburl,
svg: svg,
issues: issues,
issueCodes: issueCodes,
type: imgType,
mimeType: mimeType,
quality: quality,
exportWindowName: exportWindowName
};
if (!opts.avoidEvent) {
call('exported', obj);
}
resolve(obj);
}
if (c.toBlob) {
c.toBlob(function (blob) {
bloburl = createObjectURL(blob);
done();
}, mimeType, quality);
return;
}
bloburl = dataURLToObjectURL(datauri);
done();
}));
case 17:
case "end":
return _context2.stop();
}
}
}, _callee2, this);
}));
return function (_x4, _x5, _x6) {
return _ref5.apply(this, arguments);
};
}();
/**
* @external jsPDF
*/
/**
* @typedef {void|"save"|"arraybuffer"|"blob"|"datauristring"|"dataurlstring"|"dataurlnewwindow"|"datauri"|"dataurl"} external:jsPDF.OutputType
* @todo Newer version to add also allows these `outputType` values "bloburi"|"bloburl" which return strings, so document here and for `outputType` of `module:svgcanvas.PDFExportedResults` below if added
*/
/**
* @typedef {PlainObject} module:svgcanvas.PDFExportedResults
* @property {string} svg The SVG PDF output
* @property {string|ArrayBuffer|Blob|window} output The output based on the `outputType`;
* if `undefined`, "datauristring", "dataurlstring", "datauri",
* or "dataurl", will be a string (`undefined` gives a document, while the others
* build as Data URLs; "datauri" and "dataurl" change the location of the current page); if
* "arraybuffer", will return `ArrayBuffer`; if "blob", returns a `Blob`;
* if "dataurlnewwindow", will change the current page's location and return a string
* if in Safari and no window object is found; otherwise opens in, and returns, a new `window`
* object; if "save", will have the same return as "dataurlnewwindow" if
* `navigator.getUserMedia` support is found without `URL.createObjectURL` support; otherwise
* returns `undefined` but attempts to save
* @property {external:jsPDF.OutputType} outputType
* @property {string[]} issues The human-readable localization messages of corresponding `issueCodes`
* @property {module:svgcanvas.IssueCode[]} issueCodes
* @property {string} exportWindowName
*/
/**
* Generates a PDF based on the current image, then calls "exportedPDF" with
* an object including the string, the data URL, and any issues found.
* @function module:svgcanvas.SvgCanvas#exportPDF
* @param {string} [exportWindowName] Will also be used for the download file name here
* @param {external:jsPDF.OutputType} [outputType="dataurlstring"]
* @fires module:svgcanvas.SvgCanvas#event:exportedPDF
* @returns {Promise<module:svgcanvas.PDFExportedResults>} Resolves to {@link module:svgcanvas.PDFExportedResults}
*/
this.exportPDF = /*#__PURE__*/function () {
var _ref7 = _asyncToGenerator( /*#__PURE__*/regeneratorRuntime.mark(function _callee3(exportWindowName) {
var outputType,
modularVersion,
res,
orientation,
unit,
doc,
docTitle,
_getIssues2,
issues,
issueCodes,
svg,
obj,
_args3 = arguments;
return regeneratorRuntime.wrap(function _callee3$(_context3) {
while (1) {
switch (_context3.prev = _context3.next) {
case 0:
outputType = _args3.length > 1 && _args3[1] !== undefined ? _args3[1] : isChrome() ? 'save' : undefined;
if (window.jsPDF) {
_context3.next = 7;
break;
}
_context3.next = 4;
return importScript([// We do not currently have these paths configurable as they are
// currently global-only, so not Rolled-up
2020-01-25 02:32:24 +00:00
'jspdf/underscore-min.js', // 'jspdf/jspdf.min.js',
'../../svgedit-myfix/editor/jspdf/jspdf-1.0.150.debug.js']);
case 4:
modularVersion = !('svgEditor' in window) || !window.svgEditor || window.svgEditor.modules !== false; // Todo: Switch to `import()` when widely supported and available (also allow customization of path)
_context3.next = 7;
return importScript(curConfig.jspdfPath + 'jspdf.plugin.svgToPdf.js', {
type: modularVersion ? 'module' : 'text/javascript'
});
case 7:
res = getResolution();
orientation = res.w > res.h ? 'landscape' : 'portrait';
unit = 'pt'; // curConfig.baseUnit; // We could use baseUnit, but that is presumably not intended for export purposes
// Todo: Give options to use predefined jsPDF formats like "a4", etc. from pull-down (with option to keep customizable)
doc = jsPDF({
orientation: orientation,
unit: unit,
format: [res.w, res.h] // , compressPdf: true
});
docTitle = getDocumentTitle();
doc.setProperties({
title: docTitle
/* ,
subject: '',
author: '',
keywords: '',
creator: '' */
});
_getIssues2 = getIssues(), issues = _getIssues2.issues, issueCodes = _getIssues2.issueCodes;
svg = this.svgCanvasToString();
doc.addSVG(svg, 0, 0); // doc.output('save'); // Works to open in a new
// window; todo: configure this and other export
// options to optionally work in this manner as
// opposed to opening a new tab
outputType = outputType || 'dataurlstring';
obj = {
svg: svg,
issues: issues,
issueCodes: issueCodes,
exportWindowName: exportWindowName,
outputType: outputType
};
obj.output = doc.output(outputType, outputType === 'save' ? exportWindowName || 'svg.pdf' : undefined);
call('exportedPDF', obj);
return _context3.abrupt("return", obj);
case 21:
case "end":
return _context3.stop();
}
}
}, _callee3, this);
}));
return function (_x7) {
return _ref7.apply(this, arguments);
};
}();
/**
* Returns the current drawing as raw SVG XML text.
* @function module:svgcanvas.SvgCanvas#getSvgString
* @returns {string} The current drawing as raw SVG XML text.
*/
this.getSvgString = function () {
saveOptions.apply = false;
return this.svgCanvasToString();
};
/**
* This function determines whether to use a nonce in the prefix, when
* generating IDs for future documents in SVG-Edit.
* If you're controlling SVG-Edit externally, and want randomized IDs, call
* this BEFORE calling `svgCanvas.setSvgString`.
* @function module:svgcanvas.SvgCanvas#randomizeIds
* @param {boolean} [enableRandomization] If true, adds a nonce to the prefix. Thus
* `svgCanvas.randomizeIds() <==> svgCanvas.randomizeIds(true)`
* @returns {void}
*/
this.randomizeIds = function (enableRandomization) {
if (arguments.length > 0 && enableRandomization === false) {
randomizeIds(false, getCurrentDrawing());
} else {
randomizeIds(true, getCurrentDrawing());
}
};
/**
* Ensure each element has a unique ID.
* @function module:svgcanvas.SvgCanvas#uniquifyElems
* @param {Element} g - The parent element of the tree to give unique IDs
* @returns {void}
*/
var uniquifyElems = this.uniquifyElems = function (g) {
var ids = {}; // TODO: Handle markers and connectors. These are not yet re-identified properly
// as their referring elements do not get remapped.
//
// <marker id='se_marker_end_svg_7'/>
// <polyline id='svg_7' se:connector='svg_1 svg_6' marker-end='url(#se_marker_end_svg_7)'/>
//
// Problem #1: if svg_1 gets renamed, we do not update the polyline's se:connector attribute
// Problem #2: if the polyline svg_7 gets renamed, we do not update the marker id nor the polyline's marker-end attribute
var refElems = ['filter', 'linearGradient', 'pattern', 'radialGradient', 'symbol', 'textPath', 'use'];
walkTree(g, function (n) {
// if it's an element node
if (n.nodeType === 1) {
// and the element has an ID
if (n.id) {
// and we haven't tracked this ID yet
if (!(n.id in ids)) {
// add this id to our map
ids[n.id] = {
elem: null,
attrs: [],
hrefs: []
};
}
ids[n.id].elem = n;
} // now search for all attributes on this element that might refer
// to other elements
$$8.each(refAttrs, function (i, attr) {
var attrnode = n.getAttributeNode(attr);
if (attrnode) {
// the incoming file has been sanitized, so we should be able to safely just strip off the leading #
var url = getUrlFromAttr(attrnode.value),
refid = url ? url.substr(1) : null;
if (refid) {
if (!(refid in ids)) {
// add this id to our map
ids[refid] = {
elem: null,
attrs: [],
hrefs: []
};
}
ids[refid].attrs.push(attrnode);
}
}
}); // check xlink:href now
var href = getHref(n); // TODO: what if an <image> or <a> element refers to an element internally?
if (href && refElems.includes(n.nodeName)) {
var refid = href.substr(1);
if (refid) {
if (!(refid in ids)) {
// add this id to our map
ids[refid] = {
elem: null,
attrs: [],
hrefs: []
};
}
ids[refid].hrefs.push(n);
}
}
}
}); // in ids, we now have a map of ids, elements and attributes, let's re-identify
for (var oldid in ids) {
if (!oldid) {
continue;
}
var elem = ids[oldid].elem;
if (elem) {
var newid = getNextId(); // assign element its new id
elem.id = newid; // remap all url() attributes
var attrs = ids[oldid].attrs;
var j = attrs.length;
while (j--) {
var attr = attrs[j];
attr.ownerElement.setAttribute(attr.name, 'url(#' + newid + ')');
} // remap all href attributes
var hreffers = ids[oldid].hrefs;
var k = hreffers.length;
while (k--) {
var hreffer = hreffers[k];
setHref(hreffer, '#' + newid);
}
}
}
};
/**
* Assigns reference data for each use element.
* @function module:svgcanvas.SvgCanvas#setUseData
* @param {Element} parent
* @returns {void}
*/
var setUseData = this.setUseData = function (parent) {
var elems = $$8(parent);
if (parent.tagName !== 'use') {
elems = elems.find('use');
}
elems.each(function () {
var id = getHref(this).substr(1);
var refElem = getElem(id);
if (!refElem) {
return;
}
$$8(this).data('ref', refElem);
if (refElem.tagName === 'symbol' || refElem.tagName === 'svg') {
$$8(this).data('symbol', refElem).data('ref', refElem);
}
});
};
/**
* Converts gradients from userSpaceOnUse to objectBoundingBox.
* @function module:svgcanvas.SvgCanvas#convertGradients
* @param {Element} elem
* @returns {void}
*/
var convertGradients = this.convertGradients = function (elem) {
var elems = $$8(elem).find('linearGradient, radialGradient');
if (!elems.length && isWebkit()) {
// Bug in webkit prevents regular *Gradient selector search
elems = $$8(elem).find('*').filter(function () {
return this.tagName.includes('Gradient');
});
}
elems.each(function () {
var grad = this; // eslint-disable-line consistent-this
if ($$8(grad).attr('gradientUnits') === 'userSpaceOnUse') {
// TODO: Support more than one element with this ref by duplicating parent grad
var fillStrokeElems = $$8(svgcontent).find('[fill="url(#' + grad.id + ')"],[stroke="url(#' + grad.id + ')"]');
if (!fillStrokeElems.length) {
return;
} // get object's bounding box
var bb = getBBox(fillStrokeElems[0]); // This will occur if the element is inside a <defs> or a <symbol>,
// in which we shouldn't need to convert anyway.
if (!bb) {
return;
}
if (grad.tagName === 'linearGradient') {
var gCoords = $$8(grad).attr(['x1', 'y1', 'x2', 'y2']); // If has transform, convert
var tlist = grad.gradientTransform.baseVal;
if (tlist && tlist.numberOfItems > 0) {
var m = transformListToTransform(tlist).matrix;
var pt1 = transformPoint(gCoords.x1, gCoords.y1, m);
var pt2 = transformPoint(gCoords.x2, gCoords.y2, m);
gCoords.x1 = pt1.x;
gCoords.y1 = pt1.y;
gCoords.x2 = pt2.x;
gCoords.y2 = pt2.y;
grad.removeAttribute('gradientTransform');
}
$$8(grad).attr({
x1: (gCoords.x1 - bb.x) / bb.width,
y1: (gCoords.y1 - bb.y) / bb.height,
x2: (gCoords.x2 - bb.x) / bb.width,
y2: (gCoords.y2 - bb.y) / bb.height
});
grad.removeAttribute('gradientUnits');
} // else {
// Note: radialGradient elements cannot be easily converted
// because userSpaceOnUse will keep circular gradients, while
// objectBoundingBox will x/y scale the gradient according to
// its bbox.
//
// For now we'll do nothing, though we should probably have
// the gradient be updated as the element is moved, as
// inkscape/illustrator do.
//
// const gCoords = $(grad).attr(['cx', 'cy', 'r']);
//
// $(grad).attr({
// cx: (gCoords.cx - bb.x) / bb.width,
// cy: (gCoords.cy - bb.y) / bb.height,
// r: gCoords.r
// });
//
// grad.removeAttribute('gradientUnits');
// }
}
});
};
/**
* Converts selected/given `<use>` or child SVG element to a group.
* @function module:svgcanvas.SvgCanvas#convertToGroup
* @param {Element} elem
* @fires module:svgcanvas.SvgCanvas#event:selected
* @returns {void}
*/
var convertToGroup = this.convertToGroup = function (elem) {
if (!elem) {
elem = selectedElements[0];
}
var $elem = $$8(elem);
var batchCmd = new BatchCommand$1();
var ts;
if ($elem.data('gsvg')) {
// Use the gsvg as the new group
var svg = elem.firstChild;
var pt = $$8(svg).attr(['x', 'y']);
$$8(elem.firstChild.firstChild).unwrap();
$$8(elem).removeData('gsvg');
var tlist = getTransformList(elem);
var xform = svgroot.createSVGTransform();
xform.setTranslate(pt.x, pt.y);
tlist.appendItem(xform);
recalculateDimensions(elem);
call('selected', [elem]);
} else if ($elem.data('symbol')) {
elem = $elem.data('symbol');
ts = $elem.attr('transform');
var pos = $elem.attr(['x', 'y']);
var vb = elem.getAttribute('viewBox');
if (vb) {
var nums = vb.split(' ');
pos.x -= Number(nums[0]);
pos.y -= Number(nums[1]);
} // Not ideal, but works
ts += ' translate(' + (pos.x || 0) + ',' + (pos.y || 0) + ')';
var prev = $elem.prev(); // Remove <use> element
batchCmd.addSubCommand(new RemoveElementCommand$1($elem[0], $elem[0].nextSibling, $elem[0].parentNode));
$elem.remove(); // See if other elements reference this symbol
var hasMore = $$8(svgcontent).find('use:data(symbol)').length;
var g = svgdoc.createElementNS(NS.SVG, 'g');
var childs = elem.childNodes;
var i;
for (i = 0; i < childs.length; i++) {
g.append(childs[i].cloneNode(true));
} // Duplicate the gradients for Gecko, since they weren't included in the <symbol>
if (isGecko()) {
var dupeGrads = $$8(findDefs()).children('linearGradient,radialGradient,pattern').clone();
$$8(g).append(dupeGrads);
}
if (ts) {
g.setAttribute('transform', ts);
}
var parent = elem.parentNode;
uniquifyElems(g); // Put the dupe gradients back into <defs> (after uniquifying them)
if (isGecko()) {
$$8(findDefs()).append($$8(g).find('linearGradient,radialGradient,pattern'));
} // now give the g itself a new id
g.id = getNextId();
prev.after(g);
if (parent) {
if (!hasMore) {
// remove symbol/svg element
var _elem = elem,
nextSibling = _elem.nextSibling;
elem.remove();
batchCmd.addSubCommand(new RemoveElementCommand$1(elem, nextSibling, parent));
}
batchCmd.addSubCommand(new InsertElementCommand$1(g));
}
setUseData(g);
if (isGecko()) {
convertGradients(findDefs());
} else {
convertGradients(g);
} // recalculate dimensions on the top-level children so that unnecessary transforms
// are removed
walkTreePost(g, function (n) {
try {
recalculateDimensions(n);
} catch (e) {
console.log(e); // eslint-disable-line no-console
}
}); // Give ID for any visible element missing one
$$8(g).find(visElems).each(function () {
if (!this.id) {
this.id = getNextId();
}
});
selectOnly([g]);
var cm = pushGroupProperties(g, true);
if (cm) {
batchCmd.addSubCommand(cm);
}
addCommandToHistory(batchCmd);
} else {
console.log('Unexpected element to ungroup:', elem); // eslint-disable-line no-console
}
};
/**
* This function sets the current drawing as the input SVG XML.
* @function module:svgcanvas.SvgCanvas#setSvgString
* @param {string} xmlString - The SVG as XML text.
* @param {boolean} [preventUndo=false] - Indicates if we want to do the
* changes without adding them to the undo stack - e.g. for initializing a
* drawing on page load.
* @fires module:svgcanvas.SvgCanvas#event:setnonce
* @fires module:svgcanvas.SvgCanvas#event:unsetnonce
* @fires module:svgcanvas.SvgCanvas#event:changed
* @returns {boolean} This function returns `false` if the set was
* unsuccessful, `true` otherwise.
*/
this.setSvgString = function (xmlString, preventUndo) {
try {
// convert string into XML document
var newDoc = text2xml(xmlString);
if (newDoc.firstElementChild && newDoc.firstElementChild.namespaceURI !== NS.SVG) {
return false;
}
this.prepareSvg(newDoc);
var batchCmd = new BatchCommand$1('Change Source'); // remove old svg document
var _svgcontent = svgcontent,
nextSibling = _svgcontent.nextSibling;
svgcontent.remove();
var oldzoom = svgcontent;
batchCmd.addSubCommand(new RemoveElementCommand$1(oldzoom, nextSibling, svgroot)); // set new svg document
// If DOM3 adoptNode() available, use it. Otherwise fall back to DOM2 importNode()
if (svgdoc.adoptNode) {
svgcontent = svgdoc.adoptNode(newDoc.documentElement);
} else {
svgcontent = svgdoc.importNode(newDoc.documentElement, true);
}
svgroot.append(svgcontent);
var content = $$8(svgcontent);
canvas.current_drawing_ = new Drawing(svgcontent, idprefix); // retrieve or set the nonce
var nonce = getCurrentDrawing().getNonce();
if (nonce) {
call('setnonce', nonce);
} else {
call('unsetnonce');
} // change image href vals if possible
content.find('image').each(function () {
var image = this; // eslint-disable-line consistent-this
preventClickDefault(image);
var val = getHref(this);
if (val) {
if (val.startsWith('data:')) {
// Check if an SVG-edit data URI
var m = val.match(/svgedit_url=(.*?);/); // const m = val.match(/svgedit_url=(?<url>.*?);/);
if (m) {
var url = decodeURIComponent(m[1]); // const url = decodeURIComponent(m.groups.url);
$$8(new Image()).load(function () {
image.setAttributeNS(NS.XLINK, 'xlink:href', url);
}).attr('src', url);
}
} // Add to encodableImages if it loads
canvas.embedImage(val);
}
}); // Wrap child SVGs in group elements
content.find('svg').each(function () {
// Skip if it's in a <defs>
if ($$8(this).closest('defs').length) {
return;
}
uniquifyElems(this); // Check if it already has a gsvg group
var pa = this.parentNode;
if (pa.childNodes.length === 1 && pa.nodeName === 'g') {
$$8(pa).data('gsvg', this);
pa.id = pa.id || getNextId();
} else {
groupSvgElem(this);
}
}); // For Firefox: Put all paint elems in defs
if (isGecko()) {
content.find('linearGradient, radialGradient, pattern').appendTo(findDefs());
} // Set ref element for <use> elements
// TODO: This should also be done if the object is re-added through "redo"
setUseData(content);
convertGradients(content[0]);
var attrs = {
id: 'svgcontent',
overflow: curConfig.show_outside_canvas ? 'visible' : 'hidden'
};
var percs = false; // determine proper size
if (content.attr('viewBox')) {
var vb = content.attr('viewBox').split(' ');
attrs.width = vb[2];
attrs.height = vb[3]; // handle content that doesn't have a viewBox
} else {
$$8.each(['width', 'height'], function (i, dim) {
// Set to 100 if not given
var val = content.attr(dim) || '100%';
if (String(val).substr(-1) === '%') {
// Use user units if percentage given
percs = true;
} else {
attrs[dim] = convertToNum(dim, val);
}
});
} // identify layers
identifyLayers(); // Give ID for any visible layer children missing one
content.children().find(visElems).each(function () {
if (!this.id) {
this.id = getNextId();
}
}); // Percentage width/height, so let's base it on visible elements
if (percs) {
var bb = getStrokedBBoxDefaultVisible();
attrs.width = bb.width + bb.x;
attrs.height = bb.height + bb.y;
} // Just in case negative numbers are given or
// result from the percs calculation
if (attrs.width <= 0) {
attrs.width = 100;
}
if (attrs.height <= 0) {
attrs.height = 100;
}
content.attr(attrs);
this.contentW = attrs.width;
this.contentH = attrs.height;
batchCmd.addSubCommand(new InsertElementCommand$1(svgcontent)); // update root to the correct size
var changes = content.attr(['width', 'height']);
batchCmd.addSubCommand(new ChangeElementCommand$1(svgroot, changes)); // reset zoom
currentZoom = 1; // reset transform lists
resetListMap();
clearSelection();
clearData();
svgroot.append(selectorManager.selectorParentGroup);
if (!preventUndo) addCommandToHistory(batchCmd);
call('changed', [svgcontent]);
} catch (e) {
console.log(e); // eslint-disable-line no-console
return false;
}
return true;
};
/**
* This function imports the input SVG XML as a `<symbol>` in the `<defs>`, then adds a
* `<use>` to the current layer.
* @function module:svgcanvas.SvgCanvas#importSvgString
* @param {string} xmlString - The SVG as XML text.
* @fires module:svgcanvas.SvgCanvas#event:changed
* @returns {null|Element} This function returns null if the import was unsuccessful, or the element otherwise.
* @todo
* - properly handle if namespace is introduced by imported content (must add to svgcontent
* and update all prefixes in the imported node)
* - properly handle recalculating dimensions, `recalculateDimensions()` doesn't handle
* arbitrary transform lists, but makes some assumptions about how the transform list
* was obtained
*/
this.importSvgString = function (xmlString) {
var j, ts, useEl;
try {
// Get unique ID
var uid = encode64(xmlString.length + xmlString).substr(0, 32);
var useExisting = false; // Look for symbol and make sure symbol exists in image
if (importIds[uid]) {
if ($$8(importIds[uid].symbol).parents('#svgroot').length) {
useExisting = true;
}
}
var batchCmd = new BatchCommand$1('Import Image');
var symbol;
if (useExisting) {
symbol = importIds[uid].symbol;
ts = importIds[uid].xform;
} else {
// convert string into XML document
var newDoc = text2xml(xmlString);
this.prepareSvg(newDoc); // import new svg document into our document
var svg; // If DOM3 adoptNode() available, use it. Otherwise fall back to DOM2 importNode()
if (svgdoc.adoptNode) {
svg = svgdoc.adoptNode(newDoc.documentElement);
} else {
svg = svgdoc.importNode(newDoc.documentElement, true);
}
uniquifyElems(svg);
var innerw = convertToNum('width', svg.getAttribute('width')),
innerh = convertToNum('height', svg.getAttribute('height')),
innervb = svg.getAttribute('viewBox'),
// if no explicit viewbox, create one out of the width and height
vb = innervb ? innervb.split(' ') : [0, 0, innerw, innerh];
for (j = 0; j < 4; ++j) {
vb[j] = Number(vb[j]);
} // TODO: properly handle preserveAspectRatio
var // canvasw = +svgcontent.getAttribute('width'),
canvash = Number(svgcontent.getAttribute('height')); // imported content should be 1/3 of the canvas on its largest dimension
if (innerh > innerw) {
ts = 'scale(' + canvash / 3 / vb[3] + ')';
} else {
ts = 'scale(' + canvash / 3 / vb[2] + ')';
} // Hack to make recalculateDimensions understand how to scale
ts = 'translate(0) ' + ts + ' translate(0)';
symbol = svgdoc.createElementNS(NS.SVG, 'symbol');
var defs = findDefs();
if (isGecko()) {
// Move all gradients into root for Firefox, workaround for this bug:
// https://bugzilla.mozilla.org/show_bug.cgi?id=353575
// TODO: Make this properly undo-able.
$$8(svg).find('linearGradient, radialGradient, pattern').appendTo(defs);
}
while (svg.firstChild) {
var first = svg.firstChild;
symbol.append(first);
}
var attrs = svg.attributes;
var _iteratorNormalCompletion2 = true;
var _didIteratorError2 = false;
var _iteratorError2 = undefined;
try {
for (var _iterator2 = attrs[Symbol.iterator](), _step2; !(_iteratorNormalCompletion2 = (_step2 = _iterator2.next()).done); _iteratorNormalCompletion2 = true) {
var attr = _step2.value;
// Ok for `NamedNodeMap`
symbol.setAttribute(attr.nodeName, attr.value);
}
} catch (err) {
_didIteratorError2 = true;
_iteratorError2 = err;
} finally {
try {
if (!_iteratorNormalCompletion2 && _iterator2["return"] != null) {
_iterator2["return"]();
}
} finally {
if (_didIteratorError2) {
throw _iteratorError2;
}
}
}
symbol.id = getNextId(); // Store data
importIds[uid] = {
symbol: symbol,
xform: ts
};
findDefs().append(symbol);
batchCmd.addSubCommand(new InsertElementCommand$1(symbol));
}
useEl = svgdoc.createElementNS(NS.SVG, 'use');
useEl.id = getNextId();
setHref(useEl, '#' + symbol.id);
(currentGroup || getCurrentDrawing().getCurrentLayer()).append(useEl);
batchCmd.addSubCommand(new InsertElementCommand$1(useEl));
clearSelection();
useEl.setAttribute('transform', ts);
recalculateDimensions(useEl);
$$8(useEl).data('symbol', symbol).data('ref', symbol);
addToSelection([useEl]); // TODO: Find way to add this in a recalculateDimensions-parsable way
// if (vb[0] !== 0 || vb[1] !== 0) {
// ts = 'translate(' + (-vb[0]) + ',' + (-vb[1]) + ') ' + ts;
// }
addCommandToHistory(batchCmd);
call('changed', [svgcontent]);
} catch (e) {
console.log(e); // eslint-disable-line no-console
return null;
} // we want to return the element so we can automatically select it
return useEl;
}; // Could deprecate, but besides external uses, their usage makes clear that
// canvas is a dependency for all of these
var dr = {
identifyLayers: identifyLayers,
createLayer: createLayer,
cloneLayer: cloneLayer,
deleteCurrentLayer: deleteCurrentLayer,
setCurrentLayer: setCurrentLayer,
renameCurrentLayer: renameCurrentLayer,
setCurrentLayerPosition: setCurrentLayerPosition,
setLayerVisibility: setLayerVisibility,
moveSelectedToLayer: moveSelectedToLayer,
mergeLayer: mergeLayer,
mergeAllLayers: mergeAllLayers,
leaveContext: leaveContext,
setContext: setContext
};
Object.entries(dr).forEach(function (_ref8) {
var _ref9 = _slicedToArray(_ref8, 2),
prop = _ref9[0],
propVal = _ref9[1];
canvas[prop] = propVal;
});
init$3(
/**
* @implements {module:draw.DrawCanvasInit}
*/
{
pathActions: pathActions$1,
getCurrentGroup: function getCurrentGroup() {
return currentGroup;
},
setCurrentGroup: function setCurrentGroup(cg) {
currentGroup = cg;
},
getSelectedElements: getSelectedElements,
getSVGContent: getSVGContent,
undoMgr: undoMgr,
elData: elData,
getCurrentDrawing: getCurrentDrawing,
clearSelection: clearSelection,
call: call,
addCommandToHistory: addCommandToHistory,
/**
* @fires module:svgcanvas.SvgCanvas#event:changed
* @returns {void}
*/
changeSVGContent: function changeSVGContent() {
call('changed', [svgcontent]);
}
});
/**
* Group: Document functions.
*/
/**
* Clears the current document. This is not an undoable action.
* @function module:svgcanvas.SvgCanvas#clear
* @fires module:svgcanvas.SvgCanvas#event:cleared
* @returns {void}
*/
this.clear = function () {
pathActions$1.clear();
clearSelection(); // clear the svgcontent node
canvas.clearSvgContentElement(); // create new document
canvas.current_drawing_ = new Drawing(svgcontent); // create empty first layer
canvas.createLayer('Layer 1'); // clear the undo stack
canvas.undoMgr.resetUndoStack(); // reset the selector manager
selectorManager.initGroup(); // reset the rubber band box
rubberBox = selectorManager.getRubberBandBox();
call('cleared');
}; // Alias function
this.linkControlPoints = pathActions$1.linkControlPoints;
/**
* @function module:svgcanvas.SvgCanvas#getContentElem
* @returns {Element} The content DOM element
*/
this.getContentElem = function () {
return svgcontent;
};
/**
* @function module:svgcanvas.SvgCanvas#getRootElem
* @returns {SVGSVGElement} The root DOM element
*/
this.getRootElem = function () {
return svgroot;
};
/**
* @typedef {PlainObject} DimensionsAndZoom
* @property {Float} w Width
* @property {Float} h Height
* @property {Float} zoom Zoom
*/
/**
* @function module:svgcanvas.SvgCanvas#getResolution
* @returns {DimensionsAndZoom} The current dimensions and zoom level in an object
*/
var getResolution = this.getResolution = function () {
// const vb = svgcontent.getAttribute('viewBox').split(' ');
// return {w:vb[2], h:vb[3], zoom: currentZoom};
var w = svgcontent.getAttribute('width') / currentZoom;
var h = svgcontent.getAttribute('height') / currentZoom;
return {
w: w,
h: h,
zoom: currentZoom
};
};
/**
* @function module:svgcanvas.SvgCanvas#getSnapToGrid
* @returns {boolean} The current snap to grid setting
*/
this.getSnapToGrid = function () {
return curConfig.gridSnapping;
};
/**
* @function module:svgcanvas.SvgCanvas#getVersion
* @returns {string} A string which describes the revision number of SvgCanvas.
*/
this.getVersion = function () {
return 'svgcanvas.js ($Rev$)';
};
/**
* Update interface strings with given values.
* @function module:svgcanvas.SvgCanvas#setUiStrings
* @param {module:path.uiStrings} strs - Object with strings (see the [locales API]{@link module:locale.LocaleStrings} and the [tutorial]{@tutorial LocaleDocs})
* @returns {void}
*/
this.setUiStrings = function (strs) {
Object.assign(uiStrings, strs.notification);
$$8 = jQueryPluginDBox($$8, strs.common);
setUiStrings(strs);
};
/**
* Update configuration options with given values.
* @function module:svgcanvas.SvgCanvas#setConfig
* @param {module:SVGEditor.Config} opts - Object with options
* @returns {void}
*/
this.setConfig = function (opts) {
Object.assign(curConfig, opts);
};
/**
* @function module:svgcanvas.SvgCanvas#getTitle
* @param {Element} [elem]
* @returns {string|void} the current group/SVG's title contents or
* `undefined` if no element is passed nd there are no selected elements.
*/
this.getTitle = function (elem) {
elem = elem || selectedElements[0];
if (!elem) {
return undefined;
}
elem = $$8(elem).data('gsvg') || $$8(elem).data('symbol') || elem;
var childs = elem.childNodes;
var _iteratorNormalCompletion3 = true;
var _didIteratorError3 = false;
var _iteratorError3 = undefined;
try {
for (var _iterator3 = childs[Symbol.iterator](), _step3; !(_iteratorNormalCompletion3 = (_step3 = _iterator3.next()).done); _iteratorNormalCompletion3 = true) {
var child = _step3.value;
if (child.nodeName === 'title') {
return child.textContent;
}
}
} catch (err) {
_didIteratorError3 = true;
_iteratorError3 = err;
} finally {
try {
if (!_iteratorNormalCompletion3 && _iterator3["return"] != null) {
_iterator3["return"]();
}
} finally {
if (_didIteratorError3) {
throw _iteratorError3;
}
}
}
return '';
};
/**
* Sets the group/SVG's title content.
* @function module:svgcanvas.SvgCanvas#setGroupTitle
* @param {string} val
* @todo Combine this with `setDocumentTitle`
* @returns {void}
*/
this.setGroupTitle = function (val) {
var elem = selectedElements[0];
elem = $$8(elem).data('gsvg') || elem;
var ts = $$8(elem).children('title');
var batchCmd = new BatchCommand$1('Set Label');
var title;
if (!val.length) {
// Remove title element
var tsNextSibling = ts.nextSibling;
batchCmd.addSubCommand(new RemoveElementCommand$1(ts[0], tsNextSibling, elem));
ts.remove();
} else if (ts.length) {
// Change title contents
title = ts[0];
batchCmd.addSubCommand(new ChangeElementCommand$1(title, {
'#text': title.textContent
}));
title.textContent = val;
} else {
// Add title element
title = svgdoc.createElementNS(NS.SVG, 'title');
title.textContent = val;
$$8(elem).prepend(title);
batchCmd.addSubCommand(new InsertElementCommand$1(title));
}
addCommandToHistory(batchCmd);
};
/**
* @function module:svgcanvas.SvgCanvas#getDocumentTitle
* @returns {string|void} The current document title or an empty string if not found
*/
var getDocumentTitle = this.getDocumentTitle = function () {
return canvas.getTitle(svgcontent);
};
/**
* Adds/updates a title element for the document with the given name.
* This is an undoable action.
* @function module:svgcanvas.SvgCanvas#setDocumentTitle
* @param {string} newTitle - String with the new title
* @returns {void}
*/
this.setDocumentTitle = function (newTitle) {
var childs = svgcontent.childNodes;
var docTitle = false,
oldTitle = '';
var batchCmd = new BatchCommand$1('Change Image Title');
var _iteratorNormalCompletion4 = true;
var _didIteratorError4 = false;
var _iteratorError4 = undefined;
try {
for (var _iterator4 = childs[Symbol.iterator](), _step4; !(_iteratorNormalCompletion4 = (_step4 = _iterator4.next()).done); _iteratorNormalCompletion4 = true) {
var child = _step4.value;
if (child.nodeName === 'title') {
docTitle = child;
oldTitle = docTitle.textContent;
break;
}
}
} catch (err) {
_didIteratorError4 = true;
_iteratorError4 = err;
} finally {
try {
if (!_iteratorNormalCompletion4 && _iterator4["return"] != null) {
_iterator4["return"]();
}
} finally {
if (_didIteratorError4) {
throw _iteratorError4;
}
}
}
if (!docTitle) {
docTitle = svgdoc.createElementNS(NS.SVG, 'title');
svgcontent.insertBefore(docTitle, svgcontent.firstChild); // svgcontent.firstChild.before(docTitle); // Ok to replace above with this?
}
if (newTitle.length) {
docTitle.textContent = newTitle;
} else {
// No title given, so element is not necessary
docTitle.remove();
}
batchCmd.addSubCommand(new ChangeElementCommand$1(docTitle, {
'#text': oldTitle
}));
addCommandToHistory(batchCmd);
};
/**
* Returns the editor's namespace URL, optionally adding it to the root element.
* @function module:svgcanvas.SvgCanvas#getEditorNS
* @param {boolean} [add] - Indicates whether or not to add the namespace value
* @returns {string} The editor's namespace URL
*/
this.getEditorNS = function (add) {
if (add) {
svgcontent.setAttribute('xmlns:se', NS.SE);
}
return NS.SE;
};
/**
* Changes the document's dimensions to the given size.
* @function module:svgcanvas.SvgCanvas#setResolution
* @param {Float|"fit"} x - Number with the width of the new dimensions in user units.
* Can also be the string "fit" to indicate "fit to content".
* @param {Float} y - Number with the height of the new dimensions in user units.
* @fires module:svgcanvas.SvgCanvas#event:changed
* @returns {boolean} Indicates if resolution change was successful.
* It will fail on "fit to content" option with no content to fit to.
*/
this.setResolution = function (x, y) {
var res = getResolution();
var w = res.w,
h = res.h;
var batchCmd;
if (x === 'fit') {
// Get bounding box
var bbox = getStrokedBBoxDefaultVisible();
if (bbox) {
batchCmd = new BatchCommand$1('Fit Canvas to Content');
var visEls = getVisibleElements();
addToSelection(visEls);
var dx = [],
dy = [];
$$8.each(visEls, function (i, item) {
dx.push(bbox.x * -1);
dy.push(bbox.y * -1);
});
var cmd = canvas.moveSelectedElements(dx, dy, true);
batchCmd.addSubCommand(cmd);
clearSelection();
x = Math.round(bbox.width);
y = Math.round(bbox.height);
} else {
return false;
}
}
if (x !== w || y !== h) {
if (!batchCmd) {
batchCmd = new BatchCommand$1('Change Image Dimensions');
}
x = convertToNum('width', x);
y = convertToNum('height', y);
svgcontent.setAttribute('width', x);
svgcontent.setAttribute('height', y);
this.contentW = x;
this.contentH = y;
batchCmd.addSubCommand(new ChangeElementCommand$1(svgcontent, {
width: w,
height: h
}));
svgcontent.setAttribute('viewBox', [0, 0, x / currentZoom, y / currentZoom].join(' '));
batchCmd.addSubCommand(new ChangeElementCommand$1(svgcontent, {
viewBox: ['0 0', w, h].join(' ')
}));
addCommandToHistory(batchCmd);
call('changed', [svgcontent]);
}
return true;
};
/**
* @typedef {module:jQueryAttr.Attributes} module:svgcanvas.ElementPositionInCanvas
* @property {Float} x
* @property {Float} y
*/
/**
* @function module:svgcanvas.SvgCanvas#getOffset
* @returns {module:svgcanvas.ElementPositionInCanvas} An object with `x`, `y` values indicating the svgcontent element's
* position in the editor's canvas.
*/
this.getOffset = function () {
return $$8(svgcontent).attr(['x', 'y']);
};
/**
* @typedef {PlainObject} module:svgcanvas.ZoomAndBBox
* @property {Float} zoom
* @property {module:utilities.BBoxObject} bbox
*/
/**
* Sets the zoom level on the canvas-side based on the given value.
* @function module:svgcanvas.SvgCanvas#setBBoxZoom
* @param {"selection"|"canvas"|"content"|"layer"|module:SVGEditor.BBoxObjectWithFactor} val - Bounding box object to zoom to or string indicating zoom option. Note: the object value type is defined in `svg-editor.js`
* @param {Integer} editorW - The editor's workarea box's width
* @param {Integer} editorH - The editor's workarea box's height
* @returns {module:svgcanvas.ZoomAndBBox|void}
*/
this.setBBoxZoom = function (val, editorW, editorH) {
var spacer = 0.85;
var bb;
var calcZoom = function calcZoom(bb) {
// eslint-disable-line no-shadow
if (!bb) {
return false;
}
var wZoom = Math.round(editorW / bb.width * 100 * spacer) / 100;
var hZoom = Math.round(editorH / bb.height * 100 * spacer) / 100;
var zoom = Math.min(wZoom, hZoom);
canvas.setZoom(zoom);
return {
zoom: zoom,
bbox: bb
};
};
if (_typeof(val) === 'object') {
bb = val;
if (bb.width === 0 || bb.height === 0) {
var newzoom = bb.zoom ? bb.zoom : currentZoom * bb.factor;
canvas.setZoom(newzoom);
return {
zoom: currentZoom,
bbox: bb
};
}
return calcZoom(bb);
}
switch (val) {
case 'selection':
{
if (!selectedElements[0]) {
return undefined;
}
var selectedElems = $$8.map(selectedElements, function (n) {
if (n) {
return n;
}
return undefined;
});
bb = getStrokedBBoxDefaultVisible(selectedElems);
break;
}
case 'canvas':
{
var res = getResolution();
spacer = 0.95;
bb = {
width: res.w,
height: res.h,
x: 0,
y: 0
};
break;
}
case 'content':
bb = getStrokedBBoxDefaultVisible();
break;
case 'layer':
bb = getStrokedBBoxDefaultVisible(getVisibleElements(getCurrentDrawing().getCurrentLayer()));
break;
default:
return undefined;
}
return calcZoom(bb);
};
/**
* The zoom level has changed. Supplies the new zoom level as a number (not percentage).
* @event module:svgcanvas.SvgCanvas#event:ext_zoomChanged
* @type {Float}
*/
/**
* The bottom panel was updated.
* @event module:svgcanvas.SvgCanvas#event:ext_toolButtonStateUpdate
* @type {PlainObject}
* @property {boolean} nofill Indicates fill is disabled
* @property {boolean} nostroke Indicates stroke is disabled
*/
/**
* The element selection has changed (elements were added/removed from selection).
* @event module:svgcanvas.SvgCanvas#event:ext_selectedChanged
* @type {PlainObject}
* @property {Element[]} elems Array of the newly selected elements
* @property {Element|null} selectedElement The single selected element
* @property {boolean} multiselected Indicates whether one or more elements were selected
*/
/**
* Called when part of element is in process of changing, generally on
* mousemove actions like rotate, move, etc.
* @event module:svgcanvas.SvgCanvas#event:ext_elementTransition
* @type {PlainObject}
* @property {Element[]} elems Array of transitioning elements
*/
/**
* One or more elements were changed.
* @event module:svgcanvas.SvgCanvas#event:ext_elementChanged
* @type {PlainObject}
* @property {Element[]} elems Array of the affected elements
*/
/**
* Invoked as soon as the locale is ready.
* @event module:svgcanvas.SvgCanvas#event:ext_langReady
* @type {PlainObject}
* @property {string} lang The two-letter language code
* @property {module:SVGEditor.uiStrings} uiStrings
* @property {module:SVGEditor~ImportLocale} importLocale
*/
/**
* The language was changed. Two-letter code of the new language.
* @event module:svgcanvas.SvgCanvas#event:ext_langChanged
* @type {string}
*/
/**
* Means for an extension to add locale data. The two-letter language code.
* @event module:svgcanvas.SvgCanvas#event:ext_addLangData
* @type {PlainObject}
* @property {string} lang
* @property {module:SVGEditor~ImportLocale} importLocale
*/
/**
* Called when new image is created.
* @event module:svgcanvas.SvgCanvas#event:ext_onNewDocument
* @type {void}
*/
/**
* Called when sidepanel is resized or toggled.
* @event module:svgcanvas.SvgCanvas#event:ext_workareaResized
* @type {void}
*/
/**
* Called upon addition of the extension, or, if svgicons are set,
* after the icons are ready when extension SVG icons have loaded.
* @event module:svgcanvas.SvgCanvas#event:ext_callback
* @type {void}
*/
/**
* Sets the zoom to the given level.
* @function module:svgcanvas.SvgCanvas#setZoom
* @param {Float} zoomLevel - Float indicating the zoom level to change to
* @fires module:svgcanvas.SvgCanvas#event:ext_zoomChanged
* @returns {void}
*/
this.setZoom = function (zoomLevel) {
var res = getResolution();
svgcontent.setAttribute('viewBox', '0 0 ' + res.w / zoomLevel + ' ' + res.h / zoomLevel);
currentZoom = zoomLevel;
$$8.each(selectedElements, function (i, elem) {
if (!elem) {
return;
}
selectorManager.requestSelector(elem).resize();
});
pathActions$1.zoomChange();
runExtensions('zoomChanged',
/** @type {module:svgcanvas.SvgCanvas#event:ext_zoomChanged} */
zoomLevel);
};
/**
* @function module:svgcanvas.SvgCanvas#getMode
* @returns {string} The current editor mode string
*/
this.getMode = function () {
return currentMode;
};
/**
* Sets the editor's mode to the given string.
* @function module:svgcanvas.SvgCanvas#setMode
* @param {string} name - String with the new mode to change to
* @returns {void}
*/
this.setMode = function (name) {
pathActions$1.clear(true);
textActions.clear();
curProperties = selectedElements[0] && selectedElements[0].nodeName === 'text' ? curText : curShape;
currentMode = name;
};
/**
* Group: Element Styling.
*/
/**
* @typedef {PlainObject} module:svgcanvas.PaintOptions
* @property {"solidColor"} type
*/
/**
* @function module:svgcanvas.SvgCanvas#getColor
* @param {string} type
* @returns {string|module:svgcanvas.PaintOptions|Float|module:jGraduate~Paint} The current fill/stroke option
*/
this.getColor = function (type) {
return curProperties[type];
};
/**
* Change the current stroke/fill color/gradient value.
* @function module:svgcanvas.SvgCanvas#setColor
* @param {string} type - String indicating fill or stroke
* @param {string} val - The value to set the stroke attribute to
* @param {boolean} preventUndo - Boolean indicating whether or not this should be an undoable option
* @fires module:svgcanvas.SvgCanvas#event:changed
* @returns {void}
*/
this.setColor = function (type, val, preventUndo) {
curShape[type] = val;
curProperties[type + '_paint'] = {
type: 'solidColor'
};
var elems = [];
/**
*
* @param {Element} e
* @returns {void}
*/
function addNonG(e) {
if (e.nodeName !== 'g') {
elems.push(e);
}
}
var i = selectedElements.length;
while (i--) {
var elem = selectedElements[i];
if (elem) {
if (elem.tagName === 'g') {
walkTree(elem, addNonG);
} else if (type === 'fill') {
if (elem.tagName !== 'polyline' && elem.tagName !== 'line') {
elems.push(elem);
}
} else {
elems.push(elem);
}
}
}
if (elems.length > 0) {
if (!preventUndo) {
changeSelectedAttribute(type, val, elems);
call('changed', elems);
} else {
changeSelectedAttributeNoUndo(type, val, elems);
}
}
};
/**
* Apply the current gradient to selected element's fill or stroke.
* @function module:svgcanvas.SvgCanvas#setGradient
* @param {"fill"|"stroke"} type - String indicating "fill" or "stroke" to apply to an element
* @returns {void}
*/
var setGradient = this.setGradient = function (type) {
if (!curProperties[type + '_paint'] || curProperties[type + '_paint'].type === 'solidColor') {
return;
}
var grad = canvas[type + 'Grad']; // find out if there is a duplicate gradient already in the defs
var duplicateGrad = findDuplicateGradient(grad);
var defs = findDefs(); // no duplicate found, so import gradient into defs
if (!duplicateGrad) {
// const origGrad = grad;
grad = defs.appendChild(svgdoc.importNode(grad, true)); // get next id and set it on the grad
grad.id = getNextId();
} else {
// use existing gradient
grad = duplicateGrad;
}
canvas.setColor(type, 'url(#' + grad.id + ')');
};
/**
* Check if exact gradient already exists.
* @function module:svgcanvas~findDuplicateGradient
* @param {SVGGradientElement} grad - The gradient DOM element to compare to others
* @returns {SVGGradientElement} The existing gradient if found, `null` if not
*/
var findDuplicateGradient = function findDuplicateGradient(grad) {
var defs = findDefs();
var existingGrads = $$8(defs).find('linearGradient, radialGradient');
var i = existingGrads.length;
var radAttrs = ['r', 'cx', 'cy', 'fx', 'fy'];
while (i--) {
var og = existingGrads[i];
if (grad.tagName === 'linearGradient') {
if (grad.getAttribute('x1') !== og.getAttribute('x1') || grad.getAttribute('y1') !== og.getAttribute('y1') || grad.getAttribute('x2') !== og.getAttribute('x2') || grad.getAttribute('y2') !== og.getAttribute('y2')) {
continue;
}
} else {
var _ret = function () {
var gradAttrs = $$8(grad).attr(radAttrs);
var ogAttrs = $$8(og).attr(radAttrs);
var diff = false;
$$8.each(radAttrs, function (j, attr) {
if (gradAttrs[attr] !== ogAttrs[attr]) {
diff = true;
}
});
if (diff) {
return "continue";
}
}();
if (_ret === "continue") continue;
} // else could be a duplicate, iterate through stops
var stops = grad.getElementsByTagNameNS(NS.SVG, 'stop');
var ostops = og.getElementsByTagNameNS(NS.SVG, 'stop');
if (stops.length !== ostops.length) {
continue;
}
var j = stops.length;
while (j--) {
var stop = stops[j];
var ostop = ostops[j];
if (stop.getAttribute('offset') !== ostop.getAttribute('offset') || stop.getAttribute('stop-opacity') !== ostop.getAttribute('stop-opacity') || stop.getAttribute('stop-color') !== ostop.getAttribute('stop-color')) {
break;
}
}
if (j === -1) {
return og;
}
} // for each gradient in defs
return null;
};
/**
* Set a color/gradient to a fill/stroke.
* @function module:svgcanvas.SvgCanvas#setPaint
* @param {"fill"|"stroke"} type - String with "fill" or "stroke"
* @param {module:jGraduate.jGraduatePaintOptions} paint - The jGraduate paint object to apply
* @returns {void}
*/
this.setPaint = function (type, paint) {
// make a copy
var p = new $$8.jGraduate.Paint(paint);
this.setPaintOpacity(type, p.alpha / 100, true); // now set the current paint object
curProperties[type + '_paint'] = p;
switch (p.type) {
case 'solidColor':
this.setColor(type, p.solidColor !== 'none' ? '#' + p.solidColor : 'none');
break;
case 'linearGradient':
case 'radialGradient':
canvas[type + 'Grad'] = p[p.type];
setGradient(type);
break;
}
};
/**
* @function module:svgcanvas.SvgCanvas#setStrokePaint
* @param {module:jGraduate~Paint} paint
* @returns {void}
*/
this.setStrokePaint = function (paint) {
this.setPaint('stroke', paint);
};
/**
* @function module:svgcanvas.SvgCanvas#setFillPaint
* @param {module:jGraduate~Paint} paint
* @returns {void}
*/
this.setFillPaint = function (paint) {
this.setPaint('fill', paint);
};
/**
* @function module:svgcanvas.SvgCanvas#getStrokeWidth
* @returns {Float|string} The current stroke-width value
*/
this.getStrokeWidth = function () {
return curProperties.stroke_width;
};
/**
* Sets the stroke width for the current selected elements.
* When attempting to set a line's width to 0, this changes it to 1 instead.
* @function module:svgcanvas.SvgCanvas#setStrokeWidth
* @param {Float} val - A Float indicating the new stroke width value
* @fires module:svgcanvas.SvgCanvas#event:changed
* @returns {void}
*/
this.setStrokeWidth = function (val) {
if (val === 0 && ['line', 'path'].includes(currentMode)) {
canvas.setStrokeWidth(1);
return;
}
curProperties.stroke_width = val;
var elems = [];
/**
*
* @param {Element} e
* @returns {void}
*/
function addNonG(e) {
if (e.nodeName !== 'g') {
elems.push(e);
}
}
var i = selectedElements.length;
while (i--) {
var elem = selectedElements[i];
if (elem) {
if (elem.tagName === 'g') {
walkTree(elem, addNonG);
} else {
elems.push(elem);
}
}
}
if (elems.length > 0) {
changeSelectedAttribute('stroke-width', val, elems);
call('changed', selectedElements);
}
};
/**
* Set the given stroke-related attribute the given value for selected elements.
* @function module:svgcanvas.SvgCanvas#setStrokeAttr
* @param {string} attr - String with the attribute name
* @param {string|Float} val - String or number with the attribute value
* @fires module:svgcanvas.SvgCanvas#event:changed
* @returns {void}
*/
this.setStrokeAttr = function (attr, val) {
curShape[attr.replace('-', '_')] = val;
var elems = [];
var i = selectedElements.length;
while (i--) {
var elem = selectedElements[i];
if (elem) {
if (elem.tagName === 'g') {
walkTree(elem, function (e) {
if (e.nodeName !== 'g') {
elems.push(e);
}
});
} else {
elems.push(elem);
}
}
}
if (elems.length > 0) {
changeSelectedAttribute(attr, val, elems);
call('changed', selectedElements);
}
};
/**
* @typedef {PlainObject} module:svgcanvas.StyleOptions
* @property {string} fill
* @property {Float} fill_opacity
* @property {string} stroke
* @property {Float} stroke_width
* @property {string} stroke_dasharray
* @property {string} stroke_linejoin
* @property {string} stroke_linecap
* @property {Float} stroke_opacity
* @property {Float} opacity
*/
/**
* @function module:svgcanvas.SvgCanvas#getStyle
* @returns {module:svgcanvas.StyleOptions} current style options
*/
this.getStyle = function () {
return curShape;
};
/**
* @function module:svgcanvas.SvgCanvas#getOpacity
* @returns {Float} the current opacity
*/
this.getOpacity = getOpacity;
/**
* Sets the given opacity on the current selected elements.
* @function module:svgcanvas.SvgCanvas#setOpacity
* @param {string} val
* @returns {void}
*/
this.setOpacity = function (val) {
curShape.opacity = val;
changeSelectedAttribute('opacity', val);
};
/**
* @function module:svgcanvas.SvgCanvas#getFillOpacity
* @returns {Float} the current fill opacity
*/
this.getFillOpacity = function () {
return curShape.fill_opacity;
};
/**
* @function module:svgcanvas.SvgCanvas#getStrokeOpacity
* @returns {string} the current stroke opacity
*/
this.getStrokeOpacity = function () {
return curShape.stroke_opacity;
};
/**
* Sets the current fill/stroke opacity.
* @function module:svgcanvas.SvgCanvas#setPaintOpacity
* @param {string} type - String with "fill" or "stroke"
* @param {Float} val - Float with the new opacity value
* @param {boolean} preventUndo - Indicates whether or not this should be an undoable action
* @returns {void}
*/
this.setPaintOpacity = function (type, val, preventUndo) {
curShape[type + '_opacity'] = val;
if (!preventUndo) {
changeSelectedAttribute(type + '-opacity', val);
} else {
changeSelectedAttributeNoUndo(type + '-opacity', val);
}
};
/**
* Gets the current fill/stroke opacity.
* @function module:svgcanvas.SvgCanvas#getPaintOpacity
* @param {"fill"|"stroke"} type - String with "fill" or "stroke"
* @returns {Float} Fill/stroke opacity
*/
this.getPaintOpacity = function (type) {
return type === 'fill' ? this.getFillOpacity() : this.getStrokeOpacity();
};
/**
* Gets the `stdDeviation` blur value of the given element.
* @function module:svgcanvas.SvgCanvas#getBlur
* @param {Element} elem - The element to check the blur value for
* @returns {string} stdDeviation blur attribute value
*/
this.getBlur = function (elem) {
var val = 0; // const elem = selectedElements[0];
if (elem) {
var filterUrl = elem.getAttribute('filter');
if (filterUrl) {
var blur = getElem(elem.id + '_blur');
if (blur) {
val = blur.firstChild.getAttribute('stdDeviation');
}
}
}
return val;
};
(function () {
var curCommand = null;
var filter = null;
var filterHidden = false;
/**
* Sets the `stdDeviation` blur value on the selected element without being undoable.
* @function module:svgcanvas.SvgCanvas#setBlurNoUndo
* @param {Float} val - The new `stdDeviation` value
* @returns {void}
*/
canvas.setBlurNoUndo = function (val) {
if (!filter) {
canvas.setBlur(val);
return;
}
if (val === 0) {
// Don't change the StdDev, as that will hide the element.
// Instead, just remove the value for "filter"
changeSelectedAttributeNoUndo('filter', '');
filterHidden = true;
} else {
var elem = selectedElements[0];
if (filterHidden) {
changeSelectedAttributeNoUndo('filter', 'url(#' + elem.id + '_blur)');
}
if (isWebkit()) {
// console.log('e', elem); // eslint-disable-line no-console
elem.removeAttribute('filter');
elem.setAttribute('filter', 'url(#' + elem.id + '_blur)');
}
changeSelectedAttributeNoUndo('stdDeviation', val, [filter.firstChild]);
canvas.setBlurOffsets(filter, val);
}
};
/**
*
* @returns {void}
*/
function finishChange() {
var bCmd = canvas.undoMgr.finishUndoableChange();
curCommand.addSubCommand(bCmd);
addCommandToHistory(curCommand);
curCommand = null;
filter = null;
}
/**
* Sets the `x`, `y`, `width`, `height` values of the filter element in order to
* make the blur not be clipped. Removes them if not neeeded.
* @function module:svgcanvas.SvgCanvas#setBlurOffsets
* @param {Element} filterElem - The filter DOM element to update
* @param {Float} stdDev - The standard deviation value on which to base the offset size
* @returns {void}
*/
canvas.setBlurOffsets = function (filterElem, stdDev) {
if (stdDev > 3) {
// TODO: Create algorithm here where size is based on expected blur
assignAttributes(filterElem, {
x: '-50%',
y: '-50%',
width: '200%',
height: '200%'
}); // Removing these attributes hides text in Chrome (see Issue 579)
} else if (!isWebkit()) {
filterElem.removeAttribute('x');
filterElem.removeAttribute('y');
filterElem.removeAttribute('width');
filterElem.removeAttribute('height');
}
};
/**
* Adds/updates the blur filter to the selected element.
* @function module:svgcanvas.SvgCanvas#setBlur
* @param {Float} val - Float with the new `stdDeviation` blur value
* @param {boolean} complete - Whether or not the action should be completed (to add to the undo manager)
* @returns {void}
*/
canvas.setBlur = function (val, complete) {
if (curCommand) {
finishChange();
return;
} // Looks for associated blur, creates one if not found
var elem = selectedElements[0];
var elemId = elem.id;
filter = getElem(elemId + '_blur');
val -= 0;
var batchCmd = new BatchCommand$1(); // Blur found!
if (filter) {
if (val === 0) {
filter = null;
}
} else {
// Not found, so create
var newblur = addSVGElementFromJson({
element: 'feGaussianBlur',
attr: {
"in": 'SourceGraphic',
stdDeviation: val
}
});
filter = addSVGElementFromJson({
element: 'filter',
attr: {
id: elemId + '_blur'
}
});
filter.append(newblur);
findDefs().append(filter);
batchCmd.addSubCommand(new InsertElementCommand$1(filter));
}
var changes = {
filter: elem.getAttribute('filter')
};
if (val === 0) {
elem.removeAttribute('filter');
batchCmd.addSubCommand(new ChangeElementCommand$1(elem, changes));
return;
}
changeSelectedAttribute('filter', 'url(#' + elemId + '_blur)');
batchCmd.addSubCommand(new ChangeElementCommand$1(elem, changes));
canvas.setBlurOffsets(filter, val);
curCommand = batchCmd;
canvas.undoMgr.beginUndoableChange('stdDeviation', [filter ? filter.firstChild : null]);
if (complete) {
canvas.setBlurNoUndo(val);
finishChange();
}
};
})();
/**
* Check whether selected element is bold or not.
* @function module:svgcanvas.SvgCanvas#getBold
* @returns {boolean} Indicates whether or not element is bold
*/
this.getBold = function () {
// should only have one element selected
var selected = selectedElements[0];
if (!isNullish(selected) && selected.tagName === 'text' && isNullish(selectedElements[1])) {
return selected.getAttribute('font-weight') === 'bold';
}
return false;
};
/**
* Make the selected element bold or normal.
* @function module:svgcanvas.SvgCanvas#setBold
* @param {boolean} b - Indicates bold (`true`) or normal (`false`)
* @returns {void}
*/
this.setBold = function (b) {
var selected = selectedElements[0];
if (!isNullish(selected) && selected.tagName === 'text' && isNullish(selectedElements[1])) {
changeSelectedAttribute('font-weight', b ? 'bold' : 'normal');
}
if (!selectedElements[0].textContent) {
textActions.setCursor();
}
};
/**
* Check whether selected element is in italics or not.
* @function module:svgcanvas.SvgCanvas#getItalic
* @returns {boolean} Indicates whether or not element is italic
*/
this.getItalic = function () {
var selected = selectedElements[0];
if (!isNullish(selected) && selected.tagName === 'text' && isNullish(selectedElements[1])) {
return selected.getAttribute('font-style') === 'italic';
}
return false;
};
/**
* Make the selected element italic or normal.
* @function module:svgcanvas.SvgCanvas#setItalic
* @param {boolean} i - Indicates italic (`true`) or normal (`false`)
* @returns {void}
*/
this.setItalic = function (i) {
var selected = selectedElements[0];
if (!isNullish(selected) && selected.tagName === 'text' && isNullish(selectedElements[1])) {
changeSelectedAttribute('font-style', i ? 'italic' : 'normal');
}
if (!selectedElements[0].textContent) {
textActions.setCursor();
}
};
/**
* @function module:svgcanvas.SvgCanvas#getFontFamily
* @returns {string} The current font family
*/
this.getFontFamily = function () {
return curText.font_family;
};
/**
* Set the new font family.
* @function module:svgcanvas.SvgCanvas#setFontFamily
* @param {string} val - String with the new font family
* @returns {void}
*/
this.setFontFamily = function (val) {
curText.font_family = val;
changeSelectedAttribute('font-family', val);
if (selectedElements[0] && !selectedElements[0].textContent) {
textActions.setCursor();
}
};
/**
* Set the new font color.
* @function module:svgcanvas.SvgCanvas#setFontColor
* @param {string} val - String with the new font color
* @returns {void}
*/
this.setFontColor = function (val) {
curText.fill = val;
changeSelectedAttribute('fill', val);
};
/**
* @function module:svgcanvas.SvgCanvas#getFontColor
* @returns {string} The current font color
*/
this.getFontColor = function () {
return curText.fill;
};
/**
* @function module:svgcanvas.SvgCanvas#getFontSize
* @returns {Float} The current font size
*/
this.getFontSize = function () {
return curText.font_size;
};
/**
* Applies the given font size to the selected element.
* @function module:svgcanvas.SvgCanvas#setFontSize
* @param {Float} val - Float with the new font size
* @returns {void}
*/
this.setFontSize = function (val) {
curText.font_size = val;
changeSelectedAttribute('font-size', val);
if (!selectedElements[0].textContent) {
textActions.setCursor();
}
};
/**
* @function module:svgcanvas.SvgCanvas#getText
* @returns {string} The current text (`textContent`) of the selected element
*/
this.getText = function () {
var selected = selectedElements[0];
if (isNullish(selected)) {
return '';
}
return selected.textContent;
};
/**
* Updates the text element with the given string.
* @function module:svgcanvas.SvgCanvas#setTextContent
* @param {string} val - String with the new text
* @returns {void}
*/
this.setTextContent = function (val) {
changeSelectedAttribute('#text', val);
textActions.init(val);
textActions.setCursor();
};
/**
* Sets the new image URL for the selected image element. Updates its size if
* a new URL is given.
* @function module:svgcanvas.SvgCanvas#setImageURL
* @param {string} val - String with the image URL/path
* @fires module:svgcanvas.SvgCanvas#event:changed
* @returns {void}
*/
this.setImageURL = function (val) {
var elem = selectedElements[0];
if (!elem) {
return;
}
var attrs = $$8(elem).attr(['width', 'height']);
var setsize = !attrs.width || !attrs.height;
var curHref = getHref(elem); // Do nothing if no URL change or size change
if (curHref === val && !setsize) {
return;
}
var batchCmd = new BatchCommand$1('Change Image URL');
setHref(elem, val);
batchCmd.addSubCommand(new ChangeElementCommand$1(elem, {
'#href': curHref
}));
$$8(new Image()).load(function () {
var changes = $$8(elem).attr(['width', 'height']);
$$8(elem).attr({
width: this.width,
height: this.height
});
selectorManager.requestSelector(elem).resize();
batchCmd.addSubCommand(new ChangeElementCommand$1(elem, changes));
addCommandToHistory(batchCmd);
call('changed', [elem]);
}).attr('src', val);
};
/**
* Sets the new link URL for the selected anchor element.
* @function module:svgcanvas.SvgCanvas#setLinkURL
* @param {string} val - String with the link URL/path
* @returns {void}
*/
this.setLinkURL = function (val) {
var elem = selectedElements[0];
if (!elem) {
return;
}
if (elem.tagName !== 'a') {
// See if parent is an anchor
var parentsA = $$8(elem).parents('a');
if (parentsA.length) {
elem = parentsA[0];
} else {
return;
}
}
var curHref = getHref(elem);
if (curHref === val) {
return;
}
var batchCmd = new BatchCommand$1('Change Link URL');
setHref(elem, val);
batchCmd.addSubCommand(new ChangeElementCommand$1(elem, {
'#href': curHref
}));
addCommandToHistory(batchCmd);
};
/**
* Sets the `rx` and `ry` values to the selected `rect` element
* to change its corner radius.
* @function module:svgcanvas.SvgCanvas#setRectRadius
* @param {string|Float} val - The new radius
* @fires module:svgcanvas.SvgCanvas#event:changed
* @returns {void}
*/
this.setRectRadius = function (val) {
var selected = selectedElements[0];
if (!isNullish(selected) && selected.tagName === 'rect') {
var r = selected.getAttribute('rx');
if (r !== String(val)) {
selected.setAttribute('rx', val);
selected.setAttribute('ry', val);
addCommandToHistory(new ChangeElementCommand$1(selected, {
rx: r,
ry: r
}, 'Radius'));
call('changed', [selected]);
}
}
};
/**
* Wraps the selected element(s) in an anchor element or converts group to one.
* @function module:svgcanvas.SvgCanvas#makeHyperlink
* @param {string} url
* @returns {void}
*/
this.makeHyperlink = function (url) {
canvas.groupSelectedElements('a', url); // TODO: If element is a single "g", convert to "a"
// if (selectedElements.length > 1 && selectedElements[1]) {
};
/**
* @function module:svgcanvas.SvgCanvas#removeHyperlink
* @returns {void}
*/
this.removeHyperlink = function () {
canvas.ungroupSelectedElement();
};
/**
* Group: Element manipulation.
*/
/**
* Sets the new segment type to the selected segment(s).
* @function module:svgcanvas.SvgCanvas#setSegType
* @param {Integer} newType - New segment type. See {@link https://www.w3.org/TR/SVG/paths.html#InterfaceSVGPathSeg} for list
* @returns {void}
*/
this.setSegType = function (newType) {
pathActions$1.setSegType(newType);
};
/**
* Convert selected element to a path, or get the BBox of an element-as-path.
* @function module:svgcanvas.SvgCanvas#convertToPath
* @todo (codedread): Remove the getBBox argument and split this function into two.
* @param {Element} elem - The DOM element to be converted
* @param {boolean} getBBox - Boolean on whether or not to only return the path's BBox
* @returns {void|DOMRect|false|SVGPathElement|null} If the getBBox flag is true, the resulting path's bounding box object.
* Otherwise the resulting path element is returned.
*/
this.convertToPath = function (elem, getBBox) {
if (isNullish(elem)) {
var elems = selectedElements;
$$8.each(elems, function (i, el) {
if (el) {
canvas.convertToPath(el);
}
});
return undefined;
}
if (getBBox) {
return getBBoxOfElementAsPath(elem, addSVGElementFromJson, pathActions$1);
} // TODO: Why is this applying attributes from curShape, then inside utilities.convertToPath it's pulling addition attributes from elem?
// TODO: If convertToPath is called with one elem, curShape and elem are probably the same; but calling with multiple is a bug or cool feature.
var attrs = {
fill: curShape.fill,
'fill-opacity': curShape.fill_opacity,
stroke: curShape.stroke,
'stroke-width': curShape.stroke_width,
'stroke-dasharray': curShape.stroke_dasharray,
'stroke-linejoin': curShape.stroke_linejoin,
'stroke-linecap': curShape.stroke_linecap,
'stroke-opacity': curShape.stroke_opacity,
opacity: curShape.opacity,
visibility: 'hidden'
};
return convertToPath(elem, attrs, addSVGElementFromJson, pathActions$1, clearSelection, addToSelection, hstry, addCommandToHistory);
};
/**
* This function makes the changes to the elements. It does not add the change
* to the history stack.
* @param {string} attr - Attribute name
* @param {string|Float} newValue - String or number with the new attribute value
* @param {Element[]} elems - The DOM elements to apply the change to
* @returns {void}
*/
var changeSelectedAttributeNoUndo = function changeSelectedAttributeNoUndo(attr, newValue, elems) {
if (currentMode === 'pathedit') {
// Editing node
pathActions$1.moveNode(attr, newValue);
}
elems = elems || selectedElements;
var i = elems.length;
var noXYElems = ['g', 'polyline', 'path']; // const goodGAttrs = ['transform', 'opacity', 'filter'];
var _loop = function _loop() {
var elem = elems[i];
if (isNullish(elem)) {
return "continue";
} // Set x,y vals on elements that don't have them
if ((attr === 'x' || attr === 'y') && noXYElems.includes(elem.tagName)) {
var bbox = getStrokedBBoxDefaultVisible([elem]);
var diffX = attr === 'x' ? newValue - bbox.x : 0;
var diffY = attr === 'y' ? newValue - bbox.y : 0;
canvas.moveSelectedElements(diffX * currentZoom, diffY * currentZoom, true);
return "continue";
} // only allow the transform/opacity/filter attribute to change on <g> elements, slightly hacky
// TODO: Missing statement body
// if (elem.tagName === 'g' && goodGAttrs.includes(attr)) {}
var oldval = attr === '#text' ? elem.textContent : elem.getAttribute(attr);
if (isNullish(oldval)) {
oldval = '';
}
if (oldval !== String(newValue)) {
if (attr === '#text') {
// const oldW = utilsGetBBox(elem).width;
elem.textContent = newValue; // FF bug occurs on on rotated elements
if (/rotate/.test(elem.getAttribute('transform'))) {
elem = ffClone(elem);
} // Hoped to solve the issue of moving text with text-anchor="start",
// but this doesn't actually fix it. Hopefully on the right track, though. -Fyrd
// const box = getBBox(elem), left = box.x, top = box.y, {width, height} = box,
// dx = width - oldW, dy = 0;
// const angle = getRotationAngle(elem, true);
// if (angle) {
// const r = Math.sqrt(dx * dx + dy * dy);
// const theta = Math.atan2(dy, dx) - angle;
// dx = r * Math.cos(theta);
// dy = r * Math.sin(theta);
//
// elem.setAttribute('x', elem.getAttribute('x') - dx);
// elem.setAttribute('y', elem.getAttribute('y') - dy);
// }
} else if (attr === '#href') {
setHref(elem, newValue);
} else {
elem.setAttribute(attr, newValue);
} // Go into "select" mode for text changes
// NOTE: Important that this happens AFTER elem.setAttribute() or else attributes like
// font-size can get reset to their old value, ultimately by svgEditor.updateContextPanel(),
// after calling textActions.toSelectMode() below
if (currentMode === 'textedit' && attr !== '#text' && elem.textContent.length) {
textActions.toSelectMode(elem);
} // if (i === 0) {
// selectedBBoxes[0] = utilsGetBBox(elem);
// }
// Use the Firefox ffClone hack for text elements with gradients or
// where other text attributes are changed.
if (isGecko() && elem.nodeName === 'text' && /rotate/.test(elem.getAttribute('transform'))) {
if (String(newValue).startsWith('url') || ['font-size', 'font-family', 'x', 'y'].includes(attr) && elem.textContent) {
elem = ffClone(elem);
}
} // Timeout needed for Opera & Firefox
// codedread: it is now possible for this function to be called with elements
// that are not in the selectedElements array, we need to only request a
// selector if the element is in that array
if (selectedElements.includes(elem)) {
setTimeout(function () {
// Due to element replacement, this element may no longer
// be part of the DOM
if (!elem.parentNode) {
return;
}
selectorManager.requestSelector(elem).resize();
}, 0);
} // if this element was rotated, and we changed the position of this element
// we need to update the rotational transform attribute
var angle = getRotationAngle(elem);
if (angle !== 0 && attr !== 'transform') {
var tlist = getTransformList(elem);
var n = tlist.numberOfItems;
while (n--) {
var xform = tlist.getItem(n);
if (xform.type === 4) {
// remove old rotate
tlist.removeItem(n);
var box = getBBox(elem);
var center = transformPoint(box.x + box.width / 2, box.y + box.height / 2, transformListToTransform(tlist).matrix);
var cx = center.x,
cy = center.y;
var newrot = svgroot.createSVGTransform();
newrot.setRotate(angle, cx, cy);
tlist.insertItemBefore(newrot, n);
break;
}
}
}
} // if oldValue != newValue
};
while (i--) {
var _ret2 = _loop();
if (_ret2 === "continue") continue;
} // for each elem
};
/**
* Change the given/selected element and add the original value to the history stack.
* If you want to change all `selectedElements`, ignore the `elems` argument.
* If you want to change only a subset of `selectedElements`, then send the
* subset to this function in the `elems` argument.
* @function module:svgcanvas.SvgCanvas#changeSelectedAttribute
* @param {string} attr - String with the attribute name
* @param {string|Float} val - String or number with the new attribute value
* @param {Element[]} elems - The DOM elements to apply the change to
* @returns {void}
*/
var changeSelectedAttribute = this.changeSelectedAttribute = function (attr, val, elems) {
elems = elems || selectedElements;
canvas.undoMgr.beginUndoableChange(attr, elems); // const i = elems.length;
changeSelectedAttributeNoUndo(attr, val, elems);
var batchCmd = canvas.undoMgr.finishUndoableChange();
if (!batchCmd.isEmpty()) {
addCommandToHistory(batchCmd);
}
};
/**
* Removes all selected elements from the DOM and adds the change to the
* history stack.
* @function module:svgcanvas.SvgCanvas#deleteSelectedElements
* @fires module:svgcanvas.SvgCanvas#event:changed
* @returns {void}
*/
this.deleteSelectedElements = function () {
var batchCmd = new BatchCommand$1('Delete Elements');
var len = selectedElements.length;
var selectedCopy = []; // selectedElements is being deleted
for (var i = 0; i < len; ++i) {
var selected = selectedElements[i];
if (isNullish(selected)) {
break;
}
var parent = selected.parentNode;
var t = selected; // this will unselect the element and remove the selectedOutline
selectorManager.releaseSelector(t); // Remove the path if present.
removePath_(t.id); // Get the parent if it's a single-child anchor
if (parent.tagName === 'a' && parent.childNodes.length === 1) {
t = parent;
parent = parent.parentNode;
}
var _t = t,
nextSibling = _t.nextSibling;
t.remove();
var elem = t;
selectedCopy.push(selected); // for the copy
batchCmd.addSubCommand(new RemoveElementCommand$1(elem, nextSibling, parent));
}
selectedElements = [];
if (!batchCmd.isEmpty()) {
addCommandToHistory(batchCmd);
}
call('changed', selectedCopy);
clearSelection();
};
/**
* Removes all selected elements from the DOM and adds the change to the
* history stack. Remembers removed elements on the clipboard.
* @function module:svgcanvas.SvgCanvas#cutSelectedElements
* @returns {void}
*/
this.cutSelectedElements = function () {
canvas.copySelectedElements();
canvas.deleteSelectedElements();
};
var CLIPBOARD_ID = 'svgedit_clipboard';
/**
* Flash the clipboard data momentarily on localStorage so all tabs can see.
* @returns {void}
*/
function flashStorage() {
var data = sessionStorage.getItem(CLIPBOARD_ID);
localStorage.setItem(CLIPBOARD_ID, data);
setTimeout(function () {
localStorage.removeItem(CLIPBOARD_ID);
}, 1);
}
/**
* Transfers sessionStorage from one tab to another.
* @param {!Event} ev Storage event.
* @returns {void}
*/
function storageChange(ev) {
if (!ev.newValue) return; // This is a call from removeItem.
if (ev.key === CLIPBOARD_ID + '_startup') {
// Another tab asked for our sessionStorage.
localStorage.removeItem(CLIPBOARD_ID + '_startup');
flashStorage();
} else if (ev.key === CLIPBOARD_ID) {
// Another tab sent data.
sessionStorage.setItem(CLIPBOARD_ID, ev.newValue);
}
} // Listen for changes to localStorage.
window.addEventListener('storage', storageChange, false); // Ask other tabs for sessionStorage (this is ONLY to trigger event).
localStorage.setItem(CLIPBOARD_ID + '_startup', Math.random());
/**
* Remembers the current selected elements on the clipboard.
* @function module:svgcanvas.SvgCanvas#copySelectedElements
* @returns {void}
*/
this.copySelectedElements = function () {
var data = JSON.stringify(selectedElements.map(function (x) {
return getJsonFromSvgElement(x);
})); // Use sessionStorage for the clipboard data.
sessionStorage.setItem(CLIPBOARD_ID, data);
flashStorage();
var menu = $$8('#cmenu_canvas'); // Context menu might not exist (it is provided by editor.js).
if (menu.enableContextMenuItems) {
menu.enableContextMenuItems('#paste,#paste_in_place');
}
};
/**
* @function module:svgcanvas.SvgCanvas#pasteElements
* @param {"in_place"|"point"|void} type
* @param {Integer|void} x Expected if type is "point"
* @param {Integer|void} y Expected if type is "point"
* @fires module:svgcanvas.SvgCanvas#event:changed
* @fires module:svgcanvas.SvgCanvas#event:ext_IDsUpdated
* @returns {void}
*/
this.pasteElements = function (type, x, y) {
var clipb = JSON.parse(sessionStorage.getItem(CLIPBOARD_ID));
if (!clipb) return;
var len = clipb.length;
if (!len) return;
var pasted = [];
var batchCmd = new BatchCommand$1('Paste elements'); // const drawing = getCurrentDrawing();
/**
* @typedef {PlainObject<string, string>} module:svgcanvas.ChangedIDs
*/
/**
* @type {module:svgcanvas.ChangedIDs}
*/
var changedIDs = {}; // Recursively replace IDs and record the changes
/**
*
* @param {module:svgcanvas.SVGAsJSON} elem
* @returns {void}
*/
function checkIDs(elem) {
if (elem.attr && elem.attr.id) {
changedIDs[elem.attr.id] = getNextId();
elem.attr.id = changedIDs[elem.attr.id];
}
if (elem.children) elem.children.forEach(checkIDs);
}
clipb.forEach(checkIDs); // Give extensions like the connector extension a chance to reflect new IDs and remove invalid elements
/**
* Triggered when `pasteElements` is called from a paste action (context menu or key).
* @event module:svgcanvas.SvgCanvas#event:ext_IDsUpdated
* @type {PlainObject}
* @property {module:svgcanvas.SVGAsJSON[]} elems
* @property {module:svgcanvas.ChangedIDs} changes Maps past ID (on attribute) to current ID
*/
runExtensions('IDsUpdated',
/** @type {module:svgcanvas.SvgCanvas#event:ext_IDsUpdated} */
{
elems: clipb,
changes: changedIDs
}, true).forEach(function (extChanges) {
if (!extChanges || !('remove' in extChanges)) return;
extChanges.remove.forEach(function (removeID) {
clipb = clipb.filter(function (clipBoardItem) {
return clipBoardItem.attr.id !== removeID;
});
});
}); // Move elements to lastClickPoint
while (len--) {
var elem = clipb[len];
if (!elem) {
continue;
}
var copy = addSVGElementFromJson(elem);
pasted.push(copy);
batchCmd.addSubCommand(new InsertElementCommand$1(copy));
restoreRefElems(copy);
}
selectOnly(pasted);
if (type !== 'in_place') {
var ctrX, ctrY;
if (!type) {
ctrX = lastClickPoint.x;
ctrY = lastClickPoint.y;
} else if (type === 'point') {
ctrX = x;
ctrY = y;
}
var bbox = getStrokedBBoxDefaultVisible(pasted);
var cx = ctrX - (bbox.x + bbox.width / 2),
cy = ctrY - (bbox.y + bbox.height / 2),
dx = [],
dy = [];
$$8.each(pasted, function (i, item) {
dx.push(cx);
dy.push(cy);
});
var cmd = canvas.moveSelectedElements(dx, dy, false);
if (cmd) batchCmd.addSubCommand(cmd);
}
addCommandToHistory(batchCmd);
call('changed', pasted);
};
/**
* Wraps all the selected elements in a group (`g`) element.
* @function module:svgcanvas.SvgCanvas#groupSelectedElements
* @param {"a"|"g"} [type="g"] - type of element to group into, defaults to `<g>`
* @param {string} [urlArg]
* @returns {void}
*/
this.groupSelectedElements = function (type, urlArg) {
if (!type) {
type = 'g';
}
var cmdStr = '';
var url;
switch (type) {
case 'a':
{
cmdStr = 'Make hyperlink';
url = urlArg || '';
break;
}
default:
{
type = 'g';
cmdStr = 'Group Elements';
break;
}
}
var batchCmd = new BatchCommand$1(cmdStr); // create and insert the group element
var g = addSVGElementFromJson({
element: type,
attr: {
id: getNextId()
}
});
if (type === 'a') {
setHref(g, url);
}
batchCmd.addSubCommand(new InsertElementCommand$1(g)); // now move all children into the group
var i = selectedElements.length;
while (i--) {
var elem = selectedElements[i];
if (isNullish(elem)) {
continue;
}
if (elem.parentNode.tagName === 'a' && elem.parentNode.childNodes.length === 1) {
elem = elem.parentNode;
}
var oldNextSibling = elem.nextSibling;
var oldParent = elem.parentNode;
g.append(elem);
batchCmd.addSubCommand(new MoveElementCommand$1(elem, oldNextSibling, oldParent));
}
if (!batchCmd.isEmpty()) {
addCommandToHistory(batchCmd);
} // update selection
selectOnly([g], true);
};
/**
* Pushes all appropriate parent group properties down to its children, then
* removes them from the group.
* @function module:svgcanvas.SvgCanvas#pushGroupProperties
* @param {SVGAElement|SVGGElement} g
* @param {boolean} undoable
* @returns {BatchCommand|void}
*/
var pushGroupProperties = this.pushGroupProperties = function (g, undoable) {
var children = g.childNodes;
var len = children.length;
var xform = g.getAttribute('transform');
var glist = getTransformList(g);
var m = transformListToTransform(glist).matrix;
var batchCmd = new BatchCommand$1('Push group properties'); // TODO: get all fill/stroke properties from the group that we are about to destroy
// "fill", "fill-opacity", "fill-rule", "stroke", "stroke-dasharray", "stroke-dashoffset",
// "stroke-linecap", "stroke-linejoin", "stroke-miterlimit", "stroke-opacity",
// "stroke-width"
// and then for each child, if they do not have the attribute (or the value is 'inherit')
// then set the child's attribute
var gangle = getRotationAngle(g);
var gattrs = $$8(g).attr(['filter', 'opacity']);
var gfilter, gblur, changes;
var drawing = getCurrentDrawing();
for (var i = 0; i < len; i++) {
var elem = children[i];
if (elem.nodeType !== 1) {
continue;
}
if (gattrs.opacity !== null && gattrs.opacity !== 1) {
// const c_opac = elem.getAttribute('opacity') || 1;
var newOpac = Math.round((elem.getAttribute('opacity') || 1) * gattrs.opacity * 100) / 100;
changeSelectedAttribute('opacity', newOpac, [elem]);
}
if (gattrs.filter) {
var cblur = this.getBlur(elem);
var origCblur = cblur;
if (!gblur) {
gblur = this.getBlur(g);
}
if (cblur) {
// Is this formula correct?
cblur = Number(gblur) + Number(cblur);
} else if (cblur === 0) {
cblur = gblur;
} // If child has no current filter, get group's filter or clone it.
if (!origCblur) {
// Set group's filter to use first child's ID
if (!gfilter) {
gfilter = getRefElem(gattrs.filter);
} else {
// Clone the group's filter
gfilter = drawing.copyElem(gfilter);
findDefs().append(gfilter);
}
} else {
gfilter = getRefElem(elem.getAttribute('filter'));
} // Change this in future for different filters
var suffix = gfilter.firstChild.tagName === 'feGaussianBlur' ? 'blur' : 'filter';
gfilter.id = elem.id + '_' + suffix;
changeSelectedAttribute('filter', 'url(#' + gfilter.id + ')', [elem]); // Update blur value
if (cblur) {
changeSelectedAttribute('stdDeviation', cblur, [gfilter.firstChild]);
canvas.setBlurOffsets(gfilter, cblur);
}
}
var chtlist = getTransformList(elem); // Don't process gradient transforms
if (elem.tagName.includes('Gradient')) {
chtlist = null;
} // Hopefully not a problem to add this. Necessary for elements like <desc/>
if (!chtlist) {
continue;
} // Apparently <defs> can get get a transformlist, but we don't want it to have one!
if (elem.tagName === 'defs') {
continue;
}
if (glist.numberOfItems) {
// TODO: if the group's transform is just a rotate, we can always transfer the
// rotate() down to the children (collapsing consecutive rotates and factoring
// out any translates)
if (gangle && glist.numberOfItems === 1) {
// [Rg] [Rc] [Mc]
// we want [Tr] [Rc2] [Mc] where:
// - [Rc2] is at the child's current center but has the
// sum of the group and child's rotation angles
// - [Tr] is the equivalent translation that this child
// undergoes if the group wasn't there
// [Tr] = [Rg] [Rc] [Rc2_inv]
// get group's rotation matrix (Rg)
var rgm = glist.getItem(0).matrix; // get child's rotation matrix (Rc)
var rcm = svgroot.createSVGMatrix();
var cangle = getRotationAngle(elem);
if (cangle) {
rcm = chtlist.getItem(0).matrix;
} // get child's old center of rotation
var cbox = getBBox(elem);
var ceqm = transformListToTransform(chtlist).matrix;
var coldc = transformPoint(cbox.x + cbox.width / 2, cbox.y + cbox.height / 2, ceqm); // sum group and child's angles
var sangle = gangle + cangle; // get child's rotation at the old center (Rc2_inv)
var r2 = svgroot.createSVGTransform();
r2.setRotate(sangle, coldc.x, coldc.y); // calculate equivalent translate
var trm = matrixMultiply(rgm, rcm, r2.matrix.inverse()); // set up tlist
if (cangle) {
chtlist.removeItem(0);
}
if (sangle) {
if (chtlist.numberOfItems) {
chtlist.insertItemBefore(r2, 0);
} else {
chtlist.appendItem(r2);
}
}
if (trm.e || trm.f) {
var tr = svgroot.createSVGTransform();
tr.setTranslate(trm.e, trm.f);
if (chtlist.numberOfItems) {
chtlist.insertItemBefore(tr, 0);
} else {
chtlist.appendItem(tr);
}
}
} else {
// more complicated than just a rotate
// transfer the group's transform down to each child and then
// call recalculateDimensions()
var oldxform = elem.getAttribute('transform');
changes = {};
changes.transform = oldxform || '';
var newxform = svgroot.createSVGTransform(); // [ gm ] [ chm ] = [ chm ] [ gm' ]
// [ gm' ] = [ chmInv ] [ gm ] [ chm ]
var chm = transformListToTransform(chtlist).matrix,
chmInv = chm.inverse();
var gm = matrixMultiply(chmInv, m, chm);
newxform.setMatrix(gm);
chtlist.appendItem(newxform);
}
var cmd = recalculateDimensions(elem);
if (cmd) {
batchCmd.addSubCommand(cmd);
}
}
} // remove transform and make it undo-able
if (xform) {
changes = {};
changes.transform = xform;
g.setAttribute('transform', '');
g.removeAttribute('transform');
batchCmd.addSubCommand(new ChangeElementCommand$1(g, changes));
}
if (undoable && !batchCmd.isEmpty()) {
return batchCmd;
}
return undefined;
};
/**
* Unwraps all the elements in a selected group (`g`) element. This requires
* significant recalculations to apply group's transforms, etc. to its children.
* @function module:svgcanvas.SvgCanvas#ungroupSelectedElement
* @returns {void}
*/
this.ungroupSelectedElement = function () {
var g = selectedElements[0];
if (!g) {
return;
}
if ($$8(g).data('gsvg') || $$8(g).data('symbol')) {
// Is svg, so actually convert to group
convertToGroup(g);
return;
}
if (g.tagName === 'use') {
// Somehow doesn't have data set, so retrieve
var symbol = getElem(getHref(g).substr(1));
$$8(g).data('symbol', symbol).data('ref', symbol);
convertToGroup(g);
return;
}
var parentsA = $$8(g).parents('a');
if (parentsA.length) {
g = parentsA[0];
} // Look for parent "a"
if (g.tagName === 'g' || g.tagName === 'a') {
var batchCmd = new BatchCommand$1('Ungroup Elements');
var cmd = pushGroupProperties(g, true);
if (cmd) {
batchCmd.addSubCommand(cmd);
}
var parent = g.parentNode;
var anchor = g.nextSibling;
var children = new Array(g.childNodes.length);
var i = 0;
while (g.firstChild) {
var elem = g.firstChild;
var oldNextSibling = elem.nextSibling;
var oldParent = elem.parentNode; // Remove child title elements
if (elem.tagName === 'title') {
var nextSibling = elem.nextSibling;
batchCmd.addSubCommand(new RemoveElementCommand$1(elem, nextSibling, oldParent));
elem.remove();
continue;
}
if (anchor) {
anchor.before(elem);
} else {
g.after(elem);
}
children[i++] = elem;
batchCmd.addSubCommand(new MoveElementCommand$1(elem, oldNextSibling, oldParent));
} // remove the group from the selection
clearSelection(); // delete the group element (but make undo-able)
var gNextSibling = g.nextSibling;
g.remove();
batchCmd.addSubCommand(new RemoveElementCommand$1(g, gNextSibling, parent));
if (!batchCmd.isEmpty()) {
addCommandToHistory(batchCmd);
} // update selection
addToSelection(children);
}
};
/**
* Repositions the selected element to the bottom in the DOM to appear on top of
* other elements.
* @function module:svgcanvas.SvgCanvas#moveToTopSelectedElement
* @fires module:svgcanvas.SvgCanvas#event:changed
* @returns {void}
*/
this.moveToTopSelectedElement = function () {
var _selectedElements = selectedElements,
_selectedElements2 = _slicedToArray(_selectedElements, 1),
selected = _selectedElements2[0];
if (!isNullish(selected)) {
var t = selected;
var oldParent = t.parentNode;
var oldNextSibling = t.nextSibling;
t = t.parentNode.appendChild(t); // If the element actually moved position, add the command and fire the changed
// event handler.
if (oldNextSibling !== t.nextSibling) {
addCommandToHistory(new MoveElementCommand$1(t, oldNextSibling, oldParent, 'top'));
call('changed', [t]);
}
}
};
/**
* Repositions the selected element to the top in the DOM to appear under
* other elements.
* @function module:svgcanvas.SvgCanvas#moveToBottomSelectedElement
* @fires module:svgcanvas.SvgCanvas#event:changed
* @returns {void}
*/
this.moveToBottomSelectedElement = function () {
var _selectedElements3 = selectedElements,
_selectedElements4 = _slicedToArray(_selectedElements3, 1),
selected = _selectedElements4[0];
if (!isNullish(selected)) {
var t = selected;
var oldParent = t.parentNode;
var oldNextSibling = t.nextSibling;
var firstChild = t.parentNode.firstChild;
if (firstChild.tagName === 'title') {
firstChild = firstChild.nextSibling;
} // This can probably be removed, as the defs should not ever apppear
// inside a layer group
if (firstChild.tagName === 'defs') {
firstChild = firstChild.nextSibling;
}
t = t.parentNode.insertBefore(t, firstChild); // If the element actually moved position, add the command and fire the changed
// event handler.
if (oldNextSibling !== t.nextSibling) {
addCommandToHistory(new MoveElementCommand$1(t, oldNextSibling, oldParent, 'bottom'));
call('changed', [t]);
}
}
};
/**
* Moves the select element up or down the stack, based on the visibly
* intersecting elements.
* @function module:svgcanvas.SvgCanvas#moveUpDownSelected
* @param {"Up"|"Down"} dir - String that's either 'Up' or 'Down'
* @fires module:svgcanvas.SvgCanvas#event:changed
* @returns {void}
*/
this.moveUpDownSelected = function (dir) {
var selected = selectedElements[0];
if (!selected) {
return;
}
curBBoxes = [];
var closest, foundCur; // jQuery sorts this list
var list = $$8(getIntersectionList(getStrokedBBoxDefaultVisible([selected]))).toArray();
if (dir === 'Down') {
list.reverse();
}
$$8.each(list, function () {
if (!foundCur) {
if (this === selected) {
foundCur = true;
}
return true;
}
closest = this; // eslint-disable-line consistent-this
return false;
});
if (!closest) {
return;
}
var t = selected;
var oldParent = t.parentNode;
var oldNextSibling = t.nextSibling;
$$8(closest)[dir === 'Down' ? 'before' : 'after'](t); // If the element actually moved position, add the command and fire the changed
// event handler.
if (oldNextSibling !== t.nextSibling) {
addCommandToHistory(new MoveElementCommand$1(t, oldNextSibling, oldParent, 'Move ' + dir));
call('changed', [t]);
}
};
/**
* Moves selected elements on the X/Y axis.
* @function module:svgcanvas.SvgCanvas#moveSelectedElements
* @param {Float} dx - Float with the distance to move on the x-axis
* @param {Float} dy - Float with the distance to move on the y-axis
* @param {boolean} undoable - Boolean indicating whether or not the action should be undoable
* @fires module:svgcanvas.SvgCanvas#event:changed
* @returns {BatchCommand|void} Batch command for the move
*/
this.moveSelectedElements = function (dx, dy, undoable) {
// if undoable is not sent, default to true
// if single values, scale them to the zoom
if (dx.constructor !== Array) {
dx /= currentZoom;
dy /= currentZoom;
}
undoable = undoable || true;
var batchCmd = new BatchCommand$1('position');
var i = selectedElements.length;
while (i--) {
var selected = selectedElements[i];
if (!isNullish(selected)) {
// if (i === 0) {
// selectedBBoxes[0] = utilsGetBBox(selected);
// }
// const b = {};
// for (const j in selectedBBoxes[i]) b[j] = selectedBBoxes[i][j];
// selectedBBoxes[i] = b;
var xform = svgroot.createSVGTransform();
var tlist = getTransformList(selected); // dx and dy could be arrays
if (dx.constructor === Array) {
// if (i === 0) {
// selectedBBoxes[0].x += dx[0];
// selectedBBoxes[0].y += dy[0];
// }
xform.setTranslate(dx[i], dy[i]);
} else {
// if (i === 0) {
// selectedBBoxes[0].x += dx;
// selectedBBoxes[0].y += dy;
// }
xform.setTranslate(dx, dy);
}
if (tlist.numberOfItems) {
tlist.insertItemBefore(xform, 0);
} else {
tlist.appendItem(xform);
}
var cmd = recalculateDimensions(selected);
if (cmd) {
batchCmd.addSubCommand(cmd);
}
selectorManager.requestSelector(selected).resize();
}
}
if (!batchCmd.isEmpty()) {
if (undoable) {
addCommandToHistory(batchCmd);
}
call('changed', selectedElements);
return batchCmd;
}
return undefined;
};
/**
* Create deep DOM copies (clones) of all selected elements and move them slightly
* from their originals.
* @function module:svgcanvas.SvgCanvas#cloneSelectedElements
* @param {Float} x Float with the distance to move on the x-axis
* @param {Float} y Float with the distance to move on the y-axis
* @returns {void}
*/
this.cloneSelectedElements = function (x, y) {
var i, elem;
var batchCmd = new BatchCommand$1('Clone Elements'); // find all the elements selected (stop at first null)
var len = selectedElements.length;
/**
* Sorts an array numerically and ascending.
* @param {Element} a
* @param {Element} b
* @returns {Integer}
*/
function sortfunction(a, b) {
return $$8(b).index() - $$8(a).index();
}
selectedElements.sort(sortfunction);
for (i = 0; i < len; ++i) {
elem = selectedElements[i];
if (isNullish(elem)) {
break;
}
} // use slice to quickly get the subset of elements we need
var copiedElements = selectedElements.slice(0, i);
this.clearSelection(true); // note that we loop in the reverse way because of the way elements are added
// to the selectedElements array (top-first)
var drawing = getCurrentDrawing();
i = copiedElements.length;
while (i--) {
// clone each element and replace it within copiedElements
elem = copiedElements[i] = drawing.copyElem(copiedElements[i]);
(currentGroup || drawing.getCurrentLayer()).append(elem);
batchCmd.addSubCommand(new InsertElementCommand$1(elem));
}
if (!batchCmd.isEmpty()) {
addToSelection(copiedElements.reverse()); // Need to reverse for correct selection-adding
this.moveSelectedElements(x, y, false);
addCommandToHistory(batchCmd);
}
};
/**
* Aligns selected elements.
* @function module:svgcanvas.SvgCanvas#alignSelectedElements
* @param {string} type - String with single character indicating the alignment type
* @param {"selected"|"largest"|"smallest"|"page"} relativeTo
* @returns {void}
*/
this.alignSelectedElements = function (type, relativeTo) {
var bboxes = []; // angles = [];
var len = selectedElements.length;
if (!len) {
return;
}
var minx = Number.MAX_VALUE,
maxx = Number.MIN_VALUE,
miny = Number.MAX_VALUE,
maxy = Number.MIN_VALUE;
var curwidth = Number.MIN_VALUE,
curheight = Number.MIN_VALUE;
for (var i = 0; i < len; ++i) {
if (isNullish(selectedElements[i])) {
break;
}
var elem = selectedElements[i];
bboxes[i] = getStrokedBBoxDefaultVisible([elem]); // now bbox is axis-aligned and handles rotation
switch (relativeTo) {
case 'smallest':
if ((type === 'l' || type === 'c' || type === 'r') && (curwidth === Number.MIN_VALUE || curwidth > bboxes[i].width) || (type === 't' || type === 'm' || type === 'b') && (curheight === Number.MIN_VALUE || curheight > bboxes[i].height)) {
minx = bboxes[i].x;
miny = bboxes[i].y;
maxx = bboxes[i].x + bboxes[i].width;
maxy = bboxes[i].y + bboxes[i].height;
curwidth = bboxes[i].width;
curheight = bboxes[i].height;
}
break;
case 'largest':
if ((type === 'l' || type === 'c' || type === 'r') && (curwidth === Number.MIN_VALUE || curwidth < bboxes[i].width) || (type === 't' || type === 'm' || type === 'b') && (curheight === Number.MIN_VALUE || curheight < bboxes[i].height)) {
minx = bboxes[i].x;
miny = bboxes[i].y;
maxx = bboxes[i].x + bboxes[i].width;
maxy = bboxes[i].y + bboxes[i].height;
curwidth = bboxes[i].width;
curheight = bboxes[i].height;
}
break;
default:
// 'selected'
if (bboxes[i].x < minx) {
minx = bboxes[i].x;
}
if (bboxes[i].y < miny) {
miny = bboxes[i].y;
}
if (bboxes[i].x + bboxes[i].width > maxx) {
maxx = bboxes[i].x + bboxes[i].width;
}
if (bboxes[i].y + bboxes[i].height > maxy) {
maxy = bboxes[i].y + bboxes[i].height;
}
break;
}
} // loop for each element to find the bbox and adjust min/max
if (relativeTo === 'page') {
minx = 0;
miny = 0;
maxx = canvas.contentW;
maxy = canvas.contentH;
}
var dx = new Array(len);
var dy = new Array(len);
for (var _i6 = 0; _i6 < len; ++_i6) {
if (isNullish(selectedElements[_i6])) {
break;
} // const elem = selectedElements[i];
var bbox = bboxes[_i6];
dx[_i6] = 0;
dy[_i6] = 0;
switch (type) {
case 'l':
// left (horizontal)
dx[_i6] = minx - bbox.x;
break;
case 'c':
// center (horizontal)
dx[_i6] = (minx + maxx) / 2 - (bbox.x + bbox.width / 2);
break;
case 'r':
// right (horizontal)
dx[_i6] = maxx - (bbox.x + bbox.width);
break;
case 't':
// top (vertical)
dy[_i6] = miny - bbox.y;
break;
case 'm':
// middle (vertical)
dy[_i6] = (miny + maxy) / 2 - (bbox.y + bbox.height / 2);
break;
case 'b':
// bottom (vertical)
dy[_i6] = maxy - (bbox.y + bbox.height);
break;
}
}
this.moveSelectedElements(dx, dy);
};
/**
* Group: Additional editor tools.
*/
/**
* @name module:svgcanvas.SvgCanvas#contentW
* @type {Float}
*/
this.contentW = getResolution().w;
/**
* @name module:svgcanvas.SvgCanvas#contentH
* @type {Float}
*/
this.contentH = getResolution().h;
/**
* @typedef {PlainObject} module:svgcanvas.CanvasInfo
* @property {Float} x - The canvas' new x coordinate
* @property {Float} y - The canvas' new y coordinate
* @property {string} oldX - The canvas' old x coordinate
* @property {string} oldY - The canvas' old y coordinate
* @property {Float} d_x - The x position difference
* @property {Float} d_y - The y position difference
*/
/**
* Updates the editor canvas width/height/position after a zoom has occurred.
* @function module:svgcanvas.SvgCanvas#updateCanvas
* @param {Float} w - Float with the new width
* @param {Float} h - Float with the new height
* @fires module:svgcanvas.SvgCanvas#event:ext_canvasUpdated
* @returns {module:svgcanvas.CanvasInfo}
*/
this.updateCanvas = function (w, h) {
svgroot.setAttribute('width', w);
svgroot.setAttribute('height', h);
var bg = $$8('#canvasBackground')[0];
var oldX = svgcontent.getAttribute('x');
var oldY = svgcontent.getAttribute('y');
var x = (w - this.contentW * currentZoom) / 2;
var y = (h - this.contentH * currentZoom) / 2;
assignAttributes(svgcontent, {
width: this.contentW * currentZoom,
height: this.contentH * currentZoom,
x: x,
y: y,
viewBox: '0 0 ' + this.contentW + ' ' + this.contentH
});
assignAttributes(bg, {
width: svgcontent.getAttribute('width'),
height: svgcontent.getAttribute('height'),
x: x,
y: y
});
var bgImg = getElem('background_image');
if (bgImg) {
assignAttributes(bgImg, {
width: '100%',
height: '100%'
});
}
selectorManager.selectorParentGroup.setAttribute('transform', 'translate(' + x + ',' + y + ')');
/**
* Invoked upon updates to the canvas.
* @event module:svgcanvas.SvgCanvas#event:ext_canvasUpdated
* @type {PlainObject}
* @property {Integer} new_x
* @property {Integer} new_y
* @property {string} old_x (Of Integer)
* @property {string} old_y (Of Integer)
* @property {Integer} d_x
* @property {Integer} d_y
*/
runExtensions('canvasUpdated',
/**
* @type {module:svgcanvas.SvgCanvas#event:ext_canvasUpdated}
*/
{
new_x: x,
new_y: y,
old_x: oldX,
old_y: oldY,
d_x: x - oldX,
d_y: y - oldY
});
return {
x: x,
y: y,
old_x: oldX,
old_y: oldY,
d_x: x - oldX,
d_y: y - oldY
};
};
/**
* Set the background of the editor (NOT the actual document).
* @function module:svgcanvas.SvgCanvas#setBackground
* @param {string} color - String with fill color to apply
* @param {string} url - URL or path to image to use
* @returns {void}
*/
this.setBackground = function (color, url) {
var bg = getElem('canvasBackground');
var border = $$8(bg).find('rect')[0];
var bgImg = getElem('background_image');
2020-01-08 18:27:56 +00:00
var bgPattern = getElem('background_pattern');
border.setAttribute('fill', color === 'chessboard' ? '#fff' : color);
if (color === 'chessboard') {
if (!bgPattern) {
bgPattern = svgdoc.createElementNS(NS.SVG, 'foreignObject');
assignAttributes(bgPattern, {
id: 'background_pattern',
width: '100%',
height: '100%',
preserveAspectRatio: 'xMinYMin',
style: 'pointer-events:none'
});
var div = document.createElement('div');
assignAttributes(div, {
style: 'pointer-events:none;width:100%;height:100%;background-image:url(data:image/gif;base64,R0lGODlhEAAQAIAAAP///9bW1iH5BAAAAAAALAAAAAAQABAAAAIfjG+gq4jM3IFLJgpswNly/XkcBpIiVaInlLJr9FZWAQA7);'
});
bgPattern.appendChild(div);
bg.append(bgPattern);
}
} else if (bgPattern) {
bgPattern.remove();
}
if (url) {
if (!bgImg) {
bgImg = svgdoc.createElementNS(NS.SVG, 'image');
assignAttributes(bgImg, {
id: 'background_image',
width: '100%',
height: '100%',
preserveAspectRatio: 'xMinYMin',
style: 'pointer-events:none'
});
}
setHref(bgImg, url);
bg.append(bgImg);
} else if (bgImg) {
bgImg.remove();
}
};
/**
* Select the next/previous element within the current layer.
* @function module:svgcanvas.SvgCanvas#cycleElement
* @param {boolean} next - true = next and false = previous element
* @fires module:svgcanvas.SvgCanvas#event:selected
* @returns {void}
*/
this.cycleElement = function (next) {
var num;
var curElem = selectedElements[0];
var elem = false;
var allElems = getVisibleElements(currentGroup || getCurrentDrawing().getCurrentLayer());
if (!allElems.length) {
return;
}
if (isNullish(curElem)) {
num = next ? allElems.length - 1 : 0;
elem = allElems[num];
} else {
var i = allElems.length;
while (i--) {
if (allElems[i] === curElem) {
num = next ? i - 1 : i + 1;
if (num >= allElems.length) {
num = 0;
} else if (num < 0) {
num = allElems.length - 1;
}
elem = allElems[num];
break;
}
}
}
selectOnly([elem], true);
call('selected', selectedElements);
};
this.clear();
/**
* @interface module:svgcanvas.PrivateMethods
* @type {PlainObject}
* @property {module:svgcanvas~addCommandToHistory} addCommandToHistory
* @property {module:history.HistoryCommand} BatchCommand
* @property {module:history.HistoryCommand} ChangeElementCommand
* @property {module:utilities.decode64} decode64
* @property {module:utilities.dropXMLInternalSubset} dropXMLInternalSubset
* @property {module:utilities.encode64} encode64
* @property {module:svgcanvas~ffClone} ffClone
* @property {module:svgcanvas~findDuplicateGradient} findDuplicateGradient
* @property {module:utilities.getPathBBox} getPathBBox
* @property {module:units.getTypeMap} getTypeMap
* @property {module:draw.identifyLayers} identifyLayers
* @property {module:history.HistoryCommand} InsertElementCommand
* @property {module:browser.isChrome} isChrome
* @property {module:math.isIdentity} isIdentity
* @property {module:browser.isIE} isIE
* @property {module:svgcanvas~logMatrix} logMatrix
* @property {module:history.HistoryCommand} MoveElementCommand
* @property {module:namespaces.NS} NS
* @property {module:utilities.preventClickDefault} preventClickDefault
* @property {module:history.HistoryCommand} RemoveElementCommand
* @property {module:SVGTransformList.SVGEditTransformList} SVGEditTransformList
* @property {module:utilities.text2xml} text2xml
* @property {module:math.transformBox} transformBox
* @property {module:math.transformPoint} transformPoint
* @property {module:utilities.walkTree} walkTree
*/
/**
* @deprecated getPrivateMethods
* Since all methods are/should be public somehow, this function should be removed;
* we might require `import` in place of this in the future once ES6 Modules
* widespread
* Being able to access private methods publicly seems wrong somehow,
* but currently appears to be the best way to allow testing and provide
* access to them to plugins.
* @function module:svgcanvas.SvgCanvas#getPrivateMethods
* @returns {module:svgcanvas.PrivateMethods}
*/
this.getPrivateMethods = function () {
var obj = {
addCommandToHistory: addCommandToHistory,
BatchCommand: BatchCommand$1,
ChangeElementCommand: ChangeElementCommand$1,
decode64: decode64,
dropXMLInternalSubset: dropXMLInternalSubset,
encode64: encode64,
ffClone: ffClone,
findDefs: findDefs,
findDuplicateGradient: findDuplicateGradient,
getElem: getElem,
getPathBBox: getPathBBox,
getTypeMap: getTypeMap,
getUrlFromAttr: getUrlFromAttr,
identifyLayers: identifyLayers,
InsertElementCommand: InsertElementCommand$1,
isChrome: isChrome,
isIdentity: isIdentity,
isIE: isIE,
logMatrix: logMatrix,
MoveElementCommand: MoveElementCommand$1,
NS: NS,
preventClickDefault: preventClickDefault,
RemoveElementCommand: RemoveElementCommand$1,
SVGEditTransformList: SVGTransformList,
text2xml: text2xml,
transformBox: transformBox,
transformPoint: transformPoint,
walkTree: walkTree
};
return obj;
};
} // End constructor
; // End class
return SvgCanvas;
}());