Home | History | Annotate | Download | only in opp
      1 /*
      2  * Copyright (c) 2008-2009, Motorola, Inc.
      3  *
      4  * All rights reserved.
      5  *
      6  * Redistribution and use in source and binary forms, with or without
      7  * modification, are permitted provided that the following conditions are met:
      8  *
      9  * - Redistributions of source code must retain the above copyright notice,
     10  * this list of conditions and the following disclaimer.
     11  *
     12  * - Redistributions in binary form must reproduce the above copyright notice,
     13  * this list of conditions and the following disclaimer in the documentation
     14  * and/or other materials provided with the distribution.
     15  *
     16  * - Neither the name of the Motorola, Inc. nor the names of its contributors
     17  * may be used to endorse or promote products derived from this software
     18  * without specific prior written permission.
     19  *
     20  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
     21  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
     22  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
     23  * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
     24  * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
     25  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
     26  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
     27  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
     28  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
     29  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
     30  * POSSIBILITY OF SUCH DAMAGE.
     31  */
     32 
     33 package com.android.bluetooth.opp;
     34 
     35 import com.android.bluetooth.R;
     36 import com.google.android.collect.Lists;
     37 
     38 
     39 import android.bluetooth.BluetoothAdapter;
     40 import android.bluetooth.BluetoothDevice;
     41 import android.net.Uri;
     42 import android.content.ContentValues;
     43 import android.content.Context;
     44 import android.content.ActivityNotFoundException;
     45 import android.content.Intent;
     46 import android.content.pm.PackageManager;
     47 import android.content.pm.ResolveInfo;
     48 import android.database.Cursor;
     49 import android.util.Log;
     50 
     51 import java.io.File;
     52 import java.util.ArrayList;
     53 import java.util.List;
     54 
     55 /**
     56  * This class has some utilities for Opp application;
     57  */
     58 public class BluetoothOppUtility {
     59     private static final String TAG = "BluetoothOppUtility";
     60     private static final boolean D = Constants.DEBUG;
     61     private static final boolean V = Constants.VERBOSE;
     62 
     63     public static BluetoothOppTransferInfo queryRecord(Context context, Uri uri) {
     64         BluetoothAdapter adapter = BluetoothAdapter.getDefaultAdapter();
     65         BluetoothOppTransferInfo info = new BluetoothOppTransferInfo();
     66         Cursor cursor = context.getContentResolver().query(uri, null, null, null, null);
     67         if (cursor != null) {
     68             if (cursor.moveToFirst()) {
     69                 info.mID = cursor.getInt(cursor.getColumnIndexOrThrow(BluetoothShare._ID));
     70                 info.mStatus = cursor.getInt(cursor.getColumnIndexOrThrow(BluetoothShare.STATUS));
     71                 info.mDirection = cursor.getInt(cursor
     72                         .getColumnIndexOrThrow(BluetoothShare.DIRECTION));
     73                 info.mTotalBytes = cursor.getInt(cursor
     74                         .getColumnIndexOrThrow(BluetoothShare.TOTAL_BYTES));
     75                 info.mCurrentBytes = cursor.getInt(cursor
     76                         .getColumnIndexOrThrow(BluetoothShare.CURRENT_BYTES));
     77                 info.mTimeStamp = cursor.getLong(cursor
     78                         .getColumnIndexOrThrow(BluetoothShare.TIMESTAMP));
     79                 info.mDestAddr = cursor.getString(cursor
     80                         .getColumnIndexOrThrow(BluetoothShare.DESTINATION));
     81 
     82                 info.mFileName = cursor.getString(cursor
     83                         .getColumnIndexOrThrow(BluetoothShare._DATA));
     84                 if (info.mFileName == null) {
     85                     info.mFileName = cursor.getString(cursor
     86                             .getColumnIndexOrThrow(BluetoothShare.FILENAME_HINT));
     87                 }
     88                 if (info.mFileName == null) {
     89                     info.mFileName = context.getString(R.string.unknown_file);
     90                 }
     91 
     92                 info.mFileUri = cursor.getString(cursor.getColumnIndexOrThrow(BluetoothShare.URI));
     93 
     94                 if (info.mFileUri != null) {
     95                     Uri u = Uri.parse(info.mFileUri);
     96                     info.mFileType = context.getContentResolver().getType(u);
     97                 } else {
     98                     Uri u = Uri.parse(info.mFileName);
     99                     info.mFileType = context.getContentResolver().getType(u);
    100                 }
    101                 if (info.mFileType == null) {
    102                     info.mFileType = cursor.getString(cursor
    103                             .getColumnIndexOrThrow(BluetoothShare.MIMETYPE));
    104                 }
    105 
    106                 BluetoothDevice remoteDevice = adapter.getRemoteDevice(info.mDestAddr);
    107                 info.mDeviceName =
    108                         BluetoothOppManager.getInstance(context).getDeviceName(remoteDevice);
    109 
    110                 if (V) Log.v(TAG, "Get data from db:" + info.mFileName + info.mFileType
    111                             + info.mDestAddr);
    112             }
    113             cursor.close();
    114         } else {
    115             info = null;
    116             if (V) Log.v(TAG, "BluetoothOppManager Error: not got data from db for uri:" + uri);
    117         }
    118         return info;
    119     }
    120 
    121     /**
    122      * Organize Array list for transfers in one batch
    123      */
    124     // This function is used when UI show batch transfer. Currently only show single transfer.
    125     public static ArrayList<String> queryTransfersInBatch(Context context, Long timeStamp) {
    126         ArrayList<String> uris = Lists.newArrayList();
    127         final String WHERE = BluetoothShare.TIMESTAMP + " == " + timeStamp;
    128 
    129         Cursor metadataCursor = context.getContentResolver().query(BluetoothShare.CONTENT_URI,
    130                 new String[] {
    131                     BluetoothShare._DATA
    132                 }, WHERE, null, BluetoothShare._ID);
    133 
    134         if (metadataCursor == null) {
    135             return null;
    136         }
    137 
    138         for (metadataCursor.moveToFirst(); !metadataCursor.isAfterLast(); metadataCursor
    139                 .moveToNext()) {
    140             String fileName = metadataCursor.getString(0);
    141             Uri path = Uri.parse(fileName);
    142             // If there is no scheme, then it must be a file
    143             if (path.getScheme() == null) {
    144                 path = Uri.fromFile(new File(fileName));
    145             }
    146             uris.add(path.toString());
    147             if (V) Log.d(TAG, "Uri in this batch: " + path.toString());
    148         }
    149         metadataCursor.close();
    150         return uris;
    151     }
    152 
    153     /**
    154      * Open the received file with appropriate application, if can not find
    155      * application to handle, display error dialog.
    156      */
    157     public static void openReceivedFile(Context context, String fileName, String mimetype,
    158             Long timeStamp, Uri uri) {
    159         if (fileName == null || mimetype == null) {
    160             Log.e(TAG, "ERROR: Para fileName ==null, or mimetype == null");
    161             return;
    162         }
    163 
    164         File f = new File(fileName);
    165         if (!f.exists()) {
    166             Intent in = new Intent(context, BluetoothOppBtErrorActivity.class);
    167             in.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
    168             in.putExtra("title", context.getString(R.string.not_exist_file));
    169             in.putExtra("content", context.getString(R.string.not_exist_file_desc));
    170             context.startActivity(in);
    171 
    172             // Due to the file is not existing, delete related info in btopp db
    173             // to prevent this file from appearing in live folder
    174             if (V) Log.d(TAG, "This uri will be deleted: " + uri);
    175             context.getContentResolver().delete(uri, null, null);
    176             return;
    177         }
    178 
    179         Uri path = Uri.parse(fileName);
    180         // If there is no scheme, then it must be a file
    181         if (path.getScheme() == null) {
    182             path = Uri.fromFile(new File(fileName));
    183         }
    184 
    185         if (isRecognizedFileType(context, path, mimetype)) {
    186             Intent activityIntent = new Intent(Intent.ACTION_VIEW);
    187             activityIntent.setDataAndType(path, mimetype);
    188 
    189             activityIntent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
    190             try {
    191                 if (V) Log.d(TAG, "ACTION_VIEW intent sent out: " + path + " / " + mimetype);
    192                 context.startActivity(activityIntent);
    193             } catch (ActivityNotFoundException ex) {
    194                 if (V) Log.d(TAG, "no activity for handling ACTION_VIEW intent:  " + mimetype, ex);
    195             }
    196         } else {
    197             Intent in = new Intent(context, BluetoothOppBtErrorActivity.class);
    198             in.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
    199             in.putExtra("title", context.getString(R.string.unknown_file));
    200             in.putExtra("content", context.getString(R.string.unknown_file_desc));
    201             context.startActivity(in);
    202         }
    203     }
    204 
    205     /**
    206      * To judge if the file type supported (can be handled by some app) by phone
    207      * system.
    208      */
    209     public static boolean isRecognizedFileType(Context context, Uri fileUri, String mimetype) {
    210         boolean ret = true;
    211 
    212         if (D) Log.d(TAG, "RecognizedFileType() fileUri: " + fileUri + " mimetype: " + mimetype);
    213 
    214         Intent mimetypeIntent = new Intent(Intent.ACTION_VIEW);
    215         mimetypeIntent.setDataAndType(fileUri, mimetype);
    216         List<ResolveInfo> list = context.getPackageManager().queryIntentActivities(mimetypeIntent,
    217                 PackageManager.MATCH_DEFAULT_ONLY);
    218 
    219         if (list.size() == 0) {
    220             if (D) Log.d(TAG, "NO application to handle MIME type " + mimetype);
    221             ret = false;
    222         }
    223         return ret;
    224     }
    225 
    226     /**
    227      * update visibility to Hidden
    228      */
    229     public static void updateVisibilityToHidden(Context context, Uri uri) {
    230         ContentValues updateValues = new ContentValues();
    231         updateValues.put(BluetoothShare.VISIBILITY, BluetoothShare.VISIBILITY_HIDDEN);
    232         context.getContentResolver().update(uri, updateValues, null, null);
    233     }
    234 
    235     /**
    236      * Helper function to build the progress text.
    237      */
    238     public static String formatProgressText(long totalBytes, long currentBytes) {
    239         if (totalBytes <= 0) {
    240             return "0%";
    241         }
    242         long progress = currentBytes * 100 / totalBytes;
    243         StringBuilder sb = new StringBuilder();
    244         sb.append(progress);
    245         sb.append('%');
    246         return sb.toString();
    247     }
    248 
    249     /**
    250      * Get status description according to status code.
    251      */
    252     public static String getStatusDescription(Context context, int statusCode, String deviceName) {
    253         String ret;
    254         if (statusCode == BluetoothShare.STATUS_PENDING) {
    255             ret = context.getString(R.string.status_pending);
    256         } else if (statusCode == BluetoothShare.STATUS_RUNNING) {
    257             ret = context.getString(R.string.status_running);
    258         } else if (statusCode == BluetoothShare.STATUS_SUCCESS) {
    259             ret = context.getString(R.string.status_success);
    260         } else if (statusCode == BluetoothShare.STATUS_NOT_ACCEPTABLE) {
    261             ret = context.getString(R.string.status_not_accept);
    262         } else if (statusCode == BluetoothShare.STATUS_FORBIDDEN) {
    263             ret = context.getString(R.string.status_forbidden);
    264         } else if (statusCode == BluetoothShare.STATUS_CANCELED) {
    265             ret = context.getString(R.string.status_canceled);
    266         } else if (statusCode == BluetoothShare.STATUS_FILE_ERROR) {
    267             ret = context.getString(R.string.status_file_error);
    268         } else if (statusCode == BluetoothShare.STATUS_ERROR_NO_SDCARD) {
    269             ret = context.getString(R.string.status_no_sd_card);
    270         } else if (statusCode == BluetoothShare.STATUS_CONNECTION_ERROR) {
    271             ret = context.getString(R.string.status_connection_error);
    272         } else if (statusCode == BluetoothShare.STATUS_ERROR_SDCARD_FULL) {
    273             ret = context.getString(R.string.bt_sm_2_1, deviceName);
    274         } else if ((statusCode == BluetoothShare.STATUS_BAD_REQUEST)
    275                 || (statusCode == BluetoothShare.STATUS_LENGTH_REQUIRED)
    276                 || (statusCode == BluetoothShare.STATUS_PRECONDITION_FAILED)
    277                 || (statusCode == BluetoothShare.STATUS_UNHANDLED_OBEX_CODE)
    278                 || (statusCode == BluetoothShare.STATUS_OBEX_DATA_ERROR)) {
    279             ret = context.getString(R.string.status_protocol_error);
    280         } else {
    281             ret = context.getString(R.string.status_unknown_error);
    282         }
    283         return ret;
    284     }
    285 
    286     /**
    287      * Retry the failed transfer: Will insert a new transfer session to db
    288      */
    289     public static void retryTransfer(Context context, BluetoothOppTransferInfo transInfo) {
    290         ContentValues values = new ContentValues();
    291         values.put(BluetoothShare.URI, transInfo.mFileUri);
    292         values.put(BluetoothShare.MIMETYPE, transInfo.mFileType);
    293         values.put(BluetoothShare.DESTINATION, transInfo.mDestAddr);
    294 
    295         final Uri contentUri = context.getContentResolver().insert(BluetoothShare.CONTENT_URI,
    296                 values);
    297         if (V) Log.v(TAG, "Insert contentUri: " + contentUri + "  to device: " +
    298                 transInfo.mDeviceName);
    299     }
    300 
    301 }
    302