Django:必须使用对象pk或slug调用通用详细信息视图

时间:2014-04-18 12:49:31

标签: python django

提交与此视图关联的表单时出现此错误。不确定究竟是什么问题,考虑到我有一个结构非常相似的表格并且工作正常。

#views.py
class Facture_Creer(SuccessMessageMixin, CreateView):
    model = Facture
    template_name = "facturation/nouvelle_facture.html"
    form_class= FormulaireFacture

    # permet de retourner a l'URL pointant vers le membre modifie
    def get_success_url(self):
        return reverse_lazy('facture_consulter',kwargs={'pk': self.get_object().id})

class Facture_Update(SuccessMessageMixin, UpdateView):
    model = Facture
    template_name = "facturation/nouvelle_facture.html"
    form_class= FormulaireFacture
    success_message = "Facture mise à jour avec succes"

    # permet de retourner a l'URL pointant vers le membre modifie
    def get_success_url(self):
        return reverse_lazy('facture_consulter',kwargs={'pk': self.get_object().id})

#urls.py
urlpatterns = patterns('',
    url(r'^$', TemplateView.as_view(template_name="facturation/index.html")),
    url(r'^facture/$', FactureView.as_view()),
    url(r'^facture/(?P<id>\d+)', FactureView.as_view(), name='facture_consulter'),
    url(r'^facture/ajouter/$', Facture_Creer.as_view(), name='facture_creer'),
    url(r'^facture/modifier/(?P<pk>\d+)/$', Facture_Update.as_view(), name='facture_update'),
    url(r'^membre/ajouter/$', Membre_Creer.as_view(), name='membre_creer'),
    url(r'^membre/modifier/(?P<pk>\d+)/$', Membre_Update.as_view(), name='membre_update'),
    #url(r'membre/(?P<pk>\d+)/supprimer/$', Membre_Supp.as_view(), name='membre_delete')
)

urlpatterns += staticfiles_urlpatterns()

5 个答案:

答案 0 :(得分:34)

您需要传递一个对象标识符(pk或slug),以便您的视图知道它们正在操作哪个对象。

只是从您的urls.py

中举例说明
url(r'^facture/ajouter/$', Facture_Creer.as_view(), name='facture_creer'),
url(r'^facture/modifier/(?P<pk>\d+)/$', Facture_Update.as_view(), name='facture_update'),

了解第二个人如何(?P<pk>\d+)/?这是将pk传递给UpdateView,因此它知道要使用哪个对象。因此,如果您转到facture/modifier/5/,则UpdateView将修改pk为5的对象。

如果您不想在网址中传递pk或slug,则需要覆盖get_object方法并以另一种方式获取对象。网址here

答案 1 :(得分:8)

正如Alex建议的那样:对于默认的Django行为,你必须使用&#34; pk&#34;在你的网址模式。

如果您想更改主键的对象标识符&#34; pk&#34;使用其他名称,您可以定义pk_url_kwarg。这是从Django 1.4开始提供的。

答案 2 :(得分:2)

嘿,我全部使用了新的path()函数,这是我确定会有所帮助的工作示例:

views.py:

from django.views.generic.detail import DetailView

class ContentAmpView(DetailView):

    model = Content
    template_name = 'content_amp.html'  # Defaults to content_detail.html

urls.py:

from django.urls import path

from .views import ContentAmpView

# My pk is a string so using a slug converter here intead of int
urlpatterns = [
    path('<slug:pk>/amp', ContentAmpView.as_view(), name='content-amp'),
]

templates / content_amp.html

<!doctype html>
<html amp lang="en">
<head>
    <meta charset="utf-8">
    <script async src="https://cdn.ampproject.org/v0.js"></script>
    <title>Hello, AMPs</title>
    <link rel="canonical" href="http://example.ampproject.org/article-metadata.html">
    <meta name="viewport" content="width=device-width,minimum-scale=1,initial-scale=1">
    <script type="application/ld+json">
      {
        "@context": "http://schema.org",
        "@type": "NewsArticle",
        "headline": "Open-source framework for publishing content",
        "datePublished": "2015-10-07T12:02:41Z",
        "image": [
          "logo.jpg"
        ]
      }

    </script>
    <style amp-boilerplate>
        body{-webkit-animation:-amp-start 8s steps(1,end) 0s 1 normal both;-moz-animation:-amp-start 8s steps(1,end) 0s 1 normal both;-ms-animation:-amp-start 8s steps(1,end) 0s 1 normal both;animation:-amp-start 8s steps(1,end) 0s 1 normal both}@-webkit-keyframes -amp-start{from{visibility:hidden}to{visibility:visible}}@-moz-keyframes -amp-start{from{visibility:hidden}to{visibility:visible}}@-ms-keyframes -amp-start{from{visibility:hidden}to{visibility:visible}}@-o-keyframes -amp-start{from{visibility:hidden}to{visibility:visible}}@keyframes -amp-start{from{visibility:hidden}to{visibility:visible}}
    </style>
    <noscript>
        <style amp-boilerplate>body{-webkit-animation:none;-moz-animation:none;-ms-animation:none;animation:none}
        </style>
    </noscript>
</head>
<body>
<h1>Welcome to AMP - {{ object.pk }}</h1>
<p>{{ object.titles.main }}</p>
<p>Reporter: {{ object.reporter }}</p>
<p>Date: {{ object.created_at|date }}</p>
</body>
</html>

还要注意,在我的settings.py的{​​{1}}下,我有TEMPLATES。有关路径here的更多信息。

答案 3 :(得分:2)

像罗宾(Robin)所说,如果您想要自定义pk名称,则可以使用pk_url_kwarg

但是,此外,问题一般详细视图必须使用对象pk或子弹来调用也在子弹中引发。

因此,如果您要创建自定义弹头字段(无需使用 pk 弹头名称)。您可以在DetailView类中覆盖slug_fieldslug_url_kwarg

如果要将url作为Slug Field,这只是另一个示例。

models.py

class Post(models.Model):
    title = models.CharField(max_length=255)
    url = models.SlugField()
    body = models.TextField()
    image = models.ImageField(upload_to='blog/', blank=True, null=True)

views.py

class ListPost(ListView):
    model = Post
    template_name = 'blog/list.html'
    paginate_by = 2
    context_object_name = 'posts'

class DetailPost(DetailView):
    model = Post
    template_name = 'blog/detail.html'
    slug_field = 'url'
    slug_url_kwarg = 'url'

urls.py

urlpatterns = [
    path('', ListPost.as_view()),
    path('<slug:url>/', DetailPost.as_view()),
]

答案 4 :(得分:0)

我正在使用Django 2.2.12版,

这只是另一个例子,对我来说很不错:

models.py

<div class="jumbotron item_list">
     <h3>Item(s) in cart: {{ count }}</h3>
     <table class="table table-striped">
         <thead>
            <th>Name</th>
            <th>Price</th>
            <th>Remove</th>
         </thead>
         <tbody>
             {% for item in cart %}
             <tr>
                 <td>
                     {{ item.cart_item }}
                     {% if item.extras != "" %}
                     <br><sup>({{ item.extras }})</sup>
                     {% endif %}
                 </td>
                 <td>${{ item.item_price }}</td>
            {% endfor %}
        </tbody>
   </table>
</div>

views.py

from django.db import models

class Book(models.Model):
    title = models.CharField(max_length=256)
    author = models.CharField(max_length=256)
    pages = models.IntegerField()
    price = models.FloatField()

urls.py

from django.views.generic import ListView,DetailView
from testapp.models import Book

class BookListView(ListView):
    model=Book

class BookDetailView(DetailView):
    model=Book