Навигация по маршруту с одним пунктом назначения

Следуйте этому руководству, чтобы проложить маршрут в своем приложении с помощью Navigation SDK для Android. В этом руководстве предполагается, что вы уже интегрировали Navigation SDK в свое приложение, как описано в разделе «Настройка проекта» .

Краткое содержание

  1. Добавьте элемент пользовательского интерфейса в свое приложение либо в виде фрагмента навигации, либо в виде представления навигации. Этот элемент пользовательского интерфейса добавляет в вашу деятельность интерактивную карту и пошаговую навигацию.
  2. Запросить разрешения на определение местоположения. Ваше приложение должно запросить разрешение на определение местоположения, чтобы определить местоположение устройства.
  3. Инициализируйте SDK, используя класс NavigationApi .
  4. Задайте пункт назначения и управляйте пошаговой навигацией с помощью класса Navigator . Это включает в себя три шага:

    • Установите пункт назначения, используя setDestination() .
    • Запустите навигацию с помощью startGuidance() .
    • Используйте getSimulator() для моделирования движения транспортного средства по маршруту для тестирования, отладки и демонстрации вашего приложения.
  5. Создайте и запустите свое приложение.

Посмотреть код

Добавьте элемент пользовательского интерфейса в свое приложение

В этом разделе рассматриваются два способа добавления интерактивной карты и пользовательского интерфейса для отображения пошаговой навигации. В большинстве случаев мы рекомендуем использовать SupportNavigationFragment , который является оболочкой для NavigationView , вместо прямого взаимодействия с NavigationView . Дополнительные сведения см. в разделе «Рекомендации по взаимодействию с навигационной картой» .

SupportNavigationFragment — это компонент пользовательского интерфейса, который отображает визуальные результаты навигации, включая интерактивную карту и пошаговые инструкции. Вы можете объявить фрагмент в файле макета XML, как показано здесь:

<?xml version="1.0" encoding="utf-8"?>
<fragment xmlns:android="http://schemas.android.com/apk/res/android"
    android:name="com.google.android.libraries.navigation.SupportNavigationFragment"
    android:id="@+id/navigation_fragment"
    android:layout_width="match_parent"
    android:layout_height="match_parent"/>

Альтернативно, вы можете сконструировать фрагмент программно, как описано в документации Android , используя FragmentActivity.getSupportFragmentManager() .

В качестве альтернативы фрагменту компонент пользовательского интерфейса для отображения карты для навигации также доступен в виде NavigationView .

Запросить разрешение на определение местоположения

В этом разделе показано, как запросить разрешение на точное определение местоположения. Более подробную информацию можно найти в руководстве по разрешениям Android .

  1. Добавьте разрешение в качестве дочернего элемента <manifest> в манифесте Android:

    <manifest xmlns:android="http://schemas.android.com/apk/res/android"
        package="com.example.navsdksingledestination">
        <uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" />
    </manifest>
    
  2. Запросите разрешения во время выполнения в своем приложении, предоставив пользователю возможность предоставить или отказать в разрешении на определение местоположения. Следующий код проверяет, предоставил ли пользователь разрешение на точное определение местоположения. Если нет, он запрашивает разрешение:

    if (ContextCompat.checkSelfPermission(this.getApplicationContext(),
            android.Manifest.permission.ACCESS_FINE_LOCATION)
                == PackageManager.PERMISSION_GRANTED) {
        mLocationPermissionGranted = true;
    } else {
        ActivityCompat.requestPermissions(this,
                new String[] { android.Manifest.permission.ACCESS_FINE_LOCATION },
                PERMISSIONS_REQUEST_ACCESS_FINE_LOCATION);
    }
    
    if (!mLocationPermissionGranted) {
        displayMessage("Error loading Navigation SDK: "
                + "The user has not granted location permission.");
        return;
    }
    
  3. Переопределите обратный вызов onRequestPermissionsResult() для обработки результата запроса разрешения:

    @Override
    public void onRequestPermissionsResult(int requestCode, @NonNull String permissions[],
                                           @NonNull int[] grantResults) {
        mLocationPermissionGranted = false;
        switch (requestCode) {
            case PERMISSIONS_REQUEST_ACCESS_FINE_LOCATION: {
                // If request is canceled, the result arrays are empty.
                if (grantResults.length > 0
                        && grantResults[0] == PackageManager.PERMISSION_GRANTED) {
                    mLocationPermissionGranted = true;
                }
            }
        }
    }
    

