| /* |
| * Copyright (C) 2017 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.googlecode.android_scripting.facade; |
| |
| import android.app.AlarmManager; |
| import android.app.NotificationManager; |
| import android.app.Service; |
| import android.content.Context; |
| import android.content.Intent; |
| import android.media.AudioManager; |
| import android.os.PowerManager; |
| import android.os.SystemClock; |
| import android.os.UserHandle; |
| import android.provider.Settings; |
| import android.provider.Settings.SettingNotFoundException; |
| import android.telephony.SubscriptionManager; |
| import android.telephony.TelephonyManager; |
| import android.view.WindowManager; |
| |
| import com.android.internal.widget.LockPatternUtils; |
| import com.android.internal.widget.LockscreenCredential; |
| |
| import com.googlecode.android_scripting.BaseApplication; |
| import com.googlecode.android_scripting.FutureActivityTaskExecutor; |
| import com.googlecode.android_scripting.Log; |
| import com.googlecode.android_scripting.future.FutureActivityTask; |
| import com.googlecode.android_scripting.jsonrpc.RpcReceiver; |
| import com.googlecode.android_scripting.rpc.Rpc; |
| import com.googlecode.android_scripting.rpc.RpcDefault; |
| import com.googlecode.android_scripting.rpc.RpcOptional; |
| import com.googlecode.android_scripting.rpc.RpcParameter; |
| |
| /** |
| * Exposes phone settings functionality. |
| * |
| */ |
| public class SettingsFacade extends RpcReceiver { |
| |
| private final Service mService; |
| private final AndroidFacade mAndroidFacade; |
| private final AudioManager mAudio; |
| private final PowerManager mPower; |
| private final AlarmManager mAlarm; |
| private final LockPatternUtils mLockPatternUtils; |
| private final NotificationManager mNotificationManager; |
| private final DataUsageController mDataController; |
| private final TelephonyManager mTelephonyManager; |
| |
| /** |
| * Creates a new SettingsFacade. |
| * |
| * @param manager is the {@link Context} the APIs will run under |
| */ |
| public SettingsFacade(FacadeManager manager) { |
| super(manager); |
| mService = manager.getService(); |
| mAndroidFacade = manager.getReceiver(AndroidFacade.class); |
| mAudio = (AudioManager) mService.getSystemService(Context.AUDIO_SERVICE); |
| mPower = (PowerManager) mService.getSystemService(Context.POWER_SERVICE); |
| mAlarm = (AlarmManager) mService.getSystemService(Context.ALARM_SERVICE); |
| mLockPatternUtils = new LockPatternUtils(mService); |
| mNotificationManager = |
| (NotificationManager) mService.getSystemService(Context.NOTIFICATION_SERVICE); |
| if (!mNotificationManager.isNotificationPolicyAccessGranted()) { |
| Intent intent = new Intent(Settings.ACTION_NOTIFICATION_POLICY_ACCESS_SETTINGS); |
| intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK); |
| mService.startActivity(intent); |
| } |
| mDataController = new DataUsageController(mService); |
| mTelephonyManager = |
| (TelephonyManager) mService.getSystemService(Context.TELEPHONY_SERVICE); |
| } |
| |
| @Rpc(description = "Sets the screen timeout to this number of seconds.", |
| returns = "The original screen timeout.") |
| public Integer setScreenTimeout(@RpcParameter(name = "value") Integer value) { |
| Integer oldValue = getScreenTimeout(); |
| android.provider.Settings.System.putInt(mService.getContentResolver(), |
| android.provider.Settings.System.SCREEN_OFF_TIMEOUT, value * 1000); |
| return oldValue; |
| } |
| |
| @Rpc(description = "Returns the current screen timeout in seconds.", |
| returns = "the current screen timeout in seconds.") |
| public Integer getScreenTimeout() { |
| try { |
| return android.provider.Settings.System.getInt(mService.getContentResolver(), |
| android.provider.Settings.System.SCREEN_OFF_TIMEOUT) / 1000; |
| } catch (SettingNotFoundException e) { |
| return 0; |
| } |
| } |
| |
| @Rpc(description = "Checks the ringer silent mode setting.", |
| returns = "True if ringer silent mode is enabled.") |
| public Boolean checkRingerSilentMode() { |
| return mAudio.getRingerMode() == AudioManager.RINGER_MODE_SILENT; |
| } |
| |
| @Rpc(description = "Toggles ringer silent mode on and off.", |
| returns = "True if ringer silent mode is enabled.") |
| public Boolean toggleRingerSilentMode( |
| @RpcParameter(name = "enabled") @RpcOptional Boolean enabled) { |
| if (enabled == null) { |
| enabled = !checkRingerSilentMode(); |
| } |
| mAudio.setRingerMode(enabled ? AudioManager.RINGER_MODE_SILENT |
| : AudioManager.RINGER_MODE_NORMAL); |
| return enabled; |
| } |
| |
| @Rpc(description = "Set the ringer to a specified mode") |
| public void setRingerMode(@RpcParameter(name = "mode") Integer mode) throws Exception { |
| if (AudioManager.isValidRingerMode(mode)) { |
| mAudio.setRingerMode(mode); |
| } else { |
| throw new Exception("Ringer mode " + mode + " does not exist."); |
| } |
| } |
| |
| @Rpc(description = "Returns the current ringtone mode.", |
| returns = "An integer representing the current ringer mode") |
| public Integer getRingerMode() { |
| return mAudio.getRingerMode(); |
| } |
| |
| @Rpc(description = "Returns the maximum ringer volume.") |
| public int getMaxRingerVolume() { |
| return mAudio.getStreamMaxVolume(AudioManager.STREAM_RING); |
| } |
| |
| @Rpc(description = "Returns the current ringer volume.") |
| public int getRingerVolume() { |
| return mAudio.getStreamVolume(AudioManager.STREAM_RING); |
| } |
| |
| @Rpc(description = "Sets the ringer volume.") |
| public void setRingerVolume(@RpcParameter(name = "volume") Integer volume) { |
| mAudio.setStreamVolume(AudioManager.STREAM_RING, volume, 0); |
| } |
| |
| @Rpc(description = "Returns the maximum media volume.") |
| public int getMaxMediaVolume() { |
| return mAudio.getStreamMaxVolume(AudioManager.STREAM_MUSIC); |
| } |
| |
| @Rpc(description = "Returns the current media volume.") |
| public int getMediaVolume() { |
| return mAudio.getStreamVolume(AudioManager.STREAM_MUSIC); |
| } |
| |
| @Rpc(description = "Sets the media volume.") |
| public void setMediaVolume(@RpcParameter(name = "volume") Integer volume) { |
| mAudio.setStreamVolume(AudioManager.STREAM_MUSIC, volume, 0); |
| } |
| |
| /** |
| * Sets the voice call volume. |
| * @param volume the volume number to be set for voice call. |
| */ |
| @Rpc(description = "Sets the voice call volume.") |
| public void setVoiceCallVolume(@RpcParameter(name = "volume") Integer volume) { |
| mAudio.setStreamVolume(AudioManager.STREAM_VOICE_CALL, volume, 0); |
| } |
| |
| /** |
| * Sets the alarm volume. |
| * @param volume the volume number to be set for alarm. |
| */ |
| @Rpc(description = "Sets the alarm volume.") |
| public void setAlarmVolume(@RpcParameter(name = "volume") Integer volume) { |
| mAudio.setStreamVolume(AudioManager.STREAM_ALARM, volume, 0); |
| } |
| |
| @Rpc(description = "Returns the screen backlight brightness.", |
| returns = "the current screen brightness between 0 and 255") |
| public Integer getScreenBrightness() { |
| try { |
| return android.provider.Settings.System.getInt(mService.getContentResolver(), |
| android.provider.Settings.System.SCREEN_BRIGHTNESS); |
| } catch (SettingNotFoundException e) { |
| return 0; |
| } |
| } |
| |
| @Rpc(description = "return the system time since boot in nanoseconds") |
| public long getSystemElapsedRealtimeNanos() { |
| return SystemClock.elapsedRealtimeNanos(); |
| } |
| |
| @Rpc(description = "Sets the the screen backlight brightness.", |
| returns = "the original screen brightness.") |
| public Integer setScreenBrightness( |
| @RpcParameter(name = "value", description = "brightness value between 0 and 255") Integer value) { |
| if (value < 0) { |
| value = 0; |
| } else if (value > 255) { |
| value = 255; |
| } |
| final int brightness = value; |
| Integer oldValue = getScreenBrightness(); |
| android.provider.Settings.System.putInt(mService.getContentResolver(), |
| android.provider.Settings.System.SCREEN_BRIGHTNESS, brightness); |
| |
| FutureActivityTask<Object> task = new FutureActivityTask<Object>() { |
| @Override |
| public void onCreate() { |
| super.onCreate(); |
| WindowManager.LayoutParams lp = getActivity().getWindow().getAttributes(); |
| lp.screenBrightness = brightness * 1.0f / 255; |
| getActivity().getWindow().setAttributes(lp); |
| setResult(null); |
| finish(); |
| } |
| }; |
| |
| FutureActivityTaskExecutor taskExecutor = |
| ((BaseApplication) mService.getApplication()).getTaskExecutor(); |
| taskExecutor.execute(task); |
| |
| return oldValue; |
| } |
| |
| @Rpc(description = "Returns true if the device is in an interactive state.") |
| public Boolean isDeviceInteractive() throws Exception { |
| return mPower.isInteractive(); |
| } |
| |
| @Rpc(description = "Issues a request to put the device to sleep after a delay.") |
| public void goToSleep(Integer delay) { |
| mPower.goToSleep(SystemClock.uptimeMillis() + delay); |
| } |
| |
| @Rpc(description = "Issues a request to put the device to sleep right away.") |
| public void goToSleepNow() { |
| mPower.goToSleep(SystemClock.uptimeMillis()); |
| } |
| |
| @Rpc(description = "Issues a request to wake the device up right away.") |
| public void wakeUpNow() { |
| mPower.wakeUp(SystemClock.uptimeMillis()); |
| } |
| |
| @Rpc(description = "Get Up time of device.", |
| returns = "Long value of device up time in milliseconds.") |
| public long getDeviceUpTime() throws Exception { |
| return SystemClock.elapsedRealtime(); |
| } |
| |
| @Rpc(description = "Set a string password to the device.") |
| public void setDevicePassword(@RpcParameter(name = "password") String password, |
| @RpcParameter(name = "previousPassword") |
| @RpcDefault("") |
| @RpcOptional |
| String previousPassword) { |
| // mLockPatternUtils.setLockPatternEnabled(true, UserHandle.myUserId()); |
| mLockPatternUtils.setLockScreenDisabled(false, UserHandle.myUserId()); |
| mLockPatternUtils.setLockCredential(LockscreenCredential.createPassword(password), |
| LockscreenCredential.createPasswordOrNone(previousPassword), |
| UserHandle.myUserId()); |
| } |
| |
| @Rpc(description = "Disable screen lock password on the device. Note that disabling the " + |
| "screen lock while the screen is locked will still require the screen to be " + |
| "unlocked via pressing the back button and swiping up. To get around this, make sure " + |
| "the screen is on/unlocked when calling this method.") |
| public void disableDevicePassword( |
| @RpcParameter(name = "currentPassword", |
| description = "The current password used to lock the device") |
| @RpcDefault("1111") |
| @RpcOptional String currentPassword) { |
| mLockPatternUtils.setLockCredential(LockscreenCredential.createNone(), |
| LockscreenCredential.createPasswordOrNone(currentPassword), |
| UserHandle.myUserId()); |
| mLockPatternUtils.setLockScreenDisabled(true, UserHandle.myUserId()); |
| } |
| |
| @Rpc(description = "Set the system time in epoch.") |
| public void setTime(Long currentTime) { |
| mAlarm.setTime(currentTime); |
| } |
| |
| @Rpc(description = "Set the system time zone.") |
| public void setTimeZone(@RpcParameter(name = "timeZone") String timeZone) { |
| mAlarm.setTimeZone(timeZone); |
| } |
| |
| @Rpc(description = "Show Home Screen") |
| public void showHomeScreen() { |
| Intent intent = new Intent(Intent.ACTION_MAIN); |
| intent.addCategory(Intent.CATEGORY_HOME); |
| intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK); |
| try { |
| mAndroidFacade.startActivityIntent(intent, false); |
| } catch (RuntimeException e) { |
| Log.d("showHomeScreen RuntimeException" + e); |
| } catch (Exception e){ |
| Log.d("showHomeScreen exception" + e); |
| } |
| } |
| |
| @Rpc(description = "Set private DNS mode") |
| public void setPrivateDnsMode(@RpcParameter(name = "useTls") Boolean useTls, |
| @RpcParameter(name = "hostname") @RpcOptional String hostname) { |
| String dnsMode = ConnectivityConstants.PrivateDnsModeOpportunistic; |
| if(useTls == false) dnsMode = ConnectivityConstants.PrivateDnsModeOff; |
| if(hostname != null) dnsMode = ConnectivityConstants.PrivateDnsModeStrict; |
| android.provider.Settings.Global.putString(mService.getContentResolver(), |
| android.provider.Settings.Global.PRIVATE_DNS_MODE, dnsMode); |
| if(hostname != null) |
| android.provider.Settings.Global.putString(mService.getContentResolver(), |
| android.provider.Settings.Global.PRIVATE_DNS_SPECIFIER, hostname); |
| } |
| |
| @Rpc(description = "Get private DNS mode", returns = "Private DNS mode setting") |
| public String getPrivateDnsMode() { |
| return android.provider.Settings.Global.getString(mService.getContentResolver(), |
| android.provider.Settings.Global.PRIVATE_DNS_MODE); |
| } |
| |
| @Rpc(description = "Get private DNS specifier", returns = "DNS hostname set in strict mode") |
| public String getPrivateDnsSpecifier() { |
| if(!getPrivateDnsMode().equals(ConnectivityConstants.PrivateDnsModeStrict)) return null; |
| return android.provider.Settings.Global.getString(mService.getContentResolver(), |
| android.provider.Settings.Global.PRIVATE_DNS_SPECIFIER); |
| } |
| |
| /** |
| * Enable or disable mobile data. |
| * @param enabled Enable data: True: Disable data: False. |
| */ |
| @Rpc(description = "Set Mobile Data Enabled.") |
| public void setMobileDataEnabled(@RpcParameter(name = "enabled") |
| @RpcOptional @RpcDefault(value = "true") Boolean enabled) { |
| mDataController.setMobileDataEnabled(enabled); |
| } |
| |
| /** |
| * Get mobile data usage info for subscription. |
| * @return DataUsageInfo: The Mobile data usage information. |
| */ |
| @Rpc(description = "Get mobile data usage info", returns = "Mobile Data DataUsageInfo") |
| public DataUsageController.DataUsageInfo getMobileDataUsageInfo( |
| @RpcOptional @RpcParameter(name = "subId") Integer subId) { |
| if (subId == null) { |
| subId = SubscriptionManager.getDefaultDataSubscriptionId(); |
| } |
| String subscriberId = mTelephonyManager.getSubscriberId(subId); |
| return mDataController.getMobileDataUsageInfoForSubscriber(subscriberId); |
| } |
| |
| /** |
| * Get mobile data usage info for for uid. |
| * @return DataUsageInfo: The Mobile data usage information. |
| */ |
| @Rpc(description = "Get mobile data usage info", returns = "Mobile Data DataUsageInfo") |
| public DataUsageController.DataUsageInfo getMobileDataUsageInfoForUid( |
| @RpcParameter(name = "uId") Integer uId, |
| @RpcOptional @RpcParameter(name = "subId") Integer subId) { |
| if (subId == null) { |
| subId = SubscriptionManager.getDefaultDataSubscriptionId(); |
| } |
| String subscriberId = mTelephonyManager.getSubscriberId(subId); |
| return mDataController.getMobileDataUsageInfoForUid(uId, subscriberId); |
| } |
| |
| /** |
| * Get Wifi data usage info. |
| * @return DataUsageInfo: The Wifi data usage information. |
| */ |
| @Rpc(description = "Get wifi data usage info", returns = "Wifi Data DataUsageInfo") |
| public DataUsageController.DataUsageInfo getWifiDataUsageInfo() { |
| return mDataController.getWifiDataUsageInfo(); |
| } |
| |
| @Override |
| public void shutdown() { |
| // Nothing to do yet. |
| } |
| } |