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

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

ただいまの
回答率

90.51%

  • Python

    11708questions

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

  • Django

    1608questions

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

GETパラメータが更新される度に結果を絞り込んで表示したい

解決済

回答 1

投稿

  • 評価
  • クリップ 0
  • VIEW 648

mickey

score 18

テンプレートに

<ul>
    {% if big_cat %}
        <li><a href="?{% url_replace request 'category' big_cat.pk %}">{{ big_cat.name }}</a></li>
        {% for small_cat in big_cat.children.all %}
            <li>---<a href="?{% url_replace request 'category' small_cat.pk %}">{{ small_cat.name }}</a></li>
        {% endfor %}
    {% else %}
        {% for big_cat in big_cats %}
            <li><a href="?{% url_replace request 'category' big_cat.pk %}">{{ big_cat.name }}</a></li>
        {% endfor %}
    {% endif %}
</ul>
<p>[ 場所 ]</p>
<ul>
    {% if state %}
        <li><a href="?{% url_replace request 'location' state.pk %}">{{ state.name }}</a></li>
        {% if region %}
            <li>---<a href="?{% url_replace request 'location' region.pk %}">{{ region.name }}</a></li>
            {% for city in region.children.all %}
                <li>------<a href="?{% url_replace request 'location' city.pk %}">{{ city.name }}</a></li>
            {% endfor %}
        {% else %}
            {% for region in regions %}
                <li>---<a href="?{% url_replace request 'location' region.pk %}">{{ region.name }}</a></li>
            {% endfor %}
        {% endif %}
    {% else %}
        {% for state in states %}
            <li><a href="?{% url_replace request 'location' state.pk %}">{{ state.name }}</a></li>
        {% endfor %}
    {% endif %}
</ul>


のような絞り込みをするサイドバーがあり、各リンクをクリックすることによって

class PostQuerySet(models.QuerySet):

    def filter_category(self, category):
        if not category.parent:
            return self.filter(category__in=category.children.all())
        else:
            return self.filter(category=category)

    def filter_location(self, location):
        if not location.parent:
            # first
            return self.filter(location__parent__in=location.children.all())
        else:
            if not location.dispatch.parent:
                # second
                return self.filter(location__in=location.children.all())
            else:
                # third
                return self.filter(location=location)

class Post(models.Model):
    category = models.ForeignKey(Category)
    location = models.ForeignKey(Location)
    price = models.DecimalField(max_digits=10, decimal_places=2)
    title = models.CharField(max_length=255)
    text = models.CharField(max_length=255)

    objects = PostQuerySet.as_manager()

    def __str__(self):
        return self.title


のようにcategoryやlocationのForeignKeyをもったPostが絞り込みされて表示されるようにいたいのですが、viewでどのように記述すればいいのでしょうか?

現在

    category_pk = request.GET.get('category')
    location_pk = request.GET.get('location')

    c = {'posts': Post.objects.all()}

    if category_pk:
        category = get_object_or_404(Category, pk=category_pk)
        c.update({'posts': Post.objects.filter_category(category)})

    if location_pk:
        location = get_object_or_404(Location, pk=location_pk)
        c.update({'posts': Post.objects.filter_location(location)})


のように記述しており、もしURLに?category=(カテゴリのpk)?category=(カテゴリのpk)&location=(場所のpk)とあれば、viewにてPostを絞り込んで結果を返すようにしていますが、もし?category=(カテゴリのpk)&location=(場所のpk)というようにGETパラメータが複数存在すると最後のGETパラメータのみ(この場合locationのみ)で絞り込みされます。

GETパラメータが1つでも複数でもその都度Postを絞り込み表示したいです。

一応

Post.objects.filter_category(category).filter_location(location)


とすればGETパラメータがcategoryとlocationの複数あってもそれに応じて絞り込みはできるのですが、この場合

if category_pk and location_pk:
        category = get_object_or_404(Category, pk=category_pk)
        location = get_object_or_404(Location, pk=location_pk)
        posts = Post.objects.filter_category(category).filter_location(location)

    elif category_pk:
        category = get_object_or_404(Category, pk=category_pk)
        posts = Post.objects.filter_category(category)

    elif location_pk:
        location = get_object_or_404(Location, pk=location_pk)
        posts = Post.objects.filter_location(location)

    else:
        posts = Post.objects.all()

    c.update({'posts': posts})


と多岐に渡る条件分岐をしなくてはいけません。また将来的にcategoryやlocationのみならず、他にも絞り込み条件を増やした場合、より複雑な条件分岐を記述しなくてはいけないので他の方法を模索しているのですが僕の今の知識では到底実装できそうにありません。
理想としては

# GETパラメータがないときは
Post.objects.all()
# categoryがあるときは
Post.objects.filter_category(category)
# locationがあるときは
Post.objects.filter_location(location)
# 複数個あるときはその都度自動的にfilterを追加する
Post.objects.filter().filter().filter()......


という具合です。
filter_categoryやfilter_locationといった仕組み自体に問題があればこれも改善したいのですが、絞り込み元のモデルが親をもつ時や親の親を持つ時等、複雑なので他にいいアイデアが思いつきません。
どうかご教授願います。

categoryとlocationのモデルです。

class Category(models.Model):
    parent = models.ForeignKey('Category', related_name='children', null=True, blank=True)
    name = models.CharField(max_length=255)
    slug = models.SlugField()

    @property
    def dispatch(self):
        # parent が nullであれば親カテゴリなので自分自身を返す
        if not self.parent:
            return self
        # そうでなければ親カテゴリ(parent)を返す
        else:
            return self.parent

    def __str__(self):
        return self.name

class Location(models.Model):
    parent = models.ForeignKey('Location', related_name='children', null=True, blank=True)
    name = models.CharField(max_length=255)

    @property
    def dispatch(self):
        if not self.parent:
            return self
        else:
            return self.parent

    def __str__(self):
        return self.name
  • 気になる質問をクリップする

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

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

    クリップを取り消します

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

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

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

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

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

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

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

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

    質問の評価を下げる

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

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

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

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

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

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

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

    詳細な説明はこちら

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

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

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

回答 1

checkベストアンサー

0

とりあえず思いつきですが、

##GETのキーとフィルタ関数を関連付け
filter_keys={
  "category":PostQuerySet.filter_category,
  "location":PostQuerySet.filter_location,
}


queryset = Post.object.all()
for search_key in request.GET:
  if search_key in filter_keys:
    ##filter_keysにあるものだけフィルタ実行
    filter_method = filter_keys[search_key]
    foreign_key = request.GET[search_key]
    queryset = filter_method(queryset, foreign_key)

filter_keysだけメンテナンスする感じで。

投稿

  • 回答の評価を上げる

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

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

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

  • 回答の評価を下げる

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

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

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

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

  • Python

    11708questions

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

  • Django

    1608questions

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