Android捕获View焦点事件,LinearLayout结合HorizontalScrollView实现ViewPgaer和选项卡Tabs联动

简介: 《Android捕获View焦点事件,LinearLayout结合HorizontalScrollView实现ViewPgaer和选项卡Tabs联动。


《Android捕获View焦点事件,LinearLayout结合HorizontalScrollView实现ViewPgaer和选项卡Tabs联动。》

如图:


package zhangphil.tabs;

import java.util.ArrayList;

import android.support.v4.app.Fragment;
import android.support.v4.app.FragmentManager;
import android.support.v4.app.FragmentPagerAdapter;
import android.support.v4.view.ViewPager;
import android.support.v4.view.ViewPager.OnPageChangeListener;
import android.support.v7.app.ActionBarActivity;
import android.view.Gravity;
import android.view.LayoutInflater;
import android.view.View;
import android.view.View.OnFocusChangeListener;
import android.view.ViewGroup;
import android.widget.LinearLayout;
import android.widget.TextView;
import android.graphics.Color;
import android.os.Bundle;

/**
 * 
 * Android捕获View焦点事件,LinearLayout结合HorizontalScrollView实现ViewPgaer和选项卡Tabs联动。
 * 
 * 思路:实现方案很多。本例是把一个水平的LinearLayout在外层用HorizontalScrollView包裹起来(之所以使用HorizontalScrollView再包裹一层,是考虑到选项卡条目可能很多的情况下,让用户可以像ListView一样滑动选择。否则,单纯的使用LinearLayout添加有限条目超过边界后多的元素不可见),然后在LinearLayout里面添加若干子元素(简单期间,就比如TextView),做成一个横向可滑动且响应触发点击事件的"ListView"。该“ListView”响应点击事件,触发ViewPager页面的切换。
 * 实现目的:tab栏中的选项和ViewPager中的Fragment实时联动。比如,当ViewPager滑动到第10的位置时候,相应的位于顶部(或者底部)的tab也要切换到第十的位置。反之亦然。
 * 这种控件效果使用极为广泛,尤其是在一些新闻客户端中常见,比如新浪新闻、腾讯新闻新闻客户端的头部标签导航选项卡。
 * 
 * 
 * 附参考文章: 如何让一个View的焦点改变时同时改变其附属元素,可参考我的另一篇文章:
 * 《Android实现连续并排的若干个TextView单击改变背景颜色达到选项卡Tab栏切换效果 》
 * 链接地址:http://blog.csdn.net/zhangphil/article/details/46547561
 * 
 * 
 * */

public class MainActivity extends ActionBarActivity {

	private ArrayList<Fragment> fragments;

	// 测试的Fragment数量,也即是选项卡片的数量。
	private final int SIZE = 15;

	@Override
	protected void onCreate(Bundle savedInstanceState) {
		super.onCreate(savedInstanceState);
		setContentView(R.layout.activity_main);

		fragments = new ArrayList<Fragment>();
		for (int i = 0; i < SIZE; i++) {
			Fragment f = TestFragment.newInstance(i);
			fragments.add(f);
		}

		// 将在tabs_LinearLayout里面添加需要的若干选项卡片。
		final LinearLayout tabs_LinearLayout = (LinearLayout) findViewById(R.id.tabs_LinearLayout);

		final ViewPager mViewPager = (ViewPager) findViewById(R.id.viewpager);

		for (int i = 0; i < SIZE; i++) {
			View v = LayoutInflater.from(this).inflate(R.layout.view, null);
			TextView tv = (TextView) v;
			tv.setText("tab" + i);

			v.setOnFocusChangeListener(new OnFocusChangeListener() {

				@Override
				public void onFocusChange(View v, boolean hasFocus) {

					// 当用户选择了tab选项卡上面的子元素时候,相应的把ViewPager显示的页面调整到相应位置。

					int count = tabs_LinearLayout.getChildCount();
					for (int i = 0; i < count; i++) {
						View cv = tabs_LinearLayout.getChildAt(i);
						if (v == cv) {
							if (hasFocus) {
								mViewPager.setCurrentItem(i);
								break;
							}
						}
					}
				}
			});

			tabs_LinearLayout.addView(v, i);
		}

		mViewPager.setAdapter(new MyFragmentPagerAdapter(this
				.getSupportFragmentManager()));
		mViewPager.setOnPageChangeListener(new OnPageChangeListener() {

			@Override
			public void onPageScrollStateChanged(int arg0) {

			}

			@Override
			public void onPageScrolled(int arg0, float arg1, int arg2) {

			}

			@Override
			public void onPageSelected(int pos) {
				// 在这里,当用户翻动ViewPager页面时候,相应的把选项卡显示对应的位置。
				// 最轻巧的实现就是让tab选项卡栏中的子元素获得焦点即可。
				View v = tabs_LinearLayout.getChildAt(pos);
				v.requestFocus();
			}
		});
	}

	// ViewPager的适配器。
	private class MyFragmentPagerAdapter extends FragmentPagerAdapter {

		public MyFragmentPagerAdapter(FragmentManager fm) {
			super(fm);
		}

		@Override
		public Fragment getItem(int pos) {
			return fragments.get(pos);
		}

		@Override
		public int getCount() {
			return fragments.size();
		}
	}

	// 仅仅用于测试的Fragment。
	public static class TestFragment extends Fragment {

