พารามิเตอร์ URL และตรรกะในมุมมองตามคลาส Django (TemplateView)


97

ฉันไม่ชัดเจนว่าวิธีที่ดีที่สุดในการเข้าถึงพารามิเตอร์ URL ในมุมมองตามคลาสใน Django 1.5

พิจารณาสิ่งต่อไปนี้:

ดู:

from django.views.generic.base import TemplateView


class Yearly(TemplateView):
    template_name = "calendars/yearly.html"

    current_year = datetime.datetime.now().year
    current_month = datetime.datetime.now().month

    def get_context_data(self, **kwargs):
        context = super(Yearly, self).get_context_data(**kwargs)
        context['current_year'] = self.current_year
        context['current_month'] = self.current_month
        return context

URLCONF:

from .views import Yearly


urlpatterns = patterns('',
    url(
        regex=r'^(?P<year>\d+)/$',
        view=Yearly.as_view(),
        name='yearly-view'
    ),
)

ฉันต้องการเข้าถึงyearพารามิเตอร์ในมุมมองของฉันดังนั้นฉันจึงสามารถใช้ตรรกะเช่น:

month_names = [
    "January", "February", "March", "April", 
    "May", "June", "July", "August", 
    "September", "October", "November", "December"
]

for month, month_name in enumerate(month_names, start=1):
    is_current = False
    if year == current_year and month == current_month:
        is_current = True
        months.append({
            'month': month,
            'name': month_name,
            'is_current': is_current
        })

วิธีที่ดีที่สุดในการเข้าถึงพารามิเตอร์ url ใน CBV เช่นด้านบนที่เป็นคลาสย่อยTemplateViewและตำแหน่งใดที่ควรวางตรรกะเช่นนี้เช่น ในวิธีการ?


มีตัวเลือกของที่เรียบง่ายเป็นextra_contextDict ในdjango2ดูที่นี่
Timo

คำตอบ:


117

ในการเข้าถึงพารามิเตอร์ url ในมุมมองตามคลาสให้ใช้self.argsหรือself.kwargsมากกว่านั้นคุณสามารถเข้าถึงได้โดยทำself.kwargs['year']


1
เข้าใจถูกต้องหรือไม่ว่าฉันไม่ควรสร้างตัวแปรโดยตรงในมุมมองเหมือนที่ฉันมีด้านบน (บางอย่างเกี่ยวกับพวกเขาที่คงอยู่) นอกจากนี้ฉันไม่เข้าใจว่าฉันควรวางตรรกะไว้ตรงไหนเช่นข้างบน ด้วยวิธีใด นอกจากนี้เมื่อฉันทำในมุมมองที่ฉันได้รับyear = self.kwargs['year'] NameError: self not defined

2
ในทางเทคนิคคุณไม่ควรเนื่องจากพวกเขาอยู่ในระดับชั้นเรียนและเป็นตัวแปรของคลาส ส่วนNameErrorคุณจะพยายามทำyear = self.kwargs['year']ที่ไหน? คุณควรทำด้วยวิธีการที่คุณไม่สามารถทำได้ในระดับชั้นเรียน ตัวอย่างเช่นคุณกำลังใช้TemplateViewซึ่งหมายความว่าคุณจะใช้ตรรกะในการget_context_dataแทนที่ของคุณ
Ngenator

4
สำหรับการอ้างอิง: เอกสารเกี่ยวกับ self.request, self.args และอื่น ๆ สามารถพบได้ในdocs.djangoproject.com/th/1.10/topics/class-based-views/…
LShi

นอกจากนี้คุณสามารถทำได้ในdef __init__(self):ฟังก์ชันในคลาสหากคุณต้องการเข้าถึงนอกฟังก์ชันอื่น ๆ
Rahat Zaman

61

ในกรณีที่คุณส่งพารามิเตอร์ URL เช่นนี้:

http://<my_url>/?order_by=created

คุณสามารถเข้าถึงได้ในมุมมองตามคลาสโดยใช้self.request.GET(ไม่ได้นำเสนอในself.argsหรือในself.kwargs):

class MyClassBasedView(ObjectList):
    ...
    def get_queryset(self):
        order_by = self.request.GET.get('order_by') or '-created'
        qs = super(MyClassBasedView, self).get_queryset()
        return qs.order_by(order_by)

