慕课网QQ侧滑界面一

1.    QQ侧滑原理(第一种):

HorizontalScorllView

如图,


中间黑色的是手机屏幕,左边绿色的是手机的Menu,右边紫色的是手机的Content,在HorizontalScorllView中,当一开始是可以是手机的Menu在手机屏幕中,也就是黑色区域,如果向左滑动手机屏幕则紫色区域被拖进手机屏幕,

用HorizontalScorllView的好处是当有ListView时不用判断滑动的方向,我们只需要判断滑动的偏移量,如果偏移量足够大则另一半区域拖入,否则不拖入。

而滑动的监听为:OnTouchEvent

蓝色的是手机屏幕即我们看到的区域,而Menu,Content都在绿色的区域,我们的滑动其实就是将绿色的区域左右移动

1.    代码区域:

1.  res/left_menu.xml


 <?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:background="@drawable/img_frame_background" >

    <LinearLayout
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_centerInParent="true"
        android:orientation="vertical" >

        <RelativeLayout
            android:layout_width="match_parent"
            android:layout_height="wrap_content" >

            <ImageView
                android:id="@+id/img01"
                android:layout_width="50dp"
                android:layout_height="50dp"
                android:layout_marginLeft="20dp"
                android:layout_marginTop="20dp"
                android:src="@drawable/img_1" />

            <TextView
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:layout_centerVertical="true"
                android:layout_marginLeft="20dp"
                android:layout_toRightOf="@id/img01"
                android:text="first item"
                android:textColor="#fff"
                android:textSize="20sp" />
        </RelativeLayout>

        <RelativeLayout
            android:layout_width="match_parent"
            android:layout_height="wrap_content" >

            <ImageView
                android:id="@+id/img02"
                android:layout_width="50dp"
                android:layout_height="50dp"
                android:layout_marginLeft="20dp"
                android:layout_marginTop="20dp"
                android:src="@drawable/img_2" />

            <TextView
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:layout_centerVertical="true"
                android:layout_marginLeft="20dp"
                android:layout_toRightOf="@id/img02"
                android:text="second item"
                android:textColor="#fff"
                android:textSize="20sp" />
        </RelativeLayout>

        <RelativeLayout
            android:layout_width="match_parent"
            android:layout_height="wrap_content" >

            <ImageView
                android:id="@+id/img03"
                android:layout_width="50dp"
                android:layout_height="50dp"
                android:layout_marginLeft="20dp"
                android:layout_marginTop="20dp"
                android:src="@drawable/img_3" />

            <TextView
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:layout_centerVertical="true"
                android:layout_marginLeft="20dp"
                android:layout_toRightOf="@id/img03"
                android:text="third item"
                android:textColor="#fff"
                android:textSize="20sp" />
        </RelativeLayout>

        <RelativeLayout
            android:layout_width="match_parent"
            android:layout_height="wrap_content" >

            <ImageView
                android:id="@+id/img04"
                android:layout_width="50dp"
                android:layout_height="50dp"
                android:layout_marginLeft="20dp"
                android:layout_marginTop="20dp"
                android:src="@drawable/img_4" />

            <TextView
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:layout_centerVertical="true"
                android:layout_marginLeft="20dp"
                android:layout_toRightOf="@id/img04"
                android:text="forth item"
                android:textColor="#fff"
                android:textSize="20sp" />
        </RelativeLayout>

        <RelativeLayout
            android:layout_width="match_parent"
            android:layout_height="wrap_content" >

            <ImageView
                android:id="@+id/img05"
                android:layout_width="50dp"
                android:layout_height="50dp"
                android:layout_marginLeft="20dp"
                android:layout_marginTop="20dp"
                android:src="@drawable/img_5" />

            <TextView
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:layout_centerVertical="true"
                android:layout_marginLeft="20dp"
                android:layout_toRightOf="@id/img05"
                android:text="fifth item"
                android:textColor="#fff"
                android:textSize="20sp" />
        </RelativeLayout>
    </LinearLayout>

</RelativeLayout>

  

界面效果如下所示:


