ฉันเพิ่งอ่านซอร์สโค้ดImageView
และโดยพื้นฐานแล้วเป็นไปไม่ได้เลยหากไม่ใช้โซลูชันคลาสย่อยในเธรดนี้ ในImageView.onMeasure
ที่เราจะไปเส้นเหล่านี้:
// Get the max possible width given our constraints
widthSize = resolveAdjustedSize(w + pleft + pright, mMaxWidth, widthMeasureSpec);
// Get the max possible height given our constraints
heightSize = resolveAdjustedSize(h + ptop + pbottom, mMaxHeight, heightMeasureSpec);
ขนาดของรูปภาพอยู่ที่ไหนh
และอยู่ที่ไหนและw
เป็นp*
ช่องว่างภายใน
แล้ว:
private int resolveAdjustedSize(int desiredSize, int maxSize,
int measureSpec) {
...
switch (specMode) {
case MeasureSpec.UNSPECIFIED:
/* Parent says we can be as big as we want. Just don't be larger
than max size imposed on ourselves.
*/
result = Math.min(desiredSize, maxSize);
ดังนั้นหากคุณlayout_height="wrap_content"
ตั้งค่าไว้widthSize = w + pleft + pright
หรือกล่าวอีกนัยหนึ่งความกว้างสูงสุดจะเท่ากับความกว้างของรูปภาพ
ซึ่งหมายความว่าถ้าคุณตั้งขนาดที่แน่นอนภาพจะไม่ขยาย ฉันคิดว่านี่เป็นข้อบกพร่อง แต่โชคดีที่ Google รับแจ้งหรือแก้ไข แก้ไข: กินคำพูดของตัวเองฉันส่งรายงานข้อผิดพลาดและพวกเขาบอกว่าได้รับการแก้ไขแล้วในการเปิดตัวในอนาคต!
อีกวิธีหนึ่ง
นี่เป็นอีกหนึ่งวิธีแก้ปัญหา subclassed แต่คุณควร (ในทางทฤษฎีฉันไม่ได้ทดสอบจริงๆมันมาก!) ImageView
จะสามารถใช้งานได้ทุกที่ที่คุณ เพื่อใช้มันตั้งค่าlayout_width="match_parent"
และlayout_height="wrap_content"
. มันค่อนข้างทั่วไปมากกว่าโซลูชันที่ยอมรับเช่นกัน เช่นคุณสามารถปรับขนาดให้พอดีกับความสูงและพอดีกับความกว้าง
import android.content.Context;
import android.util.AttributeSet;
import android.widget.ImageView;
// This works around the issue described here: http://stackoverflow.com/a/12675430/265521
public class StretchyImageView extends ImageView
{
public StretchyImageView(Context context)
{
super(context);
}
public StretchyImageView(Context context, AttributeSet attrs)
{
super(context, attrs);
}
public StretchyImageView(Context context, AttributeSet attrs, int defStyle)
{
super(context, attrs, defStyle);
}
@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec)
{
// Call super() so that resolveUri() is called.
super.onMeasure(widthMeasureSpec, heightMeasureSpec);
// If there's no drawable we can just use the result from super.
if (getDrawable() == null)
return;
final int widthSpecMode = MeasureSpec.getMode(widthMeasureSpec);
final int heightSpecMode = MeasureSpec.getMode(heightMeasureSpec);
int w = getDrawable().getIntrinsicWidth();
int h = getDrawable().getIntrinsicHeight();
if (w <= 0)
w = 1;
if (h <= 0)
h = 1;
// Desired aspect ratio of the view's contents (not including padding)
float desiredAspect = (float) w / (float) h;
// We are allowed to change the view's width
boolean resizeWidth = widthSpecMode != MeasureSpec.EXACTLY;
// We are allowed to change the view's height
boolean resizeHeight = heightSpecMode != MeasureSpec.EXACTLY;
int pleft = getPaddingLeft();
int pright = getPaddingRight();
int ptop = getPaddingTop();
int pbottom = getPaddingBottom();
// Get the sizes that ImageView decided on.
int widthSize = getMeasuredWidth();
int heightSize = getMeasuredHeight();
if (resizeWidth && !resizeHeight)
{
// Resize the width to the height, maintaining aspect ratio.
int newWidth = (int) (desiredAspect * (heightSize - ptop - pbottom)) + pleft + pright;
setMeasuredDimension(newWidth, heightSize);
}
else if (resizeHeight && !resizeWidth)
{
int newHeight = (int) ((widthSize - pleft - pright) / desiredAspect) + ptop + pbottom;
setMeasuredDimension(widthSize, newHeight);
}
}
}