`

一个简单的界面拖动切换效果类ScrollViewGroup

阅读更多
网上找的一个简单的界面平滑切换类,我只改动了一点点代码。
该类不能循环切换!

import android.content.Context;
import android.util.AttributeSet;
import android.util.Log;
import android.view.GestureDetector;
import android.view.GestureDetector.OnGestureListener;
import android.view.MotionEvent;
import android.view.View;
import android.view.ViewConfiguration;
import android.view.ViewGroup;
import android.widget.Scroller;

public class ScrollViewGroup extends ViewGroup {

	private static final String TAG = "scroller";

	private Scroller scroller;

	private int currentScreenIndex;

	public int getCurrentScreenIndex() {
		return currentScreenIndex;
	}

	public void setCurrentScreenIndex(int currentScreenIndex) {
		this.currentScreenIndex = currentScreenIndex;
	}

	private GestureDetector gestureDetector;

	// 设置一个标志位,防止底层的onTouch事件重复处理UP事件
	private boolean fling;

	public ScrollViewGroup(Context context, AttributeSet attrs, int defStyle) {
		super(context, attrs, defStyle);
		initView(context);
	}

	public ScrollViewGroup(Context context, AttributeSet attrs) {
		super(context, attrs);
		initView(context);
	}

	public ScrollViewGroup(Context context) {
		super(context);
		initView(context);
	}

	private void initView(final Context context) {
		this.scroller = new Scroller(context);

		this.gestureDetector = new GestureDetector(new OnGestureListener() {

			@Override
			public boolean onSingleTapUp(MotionEvent e) {
				return false;
			}

			@Override
			public void onShowPress(MotionEvent e) {
			}

			@Override
			public boolean onScroll(MotionEvent e1, MotionEvent e2,
					float distanceX, float distanceY) {
				if ((distanceX > 0 && currentScreenIndex < getChildCount() - 1)// 防止移动过最后一页
						|| (distanceX < 0 && getScrollX() > 0)) {// 防止向第一页之前移动
					scrollBy((int) distanceX, 0);
				}
				return true;
			}

			@Override
			public void onLongPress(MotionEvent e) {
			}

			@Override
			public boolean onFling(MotionEvent e1, MotionEvent e2,
					float velocityX, float velocityY) {
				Log.d(TAG, "min velocity >>>"
						+ ViewConfiguration.get(context).getScaledMinimumFlingVelocity()
						+ " current velocity>>" + velocityX);
				// 判断是否达到最小轻松速度,取绝对值的
				if (Math.abs(velocityX) > ViewConfiguration.get(context).getScaledMinimumFlingVelocity()) {
						if (velocityX > 0 && currentScreenIndex > 0) {//手指从左往右划
							Log.d(TAG, ">>>>fling to left");
							fling = true;
							scrollToScreen(currentScreenIndex - 1);
						} else if (velocityX < 0 && currentScreenIndex < getChildCount() - 1) {
							Log.d(TAG, ">>>>fling to right");
							fling = true;
							scrollToScreen(currentScreenIndex + 1);
						}
				}

				return true;
			}

			@Override
			public boolean onDown(MotionEvent e) {
				return false;
			}
		});
		
	}

	@Override
	protected void onLayout(boolean changed, int left, int top, int right,int bottom) {
		Log.d(TAG, ">>left: " + left + " top: " + top + " right: " + right
				+ " bottom:" + bottom);

		/**
		 * 设置布局,将子视图顺序横屏排列
		 */
		for (int i = 0; i < getChildCount(); i++) {
			View child = getChildAt(i);
			child.setVisibility(View.VISIBLE);
			child.measure(right - left, bottom - top);
			child.layout(0 + i * getWidth(), 0, getWidth() + i * getWidth(),
					getHeight());
		}
		
		//初始化显示第几个界面
		int delta = currentScreenIndex * getWidth() - getScrollX();
		scroller.startScroll(getScrollX(), 0, delta, 0, 0);
		invalidate();
	}

	@Override
	public void computeScroll() {
		if (scroller.computeScrollOffset()) {
			scrollTo(scroller.getCurrX(), 0);
			postInvalidate();
		}
	}

	@Override
	public boolean onTouchEvent(MotionEvent event) {
		gestureDetector.onTouchEvent(event);

		switch (event.getAction()) {
		case MotionEvent.ACTION_DOWN:
			break;
		case MotionEvent.ACTION_MOVE:
			break;
		case MotionEvent.ACTION_UP:
			if (!fling) {
				snapToDestination();
			}
			fling = false;
			break;
		default:
			break;
		}
		return true;
	}

	/**
	 * 切换到指定屏
	 * 
	 * @param whichScreen
	 */
	private void scrollToScreen(int whichScreen) {
		if (getFocusedChild() != null && whichScreen != currentScreenIndex
				&& getFocusedChild() == getChildAt(currentScreenIndex)) {
			getFocusedChild().clearFocus();
		}

		final int delta = whichScreen * getWidth() - getScrollX();
		scroller.startScroll(getScrollX(), 0, delta, 0, Math.abs(delta) * 2);
		invalidate();

		currentScreenIndex = whichScreen;
		if (onScreenChangeListener != null) {
			onScreenChangeListener.onScreenChange(currentScreenIndex);
		}
	}

	/**
	 * 根据当前x坐标位置确定切换到第几屏
	 */
	private void snapToDestination() {
		scrollToScreen((getScrollX() + (getWidth() / 2)) / getWidth());
	}

	public interface OnScreenChangeListener {
		void onScreenChange(int currentIndex);
	}
	private OnScreenChangeListener onScreenChangeListener;

	public void setOnScreenChangeListener(OnScreenChangeListener onScreenChangeListener) {
		this.onScreenChangeListener = onScreenChangeListener;
	}
}


三个切换小点:
import com.easymorse.scroll.ScrollViewGroup.OnScreenChangeListener;

import android.content.Context;
import android.util.AttributeSet;
import android.widget.ImageView;
import android.widget.LinearLayout;

public class PageControlView extends LinearLayout {
	private Context context;

	private int count;

	public void bindScrollViewGroup(ScrollViewGroup scrollViewGroup) {
		this.count=scrollViewGroup.getChildCount();
		generatePageControl(scrollViewGroup.getCurrentScreenIndex());
		
		scrollViewGroup.setOnScreenChangeListener(new OnScreenChangeListener() {
			
			@Override
			public void onScreenChange(int currentIndex) {
				// TODO Auto-generated method stub
				generatePageControl(currentIndex);
			}
		});
	}

	public PageControlView(Context context) {
		super(context);
		this.init(context);
	}
	public PageControlView(Context context, AttributeSet attrs) {
		super(context, attrs);
		this.init(context);
	}

	private void init(Context context) {
		this.context=context;
	}

	private void generatePageControl(int currentIndex) {
		this.removeAllViews();

		for (int i = 0; i < this.count; i++) {
			ImageView imageView = new ImageView(context);
			if (currentIndex == i) {
				imageView.setImageResource(R.drawable.page_indicator_focused);
			} else {
				imageView.setImageResource(R.drawable.page_indicator);
			}
			this.addView(imageView);
		}
	}
}


用法:
import android.app.Activity;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.ImageView;

public class ScrollDemosActivity extends Activity {

	@SuppressWarnings("unused")
	private static final String TAG = "scroller";

	private ScrollViewGroup viewGroup;

	private PageControlView pageControl;

	/** Called when the activity is first created. */
	@Override
	public void onCreate(Bundle savedInstanceState) {
		super.onCreate(savedInstanceState);

		setContentView(R.layout.main);
		viewGroup = (ScrollViewGroup) findViewById(R.id.scrollViewGroup);

		ImageView imageView = new ImageView(this);
		imageView.setImageDrawable(getResources().getDrawable(R.drawable.a1));
		viewGroup.addView(imageView);

		viewGroup.addView(View.inflate(this, R.layout.layout_0, null));

		imageView = new ImageView(this);
		imageView.setImageDrawable(getResources().getDrawable(R.drawable.a2));
		viewGroup.addView(imageView);

		viewGroup.setCurrentScreenIndex(2);
		
		pageControl=(PageControlView) findViewById(R.id.pageControl);
		pageControl.bindScrollViewGroup(viewGroup);
		
	}
	
}


布局:
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
	android:orientation="vertical" 
	android:layout_width="fill_parent"
	android:layout_height="fill_parent"
	>
	<EditText android:layout_width="fill_parent"
		android:layout_height="wrap_content"
		/>
	<LinearLayout
		android:orientation="horizontal" 
		android:layout_width="fill_parent"
		android:layout_height="fill_parent"
		android:layout_weight="1"
		>
		<EditText 
		android:layout_width="fill_parent"
		android:layout_height="wrap_content"
		android:layout_weight="2"
		/>
		<RelativeLayout
			android:id="@+id/myView"
			android:layout_width="fill_parent"
			android:layout_height="fill_parent" 
			android:layout_weight="1"
			>
			<com.easymorse.scroll.ScrollViewGroup
				android:id="@+id/scrollViewGroup" 
				android:layout_width="fill_parent"
				android:layout_height="fill_parent" 
				/>
			<com.easymorse.scroll.PageControlView 
				android:id="@+id/pageControl"
				android:layout_width="fill_parent" 
				android:layout_height="40px"
				android:background="#8f00000f" 
				android:layout_alignParentBottom="true"
				android:gravity="center" 
				/>
		</RelativeLayout>
	</LinearLayout>
	<EditText android:layout_width="fill_parent"
		android:layout_height="wrap_content"
		/>
</LinearLayout>
  • 大小: 39.8 KB
分享到:
评论
2 楼 114624915 2011-11-10  
为什么动态布局只显示左上角啊
1 楼 mailyiran200101 2011-11-08  
学习了,感谢

相关推荐

    ScrollViewGroup-master自定义动画框架原版注解

    `ScrollViewGroup-master`项目是一个专门针对ScrollView的自定义动画框架,它提供了丰富的注解,使得开发者能够更方便地实现各种滚动动画效果。这个框架的核心在于通过注解简化了动画逻辑,让开发者无需深入底层就能...

    android 自定义动画框架

    在Android开发中,自定义动画框架是一个非常关键的领域,它可以帮助开发者创造出独特且吸引人的用户界面,提升应用的用户体验。"android 自定义动画框架"这个主题主要涉及到Android平台上的动画实现技术,以及如何...

    Android自定义viewgroup 使用adapter适配数据(6)

    在这个场景中,我们看到一个自定义的ViewGroup子类`ScrollViewGroup`,它扩展了`ViewGroup`,目的是实现横向滚动的效果。`ScrollViewGroup`的核心是通过`Scroller`类来管理平滑滚动,以及使用`BaseAdapter`来适配...

    Android自定义ViewGroup实现可滚动的横向布局(2)

    首先,`ScrollViewGroup` 类继承了 `ViewGroup`,这表明它是一个可以包含多个子视图的容器,并且它添加了滚动功能。`ScrollViewGroup` 使用 `Scroller` 对象来处理平滑的滚动动画,这是Android系统提供的一种用于...

    Android自定义ViewGroup横向布局(1)

    在这个例子中,我们看到的是一个自定义的ViewGroup,名为ScrollViewGroup,它的目的是创建一个水平布局,能够容纳并排列多个子View。 首先,`ScrollViewGroup`继承自`ViewGroup`,这意味着它可以包含多个子View,并...

    Android自定义viewgroup可滚动布局 GestureDetector手势监听(5)

    - ScrollViewGroup继承自ViewGroup类,是一个自定义的横向滚动布局。 - 在init方法中,首先创建Scroller实例mScroller来控制滚动。接着获取屏幕宽度screenWidth,用于之后计算滚动的最大距离。 - 创建自定义的...

    Android自定义ViewGroup实现受边界限制的滚动操作(3)

    为了实现滚动功能,它使用了一个Scroller类的实例,Scroller是Android提供的一个辅助类,用于处理各种动画效果,包括滚动。在构造函数中,我们初始化了Scroller对象,并通过WindowManager获取了屏幕的宽度,这将在...

    深入理解Android中Scroller的滚动原理

    Scroller在Android中是一个核心组件,它主要用于实现View的平滑滚动效果,使得视图能够按照预设的时间和轨迹进行连续、平滑的移动,而不是简单地瞬移。Scroller本身并不直接改变View的位置,而是提供计算滚动过程的...

Global site tag (gtag.js) - Google Analytics