提问



是否有一种仅限CSS的方式来设置<select>下拉列表的样式?


我需要尽可能地设置<select>形式,而不需要任何JavaScript。我可以在CSS中使用哪些属性?


此代码需要与所有主流浏览器兼容:



  • Internet  Explorer 6,7和8

  • 火狐

  • Safari浏览器



我知道我可以使用JavaScript:示例。[53]


我并不是在谈论简单的样式。我想知道,只有CSS才能做到最好。


我在Stack  Overflow上发现了类似的问题。


这个在Doctype.com上。[55]

最佳参考


这里有3个解决方案:


解决方案#1 - 外观:无 - 使用ie10-11解决方法(演示)



要隐藏选择元素上的默认箭头集appearance: none,请使用background-image [56]添加自己的自定义箭头


select {
   -webkit-appearance: none; 
   -moz-appearance: none;
   appearance: none;       /* remove default arrow */
   background-image: url(...);   /* add custom arrow */
}


浏览器支持:


appearance: none有非常好的浏览器支持(caniuse) - 除了ie11-和firefox 34 - [57]


我们可以通过添加来改进这种技术并添加对ie10和ie11的支持


select::-ms-expand { 
    display: none; /* hide the default arrow in ie10 and ie11 */
}


如果ie9是一个问题 - 我们无法删除默认箭头(这意味着我们现在有两个箭头),但是,我们可以使用一个时髦的ie9选择器
至少撤消我们的自定义箭头 - 保持默认选择箭头不变。


/* target Internet Explorer 9 to undo the custom arrow */
@media screen and (min-width:0\0) {
    select {
        background-image:none\9;
        padding: 5px\9;
    } 
}


所有在一起:





