| /* |
| * Copyright (C) 2013 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.printservice; |
| |
| import android.annotation.NonNull; |
| import android.annotation.Nullable; |
| import android.app.Activity; |
| import android.app.PendingIntent; |
| import android.app.Service; |
| import android.content.ComponentName; |
| import android.content.Context; |
| import android.content.Intent; |
| import android.os.Handler; |
| import android.os.IBinder; |
| import android.os.Looper; |
| import android.os.Message; |
| import android.os.RemoteException; |
| import android.print.PrintJobInfo; |
| import android.print.PrinterId; |
| import android.print.PrinterInfo; |
| import android.util.Log; |
| |
| import com.android.internal.util.Preconditions; |
| |
| import java.util.ArrayList; |
| import java.util.Collections; |
| import java.util.List; |
| |
| /** |
| * <p> |
| * This is the base class for implementing print services. A print service knows |
| * how to discover and interact one or more printers via one or more protocols. |
| * </p> |
| * <h3>Printer discovery</h3> |
| * <p> |
| * A print service is responsible for discovering printers, adding discovered printers, |
| * removing added printers, and updating added printers. When the system is interested |
| * in printers managed by your service it will call {@link |
| * #onCreatePrinterDiscoverySession()} from which you must return a new {@link |
| * PrinterDiscoverySession} instance. The returned session encapsulates the interaction |
| * between the system and your service during printer discovery. For description of this |
| * interaction refer to the documentation for {@link PrinterDiscoverySession}. |
| * </p> |
| * <p> |
| * For every printer discovery session all printers have to be added since system does |
| * not retain printers across sessions. Hence, each printer known to this print service |
| * should be added only once during a discovery session. Only an already added printer |
| * can be removed or updated. Removed printers can be added again. |
| * </p> |
| * <h3>Print jobs</h3> |
| * <p> |
| * When a new print job targeted to a printer managed by this print service is is queued, |
| * i.e. ready for processing by the print service, you will receive a call to {@link |
| * #onPrintJobQueued(PrintJob)}. The print service may handle the print job immediately |
| * or schedule that for an appropriate time in the future. The list of all active print |
| * jobs for this service is obtained by calling {@link #getActivePrintJobs()}. Active |
| * print jobs are ones that are queued or started. |
| * </p> |
| * <p> |
| * A print service is responsible for setting a print job's state as appropriate |
| * while processing it. Initially, a print job is queued, i.e. {@link PrintJob#isQueued() |
| * PrintJob.isQueued()} returns true, which means that the document to be printed is |
| * spooled by the system and the print service can begin processing it. You can obtain |
| * the printed document by calling {@link PrintJob#getDocument() PrintJob.getDocument()} |
| * whose data is accessed via {@link PrintDocument#getData() PrintDocument.getData()}. |
| * After the print service starts printing the data it should set the print job's |
| * state to started by calling {@link PrintJob#start()} after which |
| * {@link PrintJob#isStarted() PrintJob.isStarted()} would return true. Upon successful |
| * completion, the print job should be marked as completed by calling {@link |
| * PrintJob#complete() PrintJob.complete()} after which {@link PrintJob#isCompleted() |
| * PrintJob.isCompleted()} would return true. In case of a failure, the print job should |
| * be marked as failed by calling {@link PrintJob#fail(String) PrintJob.fail( |
| * String)} after which {@link PrintJob#isFailed() PrintJob.isFailed()} would |
| * return true. |
| * </p> |
| * <p> |
| * If a print job is queued or started and the user requests to cancel it, the print |
| * service will receive a call to {@link #onRequestCancelPrintJob(PrintJob)} which |
| * requests from the service to do best effort in canceling the job. In case the job |
| * is successfully canceled, its state has to be marked as cancelled by calling {@link |
| * PrintJob#cancel() PrintJob.cancel()} after which {@link PrintJob#isCancelled() |
| * PrintJob.isCacnelled()} would return true. |
| * </p> |
| * <h3>Lifecycle</h3> |
| * <p> |
| * The lifecycle of a print service is managed exclusively by the system and follows |
| * the established service lifecycle. Additionally, starting or stopping a print service |
| * is triggered exclusively by an explicit user action through enabling or disabling it |
| * in the device settings. After the system binds to a print service, it calls {@link |
| * #onConnected()}. This method can be overriden by clients to perform post binding setup. |
| * Also after the system unbinds from a print service, it calls {@link #onDisconnected()}. |
| * This method can be overriden by clients to perform post unbinding cleanup. Your should |
| * not do any work after the system disconnected from your print service since the |
| * service can be killed at any time to reclaim memory. The system will not disconnect |
| * from a print service if there are active print jobs for the printers managed by it. |
| * </p> |
| * <h3>Declaration</h3> |
| * <p> |
| * A print service is declared as any other service in an AndroidManifest.xml but it must |
| * also specify that it handles the {@link android.content.Intent} with action {@link |
| * #SERVICE_INTERFACE android.printservice.PrintService}. Failure to declare this intent |
| * will cause the system to ignore the print service. Additionally, a print service must |
| * request the {@link android.Manifest.permission#BIND_PRINT_SERVICE |
| * android.permission.BIND_PRINT_SERVICE} permission to ensure that only the system can |
| * bind to it. Failure to declare this intent will cause the system to ignore the print |
| * service. Following is an example declaration: |
| * </p> |
| * <pre> |
| * <service android:name=".MyPrintService" |
| * android:permission="android.permission.BIND_PRINT_SERVICE"> |
| * <intent-filter> |
| * <action android:name="android.printservice.PrintService" /> |
| * </intent-filter> |
| * . . . |
| * </service> |
| * </pre> |
| * <h3>Configuration</h3> |
| * <p> |
| * A print service can be configured by specifying an optional settings activity which |
| * exposes service specific settings, an optional add printers activity which is used for |
| * manual addition of printers, vendor name ,etc. It is a responsibility of the system |
| * to launch the settings and add printers activities when appropriate. |
| * </p> |
| * <p> |
| * A print service is configured by providing a {@link #SERVICE_META_DATA meta-data} |
| * entry in the manifest when declaring the service. A service declaration with a meta-data |
| * tag is presented below: |
| * <pre> <service android:name=".MyPrintService" |
| * android:permission="android.permission.BIND_PRINT_SERVICE"> |
| * <intent-filter> |
| * <action android:name="android.printservice.PrintService" /> |
| * </intent-filter> |
| * <meta-data android:name="android.printservice" android:resource="@xml/printservice" /> |
| * </service></pre> |
| * </p> |
| * <p> |
| * For more details for how to configure your print service via the meta-data refer to |
| * {@link #SERVICE_META_DATA} and <code><{@link android.R.styleable#PrintService |
| * print-service}></code>. |
| * </p> |
| * <p> |
| * <strong>Note: </strong> All callbacks in this class are executed on the main |
| * application thread. You should also invoke any method of this class on the main |
| * application thread. |
| * </p> |
| */ |
| public abstract class PrintService extends Service { |
| |
| private static final String LOG_TAG = "PrintService"; |
| |
| private static final boolean DEBUG = false; |
| |
| /** |
| * The {@link Intent} action that must be declared as handled by a service |
| * in its manifest for the system to recognize it as a print service. |
| */ |
| public static final String SERVICE_INTERFACE = "android.printservice.PrintService"; |
| |
| /** |
| * Name under which a {@link PrintService} component publishes additional information |
| * about itself. This meta-data must reference a XML resource containing a <code> |
| * <{@link android.R.styleable#PrintService print-service}></code> tag. This is |
| * a sample XML file configuring a print service: |
| * <pre> <print-service |
| * android:vendor="SomeVendor" |
| * android:settingsActivity="foo.bar.MySettingsActivity" |
| * android:addPrintersActivity="foo.bar.MyAddPrintersActivity." |
| * . . . |
| * /></pre> |
| * <p> |
| * For detailed configuration options that can be specified via the meta-data |
| * refer to {@link android.R.styleable#PrintService android.R.styleable.PrintService}. |
| * </p> |
| * <p> |
| * If you declare a settings or add a printers activity, they have to be exported, |
| * by setting the {@link android.R.attr#exported} activity attribute to <code>true |
| * </code>. Also in case you want only the system to be able to start any of these |
| * activities you can specify that they request the android.permission |
| * .START_PRINT_SERVICE_CONFIG_ACTIVITY permission by setting the |
| * {@link android.R.attr#permission} activity attribute. |
| * </p> |
| */ |
| public static final String SERVICE_META_DATA = "android.printservice"; |
| |
| /** |
| * If you declared an optional activity with advanced print options via the |
| * {@link android.R.attr#advancedPrintOptionsActivity advancedPrintOptionsActivity} attribute, |
| * this extra is used to pass in the currently constructed {@link PrintJobInfo} to your activity |
| * allowing you to modify it. After you are done, you must return the modified |
| * {@link PrintJobInfo} via the same extra. |
| * <p> |
| * You cannot modify the passed in {@link PrintJobInfo} directly, rather you should build |
| * another one using the {@link android.print.PrintJobInfo.Builder PrintJobInfo.Builder} class. |
| * You can specify any standard properties and add advanced, printer specific, ones via |
| * {@link android.print.PrintJobInfo.Builder#putAdvancedOption(String, String) |
| * PrintJobInfo.Builder.putAdvancedOption(String, String)} and |
| * {@link android.print.PrintJobInfo.Builder#putAdvancedOption(String, int) |
| * PrintJobInfo.Builder.putAdvancedOption(String, int)}. The advanced options are not |
| * interpreted by the system, they will not be visible to applications, and can only be accessed |
| * by your print service via {@link PrintJob#getAdvancedStringOption(String) |
| * PrintJob.getAdvancedStringOption(String)} and {@link PrintJob#getAdvancedIntOption(String) |
| * PrintJob.getAdvancedIntOption(String)}. |
| * </p> |
| * <p> |
| * If the advanced print options activity offers changes to the standard print options, you can |
| * get the current {@link android.print.PrinterInfo PrinterInfo} using the |
| * {@link #EXTRA_PRINTER_INFO} extra which will allow you to present the user with UI options |
| * supported by the current printer. For example, if the current printer does not support a |
| * given media size, you should not offer it in the advanced print options UI. |
| * </p> |
| * |
| * @see #EXTRA_PRINTER_INFO |
| */ |
| public static final String EXTRA_PRINT_JOB_INFO = "android.intent.extra.print.PRINT_JOB_INFO"; |
| |
| /** |
| * If you declared an optional activity with advanced print options via the |
| * {@link android.R.attr#advancedPrintOptionsActivity advancedPrintOptionsActivity} |
| * attribute, this extra is used to pass in the currently selected printer's |
| * {@link android.print.PrinterInfo} to your activity allowing you to inspect it. |
| * |
| * @see #EXTRA_PRINT_JOB_INFO |
| */ |
| public static final String EXTRA_PRINTER_INFO = "android.intent.extra.print.EXTRA_PRINTER_INFO"; |
| |
| /** |
| * If you declared an optional activity with advanced print options via the |
| * {@link android.R.attr#advancedPrintOptionsActivity advancedPrintOptionsActivity} |
| * attribute, this extra is used to pass in the meta-data for the currently printed |
| * document as a {@link android.print.PrintDocumentInfo} to your activity allowing |
| * you to inspect it. |
| * |
| * @see #EXTRA_PRINT_JOB_INFO |
| * @see #EXTRA_PRINTER_INFO |
| */ |
| public static final String EXTRA_PRINT_DOCUMENT_INFO = |
| "android.printservice.extra.PRINT_DOCUMENT_INFO"; |
| |
| /** |
| * When the {@link PendingIntent} declared via |
| * {@link PrinterInfo.Builder#setInfoIntent(PendingIntent)} is called this boolean extra |
| * will be filled in if the activity can select the printer. |
| * |
| * @see #EXTRA_SELECT_PRINTER |
| */ |
| public static final String EXTRA_CAN_SELECT_PRINTER = |
| "android.printservice.extra.CAN_SELECT_PRINTER"; |
| |
| /** |
| * If this boolean extra is set to {@code true} in the {@link Activity#setResult(int, Intent) |
| * result data} from the activity specified in |
| * {@link PrinterInfo.Builder#setInfoIntent(PendingIntent)} the printer will be selected. |
| * |
| * @see #EXTRA_CAN_SELECT_PRINTER |
| */ |
| public static final String EXTRA_SELECT_PRINTER = |
| "android.printservice.extra.SELECT_PRINTER"; |
| |
| private Handler mHandler; |
| |
| private IPrintServiceClient mClient; |
| |
| private int mLastSessionId = -1; |
| |
| private PrinterDiscoverySession mDiscoverySession; |
| |
| @Override |
| protected final void attachBaseContext(Context base) { |
| super.attachBaseContext(base); |
| mHandler = new ServiceHandler(base.getMainLooper()); |
| } |
| |
| /** |
| * The system has connected to this service. |
| */ |
| protected void onConnected() { |
| /* do nothing */ |
| } |
| |
| /** |
| * The system has disconnected from this service. |
| */ |
| protected void onDisconnected() { |
| /* do nothing */ |
| } |
| |
| /** |
| * Callback asking you to create a new {@link PrinterDiscoverySession}. |
| * |
| * @return The created session. |
| * @see PrinterDiscoverySession |
| */ |
| protected abstract @Nullable PrinterDiscoverySession onCreatePrinterDiscoverySession(); |
| |
| /** |
| * Called when cancellation of a print job is requested. The service |
| * should do best effort to fulfill the request. After the cancellation |
| * is performed, the print job should be marked as cancelled state by |
| * calling {@link PrintJob#cancel()}. |
| * |
| * @param printJob The print job to cancel. |
| * |
| * @see PrintJob#cancel() PrintJob.cancel() |
| * @see PrintJob#isCancelled() PrintJob.isCancelled() |
| */ |
| protected abstract void onRequestCancelPrintJob(PrintJob printJob); |
| |
| /** |
| * Called when there is a queued print job for one of the printers |
| * managed by this print service. |
| * |
| * @param printJob The new queued print job. |
| * |
| * @see PrintJob#isQueued() PrintJob.isQueued() |
| * @see #getActivePrintJobs() |
| */ |
| protected abstract void onPrintJobQueued(PrintJob printJob); |
| |
| /** |
| * Gets the active print jobs for the printers managed by this service. |
| * Active print jobs are ones that are not in a final state, i.e. whose |
| * state is queued or started. |
| * |
| * @return The active print jobs. |
| * |
| * @see PrintJob#isQueued() PrintJob.isQueued() |
| * @see PrintJob#isStarted() PrintJob.isStarted() |
| */ |
| public final List<PrintJob> getActivePrintJobs() { |
| throwIfNotCalledOnMainThread(); |
| if (mClient == null) { |
| return Collections.emptyList(); |
| } |
| try { |
| List<PrintJob> printJobs = null; |
| List<PrintJobInfo> printJobInfos = mClient.getPrintJobInfos(); |
| if (printJobInfos != null) { |
| final int printJobInfoCount = printJobInfos.size(); |
| printJobs = new ArrayList<PrintJob>(printJobInfoCount); |
| for (int i = 0; i < printJobInfoCount; i++) { |
| printJobs.add(new PrintJob(this, printJobInfos.get(i), mClient)); |
| } |
| } |
| if (printJobs != null) { |
| return printJobs; |
| } |
| } catch (RemoteException re) { |
| Log.e(LOG_TAG, "Error calling getPrintJobs()", re); |
| } |
| return Collections.emptyList(); |
| } |
| |
| /** |
| * Generates a global printer id given the printer's locally unique one. |
| * |
| * @param localId A locally unique id in the context of your print service. |
| * @return Global printer id. |
| */ |
| public @NonNull final PrinterId generatePrinterId(String localId) { |
| throwIfNotCalledOnMainThread(); |
| localId = Preconditions.checkNotNull(localId, "localId cannot be null"); |
| return new PrinterId(new ComponentName(getPackageName(), |
| getClass().getName()), localId); |
| } |
| |
| static void throwIfNotCalledOnMainThread() { |
| if (!Looper.getMainLooper().isCurrentThread()) { |
| throw new IllegalAccessError("must be called from the main thread"); |
| } |
| } |
| |
| @Override |
| public final IBinder onBind(Intent intent) { |
| return new IPrintService.Stub() { |
| @Override |
| public void createPrinterDiscoverySession() { |
| mHandler.sendEmptyMessage(ServiceHandler.MSG_CREATE_PRINTER_DISCOVERY_SESSION); |
| } |
| |
| @Override |
| public void destroyPrinterDiscoverySession() { |
| mHandler.sendEmptyMessage(ServiceHandler.MSG_DESTROY_PRINTER_DISCOVERY_SESSION); |
| } |
| |
| @Override |
| public void startPrinterDiscovery(List<PrinterId> priorityList) { |
| mHandler.obtainMessage(ServiceHandler.MSG_START_PRINTER_DISCOVERY, |
| priorityList).sendToTarget(); |
| } |
| |
| @Override |
| public void stopPrinterDiscovery() { |
| mHandler.sendEmptyMessage(ServiceHandler.MSG_STOP_PRINTER_DISCOVERY); |
| } |
| |
| @Override |
| public void validatePrinters(List<PrinterId> printerIds) { |
| mHandler.obtainMessage(ServiceHandler.MSG_VALIDATE_PRINTERS, |
| printerIds).sendToTarget(); |
| } |
| |
| @Override |
| public void startPrinterStateTracking(PrinterId printerId) { |
| mHandler.obtainMessage(ServiceHandler.MSG_START_PRINTER_STATE_TRACKING, |
| printerId).sendToTarget(); |
| } |
| |
| @Override |
| public void requestCustomPrinterIcon(PrinterId printerId) { |
| mHandler.obtainMessage(ServiceHandler.MSG_REQUEST_CUSTOM_PRINTER_ICON, |
| printerId).sendToTarget(); |
| } |
| |
| @Override |
| public void stopPrinterStateTracking(PrinterId printerId) { |
| mHandler.obtainMessage(ServiceHandler.MSG_STOP_PRINTER_STATE_TRACKING, |
| printerId).sendToTarget(); |
| } |
| |
| @Override |
| public void setClient(IPrintServiceClient client) { |
| mHandler.obtainMessage(ServiceHandler.MSG_SET_CLIENT, client) |
| .sendToTarget(); |
| } |
| |
| @Override |
| public void requestCancelPrintJob(PrintJobInfo printJobInfo) { |
| mHandler.obtainMessage(ServiceHandler.MSG_ON_REQUEST_CANCEL_PRINTJOB, |
| printJobInfo).sendToTarget(); |
| } |
| |
| @Override |
| public void onPrintJobQueued(PrintJobInfo printJobInfo) { |
| mHandler.obtainMessage(ServiceHandler.MSG_ON_PRINTJOB_QUEUED, |
| printJobInfo).sendToTarget(); |
| } |
| }; |
| } |
| |
| private final class ServiceHandler extends Handler { |
| public static final int MSG_CREATE_PRINTER_DISCOVERY_SESSION = 1; |
| public static final int MSG_DESTROY_PRINTER_DISCOVERY_SESSION = 2; |
| public static final int MSG_START_PRINTER_DISCOVERY = 3; |
| public static final int MSG_STOP_PRINTER_DISCOVERY = 4; |
| public static final int MSG_VALIDATE_PRINTERS = 5; |
| public static final int MSG_START_PRINTER_STATE_TRACKING = 6; |
| public static final int MSG_REQUEST_CUSTOM_PRINTER_ICON = 7; |
| public static final int MSG_STOP_PRINTER_STATE_TRACKING = 8; |
| public static final int MSG_ON_PRINTJOB_QUEUED = 9; |
| public static final int MSG_ON_REQUEST_CANCEL_PRINTJOB = 10; |
| public static final int MSG_SET_CLIENT = 11; |
| |
| public ServiceHandler(Looper looper) { |
| super(looper, null, true); |
| } |
| |
| @Override |
| @SuppressWarnings("unchecked") |
| public void handleMessage(Message message) { |
| final int action = message.what; |
| switch (action) { |
| case MSG_CREATE_PRINTER_DISCOVERY_SESSION: { |
| if (DEBUG) { |
| Log.i(LOG_TAG, "MSG_CREATE_PRINTER_DISCOVERY_SESSION " |
| + getPackageName()); |
| } |
| PrinterDiscoverySession session = onCreatePrinterDiscoverySession(); |
| if (session == null) { |
| throw new NullPointerException("session cannot be null"); |
| } |
| if (session.getId() == mLastSessionId) { |
| throw new IllegalStateException("cannot reuse session instances"); |
| } |
| mDiscoverySession = session; |
| mLastSessionId = session.getId(); |
| session.setObserver(mClient); |
| } break; |
| |
| case MSG_DESTROY_PRINTER_DISCOVERY_SESSION: { |
| if (DEBUG) { |
| Log.i(LOG_TAG, "MSG_DESTROY_PRINTER_DISCOVERY_SESSION " |
| + getPackageName()); |
| } |
| if (mDiscoverySession != null) { |
| mDiscoverySession.destroy(); |
| mDiscoverySession = null; |
| } |
| } break; |
| |
| case MSG_START_PRINTER_DISCOVERY: { |
| if (DEBUG) { |
| Log.i(LOG_TAG, "MSG_START_PRINTER_DISCOVERY " |
| + getPackageName()); |
| } |
| if (mDiscoverySession != null) { |
| List<PrinterId> priorityList = (ArrayList<PrinterId>) message.obj; |
| mDiscoverySession.startPrinterDiscovery(priorityList); |
| } |
| } break; |
| |
| case MSG_STOP_PRINTER_DISCOVERY: { |
| if (DEBUG) { |
| Log.i(LOG_TAG, "MSG_STOP_PRINTER_DISCOVERY " |
| + getPackageName()); |
| } |
| if (mDiscoverySession != null) { |
| mDiscoverySession.stopPrinterDiscovery(); |
| } |
| } break; |
| |
| case MSG_VALIDATE_PRINTERS: { |
| if (DEBUG) { |
| Log.i(LOG_TAG, "MSG_VALIDATE_PRINTERS " |
| + getPackageName()); |
| } |
| if (mDiscoverySession != null) { |
| List<PrinterId> printerIds = (List<PrinterId>) message.obj; |
| mDiscoverySession.validatePrinters(printerIds); |
| } |
| } break; |
| |
| case MSG_START_PRINTER_STATE_TRACKING: { |
| if (DEBUG) { |
| Log.i(LOG_TAG, "MSG_START_PRINTER_STATE_TRACKING " |
| + getPackageName()); |
| } |
| if (mDiscoverySession != null) { |
| PrinterId printerId = (PrinterId) message.obj; |
| mDiscoverySession.startPrinterStateTracking(printerId); |
| } |
| } break; |
| |
| case MSG_REQUEST_CUSTOM_PRINTER_ICON: { |
| if (DEBUG) { |
| Log.i(LOG_TAG, "MSG_REQUEST_CUSTOM_PRINTER_ICON " |
| + getPackageName()); |
| } |
| if (mDiscoverySession != null) { |
| PrinterId printerId = (PrinterId) message.obj; |
| mDiscoverySession.requestCustomPrinterIcon(printerId); |
| } |
| } break; |
| |
| case MSG_STOP_PRINTER_STATE_TRACKING: { |
| if (DEBUG) { |
| Log.i(LOG_TAG, "MSG_STOP_PRINTER_STATE_TRACKING " |
| + getPackageName()); |
| } |
| if (mDiscoverySession != null) { |
| PrinterId printerId = (PrinterId) message.obj; |
| mDiscoverySession.stopPrinterStateTracking(printerId); |
| } |
| } break; |
| |
| case MSG_ON_REQUEST_CANCEL_PRINTJOB: { |
| if (DEBUG) { |
| Log.i(LOG_TAG, "MSG_ON_REQUEST_CANCEL_PRINTJOB " |
| + getPackageName()); |
| } |
| PrintJobInfo printJobInfo = (PrintJobInfo) message.obj; |
| onRequestCancelPrintJob(new PrintJob(PrintService.this, printJobInfo, mClient)); |
| } break; |
| |
| case MSG_ON_PRINTJOB_QUEUED: { |
| if (DEBUG) { |
| Log.i(LOG_TAG, "MSG_ON_PRINTJOB_QUEUED " |
| + getPackageName()); |
| } |
| PrintJobInfo printJobInfo = (PrintJobInfo) message.obj; |
| if (DEBUG) { |
| Log.i(LOG_TAG, "Queued: " + printJobInfo); |
| } |
| onPrintJobQueued(new PrintJob(PrintService.this, printJobInfo, mClient)); |
| } break; |
| |
| case MSG_SET_CLIENT: { |
| if (DEBUG) { |
| Log.i(LOG_TAG, "MSG_SET_CLIENT " |
| + getPackageName()); |
| } |
| mClient = (IPrintServiceClient) message.obj; |
| if (mClient != null) { |
| onConnected(); |
| } else { |
| onDisconnected(); |
| } |
| } break; |
| |
| default: { |
| throw new IllegalArgumentException("Unknown message: " + action); |
| } |
| } |
| } |
| } |
| } |