From 66d6998cb74020b93fcdf1c9a80b73169665f944 Mon Sep 17 00:00:00 2001 From: marunjar Date: Tue, 21 Nov 2023 23:38:07 +0100 Subject: [PATCH 1/2] fix_flickering_shortcuts - cache shortcut icon - cache app icon --- .../java/fr/neamar/kiss/result/Result.java | 4 +- .../neamar/kiss/result/ShortcutsResult.java | 164 ++++++++++-------- 2 files changed, 92 insertions(+), 76 deletions(-) diff --git a/app/src/main/java/fr/neamar/kiss/result/Result.java b/app/src/main/java/fr/neamar/kiss/result/Result.java index 4cfdbd8fe..6296ae71d 100644 --- a/app/src/main/java/fr/neamar/kiss/result/Result.java +++ b/app/src/main/java/fr/neamar/kiss/result/Result.java @@ -355,7 +355,7 @@ void setAsyncDrawable(ImageView view) { setAsyncDrawable(view, android.R.color.transparent); } - void setAsyncDrawable(ImageView view, @DrawableRes int resId) { + void setAsyncDrawable(ImageView view, @DrawableRes int resId) { // getting this called multiple times in parallel may result in empty icons synchronized (this) { // the ImageView tag will store the async task if it's running @@ -371,7 +371,7 @@ void setAsyncDrawable(ImageView view, @DrawableRes int resId) { } // the ImageView will store the Result after the AsyncTask finished else if (this.equals(view.getTag())) { - ((Result) view.getTag()).setDrawableCache(view.getDrawable()); + ((Result) view.getTag()).setDrawableCache(view.getDrawable()); return; } if (isDrawableCached()) { diff --git a/app/src/main/java/fr/neamar/kiss/result/ShortcutsResult.java b/app/src/main/java/fr/neamar/kiss/result/ShortcutsResult.java index 600549791..bcaf91e1e 100644 --- a/app/src/main/java/fr/neamar/kiss/result/ShortcutsResult.java +++ b/app/src/main/java/fr/neamar/kiss/result/ShortcutsResult.java @@ -47,6 +47,9 @@ public class ShortcutsResult extends Result { private static final String TAG = ShortcutsResult.class.getSimpleName(); + private volatile Drawable icon = null; + private volatile Drawable appDrawable = null; + private Utilities.AsyncRun mLoadIconTask = null; ShortcutsResult(@NonNull ShortcutPojo pojo) { @@ -81,47 +84,40 @@ public View display(final Context context, View view, @NonNull ViewGroup parent, final ImageView appIcon = view.findViewById(R.id.item_app_icon); if (!prefs.getBoolean("icons-hide", false)) { + // set shortcut icon + this.setAsyncDrawable(shortcutIcon); + + // set app icon if (mLoadIconTask != null) { mLoadIconTask.cancel(); + mLoadIconTask = null; } boolean subIconVisible = prefs.getBoolean("subicon-visible", true); AtomicReference appDrawable = new AtomicReference<>(null); - AtomicReference shortcutDrawable = new AtomicReference<>(null); // Prepare - shortcutIcon.setImageResource(android.R.color.transparent); if (subIconVisible) { appIcon.setVisibility(View.VISIBLE); - appIcon.setImageResource(android.R.color.transparent); - } else { - appIcon.setVisibility(View.GONE); - } - - mLoadIconTask = Utilities.runAsync((task) -> { - if (task == mLoadIconTask) { - // Retrieve icon for this shortcut - appDrawable.set(getAppDrawable(context)); - shortcutDrawable.set(getDrawable(context)); - } - }, (task) -> { - if (!task.isCancelled() && task == mLoadIconTask) { - // set icons - if (shortcutDrawable.get() != null) { - shortcutIcon.setImageDrawable(shortcutDrawable.get()); - if (subIconVisible) { - appIcon.setImageDrawable(appDrawable.get()); + if (this.appDrawable != null) { + appIcon.setImageDrawable(getAppDrawable(context)); + } else { + appIcon.setImageResource(android.R.color.transparent); + mLoadIconTask = Utilities.runAsync((task) -> { + if (task == mLoadIconTask) { + // Retrieve icon for this shortcut + appDrawable.set(getAppDrawable(context)); } - } else { - // No icon for this shortcut, use app icon - shortcutIcon.setImageDrawable(appDrawable.get()); - if (subIconVisible) { - appIcon.setImageResource(android.R.drawable.ic_menu_send); + }, (task) -> { + if (!task.isCancelled() && task == mLoadIconTask) { + appIcon.setImageDrawable(appDrawable.get()); } - } + }); } - }); + } else { + appIcon.setVisibility(View.GONE); + } } else { appIcon.setImageDrawable(null); shortcutIcon.setImageDrawable(null); @@ -131,67 +127,87 @@ public View display(final Context context, View view, @NonNull ViewGroup parent, } private Drawable getAppDrawable(Context context) { - Drawable appDrawable = null; - IconsHandler iconsHandler = KissApplication.getApplication(context).getIconsHandler(); + if (appDrawable == null) { + synchronized (this) { + if (appDrawable == null) { + IconsHandler iconsHandler = KissApplication.getApplication(context).getIconsHandler(); - if (pojo.isOreoShortcut()) { - // Retrieve activity icon from oreo shortcut - appDrawable = getDrawableFromOreoShortcut(context); - } + if (pojo.isOreoShortcut()) { + // Retrieve activity icon from oreo shortcut + appDrawable = getDrawableFromOreoShortcut(context); + } - if (appDrawable == null) { - // Retrieve activity icon by intent URI - try { - Intent intent = Intent.parseUri(pojo.intentUri, 0); - ComponentName componentName = PackageManagerUtils.getComponentName(context, intent); - if (componentName != null) { - appDrawable = iconsHandler.getDrawableIconForPackage(PackageManagerUtils.getLaunchingComponent(context, componentName), new fr.neamar.kiss.utils.UserHandle()); - } - } catch (NullPointerException e) { - Log.e(TAG, "Unable to get activity icon for '" + pojo.getName() + "'", e); - } catch (URISyntaxException e) { - Log.e(TAG, "Unable to parse uri for '" + pojo.getName() + "'", e); - } - } + if (appDrawable == null) { + // Retrieve activity icon by intent URI + try { + Intent intent = Intent.parseUri(pojo.intentUri, 0); + ComponentName componentName = PackageManagerUtils.getComponentName(context, intent); + if (componentName != null) { + appDrawable = iconsHandler.getDrawableIconForPackage(PackageManagerUtils.getLaunchingComponent(context, componentName), new fr.neamar.kiss.utils.UserHandle()); + } + } catch (NullPointerException e) { + Log.e(TAG, "Unable to get activity icon for '" + pojo.getName() + "'", e); + } catch (URISyntaxException e) { + Log.e(TAG, "Unable to parse uri for '" + pojo.getName() + "'", e); + } + } - if (appDrawable == null) { - // Retrieve app icon (no Oreo shortcut or a shortcut from an activity that was removed from an installed app) - appDrawable = PackageManagerUtils.getApplicationIcon(context, pojo.packageName); - if (appDrawable != null) { - appDrawable = iconsHandler.applyIconMask(context, appDrawable); + if (appDrawable == null) { + // Retrieve app icon (no Oreo shortcut or a shortcut from an activity that was removed from an installed app) + appDrawable = PackageManagerUtils.getApplicationIcon(context, pojo.packageName); + if (appDrawable != null) { + appDrawable = iconsHandler.applyIconMask(context, appDrawable); + } + } + } } } return appDrawable; } + @Override + boolean isDrawableCached() { + return icon != null; + } + + @Override + void setDrawableCache(Drawable drawable) { + icon = drawable; + } + public Drawable getDrawable(Context context) { - Drawable shortcutDrawable = null; - if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) { - ShortcutInfo shortcutInfo = getShortCut(context); - if (shortcutInfo != null) { - final LauncherApps launcherApps = (LauncherApps) context.getSystemService(Context.LAUNCHER_APPS_SERVICE); - assert launcherApps != null; - try { - shortcutDrawable = launcherApps.getShortcutIconDrawable(shortcutInfo, 0); - } catch (IllegalStateException e) { - // do nothing if user is locked or not running - Log.w(TAG, "Unable to get shortcut icon for '" + pojo.getName() + "', user is locked or not running", e); - } catch (NullPointerException e) { - // shortcuts may use invalid icons, see https://github.com/Neamar/KISS/issues/2158 - Log.e(TAG, "Unable to get shortcut icon for '" + pojo.getName() + "'", e); + if (!isDrawableCached()) { + synchronized (this) { + if (!isDrawableCached()) { + if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) { + ShortcutInfo shortcutInfo = getShortCut(context); + if (shortcutInfo != null) { + final LauncherApps launcherApps = (LauncherApps) context.getSystemService(Context.LAUNCHER_APPS_SERVICE); + assert launcherApps != null; + try { + icon = launcherApps.getShortcutIconDrawable(shortcutInfo, 0); + } catch (IllegalStateException e) { + // do nothing if user is locked or not running + Log.w(TAG, "Unable to get shortcut icon for '" + pojo.getName() + "', user is locked or not running", e); + } catch (NullPointerException e) { + // shortcuts may use invalid icons, see https://github.com/Neamar/KISS/issues/2158 + Log.e(TAG, "Unable to get shortcut icon for '" + pojo.getName() + "'", e); + } + } + } + if (icon == null) { + icon = context.getResources().getDrawable(android.R.drawable.ic_menu_send); + } + if (icon != null) { + icon = DrawableUtils.getThemedDrawable(context, icon); + icon = KissApplication.getApplication(context).getIconsHandler().applyIconMask(context, icon); + } } } } - - if (shortcutDrawable != null) { - shortcutDrawable = DrawableUtils.getThemedDrawable(context, shortcutDrawable); - shortcutDrawable = KissApplication.getApplication(context).getIconsHandler().applyIconMask(context, shortcutDrawable); - } - - return shortcutDrawable; + return icon; } - @Override protected void doLaunch(Context context, View v) { if (pojo.isOreoShortcut()) { From 82857c3ff3587284ecf80ac8476fe17510334e11 Mon Sep 17 00:00:00 2001 From: marunjar Date: Fri, 24 Nov 2023 14:10:28 +0100 Subject: [PATCH 2/2] add changelog --- fastlane/metadata/android/en-US/changelogs/209.txt | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/fastlane/metadata/android/en-US/changelogs/209.txt b/fastlane/metadata/android/en-US/changelogs/209.txt index 89571cd92..6950b7482 100644 --- a/fastlane/metadata/android/en-US/changelogs/209.txt +++ b/fastlane/metadata/android/en-US/changelogs/209.txt @@ -1 +1,2 @@ -* Fix use of broken icon packs \ No newline at end of file +* Fix use of broken icon packs +* Fix flickering shortcut icons