Skip to content

Commit

Permalink
fix: Missing image in survey question (#621)
Browse files Browse the repository at this point in the history
* fix: update condition to not empty input value if image is present

* test: add system test to check for input value

* test: update check for image

* test: update img check again

* test: last update check img

* test: update

* test: another update

* test: if img is present

* test: update other test to avoid ambiguous selector error

* test: update to see if image is presnet after save

* fix: override editor js in decidim_awesome

* test: update system test
  • Loading branch information
Stef-Rousset authored Nov 13, 2024
1 parent 27b6e4a commit c1e6142
Show file tree
Hide file tree
Showing 4 changed files with 432 additions and 10 deletions.
214 changes: 214 additions & 0 deletions app/packs/src/decidim/decidim_awesome/editors/editor.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,214 @@
/* eslint-disable require-jsdoc, func-style */

/*
* Since version 0.25 we follow a different strategy and opt to destroy and override completely the original editor
* That's because editors are instantiated directly instead of creating a global function to instantiate them
*/

import lineBreakButtonHandler from "src/decidim/editor/linebreak_module"
import InscrybMDE from "inscrybmde"
import Europa from "europa"
import "inline-attachment/src/inline-attachment";
import "inline-attachment/src/codemirror-4.inline-attachment";
import "inline-attachment/src/jquery.inline-attachment";
import hljs from "highlight.js";
import "highlight.js/styles/github.css";
import "src/decidim/editor/clipboard_override"
import "src/decidim/vendor/image-resize.min"
import "src/decidim/vendor/image-upload.min"
import { marked } from "marked";

const DecidimAwesome = window.DecidimAwesome || {};
const quillFormats = ["bold", "italic", "link", "underline", "header", "list", "video", "image", "alt", "break", "width", "style", "code", "blockquote", "indent"];

// A tricky way to destroy the quill editor
export function destroyQuillEditor(container) {
if (container) {
const content = $(container).find(".ql-editor").html();
$(container).html(content);
$(container).siblings(".ql-toolbar").remove();
$(container).find("*[class*='ql-']").removeClass((index, className) => (className.match(/(^|\s)ql-\S+/g) || []).join(" "));
$(container).removeClass((index, className) => (className.match(/(^|\s)ql-\S+/g) || []).join(" "));
if ($(container).next().is("p.help-text")) {
$(container).next().remove();
}
}
else {
console.error(`editor [${container}] not exists`);
}
}

export function createQuillEditor(container) {
const toolbar = $(container).data("toolbar");
const disabled = $(container).data("disabled");
const allowedEmptyContentSelector = "iframe,img";

let quillToolbar = [
["bold", "italic", "underline", "linebreak"],
[{ list: "ordered" }, { list: "bullet" }],
["link", "clean"],
["code", "blockquote"],
[{ "indent": "-1"}, { "indent": "+1" }]
];

let addImage = false;

if (toolbar === "full") {
quillToolbar = [
[{ header: [2, 3, 4, 5, 6, false] }],
...quillToolbar
];
if (DecidimAwesome.allow_images_in_full_editor) {
quillToolbar.push(["video", "image"]);
addImage = true;
} else {
quillToolbar.push(["video"]);
}
} else if (toolbar === "basic") {
if (DecidimAwesome.allow_images_in_small_editor) {
quillToolbar.push(["video", "image"]);
addImage = true;
} else {
quillToolbar.push(["video"]);
}
} else if (DecidimAwesome.allow_images_in_small_editor) {
quillToolbar.push(["image"]);
addImage = true;
}

let modules = {
linebreak: {},
toolbar: {
container: quillToolbar,
handlers: {
"linebreak": lineBreakButtonHandler
}
}
};

const $input = $(container).siblings('input[type="hidden"]');
container.innerHTML = $input.val() || "";
const token = $('meta[name="csrf-token"]').attr("content");
if (addImage) {
modules.imageResize = {
modules: ["Resize", "DisplaySize"]
}
modules.imageUpload = {
url: DecidimAwesome.editor_uploader_path,
method: "POST",
name: "image",
withCredentials: false,
headers: { "X-CSRF-Token": token },
callbackOK: (serverResponse, next) => {
$("div.ql-toolbar").last().removeClass("editor-loading")
next(serverResponse.url);
},
callbackKO: (serverError) => {
$("div.ql-toolbar").last().removeClass("editor-loading")
let msg = serverError && serverError.body;
try {
msg = JSON.parse(msg).message;
} catch (evt) { console.error("Parsing error", evt); }
console.error(`Image upload error: ${msg}`);
let $p = $(`<p class="text-alert help-text">${msg}</p>`);
$(container).after($p)
setTimeout(() => {
$p.fadeOut(1000, () => {
$p.destroy();
});
}, 3000);
},
checkBeforeSend: (file, next) => {
$("div.ql-toolbar").last().addClass("editor-loading")
next(file);
}
}
}
const quill = new Quill(container, {
modules: modules,
formats: quillFormats,
theme: "snow"
});

if (disabled) {
quill.disable();
}

quill.on("text-change", () => {
const text = quill.getText();

// Triggers CustomEvent with the cursor position
// It is required in input_mentions.js
let event = new CustomEvent("quill-position", {
detail: quill.getSelection()
});
container.dispatchEvent(event);

if ((text === "\n" || text === "\n\n") && quill.root.querySelectorAll(allowedEmptyContentSelector).length === 0
&& !$input.val().match(/img/)) {
$input.val("");
} else {
const emptyParagraph = "<p><br></p>";
const cleanHTML = quill.root.innerHTML.replace(
new RegExp(`^${emptyParagraph}|${emptyParagraph}$`, "g"),
""
);
$input.val(cleanHTML);
}
});
// After editor is ready, linebreak_module deletes two extraneous new lines
quill.emitter.emit("editor-ready");

if (addImage) {
const text = $(container).data("dragAndDropHelpText") || DecidimAwesome.texts.drag_and_drop_image;
$(container).after(`<p class="help-text" style="margin-top:-1.5rem;">${text}</p>`);
}

// After editor is ready, linebreak_module deletes two extraneous new lines
quill.emitter.emit("editor-ready");

return quill;
}

export function createMarkdownEditor(container) {
const text = DecidimAwesome.texts.drag_and_drop_image;
const token = $('meta[name="csrf-token"]').attr("content");
const $input = $(container).siblings('input[type="hidden"]');
const $faker = $('<textarea name="faker-inscrybmde"/>');
const $form = $(container).closest("form");
const europa = new Europa();
$faker.val(europa.convert($input.val()));
$faker.insertBefore($(container));
$(container).hide();
const inscrybmde = new InscrybMDE({
element: $faker[0],
spellChecker: false,
renderingConfig: {
codeSyntaxHighlighting: true,
hljs: hljs
}
});
$faker[0].InscrybMDE = inscrybmde;

// Allow image upload
if (DecidimAwesome.allow_images_in_markdown_editor) {
$(inscrybmde.gui.statusbar).prepend(`<span class="help-text" style="float:left;margin:0;text-align:left;">${text}</span>`);
window.inlineAttachment.editors.codemirror4.attach(inscrybmde.codemirror, {
uploadUrl: DecidimAwesome.editor_uploader_path,
uploadFieldName: "image",
jsonFieldName: "url",
extraHeaders: { "X-CSRF-Token": token }
});
}

// Allow linebreaks
marked.setOptions({
breaks: true
});

// convert to html on submit
$form.on("submit", () => {
// e.preventDefault();
$input.val(marked(inscrybmde.value()));
});
}
152 changes: 152 additions & 0 deletions app/packs/src/decidim/editor.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,152 @@
/* eslint-disable require-jsdoc */

import lineBreakButtonHandler from "src/decidim/editor/linebreak_module";
import "src/decidim/editor/clipboard_override";
import "src/decidim/vendor/image-resize.min";
import "src/decidim/vendor/image-upload.min";

const quillFormats = [
"bold",
"italic",
"link",
"underline",
"header",
"list",
"alt",
"break",
"width",
"style",
"code",
"blockquote",
"indent"
];

export default function createQuillEditor(container) {
const toolbar = $(container).data("toolbar");
const disabled = $(container).data("disabled");

const allowedEmptyContentSelector = "iframe";
let quillToolbar = [
["bold", "italic", "underline", "linebreak"],
[{ list: "ordered" }, { list: "bullet" }],
["link", "clean"],
["code", "blockquote"],
[{ indent: "-1" }, { indent: "+1" }]
];

let addImage = false;
let addVideo = false;

/**
* - basic = only basic controls without titles
* - content = basic + headings
* - full = basic + headings + image + video
*/
if (toolbar === "content") {
quillToolbar = [[{ header: [2, 3, 4, 5, 6, false] }], ...quillToolbar];
} else if (toolbar === "full") {
addImage = true;
addVideo = true;
quillToolbar = [
[{ header: [2, 3, 4, 5, 6, false] }],
...quillToolbar,
["video"],
["image"]
];
}

let modules = {
linebreak: {},
toolbar: {
container: quillToolbar,
handlers: {
linebreak: lineBreakButtonHandler
}
}
};

const $input = $(container).siblings('input[type="hidden"]');
container.innerHTML = $input.val() || "";
const token = $('meta[name="csrf-token"]').attr("content");

if (addVideo) {
quillFormats.push("video");
}

if (addImage) {
// Attempt to allow images only if the image support is enabled at editor support.
// see: https://github.com/quilljs/quill/issues/1108
quillFormats.push("image");

modules.imageResize = {
modules: ["Resize", "DisplaySize"]
};
modules.imageUpload = {
url: $(container).data("uploadImagesPath"),
method: "POST",
name: "image",
withCredentials: false,
headers: { "X-CSRF-Token": token },
callbackOK: (serverResponse, next) => {
$("div.ql-toolbar").last().removeClass("editor-loading");
next(serverResponse.url);
},
callbackKO: (serverError) => {
$("div.ql-toolbar").last().removeClass("editor-loading");
console.log(`Image upload error: ${serverError.message}`);
},
checkBeforeSend: (file, next) => {
$("div.ql-toolbar").last().addClass("editor-loading");
next(file);
}
};

const text = $(container).data("dragAndDropHelpText");
$(container).after(
`<p class="help-text" style="margin-top:-1.5rem;">${text}</p>`
);
}
const quill = new Quill(container, {
modules: modules,
formats: quillFormats,
theme: "snow"
});

if (addImage === false) {
// Firefox natively implements image drop in contenteditable which is why we need to disable that
quill.root.addEventListener("drop", (ev) => ev.preventDefault());
}

if (disabled) {
quill.disable();
}

quill.on("text-change", () => {

const text = quill.getText();
// Triggers CustomEvent with the cursor position
// It is required in input_mentions.js
let event = new CustomEvent("quill-position", {
detail: quill.getSelection()
});
container.dispatchEvent(event);

if (
(text === "\n" || text === "\n\n") &&
quill.root.querySelectorAll(allowedEmptyContentSelector).length === 0 && !$input.val().match(/img/)
) {
$input.val("");
} else {
const emptyParagraph = "<p><br></p>";
const cleanHTML = quill.root.innerHTML.replace(
new RegExp(`^${emptyParagraph}|${emptyParagraph}$`, "g"),
""
);
$input.val(cleanHTML);
}
});
// After editor is ready, linebreak_module deletes two extraneous new lines
quill.emitter.emit("editor-ready");

return quill;
}
Loading

0 comments on commit c1e6142

Please sign in to comment.