`

网上流传的workspace类的解析

 
阅读更多

 

几个重要类的解释(每天增加,现在不完整):
  1. Scroller :该类封装了一个滑动的过程。在构造函数中传递滑动的持续时间以及指定滑动动作可以持续的最大时间。从设计模式上讲,它是一个model类型,也就是,当我们调用它的某个函数(设为afun)来记录我们移动到哪里(此时整个布局并没有实际移动),而在绘图的时候,通过某个函数(设为bfun)根据afun的设置的model来描绘出view现在应该在哪里(其实是将整个viewgroup进行了移动)。这里的afun函数就是Scroller的startScroll()函数,而bfun就是view类中有的computeScroll()函数(看下面的代码,它内部调用scrollTo函数来实现实际移动),从它的doc可以看出,我们可以把它看成是专门为view类移动而设计的函数。当我们重载了computeScroll时,系统就会在重绘时调用该函数。这里有个问题就是每次computeScroll时,是一次性的到达原先指定的目的还是分多次的进行调用?显然,为了达到一个“动画”的过程,就好像我一次向左移动x个px,分n次完成,这样就出现了一个速度,有了速度,我们才能看见有“移动”的效果。这个速度就是根据Scroller的computeScrollOffset()函数算出来的,也就是每次我们要移动整个视图到哪里时,都通过先调用该方法来计算,然后调用Scroller的getCurrX()和getCurrY()明确此次的目标,而Scroller的状态isFinshed()函数将返回false,直到getCurrX()和getCurrY()返回的值是通过startScroll设置的值,此时isFinshed()返回true。还有,当我们在两次computeScroll()之间调用了或者发生了startScroll时,将会产生新的移动方案,也就是原来的方案将会被覆盖而使用新的移动方案。
  2. ViewConfiguration类: View类及子类的配置信息(所以有很多静态函数):
    包含给Ui使用的标准静态函数及函数,比如超时、大小和距离。比如,长按(LongPressed)的标准是多少毫秒才算,两次tap的间隔多长才算作是两次的tap。
    使用ViewConfiguration.get(context)来获取该实例对象。
    方法列举:
    getScaledTouchSlop()返回对一个视图touch时应该忽略的最小移动像素。因为即使你是tap,也可能造成移动了几个像素,那么这里将定义移动了大于该函数返回值的像素点数才被定义为移动操作。
修剪后的源代码如下:
package cn.anycall.ju;

import android.content.Context;
import android.graphics.Canvas;
import android.util.AttributeSet;
import android.util.Log;
import android.view.MotionEvent;
import android.view.VelocityTracker;
import android.view.View;
import android.view.ViewConfiguration;
import android.view.ViewGroup;
import android.widget.Scroller;

/**
* 仿Launcher中的WorkSapce,可以左右滑动切换屏幕的类
* 
*/
public class ScrollLayout extends ViewGroup {

private static final String TAG = "ScrollLayout";
private Scroller mScroller;//滚动封装类
private VelocityTracker mVelocityTracker;//测量touch事件的触发速度

private int mCurScreen;//当前屏幕
private int mDefaultScreen = 0;//默认屏幕

private static final int TOUCH_STATE_REST = 0;//TOUCH状态为静止
private static final int TOUCH_STATE_SCROLLING = 1;//TOUCH状态为滚动

private static final int SNAP_VELOCITY = 600;//瞬时速度

private int mTouchState = TOUCH_STATE_REST;//当前状态
private int mTouchSlop;
private float mLastMotionX;//最终的X方向坐标
private float mLastMotionY;//最终的Y方向坐标

private int currentScreenIndex = 0;//当前屏幕索引值

public int getCurrentScreenIndex() {
return currentScreenIndex;
}

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

public ScrollLayout(Context context, AttributeSet attrs) {
this(context, attrs, 0);
// TODO Auto-generated constructor stub
}

public ScrollLayout(Context context, AttributeSet attrs, int defStyle) {
super(context, attrs, defStyle);
// TODO Auto-generated constructor stub
mScroller = new Scroller(context);

mCurScreen = mDefaultScreen;
mTouchSlop = ViewConfiguration.get(getContext()).getScaledTouchSlop();
}

@Override
protected void onLayout(boolean changed, int l, int t, int r, int b) {
// TODO Auto-generated method stub
int childLeft = 0;
final int childCount = getChildCount();
System.out.println("childCount=" + childCount);
for (int i = 0; i < childCount; i++) {
final View childView = getChildAt(i);
if (childView.getVisibility() != View.GONE) {
final int childWidth = childView.getMeasuredWidth();
childView.layout(childLeft, 0, childLeft + childWidth,
childView.getMeasuredHeight());
childLeft += childWidth;
}
}
}

@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
Log.e(TAG, "onMeasure");
super.onMeasure(widthMeasureSpec, heightMeasureSpec);

final int width = MeasureSpec.getSize(widthMeasureSpec);
final int widthMode = MeasureSpec.getMode(widthMeasureSpec);
if (widthMode != MeasureSpec.EXACTLY) {
throw new IllegalStateException(
"ScrollLayout only canmCurScreen run at EXACTLY mode!");
}

final int heightMode = MeasureSpec.getMode(heightMeasureSpec);
if (heightMode != MeasureSpec.EXACTLY) {
throw new IllegalStateException(
"ScrollLayout only can run at EXACTLY mode!");
}

// The children are given the same width and height as the scrollLayout
final int count = getChildCount();
for (int i = 0; i < count; i++) {
getChildAt(i).measure(widthMeasureSpec, heightMeasureSpec);
}
System.out.println("moving to screen " + mCurScreen);
scrollTo(mCurScreen * width, 0);//移动到当前屏幕,屏幕index*屏幕宽
}

/**
* 拖动移动(非快速移动)
* According to the position of current layout scroll to the destination
* page.getScrollX返回值即为拖动以后停留的位置,通过该位置与屏幕宽度相除就可以确定该位置在哪一页,
* 若在该基础上添加半页的大小,再舍去小数,就可以确定是停留在原来页还是向左或则右移动到邻居页
*/
public void snapToDestination() {
final int screenWidth = getWidth();
int sx=getScrollX();
final int destScreen = (sx + screenWidth / 2) / screenWidth;//添加半块屏幕的大小,使得滑动超过半块屏幕算一块,少于半块则回至原处
snapToScreen(destScreen);
}
/**
* 封装scroller的startScroll来实现
* @param whichScreen
*/
public void snapToScreen(int whichScreen) {
// get the valid layout page
whichScreen = Math.max(0, Math.min(whichScreen, getChildCount() - 1));
if (getScrollX() != (whichScreen * getWidth())) {//当拖动后的位置和目的页起始位置不一致,则进行移动

final int delta = whichScreen * getWidth() - getScrollX();
mScroller.startScroll(getScrollX(), 0, delta, 0,
Math.abs(delta) * 2);//相当于移动一个像素点要2毫秒
mCurScreen = whichScreen;
invalidate(); // Redraw the layout
}
}

public void setToScreen(int whichScreen) {//直接滑动一个屏幕
whichScreen = Math.max(0, Math.min(whichScreen, getChildCount() - 1));
mCurScreen = whichScreen;
scrollTo(whichScreen * getWidth(), 0);
}

public int getCurScreen() {
return mCurScreen;
}

@Override
public void computeScroll() {
// TODO Auto-generated method stub
if (mScroller.computeScrollOffset()) {
scrollTo(mScroller.getCurrX(), mScroller.getCurrY());
postInvalidate();
}
}

@Override
public boolean onTouchEvent(MotionEvent event) {
// TODO Auto-generated method stub

if (mVelocityTracker == null) {
mVelocityTracker = VelocityTracker.obtain();
}
mVelocityTracker.addMovement(event);

final int action = event.getAction();
final float x = event.getX();
final float y = event.getY();

switch (action) {
case MotionEvent.ACTION_DOWN:
Log.e(TAG, "event down!");
if (!mScroller.isFinished()) {
mScroller.abortAnimation();
}
mLastMotionX = x;
break;

case MotionEvent.ACTION_MOVE:
int deltaX = (int) (mLastMotionX - x);
mLastMotionX = x;

scrollBy(deltaX, 0);
break;

case MotionEvent.ACTION_UP:
Log.e(TAG, "event : up");

// if (mTouchState == TOUCH_STATE_SCROLLING) {
final VelocityTracker velocityTracker = mVelocityTracker;
velocityTracker.computeCurrentVelocity(1000);
int velocityX = (int) velocityTracker.getXVelocity();

Log.e(TAG, "velocityX:" + velocityX);

if (velocityX > SNAP_VELOCITY && mCurScreen > 0) {
// Fling enough to move left
Log.e(TAG, "snap left");
onScreenChangeListener.onScreenChange(mCurScreen - 1);
System.out.println("mCurScreen=" + (mCurScreen - 1));
snapToScreen(mCurScreen - 1);
} else if (velocityX < -SNAP_VELOCITY
&& mCurScreen < getChildCount() - 1) {
// Fling enough to move right
Log.e(TAG, "snap right");
onScreenChangeListener.onScreenChange(mCurScreen + 1);
//只往右移动才加载数据
onScreenChangeListenerDataLoad.onScreenChange(mCurScreen+1);
snapToScreen(mCurScreen + 1);
} else {
snapToDestination();
}

if (mVelocityTracker != null) {
mVelocityTracker.recycle();
mVelocityTracker = null;
}
// }
mTouchState = TOUCH_STATE_REST;
break;
case MotionEvent.ACTION_CANCEL:
mTouchState = TOUCH_STATE_REST;
break;
}

return true;
}
//截获(返回true)所有事件,当当前状态是滑动状态时,在此截获将会使事件停留在该viewgroup中被处理
//否则将被传递到下一个viewgroup中处理。
@Override
public boolean onInterceptTouchEvent(MotionEvent ev) {
// TODO Auto-generated method stub
Log.e(TAG, "onInterceptTouchEvent-slop:" + mTouchSlop);

final int action = ev.getAction();
if ((action == MotionEvent.ACTION_MOVE)
&& (mTouchState != TOUCH_STATE_REST)) {
return true;
}

final float x = ev.getX();
final float y = ev.getY();

switch (action) {
case MotionEvent.ACTION_MOVE:
final int xDiff = (int) Math.abs(mLastMotionX - x);
if (xDiff > mTouchSlop) {
mTouchState = TOUCH_STATE_SCROLLING;

}
break;

case MotionEvent.ACTION_DOWN:
mLastMotionX = x;
mLastMotionY = y;
mTouchState = mScroller.isFinished() ? TOUCH_STATE_REST
: TOUCH_STATE_SCROLLING;
break;

case MotionEvent.ACTION_CANCEL:
case MotionEvent.ACTION_UP:
mTouchState = TOUCH_STATE_REST;
break;
}

return mTouchState != TOUCH_STATE_REST;
}

//分页监听
public interface OnScreenChangeListener {
void onScreenChange(int currentIndex);
}

private OnScreenChangeListener onScreenChangeListener;

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


//动态数据监听
public interface OnScreenChangeListenerDataLoad {
void onScreenChange(int currentIndex);
}
private OnScreenChangeListenerDataLoad onScreenChangeListenerDataLoad;

public void setOnScreenChangeListenerDataLoad(OnScreenChangeListenerDataLoad onScreenChangeListenerDataLoad) {
this.onScreenChangeListenerDataLoad = onScreenChangeListenerDataLoad;
}

}
 

 

分享到:
评论

相关推荐

    java web workspace 02

    9. **Eclipse或IntelliJ IDEA项目结构**:`workspace`可能基于这些流行的Java IDE之一,包含相应的项目文件和配置,如`.project`、`.classpath`等。 10. **版本控制文件**:如`.git`目录,可能表示这个工作空间使用...

    Java 解析国标hj212协议

    5. **直接调用**:这可能指的是项目中提供了一些预设好的类或方法,可以直接在其他代码中调用来解析hj212协议的数据。例如,可能有一个名为`Hj212Parser`的类,包含`parseData()`方法,用于解析接收到的协议数据。 ...

    Android Blockly积木编程源码对Workspace中的block数据保存及读取的流程,及改造原生代码实现Trash垃圾桶中的block保存及读取

    在Android平台上,Blockly是一款流行的图形化编程工具,它允许用户通过拖放积木块来创建程序,特别适合教育和初学者。"Android Blockly积木编程源码对Workspace中的block数据保存及读取的流程,及改造原生代码实现...

    基于C++的mfc框架的酒店客房管理系统(源码+数据库).zip

    本篇将深入解析一款基于C++编程语言,采用Microsoft Foundation Classes (MFC)框架开发的酒店客房管理系统。通过这款系统的源码分析,我们可以了解到C++在开发桌面应用程序时的强大功能以及MFC框架的应用技巧。 MFC...

    java提取补丁文件

    1. **Eclipse**:开发者可以通过导入补丁项目(`Import -&gt; Existing Projects into Workspace`),然后将补丁中的类文件复制到目标项目的相应位置。此外,Eclipse的插件系统允许开发人员创建和应用特定的更新站点,...

    16秋北邮大学英语3阶段作业.doc

    Eclipse是一款流行的Java集成开发环境(IDE),它提供了多种功能,如代码编辑、调试、构建工具等。在Eclipse中,透视图(Perspective)是一种工作区布局,每个透视图定义了特定任务所需视图的配置。用户可以有多个...

    nutch1.2 java的project

    1. **导入项目**:在Eclipse中选择“File” &gt; “Import” &gt; “Existing Projects into Workspace”,然后浏览到下载的`nutch1.2+Project`目录,导入项目。 2. **添加库**:确保你的Eclipse环境中已经安装了Apache ...

    不依赖OFFICE的XLS的高速导入程序(未经测试)

    PowerBuilder是一种流行的、基于对象的开发环境,主要用于构建企业级的Windows应用程序。而“XLS导入”功能对于许多业务系统来说是重要的,因为它们经常需要与Excel数据交换,如数据分析、报告生成等。 压缩包内的...

    java常用英语

    - **Eclipse**:一款流行的开源集成开发环境(IDE),主要用于Java应用程序的开发。 - **Existing Project into Workspace**:将已存在的项目导入到工作空间中。 - **File**:文件菜单,包含创建、打开、保存等文件...

    Jupyter notebook使用详解

    Jupyter Notebook 是一款非常流行的开源 Web 应用程序,用于创建和共享包含实时代码、方程式、可视化效果和叙述文本的文档。它广泛应用于数据分析、统计建模、机器学习等领域。 - **安装**: - **Anaconda 自带 ...

    Python库 | kodexa-4.1.59.tar.gz

    - **文档理解**:通过自然语言处理(NLP)和计算机视觉技术,Kodexa能解析和理解各种格式的文档,包括PDF、Word、Excel等,提取关键信息并结构化存储。 - **智能分析**:Kodexa库包含机器学习模型,用于识别模式、...

    PB编写的电子地图源码.zip

    PowerBuilder是一种流行的、基于事件驱动的编程环境,主要用于创建企业级的应用程序,尤其在数据库应用方面有广泛应用。在这个场景中,PB被用来构建一个能够显示、操作和分析地图数据的系统。 描述中的“PB源码”...

    eclipse中class乱码GBK-UTF-8转换工具

    Eclipse是一个流行的Java开发集成环境,它提供了一流的代码编辑、调试和项目管理功能。然而,当涉及到字符编码时,如GBK和UTF-8之间的转换,可能会遇到乱码问题。这篇博客文章“eclipse中class乱码GBK-UTF-8转换工具...

    MyEclipse6中文教程

    MyEclipse是一款流行的集成开发环境(IDE),由Genuitec公司开发,基于Eclipse平台提供对Java EE的支持。MyEclipse6中文教程是为了解决学习者在使用MyEclipse进行Java应用程序开发过程中的种种问题而编写的指南。...

    MyEclipse经典教材

    - 作为一款流行的Java开发工具,MyEclipse支持多种开发框架和技术栈,例如Struts2、Spring2、Hibernate3等,使其成为进行企业级Java应用开发的理想选择。 #### 二、MyEclipse6 Java开发中文教程概述 - **作者信息*...

    java打包步骤

    #### 标题解析 标题“Java打包步骤”指明了本文的主题,即介绍如何将一个Java项目通过特定的步骤打包成可执行文件,这里的可执行文件特指Windows平台下的`.exe`文件。 #### 描述分析 描述部分进一步明确了文章的...

Global site tag (gtag.js) - Google Analytics