手记

MaterialDesign+FullscreenImageView实现漂亮的登陆界面

最近正在完成一个摇一摇的项目,打算加入一点社交元素。第一步是实现登陆界面,一个人独自设计加开发。最后的效果还是比较满意,但还是有性能上的损失,应该可以做得更好。

背景图片可以循环左右移动,使用MaterrialEdittext 实现了Edittext的向下兼容。此库继承自AppCompatEdittext,所以支持Edittext的所有特性,而且添加了字符检测,标签动画,组件颜色的定制等。有了它基本上可以忘记系统提供的Edittext了。API和下载地址

  1. FullscreenImageView
    主要是基于ImageView添加了自动缩放图片和自动滚动图片的动画。我的想法是:要让图片铺满ImageView,图片的高度一定要正好是ImageView的高度,但是图片宽度一定要比ImageView宽,这样动画播放才会把其余部分显示出来。所以,可以在onSizeChanged() 方法中得到组件的宽高(不能在其它地方尤其是构造方法中调用getWidth()或getHeight(),因为此时View的绘制过程不一定完成了)。如果图片高度和ImageView高度不符合,就进行缩放。下面是方法内部的代码。
 new AsyncTask<Object,Object,Bitmap>(){
            @Override
            protected Bitmap doInBackground(Object... params) {
                //预加载获取图片宽高
                BitmapFactory.Options options = new BitmapFactory.Options();
                options.inJustDecodeBounds = true;
                BitmapFactory.decodeResource((Resources) params[0],R.drawable.flowbloom,options);
                imageWidth = options.outWidth;
                imageHeight = options.outHeight;
                int inSampleSize = 1;
                //防止图片过大,出现OOM
                if (imageHeight > (Integer) params[2] ) {
                    final int halfHeight = imageHeight / 2;

                    while ((halfHeight / inSampleSize) > (Integer) params[2]) {
                        inSampleSize *= 2;
                    }
                }
                options.inSampleSize = inSampleSize;
                options.inJustDecodeBounds = false;

                return BitmapFactory.decodeResource((Resources)params[0],R.drawable
                        .flowbloom,options);
            }

            @Override
            protected void onPostExecute(Bitmap background) {
                setImageBitmap(background);

                Drawable d = getDrawable();
                if (d == null) return;

                int dw = d.getIntrinsicWidth();
                int dh = d.getIntrinsicHeight();

                Log.i(TAG,"onPostExecute: dw: " + dw + " dh :" + dh);

                float scale; //缩放值
                scale = getHeight() * 1f / dh;

                matrix.postScale(scale,scale);
                matrix.postTranslate(-100,0);
                setImageMatrix(matrix);
            }
        }.execute(getResources(),w,h);

这里使用了 AsyncTask 新开了一个线程加载图片,然而效果并不好,反而在打开登陆界面时有一段时间的黑屏过程,还不如直接在UI线程中加载。

这里遇到了两个坑:
1、使用Matrix控制图片必须在xml文件中为ImageView添加android:scaleType="matrix"
2、放在不同资源限定符的drawable文件夹中的Bitmap,它的Density属性是不同的。而每个设备的TargetDensity属性也是不同的(小米note是440,Vivo X3t 是320) ,系统就是根据TargetDensity到不同的drawable文件夹找图片的。举个例子,我只在drawable-mdpi中放了一张图片,那它的Density是160(图片像素1440x900)。小米note加载这张图片,因为drawable-xxhdpi没图片,于是它会对其进行缩放1440 x (440 \ 160) = 3960 , 900 x (440 \ 160 ) = 2475。于是内存中图片的真正像素是(3960 x 2475)。所以,如果不注意为每个drawable文件夹配上图片,可能不同设备上的图片像素会不一致。(注意:测试系统 API23、AndroidStudio2.1)

自动滚动图片是由一个MoveImageAnimator实现的动画的效果。

private class MoveImageAnimator extends Animation {
        @Override
        protected void applyTransformation(float interpolatedTime,Transformation t) {
            super.applyTransformation(interpolatedTime,t);
            RectF r = getMatrixRectF();

            if (r.left > 0 || r.right < getWidth()) {
                back = !back;
            }

            if (back) {
                matrix.postTranslate(2,0);
            } else {
                matrix.postTranslate(-2,0);
            }
            setImageMatrix(matrix);

        }
    }

移动方法很简单,使用Matrix对图片进行水平位移操作。这里需要注意的就是对图片左右边框的判断,如果到达图片的左右边框,让back=!back 就是碰撞到边框马上返回。检测图片的左右边框可以使用下面的方法:

/**
     * 此方法能获取View中图片的描述RectF
     * @return
     */
    private RectF getMatrixRectF() {
        Matrix m = matrix;
        RectF rectF = new RectF();

        Drawable d = getDrawable();
        if (d != null) {
            rectF.set(0,0,d.getIntrinsicWidth(),d.getIntrinsicHeight());//当前图片宽高(不改变的)
            m.mapRect(rectF);//将matrix中的值映射到rectF中
        }

        return rectF;
    }
8人推荐
随时随地看视频
慕课网APP

热门评论

很不错!学习了!正好最近在考虑用googlemd风格

查看全部评论