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()));