具有ManyToManyField的Django自定义表单:如何从网络添加新项

时间:2019-11-04 06:17:04

标签: django-forms django-admin

我正在为使用Django的Blog Post添加自定义表单。我定义的Post模型具有ManytoManyField,并且尝试在django admin.ModelAdmin中创建Post时,它具有“ +”按钮,用户可以添加它。但是,当我创建自定义窗体视图时,它没有出现。有什么方法可以创建“ +”按钮,以便用户可以创建新项目并通过单击将其添加?

使用django admin.ModelAdmin添加Post Model时,它具有如下屏幕截图(classPostForm中的keywords = ...已注释掉以获得此信息)

Creating with django admin.ModelAdmin

但是,在自定义视图(下面包含源代码)中,“ +”按钮不存在。 Creating with custom form

# models.py
class Post(models.Model):
    title = models.CharField(max_length=255)
    slug = models.SlugField(max_length=200, unique=True)
    keywords = models.ManyToManyField(Keyword, blank=True)
# forms.py
class PostForm(forms.ModelForm):
    keywords = forms.ModelMultipleChoiceField(
        Keyword.objects.all(),
        widget=widgets.FilteredSelectMultiple('Keywords', False),
        required=False,
    )

    def __init__(self, *args, **kwargs):
        super(PostForm, self).__init__(*args, **kwargs)

    class Meta:
        model = Post
        fields = ('title', 'keywords')

        querysets = {
            'keywords': Keyword.objects.all()
        }

    class Media:
        css = {'all': ('/admin/css/widgets.css',
                       'admin/css/overrides.css'), }
        js = ('/admin/jquery.js', '/admin/jsi18n/')
# views.py
class CreatePost(generic.CreateView):
    model = Post
    form_class = PostForm
    template_name = 'post_create.html'
<!---base.html---!>
<!DOCTYPE html>
<html>
<head>
    <title>Doc Flow</title>

    <link
            href="https://fonts.googleapis.com/css?family=Roboto:400,700"
            rel="stylesheet">
    <meta content="notranslate" name="google"/>
    <meta content="width=device-width, initial-scale=1" name="viewport"/>
    <link
            crossorigin="anonymous"
            href="https://maxcdn.bootstrapcdn.com/bootstrap/4.0.0/css/bootstrap.min.css"
            integrity="sha384-Gn5384xqQ1aoWXA+058RXPxPg6fy4IWvTNh0E263XmFcJlSAwiGgFAW/dAiS6JXm"
            rel="stylesheet"
    />
    <meta content="width=device-width, initial-scale=1" name="viewport">
    <link href="https://maxcdn.bootstrapcdn.com/bootstrap/3.4.0/css/bootstrap.min.css" rel="stylesheet">
    <script src="https://ajax.googleapis.com/ajax/libs/jquery/3.4.1/jquery.min.js"></script>
    <script src="https://maxcdn.bootstrapcdn.com/bootstrap/3.4.0/js/bootstrap.min.js"></script>

    <a href="/admin/blog/post/add" class="top-menu"><span class="glyphicon glyphicon-plus"></span></a>
    {{ form.media }}
        <meta charset="utf-8">
    {% block extra_head %}
    {% endblock %}
</head>

<body>
<style>
      body {
        font-family: "Roboto", sans-serif;
        font-size: 17px;
        background-color: #fdfdfd;
      }

    .shadow{
           box-shadow: 0 4px 2px -2px rgba(0,0,0,0.1);
       }
      .btn-danger {
        color: #fff;
        background-color: #f00000;
        border-color: #dc281e;
      }

     .masthead {
              background:#3398E1;
              height: auto;
              padding-bottom: 15px;
              box-shadow: 0 16px 48px #E3E7EB;
              padding-top: 10px;
    }

    .div_slide {
        width: 30%;
        float: left;
    }
    .slide-container {
      overflow: auto;
      white-space: nowrap;
    }

</style>

<!-- Navigation -->
<nav class="navbar navbar-expand-lg navbar-light bg-light shadow" id="mainNav">
    <div class="container-fluid">
        <a class="navbar-brand" href="{% url 'home' %}">Front</a>
        <button
                aria-controls="navbarResponsive"
                aria-expanded="false"
                aria-label="Toggle navigation"
                class="navbar-toggler navbar-toggler-right"
                data-target="#navbarResponsive"
                data-toggle="collapse"
                type="button"
        >
            <span class="navbar-toggler-icon"></span>
        </button>

    </div>
    </div>
</nav>
{% block content %}
<!-- Content Goes here -->
{% endblock content %}

<!-- Footer -->
<footer class="py-3 bg-grey">
    <p class="m-0 text-dark text-center ">Copyright &copy; Workhorse</p>
</footer>

</body>
</html>
<!---post_create.html---!>
{% extends 'base.html' %}
{% block extra_head %}
  <link rel="stylesheet" type="text/css" href="/static/admin/css/base.css">
  <link rel="stylesheet" type="text/css" href="/static/admin/css/forms.css">
  <link rel="stylesheet" type="text/css" href="/static/admin/css/responsive.css">
{% endblock %}

{% block content %}
  <form method="post" autocomplete="off" novalidate>
    {% csrf_token %}
    {{ form|crispy }}
    <button type="submit" class="btn btn-success">Create event</button>
  </form>
{% endblock %}

1 个答案:

答案 0 :(得分:0)

lateinit var userId: String val sessionScope = getKoin().createScope(userId,name("SESSION")) val userKey : SecretKey by scope.inject(name("sessionKey")) 负责在窗口小部件上添加“ +”号。因此,您需要将RelatedFieldWidgetWrapper包装在其中,以使其正常工作。

FilteredSelectMultiple

尽管在文档中不容易找到,但我必须在Django代码库中进行搜索。