Show datalist labels but submit the actual value

140,669

Solution 1

Note that datalist is not the same as a select. It allows users to enter a custom value that is not in the list, and it would be impossible to fetch an alternate value for such input without defining it first.

Possible ways to handle user input are to submit the entered value as is, submit a blank value, or prevent submitting. This answer handles only the first two options.

If you want to disallow user input entirely, maybe select would be a better choice.


To show only the text value of the option in the dropdown, we use the inner text for it and leave out the value attribute. The actual value that we want to send along is stored in a custom data-value attribute:

To submit this data-value we have to use an <input type="hidden">. In this case we leave out the name="answer" on the regular input and move it to the hidden copy.

<input list="suggestionList" id="answerInput">
<datalist id="suggestionList">
    <option data-value="42">The answer</option>
</datalist>
<input type="hidden" name="answer" id="answerInput-hidden">

This way, when the text in the original input changes we can use javascript to check if the text also present in the datalist and fetch its data-value. That value is inserted into the hidden input and submitted.

document.querySelector('input[list]').addEventListener('input', function(e) {
    var input = e.target,
        list = input.getAttribute('list'),
        options = document.querySelectorAll('#' + list + ' option'),
        hiddenInput = document.getElementById(input.getAttribute('id') + '-hidden'),
        inputValue = input.value;

    hiddenInput.value = inputValue;

    for(var i = 0; i < options.length; i++) {
        var option = options[i];

        if(option.innerText === inputValue) {
            hiddenInput.value = option.getAttribute('data-value');
            break;
        }
    }
});

The id answer and answer-hidden on the regular and hidden input are needed for the script to know which input belongs to which hidden version. This way it's possible to have multiple inputs on the same page with one or more datalists providing suggestions.

Any user input is submitted as is. To submit an empty value when the user input is not present in the datalist, change hiddenInput.value = inputValue to hiddenInput.value = ""


Working jsFiddle examples: plain javascript and jQuery

Solution 2

The solution I use is the following:

<input list="answers" id="answer">
<datalist id="answers">
  <option data-value="42" value="The answer">
</datalist>

Then access the value to be sent to the server using JavaScript like this:

var shownVal = document.getElementById("answer").value;
var value2send = document.querySelector("#answers option[value='"+shownVal+"']").dataset.value;


Hope it helps.

Solution 3

