23

Class Based ListViewはテーブルセットの選択を表示しています。サイトが初めて要求された場合は、データセットが表示されます。私はPOSTの提出を好むだろうが、GETもうまくいきます。Django:クラスベースのListViewの検索フォーム

これは問題です。これはfunction based viewsで扱いやすくなりましたが、クラスベースのビューでは頭がおかしくなりません。

私の問題は、クラスベースのビューの制限された理解に起因するさまざまなエラーが発生することです。私は様々なドキュメントを読んできましたが、直接クエリ要求のビューを理解していますが、クエリステートメントにフォームを追加するとすぐに、別のエラーが発生します。以下のコードでは、ValueError: Cannot use None as a query valueが届きます。

フォームエントリに応じてクラスベースのListViewのベストプラクティスワークフローは何ですか(それ以外の場合はデータベース全体を選択します)。

これは私のサンプルコードです:

models.py

class Profile(models.Model): 
    name = models.CharField(_('Name'), max_length=255) 

    def __unicode__(self): 
     return '%name' % {'name': self.name} 

    @staticmethod 
    def get_queryset(params): 

     date_created = params.get('date_created') 
     keyword = params.get('keyword') 
     qset = Q(pk__gt = 0) 
     if keyword: 
      qset &= Q(title__icontains = keyword) 
     if date_created: 
      qset &= Q(date_created__gte = date_created) 
     return qset 

forms.py

class ProfileSearchForm(forms.Form): 
    name = forms.CharField(required=False) 

views.py

class ProfileList(ListView): 
    model = Profile 
    form_class = ProfileSearchForm 
    context_object_name = 'profiles' 
    template_name = 'pages/profile/list_profiles.html' 
    profiles = [] 


    def post(self, request, *args, **kwargs): 
     self.show_results = False 
     self.object_list = self.get_queryset() 
     form = form_class(self.request.POST or None) 
     if form.is_valid(): 
      self.show_results = True 
      self.profiles = Profile.objects.filter(name__icontains=form.cleaned_data['name']) 
     else: 
      self.profiles = Profile.objects.all() 
     return self.render_to_response(self.get_context_data(object_list=self.object_list, form=form)) 

    def get_context_data(self, **kwargs): 
     context = super(ProfileList, self).get_context_data(**kwargs) 
     if not self.profiles: 
      self.profiles = Profile.objects.all() 
     context.update({ 
      'profiles': self.profiles 
     }) 
     return context 

以下、ジョブを実行するFBVを追加しました。 どのようにしてこの機能をCBVに変換できますか? 関数ベースのビューではとてもシンプルなようですが、クラスベースのビューではそうではありません。

def list_profiles(request): 
    form_class = ProfileSearchForm 
    model = Profile 
    template_name = 'pages/profile/list_profiles.html' 
    paginate_by = 10 

    form = form_class(request.POST or None) 
    if form.is_valid(): 
     profile_list = model.objects.filter(name__icontains=form.cleaned_data['name']) 
    else: 
     profile_list = model.objects.all() 

    paginator = Paginator(profile_list, 10) # Show 10 contacts per page 
    page = request.GET.get('page') 
    try: 
     profiles = paginator.page(page) 
    except PageNotAnInteger: 
     profiles = paginator.page(1) 
    except EmptyPage: 
     profiles = paginator.page(paginator.num_pages) 

    return render_to_response(template_name, 
      {'form': form, 'profiles': suppliers,}, 
      context_instance=RequestContext(request)) 
+1

質問:フォーム提出に基づいてクエリセットから値を表示しようとしていますか? –

+0

はい、それが目標でした。 – neurix

+0

この一般的なmixinアプローチを見てみましょう: http://stackoverflow.com/questions/7011773/how-to-create-a-filter-form-for-a-class-based-generic-object- list-in-django –

答えて

35

私は、もしそうなら、あなたの目標は、GETを使用して、フォームの送信に基づいてクエリセットをフィルタリングしようとしていると思う:

class ProfileSearchView(ListView) 
    template_name = '/your/template.html' 
    model = Person 

    def get_queryset(self): 
     try: 
      name = self.kwargs['name'] 
     except: 
      name = '' 
     if (name != ''): 
      object_list = self.model.objects.filter(name__icontains = name) 
     else: 
      object_list = self.model.objects.all() 
     return object_list 

次にあなたがする必要があるすべては、テンプレートをレンダリングするgetメソッドを記述し、コンテキスト。

おそらく最良の方法ではありません。上記のコードを使用することで、django形式を定義する必要はありません。

これはどのように動作するのですか:クラスベースのビューは、テンプレートのレンダリング、フォームの処理などの方法を切り分けます。同様に、getはGET応答を処理し、postはPOST応答を処理します。get_querysetget_objectは自明です。あなたはおよそListView

を知っているし、その後dir(ListView)を入力したい場合は

from django.views.generic import ListView:シェルと種類を起動し、メソッド利用できるかを知るための簡単な方法。そこに定義されているすべてのメソッドが表示され、ソースコードを参照して理解することができます。クエリーセットの取得にはget_querysetメソッドが使用されました。

class FooView(ListView): 
    template_name = 'foo.html' 
    queryset = Photo.objects.all() # or anything 

