/*
|
* 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"
|
* andorid: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);
|
}
|
}
|
}
|
}
|
}
|