/* * Copyright (C) 2008 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.systemui.statusbar.policy; import android.app.ActivityManager; import android.app.AppOpsManager; import android.app.StatusBarManager; import android.content.BroadcastReceiver; import android.content.ContentResolver; import android.content.Context; import android.content.Intent; import android.content.IntentFilter; import android.database.ContentObserver; import android.location.LocationManager; import android.os.Handler; import android.os.UserHandle; import android.os.UserManager; import android.provider.Settings; import com.android.systemui.R; import java.util.ArrayList; import java.util.List; /** * A controller to manage changes of location related states and update the views accordingly. */ public class LocationController extends BroadcastReceiver { // The name of the placeholder corresponding to the location request status icon. // This string corresponds to config_statusBarIcons in core/res/res/values/config.xml. public static final String LOCATION_STATUS_ICON_PLACEHOLDER = "location"; public static final int LOCATION_STATUS_ICON_ID = R.drawable.stat_sys_device_access_location_found; private static final int[] mHighPowerRequestAppOpArray = new int[] {AppOpsManager.OP_MONITOR_HIGH_POWER_LOCATION}; private Context mContext; private AppOpsManager mAppOpsManager; private StatusBarManager mStatusBarManager; private boolean mAreActiveLocationRequests; private ArrayList mSettingsChangeCallbacks = new ArrayList(); /** * A callback for change in location settings (the user has enabled/disabled location). */ public interface LocationSettingsChangeCallback { /** * Called whenever location settings change. * * @param locationEnabled A value of true indicates that at least one type of location * is enabled in settings. */ public void onLocationSettingsChanged(boolean locationEnabled); } public LocationController(Context context) { mContext = context; IntentFilter filter = new IntentFilter(); filter.addAction(LocationManager.HIGH_POWER_REQUEST_CHANGE_ACTION); context.registerReceiver(this, filter); mAppOpsManager = (AppOpsManager) context.getSystemService(Context.APP_OPS_SERVICE); mStatusBarManager = (StatusBarManager) context.getSystemService(Context.STATUS_BAR_SERVICE); // Register to listen for changes in location settings. IntentFilter intentFilter = new IntentFilter(); intentFilter.addAction(LocationManager.MODE_CHANGED_ACTION); context.registerReceiverAsUser(new BroadcastReceiver() { @Override public void onReceive(Context context, Intent intent) { String action = intent.getAction(); if (LocationManager.MODE_CHANGED_ACTION.equals(action)) { locationSettingsChanged(); } } }, UserHandle.ALL, intentFilter, null, new Handler()); // Examine the current location state and initialize the status view. updateActiveLocationRequests(); refreshViews(); } /** * Add a callback to listen for changes in location settings. */ public void addSettingsChangedCallback(LocationSettingsChangeCallback cb) { mSettingsChangeCallbacks.add(cb); } /** * Enable or disable location in settings. * *

This will attempt to enable/disable every type of location setting * (e.g. high and balanced power). * *

If enabling, a user consent dialog will pop up prompting the user to accept. * If the user doesn't accept, network location won't be enabled. * * @return true if attempt to change setting was successful. */ public boolean setLocationEnabled(boolean enabled) { int currentUserId = ActivityManager.getCurrentUser(); if (isUserLocationRestricted(currentUserId)) { return false; } final ContentResolver cr = mContext.getContentResolver(); // When enabling location, a user consent dialog will pop up, and the // setting won't be fully enabled until the user accepts the agreement. int mode = enabled ? Settings.Secure.LOCATION_MODE_HIGH_ACCURACY : Settings.Secure.LOCATION_MODE_OFF; // QuickSettings always runs as the owner, so specifically set the settings // for the current foreground user. return Settings.Secure .putIntForUser(cr, Settings.Secure.LOCATION_MODE, mode, currentUserId); } /** * Returns true if location isn't disabled in settings. */ public boolean isLocationEnabled() { ContentResolver resolver = mContext.getContentResolver(); // QuickSettings always runs as the owner, so specifically retrieve the settings // for the current foreground user. int mode = Settings.Secure.getIntForUser(resolver, Settings.Secure.LOCATION_MODE, Settings.Secure.LOCATION_MODE_OFF, ActivityManager.getCurrentUser()); return mode != Settings.Secure.LOCATION_MODE_OFF; } /** * Returns true if the current user is restricted from using location. */ private boolean isUserLocationRestricted(int userId) { final UserManager um = (UserManager) mContext.getSystemService(Context.USER_SERVICE); return um.hasUserRestriction( UserManager.DISALLOW_SHARE_LOCATION, new UserHandle(userId)); } /** * Returns true if there currently exist active high power location requests. */ private boolean areActiveHighPowerLocationRequests() { List packages = mAppOpsManager.getPackagesForOps(mHighPowerRequestAppOpArray); // AppOpsManager can return null when there is no requested data. if (packages != null) { final int numPackages = packages.size(); for (int packageInd = 0; packageInd < numPackages; packageInd++) { AppOpsManager.PackageOps packageOp = packages.get(packageInd); List opEntries = packageOp.getOps(); if (opEntries != null) { final int numOps = opEntries.size(); for (int opInd = 0; opInd < numOps; opInd++) { AppOpsManager.OpEntry opEntry = opEntries.get(opInd); // AppOpsManager should only return OP_MONITOR_HIGH_POWER_LOCATION because // of the mHighPowerRequestAppOpArray filter, but checking defensively. if (opEntry.getOp() == AppOpsManager.OP_MONITOR_HIGH_POWER_LOCATION) { if (opEntry.isRunning()) { return true; } } } } } } return false; } // Updates the status view based on the current state of location requests. private void refreshViews() { if (mAreActiveLocationRequests) { mStatusBarManager.setIcon(LOCATION_STATUS_ICON_PLACEHOLDER, LOCATION_STATUS_ICON_ID, 0, mContext.getString(R.string.accessibility_location_active)); } else { mStatusBarManager.removeIcon(LOCATION_STATUS_ICON_PLACEHOLDER); } } // Reads the active location requests and updates the status view if necessary. private void updateActiveLocationRequests() { boolean hadActiveLocationRequests = mAreActiveLocationRequests; mAreActiveLocationRequests = areActiveHighPowerLocationRequests(); if (mAreActiveLocationRequests != hadActiveLocationRequests) { refreshViews(); } } private void locationSettingsChanged() { boolean isEnabled = isLocationEnabled(); for (LocationSettingsChangeCallback cb : mSettingsChangeCallbacks) { cb.onLocationSettingsChanged(isEnabled); } } @Override public void onReceive(Context context, Intent intent) { final String action = intent.getAction(); if (LocationManager.HIGH_POWER_REQUEST_CHANGE_ACTION.equals(action)) { updateActiveLocationRequests(); } } }