Skip to content

Latest commit

 

History

History
294 lines (255 loc) · 11.5 KB

getting-directions.md

File metadata and controls

294 lines (255 loc) · 11.5 KB

Getting Directions

Now we have a simple map where you can search for locations. When finishing this step you'll be able to create a directions between two points and change the transportation mode.

Get Directions Between Two Locations

After having created our list of search results, we have a good starting point for creating directions between two Locations. Since our search only supports searching for a single location, we will hardcode a Location's coordinate into our app, and use that as the basis for our Origin. Then we'll create a route, show the navigation details, and display the route on the map.

We have already created a point in the basic example, called _userPosition to use as a starting point for directions on MapState. The position given with the example is for the main entrance to the White House, replace it with some location that fits your usecase.

final _userPosition = MPPoint.withCoordinates(
    longitude: -77.03740973527613,
    latitude: 38.897389429704695,
    floorIndex: 0);

We can extend the search functionality we made earlier by implementing the onLocationSelected listener on the MapsIndoorsWidget, and showing the selected location's details on the bottom sheet:

void onMapControlReady(MPError? error) async {
  if (error == null) {
    ...
    // Add a listener for location selection events, we do not want to stop the SDK from moving the camera, so we do not comsume the event
    _mapControl
      ..setOnLocationSelectedListener(onLocationSelected, false)
      ..goTo(await getDefaultVenue());
  } else {
    ...
  }
}

and implementing the onLocationSelected listener:

void onLocationSelected(MPLocation? location) {
  // if no location is selected, close the sheet
  if (location == null) {
    _controller?.close();
    _controller = null;
    return;
  }
  // show location details
  _controller = _scaffoldKey.currentState?.showBottomSheet((context) {
    return Container(
      color: Colors.white,
      padding: const EdgeInsets.only(bottom: 50.0, left: 100, right: 100),
      child: Column(
        mainAxisSize: MainAxisSize.min,
        children: [
          const SizedBox(
            height: 30,
          ),
          Text(location.name),
          const SizedBox(
            height: 30,
          ),
          Text("Description: ${location.description}"),
          const SizedBox(
            height: 30,
          ),
          Text("Building: ${location.buildingName} - ${location.floorName}"),
          const SizedBox(
            height: 30,
          ),
          ...
        ],
      ),
    );
  });
  _controller?.closed.then((value) => _mapControl.selectLocation(null));
}

Now we will create a class that takes a origin and destination and can handle the controls of a route. This is a rather complex class that creates a route, lets us traverse it without errors and displays the steps of the route in the bottom sheet. A minimal example would only require the constructor, the leg selected listener and the UI.

First lets set up the constructor and services, we want to create the RouteHandler with an origin, destination and the state of our Scaffold (to create a bottom sheet later). We will also set up our MPDirectionsService and MPDirectionsRenderer and generate a route immediately. finally we show the route on the Renderer.

class RouteHandler {
  RouteHandler(
      {required MPPoint origin,
      required MPPoint destination,
      required ScaffoldState scaffold}) {
    _service.getRoute(origin: origin, destination: destination).then((route) {
      _route = route;
      _renderer.setRoute(route);
      // Start some UI handling here later
    });
  }
  final _service = MPDirectionsService();
  final _renderer = MPDirectionsRenderer();
  late final MPRoute _route;
}

Now we want some controllers for the state of our route, we want a method to safely update the leg index, we want to listen to when the leg index changes and we want to be able to remove the route again. First we add a onLegSelectedListener to the class and set it on the renderer. Then we add a getter/setter for the current leg index, where we ensure that we cannot get an index that is out of bounds for the current route. and finally we add a removeRoute method that clears the route from the map.

class RouteHandler {
  RouteHandler(
      {required MPPoint origin,
      required MPPoint destination,
      required ScaffoldState scaffold}) {
    _service.getRoute(origin: origin, destination: destination).then((route) {
      ...
      _renderer.setOnLegSelectedListener(onLegSelected);
      // Start some UI handling here later
    });
  }

  ...

  // backing field for the current route leg index
  int _currentIndex = 0;

  int get currentIndex {
    return _currentIndex;
  }

  // clamp the index to be in the correct range
  set currentIndex(int index) {
    _currentIndex = index.clamp(0, _route.legs!.length - 1);
  }

  // updates the state of the routehandler if the route is updated externally, eg. by tapping the next marker on the route
  void onLegSelected(int legIndex) {
    _controller?.setState!(() => currentIndex = legIndex);
  }

  // external handle to clear the route
  void removeRoute() {
    _renderer.clear();
  }
}

