মাল্টি-ডিসপ্লে কমিউনিকেশন এপিআই AAOS-এর একটি সিস্টেমের সুবিধাপ্রাপ্ত অ্যাপ ব্যবহার করতে পারে একটি গাড়িতে ভিন্ন অকুপ্যান্ট জোনে চলমান একই অ্যাপের (একই প্যাকেজের নাম) সাথে যোগাযোগ করতে। এই পৃষ্ঠাটি বর্ণনা করে কিভাবে API একত্রিত করতে হয়। আরও জানতে, আপনি CarOccupantZoneManager.OccupantZoneInfo দেখতে পারেন।
দখলদার অঞ্চল
একটি দখলকারী অঞ্চলের ধারণা একজন ব্যবহারকারীকে প্রদর্শনের একটি সেটে ম্যাপ করে। প্রতিটি দখলকারী অঞ্চলে DISPLAY_TYPE_MAIN টাইপ সহ একটি প্রদর্শন রয়েছে। একটি অকুপ্যান্ট জোনে অতিরিক্ত ডিসপ্লে থাকতে পারে, যেমন একটি ক্লাস্টার ডিসপ্লে। প্রতিটি দখলকারী অঞ্চলে একজন অ্যান্ড্রয়েড ব্যবহারকারীকে বরাদ্দ করা হয়। প্রতিটি ব্যবহারকারীর নিজস্ব অ্যাকাউন্ট এবং অ্যাপ রয়েছে।
হার্ডওয়্যার কনফিগারেশন
Comms API শুধুমাত্র একটি একক SoC সমর্থন করে। একক SoC মডেলে, সমস্ত দখলকারী অঞ্চল এবং ব্যবহারকারীরা একই SoC-তে চলে। Comms API তিনটি উপাদান নিয়ে গঠিত:
পাওয়ার ম্যানেজমেন্ট API ক্লায়েন্টকে অকুপ্যান্ট জোনে প্রদর্শনের শক্তি পরিচালনা করতে দেয়।
ডিসকভারি এপিআই ক্লায়েন্টকে গাড়িতে থাকা অন্যান্য অকুপ্যান্ট জোনের অবস্থা পর্যবেক্ষণ করতে এবং সেই অকুপ্যান্ট জোনে পিয়ার ক্লায়েন্টদের নিরীক্ষণ করতে দেয়। সংযোগ API ব্যবহার করার আগে আবিষ্কার API ব্যবহার করুন.
কানেকশন এপিআই ক্লায়েন্টকে তার পিয়ার ক্লায়েন্টের সাথে অন্য অকুপ্যান্ট জোনে সংযোগ করতে এবং পিয়ার ক্লায়েন্টকে একটি পেলোড পাঠাতে দেয়।
সংযোগের জন্য ডিসকভারি API এবং সংযোগ API প্রয়োজন৷ পাওয়ার ম্যানেজমেন্ট API ঐচ্ছিক।
Comms API বিভিন্ন অ্যাপের মধ্যে যোগাযোগ সমর্থন করে না। পরিবর্তে, এটি শুধুমাত্র একই প্যাকেজ নামের অ্যাপগুলির মধ্যে যোগাযোগের জন্য ডিজাইন করা হয়েছে এবং শুধুমাত্র বিভিন্ন দৃশ্যমান ব্যবহারকারীদের মধ্যে যোগাযোগের জন্য ব্যবহার করা হয়েছে৷
ইন্টিগ্রেশন গাইড
AbstractReceiverService প্রয়োগ করুন
Payload
পেতে, রিসিভার অ্যাপটিকে অবশ্যই AbstractReceiverService
এ সংজ্ঞায়িত বিমূর্ত পদ্ধতি প্রয়োগ করতে হবে। যেমন:
public class MyReceiverService extends AbstractReceiverService {
@Override
public void onConnectionInitiated(@NonNull OccupantZoneInfo senderZone) {
}
@Override
public void onPayloadReceived(@NonNull OccupantZoneInfo senderZone,
@NonNull Payload payload) {
}
}
onConnectionInitiated()
ডাকা হয় যখন প্রেরক ক্লায়েন্ট এই রিসিভার ক্লায়েন্টের সাথে একটি সংযোগের অনুরোধ করে। সংযোগ স্থাপনের জন্য ব্যবহারকারীর নিশ্চিতকরণের প্রয়োজন হলে, MyReceiverService
একটি অনুমতি কার্যকলাপ চালু করতে এই পদ্ধতিটিকে ওভাররাইড করতে পারে এবং ফলাফলের উপর ভিত্তি করে acceptConnection()
অথবা rejectConnection()
কল করতে পারে। অন্যথায়, MyReceiverService
শুধুমাত্র acceptConnection()
কল করতে পারে ।
onPayloadReceived()
ডাকা হয় যখন MyReceiverService
প্রেরক ক্লায়েন্টের কাছ থেকে একটি Payload
পেয়েছে। MyReceiverService
এই পদ্ধতিতে ওভাররাইড করতে পারে :
-
Payload
সংশ্লিষ্ট রিসিভার এন্ডপয়েন্টে ফরোয়ার্ড করুন, যদি থাকে। নিবন্ধিত রিসিভার এন্ডপয়েন্ট পেতে,getAllReceiverEndpoints()
কল করুন। একটি প্রদত্ত রিসিভার এন্ডপয়েন্টেPayload
ফরোয়ার্ড করতে,forwardPayload()
বা,
-
Payload
ক্যাশে করুন এবং প্রত্যাশিত রিসিভার এন্ডপয়েন্ট নিবন্ধিত হলে এটি পাঠান, যার জন্যMyReceiverService
onReceiverRegistered()
এর মাধ্যমে অবহিত করা হয়।
AbstractReceiverService ঘোষণা করুন
রিসিভার অ্যাপটিকে অবশ্যই তার ম্যানিফেস্ট ফাইলে বাস্তবায়িত AbstractReceiverService
ঘোষণা করতে হবে, এই পরিষেবার জন্য android.car.intent.action.RECEIVER_SERVICE
অ্যাকশন সহ একটি ইন্টেন্ট ফিল্টার যোগ করতে হবে এবং android.car.occupantconnection.permission.BIND_RECEIVER_SERVICE
অনুমতির প্রয়োজন হবে।
<service android:name=".MyReceiverService"
android:permission="android.car.occupantconnection.permission.BIND_RECEIVER_SERVICE"
android:exported="true">
<intent-filter>
<action android:name="android.car.intent.action.RECEIVER_SERVICE" />
</intent-filter>
</service>
android.car.occupantconnection.permission.BIND_RECEIVER_SERVICE
অনুমতি নিশ্চিত করে যে শুধুমাত্র ফ্রেমওয়ার্ক এই পরিষেবার সাথে আবদ্ধ হতে পারে। এই পরিষেবাটির অনুমতির প্রয়োজন না হলে, একটি ভিন্ন অ্যাপ এই পরিষেবার সাথে আবদ্ধ হতে পারে এবং সরাসরি এটিতে একটি Payload
পাঠাতে পারে৷
অনুমতি ঘোষণা করুন
ক্লায়েন্ট অ্যাপটিকে অবশ্যই তার ম্যানিফেস্ট ফাইলে অনুমতিগুলি ঘোষণা করতে হবে৷
<!-- This permission is needed for connection API -->
<uses-permission android:name="android.car.permission.MANAGE_OCCUPANT_CONNECTION"/>
<!-- This permission is needed for discovery API -->
<uses-permission android:name="android.car.permission.MANAGE_REMOTE_DEVICE"/>
<!-- This permission is needed if the client app calls CarRemoteDeviceManager#setOccupantZonePower() -->
<uses-permission android:name="android.car.permission.CAR_POWER"/>
উপরের তিনটি অনুমতির প্রত্যেকটিই বিশেষ সুবিধাপ্রাপ্ত অনুমতি, যেগুলিকে অনুমোদনের তালিকার ফাইলগুলিকে আগে থেকে দেওয়া আবশ্যক৷ উদাহরণস্বরূপ, এখানে MultiDisplayTest
অ্যাপের মঞ্জুরি তালিকা ফাইল রয়েছে:
// packages/services/Car/data/etc/com.google.android.car.multidisplaytest.xml
<permissions>
<privapp-permissions package="com.google.android.car.multidisplaytest">
… …
<permission name="android.car.permission.MANAGE_OCCUPANT_CONNECTION"/>
<permission name="android.car.permission.MANAGE_REMOTE_DEVICE"/>
<permission name="android.car.permission.CAR_POWER"/>
</privapp-permissions>
</permissions>
কার ম্যানেজার পান
API ব্যবহার করার জন্য, ক্লায়েন্ট অ্যাপটিকে অবশ্যই একটি CarServiceLifecycleListener
নিবন্ধন করতে হবে সংশ্লিষ্ট গাড়ি পরিচালকদের পেতে:
private CarRemoteDeviceManager mRemoteDeviceManager;
private CarOccupantConnectionManager mOccupantConnectionManager;
private final Car.CarServiceLifecycleListener mCarServiceLifecycleListener = (car, ready) -> {
if (!ready) {
Log.w(TAG, "Car service crashed");
mRemoteDeviceManager = null;
mOccupantConnectionManager = null;
return;
}
mRemoteDeviceManager = car.getCarManager(CarRemoteDeviceManager.class);
mOccupantConnectionManager = car.getCarManager(CarOccupantConnectionManager.class);
};
Car.createCar(getContext(), /* handler= */ null, Car.CAR_WAIT_TIMEOUT_WAIT_FOREVER,
mCarServiceLifecycleListener);
(প্রেরক) আবিষ্কার করুন
রিসিভার ক্লায়েন্টের সাথে সংযোগ করার আগে, প্রেরক ক্লায়েন্টকে একটি CarRemoteDeviceManager.StateCallback
নিবন্ধন করে রিসিভার ক্লায়েন্ট আবিষ্কার করতে হবে:
// The maps are accessed by the main thread only, so there is no multi-thread issue.
private final ArrayMap<OccupantZoneInfo, Integer> mOccupantZoneStateMap = new ArrayMap<>();
private final ArrayMap<OccupantZoneInfo, Integer> mAppStateMap = new ArrayMap<>();
private final StateCallback mStateCallback = new StateCallback() {
@Override
public void onOccupantZoneStateChanged(
@androidx.annotation.NonNull OccupantZoneInfo occupantZone,
int occupantZoneStates) {
mOccupantZoneStateMap.put(occupantZone, occupantZoneStates);
}
@Override
public void onAppStateChanged(
@androidx.annotation.NonNull OccupantZoneInfo occupantZone,
int appStates) {
mAppStateMap.put(occupantZone, appStates);
}
};
if (mRemoteDeviceManager != null) {
mRemoteDeviceManager.registerStateCallback(getActivity().getMainExecutor(),
mStateCallback);
}
রিসিভারের সাথে সংযোগের অনুরোধ করার আগে, প্রেরককে নিশ্চিত করতে হবে যে রিসিভার অকুপ্যান্ট জোন এবং রিসিভার অ্যাপের সমস্ত পতাকা সেট করা আছে। অন্যথায়, ত্রুটি ঘটতে পারে। যেমন:
private boolean canRequestConnectionToReceiver(OccupantZoneInfo receiverZone) {
Integer zoneState = mOccupantZoneStateMap.get(receiverZone);
if ((zoneState == null) || (zoneState.intValue() & (FLAG_OCCUPANT_ZONE_POWER_ON
// FLAG_OCCUPANT_ZONE_SCREEN_UNLOCKED is not implemented yet. Right now
// just ignore this flag.
// | FLAG_OCCUPANT_ZONE_SCREEN_UNLOCKED
| FLAG_OCCUPANT_ZONE_CONNECTION_READY)) == 0) {
return false;
}
Integer appState = mAppStateMap.get(receiverZone);
if ((appState == null) ||
(appState.intValue() & (FLAG_CLIENT_INSTALLED
| FLAG_CLIENT_SAME_LONG_VERSION | FLAG_CLIENT_SAME_SIGNATURE
| FLAG_CLIENT_RUNNING | FLAG_CLIENT_IN_FOREGROUND)) == 0) {
return false;
}
return true;
}
আমরা প্রেরককে রিসিভারের সাথে সংযোগের অনুরোধ করার পরামর্শ দিই যখন রিসিভারের সমস্ত পতাকা সেট করা থাকে। যে বলেছে, ব্যতিক্রম আছে:
FLAG_OCCUPANT_ZONE_CONNECTION_READY
এবংFLAG_CLIENT_INSTALLED
হল একটি সংযোগ স্থাপনের জন্য প্রয়োজনীয় নূন্যতম প্রয়োজনীয়তা৷সংযোগের ব্যবহারকারীর অনুমোদন পেতে যদি রিসিভার অ্যাপটিকে একটি UI প্রদর্শন করতে হয়, তাহলে
FLAG_OCCUPANT_ZONE_POWER_ON
এবংFLAG_OCCUPANT_ZONE_SCREEN_UNLOCKED
অতিরিক্ত প্রয়োজনীয়তা হয়ে যাবে। একটি ভাল ব্যবহারকারীর অভিজ্ঞতার জন্য,FLAG_CLIENT_RUNNING
এবংFLAG_CLIENT_IN_FOREGROUND
ও সুপারিশ করা হয়, অন্যথায় ব্যবহারকারী অবাক হতে পারেন৷আপাতত (Android 15),
FLAG_OCCUPANT_ZONE_SCREEN_UNLOCKED
প্রয়োগ করা হয়নি৷ ক্লায়েন্ট অ্যাপ্লিকেশন শুধু এটি উপেক্ষা করতে পারেন.আপাতত (Android 15), Comms API শুধুমাত্র একই অ্যান্ড্রয়েড ইন্সট্যান্সে একাধিক ব্যবহারকারীকে সমর্থন করে যাতে পিয়ার অ্যাপগুলির একই দীর্ঘ সংস্করণ কোড (
FLAG_CLIENT_SAME_LONG_VERSION
) এবং স্বাক্ষর (FLAG_CLIENT_SAME_SIGNATURE
) থাকতে পারে৷ ফলস্বরূপ, অ্যাপগুলিকে যাচাই করতে হবে না যে দুটি মান একমত।
একটি ভাল ব্যবহারকারীর অভিজ্ঞতার জন্য, প্রেরক ক্লায়েন্ট একটি UI দেখাতে পারে যদি একটি পতাকা সেট না থাকে৷ উদাহরণস্বরূপ, যদি FLAG_OCCUPANT_ZONE_SCREEN_UNLOCKED
সেট করা না থাকে, তাহলে প্রেরক একটি টোস্ট বা একটি ডায়ালগ দেখাতে পারেন যাতে ব্যবহারকারীকে রিসিভার অকুপ্যান্ট জোনের স্ক্রীনটি আনলক করার অনুরোধ জানানো হয়।
যখন প্রেরকের আর রিসিভারগুলি আবিষ্কার করার প্রয়োজন হয় না (উদাহরণস্বরূপ, যখন এটি সমস্ত রিসিভার এবং প্রতিষ্ঠিত সংযোগগুলি খুঁজে পায় বা নিষ্ক্রিয় হয়ে যায়), তখন এটি আবিষ্কার বন্ধ করতে পারে৷
if (mRemoteDeviceManager != null) {
mRemoteDeviceManager.unregisterStateCallback();
}
আবিষ্কার বন্ধ হয়ে গেলে, বিদ্যমান সংযোগগুলি প্রভাবিত হয় না। প্রেরক সংযুক্ত রিসিভারদের Payload
পাঠানো চালিয়ে যেতে পারেন।
(প্রেরক) সংযোগের অনুরোধ করুন
যখন রিসিভারের সমস্ত পতাকা সেট করা হয়, প্রেরক রিসিভারের সাথে একটি সংযোগের অনুরোধ করতে পারেন:
private final ConnectionRequestCallback mRequestCallback = new ConnectionRequestCallback() {
@Override
public void onConnected(OccupantZoneInfo receiverZone) {
}
@Override
public void onFailed(OccupantZoneInfo receiverZone, int connectionError) {
}
@Override
public void onDisconnected(OccupantZoneInfo receiverZone) {
}
};
if (mOccupantConnectionManager != null && canRequestConnectionToReceiver(receiverZone)) {
mOccupantConnectionManager.requestConnection(receiverZone,
getActivity().getMainExecutor(), mRequestCallback);
}
(রিসিভার পরিষেবা) সংযোগ গ্রহণ করুন
প্রেরক একবার রিসিভারের সাথে সংযোগের অনুরোধ করলে, রিসিভার অ্যাপের AbstractReceiverService
গাড়ি পরিষেবা দ্বারা আবদ্ধ হবে এবং AbstractReceiverService.onConnectionInitiated()
আহ্বান করা হবে। যেমনটি (প্রেরক) অনুরোধ সংযোগে ব্যাখ্যা করা হয়েছে, onConnectionInitiated()
একটি বিমূর্ত পদ্ধতি এবং ক্লায়েন্ট অ্যাপ দ্বারা প্রয়োগ করা আবশ্যক।
যখন প্রাপক সংযোগের অনুরোধ গ্রহণ করে, প্রেরকের ConnectionRequestCallback.onConnected()
ডাকা হবে, তারপর সংযোগ স্থাপন করা হবে।
(প্রেরক) পেলোড পাঠান
সংযোগ স্থাপন হয়ে গেলে, প্রেরক রিসিভারের কাছে Payload
পাঠাতে পারেন:
if (mOccupantConnectionManager != null) {
Payload payload = ...;
try {
mOccupantConnectionManager.sendPayload(receiverZone, payload);
} catch (CarOccupantConnectionManager.PayloadTransferException e) {
Log.e(TAG, "Failed to send Payload to " + receiverZone);
}
}
প্রেরক Payload
একটি Binder
অবজেক্ট বা একটি বাইট অ্যারে রাখতে পারেন। প্রেরকের যদি অন্য ডেটা টাইপ পাঠাতে হয়, তবে এটিকে অবশ্যই একটি বাইট অ্যারেতে ডেটা সিরিয়ালাইজ করতে হবে, একটি Payload
অবজেক্ট তৈরি করতে বাইট অ্যারে ব্যবহার করতে হবে এবং Payload
পাঠাতে হবে। তারপর রিসিভার ক্লায়েন্ট প্রাপ্ত Payload
থেকে বাইট অ্যারে পায় এবং প্রত্যাশিত ডেটা অবজেক্টে বাইট অ্যারেকে ডিসিরিয়ালাইজ করে। উদাহরণস্বরূপ, যদি প্রেরক আইডি FragmentB
সহ রিসিভার এন্ডপয়েন্টে একটি স্ট্রিং hello
পাঠাতে চান, তাহলে এটি এই ধরনের ডেটা টাইপ সংজ্ঞায়িত করতে প্রোটো বাফার ব্যবহার করতে পারে:
message MyData {
required string receiver_endpoint_id = 1;
required string data = 2;
}
চিত্র 1 Payload
প্রবাহকে চিত্রিত করে:
(রিসিভার পরিষেবা) পেলোড গ্রহণ করুন এবং প্রেরণ করুন
একবার রিসিভার অ্যাপ Payload
গ্রহণ করলে, এর AbstractReceiverService.onPayloadReceived()
চালু করা হবে। Send the payload- এ যেমন ব্যাখ্যা করা হয়েছে, onPayloadReceived()
হল একটি বিমূর্ত পদ্ধতি এবং ক্লায়েন্ট অ্যাপ দ্বারা প্রয়োগ করা আবশ্যক। এই পদ্ধতিতে, ক্লায়েন্ট সংশ্লিষ্ট রিসিভার এন্ডপয়েন্টে Payload
ফরোয়ার্ড করতে পারে, অথবা Payload
ক্যাশে করে প্রত্যাশিত রিসিভার এন্ডপয়েন্ট নিবন্ধিত হয়ে গেলে এটি পাঠাতে পারে।
(রিসিভার এন্ডপয়েন্ট) নিবন্ধন করুন এবং নিবন্ধনমুক্ত করুন
রিসিভার এন্ডপয়েন্ট রেজিস্টার করতে রিসিভার অ্যাপটিকে registerReceiver()
কল করা উচিত। একটি সাধারণ ব্যবহারের ক্ষেত্রে একটি ফ্র্যাগমেন্টের রিসিভার Payload
প্রয়োজন হয়, তাই এটি একটি রিসিভার এন্ডপয়েন্ট নিবন্ধন করে:
private final PayloadCallback mPayloadCallback = (senderZone, payload) -> {
…
};
if (mOccupantConnectionManager != null) {
mOccupantConnectionManager.registerReceiver("FragmentB",
getActivity().getMainExecutor(), mPayloadCallback);
}
রিসিভার ক্লায়েন্টের AbstractReceiverService
একবার রিসিভার এন্ডপয়েন্টে Payload
পাঠালে, সংশ্লিষ্ট PayloadCallback
আহ্বান করা হবে।
ক্লায়েন্ট অ্যাপ একাধিক রিসিভার এন্ডপয়েন্ট নিবন্ধন করতে পারে যতক্ষণ না তাদের receiverEndpointId
আইডি ক্লায়েন্ট অ্যাপের মধ্যে অনন্য। কোন রিসিভার এন্ডপয়েন্ট(গুলি) পেলোডটি পাঠাতে হবে তা নির্ধারণ করতে AbstractReceiverService
ব্যবহার করবে receiverEndpointId
। যেমন:
- প্রেরক
Payload
receiver_endpoint_id:FragmentB
নির্দিষ্ট করে।Payload
গ্রহণ করার সময়, রিসিভারে থাকাAbstractReceiverService
FragmentB
-তে পেলোড পাঠানোর জন্যforwardPayload("FragmentB", payload)
কল করে - প্রেরক
Payload
data_type:VOLUME_CONTROL
নির্দিষ্ট করে।Payload
গ্রহণ করার সময়, রিসিভারে থাকাAbstractReceiverService
জানে যে এই ধরনেরPayload
FragmentB
তে পাঠানো উচিত, তাই এটিforwardPayload("FragmentB", payload)
কল করে
if (mOccupantConnectionManager != null) {
mOccupantConnectionManager.unregisterReceiver("FragmentB");
}
(প্রেরক) সংযোগটি বন্ধ করুন
একবার প্রেরকের আর রিসিভারের কাছে Payload
পাঠাতে হবে না (উদাহরণস্বরূপ, এটি নিষ্ক্রিয় হয়ে যায়), এটি সংযোগটি বন্ধ করে দিতে হবে।
if (mOccupantConnectionManager != null) {
mOccupantConnectionManager.disconnect(receiverZone);
}
একবার সংযোগ বিচ্ছিন্ন হয়ে গেলে, প্রেরক আর রিসিভারের কাছে Payload
পাঠাতে পারবেন না।
সংযোগ প্রবাহ
একটি সংযোগ প্রবাহ চিত্র 2 এ চিত্রিত করা হয়েছে।
সমস্যা সমাধান
লগ চেক করুন
সংশ্লিষ্ট লগ চেক করতে:
লগিংয়ের জন্য এই কমান্ডটি চালান:
adb shell setprop log.tag.CarRemoteDeviceService VERBOSE && adb shell setprop log.tag.CarOccupantConnectionService VERBOSE && adb logcat -s "AbstractReceiverService","CarOccupantConnectionManager","CarRemoteDeviceManager","CarRemoteDeviceService","CarOccupantConnectionService"
CarRemoteDeviceService
এবংCarOccupantConnectionService
এর অভ্যন্তরীণ অবস্থা ডাম্প করতে:adb shell dumpsys car_service --services CarRemoteDeviceService && adb shell dumpsys car_service --services CarOccupantConnectionService
নাল CarRemoteDeviceManager এবং CarOccupantConnectionManager
এই সম্ভাব্য মূল কারণগুলি দেখুন:
গাড়ি পরিষেবা বিপর্যস্ত। পূর্বে চিত্রিত হিসাবে, গাড়ি পরিষেবা ক্র্যাশ হলে দুই পরিচালক ইচ্ছাকৃতভাবে
null
হতে পুনরায় সেট করা হয়। যখন গাড়ি পরিষেবা পুনঃসূচনা করা হয়, তখন দুটি পরিচালক নন-নাল মানগুলিতে সেট করা হয়৷হয়
CarRemoteDeviceService
বাCarOccupantConnectionService
সক্ষম করা নেই৷ এক বা অন্যটি সক্ষম কিনা তা নির্ধারণ করতে, চালান:adb shell dumpsys car_service --services CarFeatureController
mDefaultEnabledFeaturesFromConfig
দেখুন, যাতেcar_remote_device_service
এবংcar_occupant_connection_service
থাকা উচিত। যেমন:mDefaultEnabledFeaturesFromConfig:[car_evs_service, car_navigation_service, car_occupant_connection_service, car_remote_device_service, car_telemetry_service, cluster_home_service, com.android.car.user.CarUserNoticeService, diagnostic, storage_monitoring, vehicle_map_service]
ডিফল্টরূপে, এই দুটি পরিষেবা অক্ষম করা হয়। যখন একটি ডিভাইস মাল্টি-ডিসপ্লে সমর্থন করে, তখন আপনাকে অবশ্যই এই কনফিগারেশন ফাইলটি ওভারলে করতে হবে। আপনি একটি কনফিগারেশন ফাইলে দুটি পরিষেবা সক্ষম করতে পারেন:
// packages/services/Car/service/res/values/config.xml <string-array translatable="false" name="config_allowed_optional_car_features"> <item>car_occupant_connection_service</item> <item>car_remote_device_service</item> … … </string-array>
API কল করার সময় ব্যতিক্রম
যদি ক্লায়েন্ট অ্যাপটি উদ্দেশ্য অনুযায়ী API ব্যবহার না করে, তাহলে একটি ব্যতিক্রম ঘটতে পারে। এই ক্ষেত্রে, ক্লায়েন্ট অ্যাপ সমস্যাটি সমাধান করতে ব্যতিক্রম এবং ক্র্যাশ স্ট্যাকের বার্তাটি পরীক্ষা করতে পারে। API অপব্যবহারের উদাহরণ হল:
-
registerStateCallback()
এই ক্লায়েন্টটি ইতিমধ্যে একটিStateCallback
নিবন্ধন করেছে। -
unregisterStateCallback()
এইCarRemoteDeviceManager
দৃষ্টান্ত দ্বারা কোনStateCallback
নিবন্ধিত হয়নি। -
registerReceiver()
receiverEndpointId
ইতিমধ্যেই নিবন্ধিত। -
unregisterReceiver()
receiverEndpointId
নিবন্ধিত নয়। -
requestConnection()
একটি মুলতুবি বা প্রতিষ্ঠিত সংযোগ ইতিমধ্যেই বিদ্যমান। -
cancelConnection()
বাতিল করার জন্য কোনো মুলতুবি সংযোগ নেই। -
sendPayload()
কোনো প্রতিষ্ঠিত সংযোগ নেই। -
disconnect()
কোনো প্রতিষ্ঠিত সংযোগ নেই।
ক্লায়েন্ট 1 পেলোড ক্লায়েন্ট 2 এ পাঠাতে পারে, তবে অন্যভাবে নয়
সংযোগ নকশা দ্বারা এক উপায়. দ্বি-মুখী সংযোগ স্থাপনের জন্য, client1
এবং client2
উভয়কেই একে অপরের সাথে সংযোগের অনুরোধ করতে হবে এবং তারপরে অনুমোদন পেতে হবে।
মাল্টি-ডিসপ্লে কমিউনিকেশন এপিআই AAOS-এর একটি সিস্টেমের সুবিধাপ্রাপ্ত অ্যাপ ব্যবহার করতে পারে একটি গাড়িতে ভিন্ন অকুপ্যান্ট জোনে চলমান একই অ্যাপের (একই প্যাকেজের নাম) সাথে যোগাযোগ করতে। এই পৃষ্ঠাটি বর্ণনা করে কিভাবে API একত্রিত করতে হয়। আরও জানতে, আপনি CarOccupantZoneManager.OccupantZoneInfo দেখতে পারেন।
দখলদার অঞ্চল
একটি দখলকারী অঞ্চলের ধারণা একজন ব্যবহারকারীকে প্রদর্শনের একটি সেটে ম্যাপ করে। প্রতিটি দখলকারী অঞ্চলে DISPLAY_TYPE_MAIN টাইপ সহ একটি প্রদর্শন রয়েছে। একটি অকুপ্যান্ট জোনে অতিরিক্ত ডিসপ্লে থাকতে পারে, যেমন একটি ক্লাস্টার ডিসপ্লে। প্রতিটি দখলকারী অঞ্চলে একজন অ্যান্ড্রয়েড ব্যবহারকারীকে বরাদ্দ করা হয়। প্রতিটি ব্যবহারকারীর নিজস্ব অ্যাকাউন্ট এবং অ্যাপ রয়েছে।
হার্ডওয়্যার কনফিগারেশন
Comms API শুধুমাত্র একটি একক SoC সমর্থন করে। একক SoC মডেলে, সমস্ত দখলকারী অঞ্চল এবং ব্যবহারকারীরা একই SoC-তে চলে। Comms API তিনটি উপাদান নিয়ে গঠিত:
পাওয়ার ম্যানেজমেন্ট API ক্লায়েন্টকে অকুপ্যান্ট জোনে প্রদর্শনের শক্তি পরিচালনা করতে দেয়।
ডিসকভারি এপিআই ক্লায়েন্টকে গাড়িতে থাকা অন্যান্য অকুপ্যান্ট জোনের অবস্থা পর্যবেক্ষণ করতে এবং সেই অকুপ্যান্ট জোনে পিয়ার ক্লায়েন্টদের নিরীক্ষণ করতে দেয়। সংযোগ API ব্যবহার করার আগে আবিষ্কার API ব্যবহার করুন.
কানেকশন এপিআই ক্লায়েন্টকে তার পিয়ার ক্লায়েন্টের সাথে অন্য অকুপ্যান্ট জোনে সংযোগ করতে এবং পিয়ার ক্লায়েন্টকে একটি পেলোড পাঠাতে দেয়।
সংযোগের জন্য ডিসকভারি API এবং সংযোগ API প্রয়োজন৷ পাওয়ার ম্যানেজমেন্ট API ঐচ্ছিক।
Comms API বিভিন্ন অ্যাপের মধ্যে যোগাযোগ সমর্থন করে না। পরিবর্তে, এটি শুধুমাত্র একই প্যাকেজ নামের অ্যাপগুলির মধ্যে যোগাযোগের জন্য ডিজাইন করা হয়েছে এবং শুধুমাত্র বিভিন্ন দৃশ্যমান ব্যবহারকারীদের মধ্যে যোগাযোগের জন্য ব্যবহার করা হয়েছে৷
ইন্টিগ্রেশন গাইড
AbstractReceiverService প্রয়োগ করুন
Payload
পেতে, রিসিভার অ্যাপটিকে অবশ্যই AbstractReceiverService
এ সংজ্ঞায়িত বিমূর্ত পদ্ধতি প্রয়োগ করতে হবে। যেমন:
public class MyReceiverService extends AbstractReceiverService {
@Override
public void onConnectionInitiated(@NonNull OccupantZoneInfo senderZone) {
}
@Override
public void onPayloadReceived(@NonNull OccupantZoneInfo senderZone,
@NonNull Payload payload) {
}
}
onConnectionInitiated()
ডাকা হয় যখন প্রেরক ক্লায়েন্ট এই রিসিভার ক্লায়েন্টের সাথে একটি সংযোগের অনুরোধ করে। সংযোগ স্থাপনের জন্য ব্যবহারকারীর নিশ্চিতকরণের প্রয়োজন হলে, MyReceiverService
একটি অনুমতি কার্যকলাপ চালু করতে এই পদ্ধতিটিকে ওভাররাইড করতে পারে এবং ফলাফলের উপর ভিত্তি করে acceptConnection()
অথবা rejectConnection()
কল করতে পারে। অন্যথায়, MyReceiverService
শুধুমাত্র acceptConnection()
কল করতে পারে ।
onPayloadReceived()
ডাকা হয় যখন MyReceiverService
প্রেরক ক্লায়েন্টের কাছ থেকে একটি Payload
পেয়েছে। MyReceiverService
এই পদ্ধতিতে ওভাররাইড করতে পারে :
-
Payload
সংশ্লিষ্ট রিসিভার এন্ডপয়েন্টে ফরোয়ার্ড করুন, যদি থাকে। নিবন্ধিত রিসিভার এন্ডপয়েন্ট পেতে,getAllReceiverEndpoints()
কল করুন। একটি প্রদত্ত রিসিভার এন্ডপয়েন্টেPayload
ফরোয়ার্ড করতে,forwardPayload()
বা,
-
Payload
ক্যাশে করুন এবং প্রত্যাশিত রিসিভার এন্ডপয়েন্ট নিবন্ধিত হলে এটি পাঠান, যার জন্যMyReceiverService
onReceiverRegistered()
এর মাধ্যমে অবহিত করা হয়।
AbstractReceiverService ঘোষণা করুন
রিসিভার অ্যাপটিকে অবশ্যই তার ম্যানিফেস্ট ফাইলে বাস্তবায়িত AbstractReceiverService
ঘোষণা করতে হবে, এই পরিষেবার জন্য android.car.intent.action.RECEIVER_SERVICE
অ্যাকশন সহ একটি ইন্টেন্ট ফিল্টার যোগ করতে হবে এবং android.car.occupantconnection.permission.BIND_RECEIVER_SERVICE
অনুমতির প্রয়োজন হবে।
<service android:name=".MyReceiverService"
android:permission="android.car.occupantconnection.permission.BIND_RECEIVER_SERVICE"
android:exported="true">
<intent-filter>
<action android:name="android.car.intent.action.RECEIVER_SERVICE" />
</intent-filter>
</service>
android.car.occupantconnection.permission.BIND_RECEIVER_SERVICE
অনুমতি নিশ্চিত করে যে শুধুমাত্র ফ্রেমওয়ার্ক এই পরিষেবার সাথে আবদ্ধ হতে পারে। এই পরিষেবাটির অনুমতির প্রয়োজন না হলে, একটি ভিন্ন অ্যাপ এই পরিষেবার সাথে আবদ্ধ হতে পারে এবং সরাসরি এটিতে একটি Payload
পাঠাতে পারে৷
অনুমতি ঘোষণা করুন
ক্লায়েন্ট অ্যাপটিকে অবশ্যই তার ম্যানিফেস্ট ফাইলে অনুমতিগুলি ঘোষণা করতে হবে৷
<!-- This permission is needed for connection API -->
<uses-permission android:name="android.car.permission.MANAGE_OCCUPANT_CONNECTION"/>
<!-- This permission is needed for discovery API -->
<uses-permission android:name="android.car.permission.MANAGE_REMOTE_DEVICE"/>
<!-- This permission is needed if the client app calls CarRemoteDeviceManager#setOccupantZonePower() -->
<uses-permission android:name="android.car.permission.CAR_POWER"/>
উপরের তিনটি অনুমতির প্রত্যেকটিই বিশেষ সুবিধাপ্রাপ্ত অনুমতি, যেগুলিকে অনুমোদনের তালিকার ফাইলগুলিকে আগে থেকে দেওয়া আবশ্যক৷ উদাহরণস্বরূপ, এখানে MultiDisplayTest
অ্যাপের মঞ্জুরি তালিকা ফাইল রয়েছে:
// packages/services/Car/data/etc/com.google.android.car.multidisplaytest.xml
<permissions>
<privapp-permissions package="com.google.android.car.multidisplaytest">
… …
<permission name="android.car.permission.MANAGE_OCCUPANT_CONNECTION"/>
<permission name="android.car.permission.MANAGE_REMOTE_DEVICE"/>
<permission name="android.car.permission.CAR_POWER"/>
</privapp-permissions>
</permissions>
কার ম্যানেজার পান
API ব্যবহার করার জন্য, ক্লায়েন্ট অ্যাপটিকে অবশ্যই একটি CarServiceLifecycleListener
নিবন্ধন করতে হবে সংশ্লিষ্ট গাড়ি পরিচালকদের পেতে:
private CarRemoteDeviceManager mRemoteDeviceManager;
private CarOccupantConnectionManager mOccupantConnectionManager;
private final Car.CarServiceLifecycleListener mCarServiceLifecycleListener = (car, ready) -> {
if (!ready) {
Log.w(TAG, "Car service crashed");
mRemoteDeviceManager = null;
mOccupantConnectionManager = null;
return;
}
mRemoteDeviceManager = car.getCarManager(CarRemoteDeviceManager.class);
mOccupantConnectionManager = car.getCarManager(CarOccupantConnectionManager.class);
};
Car.createCar(getContext(), /* handler= */ null, Car.CAR_WAIT_TIMEOUT_WAIT_FOREVER,
mCarServiceLifecycleListener);
(প্রেরক) আবিষ্কার করুন
রিসিভার ক্লায়েন্টের সাথে সংযোগ করার আগে, প্রেরক ক্লায়েন্টকে একটি CarRemoteDeviceManager.StateCallback
নিবন্ধন করে রিসিভার ক্লায়েন্ট আবিষ্কার করতে হবে:
// The maps are accessed by the main thread only, so there is no multi-thread issue.
private final ArrayMap<OccupantZoneInfo, Integer> mOccupantZoneStateMap = new ArrayMap<>();
private final ArrayMap<OccupantZoneInfo, Integer> mAppStateMap = new ArrayMap<>();
private final StateCallback mStateCallback = new StateCallback() {
@Override
public void onOccupantZoneStateChanged(
@androidx.annotation.NonNull OccupantZoneInfo occupantZone,
int occupantZoneStates) {
mOccupantZoneStateMap.put(occupantZone, occupantZoneStates);
}
@Override
public void onAppStateChanged(
@androidx.annotation.NonNull OccupantZoneInfo occupantZone,
int appStates) {
mAppStateMap.put(occupantZone, appStates);
}
};
if (mRemoteDeviceManager != null) {
mRemoteDeviceManager.registerStateCallback(getActivity().getMainExecutor(),
mStateCallback);
}
রিসিভারের সাথে সংযোগের অনুরোধ করার আগে, প্রেরককে নিশ্চিত করতে হবে যে রিসিভার অকুপ্যান্ট জোন এবং রিসিভার অ্যাপের সমস্ত পতাকা সেট করা আছে। অন্যথায়, ত্রুটি ঘটতে পারে। যেমন:
private boolean canRequestConnectionToReceiver(OccupantZoneInfo receiverZone) {
Integer zoneState = mOccupantZoneStateMap.get(receiverZone);
if ((zoneState == null) || (zoneState.intValue() & (FLAG_OCCUPANT_ZONE_POWER_ON
// FLAG_OCCUPANT_ZONE_SCREEN_UNLOCKED is not implemented yet. Right now
// just ignore this flag.
// | FLAG_OCCUPANT_ZONE_SCREEN_UNLOCKED
| FLAG_OCCUPANT_ZONE_CONNECTION_READY)) == 0) {
return false;
}
Integer appState = mAppStateMap.get(receiverZone);
if ((appState == null) ||
(appState.intValue() & (FLAG_CLIENT_INSTALLED
| FLAG_CLIENT_SAME_LONG_VERSION | FLAG_CLIENT_SAME_SIGNATURE
| FLAG_CLIENT_RUNNING | FLAG_CLIENT_IN_FOREGROUND)) == 0) {
return false;
}
return true;
}
আমরা প্রেরককে রিসিভারের সাথে সংযোগের অনুরোধ করার পরামর্শ দিই যখন রিসিভারের সমস্ত পতাকা সেট করা থাকে। যে বলেছে, ব্যতিক্রম আছে:
FLAG_OCCUPANT_ZONE_CONNECTION_READY
এবংFLAG_CLIENT_INSTALLED
হল একটি সংযোগ স্থাপনের জন্য প্রয়োজনীয় নূন্যতম প্রয়োজনীয়তা৷সংযোগের ব্যবহারকারীর অনুমোদন পেতে যদি রিসিভার অ্যাপটিকে একটি UI প্রদর্শন করতে হয়, তাহলে
FLAG_OCCUPANT_ZONE_POWER_ON
এবংFLAG_OCCUPANT_ZONE_SCREEN_UNLOCKED
অতিরিক্ত প্রয়োজনীয়তা হয়ে যাবে। একটি ভাল ব্যবহারকারীর অভিজ্ঞতার জন্য,FLAG_CLIENT_RUNNING
এবংFLAG_CLIENT_IN_FOREGROUND
ও সুপারিশ করা হয়, অন্যথায় ব্যবহারকারী অবাক হতে পারেন৷আপাতত (Android 15),
FLAG_OCCUPANT_ZONE_SCREEN_UNLOCKED
প্রয়োগ করা হয়নি৷ ক্লায়েন্ট অ্যাপ্লিকেশন শুধু এটি উপেক্ষা করতে পারেন.আপাতত (Android 15), Comms API শুধুমাত্র একই অ্যান্ড্রয়েড ইন্সট্যান্সে একাধিক ব্যবহারকারীকে সমর্থন করে যাতে পিয়ার অ্যাপগুলির একই দীর্ঘ সংস্করণ কোড (
FLAG_CLIENT_SAME_LONG_VERSION
) এবং স্বাক্ষর (FLAG_CLIENT_SAME_SIGNATURE
) থাকতে পারে৷ ফলস্বরূপ, অ্যাপগুলিকে যাচাই করতে হবে না যে দুটি মান একমত।
একটি ভাল ব্যবহারকারীর অভিজ্ঞতার জন্য, প্রেরক ক্লায়েন্ট একটি UI দেখাতে পারে যদি একটি পতাকা সেট না থাকে৷ উদাহরণস্বরূপ, যদি FLAG_OCCUPANT_ZONE_SCREEN_UNLOCKED
সেট করা না থাকে, তাহলে প্রেরক একটি টোস্ট বা একটি ডায়ালগ দেখাতে পারেন যাতে ব্যবহারকারীকে রিসিভার অকুপ্যান্ট জোনের স্ক্রীনটি আনলক করার অনুরোধ জানানো হয়।
যখন প্রেরকের আর রিসিভারগুলি আবিষ্কার করার প্রয়োজন হয় না (উদাহরণস্বরূপ, যখন এটি সমস্ত রিসিভার এবং প্রতিষ্ঠিত সংযোগগুলি খুঁজে পায় বা নিষ্ক্রিয় হয়ে যায়), তখন এটি আবিষ্কার বন্ধ করতে পারে৷
if (mRemoteDeviceManager != null) {
mRemoteDeviceManager.unregisterStateCallback();
}
আবিষ্কার বন্ধ হয়ে গেলে, বিদ্যমান সংযোগগুলি প্রভাবিত হয় না। প্রেরক সংযুক্ত রিসিভারদের Payload
পাঠানো চালিয়ে যেতে পারেন।
(প্রেরক) সংযোগের অনুরোধ করুন
যখন রিসিভারের সমস্ত পতাকা সেট করা হয়, প্রেরক রিসিভারের সাথে একটি সংযোগের অনুরোধ করতে পারেন:
private final ConnectionRequestCallback mRequestCallback = new ConnectionRequestCallback() {
@Override
public void onConnected(OccupantZoneInfo receiverZone) {
}
@Override
public void onFailed(OccupantZoneInfo receiverZone, int connectionError) {
}
@Override
public void onDisconnected(OccupantZoneInfo receiverZone) {
}
};
if (mOccupantConnectionManager != null && canRequestConnectionToReceiver(receiverZone)) {
mOccupantConnectionManager.requestConnection(receiverZone,
getActivity().getMainExecutor(), mRequestCallback);
}
(রিসিভার পরিষেবা) সংযোগ গ্রহণ করুন
প্রেরক একবার রিসিভারের সাথে সংযোগের অনুরোধ করলে, রিসিভার অ্যাপের AbstractReceiverService
গাড়ি পরিষেবা দ্বারা আবদ্ধ হবে এবং AbstractReceiverService.onConnectionInitiated()
আহ্বান করা হবে। যেমনটি (প্রেরক) অনুরোধ সংযোগে ব্যাখ্যা করা হয়েছে, onConnectionInitiated()
একটি বিমূর্ত পদ্ধতি এবং ক্লায়েন্ট অ্যাপ দ্বারা প্রয়োগ করা আবশ্যক।
যখন প্রাপক সংযোগের অনুরোধ গ্রহণ করে, প্রেরকের ConnectionRequestCallback.onConnected()
ডাকা হবে, তারপর সংযোগ স্থাপন করা হবে।
(প্রেরক) পেলোড পাঠান
সংযোগ স্থাপন হয়ে গেলে, প্রেরক রিসিভারের কাছে Payload
পাঠাতে পারেন:
if (mOccupantConnectionManager != null) {
Payload payload = ...;
try {
mOccupantConnectionManager.sendPayload(receiverZone, payload);
} catch (CarOccupantConnectionManager.PayloadTransferException e) {
Log.e(TAG, "Failed to send Payload to " + receiverZone);
}
}
প্রেরক Payload
একটি Binder
অবজেক্ট বা একটি বাইট অ্যারে রাখতে পারেন। প্রেরকের যদি অন্য ডেটা টাইপ পাঠাতে হয়, তবে এটিকে অবশ্যই একটি বাইট অ্যারেতে ডেটা সিরিয়ালাইজ করতে হবে, একটি Payload
অবজেক্ট তৈরি করতে বাইট অ্যারে ব্যবহার করতে হবে এবং Payload
পাঠাতে হবে। তারপর রিসিভার ক্লায়েন্ট প্রাপ্ত Payload
থেকে বাইট অ্যারে পায় এবং প্রত্যাশিত ডেটা অবজেক্টে বাইট অ্যারেকে ডিসিরিয়ালাইজ করে। উদাহরণস্বরূপ, যদি প্রেরক আইডি FragmentB
সহ রিসিভার এন্ডপয়েন্টে একটি স্ট্রিং hello
পাঠাতে চান, তাহলে এটি এই ধরনের ডেটা টাইপ সংজ্ঞায়িত করতে প্রোটো বাফার ব্যবহার করতে পারে:
message MyData {
required string receiver_endpoint_id = 1;
required string data = 2;
}
চিত্র 1 Payload
প্রবাহকে চিত্রিত করে:
(রিসিভার পরিষেবা) পেলোড গ্রহণ করুন এবং প্রেরণ করুন
একবার রিসিভার অ্যাপ Payload
গ্রহণ করলে, এর AbstractReceiverService.onPayloadReceived()
চালু করা হবে। Send the payload- এ যেমন ব্যাখ্যা করা হয়েছে, onPayloadReceived()
হল একটি বিমূর্ত পদ্ধতি এবং ক্লায়েন্ট অ্যাপ দ্বারা প্রয়োগ করা আবশ্যক। এই পদ্ধতিতে, ক্লায়েন্ট সংশ্লিষ্ট রিসিভার এন্ডপয়েন্টে Payload
ফরোয়ার্ড করতে পারে, অথবা Payload
ক্যাশে করে প্রত্যাশিত রিসিভার এন্ডপয়েন্ট নিবন্ধিত হয়ে গেলে এটি পাঠাতে পারে।
(রিসিভার এন্ডপয়েন্ট) নিবন্ধন করুন এবং নিবন্ধনমুক্ত করুন
রিসিভার এন্ডপয়েন্ট রেজিস্টার করতে রিসিভার অ্যাপটিকে registerReceiver()
কল করা উচিত। একটি সাধারণ ব্যবহারের ক্ষেত্রে একটি ফ্র্যাগমেন্টের রিসিভার Payload
প্রয়োজন হয়, তাই এটি একটি রিসিভার এন্ডপয়েন্ট নিবন্ধন করে:
private final PayloadCallback mPayloadCallback = (senderZone, payload) -> {
…
};
if (mOccupantConnectionManager != null) {
mOccupantConnectionManager.registerReceiver("FragmentB",
getActivity().getMainExecutor(), mPayloadCallback);
}
রিসিভার ক্লায়েন্টের AbstractReceiverService
একবার রিসিভার এন্ডপয়েন্টে Payload
পাঠালে, সংশ্লিষ্ট PayloadCallback
আহ্বান করা হবে।
ক্লায়েন্ট অ্যাপ একাধিক রিসিভার এন্ডপয়েন্ট নিবন্ধন করতে পারে যতক্ষণ না তাদের receiverEndpointId
আইডি ক্লায়েন্ট অ্যাপের মধ্যে অনন্য। কোন রিসিভার এন্ডপয়েন্ট(গুলি) পেলোডটি পাঠাতে হবে তা নির্ধারণ করতে AbstractReceiverService
ব্যবহার করবে receiverEndpointId
। যেমন:
- প্রেরক
Payload
receiver_endpoint_id:FragmentB
নির্দিষ্ট করে।Payload
গ্রহণ করার সময়, রিসিভারে থাকাAbstractReceiverService
FragmentB
-তে পেলোড পাঠানোর জন্যforwardPayload("FragmentB", payload)
কল করে - প্রেরক
Payload
data_type:VOLUME_CONTROL
নির্দিষ্ট করে।Payload
গ্রহণ করার সময়, রিসিভারে থাকাAbstractReceiverService
জানে যে এই ধরনেরPayload
FragmentB
তে পাঠানো উচিত, তাই এটিforwardPayload("FragmentB", payload)
কল করে
if (mOccupantConnectionManager != null) {
mOccupantConnectionManager.unregisterReceiver("FragmentB");
}
(প্রেরক) সংযোগটি বন্ধ করুন
একবার প্রেরকের আর রিসিভারের কাছে Payload
পাঠাতে হবে না (উদাহরণস্বরূপ, এটি নিষ্ক্রিয় হয়ে যায়), এটি সংযোগটি বন্ধ করে দিতে হবে।
if (mOccupantConnectionManager != null) {
mOccupantConnectionManager.disconnect(receiverZone);
}
একবার সংযোগ বিচ্ছিন্ন হয়ে গেলে, প্রেরক আর রিসিভারের কাছে Payload
পাঠাতে পারবেন না।
সংযোগ প্রবাহ
একটি সংযোগ প্রবাহ চিত্র 2 এ চিত্রিত করা হয়েছে।
সমস্যা সমাধান
লগ চেক করুন
সংশ্লিষ্ট লগ চেক করতে:
লগিংয়ের জন্য এই কমান্ডটি চালান:
adb shell setprop log.tag.CarRemoteDeviceService VERBOSE && adb shell setprop log.tag.CarOccupantConnectionService VERBOSE && adb logcat -s "AbstractReceiverService","CarOccupantConnectionManager","CarRemoteDeviceManager","CarRemoteDeviceService","CarOccupantConnectionService"
CarRemoteDeviceService
এবংCarOccupantConnectionService
এর অভ্যন্তরীণ অবস্থা ডাম্প করতে:adb shell dumpsys car_service --services CarRemoteDeviceService && adb shell dumpsys car_service --services CarOccupantConnectionService
নাল CarRemoteDeviceManager এবং CarOccupantConnectionManager
এই সম্ভাব্য মূল কারণগুলি দেখুন:
গাড়ি পরিষেবা বিপর্যস্ত। পূর্বে চিত্রিত হিসাবে, গাড়ি পরিষেবা ক্র্যাশ হলে দুই পরিচালক ইচ্ছাকৃতভাবে
null
হতে পুনরায় সেট করা হয়। যখন গাড়ি পরিষেবা পুনঃসূচনা করা হয়, তখন দুটি পরিচালক নন-নাল মানগুলিতে সেট করা হয়৷হয়
CarRemoteDeviceService
বাCarOccupantConnectionService
সক্ষম করা নেই৷ এক বা অন্যটি সক্ষম কিনা তা নির্ধারণ করতে, চালান:adb shell dumpsys car_service --services CarFeatureController
mDefaultEnabledFeaturesFromConfig
দেখুন, যাতেcar_remote_device_service
এবংcar_occupant_connection_service
থাকা উচিত। যেমন:mDefaultEnabledFeaturesFromConfig:[car_evs_service, car_navigation_service, car_occupant_connection_service, car_remote_device_service, car_telemetry_service, cluster_home_service, com.android.car.user.CarUserNoticeService, diagnostic, storage_monitoring, vehicle_map_service]
ডিফল্টরূপে, এই দুটি পরিষেবা অক্ষম করা হয়। যখন একটি ডিভাইস মাল্টি-ডিসপ্লে সমর্থন করে, তখন আপনাকে অবশ্যই এই কনফিগারেশন ফাইলটি ওভারলে করতে হবে। আপনি একটি কনফিগারেশন ফাইলে দুটি পরিষেবা সক্ষম করতে পারেন:
// packages/services/Car/service/res/values/config.xml <string-array translatable="false" name="config_allowed_optional_car_features"> <item>car_occupant_connection_service</item> <item>car_remote_device_service</item> … … </string-array>
API কল করার সময় ব্যতিক্রম
যদি ক্লায়েন্ট অ্যাপটি উদ্দেশ্য অনুযায়ী API ব্যবহার না করে, তাহলে একটি ব্যতিক্রম ঘটতে পারে। এই ক্ষেত্রে, ক্লায়েন্ট অ্যাপ সমস্যাটি সমাধান করতে ব্যতিক্রম এবং ক্র্যাশ স্ট্যাকের বার্তাটি পরীক্ষা করতে পারে। API অপব্যবহারের উদাহরণ হল:
-
registerStateCallback()
এই ক্লায়েন্টটি ইতিমধ্যে একটিStateCallback
নিবন্ধন করেছে। -
unregisterStateCallback()
এইCarRemoteDeviceManager
দৃষ্টান্ত দ্বারা কোনStateCallback
নিবন্ধিত হয়নি। -
registerReceiver()
receiverEndpointId
ইতিমধ্যেই নিবন্ধিত। -
unregisterReceiver()
receiverEndpointId
নিবন্ধিত নয়। -
requestConnection()
একটি মুলতুবি বা প্রতিষ্ঠিত সংযোগ ইতিমধ্যেই বিদ্যমান। -
cancelConnection()
বাতিল করার জন্য কোনো মুলতুবি সংযোগ নেই। -
sendPayload()
কোনো প্রতিষ্ঠিত সংযোগ নেই। -
disconnect()
কোনো প্রতিষ্ঠিত সংযোগ নেই।
ক্লায়েন্ট 1 পেলোড ক্লায়েন্ট 2 এ পাঠাতে পারে, তবে অন্যভাবে নয়
সংযোগ নকশা দ্বারা এক উপায়. দ্বি-মুখী সংযোগ স্থাপনের জন্য, client1
এবং client2
উভয়কেই একে অপরের সাথে সংযোগের অনুরোধ করতে হবে এবং তারপরে অনুমোদন পেতে হবে।