2025-06-13 17:10:56 +02:00
|
|
|
from django.contrib.auth.models import User
|
|
|
|
from django.db import models
|
2025-06-14 10:35:42 +02:00
|
|
|
from django.db.models.functions import Lower
|
2025-06-14 16:22:25 +02:00
|
|
|
from django.db.models.signals import post_delete, post_save
|
2025-06-14 16:03:51 +02:00
|
|
|
from django.dispatch import receiver
|
2025-06-13 17:10:56 +02:00
|
|
|
from django.urls import reverse
|
|
|
|
|
2025-06-14 16:22:25 +02:00
|
|
|
from . import tasks
|
|
|
|
|
2025-06-13 17:10:56 +02:00
|
|
|
|
2025-06-13 22:23:18 +02:00
|
|
|
class YoutubeCredentials(models.Model):
|
|
|
|
user = models.OneToOneField(User, on_delete=models.CASCADE)
|
|
|
|
credentials = models.JSONField()
|
2025-06-15 10:25:40 +02:00
|
|
|
title = models.CharField(blank=True)
|
2025-06-13 22:23:18 +02:00
|
|
|
|
|
|
|
|
2025-06-13 17:10:56 +02:00
|
|
|
class Group(models.Model):
|
|
|
|
name = models.CharField(verbose_name="Nom du groupe")
|
|
|
|
owner = models.ForeignKey(
|
|
|
|
User, on_delete=models.CASCADE, related_name="owned_group_set"
|
|
|
|
)
|
|
|
|
members = models.ManyToManyField(User, blank=True)
|
|
|
|
|
|
|
|
def get_absolute_url(self):
|
|
|
|
return reverse("group_detail", kwargs={"pk": self.pk})
|
|
|
|
|
2025-06-14 21:58:33 +02:00
|
|
|
def is_owner(self, user):
|
|
|
|
return user == self.owner
|
|
|
|
|
|
|
|
def is_leader(self, user):
|
|
|
|
return (
|
|
|
|
self.is_owner(user)
|
|
|
|
or self.members.through.objects.filter(
|
|
|
|
lead__is_leader=True, user=user
|
|
|
|
).exists()
|
|
|
|
)
|
|
|
|
|
|
|
|
def is_member(self, user):
|
|
|
|
return self.is_owner(user) or self.members.filter(user=user).exists()
|
|
|
|
|
2025-06-13 17:10:56 +02:00
|
|
|
class Meta:
|
2025-06-14 10:35:42 +02:00
|
|
|
constraints = [
|
|
|
|
models.UniqueConstraint(Lower("name"), "owner", name="unique_group_name")
|
|
|
|
]
|
2025-06-13 17:10:56 +02:00
|
|
|
|
|
|
|
|
2025-06-14 21:58:33 +02:00
|
|
|
class GroupLeader(models.Model):
|
|
|
|
member = models.OneToOneField(
|
|
|
|
Group.members.through, on_delete=models.CASCADE, related_name="lead"
|
|
|
|
)
|
|
|
|
is_leader = models.BooleanField(default=False)
|
|
|
|
|
|
|
|
|
2025-06-13 17:10:56 +02:00
|
|
|
class MusicVideo(models.Model):
|
|
|
|
yt_id = models.CharField(max_length=16)
|
2025-06-13 18:55:50 +02:00
|
|
|
title = models.CharField(blank=True)
|
2025-06-13 21:06:23 +02:00
|
|
|
date_added = models.DateTimeField(auto_now_add=True)
|
2025-06-13 18:55:50 +02:00
|
|
|
owner = models.ForeignKey(User, on_delete=models.CASCADE)
|
2025-06-13 17:10:56 +02:00
|
|
|
group = models.ForeignKey(Group, on_delete=models.CASCADE)
|
|
|
|
blacklisted = models.BooleanField(default=False)
|
2025-06-13 18:55:50 +02:00
|
|
|
|
|
|
|
class Meta:
|
2025-06-14 10:35:42 +02:00
|
|
|
constraints = [
|
|
|
|
models.UniqueConstraint(
|
|
|
|
fields=("yt_id", "owner", "group"), name="unique_music_in_group"
|
|
|
|
)
|
|
|
|
]
|
2025-06-13 21:06:23 +02:00
|
|
|
|
|
|
|
|
|
|
|
class MusikGame(models.Model):
|
|
|
|
group = models.ForeignKey(Group, on_delete=models.CASCADE)
|
|
|
|
date = models.DateTimeField(auto_now_add=True)
|
|
|
|
n = models.PositiveIntegerField(default=2, verbose_name="Nombre de musiques")
|
|
|
|
players = models.ManyToManyField(User, verbose_name="Joueurs")
|
2025-06-13 23:08:41 +02:00
|
|
|
playlist = models.CharField(blank=True, verbose_name="Playlist YouTube")
|
2025-06-14 11:49:04 +02:00
|
|
|
playlist_loading = models.BooleanField(default=False)
|
2025-06-13 21:06:23 +02:00
|
|
|
|
|
|
|
def get_absolute_url(self):
|
|
|
|
return reverse("game_detail", kwargs={"pk": self.pk})
|
|
|
|
|
|
|
|
|
2025-06-14 16:22:25 +02:00
|
|
|
@receiver(post_save, sender=MusikGame)
|
|
|
|
def generateYoutubePlaylist(sender, instance, created, **kwargs):
|
|
|
|
if not instance.playlist_loading or instance.playlist:
|
|
|
|
return
|
|
|
|
|
|
|
|
if creds := instance.group.owner.youtubecredentials:
|
|
|
|
tasks.generate_playlist.delay_on_commit(creds.credentials, instance.pk)
|
|
|
|
|
|
|
|
|
2025-06-14 16:03:51 +02:00
|
|
|
@receiver(post_delete, sender=MusikGame)
|
|
|
|
def deleteYoutubePlaylist(sender, instance, using, **kwargs):
|
|
|
|
if not instance.playlist:
|
|
|
|
return
|
|
|
|
|
2025-06-14 16:22:25 +02:00
|
|
|
if creds := instance.group.owner.youtubecredentials:
|
|
|
|
tasks.delete_playlist.delay_on_commit(creds.credentials, instance.playlist)
|
2025-06-14 16:03:51 +02:00
|
|
|
|
|
|
|
|
2025-06-13 21:06:23 +02:00
|
|
|
class MusicGameOrder(models.Model):
|
|
|
|
game = models.ForeignKey(MusikGame, on_delete=models.CASCADE)
|
|
|
|
player = models.ForeignKey(User, on_delete=models.CASCADE)
|
|
|
|
music_video = models.ForeignKey(MusicVideo, on_delete=models.CASCADE)
|
|
|
|
order = models.PositiveIntegerField()
|
|
|
|
|
|
|
|
class Meta:
|
2025-06-14 10:35:42 +02:00
|
|
|
constraints = [
|
|
|
|
models.UniqueConstraint(
|
|
|
|
fields=("game", "player", "music_video"), name="unique_music_in_game"
|
|
|
|
),
|
|
|
|
models.UniqueConstraint(fields=("game", "order"), name="unique_order"),
|
|
|
|
]
|
2025-06-13 21:06:23 +02:00
|
|
|
ordering = ["order"]
|
2025-06-15 12:04:28 +02:00
|
|
|
|
|
|
|
|
|
|
|
class MusicGameAnswer(models.Model):
|
|
|
|
game = models.ForeignKey(MusikGame, on_delete=models.CASCADE)
|
|
|
|
player = models.ForeignKey(User, on_delete=models.CASCADE)
|
|
|
|
order = models.PositiveIntegerField()
|
|
|
|
answer = models.ForeignKey(
|
|
|
|
User, on_delete=models.SET_NULL, null=True, related_name="+"
|
|
|
|
)
|
|
|
|
|
|
|
|
class Meta:
|
|
|
|
constraints = [
|
|
|
|
models.UniqueConstraint(
|
|
|
|
fields=("game", "player", "order"), name="unique_answer"
|
|
|
|
),
|
|
|
|
]
|
|
|
|
ordering = ["order"]
|