ContactsPreferenceActivity.java 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380
  1. /**
  2. * Nextcloud Android client application
  3. *
  4. * @author Tobias Kaminsky
  5. * Copyright (C) 2017 Tobias Kaminsky
  6. * Copyright (C) 2017 Nextcloud GmbH.
  7. * <p>
  8. * This program is free software: you can redistribute it and/or modify
  9. * it under the terms of the GNU Affero General Public License as published by
  10. * the Free Software Foundation, either version 3 of the License, or
  11. * at your option) any later version.
  12. * <p>
  13. * This program is distributed in the hope that it will be useful,
  14. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  15. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  16. * GNU Affero General Public License for more details.
  17. * <p>
  18. * You should have received a copy of the GNU Affero General Public License
  19. * along with this program. If not, see <http://www.gnu.org/licenses/>.
  20. */
  21. package com.owncloud.android.ui.activity;
  22. import android.Manifest;
  23. import android.accounts.Account;
  24. import android.app.DatePickerDialog;
  25. import android.content.Context;
  26. import android.content.Intent;
  27. import android.os.Bundle;
  28. import android.support.annotation.NonNull;
  29. import android.support.design.widget.BottomNavigationView;
  30. import android.support.design.widget.Snackbar;
  31. import android.support.v4.app.Fragment;
  32. import android.support.v4.app.FragmentTransaction;
  33. import android.support.v7.widget.SwitchCompat;
  34. import android.view.MenuItem;
  35. import android.view.View;
  36. import android.widget.CompoundButton;
  37. import android.widget.DatePicker;
  38. import android.widget.TextView;
  39. import android.widget.Toast;
  40. import com.evernote.android.job.JobManager;
  41. import com.evernote.android.job.JobRequest;
  42. import com.evernote.android.job.util.support.PersistableBundleCompat;
  43. import com.owncloud.android.R;
  44. import com.owncloud.android.datamodel.ArbitraryDataProvider;
  45. import com.owncloud.android.datamodel.OCFile;
  46. import com.owncloud.android.lib.common.utils.Log_OC;
  47. import com.owncloud.android.services.ContactsBackupJob;
  48. import com.owncloud.android.ui.fragment.FileFragment;
  49. import com.owncloud.android.utils.DisplayUtils;
  50. import com.owncloud.android.utils.PermissionUtil;
  51. import java.util.Calendar;
  52. import java.util.Collections;
  53. import java.util.Comparator;
  54. import java.util.Set;
  55. import java.util.Vector;
  56. /**
  57. * This activity shows all settings for contact backup/restore
  58. */
  59. public class ContactsPreferenceActivity extends FileActivity implements FileFragment.ContainerActivity {
  60. public static final String TAG = ContactsPreferenceActivity.class.getSimpleName();
  61. public static final String PREFERENCE_CONTACTS_AUTOMATIC_BACKUP = "PREFERENCE_CONTACTS_AUTOMATIC_BACKUP";
  62. public static final String PREFERENCE_CONTACTS_LAST_BACKUP = "PREFERENCE_CONTACTS_LAST_BACKUP";
  63. private SwitchCompat backupSwitch;
  64. private ArbitraryDataProvider arbitraryDataProvider;
  65. @Override
  66. protected void onCreate(Bundle savedInstanceState) {
  67. super.onCreate(savedInstanceState);
  68. setContentView(R.layout.contacts_preference);
  69. // setup toolbar
  70. setupToolbar();
  71. // setup drawer
  72. setupDrawer(R.id.nav_contacts);
  73. getSupportActionBar().setTitle(R.string.actionbar_contacts);
  74. getSupportActionBar().setDisplayHomeAsUpEnabled(true);
  75. arbitraryDataProvider = new ArbitraryDataProvider(getContentResolver());
  76. backupSwitch = (SwitchCompat) findViewById(R.id.contacts_automatic_backup);
  77. backupSwitch.setChecked(arbitraryDataProvider.getValue(getAccount(), PREFERENCE_CONTACTS_AUTOMATIC_BACKUP)
  78. .equals("true"));
  79. backupSwitch.setOnCheckedChangeListener(new CompoundButton.OnCheckedChangeListener() {
  80. @Override
  81. public void onCheckedChanged(CompoundButton buttonView, boolean isChecked) {
  82. if (isChecked &&
  83. checkAndAskForContactsReadPermission(PermissionUtil.PERMISSIONS_READ_CONTACTS_AUTOMATIC)) {
  84. // store value
  85. setAutomaticBackup(backupSwitch, true);
  86. // enable daily job
  87. startContactBackupJob(getAccount());
  88. } else {
  89. setAutomaticBackup(backupSwitch, false);
  90. // cancel pending jobs
  91. cancelContactBackupJob(getBaseContext());
  92. }
  93. }
  94. });
  95. // display last backup
  96. TextView lastBackup = (TextView) findViewById(R.id.contacts_last_backup_timestamp);
  97. Long lastBackupTimestamp = arbitraryDataProvider.getLongValue(getAccount(), PREFERENCE_CONTACTS_LAST_BACKUP);
  98. if (lastBackupTimestamp == -1) {
  99. lastBackup.setText(R.string.contacts_preference_backup_never);
  100. } else {
  101. lastBackup.setText(DisplayUtils.getRelativeTimestamp(getBaseContext(), lastBackupTimestamp));
  102. }
  103. BottomNavigationView bottomNavigationView = (BottomNavigationView) findViewById(R.id.bottom_navigation_view);
  104. if (getResources().getBoolean(R.bool.bottom_toolbar_enabled)) {
  105. bottomNavigationView.setVisibility(View.VISIBLE);
  106. DisplayUtils.setupBottomBar(bottomNavigationView, getResources(), this, -1);
  107. }
  108. }
  109. @Override
  110. public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
  111. super.onRequestPermissionsResult(requestCode, permissions, grantResults);
  112. if (requestCode == PermissionUtil.PERMISSIONS_READ_CONTACTS_AUTOMATIC) {
  113. for (int index = 0; index < permissions.length; index++) {
  114. if (Manifest.permission.READ_CONTACTS.equalsIgnoreCase(permissions[index])) {
  115. if (grantResults[index] >= 0) {
  116. setAutomaticBackup(backupSwitch, true);
  117. } else {
  118. setAutomaticBackup(backupSwitch, false);
  119. }
  120. break;
  121. }
  122. }
  123. }
  124. if (requestCode == PermissionUtil.PERMISSIONS_READ_CONTACTS_MANUALLY) {
  125. for (int index = 0; index < permissions.length; index++) {
  126. if (Manifest.permission.READ_CONTACTS.equalsIgnoreCase(permissions[index])) {
  127. if (grantResults[index] >= 0) {
  128. startContactsBackupJob();
  129. }
  130. break;
  131. }
  132. }
  133. }
  134. }
  135. public void backupContacts(View v) {
  136. if (checkAndAskForContactsReadPermission(PermissionUtil.PERMISSIONS_READ_CONTACTS_MANUALLY)) {
  137. startContactsBackupJob();
  138. }
  139. }
  140. private void startContactsBackupJob() {
  141. PersistableBundleCompat bundle = new PersistableBundleCompat();
  142. bundle.putString(ContactsBackupJob.ACCOUNT, getAccount().name);
  143. bundle.putBoolean(ContactsBackupJob.FORCE, true);
  144. new JobRequest.Builder(ContactsBackupJob.TAG)
  145. .setExtras(bundle)
  146. .setExecutionWindow(3_000L, 10_000L)
  147. .setRequiresCharging(false)
  148. .setPersisted(false)
  149. .setUpdateCurrent(false)
  150. .build()
  151. .schedule();
  152. Snackbar.make(findViewById(R.id.contacts_linear_layout), R.string.contacts_preferences_backup_scheduled,
  153. Snackbar.LENGTH_LONG).show();
  154. }
  155. private void setAutomaticBackup(SwitchCompat backupSwitch, boolean bool) {
  156. backupSwitch.setChecked(bool);
  157. arbitraryDataProvider.storeOrUpdateKeyValue(getAccount(),
  158. PREFERENCE_CONTACTS_AUTOMATIC_BACKUP, String.valueOf(bool));
  159. }
  160. private boolean checkAndAskForContactsReadPermission(final int permission) {
  161. // check permissions
  162. if ((PermissionUtil.checkSelfPermission(this, Manifest.permission.READ_CONTACTS))) {
  163. return true;
  164. } else {
  165. // Check if we should show an explanation
  166. if (PermissionUtil.shouldShowRequestPermissionRationale(ContactsPreferenceActivity.this,
  167. android.Manifest.permission.READ_CONTACTS)) {
  168. // Show explanation to the user and then request permission
  169. Snackbar snackbar = Snackbar.make(findViewById(R.id.contacts_linear_layout),
  170. R.string.contacts_read_permission, Snackbar.LENGTH_INDEFINITE)
  171. .setAction(R.string.common_ok, new View.OnClickListener() {
  172. @Override
  173. public void onClick(View v) {
  174. PermissionUtil.requestReadContactPermission(ContactsPreferenceActivity.this,
  175. permission);
  176. }
  177. });
  178. DisplayUtils.colorSnackbar(this, snackbar);
  179. snackbar.show();
  180. return false;
  181. } else {
  182. // No explanation needed, request the permission.
  183. PermissionUtil.requestReadContactPermission(ContactsPreferenceActivity.this, permission);
  184. return false;
  185. }
  186. }
  187. }
  188. public void openDate(View v) {
  189. String backupFolderString = getResources().getString(R.string.contacts_backup_folder) + OCFile.PATH_SEPARATOR;
  190. OCFile backupFolder = getStorageManager().getFileByPath(backupFolderString);
  191. Vector<OCFile> backupFiles = getStorageManager().getFolderContent(backupFolder, false);
  192. Collections.sort(backupFiles, new Comparator<OCFile>() {
  193. @Override
  194. public int compare(OCFile o1, OCFile o2) {
  195. if (o1.getModificationTimestamp() == o2.getModificationTimestamp()) {
  196. return 0;
  197. }
  198. if (o1.getModificationTimestamp() > o2.getModificationTimestamp()) {
  199. return 1;
  200. } else {
  201. return -1;
  202. }
  203. }
  204. });
  205. Calendar cal = Calendar.getInstance();
  206. int year = cal.get(Calendar.YEAR);
  207. int month = cal.get(Calendar.MONTH) + 1;
  208. int day = cal.get(Calendar.DAY_OF_MONTH);
  209. DatePickerDialog.OnDateSetListener dateSetListener = new DatePickerDialog.OnDateSetListener() {
  210. @Override
  211. public void onDateSet(DatePicker view, int year, int month, int dayOfMonth) {
  212. String backupFolderString = getResources().getString(R.string.contacts_backup_folder) + OCFile.PATH_SEPARATOR;
  213. OCFile backupFolder = getStorageManager().getFileByPath(backupFolderString);
  214. Vector<OCFile> backupFiles = getStorageManager().getFolderContent(backupFolder, false);
  215. // find file with modification with date and time between 00:00 and 23:59
  216. // if more than one file exists, take oldest
  217. Calendar date = Calendar.getInstance();
  218. date.set(year, month, dayOfMonth);
  219. // start
  220. date.set(Calendar.HOUR, 0);
  221. date.set(Calendar.MINUTE, 0);
  222. date.set(Calendar.SECOND, 1);
  223. date.set(Calendar.MILLISECOND, 0);
  224. date.set(Calendar.AM_PM, Calendar.AM);
  225. Long start = date.getTimeInMillis();
  226. // end
  227. date.set(Calendar.HOUR, 23);
  228. date.set(Calendar.MINUTE, 59);
  229. date.set(Calendar.SECOND, 59);
  230. Long end = date.getTimeInMillis();
  231. OCFile backupToRestore = null;
  232. for (OCFile file : backupFiles) {
  233. if (start < file.getModificationTimestamp() && end > file.getModificationTimestamp()) {
  234. if (backupToRestore == null) {
  235. backupToRestore = file;
  236. } else if (backupToRestore.getModificationTimestamp() < file.getModificationTimestamp()) {
  237. backupToRestore = file;
  238. }
  239. }
  240. }
  241. if (backupToRestore != null) {
  242. Fragment contactListFragment = ContactListFragment.newInstance(backupToRestore, getAccount());
  243. FragmentTransaction transaction = getSupportFragmentManager().beginTransaction();
  244. transaction.replace(R.id.contacts_linear_layout, contactListFragment);
  245. transaction.commit();
  246. } else {
  247. Toast.makeText(ContactsPreferenceActivity.this, R.string.contacts_preferences_no_file_found,
  248. Toast.LENGTH_SHORT).show();
  249. }
  250. }
  251. };
  252. DatePickerDialog datePickerDialog = new DatePickerDialog(this, dateSetListener, year, month, day);
  253. datePickerDialog.getDatePicker().setMaxDate(backupFiles.lastElement().getModificationTimestamp());
  254. datePickerDialog.getDatePicker().setMinDate(backupFiles.firstElement().getModificationTimestamp());
  255. datePickerDialog.show();
  256. }
  257. public static void startContactBackupJob(Account account) {
  258. Log_OC.d(TAG, "start daily contacts backup job");
  259. PersistableBundleCompat bundle = new PersistableBundleCompat();
  260. bundle.putString(ContactsBackupJob.ACCOUNT, account.name);
  261. new JobRequest.Builder(ContactsBackupJob.TAG)
  262. .setExtras(bundle)
  263. .setRequiresCharging(false)
  264. .setPersisted(true)
  265. .setUpdateCurrent(true)
  266. .setPeriodic(24 * 60 * 60 * 1000)
  267. .build()
  268. .schedule();
  269. }
  270. public static void cancelContactBackupJob(Context context) {
  271. Log_OC.d(TAG, "disabling contacts backup job");
  272. JobManager jobManager = JobManager.create(context);
  273. Set<JobRequest> jobs = jobManager.getAllJobRequestsForTag(ContactsBackupJob.TAG);
  274. for (JobRequest jobRequest : jobs) {
  275. jobManager.cancel(jobRequest.getJobId());
  276. }
  277. }
  278. @Override
  279. public boolean onOptionsItemSelected(MenuItem item) {
  280. boolean retval;
  281. switch (item.getItemId()) {
  282. case android.R.id.home:
  283. if (isDrawerOpen()) {
  284. closeDrawer();
  285. } else {
  286. openDrawer();
  287. }
  288. retval = true;
  289. break;
  290. default:
  291. retval = super.onOptionsItemSelected(item);
  292. break;
  293. }
  294. return retval;
  295. }
  296. @Override
  297. public void showFiles(boolean onDeviceOnly) {
  298. super.showFiles(onDeviceOnly);
  299. Intent fileDisplayActivity = new Intent(getApplicationContext(), FileDisplayActivity.class);
  300. fileDisplayActivity.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
  301. startActivity(fileDisplayActivity);
  302. }
  303. @Override
  304. public void showDetails(OCFile file) {
  305. // not needed
  306. }
  307. @Override
  308. public void onBrowsedDownTo(OCFile folder) {
  309. // not needed
  310. }
  311. @Override
  312. public void onTransferStateChanged(OCFile file, boolean downloading, boolean uploading) {
  313. // not needed
  314. }
  315. }