tutorial.txt 27 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788
  1. ==================
  2. GeoDjango Tutorial
  3. ==================
  4. Introduction
  5. ============
  6. GeoDjango is an included contrib module for Django that turns it into a
  7. world-class geographic Web framework. GeoDjango strives to make it as simple
  8. as possible to create geographic Web applications, like location-based services.
  9. Its features include:
  10. * Django model fields for `OGC`_ geometries.
  11. * Extensions to Django's ORM for querying and manipulating spatial data.
  12. * Loosely-coupled, high-level Python interfaces for GIS geometry operations and
  13. data formats.
  14. * Editing geometry fields from the admin.
  15. This tutorial assumes familiarity with Django; thus, if you're brand new to
  16. Django, please read through the :doc:`regular tutorial </intro/tutorial01>` to
  17. familiarize yourself with Django first.
  18. .. note::
  19. GeoDjango has additional requirements beyond what Django requires --
  20. please consult the :ref:`installation documentation <ref-gis-install>`
  21. for more details.
  22. This tutorial will guide you through the creation of a geographic web
  23. application for viewing the `world borders`_. [#]_ Some of the code
  24. used in this tutorial is taken from and/or inspired by the `GeoDjango
  25. basic apps`_ project. [#]_
  26. .. note::
  27. Proceed through the tutorial sections sequentially for step-by-step
  28. instructions.
  29. .. _OGC: http://www.opengeospatial.org/
  30. .. _world borders: http://thematicmapping.org/downloads/world_borders.php
  31. .. _GeoDjango basic apps: http://code.google.com/p/geodjango-basic-apps/
  32. Setting Up
  33. ==========
  34. Create a Spatial Database
  35. -------------------------
  36. .. note::
  37. MySQL and Oracle users can skip this section because spatial types
  38. are already built into the database.
  39. First, create a spatial database for your project.
  40. If you are using PostGIS, create the database from the :ref:`spatial database
  41. template <spatialdb_template>`:
  42. .. code-block:: bash
  43. $ createdb -T template_postgis geodjango
  44. .. note::
  45. This command must be issued by a database user with enough privileges to
  46. create a database. To create a user with ``CREATE DATABASE`` privileges in
  47. PostgreSQL, use the following commands:
  48. .. code-block:: bash
  49. $ sudo su - postgres
  50. $ createuser --createdb geo
  51. $ exit
  52. Replace ``geo`` with your Postgres database user's username.
  53. (In PostgreSQL, this user will also be an OS-level user.)
  54. If you are using SQLite and SpatiaLite, consult the instructions on how
  55. to create a :ref:`SpatiaLite database <create_spatialite_db>`.
  56. Create a New Project
  57. ------------------------
  58. Use the standard ``django-admin.py`` script to create a project called
  59. ``geodjango``:
  60. .. code-block:: bash
  61. $ django-admin.py startproject geodjango
  62. This will initialize a new project. Now, create a ``world`` Django application
  63. within the ``geodjango`` project:
  64. .. code-block:: bash
  65. $ cd geodjango
  66. $ python manage.py startapp world
  67. Configure ``settings.py``
  68. -------------------------
  69. The ``geodjango`` project settings are stored in the ``geodjango/settings.py``
  70. file. Edit the database connection settings to match your setup::
  71. DATABASES = {
  72. 'default': {
  73. 'ENGINE': 'django.contrib.gis.db.backends.postgis',
  74. 'NAME': 'geodjango',
  75. 'USER': 'geo',
  76. }
  77. }
  78. In addition, modify the :setting:`INSTALLED_APPS` setting to include
  79. :mod:`django.contrib.admin`, :mod:`django.contrib.gis`,
  80. and ``world`` (your newly created application)::
  81. INSTALLED_APPS = (
  82. 'django.contrib.admin',
  83. 'django.contrib.auth',
  84. 'django.contrib.contenttypes',
  85. 'django.contrib.sessions',
  86. 'django.contrib.messages',
  87. 'django.contrib.staticfiles',
  88. 'django.contrib.gis',
  89. 'world'
  90. )
  91. Geographic Data
  92. ===============
  93. .. _worldborders:
  94. World Borders
  95. -------------
  96. The world borders data is available in this `zip file`__. Create a ``data``
  97. directory in the ``world`` application, download the world borders data, and
  98. unzip. On GNU/Linux platforms, use the following commands:
  99. .. code-block:: bash
  100. $ mkdir world/data
  101. $ cd world/data
  102. $ wget http://thematicmapping.org/downloads/TM_WORLD_BORDERS-0.3.zip
  103. $ unzip TM_WORLD_BORDERS-0.3.zip
  104. $ cd ../..
  105. The world borders ZIP file contains a set of data files collectively known as
  106. an `ESRI Shapefile`__, one of the most popular geospatial data formats. When
  107. unzipped, the world borders dataset includes files with the following
  108. extensions:
  109. * ``.shp``: Holds the vector data for the world borders geometries.
  110. * ``.shx``: Spatial index file for geometries stored in the ``.shp``.
  111. * ``.dbf``: Database file for holding non-geometric attribute data
  112. (e.g., integer and character fields).
  113. * ``.prj``: Contains the spatial reference information for the geographic
  114. data stored in the shapefile.
  115. __ http://thematicmapping.org/downloads/TM_WORLD_BORDERS-0.3.zip
  116. __ http://en.wikipedia.org/wiki/Shapefile
  117. Use ``ogrinfo`` to examine spatial data
  118. ---------------------------------------
  119. The GDAL ``ogrinfo`` utility allows examining the metadata of shapefiles or
  120. other vector data sources:
  121. .. code-block:: bash
  122. $ ogrinfo world/data/TM_WORLD_BORDERS-0.3.shp
  123. INFO: Open of `world/data/TM_WORLD_BORDERS-0.3.shp'
  124. using driver `ESRI Shapefile' successful.
  125. 1: TM_WORLD_BORDERS-0.3 (Polygon)
  126. ``ogrinfo`` tells us that the shapefile has one layer, and that this
  127. layer contains polygon data. To find out more, we'll specify the layer name
  128. and use the ``-so`` option to get only the important summary information:
  129. .. code-block:: bash
  130. $ ogrinfo -so world/data/TM_WORLD_BORDERS-0.3.shp TM_WORLD_BORDERS-0.3
  131. INFO: Open of `world/data/TM_WORLD_BORDERS-0.3.shp'
  132. using driver `ESRI Shapefile' successful.
  133. Layer name: TM_WORLD_BORDERS-0.3
  134. Geometry: Polygon
  135. Feature Count: 246
  136. Extent: (-180.000000, -90.000000) - (180.000000, 83.623596)
  137. Layer SRS WKT:
  138. GEOGCS["GCS_WGS_1984",
  139. DATUM["WGS_1984",
  140. SPHEROID["WGS_1984",6378137.0,298.257223563]],
  141. PRIMEM["Greenwich",0.0],
  142. UNIT["Degree",0.0174532925199433]]
  143. FIPS: String (2.0)
  144. ISO2: String (2.0)
  145. ISO3: String (3.0)
  146. UN: Integer (3.0)
  147. NAME: String (50.0)
  148. AREA: Integer (7.0)
  149. POP2005: Integer (10.0)
  150. REGION: Integer (3.0)
  151. SUBREGION: Integer (3.0)
  152. LON: Real (8.3)
  153. LAT: Real (7.3)
  154. This detailed summary information tells us the number of features in the layer
  155. (246), the geographic bounds of the data, the spatial reference system
  156. ("SRS WKT"), as well as type information for each attribute field. For example,
  157. ``FIPS: String (2.0)`` indicates that the ``FIPS`` character field has
  158. a maximum length of 2. Similarly, ``LON: Real (8.3)`` is a floating-point
  159. field that holds a maximum of 8 digits up to three decimal places.
  160. Geographic Models
  161. =================
  162. Defining a Geographic Model
  163. ---------------------------
  164. Now that you've examined your dataset using ``ogrinfo``, create a GeoDjango
  165. model to represent this data::
  166. from django.contrib.gis.db import models
  167. class WorldBorder(models.Model):
  168. # Regular Django fields corresponding to the attributes in the
  169. # world borders shapefile.
  170. name = models.CharField(max_length=50)
  171. area = models.IntegerField()
  172. pop2005 = models.IntegerField('Population 2005')
  173. fips = models.CharField('FIPS Code', max_length=2)
  174. iso2 = models.CharField('2 Digit ISO', max_length=2)
  175. iso3 = models.CharField('3 Digit ISO', max_length=3)
  176. un = models.IntegerField('United Nations Code')
  177. region = models.IntegerField('Region Code')
  178. subregion = models.IntegerField('Sub-Region Code')
  179. lon = models.FloatField()
  180. lat = models.FloatField()
  181. # GeoDjango-specific: a geometry field (MultiPolygonField), and
  182. # overriding the default manager with a GeoManager instance.
  183. mpoly = models.MultiPolygonField()
  184. objects = models.GeoManager()
  185. # Returns the string representation of the model.
  186. # On Python 3: def __str__(self):
  187. def __unicode__(self):
  188. return self.name
  189. Please note two important things:
  190. 1. The ``models`` module is imported from ``django.contrib.gis.db``.
  191. 2. You must override the model's default manager with
  192. :class:`~django.contrib.gis.db.models.GeoManager` to perform spatial queries.
  193. The default spatial reference system for geometry fields is WGS84 (meaning
  194. the `SRID`__ is 4326) -- in other words, the field coordinates are in
  195. longitude, latitude pairs in units of degrees. To use a different
  196. coordinate system, set the SRID of the geometry field with the ``srid``
  197. argument. Use an integer representing the coordinate system's EPSG code.
  198. __ http://en.wikipedia.org/wiki/SRID
  199. Run ``migrate``
  200. ---------------
  201. After defining your model, you need to sync it with the database. First,
  202. let's look at the SQL that will generate the table for the
  203. ``WorldBorder`` model::
  204. $ python manage.py sqlall world
  205. This command should produce the following output:
  206. .. code-block:: sql
  207. BEGIN;
  208. CREATE TABLE "world_worldborder" (
  209. "id" serial NOT NULL PRIMARY KEY,
  210. "name" varchar(50) NOT NULL,
  211. "area" integer NOT NULL,
  212. "pop2005" integer NOT NULL,
  213. "fips" varchar(2) NOT NULL,
  214. "iso2" varchar(2) NOT NULL,
  215. "iso3" varchar(3) NOT NULL,
  216. "un" integer NOT NULL,
  217. "region" integer NOT NULL,
  218. "subregion" integer NOT NULL,
  219. "lon" double precision NOT NULL,
  220. "lat" double precision NOT NULL
  221. )
  222. ;
  223. SELECT AddGeometryColumn('world_worldborder', 'mpoly', 4326, 'MULTIPOLYGON', 2);
  224. ALTER TABLE "world_worldborder" ALTER "mpoly" SET NOT NULL;
  225. CREATE INDEX "world_worldborder_mpoly_id" ON "world_worldborder" USING GIST ( "mpoly" GIST_GEOMETRY_OPS );
  226. COMMIT;
  227. If this looks correct, run :djadmin:`migrate` to create this table in the database::
  228. $ python manage.py migrate
  229. Creating table world_worldborder
  230. Installing custom SQL for world.WorldBorder model
  231. The :djadmin:`migrate` command may also prompt you to create an admin user.
  232. Either do so now, or later by running ``django-admin.py createsuperuser``.
  233. Importing Spatial Data
  234. ======================
  235. This section will show you how to import the world borders
  236. shapefile into the database via GeoDjango models using the
  237. :ref:`ref-layermapping`.
  238. There are many different ways to import data into a spatial database --
  239. besides the tools included within GeoDjango, you may also use the following:
  240. * `ogr2ogr`_: A command-line utility included with GDAL that
  241. can import many vector data formats into PostGIS, MySQL, and Oracle databases.
  242. * `shp2pgsql`_: This utility included with PostGIS imports ESRI shapefiles into
  243. PostGIS.
  244. .. _ogr2ogr: http://www.gdal.org/ogr2ogr.html
  245. .. _shp2pgsql: http://postgis.refractions.net/documentation/manual-1.5/ch04.html#shp2pgsql_usage
  246. .. _gdalinterface:
  247. GDAL Interface
  248. --------------
  249. Earlier, you used ``ogrinfo`` to examine the contents of the world borders
  250. shapefile. GeoDjango also includes a Pythonic interface to GDAL's powerful OGR
  251. library that can work with all the vector data sources that OGR supports.
  252. First, invoke the Django shell:
  253. .. code-block:: bash
  254. $ python manage.py shell
  255. If you downloaded the :ref:`worldborders` data earlier in the
  256. tutorial, then you can determine its path using Python's built-in
  257. ``os`` module::
  258. >>> import os
  259. >>> import world
  260. >>> world_shp = os.path.abspath(os.path.join(os.path.dirname(world.__file__),
  261. ... 'data/TM_WORLD_BORDERS-0.3.shp'))
  262. Now, open the world borders shapefile using GeoDjango's
  263. :class:`~django.contrib.gis.gdal.DataSource` interface::
  264. >>> from django.contrib.gis.gdal import DataSource
  265. >>> ds = DataSource(world_shp)
  266. >>> print(ds)
  267. / ... /geodjango/world/data/TM_WORLD_BORDERS-0.3.shp (ESRI Shapefile)
  268. Data source objects can have different layers of geospatial features; however,
  269. shapefiles are only allowed to have one layer::
  270. >>> print(len(ds))
  271. 1
  272. >>> lyr = ds[0]
  273. >>> print(lyr)
  274. TM_WORLD_BORDERS-0.3
  275. You can see the layer's geometry type and how many features it contains::
  276. >>> print(lyr.geom_type)
  277. Polygon
  278. >>> print(len(lyr))
  279. 246
  280. .. note::
  281. Unfortunately, the shapefile data format does not allow for greater
  282. specificity with regards to geometry types. This shapefile, like
  283. many others, actually includes ``MultiPolygon`` geometries, not Polygons.
  284. It's important to use a more general field type in models: a
  285. GeoDjango ``MultiPolygonField`` will accept a ``Polygon`` geometry, but a
  286. ``PolygonField`` will not accept a ``MultiPolygon`` type geometry. This
  287. is why the ``WorldBorder`` model defined above uses a ``MultiPolygonField``.
  288. The :class:`~django.contrib.gis.gdal.Layer` may also have a spatial reference
  289. system associated with it. If it does, the ``srs`` attribute will return a
  290. :class:`~django.contrib.gis.gdal.SpatialReference` object::
  291. >>> srs = lyr.srs
  292. >>> print(srs)
  293. GEOGCS["GCS_WGS_1984",
  294. DATUM["WGS_1984",
  295. SPHEROID["WGS_1984",6378137.0,298.257223563]],
  296. PRIMEM["Greenwich",0.0],
  297. UNIT["Degree",0.0174532925199433]]
  298. >>> srs.proj4 # PROJ.4 representation
  299. '+proj=longlat +ellps=WGS84 +datum=WGS84 +no_defs '
  300. This shapefile is in the popular WGS84 spatial reference
  301. system -- in other words, the data uses longitude, latitude pairs in
  302. units of degrees.
  303. In addition, shapefiles also support attribute fields that may contain
  304. additional data. Here are the fields on the World Borders layer:
  305. >>> print(lyr.fields)
  306. ['FIPS', 'ISO2', 'ISO3', 'UN', 'NAME', 'AREA', 'POP2005', 'REGION', 'SUBREGION', 'LON', 'LAT']
  307. The following code will let you examine the OGR types (e.g. integer or
  308. string) associated with each of the fields:
  309. >>> [fld.__name__ for fld in lyr.field_types]
  310. ['OFTString', 'OFTString', 'OFTString', 'OFTInteger', 'OFTString', 'OFTInteger', 'OFTInteger', 'OFTInteger', 'OFTInteger', 'OFTReal', 'OFTReal']
  311. You can iterate over each feature in the layer and extract information from both
  312. the feature's geometry (accessed via the ``geom`` attribute) as well as the
  313. feature's attribute fields (whose **values** are accessed via ``get()``
  314. method)::
  315. >>> for feat in lyr:
  316. ... print(feat.get('NAME'), feat.geom.num_points)
  317. ...
  318. Guernsey 18
  319. Jersey 26
  320. South Georgia South Sandwich Islands 338
  321. Taiwan 363
  322. :class:`~django.contrib.gis.gdal.Layer` objects may be sliced::
  323. >>> lyr[0:2]
  324. [<django.contrib.gis.gdal.feature.Feature object at 0x2f47690>, <django.contrib.gis.gdal.feature.Feature object at 0x2f47650>]
  325. And individual features may be retrieved by their feature ID::
  326. >>> feat = lyr[234]
  327. >>> print(feat.get('NAME'))
  328. San Marino
  329. Boundary geometries may be exported as WKT and GeoJSON::
  330. >>> geom = feat.geom
  331. >>> print(geom.wkt)
  332. POLYGON ((12.415798 43.957954,12.450554 ...
  333. >>> print(geom.json)
  334. { "type": "Polygon", "coordinates": [ [ [ 12.415798, 43.957954 ], [ 12.450554, 43.979721 ], ...
  335. ``LayerMapping``
  336. ----------------
  337. To import the data, use a LayerMapping in a Python script.
  338. Create a file called ``load.py`` inside the ``world`` application,
  339. with the following code::
  340. import os
  341. from django.contrib.gis.utils import LayerMapping
  342. from models import WorldBorder
  343. world_mapping = {
  344. 'fips' : 'FIPS',
  345. 'iso2' : 'ISO2',
  346. 'iso3' : 'ISO3',
  347. 'un' : 'UN',
  348. 'name' : 'NAME',
  349. 'area' : 'AREA',
  350. 'pop2005' : 'POP2005',
  351. 'region' : 'REGION',
  352. 'subregion' : 'SUBREGION',
  353. 'lon' : 'LON',
  354. 'lat' : 'LAT',
  355. 'mpoly' : 'MULTIPOLYGON',
  356. }
  357. world_shp = os.path.abspath(os.path.join(os.path.dirname(__file__), 'data/TM_WORLD_BORDERS-0.3.shp'))
  358. def run(verbose=True):
  359. lm = LayerMapping(WorldBorder, world_shp, world_mapping,
  360. transform=False, encoding='iso-8859-1')
  361. lm.save(strict=True, verbose=verbose)
  362. A few notes about what's going on:
  363. * Each key in the ``world_mapping`` dictionary corresponds to a field in the
  364. ``WorldBorder`` model. The value is the name of the shapefile field
  365. that data will be loaded from.
  366. * The key ``mpoly`` for the geometry field is ``MULTIPOLYGON``, the
  367. geometry type GeoDjango will import the field as. Even simple polygons in
  368. the shapefile will automatically be converted into collections prior to
  369. insertion into the database.
  370. * The path to the shapefile is not absolute -- in other words, if you move the
  371. ``world`` application (with ``data`` subdirectory) to a different location,
  372. the script will still work.
  373. * The ``transform`` keyword is set to ``False`` because the data in the
  374. shapefile does not need to be converted -- it's already in WGS84 (SRID=4326).
  375. * The ``encoding`` keyword is set to the character encoding of the string
  376. values in the shapefile. This ensures that string values are read and saved
  377. correctly from their original encoding system.
  378. Afterwards, invoke the Django shell from the ``geodjango`` project directory:
  379. .. code-block:: bash
  380. $ python manage.py shell
  381. Next, import the ``load`` module, call the ``run`` routine, and watch
  382. ``LayerMapping`` do the work::
  383. >>> from world import load
  384. >>> load.run()
  385. .. _ogrinspect-intro:
  386. Try ``ogrinspect``
  387. ------------------
  388. Now that you've seen how to define geographic models and import data with the
  389. :ref:`ref-layermapping`, it's possible to further automate this process with
  390. use of the :djadmin:`ogrinspect` management command. The :djadmin:`ogrinspect`
  391. command introspects a GDAL-supported vector data source (e.g., a shapefile)
  392. and generates a model definition and ``LayerMapping`` dictionary automatically.
  393. The general usage of the command goes as follows:
  394. .. code-block:: bash
  395. $ python manage.py ogrinspect [options] <data_source> <model_name> [options]
  396. ``data_source`` is the path to the GDAL-supported data source and
  397. ``model_name`` is the name to use for the model. Command-line options may
  398. be used to further define how the model is generated.
  399. For example, the following command nearly reproduces the ``WorldBorder`` model
  400. and mapping dictionary created above, automatically:
  401. .. code-block:: bash
  402. $ python manage.py ogrinspect world/data/TM_WORLD_BORDERS-0.3.shp WorldBorder \
  403. --srid=4326 --mapping --multi
  404. A few notes about the command-line options given above:
  405. * The ``--srid=4326`` option sets the SRID for the geographic field.
  406. * The ``--mapping`` option tells ``ogrinspect`` to also generate a
  407. mapping dictionary for use with
  408. :class:`~django.contrib.gis.utils.LayerMapping`.
  409. * The ``--multi`` option is specified so that the geographic field is a
  410. :class:`~django.contrib.gis.db.models.MultiPolygonField` instead of just a
  411. :class:`~django.contrib.gis.db.models.PolygonField`.
  412. The command produces the following output, which may be copied
  413. directly into the ``models.py`` of a GeoDjango application::
  414. # This is an auto-generated Django model module created by ogrinspect.
  415. from django.contrib.gis.db import models
  416. class WorldBorder(models.Model):
  417. fips = models.CharField(max_length=2)
  418. iso2 = models.CharField(max_length=2)
  419. iso3 = models.CharField(max_length=3)
  420. un = models.IntegerField()
  421. name = models.CharField(max_length=50)
  422. area = models.IntegerField()
  423. pop2005 = models.IntegerField()
  424. region = models.IntegerField()
  425. subregion = models.IntegerField()
  426. lon = models.FloatField()
  427. lat = models.FloatField()
  428. geom = models.MultiPolygonField(srid=4326)
  429. objects = models.GeoManager()
  430. # Auto-generated `LayerMapping` dictionary for WorldBorder model
  431. worldborders_mapping = {
  432. 'fips' : 'FIPS',
  433. 'iso2' : 'ISO2',
  434. 'iso3' : 'ISO3',
  435. 'un' : 'UN',
  436. 'name' : 'NAME',
  437. 'area' : 'AREA',
  438. 'pop2005' : 'POP2005',
  439. 'region' : 'REGION',
  440. 'subregion' : 'SUBREGION',
  441. 'lon' : 'LON',
  442. 'lat' : 'LAT',
  443. 'geom' : 'MULTIPOLYGON',
  444. }
  445. Spatial Queries
  446. ===============
  447. Spatial Lookups
  448. ---------------
  449. GeoDjango adds spatial lookups to the Django ORM. For example, you
  450. can find the country in the ``WorldBorder`` table that contains
  451. a particular point. First, fire up the management shell:
  452. .. code-block:: bash
  453. $ python manage.py shell
  454. Now, define a point of interest [#]_::
  455. >>> pnt_wkt = 'POINT(-95.3385 29.7245)'
  456. The ``pnt_wkt`` string represents the point at -95.3385 degrees longitude,
  457. 29.7245 degrees latitude. The geometry is in a format known as
  458. Well Known Text (WKT), a standard issued by the Open Geospatial
  459. Consortium (OGC). [#]_ Import the ``WorldBorder`` model, and perform
  460. a ``contains`` lookup using the ``pnt_wkt`` as the parameter::
  461. >>> from world.models import WorldBorder
  462. >>> qs = WorldBorder.objects.filter(mpoly__contains=pnt_wkt)
  463. >>> qs
  464. [<WorldBorder: United States>]
  465. Here, you retrieved a ``GeoQuerySet`` with only one model: the border of
  466. the United States (exactly what you would expect).
  467. Similarly, you may also use a :ref:`GEOS geometry object <ref-geos>`.
  468. Here, you can combine the ``intersects`` spatial lookup with the ``get``
  469. method to retrieve only the ``WorldBorder`` instance for San Marino instead
  470. of a queryset::
  471. >>> from django.contrib.gis.geos import Point
  472. >>> pnt = Point(12.4604, 43.9420)
  473. >>> sm = WorldBorder.objects.get(mpoly__intersects=pnt)
  474. >>> sm
  475. <WorldBorder: San Marino>
  476. The ``contains`` and ``intersects`` lookups are just a subset of the
  477. available queries -- the :ref:`ref-gis-db-api` documentation has more.
  478. Automatic Spatial Transformations
  479. ---------------------------------
  480. When doing spatial queries, GeoDjango automatically transforms
  481. geometries if they're in a different coordinate system. In the following
  482. example, coordinates will be expressed in `EPSG SRID 32140`__,
  483. a coordinate system specific to south Texas **only** and in units of
  484. **meters**, not degrees::
  485. >>> from django.contrib.gis.geos import Point, GEOSGeometry
  486. >>> pnt = Point(954158.1, 4215137.1, srid=32140)
  487. Note that ``pnt`` may also be constructed with EWKT, an "extended" form of
  488. WKT that includes the SRID::
  489. >>> pnt = GEOSGeometry('SRID=32140;POINT(954158.1 4215137.1)')
  490. GeoDjango's ORM will automatically wrap geometry values
  491. in transformation SQL, allowing the developer to work at a higher level
  492. of abstraction::
  493. >>> qs = WorldBorder.objects.filter(mpoly__intersects=pnt)
  494. >>> print(qs.query) # Generating the SQL
  495. SELECT "world_worldborder"."id", "world_worldborder"."name", "world_worldborder"."area",
  496. "world_worldborder"."pop2005", "world_worldborder"."fips", "world_worldborder"."iso2",
  497. "world_worldborder"."iso3", "world_worldborder"."un", "world_worldborder"."region",
  498. "world_worldborder"."subregion", "world_worldborder"."lon", "world_worldborder"."lat",
  499. "world_worldborder"."mpoly" FROM "world_worldborder"
  500. WHERE ST_Intersects("world_worldborder"."mpoly", ST_Transform(%s, 4326))
  501. >>> qs # printing evaluates the queryset
  502. [<WorldBorder: United States>]
  503. __ http://spatialreference.org/ref/epsg/32140/
  504. .. admonition:: Raw queries
  505. When using :doc:`raw queries </topics/db/sql>`, you should generally wrap
  506. your geometry fields with the ``asText()`` SQL function (or ``ST_AsText``
  507. for PostGIS) so that the field value will be recognized by GEOS::
  508. City.objects.raw('SELECT id, name, asText(point) from myapp_city')
  509. This is not absolutely required by PostGIS, but generally you should only
  510. use raw queries when you know exactly what you are doing.
  511. Lazy Geometries
  512. ---------------
  513. GeoDjango loads geometries in a standardized textual representation. When the
  514. geometry field is first accessed, GeoDjango creates a `GEOS geometry object
  515. <ref-geos>`, exposing powerful functionality, such as serialization properties
  516. for popular geospatial formats::
  517. >>> sm = WorldBorder.objects.get(name='San Marino')
  518. >>> sm.mpoly
  519. <MultiPolygon object at 0x24c6798>
  520. >>> sm.mpoly.wkt # WKT
  521. MULTIPOLYGON (((12.4157980000000006 43.9579540000000009, 12.4505540000000003 43.9797209999999978, ...
  522. >>> sm.mpoly.wkb # WKB (as Python binary buffer)
  523. <read-only buffer for 0x1fe2c70, size -1, offset 0 at 0x2564c40>
  524. >>> sm.mpoly.geojson # GeoJSON (requires GDAL)
  525. '{ "type": "MultiPolygon", "coordinates": [ [ [ [ 12.415798, 43.957954 ], [ 12.450554, 43.979721 ], ...
  526. This includes access to all of the advanced geometric operations provided by
  527. the GEOS library::
  528. >>> pnt = Point(12.4604, 43.9420)
  529. >>> sm.mpoly.contains(pnt)
  530. True
  531. >>> pnt.contains(sm.mpoly)
  532. False
  533. ``GeoQuerySet`` Methods
  534. -----------------------
  535. Putting your data on the map
  536. ============================
  537. Geographic Admin
  538. ----------------
  539. GeoDjango extends :doc:`Django's admin application </ref/contrib/admin/index>`
  540. with support for editing geometry fields.
  541. Basics
  542. ^^^^^^
  543. GeoDjango also supplements the Django admin by allowing users to create
  544. and modify geometries on a JavaScript slippy map (powered by `OpenLayers`_).
  545. Let's dive right in. Create a file called ``admin.py`` inside the
  546. ``world`` application with the following code::
  547. from django.contrib.gis import admin
  548. from models import WorldBorder
  549. admin.site.register(WorldBorder, admin.GeoModelAdmin)
  550. Next, edit your ``urls.py`` in the ``geodjango`` application folder as follows::
  551. from django.conf.urls import patterns, url, include
  552. from django.contrib.gis import admin
  553. admin.autodiscover()
  554. urlpatterns = patterns('',
  555. url(r'^admin/', include(admin.site.urls)),
  556. )
  557. Start up the Django development server:
  558. .. code-block:: bash
  559. $ python manage.py runserver
  560. Finally, browse to ``http://localhost:8000/admin/``, and log in with the admin
  561. user created after running :djadmin:`migrate`. Browse to any of the ``WorldBorder``
  562. entries -- the borders may be edited by clicking on a polygon and dragging
  563. the vertexes to the desired position.
  564. .. _OpenLayers: http://openlayers.org/
  565. .. _Open Street Map: http://openstreetmap.org/
  566. .. _Vector Map Level 0: http://earth-info.nga.mil/publications/vmap0.html
  567. .. _OSGeo: http://www.osgeo.org
  568. .. _osmgeoadmin-intro:
  569. ``OSMGeoAdmin``
  570. ^^^^^^^^^^^^^^^
  571. With the :class:`~django.contrib.gis.admin.OSMGeoAdmin`, GeoDjango uses
  572. a `Open Street Map`_ layer in the admin.
  573. This provides more context (including street and thoroughfare details) than
  574. available with the :class:`~django.contrib.gis.admin.GeoModelAdmin`
  575. (which uses the `Vector Map Level 0`_ WMS dataset hosted at `OSGeo`_).
  576. First, there are some important requirements:
  577. * :class:`~django.contrib.gis.admin.OSMGeoAdmin` requires that the
  578. :ref:`spherical mercator projection be added <addgoogleprojection>`
  579. to the ``spatial_ref_sys`` table (PostGIS 1.3 and below, only).
  580. * The PROJ.4 datum shifting files must be installed (see the
  581. :ref:`PROJ.4 installation instructions <proj4>` for more details).
  582. If you meet these requirements, then just substitute the ``OSMGeoAdmin``
  583. option class in your ``admin.py`` file::
  584. admin.site.register(WorldBorder, admin.OSMGeoAdmin)
  585. .. rubric:: Footnotes
  586. .. [#] Special thanks to Bjørn Sandvik of `thematicmapping.org
  587. <http://thematicmapping.org>`_ for providing and maintaining this
  588. dataset.
  589. .. [#] GeoDjango basic apps was written by Dane Springmeyer, Josh Livni, and
  590. Christopher Schmidt.
  591. .. [#] This point is the `University of Houston Law Center
  592. <http://www.law.uh.edu/>`_.
  593. .. [#] Open Geospatial Consortium, Inc., `OpenGIS Simple Feature Specification
  594. For SQL <http://www.opengeospatial.org/standards/sfs>`_.