diff --git a/templates/tutorialv2/edit/conclusion.html b/templates/tutorialv2/edit/conclusion.html
new file mode 100644
index 0000000000..a82b24e4e5
--- /dev/null
+++ b/templates/tutorialv2/edit/conclusion.html
@@ -0,0 +1,26 @@
+{% extends "tutorialv2/base.html" %}
+{% load crispy_forms_tags %}
+{% load i18n %}
+
+{% block title %}
+ {% trans "Modifier la conclusion de " %}{{ content.title }}
+{% endblock %}
+
+{% block breadcrumb %}
+
{{ content.title }}
+ {% trans "Modifier la conclusion" %}
+{% endblock %}
+
+{% block headline %}
+
+ {% if content.image %}
+
+ {% endif %}
+ {% blocktrans with title=content.title %}Modifier la conclusion de « {{ title }} »{% endblocktrans %}
+
+{% endblock %}
+
+
+{% block content %}
+ {% crispy form %}
+{% endblock %}
diff --git a/templates/tutorialv2/edit/content.html b/templates/tutorialv2/edit/introduction.html
similarity index 80%
rename from templates/tutorialv2/edit/content.html
rename to templates/tutorialv2/edit/introduction.html
index 21f42caf28..8162f3d7fc 100644
--- a/templates/tutorialv2/edit/content.html
+++ b/templates/tutorialv2/edit/introduction.html
@@ -5,12 +5,12 @@
{% load feminize %}
{% block title %}
- {% trans "Éditer " %}{{ content.textual_type }}
+ {% trans "Modifier l'introduction de " %}{{ content.title }}
{% endblock %}
{% block breadcrumb %}
{{ content.title }}
- {% trans "Éditer " %}{{ content.textual_type|lower }}
+ {% trans "Modifier l'introduction" %}
{% endblock %}
{% block headline %}
@@ -18,14 +18,10 @@
{% if content.image %}
{% endif %}
- {% trans "Éditer" %} : {{ content.title }}
+ {% blocktrans with title=content.title %}Modifier l'introduction de « {{ title }} »{% endblocktrans %}
{% endblock %}
-{% block headline_sub %}
- {{ content.description }}
-{% endblock %}
-
{% block content %}
{% if new_version %}
diff --git a/templates/tutorialv2/includes/content/content.part.html b/templates/tutorialv2/includes/content/content.part.html
index 3e8ba4e0e0..d01bbd5805 100644
--- a/templates/tutorialv2/includes/content/content.part.html
+++ b/templates/tutorialv2/includes/content/content.part.html
@@ -4,7 +4,7 @@
{% if content.get_introduction %}
{% if display_config.draft_actions.enable_edit %}
@@ -15,10 +15,12 @@
{% trans "Il n’y a pas d’introduction." %}
{% if display_config.draft_actions.enable_edit %}
- {% trans "Vous pouvez " %}{% trans "en ajouter une" %}.
+ {% trans "Vous pouvez " %}{% trans "en ajouter une" %}.
{% endif %}
+
+
{% endif %}
{% if content.has_extracts or content.can_add_extract %}
@@ -108,7 +110,7 @@
{% if content.get_conclusion %}
{% if display_config.draft_actions.enable_edit %}
@@ -119,7 +121,7 @@
{% trans "Il n’y a pas de conclusion." %}
{% if display_config.draft_actions.enable_edit %}
- {% trans "Vous pouvez " %}{% trans "en ajouter une" %}.
+ {% trans "Vous pouvez " %}{% trans "en ajouter une" %}.
{% endif %}
diff --git a/templates/tutorialv2/view/content.html b/templates/tutorialv2/view/content.html
index 4f4b0c7462..bb197bdc86 100644
--- a/templates/tutorialv2/view/content.html
+++ b/templates/tutorialv2/view/content.html
@@ -170,13 +170,6 @@
{% endfor %}
{% endif %}
- {% if display_config.draft_actions.show_license_edit %}
-
- {% url "content:edit" content.pk content.slug as edit_url %}
- {% trans "Éditer" %}
-
- {% endif %}
-
{% if display_config.draft_actions.show_import_link %}
{% url "content:import" content.pk content.slug as import_url %}
diff --git a/zds/tutorialv2/forms.py b/zds/tutorialv2/forms.py
index 63ed611bb9..a9d73013e2 100644
--- a/zds/tutorialv2/forms.py
+++ b/zds/tutorialv2/forms.py
@@ -14,7 +14,7 @@
from zds.tutorialv2.models.database import PublishableContent
from django.utils.translation import gettext_lazy as _
from zds.utils.forms import IncludeEasyMDE
-from zds.utils.validators import with_svg_validator, slugify_raise_on_invalid, InvalidSlugError
+from zds.utils.validators import slugify_raise_on_invalid, InvalidSlugError
class FormWithTitle(forms.Form):
@@ -108,32 +108,14 @@ def __init__(self, *args, **kwargs):
class ContentForm(ContainerForm):
- type = forms.ChoiceField(choices=TYPE_CHOICES, required=False)
+ type = forms.ChoiceField(choices=TYPE_CHOICES, required=True)
def _create_layout(self):
self.helper.layout = Layout(
IncludeEasyMDE(),
Field("title"),
Field("type"),
- Field("introduction", css_class="md-editor preview-source"),
- ButtonHolder(
- StrictButton(_("Aperçu"), type="preview", name="preview", css_class="btn btn-grey preview-btn"),
- ),
- HTML(
- '{% if form.introduction.value %}{% include "misc/preview.part.html" \
- with text=form.introduction.value %}{% endif %}'
- ),
- Field("conclusion", css_class="md-editor preview-source"),
- ButtonHolder(
- StrictButton(_("Aperçu"), type="preview", name="preview", css_class="btn btn-grey preview-btn"),
- ),
- HTML(
- '{% if form.conclusion.value %}{% include "misc/preview.part.html" \
- with text=form.conclusion.value %}{% endif %}'
- ),
- Field("last_hash"),
- Field("msg_commit"),
- ButtonHolder(StrictButton("Valider", type="submit")),
+ StrictButton("Valider", type="submit"),
)
def __init__(self, *args, **kwargs):
@@ -148,32 +130,6 @@ def __init__(self, *args, **kwargs):
self.helper["type"].wrap(Field, disabled=True)
-class EditContentForm(ContentForm):
- title = None
- description = None
- type = None
-
- def _create_layout(self):
- self.helper.layout = Layout(
- IncludeEasyMDE(),
- Field("introduction", css_class="md-editor preview-source"),
- StrictButton(_("Aperçu"), type="preview", name="preview", css_class="btn btn-grey preview-btn"),
- HTML(
- '{% if form.introduction.value %}{% include "misc/preview.part.html" \
- with text=form.introduction.value %}{% endif %}'
- ),
- Field("conclusion", css_class="md-editor preview-source"),
- StrictButton(_("Aperçu"), type="preview", name="preview", css_class="btn btn-grey preview-btn"),
- HTML(
- '{% if form.conclusion.value %}{% include "misc/preview.part.html" \
- with text=form.conclusion.value %}{% endif %}'
- ),
- Field("last_hash"),
- Field("msg_commit"),
- ButtonHolder(StrictButton("Valider", type="submit")),
- )
-
-
class ExtractForm(FormWithTitle):
text = forms.CharField(
label=_("Texte"),
diff --git a/zds/tutorialv2/mixins.py b/zds/tutorialv2/mixins.py
index 2d1b80a6bb..520309efdb 100644
--- a/zds/tutorialv2/mixins.py
+++ b/zds/tutorialv2/mixins.py
@@ -182,7 +182,7 @@ def form_invalid(self, form):
class FormWithPreview(FormView):
def post(self, request, *args, **kwargs):
- form = self.form_class(request.POST)
+ form = self.get_form()
if "preview" in request.POST:
self.form_invalid(form)
diff --git a/zds/tutorialv2/tests/models/tests_models.py b/zds/tutorialv2/tests/models/tests_models.py
index 465a670516..e73cb70fd5 100644
--- a/zds/tutorialv2/tests/models/tests_models.py
+++ b/zds/tutorialv2/tests/models/tests_models.py
@@ -449,26 +449,34 @@ def test_publication_and_attributes_consistency(self):
old_description = article.public_version.description()
article.licence = LicenceFactory()
article.save()
+
self.client.force_login(self.user_author)
- self.client.post(
- reverse("content:edit", args=[article.pk, article.slug]),
- {
- "title": old_title + "bla",
- "description": old_description + "bla",
- "type": "ARTICLE",
- "licence": article.licence.pk,
- "subcategory": SubCategoryFactory().pk,
- "last_hash": article.sha_draft,
- },
+
+ new_title = old_title + "bla"
+ result = self.client.post(
+ reverse("content:edit-title", args=[article.pk]),
+ {"title": new_title},
+ follow=True,
)
+ self.assertEqual(result.status_code, 200)
+
+ new_description = old_description + "bla"
+ result = self.client.post(
+ reverse("content:edit-subtitle", args=[article.pk]),
+ {"subtitle": new_description},
+ follow=True,
+ )
+ self.assertEqual(result.status_code, 200)
+
article = PublishableContent.objects.prefetch_related("public_version").get(pk=article.pk)
article.public_version.load_public_version()
self.assertEqual(old_title, article.public_version.title())
self.assertEqual(old_description, article.public_version.description())
self.assertEqual(old_date, article.public_version.publication_date)
+
publish_content(article, article.load_version(), False)
- article = PublishableContent.objects.get(pk=article.pk)
- article.public_version.load_public_version()
+
+ article = PublishableContent.objects.prefetch_related("public_version").get(pk=article.pk)
self.assertEqual(old_date, article.public_version.publication_date)
self.assertNotEqual(old_date, article.public_version.update_date)
diff --git a/zds/tutorialv2/tests/tests_front.py b/zds/tutorialv2/tests/tests_front.py
index 02e0086629..5d8eb5931f 100644
--- a/zds/tutorialv2/tests/tests_front.py
+++ b/zds/tutorialv2/tests/tests_front.py
@@ -121,7 +121,7 @@ def test_collaborative_article_edition_and_editor_persistence(self):
article.sha_draft = versioned_article.repo_update("article", "", "", update_slug=False)
article.save()
- article_edit_url = reverse("content:edit", args=[article.pk, article.slug])
+ article_edit_url = reverse("content:edit-introduction", args=[article.pk])
self.login(author)
selenium.execute_script('localStorage.setItem("editor_choice", "new")') # we want the new editor
@@ -130,7 +130,7 @@ def test_collaborative_article_edition_and_editor_persistence(self):
intro = self.find_element("div#div_id_introduction div.CodeMirror")
# ActionChains: Support for CodeMirror https://stackoverflow.com/a/48969245/2226755
action_chains = ActionChains(selenium)
- scrollDriverTo(selenium, 0, 312)
+ scroll_driver_to(selenium, 0, 312)
action_chains.click(intro).perform()
action_chains.send_keys("intro").perform()
@@ -144,36 +144,7 @@ def test_collaborative_article_edition_and_editor_persistence(self):
self.assertEqual("new intro", self.find_element(".md-editor#id_introduction").get_attribute("value"))
- def test_the_editor_forgets_its_content_on_form_submission(self):
- selenium = self.selenium
-
- author = ProfileFactory()
-
- self.login(author)
- selenium.execute_script('localStorage.setItem("editor_choice", "new")') # we want the new editor
- new_article_url = self.live_server_url + reverse(
- "content:create-content", kwargs={"created_content_type": "ARTICLE"}
- )
- selenium.get(new_article_url)
- WebDriverWait(self.selenium, 10).until(ec.element_to_be_clickable((By.CSS_SELECTOR, "#id_title"))).click()
-
- self.find_element("#id_title").send_keys("Oulipo")
-
- intro = self.find_element("div#div_id_introduction div.CodeMirror")
- action_chains = ActionChains(selenium)
- scrollDriverTo(selenium, 0, 312)
- action_chains.click(intro).perform()
- action_chains.send_keys("Le cadavre exquis boira le vin nouveau.").perform()
-
- self.find_element(".content-container button[type=submit]").click()
-
- self.assertTrue(WebDriverWait(selenium, 10).until(ec.title_contains("Oulipo")))
-
- selenium.get(new_article_url)
-
- self.assertEqual("", self.find_element(".md-editor#id_introduction").get_attribute("value"))
-
-def scrollDriverTo(driver, x, y):
- scriptScrollTo = f"window.scrollTo({x}, {y});"
- driver.execute_script(scriptScrollTo)
+def scroll_driver_to(driver, x, y):
+ script_scroll_to = f"window.scrollTo({x}, {y});"
+ driver.execute_script(script_scroll_to)
diff --git a/zds/tutorialv2/tests/tests_lists.py b/zds/tutorialv2/tests/tests_lists.py
index fbaa04603e..0012802b05 100644
--- a/zds/tutorialv2/tests/tests_lists.py
+++ b/zds/tutorialv2/tests/tests_lists.py
@@ -118,7 +118,7 @@ def test_list_categories(self):
context_categories = list(resp.context_data["categories"])
self.assertEqual(context_categories[0].contents_count, 10)
- self.assertEqual(context_categories[0].subcategories, [subcategory_1, subcategory_2])
+ self.assertCountEqual(context_categories[0].subcategories, [subcategory_1, subcategory_2])
self.assertIn(category_1, context_categories)
def test_private_lists(self):
diff --git a/zds/tutorialv2/tests/tests_views/tests_content.py b/zds/tutorialv2/tests/tests_views/tests_content.py
index 14fe2429d9..2c629e1109 100644
--- a/zds/tutorialv2/tests/tests_views/tests_content.py
+++ b/zds/tutorialv2/tests/tests_views/tests_content.py
@@ -230,89 +230,39 @@ def test_ensure_access(self):
)
self.assertEqual(result.status_code, 200)
- def test_basic_tutorial_workflow(self):
- """General test on the basic workflow of a tutorial: creation, edition, deletion for the author"""
+ def test_create_tutorial(self):
+ """Test the creation of a new content."""
self.client.force_login(self.user_author)
- # create tutorial
- intro = "une intro"
- conclusion = "une conclusion"
- description = "une description"
title = "un titre"
- random = "un truc à la rien à voir"
- random_with_md = "un text contenant du **markdown** ."
-
- response = self.client.post(
- reverse("content:create-content", kwargs={"created_content_type": "TUTORIAL"}),
- {
- "text": random_with_md,
- "preview": "",
- },
- HTTP_X_REQUESTED_WITH="XMLHttpRequest",
- )
-
- self.assertEqual(200, response.status_code)
-
- result_string = "".join(str(a, "utf-8") for a in response.streaming_content)
- self.assertIn("markdown", result_string, "We need the text to be properly formatted")
-
result = self.client.post(
reverse("content:create-content", kwargs={"created_content_type": "TUTORIAL"}),
- {
- "title": title,
- "introduction": intro,
- "conclusion": conclusion,
- "type": "TUTORIAL",
- "licence": self.licence.pk,
- },
+ {"title": title, "type": "TUTORIAL"},
follow=False,
)
self.assertEqual(result.status_code, 302)
self.assertEqual(PublishableContent.objects.all().count(), 2)
tuto = PublishableContent.objects.last()
- pk = tuto.pk
- slug = tuto.slug
- versioned = tuto.load_version()
-
self.assertEqual(Gallery.objects.filter(pk=tuto.gallery.pk).count(), 1)
self.assertEqual(UserGallery.objects.filter(gallery__pk=tuto.gallery.pk).count(), tuto.authors.count())
- # access to tutorial
- result = self.client.get(reverse("content:edit", args=[pk, slug]), follow=False)
- self.assertEqual(result.status_code, 200)
-
- # preview tutorial
- result = self.client.post(
- reverse("content:edit", args=[pk, slug]),
- {"text": random_with_md, "last_hash": versioned.compute_hash(), "preview": ""},
- HTTP_X_REQUESTED_WITH="XMLHttpRequest",
- )
-
- self.assertEqual(result.status_code, 200)
-
- result_string = "".join(a.decode() for a in result.streaming_content)
- self.assertIn("markdown", result_string, "We need the text to be properly formatted")
+ def test_basic_tutorial_workflow(self):
+ """General test on the basic workflow of a tutorial: edition, deletion for the author"""
+ self.client.force_login(self.user_author)
- result = self.client.post(
- reverse("content:edit", args=[pk, slug]),
- {
- "introduction": random,
- "conclusion": random,
- "type": "TUTORIAL",
- "subcategory": self.subcategory.pk,
- "last_hash": versioned.compute_hash(),
- },
- follow=False,
- )
- self.assertEqual(result.status_code, 302)
+ # create tutorial
+ intro = "une intro"
+ conclusion = "une conclusion"
+ description = "une description"
+ title = "un titre"
+ random = "un truc à la rien à voir"
+ random_with_md = "un text contenant du **markdown** ."
- tuto = PublishableContent.objects.get(pk=pk)
- self.assertEqual(tuto.licence, None)
- versioned = tuto.load_version()
- self.assertEqual(versioned.get_introduction(), random)
- self.assertEqual(versioned.get_conclusion(), random)
- self.assertEqual(versioned.licence, None)
+ tuto = PublishableContentFactory(type="TUTORIAL")
+ tuto.authors.add(self.user_author)
+ pk = tuto.pk
+ slug = tuto.slug
# preview container
result = self.client.post(
@@ -930,18 +880,10 @@ def test_export_content(self):
given_title = "Oh, le beau titre à lire !"
some_text = "À lire à un moment ou un autre, Über utile" # accentuated characters are important for the test
- # create a tutorial
+ # Create a tutorial and modify its introduction and conclusion
result = self.client.post(
reverse("content:create-content", kwargs={"created_content_type": "TUTORIAL"}),
- {
- "title": given_title,
- "description": some_text,
- "introduction": some_text,
- "conclusion": some_text,
- "type": "TUTORIAL",
- "licence": self.licence.pk,
- "subcategory": self.subcategory.pk,
- },
+ {"title": given_title, "type": "TUTORIAL"},
follow=False,
)
self.assertEqual(result.status_code, 302)
@@ -951,6 +893,20 @@ def test_export_content(self):
tuto_pk = tuto.pk
tuto_slug = tuto.slug
+ result = self.client.post(
+ reverse("content:edit-introduction", args=[tuto.pk]),
+ {"introduction": some_text},
+ follow=False,
+ )
+ self.assertEqual(result.status_code, 302)
+
+ result = self.client.post(
+ reverse("content:edit-conclusion", args=[tuto.pk]),
+ {"conclusion": some_text},
+ follow=False,
+ )
+ self.assertEqual(result.status_code, 302)
+
# add a chapter
result = self.client.post(
reverse("content:create-container", args=[tuto_pk, tuto_slug]),
@@ -1093,14 +1049,7 @@ def test_import_create_content(self):
# create a tutorial
result = self.client.post(
reverse("content:create-content", kwargs={"created_content_type": "TUTORIAL"}),
- {
- "title": given_title,
- "description": some_text,
- "introduction": some_text,
- "conclusion": some_text,
- "type": "TUTORIAL",
- "subcategory": self.subcategory.pk,
- },
+ {"title": given_title, "type": "TUTORIAL"},
follow=False,
)
self.assertEqual(result.status_code, 302)
@@ -1110,6 +1059,20 @@ def test_import_create_content(self):
tuto_pk = tuto.pk
tuto_slug = tuto.slug
+ result = self.client.post(
+ reverse("content:edit-introduction", args=[tuto.pk]),
+ {"introduction": some_text},
+ follow=False,
+ )
+ self.assertEqual(result.status_code, 302)
+
+ result = self.client.post(
+ reverse("content:edit-conclusion", args=[tuto.pk]),
+ {"conclusion": some_text},
+ follow=False,
+ )
+ self.assertEqual(result.status_code, 302)
+
# add a chapter
result = self.client.post(
reverse("content:create-container", args=[tuto_pk, tuto_slug]),
@@ -1209,15 +1172,7 @@ def test_import_in_existing_content(self):
# create a tutorial
result = self.client.post(
reverse("content:create-content", kwargs={"created_content_type": "TUTORIAL"}),
- {
- "title": given_title,
- "description": some_text,
- "introduction": some_text,
- "conclusion": some_text,
- "type": "TUTORIAL",
- "licence": self.licence.pk,
- "subcategory": self.subcategory.pk,
- },
+ {"title": given_title, "type": "TUTORIAL"},
follow=False,
)
self.assertEqual(result.status_code, 302)
@@ -1227,6 +1182,20 @@ def test_import_in_existing_content(self):
tuto_pk = tuto.pk
tuto_slug = tuto.slug
+ result = self.client.post(
+ reverse("content:edit-introduction", args=[tuto.pk]),
+ {"introduction": some_text},
+ follow=False,
+ )
+ self.assertEqual(result.status_code, 302)
+
+ result = self.client.post(
+ reverse("content:edit-conclusion", args=[tuto.pk]),
+ {"conclusion": some_text},
+ follow=False,
+ )
+ self.assertEqual(result.status_code, 302)
+
# add a chapter
result = self.client.post(
reverse("content:create-container", args=[tuto_pk, tuto_slug]),
@@ -1652,21 +1621,12 @@ def test_validation_subscription(self):
# Re-ask a new validation
self.client.force_login(self.user_author)
+ # Update the title to spice things up
tuto = PublishableContent.objects.get(pk=tuto.pk)
versioned = tuto.load_version()
self.client.post(
- reverse("content:edit", args=[tuto.pk, tuto.slug]),
- {
- "title": "new title so that everything explode",
- "description": tuto.description,
- "introduction": tuto.load_version().get_introduction(),
- "conclusion": tuto.load_version().get_conclusion(),
- "type": "ARTICLE",
- "licence": tuto.licence.pk,
- "subcategory": self.subcategory.pk,
- "last_hash": tuto.load_version(tuto.sha_draft).compute_hash(),
- "image": (settings.BASE_DIR / "fixtures" / "logo.png").open("rb"),
- },
+ reverse("content:edit-title", args=[tuto.pk]),
+ {"title": "new title so that everything explode"},
follow=False,
)
@@ -2532,55 +2492,6 @@ def test_concurent_edition(self):
self.client.force_login(self.user_author)
- # no hash, no edition
- result = self.client.post(
- reverse("content:edit", args=[tuto.pk, tuto.slug]),
- {
- "title": tuto.title,
- "description": tuto.description,
- "introduction": random,
- "conclusion": random,
- "type": "TUTORIAL",
- "licence": self.licence.pk,
- "subcategory": self.subcategory.pk,
- "last_hash": "",
- },
- follow=True,
- )
- self.assertEqual(result.status_code, 200)
-
- msgs = result.context["messages"]
- last = None
- for msg in msgs:
- last = msg
- self.assertEqual(last.level, messages.ERROR)
-
- tuto = PublishableContent.objects.get(pk=tuto.pk)
- versioned = tuto.load_version()
- self.assertNotEqual(versioned.get_introduction(), random)
- self.assertNotEqual(versioned.get_conclusion(), random)
-
- result = self.client.post(
- reverse("content:edit", args=[tuto.pk, tuto.slug]),
- {
- "title": tuto.title,
- "description": tuto.description,
- "introduction": random,
- "conclusion": random,
- "type": "TUTORIAL",
- "licence": self.licence.pk,
- "subcategory": self.subcategory.pk,
- "last_hash": versioned.compute_hash(), # good hash
- },
- follow=True,
- )
- self.assertEqual(result.status_code, 200)
-
- tuto = PublishableContent.objects.get(pk=tuto.pk)
- versioned = tuto.load_version()
- self.assertEqual(versioned.get_introduction(), random)
- self.assertEqual(versioned.get_conclusion(), random)
-
# edit container:
result = self.client.post(
reverse(
diff --git a/zds/tutorialv2/tests/tests_views/tests_display.py b/zds/tutorialv2/tests/tests_views/tests_display.py
index feeaeb3c1c..a58800ab0f 100644
--- a/zds/tutorialv2/tests/tests_views/tests_display.py
+++ b/zds/tutorialv2/tests/tests_views/tests_display.py
@@ -4,7 +4,6 @@
from django.test import TestCase
from django.test.utils import override_settings
from django.urls import reverse
-from django.utils.translation import gettext_lazy as _
from zds.member.tests.factories import ProfileFactory
from zds.tutorialv2.tests import TutorialTestMixin
@@ -34,23 +33,14 @@ def setUp(self):
self.user_author = ProfileFactory().user
self.client.force_login(self.user_author)
- # Publish an article:
+ # Publish an article
self.article = PublishedContentFactory(author_list=[self.user_author], type="ARTICLE")
def _new_draft_version(self, text):
- # Create a new draft version:
- versioned = self.article.load_version()
+ """Create a new draft version."""
result = self.client.post(
- reverse("content:edit", args=[self.article.pk, self.article.slug]),
- {
- "title": self.article.title,
- "description": self.article.description,
- "introduction": text,
- "conclusion": "Modified conclusion",
- "type": self.article.type,
- "subcategory": self.article.subcategory.first().pk,
- "last_hash": versioned.compute_hash(),
- },
+ reverse("content:edit-introduction", args=[self.article.pk]),
+ {"introduction": text},
follow=False,
)
self.assertEqual(result.status_code, 302)
@@ -84,7 +74,7 @@ def common_tests():
self.assertNotContains(public_page, PublicActionsState.messages["draft_is_same"])
self.assertContains(public_page, PublicActionsState.messages["draft_is_more_recent"])
- # Now a new draft version, to have different version from validation:
+ # Now a new draft version, to have different version from validation
self._new_draft_version(self.TEXT_SECOND_MODIFICATION)
public_page = common_tests()
@@ -123,7 +113,7 @@ def common_tests():
self.assertNotContains(draft_page, PublicActionsState.messages["public_is_same"])
self.assertContains(draft_page, ValidationActions.messages["validation_is_same"])
- # Now a new draft version, to have different version from validation:
+ # Now a new draft version, to have different version from validation
self._new_draft_version(self.TEXT_SECOND_MODIFICATION)
draft_page = common_tests()
@@ -150,9 +140,7 @@ def common_tests():
self._new_draft_version(self.TEXT_FIRST_MODIFICATION)
request_validation(self.article)
- validation_page = common_tests()
-
- # Now a new draft version, to have different version from validation:
+ # Now a new draft version, to have different version from validation
self._new_draft_version(self.TEXT_SECOND_MODIFICATION)
validation_page = common_tests()
diff --git a/zds/tutorialv2/tests/tests_views/tests_editconclusionview.py b/zds/tutorialv2/tests/tests_views/tests_editconclusionview.py
new file mode 100644
index 0000000000..fcd99142d0
--- /dev/null
+++ b/zds/tutorialv2/tests/tests_views/tests_editconclusionview.py
@@ -0,0 +1,96 @@
+from datetime import datetime
+
+from django.test import TestCase
+from django.urls import reverse
+
+from zds.tutorialv2.tests import TutorialTestMixin, override_for_contents
+from zds.tutorialv2.tests.factories import PublishableContentFactory
+from zds.member.tests.factories import ProfileFactory, StaffProfileFactory
+
+
+view_name = "content:edit-conclusion"
+
+
+@override_for_contents()
+class PermissionTests(TutorialTestMixin, TestCase):
+ """Test permissions and associated behaviors, such as redirections and status codes."""
+
+ def setUp(self):
+ # Create users
+ self.author = ProfileFactory().user
+ self.staff = StaffProfileFactory().user
+ self.outsider = ProfileFactory().user
+
+ # Create a content
+ self.content = PublishableContentFactory(author_list=[self.author])
+
+ # Get information to be reused in tests
+ self.form_url = reverse(view_name, kwargs={"pk": self.content.pk})
+ self.form_data = {"conclusion": "conclusion content"} # avoids changing the slug
+ self.content_data = {"pk": self.content.pk, "slug": self.content.slug}
+ self.content_url = reverse("content:view", kwargs=self.content_data)
+ self.login_url = reverse("member-login") + "?next=" + self.form_url
+
+ def test_not_authenticated(self):
+ """Test that on form submission, unauthenticated users are redirected to the login page."""
+ self.client.logout() # ensure no user is authenticated
+ response = self.client.post(self.form_url, self.form_data)
+ self.assertRedirects(response, self.login_url)
+
+ def test_authenticated_author(self):
+ """Test that on form submission, authors are redirected to the content page."""
+ self.client.force_login(self.author)
+ response = self.client.post(self.form_url, self.form_data)
+ self.assertRedirects(response, self.content_url)
+
+ def test_authenticated_staff(self):
+ """Test that on form submission, staffs are redirected to the content page."""
+ self.client.force_login(self.staff)
+ response = self.client.post(self.form_url, self.form_data)
+ self.assertRedirects(response, self.content_url)
+
+ def test_authenticated_outsider(self):
+ """Test that on form submission, unauthorized users get a 403."""
+ self.client.force_login(self.outsider)
+ response = self.client.post(self.form_url, self.form_data)
+ self.assertEqual(response.status_code, 403)
+
+
+@override_for_contents()
+class FunctionalTests(TutorialTestMixin, TestCase):
+ """Test the detailed behavior of the feature, such as updates of the database or repositories."""
+
+ def setUp(self):
+ self.author = ProfileFactory()
+ self.content = PublishableContentFactory(author_list=[self.author.user])
+ self.form_url = reverse(view_name, kwargs={"pk": self.content.pk})
+ self.client.force_login(self.author.user)
+
+ def test_normal(self):
+ start_date = datetime.now()
+ self.assertTrue(self.content.update_date < start_date)
+
+ new_conclusion = "Ceci n'est pas l'ancienne conclusion"
+ response = self.client.post(self.form_url, {"conclusion": new_conclusion}, follow=True)
+ self.assertEqual(response.status_code, 200)
+
+ self.content.refresh_from_db()
+
+ # Database update
+ self.assertTrue(self.content.update_date > start_date)
+
+ # Update in repository
+ versioned_content = self.content.load_version()
+ self.assertEqual(versioned_content.get_conclusion(), new_conclusion)
+
+ def test_preview(self):
+ some_markdown = "Ceci est un texte avec du **markdown**"
+ expected_preview = "Ceci est un texte avec du markdown"
+
+ response = self.client.post(
+ self.form_url, {"text": some_markdown, "preview": ""}, HTTP_X_REQUESTED_WITH="XMLHttpRequest"
+ )
+
+ self.assertEqual(200, response.status_code)
+ result_string = "".join(str(a, "utf-8") for a in response.streaming_content)
+ self.assertIn(expected_preview, result_string, "We need the text to be properly formatted")
diff --git a/zds/tutorialv2/tests/tests_views/tests_editintroductionview.py b/zds/tutorialv2/tests/tests_views/tests_editintroductionview.py
new file mode 100644
index 0000000000..94b9bd1159
--- /dev/null
+++ b/zds/tutorialv2/tests/tests_views/tests_editintroductionview.py
@@ -0,0 +1,96 @@
+from datetime import datetime
+
+from django.test import TestCase
+from django.urls import reverse
+
+from zds.tutorialv2.tests import TutorialTestMixin, override_for_contents
+from zds.tutorialv2.tests.factories import PublishableContentFactory
+from zds.member.tests.factories import ProfileFactory, StaffProfileFactory
+
+
+view_name = "content:edit-introduction"
+
+
+@override_for_contents()
+class PermissionTests(TutorialTestMixin, TestCase):
+ """Test permissions and associated behaviors, such as redirections and status codes."""
+
+ def setUp(self):
+ # Create users
+ self.author = ProfileFactory().user
+ self.staff = StaffProfileFactory().user
+ self.outsider = ProfileFactory().user
+
+ # Create a content
+ self.content = PublishableContentFactory(author_list=[self.author])
+
+ # Get information to be reused in tests
+ self.form_url = reverse(view_name, kwargs={"pk": self.content.pk})
+ self.form_data = {"introduction": "introduction content"} # avoids changing the slug
+ self.content_data = {"pk": self.content.pk, "slug": self.content.slug}
+ self.content_url = reverse("content:view", kwargs=self.content_data)
+ self.login_url = reverse("member-login") + "?next=" + self.form_url
+
+ def test_not_authenticated(self):
+ """Test that on form submission, unauthenticated users are redirected to the login page."""
+ self.client.logout() # ensure no user is authenticated
+ response = self.client.post(self.form_url, self.form_data)
+ self.assertRedirects(response, self.login_url)
+
+ def test_authenticated_author(self):
+ """Test that on form submission, authors are redirected to the content page."""
+ self.client.force_login(self.author)
+ response = self.client.post(self.form_url, self.form_data)
+ self.assertRedirects(response, self.content_url)
+
+ def test_authenticated_staff(self):
+ """Test that on form submission, staffs are redirected to the content page."""
+ self.client.force_login(self.staff)
+ response = self.client.post(self.form_url, self.form_data)
+ self.assertRedirects(response, self.content_url)
+
+ def test_authenticated_outsider(self):
+ """Test that on form submission, unauthorized users get a 403."""
+ self.client.force_login(self.outsider)
+ response = self.client.post(self.form_url, self.form_data)
+ self.assertEqual(response.status_code, 403)
+
+
+@override_for_contents()
+class FunctionalTests(TutorialTestMixin, TestCase):
+ """Test the detailed behavior of the feature, such as updates of the database or repositories."""
+
+ def setUp(self):
+ self.author = ProfileFactory()
+ self.content = PublishableContentFactory(author_list=[self.author.user])
+ self.form_url = reverse(view_name, kwargs={"pk": self.content.pk})
+ self.client.force_login(self.author.user)
+
+ def test_normal(self):
+ start_date = datetime.now()
+ self.assertTrue(self.content.update_date < start_date)
+
+ new_introduction = "Ceci n'est pas l'ancienne introduction"
+ response = self.client.post(self.form_url, {"introduction": new_introduction}, follow=True)
+ self.assertEqual(response.status_code, 200)
+
+ self.content.refresh_from_db()
+
+ # Database update
+ self.assertTrue(self.content.update_date > start_date)
+
+ # Update in repository
+ versioned_content = self.content.load_version()
+ self.assertEqual(versioned_content.get_introduction(), new_introduction)
+
+ def test_preview(self):
+ some_markdown = "Ceci est un texte avec du **markdown**"
+ expected_preview = "Ceci est un texte avec du markdown"
+
+ response = self.client.post(
+ self.form_url, {"text": some_markdown, "preview": ""}, HTTP_X_REQUESTED_WITH="XMLHttpRequest"
+ )
+
+ self.assertEqual(200, response.status_code)
+ result_string = "".join(str(a, "utf-8") for a in response.streaming_content)
+ self.assertIn(expected_preview, result_string, "We need the text to be properly formatted")
diff --git a/zds/tutorialv2/tests/tests_views/tests_published.py b/zds/tutorialv2/tests/tests_views/tests_published.py
index ee8cb71250..a09c5af73f 100644
--- a/zds/tutorialv2/tests/tests_views/tests_published.py
+++ b/zds/tutorialv2/tests/tests_views/tests_published.py
@@ -1287,17 +1287,8 @@ def test_validation_list_has_good_title(self):
old_title = tuto.title
new_title = "a brand new title"
self.client.post(
- reverse("content:edit", args=[tuto.pk, tuto.slug]),
- {
- "title": new_title,
- "description": tuto.description,
- "introduction": "a",
- "conclusion": "b",
- "type": "TUTORIAL",
- "licence": self.licence.pk,
- "subcategory": self.subcategory.pk,
- "last_hash": tuto.sha_draft,
- },
+ reverse("content:edit-title", args=[tuto.pk]),
+ {"title": new_title},
follow=False,
)
self.client.logout()
@@ -1332,18 +1323,8 @@ def test_unpublish_with_title_change(self):
registered_validation.save()
self.client.force_login(self.user_staff)
self.client.post(
- reverse("content:edit", args=[article.pk, article.slug]),
- {
- "title": "new title so that everything explode",
- "description": article.description,
- "introduction": article.load_version().get_introduction(),
- "conclusion": article.load_version().get_conclusion(),
- "type": "ARTICLE",
- "licence": article.licence.pk,
- "subcategory": self.subcategory.pk,
- "last_hash": article.load_version(article.sha_draft).compute_hash(),
- "image": (settings.BASE_DIR / "fixtures" / "logo.png").open("rb"),
- },
+ reverse("content:edit-title", args=[article.pk]),
+ {"title": "new title so that everything explode"},
follow=False,
)
public_count = PublishedContent.objects.count()
@@ -1418,17 +1399,8 @@ def test_validation_history(self):
published = PublishedContentFactory(author_list=[self.user_author])
self.client.force_login(self.user_author)
result = self.client.post(
- reverse("content:edit", args=[published.pk, published.slug]),
- {
- "title": published.title,
- "description": published.description,
- "introduction": "crappy crap",
- "conclusion": "crappy crap",
- "type": "TUTORIAL",
- "licence": self.licence.pk,
- "subcategory": self.subcategory.pk,
- "last_hash": published.load_version().compute_hash(), # good hash
- },
+ reverse("content:edit-subtitle", args=[published.pk]),
+ {"subtitle": "Sous-titre qui fait une nouvelle version"},
follow=True,
)
self.assertEqual(result.status_code, 200)
@@ -1492,18 +1464,8 @@ def test_ask_validation_update(self):
# login with user, edit content and ask validation for update
self.client.force_login(self.user_author)
result = self.client.post(
- reverse("content:edit", args=[content_draft.pk, content_draft.slug]),
- {
- "title": content_draft.title + "2",
- "description": content_draft.description,
- "introduction": content_draft.introduction,
- "conclusion": content_draft.conclusion,
- "type": content_draft.type,
- "licence": self.licence.pk,
- "subcategory": self.subcategory.pk,
- "last_hash": content_draft.compute_hash(),
- "image": content_draft.image or "None",
- },
+ reverse("content:edit-title", args=[content_draft.pk]),
+ {"title": content_draft.title + "2"},
follow=False,
)
self.assertEqual(result.status_code, 302)
diff --git a/zds/tutorialv2/urls/urls_contents.py b/zds/tutorialv2/urls/urls_contents.py
index 32cb55f869..89da39e1e1 100644
--- a/zds/tutorialv2/urls/urls_contents.py
+++ b/zds/tutorialv2/urls/urls_contents.py
@@ -4,11 +4,12 @@
from zds.tutorialv2.views.canonical import EditCanonicalLinkView
from zds.tutorialv2.views.categories import EditCategoriesView
from zds.tutorialv2.views.contents import (
- CreateContent,
- EditContent,
+ CreateContentView,
DeleteContent,
EditTitle,
EditSubtitle,
+ EditIntroductionView,
+ EditConclusionView,
)
from zds.tutorialv2.views.thumbnail import EditThumbnailView
from zds.tutorialv2.views.display.container import ContainerValidationView
@@ -168,7 +169,7 @@ def get_version_pages():
# typo:
path("reactions/typo/", WarnTypo.as_view(), name="warn-typo"),
# create:
- path("nouveau-contenu//", CreateContent.as_view(), name="create-content"),
+ path("nouveau-contenu//", CreateContentView.as_view(), name="create-content"),
path(
"nouveau-conteneur////",
CreateContainer.as_view(),
@@ -208,7 +209,6 @@ def get_version_pages():
name="edit-extract",
),
path("editer-section////", EditExtract.as_view(), name="edit-extract"),
- path("editer///", EditContent.as_view(), name="edit"),
path("deplacer/", MoveChild.as_view(), name="move-element"),
path("historique///", DisplayHistory.as_view(), name="history"),
path("comparaison///", DisplayDiff.as_view(), name="diff"),
@@ -219,6 +219,8 @@ def get_version_pages():
path("modifier-titre//", EditTitle.as_view(), name="edit-title"),
path("modifier-sous-titre//", EditSubtitle.as_view(), name="edit-subtitle"),
path("modifier-miniature//", EditThumbnailView.as_view(), name="edit-thumbnail"),
+ path("modifier-introduction//", EditIntroductionView.as_view(), name="edit-introduction"),
+ path("modifier-conclusion//", EditConclusionView.as_view(), name="edit-conclusion"),
path("modifier-licence//", EditContentLicense.as_view(), name="edit-license"),
path("modifier-tags//", EditTags.as_view(), name="edit-tags"),
path("modifier-lien-canonique/", EditCanonicalLinkView.as_view(), name="edit-canonical-link"),
diff --git a/zds/tutorialv2/utils.py b/zds/tutorialv2/utils.py
index eaa0739934..a7cfce605d 100644
--- a/zds/tutorialv2/utils.py
+++ b/zds/tutorialv2/utils.py
@@ -507,7 +507,7 @@ def fill_containers_from_json(json_sub, parent):
raise BadManifestError(_("Type d'objet inconnu : « {} »").format(child["object"]))
-def init_new_repo(db_object, introduction_text, conclusion_text, commit_message="", do_commit=True):
+def init_new_repo(db_object, introduction_text="", conclusion_text="", commit_message="", do_commit=True):
"""Create a new repository in ``settings.ZDS_APP['contents']['private_repo']``\
to store the files for a new content. Note that ``db_object.sha_draft`` will be set to the good value
diff --git a/zds/tutorialv2/views/contents.py b/zds/tutorialv2/views/contents.py
index 21c9380110..036258b610 100644
--- a/zds/tutorialv2/views/contents.py
+++ b/zds/tutorialv2/views/contents.py
@@ -3,27 +3,23 @@
from crispy_forms.bootstrap import StrictButton
from crispy_forms.helper import FormHelper
-from crispy_forms.layout import Layout, Field
+from crispy_forms.layout import Layout, Field, HTML, ButtonHolder
from django.conf import settings
from django.contrib import messages
from django.contrib.auth.mixins import LoginRequiredMixin
from django.db import transaction
-from django.forms import forms, CharField
+from django.forms import forms, CharField, Textarea, TextInput
from django.shortcuts import redirect
from django.template.loader import render_to_string
from django.urls import reverse
from django.utils.decorators import method_decorator
from django.utils.translation import gettext_lazy as _
-from django.views.generic import DeleteView
+from django.views.generic import DeleteView, FormView
from zds.gallery.models import Gallery
from zds.member.decorator import LoggedWithReadWriteHability
from zds.member.utils import get_bot_account
-from zds.tutorialv2.forms import (
- ContentForm,
- FormWithTitle,
- EditContentForm,
-)
+from zds.tutorialv2.forms import ContentForm, FormWithTitle
from zds.tutorialv2.mixins import (
SingleContentFormViewMixin,
SingleContentViewMixin,
@@ -32,6 +28,7 @@
from zds.tutorialv2.models.database import PublishableContent, Validation
from zds.tutorialv2.utils import init_new_repo
from zds.tutorialv2.views.authors import RemoveAuthorFromContent
+from zds.utils.forms import IncludeEasyMDE
from zds.utils.models import get_hat_from_settings
from zds.mp.utils import send_mp, send_message_mp
from zds.utils.uuslug_wrapper import slugify
@@ -40,25 +37,15 @@
logger = logging.getLogger(__name__)
-class CreateContent(LoggedWithReadWriteHability, FormWithPreview):
- """
- Handle content creation. Since v22 a licence must be explicitly selected
- instead of defaulting to "All rights reserved". Users can however
- set a default licence in their profile.
- """
-
+class CreateContentView(LoggedWithReadWriteHability, FormView):
template_name = "tutorialv2/create/content.html"
model = PublishableContent
form_class = ContentForm
- content = None
- created_content_type = "TUTORIAL"
def get_form(self, form_class=ContentForm):
form = super().get_form(form_class)
content_type = self.kwargs["created_content_type"]
- if content_type in CONTENT_TYPE_LIST:
- self.created_content_type = content_type
- form.initial["type"] = self.created_content_type
+ form.initial["type"] = content_type if content_type in CONTENT_TYPE_LIST else "TUTORIAL"
return form
def get_context_data(self, **kwargs):
@@ -88,12 +75,7 @@ def form_valid(self, form):
self.content.ensure_author_gallery()
# Create a new git repository
- init_new_repo(
- self.content,
- form.cleaned_data["introduction"],
- form.cleaned_data["conclusion"],
- form.cleaned_data["msg_commit"],
- )
+ init_new_repo(self.content)
return super().form_valid(form)
@@ -101,62 +83,6 @@ def get_success_url(self):
return reverse("content:view", args=[self.content.pk, self.content.slug])
-class EditContent(LoggedWithReadWriteHability, SingleContentFormViewMixin, FormWithPreview):
- template_name = "tutorialv2/edit/content.html"
- model = PublishableContent
- form_class = EditContentForm
-
- def get_initial(self):
- """rewrite function to pre-populate form"""
- initial = super().get_initial()
- versioned = self.versioned_object
-
- initial["introduction"] = versioned.get_introduction()
- initial["conclusion"] = versioned.get_conclusion()
- initial["subcategory"] = self.object.subcategory.all()
- initial["last_hash"] = versioned.compute_hash()
-
- return initial
-
- def get_context_data(self, **kwargs):
- context = super().get_context_data(**kwargs)
- if "preview" not in self.request.POST:
- context["gallery"] = self.object.gallery
-
- return context
-
- def form_valid(self, form):
- versioned = self.versioned_object
- publishable = self.object
-
- # check if content has changed:
- current_hash = versioned.compute_hash()
- if current_hash != form.cleaned_data["last_hash"]:
- data = form.data.copy()
- data["last_hash"] = current_hash
- data["introduction"] = versioned.get_introduction()
- data["conclusion"] = versioned.get_conclusion()
- form.data = data
- messages.error(self.request, _("Une nouvelle version a été postée avant que vous ne validiez."))
- return self.form_invalid(form)
-
- publishable.update_date = datetime.now()
-
- # now, update the versioned information
- sha = versioned.repo_update_top_container(
- publishable.title,
- publishable.slug,
- form.cleaned_data["introduction"],
- form.cleaned_data["conclusion"],
- form.cleaned_data["msg_commit"],
- )
- publishable.sha_draft = sha
- publishable.save()
-
- self.success_url = reverse("content:view", args=[publishable.pk, publishable.slug])
- return super().form_valid(form)
-
-
class EditTitleForm(FormWithTitle):
def __init__(self, versioned_content, *args, **kwargs):
kwargs["initial"] = {"title": versioned_content.title}
@@ -313,6 +239,198 @@ def update_sha_draft(publishable_content, sha):
publishable_content.save()
+class EditIntroductionForm(forms.Form):
+ introduction = CharField(
+ label=_("Introduction"),
+ required=False,
+ widget=Textarea(
+ attrs={"placeholder": _("Votre introduction, au format Markdown."), "class": "md-editor preview-source"}
+ ),
+ )
+
+ commit_message = CharField(
+ label=_("Message de suivi"),
+ max_length=400,
+ required=False,
+ widget=TextInput(attrs={"placeholder": _("Un résumé de vos ajouts et modifications.")}),
+ )
+
+ def __init__(self, versioned_content, *args, **kwargs):
+ super().__init__(*args, **kwargs)
+
+ self.helper = FormHelper()
+ self.helper.form_class = "content-wrapper"
+ self.helper.form_method = "post"
+ self.helper.form_action = reverse("content:edit-introduction", kwargs={"pk": versioned_content.pk})
+ self.helper.layout = Layout(
+ IncludeEasyMDE(),
+ Field("introduction"),
+ ButtonHolder(
+ StrictButton(_("Aperçu"), type="preview", name="preview", css_class="btn btn-grey preview-btn"),
+ ),
+ HTML(
+ """{% if form.introduction.value %}{% include "misc/preview.part.html" with text=form.introduction.value %}{% endif %}"""
+ ),
+ Field("commit_message"),
+ ButtonHolder(StrictButton(_("Modifier"), type="submit")),
+ )
+
+
+class EditIntroductionView(LoginRequiredMixin, SingleContentFormViewMixin, FormWithPreview):
+ model = PublishableContent
+ form_class = EditIntroductionForm
+ template_name = "tutorialv2/edit/introduction.html"
+ success_message = _("L'introduction de la publication a bien été changée.")
+
+ def get_context_data(self, **kwargs):
+ context = super().get_context_data(**kwargs)
+ if "preview" not in self.request.POST:
+ context["gallery"] = self.object.gallery
+ return context
+
+ def get_initial(self):
+ initial = super().get_initial()
+ introduction = self.versioned_object.get_introduction()
+ initial["introduction"] = introduction
+ return initial
+
+ def get_form_kwargs(self):
+ kwargs = super().get_form_kwargs()
+ kwargs["versioned_content"] = self.versioned_object
+ return kwargs
+
+ def form_valid(self, form):
+ publishable = self.object
+ versioned = self.versioned_object
+
+ commit_message = "Modification de l'introduction"
+ if form.cleaned_data["commit_message"] != "":
+ commit_message = form.cleaned_data["commit_message"]
+
+ sha = self.update_introduction_in_repository(
+ publishable, versioned, form.cleaned_data["introduction"], commit_message
+ )
+ self.update_sha_draft(publishable, sha)
+
+ messages.success(self.request, self.success_message)
+ return super().form_valid(form)
+
+ def get_success_url(self):
+ return reverse("content:view", args=[self.object.pk, self.object.slug])
+
+ @staticmethod
+ def update_introduction_in_repository(publishable_content, versioned_content, introduction, commit_message):
+ sha = versioned_content.repo_update_top_container(
+ publishable_content.title,
+ publishable_content.slug,
+ introduction,
+ versioned_content.get_conclusion(),
+ commit_message,
+ )
+ return sha
+
+ @staticmethod
+ def update_sha_draft(publishable_content, sha):
+ publishable_content.sha_draft = sha
+ publishable_content.save()
+
+
+class EditConclusionForm(forms.Form):
+ conclusion = CharField(
+ label=_("Conclusion"),
+ required=False,
+ widget=Textarea(
+ attrs={"placeholder": _("Votre conclusion, au format Markdown."), "class": "md-editor preview-source"}
+ ),
+ )
+
+ commit_message = CharField(
+ label=_("Message de suivi"),
+ max_length=400,
+ required=False,
+ widget=TextInput(attrs={"placeholder": _("Un résumé de vos ajouts et modifications.")}),
+ )
+
+ def __init__(self, versioned_content, *args, **kwargs):
+ super().__init__(*args, **kwargs)
+
+ self.helper = FormHelper()
+ self.helper.form_class = "content-wrapper"
+ self.helper.form_method = "post"
+ self.helper.form_action = reverse("content:edit-conclusion", kwargs={"pk": versioned_content.pk})
+ self.helper.layout = Layout(
+ IncludeEasyMDE(),
+ Field("conclusion"),
+ ButtonHolder(
+ StrictButton(_("Aperçu"), type="preview", name="preview", css_class="btn btn-grey preview-btn")
+ ),
+ HTML(
+ """{% if form.conclusion.value %}{% include "misc/preview.part.html" with text=form.conclusion.value %}{% endif %}"""
+ ),
+ Field("commit_message"),
+ ButtonHolder(StrictButton(_("Modifier"), type="submit")),
+ )
+
+
+class EditConclusionView(LoginRequiredMixin, SingleContentFormViewMixin, FormWithPreview):
+ model = PublishableContent
+ form_class = EditConclusionForm
+ template_name = "tutorialv2/edit/conclusion.html"
+ success_message = _("La conclusion de la publication a bien été changée.")
+
+ def get_context_data(self, **kwargs):
+ context = super().get_context_data(**kwargs)
+ if "preview" not in self.request.POST:
+ context["gallery"] = self.object.gallery
+ return context
+
+ def get_initial(self):
+ initial = super().get_initial()
+ conclusion = self.versioned_object.get_conclusion()
+ initial["conclusion"] = conclusion
+ return initial
+
+ def get_form_kwargs(self):
+ kwargs = super().get_form_kwargs()
+ kwargs["versioned_content"] = self.versioned_object
+ return kwargs
+
+ def form_valid(self, form):
+ publishable = self.object
+ versioned = self.versioned_object
+
+ commit_message = "Modification de la conclusion"
+ if form.cleaned_data["commit_message"] != "":
+ commit_message = form.cleaned_data["commit_message"]
+
+ sha = self.update_conclusion_in_repository(
+ publishable, versioned, form.cleaned_data["conclusion"], commit_message
+ )
+ self.update_sha_draft(publishable, sha)
+
+ messages.success(self.request, self.success_message)
+ return super().form_valid(form)
+
+ def get_success_url(self):
+ return reverse("content:view", args=[self.object.pk, self.object.slug])
+
+ @staticmethod
+ def update_conclusion_in_repository(publishable_content, versioned_content, conclusion, commit_message):
+ sha = versioned_content.repo_update_top_container(
+ publishable_content.title,
+ publishable_content.slug,
+ versioned_content.get_introduction(),
+ conclusion,
+ commit_message,
+ )
+ return sha
+
+ @staticmethod
+ def update_sha_draft(publishable_content, sha):
+ publishable_content.sha_draft = sha
+ publishable_content.save()
+
+
class DeleteContent(LoginRequiredMixin, SingleContentViewMixin, DeleteView):
model = PublishableContent
http_method_names = ["post"]