		// 用一个id标明,否则难以识别效果。
		private static final String ID = "id";

		public static Fragment newInstance(int id) {
			Fragment f = new TestFragment();
			Bundle b = new Bundle();
			b.putInt(ID, id);
			f.setArguments(b);
			return f;
		}

		@Override
		public View onCreateView(LayoutInflater inflater, ViewGroup container,
				Bundle savedInstanceState) {

			Bundle bundle = this.getArguments();
			int id = (Integer) bundle.get(ID);

			TextView tv = new TextView(this.getActivity());
			tv.setGravity(Gravity.CENTER);
			tv.setText("Fragment:" + id);
			tv.setTextSize(50.0f);
			tv.setTextColor(Color.LTGRAY);

			return tv;
		}
	}
}

activity_main.xml:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical" >

    <HorizontalScrollView
        android:id="@+id/tabs_HorizontalScrollView"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:scrollbars="none" >

        <LinearLayout
            android:id="@+id/tabs_LinearLayout"
            android:layout_width="match_parent"
            android:layout_height="match_parent"
            android:orientation="horizontal"
            android:layout_weight="1" >
            
        </LinearLayout>
        
    </HorizontalScrollView>

    <View
        android:layout_width="match_parent"
        android:layout_height="5dip"
        android:background="@android:color/black" />

    <android.support.v4.view.ViewPager
        android:id="@+id/viewpager"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:layout_weight="9" />

</LinearLayout>




view.xml:

<?xml version="1.0" encoding="utf-8"?>
<TextView xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:background="@drawable/selector"
    android:clickable="true"
    android:focusable="true"
    android:focusableInTouchMode="true"
    android:gravity="center"
    android:padding="5dip" >

</TextView>


selector.xml:

<?xml version="1.0" encoding="utf-8"?>
<selector xmlns:android="http://schemas.android.com/apk/res/android">

    <item android:drawable="@drawable/red" android:state_focused="true"/>
    <item android:drawable="@drawable/orange" android:state_pressed="true"/>
    <item android:drawable="@drawable/green"/>

</selector>

剩余的是在drawable目录下的一些基础的图片资源,可以根据自己需要添加。
相关文章
|
2月前
|
XML 前端开发 Android开发
Android:UI:Drawable:View/ImageView与Drawable
通过本文的介绍,我们详细探讨了Android中Drawable、View和ImageView的使用方法及其相互关系。Drawable作为图像和图形的抽象表示,提供了丰富的子类和自定义能力,使得开发者能够灵活地实现各种UI效果。View和ImageView则通过使用Drawable实现了各种图像和图形的显示需求。希望本文能为您在Android开发中使用Drawable提供有价值的参考和指导。
52 2
|
2月前
|
Android开发 开发者 UED
安卓开发中自定义View的实现与性能优化
【10月更文挑战第28天】在安卓开发领域,自定义View是提升应用界面独特性和用户体验的重要手段。本文将深入探讨如何高效地创建和管理自定义View,以及如何通过代码和性能调优来确保流畅的交互体验。我们将一起学习自定义View的生命周期、绘图基础和事件处理,进而探索内存和布局优化技巧,最终实现既美观又高效的安卓界面。
51 5
|
3月前
|
缓存 数据处理 Android开发
在 Android 中使用 RxJava 更新 View
【10月更文挑战第20天】使用 RxJava 来更新 View 可以提供更优雅、更高效的解决方案。通过合理地运用操作符和订阅机制,我们能够轻松地处理异步数据并在主线程中进行 View 的更新。在实际应用中,需要根据具体情况进行灵活运用,并注意相关的注意事项和性能优化,以确保应用的稳定性和流畅性。可以通过不断的实践和探索,进一步掌握在 Android 中使用 RxJava 更新 View 的技巧和方法,为开发高质量的 Android 应用提供有力支持。
|
3月前
|
缓存 调度 Android开发
Android 在子线程更新 View
【10月更文挑战第21天】在 Android 开发中,虽然不能直接在子线程更新 View,但通过使用 Handler、AsyncTask 或 RxJava 等方法,可以实现子线程操作并在主线程更新 View 的目的。在实际应用中,需要根据具体情况选择合适的方法,并注意相关的注意事项和性能优化,以确保应用的稳定性和流畅性。可以通过不断的实践和探索,进一步掌握在子线程更新 View 的技巧和方法,为开发高质量的 Android 应用提供支持。
58 2
|
3月前
|
Android开发
Android面试高频知识点(1) 图解Android事件分发机制
Android面试高频知识点(1) 图解Android事件分发机制
|
3月前
|
XML 前端开发 Android开发
Android面试高频知识点(3) 详解Android View的绘制流程
Android面试高频知识点(3) 详解Android View的绘制流程
Android面试高频知识点(3) 详解Android View的绘制流程
|
3月前
|
XML 前端开发 Android开发
Android面试高频知识点(3) 详解Android View的绘制流程
Android面试高频知识点(3) 详解Android View的绘制流程
35 2
|
3月前
|
Android开发
Android面试高频知识点(1) 图解 Android 事件分发机制
Android面试高频知识点(1) 图解 Android 事件分发机制
56 1
|
3月前
|
XML 前端开发 Android开发
Android面试高频知识点(1) 图解Android事件分发机制
Android面试高频知识点(1) 图解Android事件分发机制
Android面试高频知识点(1) 图解Android事件分发机制

热门文章

最新文章