Angular4_层级表格,表头固定

看图效果Angular4_层级表格,表头固定

描述不好,直接上代码:

html代码:

<div [ngStyle]="alertRight" class="bottom-alert topAlter">
    <ngb-alert *ngIf="successMessage" type="success" (close)="successMessage = null">{{ successMessage }}</ngb-alert>
    <ngb-alert *ngIf="errorMessage" type="danger" (close)="errorMessage = null">{{ errorMessage }}</ngb-alert>
</div>
<div class="subNav">
    <div class="container-fluid">
        <nav [type]="'loadShipment'"></nav>
    </div>
</div>
<div class="table-responsive stickyTable"  *ngIf="!noRecord">
    <table class="table table-striped" [rtList]="getHus" #list="rtList">
        <thead>
            <tr>
                <th>
                    <span></span>
                </th>
                <th>
                    <span>Handling Unit ID</span>
                </th>
                <th>
                    <span>Stage Location</span>
                </th>
                <th>
                    <span>Truck</span>
                </th>
                <th>
                    <span>Delivery Date</span>
                </th>
                <th>
                    <span>Order Type</span>
                </th>
                <th>
                    <span>Load Position</span>
                </th>
                <th  class="rightjustified" [style.top]="columnTop">
                    <span>Weight</span>
                </th>
                <th>
                    <span>Operator</span>
                </th>
            </tr>
        </thead>
        <tbody>
            <ng-template ngFor let-pallet [ngForOf]="list.items" let-i="index">
                <tr>
                    <td (click)="expandOrders(pallet)" >
                        <i [ngClass]="{'fa': true, 'fa-caret-right': !pallet.expandOrder,'fa-caret-down': pallet.expandOrder, 'fa-lg': true}"></i>
                    </td>
                    <td>{{pallet.handlingUnitId}}</td>
                    <td>{{pallet.stageLocation}}</td>
                    <td>{{pallet.carrier}}</td>
                    <td>{{pallet.deliveryDate| date:'MM/dd/yyyy'}}</td>
                    <td>{{pallet.orderCode}}</td>
                    <td>{{pallet.loadPosition}}</td>
                    <td class="rightjustified">{{pallet.weight| number:'1.3'}}</td>
                    <td>{{pallet.operator}}</td>
                </tr>
                <tr [hidden]='!pallet.expandOrder'>
                    <td ></td>
                    <td colspan="10" >
                        <table class="table table-striped" >
                            <thead>
                                <tr>
                                    <th ></th>
                                    <th >Delivery</th>
                                    <th >Customer</th>
                                    <th >Ship To</th>
                                    <th >SO Number</th>
                                    <th >Delivery Date/Time</th>
                                    <th >Truck</th>
                                </tr>
                            </thead>
                            <tbody>
                                <ng-template ngFor let-d [ngForOf]="pallet.deliveries" let-i="index">
                                    <tr>
                                        <td (click)="expandLineItems($event,pallet,d)" >
                                            <i [ngClass]="{'fa': true, 'fa-caret-right': !d.expandItem,'fa-caret-down': d.expandItem, 'fa-lg': true}"></i>
                                        </td>
                                        <td>{{d.delivery|removeLeadingZeros}}</td>
                                        <td>{{d.customer}}</td>
                                        <td>{{d.shipTo|removeLeadingZeros}}</td>
                                        <td>{{d.soNumber|removeLeadingZeros}}</td>
                                        <td>{{d.dispatchTime| date:'MM/dd/yyyy hh:mm a'}}</td>
                                        <td>{{d.carrier}}</td>
                                    </tr>
                                    <tr [hidden]='!d.expandItem'>
                                        <td ></td>
                                        <td colspan="7" >
                                            <table class="table table-striped" >
                                                <thead>
                                                    <tr>
                                                        <th  class="rightjustified">Line</th>
                                                        <th >Material</th>
                                                        <th >Description</th>
                                                        <th  class="rightjustified">Qty</th>
                                                        <th >UoM</th>
                                                        <th ></th>
                                                    </tr>
                                                </thead>
                                                <tbody>
                                                    <ng-template ngFor let-line [ngForOf]="d.lineItems" let-i="index">
                                                        <tr>
                                                            <td class="rightjustified">{{line.line}}</td>
                                                            <td>{{line.material}}</td>
                                                            <td>{{line.description}}</td>
                                                            <td class="rightjustified">{{line.qty | number:'1.3'}}</td>
                                                            <td>{{line.uoM}}</td>
                                                            <td></td>
                                                        </tr>
                                                    </ng-template>
                                                </tbody>
                                            </table>
                                        </td>
                                    </tr>
                                </ng-template>
                            </tbody>
                        </table>
                    </td>
                </tr>
            </ng-template>
        </tbody>
        <tfoot>
            <tr>
                <td colspan="11">
                    <rt-status-progress>
                        <progress-bar></progress-bar>
                    </rt-status-progress>
                    <rt-status-no-data class="text-center">
                        0 records found
                    </rt-status-no-data>
                    <div>
                        <rt-status-done>
                            <rt-buffered-pager [defaultRowCount]="20" #rtPager>
                                <span>{{rtPager.pager.loadedCount}} record{{rtPager.pager.loadedCount > 1 ? 's' : ''}}
                                    found</span>
                                <a *ngIf="rtPager.canLoadMore" tabindex="0" (click)="rtPager.loadMore()"
                                    rtPreventDefaults="click" [class.hidden]="!rtPager.canLoadMore" [rtInfinite]="window"
                                    href="#">Show more</a>
                            </rt-buffered-pager>
                        </rt-status-done>
                    </div>
                </td>
            </tr>
        </tfoot>
    </table>
