Tabella HTML con intestazioni fisse?


231

Esiste una tecnica CSS / JavaScript tra browser per visualizzare una tabella HTML lunga in modo che le intestazioni di colonna rimangano fisse sullo schermo e non scorrano con il corpo della tabella. Pensa all'effetto "Blocca riquadri" in Microsoft Excel.

Voglio essere in grado di scorrere i contenuti della tabella, ma di poter sempre vedere le intestazioni delle colonne in alto.


3
Prova questo: Tabella scorrevole CSS pura con intestazione fissa EDIT : questa dovrebbe funzionare in Internet Explorer 7 come mostrato nell'esempio : Scorrimento della tabella HTML con intestazione fissa EDIT 2: ho trovato un paio di collegamenti extra che potrebbero essere utili: - Stupido header fisso - Un plugin jQuery con alcune limitazioni. - [Fixed Table Headers] ( cross-browser.com/x/examp
gcores

Ho trovato molte soluzioni che generalmente funzionano, ma nessuna di esse ha funzionato div a scorrimento. Voglio dire, il tuo tavolo è all'interno di un div scorrevole e comunque vuoi che l'intestazione del tuo tavolo sia ancora dentro quel div. L'ho risolto e condivido la soluzione qui .
Yogee,

9
Nel 2018, tutti i browser possono utilizzare la seguente soluzione semplice: thead th { position: sticky; top: 0; }. Safari ha bisogno di un prefisso fornitore:-webkit-sticky
Daniel Waltrip,

1
@DanielWaltrip dovresti aggiungere una risposta in modo che possa essere votato al primo posto - tutte le altre risposte sono ridondanti con la posizione: appiccicoso è un supporto migliore al giorno d'oggi
Peter Kerr

Risposte:


88

Stavo cercando una soluzione per questo da un po 'e ho scoperto che la maggior parte delle risposte non funzionano o non sono adatte alla mia situazione, quindi ho scritto una soluzione semplice con jQuery.

Questo è lo schema della soluzione:

  1. Clonare la tabella che deve avere un'intestazione fissa e posizionare la copia clonata sopra l'originale.
  2. Rimuovere il corpo del tavolo dal tavolo superiore.
  3. Rimuovere l'intestazione della tabella dalla tabella inferiore.
  4. Regola le larghezze delle colonne. (Teniamo traccia delle larghezze delle colonne originali)

Di seguito è riportato il codice in una demo eseguibile.

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>

Questa soluzione funziona in Chrome e IE. Poiché si basa su jQuery, questo dovrebbe funzionare anche con altri browser supportati da jQuery.


4
e come possiamo risolvere il problema quando il contenuto è maggiore della larghezza?
Maertz,

1
@tetra td {larghezza massima: 30px; } questo ti permetterà allo sviluppatore di controllare come vengono visualizzate le righe.
Lyuben Todorov,

Ma cosa succede se i contenuti in alcune celle di intestazione sono più lunghi rispetto alle celle td? L'ho provato in IE7, e width () rompe tutto. IE8 e IE9 funzionano bene, però ...
JustAMartin

4
Sfortunatamente, se hai bisogno di un allineamento pixel perfetto delle colonne, questo non funziona: jsbin.com/elekiq/1 ( codice sorgente ). Puoi vedere che alcune intestazioni sono sfalsate rispetto a dove dovrebbero essere, solo leggermente. L'effetto è più evidente se si utilizzano gli sfondi: jsbin.com/elekiq/2 ( codice sorgente ). (Lavoravo seguendo queste stesse linee, mi sono imbattuto in questo nel mio codice, ho trovato il tuo e ho pensato "Oh, mi chiedo se lo ha risolto per me!" Purtroppo no. :-)) I browser sono TANTO un dolore nel voler controllare il larghezze di celle ...
TJ Crowder,

Questo non sembra funzionare con lo scorrimento orizzontale: crea l'intestazione, ma si estende oltre l'area scorrevole (visibilmente) e non scorre con il contenuto.
Arresto

183

Questo può essere risolto in modo pulito in quattro righe di codice.

Se ti preoccupi solo dei browser moderni, un'intestazione fissa può essere ottenuta molto più facilmente utilizzando le trasformazioni CSS. Sembra strano, ma funziona benissimo:

  • HTML e CSS rimangono così come sono.
  • Nessuna dipendenza JavaScript esterna.
  • Quattro righe di codice.
  • Funziona con tutte le configurazioni (layout tabella: fisso, ecc.).
