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