views.py 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127
  1. from __future__ import unicode_literals
  2. import warnings
  3. from django.apps import apps
  4. from django.http import HttpResponse, Http404
  5. from django.template import loader
  6. from django.contrib.sites.shortcuts import get_current_site
  7. from django.core import urlresolvers
  8. from django.core.paginator import EmptyPage, PageNotAnInteger
  9. from django.contrib.gis.db.models.fields import GeometryField
  10. from django.db import connections, DEFAULT_DB_ALIAS
  11. from django.db.models.fields import FieldDoesNotExist
  12. from django.utils import six
  13. from django.utils.deprecation import RemovedInDjango18Warning
  14. from django.utils.translation import ugettext as _
  15. from django.contrib.gis.shortcuts import render_to_kml, render_to_kmz
  16. def index(request, sitemaps):
  17. """
  18. This view generates a sitemap index that uses the proper view
  19. for resolving geographic section sitemap URLs.
  20. """
  21. warnings.warn("Geo Sitemaps are deprecated. Use plain sitemaps from "
  22. "django.contrib.sitemaps instead", RemovedInDjango18Warning, stacklevel=2)
  23. current_site = get_current_site(request)
  24. sites = []
  25. protocol = request.scheme
  26. for section, site in sitemaps.items():
  27. if callable(site):
  28. pages = site().paginator.num_pages
  29. else:
  30. pages = site.paginator.num_pages
  31. sitemap_url = urlresolvers.reverse('django.contrib.gis.sitemaps.views.sitemap', kwargs={'section': section})
  32. sites.append('%s://%s%s' % (protocol, current_site.domain, sitemap_url))
  33. if pages > 1:
  34. for page in range(2, pages + 1):
  35. sites.append('%s://%s%s?p=%s' % (protocol, current_site.domain, sitemap_url, page))
  36. xml = loader.render_to_string('sitemap_index.xml', {'sitemaps': sites})
  37. return HttpResponse(xml, content_type='application/xml')
  38. def sitemap(request, sitemaps, section=None):
  39. """
  40. This view generates a sitemap with additional geographic
  41. elements defined by Google.
  42. """
  43. warnings.warn("Geo Sitemaps are deprecated. Use plain sitemaps from "
  44. "django.contrib.sitemaps instead", RemovedInDjango18Warning, stacklevel=2)
  45. maps, urls = [], []
  46. if section is not None:
  47. if section not in sitemaps:
  48. raise Http404(_("No sitemap available for section: %r") % section)
  49. maps.append(sitemaps[section])
  50. else:
  51. maps = list(six.itervalues(sitemaps))
  52. page = request.GET.get("p", 1)
  53. current_site = get_current_site(request)
  54. for site in maps:
  55. try:
  56. if callable(site):
  57. urls.extend(site().get_urls(page=page, site=current_site))
  58. else:
  59. urls.extend(site.get_urls(page=page, site=current_site))
  60. except EmptyPage:
  61. raise Http404(_("Page %s empty") % page)
  62. except PageNotAnInteger:
  63. raise Http404(_("No page '%s'") % page)
  64. xml = loader.render_to_string('gis/sitemaps/geo_sitemap.xml', {'urlset': urls})
  65. return HttpResponse(xml, content_type='application/xml')
  66. def kml(request, label, model, field_name=None, compress=False, using=DEFAULT_DB_ALIAS):
  67. """
  68. This view generates KML for the given app label, model, and field name.
  69. The model's default manager must be GeoManager, and the field name
  70. must be that of a geographic field.
  71. """
  72. placemarks = []
  73. try:
  74. klass = apps.get_model(label, model)
  75. except LookupError:
  76. raise Http404('You must supply a valid app label and module name. Got "%s.%s"' % (label, model))
  77. if field_name:
  78. try:
  79. field, _, _, _ = klass._meta.get_field_by_name(field_name)
  80. if not isinstance(field, GeometryField):
  81. raise FieldDoesNotExist
  82. except FieldDoesNotExist:
  83. raise Http404('Invalid geometry field.')
  84. connection = connections[using]
  85. if connection.ops.postgis:
  86. # PostGIS will take care of transformation.
  87. placemarks = klass._default_manager.using(using).kml(field_name=field_name)
  88. else:
  89. # There's no KML method on Oracle or MySQL, so we use the `kml`
  90. # attribute of the lazy geometry instead.
  91. placemarks = []
  92. if connection.ops.oracle:
  93. qs = klass._default_manager.using(using).transform(4326, field_name=field_name)
  94. else:
  95. qs = klass._default_manager.using(using).all()
  96. for mod in qs:
  97. mod.kml = getattr(mod, field_name).kml
  98. placemarks.append(mod)
  99. # Getting the render function and rendering to the correct.
  100. if compress:
  101. render = render_to_kmz
  102. else:
  103. render = render_to_kml
  104. return render('gis/kml/placemarks.kml', {'places': placemarks})
  105. def kmz(request, label, model, field_name=None, using=DEFAULT_DB_ALIAS):
  106. """
  107. This view returns KMZ for the given app label, model, and field name.
  108. """
  109. return kml(request, label, model, field_name, compress=True, using=using)