0

Positionモデルに関連する ForeignKey フィールドを持つモデルがありますEmplacement。の管理者でEmplacement、のインライン管理者を設定しましたPosition

私が達成したいのは、num_positionフィールドがインスタンスで変更されるEmplacementと、このインスタンスに関連する位置が削除され、新しい位置が作成されることです。

これを実現するために、Emplacementモデルのsave()メソッドをオーバーライドし、num_position変更されたかどうかを確認します。

Emplacement問題は、フィールドの変更後にモデルを保存するnum_positionと、次のエラーが発生することです。

ValidationError: Select a valid choice. That choice is not one of the available choices.

位置の削除/再作成に関係していると思いますが、理由がわかりません。

どうしたの?

models.py

class Emplacement(models.Model): 

    num_position = models.IntegerField()
    # more fields

    __original_num_position = None

    def __init__(self, *args, **kwargs):
        super(Emplacement, self).__init__(*args, **kwargs)
        self.__original_num_position = self.num_position

    def save(self, *args, **kwargs):

        if self.num_position != self._Emplacement__original_num_position:
            # if num_position has changed, delete existing positions and recreate <num_position> positions
            Position.objects.filter(emplacement=self).delete()            
            for i in range(self.num_position):
                position = Position()
                position.emplacement = self
                position.number = i+1
                position.save()

        super(Emplacement, self).save(*args, **kwargs)

class Position(models.Model):
    emplacement = models.ForeignKey(Emplacement)
    number = models.IntegerField()

admin.py

class PositionInline(admin.TabularInline):
    model = Position

class EmplacementAdmin(admin.ModelAdmin):
    inlines = [PositionInline]

my_site.register(Emplacement, EmplacementAdmin)

編集

シグナルでも試しましたが、同じエラーが発生しました:

@receiver(post_save, sender=Emplacement)
def create_positions(sender, instance, created, **kwargs):
    """Create positions when num_position has changed."""
    if instance.num_position != instance.old_num_position:
        Position.objects.filter(emplacement=instance).delete()
        for i in range(instance.num_position):
            position = Position()
            position.emplacement = instance
            position.numero = i+1            
            position.save()
4

1 に答える 1

1

まったく同じ問題が発生しました。最後に、モデル側ではなくフォーム/フォームセットで作成/削除ロジックを実装することで解決しました。

したがって、「メイン」管理クラスでは、あなたの場合は「EmplacementAdmin」で、save_formset メソッドを拡張しました。

def save_formset(self, request, form, formset, change):
    instance = form.instance
    deleted = []
    if instance and instance.__original_num_position != instance.num_position:
        ''' create new Positions if needed '''

        formset.delete_positions = None # pass on a list, QuerySet, whatever

    super(EmplacementAdmin, self).save_formset(request, form, formset, change)

そして、次のように PositionFormSet を作成します。

class PositionFormSet(BaseInlineFormSet):
    delete_positions = []

    @property
    def deleted_forms(self):
        deleted_forms = []
        try:
            deleted_forms = super(PositionFormSet, self).deleted_forms
        except AttributeError:
            pass
        for form in self.forms:
            if form.instance in self.delete_positions:
                 deleted_forms.append(form)
        self.delete_positions = []
        return deleted_forms

inlineadmin でこのフォームセットを設定します。

class PositionInline(admin.TabularInline):
    model = Position
    formset = PositionFormSet

私が考えていた方法とはまったく異なりますが、トリックはうまくいきました。もっとエレガントな解決策があるかどうか知りたいです:-)

于 2013-10-24T12:10:02.603 に答える