2024年最全Android自定义控件:一款多特效的智能loadingView,安卓性能优化面试题

自学编程路线、面试题集合/面经、及系列技术文章等,资源持续更新中…

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化学习资料的朋友,可以戳这里获取

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

//自定义控件的3参方法的attrs就是我们设置自定义属性的关键

//比如我们再attrs.xml里自定义了我们的属性,

TypedArray typedArray = getContext().obtainStyledAttributes(attrs, R.styleable.SmartLoadingView);

//这里是获取用户有没有设置整个属性

//这里是从用户那里获取有没有设置文案

String title = typedArray.getString(R.styleable.SmartLoadingView_textStr);

if (TextUtils.isEmpty(title)){

//如果获取来的属性是空,那么可以默认一个属性

//(作者忘记设置了!因为已经发布后期优化,老尴尬了)

buttonString =“默认文案”;

}else{

//如果有设置文案

buttonString = title;

}

}

5、【设置点击事件,启动动画】

为了点击事件的直观,也可以把处理防止重复点击事件封装在里面

//这是我自定义登录点击的接口

public interface LoginClickListener {

void click();

}

public void setLoginClickListener(final LoginClickListener loginClickListener) {

this.setOnClickListener(new OnClickListener() {

@Override

public void onClick(View v) {

if (loginClickListener != null) {

//防止重复点击

if (!isAnimRuning) {

start();

loginClickListener.click();

}

}

}

});

}

6、【动画讲解】
6.1、第一个动画,矩形到正方形,以及矩形到圆角矩形(这里是2个动画,只是同时进行)

矩形到正方形(为了简化,我把源码一些其他属性去掉了,这样方便理解)

//其中 default_all_distance = (w - h) / 2;除以2是因为2遍都往中间缩短

private void set_rect_to_circle_animation() {

//这是一个属性动画,current_left 会在duration时间内,从0到default_all_distance匀速变化

//想添加多样化的话 还可以加入插值器。

animator_rect_to_square = ValueAnimator.ofInt(0, default_all_distance);

animator_rect_to_square.setDuration(duration);

animator_rect_to_square.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {

@Override

public void onAnimationUpdate(ValueAnimator animation) {

//这里的current_left跟onDraw相关,还记得吗

//onDraw里的控件区域

//控件左边区域 rectf.left = current_left;

//控件右边区域 rectf.right = width - current_left;

current_left = (int) animation.getAnimatedValue();

//刷新绘制

invalidate();

}

});

矩形到圆角矩形。就是从一个没有圆角的变成完全圆角的矩形,当然我展示的时候只有第三个图,最后一个按钮才明显了。

其他的我直接设置成了圆角按钮,因为我把圆角做成了一个属性。

还记得onDraw里的canvas.drawRoundRect(rectf, circleAngle, circleAngle, paint);circleAngle就是圆角的半径

可以想象一下如果全是圆角,那么circleAngle会是多少,当然是height/2;没错吧,所以

因为我把圆角做成了属性obtainCircleAngle是从xml文件获取的属性,如果不设置,则为0,就没有任何圆角效果

animator_rect_to_angle = ValueAnimator.ofInt(obtainCircleAngle, height / 2);

animator_rect_to_angle.setDuration(duration);

animator_rect_to_angle.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {

@Override

public void onAnimationUpdate(ValueAnimator animation) {

//这里试想下如果是一个正方形,刚好是圆形的圆角,那就是一个圆

circleAngle = (int) animation.getAnimatedValue();

//刷新绘画

invalidate();

}

});

2个属性动画做好后,用

private AnimatorSet animatorSet = new AnimatorSet();把属性动画加进去,可以设置2个动画同时进行,还是先后顺序

这里是同时进行所用用with

animatorSet

.play(animator_rect_to_square).with(animator_rect_to_angle);

6.2、变成圆形后,有一个loading加载动画

这里就是画圆弧,只是不断改变,圆弧的起始点和终点,最终呈现loading状态,也是在onDraw里

//绘制加载进度

if (isLoading) {

//参数1:绘制圆弧区域

//参数2,3:绘制圆弧起始点和终点

canvas.drawArc(new RectF(width / 2 - height / 2 + height / 4, height / 4, width / 2 + height / 2 - height / 4, height / 2 + height / 2 - height / 4), startAngle, progAngle, false, okPaint);

//这里是我通过实践,实现最佳loading动画

//当然这里有很多方式,因为我自定义这个view想把所有东西都放在这个类里面,你也可以有你的方式

//如果有更好的方式,欢迎留言,告知我一下

startAngle += 6;

if (progAngle >= 270) {

progAngle -= 2;

isAdd = false;

} else if (progAngle <= 45) {

progAngle += 6;

isAdd = true;

} else {

if (isAdd) {

progAngle += 6;

} else {

progAngle -= 2;

}

}

//刷新绘制,这里不用担心有那么多刷新绘制,会不会影响性能

//

postInvalidate();

}

6.3、loading状态,到打勾动画

那么这里首先要把loading动画取消,那么直接改变isLoading=false;不会只它同时启动打勾动画;打勾动画的动画,这里比较麻烦,也是我在别人自定义动画里学习的,通过PathMeasure,实现路径动画

/**

  • 路径–用来获取对勾的路径

*/

private Path path = new Path();

/**

  • 取路径的长度

*/

private PathMeasure pathMeasure;

//初始化打勾动画路径;