Finally we want some UI interaction, so we add a ShowRoute method that creates a BottomSheet which contains some navigation icons, to move between legs of the route, and a text field to display the steps of the current leg. We also add a method to expand the route steps in a way that we can show the leg in a single string.

Then we call showRoute from our constructor. Additionally we close the BottomSheet when removing the map.

class RouteHandler {
  RouteHandler(
      {required MPPoint origin,
      required MPPoint destination,
      required ScaffoldState scaffold}) {
    _service.setTravelMode(MPDirectionsService.travelModeDriving);
    _service.getRoute(origin: origin, destination: destination).then((route) {
      ...
      showRoute(scaffold);
    });
  }
  PersistentBottomSheetController? _controller;

  ...

  // opens the route on a bottom sheet
  void showRoute(ScaffoldState scaffold) {
    _controller = scaffold.showBottomSheet((context) {
      return Container(
        color: Colors.white,
        padding: const EdgeInsets.only(top: 50.0, bottom: 50.0),
        child: Row(
          mainAxisAlignment: MainAxisAlignment.center,
          mainAxisSize: MainAxisSize.min,
          children: [
            // goes a step back on the route
            IconButton(
              onPressed: () async {
                currentIndex--;
                await _renderer.selectLegIndex(currentIndex);
              },
              icon: const Icon(Icons.keyboard_arrow_left),
              iconSize: 50,
            ),
            // displays the route instructions
            Expanded(
              child: Text(
                expandRouteSteps(_route.legs![currentIndex].steps!),
                softWrap: true,
                textAlign: TextAlign.center,
              ),
            ),
            // goes a step forward on the route
            IconButton(
              onPressed: () async {
                currentIndex++;
                await _renderer.selectLegIndex(currentIndex);
              },
              icon: const Icon(Icons.keyboard_arrow_right),
              iconSize: 50,
            ),
          ],
        ),
      );
    });
    // if the bottom sheet is closed, then clear the route
    _controller?.closed.then((val) {
      _renderer.clear();
    });
  }

  // external handle to clear the route
  void removeRoute() {
    _renderer.clear();
    _controller?.close();
  }

  // expands the step instructions into a single string for the entire leg
  String expandRouteSteps(List<MPRouteStep> steps) {
    String sum = "${steps[0].maneuver}";
    for (final step in steps.skip(1)) {
      sum += ", ${step.maneuver}";
    }
    return sum;
  }
}

With this we now have a class that will handle creating a Route and displaying it on the map for us. You can see the entire implementation of RouteHandler here: main.dart

We can now add a button on our onLocationSelected bottomsheet to generate a route from _userPosition to the selected location:

final _scaffoldKey = GlobalKey<ScaffoldState>();
RouteHandler? _routeHandler;
PersistentBottomSheetController? _controller;

void onLocationSelected(MPLocation? location) {
  ...
  // if an active route is displayed, remove it from view
  _routeHandler?.removeRoute();
  // show location details
  _controller = _scaffoldKey.currentState?.showBottomSheet((context) {
    return Container(
      ...
      child: Column(
        mainAxisSize: MainAxisSize.min,
        children: [
          ...
          // when clicked will create a route from the user position to the location
          ElevatedButton(
            onPressed: () => _routeHandler = RouteHandler(
                origin: _userPosition,
                destination: location.point,
                scaffold: _scaffoldKey.currentState!),
            child: const Row(
              children: [
                Icon(Icons.keyboard_arrow_left_rounded),
                SizedBox(
                  width: 5,
                ),
                Text("directions")
              ],
            ),
          ),
        ],
      ),
    );
  });
  _controller?.closed.then((value) => _mapControl.selectLocation(null));
}

We can now show and navigate a route from our _userPosition to a selected location.

Change Transportation Mode

In MapsIndoors, the transportation mode is referred to as travel mode. There are four travel modes, walking, bicycling, driving and transit (public transportation). The travel modes generally applies for outdoor navigation. Indoor navigation calculations are based on walking travel mode.

To swap Travel Modes you set the Travel Mode before making a query for the route:

RouteHandler(
    {required MPPoint origin,
    required MPPoint destination,
    required ScaffoldState scaffold}) {
  _service.setTravelMode(MPDirectionsService.travelModeDriving);
  _service.getRoute(origin: origin, destination: destination).then((route) {
    _route = route;
    _renderer.setRoute(route);
    _renderer.setOnLegSelectedListener(onLegSelected);
    showRoute(scaffold);
  });
}

Expected result:

The accompanying UI and implementation of this directions experience can be found in the getting started app sample. Getting Started App sample