| /* |
| * Copyright (C) 2006 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 com.android.internal.telephony; |
| |
| import android.content.ContentProvider; |
| import android.content.UriMatcher; |
| import android.content.ContentValues; |
| import android.database.AbstractCursor; |
| import android.database.Cursor; |
| import android.database.CursorWindow; |
| import android.net.Uri; |
| import android.os.SystemProperties; |
| import android.os.RemoteException; |
| import android.os.ServiceManager; |
| import android.text.TextUtils; |
| import android.util.Log; |
| |
| import java.util.ArrayList; |
| import java.util.List; |
| |
| import com.android.internal.telephony.IccConstants; |
| import com.android.internal.telephony.AdnRecord; |
| import com.android.internal.telephony.IIccPhoneBook; |
| |
| /** |
| * XXX old code -- should be replaced with MatrixCursor. |
| * @deprecated This is has been replaced by MatrixCursor. |
| */ |
| class ArrayListCursor extends AbstractCursor { |
| private String[] mColumnNames; |
| private ArrayList<Object>[] mRows; |
| |
| @SuppressWarnings({"unchecked"}) |
| public ArrayListCursor(String[] columnNames, ArrayList<ArrayList> rows) { |
| int colCount = columnNames.length; |
| boolean foundID = false; |
| // Add an _id column if not in columnNames |
| for (int i = 0; i < colCount; ++i) { |
| if (columnNames[i].compareToIgnoreCase("_id") == 0) { |
| mColumnNames = columnNames; |
| foundID = true; |
| break; |
| } |
| } |
| |
| if (!foundID) { |
| mColumnNames = new String[colCount + 1]; |
| System.arraycopy(columnNames, 0, mColumnNames, 0, columnNames.length); |
| mColumnNames[colCount] = "_id"; |
| } |
| |
| int rowCount = rows.size(); |
| mRows = new ArrayList[rowCount]; |
| |
| for (int i = 0; i < rowCount; ++i) { |
| mRows[i] = rows.get(i); |
| if (!foundID) { |
| mRows[i].add(i); |
| } |
| } |
| } |
| |
| @Override |
| public void fillWindow(int position, CursorWindow window) { |
| if (position < 0 || position > getCount()) { |
| return; |
| } |
| |
| window.acquireReference(); |
| try { |
| int oldpos = mPos; |
| mPos = position - 1; |
| window.clear(); |
| window.setStartPosition(position); |
| int columnNum = getColumnCount(); |
| window.setNumColumns(columnNum); |
| while (moveToNext() && window.allocRow()) { |
| for (int i = 0; i < columnNum; i++) { |
| final Object data = mRows[mPos].get(i); |
| if (data != null) { |
| if (data instanceof byte[]) { |
| byte[] field = (byte[]) data; |
| if (!window.putBlob(field, mPos, i)) { |
| window.freeLastRow(); |
| break; |
| } |
| } else { |
| String field = data.toString(); |
| if (!window.putString(field, mPos, i)) { |
| window.freeLastRow(); |
| break; |
| } |
| } |
| } else { |
| if (!window.putNull(mPos, i)) { |
| window.freeLastRow(); |
| break; |
| } |
| } |
| } |
| } |
| |
| mPos = oldpos; |
| } catch (IllegalStateException e){ |
| // simply ignore it |
| } finally { |
| window.releaseReference(); |
| } |
| } |
| |
| @Override |
| public int getCount() { |
| return mRows.length; |
| } |
| |
| @Override |
| public String[] getColumnNames() { |
| return mColumnNames; |
| } |
| |
| @Override |
| public byte[] getBlob(int columnIndex) { |
| return (byte[]) mRows[mPos].get(columnIndex); |
| } |
| |
| @Override |
| public String getString(int columnIndex) { |
| Object cell = mRows[mPos].get(columnIndex); |
| return (cell == null) ? null : cell.toString(); |
| } |
| |
| @Override |
| public short getShort(int columnIndex) { |
| Number num = (Number) mRows[mPos].get(columnIndex); |
| return num.shortValue(); |
| } |
| |
| @Override |
| public int getInt(int columnIndex) { |
| Number num = (Number) mRows[mPos].get(columnIndex); |
| return num.intValue(); |
| } |
| |
| @Override |
| public long getLong(int columnIndex) { |
| Number num = (Number) mRows[mPos].get(columnIndex); |
| return num.longValue(); |
| } |
| |
| @Override |
| public float getFloat(int columnIndex) { |
| Number num = (Number) mRows[mPos].get(columnIndex); |
| return num.floatValue(); |
| } |
| |
| @Override |
| public double getDouble(int columnIndex) { |
| Number num = (Number) mRows[mPos].get(columnIndex); |
| return num.doubleValue(); |
| } |
| |
| @Override |
| public boolean isNull(int columnIndex) { |
| return mRows[mPos].get(columnIndex) == null; |
| } |
| } |
| |
| |
| /** |
| * {@hide} |
| */ |
| public class IccProvider extends ContentProvider { |
| private static final String TAG = "IccProvider"; |
| private static final boolean DBG = false; |
| |
| |
| private static final String[] ADDRESS_BOOK_COLUMN_NAMES = new String[] { |
| "name", |
| "number", |
| "emails" |
| }; |
| |
| private static final int ADN = 1; |
| private static final int FDN = 2; |
| private static final int SDN = 3; |
| |
| private static final String STR_TAG = "tag"; |
| private static final String STR_NUMBER = "number"; |
| private static final String STR_EMAILS = "emails"; |
| private static final String STR_PIN2 = "pin2"; |
| |
| private static final UriMatcher URL_MATCHER = |
| new UriMatcher(UriMatcher.NO_MATCH); |
| |
| static { |
| URL_MATCHER.addURI("icc", "adn", ADN); |
| URL_MATCHER.addURI("icc", "fdn", FDN); |
| URL_MATCHER.addURI("icc", "sdn", SDN); |
| } |
| |
| |
| private boolean mSimulator; |
| |
| @Override |
| public boolean onCreate() { |
| String device = SystemProperties.get("ro.product.device"); |
| if (!TextUtils.isEmpty(device)) { |
| mSimulator = false; |
| } else { |
| // simulator |
| mSimulator = true; |
| } |
| |
| return true; |
| } |
| |
| @Override |
| public Cursor query(Uri url, String[] projection, String selection, |
| String[] selectionArgs, String sort) { |
| ArrayList<ArrayList> results; |
| |
| if (!mSimulator) { |
| switch (URL_MATCHER.match(url)) { |
| case ADN: |
| results = loadFromEf(IccConstants.EF_ADN); |
| break; |
| |
| case FDN: |
| results = loadFromEf(IccConstants.EF_FDN); |
| break; |
| |
| case SDN: |
| results = loadFromEf(IccConstants.EF_SDN); |
| break; |
| |
| default: |
| throw new IllegalArgumentException("Unknown URL " + url); |
| } |
| } else { |
| // Fake up some data for the simulator |
| results = new ArrayList<ArrayList>(4); |
| ArrayList<String> contact; |
| |
| contact = new ArrayList<String>(); |
| contact.add("Ron Stevens/H"); |
| contact.add("512-555-5038"); |
| results.add(contact); |
| |
| contact = new ArrayList<String>(); |
| contact.add("Ron Stevens/M"); |
| contact.add("512-555-8305"); |
| results.add(contact); |
| |
| contact = new ArrayList<String>(); |
| contact.add("Melissa Owens"); |
| contact.add("512-555-8305"); |
| results.add(contact); |
| |
| contact = new ArrayList<String>(); |
| contact.add("Directory Assistence"); |
| contact.add("411"); |
| results.add(contact); |
| } |
| |
| return new ArrayListCursor(ADDRESS_BOOK_COLUMN_NAMES, results); |
| } |
| |
| @Override |
| public String getType(Uri url) { |
| switch (URL_MATCHER.match(url)) { |
| case ADN: |
| case FDN: |
| case SDN: |
| return "vnd.android.cursor.dir/sim-contact"; |
| |
| default: |
| throw new IllegalArgumentException("Unknown URL " + url); |
| } |
| } |
| |
| @Override |
| public Uri insert(Uri url, ContentValues initialValues) { |
| Uri resultUri; |
| int efType; |
| String pin2 = null; |
| |
| if (DBG) log("insert"); |
| |
| int match = URL_MATCHER.match(url); |
| switch (match) { |
| case ADN: |
| efType = IccConstants.EF_ADN; |
| break; |
| |
| case FDN: |
| efType = IccConstants.EF_FDN; |
| pin2 = initialValues.getAsString("pin2"); |
| break; |
| |
| default: |
| throw new UnsupportedOperationException( |
| "Cannot insert into URL: " + url); |
| } |
| |
| String tag = initialValues.getAsString("tag"); |
| String number = initialValues.getAsString("number"); |
| // TODO(): Read email instead of sending null. |
| boolean success = addIccRecordToEf(efType, tag, number, null, pin2); |
| |
| if (!success) { |
| return null; |
| } |
| |
| StringBuilder buf = new StringBuilder("content://icc/"); |
| switch (match) { |
| case ADN: |
| buf.append("adn/"); |
| break; |
| |
| case FDN: |
| buf.append("fdn/"); |
| break; |
| } |
| |
| // TODO: we need to find out the rowId for the newly added record |
| buf.append(0); |
| |
| resultUri = Uri.parse(buf.toString()); |
| |
| /* |
| // notify interested parties that an insertion happened |
| getContext().getContentResolver().notifyInsert( |
| resultUri, rowID, null); |
| */ |
| |
| return resultUri; |
| } |
| |
| private String normalizeValue(String inVal) { |
| int len = inVal.length(); |
| String retVal = inVal; |
| |
| if (inVal.charAt(0) == '\'' && inVal.charAt(len-1) == '\'') { |
| retVal = inVal.substring(1, len-1); |
| } |
| |
| return retVal; |
| } |
| |
| @Override |
| public int delete(Uri url, String where, String[] whereArgs) { |
| int efType; |
| |
| if (DBG) log("delete"); |
| |
| int match = URL_MATCHER.match(url); |
| switch (match) { |
| case ADN: |
| efType = IccConstants.EF_ADN; |
| break; |
| |
| case FDN: |
| efType = IccConstants.EF_FDN; |
| break; |
| |
| default: |
| throw new UnsupportedOperationException( |
| "Cannot insert into URL: " + url); |
| } |
| |
| // parse where clause |
| String tag = null; |
| String number = null; |
| String[] emails = null; |
| String pin2 = null; |
| |
| String[] tokens = where.split("AND"); |
| int n = tokens.length; |
| |
| while (--n >= 0) { |
| String param = tokens[n]; |
| if (DBG) log("parsing '" + param + "'"); |
| |
| String[] pair = param.split("="); |
| |
| if (pair.length != 2) { |
| Log.e(TAG, "resolve: bad whereClause parameter: " + param); |
| continue; |
| } |
| |
| String key = pair[0].trim(); |
| String val = pair[1].trim(); |
| |
| if (STR_TAG.equals(key)) { |
| tag = normalizeValue(val); |
| } else if (STR_NUMBER.equals(key)) { |
| number = normalizeValue(val); |
| } else if (STR_EMAILS.equals(key)) { |
| //TODO(): Email is null. |
| emails = null; |
| } else if (STR_PIN2.equals(key)) { |
| pin2 = normalizeValue(val); |
| } |
| } |
| |
| if (TextUtils.isEmpty(number)) { |
| return 0; |
| } |
| |
| if (efType == IccConstants.EF_FDN && TextUtils.isEmpty(pin2)) { |
| return 0; |
| } |
| |
| boolean success = deleteIccRecordFromEf(efType, tag, number, emails, pin2); |
| if (!success) { |
| return 0; |
| } |
| |
| return 1; |
| } |
| |
| @Override |
| public int update(Uri url, ContentValues values, String where, String[] whereArgs) { |
| int efType; |
| String pin2 = null; |
| |
| if (DBG) log("update"); |
| |
| int match = URL_MATCHER.match(url); |
| switch (match) { |
| case ADN: |
| efType = IccConstants.EF_ADN; |
| break; |
| |
| case FDN: |
| efType = IccConstants.EF_FDN; |
| pin2 = values.getAsString("pin2"); |
| break; |
| |
| default: |
| throw new UnsupportedOperationException( |
| "Cannot insert into URL: " + url); |
| } |
| |
| String tag = values.getAsString("tag"); |
| String number = values.getAsString("number"); |
| String[] emails = null; |
| String newTag = values.getAsString("newTag"); |
| String newNumber = values.getAsString("newNumber"); |
| String[] newEmails = null; |
| // TODO(): Update for email. |
| boolean success = updateIccRecordInEf(efType, tag, number, |
| newTag, newNumber, pin2); |
| |
| if (!success) { |
| return 0; |
| } |
| |
| return 1; |
| } |
| |
| private ArrayList<ArrayList> loadFromEf(int efType) { |
| ArrayList<ArrayList> results = new ArrayList<ArrayList>(); |
| List<AdnRecord> adnRecords = null; |
| |
| if (DBG) log("loadFromEf: efType=" + efType); |
| |
| try { |
| IIccPhoneBook iccIpb = IIccPhoneBook.Stub.asInterface( |
| ServiceManager.getService("simphonebook")); |
| if (iccIpb != null) { |
| adnRecords = iccIpb.getAdnRecordsInEf(efType); |
| } |
| } catch (RemoteException ex) { |
| // ignore it |
| } catch (SecurityException ex) { |
| if (DBG) log(ex.toString()); |
| } |
| if (adnRecords != null) { |
| // Load the results |
| |
| int N = adnRecords.size(); |
| if (DBG) log("adnRecords.size=" + N); |
| for (int i = 0; i < N ; i++) { |
| loadRecord(adnRecords.get(i), results); |
| } |
| } else { |
| // No results to load |
| Log.w(TAG, "Cannot load ADN records"); |
| results.clear(); |
| } |
| if (DBG) log("loadFromEf: return results"); |
| return results; |
| } |
| |
| private boolean |
| addIccRecordToEf(int efType, String name, String number, String[] emails, String pin2) { |
| if (DBG) log("addIccRecordToEf: efType=" + efType + ", name=" + name + |
| ", number=" + number + ", emails=" + emails); |
| |
| boolean success = false; |
| |
| // TODO: do we need to call getAdnRecordsInEf() before calling |
| // updateAdnRecordsInEfBySearch()? In any case, we will leave |
| // the UI level logic to fill that prereq if necessary. But |
| // hopefully, we can remove this requirement. |
| |
| try { |
| IIccPhoneBook iccIpb = IIccPhoneBook.Stub.asInterface( |
| ServiceManager.getService("simphonebook")); |
| if (iccIpb != null) { |
| success = iccIpb.updateAdnRecordsInEfBySearch(efType, "", "", |
| name, number, pin2); |
| } |
| } catch (RemoteException ex) { |
| // ignore it |
| } catch (SecurityException ex) { |
| if (DBG) log(ex.toString()); |
| } |
| if (DBG) log("addIccRecordToEf: " + success); |
| return success; |
| } |
| |
| private boolean |
| updateIccRecordInEf(int efType, String oldName, String oldNumber, |
| String newName, String newNumber, String pin2) { |
| if (DBG) log("updateIccRecordInEf: efType=" + efType + |
| ", oldname=" + oldName + ", oldnumber=" + oldNumber + |
| ", newname=" + newName + ", newnumber=" + newNumber); |
| boolean success = false; |
| |
| try { |
| IIccPhoneBook iccIpb = IIccPhoneBook.Stub.asInterface( |
| ServiceManager.getService("simphonebook")); |
| if (iccIpb != null) { |
| success = iccIpb.updateAdnRecordsInEfBySearch(efType, |
| oldName, oldNumber, newName, newNumber, pin2); |
| } |
| } catch (RemoteException ex) { |
| // ignore it |
| } catch (SecurityException ex) { |
| if (DBG) log(ex.toString()); |
| } |
| if (DBG) log("updateIccRecordInEf: " + success); |
| return success; |
| } |
| |
| |
| private boolean deleteIccRecordFromEf(int efType, String name, String number, String[] emails, |
| String pin2) { |
| if (DBG) log("deleteIccRecordFromEf: efType=" + efType + |
| ", name=" + name + ", number=" + number + ", emails=" + emails + ", pin2=" + pin2); |
| |
| boolean success = false; |
| |
| try { |
| IIccPhoneBook iccIpb = IIccPhoneBook.Stub.asInterface( |
| ServiceManager.getService("simphonebook")); |
| if (iccIpb != null) { |
| success = iccIpb.updateAdnRecordsInEfBySearch(efType, |
| name, number, "", "", pin2); |
| } |
| } catch (RemoteException ex) { |
| // ignore it |
| } catch (SecurityException ex) { |
| if (DBG) log(ex.toString()); |
| } |
| if (DBG) log("deleteIccRecordFromEf: " + success); |
| return success; |
| } |
| |
| /** |
| * Loads an AdnRecord into an ArrayList. Must be called with mLock held. |
| * |
| * @param record the ADN record to load from |
| * @param results the array list to put the results in |
| */ |
| private void loadRecord(AdnRecord record, |
| ArrayList<ArrayList> results) { |
| if (!record.isEmpty()) { |
| ArrayList<String> contact = new ArrayList<String>(); |
| String alphaTag = record.getAlphaTag(); |
| String number = record.getNumber(); |
| String[] emails = record.getEmails(); |
| |
| if (DBG) log("loadRecord: " + alphaTag + ", " + number + ","); |
| contact.add(alphaTag); |
| contact.add(number); |
| StringBuilder emailString = new StringBuilder(); |
| |
| if (emails != null) { |
| for (String email: emails) { |
| if (DBG) log("Adding email:" + email); |
| emailString.append(email); |
| emailString.append(","); |
| } |
| contact.add(emailString.toString()); |
| } else { |
| contact.add(null); |
| } |
| results.add(contact); |
| } |
| } |
| |
| private void log(String msg) { |
| Log.d(TAG, "[IccProvider] " + msg); |
| } |
| |
| } |