IdentifiantMot de passe
Loading...
Mot de passe oublié ?Je m'inscris ! (gratuit)

Places (C++)

Qt Location Reference Documentation.

Article lu   fois.

L'auteur

Liens sociaux

Viadeo Twitter Facebook Share on Google+   

I. Overview

The Places API allows users to discover places/points of interest and view details about them such as address and contact information; some places may even have rich content such as images and reviews. The Places API also facilitates management of places and categories, allowing users to save and remove them.

II. Place Definition

A place is a point of interest, it could be a favorite restaurant, a park or someone's home. A QPlace object represents a place by acting as a container for various information about that place.

This information can be divided into 2 broad classifications

  • Details

  • Rich content

The place details consist of properties of the place, such as the name, location, contact information and so on. When a place is returned during a search, these details are filled in. Sometimes in order to save bandwidth, there are further details about the place that can be retrieved on an individual place by place basis, if the user is interested. The QPlace::detailsFetched() function can be queried to see if all available details have been fetched, and if not, QPlaceManager::getPlaceDetails() can be used to retrieve them. Precisely which details are populated during a search and which need to be fetched individually may vary from provider to provider. See plugin documentation for more details.

The rich content of a place consists of items such as images, reviews and editorials. Potentially there may be many rich content items, so they are treated separately from the place details. They can be retrieved in a paged fashion via QPlaceManager::getPlaceContent(). If necessary, the content may be assigned to a place so it can act as a convenient container.

III. Common Operations

 

III-1. Initializing a Manager

All places functionality is facilitated by a QPlaceManager instance. One must specify a QGeoServiceProvider in order to create the QPlaceManager

 
Sélectionnez
//The "provider name" is used to select a particular provider
QGeoServiceProvider *provider = new QGeoServiceProvider("provider name");
QPlaceManager *manager = provider->placeManager();

III-2. Discovery/Search

 
Sélectionnez
//instantiate request and set parameters
QPlaceSearchRequest searchRequest;
searchRequest.setSearchTerm("ice cream");
searchRequest.setSearchArea(QGeoCircle(QGeoCoordinate(12.34, 56.78)));

//send off a search request
/*QPlaceSearchReply * */ searchReply = manager->search(searchRequest);

//connect a slot to handle the reply
connect(searchReply, &QPlaceSearchReply::finished, this, &RequestHandler::handleSearchReply);

The request is an asynchronous operation so we need a slot to handle the completion of the request. In the handler we check that there are no errors and that our search result type is a place. If so we can then retrieve some of the core details of the place. At the end of the slot, we delete the reply since they are for single use only.

 
Sélectionnez
void handleSearchReply() {
    if (searchReply->error() == QPlaceReply::NoError) {
        for (const QPlaceSearchResult &result : searchReply->results()) {
            if (result.type() == QPlaceSearchResult::PlaceResult) {
                QPlaceResult placeResult = result;
                qDebug() << "Name: " << placeResult.place().name();
                qDebug() << "Coordinate " << placeResult.place().location().coordinate().toString();
                qDebug() << "Street: " << placeResult.place().location().address().street();
                qDebug() << "Distance: " << placeResult.distance();
            }
        }
    }
    searchReply->deleteLater();  //discard reply
    searchReply = nullptr;
}

Note: Depending upon the plugin backend that was chosen, the search results may contain places which have further details that can be fetched on a place by place basis. To fetch these other details see Fetching Place Details.

III-2-1. Recommendations

Recommendations can be retrieved by supplying a place id via QPlaceSearchRequest::setRecommendationId(). Any places similar to the given place are retrieved.

III-2-2. Paging

If the plugin supports paging, the limit parameter may be provided to the search request.

 
Sélectionnez
QPlaceSearchRequest searchRequest;
searchRequest.setLimit(15); //specify how many results are to be retrieved.

III-3. Fetching Place Details

