Заголовки таблиц HTML всегда видны в верхней части окна при просмотре большой таблицы - PullRequest
165 голосов
/ 23 июня 2009

Я хотел бы иметь возможность «настроить» презентацию таблицы HTML, чтобы добавить одну функцию: при прокрутке страницы вниз, чтобы таблица находилась на экране, но строки заголовков были вне экрана, мне бы хотелось, чтобы заголовки остаются видимыми в верхней части области просмотра.

Концептуально это будет похоже на функцию «стоп-панели» в Excel. Однако HTML-страница может содержать несколько таблиц, и я хотел бы, чтобы это происходило только для той таблицы, которая просматривается в данный момент, только пока она находится в поле зрения.

Примечание. Я видел одно решение, в котором область данных таблицы сделана прокручиваемой, а заголовки не прокручиваются. Это не решение, которое я ищу.

Ответы [ 11 ]

135 голосов
/ 25 июня 2009

Я разработал проверочное решение, используя jQuery .

Посмотреть образец здесь.

Теперь я получил этот код в Mercurial Bitbucket хранилище . Основной файл tables.html.

Мне известна одна проблема, связанная с этим: если таблица содержит якоря и если вы открываете URL-адрес с указанным якорем в браузере, при загрузке страницы строка с этим якорем, вероятно, будет скрыта плавающей точкой заголовок.

Обновление 2017-12-11: Я вижу, что это не работает с текущими Firefox (57) и Chrome (63). Не уверен, когда и почему это перестало работать, или как это исправить. Но теперь я думаю, что принятый ответ Хенди Ираван выше.

68 голосов
/ 11 октября 2011

Крейг, я немного усовершенствовал ваш код (среди прочего он теперь использует position: fixed) и обернул его как плагин jQuery.

Попробуйте здесь: http://jsfiddle.net/jmosbech/stFcx/

И получите источник здесь: https://github.com/jmosbech/StickyTableHeaders

65 голосов
/ 02 октября 2013

Проверьте jQuery.floatThead ( доступно демо ), что очень круто, может работать с DataTables и даже может работать внутри overflow: auto контейнера.

7 голосов
/ 22 мая 2018

Если вы ориентируетесь на современные css3-совместимые браузеры ( Поддержка браузеров: https://caniuse.com/#feat=css-sticky), вы можете использовать position:sticky, который не требует JS и не нарушит компоновку таблицы Выравнивание th и td одного и того же столбца, а также не требует фиксированной ширины столбца для правильной работы.

Пример для отдельной строки заголовка:

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

Для жаб с 1 или 2 рядами вы можете использовать что-то вроде этого:

thead > :last-child th
{
    position: sticky;
    top: 30px; /* This is for all the the "th" elements in the second row, (in this casa is the last child element into the thead) */
}

thead > :first-child th
{
    position: sticky;
    top: 0px; /* This is for all the the "th" elements in the first child row */
}

Возможно, вам придется немного поиграть со свойством top последнего потомка, изменив количество пикселей в соответствии с высотой первой строки (+ поле + граница + граница, если есть) ), поэтому второй ряд придерживается ниже первого.

Также оба решения работают, даже если у вас есть более одной таблицы на одной странице: элемент th каждого из них начинает зависать, когда его верхняя позиция совпадает с указанным в определении CSS, и просто исчезает, когда все стол прокручивается вниз. Так что если есть еще несколько таблиц, все прекрасно работают одинаково.

Зачем использовать последний ребенок до и первый ребенок после в css?

Поскольку правила CSS отображаются браузером в том же порядке, в каком вы записываете их в файл CSS, и из-за этого, если у вас есть только 1 строка в элементе thead, первая строка одновременно является последней строкой, а первая - дочернее правило должно переопределить последнее-дочернее. Если нет, у вас будет смещение строки в 30 пикселей от верхнего поля, которое, я полагаю, вам не нужно.

Известная проблема положения: залипание заключается в том, что он не работает с элементами thead или строками таблицы: вы должны нацеливать th-ые элементы. Эта проблема будет решена в будущих версиях браузера.

5 голосов
/ 02 июля 2009

Возможные альтернативы

JS-плавающей таблицы заголовки

js-плавающие заголовки таблиц (Google Code)

В Drupal

У меня есть сайт Drupal 6. Я был на странице администратора "модули", и заметил, что таблицы имеют именно эту функцию!

Глядя на код, кажется, что он реализован файлом с именем tableheader.js. Применяет эту функцию ко всем таблицам с классом sticky-enabled.

Для сайта Drupal я бы хотел использовать этот модуль tableheader.js как есть для пользовательского контента. tableheader.js не отображается на страницах с пользовательским контентом в Drupal. Я разместил сообщение на форуме , чтобы спросить, как изменить тему Drupal, чтобы она была доступна. Согласно ответу, tableheader.js можно добавить в тему Drupal, используя drupal_add_js() в теме template.php следующим образом:

drupal_add_js('misc/tableheader.js', 'core');
4 голосов
/ 23 июня 2009

