Получение растрового изображения из вектора с возможностью рисования

134

В моем приложении я должен установить большой значок для уведомления. LargeIcon должен быть растровым изображением, а мои чертежи - это векторные изображения (новая функция в Android, см. Эту ссылку ). Проблема в том, что когда я пытаюсь декодировать ресурс, который является векторным изображением, я получаю значение null.

Вот пример кода:

if (BitmapFactory.decodeResource(arg0.getResources(), R.drawable.vector_menu_objectifs) == null)
        Log.d("ISNULL", "NULL");
    else
        Log.d("ISNULL", "NOT NULL");

В этом примере, когда я заменяю R.drawable.vector_menu_objectifs «нормальным» изображением, например png, результат не равен нулю (я получаю правильное растровое изображение) Что-то мне не хватает?

liltof
источник
1
Имелась похожая проблема, не решение, а обходной путь: stackoverflow.com/questions/33548447/…
Чем

Ответы:

233

Проверено по API: 17, 21, 23

public static Bitmap getBitmapFromVectorDrawable(Context context, int drawableId) {
    Drawable drawable = ContextCompat.getDrawable(context, drawableId);
    if (Build.VERSION.SDK_INT < Build.VERSION_CODES.LOLLIPOP) {
        drawable = (DrawableCompat.wrap(drawable)).mutate();
    }

    Bitmap bitmap = Bitmap.createBitmap(drawable.getIntrinsicWidth(),
            drawable.getIntrinsicHeight(), Bitmap.Config.ARGB_8888);
    Canvas canvas = new Canvas(bitmap);
    drawable.setBounds(0, 0, canvas.getWidth(), canvas.getHeight());
    drawable.draw(canvas);

    return bitmap;
}

ОБНОВИТЬ:

Градл проекта:

dependencies {
        classpath 'com.android.tools.build:gradle:2.2.0-alpha5'
    }

Модуль gradle:

android {
    compileSdkVersion 23
    buildToolsVersion '23.0.3'
    defaultConfig {
        minSdkVersion 16
        targetSdkVersion 23
        vectorDrawables.useSupportLibrary = true
    }
    ...
}
...
Алексей
источник
Спасибо. Предыдущий ответ не работает с версией sdk, начиная с 23
Паха,
2
Это отлично работает для меня. Работает на API 15 без проблем
Вера
4
AppCompatDrawableManagerпомечен как @RestrictTo(LIBRARY_GROUP)внутренний, и вам не следует его использовать (API может быть изменен без уведомления). ContextCompatВместо этого используйте .
mradzinski
не работает Caused by: java.lang.NullPointerException: Attempt to invoke virtual method 'void android.graphics.Bitmap.setHasAlpha(boolean)' on a null object referenceна этой линииBitmap bitmap = Bitmap.createBitmap(drawable.getIntrinsicWidth(), drawable.getIntrinsicHeight(), Bitmap.Config.ARGB_8888);
user25
5
У меня были проблемы с этим решением. Для меня, использующего: AppCompatResources вместо ContextCompat исправил его: Drawable drawable = AppCompatResources.getDrawable (context, drawableId);
Mike T
64

Вы можете использовать следующий метод:

@TargetApi(Build.VERSION_CODES.LOLLIPOP)
private static Bitmap getBitmap(VectorDrawable vectorDrawable) {
    Bitmap bitmap = Bitmap.createBitmap(vectorDrawable.getIntrinsicWidth(),
            vectorDrawable.getIntrinsicHeight(), Bitmap.Config.ARGB_8888);
    Canvas canvas = new Canvas(bitmap);
    vectorDrawable.setBounds(0, 0, canvas.getWidth(), canvas.getHeight());
    vectorDrawable.draw(canvas);
    return bitmap;
}

которые я иногда комбинирую с:

private static Bitmap getBitmap(Context context, int drawableId) {
    Drawable drawable = ContextCompat.getDrawable(context, drawableId);
    if (drawable instanceof BitmapDrawable) {
        return ((BitmapDrawable) drawable).getBitmap();
    } else if (drawable instanceof VectorDrawable) {
        return getBitmap((VectorDrawable) drawable);
    } else {
        throw new IllegalArgumentException("unsupported drawable type");
    }
}
snodnipper
источник
2
надеюсь, @liltof вернется и пометит это как ответ. Следует отметить, что обоим методам нужна оболочка targetAPi, но студия Android сообщит вам об этом.
Роберто Томас
1
Я потратил около двух дней, пытаясь сделать это, теперь думая о проблеме с файлом svg. Спасибо!
sparkly_frog
1
не работает Caused by: java.lang.NullPointerException: Attempt to invoke virtual method 'void android.graphics.Bitmap.setHasAlpha(boolean)' on a null object referenceна этой линииBitmap bitmap = Bitmap.createBitmap(vectorDrawable.getIntrinsicWidth(), vectorDrawable.getIntrinsicHeight(), Bitmap.Config.ARGB_8888);
user25
45

