/* * Copyright (C) 2019 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.car.projection; import static com.android.car.internal.ExcludeFromCodeCoverageGeneratedReport.BOILERPLATE_CODE; import android.annotation.IntDef; import android.annotation.NonNull; import android.annotation.SystemApi; import android.os.Bundle; import android.os.Parcel; import android.os.Parcelable; import com.android.car.internal.ExcludeFromCodeCoverageGeneratedReport; import com.android.car.internal.util.IntArray; import java.lang.annotation.Retention; import java.lang.annotation.RetentionPolicy; import java.util.ArrayList; import java.util.Arrays; import java.util.List; /** * This class encapsulates information about projection status and connected mobile devices. * *

Since the number of connected devices expected to be small we include information about * connected devices in every status update. * * @hide */ @SystemApi public final class ProjectionStatus implements Parcelable { /** This state indicates that projection is not actively running and no compatible mobile * devices available. */ public static final int PROJECTION_STATE_INACTIVE = 0; /** At least one phone connected and ready to project. */ public static final int PROJECTION_STATE_READY_TO_PROJECT = 1; /** Projecting in the foreground */ public static final int PROJECTION_STATE_ACTIVE_FOREGROUND = 2; /** Projection is running in the background */ public static final int PROJECTION_STATE_ACTIVE_BACKGROUND = 3; /** This state indicates that at least one phone is connected and attempting to start * the projection. If one mobile device is already attempting to start the projection, * it should not be overridden by starting other projection technologies on top of it. */ public static final int PROJECTION_STATE_ATTEMPTING = 4; /** This state indicates that at least one phone is connected and in the process of finishing * the projection. If one mobile device is already finishing the projection, * another mobile device should not start other projection technologies on top of it. */ public static final int PROJECTION_STATE_FINISHING = 5; private static final int PROJECTION_STATE_MAX = PROJECTION_STATE_FINISHING; /** This status is used when projection is not actively running */ public static final int PROJECTION_TRANSPORT_NONE = 0; /** This status is used when projection is not actively running */ public static final int PROJECTION_TRANSPORT_USB = 1; /** This status is used when projection is not actively running */ public static final int PROJECTION_TRANSPORT_WIFI = 2; private static final int PROJECTION_TRANSPORT_MAX = PROJECTION_TRANSPORT_WIFI; /** @hide */ @IntDef(value = { PROJECTION_TRANSPORT_NONE, PROJECTION_TRANSPORT_USB, PROJECTION_TRANSPORT_WIFI, }) @Retention(RetentionPolicy.SOURCE) public @interface ProjectionTransport {} /** @hide */ @IntDef(value = { PROJECTION_STATE_INACTIVE, PROJECTION_STATE_READY_TO_PROJECT, PROJECTION_STATE_ACTIVE_FOREGROUND, PROJECTION_STATE_ACTIVE_BACKGROUND, PROJECTION_STATE_ATTEMPTING, PROJECTION_STATE_FINISHING, }) @Retention(RetentionPolicy.SOURCE) public @interface ProjectionState {} private final String mPackageName; private final int mState; private final int mTransport; private final List mConnectedMobileDevices; private final Bundle mExtras; /** Creator for this class. Required to have in parcelable implementations. */ public static final Creator CREATOR = new Creator() { @Override public ProjectionStatus createFromParcel(Parcel source) { return new ProjectionStatus(source); } @Override public ProjectionStatus[] newArray(int size) { return new ProjectionStatus[size]; } }; private ProjectionStatus(Builder builder) { mPackageName = builder.mPackageName; mState = builder.mState; mTransport = builder.mTransport; mConnectedMobileDevices = new ArrayList<>(builder.mMobileDevices); mExtras = builder.mExtras == null ? null : new Bundle(builder.mExtras); } private ProjectionStatus(Parcel source) { mPackageName = source.readString(); mState = source.readInt(); mTransport = source.readInt(); mExtras = source.readBundle(getClass().getClassLoader()); mConnectedMobileDevices = source.createTypedArrayList(MobileDevice.CREATOR); } /** Parcelable implementation */ @Override @ExcludeFromCodeCoverageGeneratedReport(reason = BOILERPLATE_CODE) public int describeContents() { return 0; } @Override public void writeToParcel(Parcel dest, int flags) { dest.writeString(mPackageName); dest.writeInt(mState); dest.writeInt(mTransport); dest.writeBundle(mExtras); dest.writeTypedList(mConnectedMobileDevices); } /** Returns projection state which could be one of the constants starting with * {@code #PROJECTION_STATE_}. */ public @ProjectionState int getState() { return mState; } /** Returns package name of the projection receiver app. */ public @NonNull String getPackageName() { return mPackageName; } /** Returns extra information provided by projection receiver app */ public @NonNull Bundle getExtras() { return mExtras == null ? new Bundle() : new Bundle(mExtras); } /** Returns true if currently projecting either in the foreground or in the background. */ public boolean isActive() { return mState == PROJECTION_STATE_ACTIVE_BACKGROUND || mState == PROJECTION_STATE_ACTIVE_FOREGROUND; } /** Returns transport which is used for active projection or * {@link #PROJECTION_TRANSPORT_NONE} if projection is not running. */ public @ProjectionTransport int getTransport() { return mTransport; } /** Returns a list of currently connected mobile devices. */ public @NonNull List getConnectedMobileDevices() { return new ArrayList<>(mConnectedMobileDevices); } /** * Returns new {@link Builder} instance. * * @param packageName package name that will be associated with this status * @param state current projection state, must be one of the {@code PROJECTION_STATE_*} */ @NonNull public static Builder builder(String packageName, @ProjectionState int state) { return new Builder(packageName, state); } /** Builder class for {@link ProjectionStatus} */ public static final class Builder { private final int mState; private final String mPackageName; private int mTransport = PROJECTION_TRANSPORT_NONE; private List mMobileDevices = new ArrayList<>(); private Bundle mExtras; private Builder(String packageName, @ProjectionState int state) { if (packageName == null) { throw new IllegalArgumentException("Package name can't be null"); } if (state < 0 || state > PROJECTION_STATE_MAX) { throw new IllegalArgumentException("Invalid projection state: " + state); } mPackageName = packageName; mState = state; } /** * Sets the transport which is used for currently projecting phone if any. * * @param transport transport of current projection, must be one of the * {@code PROJECTION_TRANSPORT_*} */ public @NonNull Builder setProjectionTransport(@ProjectionTransport int transport) { checkProjectionTransport(transport); mTransport = transport; return this; } /** * Add connected mobile device * * @param mobileDevice connected mobile device * @return this builder */ public @NonNull Builder addMobileDevice(MobileDevice mobileDevice) { mMobileDevices.add(mobileDevice); return this; } /** * Add extra information. * * @param extras may contain an extra information that can be passed from the projection * app to the projection status listeners * @return this builder */ public @NonNull Builder setExtras(Bundle extras) { mExtras = extras; return this; } /** Creates {@link ProjectionStatus} object. */ public ProjectionStatus build() { return new ProjectionStatus(this); } } private static void checkProjectionTransport(@ProjectionTransport int transport) { if (transport < 0 || transport > PROJECTION_TRANSPORT_MAX) { throw new IllegalArgumentException("Invalid projection transport: " + transport); } } @Override public String toString() { return "ProjectionStatus{" + "mPackageName='" + mPackageName + '\'' + ", mState=" + mState + ", mTransport=" + mTransport + ", mConnectedMobileDevices=" + mConnectedMobileDevices + (mExtras != null ? " (has extras)" : "") + '}'; } /** Class that represents information about connected mobile device. */ public static final class MobileDevice implements Parcelable { private final int mId; private final String mName; private final int[] mAvailableTransports; private final boolean mProjecting; private final Bundle mExtras; /** Creator for this class. Required to have in parcelable implementations. */ public static final Creator CREATOR = new Creator() { @Override public MobileDevice createFromParcel(Parcel source) { return new MobileDevice(source); } @Override public MobileDevice[] newArray(int size) { return new MobileDevice[size]; } }; private MobileDevice(Builder builder) { mId = builder.mId; mName = builder.mName; mAvailableTransports = builder.mAvailableTransports.toArray(); mProjecting = builder.mProjecting; mExtras = builder.mExtras == null ? null : new Bundle(builder.mExtras); } private MobileDevice(Parcel source) { mId = source.readInt(); mName = source.readString(); mAvailableTransports = source.createIntArray(); mProjecting = source.readBoolean(); mExtras = source.readBundle(getClass().getClassLoader()); } @Override public void writeToParcel(Parcel dest, int flags) { dest.writeInt(mId); dest.writeString(mName); dest.writeIntArray(mAvailableTransports); dest.writeBoolean(mProjecting); dest.writeBundle(mExtras); } /** Returns the device id which uniquely identifies the mobile device within projection */ public int getId() { return mId; } /** Returns the name of the device */ public @NonNull String getName() { return mName; } /** Returns a list of available projection transports. See {@code PROJECTION_TRANSPORT_*} * for possible values. */ public @NonNull List getAvailableTransports() { List transports = new ArrayList<>(mAvailableTransports.length); for (int transport : mAvailableTransports) { transports.add(transport); } return transports; } /** Indicates whether this mobile device is currently projecting */ public boolean isProjecting() { return mProjecting; } /** Returns extra information for mobile device */ public @NonNull Bundle getExtras() { return mExtras == null ? new Bundle() : new Bundle(mExtras); } /** Parcelable implementation */ @Override public int describeContents() { return 0; } /** * Creates new instance of {@link Builder} * * @param id uniquely identifies the device * @param name name of the connected device * @return the instance of {@link Builder} */ public static @NonNull Builder builder(int id, String name) { return new Builder(id, name); } @Override public String toString() { return "MobileDevice{" + "mId=" + mId + ", mName='" + mName + '\'' + ", mAvailableTransports=" + Arrays.toString(mAvailableTransports) + ", mProjecting=" + mProjecting + (mExtras != null ? ", (has extras)" : "") + '}'; } /** * Builder class for {@link MobileDevice} */ public static final class Builder { private int mId; private String mName; private IntArray mAvailableTransports = new IntArray(); private boolean mProjecting; private Bundle mExtras; private Builder(int id, String name) { mId = id; if (name == null) { throw new IllegalArgumentException("Name of the device can't be null"); } mName = name; } /** * Add supported transport * * @param transport supported transport by given device, must be one of the * {@code PROJECTION_TRANSPORT_*} * @return this builder */ public @NonNull Builder addTransport(@ProjectionTransport int transport) { checkProjectionTransport(transport); mAvailableTransports.add(transport); return this; } /** * Indicate whether the mobile device currently projecting or not. * * @param projecting {@code True} if this mobile device currently projecting * @return this builder */ public @NonNull Builder setProjecting(boolean projecting) { mProjecting = projecting; return this; } /** * Add extra information for mobile device * * @param extras provides an arbitrary extra information about this mobile device * @return this builder */ public @NonNull Builder setExtras(Bundle extras) { mExtras = extras; return this; } /** Creates new instance of {@link MobileDevice} */ public @NonNull MobileDevice build() { return new MobileDevice(this); } } } }