private void initOk() {

//对勾的路径

path.moveTo(default_all_distance + height / 8 * 3, height / 2);

path.lineTo(default_all_distance + height / 2, height / 5 * 3);

path.lineTo(default_all_distance + height / 3 * 2, height / 5 * 2);

pathMeasure = new PathMeasure(path, true);

}

//初始化打勾动画

private void set_draw_ok_animation() {

animator_draw_ok = ValueAnimator.ofFloat(1, 0);

animator_draw_ok.setDuration(duration);

animator_draw_ok.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {

@Override

public void onAnimationUpdate(ValueAnimator animation) {

startDrawOk = true;

isLoading = false;

float value = (Float) animation.getAnimatedValue();

effect = new DashPathEffect(new float[]{pathMeasure.getLength(), pathMeasure.getLength()}, value * pathMeasure.getLength());

okPaint.setPathEffect(effect);

invalidate();

}

});

}

//启动打勾动画只需要调用

animator_draw_ok.start();

onDraw里绘制打勾动画

//绘制打勾,这是onDraw的,startDrawOk是判断是否开启打勾动画的标识

if (startDrawOk) {

canvas.drawPath(path, okPaint);

}

6.4、loading状态下回到失败样子(有点类似联网失败了)

之前6.1提到了矩形到圆角矩形和矩形到正方形的动画,那么这里只是前面2个动画反过来,再加上联网失败的文案,和联网失败的背景图即刻

6.5、loading状态下启动扩散全屏动画(重点)

这里我通过loginSuccess里参数的类型启动不同效果:

1、启动扩散全屏动画

public void loginSuccess(Animator.AnimatorListener endListener) {}

2、启动打勾动画

public void loginSuccess(AnimationOKListener animationOKListener) {}

启动扩散全屏是本文的重点,里面还涉及到了一个自定义view

CirclBigView,这个控件是全屏的,而且是从一个小圆不断改变半径变成大圆的动画,那么有人会问,全屏肯定不好啊,会影响布局,但是这里,我把它放在了activity的视图层:

ViewGroup activityDecorView = (ViewGroup) ((Activity) getContext()).getWindow().getDecorView();

ViewGroup.LayoutParams layoutParams = new ViewGroup.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.MATCH_PARENT);

activityDecorView.addView(circlBigView, layoutParams);

这个灵感也是前不久在学习微信,拖拽退出的思路里发现的。全部代码如下:

public void toBigCircle(Animator.AnimatorListener endListener) {

//把缩小到圆的半径,告诉circlBigView

circlBigView.setRadius(this.getMeasuredHeight() / 2);

//把当前背景颜色告诉circlBigView

circlBigView.setColorBg(normal_color);

int[] location = new int[2];

//测量当前控件所在的屏幕坐标x,y

this.getLocationOnScreen(location);

//把当前坐标告诉circlBigView,同时circlBigView会计算当前点,到屏幕4个点的最大距离,即是当前控件要扩散到的半径

//具体建议读者看完本博客后,去下载玩耍下。

circlBigView.setXY(location[0] + this.getMeasuredWidth() / 2, location[1]);

if (circlBigView.getParent() == null) {

ViewGroup activityDecorView = (ViewGroup) ((Activity) getContext()).getWindow().getDecorView();

ViewGroup.LayoutParams layoutParams = new ViewGroup.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.MATCH_PARENT);

activityDecorView.addView(circlBigView, layoutParams);

}

circlBigView.startShowAni(endListener);

isAnimRuning = false;

}

总结

开发是面向对象。我们找工作应该更多是面向面试。哪怕进大厂真的只是去宁螺丝,但你要进去得先学会面试的时候造飞机不是么?

作者13年java转Android开发,在小厂待过,也去过华为,OPPO等,去年四月份进了阿里一直到现在。等大厂待过也面试过很多人。深知大多数初中级Android工程师,想要提升技能,往往是自己摸索成长,不成体系的学习效果低效漫长且无助。

这里附上上述的技术体系图相关的几十套腾讯、头条、阿里、美团等公司的面试题,把技术点整理成了视频和PDF(实际上比预期多花了不少精力),包含知识脉络 + 诸多细节,由于篇幅有限,这里以图片的形式给大家展示一部分。

相信它会给大家带来很多收获:

960页全网最全Android开发笔记

资料太多,全部展示会影响篇幅,暂时就先列举这些部分截图

当程序员容易,当一个优秀的程序员是需要不断学习的,从初级程序员到高级程序员,从初级架构师到资深架构师,或者走向管理,从技术经理到技术总监,每个阶段都需要掌握不同的能力。早早确定自己的职业方向,才能在工作和能力提升中甩开同龄人。

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化学习资料的朋友,可以戳这里获取

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

、美团等公司的面试题**,把技术点整理成了视频和PDF(实际上比预期多花了不少精力),包含知识脉络 + 诸多细节,由于篇幅有限,这里以图片的形式给大家展示一部分。

相信它会给大家带来很多收获:

[外链图片转存中…(img-T8giDXx4-1715855736404)]

[外链图片转存中…(img-7lCjafL4-1715855736404)]

资料太多,全部展示会影响篇幅,暂时就先列举这些部分截图

当程序员容易,当一个优秀的程序员是需要不断学习的,从初级程序员到高级程序员,从初级架构师到资深架构师,或者走向管理,从技术经理到技术总监,每个阶段都需要掌握不同的能力。早早确定自己的职业方向,才能在工作和能力提升中甩开同龄人。

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化学习资料的朋友,可以戳这里获取

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

  • 4
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值