19

我正在使用Twitter Bootstrap 2.1.1 和 jQuery 1.8.1的Typeahead组件

我正在尝试从 typeahead 的updater函数中访问文本框元素。这是我当前的代码,效果很好:

$('#client-search').typeahead({
    source: function (query, process) {
        $.get(url, { query: query }, function (data) {
            labels = [];
            mapped = {};
            $.each(data, function(i,item) {
                mapped[item.label] = item.value;
                labels.push(item.label);
            });
            process(labels);
        });
    }
    ,updater: function (item) {
        $('#client-id').val(mapped[item]);
        return item;
    }
    ,items: 10
    ,minLength: 2
});

我在同一页面上有许多预先输入的搜索框。每个搜索框都有一个 id#xxx-search和一个对应的 id 隐藏输入#xxx-id。我想通过执行以下操作对所有内容重新使用相同的代码:

$('#client-search, #endClient-search, #other-search').typeahead({

    ...

    ,updater: function (item) {
        var target = $(this).attr('id').split('-')[0] + '-id';
        $('#'+target).val(mapped[item]);
        return item;
    }

    ...

我认为这this将引用正在使用的文本框元素,但显然不是,因为我在 firebug 中得到一个未定义的错误:

$(this).attr("id") is undefined

当我使用this而不是$(this),我得到:

this.attr is not a function

任何人都可以帮助完成这项工作吗?


更新:答案,还有更多!

感谢 benedict_w 在下面的回答,这不仅现在可以完美地工作,而且在可重用性方面我也做得更好。

这是我<input>的样子:

<input type="text" autocomplete="off"
    id="client-search"
    data-typeahead-url="/path/to/json"
    data-typeahead-target="client-id">
<input type="hidden" id="client-id" name="client-id">

请注意搜索框如何引用隐藏的 ID。这是新的js:

$(':input[data-typeahead-url]').each(function(){
    var $this = $(this);
    $this.typeahead({
        source: function (query, process) {
            $.get($this.attr('data-typeahead-url'), { query: query }, function (data) {
                labels = [];
                mapped = {};
                $.each(data, function(i,item) {
                    mapped[item.label] = item.value;
                    labels.push(item.label);
                });
                process(labels);
            });
        }
        ,updater: function (item) {
            $('#'+$this.attr('data-typeahead-target')).val(mapped[item]);
            return item;
        }
        ,items: 10
        ,minLength: 2
    });
});
4

2 回答 2

24

您在另一个函数内部,因此您需要保存对外部闭包中元素的引用,然后可以将其传递给内部闭包的事件处理程序。您可以通过$.each()调用循环选择器来执行此操作,每次都保存对元素 (this) 的引用 - 例如

$('#client-search, #endClient-search, #other-search').each(function() {
    var $this = $(this);
    $this.typeahead({
        ...

        ,updater: function (item) {
            var target = $this.attr('id').split('-')[0] + '-id';
            $('#'+target).val(mapped[item]);
            return item;
        }
        ...
     });
于 2012-09-13T19:44:26.177 回答
6

接受的答案是一个很好的答案,但只是想分享一个实际上不需要外部封闭的替代方案。我正在使用 bootstrap v2.3.1,您可以从this.$element.

例如:

$(':input[data-typeahead-url]').typeahead({
    source: function (query, process) {
        $.get(this.$element.attr('data-typeahead-url'), { query: query }, function (data) {
            labels = [];
            mapped = {};
            $.each(data, function(i,item) {
                mapped[item.label] = item.value;
                labels.push(item.label);
            });
            process(labels);
        });
    }
    ,updater: function (item) {
        $('#'+this.$element.attr('data-typeahead-target')).val(mapped[item]);
        return item;
    }
    ,items: 10
    ,minLength: 2
});
于 2013-07-03T10:20:04.137 回答