1.    res/activity_main.xml

 <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="wrap_content"
    android:layout_height="match_parent" >

    <com.example.customview.CustomHorizontalScorllView这个是下面的自定义组件,为了显现滑动效果
        android:layout_width="match_parent"
        android:layout_height="match_parent" >

        <LinearLayout
            android:layout_width="wrap_content"
            android:layout_height="match_parent"
            android:orientation="horizontal" >

            <include layout="@layout/left_menuxml" />

            <LinearLayout
                android:layout_width="match_parent"
                android:layout_height="match_parent"
                android:background="@drawable/qq" >
            </LinearLayout>
        </LinearLayout>
    </ com.example.customview.CustomHorizontalScorllView >
</RelativeLayout>

1.    自定义的HorizontalScrollView:

 package com.example.customview;

import android.app.Notification.Action;
import android.content.Context;
import android.util.AttributeSet;
import android.util.DisplayMetrics;
import android.util.TypedValue;
import android.view.MotionEvent;
import android.view.ViewGroup;
import android.view.WindowManager;
import android.widget.HorizontalScrollView;
import android.widget.LinearLayout;

public class CustomHorizontalView extends HorizontalScrollView {

	private int mScreenWidth;
	private int mRightPadding = 50;// 单位为dp
	private int mMenuWidth;// 菜单的宽度

	private LinearLayout mWrapper = null;
	private ViewGroup mMenu = null;
	private ViewGroup mContent = null;

	private boolean once = false;// 指标,设置onMeasure方法中的设置宽高只设置一次

	public CustomHorizontalView(Context context, AttributeSet attrs) {
		super(context, attrs);
		// TODO Auto-generated constructor stub
		WindowManager wm = (WindowManager) context
				.getSystemService(Context.WINDOW_SERVICE);
		DisplayMetrics metrics = new DisplayMetrics();
		wm.getDefaultDisplay().getMetrics(metrics);
		this.mScreenWidth = metrics.widthPixels;

		// 将dp转化为px的单位
		this.mRightPadding = (int) TypedValue.applyDimension(
				TypedValue.COMPLEX_UNIT_DIP, 50, context.getResources()
						.getDisplayMetrics());
	}

	/**
	 * 设置子View的宽高 设置自己的宽高
	 */
	@Override
	protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
		// TODO Auto-generated method stub
		if (!once) {
			this.mWrapper = (LinearLayout) getChildAt(0);// 得到XML中HorizontalScrollView中的第一个LinearLayout
			this.mMenu = (ViewGroup) this.mWrapper.getChildAt(0);// 到到上面LinearLayout中的第一个布局即include的那个
			this.mContent = (ViewGroup) this.mWrapper.getChildAt(1);// 得到上面的第二个布局也就是XML中的LinearLayout那个
			this.mMenuWidth = this.mMenu.getLayoutParams().width = this.mScreenWidth
					- this.mRightPadding;
			this.mContent.getLayoutParams().width = this.mScreenWidth;
			once = true;
		}
		super.onMeasure(widthMeasureSpec, heightMeasureSpec);
	}
	
	/**
	 * 通过设置偏移量使Menu隐藏或出现
	 */
	@Override
	protected void onLayout(boolean changed, int l, int t, int r, int b) {
		// TODO Auto-generated method stub
		super.onLayout(changed, l, t, r, b);
		if(changed)
		{
			this.scrollTo(this.mMenuWidth, 0);//通过此方法(滚动到mMenuWidth,0 这个坐标)使Menu隐藏
		}
	}
	
	@Override
	public boolean onTouchEvent(MotionEvent ev) {
		// TODO Auto-generated method stub
		int action = ev.getAction();//通过这个得知用户的动作
		switch(action)
		{
		case MotionEvent.ACTION_UP:
			int scrollX = getScrollX();//此处为Menu隐藏的在手机屏幕可见区域外的区域,即隐藏在左边的区域
			if(scrollX>=this.mMenuWidth/2)
			{
				this.smoothScrollTo(this.mMenuWidth, 0);//继续隐藏左边的Menu区域
			}
			else
			{
				this.smoothScrollTo(0, 0);
			}
			return true;
		}
		return super.onTouchEvent(ev);
	}

}

1.    用户可调滑动边距的设置(ViewGroup支持自定义属性)

1.    编写values/attr.xml文件:

 <?xml version="1.0" encoding="utf-8"?>
<resources>

    <attr name="rightPadding" format="dimension"></attr>

    <declare-styleable name="CustomHorizontalScorllView"有文章说的name要和自定义组件的名称一样(但是我实验时即使name 和自定义组件不一样也可以)>
        <attr name="rightPadding"></attr>
    </declare-styleable>

