From 233c3a357a1af50f80c232492ad6850d25fc16e5 Mon Sep 17 00:00:00 2001 From: Luke Murray Date: Fri, 12 Jul 2013 16:51:56 +1000 Subject: [PATCH 1/4] Add the option for a zoom handler callback --- src/drawer.js | 5 ++++- src/overlay.js | 5 +++++ 2 files changed, 9 insertions(+), 1 deletion(-) diff --git a/src/drawer.js b/src/drawer.js index a5cc8ac9..10281190 100644 --- a/src/drawer.js +++ b/src/drawer.js @@ -430,7 +430,10 @@ $.Drawer.prototype = { $.OverlayPlacement[overlay.placement.toUpperCase()] ); }else{ - return new $.Overlay( element, rect ); + var newOverlay = new $.Overlay(element, rect); + if (overlay.zoomHandler) + newOverlay.zoomHandler = overlay.zoomHandler; + return newOverlay; } } diff --git a/src/overlay.js b/src/overlay.js index d175eb7c..acee3316 100644 --- a/src/overlay.js +++ b/src/overlay.js @@ -176,6 +176,11 @@ this.size = $.getElementSize( element ); } + if (this.zoomHandler) { + this.zoomHandler(this.position, this.size, element); + return; + } + position = this.position; size = this.size; From df877493a9c9eaec9cd16afd356043bfbbb2d90a Mon Sep 17 00:00:00 2001 From: Luke Murray Date: Tue, 16 Jul 2013 12:04:37 +1000 Subject: [PATCH 2/4] Rename drawHandler to onDraw. Also make it a notification style callback (does not override the draw functionality). Add an overlay option 'useTransform' that will transform the overlay element instead of moving it's postion - useful for SVG elements --- src/drawer.js | 44 ++++++++++++++++++++--------- src/overlay.js | 77 ++++++++++++++++++++++++++++++++------------------ 2 files changed, 81 insertions(+), 40 deletions(-) diff --git a/src/drawer.js b/src/drawer.js index 10281190..dad5c617 100644 --- a/src/drawer.js +++ b/src/drawer.js @@ -144,7 +144,7 @@ $.Drawer = function( options ) { for( i = 0; i < this.overlays.length; i++ ){ if( $.isPlainObject( this.overlays[ i ] ) ){ - this.overlays[ i ] = addOverlayFromConfiguration( this, this.overlays[ i ]); + this.overlays[ i ] = addOverlayFromConfiguration( this, this.overlays[ i ], this.source.dimensions); } else if ( $.isFunction( this.overlays[ i ] ) ){ //TODO @@ -168,8 +168,12 @@ $.Drawer.prototype = { * @param {OpenSeadragon.OverlayPlacement} placement - The position of the * viewport which the location coordinates will be treated as relative * to. + * @param {function} onDraw - A callback that is called when the overlay + * needs to be drawn. It is passed position, size and element. + * @param {boolean} useTransform - Overlay will be scaled and moved using + * the SVG transform argument. Overlay should be a SVG g element. */ - addOverlay: function( element, location, placement ) { + addOverlay: function( element, location, placement, onDraw, useTransform ) { element = $.getElement( element ); if ( getOverlayIndex( this.overlays, element ) >= 0 ) { @@ -177,7 +181,14 @@ $.Drawer.prototype = { return; } - this.overlays.push( new $.Overlay( element, location, placement ) ); + this.overlays.push( new $.Overlay({ + element: element, + location: location, + placement: placement, + onDraw: onDraw, + useTransform: useTransform, + imageFullSize: this.source.dimensions + }) ); this.updateAgain = true; if( this.viewer ){ this.viewer.raiseEvent( 'add-overlay', { @@ -390,7 +401,7 @@ $.Drawer.prototype = { * @private * @inner */ - function addOverlayFromConfiguration( drawer, overlay ){ + function addOverlayFromConfiguration( drawer, overlay, dimensions ){ var element = null, rect = ( overlay.height && overlay.width ) ? new $.Rect( @@ -423,17 +434,24 @@ $.Drawer.prototype = { //we need to translate to viewport coordinates rect = drawer.viewport.imageToViewportRectangle( rect ); } + if( overlay.placement ){ - return new $.Overlay( - element, - drawer.viewport.pointFromPixel(rect), - $.OverlayPlacement[overlay.placement.toUpperCase()] - ); + return new $.Overlay({ + element: element, + location: drawer.viewport.pointFromPixel(rect), + placement: $.OverlayPlacement[overlay.placement.toUpperCase()], + onDraw: overlay.onDraw, + useTransform: overlay.useTransform, + imageFullSize: dimensions + }); }else{ - var newOverlay = new $.Overlay(element, rect); - if (overlay.zoomHandler) - newOverlay.zoomHandler = overlay.zoomHandler; - return newOverlay; + return new $.Overlay({ + element: element, + location: rect, + onDraw: overlay.onDraw, + useTransform: overlay.useTransform, + imageFullSize: dimensions + }); } } diff --git a/src/overlay.js b/src/overlay.js index acee3316..a4ae2f9e 100644 --- a/src/overlay.js +++ b/src/overlay.js @@ -57,27 +57,42 @@ * @class */ $.Overlay = function( element, location, placement ) { - this.element = element; - this.scales = location instanceof $.Rect; + + var options; + if( $.isPlainObject( element ) ){ + options = element; + } else{ + options = { + element: element, + location: location, + placement: placement + }; + } + + this.element = options.element; + this.scales = options.location instanceof $.Rect; this.bounds = new $.Rect( - location.x, - location.y, - location.width, - location.height + options.location.x, + options.location.y, + options.location.width, + options.location.height ); this.position = new $.Point( - location.x, - location.y + options.location.x, + options.location.y ); this.size = new $.Point( - location.width, - location.height + options.location.width, + options.location.height ); - this.style = element.style; + this.style = options.element.style; // rects are always top-left - this.placement = location instanceof $.Point ? - placement : + this.placement = options.location instanceof $.Point ? + options.placement : $.OverlayPlacement.TOP_LEFT; + this.onDraw = options.onDraw; + this.useTransform = options.useTransform; + this.imageFullSize = options.imageFullSize; }; $.Overlay.prototype = { @@ -176,27 +191,35 @@ this.size = $.getElementSize( element ); } - if (this.zoomHandler) { - this.zoomHandler(this.position, this.size, element); - return; - } - position = this.position; size = this.size; this.adjust( position, size ); - position = position.apply( Math.floor ); - size = size.apply( Math.ceil ); + if(this.useTransform){ + var scale = Math.min(this.size.x / this.imageFullSize.x, this.size.y / this.imageFullSize.y); - style.left = position.x + "px"; - style.top = position.y + "px"; - style.position = "absolute"; - style.display = 'block'; + var attrValue = 'translate(' + position.x + ', ' + position.y + ') scale(' + scale + ')'; + // we expect the first element to be the g element of the SVG element that we can transform + element.firstElementChild.setAttribute('transform', attrValue); + }else{ + position = position.apply( Math.floor ); + size = size.apply( Math.ceil ); - if ( scales ) { - style.width = size.x + "px"; - style.height = size.y + "px"; + style.left = position.x + "px"; + style.top = position.y + "px"; + style.position = "absolute"; + style.display = 'block'; + + if ( scales ) { + style.width = size.x + "px"; + style.height = size.y + "px"; + } + } + + // call the onDraw callback if there is one to allow them to dping + if (this.onDraw) { + this.onDraw(this.position, this.size, element); } }, From 4239bb7adcba8b8b7119220b2bbf879ccf7cb0f9 Mon Sep 17 00:00:00 2001 From: Luke Murray Date: Wed, 31 Jul 2013 17:01:48 +1000 Subject: [PATCH 3/4] fix: remove the useTransform/SVG stuff and move back to a simple onDraw callback only so people can handle their own custom drawing and sizing --- src/drawer.js | 51 +++++++++++++++++++++++++++----------------------- src/overlay.js | 21 ++++++--------------- 2 files changed, 34 insertions(+), 38 deletions(-) diff --git a/src/drawer.js b/src/drawer.js index dad5c617..e5b5c0d1 100644 --- a/src/drawer.js +++ b/src/drawer.js @@ -144,7 +144,7 @@ $.Drawer = function( options ) { for( i = 0; i < this.overlays.length; i++ ){ if( $.isPlainObject( this.overlays[ i ] ) ){ - this.overlays[ i ] = addOverlayFromConfiguration( this, this.overlays[ i ], this.source.dimensions); + this.overlays[ i ] = addOverlayFromConfiguration( this, this.overlays[ i ]); } else if ( $.isFunction( this.overlays[ i ] ) ){ //TODO @@ -161,20 +161,31 @@ $.Drawer.prototype = { * highlighting words or areas of interest on an image or other zoomable * interface. * @method - * @param {Element|String} element - A reference to an element or an id for - * the element which will overlayed. + * @param {Element|String|Object} element - A reference to an element or an id for + * the element which will overlayed. Or an Object specifying the configuration for the overlay * @param {OpenSeadragon.Point|OpenSeadragon.Rect} location - The point or * rectangle which will be overlayed. * @param {OpenSeadragon.OverlayPlacement} placement - The position of the * viewport which the location coordinates will be treated as relative * to. - * @param {function} onDraw - A callback that is called when the overlay - * needs to be drawn. It is passed position, size and element. - * @param {boolean} useTransform - Overlay will be scaled and moved using - * the SVG transform argument. Overlay should be a SVG g element. + * @param {function} onDraw - If supplied the callback is called when the overlay + * needs to be drawn. It it the responsibility of the callback to do any drawing/positioning. + * It is passed position, size and element. */ - addOverlay: function( element, location, placement, onDraw, useTransform ) { - element = $.getElement( element ); + addOverlay: function( element, location, placement, onDraw ) { + var options; + if( $.isPlainObject( element ) ){ + options = element; + } else { + options = { + element: element, + location: location, + placement: placement, + onDraw: onDraw + }; + } + + element = $.getElement(options.element); if ( getOverlayIndex( this.overlays, element ) >= 0 ) { // they're trying to add a duplicate overlay @@ -183,19 +194,17 @@ $.Drawer.prototype = { this.overlays.push( new $.Overlay({ element: element, - location: location, - placement: placement, - onDraw: onDraw, - useTransform: useTransform, - imageFullSize: this.source.dimensions + location: options.location, + placement: options.placement, + onDraw: onDraw }) ); this.updateAgain = true; if( this.viewer ){ this.viewer.raiseEvent( 'add-overlay', { viewer: this.viewer, element: element, - location: location, - placement: placement + location: options.location, + placement: options.placement }); } return this; @@ -401,7 +410,7 @@ $.Drawer.prototype = { * @private * @inner */ - function addOverlayFromConfiguration( drawer, overlay, dimensions ){ + function addOverlayFromConfiguration( drawer, overlay ){ var element = null, rect = ( overlay.height && overlay.width ) ? new $.Rect( @@ -440,17 +449,13 @@ $.Drawer.prototype = { element: element, location: drawer.viewport.pointFromPixel(rect), placement: $.OverlayPlacement[overlay.placement.toUpperCase()], - onDraw: overlay.onDraw, - useTransform: overlay.useTransform, - imageFullSize: dimensions + onDraw: overlay.onDraw }); }else{ return new $.Overlay({ element: element, location: rect, - onDraw: overlay.onDraw, - useTransform: overlay.useTransform, - imageFullSize: dimensions + onDraw: overlay.onDraw }); } diff --git a/src/overlay.js b/src/overlay.js index a4ae2f9e..a71db596 100644 --- a/src/overlay.js +++ b/src/overlay.js @@ -91,7 +91,6 @@ options.placement : $.OverlayPlacement.TOP_LEFT; this.onDraw = options.onDraw; - this.useTransform = options.useTransform; this.imageFullSize = options.imageFullSize; }; @@ -196,16 +195,13 @@ this.adjust( position, size ); - if(this.useTransform){ - var scale = Math.min(this.size.x / this.imageFullSize.x, this.size.y / this.imageFullSize.y); - - var attrValue = 'translate(' + position.x + ', ' + position.y + ') scale(' + scale + ')'; - // we expect the first element to be the g element of the SVG element that we can transform - element.firstElementChild.setAttribute('transform', attrValue); - }else{ - position = position.apply( Math.floor ); - size = size.apply( Math.ceil ); + position = position.apply( Math.floor ); + size = size.apply( Math.ceil ); + // call the onDraw callback if there is one to allow them to dping + if (this.onDraw) { + this.onDraw(this.position, this.size, element); + } else { style.left = position.x + "px"; style.top = position.y + "px"; style.position = "absolute"; @@ -216,11 +212,6 @@ style.height = size.y + "px"; } } - - // call the onDraw callback if there is one to allow them to dping - if (this.onDraw) { - this.onDraw(this.position, this.size, element); - } }, /** From 271739818125b5e25f33c40f82fa2e0e1ea8ceb9 Mon Sep 17 00:00:00 2001 From: Luke Murray Date: Thu, 1 Aug 2013 13:36:00 +1000 Subject: [PATCH 4/4] Use the correct onDraw in drawer.js. Remove imageFullSize and fix up the comment for onDraw --- src/drawer.js | 2 +- src/overlay.js | 6 +++--- 2 files changed, 4 insertions(+), 4 deletions(-) diff --git a/src/drawer.js b/src/drawer.js index e5b5c0d1..75c18631 100644 --- a/src/drawer.js +++ b/src/drawer.js @@ -196,7 +196,7 @@ $.Drawer.prototype = { element: element, location: options.location, placement: options.placement, - onDraw: onDraw + onDraw: options.onDraw }) ); this.updateAgain = true; if( this.viewer ){ diff --git a/src/overlay.js b/src/overlay.js index a71db596..8635499f 100644 --- a/src/overlay.js +++ b/src/overlay.js @@ -91,7 +91,6 @@ options.placement : $.OverlayPlacement.TOP_LEFT; this.onDraw = options.onDraw; - this.imageFullSize = options.imageFullSize; }; $.Overlay.prototype = { @@ -198,9 +197,10 @@ position = position.apply( Math.floor ); size = size.apply( Math.ceil ); - // call the onDraw callback if there is one to allow them to dping + // call the onDraw callback if there is one to allow, this allows someone to overwrite + // the drawing/positioning/sizing of the overlay if (this.onDraw) { - this.onDraw(this.position, this.size, element); + this.onDraw(position, size, element); } else { style.left = position.x + "px"; style.top = position.y + "px";