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

ViewPager的setOnPageChangeListener方法详解

程序员文章站 2024-02-27 12:18:51
背景 最近需要实现一个自定义notification的功能。网上找了找代码,解决方案就是通过remoteviews来实现。但是在实现过程中遇到不少问题,网上也没有很好的文...

背景

最近需要实现一个自定义notification的功能。网上找了找代码,解决方案就是通过remoteviews来实现。但是在实现过程中遇到不少问题,网上也没有很好的文章描述这些问题,所以在这里做个总结,希望大家能少走点弯路。

实现

remoteviews 自定义view

这是最基础的知识点,虽然做过自定义通知的应该都清楚,但我觉得还是有必要带一下。它主要被用于appwidget和notification,它描述一个在其它进程中显示的view。以下是例子代码。从中我们可以看到remoteviews提供了一些方法来改变它的子view的值,如设置textview的文字等。

remoteviews remoteviews = new remoteviews(context.getpackagename(), r.layout.view_notification_type_0);
  remoteviews.settextviewtext(r.id.title_tv, title);
  remoteviews.settextviewtext(r.id.content_tv, content);
  remoteviews.settextviewtext(r.id.time_tv, gettime());
  remoteviews.setimageviewresource(r.id.icon_iv, r.drawable.logo);
  remoteviews.setint(r.id.close_iv, "setcolorfilter", geticoncolor());
  intent intent = new intent(context, mainactivity.class);
  intent.putextra(notice_id_key, notice_id_type_0);
  intent.setflags(intent.flag_activity_new_task | intent.flag_activity_clear_task);
  int requestcode = (int) systemclock.uptimemillis();
  pendingintent pendingintent = pendingintent.getactivity(context, requestcode, intent, pendingintent.flag_update_current);
  remoteviews.setonclickpendingintent(r.id.notice_view_type_0, pendingintent);
  int requestcode1 = (int) systemclock.uptimemillis();
  intent intent1 = new intent(action_close_notice);
  intent1.putextra(notice_id_key, notice_id_type_0);
  pendingintent pendingintent1 = pendingintent.getbroadcast(context, requestcode1, intent1, pendingintent.flag_update_current);
  remoteviews.setonclickpendingintent(r.id.close_iv, pendingintent1);

这里有几点需要注意的。

setint

这个方法被用来调用子view中需要一个int型参数的方法。如下面这句代码,调用了id为close_iv的setcolorfilter方法,参数为geticoncolor()的返回值。

remoteviews.setint(r.id.close_iv, "setcolorfilter", geticoncolor());

设置不同区域的点击pendingintent

默认的notification只能通过setcontentintent设置整体的点击事件。不过通过remoteviews我们可以设置不同地方不同的点击事件,当然这里的事件指的是pendingintent。如下,设置了点击r.id.notice_view_type_0打开一个activity,而点击r.id.close_iv会发出一个广播,可以通过这个广播的广播接收器来做一些事情,如这里是关闭当前的notification。另外还可以打开一个service。

pendingintent.getactivity(context, requestcode, intent, pendingintent.flag_update_current);
  remoteviews.setonclickpendingintent(r.id.notice_view_type_0, pendingintent);
  int requestcode1 = (int) systemclock.uptimemillis();
  intent intent1 = new intent(action_close_notice);
  intent1.putextra(notice_id_key, notice_id_type_0);
  pendingintent pendingintent1 = pendingintent.getbroadcast(context, requestcode1, intent1, pendingintent.flag_update_current);
  remoteviews.setonclickpendingintent(r.id.close_iv, pendingintent1);

设置通知的自定义view

以上我们得到了自定义的remoteviews。通过下面这段代码就能生成自定义view的notification,注意这里使用了setcontent()方法。这是网上自定义notification都会使用的方法。

notification notification = new notificationcompat.builder(context).setcontent(remoteviews).build();

但是它会有一个问题。

通过setcontent()方法获得的notification是定高的。如果view的高度比默认高度要大的话,就有一部分显示不出来。如下图

ViewPager的setOnPageChangeListener方法详解

默认情况下通知高度为64dp,当然rom不同可能会有些区别。一般文字在小于两行的情况下都是可以显示。

那么如何做到wrap_content。需要使用一些黑科技。如下:

notificationcompat.builder builder = new notificationcompat.builder(context);
if(android.os.build.version.sdk_int >= 16) {
   notification = builder.build();
   notification.bigcontentview = remoteviews;
}
notification.contentview = remoteviews;

