Android Open Source - android-http Image Processor






From Project

Back to project page android-http.

License

The source code is released under:

Apache License

If you think the Android project android-http 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) 2012, 2013 the diamond:dogs|group
 */* w ww.  j  av  a 2s  .  co m*/
 * 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 at.diamonddogs.service.processor;

import java.io.File;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.net.URL;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import android.content.Context;
import android.graphics.Bitmap;
import android.graphics.Bitmap.CompressFormat;
import android.graphics.BitmapFactory;
import android.os.Bundle;
import android.os.Handler;
import android.os.Message;
import android.view.animation.Animation;
import android.view.animation.AnimationUtils;
import android.widget.ImageView;
import at.diamonddogs.data.adapter.ReplyAdapter;
import at.diamonddogs.data.adapter.ReplyAdapter.Status;
import at.diamonddogs.data.dataobjects.CacheInformation;
import at.diamonddogs.data.dataobjects.Request;
import at.diamonddogs.data.dataobjects.WebRequest;
import at.diamonddogs.exception.ProcessorExeception;
import at.diamonddogs.util.CacheManager;
import at.diamonddogs.util.CacheManager.CachedObject;
import at.diamonddogs.util.Utils;

/**
 * This processor should be used if the data returned by the {@link WebRequest}
 * contains an image. This {@link ServiceProcessor} handles caching and handles
 * image creation.
 */
public class ImageProcessor extends DataProcessor<Bitmap, Bitmap> {

  /**
   * The processor id
   */
  public static final int ID = 13452;

  /**
   * {@link Bundle} key that is used to store the image
   */
  public static final String BUNDLE_EXTRA_BITMAP = "BUNDLE_EXTRA_BITMAP";

  private static final Logger LOGGER = LoggerFactory.getLogger(ImageProcessor.class);

  private BitmapFactory.Options bitmapOptions;

  /**
   * Default constructor using default {@link BitmapFactory.Options}
   */
  public ImageProcessor() {
    super();
    this.bitmapOptions = new BitmapFactory.Options();
  }

  /**
   * Constructor for providing {@link BitmapFactory.Options}
   * 
   * @param bitmapOptions
   *            custom {@link BitmapFactory.Options}
   */
  public ImageProcessor(BitmapFactory.Options bitmapOptions) {
    super();
    this.bitmapOptions = bitmapOptions;
  }

  /**
   * Determines if the request should be stored in memory, might cause OOM
   * errors if the image is very large
   */
  protected boolean useMemCache = true;

  public void setUseMemCache(boolean useMemCache) {
    this.useMemCache = useMemCache;
  }

  @Override
  public void processWebReply(Context c, ReplyAdapter r, Handler handler) {
    LOGGER.info("processing imagerequest from: " + r.getRequest().getUrl());
    if (r.getStatus() == Status.OK) {
      ProcessingData<Bitmap> processingData = processData(r);
      Bitmap b = processingData.output;
      saveBitmapToFile(c, r, handler, b);
    } else if (r.getStatus() == Status.FAILED) {
      handler.sendMessage(createErrorMessage(r.getThrowable(), r));
    }
  }

  private void saveBitmapToFile(Context c, ReplyAdapter r, Handler handler, Bitmap b) {
    try {
      saveBitmapToFile(c, r, b);
      handler.sendMessage(createReturnMessage(r, b));
    } catch (FileNotFoundException e) {
      LOGGER.error(e.getMessage(), e);
      handler.sendMessage(createErrorMessage(e, r));
    }
  }

  private void saveBitmapToFile(Context c, ReplyAdapter r, Bitmap b) throws FileNotFoundException {
    WebRequest request = (WebRequest) r.getRequest();
    String filename = Utils.getMD5Hash(request.getUrl().toString());
    if (filename != null && b != null) {
      if (request.getCacheTime() != CacheInformation.CACHE_NO) {
        File path = Utils.getCacheDir(c);
        FileOutputStream fos = new FileOutputStream(new File(path, filename));
        b.compress(CompressFormat.PNG, 0, fos);

        CacheInformation ci = createImage(request, path.toString(), filename);

        CacheManager cm = CacheManager.getInstance();
        cm.addToCache(c, ci);
        if (useMemCache) {
          cm.addToMemoryCache(request.getUrl().toString(), ID, b);
        }
      }
    }
  }

