Android开发实现自定义水平滚动的容器示例
本文实例讲述了Android开发实现自定义水平滚动的容器。分享给大家供大家参考,具体如下:
public class HorizontalScrollView extends ViewGroup { //手势 private GestureDetector mGestureDetector; private HorizontalScroller mScroller; private int curID; //快速滑动 private boolean isFlying; //--回调函数------------------------------------- private OnChangeListener mListener; public void setOnChangeListener(OnChangeListener listener) { if (listener != null) { mListener = listener; } } public interface OnChangeListener{ void move2dest(int curID); } public HorizontalScrollView(Context context) { this(context, null); } public HorizontalScrollView(Context context, AttributeSet attrs) { this(context, attrs, 0); } public HorizontalScrollView(Context context, AttributeSet attrs, int defStyle) { super(context, attrs, defStyle); mScroller = new HorizontalScroller(); isFlying = false; initGesture(context); } @Override protected void onLayout(boolean changed, int l, int t, int r, int b) { // 模向移动, for (int i = 0; i < getChildCount(); i++) { View view = getChildAt(i); //给水平方向的每个view定位 view.layout(i * getWidth(), 0, getWidth() + i * getWidth(), getHeight()); } } @Override protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { for (int i = 0; i < getChildCount(); i++) { View view = getChildAt(i); view.measure(widthMeasureSpec, heightMeasureSpec); } super.onMeasure(widthMeasureSpec, heightMeasureSpec); } @Override public boolean onTouchEvent(MotionEvent event) { mGestureDetector.onTouchEvent(event); switch (event.getAction()) { case MotionEvent.ACTION_DOWN: if (!isFlying) { move2dest(); } isFlying = false; break; case MotionEvent.ACTION_MOVE: break; case MotionEvent.ACTION_UP: break; default: break; } return true; } public void move2dest() { // int destID = (getScrollX() + getWidth() / 2) / getWidth(); move2dest(destID); } public void move2dest(int destID) { curID = destID; if (destID > getChildCount() - 1) { destID = getChildCount() - 1; } if (mListener != null) { mListener.move2dest(curID); } int distance = (int) (destID * getWidth() - getScrollX()); // scrollBy(distance, 0); mScroller.startScroll(getScrollX(), getScrollY(), distance, 0); invalidate(); } /** * invalidate()此方法会触发下面的方法 */ @Override public void computeScroll() { // 如果存在偏移,就不断刷新 if (mScroller.computeScrollOffset()) { scrollTo(mScroller.getCurrX(), 0); invalidate(); } super.computeScroll(); } private void initGesture(Context context) { mGestureDetector = new GestureDetector(context, 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) { scrollBy((int) distanceX, 0); return false; } @Override public void onLongPress(MotionEvent e) { } @Override /** * 快速滑动时 */ public boolean onFling(MotionEvent e1, MotionEvent e2, float velocityX, float velocityY) { isFlying = true; if (curID > 0 && velocityX > 0) {// 表示向左移 move2dest(curID - 1); } else if (curID < getChildCount() && velocityX < 0) { move2dest(curID + 1);// 向右 } else { move2dest();// 移到原位 } return false; } @Override public boolean onDown(MotionEvent e) { return false; } }); } }
/** * 位移计算工具类 * * @author chenlin * */ public class HorizontalScroller { private int startX; private int startY; private int distanceX; private int distanceY; private int currentX; private int currentY; private long startTime; private long duration = 1000L; private boolean isFinish; /** * * @param scrollX * x坐标 * @param scrollY * y坐标 * @param distanceX * X方向移动的距离 * @param distanceY * y方向移动的距离 */ public void startScroll(int scrollX, int scrollY, int distanceX, int distanceY) { startX = scrollX; startY = scrollY; this.distanceX = distanceX; this.distanceY = distanceY; isFinish = false; startTime = SystemClock.uptimeMillis(); } /** * 计算偏移量, * * @return true 还在移动 false:移动已经停止 */ public boolean computeScrollOffset() { if (isFinish) { return false; } long timePassed = SystemClock.uptimeMillis() - startTime; if (timePassed < duration) { currentX = (int) (startX + distanceX * timePassed / duration); currentY = (int) (startY + distanceY * timePassed / duration); System.out.println("currentX:::" + currentX); } else if (timePassed >= duration) { currentX = startX + distanceX; currentY = startY + distanceY; isFinish = true; } return true; } public int getCurrX() { return currentX; } public void setCurrentX(int currentX) { this.currentX = currentX; } public int getCurrentY() { return currentY; } public void setCurrentY(int currentY) { this.currentY = currentY; } }
使用方法
public class ScrollActivity extends Activity implements OnCheckedChangeListener, OnChangeListener { private int[] ids = { R.drawable.a1, R.drawable.a2, R.drawable.a3, R.drawable.a4, R.drawable.a5, R.drawable.a6 }; private HorizontalScrollView mView; private LinearLayout mLayout; private RadioGroup mGroup; @Override protected void onCreate(Bundle savedInstanceState) { requestWindowFeature(Window.FEATURE_NO_TITLE); super.onCreate(savedInstanceState); setContentView(R.layout.activity_myscrollview); init(); } private void init() { mLayout = (LinearLayout) findViewById(R.id.body_layout); mGroup = (RadioGroup) findViewById(R.id.radiogroup); mView = new HorizontalScrollView(this); for (int i = 0; i < ids.length; i++) { ImageView imageView = new ImageView(this); imageView.setBackgroundResource(ids[i]); mView.addView(imageView); } mLayout.addView(mView); // 随便添加一个view View view = getLayoutInflater().inflate(R.layout.activity_progressview, null); mView.addView(view, 3); for (int i = 0; i < mView.getChildCount(); i++) { RadioButton radioButton = new RadioButton(this); // 设置id的编号 radioButton.setId(i); mGroup.setOrientation(LinearLayout.HORIZONTAL); mGroup.addView(radioButton); if (i == 0) { radioButton.setChecked(true); } } mGroup.setOnCheckedChangeListener(this); mView.setOnChangeListener(this); } @Override public void onCheckedChanged(RadioGroup group, int checkedId) { mView.move2dest(checkedId); } @Override public void move2dest(int curID) { RadioButton radioButton = (RadioButton) mGroup.getChildAt(curID); radioButton.setChecked(true); } }
布局文件
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="fill_parent" android:layout_height="fill_parent" android:orientation="vertical" > <RadioGroup android:id="@+id/radiogroup" android:layout_width="wrap_content" android:layout_height="wrap_content" > </RadioGroup> <LinearLayout android:id="@+id/body_layout" android:layout_width="match_parent" android:layout_height="match_parent" android:orientation="vertical" > </LinearLayout> </LinearLayout>
更多关于Android相关内容感兴趣的读者可查看本站专题:《Android基本组件用法总结》、《Android开发入门与进阶教程》、《Android布局layout技巧总结》、《Android视图View技巧总结》、《Android编程之activity操作技巧总结》、《Android资源操作技巧汇总》及《Android控件用法总结》
希望本文所述对大家Android程序设计有所帮助。
相关文章
Flutter StreamBuilder组件实现局部刷新示例讲解
日常使用最多的局部刷新为Provider状态管理 Selector,今天分享flutter框架自带的StreamBuilder组件,该组件可做到局部刷新,使用简单且轻便2022-11-11Android实现3种侧滑效果(仿qq侧滑、抽屉侧滑、普通侧滑)
这篇文章主要为大家详细介绍了Android实现多种侧滑效果,包括仿qq侧滑,抽屉侧滑,普通侧滑三种效果,具有一定的参考价值,感兴趣的小伙伴们可以参考一下2017-04-04Android package属性、package name和Application ID三者的联系及区别
这篇文章主要介绍了Android package属性、package name和Application ID三者的联系及区别的相关资料,需要的朋友可以参考下2016-12-12Android编程使用自定义shape实现shadow阴影效果的方法
这篇文章主要介绍了Android编程使用自定义shape实现shadow阴影效果的方法,涉及Android中xml文件布局的相关技巧,具有一定参考借鉴价值,需要的朋友可以参考下2015-11-11android studio3.0以上如何通过navicat访问SQLite数据库文件
这篇文章主要介绍了android studio3.0以上如何通过navicat访问SQLite数据库文件,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧2020-06-06什么是Android静默拍摄 Android静默拍摄app制作方法
这篇文章主要告诉大家什么是Android静默拍摄,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下2017-03-03Android 中从屏幕左下角弹出Dialog动画效果的实现代码
这篇文章主要介绍了Android 中从屏幕左下角弹出Dialog动画效果的实现代码,非常不错,具有参考借鉴价值,需要的朋友可以参考下2016-12-12
最新评论