सुविधाएं और एपीआई

Android 17 में डेवलपर के लिए, कई नई सुविधाएं और एपीआई उपलब्ध कराए गए हैं. यहां दिए गए सेक्शन में, इन सुविधाओं के बारे में खास जानकारी दी गई है. इससे आपको इनसे जुड़े एपीआई का इस्तेमाल शुरू करने में मदद मिलेगी.

नए, बदले गए, और हटाए गए एपीआई की पूरी सूची देखने के लिए, एपीआई में अंतर दिखाने वाली रिपोर्ट पढ़ें. नए एपीआई के बारे में ज़्यादा जानने के लिए, Android API रेफ़रंस पर जाएं. नए एपीआई को हाइलाइट किया गया है, ताकि वे आसानी से दिख सकें.

आपको उन क्षेत्रों की भी समीक्षा करनी चाहिए जहां प्लैटफ़ॉर्म में हुए बदलावों का असर आपके ऐप्लिकेशन पर पड़ सकता है. ज़्यादा जानकारी के लिए, ये पेज देखें:

मुख्य फ़ंक्शन

Android 17 में, Android की मुख्य सुविधाओं से जुड़ी ये नई सुविधाएं जोड़ी गई हैं.

ProfilingManager के नए ट्रिगर

Android 17 में, ProfilingManager में कई नए सिस्टम ट्रिगर जोड़े गए हैं. इनसे आपको परफ़ॉर्मेंस से जुड़ी समस्याओं को ठीक करने के लिए ज़्यादा जानकारी वाला डेटा इकट्ठा करने में मदद मिलेगी.

नए ट्रिगर ये हैं:

  • TRIGGER_TYPE_COLD_START: ऐप्लिकेशन के कोल्ड स्टार्ट के दौरान ट्रिगर होता है. यह जवाब में कॉल स्टैक का सैंपल और सिस्टम ट्रेस, दोनों उपलब्ध कराता है.
  • TRIGGER_TYPE_OOM: यह ट्रिगर तब चालू होता है, जब कोई ऐप्लिकेशन OutOfMemoryError दिखाता है और जवाब में Java हीप डंप उपलब्ध कराता है.
  • TRIGGER_TYPE_KILL_EXCESSIVE_CPU_USAGE: यह ट्रिगर तब होता है, जब सीपीयू का इस्तेमाल असामान्य और बहुत ज़्यादा होने की वजह से किसी ऐप्लिकेशन को बंद कर दिया जाता है. साथ ही, यह जवाब में कॉल स्टैक का सैंपल उपलब्ध कराता है.

सिस्टम ट्रिगर सेट अप करने का तरीका जानने के लिए, ट्रिगर पर आधारित प्रोफ़ाइलिंग और प्रोफ़ाइलिंग डेटा को वापस पाने और उसका विश्लेषण करने के तरीके से जुड़े दस्तावेज़ देखें.

JobDebugInfo API

Android 17 में, JobDebugInfo नए एपीआई जोड़े गए हैं. इनसे डेवलपर को JobScheduler के जॉब को डीबग करने में मदद मिलेगी. जैसे, वे क्यों नहीं चल रहे हैं, वे कितने समय तक चले, और अन्य एग्रीगेट की गई जानकारी.

JobDebugInfo APIs के पहले तरीके का नाम getPendingJobReasonStats() है. यह उन वजहों का मैप दिखाता है जिनकी वजह से जॉब को अभी तक पूरा नहीं किया गया है. साथ ही, यह उन वजहों के हिसाब से, जॉब को पूरा करने में लगने वाले कुल समय की जानकारी भी दिखाता है. इस तरीके में, getPendingJobReasonsHistory() और getPendingJobReasons() तरीकों को एक साथ इस्तेमाल किया जाता है. इससे आपको यह जानकारी मिलती है कि शेड्यूल किया गया कोई जॉब उम्मीद के मुताबिक क्यों नहीं चल रहा है. हालांकि, यह जानकारी को आसानी से पाने में मदद करता है. ऐसा इसलिए, क्योंकि इसमें अवधि और जॉब के न चलने की वजह, दोनों को एक ही तरीके में उपलब्ध कराया जाता है.

