Tabla HTML con encabezados fijos?


231

¿Existe una técnica CSS / JavaScript entre navegadores para mostrar una tabla HTML larga de modo que los encabezados de las columnas permanezcan fijos en la pantalla y no se desplace con el cuerpo de la tabla? Piense en el efecto de "congelar paneles" en Microsoft Excel.

Quiero poder desplazarme por el contenido de la tabla, pero siempre poder ver los encabezados de columna en la parte superior.


3
Pruebe esto: Tabla desplazable de CSS puro con encabezado fijo EDITAR : Esta debería funcionar en Internet Explorer 7 como se ve en el ejemplo : Desplazar tabla HTML con encabezado fijo EDITAR 2: Encontré un par de enlaces adicionales que podrían ser útiles: - Estúpido encabezado fijo : un complemento de jQuery con algunas limitaciones. - [Encabezados de tabla fijos] ( cross-browser.com/x/examp
gcores

Me he encontrado con muchas soluciones que generalmente funcionan, pero ninguna de ellas funcionaba con el desplazamiento de div. Quiero decir, su tabla está dentro de un div desplazable y aún desea que el encabezado de su tabla siga dentro de ese div. Lo he resuelto y comparto la solución aquí .
Yogee

99
En 2018, todos los navegadores pueden usar la siguiente solución sencilla: thead th { position: sticky; top: 0; }. Safari necesita un prefijo de proveedor:-webkit-sticky
Daniel Waltrip

1
@DanielWaltrip, debe agregarle una respuesta para que pueda ser votado al primer puesto: todas las demás respuestas son redundantes con la posición: pegajoso es un mejor soporte hoy en día
Peter Kerr

Respuestas:


88

Estuve buscando una solución para esto por un tiempo y descubrí que la mayoría de las respuestas no funcionan o no son adecuadas para mi situación, así que escribí una solución simple con jQuery.

Este es el esquema de la solución:

  1. Clone la tabla que necesita tener un encabezado fijo y coloque la copia clonada encima del original.
  2. Retire el cuerpo de la mesa de la mesa superior.
  3. Retire el encabezado de la tabla de la tabla inferior.
  4. Ajusta los anchos de columna. (Hacemos un seguimiento de los anchos de columna originales)

A continuación se muestra el código en una demostración ejecutable.

function scrolify(tblAsJQueryObject, height) {
  var oTbl = tblAsJQueryObject;

  // for very large tables you can remove the four lines below
  // and wrap the table with <div> in the mark-up and assign
  // height and overflow property  
  var oTblDiv = $("<div/>");
  oTblDiv.css('height', height);
  oTblDiv.css('overflow', 'scroll');
  oTbl.wrap(oTblDiv);

  // save original width
  oTbl.attr("data-item-original-width", oTbl.width());
  oTbl.find('thead tr td').each(function() {
    $(this).attr("data-item-original-width", $(this).width());
  });
  oTbl.find('tbody tr:eq(0) td').each(function() {
    $(this).attr("data-item-original-width", $(this).width());
  });


  // clone the original table
  var newTbl = oTbl.clone();

  // remove table header from original table
  oTbl.find('thead tr').remove();
  // remove table body from new table
  newTbl.find('tbody tr').remove();

  oTbl.parent().parent().prepend(newTbl);
  newTbl.wrap("<div/>");

  // replace ORIGINAL COLUMN width				
  newTbl.width(newTbl.attr('data-item-original-width'));
  newTbl.find('thead tr td').each(function() {
    $(this).width($(this).attr("data-item-original-width"));
  });
  oTbl.width(oTbl.attr('data-item-original-width'));
  oTbl.find('tbody tr:eq(0) td').each(function() {
    $(this).width($(this).attr("data-item-original-width"));
  });
}

$(document).ready(function() {
  scrolify($('#tblNeedsScrolling'), 160); // 160 is height
});
<script src="https://ajax.googleapis.com/ajax/libs/jquery/1.6.4/jquery.min.js"></script>

<div style="width:300px;border:6px green solid;">
  <table border="1" width="100%" id="tblNeedsScrolling">
    <thead>
      <tr><th>Header 1</th><th>Header 2</th></tr>
    </thead>
    <tbody>
      <tr><td>row 1, cell 1</td><td>row 1, cell 2</td></tr>
      <tr><td>row 2, cell 1</td><td>row 2, cell 2</td></tr>
      <tr><td>row 3, cell 1</td><td>row 3, cell 2</td></tr>
      <tr><td>row 4, cell 1</td><td>row 4, cell 2</td></tr>			
      <tr><td>row 5, cell 1</td><td>row 5, cell 2</td></tr>
      <tr><td>row 6, cell 1</td><td>row 6, cell 2</td></tr>
      <tr><td>row 7, cell 1</td><td>row 7, cell 2</td></tr>
      <tr><td>row 8, cell 1</td><td>row 8, cell 2</td></tr>			
    </tbody>
  </table>
</div>

Esta solución funciona en Chrome e IE. Como se basa en jQuery, esto también debería funcionar en otros navegadores compatibles con jQuery.


44
¿Y cómo podemos resolver el problema cuando el contenido es mayor que el ancho?
Maertz

1
@tetra td {ancho máximo: 30px; } esto le permitirá al desarrollador controlar cómo se muestran las filas.
Lyuben Todorov

Pero, ¿qué pasa si los contenidos en alguna celda de encabezado son más largos que en las celdas td? Lo intenté en IE7, y width () rompe todo. Sin embargo, IE8 e IE9 funcionan bien ...
JustAMartin

44
Desafortunadamente, si necesita una alineación perfecta de píxeles de las columnas, esto no funciona: jsbin.com/elekiq/1 ( código fuente ). Puede ver que algunos encabezados están desplazados de donde deberían estar, solo un poco. El efecto es más obvio si usa fondos: jsbin.com/elekiq/2 ( código fuente ). (Estaba trabajando en esta misma línea, me encontré con esto en mi código, encontré el tuyo y pensé "¡Oh, me pregunto si me lo resolvió!" Lamentablemente no. :-)) Los navegadores son TAN dolorosos por querer controlar el anchuras de celdas ...
TJ Crowder

Esto no parece funcionar con desplazamiento horizontal: crea el encabezado, pero se extiende más allá del área desplazable (visiblemente) y no se desplaza con el contenido.
Choque el

183

Esto se puede resolver limpiamente en cuatro líneas de código.

Si solo le interesan los navegadores modernos, un encabezado fijo se puede lograr mucho más fácilmente mediante el uso de transformaciones CSS. Suena extraño, pero funciona muy bien:

  • HTML y CSS se mantienen como están.
  • Sin dependencias externas de JavaScript.
  • Cuatro líneas de código.
  • Funciona para todas las configuraciones (diseño de tabla: fijo, etc.).
document.getElementById("wrap").addEventListener("scroll", function(){
   var translate = "translate(0,"+this.scrollTop+"px)";
   this.querySelector("thead").style.transform = translate;
});

El soporte para transformaciones CSS está ampliamente disponible, excepto para Internet Explorer 8-.

Aquí está el ejemplo completo para referencia:


8
Debo decir que, a pesar de mi comentario anterior, esto es lo más parecido a una solución perfecta que he visto. Incluso el desplazamiento horizontal es perfecto (mejor que mi propia solución). Aquí hay un ejemplo con bordes (no puede usar el colapso del borde) y una barra de desplazamiento que se adhiere a la mesa en lugar del contenedor: jsfiddle
DoctorDestructo

11
Descubierto, funciona pero tiene que aplicar la transformación a th / td, no thead.
pelirroja

55
@AlexAlexeev, su solución es increíble. Gracias. Noté que, el encabezado fijo resultante no tiene las líneas de borde que distinguen las columnas. Se pierde el estilo CSS predeterminado. Incluso cuando incluyo esto ... $(this).addClass('border')cambia el resto de las características de la tabla fuentes, tamaño, color que paso en la clase de borde. Pero, no agrega líneas al encabezado fijo. Aprecio, cualquier entrada sobre cómo solucionar esto
user5249203

55
@ user5249203 Sé que preguntaste hace unos meses, pero tuve el mismo problema y se debió al colapso del borde: mira esto: stackoverflow.com/questions/33777751/… .
archz

66
Esto no funciona en ninguna versión de IE o en Edge. Aquí está una versión que basa en comentario de @ pelirroja jsfiddle.net/n6o8ocwb/2
Rob

58

Acabo de terminar de armar un complemento jQuery que tomará una tabla única válida usando HTML válido (debe tener un thead y tbody) y generará una tabla que tenga encabezados fijos, pie de página fijo opcional que puede ser un encabezado clonado o cualquier contenido que eligió (paginación, etc.). Si desea aprovechar los monitores más grandes, también cambiará el tamaño de la tabla cuando se cambie el tamaño del navegador. Otra característica adicional es poder desplazarse lateralmente si las columnas de la tabla no pueden caber todas a la vista.

http://fixedheadertable.com/

en github: http://markmalek.github.com/Fixed-Header-Table/

Es extremadamente fácil de configurar y puede crear sus propios estilos personalizados. También utiliza esquinas redondeadas en todos los navegadores. Tenga en cuenta que acabo de lanzarlo, por lo que todavía es técnicamente beta y hay muy pocos problemas menores que estoy solucionando.

Funciona en Internet Explorer 7, Internet Explorer 8, Safari, Firefox y Chrome.


¡Gracias! Agregaré una nueva versión más tarde hoy cuando llegue a casa del trabajo. Aquí hay un enlace a mi entrada de blog con lo que estoy agregando: fixedheadertable.mmalek.com/2009/10/07/…
Mark

Gracias por esto. Sé que esta pregunta tiene más de un año, pero incluso a riesgo de agitar el sedimento asentado, me gustaría decirle que su trabajo es apreciado
Sova

