Django Admin 实现外键过滤的方法

发布时间 - 2026-01-11 03:28:17    点击率:

说明和 Model

环境:

➜ python

Python 3.6.3 |Anaconda custom (x86_64)| (default, Oct 6 2017, 12:04:38)
[GCC 4.2.1 Compatible Clang 4.0.1 (tags/RELEASE_401/final)] on darwin
Type "help", "copyright", "credits" or "license" for more information.
>>> import django
>>> print(django.get_version())
2.0.1
>>>

2018年05月23日更新:

可以通过get_changeform_initial_data 函数来传递initial参数.

# admin.py
@admin.register(Score)
class ScoreConfigAdmin(FilterUserAdmin):
  # fields = ('id','name')
  form = ScoreConfigAdminForm

  def get_changeform_initial_data(self, request):
    initial = super().get_changeform_initial_data(request)
    initial.update({'uid': request.user.id})
    return initial

# forms.py
class ScoreConfigAdminForm(forms.ModelForm):
  def __init__(self, *args, **kwargs):
    super().__init__(*args, **kwargs)
    if not kwargs.get('initial'):
      return
    self.uid = kwargs.get('initial').get('uid')

  class Meta:
    model = Score
    fields = '__all__'

有一个支持多用户(使用 django admin)的 Blog,每一篇 Post 都需要记录是谁发表的并且属于那个 Blog。

user 与 Blog 的关系、 Blog 与 Post 有2种定义方式,一种是使用独立关系表,另外一种是直接在 Model 中定义中使用外键。

后面一种的 model 定义如下:

from django.contrib.auth.models import User
from django.db import models

class Blog(models.Model):
  '''
  Blog
  '''
  id = models.AutoField(unique=True, primary_key = True, verbose_name="序号")
  name = models.CharField(max_length=255, blank=True, null=True, verbose_name="名称")
  user = models.ForeignKey(User, on_delete=models.CASCADE)
  create_time = models.DateTimeField(verbose_name='添加时间', auto_now_add=True, blank=True)

  class Meta:
    verbose_name = 'Blog'
    verbose_name_plural = 'Blog管理'

  def __str__(self):
    return self.name

class Post(models.Model):
  '''
  Post 内容
  '''
  id = models.AutoField(unique=True, primary_key = True, verbose_name="序号")
  title = models.CharField(max_length=255, blank=True, null=True, verbose_name="标题")
  content = models.TextField(max_length=1024, blank=True, null=True, verbose_name="内容")
  blog = models.ForeignKey(Blog, on_delete=models.CASCADE, verbose_name="所属Blog")
  user = models.ForeignKey(User, on_delete=models.CASCADE)
  create_time = models.DateTimeField(verbose_name='添加时间', auto_now_add=True, blank=True)

  class Meta:
    verbose_name = '文章'
    verbose_name_plural = '文章管理'

  def __str__(self):
    return self.title

Admin 中实现

admin 中有2处,一处是 Blog 和 Post 列表中按 user 过滤,另外一处是新增 Post 时需要按当前 user 过滤。完整代码如下:

from django.contrib import admin
from django import forms

# Register your models here.
from django_summernote.admin import SummernoteModelAdmin
from .models import Team, Member, Activity, Score



from .models import Blog, Post
class FilterUserAdmin(admin.ModelAdmin):
  '''
  按所属用户过滤的 base, class
  '''
  def save_model(self, request, obj, form, change):
    # TODO 需要考虑不同用户对同一数据进行修改。
    obj.user = request.user
    obj.save()

  def get_queryset(self, request):
    # For Django < 1.6, override queryset instead of get_queryset
    qs = super(FilterUserAdmin, self).get_queryset(request) 
    # 不能加这个,加了这个会导致 superuser 更新普通用户的数据。
    # if request.user.is_superuser:
    #   return qs
    return qs.filter(user=request.user)

  def has_change_permission(self, request, obj=None):
    has_class_permission = super(FilterUserAdmin, self).has_change_permission(request, obj)
    if not has_class_permission:
      return False
    if obj is not None and not request.user.is_superuser and request.user.id != obj.user.id:
      return False
    return True


class BlogConfigAdmin(FilterUserAdmin):
  list_display = ('id','name', 'create_time')
  exclude = ['user']
  list_per_page = 50

admin.site.register(Blog, BlogConfigAdmin)


