515 lines
15 KiB
JavaScript
515 lines
15 KiB
JavaScript
define([
|
|
"settings!user",
|
|
"command",
|
|
"sessions",
|
|
"storage/file",
|
|
"util/manos",
|
|
"ui/dialog",
|
|
"ui/contextMenus",
|
|
"editor",
|
|
"util/template!templates/projectDir.html,templates/projectFile.html",
|
|
"util/i18n",
|
|
"util/dom2"
|
|
], function(Settings, command, sessions, File, M, dialog, context, editor, inflate, i18n) {
|
|
|
|
/*
|
|
It's tempting to store projects in local storage, similar to the way that we
|
|
retain files for tabs, but this would be a mistake. Reading from storage is a
|
|
pain, because it wants to store a single level deep, and we'll want to alter
|
|
parts of the setup individually.
|
|
|
|
Instead, we'll retain a single file handle to the project file, which (as
|
|
JSON) will store the IDs of individual directories, the project-specific
|
|
settings, and (hopefully, one day) build systems. This also gets us around
|
|
the issues of restored directory order and constantly updating the retained
|
|
file list--we'll just update it when the project file is saved.
|
|
*/
|
|
|
|
var guidCounter = 0;
|
|
|
|
//pseudo-worker to let the UI thread breathe
|
|
var queue = [];
|
|
var working = false;
|
|
var tick = function(fn) {
|
|
if (fn) queue.push(fn);
|
|
if (fn && working) return;
|
|
working = true;
|
|
//start work on the next frame
|
|
var process = function() {
|
|
var then = Date.now();
|
|
while (queue.length) {
|
|
var now = Date.now();
|
|
if (now - then > 10) {
|
|
return setTimeout(process);
|
|
}
|
|
var next = queue.shift();
|
|
next();
|
|
}
|
|
working = false;
|
|
};
|
|
setTimeout(process);
|
|
};
|
|
|
|
//FSNodes are used to track filesystem state inside projects
|
|
//We don't use the typical File object, because we're not really reading them
|
|
//Nodes form a tree starting at the root directory
|
|
var FSNode = function(entry) {
|
|
this.children = [];
|
|
this.id = guidCounter++;
|
|
if (entry) this.setEntry(entry);
|
|
};
|
|
|
|
FSNode.prototype = {
|
|
isDirectory: false,
|
|
entry: null,
|
|
tab: null,
|
|
id: null,
|
|
label: null,
|
|
setEntry: function(entry, c) {
|
|
this.entry = entry;
|
|
this.label = entry.name;
|
|
this.isDirectory = entry.isDirectory;
|
|
},
|
|
|
|
//walk will asynchronously collect the file tree
|
|
walk: function(blacklist, done) {
|
|
var self = this;
|
|
var entries = [];
|
|
var reader = this.entry.createReader();
|
|
var inc = 1;
|
|
|
|
var check = function() {
|
|
inc--;
|
|
if (inc == 0) {
|
|
return done(self);
|
|
}
|
|
};
|
|
|
|
var collect = function(list) {
|
|
if (list.length == 0) return complete();
|
|
entries.push.apply(entries, list);
|
|
reader.readEntries(collect);
|
|
};
|
|
|
|
var complete = function() {
|
|
self.children = [];
|
|
entries.forEach(function(entry) {
|
|
//skip dot dirs, but not files
|
|
if (entry.name[0] == "." && entry.isDirectory) return;
|
|
//skip ignored files
|
|
if (blacklist) {
|
|
if (blacklist.test(entry.name)) return;
|
|
}
|
|
|
|
var node = new FSNode(entry);
|
|
self.children.push(node);
|
|
if (node.isDirectory) {
|
|
inc++;
|
|
//give the UI thread a chance to breathe
|
|
tick(function() { node.walk(blacklist, check); });
|
|
}
|
|
});
|
|
check();
|
|
};
|
|
reader.readEntries(collect);
|
|
}
|
|
};
|
|
|
|
// The Project Manager actually handles rendering and interfacing with the rest
|
|
// of the code. Commands are bound to a singleton instance, but it's technically
|
|
// not picky about being the only one.
|
|
|
|
var ProjectManager = function(element) {
|
|
this.directories = [];
|
|
this.pathMap = {};
|
|
this.expanded = {};
|
|
this.project = null;
|
|
this.projectFile = null;
|
|
if (element) {
|
|
this.setElement(element);
|
|
}
|
|
this.loading = false;
|
|
var self = this;
|
|
chrome.storage.local.get("retainedProject", function(data) {
|
|
if (data.retainedProject) {
|
|
self.loading = true;
|
|
self.render();
|
|
var file = new File();
|
|
var onFail = function() {
|
|
self.loading = false;
|
|
self.render();
|
|
chrome.storage.local.remove("retainedProject");
|
|
}
|
|
file.onWrite = self.watchProjectFile.bind(self);
|
|
file.restore(data.retainedProject.id, function(err, f) {
|
|
if (err) {
|
|
return onFail();
|
|
}
|
|
file.read(function(err, data) {
|
|
if (err) {
|
|
return onFail();
|
|
}
|
|
self.projectFile = file;
|
|
self.loadProject(JSON.parse(data));
|
|
});
|
|
});
|
|
}
|
|
});
|
|
};
|
|
|
|
var blacklistRegExp = function() {
|
|
var blacklist = Settings.get("user").ignoreFiles;
|
|
if (blacklist) {
|
|
return new RegExp(blacklist);
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
ProjectManager.prototype = {
|
|
element: null,
|
|
|
|
addDirectory: function(c) {
|
|
var self = this;
|
|
chrome.fileSystem.chooseEntry({ type: "openDirectory" }, function(d) {
|
|
if (!d) return;
|
|
self.insertDirectory(d);
|
|
});
|
|
},
|
|
|
|
insertDirectory: function(entry) {
|
|
var root;
|
|
this.element.addClass("loading");
|
|
//ensure we aren't duplicating
|
|
this.directories.forEach(function(directoryNode){
|
|
if (directoryNode.entry.fullPath === entry.fullPath) {
|
|
root = directoryNode;
|
|
}
|
|
});
|
|
|
|
//if this is the first, go ahead and start the slideout
|
|
if (!this.directories.length) {
|
|
this.element.addClass("show");
|
|
}
|
|
|
|
if (!root) {
|
|
root = new FSNode(entry);
|
|
this.directories.push(root);
|
|
}
|
|
|
|
//if the directory was there, we still want
|
|
//to refresh it, in response to the users
|
|
//interaction
|
|
var self = this;
|
|
tick(function() {
|
|
root.walk(blacklistRegExp(), function() {
|
|
self.render()
|
|
});
|
|
});
|
|
},
|
|
|
|
removeDirectory: function(args) {
|
|
this.directories = this.directories.filter(function(node) {
|
|
return node.id != args.id;
|
|
});
|
|
this.render();
|
|
},
|
|
|
|
removeAllDirectories: function() {
|
|
this.directories = [];
|
|
this.render();
|
|
},
|
|
|
|
refresh: function() {
|
|
var counter = 0;
|
|
var self = this;
|
|
this.element.addClass("loading");
|
|
var check = function() {
|
|
counter++;
|
|
if (counter == self.directories.length) {
|
|
//render() should get rid of the class, but let's be sure
|
|
self.element.removeClass("loading");
|
|
self.render();
|
|
}
|
|
};
|
|
blacklist = blacklistRegExp();
|
|
this.directories.forEach(function(d) {
|
|
d.walk(blacklist, check);
|
|
});
|
|
},
|
|
|
|
render: function() {
|
|
if (!this.element) return;
|
|
|
|
//Ace doesn't know about non-window resize events
|
|
//moving the panel will screw up its dimensions
|
|
setTimeout(function() {
|
|
editor.resize();
|
|
}, 500);
|
|
|
|
var tree = this.element.find(".tree");
|
|
this.pathMap = {};
|
|
if (this.directories.length == 0 && !this.loading) {
|
|
this.element.removeClass("show");
|
|
tree.innerHTML = "";
|
|
return;
|
|
}
|
|
var self = this;
|
|
this.element.addClass("show");
|
|
if (this.loading) {
|
|
this.element.addClass("loading");
|
|
}
|
|
|
|
var walker = function(node) {
|
|
var li = document.createElement("li");
|
|
if (node.isDirectory) {
|
|
var isRoot = self.directories.indexOf(node) != -1;
|
|
var nodeData = {
|
|
label: node.label,
|
|
path: node.entry.fullPath,
|
|
contextMenu: context.makeURL(isRoot ? "root/directory" : "directory", node.id)
|
|
};
|
|
var a = inflate.get("templates/projectDir.html", nodeData);
|
|
li.append(a);
|
|
if (self.expanded[node.entry.fullPath]) {
|
|
li.addClass("expanded");
|
|
}
|
|
var ul = document.createElement("ul");
|
|
node.children.sort(function(a, b) {
|
|
if (a.isDirectory != b.isDirectory) {
|
|
//sneaky casting trick
|
|
return ~~b.isDirectory - ~~a.isDirectory;
|
|
}
|
|
if (a.label < b.label) return -1;
|
|
if (a.label > b.label) return 1;
|
|
return 0;
|
|
});
|
|
for (var i = 0; i < node.children.length; i++) {
|
|
ul.append(walker(node.children[i]));
|
|
}
|
|
li.append(ul);
|
|
} else {
|
|
var nodeData = {
|
|
path: node.entry.fullPath,
|
|
contextMenu: context.makeURL("file", node.entry.fullPath.replace(/[\/\\]/g, "@")),
|
|
label: node.label
|
|
};
|
|
var a = inflate.get("templates/projectFile.html", nodeData)
|
|
li.append(a);
|
|
self.pathMap[node.entry.fullPath] = node;
|
|
}
|
|
return li;
|
|
};
|
|
|
|
//we give the load bar a chance to display before rendering
|
|
tick(function() {
|
|
var trees = self.directories.map(walker);
|
|
var list = document.createElement("ul");
|
|
trees.forEach(function(dir) {
|
|
dir.classList.add("root");
|
|
dir.classList.add("expanded");
|
|
list.appendChild(dir);
|
|
});
|
|
|
|
tree.innerHTML = "";
|
|
tree.appendChild(list);
|
|
if (!self.loading) {
|
|
self.element.removeClass("loading");
|
|
}
|
|
});
|
|
},
|
|
|
|
setElement: function(el) {
|
|
this.element = el;
|
|
this.bindEvents();
|
|
},
|
|
|
|
bindEvents: function() {
|
|
var self = this;
|
|
this.element.on("click", function(e) {
|
|
e.preventDefault();
|
|
var target = e.target;
|
|
if (target.hasClass("directory")) {
|
|
target.parentElement.toggle("expanded");
|
|
var path = target.getAttribute("data-full-path");
|
|
self.expanded[path] = !!!self.expanded[path];
|
|
}
|
|
editor.focus();
|
|
});
|
|
},
|
|
|
|
openFile: function(path) {
|
|
var self = this;
|
|
var found = false;
|
|
var node = this.pathMap[path];
|
|
if (!node) return;
|
|
//walk through existing tabs to see if it's already open
|
|
var tabs = sessions.getAllTabs();
|
|
chrome.fileSystem.getDisplayPath(node.entry, function(path) {
|
|
//look through the tabs for matching display paths
|
|
M.map(
|
|
tabs,
|
|
function(tab, i, c) {
|
|
if (!tab.file || tab.file.virtual) {
|
|
return c(false);
|
|
}
|
|
tab.file.getPath(function(err, p) {
|
|
if (p == path) {
|
|
sessions.setCurrent(tab);
|
|
found = true;
|
|
}
|
|
//we don't actually use the result
|
|
c();
|
|
});
|
|
},
|
|
//if no match found, create a tab
|
|
function() {
|
|
if (found) return;
|
|
var file = new File(node.entry);
|
|
file.read(function(err, data) {
|
|
sessions.addFile(data, file);
|
|
});
|
|
}
|
|
);
|
|
});
|
|
},
|
|
|
|
generateProject: function() {
|
|
var project = this.project || {};
|
|
//everything but "folders" is left as-is
|
|
//run through all directories, retain them, and add to the structure
|
|
project.folders = this.directories.map(function(node) {
|
|
var id = chrome.fileSystem.retainEntry(node.entry);
|
|
return {
|
|
retained: id,
|
|
path: node.entry.fullPath
|
|
};
|
|
});
|
|
var json = JSON.stringify(project, null, 2);
|
|
if (this.projectFile) {
|
|
this.projectFile.write(json);
|
|
} else {
|
|
var file = new File();
|
|
var watch = this.watchProjectFile.bind(this);
|
|
var self = this;
|
|
file.open("save", function() {
|
|
file.write(json);
|
|
var id = file.retain();
|
|
chrome.storage.local.set({retainedProject: id});
|
|
file.onWrite = watch;
|
|
self.projectFile = file;
|
|
});
|
|
}
|
|
return json;
|
|
},
|
|
|
|
openProjectFile: function() {
|
|
var file = new File();
|
|
var self = this;
|
|
file.open(function() {
|
|
file.read(function(err, data) {
|
|
self.loadProject(data);
|
|
var retained = file.retain();
|
|
chrome.storage.local.set({retainedProject: retained});
|
|
self.projectFile = file;
|
|
file.onWrite = self.watchProjectFile.bind(self);
|
|
});
|
|
});
|
|
},
|
|
|
|
watchProjectFile: function() {
|
|
var self = this;
|
|
this.projectFile.read(function(err, data) {
|
|
self.loadProject(data);
|
|
});
|
|
},
|
|
|
|
loadProject: function(project) {
|
|
var self = this;
|
|
//project is the JSON from a project file
|
|
if (typeof project == "string") {
|
|
project = JSON.parse(project);
|
|
}
|
|
this.project = project;
|
|
//assign settings
|
|
if (project.settings) {
|
|
Settings.setProject(project.settings);
|
|
}
|
|
this.loading = true;
|
|
this.element.addClass("loading");
|
|
//restore directory entries that can be restored
|
|
this.directories = [];
|
|
blacklist = blacklistRegExp();
|
|
M.map(
|
|
project.folders,
|
|
function(folder, index, c) {
|
|
chrome.fileSystem.restoreEntry(folder.retained, function(entry) {
|
|
//remember, you can only restore project directories you'd previously opened
|
|
if (!entry) return c();
|
|
var node = new FSNode(entry);
|
|
//if this is the first, go ahead and start the slideout
|
|
if (!self.directories.length) {
|
|
self.element.addClass("show");
|
|
}
|
|
self.directories.push(node);
|
|
node.walk(blacklist, c);
|
|
});
|
|
},
|
|
function() {
|
|
self.loading = false;
|
|
self.render();
|
|
}
|
|
);
|
|
},
|
|
|
|
editProjectFile: function() {
|
|
if (!this.projectFile) {
|
|
return dialog(i18n.get("projectNoCurrentProject"));
|
|
}
|
|
var self = this;
|
|
this.projectFile.read(function(err, data) {
|
|
sessions.addFile(data, self.projectFile);
|
|
});
|
|
},
|
|
|
|
clearProject: function(keepRetained) {
|
|
this.projectFile = null;
|
|
this.directories = [];
|
|
this.project = {};
|
|
Settings.clearProject();
|
|
if (!keepRetained) chrome.storage.local.remove("retainedProject");
|
|
this.render();
|
|
},
|
|
|
|
getPaths: function() {
|
|
return Object.keys(this.pathMap);
|
|
}
|
|
};
|
|
|
|
var pm = new ProjectManager(document.find(".project"));
|
|
command.on("project:add-dir", pm.addDirectory.bind(pm));
|
|
command.on("project:remove-all", pm.removeAllDirectories.bind(pm));
|
|
command.on("project:generate", pm.generateProject.bind(pm));
|
|
command.on("project:open-file", pm.openFile.bind(pm));
|
|
command.on("project:refresh-dir", pm.refresh.bind(pm));
|
|
command.on("project:open", pm.openProjectFile.bind(pm));
|
|
command.on("project:edit", pm.editProjectFile.bind(pm));
|
|
command.on("project:clear", pm.clearProject.bind(pm));
|
|
|
|
context.register("Remove from Project", "removeDirectory", "root/directory/:id", pm.removeDirectory.bind(pm));
|
|
|
|
var setAutoHide = function() {
|
|
var hide = Settings.get("user").autoHideProject;
|
|
if (hide) {
|
|
pm.element.classList.add("autohide");
|
|
} else {
|
|
pm.element.classList.remove("autohide");
|
|
}
|
|
}
|
|
|
|
command.on("init:startup", setAutoHide);
|
|
command.on("init:restart", setAutoHide);
|
|
|
|
return pm;
|
|
|
|
});
|