उदाहरण के लिए, किसी तय किए गए jobId के लिए, यह तरीका PENDING_JOB_REASON_CONSTRAINT_CHARGING और 60,000 मि॰से॰ की अवधि दिखा सकता है. इससे पता चलता है कि बैटरी चार्ज करने की ज़रूरी शर्त पूरी न होने की वजह से, काम 60,000 मि॰से॰ तक पूरा नहीं हुआ.

निजता

Android 17 में, उपयोगकर्ता की निजता को बेहतर बनाने के लिए ये नई सुविधाएं शामिल हैं.

Android कॉन्टैक्ट पिकर

Android Contact Picker, एक स्टैंडर्ड इंटरफ़ेस है. इससे उपयोगकर्ता, आपके ऐप्लिकेशन के साथ संपर्क शेयर कर सकते हैं. यह Android 17 या इसके बाद के वर्शन वाले डिवाइसों पर उपलब्ध है. यह पिकर, READ_CONTACTS अनुमति के मुकाबले, निजता बनाए रखने का बेहतर विकल्प है. उपयोगकर्ता की पूरी पता पुस्तिका का ऐक्सेस मांगने के बजाय, आपका ऐप्लिकेशन उन डेटा फ़ील्ड के बारे में बताता है जिनकी उसे ज़रूरत है. जैसे, फ़ोन नंबर या ईमेल पते. इसके बाद, उपयोगकर्ता शेयर करने के लिए कुछ खास संपर्क चुनता है. इससे आपके ऐप्लिकेशन को सिर्फ़ चुने गए डेटा को पढ़ने का ऐक्सेस मिलता है. इससे आपको बेहतर कंट्रोल मिलता है. साथ ही, आपको एक जैसा उपयोगकर्ता अनुभव मिलता है. इसमें खोज करने, प्रोफ़ाइल स्विच करने, और एक से ज़्यादा आइटम चुनने की सुविधाएं शामिल होती हैं. इसके लिए, आपको यूज़र इंटरफ़ेस (यूआई) बनाने या उसे बनाए रखने की ज़रूरत नहीं होती.

ज़्यादा जानकारी के लिए, संपर्क चुनने वाले टूल का दस्तावेज़ देखें.

सुरक्षा

Android 17 में, डिवाइस और ऐप्लिकेशन की सुरक्षा को बेहतर बनाने के लिए ये नई सुविधाएं जोड़ी गई हैं.

Android का ऐडवांस सुरक्षा मोड (एएपीएम)

Android के ऐडवांस सुरक्षा मोड में, Android उपयोगकर्ताओं को सुरक्षा से जुड़ी नई सुविधाएं मिलती हैं. इससे उपयोगकर्ताओं को सुरक्षित रखने में मदद मिलती है. खास तौर पर, उन लोगों को जिनके खातों पर, ऑनलाइन हमलों का खतरा ज़्यादा होता है. AAPM को ऑप्ट-इन सुविधा के तौर पर डिज़ाइन किया गया है. इसे कॉन्फ़िगरेशन की एक सेटिंग से चालू किया जाता है. उपयोगकर्ता इसे किसी भी समय चालू कर सकते हैं, ताकि सुरक्षा से जुड़ी कुछ खास सेटिंग लागू की जा सकें.

इन मुख्य कॉन्फ़िगरेशन में, अज्ञात सोर्स से ऐप्लिकेशन इंस्टॉल करने (साइडलोडिंग) पर रोक लगाना, यूएसबी डेटा सिग्नलिंग को सीमित करना, और Google Play Protect की स्कैनिंग को ज़रूरी बनाना शामिल है. इससे डिवाइस पर हमला होने की आशंका काफ़ी कम हो जाती है. डेवलपर, इस सुविधा को AdvancedProtectionManager एपीआई का इस्तेमाल करके इंटिग्रेट कर सकते हैं. इससे मोड की स्थिति का पता लगाया जा सकता है. इससे ऐप्लिकेशन, उपयोगकर्ता के ऑप्ट-इन करने पर सुरक्षा से जुड़ी सेटिंग को अपने-आप लागू कर सकते हैं या ज़्यादा जोखिम वाले फ़ंक्शन को सीमित कर सकते हैं.

