news-editor.js 8.3 KB

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