10分钟让你实如今APP中对网络状态变化进行全局提示

永远不要指望用户按照你预设的步骤操做 APPjava

一个新项目刚刚开始推广工做,市场人员向我抱怨用户使用时总会出现各类各样的问题,大部分问题都是由于用户操做不当致使的,可是在用户眼中的结论就是“大家的 APP 很差用”。android

举一个例子,有的用户在使用时禁用了 APP 访问移动网络,或者有的用户干脆都没有打开移动数据开关或者 WIFI 开关。可是做为开发人员,咱们应该避免用户思考,当用户使用出现问题时,APP 应该可以引导用户前往设置,故有此文。网络

咱们但愿当用户网络链接不可用时,及时提醒用户当前的网络状态。当链接恢复时,将提示用的视图隐藏,而且咱们但愿这个提示视图能够工做在全部须要网络的页面中。ide

思路以下:使用 BaseActivity ,全部页面继承该文件,在该文件中实现根据网络状态显示提示、隐藏提示。布局

好了,废话少说,show u the code。post

1. 实现监听网络状态变动的广播接收器

咱们使用广播接收器接收网络变化的 Intent,这里直接使用静态注册的方法,由于咱们不须要在每一个页面单独注册这个 Receiver,那过重量级了。this

NetworkConnectChangedReceiver.java
    
    public class NetworkConnectChangedReceiver extends BroadcastReceiver {
        private static final String TAG = "NetworkConnectChanged";
        @Override
        public void onReceive(Context context, Intent intent) {
            //**判断当前的网络链接状态是否可用*/
            boolean isConnected = NetUtils.isConnected(context);
            Log.d(TAG, "onReceive: 当前网络 " + isConnected);
            EventBus.getDefault().post(new NetworkChangeEvent(isConnected));
        }
    }
复制代码

事件Event:spa

public class NetworkChangeEvent {
        public boolean isConnected; //是否存在网络
    
        public NetworkChangeEvent(boolean isConnected) {
            this.isConnected = isConnected;
        }
    }
复制代码

判断网络链接是否可用:.net

/** * 判断网络是否链接 * @param context * @return */
        public static boolean isConnected(Context context) {
            ConnectivityManager connectivity = (ConnectivityManager) context
                    .getSystemService(Context.CONNECTIVITY_SERVICE);
    
            if (null != connectivity) {
                NetworkInfo info = connectivity.getActiveNetworkInfo();
                if (null != info && info.isConnected()) {
                    if (info.getState() == NetworkInfo.State.CONNECTED) {
                        return true;
                    }
                }
            }
            return false;
        }
复制代码

静态注册Receiver:code

<receiver android:name=".receiver.NetworkConnectChangedReceiver">
        <intent-filter>
            <action android:name="android.NET.conn.CONNECTIVITY_CHANGE" />
            <action android:name="android.Net.wifi.WIFI_STATE_CHANGED" />
            <action android:name="android.net.wifi.STATE_CHANGE" />
        </intent-filter>
    </receiver>
复制代码

2. 在 BaseActivity中监听事件并处理提示视图

看到 EventBus 的时候你是否是已经知道个人实现方式了(笑 XD),是的就是那个已经好久没人提了的 EventBus。固然还可使用观察者模式来实现,这样就不用依赖第三方库了,可是咱们须要的是快速实现,且对原有代码尽量少的改动,引入观察者模式显然不如直接拿 EventBus来的方便。

BaseActivity.java

public class BaseActivity extends Activity {
    
        protected Context mContext;
        protected ACache mACache;
        protected boolean mCheckNetWork = true; //默认检查网络状态
        View mTipView;
        WindowManager mWindowManager;
        WindowManager.LayoutParams mLayoutParams;
    
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            mContext = this;
            this.mACache = ACache.get(mContext);
            MyApp.addActivity(this);
            initTipView();//初始化提示View
            EventBus.getDefault().register(this);
        }
    
        @Override
        protected void onResume() {
            super.onResume();
            MobclickAgent.onResume(this);
            //在无网络状况下打开APP时,系统不会发送网络情况变动的Intent,须要本身手动检查
            hasNetWork(NetUtils.isConnected(mContext));
        }
    
        @Override
        protected void onPause() {
            super.onPause();
            MobclickAgent.onPause(this);
        }
    
        @Override
        protected void onDestroy() {
            super.onDestroy();
            MyApp.removeActivity(this);
            EventBus.getDefault().unregister(this);
        }
    
        @Override
        public void finish() {
            super.finish();
            //当提示View被动态添加后直接关闭页面会致使该View内存溢出,因此须要在finish时移除
            if (mTipView != null && mTipView.getParent() != null) {
                mWindowManager.removeView(mTipView);
            }
        }
    
        @Subscribe(threadMode = ThreadMode.MAIN)
        public void onNetworkChangeEvent(NetworkChangeEvent event) {
            hasNetWork(event.isConnected);
        }
    
        private void hasNetWork(boolean has) {
            if (isCheckNetWork()) {
                if (has) {
                    if (mTipView != null && mTipView.getParent() != null) {
                        mWindowManager.removeView(mTipView);
                    }
                } else {
                    if (mTipView.getParent() == null) {
                        mWindowManager.addView(mTipView, mLayoutParams);
                    }
                }
            }
        }
    
        public void setCheckNetWork(boolean checkNetWork) {
            mCheckNetWork = checkNetWork;
        }
    
        public boolean isCheckNetWork() {
            return mCheckNetWork;
        }
    
        private void initTipView() {
            LayoutInflater inflater = getLayoutInflater();
            mTipView = inflater.inflate(R.layout.layout_network_tip, null); //提示View布局
            mWindowManager = (WindowManager) this.getSystemService(Context.WINDOW_SERVICE);
            mLayoutParams = new WindowManager.LayoutParams(
                    ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.WRAP_CONTENT,
                    WindowManager.LayoutParams.TYPE_APPLICATION,
                    WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE
                            | WindowManager.LayoutParams.FLAG_NOT_TOUCHABLE,
                    PixelFormat.TRANSLUCENT);
            //使用非CENTER时,能够经过设置XY的值来改变View的位置
            mLayoutParams.gravity = Gravity.TOP;
            mLayoutParams.x = 0;
            mLayoutParams.y = 0;
        }
    }
复制代码

默认全部继承 BaseActivity 的页面当网络情况变化活无网络时都会显示提示,若是某个页面不须要网络状态提示,能够在该页面 onCreate 方法中调用 setCheckNetWork(false) 便可。

因为我所有页面都有一个50dp高度的 toolbar,因此我直接在 R.layout.layout_network_tip 文件中设置了上边距。你也能够在 BaseActivity 中经过方法来设置 mLayoutParams.x = 0;mLayoutParams.y = 0; 来使每一个页面动态设置提示的位置。

最终效果以下图:

在这里插入图片描述

ToDo

全部页面在网络连接恢复后应该能够自动从新发起网络请求,实现原理其实也很简单,在BaseActivity中增长一个reConnect()的方法,在网络恢复去除提示View的时候调用。在各个页面中重写该方法便可。

相关文章
相关标签/搜索