/* * Copyright (C) 2007 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package com.android.settings; import android.app.settings.SettingsEnums; import android.content.Context; import android.content.Intent; import android.os.Looper; import android.os.UserHandle; import android.provider.Settings; import android.telephony.PhoneStateListener; import android.telephony.SubscriptionInfo; import android.telephony.TelephonyManager; import android.util.Log; import androidx.annotation.VisibleForTesting; import com.android.internal.telephony.flags.Flags; import com.android.settings.network.GlobalSettingsChangeListener; import com.android.settings.network.ProxySubscriptionManager; import com.android.settings.overlay.FeatureFactory; import com.android.settingslib.WirelessUtils; import com.android.settingslib.core.instrumentation.MetricsFeatureProvider; import java.util.List; /** * Monitor and update configuration of airplane mode settings */ public class AirplaneModeEnabler extends GlobalSettingsChangeListener { private static final String LOG_TAG = "AirplaneModeEnabler"; private static final boolean DEBUG = false; private final Context mContext; private final MetricsFeatureProvider mMetricsFeatureProvider; private OnAirplaneModeChangedListener mOnAirplaneModeChangedListener; public interface OnAirplaneModeChangedListener { /** * Called when airplane mode status is changed. * * @param isAirplaneModeOn the airplane mode is on */ void onAirplaneModeChanged(boolean isAirplaneModeOn); } private TelephonyManager mTelephonyManager; @VisibleForTesting PhoneStateListener mPhoneStateListener; public AirplaneModeEnabler(Context context, OnAirplaneModeChangedListener listener) { super(context, Settings.Global.AIRPLANE_MODE_ON); mContext = context; mMetricsFeatureProvider = FeatureFactory.getFeatureFactory().getMetricsFeatureProvider(); mOnAirplaneModeChangedListener = listener; mTelephonyManager = context.getSystemService(TelephonyManager.class); mPhoneStateListener = new PhoneStateListener(Looper.getMainLooper()) { @Override public void onRadioPowerStateChanged(int state) { if (DEBUG) { Log.d(LOG_TAG, "RadioPower: " + state); } onAirplaneModeChanged(); } }; } /** * Implementation of GlobalSettingsChangeListener.onChanged */ @Override public void onChanged(String field) { if (DEBUG) { Log.d(LOG_TAG, "Airplane mode configuration update"); } onAirplaneModeChanged(); } /** * Start listening to the phone state change */ public void start() { mTelephonyManager.listen(mPhoneStateListener, PhoneStateListener.LISTEN_RADIO_POWER_STATE_CHANGED); } /** * Stop listening to the phone state change */ public void stop() { mTelephonyManager.listen(mPhoneStateListener, PhoneStateListener.LISTEN_NONE); } private void setAirplaneModeOn(boolean enabling) { // Change the system setting Settings.Global.putInt(mContext.getContentResolver(), Settings.Global.AIRPLANE_MODE_ON, enabling ? 1 : 0); // Notify listener the system setting is changed. if (mOnAirplaneModeChangedListener != null) { mOnAirplaneModeChangedListener.onAirplaneModeChanged(enabling); } // Post the intent final Intent intent = new Intent(Intent.ACTION_AIRPLANE_MODE_CHANGED); intent.putExtra("state", enabling); mContext.sendBroadcastAsUser(intent, UserHandle.ALL); } /** * Called when we've received confirmation that the airplane mode was set. * TODO: We update the checkbox summary when we get notified * that mobile radio is powered up/down. We should not have dependency * on one radio alone. We need to do the following: * - handle the case of wifi/bluetooth failures * - mobile does not send failure notification, fail on timeout. */ private void onAirplaneModeChanged() { if (mOnAirplaneModeChangedListener != null) { mOnAirplaneModeChangedListener.onAirplaneModeChanged(isAirplaneModeOn()); } } /** * Check the status of ECM mode * * @return any subscription within device is under ECM mode */ public boolean isInEcmMode() { if (Flags.enforceTelephonyFeatureMappingForPublicApis()) { try { if (mTelephonyManager.getEmergencyCallbackMode()) { return true; } } catch (UnsupportedOperationException e) { // Device doesn't support FEATURE_TELEPHONY_CALLING // Ignore exception, device is not in ECM mode. } } else { if (mTelephonyManager.getEmergencyCallbackMode()) { return true; } } final List subInfoList = ProxySubscriptionManager.getInstance(mContext).getActiveSubscriptionsInfo(); if (subInfoList == null) { return false; } for (SubscriptionInfo subInfo : subInfoList) { final TelephonyManager telephonyManager = mTelephonyManager.createForSubscriptionId(subInfo.getSubscriptionId()); if (telephonyManager != null) { if (!Flags.enforceTelephonyFeatureMappingForPublicApis()) { if (telephonyManager.getEmergencyCallbackMode()) { return true; } } else { try { if (telephonyManager.getEmergencyCallbackMode()) { return true; } } catch (UnsupportedOperationException e) { // Ignore exception, device is not in ECM mode. } } } } return false; } public void setAirplaneMode(boolean isAirplaneModeOn) { if (isInEcmMode()) { // In ECM mode, do not update database at this point Log.d(LOG_TAG, "ECM airplane mode=" + isAirplaneModeOn); } else { mMetricsFeatureProvider.action(mContext, SettingsEnums.ACTION_AIRPLANE_TOGGLE, isAirplaneModeOn); setAirplaneModeOn(isAirplaneModeOn); } } public void setAirplaneModeInECM(boolean isECMExit, boolean isAirplaneModeOn) { Log.d(LOG_TAG, "Exist ECM=" + isECMExit + ", with airplane mode=" + isAirplaneModeOn); if (isECMExit) { // update database based on the current checkbox state setAirplaneModeOn(isAirplaneModeOn); } else { // update summary onAirplaneModeChanged(); } } public boolean isAirplaneModeOn() { return WirelessUtils.isAirplaneModeOn(mContext); } }