Android: масштабировать рисованное или фоновое изображение? - PullRequest
124 голосов
/ 09 сентября 2009

В макете я хочу масштабировать фоновое изображение (сохраняя его соотношение сторон) до пространства, выделенного при создании страницы. У кого-нибудь есть идеи, как это сделать?

Я использую layout.setBackgroundDrawable () и использую BitmapDrawable для setGravity для отсечения и заполнения, но не вижу никакой опции для масштабирования.

Ответы [ 13 ]

86 голосов
/ 20 февраля 2012

Чтобы настроить масштабирование фонового изображения, создайте ресурс, подобный этому:

<?xml version="1.0" encoding="utf-8"?>
<bitmap xmlns:android="http://schemas.android.com/apk/res/android"
    android:gravity="center"
    android:src="@drawable/list_bkgnd" />

Тогда он будет центрирован в виде, если используется в качестве фона. Есть и другие флаги: http://developer.android.com/guide/topics/resources/drawable-resource.html

54 голосов
/ 12 сентября 2009

Не пытался делать именно то, что вы хотите, но вы можете масштабировать ImageView, используя android:scaleType="fitXY"
и его размер будет соответствовать размеру, указанному вами в ImageView.

Таким образом, вы можете создать FrameLayout для своего макета, поместить в него ImageView, а затем любой другой контент, который вам нужен в FrameLayout, также с прозрачным фоном.

<FrameLayout  
android:layout_width="fill_parent" android:layout_height="fill_parent">

  <ImageView 
android:layout_width="fill_parent" android:layout_height="fill_parent"
android:src="@drawable/back" android:scaleType="fitXY" />

  <LinearLayout>your views</LinearLayout>
</FrameLayout>
28 голосов
/ 21 февраля 2014

Существует простой способ сделать это из отрисовки:

your_drawable.xml

<?xml version="1.0" encoding="utf-8"?>
<layer-list xmlns:android="http://schemas.android.com/apk/res/android" >

    <item android:drawable="@color/bg_color"/>
    <item>
        <bitmap
            android:gravity="center|bottom|clip_vertical"
            android:src="@drawable/your_image" />
    </item>

</layer-list>

Единственным недостатком является то, что если недостаточно места, ваше изображение не будет отображаться полностью, но оно будет обрезано, я не смог найти способ сделать это непосредственно из рисования. Но из тестов, которые я сделал, это работает довольно хорошо, и это не обрезает большую часть изображения. Вы можете играть больше с опциями гравитации.

Другим способом будет просто создать макет, где вы будете использовать ImageView и установить для scaleType значение fitCenter.

17 голосов
/ 23 декабря 2010

Чтобы сохранить соотношение сторон, вы должны использовать android:scaleType=fitCenter или fitStart и т. Д. Использование fitXY не сохранит исходное соотношение сторон изображения!

Обратите внимание, что это работает только для изображений с атрибутом src, но не для фонового изображения.

15 голосов
/ 24 сентября 2014

Использовать изображение в качестве фона для макета:

<?xml version="1.0" encoding="utf-8"?>
<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="wrap_content" >

    <ImageView
        android:id="@+id/imgPlaylistItemBg"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:adjustViewBounds="true"
        android:maxHeight="0dp"
        android:scaleType="fitXY"
        android:src="@drawable/img_dsh" />

    <LinearLayout
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:orientation="vertical" >


    </LinearLayout>

</FrameLayout>
5 голосов
/ 08 марта 2012

Когда вы устанавливаете Drawable для ImageView с помощью метода setBackgroundDrawable, изображение всегда будет масштабироваться. Параметры как adjustViewBounds или другие ScaleTypes будут просто игнорироваться. Единственное решение для сохранения соотношения сторон, которое я нашел, - изменить размер ImageView после загрузки вашего объекта. Вот фрагмент кода, который я использовал:

// bmp is your Bitmap object
int imgHeight = bmp.getHeight();
int imgWidth = bmp.getWidth();
int containerHeight = imageView.getHeight();
int containerWidth = imageView.getWidth();
boolean ch2cw = containerHeight > containerWidth;
float h2w = (float) imgHeight / (float) imgWidth;
float newContainerHeight, newContainerWidth;