私たちは、上記のようにそれを行うことができますが、我々はそのアプローチを使用して、動的フィルタリングを行うことはできません。なぜ、ちょうどそれがあまりにも動作しますが、このようにそれを定義しません。 get_querysetを使用することで、私たちはすべてのデータ/値/情報を使用して動的フィルタリングを行うことができます。GETによって送信されるnameパラメータを使用することができ、kwargsまたはself.kwargs["some_key"]some_keyあなたが指定したパラメータ

+2

tryを使用する代わりにgetデフォルト値では、読みやすさに役立つと思います。 'name = self.kargs.get( 'name'、None)'それから 'name:#do some stuff' – agmezr

2

これは一般的なビューのトピックDynamic filteringでうまく説明されています。

GETでフィルタリングできますが、ListViewは編集ミックスから継承されていないため、POSTメソッドを使用することはできません。あなたは何ができるか

は次のとおりです。

urls.py

urlpatterns = patterns('', 
       (r'^search/(\w+)/$', ProfileSearchListView.as_view()), 
      ) 

views.py

class ProfileSearchListView(ListView): 
    model = Profile 
    context_object_name = 'profiles' 
    template_name = 'pages/profile/list_profiles.html' 
    profiles = [] 

    def get_queryset(self): 
     if len(self.args) > 0: 
       return Profile.objects.filter(name__icontains=self.args[0]) 
     else: 
       return Profile.objects.filter() 
9

まあ、私は形成するための検証を残すことはいいアイデアだと思います。多分それは非常にシンプルな形があるので、この特定のケースでは、それを価値がない - しかし、より複雑なものとしてください(そしておそらくあなたも成長します)ので、私は何かのようにするだろう:私はと思い

class ProfileList(ListView): 
    model = Profile 
    form_class = ProfileSearchForm 
    context_object_name = 'profiles' 
    template_name = 'pages/profile/list_profiles.html' 
    profiles = [] 


    def get_queryset(self): 
     form = self.form_class(self.request.GET) 
     if form.is_valid(): 
      return Profile.objects.filter(name__icontains=form.cleaned_data['name']) 
     return Profile.objects.all() 
+0

form_classが定義されていないというエラーが出ました。 –

+0

Vic Nicethemer、それを定義することができます – meteor

+3

@VicNicethemerフォーム= self.form_class(s​​elf.request.GET)にする必要があります。 –

1

あなたのフォームは名前フィールドを必要としないため、エラーが発生しています。したがって、フォームは有効ですが、nameフィールドのcleaned_dataは空です。あなたが停止した場合

self.profiles = Profile.objects.none() 

これらは問題の行のようになります。これに

self.profiles = Profile.objects.filter(name__icontains=form.cleaned_data['name']) 

:私があなただったら

if form.is_valid(): 
    self.show_results = True 
    self.profiles = Profile.objects.filter(name__icontains=form.cleaned_data['name']) 

、私はラインを変更しようとするだろうエラーを受け取りました(あなたのテンプレートは空のobject_listを受け取ります)、問題は私が言ったことですbefore:nameフィールドは必須ではありません。

これが機能しない場合は、お知らせください。モデル

class SearchListView(ItemsListView): 

# Display a Model List page filtered by the search query. 

def get_queryset(self): 
    fields = [m.name for m in super(SearchListView, self).model._meta.fields] 
    result = super(SearchListView, self).get_queryset() 
    query = self.request.GET.get('q') 
    if query: 
     result = result.filter(
      reduce(lambda x, y: x | Q(**{"{}__icontains".format(y): query}), fields, Q()) 
     ) 
    return result 
0

検索私はあなたがget_context_data経由でこれをやったほうが良いと思います。手動でHTMLフォームを作成し、GETを使用してこのデータを取得します。私が書いた例の例は以下の通りです。フォームを送信すると、取得データを使用してコンテキストデータを介して返信することができます。この例はあなたの要求に合わせたものではありませんが、他のユーザーには役立ちます。

def get_context_data(self, **kwargs): 
    context = super(Search, self).get_context_data(**kwargs) 
    filter_set = Gauges.objects.all() 
    if self.request.GET.get('gauge_id'): 
     gauge_id = self.request.GET.get('gauge_id') 
     filter_set = filter_set.filter(gauge_id=gauge_id) 

    if self.request.GET.get('type'): 
     type = self.request.GET.get('type') 
     filter_set = filter_set.filter(type=type) 

    if self.request.GET.get('location'): 
     location = self.request.GET.get('location') 
     filter_set = filter_set.filter(location=location) 

    if self.request.GET.get('calibrator'): 
     calibrator = self.request.GET.get('calibrator') 
     filter_set = filter_set.filter(calibrator=calibrator) 

    if self.request.GET.get('next_cal_date'): 
     next_cal_date = self.request.GET.get('next_cal_date') 
     filter_set = filter_set.filter(next_cal_date__lte=next_cal_date) 

    context['gauges'] = filter_set 
    context['title'] = "Gauges " 
    context['types'] = Gauge_Types.objects.all() 
    context['locations'] = Locations.objects.all() 
    context['calibrators'] = Calibrator.objects.all() 
    # And so on for more models 
    return context 
0

内のすべてのフィールド上の

関連する問題