質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

新規登録して質問してみよう
ただいま回答率
85.30%
Django

DjangoはPythonで書かれた、オープンソースウェブアプリケーションのフレームワークです。複雑なデータベースを扱うウェブサイトを開発する際に必要な労力を減らす為にデザインされました。

Python

Pythonは、コードの読みやすさが特徴的なプログラミング言語の1つです。 強い型付け、動的型付けに対応しており、後方互換性がないバージョン2系とバージョン3系が使用されています。 商用製品の開発にも無料で使用でき、OSだけでなく仮想環境にも対応。Unicodeによる文字列操作をサポートしているため、日本語処理も標準で可能です。

Q&A

解決済

2回答

780閲覧

【Django】CreateViewでユーザー入力項目以外のフィールド値をセットしたい

Mr_PONPON_MARU

総合スコア35

Django

DjangoはPythonで書かれた、オープンソースウェブアプリケーションのフレームワークです。複雑なデータベースを扱うウェブサイトを開発する際に必要な労力を減らす為にデザインされました。

Python

Pythonは、コードの読みやすさが特徴的なプログラミング言語の1つです。 強い型付け、動的型付けに対応しており、後方互換性がないバージョン2系とバージョン3系が使用されています。 商用製品の開発にも無料で使用でき、OSだけでなく仮想環境にも対応。Unicodeによる文字列操作をサポートしているため、日本語処理も標準で可能です。

0グッド

0クリップ

投稿2023/06/29 01:26

実現したいこと

会員制の記事投稿アプリをDjangoで作成しています。
記事へコメントを入力し送信する機能を実装しています。
そこで、ユーザーが入力する項目以外のフィールドをビュー側であらかじめセットできる方法について教えていただければなと思います。

前提

モデルはユーザ情報に関するCustomUser、記事に関するArticle、記事へのコメントに関するCommentモデルを用意しています。

models.py

1class CustomUser(AbstractUser): 2 3 username = models.CharField( 4 _("username"), 5 max_length=30, 6 help_text='Required 30 characters or fewer.', 7 unique=True, 8 error_messages={ 9 'unique': _("This Username already exists."), 10 },) 11 12 email = models.EmailField( 13 _('email'), 14 unique=True, 15 error_messages={ 16 'unique': _("A user with that email address already exists."), 17 },) 18 19 class Meta: 20 verbose_name_plural = 'CustomUser' 21 22 23class Article(models.Model): 24 25 post_user = models.ForeignKey(CustomUser, verbose_name='Post User', on_delete=models.CASCADE, related_name='name',) 26 title = models.CharField(verbose_name='title', max_length=50,) 27 content = MDTextField() 28 created_at = models.DateField(verbose_name='created_at', auto_now_add=True,) 29 30 class Meta: 31 verbose_name_plural = 'Article' 32 33 def __str__(self): 34 return self.title 35 36class Comment(models.Model): 37 #記事に対するコメント 38 writer = models.ForeignKey(CustomUser, on_delete=models.CASCADE,) 39 text = MDTextField() 40 target = models.ForeignKey(Article, on_delete=models.CASCADE,) 41 created_at = models.DateField(verbose_name='created_at', auto_now_add=True,) 42 updated_at = models.DateField(verbose_name='updated_at', auto_now=True,) 43 44 def __str__(self): 45 return self.text[:20]

コメントフォームは記事詳細ページ(article_detail.html)に配置しており、コメントフォーム送信後のリダイレクト先も記事詳細ページを指定しています。

views.py

1class CommentCreateView(generic.CreateView, LoginRequiredMixin): 2 model = Comment 3 form_class = CommentCreateForm 4 template_name = 'article_detail.html' 5 6 def form_valid(self, form): 7 comment = form.save(commit=False) 8 comment.writer = self.request.user 9 comment.target = Article.objects.get(pk=self.pk).pk #ここの部分がわからない 10 comment.save() 11 messages.success(self.request, 'You added a comment!') 12 return super().form_valid(form) 13 14 def get_success_url(self): 15 return reverse_lazy('article:article_detail', kwargs={'pk': self.kwargs['pk']})

urls.py

1urlpatterns = [ 2 path('article_detail/<int:pk>', views.ArticleDetailView.as_view(), name='article_detail'), 3 path('<int:pk>/comment', views.CommentCreateView.as_view(), name='comment_create'), 4]

発生している問題・エラーメッセージ

ブラウザ上でAttributeErrorが発生しました。

AttributeError at /19/comment 'CommentCreateView' object has no attribute 'pk'

考えてみたこと

comment.writer = self.request.userとしてユーザー情報をセットしている部分はうまくいっているようですが、comment.target = Article.objects.get(pk=self.pk).pkでコメント先の記事をセットしている部分に失敗しているようです。この部分の設定の仕方について、どうすればいいか教えていただけると幸いです。

気になる質問をクリップする

クリップした質問は、後からいつでもMYページで確認できます。

またクリップした質問に回答があった際、通知やメールを受け取ることができます。

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

guest

回答2

0

各種フォームへの初期値設定はフォームモデルに対してディクショナリをinitialプロパティにセットします。

py:views.py

1initial_dict = dict(id=id) #初期値設定したいプロパティを代入し、初期値ディクショナリを作成 2form = HogeForm(initial = initial_dict) #任意のフォームモデルに対して、初期値をinitialプロパティに代入

投稿2023/06/29 02:12

FKM

総合スコア3660

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

Mr_PONPON_MARU

2023/07/01 02:15

コメントありがとうございます。 フォームで表示はさせない項目だがモデルに登録する際には必須となる項目(今回はwriterとtarget)を扱ったため、下記の手順でform_valid()をオーバーライドしました。 ・`comment = form.save(commit=False)`でDBへの保存は行わずにフォーム入力値を保持する ・`comment.writer = self.request.user`と`comment.target = post`でフォームの非入力項目の値をセット ・`comment.save()`でDBへの保存を行う ```py:views.py initial_dict = dict(id=id) #初期値設定したいプロパティを代入し、初期値ディクショナリを作成 form = HogeForm(initial = initial_dict) #任意のフォームモデルに対して、初期値をinitialプロパティに代入 ``` こちらの方法は「フォームで表示させる項目を、あらかじめ規定値として設定しておく手法」という認識でよろしいでしょうか?
FKM

2023/07/01 02:23

その認識で大丈夫です。例ではidだけですがカンマを増やしていけば、いくつでも設定できます(フォームモデルで定義したプロパティに限る)。hiddenでも使えたはずです。 あと、もう一つフォーム転送時に、フォームモデルに対し既定値を設定する方法があったのでそれは職場に戻ってから確認してみます。
Mr_PONPON_MARU

2023/07/01 02:43

教えていただきありがとうございます。勉強になります。
guest

0

自己解決

下記にようにすることで解決しました。

views.py

1class CommentCreateView(generic.CreateView, LoginRequiredMixin): 2 model = Comment 3 form_class = CommentCreateForm 4 template_name = 'article_detail.html' 5 6 def form_valid(self, form): 7 comment = form.save(commit=False) 8 9 post_pk = self.kwargs['pk'] 10 post = get_object_or_404(Article, pk=post_pk) 11 12 comment.writer = self.request.user 13 comment.target = post 14 comment.save() 15 messages.success(self.request, 'You added a comment!') 16 return super().form_valid(form) 17 18 def get_success_url(self): 19 return reverse_lazy('article:article_detail', kwargs={'pk': self.kwargs['pk']})

投稿2023/06/29 02:06

Mr_PONPON_MARU

総合スコア35

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

15分調べてもわからないことは
teratailで質問しよう!

ただいまの回答率
85.30%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問