</div>
<div class="text-center text-muted" *ngIf="noRecord" >
    <img src="/assets/queue.png" />
    <br>
    <br>
    <p >There are no Handling Units in this queue.</p>
</div>

css:

.btn-group {
  a {
    width: 180px;
  }
}

.wiz-container {
  background: #4b4f45;
  padding: 18px 0;
  .badge {
    background-color: orange;
    color: #fff;
    display: inline-block;
    text-align: center;
    margin-left: 10px;
  }
  .action-container {
    position: absolute;
    right: 20px;
  }
  .btn-secondary {
    box-shadow: none;
    &:first-child {
      border-top-left-radius: 20px;
      border-bottom-left-radius: 20px;
    }
    &:last-child {
      border-top-right-radius: 20px;
      border-bottom-right-radius: 20px;
    }
  }
  .btn {
    position: relative;
    background: transparent;
    color: #fff;
    border-color: #fff;
    cursor: pointer;
    overflow: hidden;
    span {
      display: block;
      position: absolute;
      top: -1.5px;
      left: -0.2px;
      border: solid transparent;
      border-left-color: #fff;
      border-width: 20px;
      z-index: 49;
      i {
        display: block;
        position: absolute;
        top: -18px;
        left: -20px;
        border: solid transparent;
        border-left-color: #4b4f45;
        border-width: 18px;
        z-index:50;
      }
    }
    &.active {
      background: #fff;
      color: #4b4f45;
      &+.btn span {
        border-left-color: #fff;
        i {
          border-left-color: #fff;
        }
      }
    }
  }
}
/*variables*/
$color-blue: #039be5;
$color-hover-blue: #06abfc;
$color-light-blue: #4fc3f7;
$color-btn-primary-focus: #67cbf8;
$color-btn-primary-active: #80d4f9;
$color-selected: #ccedfd;
$color-black: #212121;
$color-gray: #eeeeee;
$color-dark-grey: #5C707A;
$color-light-grey: #fafafa;
$color-muted-grey: #efefef;
$color-muted: #555;
$color-striped: #fcfcfc;
$color-white: #fff;
$font-medium: 14px;
$font-small: 12px;
$indent-unit: 8px;

/*variables*/
.filter-container {
  background: #4b4f45;
  .form-group>label {
    color: #FFFFFF;
    font-size: 12px;
    font-weight: normal;
  }

  select {
    border-color: #fff;
    color: #fff;
    background: url('/assets/select.png') no-repeat right center;
    appearance: none;
    -moz-appearance: none;
    -webkit-appearance: none;
    border-radius: 20px;
    font-size: 12px;
    font-weight: normal;
    option {
      background: #fff;
      color: #000;
    }
  }

  select::-ms-expand {
    display: none;
  }
}
.table-responsive {
  margin-bottom: 80px;
}

