Android:若wifi未开启给出相应弹框

预期效果

用户正在与本应用程序界面交互时,或本应用程序退到后台时,只要检测wifi被关闭,都能弹出相应对话框提示用户“请打开wifi”,用户必须点击对话框中的确定按钮跳转到“网络设置界面”。


请打开wifi.png

思路

1. 监听wifi开关状态改变广播、在广播接收器中进行逻辑判断

想要实现上述效果的话,存在这样一个问题,因为当我们被通知需要打开wifi时可能处于任何一个界面,难道要为每个界面上都编写一个弹出对话框的逻辑吗?

不能呀!借助广播的知识,当wifi开关状态改变时会收到一条系统广播,这条广播就是用于进行处理弹框逻辑的通知。所以“请打开wifi”弹框的逻辑写在接收这条广播的广播接收器中。这样该逻辑判断不会依附于任何界面,不管在程序的任何地方,只要发出这样一条广播,就可以完成这个逻辑判断。

系统广播WifiManager.WIFI_STATE_CHANGED_ACTION,通过监听wifi开关状态变化的广播,
wifi开关状态

WifiManager.WIFI_STATE_DISABLED ;  (1)  关闭
WifiManager..WIFI_STATE_ENABLED ;  (3)  打开
WifiManager..WIFI_STATE_DISABLING ;(0)  关闭中
WifiManager..WIFI_STATE_ENABLING  ;  (2)  打开中
WifiManager..WIFI_STATE_UNKONW  ;  (4)  未知

2. 广播接收器注册、取消注册的地方

一般广播接收器是在onResume()中注册,onPause()中取消注册。从Activity的生命周期特点可知,onResume()方法被回调后Activity可与用户交互,onPause()被回调后Activity不可与用户交互。这样的做法:保证了始终只有栈顶的活动才能接收到广播,非栈顶的活动不应该也没必要去接收这条广播,当一个活动失去栈顶位置时自动取消广播接收器的注册。

但现在,我们的需求是,在Activity可与用户交互,或不可与用户交互时都可弹框提示用户。故选择在onCreate()中注册广播接收器(Activity始终可以接收这条广播),在onDestroy()中取消注册。

3.广播接收器onReceive()方法中的逻辑

onReceive()方法中的逻辑.png
public class WifiListenerReceiver extends BroadcastReceiver {
    public static final String MY_WIFI_BROADCAST="MY_WIFI_BROADCAST";

    @Override
    public void onReceive(Context context, Intent intent) {
        String action=intent.getAction();
        if (WifiManager.WIFI_STATE_CHANGED_ACTION.equals(action) || Constant.WIFI_LISTENER_ACTION.equals(action)) {
            int wifiState = DeviceInfoUtil.getWifiState(context);
            Log.e("LoginActivity", "收到wifi广播"+wifiState);
            switch (wifiState) {
                case WifiManager.WIFI_STATE_DISABLED:
                case WifiManager.WIFI_STATE_DISABLING:
                case WifiManager.WIFI_STATE_UNKNOWN:
                        /*
                         * wifi处于关闭状态:若“请打开wifi对话框”正在显示,将该对话框显示在前台;
                         *  未正在显示:创建新的并显示
                          */
                    Intent intent1 = new Intent(context, OpenWifiActivity.class);
                    // 以SingleTask的模式启动(栈内复用)
                    intent1.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
                    context.startActivity(intent1);
                    break;
                case WifiManager.WIFI_STATE_ENABLED:
                case WifiManager.WIFI_STATE_ENABLING:
                                /*
                                 * wifi处于开启状态:若“请打开wifi对话框”正在显示,将该对话框关闭;
                         *  未正在显示:Nothing
                                 */
                    OpenWifiActivityCollector.finishAll();
                    break;
            }
        }


    }

}

3.1“请打开wifi”框的制作:用Activity做成Dialog的样式

将OpenWifiActivity主题设置为

<style name="FDialogStyle" parent="AlertDialog.AppCompat">
        <item name="android:windowBackground">@android:color/transparent</item> 设置dialog的背景,此处为系统给定的透明值
        <item name="android:windowFrame">@null</item>                Dialog的windowFrame框为无
        <item name="android:windowNoTitle">true</item>         是否显示标题
        <item name="android:windowIsFloating">true</item>            是否浮现在activity之上
        <item name="android:windowIsTranslucent">true</item>         是否半透明
        <item name="android:windowContentOverlay">@null</item>       是否有覆盖
        <item name="android:windowAnimationStyle">@android:style/Animation.Dialog</item>   设置Activity出现方式
        <item name="android:backgroundDimEnabled">true</item>        背景是否模糊显示
    </style>

