1 /* 2 * Copyright (C) 2014 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 package android.os; 18 19 import android.annotation.Nullable; 20 import android.util.ArrayMap; 21 import com.android.internal.util.XmlUtils; 22 import org.xmlpull.v1.XmlPullParser; 23 import org.xmlpull.v1.XmlPullParserException; 24 import org.xmlpull.v1.XmlSerializer; 25 26 import java.io.IOException; 27 import java.util.Iterator; 28 import java.util.Map; 29 import java.util.Set; 30 31 /** 32 * A mapping from String values to various types that can be saved to persistent and later 33 * restored. 34 * 35 */ 36 public final class PersistableBundle extends BaseBundle implements Cloneable, Parcelable, 37 XmlUtils.WriteMapCallback { 38 private static final String TAG_PERSISTABLEMAP = "pbundle_as_map"; 39 public static final PersistableBundle EMPTY; 40 static final Parcel EMPTY_PARCEL; 41 42 static { 43 EMPTY = new PersistableBundle(); 44 EMPTY.mMap = ArrayMap.EMPTY; 45 EMPTY_PARCEL = BaseBundle.EMPTY_PARCEL; 46 } 47 48 /** @hide */ isValidType(Object value)49 public static boolean isValidType(Object value) { 50 return (value instanceof Integer) || (value instanceof Long) || 51 (value instanceof Double) || (value instanceof String) || 52 (value instanceof int[]) || (value instanceof long[]) || 53 (value instanceof double[]) || (value instanceof String[]) || 54 (value instanceof PersistableBundle) || (value == null) || 55 (value instanceof Boolean) || (value instanceof boolean[]); 56 } 57 58 /** 59 * Constructs a new, empty PersistableBundle. 60 */ PersistableBundle()61 public PersistableBundle() { 62 super(); 63 } 64 65 /** 66 * Constructs a new, empty PersistableBundle sized to hold the given number of 67 * elements. The PersistableBundle will grow as needed. 68 * 69 * @param capacity the initial capacity of the PersistableBundle 70 */ PersistableBundle(int capacity)71 public PersistableBundle(int capacity) { 72 super(capacity); 73 } 74 75 /** 76 * Constructs a PersistableBundle containing a copy of the mappings from the given 77 * PersistableBundle. 78 * 79 * @param b a PersistableBundle to be copied. 80 */ PersistableBundle(PersistableBundle b)81 public PersistableBundle(PersistableBundle b) { 82 super(b); 83 } 84 85 /** 86 * Constructs a PersistableBundle containing the mappings passed in. 87 * 88 * @param map a Map containing only those items that can be persisted. 89 * @throws IllegalArgumentException if any element of #map cannot be persisted. 90 */ PersistableBundle(ArrayMap<String, Object> map)91 private PersistableBundle(ArrayMap<String, Object> map) { 92 super(); 93 94 // First stuff everything in. 95 putAll(map); 96 97 // Now verify each item throwing an exception if there is a violation. 98 final int N = mMap.size(); 99 for (int i=0; i<N; i++) { 100 Object value = mMap.valueAt(i); 101 if (value instanceof ArrayMap) { 102 // Fix up any Maps by replacing them with PersistableBundles. 103 mMap.setValueAt(i, new PersistableBundle((ArrayMap<String, Object>) value)); 104 } else if (!isValidType(value)) { 105 throw new IllegalArgumentException("Bad value in PersistableBundle key=" 106 + mMap.keyAt(i) + " value=" + value); 107 } 108 } 109 } 110 PersistableBundle(Parcel parcelledData, int length)111 /* package */ PersistableBundle(Parcel parcelledData, int length) { 112 super(parcelledData, length); 113 } 114 115 /** 116 * Make a PersistableBundle for a single key/value pair. 117 * 118 * @hide 119 */ forPair(String key, String value)120 public static PersistableBundle forPair(String key, String value) { 121 PersistableBundle b = new PersistableBundle(1); 122 b.putString(key, value); 123 return b; 124 } 125 126 /** 127 * Clones the current PersistableBundle. The internal map is cloned, but the keys and 128 * values to which it refers are copied by reference. 129 */ 130 @Override clone()131 public Object clone() { 132 return new PersistableBundle(this); 133 } 134 135 /** 136 * Inserts a PersistableBundle value into the mapping of this Bundle, replacing 137 * any existing value for the given key. Either key or value may be null. 138 * 139 * @param key a String, or null 140 * @param value a Bundle object, or null 141 */ putPersistableBundle(@ullable String key, @Nullable PersistableBundle value)142 public void putPersistableBundle(@Nullable String key, @Nullable PersistableBundle value) { 143 unparcel(); 144 mMap.put(key, value); 145 } 146 147 /** 148 * Returns the value associated with the given key, or null if 149 * no mapping of the desired type exists for the given key or a null 150 * value is explicitly associated with the key. 151 * 152 * @param key a String, or null 153 * @return a Bundle value, or null 154 */ 155 @Nullable getPersistableBundle(@ullable String key)156 public PersistableBundle getPersistableBundle(@Nullable String key) { 157 unparcel(); 158 Object o = mMap.get(key); 159 if (o == null) { 160 return null; 161 } 162 try { 163 return (PersistableBundle) o; 164 } catch (ClassCastException e) { 165 typeWarning(key, o, "Bundle", e); 166 return null; 167 } 168 } 169 170 public static final Parcelable.Creator<PersistableBundle> CREATOR = 171 new Parcelable.Creator<PersistableBundle>() { 172 @Override 173 public PersistableBundle createFromParcel(Parcel in) { 174 return in.readPersistableBundle(); 175 } 176 177 @Override 178 public PersistableBundle[] newArray(int size) { 179 return new PersistableBundle[size]; 180 } 181 }; 182 183 /** @hide */ 184 @Override writeUnknownObject(Object v, String name, XmlSerializer out)185 public void writeUnknownObject(Object v, String name, XmlSerializer out) 186 throws XmlPullParserException, IOException { 187 if (v instanceof PersistableBundle) { 188 out.startTag(null, TAG_PERSISTABLEMAP); 189 out.attribute(null, "name", name); 190 ((PersistableBundle) v).saveToXml(out); 191 out.endTag(null, TAG_PERSISTABLEMAP); 192 } else { 193 throw new XmlPullParserException("Unknown Object o=" + v); 194 } 195 } 196 197 /** @hide */ saveToXml(XmlSerializer out)198 public void saveToXml(XmlSerializer out) throws IOException, XmlPullParserException { 199 unparcel(); 200 XmlUtils.writeMapXml(mMap, out, this); 201 } 202 203 /** @hide */ 204 static class MyReadMapCallback implements XmlUtils.ReadMapCallback { 205 @Override readThisUnknownObjectXml(XmlPullParser in, String tag)206 public Object readThisUnknownObjectXml(XmlPullParser in, String tag) 207 throws XmlPullParserException, IOException { 208 if (TAG_PERSISTABLEMAP.equals(tag)) { 209 return restoreFromXml(in); 210 } 211 throw new XmlPullParserException("Unknown tag=" + tag); 212 } 213 } 214 215 /** 216 * Report the nature of this Parcelable's contents 217 */ 218 @Override describeContents()219 public int describeContents() { 220 return 0; 221 } 222 223 /** 224 * Writes the PersistableBundle contents to a Parcel, typically in order for 225 * it to be passed through an IBinder connection. 226 * @param parcel The parcel to copy this bundle to. 227 */ 228 @Override writeToParcel(Parcel parcel, int flags)229 public void writeToParcel(Parcel parcel, int flags) { 230 final boolean oldAllowFds = parcel.pushAllowFds(false); 231 try { 232 writeToParcelInner(parcel, flags); 233 } finally { 234 parcel.restoreAllowFds(oldAllowFds); 235 } 236 } 237 238 /** @hide */ restoreFromXml(XmlPullParser in)239 public static PersistableBundle restoreFromXml(XmlPullParser in) throws IOException, 240 XmlPullParserException { 241 final int outerDepth = in.getDepth(); 242 final String startTag = in.getName(); 243 final String[] tagName = new String[1]; 244 int event; 245 while (((event = in.next()) != XmlPullParser.END_DOCUMENT) && 246 (event != XmlPullParser.END_TAG || in.getDepth() < outerDepth)) { 247 if (event == XmlPullParser.START_TAG) { 248 return new PersistableBundle((ArrayMap<String, Object>) 249 XmlUtils.readThisArrayMapXml(in, startTag, tagName, 250 new MyReadMapCallback())); 251 } 252 } 253 return EMPTY; 254 } 255 256 @Override toString()257 synchronized public String toString() { 258 if (mParcelledData != null) { 259 if (mParcelledData == EMPTY_PARCEL) { 260 return "PersistableBundle[EMPTY_PARCEL]"; 261 } else { 262 return "PersistableBundle[mParcelledData.dataSize=" + 263 mParcelledData.dataSize() + "]"; 264 } 265 } 266 return "PersistableBundle[" + mMap.toString() + "]"; 267 } 268 269 } 270