diff --git a/.eslintrc.hound.json b/.eslintrc.hound.json new file mode 100644 index 00000000..027542f4 --- /dev/null +++ b/.eslintrc.hound.json @@ -0,0 +1,285 @@ +{ + "env": { + "browser": true + }, + "extends": "eslint:recommended", + "rules": { + "indent": [ + "error", + 4 + ], + "linebreak-style": [ + "error", + "unix" + ], + "quotes": [ + "error", + "double" + ], + "semi": [ + "error", + "always" + ], + "no-unused-vars": [ + "error", + { + "args": "none" + } + ], + "block-scoped-var": [ + "error" + ], + "consistent-return": [ + "error" + ], + "curly": [ + "error", + "all" + ], + "eqeqeq": [ + "error" + ], + "no-eval": [ + "error" + ], + "no-implicit-globals": [ + "error" + ], + "no-implied-eval": [ + "error" + ], + "no-invalid-this": [ + "error" + ], + "no-multi-spaces": [ + "error", + { + "exceptions": { + "Property": true, + "VariableDeclarator": true, + "AssignmentExpression": true + } + } + ], + "no-new-wrappers": [ + "error" + ], + "no-new": [ + "error" + ], + "no-return-assign": [ + "error" + ], + "no-self-compare": [ + "error" + ], + "no-unmodified-loop-condition": [ + "error" + ], + "no-unused-expressions": [ + "error" + ], + "no-useless-call": [ + "error" + ], + "no-useless-concat": [ + "error" + ], + "no-useless-escape": [ + "error" + ], + "no-useless-return": [ + "error" + ], + "no-with": [ + "error" + ], + "radix": [ + "error" + ], + "yoda": [ + "error" + ], + "no-undef-init": [ + "error" + ], + "no-use-before-define": [ + "error", + { + "functions": false + } + ], + "array-bracket-spacing": [ + "error", + "never" + ], + "block-spacing": [ + "error" + ], + "brace-style": [ + "error" + ], + "camelcase": [ + "error" + ], + "comma-spacing": [ + "error" + ], + "comma-style": [ + "error" + ], + "computed-property-spacing": [ + "error" + ], + "consistent-this": [ + "error", + "self" + ], + "eol-last": [ + "error" + ], + "func-call-spacing": [ + "error" + ], + "func-name-matching": [ + "error" + ], + "key-spacing": [ + "error", + { + "mode": "minimum" + } + ], + "keyword-spacing": [ + "error" + ], + "max-len": [ + "error", + 80 + ], + "max-statements-per-line": [ + "error", + { + "max": 1 + } + ], + "new-cap": [ + "error" + ], + "new-parens": [ + "error" + ], + "no-array-constructor": [ + "error" + ], + "no-mixed-operators": [ + "error", + { + "groups": [ + [ + "&", + "|", + "^", + "~", + "<<", + ">>", + ">>>" + ], + [ + "==", + "!=", + "===", + "!==", + ">", + ">=", + "<", + "<=" + ], + [ + "&&", + "||" + ], + [ + "in", + "instanceof" + ] + ] + } + ], + "no-new-object": [ + "error" + ], + "no-tabs": [ + "error" + ], + "no-trailing-spaces": [ + "error" + ], + "no-unneeded-ternary": [ + "error" + ], + "no-whitespace-before-property": [ + "error" + ], + "object-curly-spacing": [ + "error", + "always" + ], + "one-var-declaration-per-line": [ + "error" + ], + "one-var": [ + "error", + "never" + ], + "operator-assignment": [ + "error" + ], + "operator-linebreak": [ + "error", + "after" + ], + "quote-props": [ + "error", + "as-needed" + ], + "semi-spacing": [ + "error" + ], + "space-before-blocks": [ + "error" + ], + "space-before-function-paren": [ + "error", + "never" + ], + "space-in-parens": [ + "error", + "never" + ], + "space-infix-ops": [ + "error" + ], + "space-unary-ops": [ + "error", + { + "words": false, + "nonwords": false + } + ], + "unicode-bom": [ + "error" + ], + "no-caller": [ + "error" + ], + "no-loop-func": [ + "error" + ] + }, + "globals": { + "OpenSeadragon": true, + "define": false, + "module": false + } +} diff --git a/.eslintrc.json b/.eslintrc.json new file mode 100644 index 00000000..7e027656 --- /dev/null +++ b/.eslintrc.json @@ -0,0 +1,285 @@ +{ + "env": { + "browser": true + }, + "extends": "eslint:recommended", + "rules": { + "indent": [ + "off", + 4 + ], + "linebreak-style": [ + "error", + "unix" + ], + "quotes": [ + "off", + "double" + ], + "semi": [ + "error", + "always" + ], + "no-unused-vars": [ + "error", + { + "args": "none" + } + ], + "block-scoped-var": [ + "error" + ], + "consistent-return": [ + "off" + ], + "curly": [ + "error", + "all" + ], + "eqeqeq": [ + "off" + ], + "no-eval": [ + "error" + ], + "no-implicit-globals": [ + "error" + ], + "no-implied-eval": [ + "error" + ], + "no-invalid-this": [ + "error" + ], + "no-multi-spaces": [ + "error", + { + "exceptions": { + "Property": true, + "VariableDeclarator": true, + "AssignmentExpression": true + } + } + ], + "no-new-wrappers": [ + "error" + ], + "no-new": [ + "error" + ], + "no-return-assign": [ + "error" + ], + "no-self-compare": [ + "error" + ], + "no-unmodified-loop-condition": [ + "error" + ], + "no-unused-expressions": [ + "error" + ], + "no-useless-call": [ + "error" + ], + "no-useless-concat": [ + "error" + ], + "no-useless-escape": [ + "off" + ], + "no-useless-return": [ + "error" + ], + "no-with": [ + "error" + ], + "radix": [ + "error" + ], + "yoda": [ + "off" + ], + "no-undef-init": [ + "error" + ], + "no-use-before-define": [ + "error", + { + "functions": false + } + ], + "array-bracket-spacing": [ + "off", + "never" + ], + "block-spacing": [ + "off" + ], + "brace-style": [ + "off" + ], + "camelcase": [ + "error" + ], + "comma-spacing": [ + "error" + ], + "comma-style": [ + "error" + ], + "computed-property-spacing": [ + "off" + ], + "consistent-this": [ + "off", + "self" + ], + "eol-last": [ + "error" + ], + "func-call-spacing": [ + "error" + ], + "func-name-matching": [ + "error" + ], + "key-spacing": [ + "error", + { + "mode": "minimum" + } + ], + "keyword-spacing": [ + "off" + ], + "max-len": [ + "off", + 80 + ], + "max-statements-per-line": [ + "error", + { + "max": 1 + } + ], + "new-cap": [ + "error" + ], + "new-parens": [ + "error" + ], + "no-array-constructor": [ + "error" + ], + "no-mixed-operators": [ + "error", + { + "groups": [ + [ + "&", + "|", + "^", + "~", + "<<", + ">>", + ">>>" + ], + [ + "==", + "!=", + "===", + "!==", + ">", + ">=", + "<", + "<=" + ], + [ + "&&", + "||" + ], + [ + "in", + "instanceof" + ] + ] + } + ], + "no-new-object": [ + "error" + ], + "no-tabs": [ + "error" + ], + "no-trailing-spaces": [ + "error" + ], + "no-unneeded-ternary": [ + "error" + ], + "no-whitespace-before-property": [ + "error" + ], + "object-curly-spacing": [ + "off", + "always" + ], + "one-var-declaration-per-line": [ + "error" + ], + "one-var": [ + "off", + "never" + ], + "operator-assignment": [ + "error" + ], + "operator-linebreak": [ + "error", + "after" + ], + "quote-props": [ + "off", + "as-needed" + ], + "semi-spacing": [ + "error" + ], + "space-before-blocks": [ + "off" + ], + "space-before-function-paren": [ + "off", + "never" + ], + "space-in-parens": [ + "off", + "never" + ], + "space-infix-ops": [ + "error" + ], + "space-unary-ops": [ + "error", + { + "words": false, + "nonwords": false + } + ], + "unicode-bom": [ + "error" + ], + "no-caller": [ + "error" + ], + "no-loop-func": [ + "error" + ] + }, + "globals": { + "OpenSeadragon": true, + "define": false, + "module": false + } +} diff --git a/.hound.yml b/.hound.yml new file mode 100644 index 00000000..8355a088 --- /dev/null +++ b/.hound.yml @@ -0,0 +1,3 @@ +eslint: + enabled: true + config_file: .eslintrc.hound.json diff --git a/.jshintrc b/.jshintrc deleted file mode 100644 index 013e31e8..00000000 --- a/.jshintrc +++ /dev/null @@ -1,16 +0,0 @@ -{ - "browser": true, - "curly": true, - "eqeqeq": false, - "loopfunc": false, - "noarg": true, - "trailing": true, - "undef": true, - "unused": false, - - "globals": { - "OpenSeadragon": true, - "define": false, - "module": false - } -} diff --git a/.vscode/settings.json b/.vscode/settings.json index 752fa89f..686e7452 100644 --- a/.vscode/settings.json +++ b/.vscode/settings.json @@ -1,8 +1,5 @@ // Place your settings in this file to overwrite default and user settings. { - // Controls the font size. - "editor.fontSize": 14, - // The number of spaces a tab is equal to. "editor.tabSize": 4, @@ -15,12 +12,6 @@ // Columns at which to show vertical rulers "editor.rulers": [80], - // Controls after how many characters the editor will wrap to the next line. Setting this to 0 turns on viewport width wrapping - "editor.wrappingColumn": 0, - - // Controls the indentation of wrapped lines. Can be one of 'none', 'same' or 'indent'. - "editor.wrappingIndent": "none", - // The default character set encoding to use when reading and writing files. "files.encoding": "utf8", diff --git a/Gruntfile.js b/Gruntfile.js index d1e86ae2..c2e821ad 100644 --- a/Gruntfile.js +++ b/Gruntfile.js @@ -5,12 +5,12 @@ module.exports = function(grunt) { // ---------- grunt.loadNpmTasks("grunt-contrib-compress"); grunt.loadNpmTasks("grunt-contrib-concat"); - grunt.loadNpmTasks("grunt-contrib-jshint"); grunt.loadNpmTasks("grunt-contrib-uglify"); grunt.loadNpmTasks("grunt-qunit-istanbul"); grunt.loadNpmTasks("grunt-contrib-connect"); grunt.loadNpmTasks("grunt-contrib-watch"); grunt.loadNpmTasks("grunt-contrib-clean"); + grunt.loadNpmTasks("grunt-eslint"); grunt.loadNpmTasks("grunt-git-describe"); grunt.loadNpmTasks('grunt-text-replace'); @@ -123,10 +123,11 @@ module.exports = function(grunt) { join_vars: false }, sourceMap: true, - sourceMapName: 'build/openseadragon/openseadragon.min.js.map' + sourceMapName: 'build/openseadragon/openseadragon.min.js.map', + sourceMapIn: 'build/openseadragon/openseadragon.js.map' }, openseadragon: { - src: sources, + src: distribution, dest: minified } }, @@ -186,12 +187,11 @@ module.exports = function(grunt) { files: [ "Gruntfile.js", "src/*.js", "images/*" ], tasks: "watchTask" }, - jshint: { + eslint: { options: { - jshintrc: '.jshintrc' + configFile: '.eslintrc.json' }, - beforeconcat: sources, - afterconcat: [ distribution ] + target: sources }, "git-describe": { build: {} @@ -264,8 +264,8 @@ module.exports = function(grunt) { // Build task. // Cleans out the build folder and builds the code and images into it, checking lint. grunt.registerTask("build", [ - "clean:build", "jshint:beforeconcat", "git-describe", "concat", "jshint:afterconcat", - "uglify", "replace:cleanPaths", "copy:build" + "clean:build", "git-describe", "eslint", "concat", "uglify", + "replace:cleanPaths", "copy:build" ]); // ---------- diff --git a/changelog.txt b/changelog.txt index 99ee6973..3d2767bf 100644 --- a/changelog.txt +++ b/changelog.txt @@ -1,25 +1,48 @@ OPENSEADRAGON CHANGELOG ======================= -2.2.2: (in progress) +2.3.0: (in progress) * BREAKING CHANGE: Tile.distance has been removed (#1027) +* BREAKING CHANGE: Viewer's canvas-click event is now fired before it initiates the zoom (#1148) +* BREAKING CHANGE: Viewer's canvas-drag event is now fired before it pans (#1149) +* Optimization: Use the squared distance when comparing tiles (#1027) +* You can now prevent canvas-click events from zooming on a per-event basis (#1148) +* You can now prevent canvas-drag events from panning on a per-event basis (#1149) * You can now set the rotation of individual tiled images (#1006) * Fixed CORS bug in IE 10 (#967) * Added support for commonjs (#984) * Added an option to addTiledImage to change the crossOriginPolicy (#981) -* Fixed issue with tiles not appearing with wrapHorizontal/wrapVertical if you pan too far away from the origin (#987) +* Fixed issue with tiles not appearing with wrapHorizontal/wrapVertical if you pan too far away from the origin (#987, #1066) * The Viewer's tileSources option is now smarter about detecting JSON vs XML vs URL (#999) * The navigationControlAnchor option now works for custom toolbar as well (#1004) -* Added getFullyLoaded method and "fully-loaded-change" event to TiledImage to know when tiles are fully loaded (#837) +* Added getFullyLoaded method and "fully-loaded-change" event to TiledImage to know when tiles are fully loaded (#837, #1073) * Fixed: Initial tile load wasn't happening in parallel (#1014) * Added Zoomify tile source (#863) * Fixed problem with "sparse image" DZI files (#995) -* Optimization: Use the squared distance when comparing tiles (#1027) * Fix IndexSizeError on IE and Edge that occurred under certain circumstances (e.g. multi-image with transparency) (#1035) * ImageTileSource now works in IE8 (#1041) * LegacyTileSource now allows any image URLs regardless of type (#1056) * Fixed error in IE8 when zooming in (due to edge smoothing) (#1064) +* Improved DziTileSource guessing of tilesUrl (#1074) +* Fixed issue with OpenSeadragon.version in the minified JavaScript (#1099) +* Fixed smoothTileEdgesMinZoom performance degradation on single-tile images (#1101) +* Fixed issue with tiles not appearing after rotation (#1102) +* Fixed: The navigator wasn't respecting the constrainDuringPan setting (#1104) +* Fixed an issue causing overlays to be mis-positioned in some circumstances (#1119) +* Fixed: ImageTileSource would sometimes produce a double image (#1123) +* Fixed: console.debug caused exceptions on IE10 (#1129) +* Better compression for our UI images (#1134) +* Fixed: the reference strip would leak memory when opening new sets of images (#1175) +* You can now load tiles via AJAX and custom AJAX request headers (#1055) +* Added fix for supporting weird filenames that look like JSONs (#1189) +* Fixed: zoomTo/zoomBy ignore refPoint if immediately is true (#1184) +* Enabled configuration of ImageLoader timeout (#1192) +* Viewer.open() now supports an initialPage argument for sequenceMode (#1196) +* Fixed: IIPImageServer didn't work with the latest OSD release (#1199) +* Now clamping pixel ratio density to a minimum of 1, fixing display issues on low density devices (#1200) +* Improved calculation for determining which level to load first (#1198) +* Fixed: setItemIndex method not working with navigator inside "open" event (#1201) 2.2.1: diff --git a/images/button_grouphover.png b/images/button_grouphover.png new file mode 100755 index 00000000..9db590ea Binary files /dev/null and b/images/button_grouphover.png differ diff --git a/images/button_hover.png b/images/button_hover.png new file mode 100755 index 00000000..645c241b Binary files /dev/null and b/images/button_hover.png differ diff --git a/images/button_pressed.png b/images/button_pressed.png new file mode 100755 index 00000000..d5b1d7d0 Binary files /dev/null and b/images/button_pressed.png differ diff --git a/images/button_rest.png b/images/button_rest.png new file mode 100755 index 00000000..e232387d Binary files /dev/null and b/images/button_rest.png differ diff --git a/images/fullpage_grouphover.png b/images/fullpage_grouphover.png index 3ca4e1e3..da9002b1 100644 Binary files a/images/fullpage_grouphover.png and b/images/fullpage_grouphover.png differ diff --git a/images/fullpage_hover.png b/images/fullpage_hover.png index d08eb2d0..705b3d3e 100644 Binary files a/images/fullpage_hover.png and b/images/fullpage_hover.png differ diff --git a/images/fullpage_pressed.png b/images/fullpage_pressed.png index 0ee45b5f..6fa182ae 100644 Binary files a/images/fullpage_pressed.png and b/images/fullpage_pressed.png differ diff --git a/images/fullpage_rest.png b/images/fullpage_rest.png index 3172005c..bfab6433 100644 Binary files a/images/fullpage_rest.png and b/images/fullpage_rest.png differ diff --git a/images/home_grouphover.png b/images/home_grouphover.png index 204e1cc9..cb412ba4 100644 Binary files a/images/home_grouphover.png and b/images/home_grouphover.png differ diff --git a/images/home_hover.png b/images/home_hover.png index ec218a00..c8f860ba 100644 Binary files a/images/home_hover.png and b/images/home_hover.png differ diff --git a/images/home_pressed.png b/images/home_pressed.png index 4439508b..00c349b0 100644 Binary files a/images/home_pressed.png and b/images/home_pressed.png differ diff --git a/images/home_rest.png b/images/home_rest.png index 009d1bbf..6ac397da 100644 Binary files a/images/home_rest.png and b/images/home_rest.png differ diff --git a/images/next_grouphover.png b/images/next_grouphover.png index 8d83d8a1..18c1a925 100644 Binary files a/images/next_grouphover.png and b/images/next_grouphover.png differ diff --git a/images/next_hover.png b/images/next_hover.png index ba24ca98..fdce5820 100644 Binary files a/images/next_hover.png and b/images/next_hover.png differ diff --git a/images/next_pressed.png b/images/next_pressed.png index 95f169d6..5297c526 100644 Binary files a/images/next_pressed.png and b/images/next_pressed.png differ diff --git a/images/next_rest.png b/images/next_rest.png index 5ead544b..e3c5a3ca 100644 Binary files a/images/next_rest.png and b/images/next_rest.png differ diff --git a/images/previous_grouphover.png b/images/previous_grouphover.png index 016e6395..5e0fda1b 100644 Binary files a/images/previous_grouphover.png and b/images/previous_grouphover.png differ diff --git a/images/previous_hover.png b/images/previous_hover.png index d4a5c155..9f9efe61 100644 Binary files a/images/previous_hover.png and b/images/previous_hover.png differ diff --git a/images/previous_pressed.png b/images/previous_pressed.png index f999fe49..75c7e7d2 100644 Binary files a/images/previous_pressed.png and b/images/previous_pressed.png differ diff --git a/images/previous_rest.png b/images/previous_rest.png index 9716dac6..902a7b45 100644 Binary files a/images/previous_rest.png and b/images/previous_rest.png differ diff --git a/images/rotateleft_grouphover.png b/images/rotateleft_grouphover.png index 9aec7ac9..302ac628 100644 Binary files a/images/rotateleft_grouphover.png and b/images/rotateleft_grouphover.png differ diff --git a/images/rotateleft_hover.png b/images/rotateleft_hover.png index ba32c5a4..e757d87a 100644 Binary files a/images/rotateleft_hover.png and b/images/rotateleft_hover.png differ diff --git a/images/rotateleft_pressed.png b/images/rotateleft_pressed.png index b968ebf8..1480b1ae 100644 Binary files a/images/rotateleft_pressed.png and b/images/rotateleft_pressed.png differ diff --git a/images/rotateleft_rest.png b/images/rotateleft_rest.png index ebbf081b..f0b86541 100644 Binary files a/images/rotateleft_rest.png and b/images/rotateleft_rest.png differ diff --git a/images/rotateright_grouphover.png b/images/rotateright_grouphover.png index 86e8689c..9e713712 100644 Binary files a/images/rotateright_grouphover.png and b/images/rotateright_grouphover.png differ diff --git a/images/rotateright_hover.png b/images/rotateright_hover.png index d22a728f..08f14160 100644 Binary files a/images/rotateright_hover.png and b/images/rotateright_hover.png differ diff --git a/images/rotateright_pressed.png b/images/rotateright_pressed.png index fc2ead64..351f8243 100644 Binary files a/images/rotateright_pressed.png and b/images/rotateright_pressed.png differ diff --git a/images/rotateright_rest.png b/images/rotateright_rest.png index 07219678..d70468ac 100644 Binary files a/images/rotateright_rest.png and b/images/rotateright_rest.png differ diff --git a/images/zoomin_grouphover.png b/images/zoomin_grouphover.png index c985d0f9..98ecd291 100644 Binary files a/images/zoomin_grouphover.png and b/images/zoomin_grouphover.png differ diff --git a/images/zoomin_hover.png b/images/zoomin_hover.png index 3cab721f..c25bda42 100644 Binary files a/images/zoomin_hover.png and b/images/zoomin_hover.png differ diff --git a/images/zoomin_pressed.png b/images/zoomin_pressed.png index 9c3a7516..e617e034 100644 Binary files a/images/zoomin_pressed.png and b/images/zoomin_pressed.png differ diff --git a/images/zoomin_rest.png b/images/zoomin_rest.png index f4219a50..4380589e 100644 Binary files a/images/zoomin_rest.png and b/images/zoomin_rest.png differ diff --git a/images/zoomout_grouphover.png b/images/zoomout_grouphover.png index 46d21b3e..b588ecf7 100644 Binary files a/images/zoomout_grouphover.png and b/images/zoomout_grouphover.png differ diff --git a/images/zoomout_hover.png b/images/zoomout_hover.png index 7b924c26..a132cb45 100644 Binary files a/images/zoomout_hover.png and b/images/zoomout_hover.png differ diff --git a/images/zoomout_pressed.png b/images/zoomout_pressed.png index c028db72..679c5cda 100644 Binary files a/images/zoomout_pressed.png and b/images/zoomout_pressed.png differ diff --git a/images/zoomout_rest.png b/images/zoomout_rest.png index a13e07de..e3ac4abd 100644 Binary files a/images/zoomout_rest.png and b/images/zoomout_rest.png differ diff --git a/package.json b/package.json index 19bd4d21..2fd0cad7 100644 --- a/package.json +++ b/package.json @@ -18,12 +18,12 @@ "grunt": "^0.4.5", "grunt-contrib-clean": "^0.7.0", "grunt-contrib-compress": "^0.13.0", - "grunt-contrib-concat": "^0.5.1", + "grunt-contrib-concat": "^1.0.1", "grunt-contrib-connect": "^0.11.2", - "grunt-contrib-jshint": "^0.11.0", - "grunt-contrib-uglify": "^0.11.0", + "grunt-contrib-uglify": "^2.0.0", "grunt-contrib-watch": "^0.6.1", "grunt-git-describe": "^2.3.2", + "grunt-eslint": "^19.0.0", "grunt-qunit-istanbul": "^0.6.0", "grunt-text-replace": "^0.4.0", "qunitjs": "^1.20.0" diff --git a/psd/button.psd b/psd/button.psd new file mode 100755 index 00000000..d1574b81 Binary files /dev/null and b/psd/button.psd differ diff --git a/src/button.js b/src/button.js index 42f68afc..0ddb919a 100644 --- a/src/button.js +++ b/src/button.js @@ -61,7 +61,7 @@ $.ButtonState = { * @memberof OpenSeadragon * @extends OpenSeadragon.EventSource * @param {Object} options - * @param {Element} [options.element=null] Element to use as the button. If not specified, an HTML <button> element is created. + * @param {Element} [options.element=null] Element to use as the button. If not specified, an HTML <div> element is created. * @param {String} [options.tooltip=null] Provides context help for the button when the * user hovers over it. * @param {String} [options.srcRest=null] URL of image to use in 'rest' state. @@ -120,7 +120,7 @@ $.Button = function( options ) { * @member {Element} element * @memberof OpenSeadragon.Button# */ - this.element = options.element || $.makeNeutralElement( "div" ); + this.element = options.element || $.makeNeutralElement("div"); //if the user has specified the element to bind the control to explicitly //then do not add the default control images @@ -158,7 +158,7 @@ $.Button = function( options ) { this.imgDown.style.visibility = "hidden"; - if ( $.Browser.vendor == $.BROWSERS.FIREFOX && $.Browser.version < 3 ){ + if ($.Browser.vendor == $.BROWSERS.FIREFOX && $.Browser.version < 3) { this.imgGroup.style.top = this.imgHover.style.top = this.imgDown.style.top = @@ -172,13 +172,13 @@ $.Button = function( options ) { } - this.addHandler( "press", this.onPress ); - this.addHandler( "release", this.onRelease ); - this.addHandler( "click", this.onClick ); - this.addHandler( "enter", this.onEnter ); - this.addHandler( "exit", this.onExit ); - this.addHandler( "focus", this.onFocus ); - this.addHandler( "blur", this.onBlur ); + this.addHandler("press", this.onPress); + this.addHandler("release", this.onRelease); + this.addHandler("click", this.onClick); + this.addHandler("enter", this.onEnter); + this.addHandler("exit", this.onExit); + this.addHandler("focus", this.onFocus); + this.addHandler("blur", this.onBlur); /** * The button's current state. diff --git a/src/control.js b/src/control.js index c1ba2186..5a63c517 100644 --- a/src/control.js +++ b/src/control.js @@ -115,10 +115,10 @@ $.Control = function ( element, options, container ) { if ( this.anchor == $.ControlAnchor.ABSOLUTE ) { this.wrapper = $.makeNeutralElement( "div" ); this.wrapper.style.position = "absolute"; - this.wrapper.style.top = typeof ( options.top ) == "number" ? ( options.top + 'px' ) : options.top; - this.wrapper.style.left = typeof ( options.left ) == "number" ? (options.left + 'px' ) : options.left; - this.wrapper.style.height = typeof ( options.height ) == "number" ? ( options.height + 'px' ) : options.height; - this.wrapper.style.width = typeof ( options.width ) == "number" ? ( options.width + 'px' ) : options.width; + this.wrapper.style.top = typeof (options.top) == "number" ? (options.top + 'px') : options.top; + this.wrapper.style.left = typeof (options.left) == "number" ? (options.left + 'px') : options.left; + this.wrapper.style.height = typeof (options.height) == "number" ? (options.height + 'px') : options.height; + this.wrapper.style.width = typeof (options.width) == "number" ? (options.width + 'px') : options.width; this.wrapper.style.margin = "0px"; this.wrapper.style.padding = "0px"; diff --git a/src/controldock.js b/src/controldock.js index 2ee2d0f1..81443bf9 100644 --- a/src/controldock.js +++ b/src/controldock.js @@ -45,7 +45,7 @@ i; $.extend( true, this, { - id: 'controldock-'+$.now()+'-'+Math.floor(Math.random()*1000000), + id: 'controldock-' + $.now() + '-' + Math.floor(Math.random() * 1000000), container: $.makeNeutralElement( 'div' ), controls: [] }, options ); diff --git a/src/drawer.js b/src/drawer.js index 02a036aa..4485da98 100644 --- a/src/drawer.js +++ b/src/drawer.js @@ -285,8 +285,8 @@ $.Drawer.prototype = { return new $.Rect( topLeft.x * $.pixelDensityRatio, topLeft.y * $.pixelDensityRatio, - size.x * $.pixelDensityRatio, - size.y * $.pixelDensityRatio + size.x * $.pixelDensityRatio, + size.y * $.pixelDensityRatio ); }, @@ -328,7 +328,11 @@ $.Drawer.prototype = { // the viewport get rotated later on, we will need to resize it. if (this.viewport.getRotation() === 0) { var self = this; - this.viewer.addOnceHandler('rotate', function resizeSketchCanvas() { + this.viewer.addHandler('rotate', function resizeSketchCanvas() { + if (self.viewport.getRotation() === 0) { + return; + } + self.viewer.removeHandler('rotate', resizeSketchCanvas); var sketchCanvasSize = self._calculateSketchCanvasSize(); self.sketchCanvas.width = sketchCanvasSize.x; self.sketchCanvas.height = sketchCanvasSize.y; @@ -422,8 +426,8 @@ $.Drawer.prototype = { this.context.globalCompositeOperation = compositeOperation; } if (bounds) { - // Internet Explorer and Microsoft Edge throw IndexSizeError - // when you call context.drawImage with negative x or y + // Internet Explorer and Microsoft Edge throw IndexSizeError + // when you call context.drawImage with negative x or y // or width or height greater than the canvas width or height respectively if (bounds.x < 0) { bounds.width += bounds.x; @@ -439,7 +443,7 @@ $.Drawer.prototype = { if (bounds.height > this.canvas.height) { bounds.height = this.canvas.height; } - + this.context.drawImage( this.sketchCanvas, bounds.x, @@ -470,7 +474,7 @@ $.Drawer.prototype = { position.x - widthExt * scale, position.y - heightExt * scale, (this.canvas.width + 2 * widthExt) * scale, - (this.canvas.height + 2 * heightExt) * scale, + (this.canvas.height + 2 * heightExt) * scale, -widthExt, -heightExt, this.canvas.width + 2 * widthExt, @@ -496,9 +500,9 @@ $.Drawer.prototype = { if ( this.viewport.degrees !== 0 ) { this._offsetForRotation({degrees: this.viewport.degrees}); } - if (tiledImage.getRotation() !== 0) { + if (tiledImage.getRotation(true) % 360 !== 0) { this._offsetForRotation({ - degrees: tiledImage.getRotation(), + degrees: tiledImage.getRotation(true), point: tiledImage.viewport.pixelFromPointNoRotate( tiledImage._getRotationPoint(true), true) }); @@ -565,7 +569,7 @@ $.Drawer.prototype = { if ( this.viewport.degrees !== 0 ) { this._restoreRotationChanges(); } - if (tiledImage.getRotation() !== 0) { + if (tiledImage.getRotation(true) % 360 !== 0) { this._restoreRotationChanges(); } context.restore(); diff --git a/src/dzitilesource.js b/src/dzitilesource.js index 6c95dbee..443bfaca 100644 --- a/src/dzitilesource.js +++ b/src/dzitilesource.js @@ -140,7 +140,7 @@ $.extend( $.DziTileSource.prototype, $.TileSource.prototype, /** @lends OpenSead if (url && !options.tilesUrl) { options.tilesUrl = url.replace( - /([^\/]+?)(\.(dzi|xml|js))?\/?(\?.*)?$/, '$1_files/'); + /([^\/]+?)(\.(dzi|xml|js)?(\?[^\/]*)?)?\/?$/, '$1_files/'); if (url.search(/\.(dzi|xml|js)\?/) != -1) { options.queryParams = url.match(/\?.*/); diff --git a/src/iiiftilesource.js b/src/iiiftilesource.js index 4584f1cd..0af45c8b 100644 --- a/src/iiiftilesource.js +++ b/src/iiiftilesource.js @@ -45,6 +45,7 @@ */ $.IIIFTileSource = function( options ){ + /* eslint-disable camelcase */ $.extend( true, this, options ); @@ -86,7 +87,7 @@ $.IIIFTileSource = function( options ){ } else if ( canBeTiled(options.profile) ) { // use the largest of tileOptions that is smaller than the short dimension var shortDim = Math.min( this.height, this.width ), - tileOptions = [256,512,1024], + tileOptions = [256, 512, 1024], smallerTiles = []; for ( var c = 0; c < tileOptions.length; c++ ) { @@ -102,11 +103,11 @@ $.IIIFTileSource = function( options ){ options.tileSize = shortDim; } } else if (this.sizes && this.sizes.length > 0) { - // This info.json can't be tiled, but we can still construct a legacy pyramid from the sizes array. - // In this mode, IIIFTileSource will call functions from the abstract baseTileSource or the - // LegacyTileSource instead of performing IIIF tiling. + // This info.json can't be tiled, but we can still construct a legacy pyramid from the sizes array. + // In this mode, IIIFTileSource will call functions from the abstract baseTileSource or the + // LegacyTileSource instead of performing IIIF tiling. this.emulateLegacyImagePyramid = true; - + options.levels = constructLevels( this ); // use the largest available size to define tiles $.extend( true, options, { @@ -141,7 +142,7 @@ $.extend( $.IIIFTileSource.prototype, $.TileSource.prototype, /** @lends OpenSea * @param {Object|Array} data * @param {String} optional - url */ - + supports: function( data, url ) { // Version 2.0 and forwards if (data.protocol && data.protocol == 'http://iiif.io/api/image') { @@ -393,14 +394,16 @@ $.extend( $.IIIFTileSource.prototype, $.TileSource.prototype, /** @lends OpenSea */ function constructLevels(options) { var levels = []; - for(var i=0; i 0) { + if ((!loader.jobLimit || loader.jobsInProgress < loader.jobLimit) && loader.jobQueue.length > 0) { nextJob = loader.jobQueue.shift(); nextJob.start(); loader.jobsInProgress++; } - callback( job.image, job.errorMsg ); + callback(job.image, job.errorMsg, job.request); } -}( OpenSeadragon )); +}(OpenSeadragon)); diff --git a/src/legacytilesource.js b/src/legacytilesource.js index ad2a0ffe..e24dbbe6 100644 --- a/src/legacytilesource.js +++ b/src/legacytilesource.js @@ -217,7 +217,7 @@ function filterFiles( files ){ } } - return filtered.sort(function(a,b){ + return filtered.sort(function(a, b) { return a.height - b.height; }); @@ -253,7 +253,7 @@ function configureFromXML( tileSource, xmlDoc ){ for ( i = 0; i < levels.length; i++ ) { level = levels[ i ]; - conf.levels .push({ + conf.levels.push({ url: level.getAttribute( "url" ), width: parseInt( level.getAttribute( "width" ), 10 ), height: parseInt( level.getAttribute( "height" ), 10 ) diff --git a/src/mousetracker.js b/src/mousetracker.js index 498a72ac..3dbcf595 100644 --- a/src/mousetracker.js +++ b/src/mousetracker.js @@ -160,6 +160,7 @@ * @memberof OpenSeadragon.MouseTracker# */ this.dblClickDistThreshold = options.dblClickDistThreshold || $.DEFAULT_SETTINGS.dblClickDistThreshold; + /*eslint-disable no-multi-spaces*/ this.userData = options.userData || null; this.stopDelay = options.stopDelay || 50; @@ -182,6 +183,7 @@ this.keyHandler = options.keyHandler || null; this.focusHandler = options.focusHandler || null; this.blurHandler = options.blurHandler || null; + /*eslint-enable no-multi-spaces*/ //Store private properties in a scope sealed hash map var _this = this; @@ -1370,6 +1372,7 @@ eventParams = getCaptureEventParams( tracker, $.MouseTracker.havePointerEvents ? 'pointerevent' : pointerType ); // We emulate mouse capture by hanging listeners on the document object. // (Note we listen on the capture phase so the captured handlers will get called first) + // eslint-disable-next-line no-use-before-define if (isInIframe && canAccessEvents(window.top)) { $.addEvent( window.top, @@ -1413,6 +1416,7 @@ eventParams = getCaptureEventParams( tracker, $.MouseTracker.havePointerEvents ? 'pointerevent' : pointerType ); // We emulate mouse capture by hanging listeners on the document object. // (Note we listen on the capture phase so the captured handlers will get called first) + // eslint-disable-next-line no-use-before-define if (isInIframe && canAccessEvents(window.top)) { $.removeEvent( window.top, @@ -1703,7 +1707,7 @@ // Calculate deltaY if ( $.MouseTracker.wheelEventName == "mousewheel" ) { - simulatedEvent.deltaY = - 1 / $.DEFAULT_SETTINGS.pixelsPerWheelLine * event.wheelDelta; + simulatedEvent.deltaY = -event.wheelDelta / $.DEFAULT_SETTINGS.pixelsPerWheelLine; } else { simulatedEvent.deltaY = event.detail; } @@ -2207,11 +2211,8 @@ * @inner */ function onTouchCancel( tracker, event ) { - var i, - touchCount = event.changedTouches.length, - gPoints = [], - pointsList = tracker.getActivePointersListByType( 'touch' ); - + var pointsList = tracker.getActivePointersListByType('touch'); + abortTouchContacts( tracker, event, pointsList ); } @@ -2565,8 +2566,7 @@ * Gesture points associated with the event. */ function updatePointersExit( tracker, event, gPoints ) { - var delegate = THIS[ tracker.hash ], - pointsList = tracker.getActivePointersListByType( gPoints[ 0 ].type ), + var pointsList = tracker.getActivePointersListByType(gPoints[0].type), i, gPointCount = gPoints.length, curGPoint, @@ -2801,7 +2801,6 @@ var delegate = THIS[ tracker.hash ], pointsList = tracker.getActivePointersListByType( gPoints[ 0 ].type ), propagate, - insideElementReleased, releasePoint, releaseTime, i, @@ -2866,7 +2865,7 @@ { eventSource: tracker, pointerType: gPoints[ 0 ].type, - position: getPointRelativeToAbsolute( gPoints[ 0 ].currentPos, tracker.element ), + position: getPointRelativeToAbsolute(gPoints[0].currentPos, tracker.element), button: buttonChanged, buttons: pointsList.buttons, isTouchEvent: gPoints[ 0 ].type === 'touch', @@ -3267,7 +3266,7 @@ } ); } } - + // True if inside an iframe, otherwise false. // @member {Boolean} isInIframe // @private @@ -3279,7 +3278,7 @@ return true; } })(); - + // @function // @private // @inner @@ -3292,4 +3291,4 @@ } } -} ( OpenSeadragon ) ); +}(OpenSeadragon)); diff --git a/src/navigator.js b/src/navigator.js index fe977d0c..12273b79 100644 --- a/src/navigator.js +++ b/src/navigator.js @@ -121,7 +121,7 @@ $.Navigator = function( options ){ //At some browser magnification levels the display regions lines up correctly, but at some there appears to //be a one pixel gap. this.fudge = new $.Point(1, 1); - this.totalBorderWidths = new $.Point(this.borderWidth*2, this.borderWidth*2).minus(this.fudge); + this.totalBorderWidths = new $.Point(this.borderWidth * 2, this.borderWidth * 2).minus(this.fudge); if ( options.controlOptions.anchor != $.ControlAnchor.NONE ) { @@ -180,8 +180,8 @@ $.Navigator = function( options ){ if ( this._resizeWithViewer ) { if ( options.width && options.height ) { - this.element.style.height = typeof ( options.height ) == "number" ? ( options.height + 'px' ) : options.height; - this.element.style.width = typeof ( options.width ) == "number" ? ( options.width + 'px' ) : options.width; + this.element.style.height = typeof (options.height) == "number" ? (options.height + 'px') : options.height; + this.element.style.width = typeof (options.width) == "number" ? (options.width + 'px') : options.width; } else { viewerSize = $.getElementSize( viewer.element ); this.element.style.height = Math.round( viewerSize.y * options.sizeRatio ) + 'px'; @@ -231,8 +231,10 @@ $.Navigator = function( options ){ }); viewer.world.addHandler("item-index-change", function(event) { - var item = _this.world.getItemAt(event.previousIndex); - _this.world.setItemIndex(item, event.newIndex); + window.setTimeout(function(){ + var item = _this.world.getItemAt(event.previousIndex); + _this.world.setItemIndex(item, event.newIndex); + }, 1); }); viewer.world.addHandler("remove-item", function(event) { @@ -369,9 +371,11 @@ $.extend( $.Navigator.prototype, $.EventSource.prototype, $.Viewer.prototype, /* // private _matchBounds: function(myItem, theirItem, immediately) { - var bounds = theirItem.getBounds(); + var bounds = theirItem.getBoundsNoRotate(); myItem.setPosition(bounds.getTopLeft(), immediately); myItem.setWidth(bounds.width, immediately); + myItem.setRotation(theirItem.getRotation(), immediately); + myItem.setClip(theirItem.getClip()); } }); @@ -405,6 +409,9 @@ function onCanvasDrag( event ) { event.delta ) ); + if( this.viewer.constrainDuringPan ){ + this.viewer.viewport.applyConstraints(); + } } } diff --git a/src/openseadragon.js b/src/openseadragon.js index cdaee4e5..041dc7aa 100644 --- a/src/openseadragon.js +++ b/src/openseadragon.js @@ -415,6 +415,7 @@ * The max number of images we should keep in memory (per drawer). * * @property {Number} [timeout=30000] + * The max number of milliseconds that an image job may take to complete. * * @property {Boolean} [useCanvas=true] * Set to false to not use an HTML canvas element for image rendering even if canvas is supported. @@ -588,9 +589,16 @@ * not use CORS, and the canvas will be tainted. * * @property {Boolean} [ajaxWithCredentials=false] - * Whether to set the withCredentials XHR flag for AJAX requests (when loading tile sources). + * Whether to set the withCredentials XHR flag for AJAX requests. * Note that this can be overridden at the {@link OpenSeadragon.TileSource} level. * + * @property {Boolean} [loadTilesWithAjax=false] + * Whether to load tile data using AJAX requests. + * Note that this can be overridden at the {@link OpenSeadragon.TileSource} level. + * + * @property {Object} [ajaxHeaders={}] + * A set of headers to include when making AJAX requests for tile sources or tiles. + * */ /** @@ -863,7 +871,7 @@ function OpenSeadragon( options ){ try { // We test if the canvas is tainted by retrieving data from it. // An exception will be raised if the canvas is tainted. - var data = canvas.getContext('2d').getImageData(0, 0, 1, 1); + canvas.getContext('2d').getImageData(0, 0, 1, 1); } catch (e) { isTainted = true; } @@ -871,7 +879,8 @@ function OpenSeadragon( options ){ }; /** - * A ratio comparing the device screen's pixel density to the canvas's backing store pixel density. Defaults to 1 if canvas isn't supported by the browser. + * A ratio comparing the device screen's pixel density to the canvas's backing store pixel density, + * clamped to a minimum of 1. Defaults to 1 if canvas isn't supported by the browser. * @member {Number} pixelDensityRatio * @memberof OpenSeadragon */ @@ -884,7 +893,7 @@ function OpenSeadragon( options ){ context.msBackingStorePixelRatio || context.oBackingStorePixelRatio || context.backingStorePixelRatio || 1; - return devicePixelRatio / backingStoreRatio; + return Math.max(devicePixelRatio, 1) / backingStoreRatio; } else { return 1; } @@ -1009,6 +1018,8 @@ function OpenSeadragon( options ){ initialPage: 0, crossOriginPolicy: false, ajaxWithCredentials: false, + loadTilesWithAjax: false, + ajaxHeaders: {}, //PAN AND ZOOM SETTINGS AND CONSTRAINTS panHorizontal: true, @@ -1030,10 +1041,46 @@ function OpenSeadragon( options ){ dblClickDistThreshold: 20, springStiffness: 6.5, animationTime: 1.2, - gestureSettingsMouse: { scrollToZoom: true, clickToZoom: true, dblClickToZoom: false, pinchToZoom: false, flickEnabled: false, flickMinSpeed: 120, flickMomentum: 0.25, pinchRotate: false }, - gestureSettingsTouch: { scrollToZoom: false, clickToZoom: false, dblClickToZoom: true, pinchToZoom: true, flickEnabled: true, flickMinSpeed: 120, flickMomentum: 0.25, pinchRotate: false }, - gestureSettingsPen: { scrollToZoom: false, clickToZoom: true, dblClickToZoom: false, pinchToZoom: false, flickEnabled: false, flickMinSpeed: 120, flickMomentum: 0.25, pinchRotate: false }, - gestureSettingsUnknown: { scrollToZoom: false, clickToZoom: false, dblClickToZoom: true, pinchToZoom: true, flickEnabled: true, flickMinSpeed: 120, flickMomentum: 0.25, pinchRotate: false }, + gestureSettingsMouse: { + scrollToZoom: true, + clickToZoom: true, + dblClickToZoom: false, + pinchToZoom: false, + flickEnabled: false, + flickMinSpeed: 120, + flickMomentum: 0.25, + pinchRotate: false + }, + gestureSettingsTouch: { + scrollToZoom: false, + clickToZoom: false, + dblClickToZoom: true, + pinchToZoom: true, + flickEnabled: true, + flickMinSpeed: 120, + flickMomentum: 0.25, + pinchRotate: false + }, + gestureSettingsPen: { + scrollToZoom: false, + clickToZoom: true, + dblClickToZoom: false, + pinchToZoom: false, + flickEnabled: false, + flickMinSpeed: 120, + flickMomentum: 0.25, + pinchRotate: false + }, + gestureSettingsUnknown: { + scrollToZoom: false, + clickToZoom: false, + dblClickToZoom: true, + pinchToZoom: true, + flickEnabled: true, + flickMinSpeed: 120, + flickMomentum: 0.25, + pinchRotate: false + }, zoomPerClick: 2, zoomPerScroll: 1.2, zoomPerSecond: 1.0, @@ -1509,7 +1556,7 @@ function OpenSeadragon( options ){ }; } else { // We can't reassign the function yet, as there was no scroll. - return new $.Point(0,0); + return new $.Point(0, 0); } return $.getPageScroll(); @@ -1677,13 +1724,15 @@ function OpenSeadragon( options ){ * @function */ now: function( ) { - if (Date.now) { - $.now = Date.now; - } else { - $.now = function() { return new Date().getTime(); }; - } + if (Date.now) { + $.now = Date.now; + } else { + $.now = function() { + return new Date().getTime(); + }; + } - return $.now(); + return $.now(); }, @@ -1793,7 +1842,7 @@ function OpenSeadragon( options ){ addClass: function( element, className ) { element = $.getElement( element ); - if ( ! element.className ) { + if (!element.className) { element.className = className; } else if ( ( ' ' + element.className + ' ' ). indexOf( ' ' + className + ' ' ) === -1 ) { @@ -2017,6 +2066,7 @@ function OpenSeadragon( options ){ * @returns {String} The value of the url parameter or null if no param matches. */ getUrlParameter: function( key ) { + // eslint-disable-next-line no-use-before-define var value = URLPARAMS[ key ]; return value ? value : null; }, @@ -2086,11 +2136,16 @@ function OpenSeadragon( options ){ * @param {String} options.url - the url to request * @param {Function} options.success - a function to call on a successful response * @param {Function} options.error - a function to call on when an error occurs + * @param {Object} options.headers - headers to add to the AJAX request + * @param {String} options.responseType - the response type of the the AJAX request * @param {Boolean} [options.withCredentials=false] - whether to set the XHR's withCredentials * @throws {Error} + * @returns {XMLHttpRequest} */ makeAjaxRequest: function( url, onSuccess, onError ) { var withCredentials; + var headers; + var responseType; // Note that our preferred API is that you pass in a single object; the named // arguments are for legacy support. @@ -2098,6 +2153,8 @@ function OpenSeadragon( options ){ onSuccess = url.success; onError = url.error; withCredentials = url.withCredentials; + headers = url.headers; + responseType = url.responseType || null; url = url.url; } @@ -2113,9 +2170,9 @@ function OpenSeadragon( options ){ if ( request.readyState == 4 ) { request.onreadystatechange = function(){}; - // With protocols other than http/https, the status is 200 - // on Firefox and 0 on other browsers - if ( request.status === 200 || + // With protocols other than http/https, a successful request status is in + // the 200's on Firefox and 0 on other browsers + if ( (request.status >= 200 && request.status < 300) || ( request.status === 0 && protocol !== "http:" && protocol !== "https:" )) { @@ -2133,11 +2190,23 @@ function OpenSeadragon( options ){ try { request.open( "GET", url, true ); + if (responseType) { + request.responseType = responseType; + } + + if (headers) { + for (var headerName in headers) { + if (headers.hasOwnProperty(headerName) && headers[headerName]) { + request.setRequestHeader(headerName, headers[headerName]); + } + } + } + if (withCredentials) { request.withCredentials = true; } - request.send( null ); + request.send(null); } catch (e) { var msg = e.message; @@ -2174,7 +2243,7 @@ function OpenSeadragon( options ){ } }; xdr.onerror = function (e) { - if ( $.isFunction ( onError ) ) { + if ($.isFunction(onError)) { onError({ // Faking an xhr object responseText: xdr.responseText, status: 444, // 444 No Response @@ -2197,6 +2266,8 @@ function OpenSeadragon( options ){ } } } + + return request; }, /** @@ -2337,6 +2408,7 @@ function OpenSeadragon( options ){ // Should only be used by IE8 in non standards mode $.parseJSON = function(string) { /*jshint evil:true*/ + //eslint-disable-next-line no-eval return eval('(' + string + ')'); }; } @@ -2352,6 +2424,7 @@ function OpenSeadragon( options ){ */ imageFormatSupported: function( extension ) { extension = extension ? extension : ""; + // eslint-disable-next-line no-use-before-define return !!FILEFORMATS[ extension.toLowerCase() ]; } @@ -2388,8 +2461,7 @@ function OpenSeadragon( options ){ (function() { //A small auto-executing routine to determine the browser vendor, //version and supporting feature sets. - var app = navigator.appName, - ver = navigator.appVersion, + var ver = navigator.appVersion, ua = navigator.userAgent, regex; @@ -2411,7 +2483,7 @@ function OpenSeadragon( options ){ } break; case "Netscape": - if( !!window.addEventListener ){ + if (window.addEventListener) { if ( ua.indexOf( "Firefox" ) >= 0 ) { $.Browser.vendor = $.BROWSERS.FIREFOX; $.Browser.version = parseFloat( diff --git a/src/overlay.js b/src/overlay.js index 25cc50d2..9d9edd88 100644 --- a/src/overlay.js +++ b/src/overlay.js @@ -41,6 +41,7 @@ * compatibility. * @member OverlayPlacement * @memberof OpenSeadragon + * @see OpenSeadragon.Placement * @static * @readonly * @type {Object} @@ -247,6 +248,8 @@ element.prevNextSibling = element.nextSibling; container.appendChild(element); + // have to set position before calculating size, fix #1116 + this.style.position = "absolute"; // this.size is used by overlays which don't get scaled in at // least one direction when this.checkResize is set to false. this.size = $.getElementSize(element); @@ -285,7 +288,6 @@ style[transformProp] = ""; } } - style.position = "absolute"; if (style.display !== 'none') { style.display = 'block'; @@ -400,7 +402,7 @@ * @param {OpenSeadragon.Point|OpenSeadragon.Rect|Object} location * If an object is specified, the options are the same than the constructor * except for the element which can not be changed. - * @param {OpenSeadragon.Placement} position + * @param {OpenSeadragon.Placement} placement */ update: function(location, placement) { var options = $.isPlainObject(location) ? location : { diff --git a/src/referencestrip.js b/src/referencestrip.js index 9d88a6a3..69fa8e29 100644 --- a/src/referencestrip.js +++ b/src/referencestrip.js @@ -178,6 +178,7 @@ $.ReferenceStrip = function ( options ) { this.panelWidth = ( viewerSize.x * this.sizeRatio ) + 8; this.panelHeight = ( viewerSize.y * this.sizeRatio ) + 8; this.panels = []; + this.miniViewers = {}; /*jshint loopfunc:true*/ for ( i = 0; i < viewer.tileSources.length; i++ ) { @@ -293,6 +294,12 @@ $.extend( $.ReferenceStrip.prototype, $.EventSource.prototype, $.Viewer.prototyp // Overrides Viewer.destroy destroy: function() { + if (this.miniViewers) { + for (var key in this.miniViewers) { + this.miniViewers[key].destroy(); + } + } + if (this.element) { this.element.parentNode.removeChild(this.element); } @@ -463,6 +470,8 @@ function loadPanels( strip, viewerSize, scroll ) { miniViewer.displayRegion ); + strip.miniViewers[element.id] = miniViewer; + element.activePanel = true; } } @@ -592,6 +601,4 @@ function onKeyPress( event ) { } } - - -} ( OpenSeadragon ) ); +}(OpenSeadragon)); diff --git a/src/spring.js b/src/spring.js index 71c94d06..30a8956d 100644 --- a/src/spring.js +++ b/src/spring.js @@ -206,6 +206,7 @@ $.Spring.prototype = { /** * @function + * @returns true if the value got updated, false otherwise */ update: function() { this.current.time = $.now(); @@ -226,14 +227,17 @@ $.Spring.prototype = { transform( this.springStiffness, ( this.current.time - this.start.time ) / - ( this.target.time - this.start.time ) + ( this.target.time - this.start.time ) ); + var oldValue = this.current.value; if (this._exponential) { this.current.value = Math.exp(currentValue); } else { this.current.value = currentValue; } + + return oldValue != this.current.value; }, /** diff --git a/src/strings.js b/src/strings.js index 03f4ed42..cd3c7907 100644 --- a/src/strings.js +++ b/src/strings.js @@ -75,14 +75,14 @@ $.extend( $, /** @lends OpenSeadragon */{ container = I18N, i; - for ( i = 0; i < props.length-1; i++ ) { + for (i = 0; i < props.length - 1; i++) { // in case not a subproperty container = container[ props[ i ] ] || {}; } string = container[ props[ i ] ]; if ( typeof( string ) != "string" ) { - $.console.debug( "Untranslated source string:", prop ); + $.console.log( "Untranslated source string:", prop ); string = ""; // FIXME: this breaks gettext()-style convention, which would return source } diff --git a/src/tile.js b/src/tile.js index 72776aac..61ba2e64 100644 --- a/src/tile.js +++ b/src/tile.js @@ -47,8 +47,10 @@ * @param {String} url The URL of this tile's image. * @param {CanvasRenderingContext2D} context2D The context2D of this tile if it * is provided directly by the tile source. + * @param {Boolean} loadWithAjax Whether this tile image should be loaded with an AJAX request . + * @param {Object} ajaxHeaders The headers to send with this tile's AJAX request (if applicable). */ -$.Tile = function(level, x, y, bounds, exists, url, context2D) { +$.Tile = function(level, x, y, bounds, exists, url, context2D, loadWithAjax, ajaxHeaders) { /** * The zoom level this tile belongs to. * @member {Number} level @@ -91,6 +93,29 @@ $.Tile = function(level, x, y, bounds, exists, url, context2D) { * @memberOf OpenSeadragon.Tile# */ this.context2D = context2D; + /** + * Whether to load this tile's image with an AJAX request. + * @member {Boolean} loadWithAjax + * @memberof OpenSeadragon.Tile# + */ + this.loadWithAjax = loadWithAjax; + /** + * The headers to be used in requesting this tile's image. + * Only used if loadWithAjax is set to true. + * @member {Object} ajaxHeaders + * @memberof OpenSeadragon.Tile# + */ + this.ajaxHeaders = ajaxHeaders; + /** + * The unique cache key for this tile. + * @member {String} cacheKey + * @memberof OpenSeadragon.Tile# + */ + if (this.ajaxHeaders) { + this.cacheKey = this.url + "+" + JSON.stringify(this.ajaxHeaders); + } else { + this.cacheKey = this.url; + } /** * Is this tile loaded? * @member {Boolean} loaded diff --git a/src/tilecache.js b/src/tilecache.js index ee3a4662..05d4e9cd 100644 --- a/src/tilecache.js +++ b/src/tilecache.js @@ -140,6 +140,7 @@ $.TileCache.prototype = { * may temporarily surpass that number, but should eventually come back down to the max specified. * @param {Object} options - Tile info. * @param {OpenSeadragon.Tile} options.tile - The tile to cache. + * @param {String} options.tile.cacheKey - The unique key used to identify this tile in the cache. * @param {Image} options.image - The image of the tile to cache. * @param {OpenSeadragon.TiledImage} options.tiledImage - The TiledImage that owns that tile. * @param {Number} [options.cutoff=0] - If adding this tile goes over the cache max count, this @@ -149,16 +150,16 @@ $.TileCache.prototype = { cacheTile: function( options ) { $.console.assert( options, "[TileCache.cacheTile] options is required" ); $.console.assert( options.tile, "[TileCache.cacheTile] options.tile is required" ); - $.console.assert( options.tile.url, "[TileCache.cacheTile] options.tile.url is required" ); + $.console.assert( options.tile.cacheKey, "[TileCache.cacheTile] options.tile.cacheKey is required" ); $.console.assert( options.tiledImage, "[TileCache.cacheTile] options.tiledImage is required" ); var cutoff = options.cutoff || 0; var insertionIndex = this._tilesLoaded.length; - var imageRecord = this._imagesLoaded[options.tile.url]; + var imageRecord = this._imagesLoaded[options.tile.cacheKey]; if (!imageRecord) { $.console.assert( options.image, "[TileCache.cacheTile] options.image is required to create an ImageRecord" ); - imageRecord = this._imagesLoaded[options.tile.url] = new ImageRecord({ + imageRecord = this._imagesLoaded[options.tile.cacheKey] = new ImageRecord({ image: options.image }); @@ -232,9 +233,9 @@ $.TileCache.prototype = { }, // private - getImageRecord: function(url) { - $.console.assert(url, '[TileCache.getImageRecord] url is required'); - return this._imagesLoaded[url]; + getImageRecord: function(cacheKey) { + $.console.assert(cacheKey, '[TileCache.getImageRecord] cacheKey is required'); + return this._imagesLoaded[cacheKey]; }, // private @@ -246,11 +247,11 @@ $.TileCache.prototype = { tile.unload(); tile.cacheImageRecord = null; - var imageRecord = this._imagesLoaded[tile.url]; + var imageRecord = this._imagesLoaded[tile.cacheKey]; imageRecord.removeTile(tile); if (!imageRecord.getTileCount()) { imageRecord.destroy(); - delete this._imagesLoaded[tile.url]; + delete this._imagesLoaded[tile.cacheKey]; this._imagesLoadedCount--; } diff --git a/src/tiledimage.js b/src/tiledimage.js index bfedc96d..82f722bc 100644 --- a/src/tiledimage.js +++ b/src/tiledimage.js @@ -76,6 +76,12 @@ * @param {Boolean} [options.debugMode] - See {@link OpenSeadragon.Options}. * @param {String|CanvasGradient|CanvasPattern|Function} [options.placeholderFillStyle] - See {@link OpenSeadragon.Options}. * @param {String|Boolean} [options.crossOriginPolicy] - See {@link OpenSeadragon.Options}. + * @param {Boolean} [options.ajaxWithCredentials] - See {@link OpenSeadragon.Options}. + * @param {Boolean} [options.loadTilesWithAjax] + * Whether to load tile data using AJAX requests. + * Defaults to the setting in {@link OpenSeadragon.Options}. + * @param {Object} [options.ajaxHeaders={}] + * A set of headers to include when making tile AJAX requests. */ $.TiledImage = function( options ) { var _this = this; @@ -133,7 +139,7 @@ $.TiledImage = function( options ) { var fitBoundsPlacement = options.fitBoundsPlacement || OpenSeadragon.Placement.CENTER; delete options.fitBoundsPlacement; - this._degrees = $.positiveModulo(options.degrees || 0, 360); + var degrees = options.degrees || 0; delete options.degrees; $.extend( true, this, { @@ -147,6 +153,7 @@ $.TiledImage = function( options ) { _midDraw: false, // Is the tiledImage currently updating the viewport? _needsDraw: 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. //configurable settings springStiffness: $.DEFAULT_SETTINGS.springStiffness, animationTime: $.DEFAULT_SETTINGS.animationTime, @@ -161,6 +168,7 @@ $.TiledImage = function( options ) { iOSDevice: $.DEFAULT_SETTINGS.iOSDevice, debugMode: $.DEFAULT_SETTINGS.debugMode, crossOriginPolicy: $.DEFAULT_SETTINGS.crossOriginPolicy, + ajaxWithCredentials: $.DEFAULT_SETTINGS.ajaxWithCredentials, placeholderFillStyle: $.DEFAULT_SETTINGS.placeholderFillStyle, opacity: $.DEFAULT_SETTINGS.opacity, preload: $.DEFAULT_SETTINGS.preload, @@ -190,6 +198,12 @@ $.TiledImage = function( options ) { animationTime: this.animationTime }); + this._degreesSpring = new $.Spring({ + initial: degrees, + springStiffness: this.springStiffness, + animationTime: this.animationTime + }); + this._updateForScale(); if (fitBounds) { @@ -273,16 +287,12 @@ $.extend($.TiledImage.prototype, $.EventSource.prototype, /** @lends OpenSeadrag * @returns {Boolean} Whether the TiledImage animated. */ update: function() { - var oldX = this._xSpring.current.value; - var oldY = this._ySpring.current.value; - var oldScale = this._scaleSpring.current.value; + var xUpdated = this._xSpring.update(); + var yUpdated = this._ySpring.update(); + var scaleUpdated = this._scaleSpring.update(); + var degreesUpdated = this._degreesSpring.update(); - this._xSpring.update(); - this._ySpring.update(); - this._scaleSpring.update(); - - if (this._xSpring.current.value !== oldX || this._ySpring.current.value !== oldY || - this._scaleSpring.current.value !== oldScale) { + if (xUpdated || yUpdated || scaleUpdated || degreesUpdated) { this._updateForScale(); this._needsDraw = true; return true; @@ -317,7 +327,7 @@ $.extend($.TiledImage.prototype, $.EventSource.prototype, /** @lends OpenSeadrag */ getBounds: function(current) { return this.getBoundsNoRotate(current) - .rotate(this._degrees, this._getRotationPoint(current)); + .rotate(this.getRotation(current), this._getRotationPoint(current)); }, /** @@ -366,7 +376,7 @@ $.extend($.TiledImage.prototype, $.EventSource.prototype, /** @lends OpenSeadrag clip.width, clip.height); } - return bounds.rotate(this._degrees, this._getRotationPoint(current)); + return bounds.rotate(this.getRotation(current), this._getRotationPoint(current)); }, /** @@ -401,7 +411,7 @@ $.extend($.TiledImage.prototype, $.EventSource.prototype, /** @lends OpenSeadrag point = new $.Point(viewerX, viewerY); } - point = point.rotate(-this._degrees, this._getRotationPoint(current)); + point = point.rotate(-this.getRotation(current), this._getRotationPoint(current)); return current ? this._viewportToImageDelta( point.x - this._xSpring.current.value, @@ -443,7 +453,7 @@ $.extend($.TiledImage.prototype, $.EventSource.prototype, /** @lends OpenSeadrag point.y += this._ySpring.target.value; } - return point.rotate(this._degrees, this._getRotationPoint(current)); + return point.rotate(this.getRotation(current), this._getRotationPoint(current)); }, /** @@ -457,7 +467,7 @@ $.extend($.TiledImage.prototype, $.EventSource.prototype, /** @lends OpenSeadrag * @param {Boolean} [current=false] - Pass true to use the current location; false for target location. * @return {OpenSeadragon.Rect} A rect representing the coordinates in the viewport. */ - imageToViewportRectangle: function( imageX, imageY, pixelWidth, pixelHeight, current ) { + imageToViewportRectangle: function(imageX, imageY, pixelWidth, pixelHeight, current) { var rect = imageX; if (rect instanceof $.Rect) { //they passed a rect instead of individual components @@ -474,7 +484,7 @@ $.extend($.TiledImage.prototype, $.EventSource.prototype, /** @lends OpenSeadrag coordA.y, coordB.x, coordB.y, - rect.degrees + this._degrees + rect.degrees + this.getRotation(current) ); }, @@ -506,7 +516,7 @@ $.extend($.TiledImage.prototype, $.EventSource.prototype, /** @lends OpenSeadrag coordA.y, coordB.x, coordB.y, - rect.degrees - this._degrees + rect.degrees - this.getRotation(current) ); }, @@ -559,7 +569,7 @@ $.extend($.TiledImage.prototype, $.EventSource.prototype, /** @lends OpenSeadrag // coordinates (x in [0, 1] and y in [0, aspectRatio]) _viewportToTiledImageRectangle: function(rect) { var scale = this._scaleSpring.current.value; - rect = rect.rotate(-this.getRotation(), this._getRotationPoint(true)); + rect = rect.rotate(-this.getRotation(true), this._getRotationPoint(true)); return new $.Rect( (rect.x - this._xSpring.current.value) / scale, (rect.y - this._ySpring.current.value) / scale, @@ -581,7 +591,7 @@ $.extend($.TiledImage.prototype, $.EventSource.prototype, /** @lends OpenSeadrag viewportToImageZoom: function( viewportZoom ) { var ratio = this._scaleSpring.current.value * this.viewport._containerInnerSize.x / this.source.dimensions.x; - return ratio * viewportZoom ; + return ratio * viewportZoom; }, /** @@ -788,24 +798,33 @@ $.extend($.TiledImage.prototype, $.EventSource.prototype, /** @lends OpenSeadrag }, /** - * Get the current rotation of this tiled image in degrees. - * @returns {Number} the current rotation of this tiled image in degrees. + * Get the rotation of this tiled image in degrees. + * @param {Boolean} [current=false] True for current rotation, false for target. + * @returns {Number} the rotation of this tiled image in degrees. */ - getRotation: function() { - return this._degrees; + getRotation: function(current) { + return current ? + this._degreesSpring.current.value : + this._degreesSpring.target.value; }, /** * Set the current rotation of this tiled image in degrees. * @param {Number} degrees the rotation in degrees. + * @param {Boolean} [immediately=false] Whether to animate to the new angle + * or rotate immediately. * @fires OpenSeadragon.TiledImage.event:bounds-change */ - setRotation: function(degrees) { - degrees = $.positiveModulo(degrees, 360); - if (this._degrees === degrees) { + setRotation: function(degrees, immediately) { + if (this._degreesSpring.target.value === degrees && + this._degreesSpring.isAtTargetValue()) { return; } - this._degrees = degrees; + if (immediately) { + this._degreesSpring.resetTo(degrees); + } else { + this._degreesSpring.springTo(degrees); + } this._needsDraw = true; this._raiseBoundsChange(); }, @@ -915,9 +934,16 @@ $.extend($.TiledImage.prototype, $.EventSource.prototype, /** @lends OpenSeadrag }; }, - // private + /** + * @private + * @inner + * Pretty much every other line in this needs to be documented so it's clear + * how each piece of this routine contributes to the drawing process. That's + * why there are so many TODO's inside this function. + */ _updateViewport: function() { this._needsDraw = false; + this._tilesLoading = 0; // Reset tile's internal drawn state while (this.lastDrawn.length > 0) { @@ -972,7 +998,7 @@ $.extend($.TiledImage.prototype, $.EventSource.prototype, /** @lends OpenSeadrag var targetZeroRatio = viewport.deltaPixelsFromPointsNoRotate( this.source.getPixelRatio( Math.max( - this.source.getClosestLevel(viewport.containerSize) - 1, + this.source.getClosestLevel(), 0 ) ), @@ -1014,11 +1040,67 @@ $.extend($.TiledImage.prototype, $.EventSource.prototype, /** @lends OpenSeadrag this._needsDraw = true; this._setFullyLoaded(false); } else { - this._setFullyLoaded(true); + this._setFullyLoaded(this._tilesLoading === 0); } + }, + + // private + _getCornerTiles: function(level, topLeftBound, bottomRightBound) { + var leftX; + var rightX; + if (this.wrapHorizontal) { + leftX = $.positiveModulo(topLeftBound.x, 1); + rightX = $.positiveModulo(bottomRightBound.x, 1); + } else { + leftX = Math.max(0, topLeftBound.x); + rightX = Math.min(1, bottomRightBound.x); + } + var topY; + var bottomY; + var aspectRatio = 1 / this.source.aspectRatio; + if (this.wrapVertical) { + topY = $.positiveModulo(topLeftBound.y, aspectRatio); + bottomY = $.positiveModulo(bottomRightBound.y, aspectRatio); + } else { + topY = Math.max(0, topLeftBound.y); + bottomY = Math.min(aspectRatio, bottomRightBound.y); + } + + var topLeftTile = this.source.getTileAtPoint(level, new $.Point(leftX, topY)); + var bottomRightTile = this.source.getTileAtPoint(level, new $.Point(rightX, bottomY)); + var numTiles = this.source.getNumTiles(level); + + if (this.wrapHorizontal) { + topLeftTile.x += numTiles.x * Math.floor(topLeftBound.x); + bottomRightTile.x += numTiles.x * Math.floor(bottomRightBound.x); + } + if (this.wrapVertical) { + topLeftTile.y += numTiles.y * Math.floor(topLeftBound.y / aspectRatio); + bottomRightTile.y += numTiles.y * Math.floor(bottomRightBound.y / aspectRatio); + } + + return { + topLeft: topLeftTile, + bottomRight: bottomRightTile, + }; } }); +/** + * @private + * @inner + * Updates all tiles at a given resolution level. + * @param {OpenSeadragon.TiledImage} tiledImage - Which TiledImage is being drawn. + * @param {Boolean} haveDrawn + * @param {Boolean} drawLevel + * @param {Number} level + * @param {Number} levelOpacity + * @param {Number} levelVisibility + * @param {OpenSeadragon.Point} viewportTL - The index of the most top-left visible tile. + * @param {OpenSeadragon.Point} viewportBR - The index of the most bottom-right visible tile. + * @param {Number} currentTime + * @param {OpenSeadragon.Tile} best - The current "best" tile to draw. + */ function updateLevel(tiledImage, haveDrawn, drawLevel, level, levelOpacity, levelVisibility, drawArea, currentTime, best) { @@ -1059,23 +1141,13 @@ function updateLevel(tiledImage, haveDrawn, drawLevel, level, levelOpacity, }); } - //OK, a new drawing so do your calculations - var topLeftTile = tiledImage.source.getTileAtPoint(level, topLeftBound); - var bottomRightTile = tiledImage.source.getTileAtPoint(level, bottomRightBound); - var numberOfTiles = tiledImage.source.getNumTiles(level); - resetCoverage(tiledImage.coverage, level); - if (!tiledImage.wrapHorizontal) { - // Adjust for floating point error - topLeftTile.x = Math.max(topLeftTile.x, 0); - bottomRightTile.x = Math.min(bottomRightTile.x, numberOfTiles.x - 1); - } - if (!tiledImage.wrapVertical) { - // Adjust for floating point error - topLeftTile.y = Math.max(topLeftTile.y, 0); - bottomRightTile.y = Math.min(bottomRightTile.y, numberOfTiles.y - 1); - } + //OK, a new drawing so do your calculations + var cornerTiles = tiledImage._getCornerTiles(level, topLeftBound, bottomRightBound); + var topLeftTile = cornerTiles.topLeft; + var bottomRightTile = cornerTiles.bottomRight; + var numberOfTiles = tiledImage.source.getNumTiles(level); var viewportCenter = tiledImage.viewport.pixelFromPoint( tiledImage.viewport.getCenter()); @@ -1112,11 +1184,29 @@ function updateLevel(tiledImage, haveDrawn, drawLevel, level, levelOpacity, return best; } -function updateTile( tiledImage, drawLevel, haveDrawn, x, y, level, levelOpacity, levelVisibility, viewportCenter, numberOfTiles, currentTime, best){ +/** + * @private + * @inner + * Update a single tile at a particular resolution level. + * @param {OpenSeadragon.TiledImage} tiledImage - Which TiledImage is being drawn. + * @param {Boolean} haveDrawn + * @param {Boolean} drawLevel + * @param {Number} x + * @param {Number} y + * @param {Number} level + * @param {Number} levelOpacity + * @param {Number} levelVisibility + * @param {OpenSeadragon.Point} viewportCenter + * @param {Number} numberOfTiles + * @param {Number} currentTime + * @param {OpenSeadragon.Tile} best - The current "best" tile to draw. + */ +function updateTile( tiledImage, haveDrawn, drawLevel, x, y, level, levelOpacity, levelVisibility, viewportCenter, numberOfTiles, currentTime, best){ var tile = getTile( x, y, level, + tiledImage, tiledImage.source, tiledImage.tilesMatrix, currentTime, @@ -1175,7 +1265,7 @@ function updateTile( tiledImage, drawLevel, haveDrawn, x, y, level, levelOpacity if (tile.context2D) { setTileLoaded(tiledImage, tile); } else { - var imageRecord = tiledImage._tileCache.getImageRecord(tile.url); + var imageRecord = tiledImage._tileCache.getImageRecord(tile.cacheKey); if (imageRecord) { var image = imageRecord.getImage(); setTileLoaded(tiledImage, tile, image); @@ -1198,7 +1288,7 @@ function updateTile( tiledImage, drawLevel, haveDrawn, x, y, level, levelOpacity } } else if ( tile.loading ) { // the tile is already in the download queue - // thanks josh1093 for finally translating this typo + tiledImage._tilesLoading++; } else { best = compareTiles( best, tile ); } @@ -1206,12 +1296,39 @@ function updateTile( tiledImage, drawLevel, haveDrawn, x, y, level, levelOpacity return best; } -function getTile( x, y, level, tileSource, tilesMatrix, time, numTiles, worldWidth, worldHeight ) { +/** + * @private + * @inner + * Obtains a tile at the given location. + * @param {Number} x + * @param {Number} y + * @param {Number} level + * @param {OpenSeadragon.TiledImage} tiledImage + * @param {OpenSeadragon.TileSource} tileSource + * @param {Object} tilesMatrix - A '3d' dictionary [level][x][y] --> Tile. + * @param {Number} time + * @param {Number} numTiles + * @param {Number} worldWidth + * @param {Number} worldHeight + * @returns {OpenSeadragon.Tile} + */ +function getTile( + x, y, + level, + tiledImage, + tileSource, + tilesMatrix, + time, + numTiles, + worldWidth, + worldHeight +) { var xMod, yMod, bounds, exists, url, + ajaxHeaders, context2D, tile; @@ -1228,6 +1345,18 @@ function getTile( x, y, level, tileSource, tilesMatrix, time, numTiles, worldWid bounds = tileSource.getTileBounds( level, xMod, yMod ); exists = tileSource.tileExists( level, xMod, yMod ); url = tileSource.getTileUrl( level, xMod, yMod ); + + // Headers are only applicable if loadTilesWithAjax is set + if (tiledImage.loadTilesWithAjax) { + ajaxHeaders = tileSource.getTileAjaxHeaders( level, xMod, yMod ); + // Combine tile AJAX headers with tiled image AJAX headers (if applicable) + if ($.isPlainObject(tiledImage.ajaxHeaders)) { + ajaxHeaders = $.extend({}, tiledImage.ajaxHeaders, ajaxHeaders); + } + } else { + ajaxHeaders = null; + } + context2D = tileSource.getContext2D ? tileSource.getContext2D(level, xMod, yMod) : undefined; @@ -1241,7 +1370,9 @@ function getTile( x, y, level, tileSource, tilesMatrix, time, numTiles, worldWid bounds, exists, url, - context2D + context2D, + tiledImage.loadTilesWithAjax, + ajaxHeaders ); } @@ -1251,13 +1382,24 @@ function getTile( x, y, level, tileSource, tilesMatrix, time, numTiles, worldWid return tile; } +/** + * @private + * @inner + * Dispatch a job to the ImageLoader to load the Image for a Tile. + * @param {OpenSeadragon.TiledImage} tiledImage + * @param {OpenSeadragon.Tile} tile + * @param {Number} time + */ function loadTile( tiledImage, tile, time ) { tile.loading = true; tiledImage._imageLoader.addJob({ src: tile.url, + loadWithAjax: tile.loadWithAjax, + ajaxHeaders: tile.ajaxHeaders, crossOriginPolicy: tiledImage.crossOriginPolicy, - callback: function( image, errorMsg ){ - onTileLoad( tiledImage, tile, time, image, errorMsg ); + ajaxWithCredentials: tiledImage.ajaxWithCredentials, + callback: function( image, errorMsg, tileRequest ){ + onTileLoad( tiledImage, tile, time, image, errorMsg, tileRequest ); }, abort: function() { tile.loading = false; @@ -1265,7 +1407,18 @@ function loadTile( tiledImage, tile, time ) { }); } -function onTileLoad( tiledImage, tile, time, image, errorMsg ) { +/** + * @private + * @inner + * Callback fired when a Tile's Image finished downloading. + * @param {OpenSeadragon.TiledImage} tiledImage + * @param {OpenSeadragon.Tile} tile + * @param {Number} time + * @param {Image} image + * @param {String} errorMsg + * @param {XMLHttpRequest} tileRequest + */ +function onTileLoad( tiledImage, tile, time, image, errorMsg, tileRequest ) { if ( !image ) { $.console.log( "Tile %s failed to load: %s - error: %s", tile, tile.url, errorMsg ); /** @@ -1278,8 +1431,15 @@ function onTileLoad( tiledImage, tile, time, image, errorMsg ) { * @property {OpenSeadragon.TiledImage} tiledImage - The tiled image the tile belongs to. * @property {number} time - The time in milliseconds when the tile load began. * @property {string} message - The error message. + * @property {XMLHttpRequest} tileRequest - The XMLHttpRequest used to load the tile if available. */ - tiledImage.viewer.raiseEvent("tile-load-failed", {tile: tile, tiledImage: tiledImage, time: time, message: errorMsg}); + tiledImage.viewer.raiseEvent("tile-load-failed", { + tile: tile, + tiledImage: tiledImage, + time: time, + message: errorMsg, + tileRequest: tileRequest + }); tile.loading = false; tile.exists = false; return; @@ -1294,7 +1454,7 @@ function onTileLoad( tiledImage, tile, time, image, errorMsg ) { var finish = function() { var cutoff = Math.ceil( Math.log( tiledImage.source.getTileWidth(tile.level) ) / Math.log( 2 ) ); - setTileLoaded(tiledImage, tile, image, cutoff); + setTileLoaded(tiledImage, tile, image, cutoff, tileRequest); }; // Check if we're mid-update; this can happen on IE8 because image load events for @@ -1307,7 +1467,15 @@ function onTileLoad( tiledImage, tile, time, image, errorMsg ) { } } -function setTileLoaded(tiledImage, tile, image, cutoff) { +/** + * @private + * @inner + * @param {OpenSeadragon.TiledImage} tiledImage + * @param {OpenSeadragon.Tile} tile + * @param {Image} image + * @param {Number} cutoff + */ +function setTileLoaded(tiledImage, tile, image, cutoff, tileRequest) { var increment = 0; function getCompletionCallback() { @@ -1342,6 +1510,7 @@ function setTileLoaded(tiledImage, tile, image, cutoff) { * @property {Image} image - The image of the tile. * @property {OpenSeadragon.TiledImage} tiledImage - The tiled image of the loaded tile. * @property {OpenSeadragon.Tile} tile - The tile which has been loaded. + * @property {XMLHttpRequest} tiledImage - The AJAX request that loaded this tile (if applicable). * @property {function} getCompletionCallback - A function giving a callback to call * when the asynchronous processing of the image is done. The image will be * marked as entirely loaded when the callback has been called once for each @@ -1350,6 +1519,7 @@ function setTileLoaded(tiledImage, tile, image, cutoff) { tiledImage.viewer.raiseEvent("tile-loaded", { tile: tile, tiledImage: tiledImage, + tileRequest: tileRequest, image: image, getCompletionCallback: getCompletionCallback }); @@ -1357,6 +1527,16 @@ function setTileLoaded(tiledImage, tile, image, cutoff) { getCompletionCallback()(); } +/** + * @private + * @inner + * @param {OpenSeadragon.Tile} tile + * @param {Boolean} overlap + * @param {OpenSeadragon.Viewport} viewport + * @param {OpenSeadragon.Point} viewportCenter + * @param {Number} levelVisibility + * @param {OpenSeadragon.TiledImage} tiledImage + */ function positionTile( tile, overlap, viewport, viewportCenter, levelVisibility, tiledImage ){ var boundsTL = tile.bounds.getTopLeft(); @@ -1387,7 +1567,23 @@ function positionTile( tile, overlap, viewport, viewportCenter, levelVisibility, tile.visibility = levelVisibility; } - +/** + * @private + * @inner + * Updates the opacity of a tile according to the time it has been on screen + * to perform a fade-in. + * Updates coverage once a tile is fully opaque. + * Returns whether the fade-in has completed. + * + * @param {OpenSeadragon.TiledImage} tiledImage + * @param {OpenSeadragon.Tile} tile + * @param {Number} x + * @param {Number} y + * @param {Number} level + * @param {Number} levelOpacity + * @param {Number} currentTime + * @returns {Boolean} + */ function blendTile( tiledImage, tile, x, y, level, levelOpacity, currentTime ){ var blendTimeMillis = 1000 * tiledImage.blendTime, deltaTime, @@ -1428,6 +1624,12 @@ function blendTile( tiledImage, tile, x, y, level, levelOpacity, currentTime ){ * Note that out-of-bounds tiles provide coverage in this sense, since * there's no content that they would need to cover. Tiles at non-existent * levels that are within the image bounds, however, do not. + * + * @param {Object} coverage - A '3d' dictionary [level][x][y] --> Boolean. + * @param {Number} level - The resolution level of the tile. + * @param {Number} x - The X position of the tile. + * @param {Number} y - The Y position of the tile. + * @returns {Boolean} */ function providesCoverage( coverage, level, x, y ) { var rows, @@ -1467,6 +1669,12 @@ function providesCoverage( coverage, level, x, y ) { * Returns true if the given tile is completely covered by higher-level * tiles of higher resolution representing the same content. If neither x * nor y is given, returns true if the entire visible level is covered. + * + * @param {Object} coverage - A '3d' dictionary [level][x][y] --> Boolean. + * @param {Number} level - The resolution level of the tile. + * @param {Number} x - The X position of the tile. + * @param {Number} y - The Y position of the tile. + * @returns {Boolean} */ function isCovered( coverage, level, x, y ) { if ( x === undefined || y === undefined ) { @@ -1485,6 +1693,12 @@ function isCovered( coverage, level, x, y ) { * @private * @inner * Sets whether the given tile provides coverage or not. + * + * @param {Object} coverage - A '3d' dictionary [level][x][y] --> Boolean. + * @param {Number} level - The resolution level of the tile. + * @param {Number} x - The X position of the tile. + * @param {Number} y - The Y position of the tile. + * @param {Boolean} covers - Whether the tile provides coverage. */ function setCoverage( coverage, level, x, y, covers ) { if ( !coverage[ level ] ) { @@ -1508,6 +1722,9 @@ function setCoverage( coverage, level, x, y, covers ) { * Resets coverage information for the given level. This should be called * after every draw routine. Note that at the beginning of the next draw * routine, coverage for every visible tile should be explicitly set. + * + * @param {Object} coverage - A '3d' dictionary [level][x][y] --> Boolean. + * @param {Number} level - The resolution level of tiles to completely reset. */ function resetCoverage( coverage, level ) { coverage[ level ] = {}; @@ -1518,6 +1735,10 @@ function resetCoverage( coverage, level ) { * @inner * Determines whether the 'last best' tile for the area is better than the * tile in question. + * + * @param {OpenSeadragon.Tile} previousBest + * @param {OpenSeadragon.Tile} tile + * @returns {OpenSeadragon.Tile} The new best tile. */ function compareTiles( previousBest, tile ) { if ( !previousBest ) { @@ -1535,6 +1756,13 @@ function compareTiles( previousBest, tile ) { return previousBest; } +/** + * @private + * @inner + * Draws a TiledImage. + * @param {OpenSeadragon.TiledImage} tiledImage + * @param {OpenSeadragon.Tile[]} lastDrawn - An unordered list of Tiles drawn last frame. + */ function drawTiles( tiledImage, lastDrawn ) { if (tiledImage.opacity === 0 || lastDrawn.length === 0) { return; @@ -1551,9 +1779,12 @@ function drawTiles( tiledImage, lastDrawn ) { var zoom = tiledImage.viewport.getZoom(true); var imageZoom = tiledImage.viewportToImageZoom(zoom); - // TODO: support tile edge smoothing with tiled image rotation. - if (imageZoom > tiledImage.smoothTileEdgesMinZoom && !tiledImage.iOSDevice && - tiledImage.getRotation() === 0 && $.supportsCanvas) { + + if (lastDrawn.length > 1 && + imageZoom > tiledImage.smoothTileEdgesMinZoom && + !tiledImage.iOSDevice && + tiledImage.getRotation(true) % 360 === 0 && // TODO: support tile edge smoothing with tiled image rotation. + $.supportsCanvas) { // When zoomed in a lot (>100%) the tile edges are visible. // So we have to composite them at ~100% and scale them up together. // Note: Disabled on iOS devices per default as it causes a native crash @@ -1586,9 +1817,9 @@ function drawTiles( tiledImage, lastDrawn ) { useSketch: useSketch }); } - if (tiledImage._degrees !== 0) { + if (tiledImage.getRotation(true) % 360 !== 0) { tiledImage._drawer._offsetForRotation({ - degrees: tiledImage._degrees, + degrees: tiledImage.getRotation(true), point: tiledImage.viewport.pixelFromPointNoRotate( tiledImage._getRotationPoint(true), true), useSketch: useSketch @@ -1601,7 +1832,7 @@ function drawTiles( tiledImage, lastDrawn ) { tiledImage._drawer.saveContext(useSketch); var box = tiledImage.imageToViewportRectangle(tiledImage._clip, true); - box = box.rotate(-tiledImage._degrees, tiledImage._getRotationPoint()); + box = box.rotate(-tiledImage.getRotation(true), tiledImage._getRotationPoint(true)); var clipRect = tiledImage._drawer.viewportToDrawerRectangle(box); if (sketchScale) { clipRect = clipRect.times(sketchScale); @@ -1663,7 +1894,7 @@ function drawTiles( tiledImage, lastDrawn ) { } if (!sketchScale) { - if (tiledImage._degrees !== 0) { + if (tiledImage.getRotation(true) % 360 !== 0) { tiledImage._drawer._restoreRotationChanges(useSketch); } if (tiledImage.viewport.degrees !== 0) { @@ -1679,9 +1910,9 @@ function drawTiles( tiledImage, lastDrawn ) { useSketch: false }); } - if (tiledImage._degrees !== 0) { + if (tiledImage.getRotation(true) % 360 !== 0) { tiledImage._drawer._offsetForRotation({ - degrees: tiledImage._degrees, + degrees: tiledImage.getRotation(true), point: tiledImage.viewport.pixelFromPointNoRotate( tiledImage._getRotationPoint(true), true), useSketch: false @@ -1696,7 +1927,7 @@ function drawTiles( tiledImage, lastDrawn ) { bounds: bounds }); if (sketchScale) { - if (tiledImage._degrees !== 0) { + if (tiledImage.getRotation(true) % 360 !== 0) { tiledImage._drawer._restoreRotationChanges(false); } if (tiledImage.viewport.degrees !== 0) { @@ -1707,6 +1938,13 @@ function drawTiles( tiledImage, lastDrawn ) { drawDebugInfo( tiledImage, lastDrawn ); } +/** + * @private + * @inner + * Draws special debug information for a TiledImage if in debug mode. + * @param {OpenSeadragon.TiledImage} tiledImage + * @param {OpenSeadragon.Tile[]} lastDrawn - An unordered list of Tiles drawn last frame. + */ function drawDebugInfo( tiledImage, lastDrawn ) { if( tiledImage.debugMode ) { for ( var i = lastDrawn.length - 1; i >= 0; i-- ) { diff --git a/src/tilesource.js b/src/tilesource.js index fa52f61e..e83c4fab 100644 --- a/src/tilesource.js +++ b/src/tilesource.js @@ -65,6 +65,8 @@ * @param {Boolean} [options.ajaxWithCredentials] * If this TileSource needs to make an AJAX call, this specifies whether to set * the XHR's withCredentials (for accessing secure data). + * @param {Object} [options.ajaxHeaders] + * A set of headers to include in AJAX requests. * @param {Number} [options.width] * Width of the source image at max resolution in pixels. * @param {Number} [options.height] @@ -187,8 +189,8 @@ $.TileSource = function( width, height, tileSize, tileOverlap, minLevel, maxLeve //explicit configuration via positional args in constructor //or the more idiomatic 'options' object this.ready = true; - this.aspectRatio = ( options.width && options.height ) ? - ( options.width / options.height ) : 1; + this.aspectRatio = (options.width && options.height) ? + (options.width / options.height) : 1; this.dimensions = new $.Point( options.width, options.height ); if ( this.tileSize ){ @@ -318,25 +320,20 @@ $.TileSource.prototype = { /** * @function - * @param {Number} level + * @returns {Number} The highest level in this tile source that can be contained in a single tile. */ - getClosestLevel: function( rect ) { + getClosestLevel: function() { var i, - tilesPerSide, tiles; - for( i = this.minLevel; i < this.maxLevel; i++ ){ - tiles = this.getNumTiles( i ); - tilesPerSide = new $.Point( - Math.floor( rect.x / this.getTileWidth(i) ), - Math.floor( rect.y / this.getTileHeight(i) ) - ); - - if( tiles.x + 1 >= tilesPerSide.x && tiles.y + 1 >= tilesPerSide.y ){ + for (i = this.minLevel + 1; i <= this.maxLevel; i++){ + tiles = this.getNumTiles(i); + if (tiles.x > 1 || tiles.y > 1) { break; } } - return Math.max( 0, i - 1 ); + + return i - 1; }, /** @@ -345,17 +342,25 @@ $.TileSource.prototype = { * @param {OpenSeadragon.Point} point */ getTileAtPoint: function(level, point) { + var validPoint = point.x >= 0 && point.x <= 1 && + point.y >= 0 && point.y <= 1 / this.aspectRatio; + $.console.assert(validPoint, "[TileSource.getTileAtPoint] must be called with a valid point."); + var widthScaled = this.dimensions.x * this.getLevelScale(level); - var pixelX = $.positiveModulo(point.x, 1) * widthScaled; - var pixelY = $.positiveModulo(point.y, 1 / this.aspectRatio) * widthScaled; + var pixelX = point.x * widthScaled; + var pixelY = point.y * widthScaled; - var x = Math.floor(pixelX / this.getTileWidth()); - var y = Math.floor(pixelY / this.getTileHeight()); + var x = Math.floor(pixelX / this.getTileWidth(level)); + var y = Math.floor(pixelY / this.getTileHeight(level)); - // Fix for wrapping - var numTiles = this.getNumTiles(level); - x += numTiles.x * Math.floor(point.x); - y += numTiles.y * Math.floor(point.y * this.aspectRatio); + // When point.x == 1 or point.y == 1 / this.aspectRatio we want to + // return the last tile of the row/column + if (point.x >= 1) { + x = this.getNumTiles(level).x - 1; + } + if (point.y >= 1 / this.aspectRatio) { + y = this.getNumTiles(level).y - 1; + } return new $.Point(x, y); }, @@ -455,7 +460,7 @@ $.TileSource.prototype = { //TODO: Its not very flexible to require tile sources to end jsonp // request for info with a url that ends with '.js' but for // now it's the only way I see to distinguish uniformly. - callbackName = url.split( '/' ).pop().replace('.js',''); + callbackName = url.split('/').pop().replace('.js', ''); $.jsonp({ url: url, async: false, @@ -467,6 +472,7 @@ $.TileSource.prototype = { $.makeAjaxRequest( { url: url, withCredentials: this.ajaxWithCredentials, + headers: this.ajaxHeaders, success: function( xhr ) { var data = processResponse( xhr ); callback( data ); @@ -551,7 +557,7 @@ $.TileSource.prototype = { }, /** - * Responsible for retriving the url which will return an image for the + * Responsible for retrieving the url which will return an image for the * region specified by the given x, y, and level components. * This method is not implemented by this class other than to throw an Error * announcing you have to implement it. Because of the variety of tile @@ -567,6 +573,23 @@ $.TileSource.prototype = { throw new Error( "Method not implemented." ); }, + /** + * Responsible for retrieving the headers which will be attached to the image request for the + * region specified by the given x, y, and level components. + * This option is only relevant if {@link OpenSeadragon.Options}.loadTilesWithAjax is set to true. + * The headers returned here will override headers specified at the Viewer or TiledImage level. + * Specifying a falsy value for a header will clear its existing value set at the Viewer or + * TiledImage level (if any). + * @function + * @param {Number} level + * @param {Number} x + * @param {Number} y + * @returns {Object} + */ + getTileAjaxHeaders: function( level, x, y ) { + return {}; + }, + /** * @function * @param {Number} level @@ -575,12 +598,12 @@ $.TileSource.prototype = { */ tileExists: function( level, x, y ) { var numTiles = this.getNumTiles( level ); - return level >= this.minLevel && - level <= this.maxLevel && - x >= 0 && - y >= 0 && - x < numTiles.x && - y < numTiles.y; + return level >= this.minLevel && + level <= this.maxLevel && + x >= 0 && + y >= 0 && + x < numTiles.x && + y < numTiles.y; } }; @@ -621,7 +644,11 @@ function processResponse( xhr ){ data = xhr.responseText; } }else if( responseText.match(/\s*[\{\[].*/) ){ - data = $.parseJSON(responseText); + try{ + data = $.parseJSON(responseText); + } catch(e){ + data = responseText; + } }else{ data = responseText; } diff --git a/src/tilesourcecollection.js b/src/tilesourcecollection.js index a180aa52..e019741f 100644 --- a/src/tilesourcecollection.js +++ b/src/tilesourcecollection.js @@ -32,11 +32,11 @@ * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ -(function( $ ){ +(function($) { // deprecated -$.TileSourceCollection = function( tileSize, tileSources, rows, layout ) { +$.TileSourceCollection = function(tileSize, tileSources, rows, layout) { $.console.error('TileSourceCollection is deprecated; use World instead'); }; -}( OpenSeadragon )); +}(OpenSeadragon)); diff --git a/src/tmstilesource.js b/src/tmstilesource.js index 4c3d0ab0..e7640b15 100644 --- a/src/tmstilesource.js +++ b/src/tmstilesource.js @@ -83,7 +83,7 @@ $.TmsTileSource = function( width, height, tileSize, tileOverlap, tilesUrl ) { } else { max = bufferedHeight / 256; } - options.maxLevel = Math.ceil(Math.log(max)/Math.log(2)) - 1; + options.maxLevel = Math.ceil(Math.log(max) / Math.log(2)) - 1; options.tileSize = 256; options.width = bufferedWidth; options.height = bufferedHeight; @@ -129,7 +129,7 @@ $.extend( $.TmsTileSource.prototype, $.TileSource.prototype, /** @lends OpenSead // Convert from Deep Zoom definition to TMS zoom definition var yTiles = this.getNumTiles( level ).y - 1; - return this.tilesUrl + level + "/" + x + "/" + (yTiles - y) + ".png"; + return this.tilesUrl + level + "/" + x + "/" + (yTiles - y) + ".png"; } }); diff --git a/src/viewer.js b/src/viewer.js index 2c97d31a..0f7e92d9 100644 --- a/src/viewer.js +++ b/src/viewer.js @@ -96,6 +96,12 @@ $.Viewer = function( options ) { //internal state and dom identifiers id: options.id, hash: options.hash || nextHash++, + /** + * Index for page to be shown first next time open() is called (only used in sequenceMode). + * @member {Number} initialPage + * @memberof OpenSeadragon.Viewer# + */ + initialPage: 0, //dom nodes /** @@ -221,7 +227,7 @@ $.Viewer = function( options ) { $.ControlDock.call( this, options ); //Deal with tile sources - if ( this.xmlPath ){ + if (this.xmlPath) { //Deprecated option. Now it is preferred to use the tileSources option this.tileSources = [ this.xmlPath ]; } @@ -268,7 +274,7 @@ $.Viewer = function( options ) { this.innerTracker = new $.MouseTracker({ element: this.canvas, - startDisabled: this.mouseNavEnabled ? false : true, + startDisabled: !this.mouseNavEnabled, clickTimeThreshold: this.clickTimeThreshold, clickDistThreshold: this.clickDistThreshold, dblClickTimeThreshold: this.dblClickTimeThreshold, @@ -291,7 +297,7 @@ $.Viewer = function( options ) { this.outerTracker = new $.MouseTracker({ element: this.container, - startDisabled: this.mouseNavEnabled ? false : true, + startDisabled: !this.mouseNavEnabled, clickTimeThreshold: this.clickTimeThreshold, clickDistThreshold: this.clickDistThreshold, dblClickTimeThreshold: this.dblClickTimeThreshold, @@ -370,7 +376,8 @@ $.Viewer = function( options ) { // Create the image loader this.imageLoader = new $.ImageLoader({ - jobLimit: this.imageLoaderLimit + jobLimit: this.imageLoaderLimit, + timeout: options.timeout }); // Create the tile cache @@ -481,11 +488,13 @@ $.extend( $.Viewer.prototype, $.EventSource.prototype, $.ControlDock.prototype, * except for the index property; images are added in sequence. * A TileSource specifier is anything you could pass as the tileSource property * of the options parameter for {@link OpenSeadragon.Viewer#addTiledImage}. + * @param {Number} initialPage - If sequenceMode is true, display this page initially + * for the given tileSources. If specified, will overwrite the Viewer's existing initialPage property. * @return {OpenSeadragon.Viewer} Chainable. * @fires OpenSeadragon.Viewer.event:open * @fires OpenSeadragon.Viewer.event:open-failed */ - open: function (tileSources) { + open: function (tileSources, initialPage) { var _this = this; this.close(); @@ -500,6 +509,10 @@ $.extend( $.Viewer.prototype, $.EventSource.prototype, $.ControlDock.prototype, this.referenceStrip = null; } + if (typeof initialPage != 'undefined' && !isNaN(initialPage)) { + this.initialPage = initialPage; + } + this.tileSources = tileSources; this._sequenceIndex = Math.max(0, Math.min(this.tileSources.length - 1, this.initialPage)); if (this.tileSources.length) { @@ -674,7 +687,7 @@ $.extend( $.Viewer.prototype, $.EventSource.prototype, $.ControlDock.prototype, this.navigator.close(); } - if( ! this.preserveOverlays) { + if (!this.preserveOverlays) { this.clearOverlays(); this.overlaysContainer.innerHTML = ""; } @@ -869,7 +882,6 @@ $.extend( $.Viewer.prototype, $.EventSource.prototype, $.ControlDock.prototype, bodyStyle = body.style, docStyle = document.documentElement.style, _this = this, - hash, nodes, i; @@ -1031,9 +1043,9 @@ $.extend( $.Viewer.prototype, $.EventSource.prototype, $.ControlDock.prototype, $.setPageScroll( _this.pageScroll ); var pageScroll = $.getPageScroll(); restoreScrollCounter++; - if ( restoreScrollCounter < 10 && - pageScroll.x !== _this.pageScroll.x || - pageScroll.y !== _this.pageScroll.y ) { + if (restoreScrollCounter < 10 && + (pageScroll.x !== _this.pageScroll.x || + pageScroll.y !== _this.pageScroll.y)) { $.requestAnimationFrame( restoreScroll ); } }; @@ -1234,6 +1246,15 @@ $.extend( $.Viewer.prototype, $.EventSource.prototype, $.ControlDock.prototype, * @param {String} [options.compositeOperation] How the image is composited onto other images. * @param {String} [options.crossOriginPolicy] The crossOriginPolicy for this specific image, * overriding viewer.crossOriginPolicy. + * @param {Boolean} [options.ajaxWithCredentials] Whether to set withCredentials on tile AJAX + * @param {Boolean} [options.loadTilesWithAjax] + * Whether to load tile data using AJAX requests. + * Defaults to the setting in {@link OpenSeadragon.Options}. + * @param {Object} [options.ajaxHeaders] + * A set of headers to include when making tile AJAX requests. + * Note that these headers will be merged over any headers specified in {@link OpenSeadragon.Options}. + * Specifying a falsy value for a header will clear its existing value set at the Viewer level (if any). + * requests. * @param {Function} [options.success] A function that gets called when the image is * successfully added. It's passed the event object which contains a single property: * "item", the resulting TiledImage. @@ -1275,6 +1296,17 @@ $.extend( $.Viewer.prototype, $.EventSource.prototype, $.ControlDock.prototype, if (options.crossOriginPolicy === undefined) { options.crossOriginPolicy = options.tileSource.crossOriginPolicy !== undefined ? options.tileSource.crossOriginPolicy : this.crossOriginPolicy; } + if (options.ajaxWithCredentials === undefined) { + options.ajaxWithCredentials = this.ajaxWithCredentials; + } + if (options.loadTilesWithAjax === undefined) { + options.loadTilesWithAjax = this.loadTilesWithAjax; + } + if (options.ajaxHeaders === undefined || options.ajaxHeaders === null) { + options.ajaxHeaders = this.ajaxHeaders; + } else if ($.isPlainObject(options.ajaxHeaders) && $.isPlainObject(this.ajaxHeaders)) { + options.ajaxHeaders = $.extend({}, this.ajaxHeaders, options.ajaxHeaders); + } var myQueueItem = { options: options @@ -1390,6 +1422,9 @@ $.extend( $.Viewer.prototype, $.EventSource.prototype, $.ControlDock.prototype, smoothTileEdgesMinZoom: _this.smoothTileEdgesMinZoom, iOSDevice: _this.iOSDevice, crossOriginPolicy: queueItem.options.crossOriginPolicy, + ajaxWithCredentials: queueItem.options.ajaxWithCredentials, + loadTilesWithAjax: queueItem.options.loadTilesWithAjax, + ajaxHeaders: queueItem.options.ajaxHeaders, debugMode: _this.debugMode }); @@ -1531,7 +1566,7 @@ $.extend( $.Viewer.prototype, $.EventSource.prototype, $.ControlDock.prototype, onNextHandler = $.delegate( this, onNext ), onPreviousHandler = $.delegate( this, onPrevious ), navImages = this.navImages, - useGroup = true ; + useGroup = true; if( this.showSequenceControl ){ @@ -1627,7 +1662,7 @@ $.extend( $.Viewer.prototype, $.EventSource.prototype, $.ControlDock.prototype, onBlurHandler = $.delegate( this, onBlur ), navImages = this.navImages, buttons = [], - useGroup = true ; + useGroup = true; if ( this.showNavigationControl ) { @@ -1998,7 +2033,7 @@ $.extend( $.Viewer.prototype, $.EventSource.prototype, $.ControlDock.prototype, element = $.getElement( element ); i = getOverlayIndex( this.currentOverlays, element ); - if (i>=0) { + if (i >= 0) { return this.currentOverlays[i]; } else { return null; @@ -2119,6 +2154,7 @@ function _getSafeElemSize (oElement) { ); } + /** * @function * @private @@ -2134,7 +2170,12 @@ function getTileSourceImplementation( viewer, tileSource, imgOptions, successCal tileSource = $.parseXml( tileSource ); //json should start with "{" or "[" and end with "}" or "]" } else if ( tileSource.match(/^\s*[\{\[].*[\}\]]\s*$/ ) ) { - tileSource = $.parseJSON(tileSource); + try { + var tileSourceJ = $.parseJSON(tileSource); + tileSource = tileSourceJ; + } catch (e) { + //tileSource = tileSource; + } } } @@ -2162,6 +2203,7 @@ function getTileSourceImplementation( viewer, tileSource, imgOptions, successCal crossOriginPolicy: imgOptions.crossOriginPolicy !== undefined ? imgOptions.crossOriginPolicy : viewer.crossOriginPolicy, ajaxWithCredentials: viewer.ajaxWithCredentials, + ajaxHeaders: viewer.ajaxHeaders, useCanvas: viewer.useCanvas, success: function( event ) { successCallback( event.tileSource ); @@ -2469,16 +2511,15 @@ function onCanvasClick( event ) { this.canvas.focus(); } - if ( !event.preventDefaultAction && this.viewport && event.quick ) { - gestureSettings = this.gestureSettingsByDeviceType( event.pointerType ); - if ( gestureSettings.clickToZoom ) { - this.viewport.zoomBy( - event.shift ? 1.0 / this.zoomPerClick : this.zoomPerClick, - this.viewport.pointFromPixel( event.position, true ) - ); - this.viewport.applyConstraints(); - } - } + var canvasClickEventArgs = { + tracker: event.eventSource, + position: event.position, + quick: event.quick, + shift: event.shift, + originalEvent: event.originalEvent, + preventDefaultAction: event.preventDefaultAction + }; + /** * Raised when a mouse press/release or touch/remove occurs on the {@link OpenSeadragon.Viewer#canvas} element. * @@ -2491,15 +2532,21 @@ function onCanvasClick( event ) { * @property {Boolean} quick - True only if the clickDistThreshold and clickTimeThreshold are both passed. Useful for differentiating between clicks and drags. * @property {Boolean} shift - True if the shift key was pressed during this event. * @property {Object} originalEvent - The original DOM event. + * @property {Boolean} preventDefaultAction - Set to true to prevent default click to zoom behaviour. Default: false. * @property {?Object} userData - Arbitrary subscriber-defined object. */ - this.raiseEvent( 'canvas-click', { - tracker: event.eventSource, - position: event.position, - quick: event.quick, - shift: event.shift, - originalEvent: event.originalEvent - }); + this.raiseEvent( 'canvas-click', canvasClickEventArgs); + + if ( !canvasClickEventArgs.preventDefaultAction && this.viewport && event.quick ) { + gestureSettings = this.gestureSettingsByDeviceType( event.pointerType ); + if ( gestureSettings.clickToZoom ) { + this.viewport.zoomBy( + event.shift ? 1.0 / this.zoomPerClick : this.zoomPerClick, + this.viewport.pointFromPixel( event.position, true ) + ); + this.viewport.applyConstraints(); + } + } } function onCanvasDblClick( event ) { @@ -2538,20 +2585,16 @@ function onCanvasDblClick( event ) { function onCanvasDrag( event ) { var gestureSettings; - - if ( !event.preventDefaultAction && this.viewport ) { - gestureSettings = this.gestureSettingsByDeviceType( event.pointerType ); - if( !this.panHorizontal ){ - event.delta.x = 0; - } - if( !this.panVertical ){ - event.delta.y = 0; - } - this.viewport.panBy( this.viewport.deltaPointsFromPixels( event.delta.negate() ), gestureSettings.flickEnabled ); - if( this.constrainDuringPan ){ - this.viewport.applyConstraints(); - } - } + var canvasDragEventArgs = { + tracker: event.eventSource, + position: event.position, + delta: event.delta, + speed: event.speed, + direction: event.direction, + shift: event.shift, + originalEvent: event.originalEvent, + preventDefaultAction: event.preventDefaultAction + }; /** * Raised when a mouse or touch drag operation occurs on the {@link OpenSeadragon.Viewer#canvas} element. * @@ -2566,17 +2609,25 @@ function onCanvasDrag( event ) { * @property {Number} direction - Current computed direction, expressed as an angle counterclockwise relative to the positive X axis (-pi to pi, in radians). Only valid if speed > 0. * @property {Boolean} shift - True if the shift key was pressed during this event. * @property {Object} originalEvent - The original DOM event. + * @property {Boolean} preventDefaultAction - Set to true to prevent default drag behaviour. Default: false. * @property {?Object} userData - Arbitrary subscriber-defined object. */ - this.raiseEvent( 'canvas-drag', { - tracker: event.eventSource, - position: event.position, - delta: event.delta, - speed: event.speed, - direction: event.direction, - shift: event.shift, - originalEvent: event.originalEvent - }); + this.raiseEvent( 'canvas-drag', canvasDragEventArgs); + if ( !canvasDragEventArgs.preventDefaultAction && this.viewport ) { + gestureSettings = this.gestureSettingsByDeviceType( event.pointerType ); + if( !this.panHorizontal ){ + event.delta.x = 0; + } + if( !this.panVertical ){ + event.delta.y = 0; + } + if( event.delta.x !== 0 || event.delta.y !== 0 ){ + this.viewport.panBy( this.viewport.deltaPointsFromPixels( event.delta.negate() ), gestureSettings.flickEnabled ); + if( this.constrainDuringPan ){ + this.viewport.applyConstraints(); + } + } + } } function onCanvasDragEnd( event ) { diff --git a/src/viewport.js b/src/viewport.js index 4cac87c3..55851dc4 100644 --- a/src/viewport.js +++ b/src/viewport.js @@ -60,11 +60,11 @@ $.Viewport = function( options ) { //backward compatibility for positional args while prefering more //idiomatic javascript options object as the only argument var args = arguments; - if( args.length && args[ 0 ] instanceof $.Point ){ + if (args.length && args[0] instanceof $.Point) { options = { - containerSize: args[ 0 ], - contentSize: args[ 1 ], - config: args[ 2 ] + containerSize: args[0], + contentSize: args[1], + config: args[2] }; } @@ -805,7 +805,8 @@ $.Viewport.prototype = { * @return {OpenSeadragon.Viewport} Chainable. * @fires OpenSeadragon.Viewer.event:zoom */ - zoomTo: function( zoom, refPoint, immediately ) { + zoomTo: function(zoom, refPoint, immediately) { + var _this = this; this.zoomPoint = refPoint instanceof $.Point && !isNaN(refPoint.x) && @@ -813,13 +814,15 @@ $.Viewport.prototype = { refPoint : null; - if ( immediately ) { - this.zoomSpring.resetTo( zoom ); + if (immediately) { + this._adjustCenterSpringsForZoomPoint(function() { + _this.zoomSpring.resetTo(zoom); + }); } else { - this.zoomSpring.springTo( zoom ); + this.zoomSpring.springTo(zoom); } - if( this.viewer ){ + if (this.viewer) { /** * Raised when the viewport zoom level changes (see {@link OpenSeadragon.Viewport#zoomBy} and {@link OpenSeadragon.Viewport#zoomTo}). * @@ -832,7 +835,7 @@ $.Viewport.prototype = { * @property {Boolean} immediately * @property {?Object} userData - Arbitrary subscriber-defined object. */ - this.viewer.raiseEvent( 'zoom', { + this.viewer.raiseEvent('zoom', { zoom: zoom, refPoint: refPoint, immediately: immediately @@ -938,25 +941,10 @@ $.Viewport.prototype = { * @returns {Boolean} True if any change has been made, false otherwise. */ update: function() { - - if (this.zoomPoint) { - var oldZoomPixel = this.pixelFromPoint(this.zoomPoint, true); - this.zoomSpring.update(); - var newZoomPixel = this.pixelFromPoint(this.zoomPoint, true); - - var deltaZoomPixels = newZoomPixel.minus(oldZoomPixel); - var deltaZoomPoints = this.deltaPointsFromPixels( - deltaZoomPixels, true); - - this.centerSpringX.shiftBy(deltaZoomPoints.x); - this.centerSpringY.shiftBy(deltaZoomPoints.y); - - if (this.zoomSpring.isAtTargetValue()) { - this.zoomPoint = null; - } - } else { - this.zoomSpring.update(); - } + var _this = this; + this._adjustCenterSpringsForZoomPoint(function() { + _this.zoomSpring.update(); + }); this.centerSpringX.update(); this.centerSpringY.update(); @@ -972,6 +960,27 @@ $.Viewport.prototype = { return changed; }, + _adjustCenterSpringsForZoomPoint: function(zoomSpringHandler) { + if (this.zoomPoint) { + var oldZoomPixel = this.pixelFromPoint(this.zoomPoint, true); + zoomSpringHandler(); + var newZoomPixel = this.pixelFromPoint(this.zoomPoint, true); + + var deltaZoomPixels = newZoomPixel.minus(oldZoomPixel); + var deltaZoomPoints = this.deltaPointsFromPixels( + deltaZoomPixels, true); + + this.centerSpringX.shiftBy(deltaZoomPoints.x); + this.centerSpringY.shiftBy(deltaZoomPoints.y); + + if (this.zoomSpring.isAtTargetValue()) { + this.zoomPoint = null; + } + } else { + zoomSpringHandler(); + } + }, + /** * Convert a delta (translation vector) from viewport coordinates to pixels * coordinates. This method does not take rotation into account. @@ -1208,6 +1217,7 @@ $.Viewport.prototype = { * in image coordinate system. * @param {Number} [pixelWidth] the width in pixel of the rectangle. * @param {Number} [pixelHeight] the height in pixel of the rectangle. + * @returns {OpenSeadragon.Rect} This image's bounds in viewport coordinates */ imageToViewportRectangle: function(imageX, imageY, pixelWidth, pixelHeight) { var rect = imageX; diff --git a/src/world.js b/src/world.js index e68590b3..213aeaaa 100644 --- a/src/world.js +++ b/src/world.js @@ -212,7 +212,8 @@ $.extend( $.World.prototype, $.EventSource.prototype, /** @lends OpenSeadragon.W // We need to make sure any pending images are canceled so the world items don't get messed up this.viewer._cancelPendingImages(); var item; - for (var i = 0; i < this._items.length; i++) { + var i; + for (i = 0; i < this._items.length; i++) { item = this._items[i]; item.removeHandler('bounds-change', this._delegatedFigureSizes); item.removeHandler('clip-change', this._delegatedFigureSizes); diff --git a/src/zoomifytilesource.js b/src/zoomifytilesource.js index ad6aee52..a9f44ac2 100644 --- a/src/zoomifytilesource.js +++ b/src/zoomifytilesource.js @@ -18,7 +18,7 @@ * tilesUrl: "/test/data/zoomify/" * } * - * The tileSize is currently hardcoded to 256 (the usual Zoomify default). The tileUrl must the the path to the image _directory_. + * The tileSize is currently hardcoded to 256 (the usual Zoomify default). The tileUrl must the path to the image _directory_. * * 2) Loading image metadata from xml file: (CURRENTLY NOT SUPPORTED) * diff --git a/test/coverage.html b/test/coverage.html index b04d5fda..7fda4329 100644 --- a/test/coverage.html +++ b/test/coverage.html @@ -32,6 +32,7 @@ + @@ -79,6 +80,8 @@ + + diff --git a/test/data/testpattern.blob b/test/data/testpattern.blob new file mode 100644 index 00000000..fe027d91 Binary files /dev/null and b/test/data/testpattern.blob differ diff --git a/test/demo/customheaders.html b/test/demo/customheaders.html new file mode 100644 index 00000000..95063052 --- /dev/null +++ b/test/demo/customheaders.html @@ -0,0 +1,75 @@ + + + + OpenSeadragon Custom Request Headers Demo + + + + + +

