我在settings.py中有一些东西,我希望能够从模板访问,但我无法弄清楚如何做到这一点.我已经试过了
{{CONSTANT_NAME}}
但这似乎不起作用.这可能吗?
如果它是您希望为每个请求和模板提供的值,则使用上下文处理器更合适.
这是如何做:
context_processors.py
在您的app目录中创建一个文件.假设我想ADMIN_PREFIX_VALUE
在每个上下文中都有价值:
from django.conf import settings # import the settings file def admin_media(request): # return the value you want as a dictionnary. you may add multiple values in there. return {'ADMIN_MEDIA_URL': settings.ADMIN_MEDIA_PREFIX}
将您的上下文处理器添加到settings.py文件中:
TEMPLATES = [{ # whatever comes before 'OPTIONS': { 'context_processors': [ # whatever comes before "your_app.context_processors.admin_media", ], } }]
RequestContext
在视图中使用您的模板中添加上下文处理器.该render
快捷方式自动执行此操作:
from django.shortcuts import render def my_view(request): return render(request, "index.html")
最后,在你的模板中:
... path to admin media ...
我发现最简单的方法是单个模板标记:
from django import template from django.conf import settings register = template.Library() # settings value @register.simple_tag def settings_value(name): return getattr(settings, name, "")
用法:
{% settings_value "LANGUAGE_CODE" %}
settings.MEDIA_URL
如果您使用django的内置通用视图或在render_to_response
快捷方式函数中传入上下文实例关键字参数,Django可以访问模板的某些常用设置常量,例如某些语言设置.以下是每个案例的示例:
from django.shortcuts import render_to_response from django.template import RequestContext from django.views.generic.simple import direct_to_template def my_generic_view(request, template='my_template.html'): return direct_to_template(request, template) def more_custom_view(request, template='my_template.html'): return render_to_response(template, {}, context_instance=RequestContext(request))
这些视图都有几个常用的设置,如settings.MEDIA_URL
模板可用{{ MEDIA_URL }}
,等等.
如果您正在寻找设置中其他常量的访问权限,那么只需解压缩您想要的常量并将它们添加到您在视图函数中使用的上下文字典中,如下所示:
from django.conf import settings from django.shortcuts import render_to_response def my_view_function(request, template='my_template.html'): context = {'favorite_color': settings.FAVORITE_COLOR} return render_to_response(template, context)
现在,您可以settings.FAVORITE_COLOR
在模板上访问{{ favorite_color }}
.
退房django-settings-export
(免责声明:我是这个项目的作者).
例如...
$ pip install django-settings-export
TEMPLATES = [ { 'OPTIONS': { 'context_processors': [ 'django_settings_export.settings_export', ], }, }, ] MY_CHEESE = 'Camembert'; SETTINGS_EXPORT = [ 'MY_CHEESE', ]
另一种方法是创建一个自定义模板标记,它可以让您从设置中捕获值.
@register.tag def value_from_settings(parser, token): try: # split_contents() knows not to split quoted strings. tag_name, var = token.split_contents() except ValueError: raise template.TemplateSyntaxError, "%r tag requires a single argument" % token.contents.split()[0] return ValueFromSettings(var) class ValueFromSettings(template.Node): def __init__(self, var): self.arg = template.Variable(var) def render(self, context): return settings.__getattr__(str(self.arg))
然后你可以使用:
{% value_from_settings "FQDN" %}
在任何页面上打印它,而不跳过上下文处理器箍.
我喜欢Berislav的解决方案,因为在简单的网站上,它是干净而有效的.我不喜欢的是无所不在地暴露所有设置常量.所以我最终做的是这样的:
from django import template from django.conf import settings register = template.Library() ALLOWABLE_VALUES = ("CONSTANT_NAME_1", "CONSTANT_NAME_2",) # settings value @register.simple_tag def settings_value(name): if name in ALLOWABLE_VALUES: return getattr(settings, name, '') return ''
用法:
{% settings_value "CONSTANT_NAME_1" %}
这可以保护您未在模板中使用的任何常量,如果您想真正了解,可以在设置中设置元组,并为不同的页面,应用或区域创建多个模板标记,并简单地根据需要将本地元组与设置元组组合,然后执行列表推导以查看该值是否可接受.
我同意,在一个复杂的网站上,这有点过于简单,但有一些值很高兴在模板中普遍使用,这看起来效果很好.感谢Berislav最初的想法!
我改进了chrisdew的答案(创建你自己的标签).
首先,创建yourapp/templatetags/value_from_settings.py
用于定义自己的新标记的文件value_from_settings
:
from django.template import TemplateSyntaxError, Variable, Node, Variable, Library from yourapp import settings register = Library() # I found some tricks in URLNode and url from defaulttags.py: # https://code.djangoproject.com/browser/django/trunk/django/template/defaulttags.py @register.tag def value_from_settings(parser, token): bits = token.split_contents() if len(bits) < 2: raise TemplateSyntaxError("'%s' takes at least one " \ "argument (settings constant to retrieve)" % bits[0]) settingsvar = bits[1] settingsvar = settingsvar[1:-1] if settingsvar[0] == '"' else settingsvar asvar = None bits = bits[2:] if len(bits) >= 2 and bits[-2] == 'as': asvar = bits[-1] bits = bits[:-2] if len(bits): raise TemplateSyntaxError("'value_from_settings' didn't recognise " \ "the arguments '%s'" % ", ".join(bits)) return ValueFromSettings(settingsvar, asvar) class ValueFromSettings(Node): def __init__(self, settingsvar, asvar): self.arg = Variable(settingsvar) self.asvar = asvar def render(self, context): ret_val = getattr(settings,str(self.arg)) if self.asvar: context[self.asvar] = ret_val return '' else: return ret_val
您可以通过以下方式在模板中使用此标记:
{% load value_from_settings %} [...] {% value_from_settings "FQDN" %}
或通过
{% load value_from_settings %} [...] {% value_from_settings "FQDN" as my_fqdn %}
这种as ...
符号的优点在于,blocktrans
通过简单的方法可以很容易地在块中使用{{my_fqdn}}
.
使用Django 2.0+,添加包含创建可解决此问题的自定义模板标签的完整说明的答案
在您的应用程序文件夹中,创建一个名为templatetags的文件夹。在其中创建__init__.py和custom_tags.py:
在custom_tags.py中,创建一个自定义标签函数,该函数可访问设置常量中的任意键:
from django import template from django.conf import settings register = template.Library() @register.simple_tag def get_setting(name): return getattr(settings, name, "")
为了理解此代码,我建议阅读Django文档中有关简单标签的部分。
然后,您需要通过在要使用它的任何模板中加载此文件来使Django知道此(以及任何其他)自定义标签。就像您需要加载内置的静态标签一样:
{% load custom_tags %}
加载后,它可以像其他标签一样使用,只需提供您需要返回的特定设置即可。因此,如果您的设置中包含BUILD_VERSION变量:
{% get_setting "BUILD_VERSION" %}
此解决方案不适用于数组,但是如果需要,可以在模板中增加很多逻辑。
bchhun上面的例子很好,除了你需要从settings.py显式构建你的上下文字典.下面是一个UNTESTED示例,说明如何从settings.py的所有大写属性自动构建上下文字典(re:"^ [A-Z0-9 _] + $").
在settings.py结束时:
_context = {} local_context = locals() for (k,v) in local_context.items(): if re.search('^[A-Z0-9_]+$',k): _context[k] = str(v) def settings_context(context): return _context TEMPLATE_CONTEXT_PROCESSORS = ( ... 'myproject.settings.settings_context', ... )
如果有人像我一样找到这个问题,那么我将发布适用于Django 2.0的解决方案:
此标记为模板变量分配一些settings.py变量值:
用法: {% get_settings_value template_var "SETTINGS_VAR" %}
from django import template from django.conf import settings register = template.Library() class AssignNode(template.Node): def __init__(self, name, value): self.name = name self.value = value def render(self, context): context[self.name] = getattr(settings, self.value.resolve(context, True), "") return '' @register.tag('get_settings_value') def do_assign(parser, token): bits = token.split_contents() if len(bits) != 3: raise template.TemplateSyntaxError("'%s' tag takes two arguments" % bits[0]) value = parser.compile_filter(bits[2]) return AssignNode(bits[1], value)
{% load my_custom_tags %} # Set local template variable: {% get_settings_value settings_debug "DEBUG" %} # Output settings_debug variable: {{ settings_debug }} # Use variable in if statement: {% if settings_debug %} ... do something ... {% else %} ... do other stuff ... {% endif %}
请参阅Django文档,了解如何在此处创建自定义模板标签:https : //docs.djangoproject.com/en/2.0/howto/custom-template-tags/
将此代码添加到名为的文件中context_processors.py
:
from django.conf import settings as django_settings def settings(request): return { 'settings': django_settings, }
然后,在你的设置文件,包括路径,例如'speedy.core.base.context_processors.settings'
(与你的应用程序名称和路径)中'context_processors'
的设置TEMPLATES
。
(例如,您可以查看https://github.com/speedy-net/speedy-net/blob/staging/speedy/core/settings/base.py和https://github.com/speedy-net/speedy- net / blob / staging / speedy / core / base / context_processors.py)。