QtLocation.qtlocation-places-example

src="https://assets.ubuntu.com/v1/0acb4a71-qml-places.png" alt="" />

Overview

The Places example presents an application window displaying a map. At the top of the window is a search box, which is used to enter a place search query. To search for a place enter a search term into the text box and click the magnifying glass icon. To search for a place by category, click the category icon to display the list of available categories and select the desired category. The place search query will be for places that are near the current location shown on the map.

The search box provides search term suggestions when three or more characters are entered. Selecting one of the suggestions will cause a place search to be performed with the selected search text.

Search results are available from the slide out tab on the left. Clicking on a search result will display details about the place. If a places has rich content (editorials, reviews and images), these can be accessed by the buttons on the details page. To find similar places click the "Find similar" button. If the current Geo service provider supports it, buttons to edit and remove a place will also be available.

The geo service provider can be changed by accessing the "Provider" menu at the bottom of the window. Depending on the features supported by the provider, the "New" menu allows creating new Places and Categories. To create a new place, select "Place" from the "New" menu and fill in the fields. Click "Go!" to save the place. To create a new category, select "Category" from the "New" menu and fill in the fields. Click "Go!" to save the category.

The Places example can work with any of the available geo services plugins. However, some plugins may require additional plugin parameters in order to function correctly. Plugin parameters can be passed on the command line using the --plugin argument, which takes the form:

--plugin.<parameter name> <parameter value>

Refer to the documentation for each of the geo services plugins for details on what plugin parameters they support. The Nokia services plugin supplied with Qt requires an app_id and token pair. See "Qt Location Nokia Plugin" for details.

Displaying Categories

Before search by category can be performed, the list of available categories needs to be retrieved. This is achieved by creating a CategoryModel.

CategoryModel {
id: categoryModel
plugin: placesPlugin
hierarchical: true
}

The CategoryModel type provides a model of the available categories. It can provide either a flat list or a hierarchical tree model. In this example, we use a hierarchical tree model, by setting the hierarchical property to true. The plugin property is set to placesPlugin which is the identifier of the Plugin object used for place search throughout the example.

Next we create a view to display the category model.

ListView {
id: root
property bool showSave: true
property bool showRemove: true
property bool showChildren: true
signal categoryClicked(variant category)
signal editClicked(variant category)
header: IconButton {
source: "../../resources/left.png"
pressedSource: "../../resources/left_pressed.png"
onClicked: categoryListModel.rootIndex = categoryListModel.parentModelIndex()
}
model: VisualDataModel {
id: categoryListModel
model: categoryModel
delegate: CategoryDelegate {
id: categoryDelegate
showSave: root.showSave
showRemove: root.showRemove
showChildren: root.showChildren
onClicked: root.categoryClicked(category);
onArrowClicked: categoryListModel.rootIndex = categoryListModel.modelIndex(index)
onCrossClicked: category.remove();
onEditClicked: root.editClicked(category);
}
}
}

Because a hierarchical model is being used, a DelegateModel is needed to provide navigation functionality. If flat list model was being used the view could use the CategoryModel directly.

The view contains a header item that is used as a back button to navigate up the category tree. The onClicked handler sets the root index of the DelegateModel to the parent of the current index. Categories are displayed by the CategoryDelegate, which provides four signals. The onArrowClicked handler sets the root index to the current index causing the sub categories of the selected category to be displayed. The onClicked handler emits the categoryClicked() signal with a category parameter indicating which specific category has been chosen. The onCrossClicked handler will invoke the categories remove() method. The onEditClicked handler invokes the editClicked() signal of the root item, this is used to notify which particular category is to be edited.

The CategoryDelegate displays the category name and emits the clicked signal when the text is clicked:

Text {
id: name
anchors.left: icon.right
anchors.verticalCenter: parent.verticalCenter
anchors.right: parent.right
verticalAlignment: Text.AlignVCenter
text: category.name
elide: Text.ElideRight
}
MouseArea {
anchors.fill: parent
onClicked: root.clicked()
}

The CategoryDelegate also displays icons for editing, removing and displaying child categories. These icons are shown as desired when the showSave and showRemove and showChildren properties are set and only then in cases where the function is supported.

IconButton {
id: edit
anchors.right: cross.left
anchors.verticalCenter: parent.verticalCenter
visible: (placesPlugin.name != "" ? placesPlugin.supportsPlaces(Plugin.SaveCategoryFeature) : false)
&& showSave
source: "../../resources/pencil.png"
hoveredSource: "../../resources/pencil_hovered.png"
pressedSource: "../../resources/pencil_pressed.png"
onClicked: root.editClicked()
}
IconButton {
id: cross
anchors.right: arrow.left
anchors.verticalCenter: parent.verticalCenter
visible: (placesPlugin.name != "" ? placesPlugin.supportsPlaces(Plugin.RemoveCategoryFeature) : false)
&& showRemove
source: "../../resources/cross.png"
hoveredSource: "../../resources/cross_hovered.png"
pressedSource: "../../resources/cross_pressed.png"
onClicked: root.crossClicked()
}
IconButton {
id: arrow
anchors.right: parent.right
anchors.verticalCenter: parent.verticalCenter
visible: model.hasModelChildren && showChildren
source: "../../resources/right.png"
pressedSource: "../../resources/right_pressed.png"
onClicked: root.arrowClicked()
}

