Select2的Ajax方法无法选择问题

73

好的,我确定这里有些简单的设置出了问题,但我不确定是什么。

所以我正在尝试使用Select2 AJAX方法作为一种让用户搜索数据库并选择结果的方式。调用本身返回结果,但它不允许我从列表中选择答案。它似乎也不允许您在悬停或上/下箭头时“选择”它。因此,这是我的代码:

index.html

<html>
    <head>
        <script src="http://ajax.googleapis.com/ajax/libs/jquery/1.9.0/jquery.min.js"></script>
        <link rel="stylesheet" type="text/css" href="select2/select2.css" media="screen" />
        <script src="select2/select2.js"></script>
        <script src="select.js"></script>
    </head>
    <body>
        <input type="text" style="width: 500px" class="select2">
    </body>
</html>

选择.js

jQuery(function() {

  var formatSelection = function(bond) {
    console.log(bond)
    return bond.name
  }

  var formatResult = function(bond) {
    return '<div class="select2-user-result">' + bond.name + '</div>'
  }

  var initSelection = function(elem, cb) {
    console.log(elem)
    return elem
  }

    $('.select2').select2({
      placeholder: "Policy Name",
      minimumInputLength: 3,
      multiple: false,
      quietMillis: 100,
      ajax: {
        url: "http://localhost:3000/search",
        dataType: 'json',
        type: 'POST',
        data: function(term, page) {
          return {
            search: term,
            page: page || 1
          }
        },
        results: function(bond, page) {
          return {results: bond.results, more: (bond.results && bond.results.length == 10 ? true: false)}
        }
      },
      formatResult: formatResult,
      formatSelection: formatSelection,
      initSelection: initSelection
    })
})

JSON响应

{
  error: null,
  results: [
    {
      name: 'Some Name',
      _id: 'Some Id'
    },
    {
      name: 'Some Name',
      _id: 'Some Id'
    }
  ]
}

所有的东西看起来都正确,但是我无法实际选择答案并将其输入到框中。我的代码有问题吗?


