Pregunta

¿Existe una forma exclusiva de CSS para diseñar un <select> ¿desplegable?

Necesito diseñar un <select> forme tanto como sea humanamente posible, sin ningún JavaScript.¿Cuáles son las propiedades que puedo usar para hacerlo en CSS?

Este código debe ser compatible con todos los navegadores principales:

  • Internet Explorer 6, 7 y 8
  • Firefox
  • Safari

Sé que puedo hacerlo con JavaScript: Ejemplo.

Y no me refiero a un estilo simple.Quiero saber qué es lo mejor que podemos hacer solo con CSS.

encontré preguntas similares en Stack Overflow.

Y Éste en Doctype.com.

¿Fue útil?

Solución

Aquí hay tres soluciones:

Solución # 1 - Aspecto: ninguno - con Internet Explorer 10 - 11 Solución ( Demostración )

-

Para ocultar el conjunto predeterminado flecha appearance: none en el elemento de selección, a continuación, añadir su flecha a medida con background-image

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

Soporte para el navegador:

appearance: none tiene muy buen soporte de los navegadores ( caniuse ) - a excepción de Internet Explorer 11 ( y más tarde) y Firefox 34 (y más adelante).

Podemos mejorar esta técnica y añadir soporte para Internet Explorer 10 y 11 de Internet Explorer añadiendo

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

Si Internet Explorer 9 es una preocupación, no tenemos ninguna manera de eliminar la flecha por defecto (lo que significaría que ahora tendríamos dos flechas), pero, podríamos utilizar un selector cobarde Internet Explorer 9.

Para deshacer al menos nuestra flecha costumbre -. Dejando intacta la flecha de selección por defecto

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

Todos juntos:

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>

Esta solución es fácil y tiene un buen soporte del navegador -. Bastaría general


Si el apoyo del navegador Internet Explorer 9 (y posteriores) y Firefox 34 (y más adelante) es necesario entonces sigue leyendo ...

Solución # 2 truncar el elemento de selección para ocultar la flecha por defecto ( demostración )

-

(Lea más aquí)

Envolver el elemento select en un div con un ancho fijo y overflow:hidden.

A continuación, dar al elemento select una anchura de unos 20 píxeles mayor que el div .

El resultado es que la flecha desplegable por defecto del elemento select se ocultará (debido a la overflow:hidden en el envase), y se puede colocar cualquier imagen de fondo que desee en la mano del lado derecho de la div.

La ventaja de este enfoque es que es multi-navegador (Internet Explorer 8 y versiones posteriores, WebKit y Gecko ). Sin embargo, la desventaja de este enfoque es que las opciones desplegables se adentra en la mano del lado derecho (por los 20 píxeles que nos escondimos ... porque los elementos de opciones tienen el ancho de la seleccione elemento).

Enter descripción de la imagen aquí

[Cabe señalar, sin embargo, que si el elemento personalizado de selección es necesario sólo para móviles dispositivos - a continuación, el problema anterior no se aplica - debido a la forma de cada teléfono se abre de forma nativa el selecto elemento. Así que para móvil, esta puede ser la mejor solución.]

.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>


Si la flecha a medida es necesaria en Firefox - antes de la Versión 35 -, pero no es necesario para apoyar las versiones antiguas de Internet Explorer - entonces sigue leyendo ...

Solución # 3 - Utilice la propiedad pointer-events ( demostración )

-

(Lea más aquí)

La idea aquí es superponer un elemento más de la caída nativa flecha hacia abajo (para crear nuestra costumbre uno) y luego no permitir eventos de puntero en él.

Ventaja: Funciona bien en WebKit y Gecko. Se ve bien también (hay que sobresale elementos option).

Desventaja: Internet Explorer (Internet Explorer 10 y hacia abajo) no es compatible con pointer-events, lo que significa que no puede hacer clic en la flecha personalizado. Además, otra desventaja (obvio) con este método es que no se puede orientar su imagen Flecha de nuevo con un efecto de vuelo estacionario o cursor de la mano, porque tenemos eventos de puntero acaba con discapacidad en ellos!

Sin embargo, con este método se puede utilizar renovador o comentarios condicionales para que Internet Explorer volver a la norma construida en flecha.

Nota: Ser que Internet Explorer 10 no soporta más conditional comments: Si desea utilizar este enfoque, probablemente debería utilizar Modernizr . Sin embargo, todavía es posible excluir el puntero CSS-eventos de Internet Explorer 10 con un hack CSS descrito aquí .

.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]-->

Otros consejos

Es posible, pero desafortunadamente principalmente en navegadores basados ​​en WebKit en la medida en que nosotros, como desarrolladores, lo necesitemos.Este es el ejemplo de estilo CSS recopilado del panel de opciones de Chrome a través del inspector de herramientas de desarrollador integrado, mejorado para que coincida con las propiedades CSS actualmente admitidas en la mayoría de los navegadores modernos:

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

Cuando ejecuta este código en cualquier página dentro de un navegador basado en WebKit, debería cambiar la apariencia del cuadro de selección, eliminar la flecha estándar del sistema operativo y agregar una flecha PNG, poner algo de espacio antes y después de la etiqueta, casi cualquier cosa que desee.

