Skip to content

Commit 95ab10e

Browse files
Android tv-casting-app: simplified Endpoint API
1 parent bf7241d commit 95ab10e

38 files changed

+1713
-537
lines changed

examples/tv-casting-app/APIs.md

+111-31
Original file line numberDiff line numberDiff line change
@@ -34,15 +34,11 @@ samples so you can see the experience end to end.
3434

3535
A Casting Client (e.g. a mobile phone app) is expected to be a Matter
3636
Commissionable Node and a `CastingPlayer` (i.e. a TV) is expected to be a Matter
37-
Commissioner. In the context of the
38-
[Matter Video Player architecture](https://github.com/CHIP-Specifications/connectedhomeip-spec/blob/master/src/app_clusters/media/VideoPlayerArchitecture.adoc),
39-
a `CastingPlayer` would map to
40-
[Casting "Video" Player](https://github.com/CHIP-Specifications/connectedhomeip-spec/blob/master/src/app_clusters/media/VideoPlayerArchitecture.adoc#1-introduction).
41-
The `CastingPlayer` is expected to be hosting one or more `Endpoints` (some of
42-
which can represent
43-
[Content Apps](https://github.com/CHIP-Specifications/connectedhomeip-spec/blob/master/src/app_clusters/media/VideoPlayerArchitecture.adoc#1-introduction)
44-
in the Matter Video Player architecture) that support one or more Matter Media
45-
`Clusters`.
37+
Commissioner. In the context of the Matter Video Player architecture, a
38+
`CastingPlayer` would map to Casting "Video" Player. The `CastingPlayer` is
39+
expected to be hosting one or more `Endpoints` (some of which can represent
40+
Content Apps in the Matter Video Player architecture) that support one or more
41+
Matter Media `Clusters`.
4642

4743
The steps to start a casting session are:
4844

@@ -80,6 +76,8 @@ consume each platform's specific libraries. The libraries MUST be built with the
8076
client's specific values for `CHIP_DEVICE_CONFIG_DEVICE_VENDOR_ID` and
8177
`CHIP_DEVICE_CONFIG_DEVICE_PRODUCT_ID` updated in the
8278
[CHIPProjectAppConfig.h](tv-casting-common/include/CHIPProjectAppConfig.h) file.
79+
Other values like the `CHIP_DEVICE_CONFIG_DEVICE_NAME` may be updated as well to
80+
correspond to the client being built.
8381

8482
### Initialize the Casting Client
8583

@@ -91,10 +89,10 @@ A Casting Client must first initialize the Matter SDK and define the following
9189
`DataProvider` objects for the the Matter Casting library to use throughout the
9290
client's lifecycle:
9391

94-
1. **Rotating Device Identifier** - Refer to the Matter specification for
95-
details on how to generate the
96-
[Rotating Device Identifier](https://github.com/CHIP-Specifications/connectedhomeip-spec/blob/master/src/rendezvous/DeviceDiscovery.adoc#245-rotating-device-identifier)).
97-
Then, instantiate a `DataProvider` object as described below.
92+
1. **Rotating Device Identifier** - "This unique per-device identifier SHALL
93+
consist of a randomly-generated 128-bit or longer octet string." Refer to
94+
the Matter specification for more details. Instantiate a `DataProvider`
95+
object as described below to provide this identifier.
9896

9997
On Linux, define a `RotatingDeviceIdUniqueIdProvider` to provide the Casting
10098
Client's `RotatingDeviceIdUniqueId`, by implementing a
@@ -152,10 +150,13 @@ client's lifecycle:
152150
```
153151

154152
2. **Commissioning Data** - This object contains the passcode, discriminator,
155-
etc which identify the app and are provided to the `CastingPlayer` during
156-
the commissioning process. Refer to the Matter specification's
157-
[Onboarding Payload](https://github.com/CHIP-Specifications/connectedhomeip-spec/blob/master/src/qr_code/OnboardingPayload.adoc#ref_OnboardingPayload)
158-
section for details on commissioning data.
153+
etc. which identify the app and are provided to the `CastingPlayer` during
154+
the commissioning process. "A Passcode SHALL be included as a 27-bit
155+
unsigned integer, which serves as proof of possession during commissioning."
156+
"A Discriminator SHALL be included as a 12-bit unsigned integer, which SHALL
157+
match the value which a device advertises during commissioning." Refer to
158+
the Matter specification's "Onboarding Payload" section for more details on
159+
commissioning data.
159160

160161
On Linux, define a function `InitCommissionableDataProvider` to initialize
161162
initialize a `LinuxCommissionableDataProvider` that can provide the required
@@ -217,9 +218,8 @@ client's lifecycle:
217218
3. **Device Attestation Credentials** - This object contains the
218219
`DeviceAttestationCertificate`, `ProductAttestationIntermediateCertificate`,
219220
etc. and implements a way to sign messages when called upon by the Matter TV
220-
Casting Library as part of the
221-
[Device Attestation process](https://github.com/CHIP-Specifications/connectedhomeip-spec/blob/master/src/device_attestation/Device_Attestation_Specification.adoc)
222-
during commissioning.
221+
Casting Library as part of the Matter Device Attestation process during
222+
commissioning.
223223

224224
On Linux, implement a define a `dacProvider` to provide the Casting Client's
225225
Device Attestation Credentials, by implementing a
@@ -487,8 +487,8 @@ potentially skipping the longer commissioning process and instead, simply
487487
re-establishing the CASE session. This cache can be cleared by calling the
488488
`ClearCache` API on the `CastingApp`, say when the user signs out of the app.
489489
See API and its documentation for [Linux](tv-casting-common/core/CastingApp.h),
490-
Android and
491-
[iOS](darwin/MatterTvCastingBridge/MatterTvCastingBridge/MCCastingApp.h).
490+
[Android](android/App/app/src/main/jni/com/matter/casting/core/CastingApp.java)
491+
and [iOS](darwin/MatterTvCastingBridge/MatterTvCastingBridge/MCCastingApp.h).
492492

493493
### Discover Casting Players
494494

@@ -702,10 +702,9 @@ Each `CastingPlayer` object created during
702702
[Discovery](#discover-casting-players) contains information such as
703703
`deviceName`, `vendorId`, `productId`, etc. which can help the user pick the
704704
right `CastingPlayer`. A Casting Client can attempt to connect to the
705-
`selectedCastingPlayer` using
706-
[Matter User Directed Commissioning (UDC)](https://github.com/CHIP-Specifications/connectedhomeip-spec/blob/master/src/rendezvous/UserDirectedCommissioning.adoc).
707-
The Matter TV Casting library locally caches information required to reconnect
708-
to a `CastingPlayer`, once the Casting client has been commissioned by it. After
705+
`selectedCastingPlayer` using Matter User Directed Commissioning (UDC). The
706+
Matter TV Casting library locally caches information required to reconnect to a
707+
`CastingPlayer`, once the Casting client has been commissioned by it. After
709708
that, the Casting client is able to skip the full UDC process by establishing
710709
CASE with the `CastingPlayer` directly. Once connected, the `CastingPlayer`
711710
object will contain the list of available Endpoints on that `CastingPlayer`.
@@ -743,6 +742,60 @@ targetCastingPlayer->VerifyOrEstablishConnection(ConnectionHandler,
743742
...
744743
```
745744
745+
On Android, the Casting Client may call `verifyOrEstablishConnection` on the
746+
`CastingPlayer` object it wants to connect to.
747+
748+
```java
749+
private static final long MIN_CONNECTION_TIMEOUT_SEC = 3 * 60;
750+
751+
EndpointFilter desiredEndpointFilter = new EndpointFilter();
752+
desiredEndpointFilter.vendorId = DESIRED_ENDPOINT_VENDOR_ID;
753+
754+
MatterError err = targetCastingPlayer.verifyOrEstablishConnection(
755+
MIN_CONNECTION_TIMEOUT_SEC,
756+
desiredEndpointFilter,
757+
new MatterCallback<Void>() {
758+
@Override
759+
public void handle(Void v) {
760+
Log.i(
761+
TAG,
762+
"Connected to CastingPlayer with deviceId: "
763+
+ targetCastingPlayer.getDeviceId());
764+
getActivity()
765+
.runOnUiThread(
766+
() -> {
767+
connectionFragmentStatusTextView.setText(
768+
"Connected to Casting Player with device name: "
769+
+ targetCastingPlayer.getDeviceName()
770+
+ "\n\n");
771+
connectionFragmentNextButton.setEnabled(true);
772+
});
773+
}
774+
},
775+
new MatterCallback<MatterError>() {
776+
@Override
777+
public void handle(MatterError err) {
778+
Log.e(TAG, "CastingPLayer connection failed: " + err);
779+
getActivity()
780+
.runOnUiThread(
781+
() -> {
782+
connectionFragmentStatusTextView.setText(
783+
"Casting Player connection failed due to: " + err + "\n\n");
784+
});
785+
}
786+
});
787+
788+
if (err.hasError())
789+
{
790+
getActivity()
791+
.runOnUiThread(
792+
() -> {
793+
connectionFragmentStatusTextView.setText(
794+
"Casting Player connection failed due to: " + err + "\n\n");
795+
});
796+
}
797+
```
798+
746799
On iOS, the Casting Client may call `verifyOrEstablishConnection` on the
747800
`MCCastingPlayer` object it wants to connect to and handle any `NSErrors` that
748801
may happen in the process.
@@ -777,6 +830,8 @@ func connect(selectedCastingPlayer: MCCastingPlayer?) {
777830
### Select an Endpoint on the Casting Player
778831
779832
_{Complete Endpoint selection examples: [Linux](linux/simple-app-helper.cpp) |
833+
[Android](android/App/app/src/main/java/com/matter/casting/ContentLauncherLaunchURLExampleFragment.java)
834+
|
780835
[iOS](darwin/TvCasting/TvCasting/MCContentLauncherLaunchURLExampleViewModel.swift)}_
781836
782837
On a successful connection with a `CastingPlayer`, a Casting Client may select
@@ -803,6 +858,34 @@ if (it != endpoints.end())
803858
}
804859
```
805860
861+
On Android, it can select an `Endpoint` as shown below.
862+
863+
```java
864+
private static final Integer SAMPLE_ENDPOINT_VID = 65521;
865+
866+
private Endpoint selectEndpoint()
867+
{
868+
Endpoint endpoint = null;
869+
if(selectedCastingPlayer != null)
870+
{
871+
List<Endpoint> endpoints = selectedCastingPlayer.getEndpoints();
872+
if (endpoints == null)
873+
{
874+
Log.e(TAG, "No Endpoints found on CastingPlayer");
875+
}
876+
else
877+
{
878+
endpoint = endpoints
879+
.stream()
880+
.filter(e -> SAMPLE_ENDPOINT_VID.equals(e.getVendorId()))
881+
.findFirst()
882+
.get();
883+
}
884+
}
885+
return endpoint;
886+
}
887+
```
888+
806889
On iOS, it can select an `MCEndpoint` similarly and as shown below.
807890
808891
```swift
@@ -1045,11 +1128,8 @@ vendorIDAttribute!.read(nil) { context, before, after, err in
10451128
10461129
### Subscriptions
10471130
1048-
_{Complete Attribute subscription examples:
1049-
[Linux](linux/simple-app-helper.cpp)}_
1050-
1051-
_{Complete Attribute Read examples: [Linux](linux/simple-app-helper.cpp) |
1052-
[iOS](darwin/TvCasting/TvCasting/MCMediaPlaybackSubscribeToCurrentStateExampleViewModel.swift)}_
1131+
_{Complete Attribute subscription examples: [Linux](linux/simple-app-helper.cpp)
1132+
|[iOS](darwin/TvCasting/TvCasting/MCMediaPlaybackSubscribeToCurrentStateExampleViewModel.swift)}_
10531133
10541134
A Casting Client may subscribe to an attribute on an `Endpoint` of the
10551135
`CastingPlayer` to get data reports when the attributes change.

examples/tv-casting-app/android/App/app/src/main/java/com/chip/casting/app/MainActivity.java

+14-5
Original file line numberDiff line numberDiff line change
@@ -10,10 +10,12 @@
1010
import com.chip.casting.DiscoveredNodeData;
1111
import com.chip.casting.TvCastingApp;
1212
import com.chip.casting.util.GlobalCastingConstants;
13-
import com.chip.casting.util.PreferencesConfigurationManager;
13+
import com.matter.casting.ActionSelectorFragment;
1414
import com.matter.casting.ConnectionExampleFragment;
15+
import com.matter.casting.ContentLauncherLaunchURLExampleFragment;
1516
import com.matter.casting.DiscoveryExampleFragment;
1617
import com.matter.casting.InitializationExample;
18+
import com.matter.casting.PreferencesConfigurationManager;
1719
import com.matter.casting.core.CastingPlayer;
1820
import java.util.Random;
1921

@@ -22,7 +24,8 @@ public class MainActivity extends AppCompatActivity
2224
ConnectionFragment.Callback,
2325
SelectClusterFragment.Callback,
2426
DiscoveryExampleFragment.Callback,
25-
ConnectionExampleFragment.Callback {
27+
ConnectionExampleFragment.Callback,
28+
ActionSelectorFragment.Callback {
2629

2730
private static final String TAG = MainActivity.class.getSimpleName();
2831

@@ -73,9 +76,12 @@ public void handleCommissioningComplete() {
7376
@Override
7477
public void handleConnectionComplete(CastingPlayer castingPlayer) {
7578
Log.i(TAG, "MainActivity.handleConnectionComplete() called ");
79+
showFragment(ActionSelectorFragment.newInstance(castingPlayer));
80+
}
7681

77-
// TODO: Implement in following PRs. Select Cluster Fragment.
78-
// showFragment(SelectClusterFragment.newInstance(tvCastingApp));
82+
@Override
83+
public void handleContentLauncherLaunchURLSelected(CastingPlayer selectedCastingPlayer) {
84+
showFragment(ContentLauncherLaunchURLExampleFragment.newInstance(selectedCastingPlayer));
7985
}
8086

8187
@Override
@@ -95,7 +101,10 @@ public void handleMediaPlaybackSelected() {
95101

96102
@Override
97103
public void handleDisconnect() {
98-
showFragment(CommissionerDiscoveryFragment.newInstance(tvCastingApp));
104+
showFragment(
105+
GlobalCastingConstants.ChipCastingSimplified
106+
? DiscoveryExampleFragment.newInstance()
107+
: CommissionerDiscoveryFragment.newInstance(tvCastingApp));
99108
}
100109

101110
/**
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,97 @@
1+
/*
2+
* Copyright (c) 2024 Project CHIP Authors
3+
* All rights reserved.
4+
*
5+
* Licensed under the Apache License, Version 2.0 (the "License");
6+
* you may not use this file except in compliance with the License.
7+
* You may obtain a copy of the License at
8+
*
9+
* http://www.apache.org/licenses/LICENSE-2.0
10+
*
11+
* Unless required by applicable law or agreed to in writing, software
12+
* distributed under the License is distributed on an "AS IS" BASIS,
13+
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14+
* See the License for the specific language governing permissions and
15+
* limitations under the License.
16+
*/
17+
package com.matter.casting;
18+
19+
import android.os.Bundle;
20+
import android.util.Log;
21+
import android.view.LayoutInflater;
22+
import android.view.View;
23+
import android.view.ViewGroup;
24+
import androidx.annotation.Nullable;
25+
import androidx.fragment.app.Fragment;
26+
import com.R;
27+
import com.matter.casting.core.CastingPlayer;
28+
29+
/** An interstitial {@link Fragment} to select one of the supported media actions to perform */
30+
public class ActionSelectorFragment extends Fragment {
31+
private static final String TAG = ActionSelectorFragment.class.getSimpleName();
32+
33+
private final CastingPlayer selectedCastingPlayer;
34+
35+
private View.OnClickListener selectContentLauncherButtonClickListener;
36+
private View.OnClickListener disconnectButtonClickListener;
37+
38+
public ActionSelectorFragment(CastingPlayer selectedCastingPlayer) {
39+
this.selectedCastingPlayer = selectedCastingPlayer;
40+
}
41+
42+
/**
43+
* Use this factory method to create a new instance of this fragment using the provided
44+
* parameters.
45+
*
46+
* @param selectedCastingPlayer CastingPlayer that the casting app connected to
47+
* @return A new instance of fragment SelectActionFragment.
48+
*/
49+
public static ActionSelectorFragment newInstance(CastingPlayer selectedCastingPlayer) {
50+
return new ActionSelectorFragment(selectedCastingPlayer);
51+
}
52+
53+
@Override
54+
public void onCreate(Bundle savedInstanceState) {
55+
super.onCreate(savedInstanceState);
56+
}
57+
58+
@Override
59+
public View onCreateView(
60+
LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
61+
ActionSelectorFragment.Callback callback = (ActionSelectorFragment.Callback) this.getActivity();
62+
this.selectContentLauncherButtonClickListener =
63+
v -> {
64+
Log.d(TAG, "handle() called on selectContentLauncherButtonClickListener");
65+
callback.handleContentLauncherLaunchURLSelected(selectedCastingPlayer);
66+
};
67+
68+
this.disconnectButtonClickListener =
69+
v -> {
70+
Log.d(TAG, "Disconnecting from current casting player");
71+
selectedCastingPlayer.disconnect();
72+
callback.handleDisconnect();
73+
};
74+
75+
return inflater.inflate(R.layout.fragment_matter_action_selector, container, false);
76+
}
77+
78+
@Override
79+
public void onViewCreated(View view, @Nullable Bundle savedInstanceState) {
80+
super.onViewCreated(view, savedInstanceState);
81+
Log.d(TAG, "ActionSelectorFragment.onViewCreated called");
82+
getView()
83+
.findViewById(R.id.selectContentLauncherLaunchURLButton)
84+
.setOnClickListener(selectContentLauncherButtonClickListener);
85+
86+
getView().findViewById(R.id.disconnectButton).setOnClickListener(disconnectButtonClickListener);
87+
}
88+
89+
/** Interface for notifying the host. */
90+
public interface Callback {
91+
/** Notifies listener to trigger transition on selection of Content Launcher cluster */
92+
void handleContentLauncherLaunchURLSelected(CastingPlayer selectedCastingPlayer);
93+
94+
/** Notifies listener to trigger transition on click of the Disconnect button */
95+
void handleDisconnect();
96+
}
97+
}

0 commit comments

Comments
 (0)