专注于搜索文本字段并使搜索列表可导航

Focus on search text field and making search list navigable

提问人:RaviTeja 提问时间:10/26/2012 最后编辑:Brian Tompsett - 汤莱恩RaviTeja 更新时间:9/12/2020 访问量:2854

问:

我在其中一个代码中有一个搜索功能,理想情况下,它可以对值列表(小数字)执行搜索。

<li>abc</li> 
<li>def</li> 
...

我在底部有一个搜索按钮,我放了一个事件侦听器来获得即时搜索。现在的问题是如何使它可导航。 理想的场景是搜索一些文本,按回车键,它应该会打开第一个元素,否则您应该能够浏览。keyup

如果我在上面不清楚,我实际上是在要求一个与Facebook Chat侧边栏搜索非常相似的功能。

javascript 用户界面 google-chrome-extension dom-events unobtrusive-javascript

评论

1赞 Rob W 11/10/2012
请出示必要的信息: 1.此功能位于何处(例如:选项页面)?2. 你的HTML的真实结构是什么。
0赞 11/16/2012
事实上。你也可以发布你已经写过的东西吗?目前尚不清楚您的基本搜索过滤功能是否有效。此外,“可导航”是什么意思?你的意思是你应该能够点击/键盘浏览结果,还是其他什么?
0赞 RaviTeja 12/6/2012
@RobW 1:它在弹出窗口中.html 2:它在 div 中非常普通的 li 元素。
0赞 RaviTeja 12/6/2012
@lunchmeat317 对不起,如果我不清楚,基本上我需要的是先导航列表。然后我需要专注于搜索框,而列表仍然可以导航

答:

0赞 th3byrdm4n 11/14/2012 #1

如果这是我的页面,我会用一个类标记可搜索元素,例如,我将使用 jQuery 通过选择器查找页面中的所有可搜索元素,例如我们之前定义 var SEARCHTERM = $(“#searchbox”).text();<li class="searchable">abc</li>$("li.searchable:contains('" + SEARCHTERM + "')")

您可能希望对此进行一些微调(例如:contains区分大小写),但这听起来像是朝着正确的方向发展吗?

0赞 ranjez 11/14/2012 #2

您是否正在寻找自动完成功能...?如果是这样,您应该查看 jQuery UI 自动完成: http://jqueryui.com/autocomplete/

0赞 Imran Kabir 11/15/2012 #3
<html lang="html">
<head>
<meta charset="utf-8" />
<title>Search Users</title>
<script type="text/javascript" src="http://code.jquery.com/jquery.min.js"></script>
<script type="text/javascript">
$(function(){
    $("#searched_users_list").hide();
    $("#search_user").keyup(function(){
        var what_to_search = $(this).val().toLowerCase();
        if(what_to_search != ''){
            $("#users_list").hide();
            $("#searched_users_list").show();
        }else{
            $("#users_list").show();
            $("#searched_users_list").hide();
        }
        var searched_users = '';
        var user_name = '';
        $("#users_list").children('li:contains("'+what_to_search+'")').each(function(){
            user_name = $(this).html();
            searched_users += "<li>"+user_name+"</li>";
        });

        if(searched_users != '')
            $("#searched_users_list").html(searched_users);
        else
            $("#searched_users_list").html('<li>No user found....</li>');
    })
});
</script>
</head>
<body>
<ul id="users_list">
    <li>abc</li>
    <li>defa</li>
    <li>ghife</li>
</ul>
<ul id="searched_users_list">
    <li>No user found....</li>
</ul>
<input type="text" name="search_users" id="search_user" value="" />
</body>
</html>

It's my first post... Sorry If something is odd.

评论

0赞 Imran Kabir 11/15/2012
请检查一下。我希望这会有所帮助。
0赞 Imran Kabir 11/20/2012
这是一个完整的工作示例,例如 FB 聊天搜索......试试:)
0赞 RaviTeja 12/6/2012
看起来这更像是我已经拥有的搜索功能,我正在寻找的是浏览列表,第一个答案解决查看 jsfiddle。
1赞 Jibi Abraham 11/16/2012 #4

为了没有将jQuery添加到标签列表中的OP(我恭喜你),我将添加一个纯JavaScript的小型导航插件。

披露 - 实际逻辑的功劳将归功于 Facebook 成名的 Eric Priestley,因为它是从他的 Javelin JS 库中移除的。

在这里发布了一个小提琴的链接。我还从您的问题中假设您已经有一个搜索解决方案,并且只是在寻找导航解决方案。

从小提琴交叉发布。我在代码中注释了很多,所以应该是不言自明的。不过,如果您确实有疑问,请与我们联系。

[HTML全文]可以是任何你想要的东西。为简洁起见,我添加了这个

<input id='control' type='text' placeholder='Search text here' />
<!-- List of results - hardcoded for now -->
<ul id='hardpoint' class='hidden'>
    <li class='item'>Search item 1</li>
    <li class='item'>Search item 2</li>
    <li class='item'>Search item 3</li>
    <li class='item'>Search item 4</li>
</ul>​

代码

