Книга: JavaScript. Подробное руководство, 6-е издание

22.1. Геопозиционирование

22.1. Геопозиционирование

Прикладной интерфейс объекта Geolocationhttp://www.iv3.org/TR/geolocation-API/) позволяет программам на языке JavaScript запрашивать у броузера географическое местонахождение пользователя. Такие приложения могут отображать карты, маршруты и другую информацию, связанную с текущим местонахождением пользователя. При этом, конечно, возникает важная проблема соблюдения тайны частной информации, поэтому броузеры, поддерживающие прикладной интерфейс Geolocation, всегда запрашивают у пользователя подтверждение, прежде чем передать JavaScript-программе информацию о физическом местонахождении пользователя.

Броузеры с поддержкой интерфейса Geolocation определяют свойство navigator.geolocation. Это свойство ссылается на объект с тремя методами:

navigator.geolocation.getCurrentPosition()

Запрашивает текущее географическое местонахождение пользователя.

navigator.geolocation.watchPosition()

Не только запрашивает текущее местонахождение, но и продолжает следить за координатами, вызывая указанную функцию обратного вызова при изменении местонахождения пользователя.

navigator.geolocation.clearWatch()

Останавливает слежение за местонахождением пользователя. В аргументе этому методу следует передавать число, возвращаемое соответствующим вызовом метода watchPosition().

В устройствах, включающих аппаратную поддержку GPS, имеется возможность определять местонахождение с высокой степенью точности с помощью устройства GPS. Однако чаще информация о местонахождении поступает из Всемирной паутины. Если броузер отправит ІР-адрес специализированной веб-службе, она в большинстве случаев сможет определить (на основе информации о поставщиках услуг Интернета), в каком городе находится пользователь (и рекламодатели часто пользуются этой возможностью, реализуя определение местонахождения на стороне сервера). Броузер часто в состоянии получить еще более точную информацию о местонахождении, запросив у операционной системы список ближайших беспроводных сетей и силы их сигналов. Затем эта информация отправляется веб-службе, которая позволяет вычислить местонахождение с большой точностью (обычно с точностью до микрорайона в городе).

Эти технологии определения географического местонахождения связаны либо с обменом данными по сети, либо с взаимодействием с несколькими спутниками, поэтому прикладной интерфейс объекта Geolocation является асинхронным: методы getCurrentPosition() и watchPosition() возвращают управление немедленно, но они принимают функцию, которая будет вызвана броузером, когда он определит местонахождение пользователя (или когда местонахождение изменится). В простейшем случае запрос местонахождения выглядит так:

