У меня изображение большого размера. Во время выполнения я хочу прочитать изображение из хранилища и масштабировать его, чтобы уменьшить его вес и размер, и я могу использовать его как миниатюру. Когда пользователь нажимает на миниатюру, я хочу отобразить полноразмерное изображение.
Android, как создать миниатюру времени выполнения
Ответ 1
Мое решение
byte[] imageData = null;
try
{
final int THUMBNAIL_SIZE = 64;
FileInputStream fis = new FileInputStream(fileName);
Bitmap imageBitmap = BitmapFactory.decodeStream(fis);
imageBitmap = Bitmap.createScaledBitmap(imageBitmap, THUMBNAIL_SIZE, THUMBNAIL_SIZE, false);
ByteArrayOutputStream baos = new ByteArrayOutputStream();
imageBitmap.compress(Bitmap.CompressFormat.JPEG, 100, baos);
imageData = baos.toByteArray();
}
catch(Exception ex) {
}
Ответ 2
Попробуйте это
Bitmap ThumbImage = ThumbnailUtils.extractThumbnail(BitmapFactory.decodeFile(imagePath), THUMBSIZE, THUMBSIZE);
Эта утилита доступна из API_LEVEl 8. [Источник]
Ответ 3
Лучшее решение, которое я нашел, заключается в следующем. По сравнению с другими решениями этому не нужно загружать полное изображение для создания миниатюры, так что это более эффективно! Его ограничение состоит в том, что вы не можете иметь миниатюру с точной шириной и высотой, но решение должно быть как можно ближе.
File file = ...; // the image file
Options bitmapOptions = new Options();
bitmapOptions.inJustDecodeBounds = true; // obtain the size of the image, without loading it in memory
BitmapFactory.decodeFile(file.getAbsolutePath(), bitmapOptions);
// find the best scaling factor for the desired dimensions
int desiredWidth = 400;
int desiredHeight = 300;
float widthScale = (float)bitmapOptions.outWidth/desiredWidth;
float heightScale = (float)bitmapOptions.outHeight/desiredHeight;
float scale = Math.min(widthScale, heightScale);
int sampleSize = 1;
while (sampleSize < scale) {
sampleSize *= 2;
}
bitmapOptions.inSampleSize = sampleSize; // this value must be a power of 2,
// this is why you can not have an image scaled as you would like
bitmapOptions.inJustDecodeBounds = false; // now we want to load the image
// Let load just the part of the image necessary for creating the thumbnail, not the whole image
Bitmap thumbnail = BitmapFactory.decodeFile(file.getAbsolutePath(), bitmapOptions);
// Save the thumbnail
File thumbnailFile = ...;
FileOutputStream fos = new FileOutputStream(thumbnailFile);
thumbnail.compress(Bitmap.CompressFormat.JPEG, 90, fos);
fos.flush();
fos.close();
// Use the thumbail on an ImageView or recycle it!
thumbnail.recycle();
Ответ 4
Вот более полное решение для уменьшения размера растрового изображения в миниатюре. Он расширяется в решении Bitmap.createScaledBitmap, поддерживая пропорции изображений, а также дополняя их одинаковой шириной, чтобы они хорошо выглядели в ListView.
Кроме того, было бы лучше всего сделать это масштабирование один раз и сохранить полученный Bitmap как blob в вашей базе данных Sqlite. Я включил фрагмент кода о том, как преобразовать Bitmap в массив байтов для этой цели.
public static final int THUMBNAIL_HEIGHT = 48;
public static final int THUMBNAIL_WIDTH = 66;
imageBitmap = BitmapFactory.decodeByteArray(mImageData, 0, mImageData.length);
Float width = new Float(imageBitmap.getWidth());
Float height = new Float(imageBitmap.getHeight());
Float ratio = width/height;
imageBitmap = Bitmap.createScaledBitmap(imageBitmap, (int)(THUMBNAIL_HEIGHT*ratio), THUMBNAIL_HEIGHT, false);
int padding = (THUMBNAIL_WIDTH - imageBitmap.getWidth())/2;
imageView.setPadding(padding, 0, padding, 0);
imageView.setImageBitmap(imageBitmap);
ByteArrayOutputStream baos = new ByteArrayOutputStream();
imageBitmap.compress(Bitmap.CompressFormat.PNG, 100, baos);
byte[] byteArray = baos.toByteArray();
Ответ 5
Используйте BitmapFactory.decodeFile(...)
, чтобы получить свой объект Bitmap
и установите его в ImageView
с помощью ImageView.setImageBitmap()
.
На ImageView
установите размеры макета на что-то маленькое, например:
android:layout_width="66dip" android:layout_height="48dip"
Добавьте onClickListener
в ImageView
и запустите новое действие, где вы показываете изображение в полном размере с помощью
android:layout_width="wrap_content" android:layout_height="wrap_content"
или укажите более крупный размер.
Ответ 6
/**
* Creates a centered bitmap of the desired size.
*
* @param source original bitmap source
* @param width targeted width
* @param height targeted height
* @param options options used during thumbnail extraction
*/
public static Bitmap extractThumbnail(
Bitmap source, int width, int height, int options) {
if (source == null) {
return null;
}
float scale;
if (source.getWidth() < source.getHeight()) {
scale = width / (float) source.getWidth();
} else {
scale = height / (float) source.getHeight();
}
Matrix matrix = new Matrix();
matrix.setScale(scale, scale);
Bitmap thumbnail = transform(matrix, source, width, height,
OPTIONS_SCALE_UP | options);
return thumbnail;
}
Ответ 7
Я нашел простой способ сделать это
Bitmap thumbnail = ThumbnailUtils.extractThumbnail(BitmapFactory.decodeFile(mPath),200,200)
Синтаксис
Bitmap thumbnail = ThumbnailUtils.extractThumbnail(Bitmap source,int width,int height)
ИЛИ
использовать зависимость Пикассо
compile 'com.squareup.picasso: picasso: 2.5.2'
Picasso.with(context)
.load("file:///android_asset/DvpvklR.png")
.resize(50, 50)
.into(imageView2);
Ссылка Picasso
Ответ 8
Если вы хотите получить качественный результат, используйте библиотеку [RapidDecoder] [1]. Это просто:
import rapid.decoder.BitmapDecoder;
...
Bitmap bitmap = BitmapDecoder.from(getResources(), R.drawable.image)
.scale(width, height)
.useBuiltInDecoder(true)
.decode();
Не забудьте использовать встроенный декодер, если вы хотите уменьшить масштаб менее 50% и результат HQ.
Ответ 9
Этот ответ основан на решении, представленном в https://developer.android.com/topic/performance/graphics/load-bitmap.html (без использования внешних библиотек) с некоторыми изменениями, сделанными мной, чтобы сделать его функциональность лучше и практичнее.
Некоторые примечания об этом решении:
-
Предполагается, что вы хотите сохранить соотношение сторон. Другими словами:
finalWidth / finalHeight == sourceBitmap.getWidth() / sourceBitmap.getWidth()
(Независимо от проблем с литьем и округлением) -
Предполагается, что у вас есть два значения (
maxWidth
иmaxHeight
), которые вы хотите, чтобы любой размер вашего окончательного растрового изображения не превышал его соответствующего значения. Другими словами:finalWidth <= maxWidth && finalHeight <= maxHeight
Итак,
minRatio
был помещен в основу расчетов (см. реализацию). UNLIKE основное решение, которое положилоmaxRatio
в качестве основы вычислений в действительности. Кроме того, расчетinSampleSize
был намного лучше (более логичный, короткий и эффективный). -
Предполагается, что , который вы хотите (по крайней мере), чтобы один из окончательных измерений имел точно значение своего соответствующего maxValue (каждый из них был возможен, учитывая вышеизложенное предположения)суб > . Другими словами:
finalWidth == maxWidth || finalHeight == maxHeight
Последний дополнительный шаг в сравнении с основным решением (
Bitmap.createScaledBitmap(...)
) для этого "точно" ограничения. Очень важно отметить, что вам не следует сначала делать этот шаг (например, принятый ответ) из-за его значительного потребления памяти в случай огромных изображений! -
Он предназначен для декодирования a
file
. Вы можете изменить его, как базовое решение для декодированияresource
(или всего, что поддерживаетBitmapFactory
).
Реализация:
public static Bitmap decodeSampledBitmap(String pathName, int maxWidth, int maxHeight) {
// First decode with inJustDecodeBounds=true to check dimensions
final BitmapFactory.Options options = new BitmapFactory.Options();
options.inJustDecodeBounds = true;
BitmapFactory.decodeFile(pathName, options);
final float wRatio_inv = (float) options.outWidth / maxWidth,
hRatio_inv = (float) options.outHeight / maxHeight; // Working with inverse ratios is more comfortable
final int finalW, finalH, minRatio_inv /* = max{Ratio_inv} */;
if (wRatio_inv > hRatio_inv) {
minRatio_inv = (int) wRatio_inv;
finalW = maxWidth;
finalH = Math.round(options.outHeight / wRatio_inv);
} else {
minRatio_inv = (int) hRatio_inv;
finalH = maxHeight;
finalW = Math.round(options.outWidth / hRatio_inv);
}
options.inSampleSize = pow2Ceil(minRatio_inv); // pow2Ceil: A utility function that comes later
options.inJustDecodeBounds = false; // Decode bitmap with inSampleSize set
return Bitmap.createScaledBitmap(BitmapFactory.decodeFile(pathName, options),
finalW, finalH, true);
}
/**
* @return the largest power of 2 that is smaller than or equal to number.
* WARNING: return {0b1000000...000} for ZERO input.
*/
public static int pow2Ceil(int number) {
return 1 << -(Integer.numberOfLeadingZeros(number) + 1); // is equivalent to:
// return Integer.rotateRight(1, Integer.numberOfLeadingZeros(number) + 1);
}
Пример использования, если у вас есть imageView
с определенным значением для layout_width
(match_parent
или явным значением) и неопределенным значением для layout_height
(wrap_content
) и вместо этого определено значение для maxHeight
:
imageView.setImageBitmap(decodeSampledBitmap(filePath,
imageView.getWidth(), imageView.getMaxHeight()));