我花了相当多的时间试图提出一个可以跨网站重复使用的解决方案。 James 的帖子包含扩展 BaseInlineFormSet 的关键智慧,但战略性地调用了针对 BaseFormSet 的调用。
以下解决方案分为两部分:AdminInline 和 BaseInlineFormSet。
-
InlineAdmin 根据公开的请求对象动态生成初始值。
- 它使用 currying 通过传递给构造函数的关键字参数将初始值公开给自定义
BaseInlineFormSet。
-
BaseInlineFormSet 构造函数将初始值从关键字参数列表中弹出并正常构造。
- 最后一点是通过更改表单的最大总数并使用
BaseFormSet._construct_form 和BaseFormSet._construct_forms 方法来覆盖表单构造过程
这里有一些使用 OP 类的具体 sn-ps。我已经针对 Django 1.2.3 对此进行了测试。我强烈建议在开发时将formset 和admin 文档放在手边。
admin.py
from django.utils.functional import curry
from django.contrib import admin
from example_app.forms import *
from example_app.models import *
class AttendanceInline(admin.TabularInline):
model = Attendance
formset = AttendanceFormSet
extra = 5
def get_formset(self, request, obj=None, **kwargs):
"""
Pre-populating formset using GET params
"""
initial = []
if request.method == "GET":
#
# Populate initial based on request
#
initial.append({
'foo': 'bar',
})
formset = super(AttendanceInline, self).get_formset(request, obj, **kwargs)
formset.__init__ = curry(formset.__init__, initial=initial)
return formset
forms.py
from django.forms import formsets
from django.forms.models import BaseInlineFormSet
class BaseAttendanceFormSet(BaseInlineFormSet):
def __init__(self, *args, **kwargs):
"""
Grabs the curried initial values and stores them into a 'private'
variable. Note: the use of self.__initial is important, using
self.initial or self._initial will be erased by a parent class
"""
self.__initial = kwargs.pop('initial', [])
super(BaseAttendanceFormSet, self).__init__(*args, **kwargs)
def total_form_count(self):
return len(self.__initial) + self.extra
def _construct_forms(self):
return formsets.BaseFormSet._construct_forms(self)
def _construct_form(self, i, **kwargs):
if self.__initial:
try:
kwargs['initial'] = self.__initial[i]
except IndexError:
pass
return formsets.BaseFormSet._construct_form(self, i, **kwargs)
AttendanceFormSet = formsets.formset_factory(AttendanceForm, formset=BaseAttendanceFormSet)