navigator.geolocation.getCurrentPosition(function(pos) {
  var latitude = pos.coords.latitude;
  var longitude = pos.coords.longitude;
  alert("Ваши координаты: " + latitude + ", + longitude);
});

В дополнение к широте и долготе в ответ на каждый успешный запрос возвращается также значение (в метрах), указывающее точность определения местонахождения. Пример 22.1 демонстрирует получение информации о местонахождении: он вызывает метод getCurrentPosition(), чтобы определить текущее местонахождение, и использует полученную информацию для отображения карты (полученной от службы Google Maps) текущего местонахождения в масштабе, примерно соответствующем точности определения местонахождения.

Пример 22.1. Использование информации о местонахождении для отображения карты

// Возвращает вновь созданный элемент <img>, настроенный (в случае успешного
// определения местонахождения) на отображение карты для текущего местонахождения.
// Обратите внимание, что вызывающая программа сама должна вставить возвращаемый
// элемент в документ, чтобы отобразить его. Возбуждает исключение, если возможность
// определения местонахождения не поддерживается броузером,
function getmap() {
  // Проверить поддержку объекта geolocation
  if (!navigator.geolocation)
    throw "Определение местонахождения не поддерживается";
  // Создать новый элемент <img>, отправить запрос определения местонахождения,
  // чтобы в img отобразить карту местонахождения и вернуть изображение,
  var image = document.createElement("img");
  navigator.geolocation.getCurrentPosition(setMapURL);
  return image;
  // Эта функция будет вызвана после того, как вызывающая программа получит объект
  // изображения, в случае успешного выполнения запроса определения местонахождения,
  function setMapURL(pos) {
    // Получить информацию о местонахождении из объекта аргумента
    var latitude = pos.coords.latitude; // Градусы к северу от экватора
    var longitude = pos.coords.longitude; // Градусы к востоку от Гринвича
    var accuracy = pos.coords.accuracy; // Метры
    // Сконструировать URL для получения статического изображения карты
    // от службы Google Мар для этого местонахождения
    var url = "http://maps.google.com/maps/api/staticmap" +
      "?center=" + latitude + + longitude +
      "&size=640x640&sensor=t rue";
    // Установить масштаб карты, используя грубое приближение
    var zoomlevel=20; // Для начала установить самый крупный масштаб
    if (accuracy > 80) // Уменьшить масштаб для низкой точности
      zoomlevel -= Math.round(Math.log(accuracy/50)/Math.LN2);
    url += "&zoom=" + zoomlevel; // Добавить масштаб в URL
    // Отобразить карту в объекте изображения. Спасибо, Google!
    image.src = url;
  }
}

Прикладной интерфейс Geolocation обладает несколькими особенностями, которые не были продемонстрированы в примере 22.1:

• В дополнение к первому аргументу с функцией обратного вызова методы getCurrentPosition() и watchPosition() принимают вторую необязательную функцию, которая будет вызвана в случае неудачного выполнения запроса.

• Помимо функций обработчиков успешного и неудачного выполнения запроса эти два метода принимают в третьем необязательном аргументе объект с параметрами. Свойства этого объекта определяют: желательна ли высокая точность определения местонахождения, насколько «устаревшей» может быть информация о местонахождении и предельное время ожидания определения местонахождения.

• Объект, который передается обработчику в случае успешного выполнения запроса, также включает время и может (на некоторых устройствах) содержать дополнительную информацию, такую как высота над уровнем моря, скорость и направление перемещения.

Эти дополнительные возможности демонстрируются в примере 22.2.

Пример 22.2. Демонстрация всех возможностей определения местонахождения

// Асинхронно определяет местонахождение и отображает его в указанном элементе,
function whereami(elt) {
  // Этот объект передается методу getCurrentPosition() в 3 аргументе
  var options = {
    // Чтобы получить координаты с высокой точностью (например, с устройства GPS),
    // присвойте этому свойству значение true. Отметьте, однако, что это может
    // увеличить расход энергии в аккумуляторах.
    enableHighAccuracy: false, // Приблизительно: по умолчанию
    // Определите свое значение, если допустимо брать координаты из кэша.
    // По умолчанию имеет значение 0, что обеспечивает получение самой
    // свежей информации.
    maximumAge: 300000, // Пригодна информация, полученная в течение последних 5 минут
    // Предельное время ожидания выполнения запроса.
    // По умолчанию имеет значение Infinity, что соответствует бесконечному
    // времени ожидания выполнения запроса вызовом метода getCurrentPosition()
    timeout: 15000 // Ждать не более 15 секунд
  };
  if (navigator.geolocation) // Запросить координаты, если возможно
    navigator.geolocation.getCurrentPosition(success, error, options):
  else
    elt.innerHTML = "Возможность определения местонахождения " +
                    "не поддерживается этим броузером":
  // Эта функция будет вызвана в случае неудачного выполнения запроса
  function error(e) {
    // Объект ошибки содержит числовой код и текстовое сообщение. Коды:
    // 1: пользователь не дал разрешения на определение местонахождения
    // 2: броузер не смог определить местонахождение
    // 3: истекло предельное время ожидания
    elt.innerHTML = "Ошибка определения местонахождения " + e.code + ": + е.message;
  }
  // Эта функция будет вызвана в случае успешного выполнения запроса
  function success(pos) {
    // Эти поля возвращаются всегда. Обратите внимание, что поле timestamp
    // принадлежит внешнему объекту pos, а не вложенному coords,
    var msg = "At " +
      new Date(pos.timestamp).toLocaleString() +
        " вы находились в " + pos.coords.accuracy +  " метрах от точки " +
        pos.coords.latitude + " северной широты " +
        pos.coords.longitude + " восточной долготы.":
    // Если устройство возвращает высоту над уровнем моря, добавить эту информацию,
    if (pos.coords.altitude) {
      msg += " Вы находитесь на высоте " +
        pos.coords.altitude + " ± " + pos.coords.altitudeAccuracy +
                    " метров над уровнем моря.";
    }
    // Если устройство возвращает направление и скорость движения,
    // добавить и эту информацию.
    if (pos.coords.speed) {
      msg += " Вы перемещаетесь со скоростью " +
        pos.coords.speed + "м/сек в направлении " + pos.coords.heading + ".";
    }
    elt.innerHTML = msg; // Отобразить информацию о местонахождении
  }
}

Оглавление книги


Генерация: 2.342. Запросов К БД/Cache: 3 / 1
поделиться
Вверх Вниз