  @Override
  public Bitmap obtainDataObjectFromWebReply(Context c, ReplyAdapter reply) {
    Bitmap b = super.obtainDataObjectFromWebReply(c, reply);
    try {
      saveBitmapToFile(c, reply, b);
    } catch (Throwable tr) {
      LOGGER.warn("Could not cache bitmap (sync webrequest)", tr);
    }
    return b;
  }

  /**
   * Obtains a {@link Bitmap} from cache.
   * 
   * @param c
   *            a {@link Context}
   * @param object
   *            the {@link CachedObject} obtained from the cache database
   * @return returns the {@link Bitmap}, either from file or from memory cache
   *         or <code>null</code> if something goes awefully wrong
   */
  @Override
  public Bitmap obtainDataObjectFromCachedObject(Context c, WebRequest wr, CachedObject object) {
    switch (object.getFrom()) {
    case MEMORY:
      return (Bitmap) object.getCachedObject();
    case FILE:
      byte[] data = (byte[]) object.getCachedObject();
      return BitmapFactory.decodeByteArray(data, 0, data.length, bitmapOptions);
    default:
      return null;
    }
  }

  @Override
  public void processCachedObject(CachedObject cachedObject, Handler handler, Request request) {
    WebRequest wr = (WebRequest) request;
    switch (cachedObject.getFrom()) {
    case MEMORY:
      LOGGER.info("processing image from memory cache");
      processMemCache(cachedObject.getCachedObject(), handler, wr);
      break;
    case FILE:
      LOGGER.info("processing image from file cache");
      processFileCache((byte[]) cachedObject.getCachedObject(), handler, wr);
      break;
    }
  }

  private void processFileCache(byte[] data, Handler handler, WebRequest webRequest) throws IllegalArgumentException {
    Bitmap b = BitmapFactory.decodeByteArray(data, 0, data.length, bitmapOptions);

    if (b == null) {
      handler.sendMessage(createErrorMessage(new IllegalArgumentException("Couldn't decode Bitmap from data"), webRequest));
      return;
    }
    if (useMemCache) {
      CacheManager.getInstance().addToMemoryCache(webRequest.getUrl().toString(), webRequest.getUrl().toString(), b);
    }
    handler.sendMessage(createReturnMessage(webRequest, b));
  }

  private void processMemCache(Object data, Handler handler, WebRequest webRequest) throws IllegalArgumentException {
    if (data instanceof Bitmap) {
      Bitmap b = (Bitmap) data;
      handler.sendMessage(createReturnMessage(webRequest, b));
    } else {
      handler.sendMessage(createErrorMessage(new IllegalArgumentException("Data must be a Bitmap object"), webRequest));
    }
  }

  private CacheInformation createImage(WebRequest request, String filePath, String fileName) {
    CacheInformation c = new CacheInformation();
    c.setCacheTime(request.getCacheTime());
    c.setCreationTimeStamp(System.currentTimeMillis());
    c.setFileName(fileName);
    c.setFilePath(filePath);
    c.setUseOfflineCache(true);
    return c;
  }

  /**
   * Returns the absolute path of an image file as stored on the SDCard
   * 
   * @param url
   *            the url of the image file (web url)
   * @param context
   *            a {@link Context}
   * @return the absolute path of the image file on the file system or null if
   *         it does not exist
   */
  public static String getImageFileUrl(String url, Context context) {
    String filename = Utils.getMD5Hash(url);
    File dir = context.getExternalCacheDir();
    File file = new File(dir, filename);
    if (file.exists()) {
      return file.getAbsolutePath();
    }
    return null;
  }

  @Override
  public int getProcessorID() {
    return ID;
  }

  @Override
  protected Bitmap createParsedObjectFromByteArray(byte[] data) {
    if (data == null) {
      return null;
    }
    return BitmapFactory.decodeByteArray(data, 0, data.length, bitmapOptions);

  }

  @Override
  protected Bitmap parse(Bitmap inputObject) {
    return inputObject;
  }

  @Override
  protected Message createReturnMessage(ReplyAdapter replyAdapter, Bitmap payload) {
    Message m = super.createReturnMessage(replyAdapter, payload);
    m.getData().putParcelable(BUNDLE_EXTRA_BITMAP, payload);
    return m;
  }

  @Override
  protected Message createReturnMessage(WebRequest webRequest, Bitmap payload) {
    Message m = super.createReturnMessage(webRequest, payload);
    m.getData().putParcelable(BUNDLE_EXTRA_BITMAP, payload);
    return m;
  }

