SyncedFoldersActivity.java 38 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930
  1. /*
  2. * Nextcloud Android client application
  3. *
  4. * @author Andy Scherzinger
  5. * Copyright (C) 2016 Andy Scherzinger
  6. * Copyright (C) 2016 Nextcloud
  7. * Copyright (C) 2020 Chris Narkiewicz <hello@ezaquarii.com>
  8. *
  9. * This program is free software; you can redistribute it and/or
  10. * modify it under the terms of the GNU AFFERO GENERAL PUBLIC LICENSE
  11. * License as published by the Free Software Foundation; either
  12. * version 3 of the License, or any later version.
  13. *
  14. * This program is distributed in the hope that it will be useful,
  15. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  16. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  17. * GNU AFFERO GENERAL PUBLIC LICENSE for more details.
  18. *
  19. * You should have received a copy of the GNU Affero General Public
  20. * License along with this program. If not, see <http://www.gnu.org/licenses/>.
  21. */
  22. package com.owncloud.android.ui.activity;
  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.ImageView;
  40. import android.widget.LinearLayout;
  41. import android.widget.ProgressBar;
  42. import android.widget.TextView;
  43. import com.google.android.material.button.MaterialButton;
  44. import com.nextcloud.client.account.User;
  45. import com.nextcloud.client.core.Clock;
  46. import com.nextcloud.client.device.PowerManagementService;
  47. import com.nextcloud.client.di.Injectable;
  48. import com.nextcloud.client.jobs.BackgroundJobManager;
  49. import com.nextcloud.client.jobs.MediaFoldersDetectionWork;
  50. import com.nextcloud.client.jobs.NotificationWork;
  51. import com.nextcloud.client.preferences.AppPreferences;
  52. import com.nextcloud.java.util.Optional;
  53. import com.owncloud.android.BuildConfig;
  54. import com.owncloud.android.MainApp;
  55. import com.owncloud.android.R;
  56. import com.owncloud.android.datamodel.ArbitraryDataProvider;
  57. import com.owncloud.android.datamodel.MediaFolder;
  58. import com.owncloud.android.datamodel.MediaFolderType;
  59. import com.owncloud.android.datamodel.MediaProvider;
  60. import com.owncloud.android.datamodel.OCFile;
  61. import com.owncloud.android.datamodel.SyncedFolder;
  62. import com.owncloud.android.datamodel.SyncedFolderDisplayItem;
  63. import com.owncloud.android.datamodel.SyncedFolderProvider;
  64. import com.owncloud.android.files.services.FileUploader;
  65. import com.owncloud.android.ui.adapter.SyncedFolderAdapter;
  66. import com.owncloud.android.ui.decoration.MediaGridItemDecoration;
  67. import com.owncloud.android.ui.dialog.SyncedFolderPreferencesDialogFragment;
  68. import com.owncloud.android.ui.dialog.parcel.SyncedFolderParcelable;
  69. import com.owncloud.android.utils.DisplayUtils;
  70. import com.owncloud.android.utils.PermissionUtil;
  71. import com.owncloud.android.utils.ThemeUtils;
  72. import java.io.File;
  73. import java.util.ArrayList;
  74. import java.util.Arrays;
  75. import java.util.Collections;
  76. import java.util.HashMap;
  77. import java.util.List;
  78. import java.util.Locale;
  79. import java.util.Map;
  80. import javax.inject.Inject;
  81. import androidx.annotation.NonNull;
  82. import androidx.appcompat.app.AlertDialog;
  83. import androidx.drawerlayout.widget.DrawerLayout;
  84. import androidx.fragment.app.FragmentManager;
  85. import androidx.fragment.app.FragmentTransaction;
  86. import androidx.lifecycle.Lifecycle;
  87. import androidx.recyclerview.widget.GridLayoutManager;
  88. import androidx.recyclerview.widget.RecyclerView;
  89. import butterknife.BindView;
  90. import butterknife.ButterKnife;
  91. import butterknife.OnClick;
  92. import static android.provider.Settings.ACTION_REQUEST_IGNORE_BATTERY_OPTIMIZATIONS;
  93. import static com.owncloud.android.datamodel.SyncedFolderDisplayItem.UNPERSISTED_ID;
  94. /**
  95. * Activity displaying all auto-synced folders and/or instant upload media folders.
  96. */
  97. public class SyncedFoldersActivity extends FileActivity implements SyncedFolderAdapter.ClickListener,
  98. SyncedFolderPreferencesDialogFragment.OnSyncedFolderPreferenceListener, Injectable {
  99. private static final String[] PRIORITIZED_FOLDERS = new String[]{"Camera", "Screenshots"};
  100. public static final String EXTRA_SHOW_SIDEBAR = "SHOW_SIDEBAR";
  101. private static final String SYNCED_FOLDER_PREFERENCES_DIALOG_TAG = "SYNCED_FOLDER_PREFERENCES_DIALOG";
  102. private static final String TAG = SyncedFoldersActivity.class.getSimpleName();
  103. @BindView(R.id.empty_list_view)
  104. public LinearLayout emptyContentContainer;
  105. @BindView(R.id.empty_list_icon)
  106. public ImageView emptyContentIcon;
  107. @BindView(R.id.empty_list_progress)
  108. public ProgressBar emptyContentProgressBar;
  109. @BindView(R.id.empty_list_view_headline)
  110. public TextView emptyContentHeadline;
  111. @BindView(R.id.empty_list_view_text)
  112. public TextView emptyContentMessage;
  113. @BindView(R.id.empty_list_view_action)
  114. public MaterialButton emptyContentActionButton;
  115. @BindView(android.R.id.list)
  116. public RecyclerView mRecyclerView;
  117. private SyncedFolderAdapter adapter;
  118. private SyncedFolderProvider syncedFolderProvider;
  119. private SyncedFolderPreferencesDialogFragment syncedFolderPreferencesDialogFragment;
  120. private boolean showSidebar = true;
  121. private String path;
  122. private int type;
  123. @Inject AppPreferences preferences;
  124. @Inject PowerManagementService powerManagementService;
  125. @Inject Clock clock;
  126. @Inject BackgroundJobManager backgroundJobManager;
  127. @Override
  128. protected void onCreate(Bundle savedInstanceState) {
  129. super.onCreate(savedInstanceState);
  130. if (getIntent().getExtras() != null) {
  131. showSidebar = getIntent().getExtras().getBoolean(EXTRA_SHOW_SIDEBAR);
  132. }
  133. setContentView(R.layout.synced_folders_layout);
  134. ButterKnife.bind(this);
  135. if (getIntent() != null && getIntent().getExtras() != null) {
  136. final String accountName = getIntent().getExtras().getString(NotificationWork.KEY_NOTIFICATION_ACCOUNT);
  137. Optional<User> optionalUser = getUser();
  138. if (optionalUser.isPresent() && accountName != null) {
  139. User user = optionalUser.get();
  140. if (!accountName.equalsIgnoreCase(user.getAccountName())) {
  141. accountManager.setCurrentOwnCloudAccount(accountName);
  142. setUser(getUserAccountManager().getUser());
  143. }
  144. }
  145. path = getIntent().getStringExtra(MediaFoldersDetectionWork.KEY_MEDIA_FOLDER_PATH);
  146. type = getIntent().getIntExtra(MediaFoldersDetectionWork.KEY_MEDIA_FOLDER_TYPE, -1);
  147. // Cancel notification
  148. int notificationId = getIntent().getIntExtra(MediaFoldersDetectionWork.NOTIFICATION_ID, 0);
  149. NotificationManager notificationManager =
  150. (NotificationManager) getSystemService(Activity.NOTIFICATION_SERVICE);
  151. notificationManager.cancel(notificationId);
  152. }
  153. // setup toolbar
  154. setupToolbar();
  155. updateActionBarTitleAndHomeButtonByString(getString(R.string.drawer_synced_folders));
  156. // setup drawer
  157. setupDrawer(R.id.nav_synced_folders);
  158. if (!showSidebar) {
  159. setDrawerLockMode(DrawerLayout.LOCK_MODE_LOCKED_CLOSED);
  160. mDrawerToggle.setDrawerIndicatorEnabled(false);
  161. }
  162. setupContent();
  163. if (ThemeUtils.themingEnabled(this)) {
  164. setTheme(R.style.FallbackThemingTheme);
  165. }
  166. }
  167. @Override
  168. public boolean onCreateOptionsMenu(Menu menu) {
  169. MenuInflater inflater = getMenuInflater();
  170. inflater.inflate(R.menu.activity_synced_folders, menu);
  171. if (powerManagementService.isPowerSavingExclusionAvailable()) {
  172. MenuItem item = menu.findItem(R.id.action_disable_power_save_check);
  173. item.setVisible(true);
  174. item.setChecked(preferences.isPowerCheckDisabled());
  175. item.setOnMenuItemClickListener(this::onDisablePowerSaveCheckClicked);
  176. }
  177. return true;
  178. }
  179. private boolean onDisablePowerSaveCheckClicked(MenuItem powerCheck) {
  180. if (!powerCheck.isChecked()) {
  181. showPowerCheckDialog();
  182. }
  183. preferences.setPowerCheckDisabled(!powerCheck.isChecked());
  184. powerCheck.setChecked(!powerCheck.isChecked());
  185. return true;
  186. }
  187. private void showPowerCheckDialog() {
  188. AlertDialog alertDialog = new AlertDialog.Builder(this)
  189. .setView(findViewById(R.id.root_layout))
  190. .setPositiveButton(R.string.common_ok, (dialog, which) -> dialog.dismiss())
  191. .setTitle(R.string.autoupload_disable_power_save_check)
  192. .setMessage(getString(R.string.power_save_check_dialog_message))
  193. .show();
  194. alertDialog.getButton(AlertDialog.BUTTON_POSITIVE).setTextColor(ThemeUtils.primaryAccentColor(this));
  195. }
  196. /**
  197. * sets up the UI elements and loads all media/synced folders.
  198. */
  199. private void setupContent() {
  200. final int gridWidth = getResources().getInteger(R.integer.media_grid_width);
  201. boolean lightVersion = getResources().getBoolean(R.bool.syncedFolder_light);
  202. adapter = new SyncedFolderAdapter(this, clock, gridWidth, this, lightVersion);
  203. syncedFolderProvider = new SyncedFolderProvider(getContentResolver(), preferences, clock);
  204. emptyContentIcon.setImageResource(R.drawable.nav_synced_folders);
  205. ThemeUtils.colorPrimaryButton(emptyContentActionButton, this);
  206. final GridLayoutManager lm = new GridLayoutManager(this, gridWidth);
  207. adapter.setLayoutManager(lm);
  208. int spacing = getResources().getDimensionPixelSize(R.dimen.media_grid_spacing);
  209. mRecyclerView.addItemDecoration(new MediaGridItemDecoration(spacing));
  210. mRecyclerView.setLayoutManager(lm);
  211. mRecyclerView.setAdapter(adapter);
  212. load(gridWidth * 2, false);
  213. }
  214. @OnClick(R.id.empty_list_view_action)
  215. public void showHiddenItems() {
  216. if (adapter.getSectionCount() == 0 && adapter.getUnfilteredSectionCount() > adapter.getSectionCount()) {
  217. adapter.toggleHiddenItemsVisibility();
  218. emptyContentContainer.setVisibility(View.GONE);
  219. mRecyclerView.setVisibility(View.VISIBLE);
  220. }
  221. }
  222. /**
  223. * loads all media/synced folders, adds them to the recycler view adapter and shows the list.
  224. *
  225. * @param perFolderMediaItemLimit the amount of media items to be loaded/shown per media folder
  226. */
  227. private void load(final int perFolderMediaItemLimit, boolean force) {
  228. if (adapter.getItemCount() > 0 && !force) {
  229. return;
  230. }
  231. showLoadingContent();
  232. final List<MediaFolder> mediaFolders = MediaProvider.getImageFolders(getContentResolver(),
  233. perFolderMediaItemLimit, this, false);
  234. mediaFolders.addAll(MediaProvider.getVideoFolders(getContentResolver(), perFolderMediaItemLimit,
  235. this, false));
  236. List<SyncedFolder> syncedFolderArrayList = syncedFolderProvider.getSyncedFolders();
  237. List<SyncedFolder> currentAccountSyncedFoldersList = new ArrayList<>();
  238. User user = getUserAccountManager().getUser();
  239. for (SyncedFolder syncedFolder : syncedFolderArrayList) {
  240. if (syncedFolder.getAccount().equals(user.getAccountName())) {
  241. // delete non-existing & disabled synced folders
  242. if (!new File(syncedFolder.getLocalPath()).exists() && !syncedFolder.isEnabled()) {
  243. syncedFolderProvider.deleteSyncedFolder(syncedFolder.getId());
  244. } else {
  245. currentAccountSyncedFoldersList.add(syncedFolder);
  246. }
  247. }
  248. }
  249. List<SyncedFolderDisplayItem> syncFolderItems = sortSyncedFolderItems(
  250. mergeFolderData(currentAccountSyncedFoldersList, mediaFolders));
  251. adapter.setSyncFolderItems(syncFolderItems);
  252. adapter.notifyDataSetChanged();
  253. showList();
  254. if (!TextUtils.isEmpty(path)) {
  255. int section = adapter.getSectionByLocalPathAndType(path, type);
  256. if (section >= 0) {
  257. onSyncFolderSettingsClick(section, adapter.get(section));
  258. }
  259. }
  260. }
  261. /**
  262. * Sorts list of {@link SyncedFolderDisplayItem}s.
  263. *
  264. * @param syncFolderItemList list of items to be sorted
  265. * @return sorted list of items
  266. */
  267. public static List<SyncedFolderDisplayItem> sortSyncedFolderItems(List<SyncedFolderDisplayItem>
  268. syncFolderItemList) {
  269. Collections.sort(syncFolderItemList, (f1, f2) -> {
  270. if (f1 == null && f2 == null) {
  271. return 0;
  272. } else if (f1 == null) {
  273. return -1;
  274. } else if (f2 == null) {
  275. return 1;
  276. } else if (f1.isEnabled() && f2.isEnabled()) {
  277. if (f1.getFolderName() == null) {
  278. return -1;
  279. }
  280. if (f2.getFolderName() == null) {
  281. return 1;
  282. }
  283. return f1.getFolderName().toLowerCase(Locale.getDefault()).compareTo(
  284. f2.getFolderName().toLowerCase(Locale.getDefault()));
  285. } else if (f1.getFolderName() == null && f2.getFolderName() == null) {
  286. return 0;
  287. } else if (f1.isEnabled()) {
  288. return -1;
  289. } else if (f2.isEnabled()) {
  290. return 1;
  291. } else if (f1.getFolderName() == null) {
  292. return -1;
  293. } else if (f2.getFolderName() == null) {
  294. return 1;
  295. }
  296. for (String folder : PRIORITIZED_FOLDERS) {
  297. if (folder.equals(f1.getFolderName()) && folder.equals(f2.getFolderName())) {
  298. return 0;
  299. } else if (folder.equals(f1.getFolderName())) {
  300. return -1;
  301. } else if (folder.equals(f2.getFolderName())) {
  302. return 1;
  303. }
  304. }
  305. return f1.getFolderName().toLowerCase(Locale.getDefault()).compareTo(
  306. f2.getFolderName().toLowerCase(Locale.getDefault()));
  307. });
  308. return syncFolderItemList;
  309. }
  310. /**
  311. * merges two lists of {@link SyncedFolder} and {@link MediaFolder} items into one of SyncedFolderItems.
  312. *
  313. * @param syncedFolders the synced folders
  314. * @param mediaFolders the media folders
  315. * @return the merged list of SyncedFolderItems
  316. */
  317. @NonNull
  318. private List<SyncedFolderDisplayItem> mergeFolderData(List<SyncedFolder> syncedFolders,
  319. @NonNull List<MediaFolder> mediaFolders) {
  320. Map<String, SyncedFolder> syncedFoldersMap = createSyncedFoldersMap(syncedFolders);
  321. List<SyncedFolderDisplayItem> result = new ArrayList<>();
  322. for (MediaFolder mediaFolder : mediaFolders) {
  323. if (syncedFoldersMap.containsKey(mediaFolder.absolutePath + "-" + mediaFolder.type)) {
  324. SyncedFolder syncedFolder = syncedFoldersMap.get(mediaFolder.absolutePath + "-" + mediaFolder.type);
  325. syncedFoldersMap.remove(mediaFolder.absolutePath + "-" + mediaFolder.type);
  326. if (MediaFolderType.CUSTOM == syncedFolder.getType()) {
  327. result.add(createSyncedFolderWithoutMediaFolder(syncedFolder));
  328. } else {
  329. result.add(createSyncedFolder(syncedFolder, mediaFolder));
  330. }
  331. } else {
  332. result.add(createSyncedFolderFromMediaFolder(mediaFolder));
  333. }
  334. }
  335. for (SyncedFolder syncedFolder : syncedFoldersMap.values()) {
  336. result.add(createSyncedFolderWithoutMediaFolder(syncedFolder));
  337. }
  338. return result;
  339. }
  340. @NonNull
  341. private SyncedFolderDisplayItem createSyncedFolderWithoutMediaFolder(@NonNull SyncedFolder syncedFolder) {
  342. File localFolder = new File(syncedFolder.getLocalPath());
  343. File[] files = getFileList(localFolder);
  344. List<String> filePaths = getDisplayFilePathList(files);
  345. return new SyncedFolderDisplayItem(
  346. syncedFolder.getId(),
  347. syncedFolder.getLocalPath(),
  348. syncedFolder.getRemotePath(),
  349. syncedFolder.isWifiOnly(),
  350. syncedFolder.isChargingOnly(),
  351. syncedFolder.isExisting(),
  352. syncedFolder.isSubfolderByDate(),
  353. syncedFolder.getAccount(),
  354. syncedFolder.getUploadAction(),
  355. syncedFolder.getNameCollisionPolicyInt(),
  356. syncedFolder.isEnabled(),
  357. clock.getCurrentTime(),
  358. filePaths,
  359. localFolder.getName(),
  360. files.length,
  361. syncedFolder.getType(),
  362. syncedFolder.isHidden());
  363. }
  364. /**
  365. * creates a SyncedFolderDisplayItem merging a {@link SyncedFolder} and a {@link MediaFolder} object instance.
  366. *
  367. * @param syncedFolder the synced folder object
  368. * @param mediaFolder the media folder object
  369. * @return the created SyncedFolderDisplayItem
  370. */
  371. @NonNull
  372. private SyncedFolderDisplayItem createSyncedFolder(@NonNull SyncedFolder syncedFolder, @NonNull MediaFolder mediaFolder) {
  373. return new SyncedFolderDisplayItem(
  374. syncedFolder.getId(),
  375. syncedFolder.getLocalPath(),
  376. syncedFolder.getRemotePath(),
  377. syncedFolder.isWifiOnly(),
  378. syncedFolder.isChargingOnly(),
  379. syncedFolder.isExisting(),
  380. syncedFolder.isSubfolderByDate(),
  381. syncedFolder.getAccount(),
  382. syncedFolder.getUploadAction(),
  383. syncedFolder.getNameCollisionPolicyInt(),
  384. syncedFolder.isEnabled(),
  385. clock.getCurrentTime(),
  386. mediaFolder.filePaths,
  387. mediaFolder.folderName,
  388. mediaFolder.numberOfFiles,
  389. mediaFolder.type,
  390. syncedFolder.isHidden());
  391. }
  392. /**
  393. * creates a {@link SyncedFolderDisplayItem} based on a {@link MediaFolder} object instance.
  394. *
  395. * @param mediaFolder the media folder object
  396. * @return the created SyncedFolderDisplayItem
  397. */
  398. @NonNull
  399. private SyncedFolderDisplayItem createSyncedFolderFromMediaFolder(@NonNull MediaFolder mediaFolder) {
  400. return new SyncedFolderDisplayItem(
  401. UNPERSISTED_ID,
  402. mediaFolder.absolutePath,
  403. getString(R.string.instant_upload_path) + "/" + mediaFolder.folderName,
  404. true,
  405. false,
  406. true,
  407. false,
  408. getAccount().name,
  409. FileUploader.LOCAL_BEHAVIOUR_FORGET,
  410. FileUploader.NameCollisionPolicy.ASK_USER.serialize(),
  411. false,
  412. clock.getCurrentTime(),
  413. mediaFolder.filePaths,
  414. mediaFolder.folderName,
  415. mediaFolder.numberOfFiles,
  416. mediaFolder.type,
  417. false);
  418. }
  419. private File[] getFileList(File localFolder) {
  420. File[] files = localFolder.listFiles(pathname -> !pathname.isDirectory());
  421. if (files != null) {
  422. Arrays.sort(files, (f1, f2) -> Long.compare(f1.lastModified(), f2.lastModified()));
  423. } else {
  424. files = new File[]{};
  425. }
  426. return files;
  427. }
  428. private List<String> getDisplayFilePathList(File... files) {
  429. List<String> filePaths = null;
  430. if (files != null && files.length > 0) {
  431. filePaths = new ArrayList<>();
  432. for (int i = 0; i < 7 && i < files.length; i++) {
  433. filePaths.add(files[i].getAbsolutePath());
  434. }
  435. }
  436. return filePaths;
  437. }
  438. /**
  439. * creates a lookup map for a list of given {@link SyncedFolder}s with their local path as the key.
  440. *
  441. * @param syncFolders list of {@link SyncedFolder}s
  442. * @return the lookup map for {@link SyncedFolder}s
  443. */
  444. @NonNull
  445. private Map<String, SyncedFolder> createSyncedFoldersMap(List<SyncedFolder> syncFolders) {
  446. Map<String, SyncedFolder> result = new HashMap<>();
  447. if (syncFolders != null) {
  448. for (SyncedFolder syncFolder : syncFolders) {
  449. result.put(syncFolder.getLocalPath() + "-" + syncFolder.getType(), syncFolder);
  450. }
  451. }
  452. return result;
  453. }
  454. /**
  455. * show recycler view list or the empty message info (in case list is empty).
  456. */
  457. private void showList() {
  458. if (mRecyclerView != null) {
  459. mRecyclerView.setVisibility(View.VISIBLE);
  460. emptyContentProgressBar.setVisibility(View.GONE);
  461. checkAndShowEmptyListContent();
  462. }
  463. }
  464. private void checkAndShowEmptyListContent() {
  465. if (adapter.getSectionCount() == 0 && adapter.getUnfilteredSectionCount() > adapter.getSectionCount()) {
  466. emptyContentContainer.setVisibility(View.VISIBLE);
  467. int hiddenFoldersCount = adapter.getHiddenFolderCount();
  468. showEmptyContent(getString(R.string.drawer_synced_folders),
  469. getResources().getQuantityString(R.plurals.synced_folders_show_hidden_folders,
  470. hiddenFoldersCount,
  471. hiddenFoldersCount),
  472. getResources().getQuantityString(R.plurals.synced_folders_show_hidden_folders,
  473. hiddenFoldersCount,
  474. hiddenFoldersCount));
  475. } else if (adapter.getSectionCount() == 0 && adapter.getUnfilteredSectionCount() == 0) {
  476. emptyContentContainer.setVisibility(View.VISIBLE);
  477. showEmptyContent(getString(R.string.drawer_synced_folders),
  478. getString(R.string.synced_folders_no_results));
  479. } else {
  480. emptyContentContainer.setVisibility(View.GONE);
  481. }
  482. }
  483. @Override
  484. public boolean onOptionsItemSelected(MenuItem item) {
  485. boolean result = true;
  486. switch (item.getItemId()) {
  487. case android.R.id.home: {
  488. if (showSidebar) {
  489. if (isDrawerOpen()) {
  490. closeDrawer();
  491. } else {
  492. openDrawer();
  493. }
  494. } else {
  495. Intent settingsIntent = new Intent(getApplicationContext(), SettingsActivity.class);
  496. startActivity(settingsIntent);
  497. }
  498. break;
  499. }
  500. case R.id.action_create_custom_folder: {
  501. Log.d(TAG, "Show custom folder dialog");
  502. SyncedFolderDisplayItem emptyCustomFolder = new SyncedFolderDisplayItem(
  503. SyncedFolder.UNPERSISTED_ID,
  504. null,
  505. null,
  506. true,
  507. false,
  508. true,
  509. false,
  510. getAccount().name,
  511. FileUploader.LOCAL_BEHAVIOUR_FORGET,
  512. FileUploader.NameCollisionPolicy.ASK_USER.serialize(),
  513. false,
  514. clock.getCurrentTime(),
  515. null,
  516. MediaFolderType.CUSTOM,
  517. false);
  518. onSyncFolderSettingsClick(0, emptyCustomFolder);
  519. }
  520. default:
  521. result = super.onOptionsItemSelected(item);
  522. break;
  523. }
  524. return result;
  525. }
  526. @Override
  527. public void restart() {
  528. Intent i = new Intent(this, FileDisplayActivity.class);
  529. i.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
  530. startActivity(i);
  531. }
  532. @Override
  533. public void showFiles(boolean onDeviceOnly) {
  534. MainApp.showOnlyFilesOnDevice(onDeviceOnly);
  535. Intent fileDisplayActivity = new Intent(getApplicationContext(), FileDisplayActivity.class);
  536. fileDisplayActivity.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
  537. startActivity(fileDisplayActivity);
  538. }
  539. @Override
  540. public void onSyncStatusToggleClick(int section, SyncedFolderDisplayItem syncedFolderDisplayItem) {
  541. if (syncedFolderDisplayItem.getId() > UNPERSISTED_ID) {
  542. syncedFolderProvider.updateSyncedFolderEnabled(syncedFolderDisplayItem.getId(),
  543. syncedFolderDisplayItem.isEnabled());
  544. } else {
  545. long storedId = syncedFolderProvider.storeSyncedFolder(syncedFolderDisplayItem);
  546. if (storedId != -1) {
  547. syncedFolderDisplayItem.setId(storedId);
  548. }
  549. }
  550. if (syncedFolderDisplayItem.isEnabled()) {
  551. backgroundJobManager.startImmediateFilesSyncJob(false, false);
  552. showBatteryOptimizationInfo();
  553. }
  554. }
  555. @Override
  556. public void onSyncFolderSettingsClick(int section, SyncedFolderDisplayItem syncedFolderDisplayItem) {
  557. FragmentManager fm = getSupportFragmentManager();
  558. FragmentTransaction ft = fm.beginTransaction();
  559. ft.addToBackStack(null);
  560. syncedFolderPreferencesDialogFragment = SyncedFolderPreferencesDialogFragment.newInstance(
  561. syncedFolderDisplayItem, section);
  562. syncedFolderPreferencesDialogFragment.show(ft, SYNCED_FOLDER_PREFERENCES_DIALOG_TAG);
  563. }
  564. @Override
  565. public void onVisibilityToggleClick(int section, SyncedFolderDisplayItem syncedFolder) {
  566. syncedFolder.setHidden(!syncedFolder.isHidden());
  567. saveOrUpdateSyncedFolder(syncedFolder);
  568. adapter.setSyncFolderItem(section, syncedFolder);
  569. checkAndShowEmptyListContent();
  570. }
  571. private void showEmptyContent(String headline, String message) {
  572. showEmptyContent(headline, message, false);
  573. emptyContentActionButton.setVisibility(View.GONE);
  574. }
  575. private void showEmptyContent(String headline, String message, String action) {
  576. showEmptyContent(headline, message, false);
  577. emptyContentActionButton.setText(action);
  578. emptyContentActionButton.setVisibility(View.VISIBLE);
  579. emptyContentMessage.setVisibility(View.GONE);
  580. }
  581. private void showLoadingContent() {
  582. showEmptyContent(
  583. getString(R.string.drawer_synced_folders),
  584. getString(R.string.synced_folders_loading_folders),
  585. true
  586. );
  587. emptyContentActionButton.setVisibility(View.GONE);
  588. }
  589. private void showEmptyContent(String headline, String message, boolean loading) {
  590. if (emptyContentContainer != null) {
  591. emptyContentContainer.setVisibility(View.VISIBLE);
  592. mRecyclerView.setVisibility(View.GONE);
  593. emptyContentHeadline.setText(headline);
  594. emptyContentMessage.setText(message);
  595. emptyContentMessage.setVisibility(View.VISIBLE);
  596. if (loading) {
  597. emptyContentProgressBar.setVisibility(View.VISIBLE);
  598. emptyContentIcon.setVisibility(View.GONE);
  599. } else {
  600. emptyContentProgressBar.setVisibility(View.GONE);
  601. emptyContentIcon.setVisibility(View.VISIBLE);
  602. }
  603. }
  604. }
  605. @Override
  606. protected void onActivityResult(int requestCode, int resultCode, Intent data) {
  607. if (requestCode == SyncedFolderPreferencesDialogFragment.REQUEST_CODE__SELECT_REMOTE_FOLDER
  608. && resultCode == RESULT_OK && syncedFolderPreferencesDialogFragment != null) {
  609. OCFile chosenFolder = data.getParcelableExtra(FolderPickerActivity.EXTRA_FOLDER);
  610. syncedFolderPreferencesDialogFragment.setRemoteFolderSummary(chosenFolder.getRemotePath());
  611. }
  612. if (requestCode == SyncedFolderPreferencesDialogFragment.REQUEST_CODE__SELECT_LOCAL_FOLDER
  613. && resultCode == RESULT_OK && syncedFolderPreferencesDialogFragment != null) {
  614. String localPath = data.getStringExtra(UploadFilesActivity.EXTRA_CHOSEN_FILES);
  615. syncedFolderPreferencesDialogFragment.setLocalFolderSummary(localPath);
  616. } else {
  617. super.onActivityResult(requestCode, resultCode, data);
  618. }
  619. }
  620. @Override
  621. public void onSaveSyncedFolderPreference(SyncedFolderParcelable syncedFolder) {
  622. // custom folders newly created aren't in the list already,
  623. // so triggering a refresh
  624. if (MediaFolderType.CUSTOM == syncedFolder.getType() && syncedFolder.getId() == UNPERSISTED_ID) {
  625. SyncedFolderDisplayItem newCustomFolder = new SyncedFolderDisplayItem(
  626. SyncedFolder.UNPERSISTED_ID,
  627. syncedFolder.getLocalPath(),
  628. syncedFolder.getRemotePath(),
  629. syncedFolder.isWifiOnly(),
  630. syncedFolder.isChargingOnly(),
  631. syncedFolder.isExisting(),
  632. syncedFolder.isSubfolderByDate(),
  633. syncedFolder.getAccount(),
  634. syncedFolder.getUploadAction(),
  635. syncedFolder.getNameCollisionPolicy().serialize(),
  636. syncedFolder.isEnabled(),
  637. clock.getCurrentTime(),
  638. new File(syncedFolder.getLocalPath()).getName(),
  639. syncedFolder.getType(),
  640. syncedFolder.isHidden());
  641. saveOrUpdateSyncedFolder(newCustomFolder);
  642. adapter.addSyncFolderItem(newCustomFolder);
  643. } else {
  644. SyncedFolderDisplayItem item = adapter.get(syncedFolder.getSection());
  645. updateSyncedFolderItem(item,
  646. syncedFolder.getId(),
  647. syncedFolder.getLocalPath(),
  648. syncedFolder.getRemotePath(),
  649. syncedFolder.isWifiOnly(),
  650. syncedFolder.isChargingOnly(),
  651. syncedFolder.isExisting(),
  652. syncedFolder.isSubfolderByDate(),
  653. syncedFolder.getUploadAction(),
  654. syncedFolder.getNameCollisionPolicy().serialize(),
  655. syncedFolder.isEnabled());
  656. saveOrUpdateSyncedFolder(item);
  657. // TODO test if notifyItemChanged is sufficient (should improve performance)
  658. adapter.notifyDataSetChanged();
  659. }
  660. syncedFolderPreferencesDialogFragment = null;
  661. if (syncedFolder.isEnabled()) {
  662. showBatteryOptimizationInfo();
  663. }
  664. }
  665. private void saveOrUpdateSyncedFolder(SyncedFolderDisplayItem item) {
  666. if (item.getId() == UNPERSISTED_ID) {
  667. // newly set up folder sync config
  668. storeSyncedFolder(item);
  669. } else {
  670. // existing synced folder setup to be updated
  671. syncedFolderProvider.updateSyncFolder(item);
  672. if (item.isEnabled()) {
  673. backgroundJobManager.startImmediateFilesSyncJob(false, false);
  674. } else {
  675. String syncedFolderInitiatedKey = "syncedFolderIntitiated_" + item.getId();
  676. ArbitraryDataProvider arbitraryDataProvider = new ArbitraryDataProvider(MainApp.getAppContext().
  677. getContentResolver());
  678. arbitraryDataProvider.deleteKeyForAccount("global", syncedFolderInitiatedKey);
  679. }
  680. }
  681. }
  682. private void storeSyncedFolder(SyncedFolderDisplayItem item) {
  683. ArbitraryDataProvider arbitraryDataProvider = new ArbitraryDataProvider(MainApp.getAppContext().
  684. getContentResolver());
  685. long storedId = syncedFolderProvider.storeSyncedFolder(item);
  686. if (storedId != -1) {
  687. item.setId(storedId);
  688. if (item.isEnabled()) {
  689. backgroundJobManager.startImmediateFilesSyncJob(false, false);
  690. } else {
  691. String syncedFolderInitiatedKey = "syncedFolderIntitiated_" + item.getId();
  692. arbitraryDataProvider.deleteKeyForAccount("global", syncedFolderInitiatedKey);
  693. }
  694. }
  695. }
  696. @Override
  697. public void onCancelSyncedFolderPreference() {
  698. syncedFolderPreferencesDialogFragment = null;
  699. }
  700. @Override
  701. public void onDeleteSyncedFolderPreference(SyncedFolderParcelable syncedFolder) {
  702. syncedFolderProvider.deleteSyncedFolder(syncedFolder.getId());
  703. adapter.removeItem(syncedFolder.getSection());
  704. }
  705. /**
  706. * update given synced folder with the given values.
  707. *
  708. * @param item the synced folder to be updated
  709. * @param localPath the local path
  710. * @param remotePath the remote path
  711. * @param wifiOnly upload on wifi only
  712. * @param chargingOnly upload on charging only
  713. * @param existing also upload existing
  714. * @param subfolderByDate created sub folders
  715. * @param uploadAction upload action
  716. * @param nameCollisionPolicy what to do on name collision
  717. * @param enabled is sync enabled
  718. */
  719. private void updateSyncedFolderItem(SyncedFolderDisplayItem item,
  720. long id,
  721. String localPath,
  722. String remotePath,
  723. boolean wifiOnly,
  724. boolean chargingOnly,
  725. boolean existing,
  726. boolean subfolderByDate,
  727. Integer uploadAction,
  728. Integer nameCollisionPolicy,
  729. boolean enabled) {
  730. item.setId(id);
  731. item.setLocalPath(localPath);
  732. item.setRemotePath(remotePath);
  733. item.setWifiOnly(wifiOnly);
  734. item.setChargingOnly(chargingOnly);
  735. item.setExisting(existing);
  736. item.setSubfolderByDate(subfolderByDate);
  737. item.setUploadAction(uploadAction);
  738. item.setNameCollisionPolicy(nameCollisionPolicy);
  739. item.setEnabled(enabled, clock.getCurrentTime());
  740. }
  741. @Override
  742. public void onRequestPermissionsResult(int requestCode, @NonNull String permissions[],
  743. @NonNull int[] grantResults) {
  744. switch (requestCode) {
  745. case PermissionUtil.PERMISSIONS_WRITE_EXTERNAL_STORAGE: {
  746. // If request is cancelled, result arrays are empty.
  747. if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) {
  748. // permission was granted
  749. int gridWidth = getResources().getInteger(R.integer.media_grid_width);
  750. load(gridWidth * 2, true);
  751. } else {
  752. // permission denied --> do nothing
  753. return;
  754. }
  755. return;
  756. }
  757. default:
  758. super.onRequestPermissionsResult(requestCode, permissions, grantResults);
  759. }
  760. }
  761. @Override
  762. protected void onResume() {
  763. super.onResume();
  764. setDrawerMenuItemChecked(R.id.nav_synced_folders);
  765. }
  766. private void showBatteryOptimizationInfo() {
  767. if (powerManagementService.isPowerSavingExclusionAvailable() || checkIfBatteryOptimizationEnabled()) {
  768. AlertDialog.Builder alertDialogBuilder = new AlertDialog.Builder(this, R.style.Theme_ownCloud_Dialog)
  769. .setTitle(getString(R.string.battery_optimization_title))
  770. .setMessage(getString(R.string.battery_optimization_message))
  771. .setPositiveButton(getString(R.string.battery_optimization_disable), (dialog, which) -> {
  772. // show instant upload
  773. if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
  774. @SuppressLint("BatteryLife")
  775. Intent intent = new Intent(ACTION_REQUEST_IGNORE_BATTERY_OPTIMIZATIONS,
  776. Uri.parse("package:" + BuildConfig.APPLICATION_ID));
  777. if (intent.resolveActivity(getPackageManager()) != null) {
  778. startActivity(intent);
  779. }
  780. } else {
  781. Intent powerUsageIntent = new Intent(Intent.ACTION_POWER_USAGE_SUMMARY);
  782. if (getPackageManager().resolveActivity(powerUsageIntent, 0) != null) {
  783. startActivity(powerUsageIntent);
  784. } else {
  785. dialog.dismiss();
  786. DisplayUtils.showSnackMessage(this, getString(R.string.battery_optimization_no_setting));
  787. }
  788. }
  789. })
  790. .setNegativeButton(getString(R.string.battery_optimization_close), (dialog, which) -> dialog.dismiss())
  791. .setIcon(R.drawable.ic_battery_alert);
  792. if (getLifecycle().getCurrentState().isAtLeast(Lifecycle.State.RESUMED)) {
  793. AlertDialog alertDialog = alertDialogBuilder.show();
  794. int color = ThemeUtils.primaryAccentColor(this);
  795. alertDialog.getButton(AlertDialog.BUTTON_POSITIVE).setTextColor(color);
  796. alertDialog.getButton(AlertDialog.BUTTON_NEGATIVE).setTextColor(color);
  797. }
  798. }
  799. }
  800. /**
  801. * Check if battery optimization is enabled. If unknown, fallback to true.
  802. *
  803. * @return true if battery optimization is enabled
  804. */
  805. private boolean checkIfBatteryOptimizationEnabled() {
  806. if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
  807. PowerManager powerManager = (PowerManager) getSystemService(Context.POWER_SERVICE);
  808. if (powerManager == null) {
  809. return true;
  810. }
  811. return !powerManager.isIgnoringBatteryOptimizations(BuildConfig.APPLICATION_ID);
  812. } else {
  813. return true;
  814. }
  815. }
  816. }