仿马蜂窝TabLayout 波纹滑动菜单导航栏

先看一下马蜂窝的样式
在这里插入图片描述
我刚看到觉得下面的样式好神奇,自己就像动手做一做。

由于数学方面我不是很精通,我的实现方式没有复杂的公式计算,而是通过绘制+覆盖的形式。

自己写一个TabLayout过于麻烦,这里就借用了MaterialDesign 的TabLayout改动了其中的draw的代码;

这里绘制路径主要用的是贝塞尔曲线不了解的可以先去网上搜下用法
1.首先设置Paint绘制线条

int lineWidth = 8;
paint.setColor(Color.RED);
paint.setStyle(Paint.Style.STROKE);
paint.setStrokeWidth(lineWidth);

2.TabLayout中又一个方法可以计算出每个tab的边界坐标

private void calculateTabViewContentBounds(
                @NonNull TabView tabView, @NonNull RectF contentBounds) {
            int tabViewContentWidth = tabView.getContentWidth();
            int minIndicatorWidth = (int) ViewUtils.dpToPx(getContext(), MIN_INDICATOR_WIDTH);

            if (tabViewContentWidth < minIndicatorWidth) {
                tabViewContentWidth = minIndicatorWidth;
            }

            int tabViewCenter = (tabView.getLeft() + tabView.getRight()) / 2;
            int contentLeftBounds = tabViewCenter - (tabViewContentWidth / 2);
            int contentRightBounds = tabViewCenter + (tabViewContentWidth / 2);

            contentBounds.set(contentLeftBounds, 0, contentRightBounds, 0);
        }

3.从起点开始绘制所有Tab的底部曲线

//设置本文的宽度默认为100,后期可以自己根据情况进行设置
int total = 100;
for (int i = 0; i < getTabCount(); i++) {
                    View tabView = getChildAt(i);
                    calculateTabViewContentBounds((TabView) tabView, tabViewContentBounds);
                    int left = (int) tabViewContentBounds.left;
                    int right = (int) tabViewContentBounds.right;
                    //底部曲线起点、终点距Tab边界的距离
                    smDim = (right - left - total) / 2;
                    //绘制第一个tab的曲线
                    if (i == 0) {
                    //移动到曲线绘制起点
                        mPath.moveTo(left + smDim, startY);
                        //开始规划路径
                        mPath.rQuadTo(total / 2, indicatorHeight / 2, total, 0);
                    } else {
                        int cx = smDim + (left - lastRight) / 2;
                        //绘制两个tab之间的曲线
                        mPath.rQuadTo(cx, -indicatorHeight + lineWidth, smDim + lastsmDim + (left - lastRight), 0);
                        //绘制tab下面的曲线
                        mPath.rQuadTo(total / 2, indicatorHeight / 2, total, 0);

                    }
                    //记录上次的边界坐标
                    lastRight = right;
                    lastsmDim = smDim;
                }
//开始绘制路径                                canvas.drawPath(mPath, paint);

这样整个tab下面的曲线就完成了,如下图所示:
在这里插入图片描述
4.开始绘制中遮罩层,覆盖曲线只留下当前位置tab的总宽度

                blockPaint.setStyle(Paint.Style.FILL);
                blockPaint.setColor(Color.parseColor("#f8f8f8"));
                int dimen = (indicatorRight - indicatorLeft - total) / 2;
                canvas.drawRect(0, indicatorTop, indicatorLeft + dimen, indicatorBottom, blockPaint);
                canvas.drawRect(indicatorRight - dimen, indicatorTop, getWidth(), indicatorBottom, blockPaint);

根据tab的移动 动态绘制遮罩层就实现了曲线移动的视觉
最终实现结果如下图所示:
在这里插入图片描述
这里有一个问题就是当两个相邻tab的宽度相差比较大的时候曲线在某一点拐点比较生硬。

当然如果数学功底比较好的,可以根据坐标计算出贝塞尔曲线在某一点的坐标值,这样可能会解决上面的问题。

