Não pode escolher mais do que 25 tópicos Os tópicos devem começar com uma letra ou um número, podem incluir traços ('-') e podem ter até 35 caracteres.
 
 
 
 
 
 

498 linhas
18 KiB

  1. /*! Copyright (c) 2011 Piotr Rochala (http://rocha.la)
  2. * Dual licensed under the MIT (http://www.opensource.org/licenses/mit-license.php)
  3. * and GPL (http://www.opensource.org/licenses/gpl-license.php) licenses.
  4. *
  5. * Improved by keenthemes for Metronic Theme
  6. * Version: 1.3.2
  7. *
  8. */
  9. (function($) {
  10. jQuery.fn.extend({
  11. slimScroll: function(options) {
  12. var defaults = {
  13. // width in pixels of the visible scroll area
  14. width: 'auto',
  15. // height in pixels of the visible scroll area
  16. height: '250px',
  17. // width in pixels of the scrollbar and rail
  18. size: '7px',
  19. // scrollbar color, accepts any hex/color value
  20. color: '#000',
  21. // scrollbar position - left/right
  22. position: 'right',
  23. // distance in pixels between the side edge and the scrollbar
  24. distance: '1px',
  25. // default scroll position on load - top / bottom / $('selector')
  26. start: 'top',
  27. // sets scrollbar opacity
  28. opacity: .4,
  29. // enables always-on mode for the scrollbar
  30. alwaysVisible: false,
  31. // check if we should hide the scrollbar when user is hovering over
  32. disableFadeOut: false,
  33. // sets visibility of the rail
  34. railVisible: false,
  35. // sets rail color
  36. railColor: '#333',
  37. // sets rail opacity
  38. railOpacity: .2,
  39. // whether we should use jQuery UI Draggable to enable bar dragging
  40. railDraggable: true,
  41. // defautlt CSS class of the slimscroll rail
  42. railClass: 'slimScrollRail',
  43. // defautlt CSS class of the slimscroll bar
  44. barClass: 'slimScrollBar',
  45. // defautlt CSS class of the slimscroll wrapper
  46. wrapperClass: 'slimScrollDiv',
  47. // check if mousewheel should scroll the window if we reach top/bottom
  48. allowPageScroll: false,
  49. // scroll amount applied to each mouse wheel step
  50. wheelStep: 20,
  51. // scroll amount applied when user is using gestures
  52. touchScrollStep: 200,
  53. // sets border radius
  54. borderRadius: '7px',
  55. // sets border radius of the rail
  56. railBorderRadius: '7px',
  57. // sets animation status on a given scroll(added my keenthemes)
  58. animate: true
  59. };
  60. var o = $.extend(defaults, options);
  61. // do it for every element that matches selector
  62. this.each(function() {
  63. var isOverPanel, isOverBar, isDragg, queueHide, touchDif,
  64. barHeight, percentScroll, lastScroll,
  65. divS = '<div></div>',
  66. minBarHeight = 30,
  67. releaseScroll = false;
  68. // used in event handlers and for better minification
  69. var me = $(this);
  70. //begin: windows phone fix added by keenthemes
  71. if ('ontouchstart' in window && window.navigator.msPointerEnabled) {
  72. me.css("-ms-touch-action", "none");
  73. }
  74. //end: windows phone fix added by keenthemes
  75. // ensure we are not binding it again
  76. if (me.parent().hasClass(o.wrapperClass)) {
  77. // start from last bar position
  78. var offset = me.scrollTop();
  79. // find bar and rail
  80. bar = me.parent().find('.' + o.barClass);
  81. rail = me.parent().find('.' + o.railClass);
  82. getBarHeight();
  83. // check if we should scroll existing instance
  84. if ($.isPlainObject(options)) {
  85. // Pass height: auto to an existing slimscroll object to force a resize after contents have changed
  86. if ('height' in options && options.height == 'auto') {
  87. me.parent().css('height', 'auto');
  88. me.css('height', 'auto');
  89. var height = me.parent().parent().height();
  90. me.parent().css('height', height);
  91. me.css('height', height);
  92. }
  93. if ('scrollTo' in options) {
  94. // jump to a static point
  95. offset = parseInt(o.scrollTo);
  96. } else if ('scrollBy' in options) {
  97. // jump by value pixels
  98. offset += parseInt(o.scrollBy);
  99. } else if ('destroy' in options) {
  100. // remove slimscroll elements
  101. bar.remove();
  102. rail.remove();
  103. me.unwrap();
  104. return;
  105. }
  106. // scroll content by the given offset
  107. scrollContent(offset, false, true);
  108. }
  109. return;
  110. }
  111. // optionally set height to the parent's height
  112. o.height = (options.height == 'auto') ? me.parent().height() : options.height;
  113. // wrap content
  114. var wrapper = $(divS)
  115. .addClass(o.wrapperClass)
  116. .css({
  117. position: 'relative',
  118. overflow: 'hidden',
  119. width: o.width,
  120. height: o.height
  121. });
  122. // update style for the div
  123. me.css({
  124. overflow: 'hidden',
  125. width: o.width,
  126. height: o.height
  127. });
  128. // create scrollbar rail
  129. var rail = $(divS)
  130. .addClass(o.railClass)
  131. .css({
  132. width: o.size,
  133. height: '100%',
  134. position: 'absolute',
  135. top: 0,
  136. display: (o.alwaysVisible && o.railVisible) ? 'block' : 'none',
  137. 'border-radius': o.railBorderRadius,
  138. background: o.railColor,
  139. opacity: o.railOpacity,
  140. zIndex: 90
  141. });
  142. // create scrollbar
  143. var bar = $(divS)
  144. .addClass(o.barClass)
  145. .css({
  146. background: o.color,
  147. width: o.size,
  148. position: 'absolute',
  149. top: 0,
  150. opacity: o.opacity,
  151. display: o.alwaysVisible ? 'block' : 'none',
  152. 'border-radius': o.borderRadius,
  153. BorderRadius: o.borderRadius,
  154. MozBorderRadius: o.borderRadius,
  155. WebkitBorderRadius: o.borderRadius,
  156. zIndex: 99
  157. });
  158. // set position
  159. var posCss = (o.position == 'right') ? {
  160. right: o.distance
  161. } : {
  162. left: o.distance
  163. };
  164. rail.css(posCss);
  165. bar.css(posCss);
  166. // wrap it
  167. me.wrap(wrapper);
  168. // append to parent div
  169. me.parent().append(bar);
  170. me.parent().append(rail);
  171. // make it draggable and no longer dependent on the jqueryUI
  172. if (o.railDraggable) {
  173. bar.bind("mousedown", function(e) {
  174. var $doc = $(document);
  175. isDragg = true;
  176. t = parseFloat(bar.css('top'));
  177. pageY = e.pageY;
  178. $doc.bind("mousemove.slimscroll", function(e) {
  179. currTop = t + e.pageY - pageY;
  180. bar.css('top', currTop);
  181. scrollContent(0, bar.position().top, false); // scroll content
  182. });
  183. $doc.bind("mouseup.slimscroll", function(e) {
  184. isDragg = false;
  185. hideBar();
  186. $doc.unbind('.slimscroll');
  187. });
  188. return false;
  189. }).bind("selectstart.slimscroll", function(e) {
  190. e.stopPropagation();
  191. e.preventDefault();
  192. return false;
  193. });
  194. }
  195. //begin: windows phone fix added by keenthemes
  196. if ('ontouchstart' in window && window.navigator.msPointerEnabled) {
  197. me.bind('MSPointerDown', function(e, b) {
  198. // record where touch started
  199. touchDif = e.originalEvent.pageY;
  200. });
  201. me.bind('MSPointerMove', function(e) {
  202. // prevent scrolling the page if necessary
  203. e.originalEvent.preventDefault();
  204. // see how far user swiped
  205. var diff = (touchDif - e.originalEvent.pageY) / o.touchScrollStep;
  206. // scroll content
  207. scrollContent(diff, true);
  208. touchDif = e.originalEvent.pageY;
  209. });
  210. }
  211. //end: windows phone fix added by keenthemes
  212. // on rail over
  213. rail.hover(function() {
  214. showBar();
  215. }, function() {
  216. hideBar();
  217. });
  218. // on bar over
  219. bar.hover(function() {
  220. isOverBar = true;
  221. }, function() {
  222. isOverBar = false;
  223. });
  224. // show on parent mouseover
  225. me.hover(function() {
  226. isOverPanel = true;
  227. showBar();
  228. hideBar();
  229. }, function() {
  230. isOverPanel = false;
  231. hideBar();
  232. });
  233. // support for mobile
  234. me.bind('touchstart', function(e, b) {
  235. if (e.originalEvent.touches.length) {
  236. // record where touch started
  237. touchDif = e.originalEvent.touches[0].pageY;
  238. }
  239. });
  240. me.bind('touchmove', function(e) {
  241. // prevent scrolling the page if necessary
  242. if (!releaseScroll) {
  243. e.originalEvent.preventDefault();
  244. }
  245. if (e.originalEvent.touches.length) {
  246. // see how far user swiped
  247. var diff = (touchDif - e.originalEvent.touches[0].pageY) / o.touchScrollStep;
  248. // scroll content
  249. scrollContent(diff, true);
  250. touchDif = e.originalEvent.touches[0].pageY;
  251. }
  252. });
  253. // set up initial height
  254. getBarHeight();
  255. // check start position
  256. if (o.start === 'bottom') {
  257. // scroll content to bottom
  258. bar.css({
  259. top: me.outerHeight() - bar.outerHeight()
  260. });
  261. scrollContent(0, true);
  262. } else if (o.start !== 'top') {
  263. // assume jQuery selector
  264. scrollContent($(o.start).position().top, null, true);
  265. // make sure bar stays hidden
  266. if (!o.alwaysVisible) {
  267. bar.hide();
  268. }
  269. }
  270. // attach scroll events
  271. attachWheel();
  272. function _onWheel(e) {
  273. // use mouse wheel only when mouse is over
  274. if (!isOverPanel) {
  275. return;
  276. }
  277. var e = e || window.event;
  278. var delta = 0;
  279. if (e.wheelDelta) {
  280. delta = -e.wheelDelta / 120;
  281. }
  282. if (e.detail) {
  283. delta = e.detail / 3;
  284. }
  285. var target = e.target || e.srcTarget || e.srcElement;
  286. if ($(target).closest('.' + o.wrapperClass).is(me.parent())) {
  287. // scroll content
  288. scrollContent(delta, true);
  289. }
  290. // stop window scroll
  291. if (e.preventDefault && !releaseScroll) {
  292. e.preventDefault();
  293. }
  294. if (!releaseScroll) {
  295. e.returnValue = false;
  296. }
  297. }
  298. function scrollContent(y, isWheel, isJump) {
  299. releaseScroll = false;
  300. var delta = y;
  301. var maxTop = me.outerHeight() - bar.outerHeight();
  302. if (isWheel) {
  303. // move bar with mouse wheel
  304. delta = parseInt(bar.css('top')) + y * parseInt(o.wheelStep) / 100 * bar.outerHeight();
  305. // move bar, make sure it doesn't go out
  306. delta = Math.min(Math.max(delta, 0), maxTop);
  307. // if scrolling down, make sure a fractional change to the
  308. // scroll position isn't rounded away when the scrollbar's CSS is set
  309. // this flooring of delta would happened automatically when
  310. // bar.css is set below, but we floor here for clarity
  311. delta = (y > 0) ? Math.ceil(delta) : Math.floor(delta);
  312. // scroll the scrollbar
  313. bar.css({
  314. top: delta + 'px'
  315. });
  316. }
  317. // calculate actual scroll amount
  318. percentScroll = parseInt(bar.css('top')) / (me.outerHeight() - bar.outerHeight());
  319. delta = percentScroll * (me[0].scrollHeight - me.outerHeight());
  320. if (isJump) {
  321. delta = y;
  322. var offsetTop = delta / me[0].scrollHeight * me.outerHeight();
  323. offsetTop = Math.min(Math.max(offsetTop, 0), maxTop);
  324. bar.css({
  325. top: offsetTop + 'px'
  326. });
  327. }
  328. // scroll content
  329. if ('scrollTo' in o && o.animate) {
  330. me.animate({
  331. scrollTop: delta
  332. });
  333. } else {
  334. me.scrollTop(delta);
  335. }
  336. // fire scrolling event
  337. me.trigger('slimscrolling', ~~delta);
  338. // ensure bar is visible
  339. showBar();
  340. // trigger hide when scroll is stopped
  341. hideBar();
  342. }
  343. function attachWheel() {
  344. if (window.addEventListener) {
  345. this.addEventListener('DOMMouseScroll', _onWheel, false);
  346. this.addEventListener('mousewheel', _onWheel, false);
  347. } else {
  348. document.attachEvent("onmousewheel", _onWheel)
  349. }
  350. }
  351. function getBarHeight() {
  352. // calculate scrollbar height and make sure it is not too small
  353. barHeight = Math.max((me.outerHeight() / me[0].scrollHeight) * me.outerHeight(), minBarHeight);
  354. bar.css({
  355. height: barHeight + 'px'
  356. });
  357. // hide scrollbar if content is not long enough
  358. var display = barHeight == me.outerHeight() ? 'none' : 'block';
  359. bar.css({
  360. display: display
  361. });
  362. }
  363. function showBar() {
  364. // recalculate bar height
  365. getBarHeight();
  366. clearTimeout(queueHide);
  367. // when bar reached top or bottom
  368. if (percentScroll == ~~percentScroll) {
  369. //release wheel
  370. releaseScroll = o.allowPageScroll;
  371. // publish approporiate event
  372. if (lastScroll != percentScroll) {
  373. var msg = (~~percentScroll == 0) ? 'top' : 'bottom';
  374. me.trigger('slimscroll', msg);
  375. }
  376. } else {
  377. releaseScroll = false;
  378. }
  379. lastScroll = percentScroll;
  380. // show only when required
  381. if (barHeight >= me.outerHeight()) {
  382. //allow window scroll
  383. releaseScroll = true;
  384. return;
  385. }
  386. bar.stop(true, true).fadeIn('fast');
  387. if (o.railVisible) {
  388. rail.stop(true, true).fadeIn('fast');
  389. }
  390. }
  391. function hideBar() {
  392. // only hide when options allow it
  393. if (!o.alwaysVisible) {
  394. queueHide = setTimeout(function() {
  395. if (!(o.disableFadeOut && isOverPanel) && !isOverBar && !isDragg) {
  396. bar.fadeOut('slow');
  397. rail.fadeOut('slow');
  398. }
  399. }, 1000);
  400. }
  401. }
  402. });
  403. // maintain chainability
  404. return this;
  405. }
  406. });
  407. jQuery.fn.extend({
  408. slimscroll: jQuery.fn.slimScroll
  409. });
  410. })(jQuery);