Android Studio中用List Fragments創(chuàng)建相冊(cè)Gallery
這是關(guān)于在Android Studio中通過(guò) fragments 進(jìn)行相機(jī)方面開(kāi)發(fā)的五篇系列文章的第二篇。如果你還沒(méi)有把我放在 GitHub 上的范例程序克隆下來(lái),那么請(qǐng)先去這里獲取***代碼。本文主要包含的是 “SimplePhotoGalleryListFragment” 這個(gè) Fragment。
注意:本范例中所涉及的 List Fragment 的用法,可以在 list fragments 這篇文章中找到詳細(xì)的講解。
AsyncTaskLoaders 以及 Fragments
加載整個(gè)圖片庫(kù)到List是一個(gè)運(yùn)算量比較密集,強(qiáng)度很高任務(wù)。因此,我們希望利用 Android 提供的AsyncTaskLoader 通過(guò)異步加載解決這個(gè)問(wèn)題。在這里,我已經(jīng)寫(xiě)好了一個(gè)自定義AsyncTaskLoader工具類用加載圖庫(kù)中圖片的。我把它命名為:PhotoGalleryImageProvider,可以在源碼中找到。
Fragments 提供了一種特殊的接口給異步任務(wù)的 Loader 以便于自動(dòng)觸發(fā)異步加載任務(wù)。我們的圖庫(kù)列表在Fragment中看起來(lái)如下面的代碼:
- @Override
- public void onCreate(Bundle savedInstanceState) {
- super.onCreate(savedInstanceState);
- // Create an empty loader and pre-initialize the photo list items as an empty list.
- Context context = getActivity().getBaseContext();
- // Set up empty mAdapter
- mPhotoListItem = new ArrayList() ;
- mAdapter = new PhotoAdapter(context,
- R.layout.photo_item,
- mPhotoListItem, false);
- // Prepare the loader. Either re-connect with an existing one,
- // or start a new one.
- getLoaderManager().initLoader(0, null, this);
- }
請(qǐng)注意***的這一行:
- getLoaderManager().initLoader(0, null, this);
這一行的作用就是自動(dòng)啟用 AsyncLoader。AsyncLoader的相關(guān)代碼放在這個(gè)Class文件的后面。
- /**
- * Loader Handlers for loading the photos in the background.
- */
- @Override
- public Loader<List> onCreateLoader(int id, Bundle args) {
- // This is called when a new Loader needs to be created. This
- // sample only has one Loader with no arguments, so it is simple.
- return new PhotoGalleryAsyncLoader(getActivity());
- }
每次后臺(tái)任務(wù)成功獲取到圖庫(kù)中的圖片時(shí),則會(huì)回調(diào)下面這個(gè)函數(shù):
- @Override
- public void onLoadFinished(Loader<List> loader, List data) {
- // Set the new data in the mAdapter.
- mPhotoListItem.clear();
- for(int i = 0; i < data.size();i++){
- PhotoItem item = data.get(i);
- mPhotoListItem.add(item);
- }
- mAdapter.notifyDataSetChanged();
- resolveEmptyText();
- cancelProgressDialog();
- }
其中,PhotoItem(用作給 Adapter 存儲(chǔ)數(shù)據(jù))的數(shù)組包含了指向所有圖庫(kù)中圖片的縮略圖以及全尺寸圖片的URL。一旦獲取這些數(shù)據(jù),Adapter必定會(huì)通過(guò) “notifyDataSetChanged” 回調(diào)來(lái)通知出去,從而刷新當(dāng)前的圖片列表。
通過(guò)游標(biāo)(cursor)來(lái)獲取縮略圖
之前我提到過(guò),我已經(jīng)提供了一個(gè)工具類可以用游標(biāo)方便的去獲取圖庫(kù)中圖片的縮略圖,這個(gè)工具類叫做“PhotoGalleryImageProvider”。這個(gè)類的主要用法如下:
- /**
- * Fetch both full sized images and thumbnails via a single query.
- * Returns all images not in the Camera Roll.
- * @param context
- * @return
- */
- public static List getAlbumThumbnails(Context context){
- final String[] projection = {MediaStore.Images.Thumbnails.DATA,MediaStore.Images.Thumbnails.IMAGE_ID};
- Cursor thumbnailsCursor = context.getContentResolver().query( MediaStore.Images.Thumbnails.EXTERNAL_CONTENT_URI,
- projection, // Which columns to return
- null, // Return all rows
- null,
- null);
- ...
- return result;
- }
直接用Android 提供的Cursors 去后臺(tái)獲取圖片是比較簡(jiǎn)單的一種用法。另一種高級(jí)的用法則是使用 CursorLoader 來(lái)操作Cursor。CursorLoader 內(nèi)建了一個(gè) AsyncTaskLoader 可以用來(lái)自動(dòng)處理后臺(tái)的加載進(jìn)程。由于需要同時(shí)渲染獲取到圖片的縮略圖和完整尺寸的圖片,所以盡管使用帶有CursorLoader的AsyncTask 也可以得到同樣的結(jié)果,但我還是選擇了寫(xiě)一個(gè)自定義的Task Loader。
除了這些必須了解的內(nèi)容,你現(xiàn)在可以獲取范例代碼然后好好享受學(xué)習(xí)的旅程了!
相關(guān)連接
- List all camera images (用作分部創(chuàng)建cursor loader類)