diff --git a/Changes.md b/Changes.md
index 78b1774c..b97100e6 100644
--- a/Changes.md
+++ b/Changes.md
@@ -1,7 +1,6 @@
-HEAD
+Next
-----------
-- Web UI now has keyboard shortcuts, hit '?' to see them. [#2540]
- Add middleware stack to testing harness; see [wiki documentation](https://github.com/mperham/sidekiq/wiki/Testing#testing-server-middleware) [#2534, ryansch]
3.5.0
diff --git a/lib/sidekiq/web.rb b/lib/sidekiq/web.rb
index 5f04f0c8..bf2b8406 100644
--- a/lib/sidekiq/web.rb
+++ b/lib/sidekiq/web.rb
@@ -30,18 +30,6 @@ module Sidekiq
"Dead" => 'morgue',
}
- SHORTCUTS = {
- 'Dashboard' => 'h',
- 'Busy' => 'b',
- 'Queues' => 'q',
- 'Retries' => 'r',
- 'Scheduled' => 's',
- 'Dead' => 'd',
- 'Limits' => 'l',
- 'Batches' => 'a',
- 'Cron' => 'c',
- }
-
class << self
def default_tabs
DEFAULT_TABS
diff --git a/web/assets/javascripts/jquery.chaves.js b/web/assets/javascripts/jquery.chaves.js
deleted file mode 100644
index 30a8bee9..00000000
--- a/web/assets/javascripts/jquery.chaves.js
+++ /dev/null
@@ -1,133 +0,0 @@
-// Generated by CoffeeScript 1.3.3
-(function() {
-
- $.fn.extend({
- chaves: function(options) {
- var self;
- self = $.fn.chaves;
- options = $.extend({}, self.default_options, options);
- return $(this).each(function(i, el) {
- return self.init(el, options);
- });
- }
- });
-
- $.extend($.fn.chaves, {
- version: '0.1.1',
- default_options: {
- activeClass: 'active',
- bindings: [],
- childSelector: '> *',
- className: 'jquery-chaves',
- enableUpDown: false,
- helpModalClass: 'jquery-chaves-help',
- linkSelector: 'a:first',
- scope: 'all',
- searchSelector: '.search,\
- #search,\
- input[type="search"],\
- input[type="text"][value*="earch"],\
- input[type="text"][placeholder*="earch"]'
- },
- init: function(el, options) {
- var addToHelp, clickActive, downkeys, goDown, goUp, hideHelp, register_all_bindings, searchFocus, showHelp, upkeys,
- _this = this;
- this.options = options;
- this.bindings = $.extend([], options.bindings);
- this.el = $(el).addClass(options.className);
- this.children = this.el.find(options.childSelector);
- this.active = this.children.first().addClass(options.activeClass);
- this.index = 0;
- this.help = this.findOrCreateHelp();
- downkeys = 'j';
- upkeys = 'k';
- if (options.enableUpDown) {
- downkeys += ", down";
- upkeys += ", up";
- }
- register_all_bindings = function() {
- var binding, _i, _len, _ref, _results;
- _ref = _this.bindings;
- _results = [];
- for (_i = 0, _len = _ref.length; _i < _len; _i++) {
- binding = _ref[_i];
- key(binding[0], _this.options.scope, binding[2]);
- _results.push(addToHelp(binding[0], binding[1]));
- }
- return _results;
- };
- addToHelp = function(keys, description) {
- return _this.help.find('dl').append("
" + keys + "" + description + "");
- };
- goUp = function() {
- var prev;
- if (_this.index > 0) {
- _this.index = _this.index - 1;
- prev = $(_this.children[_this.index]).addClass(_this.options.activeClass);
- _this.active.removeClass(_this.options.activeClass);
- _this.active = prev;
- return _this.readjust(150);
- }
- };
- goDown = function() {
- var next;
- if (_this.index < _this.children.length - 1) {
- _this.index = _this.index + 1;
- next = $(_this.children[_this.index]).addClass(_this.options.activeClass);
- _this.active.removeClass(_this.options.activeClass);
- _this.active = next;
- return _this.readjust(0);
- }
- };
- showHelp = function() {
- return _this.help.toggleClass('visible');
- };
- hideHelp = function() {
- return _this.help.removeClass('visible');
- };
- searchFocus = function() {
- _this.search = $(_this.options.searchSelector);
- return window.setTimeout((function() {
- return _this.search.focus();
- }), 10);
- };
- clickActive = function() {
- var link;
- link = _this.active.find(_this.options.linkSelector);
- if (link.trigger('click').attr('target') === '_blank') {
- return window.open(link.attr('href'), 'popped');
- } else {
- return window.location.href = link.attr('href');
- }
- };
- this.bindings.push(['/', 'Focus on filter.', searchFocus]);
- this.bindings.push(['shift+/', 'Toggle help dialog.', showHelp]);
- this.bindings.push(['esc', 'Close help dialog.', hideHelp]);
- return register_all_bindings();
- },
- readjust: function(buffer) {
- var top;
- if (this.elementOutOfViewport(this.active[0])) {
- top = this.active.offset().top - buffer;
- return $(window).scrollTop(top).trigger('scroll');
- }
- },
- elementOutOfViewport: function(el) {
- var rect;
- if (el) {
- rect = el.getBoundingClientRect();
- return !(rect.top >= 0 && rect.left >= 0 && rect.bottom <= window.innerHeight && rect.right <= window.innerWidth);
- }
- },
- findOrCreateHelp: function() {
- var help, helpSelector;
- helpSelector = "." + this.options.helpModalClass;
- if (!$(helpSelector).length) {
- $('body').append("");
- help = $(helpSelector).append('Keyboard Shortcuts
');
- }
- return $(helpSelector);
- }
- });
-
-}).call(this);
diff --git a/web/assets/javascripts/keymaster.js b/web/assets/javascripts/keymaster.js
deleted file mode 100644
index 8f5b5fc5..00000000
--- a/web/assets/javascripts/keymaster.js
+++ /dev/null
@@ -1,296 +0,0 @@
-// keymaster.js
-// (c) 2011-2013 Thomas Fuchs
-// keymaster.js may be freely distributed under the MIT license.
-
-;(function(global){
- var k,
- _handlers = {},
- _mods = { 16: false, 18: false, 17: false, 91: false },
- _scope = 'all',
- // modifier keys
- _MODIFIERS = {
- '⇧': 16, shift: 16,
- '⌥': 18, alt: 18, option: 18,
- '⌃': 17, ctrl: 17, control: 17,
- '⌘': 91, command: 91
- },
- // special keys
- _MAP = {
- backspace: 8, tab: 9, clear: 12,
- enter: 13, 'return': 13,
- esc: 27, escape: 27, space: 32,
- left: 37, up: 38,
- right: 39, down: 40,
- del: 46, 'delete': 46,
- home: 36, end: 35,
- pageup: 33, pagedown: 34,
- ',': 188, '.': 190, '/': 191,
- '`': 192, '-': 189, '=': 187,
- ';': 186, '\'': 222,
- '[': 219, ']': 221, '\\': 220
- },
- code = function(x){
- return _MAP[x] || x.toUpperCase().charCodeAt(0);
- },
- _downKeys = [];
-
- for(k=1;k<20;k++) _MAP['f'+k] = 111+k;
-
- // IE doesn't support Array#indexOf, so have a simple replacement
- function index(array, item){
- var i = array.length;
- while(i--) if(array[i]===item) return i;
- return -1;
- }
-
- // for comparing mods before unassignment
- function compareArray(a1, a2) {
- if (a1.length != a2.length) return false;
- for (var i = 0; i < a1.length; i++) {
- if (a1[i] !== a2[i]) return false;
- }
- return true;
- }
-
- var modifierMap = {
- 16:'shiftKey',
- 18:'altKey',
- 17:'ctrlKey',
- 91:'metaKey'
- };
- function updateModifierKey(event) {
- for(k in _mods) _mods[k] = event[modifierMap[k]];
- };
-
- // handle keydown event
- function dispatch(event) {
- var key, handler, k, i, modifiersMatch, scope;
- key = event.keyCode;
-
- if (index(_downKeys, key) == -1) {
- _downKeys.push(key);
- }
-
- // if a modifier key, set the key. property to true and return
- if(key == 93 || key == 224) key = 91; // right command on webkit, command on Gecko
- if(key in _mods) {
- _mods[key] = true;
- // 'assignKey' from inside this closure is exported to window.key
- for(k in _MODIFIERS) if(_MODIFIERS[k] == key) assignKey[k] = true;
- return;
- }
- updateModifierKey(event);
-
- // see if we need to ignore the keypress (filter() can can be overridden)
- // by default ignore key presses if a select, textarea, or input is focused
- if(!assignKey.filter.call(this, event)) return;
-
- // abort if no potentially matching shortcuts found
- if (!(key in _handlers)) return;
-
- scope = getScope();
-
- // for each potential shortcut
- for (i = 0; i < _handlers[key].length; i++) {
- handler = _handlers[key][i];
-
- // see if it's in the current scope
- if(handler.scope == scope || handler.scope == 'all'){
- // check if modifiers match if any
- modifiersMatch = handler.mods.length > 0;
- for(k in _mods)
- if((!_mods[k] && index(handler.mods, +k) > -1) ||
- (_mods[k] && index(handler.mods, +k) == -1)) modifiersMatch = false;
- // call the handler and stop the event if neccessary
- if((handler.mods.length == 0 && !_mods[16] && !_mods[18] && !_mods[17] && !_mods[91]) || modifiersMatch){
- if(handler.method(event, handler)===false){
- if(event.preventDefault) event.preventDefault();
- else event.returnValue = false;
- if(event.stopPropagation) event.stopPropagation();
- if(event.cancelBubble) event.cancelBubble = true;
- }
- }
- }
- }
- };
-
- // unset modifier keys on keyup
- function clearModifier(event){
- var key = event.keyCode, k,
- i = index(_downKeys, key);
-
- // remove key from _downKeys
- if (i >= 0) {
- _downKeys.splice(i, 1);
- }
-
- if(key == 93 || key == 224) key = 91;
- if(key in _mods) {
- _mods[key] = false;
- for(k in _MODIFIERS) if(_MODIFIERS[k] == key) assignKey[k] = false;
- }
- };
-
- function resetModifiers() {
- for(k in _mods) _mods[k] = false;
- for(k in _MODIFIERS) assignKey[k] = false;
- };
-
- // parse and assign shortcut
- function assignKey(key, scope, method){
- var keys, mods;
- keys = getKeys(key);
- if (method === undefined) {
- method = scope;
- scope = 'all';
- }
-
- // for each shortcut
- for (var i = 0; i < keys.length; i++) {
- // set modifier keys if any
- mods = [];
- key = keys[i].split('+');
- if (key.length > 1){
- mods = getMods(key);
- key = [key[key.length-1]];
- }
- // convert to keycode and...
- key = key[0]
- key = code(key);
- // ...store handler
- if (!(key in _handlers)) _handlers[key] = [];
- _handlers[key].push({ shortcut: keys[i], scope: scope, method: method, key: keys[i], mods: mods });
- }
- };
-
- // unbind all handlers for given key in current scope
- function unbindKey(key, scope) {
- var multipleKeys, keys,
- mods = [],
- i, j, obj;
-
- multipleKeys = getKeys(key);
-
- for (j = 0; j < multipleKeys.length; j++) {
- keys = multipleKeys[j].split('+');
-
- if (keys.length > 1) {
- mods = getMods(keys);
- }
-
- key = keys[keys.length - 1];
- key = code(key);
-
- if (scope === undefined) {
- scope = getScope();
- }
- if (!_handlers[key]) {
- return;
- }
- for (i = 0; i < _handlers[key].length; i++) {
- obj = _handlers[key][i];
- // only clear handlers if correct scope and mods match
- if (obj.scope === scope && compareArray(obj.mods, mods)) {
- _handlers[key][i] = {};
- }
- }
- }
- };
-
- // Returns true if the key with code 'keyCode' is currently down
- // Converts strings into key codes.
- function isPressed(keyCode) {
- if (typeof(keyCode)=='string') {
- keyCode = code(keyCode);
- }
- return index(_downKeys, keyCode) != -1;
- }
-
- function getPressedKeyCodes() {
- return _downKeys.slice(0);
- }
-
- function filter(event){
- var tagName = (event.target || event.srcElement).tagName;
- // ignore keypressed in any elements that support keyboard data input
- return !(tagName == 'INPUT' || tagName == 'SELECT' || tagName == 'TEXTAREA');
- }
-
- // initialize key. to false
- for(k in _MODIFIERS) assignKey[k] = false;
-
- // set current scope (default 'all')
- function setScope(scope){ _scope = scope || 'all' };
- function getScope(){ return _scope || 'all' };
-
- // delete all handlers for a given scope
- function deleteScope(scope){
- var key, handlers, i;
-
- for (key in _handlers) {
- handlers = _handlers[key];
- for (i = 0; i < handlers.length; ) {
- if (handlers[i].scope === scope) handlers.splice(i, 1);
- else i++;
- }
- }
- };
-
- // abstract key logic for assign and unassign
- function getKeys(key) {
- var keys;
- key = key.replace(/\s/g, '');
- keys = key.split(',');
- if ((keys[keys.length - 1]) == '') {
- keys[keys.length - 2] += ',';
- }
- return keys;
- }
-
- // abstract mods logic for assign and unassign
- function getMods(key) {
- var mods = key.slice(0, key.length - 1);
- for (var mi = 0; mi < mods.length; mi++)
- mods[mi] = _MODIFIERS[mods[mi]];
- return mods;
- }
-
- // cross-browser events
- function addEvent(object, event, method) {
- if (object.addEventListener)
- object.addEventListener(event, method, false);
- else if(object.attachEvent)
- object.attachEvent('on'+event, function(){ method(window.event) });
- };
-
- // set the handlers globally on document
- addEvent(document, 'keydown', function(event) { dispatch(event) }); // Passing _scope to a callback to ensure it remains the same by execution. Fixes #48
- addEvent(document, 'keyup', clearModifier);
-
- // reset modifiers to false whenever the window is (re)focused.
- addEvent(window, 'focus', resetModifiers);
-
- // store previously defined key
- var previousKey = global.key;
-
- // restore previously defined key and return reference to our key object
- function noConflict() {
- var k = global.key;
- global.key = previousKey;
- return k;
- }
-
- // set window.key and window.key.set/get/deleteScope, and the default filter
- global.key = assignKey;
- global.key.setScope = setScope;
- global.key.getScope = getScope;
- global.key.deleteScope = deleteScope;
- global.key.filter = filter;
- global.key.isPressed = isPressed;
- global.key.getPressedKeyCodes = getPressedKeyCodes;
- global.key.noConflict = noConflict;
- global.key.unbind = unbindKey;
-
- if(typeof module !== 'undefined') module.exports = assignKey;
-
-})(this);
diff --git a/web/assets/stylesheets/application.css b/web/assets/stylesheets/application.css
index bda0a6f5..a832143c 100755
--- a/web/assets/stylesheets/application.css
+++ b/web/assets/stylesheets/application.css
@@ -744,57 +744,3 @@ div.interval-slider input {
max-width: 350px;
}
}
-
-.jquery-chaves-help {
- width: 250px;
- position: fixed;
- left: 50%;
- top: 20%;
- padding: 15px 0px;
- margin-left: -110px;
- margin-top: -200px;
- background: white;
- display: none;
- border: 3px solid #efefef;
- border-radius: 5px;
- box-shadow: 0 0 18px rgba(0,0,0,0.4);
- z-index: 9000;
- font-family: Helvetica,arial,freesans,clean,sans-serif;
-}
-
-.jquery-chaves-help h3 {
- border-bottom: 1px solid #DDD;
- margin: 0 0 20px;
- padding: 5px 20px 10px;
- font-size: 16px;
- font-weight: bold;
-}
-.jquery-chaves-help dl {
- padding: 0 20px;
- margin: 0;
-}
-.jquery-chaves-help dt {
- display: inline-block;
- font-size: 11px;
- margin: 0 0 15px 0;
- padding: 3px 6px;
- min-width: 10px;
- text-align: center;
- font-family: Monaco,"Courier New","DejaVu Sans Mono","Bitstream Vera Sans Mono",monospace;
- background: #333;
- color: #EEE;
- border-radius: 2px;
- text-shadow: 1px 1px 0 black;
-}
-.jquery-chaves-help dd {
- display: inline;
- margin: 0 0 0 10px;
- font-size: 13px;
-}
-.jquery-chaves-help dd:after {
- content: '\A';
- white-space: pre;
-}
-.jquery-chaves-help.visible {
- display: block;
-}
diff --git a/web/views/_nav.erb b/web/views/_nav.erb
index 88a3ccf2..e7c3da1c 100644
--- a/web/views/_nav.erb
+++ b/web/views/_nav.erb
@@ -23,11 +23,11 @@
<% Sidekiq::Web.default_tabs.each do |title, url| %>
<% if url == '' %>
- <%= t(title) %>
+ <%= t(title) %>
<% else %>
- <%= t(title) %>
+ <%= t(title) %>
<% end %>
<% end %>
@@ -39,7 +39,7 @@
@@ -47,7 +47,7 @@
<% Sidekiq::Web.custom_tabs.each do |title, url| %>
- <%= t(title) %>
+ <%= t(title) %>
<% end %>
diff --git a/web/views/layout.erb b/web/views/layout.erb
index df365745..86447612 100644
--- a/web/views/layout.erb
+++ b/web/views/layout.erb
@@ -7,8 +7,6 @@
-
-
<%= display_custom_head %>
@@ -29,17 +27,5 @@
<%= erb :_footer %>
<%= erb :_poll_js %>
-