En su demostración, los anchos están desactivados en ie6 :-( el encabezado y el cuerpo de la tabla no están alineados.
Cheekysoft

44
La última versión no funciona en IE6. Ya no soporto IE6.
Mark

gran trabajo Mark: desafortunadamente, hay algunos problemas con el desplazamiento del encabezado fijo y la columna en dispositivos móviles (iPad, tableta Android), cuando desplazo el contenido esas partes fijas no se desplazan, cuando dejo de desplazarme y toco una vez la tabla , las partes fijas "saltan" a las posiciones adecuadas, ¿hay alguna manera simple de arreglar esto?
Okizb

23

También creé un complemento que soluciona este problema. Mi proyecto: jQuery.floatThead ha existido durante más de 4 años y es muy maduro.

No requiere estilos externos y no espera que su tabla tenga un estilo particular. Es compatible con Internet Explorer9 + y Firefox / Chrome.

Actualmente (2018-05) tiene:

405 commits y 998 estrellas en GitHub


Muchas (no todas) de las respuestas aquí son trucos rápidos que pueden haber resuelto el problema que tenía una persona, pero no funcionarán para todas las mesas.

Algunos de los otros complementos son antiguos y probablemente funcionen muy bien con Internet Explorer, pero se romperán en Firefox y Chrome.


1
Gran complemento, admite tablas anidadas y compensaciones.
Mihai Alex

2
Excelente. Muchas gracias. El complemento funcionó bien en Firefox 45.2, Chromium 51 e IE 11. Además, no interfiere con una gran cantidad de código JS y jQuery integrado en la misma página.
Aldo Paradiso

Gracias. Me complace informar que el proyecto obtiene alrededor de 1 informe de error cada 4 meses en este momento. No estoy haciendo muchos cambios importantes. Es bastante sólido y funciona.
mkoryak

20

TL; DR

Si apuntas a los navegadores modernos y no tienes necesidades de estilo extravagantes: http://jsfiddle.net/dPixie/byB9d/3/ ... Aunque la versión grande cuatro también es bastante dulce, esta versión maneja el ancho del fluido mucho mejor.

¡Buenas noticias para todos!

Con los avances de HTML5 y CSS3 esto ahora es posible, al menos para los navegadores modernos. La implementación ligeramente hackeada que se me ocurrió se puede encontrar aquí: http://jsfiddle.net/dPixie/byB9d/3/ . Lo he probado en FX 25, Chrome 31 e IE 10 ...

HTML relevante (inserte un doctype HTML5 en la parte superior de su documento):

html,
body {
  margin: 0;
  padding: 0;
  height: 100%;
}

section {
  position: relative;
  border: 1px solid #000;
  padding-top: 37px;
  background: #500;
}

section.positioned {
  position: absolute;
  top: 100px;
  left: 100px;
  width: 800px;
  box-shadow: 0 0 15px #333;
}

.container {
  overflow-y: auto;
  height: 200px;
}

table {
  border-spacing: 0;
  width: 100%;
}

td+td {
  border-left: 1px solid #eee;
}

td,
th {
  border-bottom: 1px solid #eee;
  background: #ddd;
  color: #000;
  padding: 10px 25px;
}

th {
  height: 0;
  line-height: 0;
  padding-top: 0;
  padding-bottom: 0;
  color: transparent;
  border: none;
  white-space: nowrap;
}

th div {
  position: absolute;
  background: transparent;
  color: #fff;
  padding: 9px 25px;
  top: 0;
  margin-left: -25px;
  line-height: normal;
  border-left: 1px solid #800;
}

th:first-child div {
  border: none;
}
<section class="positioned">
  <div class="container">
    <table>
      <thead>
        <tr class="header">
          <th>
            Table attribute name
            <div>Table attribute name</div>
          </th>
          <th>
            Value
            <div>Value</div>
          </th>
          <th>
            Description
            <div>Description</div>
          </th>
        </tr>
      </thead>
      <tbody>
        <tr>
          <td>align</td>
          <td>left, center, right</td>
          <td>Not supported in HTML5. Deprecated in HTML 4.01. Specifies the alignment of a table according to surrounding text</td>
        </tr>
        <tr>
          <td>bgcolor</td>
          <td>rgb(x,x,x), #xxxxxx, colorname</td>
          <td>Not supported in HTML5. Deprecated in HTML 4.01. Specifies the background color for a table</td>
        </tr>
        <tr>
          <td>border</td>
          <td>1,""</td>
          <td>Specifies whether the table cells should have borders or not</td>
        </tr>
        <tr>
          <td>cellpadding</td>
          <td>pixels</td>
          <td>Not supported in HTML5. Specifies the space between the cell wall and the cell content</td>
        </tr>
        <tr>
          <td>cellspacing</td>
          <td>pixels</td>
          <td>Not supported in HTML5. Specifies the space between cells</td>
        </tr>
        <tr>
          <td>frame</td>
          <td>void, above, below, hsides, lhs, rhs, vsides, box, border</td>
          <td>Not supported in HTML5. Specifies which parts of the outside borders that should be visible</td>
        </tr>
        <tr>
          <td>rules</td>
          <td>none, groups, rows, cols, all</td>
          <td>Not supported in HTML5. Specifies which parts of the inside borders that should be visible</td>
        </tr>
        <tr>
          <td>summary</td>
          <td>text</td>
          <td>Not supported in HTML5. Specifies a summary of the content of a table</td>
        </tr>
        <tr>
          <td>width</td>
          <td>pixels, %</td>
          <td>Not supported in HTML5. Specifies the width of a table</td>
        </tr>
      </tbody>
    </table>
  </div>
</section>

¡¿Pero cómo?!

En pocas palabras, tiene un encabezado de tabla, que oculta visualmente haciendo 0px de alto, que también contiene divs utilizados como encabezado fijo. El contenedor de la tabla deja suficiente espacio en la parte superior para permitir el encabezado absolutamente posicionado, y la tabla con barras de desplazamiento aparece como cabría esperar.

El código anterior usa la clase posicionada para posicionar la tabla absolutamente (la estoy usando en un cuadro de diálogo de estilo emergente) pero también puede usarla en el flujo del documento eliminando la positionedclase del contenedor.

Pero ...

No es perfecto Firefox se niega a hacer que la fila del encabezado sea de 0px (al menos no encontré ninguna forma) pero obstinadamente la mantiene en un mínimo de 4px ... No es un gran problema, pero dependiendo de su estilo se enredará con sus bordes, etc.

La tabla también está utilizando un enfoque de columna falsa donde el color de fondo del contenedor en sí se utiliza como fondo para los divs de encabezado, que son transparentes.

Resumen

En general, puede haber problemas de estilo según sus requisitos, especialmente bordes o fondos complicados. También puede haber problemas con la computabilidad, aún no lo he verificado en una amplia variedad de navegadores (comente con sus experiencias si lo prueba), pero no encontré nada parecido, así que pensé que valía la pena publicarlo de todas formas ...


Si reduce el ancho de la ventana hasta que se activa el desplazamiento horizontal, el encabezado no se desplaza horizontalmente con el cuerpo. Maldito.
dlaliberte

@dlaliberte - Bueno, ya que el encabezado y la tabla son en realidad dos elementos diferentes que, por supuesto, puedes encontrar en la rareza. Pero mi ejemplo no permite el desbordamiento en las columnas de la tabla y los encabezados suelen ser más fáciles de controlar que el contenido de la tabla. Dicho esto, si hace que el encabezado se "desborde", sobresaldrá a la derecha de la tabla y se verá severamente roto. Puede solucionar esto estableciendo un ancho mínimo en la mesa, forzándolo a desbordar la página también ... Pero es un truco, por lo que nunca será perfecto ...
Jonas Schubert Erlandsson

1
Vale la pena señalar que esto requiere un diseño donde se pueda especificar una tabla de altura fija.
Cheekysoft

1
@Cheekysoft: no, la tabla y el contenido de la fila pueden fluir libremente. El contenedor, en mi ejemplo el <section>elemento, debe tener una altura limitada solo para forzarlo a desbordarse y mostrar el desplazamiento. Cualquier diseño que haga que el contenedor se desborde funcionaría. Si encuentra un caso en el que no aparece, publique un enlace a un violín.
Jonas Schubert Erlandsson

El padding-topvalor codificado también significa que si el texto del encabezado de la tabla está en más de una línea, aparecerá en la parte superior de las celdas de la tabla. Lástima, porque esto funciona como un encanto la mayor parte del tiempo. Muy buen truco con el diven el thde moverse por el tema columna de dimensionamiento mayoría de las otras soluciones tienen.
Bernhard Hofmann

19

Todos los intentos de resolver esto desde fuera de la especificación CSS son sombras pálidas de lo que realmente queremos: cumplir con la promesa implícita de THEAD.

Este problema de encabezados congelados para una tabla ha sido una herida abierta en HTML / CSS durante mucho tiempo.

En un mundo perfecto, habría una solución de CSS puro para este problema. Desafortunadamente, no parece haber una buena en su lugar.

Las discusiones sobre estándares relevantes sobre este tema incluyen:

ACTUALIZACIÓN : Firefox se envió position:stickyen la versión 32. ¡Todos ganan!


Sería genial tener columnas zorro de la misma manera
Csaba Toth

44
Re. Firefox y posición: pegajoso, no funciona para encabezados de tabla: bugzilla.mozilla.org/show_bug.cgi?id=925259#c8 ... El parche para ese error declara explícitamente: "Actualmente no admitimos posicionamiento relativo de elementos internos de la tabla, por lo que también los excluimos del posicionamiento fijo ".
Jonas Schubert Erlandsson

2
Esto funciona en todos los navegadores ahora: thead th { position: sticky; top: 0; }. ¿Podemos actualizar esta respuesta para indicar esto claramente?
Daniel Waltrip

1
@DanielWaltrip todos los navegadores? stackoverflow.com/a/37646284/3640407 Todavía hay más MSIE que bordes
edc65

Punto justo. Es compatible con el 86% de los usuarios web globales, de acuerdo con caniuse.com/#search=position%3Asticky
Daniel Waltrip

14

Aquí hay un complemento jQuery para encabezados de tabla fijos. Permite que toda la página se desplace, congelando el encabezado cuando llega a la parte superior. Funciona bien con las tablas de Bootstrap de Twitter .

Repositorio de GitHub: https://github.com/oma/table-fixed-header

No , no desplazarse únicamente contenido de la tabla. Busque otras herramientas para eso, como una de estas otras respuestas. Usted decide qué se adapta mejor a su caso.


1
Bummer: el enlace de ejemplo está muerto. "¡Vaya! Denne siden ble ikke funnet ..." Ojalá el código se haya pegado aquí.
JosephK

Sí ... lo siento por eso. eliminado el enlace El proyecto ya no se mantiene
oma

No se preocupe: probé varias de estas supuestas soluciones prefabricadas; ninguna funcionó con una tabla de ancho de columna flexible que excediera el ancho de pantalla. Terminé escribiendo mi propia solución.
JosephK

9

La mayoría de las soluciones publicadas aquí requieren jQuery. Si está buscando una solución independiente del marco, pruebe Grid: http://www.matts411.com/post/grid/

Está alojado en Github aquí: https://github.com/mmurph211/Grid

No solo admite encabezados fijos, también admite columnas y pies de página fijos a la izquierda, entre otras cosas.


Esto es realmente genial si satisface tus necesidades, solo jugué con él hoy. Desafortunadamente, es más bien una cuadrícula rectangular (como su nombre lo indica, en realidad) y no una tabla verdadera con altura de fila ajustada por contenido. Y diseñar filas individuales parecía difícil. No pude crear una tabla de rayas de cebra, pero no me esforcé mucho porque mis necesidades eran en realidad más complejas. De todos modos, buen trabajo.
mplwork

1
¡Oye, te conozco! Parecíamos haber escrito una mierda muy similar ( github.com/mkoryak/floatThead ) - Misha
mkoryak

9

La propiedad CSS position: stickytiene un gran soporte en la mayoría de los navegadores modernos (tuve problemas con Edge, ver más abajo).

Esto nos permite resolver el problema de los encabezados fijos con bastante facilidad:

thead th { position: sticky; top: 0; }

Safari necesita un prefijo de proveedor: -webkit-sticky.

Para Firefox, tuve que agregar min-height: 0a uno los elementos principales. Olvidé exactamente por qué esto era necesario.

Lamentablemente, la implementación de Microsoft Edge parece ser solo semi-funcional. Al menos, tuve algunas celdas de tabla parpadeantes y desalineadas en mis pruebas. La mesa todavía era utilizable, pero tenía importantes problemas estéticos.


Usando position: sticky;la tabla dentro de un div que tiene overflow: scroll;, overflow-x: scroll;o overflow-y: scroll;. parece ser la mejor y más simple solución para encabezados de tabla fijos y columnas en navegadores modernos. Esta respuesta necesita ser votada a la cima.
Aberrante

Es simple pero efectivo. Es lo que estoy buscando. Gracias.
Catbuilts

6

Una tabla de desplazamiento CSS pura más refinada

Todas las soluciones CSS puras que he visto hasta ahora, por inteligentes que sean, carecen de cierto nivel de pulido o simplemente no funcionan correctamente en algunas situaciones. Entonces, decidí crear el mío ...

caracteristicas:

  • Es puro CSS, por lo que no se requiere jQuery (o cualquier código JavaScript, para el caso)
  • Puede establecer el ancho de la tabla en un porcentaje (también conocido como "fluido") o un valor fijo, o dejar que el contenido determine su ancho (también conocido como "automático")
  • Los anchos de columna también pueden ser fluidos, fijos o automáticos.
  • Las columnas nunca se desalinearán con los encabezados debido al desplazamiento horizontal (un problema que ocurre en todas las demás soluciones basadas en CSS que he visto que no requieren anchos fijos).
  • Compatible con todos los navegadores de escritorio populares, incluido Internet Explorer hasta la versión 8
  • Aspecto limpio y pulido; sin espacios de 1 píxel de aspecto descuidado o bordes desalineados; se ve igual en todos los navegadores

Aquí hay un par de violines que muestran las opciones de ancho automático y fluido:

  • Ancho y altura del fluido (se adapta al tamaño de la pantalla): jsFiddle ( tenga en cuenta que la barra de desplazamiento solo aparece cuando es necesario en esta configuración, por lo que es posible que tenga que reducir el marco para verlo)

  • Ancho automático, altura fija (más fácil de integrar con otro contenido): jsFiddle

La configuración de Ancho automático, Altura fija probablemente tenga más casos de uso, por lo que publicaré el código a continuación.

/* The following 'html' and 'body' rule sets are required only
   if using a % width or height*/

/*html {
  width: 100%;
  height: 100%;
}*/

body {
  box-sizing: border-box;
  width: 100%;
  height: 100%;
  margin: 0;
  padding: 0 20px 0 20px;
  text-align: center;
}
.scrollingtable {
  box-sizing: border-box;
  display: inline-block;
  vertical-align: middle;
  overflow: hidden;
  width: auto; /* If you want a fixed width, set it here, else set to auto */
  min-width: 0/*100%*/; /* If you want a % width, set it here, else set to 0 */
  height: 188px/*100%*/; /* Set table height here; can be fixed value or % */
  min-height: 0/*104px*/; /* If using % height, make this large enough to fit scrollbar arrows + caption + thead */
  font-family: Verdana, Tahoma, sans-serif;
  font-size: 16px;
  line-height: 20px;
  padding: 20px 0 20px 0; /* Need enough padding to make room for caption */
  text-align: left;
  color: black;
}
.scrollingtable * {box-sizing: border-box;}
.scrollingtable > div {
  position: relative;
  border-top: 1px solid black;
  height: 100%;
  padding-top: 20px; /* This determines column header height */
}
.scrollingtable > div:before {
  top: 0;
  background: cornflowerblue; /* Header row background color */
}
.scrollingtable > div:before,
.scrollingtable > div > div:after {
  content: "";
  position: absolute;
  z-index: -1;
  width: 100%;
  height: 100%;
  left: 0;
}
.scrollingtable > div > div {
  min-height: 0/*43px*/; /* If using % height, make this large
                            enough to fit scrollbar arrows */
  max-height: 100%;
  overflow: scroll/*auto*/; /* Set to auto if using fixed
                               or % width; else scroll */
  overflow-x: hidden;
  border: 1px solid black; /* Border around table body */
}
.scrollingtable > div > div:after {background: white;} /* Match page background color */
.scrollingtable > div > div > table {
  width: 100%;
  border-spacing: 0;
  margin-top: -20px; /* Inverse of column header height */
  /*margin-right: 17px;*/ /* Uncomment if using % width */
}
.scrollingtable > div > div > table > caption {
  position: absolute;
  top: -20px; /*inverse of caption height*/
  margin-top: -1px; /*inverse of border-width*/
  width: 100%;
  font-weight: bold;
  text-align: center;
}
.scrollingtable > div > div > table > * > tr > * {padding: 0;}
.scrollingtable > div > div > table > thead {
  vertical-align: bottom;
  white-space: nowrap;
  text-align: center;
}
.scrollingtable > div > div > table > thead > tr > * > div {
  display: inline-block;
  padding: 0 6px 0 6px; /*header cell padding*/
}
.scrollingtable > div > div > table > thead > tr > :first-child:before {
  content: "";
  position: absolute;
  top: 0;
  left: 0;
  height: 20px; /*match column header height*/
  border-left: 1px solid black; /*leftmost header border*/
}
.scrollingtable > div > div > table > thead > tr > * > div[label]:before,
.scrollingtable > div > div > table > thead > tr > * > div > div:first-child,
.scrollingtable > div > div > table > thead > tr > * + :before {
  position: absolute;
  top: 0;
  white-space: pre-wrap;
  color: white; /*header row font color*/
}
.scrollingtable > div > div > table > thead > tr > * > div[label]:before,
.scrollingtable > div > div > table > thead > tr > * > div[label]:after {content: attr(label);}
.scrollingtable > div > div > table > thead > tr > * + :before {
  content: "";
  display: block;
  min-height: 20px; /* Match column header height */
  padding-top: 1px;
  border-left: 1px solid black; /* Borders between header cells */
}
.scrollingtable .scrollbarhead {float: right;}
.scrollingtable .scrollbarhead:before {
  position: absolute;
  width: 100px;
  top: -1px; /* Inverse border-width */
  background: white; /* Match page background color */
}
.scrollingtable > div > div > table > tbody > tr:after {
  content: "";
  display: table-cell;
  position: relative;
  padding: 0;
  border-top: 1px solid black;
  top: -1px; /* Inverse of border width */
}
.scrollingtable > div > div > table > tbody {vertical-align: top;}
.scrollingtable > div > div > table > tbody > tr {background: white;}
.scrollingtable > div > div > table > tbody > tr > * {
  border-bottom: 1px solid black;
  padding: 0 6px 0 6px;
  height: 20px; /* Match column header height */
}
.scrollingtable > div > div > table > tbody:last-of-type > tr:last-child > * {border-bottom: none;}
.scrollingtable > div > div > table > tbody > tr:nth-child(even) {background: gainsboro;} /* Alternate row color */
.scrollingtable > div > div > table > tbody > tr > * + * {border-left: 1px solid black;} /* Borders between body cells */
<div class="scrollingtable">
  <div>
    <div>
      <table>
        <caption>Top Caption</caption>
        <thead>
          <tr>
            <th><div label="Column 1"/></th>
            <th><div label="Column 2"/></th>
            <th><div label="Column 3"/></th>
            <th>
              <!-- More versatile way of doing column label; requires two identical copies of label -->
              <div><div>Column 4</div><div>Column 4</div></div>
            </th>
            <th class="scrollbarhead"/> <!-- ALWAYS ADD THIS EXTRA CELL AT END OF HEADER ROW -->
          </tr>
        </thead>
        <tbody>
          <tr><td>Lorem ipsum</td><td>Dolor</td><td>Sit</td><td>Amet consectetur</td></tr>
          <tr><td>Lorem ipsum</td><td>Dolor</td><td>Sit</td><td>Amet consectetur</td></tr>
          <tr><td>Lorem ipsum</td><td>Dolor</td><td>Sit</td><td>Amet consectetur</td></tr>
          <tr><td>Lorem ipsum</td><td>Dolor</td><td>Sit</td><td>Amet consectetur</td></tr>
          <tr><td>Lorem ipsum</td><td>Dolor</td><td>Sit</td><td>Amet consectetur</td></tr>
          <tr><td>Lorem ipsum</td><td>Dolor</td><td>Sit</td><td>Amet consectetur</td></tr>
          <tr><td>Lorem ipsum</td><td>Dolor</td><td>Sit</td><td>Amet consectetur</td></tr>
          <tr><td>Lorem ipsum</td><td>Dolor</td><td>Sit</td><td>Amet consectetur</td></tr>
          <tr><td>Lorem ipsum</td><td>Dolor</td><td>Sit</td><td>Amet consectetur</td></tr>
          <tr><td>Lorem ipsum</td><td>Dolor</td><td>Sit</td><td>Amet consectetur</td></tr>
          <tr><td>Lorem ipsum</td><td>Dolor</td><td>Sit</td><td>Amet consectetur</td></tr>
          <tr><td>Lorem ipsum</td><td>Dolor</td><td>Sit</td><td>Amet consectetur</td></tr>
        </tbody>
      </table>
    </div>
    Faux bottom caption
  </div>
</div>

<!--[if lte IE 9]><style>.scrollingtable > div > div > table {margin-right: 17px;}</style><![endif]-->

El método que utilicé para congelar la fila del encabezado es similar al de d-Pixie, así que consulte su publicación para obtener una explicación. Hubo una gran cantidad de errores y limitaciones con esa técnica que solo se pudieron solucionar con un montón de CSS adicional y un contenedor div adicional o dos.


¡Esta respuesta es muy poco apreciada! Pasé días tratando de obtener otras soluciones para trabajar para mi caso especialmente molesto. Cada uno de ellos no pudo mantenerse alineado de una forma u otra. Esto finalmente lo hizo! Parece demasiado complicado al principio, pero una vez que lo dominas, es increíble. Se puede quitar un poco de cosas que no es necesario que al final, cuando no se utiliza el ancho de fluidos, etc.
Justin Sane

1
@JustinSane Me alegro de que te guste! Supongo que la falta de aprecio se debe al hecho de que comparte la página con la increíble solución de Maximilian Hils . Si no te opones a usar un poco de JS, definitivamente deberías echarle un vistazo.
DoctorDestructo

Maldición, esa es una solución casi perfecta. Estaba usando jQuery de todos modos, traté de hacerlo funcionar antes de encontrar el tuyo (a través de tu comentario a otra pregunta). No estaba pensando en un oyente de desplazamiento y traducir ... Bueno, dicen que se necesita un genio para encontrar soluciones simples ...;) Terminé el proyecto y funciona perfectamente sin js, pero lo guardaré en mente para el futuro ¡Aún así, felicitaciones por ser increíble!
Justin Sane

