notice-editor.js 8.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327
  1. let NoticeEditor = (function () {
  2. let module = {};
  3. let __ButtonAddNotice = document.querySelector(".notice-add");
  4. if (__ButtonAddNotice) {
  5. __ButtonAddNotice.onclick = addNoticeModal;
  6. }
  7. let __SelectedManipulator = undefined;
  8. /**
  9. * Собрать все данные и отправить на БД
  10. *
  11. * @param {HTMLInputElement} caption
  12. * @param {HTMLInputElement} link
  13. * @param {HTMLDivElement} content
  14. * @param {HTMLDivElement} images
  15. * @param {HTMLInputElement} file
  16. */
  17. function create(caption, link, content, images, file) {
  18. if (!caption || !link || !content || !images) {
  19. Messenger.Show("Некоторые элементы не были найдены, невозможно создать новость!");
  20. return;
  21. }
  22. if (caption.value.length == 0) {
  23. Messenger.Show("Необходимо заполнить заголовок!");
  24. return;
  25. }
  26. if (link.value.length == 0) {
  27. link.value = Main.GenerateHash256();
  28. }
  29. let preview = {
  30. Valid: true,
  31. String: "",
  32. };
  33. let contentHtml = "";
  34. for (const c of content.blocks) {
  35. switch (c.type) {
  36. case "paragraph":
  37. contentHtml += "<p>" + c.data.text + "</p>";
  38. preview.String += c.data.text;
  39. break;
  40. case "header":
  41. contentHtml += "<h" + c.data.level + ">" + c.data.text + "/h" + c.data.level + ">";
  42. preview.String += c.data.text;
  43. break;
  44. case "list":
  45. if (c.data.style === "ordered") {
  46. contentHtml += "<ol>";
  47. for (const li of c.data.items) {
  48. contentHtml += "<li>" + li + "</li>";
  49. preview.String += li;
  50. }
  51. contentHtml += "</ol>";
  52. } else {
  53. contentHtml += "<ul>";
  54. for (const li of c.data.items) {
  55. contentHtml += "<li>" + li + "</li>";
  56. preview.String += li;
  57. }
  58. contentHtml += "</ul>";
  59. }
  60. break;
  61. case "delimiter":
  62. contentHtml += "<hr>";
  63. break;
  64. case "table":
  65. contentHtml += "<table>";
  66. for (const row of c.data.content) {
  67. contentHtml += "<tr>";
  68. for (const col of row) {
  69. contentHtml += "<td>" + col + "</td>";
  70. preview.String += col;
  71. }
  72. contentHtml += "</tr>";
  73. }
  74. contentHtml += "</table>";
  75. break;
  76. }
  77. }
  78. if (preview.String.length > 260) {
  79. preview.String = preview.String.slice(0, 260);
  80. }
  81. var date = new Date();
  82. let m = ("00" + (date.getMonth() + 1)).slice(-2);
  83. let d = ("00" + date.getDate()).slice(-2);
  84. var imageFullpath = "image/notice/" + date.getUTCFullYear().toString() + "-" + m + "-" + d + "/";
  85. var thumbFullpath = "thumb/notice/" + date.getUTCFullYear().toString() + "-" + m + "-" + m + "/";
  86. let selectedImage = {
  87. Valid: false,
  88. String: "",
  89. };
  90. let selectedInput = images.querySelector("input:checked");
  91. if (selectedInput) {
  92. let img = selectedInput.closest(".image").querySelector("img");
  93. if (img) {
  94. selectedImage.String = thumbFullpath + img.alt.toLowerCase();
  95. selectedImage.Valid = true;
  96. }
  97. }
  98. let data = new FormData();
  99. if (file.files.length > 0) {
  100. contentHtml += '<div class="news-images">';
  101. for (const f of file.files) {
  102. let filename = f.name.toLowerCase();
  103. contentHtml +=
  104. '<a href="/' +
  105. imageFullpath +
  106. filename +
  107. '" data-fslightbox="' +
  108. link.value +
  109. '"><img src="/' +
  110. thumbFullpath +
  111. filename +
  112. '" alt="' +
  113. filename +
  114. '"></a>';
  115. data.append("Files", f, filename);
  116. }
  117. contentHtml += "</div>";
  118. }
  119. data.set("Caption", caption.value);
  120. data.set("Link", link.value);
  121. data.set("Content", contentHtml);
  122. data.set("Source", JSON.stringify(content));
  123. data.set("PreviewValid", preview.Valid);
  124. data.set("PreviewString", preview.String);
  125. data.set("ImageValid", selectedImage.Valid);
  126. data.set("ImageString", selectedImage.String);
  127. data.set("ImagePath", imageFullpath);
  128. data.set("ThumbPath", thumbFullpath);
  129. let closeModal = this.closest(".modal").querySelector(".close");
  130. XHR.POST(
  131. function (result) {
  132. if (result && "Error" in result) {
  133. if (result.Error === null) {
  134. Messenger.Show("Объявление успешно добавлено");
  135. closeModal.click();
  136. } else {
  137. Messenger.Show("Возникла ошибка добавления объявления. " + result.Error);
  138. }
  139. }
  140. },
  141. "/material-insert/" + MATERIAL_CATEGORY.NOTICE,
  142. data,
  143. null,
  144. true,
  145. true
  146. );
  147. }
  148. /**
  149. * Создать модальное окно для добавления новости
  150. */
  151. function addNoticeModal() {
  152. let body = document.createElement("div"),
  153. inputCaption = document.createElement("input"),
  154. inputLink = document.createElement("input"),
  155. contentBlock = document.createElement("div"),
  156. previewBlock = document.createElement("div"),
  157. preview = document.createElement("div"),
  158. label = document.createElement("label"),
  159. file = document.createElement("input"),
  160. fileButton = document.createElement("div"),
  161. buttonSend = document.createElement("button");
  162. label.append(file, fileButton);
  163. label.className = "select-file";
  164. fileButton.className = "button";
  165. fileButton.textContent = "Выбрать изображениие";
  166. file.type = "file";
  167. file.multiple = false;
  168. file.accept = "image/*";
  169. file.onchange = changeImages;
  170. inputCaption.type = "text";
  171. inputCaption.placeholder = "Заголовок/название (обязательно)";
  172. inputLink.type = "text";
  173. inputLink.disabled = true;
  174. inputLink.placeholder = "Ссылка (не обязательно)";
  175. buttonSend.textContent = "Создать";
  176. // buttonSend.onclick = create.bind(buttonSend, inputCaption, inputLink, contentBlock, preview, file);
  177. preview.className = "preview-images";
  178. contentBlock.id = "editorjs";
  179. contentBlock.className = "news-content-editor";
  180. // contentBlock.append(createManipulator());
  181. body.append(inputCaption, inputLink, contentBlock, previewBlock, label, preview, buttonSend);
  182. const editor = new EditorJS({
  183. holderOd: "editorjs",
  184. tools: {
  185. header: {
  186. class: Header,
  187. inlineToolbar: true,
  188. config: {
  189. placeholder: "Заголовок",
  190. },
  191. shortcut: "CMD+SHIFT+H",
  192. },
  193. list: {
  194. class: List,
  195. inlineToolbar: true,
  196. shortcut: "CMD+SHIFT+L",
  197. },
  198. marker: {
  199. class: Marker,
  200. shortcut: "CMD+SHIFT+M",
  201. },
  202. delimiter: Delimiter,
  203. inlineCode: {
  204. class: InlineCode,
  205. shortcut: "CMD+SHIFT+C",
  206. },
  207. linkTool: LinkTool,
  208. table: {
  209. class: Table,
  210. inlineToolbar: true,
  211. shortcut: "CMD+ALT+T",
  212. },
  213. onReady: function () {
  214. // saveButton.click();
  215. },
  216. onChange: function () {
  217. console.log("something changed");
  218. },
  219. },
  220. data: null,
  221. });
  222. buttonSend.addEventListener("click", function () {
  223. editor.save().then((savedData) => {
  224. console.log(savedData);
  225. create.call(buttonSend, inputCaption, inputLink, savedData, preview, file);
  226. });
  227. });
  228. Modal.Create("Создать объявление", body);
  229. }
  230. /**
  231. * Функция выбора файлов изображения для новости
  232. */
  233. function changeImages() {
  234. let preview = document.querySelector(".preview-images");
  235. if (!preview) {
  236. console.warn("Элемент [.preview-images] не найжен");
  237. return;
  238. }
  239. while (preview.children.length > 0) {
  240. preview.children[0].remove();
  241. }
  242. for (const file of this.files) {
  243. /**
  244. * 1. Контейнер под изображение
  245. */
  246. let div = document.createElement("div");
  247. div.className = "image";
  248. /**
  249. * 2. Создать элемент для изображения
  250. */
  251. let img = document.createElement("img");
  252. img.src = window.URL.createObjectURL(file);
  253. img.alt = file.name;
  254. img.height = 100;
  255. /**
  256. * 3. Создать label
  257. */
  258. let label = document.createElement("label");
  259. /**
  260. * 4. Создать input + span
  261. */
  262. let input = document.createElement("input");
  263. input.type = "radio";
  264. input.dataset.image = file.name;
  265. input.name = "preview-images";
  266. let span = document.createElement("span");
  267. div.append(img, label);
  268. label.append(input, span);
  269. preview.append(div);
  270. }
  271. let firstImage = preview.querySelector("input");
  272. firstImage.click();
  273. }
  274. return module;
  275. })();