Arrow
14 января 2019 г. 22:42

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

select, choices

Добрый день!

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

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

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

  1. from django.db import models
  2.  
  3. # Create your models here.
  4. class Test(models.Model):
  5. METRIC = 'metric'
  6. IMPERIAL = 'imperial'
  7. ATHER = ''
  8. UNIT = (
  9. (METRIC, 'kg'),
  10. (IMPERIAL, 'lb'),
  11. (ATHER, 'none')
  12. )
  13. name = models.CharField(max_length=25)
  14. temperature_unit = models.CharField(max_length=8,
  15. choices=UNIT,
  16. default=METRIC)
  17.  
  18. def __str__(self):
  19. return self.name
  20.  
  21. class Meta:
  22. verbose_name_plural = 'tests'
  23.  

Файл forms.py

  1. from django.forms import ModelForm, TextInput
  2. from .models import Test
  3.  
  4. class TestForm(ModelForm):
  5. class Meta:
  6. model = Test
  7. fields = ['name', 'unit']
  8. widgets = {'name': TextInput(attrs={'class': 'input', 'placeholder': 'Name'})}
  9.  

Файл *.html

  1. <form method="POST">
  2. {% csrf_token %}
  3. <div class="field has-addons">
  4. <div class="control is-expanded">
  5. {{ form.name }}
  6. </div>
  7. <select>
  8. {% for temp in form.tunit %}
  9. <option value=""{{ temp.data.value }}>{{ temp.data.label }}</option>
  10. {% endfor %}
  11. </select>
  12. <div class="control">
  13. <button type="submit" class="button is-info">
  14. Add Unit
  15. </button>
  16. </div>
  17. </div>
  18. </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

  1. import requests
  2. from django.shortcuts import render
  3. from .models import Test
  4. from .forms import TestForm
  5.  
  6. def index(request):
  7. url = 'http://api.site.org/data/2.5/?q={}&units={}&id=1111&appid=111111111'
  8.  
  9. if request.method == 'POST':
  10. # Здесь request.POST нет данных unit только name
  11. # Вылетает ошибка
  12. form = TestForm(request.POST)
  13. form.save()
  14.  
  15. tests = Test.objects.all()
  16.  
  17. test_data = []
  18.  
  19. for test in tests:
  20. # Преобразование данных из metric в kg итд.
  21. r = requests.get(url.format(test, test.unit)).json()
  22. res = [data for (t_type, data) in Test.UNIT if t_type == test.unit]
  23.  
  24. test_w = {
  25. 'name': test.name,
  26. 'unit': res[0],
  27. 'temp': r['main']['temp'],
  28. 'description': r['data'][0]['description'],
  29. 'icon': r['data'][0]['icon'],
  30. }
  31.  
  32. test_data.append(test_w)
  33.  
  34. context = {'test_data': test_data, 'form': form}
  35.  
  36. return render(request, 'module/module.html', context)
  37.  

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

2

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

18
Evgenii Legotckoi
  • 14 января 2019 г. 22:46

Добрый день!

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

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

