1 /* ownCloud Android client application
2 * Copyright (C) 2012-2014 ownCloud Inc.
4 * This program is free software: you can redistribute it and/or modify
5 * it under the terms of the GNU General Public License version 2,
6 * as published by the Free Software Foundation.
8 * This program is distributed in the hope that it will be useful,
9 * but WITHOUT ANY WARRANTY; without even the implied warranty of
10 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11 * GNU General Public License for more details.
13 * You should have received a copy of the GNU General Public License
14 * along with this program. If not, see <http://www.gnu.org/licenses/>.
17 package com
.owncloud
.android
.utils
;
19 import com
.owncloud
.android
.lib
.common
.utils
.Log_OC
;
21 import android
.graphics
.Bitmap
;
22 import android
.graphics
.BitmapFactory
;
23 import android
.graphics
.Matrix
;
24 import android
.graphics
.BitmapFactory
.Options
;
25 import android
.media
.ExifInterface
;
28 * Utility class with methods for decoding Bitmaps.
30 * @author David A. Velasco
32 public class BitmapUtils
{
36 * Decodes a bitmap from a file containing it minimizing the memory use, known that the bitmap
37 * will be drawn in a surface of reqWidth x reqHeight
39 * @param srcPath Absolute path to the file containing the image.
40 * @param reqWidth Width of the surface where the Bitmap will be drawn on, in pixels.
41 * @param reqHeight Height of the surface where the Bitmap will be drawn on, in pixels.
44 public static Bitmap
decodeSampledBitmapFromFile(String srcPath
, int reqWidth
, int reqHeight
) {
46 // set desired options that will affect the size of the bitmap
47 final Options options
= new Options();
48 options
.inScaled
= true
;
49 options
.inPurgeable
= true
;
50 if (android
.os
.Build
.VERSION
.SDK_INT
>= android
.os
.Build
.VERSION_CODES
.GINGERBREAD_MR1
) {
51 options
.inPreferQualityOverSpeed
= false
;
53 if (android
.os
.Build
.VERSION
.SDK_INT
>= android
.os
.Build
.VERSION_CODES
.HONEYCOMB
) {
54 options
.inMutable
= false
;
57 // make a false load of the bitmap to get its dimensions
58 options
.inJustDecodeBounds
= true
;
60 BitmapFactory
.decodeFile(srcPath
, options
);
62 // calculate factor to subsample the bitmap
63 options
.inSampleSize
= calculateSampleFactor(options
, reqWidth
, reqHeight
);
65 // decode bitmap with inSampleSize set
66 options
.inJustDecodeBounds
= false
;
67 return BitmapFactory
.decodeFile(srcPath
, options
);
73 * Calculates a proper value for options.inSampleSize in order to decode a Bitmap minimizing
74 * the memory overload and covering a target surface of reqWidth x reqHeight if the original
75 * image is big enough.
77 * @param options Bitmap decoding options; options.outHeight and options.inHeight should
79 * @param reqWidth Width of the surface where the Bitmap will be drawn on, in pixels.
80 * @param reqHeight Height of the surface where the Bitmap will be drawn on, in pixels.
81 * @return The largest inSampleSize value that is a power of 2 and keeps both
82 * height and width larger than reqWidth and reqHeight.
84 private static int calculateSampleFactor(Options options
, int reqWidth
, int reqHeight
) {
86 final int height
= options
.outHeight
;
87 final int width
= options
.outWidth
;
90 if (height
> reqHeight
|| width
> reqWidth
) {
91 final int halfHeight
= height
/ 2;
92 final int halfWidth
= width
/ 2;
94 while ((halfHeight
/ inSampleSize
) > reqHeight
95 && (halfWidth
/ inSampleSize
) > reqWidth
) {
104 * Rotate bitmap according to EXIF orientation.
105 * Cf. http://www.daveperrett.com/articles/2012/07/28/exif-orientation-handling-is-a-ghetto/
106 * @param bitmap Bitmap to be rotated
107 * @param storagePath Path to source file of bitmap. Needed for EXIF information.
108 * @return correctly EXIF-rotated bitmap
110 public static Bitmap
rotateImage(Bitmap bitmap
, String storagePath
){
111 Bitmap resultBitmap
= bitmap
;
115 ExifInterface exifInterface
= new ExifInterface(storagePath
);
116 int orientation
= exifInterface
.getAttributeInt(ExifInterface
.TAG_ORIENTATION
, 1);
118 Matrix matrix
= new Matrix();
123 if (orientation
== ExifInterface
.ORIENTATION_FLIP_HORIZONTAL
)
125 matrix
.postScale(-1.0f
, 1.0f
);
128 else if (orientation
== ExifInterface
.ORIENTATION_ROTATE_180
)
130 matrix
.postRotate(180);
133 else if (orientation
== ExifInterface
.ORIENTATION_FLIP_VERTICAL
)
135 matrix
.postScale(1.0f
, -1.0f
);
138 else if (orientation
== ExifInterface
.ORIENTATION_TRANSPOSE
)
140 matrix
.postRotate(-90);
141 matrix
.postScale(1.0f
, -1.0f
);
144 else if (orientation
== ExifInterface
.ORIENTATION_ROTATE_90
)
146 matrix
.postRotate(90);
149 else if (orientation
== ExifInterface
.ORIENTATION_TRANSVERSE
)
151 matrix
.postRotate(90);
152 matrix
.postScale(1.0f
, -1.0f
);
155 else if (orientation
== ExifInterface
.ORIENTATION_ROTATE_270
)
157 matrix
.postRotate(270);
161 resultBitmap
= Bitmap
.createBitmap(bitmap
, 0, 0, bitmap
.getWidth(), bitmap
.getHeight(), matrix
, true
);
162 if (resultBitmap
!= bitmap
) {
166 catch (Exception exception
)
168 Log_OC
.e("BitmapUtil", "Could not rotate the image: " + storagePath
);
174 * Converts an HSL color value to RGB. Conversion formula
175 * adapted from http://en.wikipedia.org/wiki/HSL_color_space.
176 * Assumes h, s, and l are contained in the set [0, 1] and
177 * returns r, g, and b in the set [0, 255].
178 * from: http://axonflux.com/handy-rgb-to-hsl-and-rgb-to-hsv-color-model-c
180 * @param integer h The hue
181 * @param Integer s The saturation
182 * @param Integer l The lightness
183 * @return Array The RGB representation
185 public static int[] hslToRgb(Double h
, Double s
, Double l
){
189 r
= g
= b
= l
; // achromatic
191 Double q
= l
< 0.5 ? l
* (1 + s
) : l
+ s
- l
* s
;
192 Double p
= 2 * l
- q
;
193 r
= hue2rgb(p
, q
, h
+ 1/3) * 255;
194 g
= hue2rgb(p
, q
, h
) * 255;
195 b
= hue2rgb(p
, q
, h
- 1/3) * 255;
199 int[] array
= {r
.intValue(), g
.intValue(), b
.intValue()};
203 private static Double
hue2rgb(Double p
, Double q
, Double t
){
206 if(t
< 1/6) return p
+ (q
- p
) * 6 * t
;
207 if(t
< 1/2) return q
;
208 if(t
< 2/3) return p
+ (q
- p
) * (2/3 - t
) * 6;