1 /* 2 * Copyright (C) 2013 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.printservice; 18 19 import android.annotation.NonNull; 20 import android.annotation.SystemApi; 21 import android.content.ComponentName; 22 import android.content.Context; 23 import android.content.pm.PackageManager; 24 import android.content.pm.PackageManager.NameNotFoundException; 25 import android.content.pm.ResolveInfo; 26 import android.content.res.Resources; 27 import android.content.res.TypedArray; 28 import android.content.res.XmlResourceParser; 29 import android.os.Parcel; 30 import android.os.Parcelable; 31 import android.util.AttributeSet; 32 import android.util.Log; 33 import android.util.Xml; 34 35 import org.xmlpull.v1.XmlPullParser; 36 import org.xmlpull.v1.XmlPullParserException; 37 38 import java.io.IOException; 39 40 /** 41 * This class describes a {@link PrintService}. A print service knows 42 * how to communicate with one or more printers over one or more protocols 43 * and exposes printers for use by the applications via the platform print 44 * APIs. 45 * 46 * @see PrintService 47 * @see android.print.PrintManager 48 * 49 * @hide 50 */ 51 @SystemApi 52 public final class PrintServiceInfo implements Parcelable { 53 54 private static final String LOG_TAG = PrintServiceInfo.class.getSimpleName(); 55 56 private static final String TAG_PRINT_SERVICE = "print-service"; 57 58 private final String mId; 59 60 private boolean mIsEnabled; 61 62 private final ResolveInfo mResolveInfo; 63 64 private final String mSettingsActivityName; 65 66 private final String mAddPrintersActivityName; 67 68 private final String mAdvancedPrintOptionsActivityName; 69 70 /** 71 * Creates a new instance. 72 * 73 * @hide 74 */ PrintServiceInfo(Parcel parcel)75 public PrintServiceInfo(Parcel parcel) { 76 mId = parcel.readString(); 77 mIsEnabled = parcel.readByte() != 0; 78 mResolveInfo = parcel.readParcelable(null); 79 mSettingsActivityName = parcel.readString(); 80 mAddPrintersActivityName = parcel.readString(); 81 mAdvancedPrintOptionsActivityName = parcel.readString(); 82 } 83 84 /** 85 * Creates a new instance. 86 * 87 * @param resolveInfo The service resolve info. 88 * @param settingsActivityName Optional settings activity name. 89 * @param addPrintersActivityName Optional add printers activity name. 90 * @param advancedPrintOptionsActivityName Optional advanced print options activity. 91 * 92 * @hide 93 */ PrintServiceInfo(ResolveInfo resolveInfo, String settingsActivityName, String addPrintersActivityName, String advancedPrintOptionsActivityName)94 public PrintServiceInfo(ResolveInfo resolveInfo, String settingsActivityName, 95 String addPrintersActivityName, String advancedPrintOptionsActivityName) { 96 mId = new ComponentName(resolveInfo.serviceInfo.packageName, 97 resolveInfo.serviceInfo.name).flattenToString(); 98 mResolveInfo = resolveInfo; 99 mSettingsActivityName = settingsActivityName; 100 mAddPrintersActivityName = addPrintersActivityName; 101 mAdvancedPrintOptionsActivityName = advancedPrintOptionsActivityName; 102 } 103 104 /** 105 * Return the component name for this print service. 106 * 107 * @return The component name for this print service. 108 */ getComponentName()109 public @NonNull ComponentName getComponentName() { 110 return new ComponentName(mResolveInfo.serviceInfo.packageName, 111 mResolveInfo.serviceInfo.name); 112 } 113 114 /** 115 * Creates a new instance. 116 * 117 * @param context Context for accessing resources. 118 * @param resolveInfo The service resolve info. 119 * @return The created instance. 120 * 121 * @hide 122 */ create(Context context, ResolveInfo resolveInfo)123 public static PrintServiceInfo create(Context context, ResolveInfo resolveInfo) { 124 String settingsActivityName = null; 125 String addPrintersActivityName = null; 126 String advancedPrintOptionsActivityName = null; 127 128 XmlResourceParser parser = null; 129 PackageManager packageManager = context.getPackageManager(); 130 parser = resolveInfo.serviceInfo.loadXmlMetaData(packageManager, 131 PrintService.SERVICE_META_DATA); 132 if (parser != null) { 133 try { 134 int type = 0; 135 while (type != XmlPullParser.END_DOCUMENT && type != XmlPullParser.START_TAG) { 136 type = parser.next(); 137 } 138 139 String nodeName = parser.getName(); 140 if (!TAG_PRINT_SERVICE.equals(nodeName)) { 141 Log.e(LOG_TAG, "Ignoring meta-data that does not start with " 142 + TAG_PRINT_SERVICE + " tag"); 143 } else { 144 Resources resources = packageManager.getResourcesForApplication( 145 resolveInfo.serviceInfo.applicationInfo); 146 AttributeSet allAttributes = Xml.asAttributeSet(parser); 147 TypedArray attributes = resources.obtainAttributes(allAttributes, 148 com.android.internal.R.styleable.PrintService); 149 150 settingsActivityName = attributes.getString( 151 com.android.internal.R.styleable.PrintService_settingsActivity); 152 153 addPrintersActivityName = attributes.getString( 154 com.android.internal.R.styleable.PrintService_addPrintersActivity); 155 156 advancedPrintOptionsActivityName = attributes.getString(com.android.internal 157 .R.styleable.PrintService_advancedPrintOptionsActivity); 158 159 attributes.recycle(); 160 } 161 } catch (IOException ioe) { 162 Log.w(LOG_TAG, "Error reading meta-data:" + ioe); 163 } catch (XmlPullParserException xppe) { 164 Log.w(LOG_TAG, "Error reading meta-data:" + xppe); 165 } catch (NameNotFoundException e) { 166 Log.e(LOG_TAG, "Unable to load resources for: " 167 + resolveInfo.serviceInfo.packageName); 168 } finally { 169 if (parser != null) { 170 parser.close(); 171 } 172 } 173 } 174 175 return new PrintServiceInfo(resolveInfo, settingsActivityName, 176 addPrintersActivityName, advancedPrintOptionsActivityName); 177 } 178 179 /** 180 * The accessibility service id. 181 * <p> 182 * <strong>Generated by the system.</strong> 183 * </p> 184 * 185 * @return The id. 186 * 187 * @hide 188 */ getId()189 public String getId() { 190 return mId; 191 } 192 193 /** 194 * If the service was enabled when it was read from the system. 195 * 196 * @return The id. 197 * 198 * @hide 199 */ isEnabled()200 public boolean isEnabled() { 201 return mIsEnabled; 202 } 203 204 /** 205 * Mark a service as enabled or not 206 * 207 * @param isEnabled If the service should be marked as enabled. 208 * 209 * @hide 210 */ setIsEnabled(boolean isEnabled)211 public void setIsEnabled(boolean isEnabled) { 212 mIsEnabled = isEnabled; 213 } 214 215 /** 216 * The service {@link ResolveInfo}. 217 * 218 * @return The info. 219 * 220 * @hide 221 */ getResolveInfo()222 public ResolveInfo getResolveInfo() { 223 return mResolveInfo; 224 } 225 226 /** 227 * The settings activity name. 228 * <p> 229 * <strong>Statically set from 230 * {@link PrintService#SERVICE_META_DATA meta-data}.</strong> 231 * </p> 232 * 233 * @return The settings activity name. 234 * 235 * @hide 236 */ getSettingsActivityName()237 public String getSettingsActivityName() { 238 return mSettingsActivityName; 239 } 240 241 /** 242 * The add printers activity name. 243 * <p> 244 * <strong>Statically set from 245 * {@link PrintService#SERVICE_META_DATA meta-data}.</strong> 246 * </p> 247 * 248 * @return The add printers activity name. 249 * 250 * @hide 251 */ getAddPrintersActivityName()252 public String getAddPrintersActivityName() { 253 return mAddPrintersActivityName; 254 } 255 256 /** 257 * The advanced print options activity name. 258 * <p> 259 * <strong>Statically set from 260 * {@link PrintService#SERVICE_META_DATA meta-data}.</strong> 261 * </p> 262 * 263 * @return The advanced print options activity name. 264 * 265 * @hide 266 */ getAdvancedOptionsActivityName()267 public String getAdvancedOptionsActivityName() { 268 return mAdvancedPrintOptionsActivityName; 269 } 270 271 /** 272 * {@inheritDoc} 273 */ 274 @Override describeContents()275 public int describeContents() { 276 return 0; 277 } 278 279 @Override writeToParcel(Parcel parcel, int flagz)280 public void writeToParcel(Parcel parcel, int flagz) { 281 parcel.writeString(mId); 282 parcel.writeByte((byte)(mIsEnabled ? 1 : 0)); 283 parcel.writeParcelable(mResolveInfo, 0); 284 parcel.writeString(mSettingsActivityName); 285 parcel.writeString(mAddPrintersActivityName); 286 parcel.writeString(mAdvancedPrintOptionsActivityName); 287 } 288 289 @Override hashCode()290 public int hashCode() { 291 return 31 + ((mId == null) ? 0 : mId.hashCode()); 292 } 293 294 @Override equals(Object obj)295 public boolean equals(Object obj) { 296 if (this == obj) { 297 return true; 298 } 299 if (obj == null) { 300 return false; 301 } 302 if (getClass() != obj.getClass()) { 303 return false; 304 } 305 PrintServiceInfo other = (PrintServiceInfo) obj; 306 if (mId == null) { 307 if (other.mId != null) { 308 return false; 309 } 310 } else if (!mId.equals(other.mId)) { 311 return false; 312 } 313 return true; 314 } 315 316 @Override toString()317 public String toString() { 318 StringBuilder builder = new StringBuilder(); 319 builder.append("PrintServiceInfo{"); 320 builder.append("id=").append(mId); 321 builder.append("isEnabled=").append(mIsEnabled); 322 builder.append(", resolveInfo=").append(mResolveInfo); 323 builder.append(", settingsActivityName=").append(mSettingsActivityName); 324 builder.append(", addPrintersActivityName=").append(mAddPrintersActivityName); 325 builder.append(", advancedPrintOptionsActivityName=") 326 .append(mAdvancedPrintOptionsActivityName); 327 builder.append("}"); 328 return builder.toString(); 329 } 330 331 public static final @android.annotation.NonNull Parcelable.Creator<PrintServiceInfo> CREATOR = 332 new Parcelable.Creator<PrintServiceInfo>() { 333 @Override 334 public PrintServiceInfo createFromParcel(Parcel parcel) { 335 return new PrintServiceInfo(parcel); 336 } 337 338 @Override 339 public PrintServiceInfo[] newArray(int size) { 340 return new PrintServiceInfo[size]; 341 } 342 }; 343 } 344