5
ขอบคุณ! สิ่งนี้ทำให้ฉันสับสน ... ฉันอ่านสิ่งที่บ่งบอกถึงพารามิเตอร์ HTTP จะอยู่ใน kwargs
foobarbecue

คุณสามารถแสดง get_queryset () ของ superclass ของ MyClassBasedView ได้หรือไม่ ฉันจะทำ qs=<Object>.objects.<method>
Timo

24

ฉันพบโซลูชันที่ยอดเยี่ยมนี้และสำหรับ django 1.5 หรือสูงกว่าตามที่ระบุไว้ที่นี่ :

ขณะนี้มุมมองตามคลาสทั่วไปของ Django รวมตัวแปรมุมมองในบริบทโดยอัตโนมัติ ตัวแปรนี้ชี้ไปที่วัตถุมุมมองของคุณ

ใน views.py ของคุณ:

from django.views.generic.base import TemplateView    

class Yearly(TemplateView):
    template_name = "calendars/yearly.html"
    # Not here 
    current_year = datetime.datetime.now().year
    current_month = datetime.datetime.now().month

    # dispatch is called when the class instance loads
    def dispatch(self, request, *args, **kwargs):
        self.year = kwargs.get('year', "any_default")

    # other code

    # needed to have an HttpResponse
    return super(Yearly, self).dispatch(request, *args, **kwargs)

พบโซลูชันการจัดส่งในคำถามนี้
เนื่องจากมุมมองถูกส่งผ่านไปแล้วภายในบริบทเทมเพลตคุณจึงไม่จำเป็นต้องกังวลเกี่ยวกับเรื่องนี้ ในไฟล์เทมเพลตของคุณ yearly.html คุณสามารถเข้าถึงแอตทริบิวต์มุมมองเหล่านั้นได้ง่ายๆโดย:

{{ view.year }}
{{ view.current_year }}
{{ view.current_month }}

คุณสามารถเก็บ urlconfไว้ได้เหมือนเดิม

มันมูลค่าการกล่าวขวัญว่าได้รับข้อมูลในบริบทของเทมเพลตเขียนทับ get_context_data () ดังนั้นจึงเป็นอย่างใดที่จะหมด Django ของถั่วกระทำการไหล


8

จนถึงตอนนี้ฉันสามารถเข้าถึงพารามิเตอร์ url เหล่านี้ได้จากภายในเมธอด get_queryset เท่านั้นแม้ว่าฉันจะลองใช้ ListView เท่านั้นไม่ใช่ TemplateView ฉันจะใช้พารามิเตอร์ url เพื่อสร้างแอตทริบิวต์บนอินสแตนซ์วัตถุจากนั้นใช้แอตทริบิวต์นั้นใน get_context_data เพื่อเติมข้อมูลบริบท:

class Yearly(TemplateView):
    template_name = "calendars/yearly.html"

    current_year = datetime.datetime.now().year
    current_month = datetime.datetime.now().month

    def get_queryset(self):
        self.year = self.kwargs['year']
        queryset = super(Yearly, self).get_queryset()
        return queryset

    def get_context_data(self, **kwargs):
        context = super(Yearly, self).get_context_data(**kwargs)
        context['current_year'] = self.current_year
        context['current_month'] = self.current_month
        context['year'] = self.year
        return context

ฉันพบว่าแปลกมีข้อผิดพลาดหรืออะไรบางอย่างเมื่อคุณพยายามทำcontext['year'] = self.kwargs['year']? ควรเข้าถึงได้ทุกที่ในชั้นเรียน
Ngenator

@Ngenator: ฉันเพิ่งตั้งค่าโครงการ django ที่สะอาดเพื่อตรวจสอบอีกครั้งและปรากฎว่าคุณถูกต้อง ฉันไม่แน่ใจว่ามีอะไรป้องกันสิ่งนี้ในรหัสเดิมของฉัน แต่ฉันจะหาคำตอบ :) ขอบคุณสำหรับการ
แจ้งเตือน

7

ลองใช้มัณฑนากร Python เพื่อให้เข้าใจสิ่งนี้:

class Yearly(TemplateView):

    @property
    def year(self):
       return self.kwargs['year']

ฉันชอบอันนี้ คุณสมบัตินี้สามารถใช้ซ้ำได้
cezar
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.