How to implement breadcrumbs in a Django template?

前端 未结 12 2058
無奈伤痛
無奈伤痛 2020-12-07 09:45

Some solutions provided on doing a Google search for \"Django breadcrumbs\" include using templates and block.super, basically just extending the base blocks and adding the

相关标签:
12条回答
  • 2020-12-07 10:16

    I've created template filter for this.

    Apply your custom filter (I've named it 'makebreadcrumbs') to the request.path like this:

    {% with request.resolver_match.namespace as name_space %}
        {{ request.path|makebreadcrumbs:name_space|safe }}
    {% endwith %}
    

    We need to pass url namespace as an arg to our filter.

    Also use safe filter, because our filter will be returning string that needs to be resolved as html content.

    Custom filter should look like this:

    @register.filter
    def makebreadcrumbs(value, arg):
        my_crumbs = []
        crumbs = value.split('/')[1:-1]  # slice domain and last empty value
        for index, c in enumerate(crumbs):
            if c == arg and len(crumbs) != 1:  
            # check it is a index of the app. example: /users/user/change_password - /users/ is the index.
                link = '<a href="{}">{}</a>'.format(reverse(c+':index'), c)
            else:
                if index == len(crumbs)-1:
                    link = '<span>{}</span>'.format(c)  
                    # the current bread crumb should not be a link.
                else:
                    link = '<a href="{}">{}</a>'.format(reverse(arg+':' + c), c)
            my_crumbs.append(link)
        return ' &gt; '.join(my_crumbs)  
        # return whole list of crumbs joined by the right arrow special character.
    

    Important:

    splited parts of the 'value' in our filter should be equal to the namespace in the urls.py, so the reverse method can be called.

    Hope it helped.

    0 讨论(0)
  • 2020-12-07 10:18

    The Django admin view module have automatic breadcumbs, which are implemented like this:

    {% block breadcrumbs %}
        <div class="breadcrumbs">
            <a href="{% url 'admin:index' %}">{% trans 'Home' %}</a>
            {% block crumbs %}
                {% if title %} &rsaquo; {{ title }}{% endif %}
            {% endblock %}
        </div>
    {% endblock %}
    

    So there is some kind of built-in support for this..

    0 讨论(0)
  • 2020-12-07 10:18

    Something like this may work for your situation:

    Capture the entire URL in your view and make links from it. This will require modifying your urls.py, each view that needs to have breadcrumbs, and your templates.

    First you would capture the entire URL in your urls.py file

    original urls.py
    ...
    (r'^myapp/$', 'myView'),
    (r'^myapp/(?P<pk>.+)/$', 'myOtherView'),
    ...
    
    new urls.py
    ...
    (r'^(?P<whole_url>myapp/)$', 'myView'),
    (r'^(?P<whole_url>myapp/(?P<pk>.+)/)$', 'myOtherView'),
    ...
    

    Then in your view something like:

    views.py
    ...
    def myView(request, whole_url):
        # dissect the url
        slugs = whole_url.split('/')
        # for each 'directory' in the url create a piece of bread
        breadcrumbs = []
        url = '/'
        for slug in slugs:
            if slug != '':
                url = '%s%s/' % (url, slug)
                breadcrumb = { 'slug':slug, 'url':url }
                breadcrumbs.append(breadcrumb)
        
        objects = {
            'breadcrumbs': breadcrumbs,
        }
        return render_to_response('myTemplate.html', objects)
    ...
    

    Which should be pulled out into a function that gets imported into the views that need it

    Then in your template print out the breadcrumbs

    myTemplate.html
    ...
    <div class="breadcrumb-nav">
        <ul>
        {% for breadcrumb in breadcrumbs %}
            <li><a href="{{ breadcrumb.url }}">{{ breadcrumb.slug }}</a></li>
        {% endfor %}
        </ul>
    </div>
    ...
    

    One shortcoming of doing it this way is that as it stands you can only show the 'directory' part of the url as the link text. One fix for this off the top of my head (probably not a good one) would be to keep a dictionary in the file that defines the breadcrumb function.

    Anyways that's one way you could accomplish breadcrumbs, cheers :)

    0 讨论(0)
  • 2020-12-07 10:25

    Note: I provide the full snippet below, since djangosnippets has been finicky lately.

    Cool, someone actually found my snippet :-) The use of my template tag is rather simple.

    To answer your question there is no "built-in" django mechanism for dealing with breadcrumbs, but it does provide us with the next best thing: custom template tags.

    Imagine you want to have breadcrumbs like so:

    Services -> Programming
    Services -> Consulting
    

    Then you will probably have a few named urls: "services", and "programming", "consulting":

        (r'^services/$',
         'core.views.services',
         {},
         'services'),
    
        (r'^services/programming$',
         'core.views.programming',
         {},
         'programming'),
    
        (r'^services/consulting$',
         'core.views.consulting',
         {},
         'consulting'),
    

    Now inside your html template (lets just look at consulting page) all you have to put is:

    //consulting.html
    {% load breadcrumbs %}
    
    {% block breadcrumbs %}
    {% breadcrumb_url 'Services' services %}
    {% breadcrumb_url 'Consulting' consulting %}
    
    {% endblock %}
    

    If you want to use some kind of custom text within the breadcrumb, and don't want to link it, you can use breadcrumb tag instead.

    //consulting.html
    {% load breadcrumbs %}
    
    {% block breadcrumbs %}
      {% breadcrumb_url 'Services' services %}
      {% breadcrumb_url 'Consulting' consulting %}
      {% breadcrumb 'We are great!' %}  
    {% endblock %}
    

    There are more involved situations where you might want to include an id of a particular object, which is also easy to do. This is an example that is more realistic:

    {% load breadcrumbs %}
    
    {% block breadcrumbs %}
    {% breadcrumb_url 'Employees' employee_list %}
    {% if employee.id %}
        {% breadcrumb_url employee.company.name company_detail employee.company.id %}
        {% breadcrumb_url employee.full_name employee_detail employee.id %}
        {% breadcrumb 'Edit Employee ' %}
    {% else %}
        {% breadcrumb 'New Employee' %}
    {% endif %}
    
    {% endblock %}
    

    DaGood breadcrumbs snippet

    Provides two template tags to use in your HTML templates: breadcrumb and breadcrumb_url. The first allows creating of simple url, with the text portion and url portion. Or only unlinked text (as the last item in breadcrumb trail for example). The second, can actually take the named url with arguments! Additionally it takes a title as the first argument.

    This is a templatetag file that should go into your /templatetags directory.

    Just change the path of the image in the method create_crumb and you are good to go!

    Don't forget to {% load breadcrumbs %} at the top of your html template!

    from django import template
    from django.template import loader, Node, Variable
    from django.utils.encoding import smart_str, smart_unicode
    from django.template.defaulttags import url
    from django.template import VariableDoesNotExist
    
    register = template.Library()
    
    @register.tag
    def breadcrumb(parser, token):
        """
        Renders the breadcrumb.
        Examples:
            {% breadcrumb "Title of breadcrumb" url_var %}
            {% breadcrumb context_var  url_var %}
            {% breadcrumb "Just the title" %}
            {% breadcrumb just_context_var %}
    
        Parameters:
        -First parameter is the title of the crumb,
        -Second (optional) parameter is the url variable to link to, produced by url tag, i.e.:
            {% url person_detail object.id as person_url %}
            then:
            {% breadcrumb person.name person_url %}
    
        @author Andriy Drozdyuk
        """
        return BreadcrumbNode(token.split_contents()[1:])
    
    
    @register.tag
    def breadcrumb_url(parser, token):
        """
        Same as breadcrumb
        but instead of url context variable takes in all the
        arguments URL tag takes.
            {% breadcrumb "Title of breadcrumb" person_detail person.id %}
            {% breadcrumb person.name person_detail person.id %}
        """
    
        bits = token.split_contents()
        if len(bits)==2:
            return breadcrumb(parser, token)
    
        # Extract our extra title parameter
        title = bits.pop(1)
        token.contents = ' '.join(bits)
    
        url_node = url(parser, token)
    
        return UrlBreadcrumbNode(title, url_node)
    
    
    class BreadcrumbNode(Node):
        def __init__(self, vars):
            """
            First var is title, second var is url context variable
            """
            self.vars = map(Variable,vars)
    
        def render(self, context):
            title = self.vars[0].var
    
            if title.find("'")==-1 and title.find('"')==-1:
                try:
                    val = self.vars[0]
                    title = val.resolve(context)
                except:
                    title = ''
    
            else:
                title=title.strip("'").strip('"')
                title=smart_unicode(title)
    
            url = None
    
            if len(self.vars)>1:
                val = self.vars[1]
                try:
                    url = val.resolve(context)
                except VariableDoesNotExist:
                    print 'URL does not exist', val
                    url = None
    
            return create_crumb(title, url)
    
    
    class UrlBreadcrumbNode(Node):
        def __init__(self, title, url_node):
            self.title = Variable(title)
            self.url_node = url_node
    
        def render(self, context):
            title = self.title.var
    
            if title.find("'")==-1 and title.find('"')==-1:
                try:
                    val = self.title
                    title = val.resolve(context)
                except:
                    title = ''
            else:
                title=title.strip("'").strip('"')
                title=smart_unicode(title)
    
            url = self.url_node.render(context)
            return create_crumb(title, url)
    
    
    def create_crumb(title, url=None):
        """
        Helper function
        """
        crumb = """<span class="breadcrumbs-arrow">""" \
                """<img src="/media/images/arrow.gif" alt="Arrow">""" \
                """</span>"""
        if url:
            crumb = "%s<a href='%s'>%s</a>" % (crumb, url, title)
        else:
            crumb = "%s&nbsp;&nbsp;%s" % (crumb, title)
    
        return crumb
    
    0 讨论(0)
  • 2020-12-07 10:29

    My view functions emit the breadcrumbs as a simple list.

    Some information is kept in the user's session. Indirectly, however, it comes from the URL's.

    Breadcrumbs are not a simple linear list of where they've been -- that's what browser history is for. A simple list of where they've been doesn't make a good breadcrumb trail because it doesn't reflect any meaning.

    For most of our view functions, the navigation is pretty fixed, and based on template/view/URL design. In our cases, there's a lot of drilling into details, and the breadcrumbs reflect that narrowing -- we have a "realm", a "list", a "parent" and a "child". They form a simple hierarchy from general to specific.

    In most cases, a well-defined URL can be trivially broken into a nice trail of breadcrumbs. Indeed, that's one test for good URL design -- the URL can be interpreted as breadcrumbs and displayed meaningfully to the users.

    For a few view functions, where we present information that's part of a "many-to-many" join, for example, there are two candidate parents. The URL may say one thing, but the session's context stack says another.

    For that reason, our view functions have to leave context clues in the session so we can emit breadcrumbs.

    0 讨论(0)
  • 2020-12-07 10:29

    You could also reduce the boiler plate required to manage breadcrumbs using django-view-breadcrumbs, by adding a crumbs property to the view.

    urls.py

    urlpatterns = [
        ...
        path('posts/<slug:slug>', views.PostDetail.as_view(), name='post_detail'),
        ...
    ] 
    

    views.py

    from django.views.generic import DetailView
    from view_breadcrumbs import DetailBreadcrumbMixin
    
    
    class PostDetail(DetailBreadcrumbMixin, DetailView):
        model = Post
        template_name = 'app/post/detail.html'
    

    base.html

    {% load django_bootstrap_breadcrumbs %}
    
    {% block breadcrumbs %}
        {% render_breadcrumbs %}
    {% endblock %}
    
    0 讨论(0)
提交回复
热议问题