. .return render(request, 'learning_logs/new_entry.html', context)
Мы обновляем команду import и включаем в нее только что созданный класс EntryForm. Определение new_entry() содержит параметр topic_id для сохранения полученного значения из URL. Идентификатор темы понадобится для отображения страницы и обработки данных формы, поэтому мы используем topic_id для получения правильного объекта темы (1) .
В точке (2) проверяется метод запроса: POST или GET. Блок if выполняется для запроса GET, и мы создаем пустой экземпляр EntryForm (3). Для метода запроса POST мы обрабатываем данные, создавая экземпляр EntryForm, заполненный данными POST из объекта запроса (4). Затем проверяется корректность данных формы. Если данные корректны, необходимо задать атрибут topic объекта записи перед сохранением его в базе данных.
При вызове save() мы включаем аргумент commit=False (5) для того, чтобы создать новый объект записи и сохранить его в new_entry, не сохраняя пока в базе данных. Мы присваиваем атрибуту topic объекта new_entry тему, прочитанную из базы данных в начале функции ?, после чего вызываем save() без аргументов. В результате запись сохраняется в базе данных с правильной ассоциированной темой.
В точке ? пользователь перенаправляется на страницу темы. При вызове reverse() должны передаваться два аргумента: имя схемы URL, для которой генерируется URL-адрес, и список аргументов со всеми аргументами, которые должны быть включены в URL. Список аргументов содержит всего один элемент topic_id. Вызов HttpResponseRedirect() перенаправляет пользователя на страницу темы, для которой была создана запись, и пользователь видит новую запись в списке записей.
Шаблон new_entry
Как видно из следующего кода, шаблон new_entry похож на шаблон new_topic:
new_entry.html
{% extends "learning_logs/base.html" %}
{% block content %}
(1)
. .
Add a new entry:
(2)
. .{% csrf_token %}
. .{{ form.as_p }}
. .
. .
{% endblock content %}
В начале страницы выводится тема (1) , чтобы пользователь мог видеть, в какую тему добавляется новая запись. Тема также служат ссылкой для возврата к основной странице этой темы.
Аргумент action формы включает значение topic_id из URL, чтобы функция представления могла связать новую запись с правильной темой (2). В остальном этот шаблон почти не отличается от new_topic.html.
Создание ссылки на страницу new_entry
Затем необходимо создать ссылку на страницу new_entry на каждой странице темы:
topic.html
{% extends "learning_logs/base.html" %}
{% block content %}
Topic: {{ topic }}
Entries:
...
{% endblock content %}
Ссылка добавляется перед выводом записей, потому что добавление новой записи является самым частым действием на этой странице. На рис. 19.2 изображена страница new_entry. Теперь пользователь может добавить сколько угодно новых тем и новых записей по каждой теме. Опробуйте страницу new_entry, добавив несколько записей для каждой из созданных вами тем.
Рис. 19.2. Страница new_entry
Добавление записей
А теперь мы создадим страницу, на которой пользователи смогут редактировать ранее добавленные записи.
URL-адрес для edit_entry
В URL-адресе страницы должен передаваться идентификатор редактируемой записи. В файл learning_logs/urls.py для этого вносятся следующие изменения:
urls.py
...
urlpatterns = [
...
. .# Страница для редактирования записи
. .url(r'^edit_entry/(?P
. . . .name='edit_entry'),
]
Идентификатор, переданный в URL (например, http://localhost:8000/edit_entry/1/), сохраняется в параметре entry_id. Схема The URL отправляет запросы, соответствующие этому формату, функции представления edit_entry().
Функция представления edit_entry()
Когда страница edit_entry получает запрос GET, edit_entry() возвращает форму для редактирования записи. При получении запроса POST с отредактированной записью страница сохраняет измененный текст в базе данных:
views.py
from django.shortcuts import render
...
from .models import Topic, Entry
from .forms import TopicForm, EntryForm
...
def edit_entry(request, entry_id):
. ."""Редактирует существующую запись."""
(1) . .entry = Entry.objects.get(id=entry_id)
. .topic = entry.topic
. .
. .if request.method != 'POST':
. . . .# Исходный запрос; форма заполняется данными текущей записи.
(2) . . . .form = EntryForm(instance=entry)
. .else:
. . . .# Отправка данных POST; обработать данные.
(3) . . . .form = EntryForm(instance=entry, data=request.POST)
. . . .if form.is_valid():
(4) . . . . . .form.save()
(5) . . . . . .return HttpResponseRedirect(reverse('learning_logs:topic',
. . . . . . . . . . . . . . . . . . . .args=[topic.id]))
. .
. .context = {'entry': entry, 'topic': topic, 'form': form}
. .return render(request, 'learning_logs/edit_entry.html', context)