2010-01-29 4 views
62

Mon code myapp_extras.py:demande d'accès dans les balises de modèle django personnalisés

from django import template 

register = template.Library() 

@register.inclusion_tag('new/userinfo.html') 
def address(): 
    address = request.session['address'] 
    return {'address':address} 

dans 'settings.py':

TEMPLATE_CONTEXT_PROCESSORS =(
    "django.core.context_processors.auth", 
    "django.core.context_processors.debug", 
    "django.core.context_processors.i18n", 
    "django.core.context_processors.media", 
    'django.core.context_processors.request' 
) 

mais je suis une erreur:

TemplateSyntaxError at /items/ 

Caught an exception while rendering: global name 'request' is not defined 

Original Traceback (most recent call last): 
    File "C:\Python25\lib\site-packages\django\template\debug.py", line 71, in render_node 
    result = node.render(context) 
    File "C:\Python25\lib\site-packages\django\template\__init__.py", line 915, in render 
    dict = func(*args) 
    File "C:\p4\projects\myproject\..\myproject\invoice\templatetags\myapp_extras.py", line 9, in address 
    address = request.session['address'] 
NameError: global name 'request' is not defined 

J'ai référencé celui-ci In Django, is it possible to access the current user session from within a custom tag?.

+1

Hmmm ... Les réponses à cette question liée semblent erronées ... –

+0

si vous voulez accéder à la demande cochez cette réponse http://stackoverflow.com/a/2567234/1153703 –

Répondre

140

request n'est pas une variable dans cette portée. Vous devrez l'obtenir du contexte d'abord. Pass takes_context to the decorator and add context to the tag arguments.

Comme ceci:

@register.inclusion_tag('new/userinfo.html', takes_context=True) 
def address(context): 
    request = context['request'] 
    address = request.session['address'] 
    return {'address':address} 
+0

il semble que j'ai encore des erreurs TemplateSyntaxError à/ Caught une exception tout en rendant: 'demande' originale retraçage (appel le plus récent en dernier): fichier "C: \ Python25 \ lib \ site-packages \ django \ template \ debug.py", ligne 71 , dans render_node result = node.render (context) Fichier "C: \ Python25 \ lib \ paquets-de-site \ django \ template \ __ init__.py", ligne 936, dans le rendu dict = func (* args) Fichier "c: \ ... \ myapp_extras.py", ligne 7, in login request = context ['request'] Fichier "C: \ Python25 \ lib \ paquets-de-site \ django \ template \ context.py", ligne 44, dans __getitem__ raise KeyError (clé) KeyError: « demande » – icn

+5

i remarqué dans cette page http://docs.djangoproject.com/en/dev/ref/templates/api/ « DJANGO.CORE.CONTEXT_PROCESSORS.REQUEST Si TEMPLATE_CONTEXT_PROCESSORS contient ce processeur, chaque RequestContext contiendra une requête variable, qui est l'actuelle HttpRequest. Notez que ce processeur n'est pas activé par défaut. vous devrez l'activer. " Quelque part je dois configurer pour activer le processeur de requêtes – icn

+1

Pourquoi est-ce le cas avec inclusion_tag? – aehlke

11

J'ai essayé solution d'en haut (de Ignacio Vazquez-Abrams) et il ne fait pas travailler jusqu'à ce que je l'ai découvert que les processeurs de contexte fonctionne uniquement avec RequestContext classe wrapper .

donc dans la méthode de la vue principale, vous devez ajouter la ligne suivante:

from django.template import RequestContext   
return render_to_response('index.html', {'form': form, }, 
           context_instance = RequestContext(request)) 
+0

En effet, context_instance = RequestContext (request) est une pièce manquante. Utilise le! –

+3

Si vous affichez la réponse à string/html, vous pouvez faire sans RequestContext. '@ register.simple_tag (takes_context = True) ', puis 'def my_tag (contexte, ...) ... return render_to_response ('index.html', {'form': formulaire,}, context_instance = contexte) '. – osa

7

Je l'ai fait de cette façon:

from django import template 
register = template.Library() 

def do_test_request(parser,token): 
    try: 
     tag_name = token.split_contents() # Not really useful 
    except ValueError: 
     raise template.TemplateSyntaxError("%r error" % token.contents.split()[0]) 
    return RequestTestNode() 

class RequestTestNode(template.Node): 
    def __init__(self,): 
     self.request = template.Variable('request') 
    def render(self, context): 
     rqst = self.request.resolve(context) 
     return "The URL is: %s" % rqst.get_full_path() 

register.tag('test_request', do_test_request) 

Il y a aussi une fonction appelée resolve_variable, mais il est dépréciée.

J'espère que ça aide!

Questions connexes