2010-05-02 21 views
7

Próbuję owinąć głowę wokół formularzy django i sposobu działania django. Chcę utworzyć podstawowy formularz internetowy, który pozwala użytkownikowi wprowadzić adres i mieć ten geokodowany i zapisany w bazie danych.Geokodowanie adresu przy składaniu formularza?

stworzyłem model Lokalizacja:

class Location(models.Model): 
    address = models.CharField(max_length=200) 
    city = models.CharField(max_length=100) 
    state = models.CharField(max_length=100, null=True) 
    postal_code = models.CharField(max_length=100, null=True) 
    country = models.CharField(max_length=100) 
    latitude = models.DecimalField(max_digits=18, decimal_places=10, null=True) 
    longitude = models.DecimalField(max_digits=18, decimal_places=10, null=True) 

i określone formy:

class LocationForm(forms.ModelForm): 
    class Meta: 
     model = models.Location 
     exclude = ('latitude','longitude') 

Moim zdaniem Używam form.save() aby zachować formę. To działa i zapisuje adres w bazie danych.

Stworzyłem moduł do geokodowania adresu. Nie jestem pewien, jaki jest sposób działania django, ale według mnie, zanim zapiszę formularz, muszę geokodować adres i ustawić czas i długość. Jak ustawić szerokość i długość geograficzną przed zapisaniem?

Odpowiedz

8

Można zastąpić metodę zapisu modelu. Przed zapisaniem geokoduję dane. Używa się googleapi, ale można go odpowiednio zmodyfikować.

import urllib 

def save(self): 
    location = "%s, %s, %s, %s" % (self.address, self.city, self.state, self.zip) 

    if not self.latitude or not self.longitude: 
     latlng = self.geocode(location) 
     latlng = latlng.split(',') 
     self.latitude = latlng[0] 
     self.longitude = latlng[1] 

    super(Marker, self).save() 

def geocode(self, location): 
    output = "csv" 
    location = urllib.quote_plus(location) 
    request = "http://maps.google.com/maps/geo?q=%s&output=%s&key=%s" % (location, output, settings.GOOGLE_API_KEY) 
    data = urllib.urlopen(request).read() 
    dlist = data.split(',') 
    if dlist[0] == '200': 
     return "%s,%s" % (dlist[2], dlist[3]) 
    else: 
     return ',' 
+0

Wielkie dzięki, wypróbuję to i skontaktuję się z Tobą. Jakie jest twoje zdanie na temat geocode() zwracającej krotkę lub obiekt współrzędnych zamiast ciągu? – User

+0

To świetny pomysł. Byłoby czystsze, aby lepiej spakować dane przed powrotem. Generalnie lubię robić takie rzeczy. Zostawiłem to w ten sposób, ponieważ odtworzyłem oryginalne równanie. Myślałem, że CSV jest po prostu dlatego, że tak powraca z google, ale podoba mi się, jak myślisz. Zamierzam to zmienić, aby zwrócić obiekt. – digitaldreamer

+0

Aktualizacja aplikacji Google Maps API v3 poniżej. –

0

robię to samo jak ten może być cis szukasz czegoś similarstrong tekstu

# forms.py 
from django.contrib.gis.utils import GeoIP 


class registerForm(forms.ModelForm): 
class Meta: 
    model=register 
    fields = ('Availability', 'Status') 

def save(self,ip_address, *args, **kwargs): 
    g = GeoIP() 
    lat, lon = g.lat_lon(ip_address) 
    user_location = super(registerForm, self).save(commit=False) 
    user_location.latitude = lat 
    user_location.longitude = lon 
    user_location.save(*args, **kwargs) 

i moich poglądów

#views.py 
    if request.method == "POST": 
ip_address=request.META['REMOTE_ADDR'] 
    rform = registerForm(data = request.POST) 
    if rform.is_valid(): 
    register = rform.save(commit=False) 
    register.user=request.user 
    register.save(ip_address) 
     return render_to_response('home.html') 
else: 
    rform = registerForm() 
return render_to_response('status_set.html',{'rform':rform}) 
3

aktualizacji dla Google Maps API v3:

import json 
import urllib.parse, 
from decimal import Decimal 

def save(self): 
    if not self.lat or not self.lng: 
     self.lat, self.lng = self.geocode(self.address) 

    super(Location, self).save() 

def geocode(self, address): 
    address = urllib.parse.quote_plus(address) 
    maps_api_url = "?".join([ 
     "http://maps.googleapis.com/maps/api/geocode/json", 
     urllib.parse.urlencode({"address"=address, "sensor"=False}) 
    ]) 
    response = urllib.urlopen(maps_api_url) 
    data = json.loads(response.read().decode('utf8')) 

    if data['status'] == 'OK': 
     lat = data['results'][0]['geometry']['location']['lat'] 
     lng = data['results'][0]['geometry']['location']['lng'] 
     return Decimal(lat), Decimal(lng) 
0

Podobna do innych odpowiedzi:

def geocode(address, city, state, zip_code): 
    try: 
     location_param = urllib.request.quote("%s, %s, %s, %s" % (address, city, state, zip_code)) 
     url_request = "http://maps.googleapis.com/maps/api/geocode/json?address=%s&sensor=false" % location_param 
     result = requests.get(url_request) 
     data = result.json() 
     location = data['results'][0]['geometry']['location'] 
     lat = location['lat'] 
     lng = location['lng'] 
     return lat, lng 
    except Exception: 
     return None 
Powiązane problemy