UserInfoActivity.java 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417
  1. /*
  2. * Nextcloud Android client application
  3. *
  4. * @author Mario Danic
  5. * @author Andy Scherzinger
  6. * @author Chris Narkiewicz <hello@ezaquarii.com>
  7. * @author Chawki Chouib <chouibc@gmail.com>
  8. * Copyright (C) 2017 Mario Danic
  9. * Copyright (C) 2017 Andy Scherzinger
  10. * Copyright (C) 2017 Nextcloud GmbH.
  11. * Copyright (C) 2020 Chris Narkiewicz <hello@ezaquarii.com>
  12. * Copyright (C) 2020 Chawki Chouib <chouibc@gmail.com>
  13. *
  14. * This program is free software: you can redistribute it and/or modify
  15. * it under the terms of the GNU Affero General Public License as published by
  16. * the Free Software Foundation, either version 3 of the License, or
  17. * at your option) any later version.
  18. *
  19. * This program is distributed in the hope that it will be useful,
  20. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  21. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  22. * GNU Affero General Public License for more details.
  23. *
  24. * You should have received a copy of the GNU Affero General Public License
  25. * along with this program. If not, see <http://www.gnu.org/licenses/>.
  26. */
  27. package com.owncloud.android.ui.activity;
  28. import android.graphics.drawable.ColorDrawable;
  29. import android.graphics.drawable.Drawable;
  30. import android.graphics.drawable.LayerDrawable;
  31. import android.os.Bundle;
  32. import android.text.TextUtils;
  33. import android.view.LayoutInflater;
  34. import android.view.Menu;
  35. import android.view.MenuInflater;
  36. import android.view.MenuItem;
  37. import android.view.View;
  38. import android.view.ViewGroup;
  39. import android.webkit.URLUtil;
  40. import android.widget.ImageView;
  41. import android.widget.TextView;
  42. import com.bumptech.glide.Glide;
  43. import com.bumptech.glide.request.animation.GlideAnimation;
  44. import com.bumptech.glide.request.target.SimpleTarget;
  45. import com.nextcloud.client.account.User;
  46. import com.nextcloud.client.di.Injectable;
  47. import com.nextcloud.client.preferences.AppPreferences;
  48. import com.owncloud.android.R;
  49. import com.owncloud.android.databinding.UserInfoLayoutBinding;
  50. import com.owncloud.android.lib.common.UserInfo;
  51. import com.owncloud.android.lib.common.operations.RemoteOperation;
  52. import com.owncloud.android.lib.common.operations.RemoteOperationResult;
  53. import com.owncloud.android.lib.common.utils.Log_OC;
  54. import com.owncloud.android.lib.resources.users.GetUserInfoRemoteOperation;
  55. import com.owncloud.android.ui.dialog.AccountRemovalConfirmationDialog;
  56. import com.owncloud.android.ui.events.TokenPushEvent;
  57. import com.owncloud.android.utils.DisplayUtils;
  58. import com.owncloud.android.utils.PushUtils;
  59. import com.owncloud.android.utils.theme.ThemeColorUtils;
  60. import com.owncloud.android.utils.theme.ThemeToolbarUtils;
  61. import org.greenrobot.eventbus.Subscribe;
  62. import org.greenrobot.eventbus.ThreadMode;
  63. import org.parceler.Parcels;
  64. import java.util.LinkedList;
  65. import java.util.List;
  66. import javax.inject.Inject;
  67. import androidx.annotation.ColorInt;
  68. import androidx.annotation.DrawableRes;
  69. import androidx.annotation.NonNull;
  70. import androidx.annotation.StringRes;
  71. import androidx.appcompat.app.ActionBar;
  72. import androidx.core.content.res.ResourcesCompat;
  73. import androidx.core.graphics.drawable.DrawableCompat;
  74. import androidx.fragment.app.FragmentManager;
  75. import androidx.lifecycle.Lifecycle;
  76. import androidx.recyclerview.widget.RecyclerView;
  77. import butterknife.BindView;
  78. import butterknife.ButterKnife;
  79. /**
  80. * This Activity presents the user information.
  81. */
  82. public class UserInfoActivity extends DrawerActivity implements Injectable {
  83. public static final String KEY_ACCOUNT = "ACCOUNT";
  84. private static final String TAG = UserInfoActivity.class.getSimpleName();
  85. public static final String KEY_USER_DATA = "USER_DATA";
  86. @Inject AppPreferences preferences;
  87. private float mCurrentAccountAvatarRadiusDimension;
  88. private UserInfo userInfo;
  89. private User user;
  90. private UserInfoLayoutBinding binding;
  91. @Override
  92. public void onCreate(Bundle savedInstanceState) {
  93. Log_OC.v(TAG, "onCreate() start");
  94. super.onCreate(savedInstanceState);
  95. Bundle bundle = getIntent().getExtras();
  96. if (bundle == null) {
  97. finish();
  98. return;
  99. }
  100. user = bundle.getParcelable(KEY_ACCOUNT);
  101. if(user == null) {
  102. finish();
  103. return;
  104. }
  105. if (savedInstanceState != null && savedInstanceState.containsKey(KEY_USER_DATA)) {
  106. userInfo = Parcels.unwrap(savedInstanceState.getParcelable(KEY_USER_DATA));
  107. } else if (bundle.containsKey(KEY_ACCOUNT)) {
  108. userInfo = Parcels.unwrap(bundle.getParcelable(KEY_USER_DATA));
  109. }
  110. mCurrentAccountAvatarRadiusDimension = getResources().getDimension(R.dimen.user_icon_radius);
  111. binding = UserInfoLayoutBinding.inflate(getLayoutInflater());
  112. setContentView(binding.getRoot());
  113. setupToolbar();
  114. // set the back button from action bar
  115. ActionBar actionBar = getSupportActionBar();
  116. // check if is not null
  117. if (actionBar != null) {
  118. actionBar.setDisplayHomeAsUpEnabled(true);
  119. actionBar.setDisplayShowHomeEnabled(true);
  120. ThemeToolbarUtils.tintBackButton(actionBar, this);
  121. }
  122. binding.userinfoList.setAdapter(
  123. new UserInfoAdapter(null, ThemeColorUtils.primaryColor(getAccount(), true, this)));
  124. if (userInfo != null) {
  125. populateUserInfoUi(userInfo);
  126. } else {
  127. setMultiListLoadingMessage();
  128. fetchAndSetData();
  129. }
  130. setHeaderImage();
  131. }
  132. @Override
  133. public boolean onPrepareOptionsMenu(Menu menu) {
  134. if (accountManager.getUser().equals(user)) {
  135. menu.findItem(R.id.action_open_account).setVisible(false);
  136. }
  137. return super.onPrepareOptionsMenu(menu);
  138. }
  139. @Override
  140. public boolean onCreateOptionsMenu(Menu menu) {
  141. MenuInflater inflater = getMenuInflater();
  142. inflater.inflate(R.menu.item_account, menu);
  143. return true;
  144. }
  145. @Override
  146. public boolean onOptionsItemSelected(MenuItem item) {
  147. boolean retval = true;
  148. switch (item.getItemId()) {
  149. case android.R.id.home:
  150. onBackPressed();
  151. break;
  152. case R.id.action_open_account:
  153. accountClicked(user.hashCode());
  154. break;
  155. case R.id.action_delete_account:
  156. openAccountRemovalConfirmationDialog(user, getSupportFragmentManager());
  157. break;
  158. default:
  159. retval = super.onOptionsItemSelected(item);
  160. break;
  161. }
  162. return retval;
  163. }
  164. private void setMultiListLoadingMessage() {
  165. binding.userinfoList.setVisibility(View.GONE);
  166. binding.emptyList.emptyListView.setVisibility(View.GONE);
  167. }
  168. private void setErrorMessageForMultiList(String headline, String message, @DrawableRes int errorResource) {
  169. binding.emptyList.emptyListViewHeadline.setText(headline);
  170. binding.emptyList.emptyListViewText.setText(message);
  171. binding.emptyList.emptyListIcon.setImageResource(errorResource);
  172. binding.emptyList.emptyListIcon.setVisibility(View.VISIBLE);
  173. binding.emptyList.emptyListViewText.setVisibility(View.VISIBLE);
  174. binding.userinfoList.setVisibility(View.GONE);
  175. binding.loadingContent.setVisibility(View.GONE);
  176. }
  177. private void setHeaderImage() {
  178. if (getStorageManager().getCapability(user.getAccountName()).getServerBackground() != null) {
  179. ImageView backgroundImageView = findViewById(R.id.userinfo_background);
  180. if (backgroundImageView != null) {
  181. String background = getStorageManager().getCapability(user.getAccountName()).getServerBackground();
  182. int primaryColor = ThemeColorUtils.primaryColor(getAccount(), false, this);
  183. if (URLUtil.isValidUrl(background)) {
  184. // background image
  185. SimpleTarget target = new SimpleTarget<Drawable>() {
  186. @Override
  187. public void onResourceReady(Drawable resource, GlideAnimation glideAnimation) {
  188. Drawable[] drawables = {new ColorDrawable(primaryColor), resource};
  189. LayerDrawable layerDrawable = new LayerDrawable(drawables);
  190. backgroundImageView.setImageDrawable(layerDrawable);
  191. }
  192. @Override
  193. public void onLoadFailed(Exception e, Drawable errorDrawable) {
  194. Drawable[] drawables = {new ColorDrawable(primaryColor),
  195. ResourcesCompat.getDrawable(getResources(),
  196. R.drawable.background,
  197. null)};
  198. LayerDrawable layerDrawable = new LayerDrawable(drawables);
  199. backgroundImageView.setImageDrawable(layerDrawable);
  200. }
  201. };
  202. Glide.with(this)
  203. .load(background)
  204. .centerCrop()
  205. .placeholder(R.drawable.background)
  206. .error(R.drawable.background)
  207. .crossFade()
  208. .into(target);
  209. } else {
  210. // plain color
  211. backgroundImageView.setImageDrawable(new ColorDrawable(primaryColor));
  212. }
  213. }
  214. }
  215. }
  216. private void populateUserInfoUi(UserInfo userInfo) {
  217. binding.userinfoUsername.setText(user.getAccountName());
  218. binding.userinfoIcon.setTag(user.getAccountName());
  219. DisplayUtils.setAvatar(user,
  220. this,
  221. mCurrentAccountAvatarRadiusDimension,
  222. getResources(),
  223. binding.userinfoIcon,
  224. this);
  225. int tint = ThemeColorUtils.primaryColor(user.toPlatformAccount(), true, this);
  226. if (!TextUtils.isEmpty(userInfo.getDisplayName())) {
  227. binding.userinfoFullName.setText(userInfo.getDisplayName());
  228. }
  229. if (TextUtils.isEmpty(userInfo.getPhone()) && TextUtils.isEmpty(userInfo.getEmail())
  230. && TextUtils.isEmpty(userInfo.getAddress()) && TextUtils.isEmpty(userInfo.getTwitter())
  231. && TextUtils.isEmpty(userInfo.getWebsite())) {
  232. binding.userinfoList.setVisibility(View.GONE);
  233. binding.loadingContent.setVisibility(View.GONE);
  234. binding.emptyList.emptyListView.setVisibility(View.VISIBLE);
  235. setErrorMessageForMultiList(getString(R.string.userinfo_no_info_headline),
  236. getString(R.string.userinfo_no_info_text), R.drawable.ic_user);
  237. } else {
  238. binding.loadingContent.setVisibility(View.VISIBLE);
  239. binding.emptyList.emptyListView.setVisibility(View.GONE);
  240. if (binding.userinfoList.getAdapter() instanceof UserInfoAdapter) {
  241. binding.userinfoList.setAdapter(new UserInfoAdapter(createUserInfoDetails(userInfo), tint));
  242. }
  243. binding.loadingContent.setVisibility(View.GONE);
  244. binding.userinfoList.setVisibility(View.VISIBLE);
  245. }
  246. }
  247. private List<UserInfoDetailsItem> createUserInfoDetails(UserInfo userInfo) {
  248. List<UserInfoDetailsItem> result = new LinkedList<>();
  249. addToListIfNeeded(result, R.drawable.ic_phone, userInfo.getPhone(), R.string.user_info_phone);
  250. addToListIfNeeded(result, R.drawable.ic_email, userInfo.getEmail(), R.string.user_info_email);
  251. addToListIfNeeded(result, R.drawable.ic_map_marker, userInfo.getAddress(), R.string.user_info_address);
  252. addToListIfNeeded(result, R.drawable.ic_web, DisplayUtils.beautifyURL(userInfo.getWebsite()),
  253. R.string.user_info_website);
  254. addToListIfNeeded(result, R.drawable.ic_twitter, DisplayUtils.beautifyTwitterHandle(userInfo.getTwitter()),
  255. R.string.user_info_twitter);
  256. return result;
  257. }
  258. private void addToListIfNeeded(List<UserInfoDetailsItem> info, @DrawableRes int icon, String text,
  259. @StringRes int contentDescriptionInt) {
  260. if (!TextUtils.isEmpty(text)) {
  261. info.add(new UserInfoDetailsItem(icon, text, getResources().getString(contentDescriptionInt)));
  262. }
  263. }
  264. public static void openAccountRemovalConfirmationDialog(User user, FragmentManager fragmentManager) {
  265. AccountRemovalConfirmationDialog dialog = AccountRemovalConfirmationDialog.newInstance(user);
  266. dialog.show(fragmentManager, "dialog");
  267. }
  268. private void fetchAndSetData() {
  269. Thread t = new Thread(() -> {
  270. RemoteOperation getRemoteUserInfoOperation = new GetUserInfoRemoteOperation();
  271. RemoteOperationResult result = getRemoteUserInfoOperation.execute(user.toPlatformAccount(), this);
  272. if (getLifecycle().getCurrentState().isAtLeast(Lifecycle.State.RESUMED)) {
  273. if (result.isSuccess() && result.getData() != null) {
  274. userInfo = (UserInfo) result.getData().get(0);
  275. runOnUiThread(() -> populateUserInfoUi(userInfo));
  276. } else {
  277. // show error
  278. runOnUiThread(() -> setErrorMessageForMultiList(
  279. getString(R.string.user_information_retrieval_error),
  280. result.getLogMessage(),
  281. R.drawable.ic_list_empty_error)
  282. );
  283. Log_OC.d(TAG, result.getLogMessage());
  284. }
  285. }
  286. });
  287. t.start();
  288. }
  289. @Override
  290. protected void onSaveInstanceState(@NonNull Bundle outState) {
  291. super.onSaveInstanceState(outState);
  292. if (userInfo != null) {
  293. outState.putParcelable(KEY_USER_DATA, Parcels.wrap(userInfo));
  294. }
  295. }
  296. @Subscribe(threadMode = ThreadMode.BACKGROUND)
  297. public void onMessageEvent(TokenPushEvent event) {
  298. PushUtils.pushRegistrationToServer(getUserAccountManager(), preferences.getPushToken());
  299. }
  300. protected static class UserInfoDetailsItem {
  301. @DrawableRes public int icon;
  302. public String text;
  303. public String iconContentDescription;
  304. public UserInfoDetailsItem(@DrawableRes int icon, String text, String iconContentDescription) {
  305. this.icon = icon;
  306. this.text = text;
  307. this.iconContentDescription = iconContentDescription;
  308. }
  309. }
  310. protected static class UserInfoAdapter extends RecyclerView.Adapter<UserInfoAdapter.ViewHolder> {
  311. protected List<UserInfoDetailsItem> mDisplayList;
  312. @ColorInt protected int mTintColor;
  313. public static class ViewHolder extends RecyclerView.ViewHolder {
  314. @BindView(R.id.icon) protected ImageView icon;
  315. @BindView(R.id.text) protected TextView text;
  316. public ViewHolder(View itemView) {
  317. super(itemView);
  318. ButterKnife.bind(this, itemView);
  319. }
  320. }
  321. public UserInfoAdapter(List<UserInfoDetailsItem> displayList, @ColorInt int tintColor) {
  322. mDisplayList = displayList == null ? new LinkedList<>() : displayList;
  323. mTintColor = tintColor;
  324. }
  325. public void setData(List<UserInfoDetailsItem> displayList) {
  326. mDisplayList = displayList == null ? new LinkedList<>() : displayList;
  327. notifyDataSetChanged();
  328. }
  329. @NonNull
  330. @Override
  331. public ViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) {
  332. LayoutInflater inflater = LayoutInflater.from(parent.getContext());
  333. View view = inflater.inflate(R.layout.user_info_details_table_item, parent, false);
  334. return new ViewHolder(view);
  335. }
  336. @Override
  337. public void onBindViewHolder(@NonNull ViewHolder holder, int position) {
  338. UserInfoDetailsItem item = mDisplayList.get(position);
  339. holder.icon.setImageResource(item.icon);
  340. holder.text.setText(item.text);
  341. holder.icon.setContentDescription(item.iconContentDescription);
  342. DrawableCompat.setTint(holder.icon.getDrawable(), mTintColor);
  343. }
  344. @Override
  345. public int getItemCount() {
  346. return mDisplayList.size();
  347. }
  348. }
  349. }