Android实现网易新闻客户端侧滑菜单(二)

前面已经讲过通过三方开源库SlideMenu来实现这种效果,请参考Android实现网易新闻客户端侧滑菜单(一)

今天通过自定义View来实现这种功能

代码如下:

SlideMenu.java

<span style="font-family:SimSun;font-size:14px;">package com.jackie.slidemenu.view;

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

public class SlideMenu extends ViewGroup {

	private int mMostRecentX;		// 最后一次x轴的偏移量

	private final int MENU_SCREEN = 0;		// 菜单界面
	private final int MAIN_SCREEN = 1;		// 主界面
	private int mCurrentScreen = MAIN_SCREEN;		// 当前屏幕显示的是主界面
	private Scroller mScroller;

	private int touchSlop;

	public SlideMenu(Context context, AttributeSet attrs) {
		super(context, attrs);
		mScroller = new Scroller(context);

		touchSlop = ViewConfiguration.get(context).getScaledTouchSlop();
	}

	/**
	 * 测量出所有子布局的宽和高
	 */
	@Override
	protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
		super.onMeasure(widthMeasureSpec, heightMeasureSpec);

		measureView(widthMeasureSpec, heightMeasureSpec);
	}

	/**
	 * 测量所有子布局的宽和高
	 * @param widthMeasureSpec 父布局也就是ViewGroup的宽度测量规格
	 * @param heightMeasureSpec 父布局也就是ViewGroup的高度测量规格
	 */
	private void measureView(int widthMeasureSpec, int heightMeasureSpec) {
		// 测量菜单的宽和高
		View menuView = getChildAt(0);
		menuView.measure(menuView.getLayoutParams().width, heightMeasureSpec);

		// 测量主界面的宽和高
		View mainView = getChildAt(1);
		mainView.measure(widthMeasureSpec, heightMeasureSpec);		// 主界面的宽和高和父控件viewgroup的宽高一样
	}

	@Override
	protected void onLayout(boolean changed, int l, int t, int r, int b) {
		// 布置菜单的位置
		View menuView = getChildAt(0);
		menuView.layout(-menuView.getMeasuredWidth(), 0, 0, b);

		// 布置主界面的位置
		View mainView = getChildAt(1);
		mainView.layout(0, 0, r, b);
	}

	@Override
	public boolean onTouchEvent(MotionEvent event) {
		switch (event.getAction()) {
		case MotionEvent.ACTION_DOWN:
			mMostRecentX = (int) event.getX();
			break;
		case MotionEvent.ACTION_MOVE:
			// 最新的x轴偏移量
			int moveX = (int) event.getX();

			// 增量值
			int deltaX = mMostRecentX - moveX;

			// 把最新的x轴偏移量赋值给成员变量
			mMostRecentX = moveX;

			// 得到x轴移动后的偏移量
			int newScrollX = getScrollX() + deltaX;

			if(newScrollX < -getChildAt(0).getWidth()) {		// 当前屏幕x轴的偏移量超过了菜单的左边界
				// 回到菜单的左边界位置
				scrollTo(-getChildAt(0).getWidth(), 0);
			} else if(newScrollX > 0) {		// 超过了主界面的右边界
				// 回到主界面的右边界
				scrollTo(0, 0);
			} else {
				scrollBy(deltaX, 0);
			}
			break;
		case MotionEvent.ACTION_UP:
			int scrollX = getScrollX(); 	// x轴最新的偏移量

			int menuXCenter = -getChildAt(0).getWidth() / 2;		// 菜单x轴的中心点

			if(scrollX > menuXCenter) {	// 切换到主界面
				mCurrentScreen = MAIN_SCREEN;
			} else {	// 切换到菜单界面
				mCurrentScreen = MENU_SCREEN;
			}
			switchScreen();
			break;
		default:
			break;
		}
		return true;
	}

	/**
	 * 根据mCurrentScreen切换屏幕
	 */
	private void switchScreen() {
		int scrollX = getScrollX(); // 当前x轴的偏移量
		int dx = 0;

		if(mCurrentScreen == MAIN_SCREEN) {	// 切换到主界面
//			scrollTo(0, 0);
			dx = 0 - scrollX;
		} else if(mCurrentScreen == MENU_SCREEN) {	// 切换到菜单界面
//			scrollTo(-getChildAt(0).getWidth(), 0);
			dx = -getChildAt(0).getWidth() - scrollX;
		}

		mScroller.startScroll(scrollX, 0, dx, 0, Math.abs(dx) * 5);

		invalidate();		// invalidate -> drawChild -> child.draw -> computeScroll
 	}

	/**
	 * invalidate出发此方法, 更新屏幕的x轴的偏移量
	 */
	@Override
	public void computeScroll() {
		if(mScroller.computeScrollOffset()) {		// 判断是否正在模拟数据中, true 正在进行 false 数据模拟完毕
			scrollTo(mScroller.getCurrX(), 0);

			invalidate();		// 引起computeScroll的调用
		}
	}

	/**
	 * 是否显示菜单
	 * @return
	 */
	public boolean isShowMenu() {
		return mCurrentScreen == MENU_SCREEN;
	}

	/**
	 * 隐藏菜单
	 */
	public void hideMenu() {
		mCurrentScreen = MAIN_SCREEN;
		switchScreen();
	}

	/**
	 * 显示菜单
	 */
	public void showMenu() {
		mCurrentScreen = MENU_SCREEN;
		switchScreen();
	}

	/**
	 * 拦截事件的方法
	 */
	@Override
	public boolean onInterceptTouchEvent(MotionEvent ev) {
		switch (ev.getAction()) {
		case MotionEvent.ACTION_DOWN:
			mMostRecentX = (int) ev.getX();
			break;
		case MotionEvent.ACTION_MOVE:

			int diffX = (int) (ev.getX() - mMostRecentX);
			if(Math.abs(diffX) > touchSlop) {
				return true;
			}
			break;
		default:
			break;
		}
		return super.onInterceptTouchEvent(ev);
	}

}</span>

