欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  移动技术

Android高仿微信5.2.1主界面及消息提醒

程序员文章站 2024-03-06 14:55:50
好久没更新博客了,最近在做公司的项目,这也算是我接触的第一个正式项目。通过项目的检验,发现自己积累了一年的知识还是远远不够,想要提高,好的方法是 :项目+书+视频+博客。最...

好久没更新博客了,最近在做公司的项目,这也算是我接触的第一个正式项目。通过项目的检验,发现自己积累了一年的知识还是远远不够,想要提高,好的方法是 :项目+书+视频+博客。最重要一点:勤动手。最近发现了慕课网的视频,居然都是高清无码免费的!而且满满的干货!我用业余时间跟着视频中大神的讲解学习了不少知识,下面就将这些小demo与大家分享,当然,我做了一些优化,代码与视频中有些出入,但功能可以完全实现。

这是一个模仿5.2.1版本的显示界面,如下图所示:

Android高仿微信5.2.1主界面及消息提醒Android高仿微信5.2.1主界面及消息提醒

功能及实现思路简介

主要功能很简单:
1、上面有一个自定义的标题栏;
2、往下是聊天、发现、通讯录选项卡;
3、手指滑动时,文字下方蓝色的indicator可以跟随滑动;
4、在聊天的右侧,有一个未读消息的红色提醒圆点。

自定义的标题栏就是一个linearlayout,同时将系统自带的titlebar(或是actionbar)隐藏;

由于是选项卡,自然想到了fragment;

手指可以滑动,显然,黑色的区域是一个viewpager,数据源就是fragment组成的集合,并通过fragmentpageradapter进行管理;

要实现蓝色的indicator随选项卡的滑动而滑动,可以为viewpager设置监听,并根据回调方法的回传值控制该indicator的marginleft属性值可以实现该效果。

最后消息提醒的小圆点是一个badgeview ,它是一个第三方开源控件。

主布局

mainactivity布局如下,首先是自定义的titlebar:

<!-- top1.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="50dp"
 android:background="@drawable/topone_bg"
 android:paddingleft="12dp"
 android:paddingright="12dp">

 <linearlayout
  android:layout_width="wrap_content"
  android:layout_height="wrap_content"
  android:layout_centervertical="true"
  android:gravity="center"
  android:orientation="horizontal">

  <imageview
   android:layout_width="30dp"
   android:layout_height="30dp"
   android:background="@drawable/actionbar_icon" />

  <textview
   android:layout_width="wrap_content"
   android:layout_height="wrap_content"
   android:layout_marginleft="12dp"
   android:text="微信"
   android:textcolor="#d3d3d3"
   android:textsize="18sp" />
 </linearlayout>

 <linearlayout
  android:layout_width="wrap_content"
  android:layout_height="wrap_content"
  android:layout_alignparentright="true"
  android:layout_centervertical="true"
  android:orientation="horizontal">


  <imageview
   android:layout_width="30dp"
   android:layout_height="30dp"
   android:background="@drawable/actionbar_search_icon" />

  <imageview
   android:layout_width="30dp"
   android:layout_height="30dp"
   android:background="@drawable/actionbar_add_icon" />

  <imageview
   android:layout_width="30dp"
   android:layout_height="30dp"
   android:background="@drawable/actionbar_more_icon" />


 </linearlayout>


</relativelayout>

效果如下所示:

Android高仿微信5.2.1主界面及消息提醒

接着是三个选项卡的布局:

