From a9d39862a6c3996eb0df687d7d567e963db69d96 Mon Sep 17 00:00:00 2001 From: SagarMakhar Date: Sun, 3 Apr 2022 15:14:02 +0300 Subject: [PATCH] Cherish: Bring back UDFPS icons and Animations [2/3] MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Signed-off-by: Hưng Phan --- res/values/cherish_strings.xml | 13 +- res/xml/cherish_settings_udfps.xml | 18 ++ src/com/cherish/settings/fragments/Udfps.java | 8 + .../settings/fragments/ui/UdfpsAnimation.java | 257 ++++++++++++++++++ .../fragments/ui/UdfpsIconPicker.java | 210 ++++++++++++++ 5 files changed, 505 insertions(+), 1 deletion(-) create mode 100644 src/com/cherish/settings/fragments/ui/UdfpsAnimation.java create mode 100644 src/com/cherish/settings/fragments/ui/UdfpsIconPicker.java diff --git a/res/values/cherish_strings.xml b/res/values/cherish_strings.xml index 676ea14..d48af2d 100644 --- a/res/values/cherish_strings.xml +++ b/res/values/cherish_strings.xml @@ -617,7 +617,18 @@ UDFPS haptic feedback Vibrate when touching UDFPS icon + + + UDFPS customization + UDFPS animation + Shows an animation while reading your finger + UDFPS custom animation + Choose animation effect UDFPS settings - Customizations for the UDFPS + Settings for customizing the UDFPS + + + UDFPS icon picker + Choose your favorite fingerprint icon diff --git a/res/xml/cherish_settings_udfps.xml b/res/xml/cherish_settings_udfps.xml index f44a253..8b3c296 100644 --- a/res/xml/cherish_settings_udfps.xml +++ b/res/xml/cherish_settings_udfps.xml @@ -14,4 +14,22 @@ android:summary="@string/udfps_haptic_feedback_summary" android:defaultValue="true" /> + + + + + + + + diff --git a/src/com/cherish/settings/fragments/Udfps.java b/src/com/cherish/settings/fragments/Udfps.java index ee373b4..80c2759 100644 --- a/src/com/cherish/settings/fragments/Udfps.java +++ b/src/com/cherish/settings/fragments/Udfps.java @@ -48,6 +48,7 @@ import com.android.settingslib.search.SearchIndexable; public class Udfps extends SettingsPreferenceFragment implements Preference.OnPreferenceChangeListener { + private static final String UDFPS_CUSTOMIZATION = "udfps_customization"; @Override public void onCreate(Bundle icicle) { @@ -57,6 +58,13 @@ public class Udfps extends SettingsPreferenceFragment implements final PreferenceScreen prefSet = getPreferenceScreen(); Resources resources = getResources(); + final boolean udfpsResPkgInstalled = CherishUtils.isPackageInstalled(getContext(), + "com.cherish.udfps.resources"); + mUdfpsCustomization = (PreferenceCategory) findPreference(UDFPS_CUSTOMIZATION); + if (!udfpsResPkgInstalled) { + prefSet.removePreference(mUdfpsCustomization); + } + } @Override diff --git a/src/com/cherish/settings/fragments/ui/UdfpsAnimation.java b/src/com/cherish/settings/fragments/ui/UdfpsAnimation.java new file mode 100644 index 0000000..dfc1e5e --- /dev/null +++ b/src/com/cherish/settings/fragments/ui/UdfpsAnimation.java @@ -0,0 +1,257 @@ +/* + * Copyright (C) 2021 AospExtended ROM 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.content.ContentResolver; +import android.content.Context; +import android.content.pm.PackageManager; +import android.content.res.Resources; +import android.graphics.drawable.AnimationDrawable; +import android.graphics.drawable.Drawable; +import android.net.Uri; +import android.os.Bundle; +import android.provider.SearchIndexableResource; +import android.provider.Settings; +import android.text.TextUtils; +import android.view.Gravity; +import android.view.LayoutInflater; +import android.view.View; +import android.view.ViewGroup.LayoutParams; +import android.view.ViewGroup; +import android.widget.FrameLayout; +import android.widget.ImageView; +import android.widget.LinearLayout; +import android.widget.Switch; +import android.widget.TextView; +import android.widget.Toast; + +import androidx.annotation.NonNull; +import androidx.annotation.Nullable; +import androidx.core.content.res.ResourcesCompat; +import androidx.preference.Preference.OnPreferenceChangeListener; +import androidx.preference.Preference; +import androidx.preference.PreferenceScreen; +import androidx.preference.PreferenceViewHolder; +import androidx.recyclerview.widget.GridLayoutManager; +import androidx.recyclerview.widget.RecyclerView.ViewHolder; +import androidx.recyclerview.widget.RecyclerView; + +import com.bumptech.glide.Glide; + +import com.android.internal.logging.nano.MetricsProto.MetricsEvent; +import com.android.settings.R; +import com.android.settings.SettingsActivity; +import com.android.settings.SettingsPreferenceFragment; +import com.android.settings.search.BaseSearchIndexProvider; +import com.android.settings.widget.SettingsMainSwitchBar; +import com.android.settingslib.search.Indexable; +import com.android.settingslib.widget.OnMainSwitchChangeListener; + +import java.util.ArrayList; +import java.util.Arrays; +import java.util.List; + +import org.json.JSONException; +import org.json.JSONObject; + +public class UdfpsAnimation extends SettingsPreferenceFragment implements + OnMainSwitchChangeListener { + + private Switch mSwitch; + + private RecyclerView mRecyclerView; + private String mPkg = "com.cherish.udfps.resources"; + private AnimationDrawable animation; + + private Resources udfpsRes; + + private String[] mAnims; + private String[] mAnimPreviews; + private String[] mTitles; + + private boolean mEnabled; + private UdfpsAnimAdapter mUdfpsAnimAdapter; + + @Override + public void onCreate(Bundle savedInstanceState) { + super.onCreate(savedInstanceState); + getActivity().setTitle(R.string.udfps_recog_animation_effect_title); + + loadResources(); + } + + private void loadResources() { + try { + PackageManager pm = getActivity().getPackageManager(); + udfpsRes = pm.getResourcesForApplication(mPkg); + } catch (PackageManager.NameNotFoundException e) { + e.printStackTrace(); + } + + mAnims = udfpsRes.getStringArray(udfpsRes.getIdentifier("udfps_animation_styles", + "array", mPkg)); + mAnimPreviews = udfpsRes.getStringArray(udfpsRes.getIdentifier("udfps_animation_previews", + "array", mPkg)); + mTitles = udfpsRes.getStringArray(udfpsRes.getIdentifier("udfps_animation_titles", + "array", mPkg)); + } + + @Override + public View onCreateView(@NonNull LayoutInflater inflater, @Nullable ViewGroup container, + @Nullable Bundle savedInstanceState) { + View view = inflater.inflate( + R.layout.item_view, container, false); + + mRecyclerView = (RecyclerView) view.findViewById(R.id.recycler_view); + GridLayoutManager gridLayoutManager = new GridLayoutManager(getActivity(), 3); + mRecyclerView.setLayoutManager(gridLayoutManager); + mUdfpsAnimAdapter = new UdfpsAnimAdapter(getActivity()); + + return view; + } + + @Override + public void onActivityCreated(Bundle savedInstanceState) { + super.onActivityCreated(savedInstanceState); + final SettingsActivity activity = (SettingsActivity) getActivity(); + final SettingsMainSwitchBar switchBar = activity.getSwitchBar(); + mSwitch = switchBar.getSwitch(); + mEnabled = Settings.System.getInt(getActivity().getContentResolver(), + Settings.System.UDFPS_ANIM, 0) == 1; + mSwitch.setChecked(mEnabled); + setEnabled(mEnabled); + switchBar.setTitle(getActivity().getString(R.string.udfps_recog_animation)); + switchBar.addOnSwitchChangeListener(this); + switchBar.show(); + } + + @Override + public void onSwitchChanged(Switch switchView, boolean isChecked) { + Settings.System.putInt(getActivity().getContentResolver(), + Settings.System.UDFPS_ANIM, isChecked ? 1 : 0); + mSwitch.setChecked(isChecked); + setEnabled(isChecked); + } + + public void setEnabled(boolean enabled) { + if (enabled) { + mRecyclerView.setAdapter(mUdfpsAnimAdapter); + } else { + mRecyclerView.setAdapter(null); + } + } + + @Override + public int getMetricsCategory() { + return MetricsEvent.CHERISH_SETTINGS; + } + + @Override + public void onResume() { + super.onResume(); + } + + public class UdfpsAnimAdapter extends RecyclerView.Adapter { + Context context; + String mSelectedAnim; + String mAppliedAnim; + + public UdfpsAnimAdapter(Context context) { + this.context = context; + } + + @Override + public UdfpsAnimViewHolder onCreateViewHolder(ViewGroup parent, int viewType) { + View v = LayoutInflater.from(parent.getContext()).inflate(R.layout.item_option, parent, false); + UdfpsAnimViewHolder vh = new UdfpsAnimViewHolder(v); + return vh; + } + + @Override + public void onBindViewHolder(UdfpsAnimViewHolder holder, final int position) { + String animName = mAnims[position]; + + Glide.with(holder.image.getContext()) + .load("") + .placeholder(getDrawable(holder.image.getContext(), mAnimPreviews[position])) + .into(holder.image); + + holder.name.setText(mTitles[position]); + + if (position == Settings.System.getInt(context.getContentResolver(), + Settings.System.UDFPS_ANIM_STYLE, 0)) { + mAppliedAnim = animName; + if (mSelectedAnim == null) { + mSelectedAnim = animName; + } + } + + holder.itemView.setActivated(animName == mSelectedAnim); + holder.itemView.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View v) { + updateActivatedStatus(mSelectedAnim, false); + updateActivatedStatus(animName, true); + mSelectedAnim = animName; + holder.image.setBackgroundDrawable(getDrawable(v.getContext(), mAnims[position])); + animation = (AnimationDrawable) holder.image.getBackground(); + animation.setOneShot(true); + animation.start(); + Settings.System.putInt(getActivity().getContentResolver(), + Settings.System.UDFPS_ANIM_STYLE, position); + } + }); + } + + @Override + public int getItemCount() { + return mAnims.length; + } + + public class UdfpsAnimViewHolder extends RecyclerView.ViewHolder { + TextView name; + ImageView image; + public UdfpsAnimViewHolder(View itemView) { + super(itemView); + name = (TextView) itemView.findViewById(R.id.option_label); + image = (ImageView) itemView.findViewById(R.id.option_thumbnail); + } + } + + private void updateActivatedStatus(String anim, boolean isActivated) { + int index = Arrays.asList(mAnims).indexOf(anim); + if (index < 0) { + return; + } + RecyclerView.ViewHolder holder = mRecyclerView.findViewHolderForAdapterPosition(index); + if (holder != null && holder.itemView != null) { + holder.itemView.setActivated(isActivated); + } + } + } + + public Drawable getDrawable(Context context, String drawableName) { + try { + PackageManager pm = context.getPackageManager(); + Resources res = pm.getResourcesForApplication(mPkg); + return res.getDrawable(res.getIdentifier(drawableName, "drawable", mPkg)); + } + catch (PackageManager.NameNotFoundException e) { + e.printStackTrace(); + } + return null; + } +} diff --git a/src/com/cherish/settings/fragments/ui/UdfpsIconPicker.java b/src/com/cherish/settings/fragments/ui/UdfpsIconPicker.java new file mode 100644 index 0000000..07afb22 --- /dev/null +++ b/src/com/cherish/settings/fragments/ui/UdfpsIconPicker.java @@ -0,0 +1,210 @@ +/* + * Copyright (C) 2021 AospExtended ROM 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.content.ContentResolver; +import android.content.Context; +import android.content.pm.PackageManager; +import android.content.res.Resources; +import android.graphics.drawable.AnimationDrawable; +import android.graphics.drawable.Drawable; +import android.os.Bundle; +import android.net.Uri; +import android.provider.SearchIndexableResource; +import android.provider.Settings; +import android.text.TextUtils; +import android.view.Gravity; +import android.view.LayoutInflater; +import android.view.View; +import android.view.ViewGroup.LayoutParams; +import android.view.ViewGroup; +import android.widget.FrameLayout; +import android.widget.ImageView; +import android.widget.LinearLayout; +import android.widget.TextView; +import android.widget.Toast; + +import androidx.annotation.NonNull; +import androidx.annotation.Nullable; +import androidx.core.content.res.ResourcesCompat; +import androidx.preference.Preference.OnPreferenceChangeListener; +import androidx.preference.Preference; +import androidx.preference.PreferenceScreen; +import androidx.preference.PreferenceViewHolder; +import androidx.recyclerview.widget.GridLayoutManager; +import androidx.recyclerview.widget.RecyclerView.ViewHolder; +import androidx.recyclerview.widget.RecyclerView; + +import com.bumptech.glide.Glide; + +import com.android.internal.logging.nano.MetricsProto.MetricsEvent; +import com.android.settings.R; +import com.android.settings.SettingsPreferenceFragment; +import com.android.settings.search.BaseSearchIndexProvider; +import com.android.settingslib.search.Indexable; + +import java.util.ArrayList; +import java.util.Arrays; +import java.util.List; + +import org.json.JSONException; +import org.json.JSONObject; + +public class UdfpsIconPicker extends SettingsPreferenceFragment { + + private RecyclerView mRecyclerView; + + private Resources udfpsRes; + + private String mPkg = "com.cherish.udfps.resources"; + + private String[] mIcons; + + @Override + public void onCreate(Bundle savedInstanceState) { + super.onCreate(savedInstanceState); + getActivity().setTitle(R.string.udfps_icon_picker_title); + + loadResources(); + } + + private void loadResources() { + try { + PackageManager pm = getActivity().getPackageManager(); + udfpsRes = pm.getResourcesForApplication(mPkg); + } catch (PackageManager.NameNotFoundException e) { + e.printStackTrace(); + } + + mIcons = udfpsRes.getStringArray(udfpsRes.getIdentifier("udfps_icons", + "array", mPkg)); + } + + @Override + public View onCreateView(@NonNull LayoutInflater inflater, @Nullable ViewGroup container, + @Nullable Bundle savedInstanceState) { + View view = inflater.inflate( + R.layout.item_view, container, false); + + mRecyclerView = (RecyclerView) view.findViewById(R.id.recycler_view); + GridLayoutManager gridLayoutManager = new GridLayoutManager(getActivity(), 3); + mRecyclerView.setLayoutManager(gridLayoutManager); + UdfpsIconAdapter mUdfpsIconAdapter = new UdfpsIconAdapter(getActivity()); + mRecyclerView.setAdapter(mUdfpsIconAdapter); + + return view; + } + + @Override + public int getMetricsCategory() { + return MetricsEvent.CHERISH_SETTINGS; + } + + @Override + public void onResume() { + super.onResume(); + } + + public class UdfpsIconAdapter extends RecyclerView.Adapter { + Context context; + String mSelectedIcon; + String mAppliedIcon; + + public UdfpsIconAdapter(Context context) { + this.context = context; + } + + @Override + public UdfpsIconViewHolder onCreateViewHolder(ViewGroup parent, int viewType) { + View v = LayoutInflater.from(parent.getContext()).inflate(R.layout.item_option, parent, false); + UdfpsIconViewHolder vh = new UdfpsIconViewHolder(v); + return vh; + } + + @Override + public void onBindViewHolder(UdfpsIconViewHolder holder, final int position) { + String iconRes = mIcons[position]; + + Glide.with(holder.image.getContext()) + .load("") + .placeholder(getDrawable(holder.image.getContext(), mIcons[position])) + .into(holder.image); + + holder.image.setPadding(20,20,20,20); + + holder.name.setVisibility(View.GONE); + + if (position == Settings.System.getInt(context.getContentResolver(), + Settings.System.UDFPS_ICON, 0)) { + mAppliedIcon = iconRes; + if (mSelectedIcon == null) { + mSelectedIcon = iconRes; + } + } + holder.itemView.setActivated(iconRes == mSelectedIcon); + holder.itemView.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View v) { + updateActivatedStatus(mSelectedIcon, false); + updateActivatedStatus(iconRes, true); + mSelectedIcon = iconRes; + Settings.System.putInt(getActivity().getContentResolver(), + Settings.System.UDFPS_ICON, position); + } + }); + } + + @Override + public int getItemCount() { + return mIcons.length; + } + + public class UdfpsIconViewHolder extends RecyclerView.ViewHolder { + TextView name; + ImageView image; + public UdfpsIconViewHolder(View itemView) { + super(itemView); + name = (TextView) itemView.findViewById(R.id.option_label); + image = (ImageView) itemView.findViewById(R.id.option_thumbnail); + } + } + + private void updateActivatedStatus(String icon, boolean isActivated) { + int index = Arrays.asList(mIcons).indexOf(icon); + if (index < 0) { + return; + } + RecyclerView.ViewHolder holder = mRecyclerView.findViewHolderForAdapterPosition(index); + if (holder != null && holder.itemView != null) { + holder.itemView.setActivated(isActivated); + } + } + } + + public Drawable getDrawable(Context context, String drawableName) { + try { + PackageManager pm = context.getPackageManager(); + Resources res = pm.getResourcesForApplication(mPkg); + Context ctx = context.createPackageContext( + mPkg, Context.CONTEXT_IGNORE_SECURITY); + return ctx.getDrawable(res.getIdentifier(drawableName, "drawable", mPkg)); + } + catch (PackageManager.NameNotFoundException e) { + e.printStackTrace(); + } + return null; + } +}