swipe.js 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480
  1. /*
  2. * Swipe 2.0
  3. *
  4. * Brad Birdsall
  5. * Copyright 2013, MIT License
  6. *
  7. */
  8. function Swipe(container, options) {
  9. "use strict";
  10. // utilities
  11. var noop = function() {}; // 简单的无操作功能
  12. var offloadFn = function(fn) { setTimeout(fn || noop, 0) }; // 卸载功能的执行
  13. // 检查浏览器的功能
  14. var browser = {
  15. addEventListener: !!window.addEventListener,
  16. touch: ('ontouchstart' in window) || window.DocumentTouch && document instanceof DocumentTouch,
  17. transitions: (function(temp) {
  18. var props = ['transformProperty', 'WebkitTransform', 'MozTransform', 'OTransform', 'msTransform'];
  19. for ( var i in props ) if (temp.style[ props[i] ] !== undefined) return true;
  20. return false;
  21. })(document.createElement('swipe'))
  22. };
  23. // 如果没有根元素退出
  24. if (!container) return;
  25. var element = container.children[0];
  26. var slides, slidePos, width;
  27. options = options || {};
  28. var index = parseInt(options.startSlide, 10) || 0;
  29. var speed = options.speed || 300;
  30. options.continuous = options.continuous ? options.continuous : true;
  31. function setup() {
  32. // 缓存的幻灯片
  33. slides = element.children;
  34. //创建一个数组来存储每个幻灯片的当前位置
  35. slidePos = new Array(slides.length);
  36. // 确定每个幻灯片的宽度
  37. width = container.getBoundingClientRect().width || container.offsetWidth;
  38. element.style.width = (slides.length * width) + 'px';
  39. // 栈元素
  40. var pos = slides.length;
  41. while(pos--) {
  42. var slide = slides[pos];
  43. slide.style.width = width + 'px';
  44. slide.setAttribute('data-index', pos);
  45. if (browser.transitions) {
  46. slide.style.left = (pos * -width) + 'px';
  47. move(pos, index > pos ? -width : (index < pos ? width : 0), 0);
  48. }
  49. }
  50. if (!browser.transitions) element.style.left = (index * -width) + 'px';
  51. container.style.visibility = 'visible';
  52. }
  53. function prev() {
  54. if (index) slide(index-1);
  55. else if (options.continuous) slide(slides.length-1);
  56. }
  57. function next() {
  58. if (index < slides.length - 1) slide(index+1);
  59. else if (options.continuous) slide(0);
  60. }
  61. function slide(to, slideSpeed) {
  62. // 如果已经滑不要求
  63. if (index == to) return;
  64. if (browser.transitions) {
  65. var diff = Math.abs(index-to) - 1;
  66. var direction = Math.abs(index-to) / (index-to); // 1:right -1:left
  67. while (diff--) move((to > index ? to : index) - diff - 1, width * direction, 0);
  68. move(index, width * direction, slideSpeed || speed);
  69. move(to, 0, slideSpeed || speed);
  70. } else {
  71. animate(index * -width, to * -width, slideSpeed || speed);
  72. }
  73. index = to;
  74. offloadFn(options.callback && options.callback(index, slides[index]));
  75. }
  76. function move(index, dist, speed) {
  77. translate(index, dist, speed);
  78. slidePos[index] = dist;
  79. }
  80. function translate(index, dist, speed) {
  81. var slide = slides[index];
  82. var style = slide && slide.style;
  83. if (!style) return;
  84. style.webkitTransitionDuration =
  85. style.MozTransitionDuration =
  86. style.msTransitionDuration =
  87. style.OTransitionDuration =
  88. style.transitionDuration = speed + 'ms';
  89. style.webkitTransform = 'translate(' + dist + 'px,0)' + 'translateZ(0)';
  90. style.msTransform =
  91. style.MozTransform =
  92. style.OTransform = 'translateX(' + dist + 'px)';
  93. }
  94. function animate(from, to, speed) {
  95. // 如果不是动画,只是重新定位
  96. if (!speed) {
  97. element.style.left = to + 'px';
  98. return;
  99. }
  100. var start = +new Date;
  101. var timer = setInterval(function() {
  102. var timeElap = +new Date - start;
  103. if (timeElap > speed) {
  104. element.style.left = to + 'px';
  105. if (delay) begin();
  106. options.transitionEnd && options.transitionEnd.call(event, index, slides[index]);
  107. clearInterval(timer);
  108. return;
  109. }
  110. element.style.left = (( (to - from) * (Math.floor((timeElap / speed) * 100) / 100) ) + from) + 'px';
  111. }, 4);
  112. }
  113. // 安装程序自动幻灯片
  114. var delay = options.auto || 0;
  115. var interval;
  116. function begin() {
  117. interval = setTimeout(next, delay);
  118. }
  119. function stop() {
  120. delay = 0;
  121. clearTimeout(interval);
  122. }
  123. // 设置初始变量
  124. var start = {};
  125. var delta = {};
  126. var isScrolling;
  127. // 设置事件捕获
  128. var events = {
  129. handleEvent: function(event) {
  130. switch (event.type) {
  131. case 'touchstart': this.start(event); break;
  132. case 'touchmove': this.move(event); break;
  133. case 'touchend': offloadFn(this.end(event)); break;
  134. case 'webkitTransitionEnd':
  135. case 'msTransitionEnd':
  136. case 'oTransitionEnd':
  137. case 'otransitionend':
  138. case 'transitionend': offloadFn(this.transitionEnd(event)); break;
  139. case 'resize': offloadFn(setup.call()); break;
  140. }
  141. if (options.stopPropagation) event.stopPropagation();
  142. },
  143. start: function(event) {
  144. var touches = event.touches[0];
  145. // 测量的起始值
  146. start = {
  147. // 得到初始的触摸坐标
  148. x: touches.pageX,
  149. y: touches.pageY,
  150. // 存储时间确定接触时间
  151. time: +new Date
  152. };
  153. // 用于测试的第一移动事件
  154. isScrolling = undefined;
  155. // 复位三角洲和最后计算值
  156. delta = {};
  157. // 设置touchmove和touchend监听
  158. element.addEventListener('touchmove', this, false);
  159. element.addEventListener('touchend', this, false);
  160. },
  161. move: function(event) {
  162. // 确保一个触摸不捏刷
  163. if ( event.touches.length > 1 || event.scale && event.scale !== 1) return
  164. if (options.disableScroll) event.preventDefault();
  165. var touches = event.touches[0];
  166. // 计算改变后的 x 和 y
  167. delta = {
  168. x: touches.pageX - start.x,
  169. y: touches.pageY - start.y
  170. }
  171. // 确定测试运行——一个滚动时间测试
  172. if ( typeof isScrolling == 'undefined') {
  173. isScrolling = !!( isScrolling || Math.abs(delta.x) < Math.abs(delta.y) );
  174. }
  175. // 如果用户没有试图垂直滚动
  176. if (!isScrolling) {
  177. // 防止本机滚动
  178. event.preventDefault();
  179. // 停止幻灯片显示
  180. stop();
  181. // 如果第一个或最后一个滑动阻力增加
  182. delta.x =
  183. delta.x /
  184. ( (!index && delta.x > 0 // if first slide and sliding left
  185. || index == slides.length - 1 // or if last slide and sliding right
  186. && delta.x < 0 // and if sliding at all
  187. ) ?
  188. ( Math.abs(delta.x) / width + 1 ) // determine resistance level
  189. : 1 ); // no resistance if false
  190. // 转化 1:1
  191. translate(index-1, delta.x + slidePos[index-1], 0);
  192. translate(index, delta.x + slidePos[index], 0);
  193. translate(index+1, delta.x + slidePos[index+1], 0);
  194. }
  195. },
  196. end: function(event) {
  197. // 计算持续时间
  198. var duration = +new Date - start.time;
  199. // 确定滑动尝试触发下一个/上一页滑动
  200. var isValidSlide =
  201. Number(duration) < 250 // if slide duration is less than 250ms
  202. && Math.abs(delta.x) > 20 // and if slide amt is greater than 20px
  203. || Math.abs(delta.x) > width/2; // or if slide amt is greater than half the width
  204. // 如果尝试确定滑过去的开始和结束
  205. var isPastBounds =
  206. !index && delta.x > 0 // 如果第一个幻灯片和幻灯片AMT大于0
  207. || index == slides.length - 1 && delta.x < 0; // 或者如果最后一张幻灯片,幻灯片amt小于0
  208. // 确定滑动方向(true:right, false:left)
  209. var direction = delta.x < 0;
  210. // 如果不垂直滚动
  211. if (!isScrolling) {
  212. if (isValidSlide && !isPastBounds) {
  213. if (direction) {
  214. move(index-1, -width, 0);
  215. move(index, slidePos[index]-width, speed);
  216. move(index+1, slidePos[index+1]-width, speed);
  217. index += 1;
  218. } else {
  219. move(index+1, width, 0);
  220. move(index, slidePos[index]+width, speed);
  221. move(index-1, slidePos[index-1]+width, speed);
  222. index += -1;
  223. }
  224. options.callback && options.callback(index, slides[index]);
  225. } else {
  226. move(index-1, -width, speed);
  227. move(index, 0, speed);
  228. move(index+1, width, speed);
  229. }
  230. }
  231. // 取消touchmove和touchend事件监听器,直到touchstart再次调用
  232. element.removeEventListener('touchmove', events, false)
  233. element.removeEventListener('touchend', events, false)
  234. },
  235. transitionEnd: function(event) {
  236. if (parseInt(event.target.getAttribute('data-index'), 10) == index) {
  237. if (delay) begin();
  238. options.transitionEnd && options.transitionEnd.call(event, index, slides[index]);
  239. }
  240. }
  241. }
  242. // 触发设置
  243. setup();
  244. // 如果适用则开始自动幻灯片
  245. if (delay) begin();
  246. // 添加事件监听器
  247. if (browser.addEventListener) {
  248. // 设置touchstart事件元素
  249. if (browser.touch) element.addEventListener('touchstart', events, false);
  250. if (browser.transitions) {
  251. element.addEventListener('webkitTransitionEnd', events, false);
  252. element.addEventListener('msTransitionEnd', events, false);
  253. element.addEventListener('oTransitionEnd', events, false);
  254. element.addEventListener('otransitionend', events, false);
  255. element.addEventListener('transitionend', events, false);
  256. }
  257. //设置在窗口调整大小事件
  258. window.addEventListener('resize', events, false);
  259. } else {
  260. window.onresize = function () { setup() }; // to play nice with old IE
  261. }
  262. // 公开Swipe API
  263. return {
  264. setup: function() {
  265. setup();
  266. },
  267. slide: function(to, speed) {
  268. slide(to, speed);
  269. },
  270. prev: function() {
  271. // cancel slideshow
  272. stop();
  273. prev();
  274. },
  275. next: function() {
  276. stop();
  277. next();
  278. },
  279. getPos: function() {
  280. // return current index position
  281. return index;
  282. },
  283. kill: function() {
  284. // 取消幻灯片
  285. stop();
  286. // reset element
  287. element.style.width = 'auto';
  288. element.style.left = 0;
  289. // reset slides
  290. var pos = slides.length;
  291. while(pos--) {
  292. var slide = slides[pos];
  293. slide.style.width = '100%';
  294. slide.style.left = 0;
  295. if (browser.transitions) translate(pos, 0, 0);
  296. }
  297. // 删除事件侦听器
  298. if (browser.addEventListener) {
  299. // remove current event listeners
  300. element.removeEventListener('touchstart', events, false);
  301. element.removeEventListener('webkitTransitionEnd', events, false);
  302. element.removeEventListener('msTransitionEnd', events, false);
  303. element.removeEventListener('oTransitionEnd', events, false);
  304. element.removeEventListener('otransitionend', events, false);
  305. element.removeEventListener('transitionend', events, false);
  306. window.removeEventListener('resize', events, false);
  307. }
  308. else {
  309. window.onresize = null;
  310. }
  311. }
  312. }
  313. }
  314. if ( window.jQuery || window.Zepto ) {
  315. (function($) {
  316. $.fn.Swipe = function(params) {
  317. return this.each(function() {
  318. $(this).data('Swipe', new Swipe($(this)[0], params));
  319. });
  320. }
  321. })( window.jQuery || window.Zepto )
  322. }