कनेक्टिविटी

Android 17 में, डिवाइस और ऐप्लिकेशन की कनेक्टिविटी को बेहतर बनाने के लिए ये सुविधाएं जोड़ी गई हैं.

सैटलाइट नेटवर्क की उपलब्धता सीमित होना

इस वर्शन को इस तरह से ऑप्टिमाइज़ किया गया है, ताकि ऐप्लिकेशन कम बैंडविथ वाले सैटलाइट नेटवर्क पर भी असरदार तरीके से काम कर सकें.

उपयोगकर्ता अनुभव और सिस्टम यूज़र इंटरफ़ेस (यूआई)

Android 17 में, उपयोगकर्ता अनुभव को बेहतर बनाने के लिए ये बदलाव किए गए हैं.

हैंडऑफ़

हैंडऑफ़, Android 17 में उपलब्ध होने वाली एक नई सुविधा और एपीआई है. ऐप्लिकेशन डेवलपर इसे इंटिग्रेट कर सकते हैं, ताकि वे अपने उपयोगकर्ताओं को क्रॉस-डिवाइस पर एक जैसा अनुभव दे सकें. इस कुकी की मदद से, उपयोगकर्ता किसी Android डिवाइस पर ऐप्लिकेशन से जुड़ी गतिविधि शुरू कर सकता है और उसे किसी दूसरे Android डिवाइस पर ट्रांसफ़र कर सकता है. हैंडऑफ़, उपयोगकर्ता के डिवाइस के बैकग्राउंड में काम करता है. साथ ही, उपयोगकर्ता के आस-पास मौजूद अन्य डिवाइसों पर उपलब्ध गतिविधियों को दिखाता है. ऐसा अलग-अलग एंट्री पॉइंट के ज़रिए किया जाता है. जैसे, जिस डिवाइस पर गतिविधि को ट्रांसफ़र किया जा रहा है उस पर लॉन्चर और टास्कबार.

अगर Android का नेटिव ऐप्लिकेशन, डेटा पाने वाले डिवाइस पर इंस्टॉल है और उपलब्ध है, तो ऐप्लिकेशन उसे लॉन्च करने के लिए हैंडऑफ़ की सुविधा का इस्तेमाल कर सकते हैं. ऐप्लिकेशन-टू-ऐप्लिकेशन फ़्लो में, उपयोगकर्ता को तय की गई गतिविधि से डीप लिंक किया जाता है. इसके अलावा, ऐप्लिकेशन से वेब पर रीडायरेक्ट करने की सुविधा को फ़ॉलबैक विकल्प के तौर पर उपलब्ध कराया जा सकता है. इसे यूआरएल रीडायरेक्ट करने की सुविधा के साथ सीधे तौर पर लागू किया जा सकता है.

हैंडऑफ़ की सुविधा, हर गतिविधि के हिसाब से लागू की जाती है. हैंडऑफ़ की सुविधा चालू करने के लिए, गतिविधि के लिए setHandoffEnabled() तरीके का इस्तेमाल करें. हैंडऑफ़ के साथ-साथ, कुछ और डेटा भी पास करना पड़ सकता है, ताकि डेटा पाने वाले डिवाइस पर फिर से बनाई गई ऐक्टिविटी, सही स्थिति को वापस ला सके. onHandoffActivityRequested() कॉलबैक लागू करें, ताकि HandoffActivityData ऑब्जेक्ट वापस किया जा सके. इस ऑब्जेक्ट में ऐसी जानकारी होती है जिससे यह पता चलता है कि Handoff को, गतिविधि को पाने वाले डिवाइस पर कैसे हैंडल करना चाहिए और उसे फिर से कैसे बनाना चाहिए.

लाइव अपडेट - सिमैंटिक कलर एपीआई

Android 17 में, लाइव अपडेट सुविधा के साथ सिमैंटिक कलरिंग एपीआई लॉन्च किए गए हैं. इनकी मदद से, ऐसे रंगों का इस्तेमाल किया जा सकता है जिनका मतलब सभी के लिए एक जैसा हो.

