From 16a1f3dd5d1e8fa2a38f864d9bcce749e9a7ec32 Mon Sep 17 00:00:00 2001 From: charludo Date: Wed, 8 Jan 2025 08:37:57 +0100 Subject: [PATCH] Allow users to select shown contact details --- integreat_cms/cms/models/contact/contact.py | 28 +++++++ .../cms/templates/contacts/contact_card.html | 76 +++++++++-------- integreat_cms/cms/utils/content_utils.py | 16 +++- .../cms/views/utils/contact_utils.py | 19 ++++- integreat_cms/locale/de/LC_MESSAGES/django.po | 24 ++++++ .../custom_contact_input/plugin.js | 84 ++++++++++++++++--- 6 files changed, 194 insertions(+), 53 deletions(-) diff --git a/integreat_cms/cms/models/contact/contact.py b/integreat_cms/cms/models/contact/contact.py index d21450c22d..6008f81901 100644 --- a/integreat_cms/cms/models/contact/contact.py +++ b/integreat_cms/cms/models/contact/contact.py @@ -236,6 +236,34 @@ def referring_objects(self) -> List[AbstractContentTranslation]: for link in Link.objects.filter(url__url=self.absolute_url) ] + @cached_property + def available_details(self) -> dict: + """ + Returns the available details and their human-readable representation + + :return: key-value pairs of available detail and human-readable representation + """ + details = { + "address": _("show address"), + } + + if self.point_of_contact_for: + details["point_of_contact_for"] = _("show point of contact") + + if self.name: + details["name"] = _("show name") + + if self.email: + details["email"] = _("show email") + + if self.phone_number: + details["phone_number"] = _("show phone number") + + if self.website: + details["website"] = _("show website") + + return details + def archive(self) -> None: """ Archives the contact diff --git a/integreat_cms/cms/templates/contacts/contact_card.html b/integreat_cms/cms/templates/contacts/contact_card.html index afe0de11f0..6c64cdaf1e 100644 --- a/integreat_cms/cms/templates/contacts/contact_card.html +++ b/integreat_cms/cms/templates/contacts/contact_card.html @@ -3,52 +3,54 @@ {% spaceless %}
{# djlint:off H021 #} - Contact + Contact {# djlint:on #} - {% if contact %} - {% if contact.name or contact.point_of_contact_for %} -

- {% if contact.name %} - {{ contact.name }} - {% endif %} - {% if contact.name and contact.point_of_contact_for %} - | - {% endif %} - {% if contact.point_of_contact_for %} - {{ contact.point_of_contact_for }} - {% endif %} -

- {% endif %} + {# "and" takes precedent: https://docs.djangoproject.com/en/4.2/ref/templates/builtins/#boolean-operators #} + {% if contact.name and "name" in wanted or contact.point_of_contact_for and "point_of_contact_for" in wanted %} +

+ {% if contact.name and "name" in wanted %} + {{ contact.name }} + {% endif %} + {% if contact.name and "name" in wanted and contact.point_of_contact_for and "point_of_contact_for" in wanted %} + | + {% endif %} + {% if contact.point_of_contact_for and "point_of_contact_for" in wanted %} + {{ contact.point_of_contact_for }} + {% endif %} +

+ {% endif %} + {% if "address" in wanted %}

Address: {{ contact.location.short_address }}

- {% if contact.email %} -

- Email: - {{ contact.email }} -

- {% endif %} - {% if contact.phone_number %} -

- Phone Number: - {{ contact.phone_number }} -

- {% endif %} - {% if contact.website %} -

- Website: - {{ contact.website }} -

- {% endif %} + {% endif %} + {% if contact.email and "email" in wanted %} +

+ Email: + {{ contact.email }} +

+ {% endif %} + {% if contact.phone_number and "phone_number" in wanted %} +

+ Phone Number: + {{ contact.phone_number }} +

+ {% endif %} + {% if contact.website and "website" in wanted %} +

+ Website: + {{ contact.website }} +

{% endif %}
{% endspaceless %} diff --git a/integreat_cms/cms/utils/content_utils.py b/integreat_cms/cms/utils/content_utils.py index 167b481b3d..1906a889ae 100644 --- a/integreat_cms/cms/utils/content_utils.py +++ b/integreat_cms/cms/utils/content_utils.py @@ -142,16 +142,18 @@ def update_internal_links(link: HtmlElement, language_slug: str) -> None: link.append(new_html) -def render_contact_card(contact_id: int) -> HtmlElement: +def render_contact_card(contact_id: int, wanted_details: list[str]) -> HtmlElement: """ Produces a rendered html element for the contact. :param contact_id: The id of the contact to render the card for + :param wanted_details: list of details to be shown in the rendered card """ template = loader.get_template("contacts/contact_card.html") try: context = { "contact": Contact.objects.get(pk=contact_id), + "wanted": wanted_details, } raw_element = template.render(context) return fromstring(raw_element) @@ -175,9 +177,17 @@ def update_contacts(content: HtmlElement) -> None: """ contact_cards = content.xpath("//div[@data-contact-id]") contact_ids = [int(card.get("data-contact-id")) for card in contact_cards] + contact_urls = [card.get("data-contact-url") for card in contact_cards] - for contact_id, contact_card in zip(contact_ids, contact_cards): - contact_card_new = render_contact_card(contact_id) + for contact_id, contact_url, contact_card in zip( + contact_ids, contact_urls, contact_cards + ): + try: + wanted_details = contact_url.split("details=", 1)[1].split(",") + except IndexError: + wanted_details = [] + + contact_card_new = render_contact_card(contact_id, wanted_details) contact_card.getparent().replace(contact_card, contact_card_new) diff --git a/integreat_cms/cms/views/utils/contact_utils.py b/integreat_cms/cms/views/utils/contact_utils.py index b35ac315a0..be103ead84 100644 --- a/integreat_cms/cms/views/utils/contact_utils.py +++ b/integreat_cms/cms/views/utils/contact_utils.py @@ -5,7 +5,7 @@ from typing import TYPE_CHECKING from django.core.exceptions import PermissionDenied -from django.http import HttpResponse, JsonResponse +from django.http import JsonResponse from django.shortcuts import get_object_or_404, render from django.views.decorators.http import require_POST @@ -56,6 +56,7 @@ def search_contact_ajax( { "url": result.absolute_url, "name": result.get_repr_short, + "details": result.available_details, } for result in results ] @@ -77,7 +78,11 @@ def get_contact( """ # pylint: disable=unused-argument contact = get_object_or_404(Contact, pk=contact_id) - return render(request, "contacts/contact_card.html", {"contact": contact}) + + wanted = request.GET.get("details", "").split(",") + return render( + request, "contacts/contact_card.html", {"contact": contact, "wanted": wanted} + ) @permission_required("cms.view_contact") @@ -95,4 +100,12 @@ def get_contact_raw( """ # pylint: disable=unused-argument contact = get_object_or_404(Contact, pk=contact_id) - return HttpResponse(contact.get_repr_short) + return JsonResponse( + { + "data": { + "url": contact.full_url, + "name": contact.get_repr_short, + "details": contact.available_details, + } + } + ) diff --git a/integreat_cms/locale/de/LC_MESSAGES/django.po b/integreat_cms/locale/de/LC_MESSAGES/django.po index 88ad2f3981..7591c86041 100644 --- a/integreat_cms/locale/de/LC_MESSAGES/django.po +++ b/integreat_cms/locale/de/LC_MESSAGES/django.po @@ -2817,6 +2817,30 @@ msgstr "mit Website: {}" msgid "General contact information" msgstr "Allgemeine Kontaktinformationen" +#: cms/models/contact/contact.py +msgid "show address" +msgstr "Adresse anzeigen" + +#: cms/models/contact/contact.py +msgid "show point of contact" +msgstr "Ansprechperson anzeigen" + +#: cms/models/contact/contact.py +msgid "show name" +msgstr "Name anzeigen" + +#: cms/models/contact/contact.py +msgid "show email" +msgstr "Email anzeigen" + +#: cms/models/contact/contact.py +msgid "show phone number" +msgstr "Telefonnummer anzeigen" + +#: cms/models/contact/contact.py +msgid "show website" +msgstr "Webseite anzeigen" + #: cms/models/contact/contact.py msgid "(Copy)" msgstr "(Kopie)" diff --git a/integreat_cms/static/src/js/tinymce-plugins/custom_contact_input/plugin.js b/integreat_cms/static/src/js/tinymce-plugins/custom_contact_input/plugin.js index 26afbbe077..18682bd9f0 100644 --- a/integreat_cms/static/src/js/tinymce-plugins/custom_contact_input/plugin.js +++ b/integreat_cms/static/src/js/tinymce-plugins/custom_contact_input/plugin.js @@ -39,7 +39,21 @@ import { getCsrfToken } from "../../utils/csrf-token"; return data; }; - const getContactRaw = async (url) => getContactHtml(`${url}raw/`); + const getContactRaw = async (url) => { + const response = await fetch(`${url.split("?")[0]}raw/`, { + method: "GET", + headers: { + "X-CSRFToken": getCsrfToken(), + }, + }); + + if (response.status !== HTTP_STATUS_OK) { + return {}; + } + + const data = await response.json(); + return data.data; + }; tinymce.PluginManager.add("custom_contact_input", (editor, _url) => { const isContact = (node) => "contactId" in node.dataset; @@ -58,7 +72,7 @@ import { getCsrfToken } from "../../utils/csrf-token"; let tomSelectInstance; - const openDialog = (_button, initialSelection) => { + const openDialog = (_button, initialSelection, selectedDetails) => { const contact = getContact(); const dialogConfig = { @@ -72,6 +86,7 @@ import { getCsrfToken } from "../../utils/csrf-token"; // so we have to do with an htmlpanel and initializing TomSelect separately html: `