summaryrefslogtreecommitdiffstats
path: root/src/com/android/photos/data/BitmapDecoder.java
blob: f19808d065713aa4ea777fb7c18b77cf9d5ea876 (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
109
110
111
112
113
114
115
116
117
118
119
120
121
122
/*
 * Copyright (C) 2013 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.photos.data;

import android.graphics.Bitmap;
import android.graphics.Bitmap.Config;
import android.graphics.BitmapFactory;
import android.util.Log;
import android.util.Pools.Pool;
import android.util.Pools.SynchronizedPool;

import com.android.gallery3d.common.Utils;

import java.io.BufferedInputStream;
import java.io.File;
import java.io.IOException;
import java.io.InputStream;

/**
 * BitmapDecoder keeps a pool of temporary storage to reuse for decoding
 * bitmaps. It also simplifies the multi-stage decoding required to efficiently
 * use GalleryBitmapPool. The static methods decode and decodeFile can be used
 * to decode a bitmap from GalleryBitmapPool. The bitmap may be returned
 * directly to GalleryBitmapPool or use the put method here when the bitmap is
 * ready to be recycled.
 */
public class BitmapDecoder {
    private static final String TAG = BitmapDecoder.class.getSimpleName();
    private static final int POOL_SIZE = 4;
    private static final int TEMP_STORAGE_SIZE_BYTES = 16 * 1024;
    private static final int HEADER_MAX_SIZE = 128 * 1024;

    private static final Pool<BitmapFactory.Options> sOptions =
            new SynchronizedPool<BitmapFactory.Options>(POOL_SIZE);

    public static Bitmap decode(InputStream in) {
        BitmapFactory.Options opts = getOptions();
        try {
            if (!in.markSupported()) {
                in = new BufferedInputStream(in);
            }
            opts.inJustDecodeBounds = true;
            in.mark(HEADER_MAX_SIZE);
            BitmapFactory.decodeStream(in, null, opts);
            in.reset();
            opts.inJustDecodeBounds = false;
            GalleryBitmapPool pool = GalleryBitmapPool.getInstance();
            Bitmap reuseBitmap = pool.get(opts.outWidth, opts.outHeight);
            opts.inBitmap = reuseBitmap;
            Bitmap decodedBitmap = BitmapFactory.decodeStream(in, null, opts);
            if (reuseBitmap != null && decodedBitmap != reuseBitmap) {
                pool.put(reuseBitmap);
            }
            return decodedBitmap;
        } catch (IOException e) {
            Log.e(TAG, "Could not decode stream to bitmap", e);
            return null;
        } finally {
            Utils.closeSilently(in);
            release(opts);
        }
    }

    public static Bitmap decode(File in) {
        return decodeFile(in.toString());
    }

    public static Bitmap decodeFile(String in) {
        BitmapFactory.Options opts = getOptions();
        try {
            opts.inJustDecodeBounds = true;
            BitmapFactory.decodeFile(in, opts);
            opts.inJustDecodeBounds = false;
            GalleryBitmapPool pool = GalleryBitmapPool.getInstance();
            Bitmap reuseBitmap = pool.get(opts.outWidth, opts.outHeight);
            opts.inBitmap = reuseBitmap;
            Bitmap decodedBitmap = BitmapFactory.decodeFile(in, opts);
            if (reuseBitmap != null && decodedBitmap != reuseBitmap) {
                pool.put(reuseBitmap);
            }
            return decodedBitmap;
        } finally {
            release(opts);
        }
    }

    public static void put(Bitmap bitmap) {
        GalleryBitmapPool.getInstance().put(bitmap);
    }

    private static BitmapFactory.Options getOptions() {
        BitmapFactory.Options opts = sOptions.acquire();
        if (opts == null) {
            opts = new BitmapFactory.Options();
            opts.inMutable = true;
            opts.inPreferredConfig = Config.ARGB_8888;
            opts.inSampleSize = 1;
            opts.inTempStorage = new byte[TEMP_STORAGE_SIZE_BYTES];
        }

        return opts;
    }

    private static void release(BitmapFactory.Options opts) {
        opts.inBitmap = null;
        opts.inJustDecodeBounds = false;
        sOptions.release(opts);
    }
}