इन क्लास के लिए, सिमैंटिक कलरिंग की सुविधा उपलब्ध है:

रंग भरने से जुड़े गेम

  • हरा: सुरक्षा से जुड़ा है. इस रंग का इस्तेमाल तब किया जाना चाहिए, जब लोगों को यह बताना हो कि आप सुरक्षित हैं.
  • नारंगी: इसका इस्तेमाल, चेतावनी देने और शारीरिक खतरों को मार्क करने के लिए किया जाता है. इस रंग का इस्तेमाल तब किया जाना चाहिए, जब उपयोगकर्ताओं को बेहतर सुरक्षा सेटिंग सेट करने के लिए ध्यान देने की ज़रूरत हो.
  • लाल: आम तौर पर, यह खतरे का संकेत देता है. इसका मतलब है कि रुक जाएं. इसे ऐसे मामलों में इस्तेमाल किया जाना चाहिए जिनमें लोगों का ध्यान तुरंत खींचना ज़रूरी हो.
  • नीला: यह रंग ऐसे कॉन्टेंट के लिए इस्तेमाल किया जाता है जिसमें जानकारी दी गई हो और जिसे दूसरे कॉन्टेंट से अलग दिखाना हो.

यहां दिए गए उदाहरण में, सूचना में मौजूद टेक्स्ट पर सिमैंटिक स्टाइल लागू करने का तरीका बताया गया है:

  val ssb = SpannableStringBuilder()
        .append("Colors: ")
        .append("NONE", Notification.createSemanticStyleAnnotation(SEMANTIC_STYLE_UNSPECIFIED), 0)
        .append(", ")
        .append("INFO", Notification.createSemanticStyleAnnotation(SEMANTIC_STYLE_INFO), 0)
        .append(", ")
        .append("SAFE", Notification.createSemanticStyleAnnotation(SEMANTIC_STYLE_SAFE), 0)
        .append(", ")
        .append("CAUTION", Notification.createSemanticStyleAnnotation(SEMANTIC_STYLE_CAUTION), 0)
        .append(", ")
        .append("DANGER", Notification.createSemanticStyleAnnotation(SEMANTIC_STYLE_DANGER), 0)

    Notification.Builder(context, channelId)
          .setSmallIcon(R.drawable.ic_icon)
          .setContentTitle("Hello World!")
          .setContentText(ssb)
          .setOngoing(true)
              .setRequestPromotedOngoing(true)

Android 17 के लिए UWB Downlink-TDoA API

डाउनलिंक टाइम डिफ़रेंस ऑफ़ अराइवल (डीएल-टीडीओए) रेंजिंग की मदद से, कोई डिवाइस सिग्नल के पहुंचने के समय का पता लगाकर, कई ऐंकर के हिसाब से अपनी जगह की जानकारी का पता लगा सकता है.

इस स्निपेट में, Ranging Manager को शुरू करने, डिवाइस की क्षमताओं की पुष्टि करने, और DL-TDoA सेशन शुरू करने का तरीका दिखाया गया है:

Kotlin

class RangingApp {

    fun initDlTdoa(context: Context) {
        // Initialize the Ranging Manager
        val rangingManager = context.getSystemService(RangingManager::class.java)

        // Register for device capabilities
        val capabilitiesCallback = object : RangingManager.CapabilitiesCallback {
            override fun onRangingCapabilities(capabilities: RangingCapabilities) {
                // Make sure Dl-TDoA is supported before starting the session
                if (capabilities.uwbCapabilities != null && capabilities.uwbCapabilities!!.isDlTdoaSupported) {
                    startDlTDoASession(context)
                }
            }
        }
        rangingManager.registerCapabilitiesCallback(Executors.newSingleThreadExecutor(), capabilitiesCallback)
    }