  /**
   * Constructs a default image {@link WebRequest}
   * 
   * @param url
   * @return
   */
  public static WebRequest getDefaultImageRequest(String url) {
    WebRequest wr = new WebRequest();
    wr.setUrl(url);
    wr.setProcessorId(ID);
    wr.setCacheTime(CacheInformation.CACHE_1MO);
    return wr;
  }

  /**
   * A default {@link Handler} for image {@link WebRequest} will take care of
   * displaying the image on an {@link ImageView}
   */
  public static class ImageProcessHandler extends Handler {
    protected ImageView imageView;
    private String url;
    private Animation fadeInAnimation;
    private boolean useDrawingCache = false;
    private int defaultImage = -1;

    /**
     * Constructor
     * 
     * @param imageView
     *            the {@link ImageView} the image should be displayed on
     * @param url
     *            the of the image, will be set as {@link ImageView} tag in
     *            order to identify the correct {@link ImageView}
     * @param fadeInAnimation
     *            an optional {@link Animation}
     * @param defaultImage
     *            a default image to be displayed if the real image cannot
     *            be displayed
     */
    public ImageProcessHandler(ImageView imageView, String url, Animation fadeInAnimation, int defaultImage) {
      if (imageView == null) {
        throw new IllegalArgumentException("ImageView must not be null");
      }
      this.imageView = imageView;
      this.imageView.setTag(url);
      this.fadeInAnimation = fadeInAnimation;
      this.url = url;
      this.defaultImage = defaultImage;
    }

    /**
     * Convenience constructor for {@link URL} url types
     * 
     * @param imageView
     *            the {@link ImageView} the image should be displayed on
     * @param url
     *            the of the image, will be set as {@link ImageView} tag in
     *            order to identify the correct {@link ImageView}
     * @param fadeInAnimation
     *            an optional {@link Animation}
     */
    public ImageProcessHandler(ImageView imageView, URL url, Animation fadeInAnimation) {
      this(imageView, url.toString(), fadeInAnimation, -1);
    }

    /**
     * 
     * @param imageView
     *            the {@link ImageView} the image should be displayed on
     * @param url
     *            the of the image, will be set as {@link ImageView} tag in
     *            order to identify the correct {@link ImageView}
     * @param fadeInAnimation
     *            an optional {@link Animation}
     * @param defaultImage
     *            a default image to display if the image returned by the
     *            {@link WebRequest} is null
     */
    public ImageProcessHandler(ImageView imageView, URL url, Animation fadeInAnimation, int defaultImage) {
      this(imageView, url.toString(), fadeInAnimation, defaultImage);
    }

    /**
     * Convenience constructor that causes {@link ImageProcessHandler} to
     * use a default animation and {@link URL} type
     * 
     * @param imageView
     *            the {@link ImageView} the image should be displayed on
     * @param url
     *            the of the image, will be set as {@link ImageView} tag in
     *            order to identify the correct {@link ImageView}
     */
    public ImageProcessHandler(ImageView imageView, URL url) {
      this(imageView, url.toString(), AnimationUtils.loadAnimation(imageView.getContext(), android.R.anim.fade_in), -1);
    }

    /**
     * 
     * @param imageView
     *            the {@link ImageView} the image should be displayed on
     * @param url
     *            the of the image, will be set as {@link ImageView} tag in
     *            order to identify the correct {@link ImageView}
     * @param defaultImage
     *            a default image to display if the image returned by the
     *            {@link WebRequest} is null
     */
    public ImageProcessHandler(ImageView imageView, URL url, int defaultImage) {
      this(imageView, url, AnimationUtils.loadAnimation(imageView.getContext(), android.R.anim.fade_in), defaultImage);
    }

    /**
     * Convenience constructor that causes {@link ImageProcessHandler} to
     * use a default animation
     * 
     * @param imageView
     *            the {@link ImageView} the image should be displayed on
     * @param url
     *            the of the image, will be set as {@link ImageView} tag in
     *            order to identify the correct {@link ImageView}
     */
    public ImageProcessHandler(ImageView imageView, String url) {
      this(imageView, url, AnimationUtils.loadAnimation(imageView.getContext(), android.R.anim.fade_in), -1);
    }