Если вы хотите использовать Android KTX для Kotlin, вы можете использовать метод расширения Drawable#toBitmap()для достижения того же эффекта, что и другие ответы:

val bitmap = AppCompatResources.getDrawable(requireContext(), drawableId).toBitmap() 

или

val bitmap = AppCompatResources.getDrawable(context, drawableId).toBitmap() 

Чтобы добавить этот и другие полезные методы расширения, вам необходимо добавить следующее на свой уровень модуля build.gradle

repositories {
    google()
}

dependencies {
    implementation "androidx.core:core-ktx:1.2.0"
}

См. Здесь последние инструкции по добавлению зависимости в ваш проект.

Обратите внимание, что это будет работать для любого подкласса, Drawableи если это Drawable- BitmapDrawableярлык для использования базового класса Bitmap.

Дэвид Роусон
источник
Это самое простое решение для Kotlin.
Адам Гурвиц
1
Для меня он отлично работает с VectorDrawable.
ubuntudroid
Это лучший вариант .. по умолчанию androidx предоставляет функциональность
Решма
27

Основываясь на предыдущих ответах, его можно упростить таким образом, чтобы он соответствовал как VectorDrawable, так и BitmapDrawable и был совместим как минимум с API 15.

public static Bitmap getBitmapFromDrawable(Context context, @DrawableRes int drawableId) {
    Drawable drawable = AppCompatResources.getDrawable(context, drawableId);

    if (drawable instanceof BitmapDrawable) {
        return ((BitmapDrawable) drawable).getBitmap();
    } else if (drawable instanceof VectorDrawableCompat || drawable instanceof VectorDrawable) {
        Bitmap bitmap = Bitmap.createBitmap(drawable.getIntrinsicWidth(), drawable.getIntrinsicHeight(), Bitmap.Config.ARGB_8888);
        Canvas canvas = new Canvas(bitmap);
        drawable.setBounds(0, 0, canvas.getWidth(), canvas.getHeight());
        drawable.draw(canvas);

        return bitmap;
    } else {
        throw new IllegalArgumentException("unsupported drawable type");
    }
}

Затем вам нужно добавить в свой файл gradle:

android {
    defaultConfig {
        vectorDrawables.useSupportLibrary = true
    }
}

В версии до Lollipop он будет использовать VectorDrawableCompat, а в Lollipop - VectorDrawable.

РЕДАКТИРОВАТЬ

Я отредактировал условие после комментария @ user3109468

Eselfar
источник
1
drawable instanceof VectorDrawable || В drawable instanceof VectorDrawableCompat стороны должны быть поменяны местами. Результаты в классе Not Found, когда VectorDrawable не существует, когда VectorDrawableCompat должен быть сначала проверен, потому что он действительно существует.
Warrick
проверку типов VertorDrawable можно описать как(Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP && drawable instanceof VectorDrawable)
chmin
6

Престижность @Alexey

Вот Kotlinверсия, использующая расширения дляContext

fun Context.getBitmapFromVectorDrawable(drawableId: Int): Bitmap? {
    var drawable = ContextCompat.getDrawable(this, drawableId) ?: return null

    if (Build.VERSION.SDK_INT < Build.VERSION_CODES.LOLLIPOP) {
        drawable = DrawableCompat.wrap(drawable).mutate()
    }

    val bitmap = Bitmap.createBitmap(
            drawable.intrinsicWidth,
            drawable.intrinsicHeight,
            Bitmap.Config.ARGB_8888) ?: return null
    val canvas = Canvas(bitmap)
    drawable.setBounds(0, 0, canvas.width, canvas.height)
    drawable.draw(canvas)

    return bitmap
}

Пример использования в Activity:

val bitmap = this.getBitmapFromVectorDrawable(R.drawable.ic_done_white_24dp)
Gunhan
источник
1

Протестировано на API 16 - JellyBean с векторными чертежами

public static Bitmap getBitmapFromVectorDrawable(Context context, int drawableId) {
    Drawable drawable = AppCompatResources.getDrawable(context, drawableId);
    if (Build.VERSION.SDK_INT < Build.VERSION_CODES.LOLLIPOP) {
        drawable = (DrawableCompat.wrap(drawable)).mutate();
    }

    Bitmap bitmap = Bitmap.createBitmap(drawable.getIntrinsicWidth(),
            drawable.getIntrinsicHeight(), Bitmap.Config.ARGB_8888);
    Canvas canvas = new Canvas(bitmap);
    drawable.setBounds(0, 0, canvas.getWidth(), canvas.getHeight());
    drawable.draw(canvas);

    return bitmap;
}   
tomasmark79
источник
1

