Caret/js/storage/file.js

157 lines
3.7 KiB
JavaScript
Raw Normal View History

define([
"util/manos"
], function(M) {
2013-08-20 00:53:03 +00:00
/*
The File object provides storage backed by the local HDD via the chrome.fileSystem API.
2013-08-20 00:53:03 +00:00
*/
2013-11-06 20:02:20 +00:00
var noop = function() {};
var File = function(entry) {
this.entry = entry || null;
2013-11-06 20:02:20 +00:00
this.onWrite = noop;
2013-08-20 00:53:03 +00:00
};
File.prototype = {
open: function(mode, c) {
2013-08-20 00:53:03 +00:00
var self = this;
if (typeof mode == "function") {
c = mode;
mode = "open";
}
2013-08-20 00:53:03 +00:00
//mode is "open" or "save"
var modes = {
"open": "openWritableFile",
"save": "saveFile"
};
chrome.fileSystem.chooseEntry({
type: modes[mode]
}, function(entry) {
//cancelling acts like an error, but isn't.
if (!entry) return c(chrome.runtime.lastError);
self.entry = entry;
c(null, self);
2013-08-20 00:53:03 +00:00
});
},
read: function(c) {
var self = this;
if (!self.entry) {
console.error(self);
c("File not opened");
}
var reader = new FileReader();
reader.onload = function() {
c(null, reader.result);
};
reader.onerror = function(err) {
console.error("File read error!");
c(err);
};
self.entry.file(function(f) {
reader.readAsText(f);
2013-08-20 00:53:03 +00:00
});
},
write: function(data, c) {
2013-08-20 00:53:03 +00:00
var self = this;
c = c || function() {};
if (!self.entry) {
//guard against cases where we accidentally write before opening
return self.open("save").then(function() {
return self.write(data, c);
});
}
M.chain(
//check permissions
function(next) {
chrome.fileSystem.isWritableEntry(self.entry, next);
},
//if read-only, try to open as writable
function(ok, next) {
if (!ok) {
return chrome.fileSystem.getWritableEntry(self.entry, function(entry) {
if (entry) {
self.entry = entry;
next();
} else {
c("Couldn't open file as writable");
}
});
2013-08-20 00:53:03 +00:00
}
next();
},
//write file
function() {
self.entry.createWriter(function(writer) {
writer.onerror = function(err) {
console.error(err);
c(err);
}
writer.onwriteend = function() {
//after truncation, actually write the file
writer.onwriteend = function() {
c();
self.onWrite();
}
var blob = new Blob([data]);
writer.write(blob);
};
writer.truncate(0);
});
}
);
2013-08-20 00:53:03 +00:00
},
stat: function(c) {
var self = this;
if (self.entry) {
return self.entry.file(function(f) {
c(null, f);
});
}
c("No file entry");
2013-08-20 00:53:03 +00:00
},
2013-08-20 00:53:03 +00:00
retain: function() {
var id = chrome.fileSystem.retainEntry(this.entry);
return {
type: "file",
id: id
};
2013-08-20 00:53:03 +00:00
},
restore: function(id, c) {
2013-08-20 00:53:03 +00:00
var self = this;
chrome.fileSystem.isRestorable(id, function(is) {
if (is) {
chrome.fileSystem.restoreEntry(id, function(entry) {
if (!entry) return c("restoreEntry() failed for " + id);
self.entry = entry;
c();
});
} else {
c("isRestorable() returned false for " + id);
}
2013-08-20 00:53:03 +00:00
});
2013-11-06 15:37:34 +00:00
},
getPath: function(c) {
var self = this;
if (!self.entry) return fail("No backing entry, cannot get path");
chrome.fileSystem.getDisplayPath(self.entry, function(path) {
c(null, path);
});
2013-08-20 00:53:03 +00:00
}
};
return File;
});