1 /* 2 * Copyright (C) 2010 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 com.android.gallery3d.ingest.data; 18 19 import android.annotation.TargetApi; 20 import android.app.PendingIntent; 21 import android.content.BroadcastReceiver; 22 import android.content.Context; 23 import android.content.Intent; 24 import android.content.IntentFilter; 25 import android.hardware.usb.UsbConstants; 26 import android.hardware.usb.UsbDevice; 27 import android.hardware.usb.UsbDeviceConnection; 28 import android.hardware.usb.UsbInterface; 29 import android.hardware.usb.UsbManager; 30 import android.mtp.MtpDevice; 31 import android.os.Build; 32 import android.util.Log; 33 34 import java.util.ArrayList; 35 import java.util.HashMap; 36 import java.util.List; 37 38 /** 39 * This class helps an application manage a list of connected MTP or PTP devices. 40 * It listens for MTP devices being attached and removed from the USB host bus 41 * and notifies the application when the MTP device list changes. 42 */ 43 @TargetApi(Build.VERSION_CODES.HONEYCOMB_MR1) 44 public class MtpClient { 45 46 private static final String TAG = "MtpClient"; 47 48 private static final String ACTION_USB_PERMISSION = 49 "com.android.gallery3d.ingest.action.USB_PERMISSION"; 50 51 private final Context mContext; 52 private final UsbManager mUsbManager; 53 private final ArrayList<Listener> mListeners = new ArrayList<Listener>(); 54 // mDevices contains all MtpDevices that have been seen by our client, 55 // so we can inform when the device has been detached. 56 // mDevices is also used for synchronization in this class. 57 private final HashMap<String, MtpDevice> mDevices = new HashMap<String, MtpDevice>(); 58 // List of MTP devices we should not try to open for which we are currently 59 // asking for permission to open. 60 private final ArrayList<String> mRequestPermissionDevices = new ArrayList<String>(); 61 // List of MTP devices we should not try to open. 62 // We add devices to this list if the user canceled a permission request or we were 63 // unable to open the device. 64 private final ArrayList<String> mIgnoredDevices = new ArrayList<String>(); 65 66 private final PendingIntent mPermissionIntent; 67 68 private final BroadcastReceiver mUsbReceiver = new BroadcastReceiver() { 69 @Override 70 public void onReceive(Context context, Intent intent) { 71 String action = intent.getAction(); 72 UsbDevice usbDevice = (UsbDevice) intent.getParcelableExtra(UsbManager.EXTRA_DEVICE); 73 String deviceName = usbDevice.getDeviceName(); 74 75 synchronized (mDevices) { 76 MtpDevice mtpDevice = mDevices.get(deviceName); 77 78 if (UsbManager.ACTION_USB_DEVICE_ATTACHED.equals(action)) { 79 if (mtpDevice == null) { 80 mtpDevice = openDeviceLocked(usbDevice); 81 } 82 if (mtpDevice != null) { 83 for (Listener listener : mListeners) { 84 listener.deviceAdded(mtpDevice); 85 } 86 } 87 } else if (UsbManager.ACTION_USB_DEVICE_DETACHED.equals(action)) { 88 if (mtpDevice != null) { 89 mDevices.remove(deviceName); 90 mRequestPermissionDevices.remove(deviceName); 91 mIgnoredDevices.remove(deviceName); 92 for (Listener listener : mListeners) { 93 listener.deviceRemoved(mtpDevice); 94 } 95 } 96 } else if (ACTION_USB_PERMISSION.equals(action)) { 97 mRequestPermissionDevices.remove(deviceName); 98 boolean permission = intent.getBooleanExtra(UsbManager.EXTRA_PERMISSION_GRANTED, 99 false); 100 Log.d(TAG, "ACTION_USB_PERMISSION: " + permission); 101 if (permission) { 102 if (mtpDevice == null) { 103 mtpDevice = openDeviceLocked(usbDevice); 104 } 105 if (mtpDevice != null) { 106 for (Listener listener : mListeners) { 107 listener.deviceAdded(mtpDevice); 108 } 109 } 110 } else { 111 // so we don't ask for permission again 112 mIgnoredDevices.add(deviceName); 113 } 114 } 115 } 116 } 117 }; 118 119 /** 120 * An interface for being notified when MTP or PTP devices are attached 121 * or removed. In the current implementation, only PTP devices are supported. 122 */ 123 public interface Listener { 124 /** 125 * Called when a new device has been added 126 * 127 * @param device the new device that was added 128 */ deviceAdded(MtpDevice device)129 public void deviceAdded(MtpDevice device); 130 131 /** 132 * Called when a new device has been removed 133 * 134 * @param device the device that was removed 135 */ deviceRemoved(MtpDevice device)136 public void deviceRemoved(MtpDevice device); 137 } 138 139 /** 140 * Tests to see if a {@link android.hardware.usb.UsbDevice} 141 * supports the PTP protocol (typically used by digital cameras) 142 * 143 * @param device the device to test 144 * @return true if the device is a PTP device. 145 */ isCamera(UsbDevice device)146 public static boolean isCamera(UsbDevice device) { 147 int count = device.getInterfaceCount(); 148 for (int i = 0; i < count; i++) { 149 UsbInterface intf = device.getInterface(i); 150 if (intf.getInterfaceClass() == UsbConstants.USB_CLASS_STILL_IMAGE && 151 intf.getInterfaceSubclass() == 1 && 152 intf.getInterfaceProtocol() == 1) { 153 return true; 154 } 155 } 156 return false; 157 } 158 159 /** 160 * MtpClient constructor 161 * 162 * @param context the {@link android.content.Context} to use for the MtpClient 163 */ MtpClient(Context context)164 public MtpClient(Context context) { 165 mContext = context; 166 mUsbManager = (UsbManager) context.getSystemService(Context.USB_SERVICE); 167 mPermissionIntent = PendingIntent.getBroadcast(mContext, 0, 168 new Intent(ACTION_USB_PERMISSION), 0); 169 IntentFilter filter = new IntentFilter(ACTION_USB_PERMISSION); 170 filter.addAction(UsbManager.ACTION_USB_DEVICE_ATTACHED); 171 filter.addAction(UsbManager.ACTION_USB_DEVICE_DETACHED); 172 filter.addAction(ACTION_USB_PERMISSION); 173 context.registerReceiver(mUsbReceiver, filter, Context.RECEIVER_EXPORTED/*UNAUDITED*/); 174 } 175 176 /** 177 * Opens the {@link android.hardware.usb.UsbDevice} for an MTP or PTP 178 * device and return an {@link android.mtp.MtpDevice} for it. 179 * 180 * @param usbDevice the device to open 181 * @return an MtpDevice for the device. 182 */ openDeviceLocked(UsbDevice usbDevice)183 private MtpDevice openDeviceLocked(UsbDevice usbDevice) { 184 String deviceName = usbDevice.getDeviceName(); 185 186 // don't try to open devices that we have decided to ignore 187 // or are currently asking permission for 188 if (isCamera(usbDevice) && !mIgnoredDevices.contains(deviceName) 189 && !mRequestPermissionDevices.contains(deviceName)) { 190 if (!mUsbManager.hasPermission(usbDevice)) { 191 mUsbManager.requestPermission(usbDevice, mPermissionIntent); 192 mRequestPermissionDevices.add(deviceName); 193 } else { 194 UsbDeviceConnection connection = mUsbManager.openDevice(usbDevice); 195 if (connection != null) { 196 MtpDevice mtpDevice = new MtpDevice(usbDevice); 197 if (mtpDevice.open(connection)) { 198 mDevices.put(usbDevice.getDeviceName(), mtpDevice); 199 return mtpDevice; 200 } else { 201 // so we don't try to open it again 202 mIgnoredDevices.add(deviceName); 203 } 204 } else { 205 // so we don't try to open it again 206 mIgnoredDevices.add(deviceName); 207 } 208 } 209 } 210 return null; 211 } 212 213 /** 214 * Closes all resources related to the MtpClient object 215 */ close()216 public void close() { 217 mContext.unregisterReceiver(mUsbReceiver); 218 } 219 220 /** 221 * Registers a {@link com.android.gallery3d.data.MtpClient.Listener} interface to receive 222 * notifications when MTP or PTP devices are added or removed. 223 * 224 * @param listener the listener to register 225 */ addListener(Listener listener)226 public void addListener(Listener listener) { 227 synchronized (mDevices) { 228 if (!mListeners.contains(listener)) { 229 mListeners.add(listener); 230 } 231 } 232 } 233 234 /** 235 * Unregisters a {@link com.android.gallery3d.data.MtpClient.Listener} interface. 236 * 237 * @param listener the listener to unregister 238 */ removeListener(Listener listener)239 public void removeListener(Listener listener) { 240 synchronized (mDevices) { 241 mListeners.remove(listener); 242 } 243 } 244 245 246 /** 247 * Retrieves a list of all currently connected {@link android.mtp.MtpDevice}. 248 * 249 * @return the list of MtpDevices 250 */ getDeviceList()251 public List<MtpDevice> getDeviceList() { 252 synchronized (mDevices) { 253 // Query the USB manager since devices might have attached 254 // before we added our listener. 255 for (UsbDevice usbDevice : mUsbManager.getDeviceList().values()) { 256 if (mDevices.get(usbDevice.getDeviceName()) == null) { 257 openDeviceLocked(usbDevice); 258 } 259 } 260 261 return new ArrayList<MtpDevice>(mDevices.values()); 262 } 263 } 264 265 266 } 267