class PostConfigAdmin(FilterUserAdmin):
  list_display = ('id','title', 'create_time')
  exclude = ['user']
  list_per_page = 50

  def render_change_form(self, request, context, add=False, change=False, form_url='', obj=None):
    # 新增 Post 时,相关联的 Blog 需要过滤,关键就在下面这句。
    context['adminform'].form.fields['blog'].queryset = Team.objects.filter(user=request.user)
    return super(MemberConfigAdmin, self).render_change_form(request, context, add, change, form_url, obj)


admin.site.register(Post, PostConfigAdmin)

说2句

在render_change_form中下断点,直接调试下会发现更多有趣的内容。

以上这篇Django Admin 实现外键过滤的方法就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持。


# Django  # Admin  # 外键过滤  # django admin 后台实现三级联动的示例代码  # Django 中自定义 Admin 样式与功能的实现方法  # 对Django外键关系的描述  # 使用django-guardian实现django-admin的行级权限控制的方法  # Django后台admin的使用详解  # 给大家  # 添加时间  # 一处  # 就在  # 中有  # 可以通过  # 希望能  # 相关联  # 这篇  # 这句  # 多用户  # 小编  # 普通用户  # 大家多多  # 有一个  # 时需  # 列表中  # fields  # FilterUserAdmin  # ScoreConfigAdmin 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: laravel怎么用DB facade执行原生SQL查询_laravel DB facade原生SQL执行方法  JavaScript如何实现路由_前端路由原理是什么  ,怎么在广州志愿者网站注册?  Windows10如何更改计算机工作组_Win10系统属性修改Workgroup  ,交易猫的商品怎么发布到网站上去?  公司门户网站制作流程,华为官网怎么做?  Midjourney怎样加参数调细节_Midjourney参数调整技巧【指南】  Laravel如何与Pusher实现实时通信?(WebSocket示例)  Claude怎样写结构化提示词_Claude结构化提示词写法【教程】  Laravel怎么导出Excel文件_Laravel Excel插件使用教程  做企业网站制作流程,企业网站制作基本流程有哪些?  Laravel如何使用Service Provider服务提供者_Laravel依赖注入与容器绑定【深度】  手机怎么制作网站教程步骤,手机怎么做自己的网页链接?  JavaScript如何实现错误处理_try...catch如何捕获异常?  常州企业网站制作公司,全国继续教育网怎么登录?  Python面向对象测试方法_mock解析【教程】  Laravel Eloquent访问器与修改器是什么_Laravel Accessors & Mutators数据处理技巧  免费视频制作网站,更新又快又好的免费电影网站?  iOS中将个别页面强制横屏其他页面竖屏  阿里云高弹*务器配置方案|支持分布式架构与多节点部署  如何快速辨别茅台真假?关键步骤解析  深圳防火门网站制作公司,深圳中天明防火门怎么编码?  如何在IIS管理器中快速创建并配置网站?  *服务器网站为何频现安全漏洞?  零基础网站服务器架设实战:轻量应用与域名解析配置指南  实例解析angularjs的filter过滤器  移动端脚本框架Hammer.js  微信公众帐号开发教程之图文消息全攻略  Laravel怎么进行浏览器测试_Laravel Dusk自动化浏览器测试入门  Laravel如何保护应用免受CSRF攻击?(原理和示例)  如何续费美橙建站之星域名及服务?  如何用狗爹虚拟主机快速搭建网站?  Laravel如何实现API版本控制_Laravel版本化API设计方案  Laravel如何与Vue.js集成_Laravel + Vue前后端分离项目搭建指南  Mybatis 中的insertOrUpdate操作  如何在宝塔面板中修改默认建站目录?  JavaScript实现Fly Bird小游戏  西安专业网站制作公司有哪些,陕西省建行官方网站?  Laravel怎么实现验证码(Captcha)功能  如何在万网自助建站平台快速创建网站?  如何快速搭建个人网站并优化SEO?  如何在云主机快速搭建网站站点?  电商网站制作多少钱一个,电子商务公司的网站制作费用计入什么科目?  Laravel怎么创建控制器Controller_Laravel路由绑定与控制器逻辑编写【指南】  uc浏览器二维码扫描入口_uc浏览器扫码功能使用地址  linux top下的 minerd 木马清除方法  如何挑选优质建站一级代理提升网站排名?  手机网站制作平台,手机靓号代理商怎么制作属于自己的手机靓号网站?  厦门模型网站设计制作公司,厦门航空飞机模型掉色怎么办?  Edge浏览器怎么启用睡眠标签页_节省电脑内存占用优化技巧