18

出于样式原因,我希望能够使用 ul 列表作为选择表单元素。

我可以用我的代码(不包含在这个 jsfiddle 中)填充隐藏的输入,到目前为止一切都很好。但是现在我试图让我的 ul 在按下键盘或鼠标时表现得像选择输入用来。

在我之前的问题中,我在键盘控制方面遇到了一些问题。它们现在已修复。请参阅:键盘上/下箭头上的自动滚动

剩下的问题是按下键盘按钮时鼠标没有被忽略。这导致“悬停效果”首先听键盘输入,但不是立即转到鼠标并选择此 li 项目作为被选中。

这可以在我的 jsfiddle 示例中看到:http: //jsfiddle.net/JVDXT/3/

我的 JavaScript 代码:

// scrollTo plugin 
  $.fn.scrollTo = function( target, options, callback ){
  if(typeof options == 'function' && arguments.length == 2){ callback = options; options = target; }
  var settings = $.extend({
    scrollTarget  : target,
    offsetTop     : 100,
    duration      : 0,
    easing        : 'linear'
  }, options);
  return this.each(function(){
    var scrollPane = $(this);
    var scrollTarget = (typeof settings.scrollTarget == "number") ? settings.scrollTarget : $(settings.scrollTarget);
    var scrollY = (typeof scrollTarget == "number") ? scrollTarget : scrollTarget.offset().top + scrollPane.scrollTop() - parseInt(settings.offsetTop);
    scrollPane.animate({scrollTop : scrollY }, parseInt(settings.duration), settings.easing, function(){
      if (typeof callback == 'function') { callback.call(this); }
    });
  });
}


//My code
//The function that is listing the the mouse
jQuery(".btn-group .dropdown-menu li").mouseover(function() {
        console.log('mousie')
        jQuery(".btn-group .dropdown-menu li").removeClass('selected');
        jQuery(this).addClass('selected');
})  

//What to do when the keyboard is pressed
jQuery(".btn-group").keydown(function(e) {
    if (e.keyCode == 38) { // up
        console.log('keyup pressed');
        var selected = jQuery('.selected');
        jQuery(".btn-group .dropdown-menu li").removeClass('selected');
        if (selected.prev().length == 0) {
            selected.siblings().last().addClass('selected');
        } else {
            selected.prev().addClass('selected');
            jQuery('.btn-group .dropdown-menu').scrollTo('.selected');
        }
    }
    if (e.keyCode == 40) { // down
        console.log('keydown');
        var selected = jQuery('.selected');
        jQuery(".btn-group .dropdown-menu li").removeClass('selected');
        if (selected.next().length == 0) {
            selected.siblings().first().addClass('selected');
        } else {
            selected.next().addClass('selected');
            jQuery('.btn-group .dropdown-menu').scrollTo('.selected');
        }
    }
});

因此,任何人都可以教我如何在按下键盘按钮时忽略鼠标,但在用户再次触摸鼠标时列出鼠标。就像默认的选择输入表单字段一样。

更新

这是一个新的jsfiddle

4

7 回答 7

13

看一下这个:

http://jsfiddle.net/coma/9KvhL/25/

