blob: 161fd2fb7f961c829816a284118eb1be9536a3c0 [file] [log] [blame]
/*
* Copyright 2020 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package android.media;
import android.annotation.NonNull;
import android.annotation.Nullable;
import android.annotation.SystemApi;
import android.os.Bundle;
import android.os.Parcel;
import android.os.Parcelable;
import android.text.TextUtils;
import java.io.FileDescriptor;
import java.io.PrintWriter;
import java.util.Collection;
import java.util.Collections;
import java.util.HashSet;
import java.util.List;
import java.util.Objects;
import java.util.Set;
import java.util.stream.Collectors;
/**
* A media route discovery preference describing the features of routes that media router
* would like to discover and whether to perform active scanning.
* <p>
* When {@link MediaRouter2} instances set discovery preferences by calling
* {@link MediaRouter2#registerRouteCallback}, they are merged into a single discovery preference
* and it is delivered to call {@link MediaRoute2ProviderService#onDiscoveryPreferenceChanged}.
* </p><p>
* According to the given discovery preference, {@link MediaRoute2ProviderService} discovers
* routes and publishes them.
* </p>
*
* @see MediaRouter2#registerRouteCallback
*/
public final class RouteDiscoveryPreference implements Parcelable {
@NonNull
public static final Creator<RouteDiscoveryPreference> CREATOR =
new Creator<RouteDiscoveryPreference>() {
@Override
public RouteDiscoveryPreference createFromParcel(Parcel in) {
return new RouteDiscoveryPreference(in);
}
@Override
public RouteDiscoveryPreference[] newArray(int size) {
return new RouteDiscoveryPreference[size];
}
};
@NonNull
private final List<String> mPreferredFeatures;
@NonNull
private final List<String> mPackageOrder;
@NonNull
private final List<String> mAllowedPackages;
private final boolean mShouldPerformActiveScan;
@Nullable
private final Bundle mExtras;
/**
* An empty discovery preference.
* @hide
*/
@SystemApi
public static final RouteDiscoveryPreference EMPTY =
new Builder(Collections.emptyList(), false).build();
RouteDiscoveryPreference(@NonNull Builder builder) {
mPreferredFeatures = builder.mPreferredFeatures;
mPackageOrder = builder.mPackageOrder;
mAllowedPackages = builder.mAllowedPackages;
mShouldPerformActiveScan = builder.mActiveScan;
mExtras = builder.mExtras;
}
RouteDiscoveryPreference(@NonNull Parcel in) {
mPreferredFeatures = in.createStringArrayList();
mPackageOrder = in.createStringArrayList();
mAllowedPackages = in.createStringArrayList();
mShouldPerformActiveScan = in.readBoolean();
mExtras = in.readBundle();
}
/**
* Gets the features of routes that media router would like to discover.
* <p>
* Routes that have at least one of the features will be discovered.
* They may include predefined features such as
* {@link MediaRoute2Info#FEATURE_LIVE_AUDIO}, {@link MediaRoute2Info#FEATURE_LIVE_VIDEO},
* or {@link MediaRoute2Info#FEATURE_REMOTE_PLAYBACK} or custom features defined by a provider.
* </p>
*/
@NonNull
public List<String> getPreferredFeatures() {
return mPreferredFeatures;
}
/**
* Gets the ordered list of package names used to remove duplicate routes.
* <p>
* Duplicate route removal is enabled if the returned list is non-empty. Routes are deduplicated
* based on their {@link MediaRoute2Info#getDeduplicationIds() deduplication IDs}. If two routes
* have a deduplication ID in common, only the route from the provider whose package name is
* first in the provided list will remain.
*
* @see #shouldRemoveDuplicates()
* @hide
*/
@NonNull
public List<String> getDeduplicationPackageOrder() {
return mPackageOrder;
}
/**
* Gets the list of allowed packages.
* <p>
* If it's not empty, it will only discover routes from the provider whose package name
* belongs to the list.
* @hide
*/
@NonNull
public List<String> getAllowedPackages() {
return mAllowedPackages;
}
/**
* Gets whether active scanning should be performed.
* <p>
* If any of discovery preferences sets this as {@code true}, active scanning will
* be performed regardless of other discovery preferences.
* </p>
*/
public boolean shouldPerformActiveScan() {
return mShouldPerformActiveScan;
}
/**
* Gets whether duplicate routes removal is enabled.
*
* @see #getDeduplicationPackageOrder()
* @hide
*/
public boolean shouldRemoveDuplicates() {
return !mPackageOrder.isEmpty();
}
/**
* @hide
*/
public Bundle getExtras() {
return mExtras;
}
@Override
public int describeContents() {
return 0;
}
@Override
public void writeToParcel(@NonNull Parcel dest, int flags) {
dest.writeStringList(mPreferredFeatures);
dest.writeStringList(mPackageOrder);
dest.writeStringList(mAllowedPackages);
dest.writeBoolean(mShouldPerformActiveScan);
dest.writeBundle(mExtras);
}
/**
* Dumps current state of the instance. Use with {@code dumpsys}.
*
* See {@link android.os.Binder#dump(FileDescriptor, PrintWriter, String[])}.
*
* @hide
*/
public void dump(@NonNull PrintWriter pw, @NonNull String prefix) {
pw.println(prefix + "RouteDiscoveryPreference");
String indent = prefix + " ";
pw.println(indent + "mShouldPerformActiveScan=" + mShouldPerformActiveScan);
pw.println(indent + "mPreferredFeatures=" + mPreferredFeatures);
pw.println(indent + "mPackageOrder=" + mPackageOrder);
pw.println(indent + "mAllowedPackages=" + mAllowedPackages);
pw.println(indent + "mExtras=" + mExtras);
}
@Override
public String toString() {
StringBuilder result = new StringBuilder()
.append("RouteDiscoveryRequest{ ")
.append("preferredFeatures={")
.append(String.join(", ", mPreferredFeatures))
.append("}")
.append(", activeScan=")
.append(mShouldPerformActiveScan)
.append(" }");
return result.toString();
}
@Override
public boolean equals(Object o) {
if (this == o) {
return true;
}
if (!(o instanceof RouteDiscoveryPreference)) {
return false;
}
RouteDiscoveryPreference other = (RouteDiscoveryPreference) o;
return Objects.equals(mPreferredFeatures, other.mPreferredFeatures)
&& Objects.equals(mPackageOrder, other.mPackageOrder)
&& Objects.equals(mAllowedPackages, other.mAllowedPackages)
&& mShouldPerformActiveScan == other.mShouldPerformActiveScan;
}
@Override
public int hashCode() {
return Objects.hash(mPreferredFeatures, mPackageOrder, mAllowedPackages,
mShouldPerformActiveScan);
}
/**
* Builder for {@link RouteDiscoveryPreference}.
*/
public static final class Builder {
List<String> mPreferredFeatures;
List<String> mPackageOrder;
List<String> mAllowedPackages;
boolean mActiveScan;
Bundle mExtras;
public Builder(@NonNull List<String> preferredFeatures, boolean activeScan) {
Objects.requireNonNull(preferredFeatures, "preferredFeatures must not be null");
mPreferredFeatures = preferredFeatures.stream().filter(str -> !TextUtils.isEmpty(str))
.collect(Collectors.toList());
mPackageOrder = List.of();
mAllowedPackages = List.of();
mActiveScan = activeScan;
}
public Builder(@NonNull RouteDiscoveryPreference preference) {
Objects.requireNonNull(preference, "preference must not be null");
mPreferredFeatures = preference.getPreferredFeatures();
mPackageOrder = preference.getDeduplicationPackageOrder();
mAllowedPackages = preference.getAllowedPackages();
mActiveScan = preference.shouldPerformActiveScan();
mExtras = preference.getExtras();
}
/**
* A constructor to combine multiple preferences into a single preference.
* It ignores extras of preferences.
*
* @hide
*/
public Builder(@NonNull Collection<RouteDiscoveryPreference> preferences) {
Objects.requireNonNull(preferences, "preferences must not be null");
Set<String> preferredFeatures = new HashSet<>();
Set<String> allowedPackages = new HashSet<>();
mPackageOrder = List.of();
boolean activeScan = false;
for (RouteDiscoveryPreference preference : preferences) {
preferredFeatures.addAll(preference.mPreferredFeatures);
allowedPackages.addAll(preference.mAllowedPackages);
activeScan |= preference.mShouldPerformActiveScan;
// Choose one of either
if (mPackageOrder.isEmpty() && !preference.mPackageOrder.isEmpty()) {
mPackageOrder = List.copyOf(preference.mPackageOrder);
}
}
mPreferredFeatures = List.copyOf(preferredFeatures);
mAllowedPackages = List.copyOf(allowedPackages);
mActiveScan = activeScan;
}
/**
* Sets preferred route features to discover.
* @param preferredFeatures features of routes that media router would like to discover.
* May include predefined features
* such as {@link MediaRoute2Info#FEATURE_LIVE_AUDIO},
* {@link MediaRoute2Info#FEATURE_LIVE_VIDEO},
* or {@link MediaRoute2Info#FEATURE_REMOTE_PLAYBACK}
* or custom features defined by a provider.
*/
@NonNull
public Builder setPreferredFeatures(@NonNull List<String> preferredFeatures) {
Objects.requireNonNull(preferredFeatures, "preferredFeatures must not be null");
mPreferredFeatures = preferredFeatures.stream().filter(str -> !TextUtils.isEmpty(str))
.collect(Collectors.toList());
return this;
}
/**
* Sets the list of package names of providers that media router would like to discover.
* <p>
* If it's non-empty, media router only discovers route from the provider in the list.
* The default value is empty, which discovers routes from all providers.
* @hide
*/
@NonNull
public Builder setAllowedPackages(@NonNull List<String> allowedPackages) {
Objects.requireNonNull(allowedPackages, "allowedPackages must not be null");
mAllowedPackages = List.copyOf(allowedPackages);
return this;
}
/**
* Sets if active scanning should be performed.
* <p>
* Since active scanning uses more system resources, set this as {@code true} only
* when it's necessary.
* </p>
*/
@NonNull
public Builder setShouldPerformActiveScan(boolean activeScan) {
mActiveScan = activeScan;
return this;
}
/**
* Sets the order of packages to use when removing duplicate routes.
* <p>
* Routes are deduplicated based on their
* {@link MediaRoute2Info#getDeduplicationIds() deduplication IDs}.
* If two routes have a deduplication ID in common, only the route from the provider whose
* package name is first in the provided list will remain.
*
* @param packageOrder ordered list of package names used to remove duplicate routes, or an
* empty list if deduplication should not be enabled.
* @hide
*/
@NonNull
public Builder setDeduplicationPackageOrder(@NonNull List<String> packageOrder) {
Objects.requireNonNull(packageOrder, "packageOrder must not be null");
mPackageOrder = List.copyOf(packageOrder);
return this;
}
/**
* Sets the extras of the route.
* @hide
*/
@NonNull
public Builder setExtras(@Nullable Bundle extras) {
mExtras = extras;
return this;
}
/**
* Builds the {@link RouteDiscoveryPreference}.
*/
@NonNull
public RouteDiscoveryPreference build() {
return new RouteDiscoveryPreference(this);
}
}
}