<!-- top2.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="40dp"
 android:background="#eeeeee"
 android:orientation="vertical">

 <linearlayout
  android:layout_width="match_parent"
  android:layout_height="37dp"
  android:orientation="horizontal">

  <linearlayout
   android:id="@+id/ll_chat"
   android:layout_width="0dp"
   android:layout_height="match_parent"
   android:layout_weight="1"
   android:gravity="center"
   android:orientation="horizontal">

   <textview
    android:id="@+id/tv_tab_chat"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"

    android:layout_gravity="center"
    android:text="聊天"
    android:textcolor="#008000"
    android:textsize="16sp" />


  </linearlayout>

  <linearlayout
   android:layout_width="0dp"
   android:layout_height="match_parent"
   android:layout_weight="1"
   android:gravity="center">

   <textview
    android:id="@+id/tv_tab_discover"
    android:layout_width="wrap_content"

    android:layout_height="wrap_content"
    android:text="发现"
    android:textcolor="@android:color/black"
    android:textsize="16sp" />


  </linearlayout>

  <linearlayout
   android:layout_width="0dp"
   android:layout_height="match_parent"
   android:layout_weight="1"
   android:gravity="center">

   <textview
    android:id="@+id/tv_tab_contacts"

    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:text="通讯录"
    android:textcolor="@android:color/black"
    android:textsize="16sp" />


  </linearlayout>

 </linearlayout>

 <imageview
  android:id="@+id/iv_tab_line"
  android:layout_width="100dp"
  android:layout_height="3dp"
  android:background="@drawable/tabline" />

</linearlayout>

效果如下:

Android高仿微信5.2.1主界面及消息提醒

由于indicator还需要在代码中动态设置其长度,故在xml中可以附一个任意值。

最后将top1.xml、top2.xml加入至主布局中,并在主布局中引入viewpager:

<!-- activity_main.xml -->
<?xml version="1.0" encoding="utf-8"?>
<linearlayout xmlns:android="http://schemas.android.com/apk/res/android"
 xmlns:tools="http://schemas.android.com/tools"
 android:layout_width="match_parent"
 android:layout_height="match_parent"
 android:orientation="vertical"
 tools:context="com.demo.lenovo.myapplication.mainactivity">

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

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

 <android.support.v4.view.viewpager
  android:id="@+id/vp_content"
  android:layout_width="match_parent"
  android:layout_height="0dp"
  android:layout_weight="1" />
</linearlayout>


效果如下:

Android高仿微信5.2.1主界面及消息提醒

注:如您的activity继承于actionbaractivity,可以在setcontentview()方法之前调用requestwindowfeature(window.feature_no_title);隐藏标题栏;如继承于appcompactactivity,可以在androidmainfest
中的application标签中设置主题为:android:theme="@style/theme.appcompat.noactionbar",也可以实现隐藏标题栏的目的。

使用fragmentpageradapter为viewpager适配数据

在mainactivity.java 中,加入fragmentpageradapter逻辑:(在此略去三个fragment的布局及代码)

 private fragmentpageradapter adapter;
 private list<fragment> mdata;

