FileDisplayActivity.java 52 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264
  1. /* ownCloud Android client application
  2. * Copyright (C) 2011 Bartek Przybylski
  3. * Copyright (C) 2012-2013 ownCloud Inc.
  4. *
  5. * This program is free software: you can redistribute it and/or modify
  6. * it under the terms of the GNU General Public License as published by
  7. * the Free Software Foundation, either version 2 of the License, or
  8. * (at your option) any later version.
  9. *
  10. * This program is distributed in the hope that it will be useful,
  11. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  12. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  13. * GNU General Public License for more details.
  14. *
  15. * You should have received a copy of the GNU General Public License
  16. * along with this program. If not, see <http://www.gnu.org/licenses/>.
  17. *
  18. */
  19. package com.owncloud.android.ui.activity;
  20. import java.io.File;
  21. import android.accounts.Account;
  22. import android.app.AlertDialog;
  23. import android.app.AlertDialog.Builder;
  24. import android.app.Dialog;
  25. import android.app.ProgressDialog;
  26. import android.content.BroadcastReceiver;
  27. import android.content.ComponentName;
  28. import android.content.ContentResolver;
  29. import android.content.Context;
  30. import android.content.DialogInterface;
  31. import android.content.DialogInterface.OnClickListener;
  32. import android.content.Intent;
  33. import android.content.IntentFilter;
  34. import android.content.ServiceConnection;
  35. import android.content.SharedPreferences;
  36. import android.content.SharedPreferences.Editor;
  37. import android.content.pm.PackageInfo;
  38. import android.content.pm.PackageManager.NameNotFoundException;
  39. import android.content.res.Resources.NotFoundException;
  40. import android.database.Cursor;
  41. import android.graphics.Bitmap;
  42. import android.graphics.drawable.BitmapDrawable;
  43. import android.net.Uri;
  44. import android.os.Bundle;
  45. import android.os.Handler;
  46. import android.os.IBinder;
  47. import android.preference.PreferenceManager;
  48. import android.provider.MediaStore;
  49. import android.support.v4.app.FragmentTransaction;
  50. import android.util.Log;
  51. import android.view.View;
  52. import android.view.ViewGroup;
  53. import android.widget.ArrayAdapter;
  54. import android.widget.EditText;
  55. import android.widget.TextView;
  56. import android.widget.Toast;
  57. import com.actionbarsherlock.app.ActionBar;
  58. import com.actionbarsherlock.app.ActionBar.OnNavigationListener;
  59. import com.actionbarsherlock.app.SherlockFragmentActivity;
  60. import com.actionbarsherlock.view.Menu;
  61. import com.actionbarsherlock.view.MenuInflater;
  62. import com.actionbarsherlock.view.MenuItem;
  63. import com.actionbarsherlock.view.Window;
  64. import com.owncloud.android.AccountUtils;
  65. import com.owncloud.android.R;
  66. import com.owncloud.android.authenticator.AccountAuthenticator;
  67. import com.owncloud.android.datamodel.DataStorageManager;
  68. import com.owncloud.android.datamodel.FileDataStorageManager;
  69. import com.owncloud.android.datamodel.OCFile;
  70. import com.owncloud.android.files.services.FileDownloader;
  71. import com.owncloud.android.files.services.FileDownloader.FileDownloaderBinder;
  72. import com.owncloud.android.files.services.FileObserverService;
  73. import com.owncloud.android.files.services.FileUploader;
  74. import com.owncloud.android.files.services.FileUploader.FileUploaderBinder;
  75. import com.owncloud.android.network.OwnCloudClientUtils;
  76. import com.owncloud.android.operations.OnRemoteOperationListener;
  77. import com.owncloud.android.operations.RemoteOperation;
  78. import com.owncloud.android.operations.RemoteOperationResult;
  79. import com.owncloud.android.operations.RemoteOperationResult.ResultCode;
  80. import com.owncloud.android.operations.RemoveFileOperation;
  81. import com.owncloud.android.operations.RenameFileOperation;
  82. import com.owncloud.android.operations.SynchronizeFileOperation;
  83. import com.owncloud.android.syncadapter.FileSyncService;
  84. import com.owncloud.android.ui.dialog.ChangelogDialog;
  85. import com.owncloud.android.ui.dialog.SslValidatorDialog;
  86. import com.owncloud.android.ui.dialog.SslValidatorDialog.OnSslValidatorListener;
  87. import com.owncloud.android.ui.fragment.FileDetailFragment;
  88. import com.owncloud.android.ui.fragment.OCFileListFragment;
  89. import eu.alefzero.webdav.WebdavClient;
  90. /**
  91. * Displays, what files the user has available in his ownCloud.
  92. *
  93. * @author Bartek Przybylski
  94. *
  95. */
  96. public class FileDisplayActivity extends SherlockFragmentActivity implements OCFileListFragment.ContainerActivity,
  97. FileDetailFragment.ContainerActivity, OnNavigationListener, OnSslValidatorListener, OnRemoteOperationListener {
  98. private ArrayAdapter<String> mDirectories;
  99. private OCFile mCurrentDir = null;
  100. private OCFile mCurrentFile = null;
  101. private DataStorageManager mStorageManager;
  102. private SyncBroadcastReceiver mSyncBroadcastReceiver;
  103. private UploadFinishReceiver mUploadFinishReceiver;
  104. private DownloadFinishReceiver mDownloadFinishReceiver;
  105. private FileDownloaderBinder mDownloaderBinder = null;
  106. private FileUploaderBinder mUploaderBinder = null;
  107. private ServiceConnection mDownloadConnection = null, mUploadConnection = null;
  108. private RemoteOperationResult mLastSslUntrustedServerResult = null;
  109. private OCFileListFragment mFileList;
  110. private boolean mDualPane;
  111. private static final int DIALOG_SETUP_ACCOUNT = 0;
  112. private static final int DIALOG_CREATE_DIR = 1;
  113. public static final int DIALOG_SHORT_WAIT = 3;
  114. private static final int DIALOG_CHOOSE_UPLOAD_SOURCE = 4;
  115. private static final int DIALOG_SSL_VALIDATOR = 5;
  116. private static final int DIALOG_CERT_NOT_SAVED = 6;
  117. private static final String DIALOG_CHANGELOG_TAG = "DIALOG_CHANGELOG";
  118. private static final int ACTION_SELECT_CONTENT_FROM_APPS = 1;
  119. private static final int ACTION_SELECT_MULTIPLE_FILES = 2;
  120. private static final int ACTION_SELECT_FAILED_INSTANT_UPLOAD = 2;
  121. private static final String TAG = "FileDisplayActivity";
  122. @Override
  123. public void onCreate(Bundle savedInstanceState) {
  124. Log.d(getClass().toString(), "onCreate() start");
  125. super.onCreate(savedInstanceState);
  126. /// Load of parameters from received intent
  127. mCurrentDir = getIntent().getParcelableExtra(FileDetailFragment.EXTRA_FILE); // no check necessary, mCurrenDir == null if the parameter is not in the intent
  128. Account account = getIntent().getParcelableExtra(FileDetailFragment.EXTRA_ACCOUNT);
  129. if (account != null)
  130. AccountUtils.setCurrentOwnCloudAccount(this, account.name);
  131. /// Load of saved instance state: keep this always before initDataFromCurrentAccount()
  132. if(savedInstanceState != null) {
  133. // TODO - test if savedInstanceState should take precedence over file in the intent ALWAYS (now), NEVER, or SOME TIMES
  134. mCurrentDir = savedInstanceState.getParcelable(FileDetailFragment.EXTRA_FILE);
  135. }
  136. if (!AccountUtils.accountsAreSetup(this)) {
  137. /// no account available: FORCE ACCOUNT CREATION
  138. mStorageManager = null;
  139. createFirstAccount();
  140. } else { // / at least an account is available
  141. initDataFromCurrentAccount(); // it checks mCurrentDir and
  142. // mCurrentFile with the current
  143. // account
  144. }
  145. mUploadConnection = new ListServiceConnection();
  146. mDownloadConnection = new ListServiceConnection();
  147. bindService(new Intent(this, FileUploader.class), mUploadConnection, Context.BIND_AUTO_CREATE);
  148. bindService(new Intent(this, FileDownloader.class), mDownloadConnection, Context.BIND_AUTO_CREATE);
  149. // PIN CODE request ; best location is to decide, let's try this first
  150. if (getIntent().getAction() != null && getIntent().getAction().equals(Intent.ACTION_MAIN) && savedInstanceState == null) {
  151. requestPinCode();
  152. }
  153. // file observer
  154. Intent observer_intent = new Intent(this, FileObserverService.class);
  155. observer_intent.putExtra(FileObserverService.KEY_FILE_CMD, FileObserverService.CMD_INIT_OBSERVED_LIST);
  156. startService(observer_intent);
  157. /// USER INTERFACE
  158. requestWindowFeature(Window.FEATURE_INDETERMINATE_PROGRESS);
  159. // Drop-down navigation
  160. mDirectories = new CustomArrayAdapter<String>(this, R.layout.sherlock_spinner_dropdown_item);
  161. OCFile currFile = mCurrentDir;
  162. while (currFile != null && currFile.getFileName() != OCFile.PATH_SEPARATOR) {
  163. mDirectories.add(currFile.getFileName());
  164. currFile = mStorageManager.getFileById(currFile.getParentId());
  165. }
  166. mDirectories.add(OCFile.PATH_SEPARATOR);
  167. // Inflate and set the layout view
  168. setContentView(R.layout.files);
  169. mFileList = (OCFileListFragment) getSupportFragmentManager().findFragmentById(R.id.fileList);
  170. mDualPane = (findViewById(R.id.file_details_container) != null);
  171. if (mDualPane) {
  172. initFileDetailsInDualPane();
  173. }
  174. // Action bar setup
  175. ActionBar actionBar = getSupportActionBar();
  176. actionBar.setHomeButtonEnabled(true); // mandatory since Android ICS, according to the official documentation
  177. actionBar.setDisplayHomeAsUpEnabled(mCurrentDir != null && mCurrentDir.getParentId() != 0);
  178. actionBar.setDisplayShowTitleEnabled(false);
  179. actionBar.setNavigationMode(ActionBar.NAVIGATION_MODE_LIST);
  180. actionBar.setListNavigationCallbacks(mDirectories, this);
  181. setSupportProgressBarIndeterminateVisibility(false); // always AFTER setContentView(...) ; to workaround bug in its implementation
  182. // show changelog, if needed
  183. showChangeLog();
  184. Log.d(getClass().toString(), "onCreate() end");
  185. }
  186. /**
  187. * Shows a dialog with the change log of the current version after each app update
  188. *
  189. * TODO make it permanent; by now, only to advice the workaround app for 4.1.x
  190. */
  191. private void showChangeLog() {
  192. if (android.os.Build.VERSION.SDK_INT == android.os.Build.VERSION_CODES.JELLY_BEAN) {
  193. final String KEY_VERSION = "version";
  194. SharedPreferences sharedPref = PreferenceManager.getDefaultSharedPreferences(getApplicationContext());
  195. int currentVersionNumber = 0;
  196. int savedVersionNumber = sharedPref.getInt(KEY_VERSION, 0);
  197. try {
  198. PackageInfo pi = getPackageManager().getPackageInfo(getPackageName(), 0);
  199. currentVersionNumber = pi.versionCode;
  200. } catch (Exception e) {}
  201. if (currentVersionNumber > savedVersionNumber) {
  202. ChangelogDialog.newInstance(true).show(getSupportFragmentManager(), DIALOG_CHANGELOG_TAG);
  203. Editor editor = sharedPref.edit();
  204. editor.putInt(KEY_VERSION, currentVersionNumber);
  205. editor.commit();
  206. }
  207. }
  208. }
  209. /**
  210. * Launches the account creation activity. To use when no ownCloud account is available
  211. */
  212. private void createFirstAccount() {
  213. Intent intent = new Intent(android.provider.Settings.ACTION_ADD_ACCOUNT);
  214. intent.putExtra(android.provider.Settings.EXTRA_AUTHORITIES, new String[] { AccountAuthenticator.AUTH_TOKEN_TYPE });
  215. startActivity(intent); // the new activity won't be created until this.onStart() and this.onResume() are finished;
  216. }
  217. /**
  218. * Load of state dependent of the existence of an ownCloud account
  219. */
  220. private void initDataFromCurrentAccount() {
  221. /// Storage manager initialization - access to local database
  222. mStorageManager = new FileDataStorageManager(
  223. AccountUtils.getCurrentOwnCloudAccount(this),
  224. getContentResolver());
  225. /// Check if mCurrentDir is a directory
  226. if(mCurrentDir != null && !mCurrentDir.isDirectory()) {
  227. mCurrentFile = mCurrentDir;
  228. mCurrentDir = mStorageManager.getFileById(mCurrentDir.getParentId());
  229. }
  230. /// Check if mCurrentDir and mCurrentFile are in the current account, and update them
  231. if (mCurrentDir != null) {
  232. mCurrentDir = mStorageManager.getFileByPath(mCurrentDir.getRemotePath()); // mCurrentDir == null if it is not in the current account
  233. }
  234. if (mCurrentFile != null) {
  235. if (mCurrentFile.fileExists()) {
  236. mCurrentFile = mStorageManager.getFileByPath(mCurrentFile.getRemotePath()); // mCurrentFile == null if it is not in the current account
  237. } // else : keep mCurrentFile with the received value; this is currently the case of an upload in progress, when the user presses the status notification in a landscape tablet
  238. }
  239. /// Default to root if mCurrentDir was not found
  240. if (mCurrentDir == null) {
  241. mCurrentDir = mStorageManager.getFileByPath("/"); // will be NULL if the database was never synchronized
  242. }
  243. }
  244. private void initFileDetailsInDualPane() {
  245. if (mDualPane && getSupportFragmentManager().findFragmentByTag(FileDetailFragment.FTAG) == null) {
  246. FragmentTransaction transaction = getSupportFragmentManager().beginTransaction();
  247. if (mCurrentFile != null) {
  248. transaction.replace(R.id.file_details_container,
  249. new FileDetailFragment(mCurrentFile, AccountUtils.getCurrentOwnCloudAccount(this)),
  250. FileDetailFragment.FTAG); // empty FileDetailFragment
  251. mCurrentFile = null;
  252. } else {
  253. transaction.replace(R.id.file_details_container, new FileDetailFragment(null, null),
  254. FileDetailFragment.FTAG); // empty FileDetailFragment
  255. }
  256. transaction.commit();
  257. }
  258. }
  259. @Override
  260. public void onDestroy() {
  261. super.onDestroy();
  262. if (mDownloadConnection != null)
  263. unbindService(mDownloadConnection);
  264. if (mUploadConnection != null)
  265. unbindService(mUploadConnection);
  266. }
  267. @Override
  268. public boolean onCreateOptionsMenu(Menu menu) {
  269. MenuInflater inflater = getSherlock().getMenuInflater();
  270. inflater.inflate(R.menu.menu, menu);
  271. return true;
  272. }
  273. @Override
  274. public boolean onOptionsItemSelected(MenuItem item) {
  275. boolean retval = true;
  276. switch (item.getItemId()) {
  277. case R.id.createDirectoryItem: {
  278. showDialog(DIALOG_CREATE_DIR);
  279. break;
  280. }
  281. case R.id.startSync: {
  282. startSynchronization();
  283. break;
  284. }
  285. case R.id.action_upload: {
  286. showDialog(DIALOG_CHOOSE_UPLOAD_SOURCE);
  287. break;
  288. }
  289. case R.id.action_settings: {
  290. Intent settingsIntent = new Intent(this, Preferences.class);
  291. startActivity(settingsIntent);
  292. break;
  293. }
  294. case android.R.id.home: {
  295. if (mCurrentDir != null && mCurrentDir.getParentId() != 0) {
  296. onBackPressed();
  297. }
  298. break;
  299. }
  300. default:
  301. retval = super.onOptionsItemSelected(item);
  302. }
  303. return retval;
  304. }
  305. private void startSynchronization() {
  306. ContentResolver.cancelSync(null, AccountAuthenticator.AUTH_TOKEN_TYPE); // cancel the current synchronizations of any ownCloud account
  307. Bundle bundle = new Bundle();
  308. bundle.putBoolean(ContentResolver.SYNC_EXTRAS_MANUAL, true);
  309. ContentResolver.requestSync(
  310. AccountUtils.getCurrentOwnCloudAccount(this),
  311. AccountAuthenticator.AUTH_TOKEN_TYPE, bundle);
  312. }
  313. @Override
  314. public boolean onNavigationItemSelected(int itemPosition, long itemId) {
  315. int i = itemPosition;
  316. while (i-- != 0) {
  317. onBackPressed();
  318. }
  319. // the next operation triggers a new call to this method, but it's necessary to
  320. // ensure that the name exposed in the action bar is the current directory when the
  321. // user selected it in the navigation list
  322. if (itemPosition != 0)
  323. getSupportActionBar().setSelectedNavigationItem(0);
  324. return true;
  325. }
  326. /**
  327. * Called, when the user selected something for uploading
  328. */
  329. public void onActivityResult(int requestCode, int resultCode, Intent data) {
  330. if (requestCode == ACTION_SELECT_CONTENT_FROM_APPS
  331. && (resultCode == RESULT_OK || resultCode == UploadFilesActivity.RESULT_OK_AND_MOVE)) {
  332. requestSimpleUpload(data, resultCode);
  333. } else if (requestCode == ACTION_SELECT_MULTIPLE_FILES
  334. && (resultCode == RESULT_OK || resultCode == UploadFilesActivity.RESULT_OK_AND_MOVE)) {
  335. requestMultipleUpload(data, resultCode);
  336. }
  337. }
  338. private void requestMultipleUpload(Intent data, int resultCode) {
  339. String[] filePaths = data.getStringArrayExtra(UploadFilesActivity.EXTRA_CHOSEN_FILES);
  340. if (filePaths != null) {
  341. String[] remotePaths = new String[filePaths.length];
  342. String remotePathBase = "";
  343. for (int j = mDirectories.getCount() - 2; j >= 0; --j) {
  344. remotePathBase += OCFile.PATH_SEPARATOR + mDirectories.getItem(j);
  345. }
  346. if (!remotePathBase.endsWith(OCFile.PATH_SEPARATOR))
  347. remotePathBase += OCFile.PATH_SEPARATOR;
  348. for (int j = 0; j < remotePaths.length; j++) {
  349. remotePaths[j] = remotePathBase + (new File(filePaths[j])).getName();
  350. }
  351. Intent i = new Intent(this, FileUploader.class);
  352. i.putExtra(FileUploader.KEY_ACCOUNT, AccountUtils.getCurrentOwnCloudAccount(this));
  353. i.putExtra(FileUploader.KEY_LOCAL_FILE, filePaths);
  354. i.putExtra(FileUploader.KEY_REMOTE_FILE, remotePaths);
  355. i.putExtra(FileUploader.KEY_UPLOAD_TYPE, FileUploader.UPLOAD_MULTIPLE_FILES);
  356. if (resultCode == UploadFilesActivity.RESULT_OK_AND_MOVE)
  357. i.putExtra(FileUploader.KEY_LOCAL_BEHAVIOUR, FileUploader.LOCAL_BEHAVIOUR_MOVE);
  358. startService(i);
  359. } else {
  360. Log.d("FileDisplay", "User clicked on 'Update' with no selection");
  361. Toast t = Toast.makeText(this, getString(R.string.filedisplay_no_file_selected), Toast.LENGTH_LONG);
  362. t.show();
  363. return;
  364. }
  365. }
  366. private void requestSimpleUpload(Intent data, int resultCode) {
  367. String filepath = null;
  368. try {
  369. Uri selectedImageUri = data.getData();
  370. String filemanagerstring = selectedImageUri.getPath();
  371. String selectedImagePath = getPath(selectedImageUri);
  372. if (selectedImagePath != null)
  373. filepath = selectedImagePath;
  374. else
  375. filepath = filemanagerstring;
  376. } catch (Exception e) {
  377. Log.e("FileDisplay", "Unexpected exception when trying to read the result of Intent.ACTION_GET_CONTENT", e);
  378. e.printStackTrace();
  379. } finally {
  380. if (filepath == null) {
  381. Log.e("FileDisplay", "Couldnt resolve path to file");
  382. Toast t = Toast.makeText(this, getString(R.string.filedisplay_unexpected_bad_get_content), Toast.LENGTH_LONG);
  383. t.show();
  384. return;
  385. }
  386. }
  387. Intent i = new Intent(this, FileUploader.class);
  388. i.putExtra(FileUploader.KEY_ACCOUNT, AccountUtils.getCurrentOwnCloudAccount(this));
  389. String remotepath = new String();
  390. for (int j = mDirectories.getCount() - 2; j >= 0; --j) {
  391. remotepath += OCFile.PATH_SEPARATOR + mDirectories.getItem(j);
  392. }
  393. if (!remotepath.endsWith(OCFile.PATH_SEPARATOR))
  394. remotepath += OCFile.PATH_SEPARATOR;
  395. remotepath += new File(filepath).getName();
  396. i.putExtra(FileUploader.KEY_LOCAL_FILE, filepath);
  397. i.putExtra(FileUploader.KEY_REMOTE_FILE, remotepath);
  398. i.putExtra(FileUploader.KEY_UPLOAD_TYPE, FileUploader.UPLOAD_SINGLE_FILE);
  399. if (resultCode == UploadFilesActivity.RESULT_OK_AND_MOVE)
  400. i.putExtra(FileUploader.KEY_LOCAL_BEHAVIOUR, FileUploader.LOCAL_BEHAVIOUR_MOVE);
  401. startService(i);
  402. }
  403. @Override
  404. public void onBackPressed() {
  405. if (mDirectories.getCount() <= 1) {
  406. finish();
  407. return;
  408. }
  409. popDirname();
  410. mFileList.onNavigateUp();
  411. mCurrentDir = mFileList.getCurrentFile();
  412. if (mDualPane) {
  413. // Resets the FileDetailsFragment on Tablets so that it always
  414. // displays
  415. FileDetailFragment fileDetails = (FileDetailFragment) getSupportFragmentManager().findFragmentByTag(
  416. FileDetailFragment.FTAG);
  417. if (fileDetails != null && !fileDetails.isEmpty()) {
  418. FragmentTransaction transaction = getSupportFragmentManager().beginTransaction();
  419. transaction.remove(fileDetails);
  420. transaction.add(R.id.file_details_container, new FileDetailFragment(null, null),
  421. FileDetailFragment.FTAG);
  422. transaction.commit();
  423. }
  424. }
  425. if (mCurrentDir.getParentId() == 0) {
  426. ActionBar actionBar = getSupportActionBar();
  427. actionBar.setDisplayHomeAsUpEnabled(false);
  428. }
  429. }
  430. @Override
  431. protected void onSaveInstanceState(Bundle outState) {
  432. // responsibility of restore is preferred in onCreate() before than in onRestoreInstanceState when there are Fragments involved
  433. Log.d(getClass().toString(), "onSaveInstanceState() start");
  434. super.onSaveInstanceState(outState);
  435. outState.putParcelable(FileDetailFragment.EXTRA_FILE, mCurrentDir);
  436. if (mDualPane) {
  437. FileDetailFragment fragment = (FileDetailFragment) getSupportFragmentManager().findFragmentByTag(FileDetailFragment.FTAG);
  438. if (fragment != null) {
  439. OCFile file = fragment.getDisplayedFile();
  440. if (file != null) {
  441. outState.putParcelable(FileDetailFragment.EXTRA_FILE, file);
  442. }
  443. }
  444. }
  445. Log.d(getClass().toString(), "onSaveInstanceState() end");
  446. }
  447. @Override
  448. protected void onResume() {
  449. Log.d(getClass().toString(), "onResume() start");
  450. super.onResume();
  451. if (AccountUtils.accountsAreSetup(this)) {
  452. if (mStorageManager == null) {
  453. // this is necessary for handling the come back to FileDisplayActivity when the first ownCloud account is created
  454. initDataFromCurrentAccount();
  455. if (mDualPane) {
  456. initFileDetailsInDualPane();
  457. }
  458. }
  459. // Listen for sync messages
  460. IntentFilter syncIntentFilter = new IntentFilter(FileSyncService.SYNC_MESSAGE);
  461. mSyncBroadcastReceiver = new SyncBroadcastReceiver();
  462. registerReceiver(mSyncBroadcastReceiver, syncIntentFilter);
  463. // Listen for upload messages
  464. IntentFilter uploadIntentFilter = new IntentFilter(FileUploader.UPLOAD_FINISH_MESSAGE);
  465. mUploadFinishReceiver = new UploadFinishReceiver();
  466. registerReceiver(mUploadFinishReceiver, uploadIntentFilter);
  467. // Listen for download messages
  468. IntentFilter downloadIntentFilter = new IntentFilter(FileDownloader.DOWNLOAD_FINISH_MESSAGE);
  469. mDownloadFinishReceiver = new DownloadFinishReceiver();
  470. registerReceiver(mDownloadFinishReceiver, downloadIntentFilter);
  471. // List current directory
  472. mFileList.listDirectory(mCurrentDir); // TODO we should find the way to avoid the need of this (maybe it's not necessary yet; to check)
  473. } else {
  474. mStorageManager = null; // an invalid object will be there if all the ownCloud accounts are removed
  475. showDialog(DIALOG_SETUP_ACCOUNT);
  476. }
  477. Log.d(getClass().toString(), "onResume() end");
  478. }
  479. @Override
  480. protected void onPause() {
  481. Log.d(getClass().toString(), "onPause() start");
  482. super.onPause();
  483. if (mSyncBroadcastReceiver != null) {
  484. unregisterReceiver(mSyncBroadcastReceiver);
  485. mSyncBroadcastReceiver = null;
  486. }
  487. if (mUploadFinishReceiver != null) {
  488. unregisterReceiver(mUploadFinishReceiver);
  489. mUploadFinishReceiver = null;
  490. }
  491. if (mDownloadFinishReceiver != null) {
  492. unregisterReceiver(mDownloadFinishReceiver);
  493. mDownloadFinishReceiver = null;
  494. }
  495. if (!AccountUtils.accountsAreSetup(this)) {
  496. dismissDialog(DIALOG_SETUP_ACCOUNT);
  497. }
  498. Log.d(getClass().toString(), "onPause() end");
  499. }
  500. @Override
  501. protected void onPrepareDialog(int id, Dialog dialog, Bundle args) {
  502. if (id == DIALOG_SSL_VALIDATOR && mLastSslUntrustedServerResult != null) {
  503. ((SslValidatorDialog)dialog).updateResult(mLastSslUntrustedServerResult);
  504. }
  505. }
  506. @Override
  507. protected Dialog onCreateDialog(int id) {
  508. Dialog dialog = null;
  509. AlertDialog.Builder builder;
  510. switch (id) {
  511. case DIALOG_SETUP_ACCOUNT: {
  512. builder = new AlertDialog.Builder(this);
  513. builder.setTitle(R.string.main_tit_accsetup);
  514. builder.setMessage(R.string.main_wrn_accsetup);
  515. builder.setCancelable(false);
  516. builder.setPositiveButton(android.R.string.ok, new OnClickListener() {
  517. public void onClick(DialogInterface dialog, int which) {
  518. createFirstAccount();
  519. dialog.dismiss();
  520. }
  521. });
  522. String message = String.format(getString(R.string.common_exit), getString(R.string.app_name));
  523. builder.setNegativeButton(message, new OnClickListener() {
  524. public void onClick(DialogInterface dialog, int which) {
  525. dialog.dismiss();
  526. finish();
  527. }
  528. });
  529. //builder.setNegativeButton(android.R.string.cancel, this);
  530. dialog = builder.create();
  531. break;
  532. }
  533. case DIALOG_CREATE_DIR: {
  534. builder = new Builder(this);
  535. final EditText dirNameInput = new EditText(getBaseContext());
  536. builder.setView(dirNameInput);
  537. builder.setTitle(R.string.uploader_info_dirname);
  538. int typed_color = getResources().getColor(R.color.setup_text_typed);
  539. dirNameInput.setTextColor(typed_color);
  540. builder.setPositiveButton(android.R.string.ok, new OnClickListener() {
  541. public void onClick(DialogInterface dialog, int which) {
  542. String directoryName = dirNameInput.getText().toString();
  543. if (directoryName.trim().length() == 0) {
  544. dialog.cancel();
  545. return;
  546. }
  547. // Figure out the path where the dir needs to be created
  548. String path;
  549. if (mCurrentDir == null) {
  550. // this is just a patch; we should ensure that
  551. // mCurrentDir never is null
  552. if (!mStorageManager.fileExists(OCFile.PATH_SEPARATOR)) {
  553. OCFile file = new OCFile(OCFile.PATH_SEPARATOR);
  554. mStorageManager.saveFile(file);
  555. }
  556. mCurrentDir = mStorageManager.getFileByPath(OCFile.PATH_SEPARATOR);
  557. }
  558. path = FileDisplayActivity.this.mCurrentDir.getRemotePath();
  559. // Create directory
  560. path += directoryName + OCFile.PATH_SEPARATOR;
  561. Thread thread = new Thread(new DirectoryCreator(path, AccountUtils
  562. .getCurrentOwnCloudAccount(FileDisplayActivity.this), new Handler()));
  563. thread.start();
  564. dialog.dismiss();
  565. showDialog(DIALOG_SHORT_WAIT);
  566. }
  567. });
  568. builder.setNegativeButton(R.string.common_cancel, new OnClickListener() {
  569. public void onClick(DialogInterface dialog, int which) {
  570. dialog.cancel();
  571. }
  572. });
  573. dialog = builder.create();
  574. break;
  575. }
  576. case DIALOG_SHORT_WAIT: {
  577. ProgressDialog working_dialog = new ProgressDialog(this);
  578. working_dialog.setMessage(getResources().getString(R.string.wait_a_moment));
  579. working_dialog.setIndeterminate(true);
  580. working_dialog.setCancelable(false);
  581. dialog = working_dialog;
  582. break;
  583. }
  584. case DIALOG_CHOOSE_UPLOAD_SOURCE: {
  585. final String[] items = { getString(R.string.actionbar_upload_files),
  586. getString(R.string.actionbar_upload_from_apps), getString(R.string.actionbar_failed_instant_upload) };
  587. builder = new AlertDialog.Builder(this);
  588. builder.setTitle(R.string.actionbar_upload);
  589. builder.setItems(items, new DialogInterface.OnClickListener() {
  590. public void onClick(DialogInterface dialog, int item) {
  591. if (item == 0) {
  592. // if (!mDualPane) {
  593. Intent action = new Intent(FileDisplayActivity.this, UploadFilesActivity.class);
  594. action.putExtra(UploadFilesActivity.EXTRA_ACCOUNT,
  595. AccountUtils.getCurrentOwnCloudAccount(FileDisplayActivity.this));
  596. startActivityForResult(action, ACTION_SELECT_MULTIPLE_FILES);
  597. // } else {
  598. // TODO create and handle new fragment
  599. // LocalFileListFragment
  600. // }
  601. } else if (item == 1) {
  602. Intent action = new Intent(Intent.ACTION_GET_CONTENT);
  603. action = action.setType("*/*").addCategory(Intent.CATEGORY_OPENABLE);
  604. startActivityForResult(Intent.createChooser(action, getString(R.string.upload_chooser_title)),
  605. ACTION_SELECT_CONTENT_FROM_APPS);
  606. } else if (item == 2) {
  607. Account account = AccountUtils.getCurrentOwnCloudAccount(FileDisplayActivity.this);
  608. Intent action = new Intent(FileDisplayActivity.this, InstantUploadActivity.class);
  609. action.putExtra(FileUploader.KEY_ACCOUNT, account);
  610. startActivity(action);
  611. }
  612. }
  613. });
  614. dialog = builder.create();
  615. break;
  616. }
  617. case DIALOG_SSL_VALIDATOR: {
  618. dialog = SslValidatorDialog.newInstance(this, mLastSslUntrustedServerResult, this);
  619. break;
  620. }
  621. case DIALOG_CERT_NOT_SAVED: {
  622. builder = new AlertDialog.Builder(this);
  623. builder.setMessage(getResources().getString(R.string.ssl_validator_not_saved));
  624. builder.setCancelable(false);
  625. builder.setPositiveButton(R.string.common_ok, new DialogInterface.OnClickListener() {
  626. @Override
  627. public void onClick(DialogInterface dialog, int which) {
  628. dialog.dismiss();
  629. };
  630. });
  631. dialog = builder.create();
  632. break;
  633. }
  634. default:
  635. dialog = null;
  636. }
  637. return dialog;
  638. }
  639. /**
  640. * Translates a content URI of an image to a physical path
  641. * on the disk
  642. * @param uri The URI to resolve
  643. * @return The path to the image or null if it could not be found
  644. */
  645. public String getPath(Uri uri) {
  646. String[] projection = { MediaStore.Images.Media.DATA };
  647. Cursor cursor = managedQuery(uri, projection, null, null, null);
  648. if (cursor != null) {
  649. int column_index = cursor.getColumnIndexOrThrow(MediaStore.Images.Media.DATA);
  650. cursor.moveToFirst();
  651. return cursor.getString(column_index);
  652. }
  653. return null;
  654. }
  655. /**
  656. * Pushes a directory to the drop down list
  657. * @param directory to push
  658. * @throws IllegalArgumentException If the {@link OCFile#isDirectory()} returns false.
  659. */
  660. public void pushDirname(OCFile directory) {
  661. if(!directory.isDirectory()){
  662. throw new IllegalArgumentException("Only directories may be pushed!");
  663. }
  664. mDirectories.insert(directory.getFileName(), 0);
  665. mCurrentDir = directory;
  666. }
  667. /**
  668. * Pops a directory name from the drop down list
  669. * @return True, unless the stack is empty
  670. */
  671. public boolean popDirname() {
  672. mDirectories.remove(mDirectories.getItem(0));
  673. return !mDirectories.isEmpty();
  674. }
  675. private class DirectoryCreator implements Runnable {
  676. private String mTargetPath;
  677. private Account mAccount;
  678. private Handler mHandler;
  679. public DirectoryCreator(String targetPath, Account account, Handler handler) {
  680. mTargetPath = targetPath;
  681. mAccount = account;
  682. mHandler = handler;
  683. }
  684. @Override
  685. public void run() {
  686. WebdavClient wdc = OwnCloudClientUtils.createOwnCloudClient(mAccount, getApplicationContext());
  687. boolean created = wdc.createDirectory(mTargetPath);
  688. if (created) {
  689. mHandler.post(new Runnable() {
  690. @Override
  691. public void run() {
  692. dismissDialog(DIALOG_SHORT_WAIT);
  693. // Save new directory in local database
  694. OCFile newDir = new OCFile(mTargetPath);
  695. newDir.setMimetype("DIR");
  696. newDir.setParentId(mCurrentDir.getFileId());
  697. mStorageManager.saveFile(newDir);
  698. // Display the new folder right away
  699. mFileList.listDirectory();
  700. }
  701. });
  702. } else {
  703. mHandler.post(new Runnable() {
  704. @Override
  705. public void run() {
  706. dismissDialog(DIALOG_SHORT_WAIT);
  707. try {
  708. Toast msg = Toast.makeText(FileDisplayActivity.this, R.string.create_dir_fail_msg,
  709. Toast.LENGTH_LONG);
  710. msg.show();
  711. } catch (NotFoundException e) {
  712. Log.e(TAG, "Error while trying to show fail message ", e);
  713. }
  714. }
  715. });
  716. }
  717. }
  718. }
  719. // Custom array adapter to override text colors
  720. private class CustomArrayAdapter<T> extends ArrayAdapter<T> {
  721. public CustomArrayAdapter(FileDisplayActivity ctx, int view) {
  722. super(ctx, view);
  723. }
  724. public View getView(int position, View convertView, ViewGroup parent) {
  725. View v = super.getView(position, convertView, parent);
  726. ((TextView) v).setTextColor(getResources().getColorStateList(android.R.color.white));
  727. return v;
  728. }
  729. public View getDropDownView(int position, View convertView, ViewGroup parent) {
  730. View v = super.getDropDownView(position, convertView, parent);
  731. ((TextView) v).setTextColor(getResources().getColorStateList(android.R.color.white));
  732. return v;
  733. }
  734. }
  735. private class SyncBroadcastReceiver extends BroadcastReceiver {
  736. /**
  737. * {@link BroadcastReceiver} to enable syncing feedback in UI
  738. */
  739. @Override
  740. public void onReceive(Context context, Intent intent) {
  741. boolean inProgress = intent.getBooleanExtra(FileSyncService.IN_PROGRESS, false);
  742. String accountName = intent.getStringExtra(FileSyncService.ACCOUNT_NAME);
  743. Log.d("FileDisplay", "sync of account " + accountName + " is in_progress: " + inProgress);
  744. if (accountName.equals(AccountUtils.getCurrentOwnCloudAccount(context).name)) {
  745. String synchFolderRemotePath = intent.getStringExtra(FileSyncService.SYNC_FOLDER_REMOTE_PATH);
  746. boolean fillBlankRoot = false;
  747. if (mCurrentDir == null) {
  748. mCurrentDir = mStorageManager.getFileByPath("/");
  749. fillBlankRoot = (mCurrentDir != null);
  750. }
  751. if ((synchFolderRemotePath != null && mCurrentDir != null && (mCurrentDir.getRemotePath()
  752. .equals(synchFolderRemotePath))) || fillBlankRoot) {
  753. if (!fillBlankRoot)
  754. mCurrentDir = getStorageManager().getFileByPath(synchFolderRemotePath);
  755. OCFileListFragment fileListFragment = (OCFileListFragment) getSupportFragmentManager()
  756. .findFragmentById(R.id.fileList);
  757. if (fileListFragment != null) {
  758. fileListFragment.listDirectory(mCurrentDir);
  759. }
  760. }
  761. setSupportProgressBarIndeterminateVisibility(inProgress);
  762. removeStickyBroadcast(intent);
  763. }
  764. RemoteOperationResult synchResult = (RemoteOperationResult) intent
  765. .getSerializableExtra(FileSyncService.SYNC_RESULT);
  766. if (synchResult != null) {
  767. if (synchResult.getCode().equals(RemoteOperationResult.ResultCode.SSL_RECOVERABLE_PEER_UNVERIFIED)) {
  768. mLastSslUntrustedServerResult = synchResult;
  769. showDialog(DIALOG_SSL_VALIDATOR);
  770. }
  771. }
  772. }
  773. }
  774. private class UploadFinishReceiver extends BroadcastReceiver {
  775. /**
  776. * Once the file upload has finished -> update view
  777. * @author David A. Velasco
  778. * {@link BroadcastReceiver} to enable upload feedback in UI
  779. */
  780. @Override
  781. public void onReceive(Context context, Intent intent) {
  782. String uploadedRemotePath = intent.getStringExtra(FileDownloader.EXTRA_REMOTE_PATH);
  783. String accountName = intent.getStringExtra(FileUploader.ACCOUNT_NAME);
  784. boolean sameAccount = accountName.equals(AccountUtils.getCurrentOwnCloudAccount(context).name);
  785. boolean isDescendant = (mCurrentDir != null) && (uploadedRemotePath != null)
  786. && (uploadedRemotePath.startsWith(mCurrentDir.getRemotePath()));
  787. if (sameAccount && isDescendant) {
  788. OCFileListFragment fileListFragment = (OCFileListFragment) getSupportFragmentManager()
  789. .findFragmentById(R.id.fileList);
  790. if (fileListFragment != null) {
  791. fileListFragment.listDirectory();
  792. }
  793. }
  794. }
  795. }
  796. /**
  797. * Once the file download has finished -> update view
  798. */
  799. private class DownloadFinishReceiver extends BroadcastReceiver {
  800. @Override
  801. public void onReceive(Context context, Intent intent) {
  802. String downloadedRemotePath = intent.getStringExtra(FileDownloader.EXTRA_REMOTE_PATH);
  803. String accountName = intent.getStringExtra(FileDownloader.ACCOUNT_NAME);
  804. boolean sameAccount = accountName.equals(AccountUtils.getCurrentOwnCloudAccount(context).name);
  805. boolean isDescendant = (mCurrentDir != null) && (downloadedRemotePath != null)
  806. && (downloadedRemotePath.startsWith(mCurrentDir.getRemotePath()));
  807. if (sameAccount && isDescendant) {
  808. OCFileListFragment fileListFragment = (OCFileListFragment) getSupportFragmentManager()
  809. .findFragmentById(R.id.fileList);
  810. if (fileListFragment != null) {
  811. fileListFragment.listDirectory();
  812. }
  813. }
  814. }
  815. }
  816. /**
  817. * {@inheritDoc}
  818. */
  819. @Override
  820. public DataStorageManager getStorageManager() {
  821. return mStorageManager;
  822. }
  823. /**
  824. * {@inheritDoc}
  825. */
  826. @Override
  827. public void onDirectoryClick(OCFile directory) {
  828. pushDirname(directory);
  829. ActionBar actionBar = getSupportActionBar();
  830. actionBar.setDisplayHomeAsUpEnabled(true);
  831. if (mDualPane) {
  832. // Resets the FileDetailsFragment on Tablets so that it always displays
  833. FileDetailFragment fileDetails = (FileDetailFragment) getSupportFragmentManager().findFragmentByTag(FileDetailFragment.FTAG);
  834. if (fileDetails != null && !fileDetails.isEmpty()) {
  835. FragmentTransaction transaction = getSupportFragmentManager().beginTransaction();
  836. transaction.remove(fileDetails);
  837. transaction.add(R.id.file_details_container, new FileDetailFragment(null, null),
  838. FileDetailFragment.FTAG);
  839. transaction.commit();
  840. }
  841. }
  842. }
  843. /**
  844. * {@inheritDoc}
  845. */
  846. @Override
  847. public void onFileClick(OCFile file) {
  848. // If we are on a large device -> update fragment
  849. if (mDualPane) {
  850. // buttons in the details view are problematic when trying to reuse an existing fragment; create always a new one solves some of them, BUT no all; downloads are 'dangerous'
  851. FragmentTransaction transaction = getSupportFragmentManager().beginTransaction();
  852. transaction
  853. .replace(R.id.file_details_container,
  854. new FileDetailFragment(file, AccountUtils.getCurrentOwnCloudAccount(this)),
  855. FileDetailFragment.FTAG);
  856. transaction.setTransition(FragmentTransaction.TRANSIT_FRAGMENT_FADE);
  857. transaction.commit();
  858. } else { // small or medium screen device -> new Activity
  859. Intent showDetailsIntent = new Intent(this, FileDetailActivity.class);
  860. showDetailsIntent.putExtra(FileDetailFragment.EXTRA_FILE, file);
  861. showDetailsIntent.putExtra(FileDetailFragment.EXTRA_ACCOUNT, AccountUtils.getCurrentOwnCloudAccount(this));
  862. startActivity(showDetailsIntent);
  863. }
  864. }
  865. /**
  866. * {@inheritDoc}
  867. */
  868. @Override
  869. public OCFile getInitialDirectory() {
  870. return mCurrentDir;
  871. }
  872. /**
  873. * {@inheritDoc}
  874. */
  875. @Override
  876. public void onFileStateChanged() {
  877. OCFileListFragment fileListFragment = (OCFileListFragment) getSupportFragmentManager().findFragmentById(
  878. R.id.fileList);
  879. if (fileListFragment != null) {
  880. fileListFragment.listDirectory();
  881. }
  882. }
  883. /**
  884. * {@inheritDoc}
  885. */
  886. @Override
  887. public FileDownloaderBinder getFileDownloaderBinder() {
  888. return mDownloaderBinder;
  889. }
  890. /**
  891. * {@inheritDoc}
  892. */
  893. @Override
  894. public FileUploaderBinder getFileUploaderBinder() {
  895. return mUploaderBinder;
  896. }
  897. /** Defines callbacks for service binding, passed to bindService() */
  898. private class ListServiceConnection implements ServiceConnection {
  899. @Override
  900. public void onServiceConnected(ComponentName component, IBinder service) {
  901. if (component.equals(new ComponentName(FileDisplayActivity.this, FileDownloader.class))) {
  902. Log.d(TAG, "Download service connected");
  903. mDownloaderBinder = (FileDownloaderBinder) service;
  904. } else if (component.equals(new ComponentName(FileDisplayActivity.this, FileUploader.class))) {
  905. Log.d(TAG, "Upload service connected");
  906. mUploaderBinder = (FileUploaderBinder) service;
  907. } else {
  908. return;
  909. }
  910. // a new chance to get the mDownloadBinder through getFileDownloadBinder() - THIS IS A MESS
  911. if (mFileList != null)
  912. mFileList.listDirectory();
  913. if (mDualPane) {
  914. FileDetailFragment fragment = (FileDetailFragment) getSupportFragmentManager().findFragmentByTag(
  915. FileDetailFragment.FTAG);
  916. if (fragment != null)
  917. fragment.updateFileDetails(false);
  918. }
  919. }
  920. @Override
  921. public void onServiceDisconnected(ComponentName component) {
  922. if (component.equals(new ComponentName(FileDisplayActivity.this, FileDownloader.class))) {
  923. Log.d(TAG, "Download service disconnected");
  924. mDownloaderBinder = null;
  925. } else if (component.equals(new ComponentName(FileDisplayActivity.this, FileUploader.class))) {
  926. Log.d(TAG, "Upload service disconnected");
  927. mUploaderBinder = null;
  928. }
  929. }
  930. };
  931. /**
  932. * Launch an intent to request the PIN code to the user before letting him use the app
  933. */
  934. private void requestPinCode() {
  935. boolean pinStart = false;
  936. SharedPreferences appPrefs = PreferenceManager.getDefaultSharedPreferences(getApplicationContext());
  937. pinStart = appPrefs.getBoolean("set_pincode", false);
  938. if (pinStart) {
  939. Intent i = new Intent(getApplicationContext(), PinCodeActivity.class);
  940. i.putExtra(PinCodeActivity.EXTRA_ACTIVITY, "FileDisplayActivity");
  941. startActivity(i);
  942. }
  943. }
  944. @Override
  945. public void onSavedCertificate() {
  946. startSynchronization();
  947. }
  948. @Override
  949. public void onFailedSavingCertificate() {
  950. showDialog(DIALOG_CERT_NOT_SAVED);
  951. }
  952. /**
  953. * Updates the view associated to the activity after the finish of some operation over files
  954. * in the current account.
  955. *
  956. * @param operation Removal operation performed.
  957. * @param result Result of the removal.
  958. */
  959. @Override
  960. public void onRemoteOperationFinish(RemoteOperation operation, RemoteOperationResult result) {
  961. if (operation instanceof RemoveFileOperation) {
  962. onRemoveFileOperationFinish((RemoveFileOperation) operation, result);
  963. } else if (operation instanceof RenameFileOperation) {
  964. onRenameFileOperationFinish((RenameFileOperation) operation, result);
  965. } else if (operation instanceof SynchronizeFileOperation) {
  966. onSynchronizeFileOperationFinish((SynchronizeFileOperation) operation, result);
  967. }
  968. }
  969. /**
  970. * Updates the view associated to the activity after the finish of an operation trying to remove a
  971. * file.
  972. *
  973. * @param operation Removal operation performed.
  974. * @param result Result of the removal.
  975. */
  976. private void onRemoveFileOperationFinish(RemoveFileOperation operation, RemoteOperationResult result) {
  977. dismissDialog(DIALOG_SHORT_WAIT);
  978. if (result.isSuccess()) {
  979. Toast msg = Toast.makeText(this, R.string.remove_success_msg, Toast.LENGTH_LONG);
  980. msg.show();
  981. OCFile removedFile = operation.getFile();
  982. if (mDualPane) {
  983. FileDetailFragment details = (FileDetailFragment) getSupportFragmentManager().findFragmentByTag(
  984. FileDetailFragment.FTAG);
  985. if (details != null && removedFile.equals(details.getDisplayedFile())) {
  986. FragmentTransaction transaction = getSupportFragmentManager().beginTransaction();
  987. transaction.replace(R.id.file_details_container, new FileDetailFragment(null, null)); // empty
  988. // FileDetailFragment
  989. transaction.commit();
  990. }
  991. }
  992. if (mStorageManager.getFileById(removedFile.getParentId()).equals(mCurrentDir)) {
  993. mFileList.listDirectory();
  994. }
  995. } else {
  996. Toast msg = Toast.makeText(this, R.string.remove_fail_msg, Toast.LENGTH_LONG);
  997. msg.show();
  998. if (result.isSslRecoverableException()) {
  999. mLastSslUntrustedServerResult = result;
  1000. showDialog(DIALOG_SSL_VALIDATOR);
  1001. }
  1002. }
  1003. }
  1004. /**
  1005. * Updates the view associated to the activity after the finish of an operation trying to rename a
  1006. * file.
  1007. *
  1008. * @param operation Renaming operation performed.
  1009. * @param result Result of the renaming.
  1010. */
  1011. private void onRenameFileOperationFinish(RenameFileOperation operation, RemoteOperationResult result) {
  1012. dismissDialog(DIALOG_SHORT_WAIT);
  1013. OCFile renamedFile = operation.getFile();
  1014. if (result.isSuccess()) {
  1015. if (mDualPane) {
  1016. FileDetailFragment details = (FileDetailFragment) getSupportFragmentManager().findFragmentByTag(
  1017. FileDetailFragment.FTAG);
  1018. if (details != null && renamedFile.equals(details.getDisplayedFile())) {
  1019. details.updateFileDetails(renamedFile, AccountUtils.getCurrentOwnCloudAccount(this));
  1020. }
  1021. }
  1022. if (mStorageManager.getFileById(renamedFile.getParentId()).equals(mCurrentDir)) {
  1023. mFileList.listDirectory();
  1024. }
  1025. } else {
  1026. if (result.getCode().equals(ResultCode.INVALID_LOCAL_FILE_NAME)) {
  1027. Toast msg = Toast.makeText(this, R.string.rename_local_fail_msg, Toast.LENGTH_LONG);
  1028. msg.show();
  1029. // TODO throw again the new rename dialog
  1030. } else {
  1031. Toast msg = Toast.makeText(this, R.string.rename_server_fail_msg, Toast.LENGTH_LONG);
  1032. msg.show();
  1033. if (result.isSslRecoverableException()) {
  1034. mLastSslUntrustedServerResult = result;
  1035. showDialog(DIALOG_SSL_VALIDATOR);
  1036. }
  1037. }
  1038. }
  1039. }
  1040. private void onSynchronizeFileOperationFinish(SynchronizeFileOperation operation, RemoteOperationResult result) {
  1041. dismissDialog(DIALOG_SHORT_WAIT);
  1042. OCFile syncedFile = operation.getLocalFile();
  1043. if (!result.isSuccess()) {
  1044. if (result.getCode() == ResultCode.SYNC_CONFLICT) {
  1045. Intent i = new Intent(this, ConflictsResolveActivity.class);
  1046. i.putExtra(ConflictsResolveActivity.EXTRA_FILE, syncedFile);
  1047. i.putExtra(ConflictsResolveActivity.EXTRA_ACCOUNT, AccountUtils.getCurrentOwnCloudAccount(this));
  1048. startActivity(i);
  1049. } else {
  1050. Toast msg = Toast.makeText(this, R.string.sync_file_fail_msg, Toast.LENGTH_LONG);
  1051. msg.show();
  1052. }
  1053. } else {
  1054. if (operation.transferWasRequested()) {
  1055. mFileList.listDirectory();
  1056. onTransferStateChanged(syncedFile, true, true);
  1057. } else {
  1058. Toast msg = Toast.makeText(this, R.string.sync_file_nothing_to_do_msg, Toast.LENGTH_LONG);
  1059. msg.show();
  1060. }
  1061. }
  1062. }
  1063. /**
  1064. * {@inheritDoc}
  1065. */
  1066. @Override
  1067. public void onTransferStateChanged(OCFile file, boolean downloading, boolean uploading) {
  1068. /*OCFileListFragment fileListFragment = (OCFileListFragment) getSupportFragmentManager().findFragmentById(R.id.fileList);
  1069. if (fileListFragment != null) {
  1070. fileListFragment.listDirectory();
  1071. }*/
  1072. if (mDualPane) {
  1073. FileDetailFragment details = (FileDetailFragment) getSupportFragmentManager().findFragmentByTag(
  1074. FileDetailFragment.FTAG);
  1075. if (details != null && file.equals(details.getDisplayedFile())) {
  1076. if (downloading || uploading) {
  1077. details.updateFileDetails(file, AccountUtils.getCurrentOwnCloudAccount(this));
  1078. } else {
  1079. details.updateFileDetails(downloading || uploading);
  1080. }
  1081. }
  1082. }
  1083. }
  1084. }