Un pequeño problema, pero si está utilizando diferentes colores del sistema, puede ver que el color del texto no se ha configurado para nada excepto los encabezados, pero el fondo de la tabla tiene un color de fondo explícito. Tengo texto amarillo sobre un fondo blanco y gris para esta tabla.
Matt Arnold

1
@MattArnold fijo. Gracias por la propina!
DoctorDestructo

5

Un simple complemento jQuery

Esta es una variación de la solución de Mahes. Puedes llamarlo como$('table#foo').scrollableTable();

La idea es:

  • Dividir theady tbodyen tableelementos separados
  • Haga que sus anchos de celda coincidan nuevamente
  • Envuelve el segundo tableen undiv.scrollable
  • Use CSS para hacer que div.scrollablerealmente se desplace

El CSS podría ser:

div.scrollable { height: 300px; overflow-y: scroll;}

Advertencias

  • Obviamente, dividir estas tablas hace que el marcado sea menos semántico. No estoy seguro de qué efecto tiene esto en la accesibilidad.
  • Este complemento no trata con pies de página, encabezados múltiples, etc.
  • Solo lo probé en Chrome versión 20.

Dicho esto, funciona para mis propósitos y eres libre de tomarlo y modificarlo.

Aquí está el complemento:

jQuery.fn.scrollableTable = function () {
  var $newTable, $oldTable, $scrollableDiv, originalWidths;
  $oldTable = $(this);

  // Once the tables are split, their cell widths may change. 
  // Grab these so we can make the two tables match again.
  originalWidths = $oldTable.find('tr:first td').map(function() {
    return $(this).width();
  });

  $newTable = $oldTable.clone();
  $oldTable.find('tbody').remove();
  $newTable.find('thead').remove();

  $.each([$oldTable, $newTable], function(index, $table) {
    $table.find('tr:first td').each(function(i) {
      $(this).width(originalWidths[i]);
    });
  });

  $scrollableDiv = $('<div/>').addClass('scrollable');
  $newTable.insertAfter($oldTable).wrap($scrollableDiv);
};

1
Buen guión, este funcionó mejor en mi entorno. Extendí su script con soporte de pie de página fijo, consulte mi publicación a continuación.
gitaarik

4

:)

Solución no tan limpia, pero pura de HTML / CSS.

table {
    overflow-x:scroll;
}

tbody {
    max-height: /*your desired max height*/
    overflow-y:scroll;
    display:block;
}

Actualizado para IE8 + JSFiddle ejemplo


2
Buena solución, solo para mencionar, que estas celdas están flotando y, por lo tanto, de acuerdo con el contenido pueden tener diferentes alturas, es visible si establece bordes: jsfiddle.net/ZdeEH/15
Stano

3

Soporte para pie de página fijo

Extendí la función de Nathan para soportar también un pie de página fijo y una altura máxima. Además, la función establecerá el CSS en sí, y solo debe admitir un ancho.

Uso:

Altura fija:

$('table').scrollableTable({ height: 100 });

Altura máxima (si el navegador admite la opción 'altura máxima' de CSS):

$('table').scrollableTable({ maxHeight: 100 });

Guión:

jQuery.fn.scrollableTable = function(options) {

    var $originalTable, $headTable, $bodyTable, $footTable, $scrollableDiv, originalWidths;

    // Prepare the separate parts of the table
    $originalTable = $(this);
    $headTable = $originalTable.clone();

    $headTable.find('tbody').remove();
    $headTable.find('tfoot').remove();

    $bodyTable = $originalTable.clone();
    $bodyTable.find('thead').remove();
    $bodyTable.find('tfoot').remove();

    $footTable = $originalTable.clone();
    $footTable.find('thead').remove();
    $footTable.find('tbody').remove();

    // Grab the original column widths and set them in the separate tables
    originalWidths = $originalTable.find('tr:first td').map(function() {
        return $(this).width();
    });

    $.each([$headTable, $bodyTable, $footTable], function(index, $table) {
        $table.find('tr:first td').each(function(i) {
            $(this).width(originalWidths[i]);
        });
    });

    // The div that makes the body table scroll
    $scrollableDiv = $('<div/>').css({
        'overflow-y': 'scroll'
    });

    if(options.height) {
        $scrollableDiv.css({'height': options.height});
    }
    else if(options.maxHeight) {
        $scrollableDiv.css({'max-height': options.maxHeight});
    }

    // Add the new separate tables and remove the original one
    $headTable.insertAfter($originalTable);
    $bodyTable.insertAfter($headTable);
    $footTable.insertAfter($bodyTable);
    $bodyTable.wrap($scrollableDiv);
    $originalTable.remove();
};

