Android Open Source - WebImage-Android Web Image View






From Project

Back to project page WebImage-Android.

License

The source code is released under:

Copyright (c) 2011 Bohemian Wrappsody AB Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in ...

If you think the Android project WebImage-Android listed in this page is inappropriate, such as containing malicious code/tools or violating the copyright, please email info at java2s dot com, thanks.

Java Source Code

/*
 * Copyright (c) 2011 Bohemian Wrappsody AB
 *//from  w  w w  .  ja  v a 2 s.  c o m
 * Permission is hereby granted, free of charge, to any person obtaining a copy of
 * this software and associated documentation files (the "Software"), to deal in
 * the Software without restriction, including without limitation the rights to
 * use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
 * the Software, and to permit persons to whom the Software is furnished to do so,
 * subject to the following conditions:
 *
 * The above copyright notice and this permission notice shall be included in all
 * copies or substantial portions of the Software.
 *
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
 * FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL THE AUTHORS OR
 * COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
 * IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
 * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
 */

package com.wrapp.android.webimage;

import android.content.Context;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.graphics.drawable.Drawable;
import android.os.Handler;
import android.util.AttributeSet;
import android.widget.ImageView;

import java.net.MalformedURLException;
import java.net.URL;

/**
 * ImageView successor class which can load images asynchronously from the web. This class
 * is safe to use in ListAdapters or views which may trigger many simultaneous requests.
 */
public class WebImageView extends ImageView implements ImageRequest.Listener {
  Handler uiHandler;
  private Listener listener;
  // Save both a Drawable and int here. If the user wants to pass a resource ID, we can load
  // this lazily and save a bit of memory.
  private Drawable errorImage;
  private int errorImageResId;
  private Drawable placeholderImage;
  private int placeholderImageResId;
  private URL loadedImageUrl;
  private URL pendingImageUrl;

  private enum States {
    EMPTY,
    LOADING,
    LOADED,
    RELOADING,
    CANCELLED,
    ERROR,
  }
  private States currentState = States.EMPTY;

  public interface Listener {
    public void onImageLoadStarted();
    public void onImageLoadComplete();
    public void onImageLoadError();
    public void onImageLoadCancelled();
  }

  @SuppressWarnings({"UnusedDeclaration"})
  public WebImageView(Context context) {
    super(context);
    initialize();
  }

  @SuppressWarnings({"UnusedDeclaration"})
  public WebImageView(Context context, AttributeSet attrs) {
    super(context, attrs);
    initialize();
  }

  @SuppressWarnings({"UnusedDeclaration"})
  public WebImageView(Context context, AttributeSet attrs, int defStyle) {
    super(context, attrs, defStyle);
    initialize();
  }

  private void initialize() {
    uiHandler = new Handler();
  }

  /**
   * Set a listener to be informed about events from this view. If this is not set, then no events are sent.
   * @param listener Listener
   */
  public void setListener(Listener listener) {
    this.listener = listener;
  }

  /**
   * Load an image asynchronously from the web
   * @param imageUrlString Image URL to download image from
   */
  @SuppressWarnings("UnusedDeclaration")
  public void setImageUrl(String imageUrlString) {
    try {
      setImageUrl(new URL(imageUrlString));
    }
    catch(MalformedURLException e) {
      LogWrapper.logException(e);
    }
  }

  /**
   * Load an image asynchronously from the web
   * @param imageUrl Image URL to download image from
   */
  @SuppressWarnings("UnusedDeclaration")
  public void setImageUrl(URL imageUrl) {
    //noinspection NullableProblems
    setImageUrl(imageUrl, null, 0, 0);
  }

  /**
   * Load an image asynchronously from the web
   * @param imageUrl Image URL to download image from
   * @param options Options to use when loading the image. See the documentation for {@link BitmapFactory.Options}
   * for more details. Can be null.
   * @param errorImageResId Resource ID  to be displayed in case the image could not be loaded. If 0, no new image
   * will be displayed on error.
   * @param placeholderImageResId Resource ID to set for placeholder image while image is loading.
   */
  public void setImageUrl(URL imageUrl, BitmapFactory.Options options, int errorImageResId, int placeholderImageResId) {
    if(imageUrl == null) {
      return;
    }
    else if(currentState == States.LOADED && imageUrl.equals(loadedImageUrl)) {
      return;
    }
    else if(currentState == States.LOADING && imageUrl.equals(pendingImageUrl)) {
      return;
    }

    currentState = States.LOADING;
    this.errorImageResId = errorImageResId;
    if(this.placeholderImageResId > 0) {
      setImageResource(this.placeholderImageResId);
    }
    else if(this.placeholderImage != null) {
      setImageDrawable(this.placeholderImage);
    }
    else if(placeholderImageResId > 0) {
      setImageResource(placeholderImageResId);
    }
    if(this.listener != null) {
      listener.onImageLoadStarted();
    }
    pendingImageUrl = imageUrl;
    ImageLoader.load(getContext(), imageUrl, this, options);
  }

