1 /*
  2     Copyright 2008-2024
  3         Matthias Ehmann,
  4         Michael Gerhaeuser,
  5         Carsten Miller,
  6         Bianca Valentin,
  7         Alfred Wassermann,
  8         Peter Wilfahrt
  9 
 10     This file is part of JSXGraph.
 11 
 12     JSXGraph is free software dual licensed under the GNU LGPL or MIT License.
 13 
 14     You can redistribute it and/or modify it under the terms of the
 15 
 16       * GNU Lesser General Public License as published by
 17         the Free Software Foundation, either version 3 of the License, or
 18         (at your option) any later version
 19       OR
 20       * MIT License: https://github.com/jsxgraph/jsxgraph/blob/master/LICENSE.MIT
 21 
 22     JSXGraph is distributed in the hope that it will be useful,
 23     but WITHOUT ANY WARRANTY; without even the implied warranty of
 24     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 25     GNU Lesser General Public License for more details.
 26 
 27     You should have received a copy of the GNU Lesser General Public License and
 28     the MIT License along with JSXGraph. If not, see <https://www.gnu.org/licenses/>
 29     and <https://opensource.org/licenses/MIT/>.
 30  */
 31 
 32 /*global JXG: true, define: true, AMprocessNode: true, MathJax: true, document: true */
 33 /*jslint nomen: true, plusplus: true, newcap:true*/
 34 
 35 import JXG from "../jxg.js";
 36 import Options from "../options.js";
 37 import AbstractRenderer from "./abstract.js";
 38 import Const from "../base/constants.js";
 39 import Type from "../utils/type.js";
 40 import Color from "../utils/color.js";
 41 import Base64 from "../utils/base64.js";
 42 import Numerics from "../math/numerics.js";
 43 
 44 /**
 45  * Uses SVG to implement the rendering methods defined in {@link JXG.AbstractRenderer}.
 46  * @class JXG.SVGRenderer
 47  * @augments JXG.AbstractRenderer
 48  * @param {Node} container Reference to a DOM node containing the board.
 49  * @param {Object} dim The dimensions of the board
 50  * @param {Number} dim.width
 51  * @param {Number} dim.height
 52  * @see JXG.AbstractRenderer
 53  */
 54 JXG.SVGRenderer = function (container, dim) {
 55     var i;
 56 
 57     // docstring in AbstractRenderer
 58     this.type = "svg";
 59 
 60     this.isIE =
 61         navigator.appVersion.indexOf("MSIE") !== -1 || navigator.userAgent.match(/Trident\//);
 62 
 63     /**
 64      * SVG root node
 65      * @type Node
 66      */
 67     this.svgRoot = null;
 68 
 69     /**
 70      * The SVG Namespace used in JSXGraph.
 71      * @see http://www.w3.org/TR/SVG2/
 72      * @type String
 73      * @default http://www.w3.org/2000/svg
 74      */
 75     this.svgNamespace = "http://www.w3.org/2000/svg";
 76 
 77     /**
 78      * The xlink namespace. This is used for images.
 79      * @see http://www.w3.org/TR/xlink/
 80      * @type String
 81      * @default http://www.w3.org/1999/xlink
 82      */
 83     this.xlinkNamespace = "http://www.w3.org/1999/xlink";
 84 
 85     // container is documented in AbstractRenderer.
 86     // Type node
 87     this.container = container;
 88 
 89     // prepare the div container and the svg root node for use with JSXGraph
 90     this.container.style.MozUserSelect = "none";
 91     this.container.style.userSelect = "none";
 92 
 93     this.container.style.overflow = "hidden";
 94     if (this.container.style.position === "") {
 95         this.container.style.position = "relative";
 96     }
 97 
 98     this.svgRoot = this.container.ownerDocument.createElementNS(this.svgNamespace, "svg");
 99     this.svgRoot.style.overflow = "hidden";
100     this.svgRoot.style.display = "block";
101     this.resize(dim.width, dim.height);
102 
103     //this.svgRoot.setAttributeNS(null, 'shape-rendering', 'crispEdge'); //'optimizeQuality'); //geometricPrecision');
104 
105     this.container.appendChild(this.svgRoot);
106 
107     /**
108      * The <tt>defs</tt> element is a container element to reference reusable SVG elements.
109      * @type Node
110      * @see https://www.w3.org/TR/SVG2/struct.html#DefsElement
111      */
112     this.defs = this.container.ownerDocument.createElementNS(this.svgNamespace, "defs");
113     this.svgRoot.appendChild(this.defs);
114 
115     /**
116      * Filters are used to apply shadows.
117      * @type Node
118      * @see https://www.w3.org/TR/SVG2/struct.html#DefsElement
119      */
120     /**
121      * Create an SVG shadow filter. If the object's RGB color is [r,g,b], it's opacity is op, and
122      * the parameter color is given as [r', g', b'] with opacity op'
123      * the shadow will have RGB color [blend*r + r', blend*g + g', blend*b + b'] and the opacity will be equal to op * op'.
124      * Further, blur and offset can be adjusted.
125      *
126      * The shadow color is [r*ble
127      * @param {String} id Node is of the filter.
128      * @param {Array|String} rgb RGB value for the blend color or the string 'none' for default values. Default 'black'.
129      * @param {Number} opacity Value between 0 and 1, default is 1.
130      * @param {Number} blend  Value between 0 and 1, default is 0.1.
131      * @param {Number} blur  Default: 3
132      * @param {Array} offset [dx, dy]. Default is [5,5].
133      * @returns DOM node to be added to this.defs.
134      * @private
135      */
136     this.createShadowFilter = function (id, rgb, opacity, blend, blur, offset) {
137         var filter = this.container.ownerDocument.createElementNS(this.svgNamespace, 'filter'),
138             feOffset, feColor, feGaussianBlur, feBlend,
139             mat;
140 
141         filter.setAttributeNS(null, 'id', id);
142         filter.setAttributeNS(null, 'width', '300%');
143         filter.setAttributeNS(null, 'height', '300%');
144         filter.setAttributeNS(null, 'filterUnits', 'userSpaceOnUse');
145 
146         feOffset = this.container.ownerDocument.createElementNS(this.svgNamespace, 'feOffset');
147         feOffset.setAttributeNS(null, 'in', 'SourceGraphic'); // b/w: SourceAlpha, Color: SourceGraphic
148         feOffset.setAttributeNS(null, 'result', 'offOut');
149         feOffset.setAttributeNS(null, 'dx', offset[0]);
150         feOffset.setAttributeNS(null, 'dy', offset[1]);
151         filter.appendChild(feOffset);
152 
153         feColor = this.container.ownerDocument.createElementNS(this.svgNamespace, 'feColorMatrix');
154         feColor.setAttributeNS(null, 'in', 'offOut');
155         feColor.setAttributeNS(null, 'result', 'colorOut');
156         feColor.setAttributeNS(null, 'type', 'matrix');
157         // See https://developer.mozilla.org/en-US/docs/Web/SVG/Element/feColorMatrix
158         if (rgb === 'none' || !Type.isArray(rgb) || rgb.length < 3) {
159             feColor.setAttributeNS(null, 'values', '0.1 0 0 0 0  0 0.1 0 0 0  0 0 0.1 0 0  0 0 0 ' + opacity + ' 0');
160         } else {
161             rgb[0] /= 255;
162             rgb[1] /= 255;
163             rgb[2] /= 255;
164             mat = blend + ' 0 0 0 ' + rgb[0] +
165                 '  0 ' + blend + ' 0 0 ' + rgb[1] +
166                 '  0 0 ' + blend + ' 0 ' + rgb[2] +
167                 '  0 0 0 ' + opacity + ' 0';
168             feColor.setAttributeNS(null, 'values', mat);
169         }
170         filter.appendChild(feColor);
171 
172         feGaussianBlur = this.container.ownerDocument.createElementNS(this.svgNamespace, 'feGaussianBlur');
173         feGaussianBlur.setAttributeNS(null, 'in', 'colorOut');
174         feGaussianBlur.setAttributeNS(null, 'result', 'blurOut');
175         feGaussianBlur.setAttributeNS(null, 'stdDeviation', blur);
176         filter.appendChild(feGaussianBlur);
177 
178         feBlend = this.container.ownerDocument.createElementNS(this.svgNamespace, 'feBlend');
179         feBlend.setAttributeNS(null, 'in', 'SourceGraphic');
180         feBlend.setAttributeNS(null, 'in2', 'blurOut');
181         feBlend.setAttributeNS(null, 'mode', 'normal');
182         filter.appendChild(feBlend);
183 
184         return filter;
185     };
186 
187     /**
188      * Create a "unique" string id from the arguments of the function.
189      * Concatenate all arguments by "_".
190      * "Unique" is achieved by simply prepending the container id.
191      * Do not escape the string.
192      *
193      * If the id is used in an "url()" call it must be eascaped.
194      *
195      * @params {String} one or strings which will be concatenated.
196      * @return {String}
197      * @private
198      */
199     this.uniqName = function () {
200         return this.container.id + '_' +
201             Array.prototype.slice.call(arguments).join('_');
202     };
203 
204     /**
205      * Combine arguments to a string, joined by empty string.
206      * Masks the container id with CSS.escape.
207      *
208      * @params {String} str variable number of strings
209      * @returns String
210      * @see JXG.SVGRenderer#toURL
211      * @private
212      * @example
213      * this.toStr('aaa', '_', 'bbb', 'TriangleEnd')
214      * // Output:
215      * // xxx_bbbTriangleEnd
216      */
217     this.toStr = function() {
218         // ES6 would be [...arguments].join()
219         var str = Array.prototype.slice.call(arguments).join('');
220         // Mask special symbols like '/' and '\' in id
221         if (Type.exists(CSS) && Type.exists(CSS.escape)) {
222             str = CSS.escape(str);
223         }
224         return str;
225     };
226 
227     /**
228      * Combine arguments to an URL string of the form
229      * url(#...)
230      * Masks the container id. Calls {@link JXG.SVGRenderer#toStr}.
231      *
232      * @params {String} str variable number of strings
233      * @returns URL string
234      * @see JXG.SVGRenderer#toStr
235      * @private
236      * @example
237      * this.toURL('aaa', '_', 'bbb', 'TriangleEnd')
238      * // Output:
239      * // url(#xxx_bbbTriangleEnd)
240      */
241     this.toURL = function () {
242         return 'url(#' +
243             this.toStr.apply(this, arguments) + // Pass the arguments to toStr
244             ')';
245     };
246 
247     /* Default shadow filter */
248     this.defs.appendChild(this.createShadowFilter(this.uniqName('f1'), 'none', 1, 0.1, 3, [5, 5]));
249 
250     /**
251      * JSXGraph uses a layer system to sort the elements on the board. This puts certain types of elements in front
252      * of other types of elements. For the order used see {@link JXG.Options.layer}. The number of layers is documented
253      * there, too. The higher the number, the "more on top" are the elements on this layer.
254      * @type Array
255      */
256     this.layer = [];
257     for (i = 0; i < Options.layer.numlayers; i++) {
258         this.layer[i] = this.container.ownerDocument.createElementNS(this.svgNamespace, 'g');
259         this.svgRoot.appendChild(this.layer[i]);
260     }
261 
262     try {
263         this.foreignObjLayer = this.container.ownerDocument.createElementNS(
264             this.svgNamespace,
265             "foreignObject"
266         );
267         this.foreignObjLayer.setAttribute("display", "none");
268         this.foreignObjLayer.setAttribute("x", 0);
269         this.foreignObjLayer.setAttribute("y", 0);
270         this.foreignObjLayer.setAttribute("width", "100%");
271         this.foreignObjLayer.setAttribute("height", "100%");
272         this.foreignObjLayer.setAttribute("id", this.uniqName('foreignObj'));
273         this.svgRoot.appendChild(this.foreignObjLayer);
274         this.supportsForeignObject = true;
275     } catch (e) {
276         this.supportsForeignObject = false;
277     }
278 };
279 
280 JXG.SVGRenderer.prototype = new AbstractRenderer();
281 
282 JXG.extend(
283     JXG.SVGRenderer.prototype,
284     /** @lends JXG.SVGRenderer.prototype */ {
285         /**
286          * Creates an arrow DOM node. Arrows are displayed in SVG with a <em>marker</em> tag.
287          * @private
288          * @param {JXG.GeometryElement} el A JSXGraph element, preferably one that can have an arrow attached.
289          * @param {String} [idAppendix=''] A string that is added to the node's id.
290          * @returns {Node} Reference to the node added to the DOM.
291          */
292         _createArrowHead: function (el, idAppendix, type) {
293             var node2,
294                 node3,
295                 id = el.id + "Triangle",
296                 //type = null,
297                 v,
298                 h;
299 
300             if (Type.exists(idAppendix)) {
301                 id += idAppendix;
302             }
303             if (Type.exists(type)) {
304                 id += type;
305             }
306             node2 = this.createPrim("marker", id);
307 
308             node2.setAttributeNS(null, "stroke", el.evalVisProp('strokecolor'));
309             node2.setAttributeNS(
310                 null,
311                 "stroke-opacity",
312                 el.evalVisProp('strokeopacity')
313             );
314             node2.setAttributeNS(null, "fill", el.evalVisProp('strokecolor'));
315             node2.setAttributeNS(null, "fill-opacity", el.evalVisProp('strokeopacity'));
316             node2.setAttributeNS(null, "stroke-width", 0); // this is the stroke-width of the arrow head.
317             // Should be zero to simplify the calculations
318 
319             node2.setAttributeNS(null, "orient", "auto");
320             node2.setAttributeNS(null, "markerUnits", "strokeWidth"); // 'strokeWidth' 'userSpaceOnUse');
321 
322             /*
323                Types 1, 2:
324                The arrow head is an isosceles triangle with base length 10 and height 10.
325 
326                Type 3:
327                A rectangle
328 
329                Types 4, 5, 6:
330                Defined by Bezier curves from mp_arrowheads.html
331 
332                In any case but type 3 the arrow head is 10 units long,
333                type 3 is 10 units high.
334                These 10 units are scaled to strokeWidth * arrowSize pixels, see
335                this._setArrowWidth().
336 
337                See also abstractRenderer.updateLine() where the line path is shortened accordingly.
338 
339                Changes here are also necessary in setArrowWidth().
340 
341                So far, lines with arrow heads are shortenend to avoid overlapping of
342                arrow head and line. This is not the case for curves, yet.
343                Therefore, the offset refX has to be adapted to the path type.
344             */
345             node3 = this.container.ownerDocument.createElementNS(this.svgNamespace, "path");
346             h = 5;
347             if (idAppendix === "Start") {
348                 // First arrow
349                 v = 0;
350                 if (type === 2) {
351                     node3.setAttributeNS(null, "d", "M 10,0 L 0,5 L 10,10 L 5,5 z");
352                 } else if (type === 3) {
353                     node3.setAttributeNS(null, "d", "M 0,0 L 3.33,0 L 3.33,10 L 0,10 z");
354                 } else if (type === 4) {
355                     // insetRatio:0.8 tipAngle:45 wingCurve:15 tailCurve:0
356                     h = 3.31;
357                     node3.setAttributeNS(
358                         null,
359                         "d",
360                         "M 0.00,3.31 C 3.53,3.84 7.13,4.50 10.00,6.63 C 9.33,5.52 8.67,4.42 8.00,3.31 C 8.67,2.21 9.33,1.10 10.00,0.00 C 7.13,2.13 3.53,2.79 0.00,3.31"
361                     );
362                 } else if (type === 5) {
363                     // insetRatio:0.9 tipAngle:40 wingCurve:5 tailCurve:15
364                     h = 3.28;
365                     node3.setAttributeNS(
366                         null,
367                         "d",
368                         "M 0.00,3.28 C 3.39,4.19 6.81,5.07 10.00,6.55 C 9.38,5.56 9.00,4.44 9.00,3.28 C 9.00,2.11 9.38,0.99 10.00,0.00 C 6.81,1.49 3.39,2.37 0.00,3.28"
369                     );
370                 } else if (type === 6) {
371                     // insetRatio:0.9 tipAngle:35 wingCurve:5 tailCurve:0
372                     h = 2.84;
373                     node3.setAttributeNS(
374                         null,
375                         "d",
376                         "M 0.00,2.84 C 3.39,3.59 6.79,4.35 10.00,5.68 C 9.67,4.73 9.33,3.78 9.00,2.84 C 9.33,1.89 9.67,0.95 10.00,0.00 C 6.79,1.33 3.39,2.09 0.00,2.84"
377                     );
378                 } else if (type === 7) {
379                     // insetRatio:0.9 tipAngle:60 wingCurve:30 tailCurve:0
380                     h = 5.2;
381                     node3.setAttributeNS(
382                         null,
383                         "d",
384                         "M 0.00,5.20 C 4.04,5.20 7.99,6.92 10.00,10.39 M 10.00,0.00 C 7.99,3.47 4.04,5.20 0.00,5.20"
385                     );
386                 } else {
387                     // type == 1 or > 6
388                     node3.setAttributeNS(null, "d", "M 10,0 L 0,5 L 10,10 z");
389                 }
390                 if (
391                     // !Type.exists(el.rendNode.getTotalLength) &&
392                     el.elementClass === Const.OBJECT_CLASS_LINE
393                 ) {
394                     if (type === 2) {
395                         v = 4.9;
396                     } else if (type === 3) {
397                         v = 3.3;
398                     } else if (type === 4 || type === 5 || type === 6) {
399                         v = 6.66;
400                     } else if (type === 7) {
401                         v = 0.0;
402                     } else {
403                         v = 10.0;
404                     }
405                 }
406             } else {
407                 // Last arrow
408                 v = 10.0;
409                 if (type === 2) {
410                     node3.setAttributeNS(null, "d", "M 0,0 L 10,5 L 0,10 L 5,5 z");
411                 } else if (type === 3) {
412                     v = 3.3;
413                     node3.setAttributeNS(null, "d", "M 0,0 L 3.33,0 L 3.33,10 L 0,10 z");
414                 } else if (type === 4) {
415                     // insetRatio:0.8 tipAngle:45 wingCurve:15 tailCurve:0
416                     h = 3.31;
417                     node3.setAttributeNS(
418                         null,
419                         "d",
420                         "M 10.00,3.31 C 6.47,3.84 2.87,4.50 0.00,6.63 C 0.67,5.52 1.33,4.42 2.00,3.31 C 1.33,2.21 0.67,1.10 0.00,0.00 C 2.87,2.13 6.47,2.79 10.00,3.31"
421                     );
422                 } else if (type === 5) {
423                     // insetRatio:0.9 tipAngle:40 wingCurve:5 tailCurve:15
424                     h = 3.28;
425                     node3.setAttributeNS(
426                         null,
427                         "d",
428                         "M 10.00,3.28 C 6.61,4.19 3.19,5.07 0.00,6.55 C 0.62,5.56 1.00,4.44 1.00,3.28 C 1.00,2.11 0.62,0.99 0.00,0.00 C 3.19,1.49 6.61,2.37 10.00,3.28"
429                     );
430                 } else if (type === 6) {
431                     // insetRatio:0.9 tipAngle:35 wingCurve:5 tailCurve:0
432                     h = 2.84;
433                     node3.setAttributeNS(
434                         null,
435                         "d",
436                         "M 10.00,2.84 C 6.61,3.59 3.21,4.35 0.00,5.68 C 0.33,4.73 0.67,3.78 1.00,2.84 C 0.67,1.89 0.33,0.95 0.00,0.00 C 3.21,1.33 6.61,2.09 10.00,2.84"
437                     );
438                 } else if (type === 7) {
439                     // insetRatio:0.9 tipAngle:60 wingCurve:30 tailCurve:0
440                     h = 5.2;
441                     node3.setAttributeNS(
442                         null,
443                         "d",
444                         "M 10.00,5.20 C 5.96,5.20 2.01,6.92 0.00,10.39 M 0.00,0.00 C 2.01,3.47 5.96,5.20 10.00,5.20"
445                     );
446                 } else {
447                     // type == 1 or > 6
448                     node3.setAttributeNS(null, "d", "M 0,0 L 10,5 L 0,10 z");
449                 }
450                 if (
451                     // !Type.exists(el.rendNode.getTotalLength) &&
452                     el.elementClass === Const.OBJECT_CLASS_LINE
453                 ) {
454                     if (type === 2) {
455                         v = 5.1;
456                     } else if (type === 3) {
457                         v = 0.02;
458                     } else if (type === 4 || type === 5 || type === 6) {
459                         v = 3.33;
460                     } else if (type === 7) {
461                         v = 10.0;
462                     } else {
463                         v = 0.05;
464                     }
465                 }
466             }
467             if (type === 7) {
468                 node2.setAttributeNS(null, "fill", "none");
469                 node2.setAttributeNS(null, "stroke-width", 1); // this is the stroke-width of the arrow head.
470             }
471             node2.setAttributeNS(null, "refY", h);
472             node2.setAttributeNS(null, "refX", v);
473 
474             node2.appendChild(node3);
475             return node2;
476         },
477 
478         /**
479          * Updates color of an arrow DOM node.
480          * @param {Node} node The arrow node.
481          * @param {String} color Color value in a HTML compatible format, e.g. <tt>#00ff00</tt> or <tt>green</tt> for green.
482          * @param {Number} opacity
483          * @param {JXG.GeometryElement} el The element the arrows are to be attached to
484          */
485         _setArrowColor: function (node, color, opacity, el, type) {
486             if (node) {
487                 if (Type.isString(color)) {
488                     if (type !== 7) {
489                         this._setAttribute(function () {
490                             node.setAttributeNS(null, "stroke", color);
491                             node.setAttributeNS(null, "fill", color);
492                             node.setAttributeNS(null, "stroke-opacity", opacity);
493                             node.setAttributeNS(null, "fill-opacity", opacity);
494                         }, el.visPropOld.fillcolor);
495                     } else {
496                         this._setAttribute(function () {
497                             node.setAttributeNS(null, "fill", "none");
498                             node.setAttributeNS(null, "stroke", color);
499                             node.setAttributeNS(null, "stroke-opacity", opacity);
500                         }, el.visPropOld.fillcolor);
501                     }
502                 }
503 
504                 if (this.isIE) {
505                     el.rendNode.parentNode.insertBefore(el.rendNode, el.rendNode);
506                 }
507             }
508         },
509 
510         // Already documented in JXG.AbstractRenderer
511         _setArrowWidth: function (node, width, parentNode, size) {
512             var s, d;
513 
514             if (node) {
515                 // if (width === 0) {
516                 //     // display:none does not work well in webkit
517                 //     node.setAttributeNS(null, 'display', 'none');
518                 // } else {
519                 s = width;
520                 d = s * size;
521                 node.setAttributeNS(null, "viewBox", 0 + " " + 0 + " " + s * 10 + " " + s * 10);
522                 node.setAttributeNS(null, "markerHeight", d);
523                 node.setAttributeNS(null, "markerWidth", d);
524                 node.setAttributeNS(null, "display", "inherit");
525                 // }
526 
527                 if (this.isIE) {
528                     parentNode.parentNode.insertBefore(parentNode, parentNode);
529                 }
530             }
531         },
532 
533         /* ******************************** *
534          *  This renderer does not need to
535          *  override draw/update* methods
536          *  since it provides draw/update*Prim
537          *  methods except for some cases like
538          *  internal texts or images.
539          * ******************************** */
540 
541         /* **************************
542          *    Lines
543          * **************************/
544 
545         // documented in AbstractRenderer
546         updateTicks: function (ticks) {
547             var i,
548                 j,
549                 c,
550                 node,
551                 x,
552                 y,
553                 tickStr = "",
554                 len = ticks.ticks.length,
555                 len2,
556                 str,
557                 isReal = true;
558 
559             for (i = 0; i < len; i++) {
560                 c = ticks.ticks[i];
561                 x = c[0];
562                 y = c[1];
563 
564                 len2 = x.length;
565                 str = " M " + x[0] + " " + y[0];
566                 if (!Type.isNumber(x[0])) {
567                     isReal = false;
568                 }
569                 for (j = 1; isReal && j < len2; ++j) {
570                     if (Type.isNumber(x[j])) {
571                         str += " L " + x[j] + " " + y[j];
572                     } else {
573                         isReal = false;
574                     }
575                 }
576                 if (isReal) {
577                     tickStr += str;
578                 }
579             }
580 
581             node = ticks.rendNode;
582 
583             if (!Type.exists(node)) {
584                 node = this.createPrim("path", ticks.id);
585                 this.appendChildPrim(node, ticks.evalVisProp('layer'));
586                 ticks.rendNode = node;
587             }
588 
589             node.setAttributeNS(null, "stroke", ticks.evalVisProp('strokecolor'));
590             node.setAttributeNS(null, "fill", "none");
591             // node.setAttributeNS(null, 'fill', ticks.evalVisProp('fillcolor'));
592             // node.setAttributeNS(null, 'fill-opacity', ticks.evalVisProp('fillopacity'));
593             node.setAttributeNS(
594                 null,
595                 "stroke-opacity",
596                 ticks.evalVisProp('strokeopacity')
597             );
598             node.setAttributeNS(null, "stroke-width", ticks.evalVisProp('strokewidth'));
599             this.updatePathPrim(node, tickStr, ticks.board);
600         },
601 
602         /* **************************
603          *    Text related stuff
604          * **************************/
605 
606         // Already documented in JXG.AbstractRenderer
607         displayCopyright: function (str, fontsize) {
608             var node = this.createPrim("text", 'licenseText'),
609                 t;
610             node.setAttributeNS(null, 'x', '20px');
611             node.setAttributeNS(null, 'y', 2 + fontsize + 'px');
612             node.setAttributeNS(null, 'style', 'font-family:Arial,Helvetica,sans-serif; font-size:' +
613                 fontsize + 'px; fill:#356AA0;  opacity:0.3;');
614             t = this.container.ownerDocument.createTextNode(str);
615             node.appendChild(t);
616             this.appendChildPrim(node, 0);
617         },
618 
619         // Already documented in JXG.AbstractRenderer
620         drawInternalText: function (el) {
621             var node = this.createPrim("text", el.id);
622 
623             //node.setAttributeNS(null, "style", "alignment-baseline:middle"); // Not yet supported by Firefox
624             // Preserve spaces
625             //node.setAttributeNS("http://www.w3.org/XML/1998/namespace", "space", "preserve");
626             node.style.whiteSpace = "nowrap";
627 
628             el.rendNodeText = this.container.ownerDocument.createTextNode("");
629             node.appendChild(el.rendNodeText);
630             this.appendChildPrim(node, el.evalVisProp('layer'));
631 
632             return node;
633         },
634 
635         // Already documented in JXG.AbstractRenderer
636         updateInternalText: function (el) {
637             var content = el.plaintext,
638                 v, css,
639                 ev_ax = el.getAnchorX(),
640                 ev_ay = el.getAnchorY();
641 
642             css = el.evalVisProp('cssclass');
643             if (el.rendNode.getAttributeNS(null, "class") !== css) {
644                 el.rendNode.setAttributeNS(null, "class", css);
645                 el.needsSizeUpdate = true;
646             }
647 
648             if (!isNaN(el.coords.scrCoords[1] + el.coords.scrCoords[2])) {
649                 // Horizontal
650                 v = el.coords.scrCoords[1];
651                 if (el.visPropOld.left !== ev_ax + v) {
652                     el.rendNode.setAttributeNS(null, "x", v + "px");
653 
654                     if (ev_ax === "left") {
655                         el.rendNode.setAttributeNS(null, "text-anchor", "start");
656                     } else if (ev_ax === "right") {
657                         el.rendNode.setAttributeNS(null, "text-anchor", "end");
658                     } else if (ev_ax === "middle") {
659                         el.rendNode.setAttributeNS(null, "text-anchor", "middle");
660                     }
661                     el.visPropOld.left = ev_ax + v;
662                 }
663 
664                 // Vertical
665                 v = el.coords.scrCoords[2];
666                 if (el.visPropOld.top !== ev_ay + v) {
667                     el.rendNode.setAttributeNS(null, "y", v + this.vOffsetText * 0.5 + "px");
668 
669                     // Not supported by IE, edge
670                     // el.rendNode.setAttributeNS(null, "dy", "0");
671                     // if (ev_ay === "bottom") {
672                     //     el.rendNode.setAttributeNS(null, 'dominant-baseline', 'text-after-edge');
673                     // } else if (ev_ay === "top") {
674                     //     el.rendNode.setAttributeNS(null, 'dominant-baseline', 'text-before-edge');
675                     // } else if (ev_ay === "middle") {
676                     //     el.rendNode.setAttributeNS(null, 'dominant-baseline', 'middle');
677                     // }
678 
679                     if (ev_ay === "bottom") {
680                         el.rendNode.setAttributeNS(null, "dy", "0");
681                         el.rendNode.setAttributeNS(null, 'dominant-baseline', 'auto');
682                     } else if (ev_ay === "top") {
683                         el.rendNode.setAttributeNS(null, "dy", "1.6ex");
684                         el.rendNode.setAttributeNS(null, 'dominant-baseline', 'auto');
685                     } else if (ev_ay === "middle") {
686                         el.rendNode.setAttributeNS(null, "dy", "0.6ex");
687                         el.rendNode.setAttributeNS(null, 'dominant-baseline', 'auto');
688                     }
689                     el.visPropOld.top = ev_ay + v;
690                 }
691             }
692             if (el.htmlStr !== content) {
693                 el.rendNodeText.data = content;
694                 el.htmlStr = content;
695             }
696             this.transformImage(el, el.transformations);
697         },
698 
699         /**
700          * Set color and opacity of internal texts.
701          * @private
702          * @see JXG.AbstractRenderer#updateTextStyle
703          * @see JXG.AbstractRenderer#updateInternalTextStyle
704          */
705         updateInternalTextStyle: function (el, strokeColor, strokeOpacity, duration) {
706             this.setObjectFillColor(el, strokeColor, strokeOpacity);
707         },
708 
709         /* **************************
710          *    Image related stuff
711          * **************************/
712 
713         // Already documented in JXG.AbstractRenderer
714         drawImage: function (el) {
715             var node = this.createPrim("image", el.id);
716 
717             node.setAttributeNS(null, "preserveAspectRatio", "none");
718             this.appendChildPrim(node, el.evalVisProp('layer'));
719             el.rendNode = node;
720 
721             this.updateImage(el);
722         },
723 
724         // Already documented in JXG.AbstractRenderer
725         transformImage: function (el, t) {
726             var s, m, node,
727                 str = "",
728                 cx, cy,
729                 len = t.length;
730 
731             if (len > 0) {
732                 node = el.rendNode;
733                 m = this.joinTransforms(el, t);
734                 s = [m[1][1], m[2][1], m[1][2], m[2][2], m[1][0], m[2][0]].join(",");
735                 if (s.indexOf('NaN') === -1) {
736                     str += " matrix(" + s + ") ";
737                     if (el.elementClass === Const.OBJECT_CLASS_TEXT && el.visProp.display === 'html') {
738                         node.style.transform = str;
739                         cx = -el.coords.scrCoords[1];
740                         cy = -el.coords.scrCoords[2];
741                         switch (el.evalVisProp('anchorx')) {
742                             case 'right': cx += el.size[0]; break;
743                             case 'middle': cx += el.size[0] * 0.5; break;
744                         }
745                         switch (el.evalVisProp('anchory')) {
746                             case 'bottom': cy += el.size[1]; break;
747                             case 'middle': cy += el.size[1] * 0.5; break;
748                         }
749                         node.style['transform-origin'] = (cx) + 'px ' + (cy) + 'px';
750                     } else {
751                         // Images and texts with display:'internal'
752                         node.setAttributeNS(null, "transform", str);
753                     }
754                 }
755             }
756         },
757 
758         // Already documented in JXG.AbstractRenderer
759         updateImageURL: function (el) {
760             var url = el.eval(el.url);
761 
762             if (el._src !== url) {
763                 el.imgIsLoaded = false;
764                 el.rendNode.setAttributeNS(this.xlinkNamespace, "xlink:href", url);
765                 el._src = url;
766 
767                 return true;
768             }
769 
770             return false;
771         },
772 
773         // Already documented in JXG.AbstractRenderer
774         updateImageStyle: function (el, doHighlight) {
775             var css = el.evalVisProp(
776                 doHighlight ? 'highlightcssclass' : 'cssclass'
777             );
778 
779             el.rendNode.setAttributeNS(null, "class", css);
780         },
781 
782         // Already documented in JXG.AbstractRenderer
783         drawForeignObject: function (el) {
784             el.rendNode = this.appendChildPrim(
785                 this.createPrim("foreignObject", el.id),
786                 el.evalVisProp('layer')
787             );
788 
789             this.appendNodesToElement(el, "foreignObject");
790             this.updateForeignObject(el);
791         },
792 
793         // Already documented in JXG.AbstractRenderer
794         updateForeignObject: function (el) {
795             if (el._useUserSize) {
796                 el.rendNode.style.overflow = "hidden";
797             } else {
798                 el.rendNode.style.overflow = "visible";
799             }
800 
801             this.updateRectPrim(
802                 el.rendNode,
803                 el.coords.scrCoords[1],
804                 el.coords.scrCoords[2] - el.size[1],
805                 el.size[0],
806                 el.size[1]
807             );
808 
809             el.rendNode.innerHTML = el.content;
810             this._updateVisual(el, { stroke: true, dash: true }, true);
811         },
812 
813         /* **************************
814          * Render primitive objects
815          * **************************/
816 
817         // Already documented in JXG.AbstractRenderer
818         appendChildPrim: function (node, level) {
819             if (!Type.exists(level)) {
820                 // trace nodes have level not set
821                 level = 0;
822             } else if (level >= Options.layer.numlayers) {
823                 level = Options.layer.numlayers - 1;
824             }
825             this.layer[level].appendChild(node);
826 
827             return node;
828         },
829 
830         // Already documented in JXG.AbstractRenderer
831         createPrim: function (type, id) {
832             var node = this.container.ownerDocument.createElementNS(this.svgNamespace, type);
833             node.setAttributeNS(null, "id", this.uniqName(id));
834             node.style.position = "absolute";
835             if (type === "path") {
836                 node.setAttributeNS(null, "stroke-linecap", "round");
837                 node.setAttributeNS(null, "stroke-linejoin", "round");
838                 node.setAttributeNS(null, "fill-rule", "evenodd");
839             }
840 
841             return node;
842         },
843 
844         // Already documented in JXG.AbstractRenderer
845         remove: function (shape) {
846             if (Type.exists(shape) && Type.exists(shape.parentNode)) {
847                 shape.parentNode.removeChild(shape);
848             }
849         },
850 
851         // Already documented in JXG.AbstractRenderer
852         setLayer: function (el, level) {
853             if (!Type.exists(level)) {
854                 level = 0;
855             } else if (level >= Options.layer.numlayers) {
856                 level = Options.layer.numlayers - 1;
857             }
858 
859             this.layer[level].appendChild(el.rendNode);
860         },
861 
862         // Already documented in JXG.AbstractRenderer
863         makeArrows: function (el, a) {
864             var node2, str,
865                 ev_fa = a.evFirst,
866                 ev_la = a.evLast;
867 
868             if (this.isIE && el.visPropCalc.visible && (ev_fa || ev_la)) {
869                 el.rendNode.parentNode.insertBefore(el.rendNode, el.rendNode);
870                 return;
871             }
872 
873             // We can not compare against visPropOld if there is need for a new arrow head,
874             // since here visPropOld and ev_fa / ev_la already have the same value.
875             // This has been set in _updateVisual.
876             //
877             node2 = el.rendNodeTriangleStart;
878             if (ev_fa) {
879                 str = this.toStr(this.container.id, '_', el.id, 'TriangleStart', a.typeFirst);
880 
881                 // If we try to set the same arrow head as is already set, we can bail out now
882                 if (!Type.exists(node2) || node2.id !== str) {
883                     node2 = this.container.ownerDocument.getElementById(str);
884                     // Check if the marker already exists.
885                     // If not, create a new marker
886                     if (node2 === null) {
887                         node2 = this._createArrowHead(el, "Start", a.typeFirst);
888                         this.defs.appendChild(node2);
889                     }
890                     el.rendNodeTriangleStart = node2;
891                     el.rendNode.setAttributeNS(null, "marker-start", this.toURL(str));
892                 }
893             } else {
894                 if (Type.exists(node2)) {
895                     this.remove(node2);
896                     el.rendNodeTriangleStart = null;
897                 }
898                 el.rendNode.setAttributeNS(null, "marker-start", null);
899             }
900 
901             node2 = el.rendNodeTriangleEnd;
902             if (ev_la) {
903                 str = this.toStr(this.container.id, '_', el.id, 'TriangleEnd', a.typeLast);
904 
905                 // If we try to set the same arrow head as is already set, we can bail out now
906                 if (!Type.exists(node2) || node2.id !== str) {
907                     node2 = this.container.ownerDocument.getElementById(str);
908                     // Check if the marker already exists.
909                     // If not, create a new marker
910                     if (node2 === null) {
911                         node2 = this._createArrowHead(el, "End", a.typeLast);
912                         this.defs.appendChild(node2);
913                     }
914                     el.rendNodeTriangleEnd = node2;
915                     el.rendNode.setAttributeNS(null, "marker-end", this.toURL(str));
916                 }
917             } else {
918                 if (Type.exists(node2)) {
919                     this.remove(node2);
920                     el.rendNodeTriangleEnd = null;
921                 }
922                 el.rendNode.setAttributeNS(null, "marker-end", null);
923             }
924         },
925 
926         // Already documented in JXG.AbstractRenderer
927         updateEllipsePrim: function (node, x, y, rx, ry) {
928             var huge = 1000000;
929 
930             huge = 200000; // IE
931             // webkit does not like huge values if the object is dashed
932             // iE doesn't like huge values above 216000
933             x = Math.abs(x) < huge ? x : (huge * x) / Math.abs(x);
934             y = Math.abs(y) < huge ? y : (huge * y) / Math.abs(y);
935             rx = Math.abs(rx) < huge ? rx : (huge * rx) / Math.abs(rx);
936             ry = Math.abs(ry) < huge ? ry : (huge * ry) / Math.abs(ry);
937 
938             node.setAttributeNS(null, "cx", x);
939             node.setAttributeNS(null, "cy", y);
940             node.setAttributeNS(null, "rx", Math.abs(rx));
941             node.setAttributeNS(null, "ry", Math.abs(ry));
942         },
943 
944         // Already documented in JXG.AbstractRenderer
945         updateLinePrim: function (node, p1x, p1y, p2x, p2y) {
946             var huge = 1000000;
947 
948             huge = 200000; //IE
949             if (!isNaN(p1x + p1y + p2x + p2y)) {
950                 // webkit does not like huge values if the object is dashed
951                 // IE doesn't like huge values above 216000
952                 p1x = Math.abs(p1x) < huge ? p1x : (huge * p1x) / Math.abs(p1x);
953                 p1y = Math.abs(p1y) < huge ? p1y : (huge * p1y) / Math.abs(p1y);
954                 p2x = Math.abs(p2x) < huge ? p2x : (huge * p2x) / Math.abs(p2x);
955                 p2y = Math.abs(p2y) < huge ? p2y : (huge * p2y) / Math.abs(p2y);
956 
957                 node.setAttributeNS(null, "x1", p1x);
958                 node.setAttributeNS(null, "y1", p1y);
959                 node.setAttributeNS(null, "x2", p2x);
960                 node.setAttributeNS(null, "y2", p2y);
961             }
962         },
963 
964         // Already documented in JXG.AbstractRenderer
965         updatePathPrim: function (node, pointString) {
966             if (pointString === "") {
967                 pointString = "M 0 0";
968             }
969             node.setAttributeNS(null, "d", pointString);
970         },
971 
972         // Already documented in JXG.AbstractRenderer
973         updatePathStringPoint: function (el, size, type) {
974             var s = "",
975                 scr = el.coords.scrCoords,
976                 sqrt32 = size * Math.sqrt(3) * 0.5,
977                 s05 = size * 0.5;
978 
979             if (type === "x") {
980                 s =
981                     " M " +
982                     (scr[1] - size) +
983                     " " +
984                     (scr[2] - size) +
985                     " L " +
986                     (scr[1] + size) +
987                     " " +
988                     (scr[2] + size) +
989                     " M " +
990                     (scr[1] + size) +
991                     " " +
992                     (scr[2] - size) +
993                     " L " +
994                     (scr[1] - size) +
995                     " " +
996                     (scr[2] + size);
997             } else if (type === "+") {
998                 s =
999                     " M " +
1000                     (scr[1] - size) +
1001                     " " +
1002                     scr[2] +
1003                     " L " +
1004                     (scr[1] + size) +
1005                     " " +
1006                     scr[2] +
1007                     " M " +
1008                     scr[1] +
1009                     " " +
1010                     (scr[2] - size) +
1011                     " L " +
1012                     scr[1] +
1013                     " " +
1014                     (scr[2] + size);
1015             } else if (type === "|") {
1016                 s =
1017                     " M " +
1018                     scr[1] +
1019                     " " +
1020                     (scr[2] - size) +
1021                     " L " +
1022                     scr[1] +
1023                     " " +
1024                     (scr[2] + size);
1025             } else if (type === "-") {
1026                 s =
1027                     " M " +
1028                     (scr[1] - size) +
1029                     " " +
1030                     scr[2] +
1031                     " L " +
1032                     (scr[1] + size) +
1033                     " " +
1034                     scr[2];
1035             } else if (type === "<>" || type === "<<>>") {
1036                 if (type === "<<>>") {
1037                     size *= 1.41;
1038                 }
1039                 s =
1040                     " M " +
1041                     (scr[1] - size) +
1042                     " " +
1043                     scr[2] +
1044                     " L " +
1045                     scr[1] +
1046                     " " +
1047                     (scr[2] + size) +
1048                     " L " +
1049                     (scr[1] + size) +
1050                     " " +
1051                     scr[2] +
1052                     " L " +
1053                     scr[1] +
1054                     " " +
1055                     (scr[2] - size) +
1056                     " Z ";
1057                 } else if (type === "^") {
1058                     s =
1059                     " M " +
1060                     scr[1] +
1061                     " " +
1062                     (scr[2] - size) +
1063                     " L " +
1064                     (scr[1] - sqrt32) +
1065                     " " +
1066                     (scr[2] + s05) +
1067                     " L " +
1068                     (scr[1] + sqrt32) +
1069                     " " +
1070                     (scr[2] + s05) +
1071                     " Z "; // close path
1072             } else if (type === "v") {
1073                 s =
1074                     " M " +
1075                     scr[1] +
1076                     " " +
1077                     (scr[2] + size) +
1078                     " L " +
1079                     (scr[1] - sqrt32) +
1080                     " " +
1081                     (scr[2] - s05) +
1082                     " L " +
1083                     (scr[1] + sqrt32) +
1084                     " " +
1085                     (scr[2] - s05) +
1086                     " Z ";
1087             } else if (type === ">") {
1088                 s =
1089                     " M " +
1090                     (scr[1] + size) +
1091                     " " +
1092                     scr[2] +
1093                     " L " +
1094                     (scr[1] - s05) +
1095                     " " +
1096                     (scr[2] - sqrt32) +
1097                     " L " +
1098                     (scr[1] - s05) +
1099                     " " +
1100                     (scr[2] + sqrt32) +
1101                     " Z ";
1102             } else if (type === "<") {
1103                 s =
1104                     " M " +
1105                     (scr[1] - size) +
1106                     " " +
1107                     scr[2] +
1108                     " L " +
1109                     (scr[1] + s05) +
1110                     " " +
1111                     (scr[2] - sqrt32) +
1112                     " L " +
1113                     (scr[1] + s05) +
1114                     " " +
1115                     (scr[2] + sqrt32) +
1116                     " Z ";
1117             }
1118             return s;
1119         },
1120 
1121         // Already documented in JXG.AbstractRenderer
1122         updatePathStringPrim: function (el) {
1123             var i,
1124                 scr,
1125                 len,
1126                 symbm = " M ",
1127                 symbl = " L ",
1128                 symbc = " C ",
1129                 nextSymb = symbm,
1130                 maxSize = 5000.0,
1131                 pStr = "";
1132 
1133             if (el.numberPoints <= 0) {
1134                 return "";
1135             }
1136 
1137             len = Math.min(el.points.length, el.numberPoints);
1138 
1139             if (el.bezierDegree === 1) {
1140                 for (i = 0; i < len; i++) {
1141                     scr = el.points[i].scrCoords;
1142                     if (isNaN(scr[1]) || isNaN(scr[2])) {
1143                         // PenUp
1144                         nextSymb = symbm;
1145                     } else {
1146                         // Chrome has problems with values being too far away.
1147                         scr[1] = Math.max(Math.min(scr[1], maxSize), -maxSize);
1148                         scr[2] = Math.max(Math.min(scr[2], maxSize), -maxSize);
1149 
1150                         // Attention: first coordinate may be inaccurate if far way
1151                         //pStr += [nextSymb, scr[1], ' ', scr[2]].join('');
1152                         pStr += nextSymb + scr[1] + " " + scr[2]; // Seems to be faster now (webkit and firefox)
1153                         nextSymb = symbl;
1154                     }
1155                 }
1156             } else if (el.bezierDegree === 3) {
1157                 i = 0;
1158                 while (i < len) {
1159                     scr = el.points[i].scrCoords;
1160                     if (isNaN(scr[1]) || isNaN(scr[2])) {
1161                         // PenUp
1162                         nextSymb = symbm;
1163                     } else {
1164                         pStr += nextSymb + scr[1] + " " + scr[2];
1165                         if (nextSymb === symbc) {
1166                             i += 1;
1167                             scr = el.points[i].scrCoords;
1168                             pStr += " " + scr[1] + " " + scr[2];
1169                             i += 1;
1170                             scr = el.points[i].scrCoords;
1171                             pStr += " " + scr[1] + " " + scr[2];
1172                         }
1173                         nextSymb = symbc;
1174                     }
1175                     i += 1;
1176                 }
1177             }
1178             return pStr;
1179         },
1180 
1181         // Already documented in JXG.AbstractRenderer
1182         updatePathStringBezierPrim: function (el) {
1183             var i, j, k,
1184                 scr,
1185                 lx, ly,
1186                 len,
1187                 symbm = " M ",
1188                 symbl = " C ",
1189                 nextSymb = symbm,
1190                 maxSize = 5000.0,
1191                 pStr = "",
1192                 f = el.evalVisProp('strokewidth'),
1193                 isNoPlot = el.evalVisProp('curvetype') !== "plot";
1194 
1195             if (el.numberPoints <= 0) {
1196                 return "";
1197             }
1198 
1199             if (isNoPlot && el.board.options.curve.RDPsmoothing) {
1200                 el.points = Numerics.RamerDouglasPeucker(el.points, 0.5);
1201             }
1202 
1203             len = Math.min(el.points.length, el.numberPoints);
1204             for (j = 1; j < 3; j++) {
1205                 nextSymb = symbm;
1206                 for (i = 0; i < len; i++) {
1207                     scr = el.points[i].scrCoords;
1208 
1209                     if (isNaN(scr[1]) || isNaN(scr[2])) {
1210                         // PenUp
1211                         nextSymb = symbm;
1212                     } else {
1213                         // Chrome has problems with values being too far away.
1214                         scr[1] = Math.max(Math.min(scr[1], maxSize), -maxSize);
1215                         scr[2] = Math.max(Math.min(scr[2], maxSize), -maxSize);
1216 
1217                         // Attention: first coordinate may be inaccurate if far way
1218                         if (nextSymb === symbm) {
1219                             //pStr += [nextSymb, scr[1], ' ', scr[2]].join('');
1220                             pStr += nextSymb + scr[1] + " " + scr[2]; // Seems to be faster now (webkit and firefox)
1221                         } else {
1222                             k = 2 * j;
1223                             pStr += [
1224                                 nextSymb,
1225                                 lx + (scr[1] - lx) * 0.333 + f * (k * Math.random() - j),
1226                                 " ",
1227                                 ly + (scr[2] - ly) * 0.333 + f * (k * Math.random() - j),
1228                                 " ",
1229                                 lx + (scr[1] - lx) * 0.666 + f * (k * Math.random() - j),
1230                                 " ",
1231                                 ly + (scr[2] - ly) * 0.666 + f * (k * Math.random() - j),
1232                                 " ",
1233                                 scr[1],
1234                                 " ",
1235                                 scr[2]
1236                             ].join("");
1237                         }
1238 
1239                         nextSymb = symbl;
1240                         lx = scr[1];
1241                         ly = scr[2];
1242                     }
1243                 }
1244             }
1245             return pStr;
1246         },
1247 
1248         // Already documented in JXG.AbstractRenderer
1249         updatePolygonPrim: function (node, el) {
1250             var i,
1251                 pStr = "",
1252                 scrCoords,
1253                 len = el.vertices.length;
1254 
1255             node.setAttributeNS(null, "stroke", "none");
1256             node.setAttributeNS(null, "fill-rule", "evenodd");
1257             if (el.elType === "polygonalchain") {
1258                 len++;
1259             }
1260 
1261             for (i = 0; i < len - 1; i++) {
1262                 if (el.vertices[i].isReal) {
1263                     scrCoords = el.vertices[i].coords.scrCoords;
1264                     pStr = pStr + scrCoords[1] + "," + scrCoords[2];
1265                 } else {
1266                     node.setAttributeNS(null, "points", "");
1267                     return;
1268                 }
1269 
1270                 if (i < len - 2) {
1271                     pStr += " ";
1272                 }
1273             }
1274             if (pStr.indexOf("NaN") === -1) {
1275                 node.setAttributeNS(null, "points", pStr);
1276             }
1277         },
1278 
1279         // Already documented in JXG.AbstractRenderer
1280         updateRectPrim: function (node, x, y, w, h) {
1281             node.setAttributeNS(null, "x", x);
1282             node.setAttributeNS(null, "y", y);
1283             node.setAttributeNS(null, "width", w);
1284             node.setAttributeNS(null, "height", h);
1285         },
1286 
1287         /* **************************
1288          *  Set Attributes
1289          * **************************/
1290 
1291         // documented in JXG.AbstractRenderer
1292         setPropertyPrim: function (node, key, val) {
1293             if (key === "stroked") {
1294                 return;
1295             }
1296             node.setAttributeNS(null, key, val);
1297         },
1298 
1299         display: function (el, val) {
1300             var node;
1301 
1302             if (el && el.rendNode) {
1303                 el.visPropOld.visible = val;
1304                 node = el.rendNode;
1305                 if (val) {
1306                     node.setAttributeNS(null, "display", "inline");
1307                     node.style.visibility = "inherit";
1308                 } else {
1309                     node.setAttributeNS(null, "display", "none");
1310                     node.style.visibility = "hidden";
1311                 }
1312             }
1313         },
1314 
1315         // documented in JXG.AbstractRenderer
1316         show: function (el) {
1317             JXG.deprecated("Board.renderer.show()", "Board.renderer.display()");
1318             this.display(el, true);
1319             // var node;
1320             //
1321             // if (el && el.rendNode) {
1322             //     node = el.rendNode;
1323             //     node.setAttributeNS(null, 'display', 'inline');
1324             //     node.style.visibility = "inherit";
1325             // }
1326         },
1327 
1328         // documented in JXG.AbstractRenderer
1329         hide: function (el) {
1330             JXG.deprecated("Board.renderer.hide()", "Board.renderer.display()");
1331             this.display(el, false);
1332             // var node;
1333             //
1334             // if (el && el.rendNode) {
1335             //     node = el.rendNode;
1336             //     node.setAttributeNS(null, 'display', 'none');
1337             //     node.style.visibility = "hidden";
1338             // }
1339         },
1340 
1341         // documented in JXG.AbstractRenderer
1342         setBuffering: function (el, type) {
1343             el.rendNode.setAttribute("buffered-rendering", type);
1344         },
1345 
1346         // documented in JXG.AbstractRenderer
1347         setDashStyle: function (el) {
1348             var dashStyle = el.evalVisProp('dash'),
1349                 ds = el.evalVisProp('dashscale'),
1350                 sw = ds ? 0.5 * el.evalVisProp('strokewidth') : 1,
1351                 node = el.rendNode;
1352 
1353             if (dashStyle > 0) {
1354                 node.setAttributeNS(null, "stroke-dasharray",
1355                     // sw could distinguish highlighting or not.
1356                     // But it seems to preferable to ignore this.
1357                     this.dashArray[dashStyle - 1].map(function (x) { return x * sw; }).join(',')
1358                 );
1359             } else {
1360                 if (node.hasAttributeNS(null, "stroke-dasharray")) {
1361                     node.removeAttributeNS(null, "stroke-dasharray");
1362                 }
1363             }
1364         },
1365 
1366         // documented in JXG.AbstractRenderer
1367         setGradient: function (el) {
1368             var fillNode = el.rendNode,
1369                 node, node2, node3,
1370                 ev_g = el.evalVisProp('gradient');
1371 
1372             if (ev_g === "linear" || ev_g === "radial") {
1373                 node = this.createPrim(ev_g + "Gradient", el.id + "_gradient");
1374                 node2 = this.createPrim("stop", el.id + "_gradient1");
1375                 node3 = this.createPrim("stop", el.id + "_gradient2");
1376                 node.appendChild(node2);
1377                 node.appendChild(node3);
1378                 this.defs.appendChild(node);
1379                 fillNode.setAttributeNS(
1380                     null,
1381                     'style',
1382                     // "fill:url(#" + this.container.id + "_" + el.id + "_gradient)"
1383                     'fill:' + this.toURL(this.container.id + '_' + el.id + '_gradient')
1384                 );
1385                 el.gradNode1 = node2;
1386                 el.gradNode2 = node3;
1387                 el.gradNode = node;
1388             } else {
1389                 fillNode.removeAttributeNS(null, "style");
1390             }
1391         },
1392 
1393         /**
1394          * Set the gradient angle for linear color gradients.
1395          *
1396          * @private
1397          * @param {SVGnode} node SVG gradient node of an arbitrary JSXGraph element.
1398          * @param {Number} radians angle value in radians. 0 is horizontal from left to right, Pi/4 is vertical from top to bottom.
1399          */
1400         updateGradientAngle: function (node, radians) {
1401             // Angles:
1402             // 0: ->
1403             // 90: down
1404             // 180: <-
1405             // 90: up
1406             var f = 1.0,
1407                 co = Math.cos(radians),
1408                 si = Math.sin(radians);
1409 
1410             if (Math.abs(co) > Math.abs(si)) {
1411                 f /= Math.abs(co);
1412             } else {
1413                 f /= Math.abs(si);
1414             }
1415 
1416             if (co >= 0) {
1417                 node.setAttributeNS(null, "x1", 0);
1418                 node.setAttributeNS(null, "x2", co * f);
1419             } else {
1420                 node.setAttributeNS(null, "x1", -co * f);
1421                 node.setAttributeNS(null, "x2", 0);
1422             }
1423             if (si >= 0) {
1424                 node.setAttributeNS(null, "y1", 0);
1425                 node.setAttributeNS(null, "y2", si * f);
1426             } else {
1427                 node.setAttributeNS(null, "y1", -si * f);
1428                 node.setAttributeNS(null, "y2", 0);
1429             }
1430         },
1431 
1432         /**
1433          * Set circles for radial color gradients.
1434          *
1435          * @private
1436          * @param {SVGnode} node SVG gradient node
1437          * @param {Number} cx SVG value cx (value between 0 and 1)
1438          * @param {Number} cy  SVG value cy (value between 0 and 1)
1439          * @param {Number} r  SVG value r (value between 0 and 1)
1440          * @param {Number} fx  SVG value fx (value between 0 and 1)
1441          * @param {Number} fy  SVG value fy (value between 0 and 1)
1442          * @param {Number} fr  SVG value fr (value between 0 and 1)
1443          */
1444         updateGradientCircle: function (node, cx, cy, r, fx, fy, fr) {
1445             node.setAttributeNS(null, "cx", cx * 100 + "%"); // Center first color
1446             node.setAttributeNS(null, "cy", cy * 100 + "%");
1447             node.setAttributeNS(null, "r", r * 100 + "%");
1448             node.setAttributeNS(null, "fx", fx * 100 + "%"); // Center second color / focal point
1449             node.setAttributeNS(null, "fy", fy * 100 + "%");
1450             node.setAttributeNS(null, "fr", fr * 100 + "%");
1451         },
1452 
1453         // documented in JXG.AbstractRenderer
1454         updateGradient: function (el) {
1455             var col,
1456                 op,
1457                 node2 = el.gradNode1,
1458                 node3 = el.gradNode2,
1459                 ev_g = el.evalVisProp('gradient');
1460 
1461             if (!Type.exists(node2) || !Type.exists(node3)) {
1462                 return;
1463             }
1464 
1465             op = el.evalVisProp('fillopacity');
1466             op = op > 0 ? op : 0;
1467             col = el.evalVisProp('fillcolor');
1468 
1469             node2.setAttributeNS(null, "style", "stop-color:" + col + ";stop-opacity:" + op);
1470             node3.setAttributeNS(
1471                 null,
1472                 "style",
1473                 "stop-color:" +
1474                 el.evalVisProp('gradientsecondcolor') +
1475                 ";stop-opacity:" +
1476                 el.evalVisProp('gradientsecondopacity')
1477             );
1478             node2.setAttributeNS(
1479                 null,
1480                 "offset",
1481                 el.evalVisProp('gradientstartoffset') * 100 + "%"
1482             );
1483             node3.setAttributeNS(
1484                 null,
1485                 "offset",
1486                 el.evalVisProp('gradientendoffset') * 100 + "%"
1487             );
1488             if (ev_g === "linear") {
1489                 this.updateGradientAngle(el.gradNode, el.evalVisProp('gradientangle'));
1490             } else if (ev_g === "radial") {
1491                 this.updateGradientCircle(
1492                     el.gradNode,
1493                     el.evalVisProp('gradientcx'),
1494                     el.evalVisProp('gradientcy'),
1495                     el.evalVisProp('gradientr'),
1496                     el.evalVisProp('gradientfx'),
1497                     el.evalVisProp('gradientfy'),
1498                     el.evalVisProp('gradientfr')
1499                 );
1500             }
1501         },
1502 
1503         // documented in JXG.AbstractRenderer
1504         setObjectTransition: function (el, duration) {
1505             var node, props,
1506                 transitionArr = [],
1507                 transitionStr,
1508                 i,
1509                 len = 0,
1510                 nodes = ["rendNode", "rendNodeTriangleStart", "rendNodeTriangleEnd"];
1511 
1512             if (duration === undefined) {
1513                 duration = el.evalVisProp('transitionduration');
1514             }
1515 
1516             props = el.evalVisProp('transitionproperties');
1517             if (duration === el.visPropOld.transitionduration &&
1518                 props === el.visPropOld.transitionproperties) {
1519                 return;
1520             }
1521 
1522             // if (
1523             //     el.elementClass === Const.OBJECT_CLASS_TEXT &&
1524             //     el.evalVisProp('display') === "html"
1525             // ) {
1526             //     // transitionStr = " color " + duration + "ms," +
1527             //     //     " opacity " + duration + "ms";
1528             //     transitionStr = " all " + duration + "ms ease";
1529             // } else {
1530             //     transitionStr =
1531             //         " fill " + duration + "ms," +
1532             //         " fill-opacity " + duration + "ms," +
1533             //         " stroke " + duration + "ms," +
1534             //         " stroke-opacity " + duration + "ms," +
1535             //         " stroke-width " + duration + "ms," +
1536             //         " width " + duration + "ms," +
1537             //         " height " + duration + "ms," +
1538             //         " rx " + duration + "ms," +
1539             //         " ry " + duration + "ms";
1540             // }
1541 
1542             if (Type.exists(props)) {
1543                 len = props.length;
1544             }
1545             for (i = 0; i < len; i++) {
1546                 transitionArr.push(props[i] + ' ' + duration + 'ms');
1547             }
1548             transitionStr = transitionArr.join(', ');
1549 
1550             len = nodes.length;
1551             for (i = 0; i < len; ++i) {
1552                 if (el[nodes[i]]) {
1553                     node = el[nodes[i]];
1554                     node.style.transition = transitionStr;
1555                 }
1556             }
1557 
1558             el.visPropOld.transitionduration = duration;
1559             el.visPropOld.transitionproperties = props;
1560         },
1561 
1562         /**
1563          * Call user-defined function to set visual attributes.
1564          * If "testAttribute" is the empty string, the function
1565          * is called immediately, otherwise it is called in a timeOut.
1566          *
1567          * This is necessary to realize smooth transitions but avoid transitions
1568          * when first creating the objects.
1569          *
1570          * Usually, the string in testAttribute is the visPropOld attribute
1571          * of the values which are set.
1572          *
1573          * @param {Function} setFunc       Some function which usually sets some attributes
1574          * @param {String} testAttribute If this string is the empty string  the function is called immediately,
1575          *                               otherwise it is called in a setImeout.
1576          * @see JXG.SVGRenderer#setObjectFillColor
1577          * @see JXG.SVGRenderer#setObjectStrokeColor
1578          * @see JXG.SVGRenderer#_setArrowColor
1579          * @private
1580          */
1581         _setAttribute: function (setFunc, testAttribute) {
1582             if (testAttribute === "") {
1583                 setFunc();
1584             } else {
1585                 window.setTimeout(setFunc, 1);
1586             }
1587         },
1588 
1589         // documented in JXG.AbstractRenderer
1590         setARIA: function(el) {
1591             // This method is only called in abstractRenderer._updateVisual() if aria.enabled == true.
1592             var key, k, v;
1593 
1594             // this.setPropertyPrim(el.rendNode, 'aria-label', el.evalVisProp('aria.label'));
1595             // this.setPropertyPrim(el.rendNode, 'aria-live', el.evalVisProp('aria.live'));
1596             for (key in el.visProp.aria) {
1597                 if (el.visProp.aria.hasOwnProperty(key) && key !== 'enabled') {
1598                     k = 'aria.' + key;
1599                     v = el.evalVisProp('aria.' + key);
1600                     if (el.visPropOld[k] !== v) {
1601                         this.setPropertyPrim(el.rendNode, 'aria-' + key, v);
1602                         el.visPropOld[k] = v;
1603                     }
1604                 }
1605             }
1606         },
1607 
1608         // documented in JXG.AbstractRenderer
1609         setCssClass(el, cssClass) {
1610             if (el.visPropOld.cssclass !== cssClass) {
1611                 this.setPropertyPrim(el.rendNode, 'class', cssClass);
1612                 el.visPropOld.cssclass = cssClass;
1613             }
1614         },
1615 
1616         // documented in JXG.AbstractRenderer
1617         setObjectFillColor: function (el, color, opacity, rendNode) {
1618             var node, c, rgbo, oo,
1619                 rgba = color,
1620                 o = opacity,
1621                 grad = el.evalVisProp('gradient');
1622 
1623             o = o > 0 ? o : 0;
1624 
1625             // TODO  save gradient and gradientangle
1626             if (
1627                 el.visPropOld.fillcolor === rgba &&
1628                 el.visPropOld.fillopacity === o &&
1629                 grad === null
1630             ) {
1631                 return;
1632             }
1633 
1634             if (Type.exists(rgba) && rgba !== false) {
1635                 if (rgba.length !== 9) {
1636                     // RGB, not RGBA
1637                     c = rgba;
1638                     oo = o;
1639                 } else {
1640                     // True RGBA, not RGB
1641                     rgbo = Color.rgba2rgbo(rgba);
1642                     c = rgbo[0];
1643                     oo = o * rgbo[1];
1644                 }
1645 
1646                 if (rendNode === undefined) {
1647                     node = el.rendNode;
1648                 } else {
1649                     node = rendNode;
1650                 }
1651 
1652                 if (c !== "none") {
1653                     this._setAttribute(function () {
1654                         node.setAttributeNS(null, "fill", c);
1655                     }, el.visPropOld.fillcolor);
1656                 }
1657 
1658                 if (el.type === JXG.OBJECT_TYPE_IMAGE) {
1659                     this._setAttribute(function () {
1660                         node.setAttributeNS(null, "opacity", oo);
1661                     }, el.visPropOld.fillopacity);
1662                     //node.style['opacity'] = oo;  // This would overwrite values set by CSS class.
1663                 } else {
1664                     if (c === "none") {
1665                         // This is done only for non-images
1666                         // because images have no fill color.
1667                         oo = 0;
1668                         // This is necessary if there is a foreignObject below.
1669                         node.setAttributeNS(null, "pointer-events", "visibleStroke");
1670                     } else {
1671                         // This is the default
1672                         node.setAttributeNS(null, "pointer-events", "visiblePainted");
1673                     }
1674                     this._setAttribute(function () {
1675                         node.setAttributeNS(null, "fill-opacity", oo);
1676                     }, el.visPropOld.fillopacity);
1677                 }
1678 
1679                 if (grad === "linear" || grad === "radial") {
1680                     this.updateGradient(el);
1681                 }
1682             }
1683             el.visPropOld.fillcolor = rgba;
1684             el.visPropOld.fillopacity = o;
1685         },
1686 
1687         // documented in JXG.AbstractRenderer
1688         setObjectStrokeColor: function (el, color, opacity) {
1689             var rgba = color,
1690                 c, rgbo,
1691                 o = opacity,
1692                 oo, node;
1693 
1694             o = o > 0 ? o : 0;
1695 
1696             if (el.visPropOld.strokecolor === rgba && el.visPropOld.strokeopacity === o) {
1697                 return;
1698             }
1699 
1700             if (Type.exists(rgba) && rgba !== false) {
1701                 if (rgba.length !== 9) {
1702                     // RGB, not RGBA
1703                     c = rgba;
1704                     oo = o;
1705                 } else {
1706                     // True RGBA, not RGB
1707                     rgbo = Color.rgba2rgbo(rgba);
1708                     c = rgbo[0];
1709                     oo = o * rgbo[1];
1710                 }
1711 
1712                 node = el.rendNode;
1713 
1714                 if (el.elementClass === Const.OBJECT_CLASS_TEXT) {
1715                     if (el.evalVisProp('display') === "html") {
1716                         this._setAttribute(function () {
1717                             node.style.color = c;
1718                             node.style.opacity = oo;
1719                         }, el.visPropOld.strokecolor);
1720                     } else {
1721                         this._setAttribute(function () {
1722                             node.setAttributeNS(null, "style", "fill:" + c);
1723                             node.setAttributeNS(null, "style", "fill-opacity:" + oo);
1724                         }, el.visPropOld.strokecolor);
1725                     }
1726                 } else {
1727                     this._setAttribute(function () {
1728                         node.setAttributeNS(null, "stroke", c);
1729                         node.setAttributeNS(null, "stroke-opacity", oo);
1730                     }, el.visPropOld.strokecolor);
1731                 }
1732 
1733                 if (
1734                     el.elementClass === Const.OBJECT_CLASS_CURVE ||
1735                     el.elementClass === Const.OBJECT_CLASS_LINE
1736                 ) {
1737                     if (el.evalVisProp('firstarrow')) {
1738                         this._setArrowColor(
1739                             el.rendNodeTriangleStart,
1740                             c, oo, el,
1741                             el.visPropCalc.typeFirst
1742                         );
1743                     }
1744 
1745                     if (el.evalVisProp('lastarrow')) {
1746                         this._setArrowColor(
1747                             el.rendNodeTriangleEnd,
1748                             c, oo, el,
1749                             el.visPropCalc.typeLast
1750                         );
1751                     }
1752                 }
1753             }
1754 
1755             el.visPropOld.strokecolor = rgba;
1756             el.visPropOld.strokeopacity = o;
1757         },
1758 
1759         // documented in JXG.AbstractRenderer
1760         setObjectStrokeWidth: function (el, width) {
1761             var node,
1762                 w = width;
1763 
1764             if (isNaN(w) || el.visPropOld.strokewidth === w) {
1765                 return;
1766             }
1767 
1768             node = el.rendNode;
1769             this.setPropertyPrim(node, "stroked", "true");
1770             if (Type.exists(w)) {
1771                 this.setPropertyPrim(node, "stroke-width", w + "px");
1772 
1773                 // if (el.elementClass === Const.OBJECT_CLASS_CURVE ||
1774                 // el.elementClass === Const.OBJECT_CLASS_LINE) {
1775                 //     if (el.evalVisProp('firstarrow')) {
1776                 //         this._setArrowWidth(el.rendNodeTriangleStart, w, el.rendNode);
1777                 //     }
1778                 //
1779                 //     if (el.evalVisProp('lastarrow')) {
1780                 //         this._setArrowWidth(el.rendNodeTriangleEnd, w, el.rendNode);
1781                 //     }
1782                 // }
1783             }
1784             el.visPropOld.strokewidth = w;
1785         },
1786 
1787         // documented in JXG.AbstractRenderer
1788         setLineCap: function (el) {
1789             var capStyle = el.evalVisProp('linecap');
1790 
1791             if (
1792                 capStyle === undefined ||
1793                 capStyle === "" ||
1794                 el.visPropOld.linecap === capStyle ||
1795                 !Type.exists(el.rendNode)
1796             ) {
1797                 return;
1798             }
1799 
1800             this.setPropertyPrim(el.rendNode, "stroke-linecap", capStyle);
1801             el.visPropOld.linecap = capStyle;
1802         },
1803 
1804         // documented in JXG.AbstractRenderer
1805         setShadow: function (el) {
1806             var ev_s = el.evalVisProp('shadow'),
1807                 ev_s_json, c, b, bl, o, op, id, node,
1808                 use_board_filter = true,
1809                 show = false;
1810 
1811             ev_s_json = JSON.stringify(ev_s);
1812             if (ev_s_json === el.visPropOld.shadow) {
1813                 return;
1814             }
1815 
1816             if (typeof ev_s === 'boolean') {
1817                 use_board_filter = true;
1818                 show = ev_s;
1819                 c = 'none';
1820                 b = 3;
1821                 bl = 0.1;
1822                 o = [5, 5];
1823                 op = 1;
1824             } else {
1825                 if (el.evalVisProp('shadow.enabled')) {
1826                     use_board_filter = false;
1827                     show = true;
1828                     c = JXG.rgbParser(el.evalVisProp('shadow.color'));
1829                     b = el.evalVisProp('shadow.blur');
1830                     bl = el.evalVisProp('shadow.blend');
1831                     o = el.evalVisProp('shadow.offset');
1832                     op = el.evalVisProp('shadow.opacity');
1833                 } else {
1834                     show = false;
1835                 }
1836             }
1837 
1838             if (Type.exists(el.rendNode)) {
1839                 if (show) {
1840                     if (use_board_filter) {
1841                         el.rendNode.setAttributeNS(null, 'filter', this.toURL(this.container.id + '_' + 'f1'));
1842                         // 'url(#' + this.container.id + '_' + 'f1)');
1843                     } else {
1844                         node = this.container.ownerDocument.getElementById(id);
1845                         if (node) {
1846                             this.defs.removeChild(node);
1847                         }
1848                         id = el.rendNode.id + '_' + 'f1';
1849                         this.defs.appendChild(this.createShadowFilter(id, c, op, bl, b, o));
1850                         el.rendNode.setAttributeNS(null, 'filter', this.toURL(id));
1851                         // 'url(#' + id + ')');
1852                     }
1853                 } else {
1854                     el.rendNode.removeAttributeNS(null, 'filter');
1855                 }
1856             }
1857 
1858             el.visPropOld.shadow = ev_s_json;
1859         },
1860 
1861         /* **************************
1862          * renderer control
1863          * **************************/
1864 
1865         // documented in JXG.AbstractRenderer
1866         suspendRedraw: function () {
1867             // It seems to be important for the Linux version of firefox
1868             this.suspendHandle = this.svgRoot.suspendRedraw(10000);
1869         },
1870 
1871         // documented in JXG.AbstractRenderer
1872         unsuspendRedraw: function () {
1873             this.svgRoot.unsuspendRedraw(this.suspendHandle);
1874             // this.svgRoot.unsuspendRedrawAll();
1875             //this.svgRoot.forceRedraw();
1876         },
1877 
1878         // documented in AbstractRenderer
1879         resize: function (w, h) {
1880             this.svgRoot.setAttribute("width", parseFloat(w));
1881             this.svgRoot.setAttribute("height", parseFloat(h));
1882         },
1883 
1884         // documented in JXG.AbstractRenderer
1885         createTouchpoints: function (n) {
1886             var i, na1, na2, node;
1887             this.touchpoints = [];
1888             for (i = 0; i < n; i++) {
1889                 na1 = "touchpoint1_" + i;
1890                 node = this.createPrim("path", na1);
1891                 this.appendChildPrim(node, 19);
1892                 node.setAttributeNS(null, "d", "M 0 0");
1893                 this.touchpoints.push(node);
1894 
1895                 this.setPropertyPrim(node, "stroked", "true");
1896                 this.setPropertyPrim(node, "stroke-width", "1px");
1897                 node.setAttributeNS(null, "stroke", "#000000");
1898                 node.setAttributeNS(null, "stroke-opacity", 1.0);
1899                 node.setAttributeNS(null, "display", "none");
1900 
1901                 na2 = "touchpoint2_" + i;
1902                 node = this.createPrim("ellipse", na2);
1903                 this.appendChildPrim(node, 19);
1904                 this.updateEllipsePrim(node, 0, 0, 0, 0);
1905                 this.touchpoints.push(node);
1906 
1907                 this.setPropertyPrim(node, "stroked", "true");
1908                 this.setPropertyPrim(node, "stroke-width", "1px");
1909                 node.setAttributeNS(null, "stroke", "#000000");
1910                 node.setAttributeNS(null, "stroke-opacity", 1.0);
1911                 node.setAttributeNS(null, "fill", "#ffffff");
1912                 node.setAttributeNS(null, "fill-opacity", 0.0);
1913 
1914                 node.setAttributeNS(null, "display", "none");
1915             }
1916         },
1917 
1918         // documented in JXG.AbstractRenderer
1919         showTouchpoint: function (i) {
1920             if (this.touchpoints && i >= 0 && 2 * i < this.touchpoints.length) {
1921                 this.touchpoints[2 * i].setAttributeNS(null, "display", "inline");
1922                 this.touchpoints[2 * i + 1].setAttributeNS(null, "display", "inline");
1923             }
1924         },
1925 
1926         // documented in JXG.AbstractRenderer
1927         hideTouchpoint: function (i) {
1928             if (this.touchpoints && i >= 0 && 2 * i < this.touchpoints.length) {
1929                 this.touchpoints[2 * i].setAttributeNS(null, "display", "none");
1930                 this.touchpoints[2 * i + 1].setAttributeNS(null, "display", "none");
1931             }
1932         },
1933 
1934         // documented in JXG.AbstractRenderer
1935         updateTouchpoint: function (i, pos) {
1936             var x,
1937                 y,
1938                 d = 37;
1939 
1940             if (this.touchpoints && i >= 0 && 2 * i < this.touchpoints.length) {
1941                 x = pos[0];
1942                 y = pos[1];
1943 
1944                 this.touchpoints[2 * i].setAttributeNS(
1945                     null,
1946                     "d",
1947                     "M " +
1948                     (x - d) +
1949                     " " +
1950                     y +
1951                     " " +
1952                     "L " +
1953                     (x + d) +
1954                     " " +
1955                     y +
1956                     " " +
1957                     "M " +
1958                     x +
1959                     " " +
1960                     (y - d) +
1961                     " " +
1962                     "L " +
1963                     x +
1964                     " " +
1965                     (y + d)
1966                 );
1967                 this.updateEllipsePrim(this.touchpoints[2 * i + 1], pos[0], pos[1], 25, 25);
1968             }
1969         },
1970 
1971         /**
1972          * Walk recursively through the DOM subtree of a node and collect all
1973          * value attributes together with the id of that node.
1974          * <b>Attention:</b> Only values of nodes having a valid id are taken.
1975          * @param  {Node} node   root node of DOM subtree that will be searched recursively.
1976          * @return {Array}      Array with entries of the form [id, value]
1977          * @private
1978          */
1979         _getValuesOfDOMElements: function (node) {
1980             var values = [];
1981             if (node.nodeType === 1) {
1982                 node = node.firstChild;
1983                 while (node) {
1984                     if (node.id !== undefined && node.value !== undefined) {
1985                         values.push([node.id, node.value]);
1986                     }
1987                     Type.concat(values, this._getValuesOfDOMElements(node));
1988                     node = node.nextSibling;
1989                 }
1990             }
1991             return values;
1992         },
1993 
1994         // _getDataUri: function (url, callback) {
1995         //     var image = new Image();
1996         //     image.onload = function () {
1997         //         var canvas = document.createElement("canvas");
1998         //         canvas.width = this.naturalWidth; // or 'width' if you want a special/scaled size
1999         //         canvas.height = this.naturalHeight; // or 'height' if you want a special/scaled size
2000         //         canvas.getContext("2d").drawImage(this, 0, 0);
2001         //         callback(canvas.toDataURL("image/png"));
2002         //         canvas.remove();
2003         //     };
2004         //     image.src = url;
2005         // },
2006 
2007         _getImgDataURL: function (svgRoot) {
2008             var images, len, canvas, ctx, ur, i;
2009 
2010             images = svgRoot.getElementsByTagName("image");
2011             len = images.length;
2012             if (len > 0) {
2013                 canvas = document.createElement("canvas");
2014                 //img = new Image();
2015                 for (i = 0; i < len; i++) {
2016                     images[i].setAttribute("crossorigin", "anonymous");
2017                     //img.src = images[i].href;
2018                     //img.onload = function() {
2019                     // img.crossOrigin = "anonymous";
2020                     ctx = canvas.getContext("2d");
2021                     canvas.width = images[i].getAttribute("width");
2022                     canvas.height = images[i].getAttribute("height");
2023                     try {
2024                         ctx.drawImage(images[i], 0, 0, canvas.width, canvas.height);
2025 
2026                         // If the image is not png, the format must be specified here
2027                         ur = canvas.toDataURL();
2028                         images[i].setAttribute("xlink:href", ur);
2029                     } catch (err) {
2030                         console.log("CORS problem! Image can not be used", err);
2031                     }
2032                 }
2033                 //canvas.remove();
2034             }
2035             return true;
2036         },
2037 
2038         /**
2039          * Return a data URI of the SVG code representing the construction.
2040          * The SVG code of the construction is base64 encoded. The return string starts
2041          * with "data:image/svg+xml;base64,...".
2042          *
2043          * @param {Boolean} ignoreTexts If true, the foreignObject tag is set to display=none.
2044          * This is necessary for older versions of Safari. Default: false
2045          * @returns {String}  data URI string
2046          *
2047          * @example
2048          * var A = board.create('point', [2, 2]);
2049          *
2050          * var txt = board.renderer.dumpToDataURI(false);
2051          * // txt consists of a string of the form
2052          * // . base64 encoded SVG..+PC9zdmc+
2053          * // Behind the comma, there is the base64 encoded SVG code
2054          * // which is decoded with atob().
2055          * // The call of decodeURIComponent(escape(...)) is necessary
2056          * // to handle unicode strings correctly.
2057          * var ar = txt.split(',');
2058          * document.getElementById('output').value = decodeURIComponent(escape(atob(ar[1])));
2059          *
2060          * </pre><div id="JXG1bad4bec-6d08-4ce0-9b7f-d817e8dd762d" class="jxgbox" style="width: 300px; height: 300px;"></div>
2061          * <textarea id="output2023" rows="5" cols="50"></textarea>
2062          * <script type="text/javascript">
2063          *     (function() {
2064          *         var board = JXG.JSXGraph.initBoard('JXG1bad4bec-6d08-4ce0-9b7f-d817e8dd762d',
2065          *             {boundingbox: [-8, 8, 8,-8], axis: true, showcopyright: false, shownavigation: false});
2066          *     var A = board.create('point', [2, 2]);
2067          *
2068          *     var txt = board.renderer.dumpToDataURI(false);
2069          *     // txt consists of a string of the form
2070          *     // . base64 encoded SVG..+PC9zdmc+
2071          *     // Behind the comma, there is the base64 encoded SVG code
2072          *     // which is decoded with atob().
2073          *     // The call of decodeURIComponent(escape(...)) is necessary
2074          *     // to handle unicode strings correctly.
2075          *     var ar = txt.split(',');
2076          *     document.getElementById('output2023').value = decodeURIComponent(escape(atob(ar[1])));
2077          *
2078          *     })();
2079          *
2080          * </script><pre>
2081          *
2082          */
2083         dumpToDataURI: function (ignoreTexts) {
2084             var svgRoot = this.svgRoot,
2085                 btoa = window.btoa || Base64.encode,
2086                 svg, i, len,
2087                 values = [];
2088 
2089             // Move all HTML tags (beside the SVG root) of the container
2090             // to the foreignObject element inside of the svgRoot node
2091             // Problem:
2092             // input values are not copied. This can be verified by looking at an innerHTML output
2093             // of an input element. Therefore, we do it "by hand".
2094             if (this.container.hasChildNodes() && Type.exists(this.foreignObjLayer)) {
2095                 if (!ignoreTexts) {
2096                     this.foreignObjLayer.setAttribute("display", "inline");
2097                 }
2098                 while (svgRoot.nextSibling) {
2099                     // Copy all value attributes
2100                     Type.concat(values, this._getValuesOfDOMElements(svgRoot.nextSibling));
2101                     this.foreignObjLayer.appendChild(svgRoot.nextSibling);
2102                 }
2103             }
2104 
2105             this._getImgDataURL(svgRoot);
2106 
2107             // Convert the SVG graphic into a string containing SVG code
2108             svgRoot.setAttribute("xmlns", "http://www.w3.org/2000/svg");
2109             svg = new XMLSerializer().serializeToString(svgRoot);
2110 
2111             if (ignoreTexts !== true) {
2112                 // Handle SVG texts
2113                 // Insert all value attributes back into the svg string
2114                 len = values.length;
2115                 for (i = 0; i < len; i++) {
2116                     svg = svg.replace(
2117                         'id="' + values[i][0] + '"',
2118                         'id="' + values[i][0] + '" value="' + values[i][1] + '"'
2119                     );
2120                 }
2121             }
2122 
2123             // if (false) {
2124             //     // Debug: use example svg image
2125             //     svg = '<svg xmlns="http://www.w3.org/2000/svg" version="1.0" width="220" height="220"><rect width="66" height="30" x="21" y="32" stroke="#204a87" stroke-width="2" fill="none" /></svg>';
2126             // }
2127 
2128             // In IE we have to remove the namespace again.
2129             // Since 2024 we have to check if the namespace attribute appears twice in one tag, because
2130             // there might by a svg inside of the svg, e.g. the screenshot icon.
2131             if (this.isIE &&
2132                 (svg.match(/xmlns="http:\/\/www.w3.org\/2000\/svg"\s+xmlns="http:\/\/www.w3.org\/2000\/svg"/g) || []).length > 1
2133             ) {
2134                 svg = svg.replace(/xmlns="http:\/\/www.w3.org\/2000\/svg"\s+xmlns="http:\/\/www.w3.org\/2000\/svg"/g, "");
2135             }
2136 
2137             // Safari fails if the svg string contains a " "
2138             // Obsolete with Safari 12+
2139             svg = svg.replace(/ /g, " ");
2140             svg = svg.replace(/url\("(.*)"\)/g, "url($1)");
2141 
2142             // Move all HTML tags back from
2143             // the foreignObject element to the container
2144             if (Type.exists(this.foreignObjLayer) && this.foreignObjLayer.hasChildNodes()) {
2145                 // Restore all HTML elements
2146                 while (this.foreignObjLayer.firstChild) {
2147                     this.container.appendChild(this.foreignObjLayer.firstChild);
2148                 }
2149                 this.foreignObjLayer.setAttribute("display", "none");
2150             }
2151 
2152             return "data:image/svg+xml;base64," + btoa(unescape(encodeURIComponent(svg)));
2153         },
2154 
2155         /**
2156          * Convert the SVG construction into an HTML canvas image.
2157          * This works for all SVG supporting browsers. Implemented as Promise.
2158          * <p>
2159          * Might fail if any text element or foreign object element contains SVG. This
2160          * is the case e.g. for the default fullscreen symbol.
2161          * <p>
2162          * For IE, it is realized as function.
2163          * It works from version 9, with the exception that HTML texts
2164          * are ignored on IE. The drawing is done with a delay of
2165          * 200 ms. Otherwise there would be problems with IE.
2166          *
2167          * @param {String} canvasId Id of an HTML canvas element
2168          * @param {Number} w Width in pixel of the dumped image, i.e. of the canvas tag.
2169          * @param {Number} h Height in pixel of the dumped image, i.e. of the canvas tag.
2170          * @param {Boolean} ignoreTexts If true, the foreignObject tag is taken out from the SVG root.
2171          * This is necessary for older versions of Safari. Default: false
2172          * @returns {Promise}  Promise object
2173          *
2174          * @example
2175          * 	board.renderer.dumpToCanvas('canvas').then(function() { console.log('done'); });
2176          *
2177          * @example
2178          *  // IE 11 example:
2179          * 	board.renderer.dumpToCanvas('canvas');
2180          * 	setTimeout(function() { console.log('done'); }, 400);
2181          */
2182         dumpToCanvas: function (canvasId, w, h, ignoreTexts) {
2183             var svg, tmpImg,
2184                 cv, ctx,
2185                 doc = this.container.ownerDocument;
2186 
2187             // Prepare the canvas element
2188             cv = doc.getElementById(canvasId);
2189 
2190             // Clear the canvas
2191             /* eslint-disable no-self-assign */
2192             cv.width = cv.width;
2193             /* eslint-enable no-self-assign */
2194 
2195             ctx = cv.getContext("2d");
2196             if (w !== undefined && h !== undefined) {
2197                 cv.style.width = parseFloat(w) + "px";
2198                 cv.style.height = parseFloat(h) + "px";
2199                 // Scale twice the CSS size to make the image crisp
2200                 // cv.setAttribute('width', 2 * parseFloat(wOrg));
2201                 // cv.setAttribute('height', 2 * parseFloat(hOrg));
2202                 // ctx.scale(2 * wOrg / w, 2 * hOrg / h);
2203                 cv.setAttribute("width", parseFloat(w));
2204                 cv.setAttribute("height", parseFloat(h));
2205             }
2206 
2207             // Display the SVG string as data-uri in an HTML img.
2208             /**
2209              * @type {Image}
2210              * @ignore
2211              * {ignore}
2212              */
2213             tmpImg = new Image();
2214             svg = this.dumpToDataURI(ignoreTexts);
2215             tmpImg.src = svg;
2216 
2217             // Finally, draw the HTML img in the canvas.
2218             if (!("Promise" in window)) {
2219                 /**
2220                  * @function
2221                  * @ignore
2222                  */
2223                 tmpImg.onload = function () {
2224                     // IE needs a pause...
2225                     // Seems to be broken
2226                     window.setTimeout(function () {
2227                         try {
2228                             ctx.drawImage(tmpImg, 0, 0, w, h);
2229                         } catch (err) {
2230                             console.log("screenshots not longer supported on IE");
2231                         }
2232                     }, 200);
2233                 };
2234                 return this;
2235             }
2236 
2237             return new Promise(function (resolve, reject) {
2238                 try {
2239                     tmpImg.onload = function () {
2240                         ctx.drawImage(tmpImg, 0, 0, w, h);
2241                         resolve();
2242                     };
2243                 } catch (e) {
2244                     reject(e);
2245                 }
2246             });
2247         },
2248 
2249         /**
2250          * Display SVG image in html img-tag which enables
2251          * easy download for the user.
2252          *
2253          * Support:
2254          * <ul>
2255          * <li> IE: No
2256          * <li> Edge: full
2257          * <li> Firefox: full
2258          * <li> Chrome: full
2259          * <li> Safari: full (No text support in versions prior to 12).
2260          * </ul>
2261          *
2262          * @param {JXG.Board} board Link to the board.
2263          * @param {String} imgId Optional id of an img object. If given and different from the empty string,
2264          * the screenshot is copied to this img object. The width and height will be set to the values of the
2265          * JSXGraph container.
2266          * @param {Boolean} ignoreTexts If set to true, the foreignObject is taken out of the
2267          *  SVGRoot and texts are not displayed. This is mandatory for Safari. Default: false
2268          * @return {Object}       the svg renderer object
2269          */
2270         screenshot: function (board, imgId, ignoreTexts) {
2271             var node,
2272                 doc = this.container.ownerDocument,
2273                 parent = this.container.parentNode,
2274                 // cPos,
2275                 // cssTxt,
2276                 canvas, id, img,
2277                 button, buttonText,
2278                 w, h,
2279                 bas = board.attr.screenshot,
2280                 navbar, navbarDisplay, insert,
2281                 newImg = false,
2282                 _copyCanvasToImg,
2283                 isDebug = false;
2284 
2285             if (this.type === "no") {
2286                 return this;
2287             }
2288 
2289             w = bas.scale * this.container.getBoundingClientRect().width;
2290             h = bas.scale * this.container.getBoundingClientRect().height;
2291 
2292             if (imgId === undefined || imgId === "") {
2293                 newImg = true;
2294                 img = new Image(); //doc.createElement('img');
2295                 img.style.width = w + "px";
2296                 img.style.height = h + "px";
2297             } else {
2298                 newImg = false;
2299                 img = doc.getElementById(imgId);
2300             }
2301             // img.crossOrigin = 'anonymous';
2302 
2303             // Create div which contains canvas element and close button
2304             if (newImg) {
2305                 node = doc.createElement("div");
2306                 node.style.cssText = bas.css;
2307                 node.style.width = w + "px";
2308                 node.style.height = h + "px";
2309                 node.style.zIndex = this.container.style.zIndex + 120;
2310 
2311                 // Try to position the div exactly over the JSXGraph board
2312                 node.style.position = "absolute";
2313                 node.style.top = this.container.offsetTop + "px";
2314                 node.style.left = this.container.offsetLeft + "px";
2315             }
2316 
2317             if (!isDebug) {
2318                 // Create canvas element and add it to the DOM
2319                 // It will be removed after the image has been stored.
2320                 canvas = doc.createElement("canvas");
2321                 id = Math.random().toString(36).slice(2, 7);
2322                 canvas.setAttribute("id", id);
2323                 canvas.setAttribute("width", w);
2324                 canvas.setAttribute("height", h);
2325                 canvas.style.width = w + "px";
2326                 canvas.style.height = w + "px";
2327                 canvas.style.display = "none";
2328                 parent.appendChild(canvas);
2329             } else {
2330                 // Debug: use canvas element 'jxgbox_canvas' from jsxdev/dump.html
2331                 id = "jxgbox_canvas";
2332                 canvas = doc.getElementById(id);
2333             }
2334 
2335             if (newImg) {
2336                 // Create close button
2337                 button = doc.createElement("span");
2338                 buttonText = doc.createTextNode("\u2716");
2339                 button.style.cssText = bas.cssButton;
2340                 button.appendChild(buttonText);
2341                 button.onclick = function () {
2342                     node.parentNode.removeChild(node);
2343                 };
2344 
2345                 // Add all nodes
2346                 node.appendChild(img);
2347                 node.appendChild(button);
2348                 parent.insertBefore(node, this.container.nextSibling);
2349             }
2350 
2351             // Hide navigation bar in board
2352             navbar = doc.getElementById(this.uniqName('navigationbar'));
2353             if (Type.exists(navbar)) {
2354                 navbarDisplay = navbar.style.display;
2355                 navbar.style.display = "none";
2356                 insert = this.removeToInsertLater(navbar);
2357             }
2358 
2359             _copyCanvasToImg = function () {
2360                 // Show image in img tag
2361                 img.src = canvas.toDataURL("image/png");
2362 
2363                 // Remove canvas node
2364                 if (!isDebug) {
2365                     parent.removeChild(canvas);
2366                 }
2367             };
2368 
2369             // Create screenshot in image element
2370             if ("Promise" in window) {
2371                 this.dumpToCanvas(id, w, h, ignoreTexts).then(_copyCanvasToImg);
2372             } else {
2373                 // IE
2374                 this.dumpToCanvas(id, w, h, ignoreTexts);
2375                 window.setTimeout(_copyCanvasToImg, 200);
2376             }
2377 
2378             // Reinsert navigation bar in board
2379             if (Type.exists(navbar)) {
2380                 navbar.style.display = navbarDisplay;
2381                 insert();
2382             }
2383 
2384             return this;
2385         }
2386     }
2387 );
2388 
2389 export default JXG.SVGRenderer;
2390