Django在内联表单管理中获取实例

时间:2014-03-17 22:14:20

标签: django python-2.7 django-forms django-admin inline

有一个内联表单类:

class ItemColorSelectForm(forms.ModelForm):
    def __init__(self, *args, **kwargs):
        super(ItemColorSelectForm, self).__init__(*args, **kwargs)
        #here i need current object

内联课程:

class ItemColorSelectInline(generic.GenericTabularInline):
    model = ColorSelect
    extra = 1
    form = ItemColorSelectForm

管理员课程

class ItemAdmin(admin.ModelAdmin):
    inlines = [ItemColorInline,]

问题:如何在ItemColorSelectForm中获取当前对象。

print kwargs返回:

{'auto_id': u'id_%s', 'prefix': u'catalog-colorselect-content_type-object_id-__prefix__', 'empty_permitted': True}

3 个答案:

答案 0 :(得分:6)

当前接受的解决方案不是线程安全的。如果您关心线程安全,请永远不要将实例分配给静态类属性。

线程安全解决方案是:

对于 Django 1.7< 1.9 (可能是早期版本,不清楚):

from django.utils.functional import cached_property

def get_formset(self, *args, **kwargs):
    FormSet = super(InlineAdmin, self).get_formset(*args, **kwargs)

    class ProxyFormSet(FormSet):
        def __init__(self, *args, **kwargs):
            self.instance = kwargs['instance']
            super(ProxyFormSet, self).__init__(*args, **kwargs)

        @cached_property
        def forms(self):
            kwargs = {'instance': self.instance}
            forms = [self._construct_form(i, **kwargs) 
                    for i in xrange(self.total_form_count())]
            return forms
    return ProxyFormSet

Django> = 1.9 开始,也可以传递form_kwargs:

def get_formset(self, *args, **kwargs):
    FormSet = super(InlineAdmin, self).get_formset(*args, **kwargs)

    class ProxyFormSet(FormSet):
        def __init__(self, *args, **kwargs):
            form_kwargs = kwargs.pop('form_kwargs', {})
            form_kwargs['instance'] = kwargs['instance']
            super(ProxyFormSet, self).__init__(
                *args, form_kwargs=form_kwargs, **kwargs)
    return ProxyFormSet

以上解决方案将以模型形式提供实例kwarg:

class InlineForm(forms.ModelForm):
    def __init__(self, *args, **kwargs):
        super(InlineForm, self).__init__(*args, **kwargs)
        print('instance', kwargs['instance'])

答案 1 :(得分:3)

<强>解决方案: 覆盖Inline类

中的formset方法
def get_formset(self, request, obj=None, **kwargs):
        InlineForm.obj = obj
        return super(InlineAdmin, self).get_formset(request, obj, **kwargs)

答案 2 :(得分:1)

要解决:当前接受的解决方案在多线程模式下不安全

Arti的解决方案有效,另一个更好的选择可能是:

  

与其将当前的对象ID传递给内联形式,
  使用对象ID在get_formset()中创建一个内联表单字段。

# admin.py
class TransactionInline(admin.TabularInline):
    model = Transaction
    form = TransactionInlineForm

    def get_formset(self, request, obj=None, **kwargs):
        # comment Arti's solution
        # TransactionInlineForm.project_id = obj.id

        formset = super().get_formset(request, obj, **kwargs)
        field = formset.form.declared_fields['purchase']
        field.queryset = get_object_or_404(Project, pk=obj.id).products.all()
        return formset
# forms.py
class TransactionInlineForm(ModelForm):
    purchase = ModelChoiceField(queryset=None, label='Purchase', required=False)

因此,不再需要覆盖__init__()格式,当前对象也不需要。

在Django 2.1.7中工作