Виджет широты / долготы для точечного поля?

Есть ли виджет для PointField в качестве отдельных входов широты / долготы? Как SplitDateTimeWidget для DateTimeField.

Ответы на вопрос(5)

решение, но с некоторыми улучшениями

class GisForm(forms.ModelForm):
    """
    This form can be used by any model that has "coordinates" field.
    It will show a better looking map than the default one
    """
    lat = forms.FloatField(required=False)
    lng = forms.FloatField(required=False)
    coordinates = PointField(widget=CustomPointWidget(), required=False, srid=4326)

    def __init__(self, *args, **kwargs):
        super(GisForm, self).__init__(*args, **kwargs)
        coordinates = self.initial.get("coordinates", None)
        if isinstance(coordinates, Point):
            self.initial["lng"], self.initial["lat"] = coordinates.tuple

    def clean(self):
        data = super(GisForm, self).clean()
        if "lat" in self.changed_data or "lng" in self.changed_data:
            lat, lng = data.pop("lat", None), data.pop("lng", None)
            data["coordinates"] = Point(lng, lat, srid=4326)

        if not (data.get("coordinates") or data.get("lat")):
            raise forms.ValidationError({"coordinates": "Coordinates is required"})
        return data

Это в основном позволит пользователям либо устанавливать местоположение на карте, либо вручную указывать значения широты / долготы. после сохранения карта будет отражать заданное значение широты / долготы

Решение Вопроса

Мои рабочие поля и виджет:

class LatLongWidget(forms.MultiWidget):
    """
    A Widget that splits Point input into two latitude/longitude boxes.
    """

    def __init__(self, attrs=None, date_format=None, time_format=None):
        widgets = (forms.TextInput(attrs=attrs),
                   forms.TextInput(attrs=attrs))
        super(LatLongWidget, self).__init__(widgets, attrs)

    def decompress(self, value):
        if value:
            return tuple(reversed(value.coords))
        return (None, None)

class LatLongField(forms.MultiValueField):

    widget = LatLongWidget
    srid = 4326

    default_error_messages = {
        'invalid_latitude' : _('Enter a valid latitude.'),
        'invalid_longitude' : _('Enter a valid longitude.'),
    }

    def __init__(self, *args, **kwargs):
        fields = (forms.FloatField(min_value=-90, max_value=90), 
                  forms.FloatField(min_value=-180, max_value=180))
        super(LatLongField, self).__init__(fields, *args, **kwargs)

    def compress(self, data_list):
        if data_list:
            # Raise a validation error if latitude or longitude is empty
            # (possible if LatLongField has required=False).
            if data_list[0] in validators.EMPTY_VALUES:
                raise forms.ValidationError(self.error_messages['invalid_latitude'])
            if data_list[1] in validators.EMPTY_VALUES:
                raise forms.ValidationError(self.error_messages['invalid_longitude'])
            # SRID=4326;POINT(1.12345789 1.123456789)
            srid_str = 'SRID=%d'%self.srid
            point_str = 'POINT(%f %f)'%tuple(reversed(data_list))
            return ';'.join([srid_str, point_str])
        return None
 raratiru23 июл. 2014 г., 09:26
@fangsterr Вы можете использоватьFloatField или жеDecimalField сохранить ваши координаты и переопределить метод сохранения, чтобы обновитьPointFieldкак показаноВот
 fangsterr18 июл. 2014 г., 09:35
Я получаюневерное значение геометрии ' ошибка, когда я использую это

Создайте собственный виджет для этого, вы можете получить вдохновение отSelectDateWidget Вот.

но если вам нужно простое решение, которое выглядит хорошо и легко в использовании, попробуйте:http://django-map-widgets.readthedocs.io/en/latest/index.html

Вы можете:

введите координатывыберите прямо на картеискать элементы из мест Google.

 darkless18 янв. 2019 г., 17:13
Вам нужно добавить ключ API Google, чтобы иметь возможность отображать карту события (например, не используя открытые карты улиц). В противном случае это самое простое решение.

@Ramast, подходит к той же проблеме более питоническим способом. Я включил изменения в свой код, и мне всегда приятно улучшаться.

Вот как мне удалось, в конце концов, сохранить отдельные поля для широты и долготы без необходимости сохранять их в базе данных, поскольку значения уже сохранены в PointField.

Идея заключается в следующем:

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

models.py

from django.contrib.gis.db import models as geomodels


class Entry(geomodels.Model):
    point = geomodels.PointField(
        srid=4326,
        blank=True,
        )

admin.py

from myapp.forms import EntryForm
from django.contrib import admin


class EntryAdmin(admin.ModelAdmin):
    form = EntryForm


admin.site.register(Entry, EntryAdmin)

forms.py

from django import forms
from myapp.models import Entry
from django.contrib.gis.geos import Point


class MarketEntryForm(forms.ModelForm):

    latitude = forms.FloatField(
        min_value=-90,
        max_value=90,
        required=True,
    )
    longitude = forms.FloatField(
        min_value=-180,
        max_value=180,
        required=True,
    )

    class Meta(object):
        model = MarketEntry
        exclude = []
        widgets = {'point': forms.HiddenInput()}

    def __init__(self, *args, **kwargs):
        super().__init__(*args, **kwargs)
        coordinates = self.initial.get('point', None)
        if isinstance(coordinates, Point):
            self.initial['longitude'], self.initial['latitude'] = coordinates.tuple

    def clean(self):
        data = super().clean()
        latitude = data.get('latitude')
        longitude = data.get('longitude')
        point = data.get('point')
        if latitude and longitude and not point:
            data['point'] = Point(longitude, latitude)
        return data

Ваш ответ на вопрос