วิธีแก้ปัญหาเหล่านี้ไม่ได้ผลสำหรับฉันเพราะฉันต้องการคลาสที่รองรับการครอบตัดตามอำเภอใจจากทิศทางแนวนอนหรือแนวตั้งและฉันต้องการให้ฉันเปลี่ยนการครอบตัดแบบไดนามิก ฉันต้องการความเข้ากันได้ของPicassoด้วยและ Picasso ก็ตั้งค่าการวาดภาพอย่างเฉื่อยชา
การใช้งานของฉันได้รับการดัดแปลงโดยตรงจากImageView.javaใน AOSP หากต้องการใช้ให้ประกาศเช่นนี้ใน XML:
<com.yourapp.PercentageCropImageView
android:id="@+id/view"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:scaleType="matrix"/>
จากแหล่งที่มาหากคุณต้องการมีการปลูกพืชด้านบนโทร:
imageView.setCropYCenterOffsetPct(0f);
หากต้องการครอปล่างโทร:
imageView.setCropYCenterOffsetPct(1.0f);
หากคุณต้องการครอบตัด 1/3 ของทางลงให้โทร:
imageView.setCropYCenterOffsetPct(0.33f);
นอกจากนี้หากคุณเลือกที่จะใช้วิธีการครอบตัดแบบอื่นเช่น fit_center คุณสามารถทำได้และจะไม่มีการทริกเกอร์ตรรกะที่กำหนดเองนี้ (การใช้งานอื่น ๆ ให้คุณใช้วิธีการครอบตัดเท่านั้น)
สุดท้ายนี้ฉันได้เพิ่มวิธีการวาดใหม่ () ดังนั้นหากคุณเลือกที่จะเปลี่ยนวิธีการครอบตัด / scaleType แบบไดนามิกในโค้ดคุณสามารถบังคับให้มุมมองวาดใหม่ได้ ตัวอย่างเช่น:
fullsizeImageView.setScaleType(ScaleType.FIT_CENTER);
fullsizeImageView.redraw();
หากต้องการกลับไปที่การครอบตัดด้านบนตรงกลางที่สามที่กำหนดเองให้โทร:
fullsizeImageView.setScaleType(ScaleType.MATRIX);
fullsizeImageView.redraw();
นี่คือชั้นเรียน:
import android.content.Context;
import android.graphics.Matrix;
import android.graphics.drawable.Drawable;
import android.util.AttributeSet;
import android.widget.ImageView;
public class PercentageCropImageView extends ImageView{
private Float mCropYCenterOffsetPct;
private Float mCropXCenterOffsetPct;
public PercentageCropImageView(Context context) {
super(context);
}
public PercentageCropImageView(Context context, AttributeSet attrs) {
super(context, attrs);
}
public PercentageCropImageView(Context context, AttributeSet attrs,
int defStyle) {
super(context, attrs, defStyle);
}
public float getCropYCenterOffsetPct() {
return mCropYCenterOffsetPct;
}
public void setCropYCenterOffsetPct(float cropYCenterOffsetPct) {
if (cropYCenterOffsetPct > 1.0) {
throw new IllegalArgumentException("Value too large: Must be <= 1.0");
}
this.mCropYCenterOffsetPct = cropYCenterOffsetPct;
}
public float getCropXCenterOffsetPct() {
return mCropXCenterOffsetPct;
}
public void setCropXCenterOffsetPct(float cropXCenterOffsetPct) {
if (cropXCenterOffsetPct > 1.0) {
throw new IllegalArgumentException("Value too large: Must be <= 1.0");
}
this.mCropXCenterOffsetPct = cropXCenterOffsetPct;
}
private void myConfigureBounds() {
if (this.getScaleType() == ScaleType.MATRIX) {
Drawable d = this.getDrawable();
if (d != null) {
int dwidth = d.getIntrinsicWidth();
int dheight = d.getIntrinsicHeight();
Matrix m = new Matrix();
int vwidth = getWidth() - this.getPaddingLeft() - this.getPaddingRight();
int vheight = getHeight() - this.getPaddingTop() - this.getPaddingBottom();
float scale;
float dx = 0, dy = 0;
if (dwidth * vheight > vwidth * dheight) {
float cropXCenterOffsetPct = mCropXCenterOffsetPct != null ?
mCropXCenterOffsetPct.floatValue() : 0.5f;
scale = (float) vheight / (float) dheight;
dx = (vwidth - dwidth * scale) * cropXCenterOffsetPct;
} else {
float cropYCenterOffsetPct = mCropYCenterOffsetPct != null ?
mCropYCenterOffsetPct.floatValue() : 0f;
scale = (float) vwidth / (float) dwidth;
dy = (vheight - dheight * scale) * cropYCenterOffsetPct;
}
m.setScale(scale, scale);
m.postTranslate((int) (dx + 0.5f), (int) (dy + 0.5f));
this.setImageMatrix(m);
}
}
}
@Override
protected boolean setFrame(int l, int t, int r, int b) {
boolean changed = super.setFrame(l, t, r, b);
this.myConfigureBounds();
return changed;
}
@Override
public void setImageDrawable(Drawable d) {
super.setImageDrawable(d);
this.myConfigureBounds();
}
@Override
public void setImageResource(int resId) {
super.setImageResource(resId);
this.myConfigureBounds();
}
public void redraw() {
Drawable d = this.getDrawable();
if (d != null) {
this.setImageDrawable(null);
this.setImageDrawable(d);
}
}
}