自己的笔记+别人笔记我的提取
1、获取View宽高
(1)Activity/View#onWindowFocusChanged
(2)view.post(runnable)
(3)ViewTreeObserver
mLeftLayout = (RelativeLayout) findViewById(R.id.left_direction);
ViewTreeObserver treeObserver = mLeftLayout.getViewTreeObserver(); treeObserver.addOnGlobalLayoutListener(new ViewTreeObserver.OnGlobalLayoutListener() { @Override public void onGlobalLayout() {
mLeftLayout.getViewTreeObserver().removeOnGlobalLayoutListener(this);leftWidth = mLeftLayout.getWidth() ; leftHeight = mLeftLayout.getHeight(); Log.d(TAG, "onGlobalLayout: "+leftWidth); Log.d(TAG, "onGlobalLayout: "+leftHeight); }});
我们知道在 onCreate() 中 View.getWidth 和 View.getHeight 无法获得一个 view 的高度和宽度,这是因为 View 组件布局要在 onResume() 回调后完成。所以现在需要使用 getViewTreeObserver().addOnGlobalLayoutListener() 来获得宽度或者高度。这是获得一个 view 的宽度和高度的方法之一。
但是需要注意的是 OnGlobalLayoutListener 可能会被多次触发,因此在得到了高度之后,要将OnGlobalLayoutListener 注销掉。
(4)view.measure(int widthMeasureSpec,int heightMeasureSpec)
2、dip和dp之间的相互转换
public class Transform { public static int dip2px(Context context, float dipValue){ final float scale = context.getResources().getDisplayMetrics().density; return (int)(dipValue * scale + 0.5f); } public static int px2dip(Context context, float pxValue){ final float scale = context.getResources().getDisplayMetrics().density; return (int)(pxValue / scale + 0.5f); } }3.TextView设置不同字体样式
text = new SpannableString("happy everyday"); text.setSpan(new ForegroundColorSpan(getResources().getColor(R.color.colorAccent)), 0, 4, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE); text.setSpan(new AbsoluteSizeSpan(20), 6, 13, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
关键方法:setSpan()
支持设置的字体格式(继承 CharacterSty 均可):
-
字体颜色-------ForegroundColorSpan
-
字体大小-------AbsoluteSizeSpan
-
背景颜色-------BackgroundColorSpan
-
超链接----------URLSpan
-
粗体、斜体----StyleSpan
-
删除线----------StrikethroughSpan
-
下划线----------UnderlineSpan
-
图片-------------ImageSpan
9.TextView各种padding解析、长度测量
View事件
1、VelocityTracker 速度追踪,用于追踪手指在滑动过程中的速度,包括水平和竖直方向的速度。首先,在View的onTouchEvent方法中追踪当前单击事件的速度:
VelocityTracker velocityTracker = VelocityTracker.obtain(); velocityTracker.addMovement(motionEvent);接着当我们想知道当前速度时,按一下方式获取:
velocityTracker.computeCurrentVelocity(1000); int xVelocity = (int) velocityTracker.getXVelocity(); int yVelocity = (int) velocityTracker.getYVelocity();以上含义为:没1s内滑动的像素数
最后不用时,调用clear方法来重置并回收内存:
velocityTracker.clear(); velocityTracker.recycle();
2、GestureDetector
手势检测。用于辅助检测用户的单击、滑动、长按、双击等行为。
3、Scroller
弹性滑动对象,用于实现View的弹性滑动。当使用View的ScrollTo/ScrollBy方法来进行滑动时,其过程是瞬间完成的,用户体验不好。而使用Scroller来实现有过渡效果的滑动,其过程是在一定时间间隔内完成的。Scroller本身无法让View弹性滑动,他需要和View的computeScroll方法配合使用才能共同完成这个功能。
Scroller mScroller = new Scroller(mContext); private void smoothScrollTo(int destX, int destY) { int scrollX = getScrollX(); int delta = destX - scrollX; mScroller.startScroll(scrollX, 0, delta, 0, 1000);//View内容的滑动 invalidate(); } @Override public void computeScroll() { if (mScroller.computeScrollOffset()) { scrollTo(mScroller.getCurrX(), mScroller.getCurrY());//View本身的滑动 postInvalidate(); } }弹性滑动
有三种方式实现:使用Scroller、通过动画、使用延时策略
其中通过动画示列:
ObjectAnimator.ofFloat(targetView,"translationX",0,100).setDuration(100).start();
View的工作原理
ViewGroup content = findViewById(android.R.id.content);//得到content content.getChildAt(0);//得到View
理解MeasureSpec
SpecMode有三类:
UNSPECIFIED:父容器不对View有任何限制,要多大有多大,一般用于系统内部,表示一种测量状态。
EXACTLY:父容器已经检测出View所需要的精确大小,此时View的最终大小就是SpecSize所指的值。对应于LayoutParams中的match_parent和具体数值两种模式。
AT_MOST:父容器指定一个可用大小即SpecSize,View的大小不能大于这个值,要看View的具体实现。对应于LayoutParams中的wrap_content。
MeasureSpec和LayoutParams的对应关系
对于顶级View(DecorView)和普通View来说,MeasureSpec的转换过程略有不同。
对于DecorView,其MeasureSpec由窗口的尺寸和其自生的LayoutParams来共同确定;
对于普通View,其MeasureSpec由父容器的MeasureSpec和自生的LayoutParams来共同决定,MeasureSpec一旦确定后,onMeasure中就可以确定View的测量宽/高。
getMeasuredWidth和getWidth的区别 http://blog.****.net/dmk877/article/details/49734869
①getMeasuredWidth方法获得的值是setMeasuredDimension方法设置的值,它的值在measure方法运行后就会确定
②getWidth方法获得是layout方法中传递的四个参数中的mRight-mLeft,它的值是在layout方法运行后确定的
③一般情况下在onLayout方法中使用getMeasuredWidth方法,而在除onLayout方法之外的地方用getWidth方法。