I realize this may be a bit late, but I stumbled upon this and was wondering how to handle situations with multiple identical values, but different keys (as per bigbearzhu's comment).

So I modified Stephan Muller's answer slightly:

A datalist with non-unique values:

<input list="answers" name="answer" id="answerInput">
<datalist id="answers">
  <option value="42">The answer</option>
  <option value="43">The answer</option>
  <option value="44">Another Answer</option>
</datalist>
<input type="hidden" name="answer" id="answerInput-hidden">

When the user selects an option, the browser replaces input.value with the value of the datalist option instead of the innerText.

The following code then checks for an option with that value, pushes that into the hidden field and replaces the input.value with the innerText.

document.querySelector('#answerInput').addEventListener('input', function(e) {
    var input = e.target,   
        list = input.getAttribute('list'),
        options = document.querySelectorAll('#' + list + ' option[value="'+input.value+'"]'),
        hiddenInput = document.getElementById(input.getAttribute('id') + '-hidden');

    if (options.length > 0) {
      hiddenInput.value = input.value;
      input.value = options[0].innerText;
      }

});

As a consequence the user sees whatever the option's innerText says, but the unique id from option.value is available upon form submit. Demo jsFiddle

Solution 4

When clicking on the button for search you can find it without a loop.
Just add to the option an attribute with the value you need (like id) and search for it specific.

$('#search_wrapper button').on('click', function(){
console.log($('option[value="'+ 
$('#autocomplete_input').val() +'"]').data('value'));
})
Share:
140,669
Stephan Muller
Author by

Stephan Muller

Front-end developer, movie enthousiast, amateur unicorn.

Updated on May 11, 2021

Comments

  • Stephan Muller
    Stephan Muller about 3 years

    Currently the HTML5 <datalist> element is supported in most major browsers (except Safari) and seems like an interesting way to add suggestions to an input.

    However, there seem to be some discrepancies between the implementation of the value attribute and the inner text on the <option>. For example:

    <input list="answers" name="answer">
    <datalist id="answers">
      <option value="42">The answer</option>
    </datalist>
    

    This is handled differently by different browsers:

    Chrome and Opera:
    Datalist in Chrome/Opera

    FireFox and IE 11:
    Datalist in FireFox

    After selecting one, the input is filled with the value and not the inner text. I only want the user to see the text ("The answer") in the dropdown and in the input, but pass the value 42 on submit, like a select would.

    How can I make all browsers have the dropdown list show the labels (inner text) of the <option>s, but send the value attribute when the form is submitted?

    • TylerH
      TylerH about 9 years
      I think Firefox and IE11 are wrong here; according to the specs One and Two, it seems that value="" should take precedence over a string within the tags, whenever there is a value="" declared. So the suggestion would be to make "the answer" your value attribute.
    • easwee
      easwee about 9 years
      @TylerH No - Firefox and IE11 are correct here: The label attribute provides a label for element. The label of an option element is the value of the label content attribute, if there is one, or, if there is not, the value of the element's text IDL attribute. w3.org/TR/html5/forms.html#attr-option-label
    • TylerH
      TylerH about 9 years
      @easwee That just says the label should be what's in label if there is a label, and that the value should be what's in the value if there is a value. It doesn't specify which one takes precedence.
  • bigbearzhu
    bigbearzhu over 8 years
    how would you deal with options with same innertext but different data-values? e.g. jsfiddle.net/7k3sovht/78
  • Stephan Muller
    Stephan Muller over 8 years
    As far as I can tell, there's no way to differentiate between the two options. Because there is no user event to listen to it's impossible to know which of the two they actually selected; all we know is what value ended up in the input field.
  • Dejell
    Dejell over 7 years
    I am getting undefined any reason?
  • Hezbullah Shah
    Hezbullah Shah over 7 years
    Trouble shoot with console.log at each step. Make sure you are using same id value which you are calling. And check semicollons too.
  • Chris22
    Chris22 over 7 years
    @StephanMuller great answer for someone who is learning more HTML5 like me, thanks! How would you clear out the textbox for the demo after submit?
  • Piotr Kazuś
    Piotr Kazuś about 7 years
    Pure jquery document.querySelector('input[list]').addEventListener('inpu‌​t', function(e) { var value = $(this).val(); var list = $(this).attr('list'); var id = $(this).attr('id'); $('#'+list+' option').each(function(){ if($(this).val() == value){ $('#'+id+'-hidden').val($(this).attr('data-value')); } }); });
  • digitai
    digitai over 6 years
    @StephanMuller thanks for sharing, very useful, I'm running as you coded, however I get the data-value just when using backspace in the input, not when submitting or even selecting a differetn item, what can be wrong?
  • digitai
    digitai over 6 years
    @StephanMuller your answer is so clear, I was just using the <option data-value="x" value="" />, instead of <option ...>...</option>. Now it works as expected, thanks,
  • Stephane L
    Stephane L over 6 years
    Thank you this is a good solution but it doesn't work if the option has some white spaces in begining or end, or two white spaces consecutives, I don't know why but the input is trimming automatically the datalist options : <option data-value="3"> Pi </option>
  • Stephane L
    Stephane L over 6 years
    The text is not trimmed when it is in the value attribute : <option data-value="3" value=" Pi "></option>
  • Lucian Minea
    Lucian Minea about 6 years
    @bigbearzhu, for that case where the values could be identical, use both value="" and data-value="" for <option>, then get the data-value with javascript, simple.
  • Hezbullah Shah
    Hezbullah Shah about 6 years
    @ApurvG this is the variable storing the value shown in the field.
  • Fusseldieb
    Fusseldieb almost 5 years
    That seems like a mess and smells like SQL injection. Don't do that.
  • John
    John over 4 years
    @23r0, some feedback as to why this may have been voted down; this question is about frontend code specifically. Including backend code (like PHP) may be unnecessary and confusing. Regarding the HTML, I expect you'll only ever get the last value for customer_id_real submitted, not the selected value.
  • WEBjuju
    WEBjuju over 4 years
    this solution is such a great idea! - i was able to, rather simply, implement this solution in minutes. first i updated my input element with attribute data-value="1". Then i wrote the following block in the places where the value was being grabbed and used: if (typeof $(this).attr('data-value') != 'undefined') { var id = $(this).attr('name'); val = $('#'+id).find('option[value="'+val+'"]').attr('data-value')‌​; }
  • Stephan Muller
    Stephan Muller about 4 years
    Nice addition :)
  • Richard Aguirre
    Richard Aguirre almost 4 years
    But what if you have two equals captions with some different data-values
  • ABODE
    ABODE almost 4 years
    Thanks Hezbullah Shah you really saved so much hours
  • Rune Kaagaard
    Rune Kaagaard over 3 years
    @Fusseldieb as it doesn't take any user input it is perfectly safe. IMHO It's a pretty harsh statement marking it as so.
  • harbii
    harbii almost 3 years
    It doesn't work when on one page are two datalists. What I do wrong? I change name & id in forms. What else do I have to do
  • Saghachi
    Saghachi over 2 years
    And why it's voted down then?
  • aProgger
    aProgger over 2 years
    @RuneKaagaard I don't see any sqlinjection possibilities IN HIS/HER ANSWER too. But it should be noted for others: just because an input is hidden, readonly or deactivated, it still get's send to the server. Even if a normal user cannot input something, it is possible with the browsers devtools => sqlinjections are possible. Backend should always distrust frontend.