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

大概楼主理解如下:

如果在应用中有一个判断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. 总结:Mac前端开发环境的搭建(配置)

    新年新气象,在2016年的第一天,我入手了人生中第一台自己的电脑(大一时好友赠送的电脑在一次无意中烧坏了主板,此后便不断借用别人的或者网站的).macbook air,身上已无分文...接下来半年的房 ...

  2. 菜鸟学Struts2——零配置(Convention )

    又是周末,继续Struts2的学习,之前学习了,Struts的原理,Actions以及Results,今天对对Struts的Convention Plugin进行学习,如下图: Struts Conv ...

  3. BootStrap_02之全局样式及组件

    1.BootStrap指定的四种屏幕尺寸: ①超大PC屏幕--lg(large):w>=1200px: ②中等PC屏幕--md(medium):1200px>w>=992px: ③P ...

  4. 2.WindowsServer2012R2装完的一些友好化设置

    网站部署之~Windows Server | 本地部署 http://www.cnblogs.com/dunitian/p/4822808.html#iis 1.桌面图标(控制面板里面屏蔽了,得自己输 ...

  5. Matlab 高脚杯模型切片

    前言:此文为去年我替人做的一个课题,觉得比较简洁,图形也比较美观,因此放在博文里 数据源我放到了百度云盘高脚杯数据源 有兴趣的可以下载数据,跑程序试一下.也可以单独看看代码,看下实现过程. 主函数 % ...

  6. 利用PowerShell复制SQLServer账户的所有权限

    问题 对于DBA或者其他运维人员来说授权一个账户的相同权限给另一个账户是一个很普通的任务.但是随着服务器.数据库.应用.使用人员地增加就变得很枯燥乏味又耗时费力的工作.那么有什么容易的办法来实现这个任 ...

  7. CentOS上 Mono 3.2.8运行ASP.NET MVC4经验

    周一到周三,折腾了两天半的时间,经历几次周折,在小蝶惊鸿的鼎力帮助下,终于在Mono 3.2.8上运行成功MVC4.在此总结经验如下: 系统平台的版本: CentOS 6.5 Mono 3.2.8 J ...

  8. 深入浅出聊优化:从Draw Calls到GC

    前言: 刚开始写这篇文章的时候选了一个很土的题目...<Unity3D优化全解析>.因为这是一篇临时起意才写的文章,而且陈述的都是既有的事实,因而给自己“文(dou)学(bi)”加工留下的 ...

  9. lucene 基础知识点

    部分知识点的梳理,参考<lucene实战>及网络资料 1.基本概念 lucence 可以认为分为两大组件: 1)索引组件 a.内容获取:即将原始的内容材料,可以是数据库.网站(爬虫).文本 ...

  10. Vue.js——基于$.ajax实现数据的跨域增删查改

    概述 之前我们学习了Vue.js的一些基础知识,以及如何开发一个组件,然而那些示例的数据都是local的.在实际的应用中,几乎90%的数据是来源于服务端的,前端和服务端之间的数据交互一般是通过ajax ...