#132 | #121: backoffice / dev-1.0-121 (#131)

cfr #121

Co-authored-by: Christophe Siraut <d@tobald.eu.org>
Co-authored-by: bkfox <thomas bkfox net>
Co-authored-by: Thomas Kairos <thomas@bkfox.net>
Reviewed-on: rc/aircox#131
Co-authored-by: Chris Tactic <ctactic@noreply.git.radiocampus.be>
Co-committed-by: Chris Tactic <ctactic@noreply.git.radiocampus.be>
This commit is contained in:
2024-04-28 22:02:09 +02:00
committed by Thomas Kairos
parent 1e17a1334a
commit 55123c386d
348 changed files with 124397 additions and 17879 deletions

23
aircox/forms/__init__.py Normal file
View File

@ -0,0 +1,23 @@
from . import widgets
from .episode import EpisodeForm, EpisodeSoundFormSet
from .program import ProgramForm
from .page import CommentForm, ImageForm, PageForm, ChildPageForm
from .sound import SoundForm, SoundCreateForm
from .track import TrackFormSet
__all__ = (
widgets,
# ---- forms
EpisodeForm,
EpisodeSoundFormSet,
ProgramForm,
CommentForm,
ImageForm,
PageForm,
ChildPageForm,
SoundForm,
SoundCreateForm,
TrackFormSet,
)

34
aircox/forms/episode.py Normal file
View File

@ -0,0 +1,34 @@
from django import forms
from django.forms.models import modelformset_factory
from aircox import models
from .page import ChildPageForm
__all__ = ("EpisodeForm", "EpisodeSoundFormSet")
class EpisodeForm(ChildPageForm):
class Meta:
model = models.Episode
fields = ChildPageForm.Meta.fields
EpisodeSoundFormSet = modelformset_factory(
models.EpisodeSound,
fields=(
"position",
"episode",
"sound",
"broadcast",
),
widgets={
"broadcast": forms.CheckboxInput(),
"episode": forms.HiddenInput(),
# "sound": forms.HiddenInput(),
"position": forms.HiddenInput(),
},
can_delete=True,
extra=0,
)
"""Formset used in EpisodeUpdateView."""

37
aircox/forms/page.py Normal file
View File

@ -0,0 +1,37 @@
from django import forms
from aircox import models
__all__ = ("CommentForm", "ImageForm", "PageForm", "ChildPageForm")
class CommentForm(forms.ModelForm):
nickname = forms.CharField()
email = forms.EmailField(required=False)
content = forms.CharField(widget=forms.Textarea())
nickname.widget.attrs.update({"class": "input"})
email.widget.attrs.update({"class": "input"})
content.widget.attrs.update({"class": "textarea"})
class Meta:
model = models.Comment
fields = ["nickname", "email", "content"]
class ImageForm(forms.Form):
file = forms.ImageField()
class PageForm(forms.ModelForm):
class Meta:
fields = ("title", "category", "status", "cover", "content")
model = models.Page
class ChildPageForm(forms.ModelForm):
class Meta:
fields = ("title", "status", "cover", "content")
model = models.Page

11
aircox/forms/program.py Normal file
View File

@ -0,0 +1,11 @@
from aircox import models
from .page import PageForm
__all__ = ("ProgramForm",)
class ProgramForm(PageForm):
class Meta:
fields = PageForm.Meta.fields
model = models.Program

26
aircox/forms/sound.py Normal file
View File

@ -0,0 +1,26 @@
from django import forms
from aircox import models
__all__ = (
"SoundForm",
"SoundCreateForm",
)
class SoundForm(forms.ModelForm):
"""SoundForm used in EpisodeUpdateView."""
class Meta:
model = models.Sound
fields = ["name", "program", "file", "broadcast", "duration", "is_public", "is_downloadable"]
class SoundCreateForm(forms.ModelForm):
"""SoundForm used in EpisodeUpdateView."""
class Meta:
model = models.Sound
fields = ["name", "program", "file", "broadcast", "is_public", "is_downloadable"]
widgets = {"program": forms.HiddenInput()}

23
aircox/forms/track.py Normal file
View File

@ -0,0 +1,23 @@
from django import forms
from django.forms.models import modelformset_factory
from aircox import models
__all__ = ("TrackFormSet",)
TrackFormSet = modelformset_factory(
models.Track,
fields=[
"position",
"episode",
"artist",
"title",
"tags",
],
widgets={"episode": forms.HiddenInput(), "position": forms.HiddenInput()},
can_delete=True,
extra=0,
)
"""Track formset used in EpisodeUpdateView."""

89
aircox/forms/widgets.py Normal file
View File

@ -0,0 +1,89 @@
from itertools import chain
from functools import cached_property
from django import forms, http
from django.urls import reverse
__all__ = (
"VueWidget",
"VueAutoComplete",
)
class VueWidget(forms.Widget):
binds = None
"""Dict of `{attribute: value}` attrs set as bindings."""
events = None
"""Dict of `{event: value}` attrs set as events."""
v_model = ""
"""ES6 Model instance to bind to (`v-model`)."""
def __init__(self, *args, binds=None, events=None, v_model=None, **kwargs):
super().__init__(*args, **kwargs)
self.binds = binds or []
self.events = events or []
@cached_property
def vue_attrs(self):
"""Dict of Vue specific attributes."""
binds, events = self.binds, self.events
if isinstance(binds, dict):
binds = binds.items()
if isinstance(events, dict):
events = events.items()
return dict(
chain(
((":" + key, value) for key, value in binds),
(("@" + key, value) for key, value in events),
)
)
def build_attrs(self, base_attrs, extra_attrs=None):
extra_attrs = extra_attrs or {}
extra_attrs.update(self.vue_attrs)
return super().build_attrs(base_attrs, extra_attrs)
class VueAutoComplete(VueWidget, forms.TextInput):
"""Autocomplete Vue component."""
template_name = "aircox/widgets/autocomplete.html"
url: str = ""
"""Url to autocomplete API view.
If it has query parameters, does not generate it based on lookup
(see `get_url()` doc).
"""
lookup: str = ""
"""Field name used as lookup (instead as provided one)."""
params: http.QueryDict
def __init__(self, url_name, *args, lookup=None, params=None, **kwargs):
self.url_name = url_name
self.lookup = lookup
self.params = params
super().__init__(*args, **kwargs)
def get_context(self, name, value, attrs):
context = super().get_context(name, value, attrs)
context["url"] = self.get_url(name, self.lookup, self.params)
return context
def get_url(self, name, lookup, params=None):
"""Return url to autocomplete API. When query parameters are not
provided generate them using `?{lookup}=${query}&field={name}` (where
`${query} is Vue `a-autocomplete` specific).
:param str name: field name (not used by default)
:param str lookup: lookup query parameter
:param http.QueryDict params: additional mutable parameter
"""
url = reverse(self.url_name)
query = http.QueryDict(mutable=True)
if params:
query.update(params)
query.update({lookup: "${query}"})
return f"{url}?{query.urlencode()}"