为了理解以上代码,我们需要明确一个我们很容易忽略的问题,那就是通知是可以展开和收起的。请看以下两张图片。同样是网易云音乐的通知,图一比图二要大一些。其实图一展示的是网易云音乐通知的展开状态,使用两个手指上滑就可以缩起,也就是图二。

ViewPager的setOnPageChangeListener方法详解

ViewPager的setOnPageChangeListener方法详解

在上面的代码中我们分别设置了bigcontentview 这是展开的自定义视图,而contentview则是收起时的视图。

注意bigcontentview是在sdk16时引入的,所以需要判断一下。如果小于sdk16则只能定高了。

注意bigcontentview 的最大高度是256dp

注意bigcontentview和contentview的设置不能调转顺序,亲测这样会让contentview不显示。

另外需要注意某些rom可能不支持展开收起通知,在设置了bigcontentview之后就只显示展开的视图,而默认情况下只展示收起视图。如魅族的flyme,其它rom并没有测试,如果读者知道可以分享一下。

背景色适配

不同rom的通知背景色是不同的,所以在ui上需要注意。 主要分为两种情况。

  1. 背景色为有透明度的黑色,如miui、flyme。
  2. 背景色为白色,如原生的5.0之后的rom、华为部分rom。

主要有两种方案。

固定背景色

也就是设置一个固定的背景色,文字和icon颜色都可以固定。如下图。

ViewPager的setOnPageChangeListener方法详解

这有一个缺点,我们在图中也看到了,那就是某些rom的notification会有一个左右的padding,如miui的就特别明显,如果固定背景色就会很难看。

所以这种方法虽然简答,但是不建议使用。

透明背景色

另一种方法就是让背景透明。那么文字和icon的颜色怎么办呢?很简单,跟随系统的notification中文字的样式。如下设置了textview的style为默认通知中info的样式。其它相关style包括textappearance.statusbar.eventcontent.line2、textappearance.statusbar.eventcontent.info等。

<textview
   android:id="@+id/content_tv"
   android:layout_width="wrap_content"
   android:layout_height="wrap_content"
   android:textappearance="@style/textappearance.statusbar.eventcontent.info"
   tools:text="41个同校小伙伴参与讨论"
   android:layout_margintop="4dp"
   android:singleline="true"/>

需要注意的一点是android5.0之后使用了不同的style名表示通知样式。 我们需要创建一个layout-v21文件夹,并新建一个在5.0之后使用的自定义通知样式。如下同样是设置textview的style为info的样式,但我们使用的是style是@android:style/textappearance.material.notification.info。

<textview
   android:id="@+id/content_tv"
   android:layout_width="wrap_content"
   android:layout_height="wrap_content"
   android:textsize="9sp"
   android:textappearance="@android:style/textappearance.material.notification.info"
   tools:text="41个同校小伙伴参与讨论"
   android:layout_margintop="4dp"
   android:singleline="true"/>

另外如果自定义view中有icon,那么icon的颜色也需要适应背景,可以选择一个灰色,如#999999,原生安卓黑色和白色的文字内容颜色都为该值。

或者根据不同的背景色设置不同的颜色,通过上面提到的setint方法。imageview的setcolorfilter方法可以设置图案颜色为某种纯色。但是目前我还没有找到很好的方法获取默认通知的背景色,如果读者找到了望告知。

remoteviews.setint(r.id.close_iv, "setcolorfilter", geticoncolor());

另外需要注意的是很多手机的rom可能不会对以上的style做修改,而是采用自己的样式,这样就比较蛋疼。借鉴网易云音乐的方法,在api 21以下,因为大部分手机都使用黑色背景,所以采用透明背景,文字颜色跟随系统。而在api大于等于21时,因为原生android采用了白色的通知背景,而很多手机厂商的rom的通知仍然为黑色背景,因此为了通用,采用一个固定的黑色背景,文字颜色同样跟随系统。在这基础上,对某部分特定手机rom做适配,如华为等在api 21以下通知背景也为白色。这应该是比较完美的做法,但是需要花费较多的时间,所以如果为了简单起见,还是采用方案一吧。

最终效果

ViewPager的setOnPageChangeListener方法详解

ViewPager的setOnPageChangeListener方法详解

总结

以上即为我在自定义notification中遇到的一些问题以及解决方案。目前还有两点有待进一步补充和完善。

获取默认通知背景色,或者使图标颜色与背景色适配的方案。

不支持notification展开收起的rom,目前知道的仅有flyme。

示例代码地址

https://github.com/beautifulsoup/cnotification

以上所述是小编给大家介绍的viewpager的setonpagechangelistener方法详解,希望对大家有所帮助