最近正在完成一个摇一摇的项目,打算加入一点社交元素。第一步是实现登陆界面,一个人独自设计加开发。最后的效果还是比较满意,但还是有性能上的损失,应该可以做得更好。
背景图片可以循环左右移动,使用MaterrialEdittext 实现了Edittext的向下兼容。此库继承自AppCompatEdittext,所以支持Edittext的所有特性,而且添加了字符检测,标签动画,组件颜色的定制等。有了它基本上可以忘记系统提供的Edittext了。API和下载地址
- 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;
}
热门评论
很不错!学习了!正好最近在考虑用googlemd风格