@override
  protected void oncreate(bundle savedinstancestate) {
    super.oncreate(savedinstancestate);
    requestwindowfeature(window.feature_no_title);
    setcontentview(r.layout.activity_main);
    initview();

在initview()中,初始化fragment,并将fragment实例一次装入list中,接着,在初始化fragmentpageradapter时管理list的数据。最后调用viewpager的setadapter方法将fragmentpageradapter实例传入。

 mdata = new arraylist<>();
    mdata.add(new chatfragment());
    mdata.add(new discoverfragment());
    mdata.add(new contactsfragment());
    adapter = new fragmentpageradapter(getsupportfragmentmanager()) {
      @override
      public fragment getitem(int position) {
        return mdata.get(position);
      }

      @override
      public int getcount() {
        return mdata.size();
      }
    };
    vp_content.setadapter(adapter);

设置滑动时字体颜色的变化

为实现该功能,需要为viewpager设置setonpagechangelistener监听,并通过onpagechangelistener接口的回调方法onpagerselected(int position),监听当前滑动到了第几页:

@override
      public void onpageselected(int position) {
        log.e(tag, "onpageselected: " + position);
        resettextviewcolor();
        switch (position) {
          case 0:       
          addbadgeview();  tv_tab_chat.settextcolor(color.parsecolor("#008000"));
            break;

          case 1:
            tv_tab_discover.settextcolor(color.parsecolor("#008000"));
            break;

          case 2:
            tv_tab_contacts.settextcolor(color.parsecolor("#008000"));
            break;
        }

      }

//首先将每个选项卡的文字颜色置为黑色
 private void resettextviewcolor() {
    tv_tab_contacts.settextcolor(color.black);
    tv_tab_chat.settextcolor(color.black);
    tv_tab_discover.settextcolor(color.black);

添加badgeview

在addbadgeview();方法中首先判断badgeview是否为空,若不为空,首先将其移除,再添加新的badgeview,代码如下:

 private void addbadgeview()
 {
 if (mbadgeview != null) {
              ll_chat.removeview(mbadgeview);
            }
            mbadgeview = new badgeview(mainactivity.this);
            ll_chat.addview(mbadgeview);
            mbadgeview.setbadgecount(9);


}

设置滑动时字体颜色的变化

为实现该功能,需要为viewpager设置setonpagechangelistener监听,并通过onpagechangelistener接口的回调方法onpagerselected(int position),监听当前滑动到了第几页:

@override
      public void onpageselected(int position) {
        log.e(tag, "onpageselected: " + position);
        resettextviewcolor();
        switch (position) {
          case 0:       
          addbadgeview();  tv_tab_chat.settextcolor(color.parsecolor("#008000"));
            break;

          case 1:
            tv_tab_discover.settextcolor(color.parsecolor("#008000"));
            break;

          case 2:
            tv_tab_contacts.settextcolor(color.parsecolor("#008000"));
            break;
        }

      }

//首先将每个选项卡的文字颜色置为黑色
 private void resettextviewcolor() {
    tv_tab_contacts.settextcolor(color.black);
    tv_tab_chat.settextcolor(color.black);
    tv_tab_discover.settextcolor(color.black);

添加badgeview

在addbadgeview();方法中首先判断badgeview是否为空,若不为空,首先将其移除,再添加新的badgeview,代码如下:

 private void addbadgeview()
 {
 if (mbadgeview != null) {
              ll_chat.removeview(mbadgeview);
            }
            mbadgeview = new badgeview(mainactivity.this);
            ll_chat.addview(mbadgeview);
            mbadgeview.setbadgecount(9);


}

indicator的滑动

为了实现该indicator随手指的滑动而跟随的效果,需要在onpagechangelistener接口中的onpagescrolled()方法中编写逻辑,该方法的文档如下:

Android高仿微信5.2.1主界面及消息提醒

其中,第一个参数position表示滑动到了第几页,比如说,若从第0页滑动至第一页,那么position将一直为0,直到松手以后,滑动至第一页,position将变为1,第二个参数positionoffset表示滑动的百分比,取值范围是0-1,最后一个参数positionoffsetpixels表示滑动的像素数。

下面是从0—>1页面时打印的log,如下所示:

Android高仿微信5.2.1主界面及消息提醒

从1—->2页面时打印的log:

Android高仿微信5.2.1主界面及消息提醒

从2—->1页面时打印的log:

Android高仿微信5.2.1主界面及消息提醒

最后,可以根据(position+positionoffset)*1/3,来设置该indicator的marginleft。

首先,应为indicator设置宽度,其宽度应为屏幕宽度的1/3:

 windowmanager manager = getwindow().getwindowmanager();
    display display = manager.getdefaultdisplay();
    displaymetrics outmetrics = new displaymetrics();
    display.getmetrics(outmetrics);
    mscreenonethird = outmetrics.widthpixels / 3;

其中int型参数mscreenonethird 的单位是像素px。

设置到indicator上:

linearlayout.layoutparams lp = (linearlayout.layoutparams) iv_tab_line.getlayoutparams();
    lp.width = mscreenonethird;
    iv_tab_line.setlayoutparams(lp);

最终在onpagescrolled方法中动态改变indicator的marginleft属性:

@override
      public void onpagescrolled(int position, float positionoffset, int positionoffsetpixels) {
        linearlayout.layoutparams lp = (linearlayout.layoutparams) iv_tab_line.getlayoutparams();
        lp.leftmargin = (int) ((positionoffset * mscreenonethird) + (mscreenonethird * position));
        iv_tab_line.setlayoutparams(lp);

可实现最终效果。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。