A place that has been returned from a search request may have more details that can be fetched. The following demonstrates how to check if there are further details and if so how to request them.

 
Sélectionnez
if (!place.detailsFetched()) {
    /*QPlaceDetailsReply * */ detailsReply = manager->getPlaceDetails(place.placeId());
    connect(detailsReply, &QPlaceDetailsReply::finished, this, &RequestHandler::handleDetailsReply);
}
    ...
    ...
void handleDetailsReply() {
    QPlace place;
    if (detailsReply->error() == QPlaceReply::NoError)
        place = detailsReply->place();

    detailsReply->deleteLater(); //discard reply
    detailsReply = nullptr;
}

III-4. Fetching Rich Content

Rich content such as images and reviews is retrieved through the manager and then if required assigned to a place.

 
Sélectionnez
QPlaceContentRequest request;
request.setContentType(QPlaceContent::ImageType);
request.setPlaceId(place.placeId());
request.setLimit(5);
/*QPlaceContentReply * */ contentReply = manager->getPlaceContent(request);
connect(contentReply, &QPlaceContentReply::finished, this, &RequestHandler::handleImagesReply);

We can handle the content request as shown below.

 
Sélectionnez
void handleImagesReply() {
    if (contentReply->error() == QPlaceReply::NoError) {
        const auto content = contentReply->content();
        for (auto iter = content.cbegin(), end = content.cend(); iter != end; ++iter) {
            qDebug() << "Index: " << iter.key();
            QPlaceImage image = iter.value();
            qDebug() << image.url();
            qDebug() << image.mimeType();
        }

        //alternatively if indexes are irrelevant
        for (const QPlaceImage &image : contentReply->content()) {
            qDebug() << image.url();
            qDebug() << image.mimeType();
        }

        //we can assign content to the place that it belongs to.
        //the place object serves as a container where we can retrieve
        //content that has already been fetched
        place.insertContent(contentReply->request().contentType(), contentReply->content());
        place.setTotalContentCount(contentReply->request().contentType(), contentReply->totalCount());
    }

    contentReply->deleteLater();
    contentReply = nullptr;
}

It is important to note that the results in the QPlaceContentReply, is a QPlaceContent::Collection which is essentially a QMap<int, QPlaceContent>. The key int in this case is the index of the content, and the value is the content itself. Due to the way Content is implemented it is possible to convert a content type as follows

 
Sélectionnez
QPlaceImage image = content; //provided that 'content' has a type QPlace::ImageType

The usage of the QPlaceContent::Collection and the conversion between content and its subtypes means that code for handling the mechanics of paging reviews, images and editorials can be easily shared.

III-5. Search Suggestions

The retrieval of search term suggestions is very similar to performing a place search. A QPlaceSearchRequest is used just like a place search, the only difference being that the search term is set to a partially completed string.

 
Sélectionnez
QPlaceSearchRequest request;
request.setSearchTerm("piz");
request.setSearchArea(QGeoCircle(QGeoCoordinate(12.34, 56.78)));
/* QPlaceSearchSuggestion * */suggestionReply = manager-&gt;searchSuggestions(request);
connect(suggestionReply, &amp;QPlaceSearchSuggestion::finished, this, &amp;RequestHandler::handleSuggestionReply);

And when the request is done, we can use the reply to show the suggestions.

 
Sélectionnez
void handleSuggestionReply() {
    if (suggestionReply-&gt;error() == QPlaceReply::NoError) {
        for (const QString &amp;suggestion : suggestionReply-&gt;suggestions())
            qDebug() &lt;&lt; suggestion;
    }

    suggestionReply-&gt;deleteLater(); //discard reply
    suggestionReply = nullptr;
}
 

III-6. Saving a Place

The saving of a new place is performed as follows, we create a QPlace instance and populate it with information such as a name, address and coordinate. Once done we can invoke QPlaceManager::savePlace() to begin a save operation.

 
Sélectionnez
QPlace  place;
place.setName( "Fred's Ice Cream Parlor" );

QGeoLocation location;
location.setCoordinate(QGeoCoordinate(12.34, 56.78));

QGeoAddress address;
address.setStreet("111 Nother Street");
    ...
