|
@@ -23,6 +23,7 @@ import java.util.ArrayList;
|
|
|
import java.util.Collection;
|
|
|
import java.util.Collections;
|
|
|
import java.util.Iterator;
|
|
|
+import java.util.List;
|
|
|
import java.util.Vector;
|
|
|
|
|
|
import com.owncloud.android.MainApp;
|
|
@@ -33,7 +34,6 @@ import com.owncloud.android.lib.resources.shares.ShareType;
|
|
|
import com.owncloud.android.lib.resources.files.FileUtils;
|
|
|
import com.owncloud.android.utils.FileStorageUtils;
|
|
|
|
|
|
-
|
|
|
import android.accounts.Account;
|
|
|
import android.content.ContentProviderClient;
|
|
|
import android.content.ContentProviderOperation;
|
|
@@ -41,6 +41,7 @@ import android.content.ContentProviderResult;
|
|
|
import android.content.ContentResolver;
|
|
|
import android.content.ContentUris;
|
|
|
import android.content.ContentValues;
|
|
|
+import android.content.Intent;
|
|
|
import android.content.OperationApplicationException;
|
|
|
import android.database.Cursor;
|
|
|
import android.net.Uri;
|
|
@@ -346,7 +347,9 @@ public class FileDataStorageManager {
|
|
|
).withSelection(where, whereArgs).build());
|
|
|
|
|
|
if (file.isDown()) {
|
|
|
- new File(file.getStoragePath()).delete();
|
|
|
+ String path = file.getStoragePath();
|
|
|
+ new File(path).delete();
|
|
|
+ triggerMediaScan(path); // notify MediaScanner about removed file
|
|
|
}
|
|
|
}
|
|
|
}
|
|
@@ -484,8 +487,12 @@ public class FileDataStorageManager {
|
|
|
}
|
|
|
success &= (deleted > 0);
|
|
|
}
|
|
|
- if (removeLocalCopy && file.isDown() && file.getStoragePath() != null && success) {
|
|
|
- success = new File(file.getStoragePath()).delete();
|
|
|
+ String localPath = file.getStoragePath();
|
|
|
+ if (removeLocalCopy && file.isDown() && localPath != null && success) {
|
|
|
+ success = new File(localPath).delete();
|
|
|
+ if (success) {
|
|
|
+ triggerMediaScan(localPath);
|
|
|
+ }
|
|
|
if (!removeDBData && success) {
|
|
|
// maybe unnecessary, but should be checked TODO remove if unnecessary
|
|
|
file.setStoragePath(null);
|
|
@@ -542,11 +549,13 @@ public class FileDataStorageManager {
|
|
|
success &= removeLocalFolder(file);
|
|
|
} else {
|
|
|
if (file.isDown()) {
|
|
|
+ String path = file.getStoragePath();
|
|
|
File localFile = new File(file.getStoragePath());
|
|
|
success &= localFile.delete();
|
|
|
if (success) {
|
|
|
file.setStoragePath(null);
|
|
|
saveFile(file);
|
|
|
+ triggerMediaScan(path); // notify MediaScanner about removed file
|
|
|
}
|
|
|
}
|
|
|
}
|
|
@@ -568,7 +577,9 @@ public class FileDataStorageManager {
|
|
|
if (localFile.isDirectory()) {
|
|
|
success &= removeLocalFolder(localFile);
|
|
|
} else {
|
|
|
+ String path = localFile.getAbsolutePath();
|
|
|
success &= localFile.delete();
|
|
|
+ triggerMediaScan(path); // notify MediaScanner about removed file
|
|
|
}
|
|
|
}
|
|
|
}
|
|
@@ -576,109 +587,20 @@ public class FileDataStorageManager {
|
|
|
return success;
|
|
|
}
|
|
|
|
|
|
+
|
|
|
/**
|
|
|
- * Updates database for a folder that was moved to a different location.
|
|
|
+ * Updates database and file system for a file or folder that was moved to a different location.
|
|
|
*
|
|
|
* TODO explore better (faster) implementations
|
|
|
* TODO throw exceptions up !
|
|
|
*/
|
|
|
- public void moveFolder(OCFile folder, String newPath) {
|
|
|
- // TODO check newPath
|
|
|
-
|
|
|
- if ( folder != null && folder.isFolder() &&
|
|
|
- folder.fileExists() && !OCFile.ROOT_PATH.equals(folder.getFileName())
|
|
|
- ) {
|
|
|
- /// 1. get all the descendants of 'dir' in a single QUERY (including 'dir')
|
|
|
- Cursor c = null;
|
|
|
- if (getContentProviderClient() != null) {
|
|
|
- try {
|
|
|
- c = getContentProviderClient().query (
|
|
|
- ProviderTableMeta.CONTENT_URI,
|
|
|
- null,
|
|
|
- ProviderTableMeta.FILE_ACCOUNT_OWNER + "=? AND " +
|
|
|
- ProviderTableMeta.FILE_PATH + " LIKE ? ",
|
|
|
- new String[] { mAccount.name, folder.getRemotePath() + "%" },
|
|
|
- ProviderTableMeta.FILE_PATH + " ASC "
|
|
|
- );
|
|
|
- } catch (RemoteException e) {
|
|
|
- Log_OC.e(TAG, e.getMessage());
|
|
|
- }
|
|
|
- } else {
|
|
|
- c = getContentResolver().query (
|
|
|
- ProviderTableMeta.CONTENT_URI,
|
|
|
- null,
|
|
|
- ProviderTableMeta.FILE_ACCOUNT_OWNER + "=? AND " +
|
|
|
- ProviderTableMeta.FILE_PATH + " LIKE ? ",
|
|
|
- new String[] { mAccount.name, folder.getRemotePath() + "%" },
|
|
|
- ProviderTableMeta.FILE_PATH + " ASC "
|
|
|
- );
|
|
|
- }
|
|
|
-
|
|
|
- /// 2. prepare a batch of update operations to change all the descendants
|
|
|
- ArrayList<ContentProviderOperation> operations =
|
|
|
- new ArrayList<ContentProviderOperation>(c.getCount());
|
|
|
- int lengthOfOldPath = folder.getRemotePath().length();
|
|
|
- String defaultSavePath = FileStorageUtils.getSavePath(mAccount.name);
|
|
|
- int lengthOfOldStoragePath = defaultSavePath.length() + lengthOfOldPath;
|
|
|
- if (c.moveToFirst()) {
|
|
|
- do {
|
|
|
- ContentValues cv = new ContentValues(); // keep the constructor in the loop
|
|
|
- OCFile child = createFileInstance(c);
|
|
|
- cv.put(
|
|
|
- ProviderTableMeta.FILE_PATH,
|
|
|
- newPath + child.getRemotePath().substring(lengthOfOldPath)
|
|
|
- );
|
|
|
- if ( child.getStoragePath() != null &&
|
|
|
- child.getStoragePath().startsWith(defaultSavePath) ) {
|
|
|
- cv.put(
|
|
|
- ProviderTableMeta.FILE_STORAGE_PATH,
|
|
|
- defaultSavePath + newPath +
|
|
|
- child.getStoragePath().substring(lengthOfOldStoragePath)
|
|
|
- );
|
|
|
- }
|
|
|
- operations.add(
|
|
|
- ContentProviderOperation.
|
|
|
- newUpdate(ProviderTableMeta.CONTENT_URI).
|
|
|
- withValues(cv).
|
|
|
- withSelection(
|
|
|
- ProviderTableMeta._ID + "=?",
|
|
|
- new String[] { String.valueOf(child.getFileId()) }
|
|
|
- ).
|
|
|
- build()
|
|
|
- );
|
|
|
- } while (c.moveToNext());
|
|
|
- }
|
|
|
- c.close();
|
|
|
-
|
|
|
- /// 3. apply updates in batch
|
|
|
- try {
|
|
|
- if (getContentResolver() != null) {
|
|
|
- getContentResolver().applyBatch(MainApp.getAuthority(), operations);
|
|
|
-
|
|
|
- } else {
|
|
|
- getContentProviderClient().applyBatch(operations);
|
|
|
- }
|
|
|
-
|
|
|
- } catch (OperationApplicationException e) {
|
|
|
- Log_OC.e(TAG, "Fail to update descendants of " +
|
|
|
- folder.getFileId() + " in database", e);
|
|
|
-
|
|
|
- } catch (RemoteException e) {
|
|
|
- Log_OC.e(TAG, "Fail to update desendants of " +
|
|
|
- folder.getFileId() + " in database", e);
|
|
|
- }
|
|
|
-
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
-
|
|
|
public void moveLocalFile(OCFile file, String targetPath, String targetParentPath) {
|
|
|
|
|
|
if (file != null && file.fileExists() && !OCFile.ROOT_PATH.equals(file.getFileName())) {
|
|
|
|
|
|
OCFile targetParent = getFileByPath(targetParentPath);
|
|
|
if (targetParent == null) {
|
|
|
- // TODO panic
|
|
|
+ throw new IllegalStateException("Parent folder of the target path does not exist!!");
|
|
|
}
|
|
|
|
|
|
/// 1. get all the descendants of the moved element in a single QUERY
|
|
@@ -718,6 +640,8 @@ public class FileDataStorageManager {
|
|
|
ArrayList<ContentProviderOperation> operations =
|
|
|
new ArrayList<ContentProviderOperation>(c.getCount());
|
|
|
String defaultSavePath = FileStorageUtils.getSavePath(mAccount.name);
|
|
|
+ List<String> originalPathsToTriggerMediaScan = new ArrayList<String>();
|
|
|
+ List<String> newPathsToTriggerMediaScan = new ArrayList<String>();
|
|
|
if (c.moveToFirst()) {
|
|
|
int lengthOfOldPath = file.getRemotePath().length();
|
|
|
int lengthOfOldStoragePath = defaultSavePath.length() + lengthOfOldPath;
|
|
@@ -731,11 +655,14 @@ public class FileDataStorageManager {
|
|
|
if (child.getStoragePath() != null &&
|
|
|
child.getStoragePath().startsWith(defaultSavePath)) {
|
|
|
// update link to downloaded content - but local move is not done here!
|
|
|
- cv.put(
|
|
|
- ProviderTableMeta.FILE_STORAGE_PATH,
|
|
|
- defaultSavePath + targetPath +
|
|
|
- child.getStoragePath().substring(lengthOfOldStoragePath)
|
|
|
- );
|
|
|
+ String targetLocalPath = defaultSavePath + targetPath +
|
|
|
+ child.getStoragePath().substring(lengthOfOldStoragePath);
|
|
|
+
|
|
|
+ cv.put(ProviderTableMeta.FILE_STORAGE_PATH, targetLocalPath);
|
|
|
+
|
|
|
+ originalPathsToTriggerMediaScan.add(child.getStoragePath());
|
|
|
+ newPathsToTriggerMediaScan.add(targetLocalPath);
|
|
|
+
|
|
|
}
|
|
|
if (child.getRemotePath().equals(file.getRemotePath())) {
|
|
|
cv.put(
|
|
@@ -766,27 +693,35 @@ public class FileDataStorageManager {
|
|
|
}
|
|
|
|
|
|
} catch (Exception e) {
|
|
|
- Log_OC.e(
|
|
|
- TAG,
|
|
|
- "Fail to update " + file.getFileId() + " and descendants in database",
|
|
|
- e
|
|
|
- );
|
|
|
+ Log_OC.e(TAG, "Fail to update " + file.getFileId() + " and descendants in database", e);
|
|
|
}
|
|
|
|
|
|
/// 4. move in local file system
|
|
|
- String localPath = FileStorageUtils.getDefaultSavePathFor(mAccount.name, file);
|
|
|
- File localFile = new File(localPath);
|
|
|
+ String originalLocalPath = FileStorageUtils.getDefaultSavePathFor(mAccount.name, file);
|
|
|
+ String targetLocalPath = defaultSavePath + targetPath;
|
|
|
+ File localFile = new File(originalLocalPath);
|
|
|
boolean renamed = false;
|
|
|
if (localFile.exists()) {
|
|
|
- File targetFile = new File(defaultSavePath + targetPath);
|
|
|
+ File targetFile = new File(targetLocalPath);
|
|
|
File targetFolder = targetFile.getParentFile();
|
|
|
if (!targetFolder.exists()) {
|
|
|
targetFolder.mkdirs();
|
|
|
}
|
|
|
renamed = localFile.renameTo(targetFile);
|
|
|
}
|
|
|
- Log_OC.d(TAG, "Local file RENAMED : " + renamed);
|
|
|
-
|
|
|
+
|
|
|
+ if (renamed) {
|
|
|
+ Iterator<String> it = originalPathsToTriggerMediaScan.iterator();
|
|
|
+ while (it.hasNext()) {
|
|
|
+ // Notify MediaScanner about removed file
|
|
|
+ triggerMediaScan(it.next());
|
|
|
+ }
|
|
|
+ it = newPathsToTriggerMediaScan.iterator();
|
|
|
+ while (it.hasNext()) {
|
|
|
+ // Notify MediaScanner about new file/folder
|
|
|
+ triggerMediaScan(it.next());
|
|
|
+ }
|
|
|
+ }
|
|
|
}
|
|
|
|
|
|
}
|
|
@@ -1549,4 +1484,10 @@ public class FileDataStorageManager {
|
|
|
//}
|
|
|
}
|
|
|
|
|
|
+ public void triggerMediaScan(String path) {
|
|
|
+ Intent intent = new Intent(Intent.ACTION_MEDIA_SCANNER_SCAN_FILE);
|
|
|
+ intent.setData(Uri.fromFile(new File(path)));
|
|
|
+ MainApp.getAppContext().sendBroadcast(intent);
|
|
|
+ }
|
|
|
+
|
|
|
}
|