3

De alguna manera terminé Position:Stickytrabajando bien en mi caso:

table{
  width: 100%;
  border: collapse;
}

th{
    position: sticky;
    top: 0px;
    border: 1px solid black;
    background: #ff5722;
    color: #f5f5f5;
    font-weight: 600;
}
td{
    background: #d3d3d3;
    border: 1px solid black;
    color: #f5f5f5;
    font-weight: 600;
}

div{
  height: 150px
  overflow: auto;
  width: 100%
}
<div>
    <table>
        <thead>
            <tr>
                <th>header 1</th>
                <th>header 2</th>
                <th>header 3</th>
                <th>header 4</th>
                <th>header 5</th>
                <th>header 6</th>
                <th>header 7</th>
            </tr>
        </thead>
        <tbody>
            <tr>
                <td>data 1</td>
                <td>data 2</td>
                <td>data 3</td>
                <td>data 4</td>
                <td>data 5</td>
                <td>data 6</td>
                <td>data 7</td>
            </tr>
            <tr>
                <td>data 1</td>
                <td>data 2</td>
                <td>data 3</td>
                <td>data 4</td>
                <td>data 5</td>
                <td>data 6</td>
                <td>data 7</td>
            </tr>
            <tr>
                <td>data 1</td>
                <td>data 2</td>
                <td>data 3</td>
                <td>data 4</td>
                <td>data 5</td>
                <td>data 6</td>
                <td>data 7</td>
            </tr>
            <tr>
                <td>data 1</td>
                <td>data 2</td>
                <td>data 3</td>
                <td>data 4</td>
                <td>data 5</td>
                <td>data 6</td>
                <td>data 7</td>
            </tr>
            <tr>
                <td>data 1</td>
                <td>data 2</td>
                <td>data 3</td>
                <td>data 4</td>
                <td>data 5</td>
                <td>data 6</td>
                <td>data 7</td>
            </tr>
            <tr>
                <td>data 1</td>
                <td>data 2</td>
                <td>data 3</td>
                <td>data 4</td>
                <td>data 5</td>
                <td>data 6</td>
                <td>data 7</td>
            </tr>
            <tr>
                <td>data 1</td>
                <td>data 2</td>
                <td>data 3</td>
                <td>data 4</td>
                <td>data 5</td>
                <td>data 6</td>
                <td>data 7</td>
            </tr>
            <tr>
                <td>data 1</td>
                <td>data 2</td>
                <td>data 3</td>
                <td>data 4</td>
                <td>data 5</td>
                <td>data 6</td>
                <td>data 7</td>
            </tr>
            <tr>
                <td>data 1</td>
                <td>data 2</td>
                <td>data 3</td>
                <td>data 4</td>
                <td>data 5</td>
                <td>data 6</td>
                <td>data 7</td>
            </tr>
            <tr>
                <td>data 1</td>
                <td>data 2</td>
                <td>data 3</td>
                <td>data 4</td>
                <td>data 5</td>
                <td>data 6</td>
                <td>data 7</td>
            </tr>
            <tr>
                <td>data 1</td>
                <td>data 2</td>
                <td>data 3</td>
                <td>data 4</td>
                <td>data 5</td>
                <td>data 6</td>
                <td>data 7</td>
            </tr>
            <tr>
                <td>data 1</td>
                <td>data 2</td>
                <td>data 3</td>
                <td>data 4</td>
                <td>data 5</td>
                <td>data 6</td>
                <td>data 7</td>
            </tr>
            <tr>
                <td>data 1</td>
                <td>data 2</td>
                <td>data 3</td>
                <td>data 4</td>
                <td>data 5</td>
                <td>data 6</td>
                <td>data 7</td>
            </tr>
            <tr>
                <td>data 1</td>
                <td>data 2</td>
                <td>data 3</td>
                <td>data 4</td>
                <td>data 5</td>
                <td>data 6</td>
                <td>data 7</td>
            </tr>
        </tbody>
    </table>
</div>


1
Esta es la solución más limpia que he visto hasta ahora. caniuse muestra que a partir del 5/2/2020, posición no prefijada: pegajosa disfruta del 90.06% de soporte global. Entonces, esta solución funciona bien en todos los navegadores modernos.
AlienKevin

2

Dos divs, uno para encabezado, uno para datos. Haga que el div de datos se pueda desplazar y use JavaScript para establecer que el ancho de las columnas en el encabezado sea el mismo que el ancho de los datos. Creo que los anchos de las columnas de datos deben ser fijos en lugar de dinámicos.


3
Si te importa la accesibilidad, esto es un fracaso.
epascarello

1
re accesibilidad, tal vez podamos reemplazar el uso de divs con estilo en <thead> y <tbody> ??
Cheekysoft

1

Me doy cuenta de que la pregunta permite JavaScript, pero aquí hay una solución CSS pura que trabajé que también permite que la tabla se expanda horizontalmente. Fue probado con Internet Explorer 10 y los últimos navegadores Chrome y Firefox. Un enlace a jsFiddle está en la parte inferior.

El HTML:

Putting some text here to differentiate between the header
aligning with the top of the screen and the header aligning
with the top of one of its ancestor containers.

<div id="positioning-container">
<div id="scroll-container">
    <table>
        <colgroup>
            <col class="col1"></col>
            <col class="col2"></col>
        </colgroup>
        <thead>
            <th class="header-col1"><div>Header 1</div></th>
            <th class="header-col2"><div>Header 2</div></th>
        </thead>
        <tbody>
            <tr><td>Cell 1.1</td><td>Cell 1.2</td></tr>
            <tr><td>Cell 2.1</td><td>Cell 2.2</td></tr>
            <tr><td>Cell 3.1</td><td>Cell 3.2</td></tr>
            <tr><td>Cell 4.1</td><td>Cell 4.2</td></tr>
            <tr><td>Cell 5.1</td><td>Cell 5.2</td></tr>
            <tr><td>Cell 6.1</td><td>Cell 6.2</td></tr>
            <tr><td>Cell 7.1</td><td>Cell 7.2</td></tr>

        </tbody>
    </table>
</div>
</div>

Y el CSS:

table{
    border-collapse: collapse;
    table-layout: fixed;
    width: 100%;
}
/* Not required, just helps with alignment for this example */
td, th{
    padding: 0;
    margin: 0;
}

tbody{
    background-color: #ddf;
}

thead {
    /* Keeps the header in place. Don't forget top: 0 */
    position: absolute;
    top: 0;
    background-color: #ddd;

    /* The 17px is to adjust for the scrollbar width.
     * This is a new css value that makes this pure
     * css example possible */
    width: calc(100% - 17px);
    height: 20px;
}

/* Positioning container. Required to position the
 * header since the header uses position:absolute
 * (otherwise it would position at the top of the screen) */
#positioning-container{
    position: relative;
}

/* A container to set the scroll-bar and
 * includes padding to move the table contents
 * down below the header (padding = header height) */
#scroll-container{
    overflow-y: auto;
    padding-top: 20px;
    height: 100px;
}
.header-col1{
    background-color: red;
}

/* Fixed-width header columns need a div to set their width */
.header-col1 div{
    width: 100px;
}

/* Expandable columns need a width set on the th tag */
.header-col2{
    width: 100%;
}
.col1 {
    width: 100px;
}
.col2{
    width: 100%;
}

http://jsfiddle.net/HNHRv/3/


1

Para aquellos que probaron la buena solución dada por Maximilian Hils, y no lograron que funcionara con Internet Explorer, tuve el mismo problema (Internet Explorer 11) y descubrí cuál era el problema.

En Internet Explorer 11, la transformación de estilo (al menos con traducir) no funciona <THEAD>. Resolví esto aplicando el estilo a todo <TH>en un bucle. Eso funciono. Mi código JavaScript se ve así:

document.getElementById('pnlGridWrap').addEventListener("scroll", function () {
  var translate = "translate(0," + this.scrollTop + "px)";
  var myElements = this.querySelectorAll("th");
  for (var i = 0; i < myElements.length; i++) {
    myElements[i].style.transform=translate;
  }
});

En mi caso, la tabla era un GridView en ASP.NET. Primero pensé que era porque no tenía <THEAD>, pero incluso cuando lo forcé a tener uno, no funcionó. Entonces descubrí lo que escribí arriba.

Es una solución muy agradable y simple. En Chrome es perfecto, en Firefox un poco desigual, y en Internet Explorer aún más desigual. Pero en general, una buena solución.


0

Desearía haber encontrado la solución de @ Mark antes, pero fui y escribí la mía antes de ver esta pregunta SO ...

El mío es un complemento jQuery muy liviano que admite encabezado fijo, pie de página, extensión de columna (colspan), cambio de tamaño, desplazamiento horizontal y un número opcional de filas para mostrar antes de que comience el desplazamiento.

jQuery.scrollTableBody (GitHub)

Siempre que tenga una tabla con adecuada <thead>, <tbody>y (opcional) <tfoot>, todo lo que necesita hacer es esto:

$('table').scrollTableBody();

0

Encontré esta solución: mueva la fila del encabezado en una tabla sobre la tabla con datos:

<html>
<head>
	<title>Fixed header</title>
	<style>
		table td {width:75px;}
	</style>
</head>

<body>
<div style="height:auto; width:350px; overflow:auto">
<table border="1">
<tr>
	<td>header 1</td>
	<td>header 2</td>
	<td>header 3</td>
</tr>
</table>
</div>

<div style="height:50px; width:350px; overflow:auto">
<table border="1">
<tr>
	<td>row 1 col 1</td>
	<td>row 1 col 2</td>
	<td>row 1 col 3</td>		
</tr>
<tr>
	<td>row 2 col 1</td>
	<td>row 2 col 2</td>
	<td>row 2 col 3</td>		
</tr>
<tr>
	<td>row 3 col 1</td>
	<td>row 3 col 2</td>
	<td>row 3 col 3</td>		
</tr>
<tr>
	<td>row 4 col 1</td>
	<td>row 4 col 2</td>
	<td>row 4 col 3</td>		
</tr>
<tr>
	<td>row 5 col 1</td>
	<td>row 5 col 2</td>
	<td>row 5 col 3</td>		
</tr>
<tr>
	<td>row 6 col 1</td>
	<td>row 6 col 2</td>
	<td>row 6 col 3</td>		
</tr>
</table>
</div>


</body>
</html>


funciona para tablas pequeñas, pero si tiene desplazamiento horizontal, esta solución no funcionará.
crh225

Tampoco funcionará correctamente ya que las columnas de la tabla no se alinearán. Aquí está forzando ancho para td pero no debemos hacer ...
Ziggler

0

Al aplicar el complemento jQuery StickyTableHeaders a la tabla, los encabezados de las columnas se pegarán en la parte superior de la ventana gráfica a medida que se desplaza hacia abajo.

Ejemplo:

$(function () {
    $("table").stickyTableHeaders();
});

/*! Copyright (c) 2011 by Jonas Mosbech - https://github.com/jmosbech/StickyTableHeaders
	MIT license info: https://github.com/jmosbech/StickyTableHeaders/blob/master/license.txt */

