SyncedFoldersActivity.java 32 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764
  1. /*
  2. * Nextcloud Android client application
  3. *
  4. * @author Andy Scherzinger
  5. * Copyright (C) 2016 Andy Scherzinger
  6. * Copyright (C) 2016 Nextcloud
  7. *
  8. * This program is free software; you can redistribute it and/or
  9. * modify it under the terms of the GNU AFFERO GENERAL PUBLIC LICENSE
  10. * License as published by the Free Software Foundation; either
  11. * version 3 of the License, or any later version.
  12. *
  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. *
  18. * You should have received a copy of the GNU Affero General Public
  19. * License along with this program. If not, see <http://www.gnu.org/licenses/>.
  20. */
  21. package com.owncloud.android.ui.activity;
  22. import android.accounts.Account;
  23. import android.annotation.SuppressLint;
  24. import android.app.Activity;
  25. import android.app.NotificationManager;
  26. import android.content.Context;
  27. import android.content.Intent;
  28. import android.content.pm.PackageManager;
  29. import android.net.Uri;
  30. import android.os.Build;
  31. import android.os.Bundle;
  32. import android.os.PowerManager;
  33. import android.text.TextUtils;
  34. import android.util.Log;
  35. import android.view.Menu;
  36. import android.view.MenuInflater;
  37. import android.view.MenuItem;
  38. import android.view.View;
  39. import android.widget.LinearLayout;
  40. import android.widget.TextView;
  41. import com.google.android.material.bottomnavigation.BottomNavigationView;
  42. import com.nextcloud.client.di.Injectable;
  43. import com.nextcloud.client.preferences.AppPreferences;
  44. import com.owncloud.android.BuildConfig;
  45. import com.owncloud.android.MainApp;
  46. import com.owncloud.android.R;
  47. import com.owncloud.android.datamodel.ArbitraryDataProvider;
  48. import com.owncloud.android.datamodel.MediaFolder;
  49. import com.owncloud.android.datamodel.MediaFolderType;
  50. import com.owncloud.android.datamodel.MediaProvider;
  51. import com.owncloud.android.datamodel.OCFile;
  52. import com.owncloud.android.datamodel.SyncedFolder;
  53. import com.owncloud.android.datamodel.SyncedFolderDisplayItem;
  54. import com.owncloud.android.datamodel.SyncedFolderProvider;
  55. import com.owncloud.android.files.services.FileUploader;
  56. import com.owncloud.android.jobs.MediaFoldersDetectionJob;
  57. import com.owncloud.android.jobs.NotificationJob;
  58. import com.owncloud.android.ui.adapter.SyncedFolderAdapter;
  59. import com.owncloud.android.ui.decoration.MediaGridItemDecoration;
  60. import com.owncloud.android.ui.dialog.SyncedFolderPreferencesDialogFragment;
  61. import com.owncloud.android.ui.dialog.parcel.SyncedFolderParcelable;
  62. import com.owncloud.android.utils.DisplayUtils;
  63. import com.owncloud.android.utils.FilesSyncHelper;
  64. import com.owncloud.android.utils.PermissionUtil;
  65. import com.owncloud.android.utils.ThemeUtils;
  66. import java.io.File;
  67. import java.util.ArrayList;
  68. import java.util.Arrays;
  69. import java.util.Collections;
  70. import java.util.HashMap;
  71. import java.util.List;
  72. import java.util.Locale;
  73. import java.util.Map;
  74. import javax.inject.Inject;
  75. import androidx.annotation.NonNull;
  76. import androidx.appcompat.app.ActionBar;
  77. import androidx.appcompat.app.AlertDialog;
  78. import androidx.drawerlayout.widget.DrawerLayout;
  79. import androidx.fragment.app.FragmentManager;
  80. import androidx.fragment.app.FragmentTransaction;
  81. import androidx.recyclerview.widget.GridLayoutManager;
  82. import androidx.recyclerview.widget.RecyclerView;
  83. import static android.provider.Settings.ACTION_REQUEST_IGNORE_BATTERY_OPTIMIZATIONS;
  84. import static com.owncloud.android.datamodel.SyncedFolderDisplayItem.UNPERSISTED_ID;
  85. /**
  86. * Activity displaying all auto-synced folders and/or instant upload media folders.
  87. */
  88. public class SyncedFoldersActivity extends FileActivity implements SyncedFolderAdapter.ClickListener,
  89. SyncedFolderPreferencesDialogFragment.OnSyncedFolderPreferenceListener, Injectable {
  90. private static final String[] PRIORITIZED_FOLDERS = new String[]{"Camera", "Screenshots"};
  91. private static final List<String> SPECIAL_MANUFACTURER = Arrays.asList("Samsung", "Huawei", "Xiaomi");
  92. public static final String EXTRA_SHOW_SIDEBAR = "SHOW_SIDEBAR";
  93. private static final String SYNCED_FOLDER_PREFERENCES_DIALOG_TAG = "SYNCED_FOLDER_PREFERENCES_DIALOG";
  94. private static final String TAG = SyncedFoldersActivity.class.getSimpleName();
  95. private RecyclerView mRecyclerView;
  96. private SyncedFolderAdapter mAdapter;
  97. private LinearLayout mProgress;
  98. private TextView mEmpty;
  99. private SyncedFolderProvider mSyncedFolderProvider;
  100. private SyncedFolderPreferencesDialogFragment mSyncedFolderPreferencesDialogFragment;
  101. private boolean showSidebar = true;
  102. private String path;
  103. private int type;
  104. @Inject AppPreferences preferences;
  105. @Override
  106. protected void onCreate(Bundle savedInstanceState) {
  107. super.onCreate(savedInstanceState);
  108. if (getIntent().getExtras() != null) {
  109. showSidebar = getIntent().getExtras().getBoolean(EXTRA_SHOW_SIDEBAR);
  110. }
  111. setContentView(R.layout.synced_folders_layout);
  112. String account;
  113. Account currentAccount;
  114. if (getIntent() != null && getIntent().getExtras() != null) {
  115. account = getIntent().getExtras().getString(NotificationJob.KEY_NOTIFICATION_ACCOUNT);
  116. currentAccount = getAccount();
  117. if (account != null && currentAccount != null && !account.equalsIgnoreCase(currentAccount.name)) {
  118. accountManager.setCurrentOwnCloudAccount(account);
  119. setAccount(getUserAccountManager().getCurrentAccount());
  120. }
  121. path = getIntent().getStringExtra(MediaFoldersDetectionJob.KEY_MEDIA_FOLDER_PATH);
  122. type = getIntent().getIntExtra(MediaFoldersDetectionJob.KEY_MEDIA_FOLDER_TYPE, -1);
  123. // Cancel notification
  124. int notificationId = getIntent().getIntExtra(MediaFoldersDetectionJob.NOTIFICATION_ID, 0);
  125. NotificationManager notificationManager =
  126. (NotificationManager) getSystemService(Activity.NOTIFICATION_SERVICE);
  127. notificationManager.cancel(notificationId);
  128. }
  129. // setup toolbar
  130. setupToolbar();
  131. if (getSupportActionBar() != null){
  132. getSupportActionBar().setTitle(R.string.drawer_synced_folders);
  133. }
  134. // setup drawer
  135. setupDrawer(R.id.nav_synced_folders);
  136. if (!showSidebar) {
  137. setDrawerLockMode(DrawerLayout.LOCK_MODE_LOCKED_CLOSED);
  138. mDrawerToggle.setDrawerIndicatorEnabled(false);
  139. }
  140. setupContent();
  141. ActionBar actionBar = getSupportActionBar();
  142. if (actionBar != null) {
  143. ThemeUtils.setColoredTitle(getSupportActionBar(), getString(R.string.drawer_synced_folders), this);
  144. actionBar.setDisplayHomeAsUpEnabled(true);
  145. }
  146. if (ThemeUtils.themingEnabled(this)) {
  147. setTheme(R.style.FallbackThemingTheme);
  148. }
  149. }
  150. @Override
  151. public boolean onCreateOptionsMenu(Menu menu) {
  152. MenuInflater inflater = getMenuInflater();
  153. inflater.inflate(R.menu.synced_folders_menu, menu);
  154. return true;
  155. }
  156. /**
  157. * sets up the UI elements and loads all media/synced folders.
  158. */
  159. private void setupContent() {
  160. mRecyclerView = findViewById(android.R.id.list);
  161. mProgress = findViewById(android.R.id.progress);
  162. mEmpty = findViewById(android.R.id.empty);
  163. final int gridWidth = getResources().getInteger(R.integer.media_grid_width);
  164. boolean lightVersion = getResources().getBoolean(R.bool.syncedFolder_light);
  165. mAdapter = new SyncedFolderAdapter(this, gridWidth, this, lightVersion);
  166. mSyncedFolderProvider = new SyncedFolderProvider(getContentResolver(), preferences);
  167. final GridLayoutManager lm = new GridLayoutManager(this, gridWidth);
  168. mAdapter.setLayoutManager(lm);
  169. int spacing = getResources().getDimensionPixelSize(R.dimen.media_grid_spacing);
  170. mRecyclerView.addItemDecoration(new MediaGridItemDecoration(spacing));
  171. mRecyclerView.setLayoutManager(lm);
  172. mRecyclerView.setAdapter(mAdapter);
  173. BottomNavigationView bottomNavigationView = findViewById(R.id.bottom_navigation_view);
  174. if (getResources().getBoolean(R.bool.bottom_toolbar_enabled)) {
  175. bottomNavigationView.setVisibility(View.VISIBLE);
  176. DisplayUtils.setupBottomBar(
  177. getUserAccountManager().getCurrentAccount(),
  178. bottomNavigationView,
  179. getResources(),
  180. accountManager,
  181. this,
  182. -1
  183. );
  184. }
  185. load(gridWidth * 2, false);
  186. }
  187. /**
  188. * loads all media/synced folders, adds them to the recycler view adapter and shows the list.
  189. *
  190. * @param perFolderMediaItemLimit the amount of media items to be loaded/shown per media folder
  191. */
  192. private void load(final int perFolderMediaItemLimit, boolean force) {
  193. if (mAdapter.getItemCount() > 0 && !force) {
  194. return;
  195. }
  196. setListShown(false);
  197. final List<MediaFolder> mediaFolders = MediaProvider.getImageFolders(getContentResolver(),
  198. perFolderMediaItemLimit, this, false);
  199. mediaFolders.addAll(MediaProvider.getVideoFolders(getContentResolver(), perFolderMediaItemLimit,
  200. this, false));
  201. List<SyncedFolder> syncedFolderArrayList = mSyncedFolderProvider.getSyncedFolders();
  202. List<SyncedFolder> currentAccountSyncedFoldersList = new ArrayList<>();
  203. Account currentAccount = getUserAccountManager().getCurrentAccount();
  204. for (SyncedFolder syncedFolder : syncedFolderArrayList) {
  205. if (currentAccount != null && syncedFolder.getAccount().equals(currentAccount.name)) {
  206. // delete non-existing & disabled synced folders
  207. if (!new File(syncedFolder.getLocalPath()).exists() && !syncedFolder.isEnabled()) {
  208. mSyncedFolderProvider.deleteSyncedFolder(syncedFolder.getId());
  209. } else {
  210. currentAccountSyncedFoldersList.add(syncedFolder);
  211. }
  212. }
  213. }
  214. List<SyncedFolderDisplayItem> syncFolderItems = sortSyncedFolderItems(
  215. mergeFolderData(currentAccountSyncedFoldersList, mediaFolders));
  216. mAdapter.setSyncFolderItems(syncFolderItems);
  217. mAdapter.notifyDataSetChanged();
  218. setListShown(true);
  219. if (!TextUtils.isEmpty(path)) {
  220. int section = mAdapter.getSectionByLocalPathAndType(path, type);
  221. if (section >= 0) {
  222. onSyncFolderSettingsClick(section, mAdapter.get(section));
  223. }
  224. }
  225. }
  226. /**
  227. * Sorts list of {@link SyncedFolderDisplayItem}s.
  228. *
  229. * @param syncFolderItemList list of items to be sorted
  230. * @return sorted list of items
  231. */
  232. public static List<SyncedFolderDisplayItem> sortSyncedFolderItems(List<SyncedFolderDisplayItem>
  233. syncFolderItemList) {
  234. Collections.sort(syncFolderItemList, (f1, f2) -> {
  235. if (f1 == null && f2 == null) {
  236. return 0;
  237. } else if (f1 == null) {
  238. return -1;
  239. } else if (f2 == null) {
  240. return 1;
  241. } else if (f1.isEnabled() && f2.isEnabled()) {
  242. return f1.getFolderName().toLowerCase(Locale.getDefault()).compareTo(
  243. f2.getFolderName().toLowerCase(Locale.getDefault()));
  244. } else if (f1.isEnabled()) {
  245. return -1;
  246. } else if (f2.isEnabled()) {
  247. return 1;
  248. } else if (f1.getFolderName() == null && f2.getFolderName() == null) {
  249. return 0;
  250. } else if (f1.getFolderName() == null) {
  251. return -1;
  252. } else if (f2.getFolderName() == null) {
  253. return 1;
  254. }
  255. for (String folder : PRIORITIZED_FOLDERS) {
  256. if (folder.equals(f1.getFolderName()) && folder.equals(f2.getFolderName())) {
  257. return 0;
  258. } else if (folder.equals(f1.getFolderName())) {
  259. return -1;
  260. } else if (folder.equals(f2.getFolderName())) {
  261. return 1;
  262. }
  263. }
  264. return f1.getFolderName().toLowerCase(Locale.getDefault()).compareTo(
  265. f2.getFolderName().toLowerCase(Locale.getDefault()));
  266. });
  267. return syncFolderItemList;
  268. }
  269. /**
  270. * merges two lists of {@link SyncedFolder} and {@link MediaFolder} items into one of SyncedFolderItems.
  271. *
  272. * @param syncedFolders the synced folders
  273. * @param mediaFolders the media folders
  274. * @return the merged list of SyncedFolderItems
  275. */
  276. @NonNull
  277. private List<SyncedFolderDisplayItem> mergeFolderData(List<SyncedFolder> syncedFolders,
  278. @NonNull List<MediaFolder> mediaFolders) {
  279. Map<String, SyncedFolder> syncedFoldersMap = createSyncedFoldersMap(syncedFolders);
  280. List<SyncedFolderDisplayItem> result = new ArrayList<>();
  281. for (MediaFolder mediaFolder : mediaFolders) {
  282. if (syncedFoldersMap.containsKey(mediaFolder.absolutePath + "-" + mediaFolder.type)) {
  283. SyncedFolder syncedFolder = syncedFoldersMap.get(mediaFolder.absolutePath + "-" + mediaFolder.type);
  284. syncedFoldersMap.remove(mediaFolder.absolutePath + "-" + mediaFolder.type);
  285. if (MediaFolderType.CUSTOM == syncedFolder.getType()) {
  286. result.add(createSyncedFolderWithoutMediaFolder(syncedFolder));
  287. } else {
  288. result.add(createSyncedFolder(syncedFolder, mediaFolder));
  289. }
  290. } else {
  291. result.add(createSyncedFolderFromMediaFolder(mediaFolder));
  292. }
  293. }
  294. for (SyncedFolder syncedFolder : syncedFoldersMap.values()) {
  295. result.add(createSyncedFolderWithoutMediaFolder(syncedFolder));
  296. }
  297. return result;
  298. }
  299. @NonNull
  300. private SyncedFolderDisplayItem createSyncedFolderWithoutMediaFolder(@NonNull SyncedFolder syncedFolder) {
  301. File localFolder = new File(syncedFolder.getLocalPath());
  302. File[] files = getFileList(localFolder);
  303. List<String> filePaths = getDisplayFilePathList(files);
  304. return new SyncedFolderDisplayItem(
  305. syncedFolder.getId(),
  306. syncedFolder.getLocalPath(),
  307. syncedFolder.getRemotePath(),
  308. syncedFolder.getWifiOnly(),
  309. syncedFolder.getChargingOnly(),
  310. syncedFolder.getSubfolderByDate(),
  311. syncedFolder.getAccount(),
  312. syncedFolder.getUploadAction(),
  313. syncedFolder.isEnabled(),
  314. filePaths,
  315. localFolder.getName(),
  316. files.length,
  317. syncedFolder.getType());
  318. }
  319. /**
  320. * creates a SyncedFolderDisplayItem merging a {@link SyncedFolder} and a {@link MediaFolder} object instance.
  321. *
  322. * @param syncedFolder the synced folder object
  323. * @param mediaFolder the media folder object
  324. * @return the created SyncedFolderDisplayItem
  325. */
  326. @NonNull
  327. private SyncedFolderDisplayItem createSyncedFolder(@NonNull SyncedFolder syncedFolder, @NonNull MediaFolder mediaFolder) {
  328. return new SyncedFolderDisplayItem(
  329. syncedFolder.getId(),
  330. syncedFolder.getLocalPath(),
  331. syncedFolder.getRemotePath(),
  332. syncedFolder.getWifiOnly(),
  333. syncedFolder.getChargingOnly(),
  334. syncedFolder.getSubfolderByDate(),
  335. syncedFolder.getAccount(),
  336. syncedFolder.getUploadAction(),
  337. syncedFolder.isEnabled(),
  338. mediaFolder.filePaths,
  339. mediaFolder.folderName,
  340. mediaFolder.numberOfFiles,
  341. mediaFolder.type);
  342. }
  343. /**
  344. * creates a {@link SyncedFolderDisplayItem} based on a {@link MediaFolder} object instance.
  345. *
  346. * @param mediaFolder the media folder object
  347. * @return the created SyncedFolderDisplayItem
  348. */
  349. @NonNull
  350. private SyncedFolderDisplayItem createSyncedFolderFromMediaFolder(@NonNull MediaFolder mediaFolder) {
  351. return new SyncedFolderDisplayItem(
  352. UNPERSISTED_ID,
  353. mediaFolder.absolutePath,
  354. getString(R.string.instant_upload_path) + "/" + mediaFolder.folderName,
  355. true,
  356. false,
  357. false,
  358. getAccount().name,
  359. FileUploader.LOCAL_BEHAVIOUR_FORGET,
  360. false,
  361. mediaFolder.filePaths,
  362. mediaFolder.folderName,
  363. mediaFolder.numberOfFiles,
  364. mediaFolder.type);
  365. }
  366. private File[] getFileList(File localFolder) {
  367. File[] files = localFolder.listFiles(pathname -> !pathname.isDirectory());
  368. if (files != null) {
  369. Arrays.sort(files, (f1, f2) -> Long.compare(f1.lastModified(), f2.lastModified()));
  370. } else {
  371. files = new File[]{};
  372. }
  373. return files;
  374. }
  375. private List<String> getDisplayFilePathList(File... files) {
  376. List<String> filePaths = null;
  377. if (files != null && files.length > 0) {
  378. filePaths = new ArrayList<>();
  379. for (int i = 0; i < 7 && i < files.length; i++) {
  380. filePaths.add(files[i].getAbsolutePath());
  381. }
  382. }
  383. return filePaths;
  384. }
  385. /**
  386. * creates a lookup map for a list of given {@link SyncedFolder}s with their local path as the key.
  387. *
  388. * @param syncFolders list of {@link SyncedFolder}s
  389. * @return the lookup map for {@link SyncedFolder}s
  390. */
  391. @NonNull
  392. private Map<String, SyncedFolder> createSyncedFoldersMap(List<SyncedFolder> syncFolders) {
  393. Map<String, SyncedFolder> result = new HashMap<>();
  394. if (syncFolders != null) {
  395. for (SyncedFolder syncFolder : syncFolders) {
  396. result.put(syncFolder.getLocalPath() + "-" + syncFolder.getType(), syncFolder);
  397. }
  398. }
  399. return result;
  400. }
  401. /**
  402. * show/hide recycler view list or the empty message / progress info.
  403. *
  404. * @param shown flag if list should be shown
  405. */
  406. private void setListShown(boolean shown) {
  407. if (mRecyclerView != null) {
  408. mRecyclerView.setVisibility(shown ? View.VISIBLE : View.GONE);
  409. mProgress.setVisibility(shown ? View.GONE : View.VISIBLE);
  410. mEmpty.setVisibility(shown && mAdapter.getItemCount() == 0 ? View.VISIBLE : View.GONE);
  411. }
  412. }
  413. @Override
  414. public boolean onOptionsItemSelected(MenuItem item) {
  415. boolean result = true;
  416. switch (item.getItemId()) {
  417. case android.R.id.home: {
  418. if (showSidebar) {
  419. if (isDrawerOpen()) {
  420. closeDrawer();
  421. } else {
  422. openDrawer();
  423. }
  424. } else {
  425. Intent settingsIntent = new Intent(getApplicationContext(), SettingsActivity.class);
  426. startActivity(settingsIntent);
  427. }
  428. break;
  429. }
  430. case R.id.action_create_custom_folder: {
  431. Log.d(TAG, "Show custom folder dialog");
  432. SyncedFolderDisplayItem emptyCustomFolder = new SyncedFolderDisplayItem(
  433. SyncedFolder.UNPERSISTED_ID, null, null, true, false,
  434. false, getAccount().name,
  435. FileUploader.LOCAL_BEHAVIOUR_FORGET, false, null, MediaFolderType.CUSTOM);
  436. onSyncFolderSettingsClick(0, emptyCustomFolder);
  437. }
  438. default:
  439. result = super.onOptionsItemSelected(item);
  440. break;
  441. }
  442. return result;
  443. }
  444. @Override
  445. public void restart() {
  446. Intent i = new Intent(this, FileDisplayActivity.class);
  447. i.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
  448. startActivity(i);
  449. }
  450. @Override
  451. public void showFiles(boolean onDeviceOnly) {
  452. MainApp.showOnlyFilesOnDevice(onDeviceOnly);
  453. Intent fileDisplayActivity = new Intent(getApplicationContext(), FileDisplayActivity.class);
  454. fileDisplayActivity.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
  455. startActivity(fileDisplayActivity);
  456. }
  457. @Override
  458. public void onSyncStatusToggleClick(int section, SyncedFolderDisplayItem syncedFolderDisplayItem) {
  459. ArbitraryDataProvider arbitraryDataProvider = new ArbitraryDataProvider(MainApp.getAppContext().
  460. getContentResolver());
  461. if (syncedFolderDisplayItem.getId() > UNPERSISTED_ID) {
  462. mSyncedFolderProvider.updateSyncedFolderEnabled(syncedFolderDisplayItem.getId(),
  463. syncedFolderDisplayItem.isEnabled());
  464. } else {
  465. long storedId = mSyncedFolderProvider.storeSyncedFolder(syncedFolderDisplayItem);
  466. if (storedId != -1) {
  467. syncedFolderDisplayItem.setId(storedId);
  468. }
  469. }
  470. if (syncedFolderDisplayItem.isEnabled()) {
  471. FilesSyncHelper.insertAllDBEntriesForSyncedFolder(syncedFolderDisplayItem);
  472. showBatteryOptimizationInfo();
  473. } else {
  474. String syncedFolderInitiatedKey = "syncedFolderIntitiated_" + syncedFolderDisplayItem.getId();
  475. arbitraryDataProvider.deleteKeyForAccount("global", syncedFolderInitiatedKey);
  476. }
  477. }
  478. @Override
  479. public void onSyncFolderSettingsClick(int section, SyncedFolderDisplayItem syncedFolderDisplayItem) {
  480. FragmentManager fm = getSupportFragmentManager();
  481. FragmentTransaction ft = fm.beginTransaction();
  482. ft.addToBackStack(null);
  483. mSyncedFolderPreferencesDialogFragment = SyncedFolderPreferencesDialogFragment.newInstance(
  484. syncedFolderDisplayItem, section);
  485. mSyncedFolderPreferencesDialogFragment.show(ft, SYNCED_FOLDER_PREFERENCES_DIALOG_TAG);
  486. }
  487. @Override
  488. protected void onActivityResult(int requestCode, int resultCode, Intent data) {
  489. if (requestCode == SyncedFolderPreferencesDialogFragment.REQUEST_CODE__SELECT_REMOTE_FOLDER
  490. && resultCode == RESULT_OK && mSyncedFolderPreferencesDialogFragment != null) {
  491. OCFile chosenFolder = data.getParcelableExtra(FolderPickerActivity.EXTRA_FOLDER);
  492. mSyncedFolderPreferencesDialogFragment.setRemoteFolderSummary(chosenFolder.getRemotePath());
  493. }
  494. if (requestCode == SyncedFolderPreferencesDialogFragment.REQUEST_CODE__SELECT_LOCAL_FOLDER
  495. && resultCode == RESULT_OK && mSyncedFolderPreferencesDialogFragment != null) {
  496. String localPath = data.getStringExtra(UploadFilesActivity.EXTRA_CHOSEN_FILES);
  497. mSyncedFolderPreferencesDialogFragment.setLocalFolderSummary(localPath);
  498. } else {
  499. super.onActivityResult(requestCode, resultCode, data);
  500. }
  501. }
  502. @Override
  503. public void onSaveSyncedFolderPreference(SyncedFolderParcelable syncedFolder) {
  504. ArbitraryDataProvider arbitraryDataProvider = new ArbitraryDataProvider(MainApp.getAppContext().
  505. getContentResolver());
  506. // custom folders newly created aren't in the list already,
  507. // so triggering a refresh
  508. if (MediaFolderType.CUSTOM == syncedFolder.getType() && syncedFolder.getId() == UNPERSISTED_ID) {
  509. SyncedFolderDisplayItem newCustomFolder = new SyncedFolderDisplayItem(
  510. SyncedFolder.UNPERSISTED_ID, syncedFolder.getLocalPath(), syncedFolder.getRemotePath(),
  511. syncedFolder.getWifiOnly(), syncedFolder.getChargingOnly(), syncedFolder.getSubfolderByDate(),
  512. syncedFolder.getAccount(), syncedFolder.getUploadAction(), syncedFolder.getEnabled(),
  513. new File(syncedFolder.getLocalPath()).getName(), syncedFolder.getType());
  514. long storedId = mSyncedFolderProvider.storeSyncedFolder(newCustomFolder);
  515. if (storedId != -1) {
  516. newCustomFolder.setId(storedId);
  517. if (newCustomFolder.isEnabled()) {
  518. FilesSyncHelper.insertAllDBEntriesForSyncedFolder(newCustomFolder);
  519. } else {
  520. String syncedFolderInitiatedKey = "syncedFolderIntitiated_" + newCustomFolder.getId();
  521. arbitraryDataProvider.deleteKeyForAccount("global", syncedFolderInitiatedKey);
  522. }
  523. }
  524. mAdapter.addSyncFolderItem(newCustomFolder);
  525. } else {
  526. SyncedFolderDisplayItem item = mAdapter.get(syncedFolder.getSection());
  527. item = updateSyncedFolderItem(item, syncedFolder.getLocalPath(), syncedFolder.getRemotePath(), syncedFolder
  528. .getWifiOnly(), syncedFolder.getChargingOnly(), syncedFolder.getSubfolderByDate(), syncedFolder
  529. .getUploadAction(), syncedFolder.getEnabled());
  530. if (syncedFolder.getId() == UNPERSISTED_ID) {
  531. // newly set up folder sync config
  532. long storedId = mSyncedFolderProvider.storeSyncedFolder(item);
  533. if (storedId != -1) {
  534. item.setId(storedId);
  535. if (item.isEnabled()) {
  536. FilesSyncHelper.insertAllDBEntriesForSyncedFolder(item);
  537. } else {
  538. String syncedFolderInitiatedKey = "syncedFolderIntitiated_" + item.getId();
  539. arbitraryDataProvider.deleteKeyForAccount("global", syncedFolderInitiatedKey);
  540. }
  541. }
  542. } else {
  543. // existing synced folder setup to be updated
  544. mSyncedFolderProvider.updateSyncFolder(item);
  545. if (item.isEnabled()) {
  546. FilesSyncHelper.insertAllDBEntriesForSyncedFolder(item);
  547. } else {
  548. String syncedFolderInitiatedKey = "syncedFolderIntitiated_" + item.getId();
  549. arbitraryDataProvider.deleteKeyForAccount("global", syncedFolderInitiatedKey);
  550. }
  551. }
  552. mAdapter.setSyncFolderItem(syncedFolder.getSection(), item);
  553. }
  554. mSyncedFolderPreferencesDialogFragment = null;
  555. if (syncedFolder.getEnabled()) {
  556. showBatteryOptimizationInfo();
  557. }
  558. }
  559. @Override
  560. public void onCancelSyncedFolderPreference() {
  561. mSyncedFolderPreferencesDialogFragment = null;
  562. }
  563. @Override
  564. public void onDeleteSyncedFolderPreference(SyncedFolderParcelable syncedFolder) {
  565. mSyncedFolderProvider.deleteSyncedFolder(syncedFolder.getId());
  566. mAdapter.removeItem(syncedFolder.getSection());
  567. }
  568. /**
  569. * update given synced folder with the given values.
  570. *
  571. * @param item the synced folder to be updated
  572. * @param localPath the local path
  573. * @param remotePath the remote path
  574. * @param wifiOnly upload on wifi only
  575. * @param chargingOnly upload on charging only
  576. * @param subfolderByDate created sub folders
  577. * @param uploadAction upload action
  578. * @param enabled is sync enabled
  579. * @return the updated item
  580. */
  581. private SyncedFolderDisplayItem updateSyncedFolderItem(SyncedFolderDisplayItem item,
  582. String localPath,
  583. String remotePath,
  584. Boolean wifiOnly,
  585. Boolean chargingOnly,
  586. Boolean subfolderByDate,
  587. Integer uploadAction,
  588. Boolean enabled) {
  589. item.setLocalPath(localPath);
  590. item.setRemotePath(remotePath);
  591. item.setWifiOnly(wifiOnly);
  592. item.setChargingOnly(chargingOnly);
  593. item.setSubfolderByDate(subfolderByDate);
  594. item.setUploadAction(uploadAction);
  595. item.setEnabled(enabled);
  596. return item;
  597. }
  598. @Override
  599. public void onRequestPermissionsResult(int requestCode, @NonNull String permissions[],
  600. @NonNull int[] grantResults) {
  601. switch (requestCode) {
  602. case PermissionUtil.PERMISSIONS_WRITE_EXTERNAL_STORAGE: {
  603. // If request is cancelled, result arrays are empty.
  604. if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) {
  605. // permission was granted
  606. int gridWidth = getResources().getInteger(R.integer.media_grid_width);
  607. load(gridWidth * 2, true);
  608. } else {
  609. // permission denied --> do nothing
  610. return;
  611. }
  612. return;
  613. }
  614. default:
  615. super.onRequestPermissionsResult(requestCode, permissions, grantResults);
  616. }
  617. }
  618. @Override
  619. protected void onResume() {
  620. super.onResume();
  621. setDrawerMenuItemChecked(R.id.nav_synced_folders);
  622. }
  623. private void showBatteryOptimizationInfo() {
  624. boolean isSpecialManufacturer = SPECIAL_MANUFACTURER.contains(Build.MANUFACTURER.toLowerCase(Locale.ROOT));
  625. if (isSpecialManufacturer && checkIfBatteryOptimizationEnabled() || checkIfBatteryOptimizationEnabled()) {
  626. AlertDialog alertDialog = new AlertDialog.Builder(this, R.style.Theme_ownCloud_Dialog)
  627. .setTitle(getString(R.string.battery_optimization_title))
  628. .setMessage(getString(R.string.battery_optimization_message))
  629. .setPositiveButton(getString(R.string.battery_optimization_disable), (dialog, which) -> {
  630. // show instant upload
  631. if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
  632. @SuppressLint("BatteryLife")
  633. Intent intent = new Intent(ACTION_REQUEST_IGNORE_BATTERY_OPTIMIZATIONS,
  634. Uri.parse("package:" + BuildConfig.APPLICATION_ID));
  635. if (intent.resolveActivity(getPackageManager()) != null) {
  636. startActivity(intent);
  637. }
  638. } else {
  639. Intent powerUsageIntent = new Intent(Intent.ACTION_POWER_USAGE_SUMMARY);
  640. if (getPackageManager().resolveActivity(powerUsageIntent, 0) != null) {
  641. startActivity(powerUsageIntent);
  642. } else {
  643. dialog.dismiss();
  644. DisplayUtils.showSnackMessage(this, getString(R.string.battery_optimization_no_setting));
  645. }
  646. }
  647. })
  648. .setNegativeButton(getString(R.string.battery_optimization_close), (dialog, which) -> dialog.dismiss())
  649. .setIcon(R.drawable.ic_battery_alert)
  650. .show();
  651. int color = ThemeUtils.primaryAccentColor(this);
  652. alertDialog.getButton(AlertDialog.BUTTON_POSITIVE).setTextColor(color);
  653. alertDialog.getButton(AlertDialog.BUTTON_NEGATIVE).setTextColor(color);
  654. }
  655. }
  656. /**
  657. * Check if battery optimization is enabled. If unknown, fallback to true.
  658. *
  659. * @return true if battery optimization is enabled
  660. */
  661. private boolean checkIfBatteryOptimizationEnabled() {
  662. if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
  663. PowerManager powerManager = (PowerManager) getSystemService(Context.POWER_SERVICE);
  664. if (powerManager == null) {
  665. return true;
  666. }
  667. return !powerManager.isIgnoringBatteryOptimizations(BuildConfig.APPLICATION_ID);
  668. } else {
  669. return true;
  670. }
  671. }
  672. }