Используйте следующий код для преобразования изображения с правильным соотношением сторон (например, для значка уведомления):

public static Bitmap getBitmapFromVector(Context context, int drawableId) {
    Drawable drawable = ContextCompat.getDrawable(context, drawableId);
    int width = drawable.getIntrinsicWidth();
    int height = drawable.getIntrinsicHeight();
    Bitmap bitmap;
    if (width < height) {    //make a square
        bitmap = Bitmap.createBitmap(height, height, Bitmap.Config.ARGB_8888);
    } else {
        bitmap = Bitmap.createBitmap(width, width, Bitmap.Config.ARGB_8888);
    }
    Canvas canvas = new Canvas(bitmap);
    drawable.setBounds(0, 0,
            drawable.getIntrinsicWidth(),    //use dimensions of Drawable
            drawable.getIntrinsicHeight()
    );
    drawable.draw(canvas);
    return bitmap;
}
antaki93
источник
0

Если ваше vectorизображение intrinsicWidthи intrinsicHeightмаленькое, и вы пытаетесь отобразить растровое изображение в большом виде, вы увидите, что результатом является размытие.

В этом случае вы можете указать новую ширину / высоту для своего растрового изображения, чтобы получить лучшее изображение (или вы можете увеличить размер вектора в xml, но предоставить desireWidthи desireHeightможет быть более гибким).

private fun getBitmap(drawableId: Int, desireWidth: Int? = null, desireHeight: Int? = null): Bitmap? {
    val drawable = AppCompatResources.getDrawable(context, drawableId) ?: return null
    val bitmap = Bitmap.createBitmap(
        desireWidth ?: drawable.intrinsicWidth,
        desireHeight ?: drawable.intrinsicHeight,
        Bitmap.Config.ARGB_8888
    )
    val canvas = Canvas(bitmap)
    drawable.setBounds(0, 0, canvas.width, canvas.height)
    drawable.draw(canvas)
    return bitmap
}

Надеюсь, это поможет

Фан Ван Линь
источник
0
Drawable layerDrawable = (Drawable) imageBase.getDrawable();
Bitmap bitmap = Bitmap.createBitmap(layerDrawable.getIntrinsicWidth(),
        layerDrawable.getIntrinsicHeight(), Bitmap.Config.ARGB_8888);
Canvas canvas = new Canvas(bitmap);
layerDrawable.setBounds(0, 0, canvas.getWidth(), canvas.getHeight());
layerDrawable.draw(canvas);  
imageTeste.setImageBitmap(addGradient(bitmap));
амин
источник
0

Если вы хотите масштабировать вывод до желаемого размера, попробуйте следующий фрагмент:

fun getBitmapFromVectorDrawable(context: Context, drawableId: Int, outputSize: OutputSize? = null): Bitmap? {
    var drawable = ContextCompat.getDrawable(context, drawableId) ?: return null
    if (Build.VERSION.SDK_INT < Build.VERSION_CODES.LOLLIPOP) {
        drawable = DrawableCompat.wrap(drawable).mutate()
    }

    var targetBitmap: Bitmap
    if (outputSize != null) {
        targetBitmap = Bitmap.createBitmap(outputSize.width,
                outputSize.height, Bitmap.Config.ARGB_8888)
    } else {
        targetBitmap = Bitmap.createBitmap(drawable.intrinsicWidth,
            drawable.intrinsicHeight, Bitmap.Config.ARGB_8888)
    }

    val canvas = Canvas(targetBitmap)
    val scaleX =  targetBitmap.width.toFloat()/drawable.intrinsicWidth.toFloat()
    val scaleY =  targetBitmap.height.toFloat()/drawable.intrinsicHeight.toFloat()
    canvas.scale(scaleX, scaleY)
    drawable.draw(canvas)

    return targetBitmap
}

class OutputSize(val width: Int, val height: Int)
Hans
источник
0

Это дает вам растровое изображение нужного размера. Кроме того, он позволяет поддерживать или нет прозрачность в зависимости от каждого изображения для лучшей производительности с теми, которым это не нужно.

public static Bitmap drawableToBitmap(Resources res, int drawableId,
        int width, int height, boolean keepAlpha) {
    Drawable drawable = res.getDrawable(drawableId);
    Bitmap bmp = createBitmap(width, height, keepAlpha ?
            Bitmap.Config.ARGB_8888 : Bitmap.Config.RGB_565);
    Canvas cvs = new Canvas(bmp);
    drawable.setBounds(0, 0, width, height);
    drawable.draw(cvs);
    return bmp;
}
Эстид Фелипе Лосано Рейес
источник