Я столкнулся с этой проблемой совсем недавно. К сожалению, мне пришлось сделать 2 таблицы, одну для заголовка и одну для тела. Вероятно, это не самый лучший подход, но вот так:

<html>
<head>
    <title>oh hai</title>
</head>
<body>
    <table id="tableHeader">
        <tr>
            <th style="width:100px; background-color:#CCCCCC">col header</th>
            <th style="width:100px; background-color:#CCCCCC">col header</th>
        </tr>
    </table>
    <div style="height:50px; overflow:auto; width:250px">
        <table>
            <tr>
                <td style="height:50px; width:100px; background-color:#DDDDDD">data1</td>
                <td style="height:50px; width:100px; background-color:#DDDDDD">data1</td>
            </tr>
            <tr>
                <td style="height:50px; width:100px; background-color:#DDDDDD">data2</td>
                <td style="height:50px; width:100px; background-color:#DDDDDD">data2</td>
            </tr>
        </table>
    </div>
</body>
</html>

Это сработало для меня, возможно, это не элегантный способ, но он работает. Я буду исследовать, поэтому посмотрим, смогу ли я сделать что-то лучше, но это позволяет использовать несколько таблиц.

Прочтите о переполнении , чтобы понять, подходит ли оно вам

3 голосов
/ 22 августа 2012

Это действительно сложно иметь липкий заголовок на вашем столе. У меня было такое же требование, но с asp: GridView , а потом я обнаружил, что действительно думал, что у gridview есть липкий заголовок. Есть много доступных решений, и я потратил 3 дня, чтобы попробовать все решения, но ни одно из них не могло удовлетворить.

Основной проблемой, с которой я столкнулся в большинстве этих решений, была проблема выравнивания. Когда вы пытаетесь сделать заголовок плавающим, каким-то образом выравнивание ячеек заголовка и ячеек тела сбивается с пути.

В некоторых решениях у меня также возникла проблема перекрытия заголовка с первыми несколькими строками тела, из-за чего строки тела скрывались за плавающим заголовком.

Так что теперь мне пришлось реализовать свою собственную логику, чтобы достичь этого, хотя я также не считаю это идеальным решением, но это также может быть полезно для кого-то,

Ниже приведен пример таблицы.

<div class="table-holder">
        <table id="MyTable" cellpadding="4" cellspacing="0" border="1px" class="customerTable">
            <thead>
                <tr><th>ID</th><th>First Name</th><th>Last Name</th><th>DOB</th><th>Place</th></tr>
            </thead>
            <tbody>
                <tr><td>1</td><td>Customer1</td><td>LastName</td><td>1-1-1</td><td>SUN</td></tr>
                <tr><td>2</td><td>Customer2</td><td>LastName</td><td>2-2-2</td><td>Earth</td></tr>
                <tr><td>3</td><td>Customer3</td><td>LastName</td><td>3-3-3</td><td>Mars</td></tr>
                <tr><td>4</td><td>Customer4</td><td>LastName</td><td>4-4-4</td><td>Venus</td></tr>
                <tr><td>5</td><td>Customer5</td><td>LastName</td><td>5-5-5</td><td>Saturn</td></tr>
                <tr><td>6</td><td>Customer6</td><td>LastName</td><td>6-6-6</td><td>Jupitor</td></tr>
                <tr><td>7</td><td>Customer7</td><td>LastName</td><td>7-7-7</td><td>Mercury</td></tr>
                <tr><td>8</td><td>Customer8</td><td>LastName</td><td>8-8-8</td><td>Moon</td></tr>
                <tr><td>9</td><td>Customer9</td><td>LastName</td><td>9-9-9</td><td>Uranus</td></tr>
                <tr><td>10</td><td>Customer10</td><td>LastName</td><td>10-10-10</td><td>Neptune</td></tr>
            </tbody>
        </table>
    </div>

Примечание : таблица помещается в DIV с атрибутом класса, равным «table-holder».

Ниже приведен скрипт JQuery, который я добавил в заголовок своей HTML-страницы.

<script src="../Scripts/jquery-1.7.2.min.js" type="text/javascript"></script>
<script src="../Scripts/jquery-ui.min.js" type="text/javascript"></script>
<script type="text/javascript">
    $(document).ready(function () {
        //create var for table holder
        var originalTableHolder = $(".table-holder");
        // set the table holder's with
        originalTableHolder.width($('table', originalTableHolder).width() + 17);
        // Create a clone of table holder DIV
        var clonedtableHolder = originalTableHolder.clone();

        // Calculate height of all header rows.
        var headerHeight = 0;
        $('thead', originalTableHolder).each(function (index, element) {
            headerHeight = headerHeight + $(element).height();
        });

        // Set the position of cloned table so that cloned table overlapped the original
        clonedtableHolder.css('position', 'relative');
        clonedtableHolder.css('top', headerHeight + 'px');

        // Set the height of cloned header equal to header height only so that body is not visible of cloned header
        clonedtableHolder.height(headerHeight);
        clonedtableHolder.css('overflow', 'hidden');

        // reset the ID attribute of each element in cloned table
        $('*', clonedtableHolder).each(function (index, element) {
            if ($(element).attr('id')) {
                $(element).attr('id', $(element).attr('id') + '_Cloned');
            }
        });

        originalTableHolder.css('border-bottom', '1px solid #aaa');

        // Place the cloned table holder before original one
        originalTableHolder.before(clonedtableHolder);
    });