2
经过进一步调查,当我将下拉菜单类从select2-result-unselectable更改为select2-result-selectable时,一切似乎都正常工作。 - AlbertEngelB
我会避免使用类进行jquery调用,而是使用id。 $('#thisSelect').select2({...... - ganders
@ganders 我不知道这有什么关系。 - AlbertEngelB
如果您在同一表单/页面上使用了两个select2实现,那么在其中一个中输入内容会同时在两个中显示结果。类被设计为可重用,而ID应该是单一用途的。 - ganders
7个回答

140

在查看另一个答案后,我发现我需要在每次调用时传递id字段,否则它将禁用输入。

解决此问题的示例代码:

$('.select2').select2({
  placeholder: "Policy Name",
  minimumInputLength: 3,
  multiple: false,
  quietMillis: 100,
  id: function(bond){ return bond._id; },
  ajax: {
    url: "http://localhost:3000/search",
    dataType: 'json',
    type: 'POST',
    data: function(term, page) {
      return {
        search: term,
        page: page || 1
      }
    },
    results: function(bond, page) {
      return {results: bond.results, more: (bond.results && bond.results.length == 10 ? true: false)}
    }
  },
  formatResult: formatResult,
  formatSelection: formatSelection,
  initSelection: initSelection
})

编辑

由于这篇文章一直受到点赞,我将详细说明一下。 .select2()方法要求所有结果都具有唯一的id字段,但是幸运的是我们有一个解决办法。可以使用如下的函数作为id选项:

function( <INDIVIDUAL_RESULT> ) {
  // Expects you to return a unique identifier.
  // Ideally this should be from the results of the $.ajax() call. 
}

由于我的唯一标识符是<RESULT>._id,我只需 return <RESULT>._id;


2
非常感谢您的回答!真是太太太有帮助了!我没有收到任何错误信息,而且我走过的每一步都表明它应该可以工作。 - Mike Homol
@MikeHomol 没问题,我很高兴这个问题能帮助到社区中的许多人。惊讶这是一个如此普遍的问题! - AlbertEngelB
1
谢谢您。小提示,当表单被提交时,字段值将是字符串“object [Object]”。将“return {id:bond._id}”替换为“return bond._id”,一切都恢复正常了... - medowlock
16
很遗憾,这在 Select2 4.0 中不再起作用。你需要使用这里的代码来使其工作。 - awidgery
@awidgery 谢谢你提供的信息!我已经在问题中添加了 [tag:jquery-select2-3] 标签,希望这能够缩小适用于您的解决方案范围。 - AlbertEngelB
显示剩余2条评论

15

问题是JSON数据需要一个名为"id"的属性。没有必要

id: function(bond){ return bond._id; }

如果数据本身没有ID,您可以像这样在processResults中添加函数来创建它。

        $(YOUR FIELD).select2({
            placeholder: "Start typing...",
            ajax: {
                type: "POST",
                contentType: "application/json; charset=utf-8",
                url: "YOUR API ENDPOINT",
                dataType: 'json',
                data: 
                    function (params) {
                        return JSON.stringify({ username: params.term });
                },
                processResults: function (data, page) {

                    var results = [];

                    $.each(JSON.parse(data.d), function (i, v) {
                        var o = {};
                        o.id = v.key;
                        o.name = v.displayName;
                        o.value = v.key;
                        results.push(o);
                    })

                    return {
                        results: results
                    };
                },
                cache: true
            },
            escapeMarkup: function (markup) { return markup;},
            minimumInputLength: 1,
            templateResult: function (people) {
                if (people.loading)
                    return people.text;

                var markup = '<option value="' + people.value + '">' + people.name + '</option>';

                return markup;
            },
            templateSelection: function (people) { return people.value || people.text }

        });    

如果您正在使用最新版本的select2(截至2016年5月),则选择器中的ID选项似乎无法正常工作,请按照此解决方案进行操作。 - Vihari Piratla

7

同时要注意大小写。我使用的是 Id,但 select2 期望的是 id。这可能会为某些人节省时间。


6

就像Dropped.on.Caprica所说:每个结果项都需要一个唯一的ID。

因此,只需为每个结果分配一个唯一的id号码:

$('#searchDriver').select2({
    ajax: {
       dataType: 'json',
       delay: 250,
       cache: true,
       url: '/users/search',
       processResults: function (data, params) {
           //data is an array of results
           data.forEach(function (d, i) {
               //Just assign a unique number or your own unique id
               d.id = i; // or e.id = e.userId;
           })

           return {
               results: data
           };
       },
    },
    escapeMarkup: function (markup) { return markup; },
    minimumInputLength: 1,
    templateResult: formatRepo,
    templateSelection: formatRepoSelection
});

1

我遇到了以下错误: Select2不允许选项“ajax”

在我的情况下,当使用Select2版本3.5时,它会出现在选择时, 因此,您应该升级到版本4.0,并且不需要将隐藏输入附加到您的选择中。


0

enter image description here

您可以通过以编程方式打开 select2 元素,然后触发 return 键按下来触发选择选项(在此情况下为第一个选项)。

var e = jQuery.Event('keydown');
e.which = 13;

$('.offer_checkout_page_link select').select2('open');
$('.offer_checkout_page_link .select2-selection').trigger(e);

这将“设置”选项,就像您单击它一样。您可以通过触发适当数量的down按键来选择特定选项,然后再触发return

似乎在幕后,当您单击(或在本例中按enter)以选择选项时,会向选择元素添加一个选项元素。您会注意到,当页面首次加载select2选择元素时,没有任何option元素子级。这就是为什么其他解决方案建议使用jQuery添加选项元素,然后进行选择的原因。


0
我正在将 select2 绑定到一个 div 上,但是使用 ajax 作为数据源 时似乎无法工作,这可能会帮助某些人。
<div id="testSelect2"> </div>

改成了这个

<select id="testSelect2"> </select>

然后它开始保留所选项目。


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接