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