Skip to content

Commit f72bfa3

Browse files
Android tv-casting-app: simplified Endpoint API
1 parent d166650 commit f72bfa3

40 files changed

+1881
-697
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

+15-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

@@ -33,6 +36,7 @@ protected void onCreate(Bundle savedInstanceState) {
3336
super.onCreate(savedInstanceState);
3437
setContentView(R.layout.activity_main);
3538

39+
Log.i(TAG, "ChipCastingSimplified = " + GlobalCastingConstants.ChipCastingSimplified);
3640
boolean ret =
3741
GlobalCastingConstants.ChipCastingSimplified
3842
? InitializationExample.initAndStart(this.getApplicationContext()).hasNoError()
@@ -73,9 +77,12 @@ public void handleCommissioningComplete() {
7377
@Override
7478
public void handleConnectionComplete(CastingPlayer castingPlayer) {
7579
Log.i(TAG, "MainActivity.handleConnectionComplete() called ");
80+
showFragment(ActionSelectorFragment.newInstance(castingPlayer));
81+
}
7682

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

8188
@Override
@@ -95,7 +102,10 @@ public void handleMediaPlaybackSelected() {
95102

96103
@Override
97104
public void handleDisconnect() {
98-
showFragment(CommissionerDiscoveryFragment.newInstance(tvCastingApp));
105+
showFragment(
106+
GlobalCastingConstants.ChipCastingSimplified
107+
? DiscoveryExampleFragment.newInstance()
108+
: CommissionerDiscoveryFragment.newInstance(tvCastingApp));
99109
}
100110

101111
/**

0 commit comments

Comments
 (0)