(function($, undefined) {

    $.fn.dropdown = function() {

        var widget = $(this);
        var label = widget.find('span.valueOfButton');
        var list = widget.children('ul');
        var selected;
        var highlighted;

        var select = function(i) {

            selected = $(i);
            label.text(selected.text());

        };

        var highlight = function(i) {

            highlighted = $(i);

            highlighted
            .addClass('selected')
            .siblings('.selected')
            .removeClass('selected');
        };

        var scroll = function(event) {

            list.scrollTo('.selected');

        };

        var hover = function(event) {

            highlight(this);

        };

        var rebind = function(event) {

            bind();

        };

        var bind = function() {

            list.on('mouseover', 'li', hover);
            widget.off('mousemove', rebind);

        };

        var unbind = function() {

            list.off('mouseover', 'li', hover);
            widget.on('mousemove', rebind);

        };

        list.on('click', 'li', function(event) {

            select(this);

        });

        widget.keydown(function(event) {

            unbind();

            switch(event.keyCode) {

                case 38:
                    highlight((highlighted && highlighted.prev().length > 0) ? highlighted.prev() : list.children().last());

                    scroll();
                    break;

                case 40:
                    highlight((highlighted && highlighted.next().length > 0) ? highlighted.next() : list.children().first());

                    scroll();
                    break;

                case 13:
                    if(highlighted) {

                        select(highlighted);

                    }
                    break;

            }

        });

        bind();

    };

    $.fn.scrollTo = function(target, options, callback) {

        if(typeof options === 'function' && arguments.length === 2) {

            callback = options;
            options = target;
        }

        var settings = $.extend({
            scrollTarget  : target,
            offsetTop     : 185,
            duration      : 0,
            easing        : 'linear'
        }, options);

        return this.each(function(i) {

            var scrollPane = $(this);
            var scrollTarget = (typeof settings.scrollTarget === 'number') ? settings.scrollTarget : $(settings.scrollTarget);
            var scrollY = (typeof scrollTarget === 'number') ? scrollTarget : scrollTarget.offset().top + scrollPane.scrollTop() - parseInt(settings.offsetTop, 10);

            scrollPane.animate({scrollTop: scrollY}, parseInt(settings.duration, 10), settings.easing, function() {

                if (typeof callback === 'function') {

                    callback.call(this);
                }

            });

        });

    };

})(jQuery);

$('div.btn-group').dropdown();

关键是在鼠标移动时取消绑定鼠标悬停并重新绑定。

我通过使用闭包函数对其进行了一些重构,将逻辑添加到名为dropdown的 jQuery 方法中,以便您可以重用它,使用 switch 而不是一堆 if 和更多东西。

好吧,有无数插件可以将选择转换为列表:

http://ivaynberg.github.io/select2/

http://harvesthq.github.io/chosen/

http://meetselva.github.io/combobox/

