mirror of
https://github.com/mediacms-io/mediacms.git
synced 2025-11-21 13:57:57 -05:00
166 lines
5.5 KiB
Python
166 lines
5.5 KiB
Python
import uuid
|
|
|
|
from django.db import models
|
|
from django.db.models.signals import post_save, pre_delete
|
|
from django.dispatch import receiver
|
|
from django.urls import reverse
|
|
from django.utils.html import strip_tags
|
|
|
|
from .. import helpers
|
|
|
|
|
|
class Playlist(models.Model):
|
|
"""Playlists model"""
|
|
|
|
add_date = models.DateTimeField(auto_now_add=True, db_index=True)
|
|
|
|
description = models.TextField(blank=True, help_text="description")
|
|
|
|
friendly_token = models.CharField(blank=True, max_length=12, db_index=True)
|
|
|
|
media = models.ManyToManyField("Media", through="playlistmedia", blank=True)
|
|
|
|
title = models.CharField(max_length=100, db_index=True)
|
|
|
|
uid = models.UUIDField(unique=True, default=uuid.uuid4)
|
|
|
|
user = models.ForeignKey("users.User", on_delete=models.CASCADE, db_index=True, related_name="playlists")
|
|
|
|
def __str__(self):
|
|
return self.title
|
|
|
|
@property
|
|
def media_count(self):
|
|
return self.media.filter(listable=True).count()
|
|
|
|
def get_first_media(self):
|
|
"""Get the first media item in the playlist"""
|
|
pm = self.playlistmedia_set.filter(media__listable=True).first()
|
|
return pm.media if pm else None
|
|
|
|
def get_absolute_url(self, api=False, start_playback=False):
|
|
"""
|
|
Get the URL for this playlist.
|
|
|
|
Args:
|
|
api: If True, return API URL
|
|
start_playback: If True, return URL to first media with playlist context
|
|
"""
|
|
if start_playback and not api:
|
|
# Get first media and return its URL with playlist parameter
|
|
first_media = self.get_first_media()
|
|
if first_media:
|
|
return f"{first_media.get_absolute_url()}&pl={self.friendly_token}"
|
|
|
|
if api:
|
|
return reverse("api_get_playlist", kwargs={"friendly_token": self.friendly_token})
|
|
else:
|
|
return reverse("get_playlist", kwargs={"friendly_token": self.friendly_token})
|
|
|
|
@property
|
|
def url(self):
|
|
return self.get_absolute_url()
|
|
|
|
@property
|
|
def playback_url(self):
|
|
"""URL that starts playing the first media in the playlist"""
|
|
return self.get_absolute_url(start_playback=True)
|
|
|
|
@property
|
|
def api_url(self):
|
|
return self.get_absolute_url(api=True)
|
|
|
|
def user_thumbnail_url(self):
|
|
if self.user.logo:
|
|
return helpers.url_from_path(self.user.logo.path)
|
|
return None
|
|
|
|
def set_ordering(self, media, ordering):
|
|
if media not in self.media.all():
|
|
return False
|
|
pm = PlaylistMedia.objects.filter(playlist=self, media=media).first()
|
|
if pm and isinstance(ordering, int) and 0 < ordering:
|
|
pm.ordering = ordering
|
|
pm.save()
|
|
return True
|
|
return False
|
|
|
|
def save(self, *args, **kwargs):
|
|
strip_text_items = ["title", "description"]
|
|
for item in strip_text_items:
|
|
setattr(self, item, strip_tags(getattr(self, item, None)))
|
|
self.title = self.title[:100]
|
|
|
|
if not self.friendly_token:
|
|
while True:
|
|
friendly_token = helpers.produce_friendly_token()
|
|
if not Playlist.objects.filter(friendly_token=friendly_token):
|
|
self.friendly_token = friendly_token
|
|
break
|
|
super(Playlist, self).save(*args, **kwargs)
|
|
|
|
@property
|
|
def thumbnail_url(self):
|
|
pm = self.playlistmedia_set.filter(media__listable=True).first()
|
|
if pm and pm.media.thumbnail:
|
|
return helpers.url_from_path(pm.media.thumbnail.path)
|
|
return None
|
|
|
|
|
|
class PlaylistMedia(models.Model):
|
|
"""Helper model to store playlist specific media"""
|
|
|
|
action_date = models.DateTimeField(auto_now=True)
|
|
|
|
media = models.ForeignKey("Media", on_delete=models.CASCADE)
|
|
|
|
playlist = models.ForeignKey(Playlist, on_delete=models.CASCADE)
|
|
|
|
ordering = models.IntegerField(default=1)
|
|
|
|
class Meta:
|
|
ordering = ["ordering", "-action_date"]
|
|
|
|
|
|
@receiver(post_save, sender=Playlist)
|
|
def create_or_update_playlist_media(sender, instance, created, **kwargs):
|
|
"""
|
|
Automatically create or update a Media object that represents this Playlist in listings.
|
|
This allows playlists to appear alongside regular media in search results and listings.
|
|
"""
|
|
from .media import Media
|
|
|
|
# Check if a Media representation already exists for this playlist
|
|
media_representation = Media.objects.filter(linked_playlist=instance).first()
|
|
|
|
if media_representation:
|
|
# Update existing media representation
|
|
media_representation.title = instance.title
|
|
media_representation.description = instance.description
|
|
media_representation.user = instance.user
|
|
media_representation.media_type = "playlist"
|
|
media_representation.encoding_status = "success"
|
|
media_representation.save()
|
|
else:
|
|
# Create new media representation for this playlist
|
|
Media.objects.create(
|
|
title=instance.title,
|
|
description=instance.description,
|
|
user=instance.user,
|
|
linked_playlist=instance,
|
|
media_type="playlist",
|
|
encoding_status="success",
|
|
# Inherit the same state and review status defaults
|
|
)
|
|
|
|
|
|
@receiver(pre_delete, sender=Playlist)
|
|
def delete_playlist_media(sender, instance, **kwargs):
|
|
"""
|
|
Delete the associated Media representation when a Playlist is deleted.
|
|
"""
|
|
from .media import Media
|
|
|
|
# Delete any Media objects that represent this playlist
|
|
Media.objects.filter(linked_playlist=instance).delete()
|