document.getElementById("wrap").addEventListener("scroll", function(){
   var translate = "translate(0,"+this.scrollTop+"px)";
   this.querySelector("thead").style.transform = translate;
});

Il supporto per le trasformazioni CSS è ampiamente disponibile eccezione di Internet Explorer 8-.

Ecco l'esempio completo di riferimento:


8
Devo dire che, nonostante il mio commento precedente, questa è la cosa più vicina a una soluzione perfetta che abbia mai visto. Anche lo scorrimento orizzontale è perfetto (meglio della mia soluzione). Ecco un esempio con i bordi (non è possibile utilizzare border-collapse) e una barra di scorrimento che si attacca al tavolo anziché al contenitore: jsfiddle
DoctorDestructo

11
Scoperto, funziona ma deve applicare la trasformazione a th / td, non a thead.
rossa il

5
@AlexAlexeev, la tua soluzione è sorprendente. Grazie. Ho notato che, l'intestazione fissa risultante non ha le linee di confine che distinguono le colonne. Lo stile CSS predefinito viene perso. Anche quando includo questo ... $(this).addClass('border')cambia il resto della tabella presenta caratteri, dimensioni, colore che passo nella classe del bordo. Tuttavia, non aggiunge righe all'intestazione fissa. Apprezzo, qualsiasi input su come risolvere questo problema
user5249203

5
@ user5249203 So che te l'hai chiesto qualche mese fa ma ho avuto lo stesso problema ed era dovuto al collasso dei bordi: vedi questo: stackoverflow.com/questions/33777751/… .
Archz,

6
Questo non funziona in nessuna versione di IE o in Edge. Ecco una versione basata sul commento di @ redhead jsfiddle.net/n6o8ocwb/2
rob

58

Ho appena completato la compilazione di un plug-in jQuery che prenderà una singola tabella valida utilizzando HTML valido (deve avere un thead e tbody) e produrrà una tabella con intestazioni fisse, piè di pagina fisso opzionale che può essere un'intestazione clonata o qualsiasi contenuto che hai scelto (impaginazione, ecc.). Se vuoi sfruttare monitor più grandi, ridimensionerà anche la tabella quando il browser viene ridimensionato. Un'altra caratteristica aggiunta è la possibilità di scorrere lateralmente se le colonne della tabella non sono tutte in vista.

http://fixedheadertable.com/

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

È estremamente facile da configurare e puoi creare i tuoi stili personalizzati per questo. Utilizza anche gli angoli arrotondati in tutti i browser. Tieni a mente che l'ho appena rilasciato, quindi è ancora tecnicamente beta e ci sono pochissimi problemi minori che sto risolvendo.

Funziona in Internet Explorer 7, Internet Explorer 8, Safari, Firefox e Chrome.


Grazie! Oggi aggiungerò una nuova versione quando torno a casa dal lavoro. Ecco un link al mio post sul blog con quello che sto aggiungendo: fixedheadertable.mmalek.com/2009/10/07/…
Segna il

Grazie per questo. So che questa domanda ha più di un anno, ma anche a rischio di risvegliare il limo, vorrei dirti che il tuo lavoro è apprezzato
sova,