Инициализируйте навигационный SDK

Класс NavigationApi предоставляет логику инициализации, которая разрешает вашему приложению использовать навигацию Google. В этом разделе описано, как инициализировать навигатор, а также некоторые другие конфигурации, которые вы можете включить для своего приложения:

  1. Инициализируйте Navigation SDK и переопределите обратный вызов onNavigatorReady() чтобы начать навигацию, когда навигатор будет готов.

  2. Необязательный. Настройте приложение так, чтобы уведомления о указаниях и фоновые службы отключались, когда пользователь закрывает приложение со своего устройства. Этот выбор зависит от вашей бизнес-модели. Возможно, вы захотите использовать поведение навигатора по умолчанию, которое продолжает отображать указания по повороту и обновления местоположения, даже если приложение закрыто. Если вместо этого вы хотите отключить обновление навигации и местоположения, когда конечный пользователь закрыл приложение, вам следует использовать эту конфигурацию.

  3. Необязательный. Включить дорожные ограничения в поддерживаемых странах. Установите последнюю цифру номерного знака. Этот вызов необходимо выполнить только один раз: последующие запросы направлений продолжат его использовать. Этот вызов работает только в поддерживаемых регионах. См. раздел «Страны, поддерживаемые навигационным SDK» .

    NavigationApi.getNavigator(this, new NavigationApi.NavigatorListener() {
                /**
                 * Sets up the navigation UI when the navigator is ready for use.
                 */
                @Override
                public void onNavigatorReady(Navigator navigator) {
                    displayMessage("Navigator ready.");
                    mNavigator = navigator;
                    mNavFragment = (NavigationFragment) getFragmentManager()
                            .findFragmentById(R.id.navigation_fragment);
    
                    // Optional. Disable the guidance notifications and shut down the app
                    // and background service when the user closes the app.
                    // mNavigator.setTaskRemovedBehavior(Navigator.TaskRemovedBehavior.QUIT_SERVICE)
    
                    // Optional. Set the last digit of the car's license plate to get
                    // route restrictions for supported countries.
                    // mNavigator.setLicensePlateRestrictionInfo(getLastDigit(), "BZ");
    
                    // Set the camera to follow the device location with 'TILTED' driving view.
                    mNavFragment.getCamera().followMyLocation(Camera.Perspective.TILTED);
    
                    // Set the travel mode (DRIVING, WALKING, CYCLING, TWO_WHEELER, or TAXI).
                    mRoutingOptions = new RoutingOptions();
                    mRoutingOptions.travelMode(RoutingOptions.TravelMode.DRIVING);
    
                    // Navigate to a place, specified by Place ID.
                    navigateToPlace(SYDNEY_OPERA_HOUSE, mRoutingOptions);
                }
    
                /**
                 * Handles errors from the Navigation SDK.
                 * @param errorCode The error code returned by the navigator.
                 */
                @Override
                public void onError(@NavigationApi.ErrorCode int errorCode) {
                    switch (errorCode) {
                        case NavigationApi.ErrorCode.NOT_AUTHORIZED:
                            displayMessage("Error loading Navigation SDK: Your API key is "
                                    + "invalid or not authorized to use the Navigation SDK.");
                            break;
                        case NavigationApi.ErrorCode.TERMS_NOT_ACCEPTED:
                            displayMessage("Error loading Navigation SDK: User did not accept "
                                    + "the Navigation Terms of Use.");
                            break;
                        case NavigationApi.ErrorCode.NETWORK_ERROR:
                            displayMessage("Error loading Navigation SDK: Network error.");
                            break;
                        case NavigationApi.ErrorCode.LOCATION_PERMISSION_MISSING:
                            displayMessage("Error loading Navigation SDK: Location permission "
                                    + "is missing.");
                            break;
                        default:
                            displayMessage("Error loading Navigation SDK: " + errorCode);
                    }
                }
            });
    

