bernar92
bernar9225 декабря 2018 г. 23:35

Одна из возможных реализаций бокового меню для Django

Привет. Не давно была задача реализовать боковое меню со списком каталогов; что бы его можно было вставить в любое место и в любую страницу!
Проблема заключалась в том что список большой и для того что бы он не нагружал сервак я его закэшировал, но тут не буду указывать как закэшировал так как писать придется долго...
В общем буду описывать все по порядку!


Начнем с создания шаблонного тега! для того что бы его реализовать! я создал папку пакета:

назвать его можно как хотите но я его назвал его: context_processors
далее создал файл : right_menu.py
в котором будет реализована вся логика шаблонного тега.

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


далее в settings.py подключаем шаблонный тег

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', # как видете тут я его подключил
            ],
        },
    },
]

в самом шаблоне right_menu.html

{% load static %}
{% load i18n %}
<ul>
    {% for i in categorys %}
    <li>{{ i.name }}</li>
    {% endfor %}
</ul>
Рекомендуем хостинг TIMEWEB
Рекомендуем хостинг TIMEWEB
Стабильный хостинг, на котором располагается социальная сеть EVILEG. Для проектов на Django рекомендуем VDS хостинг.

Вам это нравится? Поделитесь в социальных сетях!

Evgenii Legotckoi
  • 26 декабря 2018 г. 4:53

Добрый день!

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

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

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

bernar92
  • 26 декабря 2018 г. 4:59

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

Evgenii Legotckoi
  • 26 декабря 2018 г. 5:15
  • (ред.)

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

Кстати, самого примере применения 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
  • 27 декабря 2018 г. 0:16
  • (ред.)

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

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

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

Evgenii Legotckoi
  • 27 декабря 2018 г. 1:17

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

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

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

или нет

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

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

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

bernar92
  • 27 декабря 2018 г. 1:36

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

return render(request, 'includes/header.html', context)
Evgenii Legotckoi
  • 27 декабря 2018 г. 1:46
  • (ред.)

нет, я предлагаю сделать так. В каталоге приложения добавляешь каталог 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
  • 27 декабря 2018 г. 2:03

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

Evgenii Legotckoi
  • 27 декабря 2018 г. 2:09

нет, в твоём случае он остается, это штука добавляет в контекст 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
  • 27 декабря 2018 г. 2:14

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

Evgenii Legotckoi
  • 27 декабря 2018 г. 2:17

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

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

bernar92
  • 27 декабря 2018 г. 3:39
  • (ред.)

Django 2.1

'schools' is not a registered tag library. Must be one of:
{% load schools %}
{% r_schools %}
bernar92
  • 27 декабря 2018 г. 4:37

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

'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
  • 27 декабря 2018 г. 6:42
  • (ред.)

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

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

  • schools
    • templatetags
      • __init__.py
      • schools.py
bernar92
  • 27 декабря 2018 г. 7:24

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

Evgenii Legotckoi
  • 27 декабря 2018 г. 8:18

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

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

ПК
  • 28 декабря 2018 г. 5:02

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

Комментарии

Только авторизованные пользователи могут публиковать комментарии.
Пожалуйста, авторизуйтесь или зарегистрируйтесь
d
  • dsfs
  • 26 апреля 2024 г. 14:56

C++ - Тест 004. Указатели, Массивы и Циклы

  • Результат:80баллов,
  • Очки рейтинга4
d
  • dsfs
  • 26 апреля 2024 г. 14:45

C++ - Тест 002. Константы

  • Результат:50баллов,
  • Очки рейтинга-4
d
  • dsfs
  • 26 апреля 2024 г. 14:35

C++ - Тест 001. Первая программа и типы данных

  • Результат:73баллов,
  • Очки рейтинга1
Последние комментарии
k
kmssr9 февраля 2024 г. 5:43
Qt Linux - Урок 001. Автозапуск Qt приложения под Linux как сделать автозапуск для флэтпака, который не даёт создавать файлы в ~/.config - вот это вопрос ))
АК
Анатолий Кононенко5 февраля 2024 г. 12:50
Qt WinAPI - Урок 007. Работаем с ICMP Ping в Qt Без строки #include <QRegularExpressionValidator> в заголовочном файле не работает валидатор.
EVA
EVA25 декабря 2023 г. 21:30
Boost - статическая линковка в CMake проекте под Windows Ошибка LNK1104 часто возникает, когда компоновщик не может найти или открыть файл библиотеки. В вашем случае, это файл libboost_locale-vc142-mt-gd-x64-1_74.lib из библиотеки Boost для C+…
J
JonnyJo25 декабря 2023 г. 19:38
Boost - статическая линковка в CMake проекте под Windows Сделал всё по-как у вас, но выдаёт ошибку [build] LINK : fatal error LNK1104: не удается открыть файл "libboost_locale-vc142-mt-gd-x64-1_74.lib" Хоть убей, не могу понять в чём дел…
G
Gvozdik19 декабря 2023 г. 8:01
Qt/C++ - Урок 056. Подключение библиотеки Boost в Qt для компиляторов MinGW и MSVC Для решения твой проблемы добавь в файл .pro строчку "LIBS += -lws2_32" она решит проблему , лично мне помогло.
Сейчас обсуждают на форуме
G
Gar22 апреля 2024 г. 15:46
Clipboard Как скопировать окно целиком в clipb?
DA
Dr Gangil Academics20 апреля 2024 г. 17:45
Unlock Your Aesthetic Potential: Explore MSC in Facial Aesthetics and Cosmetology in India Embark on a transformative journey with an msc in facial aesthetics and cosmetology in india . Delve into the intricate world of beauty and rejuvenation, guided by expert faculty and …
a
a_vlasov14 апреля 2024 г. 16:41
Мобильное приложение на C++Qt и бэкенд к нему на Django Rest Framework Евгений, добрый день! Такой вопрос. Верно ли следующее утверждение: Любое Android-приложение, написанное на Java/Kotlin чисто теоретически (пусть и с большими трудностями) можно написать и на C+…
Павел Дорофеев
Павел Дорофеев14 апреля 2024 г. 12:35
QTableWidget с 2 заголовками Вот тут есть кастомный QTableView с многорядностью проект поддерживается, обращайтесь
f
fastrex4 апреля 2024 г. 14:47
Вернуть старое поведение QComboBox, не менять индекс при resetModel Добрый день! У нас много проектов в которых используется QComboBox, в версии 5.5.1, когда модель испускает сигнал resetModel, currentIndex не менялся. В версии 5.15 при resetModel происходит try…

Следите за нами в социальных сетях