android 继承ViewGroup实现自定义布局

case MotionEvent.ACTION_MOVE:

int detlaX = x-mLastXIntercept;

int detlaY = y-mLastYIntercept;

if (Math.abs(detlaX) > Math.abs(detlaY)){

intercepted = true;

}else{

intercepted = false;

}

break;

case MotionEvent.ACTION_UP:

intercepted = false;

break;

default:

break;

}

Log.d(TAG,"intercepted = "+intercepted);

mLastX = x;

mLastY = y;

mLastXIntercept = x;

mLastYIntercept = y;

return intercepted;

}

@Override

public boolean onTouchEvent(MotionEvent event) {

mVelocityTracker.addMovement(event);

int x = (int)getX();

int y = (int)getY();

switch (event.getAction()){

case MotionEvent.ACTION_DOWN:

if (!mScroller.isFinished()){

mScroller.abortAnimation();

}

break;

case MotionEvent.ACTION_MOVE:

int detalX = x-mLastX;

int detalY = y-mLastY;

scrollBy(-detalX,0);

break;

case MotionEvent.ACTION_UP:

int scrollX = getScrollX();

mVelocityTracker.computeCurrentVelocity(1000);

float xVelocity = mVelocityTracker.getXVelocity();

if (Math.abs(xVelocity) >= 50){

mChildIndex = xVelocity > 0 ? mChildIndex -1 : mChildIndex + 1;

}else{

mChildIndex = (scrollX + mChildWidth / 2) / mChildWidth;

}

mChildIndex = Math.max(0,Math.min(mChildIndex,mChildrenSize -1 ));

int dx = mChildIndex * mChildWidth - scrollX;

smoothScrollBy(dx, 0);

mVelocityTracker.clear();

break;

default:

break;

}

mLastX = x;

mLastY = y;

return true;

}

@Override

protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {

super.onMeasure(widthMeasureSpec, heightMeasureSpec);

int measuredWidth = 0;

int measuredHeight = 0;

//拿到自己含有子元素的个数

final int childCount = getChildCount();

measureChildren(widthMeasureSpec,heightMeasureSpec);

int widthSpaceSize = MeasureSpec.getSize(widthMeasureSpec);

int widthSpaceMode = MeasureSpec.getMode(widthMeasureSpec);

int heightSpaceSize = MeasureSpec.getSize(heightMeasureSpec);

int heightSpaceMode = MeasureSpec.getMode(heightMeasureSpec);

/**

  • 1.判断是否含有子元素,如果没有直接把自己的高度和宽度设置为0

  • 2.判断子元素的宽度和高度是否采用的warp_content,如果采用了则把所有的子元素的宽度之和给自己并且把

  • 第一个子元素的高度给自己。

  • 3.再进行分别判断高度和宽度那个用了warp_content,就对其分别设置不一样的值。

*/

if (childCount == 0){

setMeasuredDimension(0,0);

}else if (widthSpaceMode == MeasureSpec.AT_MOST && heightSpaceMode == MeasureSpec.AT_MOST){

final View childView = getChildAt(0);

measuredWidth = childView.getMeasuredWidth() * childCount;

measuredHeight = childView.getMeasuredHeight();

setMeasuredDimension(measuredWidth,measuredHeight);

}else if (heightSpaceMode == MeasureSpec.AT_MOST){

final View childView = getChildAt(0);

measuredHeight = childView.getMeasuredHeight();

setMeasuredDimension(widthSpaceSize,childView.getMeasuredHeight());

}else if (widthSpaceMode == MeasureSpec.AT_MOST){

final View childView = getChildAt(0);

measuredWidth = childView.getMeasuredWidth() * childCount;

setMeasuredDimension(measuredWidth,heightSpaceSize);

}

/**

  • 以上代码有两个不规范的地方:

  • 1.如果没有子元素的时候,不应该直接把宽度/高度直接设置为0,而是要根据LayoutParams中的高度/宽度来做相应的处理

  • 2.在测量的时候要考虑到自己和子元素的padding和margin值,这样测出来的才是最标准的

*/

}

/**

  • 完成子元素的定位:

  • 1.遍历所有的子元素是否处于GONE的状态,不是则放置在合适的位置

  • 放置的过程也是由左向右

  • 缺点:也是没有考虑到自己个子元素的padding和maigin的值

  • @param changed

  • @param l

  • @param t

  • @param r

  • @param b

*/

@Override

protected void onLayout(boolean changed, int l, int t, int r, int b) {

int childLeft = 0;

final int childCount = getChildCount();

mChildrenSize = childCount;

for (int i = 0; i < childCount; i++) {

final View childView = getChildAt(i);

if (childView.getVisibility() != View.GONE){

final int childWidth = childView.getMeasuredWidth();

mChildWidth = childWidth;

childView.layout(childLeft,0,childLeft + childWidth,childView.getMeasuredHeight());

childLeft += childWidth;

}

}

}

private void smoothScrollBy(int dx,int dy){

//1000ms内滑向目标位置,效果是谩慢滑动

mScroller.startScroll(getScrollX(),0,dx,0,500);

invalidate();

}

@Override

public void computeScroll() {

if (mScroller.computeScrollOffset()){

scrollTo(mScroller.getCurrX(),mScroller.getCurrY());

postInvalidate();

}
自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数初中级Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则近万的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。

img

img

img

img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:Android)

总结

找工作是个很辛苦的事情,而且一般周期都比较长,有时候既看个人技术,也看运气。第一次找工作,最后的结果虽然不尽如人意,不过收获远比offer大。接下来就是针对自己的不足,好好努力了。

最后为了节约大家的时间,我把我学习所用的资料和面试遇到的问题和答案都整理成了PDF文档

喜欢文章的话请关注、点赞、转发 谢谢!

《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!

总结

找工作是个很辛苦的事情,而且一般周期都比较长,有时候既看个人技术,也看运气。第一次找工作,最后的结果虽然不尽如人意,不过收获远比offer大。接下来就是针对自己的不足,好好努力了。

最后为了节约大家的时间,我把我学习所用的资料和面试遇到的问题和答案都整理成了PDF文档

喜欢文章的话请关注、点赞、转发 谢谢!

[外链图片转存中…(img-jAJsnIVg-1712526269886)]

《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值