mirror of
https://code.eliotberriot.com/funkwhale/funkwhale.git
synced 2025-10-06 01:49:56 +02:00
See #432: can now start a radio from a tag
This commit is contained in:
parent
3ce2f0d774
commit
918e7beb7e
6 changed files with 95 additions and 8 deletions
|
@ -2,6 +2,7 @@ import random
|
|||
|
||||
from django.core.exceptions import ValidationError
|
||||
from django.db import connection
|
||||
from django.db.models import Q
|
||||
from rest_framework import serializers
|
||||
|
||||
from funkwhale_api.moderation import filters as moderation_filters
|
||||
|
@ -14,6 +15,8 @@ from .registries import registry
|
|||
|
||||
|
||||
class SimpleRadio(object):
|
||||
related_object_field = None
|
||||
|
||||
def clean(self, instance):
|
||||
return
|
||||
|
||||
|
@ -146,6 +149,8 @@ class CustomRadio(SessionRadio):
|
|||
class RelatedObjectRadio(SessionRadio):
|
||||
"""Abstract radio related to an object (tag, artist, user...)"""
|
||||
|
||||
related_object_field = serializers.IntegerField(required=True)
|
||||
|
||||
def clean(self, instance):
|
||||
super().clean(instance)
|
||||
if not instance.related_object:
|
||||
|
@ -162,10 +167,22 @@ class RelatedObjectRadio(SessionRadio):
|
|||
@registry.register(name="tag")
|
||||
class TagRadio(RelatedObjectRadio):
|
||||
model = Tag
|
||||
related_object_field = serializers.CharField(required=True)
|
||||
|
||||
def get_related_object(self, name):
|
||||
return self.model.objects.get(name=name)
|
||||
|
||||
def get_queryset(self, **kwargs):
|
||||
qs = super().get_queryset(**kwargs)
|
||||
return qs.filter(tagged_items__tag=self.session.related_object)
|
||||
query = (
|
||||
Q(tagged_items__tag=self.session.related_object)
|
||||
| Q(artist__tagged_items__tag=self.session.related_object)
|
||||
| Q(album__tagged_items__tag=self.session.related_object)
|
||||
)
|
||||
return qs.filter(query)
|
||||
|
||||
def get_related_object_id_repr(self, obj):
|
||||
return obj.name
|
||||
|
||||
|
||||
def weighted_choice(choices):
|
||||
|
|
|
@ -54,6 +54,9 @@ class RadioSessionTrackSerializer(serializers.ModelSerializer):
|
|||
|
||||
|
||||
class RadioSessionSerializer(serializers.ModelSerializer):
|
||||
|
||||
related_object_id = serializers.CharField(required=False, allow_null=True)
|
||||
|
||||
class Meta:
|
||||
model = models.RadioSession
|
||||
fields = (
|
||||
|
@ -66,7 +69,17 @@ class RadioSessionSerializer(serializers.ModelSerializer):
|
|||
)
|
||||
|
||||
def validate(self, data):
|
||||
registry[data["radio_type"]]().validate_session(data, **self.context)
|
||||
radio_conf = registry[data["radio_type"]]()
|
||||
if radio_conf.related_object_field:
|
||||
try:
|
||||
data[
|
||||
"related_object_id"
|
||||
] = radio_conf.related_object_field.to_internal_value(
|
||||
data["related_object_id"]
|
||||
)
|
||||
except KeyError:
|
||||
raise serializers.ValidationError("Radio requires a related object")
|
||||
radio_conf.validate_session(data, **self.context)
|
||||
return data
|
||||
|
||||
def create(self, validated_data):
|
||||
|
@ -77,3 +90,11 @@ class RadioSessionSerializer(serializers.ModelSerializer):
|
|||
validated_data["related_object_id"]
|
||||
)
|
||||
return super().create(validated_data)
|
||||
|
||||
def to_representation(self, instance):
|
||||
repr = super().to_representation(instance)
|
||||
radio_conf = registry[repr["radio_type"]]()
|
||||
handler = getattr(radio_conf, "get_related_object_id_repr", None)
|
||||
if handler and instance.related_object:
|
||||
repr["related_object_id"] = handler(instance.related_object)
|
||||
return repr
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue