前日,一小伙伴问我一个问题,说它解决了半天都没解决这个问题,截图如下:

大概楼主理解如下:

如果在应用中有一个判断wifi的开关和一个当前音量大小的seekbar以及一个获取当前电量多少的按钮,想知道如果按home键后调整了wifi开关信息以及媒体音量信息,再切换到前台UI如何才会实时刷新。其实这个问题不难解决,如果你了解activity的生命周期,只需要把设置开关和seekbar的信息放在onResume中就好了,因为无论是锁屏后打开或者是切换后台再前台都是会调用onResume的。但不由得滋生一个问题,大家都知道APP在前台的情况下用户依然是可以下拉状态栏设置Wifi开关信息的,对于音量信息也是可以侧边增减,那APP一直在前台,生命周期明显是无法实时更新了,那我们应该如何解决呢?没错,没当改变系统属性的时候,都会发出系统广播,我们只需要去写一个接收器,并根据它做响应的操作就好了。

分析至此,楼主就把给这位小伙伴写的一些代码分享给大家,也可以帮助不太熟悉的小伙伴更加了解android的广播以及回调机制。对于还不太明白java的回调是什么意思的小伙伴,也可以看看。

1)由于要使用到系统属性,所以先申明权限。

 <!--wifi管理必备权限-->
<uses-permission android:name="android.permission.ACCESS_WIFI_STATE"/>
<uses-permission android:name="android.permission.CHANGE_WIFI_STATE"/> <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE"/> <!--操作音频需要权限-->
<uses-permission android:name="android.permission.MODIFY_AUDIO_SETTINGS"/>

2)然后写一个广播接收器,做好过滤,并申明一个回调接口,用于当广播接收到的时候提醒主线程更新UI。

 package com.example.nanchen.maweinaitest;

 import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.media.AudioManager;
import android.net.wifi.WifiManager;
import android.util.Log; import static android.content.Intent.ACTION_BATTERY_CHANGED; /**
* @author nanchen
* @fileName MaWeiNaiTest
* @packageName com.example.nanchen.maweinaitest
* @date 2016/11/05 21:35
*/ public class MyStatusReceiver extends BroadcastReceiver { private static final String TAG = "MyStatusReceiver";
private StatusCallback mStatusCallback = MainActivity.callback; public MyStatusReceiver(){
} @Override
public void onReceive(Context context, Intent intent) { String action = intent.getAction();
Log.e(TAG,action);
Log.e(TAG,intent.getAction()+" ==== "); // 首先判断它是否是电量变化的Broadcast Action
if (ACTION_BATTERY_CHANGED.equals(action)) {//如果监听到电量改变广播
// 获取当前电量
int level = intent.getIntExtra("level", 0);
// 电量的总刻度
int scale = intent.getIntExtra("scale", 100);
// 把它转换为百分比
// mActivity.mTextView.setText(level * 100 / scale + "%");
String str = level * 100 / scale + "%"; Log.e(TAG,level+"");
Log.e(TAG,scale+"");
Log.e(TAG,str+""); mStatusCallback.onPowerChanged(level * 100 / scale + "%");
}
// 监听一下音量
if ("android.media.VOLUME_CHANGED_ACTION".equals(action)){
// mActivity.mSeekBar.setProgress(mActivity.mAudioManager.getStreamVolume(AudioManager.STREAM_SYSTEM));
AudioManager audioManager = (AudioManager) context.getSystemService(Context.AUDIO_SERVICE);
int progress = audioManager.getStreamVolume(AudioManager.STREAM_SYSTEM);
Log.e(TAG,progress+"");
mStatusCallback.onAudioChanged(audioManager.getStreamVolume(AudioManager.STREAM_SYSTEM));
}
if (WifiManager.WIFI_STATE_CHANGED_ACTION.equals(action)){
WifiManager wifiManager = (WifiManager) context.getSystemService(Context.WIFI_SERVICE);
mStatusCallback.onWifiChanged(wifiManager.isWifiEnabled());
}
} /**
* 一个回调接口
*/
public interface StatusCallback {
/**
* 当电量改变时应该调用的回调接口
* @param status 当前电量百分比
*/
void onPowerChanged(String status); /**
* 当音频音量改变时会调用的回调接口
* @param status 当前音量数值
*/
void onAudioChanged(int status); /**
* 当wifi改变时会调用的回调接口
* @param status wifi的开关 true-开 false - 关
*/
void onWifiChanged(boolean status);
}
}

3)别忘了在mainfest申明

 <receiver android:name=".MyStatusReceiver">
<intent-filter>
<action android:name="android.intent.action.BATTERY_CHANGED"/>
<action android:name="android.media.VOLUME_CHANGED_ACTION"/>
<action android:name="android.net.wifi.WIFI_STATE_CHANGED_ACTION"/>
</intent-filter>
</receiver>

4)布局就采用的这位小伙伴的布局

 <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"> <Switch
android:id="@+id/wifi"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:textOn="开"
android:textOff="关"
android:text="WiFi"/> <SeekBar
android:id="@+id/seekBar"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginTop="16dp"/> <Button
android:id="@+id/btn"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginTop="16dp"
android:text="当前电量百分比" /> <LinearLayout
android:orientation="horizontal"
android:layout_width="match_parent"
android:layout_height="wrap_content"> <TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="当前电量百分比为:"/> <TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="0%"
android:id="@+id/text"/>
</LinearLayout> </LinearLayout>

5)最后是MainActivity,注意广播注销,否则造成内存泄漏!

 package com.example.nanchen.maweinaitest;

 import android.content.Context;
import android.content.IntentFilter;
import android.media.AudioManager;
import android.net.wifi.WifiManager;
import android.os.Bundle;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.CompoundButton;
import android.widget.CompoundButton.OnCheckedChangeListener;
import android.widget.SeekBar;
import android.widget.SeekBar.OnSeekBarChangeListener;
import android.widget.Switch;
import android.widget.TextView;
import android.widget.Toast; import com.example.nanchen.maweinaitest.MyStatusReceiver.StatusCallback; import static android.content.Intent.ACTION_BATTERY_CHANGED; public class MainActivity extends ActivityBase implements StatusCallback { public static StatusCallback callback; private static final String TAG = "MainActivity";
private Switch mSwitchWifi;
private SeekBar mSeekBar;
private WifiManager mWifiManager;
private AudioManager mAudioManager;
private TextView mTextView;
private MyStatusReceiver mMyStatusReceiver; @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
callback = this; bindView(); initManager(); bindListener(); } private void initManager() {
// 获取Wifi管理器
mWifiManager = (WifiManager) getSystemService(Context.WIFI_SERVICE);
// 把动态获取的信息放在onResume设置 避免按home键后再把APP切换到前台获取不到正常的数据 // 获取音频管理器
mAudioManager = (AudioManager) getSystemService(Context.AUDIO_SERVICE);
} /**
* 绑定监听
*/
private void bindListener() {
// 为wifi开关事件设置监听
mSwitchWifi.setOnCheckedChangeListener(new OnCheckedChangeListener() {
@Override
public void onCheckedChanged(CompoundButton buttonView, boolean isChecked) {
if (!isChecked) {
buttonView.setChecked(false);
mWifiManager.setWifiEnabled(false);
Toast.makeText(MainActivity.this, "wifi关闭成功!", Toast.LENGTH_SHORT).show();
} else {
buttonView.setChecked(true);
mWifiManager.setWifiEnabled(true);
Toast.makeText(MainActivity.this, "wifi开启成功!", Toast.LENGTH_SHORT).show();
}
}
}); // 再动态监听SeekBar
mSeekBar.setOnSeekBarChangeListener(new OnSeekBarChangeListener() {
@Override
public void onProgressChanged(SeekBar seekBar, int progress, boolean fromUser) {
// 停止滑动
mSeekBar.setProgress(progress);
// 三个参数一次是 模式,值,标志位
mAudioManager.setStreamVolume(AudioManager.STREAM_SYSTEM, progress, 0);
} @Override
public void onStartTrackingTouch(SeekBar seekBar) { } @Override
public void onStopTrackingTouch(SeekBar seekBar) { }
}); // 注册广播,添加三个Action
IntentFilter intentFilter = new IntentFilter();
intentFilter.addAction(ACTION_BATTERY_CHANGED);
intentFilter.addAction(AudioManager.RINGER_MODE_CHANGED_ACTION);
intentFilter.addAction(WifiManager.WIFI_STATE_CHANGED_ACTION);
mMyStatusReceiver = new MyStatusReceiver();
registerReceiver(mMyStatusReceiver, intentFilter); // 注册监听广播
} private int max;
private int current; /**
* 设置wifi开关
*/
private void setWifiSwitch() {
if (mWifiManager.isWifiEnabled()) {
mSwitchWifi.setChecked(true);
} else {
mSwitchWifi.setChecked(false);
}
} @Override
protected void onResume() {
super.onResume();
// 先动态设置wifi
setWifiSwitch(); // 再动态设置音频音量 参数为音量模式
max = mAudioManager.getStreamMaxVolume(AudioManager.STREAM_SYSTEM); // 最大音量
current = mAudioManager.getStreamVolume(AudioManager.STREAM_SYSTEM); // 当前音量 mSeekBar.setMax(max);// 设置seekBar
mSeekBar.setProgress(current);
} @Override
protected void onPause() {
super.onPause();
// 一定记得注销广播,否则会造成内存泄漏
unregisterReceiver(mMyStatusReceiver);
} @SuppressWarnings("ConstantConditions")
private void bindView() {
mSwitchWifi = (Switch) findViewById(R.id.wifi);
mSeekBar = (SeekBar) findViewById(R.id.seekBar);
mTextView = (TextView) findViewById(R.id.text);
findViewById(R.id.btn).setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
// 在这里获取当前电量信息 // 这里就不写了,实际上监听系统广播,它会自动实时获取电量信息
}
});
} @Override
public void onPowerChanged(String status) {
mTextView.setText(status);
} @Override
public void onAudioChanged(final int status) {
mSeekBar.setProgress(status);
} @Override
public void onWifiChanged(boolean status) {
mSwitchWifi.setChecked(status);
} }

