|
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467 |
- /*
- * jQuery PanZoom Plugin
- * Pan and zoom an image within a parent div.
- *
- * version: 0.9.0
- * @requires jQuery v1.4.2 or later (earlier probably work, but untested so far)
- *
- * Copyright (c) 2011 Ben Lumley
- * Examples and documentation at: https://github.com/benlumley/jQuery-PanZoom
- *
- * Dual licensed under the MIT and GPL licenses:
- * http://www.opensource.org/licenses/mit-license.php
- * http://www.gnu.org/licenses/gpl.html
- */
-
- (function( $ ){
-
- $.fn.panZoom = function(method) {
-
- if ( methods[method] ) {
- return methods[ method ].apply( this, Array.prototype.slice.call( arguments, 1 ));
- } else if ( typeof method === 'object' || ! method ) {
- return methods.init.apply( this, arguments );
- } else {
- $.error( 'Method ' + method + ' does not exist' );
- }
-
- };
-
- $.fn.panZoom.defaults = {
- zoomIn : false,
- zoomOut : false,
- panUp : false,
- panDown : false,
- panLeft : false,
- panRight : false,
- fit : false,
- out_x1 : false,
- out_y1 : false,
- out_x2 : false,
- out_y2 : false,
- min_width : 20,
- min_height : 20,
- zoom_step : 3,
- pan_step : 3,
- debug : false,
- directedit : false,
- aspect : true,
- factor : 1,
- animate : true,
- animate_duration : 200,
- animate_easing : 'linear',
- double_click : true,
- mousewheel : true,
- mousewheel_delta : 1,
- draggable : true,
- clickandhold : true
- };
-
- var settings = {}
-
- var methods = {
- 'init': function (options) {
- $.extend(settings, $.fn.panZoom.defaults, options);
- setupCSS.apply(this);
- setupData.apply(this);
- setupBindings.apply(this);
- methods.readPosition.apply(this);
- },
-
- 'destroy': function () {
- $(window).unbind('.panZoom');
- this.removeData('panZoom');
- },
-
- 'loadImage': function () {
- var data = this.data('panZoom');
- loadTargetDimensions.apply(this);
- methods.updatePosition.apply(this);
- if (data.last_image != null && data.last_image != this.attr('src')) {
- methods.fit.apply(this);
- }
- data.last_image = this.attr('src');
- data.loaded = true;
- },
-
- 'readPosition': function () {
- var data = this.data('panZoom');
- if (settings.out_x1) { data.position.x1 = settings.out_x1.val()*settings.factor }
- if (settings.out_y1) { data.position.y1 = settings.out_y1.val()*settings.factor }
- if (settings.out_x2) { data.position.x2 = settings.out_x2.val()*settings.factor }
- if (settings.out_y2) { data.position.y2 = settings.out_y2.val()*settings.factor }
- methods.updatePosition.apply(this);
- },
-
- 'updatePosition': function() {
- validatePosition.apply(this);
- writePosition.apply(this);
- applyPosition.apply(this);
- },
-
- 'fit': function () {
- var data = this.data('panZoom');
- data.position.x1 = 0;
- data.position.y1 = 0;
- data.position.x2 = data.viewport_dimensions.x;
- data.position.y2 = data.viewport_dimensions.y;
- methods.updatePosition.apply(this);
- },
-
- 'zoomIn': function (steps) {
- var data = this.data('panZoom');
- if (typeof(steps) == 'undefined') {
- var steps = getStepDimensions.apply(this);
- }
- console.debug(data.position);
- console.debug(data.viewport_dimensions);
- data.position.x1 = data.position.x1*1 - steps.zoom.x;
- data.position.x2 = data.position.x2*1 + steps.zoom.x;
- data.position.y1 = data.position.y1*1 - steps.zoom.y;
- data.position.y2 = data.position.y2*1 + steps.zoom.y;
- methods.updatePosition.apply(this);
- },
-
- 'zoomOut': function (steps) {
- var data = this.data('panZoom');
- if (typeof(steps) == 'undefined') {
- var steps = getStepDimensions.apply(this);
- }
- data.position.x1 = data.position.x1*1 + steps.zoom.x;
- data.position.x2 = data.position.x2*1 - steps.zoom.x;
- data.position.y1 = data.position.y1*1 + steps.zoom.y;
- data.position.y2 = data.position.y2*1 - steps.zoom.y;
- methods.updatePosition.apply(this);
- },
-
- 'panUp': function () {
- var data = this.data('panZoom');
- var steps = getStepDimensions.apply(this);
- data.position.y1 -= steps.pan.y;
- data.position.y2 -= steps.pan.y;
- methods.updatePosition.apply(this);
- },
-
- 'panDown': function () {
- var data = this.data('panZoom');
- var steps = getStepDimensions.apply(this);
- data.position.y1 = data.position.y1*1 + steps.pan.y;
- data.position.y2 = data.position.y2*1 + steps.pan.y;
- methods.updatePosition.apply(this);
- },
-
- 'panLeft': function () {
- var data = this.data('panZoom');
- var steps = getStepDimensions.apply(this);
- data.position.x1 -= steps.pan.x;
- data.position.x2 -= steps.pan.x;
- methods.updatePosition.apply(this);
- },
-
- 'panRight': function () {
- var data = this.data('panZoom');
- var steps = getStepDimensions.apply(this);
- data.position.x1 = data.position.x1*1 + steps.pan.x;
- data.position.x2 = data.position.x2*1 + steps.pan.x;
- methods.updatePosition.apply(this);
- },
-
- 'mouseWheel': function (delta) {
- // first calculate how much to zoom in/out
- var steps = getStepDimensions.apply(this);
- steps.zoom.x = steps.zoom.x * (Math.abs(delta) / settings.mousewheel_delta);
- steps.zoom.y = steps.zoom.y * (Math.abs(delta) / settings.mousewheel_delta);
-
- // then do it
- if (delta > 0) {
- methods.zoomIn.apply(this, [steps]);
- } else if (delta < 0) {
- methods.zoomOut.apply(this, [steps]);
- }
- },
-
- 'dragComplete': function() {
- var data = this.data('panZoom');
- data.position.x1 = this.position().left;
- data.position.y1 = this.position().top;
- data.position.x2 = this.position().left*1 + this.width();
- data.position.y2 = this.position().top*1 + this.height();
- methods.updatePosition.apply(this);
- },
-
- 'mouseDown': function (action) {
- methods[action].apply(this);
-
- if (settings.clickandhold) {
- var data = this.data('panZoom');
- methods.mouseUp.apply(this);
- data.mousedown_interval = window.setInterval(function (that, action) {
- that.panZoom(action);
- }, settings.animate_duration, this, action);
- }
- },
-
- 'mouseUp': function() {
- var data = this.data('panZoom');
- window.clearInterval(data.mousedown_interval);
- }
-
- }
-
- function setupBindings() {
-
- eventData = { target: this }
-
- // bind up controls
- if (settings.zoomIn) {
- settings.zoomIn.bind('mousedown.panZoom', eventData, function(event) {
- event.preventDefault(); event.data.target.panZoom('mouseDown', 'zoomIn');
- }).bind('mouseleave.panZoom mouseup.panZoom', eventData, function(event) {
- event.preventDefault(); event.data.target.panZoom('mouseUp');
- });
- }
-
- if (settings.zoomOut) {
- settings.zoomOut.bind('mousedown.panZoom', eventData, function(event) {
- event.preventDefault(); event.data.target.panZoom('mouseDown', 'zoomOut');
- }).bind('mouseleave.panZoom mouseup.panZoom', eventData, function(event) {
- event.preventDefault(); event.data.target.panZoom('mouseUp');
- });
- }
-
- if (settings.panUp) {
- settings.panUp.bind('mousedown.panZoom', eventData, function(event) {
- event.preventDefault(); event.data.target.panZoom('mouseDown', 'panUp');
- }).bind('mouseleave.panZoom mouseup.panZoom', eventData, function(event) {
- event.preventDefault(); event.data.target.panZoom('mouseUp');
- });
- }
-
- if (settings.panDown) {
- settings.panDown.bind('mousedown.panZoom', eventData, function(event) {
- event.preventDefault(); event.data.target.panZoom('mouseDown', 'panDown');
- }).bind('mouseleave.panZoom mouseup.panZoom', eventData, function(event) {
- event.preventDefault(); event.data.target.panZoom('mouseUp');
- });
- }
-
- if (settings.panLeft) {
- settings.panLeft.bind('mousedown.panZoom', eventData, function(event) {
- event.preventDefault(); event.data.target.panZoom('mouseDown', 'panLeft');
- }).bind('mouseleave.panZoom mouseup.panZoom', eventData, function(event) {
- event.preventDefault(); event.data.target.panZoom('mouseUp');
- });
- }
-
- if (settings.panRight) {
- settings.panRight.bind('mousedown.panZoom', eventData, function(event) {
- event.preventDefault(); event.data.target.panZoom('mouseDown', 'panRight');
- }).bind('mouseleave.panZoom mouseup.panZoom', eventData, function(event) {
- event.preventDefault(); event.data.target.panZoom('mouseUp');
- });
- }
-
- if (settings.fit) { settings.fit.bind('click.panZoom', eventData, function(event) { event.preventDefault(); event.data.target.panZoom('fit'); } ); }
-
- // double click
- if (settings.double_click) {
- this.bind('dblclick.panZoom', eventData, function(event, delta) { event.data.target.panZoom('zoomIn') } );
- }
-
- // mousewheel
- if (settings.mousewheel && typeof(this.mousewheel) == 'function') {
- this.parent().mousewheel(function(event, delta) { event.preventDefault(); $(this).find('img').panZoom('mouseWheel', delta) } );
- } else if (settings.mousewheel) {
- alert('Mousewheel requires mousewheel from jQuery tools - please include jQuery tools or disable mousewheel to remove this warning.')
- }
-
- // direct form input
- if (settings.directedit) {
- $(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') } );
- }
-
- if (settings.draggable && typeof(this.draggable) == 'function') {
- this.draggable({
- stop: function () { $(this).panZoom('dragComplete'); }
- });
- } else if (settings.draggable) {
- alert('Draggable requires jQuery UI - please include jQuery UI or disable draggable to remove this warning.')
- }
-
- // image load
- $(this).bind('load.panZoom', eventData, function (event) { event.data.target.panZoom('loadImage') })
-
- }
-
- function setupData() {
- this.data('panZoom', {
- target_element: this,
- target_dimensions: { x: null, y: null },
- viewport_element: this.parent(),
- viewport_dimensions: { x: this.parent().width(), y: this.parent().height() },
- position: { x1: null, y1: null, x2: null, y2: null },
- last_image: null,
- loaded: false,
- mousewheel_delta: 0,
- mousedown_interval: false
- });
- if (settings.debug) {
- console.log(this.data('panZoom'));
- }
- }
-
- function setupCSS() {
- if (this.parent().css('position') == 'static') {
- this.parent().css('position', 'relative');
- }
- this.css({
- 'position': 'absolute',
- 'top': 0,
- 'left': 0
- });
- if (settings.draggable) {
- this.css({
- 'cursor': 'move'
- });
- }
- }
-
- function validatePosition() {
- var data = this.data('panZoom');
- // if dimensions are too small...
- if ( data.position.x2 - data.position.x1 < settings.min_width/settings.factor || data.position.y2 - data.position.y1 < settings.min_height/settings.factor ) {
- // and second co-ords are zero (IE: no dims set), fit image
- if (data.position.x2 == 0 || data.position.y2 == 0) {
- methods.fit.apply(this);
- }
- // otherwise, backout a bit
- else {
- if (data.position.x2 - data.position.x1 < settings.min_width/settings.factor) {
- data.position.x2 = data.position.x1*1+settings.min_width/settings.factor;
- }
- if (data.position.y2 - data.position.y1 < settings.min_height/settings.factor) {
- data.position.y2 = data.position.y1*1+settings.min_height/settings.factor;
- }
- }
- }
-
- if (settings.aspect) {
- target = data.target_dimensions.ratio;
- current = getCurrentAspectRatio.apply(this)
- if (current > target) {
- new_width = getHeight.apply(this) * target;
- diff = getWidth.apply(this) - new_width;
- data.position.x1 = data.position.x1*1 + (diff/2);
- data.position.x2 = data.position.x2*1 - (diff/2);
- } else if (current < target) {
- new_height = getWidth.apply(this) / target;
- diff = getHeight.apply(this) - new_height;
- data.position.y1 = data.position.y1*1 + (diff/2);
- data.position.y2 = data.position.y2*1 - (diff/2);
- }
- }
-
-
- }
-
- function applyPosition() {
- var data = this.data('panZoom');
-
- width = getWidth.apply(this);
- height = getHeight.apply(this);
- left_offset = getLeftOffset.apply(this);
- top_offset = getTopOffset.apply(this);
-
- properties = {
- 'top': Math.round(top_offset),
- 'left': Math.round(left_offset),
- 'width': Math.round(width),
- 'height': Math.round(height)
- }
-
- if (data.loaded && settings.animate) {
- applyAnimate.apply(this, [ properties ]);
- } else {
- applyCSS.apply(this, [ properties ]);
- }
-
- if (settings.debug) {
- console.log('--');
- console.log('width:' + width);
- console.log('height:' + height);
- console.log('left:' + left_offset);
- console.log('top:' + top_offset);
- }
- }
-
- function applyCSS() {
- this.css( properties );
- }
-
- function applyAnimate() {
- this.stop().animate( properties , settings.animate_duration, settings.animate_easing);
- }
-
- function getWidth() {
- var data = this.data('panZoom');
- width = (data.position.x2 - data.position.x1);
- return width;
- }
-
- function getLeftOffset() {
- var data = this.data('panZoom');
- return data.position.x1;
- }
-
- function getHeight() {
- var data = this.data('panZoom');
- height = (data.position.y2 - data.position.y1);
- return height;
- }
-
- function getTopOffset() {
- var data = this.data('panZoom');
- top_offset = data.position.y1;
- return top_offset;
- }
-
- function getCurrentAspectRatio() {
- return (getWidth.apply(this) / getHeight.apply(this));
- }
-
- function writePosition() {
- var data = this.data('panZoom');
- if (settings.out_x1) { settings.out_x1.val(Math.round(data.position.x1 / settings.factor)) }
- if (settings.out_y1) { settings.out_y1.val(Math.round(data.position.y1 / settings.factor)) }
- if (settings.out_x2) { settings.out_x2.val(Math.round(data.position.x2 / settings.factor)) }
- if (settings.out_y2) { settings.out_y2.val(Math.round(data.position.y2 / settings.factor)) }
- }
-
- function getStepDimensions() {
- var data = this.data('panZoom');
- ret = {
- zoom: {
- x: (settings.zoom_step/100 * data.viewport_dimensions.x),
- y: (settings.zoom_step/100 * data.viewport_dimensions.y)
- },
- pan: {
- x: (settings.pan_step/100 * data.viewport_dimensions.x),
- y: (settings.pan_step/100 * data.viewport_dimensions.y)
- }
- }
- return ret;
- }
-
- function loadTargetDimensions() {
- var data = this.data('panZoom');
- var img = document.createElement('img');
- img.src = this.attr('src');
- img.id = "jqpz-temp";
- $('body').append(img);
- data.target_dimensions.x = $('#jqpz-temp').width();
- data.target_dimensions.y = $('#jqpz-temp').height();
- $('#jqpz-temp').remove();
- data.target_dimensions.ratio = data.target_dimensions.x / data.target_dimensions.y;
- }
-
- })( jQuery );
|