jquery.wookmark.js 8.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261
  1. /*!
  2. jQuery Wookmark plugin
  3. @name jquery.wookmark.js
  4. @author Christoph Ono (chri@sto.ph or @gbks)
  5. @author Sebastian Helzle (sebastian@helzle.net or @sebobo)
  6. @version 1.1.1
  7. @date 4/14/2013
  8. @category jQuery plugin
  9. @copyright (c) 2009-2013 Christoph Ono (www.wookmark.com)
  10. @license Licensed under the MIT (http://www.opensource.org/licenses/mit-license.php) license.
  11. */
  12. (function($){
  13. var Wookmark, defaultOptions, __bind;
  14. __bind = function(fn, me) {
  15. return function() {
  16. return fn.apply(me, arguments);
  17. };
  18. };
  19. // Wookmark default options
  20. defaultOptions = {
  21. align: 'center',
  22. container: $('body'),
  23. offset: 2,
  24. autoResize: false,
  25. itemWidth: 0,
  26. flexibleWidth: 0,
  27. resizeDelay: 50,
  28. onLayoutChanged: undefined
  29. };
  30. Wookmark = (function(options) {
  31. function Wookmark(handler, options) {
  32. this.handler = handler;
  33. // Layout variables.
  34. this.columns = null;
  35. this.containerWidth = null;
  36. this.resizeTimer = null;
  37. this.direction = 'left';
  38. $.extend(true, this, defaultOptions, options);
  39. // Bind methods
  40. this.update = __bind(this.update, this);
  41. this.onResize = __bind(this.onResize, this);
  42. this.getItemWidth = __bind(this.getItemWidth, this);
  43. this.layout = __bind(this.layout, this);
  44. this.layoutFull = __bind(this.layoutFull, this);
  45. this.layoutColumns = __bind(this.layoutColumns, this);
  46. this.clear = __bind(this.clear, this);
  47. // Listen to resize event if requested.
  48. if (this.autoResize) {
  49. $(window).bind('resize.wookmark', this.onResize);
  50. this.container.bind('refreshWookmark', this.onResize);
  51. };
  52. };
  53. // Method for updating the plugins options
  54. Wookmark.prototype.update = function(options) {
  55. $.extend(true, this, options);
  56. };
  57. // This timer ensures that layout is not continuously called as window is being dragged.
  58. Wookmark.prototype.onResize = function() {
  59. clearTimeout(this.resizeTimer);
  60. this.resizeTimer = setTimeout(this.layout, this.resizeDelay);
  61. };
  62. // Method to get the standard item width
  63. Wookmark.prototype.getItemWidth = function() {
  64. if (this.itemWidth === undefined || this.itemWidth === 0) {
  65. return this.handler.eq(0).outerWidth();
  66. }
  67. else if (typeof this.itemWidth == 'string' && this.itemWidth.indexOf('%') >= 0) {
  68. return parseFloat(this.itemWidth) / 100 * this.container.width();
  69. }
  70. return this.itemWidth;
  71. };
  72. // Method to get the flexible item width
  73. Wookmark.prototype.getFlexibleWidth = function() {
  74. var containerWidth = this.container.width(),
  75. flexibleWidth = this.flexibleWidth;
  76. if (typeof flexibleWidth == 'string' && flexibleWidth.indexOf('%') >= 0) {
  77. flexibleWidth = parseFloat(flexibleWidth) / 100 * containerWidth;
  78. flexibleWidth -= this.handler.eq(0).outerWidth() - this.handler.eq(0).innerWidth();
  79. }
  80. var columns = Math.floor(1 + containerWidth / (flexibleWidth + this.offset)),
  81. columnWidth = (containerWidth - (columns - 1) * this.offset) / columns;
  82. return Math.floor(columnWidth);
  83. };
  84. // Main layout methdd.
  85. Wookmark.prototype.layout = function() {
  86. // Do nothing if container isn't visible
  87. if(!this.container.is(":visible")) {
  88. return;
  89. }
  90. // Calculate flexible item width if option is set
  91. if (this.flexibleWidth) {
  92. this.itemWidth = this.getFlexibleWidth();
  93. // Stretch items to fill calculated width
  94. this.handler.css('width', this.itemWidth);
  95. }
  96. // Calculate basic layout parameters.
  97. var columnWidth = this.getItemWidth() + this.offset,
  98. containerWidth = this.container.width(),
  99. columns = Math.floor((containerWidth + this.offset) / columnWidth),
  100. offset = 0,
  101. maxHeight = 0;
  102. // Use less columns if there are to few items
  103. columns = Math.min(columns, this.handler.length);
  104. // Calculate the offset based on the alignment of columns to the parent container
  105. if (this.align == 'left' || this.align == 'right') {
  106. offset = Math.floor((columns / columnWidth + this.offset) / 2);
  107. } else {
  108. offset = Math.round((containerWidth - (columns * columnWidth - this.offset)) / 2);
  109. }
  110. // Get direction for positioning
  111. this.direction = this.align == 'right' ? 'right' : 'left';
  112. // If container and column count hasn't changed, we can only update the columns.
  113. if(this.columns != null && this.columns.length == columns) {
  114. maxHeight = this.layoutColumns(columnWidth, offset);
  115. } else {
  116. maxHeight = this.layoutFull(columnWidth, columns, offset);
  117. }
  118. // Set container height to height of the grid.
  119. this.container.css('height', maxHeight);
  120. if (this.onLayoutChanged !== undefined && typeof this.onLayoutChanged === 'function') {
  121. this.onLayoutChanged();
  122. }
  123. };
  124. /**
  125. * Perform a full layout update.
  126. */
  127. Wookmark.prototype.layoutFull = function(columnWidth, columns, offset) {
  128. var item, top, left, i = 0, k = 0 , j = 0,
  129. length = this.handler.length,
  130. shortest = null, shortestIndex = null,
  131. itemCSS = {position: 'absolute'},
  132. sideOffset, heights = [],
  133. leftAligned = this.align == 'left' ? true : false;
  134. this.columns = [];
  135. // Prepare arrays to store height of columns and items.
  136. while (heights.length < columns) {
  137. heights.push(0);
  138. this.columns.push([]);
  139. }
  140. // Loop over items.
  141. for (; i < length; i++ ) {
  142. item = this.handler.eq(i);
  143. // Find the shortest column.
  144. shortest = heights[0];
  145. shortestIndex = 0;
  146. for (k = 0; k < columns; k++) {
  147. if (heights[k] < shortest) {
  148. shortest = heights[k];
  149. shortestIndex = k;
  150. }
  151. }
  152. // stick to left side if alignment is left and this is the first column
  153. if (shortestIndex == 0 && leftAligned) {
  154. sideOffset = 0;
  155. } else {
  156. sideOffset = shortestIndex * columnWidth + offset;
  157. }
  158. // Position the item.
  159. itemCSS[this.direction] = sideOffset;
  160. itemCSS.top = shortest;
  161. item.css(itemCSS);
  162. // Update column height and store item in shortest column
  163. heights[shortestIndex] += item.outerHeight() + this.offset;
  164. this.columns[shortestIndex].push(item);
  165. }
  166. // Return longest column
  167. return Math.max.apply(Math, heights);
  168. };
  169. /**
  170. * This layout method only updates the vertical position of the
  171. * existing column assignments.
  172. */
  173. Wookmark.prototype.layoutColumns = function(columnWidth, offset) {
  174. var heights = [],
  175. i = 0, k = 0,
  176. column, item, itemCSS, sideOffset;
  177. for (; i < this.columns.length; i++) {
  178. heights.push(0);
  179. column = this.columns[i];
  180. sideOffset = i * columnWidth + offset;
  181. for (k = 0; k < column.length; k++) {
  182. item = column[k];
  183. itemCSS = {
  184. top: heights[i]
  185. };
  186. itemCSS[this.direction] = sideOffset;
  187. item.css(itemCSS);
  188. heights[i] += item.outerHeight() + this.offset;
  189. }
  190. }
  191. // Return longest column
  192. return Math.max.apply(Math, heights);
  193. };
  194. /**
  195. * Clear event listeners and time outs.
  196. */
  197. Wookmark.prototype.clear = function() {
  198. clearTimeout(this.resizeTimer);
  199. $(window).unbind('resize.wookmark', this.onResize);
  200. this.container.unbind('refreshWookmark', this.onResize);
  201. };
  202. return Wookmark;
  203. })();
  204. $.fn.wookmark = function(options) {
  205. // Create a wookmark instance if not available
  206. if (!this.wookmarkInstance) {
  207. this.wookmarkInstance = new Wookmark(this, options || {});
  208. } else {
  209. this.wookmarkInstance.update(options || {});
  210. }
  211. // Apply layout
  212. this.wookmarkInstance.layout();
  213. // Display items (if hidden) and return jQuery object to maintain chainability
  214. return this.show();
  215. };
  216. })(jQuery);