Du kannst nicht mehr als 25 Themen auswählen Themen müssen entweder mit einem Buchstaben oder einer Ziffer beginnen. Sie können Bindestriche („-“) enthalten und bis zu 35 Zeichen lang sein.
 
 
 
 
 
 

377 Zeilen
12 KiB

  1. /*! Backstretch - v2.0.4 - 2013-06-19
  2. * http://srobbin.com/jquery-plugins/backstretch/
  3. * Copyright (c) 2013 Scott Robbin; Licensed MIT */
  4. ;(function ($, window, undefined) {
  5. 'use strict';
  6. /* PLUGIN DEFINITION
  7. * ========================= */
  8. $.fn.backstretch = function (images, options) {
  9. // We need at least one image or method name
  10. if (images === undefined || images.length === 0) {
  11. $.error("No images were supplied for Backstretch");
  12. }
  13. /*
  14. * Scroll the page one pixel to get the right window height on iOS
  15. * Pretty harmless for everyone else
  16. */
  17. if ($(window).scrollTop() === 0 ) {
  18. window.scrollTo(0, 0);
  19. }
  20. return this.each(function () {
  21. var $this = $(this)
  22. , obj = $this.data('backstretch');
  23. // Do we already have an instance attached to this element?
  24. if (obj) {
  25. // Is this a method they're trying to execute?
  26. if (typeof images == 'string' && typeof obj[images] == 'function') {
  27. // Call the method
  28. obj[images](options);
  29. // No need to do anything further
  30. return;
  31. }
  32. // Merge the old options with the new
  33. options = $.extend(obj.options, options);
  34. // Remove the old instance
  35. obj.destroy(true);
  36. }
  37. obj = new Backstretch(this, images, options);
  38. $this.data('backstretch', obj);
  39. });
  40. };
  41. // If no element is supplied, we'll attach to body
  42. $.backstretch = function (images, options) {
  43. // Return the instance
  44. return $('body')
  45. .backstretch(images, options)
  46. .data('backstretch');
  47. };
  48. // Custom selector
  49. $.expr[':'].backstretch = function(elem) {
  50. return $(elem).data('backstretch') !== undefined;
  51. };
  52. /* DEFAULTS
  53. * ========================= */
  54. $.fn.backstretch.defaults = {
  55. centeredX: true // Should we center the image on the X axis?
  56. , centeredY: true // Should we center the image on the Y axis?
  57. , duration: 5000 // Amount of time in between slides (if slideshow)
  58. , fade: 0 // Speed of fade transition between slides
  59. };
  60. /* STYLES
  61. *
  62. * Baked-in styles that we'll apply to our elements.
  63. * In an effort to keep the plugin simple, these are not exposed as options.
  64. * That said, anyone can override these in their own stylesheet.
  65. * ========================= */
  66. var styles = {
  67. wrap: {
  68. left: 0
  69. , top: 0
  70. , overflow: 'hidden'
  71. , margin: 0
  72. , padding: 0
  73. , height: '100%'
  74. , width: '100%'
  75. , zIndex: -999999
  76. }
  77. , img: {
  78. position: 'absolute'
  79. , display: 'none'
  80. , margin: 0
  81. , padding: 0
  82. , border: 'none'
  83. , width: 'auto'
  84. , height: 'auto'
  85. , maxHeight: 'none'
  86. , maxWidth: 'none'
  87. , zIndex: -999999
  88. }
  89. };
  90. /* CLASS DEFINITION
  91. * ========================= */
  92. var Backstretch = function (container, images, options) {
  93. this.options = $.extend({}, $.fn.backstretch.defaults, options || {});
  94. /* In its simplest form, we allow Backstretch to be called on an image path.
  95. * e.g. $.backstretch('/path/to/image.jpg')
  96. * So, we need to turn this back into an array.
  97. */
  98. this.images = $.isArray(images) ? images : [images];
  99. // Preload images
  100. $.each(this.images, function () {
  101. $('<img />')[0].src = this;
  102. });
  103. // Convenience reference to know if the container is body.
  104. this.isBody = container === document.body;
  105. /* We're keeping track of a few different elements
  106. *
  107. * Container: the element that Backstretch was called on.
  108. * Wrap: a DIV that we place the image into, so we can hide the overflow.
  109. * Root: Convenience reference to help calculate the correct height.
  110. */
  111. this.$container = $(container);
  112. this.$root = this.isBody ? supportsFixedPosition ? $(window) : $(document) : this.$container;
  113. // Don't create a new wrap if one already exists (from a previous instance of Backstretch)
  114. var $existing = this.$container.children(".backstretch").first();
  115. this.$wrap = $existing.length ? $existing : $('<div class="backstretch"></div>').css(styles.wrap).appendTo(this.$container);
  116. // Non-body elements need some style adjustments
  117. if (!this.isBody) {
  118. // If the container is statically positioned, we need to make it relative,
  119. // and if no zIndex is defined, we should set it to zero.
  120. var position = this.$container.css('position')
  121. , zIndex = this.$container.css('zIndex');
  122. this.$container.css({
  123. position: position === 'static' ? 'relative' : position
  124. , zIndex: zIndex === 'auto' ? 0 : zIndex
  125. , background: 'none'
  126. });
  127. // Needs a higher z-index
  128. this.$wrap.css({zIndex: -999998});
  129. }
  130. // Fixed or absolute positioning?
  131. this.$wrap.css({
  132. position: this.isBody && supportsFixedPosition ? 'fixed' : 'absolute'
  133. });
  134. // Set the first image
  135. this.index = 0;
  136. this.show(this.index);
  137. // Listen for resize
  138. $(window).on('resize.backstretch', $.proxy(this.resize, this))
  139. .on('orientationchange.backstretch', $.proxy(function () {
  140. // Need to do this in order to get the right window height
  141. if (this.isBody && window.pageYOffset === 0) {
  142. window.scrollTo(0, 1);
  143. this.resize();
  144. }
  145. }, this));
  146. };
  147. /* PUBLIC METHODS
  148. * ========================= */
  149. Backstretch.prototype = {
  150. resize: function () {
  151. try {
  152. var bgCSS = {left: 0, top: 0}
  153. , rootWidth = this.isBody ? this.$root.width() : this.$root.innerWidth()
  154. , bgWidth = rootWidth
  155. , rootHeight = this.isBody ? ( window.innerHeight ? window.innerHeight : this.$root.height() ) : this.$root.innerHeight()
  156. , bgHeight = bgWidth / this.$img.data('ratio')
  157. , bgOffset;
  158. // Make adjustments based on image ratio
  159. if (bgHeight >= rootHeight) {
  160. bgOffset = (bgHeight - rootHeight) / 2;
  161. if(this.options.centeredY) {
  162. bgCSS.top = '-' + bgOffset + 'px';
  163. }
  164. } else {
  165. bgHeight = rootHeight;
  166. bgWidth = bgHeight * this.$img.data('ratio');
  167. bgOffset = (bgWidth - rootWidth) / 2;
  168. if(this.options.centeredX) {
  169. bgCSS.left = '-' + bgOffset + 'px';
  170. }
  171. }
  172. this.$wrap.css({width: rootWidth, height: rootHeight})
  173. .find('img:not(.deleteable)').css({width: bgWidth, height: bgHeight}).css(bgCSS);
  174. } catch(err) {
  175. // IE7 seems to trigger resize before the image is loaded.
  176. // This try/catch block is a hack to let it fail gracefully.
  177. }
  178. return this;
  179. }
  180. // Show the slide at a certain position
  181. , show: function (newIndex) {
  182. // Validate index
  183. if (Math.abs(newIndex) > this.images.length - 1) {
  184. return;
  185. }
  186. // Vars
  187. var self = this
  188. , oldImage = self.$wrap.find('img').addClass('deleteable')
  189. , evtOptions = { relatedTarget: self.$container[0] };
  190. // Trigger the "before" event
  191. self.$container.trigger($.Event('backstretch.before', evtOptions), [self, newIndex]);
  192. // Set the new index
  193. this.index = newIndex;
  194. // Pause the slideshow
  195. clearInterval(self.interval);
  196. // New image
  197. self.$img = $('<img />')
  198. .css(styles.img)
  199. .bind('load', function (e) {
  200. var imgWidth = this.width || $(e.target).width()
  201. , imgHeight = this.height || $(e.target).height();
  202. // Save the ratio
  203. $(this).data('ratio', imgWidth / imgHeight);
  204. // Show the image, then delete the old one
  205. // "speed" option has been deprecated, but we want backwards compatibilty
  206. $(this).fadeIn(self.options.speed || self.options.fade, function () {
  207. oldImage.remove();
  208. // Resume the slideshow
  209. if (!self.paused) {
  210. self.cycle();
  211. }
  212. // Trigger the "after" and "show" events
  213. // "show" is being deprecated
  214. $(['after', 'show']).each(function () {
  215. self.$container.trigger($.Event('backstretch.' + this, evtOptions), [self, newIndex]);
  216. });
  217. });
  218. // Resize
  219. self.resize();
  220. })
  221. .appendTo(self.$wrap);
  222. // Hack for IE img onload event
  223. self.$img.attr('src', self.images[newIndex]);
  224. return self;
  225. }
  226. , next: function () {
  227. // Next slide
  228. return this.show(this.index < this.images.length - 1 ? this.index + 1 : 0);
  229. }
  230. , prev: function () {
  231. // Previous slide
  232. return this.show(this.index === 0 ? this.images.length - 1 : this.index - 1);
  233. }
  234. , pause: function () {
  235. // Pause the slideshow
  236. this.paused = true;
  237. return this;
  238. }
  239. , resume: function () {
  240. // Resume the slideshow
  241. this.paused = false;
  242. this.next();
  243. return this;
  244. }
  245. , cycle: function () {
  246. // Start/resume the slideshow
  247. if(this.images.length > 1) {
  248. // Clear the interval, just in case
  249. clearInterval(this.interval);
  250. this.interval = setInterval($.proxy(function () {
  251. // Check for paused slideshow
  252. if (!this.paused) {
  253. this.next();
  254. }
  255. }, this), this.options.duration);
  256. }
  257. return this;
  258. }
  259. , destroy: function (preserveBackground) {
  260. // Stop the resize events
  261. $(window).off('resize.backstretch orientationchange.backstretch');
  262. // Clear the interval
  263. clearInterval(this.interval);
  264. // Remove Backstretch
  265. if(!preserveBackground) {
  266. this.$wrap.remove();
  267. }
  268. this.$container.removeData('backstretch');
  269. }
  270. };
  271. /* SUPPORTS FIXED POSITION?
  272. *
  273. * Based on code from jQuery Mobile 1.1.0
  274. * http://jquerymobile.com/
  275. *
  276. * In a nutshell, we need to figure out if fixed positioning is supported.
  277. * Unfortunately, this is very difficult to do on iOS, and usually involves
  278. * injecting content, scrolling the page, etc.. It's ugly.
  279. * jQuery Mobile uses this workaround. It's not ideal, but works.
  280. *
  281. * Modified to detect IE6
  282. * ========================= */
  283. var supportsFixedPosition = (function () {
  284. var ua = navigator.userAgent
  285. , platform = navigator.platform
  286. // Rendering engine is Webkit, and capture major version
  287. , wkmatch = ua.match( /AppleWebKit\/([0-9]+)/ )
  288. , wkversion = !!wkmatch && wkmatch[ 1 ]
  289. , ffmatch = ua.match( /Fennec\/([0-9]+)/ )
  290. , ffversion = !!ffmatch && ffmatch[ 1 ]
  291. , operammobilematch = ua.match( /Opera Mobi\/([0-9]+)/ )
  292. , omversion = !!operammobilematch && operammobilematch[ 1 ]
  293. , iematch = ua.match( /MSIE ([0-9]+)/ )
  294. , ieversion = !!iematch && iematch[ 1 ];
  295. return !(
  296. // iOS 4.3 and older : Platform is iPhone/Pad/Touch and Webkit version is less than 534 (ios5)
  297. ((platform.indexOf( "iPhone" ) > -1 || platform.indexOf( "iPad" ) > -1 || platform.indexOf( "iPod" ) > -1 ) && wkversion && wkversion < 534) ||
  298. // Opera Mini
  299. (window.operamini && ({}).toString.call( window.operamini ) === "[object OperaMini]") ||
  300. (operammobilematch && omversion < 7458) ||
  301. //Android lte 2.1: Platform is Android and Webkit version is less than 533 (Android 2.2)
  302. (ua.indexOf( "Android" ) > -1 && wkversion && wkversion < 533) ||
  303. // Firefox Mobile before 6.0 -
  304. (ffversion && ffversion < 6) ||
  305. // WebOS less than 3
  306. ("palmGetResource" in window && wkversion && wkversion < 534) ||
  307. // MeeGo
  308. (ua.indexOf( "MeeGo" ) > -1 && ua.indexOf( "NokiaBrowser/8.5.0" ) > -1) ||
  309. // IE6
  310. (ieversion && ieversion <= 6)
  311. );
  312. }());
  313. }(jQuery, window));