;
(function ($, window, undefined) {
    'use strict';

    var name = 'stickyTableHeaders',
        id = 0,
        defaults = {
            fixedOffset: 0,
            leftOffset: 0,
            marginTop: 0,
            scrollableArea: window
        };

    function Plugin(el, options) {
        // To avoid scope issues, use 'base' instead of 'this'
        // to reference this class from internal events and functions.
        var base = this;

        // Access to jQuery and DOM versions of element
        base.$el = $(el);
        base.el = el;
        base.id = id++;
        base.$window = $(window);
        base.$document = $(document);

        // Listen for destroyed, call teardown
        base.$el.bind('destroyed',
        $.proxy(base.teardown, base));

        // Cache DOM refs for performance reasons
        base.$clonedHeader = null;
        base.$originalHeader = null;

        // Keep track of state
        base.isSticky = false;
        base.hasBeenSticky = false;
        base.leftOffset = null;
        base.topOffset = null;

        base.init = function () {
            base.$el.each(function () {
                var $this = $(this);

                // remove padding on <table> to fix issue #7
                $this.css('padding', 0);

                base.$originalHeader = $('thead:first', this);
                base.$clonedHeader = base.$originalHeader.clone();
                $this.trigger('clonedHeader.' + name, [base.$clonedHeader]);

                base.$clonedHeader.addClass('tableFloatingHeader');
                base.$clonedHeader.css('display', 'none');

                base.$originalHeader.addClass('tableFloatingHeaderOriginal');

                base.$originalHeader.after(base.$clonedHeader);

                base.$printStyle = $('<style type="text/css" media="print">' +
                    '.tableFloatingHeader{display:none !important;}' +
                    '.tableFloatingHeaderOriginal{position:static !important;}' +
                    '</style>');
                $('head').append(base.$printStyle);
            });

            base.setOptions(options);
            base.updateWidth();
            base.toggleHeaders();
            base.bind();
        };

        base.destroy = function () {
            base.$el.unbind('destroyed', base.teardown);
            base.teardown();
        };

        base.teardown = function () {
            if (base.isSticky) {
                base.$originalHeader.css('position', 'static');
            }
            $.removeData(base.el, 'plugin_' + name);
            base.unbind();

            base.$clonedHeader.remove();
            base.$originalHeader.removeClass('tableFloatingHeaderOriginal');
            base.$originalHeader.css('visibility', 'visible');
            base.$printStyle.remove();

            base.el = null;
            base.$el = null;
        };

        base.bind = function () {
            base.$scrollableArea.on('scroll.' + name, base.toggleHeaders);
            if (!base.isWindowScrolling) {
                base.$window.on('scroll.' + name + base.id, base.setPositionValues);
                base.$window.on('resize.' + name + base.id, base.toggleHeaders);
            }
            base.$scrollableArea.on('resize.' + name, base.toggleHeaders);
            base.$scrollableArea.on('resize.' + name, base.updateWidth);
        };

        base.unbind = function () {
            // unbind window events by specifying handle so we don't remove too much
            base.$scrollableArea.off('.' + name, base.toggleHeaders);
            if (!base.isWindowScrolling) {
                base.$window.off('.' + name + base.id, base.setPositionValues);
                base.$window.off('.' + name + base.id, base.toggleHeaders);
            }
            base.$scrollableArea.off('.' + name, base.updateWidth);
        };

        base.toggleHeaders = function () {
            if (base.$el) {
                base.$el.each(function () {
                    var $this = $(this),
                        newLeft,
                        newTopOffset = base.isWindowScrolling ? (
                        isNaN(base.options.fixedOffset) ? base.options.fixedOffset.outerHeight() : base.options.fixedOffset) : base.$scrollableArea.offset().top + (!isNaN(base.options.fixedOffset) ? base.options.fixedOffset : 0),
                        offset = $this.offset(),

                        scrollTop = base.$scrollableArea.scrollTop() + newTopOffset,
                        scrollLeft = base.$scrollableArea.scrollLeft(),

                        scrolledPastTop = base.isWindowScrolling ? scrollTop > offset.top : newTopOffset > offset.top,
                        notScrolledPastBottom = (base.isWindowScrolling ? scrollTop : 0) < (offset.top + $this.height() - base.$clonedHeader.height() - (base.isWindowScrolling ? 0 : newTopOffset));

                    if (scrolledPastTop && notScrolledPastBottom) {
                        newLeft = offset.left - scrollLeft + base.options.leftOffset;
                        base.$originalHeader.css({
                            'position': 'fixed',
                                'margin-top': base.options.marginTop,
                                'left': newLeft,
                                'z-index': 3 // #18: opacity bug
                        });
                        base.leftOffset = newLeft;
                        base.topOffset = newTopOffset;
                        base.$clonedHeader.css('display', '');
                        if (!base.isSticky) {
                            base.isSticky = true;
                            // make sure the width is correct: the user might have resized the browser while in static mode
                            base.updateWidth();
                        }
                        base.setPositionValues();
                    } else if (base.isSticky) {
                        base.$originalHeader.css('position', 'static');
                        base.$clonedHeader.css('display', 'none');
                        base.isSticky = false;
                        base.resetWidth($('td,th', base.$clonedHeader), $('td,th', base.$originalHeader));
                    }
                });
            }
        };

        base.setPositionValues = function () {
            var winScrollTop = base.$window.scrollTop(),
                winScrollLeft = base.$window.scrollLeft();
            if (!base.isSticky || winScrollTop < 0 || winScrollTop + base.$window.height() > base.$document.height() || winScrollLeft < 0 || winScrollLeft + base.$window.width() > base.$document.width()) {
                return;
            }
            base.$originalHeader.css({
                'top': base.topOffset - (base.isWindowScrolling ? 0 : winScrollTop),
                    'left': base.leftOffset - (base.isWindowScrolling ? 0 : winScrollLeft)
            });
        };

        base.updateWidth = function () {
            if (!base.isSticky) {
                return;
            }
            // Copy cell widths from clone
            if (!base.$originalHeaderCells) {
                base.$originalHeaderCells = $('th,td', base.$originalHeader);
            }
            if (!base.$clonedHeaderCells) {
                base.$clonedHeaderCells = $('th,td', base.$clonedHeader);
            }
            var cellWidths = base.getWidth(base.$clonedHeaderCells);
            base.setWidth(cellWidths, base.$clonedHeaderCells, base.$originalHeaderCells);

            // Copy row width from whole table
            base.$originalHeader.css('width', base.$clonedHeader.width());
        };

        base.getWidth = function ($clonedHeaders) {
            var widths = [];
            $clonedHeaders.each(function (index) {
                var width, $this = $(this);

                if ($this.css('box-sizing') === 'border-box') {
                    width = $this[0].getBoundingClientRect().width; // #39: border-box bug
                } else {
                    var $origTh = $('th', base.$originalHeader);
                    if ($origTh.css('border-collapse') === 'collapse') {
                        if (window.getComputedStyle) {
                            width = parseFloat(window.getComputedStyle(this, null).width);
                        } else {
                            // ie8 only
                            var leftPadding = parseFloat($this.css('padding-left'));
                            var rightPadding = parseFloat($this.css('padding-right'));
                            // Needs more investigation - this is assuming constant border around this cell and it's neighbours.
                            var border = parseFloat($this.css('border-width'));
                            width = $this.outerWidth() - leftPadding - rightPadding - border;
                        }
                    } else {
                        width = $this.width();
                    }
                }

                widths[index] = width;
            });
            return widths;
        };

        base.setWidth = function (widths, $clonedHeaders, $origHeaders) {
            $clonedHeaders.each(function (index) {
                var width = widths[index];
                $origHeaders.eq(index).css({
                    'min-width': width,
                        'max-width': width
                });
            });
        };

        base.resetWidth = function ($clonedHeaders, $origHeaders) {
            $clonedHeaders.each(function (index) {
                var $this = $(this);
                $origHeaders.eq(index).css({
                    'min-width': $this.css('min-width'),
                        'max-width': $this.css('max-width')
                });
            });
        };

        base.setOptions = function (options) {
            base.options = $.extend({}, defaults, options);
            base.$scrollableArea = $(base.options.scrollableArea);
            base.isWindowScrolling = base.$scrollableArea[0] === window;
        };

        base.updateOptions = function (options) {
            base.setOptions(options);
            // scrollableArea might have changed
            base.unbind();
            base.bind();
            base.updateWidth();
            base.toggleHeaders();
        };

        // Run initializer
        base.init();
    }

    // A plugin wrapper around the constructor,
    // preventing against multiple instantiations
    $.fn[name] = function (options) {
        return this.each(function () {
            var instance = $.data(this, 'plugin_' + name);
            if (instance) {
                if (typeof options === 'string') {
                    instance[options].apply(instance);
                } else {
                    instance.updateOptions(options);
                }
            } else if (options !== 'destroy') {
                $.data(this, 'plugin_' + name, new Plugin(this, options));
            }
        });
    };

})(jQuery, window);
body {
    margin: 0 auto;
    padding: 0 20px;
    font-family: Arial, Helvetica, sans-serif;
    font-size: 11px;
    color: #555;
}
table {
    border: 0;
    padding: 0;
    margin: 0 0 20px 0;
    border-collapse: collapse;
}
th {
    padding: 5px;
    /* NOTE: th padding must be set explicitly in order to support IE */
    text-align: right;
    font-weight:bold;
    line-height: 2em;
    color: #FFF;
    background-color: #555;
}
tbody td {
    padding: 10px;
    line-height: 18px;
    border-top: 1px solid #E0E0E0;
}
tbody tr:nth-child(2n) {
    background-color: #F7F7F7;
}
tbody tr:hover {
    background-color: #EEEEEE;
}
td {
    text-align: right;
}
td:first-child, th:first-child {
    text-align: left;
}
<script src="https://ajax.googleapis.com/ajax/libs/jquery/1.9.1/jquery.min.js"></script>
<div style="width:3000px">some really really wide content goes here</div>
<table>
    <thead>
        <tr>
            <th colspan="9">Companies listed on NASDAQ OMX Copenhagen.</th>
        </tr>
        <tr>
            <th>Full name</th>
            <th>CCY</th>
            <th>Last</th>
            <th>+/-</th>
            <th>%</th>
            <th>Bid</th>
            <th>Ask</th>
            <th>Volume</th>
            <th>Turnover</th>
        </tr>
    </thead>
    <tbody>
        <tr>
            <td>A.P. Møller...</td>
            <td>DKK</td>
            <td>33,220.00</td>
            <td>760</td>
            <td>2.34</td>
            <td>33,140.00</td>
            <td>33,220.00</td>
            <td>594</td>
            <td>19,791,910</td>
        </tr>
        <tr>
            <td>A.P. Møller...</td>
            <td>DKK</td>
            <td>34,620.00</td>
            <td>640</td>
            <td>1.88</td>
            <td>34,620.00</td>
            <td>34,700.00</td>
            <td>9,954</td>
            <td>346,530,246</td>
        </tr>
        <tr>
            <td>Carlsberg A</td>
            <td>DKK</td>
            <td>380</td>
            <td>0</td>
            <td>0</td>
            <td>371</td>
            <td>391.5</td>
            <td>6</td>
            <td>2,280</td>
        </tr>
        <tr>
            <td>Carlsberg B</td>
            <td>DKK</td>
            <td>364.4</td>
            <td>8.6</td>
            <td>2.42</td>
            <td>363</td>
            <td>364.4</td>
            <td>636,267</td>
            <td>228,530,601</td>
        </tr>
        <tr>
            <td>Chr. Hansen...</td>
            <td>DKK</td>
            <td>114.5</td>
            <td>-1.6</td>
            <td>-1.38</td>
            <td>114.2</td>
            <td>114.5</td>
            <td>141,822</td>
            <td>16,311,454</td>
        </tr>
        <tr>
            <td>Coloplast B</td>
            <td>DKK</td>
            <td>809.5</td>
            <td>11</td>
            <td>1.38</td>
            <td>809</td>
            <td>809.5</td>
            <td>85,840</td>
            <td>69,363,301</td>
        </tr>
        <tr>
            <td>D/S Norden</td>
            <td>DKK</td>
            <td>155</td>
            <td>-1.5</td>
            <td>-0.96</td>
            <td>155</td>
            <td>155.1</td>
            <td>51,681</td>
            <td>8,037,225</td>
        </tr>
        <tr>
            <td>Danske Bank</td>
            <td>DKK</td>
            <td>69.05</td>
            <td>2.55</td>
            <td>3.83</td>
            <td>69.05</td>
            <td>69.2</td>
            <td>1,723,719</td>
            <td>115,348,068</td>
        </tr>
        <tr>
            <td>DSV</td>
            <td>DKK</td>
            <td>105.4</td>
            <td>0.2</td>
            <td>0.19</td>
            <td>105.2</td>
            <td>105.4</td>
            <td>674,873</td>
            <td>71,575,035</td>
        </tr>
        <tr>
            <td>FLSmidth &amp; Co.</td>
            <td>DKK</td>
            <td>295.8</td>
            <td>-1.8</td>
            <td>-0.6</td>
            <td>295.1</td>
            <td>295.8</td>
            <td>341,263</td>
            <td>100,301,032</td>
        </tr>
        <tr>
            <td>G4S plc</td>
            <td>DKK</td>
            <td>22.53</td>
            <td>0.05</td>
            <td>0.22</td>
            <td>22.53</td>
            <td>22.57</td>
            <td>190,920</td>
            <td>4,338,150</td>
        </tr>
        <tr>
            <td>Jyske Bank</td>
            <td>DKK</td>
            <td>144.2</td>
            <td>1.4</td>
            <td>0.98</td>
            <td>142.8</td>
            <td>144.2</td>
            <td>78,163</td>
            <td>11,104,874</td>
        </tr>
        <tr>
            <td>Københavns ...</td>
            <td>DKK</td>
            <td>1,580.00</td>
            <td>-12</td>
            <td>-0.75</td>
            <td>1,590.00</td>
            <td>1,620.00</td>
            <td>82</td>
            <td>131,110</td>
        </tr>
        <tr>
            <td>Lundbeck</td>
            <td>DKK</td>
            <td>103.4</td>
            <td>-2.5</td>
            <td>-2.36</td>
            <td>103.4</td>
            <td>103.8</td>
            <td>157,162</td>
            <td>16,462,282</td>
        </tr>
        <tr>
            <td>Nordea Bank</td>
            <td>DKK</td>
            <td>43.22</td>
            <td>-0.06</td>
            <td>-0.14</td>
            <td>43.22</td>
            <td>43.25</td>
            <td>167,520</td>
            <td>7,310,143</td>
        </tr>
        <tr>
            <td>Novo Nordisk B</td>
            <td>DKK</td>
            <td>552.5</td>
            <td>-3.5</td>
            <td>-0.63</td>
            <td>550.5</td>
            <td>552.5</td>
            <td>843,533</td>
            <td>463,962,375</td>
        </tr>
        <tr>
            <td>Novozymes B</td>
            <td>DKK</td>
            <td>805.5</td>
            <td>5.5</td>
            <td>0.69</td>
            <td>805</td>
            <td>805.5</td>
            <td>152,188</td>
            <td>121,746,199</td>
        </tr>
        <tr>
            <td>Pandora</td>
            <td>DKK</td>
            <td>39.04</td>
            <td>0.94</td>
            <td>2.47</td>
            <td>38.8</td>
            <td>39.04</td>
            <td>350,965</td>
            <td>13,611,838</td>
        </tr>
        <tr>
            <td>Rockwool In...</td>
            <td>DKK</td>
            <td>492</td>
            <td>0</td>
            <td>0</td>
            <td>482</td>
            <td>492</td>
            <td></td>
            <td></td>
        </tr>
        <tr>
            <td>Rockwool In...</td>
            <td>DKK</td>
            <td>468</td>
            <td>12</td>
            <td>2.63</td>
            <td>465.2</td>
            <td>468</td>
            <td>9,885</td>
            <td>4,623,850</td>
        </tr>
        <tr>
            <td>Sydbank</td>
            <td>DKK</td>
            <td>95</td>
            <td>0.05</td>
            <td>0.05</td>
            <td>94.7</td>
            <td>95</td>
            <td>103,438</td>
            <td>9,802,899</td>
        </tr>
        <tr>
            <td>TDC</td>
            <td>DKK</td>
            <td>43.6</td>
            <td>0.13</td>
            <td>0.3</td>
            <td>43.5</td>
            <td>43.6</td>
            <td>845,110</td>
            <td>36,785,339</td>
        </tr>
        <tr>
            <td>Topdanmark</td>
            <td>DKK</td>
            <td>854</td>
            <td>13.5</td>
            <td>1.61</td>
            <td>854</td>
            <td>855</td>
            <td>38,679</td>
            <td>32,737,678</td>
        </tr>
        <tr>
            <td>Tryg</td>
            <td>DKK</td>
            <td>290.4</td>
            <td>0.3</td>
            <td>0.1</td>
            <td>290</td>
            <td>290.4</td>
            <td>94,587</td>
            <td>27,537,247</td>
        </tr>
        <tr>
            <td>Vestas Wind...</td>
            <td>DKK</td>
            <td>90.15</td>
            <td>-4.2</td>
            <td>-4.45</td>
            <td>90.1</td>
            <td>90.15</td>
            <td>1,317,313</td>
            <td>121,064,314</td>
        </tr>
        <tr>
            <td>William Dem...</td>
            <td>DKK</td>
            <td>417.6</td>
            <td>0.1</td>
            <td>0.02</td>
            <td>417</td>
            <td>417.6</td>
            <td>64,242</td>
            <td>26,859,554</td>
        </tr>
    </tbody>
