summaryrefslogtreecommitdiffstats
path: root/ui
diff options
context:
space:
mode:
authorsmckay <smckay@chromium.org>2015-01-06 16:29:38 -0800
committerCommit bot <commit-bot@chromium.org>2015-01-07 00:31:33 +0000
commit45e673e6ff837eeb500c2ad5bcbe5a799e8ba180 (patch)
treed92276d4f105ad901e762ee9ddfbf718fe96082b /ui
parent64c523540529f62a66f6a52e888317bf7fc262c1 (diff)
downloadchromium_src-45e673e6ff837eeb500c2ad5bcbe5a799e8ba180.zip
chromium_src-45e673e6ff837eeb500c2ad5bcbe5a799e8ba180.tar.gz
chromium_src-45e673e6ff837eeb500c2ad5bcbe5a799e8ba180.tar.bz2
Show a little circly/arrowey badge for files that have been copied locally.
Update import history code to support tracking locally copied files (and their destinations). Tracking destination is necessary as we'll eventually need to do lookups against destinations in order to mark items as imported when they are fully synced to the cloud. BUG=420680 TEST=browser_test: FileManagerJsTest.* Review URL: https://codereview.chromium.org/837563002 Cr-Commit-Position: refs/heads/master@{#310189}
Diffstat (limited to 'ui')
-rw-r--r--ui/file_manager/file_manager/background/js/import_history.js278
-rw-r--r--ui/file_manager/file_manager/background/js/import_history_unittest.html1
-rw-r--r--ui/file_manager/file_manager/background/js/import_history_unittest.js173
-rw-r--r--ui/file_manager/file_manager/background/js/media_import_handler.js6
-rw-r--r--ui/file_manager/file_manager/background/js/media_import_handler_unittest.js2
-rw-r--r--ui/file_manager/file_manager/background/js/test_import_history.js45
-rw-r--r--ui/file_manager/file_manager/common/js/importer_common.js2
-rw-r--r--ui/file_manager/file_manager/foreground/css/file_manager.css17
-rw-r--r--ui/file_manager/file_manager/foreground/images/files/ui/2x/copied_badge.pngbin0 -> 3157 bytes
-rw-r--r--ui/file_manager/file_manager/foreground/images/files/ui/copied_badge.pngbin0 -> 2944 bytes
-rw-r--r--ui/file_manager/file_manager/foreground/js/ui/file_grid.js14
11 files changed, 403 insertions, 135 deletions
diff --git a/ui/file_manager/file_manager/background/js/import_history.js b/ui/file_manager/file_manager/background/js/import_history.js
index 825ebe1..6bfbab0 100644
--- a/ui/file_manager/file_manager/background/js/import_history.js
+++ b/ui/file_manager/file_manager/background/js/import_history.js
@@ -16,6 +16,14 @@ importer.ImportHistory = function() {};
* @param {!FileEntry} entry
* @param {!importer.Destination} destination
* @return {!Promise.<boolean>} Resolves with true if the FileEntry
+ * was previously copied to the device.
+ */
+importer.ImportHistory.prototype.wasCopied;
+
+/**
+ * @param {!FileEntry} entry
+ * @param {!importer.Destination} destination
+ * @return {!Promise.<boolean>} Resolves with true if the FileEntry
* was previously imported to the specified destination.
*/
importer.ImportHistory.prototype.wasImported;
@@ -23,6 +31,13 @@ importer.ImportHistory.prototype.wasImported;
/**
* @param {!FileEntry} entry
* @param {!importer.Destination} destination
+ * @param {string} url
+ */
+importer.ImportHistory.prototype.markCopied;
+
+/**
+ * @param {!FileEntry} entry
+ * @param {!importer.Destination} destination
* @return {!Promise.<?>} Resolves when the operation is completed.
*/
importer.ImportHistory.prototype.markImported;
@@ -43,6 +58,7 @@ importer.ImportHistory.prototype.removeObserver;
/** @enum {string} */
importer.ImportHistory.State = {
+ 'COPIED': 'copied',
'IMPORTED': 'imported'
};
@@ -78,12 +94,24 @@ importer.DummyImportHistory.prototype.getHistory = function() {
};
/** @override */
+importer.DummyImportHistory.prototype.wasCopied =
+ function(entry, destination) {
+ return Promise.resolve(this.answer_);
+};
+
+/** @override */
importer.DummyImportHistory.prototype.wasImported =
function(entry, destination) {
return Promise.resolve(this.answer_);
};
/** @override */
+importer.DummyImportHistory.prototype.markCopied =
+ function(entry, destination, url) {
+ return Promise.resolve();
+};
+
+/** @override */
importer.DummyImportHistory.prototype.markImported =
function(entry, destination) {
return Promise.resolve();
@@ -96,6 +124,14 @@ importer.DummyImportHistory.prototype.addObserver = function(observer) {};
importer.DummyImportHistory.prototype.removeObserver = function(observer) {};
/**
+ * @private @enum {number}
+ */
+importer.RecordType_ = {
+ COPY: 0,
+ IMPORT: 1
+};
+
+/**
* An {@code ImportHistory} implementation that reads from and
* writes to a storage object.
*
@@ -111,12 +147,20 @@ importer.PersistentImportHistory = function(storage) {
this.storage_ = storage;
/**
+ * An in-memory representation of local copy history.
+ * The first value is the "key" (as generated internally
+ * from a file entry).
+ * @private {!Object.<string, !Object.<!importer.Destination, string>>}
+ */
+ this.copiedEntries_ = {};
+
+ /**
* An in-memory representation of import history.
* The first value is the "key" (as generated internally
* from a file entry).
* @private {!Object.<string, !Array.<importer.Destination>>}
*/
- this.entries_ = {};
+ this.importedEntries_ = {};
/** @private {!Array.<!importer.ImportHistory.Observer>} */
this.observers_ = [];
@@ -135,11 +179,15 @@ importer.PersistentImportHistory = function(storage) {
* @private
*/
importer.PersistentImportHistory.prototype.refresh_ = function() {
- var oldEntries = this.entries_;
- this.entries_ = {};
+ var oldCopiedEntries = this.copiedEntries_;
+ var oldImportedEntries = this.importedEntries_;
+
+ this.copiedEntries_ = {};
+ this.importedEntries_ = {};
return this.storage_.readAll()
.then(this.updateHistoryRecords_.bind(this))
- .then(this.mergeEntries_.bind(this, oldEntries))
+ .then(this.mergeCopiedEntries_.bind(this, oldCopiedEntries))
+ .then(this.mergeImportedEntries_.bind(this, oldImportedEntries))
.then(
/**
* @return {!importer.PersistentImportHistory}
@@ -151,32 +199,58 @@ importer.PersistentImportHistory.prototype.refresh_ = function() {
};
/**
- * Adds all entries not already present in history.
+ * Adds all copied entries into existing entries.
+ *
+ * @param {!Object.<string, !Object.<!importer.Destination, string>>} entries
+ * @return {!Promise.<?>} Resolves once all updates are completed.
+ * @private
+ */
+importer.PersistentImportHistory.prototype.mergeCopiedEntries_ =
+ function(entries) {
+ var promises = [];
+ for (var key in entries) {
+ for (var destination in entries[key]) {
+ // This method is only called when data is reloaded from disk.
+ // In such a situation we defend against loss of in-memory data
+ // by copying it out of the way, reloading data from disk, then
+ // merging that data back into the freshly loaded data. For that
+ // reason we will write newly created entries back to disk.
+
+ var path = entries[key][destination];
+ if (this.updateInMemoryRecord_(
+ importer.RecordType_.COPY, key, destination, path)) {
+ promises.push(this.storage_.write(
+ [importer.RecordType_.COPY, key, destination, path]));
+ }
+ }
+ }
+ return Promise.all(promises);
+};
+
+/**
+ * Adds all imported entries into existing entries.
*
* @param {!Object.<string, !Array.<string>>} entries
* @return {!Promise.<?>} Resolves once all updates are completed.
* @private
*/
-importer.PersistentImportHistory.prototype.mergeEntries_ = function(entries) {
+importer.PersistentImportHistory.prototype.mergeImportedEntries_ =
+ function(entries) {
var promises = [];
- Object.keys(entries).forEach(
- /**
- * @param {string} key
- * @this {importer.PersistentImportHistory}
- */
- function(key) {
- entries[key].forEach(
- /**
- * @param {string} key
- * @this {importer.PersistentImportHistory}
- */
- function(destination) {
- if (this.getDestinations_(key).indexOf(destination) >= 0) {
- this.updateHistoryRecord_(key, destination);
- promises.push(this.storage_.write([key, destination]));
- }
+ for (var key in entries) {
+ entries[key].forEach(
+ /**
+ * @param {string} key
+ * @this {importer.PersistentImportHistory}
+ */
+ function(destination) {
+ if (this.updateInMemoryRecord_(
+ importer.RecordType_.IMPORT, key, destination)) {
+ promises.push(this.storage_.write(
+ [importer.RecordType_.IMPORT, key, destination]));
+ }
}.bind(this));
- }.bind(this));
+ }
return Promise.all(promises);
};
@@ -212,26 +286,66 @@ importer.PersistentImportHistory.prototype.updateHistoryRecords_ =
* @this {importer.PersistentImportHistory}
*/
function(record) {
- this.updateHistoryRecord_(record[0], record[1]);
+ this.updateInMemoryRecord_(record[0], record[1], record[2], record[3]);
}.bind(this));
};
/**
* Adds a history entry to the in-memory history model.
+ * @param {!importer.RecordType_} type
* @param {string} key
- * @param {string} destination
+ * @param {!importer.Destination} destination
+ * @param {*=} opt_payload
+ * @return {boolean} True if a record was created.
* @private
*/
-importer.PersistentImportHistory.prototype.updateHistoryRecord_ =
- function(key, destination) {
- if (key in this.entries_) {
- this.entries_[key].push(destination);
- } else {
- this.entries_[key] = [destination];
+importer.PersistentImportHistory.prototype.updateInMemoryRecord_ =
+ function(type, key, destination, opt_payload) {
+
+ switch (type) {
+ case importer.RecordType_.COPY:
+ if (!this.copiedEntries_.hasOwnProperty(key)) {
+ this.copiedEntries_[key] = {};
+ }
+ if (!this.copiedEntries_[key].hasOwnProperty(destination)) {
+ this.copiedEntries_[key][destination] = /** @type {string} */ (
+ opt_payload);
+ return true;
+ }
+ break;
+ case importer.RecordType_.IMPORT:
+ if (!this.importedEntries_.hasOwnProperty(key)) {
+ this.importedEntries_[key] = [destination];
+ }
+ if (this.importedEntries_[key].indexOf(destination) === -1) {
+ this.importedEntries_[key].push(destination);
+ return true;
+ }
+ break;
+ default:
+ console.log('Ignoring record with unrecognized type: ' + type);
+ return false;
}
};
/** @override */
+importer.PersistentImportHistory.prototype.wasCopied =
+ function(entry, destination) {
+ return this.whenReady_
+ .then(this.createKey_.bind(this, entry))
+ .then(
+ /**
+ * @param {string} key
+ * @return {boolean}
+ * @this {importer.PersistentImportHistory}
+ */
+ function(key) {
+ return key in this.copiedEntries_ &&
+ destination in this.copiedEntries_[key];
+ }.bind(this));
+};
+
+/** @override */
importer.PersistentImportHistory.prototype.wasImported =
function(entry, destination) {
return this.whenReady_
@@ -239,7 +353,7 @@ importer.PersistentImportHistory.prototype.wasImported =
.then(
/**
* @param {string} key
- * @return {!Promise.<boolean>}
+ * @return {boolean}
* @this {importer.PersistentImportHistory}
*/
function(key) {
@@ -248,6 +362,30 @@ importer.PersistentImportHistory.prototype.wasImported =
};
/** @override */
+importer.PersistentImportHistory.prototype.markCopied =
+ function(entry, destination, url) {
+ return this.whenReady_
+ .then(this.createKey_.bind(this, entry))
+ .then(
+ /**
+ * @param {string} key
+ * @return {!Promise.<?>}
+ * @this {importer.ImportHistory}
+ */
+ function(key) {
+ return this.storeRecord_(
+ importer.RecordType_.COPY,
+ key,
+ destination,
+ url);
+ }.bind(this))
+ .then(this.notifyObservers_.bind(
+ this,
+ entry,
+ importer.ImportHistory.State.COPIED));
+};
+
+/** @override */
importer.PersistentImportHistory.prototype.markImported =
function(entry, destination) {
return this.whenReady_
@@ -259,9 +397,15 @@ importer.PersistentImportHistory.prototype.markImported =
* @this {importer.ImportHistory}
*/
function(key) {
- return this.addDestination_(destination, key);
+ return this.storeRecord_(
+ importer.RecordType_.IMPORT,
+ key,
+ destination);
}.bind(this))
- .then(this.notifyObservers_.bind(this, entry));
+ .then(this.notifyObservers_.bind(
+ this,
+ entry,
+ importer.ImportHistory.State.IMPORTED));
};
/** @override */
@@ -283,29 +427,34 @@ importer.PersistentImportHistory.prototype.removeObserver =
/**
* @param {!FileEntry} subject
+ * @param {!importer.ImportHistory.State} newState
* @private
*/
importer.PersistentImportHistory.prototype.notifyObservers_ =
- function(subject) {
+ function(subject, newState) {
this.observers_.forEach(
function(observer) {
observer({
- state: importer.ImportHistory.State.IMPORTED,
+ state: newState,
entry: subject
});
}.bind(this));
};
/**
- * @param {string} destination
+ * @param {!importer.RecordType_} type
* @param {string} key
+ * @param {!importer.Destination} destination
+ * @param {*=} opt_payload
* @return {!Promise.<?>} Resolves once the write has been completed.
* @private
*/
-importer.PersistentImportHistory.prototype.addDestination_ =
- function(destination, key) {
- this.updateHistoryRecord_(key, destination);
- return this.storage_.write([key, destination]);
+importer.PersistentImportHistory.prototype.storeRecord_ =
+ function(type, key, destination, opt_payload) {
+ this.updateInMemoryRecord_(type, key, destination, opt_payload);
+ return !!opt_payload ?
+ this.storage_.write([type, key, destination, opt_payload]) :
+ this.storage_.write([type, key, destination]);
};
/**
@@ -315,7 +464,7 @@ importer.PersistentImportHistory.prototype.addDestination_ =
* @private
*/
importer.PersistentImportHistory.prototype.getDestinations_ = function(key) {
- return key in this.entries_ ? this.entries_[key] : [];
+ return key in this.importedEntries_ ? this.importedEntries_[key] : [];
};
/**
@@ -476,7 +625,7 @@ importer.ChromeSyncFileEntryProvider = function() {
};
/** @private @const {string} */
-importer.ChromeSyncFileEntryProvider.FILE_NAME_ = 'import-history.data';
+importer.ChromeSyncFileEntryProvider.FILE_NAME_ = 'import-history.log';
/**
* Wraps chrome.syncFileSystem.onFileStatusChanged
@@ -678,6 +827,9 @@ importer.FileEntryRecordStorage.prototype.write = function(record) {
*/
importer.FileEntryRecordStorage.prototype.writeRecord_ =
function(record, writer) {
+ var blob = new Blob(
+ [JSON.stringify(record) + ',\n'],
+ {type: 'text/plain; charset=UTF-8'});
return new Promise(
/**
* @param {function()} resolve
@@ -685,10 +837,6 @@ importer.FileEntryRecordStorage.prototype.writeRecord_ =
* @this {importer.FileEntryRecordStorage}
*/
function(resolve, reject) {
- var blob = new Blob(
- [JSON.stringify(record) + ',\n'],
- {type: 'text/plain; charset=UTF-8'});
-
writer.onwriteend = resolve;
writer.onerror = reject;
@@ -759,31 +907,23 @@ importer.FileEntryRecordStorage.prototype.readFileAsText_ = function(file) {
* Parses the text.
*
* @param {string} text
- * @return {!Promise.<!Array.<!Array.<*>>>}
+ * @return {!Array.<!Array.<*>>}
* @private
*/
importer.FileEntryRecordStorage.prototype.parse_ = function(text) {
- return new Promise(
- /**
- * @param {function()} resolve
- * @param {function()} reject
- * @this {importer.FileEntryRecordStorage}
- */
- function(resolve, reject) {
- if (text.length === 0) {
- resolve([]);
- } else {
- // Dress up the contents of the file like an array,
- // so the JSON object can parse it using JSON.parse.
- // That means we need to both:
- // 1) Strip the trailing ',\n' from the last record
- // 2) Surround the whole string in brackets.
- // NOTE: JSON.parse is WAY faster than parsing this
- // ourselves in javascript.
- var json = '[' + text.substring(0, text.length - 2) + ']';
- resolve(JSON.parse(json));
- }
- }.bind(this));
+ if (text.length === 0) {
+ return [];
+ } else {
+ // Dress up the contents of the file like an array,
+ // so the JSON object can parse it using JSON.parse.
+ // That means we need to both:
+ // 1) Strip the trailing ',\n' from the last record
+ // 2) Surround the whole string in brackets.
+ // NOTE: JSON.parse is WAY faster than parsing this
+ // ourselves in javascript.
+ var json = '[' + text.substring(0, text.length - 2) + ']';
+ return /** @type {!Array.<!Array.<*>>} */ (JSON.parse(json));
+ }
};
/**
diff --git a/ui/file_manager/file_manager/background/js/import_history_unittest.html b/ui/file_manager/file_manager/background/js/import_history_unittest.html
index 4343fff..bf582b2 100644
--- a/ui/file_manager/file_manager/background/js/import_history_unittest.html
+++ b/ui/file_manager/file_manager/background/js/import_history_unittest.html
@@ -7,6 +7,7 @@
<html>
<body>
+<script src="../../common/js/volume_manager_common.js"></script>
<script src="../../common/js/importer_common.js"></script>
<script src="../../common/js/mock_entry.js"></script>
<script src="../../common/js/unittest_util.js"></script>
diff --git a/ui/file_manager/file_manager/background/js/import_history_unittest.js b/ui/file_manager/file_manager/background/js/import_history_unittest.js
index 5d720ce..8b1d45a 100644
--- a/ui/file_manager/file_manager/background/js/import_history_unittest.js
+++ b/ui/file_manager/file_manager/background/js/import_history_unittest.js
@@ -12,7 +12,10 @@ var FILE_SIZE = 1234;
var FILE_PATH = 'test/data';
/** @const {string} */
-var GOOGLE_DRIVE = 'Google Drive';
+var DEVICE = importer.Destination.DEVICE;
+
+/** @const {string} */
+var GOOGLE_DRIVE = importer.Destination.GOOGLE_DRIVE;
/**
* Space Cloud: Your source for interstellar cloud storage.
@@ -32,6 +35,9 @@ var storage;
/** @type {!Promise.<!importer.PersistentImportHistory>|undefined} */
var historyProvider;
+/** @type {Promise} */
+var testPromise;
+
// Set up the test components.
function setUp() {
testFileEntry = new MockFileEntry(
@@ -49,53 +55,95 @@ function setUp() {
historyProvider = history.refresh();
}
+function testHistoryWasCopiedFalseForUnknownEntry(callback) {
+ // TestRecordWriter is pre-configured with a Space Cloud entry
+ // but not for this file.
+ testPromise = historyProvider.then(
+ function(history) {
+ history.wasCopied(testFileEntry, SPACE_CAMP).then(assertFalse);
+ });
+
+ reportPromise(testPromise, callback);
+}
+
+function testHistoryWasCopiedTrueForKnownEntryLoadedFromStorage(callback) {
+ // TestRecordWriter is pre-configured with this entry.
+ testPromise = historyProvider.then(
+ function(history) {
+ history.wasCopied(testFileEntry, GOOGLE_DRIVE).then(assertTrue);
+ });
+
+ reportPromise(testPromise, callback);
+}
+
+function testHistoryWasImportedTrueForKnownEntrySetAtRuntime(callback) {
+ testPromise = historyProvider.then(
+ function(history) {
+ history.markImported(testFileEntry, SPACE_CAMP).then(
+ function() {
+ history.wasImported(testFileEntry, SPACE_CAMP).then(assertTrue);
+ });
+ });
+
+ reportPromise(testPromise, callback);
+}
+
+function testCopyChangeFiresChangedEvent(callback) {
+ testPromise = historyProvider.then(
+ function(history) {
+ var recorder = new TestCallRecorder();
+ history.addObserver(recorder.callback);
+ history.markCopied(testFileEntry, SPACE_CAMP, 'url1').then(
+ function() {
+ Promise.resolve()
+ .then(
+ function() {
+ recorder.assertCallCount(1);
+ assertEquals(
+ importer.ImportHistory.State.COPIED,
+ recorder.getLastArguments()[0]['state']);
+ });
+ });
+ });
+
+ reportPromise(testPromise, callback);
+}
+
function testHistoryWasImportedFalseForUnknownEntry(callback) {
// TestRecordWriter is pre-configured with a Space Cloud entry
// but not for this file.
- historyProvider.then(
+ testPromise = historyProvider.then(
function(history) {
- history.wasImported(testFileEntry, SPACE_CAMP).then(
- function(result) {
- callback(/* error */ result);
- },
- callback.bind(null, true));
- },
- callback.bind(null, true))
- .catch(handleError.bind(null, callback));
+ history.wasImported(testFileEntry, SPACE_CAMP).then(assertFalse);
+ });
+
+ reportPromise(testPromise, callback);
}
function testHistoryWasImportedTrueForKnownEntryLoadedFromStorage(callback) {
// TestRecordWriter is pre-configured with this entry.
- historyProvider.then(
+ testPromise = historyProvider.then(
function(history) {
- history.wasImported(testFileEntry, GOOGLE_DRIVE).then(
- function(result) {
- callback(/* error */ !result);
- },
- callback.bind(null, true));
- },
- callback.bind(null, true))
- .catch(handleError.bind(null, callback));
+ history.wasImported(testFileEntry, GOOGLE_DRIVE).then(assertTrue);
+ });
+
+ reportPromise(testPromise, callback);
}
function testHistoryWasImportedTrueForKnownEntrySetAtRuntime(callback) {
- historyProvider.then(
+ testPromise = historyProvider.then(
function(history) {
history.markImported(testFileEntry, SPACE_CAMP).then(
function() {
- history.wasImported(testFileEntry, SPACE_CAMP).then(
- function(result) {
- callback(/* error */ !result);
- });
- },
- callback.bind(null, true));
- },
- callback.bind(null, true))
- .catch(handleError.bind(null, callback));
+ history.wasImported(testFileEntry, SPACE_CAMP).then(assertTrue);
+ });
+ });
+
+ reportPromise(testPromise, callback);
}
-function testHistoryChangeFiresChangedEvent(callback) {
- historyProvider.then(
+function testImportChangeFiresChangedEvent(callback) {
+ testPromise = historyProvider.then(
function(history) {
var recorder = new TestCallRecorder();
history.addObserver(recorder.callback);
@@ -108,53 +156,50 @@ function testHistoryChangeFiresChangedEvent(callback) {
assertEquals(
importer.ImportHistory.State.IMPORTED,
recorder.getLastArguments()[0]['state']);
- callback(false);
- })
- .catch(handleError.bind(null, callback));
- },
- callback.bind(null, true));
- },
- callback.bind(null, true))
- .catch(handleError.bind(null, callback));
+ });
+ });
+ });
+
+ reportPromise(testPromise, callback);
}
function testHistoryObserverUnsubscribe(callback) {
- historyProvider.then(
+ testPromise = historyProvider.then(
function(history) {
var recorder = new TestCallRecorder();
history.addObserver(recorder.callback);
history.removeObserver(recorder.callback);
- history.markImported(testFileEntry, SPACE_CAMP).then(
+
+ var promises = [];
+ promises.push(history.markCopied(testFileEntry, SPACE_CAMP, 'url2'));
+ promises.push(history.markImported(testFileEntry, SPACE_CAMP));
+ Promise.all(promises).then(
function() {
Promise.resolve()
.then(
function() {
recorder.assertCallCount(0);
- callback(false);
- })
- .catch(handleError.bind(null, callback));
- },
- callback.bind(null, true));
- },
- callback.bind(null, true))
- .catch(handleError.bind(null, callback));
+ });
+ });
+ });
+
+ reportPromise(testPromise, callback);
}
function testRecordStorageRemembersPreviouslyWrittenRecords(callback) {
- createRealStorage('recordStorageTest.data')
+ testPromise = createRealStorage('recordStorageTest.data')
.then(
function(storage) {
storage.write(['abc', '123']).then(
function() {
storage.readAll().then(
function(records) {
- callback(/* error */ records.length != 1);
- },
- callback);
+ assertTrue(records.length != 1);
+ });
});
- },
- callback)
- .catch(handleError.bind(null, callback));
+ });
+
+ reportPromise(testPromise, callback);
}
function testHistoryLoaderIntegration(callback) {
@@ -171,7 +216,7 @@ function testHistoryLoaderIntegration(callback) {
/** @type {!TestSyncFileEntryProvider|undefined} */
var syncFileProvider;
- createFileEntry('historyLoaderTest.data')
+ testPromise = createFileEntry('historyLoaderTest.data')
.then(
/**
* @param {!FileEntry} fileEntry
@@ -208,6 +253,8 @@ function testHistoryLoaderIntegration(callback) {
function() {
syncFileProvider.fireSyncEvent();
})
+ // TODO(smckay): Add markCopied support once FileWriter issues
+ // are resolved.
.then(
/**
* @return {!Promise.<boolean>} Resolves with true if the
@@ -218,9 +265,10 @@ function testHistoryLoaderIntegration(callback) {
})
.then(
function(wasImported) {
- callback(/* error */ !wasImported);
- })
- .catch(handleError.bind(null, callback));
+ assertTrue(wasImported);
+ });
+
+ reportPromise(testPromise, callback);
}
/**
@@ -285,8 +333,9 @@ var TestRecordStorage = function() {
// Pre-populate the store with some "previously written" data <wink>.
/** @private {!Array.<!Array.<string>>} */
this.records_ = [
- [FILE_LAST_MODIFIED + '_' + FILE_SIZE, GOOGLE_DRIVE],
- ['99999_99999', SPACE_CAMP]
+ [1, FILE_LAST_MODIFIED + '_' + FILE_SIZE, GOOGLE_DRIVE],
+ [0, FILE_LAST_MODIFIED + '_' + FILE_SIZE, 'google-drive', 'url4'],
+ [1, '99999_99999', SPACE_CAMP]
];
/**
diff --git a/ui/file_manager/file_manager/background/js/media_import_handler.js b/ui/file_manager/file_manager/background/js/media_import_handler.js
index 4e62ee2..ac49ac7 100644
--- a/ui/file_manager/file_manager/background/js/media_import_handler.js
+++ b/ui/file_manager/file_manager/background/js/media_import_handler.js
@@ -204,10 +204,14 @@ importer.MediaImportHandler.ImportTask.prototype.onProgress_ =
/** @param {!FileEntry} entry */
importer.MediaImportHandler.ImportTask.prototype.markAsCopied_ =
function(entry) {
+ var destinationUrl = this.destination_.toURL() + '/' + entry.name;
this.historyLoader_.getHistory().then(
/** @param {!importer.ImportHistory} history */
function(history) {
- history.markImported(entry, importer.Destination.GOOGLE_DRIVE);
+ history.markCopied(
+ entry,
+ importer.Destination.GOOGLE_DRIVE,
+ destinationUrl);
});
};
diff --git a/ui/file_manager/file_manager/background/js/media_import_handler_unittest.js b/ui/file_manager/file_manager/background/js/media_import_handler_unittest.js
index b7a2d79..f3217b8 100644
--- a/ui/file_manager/file_manager/background/js/media_import_handler_unittest.js
+++ b/ui/file_manager/file_manager/background/js/media_import_handler_unittest.js
@@ -163,7 +163,7 @@ function testUpdatesHistoryAfterImport(callback) {
importedMedia.forEach(
/** @param {!CopyCapture} */
function(capture) {
- importHistory.assertImported(
+ importHistory.assertCopied(
capture.source, importer.Destination.GOOGLE_DRIVE);
});
}),
diff --git a/ui/file_manager/file_manager/background/js/test_import_history.js b/ui/file_manager/file_manager/background/js/test_import_history.js
index cb78260..880517e 100644
--- a/ui/file_manager/file_manager/background/js/test_import_history.js
+++ b/ui/file_manager/file_manager/background/js/test_import_history.js
@@ -15,8 +15,12 @@ var importer = importer || {};
* @implements {importer.ImportHistory}
*/
importer.TestImportHistory = function() {
+ /** @type {!Object.<string, !Object.<!importer.Destination, string>>} */
+ this.copiedPaths = {};
+
/** @type {!Object.<string, Array.<string>>} */
this.importedPaths = {};
+
};
/** @override */
@@ -29,6 +33,47 @@ importer.TestImportHistory.prototype.getHistory =
* @param {!FileEntry} entry
* @param {!importer.Destination} destination
*/
+importer.TestImportHistory.prototype.assertCopied =
+ function(entry, destination) {
+ assertTrue(this.wasCopied_(entry, destination));
+};
+
+/**
+ * Fully synchronous version of wasCopied.
+ * @param {!FileEntry} entry
+ * @param {!importer.Destination} destination
+ * @return {boolean}
+ */
+importer.TestImportHistory.prototype.wasCopied_ =
+ function(entry, destination) {
+ var path = entry.fullPath;
+ return path in this.copiedPaths &&
+ this.copiedPaths[path].indexOf(destination) > -1;
+};
+
+/** @override */
+importer.TestImportHistory.prototype.wasCopied =
+ function(entry, destination) {
+ var path = entry.fullPath;
+ return Promise.resolve(this.wasCopied_(entry, destination));
+};
+
+/** @override */
+importer.TestImportHistory.prototype.markCopied =
+ function(entry, destination) {
+ var path = entry.fullPath;
+ if (path in this.copiedPaths) {
+ this.copiedPaths[path].push(destination);
+ } else {
+ this.copiedPaths[path] = [destination];
+ }
+ return Promise.resolve();
+};
+
+/**
+ * @param {!FileEntry} entry
+ * @param {!importer.Destination} destination
+ */
importer.TestImportHistory.prototype.assertImported =
function(entry, destination) {
assertTrue(this.wasImported_(entry, destination));
diff --git a/ui/file_manager/file_manager/common/js/importer_common.js b/ui/file_manager/file_manager/common/js/importer_common.js
index 3b77f1e..f912c6e 100644
--- a/ui/file_manager/file_manager/common/js/importer_common.js
+++ b/ui/file_manager/file_manager/common/js/importer_common.js
@@ -29,6 +29,8 @@ importer.ELIGIBLE_VOLUME_TYPES_ = [
* @enum {string}
*/
importer.Destination = {
+ // locally copied, but not imported to cloud as of yet.
+ DEVICE: 'device',
GOOGLE_DRIVE: 'google-drive'
};
diff --git a/ui/file_manager/file_manager/foreground/css/file_manager.css b/ui/file_manager/file_manager/foreground/css/file_manager.css
index ca3fb0d..a42950f 100644
--- a/ui/file_manager/file_manager/foreground/css/file_manager.css
+++ b/ui/file_manager/file_manager/foreground/css/file_manager.css
@@ -750,6 +750,20 @@ body[type='full-page'] .detail-name .detail-icon {
height: 30px;
}
+.badge {
+ height: 16px;
+ position: absolute;
+ right: 7px;
+ top: 7px;
+ width: 16px;
+}
+
+.copied .badge {
+ background-image: -webkit-image-set(
+ url(../images/files/ui/copied_badge.png) 1x,
+ url(../images/files/ui/2x/copied_badge.png) 2x);
+}
+
.imported .badge {
background-image: -webkit-image-set(
url(../images/files/ui/drive_badge.png) 1x,
@@ -761,7 +775,8 @@ body[type='full-page'] .detail-name .detail-icon {
width: 16px;
}
-.imported .filename-label {
+.imported .filename-label,
+.copied .filename-label {
margin-right: 14px;
}
diff --git a/ui/file_manager/file_manager/foreground/images/files/ui/2x/copied_badge.png b/ui/file_manager/file_manager/foreground/images/files/ui/2x/copied_badge.png
new file mode 100644
index 0000000..a9e2bda
--- /dev/null
+++ b/ui/file_manager/file_manager/foreground/images/files/ui/2x/copied_badge.png
Binary files differ
diff --git a/ui/file_manager/file_manager/foreground/images/files/ui/copied_badge.png b/ui/file_manager/file_manager/foreground/images/files/ui/copied_badge.png
new file mode 100644
index 0000000..deb929e
--- /dev/null
+++ b/ui/file_manager/file_manager/foreground/images/files/ui/copied_badge.png
Binary files differ
diff --git a/ui/file_manager/file_manager/foreground/js/ui/file_grid.js b/ui/file_manager/file_manager/foreground/js/ui/file_grid.js
index fc20538..8c09833 100644
--- a/ui/file_manager/file_manager/foreground/js/ui/file_grid.js
+++ b/ui/file_manager/file_manager/foreground/js/ui/file_grid.js
@@ -294,11 +294,23 @@ FileGrid.applyHistoryBadges_ = function(entry, box, history) {
// a possibly-fragile sibling selector we just
// plop the imported class on the parent of both.
box.parentElement.classList.add('imported');
+ } else {
+ history.wasCopied(entry, importer.Destination.GOOGLE_DRIVE)
+ .then(
+ function(copied) {
+ if (copied) {
+ // TODO(smckay): update badges when history changes
+ // "box" is currently the sibling of the elemement
+ // we want to style. So rather than employing
+ // a possibly-fragile sibling selector we just
+ // plop the imported class on the parent of both.
+ box.parentElement.classList.add('copied');
+ }
+ });
}
});
};
-
/**
* Item for the Grid View.
* @constructor