Django Admin 实现外键过滤的方法


Posted in Python onSeptember 29, 2017

说明和 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 实现外键过滤的方法就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持三水点靠木。

Python 相关文章推荐
python实现创建新列表和新字典,并使元素及键值对全部变成小写
Jan 15 Python
python单线程文件传输的实例(C/S)
Feb 13 Python
12个Python程序员面试必备问题与答案(小结)
Jun 24 Python
python3.5 cv2 获取视频特定帧生成jpg图片
Aug 28 Python
Python进度条的制作代码实例
Aug 31 Python
Python-Flask:动态创建表的示例详解
Nov 22 Python
python隐藏类中属性的3种实现方法
Dec 19 Python
keras slice layer 层实现方式
Jun 11 Python
Python extract及contains方法代码实例
Sep 11 Python
Python图像识别+KNN求解数独的实现
Nov 13 Python
python爬虫请求头的使用
Dec 01 Python
使用Python获取字典键对应值的方法
Apr 26 Python
python 调用c语言函数的方法
Sep 29 #Python
python文件名和文件路径操作实例
Sep 29 #Python
Python 实现简单的shell sed替换功能(实例讲解)
Sep 29 #Python
Python 基础教程之闭包的使用方法
Sep 29 #Python
python下实现二叉堆以及堆排序的示例
Sep 29 #Python
Python数据结构与算法之链表定义与用法实例详解【单链表、循环链表】
Sep 28 #Python
Python实现压缩和解压缩ZIP文件的方法分析
Sep 28 #Python
You might like
英雄试炼之肉山谷—引领RPG新潮流
2020/04/20 DOTA
PHP入门
2006/10/09 PHP
PHP 杂谈《重构-改善既有代码的设计》之四 简化条件表达式
2012/04/09 PHP
DEDECMS首页调用图片集里的多张图片
2015/06/05 PHP
解读PHP的Yii框架中请求与响应的处理流程
2016/03/17 PHP
FileUpload上传图片(图片不变形)
2010/08/05 Javascript
jquery事件重复绑定的快速解决方法
2014/01/03 Javascript
页面装载js及性能分析方法介绍
2014/03/21 Javascript
JavaScript记录光标在编辑器中位置的实现方法
2016/04/22 Javascript
一个字符串中出现次数最多的字符 统计这个次数【实现代码】
2016/04/29 Javascript
JS判断浏览器是否安装flash插件的简单方法
2016/09/13 Javascript
nodejs集成sqlite使用示例
2017/06/05 NodeJs
jquery插件canvaspercent.js实现百分比圆饼效果
2017/07/18 jQuery
Vue + Vue-router 同名路由切换数据不更新的方法
2017/11/20 Javascript
vue-cli+webpack项目 修改项目名称的方法
2018/02/28 Javascript
微信小程序获取用户openid的实现
2018/12/24 Javascript
js中null与空字符串&quot;&quot;的区别讲解
2019/01/17 Javascript
vue-cli3环境变量与分环境打包的方法示例
2019/02/18 Javascript
AjaxFileUpload.js实现异步上传文件功能
2019/04/19 Javascript
jquery 遍历hash操作示例【基于ajax交互】
2019/10/12 jQuery
[01:56]生活中的妖精之七夕特别档
2016/08/09 DOTA
Python使用xlrd读取Excel格式文件的方法
2015/03/10 Python
Python GUI库PyQt5样式QSS子控件介绍
2020/02/25 Python
Python爬虫代理池搭建的方法步骤
2020/09/28 Python
python上下文管理器异常问题解决方法
2021/02/07 Python
用CSS3实现无限循环的无缝滚动的实例代码
2017/07/04 HTML / CSS
英国最大的在线照明商店:Litecraft
2020/08/31 全球购物
如何写一个Java类既可以用作applet也可以用作java应用
2016/01/18 面试题
毕业生毕业总结的自我评价范文
2013/11/02 职场文书
品质主管的岗位职责
2013/12/04 职场文书
商场父亲节活动方案
2014/08/27 职场文书
入党政审材料范文
2014/12/24 职场文书
完美解决golang go get私有仓库的问题
2021/05/05 Golang
解析CSS 提取图片主题色功能(小技巧)
2021/05/12 HTML / CSS
python控制台打印log输出重复的解决方法
2021/05/14 Python
vue二维数组循环嵌套方式 循环数组、循环嵌套数组
2022/04/24 Vue.js