【问题标题】:How to pass variable to Django Paginator()'s per_page parameter so all album tracks can be displayed on one page?如何将变量传递给 Django Paginator() 的 per_page 参数,以便所有专辑曲目都可以显示在一页上?
【发布时间】:2021-11-29 17:44:56
【问题描述】:

我正在使用 Django 和 Postgres 作为数据库构建一个网络应用音乐播放器。

我现在的设置会显示每张专辑的第一首歌曲。单击播放按钮时,视图 更改为每张专辑的第一首曲目。

我想在一个页面上显示任何给定专辑的所有曲目。

单个页面会显示这些对象:

来自专辑:album_title、artist、artifact_file 和来自曲目:专辑中每个曲目的 track_title 和 audio_file

为此,我需要为 Django 的 Paginator 类中的 self.per_page 参数提供一个整数。目前它设置为 1。

曲目的数量会因专辑而异,因此我想将其作为整数作为变量 (Album.number_tracks) 传递。 问题是我无法创建一个查询集来迭代并将每次迭代传递给 self.per_page 因为 Paginator 函数一次获取所有对象,而不是逐个对象。所以我不能使用任何条件循环 喜欢:

  queryset = Album.objects.all() 
  number_of_tracks = [a.number_tracks for a in queryset]
  for num_track in number_of_tracks:
      # pass album's number of tracks to Paginator(per_page=num_track)

如何在一个页面上显示任何给定专辑中的所有曲目,无论歌曲数量如何?

以下是 models.py 中的相关字段:

class Album(models.Model):
    title = models.TextField(blank=False, null=False)
    artist = models.ForeignKey(Artist, on_delete=models.CASCADE)
    number_tracks = models.TextField(blank=True, null=True)
    track_list = models.TextField(blank=True, null=True)
    artwork_file = models.ImageField(blank=True, null=True, max_length=500)

    def __str__(self):
        return self.title

class Track(models.Model):
    title = models.TextField(blank=False, null=False)
    album = models.ForeignKey(Album, on_delete=models.CASCADE)
    artist = models.TextField(blank=True, null=True)
    track_number = models.TextField(blank=True, null=True)
    audio_file = models.FileField(blank=True, null=True, max_length=500)

    def __str__(self):
        return self.title

views.py

def index(request):
    paginator_track = Paginator(Track.objects.order_by('album', 'track_number').all(), 1)
    page_number = request.GET.get('page')
    page_obj_track = paginator_track.get_page(page_number)

    paginator_album = Paginator(Album.objects.order_by('artist', 'title').all(), 1)
    page_number = request.GET.get('page')
    page_obj_album = paginator_album.get_page(page_number)

    context = {'page_obj_album': page_obj_album, 'page_obj_track': page_obj_track}

    return render(request, 'index.html', context)

index.html 中,我像这样显示对象:

{% for album_obj in page_obj_album %}
    <img src='{{ album_obj.artwork_file.url }}' alt='{{ album_obj.artwork_link }}' />
    <h3> {{ album_obj.artist }} </h3>
    <h3> {{ album_obj.title }} </h3>
    {% endfor %}

    {% for track_obj in page_obj_track %}
    <h1> {{ track_obj.title }} </h1>
    <a href='{% if page_obj_track.has_previous %}?page={{ 
     page_obj_track.previous_page_number }} {% endif %}'>
    <i class='fa fa-step-backward fa-2x'></i></a>

    <a href='{% if page_obj_track.has_next %}?page={{page_obj_track.next_page_number}}  {% endif %}'><i class='fa fa-step-forward fa-2x'></i></a>

    <audio class='fc-media'>
    <source src='{% if track_obj.audio_file %} {{ track_obj.audio_file.url }} 
     {% else %} {{ track_obj.audio_link }} {% endif %}'
             type='audio/mp3'/></audio>
   {% endfor %}

【问题讨论】:

  • 我认为您不需要分页,或者我可能遗漏了一些东西。如果您想在单个页面上显示与专辑相关的所有曲目,album.tracks.alll() 应该为您提供专辑的所有曲目您只需要一个通过其 ID 获取专辑并返回其所有曲目的视图(即 /albums/&lt;id&gt; )
  • album.tracks.all() 如果我只有一个 Albums 模型就可以了。但是数据库的设置方式,我有一个专辑模型,其中包含艺术家、专辑、曲目数量、专辑封面。 My Tracks 模型包含实际的音频文件。因此,我试图找到一种方法,在这两个模型之间的 views.py 中更流畅地交互,同时保持专辑和属于它的曲目同步(假设每个模型都是使用单独的 Paginator 实例收集的)。有分页是因为对于每个相册,用户点击另一个页面,并且 url 相应地发生变化。

标签: python html django audio-player webapp2


【解决方案1】:

以防万一这对将来的任何人有所帮助——我最终重构了我的代码以使其正常工作。我没有使用内置的 Paginator 函数,而是使用了 DetailView。以下是代码的相关部分:

我在 models.py 中添加了一个 slug 字段来创建 URL: slug = AutoSlugField(null=True, default=None, unique=True, max_length=500, populate_from='title')

views.py:

from django.views.generic.detail import DetailView
from datasource.models import Album, Track
class AlbumDetailView(DetailView):

model = Album
template_name = 'album_view.html'

def get_context_data(self, **kwargs):

    context = super(AlbumDetailView, self).get_context_data(**kwargs)
    context['next_album'] = Album.objects.filter(title__gt=self.object.title).order_by('title').first()
    context['prev_album'] = Album.objects.filter(title__lt=self.object.title).order_by('-title').first()
    context['track'] = Track.objects.filter(album=self.object).order_by('track_number')

    return context

在 HTML 模板中,视图呈现如下:

{% if prev_album %}
<a href='{% url "album_detail" slug=prev_album.slug %}'><i class='fa fa-step-backward fa-2x'></i></a>
{% endif %}
{% if next_album %}
<a href='{% url "album_detail" slug=next_album.slug %}'><i class='fa fa-step-forward fa-2x'></i></a>
{% endif %}

{% for t in track %}
<h1> {{ t.title }} </h1>
<audio class='fc-media' style='width: 100%;'>
<source src='{% if t.audio_file %} {{ t.audio_file.url }} {% else %} {{ t.audio_link }} {% endif %}'type='audio/mp3'/></audio>
{% endfor %}

【讨论】:

    猜你喜欢
    • 2013-05-03
    • 2013-11-14
    • 2022-06-17
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2018-01-18
    • 2020-01-02
    相关资源
    最近更新 更多