Arrow
Arrow14 січня 2019 р. 11:42

Сохранение данных из выпадающего списка в модель

select, choices

Добрый день!

Возникла проблема с выпадающим списком, а точнее с сохранением из него данных.

Приведу простой пример.

Модель имеет вид (models.py)

from django.db import models

# Create your models here.
class Test(models.Model):
    METRIC = 'metric'
    IMPERIAL = 'imperial'
    ATHER = ''
    UNIT = (
        (METRIC, 'kg'),
        (IMPERIAL, 'lb'),
        (ATHER, 'none')
    )
    name = models.CharField(max_length=25)
    temperature_unit = models.CharField(max_length=8,
                                        choices=UNIT,
                                        default=METRIC)

    def __str__(self):
        return self.name

    class Meta:
        verbose_name_plural = 'tests'

Файл forms.py

from django.forms import ModelForm, TextInput
from .models import Test

class TestForm(ModelForm):
    class Meta:
        model = Test
        fields = ['name', 'unit']
        widgets = {'name': TextInput(attrs={'class': 'input', 'placeholder': 'Name'})}

Файл *.html

<form method="POST">
    {% csrf_token %}
        <div class="field has-addons">
            <div class="control is-expanded">
                {{ form.name }}
            </div>
            <select>
                {% for temp in form.tunit %}
                    <option value=""{{ temp.data.value }}>{{ temp.data.label }}</option>
                {% endfor %}
             </select>
             <div class="control">
                <button type="submit" class="button is-info">
                    Add Unit
                </button>
            </div>
        </div>
</form>

В результате в качестве value выпадающего списка (temp.data.value) оказывается 'metric', 'imperial' и '', а в самом выпадающем списке на странице оказывается (temp.data.label): kg, lb, none. Как и должно быть.

Только при вводе данных в поле формы (name = models.CharField(max_length=25)) и попытке сохранения вылетает ошибка. В результате просмотра request.POST в нем оказывается только значение 'name' без 'unit'.

Файл views.py

import requests
from django.shortcuts import render
from .models import Test
from .forms import TestForm

def index(request):
    url = 'http://api.site.org/data/2.5/?q={}&units={}&id=1111&appid=111111111'

    if request.method == 'POST':
        # Здесь request.POST нет данных unit только name
        # Вылетает ошибка
        form = TestForm(request.POST) 
        form.save()

    tests = Test.objects.all()

    test_data = []

    for test in tests:
        # Преобразование данных из metric в kg итд.
        r = requests.get(url.format(test, test.unit)).json()
        res = [data for (t_type, data) in Test.UNIT if t_type == test.unit]

        test_w = {
            'name': test.name,
            'unit': res[0],
            'temp': r['main']['temp'],
            'description': r['data'][0]['description'],
            'icon': r['data'][0]['icon'],
        }

        test_data.append(test_w)

    context = {'test_data': test_data, 'form': form}

    return render(request, 'module/module.html', context)

Что я делаю не так?
Возможно есть другой способ более правильный:
поместить данные о unit в отдельную таблицу в базе данных и вытягивать от туда или еще как-то?

Рекомендуємо хостинг TIMEWEB
Рекомендуємо хостинг TIMEWEB
Стабільний хостинг, на якому розміщується соціальна мережа EVILEG. Для проектів на Django радимо VDS хостинг.

Вам це подобається? Поділіться в соціальних мережах!

18
Evgenii Legotckoi
  • 14 січня 2019 р. 11:46

Добрый день!

Так у вас в модели поде unit объявлено как temperature_unit

Вы имя перепутали

Так нужно

from django.forms import ModelForm, TextInput
from .models import Test

class TestForm(ModelForm):
    class Meta:
        model = Test
        fields = ['name', 'temperature_unit']
        widgets = {'name': TextInput(attrs={'class': 'input', 'placeholder': 'Name'})}