    fun startDlTDoASession(context: Context) {

        // Initialize the Ranging Manager
        val rangingManager = context.getSystemService(RangingManager::class.java)

        // Create session and configure parameters
        val executor = Executors.newSingleThreadExecutor()
        val rangingSession = rangingManager.createRangingSession(executor, RangingSessionCallback())
        val rangingRoundIndexes = intArrayOf(0)
        val config: ByteArray = byteArrayOf() // OOB config data
        val params = DlTdoaRangingParams.createFromFiraConfigPacket(config, rangingRoundIndexes)

        val rangingDevice = RangingDevice.Builder().build()
        val rawTagDevice = RawRangingDevice.Builder()
            .setRangingDevice(rangingDevice)
            .setDlTdoaRangingParams(params)
            .build()

        val dtTagConfig = RawDtTagRangingConfig.Builder(rawTagDevice).build()

        val preference = RangingPreference.Builder(DEVICE_ROLE_DT_TAG, dtTagConfig)
            .setSessionConfig(SessionConfig.Builder().build())
            .build()

        // Start the ranging session
        rangingSession.start(preference)
    }
}

private class RangingSessionCallback : RangingSession.Callback {
    override fun onDlTdoaResults(peer: RangingDevice, measurement: DlTdoaMeasurement) {
        // Process measurement results here
    }
}

Java

public class RangingApp {

    public void initDlTdoa(Context context) {

        // Initialize the Ranging Manager
        RangingManager rangingManager = context.getSystemService(RangingManager.class);

        // Register for device capabilities
        RangingManager.CapabilitiesCallback capabilitiesCallback = new RangingManager.CapabilitiesCallback() {
            @Override
            public void onRangingCapabilities(RangingCapabilities capabilities) {
                // Make sure Dl-TDoA is supported before starting the session
                if (capabilities.getUwbCapabilities() != null && capabilities.getUwbCapabilities().isDlTdoaSupported) {
                    startDlTDoASession(context);
                }
            }
        };
        rangingManager.registerCapabilitiesCallback(Executors.newSingleThreadExecutor(), capabilitiesCallback);
    }

    public void startDlTDoASession(Context context) {
        RangingManager rangingManager = context.getSystemService(RangingManager.class);

        // Create session and configure parameters
        Executor executor = Executors.newSingleThreadExecutor();
        RangingSession rangingSession = rangingManager.createRangingSession(executor, new RangingSessionCallback());
        int[] rangingRoundIndexes = new int[] {0};
        byte[] config = new byte[0]; // OOB config data
        DlTdoaRangingParams params = DlTdoaRangingParams.createFromFiraConfigPacket(config, rangingRoundIndexes);

        RangingDevice rangingDevice = new RangingDevice.Builder().build();
        RawRangingDevice rawTagDevice = new RawRangingDevice.Builder()
                .setRangingDevice(rangingDevice)
                .setDlTdoaRangingParams(params)
                .build();

        RawDtTagRangingConfig dtTagConfig = new RawDtTagRangingConfig.Builder(rawTagDevice).build();

        RangingPreference preference = new RangingPreference.Builder(DEVICE_ROLE_DT_TAG, dtTagConfig)
                .setSessionConfig(new SessionConfig.Builder().build())
                .build();

        // Start the ranging session
        rangingSession.start(preference);
    }

    private static class RangingSessionCallback implements RangingSession.Callback {

        @Override
        public void onDlTdoaResults(RangingDevice peer, DlTdoaMeasurement measurement) {
            // Process measurement results here
        }
    }
}

आउट-ऑफ़-बैंड (ओओबी) कॉन्फ़िगरेशन

यहां दिए गए स्निपेट में, वाई-फ़ाई और बीएलई के लिए DL-TDoA OOB कॉन्फ़िगरेशन डेटा का उदाहरण दिया गया है:

Java