Так нужно

  1. from django.forms import ModelForm, TextInput
  2. from .models import Test
  3.  
  4. class TestForm(ModelForm):
  5. class Meta:
  6. model = Test
  7. fields = ['name', 'temperature_unit']
  8. widgets = {'name': TextInput(attrs={'class': 'input', 'placeholder': 'Name'})}

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

    Arrow
    • 14 января 2019 г. 22:52

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

      Arrow
      • 14 января 2019 г. 22:53
      • (ред.)

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

        Arrow
        • 15 января 2019 г. 0:59

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

          Evgenii Legotckoi
          • 15 января 2019 г. 1:54

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

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

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

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

            Arrow
            • 15 января 2019 г. 1:55

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

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

            а не:

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

              Evgenii Legotckoi
              • 15 января 2019 г. 2:03
              • (ред.)

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

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

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

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

                Получаю:

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

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

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

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

                  Evgenii Legotckoi
                  • 15 января 2019 г. 2:12

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

                  Вот отсюда

                    Arrow
                    • 15 января 2019 г. 2:16

                    Все ясно:

                      Evgenii Legotckoi
                      • 15 января 2019 г. 2:18
                      • Ответ был помечен как решение.

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

                      вот ваш код

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

                      а вот как надо

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

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

                        Arrow
                        • 15 января 2019 г. 2:18
                        • (ред.)

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

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

                        Нужно:

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

                          Arrow
                          • 15 января 2019 г. 2:20

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

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

                            Evgenii Legotckoi
                            • 15 января 2019 г. 2:22

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

                              Arrow
                              • 15 января 2019 г. 2:25

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

                                Arrow
                                • 15 января 2019 г. 2:53
                                • (ред.)

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

                                models.py

                                1. from django.db import models
                                2.  
                                3. class TempUnit(models.Model):
                                4. name = models.CharField(max_length=2)
                                5. unit = models.CharField(max_length=8)
                                6.  
                                7. def __str__(self):
                                8. return self.name
                                9.  
                                10.  
                                11. class City(models.Model):
                                12. name = models.CharField(max_length=25)
                                13. temperature_unit = models.ForeignKey(TempUnit, on_delete=models.CASCADE)
                                14.  
                                15. def __str__(self):
                                16. return self.name
                                17.  
                                18. class Meta:
                                19. verbose_name_plural = 'cities'
                                20.  

                                views.py

                                1. def index(request):
                                2. url = 'http://api.site.org/data/2.5/?q={}&units={}&id=1111&appid=111111111111'
                                3.  
                                4. if request.method == 'POST':
                                5. form = CityForm(request.POST)
                                6. form.save()
                                7.  
                                8. cities = City.objects.all()
                                9. weather_data = []
                                10.  
                                11. for city in cities:
                                12. r = requests.get(url.format(city, city.temperature_unit)).json()
                                13.  
                                14. city_weather = {
                                15. 'city': city.name,
                                16. 'temperature_unit': city.temperature_unit,
                                17. 'temperature': r['main']['temp'],
                                18. 'description': r['weather'][0]['description'],
                                19. 'icon': r['weather'][0]['icon'],
                                20. }
                                21.  
                                22. weather_data.append(city_weather)
                                23.  
                                24. context = {'weather_data': weather_data, 'form': form}
                                25.  
                                26. return render(request, 'data/data.html', context)
                                27.  

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

                                  Evgenii Legotckoi
                                  • 15 января 2019 г. 3:00

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

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

                                    Arrow
                                    • 15 января 2019 г. 3:46

                                    Спасибо.

                                      Комментарии

                                      Только авторизованные пользователи могут публиковать комментарии.
                                      Пожалуйста, авторизуйтесь или зарегистрируйтесь
                                      • Последние комментарии
                                      • Evgenii Legotckoi
                                        16 апреля 2025 г. 17:08
                                        Благодарю за отзыв. И вам желаю всяческих успехов!
                                      • IscanderChe
                                        12 апреля 2025 г. 17:12
                                        Добрый день. Спасибо Вам за этот проект и отдельно за ответы на форуме, которые мне очень помогли в некоммерческих пет-проектах. Профессиональным программистом я так и не стал, но узнал мно…
                                      • AK
                                        1 апреля 2025 г. 11:41
                                        Добрый день. В данный момент работаю над проектом, где необходимо выводить звук из программы в определенное аудиоустройство (колонки, наушники, виртуальный кабель и т.д). Пишу на Qt5.12.12 поско…
                                      • Evgenii Legotckoi
                                        9 марта 2025 г. 21:02
                                        К сожалению, я этого подсказать не могу, поскольку у меня нет необходимости в обходе блокировок и т.д. Поэтому я и не задавался решением этой проблемы. Ну выглядит так, что вам действитель…
                                      • VP
                                        9 марта 2025 г. 16:14
                                        Здравствуйте! Я устанавливал Qt6 из исходников а также Qt Creator по отдельности. Все компоненты, связанные с разработкой для Android, установлены. Кроме одного... Когда пытаюсь скомпилиров…