PS:
该篇博客已经deprecated,不再维护,详情请参见
站在源码的肩膀上全解Scroller工作机制
http://blog.csdn.net/lfdfhl/article/details/53143114
MainActivity如下:
package cc.uu;
import android.os.Bundle;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;
import android.widget.TextView;
import android.app.Activity;
/**
* Demo描述:
* scrollTo()和scrollBy()的理解以及使用
*
* 原理说明:
* 1 其实View是没有边界,在屏幕上看到的只是View的一部分而已
* 2 scrollTo()和scrollBy()的本质一样只是表现形式略有不同
* 与这两个方法密切相关的两个变量mScrollX和mScrollY在
* View的源码中可以看到:
*
* //The offset,in pixels,by which the content of this view is scrolled horizontally.
* protected int mScrollX;
*
* //The offset,in pixels,by which the content of this view is scrolled vertically.
* protected int mScrollY;
*
* 通过文档描述可知:
* mScrollX和mScrollY表示:View的内容(content)相对于View本身在水平或垂直方向的偏移量.
*
* scrollTo(int x, int y)方法:
* 将一个视图的内容移动到指定位置.此时偏移量 mScrollX,mScrollY就分别等于x,y.
* 默认情况下 mScrollX和mScrollY均为0
*
* scrollBy(int x, int y)方法:
* 在现有的基础上继续移动视图的内容.
* 在该方法的源码很简单,也体现了这一点,如下:
* public void scrollBy(int x, int y) {
* scrollTo(mScrollX + x, mScrollY + y);
* }
* 默认情况下 mScrollX和mScrollY均为0
*
* 再次强调和注意:
* scrollTo()和scrollBy()移动的只是View的内容,但是View的背景是不移动的.
* 为了体现这点,在该示例中为View添加了背景色.
*
* 继续上面问题的延伸:
* 假如一个ViewGroup(比如XXXLayout)调用了scrollTo(By)()方法会怎样?
* 它的Content(即它所有的子View)都会移动,这点在下个例子中可以看到.
*
*
* 3 scrollTo(int x,int y)和scrollBy(int x,int y)方法的坐标说明
* 比如我们对于一个TextView调用scrollTo(0,25)
* 那么该TextView中的content(比如显示的文字:Hello)会怎么移动呢?
* 向下移动25个单位?不,恰好相反.
* 这是为什么呢?
* 因为调用这两个方法会导致视图重绘.
* 即调用public void invalidate(int l, int t, int r, int b)方法.
* 此处的l,t,r,b四个参数就表示View原来的坐标.
* 在该方法中最终会调用:
* tmpr.set(l - scrollX, t - scrollY, r - scrollX, b - scrollY);
* p.invalidateChild(this, tmpr);
* 其中tmpr是ViewParent,tmpr是Rect,this是原来的View.
* 通过这两行代码就把View在一个Rect中重绘.
* 请注意第一行代码:
* 原来的l和r均减去了scrollX
* 原来的t和b均减去了scrollY
* 就是说scrollX如果是正值,那么重绘后的View的宽度反而减少了;反之同理
* 就是说scrollY如果是正值,那么重绘后的View的高度反而减少了;反之同理
* 所以,TextView调用scrollTo(0,25)和我们的理解相反
*
* scrollBy(int x,int y)方法与上类似,不再赘述.
*
*
* 该示例的说明,请参加下面的代码注释
*
*
*
*
* 备注说明:
* 使用scrollTo(By)()方法移动过程较快而且比较生硬.
* 为了优化scrollTo(By)()的滑动过程可采用Scroller类.
* 该类源码第一句This class encapsulates scrolling.
* 就指明了该类的目的:封装了滑动过程.
* 在后面的示例中,将学习到Scroller的使用.
*
*/
public class MainActivity extends Activity {
private TextView mTextView;
private Button mLeftButton;
private Button mRightButon;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
init();
}
private void init(){
mTextView=(TextView) findViewById(R.id.textView);
mLeftButton=(Button) findViewById(R.id.leftButton);
mLeftButton.setOnClickListener(new ClickListenerImpl());
mRightButon=(Button) findViewById(R.id.rightButton);
mRightButon.setOnClickListener(new ClickListenerImpl());
}
/**
* 示例说明:
* 1 每次点击leftButton的时候
* 1.1 调用scrollBy()让mTextView的内容(即文字)在原本偏移的基础上往左移30
* 1.2 调用scrollBy()让mLeftButton的内容(即文字)在原本偏移的基础上也往左移30
* 2 每次点击rightButton的时候
* 2.1 调用scrollTo()让mTextView的内容(即文字)直接往右偏移30,而不管以前的基础(即 mScrollX和mScrollY)
* 3 连续几次点击leftButton会看到mTextView的内容(即文字)每点一次都会往左移动30,
* 然后再点击一次rightButton会看到mTextView的内容(即文字)直接一次性到了往右30的位置,而
* 不是慢慢移动过去.
* 这么操作
* 1 很好的体现了这两个方法的区别.
* 2 直观地看了scrollTo()方法的效用,它是不管以前的偏移量的.
* 4 在该例中也可以看到调用这两个方法时,View的背景是没有移动.移动的是内容.
*/
private class ClickListenerImpl implements OnClickListener {
@Override
public void onClick(View v) {
switch (v.getId()) {
case R.id.leftButton:
//让mTextView的内容往左移
mTextView.scrollBy(30, 0);
//让mLeftButton的内容也往左移
mLeftButton.scrollBy(20, 0);
break;
case R.id.rightButton:
//让mTextView的内容往右移直接到-30的位置
mTextView.scrollTo(-30, 0);
break;
default:
break;
}
}
}
}
main.xml如下:
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
>
<TextView
android:id="@+id/textView"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@string/hello_world"
android:layout_centerHorizontal="true"
android:layout_marginTop="20dip"
android:background="@android:color/darker_gray"
/>
<Button
android:id="@+id/leftButton"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="TextView移向屏幕的左边"
android:layout_centerHorizontal="true"
android:layout_marginTop="150dip"/>
<Button
android:id="@+id/rightButton"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="TextView移向屏幕的右边"
android:layout_centerHorizontal="true"
android:layout_marginTop="250dip"/>
</RelativeLayout>