167 lines
No EOL
5.1 KiB
JavaScript
167 lines
No EOL
5.1 KiB
JavaScript
define([
|
|
"settings!menus,keys",
|
|
"editor",
|
|
"ui/dialog",
|
|
"command",
|
|
"util/template!templates/menuItem.html",
|
|
"util/i18n",
|
|
"util/dom2"
|
|
], function(Settings, editor, dialog, command, inflate, i18n) {
|
|
|
|
//default "Windows", will be adjusted during menu creation because async
|
|
var platform = "win";
|
|
|
|
// walker() renders the menu and returns a document fragment
|
|
var walker = function(list, depth) {
|
|
var fragment = document.createDocumentFragment();
|
|
for (var i = 0; i < list.length; i++) {
|
|
var entry = list[i];
|
|
if (typeof entry == "string") {
|
|
var preset;
|
|
switch (entry) {
|
|
case "divider":
|
|
preset = document.createElement("hr");
|
|
break;
|
|
//other special string keys go here - spacer? dynamic menus?
|
|
}
|
|
fragment.append(preset);
|
|
continue;
|
|
}
|
|
//version testing to hide entries that this version of Chrome can't run
|
|
//originally for project support, should remove soon
|
|
if (entry.minVersion && entry.minVersion > chrome.version) {
|
|
continue;
|
|
}
|
|
var isAce = entry.command == "ace:command";
|
|
var keyCombo = isAce ? findKeyCombo(entry.argument) : findKeyCombo(entry.command, entry.argument);
|
|
if (keyCombo && platform == "mac") {
|
|
//special snowflake keyboard adjustment
|
|
keyCombo = keyCombo
|
|
.replace("Command-", "Cmd-")
|
|
.replace("Ctrl-", "Cmd-")
|
|
.replace("Alt-", "Option-");
|
|
}
|
|
var data = {
|
|
command: entry.command,
|
|
argument: entry.argument,
|
|
shortcut: keyCombo,
|
|
hasChildren: entry.sub && !!entry.sub.length,
|
|
isRoot: !depth,
|
|
retainFocus: entry.retainFocus,
|
|
label: i18n.get(entry.label)
|
|
};
|
|
var element = inflate.get("templates/menuItem.html", data);
|
|
if (entry.sub) {
|
|
var children = walker(entry.sub, depth + 1);
|
|
element.find("ul").append(children);
|
|
}
|
|
fragment.append(element);
|
|
}
|
|
return fragment;
|
|
};
|
|
|
|
// We load match commands to the key config, so they're always current
|
|
var findKeyCombo = function(command, arg) {
|
|
var keys = Settings.get("keys");
|
|
var aceCommands = editor.commands.commands;
|
|
//check key config
|
|
for (var key in keys) {
|
|
var action = keys[key];
|
|
if (!action) continue;
|
|
var verb = action.ace || action.command || action;
|
|
var object = action.argument;
|
|
if (verb == command) {
|
|
if (arg && object !== arg) continue;
|
|
//transform old keys and lower-case
|
|
key = key
|
|
//back-compat
|
|
.replace(/(\^|M)-([A-Z]+)$/, "$1-Shift-$2")
|
|
.replace(/\^-/g, "Ctrl-")
|
|
.replace(/M-/g, "Alt-")
|
|
//capitalize keys for lazy people
|
|
.replace(/(^|-)([a-z])/g, function(match) { return match.toUpperCase(); });
|
|
return key;
|
|
}
|
|
}
|
|
for (var cmd in aceCommands) {
|
|
if (cmd == command && aceCommands[cmd].bindKey[platform]) {
|
|
return aceCommands[cmd].bindKey[platform].split("|").shift();
|
|
}
|
|
}
|
|
return false;
|
|
};
|
|
|
|
var Menu = function() {
|
|
this.element = document.find(".toolbar");
|
|
this.active = false;
|
|
this.bindEvents();
|
|
};
|
|
Menu.prototype = {
|
|
create: function() {
|
|
var cfg = Settings.get("menus");
|
|
var self = this;
|
|
chrome.runtime.getPlatformInfo(function(info) {
|
|
if (info.os == "mac") platform = "mac";
|
|
var elements = walker(cfg, 0);
|
|
self.element.innerHTML = "";
|
|
self.element.append(elements);
|
|
});
|
|
},
|
|
bindEvents: function() {
|
|
var self = this;
|
|
var menubar = this.element;
|
|
var clickElsewhere = function(e) {
|
|
if (e.target.matches(".toolbar *")) return;
|
|
self.deactivate();
|
|
self.active = false;
|
|
document.body.off("click", clickElsewhere);
|
|
};
|
|
menubar.addEventListener("click", function(e) {
|
|
document.body.on("click", clickElsewhere);
|
|
var el = e.target;
|
|
if (el.hasClass("top")) {
|
|
el.toggle("active");
|
|
self.active = !self.active;
|
|
} else {
|
|
self.active = false;
|
|
}
|
|
if (!self.active && !el.hasClass("no-refocus")) {
|
|
editor.focus();
|
|
}
|
|
menubar
|
|
.findAll(".active")
|
|
.filter(function(n) { return n != el })
|
|
.forEach(function(n) { n.removeClass("active") });
|
|
});
|
|
menubar.addEventListener("mousemove", function(e) {
|
|
var el = e.target;
|
|
if (el.hasClass("top") && self.active) {
|
|
self.deactivate();
|
|
el.addClass("active");
|
|
}
|
|
});
|
|
},
|
|
deactivate: function() {
|
|
this.element.findAll(".active").forEach(function(node) { node.removeClass("active") });
|
|
}
|
|
};
|
|
|
|
var menu = new Menu();
|
|
|
|
command.on("init:startup", menu.create.bind(menu));
|
|
command.on("init:restart", menu.create.bind(menu));
|
|
|
|
command.on("app:about", function() {
|
|
inflate.load("templates/about.html").then(function() {
|
|
dialog(
|
|
inflate.getHTML("templates/about.html", {
|
|
version: chrome.runtime.getManifest().version
|
|
}),
|
|
["ok"]
|
|
);
|
|
});
|
|
});
|
|
|
|
return menu;
|
|
|
|
}); |