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