我也有我的!(使用与http://uniformjs.com相同的技巧为触摸设备做好准备)

https://github.com/coma/jquery.select

但是这个问题是关于采用该 HTML 并使其表现得像一个避免悬停问题的选择,对吗?

于 2013-05-12T15:55:47.420 回答
2

这是一个解决方案,我正在使用mousemove它,因为这将确保鼠标再次开始移动时选择正确mouseover的列表项,它只会在输入新列表项时开始选择列表项:

取匿名函数并为其命名:

function mousemove() {
  console.log('mousie')
  jQuery(".btn-group .dropdown-menu li").removeClass('selected');
  jQuery(this).addClass('selected');
}

声明一个全局变量mousemoved,指示鼠标是否已移到文档上并将其设置为false,在mousemove文档上,将其设置为true并将mousemove函数附加到mousemove列表项上的事件。

var mousemoved = false;

jQuery(document).mousemove(function() {
  if(!mousemoved) {
    $('.btn-group .dropdown-menu li').mousemove(mousemove);  
    mousemoved = true;    
  }
})  

一旦按下一个键(在 keydown 事件开始时),使用 jQuery 的.off()方法删除mousemove列表项上的事件(如果存在),并设置mousemovedfalse确保mousemove在鼠标移动之前不会再次附加该事件又动了。

jQuery(".btn-group").keydown(function(e) {

  $('.btn-group .dropdown-menu li').off('mousemove');
  mousemoved = false; 
  ... // Some more of your code

这是一个jsFiddle.

于 2013-05-11T14:45:56.783 回答
1

我试图通过防止自动滚动、在 li 上添加 tabindex、将焦点设置为活动以及使用标志来抑制鼠标来解决您的问题。

修复小提琴: http: //jsfiddle.net/8nKJT/ [修复了 Chrome 中的问题]

http://jsfiddle.net/RDSEt/

问题是因为scroll触发的自动keydown再次触发mouseenter混乱的选择li

注意:与其他方法的区别(此处的答案)我注意到它在每次按键时滚动,而不是仅在到达顶部或底部后滚动(正常行为)。当您并排查看演示时,您会感觉到不同。

下面是更改描述列表和一个小演示来解释它是如何修复的,

  • e.preventDefault() 阻止使用http://jsfiddle.net/TRkAb/按下向上箭头/向下箭头触发的自动滚动[按下],现在在http://jsfiddle.net/TRkAb/1/ul li上尝试相同的操作[不再滚动]
  • 添加了一个标志keydown来抑制 keydown 上的鼠标事件,这个标志被重置onmousemove
  • 添加tabindex到 li 允许您使用.focus函数设置焦点。[更多信息:https ://stackoverflow.com/a/6809236/297641 ]
  • 呼叫.focus将自动滚动到所需位置。(不需要scrollTo插件)http://jsfiddle.net/39h3J/ - [检查它如何滚动到焦点上的 li]

查看演示和代码更改(添加了一些改进)并告诉我。

还要感谢您的问题,我在我编写的一个插件中注意到了这个问题和许多其他问题。

几个月前我写了一个插件来过滤选项,它的作用也完全像一个下拉菜单。

DEMO: http: //jsfiddle.net/nxmBQ/ [filterType改为''关闭过滤]

原始插件页面是http://meetselva.github.io/combobox/

.. 更多的

于 2013-05-13T14:37:09.597 回答
0

方法一个合理的解决方案应该模仿其他具有类似目的的 UI 元素的行为。在所有选中的系统(Windows、Linux、主要浏览器)上,下拉框的行为如下:

将鼠标悬停在一个项目上会突出显示它。按箭头键更改选定的元素,并相应地滚动。移动鼠标选择下面的元素。如果选择为空,则按下down选择第一个元素。按下up选择最后一个元素。

解决方案代码说明了我模仿所描述行为的方法。挺不错的,试试看。。。

其他注意事项将有许多其他选项来抑制不必要的鼠标移动以更改所选元素。这些包括:

  • 保持上次输入法的状态。如果最后一个选择是使用键盘,将鼠标悬停在一个元素上不会选择它,只有单击会
  • mouseover如果坐标没有改变指定的距离,例如 10 像素,则忽略该事件
  • 忽略mouseover用户是否曾经使用过键盘

但是,至少对于公众可以访问的应用程序,最好始终坚持已建立的 UI 模式。

于 2013-05-11T19:49:56.250 回答
0

出现的问题是,当鼠标停留在展开列表的一部分上时,使用键进行的选择无效,因为键盘所做的选择会立即恢复到恰好位于鼠标下方的项目。

您可以解决此问题并保留所有功能,而无需执行任何复杂的条件行为或删除任何事件处理程序。

只需将 mouseover 事件处理程序更改为 mousemove 事件处理程序即可。这样,任何键盘导航和选择都会被监听,并且在用户使用键盘进行选择的任何时候鼠标位置都会被忽略。并且无论何时使用鼠标进行选择,都会听到鼠标的声音。

这听起来微不足道,但它似乎使您的 JS Fiddle 表现完美,并且鼠标和键盘之间没有任何冲突行为。像这样:

//The function that is listening to the mouse
jQuery(".btn-group .dropdown-menu li").mousemove...

(您的代码保持不变,仅用 mousemove 替换 mouseover)

于 2013-05-13T17:18:04.780 回答
0

mouseover如果最近在小部件上按下了 keydown,您可以使用全局忽略该事件。例如:

var last_key_event = 0;

jQuery(".btn-group .dropdown-menu li").mouseover(function() {
    if ((new Date).getTime() > last_key_event + 1000) {
        console.log('mousie')
        jQuery(".btn-group .dropdown-menu li").removeClass('selected');
        jQuery(this).addClass('selected');
    }
});

然后keydown处理程序可以设置处理时间以避免与鼠标交互:

//What to do when the keyboard is pressed
jQuery(".btn-group").keydown(function(e) {
    last_key_event = (new Date).getTime();
    ...
});

last_key_event将每个小部件的变量分开而不是全局变量可能是有意义的。

于 2013-05-11T14:32:40.740 回答
0

你可以试试这个解决方案。mousemove如果坐标没有改变(自上次 mousemove 事件以来),它会忽略该事件

//The function that is listing the the mouse
var lastOffsets = "";

jQuery(".btn-group .dropdown-menu li").mouseover(function(e) {
        var curOffsets = e.clientX+":"+e.clientY;
        if(curOffsets == lastOffsets) {
           // mouse did not really move
            return false;
        }

        lastOffsets = curOffsets;

        ///// rest of your code
}

更新小提琴以验证这是否是您所追求的:http: //jsfiddle.net/pdW75/1/

于 2013-05-11T14:50:03.883 回答