Revise overlaps() function to be find() function, support searching by string alone.

This commit is contained in:
Schuyler Erle 2011-08-30 17:43:00 -07:00
parent 6c7fee843c
commit a9c5064c95
2 changed files with 31 additions and 17 deletions

View File

@ -24,16 +24,23 @@ class FeatureSearchManager(models.GeoManager):
cursor = connection.cursor() cursor = connection.cursor()
cursor.execute("""SELECT set_limit(%f)""" % threshold) cursor.execute("""SELECT set_limit(%f)""" % threshold)
def overlaps(self, (minx, miny, maxx, maxy), text=None, srid=4326): def find(self, bbox=None, text=None, adm1=None, adm2=None srid=4326):
qset = self.get_query_set()
if bbox:
(minx, miny, maxx, maxy) = bbox
bbox = Polygon(((minx,miny),(minx,maxy),(maxx,maxy),(maxx,miny),(minx,miny)),srid=srid) bbox = Polygon(((minx,miny),(minx,maxy),(maxx,maxy),(maxx,miny),(minx,miny)),srid=srid)
if srid != 4326: bbox.transform(4326) # convert to lon/lat if srid != 4326: bbox.transform(4326) # convert to lon/lat
qset = models.GeoManager.get_query_set(self).filter(geometry__bboverlaps=bbox) qset = qset.filter(geometry__bboverlaps=bbox)
if text: if text:
text = text.replace("'", "''") # escape the ' text = text.replace("'", "''") # escape the '
# use the pg_trgm index # use the pg_trgm index
qset = qset.extra(select={"similarity":"similarity(preferred_name, '%s')" % text}, qset = qset.extra(select={"similarity":"similarity(preferred_name, '%s')"},
where=["preferred_name %%%% '%s'" % text], select_params=[text],
where=["preferred_name %% '%s'"],
params=[text],
order_by=["-similarity"]) order_by=["-similarity"])
if adm1: qset = qset.filter(admin1__exact=adm1)
if adm2: qset = qset.filter(admin2__exact=adm2)
return qset return qset
class Feature(models.Model): class Feature(models.Model):
@ -41,8 +48,8 @@ class Feature(models.Model):
url = models.CharField(max_length=512, unique=True, verbose_name="URI") url = models.CharField(max_length=512, unique=True, verbose_name="URI")
preferred_name = models.CharField(max_length=512) preferred_name = models.CharField(max_length=512)
feature_type = models.ForeignKey("FeatureType", null=True, blank=True) feature_type = models.ForeignKey("FeatureType", null=True, blank=True)
admin1 = models.CharField(max_length=2, blank=True, verbose_name="State") admin1 = models.CharField(max_length=2, blank=True, verbose_name="State", db_index=True)
admin2 = models.CharField(max_length=255, blank=True, verbose_name="County") admin2 = models.CharField(max_length=255, blank=True, verbose_name="County", db_index=True)
geometry = models.GeometryField() geometry = models.GeometryField()
is_primary = models.BooleanField(default=True) is_primary = models.BooleanField(default=True)
time_frame = models.ForeignKey("TimeFrame", null=True, blank=True) time_frame = models.ForeignKey("TimeFrame", null=True, blank=True)
@ -60,8 +67,8 @@ class Feature(models.Model):
return "%s: %s" % (self.feature_type.code, self.feature_type.name,) return "%s: %s" % (self.feature_type.code, self.feature_type.name,)
feature_type_name.short_description = "Feature Type" feature_type_name.short_description = "Feature Type"
def get_geojson(self): def get_geojson(self, srid=4326):
geom = json.loads(self.geometry.transform(3785, True).geojson) geom = json.loads(self.geometry.transform(srid, True).geojson)
if self.feature_type is None: if self.feature_type is None:
feature_type = '' feature_type = ''

View File

@ -11,14 +11,21 @@ def search(request):
def search_json(request): def search_json(request):
search_term = request.GET.get("search", "") search_term = request.GET.get("search", "")
bbox = request.GET.get("bbox", False) bbox = request.GET.get("bbox", False)
if not bbox: country = request.GET.get("adm0", "US") # right now, unused
return render_to_json_response({'error': 'no bbox parameter'}) adm1 = request.GET.get("adm1", "")
adm2 = request.GET.get("adm2", "")
srid = int(request.GET.get("srid", 4326))
bs = bbox.split(",") if bbox:
bs_param = (float(bs[0]), float(bs[1]), float(bs[2]), float(bs[3]),) try:
bbox = map(float, bbox.split(","))
except ValueError:
bbox = None
if not bbox and not search_term:
return render_to_json_response({'error': 'must supply either a valid `bbox` or a `search` parameter'})
features_qset = Feature.search.overlaps(bs_param, text=search_term, srid=3785)[0:20] features_qset = Feature.search.find(bbox=bbox, text=search_term, adm1=adm1, adm2=adm2, srid=srid)[0:20]
features = [f.get_geojson() for f in features_qset] features = [f.get_geojson(srid) for f in features_qset]
d = { d = {
'type': 'FeatureCollection', 'type': 'FeatureCollection',
'features': features 'features': features