1
0
mirror of synced 2024-11-22 13:06:08 +03:00

reworked how dropdown is managed in singleselect so the search field has focus even when the dropdown is not opened and can capture te first keypress. fixes #196

This commit is contained in:
Igor Vaynberg 2012-07-15 22:22:38 +03:00
parent 06ec20dec6
commit 1605a631e5

View File

@ -806,26 +806,34 @@
if (!this.shouldOpen()) return false;
window.setTimeout(this.bind(this.opening), 1);
return true;
},
/**
* Performs the opening of the dropdown
*/
// abstract
opening: function() {
this.clearDropdownAlignmentPreference();
if (this.search.val() === " ") { this.search.val(""); }
this.container.addClass("select2-dropdown-open").addClass("select2-container-active");
if(this.dropdown[0] !== this.body.children().last()[0]) {
// ensure the dropdown is the last child of body, so the z-index is always respected correctly
this.dropdown.detach().appendTo(this.body);
}
this.dropdown.addClass("select2-drop-active");
this.positionDropdown();
this.updateResults(true);
if(this.dropdown[0] !== this.body.children().last()[0]) {
this.dropdown.detach().appendTo(this.body);
}
this.dropdown.show();
this.ensureHighlightVisible();
this.focusSearch();
return true;
this.container.addClass("select2-dropdown-open").addClass("select2-container-active");
},
// abstract
@ -1132,11 +1140,11 @@
"class": "select2-container",
"style": "width: " + this.getContainerWidth()
}).html([
" <a href='javascript:void(0)' class='select2-choice'><input type='text' class='select2-offscreen select2-focusser'/>",
" <a href='javascript:void(0)' class='select2-choice'>",
" <span></span><abbr class='select2-search-choice-close' style='display:none;'></abbr>",
" <div><b></b></div>" ,
"</a>",
" <div class='select2-drop' style='display:none;'>" ,
" <div class='select2-drop select2-offscreen'>" ,
" <div class='select2-search'>" ,
" <input type='text' autocomplete='off' class='select2-input'/>" ,
" </div>" ,
@ -1145,16 +1153,23 @@
"</div>"].join(""));
},
// single
opening: function () {
this.parent.opening.apply(this, arguments);
this.dropdown.removeClass("select2-offscreen");
},
// single
close: function () {
if (!this.opened()) return;
this.parent.close.apply(this, arguments);
this.dropdown.removeAttr("style").addClass("select2-offscreen").insertAfter(this.selection).show();
},
// single
focus: function () {
this.close();
this.selection.focus();
this.search.focus();
},
// single
@ -1165,7 +1180,7 @@
// single
cancel: function () {
this.parent.cancel.apply(this, arguments);
this.selection.focus();
this.search.focus();
},
// single
@ -1174,65 +1189,66 @@
var selection,
container = this.container,
dropdown = this.dropdown,
containers = $([this.container.get(0), this.dropdown.get(0)]),
clickingInside = false,
selector = ".select2-choice",
focusser=container.find("input.select2-focusser");
clickingInside = false;
this.selection = selection = container.find(selector);
this.selection = selection = container.find(".select2-choice");
this.search.bind("keydown", this.bind(function (e) {
switch (e.which) {
case KEY.UP:
case KEY.DOWN:
this.moveHighlight((e.which === KEY.UP) ? -1 : 1);
killEvent(e);
return;
case KEY.TAB:
case KEY.ENTER:
this.selectHighlighted();
killEvent(e);
return;
case KEY.ESC:
this.cancel(e);
if (!this.enabled) return;
if (e.which === KEY.PAGE_UP || e.which === KEY.PAGE_DOWN || e.which === KEY.SPACE) {
// prevent the page from scrolling
killEvent(e);
return;
}
if (this.opened()) {
switch (e.which) {
case KEY.UP:
case KEY.DOWN:
this.moveHighlight((e.which === KEY.UP) ? -1 : 1);
killEvent(e);
return;
case KEY.TAB:
case KEY.ENTER:
this.selectHighlighted();
killEvent(e);
return;
case KEY.ESC:
this.cancel(e);
killEvent(e);
return;
}
} else {
if (e.which === KEY.TAB || KEY.isControl(e) || KEY.isFunctionKey(e) || e.which === KEY.ESC) {
return;
}
this.open();
if (e.which === KEY.ENTER) {
// do not propagate the event otherwise we open, and propagate enter which closes
killEvent(e);
return;
}
}
}));
containers.delegate(selector, "click", this.bind(function (e) {
selection.bind("click", this.bind(function (e) {
clickingInside = true;
if (this.opened()) {
this.close();
selection.focus();
this.search.focus();
} else if (this.enabled) {
this.open();
}
e.preventDefault();
killEvent(e);
clickingInside = false;
}));
containers.delegate(selector, "keydown", this.bind(function (e) {
if (!this.enabled || e.which === KEY.TAB || KEY.isControl(e) || KEY.isFunctionKey(e) || e.which === KEY.ESC) {
return;
}
this.open();
if (e.which === KEY.PAGE_UP || e.which === KEY.PAGE_DOWN || e.which === KEY.SPACE) {
// prevent the page from scrolling
killEvent(e);
}
if (e.which === KEY.ENTER) {
// do not propagate the event otherwise we open, and propagate enter which closes
killEvent(e);
}
}));
containers.delegate(selector, "focus", function () { if (this.enabled) { containers.addClass("select2-container-active"); dropdown.addClass("select2-drop-active"); }});
containers.delegate(selector, "blur", this.bind(function (e) {
if (clickingInside) return;
if (e.target===focusser.get(0)) return; // ignore blurs from focusser
if (!this.opened()) { this.blur(); }
}));
dropdown.bind("click", this.bind(function() { this.search.focus(); }));
selection.delegate("abbr", "click", this.bind(function (e) {
if (!this.enabled) return;
@ -1240,18 +1256,16 @@
killEvent(e);
this.close();
this.triggerChange();
selection.focus();
this.search.focus();
}));
selection.bind("focus", this.bind(function() { this.search.focus(); }));
this.setPlaceholder();
focusser.bind("focus", function() { selection.focus(); });
selection.bind("focus", this.bind(function() {
focusser.hide();
this.search.bind("focus", this.bind(function() {
this.container.addClass("select2-container-active");
}));
selection.bind("blur", function() { focusser.show(); });
this.opts.element.bind("open", function() { focusser.hide(); });
},
clear: function() {
@ -1358,7 +1372,7 @@
this.opts.element.val(this.id(data));
this.updateSelection(data);
this.close();
this.selection.focus();
this.search.focus();
if (!equal(old, this.id(data))) { this.triggerChange(); }
},
@ -1623,13 +1637,12 @@
},
// multi
open: function () {
if (this.parent.open.apply(this, arguments) === false) return false;
opening: function () {
this.parent.opening.apply(this, arguments);
this.clearPlaceholder();
this.resizeSearch();
this.focusSearch();
return true;
},
// multi