bernar92
bernar92Dec. 25, 2018, 11:35 p.m.

One possible side menu implementation for Django

Hello. Not long ago there was a task to implement a side menu with a list of directories; so that it can be inserted anywhere and on any page!
The problem was that the list is large and so that it would not load the server, I cached it, but here I will not indicate how I cached it, since it will take a long time to write ...
In general, I will describe everything in order!


Let's start by creating a template tag! in order to implement it! i created a package folder:

you can name it whatever you want, but I named it: context_processors
further created a file: right_menu.py
in which all the logic of the template tag will be implemented.

from products.models import Category
from django.core.cache import cache
from django.db.models.aggregates import Count
from django import template #нужен для регистрации шаблонного тега

register = template.Library() #тут мы присваем значение 
@register.inclusion_tag('includes/right_menu.html') #декоратор для регистрации шаблона для подключения (я его указал в папке templates где находится base.html

def get_right_school_menu(request):
    context = {}
    сategory_menu = cache.get('left_сategory_menu_' + request.LANGUAGE_CODE) #так как у меня мультиязычность стоит я сделал так. Оставлю для примера
    context['categorys'] = Category.objects.all()
    return context


further in settings.py we connect the template tag

TEMPLATES = [
    {
        'BACKEND': 'django.template.backends.django.DjangoTemplates',
        'DIRS': [os.path.join(BASE_DIR, 'templates')],
        'APP_DIRS': True,
        'OPTIONS': {
            'context_processors': [
                'django.template.context_processors.debug',
                'django.template.context_processors.request',
                'django.contrib.auth.context_processors.auth',
                'django.template.context_processors.i18n',
                'django.contrib.messages.context_processors.messages',
                'context_processors.right_menu.get_right_school_menu', # как видете тут я его подключил
            ],
        },
    },
]

in the right_menu.html template itself

{% load static %}
{% load i18n %}
<ul>
    {% for i in categorys %}
    <li>{{ i.name }}</li>
    {% endfor %}
</ul>
We recommend hosting TIMEWEB
We recommend hosting TIMEWEB
Stable hosting, on which the social network EVILEG is located. For projects on Django we recommend VDS hosting.

Do you like it? Share on social networks!

Evgenii Legotckoi
  • Dec. 26, 2018, 4:53 a.m.

Добрый день!

Я добавил разделитель контента в статье, чтобы скрыть под катом основной содержание статьи. Спасибо за материал.

У меня есть вопрос, а почему не использовали просто папку templatetags для этого, например как здесь ?
Вы же здесь по сути написали middleware. Если таких inclusion tag будет очень много, то придётся каждый записывать в context processors.

Впрочем вы ответили на один из моих вопросов, как избавиться от подключения модуля, которым я задавался. Но минус по-моему в том, что потом придётся всё записывать в context_processors ....

bernar92
  • Dec. 26, 2018, 4:59 a.m.

так ты там тоже используешь теги! только симпл упращенный тег! только в твоем случает ты не подгружаешь шаблон! а так суть одна! но в моем случае я просто вывожу через инклуде ) он хорош для менюшек! Я так считаю ))

Evgenii Legotckoi
  • Dec. 26, 2018, 5:15 a.m.
  • (edited)

Согласен, что для менюшек хороши, я много чего на них сделал уже

Кстати, самого примере применения get_right_school_menu нет.

Как именно он выглядит в шаблоне?

Так?

{% get_right_school_menu %}

inclusion tag делается точно так же в templatetags, вот например

@register.inclusion_tag('ecore/breadcrumb_home.html')
def breadcrumb_home(url='/', title=''):
    return {
        'url': url,
        'title': title
    }

применение в шаблоне будет таким

{% breadcrumb_home '/' 'EVILEG' %}

Просто у меня на сайте сейчас используется более 35-ти inclusion_tag, я бы свихнулся каждый прописывать в context_processros. Да и минус такого подхода в том, что каждый запрос обрабатывается всеми middleware вне зависимости, используется он или нет. То есть всё равно будет искаться тот метод, который ты используешь в шаблонах, даже если он не был там задействован. То есть, если тег не используется, то middleware работает вхолостую, тогда как templatetags будет подгружаться тогда, когда он действительно нужен.

