Friday, January 6, 2017

How do I mimic keyboard behavior in my styled select dropdown?

Leave a Comment

I’m using jQuery 1.11. I want to style my select drop-downs because, let’s face it, the default looks sucks. So I found some styles

.selectMenu {   font-family: 'Oxygen', sans-serif;   font-size: 20px;   height: 50px;   -webkit-appearance: menulist-button; }  .select-hidden {   display: none;   visibility: hidden;   padding-right: 10px; }  .select {   cursor: pointer;   display: inline-block;   position: relative;   font-size: 16px;   color: #fff;   width: 220px;   height: 42px; }  .select-styled {   position: absolute;   top: 0;   right: 0;   bottom: 0;   left: 0;   background-color: gray;   padding: 11px 12px;   -webkit-transition: all 0.2s ease-in;   transition: all 0.2s ease-in; } .select-styled:after {   content: "";   width: 0;   height: 0;   border: 7px solid transparent;   border-color: #fff transparent transparent transparent;   position: absolute;   top: 16px;   right: 10px; } .select-styled:hover {   background-color: #7b7b7b; } .select-styled:active, .select-styled.active {   background-color: #737373; } .select-styled:active:after, .select-styled.active:after {   top: 9px;   border-color: transparent transparent #fff transparent; }  .select-options {   display: none;   position: absolute;   top: 100%;   right: 0;   left: 0;   z-index: 999;   margin: 0;   padding: 0;   list-style: none;   background-color: #737373;   overflow: scroll; } .select-options li {   margin: 0;   padding: 12px 0;   text-indent: 15px;   border-top: 1px solid #676767;   -webkit-transition: all 0.15s ease-in;   transition: all 0.15s ease-in; } .select-options li:hover {   color: gray;   background: #fff; } .select-options li[rel="hide"] {   display: none; }  ul.select-options {   max-height: 15em;   overflow-y: scroll;   overflow-x: hidden; } 

and added this jQuery

$(function() {          $('select').each(function(){             styleSelectMenu($(this));         });  });  // This method applies the styles to our select menu function styleSelectMenu(selectMenu) {     var $this = $(selectMenu), numberOfOptions = $(selectMenu).children('option').length;                  /*** NEW - start ***/                 var $paddingCalculator = $('<div />', {                 'class' : "select-styled test"     }).css({                 width : 0,         visibility : "hidden"     }).appendTo("body");     $this.addClass('select-hidden');     var paddingWidth = $paddingCalculator.outerWidth() + 10;     $paddingCalculator.remove();      //var selectWidth = $this.width() + paddingWidth;     var selectWidth = $this.outerWidth() + paddingWidth;     //alert(selectWidth);      if ( !$this.parent().hasClass('select') ) {                 var $wrapper = $("<div />", {                         'class' : "select"         }).css({                         width   : selectWidth         });         $this.wrap( $wrapper );     }   // if                  /*** NEW - end ***/      if ( !$this.next().hasClass('select-styled') ) {         $this.after('<div class="select-styled"></div>');     }    // if      var $styledSelect = $this.next('div.select-styled');     $styledSelect.text($this.children('option').eq(0).text());      if ( $styledSelect.parent().find('ul').length > 0 ) {         $styledSelect.parent().find('ul').remove();     }   // if     var $list = $('<ul />', {         'class': 'select-options'     }).insertAfter($styledSelect);      for (var i = 0; i < numberOfOptions; i++) {         $('<li />', {             text: $this.children('option').eq(i).text(),             rel: $this.children('option').eq(i).val()         }).appendTo($list);     }      var $listItems = $list.children('li');      // This is the event when someone opens the list     $styledSelect.unbind('click')     $styledSelect.click(function(e) {         e.stopPropagation();         $('div.select-styled.active').each(function(){             $(this).removeClass('active').next('ul.select-options').hide();         });         $(this).toggleClass('active').next('ul.select-options').toggle();     });      // This is the event when someone actually selects something from the list     $listItems.unbind('click.selectStyledItem')     $listItems.bind('click.selectStyledItem', function(event) {          clickListItem(event, $styledSelect, $this, $(this), $list);     });      $(document).click(function(event) {         $styledSelect.removeClass('active');         $list.hide();     });       var selectedIndex = $this[0].selectedIndex;     if (selectedIndex > 0) {         var name = $this.attr("name")         var selectedText = $( "select[name='" + name + "'] option:selected" ).text();         selectItemFromStyledList($styledSelect, $this, selectedText, $list);     }   // if  }  // This is the method that will select an item from the styled list function selectItemFromStyledList(styledSelect, selectMenu, selectedText, listToHide) {     $(styledSelect).text(selectedText).removeClass('active');     $(selectMenu).val($(selectMenu).attr('rel'));     $(listToHide).hide();     // Select option in the underlying list so that the form gets submitted     // with the right values     selectedOption = $(selectMenu).find("option").filter(function () { return $(this).html() == selectedText; });     $(selectMenu).find("option[selected='selected']").removeAttr("selected");     $(selectedOption).attr("selected","selected"); }       // selectItemFromStyledList  function clickListItem(event, styledSelect, selectMenu, listItemClicked, list) {         var $styledSelect = $(styledSelect);         var $selectMenu = $(selectMenu);         var $listItem = $(listItemClicked);         var $list = $(list);          event.stopPropagation();         var selectedText = $listItem.text();         selectItemFromStyledList($styledSelect, $selectMenu, selectedText, $list)  }       // clickListItem 

The Fiddle illustrating this is here — http://jsfiddle.net/cwzjL2uw/1/ . The issue is, although I have achieved the style, I have failed to replicate the keyboard behavior that a normal select menu has. My question is, how can I make my menu behave such that when I click the letter “A,” the first “A” item is selected (in this case “Alabama”), just like a regular select menu behaves.

4 Answers

Answers 1

You could use jQuery UI's selectmenu widget, and style it to your liking. Since it already includes keyboard handling, you would only have to worry about styling.

$(function () {    $('select').selectmenu();  });
.ui-selectmenu-button  {    background: gray !important;    padding: 0.4em 0.8em !important;  }  .ui-selectmenu-button *  {     font-family: Verdana;    font-size: 12px;    color: white !important;  }  .ui-menu-item  {     font-family: Verdana;    font-size: 12px;    background: gray;    color: white;  }  .ui-selectmenu-menu ul  {    max-height: 170px;  }  /* Using base64 version of white down arrow */  .ui-icon {    background-image: url('')!important;   }
<script src="https://ajax.googleapis.com/ajax/libs/jquery/1.11.1/jquery.min.js"></script>  <link rel="stylesheet" href="https://ajax.googleapis.com/ajax/libs/jqueryui/1.12.1/themes/smoothness/jquery-ui.css">  <script src="https://ajax.googleapis.com/ajax/libs/jqueryui/1.12.1/jquery-ui.min.js"></script>  <select name="user[address_attributes][state_id]" id="user_address_attributes_state_id">    <option value="">Select State</option>    <option value="3526">Alabama</option>    <option value="3556">Alaska</option>    <option value="3547">Arizona</option>    <option value="3510">Arkansas</option>    <option value="3542">California</option>    <option value="3543">Colorado</option>    <option value="3527">Connecticut</option>    <option value="3512">Delaware</option>    <option value="3513">Florida</option>    <option value="3514">Georgia</option>    <option value="3555">Hawaii</option>    <option value="3548">Idaho</option>    <option value="3529">Illinois</option>    <option value="3530">Indiana</option>    <option value="3528">Iowa</option>    <option value="3515">Kansas</option>    <option value="3557">Kentucky</option>    <option value="3516">Louisiana</option>    <option value="3531">Maine</option>    <option value="3517">Maryland</option>    <option value="3558">Massachusetts</option>    <option value="3532">Michigan</option>    <option value="3533">Minnesota</option>    <option value="3519">Mississippi</option>    <option value="3518">Missouri</option>    <option value="3549">Montana</option>    <option value="3534">Nebraska</option>    <option value="3545">Nevada</option>    <option value="3535">New Hampshire</option>    <option value="3536">New Jersey</option>    <option value="3544">New Mexico</option>    <option value="3537">New York</option>    <option value="3520">North Carolina</option>    <option value="3550">North Dakota</option>    <option value="3538">Ohio</option>    <option value="3521">Oklahoma</option>    <option value="3551">Oregon</option>    <option value="3559">Pennsylvania</option>    <option value="3539">Rhode Island</option>    <option value="3522">South Carolina</option>    <option value="3552">South Dakota</option>    <option value="3523">Tennessee</option>    <option value="3524">Texas</option>    <option value="3546">Utah</option>    <option value="3540">Vermont</option>    <option value="3560">Virginia</option>    <option value="3553">Washington</option>    <option value="3511">Washington, D.C.</option>    <option value="3525">West Virginia</option>    <option value="3541">Wisconsin</option>    <option value="3554">Wyoming</option>  </select>

Answers 2

Here is an example of how you can accomplish this using your code. By listening for keys on your active drop down and iterating through the options to find and scroll to a match.

function styleSelectMenu(selectMenu) {         .......      $(document).click(function(event) {         $styledSelect.removeClass('active');         $list.hide();     });       //Example      var keyUps = "", timeOut, $selectOptions = $('.select-options');      $(document).keyup(function(event){       if(!$selectOptions.prev().hasClass('active')){         return false;       }       if(event.key){       keyUps += event.key;       retrieveFromOptions($selectOptions,keyUps);     }     clearTimeout(timeOut);     timeOut = setTimeout(function(){       keyUps = "";     },250);     .......  });  function retrieveFromOptions($options,val){    $options.find('li').each(function(index){     if(this.textContent.substring(0,val.length).toLowerCase() === val.toLowerCase()){         $options.scrollTop(43*index);         return false;     }    }); } 

Fiddle

Notice that this solution does not implement a selection effect equivalent to hovering over the items. That could be done but would require more work.

I would recommend using a solution such as ConnorsFan or one where only css is used to style the select element without replacing it. That way you will retain native functionality.

There are many CSS frameworks which implement the styling of the select element.

Bootstrap is an example.

Answers 3

This feature is not trivial to implement, and involves a few steps:

  1. When the user open the dropdown, attach and event to the document itself listening for keypress events
  2. When the event is fired, get the character from the event and loop over your list looking for the first element that starts with that character
  3. get the position of the list item relative to its container, and scroll the container to that position
  4. you should also check for other relevant keys like arrows up/down, spacebar etc.
  5. remove the listener from the body when the user close the drowdown

Hope I didn't miss anything

Answers 4

You can change your code to this:

$(function() {      $('select').each(function() {      styleSelectMenu($(this));    });  });    // This method applies the styles to our select menu  function styleSelectMenu(selectMenu) {    var $this = $(selectMenu),      numberOfOptions = $(selectMenu).children('option').length;      /*** NEW - start ***/    var $paddingCalculator = $('<div />', {      'class': "select-styled test"    }).css({      width: 0,      visibility: "hidden"    }).appendTo("body");    $this.addClass('select-hidden');    var paddingWidth = $paddingCalculator.outerWidth() + 10;    $paddingCalculator.remove();      //var selectWidth = $this.width() + paddingWidth;    var selectWidth = $this.outerWidth() + paddingWidth;    //alert(selectWidth);      if (!$this.parent().hasClass('select')) {      var $wrapper = $("<div />", {        'class': "select"      }).css({        width: selectWidth      });      $this.wrap($wrapper);    } // if      /*** NEW - end ***/      if (!$this.next().hasClass('select-styled')) {      $this.after('<div class="select-styled"></div>');    } // if      var $styledSelect = $this.next('div.select-styled');    $styledSelect.text($this.children('option').eq(0).text());      if ($styledSelect.parent().find('ul').length > 0) {      $styledSelect.parent().find('ul').remove();    } // if    var $list = $('<ul />', {      'class': 'select-options'    }).insertAfter($styledSelect);      for (var i = 0; i < numberOfOptions; i++) {      $('<li />', {        text: $this.children('option').eq(i).text(),        rel: $this.children('option').eq(i).val()      }).appendTo($list);    }      var $listItems = $list.children('li');      // This is the event when someone opens the list    $styledSelect.unbind('click')    $styledSelect.click(function(e) {      e.stopPropagation();      $('div.select-styled.active').each(function() {        $(this).removeClass('active').next('ul.select-options').hide();      });      $(this).toggleClass('active').next('ul.select-options').toggle();    });      // This is the event when someone actually selects something from the list    $listItems.unbind('click.selectStyledItem')    $listItems.bind('click.selectStyledItem', function(event) {      clickListItem(event, $styledSelect, $this, $(this), $list);    });      $(document).click(function(event) {      $styledSelect.removeClass('active');      $list.hide();    });        var selectedIndex = $this[0].selectedIndex;    if (selectedIndex > 0) {      var name = $this.attr("name")      var selectedText = $("select[name='" + name + "'] option:selected").text();      selectItemFromStyledList($styledSelect, $this, selectedText, $list);    } // if    }    // This is the method that will select an item from the styled list  function selectItemFromStyledList(styledSelect, selectMenu, selectedText, listToHide) {      $(styledSelect).text(selectedText).removeClass('active');      $(selectMenu).val($(selectMenu).attr('rel'));      $(listToHide).hide();      // Select option in the underlying list so that the form gets submitted      // with the right values      selectedOption = $(selectMenu).find("option").filter(function() {        return $(this).html() == selectedText;      });      $(selectMenu).find("option[selected='selected']").removeAttr("selected");      $(selectedOption).attr("selected", "selected");    } // selectItemFromStyledList    // Called when someone clicks an item from the styled list  // The event data should contain the following parameters:  //      styledSelect - the <div> element that contains the styled menu  //      selectMenu - the actual form element that contains the items  //      listItemClicked - the item that was clicked.  //      list - THe <UL> element containig the <li> options  function clickListItem(event, styledSelect, selectMenu, listItemClicked, list) {      var $styledSelect = $(styledSelect);      var $selectMenu = $(selectMenu);      var $listItem = $(listItemClicked);      var $list = $(list);        event.stopPropagation();      var selectedText = $listItem.text();      selectItemFromStyledList($styledSelect, $selectMenu, selectedText, $list)    } // clickListItem    /* New Code */  // Prepare variable to get the entered text  var text = '';    // Handle keydown  jQuery(document).on("keypress", function(e) {          // Only handle event if the menu is open    if (jQuery(".select-styled").hasClass("active")) {      if (e.which != 8) {        var letter = String.fromCharCode(e.which);          // If backspace is pressed        text = text + letter;            jQuery(document).trigger("updateSelect");      }      return false;    }  });    jQuery(document).on("keydown", function(e) {    // Only handle event if the menu is open    if (jQuery(".select-styled").hasClass("active") && e.keyCode == 8) {      text = text.substring(0, text.length - 1);          jQuery(document).trigger("updateSelect");        // If there is no match show original text      if (text.length == 0)        jQuery(".select-styled").text("Select State");      return false;    }  });    jQuery(document).on("updateSelect", function() {      jQuery(".select-styled").text(text);        // Hide all elements    jQuery(".select-options li").hide();      // Show only matching elements    var matchingElements = jQuery(".select-options li:icontains('" + text + "')");    matchingElements.show();      return false;  });    // Add case insitive contains expression  $.expr[":"].icontains = $.expr.createPseudo(function(arg) {    return function(elem) {      return $(elem).text().toUpperCase().indexOf(arg.toUpperCase()) == 0;    };  });
/* line 63, /Users/davea/Documents/workspace/runtrax/app/assets/stylesheets/profile.css.scss */    .selectMenu {    font-family: 'Oxygen', sans-serif;    font-size: 20px;    height: 50px;    -webkit-appearance: menulist-button;  }  /* line 70, /Users/davea/Documents/workspace/runtrax/app/assets/stylesheets/profile.css.scss */    .select-hidden {    display: none;    visibility: hidden;    padding-right: 10px;  }  /* line 76, /Users/davea/Documents/workspace/runtrax/app/assets/stylesheets/profile.css.scss */    .select {    cursor: pointer;    display: inline-block;    position: relative;    font-size: 16px;    color: #fff;    width: 220px;    height: 42px;  }  /* line 85, /Users/davea/Documents/workspace/runtrax/app/assets/stylesheets/profile.css.scss */    .select-styled {    position: absolute;    top: 0;    right: 0;    bottom: 0;    left: 0;    background-color: gray;    padding: 11px 12px;    -webkit-transition: all 0.2s ease-in;    transition: all 0.2s ease-in;  }  /* line 94, /Users/davea/Documents/workspace/runtrax/app/assets/stylesheets/profile.css.scss */    .select-styled:after {    content: "";    width: 0;    height: 0;    border: 7px solid transparent;    border-color: #fff transparent transparent transparent;    position: absolute;    top: 16px;    right: 10px;  }  /* line 104, /Users/davea/Documents/workspace/runtrax/app/assets/stylesheets/profile.css.scss */    .select-styled:hover {    background-color: #7b7b7b;  }  /* line 107, /Users/davea/Documents/workspace/runtrax/app/assets/stylesheets/profile.css.scss */    .select-styled:active,  .select-styled.active {    background-color: #737373;  }  /* line 109, /Users/davea/Documents/workspace/runtrax/app/assets/stylesheets/profile.css.scss */    .select-styled:active:after,  .select-styled.active:after {    top: 9px;    border-color: transparent transparent #fff transparent;  }  /* line 116, /Users/davea/Documents/workspace/runtrax/app/assets/stylesheets/profile.css.scss */    .select-options {    display: none;    position: absolute;    top: 100%;    right: 0;    left: 0;    z-index: 999;    margin: 0;    padding: 0;    list-style: none;    background-color: #737373;    overflow: scroll;  }  /* line 128, /Users/davea/Documents/workspace/runtrax/app/assets/stylesheets/profile.css.scss */    .select-options li {    margin: 0;    padding: 12px 0;    text-indent: 15px;    border-top: 1px solid #676767;    -webkit-transition: all 0.15s ease-in;    transition: all 0.15s ease-in;  }  /* line 134, /Users/davea/Documents/workspace/runtrax/app/assets/stylesheets/profile.css.scss */    .select-options li:hover {    color: gray;    background: #fff;  }  /* line 138, /Users/davea/Documents/workspace/runtrax/app/assets/stylesheets/profile.css.scss */    .select-options li[rel="hide"] {    display: none;  }  /* line 144, /Users/davea/Documents/workspace/runtrax/app/assets/stylesheets/profile.css.scss */    ul.select-options {    max-height: 15em;    overflow-y: scroll;    overflow-x: hidden;  }
<script src="https://ajax.googleapis.com/ajax/libs/jquery/2.1.1/jquery.min.js"></script>  <select class="selectField selectMenu form-control" name="user[address_attributes][state_id]" id="user_address_attributes_state_id">    <option value="">Select State</option>    <option value="3526">Alabama</option>    <option value="3556">Alaska</option>    <option value="3547">Arizona</option>    <option value="3510">Arkansas</option>    <option value="3542">California</option>    <option value="3543">Colorado</option>    <option value="3527">Connecticut</option>    <option value="3512">Delaware</option>    <option value="3513">Florida</option>    <option value="3514">Georgia</option>    <option value="3555">Hawaii</option>    <option value="3548">Idaho</option>    <option value="3529">Illinois</option>    <option value="3530">Indiana</option>    <option value="3528">Iowa</option>    <option value="3515">Kansas</option>    <option value="3557">Kentucky</option>    <option value="3516">Louisiana</option>    <option value="3531">Maine</option>    <option value="3517">Maryland</option>    <option value="3558">Massachusetts</option>    <option value="3532">Michigan</option>    <option value="3533">Minnesota</option>    <option value="3519">Mississippi</option>    <option value="3518">Missouri</option>    <option value="3549">Montana</option>    <option value="3534">Nebraska</option>    <option value="3545">Nevada</option>    <option value="3535">New Hampshire</option>    <option value="3536">New Jersey</option>    <option value="3544">New Mexico</option>    <option value="3537">New York</option>    <option value="3520">North Carolina</option>    <option value="3550">North Dakota</option>    <option value="3538">Ohio</option>    <option value="3521">Oklahoma</option>    <option value="3551">Oregon</option>    <option value="3559">Pennsylvania</option>    <option value="3539">Rhode Island</option>    <option value="3522">South Carolina</option>    <option value="3552">South Dakota</option>    <option value="3523">Tennessee</option>    <option value="3524">Texas</option>    <option value="3546">Utah</option>    <option value="3540">Vermont</option>    <option value="3560">Virginia</option>    <option value="3553">Washington</option>    <option value="3511">Washington, D.C.</option>    <option value="3525">West Virginia</option>    <option value="3541">Wisconsin</option>    <option value="3554">Wyoming</option>  </select>

And here is a JSFiddle: http://jsfiddle.net/noevLzno/2/

If You Enjoyed This, Take 5 Seconds To Share It

0 comments:

Post a Comment