Django sınıfı tabanlı görünümlerde (TemplateView) URL parametreleri ve mantık


98

Django 1.5'te sınıf tabanlı görünümlerde URL parametrelerine erişmenin en iyi nasıl olduğu bana açık değil.

Aşağıdakileri göz önünde bulundur:

Görünüm:

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'
    ),
)

yearGörünümümdeki parametreye erişmek istiyorum , böylece aşağıdaki gibi mantık yapabilirim:

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
        })

Yukarıdakine benzer CBV'lerde url parametresine en iyi nasıl erişilir ve bunun gibi TemplateViewmantığı ideal olarak nereye yerleştirir, ör. bir yöntemde?


Basit extra_contextdikte seçeneği var, burayadjango2 bakın
Timo

Yanıtlar:


117

Url parametrelerine sınıf tabanlı görünümlerde erişmek için, self.argsveya self.kwargsbu şekilde kullanın.self.kwargs['year']


1
Yukarıdaki gibi doğrudan görünümde değişkenler yaratmamam gerektiği doğru anlaşıldı mı? (ısrarcı olmaları hakkında bir şey). Ayrıca yukarıdaki gibi mantığı nereye yerleştirmem gerektiğini anlamıyorum, örn. hangi yöntemle? Ayrıca year = self.kwargs['year']görünümde yaptığımda elde ederim NameError: self not defined.

2
Teknik olarak, sınıf düzeyinde oldukları ve sınıf değişkenleri oldukları için yapmamalısınız. Gelince NameError, nerede yapmaya çalışıyorsun year = self.kwargs['year']? Bunu bir yöntemle yapmalısın, sınıf seviyesinde yapamazsın. Örneğin, TemplateViewmantığınızı get_context_datageçersiz kılmanızda yapacağınız anlamına gelen a kullanıyorsunuz .
Yarışmacı


Ayrıca, def __init__(self):diğer işlevler dışında ona erişmek istiyorsanız, sınıfta işlevde de yapabilirsiniz.
Rahat Zaman

62

URL parametresini şu şekilde geçirmeniz durumunda:

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

Aşağıdakileri kullanarak self.request.GET(içinde self.argsveya içinde sunulmaz self.kwargs) sınıf tabanlı görünümde erişebilirsiniz :

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
Teşekkürler! Bu kafamı karıştırdı ... HTTP parametrelerinin kwargs içinde olacağını ima eden şeyleri okumaya devam ediyorum.
foobarbecue

MyClassBasedView üst sınıfının get_queryset () değerini gösterebilir misiniz? Sadece yapardım qs=<Object>.objects.<method>
Timo

24

Bu zarif çözümü buldum ve burada belirtildiği gibi django 1.5 veya üstü için :

Django'nun genel sınıf tabanlı görünümleri artık bağlamda otomatik olarak bir görünüm değişkeni içeriyor. Bu değişken, görünüm nesnenizi işaret eder.

Views.py dosyanızda:

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)

Bu soruda bulunan sevk çözümü .
Gibi görünüm zaten Şablon kapsamında geçirilir, gerçekten bu konuda endişe gerekmez. Yearly.html şablon dosyanızda, bu görünüm özniteliklerine basitçe şu şekilde erişmek mümkündür:

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

Sen edebilirsiniz senin urlconf tutmak olduğu gibi.

Şablonunuzun bağlamına bilgi girmenin get_context_data () öğesinin üzerine yazdığından bahsetmeye değer, bu nedenle bu, django'nun aksiyon çekirdeği akışını bir şekilde bozuyor .


8

Şimdiye kadar bu url parametrelerine sadece get_queryset yönteminden erişebildim, ancak bunu yalnızca bir TemplateView değil ListView ile denedim. Nesne örneğinde bir öznitelik oluşturmak için url parametresini kullanacağım, sonra bu özniteliği get_context_data'da bağlamı doldurmak için kullanacağım:

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

Bunu garip buluyorum, yapmaya çalıştığınızda bir hata veya bir şey context['year'] = self.kwargs['year']mi var ? Sınıfın herhangi bir yerinde erişilebilir olmalıdır.
Yarışmacı

@Ngenator: İki kez kontrol etmek için temiz bir django projesi kurdum ve doğru olduğunuz ortaya çıktı. Orijinal kodumda bunu neyin engellediğinden emin değilim ama öğreneceğim :).
Uyarı

7

Bunu anlaşılır kılmak için Python dekoratörlerini kullanmaya ne dersiniz?

class Yearly(TemplateView):

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

Bunu beğendim. Mülkiyet yeniden kullanılabilir.
Cezar
Sitemizi kullandığınızda şunları okuyup anladığınızı kabul etmiş olursunuz: Çerez Politikası ve Gizlilik Politikası.
Licensed under cc by-sa 3.0 with attribution required.