使用 Image Loader 和自定義 schema 加載特殊來源的圖片

首先,我們問一個問題。

Image Loader 一般拿來加載哪些圖片?

不管是早期的 Universal-Image-Loader,還是后來的 PicassoGlide 以及 Fresco,它們都是很優秀的圖片加載庫,我們可以很輕松地通過它們來加載各種網絡圖片(http://, https://),甚至是從

  • 應用 Resource(android.resource://
  • 應用 Asset(file://android_asset/
  • 本地文件(file:///
  • 本地多媒體庫(content://

加載圖片。

但是,Image Loader 的作用絕不止于此。

Image Loader 還可以用來加載哪些圖片?

  • 系統已安裝動態壁紙的縮略圖
  • 系統已安裝應用的圖標
  • 系統已安裝應用里的圖片資源
  • 未安裝 apk 文件的圖標
  • 未安裝 apk 文件里的圖片資源
  • 圖片縮略圖
  • 視頻縮略圖
  • 通訊錄聯系人頭像

可以這么說,只要輸出是圖片的地方,都可以用 Image Loader 進行加載。

如何利用 Image Loader 加載這些特殊圖片?

一種通用思路是,既然 Image Loader 根據 URL 來加載各種網絡圖片或本地圖片,而它們的 schema 一般都不一樣,如網絡圖片的 schema 是 httphttps,本地文件的 schema 是 file,那我們可以通過構造自定義 schema 的 URL,來指代不同類型(來源)的圖片,只要配置 Image Loader 使其能從這些自定義 schema 的 URL 中識別并加載圖片,后續就可以像加載普通圖片一樣方便地加載這些特殊圖片了。

具體步驟如下。

1. 設計圖片 URL 的格式

舉個例子,比如我們在開發一個應用管理軟件,需要在列表中顯示系統已安裝應用的圖標,我們可以用 "application://com.elvishew.sampleapp/10" 表示包名為 com.elvishew.sampleapp、版本號為 10 的應用的縮略圖。

又比如我們在開發一個文件管理器,需要顯示 apk 文件的應用圖標,我們可以用 "apk:///sdcard/sample.apk#1470000000000" 表示存放在 /sdcard 路徑下、名為 sample.apk、最后修改時間戳為 1470000000000 的 apk 文件的應用圖標。

以上有個小細節,我們把那些可能會影響加載結果的因素,如應用程序的版本號、apk 文件的最后修改時間戳,加入到 URL 中。這樣一來,一旦這些影響因素變了,URL 也會跟著改變,Image Loader 就會視它們為不同的圖片,而不去使用錯誤的緩存。

2. 配置 Image Loader,讓其可以從自定義格式的 URL 中加載圖片

首先,定義特殊圖片的 schema。

// 系統已安裝應用的圖標 schema
public static final String URL_SCHEMA_APPLICATION = "application"

// apk 文件的應用圖標 schema
public static final String URL_SCHEMA_APK = "apk"

// 省略其他 schema
...

其次,自定義 Downloader(Universal-Image-Loader) 或 RequestHandler(Picasso),在應用初始化時配置 Image Loader。

Universal-Image-Loader

// 自定義 Downloader
private class CustomImageDownloader extends BaseImageDownloader {

  // 省略部分代碼
  ...

  @Override
  protected InputStream getStreamFromOtherSource(String imageUri, Object extra)
      throws IOException {
    InputStream is = null;
    Uri uri = Uri.parse(imageUri);
    if (URL_SCHEMA_APPLICATION.equals(uri.getScheme())) {
      String packageName = uri.getHost();
      is = getApplicationIconInputStream(packageName);
    } else if (URL_SCHEMA_APK.equals(uri.getScheme())) {
      String apkPath = uri.getAuthority();
      is = getApkIconInputStream(apkPath);
    }
    ... // 省略其他分支代碼
    return is;
  }
}

// 配置 Universal-Image-Loader
ImageLoaderConfiguration.Builder configBuilder = new ImageLoaderConfiguration.Builder(context)
      .imageDownloader(new CustomImageDownloader())
      ... // 省略其他配置代碼

ImageLoader.getInstance().init(configBuilder.build());

Picasso

// 自定義 RequestHandler
private class CustomRequestHandler extends RequestHandler {
  @Override
  public boolean canHandleRequest(Request data) {
    return URL_SCHEMA_APPLICATION.equals(data.uri.getScheme())
        || URL_SCHEMA_APK.equals(data.uri.getScheme())
        || ... /* 省略其他 schema 的檢查 */;
  }

  @Override
  public Result load(Request request, int networkPolicy) throws IOException {
    Uri uri = request.uri;
    InputStream is = null;
    if (URL_SCHEMA_APPLICATION.equals(uri.getScheme())) {
      String packageName = uri.getHost();
      int versionCode = uri.getPort();
      is = getApplicationIconInputStream(packageName, versionCode);
    } else if (URL_SCHEMA_APK.equals(uri.getScheme())) {
      String apkPath = uri.getAuthority();
      is = getApkIconInputStream(apkPath);
    }
    ... // 省略其他分支代碼
    return new Result(is, Picasso.LoadedFrom.DISK);
  }
}

// 配置 Picasso
mPicasso = new Picasso.Builder(context)
    .addRequestHandler(new CustomRequestHandler())
    .build();

3. 組裝圖片 URL

以加載 /sdcard/sample.apk 這個 apk 文件的應用圖標為例,假設它的最后修改時間戳為 1470000000000,我們可以組裝得到 URL "apk:///sdcard/sample.apk#1470000000000"

4. 使用組裝的 URL 加載圖片

Universal-Image-Loader

ImageLoader.getInstance().displayImage(uri, imageView);

Picasso

mPicasso.load(uri).into(imageView);

大功告成。

關于作者
ElvisHew 的開源項目: XLog
ElvisHew 的 Github 主頁: https://github.com/elvishew
ElvisHew 的新浪微博:http://weibo.com/elvishew

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。

推薦閱讀更多精彩內容