From 9179b14fd4d876ea8997ce39bc0ec3cd4f6a0a73 Mon Sep 17 00:00:00 2001
From: Glenn Vorhes <gavorhes@wisc.edu>
Date: Tue, 3 May 2016 12:58:15 -0500
Subject: [PATCH] set up gulp tasks, porting content

---
 .babelrc                               |   3 -
 css/all-ol-style.less                  |   3 +
 css/glrtoc/glrtoc-opsmap.less          | 132 ++++
 css/glrtoc/img/glrtoc-logo.png         | Bin 0 -> 13466 bytes
 css/glrtoc/img/tops-logo.png           | Bin 0 -> 6816 bytes
 css/itsMap.less                        | 116 +++
 css/legend.less                        | 116 +++
 css/media-control.less                 |  68 ++
 css/npmrds-heatmap.less                | 174 +++++
 css/ol-popup.less                      |  68 ++
 css/peergroup.less                     | 173 +++++
 css/tip-colors.less                    |  30 +
 css/tip-results.less                   | 136 ++++
 css/tip.less                           | 159 ++++
 gulpfile.js                            | 230 ++++++
 lib/jquery.floatThead.js               | 967 +++++++++++++++++++++++++
 package.json                           |  25 +-
 projects/glrtoc/appConfig.js           | 150 ++++
 projects/glrtoc/layerPopups.js         |  37 +
 projects/glrtoc/layerStyles.js         |  39 +
 projects/glrtoc/legendTest.js          | 129 ++++
 projects/glrtoc/main.js                | 319 ++++++++
 projects/glrtoc/mainUi.js              | 105 +++
 projects/itsMap.js                     |  27 +
 projects/npmrds/delay/delay-config.js  |   3 +
 projects/npmrds/delay/delay-main.js    |  32 +
 projects/npmrds/heatmap/appConfig.js   |  50 ++
 projects/npmrds/heatmap/layerStyles.js |  43 ++
 projects/npmrds/heatmap/main-ui.js     | 368 ++++++++++
 projects/npmrds/heatmap/main.js        |  47 ++
 projects/tsmo/TipConfig.js             | 149 ++++
 projects/tsmo/TipSegmentLayer.js       | 191 +++++
 projects/tsmo/legend-test.js           |  84 +++
 projects/tsmo/main-report.js           | 198 +++++
 projects/tsmo/main-ui.js               | 126 ++++
 projects/tsmo/main.js                  | 166 +++++
 projects/tsmo/slider-test.js           |  29 +
 projects/tsmo/tipStyleFunction.js      |  84 +++
 src/collections/ItsLayerCollection.js  |   2 +-
 src/collections/LayerLegend.js         |   2 +-
 src/collections/Sliders.js             |   2 +-
 src/jquery-plugin/dayRange.js          |   2 +-
 src/jquery-plugin/mediaControl.js      |   2 +-
 src/jquery-plugin/rangeChange.js       |   2 +-
 src/jquery-plugin/rpPicker.js          |   2 +-
 src/jquery-plugin/ssaCorridorPicker.js |   2 +-
 src/jquery.js                          |   6 +
 src/layers/LayerBase.js                |   2 +-
 src/layers/LayerBaseVector.js          |   4 +-
 src/layers/LayerBaseVectorEsri.js      |   2 +-
 src/layers/LayerBaseVectorGeoJson.js   |   2 +-
 src/layers/LayerBaseXyzTile.js         |   2 +-
 src/layers/LayerEsriMapServer.js       |   2 +-
 src/layers/LayerItsInventory.js        |   2 +-
 src/layers/LayerRealEarthTile.js       |   2 +-
 src/mixin/RealEarthAnimate.js          |   2 +-
 src/olHelpers/mapMoveCls.js            |   2 +-
 src/olHelpers/mapPopupCls.js           |   2 +-
 src/olHelpers/quickMapBase.js          |   2 +-
 59 files changed, 4799 insertions(+), 25 deletions(-)
 delete mode 100644 .babelrc
 create mode 100644 css/all-ol-style.less
 create mode 100644 css/glrtoc/glrtoc-opsmap.less
 create mode 100644 css/glrtoc/img/glrtoc-logo.png
 create mode 100644 css/glrtoc/img/tops-logo.png
 create mode 100644 css/itsMap.less
 create mode 100644 css/legend.less
 create mode 100644 css/media-control.less
 create mode 100644 css/npmrds-heatmap.less
 create mode 100644 css/ol-popup.less
 create mode 100644 css/peergroup.less
 create mode 100644 css/tip-colors.less
 create mode 100644 css/tip-results.less
 create mode 100644 css/tip.less
 create mode 100644 gulpfile.js
 create mode 100644 lib/jquery.floatThead.js
 create mode 100644 projects/glrtoc/appConfig.js
 create mode 100644 projects/glrtoc/layerPopups.js
 create mode 100644 projects/glrtoc/layerStyles.js
 create mode 100644 projects/glrtoc/legendTest.js
 create mode 100644 projects/glrtoc/main.js
 create mode 100644 projects/glrtoc/mainUi.js
 create mode 100644 projects/itsMap.js
 create mode 100644 projects/npmrds/delay/delay-config.js
 create mode 100644 projects/npmrds/delay/delay-main.js
 create mode 100644 projects/npmrds/heatmap/appConfig.js
 create mode 100644 projects/npmrds/heatmap/layerStyles.js
 create mode 100644 projects/npmrds/heatmap/main-ui.js
 create mode 100644 projects/npmrds/heatmap/main.js
 create mode 100644 projects/tsmo/TipConfig.js
 create mode 100644 projects/tsmo/TipSegmentLayer.js
 create mode 100644 projects/tsmo/legend-test.js
 create mode 100644 projects/tsmo/main-report.js
 create mode 100644 projects/tsmo/main-ui.js
 create mode 100644 projects/tsmo/main.js
 create mode 100644 projects/tsmo/slider-test.js
 create mode 100644 projects/tsmo/tipStyleFunction.js
 create mode 100644 src/jquery.js

diff --git a/.babelrc b/.babelrc
deleted file mode 100644
index c6c9efb..0000000
--- a/.babelrc
+++ /dev/null
@@ -1,3 +0,0 @@
-{
-"presets": ["es2015"]
-}
\ No newline at end of file
diff --git a/css/all-ol-style.less b/css/all-ol-style.less
new file mode 100644
index 0000000..41a007c
--- /dev/null
+++ b/css/all-ol-style.less
@@ -0,0 +1,3 @@
+@import (inline) "../node_modules/openlayers/dist/ol.css";
+@import "legend";
+@import "ol-popup";
diff --git a/css/glrtoc/glrtoc-opsmap.less b/css/glrtoc/glrtoc-opsmap.less
new file mode 100644
index 0000000..6e0df65
--- /dev/null
+++ b/css/glrtoc/glrtoc-opsmap.less
@@ -0,0 +1,132 @@
+@import "../all-ol-style";
+@import "../media-control";
+
+
+body, html {
+  height: 100%;
+  width: 100%;
+  margin: 0;
+  padding: 0;
+}
+
+#main-container {
+  height: 100%;
+}
+
+.flex-container {
+  display: flex;
+
+  //> div {
+  //  flex-grow: 1;
+  //}
+}
+
+#map{
+  flex-grow: 1;
+}
+
+
+#sidebar {
+  height: 100%;
+  width: 350px;
+  flex-grow: initial;
+  position: relative;
+}
+
+#logo-container {
+    > div {
+    flex-grow: 1;
+  }
+
+  div {
+    margin: 7px;
+    height: 150px;
+  }
+
+  div:first-child {
+    background: url('img/glrtoc-logo.png') no-repeat center;
+    background-size: contain;
+  }
+
+  div:nth-child(2) {
+    background: url('img/tops-logo.png') no-repeat center;
+    background-size: contain;
+  }
+}
+
+#map {
+  position: relative;
+}
+
+#tabs {
+  height: 100%;
+  padding: 0;
+  border: none;
+  width: 100%;
+
+
+  > div {
+    padding: 0;
+    height: 98%;
+
+    p {
+      text-align: justify;
+      padding: 7px 12px;
+      margin: 10px 0;
+    }
+
+  }
+
+  .ui-tab {
+    padding: 3px 7px !important;
+  }
+}
+
+#operations-tab {
+  h3 {
+    margin-top: 0;
+    padding-top: 5px;
+    padding-bottom: 5px;
+  }
+
+  > div{
+    padding: 0;
+  }
+}
+
+#hide-sidebar, #show-sidebar {
+  z-index: 5;
+  position: absolute;
+  color: white;
+  cursor: pointer;
+}
+
+//hide sidebar span
+#hide-sidebar {
+  right: 4px;
+  top: 9px;
+}
+
+#show-sidebar {
+  display: none;
+  left: -12px;
+  top: 74px;
+  background: linear-gradient(#74BBD8, #3FA3CA);
+  width: 40px;
+  text-align: right;
+  padding: 10px 4px;
+  border-radius: 5px;
+  border: solid #4297D7 1px;
+}
+
+#operations > div {
+  padding: 0;
+}
+
+#animation-control {
+  text-align: center;
+  padding: 10px;
+}
+
+
+
diff --git a/css/glrtoc/img/glrtoc-logo.png b/css/glrtoc/img/glrtoc-logo.png
new file mode 100644
index 0000000000000000000000000000000000000000..e703e4220e4f9bfc334eb49e95d19b8e9720aefe
GIT binary patch
literal 13466
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
Cmu+?c

literal 0
HcmV?d00001

diff --git a/css/glrtoc/img/tops-logo.png b/css/glrtoc/img/tops-logo.png
new file mode 100644
index 0000000000000000000000000000000000000000..436dbf5a4ff00a9dcea8dd51add6f68b5943c4f7
GIT binary patch
literal 6816
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

literal 0
HcmV?d00001