  /**
   * This method is called when the drawable has been downloaded (or retreived from cache) and is
   * ready to be displayed. If you override this class, then you should not call this method via
   * super.onBitmapLoaded(). Instead, handle the drawable as necessary (ie, resizing or other
   * transformations), and then call postToGuiThread() to display the image from the correct thread.
   *
   * If you only need a callback to be notified about the drawable being loaded to update other
   * GUI elements and whatnot, then you should override onImageLoaded() instead.
   *
   * @param response Request response
   */
  public void onBitmapLoaded(final RequestResponse response) {
    if(response.originalRequest.imageUrl.equals(pendingImageUrl)) {
      postToGuiThread(new Runnable() {
        public void run() {
          final Bitmap bitmap = response.bitmapReference.get();
          if(bitmap != null) {
            setImageBitmap(bitmap);
            currentState = States.LOADED;
            loadedImageUrl = response.originalRequest.imageUrl;
            pendingImageUrl = null;
            if(listener != null) {
              listener.onImageLoadComplete();
            }
          }
          else {
            // The garbage collecter has cleaned up this bitmap by now (yes, that does happen), so re-issue the request
            ImageLoader.load(getContext(), response.originalRequest.imageUrl, response.originalRequest.listener, response.originalRequest.loadOptions);
            currentState = States.RELOADING;
          }
        }
      });
    }
    else {
      if(listener != null) {
        listener.onImageLoadCancelled();
      }
      currentState = States.CANCELLED;
    }
  }

  /**
   * This method is called if the drawable could not be loaded for any reason. If you need a callback
   * to react to these events, you should override onImageError() instead.
   * @param message Error message (non-localized)
   */
  public void onBitmapLoadError(String message) {
    currentState = States.ERROR;
    LogWrapper.logMessage(message);
    postToGuiThread(new Runnable() {
      public void run() {
        // In case of error, lazily load the drawable here
        if(errorImageResId > 0) {
          errorImage = getResources().getDrawable(errorImageResId);
        }
        if(errorImage != null) {
          setImageDrawable(errorImage);
        }
      }
    });
    if(listener != null) {
      listener.onImageLoadError();
    }
  }

  @Override
  protected void onWindowVisibilityChanged(int visibility) {
    super.onWindowVisibilityChanged(visibility);
    if(visibility == VISIBLE && currentState == States.LOADING) {
      setImageUrl(pendingImageUrl);
    }
  }

  /**
   * Called when the URL which the caller asked to load was cancelled. This can happen for a number
   * of reasons, including the activity being closed or scrolling rapidly in a ListView. For this
   * reason it is recommended not to do so much work in this method.
   */
  public void onBitmapLoadCancelled() {
    currentState = States.CANCELLED;
    if(listener != null) {
      listener.onImageLoadCancelled();
    }
  }

  /**
   * Post a message to the GUI thread. This should be used for updating the component from
   * background tasks.
   * @param runnable Runnable task
   */
  public final void postToGuiThread(Runnable runnable) {
    uiHandler.post(runnable);
  }

  @SuppressWarnings("UnusedDeclaration")
  public void setErrorImageResId(int errorImageResId) {
    this.errorImageResId = errorImageResId;
  }

  @SuppressWarnings("UnusedDeclaration")
  public void setErrorImage(Drawable errorImage) {
    this.errorImage = errorImage;
  }

  @SuppressWarnings("UnusedDeclaration")
  public void setPlaceholderImage(Drawable placeholderImage) {
    this.placeholderImage = placeholderImage;
  }

  @SuppressWarnings("UnusedDeclaration")
  public void setPlaceholderImageResId(int placeholderImageResId) {
    this.placeholderImageResId = placeholderImageResId;
  }
}




Java Source Code List

com.wrapp.android.webimage.CheckTimestampThread.java
com.wrapp.android.webimage.DownloadThreadPool.java
com.wrapp.android.webimage.DownloadThread.java
com.wrapp.android.webimage.FileLoaderThread.java
com.wrapp.android.webimage.ImageCache.java
com.wrapp.android.webimage.ImageDownloader.java
com.wrapp.android.webimage.ImageLoader.java
com.wrapp.android.webimage.ImageRequest.java
com.wrapp.android.webimage.LogWrapper.java
com.wrapp.android.webimage.RequestResponse.java
com.wrapp.android.webimage.RequestRouterThread.java
com.wrapp.android.webimage.TaskQueueThread.java
com.wrapp.android.webimage.WebImageView.java
com.wrapp.android.webimage.WebImage.java
com.wrapp.android.webimagelist.WebImageContainerView.java
com.wrapp.android.webimagelist.WebImageListActivity.java
com.wrapp.android.webimagelist.WebImageListAdapter.java