ИМ
Игорь Максимов22 декабря 2018 г. 6:44

Подскажите по фильтрации объектов в админ-панели

django

Доброго времени суток. У меня ситуация следующая, имею 3 модели которые принадлежат один ко многим:

class Serial(models.Model):
    class Meta:
        db_table = 'serial'
        verbose_name = 'Сериал'
        verbose_name_plural = 'Сериалы'

    name = models.CharField(verbose_name='Название', max_length=200)
    orig_name = models.CharField(verbose_name='Оригинальное название', max_length=200, blank=True)
    poster = models.ImageField(
        # upload_to=curry(upload_to_media, prefix='posters'),
        upload_to=upload_to_media,
        blank=True,
        verbose_name='Постер'
    )
    category = models.ForeignKey(CategorySerial, on_delete=models.CASCADE, verbose_name='Категория', null=True, blank=True)
    year = models.DateField(verbose_name='Дата выхода', default='2000-01-01')
    country = models.CharField(verbose_name='Страна', max_length=300, default='Неизвестно')
    translate = models.CharField(verbose_name='Перевод', max_length=300, default='Русский')
    create = models.DateTimeField(verbose_name='Дата публикации', default=timezone.now)
    update = models.DateTimeField(verbose_name='Дата обновления', default=timezone.now)
    description = models.TextField(verbose_name='Описание')
    moder = models.BooleanField(verbose_name='Модерация', default=False)
    votes = GenericRelation(LikeDislike, related_query_name='serials')

    def public(self):
        self.date = timezone.now()
        self.save()

    def __str__(self):
        return self.name

    def get_absolute_url(self):
        return reverse('serial:serial_detail', args=[self.id])

    def get_bookmark_count(self):
        return self.bookmarkserial_set.all().count()


class Playlist(models.Model):
    class Meta:
        db_table = 'playlist'
        verbose_name = 'Плейлист'
        verbose_name_plural = 'Плейлисты'
    name = models.CharField(max_length=250, verbose_name='Имя плейлиста')
    serial = models.ForeignKey(Serial, on_delete=models.CASCADE)

    def __str__(self):
        return self.name



class Seria(models.Model):
    class Meta:
        db_table = 'seria'
        verbose_name = 'Серия'
        verbose_name_plural = 'Серии'

    playlist = models.ForeignKey(Playlist, on_delete=models.CASCADE)
    serial = models.ForeignKey(Serial, on_delete=models.CASCADE)
    video = models.FileField(
        # upload_to=curry(upload_to_media, prefix='movies'),
        upload_to=upload_to_media,
        verbose_name='Видео'
    )

В админ панели ситуация следующая:
Создал я плейлист в одной сериале, и добавляя видео в другом сериале у меня плейлисты с предыдущего

Вопрос заключается в следующем: как отображать плейлисты только принадлежащие к определенному сериалу а не весь список плейлистов.

Рекомендуем хостинг TIMEWEB
Рекомендуем хостинг TIMEWEB
Стабильный хостинг, на котором располагается социальная сеть EVILEG. Для проектов на Django рекомендуем VDS хостинг.

Вам это нравится? Поделитесь в социальных сетях!

12
Evgenii Legotckoi
  • 22 декабря 2018 г. 17:06

Добрый день.

Это делается через установку кастомной формы для model.Admin

Смысл в том, чтобы в соответствующем поле отфильтровать только те серии, или плейлисты, которые отностятся к данном сериалу

class SeriaAdminForm(forms.ModelForm):

    def __init__(self, *args, **kwargs):
        super().__init__(*args, **kwargs)

        try:
            self.fields['playlist'].queryset = self.instance.serial.playlist_set.all()
        except Playlist.DoesNotExist:
            pass


class SeriaAdmin(PostAdmin):
    form = SeriaAdminForm


admin.site.register(Seria, SeriaAdmin)