location.setAddress(address);
place.setLocation(location);

/* QPlaceIdReply * */savePlaceReply = manager-&gt;savePlace(place);
connect(savePlaceReply, &amp;QPlaceIdReply::finished, this, &amp;RequestHandler::handleSavePlaceReply);

Once a place is saved the reply contains the new identifier for that place.

 
Sélectionnez
void handleSavePlaceReply() {
    if (savePlaceReply-&gt;error() == QPlaceReply::NoError)
        qDebug() &lt;&lt; savePlaceReply-&gt;id();

    savePlaceReply-&gt;deleteLater(); //discard reply
    savePlaceReply = nullptr;
}

Note that to save an already existing place, the QPlace::placeId() must be filled in with the correct identifier. Otherwise a new place will be created if empty or the wrong place overwritten if the identifier is incorrect.

When a place is saved, the QPlaceManager may emit QPlaceManager::placedAdded() or QPlaceManager::placeUpdated() signals. However whether a manager does so or not is provider specific, managers accessing places from a web service will typically not emit these signals while managers accessing places locally stored generally will.

III-6-1. Caveats

The Places API is currently designed for only saving core details. Saving rich content like images and reviews or details like supplier and rating is not a supported use case. Typically a manager will generally ignore these fields upon save and may produce a warning message if they are populated.

The Places API only supports saving of the following core details:

  • name

  • place id

  • location

  • contact details

  • icon

  • categories (tag-like names to describe a place)

  • visibility scope

It is possible that providers may only support a subset of these. See the plugin documentation for more details.

Saving of properties such as the rating, extended attributes, images, reviews, editorials and supplier is explicitly not supported by the Places API.

III-6-2. Saving Between Managers

When saving places between managers, there are a few things to be aware of. Some fields of a place such as the id, categories and icons are manager specific entities for example the categories in one manager may not be recognized in another. Therefore trying to save a place directly from one manager to another is not possible.

The typical approach is to use the QPlaceManager::compatiblePlace() function, it creates a copy of a place, but only copies data that the manager supports. Manager specific data such as the place identifier is not copied over. The new copy is now suitable for saving into the manager. If the manager supports matching by alternative identifiers, an alternative identifier attribute is assigned to the copy (see Matching places between managers)

 
Sélectionnez
//result retrieved from a different manager)
QPlace place = manager-&gt;compatiblePlace(result.place());
saveReply = manager-&gt;savePlace(place);
 

III-7. Removing a Place

The removal of a place is performed as follows:

 
Sélectionnez
/* QPlaceIdReply * */removePlaceReply = manager-&gt;removePlace(place.placeId());
connect(removePlaceReply, &amp;QPlaceIdReply::finished, this, &amp;RequestHandler::handleRemovePlaceReply);
    ...
    ...
void handleRemovePlaceReply() {
    if (removePlaceReply-&gt;error() == QPlaceReply::NoError)
        qDebug() &lt;&lt; "Removal of place identified by"
                 &lt;&lt; removePlaceReply-&gt;id() &lt;&lt; "was successful";

    removePlaceReply-&gt;deleteLater(); //discard reply
    removePlaceReply = nullptr;
}

When a place is removed, the QPlaceManager may emit the QPlaceManager::placeRemoved() signal. Whether a manager does so is provider specific. Managers accessing places from a web service will typically not emit these signals, while managers accessing places stored locally generally will.

III-8. Using Categories

Categories are keywords that can describe a place. For example, 'park', 'theater', 'restaurant'. A place could be described by many categories, it could be a park and a music venue and a ferry or bus stop.

To use categories they must first be initialized.

 
Sélectionnez
/* QPlaceReply * */initCatReply = manager-&gt;initializeCategories();
connect(initCatReply, &amp;QPlaceReply::finished, this, &amp;RequestHandler::handleInitCatReply);
    ...
    ...