Nella tua demo, le larghezze sono disattivate in ie6 :-( l'intestazione e il corpo della tabella non sono allineati.
Cheekysoft,

4
L'ultima versione non funziona in IE6. Non supporto più IE6.
Segna il

ottimo lavoro Mark - sfortunatamente ci sono alcuni problemi con lo scorrimento dell'intestazione e della colonna fisse nei dispositivi mobili (iPad, tablet Android) - quando scorro il contenuto quelle parti fisse non scorrono - quando smetto di scorrere e tocco una volta il tavolo , le parti fisse "saltano" nelle posizioni corrette: esiste un modo semplice per risolvere questo problema?
Okizb,

23

Ho anche creato un plug-in che risolve questo problema. Il mio progetto - jQuery.floatThead è in circolazione da oltre 4 anni ed è molto maturo.

Non richiede stili esterni e non si aspetta che il tuo tavolo sia disegnato in modo particolare. Supporta Internet Explorer9 + e Firefox / Chrome.

Attualmente (2018-05) ha:

405 commit e 998 stelle su GitHub


Molte (non tutte) delle risposte qui sono trucchi rapidi che potrebbero aver risolto il problema che una persona stava avendo, ma non funzionerà per ogni tavolo.

Alcuni degli altri plugin sono vecchi e probabilmente funzionano benissimo con Internet Explorer, ma si romperanno su Firefox e Chrome.


1
Ottimo plugin, supporta tabelle nidificate e offset.
Mihai Alex,

2
Grande. Molte grazie. Il plugin ha funzionato bene in Firefox 45.2, Chromium 51 e IE 11. Inoltre, non interferisce con molti codici JS e jQuery creati sulla stessa pagina.
Aldo Paradiso,

Grazie. Sono felice di segnalare che a questo punto il progetto ottiene circa 1 segnalazione di bug ogni 4 mesi. Non sto apportando molte modifiche sostanziali. È abbastanza solido e funziona.
mkoryak,

20

TL; DR

Se scegli come target i browser moderni e non hai esigenze di stile stravaganti: http://jsfiddle.net/dPixie/byB9d/3/ ... Anche se le quattro grandi versioni è piuttosto dolce, questa versione gestisce la larghezza del fluido molto meglio.

Buone notizie a tutti!

Con i progressi di HTML5 e CSS3 questo è ora possibile, almeno per i browser moderni. L'implementazione leggermente hackish che mi è venuta in mente può essere trovata qui: http://jsfiddle.net/dPixie/byB9d/3/ . L'ho testato in FX 25, Chrome 31 e IE 10 ...

HTML pertinente (inserisci un doctype HTML5 nella parte superiore del 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>

Ma come?!

In poche parole, hai un'intestazione di tabella, che nascondi visivamente rendendola 0px alta, che contiene anche div usati come intestazione fissa. Il contenitore della tabella lascia abbastanza spazio in alto per consentire l'intestazione assolutamente posizionata e la tabella con le barre di scorrimento appare come ci si aspetterebbe.

Il codice sopra usa la classe posizionata per posizionare assolutamente la tabella (la sto usando in una finestra di dialogo in stile popup) ma puoi usarla anche nel flusso del documento rimuovendo il positioned classe dal contenitore.

Ma ...

Non è perfetto Firefox si rifiuta di rendere la riga di intestazione 0px (almeno non l'ho trovata in alcun modo) ma ostinatamente la mantiene almeno 4px ... Non è un grosso problema, ma a seconda del tuo stile si rovinerà con i tuoi bordi ecc.

La tabella utilizza anche un approccio a colonne finte in cui il colore di sfondo del contenitore stesso viene utilizzato come sfondo per i div dell'intestazione, che sono trasparenti.

Sommario

Tutto sommato potrebbero esserci problemi di stile a seconda delle tue esigenze, in particolare bordi o sfondi complicati. Potrebbero esserci anche problemi con la calcolabilità, non l'ho ancora verificato in un'ampia varietà di browser (per favore commenta le tue esperienze se lo provi), ma non ho trovato nulla di simile, quindi ho pensato che valesse la pena pubblicare Comunque ...


Se riduci la larghezza della finestra finché non inizia lo scorrimento orizzontale, l'intestazione non scorre orizzontalmente con il corpo. Darn.
dlaliberte,

@dlaliberte - Bene, dato che l'intestazione e la tabella sono in realtà due diversi elementi che puoi, spesso, entrare nella stranezza. Ma il mio esempio non consente l'overflow delle colonne della tabella e le intestazioni sono in genere più facili da controllare rispetto al contenuto della tabella. Detto questo, se causi lo "overflow" dell'intestazione, se rimarrà a destra della tabella e sembrerà gravemente rotto. Potresti risolvere questo problema impostando una larghezza minima sul tavolo, costringendolo a traboccare anche la pagina ... Ma è un trucco quindi non sarà mai perfetto ...
Jonas Schubert Erlandsson,

1
Vale la pena sottolineare che ciò richiede un design in cui è possibile specificare una tabella ad altezza fissa.
Cheekysoft,

1
@Cheekysoft - No, il contenuto della tabella e delle righe può fluire liberamente. Il contenitore, nel mio esempio l' <section>elemento, deve essere limitato in altezza solo per forzarlo a traboccare e mostrare lo scorrimento. Qualsiasi layout che renderà l'overflow del contenitore funzionerebbe. Se trovi un caso in cui non lo è, per favore pubblica un link a un violino.
Jonas Schubert Erlandsson,

Il padding-topvalore hardcoded indica anche che se il testo dell'intestazione della tabella si trova su più di una riga, verrà visualizzato in cima alle celle della tabella. Peccato, perché questo funziona come un incantesimo il più delle volte. Davvero un bel trucco con il divin thper aggirare il problema di dimensionamento della colonna che la maggior parte delle altre soluzioni ha.
Bernhard Hofmann,

19

Tutti i tentativi di risolvere questo problema al di fuori delle specifiche CSS sono ombre pallide di ciò che vogliamo veramente: consegna sulla promessa implicita di THEAD.

Questo problema relativo alle intestazioni bloccate per un tavolo è stato a lungo una ferita aperta in HTML / CSS.

In un mondo perfetto, ci sarebbe una soluzione CSS pura per questo problema. Sfortunatamente, non sembra esserci una buona soluzione.

Le discussioni sugli standard pertinenti su questo argomento includono:

AGGIORNAMENTO : Firefox è stato spedito position:stickynella versione 32. Tutti vincono!


Sarebbe bello avere colonne di volpe allo stesso modo
Csaba Toth,

4
Ri. Firefox e posizione: appiccicoso, non funziona per le intestazioni di tabella: bugzilla.mozilla.org/show_bug.cgi?id=925259#c8 ... La patch per quel bug afferma esplicitamente: "Al momento non supportiamo il posizionamento relativo degli elementi interni del tavolo, quindi li escludiamo anche dal posizionamento appiccicoso ".
Jonas Schubert Erlandsson,

2
Questo funziona in tutti i browser oggi: thead th { position: sticky; top: 0; }. Possiamo aggiornare questa risposta per affermarlo chiaramente?
Daniel Waltrip,

1
@DanielWaltrip tutti i browser? stackoverflow.com/a/37646284/3640407 Ci sono ancora più MSIE che bordi
edc65,

Punto valido. È supportato per l'86% degli utenti Web globali, secondo caniuse.com/#search=position%3Asticky
Daniel Waltrip

14

Ecco un plugin jQuery per le intestazioni di tabella fisse. Consente a tutta la pagina di scorrere, bloccando l'intestazione quando raggiunge la cima. Funziona bene con le tabelle Bootstrap di Twitter .

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

Essa non scorre solo il contenuto della tabella. Cerca altri strumenti per questo, come una di queste altre risposte. Decidi tu cosa si adatta meglio al tuo caso.


1
Peccato: il link di esempio è morto. "Oops! Denne siden ble ikke funnet ..." Vorrei che il codice fosse incollato qui.
JosephK,

si ... mi dispiace per quello. rimosso il collegamento. Il progetto non è più mantenuto
oma

Non preoccuparti - ho provato diverse di queste presunte soluzioni prefabbricate - nessuna ha funzionato con una tabella flex-col-width che ha superato la larghezza dello schermo. Ho finito per scrivere la mia soluzione.
JosephK,

9

La maggior parte delle soluzioni pubblicate qui richiede jQuery. Se stai cercando una soluzione indipendente dal framework prova Grid: http://www.matts411.com/post/grid/

È ospitato su Github qui: https://github.com/mmurph211/Grid

Non solo supporta intestazioni fisse, ma supporta anche colonne e piè di pagina fissi a sinistra, tra le altre cose.


Questo è davvero pulito se soddisfa le tue esigenze, ci ho appena giocato oggi. Sfortunatamente, è piuttosto una griglia rettangolare (come suggerisce il nome, in realtà) e non una vera tabella con l'altezza della riga regolata dal contenuto. E lo stile delle singole file sembrava difficile. Non sono riuscito a creare un tavolo zebrato, ma non ho provato molto perché le mie esigenze erano in realtà più complesse. Comunque, bel lavoro.
mplwork

1
Ehi ti conosco! Sembrava che
avessimo

9

La proprietà CSS position: stickyha un grande supporto nella maggior parte dei browser moderni (ho avuto problemi con Edge, vedi sotto).

Questo ci consente di risolvere abbastanza facilmente il problema delle intestazioni fisse:

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

Safari ha bisogno di un prefisso fornitore: -webkit-sticky.

Per Firefox, ho dovuto aggiungere min-height: 0a uno gli elementi principali. Dimentico esattamente perché fosse necessario.

Sfortunatamente, l'implementazione di Microsoft Edge sembra essere solo semi-funzionante. Almeno, ho avuto delle celle di tabella tremolanti e disallineate nei miei test. Il tavolo era ancora utilizzabile, ma presentava importanti problemi estetici.


Utilizzando position: sticky;la tabella all'interno di un div che ha overflow: scroll;, overflow-x: scroll;o overflow-y: scroll;. sembra essere la soluzione migliore e più semplice per le intestazioni e le colonne di tabelle fisse nei browser moderni. Questa risposta deve essere votata all'inizio.
Aberrant,

È semplice ma efficace. È quello che sto cercando. Grazie.
Catbuilts

6

Una tabella di scorrimento CSS pura più raffinata

Tutte le pure soluzioni CSS che ho visto finora, per quanto intelligenti possano essere, mancano di un certo livello di polacco, o semplicemente non funzionano bene in alcune situazioni. Quindi, ho deciso di creare il mio ...

Caratteristiche:

  • È puro CSS, quindi non è necessario jQuery (o qualsiasi codice JavaScript, del resto)
  • Puoi impostare la larghezza della tabella su una percentuale (aka "fluido") o su un valore fisso, oppure lasciare che il contenuto ne determini la larghezza (aka "auto")
  • Le larghezze delle colonne possono anche essere fluide, fisse o automatiche.
  • Le colonne non verranno mai disallineate con le intestazioni a causa dello scorrimento orizzontale (un problema che si verifica in ogni altra soluzione basata su CSS che ho visto che non richiede larghezze fisse).
  • Compatibile con tutti i browser desktop più diffusi, incluso Internet Explorer fino alla versione 8
  • Aspetto pulito e lucido; nessun gap di 1 pixel dall'aspetto sciatto o bordi disallineati; sembra lo stesso in tutti i browser

Ecco un paio di violini che mostrano le opzioni di larghezza fluida e automatica:

  • Larghezza e altezza del fluido (si adatta alle dimensioni dello schermo): jsFiddle (Notare che la barra di scorrimento viene visualizzata solo quando necessario in questa configurazione, quindi potrebbe essere necessario ridurre il riquadro per vederlo)

  • Larghezza automatica, altezza fissa (più facile da integrare con altri contenuti): jsFiddle

La configurazione Larghezza automatica, Altezza fissa probabilmente ha più casi d'uso, quindi posterò il codice qui sotto.

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

Il metodo che ho usato per bloccare la riga di intestazione è simile a quello di d-Pixie, quindi fai riferimento al suo post per una spiegazione. C'erano una serie di bug e limitazioni con quella tecnica che potevano essere risolti solo con un sacco di CSS aggiuntivo e un contenitore div aggiuntivo o due.


Questa risposta è decisamente sottovalutata! Ho trascorso giorni cercando di far funzionare altre soluzioni per il mio caso particolarmente fastidioso. Ognuno di loro non è riuscito a rimanere allineato in un modo o nell'altro. Questo finalmente lo ha fatto! All'inizio sembra troppo complicato, ma una volta capito, fantastico. Puoi rimuovere un bel po 'di cose che non ti servono alla fine, quando non usi la larghezza del fluido, ecc.
Justin Sane,

1
@JustinSane Sono contento che ti piaccia! Immagino che la mancanza di apprezzamento sia dovuta al fatto che condivide la pagina con la straordinaria soluzione di Maximilian Hils . Se non sei contrario all'utilizzo di un po 'di JS, dovresti assolutamente provarlo.
DoctorDestructo,

Accidenti, questa è una soluzione quasi perfetta. Stavo usando comunque jQuery, ho provato a farlo funzionare prima di trovare il tuo (tramite il tuo commento a un'altra domanda). Non stavo pensando a un ascoltatore di scroll e non traduce ... Beh, dicono che ci vuole un genio per trovare soluzioni semplici ..;) Ho finito il progetto e funziona perfettamente senza js, ma lo terrò dentro mente per il futuro. Tuttavia, ti do il cappello per essere fantastico!
Justin Sane,

Un piccolo problema, ma se stai utilizzando diversi colori di sistema puoi vedere che il colore del testo non è stato impostato per nulla tranne le intestazioni ma lo sfondo della tabella ha un colore di sfondo esplicito. Ho un testo giallo su uno sfondo bianco e grigio per questa tabella.
Matt Arnold,

1
@MattArnold Risolto. Grazie per la punta!
DoctorDestructo,

5

Un semplice plugin jQuery

Questa è una variazione sulla soluzione di Mahes. Puoi chiamarlo come$('table#foo').scrollableTable();

L'idea è:

  • Dividi il theade tbodyin tableelementi separati
  • Rendi nuovamente la larghezza delle celle corrispondente
  • Avvolgi il secondo tablein adiv.scrollable
  • Usa CSS per far div.scrollablescorrere davvero

Il CSS potrebbe essere:

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

Avvertenze

  • Ovviamente, suddividere queste tabelle rende il markup meno semantico. Non sono sicuro di quale effetto ciò abbia sull'accessibilità.
  • Questo plugin non tratta piè di pagina, intestazioni multiple, ecc.
  • L'ho provato solo in Chrome versione 20.

Detto questo, funziona per i miei scopi e sei libero di prenderlo e modificarlo.

Ecco il plugin:

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
Bella sceneggiatura, questa ha funzionato meglio nel mio ambiente. Ho esteso la tua sceneggiatura con supporto a piè di pagina fisso, controlla il mio post qui sotto.
Gitaarik,

4

:)

Soluzione HTML / CSS non così pulita, ma pura.

table {
    overflow-x:scroll;
}

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

Aggiornato per IE8 + esempio JSFiddle


2
Buona soluzione, solo per citare, che queste celle sono flottanti e quindi, in base al contenuto, possono avere altezze diverse, è visibile se si impostano i bordi: jsfiddle.net/ZdeEH/15
Stano

3

Supporto per piè di pagina fisso

Ho esteso la funzione di Nathan per supportare anche un piè di pagina fisso e l'altezza massima. Inoltre, la funzione imposterà il CSS stesso e dovrai solo supportare una larghezza.

Uso:

Altezza fissa:

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

Altezza massima (se il browser supporta l'opzione CSS "altezza massima"):

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

script:

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

In qualche modo ho finito per Position:Stickylavorare bene sul mio 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
Questa è la soluzione più pulita che ho visto finora. caniuse mostra che a partire dal 5/2/2020, posizione non prefissata: appiccicoso gode del 90,06% di supporto globale. Quindi questa soluzione funziona bene in tutti i browser moderni.
AlienKevin

2

Due div, uno per l'intestazione, uno per i dati. Rendi scorrevole il div dei dati e utilizza JavaScript per impostare la larghezza delle colonne nell'intestazione in modo che corrisponda alle larghezze nei dati. Penso che le larghezze delle colonne di dati debbano essere riparate piuttosto che dinamiche.


3
Se ti interessa l'accessibilità, questo è un fallimento.
epascarello,

1
per quanto riguarda l'accessibilità, forse possiamo sostituire l'uso dei div con lo styling su <thead> e <tbody> ??
Cheekysoft,

1

Mi rendo conto che la domanda consente JavaScript, ma qui è una soluzione CSS pura che ho elaborato che consente anche alla tabella di espandersi in orizzontale. È stato testato con Internet Explorer 10 e gli ultimi browser Chrome e Firefox. Un link a jsFiddle è in fondo.

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>

E il 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

Per coloro che hanno provato la bella soluzione fornita da Maximilian Hils e non sono riusciti a farlo funzionare con Internet Explorer, ho avuto lo stesso problema (Internet Explorer 11) e ho scoperto qual era il problema.

In Internet Explorer 11 la trasformazione dello stile (almeno con translate) non funziona <THEAD>. Ho risolto questo applicando invece lo stile a tutti <TH>in un ciclo. Ha funzionato Il mio codice JavaScript è simile al seguente:

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

Nel mio caso la tabella era una GridView in ASP.NET. All'inizio ho pensato che fosse perché non aveva <THEAD>, ma anche quando l'ho forzato ad averne uno, non ha funzionato. Poi ho scoperto quello che ho scritto sopra.

È una soluzione molto bella e semplice. Su Chrome è perfetto, su Firefox un po 'a scatti e su Internet Explorer ancora più a scatti. Ma nel complesso una buona soluzione.


0

Vorrei aver trovato la soluzione di @ Mark prima, ma sono andato a scrivere la mia prima di vedere questa domanda SO ...

Il mio è un plug-in jQuery molto leggero che supporta intestazione fissa, piè di pagina, spanning colonna (colspan), ridimensionamento, scorrimento orizzontale e un numero opzionale di righe da visualizzare prima dell'inizio dello scorrimento.

jQuery.scrollTableBody (GitHub)

Fintanto che hai una tabella con appropriate <thead>, <tbody>e (facoltativo) <tfoot>, tutto ciò che devi fare è questo:

$('table').scrollTableBody();

0

Ho trovato questa soluzione alternativa: sposta la riga di intestazione in una tabella sopra la tabella con i dati:

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


funziona per piccoli tavoli, ma se si dispone di scorrimento orizzontale questa soluzione non funzionerà.
crh225

Inoltre non funzionerà correttamente poiché le colonne della tabella non si allineano. Qui stai forzando la larghezza per td ma non dobbiamo farlo ...
Ziggler

0

Applicando il plug-in StickyTableHeaders jQuery alla tabella, le intestazioni di colonna rimarranno nella parte superiore della finestra mentre scorri verso il basso.

Esempio:

$(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

Mi piace la risposta di Maximillian Hils ma ho avuto alcuni problemi:

  1. la trasformazione non funziona in Edge o IE a meno che non la applichi a th
  2. l'intestazione lampeggia durante lo scorrimento in Edge e IE
  3. la mia tabella viene caricata usando ajax, quindi volevo collegarmi all'evento scroll della finestra piuttosto che all'evento scroll del wrapper

Per eliminare lo sfarfallio, utilizzo un timeout per attendere fino a quando l'utente ha terminato lo scorrimento, quindi applico la trasformazione, quindi l'intestazione non è visibile durante lo scorrimento.

Ho anche scritto questo usando jQuery, un vantaggio del fatto che jQuery dovrebbe gestire i prefissi dei fornitori per te

    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 tabella è racchiusa in un div con la classe table-container-fixed.

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

Ho impostato border-collapse per separare perché altrimenti perdiamo i bordi durante la traduzione e rimuovo il bordo sulla tabella per interrompere la visualizzazione del contenuto appena sopra la cella in cui si trovava il bordo durante lo scorrimento.

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

Rendo lo thsfondo bianco per coprire le celle sottostanti e aggiungo un bordo che corrisponde al bordo della tabella, che è disegnato con Bootstrap e fatto scorrere fuori dalla 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

Utilizzare l'ultima versione di jQuery e includere il seguente codice JavaScript.

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

1
Questo non sembra funzionare. Forse potresti chiarire cosa vuoi che facciamo?
Chris,

1
Quale div? Stiamo parlando di tavoli qui
isapir il

0

Questa non è una soluzione esatta alla riga di intestazione fissa, ma ho creato un metodo piuttosto ingegnoso per ripetere la riga di intestazione in tutta la tabella lunga, mantenendo comunque la possibilità di ordinare.

Questa piccola opzione richiede il plug-in jQuerytablesorter . Ecco come funziona:

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>

Ovviamente, la mia tabella ha molte più righe di questa. 193 per l'esattezza, ma puoi vedere dove si ripete la riga di intestazione. La riga di intestazione ripetuta è impostata da questa funzione:

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

Molte persone sembrano cercare questa risposta. L'ho trovato sepolto in una risposta a un'altra domanda qui: Sincronizzare la larghezza della colonna tra le tabelle in due diversi frame, ecc

Tra le dozzine di metodi che ho provato, questo è l'unico metodo che ho trovato che funziona in modo affidabile per consentire di avere una tabella inferiore a scorrimento con la tabella di intestazione con le stesse larghezze.

Ecco come l'ho fatto, prima ho migliorato il jsfiddle sopra per creare questa funzione, che funziona su entrambi tde th(nel caso in cui inciampare altri che usano thper lo stile delle loro righe di intestazione).

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

Successivamente, è necessario creare due tabelle, NOTA la tabella di intestazione dovrebbe avere un extra TDper lasciare spazio nella tabella superiore per la barra di scorrimento, in questo modo:

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

Quindi fai qualcosa del tipo:

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

Questa è l'unica soluzione che ho trovato su Stack Overflow che funziona su molte domande simili che sono state postate, che funziona in tutti i miei casi.

Ad esempio, ho provato il plug-in jQuery stickytables che non funziona con durandal, e il progetto Google Code qui https://code.google.com/p/js-scroll-table-header/issues/detail?id=2

Altre soluzioni che coinvolgono la clonazione dei tavoli, hanno scarse prestazioni o succhiano e non funzionano in tutti i casi.

Non sono necessarie soluzioni troppo complesse. Basta creare due tabelle come negli esempi seguenti e chiamare la funzione setHeaderTableWidth come descritto qui e boom, il gioco è fatto .

Se questo non funziona per te, probabilmente stavi giocando con la tua proprietà di ridimensionamento della casella CSS e devi impostarla correttamente. È facile rovinare il tuo contenuto CSS per sbaglio. Ci sono molte cose che possono andare storte, quindi fai attenzione / attenzione. Questo approccio funziona per me .


0

Ecco una soluzione con cui abbiamo finito per lavorare (al fine di gestire alcuni casi limite e versioni precedenti di Internet Explorer, alla fine abbiamo anche sbiadito la barra del titolo su scroll e poi sbiadita al termine dello scorrimento, ma nei browser Firefox e WebKit questa soluzione funziona e presuppone il collasso dei confini: il collasso.

La chiave di questa soluzione è che una volta applicato il collasso dei bordi , i CSS trasformano il lavoro sull'intestazione, quindi è solo una questione di intercettare gli eventi di scorrimento e impostare correttamente la trasformazione. Non è necessario duplicare nulla. A meno che questo comportamento non sia implementato correttamente nel browser, è difficile immaginare una soluzione più leggera.

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

È implementato come un semplice plugin jQuery. Devi semplicemente rendere la tua appiccicosa appiccicosa con una chiamata come $ ('thead'). Appiccicosa (), e loro resteranno in giro. Funziona per più tabelle su una pagina e sezioni di testa a metà dei tavoli più grandi.

$.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();

buona soluzione ma come si includono i bordi delle colonne tra le colonne (entrambi nell'intestazione fissa, allineati ai dati td)?
user5249203

Non sono sicuro di aver capito il tuo problema. border-collapse non ti impedisce di usare bordi, margini, ecc., rimuove semplicemente le metriche della tabella voodoo di un tempo.
podperson

1
Aggiungi border: 2px solid red;a th, scorri e vedrai il problema. Mi sono inventato
calandoa

Aggiungi lo stesso bordo quotato a td e non ci sono problemi. Non vedo il tuo punto. La tua versione è molto più pulita e non usa jQuery, quindi andrei sicuramente con qualcosa di più simile oggi. (Anche se, francamente, non penso che
userei

0

Ecco una risposta migliorata a quella pubblicata da Maximilian Hils .

Questo funziona in Internet Explorer 11 senza sfarfallio:

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

Ho sviluppato un semplice plug-in jQuery leggero per convertire una tabella HTML ben formattata in una tabella scorrevole con intestazione e colonne fisse.

Il plugin funziona bene per abbinare il posizionamento da pixel a pixel della sezione fissa con la sezione scorrevole. Inoltre, puoi anche congelare il numero di colonne che saranno sempre visibili durante lo scorrimento in orizzontale.

Demo e documentazione: http://meetselva.github.io/fixed-table-rows-cols/

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

Di seguito è riportato l'utilizzo di una tabella semplice con un'intestazione fissa,

$(<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'}
              ]
});

Che cos'è "una tabella HTML ben" ?
Peter Mortensen,

@PeterMortensen Avrebbe dovuto essere "ben formattato HTML". A cura, grazie.
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

In aggiunta alla risposta @Daniel Waltrip. La tabella deve essere chiusa con div position: relativeper poter funzionare position:sticky. Quindi vorrei pubblicare il mio codice di esempio qui.

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>

dimostrazione

inserisci qui la descrizione dell'immagine

Utilizzando il nostro sito, riconosci di aver letto e compreso le nostre Informativa sui cookie e Informativa sulla privacy.
Licensed under cc by-sa 3.0 with attribution required.