</resources>

1.    在activity_main.xml文件中自定义命名空间:红色部分

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
   <span style="color:#ff0000;"> xmlns:padding=http://schemas.android.com/apk/res/com.example.qqslidmenu  </span>  其中padding是随便写的,而com.example.qqslidmenu是应用所在的包名即MainActivity的包名
<com.example.customview.CustomHorizontalView
        android:layout_width="match_parent"
        android:layout_height="match_parent" 
        padding:rightPadding="100">此处是使用上面自定义属性时的

1.    在CustomHorizontalView中:

我们前面编写的构造函数 : public CustomHorizontalView(Contextcontext, AttributeSetattrs)这是在未自定义属性时调用的,现在我们自定义了属性,所以就不会调用了

   

2.    CustomHorizontalScrollView类的改动如下:

/**
	 * 未自定义属性时调用
	 * 
	 * @param context
	 * @param attrs
	 */
	public CustomHorizontalView(Context context, AttributeSet attrs) {
		this(context, attrs, 0);
		// TODO Auto-generated constructor stub

	}

	/**
	 * 当使用自定义属性时会调用此构造方法
	 * 
	 * @param context
	 * @param attrs
	 * @param defStyle
	 */
	public CustomHorizontalView(Context context, AttributeSet attrs,
			int defStyle) {
		super(context, attrs, defStyle);
		// TODO Auto-generated constructor stub

		TypedArray a = context.getTheme().obtainStyledAttributes(attrs,
				R.styleable.CustomHorizontalView, defStyle, 0);此处是获取自定义XML文件的数组,得到用户自定义的所有属性
		int n = a.getIndexCount();//得到属性(数组)的数目 
		for(int i=0;i<n;i++)
		{
			int attr = a.getIndex(i);
			switch(attr)
			{
			case R.styleable.CustomHorizontalView_rightPadding:
				this.mRightPadding = a.getDimensionPixelSize(attr, (int) TypedValue.applyDimension(
				TypedValue.COMPLEX_UNIT_DIP, 50, context.getResources()
						.getDisplayMetrics()));//后面的参数是当用户没有自定义属性时的默认值
				break;
			}
		}
		a.recycle();//释放

		WindowManager wm = (WindowManager) context
				.getSystemService(Context.WINDOW_SERVICE);
		DisplayMetrics metrics = new DisplayMetrics();
		wm.getDefaultDisplay().getMetrics(metrics);
		this.mScreenWidth = metrics.widthPixels;

//		// 将dp转化为px的单位
//		this.mRightPadding = (int) TypedValue.applyDimension(
//				TypedValue.COMPLEX_UNIT_DIP, 50, context.getResources()
//						.getDisplayMetrics());由于在此之前已经设置了默认值所以此处不需要再设置一次了
	}

	/**
	 * 在代码中new一个对象时调用
	 * 
	 * @param context
	 */
	public CustomHorizontalView(Context context) {
		this(context, null, 0);
		// TODO Auto-generated constructor stub
	}

在滑动的Content中添加一个切换屏幕的点击按钮

   avtivity_main.xml中:

 <LinearLayout
                android:layout_width="match_parent"
                android:layout_height="match_parent"
                android:background="@drawable/qq" >
                <Button 
                    android:id="@+id/chang_menu_button"
                    android:onClick="toggleMenu"—>Button的按钮事件
                    android:layout_width="wrap_content"
                    android:layout_height="wrap_content"
                    android:text="change menu"/>
            </LinearLayout>
在MainActivity类中:
public class MainActivity extends ActionBarActivity {
	private CustomHorizontalView customview = null;
	

	@Override
	protected void onCreate(Bundle savedInstanceState) {
		super.onCreate(savedInstanceState);
		super.requestWindowFeature(Window.FEATURE_NO_TITLE);
		super.setContentView(R.layout.activity_main);
		this.customview = (CustomHorizontalView) super
				.findViewById(R.id.customHorizontalView);这个是自定义组件的在avtivity_main.xml中的id
	}