</table>
<div style="height: 4000px">lots of content down here...</div>


0

Me gusta la respuesta de Maximillian Hils pero tuve algunos problemas:

  1. la transformación no funciona en Edge o IE a menos que la aplique al th
  2. el encabezado parpadea durante el desplazamiento en Edge e IE
  3. mi tabla se carga usando ajax, por lo que quería adjuntar al evento de desplazamiento de la ventana en lugar del evento de desplazamiento del contenedor

Para deshacerme del parpadeo, utilizo un tiempo de espera para esperar hasta que el usuario haya terminado de desplazarse, luego aplico la transformación, por lo que el encabezado no es visible durante el desplazamiento.

También he escrito esto usando jQuery, una ventaja de eso es que jQuery debería manejar los prefijos de proveedor por usted

    var isScrolling, lastTop, lastLeft, isLeftHidden, isTopHidden;

    //Scroll events don't bubble https://stackoverflow.com/a/19375645/150342
    //so can't use $(document).on("scroll", ".table-container-fixed", function (e) {
    document.addEventListener('scroll', function (event) {
        var $container = $(event.target);
        if (!$container.hasClass("table-container-fixed"))
            return;    

        //transform needs to be applied to th for Edge and IE
        //in this example I am also fixing the leftmost column
        var $topLeftCell = $container.find('table:first > thead > tr > th:first');
        var $headerCells = $topLeftCell.siblings();
        var $columnCells = $container
           .find('table:first > tbody > tr > td:first-child, ' +
                 'table:first > tfoot > tr > td:first-child');

        //hide the cells while returning otherwise they show on top of the data
        if (!isLeftHidden) {
            var currentLeft = $container.scrollLeft();
            if (currentLeft < lastLeft) {
                //scrolling left
                isLeftHidden = true;
                $topLeftCell.css('visibility', 'hidden');
                $columnCells.css('visibility', 'hidden');
            }
            lastLeft = currentLeft;
        }

        if (!isTopHidden) {
            var currentTop = $container.scrollTop();
            if (currentTop < lastTop) {
                //scrolling up
                isTopHidden = true;
                $topLeftCell.css('visibility', 'hidden');
                $headerCells.css('visibility', 'hidden');
            }
            lastTop = currentTop;
        }

        // Using timeout to delay transform until user stops scrolling
        // Clear timeout while scrolling
        window.clearTimeout(isScrolling);

        // Set a timeout to run after scrolling ends
        isScrolling = setTimeout(function () {
            //move the table cells. 
            var x = $container.scrollLeft();
            var y = $container.scrollTop();

            $topLeftCell.css('transform', 'translate(' + x + 'px, ' + y + 'px)');
            $headerCells.css('transform', 'translateY(' + y + 'px)');
            $columnCells.css('transform', 'translateX(' + x + 'px)');

            isTopHidden = isLeftHidden = false;
            $topLeftCell.css('visibility', 'inherit');
            $headerCells.css('visibility', 'inherit');
            $columnCells.css('visibility', 'inherit');
        }, 100);

    }, true);

La tabla está envuelta en un div con la clase table-container-fixed.

.table-container-fixed{
    overflow: auto;
    height: 400px;
}

Configuré el colapso del borde para que se separe porque, de lo contrario, perdemos bordes durante la traducción, y elimino el borde de la tabla para evitar que el contenido aparezca justo encima de la celda donde estaba el borde durante el desplazamiento.

.table-container-fixed > table {
   border-collapse: separate;
   border:none;
}

Hago el thfondo blanco para cubrir las celdas debajo, y agrego un borde que coincide con el borde de la tabla, que está diseñado con Bootstrap y se desplaza fuera de la vista.

 .table-container-fixed > table > thead > tr > th {
        border-top: 1px solid #ddd !important;
        background-color: white;        
        z-index: 10;
        position: relative;/*to make z-index work*/
    }

            .table-container-fixed > table > thead > tr > th:first-child {
                z-index: 20;
            }

.table-container-fixed > table > tbody > tr > td:first-child,
.table-container-fixed > table > tfoot > tr > td:first-child {
    background-color: white;        
    z-index: 10;
    position: relative;
}

0

Use la última versión de jQuery e incluya el siguiente código JavaScript.

$(window).scroll(function(){
  $("id of the div element").offset({top:$(window).scrollTop()});
});

1
Esto no parece funcionar. ¿Quizás podrías aclarar lo que quieres que hagamos?
Chris

1
Que div? Estamos hablando de tablas aquí
isapir

0

Esta no es una solución exacta para la fila de encabezado fija, pero he creado un método bastante ingenioso para repetir la fila de encabezado a lo largo de la tabla larga, pero manteniendo la capacidad de ordenar.

Esta pequeña opción ordenada requiere el complemento jQuerytablesorter . Así es como funciona:

HTML

<table class="tablesorter boxlist" id="pmtable">
    <thead class="fixedheader">
        <tr class="boxheadrow">
            <th width="70px" class="header">Job Number</th>
            <th width="10px" class="header">Pri</th>
            <th width="70px" class="header">CLLI</th>
            <th width="35px" class="header">Market</th>
            <th width="35px" class="header">Job Status</th>
            <th width="65px" class="header">Technology</th>
            <th width="95px;" class="header headerSortDown">MEI</th>
            <th width="95px" class="header">TEO Writer</th>
            <th width="75px" class="header">Quote Due</th>
            <th width="100px" class="header">Engineer</th>
            <th width="75px" class="header">ML Due</th>
            <th width="75px" class="header">ML Complete</th>
            <th width="75px" class="header">SPEC Due</th>
            <th width="75px" class="header">SPEC Complete</th>
            <th width="100px" class="header">Install Supervisor</th>
            <th width="75px" class="header">MasTec OJD</th>
            <th width="75px" class="header">Install Start</th>
            <th width="30px" class="header">Install Hours</th>
            <th width="75px" class="header">Revised CRCD</th>
            <th width="75px" class="header">Latest Ship-To-Site</th>
            <th width="30px" class="header">Total Parts</th>
            <th width="30px" class="header">OEM Rcvd</th>
            <th width="30px" class="header">Minor Rcvd</th>
            <th width="30px" class="header">Total Received</th>
            <th width="30px" class="header">% On Site</th>
            <th width="60px" class="header">Actions</th>
        </tr>
    </thead>
        <tbody class="scrollable">
            <tr data-job_id="3548" data-ml_id="" class="odd">
                <td class="c black">FL-8-RG9UP</td>
                <td data-pri="2" class="priority c yellow">M</td>
                <td class="c">FTLDFLOV</td>
                <td class="c">SFL</td>
                <td class="c">NOI</td>
                <td class="c">TRANSPORT</td>
                <td class="c"></td>
                <td class="c">Chris Byrd</td>
                <td class="c">Apr 13, 2013</td>
                <td class="c">Kris Hall</td>
                <td class="c">May 20, 2013</td>
                <td class="c">May 20, 2013</td>
                <td class="c">Jun 5, 2013</td>
                <td class="c">Jun 7, 2013</td>
                <td class="c">Joseph Fitz</td>
                <td class="c">Jun 10, 2013</td>
                <td class="c">TBD</td>
                <td class="c">123</td>
                <td class="c revised_crcd"><input readonly="true" name="revised_crcd" value="Jul 26, 2013" type="text" size="12" class="smInput r_crcd c hasDatepicker" id="dp1377194058616"></td>
                <td class="c">TBD</td>
                <td class="c">N/A</td>
                <td class="c">N/A</td>
                <td class="c">N/A</td>
                <td class="c">N/A</td>
                <td class="c">N/A</td>
                <td class="actions"><span style="float:left;" class="ui-icon ui-icon-folder-open editJob" title="View this job" s="" details'=""></span></td>
            </tr>
            <tr data-job_id="4264" data-ml_id="2959" class="even">
                <td class="c black">MTS13009SF</td>
                <td data-pri="2" class="priority c yellow">M</td>
                <td class="c">OJUSFLTL</td>
                <td class="c">SFL</td>
                <td class="c">NOI</td>
                <td class="c">TRANSPORT</td>
                <td class="c"></td>
                <td class="c">DeMarcus Stewart</td>
                <td class="c">May 22, 2013</td>
                <td class="c">Ryan Alsobrook</td>
                <td class="c">Jun 19, 2013</td>
                <td class="c">Jun 27, 2013</td>
                <td class="c">Jun 19, 2013</td>
                <td class="c">Jul 4, 2013</td>
                <td class="c">Randy Williams</td>
                <td class="c">Jun 21, 2013</td>
                <td class="c">TBD</td>
                <td class="c">95</td>
                <td class="c revised_crcd"><input readonly="true" name="revised_crcd" value="Aug 9, 2013" type="text" size="12" class="smInput r_crcd c hasDatepicker" id="dp1377194058632"></td><td class="c">TBD</td>
                <td class="c">0</td>
                <td class="c">0.00%</td>
                <td class="c">0.00%</td>
                <td class="c">0.00%</td>
                <td class="c">0.00%</td>
                <td class="actions"><span style="float:left;" class="ui-icon ui-icon-folder-open editJob" title="View this job" s="" details'=""></span><input style="float:left;" type="hidden" name="req_ship" class="reqShip hasDatepicker" id="dp1377194058464"><span style="float:left;" class="ui-icon ui-icon-calendar requestShip" title="Schedule this job for shipping"></span><span class="ui-icon ui-icon-info viewOrderInfo" style="float:left;" title="Show material details for this order"></span></td>
            </tr>
            .
            .
            .
            .
            <tr class="boxheadrow repeated-header">
                <th width="70px" class="header">Job Number</th>
                <th width="10px" class="header">Pri</th>
                <th width="70px" class="header">CLLI</th>
                <th width="35px" class="header">Market</th>
                <th width="35px" class="header">Job Status</th>
                <th width="65px" class="header">Technology</th>
                <th width="95px;" class="header">MEI</th>
                <th width="95px" class="header">TEO Writer</th>
                <th width="75px" class="header">Quote Due</th>
                <th width="100px" class="header">Engineer</th>
                <th width="75px" class="header">ML Due</th>
                <th width="75px" class="header">ML Complete</th>
                <th width="75px" class="header">SPEC Due</th>
                <th width="75px" class="header">SPEC Complete</th>
                <th width="100px" class="header">Install Supervisor</th>
                <th width="75px" class="header">MasTec OJD</th>
                <th width="75px" class="header">Install Start</th>
                <th width="30px" class="header">Install Hours</th>
                <th width="75px" class="header">Revised CRCD</th>
                <th width="75px" class="header">Latest Ship-To-Site</th>
                <th width="30px" class="header">Total Parts</th>
                <th width="30px" class="header">OEM Rcvd</th>
                <th width="30px" class="header">Minor Rcvd</th>
                <th width="30px" class="header">Total Received</th>
                <th width="30px" class="header">% On Site</th>
                <th width="60px" class="header">Actions</th>
            </tr>

