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