(function() {


    /**
     * Helper functions
     */
    var bind = function( context, func, more ){
        if (typeof func !== 'function') {
            throw new Error ("No function supplied to bind function");
        }
        var bound = Array.prototype.slice.call( arguments ).slice(2);
        if (func.bind) {
            return func.bind.apply(func, [context].concat(bound));
        }
        return function() {
            return func.apply(context || window, bound.concat(Array.prototype.slice.call( arguments )));
        }
    };

    var getChildElements = function(node) {
        var childNodes = Array.prototype.slice.call(node.childNodes);
        childNodes = childNodes.filter(function(item) {
            return item.tagName && item.nodeType && item.nodeType === 1 && item !== node;
        });
        return childNodes;
    };

    var hasClass = function(node, className) {
        return ((' ' + node.className + ' ').indexOf(' ' + className + ' ') > -1);
    };

    var alterClass = function(node, className, add) {
        var has = hasClass(node, className);
        if (add && !has) {
            node.className = node.className.trim() + ' ' + className;
        }
        else if (has && !add) {
            node.className = node.className.replace(
            new RegExp('(^|\\s)' + className + '(?:\\s|$)', 'g'), ' ');
        }
    };

    var getIndex = function( list, element ){
        var index = -1;
        for (var i = 0; i < list.length; i++) {
            if( list[i] === element ){
                index = i;
                break;
            }
        };
        return index;
    };


    // Start of plugin
    // Constructor
    var ListNavigator = function(config) {};

    // This can be moved within constructor if you so prefer
    ListNavigator.prototype.init = function(config) {
        // On what element do you want the controls to activate
        // Pressing up down etc on this element, triggers tha navigation
        this.control = document.getElementById(config.control);

        // The list of results that can be navigated
        this.hardpoint = document.getElementById(config.hardpoint);

        // A list of items ( usually, childNodes ) of hardpoint
        // Dynamically populated
        this.display =  getChildElements(this.hardpoint);;

        // What to set the focus on initially
        this.focus = -1;

        // Specify a function to execute when the user chooses a result
        // Keydown - Return : configured to be the choose event type now
        this.choose = config.choose;

        this.selector = config.selector;
    };

    ListNavigator.prototype.run = function() {
        var controlEvents = ['focus', 'blur', 'keydown', 'input'],
            mouseEvents = [ 'mouseover', 'mouseout', 'mousedown' ],
            self = this,
            selector = this.selector;

        controlEvents.forEach(function(event) {
            self.control.addEventListener(event, bind(self, self.handleEvent), false);
        });

        mouseEvents.forEach(function(event) {
            self.hardpoint.addEventListener(event, bind(self, self.onmouse), true);
        });
    };

    // Logic to change the focus on keydown
    ListNavigator.prototype.changeFocus = function(d) {
        var n = Math.min( Math.max( -1, this.focus + d ), this.display.length - 1);

        if (this.focus >= 0 && this.focus < this.display.length) {
            alterClass(this.display[this.focus], 'focused', false);
        }

        this.focus = n;
        this.drawFocus();
        return true;
    };

    // Set the focus on the targetted element
    ListNavigator.prototype.drawFocus = function() {
        var f = this.display[this.focus];
        if (f) {
            alterClass(f, 'focused', true);
        }
    };

    // Handle mouse events
    ListNavigator.prototype.onmouse = function(event) {
        var target = event.target, type = event.type;
        if ( hasClass( target, this.selector ) ) {
            if ( type === 'mousedown' ) {
                // Choose this element
                this.choose(target);
            }
            else if ( type === 'mouseover' ) {
                // Set the focus to element on which mouse is hovering on
                this.focus = getIndex( this.display, target );
                this.drawFocus();
            } 
            else if ( type === 'mouseout' ){
                // Reset the display to none
                this.changeFocus(Number.NEGATIVE_INFINITY);
            }
        };
    };

    ListNavigator.prototype.handleEvent = function(e) {
        var type = e.type;
        if (type === 'blur') {
            this.focused = false;
            this.hide();
        } else {
            alterClass(this.hardpoint, 'hidden', false);
            this.update(e);
        }
    };

    ListNavigator.prototype.hide = function() {
        this.changeFocus(Number.NEGATIVE_INFINITY);
        this.display = [];
        alterClass(this.hardpoint, 'hidden', true);
    };

    ListNavigator.prototype.submit = function() {
        if (this.focus >= 0 && this.display[this.focus]) {
            this.choose(this.display[this.focus]);
        } else {

            if (this.display.length) {
                this.changeFocus(1);
                this.choose(this.display[this.focus]);
            }
        }
        return false;
    };

    ListNavigator.prototype.update = function(event) {

        console.log( 'upadte' );
        this.display = getChildElements(this.hardpoint);

        if (event.type === 'focus') {
            this.focused = true;
        }

        var k = event.which;
        if (k && event.type == 'keydown') {
            switch (k) {
            case 38:
                if (this.display.length && this.changeFocus(-1)) {
                    event.stopPropagation();
                }
                break;
            case 40:
                if (this.display.length && this.changeFocus(1)) {
                    event.stopPropagation();
                }
                break;
            case 13:
                if (this.display.length) {
                    this.hide();
                    event.stopPropagation();
                    this.submit();
                }
                break;
            case 27:
                if (this.display.length) {
                    this.hide();
                    event.stopPropagation();
                }
                break;
            case 9:
                // If the user tabs out of the field, don't refresh.
                return;
            }
        }

    };

    window.ListNav = ListNavigator

})();

var nav = new ListNav();
nav.init({
    control: 'control',
    hardpoint: 'hardpoint',
    selector: 'item',
    choose: function(){
        console.log( arguments );
    }
});
nav.run();​