    /**
     * 
     * @param imageView
     *            the {@link ImageView} the image should be displayed on
     * @param url
     *            the of the image, will be set as {@link ImageView} tag in
     *            order to identify the correct {@link ImageView}
     * @param defaultImage
     *            a default image to display if the image returned by the
     *            {@link WebRequest} is null
     */
    public ImageProcessHandler(ImageView imageView, String url, int defaultImage) {
      this(imageView, url, AnimationUtils.loadAnimation(imageView.getContext(), android.R.anim.fade_in), defaultImage);
    }

    /**
     * Enables the drawing cache of the {@link ImageView}
     */
    public void enableDrawingCache() {
      useDrawingCache = true;
    }

    /**
     * This method allows {@link Bitmap} processing before the
     * {@link Bitmap} is placed into the {@link ImageView}. This method does
     * not affect the default image to be displayed if not {@link Bitmap} is
     * present.
     * 
     * @param bitmap
     *            the {@link Bitmap} to be processed
     * @return the processed {@link Bitmap}
     */
    public Bitmap postProcessBitmap(Bitmap bitmap) {
      return bitmap;
    }

    @Override
    public void handleMessage(Message msg) {
      if (msg.arg1 == ServiceProcessor.RETURN_MESSAGE_OK) {

        if (imageView == null || imageView.getTag() == null) {
          throw new ProcessorExeception("Tag for ImageView has not been set or ImageView was null");
        }
        LOGGER.info("adding image from url:" + url + " for tag: " + imageView.getTag());
        if (imageView.getTag().equals(url)) {
          Bundle b = msg.getData();
          Bitmap bitmap = b.getParcelable(BUNDLE_EXTRA_BITMAP);

          if (bitmap == null && defaultImage != -1) {
            imageView.setImageResource(defaultImage);
          } else {
            if (bitmap != null) {
              bitmap = postProcessBitmap(bitmap);
            }
            imageView.setImageBitmap(bitmap);
          }

          if (useDrawingCache) {
            imageView.setDrawingCacheEnabled(true);
            imageView.buildDrawingCache(true);
          }
          if (fadeInAnimation != null) {
            imageView.startAnimation(fadeInAnimation);
          }
          imageView.invalidate();
        }
      }
    }
  }
}




Java Source Code List

