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, window: true*/ 33 /*jslint nomen: true, plusplus: true*/ 34 35 /** 36 * @fileoverview In this file the ForeignObject element is defined. 37 */ 38 39 import JXG from "../jxg.js"; 40 import Const from "./constants.js"; 41 import Coords from "./coords.js"; 42 import GeometryElement from "./element.js"; 43 import Mat from "../math/math.js"; 44 import Type from "../utils/type.js"; 45 import CoordsElement from "./coordselement.js"; 46 47 /** 48 * Construct and handle SVG foreignObjects. 49 * 50 * @class Creates a new foreignObject object. Do not use this constructor to create a foreignObject. Use {@link JXG.Board#create} with 51 * type {@link foreignobject} instead. 52 * @augments JXG.GeometryElement 53 * @augments JXG.CoordsElement 54 * @param {string|JXG.Board} board The board the new foreignObject is drawn on. 55 * @param {Array} coordinates An array with the user coordinates of the foreignObject. 56 * @param {Object} attributes An object containing visual and - optionally - a name and an id. 57 * @param {string|function} url An URL string or a function returning an URL string. 58 * @param {Array} size Array containing width and height of the foreignObject in user coordinates. 59 * 60 */ 61 JXG.ForeignObject = function (board, coords, attributes, content, size) { 62 this.constructor( 63 board, 64 attributes, 65 Const.OBJECT_TYPE_FOREIGNOBJECT, 66 Const.OBJECT_CLASS_OTHER 67 ); 68 this.element = this.board.select(attributes.anchor); 69 this.coordsConstructor(coords); 70 71 this._useUserSize = false; 72 73 /** 74 * Array of length two containing [width, height] of the foreignObject in pixel. 75 * @type Array 76 */ 77 this.size = [1, 1]; 78 if (Type.exists(size) && size.length > 0) { 79 this._useUserSize = true; 80 81 this.W = Type.createFunction(size[0], this.board, ""); 82 this.H = Type.createFunction(size[1], this.board, ""); 83 this.addParentsFromJCFunctions([this.W, this.H]); 84 85 this.usrSize = [this.W(), this.H()]; 86 } 87 88 // this.size = [Math.abs(this.usrSize[0] * board.unitX), Math.abs(this.usrSize[1] * board.unitY)]; 89 90 /** 91 * 'href' of the foreignObject. 92 * @type {string} 93 */ 94 this.content = content; 95 96 this.elType = "foreignobject"; 97 98 // span contains the anchor point and the two vectors 99 // spanning the foreignObject rectangle. 100 // this.span = [ 101 // this.coords.usrCoords.slice(0), 102 // [this.coords.usrCoords[0], this.W(), 0], 103 // [this.coords.usrCoords[0], 0, this.H()] 104 // ]; 105 //this.parent = board.select(attributes.anchor); 106 107 this.id = this.board.setId(this, "Im"); 108 109 this.board.renderer.drawForeignObject(this); 110 this.board.finalizeAdding(this); 111 112 this.methodMap = JXG.deepCopy(this.methodMap, { 113 addTransformation: "addTransform", 114 trans: "addTransform", 115 W: "W", 116 Width: "W", 117 H: "H", 118 Height: "H" 119 }); 120 }; 121 122 JXG.ForeignObject.prototype = new GeometryElement(); 123 Type.copyPrototypeMethods(JXG.ForeignObject, CoordsElement, "coordsConstructor"); 124 125 JXG.extend( 126 JXG.ForeignObject.prototype, 127 /** @lends JXG.ForeignObject.prototype */ { 128 /** 129 * Checks whether (x,y) is over or near the image; 130 * @param {Number} x Coordinate in x direction, screen coordinates. 131 * @param {Number} y Coordinate in y direction, screen coordinates. 132 * @returns {Boolean} True if (x,y) is over the image, False otherwise. 133 */ 134 hasPoint: function (x, y) { 135 var dx, dy, r, type, prec, c, v, p, dot, 136 len = this.transformations.length; 137 138 if (Type.isObject(Type.evaluate(this.visProp.precision))) { 139 type = this.board._inputDevice; 140 prec = Type.evaluate(this.visProp.precision[type]); 141 } else { 142 // 'inherit' 143 prec = this.board.options.precision.hasPoint; 144 } 145 146 // Easy case: no transformation 147 if (len === 0) { 148 dx = x - this.coords.scrCoords[1]; 149 dy = this.coords.scrCoords[2] - y; 150 r = prec; 151 152 return dx >= -r && dx - this.size[0] <= r && dy >= -r && dy - this.size[1] <= r; 153 } 154 155 // foreignObject is transformed 156 c = new Coords(Const.COORDS_BY_SCREEN, [x, y], this.board); 157 // v is the vector from anchor point to the drag point 158 c = c.usrCoords; 159 v = [c[0] - this.span[0][0], c[1] - this.span[0][1], c[2] - this.span[0][2]]; 160 dot = Mat.innerProduct; // shortcut 161 162 // Project the drag point to the sides. 163 p = dot(v, this.span[1]); 164 if (0 <= p && p <= dot(this.span[1], this.span[1])) { 165 p = dot(v, this.span[2]); 166 167 if (0 <= p && p <= dot(this.span[2], this.span[2])) { 168 return true; 169 } 170 } 171 return false; 172 }, 173 174 /** 175 * Recalculate the coordinates of lower left corner and the width and height. 176 * 177 * @returns {JXG.ForeignObject} A reference to the element 178 * @private 179 */ 180 update: function (fromParent) { 181 if (!this.needsUpdate) { 182 return this; 183 } 184 this.updateCoords(fromParent); 185 this.updateSize(); 186 // this.updateSpan(); 187 return this; 188 }, 189 190 /** 191 * Send an update request to the renderer. 192 * @private 193 */ 194 updateRenderer: function () { 195 return this.updateRendererGeneric("updateForeignObject"); 196 }, 197 198 /** 199 * Updates the internal arrays containing size of the foreignObject. 200 * @returns {JXG.ForeignObject} A reference to the element 201 * @private 202 */ 203 updateSize: function () { 204 var bb = [0, 0]; 205 206 if (this._useUserSize) { 207 this.usrSize = [this.W(), this.H()]; 208 this.size = [ 209 Math.abs(this.usrSize[0] * this.board.unitX), 210 Math.abs(this.usrSize[1] * this.board.unitY) 211 ]; 212 } else { 213 if (this.rendNode.hasChildNodes()) { 214 bb = this.rendNode.childNodes[0].getBoundingClientRect(); 215 this.size = [bb.width, bb.height]; 216 } 217 } 218 219 return this; 220 }, 221 222 /** 223 * Update the anchor point of the foreignObject, i.e. the lower left corner 224 * and the two vectors which span the rectangle. 225 * @returns {JXG.ForeignObject} A reference to the element 226 * @private 227 * 228 */ 229 updateSpan: function () { 230 var i, 231 j, 232 len = this.transformations.length, 233 v = []; 234 235 if (len === 0) { 236 this.span = [ 237 [this.Z(), this.X(), this.Y()], 238 [this.Z(), this.W(), 0], 239 [this.Z(), 0, this.H()] 240 ]; 241 } else { 242 // v contains the three defining corners of the rectangle/image 243 v[0] = [this.Z(), this.X(), this.Y()]; 244 v[1] = [this.Z(), this.X() + this.W(), this.Y()]; 245 v[2] = [this.Z(), this.X(), this.Y() + this.H()]; 246 247 // Transform the three corners 248 for (i = 0; i < len; i++) { 249 for (j = 0; j < 3; j++) { 250 v[j] = Mat.matVecMult(this.transformations[i].matrix, v[j]); 251 } 252 } 253 // Normalize the vectors 254 for (j = 0; j < 3; j++) { 255 v[j][1] /= v[j][0]; 256 v[j][2] /= v[j][0]; 257 v[j][0] /= v[j][0]; 258 } 259 // Compute the two vectors spanning the rectangle 260 // by subtracting the anchor point. 261 for (j = 1; j < 3; j++) { 262 v[j][0] -= v[0][0]; 263 v[j][1] -= v[0][1]; 264 v[j][2] -= v[0][2]; 265 } 266 this.span = v; 267 } 268 269 return this; 270 }, 271 272 addTransform: function (transform) { 273 var i; 274 275 if (Type.isArray(transform)) { 276 for (i = 0; i < transform.length; i++) { 277 this.transformations.push(transform[i]); 278 } 279 } else { 280 this.transformations.push(transform); 281 } 282 283 return this; 284 }, 285 286 // Documented in element.js 287 getParents: function () { 288 var p = [this.url, [this.Z(), this.X(), this.Y()], this.usrSize]; 289 290 if (this.parents.length !== 0) { 291 p = this.parents; 292 } 293 294 return p; 295 }, 296 297 /** 298 * Set the width and height of the foreignObject. After setting a new size, 299 * board.update() or foreignobject.fullUpdate() 300 * has to be called to make the change visible. 301 * @param {numbe|function|string} width Number, function or string 302 * that determines the new width of the foreignObject 303 * @param {number|function|string} height Number, function or string 304 * that determines the new height of the foreignObject 305 * @returns {JXG.ForeignObject} A reference to the element 306 * 307 */ 308 setSize: function (width, height) { 309 this.W = Type.createFunction(width, this.board, ""); 310 this.H = Type.createFunction(height, this.board, ""); 311 this._useUserSize = true; 312 this.addParentsFromJCFunctions([this.W, this.H]); 313 314 return this; 315 }, 316 317 /** 318 * Returns the width of the foreignObject in user coordinates. 319 * @returns {number} width of the image in user coordinates 320 */ 321 W: function () {}, // Needed for docs, defined in constructor 322 323 /** 324 * Returns the height of the foreignObject in user coordinates. 325 * @returns {number} height of the image in user coordinates 326 */ 327 H: function () {} // Needed for docs, defined in constructor 328 } 329 ); 330 331 /** 332 * @class This element is used to provide a constructor for arbitrary content in 333 * an SVG foreignObject container. 334 * <p> 335 * Instead of board.create('foreignobject') the shortcut board.create('fo') may be used. 336 * 337 * <p style="background-color:#dddddd; padding:10px"><b>NOTE:</b> In Safari up to version 15, a foreignObject does not obey the layer structure 338 * if it contains <video> or <iframe> tags, as well as elements which are 339 * positioned with <tt>position:absolute|relative|fixed</tt>. In this case, the foreignobject will be 340 * "above" the JSXGraph construction. 341 * </p> 342 * 343 * @pseudo 344 * @name ForeignObject 345 * @augments JXG.ForeignObject 346 * @constructor 347 * @type JXG.ForeignObject 348 * 349 * @param {String} content HTML content of the foreignObject. May also be <video> or <iframe> 350 * @param {Array} position Position of the foreignObject given by [x, y] in user coordinates. Same as for images. 351 * @param {Array} [size] (Optional) argument size of the foreignObject in user coordinates. If not given, size is specified by the HTML attributes 352 * or CSS properties of the content. 353 * 354 * @see Image 355 * 356 * @example 357 * var p = board.create('point', [1, 7], {size: 16}); 358 * var fo = board.create('foreignobject', [ 359 * '<video width="300" height="200" src="https://eucbeniki.sio.si/vega2/278/Video_metanje_oge_.mp4" type="html5video" controls>', 360 * [0, -3], [9, 6]], 361 * {layer: 8, fixed: true} 362 * ); 363 * 364 * </pre><div id="JXG0c122f2c-3671-4a28-80a9-f4c523eeda89" class="jxgbox" style="width: 500px; height: 500px;"></div> 365 * <script type="text/javascript"> 366 * (function() { 367 * var board = JXG.JSXGraph.initBoard('JXG0c122f2c-3671-4a28-80a9-f4c523eeda89', 368 * {boundingbox: [-8, 8, 8,-8], axis: true, showcopyright: false, shownavigation: false}); 369 * var p = board.create('point', [1, 7], {size: 16}); 370 * var fo = board.create('foreignobject', [ 371 * '<video width="300" height="200" src="https://eucbeniki.sio.si/vega2/278/Video_metanje_oge_.mp4" type="html5video" controls>', 372 * [0, -3], [9, 6]], 373 * {layer: 8, fixed: true} 374 * ); 375 * 376 * })(); 377 * 378 * </script><pre> 379 * 380 * @example 381 * var p = board.create('point', [1, 7], {size: 16}); 382 * var fo = board.create('fo', [ 383 * '<div style="background-color:blue; color: yellow; padding:20px; width:200px; height:50px; ">Hello</div>', 384 * [-7, -6]], 385 * {layer: 1, fixed: false} 386 * ); 387 * 388 * </pre><div id="JXG1759c868-1a4a-4767-802c-91f84902e3ec" class="jxgbox" style="width: 500px; height: 500px;"></div> 389 * <script type="text/javascript"> 390 * (function() { 391 * var board = JXG.JSXGraph.initBoard('JXG1759c868-1a4a-4767-802c-91f84902e3ec', 392 * {boundingbox: [-8, 8, 8,-8], axis: true, showcopyright: false, shownavigation: false}); 393 * var p = board.create('point', [1, 7], {size: 16}); 394 * var fo = board.create('foreignobject', [ 395 * '<div style="background-color:blue; color: yellow; padding:20px; width:200px; height:50px; ">Hello</div>', 396 * [-7, -6]], 397 * {layer: 1, fixed: false} 398 * ); 399 * 400 * })(); 401 * 402 * </script><pre> 403 * 404 * @example 405 * board.renderer.container.style.backgroundColor = 'lightblue'; 406 * var points = []; 407 * points.push( board.create('point', [-2, 3.5], {fixed:false,color: 'yellow', size: 6,name:'6 am'}) ); 408 * points.push( board.create('point', [0, 3.5], {fixed:false,color: 'yellow', size: 6,name:'12 pm'}) ); 409 * points.push( board.create('point', [2, 3.5], {fixed:false,color: 'yellow', size: 6,name:'6 pm'}) ); 410 * 411 * var fo = board.create('fo', [ 412 * '<video width="100%" height="100%" src="https://benedu.net/moodle/aaimg/ajx_img/astro/tr/1vd.mp4" type="html5video" controls>', 413 * [-6, -4], [12, 8]], 414 * {layer: 0, fixed: true} 415 * ); 416 * 417 * var f = JXG.Math.Numerics.lagrangePolynomial(points); 418 * var graph = board.create('functiongraph', [f, -10, 10], {fixed:true,strokeWidth:3, layer: 8}); 419 * 420 * </pre><div id="JXGc3fc5520-13aa-4f66-abaa-42e9dc3fbf3f" class="jxgbox" style="width: 500px; height: 500px;"></div> 421 * <script type="text/javascript"> 422 * (function() { 423 * var board = JXG.JSXGraph.initBoard('JXGc3fc5520-13aa-4f66-abaa-42e9dc3fbf3f', 424 * {boundingbox: [-6,4,6,-4], axis: true, showcopyright: false, shownavigation: false}); 425 * board.renderer.container.style.backgroundColor = 'lightblue'; 426 * var points = []; 427 * points.push( board.create('point', [-2, 3.5], {fixed:false,color: 'yellow', size: 6,name:'6 am'}) ); 428 * points.push( board.create('point', [0, 3.5], {fixed:false,color: 'yellow', size: 6,name:'12 pm'}) ); 429 * points.push( board.create('point', [2, 3.5], {fixed:false,color: 'yellow', size: 6,name:'6 pm'}) ); 430 * 431 * var fo = board.create('fo', [ 432 * '<video width="100%" height="100%" src="https://benedu.net/moodle/aaimg/ajx_img/astro/tr/1vd.mp4" type="html5video" controls>', 433 * [-6, -4], [12, 8]], 434 * {layer: 0, fixed: true} 435 * ); 436 * 437 * var f = JXG.Math.Numerics.lagrangePolynomial(points); 438 * var graph = board.create('functiongraph', [f, -10, 10], {fixed:true,strokeWidth:3, layer: 8}); 439 * 440 * })(); 441 * 442 * </script><pre> 443 * 444 * Video "24-hour time-lapse in Cascais, Portugal. Produced by Nuno Miguel Duarte" adapted from 445 * <a href="https://www.pbslearningmedia.org/resource/buac18-k2-sci-ess-sunposition/changing-position-of-the-sun-in-the-sky/">https://www.pbslearningmedia.org/resource/buac18-k2-sci-ess-sunposition/changing-position-of-the-sun-in-the-sky/</a>, 446 * ©2016 Nuno Miguel Duarte. 447 * 448 */ 449 JXG.createForeignObject = function (board, parents, attributes) { 450 var attr, 451 fo, 452 content = parents[0], 453 coords = parents[1], 454 size = []; 455 456 if (parents.length >= 2) { 457 size = parents[2]; 458 } 459 460 attr = Type.copyAttributes(attributes, board.options, "foreignobject"); 461 fo = CoordsElement.create(JXG.ForeignObject, board, coords, attr, content, size); 462 if (!fo) { 463 throw new Error( 464 "JSXGraph: Can't create foreignObject with parent types '" + 465 typeof parents[0] + 466 "' and '" + 467 typeof parents[1] + 468 "'." + 469 "\nPossible parent types: [string, [x, y], [w, h]], [string, [x, y]], [element,transformation]" 470 ); 471 } 472 473 return fo; 474 }; 475 476 JXG.registerElement("foreignobject", JXG.createForeignObject); 477 JXG.registerElement("fo", JXG.createForeignObject); 478 479 export default JXG.ForeignObject; 480 // export default { 481 // ForeignObject: JXG.ForeignObject, 482 // createForeignobject: JXG.createForeignObject 483 // }; 484