Quyền truy cập Bluetooth

Để dùng các tính năng Bluetooth trong ứng dụng, bạn phải khai báo nhiều quyền. Bạn cũng nên chỉ định xem ứng dụng của mình có cần hỗ trợ Bluetooth cổ điển hay Bluetooth năng lượng thấp (BLE) Nếu ứng dụng của bạn không cần đến Bluetooth cổ điển hoặc BLE nhưng vẫn có thể hưởng lợi từ các công nghệ này, bạn có thể kiểm tra phạm vi cung cấp trong thời gian chạy.

Khai báo quyền

Tập hợp các quyền mà bạn khai báo trong ứng dụng phụ thuộc vào phiên bản SDK mục tiêu của ứng dụng.

Nhắm mục tiêu đến Android 12 trở lên

Lưu ý: Trên Android 8.0 (API cấp 26) trở lên, Trình quản lý thiết bị đồng hành (CDM) cung cấp một phương thức đơn giản hơn để kết nối với các thiết bị đồng hành, so với các quyền được mô tả trong phần này. Hệ thống CDM đại diện cho ứng dụng của bạn cung cấp giao diện người dùng ghép nối và không yêu cầu quyền truy cập thông tin vị trí.

Nếu bạn muốn có thêm quyền kiểm soát đối với trải nghiệm ghép nối và kết nối, hãy sử dụng các quyền được mô tả trong phần này.

Hộp thoại cấp quyền Bluetooth
Hộp thoại cấp quyền của hệ thống, yêu cầu người dùng cấp quyền cho ứng dụng khám phá, quảng cáo và kết nối với các thiết bị ở gần.

Nếu ứng dụng của bạn nhắm đến Android 12 (API cấp 31) trở lên, hãy khai báo các quyền sau đây trong tệp kê khai của ứng dụng:

  1. Nếu ứng dụng của bạn tìm thiết bị Bluetooth, chẳng hạn như các thiết bị ngoại vi BLE, hãy khai báo quyền BLUETOOTH_SCAN.
  2. Nếu ứng dụng của bạn giúp các thiết bị Bluetooth khác có thể phát hiện thiết bị hiện tại, hãy khai báo quyền BLUETOOTH_ADVERTISE.
  3. Nếu ứng dụng của bạn giao tiếp với thiết bị Bluetooth đã được ghép nối, hãy khai báo quyền BLUETOOTH_CONNECT.
  4. Đối với các nội dung khai báo cũ liên quan đến Bluetooth, hãy đặt android:maxSdkVersion thành 30. Bước tương thích của ứng dụng này giúp hệ thống chỉ cấp cho ứng dụng của bạn các quyền truy cập Bluetooth mà ứng dụng cần khi được cài đặt trên thiết bị chạy Android 12 trở lên.
  5. Nếu ứng dụng của bạn dùng kết quả quét Bluetooth để lấy thông tin vị trí thực tế, hãy khai báo quyền ACCESS_FINE_LOCATION. Nếu không, bạn có thể khẳng định chắc chắn rằng ứng dụng của mình không lấy được thông tin vị trí thực tế.

Các quyền BLUETOOTH_ADVERTISE, BLUETOOTH_CONNECTBLUETOOTH_SCANquyền khi bắt đầu chạy. Do đó, bạn phải yêu cầu người dùng phê duyệt một cách rõ ràng trong ứng dụng trước khi có thể tìm thiết bị Bluetooth, đặt thiết bị ở chế độ có thể phát hiện được với các thiết bị khác hoặc giao tiếp với các thiết bị Bluetooth đã được ghép nối. Khi ứng dụng của bạn yêu cầu ít nhất một trong các quyền này, hệ thống sẽ nhắc người dùng cho phép ứng dụng truy cập vào Thiết bị ở gần, như minh hoạ trong hình 1.

Đoạn mã sau đây minh hoạ cách khai báo quyền liên quan đến Bluetooth trong ứng dụng nếu ứng dụng đó nhắm đến Android 12 trở lên:

<manifest>
    <!-- Request legacy Bluetooth permissions on older devices. -->
    <uses-permission android:name="android.permission.BLUETOOTH"
                     android:maxSdkVersion="30" />
    <uses-permission android:name="android.permission.BLUETOOTH_ADMIN"
                     android:maxSdkVersion="30" />

    <!-- Needed only if your app looks for Bluetooth devices.
         If your app doesn't use Bluetooth scan results to derive physical
         location information, you can
         <a href="#assert-never-for-location">strongly assert that your app
         doesn't derive physical location</a>. -->
    <uses-permission android:name="android.permission.BLUETOOTH_SCAN" />

    <!-- Needed only if your app makes the device discoverable to Bluetooth
         devices. -->
    <uses-permission android:name="android.permission.BLUETOOTH_ADVERTISE" />

    <!-- Needed only if your app communicates with already-paired Bluetooth
         devices. -->
    <uses-permission android:name="android.permission.BLUETOOTH_CONNECT" />

    <!-- Needed only if your app uses Bluetooth scan results to derive physical location. -->
    <uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" />
    ...
</manifest>

Khẳng định chắc chắn rằng ứng dụng của bạn không lấy thông tin vị trí thực tế

Nếu ứng dụng của bạn không dùng kết quả quét Bluetooth để lấy thông tin vị trí thực tế, thì bạn có thể đưa ra khẳng định chắc chắn rằng ứng dụng của bạn không bao giờ sử dụng quyền truy cập Bluetooth để lấy thông tin vị trí thực tế. Để thực hiện điều này, vui lòng hoàn thành các bước sau:

  1. Thêm thuộc tính android:usesPermissionFlags vào phần khai báo quyền BLUETOOTH_SCAN và đặt giá trị của thuộc tính này thành neverForLocation.

  2. Nếu không cần thông tin vị trí cho ứng dụng, hãy xoá quyền ACCESS_FINE_LOCATION khỏi tệp kê khai của ứng dụng.