void handleInitCatReply() {
    if (initCatReply-&gt;error() == QPlaceReply::NoError)
        qDebug() &lt;&lt; "Categories initialized";
    else
        qDebug() &lt;&lt; "Failed to initialize categories";

    initCatReply-&gt;deleteLater();
    initCatReply = nullptr;
}

After the categories have been initialized we can then use these category functions.

To retrieve the top level categories we use the QPlaceManager::childCategories() function but do not provide a category identifier.

 
Sélectionnez
const QList&lt;QPlaceCategory&gt; topLevelCategories = manager-&gt;childCategories();
for (const QPlaceCategory &amp;category : topLevelCategories)
    qDebug() &lt;&lt; category.name();

If we did provide an identifier then we could retrieve a category's children.

 
Sélectionnez
QList&lt;QPlaceCategory&gt; childCategories = manager-&gt;childCategories(pizza.categoryId());

III-9. Saving a Category

The following shows how to save a category

 
Sélectionnez
QPlaceCategory fastFood;

QPlaceCategory category;
category.setName("pizza");
/*QPlaceIdReply */ saveCategoryReply = manager-&gt;saveCategory(category);
connect(saveCategoryReply, &amp;QPlaceIdReply::finished, this, &amp;RequestHandler::handleSaveCategoryReply);

//we could have saved a category as a child by supplying a parent identifier.
saveCategoryReply = manager-&gt;saveCategory(category, fastFood.categoryId());
    ...
    ...
void handleSaveCategoryReply() {
    if (saveCategoryReply-&gt;error() == QPlaceReply::NoError) {
        qDebug() &lt;&lt; "Saved category id =" &lt;&lt; saveCategoryReply-&gt;id();
    }

    saveCategoryReply-&gt;deleteLater();
    saveCategoryReply = nullptr;
}

When a category is saved, the QPlaceManager may emit QPlaceManager::categoryAdded() or QPlaceManager::categoryUpdated() signals. However whether a manager does so or not is provider specific, managers accessing places from a web service will typically not emit these signals while managers accessing places locally stored generally will.

III-10. Removing a Category

Category removal is very similar to removing a place

 
Sélectionnez
/* QPlaceIdReply * */removeCategoryReply = manager-&gt;removeCategory(place.placeId());
connect(removeCategoryReply, &amp;QPlaceIdReply::finished, this, &amp;RequestHandler::handleRemoveCategoryReply);
    ...
    ...
void handleRemoveCategoryReply() {
    if (removeCategoryReply-&gt;error() == QPlaceReply::NoError)
        qDebug() &lt;&lt; "Removal of category identified by"
                 &lt;&lt; removeCategoryReply-&gt;id() &lt;&lt; "was successful";

    removeCategoryReply-&gt;deleteLater(); //discard reply
    removeCategoryReply = nullptr;
}

When a category is removed, the QPlaceManager may emit the QPlaceManager::categoryRemoved() signal. Whether a manager does so is provider specific. Managers accessing places from a web service will typically not emit these signals, while managers accessing places stored locally generally will.

III-11. Matching Places Between Managers

Sometimes you may want to cross reference whether places from one manager match those from another manager. Such a situation may arise where one manager provides read-only access to places (origin manager) while another second r/w manager (destination manager) is used to save selected favorites from the first. During a search of the origin manager we may want to know which ones have been 'favorited' into the destination manager and perhaps display a customized favorite name rather than the original name.

The matching mechanism can vary between managers, but is typically accomplished through an alternative identifier. As part of the save process, the place identifier from the origin manager is saved as an alternative identifier attribute in the destination manager (which can have its own place identifier scheme). In the following example, the origin manager is from the 'here' QGeoServiceProider, therefore as part of the saving process an alternative identifier attribute, x_id_here, is set for the place saved into the destination manager (when QPlaceManager::compatiblePlace() is called)

 
Sélectionnez
origin R/O manager(here)       destination R/W manager (places_jsondb)
                        Save
Place id: ae246         ---&gt;    Place id: 0001
Attribute type: x_provider      Attribute type: x_id_here
Attribute value: here           Attribute text value: ae246