// Wifi Configuration
byte[] wifiConfig = {
    (byte) 0xDD, (byte) 0x2D, (byte) 0x5A, (byte) 0x18, (byte) 0xFF, // Header
    (byte) 0x5F, (byte) 0x19, // FiRa Sub-Element
    (byte) 0x02, (byte) 0x00, // Profile ID
    (byte) 0x06, (byte) 0x02, (byte) 0x20, (byte) 0x08, // MAC Address
    (byte) 0x14, (byte) 0x01, (byte) 0x0C, // Preamble Index
    (byte) 0x27, (byte) 0x02, (byte) 0x08, (byte) 0x07, // Vendor ID
    (byte) 0x28, (byte) 0x06, (byte) 0xCA, (byte) 0xC8, (byte) 0xA6, (byte) 0xF7, (byte) 0x6F, (byte) 0x08, // Static STS IV
    (byte) 0x08, (byte) 0x02, (byte) 0x60, (byte) 0x09, // Slot Duration
    (byte) 0x1B, (byte) 0x01, (byte) 0x0A, // Slots per RR
    (byte) 0x09, (byte) 0x04, (byte) 0xE8, (byte) 0x03, (byte) 0x00, (byte) 0x00, // Duration
    (byte) 0x9F, (byte) 0x04, (byte) 0x67, (byte) 0x45, (byte) 0x23, (byte) 0x01  // Session ID
};

// BLE Configuration
byte[] bleConfig = {
    (byte) 0x2D, (byte) 0x16, (byte) 0xF4, (byte) 0xFF, // Header
    (byte) 0x5F, (byte) 0x19, // FiRa Sub-Element
    (byte) 0x02, (byte) 0x00, // Profile ID
    (byte) 0x06, (byte) 0x02, (byte) 0x20, (byte) 0x08, // MAC Address
    (byte) 0x14, (byte) 0x01, (byte) 0x0C, // Preamble Index
    (byte) 0x27, (byte) 0x02, (byte) 0x08, (byte) 0x07, // Vendor ID
    (byte) 0x28, (byte) 0x06, (byte) 0xCA, (byte) 0xC8, (byte) 0xA6, (byte) 0xF7, (byte) 0x6F, (byte) 0x08, // Static STS IV
    (byte) 0x08, (byte) 0x02, (byte) 0x60, (byte) 0x09, // Slot Duration
    (byte) 0x1B, (byte) 0x01, (byte) 0x0A, // Slots per RR
    (byte) 0x09, (byte) 0x04, (byte) 0xE8, (byte) 0x03, (byte) 0x00, (byte) 0x00, // Duration
    (byte) 0x9F, (byte) 0x04, (byte) 0x67, (byte) 0x45, (byte) 0x23, (byte) 0x01  // Session ID
};

अगर ओओबी कॉन्फ़िगरेशन मौजूद नहीं है, तो इसका इस्तेमाल नहीं किया जा सकता. इसके अलावा, अगर आपको ओओबी कॉन्फ़िगरेशन में मौजूद डिफ़ॉल्ट वैल्यू बदलनी हैं, तो DlTdoaRangingParams.Builder का इस्तेमाल करके पैरामीटर बनाए जा सकते हैं. इसके बारे में यहां बताया गया है. DlTdoaRangingParams.createFromFiraConfigPacket() की जगह इन पैरामीटर का इस्तेमाल किया जा सकता है:

Kotlin

val dlTdoaParams = DlTdoaRangingParams.Builder(1)
    .setComplexChannel(UwbComplexChannel.Builder()
            .setChannel(9).setPreambleIndex(10).build())
    .setDeviceAddress(deviceAddress)
    .setSessionKeyInfo(byteArrayOf(0x01, 0x02, 0x03, 0x04))
    .setRangingIntervalMillis(240)
    .setSlotDuration(UwbRangingParams.DURATION_2_MS)
    .setSlotsPerRangingRound(20)
    .setRangingRoundIndexes(byteArrayOf(0x01, 0x05))
    .build()

Java

DlTdoaRangingParams dlTdoaParams = new DlTdoaRangingParams.Builder(1)
    .setComplexChannel(new UwbComplexChannel.Builder()
            .setChannel(9).setPreambleIndex(10).build())
    .setDeviceAddress(deviceAddress)
    .setSessionKeyInfo(new byte[]{0x01, 0x02, 0x03, 0x04})
    .setRangingIntervalMillis(240)
    .setSlotDuration(UwbRangingParams.DURATION_2_MS)
    .setSlotsPerRangingRound(20)
    .setRangingRoundIndexes(new byte[]{0x01, 0x05})
    .build();