Đoạn mã sau đây cho biết cách cập nhật tệp kê khai của ứng dụng:

<manifest>
    <!-- Include "neverForLocation" only if you can strongly assert that
         your app never derives physical location from Bluetooth scan results. -->
    <uses-permission android:name="android.permission.BLUETOOTH_SCAN"
                     android:usesPermissionFlags="neverForLocation" />

    <!-- Not needed if you can strongly assert that your app never derives
         physical location from Bluetooth scan results and doesn't need location
         access for any other purpose. -->
    <strike><uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" /></strike>
    ...
</manifest>

Nhắm mục tiêu đến Android 11 trở xuống

Nếu ứng dụng của bạn nhắm đến Android 11 (API cấp 30) trở xuống, hãy khai báo các quyền sau đây trong tệp kê khai của ứng dụng:

  • Bạn cần có BLUETOOTH để thực hiện mọi hoạt động giao tiếp Bluetooth hoặc BLE, chẳng hạn như yêu cầu kết nối, chấp nhận kết nối và truyền dữ liệu.
  • ACCESS_FINE_LOCATION là cần thiết vì trên Android 11 trở xuống, bạn có thể sử dụng tính năng quét Bluetooth để thu thập thông tin về vị trí của người dùng.

Vì quyền truy cập thông tin vị trí là quyền khi bắt đầu chạy, nên bạn phải yêu cầu các quyền này trong thời gian chạy cùng với việc khai báo các quyền đó trong tệp kê khai.

Khám phá thiết bị Bluetooth ở gần

Nếu muốn ứng dụng của mình bắt đầu tính năng khám phá thiết bị hoặc thao tác với các chế độ cài đặt Bluetooth, bạn phải khai báo quyền BLUETOOTH_ADMIN. Hầu hết các ứng dụng chỉ cần quyền này để có thể khám phá các thiết bị Bluetooth cục bộ. Không sử dụng các tính năng khác do quyền này cấp, trừ phi ứng dụng là "trình quản lý nguồn" có chức năng sửa đổi chế độ cài đặt Bluetooth theo yêu cầu của người dùng. Khai báo quyền trong tệp kê khai ứng dụng. Ví dụ:

<manifest>
...
<uses-permission android:name="android.permission.BLUETOOTH_ADMIN" />
...
</manifest>

Nếu ứng dụng của bạn hỗ trợ một dịch vụ và có thể chạy trên Android 10 (API cấp 29) hoặc Android 11, thì bạn cũng phải khai báo quyền ACCESS_BACKGROUND_LOCATION để khám phá các thiết bị Bluetooth. Để biết thêm thông tin về yêu cầu này, hãy xem bài viết Truy cập thông tin vị trí ở chế độ nền.

Đoạn mã sau đây cho biết cách khai báo quyền ACCESS_BACKGROUND_LOCATION:

<manifest>
...
<uses-permission android:name="android.permission.ACCESS_BACKGROUND_LOCATION" />
...
</manifest>

Xem tài liệu tham khảo về <uses-permission> để biết thêm thông tin về cách khai báo quyền cho ứng dụng.

Chỉ định mức sử dụng tính năng Bluetooth

Nếu Bluetooth là một phần quan trọng của ứng dụng, bạn có thể thêm cờ vào tệp kê khai cho biết yêu cầu này. Phần tử <uses-feature> cho phép bạn chỉ định loại phần cứng mà ứng dụng dùng và liệu có bắt buộc hay không.

Ví dụ này cho biết cách chỉ định rằng cần phải có Bluetooth phiên bản cũ cho ứng dụng của bạn.

<uses-feature android:name="android.hardware.bluetooth" android:required="true"/>

Nếu ứng dụng của bạn dựa vào Bluetooth năng lượng thấp, bạn có thể sử dụng:

<uses-feature android:name="android.hardware.bluetooth_le" android:required="true"/>

Nếu bạn nói rằng tính năng này là bắt buộc đối với ứng dụng của mình, thì Cửa hàng Google Play sẽ ẩn ứng dụng khỏi người dùng trên những thiết bị thiếu các tính năng đó. Vì lý do này, bạn chỉ nên đặt thuộc tính bắt buộc thành true nếu ứng dụng không thể hoạt động nếu không có tính năng này.

Kiểm tra phạm vi cung cấp của các tính năng trong thời gian chạy

Để cung cấp ứng dụng cho các thiết bị không hỗ trợ Bluetooth cổ điển hoặc BLE, bạn vẫn nên thêm phần tử <uses-feature> vào tệp kê khai của ứng dụng nhưng đặt required="false". Sau đó, trong thời gian chạy, bạn có thể xác định khả năng sử dụng của tính năng bằng cách sử dụng PackageManager.hasSystemFeature():

Kotlin

// Check to see if the Bluetooth classic feature is available.
val bluetoothAvailable = packageManager.hasSystemFeature(PackageManager.FEATURE_BLUETOOTH)

// Check to see if the BLE feature is available.
val bluetoothLEAvailable = packageManager.hasSystemFeature(PackageManager.FEATURE_BLUETOOTH_LE)

Java

// Use this check to determine whether Bluetooth classic is supported on the device.
// Then you can selectively disable BLE-related features.
boolean bluetoothAvailable = getPackageManager().hasSystemFeature(PackageManager.FEATURE_BLUETOOTH);

// Use this check to determine whether BLE is supported on the device. Then
// you can selectively disable BLE-related features.
boolean bluetoothLEAvailable = getPackageManager().hasSystemFeature(PackageManager.FEATURE_BLUETOOTH_LE);