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

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

ただいまの
回答率

89.06%

formsetを利用したデータの一括削除について

解決済

回答 3

投稿 編集

  • 評価
  • クリップ 1
  • VIEW 1,420

isshi

score 8

Django初心者です。
Djangoのformsetを利用してレコードを複数登録後、
同様にformsetを利用して一括削除しようとしています。
formsetに追加した削除チェックボックスにチェック後、submitしても削除される事なくエラー表示されず指定画面に遷移する状態です。

登録画面表示時にモデルに登録されているデータが全て表示されるのですが、 これを表示させない、または特定条件のデータのみ表示させるにはどうしたら良いでしょうか。

削除処理をする上でどこが誤っているのかわからない状態です。
どうぞご教授のほどよろしくお願いします。

 models.py

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

"""書籍"""
class Book(models.Model):

    book_choices = (
        ("雑誌", '雑誌'),
        ("新書", '新書'),
        ("参考書", '参考書'),
        ("漫画", '漫画'),
    )
    # id = models.AutoField(primary_key=True)
    bookType = models.CharField("本種", max_length=3, blank=False, choices=book_choices,default="雑誌")
    name = models.CharField('書籍名', max_length=255)
    publisher = models.CharField('出版社', max_length=255, blank=True)
    page = models.IntegerField('ページ数', blank=True, default=0)
    impressionCount = models.IntegerField('感想数', blank=True, default=0)
    total_readCount = models.IntegerField('総読書回数', blank=True, default=0)
    editer = models.CharField('登録者', max_length=255, blank=True)
    created_at = models.DateTimeField('登録日時', default=datetime.now)
    updated_at = models.DateTimeField('更新日時', auto_now=True, blank=True, null=True)  # 更新日時

    def __str__(self):
        return self.name

 forms.py

from django import forms
from .models import Book,Impression

class BookCreateForm(forms.ModelForm):
    """書籍のフォーム"""
    def __init__(self, *args, **kwargs):
        super(BookCreateForm, self).__init__(*args, **kwargs)
        for field in self.fields.values():
            field.widget.attrs["class"] = "form-control"

    class Meta:
        model = Book
        fields = ('bookType', 'name', 'publisher', 'page', 'impressionCount',)

BookFormSet = forms.modelformset_factory(
    Book, form=BookCreateForm, extra=0, can_delete=True)

 views.py

from django.shortcuts import render

# Create your views here.
from django.shortcuts import get_object_or_404, redirect,render,reverse
from django.contrib.auth.decorators import login_required
from django.utils.decorators import method_decorator
from django.contrib.auth.models import User
from django.contrib import messages  # メッセージフレームワーク
from django.urls import reverse_lazy
from bookmgr.models import Book,Impression
from django.views import generic
from .forms import BookCreateForm,BookFormSet #ImpressionCreateForm
from django.db.models import Q,Count,Sum

"""書籍一覧"""
@method_decorator(login_required, name='dispatch')
class IndexView(generic.ListView):
    model = Book
    context_object_name = 'books'
    template_name = 'book_list.html'
    paginate_by = 7

    def get_queryset(self):
        # 子数を親の感想数にセット
        # qs = Book.objects.filter(impressions__isnull=False).values('id').annotate(impre_count=Count('id'))
        # for i in qs:
        #     Book.objects.filter(id=i['id']).update(impressionCount=i['impre_count']) # 辞書型のキーを指定して更新
        queryset = Book.objects.all().order_by('-created_at')
        keyword = self.request.GET.get('keyword')
        if keyword:
            queryset = queryset.filter(
                Q(name__icontains=keyword) | Q(bookType__icontains=keyword)
            )
        return queryset


class multiEditView(generic.FormView):
    template_name = 'book_multiEdit.html'
    form_class = BookFormSet
    success_url = reverse_lazy('bookmgr:index')

    def get_form(self, form_class=None):
         return BookFormSet(self.request.POST or None)

    def form_valid(self, form):
        for fm in form:
            book = fm.save(commit=False)
            book.editer = str(self.request.user)
            book.save()
        return super().form_valid(form)

 book_multiEdit.html

{% block title %}一括登録{% endblock title %}

