如何使用CSS样式化<select>下拉菜单?

1638

有没有一种仅使用CSS的方法来样式化元素,而不使用任何JavaScript。在CSS中,有哪些属性可以用来实现这个需求?

这段代码需要兼容所有主流浏览器:

  • Internet Explorer 6、7和8
  • Firefox
  • Safari

我知道可以使用JavaScript实现:示例

但是我并不是要简单的样式设置。我想知道如何在仅使用CSS的情况下最大限度地进行样式定制。

在Stack Overflow上,我找到了类似的问题

还有在Doctype.com上找到了这个


74
我认为这是一个合理的问题,但答案是“不太”或“不是你想要的方式”。但是没有人(包括我)对此完全确定,这种模糊感在读者心中潜伏,进而质疑问题的合理性。 - ZJR
1
@Jitendra,我知道你的意思。如果您能更明确地提出问题,我们会很高兴。此外,我认为我找到了您可能正在寻找的内容。这是实验性的,但请查看:http://cappuccino.org/aristo/showcase/ - jeremyosborne
1
@jeremyosborne - 感谢您的回复。我知道我可以用JavaScript做到这一点。您提供的示例链接是基于JS的。我之所以问这个问题,是因为我想知道,有没有人知道我们只能使用CSS做什么最好的事情。 - Jitendra Vyas
1
@Jitendra 感谢您更新问题。在您的限制条件下(仅使用CSS和无JS),您可以可靠地进行以下修改:字体,背景和前景(文本)颜色,边框大小,外观和颜色,位置以及大小(通常通过字体设置)。即使如此,您可能还需要进行一些微调,以确保所有浏览器中的显示效果相同。我希望我知道比这更好的答案,也许我错过了一个,但我不认为有。 - jeremyosborne
但是没有人(包括我)对此有100%的把握,这就是为什么这是一个很好的问题!我们应该确信像这样的事情。它们是基础。 - Stijn de Witt
25个回答

1214

以下是三种解决方案:

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

--

要隐藏默认箭头,请在选择元素上设置appearance: none,然后使用background-image添加自定义箭头。

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

浏览器支持:

appearance: none 在大多数浏览器中有很好的支持(caniuse),但不包括Internet Explorer。

我们可以通过添加以下代码来改善这种技术并支持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(https://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,请继续阅读。
解决方案#2 将选择元素截断以隐藏默认箭头(demo

--

(点击此处阅读更多)

select 元素包裹在一个带有固定宽度和 overflow:hidden 样式的 div 中。

然后给 select 元素设置一个比 div 宽约 20 像素 的宽度。

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

这种方法的优点是它跨浏览器兼容(Internet Explorer 8及以上版本,WebKitGecko)。但是,这种方法的缺点是选项下拉菜单会向右侧突出(因为选项元素采用了 select 元素的宽度)。

Enter image description here

需要注意的是,如果自定义选择元素仅适用于移动设备,则上述问题不适用,因为每个手机本地打开选择元素的方式不同。因此,对于移动设备,这可能是最佳解决方案。

.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(https://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>


If the custom arrow is necessary on Firefox - prior to Version 35 - but you don't need to support old versions of Internet Explorer - then keep reading...

Solution #3 - Use the pointer-events property (demo)

--

(Read more here)

The idea here is to overlay an element over the native drop down arrow (to create our custom one) and then disallow pointer events on it.

Advantage: It works well in WebKit and Gecko. It looks good too (no jutting out option elements).

Disadvantage: Internet Explorer (Internet Explorer 10 and down) doesn't support pointer-events, which means you can't click the custom arrow. Also, another (obvious) disadvantage with this method is that you can't target your new arrow image with a hover effect or hand cursor, because we have just disabled pointer events on them!

However, with this method you can use Modernizer or conditional comments to make Internet Explorer revert to the standard built in arrow.

NB: Being that Internet Explorer 10 doesn't support conditional comments anymore: If you want to use this approach, you should probably use Modernizr. However, it is still possible to exclude the pointer-events CSS from Internet Explorer 10 with a CSS hack described here.

.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(https://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]-->


IE已经崩溃,所以可能需要进行重构? - Shane Muirhead

259

这是可能的,但不幸的是大多数情况下只有基于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-arrow 并添加一个 PNG-arrow,在标签前后添加一些间距,几乎可以实现任何你想要的效果。

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

这段代码在几乎所有基于 WebKit 的浏览器中都能完美地工作,包括移动端,但 Gecko 似乎不像 WebKit 那样支持 appearance


73

我遇到了与此完全相同的问题,但我无法使用图像,并且没有受到浏览器支持的限制。这应该是“在规范范围内”,并且有希望最终能够在所有地方正常工作。

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

编辑: 在此更新版本中,我正在使用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>


你好@Henrik,对于粉色的那个,我该如何使主要部分(即许多选项部分)的线性渐变颜色改变?我希望那个大部分也有多种颜色,就像这样: background-image: linear-gradient(to right, #a1c4fd 0%, #c2e9fb 51%, #a1c4fd 100%); } - r_e

72

选择元素及其下拉功能很难进行样式设计。

选择元素的样式属性由Chris Heilmann确认了Ryan Dohery在对第一个答案的评论中所说的话:

"选择元素是操作系统的一部分,而不是浏览器的界面。因此,尝试进行样式设计是非常不可靠的,而且这样做未必有意义。"


46

在样式化下拉框时,我注意到的最大不一致性是 SafariGoogle Chrome 的渲染(Firefox 可以通过 CSS 进行完全自定义)。在互联网上深入搜索后,我发现以下内容几乎完全解决了我的 WebKit 疑虑:

Safari 和 Google Chrome 修复方法

select {
  -webkit-appearance: none;
}

然而,这样做会移除下拉箭头。您可以使用附近带有背景、负边距或绝对定位于选择下拉框上方的 div 添加下拉箭头。

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


36

<select>标签可以像浏览器中渲染的其他HTML元素一样通过CSS进行样式设置。以下是一个(过于简单的)示例,将在页面上定位


33
很棒,我第一次看到“母亲们”在互联网上出现,并没有粗鄙的语言。对此给您点赞! - BaL

26

自定义选择器CSS样式

Internet Explorer(10和11)、Edge、Firefox和Chrome 中经过测试

select::-ms-expand {
  display: none;
}
select {
  display: inline-block;
  box-sizing: border-box;
  padding: 0.5em 2em 0.5em 0.5em;
  border: 1px solid #eee;
  font: inherit;
  line-height: inherit;
  -webkit-appearance: none;
  -moz-appearance: none;
  -ms-appearance: none;
  appearance: none;
  background-repeat: no-repeat;
  background-image: linear-gradient(45deg, transparent 50%, currentColor 50%), linear-gradient(135deg, currentColor 50%, transparent 50%);
  background-position: right 15px top 1em, right 10px top 1em;
  background-size: 5px 5px, 5px 5px;
}
<select name="">
  <option value="">Lorem</option>
  <option value="">Lorem Ipsum</option>
</select>

<select name="" disabled>
  <option value="">Disabled</option>
</select>

<select name="" style="color:red;">
  <option value="">Color!</option>
  <option value="">Lorem Ipsum</option>
</select>


25

以下是适用于所有现代浏览器的代码版本。关键是使用 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>


1
我喜欢这样的自包含答案,它们可以立即使用!谢谢。 - undefined

20

如何使用CSS样式表创建无需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>


20
我使用 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

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接