`
fdayok
  • 浏览: 28670 次
  • 性别: Icon_minigender_1
  • 来自: 深圳
社区版块
存档分类
最新评论

django 1.2 change_form 中编辑修改ForeignKey的值

阅读更多
Model如下
class People(models.Model):
    name = models.CharField(max_length = 200)
    def __unicode__(self):
        return force_unicode(self.name)

class Buy(models.Model):
    from_people = models.ForeignKey(People, related_name = 'from_people')#related_name must be 'from_people'
    to_people = models.ForeignKey(People, related_name = 'to_people')
    price = models.IntegerField()
    def __unicode__(self):
        return force_unicode(u'%s-%s-%s' % (self.from_people, self.to_people, self.price))


要实现的效果:在Buy的修改页面可以同时编辑from_people和to_people两个外键的值.
django admin页面的默认效果
需要的实现

django的帮忙文档中有通过TabularInline实现在修改People时同时修改Buy的方法,代码为
#admin.py
class BuyModelInline(djopt.TabularInline):
    model = mym.Buy
    fk_name = 'from_people'

class PeopleModelAdmin(djopt.ModelAdmin):
    list_display = ['name']
    inlines = [BuyModelInline]

效果:

查看了TabularInline的代码,发现可以实现一个从Buy到People反向外键内联类来达到目的.代码
#admin.py
from django.contrib.admin.util import flatten_fieldsets
from django.utils.functional import curry
import django.forms.models as djfmmdl
class ReverseFKTabularInline(djopt.TabularInline):
    reverse_fk_name = None
    def get_formset(self, request, obj = None, **kwargs):
        """Returns a BaseInlineFormSet class for use in admin add/change views."""
        if self.declared_fieldsets:
            fields = flatten_fieldsets(self.declared_fieldsets)
        else:
            fields = None
        if self.exclude is None:
            exclude = []
        else:
            exclude = list(self.exclude)
        exclude.extend(kwargs.get("exclude", []))
        exclude.extend(self.get_readonly_fields(request, obj))
        # if exclude is an empty list we use None, since that's the actual
        # default
        exclude = exclude or None
        defaults = {
            "form": self.form,
            "formset": self.formset,
            "fk_name": self.reverse_fk_name,
            "fields": fields,
            "exclude": exclude,
            "formfield_callback": curry(self.formfield_for_dbfield, request = request),
            "extra": self.extra,
            "max_num": self.max_num,
            "can_delete": self.can_delete,
        }
        defaults.update(kwargs)
        return self.inlineformset_factory(self.model, self.parent_model, **defaults)

    def inlineformset_factory(self, model, model_has_fk, form = djfmmdl.ModelForm,
                              formset = djfmmdl.BaseInlineFormSet, fk_name = None,
                              fields = None, exclude = None,
                              extra = 3, can_order = False, can_delete = True, max_num = None,
                              formfield_callback = lambda f: f.formfield()):
        """
        Returns an ``InlineFormSet`` for the given kwargs.
    
        You must provide ``fk_name`` if ``model`` has more than one ``ForeignKey``
        to ``model_has_fk``.
        """
        fk = djfmmdl._get_foreign_key(model, model_has_fk, fk_name = fk_name)
        # enforce a max_num=1 when the foreign key to the parent model is unique.
        if fk.unique:
            max_num = 1
        kwargs = {
            'form': form,
            'formfield_callback': formfield_callback,
            'formset': formset,
            'extra': extra,
            'can_delete': can_delete,
            'can_order': can_order,
            'fields': fields,
            'exclude': exclude,
            'max_num': max_num,
        }
        FormSet = djfmmdl.modelformset_factory(model, **kwargs)
        FormSet.fk = fk
        return FormSet


上面是基类,下面实现People内联类
#admin.py
class PeopleTabularInline(ReverseFKTabularInline):
    model = mym.People
    extra = 0
class SalerInline(PeopleTabularInline):
    reverse_fk_name = 'from_people'
    verbose_name_plural = 'Saler'
class BuyerInline(PeopleTabularInline):
    reverse_fk_name = 'to_people'
    verbose_name_plural = 'Buyer'


内联类的使用方法
#admin.py
class BuyModelAdmin(djopt.ModelAdmin):
    saler_name = make_func(mym.Buy, 'from_people__name')
    buyer_name = make_func(mym.Buy, 'to_people__name')
    list_display = [saler_name, buyer_name, 'price']
    readonly_fields = [saler_name]
    inlines = [SalerInline, BuyerInline]


完成后的效果
图中红框里的显示是readonly_fields = [saler_name]这行代码的结果
分享到:
评论

相关推荐

    PyPI 官网下载 | django_file_form-3.1.2-py3-none-any.whl

    标题中的"PyPI 官网下载 | django_file_form-3.1.2-py3-none-any.whl"表明这是一个从Python Package Index(PyPI)官方源下载的软件包,名为`django_file_form`,版本为3.1.2,适用于Python 3环境,且不分架构(any...

    Python库 | django_file_form-0.4.1-py3-none-any.whl

    在Django项目中使用`django_file_form`,首先需要在`settings.py`的`INSTALLED_APPS`列表中添加`django_file_form`,然后在模型中定义文件字段,在视图中处理文件上传,最后在模板中渲染表单。通过这种方式,开发者...

    django1.2文档

    文档中包含的 `django1.2.chm` 和 `djangobook2ch.chm` 文件分别提供了 Django 1.2 的中文版和英文版官方文档,对于学习和查阅 Django 1.2 的相关知识非常有帮助。CHM 文件是一种帮助文件格式,包含了索引、搜索和...

    Python库 | django_form_error_reporting-0.9-py2.py3-none-any.whl

    综上所述,`django_form_error_reporting`是一个针对Django框架的实用工具,旨在简化和增强Django应用中的表单错误处理,对于需要构建高质量、用户友好的Web应用的开发者来说,是一个有价值的资源。

    django admin 自定义替换change页面模板的方法

    举个简单的例子:(此仅限于修改change_form页面) 原来的时候,change_form_list是包含这些按钮的: 因为此页面继承了 {% extends 'admin/change_form.html' %} 所以,可以将admin/change_form.html 拷贝到指定...

    Python库 | django_celery_results-2.2.0.tar.gz

    这个库的版本为2.2.0,主要功能是为Celery任务提供持久化的结果存储,使得在Django项目中使用Celery时,可以方便地管理和检索异步任务的结果。 首先,我们来理解一下Django。Django是一款用Python编写的高级Web框架...

    PyPI 官网下载 | django_rename_app-0.1.1.tar.gz

    当开发者需要修改应用的名称时,`django_rename_app` 提供了一个方便的工具,帮助开发者自动更新与旧应用名称相关的所有引用,避免手动搜索和替换可能遗漏的地方。 `django_rename_app-0.1.1.tar.gz` 是这个库的...

    Python库 | django_mongoengine_filter-0.3.1-py2.py3-none-any.whl

    **Python库 django_mongoengine_filter-0.3.1-py2.py3-none-any.whl 深度解析** 在Python的世界中,开发者们经常利用各种库来简化工作,提高效率。`django_mongoengine_filter`是这样一个库,它专为Django框架与...

    Python库 | django_rest_form_fields-1.2.5-py2.py3-none-any.whl

    资源分类:Python库 所属语言:Python 资源全名:django_rest_form_fields-1.2.5-py2.py3-none-any.whl 资源来源:官方 安装方法:https://lanzao.blog.csdn.net/article/details/101784059

    PyPI 官网下载 | django_dnoticias_auth-1.2.7.1.tar.gz

    Django_dnoticias_auth则是Django生态中的一个授权(Authentication)模块,版本号1.2.7.1,意味着它是经过多次迭代和优化的成熟组件。在Python库中,这样的模块通常提供额外的功能或定制化服务,以增强Django的默认...

    Django python ERP_softwarecost_python项目_pythondjango_pythonERP_s

    ERP管理软件

    Python库 | django_counter_field_py3-0.0.1.2.tar.gz

    标题中的"Python库 | django_counter_field_py3-0.0.1.2.tar.gz"指出这是一个基于Python的软件包,具体来说是针对Django框架的一个扩展模块,名为`django_counter_field`. 这个版本号0.0.1.2表明它是该库的早期版本...

    PyPI 官网下载 | django_trix_fork-0.3.1.2-py3-none-any.whl

    `django_trix_fork`这个包可能是对原版Django Trix的修改或增强版,Trix通常指的是一个富文本编辑器,用于提高网站内容的编辑体验。 **Python 库和开发语言** Python库是Python编程的重要组成部分,它们提供了预定...

    django_contact_form

    1. **模型(Models)**:虽然`django_contact_form`可能没有自定义数据库模型,但在其他更复杂的Django应用中,我们可能会创建一个模型来存储表单数据。模型是数据库中的表的Python表示,可以定义字段类型、验证规则...

    PyPI 官网下载 | django_form_error_reporting-0.9-py2.py3-none-any.whl

    资源来自pypi官网。 资源全名:django_form_error_reporting-0.9-py2.py3-none-any.whl

    PyPI 官网下载 | django_comments_xtd-1.2-py3.3.egg

    标题"PyPI 官网下载 | django_comments_xtd-1.2-py3.3.egg" 提供了关键信息,表明这是一个从Python Package Index(PyPI)官方源下载的软件包,名为`django_comments_xtd`,版本为1.2,适用于Python 3.3环境。...

Global site tag (gtag.js) - Google Analytics