diff --git a/css/itsMap.less b/css/itsMap.less
new file mode 100644
index 0000000..7cbd153
--- /dev/null
+++ b/css/itsMap.less
@@ -0,0 +1,116 @@
+@import "./all-ol-style";
+
+
+@headerHeight: 80px;
+
+body, html {
+  height: 100%;
+  width: 100%;
+  margin: 0;
+  padding: 0;
+}
+
+#header {
+  height: @headerHeight;
+  background-color: #808080;
+
+  span {
+    display: inline-block;
+    margin: 9px;
+    font-size: 28px;
+    line-height: 30px;
+  }
+
+  a {
+    float: right;
+  }
+
+  img {
+    margin: 3px 10px;
+    max-height: @headerHeight - 10px;
+    width: auto;
+  }
+}
+
+#container {
+  display: flex;
+  height: ~"calc(100% - " @headerHeight ~")";
+  width: 100%;
+}
+
+#left {
+  height: 100%;
+  width: 320px;
+  overflow-y: auto;
+}
+
+#map {
+  flex: 1
+}
+
+.legend-container {
+  border: none;
+}
+
+@media screen and (max-height: 650px) {
+
+}
+
+@media screen and (max-height: 640px), (max-width: 650px) {
+
+  #header a {
+    display: none;
+  }
+
+  #header{
+    display: none;
+  }
+
+  #container {
+    //display: block;
+    height: 100%;
+  }
+
+  #left {
+    width: 246px;
+    height: auto;
+    position: absolute;
+    top: 5px;
+    right: 5px;
+    z-index: 10;
+    background-color: white;
+    -webkit-border-radius:5px;
+    -moz-border-radius:5px;
+    border-radius: 5px;
+    max-height: 250px;
+    overflow-y: scroll;
+  }
+
+  #map{
+    height: 100%;
+  }
+
+  .legend-container ul{
+    padding-left: 5px;
+  }
+
+}
+
+@media screen and (max-height: 500px), (max-width: 346px) {
+  #left{
+    width: 204px;
+    max-height: 204px;
+  }
+
+  .ol-popup{
+    width: 240px;
+    min-width: 240px;
+
+    iframe{
+      width: 240px;
+    }
+  }
+}
+
+
+
diff --git a/css/legend.less b/css/legend.less
new file mode 100644
index 0000000..504f281
--- /dev/null
+++ b/css/legend.less
@@ -0,0 +1,116 @@
+@legendKeyOffset: 7px;
+
+@rectangleImg: url('');
+
+
+.legend-container {
+  list-style: none;
+  border: solid black 1px;
+  border-radius: 5px;
+  background-color: rgba(211, 211, 211, 0.2);
+  margin: 2px;
+  padding: 2px;
+
+  ul {
+    list-style: none;
+    padding-left: 17px;
+  }
+
+  li {
+    border-radius: 7px;
+    margin: 2px;
+    padding: 2px 4px;
+  }
+
+  hr {
+    display: inline-block;
+    width: 40px;
+    height: 10px;
+    background-color: blue;
+    border: none;
+    margin: 0 0 3px @legendKeyOffset;
+  }
+
+  > li:first-child {
+    font-weight: bold;
+    min-height: 25px;
+
+    input {
+      display: none;
+    }
+
+    input + label {
+      width: 100px;
+      height: 19px;
+      margin: 5px;
+      display: inline-block;
+    }
+
+    input + label > span {
+      display: inline-block;
+      width: 59px;
+      height: 19px;
+      background: @rectangleImg no-repeat 0 0;
+    }
+
+
+
+    input:checked + label > span {
+      background: @rectangleImg no-repeat -59px 0;
+    }
+  }
+}
+
+
+.legend-layer-subitem {
+  display: inline-block;
+  max-width: 148px;
+  overflow: hidden;
+  max-height: 20px;
+}
+
+.legend-layer-icon {
+  margin-left: @legendKeyOffset;
+}
+
+.legend-layer-div li {
+  margin: 0;
+  padding-top: 0;
+  padding-bottom: 0;
+}
+
+.layer-not-visible {
+  background-color: gray;
+  display: none;
+}
+
+.layer-force-show {
+  display: inherit;
+}
+
+.layer-group-expander, .map-server-expander, .unique-symbol-expander, .legend-items-expander {
+  color: #59AFEE;
+  cursor: pointer;
+  padding-left: 4px;
+  font-size: 14px;
+}
+
+.esri-popup-table{
+  border-collapse: collapse;
+
+  td, th {
+    border: solid black 1px;
+    padding: 0 4px;
+  }
+
+  tr:nth-child(even){
+    background-color: lightblue;
+  }
+}
+
+.legend-layer-name, .legend-check{
+  cursor: pointer
+}
+
+
+
diff --git a/css/media-control.less b/css/media-control.less
new file mode 100644
index 0000000..9233513
--- /dev/null
+++ b/css/media-control.less
@@ -0,0 +1,68 @@
+@mediaImg: url('');
+
+.media-control-container {
+  padding: 10px;
+  text-align: center;
+
+  input[type=range]{
+    width: 80%;
+  }
+
+  input[type=range]:disabled{
+      cursor: not-allowed;
+      background-color: lightgray;
+  }
+}
+
+.media-player-button {
+  background: @mediaImg no-repeat;
+  height: 29px;
+  width: 30px;
+  display: inline-block;
+  margin: 0 2px;
+  cursor: pointer;
+}
+
+.media-player-button:hover {
+  background-color: lightblue;
+}
+
+.media-pause {
+  background-position: -86px 0;
+}
+
+.media-play {
+  background-position: -28px 0;
+}
+
+.media-stop {
+  background-position: -57px 0;
+}
+
+.media-ahead {
+  background-position: -116px 0;
+}
+
+.media-back {
+  background-position: 1px 0;
+}
+
+.media-disabled {
+  display: none;
+}
+
+.media-locked{
+  background-color: lightgray !important;
+  cursor: not-allowed;
+}
+
+.media-control-value-label-container{
+  display: flex;
+  justify-content: space-between;
+
+  span{
+    display: block;
+    font-size: small;
+  }
+}
+
diff --git a/css/npmrds-heatmap.less b/css/npmrds-heatmap.less
new file mode 100644
index 0000000..16ffa8b
--- /dev/null
+++ b/css/npmrds-heatmap.less
@@ -0,0 +1,174 @@
+@heatMapControlHeight: 135px;
+@heatMapSliderHeight: 30px;
+@import "./all-ol-style";
+
+
+html, body {
+  height: 100%;
+  width: 100%;
+  margin: 0;
+  /*position: relative;*/
+}
+
+#container {
+  display: flex;
+  height: 100%;
+  width: 100%;
+}
+
+#map {
+  flex: 1;
+}
+
+#left {
+  flex: initial;
+  width: 300px;
+  min-width: 100px;
+
+  > div {
+    margin: 10px;
+  }
+}
+
+#ul-hwy-dirs {
+  list-style: none;
+  padding-left: 5px;
+  max-height: 300px;
+  overflow-y: auto;
+  background-color: lightgrey;
+
+  li {
+    padding: 2px;
+  }
+}
+
+h4 {
+  margin: 10px 5px;
+}
+
+#right {
+  width: 60%;
+  height: 100%;
+
+  input[type=button] {
+    margin: 3px 0;
+  }
+}
+
+#heat-map-control {
+  display: flex;
+  flex-wrap: wrap;
+  height: @heatMapControlHeight;
+
+  > div {
+    margin: 0 5px;
+  }
+
+  ul {
+    list-style: none;
+    padding-left: 0;
+    margin-top: 14px;
+  }
+}
+
+#heat-slider-container {
+  height: @heatMapSliderHeight;
+}
+
+#heat-map-slider {
+  border: none;
+  width: 95%;
+  padding: 0;
+  margin: 0 0 0 18px;
+}
+
+#heatmap-canvas {
+  position: absolute;
+  top: 0;
+  left: 22px;
+  cursor: crosshair;
+}
+
+#heat-slider-container, #canvas-container {
+  width: 100%;
+  position: relative;
+}
+
+#canvas-container {
+  height: ~"calc(100% - " @heatMapControlHeight + @heatMapSliderHeight ~")";
+  overflow-y: auto;
+}
+
+#heat-map-vertical-bar {
+  position: absolute;
+  /*z-index: 4;*/
+  border-right: solid blue 2px;
+  left: 22px;
+  top: 0;
+  width: 0;
+  height: 0;
+  background-color: rgba(100, 100, 100, 0.3);
+}
+
+#date-ul {
+  list-style: none;
+  margin: 0;
+  padding: 0;
+  position: absolute;
+  left: 0;
+  top: 0;
+  z-index: -5;
+
+  li {
+    border-bottom: solid #000000 1px;
+    height: 575px;
+  }
+
+  li:nth-child(odd) {
+    background-color: lightblue;
+  }
+
+  li div {
+    color: blue;
+    height: 150px;
+    transform: rotate(-90deg) translate(-259px, 0px);
+    transform-origin: left top 0;
+    float: left;
+  }
+}
+
+#canvas-tooltip-span {
+  display: none;
+  z-index: 100;
+  position: fixed;
+  background-color: lightgrey;
+  border: solid black 1px;
+  border-radius: 5px;
+  padding: 5px;
+  /*height: 50px;*/
+  width: 160px;
+  overflow: hidden;
+  text-align: center;
+}
+
+#progress-indicator-div {
+  z-index: 200;
+  background-color: lightblue;
+  border: solid darkslategray 1px;
+  border-radius: 5px;
+  width: 200px;
+  padding: 10px;
+  margin: 20px;
+  display: none;
+}
+
+#sel-hwy {
+  padding-left: 14px;
+  padding-top: 9px;
+  font-weight: bold;
+  display: inline-block;
+}
+
+/*.ui-datepicker{*/
+/*z-index: 100 !important;*/
+/*}*/
\ No newline at end of file
diff --git a/css/ol-popup.less b/css/ol-popup.less
new file mode 100644
index 0000000..8e13d3e
--- /dev/null
+++ b/css/ol-popup.less
@@ -0,0 +1,68 @@
+.ol-popup {
+  position: absolute;
+  background-color: white;
+  -webkit-filter: drop-shadow(0 1px 4px rgba(0, 0, 0, 0.2));
+  filter: drop-shadow(0 1px 4px rgba(0, 0, 0, 0.2));
+  padding: 22px 5px 5px 5px;
+  border-radius: 10px;
+  border: 1px solid #cccccc;
+  bottom: 12px;
+  left: -50px;
+  min-width: 280px;
+  cursor: default;
+}
+
+.ol-popup:after, .ol-popup:before {
+  top: 100%;
+  border: solid transparent;
+  content: " ";
+  height: 0;
+  width: 0;
+  position: absolute;
+  pointer-events: none;
+}
+
+.ol-popup:after {
+  border-top-color: white;
+  border-width: 10px;
+  left: 48px;
+  margin-left: -10px;
+}
+
+.ol-popup:before {
+  border-top-color: #cccccc;
+  border-width: 11px;
+  left: 48px;
+  margin-left: -11px;
+}
+
+.ol-popup-closer {
+  text-decoration: none;
+  position: absolute;
+  top: 2px;
+  right: 8px;
+}
+
+.ol-popup-closer:after {
+  content: "X";
+}
+
+.ol-popup-nav {
+  background-color: lightgray;
+  padding: 3px;
+  margin-bottom: 7px;
+}
+
+.ol-popup-nav-arrow{
+  color: #419CC4;
+  cursor: pointer;
+  margin: 0 3px;
+  font-size: larger;
+}
+
+.ol-inner-inner {
+  height: 300px;
+  //width: 392px;
+  //padding-right: 25px;
+  overflow-y: auto;
+}
diff --git a/css/peergroup.less b/css/peergroup.less
new file mode 100644
index 0000000..e0fb5e7
--- /dev/null
+++ b/css/peergroup.less
@@ -0,0 +1,173 @@
+@import './all-ol-style';
+
+@sidebarWidth: 280px;
+
+html, body {
+  margin: 0;
+  padding: 0;
+  height: 100%;
+  width: 100%;
+}
+
+#app-container {
+  display: flex;
+  height: 100%;
+
+  > div {
+    height: 100%;
+    overflow-y: auto;
+  }
+}
+
+#left-sidebar {
+  width: @sidebarWidth;
+
+  label {
+    display: inline-block;
+    margin-left: 4px;
+  }
+
+  h6 {
+    margin-top: 5px;
+    margin-bottom: 2px;
+  }
+}
+
+#accordion {
+  font-size: 0.9em !important;
+
+  h3 {
+    padding-top: 1px;
+    padding-bottom: 1px;
+    margin-bottom: 2px;
+  }
+}
+
+#right-sidebar {
+  width: 400px;
+}
+
+#map {
+  flex: 1;
+}
+
+#dialog{
+  display: none;
+}
+//
+.ui-accordion-content {
+  padding: 5px 10px !important;
+}
+
+select {
+  margin: 4px;
+  padding: 0;
+}
+
+input[type=text] {
+  margin: 2px 7px;
+  padding: 5px;
+}
+
+.divUnd {
+  width: 100px;
+
+}
+
+.equality-compare {
+  width: 45px;
+}
+
+.class-adtVal {
+  width: 90px;
+}
+
+.class-laneVal, .class-speedVal {
+  width: 50px;
+}
+
+.comma-list {
+  width: 250px;
+
+}
+
+#summary {
+  border-collapse: collapse;
+}
+
+#summary tr {
+  height: 30px;
+}
+
+#summary th, td {
+  border: 1px solid black;
+  padding: 1px 4px;
+  text-align: center;
+  //color: #636363;
+}
+
+.invalid {
+  border: solid red 1px !important;
+}
+
+.popup-table {
+  border-collapse: collapse;
+  margin-top: 5px;
+
+}
+
+.popup-table th, td {
+  border: 1px solid black;
+}
+
+//
+//.ol-popup {
+//    display: none;
+//    position: absolute;
+//    background-color: white;
+//    -moz-box-shadow: 0 1px 4px rgba(0, 0, 0, 0.2);
+//    -webkit-filter: drop-shadow(0 1px 4px rgba(0, 0, 0, 0.2));
+//    filter: drop-shadow(0 1px 4px rgba(0, 0, 0, 0.2));
+//    padding: 15px;
+//    border-radius: 10px;
+//    border: 1px solid #cccccc;
+//    bottom: 12px;
+//    left: -50px;
+//}
+//
+//.ol-popup:after, .ol-popup:before {
+//    top: 100%;
+//    border: solid transparent;
+//    content: " ";
+//    height: 0;
+//    width: 0;
+//    position: absolute;
+//    pointer-events: none;
+//}
+//
+//.ol-popup:after {
+//    border-top-color: white;
+//    border-width: 10px;
+//    left: 48px;
+//    margin-left: -10px;
+//}
+//
+//.ol-popup:before {
+//    border-top-color: #cccccc;
+//    border-width: 11px;
+//    left: 48px;
+//    margin-left: -11px;
+//}
+//
+//.ol-popup-closer {
+//    text-decoration: none;
+//    position: absolute;
+//    top: 2px;
+//    right: 8px;
+//}
+//
+//.ol-popup-closer:after {
+//    content: "✖";
+//}
+//
+//
diff --git a/css/tip-colors.less b/css/tip-colors.less
new file mode 100644
index 0000000..669e4c5
--- /dev/null
+++ b/css/tip-colors.less
@@ -0,0 +1,30 @@
+.cell-min {
+    background-color: #00ff00 !important;
+}
+
+.cell-very-low {
+    background-color: #55ff00  !important;
+}
+
+.cell-low {
+    background-color: #aaff00  !important;
+}
+
+.cell-mid {
+    background-color: #ffff00  !important;
+}
+
+.cell-high {
+    background-color: #ffaa00  !important;
+}
+
+.cell-very-high {
+    background-color: #ff5500  !important;
+    font-weight: bold;
+}
+
+.cell-max {
+    background-color: #ff0000  !important;
+    font-weight: bold;
+    text-decoration: underline;
+}
\ No newline at end of file
diff --git a/css/tip-results.less b/css/tip-results.less
new file mode 100644
index 0000000..8a04f94
--- /dev/null
+++ b/css/tip-results.less
@@ -0,0 +1,136 @@
+@import "tip-colors";
+@import "./all-ol-style";
+
+body, html, #container {
+  height: 100%;
+  width: 100%;
+  margin: 0;
+  padding: 0;
+}
+
+#container, #summary {
+  display: flex;
+}
+
+#summary {
+  height: 38%;
+
+  > div {
+    flex-grow: 1;
+    height: 100%;
+    overflow-y: auto;
+  }
+}
+
+#crash-data {
+  table {
+    border-collapse: collapse;
+    margin: 7px 4px 0 4px;
+  }
+
+  th, td {
+    border: solid black 1px;
+    text-align: center;
+    padding: 0 5px;
+  }
+}
+
+#right {
+  flex: 1;
+}
+
+#map {
+  height: 60%;
+  position: relative;
+}
+
+#legend-container {
+  position: absolute;
+  width: 250px;
+  max-height: 80%;
+  min-height: 30px;
+  overflow-y: auto;
+  background-color: rgba(255, 255, 255, 0.8);
+  z-index: 10;
+  top: 5px;
+  right: 5px;
+  border-radius: 7px;
+
+  > ul {
+    > li:first-child {
+      display: none;
+    }
+  }
+}
+
+#summary {
+  h3, h4, ul {
+    margin: 5px 10px;
+  }
+}
+
+#results-container {
+  width: 936px;
+  padding-right: 20px;
+  height: 100%;
+  overflow: auto;
+  position: relative;
+  table {
+    border-collapse: collapse;
+    margin: 4px;
+    width: 100%;
+    //height: 100%;
+    position: relative;
+    table-layout: fixed;
+
+    th, td {
+      border: solid black 1px;
+      padding: 3px;
+      text-align: center;
+    }
+
+    > *{
+      width: 100%;
+    }
+
+    th {
+      cursor: pointer;
+    }
+
+    thead {
+
+      display: block;
+      width: 100%;
+
+      tr {
+        background-color: #4a4a4a;
+        color: #FDFDFD;
+        width: 100%;
+      }
+    }
+
+    tbody {
+      display: block;
+      height: 400px;
+      overflow-y: auto;
+      width: 100%;
+    }
+  }
+}
+
+.selectable-row:hover {
+  background-color: #00ffd0;
+  cursor: pointer;
+}
+
+.selected-row {
+  background-color: #0074ff !important;
+}
+
+.score-under-one {
+  display: none;
+}
+
+#show-all-toggle, #show-all-toggle-label {
+  cursor: pointer;
+}
diff --git a/css/tip.less b/css/tip.less
new file mode 100644
index 0000000..09ac1b5
--- /dev/null
+++ b/css/tip.less
@@ -0,0 +1,159 @@
+@import "./tip-colors";
+@import "./all-ol-style";
+
+html, body {
+  width: 100%;
+  height: 100%;
+  margin: 0;
+}
+
+#container {
+  display: flex;
+  height: 100%;
+  width: 100%;
+}
+
+#left {
+  overflow-y: scroll;
+  background-color: lightblue;
+  height: 100%;
+  width: 320px;
+}
+
+// suppress first li in legend
+.legend-container {
+  > li:first-child {
+    display: none;
+  }
+}
+
+//tip legend style
+#tip-segments-layer-li {
+  display: inherit;
+
+  > div {
+    height: 80px;
+
+    hr {
+      height: 4px;
+    }
+
+    ul:first-of-type {
+      float: right;
+      padding-left: 0;
+
+      li {
+        height: 10px;
+      }
+    }
+
+    ul:last-of-type {
+      float: right;
+      margin-top: 5px;
+
+      li {
+        margin: 3px 0;
+        text-align: right;
+        border-right: solid black 1px;
+        border-radius: 0;
+      }
+    }
+  }
+}
+
+//common border styling of legend and preset container
+#preset-wrapper, .legend-container {
+  border: solid black 1px;
+  border-radius: 5px;
+}
+
+#preset-wrapper {
+  margin: 3px;
+  padding: 3px;
+  border: solid black 1px;
+  border-radius: 5px;
+
+  h4 {
+    display: inline-block;
+    margin-top: 0;
+    margin-bottom: 0;
+  }
+
+  select {
+    width: 141px;
+  }
+
+  input[type=button] {
+    width: 88px;
+    margin: 0;
+  }
+}
+
+.slider-container {
+
+  > div {
+    padding: 2px;
+    border-bottom: solid darkblue 1px;
+    //min-height: 46px;
+  }
+
+  label {
+    display: inline-block;
+  }
+
+  .slider-label {
+    width: 97px;
+  }
+
+  .hidden-select {
+    display: none;
+  }
+
+  .percent-label {
+    padding: 2px 3px;
+    margin-left: 2px;
+    background-color: lightgray;
+    font-weight: bold;
+    color: orangered;
+  }
+
+  input[type=range] {
+    width: 153px;
+    padding: 1px;
+
+  }
+
+  input[type=range][disabled] {
+    background-color: #59afee;
+    cursor: not-allowed;
+  }
+
+  select {
+    margin-bottom: 2px;
+    width: 46px;
+  }
+
+  .low-high {
+    font-size: 12px;
+  }
+}
+
+// offset around show help button
+#tip-help {
+  margin-left: 10px;
+  margin-top: 4px;
+}
+
+#map {
+  background-color: slategray;
+  height: 100%;
+  flex: 1;
+  position: relative;
+}
+
+#loading-gif {
+  left: 10px;
+  top: 80px;
+  z-index: 10;
+  position: absolute;
+}
diff --git a/gulpfile.js b/gulpfile.js
new file mode 100644
index 0000000..8ba80e8
--- /dev/null
+++ b/gulpfile.js
@@ -0,0 +1,230 @@
+const gulp = require('gulp');
+const babel = require('gulp-babel');
+const babelify = require('babelify');
+const browserify = require('browserify');
+const buffer = require('vinyl-buffer');
+const source = require('vinyl-source-stream');
+const sourcemaps = require('gulp-sourcemaps');
+const minify = require('gulp-minify');
+const less = require('gulp-less');
+const cssmin = require('gulp-cssmin');
+const rename = require('gulp-rename');
+
+
+/**
+ *
+ * @param {string} inputFile - input file
+ * @param {string} outputFile - output file
+ * @param {boolean|*} runMinify - if should minify
+ * @returns {object} return stream
+ */
+function processJsFile(inputFile, outputFile, runMinify) {
+    "use strict";
+    runMinify = typeof runMinify == 'boolean' ? runMinify : false;
+    let pathParts = outputFile.split('/');
+    let outFileName = pathParts[pathParts.length - 1];
+    pathParts.splice(pathParts.length - 1, 1);
+    let outDir = pathParts.length === 0 ? '.' : pathParts.join('/');
+
+    let bundler = browserify({entries: inputFile, "debug": true, "extensions": ["js"]});
+
+    bundler.transform(babelify.configure({
+        presets: ["es2015"],
+        "ignore": /custom-ol-build|jquery.min/
+    }));
+
+
+    //bundler.transform(babelify.configure({
+    //    ignore: /custom-ol-build|jquery.min/
+    //}));
+    //let bundler = browserify({entries: inputFile, extensions: ['.js'], debug: true})
+    //    .transform(babelify.configure({
+    //        ignore: /custom-ol-build.js|jquery.min|/,
+    //        presets: ["es2015"],
+    //        extensions: [".ts", ".js"]
+    //    }));
+
+
+    //bundler.transform("babelify", {presets: ["es2015"], ignore: /custom-ol-build.js|jquery.min|/}, extensions: ['.js', '.ts']);
+
+
+    if (runMinify) {
+        return bundler.bundle()
+            .on('error', function (err) {
+                console.error(err);
+            })
+            .pipe(source(outFileName))
+            .pipe(buffer())
+            .pipe(sourcemaps.init({loadMaps: true}))
+            .pipe(minify({
+                ext: {
+                    src: '-debug.js',
+                    min: '.js'
+                },
+                exclude: ['tasks'],
+                ignoreFiles: ['.combo.js', '-min.js']
+            }))
+            .pipe(sourcemaps.write('./'))
+            .pipe(gulp.dest(outDir));
+    } else {
+        return bundler.bundle()
+            .on('error', function (err) {
+                console.error(err);
+            })
+            .pipe(source(outFileName))
+            .pipe(buffer())
+            .pipe(sourcemaps.init({loadMaps: true}))
+            .pipe(sourcemaps.write('./'))
+            .pipe(gulp.dest(outDir));
+    }
+}
+
+function processLessFile(inputFile, outputFile) {
+    "use strict";
+    let pathParts = outputFile.split('/');
+    let outFileName = pathParts[pathParts.length - 1];
+    pathParts.splice(pathParts.length - 1, 1);
+    let outDir = pathParts.length === 0 ? '.' : pathParts.join('/');
+
+    let fileNameParts = outFileName.split('.');
+
+    return gulp.src(inputFile)
+        .pipe(less().on('error', function (err) {
+            console.log(err);
+        }))
+        .pipe(cssmin().on('error', function (err) {
+            console.log(err);
+        }))
+        .pipe(rename({
+            basename: fileNameParts[0],
+            extname: '.' + fileNameParts[1],
+            suffix: '.min'
+        }))
+        .pipe(gulp.dest(outDir));
+}
+
+//gulp.task('default', function () {
+//    "use strict";
+//    return gulp.src('src/app.js')
+//        .pipe(babel())
+//        .pipe(gulp.dest('dist'))
+//});
+
+function _itsInventory(doMinify) {
+    "use strict";
+    //processLessFile('./flaskApp/blueprints/its_inventory/static/css/itsMap.less', './flaskApp/blueprints/its_inventory/static/_build/itsMap.css');
+
+    return processJsFile('./projects/itsMap.js', './build/itsMap.js', doMinify);
+}
+
+gulp.task('itsInventory-dev', () => {
+    return _itsInventory(false);
+});
+
+gulp.task('itsInventory-prod', () => {
+    return _itsInventory(true);
+});
+
+function _glrtoc(doMinify) {
+    "use strict";
+    processJsFile('./projects/glrtoc/main.js', './build/glrtoc/main.js', doMinify);
+
+    return processJsFile('./projects/glrtoc/legendTest.js', './build/glrtoc/legendTest.js', doMinify);
+}
+
+gulp.task('glrtoc-dev', () => {
+    "use strict";
+
+    return _glrtoc(false);
+});
+
+gulp.task('glrtoc-prod', () => {
+    "use strict";
+
+    return _glrtoc(true);
+});
+
+function _tsmo(doMinify) {
+    "use strict";
+    processJsFile('./projects/tsmo/legend-test.js', './build/legend-test.js', doMinify);
+    //processJsFile('./projects/tsmo/slider-test.js', './build/slider-test.js', doMinify);
+    //processJsFile('./projects/tsmo/main.js', './build/main.js', doMinify);
+    //return processJsFile('./projects/tsmo/main-report.js', './build/main-report.js', doMinify);
+}
+
+gulp.task('tsmo-dev', () => {
+    "use strict";
+
+    return _tsmo(false);
+});
+
+gulp.task('tsmo-prod', () => {
+    "use strict";
+
+    return _tsmo(true);
+});
+
+function _npmrds(doMinify) {
+    "use strict";
+
+    return processJsFile('./flaskApp/blueprints/npmrds/static/js/heatmap/main.js', './flaskApp/blueprints/npmrds/static/_build/heatmap-main.js', doMinify);
+}
+
+gulp.task('npmrds-dev', () => {
+    "use strict";
+
+    return _npmrds(false);
+});
+
+gulp.task('npmrds-prod', () => {
+    "use strict";
+
+    return _npmrds(true);
+});
+
+function _ssa(doMinify) {
+    "use strict";
+    processLessFile('./flaskApp/blueprints/testing/static/css/ssa-corridor.less', './flaskApp/blueprints/testing/static/_build/ssa-corridor.css');
+
+    return processJsFile('./flaskApp/blueprints/testing/static/js/ssa-main.js', './flaskApp/blueprints/testing/static/_build/ssa-main.js', doMinify);
+}
+
+gulp.task('ssa-dev', () => {
+    "use strict";
+
+    return _ssa(false);
+});
+
+gulp.task('ssa-prod', () => {
+    "use strict";
+
+    return _ssa(true);
+});
+
+gulp.task('peerGroup-dev', () => {
+    "use strict";
+
+    return processJsFile('./flaskApp/blueprints/peerGroup/static/js/main.js', './flaskApp/blueprints/peerGroup/static/_build/main.js', false);
+});
+
+gulp.task('peerGroup-prod', () => {
+    "use strict";
+
+    return processJsFile('./flaskApp/blueprints/peerGroup/static/js/main.js', './flaskApp/blueprints/peerGroup/static/_build/main.js', true);
+});
+
+function _buildTestApps() {
+    "use strict";
+    //processJsFile('./flaskApp/blueprints/testing/static/js/test-custom-build.js', './flaskApp/blueprints/testing/static/_build/test-custom-build.js', false);
+
+    return processJsFile('./flaskApp/blueprints/testing/static/js/test-corridor-layer.js', './flaskApp/blueprints/testing/static/_build/test-corridor-layer.js', false);
+
+}
+
+gulp.task('buildTestApps', () => {
+    "use strict";
+
+    return _buildTestApps(false);
+});
+
+gulp.task('build-prod', ['glrtoc-prod', 'tsmo-prod', 'npmrds-prod', 'ssa-prod', 'itsInventory-prod', 'peerGroup-prod']);
diff --git a/lib/jquery.floatThead.js b/lib/jquery.floatThead.js
new file mode 100644
index 0000000..42eba92
--- /dev/null
+++ b/lib/jquery.floatThead.js
@@ -0,0 +1,967 @@
+// @preserve jQuery.floatThead 1.3.2 - http://mkoryak.github.io/floatThead/ - Copyright (c) 2012 - 2015 Misha Koryak
+// @license MIT
+
+/* @author Misha Koryak
+ * @projectDescription lock a table header in place while scrolling - without breaking styles or events bound to the header
+ *
+ * Dependencies:
+ * jquery 1.9.0 + [required] OR jquery 1.7.0 + jquery UI core
+ *
+ * http://mkoryak.github.io/floatThead/
+ *
+ * Tested on FF13+, Chrome 21+, IE8, IE9, IE10, IE11
+ *
+ */
+let jQuery = require('jQuery');
+
+(function( $ ) {
+  /**
+   * provides a default config object. You can modify this after including this script if you want to change the init defaults
+   * @type {Object}
+   */
+  $.floatThead = $.floatThead || {};
+  $.floatThead.defaults = {
+    headerCellSelector: 'tr:visible:first>*:visible', //thead cells are this.
+    zIndex: 1001, //zindex of the floating thead (actually a container div)
+    position: 'auto', // 'fixed', 'absolute', 'auto'. auto picks the best for your table scrolling type.
+    top: 0, //String or function($table) - offset from top of window where the header should not pass above
+    bottom: 0, //String or function($table) - offset from the bottom of the table where the header should stop scrolling
+    scrollContainer: function($table){
+      return $([]); //if the table has horizontal scroll bars then this is the container that has overflow:auto and causes those scroll bars
+    },
+    getSizingRow: function($table, $cols, $fthCells){ // this is only called when using IE,
+      // override it if the first row of the table is going to contain colgroups (any cell spans greater than one col)
+      // it should return a jquery object containing a wrapped set of table cells comprising a row that contains no col spans and is visible
+      return $table.find('tbody tr:visible:first>*:visible');
+    },
+    floatTableClass: 'floatThead-table',
+    floatWrapperClass: 'floatThead-wrapper',
+    floatContainerClass: 'floatThead-container',
+    copyTableClass: true, //copy 'class' attribute from table into the floated table so that the styles match.
+    enableAria: false, //will copy header text from the floated header back into the table for screen readers. Might cause the css styling to be off. beware!
+    autoReflow: false, //(undocumented) - use MutationObserver api to reflow automatically when internal table DOM changes
+    debug: false //print possible issues (that don't prevent script loading) to console, if console exists.
+  };
+
+  var util = window._;
+
+  var canObserveMutations = typeof MutationObserver !== 'undefined';
+
+
+  //browser stuff
+  var ieVersion = function(){for(var a=3,b=document.createElement("b"),c=b.all||[];a = 1+a,b.innerHTML="<!--[if gt IE "+ a +"]><i><![endif]-->",c[0];);return 4<a?a:document.documentMode}();
+  var isFF = /Gecko\//.test(navigator.userAgent);
+  var isWebkit = /WebKit\//.test(navigator.userAgent);
+
+  //safari 7 (and perhaps others) reports table width to be parent container's width if max-width is set on table. see: https://github.com/mkoryak/floatThead/issues/108
+  var isTableWidthBug = function(){
+    if(isWebkit) {
+      var $test = $('<div style="width:0px"><table style="max-width:100%"><tr><th><div style="min-width:100px;">X</div></th></tr></table></div>');
+      $("body").append($test);
+      var ret = ($test.find("table").width() == 0);
+      $test.remove();
+      return ret;
+    }
+    return false;
+  };
+
+  var createElements = !isFF && !ieVersion; //FF can read width from <col> elements, but webkit cannot
+
+  var $window = $(window);
+
+  /**
+   * @param debounceMs
+   * @param cb
+   */
+  function windowResize(eventName, cb){
+    if(ieVersion == 8){ //ie8 is crap: https://github.com/mkoryak/floatThead/issues/65
+      var winWidth = $window.width();
+      var debouncedCb = util.debounce(function(){
+        var winWidthNew = $window.width();
+        if(winWidth != winWidthNew){
+          winWidth = winWidthNew;
+          cb();
+        }
+      }, 1);
+      $window.on(eventName, debouncedCb);
+    } else {
+      $window.on(eventName, util.debounce(cb, 1));
+    }
+  }
+
+
+  function debug(str){
+    window && window.console && window.console.error && window.console.error("jQuery.floatThead: " + str);
+  }
+
+  //returns fractional pixel widths
+  function getOffsetWidth(el) {
+    var rect = el.getBoundingClientRect();
+    return rect.width || rect.right - rect.left;
+  }
+
+  /**
+   * try to calculate the scrollbar width for your browser/os
+   * @return {Number}
+   */
+  function scrollbarWidth() {
+    var $div = $( //borrowed from anti-scroll
+        '<div style="width:50px;height:50px;overflow-y:scroll;'
+        + 'position:absolute;top:-200px;left:-200px;"><div style="height:100px;width:100%">'
+        + '</div>'
+    );
+    $('body').append($div);
+    var w1 = $div.innerWidth();
+    var w2 = $('div', $div).innerWidth();
+    $div.remove();
+    return w1 - w2;
+  }
+  /**
+   * Check if a given table has been datatableized (http://datatables.net)
+   * @param $table
+   * @return {Boolean}
+   */
+  function isDatatable($table){
+    if($table.dataTableSettings){
+      for(var i = 0; i < $table.dataTableSettings.length; i++){
+        var table = $table.dataTableSettings[i].nTable;
+        if($table[0] == table){
+          return true;
+        }
+      }
+    }
+    return false;
+  }
+
+  function tableWidth($table, $fthCells, isOuter){
+    // see: https://github.com/mkoryak/floatThead/issues/108
+    var fn = isOuter ? "outerWidth": "width";
+    if(isTableWidthBug && $table.css("max-width")){
+      var w = 0;
+      if(isOuter) {
+        w += parseInt($table.css("borderLeft"), 10);
+        w += parseInt($table.css("borderRight"), 10);
+      }
+      for(var i=0; i < $fthCells.length; i++){
+        w += $fthCells.get(i).offsetWidth;
+      }
+      return w;
+    } else {
+      return $table[fn]();
+    }
+  }
+  $.fn.floatThead = function(map){
+    map = map || {};
+    if(!util){ //may have been included after the script? lets try to grab it again.
+      util = window._ || $.floatThead._;
+      if(!util){
+        throw new Error("jquery.floatThead-slim.js requires underscore. You should use the non-lite version since you do not have underscore.");
+      }
+    }
+
+    if(ieVersion < 8){
+      return this; //no more crappy browser support.
+    }
+
+    var mObs = null; //mutation observer lives in here if we can use it / make it
+
+    if(util.isFunction(isTableWidthBug)) {
+      isTableWidthBug = isTableWidthBug();
+    }
+
+    if(util.isString(map)){
+      var command = map;
+      var ret = this;
+      this.filter('table').each(function(){
+        var $this = $(this);
+        var opts = $this.data('floatThead-lazy');
+        if(opts){
+          $this.floatThead(opts);
+        }
+        var obj = $this.data('floatThead-attached');
+        if(obj && util.isFunction(obj[command])){
+          var r = obj[command]();
+          if(typeof r !== 'undefined'){
+            ret = r;
+          }
+        }
+      });
+      return ret;
+    }
+    var opts = $.extend({}, $.floatThead.defaults || {}, map);
+
+    $.each(map, function(key, val){
+      if((!(key in $.floatThead.defaults)) && opts.debug){
+        debug("Used ["+key+"] key to init plugin, but that param is not an option for the plugin. Valid options are: "+ (util.keys($.floatThead.defaults)).join(', '));
+      }
+    });
+    if(opts.debug){
+      var v = $.fn.jquery.split(".");
+      if(parseInt(v[0], 10) == 1 && parseInt(v[1], 10) <= 7){
+        debug("jQuery version "+$.fn.jquery+" detected! This plugin supports 1.8 or better, or 1.7.x with jQuery UI 1.8.24 -> http://jqueryui.com/resources/download/jquery-ui-1.8.24.zip")
+      }
+    }
+
+    this.filter(':not(.'+opts.floatTableClass+')').each(function(){
+      var floatTheadId = util.uniqueId();
+      var $table = $(this);
+      if($table.data('floatThead-attached')){
+        return true; //continue the each loop
+      }
+      if(!$table.is('table')){
+        throw new Error('jQuery.floatThead must be run on a table element. ex: $("table").floatThead();');
+      }
+      canObserveMutations = opts.autoReflow && canObserveMutations; //option defaults to false!
+      var $header = $table.children('thead:first');
+      var $tbody = $table.children('tbody:first');
+      if($header.length == 0 || $tbody.length == 0){
+        $table.data('floatThead-lazy', opts);
+        $table.unbind("reflow").one('reflow', function(){
+          $table.floatThead(opts);
+        });
+        return;
+      }
+      if($table.data('floatThead-lazy')){
+        $table.unbind("reflow");
+      }
+      $table.data('floatThead-lazy', false);
+
+      var headerFloated = true;
+      var scrollingTop, scrollingBottom;
+      var scrollbarOffset = {vertical: 0, horizontal: 0};
+      var scWidth = scrollbarWidth();
+      var lastColumnCount = 0; //used by columnNum()
+      var $scrollContainer = opts.scrollContainer($table) || $([]); //guard against returned nulls
+      var locked = $scrollContainer.length > 0;
+
+      var useAbsolutePositioning = null;
+      if(typeof opts.useAbsolutePositioning !== 'undefined'){
+        opts.position = 'auto';
+        if(opts.useAbsolutePositioning){
+          opts.position = opts.useAbsolutePositioning ? 'absolute' : 'fixed';
+        }
+        debug("option 'useAbsolutePositioning' has been removed in v1.3.0, use `position:'"+opts.position+"'` instead. See docs for more info: http://mkoryak.github.io/floatThead/#options")
+      }
+      if(typeof opts.scrollingTop !== 'undefined'){
+        opts.top = opts.scrollingTop;
+        debug("option 'scrollingTop' has been renamed to 'top' in v1.3.0. See docs for more info: http://mkoryak.github.io/floatThead/#options");
+      }
+      if(typeof opts.scrollingBottom !== 'undefined'){
+          opts.bottom = opts.scrollingBottom;
+          debug("option 'scrollingBottom' has been renamed to 'bottom' in v1.3.0. See docs for more info: http://mkoryak.github.io/floatThead/#options");
+      }
+
+
+      if (opts.position == 'auto') {
+        useAbsolutePositioning = null;
+      } else if (opts.position == 'fixed') {
+        useAbsolutePositioning = false;
+      } else if (opts.position == 'absolute'){
+        useAbsolutePositioning = true;
+      } else if (opts.debug) {
+        debug('Invalid value given to "position" option, valid is "fixed", "absolute" and "auto". You passed: ', opts.position);
+      }
+
+      if(useAbsolutePositioning == null){ //defaults: locked=true, !locked=false
+        useAbsolutePositioning = locked;
+      }
+      var $caption = $table.find("caption");
+      var haveCaption = $caption.length == 1;
+      if(haveCaption){
+        var captionAlignTop = ($caption.css("caption-side") || $caption.attr("align") || "top") === "top";
+      }
+
+      var $fthGrp = $('<fthfoot style="display:table-footer-group;border-spacing:0;height:0;border-collapse:collapse;visibility:hidden"/>');
+
+      var wrappedContainer = false; //used with absolute positioning enabled. did we need to wrap the scrollContainer/table with a relative div?
+      var $wrapper = $([]); //used when absolute positioning enabled - wraps the table and the float container
+      var absoluteToFixedOnScroll = ieVersion <= 9 && !locked && useAbsolutePositioning; //on IE using absolute positioning doesn't look good with window scrolling, so we change position to fixed on scroll, and then change it back to absolute when done.
+      var $floatTable = $("<table/>");
+      var $floatColGroup = $("<colgroup/>");
+      var $tableColGroup = $table.children('colgroup:first');
+      var existingColGroup = true;
+      if($tableColGroup.length == 0){
+        $tableColGroup = $("<colgroup/>");
+        existingColGroup = false;
+      }
+      var $fthRow = $('<fthtr style="display:table-row;border-spacing:0;height:0;border-collapse:collapse"/>'); //created unstyled elements (used for sizing the table because chrome can't read <col> width)
+      var $floatContainer = $('<div style="overflow: hidden;" aria-hidden="true"></div>');
+      var floatTableHidden = false; //this happens when the table is hidden and we do magic when making it visible
+      var $newHeader = $("<thead/>");
+      var $sizerRow = $('<tr class="size-row"/>');
+      var $sizerCells = $([]);
+      var $tableCells = $([]); //used for sizing - either $sizerCells or $tableColGroup cols. $tableColGroup cols are only created in chrome for borderCollapse:collapse because of a chrome bug.
+      var $headerCells = $([]);
+      var $fthCells = $([]); //created elements
+
+      $newHeader.append($sizerRow);
+      $table.prepend($tableColGroup);
+      if(createElements){
+        $fthGrp.append($fthRow);
+        $table.append($fthGrp);
+      }
+
+      $floatTable.append($floatColGroup);
+      $floatContainer.append($floatTable);
+      if(opts.copyTableClass){
+        $floatTable.attr('class', $table.attr('class'));
+      }
+      $floatTable.attr({ //copy over some deprecated table attributes that people still like to use. Good thing people don't use colgroups...
+        'cellpadding': $table.attr('cellpadding'),
+        'cellspacing': $table.attr('cellspacing'),
+        'border': $table.attr('border')
+      });
+      var tableDisplayCss = $table.css('display');
+      $floatTable.css({
+        'borderCollapse': $table.css('borderCollapse'),
+        'border': $table.css('border'),
+        'display': tableDisplayCss
+      });
+      if(tableDisplayCss == 'none'){
+        floatTableHidden = true;
+      }
+
+      $floatTable.addClass(opts.floatTableClass).css({'margin': 0, 'border-bottom-width': 0}); //must have no margins or you won't be able to click on things under floating table
+
+      if(useAbsolutePositioning){
+        var makeRelative = function($container, alwaysWrap){
+          var positionCss = $container.css('position');
+          var relativeToScrollContainer = (positionCss == "relative" || positionCss == "absolute");
+          var $containerWrap = $container;
+          if(!relativeToScrollContainer || alwaysWrap){
+            var css = {"paddingLeft": $container.css('paddingLeft'), "paddingRight": $container.css('paddingRight')};
+            $floatContainer.css(css);
+            $containerWrap = $container.data('floatThead-containerWrap') || $container.wrap("<div class='"+opts.floatWrapperClass+"' style='position: relative; clear:both;'></div>").parent();
+            $container.data('floatThead-containerWrap', $containerWrap); //multiple tables inside one scrolling container - #242
+            wrappedContainer = true;
+          }
+          return $containerWrap;
+        };
+        if(locked){
+          $wrapper = makeRelative($scrollContainer, true);
+          $wrapper.prepend($floatContainer);
+        } else {
+          $wrapper = makeRelative($table);
+          $table.before($floatContainer);
+        }
+      } else {
+        $table.before($floatContainer);
+      }
+
+
+      $floatContainer.css({
+        position: useAbsolutePositioning ? 'absolute' : 'fixed',
+        marginTop: 0,
+        top:  useAbsolutePositioning ? 0 : 'auto',
+        zIndex: opts.zIndex
+      });
+      $floatContainer.addClass(opts.floatContainerClass);
+      updateScrollingOffsets();
+
+      var layoutFixed = {'table-layout': 'fixed'};
+      var layoutAuto = {'table-layout': $table.css('tableLayout') || 'auto'};
+      var originalTableWidth = $table[0].style.width || ""; //setting this to auto is bad: #70
+      var originalTableMinWidth = $table.css('minWidth') || "";
+
+      function eventName(name){
+        return name+'.fth-'+floatTheadId+'.floatTHead'
+      }
+
+      function setHeaderHeight(){
+        var headerHeight = 0;
+        $header.children("tr:visible").each(function(){
+          headerHeight += $(this).outerHeight(true);
+        });
+        if($table.css('border-collapse') == 'collapse') {
+          var tableBorderTopHeight = parseInt($table.css('border-top-width'), 10);
+          var cellBorderTopHeight = parseInt($table.find("thead tr:first").find(">*:first").css('border-top-width'), 10);
+          if(tableBorderTopHeight > cellBorderTopHeight) {
+            headerHeight -= (tableBorderTopHeight / 2); //id love to see some docs where this magic recipe is found..
+          }
+        }
+        $sizerRow.outerHeight(headerHeight);
+        $sizerCells.outerHeight(headerHeight);
+      }
+
+
+      function setFloatWidth(){
+        var tw = tableWidth($table, $fthCells, true);
+        var width = $scrollContainer.width() || tw;
+        var floatContainerWidth = $scrollContainer.css("overflow-y") != 'hidden' ? width - scrollbarOffset.vertical : width;
+        $floatContainer.width(floatContainerWidth);
+        if(locked){
+          var percent = 100 * tw / (floatContainerWidth);
+          $floatTable.css('width', percent+'%');
+        } else {
+          $floatTable.outerWidth(tw);
+        }
+      }
+
+      function updateScrollingOffsets(){
+        scrollingTop = (util.isFunction(opts.top) ? opts.top($table) : opts.top) || 0;
+        scrollingBottom = (util.isFunction(opts.bottom) ? opts.bottom($table) : opts.bottom) || 0;
+      }
+
+      /**
+       * get the number of columns and also rebuild resizer rows if the count is different than the last count
+       */
+      function columnNum(){
+        var count;
+        var $headerColumns = $header.find(opts.headerCellSelector);
+        if(existingColGroup){
+          count = $tableColGroup.find('col').length;
+        } else {
+          count = 0;
+          $headerColumns.each(function () {
+              count += parseInt(($(this).attr('colspan') || 1), 10);
+          });
+        }
+        if(count != lastColumnCount){
+          lastColumnCount = count;
+          var cells = [], cols = [], psuedo = [], content;
+          for(var x = 0; x < count; x++){
+            if (opts.enableAria && (content = $headerColumns.eq(x).text()) ) {
+              cells.push('<th scope="col" class="floatThead-col">' + content + '</th>');
+            } else {
+              cells.push('<th class="floatThead-col"/>');
+            }
+            cols.push('<col/>');
+            psuedo.push("<fthtd style='display:table-cell;height:0;width:auto;'/>");
+          }
+
+          cols = cols.join('');
+          cells = cells.join('');
+
+          if(createElements){
+            psuedo = psuedo.join('');
+            $fthRow.html(psuedo);
+            $fthCells = $fthRow.find('fthtd');
+          }
+
+          $sizerRow.html(cells);
+          $sizerCells = $sizerRow.find("th");
+          if(!existingColGroup){
+            $tableColGroup.html(cols);
+          }
+          $tableCells = $tableColGroup.find('col');
+          $floatColGroup.html(cols);
+          $headerCells = $floatColGroup.find("col");
+
+        }
+        return count;
+      }
+
+      function refloat(){ //make the thing float
+        if(!headerFloated){
+          headerFloated = true;
+          if(useAbsolutePositioning){ //#53, #56
+            var tw = tableWidth($table, $fthCells, true);
+            var wrapperWidth = $wrapper.width();
+            if(tw > wrapperWidth){
+              $table.css('minWidth', tw);
+            }
+          }
+          $table.css(layoutFixed);
+          $floatTable.css(layoutFixed);
+          $floatTable.append($header); //append because colgroup must go first in chrome
+          $tbody.before($newHeader);
+          setHeaderHeight();
+        }
+      }
+      function unfloat(){ //put the header back into the table
+        if(headerFloated){
+          headerFloated = false;
+          if(useAbsolutePositioning){ //#53, #56
+            $table.width(originalTableWidth);
+          }
+          $newHeader.detach();
+          $table.prepend($header);
+          $table.css(layoutAuto);
+          $floatTable.css(layoutAuto);
+          $table.css('minWidth', originalTableMinWidth); //this looks weird, but it's not a bug. Think about it!!
+          $table.css('minWidth', tableWidth($table, $fthCells)); //#121
+        }
+      }
+      var isHeaderFloatingLogical = false; //for the purpose of this event, the header is/isnt floating, even though the element
+                                           //might be in some other state. this is what the header looks like to the user
+      function triggerFloatEvent(isFloating){
+        if(isHeaderFloatingLogical != isFloating){
+          isHeaderFloatingLogical = isFloating;
+          $table.triggerHandler("floatThead", [isFloating, $floatContainer])
+        }
+      }
+      function changePositioning(isAbsolute){
+        if(useAbsolutePositioning != isAbsolute){
+          useAbsolutePositioning = isAbsolute;
+          $floatContainer.css({
+            position: useAbsolutePositioning ? 'absolute' : 'fixed'
+          });
+        }
+      }
+      function getSizingRow($table, $cols, $fthCells, ieVersion){
+        if(createElements){
+          return $fthCells;
+        } else if(ieVersion) {
+          return opts.getSizingRow($table, $cols, $fthCells);
+        } else {
+          return $cols;
+        }
+      }
+
+      /**
+       * returns a function that updates the floating header's cell widths.
+       * @return {Function}
+       */
+      function reflow(){
+        var i;
+        var numCols = columnNum(); //if the tables columns changed dynamically since last time (datatables), rebuild the sizer rows and get a new count
+
+        return function(){
+          $tableCells = $tableColGroup.find('col');
+          var $rowCells = getSizingRow($table, $tableCells, $fthCells, ieVersion);
+
+          if($rowCells.length == numCols && numCols > 0){
+            if(!existingColGroup){
+              for(i=0; i < numCols; i++){
+                $tableCells.eq(i).css('width', '');
+              }
+            }
+            unfloat();
+            var widths = [];
+            for(i=0; i < numCols; i++){
+              widths[i] = getOffsetWidth($rowCells.get(i));
+            }
+            for(i=0; i < numCols; i++){
+              $headerCells.eq(i).width(widths[i]);
+              $tableCells.eq(i).width(widths[i]);
+            }
+            refloat();
+          } else {
+            $floatTable.append($header);
+            $table.css(layoutAuto);
+            $floatTable.css(layoutAuto);
+            setHeaderHeight();
+          }
+          $table.triggerHandler("reflowed", [$floatContainer]);
+        };
+      }
+
+      function floatContainerBorderWidth(side){
+        var border = $scrollContainer.css("border-"+side+"-width");
+        var w = 0;
+        if (border && ~border.indexOf('px')) {
+          w = parseInt(border, 10);
+        }
+        return w;
+      }
+      /**
+       * first performs initial calculations that we expect to not change when the table, window, or scrolling container are scrolled.
+       * returns a function that calculates the floating container's top and left coords. takes into account if we are using page scrolling or inner scrolling
+       * @return {Function}
+       */
+      function calculateFloatContainerPosFn(){
+        var scrollingContainerTop = $scrollContainer.scrollTop();
+
+        //this floatEnd calc was moved out of the returned function because we assume the table height doesn't change (otherwise we must reinit by calling calculateFloatContainerPosFn)
+        var floatEnd;
+        var tableContainerGap = 0;
+        var captionHeight = haveCaption ? $caption.outerHeight(true) : 0;
+        var captionScrollOffset = captionAlignTop ? captionHeight : -captionHeight;
+
+        var floatContainerHeight = $floatContainer.height();
+        var tableOffset = $table.offset();
+        var tableLeftGap = 0; //can be caused by border on container (only in locked mode)
+        var tableTopGap = 0;
+        if(locked){
+          var containerOffset = $scrollContainer.offset();
+          tableContainerGap = tableOffset.top - containerOffset.top + scrollingContainerTop;
+          if(haveCaption && captionAlignTop){
+            tableContainerGap += captionHeight;
+          }
+          tableLeftGap = floatContainerBorderWidth('left');
+          tableTopGap = floatContainerBorderWidth('top');
+          tableContainerGap -= tableTopGap;
+        } else {
+          floatEnd = tableOffset.top - scrollingTop - floatContainerHeight + scrollingBottom + scrollbarOffset.horizontal;
+        }
+        var windowTop = $window.scrollTop();
+        var windowLeft = $window.scrollLeft();
+        var scrollContainerLeft =  $scrollContainer.scrollLeft();
+
+        return function(eventType){
+          var isTableHidden = $table[0].offsetWidth <= 0 && $table[0].offsetHeight <= 0;
+          if(!isTableHidden && floatTableHidden) {
+            floatTableHidden = false;
+            setTimeout(function(){
+              $table.triggerHandler("reflow");
+            }, 1);
+            return null;
+          }
+          if(isTableHidden){ //it's hidden
+            floatTableHidden = true;
+            if(!useAbsolutePositioning){
+              return null;
+            }
+          }
+
+          if(eventType == 'windowScroll'){
+            windowTop = $window.scrollTop();
+            windowLeft = $window.scrollLeft();
+          } else if(eventType == 'containerScroll'){
+            scrollingContainerTop = $scrollContainer.scrollTop();
+            scrollContainerLeft =  $scrollContainer.scrollLeft();
+          } else if(eventType != 'init') {
+            windowTop = $window.scrollTop();
+            windowLeft = $window.scrollLeft();
+            scrollingContainerTop = $scrollContainer.scrollTop();
+            scrollContainerLeft =  $scrollContainer.scrollLeft();
+          }
+          if(isWebkit && (windowTop < 0 || windowLeft < 0)){ //chrome overscroll effect at the top of the page - breaks fixed positioned floated headers
+            return;
+          }
+
+          if(absoluteToFixedOnScroll){
+            if(eventType == 'windowScrollDone'){
+              changePositioning(true); //change to absolute
+            } else {
+              changePositioning(false); //change to fixed
+            }
+          } else if(eventType == 'windowScrollDone'){
+            return null; //event is fired when they stop scrolling. ignore it if not 'absoluteToFixedOnScroll'
+          }
+
+          tableOffset = $table.offset();
+          if(haveCaption && captionAlignTop){
+            tableOffset.top += captionHeight;
+          }
+          var top, left;
+          var tableHeight = $table.outerHeight();
+
+          if(locked && useAbsolutePositioning){ //inner scrolling, absolute positioning
+            if (tableContainerGap >= scrollingContainerTop) {
+              var gap = tableContainerGap - scrollingContainerTop + tableTopGap;
+              top = gap > 0 ? gap : 0;
+              triggerFloatEvent(false);
+            } else {
+              top = wrappedContainer ? tableTopGap : scrollingContainerTop;
+              //headers stop at the top of the viewport
+              triggerFloatEvent(true);
+            }
+            left = tableLeftGap;
+          } else if(!locked && useAbsolutePositioning) { //window scrolling, absolute positioning
+            if(windowTop > floatEnd + tableHeight + captionScrollOffset){
+              top = tableHeight - floatContainerHeight + captionScrollOffset; //scrolled past table
+            } else if (tableOffset.top >= windowTop + scrollingTop) {
+              top = 0; //scrolling to table
+              unfloat();
+              triggerFloatEvent(false);
+            } else {
+              top = scrollingTop + windowTop - tableOffset.top + tableContainerGap + (captionAlignTop ? captionHeight : 0);
+              refloat(); //scrolling within table. header floated
+              triggerFloatEvent(true);
+            }
+            left =  0;
+          } else if(locked && !useAbsolutePositioning){ //inner scrolling, fixed positioning
+            if (tableContainerGap > scrollingContainerTop || scrollingContainerTop - tableContainerGap > tableHeight) {
+              top = tableOffset.top - windowTop;
+              unfloat();
+              triggerFloatEvent(false);
+            } else {
+              top = tableOffset.top + scrollingContainerTop  - windowTop - tableContainerGap;
+              refloat();
+              triggerFloatEvent(true);
+              //headers stop at the top of the viewport
+            }
+            left = tableOffset.left + scrollContainerLeft - windowLeft;
+          } else if(!locked && !useAbsolutePositioning) { //window scrolling, fixed positioning
+            if(windowTop > floatEnd + tableHeight + captionScrollOffset){
+              top = tableHeight + scrollingTop - windowTop + floatEnd + captionScrollOffset;
+              //scrolled past the bottom of the table
+            } else if (tableOffset.top > windowTop + scrollingTop) {
+              top = tableOffset.top - windowTop;
+              refloat();
+              triggerFloatEvent(false); //this is a weird case, the header never gets unfloated and i have no no way to know
+              //scrolled past the top of the table
+            } else {
+              //scrolling within the table
+              top = scrollingTop;
+              triggerFloatEvent(true);
+            }
+            left = tableOffset.left - windowLeft;
+          }
+          return {top: top, left: left};
+        };
+      }
+      /**
+       * returns a function that caches old floating container position and only updates css when the position changes
+       * @return {Function}
+       */
+      function repositionFloatContainerFn(){
+        var oldTop = null;
+        var oldLeft = null;
+        var oldScrollLeft = null;
+        return function(pos, setWidth, setHeight){
+          if(pos != null && (oldTop != pos.top || oldLeft != pos.left)){
+            $floatContainer.css({
+              top: pos.top,
+              left: pos.left
+            });
+            oldTop = pos.top;
+            oldLeft = pos.left;
+          }
+          if(setWidth){
+            setFloatWidth();
+          }
+          if(setHeight){
+            setHeaderHeight();
+          }
+          var scrollLeft = $scrollContainer.scrollLeft();
+          if(!useAbsolutePositioning || oldScrollLeft != scrollLeft){
+            $floatContainer.scrollLeft(scrollLeft);
+            oldScrollLeft = scrollLeft;
+          }
+        }
+      }
+
+      /**
+       * checks if THIS table has scrollbars, and finds their widths
+       */
+      function calculateScrollBarSize(){ //this should happen after the floating table has been positioned
+        if($scrollContainer.length){
+          if($scrollContainer.data().perfectScrollbar){
+            scrollbarOffset = {horizontal:0, vertical:0};
+          } else {
+            var sw = $scrollContainer.width(), sh = $scrollContainer.height(), th = $table.height(), tw = tableWidth($table, $fthCells);
+            var offseth = sw < tw ? scWidth : 0;
+            var offsetv = sh < th ? scWidth : 0;
+            scrollbarOffset.horizontal = sw - offsetv < tw ? scWidth : 0;
+            scrollbarOffset.vertical = sh - offseth < th ? scWidth : 0;
+          }
+        }
+      }
+      //finish up. create all calculation functions and bind them to events
+      calculateScrollBarSize();
+
+      var flow;
+
+      var ensureReflow = function(){
+        flow = reflow();
+        flow();
+      };
+
+      ensureReflow();
+
+      var calculateFloatContainerPos = calculateFloatContainerPosFn();
+      var repositionFloatContainer = repositionFloatContainerFn();
+
+      repositionFloatContainer(calculateFloatContainerPos('init'), true); //this must come after reflow because reflow changes scrollLeft back to 0 when it rips out the thead
+
+      var windowScrollDoneEvent = util.debounce(function(){
+        repositionFloatContainer(calculateFloatContainerPos('windowScrollDone'), false);
+      }, 1);
+
+      var windowScrollEvent = function(){
+        repositionFloatContainer(calculateFloatContainerPos('windowScroll'), false);
+        if(absoluteToFixedOnScroll){
+          windowScrollDoneEvent();
+        }
+      };
+      var containerScrollEvent = function(){
+        repositionFloatContainer(calculateFloatContainerPos('containerScroll'), false);
+      };
+
+
+      var windowResizeEvent = function(){
+        if($table.is(":hidden")){
+          return;
+        }
+        updateScrollingOffsets();
+        calculateScrollBarSize();
+        ensureReflow();
+        calculateFloatContainerPos = calculateFloatContainerPosFn();
+        repositionFloatContainer = repositionFloatContainerFn();
+        repositionFloatContainer(calculateFloatContainerPos('resize'), true, true);
+      };
+      var reflowEvent = util.debounce(function(){
+        if($table.is(":hidden")){
+          return;
+        }
+        calculateScrollBarSize();
+        updateScrollingOffsets();
+        ensureReflow();
+        calculateFloatContainerPos = calculateFloatContainerPosFn();
+        repositionFloatContainer(calculateFloatContainerPos('reflow'), true);
+      }, 1);
+      if(locked){ //internal scrolling
+        if(useAbsolutePositioning){
+          $scrollContainer.on(eventName('scroll'), containerScrollEvent);
+        } else {
+          $scrollContainer.on(eventName('scroll'), containerScrollEvent);
+          $window.on(eventName('scroll'), windowScrollEvent);
+        }
+      } else { //window scrolling
+        $window.on(eventName('scroll'), windowScrollEvent);
+      }
+
+      $window.on(eventName('load'), reflowEvent); //for tables with images
+
+      windowResize(eventName('resize'), windowResizeEvent);
+      $table.on('reflow', reflowEvent);
+      if(isDatatable($table)){
+        $table
+          .on('filter', reflowEvent)
+          .on('sort',   reflowEvent)
+          .on('page',   reflowEvent);
+      }
+
+      $window.on(eventName('shown.bs.tab'), reflowEvent); // people cant seem to figure out how to use this plugin with bs3 tabs... so this :P
+      $window.on(eventName('tabsactivate'), reflowEvent); // same thing for jqueryui
+
+
+      if (canObserveMutations) {
+        var mutationElement = null;
+        if(util.isFunction(opts.autoReflow)){
+          mutationElement = opts.autoReflow($table, $scrollContainer)
+        }
+        if(!mutationElement) {
+          mutationElement = $scrollContainer.length ? $scrollContainer[0] : $table[0]
+        }
+        mObs = new MutationObserver(function(e){
+          var wasTableRelated = function(nodes){
+            return nodes && nodes[0] && (nodes[0].nodeName == "THEAD" || nodes[0].nodeName == "TD"|| nodes[0].nodeName == "TH");
+          };
+          for(var i=0; i < e.length; i++){
+            if(!(wasTableRelated(e[i].addedNodes) || wasTableRelated(e[i].removedNodes))){
+              reflowEvent();
+              break;
+            }
+          }
+        });
+        mObs.observe(mutationElement, {
+            childList: true,
+            subtree: true
+        });
+      }
+
+      //attach some useful functions to the table.
+      $table.data('floatThead-attached', {
+        destroy: function(){
+          var ns = '.fth-'+floatTheadId;
+          unfloat();
+          $table.css(layoutAuto);
+          $tableColGroup.remove();
+          createElements && $fthGrp.remove();
+          if($newHeader.parent().length){ //only if it's in the DOM
+            $newHeader.replaceWith($header);
+          }
+          if(canObserveMutations){
+            mObs.disconnect();
+            mObs = null;
+          }
+          $table.off('reflow reflowed');
+          $scrollContainer.off(ns);
+          if (wrappedContainer) {
+            if ($scrollContainer.length) {
+              $scrollContainer.unwrap();
+            }
+            else {
+              $table.unwrap();
+            }
+          }
+          if(locked){
+            $scrollContainer.data('floatThead-containerWrap', false);
+          } else {
+            $table.data('floatThead-containerWrap', false);
+          }
+          $table.css('minWidth', originalTableMinWidth);
+          $floatContainer.remove();
+          $table.data('floatThead-attached', false);
+          $window.off(ns);
+        },
+        reflow: function(){
+          reflowEvent();
+        },
+        setHeaderHeight: function(){
+          setHeaderHeight();
+        },
+        getFloatContainer: function(){
+          return $floatContainer;
+        },
+        getRowGroups: function(){
+          if(headerFloated){
+            return $floatContainer.find('>table>thead').add($table.children("tbody,tfoot"));
+          } else {
+            return $table.children("thead,tbody,tfoot");
+          }
+        }
+      });
+    });
+    return this;
+  };
+})(jQuery);
+
+/* jQuery.floatThead.utils - http://mkoryak.github.io/floatThead/ - Copyright (c) 2012 - 2014 Misha Koryak
+ * License: MIT
+ *
+ * This file is required if you do not use underscore in your project and you want to use floatThead.
+ * It contains functions from underscore that the plugin uses.
+ *
+ * YOU DON'T NEED TO INCLUDE THIS IF YOU ALREADY INCLUDE UNDERSCORE!
+ *
+ */
+
+(function($){
+
+  $.floatThead = $.floatThead || {};
+
+  $.floatThead._  = window._ || (function(){
+    var that = {};
+    var hasOwnProperty = Object.prototype.hasOwnProperty, isThings = ['Arguments', 'Function', 'String', 'Number', 'Date', 'RegExp'];
+    that.has = function(obj, key) {
+      return hasOwnProperty.call(obj, key);
+    };
+    that.keys = function(obj) {
+      if (obj !== Object(obj)) throw new TypeError('Invalid object');
+      var keys = [];
+      for (var key in obj) if (that.has(obj, key)) keys.push(key);
+      return keys;
+    };
+    var idCounter = 0;
+    that.uniqueId = function(prefix) {
+      var id = ++idCounter + '';
+      return prefix ? prefix + id : id;
+    };
+    $.each(isThings, function(){
+      var name = this;
+      that['is' + name] = function(obj) {
+        return Object.prototype.toString.call(obj) == '[object ' + name + ']';
+      };
+    });
+    that.debounce = function(func, wait, immediate) {
+      var timeout, args, context, timestamp, result;
+      return function() {
+        context = this;
+        args = arguments;
+        timestamp = new Date();
+        var later = function() {
+          var last = (new Date()) - timestamp;
+          if (last < wait) {
+            timeout = setTimeout(later, wait - last);
+          } else {
+            timeout = null;
+            if (!immediate) result = func.apply(context, args);
+          }
+        };
+        var callNow = immediate && !timeout;
+        if (!timeout) {
+          timeout = setTimeout(later, wait);
+        }
+        if (callNow) result = func.apply(context, args);
+        return result;
+      };
+    };
+    return that;
+  })();
+})(jQuery);
+
+export default undefined;
\ No newline at end of file
diff --git a/package.json b/package.json
index df446ff..bef02ff 100644
--- a/package.json
+++ b/package.json
@@ -12,11 +12,34 @@
   "author": "TOPS Lab",
   "license": "ISC",
   "devDependencies": {
+    "angular2": "^2.0.0-beta.17",
+    "babel-cli": "^6.8.0",
     "babel-preset-es2015": "^6.6.0",
-    "openlayers": "^3.15.1"
+    "babelify": "^7.3.0",
+    "browserify": "^13.0.0",
+    "es6-mixins": "^1.0.2",
+    "es6-shim": "^0.35.0",
+    "gulp": "^3.9.1",
+    "gulp-babel": "^6.1.2",
+    "gulp-cssmin": "^0.1.7",
+    "gulp-less": "^3.0.5",
+    "gulp-minify": "0.0.11",
+    "gulp-rename": "^1.2.2",
+    "gulp-sourcemaps": "^2.0.0-alpha",
+    "jquery": "^2.2.3",
+    "jquery-ui": "^1.10.5",
+    "openlayers": "^3.15.1",
+    "reflect-metadata": "^0.1.2",
+    "rxjs": "^5.0.0-beta.6",
+    "vinyl-buffer": "^1.0.0",
+    "vinyl-source-stream": "^1.1.0",
+    "zone.js": "^0.6.12"
   },
   "repository": {
     "type": "git",
     "url": "https://github.com/glennvorhes/webmapsjs.git"
+  },
+  "dependencies": {
+    "angular2": "^2.0.0-beta.17"
   }
 }
