如何仅使用 CSS 设置 <select> 下拉列表的样式?

是否有仅 CSS 方式设置<select>下拉菜单的样式?

我需要在不使用任何 JavaScript 的情况下,尽可能地设置<select>表单的样式。我可以在 CSS 中使用哪些属性?

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

  • Internet Explorer 6、7 和 8
  • 火狐浏览器
  • 苹果浏览器

我知道我可以使用 JavaScript 实现: Example

我不是在谈论简单的样式。我想知道,仅 CSS 可以做得最好。

我在堆栈溢出中发现了类似的问题

这个在 Doctype.com 上。

答案

这是三个解决方案:

解决方案#1 - 外观:无 - 使用 Internet Explorer 10-11 解决方法( 演示

-

要隐藏默认的箭头集appearance: none select 元素上appearance: none ,然后添加带有background-image的自定义箭头

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

浏览器支持:

appearance: none除 Internet Explorer 11(及更高版本)和 Firefox 34(及更高版本)外, appearance: none具有很好的浏览器支持( caniuse )。

我们可以改进此技术,并通过添加以下内容来添加对 Internet Explorer 10 和 Internet Explorer 11 的支持:

select::-ms-expand {
    display: none; /* Hide the default arrow in Internet Explorer 10 and Internet Explorer 11 */
}

如果担心 Internet Explorer 9,则无法删除默认箭头(这意味着我们现在将有两个箭头),但是,我们可以使用时髦的 Internet Explorer 9 选择器。

至少撤消我们的自定义箭头 - 保留默认的选择箭头不变。

/* 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: Internet Explorer hackery ahead */