Presenting Search Suggestions

The PlaceSearchSuggestionModel type is used to fetch suggested search terms based on a partially entered search term.

A new suggestion search is triggered whenever the entered search term is changed.

onTextChanged: {
if (searchRectangle.suggestionsEnabled) {
if (text.length >= 3) {
if (suggestionModel != null) {
suggestionModel.searchTerm = text;
suggestionModel.update();
}
} else {
searchRectangle.state = "";
}
}
}

The suggestionsEnabled property is used to temporarily disable search suggestions when a suggestion is selected (selecting it updates the search term text). Suggestions are only queried if the length of the search term is three or more characters, otherwise the search boxes state is reset.

When the status of the PlaceSearchSuggestionModel changes, the state of the search box is changed to display the search suggestions.

PlaceSearchSuggestionModel {
id: suggestionModel
plugin: placesPlugin
searchArea: placeSearchModel.searchArea
onStatusChanged: {
if (status == PlaceSearchSuggestionModel.Ready)
searchRectangle.state = "SuggestionsShown";
}
}

The main object in the "SuggestionsShown" state is the ListView showing the search suggestions.

ListView {
id: suggestionView
model: suggestionModel
delegate: Text {
text: suggestion
width: parent.width
MouseArea {
anchors.fill: parent
onClicked: {
suggestionsEnabled = false;
searchBox.text = suggestion;
suggestionsEnabled = true;
placeSearchModel.searchForText(suggestion);
searchRectangle.state = "";
}
}
}
}

A Text object is used as the delegate to display the suggestion text. Clicking on the suggested search term updates the search term and triggers a place search using the search suggestion.

Searching for Places

The PlaceSearchModel type is used to search for places.

PlaceSearchModel {
id: placeSearchModel
plugin: placesPlugin
searchArea: searchRegion
function searchForCategory(category) {
searchTerm = "";
categories = category;
recommendationId = "";
searchArea = searchRegion
limit = -1;
update();
}
function searchForText(text) {
searchTerm = text;
categories = null;
recommendationId = "";
searchArea = searchRegion
limit = -1;
update();
}
function searchForRecommendations(placeId) {
searchTerm = "";
categories = null;
recommendationId = placeId;
searchArea = null;
limit = -1;
update();
}
onStatusChanged: {
switch (status) {
case PlaceSearchModel.Ready:
searchResultView.showSearchResults();
break;
case PlaceSearchModel.Error:
console.log(errorString());
}
}
}

First some of the model's properties are set, which will be used to form the search request. The searchArea property is set to the searchRegion object which is a GeoCircle with a center that is linked to the current location displayed on the Map.

Finally, we define two helper functions searchForCategory() and searchForText(), which set either the categories or searchTerm properties and invokes the update() method to start the place search. The search results are displayed in a ListView.

ListView {
id: searchView
anchors.fill: parent
model: placeSearchModel
delegate: SearchResultDelegate {
onDisplayPlaceDetails: showPlaceDetails(data)
onSearchFor: placeSearchModel.searchForText(query);
}
footer: Item {
width: searchView.width
height: childrenRect.height
Button {
text: qsTr("Previous")
enabled: placeSearchModel.previousPagesAvailable
onClicked: placeSearchModel.previousPage()
anchors.left: parent.left
}
Button {
text: qsTr("Clear")
onClicked: placeSearchModel.reset()
anchors.horizontalCenter: parent.horizontalCenter
}
Button {
text: qsTr("Next")
enabled: placeSearchModel.nextPagesAvailable
onClicked: placeSearchModel.nextPage()
anchors.right: parent.right
}
}
}

The delegate used in the ListView, SearchResultDelegate, is designed to handle multiple search result types via a Loader object. For results of type PlaceResult the delegate is:

Component {
id: placeComponent
Item {
id: placeRoot
height: childrenRect.height
width: parent.width
Rectangle {
anchors.fill: parent
color: "#dbffde"
visible: model.sponsored !== undefined ? model.sponsored : false
Text {
text: qsTr("Sponsored result")
horizontalAlignment: Text.AlignRight
anchors.right: parent.right
anchors.bottom: parent.bottom
font.pixelSize: 8
visible: model.sponsored !== undefined ? model.sponsored : false
}
}
Row {
Image {
source: place.favorite ? class="string">&quot;https://assets.ubuntu.com/v1/b5b8205a-star.png" : place.icon.url()
}
Column {
anchors.verticalCenter: parent.verticalCenter
Text {
id: placeName
text: place.favorite ? place.favorite.name : place.name
}
Text {
id: distanceText
font.italic: true
text: PlacesUtils.prettyDistance(distance)
}
}
}
MouseArea {
anchors.fill: parent
onPressed: placeRoot.state = "Pressed"
onReleased: placeRoot.state = ""
onCanceled: placeRoot.state = ""
onClicked: {
if (model.type === undefined || type === PlaceSearchModel.PlaceResult) {
if (!place.detailsFetched)
place.getDetails();
root.displayPlaceDetails({
distance: model.distance,
place: model.place,
});
}
}
}
states: [
State {
name: ""
},
State {
name: "Pressed"
PropertyChanges { target: placeName; color: "#1C94FC"}
PropertyChanges { target: distanceText; color: "#1C94FC"}
}
]
}
}