Obviamente, mi tabla tiene muchas más filas que esta. 193 para ser exactos, pero puede ver dónde se repite la fila del encabezado. La fila de encabezado de repetición se configura mediante esta función:

jQuery

// Clone the original header row and add the "repeated-header" class
var tblHeader = $('tr.boxheadrow').clone().addClass('repeated-header');

// Add the cloned header with the new class every 34th row (or as you see fit)
$('tbody tr:odd:nth-of-type(17n)').after(tblHeader);

// On the 'sortStart' routine, remove all the inserted header rows
$('#pmtable').bind('sortStart', function() {
    $('.repeated-header').remove();
    // On the 'sortEnd' routine, add back all the header row lines.
}).bind('sortEnd', function() {
    $('tbody tr:odd:nth-of-type(17n)').after(tblHeader);
});

0

Mucha gente parece estar buscando esta respuesta. Lo encontré enterrado en una respuesta a otra pregunta aquí: Sincronizando el ancho de columna de las tablas en dos marcos diferentes, etc.

De las docenas de métodos que he probado, este es el único método que encontré que funciona de manera confiable para permitirle tener una tabla inferior desplazable con la tabla de encabezado con los mismos anchos.

Así es como lo hice, primero mejoré el jsfiddle de arriba para crear esta función, que funciona en ambos tdy th(en caso de que tropiece con otros que usan thpara diseñar sus filas de encabezado).

var setHeaderTableWidth= function (headertableid,basetableid) {
            $("#"+headertableid).width($("#"+basetableid).width());
            $("#"+headertableid+" tr th").each(function (i) {
                $(this).width($($("#"+basetableid+" tr:first td")[i]).width());
            });
            $("#" + headertableid + " tr td").each(function (i) {
                $(this).width($($("#" + basetableid + " tr:first td")[i]).width());
            });
        }

A continuación, debe crear dos tablas, tenga en cuenta que la tabla de encabezado debe tener un TDespacio adicional para dejar espacio en la tabla superior para la barra de desplazamiento, de esta manera:

 <table id="headertable1" class="input-cells table-striped">
        <thead>
            <tr style="background-color:darkgray;color:white;"><th>header1</th><th>header2</th><th>header3</th><th>header4</th><th>header5</th><th>header6</th><th></th></tr>
        </thead>
     </table>
    <div id="resizeToBottom" style="overflow-y:scroll;overflow-x:hidden;">
        <table id="basetable1" class="input-cells table-striped">
            <tbody >
                <tr>
                    <td>testdata</td>
                    <td>2</td>
                    <td>3</td>
                    <td>4</span></td>
                    <td>55555555555555</td>
                    <td>test</td></tr>
            </tbody>
        </table>
    </div>

Luego haz algo como:

        setHeaderTableWidth('headertable1', 'basetable1');
        $(window).resize(function () {
            setHeaderTableWidth('headertable1', 'basetable1');
        });

Esta es la única solución que encontré en Stack Overflow que funciona a partir de muchas preguntas similares que se han publicado, que funciona en todos mis casos.

Por ejemplo, probé el complemento jQuery stickytables que no funciona con durandal, y el proyecto Google Code aquí https://code.google.com/p/js-scroll-table-header/issues/detail?id=2

Otras soluciones que implican la clonación de las tablas, tienen un rendimiento deficiente o son malas y no funcionan en todos los casos.

No hay necesidad de estas soluciones demasiado complejas. Simplemente haga dos tablas como los ejemplos a continuación y llame a la función setHeaderTableWidth como se describe aquí y boom, ya está .

Si esto no funciona para usted, probablemente estaba jugando con su propiedad de tamaño de caja CSS y necesita configurarla correctamente. Es fácil arruinar su contenido CSS por accidente. Hay muchas cosas que pueden salir mal, así que ten cuidado / ten cuidado con eso. Este enfoque funciona para mí .


0

