From 1a7a5ee1d626c2bf71de2ef07f33ff152eb9e6e6 Mon Sep 17 00:00:00 2001 From: Aiosa <469130@mail.muni.cz> Date: Wed, 4 Dec 2024 09:08:00 +0100 Subject: [PATCH] Respect tainted() when uploading a texture. --- src/drawerbase.js | 2 +- src/webgldrawer.js | 94 ++++++++++++++++++++++++++-------------------- 2 files changed, 54 insertions(+), 42 deletions(-) diff --git a/src/drawerbase.js b/src/drawerbase.js index e2f32338..3355b45c 100644 --- a/src/drawerbase.js +++ b/src/drawerbase.js @@ -368,7 +368,7 @@ OpenSeadragon.DrawerBase = class DrawerBase{ * @property {OpenSeadragon.DrawerBase} drawer - The drawer that raised the error. * @property {String} error - A message describing the error. * @property {?Object} userData - Arbitrary subscriber-defined object. - * @private + * @protected */ this.viewer.raiseEvent( 'drawer-error', { tiledImage: tiledImage, diff --git a/src/webgldrawer.js b/src/webgldrawer.js index 6a13f555..03b43ea4 100644 --- a/src/webgldrawer.js +++ b/src/webgldrawer.js @@ -887,51 +887,63 @@ const tex2DCompatibleLoader = (tile, data) => { let tiledImage = tile.tiledImage; let gl = this._gl; - - let sourceWidthFraction, sourceHeightFraction; - if (tile.sourceBounds) { - sourceWidthFraction = Math.min(tile.sourceBounds.width, data.width) / data.width; - sourceHeightFraction = Math.min(tile.sourceBounds.height, data.height) / data.height; - } else { - sourceWidthFraction = 1; - sourceHeightFraction = 1; - } - - // create a gl Texture for this tile and bind the canvas with the image data - let texture = gl.createTexture(); + let texture; let position; - let overlap = tiledImage.source.tileOverlap; - if( overlap > 0){ - // calculate the normalized position of the rect to actually draw - // discarding overlap. - let overlapFraction = this._calculateOverlapFraction(tile, tiledImage); - let left = (tile.x === 0 ? 0 : overlapFraction.x) * sourceWidthFraction; - let top = (tile.y === 0 ? 0 : overlapFraction.y) * sourceHeightFraction; - let right = (tile.isRightMost ? 1 : 1 - overlapFraction.x) * sourceWidthFraction; - let bottom = (tile.isBottomMost ? 1 : 1 - overlapFraction.y) * sourceHeightFraction; - position = this._makeQuadVertexBuffer(left, right, top, bottom); - } else if (sourceWidthFraction === 1 && sourceHeightFraction === 1) { - // no overlap and no padding: this texture can use the unit quad as its position data - position = this._unitQuad; - } else { - position = this._makeQuadVertexBuffer(0, sourceWidthFraction, 0, sourceHeightFraction); - } + if (!tiledImage.isTainted()) { + if((data instanceof CanvasRenderingContext2D) && $.isCanvasTainted(data)){ + tiledImage.setTainted(true); + $.console.warn('WebGL cannot be used to draw this TiledImage because it has tainted data. Does crossOriginPolicy need to be set?'); + this._raiseDrawerErrorEvent(tiledImage, 'Tainted data cannot be used by the WebGLDrawer. Falling back to CanvasDrawer for this TiledImage.'); + } else { + let sourceWidthFraction, sourceHeightFraction; + if (tile.sourceBounds) { + sourceWidthFraction = Math.min(tile.sourceBounds.width, data.width) / data.width; + sourceHeightFraction = Math.min(tile.sourceBounds.height, data.height) / data.height; + } else { + sourceWidthFraction = 1; + sourceHeightFraction = 1; + } - gl.activeTexture(gl.TEXTURE0); - gl.bindTexture(gl.TEXTURE_2D, texture); - // Set the parameters so we can render any size image. - gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE); - gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE); - gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, this._textureFilter()); - gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, this._textureFilter()); + // create a gl Texture for this tile and bind the canvas with the image data + texture = gl.createTexture(); + let overlap = tiledImage.source.tileOverlap; + if( overlap > 0){ + // calculate the normalized position of the rect to actually draw + // discarding overlap. + let overlapFraction = this._calculateOverlapFraction(tile, tiledImage); - try{ - // This depends on gl.TEXTURE_2D being bound to the texture - // associated with this canvas before calling this function - gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, gl.RGBA, gl.UNSIGNED_BYTE, data); - } catch (e){ - $.console.error('Error uploading image data to WebGL', e); + let left = (tile.x === 0 ? 0 : overlapFraction.x) * sourceWidthFraction; + let top = (tile.y === 0 ? 0 : overlapFraction.y) * sourceHeightFraction; + let right = (tile.isRightMost ? 1 : 1 - overlapFraction.x) * sourceWidthFraction; + let bottom = (tile.isBottomMost ? 1 : 1 - overlapFraction.y) * sourceHeightFraction; + position = this._makeQuadVertexBuffer(left, right, top, bottom); + } else if (sourceWidthFraction === 1 && sourceHeightFraction === 1) { + // no overlap and no padding: this texture can use the unit quad as its position data + position = this._unitQuad; + } else { + position = this._makeQuadVertexBuffer(0, sourceWidthFraction, 0, sourceHeightFraction); + } + + gl.activeTexture(gl.TEXTURE0); + gl.bindTexture(gl.TEXTURE_2D, texture); + // Set the parameters so we can render any size image. + gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE); + gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE); + gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, this._textureFilter()); + gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, this._textureFilter()); + + try { + // This depends on gl.TEXTURE_2D being bound to the texture + // associated with this canvas before calling this function + gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, gl.RGBA, gl.UNSIGNED_BYTE, data); + } catch (e){ + // Todo a bit dirty re-use of the tainted flag, but makes the code more stable + tiledImage.setTainted(true); + $.console.error('Error uploading image data to WebGL. Falling back to canvas renderer.', e); + this._raiseDrawerErrorEvent(tiledImage, 'Unknown error when uploading texture. Falling back to CanvasDrawer for this TiledImage.'); + } + } } // TextureInfo stored in the cache