diff --git a/res/values/cherish_arrays.xml b/res/values/cherish_arrays.xml
index cba1bdc..cc8315c 100644
--- a/res/values/cherish_arrays.xml
+++ b/res/values/cherish_arrays.xml
@@ -1330,4 +1330,29 @@
- 5
- 6
+
+
+
+ - @string/pulse_render_mode_fading_bars
+ - @string/pulse_render_mode_solid_lines
+
+
+
+ - 0
+ - 1
+
+
+
+ - @string/pulse_color_accent
+ - @string/pulse_color_custom
+ - @string/pulse_color_lava_lamp
+ - @string/pulse_color_auto
+
+
+
+ - 0
+ - 1
+ - 2
+ - 3
+
diff --git a/res/xml/cherish_settings_lockscreen.xml b/res/xml/cherish_settings_lockscreen.xml
index f90c81e..c11b3ac 100644
--- a/res/xml/cherish_settings_lockscreen.xml
+++ b/res/xml/cherish_settings_lockscreen.xml
@@ -245,6 +245,13 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/src/com/cherish/settings/fragments/PulseSettings.java b/src/com/cherish/settings/fragments/PulseSettings.java
new file mode 100644
index 0000000..f54c646
--- /dev/null
+++ b/src/com/cherish/settings/fragments/PulseSettings.java
@@ -0,0 +1,229 @@
+/*
+ * Copyright (C) 2020 The Dirty Unicorns 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.cherish.settings.fragments;
+
+import android.app.ActionBar;
+import android.content.Context;
+import android.content.ContentResolver;
+import android.graphics.Color;
+import android.os.Bundle;
+import android.os.UserHandle;
+import android.provider.SearchIndexableResource;
+import android.provider.Settings;
+
+import androidx.preference.ListPreference;
+import androidx.preference.Preference;
+import androidx.preference.PreferenceCategory;
+import androidx.preference.PreferenceScreen;
+import androidx.preference.Preference.OnPreferenceChangeListener;
+import androidx.preference.SwitchPreference;
+
+import com.android.internal.logging.nano.MetricsProto;
+import com.android.settings.R;
+import com.android.settings.SettingsPreferenceFragment;
+import com.android.settings.search.BaseSearchIndexProvider;
+import com.android.settingslib.search.SearchIndexable;
+
+import net.margaritov.preference.colorpicker.ColorPickerPreference;
+
+import java.util.ArrayList;
+import java.util.Arrays;
+import java.util.List;
+
+@SearchIndexable
+public class PulseSettings extends SettingsPreferenceFragment implements
+ Preference.OnPreferenceChangeListener {
+ private static final String TAG = PulseSettings.class.getSimpleName();
+
+ private static final String NAVBAR_PULSE_ENABLED_KEY = "navbar_pulse_enabled";
+ private static final String LOCKSCREEN_PULSE_ENABLED_KEY = "lockscreen_pulse_enabled";
+ private static final String PULSE_SMOOTHING_KEY = "pulse_smoothing_enabled";
+ private static final String PULSE_COLOR_MODE_KEY = "pulse_color_mode";
+ private static final String PULSE_COLOR_MODE_CHOOSER_KEY = "pulse_color_user";
+ private static final String PULSE_COLOR_MODE_LAVA_SPEED_KEY = "pulse_lavalamp_speed";
+ private static final String PULSE_RENDER_CATEGORY_SOLID = "pulse_2";
+ private static final String PULSE_RENDER_CATEGORY_FADING = "pulse_fading_bars_category";
+ private static final String PULSE_RENDER_MODE_KEY = "pulse_render_style";
+ private static final int RENDER_STYLE_FADING_BARS = 0;
+ private static final int RENDER_STYLE_SOLID_LINES = 1;
+ private static final int COLOR_TYPE_ACCENT = 0;
+ private static final int COLOR_TYPE_USER = 1;
+ private static final int COLOR_TYPE_LAVALAMP = 2;
+ private static final int COLOR_TYPE_AUTO = 3;
+
+ private SwitchPreference mNavbarPulse;
+ private SwitchPreference mLockscreenPulse;
+ private SwitchPreference mPulseSmoothing;
+ private Preference mRenderMode;
+ private ListPreference mColorModePref;
+ private ColorPickerPreference mColorPickerPref;
+ private Preference mLavaSpeedPref;
+
+ private PreferenceCategory mFadingBarsCat;
+ private PreferenceCategory mSolidBarsCat;
+
+ @Override
+ public void onCreate(Bundle savedInstanceState) {
+ super.onCreate(savedInstanceState);
+ addPreferencesFromResource(R.xml.pulse_settings);
+
+ ContentResolver resolver = getContext().getContentResolver();
+
+ mNavbarPulse = (SwitchPreference) findPreference(NAVBAR_PULSE_ENABLED_KEY);
+ boolean navbarPulse = Settings.Secure.getIntForUser(resolver,
+ Settings.Secure.NAVBAR_PULSE_ENABLED, 0, UserHandle.USER_CURRENT) != 0;
+ mNavbarPulse.setChecked(navbarPulse);
+ mNavbarPulse.setOnPreferenceChangeListener(this);
+
+ mLockscreenPulse = (SwitchPreference) findPreference(LOCKSCREEN_PULSE_ENABLED_KEY);
+ boolean lockscreenPulse = Settings.Secure.getIntForUser(resolver,
+ Settings.Secure.LOCKSCREEN_PULSE_ENABLED, 0, UserHandle.USER_CURRENT) != 0;
+ mLockscreenPulse.setChecked(lockscreenPulse);
+ mLockscreenPulse.setOnPreferenceChangeListener(this);
+
+ mColorModePref = (ListPreference) findPreference(PULSE_COLOR_MODE_KEY);
+ mColorPickerPref = (ColorPickerPreference) findPreference(PULSE_COLOR_MODE_CHOOSER_KEY);
+ int userColor = Settings.Secure.getInt(getContentResolver(),
+ Settings.Secure.PULSE_COLOR_USER, 0xffffffff);
+ mColorPickerPref.setOnPreferenceChangeListener(this);
+ mLavaSpeedPref = findPreference(PULSE_COLOR_MODE_LAVA_SPEED_KEY);
+ mColorModePref.setOnPreferenceChangeListener(this);
+
+ mRenderMode = findPreference(PULSE_RENDER_MODE_KEY);
+ mRenderMode.setOnPreferenceChangeListener(this);
+
+ mFadingBarsCat = (PreferenceCategory) findPreference(
+ PULSE_RENDER_CATEGORY_FADING);
+ mSolidBarsCat = (PreferenceCategory) findPreference(
+ PULSE_RENDER_CATEGORY_SOLID);
+
+ mPulseSmoothing = (SwitchPreference) findPreference(PULSE_SMOOTHING_KEY);
+
+ updateAllPrefs();
+ }
+
+ @Override
+ public boolean onPreferenceChange(Preference preference, Object newValue) {
+ ContentResolver resolver = getContext().getContentResolver();
+ if (preference == mNavbarPulse) {
+ boolean val = (Boolean) newValue;
+ Settings.Secure.putIntForUser(resolver,
+ Settings.Secure.NAVBAR_PULSE_ENABLED, val ? 1 : 0, UserHandle.USER_CURRENT);
+ updateAllPrefs();
+ return true;
+ } else if (preference == mLockscreenPulse) {
+ boolean val = (Boolean) newValue;
+ Settings.Secure.putIntForUser(resolver,
+ Settings.Secure.LOCKSCREEN_PULSE_ENABLED, val ? 1 : 0, UserHandle.USER_CURRENT);
+ updateAllPrefs();
+ return true;
+ } else if (preference == mColorPickerPref) {
+ int value = (Integer) newValue;
+ Settings.Secure.putInt(resolver,
+ Settings.Secure.PULSE_COLOR_USER, value);
+ return true;
+ } else if (preference == mColorModePref) {
+ updateColorPrefs(Integer.valueOf(String.valueOf(newValue)));
+ return true;
+ } else if (preference == mRenderMode) {
+ updateRenderCategories(Integer.valueOf(String.valueOf(newValue)));
+ return true;
+ }
+ return false;
+ }
+
+ private void updateAllPrefs() {
+ ContentResolver resolver = getContext().getContentResolver();
+
+ boolean navbarPulse = Settings.Secure.getIntForUser(resolver,
+ Settings.Secure.NAVBAR_PULSE_ENABLED, 0, UserHandle.USER_CURRENT) != 0;
+ boolean lockscreenPulse = Settings.Secure.getIntForUser(resolver,
+ Settings.Secure.LOCKSCREEN_PULSE_ENABLED, 1, UserHandle.USER_CURRENT) != 0;
+
+ mPulseSmoothing.setEnabled(navbarPulse || lockscreenPulse);
+
+ mColorModePref.setEnabled(navbarPulse || lockscreenPulse);
+ if (navbarPulse || lockscreenPulse) {
+ int colorMode = Settings.Secure.getIntForUser(resolver,
+ Settings.Secure.PULSE_COLOR_MODE, COLOR_TYPE_LAVALAMP, UserHandle.USER_CURRENT);
+ updateColorPrefs(colorMode);
+ } else {
+ mColorPickerPref.setEnabled(false);
+ mLavaSpeedPref.setEnabled(false);
+ }
+
+ mRenderMode.setEnabled(navbarPulse || lockscreenPulse);
+ if (navbarPulse || lockscreenPulse) {
+ int renderMode = Settings.Secure.getIntForUser(resolver,
+ Settings.Secure.PULSE_RENDER_STYLE, RENDER_STYLE_SOLID_LINES, UserHandle.USER_CURRENT);
+ updateRenderCategories(renderMode);
+ } else {
+ mFadingBarsCat.setEnabled(false);
+ mSolidBarsCat.setEnabled(false);
+ }
+ }
+
+ private void updateColorPrefs(int val) {
+ switch (val) {
+ case COLOR_TYPE_ACCENT:
+ mColorPickerPref.setEnabled(false);
+ mLavaSpeedPref.setEnabled(false);
+ break;
+ case COLOR_TYPE_USER:
+ mColorPickerPref.setEnabled(true);
+ mLavaSpeedPref.setEnabled(false);
+ break;
+ case COLOR_TYPE_LAVALAMP:
+ mColorPickerPref.setEnabled(false);
+ mLavaSpeedPref.setEnabled(true);
+ break;
+ case COLOR_TYPE_AUTO:
+ mColorPickerPref.setEnabled(false);
+ mLavaSpeedPref.setEnabled(false);
+ break;
+ }
+ }
+
+ private void updateRenderCategories(int mode) {
+ mFadingBarsCat.setEnabled(mode == RENDER_STYLE_FADING_BARS);
+ mSolidBarsCat.setEnabled(mode == RENDER_STYLE_SOLID_LINES);
+ }
+
+ @Override
+ public int getMetricsCategory() {
+ return MetricsProto.MetricsEvent.CHERISH_SETTINGS;
+ }
+
+ public static final BaseSearchIndexProvider SEARCH_INDEX_DATA_PROVIDER =
+ new BaseSearchIndexProvider() {
+
+ @Override
+ public List getXmlResourcesToIndex(
+ Context context, boolean enabled) {
+ final SearchIndexableResource sir = new SearchIndexableResource(context);
+ sir.xmlResId = R.xml.pulse_settings;
+ return Arrays.asList(sir);
+ }
+
+ @Override
+ public List getNonIndexableKeys(Context context) {
+ ArrayList result = new ArrayList();
+ return result;
+ }
+ };
+
+}