التنقّل في مسار متعدّد الوجهات

اتّبِع هذا الدليل لرسم مسار داخل تطبيقك إلى وجهات متعدّدة، تُعرف أيضًا باسم نقاط الطريق، باستخدام "حزمة تطوير البرامج للتنقّل على أجهزة Android".

نظرة عامة

  1. يمكنك دمج حزمة تطوير البرامج (SDK) الخاصة بخدمة Navigation في تطبيقك، كما هو موضّح في إعداد مشروعك.
  2. أضِف SupportNavigationFragment أو NavigationView إلى تطبيقك، لأنّ عنصر واجهة المستخدم هذا يضيف خريطة تفاعلية وواجهة مستخدم للتنقّل خطوة بخطوة إلى نشاطك.
  3. استخدِم الفئة NavigationApi لإعداد حزمة SDK.
  4. حدِّد Navigator للتحكّم في ميزة "التنقّل باتّجاهات مفصّلة":

    • أضِف وجهات باستخدام setDestinations().
    • ابدأ التنقّل باستخدام startGuidance().
    • استخدِم getSimulator() لمحاكاة تقدّم المركبة على طول المسار، وذلك لاختبار تطبيقك وتصحيح أخطائه وعرضه.
  5. إنشاء تطبيقك وتشغيله

الاطّلاع على الرمز

إضافة جزء للتنقّل

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. في معظم الحالات، ننصحك باستخدام SupportNavigationFragment، وهي أداة تغليف NavigationView، بدلاً من التفاعل مباشرةً مع NavigationView. لمزيد من المعلومات، يُرجى الاطّلاع على أفضل الممارسات للتفاعل مع خريطة التنقّل .

طلب إذن تحديد الموقع الجغرافي

يجب أن يطلب تطبيقك إذن تحديد الموقع الجغرافي لتحديد موقع الجهاز.

يوفّر هذا البرنامج التعليمي الرمز البرمجي الذي تحتاج إليه لطلب إذن تحديد الموقع الجغرافي بدقة. لمزيد من التفاصيل، يُرجى الاطّلاع على دليل أذونات Android.

  1. أضِف الإذن كعنصر فرعي من العنصر <manifest> في بيان Android:

        <manifest xmlns:android="http://schemas.android.com/apk/res/android"
            package="com.example.navsdkmultidestination">
            <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.", DISPLAY_BOTH);
            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;
                    }
                }
            }
        }
    

بدء حزمة تطوير البرامج Navigation SDK وإعداد رحلة

