luxiaol | dbe4a62 | 2013-07-19 17:14:06 +0800 | [diff] [blame] | 1 | /* |
| 2 | * Copyright (C) 2006 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.browser; |
| 18 | |
| 19 | import android.app.Activity; |
| 20 | import android.app.AlertDialog; |
| 21 | import android.app.LoaderManager; |
| 22 | import android.app.LoaderManager.LoaderCallbacks; |
| 23 | import android.content.AsyncTaskLoader; |
| 24 | import android.content.ContentResolver; |
| 25 | import android.content.ContentUris; |
| 26 | import android.content.ContentValues; |
| 27 | import android.content.Context; |
| 28 | import android.content.CursorLoader; |
| 29 | import android.content.DialogInterface; |
| 30 | import android.content.Loader; |
| 31 | import android.content.res.Resources; |
| 32 | import android.database.Cursor; |
| 33 | import android.graphics.drawable.Drawable; |
| 34 | import android.net.Uri; |
| 35 | import android.os.Bundle; |
| 36 | import android.provider.BrowserContract; |
| 37 | import android.provider.BrowserContract.Accounts; |
| 38 | import android.text.InputFilter; |
| 39 | import android.text.Spanned; |
| 40 | import android.text.TextUtils; |
| 41 | import android.util.Log; |
| 42 | import android.view.KeyEvent; |
| 43 | import android.view.LayoutInflater; |
| 44 | import android.view.View; |
| 45 | import android.view.ViewGroup; |
| 46 | import android.view.Window; |
| 47 | import android.view.inputmethod.EditorInfo; |
| 48 | import android.view.inputmethod.InputMethodManager; |
| 49 | import android.widget.AdapterView; |
| 50 | import android.widget.AdapterView.OnItemSelectedListener; |
| 51 | import android.widget.ArrayAdapter; |
| 52 | import android.widget.CursorAdapter; |
| 53 | import android.widget.EditText; |
| 54 | import android.widget.Spinner; |
| 55 | import android.widget.TextView; |
| 56 | import android.widget.Toast; |
| 57 | |
| 58 | import com.android.browser.addbookmark.FolderSpinner; |
| 59 | import com.android.browser.addbookmark.FolderSpinnerAdapter; |
| 60 | import com.android.browser.provider.BrowserProvider2; |
| 61 | |
| 62 | public class AddBookmarkFolder extends Activity implements View.OnClickListener, |
| 63 | TextView.OnEditorActionListener, AdapterView.OnItemClickListener, |
| 64 | LoaderManager.LoaderCallbacks<Cursor>, BreadCrumbView.Controller, |
| 65 | FolderSpinner.OnSetSelectionListener, OnItemSelectedListener { |
| 66 | |
| 67 | public static final long DEFAULT_FOLDER_ID = -1; |
| 68 | |
| 69 | // Place on an edited bookmark to remove the saved thumbnail |
| 70 | public static final String CHECK_FOR_DUPE = "check_for_dupe"; |
| 71 | |
| 72 | public static final String BOOKMARK_CURRENT_ID = "bookmark_current_id"; |
| 73 | |
| 74 | /* package */static final String EXTRA_EDIT_BOOKMARK = "bookmark"; |
| 75 | |
| 76 | /* package */static final String EXTRA_IS_FOLDER = "is_folder"; |
| 77 | |
| 78 | private static final int MAX_CRUMBS_SHOWN = 2; |
| 79 | |
| 80 | private long mOriginalFolder = -1; |
| 81 | |
| 82 | private boolean mIsFolderChanged = false; |
| 83 | |
| 84 | private boolean mIsOtherFolderSelected = false; |
| 85 | |
| 86 | private boolean mIsRecentFolder = false; |
| 87 | |
| 88 | // IDs for the CursorLoaders that are used. |
| 89 | private static final int LOADER_ID_ACCOUNTS = 0; |
| 90 | |
| 91 | private static final int LOADER_ID_FOLDER_CONTENTS = 1; |
| 92 | |
| 93 | private static final int LOADER_ID_EDIT_INFO = 2; |
| 94 | |
| 95 | private EditText mTitle; |
| 96 | |
| 97 | private EditText mAddress; |
| 98 | |
| 99 | private TextView mButton; |
| 100 | |
| 101 | private View mCancelButton; |
| 102 | |
| 103 | private Bundle mMap; |
| 104 | |
| 105 | private FolderSpinner mFolder; |
| 106 | |
| 107 | private View mDefaultView; |
| 108 | |
| 109 | private View mFolderSelector; |
| 110 | |
| 111 | private EditText mFolderNamer; |
| 112 | |
| 113 | private View mFolderCancel; |
| 114 | |
| 115 | private boolean mIsFolderNamerShowing; |
| 116 | |
| 117 | private View mFolderNamerHolder; |
| 118 | |
| 119 | private View mAddNewFolder; |
| 120 | |
| 121 | private View mAddSeparator; |
| 122 | |
| 123 | private long mCurrentFolder; |
| 124 | |
| 125 | private FolderAdapter mAdapter; |
| 126 | |
| 127 | private BreadCrumbView mCrumbs; |
| 128 | |
| 129 | private TextView mFakeTitle; |
| 130 | |
| 131 | private View mCrumbHolder; |
| 132 | |
| 133 | private AddBookmarkPage.CustomListView mListView; |
| 134 | |
| 135 | private long mRootFolder; |
| 136 | |
| 137 | private TextView mTopLevelLabel; |
| 138 | |
| 139 | private Drawable mHeaderIcon; |
| 140 | |
| 141 | private View mRemoveLink; |
| 142 | |
| 143 | private View mFakeTitleHolder; |
| 144 | |
| 145 | private FolderSpinnerAdapter mFolderAdapter; |
| 146 | |
| 147 | private Spinner mAccountSpinner; |
| 148 | |
| 149 | private ArrayAdapter<BookmarkAccount> mAccountAdapter; |
| 150 | |
| 151 | |
| 152 | private static class Folder { |
| 153 | String mName; |
| 154 | |
| 155 | long mId; |
| 156 | |
| 157 | Folder(String name, long id) { |
| 158 | mName = name; |
| 159 | mId = id; |
| 160 | } |
| 161 | } |
| 162 | |
| 163 | private InputMethodManager getInputMethodManager() { |
| 164 | return (InputMethodManager) getSystemService(INPUT_METHOD_SERVICE); |
| 165 | } |
| 166 | |
| 167 | private Uri getUriForFolder(long folder) { |
| 168 | BookmarkAccount account = (BookmarkAccount) mAccountSpinner.getSelectedItem(); |
| 169 | if (folder == mRootFolder && account != null) { |
| 170 | return BookmarksLoader.addAccount(BrowserContract.Bookmarks.CONTENT_URI_DEFAULT_FOLDER, |
| 171 | account.mAccountType, account.mAccountName); |
| 172 | } |
| 173 | return BrowserContract.Bookmarks.buildFolderUri(folder); |
| 174 | } |
| 175 | |
| 176 | public static long getIdFromData(Object data) { |
| 177 | if (data == null) { |
| 178 | return BrowserProvider2.FIXED_ID_ROOT; |
| 179 | } else { |
| 180 | Folder folder = (Folder) data; |
| 181 | return folder.mId; |
| 182 | } |
| 183 | } |
| 184 | |
| 185 | @Override |
| 186 | public void onTop(BreadCrumbView view, int level, Object data) { |
| 187 | if (null == data) { |
| 188 | return; |
| 189 | } |
| 190 | Folder folderData = (Folder) data; |
| 191 | long folder = folderData.mId; |
| 192 | LoaderManager manager = getLoaderManager(); |
| 193 | CursorLoader loader = (CursorLoader) ((Loader<?>) manager |
| 194 | .getLoader(LOADER_ID_FOLDER_CONTENTS)); |
| 195 | loader.setUri(getUriForFolder(folder)); |
| 196 | loader.forceLoad(); |
| 197 | if (mIsFolderNamerShowing) { |
| 198 | completeOrCancelFolderNaming(true); |
| 199 | } |
| 200 | setShowBookmarkIcon(level == 1); |
| 201 | } |
| 202 | |
| 203 | /** |
| 204 | * Show or hide the icon for bookmarks next to "Bookmarks" in the crumb |
| 205 | * view. |
| 206 | * |
| 207 | * @param show True if the icon should visible, false otherwise. |
| 208 | */ |
| 209 | private void setShowBookmarkIcon(boolean show) { |
| 210 | Drawable drawable = show ? mHeaderIcon : null; |
| 211 | mTopLevelLabel.setCompoundDrawablesWithIntrinsicBounds(drawable, null, null, null); |
| 212 | } |
| 213 | |
| 214 | @Override |
| 215 | public boolean onEditorAction(TextView v, int actionId, KeyEvent event) { |
| 216 | if (v == mFolderNamer) { |
| 217 | if (v.getText().length() > 0) { |
| 218 | if (actionId == EditorInfo.IME_NULL) { |
| 219 | // Only want to do this once. |
| 220 | if (event.getAction() == KeyEvent.ACTION_UP) { |
| 221 | completeOrCancelFolderNaming(false); |
| 222 | } |
| 223 | } |
| 224 | } |
| 225 | // Steal the key press; otherwise a newline will be added |
| 226 | return true; |
| 227 | } |
| 228 | return false; |
| 229 | } |
| 230 | |
| 231 | private void switchToDefaultView(boolean changedFolder) { |
| 232 | mFolderSelector.setVisibility(View.GONE); |
| 233 | mDefaultView.setVisibility(View.VISIBLE); |
| 234 | mCrumbHolder.setVisibility(View.GONE); |
| 235 | mFakeTitleHolder.setVisibility(View.VISIBLE); |
| 236 | if (changedFolder) { |
| 237 | Object data = mCrumbs.getTopData(); |
| 238 | if (data != null) { |
| 239 | Folder folder = (Folder) data; |
| 240 | mCurrentFolder = folder.mId; |
| 241 | if (mCurrentFolder == mRootFolder) { |
| 242 | // The Spinner changed to show "Other folder ..." Change |
| 243 | // it back to "Bookmarks", which is position 0 if we are |
| 244 | // editing a folder, 1 otherwise. |
| 245 | mFolder.setSelectionIgnoringSelectionChange(0); |
| 246 | } else { |
| 247 | mFolderAdapter.setOtherFolderDisplayText(folder.mName); |
| 248 | } |
| 249 | } |
| 250 | } else { |
| 251 | if (mCurrentFolder == mRootFolder) { |
| 252 | mFolder.setSelectionIgnoringSelectionChange(0); |
| 253 | } else { |
| 254 | Object data = mCrumbs.getTopData(); |
| 255 | if (data != null && ((Folder) data).mId == mCurrentFolder) { |
| 256 | // We are showing the correct folder hierarchy. The |
| 257 | // folder selector will say "Other folder..." Change it |
| 258 | // to say the name of the folder once again. |
| 259 | mFolderAdapter.setOtherFolderDisplayText(((Folder) data).mName); |
| 260 | } else { |
| 261 | // We are not showing the correct folder hierarchy. |
| 262 | // Clear the Crumbs and find the proper folder |
| 263 | setupTopCrumb(); |
| 264 | LoaderManager manager = getLoaderManager(); |
| 265 | manager.restartLoader(LOADER_ID_FOLDER_CONTENTS, null, this); |
| 266 | |
| 267 | } |
| 268 | } |
| 269 | } |
| 270 | } |
| 271 | |
| 272 | @Override |
| 273 | public void onClick(View v) { |
| 274 | if (v == mButton) { |
| 275 | if (mFolderSelector.getVisibility() == View.VISIBLE) { |
| 276 | // We are showing the folder selector. |
| 277 | if (mIsFolderNamerShowing) { |
| 278 | completeOrCancelFolderNaming(false); |
| 279 | } else { |
| 280 | switchToDefaultView(true); |
| 281 | } |
| 282 | } else { |
| 283 | if (save()) { |
| 284 | finish(); |
| 285 | } |
| 286 | } |
| 287 | } else if (v == mCancelButton) { |
| 288 | if (mIsFolderNamerShowing) { |
| 289 | completeOrCancelFolderNaming(true); |
| 290 | } else if (mFolderSelector.getVisibility() == View.VISIBLE) { |
| 291 | switchToDefaultView(false); |
| 292 | } else { |
| 293 | finish(); |
| 294 | } |
| 295 | } else if (v == mFolderCancel) { |
| 296 | completeOrCancelFolderNaming(true); |
| 297 | } |
| 298 | } |
| 299 | |
| 300 | private void displayToastForExistingFolder() { |
| 301 | Toast.makeText(getApplicationContext(), R.string.duplicated_folder_warning, |
| 302 | Toast.LENGTH_LONG).show(); |
| 303 | } |
| 304 | |
| 305 | @Override |
| 306 | public void onSetSelection(long id) { |
| 307 | int intId = (int) id; |
| 308 | mIsFolderChanged = true; |
| 309 | mIsOtherFolderSelected = false; |
| 310 | mIsRecentFolder = false; |
| 311 | switch (intId) { |
| 312 | case FolderSpinnerAdapter.ROOT_FOLDER: |
| 313 | mCurrentFolder = mRootFolder; |
| 314 | mOriginalFolder = mCurrentFolder; |
| 315 | break; |
| 316 | case FolderSpinnerAdapter.HOME_SCREEN: |
| 317 | |
| 318 | break; |
| 319 | case FolderSpinnerAdapter.OTHER_FOLDER: |
| 320 | mIsOtherFolderSelected = true; |
| 321 | switchToFolderSelector(); |
| 322 | break; |
| 323 | case FolderSpinnerAdapter.RECENT_FOLDER: |
| 324 | mCurrentFolder = mFolderAdapter.recentFolderId(); |
| 325 | mOriginalFolder = mCurrentFolder; |
| 326 | mIsRecentFolder = true; |
| 327 | // In case the user decides to select OTHER_FOLDER |
| 328 | // and choose a different one, so that we will start from |
| 329 | // the correct place. |
| 330 | LoaderManager manager = getLoaderManager(); |
| 331 | manager.restartLoader(LOADER_ID_FOLDER_CONTENTS, null, this); |
| 332 | break; |
| 333 | default: |
| 334 | break; |
| 335 | } |
| 336 | } |
| 337 | |
| 338 | /** |
| 339 | * Finish naming a folder, and close the IME |
| 340 | * |
| 341 | * @param cancel If true, the new folder is not created. If false, the new |
| 342 | * folder is created and the user is taken inside it. |
| 343 | */ |
| 344 | private void completeOrCancelFolderNaming(boolean cancel) { |
| 345 | if (!cancel && !TextUtils.isEmpty(mFolderNamer.getText())) { |
| 346 | String name = mFolderNamer.getText().toString(); |
| 347 | long id = addFolderToCurrent(mFolderNamer.getText().toString()); |
| 348 | descendInto(name, id); |
| 349 | } |
| 350 | setShowFolderNamer(false); |
| 351 | getInputMethodManager().hideSoftInputFromWindow(mListView.getWindowToken(), 0); |
| 352 | } |
| 353 | |
| 354 | private long addFolderToCurrent(String name) { |
| 355 | // Add the folder to the database |
| 356 | ContentValues values = new ContentValues(); |
| 357 | values.put(BrowserContract.Bookmarks.TITLE, name); |
| 358 | values.put(BrowserContract.Bookmarks.IS_FOLDER, 1); |
| 359 | long currentFolder; |
| 360 | Object data = null; |
| 361 | if (null != mCrumbs) { |
| 362 | data = mCrumbs.getTopData(); |
| 363 | } |
| 364 | if (data != null) { |
| 365 | currentFolder = ((Folder) data).mId; |
| 366 | } else { |
| 367 | currentFolder = mRootFolder; |
| 368 | } |
| 369 | currentFolder = mCurrentFolder; |
| 370 | if (mIsRecentFolder) { |
| 371 | values.put(BrowserContract.Bookmarks.PARENT, mCurrentFolder); |
| 372 | } else if (!(mIsFolderChanged && mIsOtherFolderSelected) && mOriginalFolder != -1) { |
| 373 | values.put(BrowserContract.Bookmarks.PARENT, mOriginalFolder); |
| 374 | } else { |
| 375 | values.put(BrowserContract.Bookmarks.PARENT, currentFolder); |
| 376 | } |
| 377 | Uri uri = getContentResolver().insert(BrowserContract.Bookmarks.CONTENT_URI, values); |
| 378 | if (uri != null) { |
| 379 | return ContentUris.parseId(uri); |
| 380 | } else { |
| 381 | return -1; |
| 382 | } |
| 383 | } |
| 384 | |
| 385 | private void switchToFolderSelector() { |
| 386 | // Set the list to the top in case it is scrolled. |
| 387 | mListView.setSelection(0); |
| 388 | mFakeTitleHolder.setVisibility(View.GONE); |
| 389 | // mFakeTitle.setVisibility(View.GONE); |
| 390 | mDefaultView.setVisibility(View.GONE); |
| 391 | mFolderSelector.setVisibility(View.VISIBLE); |
| 392 | mCrumbHolder.setVisibility(View.VISIBLE); |
| 393 | getInputMethodManager().hideSoftInputFromWindow(mListView.getWindowToken(), 0); |
| 394 | } |
| 395 | |
| 396 | private void descendInto(String foldername, long id) { |
| 397 | if (id != DEFAULT_FOLDER_ID) { |
| 398 | mCrumbs.pushView(foldername, new Folder(foldername, id)); |
| 399 | mCrumbs.notifyController(); |
| 400 | } else { |
| 401 | Toast.makeText(getApplicationContext(), R.string.duplicated_folder_warning, |
| 402 | Toast.LENGTH_LONG).show(); |
| 403 | } |
| 404 | } |
| 405 | |
| 406 | private LoaderCallbacks<EditBookmarkInfo> mEditInfoLoaderCallbacks = new LoaderCallbacks<EditBookmarkInfo>() { |
| 407 | |
| 408 | @Override |
| 409 | public void onLoaderReset(Loader<EditBookmarkInfo> loader) { |
| 410 | // Don't care |
| 411 | } |
| 412 | |
| 413 | @Override |
| 414 | public void onLoadFinished(Loader<EditBookmarkInfo> loader, EditBookmarkInfo info) { |
| 415 | boolean setAccount = false; |
| 416 | // TODO: Detect if lastUsedId is a subfolder of info.id in the |
| 417 | // editing folder case. For now, just don't show the last used |
| 418 | // folder at all to prevent any chance of the user adding a parent |
| 419 | // folder to a child folder |
| 420 | if (info.mLastUsedId != -1 && info.mLastUsedId != info.mId) { |
| 421 | if (setAccount && info.mLastUsedId != mRootFolder |
| 422 | && TextUtils.equals(info.mLastUsedAccountName, info.mAccountName) |
| 423 | && TextUtils.equals(info.mLastUsedAccountType, info.mAccountType)) { |
| 424 | mFolderAdapter.addRecentFolder(info.mLastUsedId, info.mLastUsedTitle); |
| 425 | } else if (!setAccount) { |
| 426 | setAccount = true; |
| 427 | setAccount(info.mLastUsedAccountName, info.mLastUsedAccountType); |
| 428 | if (info.mLastUsedId != mRootFolder) { |
| 429 | mFolderAdapter.addRecentFolder(info.mLastUsedId, info.mLastUsedTitle); |
| 430 | } |
| 431 | } |
| 432 | } |
| 433 | if (!setAccount) { |
| 434 | mAccountSpinner.setSelection(0); |
| 435 | } |
| 436 | } |
| 437 | |
| 438 | @Override |
| 439 | public Loader<EditBookmarkInfo> onCreateLoader(int id, Bundle args) { |
| 440 | return new EditBookmarkInfoLoader(AddBookmarkFolder.this, mMap); |
| 441 | } |
| 442 | }; |
| 443 | |
| 444 | void setAccount(String accountName, String accountType) { |
| 445 | for (int i = 0; i < mAccountAdapter.getCount(); i++) { |
| 446 | BookmarkAccount account = mAccountAdapter.getItem(i); |
| 447 | if (TextUtils.equals(account.mAccountName, accountName) |
| 448 | && TextUtils.equals(account.mAccountType, accountType)) { |
| 449 | mAccountSpinner.setSelection(i); |
| 450 | onRootFolderFound(account.rootFolderId); |
| 451 | return; |
| 452 | } |
| 453 | } |
| 454 | } |
| 455 | |
| 456 | @Override |
| 457 | public Loader<Cursor> onCreateLoader(int id, Bundle args) { |
| 458 | String[] projection; |
| 459 | switch (id) { |
| 460 | case LOADER_ID_ACCOUNTS: |
| 461 | return new AccountsLoader(this); |
| 462 | case LOADER_ID_FOLDER_CONTENTS: |
| 463 | projection = new String[] { |
| 464 | BrowserContract.Bookmarks._ID, BrowserContract.Bookmarks.TITLE, |
| 465 | BrowserContract.Bookmarks.IS_FOLDER |
| 466 | }; |
| 467 | String where = BrowserContract.Bookmarks.IS_FOLDER + " != 0" + " AND " |
| 468 | + BrowserContract.Bookmarks._ID + " != ?"; |
| 469 | String whereArgs[] = new String[] { |
| 470 | Long.toString(mMap.getLong(BrowserContract.Bookmarks._ID)) |
| 471 | }; |
| 472 | long currentFolder; |
| 473 | Object data = mCrumbs.getTopData(); |
| 474 | if (data != null) { |
| 475 | currentFolder = ((Folder) data).mId; |
| 476 | } else { |
| 477 | currentFolder = mRootFolder; |
| 478 | } |
| 479 | return new CursorLoader(this, getUriForFolder(currentFolder), projection, where, |
| 480 | whereArgs, BrowserContract.Bookmarks._ID + " ASC"); |
| 481 | default: |
| 482 | throw new AssertionError("Asking for nonexistant loader!"); |
| 483 | } |
| 484 | } |
| 485 | |
| 486 | @Override |
| 487 | public void onLoadFinished(Loader<Cursor> loader, Cursor cursor) { |
| 488 | switch (loader.getId()) { |
| 489 | case LOADER_ID_ACCOUNTS: |
| 490 | mAccountAdapter.clear(); |
| 491 | while (cursor.moveToNext()) { |
| 492 | mAccountAdapter.add(new BookmarkAccount(this, cursor)); |
| 493 | } |
| 494 | getLoaderManager().destroyLoader(LOADER_ID_ACCOUNTS); |
| 495 | getLoaderManager().restartLoader(LOADER_ID_EDIT_INFO, null, |
| 496 | mEditInfoLoaderCallbacks); |
| 497 | break; |
| 498 | case LOADER_ID_FOLDER_CONTENTS: |
| 499 | mAdapter.changeCursor(cursor); |
| 500 | break; |
| 501 | default: |
| 502 | break; |
| 503 | } |
| 504 | } |
| 505 | |
| 506 | public void onLoaderReset(Loader<Cursor> loader) { |
| 507 | switch (loader.getId()) { |
| 508 | case LOADER_ID_FOLDER_CONTENTS: |
| 509 | mAdapter.changeCursor(null); |
| 510 | break; |
| 511 | default: |
| 512 | break; |
| 513 | } |
| 514 | } |
| 515 | |
| 516 | /** |
| 517 | * Move cursor to the position that has folderToFind as its "_id". |
| 518 | * |
| 519 | * @param cursor Cursor containing folders in the bookmarks database |
| 520 | * @param folderToFind "_id" of the folder to move to. |
| 521 | * @param idIndex Index in cursor of "_id" |
| 522 | * @throws AssertionError if cursor is empty or there is no row with |
| 523 | * folderToFind as its "_id". |
| 524 | */ |
| 525 | void moveCursorToFolder(Cursor cursor, long folderToFind, int idIndex) throws AssertionError { |
| 526 | if (!cursor.moveToFirst()) { |
| 527 | throw new AssertionError("No folders in the database!"); |
| 528 | } |
| 529 | long folder; |
| 530 | do { |
| 531 | folder = cursor.getLong(idIndex); |
| 532 | } while (folder != folderToFind && cursor.moveToNext()); |
| 533 | if (cursor.isAfterLast()) { |
| 534 | throw new AssertionError("Folder(id=" + folderToFind |
| 535 | + ") holding this bookmark does not exist!"); |
| 536 | } |
| 537 | } |
| 538 | |
| 539 | @Override |
| 540 | public void onItemClick(AdapterView<?> parent, View view, int position, long id) { |
| 541 | TextView tv = (TextView) view.findViewById(android.R.id.text1); |
| 542 | // Switch to the folder that was clicked on. |
| 543 | descendInto(tv.getText().toString(), id); |
| 544 | } |
| 545 | |
| 546 | private void setShowFolderNamer(boolean show) { |
| 547 | if (show != mIsFolderNamerShowing) { |
| 548 | mIsFolderNamerShowing = show; |
| 549 | if (show) { |
| 550 | // Set the selection to the folder namer so it will be in |
| 551 | // view. |
| 552 | mListView.addFooterView(mFolderNamerHolder); |
| 553 | } else { |
| 554 | mListView.removeFooterView(mFolderNamerHolder); |
| 555 | } |
| 556 | // Refresh the list. |
| 557 | mListView.setAdapter(mAdapter); |
| 558 | if (show) { |
| 559 | mListView.setSelection(mListView.getCount() - 1); |
| 560 | } |
| 561 | } |
| 562 | } |
| 563 | |
| 564 | /** |
| 565 | * Shows a list of names of folders. |
| 566 | */ |
| 567 | private class FolderAdapter extends CursorAdapter { |
| 568 | public FolderAdapter(Context context) { |
| 569 | super(context, null); |
| 570 | } |
| 571 | |
| 572 | @Override |
| 573 | public void bindView(View view, Context context, Cursor cursor) { |
| 574 | ((TextView) view.findViewById(android.R.id.text1)).setText(cursor.getString(cursor |
| 575 | .getColumnIndexOrThrow(BrowserContract.Bookmarks.TITLE))); |
| 576 | } |
| 577 | |
| 578 | @Override |
| 579 | public View newView(Context context, Cursor cursor, ViewGroup parent) { |
| 580 | View view = LayoutInflater.from(context).inflate(R.layout.folder_list_item, null); |
| 581 | view.setBackgroundDrawable(context.getResources().getDrawable( |
| 582 | android.R.drawable.list_selector_background)); |
| 583 | return view; |
| 584 | } |
| 585 | |
| 586 | @Override |
| 587 | public boolean isEmpty() { |
| 588 | // Do not show the empty view if the user is creating a new folder. |
| 589 | return super.isEmpty() && !mIsFolderNamerShowing; |
| 590 | } |
| 591 | } |
| 592 | |
| 593 | @Override |
| 594 | protected void onCreate(Bundle icicle) { |
| 595 | super.onCreate(icicle); |
| 596 | requestWindowFeature(Window.FEATURE_NO_TITLE); |
| 597 | |
| 598 | mMap = getIntent().getExtras(); |
| 599 | |
| 600 | setContentView(R.layout.browser_add_bookmark); |
| 601 | |
| 602 | Window window = getWindow(); |
| 603 | |
| 604 | String title = this.getString(R.string.new_folder); |
| 605 | mFakeTitle = (TextView) findViewById(R.id.fake_title); |
| 606 | mFakeTitleHolder = findViewById(R.id.title_holder); |
| 607 | mFakeTitle.setText(this.getString(R.string.new_folder)); |
| 608 | |
| 609 | mTitle = (EditText) findViewById(R.id.title); |
| 610 | // add for cmcc test about waring limit of edit text |
| 611 | BrowserUtils.maxLengthFilter(AddBookmarkFolder.this, mTitle, BrowserUtils.FILENAME_MAX_LENGTH); |
| 612 | |
| 613 | mTitle.setText(title); |
| 614 | mAddress = (EditText) findViewById(R.id.address); |
| 615 | mAddress.setVisibility(View.GONE); |
| 616 | findViewById(R.id.row_address).setVisibility(View.GONE); |
| 617 | |
| 618 | mButton = (TextView) findViewById(R.id.OK); |
| 619 | mButton.setOnClickListener(this); |
| 620 | |
| 621 | mCancelButton = findViewById(R.id.cancel); |
| 622 | mCancelButton.setOnClickListener(this); |
| 623 | |
| 624 | mFolder = (FolderSpinner) findViewById(R.id.folder); |
| 625 | mFolderAdapter = new FolderSpinnerAdapter(this, false); |
| 626 | mFolder.setAdapter(mFolderAdapter); |
| 627 | mFolder.setOnSetSelectionListener(this); |
| 628 | |
| 629 | mDefaultView = findViewById(R.id.default_view); |
| 630 | mFolderSelector = findViewById(R.id.folder_selector); |
| 631 | |
| 632 | mFolderNamerHolder = getLayoutInflater().inflate(R.layout.new_folder_layout, null); |
| 633 | mFolderNamer = (EditText) mFolderNamerHolder.findViewById(R.id.folder_namer); |
| 634 | mFolderNamer.setOnEditorActionListener(this); |
| 635 | mFolderCancel = mFolderNamerHolder.findViewById(R.id.close); |
| 636 | mFolderCancel.setOnClickListener(this); |
| 637 | |
| 638 | mAddNewFolder = findViewById(R.id.add_new_folder); |
| 639 | mAddNewFolder.setVisibility(View.GONE); |
| 640 | mAddSeparator = findViewById(R.id.add_divider); |
| 641 | mAddSeparator.setVisibility(View.GONE); |
| 642 | |
| 643 | mCrumbs = (BreadCrumbView) findViewById(R.id.crumbs); |
| 644 | mCrumbs.setUseBackButton(true); |
| 645 | mCrumbs.setController(this); |
| 646 | mHeaderIcon = getResources().getDrawable(R.drawable.ic_folder_holo_dark); |
| 647 | mCrumbHolder = findViewById(R.id.crumb_holder); |
| 648 | mCrumbs.setMaxVisible(MAX_CRUMBS_SHOWN); |
| 649 | |
| 650 | mAdapter = new FolderAdapter(this); |
| 651 | mListView = (AddBookmarkPage.CustomListView) findViewById(R.id.list); |
| 652 | View empty = findViewById(R.id.empty); |
| 653 | mListView.setEmptyView(empty); |
| 654 | mListView.setAdapter(mAdapter); |
| 655 | mListView.setOnItemClickListener(this); |
| 656 | mListView.addEditText(mFolderNamer); |
| 657 | |
| 658 | mAccountAdapter = new ArrayAdapter<BookmarkAccount>(this, |
| 659 | android.R.layout.simple_spinner_item); |
| 660 | mAccountAdapter.setDropDownViewResource(android.R.layout.simple_spinner_dropdown_item); |
| 661 | mAccountSpinner = (Spinner) findViewById(R.id.accounts); |
| 662 | mAccountSpinner.setAdapter(mAccountAdapter); |
| 663 | mAccountSpinner.setOnItemSelectedListener(this); |
| 664 | |
| 665 | if (!window.getDecorView().isInTouchMode()) { |
| 666 | mButton.requestFocus(); |
| 667 | } |
| 668 | // getLoaderManager().restartLoader(LOADER_ID_ACCOUNTS, null, this); |
| 669 | |
| 670 | setShowFolderNamer(false); |
| 671 | mFolderNamer.setText(R.string.new_folder); |
| 672 | mFolderNamer.requestFocus(); |
| 673 | InputMethodManager imm = getInputMethodManager(); |
| 674 | imm.focusIn(mListView); |
| 675 | imm.showSoftInput(mFolderNamer, InputMethodManager.SHOW_IMPLICIT); |
| 676 | |
| 677 | mCurrentFolder = getIntent().getLongExtra( |
| 678 | BrowserContract.Bookmarks.PARENT, DEFAULT_FOLDER_ID); |
| 679 | mOriginalFolder = mCurrentFolder; |
| 680 | if (!(mCurrentFolder == -1 || mCurrentFolder == 1)) { |
| 681 | mFolder.setSelectionIgnoringSelectionChange(1); |
| 682 | mFolderAdapter.setOtherFolderDisplayText(getNameFromId(mOriginalFolder)); |
| 683 | } |
| 684 | |
| 685 | getLoaderManager().restartLoader(LOADER_ID_ACCOUNTS, null, this); |
| 686 | } |
| 687 | |
| 688 | // get folder title from folder id |
| 689 | private String getNameFromId(long mCurrentFolder2) { |
| 690 | String title = ""; |
| 691 | Cursor cursor = null; |
| 692 | try { |
| 693 | cursor = getApplicationContext().getContentResolver().query( |
| 694 | BrowserContract.Bookmarks.CONTENT_URI, |
| 695 | new String[] { |
| 696 | BrowserContract.Bookmarks.TITLE |
| 697 | }, |
| 698 | BrowserContract.Bookmarks._ID + " = ? AND " |
| 699 | + BrowserContract.Bookmarks.IS_DELETED + " = ? AND " |
| 700 | + BrowserContract.Bookmarks.IS_FOLDER + " = ? ", new String[] { |
| 701 | String.valueOf(mCurrentFolder2), 0 + "", 1 + "" |
| 702 | }, null); |
| 703 | if (cursor != null && cursor.getCount() != 0) { |
| 704 | while (cursor.moveToNext()) { |
| 705 | title = cursor.getString(0); |
| 706 | } |
| 707 | } |
| 708 | } finally { |
| 709 | if (cursor != null) { |
| 710 | cursor.close(); |
| 711 | } |
| 712 | } |
| 713 | return title; |
| 714 | } |
| 715 | |
| 716 | private void showRemoveButton() { |
| 717 | findViewById(R.id.remove_divider).setVisibility(View.VISIBLE); |
| 718 | mRemoveLink = findViewById(R.id.remove); |
| 719 | mRemoveLink.setVisibility(View.VISIBLE); |
| 720 | mRemoveLink.setOnClickListener(this); |
| 721 | } |
| 722 | |
| 723 | // Called once we have determined which folder is the root folder |
| 724 | private void onRootFolderFound(long root) { |
| 725 | mRootFolder = root; |
| 726 | mCurrentFolder = mRootFolder; |
| 727 | setupTopCrumb(); |
| 728 | onCurrentFolderFound(); |
| 729 | } |
| 730 | |
| 731 | private void setupTopCrumb() { |
| 732 | mCrumbs.clear(); |
| 733 | String name = getString(R.string.bookmarks); |
| 734 | mTopLevelLabel = (TextView) mCrumbs.pushView(name, false, new Folder(name, mRootFolder)); |
| 735 | // To better match the other folders. |
| 736 | mTopLevelLabel.setCompoundDrawablePadding(6); |
| 737 | } |
| 738 | |
| 739 | private void onCurrentFolderFound() { |
| 740 | LoaderManager manager = getLoaderManager(); |
| 741 | if (mCurrentFolder != mRootFolder) { |
| 742 | // Since we're not in the root folder, change the selection to other |
| 743 | // folder now. The text will get changed once we select the correct |
| 744 | // folder. |
| 745 | mFolder.setSelectionIgnoringSelectionChange(1); |
| 746 | } else { |
| 747 | setShowBookmarkIcon(true); |
| 748 | } |
| 749 | // Find the contents of the current folder |
| 750 | manager.restartLoader(LOADER_ID_FOLDER_CONTENTS, null, this); |
| 751 | } |
| 752 | |
| 753 | /** |
| 754 | * Parse the data entered in the dialog and post a message to update the |
| 755 | * bookmarks database. |
| 756 | */ |
| 757 | private boolean save() { |
| 758 | String title = mTitle.getText().toString().trim(); |
| 759 | |
| 760 | boolean emptyTitle = title.length() == 0; |
| 761 | Resources r = getResources(); |
| 762 | if (emptyTitle) { |
| 763 | mTitle.setError(r.getText(R.string.bookmark_needs_title)); |
| 764 | return false; |
| 765 | } |
| 766 | |
| 767 | long id = addFolderToCurrent(title); |
| 768 | if (id == -1) { |
| 769 | displayToastForExistingFolder(); |
| 770 | return false; |
| 771 | } |
| 772 | |
| 773 | setResult(RESULT_OK); |
| 774 | return true; |
| 775 | } |
| 776 | |
| 777 | @Override |
| 778 | public void onItemSelected(AdapterView<?> parent, View view, int position, long id) { |
| 779 | if (mAccountSpinner == parent) { |
| 780 | long root = mAccountAdapter.getItem(position).rootFolderId; |
| 781 | if (root != mRootFolder) { |
| 782 | onRootFolderFound(root); |
| 783 | mFolderAdapter.clearRecentFolder(); |
| 784 | } |
| 785 | } |
| 786 | } |
| 787 | |
| 788 | @Override |
| 789 | public void onNothingSelected(AdapterView<?> parent) { |
| 790 | // Don't care |
| 791 | } |
| 792 | |
| 793 | static class AccountsLoader extends CursorLoader { |
| 794 | |
| 795 | static final String[] PROJECTION = new String[] { |
| 796 | Accounts.ACCOUNT_NAME, Accounts.ACCOUNT_TYPE, Accounts.ROOT_ID, |
| 797 | }; |
| 798 | |
| 799 | static final int COLUMN_INDEX_ACCOUNT_NAME = 0; |
| 800 | |
| 801 | static final int COLUMN_INDEX_ACCOUNT_TYPE = 1; |
| 802 | |
| 803 | static final int COLUMN_INDEX_ROOT_ID = 2; |
| 804 | |
| 805 | public AccountsLoader(Context context) { |
| 806 | super(context, Accounts.CONTENT_URI, PROJECTION, null, null, null); |
| 807 | } |
| 808 | |
| 809 | } |
| 810 | |
| 811 | public static class BookmarkAccount { |
| 812 | |
| 813 | private String mLabel; |
| 814 | |
| 815 | String mAccountName; |
| 816 | String mAccountType; |
| 817 | |
| 818 | public long rootFolderId; |
| 819 | |
| 820 | public BookmarkAccount(Context context, Cursor cursor) { |
| 821 | mAccountName = cursor.getString(AccountsLoader.COLUMN_INDEX_ACCOUNT_NAME); |
| 822 | mAccountType = cursor.getString(AccountsLoader.COLUMN_INDEX_ACCOUNT_TYPE); |
| 823 | rootFolderId = cursor.getLong(AccountsLoader.COLUMN_INDEX_ROOT_ID); |
| 824 | mLabel = mAccountName; |
| 825 | if (TextUtils.isEmpty(mLabel)) { |
| 826 | mLabel = context.getString(R.string.local_bookmarks); |
| 827 | } |
| 828 | } |
| 829 | |
| 830 | @Override |
| 831 | public String toString() { |
| 832 | return mLabel; |
| 833 | } |
| 834 | } |
| 835 | |
| 836 | static class EditBookmarkInfo { |
| 837 | long mId = -1; |
| 838 | |
| 839 | long mParentId = -1; |
| 840 | |
| 841 | String mParentTitle; |
| 842 | |
| 843 | String mEBITitle; |
| 844 | |
| 845 | String mAccountName; |
| 846 | |
| 847 | String mAccountType; |
| 848 | |
| 849 | long mLastUsedId = -1; |
| 850 | |
| 851 | String mLastUsedTitle; |
| 852 | |
| 853 | String mLastUsedAccountName; |
| 854 | |
| 855 | String mLastUsedAccountType; |
| 856 | } |
| 857 | |
| 858 | static class EditBookmarkInfoLoader extends AsyncTaskLoader<EditBookmarkInfo> { |
| 859 | |
| 860 | private Context mContext; |
| 861 | |
| 862 | private Bundle mMap; |
| 863 | |
| 864 | public EditBookmarkInfoLoader(Context context, Bundle bundle) { |
| 865 | super(context); |
| 866 | mContext = context.getApplicationContext(); |
| 867 | mMap = bundle; |
| 868 | } |
| 869 | |
| 870 | @Override |
| 871 | public EditBookmarkInfo loadInBackground() { |
| 872 | final ContentResolver cr = mContext.getContentResolver(); |
| 873 | EditBookmarkInfo info = new EditBookmarkInfo(); |
| 874 | Cursor c = null; |
| 875 | try { |
| 876 | // First, let's lookup the bookmark (check for dupes, get needed |
| 877 | // info) |
| 878 | String url = mMap.getString(BrowserContract.Bookmarks.URL); |
| 879 | info.mId = mMap.getLong(BrowserContract.Bookmarks._ID, -1); |
| 880 | boolean checkForDupe = mMap.getBoolean(CHECK_FOR_DUPE); |
| 881 | if (checkForDupe && info.mId == -1 && !TextUtils.isEmpty(url)) { |
| 882 | c = cr.query(BrowserContract.Bookmarks.CONTENT_URI, new String[] { |
| 883 | BrowserContract.Bookmarks._ID |
| 884 | }, BrowserContract.Bookmarks.URL + "=?", new String[] { |
| 885 | url |
| 886 | }, null); |
| 887 | if (c.getCount() == 1 && c.moveToFirst()) { |
| 888 | info.mId = c.getLong(0); |
| 889 | } |
| 890 | c.close(); |
| 891 | } |
| 892 | if (info.mId != -1) { |
| 893 | c = cr.query(ContentUris.withAppendedId(BrowserContract.Bookmarks.CONTENT_URI, |
| 894 | info.mId), new String[] { |
| 895 | BrowserContract.Bookmarks.PARENT, |
| 896 | BrowserContract.Bookmarks.ACCOUNT_NAME, |
| 897 | BrowserContract.Bookmarks.ACCOUNT_TYPE, BrowserContract.Bookmarks.TITLE |
| 898 | }, null, null, null); |
| 899 | if (c.moveToFirst()) { |
| 900 | info.mParentId = c.getLong(0); |
| 901 | info.mAccountName = c.getString(1); |
| 902 | info.mAccountType = c.getString(2); |
| 903 | info.mEBITitle = c.getString(3); |
| 904 | } |
| 905 | c.close(); |
| 906 | c = cr.query(ContentUris.withAppendedId(BrowserContract.Bookmarks.CONTENT_URI, |
| 907 | info.mParentId), new String[] { |
| 908 | BrowserContract.Bookmarks.TITLE, |
| 909 | }, null, null, null); |
| 910 | if (c.moveToFirst()) { |
| 911 | info.mParentTitle = c.getString(0); |
| 912 | } |
| 913 | c.close(); |
| 914 | } |
| 915 | |
| 916 | // Figure out the last used folder/account |
| 917 | c = cr.query(BrowserContract.Bookmarks.CONTENT_URI, new String[] { |
| 918 | BrowserContract.Bookmarks.PARENT, |
| 919 | }, null, null, BrowserContract.Bookmarks.DATE_MODIFIED + " DESC LIMIT 1"); |
| 920 | if (c.moveToFirst()) { |
| 921 | long parent = c.getLong(0); |
| 922 | c.close(); |
| 923 | c = cr.query(BrowserContract.Bookmarks.CONTENT_URI, new String[] { |
| 924 | BrowserContract.Bookmarks.TITLE, |
| 925 | BrowserContract.Bookmarks.ACCOUNT_NAME, |
| 926 | BrowserContract.Bookmarks.ACCOUNT_TYPE |
| 927 | }, BrowserContract.Bookmarks._ID + "=?", new String[] { |
| 928 | Long.toString(parent) |
| 929 | }, null); |
| 930 | if (c.moveToFirst()) { |
| 931 | info.mLastUsedId = parent; |
| 932 | info.mLastUsedTitle = c.getString(0); |
| 933 | info.mLastUsedAccountName = c.getString(1); |
| 934 | info.mLastUsedAccountType = c.getString(2); |
| 935 | } |
| 936 | c.close(); |
| 937 | } |
| 938 | } finally { |
| 939 | if (c != null) { |
| 940 | c.close(); |
| 941 | } |
| 942 | } |
| 943 | return info; |
| 944 | } |
| 945 | |
| 946 | @Override |
| 947 | protected void onStartLoading() { |
| 948 | forceLoad(); |
| 949 | } |
| 950 | } |
| 951 | } |