	/**
	 * 此处是Button的监听的事件
	 */
	public void toggleMenu(View view) {
		this.customview.toggle();
	}

}
1. 抽屉式侧滑:
原理:通过菜单的偏移量来改变菜单显示的多少
mMenuWidth-偏移量
需要用到属性动画TranslationX 
getScrollX中:一开始Menu隐藏中:所以它的值最大为mMenuWidth
当Menu滑出时它的值变小直到为0
2. 步骤:
在CustomHorizontalView类中加入如下代码用于监听滚动事件:同时为了设置属性动画需要引入nineoldandroids-2.4.0.jar这个包,这是为了兼容Android3.0一下的版本(如果是Android3.0以上的版本则不用这个包)
/**
	 * 当滚动发生时用于监听滚动的,在滚动的全过程中都监听
	 */
	@Override
	protected void onScrollChanged(int l, int t, int oldl, int oldt) {
		// TODO Auto-generated method stub
		super.onScrollChanged(l, t, oldl, oldt);

		// 这里l的初始值为mMenuWidth,越右拖l越小
		float scale = l * 1.0f / this.mMenuWidth;// 此scale的值从1~0变化
		// 调用属性动画,设置TranslationX
		ViewHelper.setTranslationX(mMenu, this.mMenuWidth * scale);
	}

个人理解如下:

   首先是初始化当scale的值为1的时候:


这个绿色的矩形就是:


这个界面。

蓝色是


这个界面,初始时,我们设定(ViewHelper.setTranslationX(mMenu,this.mMenuWidth *scale);

),此时设置的是偏移量,mMenuWidth是屏幕的宽度,则此时绿色的矩形就会由原本的在手机屏幕外面偏移到手机屏幕内,效果就是在蓝色界面的下方。但是随着滑动蓝色的界面会往屏幕右边滑去,绿色的界面本来也会随着蓝色界面往右边滑去,但是我们此时设定绿色界面的偏移量慢慢的变小,效果便是绿色的界面慢慢的在原地不动,而蓝色的界面慢慢的从屏幕中离开露出底部的绿色界面

下面黑色的代表手机屏幕,绿色的是那个界面)

1.  这种效果图的侧滑界面制作:


区别1;内容区域1.0~0.7缩放效果

       scale的值: 1.0~0.0

       所以运算得到内容区域值:0.7+0.3*scale

区别2 :菜单的偏移量需要修改

区别3:菜单的显示时有缩放以及透明度变化

       缩放:0.7~1.0

1.0-  scale*0.3

透明度: 0.6~1.0

         0.6+0.4*(1-scale)

2. 添加代码如下:

/**
	 * 当滚动发生时用于监听滚动的,在滚动的全过程中都监听
	 */
	@Override
	protected void onScrollChanged(int l, int t, int oldl, int oldt) {
		// TODO Auto-generated method stub
		super.onScrollChanged(l, t, oldl, oldt);

		// 这里l的初始值为mMenuWidth,越右拖l越小
		float scale = l * 1.0f / this.mMenuWidth;// 此scale的值从1~0变化

		/**
		 * 区别1;内容区域1.0~0.7缩放效果 scale的值: 1.0~0.0 所以运算得到内容区域值:0.7+0.3*scale 区别2
		 * :菜单的偏移量需要修改 区别3:菜单的显示时有缩放以及透明度变化 缩放:0.7~1.0 1.0- scale*0.3 透明度:
		 * 0.6~1.0 0.6+0.4*(1-scale)
		 */
		float rightscale = 0.7f + 0.3f * scale;
		float leftscale = 1.0f-scale*0.3f;
		float leftAlpha = 0.6f+0.4f*(1-scale);
		// 调用属性动画,设置TranslationX
		ViewHelper.setTranslationX(mMenu, this.mMenuWidth * scale);// mMenu是偏移的对象,后面的是偏移的值
		ViewHelper.setScaleX(this.mMenu, leftscale);
		ViewHelper.setScaleY(this.mMenu, leftscale);
		ViewHelper.setAlpha(this.mMenu, leftAlpha);
		// 设置右侧界面的缩放中心点,设置为在左边边框的中点
		ViewHelper.setPivotX(this.mContent, 0);
		ViewHelper.setPivotY(this.mContent, this.mContent.getHeight() / 2);
		ViewHelper.setScaleX(this.mContent, rightscale);
		ViewHelper.setScaleY(this.mContent, rightscale);
	}
注:以上笔记是通过观看慕课网视频所做



posted @ 2014-10-18 16:32  伟衙内  阅读(5)  评论(0编辑  收藏  举报