package android.view;
import android.annotation.NonNull;
import android.annotation.Nullable;
import android.compat.annotation.UnsupportedAppUsage;
import java.io.ByteArrayOutputStream;
import java.io.DataOutputStream;
import java.io.IOException;
import java.nio.charset.Charset;
import java.util.HashMap;
import java.util.Map;
/**
* {@link ViewHierarchyEncoder} is a serializer that is tailored towards writing out
* view hierarchies (the view tree, along with the properties for each view) to a stream.
*
* It is typically used as follows:
*
* ViewHierarchyEncoder e = new ViewHierarchyEncoder();
*
* for (View view : views) {
* e.beginObject(view);
* e.addProperty("prop1", value);
* ...
* e.endObject();
* }
*
* // repeat above snippet for each view, finally end with:
* e.endStream();
*
*
* On the stream, a snippet such as the above gets encoded as a series of Map's (one
* corresponding to each view) with the property name as the key and the property value
* as the value.
*
*
Since the property names are practically the same across all views, rather than using
* the property name directly as the key, we use a short integer id corresponding to each
* property name as the key. A final map is added at the end which contains the mapping
* from the integer to its property name.
*
*
A value is encoded as a single byte type identifier followed by the encoding of the
* value. Only primitive types are supported as values, in addition to the Map type.
*
* @hide
*/
public class ViewHierarchyEncoder {
// Prefixes for simple primitives. These match the JNI definitions.
private static final byte SIG_BOOLEAN = 'Z';
private static final byte SIG_BYTE = 'B';
private static final byte SIG_SHORT = 'S';
private static final byte SIG_INT = 'I';
private static final byte SIG_LONG = 'J';
private static final byte SIG_FLOAT = 'F';
private static final byte SIG_DOUBLE = 'D';
// Prefixes for some commonly used objects
private static final byte SIG_STRING = 'R';
private static final byte SIG_MAP = 'M'; // a map with an short key
private static final short SIG_END_MAP = 0;
private final DataOutputStream mStream;
private final Map mPropertyNames = new HashMap(200);
private short mPropertyId = 1;
private Charset mCharset = Charset.forName("utf-8");
private boolean mUserPropertiesEnabled = true;
public ViewHierarchyEncoder(@NonNull ByteArrayOutputStream stream) {
mStream = new DataOutputStream(stream);
}
public void setUserPropertiesEnabled(boolean enabled) {
mUserPropertiesEnabled = enabled;
}
public void beginObject(@NonNull Object o) {
startPropertyMap();
addProperty("meta:__name__", o.getClass().getName());
addProperty("meta:__hash__", o.hashCode());
}
public void endObject() {
endPropertyMap();
}
public void endStream() {
// write out the string table
startPropertyMap();
addProperty("__name__", "propertyIndex");
for (Map.Entry entry : mPropertyNames.entrySet()) {
writeShort(entry.getValue());
writeString(entry.getKey());
}
endPropertyMap();
}
@UnsupportedAppUsage
public void addProperty(@NonNull String name, boolean v) {
writeShort(createPropertyIndex(name));
writeBoolean(v);
}
public void addProperty(@NonNull String name, short s) {
writeShort(createPropertyIndex(name));
writeShort(s);
}
@UnsupportedAppUsage
public void addProperty(@NonNull String name, int v) {
writeShort(createPropertyIndex(name));
writeInt(v);
}
@UnsupportedAppUsage
public void addProperty(@NonNull String name, float v) {
writeShort(createPropertyIndex(name));
writeFloat(v);
}
@UnsupportedAppUsage
public void addProperty(@NonNull String name, @Nullable String s) {
writeShort(createPropertyIndex(name));
writeString(s);
}
/**
* Encodes a user defined property if they are allowed to be encoded
*
* @see #setUserPropertiesEnabled(boolean)
*/
public void addUserProperty(@NonNull String name, @Nullable String s) {
if (mUserPropertiesEnabled) {
addProperty(name, s);
}
}
/**
* Writes the given name as the property name, and leaves it to the callee
* to fill in value for this property.
*/
public void addPropertyKey(@NonNull String name) {
writeShort(createPropertyIndex(name));
}
private short createPropertyIndex(@NonNull String name) {
Short index = mPropertyNames.get(name);
if (index == null) {
index = mPropertyId++;
mPropertyNames.put(name, index);
}
return index;
}
private void startPropertyMap() {
try {
mStream.write(SIG_MAP);
} catch (IOException e) {
// does not happen since the stream simply wraps a ByteArrayOutputStream
}
}
private void endPropertyMap() {
writeShort(SIG_END_MAP);
}
private void writeBoolean(boolean v) {
try {
mStream.write(SIG_BOOLEAN);
mStream.write(v ? 1 : 0);
} catch (IOException e) {
// does not happen since the stream simply wraps a ByteArrayOutputStream
}
}
private void writeShort(short s) {
try {
mStream.write(SIG_SHORT);
mStream.writeShort(s);
} catch (IOException e) {
// does not happen since the stream simply wraps a ByteArrayOutputStream
}
}
private void writeInt(int i) {
try {
mStream.write(SIG_INT);
mStream.writeInt(i);
} catch (IOException e) {
// does not happen since the stream simply wraps a ByteArrayOutputStream
}
}
private void writeFloat(float v) {
try {
mStream.write(SIG_FLOAT);
mStream.writeFloat(v);
} catch (IOException e) {
// does not happen since the stream simply wraps a ByteArrayOutputStream
}
}
private void writeString(@Nullable String s) {
if (s == null) {
s = "";
}
try {
mStream.write(SIG_STRING);
byte[] bytes = s.getBytes(mCharset);
short len = (short)Math.min(bytes.length, Short.MAX_VALUE);
mStream.writeShort(len);
mStream.write(bytes, 0, len);
} catch (IOException e) {
// does not happen since the stream simply wraps a ByteArrayOutputStream
}
}
}