2013-09-16 18:16:06 +00:00
|
|
|
define([
|
2013-09-24 19:16:19 +00:00
|
|
|
"sessions",
|
2014-02-20 17:00:29 +00:00
|
|
|
"editor",
|
2013-12-04 17:24:04 +00:00
|
|
|
"storage/file",
|
|
|
|
"ui/dialog",
|
2013-09-16 18:16:06 +00:00
|
|
|
"command",
|
2014-01-21 06:10:06 +00:00
|
|
|
"storage/settingsProvider",
|
2014-01-12 09:12:14 +00:00
|
|
|
"util/manos",
|
2014-06-29 23:40:40 +00:00
|
|
|
"storage/nullfile",
|
2014-07-07 16:10:37 +00:00
|
|
|
"util/i18n",
|
2014-06-29 23:40:40 +00:00
|
|
|
//these next modules are self-contained
|
|
|
|
"sessions/dragdrop",
|
|
|
|
"sessions/autosave"
|
2014-07-07 16:10:37 +00:00
|
|
|
], function(sessions, editor, File, dialog, command, Settings, M, NullFile, i18n) {
|
2014-02-27 06:34:02 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
FileManager splits out the session code that specifically deals with I/O.
|
|
|
|
Pretty much the whole file is just bindings to various commands.
|
|
|
|
|
|
|
|
Now that session.js is refactored, this could probably move into a submodule,
|
|
|
|
except that it gets loaded explicitly on startup.
|
|
|
|
*/
|
2014-01-12 09:34:47 +00:00
|
|
|
|
2013-12-17 01:35:02 +00:00
|
|
|
command.on("session:new-file", function(content) { return sessions.addFile(content) });
|
2014-02-27 06:34:02 +00:00
|
|
|
|
|
|
|
command.on("session:open-file", function(c) {
|
|
|
|
//have to call chooseEntry manually to support multiple files
|
|
|
|
var args = {
|
|
|
|
type: "openWritableFile"
|
|
|
|
};
|
|
|
|
if (chrome.version >= 30) {
|
|
|
|
args.acceptsMultiple = true;
|
|
|
|
}
|
|
|
|
chrome.fileSystem.chooseEntry(args, function(files) {
|
|
|
|
//annoying array function test, since it's not apparently a real array
|
|
|
|
if (!files.slice) {
|
|
|
|
files = [ files ];
|
|
|
|
}
|
|
|
|
files.map(function(entry) {
|
|
|
|
var f = new File(entry);
|
2014-05-04 09:09:19 +00:00
|
|
|
return f.read(function(err, data) {
|
2014-02-27 06:34:02 +00:00
|
|
|
sessions.addFile(data, f);
|
2014-06-29 23:40:40 +00:00
|
|
|
});
|
2014-02-27 06:34:02 +00:00
|
|
|
});
|
|
|
|
Promise.all(files).then(c);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2014-05-13 03:15:39 +00:00
|
|
|
command.on("session:save-file", function(c) {
|
2014-02-22 06:27:36 +00:00
|
|
|
sessions.getCurrent()
|
|
|
|
.save(c)
|
|
|
|
.then(function() {
|
|
|
|
command.fire("session:syntax");
|
2014-02-27 06:34:02 +00:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2014-05-13 03:15:39 +00:00
|
|
|
command.on("session:save-file-as", function(c) {
|
2013-11-06 16:46:20 +00:00
|
|
|
var tab = sessions.getCurrent();
|
2013-12-23 19:20:04 +00:00
|
|
|
tab.save(true).then(function() {
|
2014-01-21 06:10:06 +00:00
|
|
|
var mode = tab.detectSyntax();
|
2014-01-21 02:03:18 +00:00
|
|
|
sessions.renderTabs();
|
|
|
|
command.fire("session:syntax", mode);
|
2013-12-17 01:35:02 +00:00
|
|
|
if (c) c();
|
|
|
|
});
|
2013-11-06 16:46:20 +00:00
|
|
|
});
|
2013-09-24 02:22:03 +00:00
|
|
|
|
2013-12-17 01:35:02 +00:00
|
|
|
command.on("session:revert-file", function(c) {
|
2013-09-16 18:58:37 +00:00
|
|
|
var tab = sessions.getCurrent();
|
|
|
|
if (!tab.file) return;
|
2014-05-04 09:09:19 +00:00
|
|
|
tab.file.read(function(err, data) {
|
2013-09-16 18:58:37 +00:00
|
|
|
tab.setValue(data);
|
|
|
|
tab.modified = false;
|
2014-03-06 14:56:01 +00:00
|
|
|
tab.modifiedAt = new Date();
|
2013-09-16 18:58:37 +00:00
|
|
|
sessions.renderTabs();
|
2014-05-04 09:09:19 +00:00
|
|
|
if (c) c();
|
2013-09-16 18:58:37 +00:00
|
|
|
});
|
|
|
|
});
|
2014-02-27 06:34:02 +00:00
|
|
|
|
2014-07-07 02:30:39 +00:00
|
|
|
//we now autoretain starting after load, every n seconds
|
|
|
|
var retainInterval = 5
|
|
|
|
var retainLoop = function(c) {
|
2014-02-22 04:46:28 +00:00
|
|
|
var tabs = sessions.getAllTabs();
|
|
|
|
var keep = [];
|
|
|
|
tabs.forEach(function(tab, i) {
|
|
|
|
if (!tab.file || tab.file.virtual) return;
|
|
|
|
keep[i] = tab.file.retain();
|
|
|
|
});
|
|
|
|
keep = keep.filter(function(m) { return m });
|
2014-07-07 02:30:39 +00:00
|
|
|
chrome.storage.local.set({ retained: keep }, function() {
|
|
|
|
setTimeout(retainLoop, retainInterval * 1000);
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2013-09-21 00:39:04 +00:00
|
|
|
command.on("session:check-file", function() {
|
|
|
|
var tab = sessions.getCurrent();
|
2013-09-21 19:21:25 +00:00
|
|
|
if (!tab.file || tab.file.virtual) return;
|
2013-09-21 00:39:04 +00:00
|
|
|
tab.file.entry.file(function(entry) {
|
|
|
|
if (tab.modifiedAt && entry.lastModifiedDate > tab.modifiedAt) {
|
|
|
|
if (tab.modified) {
|
|
|
|
dialog(
|
2014-07-07 16:10:37 +00:00
|
|
|
i18n.get("dialogFileModified"),
|
|
|
|
[
|
|
|
|
{label: i18n.get("dialogReload"), value: true},
|
|
|
|
{label: i18n.get("dialogCancel"), value: false, focus: true}],
|
2013-09-21 00:39:04 +00:00
|
|
|
function(confirmed) {
|
|
|
|
if (confirmed) {
|
|
|
|
command.fire("session:revert-file");
|
|
|
|
} else {
|
|
|
|
tab.modifiedAt = new Date();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
command.fire("session:revert-file");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
2013-09-16 18:16:06 +00:00
|
|
|
|
2013-12-24 00:15:03 +00:00
|
|
|
command.on("session:open-settings-file", function(name, c) {
|
2013-09-16 18:16:06 +00:00
|
|
|
Settings.load(name, function() {
|
|
|
|
var data = Settings.getAsString(name);
|
|
|
|
var file = Settings.getAsFile(name);
|
|
|
|
sessions.addFile(data, file);
|
2013-12-24 00:15:03 +00:00
|
|
|
if (c) c();
|
2013-09-16 18:16:06 +00:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
//defaults don't get loaded as files, just as content
|
2013-12-24 00:15:03 +00:00
|
|
|
command.on("session:open-settings-defaults", function(name, c) {
|
2014-06-29 23:23:03 +00:00
|
|
|
Settings.load(name, function() {
|
|
|
|
var text = Settings.getAsString(name, true);
|
|
|
|
var tab = sessions.addFile(text);
|
|
|
|
tab.syntaxMode = "javascript";
|
|
|
|
tab.detectSyntax();
|
|
|
|
tab.fileName = name + ".json";
|
|
|
|
tab.file = new NullFile(text);
|
|
|
|
if (c) c();
|
|
|
|
});
|
2013-09-16 18:16:06 +00:00
|
|
|
});
|
|
|
|
|
2014-02-20 17:00:29 +00:00
|
|
|
command.on("session:insert-from-file", function(c) {
|
|
|
|
var f = new File();
|
2014-05-04 09:09:19 +00:00
|
|
|
f.open(function() {
|
|
|
|
f.read(function(err, text) {
|
|
|
|
editor.execCommand("insertstring", text);
|
|
|
|
});
|
2014-02-20 17:00:29 +00:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2014-06-22 07:36:25 +00:00
|
|
|
var openFromLaunchData = function() {
|
|
|
|
if (window.launchData) {
|
|
|
|
window.launchData.forEach(function(file) {
|
|
|
|
var f = new File(file.entry);
|
|
|
|
f.read(function(err, contents) {
|
|
|
|
sessions.addFile(contents, f);
|
2014-06-29 23:40:40 +00:00
|
|
|
});
|
2014-06-22 07:36:25 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
var openFromRetained = function(done) {
|
|
|
|
chrome.storage.local.get("retained", function(data) {
|
|
|
|
var failures = [];
|
|
|
|
if (!data.retained || !data.retained.length) return done();
|
|
|
|
|
|
|
|
//convert raw retained IDs into typed retention objects
|
|
|
|
var retained = data.retained.map(function(item) {
|
|
|
|
if (typeof item == "string") {
|
|
|
|
return {
|
|
|
|
type: "file",
|
|
|
|
id: item
|
|
|
|
};
|
|
|
|
}
|
2014-06-30 00:00:27 +00:00
|
|
|
return item;
|
2014-06-22 07:36:25 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
//constructors for restorable types
|
|
|
|
var restoreTypes = {
|
|
|
|
file: File,
|
|
|
|
settings: null, //not yet, will be regular SyncFile
|
|
|
|
buffer: null //not yet, will be HTML5 filesystem for scratch files
|
|
|
|
}
|
|
|
|
|
|
|
|
//try to restore items in order
|
|
|
|
M.map(
|
|
|
|
retained,
|
|
|
|
function(item, i, c) {
|
|
|
|
var Type = restoreTypes[item.type] || File;
|
|
|
|
var file = new Type();
|
2014-07-05 21:04:33 +00:00
|
|
|
file.restore(item.id, function(err) {
|
|
|
|
if (err) {
|
|
|
|
console.log("Fail restore", file);
|
|
|
|
failures.push(item)
|
|
|
|
return c(null);
|
|
|
|
}
|
2014-06-22 07:36:25 +00:00
|
|
|
file.read(function(err, data) {
|
|
|
|
if (err) {
|
2014-07-05 21:04:33 +00:00
|
|
|
console.log("Failed reading", file)
|
2014-06-22 07:36:25 +00:00
|
|
|
failures.push(item);
|
|
|
|
return c(null);
|
|
|
|
}
|
|
|
|
c({
|
|
|
|
value: data,
|
|
|
|
file: file
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
},
|
|
|
|
function(restored) {
|
|
|
|
restored = restored.filter(function(d) { return d });
|
|
|
|
for (var i = 0; i < restored.length; i++) {
|
|
|
|
var tab = restored[i];
|
|
|
|
sessions.addFile(tab.value, tab.file);
|
|
|
|
}
|
2014-07-05 21:04:33 +00:00
|
|
|
if (!failures.length) {
|
|
|
|
if (done) done();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
console.log("Removing failed restore IDs", failures);
|
2014-06-22 07:36:25 +00:00
|
|
|
chrome.storage.local.get("retained", function(data) {
|
|
|
|
if (!data.retained) return;
|
|
|
|
chrome.storage.local.set({
|
|
|
|
retained: data.retained.filter(function(d) {
|
|
|
|
if (typeof d == "string") {
|
|
|
|
d = {
|
|
|
|
type: "file",
|
|
|
|
id: d
|
|
|
|
};
|
|
|
|
}
|
|
|
|
return !failures.some(function(fail) {
|
|
|
|
return fail.type == d.type && fail.id == d.id;
|
|
|
|
});
|
|
|
|
})
|
|
|
|
});
|
|
|
|
});
|
|
|
|
if (done) done();
|
|
|
|
}
|
|
|
|
);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2013-09-16 18:16:06 +00:00
|
|
|
command.on("session:open-launch", openFromLaunchData);
|
|
|
|
|
2014-01-12 23:35:30 +00:00
|
|
|
var init = function(complete) {
|
2014-01-21 06:10:06 +00:00
|
|
|
Settings.pull("user").then(function(data) {
|
2014-10-21 14:24:59 +00:00
|
|
|
if (data.user.disableTabRestore) {
|
2014-06-22 07:36:25 +00:00
|
|
|
openFromLaunchData();
|
|
|
|
complete("fileManager");
|
2014-10-21 14:24:59 +00:00
|
|
|
} else {
|
|
|
|
openFromRetained(function() {
|
|
|
|
openFromLaunchData();
|
|
|
|
//start the retention process
|
|
|
|
retainLoop();
|
|
|
|
complete("fileManager");
|
|
|
|
});
|
|
|
|
}
|
2013-09-16 18:16:06 +00:00
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
command.on("init:startup", init);
|
2014-03-05 15:34:25 +00:00
|
|
|
|
|
|
|
window.on("focus", command.fire.bind(null, "session:check-file"));
|
2013-09-16 18:16:06 +00:00
|
|
|
|
2014-03-06 14:56:01 +00:00
|
|
|
});
|