</script>

и, наконец, ниже класс CSS для цели раскраски.

.table-holder
{
    height:200px;
    overflow:auto;
    border-width:0px;    
}

.customerTable thead
{
    background: #4b6c9e;        
    color:White;
}

Таким образом, вся идея этой логики состоит в том, чтобы поместить таблицу в div держателя таблицы и создать клон этого держателя на стороне клиента при загрузке страницы. Теперь скройте тело таблицы внутри держателя клона и поместите оставшуюся часть заголовка в исходный заголовок.

То же решение работает и для asp: gridview, вам нужно добавить еще два шага для достижения этой цели в gridview,

  1. В событии OnPrerender объекта gridview на веб-странице задайте для раздела таблицы строки заголовка значение TableHeader.

    if (this.HeaderRow != null)
    {
        this.HeaderRow.TableSection = TableRowSection.TableHeader;
    }
    
  2. И оберните свою сетку в <div class="table-holder"></div>.

Примечание : если в вашем заголовке есть интерактивные элементы управления, вам может потребоваться добавить еще несколько сценариев jQuery для передачи событий, вызванных клонированным заголовком, в исходный заголовок. Этот код уже доступен в плагине jQuery sticky-header, создаваемом jmosbech

2 голосов
/ 23 июня 2009

Использование display: fixed в разделе thead должно работать, но для того, чтобы оно работало только в текущей видимой таблице, вам потребуется помощь JavaScript. И это будет непросто, потому что ему нужно будет определить места прокрутки и расположение элементов относительно области просмотра, что является одной из главных областей несовместимости браузера.

Посмотрите на популярные JavaScript-фреймворки (jQuery, MooTools, YUI и т. Д.), Чтобы узнать, могут ли они делать то, что вы хотите, или облегчать выполнение того, что вы хотите.

2 голосов
/ 23 июня 2009

Если вы используете полноэкранную таблицу, вас может заинтересовать установка отображения th: исправлено; и верх: 0; или попробуйте очень похожий подход через CSS.

Обновление

Просто быстро создайте рабочее решение с помощью iframes (html4.0). Этот пример НЕ соответствует стандарту, однако вы легко сможете это исправить:

outer.html

<?xml version='1.0' encoding='UTF-8'?>
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.1//EN" "http://www.w3.org/TR/xhtml11/DTD/xhtml11.dtd">   
<html xmlns="http://www.w3.org/1999/xhtml" xml:lang="en">     
    <head>      
        <meta http-equiv="Content-Type" content="text/html; charset=UTF-8" />   
        <title>Outer</title>
  <body>
    <iframe src="test.html" width="200" height="100"></iframe>
    </body>
</html> 

test.html

<?xml version='1.0' encoding='UTF-8'?>
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.1//EN" "http://www.w3.org/TR/xhtml11/DTD/xhtml11.dtd">   
<html xmlns="http://www.w3.org/1999/xhtml" xml:lang="en">     
    <head>      
        <meta http-equiv="Content-Type" content="text/html; charset=UTF-8" />   
        <title>Floating</title>
    <style type="text/css">
      .content{
        position:relative; 
      }

      thead{
        background-color:red;
        position:fixed; 
        top:0;
      }
    </style>
  <body>
    <div class="content">      
      <table>
        <thead>
          <tr class="top"><td>Title</td></tr>
        </head>
        <tbody>
          <tr><td>a</td></tr>
          <tr><td>b</td></tr>
          <tr><td>c</td></tr>
          <tr><td>d</td></tr>
          <tr><td>e</td></tr>
          <tr><td>e</td></tr>
          <tr><td>e</td></tr>
          <tr><td>e</td></tr>
          <tr><td>e</td></tr>
          <tr><td>e</td></tr>
        </tbody>
      </table>
    </div>
    </body>
</html> 
0 голосов
/ 15 декабря 2011

Обидно, что то, что прекрасно работает в одном браузере, не работает в других.Следующее работает в Firefox, но не в Chrome или IE:

<table width="80%">

 <thead>

 <tr>
  <th>Column 1</th>
  <th>Column 2</th>
  <th>Column 3</th>
 </tr>

 </thead>

 <tbody style="height:50px; overflow:auto">

  <tr>
    <td>Cell A1</td>
    <td>Cell B1</td>
    <td>Cell C1</td>
  </tr>

  <tr>
    <td>Cell A2</td>
    <td>Cell B2</td>
    <td>Cell C2</td>
  </tr>

  <tr>
    <td>Cell A3</td>
    <td>Cell B3</td>
    <td>Cell C3</td>
  </tr>

 </tbody>

</table>
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...