You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 

467 lines
14 KiB

  1. /*
  2. * jQuery PanZoom Plugin
  3. * Pan and zoom an image within a parent div.
  4. *
  5. * version: 0.9.0
  6. * @requires jQuery v1.4.2 or later (earlier probably work, but untested so far)
  7. *
  8. * Copyright (c) 2011 Ben Lumley
  9. * Examples and documentation at: https://github.com/benlumley/jQuery-PanZoom
  10. *
  11. * Dual licensed under the MIT and GPL licenses:
  12. * http://www.opensource.org/licenses/mit-license.php
  13. * http://www.gnu.org/licenses/gpl.html
  14. */
  15. (function( $ ){
  16. $.fn.panZoom = function(method) {
  17. if ( methods[method] ) {
  18. return methods[ method ].apply( this, Array.prototype.slice.call( arguments, 1 ));
  19. } else if ( typeof method === 'object' || ! method ) {
  20. return methods.init.apply( this, arguments );
  21. } else {
  22. $.error( 'Method ' + method + ' does not exist' );
  23. }
  24. };
  25. $.fn.panZoom.defaults = {
  26. zoomIn : false,
  27. zoomOut : false,
  28. panUp : false,
  29. panDown : false,
  30. panLeft : false,
  31. panRight : false,
  32. fit : false,
  33. out_x1 : false,
  34. out_y1 : false,
  35. out_x2 : false,
  36. out_y2 : false,
  37. min_width : 20,
  38. min_height : 20,
  39. zoom_step : 3,
  40. pan_step : 3,
  41. debug : false,
  42. directedit : false,
  43. aspect : true,
  44. factor : 1,
  45. animate : true,
  46. animate_duration : 200,
  47. animate_easing : 'linear',
  48. double_click : true,
  49. mousewheel : true,
  50. mousewheel_delta : 1,
  51. draggable : true,
  52. clickandhold : true
  53. };
  54. var settings = {}
  55. var methods = {
  56. 'init': function (options) {
  57. $.extend(settings, $.fn.panZoom.defaults, options);
  58. setupCSS.apply(this);
  59. setupData.apply(this);
  60. setupBindings.apply(this);
  61. methods.readPosition.apply(this);
  62. },
  63. 'destroy': function () {
  64. $(window).unbind('.panZoom');
  65. this.removeData('panZoom');
  66. },
  67. 'loadImage': function () {
  68. var data = this.data('panZoom');
  69. loadTargetDimensions.apply(this);
  70. methods.updatePosition.apply(this);
  71. if (data.last_image != null && data.last_image != this.attr('src')) {
  72. methods.fit.apply(this);
  73. }
  74. data.last_image = this.attr('src');
  75. data.loaded = true;
  76. },
  77. 'readPosition': function () {
  78. var data = this.data('panZoom');
  79. if (settings.out_x1) { data.position.x1 = settings.out_x1.val()*settings.factor }
  80. if (settings.out_y1) { data.position.y1 = settings.out_y1.val()*settings.factor }
  81. if (settings.out_x2) { data.position.x2 = settings.out_x2.val()*settings.factor }
  82. if (settings.out_y2) { data.position.y2 = settings.out_y2.val()*settings.factor }
  83. methods.updatePosition.apply(this);
  84. },
  85. 'updatePosition': function() {
  86. validatePosition.apply(this);
  87. writePosition.apply(this);
  88. applyPosition.apply(this);
  89. },
  90. 'fit': function () {
  91. var data = this.data('panZoom');
  92. data.position.x1 = 0;
  93. data.position.y1 = 0;
  94. data.position.x2 = data.viewport_dimensions.x;
  95. data.position.y2 = data.viewport_dimensions.y;
  96. methods.updatePosition.apply(this);
  97. },
  98. 'zoomIn': function (steps) {
  99. var data = this.data('panZoom');
  100. if (typeof(steps) == 'undefined') {
  101. var steps = getStepDimensions.apply(this);
  102. }
  103. console.debug(data.position);
  104. console.debug(data.viewport_dimensions);
  105. data.position.x1 = data.position.x1*1 - steps.zoom.x;
  106. data.position.x2 = data.position.x2*1 + steps.zoom.x;
  107. data.position.y1 = data.position.y1*1 - steps.zoom.y;
  108. data.position.y2 = data.position.y2*1 + steps.zoom.y;
  109. methods.updatePosition.apply(this);
  110. },
  111. 'zoomOut': function (steps) {
  112. var data = this.data('panZoom');
  113. if (typeof(steps) == 'undefined') {
  114. var steps = getStepDimensions.apply(this);
  115. }
  116. data.position.x1 = data.position.x1*1 + steps.zoom.x;
  117. data.position.x2 = data.position.x2*1 - steps.zoom.x;
  118. data.position.y1 = data.position.y1*1 + steps.zoom.y;
  119. data.position.y2 = data.position.y2*1 - steps.zoom.y;
  120. methods.updatePosition.apply(this);
  121. },
  122. 'panUp': function () {
  123. var data = this.data('panZoom');
  124. var steps = getStepDimensions.apply(this);
  125. data.position.y1 -= steps.pan.y;
  126. data.position.y2 -= steps.pan.y;
  127. methods.updatePosition.apply(this);
  128. },
  129. 'panDown': function () {
  130. var data = this.data('panZoom');
  131. var steps = getStepDimensions.apply(this);
  132. data.position.y1 = data.position.y1*1 + steps.pan.y;
  133. data.position.y2 = data.position.y2*1 + steps.pan.y;
  134. methods.updatePosition.apply(this);
  135. },
  136. 'panLeft': function () {
  137. var data = this.data('panZoom');
  138. var steps = getStepDimensions.apply(this);
  139. data.position.x1 -= steps.pan.x;
  140. data.position.x2 -= steps.pan.x;
  141. methods.updatePosition.apply(this);
  142. },
  143. 'panRight': function () {
  144. var data = this.data('panZoom');
  145. var steps = getStepDimensions.apply(this);
  146. data.position.x1 = data.position.x1*1 + steps.pan.x;
  147. data.position.x2 = data.position.x2*1 + steps.pan.x;
  148. methods.updatePosition.apply(this);
  149. },
  150. 'mouseWheel': function (delta) {
  151. // first calculate how much to zoom in/out
  152. var steps = getStepDimensions.apply(this);
  153. steps.zoom.x = steps.zoom.x * (Math.abs(delta) / settings.mousewheel_delta);
  154. steps.zoom.y = steps.zoom.y * (Math.abs(delta) / settings.mousewheel_delta);
  155. // then do it
  156. if (delta > 0) {
  157. methods.zoomIn.apply(this, [steps]);
  158. } else if (delta < 0) {
  159. methods.zoomOut.apply(this, [steps]);
  160. }
  161. },
  162. 'dragComplete': function() {
  163. var data = this.data('panZoom');
  164. data.position.x1 = this.position().left;
  165. data.position.y1 = this.position().top;
  166. data.position.x2 = this.position().left*1 + this.width();
  167. data.position.y2 = this.position().top*1 + this.height();
  168. methods.updatePosition.apply(this);
  169. },
  170. 'mouseDown': function (action) {
  171. methods[action].apply(this);
  172. if (settings.clickandhold) {
  173. var data = this.data('panZoom');
  174. methods.mouseUp.apply(this);
  175. data.mousedown_interval = window.setInterval(function (that, action) {
  176. that.panZoom(action);
  177. }, settings.animate_duration, this, action);
  178. }
  179. },
  180. 'mouseUp': function() {
  181. var data = this.data('panZoom');
  182. window.clearInterval(data.mousedown_interval);
  183. }
  184. }
  185. function setupBindings() {
  186. eventData = { target: this }
  187. // bind up controls
  188. if (settings.zoomIn) {
  189. settings.zoomIn.bind('mousedown.panZoom', eventData, function(event) {
  190. event.preventDefault(); event.data.target.panZoom('mouseDown', 'zoomIn');
  191. }).bind('mouseleave.panZoom mouseup.panZoom', eventData, function(event) {
  192. event.preventDefault(); event.data.target.panZoom('mouseUp');
  193. });
  194. }
  195. if (settings.zoomOut) {
  196. settings.zoomOut.bind('mousedown.panZoom', eventData, function(event) {
  197. event.preventDefault(); event.data.target.panZoom('mouseDown', 'zoomOut');
  198. }).bind('mouseleave.panZoom mouseup.panZoom', eventData, function(event) {
  199. event.preventDefault(); event.data.target.panZoom('mouseUp');
  200. });
  201. }
  202. if (settings.panUp) {
  203. settings.panUp.bind('mousedown.panZoom', eventData, function(event) {
  204. event.preventDefault(); event.data.target.panZoom('mouseDown', 'panUp');
  205. }).bind('mouseleave.panZoom mouseup.panZoom', eventData, function(event) {
  206. event.preventDefault(); event.data.target.panZoom('mouseUp');
  207. });
  208. }
  209. if (settings.panDown) {
  210. settings.panDown.bind('mousedown.panZoom', eventData, function(event) {
  211. event.preventDefault(); event.data.target.panZoom('mouseDown', 'panDown');
  212. }).bind('mouseleave.panZoom mouseup.panZoom', eventData, function(event) {
  213. event.preventDefault(); event.data.target.panZoom('mouseUp');
  214. });
  215. }
  216. if (settings.panLeft) {
  217. settings.panLeft.bind('mousedown.panZoom', eventData, function(event) {
  218. event.preventDefault(); event.data.target.panZoom('mouseDown', 'panLeft');
  219. }).bind('mouseleave.panZoom mouseup.panZoom', eventData, function(event) {
  220. event.preventDefault(); event.data.target.panZoom('mouseUp');
  221. });
  222. }
  223. if (settings.panRight) {
  224. settings.panRight.bind('mousedown.panZoom', eventData, function(event) {
  225. event.preventDefault(); event.data.target.panZoom('mouseDown', 'panRight');
  226. }).bind('mouseleave.panZoom mouseup.panZoom', eventData, function(event) {
  227. event.preventDefault(); event.data.target.panZoom('mouseUp');
  228. });
  229. }
  230. if (settings.fit) { settings.fit.bind('click.panZoom', eventData, function(event) { event.preventDefault(); event.data.target.panZoom('fit'); } ); }
  231. // double click
  232. if (settings.double_click) {
  233. this.bind('dblclick.panZoom', eventData, function(event, delta) { event.data.target.panZoom('zoomIn') } );
  234. }
  235. // mousewheel
  236. if (settings.mousewheel && typeof(this.mousewheel) == 'function') {
  237. this.parent().mousewheel(function(event, delta) { event.preventDefault(); $(this).find('img').panZoom('mouseWheel', delta) } );
  238. } else if (settings.mousewheel) {
  239. alert('Mousewheel requires mousewheel from jQuery tools - please include jQuery tools or disable mousewheel to remove this warning.')
  240. }
  241. // direct form input
  242. if (settings.directedit) {
  243. $(settings.out_x1).add(settings.out_y1).add(settings.out_x2).add(settings.out_y2).bind('change.panZoom blur.panZoom', eventData, function(event) { event.data.target.panZoom('readPosition') } );
  244. }
  245. if (settings.draggable && typeof(this.draggable) == 'function') {
  246. this.draggable({
  247. stop: function () { $(this).panZoom('dragComplete'); }
  248. });
  249. } else if (settings.draggable) {
  250. alert('Draggable requires jQuery UI - please include jQuery UI or disable draggable to remove this warning.')
  251. }
  252. // image load
  253. $(this).bind('load.panZoom', eventData, function (event) { event.data.target.panZoom('loadImage') })
  254. }
  255. function setupData() {
  256. this.data('panZoom', {
  257. target_element: this,
  258. target_dimensions: { x: null, y: null },
  259. viewport_element: this.parent(),
  260. viewport_dimensions: { x: this.parent().width(), y: this.parent().height() },
  261. position: { x1: null, y1: null, x2: null, y2: null },
  262. last_image: null,
  263. loaded: false,
  264. mousewheel_delta: 0,
  265. mousedown_interval: false
  266. });
  267. if (settings.debug) {
  268. console.log(this.data('panZoom'));
  269. }
  270. }
  271. function setupCSS() {
  272. if (this.parent().css('position') == 'static') {
  273. this.parent().css('position', 'relative');
  274. }
  275. this.css({
  276. 'position': 'absolute',
  277. 'top': 0,
  278. 'left': 0
  279. });
  280. if (settings.draggable) {
  281. this.css({
  282. 'cursor': 'move'
  283. });
  284. }
  285. }
  286. function validatePosition() {
  287. var data = this.data('panZoom');
  288. // if dimensions are too small...
  289. if ( data.position.x2 - data.position.x1 < settings.min_width/settings.factor || data.position.y2 - data.position.y1 < settings.min_height/settings.factor ) {
  290. // and second co-ords are zero (IE: no dims set), fit image
  291. if (data.position.x2 == 0 || data.position.y2 == 0) {
  292. methods.fit.apply(this);
  293. }
  294. // otherwise, backout a bit
  295. else {
  296. if (data.position.x2 - data.position.x1 < settings.min_width/settings.factor) {
  297. data.position.x2 = data.position.x1*1+settings.min_width/settings.factor;
  298. }
  299. if (data.position.y2 - data.position.y1 < settings.min_height/settings.factor) {
  300. data.position.y2 = data.position.y1*1+settings.min_height/settings.factor;
  301. }
  302. }
  303. }
  304. if (settings.aspect) {
  305. target = data.target_dimensions.ratio;
  306. current = getCurrentAspectRatio.apply(this)
  307. if (current > target) {
  308. new_width = getHeight.apply(this) * target;
  309. diff = getWidth.apply(this) - new_width;
  310. data.position.x1 = data.position.x1*1 + (diff/2);
  311. data.position.x2 = data.position.x2*1 - (diff/2);
  312. } else if (current < target) {
  313. new_height = getWidth.apply(this) / target;
  314. diff = getHeight.apply(this) - new_height;
  315. data.position.y1 = data.position.y1*1 + (diff/2);
  316. data.position.y2 = data.position.y2*1 - (diff/2);
  317. }
  318. }
  319. }
  320. function applyPosition() {
  321. var data = this.data('panZoom');
  322. width = getWidth.apply(this);
  323. height = getHeight.apply(this);
  324. left_offset = getLeftOffset.apply(this);
  325. top_offset = getTopOffset.apply(this);
  326. properties = {
  327. 'top': Math.round(top_offset),
  328. 'left': Math.round(left_offset),
  329. 'width': Math.round(width),
  330. 'height': Math.round(height)
  331. }
  332. if (data.loaded && settings.animate) {
  333. applyAnimate.apply(this, [ properties ]);
  334. } else {
  335. applyCSS.apply(this, [ properties ]);
  336. }
  337. if (settings.debug) {
  338. console.log('--');
  339. console.log('width:' + width);
  340. console.log('height:' + height);
  341. console.log('left:' + left_offset);
  342. console.log('top:' + top_offset);
  343. }
  344. }
  345. function applyCSS() {
  346. this.css( properties );
  347. }
  348. function applyAnimate() {
  349. this.stop().animate( properties , settings.animate_duration, settings.animate_easing);
  350. }
  351. function getWidth() {
  352. var data = this.data('panZoom');
  353. width = (data.position.x2 - data.position.x1);
  354. return width;
  355. }
  356. function getLeftOffset() {
  357. var data = this.data('panZoom');
  358. return data.position.x1;
  359. }
  360. function getHeight() {
  361. var data = this.data('panZoom');
  362. height = (data.position.y2 - data.position.y1);
  363. return height;
  364. }
  365. function getTopOffset() {
  366. var data = this.data('panZoom');
  367. top_offset = data.position.y1;
  368. return top_offset;
  369. }
  370. function getCurrentAspectRatio() {
  371. return (getWidth.apply(this) / getHeight.apply(this));
  372. }
  373. function writePosition() {
  374. var data = this.data('panZoom');
  375. if (settings.out_x1) { settings.out_x1.val(Math.round(data.position.x1 / settings.factor)) }
  376. if (settings.out_y1) { settings.out_y1.val(Math.round(data.position.y1 / settings.factor)) }
  377. if (settings.out_x2) { settings.out_x2.val(Math.round(data.position.x2 / settings.factor)) }
  378. if (settings.out_y2) { settings.out_y2.val(Math.round(data.position.y2 / settings.factor)) }
  379. }
  380. function getStepDimensions() {
  381. var data = this.data('panZoom');
  382. ret = {
  383. zoom: {
  384. x: (settings.zoom_step/100 * data.viewport_dimensions.x),
  385. y: (settings.zoom_step/100 * data.viewport_dimensions.y)
  386. },
  387. pan: {
  388. x: (settings.pan_step/100 * data.viewport_dimensions.x),
  389. y: (settings.pan_step/100 * data.viewport_dimensions.y)
  390. }
  391. }
  392. return ret;
  393. }
  394. function loadTargetDimensions() {
  395. var data = this.data('panZoom');
  396. var img = document.createElement('img');
  397. img.src = this.attr('src');
  398. img.id = "jqpz-temp";
  399. $('body').append(img);
  400. data.target_dimensions.x = $('#jqpz-temp').width();
  401. data.target_dimensions.y = $('#jqpz-temp').height();
  402. $('#jqpz-temp').remove();
  403. data.target_dimensions.ratio = data.target_dimensions.x / data.target_dimensions.y;
  404. }
  405. })( jQuery );