category.js 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167
  1. // @license magnet:?xt=urn:btih:0b31508aeb0634b347b8270c7bee4d411b5d4109&dn=agpl-3.0.txt AGPL-3.0
  2. 'use strict';
  3. /* globals context */
  4. let loading = false;
  5. let dnd_successful = false;
  6. function dragend_process(t) {
  7. t.setAttribute('draggable', 'false');
  8. if (loading) {
  9. setTimeout(function () {
  10. dragend_process(t);
  11. }, 50);
  12. return;
  13. }
  14. if (!dnd_successful) {
  15. t.style.display = '';
  16. t.style.opacity = '';
  17. t.setAttribute('draggable', 'true');
  18. } else {
  19. const p = t.parentElement;
  20. t.remove();
  21. if (p.childElementCount <= 1) {
  22. p.insertAdjacentHTML('afterbegin',
  23. '<li class="item feed disabled emptyCategory"><div class="alert-warn">' + context.i18n.category_empty + '</div></li>');
  24. }
  25. }
  26. }
  27. let dragFeedId = '';
  28. let dragHtml = '';
  29. function init_draggable() {
  30. if (!window.context) {
  31. if (window.console) {
  32. console.log('FreshRSS category waiting for JS…');
  33. }
  34. setTimeout(init_draggable, 50);
  35. return;
  36. }
  37. const draggable = '[draggable="true"]';
  38. const dropzone = '[dropzone="move"]';
  39. const dropSection = document.querySelector('.drop-section');
  40. dropSection.ondragstart = function (ev) {
  41. const li_draggable = ev.target.closest ? ev.target.closest(draggable) : null;
  42. if (li_draggable) {
  43. const ulClosest = li_draggable.closest('ul');
  44. ulClosest.classList.add('drag-disallowed');
  45. ulClosest.removeAttribute('dropzone', '');
  46. const drag = ev.target.closest('[draggable]');
  47. ev.dataTransfer.effectAllowed = 'move';
  48. dragHtml = drag.outerHTML;
  49. dragFeedId = drag.getAttribute('data-feed-id');
  50. ev.dataTransfer.setData('text', dragFeedId);
  51. drag.style.opacity = 0.5;
  52. drag.classList.add('dragging');
  53. li_draggable.closest('.drop-section').classList.add('drag-active');
  54. dnd_successful = false;
  55. }
  56. };
  57. dropSection.ondragend = function (ev) {
  58. const li_draggable = ev.target.closest ? ev.target.closest(draggable) : null;
  59. if (li_draggable) {
  60. dragend_process(li_draggable);
  61. li_draggable.classList.remove('dragging');
  62. const disallowDragging = document.getElementsByClassName('drag-disallowed');
  63. for (let i = 0; i < disallowDragging.length; i++) {
  64. disallowDragging[i].setAttribute('dropzone', 'move');
  65. disallowDragging[i].classList.remove('drag-disallowed');
  66. }
  67. li_draggable.closest('.drag-active').classList.remove('drag-active');
  68. }
  69. };
  70. dropSection.ondragenter = function (ev) {
  71. const ul_dropzone = ev.target.closest ? ev.target.closest(dropzone) : null;
  72. if (ul_dropzone) {
  73. ul_dropzone.classList.add('drag-hover');
  74. return false;
  75. }
  76. };
  77. dropSection.ondragleave = function (ev) {
  78. const ul_dropzone = ev.target.closest ? ev.target.closest(dropzone) : null;
  79. if (ul_dropzone) {
  80. const scroll_top = document.documentElement.scrollTop;
  81. const top = ul_dropzone.offsetTop;
  82. const left = ul_dropzone.offsetLeft;
  83. const right = left + ul_dropzone.clientWidth;
  84. const bottom = top + ul_dropzone.clientHeight;
  85. const mouse_x = ev.screenX;
  86. const mouse_y = ev.clientY + scroll_top;
  87. if (left <= mouse_x && mouse_x <= right &&
  88. top <= mouse_y && mouse_y <= bottom) {
  89. // HACK because dragleave is triggered when hovering children!
  90. return;
  91. }
  92. ul_dropzone.classList.remove('drag-hover');
  93. }
  94. };
  95. dropSection.ondragover = function (ev) {
  96. const li = ev.target.closest ? ev.target.closest(dropzone) : null;
  97. if (li) {
  98. li.closest('ul').classList.remove('drag-drop');
  99. ev.dataTransfer.dropEffect = 'move';
  100. return false;
  101. }
  102. };
  103. dropSection.ondrop = function (ev) {
  104. if (dragFeedId) {
  105. const ul_dropzone = ev.target.closest ? ev.target.closest(dropzone) : null;
  106. if (ul_dropzone) {
  107. loading = true;
  108. const req = new XMLHttpRequest();
  109. req.open('POST', './?c=feed&a=move', true);
  110. req.responseType = 'json';
  111. req.onload = function (e) {
  112. if (this.status == 200) {
  113. ul_dropzone.insertAdjacentHTML('afterbegin', dragHtml);
  114. ul_dropzone.firstChild.classList.add('moved');
  115. ul_dropzone.scrollTop = 0;
  116. const disabledElement = ul_dropzone.getElementsByClassName('disabled');
  117. if (disabledElement.length > 0) {
  118. disabledElement[0].remove();
  119. }
  120. dnd_successful = true;
  121. ul_dropzone.closest('ul').classList.add('drag-drop');
  122. }
  123. };
  124. req.onloadend = function (e) {
  125. loading = false;
  126. dragFeedId = '';
  127. dragHtml = '';
  128. };
  129. req.setRequestHeader('Content-Type', 'application/json; charset=utf-8');
  130. req.send(JSON.stringify({
  131. f_id: dragFeedId,
  132. c_id: ul_dropzone.getAttribute('data-cat-id'),
  133. _csrf: context.csrf,
  134. }));
  135. ul_dropzone.closest('ul').classList.remove('drag-hover');
  136. return false;
  137. }
  138. }
  139. };
  140. }
  141. if (document.readyState && document.readyState !== 'loading') {
  142. init_draggable();
  143. } else if (document.addEventListener) {
  144. document.addEventListener('DOMContentLoaded', function () {
  145. init_draggable();
  146. }, false);
  147. }
  148. // @license-end