<!doctype html>
<html lang="ja">
<head>
  <!-- Required meta tags -->
  <meta charset="utf-8">
  <meta name="viewport" content="width=device-width, initial-scale=1, shrink-to-fit=no">

  <!-- Bootstrap CSS -->
  <link rel="stylesheet" href="https://stackpath.bootstrapcdn.com/bootstrap/4.1.0/css/bootstrap.min.css"
        integrity="sha384-9gVQ4dYFwwWSjIDZnLEWnxCjeSWFphJiwGPXr1jddIhOegiu1FwO5qRGvFXOdJZ4" crossorigin="anonymous">

  <title>一括登録</title>
</head>


<form action="" method="POST">
{{ form.management_form }}
{% for fm in form %}
{{ fm.id }}
{{ fm.errors }}
  <table class="table">
      <tr>
        <th>書籍タイプ</th>
        <td>{{ fm.bookType }}</td>
        <th>書籍名</th>
        <td>{{ fm.name }}</td>
        <th>出版社</th>
        <td>{{ fm.publisher }}</td>
        <th>ページ数</th>
        <td>{{ fm.page }}</td>
      {% if fm.instance.pk %}
        <th>削除</th>
        <td>{{ fm.DELETE }}</td>
      {% endif %}
      </tr>
  </table>
{% endfor %}
  <button type="submit" class="btn btn-primary">送信</button>
  <a href="{% url 'bookmgr:index' %}" class="btn btn-primary btn-sm">戻る</a>
  {% csrf_token %}
</form>

 urls.py

from django.urls import path,include
from django.conf import settings
from . import views
from django.contrib.auth import views as auth_views

app_name = 'bookmgr'

urlpatterns = [
    # 書籍
    path('booklist/', views.IndexView.as_view(), name='index'),  # 一覧
    path('add/', views.AddView.as_view(), name='book_add'),  # 追加
    path('multiadd/', views.multiAddView.as_view(), name='book_multiadd'),

    # ログアウト
    path('accounts/logout/', auth_views.LogoutView.as_view(), name='logout'),


]

 その他、補足情報

・Django:2.0.7
・Python:3.6.5

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

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

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

    クリップを取り消します

  • 良い質問の評価を上げる

    以下のような質問は評価を上げましょう

    • 質問内容が明確
    • 自分も答えを知りたい
    • 質問者以外のユーザにも役立つ

    評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。

    質問の評価を上げたことを取り消します

  • 評価を下げられる数の上限に達しました

    評価を下げることができません

    • 1日5回まで評価を下げられます
    • 1日に1ユーザに対して2回まで評価を下げられます

    質問の評価を下げる

    teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。

    • プログラミングに関係のない質問
    • やってほしいことだけを記載した丸投げの質問
    • 問題・課題が含まれていない質問
    • 意図的に内容が抹消された質問
    • 過去に投稿した質問と同じ内容の質問
    • 広告と受け取られるような投稿

    評価が下がると、TOPページの「アクティブ」「注目」タブのフィードに表示されにくくなります。

    質問の評価を下げたことを取り消します

    この機能は開放されていません

    評価を下げる条件を満たしてません

    評価を下げる理由を選択してください

    詳細な説明はこちら

    上記に当てはまらず、質問内容が明確になっていない質問には「情報の追加・修正依頼」機能からコメントをしてください。

    質問の評価を下げる機能の利用条件

    この機能を利用するためには、以下の事項を行う必要があります。

質問への追記・修正、ベストアンサー選択の依頼

  • Meganezaru

    2018/08/01 22:28

    念のため確認ですが、登録について、add/は動くけど、multiadd/がうまくいかないという質問でよかったですか?

    キャンセル

回答 3

check解決した方法

0

FormViewクラスのdef form_valid中で、
instances = form.save(commit=False)

その後、以下のように削除チェックがついたbookを取り出して削除
for schedule in form.deleted_objects:
schedule.delete()

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

0

どう上手くいかないのか 質問をもう少し詳しく書いていただけますか

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2018/08/02 12:18

    登録処理については自己解決しました。
    もう一方のフォームデータの絞り込みが分かりません。

    キャンセル

0

FormViewクラスのdef form_valid中で、
instances = form.save(commit=False)

その後、以下のように削除チェックがついたbookを取り出して削除
for schedule in form.deleted_objects:
schedule.delete()

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

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

  • ただいまの回答率 89.06%
  • 質問をまとめることで、思考を整理して素早く解決
  • テンプレート機能で、簡単に質問をまとめられる

関連した質問

同じタグがついた質問を見る