if (h2w > 1) {
    // height is greater than width
    if (ch2cw) {
        newContainerWidth = (float) containerWidth;
        newContainerHeight = newContainerWidth * h2w;
    } else {
        newContainerHeight = (float) containerHeight;
        newContainerWidth = newContainerHeight / h2w;
    }
} else {
    // width is greater than height
    if (ch2cw) {
        newContainerWidth = (float) containerWidth;
        newContainerHeight = newContainerWidth / h2w; 
    } else {
        newContainerWidth = (float) containerHeight;
        newContainerHeight = newContainerWidth * h2w;       
    }
}
Bitmap copy = Bitmap.createScaledBitmap(bmp, (int) newContainerWidth, (int) newContainerHeight, false);
imageView.setBackgroundDrawable(new BitmapDrawable(copy));
LayoutParams params = new LayoutParams(LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT);
imageView.setLayoutParams(params);
imageView.setMaxHeight((int) newContainerHeight);
imageView.setMaxWidth((int) newContainerWidth);

В приведенном выше фрагменте кода bmp должен отображаться растровый объект, а imageView - это ImageView объект

Важно отметить, что изменение параметров макета . Это необходимо, потому что setMaxHeight и setMaxWidth будут иметь значение только в том случае, если ширина и высота определены для переноса содержимого, а не для заполнения родительского элемента. С другой стороны, Fill parent является желаемой настройкой в ​​начале, потому что в противном случае containerWidth и containerHeight будут иметь значения, равные 0. Итак, в вашем файле макета у вас будет что-то вроде этого для вашего ImageView:

...
<ImageView android:id="@+id/my_image_view"
    android:layout_width="fill_parent"
    android:layout_height="fill_parent"/>
...
4 голосов
/ 15 января 2014

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

int bwidth=bitMap1.getWidth();
int bheight=bitMap1.getHeight();
int swidth=imageView_location.getWidth();
int sheight=imageView_location.getHeight();
new_width=swidth;
new_height = (int) Math.floor((double) bheight *( (double) new_width / (double) bwidth));
Bitmap newbitMap = Bitmap.createScaledBitmap(bitMap1,new_width,new_height, true);
imageView_location.setImageBitmap(newbitMap)
4 голосов
/ 21 июня 2013

Это не самое эффективное решение, но, как кто-то предложил вместо фона, вы можете создать FrameLayout или RelativeLayout и использовать ImageView в качестве псевдо-фона - другие элементы будут располагаться просто над ним:

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_height="match_parent"
    android:layout_width="match_parent">

    <ImageView
        android:id="@+id/ivBackground"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:layout_alignParentLeft="true"
        android:layout_alignParentTop="true"
        android:scaleType="fitStart"
        android:src="@drawable/menu_icon_exit" />

    <Button
        android:id="@+id/bSomeButton"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_alignParentLeft="true"
        android:layout_alignParentTop="true"
        android:layout_marginLeft="61dp"
        android:layout_marginTop="122dp"
        android:text="Button" />

</RelativeLayout>

