Использование Markdown в Django

9594bc7f65b6c25e22ad15fe831eac03

Привет, Habr!

В этой статье мы научимся использовать Markdown вместе с Django на примере блога. Она написана для новичков, для базового ознакомления. Её в формате .md можно скачать в моём развивающемся Telegram канале.

Markdown — это простой язык разметки, используемый для создания форматированного текста (например, HTML) с помощью текстового редактора. Кстати, эту статью я писал, использую синтаксис Markdown)

Создание и активация виртуального окружения

python -m venv venv

source venv/bin/activate

Установка необходимых пакетов

pip install Django==4.2

Версия Django актуальна на момент 2024 года.

pip install markdown

Создание сайта и приложения

django-admin startproject markdown_blog

cd markdown_blog

python manage.py startapp posts

Мы создаем:

Создание модели поста

Откройте файл posts/models.py, и напишите следующее:

from django.db import models
from django.contrib.auth.models import User
from django.urls import reverse


class Post(models.Model):
    title = models.CharField(max_length=250)
    slug = models.SlugField(max_length=250)
    author = models.ForeignKey(User,
    on_delete=models.CASCADE,
    related_name='blog_posts')
    body = models.TextField()

	def __str__(self):
		return self.title

	def get_absolute_url(self):
        return reverse('post_detail',
        args=[self.id])

Мы создали модель Post, в которой есть такие поля:

  • title — Заголовок, длиной до 250 символов

  • slug — Слаг, длиной до 250 символов

  • author — ForeignKey к модели User. При удалении пользователя удаляются все его посты.

  • body — Текст поста (с этим полем мы и поработаем)

Добавьте приложение в markdown_blog/settings.py:

INSTALLED_APPS = [
				  # other apps,
				  'posts.apps.PostsConfig',
]

Создайте и выполните миграции:

python manage.py makemigrations

python manage.py migrate

Регистрация модели в админ панели

Откройте файл posts/admin.py, и напишите следующее:

from django.contrib import admin
from .models import Post

admin.site.register(Post)

Мы зарегистрировали модель Post в админ панели.

Создание представлений и шаблонов

Откройте файл posts/views.py, и напишите следующее:

from django.shortcuts import render, get_object_or_404

from .models import Post


def post_list(request):
    posts = Post.objects.all()
    return render(request,
    'post_list.html',
    {'posts': posts})

def post_detail(request, id):
	post = get_object_or_404(Post,
	id=id)
	return render(request,
	'post_detail.html',
	{'post': post})
	

Мы создали два простых представления:

  • post_list — Получает из базы данных список постов, и передаёт его в шаблон post_list.html.

  • post_detail — Получает из базы данных пост по id, и передает его в шаблон post_detail.html.

Создайте директорию posts/templates и внутри создайте новый файл post_list.html. В него напишите следующее:





    
    List of posts



{% for post in posts %}

{{ post.title }}

{{ post.body|truncatewords_html:30 }}
{% endfor %}

Внутри posts/templates создайте файл post_detail.html. В него напишите следующее:





    
    Post: {{ post.title }}



List of posts

{{ post.title }}

{{ post.body }}

Откройте файл markdown_blog/urls.py, и напишите следующее:

from django.contrib import admin
from django.urls import path

from posts import views

urlpatterns = [
  path('admin/', admin.site.urls),
  path('', views.post_list, name='post_list'),
  path('/', views.post_detail, name='post_detail'),
]

Мы создали два URL для наших представлений.

Добавление синтаксиса Markdown на сайт

Один из способов добавления синтаксиса Markdown на сайт — это создание конкретно-прикладного шаблонного фильтра.

Создайте пакет posts/templatetags, создайте файлы post_tags.py, и __init.py__. Файл post_tags.py будет содержать нашу библиотеку конкретно-прикладных шаблонных фильтров и тегов. В него напишите следующее:

from django import template
from django.utils.safestring import mark_safe

import markdown

register = template.Library()


@register.filter(name='markdown')
def markdown_format(text):
    return mark_safe(markdown.markdown(text))

Мы используем предоставляемую веб-фреймворком Django функцию
mark_safe, чтобы помечать результат как безопасный для прорисовки в шаблоне исходный код HTML. По умолчанию Django не будет доверять никакому исходному коду HTML и будет экранировать его перед его вставкой
в результат. Единственными исключениями являются переменные, которые
помечены как безопасные, чтобы тем самым избежать экранирования. Такое
поведение не дает Django выводить потенциально опасный исходный код
HTML и позволяет создавать исключения, дабы возвращать безопасный исходный код HTML.

Отредактируйте файл templates/post_detail.html:

{% load post_tags %}




    
    Post: {{ post.title }}



List of posts

{{ post.title }}

{{ post.body|markdown }}

Отредактируйте файл templates/post_list.html:

{% load post_tags %}




    
    List of posts



{% for post in posts %}

{{ post.title }}

{{ post.body|markdown|truncatewords_html:30 }}
{% endfor %}

Проверка

Создайте суперпользователя:

python manage.py createsuperuser

Выполните все инструкции, и запустите сервер разработки:

python manage.py runserver

Пройдите по адресу 127.0.0.1:8000/admin, войдите, используя ранее введённые данные. Через админ панель добавьте несколько постов (используя синтаксис Markdown). Некоторые тексты есть ниже:

**Модель** — это единственный и точный источник информации о ваших данных. Он содержит основные поля и поведение данных, которые вы храните. Обычно каждая модель сопоставляется с одной таблицей базы данных.

### Основы
- - -

- Каждая модель представляет собой класс Python, который является подклассом `django.db.models.Model`.
- Каждый атрибут модели представляет поле базы данных.
- При всем этом Django предоставляет вам автоматически генерируемый API доступа к базе данных; см. [Создание запросов](https://docs.djangoproject.com/en/5.0/topics/db/queries/).


- - -
## MTV

Django подчиняется шаблону архитектурного дизайна **MTV** (Model-Template-View)

Обязанности в шаблоне архитектурного дизайна MTV Django распределе-
ны следующим образом:

- **Model** (модель) – определяет логическую структуру данных и является обработчиком данных между базой данных и их представлением

- **Template** (шаблон) – это слой представления. В Django используется система текстовых шаблонов, в которой хранится все, что браузер прорисовывает на страницах.

- **View** (представление) – взаимодействует с базой данных через модель и передает данные в шаблон для их прорисовки и просмотра.


- - -

Перейдите по адресу 127.0.0.1:8000. Вы должны увидеть список постов. Нажмите на одну из ссылок, вы увидите всё содержимое страницы.

Итог

Мы реализовали один из вариантов использования синтаксиса Markdown в Django. Целью статьи не была реализация идеального блога, а лишь простейшая реализация такой задачи. В дальнейшем можно встроить на сайт редактор для удобного написания страниц с использованием Markdown. Но это уже совсем другая история), в дальнейшем рассмотрим.

P.S. это моя первая статья. Всем пока, Control + D!

© Habrahabr.ru