select {
  margin: 50px;
  width: 150px;
  padding: 5px 35px 5px 5px;
  font-size: 16px;
  border: 1px solid #ccc;
  height: 34px;
  -webkit-appearance: none;
  -moz-appearance: none;
  appearance: none;
  background: url(http://www.stackoverflow.com/favicon.ico) 96% / 15% no-repeat #eee;
}


/* CAUTION: IE hackery ahead */


select::-ms-expand { 
    display: none; /* remove default arrow in IE 10 and 11 */
}

/* target Internet Explorer 9 to undo the custom arrow */
@media screen and (min-width:0\0) {
    select {
        background:none\9;
        padding: 5px\9;
    }
}

<select>
  <option>Apples</option>
  <option selected>Pineapples</option>
  <option>Chocklate</option>
  <option>Pancakes</option>
</select>



这是可能的,但不幸的是,在我们作为开发人员需要的程度上,主要是基于Webkit的浏览器。以下是通过内置开发人员工具检查器从Chrome选项面板收集的CSS样式示例,经过改进以匹配大多数现代浏览器中当前支持的CSS属性:


select {
    -webkit-appearance: button;
    -moz-appearance: button;
    -webkit-user-select: none;
    -moz-user-select: none;
    -webkit-padding-end: 20px;
    -moz-padding-end: 20px;
    -webkit-padding-start: 2px;
    -moz-padding-start: 2px;
    background-color: #F07575; /* fallback color if gradients are not supported */
    background-image: url(../images/select-arrow.png), -webkit-linear-gradient(top, #E5E5E5, #F4F4F4); /* For Chrome and Safari */
    background-image: url(../images/select-arrow.png), -moz-linear-gradient(top, #E5E5E5, #F4F4F4); /* For old Fx (3.6 to 15) */
    background-image: url(../images/select-arrow.png), -ms-linear-gradient(top, #E5E5E5, #F4F4F4); /* For pre-releases of IE 10*/
    background-image: url(../images/select-arrow.png), -o-linear-gradient(top, #E5E5E5, #F4F4F4); /* For old Opera (11.1 to 12.0) */ 
    background-image: url(../images/select-arrow.png), linear-gradient(to bottom, #E5E5E5, #F4F4F4); /* Standard syntax; must be last */
    background-position: center right;
    background-repeat: no-repeat;
    border: 1px solid #AAA;
    border-radius: 2px;
    box-shadow: 0px 1px 3px rgba(0, 0, 0, 0.1);
    color: #555;
    font-size: inherit;
    margin: 0;
    overflow: hidden;
    padding-top: 2px;
    padding-bottom: 2px;
    text-overflow: ellipsis;
    white-space: nowrap;
}


当您在基于Webkit的浏览器中的任何页面上运行此代码时,它应该更改选择框的外观,删除标准OS箭头并添加PNG箭头,在标签之前和之后放置一些间距,几乎任何您想要的。


最重要的部分是appearance属性,它改变了元素的行为方式。


它在几乎所有基于Webkit的浏览器中都能很好地工作,包括移动浏览器,尽管Gecko似乎并不支持appearance和Webkit。

其它参考1


select元素及其下拉功能 难以设置样式。


Chris Heilmann的选择元素的样式属性证实了Ryan Dohery在对第一个答案的评论中所说的内容:[58]



  选择元素是其中的一部分
  操作系统,而不是浏览器chrome。因此,非常
  风格不可靠,尝试不一定有意义
  无论如何。


其它参考2


我在设计选择下拉列表时注意到的最大不一致是 Safari 谷歌浏览器渲染(Firefox可以通过CSS完全自定义)。经过一些不透刻的互联网深度搜索我遇到了以下几点,几乎完全解决了我对WebKit的疑虑:[59]


Safari和Google Chrome修复:


select {
  -webkit-appearance: none;
}


但是,这会删除下拉箭头。您可以使用附近的div添加一个下拉箭头,背景为负边距或绝对位于选择下拉列表上方。


* CSS属性中提供了更多信息和其他变量:-webkit-appearance 。[60]

其它参考3


<select>标签可以通过CSS设置样式,就像在浏览器中呈现的HTML页面上的任何其他HTML元素一样。下面是一个(过于简单的)示例,它将在页面上定位一个选择元素,并以蓝色呈现选项的文本。


示例HTML文件(selectExample.html):


<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01//EN" "http://www.w3.org/TR/html4/strict.dtd">
<html>
<head>
  <title>Select Styling</title>
  <link href="selectExample.css" rel="stylesheet">
</head>
<body>
<select id="styledSelect" class="blueText">
  <option value="apple">Apple</option>
  <option value="orange">Orange</option>
  <option value="cherry">Cherry</option>
</select>
</body>
</html>


示例CSS文件(selectExample.css):


/* All select elements on page */
select {
  position: relative;
}

/* Style by class. Effects the text of the contained options. */
.blueText {
  color: #0000FF;
}

/* Style by id. Effects position of the select drop down. */
#styledSelect {
  left: 100px;
}

其它参考4


博客文章如何使用CSS形式下拉样式没有JavaScript 对我有用,但它在Opera中失败了:[61] [62]


select {
    border: 0 none;
    color: #FFFFFF;
    background: transparent;
    font-size: 20px;
    font-weight: bold;
    padding: 2px 10px;
    width: 378px;
    *width: 350px;
    *background: #58B14C;
}

#mainselection {
    overflow: hidden;
    width: 350px;
    -moz-border-radius: 9px 9px 9px 9px;
    -webkit-border-radius: 9px 9px 9px 9px;
    border-radius: 9px 9px 9px 9px;
    box-shadow: 1px 1px 11px #330033;
    background: url("arrow.gif") no-repeat scroll 319px 5px #58B14C;
}

<div id="mainselection">
    <select>
    <option>Select an Option</option>
    <option>Option 1</option>
    <option>Option 2</option>
    </select>
</div>

其它参考5


我有这个确切的问题,除了我不能使用图像并且不受浏览器支持的限制。这应该是«在规范上»并且运气开始在所有地方工作最终。


它使用分层旋转的背景图层来剪切下拉箭头,因为伪元素不适用于选择元素。用您喜欢的颜色替换«hotpink» - 我使用变量。




select {
  font: 400 12px/1.3 "Helvetica Neue", sans-serif;
  -webkit-appearance: none;
  appearance: none;
  border: 1px solid hotpink;
  line-height: 1;
  outline: 0;
  color: hotpink;
  border-color: hotpink;
  padding: 0.65em 2.5em 0.55em 0.75em;
  border-radius: 3px;
  background: linear-gradient(hotpink, hotpink) no-repeat,
              linear-gradient(-135deg, rgba(255,255,255,0) 50%, white 50%) no-repeat,
              linear-gradient(-225deg, rgba(255,255,255,0) 50%, white 50%) no-repeat,
              linear-gradient(hotpink, hotpink) no-repeat;
  background-color: white;
  background-size: 1px 100%, 20px 20px, 20px 20px, 20px 60%;
  background-position: right 20px center, right bottom, right bottom, right bottom;   
}

<select>
    <option>So many options</option>
    <option>...</option>
</select>



select  {
    outline: 0;
    overflow: hidden;
    height: 30px;
    background: #2c343c;
    color: #747a80;
    border: #2c343c;
    padding: 5px 3px 5px 10px;
    -moz-border-radius: 6px;
    -webkit-border-radius: 6px;
    border-radius: 10px;
}

select option {border: 1px solid #000; background: #010;}

其它参考6


如果样式是一个重要的问题,使用完全自定义的小部件可能会有所帮助,例如博客文章使用CSS和jQuery重新发送下拉中描述的那个。[63]

其它参考7


这是一个适用于所有现代浏览器的版本。关键是使用appearance:none删除默认格式。由于所有格式都消失了,您必须在箭头中添加回来,以便在视觉上区分选择和输入。


工作示例:https://jsfiddle.net/gs2q1c7p/[64]




select:not([multiple]) {
    -webkit-appearance: none;
    -moz-appearance: none;
    background-position: right 50%;
    background-repeat: no-repeat;
    background-image: url();
    padding: .5em;
    padding-right: 1.5em
}

#mySelect {
    border-radius: 0
}

<select id="mySelect">
    <option>Option 1</option>
    <option>Option 2</option>
</select>



我使用Bootstrap来处理你的情况。这是最简单的解决方案:[65]




select.form-control {
    -moz-appearance: none;
    -webkit-appearance: none;
    appearancce: none;
    background-position: right center;
    background-repeat: no-repeat;
    background-size: 1ex;
    background-origin: content-box;
    background-image: url("");
}

<link href="https://maxcdn.bootstrapcdn.com/bootstrap/3.3.6/css/bootstrap.min.css" rel="stylesheet" />
<section class="container">
  <form class="form-horizontal">
    <select class="form-control">
      <option>One</option>
      <option>Two</option>
    </select>
  </form>
</section>



使用clip属性裁剪边框和select元素的箭头,然后将自己的替换样式添加到包装器:




    <!DOCTYPE html>
    <html>
      <head>
        <style>
          select { position: absolute; clip:rect(2px 49px 19px 2px); z-index:2; }
          body > span { display:block; position: relative; width: 64px; height: 21px; border: 2px solid green;  background: url(http://www.stackoverflow.com/favicon.ico) right 1px no-repeat; }
        </style>
      </head>
      <span>
        <select>
          <option value="">Alpha</option>
          <option value="">Beta</option>
          <option value="">Charlie</option>
        </select>
      </span>
    </html>



在现代浏览器中,在CSS中设置<select>样式相对轻松。appearance: none唯一棘手的部分是替换箭头(如果那是你想要的)。这是一个使用带有纯文本SVG的内联data: URI的解决方案:




select {
  -moz-appearance: none;
  -webkit-appearance: none;
  appearance: none;
  
  background-repeat: no-repeat;
  background-size: 0.5em auto;
  background-position: right 0.25em center;
  padding-right: 1em;
  
  background-image: url("data:image/svg+xml;charset=utf-8, \
    <svg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 60 40'> \
      <polygon points='0,0 60,0 30,40' style='fill:black;'/> \
    </svg>");
}

<select>
  <option>Option 1</option>
  <option>Option 2</option>
</select>

<select style="font-size: 2rem;">
  <option>Option 1</option>
  <option>Option 2</option>
</select>



是。您可以按标签名称设置任何HTML元素的样式,如下所示:


select {
  font-weight: bold;
}


当然,您也可以使用CSS类来设置样式,就像任何其他元素一样:


<select class="important">
  <option>Important Option</option>
  <option>Another Important Option</option>
</select>

<style type="text/css">
  .important {
    font-weight: bold;
  }
</style>

其它参考8


不推荐编辑此元素,但如果您想尝试它,就像任何其他HTML元素一样。


编辑示例:


/*Edit select*/
select {
    /*css style here*/
}

/*Edit option*/
option {
    /*css style here*/
}

/*Edit selected option*/
/*element  attr    attr value*/
option[selected="selected"] {
    /*css style here*/
}

<select>
    <option >Something #1</option>
    <option selected="selected">Something #2</option>
    <option >Something #3</option>
</select>

其它参考9


使用:after:before来完成这个技巧的一个非常好的例子是使用CSS3的样式选择框CSSDeck [66]

其它参考10


你绝对应该像样式选择,optgroup和CSS 选项一样。在许多方面,背景颜色和颜色正是您通常需要的样式选项,而不是整个选择。[67]

其它参考11


label {
    position: relative;
    display: inline-block;
}
select {
    display: inline-block;
    padding: 4px 3px 5px 5px;
    width: 150px;
    outline: none;
    color: black;
    border: 1px solid #C8BFC4;
    border-radius: 4px;
    box-shadow: inset 1px 1px 2px #ddd8dc;
    background-color: lightblue;
}


这使用选择元素的背景颜色,我删除了图像..

其它参考12


从Internet Explorer 10开始,您可以使用::-ms-expand伪元素选择器来设置样式并隐藏下拉箭头元素。 [68]


select::-ms-expand {
    display:none;
    /* or visibility: hidden; to keep it's space/hitbox */
}


其余样式应与其他浏览器类似。


这是Danield的jsfiddle的基本分支,它支持IE10 [69]

其它参考13


这是基于我讨论中我最喜欢的想法的解决方案。这允许直接设置元素而无需任何额外的标记。


使用IE10 +为IE8/9安全后备。这些浏览器的一个警告是,背景图像必须定位并且足够小以隐藏在原生扩展控件之后。


HTML



<select name='options'>
  <option value='option-1'>Option 1</option>
  <option value='option-2'>Option 2</option>
  <option value='option-3'>Option 3</option>
</select>


SCSS



body {
  padding: 4em 40%;
  text-align: center;
}

select {
  $bg-color: lightcyan;
  $text-color: black;
  appearance: none; // using -prefix-free http://leaverou.github.io/prefixfree/
  background: {
    color: $bg-color;
    image: url("https://s3-us-west-2.amazonaws.com/s.cdpn.io/1255/caret--down-15.png");
    position: right;
    repeat: no-repeat;
  }
  border: {
    color: mix($bg-color, black, 80%);
    radius: .2em;
    style: solid;
    width: 1px;
    right-color: mix($bg-color, black, 60%);
    bottom-color: mix($bg-color, black, 60%);
  }
  color: $text-color;
  padding: .33em .5em;
  width: 100%;
}

// Removes default arrow for IE10+
// IE 8/9 get dafault arrow which covers caret image
// as long as caret image is small than and positioned
// behind default arrow
select::-ms-expand {
    display: none;
}


Codepen



http://codepen.io/ralgh/pen/gpgbGx[70]

其它参考14


CSS&仅HTML解决方案


我似乎与Chrome,Firefox和& IE11。但请留下您对其他网络浏览器的反馈。


正如@Danield回答所建议的那样,我将select包装在一个div中(甚至两个div用于x-browser兼容性)以获得预期的行为。


见http://jsfiddle.net/bjap2/[71]


HTML:


<div class="sort-options-wrapper">
    <div class="sort-options-wrapper-2">
        <select class="sort-options">
                <option value="choiceOne">choiceOne</option>
                <option value="choiceOne">choiceThree</option>
                <option value="choiceOne">choiceFour</option>
                <option value="choiceFiveLongTestPurpose">choiceFiveLongTestPurpose</option>
        </select>
    </div>
    <div class="search-select-arrow-down"></div>
</div>


注意2 div包装器。
还要注意添加额外的div以将箭头向下按钮放在任何你喜欢的位置(绝对定位),这里我们把它放在左边。


CSS


.sort-options-wrapper {
    display: inline-block;
    position: relative;
    border: 1px solid #83837f;
}
/* this second wrapper is needed for x-browser compatibility */
.sort-options-wrapper-2 {
    overflow: hidden;
}
select {
    margin-right: -19px; /* that's what hidding the default-provided browser arrow */
    padding-left: 13px;
    margin-left: 0;
    border: none;
    background: none;
    /* margin-top & margin-bottom must be set since some browser have default values for select elements */
    margin-bottom: 1px;
    margin-top: 1px;
}
select:focus {
    outline: none; /* removing default browsers outline on focus */
}
.search-select-arrow-down {
    position: absolute;
    height:10px;
    width: 12px;
    background: url(http://i.imgur.com/pHIYN06.png) scroll no-repeat 2px 0px;
    left: 1px;
    top: 5px;
}

其它参考15


Danield的答案(https://stackoverflow.com/a/13968900/280972)中的第二种方法可以改进,以处理悬停效果和其他鼠标事件。只需确保按钮元素紧随其后标记中的select元素。然后使用+ css-selector定位它:


HTML:


<select class="select-input">...</select>
<div class="select-button"></div>


CSS:


.select-input:hover+.select-button {
    [hover styles here]
}


但是,这将显示悬停在select元素上方的任何位置时的悬停效果,而不仅仅是按钮。


我将此方法与Angular结合使用(因为我的项目恰好是一个Angular-app),覆盖整个select元素,让Angular在button元素中显示所选选项的文本。在这种情况下,将悬停效果悬停在选择的任何地方时都是完全合理的。虽然没有javascript但它没有工作,所以如果你想这样做,你的网站必须在没有javascript的情况下工作,你应该确保您的脚本添加了增强所需的元素和类。这样,没有javascript的浏览器只会获得正常的,没有样式的选择,而不是没有正确更新的样式徽章。

其它参考16


有一种方法可以设置SELECT标签的样式。


如果标签中有sasize参数,那么几乎所有的CSS都会适用。使用这个技巧,我创造了一个实际上相当于普通选择标签的小提琴,加上该值可以像ComboBox一样手动编辑可视语言(除非您在输入标记中放入 readonly )。


所以这里是一个看到背后原理的最小例子:

(你需要点击机制的jQuery):


<style>

    /* only these 2 lines are truly required */
    .stylish span {position:relative;}
    .stylish select {position:absolute;left:0px;display:none}

    /* now you can style the hell out of them */
    .stylish input    { ... }
    .stylish select   { ... }
    .stylish option   { ... }
    .stylish optgroup { ... }

</style>
...
<div class="stylish">
    <label> Choose your superhero: </label>
    <span>
        <input onclick="$(this).closest('div').find('select').slideToggle(110)">
        <br>
        <select size=15 onclick="$(this).hide().closest('div').find('input').val($(this).find('option:selected').text());">

            <optgroup label="Fantasy"></optgroup>
            <option value="gandalf">Gandalf</option>
            <option value="harry">Harry Potter</option>
            <option value="jon">Jon Snow</option>

            <optgroup label="Comics"></optgroup>
            <option value="tony">Tony Stark</option>
            <option value="steve">Steven Rogers</option>
            <option value="natasha">Natasha Romanova</option>

        </select>
    </span>
</div>


这里有更多风格的小提琴:
https://jsfiddle.net/dkellner/7ac9us70/[73]


(当然,这只是为了证明其可能性。)


注意标签如何不像通常那样封装属于它们的选项;是的,这是有意的,它是造型的。 (有礼貌的方式可以减少风格。)是的,他们确实以这种方式完美地工作。


在任何人指出NO-JS部分之前:我知道问题是没有Javascript。对我来说,这更像是请不要打扰插件,我知道他们可以做到,但我需要原生的方式。理解,没有插件,没有额外的脚本,只有适合内部的标记sonclick。唯一的依赖是jQuery,以避免本机document.parentNode.getElementsByTagName的疯狂。但它可以这样工作。所以是的,这是一个带有原生样式和一些onclick处理程序的原生选择标记。它显然不是Javascript解决方案。


请享用!