如果有更好的方法还请多多指教;

Demo 地址

仿蚂蜂窝自由行和慕课网视频欢迎页一.资源准备三个比较短小的视频:视频下载二.开始编写代码1.在项目的res下新建一个raw文件夹,放入准备好的这三个视频2.自定义播放视频的CustomVideoView 在这个自定义View里面提供一个播放视频的方法。用户只需要传入播放路径就可以了,并且可一循环播放。package cn.bluemobi.dylan.welcomevideopager; import android.content.Context; import android.media.MediaPlayer; import android.net.Uri; import android.util.AttributeSet; import android.view.View; import android.widget.VideoView; /**  * 可以播放视频的View  * Created by yuandl on 2016-11-10.  */ public class CustomVideoView extends VideoView {     public CustomVideoView(Context context) {         super(context);     }     public CustomVideoView(Context context, AttributeSet attrs, int defStyleAttr) {         super(context, attrs, defStyleAttr);     }     public CustomVideoView(Context context, AttributeSet attrs) {         super(context, attrs);     }     @Override     protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {         super.onMeasure(widthMeasureSpec, heightMeasureSpec);         setMeasuredDimension(View.MeasureSpec.getSize(widthMeasureSpec), View.MeasureSpec.getSize(heightMeasureSpec));     }     /**      * 播放视频      *      * @param uri 播放地址      */     public void playVideo(Uri uri) {         if (uri == null) {             throw new IllegalArgumentException("Uri can not be null");         }         /**设置播放路径**/         setVideoURI(uri);         /**开始播放**/         start();         setOnPreparedListener(new MediaPlayer.OnPreparedListener() {             @Override             public void onPrepared(MediaPlayer mp) {                 /**设置循环播放**/                 mp.setLooping(true);             }         });         setOnErrorListener(new MediaPlayer.OnErrorListener() {             @Override             public boolean onError(MediaPlayer mp, int what, int extra) {                 return true;             }         });     } }3.建立没个欢迎页面的Fragment去加载自定义视频View的视图package cn.bluemobi.dylan.welcomevideopager; import android.net.Uri; import android.os.Bundle; import android.support.annotation.Nullable; import android.support.v4.app.Fragment; import android.view.LayoutInflater; import android.view.View; import android.view.ViewGroup; /**  * Created by yuandl on 2016-11-10.  */ public class GuildFragment extends Fragment {     private CustomVideoView customVideoView;     @Nullable     @Override     public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {         customVideoView = new CustomVideoView(getContext());         /**获取参数,根据不同的参数播放不同的视频**/         int index = getArguments().getInt("index");         Uri uri;         if (index == 1) {             uri = Uri.parse("android.resource://"   getActivity().getPackageName()   "/"   R.raw.guide_1);         } else if (index == 2) {             uri = Uri.parse("android.resource://"   getActivity().getPackageName()   "/"   R.raw.guide_2);         } else {             uri = Uri.parse("android.resource://"   getActivity().getPackageName()   "/"   R.raw.guide_3);         }         /**播放视频**/         customVideoView.playVideo(uri);         return customVideoView;     }     /**      * 记得在销毁的时候让播放的视频终止      */     @Override     public void onDestroy() {         super.onDestroy();         if (customVideoView != null) {             customVideoView.stopPlayback();         }     } }4.界面布局<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"     android:layout_width="match_parent"     android:layout_height="match_parent">     <android.support.v4.view.ViewPager         android:id="@ id/vp"         android:layout_width="match_parent"         android:layout_height="match_parent"></android.support.v4.view.ViewPager>     <LinearLayout         android:layout_width="wrap_content"         android:layout_height="wrap_content"         android:layout_alignParentBottom="true"         android:layout_centerHorizontal="true"         android:layout_marginBottom="130dp"         android:orientation="horizontal">         <ImageView             android:id="@ id/iv1"             android:layout_width="10dp"             android:layout_height="10dp"             android:src="@mipmap/dot_focus" />         <ImageView             android:id="@ id/iv2"             android:layout_width="10dp"             android:layout_height="10dp"             android:layout_marginLeft="10dp"             android:src="@mipmap/dot_normal" />         <ImageView             android:id="@ id/iv3"             android:layout_width="10dp"             android:layout_height="10dp"             android:layout_marginLeft="10dp"             android:src="@mipmap/dot_normal" />     </LinearLayout>     <Button         android:id="@ id/bt_start"         android:layout_width="wrap_content"         android:layout_height="wrap_content"         android:layout_alignParentBottom="true"         android:layout_centerHorizontal="true"         android:layout_marginBottom="50dp"         android:background="@mipmap/bt_start"         android:textColor="@android:color/white"         android:visibility="gone" /> </RelativeLayout>5.给界面添加Fragmentpackage cn.bluemobi.dylan.welcomevideopager; import android.os.Bundle; 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.v7.app.AppCompatActivity; import android.view.View; import android.widget.Button; import android.widget.ImageView; import java.util.ArrayList; import java.util.List; public class MainActivity extends AppCompatActivity {     private ViewPager vp;     private ImageView iv1;     private ImageView iv2;     private ImageView iv3;     private Button bt_start;     private List<Fragment> fragments;     private void assignViews() {         vp = (ViewPager) findViewById(R.id.vp);         iv1 = (ImageView) findViewById(R.id.iv1);         iv2 = (ImageView) findViewById(R.id.iv2);         iv3 = (ImageView) findViewById(R.id.iv3);         bt_start = (Button) findViewById(R.id.bt_start);     }     @Override     protected void onCreate(Bundle savedInstanceState) {         super.onCreate(savedInstanceState);         setContentView(R.layout.activity_main);         assignViews();         initData();         initView();     }     /**      * 初始化数据,添加三个Fragment      */     private void initData() {         fragments = new ArrayList<>();         Fragment fragment1 = new GuildFragment();         Bundle bundle1 = new Bundle();         bundle1.putInt("index", 1);         fragment1.setArguments(bundle1);         fragments.add(fragment1);         Fragment fragment2 = new GuildFragment();         Bundle bundle2 = new Bundle();         bundle2.putInt("index", 2);         fragment2.setArguments(bundle2);         fragments.add(fragment2);         Fragment fragment3 = new GuildFragment();         Bundle bundle3 = new Bundle();         bundle3.putInt("index", 3);         fragment3.setArguments(bundle3);         fragments.add(fragment3);     }     /**      * 设置ViewPager的适配器和滑动监听      */     private void initView() {         vp.setOffscreenPageLimit(3);         vp.setAdapter(new MyPageAdapter(getSupportFragmentManager()));         vp.addOnPageChangeListener(new MyPageChangeListener());     }     /**      * ViewPager适配器      */     private class MyPageAdapter extends FragmentPagerAdapter {         public MyPageAdapter(FragmentManager fm) {             super(fm);         }         @Override         public Fragment getItem(int position) {             return fragments.get(position);         }         @Override         public int getCount() {             return fragments.size();         }     }     /**      * ViewPager滑动页面监听器      */     private class MyPageChangeListener implements ViewPager.OnPageChangeListener {         @Override         public void onPageScrolled(int position, float positionOffset, int positionOffsetPixels) {         }         /**          * 根据页面不同动态改变红点和在最后一页显示立即体验按钮          *          * @param position          */         @Override         public void onPageSelected(int position) {             bt_start.setVisibility(View.GONE);             iv1.setImageResource(R.mipmap.dot_normal);             iv2.setImageResource(R.mipmap.dot_normal);             iv3.setImageResource(R.mipmap.dot_normal);             if (position == 0) {                 iv1.setImageResource(R.mipmap.dot_focus);             } else if (position == 1) {                 iv2.setImageResource(R.mipmap.dot_focus);             } else {                 iv3.setImageResource(R.mipmap.dot_focus);                 bt_start.setVisibility(View.VISIBLE);             }         }         @Override         public void onPageScrollStateChanged(int state) {         }     } }
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值