OCFileListFragment.java 27 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674
  1. /* ownCloud Android client application
  2. * Copyright (C) 2011 Bartek Przybylski
  3. * Copyright (C) 2012-2014 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 version 2,
  7. * as published by the Free Software Foundation.
  8. *
  9. * This program is distributed in the hope that it will be useful,
  10. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  11. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  12. * GNU General Public License for more details.
  13. *
  14. * You should have received a copy of the GNU General Public License
  15. * along with this program. If not, see <http://www.gnu.org/licenses/>.
  16. *
  17. */
  18. package com.owncloud.android.ui.fragment;
  19. import java.io.File;
  20. import java.util.ArrayList;
  21. import java.util.List;
  22. import com.owncloud.android.R;
  23. import com.owncloud.android.authentication.AccountUtils;
  24. import com.owncloud.android.datamodel.FileDataStorageManager;
  25. import com.owncloud.android.datamodel.FileListCursorLoader;
  26. import com.owncloud.android.datamodel.OCFile;
  27. import com.owncloud.android.db.ProviderMeta.ProviderTableMeta;
  28. import com.owncloud.android.files.services.FileDownloader.FileDownloaderBinder;
  29. import com.owncloud.android.files.services.FileUploader.FileUploaderBinder;
  30. import com.owncloud.android.ui.ExtendedListView;
  31. import com.owncloud.android.ui.adapter.FileListListAdapter;
  32. import com.owncloud.android.ui.activity.FileDisplayActivity;
  33. import com.owncloud.android.ui.dialog.ConfirmationDialogFragment;
  34. import com.owncloud.android.ui.dialog.EditNameDialog;
  35. import com.owncloud.android.ui.dialog.ConfirmationDialogFragment.ConfirmationDialogFragmentListener;
  36. import com.owncloud.android.ui.dialog.EditNameDialog.EditNameDialogListener;
  37. import com.owncloud.android.ui.preview.PreviewImageFragment;
  38. import com.owncloud.android.ui.preview.PreviewMediaFragment;
  39. import com.owncloud.android.utils.Log_OC;
  40. import android.accounts.Account;
  41. import android.app.Activity;
  42. import android.database.Cursor;
  43. import android.net.Uri;
  44. import android.os.Bundle;
  45. import android.support.v4.app.LoaderManager;
  46. import android.support.v4.app.LoaderManager.LoaderCallbacks;
  47. import android.support.v4.content.Loader;
  48. import android.view.ContextMenu;
  49. import android.view.MenuInflater;
  50. import android.view.MenuItem;
  51. import android.view.View;
  52. import android.widget.AdapterView;
  53. import android.widget.AdapterView.AdapterContextMenuInfo;
  54. /**
  55. * A Fragment that lists all files and folders in a given path.
  56. *
  57. * TODO refactorize to get rid of direct dependency on FileDisplayActivity
  58. *
  59. * @author Bartek Przybylski
  60. * @author masensio
  61. * @author David A. Velasco
  62. */
  63. public class OCFileListFragment extends ExtendedListFragment
  64. implements EditNameDialogListener, ConfirmationDialogFragmentListener,
  65. LoaderCallbacks<Cursor>{
  66. private static final String TAG = OCFileListFragment.class.getSimpleName();
  67. private static final String MY_PACKAGE = OCFileListFragment.class.getPackage() != null ? OCFileListFragment.class.getPackage().getName() : "com.owncloud.android.ui.fragment";
  68. private static final String EXTRA_FILE = MY_PACKAGE + ".extra.FILE";
  69. private static final String KEY_INDEXES = "INDEXES";
  70. private static final String KEY_FIRST_POSITIONS= "FIRST_POSITIONS";
  71. private static final String KEY_TOPS = "TOPS";
  72. private static final String KEY_HEIGHT_CELL = "HEIGHT_CELL";
  73. private static final int LOADER_ID = 0;
  74. private FileFragment.ContainerActivity mContainerActivity;
  75. private OCFile mFile = null;
  76. private FileListListAdapter mAdapter;
  77. private LoaderManager mLoaderManager;
  78. private FileListCursorLoader mCursorLoader;
  79. private OCFile mTargetFile;
  80. // Save the state of the scroll in browsing
  81. private ArrayList<Integer> mIndexes;
  82. private ArrayList<Integer> mFirstPositions;
  83. private ArrayList<Integer> mTops;
  84. private int mHeightCell = 0;
  85. /**
  86. * {@inheritDoc}
  87. */
  88. @Override
  89. public void onAttach(Activity activity) {
  90. super.onAttach(activity);
  91. Log_OC.e(TAG, "onAttach");
  92. try {
  93. mContainerActivity = (FileFragment.ContainerActivity) activity;
  94. } catch (ClassCastException e) {
  95. throw new ClassCastException(activity.toString() + " must implement " +
  96. FileFragment.ContainerActivity.class.getSimpleName());
  97. }
  98. }
  99. @Override
  100. public void onDetach() {
  101. mContainerActivity = null;
  102. super.onDetach();
  103. }
  104. /**
  105. * {@inheritDoc}
  106. */
  107. @Override
  108. public void onActivityCreated(Bundle savedInstanceState) {
  109. super.onActivityCreated(savedInstanceState);
  110. Log_OC.e(TAG, "onActivityCreated() start");
  111. mAdapter = new FileListListAdapter(getSherlockActivity(), mContainerActivity);
  112. mLoaderManager = getLoaderManager();
  113. if (savedInstanceState != null) {
  114. mFile = savedInstanceState.getParcelable(EXTRA_FILE);
  115. mIndexes = savedInstanceState.getIntegerArrayList(KEY_INDEXES);
  116. mFirstPositions = savedInstanceState.getIntegerArrayList(KEY_FIRST_POSITIONS);
  117. mTops = savedInstanceState.getIntegerArrayList(KEY_TOPS);
  118. mHeightCell = savedInstanceState.getInt(KEY_HEIGHT_CELL);
  119. } else {
  120. mIndexes = new ArrayList<Integer>();
  121. mFirstPositions = new ArrayList<Integer>();
  122. mTops = new ArrayList<Integer>();
  123. mHeightCell = 0;
  124. }
  125. // Initialize loaderManager and makes it active
  126. mLoaderManager.initLoader(LOADER_ID, null, this);
  127. setListAdapter(mAdapter);
  128. registerForContextMenu(getListView());
  129. getListView().setOnCreateContextMenuListener(this);
  130. }
  131. /**
  132. * Saves the current listed folder.
  133. */
  134. @Override
  135. public void onSaveInstanceState (Bundle outState) {
  136. super.onSaveInstanceState(outState);
  137. outState.putParcelable(EXTRA_FILE, mFile);
  138. outState.putIntegerArrayList(KEY_INDEXES, mIndexes);
  139. outState.putIntegerArrayList(KEY_FIRST_POSITIONS, mFirstPositions);
  140. outState.putIntegerArrayList(KEY_TOPS, mTops);
  141. outState.putInt(KEY_HEIGHT_CELL, mHeightCell);
  142. }
  143. /**
  144. * Call this, when the user presses the up button.
  145. *
  146. * Tries to move up the current folder one level. If the parent folder was removed from the database,
  147. * it continues browsing up until finding an existing folders.
  148. *
  149. * return Count of folder levels browsed up.
  150. */
  151. public int onBrowseUp() {
  152. OCFile parentDir = null;
  153. int moveCount = 0;
  154. if(mFile != null){
  155. FileDataStorageManager storageManager = mContainerActivity.getStorageManager();
  156. String parentPath = null;
  157. if (mFile.getParentId() != FileDataStorageManager.ROOT_PARENT_ID) {
  158. parentPath = new File(mFile.getRemotePath()).getParent();
  159. parentPath = parentPath.endsWith(OCFile.PATH_SEPARATOR) ? parentPath : parentPath + OCFile.PATH_SEPARATOR;
  160. parentDir = storageManager.getFileByPath(parentPath);
  161. moveCount++;
  162. } else {
  163. parentDir = storageManager.getFileByPath(OCFile.ROOT_PATH); // never returns null; keep the path in root folder
  164. }
  165. while (parentDir == null) {
  166. parentPath = new File(parentPath).getParent();
  167. parentPath = parentPath.endsWith(OCFile.PATH_SEPARATOR) ? parentPath : parentPath + OCFile.PATH_SEPARATOR;
  168. parentDir = storageManager.getFileByPath(parentPath);
  169. moveCount++;
  170. } // exit is granted because storageManager.getFileByPath("/") never returns null
  171. mFile = parentDir;
  172. }
  173. if (mFile != null) {
  174. listDirectory(mFile);
  175. ((FileDisplayActivity)mContainerActivity).startSyncFolderOperation(mFile);
  176. // restore index and top position
  177. restoreIndexAndTopPosition();
  178. } // else - should never happen now
  179. return moveCount;
  180. }
  181. /*
  182. * Restore index and position
  183. */
  184. private void restoreIndexAndTopPosition() {
  185. if (mIndexes.size() > 0) {
  186. // needs to be checked; not every browse-up had a browse-down before
  187. int index = mIndexes.remove(mIndexes.size() - 1);
  188. int firstPosition = mFirstPositions.remove(mFirstPositions.size() -1);
  189. int top = mTops.remove(mTops.size() - 1);
  190. ExtendedListView list = (ExtendedListView) getListView();
  191. list.setSelectionFromTop(firstPosition, top);
  192. // Move the scroll if the selection is not visible
  193. int indexPosition = mHeightCell*index;
  194. int height = list.getHeight();
  195. if (indexPosition > height) {
  196. if (android.os.Build.VERSION.SDK_INT >= 11)
  197. {
  198. list.smoothScrollToPosition(index);
  199. }
  200. else if (android.os.Build.VERSION.SDK_INT >= 8)
  201. {
  202. list.setSelectionFromTop(index, 0);
  203. }
  204. }
  205. }
  206. }
  207. /*
  208. * Save index and top position
  209. */
  210. private void saveIndexAndTopPosition(int index) {
  211. mIndexes.add(index);
  212. ExtendedListView list = (ExtendedListView) getListView();
  213. int firstPosition = list.getFirstVisiblePosition();
  214. mFirstPositions.add(firstPosition);
  215. View view = list.getChildAt(0);
  216. int top = (view == null) ? 0 : view.getTop() ;
  217. mTops.add(top);
  218. // Save the height of a cell
  219. mHeightCell = (view == null || mHeightCell != 0) ? mHeightCell : view.getHeight();
  220. }
  221. @Override
  222. public void onItemClick(AdapterView<?> l, View v, int position, long id) {
  223. OCFile file = mContainerActivity.getStorageManager().createFileInstance(
  224. (Cursor) mAdapter.getItem(position));
  225. if (file != null) {
  226. if (file.isFolder()) {
  227. // update state and view of this fragment
  228. listDirectory(file);
  229. // then, notify parent activity to let it update its state and view, and other fragments
  230. mContainerActivity.onBrowsedDownTo(file);
  231. // save index and top position
  232. saveIndexAndTopPosition(position);
  233. } else { /// Click on a file
  234. if (PreviewImageFragment.canBePreviewed(file)) {
  235. // preview image - it handles the download, if needed
  236. ((FileDisplayActivity)mContainerActivity).startImagePreview(file);
  237. } else if (file.isDown()) {
  238. if (PreviewMediaFragment.canBePreviewed(file)) {
  239. // media preview
  240. ((FileDisplayActivity)mContainerActivity).startMediaPreview(file, 0, true);
  241. } else {
  242. ((FileDisplayActivity)mContainerActivity).getFileOperationsHelper().openFile(file);
  243. }
  244. } else {
  245. // automatic download, preview on finish
  246. ((FileDisplayActivity)mContainerActivity).startDownloadForPreview(file);
  247. }
  248. }
  249. } else {
  250. Log_OC.d(TAG, "Null object in ListAdapter!!");
  251. }
  252. }
  253. /**
  254. * {@inheritDoc}
  255. */
  256. @Override
  257. public void onCreateContextMenu (ContextMenu menu, View v, ContextMenu.ContextMenuInfo menuInfo) {
  258. super.onCreateContextMenu(menu, v, menuInfo);
  259. MenuInflater inflater = getSherlockActivity().getMenuInflater();
  260. inflater.inflate(R.menu.file_actions_menu, menu);
  261. AdapterContextMenuInfo info = (AdapterContextMenuInfo) menuInfo;
  262. OCFile targetFile = mContainerActivity.getStorageManager().createFileInstance(
  263. (Cursor) mAdapter.getItem(info.position));
  264. List<Integer> toHide = new ArrayList<Integer>();
  265. List<Integer> toDisable = new ArrayList<Integer>();
  266. MenuItem item = null;
  267. if (targetFile.isFolder()) {
  268. // contextual menu for folders
  269. toHide.add(R.id.action_open_file_with);
  270. toHide.add(R.id.action_download_file);
  271. toHide.add(R.id.action_cancel_download);
  272. toHide.add(R.id.action_cancel_upload);
  273. toHide.add(R.id.action_sync_file);
  274. toHide.add(R.id.action_see_details);
  275. toHide.add(R.id.action_send_file);
  276. if ( mContainerActivity.getFileDownloaderBinder().isDownloading(AccountUtils.getCurrentOwnCloudAccount(getSherlockActivity()), targetFile) ||
  277. mContainerActivity.getFileUploaderBinder().isUploading(AccountUtils.getCurrentOwnCloudAccount(getSherlockActivity()), targetFile) ) {
  278. toDisable.add(R.id.action_rename_file);
  279. toDisable.add(R.id.action_remove_file);
  280. }
  281. } else {
  282. // contextual menu for regular files
  283. // new design: 'download' and 'open with' won't be available anymore in context menu
  284. toHide.add(R.id.action_download_file);
  285. toHide.add(R.id.action_open_file_with);
  286. if (targetFile.isDown()) {
  287. toHide.add(R.id.action_cancel_download);
  288. toHide.add(R.id.action_cancel_upload);
  289. } else {
  290. toHide.add(R.id.action_sync_file);
  291. }
  292. if ( mContainerActivity.getFileDownloaderBinder().isDownloading(AccountUtils.getCurrentOwnCloudAccount(getSherlockActivity()), targetFile)) {
  293. toHide.add(R.id.action_cancel_upload);
  294. toDisable.add(R.id.action_rename_file);
  295. toDisable.add(R.id.action_remove_file);
  296. } else if ( mContainerActivity.getFileUploaderBinder().isUploading(AccountUtils.getCurrentOwnCloudAccount(getSherlockActivity()), targetFile)) {
  297. toHide.add(R.id.action_cancel_download);
  298. toDisable.add(R.id.action_rename_file);
  299. toDisable.add(R.id.action_remove_file);
  300. } else {
  301. toHide.add(R.id.action_cancel_download);
  302. toHide.add(R.id.action_cancel_upload);
  303. }
  304. }
  305. // Options shareLink
  306. if (!targetFile.isShareByLink()) {
  307. toHide.add(R.id.action_unshare_file);
  308. }
  309. // Send file
  310. boolean sendEnabled = getString(R.string.send_files_to_other_apps).equalsIgnoreCase("on");
  311. if (!sendEnabled) {
  312. toHide.add(R.id.action_send_file);
  313. }
  314. for (int i : toHide) {
  315. item = menu.findItem(i);
  316. if (item != null) {
  317. item.setVisible(false);
  318. item.setEnabled(false);
  319. }
  320. }
  321. for (int i : toDisable) {
  322. item = menu.findItem(i);
  323. if (item != null) {
  324. item.setEnabled(false);
  325. }
  326. }
  327. }
  328. /**
  329. * {@inhericDoc}
  330. */
  331. @Override
  332. public boolean onContextItemSelected (MenuItem item) {
  333. AdapterContextMenuInfo info = (AdapterContextMenuInfo) item.getMenuInfo();
  334. mTargetFile = mContainerActivity.getStorageManager().createFileInstance(
  335. (Cursor) mAdapter.getItem(info.position));
  336. switch (item.getItemId()) {
  337. case R.id.action_share_file: {
  338. mContainerActivity.getFileOperationsHelper().shareFileWithLink(mTargetFile);
  339. return true;
  340. }
  341. case R.id.action_unshare_file: {
  342. mContainerActivity.getFileOperationsHelper().unshareFileWithLink(mTargetFile);
  343. return true;
  344. }
  345. case R.id.action_rename_file: {
  346. String fileName = mTargetFile.getFileName();
  347. int extensionStart = mTargetFile.isFolder() ? -1 : fileName.lastIndexOf(".");
  348. int selectionEnd = (extensionStart >= 0) ? extensionStart : fileName.length();
  349. EditNameDialog dialog = EditNameDialog.newInstance(getString(R.string.rename_dialog_title), fileName, 0, selectionEnd, this);
  350. dialog.show(getFragmentManager(), EditNameDialog.TAG);
  351. return true;
  352. }
  353. case R.id.action_remove_file: {
  354. int messageStringId = R.string.confirmation_remove_alert;
  355. int posBtnStringId = R.string.confirmation_remove_remote;
  356. int neuBtnStringId = -1;
  357. if (mTargetFile.isFolder()) {
  358. messageStringId = R.string.confirmation_remove_folder_alert;
  359. posBtnStringId = R.string.confirmation_remove_remote_and_local;
  360. neuBtnStringId = R.string.confirmation_remove_folder_local;
  361. } else if (mTargetFile.isDown()) {
  362. posBtnStringId = R.string.confirmation_remove_remote_and_local;
  363. neuBtnStringId = R.string.confirmation_remove_local;
  364. }
  365. ConfirmationDialogFragment confDialog = ConfirmationDialogFragment.newInstance(
  366. messageStringId,
  367. new String[]{mTargetFile.getFileName()},
  368. posBtnStringId,
  369. neuBtnStringId,
  370. R.string.common_cancel);
  371. confDialog.setOnConfirmationListener(this);
  372. confDialog.show(getFragmentManager(), FileDetailFragment.FTAG_CONFIRMATION);
  373. return true;
  374. }
  375. case R.id.action_sync_file: {
  376. mContainerActivity.getFileOperationsHelper().syncFile(mTargetFile);
  377. return true;
  378. }
  379. case R.id.action_cancel_download: {
  380. FileDownloaderBinder downloaderBinder = mContainerActivity.getFileDownloaderBinder();
  381. Account account = AccountUtils.getCurrentOwnCloudAccount(getSherlockActivity());
  382. if (downloaderBinder != null && downloaderBinder.isDownloading(account, mTargetFile)) {
  383. downloaderBinder.cancel(account, mTargetFile);
  384. listDirectory();
  385. mContainerActivity.onTransferStateChanged(mTargetFile, false, false);
  386. }
  387. return true;
  388. }
  389. case R.id.action_cancel_upload: {
  390. FileUploaderBinder uploaderBinder = mContainerActivity.getFileUploaderBinder();
  391. Account account = AccountUtils.getCurrentOwnCloudAccount(getSherlockActivity());
  392. if (uploaderBinder != null && uploaderBinder.isUploading(account, mTargetFile)) {
  393. uploaderBinder.cancel(account, mTargetFile);
  394. listDirectory();
  395. mContainerActivity.onTransferStateChanged(mTargetFile, false, false);
  396. }
  397. return true;
  398. }
  399. case R.id.action_see_details: {
  400. mContainerActivity.showDetails(mTargetFile);
  401. return true;
  402. }
  403. case R.id.action_send_file: {
  404. // Obtain the file
  405. if (!mTargetFile.isDown()) { // Download the file
  406. Log_OC.d(TAG, mTargetFile.getRemotePath() + " : File must be downloaded");
  407. ((FileDisplayActivity)mContainerActivity).startDownloadForSending(mTargetFile);
  408. } else {
  409. ((FileDisplayActivity)mContainerActivity).getFileOperationsHelper().sendDownloadedFile(mTargetFile);
  410. }
  411. return true;
  412. }
  413. default:
  414. return super.onContextItemSelected(item);
  415. }
  416. }
  417. /**
  418. * Use this to query the {@link OCFile} that is currently
  419. * being displayed by this fragment
  420. * @return The currently viewed OCFile
  421. */
  422. public OCFile getCurrentFile(){
  423. return mFile;
  424. }
  425. /**
  426. * Calls {@link OCFileListFragment#listDirectory(OCFile)} with a null parameter
  427. */
  428. public void listDirectory(){
  429. listDirectory(null);
  430. }
  431. /**
  432. * Lists the given directory on the view. When the input parameter is null,
  433. * it will either refresh the last known directory. list the root
  434. * if there never was a directory.
  435. *
  436. * @param directory File to be listed
  437. */
  438. public void listDirectory(OCFile directory) {
  439. FileDataStorageManager storageManager = mContainerActivity.getStorageManager();
  440. if (storageManager != null) {
  441. // Check input parameters for null
  442. if(directory == null){
  443. if(mFile != null){
  444. directory = mFile;
  445. } else {
  446. directory = storageManager.getFileByPath("/");
  447. if (directory == null) return; // no files, wait for sync
  448. }
  449. }
  450. // If that's not a directory -> List its parent
  451. if(!directory.isFolder()){
  452. Log_OC.w(TAG, "You see, that is not a directory -> " + directory.toString());
  453. directory = storageManager.getFileById(directory.getParentId());
  454. }
  455. swapDirectory(directory.getFileId(), storageManager);
  456. if (mFile == null || !mFile.equals(directory)) {
  457. ((ExtendedListView) getListView()).setSelectionFromTop(0, 0);
  458. }
  459. mFile = directory;
  460. }
  461. }
  462. /**
  463. * Change the adapted directory for a new one
  464. * @param folder New file to adapt. Can be NULL, meaning "no content to adapt".
  465. * @param updatedStorageManager Optional updated storage manager; used to replace mStorageManager if is different (and not NULL)
  466. */
  467. public void swapDirectory(long parentId, FileDataStorageManager updatedStorageManager) {
  468. FileDataStorageManager storageManager = null;
  469. if (updatedStorageManager != null && updatedStorageManager != storageManager) {
  470. storageManager = updatedStorageManager;
  471. }
  472. Cursor newCursor = null;
  473. if (storageManager != null) {
  474. mAdapter.setStorageManager(storageManager);
  475. mCursorLoader.setParentId(parentId);
  476. newCursor = mCursorLoader.loadInBackground();//storageManager.getContent(folder.getFileId());
  477. Uri uri = Uri.withAppendedPath(
  478. ProviderTableMeta.CONTENT_URI_DIR,
  479. String.valueOf(parentId));
  480. Log_OC.d(TAG, "swapDirectory Uri " + uri);
  481. //newCursor.setNotificationUri(getSherlockActivity().getContentResolver(), uri);
  482. }
  483. Cursor oldCursor = mAdapter.swapCursor(newCursor);
  484. if (oldCursor != null){
  485. oldCursor.close();
  486. }
  487. mAdapter.notifyDataSetChanged();
  488. }
  489. @Override
  490. public void onDismiss(EditNameDialog dialog) {
  491. if (dialog.getResult()) {
  492. String newFilename = dialog.getNewFilename();
  493. Log_OC.d(TAG, "name edit dialog dismissed with new name " + newFilename);
  494. mContainerActivity.getFileOperationsHelper().renameFile(mTargetFile, newFilename);
  495. }
  496. }
  497. @Override
  498. public void onConfirmation(String callerTag) {
  499. if (callerTag.equals(FileDetailFragment.FTAG_CONFIRMATION)) {
  500. FileDataStorageManager storageManager = mContainerActivity.getStorageManager();
  501. if (storageManager.getFileById(mTargetFile.getFileId()) != null) {
  502. mContainerActivity.getFileOperationsHelper().removeFile(mTargetFile, true);
  503. }
  504. }
  505. }
  506. @Override
  507. public void onNeutral(String callerTag) {
  508. mContainerActivity.getStorageManager().removeFile(mTargetFile, false, true); // TODO perform in background task / new thread
  509. listDirectory();
  510. mContainerActivity.onTransferStateChanged(mTargetFile, false, false);
  511. }
  512. @Override
  513. public void onCancel(String callerTag) {
  514. Log_OC.d(TAG, "REMOVAL CANCELED");
  515. }
  516. /***
  517. * LoaderManager.LoaderCallbacks<Cursor>
  518. */
  519. /**
  520. * Instantiate and return a new Loader for the given ID. This is where the cursor is created.
  521. */
  522. @Override
  523. public Loader<Cursor> onCreateLoader(int id, Bundle bundle) {
  524. Log_OC.d(TAG, "onCreateLoader start");
  525. mCursorLoader = new FileListCursorLoader(
  526. getSherlockActivity(),
  527. mContainerActivity.getStorageManager());
  528. if (mFile != null) {
  529. mCursorLoader.setParentId(mFile.getFileId());
  530. } else {
  531. mCursorLoader.setParentId(1);
  532. }
  533. Log_OC.d(TAG, "onCreateLoader end");
  534. return mCursorLoader;
  535. }
  536. /**
  537. * Called when a previously created loader has finished its load. Here, you can start using the cursor.
  538. */
  539. @Override
  540. public void onLoadFinished(Loader<Cursor> loader, Cursor cursor) {
  541. Log_OC.d(TAG, "onLoadFinished start");
  542. FileDataStorageManager storageManager = mContainerActivity.getStorageManager();
  543. if (storageManager != null) {
  544. mCursorLoader.setStorageManager(storageManager);
  545. if (mFile != null) {
  546. mCursorLoader.setParentId(mFile.getFileId());
  547. } else {
  548. mCursorLoader.setParentId(1);
  549. }
  550. mAdapter.swapCursor(mCursorLoader.loadInBackground());
  551. }
  552. // if(mAdapter != null && cursor != null)
  553. // mAdapter.swapCursor(cursor); //swap the new cursor in.
  554. // else
  555. // Log_OC.d(TAG,"OnLoadFinished: mAdapter is null");
  556. Log_OC.d(TAG, "onLoadFinished end");
  557. }
  558. /**
  559. * Called when a previously created loader is being reset, thus making its data unavailable.
  560. * It is being reset in order to create a new cursor to query different data.
  561. * This is called when the last Cursor provided to onLoadFinished() above is about to be closed.
  562. * We need to make sure we are no longer using it.
  563. */
  564. @Override
  565. public void onLoaderReset(Loader<Cursor> loader) {
  566. Log_OC.d(TAG, "onLoadReset start");
  567. if(mAdapter != null)
  568. mAdapter.swapCursor(null);
  569. else
  570. Log_OC.d(TAG,"OnLoadFinished: mAdapter is null");
  571. Log_OC.d(TAG, "onLoadReset end");
  572. }
  573. }