Aquí hay una solución con la que terminamos trabajando (para hacer frente a algunos casos extremos y versiones anteriores de Internet Explorer, eventualmente también desvanecimos la barra de título en el desplazamiento y luego la desvanecimos cuando finaliza el desplazamiento, pero en los navegadores Firefox y WebKit esta solución simplemente funciona , supone un colapso de borde: colapso.

La clave de esta solución es que una vez que aplica el colapso del borde , las transformaciones CSS funcionan en el encabezado, por lo que es solo cuestión de interceptar eventos de desplazamiento y configurar la transformación correctamente. No necesitas duplicar nada. A menos que este comportamiento se implemente correctamente en el navegador, es difícil imaginar una solución más liviana.

JSFiddle: http://jsfiddle.net/podperson/tH9VU/2/

Se implementa como un simple complemento jQuery. Simplemente hace que su thead sea pegajoso con una llamada como $ ('thead'). Sticky (), y se quedarán. Funciona para varias tablas en una página y secciones de encabezado a la mitad de tablas grandes.

$.fn.sticky = function(){
    $(this).each( function(){
        var thead = $(this),
            tbody = thead.next('tbody');

        updateHeaderPosition();

        function updateHeaderPosition(){
            if(
                thead.offset().top < $(document).scrollTop()
                && tbody.offset().top + tbody.height() > $(document).scrollTop()
            ){
                var tr = tbody.find('tr').last(),
                    y = tr.offset().top - thead.height() < $(document).scrollTop()
                        ? tr.offset().top - thead.height() - thead.offset().top
                        : $(document).scrollTop() - thead.offset().top;

                thead.find('th').css({
                    'z-index': 100,
                    'transform': 'translateY(' + y + 'px)',
                    '-webkit-transform': 'translateY(' + y + 'px)'
                });
            } else {
                thead.find('th').css({
                    'transform': 'none',
                    '-webkit-transform': 'none'
                });
            }
        }

        // See http://www.quirksmode.org/dom/events/scroll.html
        $(window).on('scroll', updateHeaderPosition);
    });
}

$('thead').sticky();

buena solución, pero ¿cómo se incluyen los bordes de columna entre columnas (ambos en encabezado fijo, alineados con los datos td)?
user5249203

No estoy seguro de entender tu problema. el colapso del borde no le impide usar bordes, márgenes, etc., solo elimina las métricas de la tabla vudú de antaño.
podperson

1
Agregar border: 2px solid red;a th, desplazarse, y verá el problema. Se me ocurrió esta solución más básica: jsfiddle.net/x6pLcor9/19
calandoa

Agregue el mismo borde dimensionado a td y no hay problema. No entiendo tu punto. Su versión es mucho más limpia y no usa jQuery, por lo que definitivamente elegiría algo más como eso hoy. (Aunque, francamente, no creo que use una mesa hoy).
podperson

0

Aquí hay una respuesta mejorada a la publicada por Maximilian Hils .

Este funciona en Internet Explorer 11 sin parpadeo alguno:

var headerCells = tableWrap.querySelectorAll("thead td");
for (var i = 0; i < headerCells.length; i++) {
    var headerCell = headerCells[i];
    headerCell.style.backgroundColor = "silver";
}
var lastSTop = tableWrap.scrollTop;
tableWrap.addEventListener("scroll", function () {
    var stop = this.scrollTop;
    if (stop < lastSTop) {
        // Resetting the transform for the scrolling up to hide the headers
        for (var i = 0; i < headerCells.length; i++) {
            headerCells[i].style.transitionDelay = "0s";
            headerCells[i].style.transform = "";
        }
    }
    lastSTop = stop;
    var translate = "translate(0," + stop + "px)";
    for (var i = 0; i < headerCells.length; i++) {
        headerCells[i].style.transitionDelay = "0.25s";
        headerCells[i].style.transform = translate;
    }
});

0

Desarrollé un plugin jQuery simple y liviano para convertir una tabla HTML bien formateada en una tabla desplazable con encabezado y columnas fijas.

El complemento funciona bien para hacer coincidir el posicionamiento de píxel a píxel de la sección fija con la sección desplazable. Además, también puede congelar la cantidad de columnas que siempre estarán a la vista al desplazarse horizontalmente.

Demostración y documentación: http://meetselva.github.io/fixed-table-rows-cols/

Repositorio de GitHub: https://github.com/meetselva/fixed-table-rows-cols

A continuación se muestra el uso de una tabla simple con un encabezado fijo,

$(<table selector>).fxdHdrCol({
    width:     "100%",
    height:    200,
    colModal: [{width: 30, align: 'center'},
               {width: 70, align: 'center'}, 
               {width: 200, align: 'left'}, 
               {width: 100, align: 'center'}, 
               {width: 70, align: 'center'}, 
               {width: 250, align: 'center'}
              ]
});

¿Qué es "una tabla HTML bien" ?
Peter Mortensen

@PeterMortensen Debería haber sido "HTML bien formateado". Editado, gracias.
Selvakumar Arumugam

0
<html>
<head>
    <script src="//cdn.jsdelivr.net/npm/jquery@3.2.1/dist/jquery.min.js"></script>
    <script>
        function stickyTableHead (tableID) {
            var $tmain = $(tableID);
            var $tScroll = $tmain.children("thead")
                .clone()
                .wrapAll('<table id="tScroll" />')
                .parent()
                .addClass($(tableID).attr("class"))
                .css("position", "fixed")
                .css("top", "0")
                .css("display", "none")
                .prependTo("#tMain");

            var pos = $tmain.offset().top + $tmain.find(">thead").height();


            $(document).scroll(function () {
                var dataScroll = $tScroll.data("scroll");
                dataScroll = dataScroll || false;
                if ($(this).scrollTop() >= pos) {
                    if (!dataScroll) {
                        $tScroll
                            .data("scroll", true)
                            .show()
                            .find("th").each(function () {
                                $(this).width($tmain.find(">thead>tr>th").eq($(this).index()).width());
                            });
                    }
                } else {
                    if (dataScroll) {
                        $tScroll
                            .data("scroll", false)
                            .hide()
                        ;
                    }
                }
            });
        }

        $(document).ready(function () {
            stickyTableHead('#tMain');
        });
    </script>
</head>

<body>
    gfgfdgsfgfdgfds<br/>
    gfgfdgsfgfdgfds<br/>
    gfgfdgsfgfdgfds<br/>
    gfgfdgsfgfdgfds<br/>
    gfgfdgsfgfdgfds<br/>
    gfgfdgsfgfdgfds<br/>

    <table id="tMain" >
        <thead>
        <tr>
            <th>1</th> <th>2</th><th>3</th> <th>4</th><th>5</th> <th>6</th><th>7</th> <th>8</th>

        </tr>
        </thead>
        <tbody>
            <tr><td>11111111111111111111111111111111111111111111111111111111</td><td>2</td><td>3</td><td>4</td><td>5555555</td><td>66666666666</td><td>77777777777</td><td>8888888888888888</td></tr>
            <tr><td>1</td><td>2</td><td>3</td><td>4</td><td>5555555</td><td>66666666666</td><td>77777777777</td><td>8888888888888888</td></tr>
            <tr><td>1</td><td>2</td><td>3</td><td>4</td><td>5555555</td><td>66666666666</td><td>77777777777</td><td>8888888888888888</td></tr>
            <tr><td>1</td><td>2</td><td>3</td><td>4</td><td>5555555</td><td>66666666666</td><td>77777777777</td><td>8888888888888888</td></tr>
            <tr><td>1</td><td>2</td><td>3</td><td>4</td><td>5555555</td><td>66666666666</td><td>77777777777</td><td>8888888888888888</td></tr>
            <tr><td>1</td><td>2</td><td>3</td><td>4</td><td>5555555</td><td>66666666666</td><td>77777777777</td><td>8888888888888888</td></tr>
            <tr><td>1</td><td>2</td><td>3</td><td>4</td><td>5555555</td><td>66666666666</td><td>77777777777</td><td>8888888888888888</td></tr>
            <tr><td>1</td><td>2</td><td>3</td><td>4</td><td>5555555</td><td>66666666666</td><td>77777777777</td><td>8888888888888888</td></tr>
            <tr><td>1</td><td>2</td><td>3</td><td>4</td><td>5555555</td><td>66666666666</td><td>77777777777</td><td>8888888888888888</td></tr>
            <tr><td>1</td><td>2</td><td>3</td><td>4</td><td>5555555</td><td>66666666666</td><td>77777777777</td><td>8888888888888888</td></tr>
            <tr><td>1</td><td>2</td><td>3</td><td>4</td><td>5555555</td><td>66666666666</td><td>77777777777</td><td>8888888888888888</td></tr>
            <tr><td>1</td><td>2</td><td>3</td><td>4</td><td>5555555</td><td>66666666666</td><td>77777777777</td><td>8888888888888888</td></tr>
            <tr><td>1</td><td>2</td><td>3</td><td>4</td><td>5555555</td><td>66666666666</td><td>77777777777</td><td>8888888888888888</td></tr>
            <tr><td>1</td><td>2</td><td>3</td><td>4</td><td>5555555</td><td>66666666666</td><td>77777777777</td><td>8888888888888888</td></tr>
            <tr><td>1</td><td>2</td><td>3</td><td>4</td><td>5555555</td><td>66666666666</td><td>77777777777</td><td>8888888888888888</td></tr>
            <tr><td>1</td><td>2</td><td>3</td><td>4</td><td>5555555</td><td>66666666666</td><td>77777777777</td><td>8888888888888888</td></tr>
            <tr><td>1</td><td>2</td><td>3</td><td>4</td><td>5555555</td><td>66666666666</td><td>77777777777</td><td>8888888888888888</td></tr>
            <tr><td>1</td><td>2</td><td>3</td><td>4</td><td>5555555</td><td>66666666666</td><td>77777777777</td><td>8888888888888888</td></tr>
            <tr><td>1</td><td>2</td><td>3</td><td>4</td><td>5555555</td><td>66666666666</td><td>77777777777</td><td>8888888888888888</td></tr>
            <tr><td>1</td><td>2</td><td>3</td><td>4</td><td>5555555</td><td>66666666666</td><td>77777777777</td><td>8888888888888888</td></tr>
            <tr><td>1</td><td>2</td><td>3</td><td>4</td><td>5555555</td><td>66666666666</td><td>77777777777</td><td>8888888888888888</td></tr>
            <tr><td>1</td><td>2</td><td>3</td><td>4</td><td>5555555</td><td>66666666666</td><td>77777777777</td><td>8888888888888888</td></tr>
            <tr><td>1</td><td>2</td><td>3</td><td>4</td><td>5555555</td><td>66666666666</td><td>77777777777</td><td>8888888888888888</td></tr>
            <tr><td>1</td><td>2</td><td>3</td><td>4</td><td>5555555</td><td>66666666666</td><td>77777777777</td><td>8888888888888888</td></tr>
            <tr><td>1</td><td>2</td><td>3</td><td>4</td><td>5555555</td><td>66666666666</td><td>77777777777</td><td>8888888888888888</td></tr>
            <tr><td>1</td><td>2</td><td>3</td><td>4</td><td>5555555</td><td>66666666666</td><td>77777777777</td><td>8888888888888888</td></tr>
            <tr><td>1</td><td>2</td><td>3</td><td>4</td><td>5555555</td><td>66666666666</td><td>77777777777</td><td>8888888888888888</td></tr>
            <tr><td>1</td><td>2</td><td>3</td><td>4</td><td>5555555</td><td>66666666666</td><td>77777777777</td><td>8888888888888888</td></tr>
            <tr><td>1</td><td>2</td><td>3</td><td>4</td><td>5555555</td><td>66666666666</td><td>77777777777</td><td>8888888888888888</td></tr>
            <tr><td>1</td><td>2</td><td>3</td><td>4</td><td>5555555</td><td>66666666666</td><td>77777777777</td><td>8888888888888888</td></tr>
            <tr><td>1</td><td>2</td><td>3</td><td>4</td><td>5555555</td><td>66666666666</td><td>77777777777</td><td>8888888888888888</td></tr>
            <tr><td>1</td><td>2</td><td>3</td><td>4</td><td>5555555</td><td>66666666666</td><td>77777777777</td><td>8888888888888888</td></tr>
            <tr><td>1</td><td>2</td><td>3</td><td>4</td><td>5555555</td><td>66666666666</td><td>77777777777</td><td>8888888888888888</td></tr>
            <tr><td>1</td><td>2</td><td>3</td><td>4</td><td>5555555</td><td>66666666666</td><td>77777777777</td><td>8888888888888888</td></tr>
            <tr><td>1</td><td>2</td><td>3</td><td>4</td><td>5555555</td><td>66666666666</td><td>77777777777</td><td>8888888888888888</td></tr>
            <tr><td>1</td><td>2</td><td>3</td><td>4</td><td>5555555</td><td>66666666666</td><td>77777777777</td><td>8888888888888888</td></tr>
        </tbody>
    </table>
</body>
</html>

0

Adicional a la respuesta de @Daniel Waltrip. La tabla debe encerrarse con div position: relativepara poder trabajar position:sticky. Así que me gustaría publicar mi código de muestra aquí.

CSS

/* Set table width/height as you want.*/
div.freeze-header {
  position: relative;
  max-height: 150px;
  max-width: 400px;
  overflow:auto;
}

/* Use position:sticky to freeze header on top*/
div.freeze-header > table > thead > tr > th {
  position: sticky;
  top: 0;
  background-color:yellow;
}

/* below is just table style decoration.*/
div.freeze-header > table {
  border-collapse: collapse;
}

div.freeze-header > table td {
  border: 1px solid black;
}

HTML

<html>
<body>
  <div>
   other contents ...
  </div>
  <div>
   other contents ...
  </div>
  <div>
   other contents ...
  </div>

  <div class="freeze-header">
    <table>
       <thead>
         <tr>
           <th> header 1 </th>
           <th> header 2 </th>
           <th> header 3 </th>
           <th> header 4 </th>
           <th> header 5 </th>
           <th> header 6 </th>
           <th> header 7 </th>
           <th> header 8 </th>
           <th> header 9 </th>
           <th> header 10 </th>
           <th> header 11 </th>
           <th> header 12 </th>
           <th> header 13 </th>
           <th> header 14 </th>
           <th> header 15 </th>
          </tr>
       </thead>
       <tbody>
         <tr>
           <td> data 1 </td>
           <td> data 2 </td>
           <td> data 3 </td>
           <td> data 4 </td>
           <td> data 5 </td>
           <td> data 6 </td>
           <td> data 7 </td>
           <td> data 8 </td>
           <td> data 9 </td>
           <td> data 10 </td>
           <td> data 11 </td>
           <td> data 12 </td>
           <td> data 13 </td>
           <td> data 14 </td>
           <td> data 15 </td>
          </tr>         
         <tr>
           <td> data 1 </td>
           <td> data 2 </td>
           <td> data 3 </td>
           <td> data 4 </td>
           <td> data 5 </td>
           <td> data 6 </td>
           <td> data 7 </td>
           <td> data 8 </td>
           <td> data 9 </td>
           <td> data 10 </td>
           <td> data 11 </td>
           <td> data 12 </td>
           <td> data 13 </td>
           <td> data 14 </td>
           <td> data 15 </td>
          </tr>         
         <tr>
           <td> data 1 </td>
           <td> data 2 </td>
           <td> data 3 </td>
           <td> data 4 </td>
           <td> data 5 </td>
           <td> data 6 </td>
           <td> data 7 </td>
           <td> data 8 </td>
           <td> data 9 </td>
           <td> data 10 </td>
           <td> data 11 </td>
           <td> data 12 </td>
           <td> data 13 </td>
           <td> data 14 </td>
           <td> data 15 </td>
          </tr>         
         <tr>
           <td> data 1 </td>
           <td> data 2 </td>
           <td> data 3 </td>
           <td> data 4 </td>
           <td> data 5 </td>
           <td> data 6 </td>
           <td> data 7 </td>
           <td> data 8 </td>
           <td> data 9 </td>
           <td> data 10 </td>
           <td> data 11 </td>
           <td> data 12 </td>
           <td> data 13 </td>
           <td> data 14 </td>
           <td> data 15 </td>
          </tr>         
         <tr>
           <td> data 1 </td>
           <td> data 2 </td>
           <td> data 3 </td>
           <td> data 4 </td>
           <td> data 5 </td>
           <td> data 6 </td>
           <td> data 7 </td>
           <td> data 8 </td>
           <td> data 9 </td>
           <td> data 10 </td>
           <td> data 11 </td>
           <td> data 12 </td>
           <td> data 13 </td>
           <td> data 14 </td>
           <td> data 15 </td>
          </tr>         
         <tr>
           <td> data 1 </td>
           <td> data 2 </td>
           <td> data 3 </td>
           <td> data 4 </td>
           <td> data 5 </td>
           <td> data 6 </td>
           <td> data 7 </td>
           <td> data 8 </td>
           <td> data 9 </td>
           <td> data 10 </td>
           <td> data 11 </td>
           <td> data 12 </td>
           <td> data 13 </td>
           <td> data 14 </td>
           <td> data 15 </td>
          </tr>         
         <tr>
           <td> data 1 </td>
           <td> data 2 </td>
           <td> data 3 </td>
           <td> data 4 </td>
           <td> data 5 </td>
           <td> data 6 </td>
           <td> data 7 </td>
           <td> data 8 </td>
           <td> data 9 </td>
           <td> data 10 </td>
           <td> data 11 </td>
           <td> data 12 </td>
           <td> data 13 </td>
           <td> data 14 </td>
           <td> data 15 </td>
          </tr>         
         <tr>
           <td> data 1 </td>
           <td> data 2 </td>
           <td> data 3 </td>
           <td> data 4 </td>
           <td> data 5 </td>
           <td> data 6 </td>
           <td> data 7 </td>
           <td> data 8 </td>
           <td> data 9 </td>
           <td> data 10 </td>
           <td> data 11 </td>
           <td> data 12 </td>
           <td> data 13 </td>
           <td> data 14 </td>
           <td> data 15 </td>
          </tr>         
       </tbody>
    </table>
  </div>
</body>
</html>

Manifestación

ingrese la descripción de la imagen aquí

Al usar nuestro sitio, usted reconoce que ha leído y comprende nuestra Política de Cookies y Política de Privacidad.
Licensed under cc by-sa 3.0 with attribution required.