Ну и так далее везде поменяйте unit на temperature_unit

    Arrow
    • 14 січня 2019 р. 11:52

    Я уже поправил. Скопировал данные из реального и тестово проектов. Ошибка вышла. Извините. :(

      Arrow
      • 14 січня 2019 р. 11:53
      • (відредаговано)

      Но вопрос остался

        Arrow
        • 14 січня 2019 р. 13:59

        Посмотрел документацию, делаю правильно, но не работает.

          Evgenii Legotckoi
          • 14 січня 2019 р. 14:54

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

          <select id="id_temperature_unit" name="temperature_unit">
              {% for temp in form.temperature_unit %}
                   <option value=""{{ temp.data.value }}>{{ temp.data.label }}</option>
              {% endfor %}
          </select>
          

          Вообще у Field есть методы для получения этих всех id и name, но я как-то не помню их, нужно в документации покопаться.

          А так я использую django_bootstrap4, там эти моменты уже реализованы, так что всё автоматически работает. Не нужно всю эту вёрстку делать вручную.

            Arrow
            • 14 січня 2019 р. 14:55

            Все исправил, все пересмотрел еще раз.
            Понять не могу почему я в качестве request.POST при попытке сохранения данных, от формы получаю:

            {'csrfmiddlewaretoken': 'BrD9OchYiCLJJAV3U8MX7NewnCO2uZ2EjmyfOIesySZp6dmcyoA1GRwnwReSch0T',
            'name': 'Unit1'}

            а не:

            {'csrfmiddlewaretoken': 'BrD9OchYiCLJJAV3U8MX7NewnCO2uZ2EjmyfOIesySZp6dmcyoA1GRwnwReSch0T',
            'name': 'Unit1', temperature_unit: 'kg'}

              Evgenii Legotckoi
              • 14 січня 2019 р. 15:03
              • (відредаговано)

              А как выглядит отрендеренная форма? В смысле html можете показать итоговый?
              То есть открываете страницу с формой и через браузер посмотрите html и покажите, что там вышло.
              Вот например у меня в одной из форм так получается.

              <select name="type" id="id_type" class="form-control" title="">
                  <option value="T" selected="">Руководство</option>
                  <option value="N">Новости</option>
                  <option value="S">Сниппет</option>
                  <option value="O">Заметка</option>
              </select>
              
                Arrow
                • 14 січня 2019 р. 15:08
                • (відредаговано)

                После исправления:

                <select id="id_temperature_unit" name="temperature_unit">
                    {% for temp in form.temperature_unit %}
                         <option value=""{{ temp.data.value }}>{{ temp.data.label }}</option>
                    {% endfor %}
                </select>
                

                Получаю:

                {'csrfmiddlewaretoken': ['EvCSJRprJbsn3Rte7AxtnpeXeNO3pIThmqxYJnmVZrG3quUnLQlxWtwOn2eT70Rw'], 'name': ['Lviv'], 'temperature_unit': ['']}

                Только что-бы я не выбрал в списке значение temperature_unit не меняется.

                Форма выглядит так:

                Пытаюсь на сайт притулить индикатор погоды с выбором в каких унитах показывать температеру.

                  Evgenii Legotckoi
                  • 14 січня 2019 р. 15:12

                  Нет, вы не поняли, покажите мне отрендеренный html из консоли браузера.

                  Вот отсюда

                    Arrow
                    • 14 січня 2019 р. 15:16

                    Все ясно:

                      Evgenii Legotckoi
                      • 14 січня 2019 р. 15:18
                      • Відповідь була позначена як рішення.

                      И кстати да, я увидел вашу ошибку,

                      вот ваш код

                      <select id="id_temperature_unit" name="temperature_unit">
                          {% for temp in form.temperature_unit %}
                               <option value=""{{ temp.data.value }}>{{ temp.data.label }}</option>
                          {% endfor %}
                      </select>
                      

                      а вот как надо

                      <select id="id_temperature_unit" name="temperature_unit">
                          {% for temp in form.temperature_unit %}
                               <option value="{{ temp.data.value }}">{{ temp.data.label }}</option>
                          {% endfor %}
                      </select>
                      

                      Найдёте, где кавычки перепутали?

                        Arrow
                        • 14 січня 2019 р. 15:18
                        • (відредаговано)

                        Нашел ошибку. Было:

                        option value=""{{ temperature.data.value }}>{{ temperature.data.label }}

                        Нужно:

                        option value="{{ temperature.data.value }}">{{ temperature.data.label }}

                          Arrow
                          • 14 січня 2019 р. 15:20

                          Оказывается все было правильно кроме кавычки :) Буду чаше заглядывать в Инспектор.

                          Огромное спасибо! :)

                            Evgenii Legotckoi
                            • 14 січня 2019 р. 15:22

                            Я с консолью браузера при разработке на django вообще не расстаюсь. Иначе можно голову долго ломать.

                              Arrow
                              • 14 січня 2019 р. 15:25

                              Теперь так и буду делать :)

                                Arrow
                                • 14 січня 2019 р. 15:53
                                • (відредаговано)

                                Если кому-то понадобится есть еще один вариант, который я думаю более верный:

                                models.py

                                from django.db import models
                                
                                class TempUnit(models.Model):
                                    name = models.CharField(max_length=2)
                                    unit = models.CharField(max_length=8)
                                
                                    def __str__(self):
                                        return self.name
                                
                                
                                class City(models.Model):
                                    name = models.CharField(max_length=25)
                                    temperature_unit = models.ForeignKey(TempUnit, on_delete=models.CASCADE)
                                
                                    def __str__(self):
                                        return self.name
                                
                                    class Meta:
                                        verbose_name_plural = 'cities'
                                
                                

                                views.py

                                def index(request):
                                    url = 'http://api.site.org/data/2.5/?q={}&units={}&id=1111&appid=111111111111'
                                
                                    if request.method == 'POST':
                                        form = CityForm(request.POST)
                                        form.save()
                                
                                    cities = City.objects.all()
                                    weather_data = []
                                
                                    for city in cities:
                                        r = requests.get(url.format(city, city.temperature_unit)).json()
                                
                                        city_weather = {
                                            'city': city.name,
                                            'temperature_unit': city.temperature_unit,
                                            'temperature': r['main']['temp'],
                                            'description': r['weather'][0]['description'],
                                            'icon': r['weather'][0]['icon'],
                                        }
                                
                                        weather_data.append(city_weather)
                                
                                    context = {'weather_data': weather_data, 'form': form}
                                
                                    return render(request, 'data/data.html', context)
                                
                                

                                Тоесть просто создал новую модель и все скинул туда.

                                  Evgenii Legotckoi
                                  • 14 січня 2019 р. 16:00

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

                                  Однако переходите к CBV (Class Based View) вместо обычных вьюшек. Так лучше будет, в будущем оцените.

                                    Arrow
                                    • 14 січня 2019 р. 16:46

                                    Спасибо.

                                      Коментарі

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

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

                                      • Результат:50бали,
                                      • Рейтинг балів-4
                                      m
                                      • molni99
                                      • 26 жовтня 2024 р. 01:37

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

                                      • Результат:80бали,
                                      • Рейтинг балів4
                                      m
                                      • molni99
                                      • 26 жовтня 2024 р. 01:29

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

                                      • Результат:20бали,
                                      • Рейтинг балів-10
                                      Останні коментарі
                                      ИМ
                                      Игорь Максимов22 листопада 2024 р. 11:51
                                      Django - Підручник 017. Налаштуйте сторінку входу до Django Добрый вечер Евгений! Я сделал себе авторизацию аналогичную вашей, все работает, кроме возврата к предидущей странице. Редеректит всегда на главную, хотя в логах сервера вижу запросы на правильн…
                                      Evgenii Legotckoi
                                      Evgenii Legotckoi31 жовтня 2024 р. 14:37
                                      Django - Урок 064. Як написати розширення для Python Markdown Добрый день. Да, можно. Либо через такие же плагины, либо с постобработкой через python библиотеку Beautiful Soup
                                      A
                                      ALO1ZE19 жовтня 2024 р. 08:19
                                      Читалка файлів fb3 на Qt Creator Подскажите как это запустить? Я не шарю в программировании и кодинге. Скачал и установаил Qt, но куча ошибок выдается и не запустить. А очень надо fb3 переконвертировать в html
                                      ИМ
                                      Игорь Максимов05 жовтня 2024 р. 07:51
                                      Django - Урок 064. Як написати розширення для Python Markdown Приветствую Евгений! У меня вопрос. Можно ли вставлять свои классы в разметку редактора markdown? Допустим имея стандартную разметку: <ul> <li></li> <li></l…
                                      d
                                      dblas505 липня 2024 р. 11:02
                                      QML - Урок 016. База даних SQLite та робота з нею в QML Qt Здравствуйте, возникает такая проблема (я новичок): ApplicationWindow неизвестный элемент. (М300) для TextField и Button аналогично. Могу предположить, что из-за более новой верси…
                                      Тепер обговоріть на форумі
                                      Evgenii Legotckoi
                                      Evgenii Legotckoi24 червня 2024 р. 15:11
                                      добавить qlineseries в функции Я тут. Работы оень много. Отправил его в бан.
                                      t
                                      tonypeachey115 листопада 2024 р. 06:04
                                      google domain [url=https://google.com/]domain[/url] domain [http://www.example.com link title]
                                      NSProject
                                      NSProject04 червня 2022 р. 03:49
                                      Всё ещё разбираюсь с кешем. В следствии прочтения данной статьи. Я принял для себя решение сделать кеширование свойств менеджера модели LikeDislike. И так как установка evileg_core для меня не была возможна, ибо он писался…
                                      9
                                      9Anonim25 жовтня 2024 р. 09:10
                                      Машина тьюринга // Начальное состояние 0 0, ,<,1 // Переход в состояние 1 при пустом символе 0,0,>,0 // Остаемся в состоянии 0, двигаясь вправо при встрече 0 0,1,>…

                                      Слідкуйте за нами в соціальних мережах