美文网首页
Android:若wifi未开启给出相应弹框

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

作者: 四喜汤圆 | 来源:发表于2018-06-05 19:16 被阅读179次

    预期效果

    用户正在与本应用程序界面交互时,或本应用程序退到后台时,只要检测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;
                }
            }
    
    
        }
    
    }
    

    参考文献

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

    相关文章

      网友评论

          本文标题:Android:若wifi未开启给出相应弹框

          本文链接:https://www.haomeiwen.com/subject/xjhmsftx.html