Подтвердить что ты не робот

Twitter bootstrap typeahead несколько значений?

Я использую Twitter Bootstrap с JQuery. Я хочу использовать функцию TYPEAHEAD для текстового поля, которое я должен работать очень легко. Но я также нуждаюсь в этом, чтобы разрешить множественный выбор.

Под этим я подразумеваю, что после того, как я выбрал одно слово из автозаполнения, он возвращает меня в textarea с дополнительным пространством после этого, а затем, если я начну вводить текст снова, он предлагает мне сделать это снова.

Вот JS bin: http://jsbin.com/ewubuk/1/edit (Ничего особенного внутри).

Есть ли простое решение для множественного выбора с помощью typeahead? Если да, то как?

Спасибо заранее.

4b9b3361

Ответ 1

Изменить. Уже было натянуто: https://github.com/twitter/bootstrap/pull/2007


Вы можете подойти к желаемому поведению, используя прокси для typeahead: Demo (jsfiddle)

var $myTextarea = $('#myTextarea');

$('.typeahead').typeahead({
    source: source,
    updater: function(item) {
        $myTextarea.append(item, ' ');
        return '';
    }
});

Я думаю, что метод updater предназначен для такого рода вещей, вы просто возвращаете то, что будет отображаться.


Или , если вы действительно хотите, чтобы все было в одном и том же элементе ввода, вам придется переопределить больше методов, чтобы он соответствовал только вложенному в данный момент элементу: Демо (jsfiddle)

function extractor(query) {
    var result = /([^,]+)$/.exec(query);
    if(result && result[1])
        return result[1].trim();
    return '';
}

$('.typeahead').typeahead({
    source: source,
    updater: function(item) {
        return this.$element.val().replace(/[^,]*$/,'')+item+',';
    },
    matcher: function (item) {
      var tquery = extractor(this.query);
      if(!tquery) return false;
      return ~item.toLowerCase().indexOf(tquery.toLowerCase())
    },
    highlighter: function (item) {
      var query = extractor(this.query).replace(/[\-\[\]{}()*+?.,\\\^$|#\s]/g, '\\$&')
      return item.replace(new RegExp('(' + query + ')', 'ig'), function ($1, match) {
        return '<strong>' + match + '</strong>'
      })
    }
});

Это не доказательство идиота, потому что вы должны вводить текст в конце после специального символа.

Ответ 2

Это отличная замена для выделенных ящиков:

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

(Если вы используете многозначную версию.)

Ответ 3

Я думаю, вы могли бы редактировать плагин, чтобы разрешить несколько выборов (просто не закрывайте раскрывающееся меню) и добавляйте выбранные значения, разделенные запятой. Единственная проблема, которую я вижу, это то, что вы не знаете, когда закрыть раскрывающийся список.

Ответ 4

Верхний ответ больше не работает с последним typeahead, поэтому я предлагаю следующее.

FIDDLE

function MultiTypeahead(id, data, trigger, vertAdjustMenu)
{
    trigger = (undefined !== trigger) ? trigger : '';
    var validChars = /^[a-zA-Z]+$/;


    function extractor(query)
    {
        var result = (new RegExp('([^,; \r\n]+)$')).exec(query);
        if(result && result[1])
            return result[1].trim();
        return '';
    }

    var lastUpper = false;
    function strMatcher(id, strs) 
    {
        return function findMatches(q, sync, async) 
        {
            var pos = $(id).caret('pos');
            q = (0 < pos) ? extractor(q.substring(0, pos)) : '';

            if (q.length <= trigger.length)
                return;

            if (trigger.length)
            {
                if(trigger != q.substr(0, trigger.length))
                    return;

                q = q.substr(trigger.length);
            }

            if (!q.match(validChars))
                return;

            var firstChar = q.substr(0, 1);
            lastUpper = (firstChar === firstChar.toUpperCase() && firstChar !== firstChar.toLowerCase());

            var cpos = $(id).caret('position');
            $(id).parent().find('.tt-menu').css('left', cpos.left + 'px');
            if (vertAdjustMenu)
                $(id).parent().find('.tt-menu').css('top', (cpos.top + cpos.height) + 'px');

            var matches = [];
            var matches = [], substrRegex = new RegExp(q, 'i');
            $.each(strs, function(i, str) 
            {
                if (str.length > q.length && substrRegex.test(str))
                    matches.push(str);
            });

            if (!matches.length)
                return;

            sync(matches);
        };
    };

    var lastVal = '';
    var lastPos = 0;
    function beforeReplace(event, data)
    {
        lastVal = $(id).val();
        lastPos = $(id).caret('pos');
        return true;
    }

    function onReplace(event, data)
    {            
        if (!data || !data.length)
            return;

        if (!lastVal.length)
            return;

        var root = lastVal.substr(0, lastPos);
        var post = lastVal.substr(lastPos);

        var typed = extractor(root);
        if (!lastUpper && typed.length >= root.length && 0 >= post.length)
            return;

        var str = root.substr(0, root.length - typed.length);

        str += lastUpper ? (data.substr(0, 1).toUpperCase() + data.substr(1)) : data;
        var cursorPos = str.length;

        str += post;

        $(id).val(str);
        $(id).caret('pos', cursorPos);      
    }

    this.typeahead = 
        $(id).typeahead({hint: false, highlight: false}, {'limit': 5, 'source': strMatcher(id, data)})
                .on('typeahead:beforeselect', beforeReplace)
                .on('typeahead:beforeautocomplete', beforeReplace)
                .on('typeahead:beforecursorchange', beforeReplace)
                .on('typeahead:selected', function(event,data){setTimeout(function(){ onReplace(event, data); }, 0);})
                .on('typeahead:autocompleted', onReplace)
                .on('typeahead:cursorchange', onReplace)
                ;
}

РЕДАКТИРОВАТЬ: Понимая, что в предыдущем коде было слишком много лишних вещей, я сузил его до минимального рабочего примера.

Это - это то, что было опубликовано ранее.