forked from rc/aircox
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:
23
aircox/forms/__init__.py
Normal file
23
aircox/forms/__init__.py
Normal 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
34
aircox/forms/episode.py
Normal 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
37
aircox/forms/page.py
Normal 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
11
aircox/forms/program.py
Normal 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
26
aircox/forms/sound.py
Normal 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
23
aircox/forms/track.py
Normal 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
89
aircox/forms/widgets.py
Normal 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()}"
|
Reference in New Issue
Block a user