/*mixins*/
@mixin border-radius($radius) {
  -webkit-border-radius: $radius;
  -moz-border-radius: $radius;
  -ms-border-radius: $radius;
  border-radius: $radius;
}
@mixin box-shadow($shadow) {
  -webkit-box-shadow: $shadow;
  box-shadow: $shadow;
}
@mixin opacity($opacity) {
  -ms-opacity: $opacity;
  opacity: $opacity;
}
@mixin transition($transition) {
  -webkit-transition: $transition;
  -moz-transition: $transition;
  -ms-transition: $transition;
  -o-transition: $transition;
  transition: $transition;
}
@mixin img-button {
  width: 44px;
  height: 44px;
  padding: 0;
  overflow: hidden;
  font-size: 0;
  line-height: 0;
  text-indent: 999px;
  background-color: transparent;
  background-position: center center;
  background-repeat: no-repeat;
  @include box-shadow(none);
}
@mixin icon-reset() {
  background-image: url('');
}
@mixin icon-load() {
  background-image: url('');
}
@mixin icon-cancel() {
  background-image: url('');
}

/* mixins */
body,
html {
  background: $color-light-grey;
  font-size: $font-medium;
  * {
    outline: none !important;
  }
}
h1,
h2 {
  color: #049AE6;
}
h3,
h4,
h5,
h6 {
  color: $color-black;
  margin-bottom: 20px;
}
ul {
  list-style: none;
  &.bulleted {
    li {
      background: url('') no-repeat left top;
      background-repeat: no-repeat;
      background-position-y: 8px;
      padding-left: 10px;
    }
  }

  &.agenda {
    padding-left: 30px;
    ul {
      padding-left: 30px;
    }
    li {
      padding-left: 0;
    }
    @media (max-width: 768px) {
      padding-left: 15px;
      li {
        padding-top: 2px;
      }
      ul {
        padding-left: 15px;
      }
    }
  }
}
a {
  color: #147391;
}
code {
  padding: 2px;
}
section {
  margin-top: -60px;
  padding-top: 60px;
  &:target>h3,
  &:target>h4 {
    text-decoration: underline;
  }
}
aside {
  p {
    vertical-align: middle;
  }

  padding: 15px;
  margin: 20px 0 20px 25px;
  border: 1px solid $color-muted-grey;
  border-left-width: 5px;
  border-left-color: $color-blue;
  background-position: 10px center;
  background-repeat: no-repeat;
  @media (max-width: 768px) {
    margin-left: 0;
  }
}
.spinner {
  display: table;
  width: 100%;
  height: 100vh;
  .spinner-pusher {
    display: table-cell;
    text-align: center;
    vertical-align: middle;
    .spinner-wraper {
      vertical-align: middle;
      width: 100px;
      height: 100px;
      position: relative;
      margin: 50px auto;
      cursor: pointer;
      border-radius: 110px;
      -webkit-border-radius: 110px;
    }
    .spinner-body {
      background-color: $color-light-grey;
      position: absolute;
      top: 5px;
      left: 5px;
      right: 5px;
      bottom: 5px;
      z-index: 2;
      border-radius: 110px;
      text-align: center;
      font-size: 10px;
      color: $color-light-grey;
      line-height: 90px;
    }
    .spinner-line {
      margin: 0 auto;
      background: $color-light-blue;
      background: -webkit-radial-gradient(20% 20%, circle cover, $color-light-blue 0%, transparent 60%, transparent 100%);
      background: radial-gradient(circle at 20% 20%, $color-light-blue 0%, transparent 60%, transparent 100%);
      border-radius: 110px;
      padding: 10px;
      position: absolute;
      top: 0;
      right: 0;
      bottom: 0;
      left: 0;
      -webkit-animation-name: spinner-rotate;
      -webkit-animation-duration: 1s;
      -webkit-animation-iteration-count: infinite;
      -webkit-animation-timing-function: linear;
      -moz-animation-name: spinner-rotate;
      -moz-animation-duration: 1s;
      -moz-animation-iteration-count: infinite;
      -moz-animation-timing-function: linear;
      animation-name: spinner-rotate;
      animation-duration: 1s;
      animation-iteration-count: infinite;
      animation-timing-function: linear;
    }
  }
}
@-webkit-keyframes spinner-rotate {
  from {
    -webkit-transform: rotate(0deg);
    -moz-transform: rotate(0deg);
    transform: rotate(0deg);
  }

  to {
    -webkit-transform: rotate(360deg);
    -moz-transform: rotate(360deg);
    transform: rotate(360deg);
  }
}
@-moz-keyframes spinner-rotate {
  from {
    -webkit-transform: rotate(0deg);
    -moz-transform: rotate(0deg);
    transform: rotate(0deg);
  }

  to {
    -webkit-transform: rotate(360deg);
    -moz-transform: rotate(360deg);
    transform: rotate(360deg);
  }
}
@keyframes spinner-rotate {
  from {
    -webkit-transform: rotate(0deg);
    -moz-transform: rotate(0deg);
    transform: rotate(0deg);
  }

  to {
    -webkit-transform: rotate(360deg);
    -moz-transform: rotate(360deg);
    transform: rotate(360deg);
  }
}
.btn-primary {
  // background-color: $color-light-blue;
  // border-color: $color-light-blue;
  @include border-radius(0);
  &:focus,
  &:hover {
    background-color: $color-btn-primary-focus;
    border-color: $color-btn-primary-focus;
  }

  &:active {
    &:focus {
      background-color: $color-btn-primary-active;
      border-color: $color-btn-primary-active;
      @include box-shadow(none);
    }
    background-color: $color-btn-primary-active;
    border-color: $color-btn-primary-active;
    @include box-shadow(none);
  }

  &.docs-link {
    padding: 10px 25px;
    &+.docs-link {
      margin-left: 15px;
    }
    @media (max-width:500px) {
      width: 100%;
      &+.docs-link {
        margin-top: 5px;
      }
    }
  }
}
.btn {
  &.btn-reset {
    margin-top: 21px;
    margin-bottom: -5px;
    @include img-button();
    @include icon-reset();
  }

  &.btn-load {
    margin-top: 21px;
    margin-bottom: -5px;
    @include img-button();
    @include icon-load();
  }

  &.btn-cancel {
    margin-top: 21px;
    margin-bottom: -5px;
    @include img-button();
    @include icon-cancel();
  }
}
div.toolbar {
  text-align: right;
  padding-bottom: 15px;
  ul {
    padding-left: 0;
    margin-left: -5px;
    list-style: none;
    text-align: right;
    margin-right: -5px;
    margin-bottom: 0;
    li {
      display: inline-block;
      padding-right: 5px;
      padding-left: 5px;
    }
  }

  @media (max-width: 768px) {
    ul {
      &:after {
        content: '';
        clear: both;
        display: table;
      }
      &:before {
        content: '';
        clear: both;
      }
      li {
        width: 50%;
        float: left;
        padding-top: 5px;
        .btn {
          width: 100%;
        }
      }
    }
  }
}
rt-status-done,
rt-status-failed,
rt-status-initial,
rt-status-no-data,
rt-status-progress,
rt-status-request-cancelled {
  display: block;
  text-align: center;
}
rt-demo-buffered-footer,
rt-demo-paged-footer {
  display: block;
}
form {
  label {
    color: $color-muted;
    font-weight: normal;
    line-height: 1.5;
  }

  margin-bottom: 15px;
  .btn {
    margin-top: 21px;
    margin-bottom: -5px;
    &+.btn {
      margin-left: 5px;
    }
  }
}
.rt-sortable {
  position: relative;
  padding-right: 45px;
  cursor: pointer;
  &:after {
    content: '';
    display: inline-block;
    width: 22px;
    height: 22px;
    position: absolute;
    top: 50%;
    right: 0;
    margin-top: -11px;
    background-position: center center;
    background-repeat: no-repeat;
  }

  &:hover {
    &:after {
      background-image: url('');
    }
  }

  &.rt-sort-asc:after {
    background-image: url('');
  }

  &.rt-sort-desc:after {
    background-image: url('');
  }
}
.table-responsive {
  border: 0;
}
.table {
  font-size: $font-small;
  thead {
    tr {
      th {
        background-color: $color-white;
        font-weight: bold;
        color: #053B75;
        font-size: $font-small;
        border-bottom-color: $color-gray;
      }
    }
  }

  tbody {
    tr {
      background-color: $color-white;
      td {
        border-top-color: $color-gray;
      }
      &:nth-of-type(even) {
        background-color: $color-striped;
      }
      &:hover {
        background-color: $color-light-grey;
      }
      &.rt-selected {
        background-color: $color-selected;
      }
    }
  }

  tfoot {
    tr {
      background-color: $color-gray;
      td {
        border-top: 0;
        padding-right: 40px;
        vertical-align: middle;
        .form-control {
          display: inline-block;
          width: 50px;
          text-align: center;
          padding: 6px;
          vertical-align: middle;
        }
      }
    }
  }
}
section.alertify-logs {
  bottom: 75px;
}
section.alertify-alert {
  padding-top: 0;
  margin-top: 0;
  @include box-shadow(none);
  @include border-radius(0);
}
.alertify-button-ok {
  @include box-shadow(none);
  @include border-radius(0);
  background: $color-light-blue;
  border-color: $color-light-blue;
  color: $color-white;
  text-decoration: none;
  text-shadow: none;
  &:focus,
  &:hover {
    @include box-shadow(none);
    @include border-radius(0);
    background: $color-btn-primary-focus;
    border-color: $color-btn-primary-focus;
    color: $color-white;
    text-decoration: none;
  }

  &:active {
    &:focus {
      background: $color-btn-primary-active;
      border-color: $color-btn-primary-active;
      color: $color-white;
      text-decoration: none;
      @include box-shadow(none);
    }
    background: $color-btn-primary-active;
    border-color: $color-btn-primary-active;
    color: $color-white;
    text-decoration: none;
    @include box-shadow(none);
  }
}
.control-group {
  display: inline-block;
  vertical-align: top;
  background: #fff;
  text-align: left;
  box-shadow: 0 1px 2px rgba(0, 0, 0, 0.1);
  padding: 30px;
  width: 200px;
  height: 210px;
  margin: 10px;
}
.control {
  display: block;
  position: relative;
  padding-left: 30px;
  margin-bottom: 18px;
  cursor: pointer;
  font-size: 18px;
}
.control input {
  position: absolute;
  z-index: -1;
  opacity: 0;
}
.control__indicator {
  position: absolute;
  top: 0;
  left: 0;
  height: 20px;
  width: 20px;
  background: #e6e6e6;
}
.control--radio .control__indicator {
  border-radius: 50%;
}
.control:hover input~.control__indicator,
.control input:focus~.control__indicator {
  background: #ccc;
}
.control input:checked~.control__indicator {
  background: #2aa1c0;
}
.control:hover input:not([disabled]):checked~.control__indicator,
.control input:checked:focus~.control__indicator {
  background: #0e647d;
}
.control input:disabled~.control__indicator {
  background: #e6e6e6;
  opacity: 0.6;
  pointer-events: none;
}
.control__indicator:after {
  content: '';
  position: absolute;
  display: none;
}
.control input:checked~.control__indicator:after {
  display: block;
}
.control--checkbox .control__indicator:after {
  left: 8px;
  top: 4px;
  width: 5px;
  height: 8px;
  border: solid #fff;
  border-width: 0 1px 1px 0;
  transform: rotate(45deg);
}
.control--checkbox input:disabled~.control__indicator:after {
  border-color: #7b7b7b;
}
.control--radio .control__indicator:after {
  left: 7px;
  top: 7px;
  height: 6px;
  width: 6px;
  border-radius: 50%;
  background: #fff;
}
.control--radio input:disabled~.control__indicator:after {
  background: #7b7b7b;
}
.select {
  position: relative;
  display: inline-block;
  margin-bottom: 15px;
  width: 100%;
}
.select select {
  display: inline-block;
  width: 100%;
  cursor: pointer;
  padding: 10px 15px;
  outline: 0;
  border: 0;
  border-radius: 0;
  background: #e6e6e6;
  color: #7b7b7b;
  appearance: none;
  -webkit-appearance: none;
  -moz-appearance: none;
}
.select select::-ms-expand {
  display: none;
}
.select select:hover,
.select select:focus {
  color: #000;
  background: #ccc;
}
.select select:disabled {
  opacity: 0.5;
  pointer-events: none;
}
.select__arrow {
  position: absolute;
  top: 16px;
  right: 15px;
  width: 0;
  height: 0;
  pointer-events: none;
  border-style: solid;
  border-width: 8px 5px 0 5px;
  border-color: #7b7b7b transparent transparent transparent;
}
.select select:hover~.select__arrow,
.select select:focus~.select__arrow {
  border-top-color: #000;
}
.select select:disabled~.select__arrow {
  border-top-color: #ccc;
}
.hideGrid {
  display: none;
}
.rightjustified{
  text-align: right;
}
// 这个是主要的。
.stickyTable{
    overflow: visible !important;
    thead {
        tr{
            th{
                position: sticky !important;
                top: 134px !important;
            }
        }
    }
}

其实就是一个

position:sticky;