+ Demo of how the loadTilesWithAjax and ajaxHeaders options as well as the getTileHeaders() method on TileSource can be applied. +

+

+ Examine the network requests in your browser developer tools to see the custom headers sent with each request. +

+
+ + + diff --git a/test/demo/zoomify.html b/test/demo/zoomify.html new file mode 100644 index 00000000..f4ba031e --- /dev/null +++ b/test/demo/zoomify.html @@ -0,0 +1,38 @@ + + + + OpenSeadragon Zoomify Demo + + + + + +
+ Simple demo page to show a default OpenSeadragon viewer with a Zoomify tile source. +
+
+ + + diff --git a/test/modules/ajax-tiles.js b/test/modules/ajax-tiles.js new file mode 100644 index 00000000..646e80b8 --- /dev/null +++ b/test/modules/ajax-tiles.js @@ -0,0 +1,242 @@ +/* global module, asyncTest, start, $, ok, equal, deepEqual, testLog */ + +(function() { + var viewer; + + // These values are generated by a script that concatenates all the tile files and records + // their byte ranges in a multi-dimensional array. + + // eslint-disable-next-line + var tileManifest = {"tileRanges":[[[[0,3467]]],[[[3467,6954]]],[[[344916,348425]]],[[[348425,351948]]],[[[351948,355576]]],[[[355576,359520]]],[[[359520,364663]]],[[[364663,374196]]],[[[374196,407307]]],[[[407307,435465],[435465,463663]],[[463663,491839],[491839,520078]]],[[[6954,29582],[29582,50315],[50315,71936],[71936,92703]],[[92703,113385],[113385,133265],[133265,154763],[154763,175710]],[[175710,197306],[197306,218807],[218807,242177],[242177,263007]],[[263007,283790],[283790,304822],[304822,325691],[325691,344916]]]],"totalSize":520078} + + function getTileRangeHeader(level, x, y) { + return 'bytes=' + tileManifest.tileRanges[level][x][y].join('-') + '/' + tileManifest.totalSize; + } + + // This tile source demonstrates how you can retrieve individual tiles from a single file + // using the Range header. + var customTileSource = { + width: 1000, + height: 1000, + tileWidth: 254, + tileHeight: 254, + tileOverlap: 1, + maxLevel: 10, + minLevel: 0, + // The tile URL is always the same. Only the Range header changes + getTileUrl: function () { + return '/test/data/testpattern.blob'; + }, + // This method will send the appropriate range header for this tile based on the data + // in tileByteRanges. + getTileAjaxHeaders: function(level, x, y) { + return { + Range: getTileRangeHeader(level, x, y) + }; + }, + }; + + module('AJAX-Tiles', { + setup: function() { + $('
').appendTo('#qunit-fixture'); + + testLog.reset(); + + viewer = OpenSeadragon({ + id: 'example', + prefixUrl: '/build/openseadragon/images/', + springStiffness: 100, // Faster animation = faster tests, + loadTilesWithAjax: true, + ajaxHeaders: { + 'X-Viewer-Header': 'ViewerHeaderValue' + } + }); + }, + teardown: function() { + if (viewer && viewer.close) { + viewer.close(); + } + + viewer = null; + } + }); + + asyncTest('tile-loaded event includes AJAX request object', function() { + var tileLoaded = function tileLoaded(evt) { + viewer.removeHandler('tile-loaded', tileLoaded); + ok(evt.tileRequest, 'Event includes tileRequest property'); + equal(evt.tileRequest.readyState, XMLHttpRequest.DONE, 'tileRequest is in completed state'); + start(); + }; + + viewer.addHandler('tile-loaded', tileLoaded); + viewer.open(customTileSource); + }); + + asyncTest('withCredentials is set in tile AJAX requests', function() { + var tileLoaded = function tileLoaded(evt) { + viewer.removeHandler('tile-loaded', tileLoaded); + ok(evt.tileRequest, 'Event includes tileRequest property'); + equal(evt.tileRequest.readyState, XMLHttpRequest.DONE, 'tileRequest is in completed state'); + equal(evt.tileRequest.withCredentials, true, 'withCredentials is set in tile request'); + start(); + }; + + viewer.addHandler('tile-loaded', tileLoaded); + viewer.addTiledImage({ + tileSource: customTileSource, + ajaxWithCredentials: true + }); + }); + + asyncTest('tile-load-failed event includes AJAX request object', function() { + // Create a tile source that points to a broken URL + var brokenTileSource = OpenSeadragon.extend({}, customTileSource, { + getTileUrl: function () { + return '/test/data/testpattern.blob.invalid'; + } + }); + + var tileLoadFailed = function tileLoadFailed(evt) { + viewer.removeHandler('tile-load-failed', tileLoadFailed); + ok(evt.tileRequest, 'Event includes tileRequest property'); + equal(evt.tileRequest.readyState, XMLHttpRequest.DONE, 'tileRequest is in completed state'); + start(); + }; + + viewer.addHandler('tile-load-failed', tileLoadFailed); + viewer.open(brokenTileSource); + }); + + asyncTest('Headers can be set per-tile', function() { + var tileLoaded = function tileLoaded(evt) { + viewer.removeHandler('tile-loaded', tileLoaded); + var tile = evt.tile; + ok(tile, 'tile property exists on event'); + ok(tile.ajaxHeaders, 'Tile has ajaxHeaders property'); + equal(tile.ajaxHeaders.Range, getTileRangeHeader(tile.level, tile.x, tile.y), 'Tile has correct range header.'); + start(); + }; + + viewer.addHandler('tile-loaded', tileLoaded); + + viewer.open(customTileSource); + }); + + asyncTest('Headers are propagated correctly', function() { + // Create a tile source that sets a static header for tiles + var staticHeaderTileSource = OpenSeadragon.extend({}, customTileSource, { + getTileAjaxHeaders: function() { + return { + 'X-Tile-Header': 'TileHeaderValue' + }; + } + }); + + var expectedHeaders = { + 'X-Viewer-Header': 'ViewerHeaderValue', + 'X-TiledImage-Header': 'TiledImageHeaderValue', + 'X-Tile-Header': 'TileHeaderValue' + }; + + var tileLoaded = function tileLoaded(evt) { + viewer.removeHandler('tile-loaded', tileLoaded); + var tile = evt.tile; + ok(tile, 'tile property exists on event'); + ok(tile.ajaxHeaders, 'Tile has ajaxHeaders property'); + deepEqual( + tile.ajaxHeaders, expectedHeaders, + 'Tile headers include headers set on Viewer and TiledImage' + ); + start(); + }; + + viewer.addHandler('tile-loaded', tileLoaded); + + viewer.addTiledImage({ + ajaxHeaders: { + 'X-TiledImage-Header': 'TiledImageHeaderValue' + }, + tileSource: staticHeaderTileSource + }); + }); + + asyncTest('Viewer headers are overwritten by TiledImage', function() { + // Create a tile source that sets a static header for tiles + var staticHeaderTileSource = OpenSeadragon.extend({}, customTileSource, { + getTileAjaxHeaders: function() { + return { + 'X-Tile-Header': 'TileHeaderValue' + }; + } + }); + + var expectedHeaders = { + 'X-Viewer-Header': 'ViewerHeaderValue-Overwritten', + 'X-TiledImage-Header': 'TiledImageHeaderValue', + 'X-Tile-Header': 'TileHeaderValue' + }; + + var tileLoaded = function tileLoaded(evt) { + viewer.removeHandler('tile-loaded', tileLoaded); + var tile = evt.tile; + ok(tile, 'tile property exists on event'); + ok(tile.ajaxHeaders, 'Tile has ajaxHeaders property'); + deepEqual( + tile.ajaxHeaders, expectedHeaders, + 'TiledImage header overwrites viewer header' + ); + start(); + }; + + viewer.addHandler('tile-loaded', tileLoaded); + + viewer.addTiledImage({ + ajaxHeaders: { + 'X-TiledImage-Header': 'TiledImageHeaderValue', + 'X-Viewer-Header': 'ViewerHeaderValue-Overwritten' + }, + tileSource: staticHeaderTileSource + }); + }); + + asyncTest('TiledImage headers are overwritten by Tile', function() { + + var expectedHeaders = { + 'X-Viewer-Header': 'ViewerHeaderValue', + 'X-TiledImage-Header': 'TiledImageHeaderValue-Overwritten', + 'X-Tile-Header': 'TileHeaderValue' + }; + + var tileLoaded = function tileLoaded(evt) { + viewer.removeHandler('tile-loaded', tileLoaded); + var tile = evt.tile; + ok(tile, 'tile property exists on event'); + ok(tile.ajaxHeaders, 'Tile has ajaxHeaders property'); + deepEqual( + tile.ajaxHeaders, expectedHeaders, + 'Tile header overwrites TiledImage header' + ); + start(); + }; + + viewer.addHandler('tile-loaded', tileLoaded); + + // Create a tile source that sets a static header for tiles + var staticHeaderTileSource = OpenSeadragon.extend({}, customTileSource, { + getTileAjaxHeaders: function() { + return { + 'X-TiledImage-Header': 'TiledImageHeaderValue-Overwritten', + 'X-Tile-Header': 'TileHeaderValue' + }; + } + }); + + viewer.addTiledImage({ + ajaxHeaders: { + 'X-TiledImage-Header': 'TiledImageHeaderValue' + }, + tileSource: staticHeaderTileSource + }); + }); +})(); diff --git a/test/modules/basic.js b/test/modules/basic.js index c6164560..7dc2a9ef 100644 --- a/test/modules/basic.js +++ b/test/modules/basic.js @@ -331,7 +331,7 @@ height: 155 } ] } ); - viewer.addHandler('tile-drawn', function() { + viewer.addOnceHandler('tile-drawn', function() { ok(OpenSeadragon.isCanvasTainted(viewer.drawer.context.canvas), "Canvas should be tainted."); start(); @@ -355,7 +355,7 @@ height: 155 } ] } ); - viewer.addHandler('tile-drawn', function() { + viewer.addOnceHandler('tile-drawn', function() { ok(!OpenSeadragon.isCanvasTainted(viewer.drawer.context.canvas), "Canvas should not be tainted."); start(); @@ -385,7 +385,7 @@ }, crossOriginPolicy : false } ); - viewer.addHandler('tile-drawn', function() { + viewer.addOnceHandler('tile-drawn', function() { ok(OpenSeadragon.isCanvasTainted(viewer.drawer.context.canvas), "Canvas should be tainted."); start(); @@ -414,7 +414,7 @@ crossOriginPolicy : "Anonymous" } } ); - viewer.addHandler('tile-drawn', function() { + viewer.addOnceHandler('tile-drawn', function() { ok(!OpenSeadragon.isCanvasTainted(viewer.drawer.context.canvas), "Canvas should not be tainted."); start(); @@ -423,4 +423,12 @@ }); } ); + + test('version object', function() { + equal(typeof OpenSeadragon.version.versionStr, "string", "versionStr should be a string"); + ok(OpenSeadragon.version.major >= 0, "major should be a positive number"); + ok(OpenSeadragon.version.minor >= 0, "minor should be a positive number"); + ok(OpenSeadragon.version.revision >= 0, "revision should be a positive number"); + }); + })(); diff --git a/test/modules/dzitilesource.js b/test/modules/dzitilesource.js index fe4677a0..25b0c265 100644 --- a/test/modules/dzitilesource.js +++ b/test/modules/dzitilesource.js @@ -32,7 +32,13 @@ 'relative link should stay the same'); testImplicitTilesUrl( '/p/foo.dzi?a=1&b=2', '/p/foo_files/', - 'querystring in dzi url should be ignored'); + 'querystring in dzi url should be ignored after slashes'); + testImplicitTilesUrl( + '/iiipsrv?DeepZoom=/path/my.dzi', '/iiipsrv?DeepZoom=/path/my_files/', + 'querystring in dzi url should not be ignored before slashes'); + testImplicitTilesUrl( + '/fcg-bin/iipsrv.fcgi?Deepzoom=123test.tif.dzi', '/fcg-bin/iipsrv.fcgi?Deepzoom=123test.tif_files/', + 'filename in querystring does not have to contain slash'); }); }()); diff --git a/test/modules/imageloader.js b/test/modules/imageloader.js new file mode 100644 index 00000000..1750ea73 --- /dev/null +++ b/test/modules/imageloader.js @@ -0,0 +1,88 @@ +/* global module, asyncTest, $, ok, equal, notEqual, start, test, Util, testLog */ + +(function() { + var viewer, + baseOptions = { + id: 'example', + prefixUrl: '/build/openseadragon/images/', + springStiffness: 100 // Faster animation = faster tests + }; + + module('ImageLoader', { + setup: function () { + var example = $('
').appendTo("#qunit-fixture"); + + testLog.reset(); + }, + teardown: function () { + if (viewer && viewer.close) { + viewer.close(); + } + + viewer = null; + } + }); + + // ---------- + + test('Default timeout', function() { + var actual, + expected = OpenSeadragon.DEFAULT_SETTINGS.timeout, + message, + options = OpenSeadragon.extend(true, baseOptions, { + imageLoaderLimit: 1 + }), + viewer = OpenSeadragon(options), + imageLoader = viewer.imageLoader; + + message = 'ImageLoader timeout should be set to the default value of ' + expected + ' when none is specified'; + actual = imageLoader.timeout; + equal(actual, expected, message); + + // Manually seize the ImageLoader + imageLoader.jobsInProgress = imageLoader.jobLimit; + imageLoader.addJob({ + src: 'test', + loadWithAjax: false, + crossOriginPolicy: 'test', + ajaxWithCredentials: false, + abort: function() {} + }); + + message = 'ImageJob should inherit the ImageLoader timeout value'; + actual = imageLoader.jobQueue.shift().timeout; + equal(actual, expected, message); + }); + + // ---------- + + test('Configure timeout', function() { + var actual, + expected = 123456, + message, + options = OpenSeadragon.extend(true, baseOptions, { + imageLoaderLimit: 1, + timeout: expected + }), + viewer = OpenSeadragon(options), + imageLoader = viewer.imageLoader; + + message = 'ImageLoader timeout should be configurable'; + actual = imageLoader.timeout; + equal(actual, expected, message); + + imageLoader.jobsInProgress = imageLoader.jobLimit; + imageLoader.addJob({ + src: 'test', + loadWithAjax: false, + crossOriginPolicy: 'test', + ajaxWithCredentials: false, + abort: function() {} + }); + + message = 'ImageJob should inherit the ImageLoader timeout value'; + actual = imageLoader.jobQueue.shift().timeout; + equal(actual, expected, message); + }); + +})(); diff --git a/test/modules/multi-image.js b/test/modules/multi-image.js index 4cbd5211..1dc9cbca 100644 --- a/test/modules/multi-image.js +++ b/test/modules/multi-image.js @@ -217,7 +217,7 @@ var firstImage = viewer.world.getItemAt(0); firstImage.addHandler('fully-loaded-change', function() { var imageData = viewer.drawer.context.getImageData(0, 0, - 500 * OpenSeadragon.pixelDensityRatio, 500 * density); + 500 * density, 500 * density); // Pixel 250,250 will be in the hole of the A var expectedVal = getPixelValue(imageData, 250 * density, 250 * density); diff --git a/test/modules/strings.js b/test/modules/strings.js index 0cc0d8d4..83f97d82 100644 --- a/test/modules/strings.js +++ b/test/modules/strings.js @@ -22,11 +22,11 @@ test("getInvalidString", function() { equal(OpenSeadragon.getString("Greeting"), "", "Handled unset string key"); - ok(testLog.debug.contains('["Untranslated source string:","Greeting"]'), + ok(testLog.log.contains('["Untranslated source string:","Greeting"]'), 'Invalid string keys are logged'); equal(OpenSeadragon.getString("Errors"), "", "Handled requesting parent key"); - ok(testLog.debug.contains('["Untranslated source string:","Errors"]'), + ok(testLog.log.contains('["Untranslated source string:","Errors"]'), 'Invalid string parent keys are logged'); }); diff --git a/test/modules/tilecache.js b/test/modules/tilecache.js index 80bb44de..ba89b73a 100644 --- a/test/modules/tilecache.js +++ b/test/modules/tilecache.js @@ -25,12 +25,14 @@ var fakeTile0 = { url: 'foo.jpg', + cacheKey: 'foo.jpg', image: {}, unload: function() {} }; var fakeTile1 = { url: 'foo.jpg', + cacheKey: 'foo.jpg', image: {}, unload: function() {} }; @@ -74,18 +76,21 @@ var fakeTile0 = { url: 'different.jpg', + cacheKey: 'different.jpg', image: {}, unload: function() {} }; var fakeTile1 = { url: 'same.jpg', + cacheKey: 'same.jpg', image: {}, unload: function() {} }; var fakeTile2 = { url: 'same.jpg', + cacheKey: 'same.jpg', image: {}, unload: function() {} }; diff --git a/test/modules/tiledimage.js b/test/modules/tiledimage.js index 20171432..b2206341 100644 --- a/test/modules/tiledimage.js +++ b/test/modules/tiledimage.js @@ -319,10 +319,16 @@ function testDefaultRotation() { var image = viewer.world.getItemAt(0); - strictEqual(image.getRotation(), 0, 'image has default rotation'); + strictEqual(image.getRotation(true), 0, 'image has default current rotation'); + strictEqual(image.getRotation(false), 0, 'image has default target rotation'); image.setRotation(400); - strictEqual(image.getRotation(), 40, 'rotation is set correctly'); + strictEqual(image.getRotation(true), 0, 'current rotation is not changed'); + strictEqual(image.getRotation(false), 400, 'target rotation is set correctly'); + + image.setRotation(200, true); + strictEqual(image.getRotation(true), 200, 'current rotation is set correctly'); + strictEqual(image.getRotation(false), 200, 'target rotation is set correctly'); viewer.addOnceHandler('open', testTileSourceRotation); viewer.open({ @@ -333,7 +339,8 @@ function testTileSourceRotation() { var image = viewer.world.getItemAt(0); - strictEqual(image.getRotation(), 300, 'image has correct rotation'); + strictEqual(image.getRotation(true), -60, 'image has correct current rotation'); + strictEqual(image.getRotation(false), -60, 'image has correct target rotation'); start(); } @@ -515,4 +522,159 @@ }]); }); + // PhantomJS is missing Function.prototype.bind + function bind(func, _this) { + return function() { + return func.apply(_this, arguments); + }; + } + + test('_getCornerTiles without wrapping', function() { + var tiledImageMock = { + wrapHorizontal: false, + wrapVertical: false, + source: new OpenSeadragon.TileSource({ + width: 1500, + height: 1000, + tileWidth: 200, + tileHeight: 150, + tileOverlap: 1, + }), + }; + var _getCornerTiles = bind( + OpenSeadragon.TiledImage.prototype._getCornerTiles, + tiledImageMock); + + function assertCornerTiles(topLeftBound, bottomRightBound, + expectedTopLeft, expectedBottomRight) { + var cornerTiles = _getCornerTiles(11, topLeftBound, bottomRightBound); + ok(cornerTiles.topLeft.equals(expectedTopLeft), + 'Top left tile should be ' + expectedTopLeft.toString() + + ' found ' + cornerTiles.topLeft.toString()); + ok(cornerTiles.bottomRight.equals(expectedBottomRight), + 'Bottom right tile should be ' + expectedBottomRight.toString() + + ' found ' + cornerTiles.bottomRight.toString()); + } + + assertCornerTiles( + new OpenSeadragon.Point(0, 0), + new OpenSeadragon.Point(1, 10 / 15), + new OpenSeadragon.Point(0, 0), + new OpenSeadragon.Point(7, 6) + ) + + // Floating point errors should be handled + assertCornerTiles( + new OpenSeadragon.Point(-1e-14, -1e-14), + new OpenSeadragon.Point(1 + 1e-14, 10 / 15 + 1e-14), + new OpenSeadragon.Point(0, 0), + new OpenSeadragon.Point(7, 6) + ) + + assertCornerTiles( + new OpenSeadragon.Point(0.3, 0.5), + new OpenSeadragon.Point(0.5, 0.6), + new OpenSeadragon.Point(2, 5), + new OpenSeadragon.Point(3, 6) + ) + }); + + test('_getCornerTiles with horizontal wrapping', function() { + var tiledImageMock = { + wrapHorizontal: true, + wrapVertical: false, + source: new OpenSeadragon.TileSource({ + width: 1500, + height: 1000, + tileWidth: 200, + tileHeight: 150, + tileOverlap: 1, + }), + }; + var _getCornerTiles = bind( + OpenSeadragon.TiledImage.prototype._getCornerTiles, + tiledImageMock); + + function assertCornerTiles(topLeftBound, bottomRightBound, + expectedTopLeft, expectedBottomRight) { + var cornerTiles = _getCornerTiles(11, topLeftBound, bottomRightBound); + ok(cornerTiles.topLeft.equals(expectedTopLeft), + 'Top left tile should be ' + expectedTopLeft.toString() + + ' found ' + cornerTiles.topLeft.toString()); + ok(cornerTiles.bottomRight.equals(expectedBottomRight), + 'Bottom right tile should be ' + expectedBottomRight.toString() + + ' found ' + cornerTiles.bottomRight.toString()); + } + + assertCornerTiles( + new OpenSeadragon.Point(0, 0), + new OpenSeadragon.Point(1, 10 / 15), + new OpenSeadragon.Point(0, 0), + new OpenSeadragon.Point(8, 6) + ) + + assertCornerTiles( + new OpenSeadragon.Point(-1, 0), + new OpenSeadragon.Point(0.5, 10 / 15 + 1e-14), + new OpenSeadragon.Point(-8, 0), + new OpenSeadragon.Point(3, 6) + ) + + assertCornerTiles( + new OpenSeadragon.Point(1.3, 0.5), + new OpenSeadragon.Point(1.5, 0.6), + new OpenSeadragon.Point(10, 5), + new OpenSeadragon.Point(11, 6) + ) + }); + + test('_getCornerTiles with vertical wrapping', function() { + var tiledImageMock = { + wrapHorizontal: false, + wrapVertical: true, + source: new OpenSeadragon.TileSource({ + width: 1500, + height: 1000, + tileWidth: 200, + tileHeight: 150, + tileOverlap: 1, + }), + }; + var _getCornerTiles = bind( + OpenSeadragon.TiledImage.prototype._getCornerTiles, + tiledImageMock); + + function assertCornerTiles(topLeftBound, bottomRightBound, + expectedTopLeft, expectedBottomRight) { + var cornerTiles = _getCornerTiles(11, topLeftBound, bottomRightBound); + ok(cornerTiles.topLeft.equals(expectedTopLeft), + 'Top left tile should be ' + expectedTopLeft.toString() + + ' found ' + cornerTiles.topLeft.toString()); + ok(cornerTiles.bottomRight.equals(expectedBottomRight), + 'Bottom right tile should be ' + expectedBottomRight.toString() + + ' found ' + cornerTiles.bottomRight.toString()); + } + + assertCornerTiles( + new OpenSeadragon.Point(0, 0), + new OpenSeadragon.Point(1, 10 / 15), + new OpenSeadragon.Point(0, 0), + new OpenSeadragon.Point(7, 7) + ) + + assertCornerTiles( + new OpenSeadragon.Point(0, -10 / 15 / 2), + new OpenSeadragon.Point(0.5, 0.5), + new OpenSeadragon.Point(0, -4), + new OpenSeadragon.Point(3, 5) + ) + + assertCornerTiles( + new OpenSeadragon.Point(0, 10 / 15 + 0.1), + new OpenSeadragon.Point(0.3, 10 / 15 + 0.3), + new OpenSeadragon.Point(0, 7), + new OpenSeadragon.Point(2, 9) + ) + }); + })(); diff --git a/test/modules/tilesource.js b/test/modules/tilesource.js index 59d8b77f..21a9f30c 100644 --- a/test/modules/tilesource.js +++ b/test/modules/tilesource.js @@ -1,32 +1,32 @@ /* global module, ok, equal, start, test, testLog, Util */ (function() { - + module('TileSource', { setup: function() { testLog.reset(); } }); - - + + test("should set sane tile size defaults", function() { var source = new OpenSeadragon.TileSource(); - + equal(source.getTileWidth(), 0, "getTileWidth() should return 0 if not provided a size"); equal(source.getTileHeight(), 0, "getTileHeight() should return 0 if not provided a size"); }); - + test("providing tileSize", function(){ var tileSize = 256, source = new OpenSeadragon.TileSource({ tileSize: tileSize }); - + equal(source.tileSize, undefined, "tileSize should not be set on the tileSource"); equal(source.getTileWidth(), tileSize, "getTileWidth() should equal tileSize"); equal(source.getTileHeight(), tileSize, "getTileHeight() should equal tileSize"); }); - - + + test("providing tileWidth and tileHeight", function(){ var tileWidth = 256, tileHeight = 512, @@ -34,7 +34,7 @@ tileWidth: tileWidth, tileHeight: tileHeight }); - + equal(source._tileWidth, tileWidth, "tileWidth option should set _tileWidth"); equal(source._tileHeight, tileHeight, "tileHeight option should set _tileHeight"); equal(source.tileWidth, undefined, "tileWidth should be renamed _tileWidth"); @@ -42,10 +42,60 @@ equal(source.getTileWidth(), tileWidth, "getTileWidth() should equal tileWidth"); equal(source.getTileHeight(), tileHeight, "getTileHeight() should equal tileHeight"); }); - + test('getTileSize() deprecation', function() { var source = new OpenSeadragon.TileSource(); Util.testDeprecation(source, 'getTileSize'); }); + test('getTileAtPoint', function() { + var tileSource = new OpenSeadragon.TileSource({ + width: 1500, + height: 1000, + tileWidth: 200, + tileHeight: 150, + tileOverlap: 1, + }); + + equal(tileSource.maxLevel, 11, "The max level should be 11."); + + function assertTileAtPoint(level, position, expected) { + var actual = tileSource.getTileAtPoint(level, position); + ok(actual.equals(expected), "The tile at level " + level + + ", position " + position.toString() + + " should be tile " + expected.toString() + + " got " + actual.toString()); + } + + assertTileAtPoint(11, new OpenSeadragon.Point(0, 0), new OpenSeadragon.Point(0, 0)); + assertTileAtPoint(11, new OpenSeadragon.Point(0.5, 0.5), new OpenSeadragon.Point(3, 5)); + assertTileAtPoint(11, new OpenSeadragon.Point(1, 10 / 15), new OpenSeadragon.Point(7, 6)); + + assertTileAtPoint(10, new OpenSeadragon.Point(0, 0), new OpenSeadragon.Point(0, 0)); + assertTileAtPoint(10, new OpenSeadragon.Point(0.5, 0.5), new OpenSeadragon.Point(1, 2)); + assertTileAtPoint(10, new OpenSeadragon.Point(1, 10 / 15), new OpenSeadragon.Point(3, 3)); + + assertTileAtPoint(9, new OpenSeadragon.Point(0, 0), new OpenSeadragon.Point(0, 0)); + assertTileAtPoint(9, new OpenSeadragon.Point(0.5, 0.5), new OpenSeadragon.Point(0, 1)); + assertTileAtPoint(9, new OpenSeadragon.Point(1, 10 / 15), new OpenSeadragon.Point(1, 1)); + + // For all other levels, there is only one tile. + for (var level = 8; level >= 0; level--) { + assertTileAtPoint(level, new OpenSeadragon.Point(0, 0), new OpenSeadragon.Point(0, 0)); + assertTileAtPoint(level, new OpenSeadragon.Point(0.5, 0.5), new OpenSeadragon.Point(0, 0)); + assertTileAtPoint(level, new OpenSeadragon.Point(1, 10 / 15), new OpenSeadragon.Point(0, 0)); + } + + // Test for issue #1113 + tileSource = new OpenSeadragon.TileSource({ + width: 1006, + height: 1009, + tileWidth: 1006, + tileHeight: 1009, + tileOverlap: 0, + maxLevel: 0, + }); + assertTileAtPoint(0, new OpenSeadragon.Point(1, 1009 / 1006), new OpenSeadragon.Point(0, 0)); + }); + }()); diff --git a/test/modules/viewport.js b/test/modules/viewport.js index e49c1e68..236d4f9c 100644 --- a/test/modules/viewport.js +++ b/test/modules/viewport.js @@ -36,7 +36,7 @@ var TALL_PATH = '/test/data/tall.dzi'; var WIDE_PATH = '/test/data/wide.dzi'; - var testZoomLevels = [-1, 0, 0.1, 0.5, 4, 10]; + var testZoomLevels = [0.1, 0.2, 0.5, 1, 4, 10]; var testPoints = [ new OpenSeadragon.Point(0, 0), @@ -59,7 +59,6 @@ var reopenViewerHelper = function(config) { var expected, level, actual, i = 0; var openHandler = function(event) { - viewer.removeHandler('open', openHandler); var viewport = viewer.viewport; expected = config.processExpected(level, expected); actual = viewport[config.method](); @@ -70,7 +69,7 @@ "Test " + config.method + " with zoom level of " + level + ". Expected : " + expected + ", got " + actual ); i++; - if(i < testZoomLevels.length){ + if (i < testZoomLevels.length) { level = expected = testZoomLevels[i]; var viewerConfig = { id: VIEWER_ID, @@ -80,15 +79,22 @@ viewerConfig[config.property] = level; viewer = OpenSeadragon(viewerConfig); - viewer.addHandler('open', openHandler); + viewer.addOnceHandler('open', openHandler); viewer.open(DZI_PATH); } else { start(); } }; - viewer.addHandler('open', openHandler); level = expected = testZoomLevels[i]; - viewer[config.property] = level; + var viewerConfig = { + id: VIEWER_ID, + prefixUrl: PREFIX_URL, + springStiffness: SPRING_STIFFNESS + }; + + viewerConfig[config.property] = level; + viewer = OpenSeadragon(viewerConfig); + viewer.addOnceHandler('open', openHandler); viewer.open(DZI_PATH); }; @@ -211,15 +217,9 @@ property: 'defaultZoomLevel', method: 'getHomeBounds', processExpected: function(level, expected) { - // Have to special case this to avoid dividing by 0 - if(level === -1 || level === 0){ - expected = new OpenSeadragon.Rect(0, 0, 1, 1); - } else { - var sideLength = 1.0 / viewer.defaultZoomLevel; // it's a square in this case - var position = 0.5 - (sideLength / 2.0); - expected = new OpenSeadragon.Rect(position, position, sideLength, sideLength); - } - return expected; + var sideLength = 1.0 / viewer.defaultZoomLevel; // it's a square in this case + var position = 0.5 - (sideLength / 2.0); + return new OpenSeadragon.Rect(position, position, sideLength, sideLength); } }); }); @@ -333,44 +333,39 @@ // I don't use the helper for this one because it sets a couple more // properties that would need special casing. asyncTest('getHomeZoomWithHomeFillsViewer', function() { - var expected, level, i = 0; + var i = 0; var openHandler = function(event) { - viewer.removeHandler('open', openHandler); var viewport = viewer.viewport; viewport.zoomTo(ZOOM_FACTOR, null, true); - // Special cases for oddball levels - if (level === -1) { - expected = 0.25; - } else if(level === 0){ - expected = 1; - } - equal( viewport.getHomeZoom(), - expected, - "Test getHomeZoom with homeFillsViewer = true and default zoom level of " + expected + testZoomLevels[i], + "Test getHomeZoom with homeFillsViewer = true and default zoom level of " + testZoomLevels[i] ); i++; - if(i < testZoomLevels.length){ - level = expected = testZoomLevels[i]; + if (i < testZoomLevels.length) { viewer = OpenSeadragon({ - id: VIEWER_ID, - prefixUrl: PREFIX_URL, + id: VIEWER_ID, + prefixUrl: PREFIX_URL, springStiffness: SPRING_STIFFNESS, - defaultZoomLevel: level, + defaultZoomLevel: testZoomLevels[i], homeFillsViewer: true }); - viewer.addHandler('open', openHandler); + viewer.addOnceHandler('open', openHandler); viewer.open(TALL_PATH); // use a different image for homeFillsViewer } else { start(); } }; - viewer.addHandler('open', openHandler); - level = expected = testZoomLevels[i]; - viewer.homeFillsViewer = true; - viewer.defaultZoomLevel = expected; + viewer = OpenSeadragon({ + id: VIEWER_ID, + prefixUrl: PREFIX_URL, + springStiffness: SPRING_STIFFNESS, + defaultZoomLevel: testZoomLevels[i], + homeFillsViewer: true + }); + viewer.addOnceHandler('open', openHandler); viewer.open(TALL_PATH); // use a different image for homeFillsViewer }); @@ -725,27 +720,18 @@ viewer.open(DZI_PATH); }); - asyncTest('zoomBy', function(){ + asyncTest('zoomBy no ref point', function() { var openHandler = function(event) { viewer.removeHandler('open', openHandler); var viewport = viewer.viewport; - for (var i = 0; i < testZoomLevels.length; i++){ + for (var i = 0; i < testZoomLevels.length; i++) { viewport.zoomBy(testZoomLevels[i], null, true); propEqual( viewport.getZoom(), testZoomLevels[i], "Zoomed by the correct amount." ); - - // now use a ref point - // TODO: check the ending position due to ref point - viewport.zoomBy(testZoomLevels[i], testPoints[i], true); - propEqual( - viewport.getZoom(), - testZoomLevels[i], - "Zoomed by the correct amount." - ); } start(); @@ -754,27 +740,88 @@ viewer.open(DZI_PATH); }); - asyncTest('zoomTo', function(){ + asyncTest('zoomBy with ref point', function() { var openHandler = function(event) { viewer.removeHandler('open', openHandler); var viewport = viewer.viewport; - for (var i = 0; i < testZoomLevels.length; i++){ + var expectedCenters = [ + new OpenSeadragon.Point(5, 5), + new OpenSeadragon.Point(6.996, 6.996), + new OpenSeadragon.Point(7.246, 6.996), + new OpenSeadragon.Point(7.246, 6.996), + new OpenSeadragon.Point(7.621, 7.371), + new OpenSeadragon.Point(7.621, 7.371), + ]; + + for (var i = 0; i < testZoomLevels.length; i++) { + viewport.zoomBy(testZoomLevels[i], testPoints[i], true); + propEqual( + viewport.getZoom(), + testZoomLevels[i], + "Zoomed by the correct amount." + ); + assertPointsEquals( + viewport.getCenter(), + expectedCenters[i], + 1e-14, + "Panned to the correct location." + ); + } + + start(); + }; + viewer.addHandler('open', openHandler); + viewer.open(DZI_PATH); + }); + + asyncTest('zoomTo no ref point', function() { + var openHandler = function(event) { + viewer.removeHandler('open', openHandler); + var viewport = viewer.viewport; + + for (var i = 0; i < testZoomLevels.length; i++) { viewport.zoomTo(testZoomLevels[i], null, true); propEqual( viewport.getZoom(), testZoomLevels[i], "Zoomed to the correct level." ); + } - // now use a ref point - // TODO: check the ending position due to ref point + start(); + }; + viewer.addHandler('open', openHandler); + viewer.open(DZI_PATH); + }); + + asyncTest('zoomTo with ref point', function() { + var openHandler = function(event) { + viewer.removeHandler('open', openHandler); + var viewport = viewer.viewport; + + var expectedCenters = [ + new OpenSeadragon.Point(5, 5), + new OpenSeadragon.Point(4.7505, 4.7505), + new OpenSeadragon.Point(4.6005, 4.7505), + new OpenSeadragon.Point(4.8455, 4.9955), + new OpenSeadragon.Point(5.2205, 5.3705), + new OpenSeadragon.Point(5.2205, 5.3705), + ]; + + for (var i = 0; i < testZoomLevels.length; i++) { viewport.zoomTo(testZoomLevels[i], testPoints[i], true); propEqual( viewport.getZoom(), testZoomLevels[i], "Zoomed to the correct level." ); + assertPointsEquals( + viewport.getCenter(), + expectedCenters[i], + 1e-14, + "Panned to the correct location." + ); } start(); diff --git a/test/test.html b/test/test.html index 74495ff6..16e7e54a 100644 --- a/test/test.html +++ b/test/test.html @@ -14,7 +14,7 @@ - + @@ -42,6 +42,8 @@ + +