MainActivity.java

<span style="font-family:SimSun;font-size:14px;">package com.jackie.slidemenu;

import com.jackie.slidemenu.view.SlideMenu;

import android.os.Bundle;
import android.app.Activity;
import android.view.Menu;
import android.view.View;
import android.view.View.OnClickListener;
import android.view.Window;
import android.widget.TextView;
import android.widget.Toast;

public class MainActivity extends Activity implements OnClickListener {

	private SlideMenu mSlideMenu;

	@Override
	protected void onCreate(Bundle savedInstanceState) {
		super.onCreate(savedInstanceState);
		// 去除标题, 需要在setContentView之前调用
		requestWindowFeature(Window.FEATURE_NO_TITLE);
		setContentView(R.layout.activity_main);

		mSlideMenu = (SlideMenu) findViewById(R.id.slidemenu);
		findViewById(R.id.iv_slidemenu_main_back).setOnClickListener(this);

	}

	@Override
	public boolean onCreateOptionsMenu(Menu menu) {
		// Inflate the menu; this adds items to the action bar if it is present.
		getMenuInflater().inflate(R.menu.main, menu);
		return true;
	}

	@Override
	public void onClick(View v) {
		if(mSlideMenu.isShowMenu()) {
			mSlideMenu.hideMenu();
		} else {
			mSlideMenu.showMenu();
		}
	}

	public void click(View v) {
		TextView tv = (TextView) v;
		Toast.makeText(this, tv.getText(), 0).show();
	}
}
</span>
时间: 2024-11-06 11:03:26

Android实现网易新闻客户端侧滑菜单(二)的相关文章

android 仿网易新闻客户端源码都有

原文:android 仿网易新闻客户端源码都有 android 仿网易新闻服务端源码 源代码下载地址: http://www.zuidaima.com/share/1550463560944640.htm http://www.zuidaima.com/share/1550463561206784.htm android 仿网易新闻 客户端和服务端 源码都有 ,有些功能还未实现,因为文件有点大,所以分为2次上传  java源代码截图:

Android实现网易新闻客户端效果

