当我在使用scollview时发现一个有趣的现象,在按钮点击的事件中,我们进行对与scollview的位置设置scollTo方法是完全ok的,但是当我们想要在初始化过程中就进行对scollview滑动到的位置进行设定时,就无效了。
网上找了一些资料,都说是用handler.postDelayed方法,进行一个延迟操作,试过了之后,并没有什么用,
由此对scollview进行研究:
1.scrollTo() 是直接指定滚动条的位置, 但是由于这个动作不是单纯关于 ScrollView 而已, 还要根据 ScrollView 里面包含的View 的实际信息. 所以这动作必须在页面加载完成以后才能执行.
2 那么如何判断页面加载完成呢,在activity中有这样一个方法:
@Override
public void onWindowFocusChanged(boolean hasFocus) {
// TODO Auto-generated method stub
super.onWindowFocusChanged(hasFocus);
if (hasFocus) {
//do something 当activity 得到或者失去焦点时,会执行的方法,得到焦点返回 true,反之
}
}
3.activity中页面初始化渲染完成之后,进行scollview位置的设定:
mScrollView.post(new Runnable() {
@Override
public void run() {
mScrollView.scrollTo(0, 1000); // 0 代表x轴移动的距离, 1000表示y轴移动的距离
}
});
4下面进行源码剖析:
scollview源码:
/**
* {@inheritDoc}
*
* <p>This version also clamps the scrolling to the bounds of our child.
*/
@Override
public void scrollTo(int x, int y) {
// we rely on the fact the View.scrollBy calls scrollTo.
if (getChildCount() > 0) {
View child = getChildAt(0);
x = clamp(x, getWidth() - mPaddingRight - mPaddingLeft, child.getWidth());
y = clamp(y, getHeight() - mPaddingBottom - mPaddingTop, child.getHeight());
if (x != mScrollX || y != mScrollY) {
super.scrollTo(x, y);
}
}
}
它的父类View:
public void scrollTo(int x, int y) {
if (mScrollX != x || mScrollY != y) {
int oldX = mScrollX;
int oldY = mScrollY;
mScrollX = x;
mScrollY = y;
onScrollChanged(mScrollX, mScrollY, oldX, oldY);
if (!awakenScrollBars()) {
invalidate();
}
}
}
由此可见:
if (mScrollX != x || mScrollY != y) 这个判断语句有没有通过是关键所在. 这也间接说明了, 在Activity 没初始化完成, ScrollView 对象获取的一些信息是不准确的, 直接导致了scrollTo() 方法无效.
5.于是我们得出结论,要想在初始化中设置horizontalscollview的位置,需要在activity页面渲染完成之后,进行设置,也就是在onWindowFocusChanged 方法中设置偏移位置,则生效。