UserInfoActivity.java 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457
  1. /**
  2. * Nextcloud Android client application
  3. *
  4. * @author Mario Danic
  5. * @author Andy Scherzinger
  6. * Copyright (C) 2017 Mario Danic
  7. * Copyright (C) 2017 Andy Scherzinger
  8. * Copyright (C) 2017 Nextcloud GmbH.
  9. *
  10. * This program is free software: you can redistribute it and/or modify
  11. * it under the terms of the GNU Affero General Public License as published by
  12. * the Free Software Foundation, either version 3 of the License, or
  13. * at your option) any later version.
  14. *
  15. * This program is distributed in the hope that it will be useful,
  16. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  17. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  18. * GNU Affero General Public License for more details.
  19. *
  20. * You should have received a copy of the GNU Affero General Public License
  21. * along with this program. If not, see <http://www.gnu.org/licenses/>.
  22. */
  23. package com.owncloud.android.ui.activity;
  24. import android.accounts.Account;
  25. import android.accounts.AccountManager;
  26. import android.app.AlertDialog;
  27. import android.app.Dialog;
  28. import android.app.DialogFragment;
  29. import android.app.FragmentManager;
  30. import android.content.ContentResolver;
  31. import android.content.DialogInterface;
  32. import android.content.Intent;
  33. import android.os.Bundle;
  34. import android.support.annotation.ColorInt;
  35. import android.support.annotation.NonNull;
  36. import android.support.annotation.Nullable;
  37. import android.support.v4.content.ContextCompat;
  38. import android.support.v4.graphics.drawable.DrawableCompat;
  39. import android.text.TextUtils;
  40. import android.view.Menu;
  41. import android.view.MenuInflater;
  42. import android.view.MenuItem;
  43. import android.view.View;
  44. import android.widget.ImageView;
  45. import android.widget.LinearLayout;
  46. import android.widget.ProgressBar;
  47. import android.widget.TextView;
  48. import com.google.gson.Gson;
  49. import com.owncloud.android.R;
  50. import com.owncloud.android.authentication.AccountUtils;
  51. import com.owncloud.android.authentication.AuthenticatorActivity;
  52. import com.owncloud.android.datamodel.ArbitraryDataProvider;
  53. import com.owncloud.android.datamodel.PushConfigurationState;
  54. import com.owncloud.android.datamodel.SyncedFolderProvider;
  55. import com.owncloud.android.datamodel.UploadsStorageManager;
  56. import com.owncloud.android.lib.common.UserInfo;
  57. import com.owncloud.android.lib.common.operations.RemoteOperation;
  58. import com.owncloud.android.lib.common.operations.RemoteOperationResult;
  59. import com.owncloud.android.lib.common.utils.Log_OC;
  60. import com.owncloud.android.lib.resources.users.GetRemoteUserInfoOperation;
  61. import com.owncloud.android.ui.events.TokenPushEvent;
  62. import com.owncloud.android.utils.DisplayUtils;
  63. import com.owncloud.android.utils.PushUtils;
  64. import org.greenrobot.eventbus.EventBus;
  65. import org.greenrobot.eventbus.Subscribe;
  66. import org.greenrobot.eventbus.ThreadMode;
  67. import org.parceler.Parcels;
  68. import butterknife.BindString;
  69. import butterknife.BindView;
  70. import butterknife.ButterKnife;
  71. import butterknife.Unbinder;
  72. /**
  73. * This Activity presents the user information.
  74. */
  75. public class UserInfoActivity extends FileActivity {
  76. private static final String TAG = UserInfoActivity.class.getSimpleName();
  77. private static final String KEY_USER_DATA = "USER_DATA";
  78. private static final String KEY_ACCOUNT = "ACCOUNT";
  79. private static final String KEY_DIRECT_REMOVE = "DIRECT_REMOVE";
  80. private static final int KEY_DELETE_CODE = 101;
  81. @BindView(R.id.empty_list_view)
  82. public LinearLayout emptyContentContainer;
  83. @BindView(R.id.empty_list_view_text)
  84. public TextView emptyContentMessage;
  85. @BindView(R.id.empty_list_view_headline)
  86. public TextView emptyContentHeadline;
  87. @BindView(R.id.empty_list_icon)
  88. public ImageView emptyContentIcon;
  89. @BindView(R.id.user_info_view)
  90. public LinearLayout userInfoView;
  91. @BindView(R.id.user_icon)
  92. public ImageView avatar;
  93. @BindView(R.id.drawer_username)
  94. public TextView userName;
  95. @BindView(R.id.drawer_username_full)
  96. public TextView fullName;
  97. @BindView(R.id.phone_container)
  98. public View mPhoneNumberContainer;
  99. @BindView(R.id.phone_number)
  100. public TextView mPhoneNumberTextView;
  101. @BindView(R.id.phone_icon)
  102. public ImageView mPhoneNumberIcon;
  103. @BindView(R.id.email_container)
  104. public View mEmailContainer;
  105. @BindView(R.id.email_address)
  106. public TextView mEmailAddressTextView;
  107. @BindView(R.id.email_icon)
  108. public ImageView mEmailIcon;
  109. @BindView(R.id.address_container)
  110. public View mAddressContainer;
  111. @BindView(R.id.address)
  112. public TextView mAddressTextView;
  113. @BindView(R.id.address_icon)
  114. public ImageView mAddressIcon;
  115. @BindView(R.id.website_container)
  116. public View mWebsiteContainer;
  117. @BindView(R.id.website_address)
  118. public TextView mWebsiteTextView;
  119. @BindView(R.id.website_icon)
  120. public ImageView mWebsiteIcon;
  121. @BindView(R.id.twitter_container)
  122. public View mTwitterContainer;
  123. @BindView(R.id.twitter_handle)
  124. public TextView mTwitterHandleTextView;
  125. @BindView(R.id.twitter_icon)
  126. public ImageView mTwitterIcon;
  127. @BindView(R.id.empty_list_progress)
  128. public ProgressBar multiListProgressBar;
  129. @BindString(R.string.preview_sorry)
  130. public String sorryMessage;
  131. private float mCurrentAccountAvatarRadiusDimension;
  132. private Unbinder unbinder;
  133. private UserInfo userInfo;
  134. private Account account;
  135. @Override
  136. public void onCreate(Bundle savedInstanceState) {
  137. Log_OC.v(TAG, "onCreate() start");
  138. super.onCreate(savedInstanceState);
  139. Bundle bundle = getIntent().getExtras();
  140. account = Parcels.unwrap(bundle.getParcelable(KEY_ACCOUNT));
  141. if (savedInstanceState != null && savedInstanceState.containsKey(KEY_USER_DATA)) {
  142. userInfo = Parcels.unwrap(savedInstanceState.getParcelable(KEY_USER_DATA));
  143. }
  144. mCurrentAccountAvatarRadiusDimension = getResources().getDimension(R.dimen.nav_drawer_header_avatar_radius);
  145. setContentView(R.layout.user_info_layout);
  146. unbinder = ButterKnife.bind(this);
  147. setupToolbar();
  148. updateActionBarTitleAndHomeButtonByString("");
  149. setAccount(AccountUtils.getCurrentOwnCloudAccount(this));
  150. onAccountSet(false);
  151. if (userInfo != null) {
  152. populateUserInfoUi(userInfo);
  153. emptyContentContainer.setVisibility(View.GONE);
  154. userInfoView.setVisibility(View.VISIBLE);
  155. } else {
  156. setMultiListLoadingMessage();
  157. fetchAndSetData();
  158. }
  159. }
  160. @Override
  161. public boolean onCreateOptionsMenu(Menu menu) {
  162. MenuInflater inflater = getMenuInflater();
  163. inflater.inflate(R.menu.user_info_menu, menu);
  164. return true;
  165. }
  166. @Override
  167. public boolean onOptionsItemSelected(MenuItem item) {
  168. boolean retval = true;
  169. switch (item.getItemId()) {
  170. case android.R.id.home:
  171. onBackPressed();
  172. break;
  173. case R.id.change_password:
  174. changeAccountPassword(account);
  175. break;
  176. case R.id.delete_account:
  177. openAccountRemovalConfirmationDialog(account, getFragmentManager(), false);
  178. break;
  179. default:
  180. retval = super.onOptionsItemSelected(item);
  181. break;
  182. }
  183. return retval;
  184. }
  185. public void onDestroy() {
  186. super.onDestroy();
  187. unbinder.unbind();
  188. }
  189. private void setMultiListLoadingMessage() {
  190. if (emptyContentContainer != null) {
  191. emptyContentHeadline.setText(R.string.file_list_loading);
  192. emptyContentMessage.setText("");
  193. emptyContentIcon.setVisibility(View.GONE);
  194. multiListProgressBar.setVisibility(View.VISIBLE);
  195. }
  196. }
  197. private void setMessageForMultiList(String headline, String message) {
  198. if (emptyContentContainer != null && emptyContentMessage != null) {
  199. emptyContentHeadline.setText(headline);
  200. emptyContentMessage.setText(message);
  201. multiListProgressBar.setVisibility(View.GONE);
  202. }
  203. }
  204. private void populateUserInfoUi(UserInfo userInfo) {
  205. userName.setText(account.name);
  206. DisplayUtils.setAvatar(account, UserInfoActivity.this,
  207. mCurrentAccountAvatarRadiusDimension, getResources(), getStorageManager(),avatar);
  208. int tint = ContextCompat.getColor(this, R.color.primary);
  209. if (userInfo != null) {
  210. if (!TextUtils.isEmpty(userInfo.getDisplayName())) {
  211. fullName.setText(userInfo.getDisplayName());
  212. }
  213. populateUserInfoElement(mPhoneNumberContainer, mPhoneNumberTextView, userInfo.getPhone(),
  214. mPhoneNumberIcon, tint);
  215. populateUserInfoElement(mEmailContainer, mEmailAddressTextView, userInfo.getEmail(), mEmailIcon, tint);
  216. populateUserInfoElement(mAddressContainer, mAddressTextView, userInfo.getAddress(), mAddressIcon, tint);
  217. populateUserInfoElement(
  218. mWebsiteContainer,
  219. mWebsiteTextView,
  220. DisplayUtils.beautifyURL(userInfo.getWebpage()),
  221. mWebsiteIcon,
  222. tint);
  223. populateUserInfoElement(
  224. mTwitterContainer,
  225. mTwitterHandleTextView,
  226. DisplayUtils.beautifyTwitterHandle(userInfo.getTwitter()),
  227. mTwitterIcon,
  228. tint);
  229. }
  230. }
  231. private void populateUserInfoElement(View container, TextView textView, String text, ImageView icon, @ColorInt int
  232. tint) {
  233. if (!TextUtils.isEmpty(text)) {
  234. textView.setText(text);
  235. DrawableCompat.setTint(icon.getDrawable(), tint);
  236. } else {
  237. container.setVisibility(View.GONE);
  238. }
  239. }
  240. private void changeAccountPassword(Account account) {
  241. // let the user update credentials with one click
  242. Intent updateAccountCredentials = new Intent(this, AuthenticatorActivity.class);
  243. updateAccountCredentials.putExtra(AuthenticatorActivity.EXTRA_ACCOUNT, account);
  244. updateAccountCredentials.putExtra(AuthenticatorActivity.EXTRA_ACTION,
  245. AuthenticatorActivity.ACTION_UPDATE_TOKEN);
  246. startActivity(updateAccountCredentials);
  247. }
  248. public static void openAccountRemovalConfirmationDialog(Account account, FragmentManager fragmentManager,
  249. boolean removeDirectly) {
  250. UserInfoActivity.AccountRemovalConfirmationDialog dialog =
  251. UserInfoActivity.AccountRemovalConfirmationDialog.newInstance(account, removeDirectly);
  252. dialog.show(fragmentManager, "dialog");
  253. }
  254. public static class AccountRemovalConfirmationDialog extends DialogFragment {
  255. private Account account;
  256. public static UserInfoActivity.AccountRemovalConfirmationDialog newInstance(Account account,
  257. boolean removeDirectly) {
  258. Bundle bundle = new Bundle();
  259. bundle.putParcelable(KEY_ACCOUNT, account);
  260. bundle.putBoolean(KEY_DIRECT_REMOVE, removeDirectly);
  261. UserInfoActivity.AccountRemovalConfirmationDialog dialog = new
  262. UserInfoActivity.AccountRemovalConfirmationDialog();
  263. dialog.setArguments(bundle);
  264. return dialog;
  265. }
  266. @Override
  267. public void onCreate(@Nullable Bundle savedInstanceState) {
  268. super.onCreate(savedInstanceState);
  269. account = getArguments().getParcelable(KEY_ACCOUNT);
  270. }
  271. @NonNull
  272. @Override
  273. public Dialog onCreateDialog(Bundle savedInstanceState) {
  274. final boolean removeDirectly = getArguments().getBoolean(KEY_DIRECT_REMOVE);
  275. return new AlertDialog.Builder(getActivity(), R.style.Theme_ownCloud_Dialog)
  276. .setTitle(R.string.delete_account)
  277. .setMessage(getResources().getString(R.string.delete_account_warning, account.name))
  278. .setIcon(R.drawable.ic_warning)
  279. .setPositiveButton(R.string.common_ok,
  280. new DialogInterface.OnClickListener() {
  281. @Override
  282. public void onClick(DialogInterface dialogInterface, int i) {
  283. // remove contact backup job
  284. ContactsPreferenceActivity.cancelContactBackupJobForAccount(getActivity(), account);
  285. ContentResolver contentResolver = getActivity().getContentResolver();
  286. // delete all synced folder for an account
  287. SyncedFolderProvider syncedFolderProvider = new SyncedFolderProvider(
  288. contentResolver);
  289. syncedFolderProvider.deleteSyncFoldersForAccount(account);
  290. UploadsStorageManager uploadsStorageManager = new UploadsStorageManager(
  291. contentResolver, getActivity());
  292. uploadsStorageManager.cancelPendingAutoUploadJobsForAccount(account);
  293. uploadsStorageManager.removeAccountUploads(account);
  294. // disable daily backup
  295. ArbitraryDataProvider arbitraryDataProvider = new ArbitraryDataProvider(
  296. contentResolver);
  297. arbitraryDataProvider.storeOrUpdateKeyValue(account,
  298. ContactsPreferenceActivity.PREFERENCE_CONTACTS_AUTOMATIC_BACKUP,
  299. "false");
  300. String arbitraryDataPushString;
  301. if (!TextUtils.isEmpty(arbitraryDataPushString = arbitraryDataProvider.getValue(
  302. account, PushUtils.KEY_PUSH)) &&
  303. !TextUtils.isEmpty(getResources().getString(R.string.push_server_url))) {
  304. Gson gson = new Gson();
  305. PushConfigurationState pushArbitraryData = gson.fromJson(arbitraryDataPushString,
  306. PushConfigurationState.class);
  307. pushArbitraryData.setShouldBeDeleted(true);
  308. arbitraryDataProvider.storeOrUpdateKeyValue(account, PushUtils.KEY_PUSH,
  309. gson.toJson(pushArbitraryData));
  310. EventBus.getDefault().post(new TokenPushEvent());
  311. }
  312. if (getActivity() != null && !removeDirectly) {
  313. Bundle bundle = new Bundle();
  314. bundle.putParcelable(KEY_ACCOUNT, Parcels.wrap(account));
  315. Intent intent = new Intent();
  316. intent.putExtras(bundle);
  317. getActivity().setResult(KEY_DELETE_CODE, intent);
  318. getActivity().finish();
  319. } else {
  320. AccountManager am = (AccountManager) getActivity()
  321. .getSystemService(ACCOUNT_SERVICE);
  322. am.removeAccount(account, null, null);
  323. Intent start = new Intent(getActivity(), FileDisplayActivity.class);
  324. start.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
  325. startActivity(start);
  326. }
  327. }
  328. })
  329. .setNegativeButton(R.string.common_cancel, null)
  330. .create();
  331. }
  332. }
  333. private void fetchAndSetData() {
  334. Thread t = new Thread(new Runnable() {
  335. public void run() {
  336. RemoteOperation getRemoteUserInfoOperation = new GetRemoteUserInfoOperation();
  337. RemoteOperationResult result = getRemoteUserInfoOperation.execute(account, UserInfoActivity.this);
  338. if (result.isSuccess() && result.getData() != null) {
  339. userInfo = (UserInfo) result.getData().get(0);
  340. runOnUiThread(new Runnable() {
  341. @Override
  342. public void run() {
  343. populateUserInfoUi(userInfo);
  344. emptyContentContainer.setVisibility(View.GONE);
  345. userInfoView.setVisibility(View.VISIBLE);
  346. }
  347. });
  348. } else {
  349. // show error
  350. setMessageForMultiList(result.getLogMessage(), sorryMessage);
  351. Log_OC.d(TAG, result.getLogMessage());
  352. }
  353. }
  354. });
  355. t.start();
  356. }
  357. @Override
  358. protected void onSaveInstanceState(Bundle outState) {
  359. super.onSaveInstanceState(outState);
  360. if (userInfo != null) {
  361. outState.putParcelable(KEY_USER_DATA, Parcels.wrap(userInfo));
  362. }
  363. }
  364. @Subscribe(threadMode = ThreadMode.BACKGROUND)
  365. public void onMessageEvent(TokenPushEvent event) {
  366. PushUtils.pushRegistrationToServer();
  367. }
  368. }