select::-ms-expand {
    display: none; /* Remove default arrow in Internet Explorer 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>

该解决方案很简单,并且具有良好的浏览器支持 - 通常应该足够。


如果需要浏览器支持 Internet Explorer 9(及更高版本)和 Firefox 34(及更高版本),请继续阅读...

解决方案#2 截断选择元素以隐藏默认箭头( demo

-

(在这里阅读更多)

select元素包装在具有固定宽度的 div 中,并overflow:hidden

然后,为select元素提供一个比 div大约20 个像素的宽度。

结果是, select元素的默认下拉箭头将被隐藏(由于在容器中的overflow:hidden ),并且您可以将所需的任何背景图像放置在 div 的右侧。

这种方法的优点是它是跨浏览器(Internet Explorer 8 和更高版本, WebKitGecko )。但是,这种方法的缺点是,选项下拉菜单在右侧突出(被我们隐藏的 20 个像素...,因为选项元素采用选择元素的宽度)。

在此处输入图片说明

[然而,应该注意的是,如果自定义选择元素仅对于移动设备是必需的,那么上述问题就不适用,因为每部电话都会以本机方式打开选择元素。因此,对于移动设备,这可能是最好的解决方案。]

.styled select {
  background: transparent;
  width: 150px;
  font-size: 16px;
  border: 1px solid #CCC;
  height: 34px;
}
.styled {
  margin: 50px;
  width: 120px;
  height: 34px;
  border: 1px solid #111;
  border-radius: 3px;
  overflow: hidden;
  background: url(http://www.stackoverflow.com/favicon.ico) 96% / 20% no-repeat #EEE;
}
<div class="styled">
  <select>
    <option>Pineapples</option>
    <option selected>Apples</option>
    <option>Chocklate</option>
    <option>Pancakes</option>
  </select>
</div>


如果在 Firefox( 版本 35之前)上需要自定义箭头,但您不需要支持旧版本的 Internet Explorer,则请继续阅读...

解决方案#3 - 使用pointer-events属性( demo

-

(在这里阅读更多)

这里的想法是在本地下拉箭头上覆盖一个元素(以创建我们的自定义箭头),然后在其上禁止指针事件。

优点:它在 WebKit 和 Gecko 中运行良好。看起来也不错(不突出option元素)。

缺点: Internet Explorer(Internet Explorer 10 及更低版本)不支持pointer-events ,这意味着您无法单击自定义箭头。另外,此方法的另一个(明显)缺点是您不能用悬停效果或手形光标作为新箭头图像的目标,因为我们刚刚禁用了它们的指针事件!

但是,通过这种方法,您可以使用 Modernizer 或条件注释使 Internet Explorer 恢复为内置的标准箭头。

注意: Internet Explorer 10 不再支持conditional comments :如果要使用此方法,则可能应该使用Modernizr 。但是,仍然可以通过此处描述的 CSS hack 从 Internet Explorer 10 中排除指针事件 CSS。

.notIE {
  position: relative;
  display: inline-block;
}
select {
  display: inline-block;
  height: 30px;
  width: 150px;
  outline: none;
  color: #74646E;
  border: 1px solid #C8BFC4;
  border-radius: 4px;
  box-shadow: inset 1px 1px 2px #DDD8DC;
  background: #FFF;
}
/* Select arrow styling */

.notIE .fancyArrow {
  width: 23px;
  height: 28px;
  position: absolute;
  display: inline-block;
  top: 1px;
  right: 3px;
  background: url(http://www.stackoverflow.com/favicon.ico) right / 90% no-repeat #FFF;
  pointer-events: none;
}
/*target Internet Explorer 9 and Internet Explorer 10:*/

@media screen and (min-width: 0\0) {
  .notIE .fancyArrow {
    display: none;
  }
}
<!--[if !IE]> -->
<div class="notIE">
  <!-- <![endif]-->
  <span class="fancyArrow"></span>
  <select>
    <option>Apples</option>
    <option selected>Pineapples</option>
    <option>Chocklate</option>
    <option>Pancakes</option>
  </select>
  <!--[if !IE]> -->
</div>
<!-- <![endif]-->

这是可能的,但不幸的是,大多数情况下,在开发人员要求的程度上基于 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 Firefox (3.6 to 15) */
    background-image: url(../images/select-arrow.png), -ms-linear-gradient(top, #E5E5E5, #F4F4F4); /* For pre-releases of Internet Explorer  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 似乎不像 WebKit 那样支持appearance

选择元素及其下拉功能难以风格。

克里斯 · 海尔曼(Chris Heilmann)的select 元素样式属性证实了 Ryan Dohery 在对第一个答案的评论中所说的话:

“select 元素是操作系统的一部分,而不是浏览器 chrome。因此,样式非常不可靠,无论如何尝试都不一定有意义。”

在样式化选择下拉列表时,我注意到的最大不一致是SafariGoogle Chrome渲染(Firefox 可以通过 CSS 完全自定义)。经过对 Internet 的晦涩深度的搜索后,我发现以下内容几乎可以完全解决 WebKit 的问题:

Safari 和 Google Chrome 浏览器修复

select {
  -webkit-appearance: none;
}

但是,这确实删除了下拉箭头。您可以使用附近的div添加下拉箭头,该div具有背景,负边距或绝对位于选择下拉列表上方。

* 更多信息和其他变量可在CSS 属性中找到:-webkit-appearance

我遇到了这个确切的问题,除了我不能使用图像并且不受浏览器支持的限制。这应该是 “按规格” 的,运气好的话最终可以在任何地方开始。

它使用分层的旋转背景层来 “切出” 下拉箭头,因为伪元素对 select 元素不起作用。

编辑:在此更新的版本中,我正在使用 CSS 变量和小型主题系统。

:root {
  --radius: 2px;
  --baseFg: dimgray;
  --baseBg: white;
  --accentFg: #006fc2;
  --accentBg: #bae1ff;
}

.theme-pink {
  --radius: 2em;
  --baseFg: #c70062;
  --baseBg: #ffe3f1;
  --accentFg: #c70062;
  --accentBg: #ffaad4;
}

.theme-construction {
  --radius: 0;
  --baseFg: white;
  --baseBg: black;
  --accentFg: black;
  --accentBg: orange;
}

select {
  font: 400 12px/1.3 sans-serif;
  -webkit-appearance: none;
  appearance: none;
  color: var(--baseFg);
  border: 1px solid var(--baseFg);
  line-height: 1;
  outline: 0;
  padding: 0.65em 2.5em 0.55em 0.75em;
  border-radius: var(--radius);
  background-color: var(--baseBg);
  background-image: linear-gradient(var(--baseFg), var(--baseFg)),
    linear-gradient(-135deg, transparent 50%, var(--accentBg) 50%),
    linear-gradient(-225deg, transparent 50%, var(--accentBg) 50%),
    linear-gradient(var(--accentBg) 42%, var(--accentFg) 42%);
  background-repeat: no-repeat, no-repeat, no-repeat, no-repeat;
  background-size: 1px 100%, 20px 22px, 20px 22px, 20px 100%;
  background-position: right 20px center, right bottom, right bottom, right bottom;   
}

select:hover {
  background-image: linear-gradient(var(--accentFg), var(--accentFg)),
    linear-gradient(-135deg, transparent 50%, var(--accentFg) 50%),
    linear-gradient(-225deg, transparent 50%, var(--accentFg) 50%),
    linear-gradient(var(--accentFg) 42%, var(--accentBg) 42%);
}

select:active {
  background-image: linear-gradient(var(--accentFg), var(--accentFg)),
    linear-gradient(-135deg, transparent 50%, var(--accentFg) 50%),
    linear-gradient(-225deg, transparent 50%, var(--accentFg) 50%),
    linear-gradient(var(--accentFg) 42%, var(--accentBg) 42%);
  color: var(--accentBg);
  border-color: var(--accentFg);
  background-color: var(--accentFg);
}
<select>
  <option>So many options</option>
  <option>...</option>
</select>

<select class="theme-pink">
  <option>So many options</option>
  <option>...</option>
</select>

<select class="theme-construction">
  <option>So many options</option>
  <option>...</option>
</select>

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

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;
}

博客文章How to CSS form drop down style 没有 JavaScript对我有用,但是在Opera 中却失败了:

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>

这是适用于所有现代浏览器的版本。关键是使用appearance:none将删除默认格式。由于所有格式都不用了,因此您必须重新添加箭头,以可视方式将选择内容与输入内容区分开。

工作示例: https//jsfiddle.net/gs2q1c7p/

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的情况。这是最简单的解决方案:

select.form-control {
    -moz-appearance: none;
    -webkit-appearance: none;
    appearance: 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>

注意:base64 的东西在 SVG 中是fa-chevron-down的。

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

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>

其余样式(边框,填充,颜色等)非常简单。

这在我刚尝试过的所有浏览器(Firefox 50,Chrome 55,Edge 38 和 Safari 10)中均可使用。关于 Firefox 的一个注意事项是,如果要在数据 URI 中使用#字符(例如fill: #000 ),则需要对其进行转义( fill: %23000 )。