美文网首页
按比例自适应-AutoSizeImageView

按比例自适应-AutoSizeImageView

作者: AdminFun | 来源:发表于2019-01-24 14:54 被阅读0次

    一、效果

    实现原理比较简单,本章就不做原理讲解。一切只因为项目中经常遇到设置图片比例,于是便想到把功能封装一下方便以后使用。

    先看效果:图1是xml代码,图2是效果


    AutoImageView-code.png AutoImageView-result.png

    二、代码

    import android.content.Context;
    import android.content.res.TypedArray;
    import android.support.v7.widget.AppCompatImageView;
    import android.util.AttributeSet;
    import android.util.Log;
    
    import com.fun.ex.app.R;
    
    /**
     * 作者: Created by AdminFun
     * 邮箱: 614484070@qq.com
     * 创建: 2019/1/24
     * 修改: 2019/1/24
     * 版本: v1.0.0
     * 描述: 自动计算并重置宽高的图片控件,如果有需求可以继承自其他功能图片控件
     */
    public class AutoSizeImageView extends AppCompatImageView {
    
        private final String TAG = "common";
    
        public AutoSizeImageView(Context context) {
            super(context);
        }
    
        public AutoSizeImageView(Context context, AttributeSet attrs) {
            this(context, attrs, 0);
        }
    
        public AutoSizeImageView(Context context, AttributeSet attrs, int defStyleAttr) {
            super(context, attrs, defStyleAttr);
            TypedArray attributes = context.obtainStyledAttributes(attrs, R.styleable.AutoSizeImageView, defStyleAttr, 0);
            mPercentWidth = attributes.getInt(R.styleable.AutoSizeImageView_widget_width_percent, 1);
            mPercentHeight = attributes.getInt(R.styleable.AutoSizeImageView_widget_height_percent, 1);
            attributes.recycle();
        }
    
        private int mPercentWidth = 0, mPercentHeight = 0; // 想要设置的宽高比例
        private int mCurrentWidth = 0, mCurrentHeight = 0; // 当前控件在视图中未处理的宽高
        private int mCurrentMode = 4;                      // 当前控件的宽高类型,详细类型见 getType 注释
    
        /**
         * 设置控件宽高比:这里只设置比例即可
         * 例如:想设置宽高比为4:3,那么即可设置 width = 40,高=30。或者width=4,高=3
         */
        public void setPercent(int width, int height) {
            this.mPercentWidth = width < 0 ? 0 : width;
            this.mPercentHeight = height <= 0 ? 1 : height;
            this.reMeaureView(mCurrentMode, mCurrentWidth, mCurrentHeight);
        }
    
        @Override
        protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
            super.onMeasure(widthMeasureSpec, heightMeasureSpec);
            // final int minimumWidth = getSuggestedMinimumWidth();
            // final int minimumHeight = getSuggestedMinimumHeight();
            final int widthSpecMode = MeasureSpec.getMode(widthMeasureSpec);
            final int heightSpecMode = MeasureSpec.getMode(heightMeasureSpec);
            this.mCurrentWidth = MeasureSpec.getSize(widthMeasureSpec);
            this.mCurrentHeight = MeasureSpec.getSize(heightMeasureSpec);
            this.mCurrentMode = getType(widthSpecMode, heightSpecMode);
            Log.d(TAG, String.format("AutoSizeImageView.onMeasure:width = %s,height = %s,mode = %s",
                    String.valueOf(mCurrentWidth), String.valueOf(mCurrentHeight), String.valueOf(mCurrentMode)));
            this.reMeaureView(mCurrentMode, mCurrentWidth, mCurrentHeight);
        }
    
        private void reMeaureView(int CurrentMode, int MeasureWidth, int MeasureHeight) {
            switch (CurrentMode) {
                case 1:// 已知宽度和比例,不理会高度。
                    setMeasuredDimension(MeasureWidth, getHeightByWidth(MeasureWidth, MeasureHeight));
                    break;
    
                case 3:// 已知高度和比例,不理会宽度
                    setMeasuredDimension(getWidthByHeight(MeasureWidth, MeasureHeight), MeasureHeight);
                    break;
    
                case 2 | 4:// 宽度和高度都已知或都未知,完全不用理会
                    // TODO  都已知的情况下是不需要处理的,暂时还未想到都未知的场景。
                    break;
            }
        }
    
        private int getHeightByWidth(int width, int height) {
            return (mPercentWidth <= 0 || mPercentHeight <= 0) ?
                    height : width * mPercentHeight / mPercentWidth;
        }
    
        private int getWidthByHeight(int width, int height) {
            return (mPercentWidth <= 0 || mPercentHeight <= 0) ?
                    height : height * mPercentWidth / mPercentHeight;
        }
    
        /**
         * 四种类型
         * 1、width = match, height = warp,已知宽度和宽高比,计算高度size
         * 2、width = match, height = match,宽和高都已知,宽高比也有,这种情况下:不予理睬
         * 3、width = warp, height = match,已知高度和宽高比,计算宽度,重置size
         * 4、width = warp, height = warp,宽和高都未知,且知道比例,这种情况下:不予理睬
         * 注意:这里不考虑宽高都未设置的情况,默认第4种情况
         */
        private int getType(int widthMode, int heightMode) {
            switch (widthMode) {
                case MeasureSpec.EXACTLY: // 已知宽度
                    if (heightMode == MeasureSpec.AT_MOST) {
                        return 1;
                    } else if (heightMode == MeasureSpec.EXACTLY) {
                        return 2;
                    } else {
                        return 1;
                    }
    
                case MeasureSpec.AT_MOST | MeasureSpec.UNSPECIFIED: // 宽度未知
                    if (heightMode == MeasureSpec.AT_MOST) {
                        return 4;
                    } else if (heightMode == MeasureSpec.EXACTLY) {
                        return 3;
                    } else {
                        return 4;
                    }
            }
            return 4;
        }
    }
    
     <declare-styleable name="AutoSizeImageView">
            <attr name="widget_width_percent" format="integer" />
            <attr name="widget_height_percent" format="integer" />
     </declare-styleable>
    

    三、使用

    方法一(设置比例并非是设置宽高)
     <LinearLayout
                    android:layout_width="match_parent"
                    android:layout_height="wrap_content"
                    android:orientation="horizontal">
    
                    <com.fun.ex.app.custom_view.AutoSizeImageView
                        xmlns:app="http://schemas.android.com/apk/res-auto"
                        android:id="@+id/image"
                        android:layout_width="0dp"
                        android:layout_height="wrap_content"
                        android:layout_weight="1"
                        android:scaleType="centerCrop"
                        android:src="@drawable/fang_"
                        app:widget_height_percent="1"
                        app:widget_width_percent="1" />
    
                    <android.support.v7.widget.AppCompatImageView
                        android:id="@+id/image1"
                        android:layout_width="0dp"
                        android:layout_height="wrap_content"
                        android:layout_weight="1"
                        android:background="#56ffff"
                        android:scaleType="centerCrop"
                        android:src="@drawable/fang_" />
      </LinearLayout>
    
    方法二
    autoSizeImageView.setPercent(1, 1);
    

    相关文章

      网友评论

          本文标题:按比例自适应-AutoSizeImageView

          本文链接:https://www.haomeiwen.com/subject/rqnyjqtx.html