PreviewImageActivity.java 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526
  1. /*
  2. * ownCloud Android client application
  3. *
  4. * @author David A. Velasco
  5. * @author Chris Narkiewicz
  6. *
  7. * Copyright (C) 2016 ownCloud Inc.
  8. * Copyright (C) 2019 Chris Narkiewicz <hello@ezaquarii.com>
  9. *
  10. * This program is free software: you can redistribute it and/or modify
  11. * it under the terms of the GNU General Public License version 2,
  12. * as published by the Free Software Foundation.
  13. *
  14. * This program is distributed in the hd 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 General Public License for more details.
  18. *
  19. * You should have received a copy of the GNU General Public License
  20. * along with this program. If not, see <http://www.gnu.org/licenses/>.
  21. *
  22. */
  23. package com.owncloud.android.ui.preview;
  24. import android.annotation.SuppressLint;
  25. import android.content.BroadcastReceiver;
  26. import android.content.ComponentName;
  27. import android.content.Context;
  28. import android.content.Intent;
  29. import android.content.IntentFilter;
  30. import android.content.ServiceConnection;
  31. import android.graphics.Color;
  32. import android.os.Bundle;
  33. import android.os.IBinder;
  34. import android.view.MenuItem;
  35. import android.view.View;
  36. import com.nextcloud.client.account.User;
  37. import com.nextcloud.client.di.Injectable;
  38. import com.nextcloud.client.preferences.AppPreferences;
  39. import com.nextcloud.java.util.Optional;
  40. import com.owncloud.android.MainApp;
  41. import com.owncloud.android.R;
  42. import com.owncloud.android.datamodel.FileDataStorageManager;
  43. import com.owncloud.android.datamodel.OCFile;
  44. import com.owncloud.android.datamodel.VirtualFolderType;
  45. import com.owncloud.android.files.services.FileDownloader;
  46. import com.owncloud.android.files.services.FileDownloader.FileDownloaderBinder;
  47. import com.owncloud.android.files.services.FileUploader;
  48. import com.owncloud.android.files.services.FileUploader.FileUploaderBinder;
  49. import com.owncloud.android.lib.common.operations.OnRemoteOperationListener;
  50. import com.owncloud.android.lib.common.operations.RemoteOperation;
  51. import com.owncloud.android.lib.common.operations.RemoteOperationResult;
  52. import com.owncloud.android.lib.common.utils.Log_OC;
  53. import com.owncloud.android.operations.RemoveFileOperation;
  54. import com.owncloud.android.operations.SynchronizeFileOperation;
  55. import com.owncloud.android.ui.activity.FileActivity;
  56. import com.owncloud.android.ui.activity.FileDisplayActivity;
  57. import com.owncloud.android.ui.fragment.FileFragment;
  58. import com.owncloud.android.utils.MimeTypeUtil;
  59. import com.owncloud.android.utils.ThemeUtils;
  60. import javax.inject.Inject;
  61. import androidx.annotation.NonNull;
  62. import androidx.appcompat.app.ActionBar;
  63. import androidx.drawerlayout.widget.DrawerLayout;
  64. import androidx.viewpager.widget.ViewPager;
  65. import edu.umd.cs.findbugs.annotations.SuppressFBWarnings;
  66. /**
  67. * Holds a swiping galley where image files contained in an ownCloud directory are shown
  68. */
  69. @SuppressWarnings("PMD.AvoidDuplicateLiterals")
  70. public class PreviewImageActivity extends FileActivity implements
  71. FileFragment.ContainerActivity,
  72. ViewPager.OnPageChangeListener,
  73. OnRemoteOperationListener,
  74. Injectable {
  75. public static final String TAG = PreviewImageActivity.class.getSimpleName();
  76. private static final String KEY_WAITING_FOR_BINDER = "WAITING_FOR_BINDER";
  77. private static final String KEY_SYSTEM_VISIBLE = "TRUE";
  78. public static final String EXTRA_VIRTUAL_TYPE = "EXTRA_VIRTUAL_TYPE";
  79. private ViewPager mViewPager;
  80. private PreviewImagePagerAdapter mPreviewImagePagerAdapter;
  81. private int mSavedPosition;
  82. private boolean mHasSavedPosition;
  83. private boolean mRequestWaitingForBinder;
  84. private DownloadFinishReceiver mDownloadFinishReceiver;
  85. private View mFullScreenAnchorView;
  86. @Inject AppPreferences preferences;
  87. @Override
  88. protected void onCreate(Bundle savedInstanceState) {
  89. super.onCreate(savedInstanceState);
  90. final ActionBar actionBar = getSupportActionBar();
  91. if (savedInstanceState != null && !savedInstanceState.getBoolean(KEY_SYSTEM_VISIBLE, true) &&
  92. actionBar != null) {
  93. actionBar.hide();
  94. }
  95. setContentView(R.layout.preview_image_activity);
  96. // Navigation Drawer
  97. setupDrawer();
  98. // ActionBar
  99. updateActionBarTitleAndHomeButton(null);
  100. ThemeUtils.tintBackButton(actionBar, this, Color.WHITE);
  101. mFullScreenAnchorView = getWindow().getDecorView();
  102. // to keep our UI controls visibility in line with system bars visibility
  103. setDrawerLockMode(DrawerLayout.LOCK_MODE_LOCKED_CLOSED);
  104. if (savedInstanceState != null) {
  105. mRequestWaitingForBinder = savedInstanceState.getBoolean(KEY_WAITING_FOR_BINDER);
  106. } else {
  107. mRequestWaitingForBinder = false;
  108. }
  109. }
  110. private void initViewPager(User user) {
  111. // virtual folder
  112. if (getIntent().getSerializableExtra(EXTRA_VIRTUAL_TYPE) != null) {
  113. VirtualFolderType type = (VirtualFolderType) getIntent().getSerializableExtra(EXTRA_VIRTUAL_TYPE);
  114. mPreviewImagePagerAdapter = new PreviewImagePagerAdapter(getSupportFragmentManager(),
  115. type,
  116. user,
  117. getStorageManager());
  118. } else {
  119. // get parent from path
  120. OCFile parentFolder = getStorageManager().getFileById(getFile().getParentId());
  121. if (parentFolder == null) {
  122. // should not be necessary
  123. parentFolder = getStorageManager().getFileByPath(OCFile.ROOT_PATH);
  124. }
  125. mPreviewImagePagerAdapter = new PreviewImagePagerAdapter(
  126. getSupportFragmentManager(),
  127. parentFolder,
  128. user,
  129. getStorageManager(),
  130. MainApp.isOnlyOnDevice(),
  131. preferences
  132. );
  133. }
  134. mViewPager = findViewById(R.id.fragmentPager);
  135. int position = mHasSavedPosition ? mSavedPosition : mPreviewImagePagerAdapter.getFilePosition(getFile());
  136. position = position >= 0 ? position : 0;
  137. mViewPager.setAdapter(mPreviewImagePagerAdapter);
  138. mViewPager.addOnPageChangeListener(this);
  139. mViewPager.setCurrentItem(position);
  140. if (position == 0 && !getFile().isDown()) {
  141. // this is necessary because mViewPager.setCurrentItem(0) just after setting the
  142. // adapter does not result in a call to #onPageSelected(0)
  143. mRequestWaitingForBinder = true;
  144. }
  145. }
  146. @Override
  147. public void onStart() {
  148. super.onStart();
  149. Optional<User> optionalUser = getUser();
  150. if (optionalUser.isPresent()) {
  151. OCFile file = getFile();
  152. /// Validate handled file (first image to preview)
  153. if (file == null) {
  154. throw new IllegalStateException("Instanced with a NULL OCFile");
  155. }
  156. if (!MimeTypeUtil.isImage(file)) {
  157. throw new IllegalArgumentException("Non-image file passed as argument");
  158. }
  159. // Update file according to DB file, if it is possible
  160. if (file.getFileId() > FileDataStorageManager.ROOT_PARENT_ID) {
  161. file = getStorageManager().getFileById(file.getFileId());
  162. }
  163. if (file != null) {
  164. /// Refresh the activity according to the Account and OCFile set
  165. setFile(file); // reset after getting it fresh from storageManager
  166. getSupportActionBar().setTitle(getFile().getFileName());
  167. //if (!stateWasRecovered) {
  168. initViewPager(optionalUser.get());
  169. //}
  170. } else {
  171. // handled file not in the current Account
  172. finish();
  173. }
  174. }
  175. }
  176. @Override
  177. protected void onSaveInstanceState(@NonNull Bundle outState) {
  178. super.onSaveInstanceState(outState);
  179. outState.putBoolean(KEY_WAITING_FOR_BINDER, mRequestWaitingForBinder);
  180. outState.putBoolean(KEY_SYSTEM_VISIBLE, isSystemUIVisible());
  181. }
  182. @Override
  183. public void onRemoteOperationFinish(RemoteOperation operation, RemoteOperationResult result) {
  184. super.onRemoteOperationFinish(operation, result);
  185. if (operation instanceof RemoveFileOperation) {
  186. finish();
  187. } else if (operation instanceof SynchronizeFileOperation) {
  188. onSynchronizeFileOperationFinish(result);
  189. }
  190. }
  191. private void onSynchronizeFileOperationFinish(RemoteOperationResult result) {
  192. if (result.isSuccess()) {
  193. supportInvalidateOptionsMenu();
  194. }
  195. }
  196. @Override
  197. protected ServiceConnection newTransferenceServiceConnection() {
  198. return new PreviewImageServiceConnection();
  199. }
  200. /** Defines callbacks for service binding, passed to bindService() */
  201. private class PreviewImageServiceConnection implements ServiceConnection {
  202. @Override
  203. public void onServiceConnected(ComponentName component, IBinder service) {
  204. if (component.equals(new ComponentName(PreviewImageActivity.this,
  205. FileDownloader.class))) {
  206. mDownloaderBinder = (FileDownloaderBinder) service;
  207. if (mRequestWaitingForBinder) {
  208. mRequestWaitingForBinder = false;
  209. Log_OC.d(TAG, "Simulating reselection of current page after connection " +
  210. "of download binder");
  211. onPageSelected(mViewPager.getCurrentItem());
  212. }
  213. } else if (component.equals(new ComponentName(PreviewImageActivity.this,
  214. FileUploader.class))) {
  215. Log_OC.d(TAG, "Upload service connected");
  216. mUploaderBinder = (FileUploaderBinder) service;
  217. }
  218. }
  219. @Override
  220. public void onServiceDisconnected(ComponentName component) {
  221. if (component.equals(new ComponentName(PreviewImageActivity.this,
  222. FileDownloader.class))) {
  223. Log_OC.d(TAG, "Download service suddenly disconnected");
  224. mDownloaderBinder = null;
  225. } else if (component.equals(new ComponentName(PreviewImageActivity.this,
  226. FileUploader.class))) {
  227. Log_OC.d(TAG, "Upload service suddenly disconnected");
  228. mUploaderBinder = null;
  229. }
  230. }
  231. }
  232. @Override
  233. public void onStop() {
  234. super.onStop();
  235. }
  236. @Override
  237. public void onDestroy() {
  238. super.onDestroy();
  239. }
  240. @Override
  241. public boolean onOptionsItemSelected(MenuItem item) {
  242. boolean returnValue = false;
  243. switch(item.getItemId()){
  244. case android.R.id.home:
  245. if (isDrawerOpen()) {
  246. closeDrawer();
  247. } else {
  248. backToDisplayActivity();
  249. }
  250. returnValue = true;
  251. break;
  252. default:
  253. returnValue = super.onOptionsItemSelected(item);
  254. break;
  255. }
  256. return returnValue;
  257. }
  258. @Override
  259. protected void onResume() {
  260. super.onResume();
  261. mDownloadFinishReceiver = new DownloadFinishReceiver();
  262. IntentFilter filter = new IntentFilter(FileDownloader.getDownloadFinishMessage());
  263. filter.addAction(FileDownloader.getDownloadAddedMessage());
  264. registerReceiver(mDownloadFinishReceiver, filter);
  265. }
  266. @Override
  267. protected void onPostResume() {
  268. super.onPostResume();
  269. }
  270. @Override
  271. public void onPause() {
  272. if (mDownloadFinishReceiver != null){
  273. unregisterReceiver(mDownloadFinishReceiver);
  274. mDownloadFinishReceiver = null;
  275. }
  276. super.onPause();
  277. }
  278. private void backToDisplayActivity() {
  279. finish();
  280. }
  281. @SuppressFBWarnings("DLS")
  282. @Override
  283. public void showDetails(OCFile file) {
  284. final Intent showDetailsIntent = new Intent(this, FileDisplayActivity.class);
  285. showDetailsIntent.setAction(FileDisplayActivity.ACTION_DETAILS);
  286. showDetailsIntent.putExtra(FileActivity.EXTRA_FILE, file);
  287. showDetailsIntent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
  288. startActivity(showDetailsIntent);
  289. finish();
  290. }
  291. @Override
  292. public void showDetails(OCFile file, int activeTab) {
  293. showDetails(file);
  294. }
  295. public void requestForDownload(OCFile file) {
  296. if (mDownloaderBinder == null) {
  297. Log_OC.d(TAG, "requestForDownload called without binder to download service");
  298. } else if (!mDownloaderBinder.isDownloading(getAccount(), file)) {
  299. final User user = getUser().orElseThrow(RuntimeException::new);
  300. Intent i = new Intent(this, FileDownloader.class);
  301. i.putExtra(FileDownloader.EXTRA_USER, user);
  302. i.putExtra(FileDownloader.EXTRA_FILE, file);
  303. startService(i);
  304. }
  305. }
  306. /**
  307. * This method will be invoked when a new page becomes selected. Animation is not necessarily
  308. * complete.
  309. *
  310. * @param position Position index of the new selected page
  311. */
  312. @Override
  313. public void onPageSelected(int position) {
  314. mSavedPosition = position;
  315. mHasSavedPosition = true;
  316. if (mDownloaderBinder == null) {
  317. mRequestWaitingForBinder = true;
  318. } else {
  319. OCFile currentFile = mPreviewImagePagerAdapter.getFileAt(position);
  320. if (currentFile != null) {
  321. if (getSupportActionBar() != null) {
  322. getSupportActionBar().setTitle(currentFile.getFileName());
  323. }
  324. setDrawerIndicatorEnabled(false);
  325. if (currentFile.isEncrypted() && !currentFile.isDown() &&
  326. !mPreviewImagePagerAdapter.pendingErrorAt(position)) {
  327. requestForDownload(currentFile);
  328. }
  329. // Call to reset image zoom to initial state
  330. // ((PreviewImagePagerAdapter) mViewPager.getAdapter()).resetZoom();
  331. }
  332. }
  333. }
  334. /**
  335. * Called when the scroll state changes. Useful for discovering when the user begins dragging,
  336. * when the pager is automatically settling to the current page. when it is fully stopped/idle.
  337. *
  338. * @param state The new scroll state (SCROLL_STATE_IDLE, _DRAGGING, _SETTLING
  339. */
  340. @Override
  341. public void onPageScrollStateChanged(int state) {
  342. // not used at the moment
  343. }
  344. /**
  345. * This method will be invoked when the current page is scrolled, either as part of a
  346. * programmatically initiated smooth scroll or a user initiated touch scroll.
  347. *
  348. * @param position Position index of the first page currently being displayed.
  349. * Page position+1 will be visible if positionOffset is
  350. * nonzero.
  351. * @param positionOffset Value from [0, 1) indicating the offset from the page
  352. * at position.
  353. * @param positionOffsetPixels Value in pixels indicating the offset from position.
  354. */
  355. @Override
  356. public void onPageScrolled(int position, float positionOffset, int positionOffsetPixels) {
  357. // not used at the moment
  358. }
  359. /**
  360. * Class waiting for broadcast events from the {@link FileDownloader} service.
  361. *
  362. * Updates the UI when a download is started or finished, provided that it is relevant for the
  363. * folder displayed in the gallery.
  364. */
  365. private class DownloadFinishReceiver extends BroadcastReceiver {
  366. @Override
  367. public void onReceive(Context context, Intent intent) {
  368. String accountName = intent.getStringExtra(FileDownloader.ACCOUNT_NAME);
  369. String downloadedRemotePath = intent.getStringExtra(FileDownloader.EXTRA_REMOTE_PATH);
  370. if (getAccount().name.equals(accountName) &&
  371. downloadedRemotePath != null) {
  372. OCFile file = getStorageManager().getFileByPath(downloadedRemotePath);
  373. int position = mPreviewImagePagerAdapter.getFilePosition(file);
  374. boolean downloadWasFine = intent.getBooleanExtra(
  375. FileDownloader.EXTRA_DOWNLOAD_RESULT, false);
  376. //boolean isOffscreen = Math.abs((mViewPager.getCurrentItem() - position))
  377. // <= mViewPager.getOffscreenPageLimit();
  378. if (position >= 0 &&
  379. intent.getAction().equals(FileDownloader.getDownloadFinishMessage())) {
  380. if (downloadWasFine) {
  381. mPreviewImagePagerAdapter.updateFile(position, file);
  382. } else {
  383. mPreviewImagePagerAdapter.updateWithDownloadError(position);
  384. }
  385. mPreviewImagePagerAdapter.notifyDataSetChanged(); // will trigger the creation
  386. // of new fragments
  387. } else {
  388. Log_OC.d(TAG, "Download finished, but the fragment is offscreen");
  389. }
  390. }
  391. removeStickyBroadcast(intent);
  392. }
  393. }
  394. public boolean isSystemUIVisible() {
  395. return getSupportActionBar() == null || getSupportActionBar().isShowing();
  396. }
  397. public void toggleFullScreen() {
  398. boolean visible = (mFullScreenAnchorView.getSystemUiVisibility()
  399. & View.SYSTEM_UI_FLAG_HIDE_NAVIGATION) == 0;
  400. if (visible) {
  401. hideSystemUI(mFullScreenAnchorView);
  402. // actionBar.hide(); // propagated through
  403. // OnSystemUiVisibilityChangeListener()
  404. } else {
  405. showSystemUI(mFullScreenAnchorView);
  406. // actionBar.show(); // propagated through
  407. // OnSystemUiVisibilityChangeListener()
  408. }
  409. }
  410. public void switchToFullScreen() {
  411. hideSystemUI(mFullScreenAnchorView);
  412. }
  413. @Override
  414. public void onBrowsedDownTo(OCFile folder) {
  415. // TODO Auto-generated method stub
  416. }
  417. @Override
  418. public void onTransferStateChanged(OCFile file, boolean downloading, boolean uploading) {
  419. // TODO Auto-generated method stub
  420. }
  421. @SuppressLint("InlinedApi")
  422. private void hideSystemUI(View anchorView) {
  423. anchorView.setSystemUiVisibility(
  424. View.SYSTEM_UI_FLAG_HIDE_NAVIGATION // hides NAVIGATION BAR; Android >= 4.0
  425. | View.SYSTEM_UI_FLAG_FULLSCREEN // hides STATUS BAR; Android >= 4.1
  426. | View.SYSTEM_UI_FLAG_IMMERSIVE // stays interactive; Android >= 4.4
  427. | View.SYSTEM_UI_FLAG_LAYOUT_STABLE // draw full window; Android >= 4.1
  428. | View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN // draw full window; Android >= 4.1
  429. | View.SYSTEM_UI_FLAG_LAYOUT_HIDE_NAVIGATION // draw full window; Android >= 4.1
  430. );
  431. }
  432. @SuppressLint("InlinedApi")
  433. private void showSystemUI(View anchorView) {
  434. anchorView.setSystemUiVisibility(
  435. View.SYSTEM_UI_FLAG_LAYOUT_STABLE // draw full window; Android >= 4.1
  436. | View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN // draw full window; Android >= 4.1
  437. | View.SYSTEM_UI_FLAG_LAYOUT_HIDE_NAVIGATION // draw full window; Android >= 4.
  438. );
  439. }
  440. }