In order to perform the matching, we create a QPlaceMatchRequest and assign it the search results from the origin manager. The QPlaceMatchRequest will be used on the destination manager to return corresponding places. We also specify matching parameters which are key value pairs. As mentioned previously, this can vary depending on the manager but typically the key is QPlaceMatchRequest::AlternativeId to indicate we are matching by alternative id, the value in this case would be x_id_here which specifies which alternative identifier attribute we are using to do the matching.

 
Sélectionnez
QPlaceMatchRequest request;
request.setResults(results);
QVariantMap parameters;
parameters.insert(QPlaceMatchRequest::AlternativeId, "x_id_here");
request.setParameters(parameters);
matchReply = manager-&gt;matchingPlaces(request);
    ...
    ...
void matchHandler() {
    if (matchReply-&gt;error() == QPlaceReply::NoError) {
        const auto places = matchReply-&gt;places();
        for (const QPlace &amp;place : places) {
            if (place != QPlace())
                qDebug() &lt;&lt; "Place is a favorite with name" &lt;&lt; place.name();
            else
                qDebug() &lt;&lt; "Place is not a favorite";
        }
    }

    matchReply-&gt;deleteLater();
    matchReply = nullptr;
}

IV. Classes in Places

 

IV-1. Data Classes

  • QPlace: The QPlace class represents a set of data about a place.

  • QPlaceAttribute: The QPlaceAttribute class represents generic attribute information about a place.

  • QPlaceCategory: The QPlaceCategory class represents a category that a QPlace can be associated with.

  • QPlaceContactDetail: The QPlaceContactDetail class represents a contact detail such as a phone number or website url.

  • QPlaceContent: The QPlaceContent class holds content about places.

  • QPlaceIcon: The QPlaceIcon class represents an icon.

  • QPlaceProposedSearchResult: The QPlaceProposedSearchResult class represents a search result containing a proposed search.

  • QPlaceRatings: The QPlaceRatings class holds rating information about a place.

  • QPlaceResult: The QPlaceResult class represents a search result containing a place.

  • QPlaceSearchResult: The QPlaceSearchResult class is the base class for search results.

  • QPlaceSupplier: The QPlaceSupplier class represents a supplier of a place or content associated with a place.

  • QPlaceUser: The QPlaceUser class represents an individual user.

IV-2. Request Classes

  • QPlaceContentRequest: The QPlaceContentRequest class represents the parameters of a content request.

  • QPlaceMatchRequest: The QPlaceMatchRequest class is used to find places from one manager that match those from another. It represents a set of request parameters.

  • QPlaceSearchRequest: The QPlaceSearchRequest class represents the set of parameters for a search request.

 

IV-3. Reply classes

  • QPlaceContentReply: The QPlaceContentReply class manages a content retrieval operation started by an instance of QPlaceManager.

  • QPlaceDetailsReply: The QPlaceDetailsReply class manages a place details fetch operation started by an instance of QPlaceManager.

  • QPlaceIdReply: The QPlaceIdReply class manages operations which return an identifier such as saving and removal operations of places and categories.

  • QPlaceMatchReply: The QPlaceMatchReply class manages a place matching operation started by an instance of QPlaceManager.

  • QPlaceReply: The QPlaceReply class manages an operation started by an instance of QPlaceManager and serves as a base class for more specialized replies.

  • QPlaceSearchReply: The QPlaceSearchReply class manages a place search operation started by an instance of QPlaceManager.

  • QPlaceSearchSuggestionReply: The QPlaceSearchSuggestionReply class manages a search suggestion operation started by an instance of QPlaceManager.

IV-4. Manager Classes

  • QPlaceManager: The QPlaceManager class provides the interface which allows clients to access places stored in a particular backend.

  • QPlaceManagerEngine: The QPlaceManagerEngine class provides an interface for implementers of QGeoServiceProvider plugins who want to provide access to place functionality.

Vous avez aimé ce tutoriel ? Alors partagez-le en cliquant sur les boutons suivants : Viadeo Twitter Facebook Share on Google+