at.diamonddogs.android.support.v4.util.LruCache.java
at.diamonddogs.builder.WebRequestBuilderConfiguration.java
at.diamonddogs.builder.WebRequestBuilderDefaultConfig.java
at.diamonddogs.builder.WebRequestBuilder.java
at.diamonddogs.contentprovider.AbstractDefaultContentProvider.java
at.diamonddogs.contentprovider.CacheContentProvider.java
at.diamonddogs.data.adapter.ReplyAdapter.java
at.diamonddogs.data.adapter.database.DataBaseAdapterCacheInformation.java
at.diamonddogs.data.adapter.database.DatabaseAdapter.java
at.diamonddogs.data.adapter.database.Query.java
at.diamonddogs.data.adapter.parcelable.ParcelableAdapterTempFile.java
at.diamonddogs.data.adapter.parcelable.ParcelableAdapterWebReply.java
at.diamonddogs.data.adapter.parcelable.ParcelableAdapterWebRequest.java
at.diamonddogs.data.adapter.parcelable.ParcelableAdapter.java
at.diamonddogs.data.adapter.soap.SoapByteArrayAdapter.java
at.diamonddogs.data.adapter.soap.SoapReplyAdapter.java
at.diamonddogs.data.adapter.soap.SoapRequestAdapter.java
at.diamonddogs.data.dataobjects.CacheInformation.java
at.diamonddogs.data.dataobjects.NonTimeCriticalTaskQueueDefaultConfiguration.java
at.diamonddogs.data.dataobjects.NonTimeCriticalTask.java
at.diamonddogs.data.dataobjects.NonTimeCriticalWebRequest.java
at.diamonddogs.data.dataobjects.Reply.java
at.diamonddogs.data.dataobjects.Request.java
at.diamonddogs.data.dataobjects.SoapReply.java
at.diamonddogs.data.dataobjects.SoapRequest.java
at.diamonddogs.data.dataobjects.TempFile.java
at.diamonddogs.data.dataobjects.WebReply.java
at.diamonddogs.data.dataobjects.WebRequest.java
at.diamonddogs.data.parser.JSONArrayProxy.java
at.diamonddogs.data.parser.JSONObjectProxy.java
at.diamonddogs.data.parser.ParserProxy.java
at.diamonddogs.example.http.activity.CachingExampleActivity.java
at.diamonddogs.example.http.activity.HttpExampleActivity.java
at.diamonddogs.example.http.activity.HttpOrderedAsyncAssisiterExampleActivity.java
at.diamonddogs.example.http.activity.HttpServiceAssisterExampleActivity.java
at.diamonddogs.example.http.activity.ImageLoadingExampleListActivity.java
at.diamonddogs.example.http.activity.NonTimeCriticalExampleActivity.java
at.diamonddogs.example.http.activity.StartActivity.java
at.diamonddogs.example.http.dataobject.Example.java
at.diamonddogs.example.http.dataobject.NonTimeCriticalExampleConfiguration.java
at.diamonddogs.example.http.dataobject.Tripple.java
at.diamonddogs.example.http.dataobject.Weather.java
at.diamonddogs.example.http.dataobject.WebComic.java
at.diamonddogs.example.http.factory.NonTimeCriticalExampleConfigFactory.java
at.diamonddogs.example.http.processor.RssProcessor.java
at.diamonddogs.example.http.processor.WeatherProcessor.java
at.diamonddogs.example.http.processor.WebComicProcessor.java
at.diamonddogs.example.http.view.adapter.ImageLoadingExampleAdapter.java
at.diamonddogs.exception.CacheManagerException.java
at.diamonddogs.exception.DatabaseAdapterException.java
at.diamonddogs.exception.ProcessorExeception.java
at.diamonddogs.exception.ServiceException.java
at.diamonddogs.exception.WebClientException.java
at.diamonddogs.net.WebClientDefaultHttpClient.java
at.diamonddogs.net.WebClientFactory.java
at.diamonddogs.net.WebClientHttpURLConnection.java
at.diamonddogs.net.WebClient.java
at.diamonddogs.net.ssl.CustomSSLSocketFactory.java
at.diamonddogs.net.ssl.CustomX509TrustManager.java
at.diamonddogs.net.ssl.SSLHelper.java
at.diamonddogs.nontimecritical.NonTimeCriticalTaskManager.java
at.diamonddogs.nontimecritical.NonTimeCriticalTaskQueueConfigurationDefaultFactory.java
at.diamonddogs.nontimecritical.NonTimeCriticalTaskQueue.java
at.diamonddogs.service.CacheService.java
at.diamonddogs.service.importservice.GenericImportService.java
at.diamonddogs.service.importservice.ImportServiceContract.java
at.diamonddogs.service.importservice.ImportService.java
at.diamonddogs.service.importservice.OrderedImportServiceContract.java
at.diamonddogs.service.importservice.OrderedImportService.java
at.diamonddogs.service.net.HttpOrderedAsyncAssister.java
at.diamonddogs.service.net.HttpServiceAssister.java
at.diamonddogs.service.net.HttpService.java
at.diamonddogs.service.net.ServiceProcessorIdGenerator.java
at.diamonddogs.service.processor.AdjustableImageProcessor.java
at.diamonddogs.service.processor.DataProcessor.java
at.diamonddogs.service.processor.DummyProcessor.java
at.diamonddogs.service.processor.HeadRequestProcessor.java
at.diamonddogs.service.processor.ImageProcessor.java
at.diamonddogs.service.processor.JSONArrayProcessor.java
at.diamonddogs.service.processor.JSONProcessor.java
at.diamonddogs.service.processor.RawDataProcessor.java
at.diamonddogs.service.processor.ServiceProcessorMessageUtil.java
at.diamonddogs.service.processor.ServiceProcessor.java
at.diamonddogs.service.processor.SoapProcessor.java
at.diamonddogs.service.processor.StreamProcessor.java
at.diamonddogs.service.processor.SynchronousProcessor.java
at.diamonddogs.service.processor.SynchronousXmlProcessorNoDom.java
at.diamonddogs.service.processor.XMLProcessorNoDom.java
at.diamonddogs.service.processor.XMLProcessor.java
at.diamonddogs.service.processor.XMLXPathProcessor.java
at.diamonddogs.util.AndroidUtils.java
at.diamonddogs.util.CacheManager.java
at.diamonddogs.util.ConnectivityHelper.java
at.diamonddogs.util.SoapUtil.java
at.diamonddogs.util.Utils.java
at.diamonddogs.util.WorkerQueue.java
org.apache.commons.codec.CharEncoding.java
org.apache.commons.codec.binary.Hex.java
org.apache.commons.codec.binary.StringUtils.java