La parte más importante es appearance propiedad, que cambia el comportamiento del elemento.

Funciona perfectamente en casi todos los navegadores basados ​​en WebKit, incluidos los móviles, aunque Gecko no es compatible. appearance además de WebKit, al parecer.

El elemento de selección y su función desplegable son difícil de peinar.

para selecto elemento por Chris Heilmann confirma lo que dijo Ryan Dohery en un comentario a la primera respuesta:

  

"El elemento de selección es parte de la   sistema operativo, no el navegador Chrome. Por lo tanto, es muy   poco fiable para el estilo, y no necesariamente tiene sentido tratar   de todos modos ".

La inconsistencia más grande que he notado durante el peinado selectos menús desplegables es Safari y Google Chrome renderizado (Firefox es totalmente personalizable a través de CSS). Después de buscar a través de oscuras profundidades de Internet me encontré con el siguiente, que resuelve casi completamente mis reparos con WebKit:

Safari y Google Chrome solución

select {
  -webkit-appearance: none;
}

Esto, sin embargo, quitar la flecha desplegable. Puede añadir una flecha de la derecha usando un div cerca, con un fondo, el margen negativo o con posición absoluta sobre el selecto desplegable.

* Más información y otras variables están disponibles en propiedad CSS: -webkit-apariencia .

etiquetas <select> pueden ser de estilo a través de CSS al igual que cualquier otro elemento HTML en una página HTML representado en un navegador. A continuación se muestra un ejemplo (demasiado simple) que posicionará a un elemento de selección de la página y hacer que el texto de las opciones en azul.

Ejemplo de un archivo 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>

Ejemplo de un archivo 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;
}

Tenía este problema exacto, excepto que no podía usar imágenes y no estaba limitado por el apoyo del navegador. Esto debería ser «por casualidad» y con suerte empezar a trabajar en todas partes finalmente .

Se utiliza capas de fondo en capas girada a «cortar» la flecha de la derecha, como pseudo-elementos no funcionarían para el elemento de selección.

Editar:. En esta versión actualizada estoy usando variables de CSS y un pequeño sistema de tematización

: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>

Aquí hay una versión que funciona en todos los navegadores modernos. La clave es usar appearance:none que elimina el formato predeterminado. Dado que todo el formato se ha ido, hay que añadir de nuevo en la flecha que se diferencia visualmente el selectivo de la entrada.

Ejemplo de trabajo: 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>

La entrada de blog Cómo forma CSS desplegable estilo sin JavaScript funciona para mí, pero falla en Opera sin embargo:

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>

llegué a su caso usando Bootstrap . Esta es la solución más simple que funciona:

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>

Nota:. El material base 64 es fa-chevron-down en SVG

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

En los navegadores modernos que es relativamente indolora con el estilo de la <select> en el CSS. Con appearance: none la única parte difícil está reemplazando la flecha (si eso es lo que quiere). He aquí una solución que utiliza una línea data: URI con texto sin formato 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>

El resto del peinado (bordes, rellenos, colores, etc.) es bastante sencillo.

Esto funciona en todos los navegadores que acaba de intentar (Firefox 50, Chrome 55, 38 de borde, y Safari 10). Una nota acerca de Firefox es que si desea utilizar el carácter # en el URI de datos (por ejemplo fill: #000) que necesita para escapar de ella (fill: %23000).

Utilice la propiedad clip para recortar los bordes y la flecha del elemento select, a continuación, añadir sus propios estilos de reemplazo a la envoltura:

    <!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>

Use un segundo SELECT con cero opacidad pueda hacer clic en el botón:

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

Las coordenadas difieren entre Webkit y otros navegadores, pero un @media consulta que puede cubrir.

Referencias

Un muy buen ejemplo que usa :after y :before para hacer el truco está en Seleccionar Styling Caja con CSS3 | CSSDeck

Sí. Es posible que el estilo de cualquier elemento HTML por su nombre de la etiqueta, así:

select {
  font-weight: bold;
}

Por supuesto, también puede utilizar una clase CSS de estilo que, como cualquier otro elemento:

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

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

Editar No se recomienda este elemento, pero si quieres probar que es como cualquier otro elemento HTML.

Editar ejemplo:

/* 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>

sin duda debería hacerlo como en Styling selecto, y optgroup opciones con CSS . En muchos sentidos, color de fondo y el color son justo lo que normalmente se necesitan opciones de estilo, no toda la selección.

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

Esto utiliza un color de fondo para los elementos seleccionados y elimina la imagen ..

A partir de Internet Explorer 10, puede utilizar el ::-ms-expand selector de pseudo-elemento de estilo, y esconderse, el menú desplegable elemento de flecha.

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

El estilo restante debe ser similar a otros navegadores.

Aquí es un tenedor básica de jsFiddle de DanielD que se aplica apoyo a IE10

Esto es una solución basada en mis ideas preferidas de esta discusión. Esto permite estilizar un elemento

El contenido tiene licencia bajo Creative Commons.

Si encuentra violaciones de derechos de autor, puede contactarnos en info@generacodice.com Para solicitar la eliminación del contenido.

scroll top