summaryrefslogtreecommitdiffstats
path: root/src/com/android/messaging/datamodel/media/FileImageRequest.java
blob: 31c053a4b2aa9a3e1fd16f17234d6f96b375b04c (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
/*
 * Copyright (C) 2015 The Android Open Source Project
 *
 * 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 com.android.messaging.datamodel.media;

import android.content.Context;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.media.ExifInterface;

import com.android.messaging.datamodel.media.PoolableImageCache.ReusableImageResourcePool;
import com.android.messaging.util.ImageUtils;
import com.android.messaging.util.LogUtil;

import java.io.IOException;

/**
 * Serves file system based image requests. Since file paths can be expressed in Uri form, this
 * extends regular UriImageRequest but performs additional optimizations such as loading thumbnails
 * directly from Exif information.
 */
public class FileImageRequest extends UriImageRequest {
    private final String mPath;
    private final boolean mCanUseThumbnail;

    public FileImageRequest(final Context context,
            final FileImageRequestDescriptor descriptor) {
        super(context, descriptor);
        mPath = descriptor.path;
        mCanUseThumbnail = descriptor.canUseThumbnail;
    }

    @Override
    protected Bitmap loadBitmapInternal()
            throws IOException {
        // Before using the FileInputStream, check if the Exif has a thumbnail that we can use.
        if (mCanUseThumbnail) {
            byte[] thumbnail = null;
            try {
                final ExifInterface exif = new ExifInterface(mPath);
                if (exif.hasThumbnail()) {
                    thumbnail = exif.getThumbnail();
                }
            } catch (final IOException e) {
                // Nothing to do
            }

            if (thumbnail != null) {
                final BitmapFactory.Options options = PoolableImageCache.getBitmapOptionsForPool(
                        false /* scaled */, 0 /* inputDensity */, 0 /* targetDensity */);
                // First, check dimensions of the bitmap.
                options.inJustDecodeBounds = true;
                BitmapFactory.decodeByteArray(thumbnail, 0, thumbnail.length, options);

                // Calculate inSampleSize
                options.inSampleSize = ImageUtils.get().calculateInSampleSize(options,
                        mDescriptor.desiredWidth, mDescriptor.desiredHeight);

                options.inJustDecodeBounds = false;

                // Actually decode the bitmap, optionally using the bitmap pool.
                try {
                    // Get the orientation. We should be able to get the orientation from
                    // the thumbnail itself but at least on some phones, the thumbnail
                    // doesn't have an orientation tag. So use the outer image's orientation
                    // tag and hope for the best.
                    mOrientation = ImageUtils.getOrientation(getInputStreamForResource());
                    if (com.android.messaging.util.exif.ExifInterface.
                            getOrientationParams(mOrientation).invertDimensions) {
                        mDescriptor.updateSourceDimensions(options.outHeight, options.outWidth);
                    } else {
                        mDescriptor.updateSourceDimensions(options.outWidth, options.outHeight);
                    }
                    final ReusableImageResourcePool bitmapPool = getBitmapPool();
                    if (bitmapPool == null) {
                        return BitmapFactory.decodeByteArray(thumbnail, 0, thumbnail.length,
                                options);
                    } else {
                        final int sampledWidth = options.outWidth / options.inSampleSize;
                        final int sampledHeight = options.outHeight / options.inSampleSize;
                        return bitmapPool.decodeByteArray(thumbnail, options, sampledWidth,
                                sampledHeight);
                    }
                } catch (IOException ex) {
                    // If the thumbnail is broken due to IOException, this will
                    // fall back to default bitmap loading.
                    LogUtil.e(LogUtil.BUGLE_IMAGE_TAG, "FileImageRequest: failed to load " +
                            "thumbnail from Exif", ex);
                }
            }
        }

        // Fall back to default InputStream-based loading if no thumbnails could be retrieved.
        return super.loadBitmapInternal();
    }
}