index.js 6.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219
  1. import $ from 'jquery';
  2. import { config, uri_params } from 'grav-config';
  3. import request from '../utils/request';
  4. export default class Filter {
  5. constructor() {
  6. this.URI = `${config.base_url_relative}/media-manager/`;
  7. }
  8. filter(name, value) {
  9. let filtered = [];
  10. let keys = Object.keys(uri_params);
  11. if (!~keys.indexOf(name)) { keys.push(name); }
  12. keys.forEach((key) => {
  13. let filter = Filter.cleanValue(key === name ? value : uri_params[key]);
  14. if (filter !== '*') {
  15. filtered.push(`${key}${config.param_sep}${filter}`);
  16. }
  17. });
  18. global.location = this.URI + filtered.join('/');
  19. }
  20. static cleanValue(value) {
  21. return encodeURIComponent(value.replace('/', '\\'));
  22. }
  23. }
  24. export let Instance = new Filter();
  25. var isLoading = false;
  26. var filters = {};
  27. var global_index = 1;
  28. var files_ended = false;
  29. const MEDIA_PAGINATION_INTERVAL = 20;
  30. /* handle changing file type / date filter */
  31. $('body').on('change', '.thumbs-list-container select.filter', (event) => {
  32. let target = $(event.currentTarget);
  33. let filterName = target.data('name');
  34. let filterValue = target.val();
  35. if (filterValue) {
  36. filters[filterName] = filterValue;
  37. } else {
  38. delete filters[filterName];
  39. }
  40. filterFiles();
  41. });
  42. /* initialize media uploader */
  43. if ($('.thumbs-list-container .dropzone')[0]) {
  44. $('.thumbs-list-container .dropzone')[0].dropzone.on('queuecomplete', function() {
  45. let body = {};
  46. if (filters.page) { body.page = filters.page; }
  47. if (filters.date) { body.date = filters.date; }
  48. if (filters.type) { body.type = filters.type; }
  49. $('.dropzone')[0].dropzone.files.forEach(function(file) { file.previewElement.remove(); });
  50. $('.dropzone').first().removeClass('dz-started');
  51. request(config.base_url_relative + '/media-manager.json/task:clearMediaCache', { method: 'post', body }, () => {
  52. filterFiles();
  53. });
  54. });
  55. }
  56. /* handle loading media */
  57. var loadMedia = function loadMedia(filters, callback) {
  58. var url = config.base_url_relative + '/media.json/tmpl:media-list-content/index:' + global_index;
  59. if (filters.page) {
  60. url += '/page:' + (filters.page).split('/').join('%5C');
  61. }
  62. if (filters.type && filters.type !== '*') {
  63. url += '/type:' + filters.type;
  64. }
  65. if (filters.date && filters.date !== '*') {
  66. url += '/date:' + filters.date;
  67. }
  68. if (!isLoading) {
  69. isLoading = true;
  70. $('.spinning-wheel').show();
  71. $.get(url, function(content) {
  72. $('.js__files').append(content);
  73. $('.spinning-wheel').hide();
  74. isLoading = false;
  75. global_index++;
  76. callback(content);
  77. });
  78. }
  79. };
  80. var cleanFilesList = function cleanFilesList() {
  81. $('.js__files .card-item').remove();
  82. };
  83. var resetActiveStateInSidebar = function resetActiveStateInSidebar() {
  84. $('.pages-list-container .row').removeClass('active'); // clear active state in sidebar
  85. };
  86. var showEmptyState = function showEmptyState() {
  87. $('.thumbs-list-container').append('<p class="card-item empty-space">No media found</p>');
  88. };
  89. var filterFiles = function filterFiles() {
  90. cleanFilesList();
  91. global_index = 0;
  92. files_ended = false;
  93. $('.empty-space').remove();
  94. loadMedia(filters, function(content) {
  95. if (!$(content).length) {
  96. showEmptyState();
  97. } else {
  98. if (!filters.page && (!filters.date || filters.date === '*') && (!filters.type || filters.type === '*')) {
  99. $('.js__files').trigger('fillView');
  100. }
  101. }
  102. });
  103. };
  104. /* handle changing page */
  105. $('body').on('click', '.pages-list-container .js__page-link', (event) => {
  106. var page = $(event.target).data('page');
  107. filters['page'] = page;
  108. $('.media-list-title .page-indicator').html(page); // set indication
  109. $('.js__reset-pages-filter').removeClass('hidden'); // activate reset pages icon
  110. resetActiveStateInSidebar();
  111. $(event.target).parents('.row').addClass('active'); // set active state in sidebar
  112. $('.js__file-uploader').removeClass('hidden');
  113. // customize processing URL, as the page changes dynamically
  114. if ($('.dropzone')[0]) {
  115. $('.dropzone')[0].dropzone.on('processing', function(file) {
  116. this.options.url = `${config.base_url_relative}/media-manager${page}.json/task${config.param_sep}addmedia`;
  117. });
  118. }
  119. $('.js__button-clear-media-cache').addClass('hidden');
  120. filterFiles();
  121. disableInfiniteScrolling(); // only infinite scroll on main list, not inside single pages
  122. });
  123. /* handle clearing page filter */
  124. $('body').on('click', '.js__reset-pages-filter', (event) => {
  125. $('.media-list-title .page-indicator').html('All Pages'); // set indication
  126. cleanFilesList();
  127. resetActiveStateInSidebar();
  128. $('.js__reset-pages-filter').addClass('hidden'); // remove reset pages icon
  129. $('.js__file-uploader').addClass('hidden');
  130. $('.js__button-clear-media-cache').removeClass('hidden');
  131. delete filters['page'];
  132. filterFiles();
  133. });
  134. /* handle infinite loading */
  135. var enableInfiniteScrolling = function enableInfiniteScrolling() {
  136. $('.spinning-wheel').hide();
  137. var view = $('.mediapicker-scroll');
  138. view.scroll(function() {
  139. if (($(this).scrollTop() + $(this).innerHeight() + 100) >= $(this)[0].scrollHeight) {
  140. fillView();
  141. }
  142. });
  143. };
  144. var loadNextBatch = function loadNextBatch(callback) {
  145. if (files_ended) {
  146. return;
  147. }
  148. loadMedia({}, function(content) {
  149. if (!$(content).length || ((content.split('card-item').length - 1) < MEDIA_PAGINATION_INTERVAL)) {
  150. files_ended = true;
  151. } else {
  152. if (callback) {
  153. callback();
  154. }
  155. }
  156. });
  157. };
  158. var fillView = function fillView() {
  159. if (!$('.js__files').find('.card-item').last().offset()) {
  160. setTimeout(function() {
  161. // retry later
  162. fillView();
  163. }, 300);
  164. return;
  165. }
  166. if ($('.js__files').find('.card-item').last().offset().top < $('.media-container').height()) {
  167. loadNextBatch(function() {
  168. fillView();
  169. });
  170. }
  171. };
  172. /* disable infinite loading */
  173. var disableInfiniteScrolling = function disableInfiniteScrolling() {
  174. $('.spinning-wheel').hide();
  175. $('.content-wrapper').unbind('scroll');
  176. };
  177. $('.js__files').on('fillView', function(event) {
  178. // the first batch got the max number of media files, try loading more
  179. if (($('.js__files')[0].innerHTML.split('card-item').length - 1) === MEDIA_PAGINATION_INTERVAL) {
  180. fillView();
  181. enableInfiniteScrolling();
  182. }
  183. });