Привет, Хабр! Хотим рассказать о том, как создать плагин Qt GeoServices и использовать его в своём приложении на ОС Аврора. В этом посте мы подробно объясним, как научить приложение определять координаты устройства на карте и прокладывать оптимальные маршруты с помощью сервиса Sight Safari. Самые нетерпеливые могут пощупать готовый код плагина и демо-приложения на GitHub, всех остальных приглашаем под кат.
Зачем писать свой плагин
Когда приложению на Qt требуется поддержка карт, то первое, что приходит на ум — использовать QML-компонент Map. Но к нему нужен плагин, реализующий работу с провайдером данных для карты. Так что если ваше приложение должно работать offline или использовать сторонний API, стандартные плагины вас могут не устроить. Придётся реализовывать либо свой плагин, либо свою карту.
Мы пойдём по первому пути: создадим плагин Qt GeoServices, который будет обращаться к offline-провайдеру тайлов для отображения карты и к Sight Safari. На Хабре уже рассказывали об этом сервисе для построения маршрутов (раз и два). Возможно, кому-то не помешает изучить базовую информацию о работе с картами в Аврора.
Подготовка: настраиваем OSM Scout Server
Для offline-доступа к тайлам карты устанавливаем на наш телефон под управлением ОС Аврора OSM Scout Server — полностью автономное решение для навигации. Ещё нам понадобится дополнительный модуль со шрифтами Noto, которые используются для рендеринга карт.
Чтобы тайлы возвращались в подходящем для отображения формате, выбираем профиль «Рекомендовано для карт с векторными и растровыми тайлами». Теперь осталось выбрать и скачать карты необходимого района в «Диспетчере карт». Они отобразятся списком, как на рисунке ниже.
С чего начинается плагин: конфигурационный файл
Театр начинается с вешалки, а Qt-плагин — с конфигурационного json-файла.
{
"Keys": ["osmscoutoffline"],
"Provider": "osmscoutoffline",
"Version": 100,
"Experimental": false,
"Features": [
"OfflineMappingFeature",
"OnlineRoutingFeature"
]
}
Keys
— уникальное имя плагина, Provider
— имя сервиса-провайдера, Version
— версия плагина, Experimental
— статус плагина, Features
— список поддерживаемых функций. Согласно нашей конфигурации, плагин osmscoutoffline
версии 1.0.0
, поддерживает функции отображения карт offline
и построения маршрутов online
и не является экспериментальным, то есть доступен для всех приложений.
Регистрируем плагин в системе
qgeoserviceproviderfactoryosmscoutoffline.h
#ifndef QGEOSERVICEPROVIDERFACTORYOSMSCOUTOFFLINE_H
#define QGEOSERVICEPROVIDERFACTORYOSMSCOUTOFFLINE_H
#include <QObject>
#include <QGeoServiceProviderFactory>
class QGeoServiceProviderFactoryOsmScoutOffline : public QObject, public QGeoServiceProviderFactory
{
Q_OBJECT
Q_INTERFACES(QGeoServiceProviderFactory)
Q_PLUGIN_METADATA(IID "org.qt-project.qt.geoservice.serviceproviderfactory/5.0"
FILE "../osmscoutoffline_plugin.json")
public:
QGeoRoutingManagerEngine *createRoutingManagerEngine(const QVariantMap ¶meters,
QGeoServiceProvider::Error *error,
QString *errorString) const;
QGeoMappingManagerEngine *createMappingManagerEngine(const QVariantMap ¶meters,
QGeoServiceProvider::Error *error,
QString *errorString) const;
};
#endif // QGEOSERVICEPROVIDERFACTORYOSMSCOUTOFFLINE_H
qgeoserviceproviderfactoryosmscoutoffline.cpp
#include "qgeoserviceproviderfactoryosmscoutoffline.h"
#include "qgeoroutingmanagerengineosmscoutoffline.h"
#include "qgeotiledmappingmanagerengineosmscoutoffline.h"
QGeoRoutingManagerEngine *QGeoServiceProviderFactoryOsmScoutOffline::createRoutingManagerEngine(
const QVariantMap ¶meters, QGeoServiceProvider::Error *error,
QString *errorString) const
{
return new QGeoRoutingManagerEngineOsmScoutOffline(parameters, error, errorString);
}
QGeoMappingManagerEngine *QGeoServiceProviderFactoryOsmScoutOffline::createMappingManagerEngine(
const QVariantMap ¶meters, QGeoServiceProvider::Error *error,
QString *errorString) const
{
return new QGeoTiledMappingManagerEngineOsmScoutOffline(parameters, error, errorString);
}
Обратите внимание на макрос Q_PLUGIN_METADATA
, который регистрирует json-файл в системе. Параметр IID
— название упомянутого выше класса для реализации интерфейса (org.qt-project.qt.geoservice.serviceproviderfactory/5.0). Параметр FILE
— путь к json-файлу.
Объявленные в public-секции методы createRoutingManagerEngine
и createMappingManagerEngine
создают объекты, отвечающие за ту или иную функцию указанного в Q_PLUGIN_METADATA
интерфейса. Класс QGeoRoutingManagerEngineOsmScoutOffline
отвечает за составление маршрутов, с ним мы разберёмся позже. Сейчас нас больше интересует QGeoTiledMappingManagerEngineOsmScoutOffline
— наследник класса QGeoTiledMappingManagerEngine
, работающий с OSM Scout Server. В конструкторе этого класса устанавливаем параметры, необходимые для работы нашего плагина.
QGeoTiledMappingManagerEngineOsmScoutOffline::QGeoTiledMappingManagerEngineOsmScoutOffline
QGeoTiledMappingManagerEngineOsmScoutOffline::QGeoTiledMappingManagerEngineOsmScoutOffline(
const QVariantMap ¶meters, QGeoServiceProvider::Error *error, QString *errorString)
{
QGeoCameraCapabilities cameraCaps;
cameraCaps.setMinimumZoomLevel(0.0);
cameraCaps.setMaximumZoomLevel(19.0);
setCameraCapabilities(cameraCaps);
setTileSize(QSize(256, 256));
QList<QGeoMapType> mapTypes;
mapTypes << QGeoMapType(QGeoMapType::StreetMap, tr("Street Map"), tr("OSM Street Map"), false, false, 1);
setSupportedMapTypes(mapTypes);
QGeoTileFetcherOsmScoutOffline *tileFetcher = new QGeoTileFetcherOsmScoutOffline(this);
tileFetcher->setParams(parameters);
setTileFetcher(tileFetcher);
*error = QGeoServiceProvider::NoError;
errorString->clear();
}
Указываем минимальный и максимальный уровни масштабирования карты, а также размер получаемых от сервера тайлов. Затем прописываем поддерживаемые типы карт — в данном случае только карты улиц (StreetMap). Наконец, создаём объект, который будет обращаться к серверу за тайлами. Далее рассмотрим его подробнее.
Объект, который обращается к серверу
QGeoTiledMapReply *QGeoTileFetcherOsmScoutOffline::getTileImage(const QGeoTileSpec &spec)
{
QUrlQuery query;
for (QString &key : m_params.keys())
query.addQueryItem(key, m_params[key].toString());
query.addQueryItem(QStringLiteral("x"), QString::number(spec.x()));
query.addQueryItem(QStringLiteral("y"), QString::number(spec.y()));
query.addQueryItem(QStringLiteral("z"), QString::number(spec.zoom()));
QUrl url(QStringLiteral("http://localhost:8553/v1/tile"));
url.setQuery(query);
QNetworkRequest remoteRequest(url);
QNetworkReply *reply = m_networkManager->get(remoteRequest);
return new QGeoMapReplyOsmScoutOffline(reply, spec);
}
Метод getTileImage
принимает параметры, которым должен соответствовать запрашиваемый тайл, и возвращает объект, содержащий изображение. Запрос к серверу создаётся внутри метода, поэтому его дополнительная синхронизация не требуется даже в случае нелокального расположения сервера.
Важно указать формат API, к которому мы обращаемся. В нашей реализации OSM Scout Server крутится непосредственно на телефоне (http://localhost) на стандартном порте (8553). Запрос тайла с сервера (метод tile
) идёт через первую версию API (v1). Параметры x
, y
и z
— координаты и масштаб запрашиваемого тайла. Цикл по m_params
в начале метода позволяет добавить к запросу параметры, переданные из определения плагина в QML.
Сохраняем тайл в наследнике класса QGeoTiledMapReply
— для отображения в компоненте Map
.
void QGeoMapReplyOsmScoutOffline::networkReplyFinished()
{
if (!m_reply)
return;
if (m_reply->error() != QNetworkReply::NoError)
return;
setMapImageData(m_reply->readAll());
setMapImageFormat("png");
setFinished(true);
m_reply->deleteLater();
m_reply = 0;
}
Здесь после проверки на наличие ошибок пришедшее изображение считывается в виде массива байтов. Затем указывается формат изображения (в данном случае — png). Сигнал finished
уведомляет компонент Map
о том, что тайл готов к отрисовке.
А что с маршрутом? Обзор Sight Safari API
Итак, наш Qt GeoServices-плагин может определять и показывать положение пользователя на карте. Теперь научим его отображать маршруты, построенные с помощью сервиса Sight Safari.
Для поиска маршрута сервис предоставляет метод findpath
, который принимает на вход шесть параметров. Впрочем, мы собираемся передавать только три:
from
— начало пути;to
— конец пути;ratio
— «интересность».
Чем больше значение последнего параметра, тем больше достопримечательностей и просто красивых мест встретится на пути. Мы зададим ratio равным единице — в качестве оптимального сочетания интересности и протяжённости маршрута.
Отладочная информация в нашем приложении не нужна, поэтому параметр debug
оставим установленным по умолчанию. Якорная точка через desiredCoordinates
не передаётся, так как в параметрах from
и to
будут указаны точные координаты. Параметр apiKey
тоже опустим: для демонстрационных целей хватает бесплатных возможностей API.
В качестве ответа findpath
возвращает json-объект, в котором нас интересует массив latLonPoints
: здесь хранятся точки, по которым проходит маршрут — его-то и надо отобразить на нашей карте. Кстати, несмотря на указанный в документации Sight Safari тип запроса POST, API прекрасно работает и через GET-запросы.
Расширение функциональности плагина
Настало время разобраться с классом QGeoServiceProviderFactoryOsmScoutOffline
, с которым мы уже сталкивались в ходе регистрации плагина в системе. В нём имплементирован метод createRoutingManagerEngine
, возвращающий указатель на объект класса, отвечающего за построение маршрутов:
QGeoRoutingManagerEngine *QGeoServiceProviderFactoryOsmScoutOffline::createRoutingManagerEngine(
const QVariantMap ¶meters, QGeoServiceProvider::Error *error, QString *errorString) const
{
return new QGeoRoutingManagerEngineOsmScoutOffline(parameters, error, errorString);
}
Основной метод в этом классе — calculateRoute
, принимающий запрос с точками, через которые проходит маршрут, и возвращающий ответ с одним или несколькими маршрутами для отображения:
QGeoRouteReply
QGeoRouteReply *QGeoRoutingManagerEngineOsmScoutOffline::calculateRoute(
const QGeoRouteRequest &request)
{
QGeoCoordinate start = request.waypoints()[0];
QGeoCoordinate end = request.waypoints()[1];
QString from = QStringLiteral("%1,%2").arg(QString::number(start.latitude()), QString::number(start.longitude()));
QString to = QStringLiteral("%1,%2").arg(QString::number(end.latitude()), QString::number(end.longitude()));
QUrlQuery query;
query.addQueryItem(QStringLiteral("from"), from);
query.addQueryItem(QStringLiteral("to"), to);
query.addQueryItem(QStringLiteral("ratio"), QStringLiteral("1"));
QUrl url(QStringLiteral("https://sightsafari.city/api/v1/routes/direct"));
url.setQuery(query);
QNetworkRequest remoteRequest(url);
QNetworkReply *reply = mNetworkManager->get(remoteRequest);
QGeoRouteReplyOsmScoutOffline routeReply = new QGeoRouteReplyOsmScoutOffline(reply, request, this);
connect(routeReply, &QGeoRouteReplyOsmScoutOffline::finished,
this, &QGeoRoutingManagerEngineOsmScoutOffline::replyFinished);
connect(routeReply, static_cast<void(QGeoRouteReplyOsmScoutOffline::)
(QGeoRouteReplyOsmScoutOffline::Error, const QString &)>(&QGeoRouteReplyOsmScoutOffline::error),
this, &QGeoRoutingManagerEngineOsmScoutOffline::replyError);
return routeReply;
}
Сначала из пришедшего запроса получаются координаты начала и конца маршрута — подразумевается, что запрос содержит только две координаты. Далее формируется и отправляется GET-запрос к описанному ранее методу API. Здесь поле mUrlPrefix
содержит endpoint сервера.
После этого формируется и возвращается указатель на объект с полученными маршрутами. У этого объекта могут быть сигналы finished
или error
, в соответствии с которыми выполняется обработка успешно либо неуспешно завершённого запроса. Обработка у нас простая:
void QGeoRoutingManagerEngineOsmScoutOffline::replyFinished()
{
QGeoRouteReply *reply = qobject_cast<QGeoRouteReply *>(sender());
if (reply)
emit finished(reply);
}
void QGeoRoutingManagerEngineOsmScoutOffline::replyError(QGeoRouteReply::Error errorCode,
const QString &errorString)
{
QGeoRouteReply *reply = qobject_cast<QGeoRouteReply *>(sender());
if (reply)
emit error(reply, errorCode, errorString);
}
Метод calculateRoute
создаёт и возвращает объект типа QGeoRouteReplyOsmScoutOffline
. Всё, что нужно для работы объекта, находится в конструкторе:
QGeoRouteReplyOsmScoutOffline::QGeoRouteReplyOsmScoutOffline
QGeoRouteReplyOsmScoutOffline::QGeoRouteReplyOsmScoutOffline(QNetworkReply *reply,
const QGeoRouteRequest &request,
QObject parent)
: QGeoRouteReply(request, parent)
{
if (reply == nullptr) {
setError(UnknownError, QStringLiteral("Null reply"));
return;
}
connect(reply, &QNetworkReply::finished,
this, &QGeoRouteReplyOsmScoutOffline::networkReplyFinished);
connect(reply, static_cast<void(QNetworkReply::)(QNetworkReply::NetworkError)>(&QNetworkReply::error),
this, &QGeoRouteReplyOsmScoutOffline::networkReplyError);
connect(this, &QGeoRouteReplyOsmScoutOffline::destroyed,
reply, &QNetworkReply::deleteLater);
}
После проверки переданного в конструктор указателя на корректность, обработчики привязываются к посылаемым сигналам. Последние две связки — для обработки ошибки сетевого соединения и для освобождения памяти — не представляют большого интереса. А обработку успешного ответа от сервера рассмотрим подробнее:
void QGeoRouteReplyOsmScoutOffline::networkReplyFinished
void QGeoRouteReplyOsmScoutOffline::networkReplyFinished()
{
QNetworkReply *reply = static_cast<QNetworkReply *>(sender());
reply->deleteLater();
if (reply->error() != QNetworkReply::NoError)
return;
QJsonDocument jsonDoc = QJsonDocument::fromJson(reply->readAll());
QJsonObject jsonBody = jsonDoc.object().value(QStringLiteral("body")).toObject();
QJsonArray jsonPath = jsonBody.value(QStringLiteral("latLonPoints")).toArray();
QList<QGeoCoordinate> coords;
for (QJsonValue value : jsonPath) {
QJsonArray coord = value.toArray();
coords.append(QGeoCoordinate(coord.at(0).toDouble(), coord.at(1).toDouble()));
}
QGeoRoute route;
route.setPath(coords);
route.setRequest(request());
setRoutes({ route });
setFinished(true);
}
В этом методе из пришедшего с сервера json-ответа извлекаем массив latLonPoints
с точками маршрута, который преобразуем в список объектов координат, сохраняем как маршрут (здесь мы подразумеваем, что маршрут один) и указываем, что обработка запроса произведена успешно.
После вызова метода setFinished
посылается сигнал finished
. Он обрабатывается упомянутым ранее методом replyFinished
, сообщающим приложению, использующему плагин, что маршрут построен.
Кстати, что там с приложением? Особенности yaml-файла
Мы почти готовы собрать и установить rpm-пакет, чтобы перейти к использованию плагина в своих приложениях. Осталось подкорректировать стандартный yaml-файл, создаваемый Аврора IDE, или написать свой с нуля.
qtgeoservices-osmscoutoffline.yaml
Name: qtgeoservices_osmscoutoffline
Summary: QtGeoServices OSM Scout Offline with Sight Safary routing
Version: 0.5.0
Release: 1
Group: System/Libraries
URL: https://github.com/osanwe/qtgeoservices-osmscoutoffline
License: BSD-3-Clause
Sources:
- '%{name}-%{version}.tar.bz2'
Description: |
QtGeoServices OSM Scout Offline with Sight Safary routing
Configure: none
Builder: qtc5
PkgConfigBR:
- Qt5Core
- Qt5Location
- Qt5Positioning
- Qt5Network
Files:
- '%{_libdir}/qt5/plugins/geoservices/libqtgeoservices_osmscoutoffline.so'
Самое важное изменение — блок Files
, в котором прописывается путь установки скомпилированной библиотеки
Ищем себя на карте
В новом проекте, созданном Aurora IDE, по умолчанию создаются четыре QML элемента:
CoverPage.qml — данный элемент отвечает за вывод наиболее важной информации о приложении когда оно свернуто.
FirstPage.qml — в новом проекте эта страница является первой.
SecondPage.qml — не будем использовать.
harbour-walking.qml — название данного файла совпадает с названием проекта; в нем задаются элемент инициализации и элемент обложки приложения.
Для нашего примера потребуется только FirstPage.qml, поэтому SecondPage.qml можно удалить. Далее мы удаляем всё содержимое FirstPage.qml, переименовываем его в MainPage.qml и приступаем к реализации нашего приложения.
Первым делом добавляем все необходимые импорты для отображения карты и получения информации с GPS-приемника устройства. Также у нас есть некоторое множество дополнительно реализованных элементов, которые расположены в директории qml/views. Для того, чтобы у нас появился доступ ко всем элементам в данном каталоге необходимо добавить соответствующий импорт.
import QtQuick 2.5
import QtLocation 5.0
import QtPositioning 5.0
import Sailfish.Silica 1.0
import "../views"
Добавляем корневой для данного QML файла элемент Page
и начинаем его заполнять.
Page
Page {
id: page
property bool mapFollowing: false
property var mapGpsPosition: positionSource.position.coordinate
property var mapCenterPosition: QtPositioning.coordinate(NaN, NaN)
property var pressCoords: QtPositioning.coordinate(NaN, NaN)
allowedOrientations: Orientation.Portrait
Drawer {
id: drawer
anchors.fill: parent
open: true
backgroundSize: background.height
background: Item {
id: background
// some code
}
Plugin {
id: mapPlugin
name: "osmscoutoffline"
}
PositionSource {
id: positionSource
updateInterval: 1000
active: true
preferredPositioningMethods: PositionSource.AllPositioningMethods
}
Map {
id: map
function initMapCenter() {
var moscowCenterPos = QtPositioning.coordinate(55.751244, 37.618423)
if (mapGpsPosition.isValid) {
map.center = mapGpsPosition
mapFollowing = true
} else {
map.center = moscowCenterPos
}
map.zoomLevel = 15
}
function setMapCenterFromGps() {
if (mapGpsPosition.isValid) {
map.zoomLevel = 17
map.center.latitude = mapGpsPosition.latitude
map.center.longitude = mapGpsPosition.longitude
mapFollowing = true
}
}
anchors.fill: parent
plugin: mapPlugin
onCenterChanged: {
mapCenterPosition = center
if (mapFollowing && center !== mapGpsPosition)
mapFollowing = false
}
Component.onCompleted: map.initMapCenter()
MapMarker {
coordinate: mapGpsPosition
visible: mapGpsPosition.isValid
source: "../images/mylocation.svg"
}
Connections {
target: page
onMapGpsPositionChanged: {
if (mapFollowing)
map.setMapCenterFromGps()
}
}
}
}
}
Что здесь было добавлено:
Дополнительные переменные:
mapFollowing
— переменная в которой хранится флаг следования за изменяющийся позицией, получаемой с GPS датчика устройства; данное поведение будет воспроизводиться в том случае, если мы вызовем далее описанную функциюsetMapCenterFromGps
у объектаmap
и не будем изменять область вывода карты вручную.mapGpsPosition
— переменная в которой хранится текущая координата, полученная от элементаPositionSource
.mapCenterPosition
— в данной переменной хранится координата центра карты.pressCoords
— переменная для хранения координаты нажатия по карте.
Drawer
— контейнер который позволяет выдвинуть некоторую область на передний план, где могут быть расположены некоторые второстепенные элементы управления; выезжающая область задаётся через свойство:background
(содержимое контейнера мы опишем позже).Plugin
— непосредственно сам плагин, описанный ранее, который будет взаимодействовать с картой и предоставлять ей необходимые тайлы. В качестве значения параметраname
устанавливаем имя плагина, указанное в json-файле.PositionSource
— объект, предоставляющий информацию о текущей позиции телефона (для его работы необходимо включить GPS датчик устройства). Установка свойстваactive
в значениеtrue
запускает работу данного элемента. СвойствоupdateInterval
равное1000
задает элементу таймаут по которому он будет сообщать о текущем положении раз в одну секунду. СвойствоpreferredPositioningMethods
со значениемPositionSource.AllPositioningMethods
говорит о том, что данным элементом будут использоваться любые методы позиционирования.Map
— элемент для отображения карты. С помощью свойстваanchors.fill: parent
указываем, что карта должна занимать все доступное пространство. Значение свойстваplugin
— идентификатор объявленного ранее плагина. По сигналуonCenterChanged
выполняется обновление переменнойmapCenterPosition
, объявленной в начале элементаPage
и в случае, если флагmapFollowing
был выставлен вtrue
, он сбрасывается (это означает, что мы не хотим чтобы центр карты автоматически обновлялся в соответствии с данными, получаемыми с GPS-приемника; для того, чтобы вернуть данное поведение обратно, необходимо нажать на кнопку в элементеDrawer
, который будет описан позже). По сигналуComponent.onCompleted
выполняются действия, которые необходимо произвести после полной инициализации карты; здесь вызывается функцияinitMapCenter
у объектаMap
, которая выставляет начальный масштаб и в случае, если у нас есть валидные данные с GPS-приемника, то они выбираются в качестве начальной позиции центра карты, в обратном случае выставляются координаты центра Москвы.MapMarker
— элемент, унаследованный отMapQuickItem
и расположенный в каталоге qml/views (с его реализацией можно ознакомиться в файлах исходника проекта) с предустановленными свойствами отображения границ изображений, их размером и якорной точкой. Таким образом, для данного элемента достаточно установить только иконку в свойствеsource
, его позицию на карте через свойствоcoordinate
и флаг отображения в свойствеvisible
(так, например, если нам по каким-либо причинам не удалось получить валидные координаты с GPS-приемника, то и отображать этот элемент смысла нет).Connections
— данный элемент позволяет задать ему объект, у которого мы хотим обрабатывать сигналы через свойствоtarget
. При подключении к сигналам в QML обычным способом является создание обработчика видаon<Signal>
. Здесь мы отслеживаем изменение текущей координаты полученной отPositionSource
и, если свойствоmapFollowing
выставлено вtrue
, то происходит автоматическое обновление центра отображаемой области карты при изменении его координат.
Собираем данные исходники и проверяем. Результат должен получиться аналогично изображению ниже.
Добавляем в приложение возможность построения маршрутов
Добавим в элемент Map
элемент BusyIndicator
, отвечающий за индикацию процесса выполнения запроса к online-сервису по построению маршрутов. Данный элемент мы центрируем по отношению к родительскому, а именно к Map
через свойство anchors.centerIn
. Мы делаем именно так, потому что, если мы отобразим выезжающую область Drawer
, то размер отображаемой области карты станет меньше, и таким образом мы избежим наложение этих элементов друг на друга. Так как на картах используется достаточно большое количество цветов, для данного индикатора лучше использовать контрастный цвет, а именно черный. Его мы получаем из системной темы и задаем свойству color
. Устанавливаем свойство size
, используя стандартное значение перечисления данного элемента. В завершении по данному элементу мы указываем, что изначально он не отображается через свойство running
, в дальнейшем это свойство будет изменяться по определенным событиям.
BusyIndicator {
id: routeLoadingIndicator
anchors.centerIn: parent
size: BusyIndicatorSize.Large
color: Theme.rgba(Theme.darkPrimaryColor, Theme.opacityOverlay)
running: false
}
Добавим в элемент Map
еще два элемента типа MapMarker
для отображения начальной и конечной точек маршрута. Данные элементы аналогичны тому, что был описан ранее и отличаются только иконкой. Туда же добавим элемент MapRoute
для вывода самого маршрута. Он аналогично MapMarker
расположен в каталоге qml/views (с его реализацией можно ознакомиться в файлах исходника проекта). Данный элемент унаследован от MapPolyline
с предустановленными свойствами отображения границ линии и ее толщиной. Этот элемент имеет свойство path
которое будет заполняться чуть дальше.
MapRoute { id: mapRoute }
MapMarker {
id: markerStart
visible: false
source: "../images/location.svg"
}
MapMarker {
id: markerFinish
visible: false
source: "../images/location.svg"
}
Добавляем в элемент Drawer
объекты, позволяющие получать информацию о маршрутах. Элемент RouteQuery
отвечает за формирование запроса на построение маршрута к онлайн сервису Sight Safari. Элемент RouteModel
хранит полученные маршруты и связывается с RouteQuery
с помощью параметра query
. В качестве значения параметра plugin
указывается наш плагин объявленный ранее. Параметру autoUpdate
присваиваем false
, чтобы маршрут перестраивался не при изменении начальных и конечных координат, а только по запросу (по нажатии на кнопку, которая будет располагаться в Drawer
). Также, при получении сигнала onRoutesChanged
мы устанавливаем параметр path
у элемента MapRoute
и останавливаем работу индикатора загрузки информации о маршруте.
RouteQuery { id: mapRouteQuery }
RouteModel {
id: mapRouteModel
plugin: mapPlugin
query: mapRouteQuery
autoUpdate: false
onRoutesChanged: {
routeLoadingIndicator.running = false
mapRoute.path = mapRouteModel.get(0).path
}
}
Теперь подготовим выезжающую область в элементе Drawer
. Для этого заполним элементами управления Item
который установили свойству background
в Drawer
. Опустим описание всех свойств следующих элементов, достаточно будет описать их назначение. В данном элементе отображаются точки начала и конца маршрута, и две кнопки. Первая кнопка перемещает центр отображаемой области карты в определившиеся GPS координаты (кнопка будет неактивна, если GPS координаты невалидны или текущий отображаемый центр совпадает с координатами полученными с GPS-приемника). Вторая кнопка выполняет построение маршрута между двумя заданными точками (кнопка становится активной только, если обе точки будут заданы). В следующем фрагменте кода мы используем элемент CoordField
, который аналогично остальным пользовательским элементам расположен в каталоге qml/views (с его реализацией можно ознакомиться в файлах исходника проекта).
Item
Item {
id: background
anchors.fill: parent
height: column.implicitHeight + column.anchors.topMargin + column.anchors.bottomMargin
Column {
id: column
anchors {
fill: parent
margins: Theme.paddingMedium
}
spacing: Theme.paddingMedium
width: parent.width
Label {
text: qsTr("Route from:")
font.bold: true
}
CoordField { id: startCoordField }
Label {
text: qsTr("Route to:")
font.bold: true
}
CoordField { id: endCoordField }
Row {
anchors.horizontalCenter: parent.horizontalCenter
spacing: Theme.paddingMedium
Button {
text: qsTr("My position")
enabled: mapGpsPosition.isValid
&& (mapGpsPosition.latitude !== mapCenterPosition.latitude
|| mapGpsPosition.longitude !== mapCenterPosition.longitude)
onClicked: map.setMapCenterFromGps()
}
Button {
text: qsTr("Route")
enabled: startCoordField.coordinate.isValid && endCoordField.coordinate.isValid
onClicked: {
routeLoadingIndicator.running = true
mapRouteQuery.clearWaypoints()
mapRouteQuery.addWaypoint(startCoordField.coordinate)
mapRouteQuery.addWaypoint(endCoordField.coordinate)
mapRouteModel.update()
}
}
}
}
}
Данный элемент будет выглядеть так, как представлено на изображении ниже.
Добавим возможность управления отображения данным выезжающим элементом. Для этого поместим в элемент Map
кнопку типа MapButton
, которая аналогично остальным пользовательским элементам расположена в каталоге qml/views (с его реализацией можно ознакомиться в файлах исходника проекта). Этот элемент унаследован от IconButton
и дополнен рамкой и фоном. Тут мы его прижимаем к правому нижнему углу и задаем отступы по краям. Устанавливаем элементу в свойстве icon.source
стандартную иконку для меню. Указываем, что иконка элемента будет подсвечиваться тогда, когда элемент Drawer
открыт. Также при получении сигнала onClicked
мы будем менять состояние отображения Drawer
на обратное.
MapButton {
anchors {
bottom: parent.bottom
right: parent.right
bottomMargin: Theme.paddingLarge
rightMargin: Theme.horizontalPageMargin
}
icon.source: "image://theme/icon-m-menu"
highlighted: drawer.open
onClicked: drawer.open ? drawer.hide() : drawer.show()
}
Раз уж мы добавили одну кнопку, добавим еще парочку для управления масштабом в элемент Map
.
Map
Column {
anchors {
right: parent.right
verticalCenter: parent.verticalCenter
rightMargin: Theme.horizontalPageMargin
}
spacing: Theme.paddingLarge
MapButton {
id: buttonZoomIn
icon.source: "../images/zoom-plus.svg"
enabled: map.zoomLevel < map.maximumZoomLevel
onClicked: map.zoomLevel = Math.min(map.zoomLevel + 1.0, map.maximumZoomLevel)
}
MapButton {
id: buttonZoomOut
icon.source: "../images/zoom-minus.svg"
enabled: map.zoomLevel > map.minimumZoomLevel
onClicked: map.zoomLevel = Math.max(map.zoomLevel - 1.0, map.minimumZoomLevel)
}
}
Теперь, когда большая часть реализована, необходимо добавить возможность выбора начальной и конечной точек на карте к элементу Map
. Для этого добавляем в него MouseArea
, отвечающий за обработку нажатий по экрану устройства. При нажатии по области карты, в случае, если диалог был закрыт, то он отображается и координата нажатия сохраняется в объявленную в самом начале переменную pressCoords
, после чего отображается диалог PointDialog
, который аналогично остальным пользовательским элементам расположен в каталоге qml/views (с его реализацией можно ознакомиться в файлах исходника проекта). В нем выводится координата нажатия, сохраненная ранее в pressCoords
, и две кнопки: “От” и “До”. При нажатии на кнопку любую из кнопок происходит передача текущей координаты в соответствующее поле в выезжающей области Drawer
. Как только будут заданы обе координаты, кнопка построения маршрута станет активной.
MouseArea {
anchors.fill: parent
z: -1
onClicked: {
if (choosePointDialog.visible) {
choosePointDialog.visible = false
} else {
pressCoords = map.toCoordinate(Qt.point(mouse.x, mouse.y))
choosePointDialog.visible = true
}
}
}
PointDialog { id: choosePointDialog }
После нажатия на кнопку построения маршрута получится что-то наподобие изображения ниже.
На этом всё, подробнее о создании плагинов можно почитать на Хабре или в документации Qt. Мы же со своей стороны готовы дать любые пояснения в комментариях.
Материалы для публикации подготовлены Петром Вытовтовым и Павлом Казеко с комментариями и правками от Кирилла Чувилина.