ИМ
Игорь МаксимовЖел. 21, 2018, 7: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, 6: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, 8:22 Т.Ж.

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

    serial.models.Seria.serial.RelatedObjectDoesNotExist: Seria has no serial.
    
      Evgenii Legotckoi
      • Жел. 22, 2018, 8: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, 9:24 Т.Ж.

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

          Evgenii Legotckoi
          • Жел. 22, 2018, 9: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, 9:39 Т.Ж.

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

              Evgenii Legotckoi
              • Жел. 22, 2018, 9: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, 10:49 Т.Ж.
                • (өңделген)

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

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

                  ИМ
                  • Жел. 22, 2018, 10:59 Т.Ж.

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

                    Evgenii Legotckoi
                    • Жел. 22, 2018, 2:01 Т.Қ.

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

                      ИМ
                      • Жел. 22, 2018, 2:05 Т.Қ.

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

                        Evgenii Legotckoi
                        • Жел. 22, 2018, 2:10 Т.Қ.

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

                          Пікірлер

                          Тек рұқсаты бар пайдаланушылар ғана пікір қалдыра алады.
                          Кіріңіз немесе Тіркеліңіз
                          Г

                          C++ - Тест 001. Первая программа и типы данных

                          • Нәтиже:66ұпай,
                          • Бағалау ұпайлары-1
                          t

                          C++ - Тест 001. Первая программа и типы данных

                          • Нәтиже:33ұпай,
                          • Бағалау ұпайлары-10
                          t

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

                          • Нәтиже:52ұпай,
                          • Бағалау ұпайлары-4
                          Соңғы пікірлер
                          G
                          GoattRockҚыр. 3, 2024, 1:50 Т.Қ.
                          Linux жүйесінде файлдарды қалай көшіруге болады Задумывались когда-нибудь о том, как мы привыкли доверять свои вещи службам грузоперевозок? Сейчас такие услуги стали неотъемлемой частью нашей жизни, особенно когда речь идет о переездах между …
                          d
                          dblas5Шілде 5, 2024, 11:02 Т.Ж.
                          QML - Сабақ 016. SQLite деректер қоры және онымен QML Qt-та жұмыс істеу Здравствуйте, возникает такая проблема (я новичок): ApplicationWindow неизвестный элемент. (М300) для TextField и Button аналогично. Могу предположить, что из-за более новой верси…
                          k
                          kmssrАқп. 8, 2024, 6:43 Т.Қ.
                          Qt Linux - Сабақ 001. Linux астында Autorun Qt қолданбасы как сделать автозапуск для флэтпака, который не даёт создавать файлы в ~/.config - вот это вопрос ))
                          АК
                          Анатолий КононенкоАқп. 5, 2024, 1:50 Т.Ж.
                          Qt WinAPI - Сабақ 007. Qt ішінде ICMP Ping арқылы жұмыс істеу Без строки #include <QRegularExpressionValidator> в заголовочном файле не работает валидатор.
                          Енді форумда талқылаңыз
                          Evgenii Legotckoi
                          Evgenii LegotckoiМаусым 24, 2024, 3:11 Т.Қ.
                          добавить qlineseries в функции Я тут. Работы оень много. Отправил его в бан.
                          F
                          FynjyШілде 22, 2024, 4:15 Т.Ж.
                          при создании qml проекта Kits есть но недоступны для выбора Поставил Qt Creator 11.0.2. Qt 6.4.3 При создании проекта Qml не могу выбрать Kits, они все недоступны, хотя настроены и при создании обычного Qt Widget приложения их можно выбрать. В чем может …
                          BlinCT
                          BlinCTМаусым 25, 2024, 1 Т.Ж.
                          Нарисовать кривую в qml Всем привет. Имеется Лист листов с тосками, точки получаны интерполяцией Лагранжа. Вопрос, как этими точками нарисовать кривую? ChartView отпадает сразу, в qt6.7 появился новый элемент…
                          BlinCT
                          BlinCTМамыр 5, 2024, 5:46 Т.Ж.
                          Написать свой GraphsView Всем привет. В Qt есть давольно старый обьект дял работы с графиками ChartsView и есть в 6.7 новый но очень сырой и со слабым функционалом GraphsView. По этой причине я хочу написать х…
                          Evgenii Legotckoi
                          Evgenii LegotckoiМамыр 2, 2024, 2:07 Т.Қ.
                          Мобильное приложение на C++Qt и бэкенд к нему на Django Rest Framework Добрый день. По моему мнению - да, но то, что будет касаться вызовов к функционалу Андроида, может создать огромные трудности.

                          Бізді әлеуметтік желілерде бақылаңыз