По идее так должно работать правильно.

    ИМ
    • 22 декабря 2018 г. 19:22

    Так то оно работает, но я использую Inline. В inline не хочет, пишет:

    serial.models.Seria.serial.RelatedObjectDoesNotExist: Seria has no serial.
    
      Evgenii Legotckoi
      • 22 декабря 2018 г. 19:31
      • (ред.)

      Ну, полагаю, что весь прикол в том, что здесь просто не обработано это исключение. Поскольку при создании нового объекта там никакой объект не создан.

      А если так? Нужно тогда обработать это исключение и всё

      class SeriaAdminForm(forms.ModelForm):
      
          def __init__(self, *args, **kwargs):
              super().__init__(*args, **kwargs)
      
              try:
                  self.fields['playlist'].queryset = self.instance.serial.playlist_set.all()
              except Playlist.DoesNotExist:
                  pass
              except Seria.serial.RelatedObjectDoesNotExist:
                  pass
      

      Там скорее всего понадобится сначала установить сериал, сохранить все объекты, а потом уже появится возможность выбора плейлистов.

      Дальше я бы покопал в сторону обновления queryset плейлистов в случае изменения внешнего ключа на сериал.

        ИМ
        • 22 декабря 2018 г. 20:24

        Я ошибся. Вышеуказанные формы не работают.
        А с добавлением except Seria.serial.RelatedObjectDoesNotExist: пропускает ошибку, и в выпадающем списке все плейлисты.

          Evgenii Legotckoi
          • 22 декабря 2018 г. 20:32

          Возможно, тогда стоит установить пустой queryset в том случае если нет внешнего ключа на сериал.

          class SeriaAdminForm(forms.ModelForm):
          
              def __init__(self, *args, **kwargs):
                  super().__init__(*args, **kwargs)
          
                  try:
                      self.fields['playlist'].queryset = self.instance.serial.playlist_set.all()
                  except Playlist.DoesNotExist:
                      pass
                  except Seria.serial.RelatedObjectDoesNotExist:
                      self.fields['playlist'].queryset = Playlist.objects.none()
          

          В любом случае, пока не установлен ключ на сериал, то по логике вы не можете установить какой-либо плейлист туда

            ИМ
            • 22 декабря 2018 г. 20:39

            А так в выпадающем списке ничего нет(

              Evgenii Legotckoi
              • 22 декабря 2018 г. 20:54
              • Ответ был помечен как решение.

              Нужно тогда усложнять код, поскольку серия ещё не создана, то нужно сначала получить внешний ключ на парента

              class MyFormSet(BaseInlineFormSet):
                  def get_form_kwargs(self, index):
                      kwargs = super().get_form_kwargs(index)
                      kwargs['parent_object'] = self.instance
                      return kwargs
              
              
              class MyForm(forms.ModelForm):
                  def __init__(self, *args, parent_object, **kwargs):
                      self.parent_object = parent_object
                      super(MyForm, self).__init__(*args, **kwargs)
              
                     try:
                          self.fields['playlist'].queryset = parent_object.playlist_set.all()
                      except Playlist.DoesNotExist:
                          pass
              
              
              class MyChildInline(admin.TabularInline):
                  formset = MyFormSet
                  form = MyForm
              

              что-то вроде такого должно получиться.

                ИМ
                • 22 декабря 2018 г. 21:49
                • (ред.)

                Вместо parent_object подставлять serial?

                UPD: У меня отступы сдвинулись.

                  ИМ
                  • 22 декабря 2018 г. 21:59

                  Так все работает. Спасибо вам огромное. Учусь с вами потихоньку.

                    Evgenii Legotckoi
                    • 23 декабря 2018 г. 1:01

                    Отступы? какие отступы? в админке? это уже детали, которые просто нужно будет дооправить в дальнейшем.

                      ИМ
                      • 23 декабря 2018 г. 1:05

                      У меня try сьехал влево, сначала не за метил и получал ошибки. Почудилось мне что нужно свои переменные подставлять))

                        Evgenii Legotckoi
                        • 23 декабря 2018 г. 1:10

                        А копировали просто с моего комментария. Там действительно съехал на один пробел. Ну я редактировал сразу в форме комментария текст.

                          Комментарии

                          Только авторизованные пользователи могут публиковать комментарии.
                          Пожалуйста, авторизуйтесь или зарегистрируйтесь
                          Ua

                          Qt - Тест 001. Сигналы и слоты

                          • Результат:84баллов,
                          • Очки рейтинга4
                          Ua

                          Qt - Тест 001. Сигналы и слоты

                          • Результат:42баллов,
                          • Очки рейтинга-8
                          ОК

                          Qt - Тест 001. Сигналы и слоты

                          • Результат:47баллов,
                          • Очки рейтинга-6
                          Последние комментарии
                          ИМ
                          Игорь Максимов22 ноября 2024 г. 21:51
                          Django - Урок 017. Кастомизированная страница авторизации на Django Добрый вечер Евгений! Я сделал себе авторизацию аналогичную вашей, все работает, кроме возврата к предидущей странице. Редеректит всегда на главную, хотя в логах сервера вижу запросы на правильн…
                          Evgenii Legotckoi
                          Evgenii Legotckoi31 октября 2024 г. 23:37
                          Django - Урок 064. Как написать расширение для Python Markdown Добрый день. Да, можно. Либо через такие же плагины, либо с постобработкой через python библиотеку Beautiful Soup
                          A
                          ALO1ZE19 октября 2024 г. 17:19
                          Читалка fb3-файлов на Qt Creator Подскажите как это запустить? Я не шарю в программировании и кодинге. Скачал и установаил Qt, но куча ошибок выдается и не запустить. А очень надо fb3 переконвертировать в html
                          ИМ
                          Игорь Максимов5 октября 2024 г. 16:51
                          Django - Урок 064. Как написать расширение для Python Markdown Приветствую Евгений! У меня вопрос. Можно ли вставлять свои классы в разметку редактора markdown? Допустим имея стандартную разметку: <ul> <li></li> <li></l…
                          d
                          dblas55 июля 2024 г. 20:02
                          QML - Урок 016. База данных SQLite и работа с ней в QML Qt Здравствуйте, возникает такая проблема (я новичок): ApplicationWindow неизвестный элемент. (М300) для TextField и Button аналогично. Могу предположить, что из-за более новой верси…
                          Сейчас обсуждают на форуме
                          f
                          firstlunoxod15 февраля 2025 г. 13:46
                          Рисование на QGraphicsScene при зажатой кнопке мыши Подскажите, пожалуйста! Как данный класс можно дополнить, чтобы созданные объекты можно было перемещать мышкой по сцене?
                          Дмитрий
                          Дмитрий3 февраля 2025 г. 16:24
                          Создание deb-пакета. Как создать ярлык на рабочем столе после установки собственного deb-пакета? Всем привет. Сделал свой deb-пакет с программой. Всё устанавливается и работает. Ставлю по пути /usr/bin/my_application. Как для пользователя при установке пакета сразу создать ярлык на раб…
                          NW
                          Nayo Wai30 января 2025 г. 19:22
                          не запускается компьютер!!! Не запускается компьютер (точнее работает блок , но сам монитор вообще жесть)В общем я ничего с интернета не скачивала в последнее время. На компе никаких левых пр…
                          n
                          nkly3 января 2025 г. 12:52
                          Нужно запретить перемещение только некоторых итемов, остальные перемещать можно. Вопрос решен. Узнать QModelIndex элемента на который мы перетаскиваем другой элемент, можно с помощью функции indexAt(event->position().toPoint()) представления QTreeViev вызываемой в переопр…
                          M
                          Marsel17 августа 2023 г. 0:26
                          OAuth2.0 через VK, получение email Спасибо большое за помощь и простите за то что отнял время своей невнимательностью.

                          Следите за нами в социальных сетях