大概运行图如下:

代码已上传至github:https://github.com/nanchen2251/ReceiverDemo

【用户交互】APP没有退出前台但改变系统属性如何实时更新UI?监听系统广播,让用户交互更舒心~的更多相关文章

  1. 后台自动运行,定期记录定位数据(Hbuilder监听 app由前台切换到后台、切换运行环境的 监听方法)

    http://ask.dcloud.net.cn/question/28090 https://blog.csdn.net/qq_37508970/article/details/86649703 各 ...

  2. 监听指定端口数据交互(HttpListenerContext )

    很怀念以前做机票的日子,,,,可惜回不去 以前的项目中的,拿来贴贴 场景:同步第三方数据,监听指定地址(指定时间间隔,否则不满足,因为需要处理粘包问题,改篇未实现) 主要内容四个文件:下面分别说下每个 ...

  3. datePicker 及 timePicker 监听事件 获取用户选择 年月日分秒信息

    public class MainActivity extends AppCompatActivity { private TimePicker timePicker; private DatePic ...

  4. 重学 Java 设计模式:实战观察者模式「模拟类似小客车指标摇号过程,监听消息通知用户中签场景」

    作者:小傅哥 博客:https://bugstack.cn - 原创系列专题文章 沉淀.分享.成长,让自己和他人都能有所收获! 一.前言 知道的越多不知道的就越多 编程开发这条路上的知识是无穷无尽的, ...

  5. Linux(Centos7)安装Oracle11.2.0数据字典初始化,监听,网络,创建用户等部分配置

    #创建数据字典和pl/sql包 @/u01/app/oracle/product/11.2.0/db_1/rdbms/admin/catalog.sql; @/u01/app/oracle/produ ...

  6. vue学习-day05 -- 案例:名字合并(监听data数据的改变)

    1.案例:名字合并(监听data数据的改变) 使用keyup事件监听data数据的改变 <!DOCTYPE html> <html> <head> <titl ...

  7. iOS 监听控件某个属性的改变observeValueForKeyPath

    创建一个测试的UIButton #import "ViewController.h" @interface ViewController () @property(nonatomi ...

  8. ionic app 监听网络功能

    安装cordova插件: cordova plugin add cordova-plugin-network-information 在app.js 的run()里面的function()注入$cor ...

  9. Android 监听APP进入后台或切换到前台方案对比

    在我们开发的过程中,经常会遇到需要我们判断app进入后台,或者切换到前台的情况.比如我们想判断app切换到前台时,显示一个解锁界面,要求用户输入解锁密码才能继续进行操作:我们想判断app切换到后台,记 ...

随机推荐

  1. 中国移动测试大会 PPT 和视频

    PPT网盘链接:http://pan.baidu.com/s/1c0prdoG优酷专辑:http://v.youku.com/v_show/id_XMTI5NjExNjIwOA==.html?f=25 ...

  2. struts2:数据校验,通过Action中的validate()方法实现校验,图解

    根据输入校验的处理场所的不同,可以将输入校验分为客户端校验和服务器端校验两种.服务器端验证目前有两种方式: 第一种 Struts2中提供了一个com.opensymphony.xwork2.Valid ...

  3. VisualVM连接远程Java进程

    jstatd是一个RMI(Remove Method Invocation)的server应用,用于监控jvm的创建和结束,并且提供接口让监控工具(如VisualVM)可以远程连接到本机的jvms . ...

  4. 轻松使用px为单位开发移动端页面

    研究移动端页面已经有许久了,一直执着于rem来开发,不谈性能怎么样,单从工作效率上看影响了不少,首先要固定设计稿的宽度,一般都是固定在640px,然后在根据根目录的字体大小来计算出每个元素的rem的值 ...

  5. 【动态规划】XMU 1583 Sequence

    题目链接: http://acm.xmu.edu.cn/JudgeOnline/problem.php?id=1583 题目大意: T组数据,对于n(n<=6000)给定序列Xn(Xn<= ...

  6. linux c++ 加载动态库常用的三种方法

    链接库时的搜索路径顺序:LD_LIBRARY_PATH --> /etc/ld.so.conf --> /lib,/usr/lib 方法1. vi .bash_profile    设置环 ...

  7. Two-phase clustering process for outliers detection 文章翻译

    基于二阶段聚集模式的异常探测 M.F .Jiang, S.S. Tseng *, C.M. Su 国立交通大学计算机与信息科学系,中国台北市新竹路100150号 1999年11月17日; 2000年4 ...

  8. MySQL Group Replication 动态添加成员节点

    前提: MySQL GR 3节点(node1.node2.node3)部署成功,模式定为多主模式,单主模式也是一样的处理. 在线修改已有GR节点配置 分别登陆node1.node2.node3,执行以 ...

  9. indexOf刚开始写成IndexOf出错

    {{# if(d.fronturlmin ==null||d.fronturlmin ==""){ }} <img src="@System.Configurati ...

  10. ASP.NET Web API中通过URI显示实体中的部分字段

    有时候我们可能不想显示某个实体中的所有字段.比如客户端发出如下请求: locaohost:43321/api/groups/1/items?fields=idlocaohost:43321/api/g ...