/* * Copyright (C) 2009 Google Inc. All rights reserved. * * 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 com.google.polo.pairing.message; import java.util.HashSet; import java.util.Iterator; import java.util.Set; /** * Object implementing the internal representation of the protocol message * 'OPTIONS'. */ public class OptionsMessage extends PoloMessage { /** * Representation of a specific role type. The numeric values * should be kept synchronized with the constants defined by * Options.Encoding.EncodingType in polo.proto. */ public static enum ProtocolRole { UNKNOWN(0), INPUT_DEVICE(1), DISPLAY_DEVICE(2); private final int mIntVal; private ProtocolRole(int intVal) { mIntVal = intVal; } public static ProtocolRole fromIntVal(int intVal) { for (ProtocolRole role : ProtocolRole.values()) { if (role.getAsInt() == intVal) { return role; } } return ProtocolRole.UNKNOWN; } public int getAsInt() { return mIntVal; } } /** * The preferred protocol role of the sender. */ private ProtocolRole mProtocolRolePreference; /** * Sender's supported input encodings. */ private Set<EncodingOption> mInputEncodings; /** * Sender's supported output encodings. */ private Set<EncodingOption> mOutputEncodings; public OptionsMessage() { super(PoloMessage.PoloMessageType.OPTIONS); mProtocolRolePreference = ProtocolRole.UNKNOWN; mInputEncodings = new HashSet<EncodingOption>(); mOutputEncodings = new HashSet<EncodingOption>(); } public void setProtocolRolePreference(ProtocolRole pref) { mProtocolRolePreference = pref; } public ProtocolRole getProtocolRolePreference() { return mProtocolRolePreference; } public void addInputEncoding(EncodingOption encoding) { mInputEncodings.add(encoding); } public void addOutputEncoding(EncodingOption encoding) { mOutputEncodings.add(encoding); } public boolean supportsInputEncoding(EncodingOption encoding) { return mInputEncodings.contains(encoding); } public boolean supportsOutputEncoding(EncodingOption encoding) { return mOutputEncodings.contains(encoding); } public Set<EncodingOption> getInputEncodingSet() { return new HashSet<EncodingOption>(mInputEncodings); } public Set<EncodingOption> getOutputEncodingSet() { return new HashSet<EncodingOption>(mOutputEncodings); } /** * Generates a ConfigurationMessage, or {@code null}, by intersecting the * local message with another message. */ public ConfigurationMessage getBestConfiguration( OptionsMessage otherOptions) { Set<EncodingOption> peerOutputs = otherOptions.getOutputEncodingSet(); peerOutputs.retainAll(mInputEncodings); Set<EncodingOption> peerInputs = otherOptions.getInputEncodingSet(); peerInputs.retainAll(mOutputEncodings); if (peerOutputs.isEmpty() && peerInputs.isEmpty()) { // No configuration possible: no common encodings. return null; } EncodingOption encoding; ProtocolRole role; EncodingOption bestInput = null; if (!peerInputs.isEmpty()) { bestInput = peerInputs.iterator().next(); } EncodingOption bestOutput = null; if (!peerOutputs.isEmpty()) { bestOutput = peerOutputs.iterator().next(); } if (mProtocolRolePreference == ProtocolRole.DISPLAY_DEVICE) { // We prefer to be the display device if (bestInput != null) { encoding = bestInput; role = ProtocolRole.DISPLAY_DEVICE; } else { encoding = bestOutput; role = ProtocolRole.INPUT_DEVICE; } } else { // We prefer to be the input device, or have no preference if (!peerOutputs.isEmpty()) { encoding = bestOutput; role = ProtocolRole.INPUT_DEVICE; } else { encoding = bestInput; role = ProtocolRole.DISPLAY_DEVICE; } } return new ConfigurationMessage(encoding, role); } @Override public String toString() { StringBuilder ret = new StringBuilder(); ret.append("[" + getType() + " "); ret.append("inputs="); Iterator<EncodingOption> inputIterator = mInputEncodings.iterator(); while (inputIterator.hasNext()) { ret.append(inputIterator.next()); if (inputIterator.hasNext()) { ret.append(","); } } ret.append(" outputs="); Iterator<EncodingOption> outputIterator = mOutputEncodings.iterator(); while (outputIterator.hasNext()) { ret.append(outputIterator.next()); if (outputIterator.hasNext()) { ret.append(","); } } ret.append(" pref=" + mProtocolRolePreference); ret.append("]"); return ret.toString(); } @Override public boolean equals(Object obj) { if (this == obj) { return true; } if (!(obj instanceof OptionsMessage)) { return false; } OptionsMessage other = (OptionsMessage) obj; if (mProtocolRolePreference == null) { if (other.mProtocolRolePreference != null) { return false; } } else if (!mProtocolRolePreference.equals(other.mProtocolRolePreference)) { return false; } return mInputEncodings.equals(other.mInputEncodings) && mOutputEncodings.equals(other.mOutputEncodings); } }