tabhost_tabwidget.xml [html] view plaincopyprint? <?xml version="1.0" encoding="utf-8"?> <TabHost xmlns:android="http://schemas.android.com/apk/res/android" android:id="@android:id/tabhost" android:layout_w

Android应用经典主界面框架之二:仿网易新闻客户端、CSDN 客户端 (Fragment ViewPager)

第二种主界面风格则是以网易新闻.凤凰新闻以及新推出的新浪博客(阅读版)为代表,使用ViewPager+Fragment,即ViewPager里适配器里放的不是一般的View,而是Fragment.所以适配器不能继承PagerAdapter,而要继承FragmentPagerAdapter,这是在android.support.v4.app.FragmentPagerAdapter包里的.有点奇葩的是,FragmentPagerAdapter只在这个包里有,在android.app.*这个包下面么

Android 开源框架ActionBarSherlock 和 ViewPager 仿网易新闻客户端

转载请注明出处:http://blog.csdn.net/xiaanming/article/details/9971721 大家都知道Android的ActionBar是在3.0以上才有的,那么在3.0以下呢,google并没有给我提供在3.0以下支持ActionBar的包,但是外国的大牛JakeWharton实现了在3.0以下使用ActionBar, JakeWharton这位大牛是ActionBarSherlock,Android-ViewPagerIndicator ,NineOldAn

Android(java)学习笔记205:网易新闻客户端应用编写逻辑过程

1.我们的项目需求是编写一个新闻RSS浏览器,RSS(Really Simple Syndication)是一种描述和同步网站内容的格式,是使用最广泛的XML应用.RSS目前广泛用于网上新闻频道,blog和wiki,主要的版本有0.91, 1.0, 2.0.使用RSS订阅能更快地获取信息,网站提供RSS输出,有利于让用户获取网站内容的最新更新.网络用户可以在客户端借助于支持RSS的聚合工具软件,在不打开网站内容页面的情况下阅读支持RSS输出的网站内容. 例如如下的网易RSS订阅: 2.由于我们这

Android 开源框架ViewPageIndicator 和 ViewPager 仿网易新闻客户端Tab标签

之前用JakeWharton的开源框架ActionBarSherlock和ViewPager实现了对网易新闻客户端Tab标签的功能,ActionBarSherlock是在3.0以下的机器支持ActionBar的功能,有兴趣的可以看看开源框架ActionBarSherlock 和 ViewPager 仿网易新闻客户端,今天用到的是JakeWharton的另一开源控件ViewPageIndicator,ViewPager想必大家都知道,Indicator指示器的意思,所以ViewPageIndica

ActionBar+DrawerLayout实现网易新闻客户端首页

一.概述 随着android版本的不断的更新,google推出了越来越多的高级组件,采用这些官方组件我们可以方便的实现一些以前需要通过复杂编码或者使用第三方组件才能实现的效果,比如slidingmenu.sherlockactionbar等.在这里,我们通过使用android的官方组件ActionBar和DrawerLayout来实现网易新闻客户端首页的效果. 由于ActionBar和DrawerLayout都是后来推出的,如果需要兼容低版本必须在项目中添加v7支持库.具体如何添加支持库在此不做

Toolbar+DrawerLayout高仿网易新闻客户端

首先看效果图,网易新闻客户端的特点是双向侧滑,并且左上角的图标会随着菜单的侧滑会有动画效果. 我们采用Toolbar和DrawerLayout实现双向侧滑以及actionbar 在菜单文件里先定义菜单 <menu xmlns:android="http://schemas.android.com/apk/res/android" xmlns:app="http://schemas.android.com/apk/res-auto" xmlns:tools=&q

一个基于Android系统的新闻客户端(一)

一.整体概述 在服务器端,通过对凤凰网的抓取存入数据库,客户端通过向服务器发送请求得到新闻. 服务端用WCF,宿主为window服务,客户端为Java写的安卓程序. 二.客户端 我在eclipse里新建了个Android项目,命名为MyNewClient,eclipse自动生成二个xml布局文件,如图: 其中,fragment_main.xml是新版的布局文件,暂时不会用,把它删掉. 新建xml文件,命名为activity_foot.xml,在这里我们要做app的底部,先上代码: <?xml v