Проблема с ImageView заключается в том, что доступны только масштабированные типы: CENTER, CENTER_CROP, CENTER_INSIDE, FIT_CENTER, FIT_END, FIT_START, FIT_XY, MATRIX (http://etcodehome.blogspot.de/2011/05/android-imageview-scaletype-samples.html)

и для «масштабирования фонового изображения (с сохранением его соотношения сторон)» в некоторых случаях, когда требуется, чтобы изображение заполняло весь экран (например, фоновое изображение) и соотношение сторон экрана отличалось от изображения, необходимое scaleType является разновидностью TOP_CROP, потому что:

CENTER_CROP центрирует масштабированное изображение вместо выравнивания верхнего края с верхним краем просмотра изображения, а FIT_START соответствует высоте экрана и не заполняет ширину. И, как заметил пользователь Anke, FIT_XY не сохраняет соотношение сторон.

Рад, что кто-то расширил ImageView для поддержки TOP_CROP

public class ImageViewScaleTypeTopCrop extends ImageView {
    public ImageViewScaleTypeTopCrop(Context context) {
        super(context);
        setup();
    }

    public ImageViewScaleTypeTopCrop(Context context, AttributeSet attrs) {
        super(context, attrs);
        setup();
    }

    public ImageViewScaleTypeTopCrop(Context context, AttributeSet attrs, int defStyle) {
        super(context, attrs, defStyle);
        setup();
    }

    private void setup() {
        setScaleType(ScaleType.MATRIX);
    }

    @Override
    protected boolean setFrame(int frameLeft, int frameTop, int frameRight, int frameBottom) {

        float frameWidth = frameRight - frameLeft;
        float frameHeight = frameBottom - frameTop;

        if (getDrawable() != null) {

            Matrix matrix = getImageMatrix();
            float scaleFactor, scaleFactorWidth, scaleFactorHeight;

            scaleFactorWidth = (float) frameWidth / (float) getDrawable().getIntrinsicWidth();
            scaleFactorHeight = (float) frameHeight / (float) getDrawable().getIntrinsicHeight();

            if (scaleFactorHeight > scaleFactorWidth) {
                scaleFactor = scaleFactorHeight;
            } else {
                scaleFactor = scaleFactorWidth;
            }

            matrix.setScale(scaleFactor, scaleFactor, 0, 0);
            setImageMatrix(matrix);
        }

        return super.setFrame(frameLeft, frameTop, frameRight, frameBottom);
    }

}

https://stackoverflow.com/a/14815588/2075875

Теперь ИМХО было бы идеально, если бы кто-нибудь написал собственный Drawable, который масштабирует изображение таким образом. Затем его можно использовать в качестве параметра фона.


Reflog предлагает предварительно масштабировать отрисовку перед ее использованием. Вот инструкция, как это сделать: Java (Android): как масштабировать чертеж без растрового изображения? Хотя он имеет недостаток, этот масштабируемый чертеж / растровое изображение будет использовать больше оперативной памяти, в то время как масштабирование на лету, используемое ImageView, не требует больше памяти. Преимущество может быть меньше загрузки процессора.

2 голосов
/ 11 апреля 2012

То, что предложил Двебо, работает. Но по моему скромному мнению это не нужно. Рисуемое фоновое изображение хорошо масштабируется само по себе. Вид должен иметь фиксированную ширину и высоту, как в следующем примере:

 < RelativeLayout 
    android:layout_width="fill_parent"
    android:layout_height="fill_parent"
    android:background="@android:color/black">

    <LinearLayout
        android:layout_width="500dip"
        android:layout_height="450dip"
        android:layout_centerInParent="true"
        android:background="@drawable/my_drawable"
        android:orientation="vertical"
        android:padding="30dip"
        >
        ...
     </LinearLayout>
 < / RelativeLayout>
0 голосов
/ 15 марта 2019

Котлин:

Если вам нужно нарисовать растровое изображение в представлении, масштабируется до FIT.

Вы можете сделать правильные вычисления, чтобы установить bm высоту, равную контейнеру, и отрегулировать ширину, если отношение ширины к высоте bm меньше, чем отношение ширины контейнера к высоте, или обратное в противоположном сценарии.

Изображения:

Example Image 1

Example Image 2

// binding.fragPhotoEditDrawCont is the RelativeLayout where is your view
// bm is the Bitmap
val ch = binding.fragPhotoEditDrawCont.height
val cw = binding.fragPhotoEditDrawCont.width
val bh = bm.height
val bw = bm.width
val rc = cw.toFloat() / ch.toFloat()
val rb = bw.toFloat() / bh.toFloat()

if (rb < rc) {
    // Bitmap Width to Height ratio is less than Container ratio
    // Means, bitmap should pin top and bottom, and have some space on sides.
    //              _____          ___
    // container = |_____|   bm = |___| 
    val bmHeight = ch - 4 //4 for container border
    val bmWidth = rb * bmHeight //new width is bm_ratio * bm_height
    binding.fragPhotoEditDraw.layoutParams = RelativeLayout.LayoutParams(bmWidth.toInt(), bmHeight)
}
else {
    val bmWidth = cw - 4 //4 for container border
    val bmHeight = 1f/rb * cw
    binding.fragPhotoEditDraw.layoutParams = RelativeLayout.LayoutParams(bmWidth, bmHeight.toInt())
}
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...