3.2可能会出现:连续弹出多个“请打开wifi”框

此处是将Activity做成Dialog的效果显示出来。故将Activity的启动模式设置成SingleTask,就可保证栈内复用,始终只有一个弹框

 <activity
     android:name=".module.wifi.OpenWifiActivity"
     android:theme="@style/FDialogStyle"
     android:launchMode="singleTask"></activity>

3.3会出现的情况:“请打开wifi”框正在显示,通过通知栏打开wifi,但该框未被销毁

如何销毁打开的Activity?此处参考郭霖-第一行代码-随时随地退出程序功能的实现

用一个专门的集合类对所有打开的OpenWifiActivity实例进行存储。当需要关闭OpenWifiActivity时,finish掉集合中保存的该活动的所有实例

public class OpenWifiActivity extends Activity {
    @BindView(R.id.btn_openwifi_sure)
    Button btn_sure;
    @BindView(R.id.btn_openwifi_cancel)
    Button btn_cancel;


    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_openwifi);
        KnifeKit.bind(this);
        // 加入到ActivityCollector中
        OpenWifiActivityCollector.addActivity(this);
        btn_sure.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                DeviceInfoUtil.openWifiSetting(getApplicationContext());
                finish();

            }
        });

        btn_cancel.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                finish();
            }
        });
    }

    @Override
    protected void onDestroy() {
        super.onDestroy();
        // 从管理类中移除实例
        OpenWifiActivityCollector.removeActivity(this);
    }
}
public class OpenWifiActivityCollector {

    public static List<Activity> sActivities = new ArrayList<>();

    public static void addActivity(Activity activity) {
        sActivities.add(activity);
    }

    public static void removeActivity(Activity activity) {
        sActivities.remove(activity);
    }

    public static void finishAll() {

        for (Activity activity : sActivities) {
            if (!activity.isFinishing()) {
                activity.finish();
            }
        }
    }
}

代码实现【下面给出相关代码文件】

1.base

base.png

MyBaseActivity

public abstract class MyBaseActivity extends AppCompatActivity implements MyUiCallback {
    public static AlertDialog mAlertDialog;
    protected Activity context;
    private Unbinder unbinder;
    // GPS检测监听
    private GPSListenerRecevicer receiver;
    // Wifi状态改变监听
    private WifiListenerReceiver mWifiListenerReceiver;

    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        Log.e(getClass().getSimpleName(), "onCreate");
        this.context = this;
        beforeSetConentView(savedInstanceState);
        if (getLayoutId() > 0) {
            setContentView(getLayoutId());
            unbinder = KnifeKit.bind(this);
        }
        setListener();
        initData(savedInstanceState);
        /*
         * 在onCreate中注册WiFi广播监听,在onDestroy中取消注册
         * ylj
         */
        registerWifiReceiver();
    }

    /**
     * 注册wifi状态改变广播接收器
     */
    protected  void registerWifiReceiver(){
        mWifiListenerReceiver = new WifiListenerReceiver();
        IntentFilter filter = new IntentFilter();
        filter.addAction(WifiManager.WIFI_STATE_CHANGED_ACTION);
        filter.addAction(Constant.WIFI_LISTENER_ACTION);
        registerReceiver(mWifiListenerReceiver, filter);
    }

    @Override
    public void beforeSetConentView(Bundle savedInstanceState) {

    }

    @Override
    public void setListener() {

    }


    @Override
    protected void onResume() {
        super.onResume();
        Log.e(getClass().getSimpleName(), "onResume");
      
        /* 
         * 
         * 在每次resume中,发送一条检测wifi状态的广播
         */
        sendCheckWifiStateBroadcast();
    }

    protected  void sendCheckWifiStateBroadcast(){
        Intent intent = new Intent();
        intent.setAction(Constant.WIFI_LISTENER_ACTION);
        sendBroadcast(intent);
        Log.e(getClass().getSimpleName(), "发出MY_WIFI_BROADCAST广播");
    }

    @Override
    protected void onDestroy() {
        super.onDestroy();
        Log.e(getClass().getSimpleName(), "onDestroy");
        /*
         *  取消注册wifi状态改变广播接收器
         *  ylj
         */
        unregisterWifiReceiver();
    }

    /**
     *  取消注册wifi状态改变广播接收器
     *  ylj
     */
    private void unregisterWifiReceiver() {
        if (mWifiListenerReceiver != null) {
            unregisterGPSListener();
            mWifiListenerReceiver = null;
        }
    }

    
}

MyUiCallback

/**
 * <pre>
 *     author : 杨丽金
 *     time   : 2018/03/23
 *     desc   :
 * </pre>
 */