bernar92
  • Dec. 27, 2018, 12:16 a.m.
  • (edited)

нет!
будет так:

{% include 'ecore/breadcrumb_home.html' %}

просто инклудишь шаблон где тебе он нужен и все!

Evgenii Legotckoi
  • Dec. 27, 2018, 1:17 a.m.

Понятно, тогда это статья не о применении тега, а о написании context processor.

Видишь ли, тег include работает независимо от того, напишешь ли ты это

register = template.Library() #тут мы присваем значение 
@register.inclusion_tag('includes/right_menu.html') #декоратор для регистрации шаблона для подключения (я его указал в папке templates где находится base.html

или нет

Тег include не смотрит на то, зарегистрирован ли шаблон в качестве имени или нет. И каждый раз при его вызове он ищет в каталогах данный шаблон.
То есть, если ты удалишь эти две строки, то ровным счётом ничего не изменится.

inclusion_tag, как я его написал в этом комментарии , служит именно для кеширования шаблона, чтобы не искать его каждый раз по каталогам, а ты его всё равно ищешь, то есть те две строки написаны впустую.

А context proccessor я понимаю почему там написан, как раз, чтобы генерить все эти категории вне зависимости от страницы, на которой находится пользователь. Но реально, отвечаю, те две строки лишние и работают вхолостую, их можно удалить и ровным счётом ничего не изменится.

bernar92
  • Dec. 27, 2018, 1:36 a.m.

то есть ты предлагаешь рендерить на прямую?

return render(request, 'includes/header.html', context)
Evgenii Legotckoi
  • Dec. 27, 2018, 1:46 a.m.
  • (edited)

нет, я предлагаю сделать так. В каталоге приложения добавляешь каталог templatetags , а внутри него два файла, init .py и файл с названием приложения. Будет выглядеть так

то есть структура app будет такая

  • myapp
    • templatetags
      • __init__.py
      • myapp.py

А далее добавляешь в файл с названием приложения свой inclusion_tag следующим образом

register = template.Library() 
@register.inclusion_tag('includes/right_menu.html', takes_context=True) 
def right_menu(context):
    return context

А потом в шаблоне вызываешь

{% load myapp %}
{% right_menu %}

Твой context processor добавит в контекст всё необходимое, а настройка takes_context=True позволит тегу его захватить.

Преимущество в данном случае будет в том, что шаблон будет реально закеширован и django не будет искать его при каждом рендеринге

bernar92
  • Dec. 27, 2018, 2:03 a.m.

а в settings тогда его нужно убрать! верно я понимаю?!

Evgenii Legotckoi
  • Dec. 27, 2018, 2:09 a.m.

нет, в твоём случае он остается, это штука добавляет в контекст category при каждой обработке запроса.

Впрочем, если переписать так

register = template.Library() 

@register.inclusion_tag('includes/right_menu.html', takes_context=True) 
def right_menu(context):
    context['categorys'] = Category.objects.all()
    return context

То тогда да, можно и удалить из settings, тогда context processor тебе и вовсе не нужен

bernar92
  • Dec. 27, 2018, 2:14 a.m.

is not a registered tag library. Must be one of:

Evgenii Legotckoi
  • Dec. 27, 2018, 2:17 a.m.

какая версия django?

Ну и структура каталогов и файлов, как написал выше, а также перезапустить инстанс джанги, если Debug=False, то автоматически не подхватывается без перезапуска

bernar92
  • Dec. 27, 2018, 3:39 a.m.
  • (edited)

Django 2.1

'schools' is not a registered tag library. Must be one of:
{% load schools %}
{% r_schools %}
bernar92
  • Dec. 27, 2018, 4:37 a.m.

пришлось зарегать его в

'OPTIONS': {
            'context_processors': [
                'django.template.context_processors.debug',
                'django.template.context_processors.request',
                'django.contrib.auth.context_processors.auth',
                'django.template.context_processors.i18n',
                'django.contrib.messages.context_processors.messages',
                'context_processors.right_menu.get_top_string',
                'context_processors.right_menu.right_school'
            ],
            'libraries': {
                'schools': 'schools.templatetags.schools'
            }
        },
Evgenii Legotckoi
  • Dec. 27, 2018, 6:42 a.m.
  • (edited)