يوفّر الصف NavigationApi منطق الإعداد الذي يمنح تطبيقك الإذن باستخدام ميزة التنقّل من Google. يوفّر صف Navigator عناصر تحكّم في عملية ضبط رحلة التنقّل وبدئها وإيقافها.

  1. أنشئ طريقة مساعدة لعرض رسالة على الشاشة وفي السجلّ.

        private void displayMessage(String errorMessage, String displayMedium) {
            if (displayMedium.equals(DISPLAY_BOTH) || displayMedium.equals(DISPLAY_TOAST)) {
                Toast.makeText(this, errorMessage, Toast.LENGTH_LONG).show();
            }
    
            if (displayMedium.equals(DISPLAY_BOTH) || displayMedium.equals(DISPLAY_LOG)) {
                Log.d(TAG, errorMessage);
            }
        }
    
  2. يمكنك إعداد حزمة تطوير البرامج للتنقّل وتجاوز معاودة الاتصال onNavigatorReady() لبدء التنقّل عندما يكون برنامج التنقّل جاهزًا:

        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.", DISPLAY_BOTH);
                        mNavigator = navigator;
                        mNavFragment = (SupportNavigationFragment) getFragmentManager()
                                .findFragmentById(R.id.navigation_fragment);
    
                        // Set the camera to follow the device location with 'TILTED' driving view.
                        mNavFragment.getCamera().followMyLocation(Camera.Perspective.TILTED);
    
                        // Navigate to the specified places.
                        navigateToPlaces();
                    }
    
                    /**
                     * 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.",
                                        DISPLAY_BOTH);
                                break;
                            case NavigationApi.ErrorCode.TERMS_NOT_ACCEPTED:
                                displayMessage("Error loading Navigation SDK: User did not accept "
                                        + "the Navigation Terms of Use.", DISPLAY_BOTH);
                                break;
                            case NavigationApi.ErrorCode.NETWORK_ERROR:
                                displayMessage("Error loading Navigation SDK: Network error.",
                                        DISPLAY_BOTH);
                                break;
                            case NavigationApi.ErrorCode.LOCATION_PERMISSION_MISSING:
                                displayMessage("Error loading Navigation SDK: Location permission "
                                        + "is missing.", DISPLAY_BOTH);
                                break;
                            default:
                                displayMessage("Error loading Navigation SDK: " + errorCode,
                                        DISPLAY_BOTH);
                        }
                    }
                });
    
  3. أضِف طريقة لإنشاء كائن Waypoint من معرّف مكان وعنوان محدّدين.

        private void createWaypoint(String placeId, String title) {
            try {
                mWaypoints.add(
                  Waypoint.builder()
                         .setPlaceIdString(placeId)
                         .setTitle(title)
                         .build()
                );
            } catch (Waypoint.UnsupportedPlaceIdException e) {
                displayMessage("Error starting navigation: Place ID is not supported: " + placeId,
                        DISPLAY_BOTH);
            }
        }
    
  4. أضِف طريقة لعرض وقت الرحلة والمسافة المحسوبَين إلى كل نقطة طريق.

        private void displayTimesAndDistances() {
            List<TimeAndDistance> timesAndDistances = mNavigator.getTimeAndDistanceList();
            int leg = 1;
            String message = "You're on your way!";
            for (TimeAndDistance timeAndDistance : timesAndDistances) {
                message = message + "\nRoute leg: " + leg++
                        + ": Travel time (seconds): " + timeAndDistance.getSeconds()
                        + ". Distance (meters): " + timeAndDistance.getMeters();
            }
            displayMessage(message, DISPLAY_BOTH);
        }
    
  5. اضبط جميع نقاط الطريق لهذه الرحلة. (يُرجى العِلم أنّه قد يظهر لك خطأ إذا استخدمت معرّفات أماكن لا يمكن لأداة التنقّل رسم مسار لها. يستخدم التطبيق النموذجي في هذا البرنامج التعليمي معرّفات الأماكن لنقاط الطريق في أستراليا. (راجِع الملاحظات أدناه حول الحصول على معرّفات أماكن مختلفة). بعد احتساب الاتجاهات، تعرض SupportNavigationFragment خطًا متعدد الأضلاع يمثّل المسار على الخريطة، مع علامة في كل نقطة طريق.

        private void navigateToPlaces() {
    
            // Set up a waypoint for each place that we want to go to.
            createWaypoint("ChIJq6qq6jauEmsRJAf7FjrKnXI", "Sydney Star");
            createWaypoint("ChIJ3S-JXmauEmsRUcIaWtf4MzE", "Sydney Opera House");
            createWaypoint("ChIJLwgLFGmuEmsRzpDhHQuyyoU", "Sydney Conservatorium of Music");
    
            // If this journey is already in progress, no need to restart navigation.
            // This can happen when the user rotates the device, or sends the app to the background.
            if (mSavedInstanceState != null
                    && mSavedInstanceState.containsKey(KEY_JOURNEY_IN_PROGRESS)
                    && mSavedInstanceState.getInt(KEY_JOURNEY_IN_PROGRESS) == 1) {
                return;
            }
    
            // Create a future to await the result of the asynchronous navigator task.
            ListenableResultFuture<Navigator.RouteStatus> pendingRoute =
                    mNavigator.setDestinations(mWaypoints);
    
            // 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:
                                    mJourneyInProgress = true;
                                    // Hide the toolbar to maximize the navigation UI.
                                    if (getActionBar() != null) {
                                        getActionBar().hide();
                                    }
    
                                    // Register some listeners for navigation events.
                                    registerNavigationListeners();
    
                                    // Display the time and distance to each waypoint.
                                    displayTimesAndDistances();
    
                                    // 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.",
                                            DISPLAY_BOTH);
                                    break;
                                case NETWORK_ERROR:
                                    displayMessage("Error starting navigation: Network error.",
                                            DISPLAY_BOTH);
                                    break;
                                case ROUTE_CANCELED:
                                    displayMessage("Error starting navigation: Route canceled.",
                                            DISPLAY_BOTH);
                                    break;
                                default:
                                    displayMessage("Error starting navigation: "
                                            + String.valueOf(code), DISPLAY_BOTH);
                            }
                        }
                    });
        }
    

إنشاء تطبيقك وتشغيله

  1. وصِّل جهاز Android بجهاز الكمبيوتر. اتّبِع التعليمات لتفعيل خيارات المطوّرين على جهاز Android وإعداد نظامك لرصد الجهاز. (يمكنك بدلاً من ذلك استخدام مدير أجهزة Android الافتراضية (AVD) لضبط جهاز افتراضي. عند اختيار محاكي، تأكَّد من اختيار صورة تتضمّن واجهات برمجة التطبيقات من Google.
  2. في Android Studio، انقر على خيار القائمة تشغيل (Run) (أو رمز زر التشغيل). اختَر جهازًا حسب التعليمات.

تلميحات لتحسين تجربة المستخدم

  • يجب أن يوافق المستخدم على بنود خدمة &quot;خرائط Google&quot; قبل أن تصبح ميزة التنقّل متاحة. يجب الموافقة على هذه الشروط مرة واحدة فقط. تطلب حزمة SDK الموافقة تلقائيًا عند استدعاء أداة التنقّل للمرة الأولى. إذا كنت تفضّل ذلك، يمكنك عرض مربّع حوار "بنود خدمة التنقّل" في مرحلة مبكرة من مسار تجربة المستخدم في تطبيقك، مثلاً أثناء الاشتراك أو تسجيل الدخول، وذلك باستخدام showTermsAndConditionsDialog().
  • تتحسّن جودة التنقّل ودقة الوقت المقدَّر للوصول بشكل كبير إذا كنت تستخدم معرّفات الأماكن لتهيئة نقطة طريق، بدلاً من استخدام خطوط الطول والعرض كوجهة.
  • يستمدّ هذا النموذج نقاط المرور من معرّفات أماكن معيّنة. تشمل الطرق الأخرى للحصول على معرّف مكان ما يلي:

  • استخدِم أداة البحث عن أرقام تعريف الأماكن للحصول على أرقام تعريف الأماكن لمواقع جغرافية محدّدة.

  • استخدِم Geocoding API للعثور على معرّف المكان لعنوان معيّن. تعمل واجهة برمجة التطبيقات Geocoding API بشكل جيد إذا كانت لديك عناوين كاملة وواضحة لنقاط الطريق. راجِع دليل أفضل الممارسات المتعلّقة بالترميز الجغرافي.

  • استخدِم البحث النصي في Places API للعثور على معرّف المكان لعنوان معيّن. تعمل Places API بشكل جيد إذا كانت لديك عناوين غير مكتملة أو غامضة لنقاط الطريق. راجِع دليل أفضل الممارسات المتعلّقة بالترميز الجغرافي.