Displaying Place Content

Places can have additional rich content, including editorials, reviews and images. Rich content is accessed via a set of models. Content models are generally not created directly by the application developer, instead models are obtained from the editorialModel, reviewModel and imageModel properties of the Place type.

ListView {
anchors.fill: parent
model: place.editorialModel
delegate: EditorialDelegate { }
}

Place and Category Creation

Some backends may support creation and saving of new places and categories. Plugin support can be checked an run-time with the Plugin::supportsPlaces() method.

To save a new place, first create a new Place object, using the Qt.createQmlObject() method. Assign the appropriate plugin and place properties and invoke the save() method.

        locationPlace.plugin = placesPlugin;
locationPlace.name = dataFieldsModel.get(0).inputText;
locationPlace.location.address.street = dataFieldsModel.get(1).inputText;
locationPlace.location.address.district = dataFieldsModel.get(2).inputText;
locationPlace.location.address.city = dataFieldsModel.get(3).inputText;
locationPlace.location.address.county = dataFieldsModel.get(4).inputText;
locationPlace.location.address.state = dataFieldsModel.get(5).inputText;
locationPlace.location.address.countryCode = dataFieldsModel.get(6).inputText;
locationPlace.location.address.country = dataFieldsModel.get(7).inputText;
locationPlace.location.address.postalCode = dataFieldsModel.get(8).inputText;
var c = QtPositioning.coordinate(parseFloat(dataFieldsModel.get(9).inputText),
parseFloat(dataFieldsModel.get(10).inputText));
locationPlace.location.coordinate = c;
var phone = Qt.createQmlObject('import QtLocation 5.3; ContactDetail { }', locationPlace);
phone.label = "Phone";
phone.value = dataFieldsModel.get(11).inputText;
locationPlace.contactDetails.phone = phone;
var fax = Qt.createQmlObject('import QtLocation 5.3; ContactDetail { }', locationPlace);
fax.label = "Fax";
fax.value = dataFieldsModel.get(12).inputText;
locationPlace.contactDetails.fax = fax;
var email = Qt.createQmlObject('import QtLocation 5.3; ContactDetail { }', locationPlace);
email.label = "Email";
email.value = dataFieldsModel.get(13).inputText;
locationPlace.contactDetails.email = email;
var website = Qt.createQmlObject('import QtLocation 5.3; ContactDetail { }', locationPlace);
website.label = "Website";
website.value = dataFieldsModel.get(14).inputText;
locationPlace.contactDetails.website = website;
locationPlace.categories = __categories;
locationPlace.statusChanged.connect(processStatus);
locationPlace.save();

Category creation is similar:

onGoButtonClicked: {
var modifiedCategory = category ? category : Qt.createQmlObject('import QtLocation 5.3; Category { }', page);
modifiedCategory.plugin = placesPlugin;
modifiedCategory.name = dialogModel.get(0).inputText;
category = modifiedCategory;
category.save();
}

Support for place and category removal can be checked at run-time by using the Plugin::supportsPlaces method, passing in a Plugin::PlacesFeatures flag and getting back true if the feature is supported. For example one would invoke supportsPlaces(Plugin.RemovePlaceFeature) to check if the Plugin.RemovePlaceFeature is supported.

To remove a place, invoke its remove() method. To remove a category, invoke its remove() method.

Running the Example

The example detects which plugins are available and has an option to show them in the via the Provider button.

Files:

  • places/places.qml
  • places/qmlplaceswrapper.cpp
  • places/content/places/CategoryDelegate.qml
  • places/content/places/CategoryDialog.qml
  • places/content/places/CategoryView.qml
  • places/content/places/EditorialDelegate.qml
  • places/content/places/EditorialPage.qml
  • places/content/places/Group.qml
  • places/content/places/MapComponent.qml
  • places/content/places/OptionsDialog.qml
  • places/content/places/PlaceDelegate.qml
  • places/content/places/PlaceDialog.qml
  • places/content/places/PlaceEditorials.qml
  • places/content/places/PlaceImages.qml
  • places/content/places/PlaceReviews.qml
  • places/content/places/PlacesUtils.js
  • places/content/places/RatingView.qml
  • places/content/places/ReviewDelegate.qml
  • places/content/places/ReviewPage.qml
  • places/content/places/SearchBox.qml
  • places/content/places/SearchResultDelegate.qml
  • places/content/places/SearchResultView.qml
  • places/places.pro
  • places/placeswrapper.qrc