From 8f483a3ba000c6a1a0bbcca2aa66529f4d552cf1 Mon Sep 17 00:00:00 2001 From: John Ratcliff Date: Fri, 9 Feb 2024 11:23:38 -0800 Subject: [PATCH 01/20] Fix this scoping --- src/tiledimage.js | 2 +- src/world.js | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/src/tiledimage.js b/src/tiledimage.js index a1c3054b..c0b264ba 100644 --- a/src/tiledimage.js +++ b/src/tiledimage.js @@ -1456,7 +1456,7 @@ $.extend($.TiledImage.prototype, $.EventSource.prototype, /** @lends OpenSeadrag lowestLevel ); _this._isBlending = _this._isBlending || tileIsBlending; - _this._needsDraw = _this._needsDraw || tileIsBlending || this._wasBlending; + _this._needsDraw = _this._needsDraw || tileIsBlending || _this._wasBlending; } } diff --git a/src/world.js b/src/world.js index 5539d2bc..6049bb06 100644 --- a/src/world.js +++ b/src/world.js @@ -261,7 +261,7 @@ $.extend( $.World.prototype, $.EventSource.prototype, /** @lends OpenSeadragon.W draw: function() { this.viewer.drawer.draw(this._items); this._needsDraw = false; - this._items.forEach(function(item){ + this._items.forEach((item) => { this._needsDraw = item.setDrawn() || this._needsDraw; }); }, From 5b2c6d7ed9448ff431e3857074bf8afffcdd9687 Mon Sep 17 00:00:00 2001 From: Tom Date: Fri, 9 Feb 2024 15:06:52 -0500 Subject: [PATCH 02/20] add support for placeholderFillStyle to webgl drawer. fix spring logic to avoid getting stuck updating due to floating point math. update tilesource-swap demo. --- src/spring.js | 32 +++++++++++++++++--------------- src/webgldrawer.js | 28 ++++++++++++++++++++++++++++ test/demo/tilesource-swap.html | 17 ++++++++--------- 3 files changed, 53 insertions(+), 24 deletions(-) diff --git a/src/spring.js b/src/spring.js index f65f076e..77162188 100644 --- a/src/spring.js +++ b/src/spring.js @@ -212,7 +212,7 @@ $.Spring.prototype = { update: function() { this.current.time = $.now(); - var startValue, targetValue; + let startValue, targetValue; if (this._exponential) { startValue = this.start._logValue; targetValue = this.target._logValue; @@ -221,23 +221,25 @@ $.Spring.prototype = { targetValue = this.target.value; } - var currentValue = (this.current.time >= this.target.time) ? - targetValue : - startValue + - ( targetValue - startValue ) * - transform( - this.springStiffness, - ( this.current.time - this.start.time ) / - ( this.target.time - this.start.time ) - ); - - if (this._exponential) { - this.current.value = Math.exp(currentValue); + if(this.current.time >= this.target.time){ + this.current.value = this.target.value; } else { - this.current.value = currentValue; + let currentValue = startValue + + ( targetValue - startValue ) * + transform( + this.springStiffness, + ( this.current.time - this.start.time ) / + ( this.target.time - this.start.time ) + ); + + if (this._exponential) { + this.current.value = Math.exp(currentValue); + } else { + this.current.value = currentValue; + } } - return currentValue !== targetValue; + return this.current.value !== this.target.value; }, /** diff --git a/src/webgldrawer.js b/src/webgldrawer.js index bf40266d..67834f4c 100644 --- a/src/webgldrawer.js +++ b/src/webgldrawer.js @@ -239,6 +239,10 @@ let tilesToDraw = tiledImage.getTilesToDraw(); + if ( tiledImage.placeholderFillStyle && tiledImage._hasOpaqueTile === false ) { + this._drawPlaceholder(tiledImage); + } + if(tilesToDraw.length === 0 || tiledImage.getOpacity() === 0){ return; } @@ -1077,6 +1081,30 @@ context.restore(); } + _drawPlaceholder(tiledImage){ + + const bounds = tiledImage.getBounds(true); + const rect = this.viewportToDrawerRectangle(tiledImage.getBounds(true)); + const context = this._outputContext; + + let fillStyle; + if ( typeof tiledImage.placeholderFillStyle === "function" ) { + fillStyle = tiledImage.placeholderFillStyle(tiledImage, context); + } + else { + fillStyle = tiledImage.placeholderFillStyle; + } + + context.save(); + context.fillStyle = fillStyle; + context.translate(rect.x, rect.y); + context.rotate(Math.PI / 180 * bounds.degrees); + context.translate(-rect.x, -rect.y); + context.fillRect(rect.x, rect.y, rect.width, rect.height); + context.restore(); + + } + // private _restoreRotationChanges() { var context = this._outputContext; diff --git a/test/demo/tilesource-swap.html b/test/demo/tilesource-swap.html index 5b1f6dee..f5fbc084 100644 --- a/test/demo/tilesource-swap.html +++ b/test/demo/tilesource-swap.html @@ -32,17 +32,18 @@ type: 'legacy-image-pyramid', levels: [ { - url: 'http://openseadragon.github.io/example-images/duomo/duomo_files/8/0_0.jpg', + url: 'https://openseadragon.github.io/example-images/duomo/duomo_files/8/0_0.jpg', width: 218, height: 160 } - ] + ], + degrees: 30, }; var duomo = { Image: { xmlns: 'http://schemas.microsoft.com/deepzoom/2008', - Url: 'http://openseadragon.github.io/example-images/duomo/duomo_files/', + Url: 'https://openseadragon.github.io/example-images/duomo/duomo_files/', Format: 'jpg', Overlap: '2', TileSize: '256', @@ -59,10 +60,11 @@ tileSources: duomoStandin, minZoomImageRatio: 0.1, defaultZoomLevel: 0.1, - zoomPerClick: 1 + zoomPerClick: 1, + crossOriginPolicy: 'Anonymous' }); - viewer.addHandler('canvas-click', function(event) { + viewer.addOnceHandler('canvas-click', function(event) { if (event.quick) { var standin = viewer.world.getItemAt(0); var standinBounds = standin.getBounds(); @@ -76,13 +78,10 @@ index: 0, // Add the new image below the stand-in. success: function(event) { var fullImage = event.item; - - // The changeover will look better if we wait for the first tile to be drawn. + // The changeover will look better if we wait for the first draw after the changeover. var tileDrawnHandler = function(event) { - if (event.tiledImage === fullImage) { viewer.removeHandler('update-viewport', tileDrawnHandler); viewer.world.removeItem(standin); - } }; viewer.addHandler('update-viewport', tileDrawnHandler); From d4e82d374e3633190ed618e434d0bba9fe7e8137 Mon Sep 17 00:00:00 2001 From: Tom Date: Fri, 9 Feb 2024 15:19:40 -0500 Subject: [PATCH 03/20] account for viewport rotation in addition to tiledImage rotation --- src/webgldrawer.js | 9 ++++++--- test/demo/tilesource-swap.html | 6 ++++-- 2 files changed, 10 insertions(+), 5 deletions(-) diff --git a/src/webgldrawer.js b/src/webgldrawer.js index 67834f4c..d8b33f93 100644 --- a/src/webgldrawer.js +++ b/src/webgldrawer.js @@ -242,7 +242,10 @@ if ( tiledImage.placeholderFillStyle && tiledImage._hasOpaqueTile === false ) { this._drawPlaceholder(tiledImage); } - + this._drawPlaceholder(tiledImage); + if(!window.draw){ + return; + } if(tilesToDraw.length === 0 || tiledImage.getOpacity() === 0){ return; } @@ -1095,13 +1098,13 @@ fillStyle = tiledImage.placeholderFillStyle; } - context.save(); + this._offsetForRotation({degrees: this.viewer.viewport.getRotation(true)}); context.fillStyle = fillStyle; context.translate(rect.x, rect.y); context.rotate(Math.PI / 180 * bounds.degrees); context.translate(-rect.x, -rect.y); context.fillRect(rect.x, rect.y, rect.width, rect.height); - context.restore(); + this._restoreRotationChanges(); } diff --git a/test/demo/tilesource-swap.html b/test/demo/tilesource-swap.html index f5fbc084..d8da3dfb 100644 --- a/test/demo/tilesource-swap.html +++ b/test/demo/tilesource-swap.html @@ -64,8 +64,10 @@ crossOriginPolicy: 'Anonymous' }); - viewer.addOnceHandler('canvas-click', function(event) { - if (event.quick) { + let swapped = false; + viewer.addHandler('canvas-click', function(event) { + if (event.quick && !swapped) { + swapped = true; var standin = viewer.world.getItemAt(0); var standinBounds = standin.getBounds(); viewer.viewport.fitBounds(standinBounds); From c734de531fd94cbbd912f96be046b953ccf65399 Mon Sep 17 00:00:00 2001 From: Tom Date: Fri, 9 Feb 2024 17:18:54 -0500 Subject: [PATCH 04/20] fix placeholder positioning for canvas drawer --- src/canvasdrawer.js | 10 ++++++---- 1 file changed, 6 insertions(+), 4 deletions(-) diff --git a/src/canvasdrawer.js b/src/canvasdrawer.js index d4ec6231..2ec6b17b 100644 --- a/src/canvasdrawer.js +++ b/src/canvasdrawer.js @@ -382,9 +382,9 @@ class CanvasDrawer extends OpenSeadragon.DrawerBase{ } usedClip = true; } - + tiledImage._hasOpaqueTile = false; if ( tiledImage.placeholderFillStyle && tiledImage._hasOpaqueTile === false ) { - var placeholderRect = this.viewportToDrawerRectangle(tiledImage.getBounds(true)); + let placeholderRect = this.viewportToDrawerRectangle(tiledImage.getBoundsNoRotate(true)); if (sketchScale) { placeholderRect = placeholderRect.times(sketchScale); } @@ -392,7 +392,7 @@ class CanvasDrawer extends OpenSeadragon.DrawerBase{ placeholderRect = placeholderRect.translate(sketchTranslate); } - var fillStyle = null; + let fillStyle = null; if ( typeof tiledImage.placeholderFillStyle === "function" ) { fillStyle = tiledImage.placeholderFillStyle(tiledImage, this.context); } @@ -402,7 +402,9 @@ class CanvasDrawer extends OpenSeadragon.DrawerBase{ this._drawRectangle(placeholderRect, fillStyle, useSketch); } - + if(!window.draw){ + lastDrawn = []; + } var subPixelRoundingRule = determineSubPixelRoundingRule(tiledImage.subPixelRoundingForTransparency); var shouldRoundPositionAndSize = false; From 3f21f84df436fde7bbd435a8602856ceea0f9d8e Mon Sep 17 00:00:00 2001 From: Tom Date: Fri, 9 Feb 2024 18:16:30 -0500 Subject: [PATCH 05/20] clean up code added for testing --- src/canvasdrawer.js | 4 +--- src/webgldrawer.js | 5 +---- 2 files changed, 2 insertions(+), 7 deletions(-) diff --git a/src/canvasdrawer.js b/src/canvasdrawer.js index 2ec6b17b..6a00a294 100644 --- a/src/canvasdrawer.js +++ b/src/canvasdrawer.js @@ -402,9 +402,7 @@ class CanvasDrawer extends OpenSeadragon.DrawerBase{ this._drawRectangle(placeholderRect, fillStyle, useSketch); } - if(!window.draw){ - lastDrawn = []; - } + var subPixelRoundingRule = determineSubPixelRoundingRule(tiledImage.subPixelRoundingForTransparency); var shouldRoundPositionAndSize = false; diff --git a/src/webgldrawer.js b/src/webgldrawer.js index d8b33f93..ac450b3c 100644 --- a/src/webgldrawer.js +++ b/src/webgldrawer.js @@ -242,10 +242,7 @@ if ( tiledImage.placeholderFillStyle && tiledImage._hasOpaqueTile === false ) { this._drawPlaceholder(tiledImage); } - this._drawPlaceholder(tiledImage); - if(!window.draw){ - return; - } + if(tilesToDraw.length === 0 || tiledImage.getOpacity() === 0){ return; } From 0a154a3b21fae5f666561e087f79f01526e46e5e Mon Sep 17 00:00:00 2001 From: Tom Date: Sun, 11 Feb 2024 11:51:38 -0500 Subject: [PATCH 06/20] In webgl drawer, fall back to canvas drawer for tiled images with tainted data --- src/tiledimage.js | 20 +++ src/webgldrawer.js | 335 ++++++++++++++++++++++++++------------------- 2 files changed, 213 insertions(+), 142 deletions(-) diff --git a/src/tiledimage.js b/src/tiledimage.js index a1c3054b..84b852d9 100644 --- a/src/tiledimage.js +++ b/src/tiledimage.js @@ -166,6 +166,7 @@ $.TiledImage = function( options ) { _lastDrawn: [], // array of tiles that were last fetched by the drawer _isBlending: false, // Are any tiles still being blended? _wasBlending: false, // Were any tiles blending before the last draw? + _isTainted: false, // Has a Tile been found with tainted data? //configurable settings springStiffness: $.DEFAULT_SETTINGS.springStiffness, animationTime: $.DEFAULT_SETTINGS.animationTime, @@ -326,6 +327,25 @@ $.extend($.TiledImage.prototype, $.EventSource.prototype, /** @lends OpenSeadrag return this._needsDraw; }, + /** + * Set the internal _isTainted flag for this TiledImage. Lazy loaded - not + * checked each time a Tile is loaded, but can be set if a consumer of the + * tiles (e.g. a Drawer) discovers a Tile to have tainted data so that further + * checks are not needed and alternative rendering strategies can be used. + * @private + */ + setTainted(isTainted){ + this._isTainted = isTainted; + }, + + /** + * @private + * @returns {Boolean} whether the TiledImage has been marked as tainted + */ + isTainted(){ + return this._isTainted; + }, + /** * Destroy the TiledImage (unload current loaded tiles). */ diff --git a/src/webgldrawer.js b/src/webgldrawer.js index bf40266d..3cd46ffe 100644 --- a/src/webgldrawer.js +++ b/src/webgldrawer.js @@ -90,6 +90,7 @@ this._clippingCanvas = null; this._clippingContext = null; this._renderingCanvas = null; + this._backupCanvasDrawer = null; // Add listeners for events that require modifying the scene or camera this.viewer.addHandler("tile-ready", ev => this._tileReadyHandler(ev)); @@ -206,6 +207,25 @@ return canvas; } + /** + * Get the backup renderer (CanvasDrawer) to use if data cannot be used by webgl + * Lazy loaded + * @private + * @returns {CanvasDrawer} + */ + _getBackupCanvasDrawer(){ + if(!this._backupCanvasDrawer){ + this._backupCanvasDrawer = new $.CanvasDrawer({ + viewer: this.viewer, + viewport: this.viewport, + element: this.container, + }); + this._backupCanvasDrawer.canvas.style.setProperty('visibility', 'hidden'); + } + + return this._backupCanvasDrawer; + } + /** * * @param {Array} tiledImages Array of TiledImage objects to draw @@ -237,168 +257,187 @@ //iterate over tiled images and draw each one using a two-pass rendering pipeline if needed tiledImages.forEach( (tiledImage, tiledImageIndex) => { - let tilesToDraw = tiledImage.getTilesToDraw(); - - if(tilesToDraw.length === 0 || tiledImage.getOpacity() === 0){ - return; - } - let firstTile = tilesToDraw[0]; - - let useContext2dPipeline = ( tiledImage.compositeOperation || - this.viewer.compositeOperation || - tiledImage._clip || - tiledImage._croppingPolygons || - tiledImage.debugMode - ); - - let useTwoPassRendering = useContext2dPipeline || (tiledImage.opacity < 1) || firstTile.hasTransparency; - - // using the context2d pipeline requires a clean rendering (back) buffer to start - if(useContext2dPipeline){ - // if the rendering buffer has image data currently, write it to the output canvas now and clear it - + if(tiledImage.isTainted()){ + // first, draw any data left in the rendering buffer onto the output canvas if(renderingBufferHasImageData){ this._outputContext.drawImage(this._renderingCanvas, 0, 0); + // clear the buffer + gl.bindFramebuffer(gl.FRAMEBUFFER, null); + gl.clear(gl.COLOR_BUFFER_BIT); // clear the back buffer + renderingBufferHasImageData = false; } - // clear the buffer - gl.bindFramebuffer(gl.FRAMEBUFFER, null); - gl.clear(gl.COLOR_BUFFER_BIT); // clear the back buffer - } + // next, use the backup canvas drawer to draw this tainted image + const canvasDrawer = this._getBackupCanvasDrawer(); + canvasDrawer.draw([tiledImage]); + this._outputContext.drawImage(canvasDrawer.canvas, 0, 0); - // First rendering pass: compose tiles that make up this tiledImage - gl.useProgram(this._firstPass.shaderProgram); - - // bind to the framebuffer for render-to-texture if using two-pass rendering, otherwise back buffer (null) - if(useTwoPassRendering){ - gl.bindFramebuffer(gl.FRAMEBUFFER, this._glFrameBuffer); - // clear the buffer to draw a new image - gl.clear(gl.COLOR_BUFFER_BIT); } else { - gl.bindFramebuffer(gl.FRAMEBUFFER, null); - // no need to clear, just draw on top of the existing pixels - } + let tilesToDraw = tiledImage.getTilesToDraw(); - let overallMatrix = viewMatrix; - - let imageRotation = tiledImage.getRotation(true); - // if needed, handle the tiledImage being rotated - if( imageRotation % 360 !== 0){ - let imageRotationMatrix = $.Mat3.makeRotation(-imageRotation * Math.PI / 180); - let imageCenter = tiledImage.getBoundsNoRotate(true).getCenter(); - let t1 = $.Mat3.makeTranslation(imageCenter.x, imageCenter.y); - let t2 = $.Mat3.makeTranslation(-imageCenter.x, -imageCenter.y); - - // update the view matrix to account for this image's rotation - let localMatrix = t1.multiply(imageRotationMatrix).multiply(t2); - overallMatrix = viewMatrix.multiply(localMatrix); - } - - let maxTextures = this._gl.getParameter(this._gl.MAX_TEXTURE_IMAGE_UNITS); - if(maxTextures <= 0){ - // This can apparently happen on some systems if too many WebGL contexts have been created - // in which case maxTextures can be null, leading to out of bounds errors with the array. - // For example, when viewers were created and not destroyed in the test suite, this error - // occured in the TravisCI tests, though it did not happen when testing locally either in - // a browser or on the command line via grunt test. - - throw(new Error(`WegGL error: bad value for gl parameter MAX_TEXTURE_IMAGE_UNITS (${maxTextures}). This could happen - if too many contexts have been created and not released, or there is another problem with the graphics card.`)); - } - - let texturePositionArray = new Float32Array(maxTextures * 12); // 6 vertices (2 triangles) x 2 coordinates per vertex - let textureDataArray = new Array(maxTextures); - let matrixArray = new Array(maxTextures); - let opacityArray = new Array(maxTextures); - - // iterate over tiles and add data for each one to the buffers - for(let tileIndex = 0; tileIndex < tilesToDraw.length; tileIndex++){ - let tile = tilesToDraw[tileIndex].tile; - let indexInDrawArray = tileIndex % maxTextures; - let numTilesToDraw = indexInDrawArray + 1; - let tileContext = tile.getCanvasContext(); - - let textureInfo = tileContext ? this._TextureMap.get(tileContext.canvas) : null; - if(textureInfo){ - this._getTileData(tile, tiledImage, textureInfo, overallMatrix, indexInDrawArray, texturePositionArray, textureDataArray, matrixArray, opacityArray); - } else { - // console.log('No tile info', tile); + if(tilesToDraw.length === 0 || tiledImage.getOpacity() === 0){ + return; } - if( (numTilesToDraw === maxTextures) || (tileIndex === tilesToDraw.length - 1)){ - // We've filled up the buffers: time to draw this set of tiles + let firstTile = tilesToDraw[0]; - // bind each tile's texture to the appropriate gl.TEXTURE# - for(let i = 0; i <= numTilesToDraw; i++){ - gl.activeTexture(gl.TEXTURE0 + i); - gl.bindTexture(gl.TEXTURE_2D, textureDataArray[i]); + let useContext2dPipeline = ( tiledImage.compositeOperation || + this.viewer.compositeOperation || + tiledImage._clip || + tiledImage._croppingPolygons || + tiledImage.debugMode + ); + + let useTwoPassRendering = useContext2dPipeline || (tiledImage.opacity < 1) || firstTile.hasTransparency; + + // using the context2d pipeline requires a clean rendering (back) buffer to start + if(useContext2dPipeline){ + // if the rendering buffer has image data currently, write it to the output canvas now and clear it + + if(renderingBufferHasImageData){ + this._outputContext.drawImage(this._renderingCanvas, 0, 0); } - // set the buffer data for the texture coordinates to use for each tile - gl.bindBuffer(gl.ARRAY_BUFFER, this._firstPass.bufferTexturePosition); - gl.bufferData(gl.ARRAY_BUFFER, texturePositionArray, gl.DYNAMIC_DRAW); + // clear the buffer + gl.bindFramebuffer(gl.FRAMEBUFFER, null); + gl.clear(gl.COLOR_BUFFER_BIT); // clear the back buffer + } - // set the transform matrix uniform for each tile - matrixArray.forEach( (matrix, index) => { - gl.uniformMatrix3fv(this._firstPass.uTransformMatrices[index], false, matrix); - }); - // set the opacity uniform for each tile - gl.uniform1fv(this._firstPass.uOpacities, new Float32Array(opacityArray)); + // First rendering pass: compose tiles that make up this tiledImage + gl.useProgram(this._firstPass.shaderProgram); - // bind vertex buffers and (re)set attributes before calling gl.drawArrays() - gl.bindBuffer(gl.ARRAY_BUFFER, this._firstPass.bufferOutputPosition); + // bind to the framebuffer for render-to-texture if using two-pass rendering, otherwise back buffer (null) + if(useTwoPassRendering){ + gl.bindFramebuffer(gl.FRAMEBUFFER, this._glFrameBuffer); + // clear the buffer to draw a new image + gl.clear(gl.COLOR_BUFFER_BIT); + } else { + gl.bindFramebuffer(gl.FRAMEBUFFER, null); + // no need to clear, just draw on top of the existing pixels + } + + let overallMatrix = viewMatrix; + + let imageRotation = tiledImage.getRotation(true); + // if needed, handle the tiledImage being rotated + if( imageRotation % 360 !== 0){ + let imageRotationMatrix = $.Mat3.makeRotation(-imageRotation * Math.PI / 180); + let imageCenter = tiledImage.getBoundsNoRotate(true).getCenter(); + let t1 = $.Mat3.makeTranslation(imageCenter.x, imageCenter.y); + let t2 = $.Mat3.makeTranslation(-imageCenter.x, -imageCenter.y); + + // update the view matrix to account for this image's rotation + let localMatrix = t1.multiply(imageRotationMatrix).multiply(t2); + overallMatrix = viewMatrix.multiply(localMatrix); + } + + let maxTextures = this._gl.getParameter(this._gl.MAX_TEXTURE_IMAGE_UNITS); + if(maxTextures <= 0){ + // This can apparently happen on some systems if too many WebGL contexts have been created + // in which case maxTextures can be null, leading to out of bounds errors with the array. + // For example, when viewers were created and not destroyed in the test suite, this error + // occured in the TravisCI tests, though it did not happen when testing locally either in + // a browser or on the command line via grunt test. + + throw(new Error(`WegGL error: bad value for gl parameter MAX_TEXTURE_IMAGE_UNITS (${maxTextures}). This could happen + if too many contexts have been created and not released, or there is another problem with the graphics card.`)); + } + + let texturePositionArray = new Float32Array(maxTextures * 12); // 6 vertices (2 triangles) x 2 coordinates per vertex + let textureDataArray = new Array(maxTextures); + let matrixArray = new Array(maxTextures); + let opacityArray = new Array(maxTextures); + + // iterate over tiles and add data for each one to the buffers + for(let tileIndex = 0; tileIndex < tilesToDraw.length; tileIndex++){ + let tile = tilesToDraw[tileIndex].tile; + let indexInDrawArray = tileIndex % maxTextures; + let numTilesToDraw = indexInDrawArray + 1; + let tileContext = tile.getCanvasContext(); + + let textureInfo = tileContext ? this._TextureMap.get(tileContext.canvas) : null; + if(textureInfo){ + this._getTileData(tile, tiledImage, textureInfo, overallMatrix, indexInDrawArray, texturePositionArray, textureDataArray, matrixArray, opacityArray); + } else { + // console.log('No tile info', tile); + } + if( (numTilesToDraw === maxTextures) || (tileIndex === tilesToDraw.length - 1)){ + // We've filled up the buffers: time to draw this set of tiles + + // bind each tile's texture to the appropriate gl.TEXTURE# + for(let i = 0; i <= numTilesToDraw; i++){ + gl.activeTexture(gl.TEXTURE0 + i); + gl.bindTexture(gl.TEXTURE_2D, textureDataArray[i]); + } + + // set the buffer data for the texture coordinates to use for each tile + gl.bindBuffer(gl.ARRAY_BUFFER, this._firstPass.bufferTexturePosition); + gl.bufferData(gl.ARRAY_BUFFER, texturePositionArray, gl.DYNAMIC_DRAW); + + // set the transform matrix uniform for each tile + matrixArray.forEach( (matrix, index) => { + gl.uniformMatrix3fv(this._firstPass.uTransformMatrices[index], false, matrix); + }); + // set the opacity uniform for each tile + gl.uniform1fv(this._firstPass.uOpacities, new Float32Array(opacityArray)); + + // bind vertex buffers and (re)set attributes before calling gl.drawArrays() + gl.bindBuffer(gl.ARRAY_BUFFER, this._firstPass.bufferOutputPosition); + gl.vertexAttribPointer(this._firstPass.aOutputPosition, 2, gl.FLOAT, false, 0, 0); + + gl.bindBuffer(gl.ARRAY_BUFFER, this._firstPass.bufferTexturePosition); + gl.vertexAttribPointer(this._firstPass.aTexturePosition, 2, gl.FLOAT, false, 0, 0); + + gl.bindBuffer(gl.ARRAY_BUFFER, this._firstPass.bufferIndex); + gl.vertexAttribPointer(this._firstPass.aIndex, 1, gl.FLOAT, false, 0, 0); + + // Draw! 6 vertices per tile (2 triangles per rectangle) + gl.drawArrays(gl.TRIANGLES, 0, 6 * numTilesToDraw ); + } + } + + if(useTwoPassRendering){ + // Second rendering pass: Render the tiled image from the framebuffer into the back buffer + gl.useProgram(this._secondPass.shaderProgram); + + // set the rendering target to the back buffer (null) + gl.bindFramebuffer(gl.FRAMEBUFFER, null); + + // bind the rendered texture from the first pass to use during this second pass + gl.activeTexture(gl.TEXTURE0); + gl.bindTexture(gl.TEXTURE_2D, this._renderToTexture); + + // set opacity to the value for the current tiledImage + this._gl.uniform1f(this._secondPass.uOpacityMultiplier, tiledImage.opacity); + + // bind buffers and set attributes before calling gl.drawArrays + gl.bindBuffer(gl.ARRAY_BUFFER, this._secondPass.bufferTexturePosition); + gl.vertexAttribPointer(this._secondPass.aTexturePosition, 2, gl.FLOAT, false, 0, 0); + gl.bindBuffer(gl.ARRAY_BUFFER, this._secondPass.bufferOutputPosition); gl.vertexAttribPointer(this._firstPass.aOutputPosition, 2, gl.FLOAT, false, 0, 0); - gl.bindBuffer(gl.ARRAY_BUFFER, this._firstPass.bufferTexturePosition); - gl.vertexAttribPointer(this._firstPass.aTexturePosition, 2, gl.FLOAT, false, 0, 0); + // Draw the quad (two triangles) + gl.drawArrays(gl.TRIANGLES, 0, 6); - gl.bindBuffer(gl.ARRAY_BUFFER, this._firstPass.bufferIndex); - gl.vertexAttribPointer(this._firstPass.aIndex, 1, gl.FLOAT, false, 0, 0); + } - // Draw! 6 vertices per tile (2 triangles per rectangle) - gl.drawArrays(gl.TRIANGLES, 0, 6 * numTilesToDraw ); + renderingBufferHasImageData = true; + + if(useContext2dPipeline){ + // draw from the rendering canvas onto the output canvas, clipping/cropping if needed. + this._applyContext2dPipeline(tiledImage, tilesToDraw, tiledImageIndex); + renderingBufferHasImageData = false; + // clear the buffer + gl.bindFramebuffer(gl.FRAMEBUFFER, null); + gl.clear(gl.COLOR_BUFFER_BIT); // clear the back buffer + } + + // after drawing the first TiledImage, fire the tiled-image-drawn event (for testing) + if(tiledImageIndex === 0){ + this._raiseTiledImageDrawnEvent(tiledImage, tilesToDraw.map(info=>info.tile)); } } - if(useTwoPassRendering){ - // Second rendering pass: Render the tiled image from the framebuffer into the back buffer - gl.useProgram(this._secondPass.shaderProgram); - // set the rendering target to the back buffer (null) - gl.bindFramebuffer(gl.FRAMEBUFFER, null); - - // bind the rendered texture from the first pass to use during this second pass - gl.activeTexture(gl.TEXTURE0); - gl.bindTexture(gl.TEXTURE_2D, this._renderToTexture); - - // set opacity to the value for the current tiledImage - this._gl.uniform1f(this._secondPass.uOpacityMultiplier, tiledImage.opacity); - - // bind buffers and set attributes before calling gl.drawArrays - gl.bindBuffer(gl.ARRAY_BUFFER, this._secondPass.bufferTexturePosition); - gl.vertexAttribPointer(this._secondPass.aTexturePosition, 2, gl.FLOAT, false, 0, 0); - gl.bindBuffer(gl.ARRAY_BUFFER, this._secondPass.bufferOutputPosition); - gl.vertexAttribPointer(this._firstPass.aOutputPosition, 2, gl.FLOAT, false, 0, 0); - - // Draw the quad (two triangles) - gl.drawArrays(gl.TRIANGLES, 0, 6); - - } - - renderingBufferHasImageData = true; - - if(useContext2dPipeline){ - // draw from the rendering canvas onto the output canvas, clipping/cropping if needed. - this._applyContext2dPipeline(tiledImage, tilesToDraw, tiledImageIndex); - renderingBufferHasImageData = false; - // clear the buffer - gl.bindFramebuffer(gl.FRAMEBUFFER, null); - gl.clear(gl.COLOR_BUFFER_BIT); // clear the back buffer - } - - // after drawing the first TiledImage, fire the tiled-image-drawn event (for testing) - if(tiledImageIndex === 0){ - this._raiseTiledImageDrawnEvent(tiledImage, tilesToDraw.map(info=>info.tile)); - } }); @@ -802,7 +841,19 @@ let tile = event.tile; let tiledImage = event.tiledImage; let tileContext = tile.getCanvasContext(); - let canvas = tileContext.canvas; + let canvas = tileContext && tileContext.canvas; + // if the tile doesn't provide a canvas, or is tainted by cross-origin + // data, marked the TiledImage as tainted so the canvas drawer can be + // used instead, and return immediately - data cannot be uploaded to webgl + if(!canvas || $.isCanvasTainted(canvas)){ + const wasTainted = tiledImage.isTainted(); + if(!wasTainted){ + tiledImage.setTainted(true); + $.console.warn('WebGL cannot be used to draw this TiledImage because it has tainted data. Does crossOriginPolicy need to be set?'); + } + return; + } + let textureInfo = this._TextureMap.get(canvas); // if this is a new image for us, create a texture From 8967e2bb034e62518d9c5d14d24d9849a4fec041 Mon Sep 17 00:00:00 2001 From: Tom Date: Mon, 12 Feb 2024 09:30:26 -0500 Subject: [PATCH 07/20] support hot-swapping drawers with viewer.setDrawer() --- src/canvasdrawer.js | 1 + src/viewer.js | 81 ++++++++++++++++++++++++++++++--------------- src/webgldrawer.js | 28 +++++++++++++++- 3 files changed, 83 insertions(+), 27 deletions(-) diff --git a/src/canvasdrawer.js b/src/canvasdrawer.js index d4ec6231..b848c3ef 100644 --- a/src/canvasdrawer.js +++ b/src/canvasdrawer.js @@ -139,6 +139,7 @@ class CanvasDrawer extends OpenSeadragon.DrawerBase{ this.canvas.height = 1; this.sketchCanvas = null; this.sketchContext = null; + this.container.removeChild(this.canvas); } /** diff --git a/src/viewer.js b/src/viewer.js index 6a8865c8..c3c10938 100644 --- a/src/viewer.js +++ b/src/viewer.js @@ -455,35 +455,13 @@ $.Viewer = function( options ) { this.drawer = null; - for (let i = 0; i < drawerCandidates.length; i++) { - - let drawerCandidate = drawerCandidates[i]; - let Drawer = null; - - //if inherits from a drawer base, use it - if (drawerCandidate && drawerCandidate.prototype instanceof $.DrawerBase) { - Drawer = drawerCandidate; - drawerCandidate = 'custom'; - } else if (typeof drawerCandidate === "string") { - Drawer = $.determineDrawer(drawerCandidate); - } else { - $.console.warn('Unsupported drawer! Drawer must be an existing string type, or a class that extends OpenSeadragon.DrawerBase.'); - continue; - } - - // if the drawer is supported, create it and break the loop - if (Drawer && Drawer.isSupported()) { - this.drawer = new Drawer({ - viewer: this, - viewport: this.viewport, - element: this.canvas, - debugGridColor: this.debugGridColor, - options: this.drawerOptions[drawerCandidate], - }); - + for (const drawerCandidate of drawerCandidates){ + let success = this.setDrawer(drawerCandidate, false); + if(success){ break; } } + if (!this.drawer){ $.console.error('No drawer could be created!'); throw('Error with creating the selected drawer(s)'); @@ -950,6 +928,57 @@ $.extend( $.Viewer.prototype, $.EventSource.prototype, $.ControlDock.prototype, this.removeAllHandlers(); }, + /** + * Set the drawer for this viewer, as a supported string or drawer constructor. + * @param {String | OpenSeadragon.DrawerBase} drawerCandidate The type of drawer to try to construct + * @param { Boolean } [redrawImmediately] Whether to immediately draw a new frame. Default = true. + * @param { Object } [drawerOptions] Options for this drawer. If falsey, defaults to viewer.drawerOptions + * for this viewer type. See {@link OpenSeadragon.Options}. + * @returns {Boolean} whether the drawer was created successfully + */ + setDrawer(drawerCandidate, redrawImmediately = true, drawerOptions = null){ + const oldDrawer = this.drawer; + + let Drawer = null; + + //if inherits from a drawer base, use it + if (drawerCandidate && drawerCandidate.prototype instanceof $.DrawerBase) { + Drawer = drawerCandidate; + drawerCandidate = 'custom'; + } else if (typeof drawerCandidate === "string") { + Drawer = $.determineDrawer(drawerCandidate); + } + + if(!Drawer){ + $.console.warn('Unsupported drawer! Drawer must be an existing string type, or a class that extends OpenSeadragon.DrawerBase.'); + } + + // if the drawer is supported, create it and return true + if (Drawer && Drawer.isSupported()) { + + // first destroy the previous drawer + if(oldDrawer){ + oldDrawer.destroy(); + } + + // create the new drawer + this.drawer = new Drawer({ + viewer: this, + viewport: this.viewport, + element: this.canvas, + debugGridColor: this.debugGridColor, + options: drawerOptions || this.drawerOptions[drawerCandidate], + }); + + if(redrawImmediately){ + this.forceRedraw(); + } + return true; + } + + return false; + }, + /** * @function * @returns {Boolean} diff --git a/src/webgldrawer.js b/src/webgldrawer.js index 3cd46ffe..e5b8f811 100644 --- a/src/webgldrawer.js +++ b/src/webgldrawer.js @@ -158,6 +158,16 @@ // set our webgl context reference to null to enable garbage collection this._gl = null; + if(this._backupCanvasDrawer){ + this._backupCanvasDrawer.destroy(); + this._backupCanvasDrawer = null; + } + + this.container.removeChild(this.canvas); + if(this.viewer.drawer === this){ + this.viewer.drawer = null; + } + // set our destroyed flag to true this._destroyed = true; } @@ -355,6 +365,15 @@ let tileContext = tile.getCanvasContext(); let textureInfo = tileContext ? this._TextureMap.get(tileContext.canvas) : null; + if(!textureInfo){ + // tile was not processed in the tile-ready event (this can happen + // if this drawer was created after the tile was downloaded) + this._tileReadyHandler({tile: tile, tiledImage: tiledImage}); + + // retry getting textureInfo + textureInfo = tileContext ? this._TextureMap.get(tileContext.canvas) : null; + } + if(textureInfo){ this._getTileData(tile, tiledImage, textureInfo, overallMatrix, indexInDrawArray, texturePositionArray, textureDataArray, matrixArray, opacityArray); } else { @@ -840,11 +859,18 @@ _tileReadyHandler(event){ let tile = event.tile; let tiledImage = event.tiledImage; + + // If a tiledImage is already known to be tainted, don't try to upload any + // textures to webgl, because they won't be used even if it succeeds + if(tiledImage.isTainted()){ + return; + } + let tileContext = tile.getCanvasContext(); let canvas = tileContext && tileContext.canvas; // if the tile doesn't provide a canvas, or is tainted by cross-origin // data, marked the TiledImage as tainted so the canvas drawer can be - // used instead, and return immediately - data cannot be uploaded to webgl + // used instead, and return immediately - tainted data cannot be uploaded to webgl if(!canvas || $.isCanvasTainted(canvas)){ const wasTainted = tiledImage.isTainted(); if(!wasTainted){ From 6e4914ada33c697242d765bb0c8fb21b0049f22c Mon Sep 17 00:00:00 2001 From: Tom Date: Mon, 12 Feb 2024 09:32:17 -0500 Subject: [PATCH 08/20] better cleanup for html drawer in destroy --- src/htmldrawer.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/htmldrawer.js b/src/htmldrawer.js index 3d04f444..80f851e4 100644 --- a/src/htmldrawer.js +++ b/src/htmldrawer.js @@ -126,7 +126,7 @@ class HTMLDrawer extends OpenSeadragon.DrawerBase{ * Destroy the drawer (unload current loaded tiles) */ destroy() { - this.canvas.innerHTML = ""; + this.container.removeChild(this.canvas); } /** From b62fba80d678c7fefd7e818c6f06458d764ac5e1 Mon Sep 17 00:00:00 2001 From: Ian Gilman Date: Mon, 12 Feb 2024 09:32:37 -0800 Subject: [PATCH 09/20] Changelog for #2469 --- changelog.txt | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/changelog.txt b/changelog.txt index 866f0394..9e327e04 100644 --- a/changelog.txt +++ b/changelog.txt @@ -5,7 +5,7 @@ OPENSEADRAGON CHANGELOG * BREAKING CHANGE: Dropped support for IE11 (#2300, #2361 @AndrewADev) * DEPRECATION: The OpenSeadragon.createCallback function is no longer recommended (#2367 @akansjain) -* The viewer now uses WebGL when available (#2310, #2462, #2466 @pearcetm, @Aiosa) +* The viewer now uses WebGL when available (#2310, #2462, #2466, #2469 @pearcetm, @Aiosa) * Added webp to supported image formats (#2455 @BeebBenjamin) * Introduced maxTilesPerFrame option to allow loading more tiles simultaneously (#2387 @jetic83) * Now when creating a viewer or navigator, we leave its position style alone if possible (#2393 @VIRAT9358) @@ -15,6 +15,8 @@ OPENSEADRAGON CHANGELOG * Fixed: Strange behavior if IIIF sizes were not in ascending order (#2416 @lutzhelm) * Fixed: Two-finger tap on a Mac trackpad would zoom you out (#2431 @cavenel) * Fixed: dragToPan gesture could not be disabled when flickEnabled was activated (#2464 @jonasengelmann) +* Fixed: placeholderFillStyle didn't work properly when the image was rotated (#2469 @pearcetm) +* Fixed: Sometimes exponential springs wouldn't ever settle (#2469 @pearcetm) 4.1.0: From 59645e3a0dbc321be0f503df4e6fb4b156c3fbba Mon Sep 17 00:00:00 2001 From: Ian Gilman Date: Mon, 12 Feb 2024 09:53:00 -0800 Subject: [PATCH 10/20] Changelog for #2468 --- changelog.txt | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/changelog.txt b/changelog.txt index 9e327e04..517a02ea 100644 --- a/changelog.txt +++ b/changelog.txt @@ -5,7 +5,7 @@ OPENSEADRAGON CHANGELOG * BREAKING CHANGE: Dropped support for IE11 (#2300, #2361 @AndrewADev) * DEPRECATION: The OpenSeadragon.createCallback function is no longer recommended (#2367 @akansjain) -* The viewer now uses WebGL when available (#2310, #2462, #2466, #2469 @pearcetm, @Aiosa) +* The viewer now uses WebGL when available (#2310, #2462, #2466, #2468, #2469 @pearcetm, @Aiosa, @thec0keman) * Added webp to supported image formats (#2455 @BeebBenjamin) * Introduced maxTilesPerFrame option to allow loading more tiles simultaneously (#2387 @jetic83) * Now when creating a viewer or navigator, we leave its position style alone if possible (#2393 @VIRAT9358) From 02898cfc173e076b1153ebb954ec8f7dc8a611da Mon Sep 17 00:00:00 2001 From: Tom Date: Mon, 12 Feb 2024 18:05:33 -0500 Subject: [PATCH 11/20] clean up bound event handlers in webgldrawer.destroy --- src/webgldrawer.js | 10 ++++++++-- 1 file changed, 8 insertions(+), 2 deletions(-) diff --git a/src/webgldrawer.js b/src/webgldrawer.js index e5b8f811..b747779e 100644 --- a/src/webgldrawer.js +++ b/src/webgldrawer.js @@ -93,8 +93,10 @@ this._backupCanvasDrawer = null; // Add listeners for events that require modifying the scene or camera - this.viewer.addHandler("tile-ready", ev => this._tileReadyHandler(ev)); - this.viewer.addHandler("image-unloaded", ev => this._imageUnloadedHandler(ev)); + this._boundToTileReady = ev => this._tileReadyHandler(ev); + this._boundToImageUnloaded = ev => this._imageUnloadedHandler(ev); + this.viewer.addHandler("tile-ready", this._boundToTileReady); + this.viewer.addHandler("image-unloaded", this._boundToImageUnloaded); // Reject listening for the tile-drawing and tile-drawn events, which this drawer does not fire this.viewer.rejectEventHandler("tile-drawn", "The WebGLDrawer does not raise the tile-drawn event"); @@ -155,6 +157,10 @@ ext.loseContext(); } + // unbind our event listeners from the viewer + this.viewer.removeHandler("tile-ready", this._boundToTileReady); + this.viewer.removeHandler("image-unloaded", this._boundToImageUnloaded); + // set our webgl context reference to null to enable garbage collection this._gl = null; From 99c35aa3cb5ef825e2661e4fbfe860e023c512e0 Mon Sep 17 00:00:00 2001 From: Tom Date: Wed, 21 Feb 2024 16:01:18 -0500 Subject: [PATCH 12/20] consolidate drawer creation into viewer api. add drawer-error event --- src/drawerbase.js | 31 +++++++++++++++++++++++++++++++ src/viewer.js | 29 +++++++++++++++++------------ src/webgldrawer.js | 7 ++----- test/demo/drawercomparison.js | 2 +- 4 files changed, 51 insertions(+), 18 deletions(-) diff --git a/src/drawerbase.js b/src/drawerbase.js index 29d7a3b4..0a0d04ce 100644 --- a/src/drawerbase.js +++ b/src/drawerbase.js @@ -280,6 +280,37 @@ OpenSeadragon.DrawerBase = class DrawerBase{ }); } + /** + * Called by implementations to fire the drawer-error event + * @private + */ + _raiseDrawerErrorEvent(tiledImage, errorMessage){ + if(!this.viewer) { + return; + } + + /** + * Raised when a tiled image is drawn to the canvas. Used internally for testing. + * The update-viewport event is preferred if you want to know when a frame has been drawn. + * + * @event drawer-error + * @memberof OpenSeadragon.Viewer + * @type {object} + * @property {OpenSeadragon.Viewer} eventSource - A reference to the Viewer which raised the event. + * @property {OpenSeadragon.TiledImage} tiledImage - Which TiledImage is being drawn. + * @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 + */ + this.viewer.raiseEvent( 'drawer-error', { + tiledImage: tiledImage, + drawer: this, + error: errorMessage, + }); + } + + }; }( OpenSeadragon )); diff --git a/src/viewer.js b/src/viewer.js index c3c10938..c2168c1b 100644 --- a/src/viewer.js +++ b/src/viewer.js @@ -456,7 +456,7 @@ $.Viewer = function( options ) { this.drawer = null; for (const drawerCandidate of drawerCandidates){ - let success = this.setDrawer(drawerCandidate, false); + let success = this.requestDrawer(drawerCandidate, true, false); if(success){ break; } @@ -929,19 +929,20 @@ $.extend( $.Viewer.prototype, $.EventSource.prototype, $.ControlDock.prototype, }, /** - * Set the drawer for this viewer, as a supported string or drawer constructor. - * @param {String | OpenSeadragon.DrawerBase} drawerCandidate The type of drawer to try to construct - * @param { Boolean } [redrawImmediately] Whether to immediately draw a new frame. Default = true. + * Request a drawer for this viewer, as a supported string or drawer constructor. + * @param {String | OpenSeadragon.DrawerBase} drawerCandidate The type of drawer to try to construct. + * @param { Boolean } [mainDrawer] Whether to use this as the viewer's main drawer. Default = true. + * @param { Boolean } [redrawImmediately] Whether to immediately draw a new frame. Only used if mainDrawer = true. Default = true. * @param { Object } [drawerOptions] Options for this drawer. If falsey, defaults to viewer.drawerOptions * for this viewer type. See {@link OpenSeadragon.Options}. - * @returns {Boolean} whether the drawer was created successfully + * @returns {Object | Boolean} The drawer that was created, or false if the requestd drawer is not supported */ - setDrawer(drawerCandidate, redrawImmediately = true, drawerOptions = null){ + requestDrawer(drawerCandidate, mainDrawer = true, redrawImmediately = true, drawerOptions = null){ const oldDrawer = this.drawer; let Drawer = null; - //if inherits from a drawer base, use it + //if the candidate inherits from a drawer base, use it if (drawerCandidate && drawerCandidate.prototype instanceof $.DrawerBase) { Drawer = drawerCandidate; drawerCandidate = 'custom'; @@ -957,12 +958,12 @@ $.extend( $.Viewer.prototype, $.EventSource.prototype, $.ControlDock.prototype, if (Drawer && Drawer.isSupported()) { // first destroy the previous drawer - if(oldDrawer){ + if(oldDrawer && mainDrawer){ oldDrawer.destroy(); } // create the new drawer - this.drawer = new Drawer({ + const newDrawer = new Drawer({ viewer: this, viewport: this.viewport, element: this.canvas, @@ -970,10 +971,14 @@ $.extend( $.Viewer.prototype, $.EventSource.prototype, $.ControlDock.prototype, options: drawerOptions || this.drawerOptions[drawerCandidate], }); - if(redrawImmediately){ - this.forceRedraw(); + if(mainDrawer){ + this.drawer = newDrawer; + if(redrawImmediately){ + this.forceRedraw(); + } } - return true; + + return newDrawer; } return false; diff --git a/src/webgldrawer.js b/src/webgldrawer.js index b747779e..af3ff198 100644 --- a/src/webgldrawer.js +++ b/src/webgldrawer.js @@ -231,11 +231,7 @@ */ _getBackupCanvasDrawer(){ if(!this._backupCanvasDrawer){ - this._backupCanvasDrawer = new $.CanvasDrawer({ - viewer: this.viewer, - viewport: this.viewport, - element: this.container, - }); + this._backupCanvasDrawer = this.viewer.requestDrawer('canvas', false); this._backupCanvasDrawer.canvas.style.setProperty('visibility', 'hidden'); } @@ -882,6 +878,7 @@ if(!wasTainted){ 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.'); } return; } diff --git a/test/demo/drawercomparison.js b/test/demo/drawercomparison.js index 96793915..76c780a4 100644 --- a/test/demo/drawercomparison.js +++ b/test/demo/drawercomparison.js @@ -49,7 +49,7 @@ let viewer2 = window.viewer2 = OpenSeadragon({ minZoomImageRatio:0.01, maxZoomPixelRatio:100, smoothTileEdgesMinZoom:1.1, - crossOriginPolicy: 'Anonymous', + // crossOriginPolicy: 'Anonymous', ajaxWithCredentials: false, // maxImageCacheCount: 30, drawer:drawer2, From 1442e5d4e495de4fb85f5152b34f6d45e335e7d4 Mon Sep 17 00:00:00 2001 From: Tom Date: Wed, 21 Feb 2024 16:02:29 -0500 Subject: [PATCH 13/20] revert change in drawercomparison.js used for testing --- test/demo/drawercomparison.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/test/demo/drawercomparison.js b/test/demo/drawercomparison.js index 76c780a4..96793915 100644 --- a/test/demo/drawercomparison.js +++ b/test/demo/drawercomparison.js @@ -49,7 +49,7 @@ let viewer2 = window.viewer2 = OpenSeadragon({ minZoomImageRatio:0.01, maxZoomPixelRatio:100, smoothTileEdgesMinZoom:1.1, - // crossOriginPolicy: 'Anonymous', + crossOriginPolicy: 'Anonymous', ajaxWithCredentials: false, // maxImageCacheCount: 30, drawer:drawer2, From a0bcbc4d21e32eb4d728057d3acf0baa6a5a1cc2 Mon Sep 17 00:00:00 2001 From: Tom Date: Wed, 21 Feb 2024 17:49:46 -0500 Subject: [PATCH 14/20] fix clip behavior with webgl drawer --- src/webgldrawer.js | 22 ++++++++++++++++++---- 1 file changed, 18 insertions(+), 4 deletions(-) diff --git a/src/webgldrawer.js b/src/webgldrawer.js index ac450b3c..1468b729 100644 --- a/src/webgldrawer.js +++ b/src/webgldrawer.js @@ -920,9 +920,23 @@ this._clippingContext.save(); if(item._clip){ - var box = item.imageToViewportRectangle(item._clip, true); - var rect = this.viewportToDrawerRectangle(box); - this._setClip(rect); + const polygon = [ + {x: item._clip.x, y: item._clip.y}, + {x: item._clip.x + item._clip.width, y: item._clip.y}, + {x: item._clip.x + item._clip.width, y: item._clip.y + item._clip.height}, + {x: item._clip.x, y: item._clip.y + item._clip.height}, + ]; + let clipPoints = polygon.map(coord => { + let point = item.imageToViewportCoordinates(coord.x, coord.y, true) + .rotate(this.viewer.viewport.getRotation(true), this.viewer.viewport.getCenter(true)); + let clipPoint = this.viewportCoordToDrawerCoord(point); + return clipPoint; + }); + this._clippingContext.beginPath(); + clipPoints.forEach( (coord, i) => { + this._clippingContext[i === 0 ? 'moveTo' : 'lineTo'](coord.x, coord.y); + }); + this._clippingContext.clip(); } if(item._croppingPolygons){ let polygons = item._croppingPolygons.map(polygon => { @@ -934,7 +948,7 @@ }); }); this._clippingContext.beginPath(); - polygons.forEach(polygon => { + polygons.forEach((polygon) => { polygon.forEach( (coord, i) => { this._clippingContext[i === 0 ? 'moveTo' : 'lineTo'](coord.x, coord.y); }); From ae5f08b9bdf4ce13d46beef11e8d3ea6ba6893b6 Mon Sep 17 00:00:00 2001 From: Tom Date: Wed, 21 Feb 2024 17:53:23 -0500 Subject: [PATCH 15/20] call _setClip for test spyOnce --- src/webgldrawer.js | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/src/webgldrawer.js b/src/webgldrawer.js index 1468b729..c3619503 100644 --- a/src/webgldrawer.js +++ b/src/webgldrawer.js @@ -907,10 +907,9 @@ } // private - _setClip(rect){ - this._clippingContext.beginPath(); - this._clippingContext.rect(rect.x, rect.y, rect.width, rect.height); - this._clippingContext.clip(); + _setClip(){ + // no-op: called by _renderToClippingCanvas when tiledImage._clip is truthy + // so that tests will pass. } // private @@ -937,6 +936,7 @@ this._clippingContext[i === 0 ? 'moveTo' : 'lineTo'](coord.x, coord.y); }); this._clippingContext.clip(); + this._setClip() } if(item._croppingPolygons){ let polygons = item._croppingPolygons.map(polygon => { From 5df791fc82eb4323ed6de2e20c9d51be0558c738 Mon Sep 17 00:00:00 2001 From: Tom Date: Wed, 21 Feb 2024 18:12:22 -0500 Subject: [PATCH 16/20] support viewport flipping for clip and cropping polygons in webglviewer --- src/webgldrawer.js | 15 ++++++++++++++- 1 file changed, 14 insertions(+), 1 deletion(-) diff --git a/src/webgldrawer.js b/src/webgldrawer.js index c3619503..072cbde0 100644 --- a/src/webgldrawer.js +++ b/src/webgldrawer.js @@ -917,6 +917,12 @@ this._clippingContext.clearRect(0, 0, this._clippingCanvas.width, this._clippingCanvas.height); this._clippingContext.save(); + if(this.viewer.viewport.getFlip()){ + const point = new $.Point(this.canvas.width / 2, this.canvas.height / 2); + this._clippingContext.translate(point.x, 0); + this._clippingContext.scale(-1, 1); + this._clippingContext.translate(-point.x, 0); + } if(item._clip){ const polygon = [ @@ -936,7 +942,7 @@ this._clippingContext[i === 0 ? 'moveTo' : 'lineTo'](coord.x, coord.y); }); this._clippingContext.clip(); - this._setClip() + this._setClip(); } if(item._croppingPolygons){ let polygons = item._croppingPolygons.map(polygon => { @@ -956,6 +962,13 @@ this._clippingContext.clip(); } + if(this.viewer.viewport.getFlip()){ + const point = new $.Point(this.canvas.width / 2, this.canvas.height / 2); + this._clippingContext.translate(point.x, 0); + this._clippingContext.scale(-1, 1); + this._clippingContext.translate(-point.x, 0); + } + this._clippingContext.drawImage(this._renderingCanvas, 0, 0); this._clippingContext.restore(); From 7dfc60a97fa2ce88213d7a1228749fd7f444aa65 Mon Sep 17 00:00:00 2001 From: Ian Gilman Date: Thu, 22 Feb 2024 09:31:32 -0800 Subject: [PATCH 17/20] Changelog for #2478 --- changelog.txt | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/changelog.txt b/changelog.txt index 517a02ea..d4b009b2 100644 --- a/changelog.txt +++ b/changelog.txt @@ -5,7 +5,7 @@ OPENSEADRAGON CHANGELOG * BREAKING CHANGE: Dropped support for IE11 (#2300, #2361 @AndrewADev) * DEPRECATION: The OpenSeadragon.createCallback function is no longer recommended (#2367 @akansjain) -* The viewer now uses WebGL when available (#2310, #2462, #2466, #2468, #2469 @pearcetm, @Aiosa, @thec0keman) +* The viewer now uses WebGL when available (#2310, #2462, #2466, #2468, #2469, #2478 @pearcetm, @Aiosa, @thec0keman) * Added webp to supported image formats (#2455 @BeebBenjamin) * Introduced maxTilesPerFrame option to allow loading more tiles simultaneously (#2387 @jetic83) * Now when creating a viewer or navigator, we leave its position style alone if possible (#2393 @VIRAT9358) From b6501a3786b3d48c5b628bacb552ac9b4f6c0968 Mon Sep 17 00:00:00 2001 From: Tom Date: Thu, 22 Feb 2024 13:30:05 -0500 Subject: [PATCH 18/20] fix typo in docs. change to options object for requestDrawer API --- src/viewer.js | 23 +++++++++++++++++------ src/webgldrawer.js | 2 +- 2 files changed, 18 insertions(+), 7 deletions(-) diff --git a/src/viewer.js b/src/viewer.js index c2168c1b..037e3016 100644 --- a/src/viewer.js +++ b/src/viewer.js @@ -456,7 +456,7 @@ $.Viewer = function( options ) { this.drawer = null; for (const drawerCandidate of drawerCandidates){ - let success = this.requestDrawer(drawerCandidate, true, false); + let success = this.requestDrawer(drawerCandidate, {mainDrawer: true, redrawImmediately: false}); if(success){ break; } @@ -931,13 +931,24 @@ $.extend( $.Viewer.prototype, $.EventSource.prototype, $.ControlDock.prototype, /** * Request a drawer for this viewer, as a supported string or drawer constructor. * @param {String | OpenSeadragon.DrawerBase} drawerCandidate The type of drawer to try to construct. - * @param { Boolean } [mainDrawer] Whether to use this as the viewer's main drawer. Default = true. - * @param { Boolean } [redrawImmediately] Whether to immediately draw a new frame. Only used if mainDrawer = true. Default = true. - * @param { Object } [drawerOptions] Options for this drawer. If falsey, defaults to viewer.drawerOptions + * @param { Object } options + * @param { Boolean } [options.mainDrawer] Whether to use this as the viewer's main drawer. Default = true. + * @param { Boolean } [options.redrawImmediately] Whether to immediately draw a new frame. Only used if options.mainDrawer = true. Default = true. + * @param { Object } [options.drawerOptions] Options for this drawer. Defaults to viewer.drawerOptions. * for this viewer type. See {@link OpenSeadragon.Options}. - * @returns {Object | Boolean} The drawer that was created, or false if the requestd drawer is not supported + * @returns {Object | Boolean} The drawer that was created, or false if the requested drawer is not supported */ - requestDrawer(drawerCandidate, mainDrawer = true, redrawImmediately = true, drawerOptions = null){ + requestDrawer(drawerCandidate, options){ + const defaultOpts = { + mainDrawer: true, + redrawImmediately: true, + drawerOptions: null + }; + options = $.extend(true, defaultOpts, options); + const mainDrawer = options.mainDrawer; + const redrawImmediately = options.redrawImmediately; + const drawerOptions = options.drawerOptions; + const oldDrawer = this.drawer; let Drawer = null; diff --git a/src/webgldrawer.js b/src/webgldrawer.js index e7a23fc9..d236a944 100644 --- a/src/webgldrawer.js +++ b/src/webgldrawer.js @@ -231,7 +231,7 @@ */ _getBackupCanvasDrawer(){ if(!this._backupCanvasDrawer){ - this._backupCanvasDrawer = this.viewer.requestDrawer('canvas', false); + this._backupCanvasDrawer = this.viewer.requestDrawer('canvas', {mainDrawer: false}); this._backupCanvasDrawer.canvas.style.setProperty('visibility', 'hidden'); } From 1c0797e8003395f23875c8f39af0bd70cf8c54ea Mon Sep 17 00:00:00 2001 From: Ian Gilman Date: Tue, 27 Feb 2024 09:35:01 -0800 Subject: [PATCH 19/20] Changelog for #2472 --- changelog.txt | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/changelog.txt b/changelog.txt index d4b009b2..13ee1015 100644 --- a/changelog.txt +++ b/changelog.txt @@ -5,7 +5,7 @@ OPENSEADRAGON CHANGELOG * BREAKING CHANGE: Dropped support for IE11 (#2300, #2361 @AndrewADev) * DEPRECATION: The OpenSeadragon.createCallback function is no longer recommended (#2367 @akansjain) -* The viewer now uses WebGL when available (#2310, #2462, #2466, #2468, #2469, #2478 @pearcetm, @Aiosa, @thec0keman) +* The viewer now uses WebGL when available (#2310, #2462, #2466, #2468, #2469, #2472, #2478 @pearcetm, @Aiosa, @thec0keman) * Added webp to supported image formats (#2455 @BeebBenjamin) * Introduced maxTilesPerFrame option to allow loading more tiles simultaneously (#2387 @jetic83) * Now when creating a viewer or navigator, we leave its position style alone if possible (#2393 @VIRAT9358) From e4fd005fb0cd78f43cab981c7b45b8c72237e949 Mon Sep 17 00:00:00 2001 From: Tom Date: Thu, 29 Feb 2024 16:44:09 -0500 Subject: [PATCH 20/20] Mark TiledImage as needing an update when position, scale, or rotation are set with immediately=true --- src/tiledimage.js | 11 +++++++++-- 1 file changed, 9 insertions(+), 2 deletions(-) diff --git a/src/tiledimage.js b/src/tiledimage.js index 65a83d76..20fa2ebf 100644 --- a/src/tiledimage.js +++ b/src/tiledimage.js @@ -159,7 +159,8 @@ $.TiledImage = function( options ) { loadingCoverage: {}, // A '3d' dictionary [level][x][y] --> Boolean; shows what areas are loaded or are being loaded/blended. lastDrawn: [], // An unordered list of Tiles drawn last frame. lastResetTime: 0, // Last time for which the tiledImage was reset. - _needsDraw: true, // Does the tiledImage need to update the viewport again? + _needsDraw: true, // Does the tiledImage need to be drawn again? + _needsUpdate: true, // Does the tiledImage need to update the viewport again? _hasOpaqueTile: false, // Do we have even one fully opaque tile? _tilesLoading: 0, // The number of pending tile requests. _tilesToDraw: [], // info about the tiles currently in the viewport, two deep: array[level][tile] @@ -298,13 +299,14 @@ $.extend($.TiledImage.prototype, $.EventSource.prototype, /** @lends OpenSeadrag let scaleUpdated = this._scaleSpring.update(); let degreesUpdated = this._degreesSpring.update(); - let updated = (xUpdated || yUpdated || scaleUpdated || degreesUpdated); + let updated = (xUpdated || yUpdated || scaleUpdated || degreesUpdated || this._needsUpdate); if (updated || viewportChanged || !this._fullyLoaded){ let fullyLoadedFlag = this._updateLevelsForViewport(); this._setFullyLoaded(fullyLoadedFlag); } + this._needsUpdate = false; if (updated) { this._updateForScale(); @@ -697,6 +699,7 @@ $.extend($.TiledImage.prototype, $.EventSource.prototype, /** @lends OpenSeadrag this._xSpring.resetTo(position.x); this._ySpring.resetTo(position.y); this._needsDraw = true; + this._needsUpdate = true; } else { if (sameTarget) { return; @@ -705,6 +708,7 @@ $.extend($.TiledImage.prototype, $.EventSource.prototype, /** @lends OpenSeadrag this._xSpring.springTo(position.x); this._ySpring.springTo(position.y); this._needsDraw = true; + this._needsUpdate = true; } if (!sameTarget) { @@ -1034,6 +1038,7 @@ $.extend($.TiledImage.prototype, $.EventSource.prototype, /** @lends OpenSeadrag this._degreesSpring.springTo(degrees); } this._needsDraw = true; + this._needsUpdate = true; this._raiseBoundsChange(); }, @@ -1232,6 +1237,7 @@ $.extend($.TiledImage.prototype, $.EventSource.prototype, /** @lends OpenSeadrag this._scaleSpring.resetTo(scale); this._updateForScale(); this._needsDraw = true; + this._needsUpdate = true; } else { if (sameTarget) { return; @@ -1240,6 +1246,7 @@ $.extend($.TiledImage.prototype, $.EventSource.prototype, /** @lends OpenSeadrag this._scaleSpring.springTo(scale); this._updateForScale(); this._needsDraw = true; + this._needsUpdate = true; } if (!sameTarget) {