Установить пункт назначения

Класс Navigator обеспечивает контроль над настройкой, запуском и остановкой навигации.

Используя Navigator полученный в предыдущем разделе, установите Waypoint назначения для этого путешествия. После расчета направлений SupportNavigationFragment отображает ломаную линию, представляющую маршрут на карте, и маркер в пункте назначения.

    private void navigateToPlace(String placeId, RoutingOptions travelMode) {
        Waypoint destination;
        try {
            destination = Waypoint.builder().setPlaceIdString(placeId).build();
        } catch (Waypoint.UnsupportedPlaceIdException e) {
            displayMessage("Error starting navigation: Place ID is not supported.");
            return;
        }

        // Create a future to await the result of the asynchronous navigator task.
        ListenableResultFuture<Navigator.RouteStatus> pendingRoute =
                mNavigator.setDestination(destination, travelMode);

        // Define the action to perform when the SDK has determined the route.
        pendingRoute.setOnResultListener(
                new ListenableResultFuture.OnResultListener<Navigator.RouteStatus>() {
                    @Override
                    public void onResult(Navigator.RouteStatus code) {
                        switch (code) {
                            case OK:
                                // Hide the toolbar to maximize the navigation UI.
                                if (getActionBar() != null) {
                                    getActionBar().hide();
                                }

                                // Enable voice audio guidance (through the device speaker).
                                mNavigator.setAudioGuidance(
                                        Navigator.AudioGuidance.VOICE_ALERTS_AND_GUIDANCE);

                                // Simulate vehicle progress along the route for demo/debug builds.
                                if (BuildConfig.DEBUG) {
                                    mNavigator.getSimulator().simulateLocationsAlongExistingRoute(
                                            new SimulationOptions().speedMultiplier(5));
                                }

                                // Start turn-by-turn guidance along the current route.
                                mNavigator.startGuidance();
                                break;
                            // Handle error conditions returned by the navigator.
                            case NO_ROUTE_FOUND:
                                displayMessage("Error starting navigation: No route found.");
                                break;
                            case NETWORK_ERROR:
                                displayMessage("Error starting navigation: Network error.");
                                break;
                            case ROUTE_CANCELED:
                                displayMessage("Error starting navigation: Route canceled.");
                                break;
                            default:
                                displayMessage("Error starting navigation: "
                                        + String.valueOf(code));
                        }
                    }
                });
    }

Создайте и запустите свое приложение

  1. Подключите Android-устройство к компьютеру. Следуйте инструкциям Android Studio о том, как запускать приложения на аппаратном устройстве. Альтернативно вы можете настроить виртуальное устройство с помощью диспетчера виртуальных устройств Android (AVD) . Выбирая эмулятор, убедитесь, что вы выбрали образ, включающий API Google.
  2. В Android Studio щелкните пункт меню «Выполнить» или значок кнопки воспроизведения. Выберите устройство, как будет предложено.

Советы по улучшению пользовательского опыта

  • Пользователь должен принять Условия использования Google Navigation, прежде чем навигация станет доступной. Это согласие требуется только один раз. По умолчанию SDK запрашивает подтверждение при первом вызове навигатора. Если вы предпочитаете, вы можете вызвать диалоговое окно «Условия обслуживания навигации» на раннем этапе процесса взаимодействия с пользователем вашего приложения, например, во время регистрации или входа в систему, TermsAndConditionsCheckOption .
  • Чтобы значительно улучшить качество навигации и точность расчетного прибытия, используйте идентификаторы мест для инициализации путевой точки, а не координат широты и долготы.
  • В этом примере путевая точка назначения извлекается из определенного идентификатора места для Сиднейского оперного театра. Вы можете использовать инструмент поиска идентификаторов мест , чтобы получить идентификаторы мест для других конкретных мест.