diff --git a/projects/glrtoc/appConfig.js b/projects/glrtoc/appConfig.js
new file mode 100644
index 0000000..bc44356
--- /dev/null
+++ b/projects/glrtoc/appConfig.js
@@ -0,0 +1,150 @@
+/**
+ * Created by gavorhes on 12/8/2015.
+ */
+
+import mapPopup from '../../src/olHelpers/mapPopup';
+import $ from '../../src/jquery';
+let homeTabId = 'home-tab';
+let coordinationTabId = 'coordination-tab';
+let operationsTabId = 'operations-tab';
+
+let operationsStaticPanelId = 'operations-static';
+let operationsAnimatedPanelId = 'operations-animated';
+
+class AppConfig {
+    /**
+     * app configuration object
+     */
+    constructor() {
+        this.$opsAccordion = $("#" + operationsTabId);
+        this.debug = true;
+        this.map = undefined;
+        this._coordinationLayer = undefined;
+        this.lyrIds = [];
+        this.lyrLookup = {};
+        this.lyrArray = [];
+        this.mediaControl = undefined;
+        this.animationLayers = [];
+        this.coordinationLayerId = 'coorindation-layer';
+
+        /**
+         *
+         * @type {Array<LayerBase>}
+         */
+        this.operationsLayersStatic = [];
+
+        /**
+         *
+         * @type {Array<LayerBase>}
+         */
+        this.operationsLayersAnimated = [];
+
+        this.currentOperationsLayers = this.operationsLayersStatic;
+    }
+
+    init() {
+        for (let l of this.operationsLayersStatic) {
+            this.map.removeLayer(l['olLayer']);
+        }
+
+        for (let l of this.operationsLayersAnimated) {
+            this.map.removeLayer(l['olLayer']);
+        }
+
+        this.currentTabId = homeTabId;
+    }
+
+    /**
+     * Add the layer to the config object
+     * @param {LayerBase|*} lyr - base layer
+     */
+    _addLayer(lyr) {
+        this.lyrIds.push(lyr.id);
+        this.lyrLookup[lyr.id] = lyr;
+        this.lyrArray.push(lyr);
+    }
+
+    get coordinationLayer() {
+        return this._coordinationLayer;
+    }
+
+    set coordinationLayer(coordLayer) {
+        this._coordinationLayer = coordLayer;
+        this._addLayer(coordLayer);
+    }
+
+    addOperationsLayerStatic(lyr) {
+        this.operationsLayersStatic.push(lyr);
+        this._addLayer(lyr);
+    }
+
+    addOperationsLayerAnimate(lyr) {
+        this.operationsLayersAnimated.push(lyr);
+        this._addLayer(lyr);
+    }
+
+    set currentTabId(tabId) {
+        mapPopup.closePopup();
+        this.mediaControl.stopPlaying();
+        if (tabId == operationsTabId) {
+            for (let l of this.currentOperationsLayers) {
+                this.map.addLayer(l['olLayer']);
+            }
+        } else {
+            for (let l of this.currentOperationsLayers) {
+                this.map.removeLayer(l['olLayer']);
+            }
+        }
+
+        switch (tabId) {
+            case homeTabId:
+                this.coordinationLayer.visible = false;
+                break;
+            case coordinationTabId:
+                this.coordinationLayer.visible = true;
+                break;
+            case operationsTabId:
+                this.coordinationLayer.visible = false;
+                this.$opsAccordion.accordion("refresh");
+                break;
+            default:
+                throw tabId + ' tab id not found'
+        }
+    }
+
+    set currentOperationsPanelId(panelId) {
+        mapPopup.closePopup();
+        this.mediaControl.stopPlaying();
+
+        for (let l of this.currentOperationsLayers) {
+            this.map.removeLayer(l['olLayer']);
+        }
+
+        switch (panelId) {
+            case operationsStaticPanelId:
+                this.currentOperationsLayers = this.operationsLayersStatic;
+                break;
+
+            case operationsAnimatedPanelId:
+                this.currentOperationsLayers = this.operationsLayersAnimated;
+                break;
+            default:
+                throw panelId + ' panel not found';
+        }
+        for (let l of this.currentOperationsLayers) {
+            this.map.addLayer(l['olLayer']);
+        }
+
+    }
+
+    ///**
+    // * trick to trigger the map move event
+    // */
+    //forceRefresh() {
+    //    if (this.map) {
+    //        this.map.getView().setZoom(this.map.getView().getZoom());
+    //    }
+    //}
+}
+
+export default new AppConfig();
diff --git a/projects/glrtoc/layerPopups.js b/projects/glrtoc/layerPopups.js
new file mode 100644
index 0000000..4362c1c
--- /dev/null
+++ b/projects/glrtoc/layerPopups.js
@@ -0,0 +1,37 @@
+/**
+ * Created by gavorhes on 12/7/2015.
+ */
+
+
+/**
+ * coordination layer popup config
+ * @param {object} props
+ * @returns {string}
+ */
+export function coordination(props) {
+    "use strict";
+    return `<iframe src="${window.location.href.replace('#', '') +
+    '/../toc?toc=' + props['toc']}" width="368" height="292"></iframe>`;
+}
+
+/**
+ * wrs layer popup config
+ * @param {object} props
+ * @returns {string}
+ */
+export function wrs(props) {
+    "use strict";
+    return `<p style="text-align: center">${props['WMS_INFO'].replace(/\n/g, '<br>')}</p>`
+}
+
+export function specialEventWorkZone(props) {
+    "use strict";
+    let startDate = new Date(props['EventStartDate']);
+    let endDate = new Date(props['EventEndDate']);
+    let theContent = '<p style="text-align:center">';
+    theContent += props['EventDescription'] + '<br>';
+    theContent += `${props['HwyName']} ${props['Location']}` + '<br>';
+    theContent += `${startDate.toLocaleDateString()}&nbsp;-&nbsp;${endDate.toLocaleDateString()}`;
+    theContent += '</p>';
+    return theContent;
+}
diff --git a/projects/glrtoc/layerStyles.js b/projects/glrtoc/layerStyles.js
new file mode 100644
index 0000000..7b2eaed
--- /dev/null
+++ b/projects/glrtoc/layerStyles.js
@@ -0,0 +1,39 @@
+/**
+ * Created by gavorhes on 12/3/2015.
+ */
+import ol from '../../src/custom-ol';
+
+export const workZoneAndEventStyle = new ol.style.Style({
+    stroke: new ol.style.Stroke({
+        color: 'red',
+        width: 7
+    }),
+    image: new ol.style.Circle({
+        radius: 8,
+        fill: new ol.style.Fill({
+            color: 'magenta'
+        }),
+        stroke: new ol.style.Stroke({
+            color: 'magenta'
+        })
+    })
+});
+
+export function wrsStyle(feature, resolution) {
+    let colorString = feature.getProperties()['COLOR'];
+
+    let c;
+    if (typeof colorString === 'string') {
+        c = colorString.split(' ');
+    } else {
+        c = [125, 125, 125];
+    }
+
+    return [new ol.style.Style({
+        stroke: new ol.style.Stroke({
+            color: `rgba(${c[0]},${c[1]},${c[2]},0)`,
+            width: 5
+        })
+    })];
+}
+
diff --git a/projects/glrtoc/legendTest.js b/projects/glrtoc/legendTest.js
new file mode 100644
index 0000000..71ecba4
--- /dev/null
+++ b/projects/glrtoc/legendTest.js
@@ -0,0 +1,129 @@
+/**
+ * Created by gavorhes on 1/4/2016.
+ */
+import quickMap from '../../src/olHelpers/quickMap';
+import LayerItsInventory from '../../src/layers/LayerItsInventory';
+import LayerBaseVectorEsri from '../../src/layers/LayerBaseVectorEsri';
+import LayerEsriMapServer from '../../src/layers/LayerEsriMapServer';
+import LayerBaseXyzTile from '../../src/layers/LayerBaseXyzTile';
+import LayerLegend from '../../src/collections/LayerLegend';
+import ItsLayerCollection from '../../src/collections/ItsLayerCollection';
+import mapPopup from '../../src/olHelpers/mapPopup';
+
+
+(function () {
+    "use strict";
+    let map = quickMap({center: {x: -9907589, y: 5232317}, zoom: 12, minZoom: 3, maxZoom: 19});
+
+    let itsLayerCollection = new ItsLayerCollection(map);
+
+    let oakRidgeCams = new LayerEsriMapServer(
+        `http://itsdpro.ornl.gov/arcgis/rest/services/ITSPublic/cameras33/MapServer`,
+        {
+            id: 'cameras33',
+            name: 'Oak Cameras',
+            visible: true,
+            minZoom: 7,
+            zIndex: 20,
+            addPopup: true
+        }
+    );
+
+    map.addLayer(oakRidgeCams.olLayer);
+
+    glob.cat = (bird) => bird * 2;
+
+
+
+
+    glob.map = map;
+
+    let itsLayer = new LayerItsInventory({itsType: 'CCTV', itsIcon: 'cctv.png', name: 'Camera', visible: false});
+
+    map.addLayer(itsLayer.olLayer);
+
+    let legend = new LayerLegend(
+        [
+            {groupName: 'its layers', collapse: true, items: itsLayerCollection.layers},
+            itsLayer, oakRidgeCams
+        ],
+        'legend-container', {});
+
+
+
+    //let legend = new LayerLegend(
+    //    [
+    //        {groupName: 'its layers', expand: true, items: itsLayerCollection.layers},
+    //        itsLayer, testEsri, msg, tower, xyzTile,
+    //        {groupName: 'new group', expand: true, items: [rwis, coord, seg, esriMap]}
+    //    ],
+    //    'legend-container', {});
+
+    //let legend = new LayerLegend(
+    //    [
+    //        itsLayerCollection.layers[1]
+    //    ],
+    //    'legend-container', {});
+
+    map.getView().setZoom(13);
+
+    return;
+
+    let testEsri = new LayerBaseVectorEsri('http://transportal.cee.wisc.edu/applications/arcgis2/rest/services/GLRTOC/LegendTest/MapServer/3',
+        {name: 'test esri', useEsriStyle: true, visible: false}
+    );
+    map.addLayer(testEsri.olLayer);
+
+    let msg = new LayerBaseVectorEsri('http://transportal.cee.wisc.edu/applications/arcgis2/rest/services/GLRTOC/LegendTest/MapServer/8',
+        {name: 'message', useEsriStyle: true, visible: false}
+    );
+    map.addLayer(msg.olLayer);
+
+    let tower = new LayerBaseVectorEsri('http://transportal.cee.wisc.edu/applications/arcgis2/rest/services/GLRTOC/LegendTest/MapServer/0',
+        {name: 'tower', useEsriStyle: true, visible: false}
+    );
+    map.addLayer(tower.olLayer);
+
+
+    let rwis = new LayerBaseVectorEsri('http://transportal.cee.wisc.edu/applications/arcgis2/rest/services/GLRTOC/LegendTest/MapServer/1',
+        {name: 'rwis', useEsriStyle: true, visible: false}
+    );
+    map.addLayer(rwis.olLayer);
+
+    let coord = new LayerBaseVectorEsri('http://transportal.cee.wisc.edu/applications/arcgis2/rest/services/GLRTOC/LegendTest/MapServer/10',
+        {name: 'coord', useEsriStyle: true, minZoom: 6, visible: false, collapseLegend: true}
+    );
+    map.addLayer(coord.olLayer);
+
+    //let coord2 = new LayerBaseVectorEsri('http://transportal.cee.wisc.edu/applications/arcgis2/rest/services/GLRTOC/LegendTest/MapServer/11',
+    //    {name: 'coord2', useEsriStyle: true}
+    //);
+    //map.addLayer(coord.olLayer);
+
+    let seg = new LayerBaseVectorEsri('http://transportal.cee.wisc.edu/applications/arcgis2/rest/services/GLRTOC/LegendTest/MapServer/9',
+        {name: 'seg', useEsriStyle: true, visible: false}
+    );
+    map.addLayer(seg.olLayer);
+
+    let esriMap = new LayerEsriMapServer('http://transportal.cee.wisc.edu/applications/arcgis2/rest/services/GLRTOC/LegendTest/MapServer', {
+        name: 'esri map',
+        legendCollapse: true,
+        legendCheckbox: false,
+        visible: false
+    });
+
+    map.addLayer(esriMap.olLayer);
+
+    let xyzTile = new LayerBaseXyzTile('http://transportal.cee.wisc.edu/applications/arcgis2/rest/services/NPMRDS/npmrds_tile/MapServer/tile/{z}/{y}/{x}',
+        {minZoom: 4, maxZoom: 11, name: "NPMRDS", useEsriStyle: true, collapseLegend: true});
+
+    //let esriMapServer = new LayerEsriMapServer('http://transportal.cee.wisc.edu/applications/arcgis2/rest/services/NPMRDS/npmrds_dynamic/MapServer',
+    //    {minZoom: 12, maxZoom: 18});
+
+    map.addLayer(xyzTile.olLayer);
+    //map.addLayer(esriMapServer.olLayer);
+
+    glob.itsCollection = itsLayerCollection;
+
+
+})();
diff --git a/projects/glrtoc/main.js b/projects/glrtoc/main.js
new file mode 100644
index 0000000..2613e08
--- /dev/null
+++ b/projects/glrtoc/main.js
@@ -0,0 +1,319 @@
+/**
+ * Created by gavorhes on 12/2/2015.
+ */
+
+// <editor-fold desc="imports">
+
+import quickMap from '../../src/olHelpers/quickMap';
+import LayerBase from '../../src/layers/LayerBase';
+import LayerBaseVectorGeoJson from '../../src/layers/LayerBaseVectorGeoJson';
+import LayerBaseVectorEsri from  '../../src/layers/LayerBaseVectorEsri';
+import LayerRealEarthTile from  '../../src/layers/LayerRealEarthTile';
+import LayerRealEarthVector from '../../src/layers/LayerRealEarthVector';
+import LayerEsriMapServer from '../../src/layers/LayerEsriMapServer';
+import * as layerStyles from './layerStyles';
+import * as layerPopups from './layerPopups';
+import mapPopup from '../../src/olHelpers/mapPopup';
+import mapMove from '../../src/olHelpers/mapMove';
+import * as dteConvert from '../../src/util/dateConvert';
+import * as colors from '../../src/util/colors';
+import appConfig from './appConfig';
+import * as uiSetup from './mainUi';
+import LayerLegend from '../../src/collections/LayerLegend';
+// </editor-fold>
+
+(function () {
+    "use strict";
+    uiSetup.uiInit();
+
+    let legendItemsStatic = [];
+    let legendItemsAnimated = [];
+    let oakRidgeGroup = {groupName: 'Oak Ridge ITS', collapse: false, addCheck: true, items: []};
+    let realEarthGroup = {groupName: 'Real Earth', collapse: false, items: []};
+    let workZoneEventGroup = {groupName: 'Work Zones / Events', collapse: false, items: []};
+    legendItemsStatic.push(oakRidgeGroup);
+    legendItemsStatic.push(realEarthGroup);
+    legendItemsStatic.push(workZoneEventGroup);
+
+    function animationLoadCallback() {
+        appConfig.animationLayers.push(this);
+    }
+
+    let map = quickMap({center: {x: -85.413, y: 43.29320}, zoom: 6, minZoom: 3, maxZoom: 19});
+    appConfig.map = map;
+
+    // <editor-fold desc="Coordination Layer">
+    let coordinationLayer = new LayerBaseVectorEsri(
+        'http://transportal.cee.wisc.edu/applications/arcgis2/rest/services/GLRTOC/GlrtocCoordination/MapServer/0',
+        {
+            id: appConfig.coordinationLayerId,
+            visible: true,
+            autoLoad: true,
+            name: 'Coordination',
+            useEsriStyle: true
+        }
+    );
+
+    map.addLayer(coordinationLayer.olLayer);
+    mapPopup.addVectorPopup(coordinationLayer, layerPopups.coordination);
+    appConfig.coordinationLayer = coordinationLayer;
+
+
+    //appConfig.addLayer(coordinationLayer);
+    // </editor-fold>
+
+    // <editor-fold desc="Oak Ridge Layers ">
+    let oakRidgeLayers = [
+        ['Cameras', 'cameras33'],
+        ['HAR', 'HAR33'],
+        ['DMS', 'MessageSigns33'],
+        //['State Summary', 'statesummary'],
+        ['Traffic Control', 'TrafficControl33'],
+        ['Traffic Detection', 'TrafficDetection33'],
+        ['Weather', 'Weather33']
+    ];
+
+
+    for (let i = 0; i < oakRidgeLayers.length; i++) {
+        let oakRidgeLayer = new LayerEsriMapServer(
+            `http://itsdpro.ornl.gov/arcgis/rest/services/ITSPublic/${oakRidgeLayers[i][1]}/MapServer`,
+            {
+                id: oakRidgeLayers[i][1],
+                name: oakRidgeLayers[i][0],
+                visible: false,
+                minZoom: 7,
+                zIndex: 20,
+                addPopup: true
+            }
+        );
+        oakRidgeGroup.items.push(oakRidgeLayer);
+        map.addLayer(oakRidgeLayer.olLayer);
+        appConfig.addOperationsLayerStatic(oakRidgeLayer);
+    }
+    // </editor-fold>
+
+    // <editor-fold desc="WRS Segments">
+
+    let wrsConfigVector = {
+        products: 'ROADS',
+        style: layerStyles.wrsStyle,
+        animate: false,
+        id: 'wrs-segments-vector-static',
+        name: 'Winter Roads',
+        minZoom: 5,
+        maxZoom: 13,
+        zIndex: 3
+    };
+
+    let wrsSegmentLayerVectorStatic = new LayerRealEarthVector(wrsConfigVector);
+    map.addLayer(wrsSegmentLayerVectorStatic.olLayer);
+    mapPopup.addVectorPopup(wrsSegmentLayerVectorStatic, layerPopups.wrs);
+    appConfig.addOperationsLayerStatic(wrsSegmentLayerVectorStatic);
+    //realEarthGroup.items.push(wrsSegmentLayerVectorStatic);
+
+    wrsConfigVector.animate = true;
+    wrsConfigVector.id = 'wrs-segments-vector-animate';
+    wrsConfigVector.loadCallback = animationLoadCallback;
+
+    let wrsSegmentLayerVectorAnimate = new LayerRealEarthVector(wrsConfigVector);
+    map.addLayer(wrsSegmentLayerVectorAnimate.olLayer);
+    mapPopup.addVectorPopup(wrsSegmentLayerVectorAnimate, layerPopups.wrs);
+    appConfig.addOperationsLayerAnimate(wrsSegmentLayerVectorAnimate);
+
+    let wrsConfigTile = {
+        products: 'ROADS',
+        id: 'roads-tile-static',
+        opacity: 0.6,
+        animate: false,
+        name: 'Winter Roads',
+        zIndex: 10
+    };
+
+    let wrsSegmentLayerTileStatic = new LayerRealEarthTile(wrsConfigTile);
+    map.addLayer(wrsSegmentLayerTileStatic.olLayer);
+    appConfig.addOperationsLayerStatic(wrsSegmentLayerTileStatic);
+    realEarthGroup.items.push(wrsSegmentLayerTileStatic);
+
+    wrsConfigTile.animate = true;
+    wrsConfigTile.id = 'roads-tile-animate';
+    wrsConfigTile.loadCallback = animationLoadCallback;
+
+    let wrsSegmentLayerTileAnimate = new LayerRealEarthTile(wrsConfigTile);
+    map.addLayer(wrsSegmentLayerTileAnimate.olLayer);
+    appConfig.addOperationsLayerAnimate(wrsSegmentLayerTileAnimate);
+    legendItemsAnimated.push(wrsSegmentLayerTileAnimate);
+
+    // </editor-fold>
+
+    // <editor-fold desc="24 hour snow">
+
+    let snow24Config = {
+        products: 'SNOWDEPTH24',
+        id: 'snowdepth24-static',
+        opacity: 0.5,
+        animate: false,
+        name: '24HR Snow',
+        maxZoom: 9
+    };
+
+
+    let snowDepthStatic = new LayerRealEarthTile(snow24Config);
+    map.addLayer(snowDepthStatic.olLayer);
+    appConfig.addOperationsLayerStatic(snowDepthStatic);
+    realEarthGroup.items.push(snowDepthStatic);
+
+    //snow24Config.animate = true;
+    //snow24Config.id = 'snowdepth24-animate';
+    //snow24Config.loadCallback = animationLoadCallback;
+    //
+    //let snowDepthAnimate = new LayerRealEarthTile(snow24Config);
+    //map.addLayer(snowDepthAnimate.olLayer);
+    //appConfig.addOperationsLayerAnimate(snowDepthAnimate);
+    //legendItemsAnimated.push(snowDepthAnimate);
+
+    // </editor-fold>
+
+    // <editor-fold desc="nexrhres and precipitation layers">
+    let nexrhresConfig = {
+        products: 'nexrhres',
+        id: 'nexrhres-static',
+        opacity: 0.6,
+        animate: false,
+        name: 'Hybrid Reflectivity',
+        maxZoom: 10
+    };
+
+    let nexrhresStatic = new LayerRealEarthTile(nexrhresConfig);
+    map.addLayer(nexrhresStatic.olLayer);
+    appConfig.addOperationsLayerStatic(nexrhresStatic);
+    realEarthGroup.items.push(nexrhresStatic);
+
+    nexrhresConfig.animate = true;
+    nexrhresConfig.id = 'nexrhres-animate';
+    nexrhresConfig.loadCallback = animationLoadCallback;
+
+    let nexrhresAnimate = new LayerRealEarthTile(nexrhresConfig);
+    map.addLayer(nexrhresAnimate.olLayer);
+    appConfig.addOperationsLayerAnimate(nexrhresAnimate);
+    legendItemsAnimated.push(nexrhresAnimate);
+
+    let precipRateConfig = {
+        products: 'nexrcomp',
+        id: 'precip-rate-static',
+        opacity: 0.6,
+        animate: false,
+        name: 'Precipitation Rate',
+        maxZoom: 10
+    };
+
+    let precipRateStatic = new LayerRealEarthTile(precipRateConfig);
+    map.addLayer(precipRateStatic.olLayer);
+    appConfig.addOperationsLayerStatic(precipRateStatic);
+    realEarthGroup.items.push(precipRateStatic);
+
+    precipRateConfig.animate = true;
+    precipRateConfig.id = 'precip-rate-animate';
+    precipRateConfig.loadCallback = animationLoadCallback;
+
+    let precipRateAnimate = new LayerRealEarthTile(precipRateConfig);
+    map.addLayer(precipRateAnimate.olLayer);
+    appConfig.addOperationsLayerAnimate(precipRateAnimate);
+    legendItemsAnimated.push(precipRateAnimate);
+
+
+    // </editor-fold>
+
+    // <editor-fold desc="Work Zones and special events">
+    let d = new Date();
+    d.setSeconds(0);
+    let endDate = dteConvert.dateToYyyyMmDdHhMmSs(d);
+    d.setYear(d.getYear() + 1901);
+    var startDate = dteConvert.dateToYyyyMmDdHhMmSs(d);
+
+    let workZoneSegLayer = new LayerBaseVectorEsri(
+        'http://transportal.cee.wisc.edu/applications/arcgis2/rest/services/GLRTOC/GLRTOC_WZ_SE/MapServer/1',
+        {
+            where: `EventStartDate < '${startDate}' AND EventEndDate > '${endDate}' AND Impact IN('High', 'XXX')`,
+            name: "Work Zone Segments",
+            style: layerStyles.workZoneAndEventStyle,
+            id: 'work-zone-segments',
+            minZoom: 5,
+            maxZoom: 13
+        }
+    );
+
+    map.addLayer(workZoneSegLayer.olLayer);
+    appConfig.addOperationsLayerStatic(workZoneSegLayer);
+    mapPopup.addVectorPopup(workZoneSegLayer, layerPopups.specialEventWorkZone);
+    workZoneEventGroup.items.push(workZoneSegLayer);
+
+    let specialEventsLayer = new LayerBaseVectorEsri(
+        'http://transportal.cee.wisc.edu/applications/arcgis2/rest/services/GLRTOC/GLRTOC_WZ_SE/MapServer/2',
+        {
+            where: `EventStartDate < '${startDate}' AND EventEndDate > '${endDate}' AND Impact IN('High', 'XXX')`,
+            //where: '1=1',
+            name: "Special Events",
+            style: layerStyles.workZoneAndEventStyle,
+            id: 'special-event-points',
+            minZoom: 5,
+            maxZoom: 13
+        }
+    );
+
+    map.addLayer(specialEventsLayer.olLayer);
+    appConfig.addOperationsLayerStatic(specialEventsLayer);
+    mapPopup.addVectorPopup(specialEventsLayer, layerPopups.specialEventWorkZone);
+    workZoneEventGroup.items.push(specialEventsLayer);
+
+    // </editor-fold>
+
+    let legendStatic = new LayerLegend(legendItemsStatic, 'legend-container-static', {});
+    let legendAnimate = new LayerLegend(legendItemsAnimated, 'legend-container-animate', {});
+
+    appConfig.init();
+
+    uiSetup.uiAfterMap();
+})();
+
+
+// <editor-fold desc="Promise example">
+
+//var promise = new Promise(function (resolve, reject) {
+//    console.log('doing stuff');
+//
+//    $.get('http://realearth.ssec.wisc.edu:80/api/products', {products: 'ROADS'}, function (d) {
+//        //if (d.length == 0) {
+//        //    console.log(`${this._products} layer not available or does not have times`);
+//        //    return;
+//        //}
+//        d = d[0];
+//        console.log(d);
+//        resolve("Stuff worked!");
+//
+//
+//        //for (let i = 0; i < d['times'].length; i++) {
+//        //    _this._loadDates.call(_this, d['times'][i]);
+//        //}
+//        //console.log(_this._localDates);
+//
+//        //_this._loadAtTimeIndex.call(_this, _this._localDates.length - 1)
+//
+//    }, 'json').fail(function () {
+//        reject(Error("It broke"));
+//    });
+//
+//
+//});
+//
+//
+//promise.then(
+//    function (result) {
+//        console.log(result); // "Stuff worked!"
+//    },
+//    function (err) {
+//        console.log(err); // Error: "It broke"
+//    }
+//);
+// </editor-fold>
+
+
diff --git a/projects/glrtoc/mainUi.js b/projects/glrtoc/mainUi.js
new file mode 100644
index 0000000..9e4f17c
--- /dev/null
+++ b/projects/glrtoc/mainUi.js
@@ -0,0 +1,105 @@
+/**
+ * Created by gavorhes on 12/7/2015.
+ */
+
+import {} from '../../src/jquery-plugin/mediaControl';
+import {} from '../../src/jquery-plugin/rangeChange';
+import appConfig from './appConfig';
+import $ from '../../src/jquery';
+require('jquery-ui/tabs');
+require('jquery-ui/accordion');
+
+
+/**
+ * Set up the UI
+ */
+export function uiInit() {
+    "use strict";
+
+    //$("#operations").accordion('refresh');
+
+
+    let $sidebar = $('#sidebar');
+    let $hideSideBar = $('#hide-sidebar');
+    let $showSideBar = $('#show-sidebar');
+
+    let $tabs = $("#tabs");
+
+    let sidebarWidth = $sidebar.width();
+
+    //apply tab layout
+    $tabs.tabs({
+        heightStyle: "fill",
+        activate: function (event, ui) {
+            appConfig.currentTabId = ui.newPanel[0].id;
+        }
+    });
+
+    //apply accordion
+    appConfig.$opsAccordion.accordion({
+        heightStyle: "fill",
+        activate: function (event, ui) {
+            appConfig.currentOperationsPanelId = ui.newPanel[0].id;
+        }
+    });
+
+    $(window).resize(function () {
+        $tabs.tabs('refresh');
+        setTimeout(function () {
+            appConfig.$opsAccordion.accordion("refresh");
+        }, 50);
+    });
+
+    $hideSideBar.click(function () {
+        var mapCenter = appConfig.map.getView().getCenter();
+
+        appConfig.map.beforeRender(function () {
+            $sidebar.animate({'margin-left': -1 * sidebarWidth}, 200,
+                function () {
+                    $hideSideBar.hide();
+                    $showSideBar.show();
+                    appConfig.map.updateSize();
+                }
+            );
+        });
+        appConfig.map.getView().setCenter(mapCenter);
+    });
+
+    $showSideBar.click(function () {
+        var mapCenter = appConfig.map.getView().getCenter();
+
+        appConfig.map.beforeRender(function () {
+            $sidebar.animate({'margin-left': 0}, 200,
+                function () {
+                    $showSideBar.hide();
+                    $hideSideBar.show();
+                    appConfig.map.updateSize();
+                }
+            );
+        });
+        appConfig.map.getView().setCenter(mapCenter);
+    });
+
+    let d = new Date();
+    let endTime = d.getTime();
+    d.setHours(d.getHours() - 4);
+    let startTime = d.getTime();
+    let rangeStep = Math.round((endTime - startTime) / 8);
+
+    appConfig.mediaControl = $('#animation-control').mediaControl(startTime, endTime, endTime, rangeStep,
+        function (t) {
+            for (let l of appConfig.animationLayers) {
+                l.setLayerTime(t);
+            }
+        },
+        750, true);
+}
+
+
+//http://realearth.ssec.wisc.edu/api/image?products=nexrhres_20160108_180000&x=5&y=5&z=4
+
+export function uiAfterMap() {
+    "use strict";
+
+
+}
\ No newline at end of file
diff --git a/projects/itsMap.js b/projects/itsMap.js
new file mode 100644
index 0000000..d31b6c5
--- /dev/null
+++ b/projects/itsMap.js
@@ -0,0 +1,27 @@
+/**
+ * Created by gavorhes on 12/18/2015.
+ */
+
+import quickMap from '../src/olHelpers/quickMap';
+import mapMove from '../src/olHelpers/mapMove';
+import mapPopup from '../src/olHelpers/mapPopup';
+import ItsLayerCollection from '../src/collections/ItsLayerCollection';
+import LayerLegend from '../src/collections/LayerLegend';
+
+let map = quickMap();
+mapMove.init(map);
+mapPopup.init(map);
+
+let itsLayerCollection = new ItsLayerCollection(map);
+
+let layerArray = [
+    {
+        groupName: 'ITS Inventory Layers',
+        collapse: false,
+        addCheck: true,
+        items: itsLayerCollection.layers
+    }
+];
+
+
+let legend = new LayerLegend(layerArray, 'legend-container', {});
diff --git a/projects/npmrds/delay/delay-config.js b/projects/npmrds/delay/delay-config.js
new file mode 100644
index 0000000..5be9a94
--- /dev/null
+++ b/projects/npmrds/delay/delay-config.js
@@ -0,0 +1,3 @@
+/**
+ * Created by gavorhes on 2/9/2016.
+ */
diff --git a/projects/npmrds/delay/delay-main.js b/projects/npmrds/delay/delay-main.js
new file mode 100644
index 0000000..b0783f4
--- /dev/null
+++ b/projects/npmrds/delay/delay-main.js
@@ -0,0 +1,32 @@
+/**
+ * Created by gavorhes on 2/9/2016.
+ */
+
+import quickMap from '../../../src/olHelpers/quickMap';
+const angular = require('angular2');
+
+let map = quickMap();
+
+let app = angular.module('myApp', []);
+
+
+app.controller('myCtrl', function($scope) {
+    $scope.firstName = "John";
+    $scope.lastName = "Doe";
+
+    $scope.myFunction = function(){
+        "use strict";
+        console.log((new Date()).getTime());
+    };
+});
+
+
+app.controller('myCtrl2', function($scope) {
+    //$scope.firstName = "John";
+    //$scope.lastName = "Doe";
+    //
+    //$scope.myFunction = function(){
+    //    "use strict";
+    //    console.log((new Date()).getTime());
+    //}
+});
diff --git a/projects/npmrds/heatmap/appConfig.js b/projects/npmrds/heatmap/appConfig.js
new file mode 100644
index 0000000..32d4fcb
--- /dev/null
+++ b/projects/npmrds/heatmap/appConfig.js
@@ -0,0 +1,50 @@
+/**
+ * Created by gavorhes on 12/23/2015.
+ */
+
+class NpmrdsHeatmapConfig {
+    constructor() {
+
+        this.map = undefined;
+
+        /**
+         *
+         * @type {LayerBaseVectorGeoJson}
+         */
+        this.featureOverlay = undefined;
+
+        /**
+         *
+         * @type {LayerBaseVectorGeoJson}
+         */
+        this.lineLayer = undefined;
+
+
+        /**
+         *
+         * @type {LayerBaseVectorGeoJson}
+         */
+        this.pointLayer = undefined;
+
+
+        /**
+         *
+         * @type {LayerBaseVectorGeoJson}
+         */
+        this.trackerLayer = undefined;
+    }
+
+
+}
+
+///**
+// *
+// * @type {NpmrdsHeatmapConfig}
+// */
+//const npmrdsHeatmapConfig = new NpmrdsHeatmapConfig();
+
+/**
+ * @type {NpmrdsHeatmapConfig}
+ */
+export default new NpmrdsHeatmapConfig();
+
diff --git a/projects/npmrds/heatmap/layerStyles.js b/projects/npmrds/heatmap/layerStyles.js
new file mode 100644
index 0000000..e16bd69
--- /dev/null
+++ b/projects/npmrds/heatmap/layerStyles.js
@@ -0,0 +1,43 @@
+/**
+ * Created by gavorhes on 12/22/2015.
+ */
+import ol from '../../../src/custom-ol';
+
+
+export const overlayStyle = new ol.style.Style({
+    fill: new ol.style.Fill({
+        color: 'rgba(255, 0, 237, 0.1)'
+    }),
+    stroke: new ol.style.Stroke({
+        color: 'rgb(255, 0, 237)',
+        width: 2
+    })
+});
+
+
+export const lineIndicator = new ol.style.Style({
+    stroke: new ol.style.Stroke({
+        color: 'red',
+        width: 7
+    })
+});
+
+
+export const pointIndices = new ol.style.Style({
+    image: new ol.style.Circle({
+        radius: 4,
+        fill: new ol.style.Fill({
+            color: 'blue'
+        })
+    })
+});
+
+
+export const trackerPoint = new ol.style.Style({
+    image: new ol.style.Circle({
+        radius: 8,
+        fill: new ol.style.Fill({
+            color: 'blue'
+        })
+    })
+});
diff --git a/projects/npmrds/heatmap/main-ui.js b/projects/npmrds/heatmap/main-ui.js
new file mode 100644
index 0000000..7f6a44a
--- /dev/null
+++ b/projects/npmrds/heatmap/main-ui.js
@@ -0,0 +1,368 @@
+/**
+ * Created by gavorhes on 12/22/2015.
+ */
+
+import npmrdsHeatmapConfig from './appConfig';
+import {} from '../../../src/jquery-plugin/dayRange';
+import {} from '../../../src/jquery-plugin/rangeChange';
+import * as colors from '../../../src/util/colors';
+import SortedFeatures from '../../../src/olHelpers/SortedFeatures';
+import $ from '../../../src/jquery';
+import ol from '../../../src/custom-ol';
+
+
+let $btnCancelArea = $('#btn-cancel-area');
+let $btnSelectArea = $('#btn-select-area');
+let $divSelectArea = $('#div-select-area');
+let $ulHwyDirs = $('#ul-hwy-dirs');
+let $btnSelectHwy = $('#btn-select-hwy');
+//let $btnSelectHwyBack = $('#btn-select-hwy-back');
+let $divHighwaySelection = $('#div-highway-selection');
+let $divRight = $('#right');
+let $divLeft = $('#left');
+let $ckmapTrack = $('#chk-map-track');
+let $canvasToolTipSpan = $('#canvas-tooltip-span');
+let $canv = $('#heatmap-canvas');
+/**
+ *
+ * @type {*|jQuery|HTMLElement}
+ */
+let $canvContainer = $('#canvas-container');
+let $dateUl = $('#date-ul');
+let $heatMapVerticalBar = $('#heat-map-vertical-bar');
+let $selectedTmcs = $('#selected-tmcs');
+
+let sldrHeatMapId = 'heat-map-slider';
+let $sldrHeatMap = $('#' + sldrHeatMapId);
+
+
+let heatMapReady = false;
+
+/**
+ * @type {SortedFeatures}
+ */
+let sortedFeatures;
+
+/**
+ * @type {DayRange}
+ */
+let dayRange;
+
+function _leadingPad(num) {
+    let out = num.toFixed();
+    if (out.length == 1) {
+        out = '0' + out;
+    }
+    
+    return out;
+}
+
+function _formatDate(d) {
+    return `${_leadingPad(d.getMonth() + 1)}/${_leadingPad(d.getDate())}/${d.getYear() + 1900} ` +
+        `${_leadingPad(d.getHours())}:${_leadingPad(d.getMinutes())}:${_leadingPad(d.getSeconds())}`;
+}
+
+
+
+function clearCanvas() {
+    $dateUl.html('');
+    $canv.off('mousemove');
+    let canv = $canv[0];
+    let ctx = canv.getContext("2d");
+    ctx.clearRect(0, 0, canv.width, canv.height);
+
+    $canv.attr('width', $('#heat-map-slider').width());
+    $canv.attr('height', 100);
+    $heatMapVerticalBar.height(0);
+}
+
+
+export function startUi() {
+    "use strict";
+    dayRange = $('#div-date-range').dayRange(10);
+}
+
+export function drawSetup() {
+    "use strict";
+    let draw = new ol.interaction.Draw({
+        source: npmrdsHeatmapConfig.featureOverlay.source,
+        type: 'Polygon'
+    });
+
+    draw.on('drawend', function (evt) {
+        //$divSelectArea.hide();
+        $btnSelectArea.prop('disabled', false);
+        let geom = evt.feature.getGeometry();
+        let geomClone = geom.clone();
+
+        geomClone.transform('EPSG:3857', 'EPSG:4326');
+
+        setTimeout(function () {
+            npmrdsHeatmapConfig.map.removeInteraction(draw);
+        }, 100);
+        $btnCancelArea.hide();
+
+        $.get('npmrds/gethighways', {polygon: JSON.stringify(geomClone.getCoordinates())},
+            function (d) {
+                $ulHwyDirs.html(d);
+                $divHighwaySelection.show();
+
+                $('input[name=hwydirs]').change(function () {
+                    $btnSelectHwy.prop('disabled', false);
+                });
+            }, 'text');
+    });
+
+    $btnCancelArea.click(function () {
+        npmrdsHeatmapConfig.map.removeInteraction(draw);
+        $btnSelectArea.prop('disabled', false);
+        $btnCancelArea.hide();
+    });
+
+    $btnSelectArea.click(function () {
+        npmrdsHeatmapConfig.featureOverlay.source.clear();
+        npmrdsHeatmapConfig.map.addInteraction(draw);
+        $btnSelectArea.prop('disabled', true);
+        $btnCancelArea.show();
+        $ulHwyDirs.html('');
+        $divHighwaySelection.hide();
+        $btnSelectHwy.prop('disabled', true);
+    });
+}
+
+
+export function endUi() {
+    "use strict";
+
+    $divSelectArea.show();
+    $btnSelectArea.prop('disabled', false);
+    $btnCancelArea.hide();
+    $divHighwaySelection.hide();
+
+    $btnSelectHwy.click(function () {
+        let selectedHwy = $('input[name=hwydirs]:checked:first').val();
+        $('#sel-hwy').html(selectedHwy);
+        $.get('npmrds/getroute',
+            {
+                hwyDir: selectedHwy
+            }, function (d) {
+                npmrdsHeatmapConfig.lineLayer.clear();
+                npmrdsHeatmapConfig.lineLayer.addFeatures(d['line']);
+
+                npmrdsHeatmapConfig.pointLayer.clear();
+                npmrdsHeatmapConfig.pointLayer.addFeatures(d['points']);
+
+                sortedFeatures = new SortedFeatures(npmrdsHeatmapConfig.pointLayer.source.getFeatures(), 'distReal');
+
+                npmrdsHeatmapConfig.trackerLayer.clear();
+                npmrdsHeatmapConfig.trackerLayer.source.addFeature(sortedFeatures.sortedFeatures[0]);
+
+
+                $divLeft.hide();
+                $divRight.show();
+                npmrdsHeatmapConfig.map.updateSize();
+
+                let mapView = npmrdsHeatmapConfig.map.getView();
+
+                let panAnimation = ol.animation.pan({
+                    duration: 500,
+                    source: mapView.getCenter()
+                });
+                npmrdsHeatmapConfig.map.beforeRender(panAnimation);
+
+                mapView.fit(
+                    npmrdsHeatmapConfig.featureOverlay.source.getFeatures()[0].getGeometry().getExtent(),
+                    npmrdsHeatmapConfig.map.getSize()
+                );
+
+                mapView.setZoom(mapView.getZoom() - 1);
+
+                $('#vehicle-all').prop('checked', true);
+
+                npmrdsHeatmapConfig.featureOverlay.visible = false;
+                npmrdsHeatmapConfig.lineLayer.visible = true;
+                npmrdsHeatmapConfig.pointLayer.visible = true;
+                npmrdsHeatmapConfig.trackerLayer.visible = true;
+
+                $sldrHeatMap.prop('max', d['totalDistance']);
+                $sldrHeatMap.val(0);
+                $heatMapVerticalBar.css('left', '22px');
+                $selectedTmcs.val(JSON.stringify(d['tmcs']));
+
+                //console.log(d);
+            }, 'json').fail(function () {
+            alert("something went wrong, try another area selection");
+        });
+    });
+}
+
+
+export function heatMapUi() {
+    "use strict";
+
+    $('#heat-map-back').click(function () {
+        $divLeft.show();
+        $divRight.hide();
+        npmrdsHeatmapConfig.map.updateSize();
+        npmrdsHeatmapConfig.featureOverlay.visible = true;
+        npmrdsHeatmapConfig.lineLayer.visible = false;
+        npmrdsHeatmapConfig.pointLayer.visible = false;
+        npmrdsHeatmapConfig.trackerLayer.visible = false;
+
+        clearCanvas();
+    });
+
+    $canv.mouseenter(function () {
+        if (heatMapReady) {
+            $canvasToolTipSpan.css('display', 'block');
+        }
+    });
+
+    $canv.mouseleave(function () {
+        $canvasToolTipSpan.css('display', 'none');
+        $canvasToolTipSpan.html('');
+    });
+
+    //add the slider change interaction
+    $('#' + sldrHeatMapId).rangeChange(function(newVal, percent, evt){
+
+        let canvWidth = parseFloat($canv.width());
+
+        $heatMapVerticalBar.css('left', (canvWidth * percent + 22).toFixed() + 'px');
+
+        let selectedFeature = sortedFeatures.getFeature(newVal);
+
+        npmrdsHeatmapConfig.trackerLayer.clear();
+        npmrdsHeatmapConfig.trackerLayer.source.addFeature(selectedFeature);
+
+        if ($ckmapTrack.prop('checked')) {
+            let panAnimation = ol.animation.pan({
+                duration: 500,
+                source: npmrdsHeatmapConfig.map.getView().getCenter()
+            });
+            npmrdsHeatmapConfig.map.beforeRender(panAnimation);
+            npmrdsHeatmapConfig.map.getView().setCenter(selectedFeature.getGeometry().getCoordinates());
+        }
+    }, 25);
+
+
+    $('#heat-map-confirm').click(function () {
+        clearCanvas();
+        let canv = $canv[0];
+        let ctx = canv.getContext("2d");
+        let $progressDiv = $('#progress-indicator-div');
+        $dateUl.html('');
+
+        $progressDiv.show();
+
+        $.get('npmrds/getheatmap',
+            {
+                vehicleType: $('input[name=vehicle-type]:checked:first').val(),
+                startDate: _formatDate(dayRange.startDate),
+                endDate: _formatDate(dayRange.endDate),
+                tmcs: $selectedTmcs.val()
+            }, function (d) {
+                if (d['error']) {
+                    alert(d['error']);
+
+                    return;
+                }
+
+                let canvasWidth = parseInt($canv.attr('width'));
+                let colorGradient = colors.makeBlueGreenRedGradient(d['minSpeedVsFree'], d['maxSpeedVsFree'], true);
+                let colorGradientStdDev = colors.makeBlueGreenRedGradientZScore(d['medianSpeedVsFree'], d['stdDevSpeedVsFree'], true);
+
+                let recordLength = d['tmcResult'][0].values.length;
+                $canv.attr('height', (recordLength * 2).toFixed());
+                $heatMapVerticalBar.height(recordLength * 2);
+
+                let columnPosition = 0;
+                let tmcEndPositionArray = [];
+
+                for (let i = 0; i < d['tmcResult'].length; i++) {
+                    let tmc = d['tmcResult'][i];
+                    let rectangleWidth = canvasWidth * tmc['distPercent'] / 100;
+                    let valLength = tmc['values'].length;
+
+                    for (let j = 0; j < valLength; j++) {
+                        let speedVFreeSpeed = (tmc['values'][j] == null ? null : tmc['values'][j] / tmc['freeFlowSpeed']);
+                        let speedVFreeZScore = (speedVFreeSpeed - d['medianSpeedVsFree']) / d['stdDevSpeedVsFree'];
+                        //console.log(colorGradientStdDev(speedVFreeSpeed));
+                        //let z_score =
+                        //let z_score =
+
+                        ctx.fillStyle = colorGradientStdDev(speedVFreeSpeed);
+                        ctx.fillRect(columnPosition, j * 2, rectangleWidth, 2);
+                    }
+
+                    columnPosition += rectangleWidth;
+                    tmcEndPositionArray.push([tmc['tmc'], columnPosition]);
+                }
+
+                for (let i = 0; i < d['dateList'].length; i++) {
+                    $dateUl.append('<li><div>' + d['dateList'][i] + '</div></ul>');
+                }
+
+                // set the date li height
+                //$('#date-ul li').css('height', (2 + (parseInt($canv.attr('height')) / d['dateList'].length) - d['dateList'].length).toFixed());
+
+                let tmcResultDict = {};
+                for (let i = 0; i < d['tmcResult'].length; i++) {
+                    tmcResultDict[d['tmcResult'][i]['tmc']] = d['tmcResult'][i];
+                }
+
+                $canv.mousemove(function (evt) {
+
+                    // coordinates relative the canvas
+                    let offsetX = evt.offsetX;
+                    let offsetY = evt.offsetY;
+
+                    // coordinates relative the window
+                    let clientX = evt.clientX;
+                    let clientY = evt.clientY;
+
+                    // get the tmc
+                    let tmcId = null;
+                    for (let i = 0; i < tmcEndPositionArray.length; i++) {
+                        if (offsetX < tmcEndPositionArray[i][1]) {
+                            tmcId = tmcEndPositionArray[i][0];
+                            break;
+                        }
+                    }
+
+                    if (tmcId == null) {
+                        tmcId = tmcEndPositionArray[tmcEndPositionArray.length - 1][0];
+                    }
+
+                    // since each retangle in the heatmap is 2px high, get the index by divding by 2, round down
+                    let rowIndex = Math.floor(offsetY / 2);
+
+                    // make a date equal to the query start date
+                    let dte = new Date(d['dateList'][0]);
+                    // add minutes as 5X the row index
+                    dte.setMinutes(dte.getMinutes() + rowIndex * 5);
+
+                    let tmcObj = tmcResultDict[tmcId];
+
+                    //set the html
+                    $canvasToolTipSpan.html(`TMC: ${tmcId}<br>${dte.toLocaleDateString()} ${dte.toLocaleTimeString()}<br>` +
+                    `Speed: ${(tmcObj['values'][rowIndex] == null ? 'null' : tmcObj['values'][rowIndex].toFixed(1))}, Free: ${tmcObj['freeFlowSpeed'].toFixed(1)}`
+                    );
+
+                    //set the position
+                    let canvasToolipSpan = $canvasToolTipSpan[0];
+                    canvasToolipSpan.style.left = (clientX - 150).toFixed() + 'px';
+                    canvasToolipSpan.style.top = (clientY - 85).toFixed() + 'px';
+
+                });
+                //console.log(d);
+
+                $canvContainer.scrollTop(0);
+                heatMapReady = true;
+            }, 'json').fail(function(){
+                alert('Something went wrong with the request');
+            }).always(function(){
+                $progressDiv.hide();
+            });
+    });
+}
diff --git a/projects/npmrds/heatmap/main.js b/projects/npmrds/heatmap/main.js
new file mode 100644
index 0000000..2397377
--- /dev/null
+++ b/projects/npmrds/heatmap/main.js
@@ -0,0 +1,47 @@
+/**
+ * Created by gavorhes on 12/22/2015.
+ */
+
+import npmrdsHeatmapConfig from './appConfig';
+import quickMap from '../../../src/olHelpers/quickMap';
+import LayerBaseXyzTile from '../../../src/layers/LayerBaseXyzTile';
+import LayerEsriMapServer from '../../../src/layers/LayerEsriMapServer';
+import LayerBaseVectorGeoJson from '../../../src/layers/LayerBaseVectorGeoJson';
+import * as layerStyles from './layerStyles';
+import * as uiSetup from './main-ui';
+
+//glob.appConfig = npmrdsHeatmapConfig;
+
+(function(){
+    "use strict";
+    uiSetup.startUi();
+    let map = quickMap({center: {x: -10012438, y: 5548095}, zoom: 8, minZoom: 5});
+
+    npmrdsHeatmapConfig.map = map;
+
+    let xyzTile = new LayerBaseXyzTile('http://transportal.cee.wisc.edu/applications/arcgis2/rest/services/NPMRDS/npmrds_tile/MapServer/tile/{z}/{y}/{x}',
+        {minZoom: 4, maxZoom: 11});
+
+    let esriMapServer = new LayerEsriMapServer('http://transportal.cee.wisc.edu/applications/arcgis2/rest/services/NPMRDS/npmrds_dynamic/MapServer',
+        {minZoom: 12, maxZoom: 18});
+
+    map.addLayer(xyzTile.olLayer);
+    map.addLayer(esriMapServer.olLayer);
+
+    let transform = {dataProjection: 'EPSG:3857', featureProjection: 'EPSG:3857'};
+
+    npmrdsHeatmapConfig.featureOverlay = new LayerBaseVectorGeoJson('', {style: layerStyles.overlayStyle, transform: transform});
+    npmrdsHeatmapConfig.lineLayer = new LayerBaseVectorGeoJson('', {style: layerStyles.lineIndicator, transform: transform});
+    npmrdsHeatmapConfig.pointLayer = new LayerBaseVectorGeoJson('', {style: layerStyles.pointIndices, transform: transform});
+    npmrdsHeatmapConfig.trackerLayer = new LayerBaseVectorGeoJson('', {style: layerStyles.trackerPoint, transform: transform});
+
+    map.addLayer(npmrdsHeatmapConfig.featureOverlay.olLayer);
+    map.addLayer(npmrdsHeatmapConfig.lineLayer.olLayer);
+    //map.addLayer(npmrdsHeatmapConfig.pointLayer.olLayer);
+    map.addLayer(npmrdsHeatmapConfig.trackerLayer.olLayer);
+
+    uiSetup.drawSetup();
+    uiSetup.heatMapUi();
+    uiSetup.endUi();
+})();
+
diff --git a/projects/tsmo/TipConfig.js b/projects/tsmo/TipConfig.js
new file mode 100644
index 0000000..705a404
--- /dev/null
+++ b/projects/tsmo/TipConfig.js
@@ -0,0 +1,149 @@
+/**
+ * Created by gavorhes on 12/14/2015.
+ */
+
+/**
+ * Tip config object
+ */
+class TipConfig {
+    constructor() {
+
+        /**
+         *
+         * @type {Sliders}
+         */
+        this.sliders = undefined;
+
+        /**
+         *
+         * @type {Map}
+         */
+        this.map = null;
+
+        this.$loadingGif = null;
+        this.$regionSelector = undefined;
+
+        /**
+         *
+         * @type {TipSegmentLayer}
+         */
+        this.tipSegmentLayer = undefined;
+
+
+        /**
+         *
+         * @type {LayerBase|*}
+         */
+        this.metamanagerSegments = undefined;
+        this.tipSegmentLayerMinZoom = 10;
+
+        /**
+         * its layer collection
+         * @type {ItsLayerCollection}
+         */
+        this.itsLayerCollection = undefined;
+
+        /**
+         * param list to set up the sliders, initial weight and drop down selection added later
+         */
+        this._sliderParamArray = [
+            ['AADT', [['aadtyr_1', '1']]],
+            ['AADT Future', [['aadtyr_5', '5'], ['aadtyr_10', '10'], ['aadtyr_15', '15'], ['aadtyr_20', '20']]],
+            ['Growth', [['growth_5', '5'], ['growth_10', '10'], ['growth_15', '15'], ['growth_20', '20']]],
+            ['Truck', [['trkdyr_1', '1'], ['trkdyr_20', '20']]],
+            ['LOS', [['losyr_1', '1']]],
+            ['LOS Future', [['losyr_5', '5'], ['losyr_10', '10'], ['losyr_15', '15'], ['losyr_20', '20']]],
+            ['Crash Rate', [['crash_rate', '1']]],
+            ['Severity', [['crash_severity', '1']]],
+            ['Weather', [['weather', '1']]],
+            ['Event', [['event', '1']]]
+        ];
+
+        /**
+         * Presets list, order of the parameters is important, must match that of the slider param list
+         */
+        this._presetArray =
+            [
+                ['Default TIP', [
+                    [10.0, 'aadtyr_1'],
+                    [7.0, 'aadtyr_20'],
+                    [7.0, 'growth_20'],
+                    [4.0, 'trkdyr_1'],
+                    [12.0, 'losyr_1'],
+                    [12.0, 'losyr_20'],
+                    [15.0, 'crash_rate'],
+                    [13.0, 'crash_severity'],
+                    [9.0, 'weather'],
+                    [11.0, 'event']]
+                ],
+                ['Safety', [
+                    [20.0, 'aadtyr_1'],
+                    [0.0, 'aadtyr_20'],
+                    [0.0, 'growth_20'],
+                    [0.0, 'trkdyr_1'],
+                    [0.0, 'losyr_1'],
+                    [0.0, 'losyr_20'],
+                    [40.0, 'crash_rate'],
+                    [40.0, 'crash_severity'],
+                    [0.0, 'weather'],
+                    [0.0, 'event']]
+                ],
+                ['Mobility Present', [
+                    [25.0, 'aadtyr_1'],
+                    [25.0, 'aadtyr_5'],
+                    [0.0, 'growth_20'],
+                    [25.0, 'trkdyr_1'],
+                    [25.0, 'losyr_1'],
+                    [0.0, 'losyr_20'],
+                    [0.0, 'crash_rate'],
+                    [0.0, 'crash_severity'],
+                    [0.0, 'weather'],
+                    [0.0, 'event']]
+                ],
+                ['Mobility Future', [
+                    [0.0, 'aadtyr_1'],
+                    [25.0, 'aadtyr_20'],
+                    [25.0, 'growth_20'],
+                    [25.0, 'trkdyr_20'],
+                    [0.0, 'losyr_1'],
+                    [25.0, 'losyr_20'],
+                    [0.0, 'crash_rate'],
+                    [0.0, 'crash_severity'],
+                    [0.0, 'weather'],
+                    [0.0, 'event']]
+                ],
+                ['Service', [
+                    [30.0, 'aadtyr_1'],
+                    [0.0, 'aadtyr_20'],
+                    [10.0, 'growth_20'],
+                    [0.0, 'trkdyr_1'],
+                    [30.0, 'losyr_1'],
+                    [30.0, 'losyr_20'],
+                    [0.0, 'crash_rate'],
+                    [0.0, 'crash_severity'],
+                    [0.0, 'weather'],
+                    [0.0, 'event']]
+                ],
+                ['Freight Performance', [
+                    [20.0, 'aadtyr_1'],
+                    [0.0, 'aadtyr_20'],
+                    [0.0, 'growth_20'],
+                    [60.0, 'trkdyr_1'],
+                    [20.0, 'losyr_1'],
+                    [0.0, 'losyr_20'],
+                    [0.0, 'crash_rate'],
+                    [0.0, 'crash_severity'],
+                    [0.0, 'weather'],
+                    [0.0, 'event']]
+                ]
+            ];
+
+        for (var i = 0; i < this._sliderParamArray.length; i++) {
+            this._sliderParamArray[i].push(this._presetArray[0][1][i][0]);
+            this._sliderParamArray[i].push(this._presetArray[0][1][i][1]);
+        }
+
+    }
+}
+
+export default new TipConfig();
diff --git a/projects/tsmo/TipSegmentLayer.js b/projects/tsmo/TipSegmentLayer.js
new file mode 100644
index 0000000..60f299b
--- /dev/null
+++ b/projects/tsmo/TipSegmentLayer.js
@@ -0,0 +1,191 @@
+/**
+ * Created by gavorhes on 12/10/2015.
+ */
+
+import LayerBaseVectorGeoJson from '../../src/layers/LayerBaseVectorGeoJson';
+import mapMove from '../../src/olHelpers/mapMove';
+import mapPopup from '../../src/olHelpers/mapPopup';
+import tipConfig from './TipConfig'
+import * as tipLayerStyles from './tipStyleFunction';
+const $ = require('jquery');
+
+/**
+ * Tip Segment layer with a bunch of configuration applied, extends base GeoJSON
+ */
+class TipSegmentLayer extends LayerBaseVectorGeoJson {
+
+    /**
+     * ITS device layer, types available at http://transportal.cee.wisc.edu/its/inventory/
+     * @param {Sliders} sliders
+     * @param {object} options
+     * @param {boolean} [options.visible=true]
+     * @param {boolean} [options.onDemand=true]
+     * @param {number} [options.onDemandDelay=300] delay before the map move callback should be called
+     * @param [options.$loadingGif=undefined] jquery reference to loading gif
+     * @param [options.$regionSelector=undefined] jquery reference to region selector
+     *
+     * @param {object} [options.selectionStyle={}] the selection style configuration
+     * @param {string} [options.selectionStyle.color=rgba(255,170,0,0.5)] the selection color
+     * @param {number} [options.selectionStyle.width=10] the selection width for linear features
+     * @param {object} [options.selectionStyle.olStyle=undefined] an openlayers style object
+     * @param {propertiesZoomStyle} [options.selectionStyle.olStyleFunction=undefined] modified OL style function
+     */
+    constructor(sliders, options) {
+        options['name'] = "TIP Segments";
+        options.onDemand = true;
+        options.style = tipLayerStyles.tipStyle;
+        options['minZoom'] = tipConfig.tipSegmentLayerMinZoom;
+        options['transform'] = {};
+        options.transform.dataProjection = 'EPSG:3857';
+        options.transform.featureProjection = 'EPSG:3857';
+        options['id'] = 'tip-segments';
+        super('tip/gettipfeatures?callback=?', options);
+        this._$regionSelector = options.$regionSelector || undefined;
+
+
+        this._$loadingGif = options.$loadingGif;
+
+        this.legendContent += '<ul><li><hr class="cell-min"></li><li><hr class="cell-very-low"></li>' +
+            '<li><hr class="cell-low"></li><li><hr class="cell-mid"></li><li><hr class="cell-high"></li>' +
+            '<li><hr class="cell-very-high"></li><li><hr class="cell-max">' +
+            '</li></ul>';
+
+        this.legendContent += '<ul>';
+        this.legendContent += '<li style="height: 15px;">low</li>';
+        this.legendContent += '<li style="height: 25px; margin-top: 4px;">medium</li>';
+        this.legendContent += '<li style="height: 15px; margin-top: 5px;">high</li>';
+        this.legendContent += '</ul>';
+
+        this.legendContent += '<span style="margin: 12px 0 0 52px; display: block">Relative Need</span>';
+
+        this._geometry_cache = [];
+
+        this.sliders = sliders;
+        tipConfig.tipSegmentLayer = this;
+
+        let _this = this;
+
+        mapPopup.addVectorPopup(this, function (props) {
+
+            let parms = {
+                pdpId: props['p'],
+                paramWeights: JSON.stringify(_this.sliders.getParams())
+            };
+
+            return '<iframe style="margin-top:5px;" ' +
+                `src="tip/getsegprops?${$.param(parms)}" ` +
+                `height="250" width="350"></iframe>`;
+        }, options.selectionStyle);
+    }
+
+    /**
+     * override base method to include show or hide loading indicator
+     * @param {number} zoom
+     * @param {string} [evtType=undefined] undefined for initial load, otherwise one of 'change:center', 'change:resolution'
+     * @returns {boolean}
+     */
+    mapMoveBefore(zoom, evtType) {
+        let returnVal = super.mapMoveBefore(zoom, evtType);
+
+        if (this._$loadingGif) {
+            if (returnVal) {
+                this._$loadingGif.fadeIn();
+            } else {
+                this._$loadingGif.fadeOut();
+            }
+        }
+        return returnVal;
+    }
+
+    /**
+     * callback to generate the parameters passed in the get request
+     * @param {object} extent
+     * @param {number} extent.minX
+     * @param {number} extent.minY
+     * @param {number} extent.maxX
+     * @param {number} extent.maxY
+     * @param {number} zoomLevel
+     */
+    mapMoveMakeGetParams(extent, zoomLevel) {
+        super.mapMoveMakeGetParams(extent, zoomLevel);
+        $.extend(this.mapMoveParams, extent);
+        this.mapMoveParams['param_weights'] = JSON.stringify(this.sliders.getParams());
+        if (this._$regionSelector){
+            this.mapMoveParams['region'] = this._$regionSelector.val();
+        }
+    }
+
+
+    /**
+     * callback function on map move
+     * @param d the json response
+     */
+    mapMoveCallback(d) {
+        if (d['geojson'].features.length > 0) {
+            this._geometry_cache = this._geometry_cache.concat(d['geojson'].features);
+            this._geometry_cache.sort(function (a, b) {
+                return (a.properties['p'] - b.properties['p']);
+            });
+        }
+
+        let passFeatures = [];
+
+        for (let s of d['scores']) {
+
+            let feat = this._set_pdp_score(s['p'], s['z']);
+            if (feat) {
+                passFeatures.push(feat);
+            }
+        }
+
+        d['geojson'].features = passFeatures;
+        super.mapMoveCallback(d['geojson']);
+        if (this._$loadingGif) {
+            this._$loadingGif.fadeOut();
+        }
+    }
+
+    /**
+     * set the score for the feature
+     * @param {number} pdp pdp_id
+     * @param {number} score to assign
+     * @param {object} [in_array=this._geometry_cache]
+     * @returns {*} the feature with the score assigned
+     * @private
+     */
+    _set_pdp_score(pdp, score, in_array) {
+        if (typeof in_array == 'undefined') {
+            in_array = this._geometry_cache;
+        }
+
+        if (in_array.length == 0) {
+            return undefined;
+        }
+
+        if (in_array.length == 1) {
+            let feat = in_array[0];
+            if (feat.properties['p'] == pdp) {
+                feat.properties['z'] = score;
+                return feat;
+            } else {
+                return undefined;
+            }
+        }
+
+        let midPoint = Math.floor(in_array.length / 2);
+
+        let midFeature = in_array[midPoint];
+        if (midFeature.properties['p'] == pdp) {
+            midFeature.properties['z'] = score;
+            return midFeature;
+        } else if (pdp < midFeature.properties['p']) {
+            return this._set_pdp_score(pdp, score, in_array.slice(0, midPoint));
+        } else {
+            return this._set_pdp_score(pdp, score, in_array.slice(midPoint + 1));
+        }
+    }
+
+}
+
+
+export default TipSegmentLayer;
diff --git a/projects/tsmo/legend-test.js b/projects/tsmo/legend-test.js
new file mode 100644
index 0000000..361da67
--- /dev/null
+++ b/projects/tsmo/legend-test.js
@@ -0,0 +1,84 @@
+/**
+ * Created by gavorhes on 12/18/2015.
+ */
+import quickMap from '../../src/olHelpers/quickMap';
+import mapMove from '../../src/olHelpers/mapMove';
+import mapPopup from '../../src/olHelpers/mapPopup';
+import ItsLayerCollection from '../../src/collections/ItsLayerCollection';
+import LayerLegend from '../../src/collections/LayerLegend';
+
+let map = quickMap();
+
+mapMove.init(map);
+mapPopup.init(map);
+
+let itsLayerCollection = new ItsLayerCollection(map);
+
+let layerArray = [
+    {
+        groupName: 'ITS Inventory Layers',
+        collapse: false,
+        addCheck: false,
+        items: itsLayerCollection.layers
+    }
+];
+
+let legend = new LayerLegend(layerArray, 'legend-container', {});
+
+//
+//let layerArry = itsLayerCollection.layers.slice(0, 3);
+//
+//layerArry.push(
+//    {
+//        groupName: 'Group 1',
+//        collapse: false,
+//        items: itsLayerCollection.layers.slice(3, 6)
+//    }
+//);
+//
+//layerArry.push(itsLayerCollection.layers[6]);
+//
+//let collection2 = itsLayerCollection.layers.slice(7, 9);
+//collection2.push({
+//
+//    groupName: 'Group 3',
+//    collapse: false,
+//    items: itsLayerCollection.layers.slice(10, 12)
+//
+//});
+//
+//layerArry.push(
+//    {
+//        groupName: 'Group 2',
+//        collapse: true,
+//        items: collection2
+//    }
+//);
+//
+//let layerArray = [
+//    itsLayerCollection.layers[0],
+//    itsLayerCollection.layers[1],
+//    {
+//
+//        groupName: 'Group 1',
+//        collapse: false,
+//        items: [itsLayerCollection.layers[2], itsLayerCollection.layers[3]]
+//    },
+//    {
+//        groupName: 'Group 2',
+//        collapse: true,
+//        addCheck: false,
+//        items: [
+//            itsLayerCollection.layers[4],
+//            itsLayerCollection.layers[5],
+//            {
+//                groupName: 'Group 3',
+//                collapse: false,
+//                items: [itsLayerCollection.layers[6], itsLayerCollection.layers[7]]
+//            }
+//        ]
+//    }
+//];
+
+
+
diff --git a/projects/tsmo/main-report.js b/projects/tsmo/main-report.js
new file mode 100644
index 0000000..6b8d948
--- /dev/null
+++ b/projects/tsmo/main-report.js
@@ -0,0 +1,198 @@
+/**
+ * Created by gavorhes on 12/22/2015.
+ */
+
+import quickMap from '../../src/olHelpers/quickMap';
+import LayerBaseVectorGeoJson from '../../src/layers/LayerBaseVectorGeoJson';
+import mapPopup from '../../src/olHelpers/mapPopup';
+import * as tipLayerStyles from './tipStyleFunction';
+import LayerEsriMapServer from '../../src/layers/LayerEsriMapServer';
+import ItsLayerCollection from '../../src/collections/ItsLayerCollection';
+import LayerLegend from '../../src/collections/LayerLegend';
+import {} from '../../lib/jquery.floatThead';
+let $ = require('jquery');
+
+let suppressLowScores = true;
+
+function tipStyleReport(feature, resolution) {
+    "use strict";
+
+    let seg_score = feature.getProperties()['z'];
+
+    if (seg_score < 1 && suppressLowScores) {
+        return null;
+    } else {
+        return tipLayerStyles.tipStyle(feature, resolution);
+    }
+}
+
+
+(function () {
+    "use strict";
+
+    let $crashData = $('#crash-data');
+
+    function getCrashData(pdpId) {
+        $crashData.html('');
+
+        $.get('getcrash', {pdpId: pdpId}, function (d) {
+            $crashData.html(d);
+        }, 'text')
+    }
+
+    // add the cell classes
+    $('.param-val').each(function () {
+        let z = parseFloat($(this).attr('data-crumb'));
+        let cellClass = 'cell-max';
+
+        if (z < -2.5) {
+            cellClass = 'cell-min';
+        } else if (z < -1.5) {
+            cellClass = 'cell-very-low';
+        } else if (z < -0.5) {
+            cellClass = 'cell-low';
+        } else if (z < 0.5) {
+            cellClass = 'cell-mid';
+        } else if (z < 1.5) {
+            cellClass = 'cell-high';
+        } else if (z < 2.5) {
+            cellClass = 'cell-very-high';
+        }
+
+        $(this).addClass(cellClass);
+    });
+
+    let map = quickMap({center: {x: -10012438, y: 5548095}, zoom: 6, minZoom: 6});
+
+    let wisDotRegions = new LayerEsriMapServer(
+        'http://transportal.cee.wisc.edu/applications/arcgis2/rest/services/MetaManager/Metamanager_regions/MapServer',
+        {
+            minZoom: 6,
+            maxZoom: 15,
+            opacity: 0.4
+        });
+
+    map.addLayer(wisDotRegions.olLayer);
+
+    let reportSegments = new LayerBaseVectorGeoJson('',
+        {
+            loadCallback: function (theLayer) {
+                map.getView().fit(theLayer.source.getExtent(), map.getSize());
+            },
+            transform: {featureProjection: 'EPSG:3857', dataProjection: 'EPSG:3857'},
+            style: tipStyleReport,
+            params: {d: (new Date()).getTime().toString()},
+            zIndex: 10
+        }
+    );
+
+    let geoJsonData = JSON.parse($('#geojson-data').val());
+    reportSegments.addFeatures(geoJsonData);
+
+    map.addLayer(reportSegments.olLayer);
+    map.getView().fit(reportSegments.source.getExtent(), map.getSize());
+
+    let itsLayerCollection = new ItsLayerCollection(map);
+
+    for (let l of itsLayerCollection.layers) {
+        l.zIndex = 10;
+        l.opacity = 0.5;
+    }
+
+    let layerArray = [
+        {
+            groupName: 'ITS Inventory Layers',
+            collapse: true,
+            addCheck: true,
+            items: itsLayerCollection.layers
+        }
+    ];
+
+    let legend = new LayerLegend(layerArray, 'legend-container', {});
+
+    let $resultsTable = $('#results-table');
+    let $resultsContainer = $('#results-container');
+
+
+
+    mapPopup.addMapClickFunction(function () {
+        $('.selected-row').removeClass('selected-row');
+    });
+
+    //return undefined in popup callback to prevent default popup display
+    let selectionLayer = mapPopup.addVectorPopup(reportSegments, function (props) {
+            let pdp = props['p'];
+            getCrashData(pdp);
+            let $selectedRow = $resultsTable.find(`#${pdp}`);
+            $selectedRow.addClass('selected-row');
+            $resultsContainer.find('table tbody').scrollTop(0);
+            $resultsContainer.find('table tbody').scrollTop($selectedRow.offset().top - 60);
+            return undefined;
+        },
+        {olStyle: tipLayerStyles.tipStyleSelection}
+    );
+
+    mapPopup.addMapClickFunction(function () {
+        $crashData.html('');
+    });
+
+    $('.selectable-row').click(function () {
+        let mapView = map.getView();
+        selectionLayer.getSource().clear();
+        $('.selected-row').removeClass('selected-row');
+        $(this).addClass('selected-row');
+        let pdp = parseInt(this.id);
+        let feats = reportSegments.source.getFeatures();
+        getCrashData(pdp);
+        for (let feature of feats) {
+            if (feature.getProperties()['p'] == pdp) {
+                selectionLayer.getSource().addFeature(feature);
+                mapView.fit(feature.getGeometry().getExtent(), map.getSize());
+                mapView.setZoom(mapView.getZoom() - 2);
+                return;
+            }
+        }
+    });
+
+    let $scoreUnderOneRows = $('.score-under-one-flag');
+
+    $('#show-all-toggle').click(function () {
+        if (this.checked) {
+            $scoreUnderOneRows.removeClass('score-under-one');
+            suppressLowScores = false;
+        } else {
+            $scoreUnderOneRows.addClass('score-under-one');
+            suppressLowScores = true;
+        }
+        reportSegments.refresh();
+        mapPopup.closePopup();
+        $('.selected-row').removeClass('selected-row');
+        $('#crash-table').html('');
+
+    }).prop('checked', false);
+
+    let oneRowTd = $resultsTable.find('tr:nth-of-type(2)').children('td');
+    let $firstTrTh = $resultsTable.find('tr:first-of-type').children('th');
+
+    function updateTableColumnWidth(){
+
+        let colWidths = [];
+
+        $firstTrTh.each(function(i, el){
+            colWidths.push($(el).width());
+        });
+
+        oneRowTd.each(function(i, el){
+            $(el).width(colWidths[i]);
+        });
+    }
+
+    updateTableColumnWidth();
+
+    $resultsTable.find('tr:first-of-type').click(function(){
+       setTimeout(updateTableColumnWidth, 50);
+
+    });
+
+
+})();
diff --git a/projects/tsmo/main-ui.js b/projects/tsmo/main-ui.js
new file mode 100644
index 0000000..48be215
--- /dev/null
+++ b/projects/tsmo/main-ui.js
@@ -0,0 +1,126 @@
+/**
+ * Created by gavorhes on 12/14/2015.
+ */
+import tipConfig from './TipConfig';
+import mapMove from '../../src/olHelpers/mapMove';
+import LayerLegend from '../../src/collections/LayerLegend';
+const $ = require('jquery');
+
+/**
+ * start the UI setup
+ */
+export function startUi() {
+
+    tipConfig.$loadingGif = $('#loading-gif');
+    tipConfig.$loadingGif.fadeIn();
+
+    tipConfig.$regionSelector = $('#region-selector');
+    tipConfig.$regionSelector.val('all');
+
+    // configure the preset selector
+    let $presetSelector = $('#preset-selector');
+    for (let i = 0; i < tipConfig._presetArray.length; i++) {
+        let weights = {};
+        let sumCheck = 0;
+        for (let j = 0; j < tipConfig._sliderParamArray.length; j++) {
+            sumCheck += tipConfig._presetArray[i][1][j][0];
+            weights[tipConfig._sliderParamArray[j][0].toLowerCase().replace(/ /g, '-')] = tipConfig._presetArray[i][1][j];
+        }
+        if (sumCheck != 100) {
+            alert('Sum not equal to 100 for preset ' + tipConfig._presetArray[i][0]);
+        }
+
+        let optionHtml = '<option value="';
+        optionHtml += JSON.stringify(weights).replace(/"/g, '&quot;');
+        optionHtml += '"';
+        optionHtml += (i == 0 ? ' selected="selected"' : '') + '>' + tipConfig._presetArray[i][0] + '</option>';
+
+        $presetSelector.append(optionHtml);
+    }
+
+    //enable get help button
+    $('#tip-help').click(function () {
+
+        let win = window.open('tip/help', '_blank');
+        if (win) {
+            //Browser has allowed it to be opened
+            win.focus();
+        } else {
+            //Browser has blocked it
+            alert('Please allow popups for this site');
+        }
+    });
+}
+
+
+/**
+ * finish the UI setup
+ */
+export function endUi() {
+    tipConfig.$loadingGif.fadeOut();
+
+    $('#make-report').click(function () {
+        let params = mapMove.mapExtent;
+        params['paramWeights'] = JSON.stringify(tipConfig.sliders.getParams());
+        params['region'] = $('#region-selector').val();
+        params['preset'] = $('#preset-selector option:selected').text();
+
+        let win = window.open('tip/getreport?' + $.param(params), '_blank');
+        if (win) {
+            //Browser has allowed it to be opened
+            win.focus();
+        } else {
+            //Browser has blocked it
+            alert('Please allow popups for this site');
+        }
+    });
+
+    //add events to the preset selector
+    let $presetSelector = $('#preset-selector');
+
+    // add preset selector change handler
+    let first = true;
+    $presetSelector.change(function () {
+
+        var weightSelection = JSON.parse(this.value);
+        tipConfig.sliders.setValues(weightSelection);
+        if (first) {
+            first = false;
+        } else {
+            mapMove.triggerLyrLoad(tipConfig.tipSegmentLayer);
+        }
+    });
+
+    $presetSelector.trigger('change');
+
+    $('#region-selector').change(function(){
+        "use strict";
+        mapMove.triggerLyrLoad(tipConfig.tipSegmentLayer);
+    });
+
+
+    tipConfig.sliders.addSlideFinishedFunction(function () {
+        $presetSelector[0].selectedIndex = 0;
+        mapMove.triggerLyrLoad(tipConfig.tipSegmentLayer);
+    });
+}
+
+export function endUiMap() {
+
+    mapMove.addCallback(function (ext, zoom, evt) {
+        $('#make-report').prop('disabled', !(zoom >= tipConfig.tipSegmentLayerMinZoom));
+    });
+
+    let layerArray = [
+        tipConfig.tipSegmentLayer,
+        tipConfig.metamanagerSegments,
+        {
+            groupName: 'ITS Inventory Layers',
+            collapse: true,
+            addCheck: false,
+            items: tipConfig.itsLayerCollection.layers
+        }
+    ];
+
+    let legend = new LayerLegend(layerArray, 'legend-container', {});
+}
diff --git a/projects/tsmo/main.js b/projects/tsmo/main.js
new file mode 100644
index 0000000..3a73607
--- /dev/null
+++ b/projects/tsmo/main.js
@@ -0,0 +1,166 @@
+/**
+ * Created by gavorhes on 12/14/2015.
+ */
+import Sliders from '../../src/collections/Sliders';
+import tipConfig from './TipConfig';
+import mapPopup from '../../src/olHelpers/mapPopup';
+import * as uiSetup from './main-ui';
+import LayerEsriMapServer from '../../src/layers/LayerEsriMapServer';
+import ItsLayerCollection from '../../src/collections/ItsLayerCollection';
+import quickMap from '../../src/olHelpers/quickMap';
+import TipSegmentLayer from './TipSegmentLayer';
+
+(function () {
+    "use strict";
+
+    glob.config = tipConfig;
+
+    uiSetup.startUi();
+
+    let sliders = new Sliders(tipConfig._sliderParamArray, 'slider-container');
+    tipConfig.sliders = sliders;
+
+    let map = quickMap({minZoom: 7});
+
+    tipConfig.map = map;
+
+    let wisDotRegions = new LayerEsriMapServer(
+        'http://transportal.cee.wisc.edu/applications/arcgis2/rest/services/MetaManager/Metamanager_regions/MapServer',
+        {
+            minZoom: 6,
+            maxZoom: 12,
+            name: 'WisDOT Regions',
+            useEsriStyle: true
+        });
+
+    map.addLayer(wisDotRegions.olLayer);
+
+    //initialize the tip segment layer
+    let tipSegmentLayer = new TipSegmentLayer(sliders,
+        {
+            selectionStyle: {
+                color: 'rgba(0,0,255,0.5)',
+                width: 7
+            },
+            $loadingGif: tipConfig.$loadingGif,
+            $regionSelector: tipConfig.$regionSelector
+        }
+    );
+    map.addLayer(tipSegmentLayer.olLayer);
+    tipConfig.tipSegmentLayer = tipSegmentLayer;
+
+    let metamanagerSegments = new LayerEsriMapServer(
+        'http://transportal.cee.wisc.edu/applications/arcgis2/rest/services/MetaManager/MM_All_Segments/MapServer',
+        {
+            minZoom: 7,
+            visible: false,
+            name: 'Metamanager Segments'
+        });
+
+    map.addLayer(metamanagerSegments.olLayer);
+    tipConfig.metamanagerSegments = metamanagerSegments;
+
+    //initialize the ITS layer collection
+    tipConfig.itsLayerCollection = new ItsLayerCollection(map);
+
+    uiSetup.endUi();
+    uiSetup.endUiMap();
+
+})();
+
+
+//
+//let sliders;
+//    let app;
+//    let extraLayers = {};
+//
+//    $(function () {
+//        return;
+//        // create the sliders object
+//        sliders = new SliderGroup(tipConfig._sliderParamArray, 'slider-container');
+//
+//        $('#btnResetDefault').click(function () {
+//            sliders.reset();
+//        });
+//
+
+//
+//        let geoJsonArray = [];
+//
+//        for (i = 0; i < itsLayerConfig.length; i++) {
+//            geoJsonArray.push(new ItsGeoJsonLayer(itsLayerConfig[i], app.map, 'its-legend-items'));
+//        }
+//
+//        $('#showVis').prop('checked', true);
+//
+//        $('input[name=showAll]').change(function () {
+//            let $legendDivs = $('#its-legend-items > div');
+//            if (this.id == 'showAll' && this.checked) {
+//                $legendDivs.addClass('force-show');
+//            } else {
+//                $legendDivs.removeClass('force-show');
+//            }
+//        });
+//
+//        function showHide() {
+//            let shown = true;
+//            let $legend = $('#its-legend-items');
+//            let $pic = $('#show-hide');
+//            return function () {
+//                if (shown) {
+//                    $legend.slideUp();
+//                    shown = false;
+//                    $pic.attr('src', $pic.attr('src').replace('up', 'down'));
+//                } else {
+//                    $legend.slideDown();
+//                    shown = true;
+//                    $pic.attr('src', $pic.attr('src').replace('down', 'up'));
+//                }
+//            }
+//        }
+//
+//
+//        $('#show-hide').click(showHide()).trigger('click');
+//
+//        $('#show-hide-all-its').prop('checked', true).change(function () {
+//
+//            for (let i = 0; i < geoJsonArray.length; i++) {
+//                geoJsonArray[i].groupVisible = this.checked;
+//            }
+//
+//        });
+//
+//
+//        let url = 'http://transportal.cee.wisc.edu/applications/arcgis2/rest/services/MetaManager/MM_All_Segments/MapServer';
+//
+//        extraLayers['mm-layer'] = new ol.layer.Tile({
+//            source: new ol.source.TileArcGISRest({
+//                url: url
+//            })
+//        });
+//
+//        let layerCheckboxes = $('#layer-switch').find('input[type="checkbox"]');
+//
+//        layerCheckboxes.each(function (idx, val) {
+//            val.checked = false;
+//        });
+//
+//        layerCheckboxes.click(function () {
+//            if (this.checked) {
+//                app.map.addLayer(extraLayers[this.id])
+//            } else {
+//                app.map.removeLayer(extraLayers[this.id])
+//            }
+//        });
+//
+//        //fade out the loading gif when complete
+//        $gif.fadeOut();
+//
+
+//
+
+//    });
+
+
+
+
diff --git a/projects/tsmo/slider-test.js b/projects/tsmo/slider-test.js
new file mode 100644
index 0000000..6bf939d
--- /dev/null
+++ b/projects/tsmo/slider-test.js
@@ -0,0 +1,29 @@
+/**
+ * Created by gavorhes on 12/14/2015.
+ */
+
+import Sliders from '../../src/collections/Sliders';
+import {} from '../../src/jquery-plugin/dayRange';
+import tipConfig from './TipConfig';
+const $ = require('jquery');
+
+(function(){
+    //alert('here');
+
+    let dayRange = $('#dates').dayRange(10);
+
+
+    //let sliders = new Sliders(tipConfig._sliderParamArray, 'slider-container');
+    //glob.sliders = sliders;
+    //
+    //
+    //
+    //
+    //console.log(tipConfig);
+})();
+
+
+
+
+
+
diff --git a/projects/tsmo/tipStyleFunction.js b/projects/tsmo/tipStyleFunction.js
new file mode 100644
index 0000000..33d6c2f
--- /dev/null
+++ b/projects/tsmo/tipStyleFunction.js
@@ -0,0 +1,84 @@
+/**
+ * Created by gavorhes on 12/22/2015.
+ */
+import * as zoomResolutionConvert from '../../src/olHelpers/zoomResolutionConvert';
+import ol from '../../src/custom-ol';
+
+
+/**
+ * tip style function
+ * @param feature
+ * @param resolution
+ * @returns {*[]}
+ */
+export function tipStyle(feature, resolution) {
+
+    let zoom = zoomResolutionConvert.resolutionToZoom(resolution);
+
+    let width = 5;
+
+    if (zoom >= 18){
+        width = 10;
+    } else if (zoom >= 14) {
+        width = 8;
+    } else if (zoom >= 12) {
+        width = 7;
+    } else if (zoom >= 10) {
+        width = 6;
+    }
+
+    let seg_score = feature.getProperties()['z'];
+    let hex_color;
+    if (seg_score < -2.5) {
+        hex_color = '#00ff00';
+    } else if (seg_score < -1.5) {
+        hex_color = '#55ff00';
+    } else if (seg_score < -0.5) {
+        hex_color = '#aaff00';
+    } else if (seg_score < 0.5) {
+        hex_color = '#ffff00';
+    } else if (seg_score < 1.5) {
+        hex_color = '#ffaa00';
+    } else if (seg_score < 2.5) {
+        hex_color = '#ff5500';
+    } else {
+        hex_color = '#ff0000'
+    }
+
+    return [new ol.style.Style({
+        stroke: new ol.style.Stroke({
+            color: hex_color,
+            width: width
+        })
+    })];
+}
+
+/**
+ * selection style
+ * @param feature
+ * @param resolution
+ * @returns {*[]}
+ */
+export function tipStyleSelection (feature, resolution) {
+
+    let zoom = zoomResolutionConvert.resolutionToZoom(resolution);
+
+    let width = 5;
+
+    if (zoom >= 18){
+        width = 10;
+    } else if (zoom >= 14) {
+        width = 8;
+    } else if (zoom >= 12) {
+        width = 7;
+    } else if (zoom >= 10) {
+        width = 6;
+    }
+
+    return [new ol.style.Style({
+        stroke: new ol.style.Stroke({
+            color: '#0074ff',
+            width: width
+        })
+    })];
+}
\ No newline at end of file
diff --git a/src/collections/ItsLayerCollection.js b/src/collections/ItsLayerCollection.js
index 78d4078..d446ae3 100644
--- a/src/collections/ItsLayerCollection.js
+++ b/src/collections/ItsLayerCollection.js
@@ -123,7 +123,7 @@ class ItsLayerCollection {
 
     /**
      * Create a collection of all ITS layers
-     * @param {Map} theMap the openlayers map
+     * @param {ol.Map} theMap the openlayers map
      * @param {Array} [exclude=[]] array of Its layer identifiers to exclude
      *
      * BLUE Bluetooth Detector - Bluetooth Detector
diff --git a/src/collections/LayerLegend.js b/src/collections/LayerLegend.js
index 6d7bb32..d852685 100644
--- a/src/collections/LayerLegend.js
+++ b/src/collections/LayerLegend.js
@@ -1,7 +1,7 @@
 /**
  * Created by gavorhes on 12/16/2015.
  */
-const $ = require('jquery');
+import $ from '../jquery';
 import provide from '../util/provide';
 import makeGuid from '../util/makeGuid';
 import mapMove from '../olHelpers/mapMove';
diff --git a/src/collections/Sliders.js b/src/collections/Sliders.js
index 3ea46e3..6254f48 100644
--- a/src/collections/Sliders.js
+++ b/src/collections/Sliders.js
@@ -3,7 +3,7 @@
  */
 import provide from '../util/provide';
 let nm = provide('collections');
-let $ = require('jquery');
+import $ from '../jquery';
 
 class _Slider {
 
diff --git a/src/jquery-plugin/dayRange.js b/src/jquery-plugin/dayRange.js
index 21037cd..75b3b51 100644
--- a/src/jquery-plugin/dayRange.js
+++ b/src/jquery-plugin/dayRange.js
@@ -1,4 +1,4 @@
-let $ = require('jquery');
+import $ from '../jquery';
 require('jquery-ui/datepicker');
 import provide from '../util/provide';
 let nm = provide('jQueryPlugin');
diff --git a/src/jquery-plugin/mediaControl.js b/src/jquery-plugin/mediaControl.js
index 10565ea..5fbe63a 100644
--- a/src/jquery-plugin/mediaControl.js
+++ b/src/jquery-plugin/mediaControl.js
@@ -1,4 +1,4 @@
-const $ = require('jquery');
+import $ from '../jquery';
 import provide from '../util/provide';
 let nm = provide('jQueryPlugin');
 
diff --git a/src/jquery-plugin/rangeChange.js b/src/jquery-plugin/rangeChange.js
index 19d8f69..eea8b88 100644
--- a/src/jquery-plugin/rangeChange.js
+++ b/src/jquery-plugin/rangeChange.js
@@ -4,7 +4,7 @@
  * Created by gavorhes on 11/2/2015.
  */
 
-const $ = require('jquery');
+import $ from '../jquery';
 let mouseIn = false;
 let mouseDown = false;
 let timeout = null;
diff --git a/src/jquery-plugin/rpPicker.js b/src/jquery-plugin/rpPicker.js
index dc7d958..05ffd49 100644
--- a/src/jquery-plugin/rpPicker.js
+++ b/src/jquery-plugin/rpPicker.js
@@ -1,4 +1,4 @@
-const $ = require('jquery');
+import $ from '../jquery';
 import ol from '../custom-ol';
 
 import quickMap from '../olHelpers/quickMap';
diff --git a/src/jquery-plugin/ssaCorridorPicker.js b/src/jquery-plugin/ssaCorridorPicker.js
index eaded16..67855fb 100644
--- a/src/jquery-plugin/ssaCorridorPicker.js
+++ b/src/jquery-plugin/ssaCorridorPicker.js
@@ -1,4 +1,4 @@
-const $ = require('jquery');
+import $ from '../jquery';
 import ol from '../custom-ol';
 
 import quickMap from '../olHelpers/quickMap';
diff --git a/src/jquery.js b/src/jquery.js
new file mode 100644
index 0000000..c863dfd
--- /dev/null
+++ b/src/jquery.js
@@ -0,0 +1,6 @@
+/**
+ * Created by gavorhes on 5/3/2016.
+ */
+const $ = require('jquery/dist/jquery.min');
+
+export default $;
diff --git a/src/layers/LayerBase.js b/src/layers/LayerBase.js
index ed07436..699e4a9 100644
--- a/src/layers/LayerBase.js
+++ b/src/layers/LayerBase.js
@@ -1,4 +1,4 @@
-const $ = require('jquery');
+import $ from '../jquery';
 import makeGuid from '../util/makeGuid';
 import * as zoomResolutionConvert from '../olHelpers/zoomResolutionConvert';
 import provide from '../util/provide';
diff --git a/src/layers/LayerBaseVector.js b/src/layers/LayerBaseVector.js
index 70af827..8dc214a 100644
--- a/src/layers/LayerBaseVector.js
+++ b/src/layers/LayerBaseVector.js
@@ -1,4 +1,4 @@
-const $ = require('jquery');
+import $ from '../jquery';
 import ol from '../custom-ol';
 import LayerBase from './LayerBase';
 import mapMove from '../olHelpers/mapMove';
@@ -183,7 +183,7 @@ class LayerBaseVector extends LayerBase {
 
     /**
      * get the style definition
-     * @type {ol.Style|function}
+     * @type {ol.Style|styleFunc}
      */
     get style() {
         return this._style;
diff --git a/src/layers/LayerBaseVectorEsri.js b/src/layers/LayerBaseVectorEsri.js
index 941cd3a..70862a3 100644
--- a/src/layers/LayerBaseVectorEsri.js
+++ b/src/layers/LayerBaseVectorEsri.js
@@ -1,7 +1,7 @@
 /**
  * Created by gavorhes on 11/2/2015.
  */
-const $ = require('jquery');
+import $ from '../jquery';
 import ol from '../custom-ol';
 import LayerBaseVector from './LayerBaseVector';
 import * as esriToOl from '../olHelpers/esriToOlStyle';
diff --git a/src/layers/LayerBaseVectorGeoJson.js b/src/layers/LayerBaseVectorGeoJson.js
index 73c381c..884a280 100644
--- a/src/layers/LayerBaseVectorGeoJson.js
+++ b/src/layers/LayerBaseVectorGeoJson.js
@@ -1,7 +1,7 @@
 /**
  * Created by gavorhes on 11/2/2015.
  */
-const $ = require('jquery');
+import $ from '../jquery';
 import ol from '../custom-ol';
 import LayerBaseVector from './LayerBaseVector';
 import provide from '../util/provide';
diff --git a/src/layers/LayerBaseXyzTile.js b/src/layers/LayerBaseXyzTile.js
index 3ccefd8..5b6ec6e 100644
--- a/src/layers/LayerBaseXyzTile.js
+++ b/src/layers/LayerBaseXyzTile.js
@@ -1,7 +1,7 @@
 /**
  * Created by gavorhes on 12/4/2015.
  */
-const $ = require('jquery');
+import $ from '../jquery';
 import ol from '../custom-ol';
 import LayerBase from './LayerBase';
 import * as esriToOl from '../olHelpers/esriToOlStyle';
diff --git a/src/layers/LayerEsriMapServer.js b/src/layers/LayerEsriMapServer.js
index 47a23d5..759775a 100644
--- a/src/layers/LayerEsriMapServer.js
+++ b/src/layers/LayerEsriMapServer.js
@@ -1,7 +1,7 @@
 /**
  * Created by gavorhes on 12/7/2015.
  */
-const $ = require('jquery');
+import $ from '../jquery';
 import ol from '../custom-ol';
 import LayerBase from './LayerBase';
 import * as esriToOl from '../olHelpers/esriToOlStyle';
diff --git a/src/layers/LayerItsInventory.js b/src/layers/LayerItsInventory.js
index cb793ff..e440543 100644
--- a/src/layers/LayerItsInventory.js
+++ b/src/layers/LayerItsInventory.js
@@ -2,7 +2,7 @@
  * Created by gavorhes on 12/8/2015.
  */
 
-const $ = require('jquery');
+import $ from '../jquery';
 import ol from '../custom-ol';
 import LayerBaseVectorGeoJson from './LayerBaseVectorGeoJson';
 import mapMove from '../olHelpers/mapMove';
diff --git a/src/layers/LayerRealEarthTile.js b/src/layers/LayerRealEarthTile.js
index c7bdddd..de86709 100644
--- a/src/layers/LayerRealEarthTile.js
+++ b/src/layers/LayerRealEarthTile.js
@@ -2,7 +2,7 @@
  * Created by gavorhes on 11/4/2015.
  */
 
-const $ = require('jquery');
+import $ from '../jquery';
 import LayerBaseXyzTile from './LayerBaseXyzTile';
 import RealEarthAnimateTile from '../mixin/RealEarthAnimateTile';
 let mixIns = require('es6-mixins');
diff --git a/src/mixin/RealEarthAnimate.js b/src/mixin/RealEarthAnimate.js
index 900105d..d2f38fc 100644
--- a/src/mixin/RealEarthAnimate.js
+++ b/src/mixin/RealEarthAnimate.js
@@ -3,7 +3,7 @@
  */
 import provide from '../util/provide';
 import mapPopup from '../olHelpers/mapPopup';
-const $ = require('jquery');
+import $ from '../jquery';
 let nm = provide('mixin');
 
 
diff --git a/src/olHelpers/mapMoveCls.js b/src/olHelpers/mapMoveCls.js
index c93ba7f..127ac8b 100644
--- a/src/olHelpers/mapMoveCls.js
+++ b/src/olHelpers/mapMoveCls.js
@@ -3,7 +3,7 @@
  */
 
 
-const $ = require('jquery');
+import $ from '../jquery';
 import MapInteractionBase from './mapInteractionBase';
 import * as checkDefined from '../util/checkDefined';
 import provide from '../util/provide';
diff --git a/src/olHelpers/mapPopupCls.js b/src/olHelpers/mapPopupCls.js
index 86c9256..8519140 100644
--- a/src/olHelpers/mapPopupCls.js
+++ b/src/olHelpers/mapPopupCls.js
@@ -2,7 +2,7 @@
  * Created by gavorhes on 11/3/2015.
  */
 
-const $ = require('jquery');
+import $ from '../jquery';
 import ol from '../custom-ol';
 import MapInteractionBase from './mapInteractionBase';
 import propertiesZoomStyle from '../olHelpers/propertiesZoomStyle';
diff --git a/src/olHelpers/quickMapBase.js b/src/olHelpers/quickMapBase.js
index 0990140..338af57 100644
--- a/src/olHelpers/quickMapBase.js
+++ b/src/olHelpers/quickMapBase.js
@@ -6,7 +6,7 @@
  * Created by gavorhes on 12/15/2015.
  */
 
-const $ = require('jquery');
+import $ from '../jquery';
 import ol from '../custom-ol';
 import provide from '../util/provide';
 import mapMove from './mapMove';
-- 
GitLab