public interface MyUiCallback {
    // 在setContentView()之前调用的方法
    void beforeSetConentView(Bundle savedInstanceState);

    // 下面的三个方法都是在setContentView()之后调用的
    void initData(Bundle savedInstanceState);

    void setListener();

    int getLayoutId();
}

2.wifi

wifi.png

OpenWifiActivity

public class OpenWifiActivity extends Activity {
    @BindView(R.id.btn_openwifi_sure)
    Button btn_sure;
    @BindView(R.id.btn_openwifi_cancel)
    Button btn_cancel;


    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_openwifi);
        KnifeKit.bind(this);
        // 加入到ActivityCollector中
        OpenWifiActivityCollector.addActivity(this);
        btn_sure.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                DeviceInfoUtil.openWifiSetting(getApplicationContext());
                finish();

            }
        });

        btn_cancel.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                finish();
            }
        });
    }

    @Override
    protected void onDestroy() {
        super.onDestroy();
        // 从管理类中移除实例
        OpenWifiActivityCollector.removeActivity(this);
    }
}

OpenWifiActivityCollector

/**
 * <pre>
 *     author : 杨丽金
 *     time   : 2018/06/05
 *     desc   : 管理OPENWiFiActivity实例
 * </pre>
 */
public class OpenWifiActivityCollector {

    public static List<Activity> sActivities = new ArrayList<>();

    public static void addActivity(Activity activity) {
        sActivities.add(activity);
    }

    public static void removeActivity(Activity activity) {
        sActivities.remove(activity);
    }

    public static void finishAll() {

        for (Activity activity : sActivities) {
            if (!activity.isFinishing()) {
                activity.finish();
            }
        }
    }
}

WifiListenerBroadcast

public class WifiListenerReceiver extends BroadcastReceiver {
    public static final String MY_WIFI_BROADCAST="MY_WIFI_BROADCAST";

    @Override
    public void onReceive(Context context, Intent intent) {
        String action=intent.getAction();
        if (WifiManager.WIFI_STATE_CHANGED_ACTION.equals(action) || Constant.WIFI_LISTENER_ACTION.equals(action)) {
            int wifiState = DeviceInfoUtil.getWifiState(context);
            Log.e("LoginActivity", "收到wifi广播"+wifiState);
            switch (wifiState) {
                case WifiManager.WIFI_STATE_DISABLED:
                case WifiManager.WIFI_STATE_DISABLING:
                case WifiManager.WIFI_STATE_UNKNOWN:
                        /*
                         * wifi处于关闭状态:若“请打开wifi对话框”正在显示,将该对话框显示在前台;
                         *  未正在显示:创建新的并显示
                          */
                    Intent intent1 = new Intent(context, OpenWifiActivity.class);
                    // 以SingleTask的模式启动(栈内复用)
                    intent1.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
                    context.startActivity(intent1);
                    break;
                case WifiManager.WIFI_STATE_ENABLED:
                case WifiManager.WIFI_STATE_ENABLING:
                                /*
                                 * wifi处于开启状态:若“请打开wifi对话框”正在显示,将该对话框关闭;
                         *  未正在显示:Nothing
                                 */
                    OpenWifiActivityCollector.finishAll();
                    break;
            }
        }


    }

}

参考文献

郭霖-第一行代码-强制下线功能的实现

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 180,161评论 5 436
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 75,785评论 2 342
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 128,551评论 0 297
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 48,708评论 1 254
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 57,479评论 4 337
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 43,558评论 1 250
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 34,544评论 3 364
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 33,100评论 0 237
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 37,231评论 1 278
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 32,788评论 2 285
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 34,457评论 1 300
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 30,484评论 3 294
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 35,750评论 3 287
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 27,507评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 28,714评论 1 240
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 38,993评论 2 318
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 38,555评论 2 318

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 170,863评论 25 707
  • 面试题总结 通用 安卓学习途径, 寻找资料学习的博客网站 AndroidStudio使用, 插件使用 安卓和苹果的...
    JingBeibei阅读 1,629评论 2 21
  • 1.什么是Activity?问的不太多,说点有深度的 四大组件之一,一般的,一个用户交互界面对应一个activit...
    JoonyLee阅读 5,708评论 2 51
  • 星期五,我向老妈请假,我说明天不来看你了,这个周六很忙,上午要去自来水厂采风,下午要听个文学讲座,晚上还有文...
    苏州的蔷薇花开阅读 580评论 11 2
  • 我虽然不能普度众生,但我可以祸害苍生。啊哈哈。我有一个让人恨得牙痒痒的本领,再好的事被我这么一说,乌鸦嘴,准灵。我...
    少音阅读 1,571评论 0 0