基于Django类的视图(TemplateView)中的URL参数和逻辑


94

我不清楚在Django 1.5中如何最好地访问基于类的视图中的URL参数。

考虑以下:

视图:

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

例如,如何最好地访问CBV中被子类化的url参数,TemplateView理想情况下应将逻辑放置在哪里?在某种方法上?


有简单的选项extra_context中字典django2,看这里
蒂莫

Answers:


113

要在基于类的视图中访问url参数,请使用self.args或,self.kwargs这样您就可以通过self.kwargs['year']


1
是否正确理解我不应该像上面一样在视图中直接创建变量?(关于它们的持久性)。另外,我也不明白应该将上述逻辑放置在何处。用哪种方法?另外,当我year = self.kwargs['year']在视图中执行操作时,也会得到提示NameError: self not defined

2
从技术上讲,您不应该这样做,因为它们是在类级别并且是类变量。至于NameError,你想在哪里做year = self.kwargs['year']?您应该在方法中执行此操作,而不能在类级别执行。因此,例如,您使用a TemplateView表示您将执行get_context_data覆盖中的逻辑。
Ngenator

4
仅供参考:关于self.request,self.args等的文档可以在docs.djangoproject.com/en/1.10/topics/class-based-views/…中
LShi

def __init__(self):如果您想在其他函数之外访问它,也可以在该类的函数中进行操作。
Rahat Zaman

60

如果您传递这样的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)

4
谢谢!这一直让我感到困惑...我一直在读一些东西,暗示HTTP参数将处在困境之中。
foob​​arbecue 2014年

您可以显示MyClassBasedView的超类的get_queryset()吗?我会做的 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)

在这个问题中找到了调度解决方案。
由于视图已经在Template上下文中传递,因此您实际上不必担心它。在模板文件Annual.html中,可以通过以下方式简单地访问这些视图属性:

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

您可以保持urlconf不变

值得一提的是,在模板上下文中获取信息会覆盖get_context_data(),因此某种程度上破坏了django的动作bean流。


8

到目前为止,尽管我只使用ListView而不是TemplateView进行尝试,但我只能从get_queryset方法中访问这些url参数。我将使用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 2013年

@Ngenator:我刚刚建立了一个干净的django项目进行仔细检查,事实证明您是正确的。我不确定是什么在我的原始代码中阻止了此操作,但是我会发现:)。感谢您的单挑
hellsgate 2013年

7

仅仅使用Python装饰器使它变得可理解的怎么样:

class Yearly(TemplateView):

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

我喜欢这一个。该属性是可重用的。
cezar
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.