Преобразование представления в Bitmap без отображения его в Android?
Я попытаюсь объяснить, что именно мне нужно сделать.
У меня есть 3 отдельных экрана: A, B, C. Существует еще один экран, называемый HomeScreen, где все 3 растровые изображения экрана должны отображаться в виде галереи, и пользователь может выбрать, в каком виде он хочет идти.
Мне удалось получить растровые изображения всех трех экранов и отобразить их в окне "Галерея", поместив весь код в ActivityScreen Activity. Теперь это сильно усложнило код, и я хотел бы его упростить.
Итак, могу ли я вызвать другое действие из HomeScreen и не отображать его и просто получить растровое изображение этого экрана. Например, скажем, я просто вызываю HomeScreen и вызывает Activity A, B, C, и ни одна из Деяний из A, B, C не отображается. Он просто дает растровое изображение этого экрана с помощью getDrawingCache(). И тогда мы можем отобразить эти растровые изображения в виде галереи в HomeScreen.
Надеюсь, я очень четко объяснил эту проблему.
Пожалуйста, дайте мне знать, если это действительно возможно.
Ответы
Ответ 1
есть способ сделать это. вам нужно создать растровое изображение и холст и вызвать view.draw(canvas);
вот код:
public static Bitmap loadBitmapFromView(View v) {
Bitmap b = Bitmap.createBitmap( v.getLayoutParams().width, v.getLayoutParams().height, Bitmap.Config.ARGB_8888);
Canvas c = new Canvas(b);
v.layout(v.getLeft(), v.getTop(), v.getRight(), v.getBottom());
v.draw(c);
return b;
}
если представление не было отображено до того, как его размер будет равен нулю. Его можно измерить следующим образом:
if (v.getMeasuredHeight() <= 0) {
v.measure(LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT);
Bitmap b = Bitmap.createBitmap(v.getMeasuredWidth(), v.getMeasuredHeight(), Bitmap.Config.ARGB_8888);
Canvas c = new Canvas(b);
v.layout(0, 0, v.getMeasuredWidth(), v.getMeasuredHeight());
v.draw(c);
return b;
}
EDIT: согласно этот пост, передача WRAP_CONTENT
как значения в makeMeasureSpec()
не делает ничего хорошего (хотя для некоторых классов классов это работает), и рекомендуемый метод:
// Either this
int specWidth = MeasureSpec.makeMeasureSpec(parentWidth, MeasureSpec.AT_MOST);
// Or this
int specWidth = MeasureSpec.makeMeasureSpec(0 /* any */, MeasureSpec.UNSPECIFIED);
view.measure(specWidth, specWidth);
int questionWidth = view.getMeasuredWidth();
Ответ 2
вот мое решение:
public static Bitmap getBitmapFromView(View view) {
Bitmap returnedBitmap = Bitmap.createBitmap(view.getWidth(), view.getHeight(),Bitmap.Config.ARGB_8888);
Canvas canvas = new Canvas(returnedBitmap);
Drawable bgDrawable =view.getBackground();
if (bgDrawable!=null)
bgDrawable.draw(canvas);
else
canvas.drawColor(Color.WHITE);
view.draw(canvas);
return returnedBitmap;
}
Наслаждайтесь:)
Ответ 3
Попробуй это,
/**
* Draw the view into a bitmap.
*/
public static Bitmap getViewBitmap(View v) {
v.clearFocus();
v.setPressed(false);
boolean willNotCache = v.willNotCacheDrawing();
v.setWillNotCacheDrawing(false);
// Reset the drawing cache background color to fully transparent
// for the duration of this operation
int color = v.getDrawingCacheBackgroundColor();
v.setDrawingCacheBackgroundColor(0);
if (color != 0) {
v.destroyDrawingCache();
}
v.buildDrawingCache();
Bitmap cacheBitmap = v.getDrawingCache();
if (cacheBitmap == null) {
Log.e(TAG, "failed getViewBitmap(" + v + ")", new RuntimeException());
return null;
}
Bitmap bitmap = Bitmap.createBitmap(cacheBitmap);
// Restore the view
v.destroyDrawingCache();
v.setWillNotCacheDrawing(willNotCache);
v.setDrawingCacheBackgroundColor(color);
return bitmap;
}
Ответ 4
Я знаю, что это может быть просроченной проблемой, но у меня были проблемы с тем, чтобы какие-либо из этих решений работали для меня.
В частности, я обнаружил, что если какие-либо изменения были внесены в представление после того, как он был завышен, эти изменения не будут включены в отображаемое растровое изображение.
Здесь метод, который закончил работать для моего дела. Однако с одной оговоркой. перед вызовом getViewBitmap(View)
я раздул свое представление и попросил его расставить с известными измерениями. Это было необходимо, так как мой макет представления делал бы его нулевой высотой/шириной до тех пор, пока содержимое не было размещено внутри.
View view = LayoutInflater.from(context).inflate(layoutID, null);
//Do some stuff to the view, like add an ImageView, etc.
view.layout(0, 0, width, height);
Bitmap getViewBitmap(View view)
{
//Get the dimensions of the view so we can re-layout the view at its current size
//and create a bitmap of the same size
int width = view.getWidth();
int height = view.getHeight();
int measuredWidth = View.MeasureSpec.makeMeasureSpec(width, View.MeasureSpec.EXACTLY);
int measuredHeight = View.MeasureSpec.makeMeasureSpec(height, View.MeasureSpec.EXACTLY);
//Cause the view to re-layout
view.measure(measuredWidth, measuredHeight);
view.layout(0, 0, view.getMeasuredWidth(), view.getMeasuredHeight());
//Create a bitmap backed Canvas to draw the view into
Bitmap b = Bitmap.createBitmap(width, height, Bitmap.Config.ARGB_8888);
Canvas c = new Canvas(b);
//Now that the view is laid out and we have a canvas, ask the view to draw itself into the canvas
view.draw(c);
return b;
}
"Волшебный соус" для меня был найден здесь:
https://groups.google.com/forum/#!topic/android-developers/BxIBAOeTA1Q
Приветствия,
Леви
Ответ 5
Надеюсь это поможет
View view="some view instance";
view.setDrawingCacheEnabled(true);
Bitmap bitmap=view.getDrawingCache();
view.setDrawingCacheEnabled(false);
Обновить
Метод getDrawingCache()
устарел на уровне API 28. Поэтому ищите другую альтернативу для уровня API> 28.
Ответ 6
В Android KTX есть отличная функция расширения Kotlin: View.drawToBitmap(Bitmap.Config)
Ответ 7
Я думаю, что это немного лучше:
/**
* draws the view content to a bitmap. code initially based on :
* http://nadavfima.com/android-snippet-inflate-a-layout-draw-to-a-bitmap/
*/
@Nullable
public static Bitmap drawToBitmap(final View viewToDrawFrom, int width, int height) {
boolean wasDrawingCacheEnabled = viewToDrawFrom.isDrawingCacheEnabled();
if (!wasDrawingCacheEnabled)
viewToDrawFrom.setDrawingCacheEnabled(true);
if (width <= 0 || height <= 0) {
if (viewToDrawFrom.getWidth() <= 0 || viewToDrawFrom.getHeight() <= 0) {
viewToDrawFrom.measure(MeasureSpec.makeMeasureSpec(0, MeasureSpec.UNSPECIFIED), MeasureSpec.makeMeasureSpec(0, MeasureSpec.UNSPECIFIED));
width = viewToDrawFrom.getMeasuredWidth();
height = viewToDrawFrom.getMeasuredHeight();
}
if (width <= 0 || height <= 0) {
final Bitmap bmp = viewToDrawFrom.getDrawingCache();
final Bitmap result = bmp == null ? null : Bitmap.createBitmap(bmp);
if (!wasDrawingCacheEnabled)
viewToDrawFrom.setDrawingCacheEnabled(false);
return result;
}
viewToDrawFrom.layout(0, 0, width, height);
} else {
viewToDrawFrom.measure(MeasureSpec.makeMeasureSpec(width, MeasureSpec.EXACTLY), MeasureSpec.makeMeasureSpec(height, MeasureSpec.EXACTLY));
viewToDrawFrom.layout(0, 0, viewToDrawFrom.getMeasuredWidth(), viewToDrawFrom.getMeasuredHeight());
}
final Bitmap drawingCache = viewToDrawFrom.getDrawingCache();
final Bitmap bmp = ThumbnailUtils.extractThumbnail(drawingCache, width, height);
final Bitmap result = bmp == null || bmp != drawingCache ? bmp : Bitmap.createBitmap(bmp);
if (!wasDrawingCacheEnabled)
viewToDrawFrom.setDrawingCacheEnabled(false);
return result;
}
Используя приведенный выше код, вам не нужно указывать размер растрового изображения (используйте 0 для ширины и высоты), если вы хотите использовать один из самого представления.
Кроме того, если вы хотите преобразовать специальные виды (например, SurfaceView, Surface или Window) в растровое изображение, вам следует рассмотреть возможность использования класса PixelCopy. Это требует API 24 и выше, хотя. Я не знаю, как это сделать раньше.
Ответ 8
Макет или вид на растровое изображение:
private Bitmap createBitmapFromLayout(View tv) {
int spec = View.MeasureSpec.makeMeasureSpec(0, View.MeasureSpec.UNSPECIFIED);
tv.measure(spec, spec);
tv.layout(0, 0, tv.getMeasuredWidth(), tv.getMeasuredHeight());
Bitmap b = Bitmap.createBitmap(tv.getMeasuredWidth(), tv.getMeasuredWidth(),
Bitmap.Config.ARGB_8888);
Canvas c = new Canvas(b);
c.translate((-tv.getScrollX()), (-tv.getScrollY()));
tv.draw(c);
return b;
}
Метод вызова:
Bitmap src = createBitmapFromLayout(View.inflate(this, R.layout.sample, null)/* or pass your view object*/);
Ответ 9
view.setDrawingCacheEnabled(true);
Bitmap bitmap = Bitmap.createBitmap(view.getDrawingCache());
view.setDrawingCacheEnabled(false);