Странно конечно, что потребовалось добавлять каталог в libraries.

Ты добавил файл __init__.py в каталог templatetags? Чтобы получилась следующая структура

  • schools
    • templatetags
      • __init__.py
      • schools.py
bernar92
  • Dec. 27, 2018, 7:24 a.m.

да. не помогло. наверное изза того что джанго 2

Evgenii Legotckoi
  • Dec. 27, 2018, 8:18 a.m.

у меня тоже Django 2, не в этом дело, скорее всего в настройках ещё что-то пропущено, не знаю, мне всегда хватало того, что я указал уже выше.

Видимо я не обращал внимание на это

ПК
  • Dec. 28, 2018, 5:02 a.m.

Шаблонные теги здесь совсем не причём. Уберите строки с template.Library, измените название на что то типа "одна из возможных реализаций бокового меню" и будет статья о контекстных процессорах в django для самых начинающих.

Comments

Only authorized users can post comments.
Please, Log in or Sign up
MB

Qt - Test 001. Signals and slots

  • Result:57points,
  • Rating points-2
MB

C++ - Test 001. The first program and data types

  • Result:60points,
  • Rating points-1
GK

C++ - Test 005. Structures and Classes

  • Result:0points,
  • Rating points-10
Last comments
J
JonnyJoJune 8, 2023, 9:14 a.m.
Qt/C++ - Lesson 019. How to paint triangle in Qt5. Positioning shapes in QGraphicsScene Евгений, здравствуйте! Решил поэкспериментировать немного с кодом из этого урока, нарисовать вместо треугольника квадрат и разобраться с координатами. В итоге, запутался. И ни документация,…
J
JonnyJoMay 25, 2023, 11:24 a.m.
How to make game using Qt - Lesson 2. Animation game hero (2D) Евгений, благодарю!
Evgenii Legotckoi
Evgenii LegotckoiMay 25, 2023, 1:49 a.m.
How to make game using Qt - Lesson 2. Animation game hero (2D) Код на строчка 184-198 вызывает перерисовку области на каждый 4-й такт счётчика. По той логике не нужно перерисовывать объект постоянно, достаточно реже, чем выполняется игровой слот. А слот вып…
J
JonnyJoMay 21, 2023, 7:49 a.m.
How to make game using Qt - Lesson 2. Animation game hero (2D) Евгений, благодарю! Всё равно не совсем понимаю :( Если муха двигает ножками только при нажатии клавиш перемещение, то что, собственно, делает код со строк 184-198 в triangle.cpp? В этих строчка…
Evgenii Legotckoi
Evgenii LegotckoiMay 21, 2023, 2:57 a.m.
How to make game using Qt - Lesson 2. Animation game hero (2D) Добрый день. slotGameTimer срабатывает по таймеру и при каждой сработке countForSteps увеличивается на 1, это не зависит от нажатия клавиш, нажатая клавиша лишь определяет положение ножек, котор…
Now discuss on the forum
T
TwangerJune 7, 2023, 8:12 a.m.
Ошибка при выполнении триггерной функции (GreenPlum) Есть 3 таблицы fact_amount со структурой: CREATE TABLE fact_amount ( id serial4 NOT NULL, fdate date NULL, type_activity_id int4 NULL, status_id int4 NULL, CONSTRAINT fact…
AR
Alexander RyabikovJune 6, 2023, 10:35 a.m.
Работа с QFileSystemModel Вопросик по теме QFileSystemModel в Linux. Он, как и положено, обновляется самостоятельно, если директория локальная. Но, вот, сетевая папка (у меня шара samba) не обновляется. Как её можно…
Evgenii Legotckoi
Evgenii LegotckoiApril 16, 2023, 1:07 a.m.
Мобильное приложение на C++Qt и бэкенд к нему на Django Rest Framework Да, это возможно. Но подобные вещи лучше запускать через celery. То есть drf принимает команду, и после этого регистрирует задачу в celery, котроый уже асинхронно всё это выполняет. В противном …
АБ
Алексей БобровDec. 14, 2021, 4:03 p.m.
Sorting the added QML elements in the ListModel I am writing an alarm clock in QML, I am required to sort the alarms in ascending order (depending on the date or time (if there are several alarms on the same day). I've done the sorting …

Follow us in social networks