DisplayUtils.java 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565
  1. /*
  2. * Nextcloud Talk application
  3. *
  4. * @author Mario Danic
  5. * Copyright (C) 2017-2020 Mario Danic <mario@lovelyhq.com>
  6. *
  7. * This program is free software: you can redistribute it and/or modify
  8. * it under the terms of the GNU General Public License as published by
  9. * the Free Software Foundation, either version 3 of the License, or
  10. * at your option) any later version.
  11. *
  12. * This program is distributed in the hope that it will be useful,
  13. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  14. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  15. * GNU General Public License for more details.
  16. *
  17. * You should have received a copy of the GNU General Public License
  18. * along with this program. If not, see <http://www.gnu.org/licenses/>.
  19. */
  20. package com.nextcloud.talk.utils;
  21. import android.annotation.SuppressLint;
  22. import android.app.Activity;
  23. import android.content.Context;
  24. import android.content.Intent;
  25. import android.content.res.ColorStateList;
  26. import android.content.res.Configuration;
  27. import android.content.res.Resources;
  28. import android.graphics.Bitmap;
  29. import android.graphics.Canvas;
  30. import android.graphics.Color;
  31. import android.graphics.Typeface;
  32. import android.graphics.drawable.Animatable;
  33. import android.graphics.drawable.BitmapDrawable;
  34. import android.graphics.drawable.Drawable;
  35. import android.graphics.drawable.VectorDrawable;
  36. import android.net.Uri;
  37. import android.os.Build;
  38. import android.text.Spannable;
  39. import android.text.SpannableString;
  40. import android.text.Spanned;
  41. import android.text.TextPaint;
  42. import android.text.TextUtils;
  43. import android.text.method.LinkMovementMethod;
  44. import android.text.style.AbsoluteSizeSpan;
  45. import android.text.style.ClickableSpan;
  46. import android.text.style.ForegroundColorSpan;
  47. import android.text.style.StyleSpan;
  48. import android.util.Log;
  49. import android.util.TypedValue;
  50. import android.view.View;
  51. import android.view.ViewGroup;
  52. import android.view.Window;
  53. import android.view.WindowManager;
  54. import android.widget.EditText;
  55. import android.widget.ImageView;
  56. import android.widget.TextView;
  57. import androidx.annotation.ColorInt;
  58. import androidx.annotation.ColorRes;
  59. import androidx.annotation.DrawableRes;
  60. import androidx.annotation.NonNull;
  61. import androidx.annotation.Nullable;
  62. import androidx.annotation.XmlRes;
  63. import androidx.appcompat.app.AppCompatDelegate;
  64. import androidx.appcompat.widget.AppCompatDrawableManager;
  65. import androidx.appcompat.widget.SearchView;
  66. import androidx.core.content.ContextCompat;
  67. import androidx.core.content.res.ResourcesCompat;
  68. import androidx.core.graphics.ColorUtils;
  69. import androidx.core.graphics.drawable.DrawableCompat;
  70. import androidx.emoji.text.EmojiCompat;
  71. import androidx.viewpager.widget.ViewPager;
  72. import com.facebook.common.executors.UiThreadImmediateExecutorService;
  73. import com.facebook.common.references.CloseableReference;
  74. import com.facebook.datasource.DataSource;
  75. import com.facebook.drawee.backends.pipeline.Fresco;
  76. import com.facebook.drawee.controller.ControllerListener;
  77. import com.facebook.drawee.interfaces.DraweeController;
  78. import com.facebook.drawee.view.SimpleDraweeView;
  79. import com.facebook.imagepipeline.common.RotationOptions;
  80. import com.facebook.imagepipeline.core.ImagePipeline;
  81. import com.facebook.imagepipeline.datasource.BaseBitmapDataSubscriber;
  82. import com.facebook.imagepipeline.image.CloseableImage;
  83. import com.facebook.imagepipeline.image.ImageInfo;
  84. import com.facebook.imagepipeline.postprocessors.RoundAsCirclePostprocessor;
  85. import com.facebook.imagepipeline.postprocessors.RoundPostprocessor;
  86. import com.facebook.imagepipeline.request.ImageRequest;
  87. import com.facebook.imagepipeline.request.ImageRequestBuilder;
  88. import com.facebook.widget.text.span.BetterImageSpan;
  89. import com.google.android.material.chip.ChipDrawable;
  90. import com.nextcloud.talk.R;
  91. import com.nextcloud.talk.application.NextcloudTalkApplication;
  92. import com.nextcloud.talk.events.UserMentionClickEvent;
  93. import com.nextcloud.talk.models.database.UserEntity;
  94. import com.nextcloud.talk.utils.preferences.AppPreferences;
  95. import com.nextcloud.talk.utils.text.Spans;
  96. import org.greenrobot.eventbus.EventBus;
  97. import java.lang.reflect.Constructor;
  98. import java.lang.reflect.InvocationTargetException;
  99. import java.lang.reflect.Method;
  100. import java.util.HashMap;
  101. import java.util.Map;
  102. import java.util.regex.Matcher;
  103. import java.util.regex.Pattern;
  104. import androidx.annotation.ColorInt;
  105. import androidx.annotation.ColorRes;
  106. import androidx.annotation.DrawableRes;
  107. import androidx.annotation.NonNull;
  108. import androidx.annotation.Nullable;
  109. import androidx.annotation.XmlRes;
  110. import androidx.appcompat.widget.AppCompatDrawableManager;
  111. import androidx.core.content.ContextCompat;
  112. import androidx.core.graphics.drawable.DrawableCompat;
  113. import androidx.emoji.text.EmojiCompat;
  114. public class DisplayUtils {
  115. private static final String TAG = "DisplayUtils";
  116. private static final int INDEX_LUMINATION = 2;
  117. private static final double MAX_LIGHTNESS = 0.92;
  118. private static final String TWITTER_HANDLE_PREFIX = "@";
  119. private static final String HTTP_PROTOCOL = "http://";
  120. private static final String HTTPS_PROTOCOL = "https://";
  121. public static void setClickableString(String string, String url, TextView textView) {
  122. SpannableString spannableString = new SpannableString(string);
  123. spannableString.setSpan(new ClickableSpan() {
  124. @Override
  125. public void onClick(@NonNull View widget) {
  126. Intent browserIntent = new Intent(Intent.ACTION_VIEW, Uri.parse(url));
  127. browserIntent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
  128. NextcloudTalkApplication.Companion.getSharedApplication().getApplicationContext().startActivity(browserIntent);
  129. }
  130. @Override
  131. public void updateDrawState(@NonNull TextPaint ds) {
  132. super.updateDrawState(ds);
  133. ds.setUnderlineText(false);
  134. }
  135. }, 0, string.length(), Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
  136. textView.setText(spannableString);
  137. textView.setMovementMethod(LinkMovementMethod.getInstance());
  138. }
  139. private static void updateViewSize(@Nullable ImageInfo imageInfo, SimpleDraweeView draweeView) {
  140. if (imageInfo != null) {
  141. int maxSize = draweeView.getContext().getResources().getDimensionPixelSize(R.dimen.maximum_file_preview_size);
  142. draweeView.getLayoutParams().width = imageInfo.getWidth() > maxSize ? maxSize : imageInfo.getWidth();
  143. draweeView.getLayoutParams().height = ViewGroup.LayoutParams.WRAP_CONTENT;
  144. draweeView.setAspectRatio((float) imageInfo.getWidth() / imageInfo.getHeight());
  145. draweeView.requestLayout();
  146. }
  147. }
  148. public static Drawable getRoundedDrawable(Drawable drawable) {
  149. Bitmap bitmap = getBitmap(drawable);
  150. new RoundAsCirclePostprocessor(true).process(bitmap);
  151. return new BitmapDrawable(bitmap);
  152. }
  153. public static Bitmap getRoundedBitmapFromVectorDrawableResource(Resources resources, int resource) {
  154. VectorDrawable vectorDrawable = (VectorDrawable) resources.getDrawable(resource);
  155. Bitmap bitmap = getBitmap(vectorDrawable);
  156. new RoundPostprocessor(true).process(bitmap);
  157. return bitmap;
  158. }
  159. public static Drawable getRoundedBitmapDrawableFromVectorDrawableResource(Resources resources, int resource) {
  160. return new BitmapDrawable(getRoundedBitmapFromVectorDrawableResource(resources, resource));
  161. }
  162. private static Bitmap getBitmap(Drawable drawable) {
  163. Bitmap bitmap = Bitmap.createBitmap(drawable.getIntrinsicWidth(),
  164. drawable.getIntrinsicHeight(), Bitmap.Config.ARGB_8888);
  165. Canvas canvas = new Canvas(bitmap);
  166. drawable.setBounds(0, 0, canvas.getWidth(), canvas.getHeight());
  167. drawable.draw(canvas);
  168. return bitmap;
  169. }
  170. public static ImageRequest getImageRequestForUrl(String url, @Nullable UserEntity userEntity) {
  171. Map<String, String> headers = new HashMap<>();
  172. if (userEntity != null && url.startsWith(userEntity.getBaseUrl()) && (url.contains("index.php/core/preview?fileId=") || url.contains("/avatar/"))) {
  173. headers.put("Authorization", ApiUtils.getCredentials(userEntity.getUsername(), userEntity.getToken()));
  174. }
  175. return ImageRequestBuilder.newBuilderWithSource(Uri.parse(url))
  176. .setProgressiveRenderingEnabled(true)
  177. .setRotationOptions(RotationOptions.autoRotate())
  178. .disableDiskCache()
  179. .setHeaders(headers)
  180. .build();
  181. }
  182. public static ControllerListener getImageControllerListener(SimpleDraweeView draweeView) {
  183. return new ControllerListener() {
  184. @Override
  185. public void onSubmit(String id, Object callerContext) {
  186. }
  187. @Override
  188. public void onFinalImageSet(String id, @androidx.annotation.Nullable Object imageInfo, @androidx.annotation.Nullable Animatable animatable) {
  189. updateViewSize((ImageInfo) imageInfo, draweeView);
  190. }
  191. @Override
  192. public void onIntermediateImageSet(String id, @androidx.annotation.Nullable Object imageInfo) {
  193. updateViewSize((ImageInfo) imageInfo, draweeView);
  194. }
  195. @Override
  196. public void onIntermediateImageFailed(String id, Throwable throwable) {
  197. }
  198. @Override
  199. public void onFailure(String id, Throwable throwable) {
  200. }
  201. @Override
  202. public void onRelease(String id) {
  203. }
  204. };
  205. }
  206. public static float convertDpToPixel(float dp, Context context) {
  207. return Math.round(TypedValue.applyDimension(TypedValue.COMPLEX_UNIT_DIP, dp,
  208. context.getResources().getDisplayMetrics()) + 0.5f);
  209. }
  210. // Solution inspired by https://stackoverflow.com/questions/34936590/why-isnt-my-vector-drawable-scaling-as-expected
  211. public static void useCompatVectorIfNeeded() {
  212. if (Build.VERSION.SDK_INT < 23) {
  213. try {
  214. @SuppressLint("RestrictedApi") AppCompatDrawableManager drawableManager = AppCompatDrawableManager.get();
  215. Class<?> inflateDelegateClass = Class.forName("android.support.v7.widget.AppCompatDrawableManager$InflateDelegate");
  216. Class<?> vdcInflateDelegateClass = Class.forName("android.support.v7.widget.AppCompatDrawableManager$VdcInflateDelegate");
  217. Constructor<?> constructor = vdcInflateDelegateClass.getDeclaredConstructor();
  218. constructor.setAccessible(true);
  219. Object vdcInflateDelegate = constructor.newInstance();
  220. Class<?> args[] = {String.class, inflateDelegateClass};
  221. Method addDelegate = AppCompatDrawableManager.class.getDeclaredMethod("addDelegate", args);
  222. addDelegate.setAccessible(true);
  223. addDelegate.invoke(drawableManager, "vector", vdcInflateDelegate);
  224. } catch (ClassNotFoundException | NoSuchMethodException | InstantiationException |
  225. InvocationTargetException | IllegalAccessException e) {
  226. Log.e(TAG, "Failed to use reflection to enable proper vector scaling");
  227. }
  228. }
  229. }
  230. public static Drawable getTintedDrawable(Resources res, @DrawableRes int drawableResId, @ColorRes int colorResId) {
  231. Drawable drawable = res.getDrawable(drawableResId);
  232. int color = res.getColor(colorResId);
  233. drawable.setTint(color);
  234. return drawable;
  235. }
  236. public static Drawable getDrawableForMentionChipSpan(Context context, String id, CharSequence label,
  237. UserEntity conversationUser, String type,
  238. @XmlRes int chipResource,
  239. @Nullable EditText emojiEditText) {
  240. ChipDrawable chip = ChipDrawable.createFromResource(context, chipResource);
  241. chip.setText(EmojiCompat.get().process(label));
  242. chip.setEllipsize(TextUtils.TruncateAt.MIDDLE);
  243. if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
  244. Configuration config = context.getResources().getConfiguration();
  245. chip.setLayoutDirection(config.getLayoutDirection());
  246. }
  247. int drawable;
  248. boolean isCall = "call".equals(type) || "calls".equals(type);
  249. if (!isCall) {
  250. if (chipResource == R.xml.chip_you) {
  251. drawable = R.drawable.mention_chip;
  252. } else {
  253. drawable = R.drawable.accent_circle;
  254. }
  255. chip.setChipIconResource(drawable);
  256. } else {
  257. chip.setChipIconResource(R.drawable.ic_circular_group);
  258. }
  259. chip.setBounds(0, 0, chip.getIntrinsicWidth(), chip.getIntrinsicHeight());
  260. if (!isCall) {
  261. String url = ApiUtils.getUrlForAvatarWithName(conversationUser.getBaseUrl(), id, R.dimen.avatar_size_big);
  262. if ("guests".equals(type) || "guest".equals(type)) {
  263. url = ApiUtils.getUrlForAvatarWithNameForGuests(conversationUser.getBaseUrl(), String.valueOf(label), R.dimen.avatar_size_big);
  264. }
  265. ImageRequest imageRequest = getImageRequestForUrl(url, null);
  266. ImagePipeline imagePipeline = Fresco.getImagePipeline();
  267. DataSource<CloseableReference<CloseableImage>> dataSource = imagePipeline.fetchDecodedImage(imageRequest, context);
  268. dataSource.subscribe(
  269. new BaseBitmapDataSubscriber() {
  270. @Override
  271. protected void onNewResultImpl(Bitmap bitmap) {
  272. if (bitmap != null) {
  273. chip.setChipIcon(getRoundedDrawable(new BitmapDrawable(bitmap)));
  274. // A hack to refresh the chip icon
  275. if (emojiEditText != null) {
  276. emojiEditText.post(() -> emojiEditText.setTextKeepState(emojiEditText.getText(), TextView.BufferType.SPANNABLE));
  277. }
  278. }
  279. }
  280. @Override
  281. protected void onFailureImpl(DataSource<CloseableReference<CloseableImage>> dataSource) {
  282. }
  283. },
  284. UiThreadImmediateExecutorService.getInstance());
  285. }
  286. return chip;
  287. }
  288. public static Spannable searchAndReplaceWithMentionSpan(Context context, Spannable text,
  289. String id, String label, String type,
  290. UserEntity conversationUser,
  291. @XmlRes int chipXmlRes) {
  292. Spannable spannableString = new SpannableString(text);
  293. String stringText = text.toString();
  294. Matcher m = Pattern.compile("@" + label,
  295. Pattern.CASE_INSENSITIVE | Pattern.LITERAL | Pattern.MULTILINE)
  296. .matcher(spannableString);
  297. ClickableSpan clickableSpan = new ClickableSpan() {
  298. @Override
  299. public void onClick(@NonNull View widget) {
  300. EventBus.getDefault().post(new UserMentionClickEvent(id));
  301. }
  302. };
  303. int lastStartIndex = -1;
  304. Spans.MentionChipSpan mentionChipSpan;
  305. while (m.find()) {
  306. int start = stringText.indexOf(m.group(), lastStartIndex);
  307. int end = start + m.group().length();
  308. lastStartIndex = end;
  309. mentionChipSpan = new Spans.MentionChipSpan(DisplayUtils.getDrawableForMentionChipSpan(context,
  310. id, label, conversationUser, type, chipXmlRes, null),
  311. BetterImageSpan.ALIGN_CENTER, id,
  312. label);
  313. spannableString.setSpan(mentionChipSpan, start, end, Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);
  314. if ("user".equals(type) && !conversationUser.getUserId().equals(id)) {
  315. spannableString.setSpan(clickableSpan, start, end, Spannable.SPAN_INCLUSIVE_EXCLUSIVE);
  316. }
  317. }
  318. return spannableString;
  319. }
  320. public static Spannable searchAndColor(Spannable text, String searchText, @ColorInt int color) {
  321. Spannable spannableString = new SpannableString(text);
  322. String stringText = text.toString();
  323. if (TextUtils.isEmpty(text) || TextUtils.isEmpty(searchText)) {
  324. return spannableString;
  325. }
  326. Matcher m = Pattern.compile(searchText,
  327. Pattern.CASE_INSENSITIVE | Pattern.LITERAL | Pattern.MULTILINE)
  328. .matcher(spannableString);
  329. int textSize = NextcloudTalkApplication.Companion.getSharedApplication().getResources().getDimensionPixelSize(R.dimen
  330. .chat_text_size);
  331. int lastStartIndex = -1;
  332. while (m.find()) {
  333. int start = stringText.indexOf(m.group(), lastStartIndex);
  334. int end = start + m.group().length();
  335. lastStartIndex = end;
  336. spannableString.setSpan(new ForegroundColorSpan(color), start, end,
  337. Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);
  338. spannableString.setSpan(new StyleSpan(Typeface.BOLD), start, end, Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);
  339. spannableString.setSpan(new AbsoluteSizeSpan(textSize), start, end, Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);
  340. }
  341. return spannableString;
  342. }
  343. public static Drawable getMessageSelector(@ColorInt int normalColor, @ColorInt int selectedColor,
  344. @ColorInt int pressedColor, @DrawableRes int shape) {
  345. Drawable vectorDrawable = ContextCompat.getDrawable(NextcloudTalkApplication.Companion.getSharedApplication()
  346. .getApplicationContext(),
  347. shape);
  348. Drawable drawable = DrawableCompat.wrap(vectorDrawable).mutate();
  349. DrawableCompat.setTintList(
  350. drawable,
  351. new ColorStateList(
  352. new int[][]{
  353. new int[]{android.R.attr.state_selected},
  354. new int[]{android.R.attr.state_pressed},
  355. new int[]{-android.R.attr.state_pressed, -android.R.attr.state_selected}
  356. },
  357. new int[]{selectedColor, pressedColor, normalColor}
  358. ));
  359. return drawable;
  360. }
  361. /**
  362. * Sets the color of the status bar to {@code color}.
  363. *
  364. * @param activity activity
  365. * @param color the color
  366. */
  367. public static void applyColorToStatusBar(Activity activity, @ColorInt int color) {
  368. Window window = activity.getWindow();
  369. boolean isLightTheme = lightTheme(color);
  370. if (window != null) {
  371. if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
  372. View decor = window.getDecorView();
  373. if (isLightTheme) {
  374. int systemUiFlagLightStatusBar;
  375. if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
  376. systemUiFlagLightStatusBar = View.SYSTEM_UI_FLAG_LIGHT_STATUS_BAR | View.SYSTEM_UI_FLAG_LIGHT_NAVIGATION_BAR;
  377. } else {
  378. systemUiFlagLightStatusBar = View.SYSTEM_UI_FLAG_LIGHT_STATUS_BAR;
  379. }
  380. decor.setSystemUiVisibility(systemUiFlagLightStatusBar);
  381. } else {
  382. decor.setSystemUiVisibility(0);
  383. }
  384. window.setStatusBarColor(color);
  385. } else if (isLightTheme) {
  386. window.setStatusBarColor(Color.BLACK);
  387. }
  388. }
  389. }
  390. /**
  391. * Tests if light color is set
  392. *
  393. * @param color the color
  394. * @return true if primaryColor is lighter than MAX_LIGHTNESS
  395. */
  396. public static boolean lightTheme(int color) {
  397. float[] hsl = colorToHSL(color);
  398. return hsl[INDEX_LUMINATION] >= MAX_LIGHTNESS;
  399. }
  400. private static float[] colorToHSL(int color) {
  401. float[] hsl = new float[3];
  402. ColorUtils.RGBToHSL(Color.red(color), Color.green(color), Color.blue(color), hsl);
  403. return hsl;
  404. }
  405. public static void applyColorToNavigationBar(Window window, @ColorInt int color) {
  406. window.setNavigationBarColor(color);
  407. }
  408. /**
  409. * Theme search view
  410. *
  411. * @param searchView searchView to be changed
  412. * @param context the app's context
  413. */
  414. public static void themeSearchView(SearchView searchView, Context context) {
  415. // hacky as no default way is provided
  416. SearchView.SearchAutoComplete editText = searchView.findViewById(R.id.search_src_text);
  417. editText.setTextSize(16);
  418. editText.setHintTextColor(context.getResources().getColor(R.color.fontSecondaryAppbar));
  419. }
  420. /**
  421. * beautifies a given URL by removing any http/https protocol prefix.
  422. *
  423. * @param url to be beautified url
  424. * @return beautified url
  425. */
  426. public static String beautifyURL(@Nullable String url) {
  427. if (TextUtils.isEmpty(url)) {
  428. return "";
  429. }
  430. if (url.length() >= 7 && HTTP_PROTOCOL.equalsIgnoreCase(url.substring(0, 7))) {
  431. return url.substring(HTTP_PROTOCOL.length()).trim();
  432. }
  433. if (url.length() >= 8 && HTTPS_PROTOCOL.equalsIgnoreCase(url.substring(0, 8))) {
  434. return url.substring(HTTPS_PROTOCOL.length()).trim();
  435. }
  436. return url.trim();
  437. }
  438. /**
  439. * beautifies a given twitter handle by prefixing it with an @ in case it is missing.
  440. *
  441. * @param handle to be beautified twitter handle
  442. * @return beautified twitter handle
  443. */
  444. public static String beautifyTwitterHandle(@Nullable String handle) {
  445. if (handle != null) {
  446. String trimmedHandle = handle.trim();
  447. if (TextUtils.isEmpty(trimmedHandle)) {
  448. return "";
  449. }
  450. if (trimmedHandle.startsWith(TWITTER_HANDLE_PREFIX)) {
  451. return trimmedHandle;
  452. } else {
  453. return TWITTER_HANDLE_PREFIX + trimmedHandle;
  454. }
  455. } else {
  456. return "";
  457. }
  458. }
  459. public static void loadAvatarImage(UserEntity user, SimpleDraweeView avatarImageView, boolean deleteCache) {
  460. String avatarId;
  461. if (!TextUtils.isEmpty(user.getUserId())) {
  462. avatarId = user.getUserId();
  463. } else {
  464. avatarId = user.getUsername();
  465. }
  466. String avatarString = ApiUtils.getUrlForAvatarWithName(user.getBaseUrl(), avatarId, R.dimen.avatar_size_big);
  467. // clear cache
  468. if (deleteCache) {
  469. Uri avatarUri = Uri.parse(avatarString);
  470. ImagePipeline imagePipeline = Fresco.getImagePipeline();
  471. imagePipeline.evictFromMemoryCache(avatarUri);
  472. imagePipeline.evictFromDiskCache(avatarUri);
  473. imagePipeline.evictFromCache(avatarUri);
  474. }
  475. DraweeController draweeController = Fresco.newDraweeControllerBuilder()
  476. .setOldController(avatarImageView.getController())
  477. .setAutoPlayAnimations(true)
  478. .setImageRequest(DisplayUtils.getImageRequestForUrl(avatarString, null))
  479. .build();
  480. avatarImageView.setController(draweeController);
  481. }
  482. }