mirror of
https://github.com/GSA/notifications-admin.git
synced 2025-12-11 23:53:52 -05:00
When an element is obscured by the sticky nav, this method allows you to scroll the page until it is revealled. The bulk of this code was added in: https://github.com/alphagov/notifications-admin/pull/2843 to ensure elements with focus were in view. This moves that into a public method so, as well as being called by the focus event handler, it can be called directly by other code. These changes include code that adds a 'sticky-scroll-area' class to scroll areas not explicitly marked as such in the base HTML but made to be a scroll area by the sticky JS.
996 lines
29 KiB
JavaScript
996 lines
29 KiB
JavaScript
;(function (global) {
|
|
'use strict';
|
|
|
|
var $ = global.jQuery;
|
|
var GOVUK = global.GOVUK || {};
|
|
var _mode = 'default';
|
|
|
|
// Constructor to make objects representing the area sticky elements can scroll in
|
|
var ScrollArea = function (el, edge, selector) {
|
|
var $el = el.$fixedEl;
|
|
var $scrollArea = $el.closest('.sticky-scroll-area');
|
|
|
|
if($scrollArea.length === 0) {
|
|
$scrollArea = $el.parent();
|
|
$scrollArea.addClass('sticky-scroll-area');
|
|
}
|
|
|
|
this._els = [el];
|
|
this.edge = edge;
|
|
this.selector = selector;
|
|
this.node = $scrollArea.get(0);
|
|
this.setEvents();
|
|
};
|
|
ScrollArea.prototype.addEl = function (el) {
|
|
this._els.push(el);
|
|
};
|
|
ScrollArea.prototype.hasEl = function (el) {
|
|
return $.inArray(el, this._els) !== -1;
|
|
};
|
|
ScrollArea.prototype.updateEls = function (usedEls) {
|
|
this._els = usedEls;
|
|
};
|
|
ScrollArea.prototype.setEvents = function () {
|
|
this.node.addEventListener('focus', this.focusHandler.bind(this), true);
|
|
$(this.node).on('keyup', 'textarea', this.focusHandler.bind(this));
|
|
};
|
|
ScrollArea.prototype.removeEvents = function () {
|
|
this.node.removeEventListener('focus', this.focusHandler.bind(this));
|
|
$(this.node).find('textarea').off('keyup', 'textarea', this.focusHandler.bind(this));
|
|
};
|
|
ScrollArea.prototype.getFocusedDetails = {
|
|
forElement: function ($focusedElement) {
|
|
var focused = {
|
|
'top': $focusedElement.offset().top,
|
|
'height': $focusedElement.outerHeight(),
|
|
'type': 'element'
|
|
};
|
|
focused.bottom = focused.top + focused.height;
|
|
|
|
return focused;
|
|
},
|
|
forCaret: function ($textarea) {
|
|
var textarea = $textarea.get(0);
|
|
var caretCoordinates = window.getCaretCoordinates(textarea, textarea.selectionEnd);
|
|
var focused = {
|
|
'top': $textarea.offset().top + caretCoordinates.top,
|
|
'height': caretCoordinates.height,
|
|
'type': 'caret'
|
|
};
|
|
|
|
focused.bottom = focused.top + focused.height;
|
|
|
|
return focused;
|
|
}
|
|
};
|
|
ScrollArea.prototype.focusHandler = function (e) {
|
|
this.scrollToRevealElement($(document.activeElement));
|
|
};
|
|
ScrollArea.prototype.scrollToRevealElement = function ($el) {
|
|
var nodeName = $el.get(0).nodeName.toLowerCase();
|
|
var endOfFurthestEl = focusOverlap.endOfFurthestEl(this._els, this.edge);
|
|
var isInSticky = function () {
|
|
return $el.closest(this.selector).length > 0;
|
|
}.bind(this);
|
|
var focused;
|
|
var overlap;
|
|
|
|
// if textarea is focused, we care about checking the caret, not the whole element
|
|
if (nodeName === 'textarea') {
|
|
focused = this.getFocusedDetails.forCaret($el);
|
|
} else {
|
|
if (isInSticky()) { return; }
|
|
focused = this.getFocusedDetails.forElement($el);
|
|
}
|
|
|
|
overlap = focusOverlap.getOverlap(focused, this.edge, endOfFurthestEl);
|
|
|
|
if (overlap > 0) {
|
|
focusOverlap.adjustForOverlap(focused, this.edge, overlap);
|
|
}
|
|
};
|
|
ScrollArea.prototype.destroy = function () {
|
|
this.removeEvents();
|
|
};
|
|
|
|
// Object collecting together methods for interacting with scrollareas
|
|
var scrollAreas = {
|
|
_scrollAreas: [],
|
|
getAreaForEl: function (el) {
|
|
var loopIdx = this._scrollAreas.length;
|
|
|
|
while(loopIdx--) {
|
|
if (this._scrollAreas[loopIdx].hasEl(el)) {
|
|
return this._scrollAreas[loopIdx];
|
|
}
|
|
}
|
|
|
|
return false;
|
|
},
|
|
getAreaByEl: function (el) {
|
|
var matches = $.grep(this._scrollAreas, function (area) {
|
|
return $.inArray(el, area.els) !== -1;
|
|
});
|
|
|
|
return matches[0] || false;
|
|
},
|
|
addEl: function (el, edge, selector) {
|
|
var scrollArea = this.getAreaForEl(el);
|
|
|
|
if (!scrollArea) {
|
|
this._scrollAreas.push(new ScrollArea(el, edge, selector));
|
|
} else {
|
|
scrollArea.addEl(el);
|
|
}
|
|
},
|
|
syncEls: function (elsInDOM) {
|
|
var self = this;
|
|
var unusedAreas = [];
|
|
|
|
var getUsed = function (area) {
|
|
var used = [];
|
|
|
|
$.each(elsInDOM, function (elIdx, el) {
|
|
if (area.hasEl(el)) {
|
|
used.push(el);
|
|
}
|
|
});
|
|
|
|
return used;
|
|
};
|
|
|
|
var deleteUnused = function (idx, areaIdx) {
|
|
// remove any events for overlap checking bound to the scrollArea
|
|
self._scrollAreas[areaIdx].destroy();
|
|
self._scrollAreas.splice(areaIdx, 1);
|
|
};
|
|
|
|
// update any scroll areas with els still in the DOM and track any with none
|
|
$.each(this._scrollAreas, function (areaIdx, area) {
|
|
var used = getUsed(area);
|
|
|
|
if (!used.length) {
|
|
unusedAreas.push(areaIdx);
|
|
}
|
|
|
|
area.updateEls(used);
|
|
});
|
|
|
|
// delete any scroll areas with no els still in DOM
|
|
$.each(unusedAreas, deleteUnused);
|
|
}
|
|
};
|
|
|
|
// Object collecting together methods for stopping sticky overlapping focused elements
|
|
var focusOverlap = {
|
|
getOverlap: function (focused, edge, endOfFurthestEl) {
|
|
if (!endOfFurthestEl) { return 0; }
|
|
|
|
if (edge === 'top') {
|
|
return endOfFurthestEl - focused.top;
|
|
} else {
|
|
return focused.bottom - endOfFurthestEl;
|
|
}
|
|
},
|
|
endOfFurthestEl: function (els, edge) {
|
|
var stuckEls = $.grep(els, function (el) { return el.isStuck(); });
|
|
var edgeOfEl;
|
|
var offsets;
|
|
|
|
if (edge === 'bottom') {
|
|
edgeOfEl = function (el) {
|
|
return el.$fixedEl.offset().top;
|
|
};
|
|
} else {
|
|
edgeOfEl = function (el) {
|
|
return el.$fixedEl.offset().top + el.height;
|
|
};
|
|
}
|
|
|
|
if (!stuckEls.length) { return false; }
|
|
|
|
offsets = $.map(stuckEls, function (el) { return edgeOfEl(el); });
|
|
|
|
return offsets.reduce(function (accumulator, offset) {
|
|
return (accumulator < offset) ? offset: accumulator;
|
|
});
|
|
},
|
|
adjustForOverlap: function (focused, edge, overlap) {
|
|
var scrollTop = $(window).scrollTop();
|
|
|
|
// scroll so element becomes visible
|
|
if (edge === 'top') {
|
|
$(window).scrollTop(scrollTop - overlap);
|
|
} else {
|
|
$(window).scrollTop(scrollTop + overlap);
|
|
}
|
|
}
|
|
};
|
|
|
|
// Object collecting together methods for dealing with marking the edge of a sticky, or group of
|
|
// sticky elements (as seen in dialog mode)
|
|
var oppositeEdge = {
|
|
_classes: {
|
|
'top': 'content-fixed__top',
|
|
'bottom': 'content-fixed__bottom'
|
|
},
|
|
_getClassForEdge: function (edge) {
|
|
return this._classes[edge];
|
|
},
|
|
mark: function (sticky) {
|
|
var edgeClass = this._getClassForEdge(sticky.edge);
|
|
var els;
|
|
|
|
if (_mode === 'dialog') {
|
|
els = [dialog.getElementAtOppositeEnd(sticky)];
|
|
} else {
|
|
els = sticky._els;
|
|
}
|
|
|
|
els = $.grep(els, function (el) { return el.isStuck(); });
|
|
|
|
$.each(els, function (i, el) {
|
|
el.$fixedEl.addClass(edgeClass);
|
|
});
|
|
},
|
|
unmark: function (sticky) {
|
|
var edgeClass = this._getClassForEdge(sticky.edge);
|
|
|
|
$.each(sticky._els, function (i, el) {
|
|
el.$fixedEl.removeClass(edgeClass);
|
|
});
|
|
}
|
|
};
|
|
|
|
// Constructor for objects holding data for each element to have sticky behaviour
|
|
var StickyElement = function ($el, sticky) {
|
|
this._sticky = sticky;
|
|
this.$fixedEl = $el;
|
|
this._initialFixedClass = 'content-fixed-onload';
|
|
this._fixedClass = 'content-fixed';
|
|
this._appliedClass = null;
|
|
this._$shim = null;
|
|
this._stopped = false;
|
|
this._hasLoaded = false;
|
|
this._canBeStuck = true;
|
|
this.verticalMargins = {
|
|
'top': parseInt(this.$fixedEl.css('margin-top'), 10),
|
|
'bottom': parseInt(this.$fixedEl.css('margin-bottom'), 10),
|
|
};
|
|
};
|
|
StickyElement.prototype._getShimCSS = function () {
|
|
return {
|
|
'width': this.horizontalSpace + 'px',
|
|
'height': this.height + 'px',
|
|
'margin-top': this.verticalMargins.top + 'px',
|
|
'margin-bottom': this.verticalMargins.bottom + 'px'
|
|
};
|
|
};
|
|
StickyElement.prototype.stickyClass = function () {
|
|
return (this._sticky._initialPositionsSet) ? this._fixedClass : this._initialFixedClass;
|
|
};
|
|
StickyElement.prototype.appliedClass = function () {
|
|
return this._appliedClass;
|
|
};
|
|
StickyElement.prototype.removeStickyClasses = function (sticky) {
|
|
this.$fixedEl.removeClass([
|
|
this._initialFixedClass,
|
|
this._fixedClass
|
|
].join(' '));
|
|
};
|
|
StickyElement.prototype.isStuck = function () {
|
|
return this._appliedClass !== null;
|
|
};
|
|
StickyElement.prototype.stick = function (sticky) {
|
|
this._appliedClass = this.stickyClass();
|
|
this.$fixedEl.addClass(this._appliedClass);
|
|
this._hasBeenCalled = true;
|
|
};
|
|
StickyElement.prototype.release = function (sticky) {
|
|
this._appliedClass = null;
|
|
this.removeStickyClasses(sticky);
|
|
this._hasBeenCalled = true;
|
|
};
|
|
// When a sticky element is moved into the 'stuck' state, a shim is inserted into the
|
|
// page to preserve the space the element occupies in the flow.
|
|
StickyElement.prototype.addShim = function (position) {
|
|
this._$shim = $('<div class="shim"> </div>');
|
|
this._$shim.css(this._getShimCSS());
|
|
this.$fixedEl[position](this._$shim);
|
|
};
|
|
StickyElement.prototype.removeShim = function () {
|
|
if (this._$shim !== null) {
|
|
this._$shim.remove();
|
|
this._$shim = null;
|
|
}
|
|
};
|
|
// Changes to the dimensions of a sticky element with a shim need to be passed on to the shim
|
|
StickyElement.prototype.updateShim = function () {
|
|
if (this._$shim) {
|
|
this._$shim.css(this._getShimCSS());
|
|
}
|
|
};
|
|
StickyElement.prototype.stop = function () {
|
|
this._stopped = true;
|
|
};
|
|
StickyElement.prototype.unstop = function () {
|
|
this._stopped = false;
|
|
};
|
|
StickyElement.prototype.isStopped = function () {
|
|
return this._stopped;
|
|
};
|
|
StickyElement.prototype.isInPage = function () {
|
|
var node = this.$fixedEl.get(0);
|
|
|
|
return (node === document.body) ? false : document.body.contains(node);
|
|
};
|
|
StickyElement.prototype.canBeStuck = function (val) {
|
|
if (val !== undefined) {
|
|
this._canBeStuck = val;
|
|
} else {
|
|
return this._canBeStuck;
|
|
}
|
|
};
|
|
StickyElement.prototype.hasLoaded = function (val) {
|
|
if (val !== undefined) {
|
|
this._hasLoaded = val;
|
|
} else {
|
|
return this._hasLoaded;
|
|
}
|
|
};
|
|
|
|
// Object collecting together methods for treating sticky elements as if they
|
|
// were wrapped by a dialog component
|
|
var dialog = {
|
|
hasResized: false,
|
|
spaceBetweenStickys: 40,
|
|
// we add padding of 20px around each sticky to give some space between it and the rest of the page
|
|
// this shouldn't apply between stickys in a stack
|
|
// (the in-page CSS handles this by each subsequent sticky in a sequence having margin: -40px)
|
|
_getPaddingBetweenEls: function (els) {
|
|
if (els.length <= 1) { return 0; }
|
|
|
|
return (els.length - 1) * this.spaceBetweenStickys;
|
|
},
|
|
_getTotalHeight: function (els) {
|
|
var reducer = function (accumulator, currentValue) {
|
|
return accumulator + currentValue;
|
|
};
|
|
var combinedHeight = $.map(els, function (el) { return el.height; }).reduce(reducer);
|
|
|
|
return combinedHeight - this._getPaddingBetweenEls(els);
|
|
},
|
|
_elsThatCanBeStuck: function (els) {
|
|
return $.grep(els, function (el) { return el.canBeStuck(); });
|
|
},
|
|
getOffsetFromEdge: function (el, sticky) {
|
|
var els = this._elsThatCanBeStuck(sticky._els).slice();
|
|
var elIdx;
|
|
|
|
// els must be arranged furtherest from window edge is stuck to first
|
|
// default direction is order in document
|
|
if (sticky.edge === 'top') {
|
|
els.reverse();
|
|
}
|
|
|
|
elIdx = els.indexOf(el);
|
|
|
|
// if next to window edge the dialog is stuck to, no offset
|
|
if (elIdx === (els.length - 1)) { return 0; }
|
|
|
|
// make els all those from this one to the window edge
|
|
els = els.slice(elIdx + 1);
|
|
|
|
// remove the space between those els and the one on the edge
|
|
return this._getTotalHeight(els) - this.spaceBetweenStickys;
|
|
},
|
|
getOffsetFromEnd: function (el, sticky) {
|
|
var els = this._elsThatCanBeStuck(sticky._els).slice();
|
|
var elIdx;
|
|
|
|
// els must be arranged furtherest from window edge is stuck to first
|
|
// default direction is order in document
|
|
if (sticky.edge === 'bottom') {
|
|
els.reverse();
|
|
}
|
|
|
|
elIdx = els.indexOf(el);
|
|
|
|
// if next to opposite edge to the one the dialog is stuck to, no offset
|
|
if (elIdx === (els.length - 1)) { return 0; }
|
|
|
|
// make els all those from this one to the window edge
|
|
els = els.slice(elIdx + 1);
|
|
|
|
return this._getTotalHeight(els) - this.spaceBetweenStickys;
|
|
},
|
|
// checks total height of all this._sticky elements against a height
|
|
// unsticks each that won't fit and marks them as unstickable
|
|
fitToHeight: function (sticky) {
|
|
var self = this;
|
|
var els = sticky._els.slice();
|
|
var height = sticky.getWindowDimensions().height;
|
|
var totalStickyHeight = function () {
|
|
return self._getTotalHeight(self._elsThatCanBeStuck(els));
|
|
};
|
|
var dialogFitsHeight = function () {
|
|
return totalStickyHeight() <= height;
|
|
};
|
|
|
|
// els must be arranged furtherest from window edge is stuck to first
|
|
// default direction is order in document
|
|
if (sticky.edge === 'top') {
|
|
els.reverse();
|
|
}
|
|
|
|
// reset elements
|
|
$.each(els, function (i, el) { el.canBeStuck(true); });
|
|
|
|
while (self._elsThatCanBeStuck(els).length && !dialogFitsHeight()) {
|
|
var currentEl = self._elsThatCanBeStuck(els)[0];
|
|
|
|
sticky.reset(currentEl);
|
|
currentEl.canBeStuck(false);
|
|
|
|
if (!self.hasResized) { self.hasResized = true; }
|
|
}
|
|
},
|
|
getElementAtStickyEdge: function (sticky) {
|
|
var els = this._elsThatCanBeStuck(sticky._els);
|
|
var idx = (sticky.edge === 'top') ? 0 : els.length - 1;
|
|
|
|
return els[idx];
|
|
},
|
|
// get element at the end opposite the sticky edge
|
|
getElementAtOppositeEnd: function (sticky) {
|
|
var els = this._elsThatCanBeStuck(sticky._els);
|
|
var idx = (sticky.edge === 'top') ? els.length - 1 : 0;
|
|
|
|
return els[idx];
|
|
},
|
|
getInPageEdgePosition: function (sticky) {
|
|
return this.getElementAtStickyEdge(sticky).inPageEdgePosition;
|
|
},
|
|
getHeight: function (els) {
|
|
return this._getTotalHeight(this._elsThatCanBeStuck(els));
|
|
},
|
|
adjustForResize: function (sticky) {
|
|
var windowHeight = sticky.getWindowDimensions().height;
|
|
|
|
if (sticky.edge === 'top') {
|
|
$(window).scrollTop(this.getInPageEdgePosition(sticky));
|
|
} else {
|
|
$(window).scrollTop(this.getInPageEdgePosition(sticky) - windowHeight);
|
|
}
|
|
|
|
this.hasResized = false;
|
|
},
|
|
releaseEl: function (el, sticky) {
|
|
el.$fixedEl.css(sticky.edge, '');
|
|
}
|
|
};
|
|
|
|
// Constructor for objects collecting together all generic behaviour for controlling the state of
|
|
// sticky elements
|
|
var Sticky = function (selector) {
|
|
this._hasScrolled = false;
|
|
this._scrollTimeout = false;
|
|
this._windowHasResized = false;
|
|
this._resizeTimeout = false;
|
|
this._elsLoaded = false;
|
|
this._initialPositionsSet = false;
|
|
this._els = [];
|
|
|
|
this.CSS_SELECTOR = selector;
|
|
this.STOP_PADDING = 10;
|
|
};
|
|
Sticky.prototype.setMode = function (mode) {
|
|
_mode = mode;
|
|
};
|
|
Sticky.prototype.getWindowDimensions = function () {
|
|
return {
|
|
height: $(global).height(),
|
|
width: $(global).width()
|
|
};
|
|
};
|
|
Sticky.prototype.getWindowPositions = function () {
|
|
return {
|
|
scrollTop: $(global).scrollTop()
|
|
};
|
|
};
|
|
// Change state of sticky elements based on their position relative to the window
|
|
Sticky.prototype.setElementPositions = function () {
|
|
var self = this,
|
|
windowDimensions = self.getWindowDimensions(),
|
|
windowTop = self.getWindowPositions().scrollTop,
|
|
windowPositions = {
|
|
'top': windowTop,
|
|
'bottom': windowTop + windowDimensions.height
|
|
};
|
|
|
|
var _setElementPosition = function (el) {
|
|
if (self.viewportIsWideEnough(windowDimensions.width)) {
|
|
|
|
if (self.windowNotPastScrolledFrom(windowPositions, self.getScrolledFrom(el))) {
|
|
self.reset(el);
|
|
} else { // past the point it sits in the document
|
|
if (self.windowNotPastScrollingTo(windowPositions, self.getScrollingTo(el))) {
|
|
self.stick(el);
|
|
if (el.isStopped()) {
|
|
self.unstop(el);
|
|
}
|
|
} else { // window past scrollingTo position
|
|
if (!el.isStuck()) {
|
|
self.stick(el);
|
|
}
|
|
self.stop(el);
|
|
}
|
|
}
|
|
|
|
} else {
|
|
|
|
self.reset(el);
|
|
|
|
}
|
|
};
|
|
|
|
// clean up any existing styles marking the edges of sticky elements
|
|
oppositeEdge.unmark(self);
|
|
|
|
$.each(self._els, function (i, el) {
|
|
if (el.canBeStuck()) {
|
|
_setElementPosition(el);
|
|
}
|
|
});
|
|
|
|
// add styles to mark the edge of sticky elements opposite to that stuck to the window
|
|
oppositeEdge.mark(self);
|
|
|
|
if (self._initialPositionsSet === false) { self._initialPositionsSet = true; }
|
|
};
|
|
// Store all the dimensions for a sticky element to limit DOM queries
|
|
Sticky.prototype.setElementDimensions = function (el, callback) {
|
|
var self = this;
|
|
var $el = el.$fixedEl;
|
|
var onHeightSet = function () {
|
|
// if element is shim'ed, pass changes in dimension on to the shim
|
|
if (el._$shim) {
|
|
el.updateShim();
|
|
}
|
|
if (callback !== undefined) {
|
|
callback();
|
|
}
|
|
};
|
|
|
|
this.setElWidth(el);
|
|
this.setElHeight(el, onHeightSet);
|
|
};
|
|
// Reset element to original state in the page
|
|
Sticky.prototype.reset = function (el) {
|
|
if (el.isStopped()) {
|
|
this.unstop(el);
|
|
}
|
|
if (el.isStuck()) {
|
|
this.release(el);
|
|
}
|
|
};
|
|
// Recalculate stored dimensions for all sticky elements
|
|
Sticky.prototype.recalculate = function () {
|
|
var self = this;
|
|
var onSyncComplete = function () {
|
|
scrollAreas.syncEls(self._els);
|
|
self.setEvents();
|
|
if (_mode === 'dialog') {
|
|
dialog.fitToHeight(self);
|
|
if (dialog.hasResized) {
|
|
dialog.adjustForResize(self);
|
|
}
|
|
}
|
|
self.setElementPositions();
|
|
};
|
|
|
|
this.syncWithDOM(onSyncComplete);
|
|
};
|
|
// Public method to scroll so an element isn't covered by the sticky nav
|
|
Sticky.prototype.scrollToRevealElement = function (el) {
|
|
var $el = $(el);
|
|
var scrollAreaNode = $el.closest('.sticky-scroll-area').get(0);
|
|
var matches = $.grep(scrollAreas._scrollAreas, function (scrollArea) {
|
|
return scrollArea.node === scrollAreaNode;
|
|
});
|
|
|
|
if (matches.length) {
|
|
matches[0].scrollToRevealElement($el);
|
|
}
|
|
};
|
|
Sticky.prototype.setElWidth = function (el) {
|
|
var $el = el.$fixedEl;
|
|
var scrollArea = scrollAreas.getAreaByEl(el);
|
|
var width = $(scrollArea.node).width();
|
|
|
|
el.horizontalSpace = width;
|
|
// if stuck, element won't inherit width from parent so set explicitly
|
|
if (el._$shim) {
|
|
$el.width(width);
|
|
}
|
|
};
|
|
Sticky.prototype.setElHeight = function (el, callback) {
|
|
var self = this;
|
|
var $el = el.$fixedEl;
|
|
var $img = $el.find('img');
|
|
var onload = function () {
|
|
el.height = $el.outerHeight();
|
|
// if element has a shim, the shim's offset represents the element's in-page position
|
|
if (el._$shim) {
|
|
el.inPageEdgePosition = self.getInPageEdgePosition(el._$shim);
|
|
} else {
|
|
el.inPageEdgePosition = self.getInPageEdgePosition($el);
|
|
}
|
|
callback();
|
|
};
|
|
|
|
if ((!el.hasLoaded()) && ($img.length > 0)) {
|
|
var image = new global.Image();
|
|
image.onload = function () {
|
|
onload();
|
|
};
|
|
image.src = $img.attr('src');
|
|
} else {
|
|
onload();
|
|
}
|
|
};
|
|
Sticky.prototype.allElementsLoaded = function (totalEls) {
|
|
return this._els.length === totalEls;
|
|
};
|
|
Sticky.prototype.getElForNode = function (node) {
|
|
var matches = $.grep(this._els, function (el) { return el.$fixedEl.is(node); });
|
|
|
|
return !!matches.length ? matches[0] : false;
|
|
};
|
|
Sticky.prototype.add = function (el, setPositions, cb) {
|
|
var self = this;
|
|
var $el = $(el);
|
|
var onDimensionsSet;
|
|
var elObj = this.getElForNode(el);
|
|
var exists = !!elObj;
|
|
|
|
onDimensionsSet = function () {
|
|
elObj.hasLoaded(true);
|
|
|
|
// guard against adding elements already stored
|
|
if (!exists) {
|
|
self._els.push(elObj);
|
|
}
|
|
|
|
if (setPositions) {
|
|
self.setElementPositions();
|
|
}
|
|
|
|
if (cb !== undefined) {
|
|
cb();
|
|
}
|
|
};
|
|
|
|
if (!exists) {
|
|
elObj = new StickyElement($el, self);
|
|
scrollAreas.addEl(elObj, self.edge, self.CSS_SELECTOR);
|
|
}
|
|
|
|
self.setElementDimensions(elObj, onDimensionsSet);
|
|
};
|
|
Sticky.prototype.remove = function (el) {
|
|
if ($.inArray(el, this._els) !== -1) {
|
|
|
|
// reset DOM node to original state
|
|
this.reset(el);
|
|
|
|
// remove sticky element object
|
|
this._els = $.grep(this._els, function (_el) { return _el !== el; });
|
|
}
|
|
};
|
|
// gets all sticky elements in the DOM and removes any in this._els no longer in attached to it
|
|
Sticky.prototype.syncWithDOM = function (callback) {
|
|
var self = this;
|
|
var $els = $(self.CSS_SELECTOR);
|
|
var numOfEls = $els.length;
|
|
var onLoaded;
|
|
|
|
onLoaded = function () {
|
|
if (self._els.length === numOfEls) {
|
|
self.endOfScrollArea = self.getEndOfScrollArea();
|
|
if (callback !== undefined) {
|
|
callback();
|
|
}
|
|
}
|
|
};
|
|
|
|
// remove any els no longer in the DOM
|
|
if (this._els.length) {
|
|
$.each(this._els, function (i, el) {
|
|
if (!el.isInPage()) {
|
|
self.remove(el);
|
|
}
|
|
});
|
|
}
|
|
|
|
if (numOfEls) {
|
|
// reset flag marking page load
|
|
this._initialPositionsSet = false;
|
|
|
|
$els.each(function (i, el) {
|
|
// delay setting position until all stickys are loaded
|
|
self.add(el, false, onLoaded);
|
|
});
|
|
}
|
|
};
|
|
Sticky.prototype.init = function () {
|
|
this.recalculate();
|
|
};
|
|
Sticky.prototype.setEvents = function () {
|
|
this._scrollEvent = this.onScroll.bind(this);
|
|
this._resizeEvent = this.onResize.bind(this);
|
|
|
|
// flag when scrolling takes place and check (and re-position) sticky elements relative to
|
|
// window position
|
|
if (this._scrollTimeout === false) {
|
|
$(global).scroll(this._scrollEvent);
|
|
this._scrollTimeout = global.setInterval(this.checkScroll.bind(this), 50);
|
|
}
|
|
|
|
// Recalculate all dimensions when the window resizes
|
|
if (this._resizeTimeout === false) {
|
|
$(global).resize(this._resizeEvent);
|
|
this._resizeTimeout = global.setInterval(this.checkResize.bind(this), 50);
|
|
}
|
|
};
|
|
Sticky.prototype.clearEvents = function () {
|
|
if (this._scrollTimeout !== false) {
|
|
$(global).off('scroll', this._scrollEvent);
|
|
global.clearInterval(this._scrollTimeout);
|
|
this._scrollTimeout = false;
|
|
}
|
|
|
|
if (this._resizeTimeout !== false) {
|
|
$(global).off('resize', this._resizeEvent);
|
|
global.clearInterval(this._resizeTimeout);
|
|
this._resizeTimeout = false;
|
|
}
|
|
|
|
};
|
|
Sticky.prototype.viewportIsWideEnough = function (windowWidth) {
|
|
return windowWidth > 768;
|
|
};
|
|
Sticky.prototype.onScroll = function () {
|
|
this._hasScrolled = true;
|
|
};
|
|
Sticky.prototype.onResize = function () {
|
|
this._windowHasResized = true;
|
|
};
|
|
Sticky.prototype.checkScroll = function () {
|
|
var self = this;
|
|
|
|
if (self._hasScrolled === true) {
|
|
self._hasScrolled = false;
|
|
self.setElementPositions();
|
|
}
|
|
};
|
|
Sticky.prototype.checkResize = function () {
|
|
var self = this,
|
|
windowWidth = self.getWindowDimensions().width;
|
|
|
|
if (self._windowHasResized === true) {
|
|
self._windowHasResized = false;
|
|
|
|
$.each(self._els, function (i, el) {
|
|
if (!self.viewportIsWideEnough(windowWidth)) {
|
|
self.reset(el);
|
|
} else {
|
|
self.setElementDimensions(el);
|
|
}
|
|
});
|
|
|
|
if (self.viewportIsWideEnough(windowWidth)) {
|
|
if (_mode === 'dialog') {
|
|
dialog.fitToHeight(self);
|
|
if (dialog.hasResized) {
|
|
dialog.adjustForResize(self);
|
|
}
|
|
}
|
|
self.setElementPositions();
|
|
}
|
|
}
|
|
};
|
|
Sticky.prototype.release = function (el) {
|
|
if (el.isStuck()) {
|
|
var $el = el.$fixedEl;
|
|
|
|
el.removeStickyClasses(this);
|
|
$el.css('width', '');
|
|
// clear styles from any elements stuck while in a dialog mode
|
|
dialog.releaseEl(el, this);
|
|
el.removeShim();
|
|
el.release(this);
|
|
}
|
|
};
|
|
|
|
// Extension of sticky object to add behaviours specific to sticking to top of window
|
|
var stickAtTop = new Sticky('.js-stick-at-top-when-scrolling');
|
|
stickAtTop.edge = 'top';
|
|
// Store furthest point sticky elements are allowed
|
|
stickAtTop.getEndOfScrollArea = function () {
|
|
var footer = $('.js-footer:eq(0)');
|
|
if (footer.length === 0) {
|
|
return 0;
|
|
}
|
|
return footer.offset().top - this.STOP_PADDING;
|
|
};
|
|
// position of the bottom edge when in the page flow
|
|
stickAtTop.getInPageEdgePosition = function ($el) {
|
|
return $el.offset().top;
|
|
};
|
|
stickAtTop.getScrolledFrom = function (el) {
|
|
if (_mode === 'dialog') {
|
|
return dialog.getInPageEdgePosition(this);
|
|
} else {
|
|
return el.inPageEdgePosition;
|
|
}
|
|
};
|
|
stickAtTop.getScrollingTo = function (el) {
|
|
var height = el.height;
|
|
|
|
if (_mode === 'dialog') {
|
|
height = dialog.getHeight(this._els);
|
|
}
|
|
|
|
return this.endOfScrollArea - height;
|
|
};
|
|
stickAtTop.getStoppingPosition = function (el) {
|
|
var offset = 0;
|
|
|
|
if (_mode === 'dialog') {
|
|
offset = dialog.getOffsetFromEnd(el, this);
|
|
}
|
|
|
|
return (this.endOfScrollArea - offset) - el.height;
|
|
};
|
|
stickAtTop.windowNotPastScrolledFrom = function (windowPositions, scrolledFrom) {
|
|
return scrolledFrom > windowPositions.top;
|
|
};
|
|
stickAtTop.windowNotPastScrollingTo = function (windowPositions, scrollingTo) {
|
|
return windowPositions.top < scrollingTo;
|
|
};
|
|
stickAtTop.stick = function (el) {
|
|
if (!el.isStuck()) {
|
|
var $el = el.$fixedEl;
|
|
var offset = 0;
|
|
|
|
if (_mode === 'dialog') {
|
|
offset = dialog.getOffsetFromEdge(el, this);
|
|
}
|
|
|
|
el.addShim('before');
|
|
$el.css({
|
|
// element will be absolutely positioned so cannot rely on parent element for width
|
|
'width': $el.width() + 'px',
|
|
'top': offset + 'px'
|
|
});
|
|
el.stick(this);
|
|
}
|
|
};
|
|
stickAtTop.stop = function (el) {
|
|
if (!el.isStopped()) {
|
|
el.$fixedEl.css({
|
|
'position': 'absolute',
|
|
'top': this.getStoppingPosition(el)
|
|
});
|
|
el.stop();
|
|
}
|
|
};
|
|
stickAtTop.unstop = function (el) {
|
|
var offset = 0;
|
|
|
|
if (_mode === 'dialog') {
|
|
offset = dialog.getOffsetFromEdge(el, this);
|
|
}
|
|
|
|
el.$fixedEl.css({
|
|
'position': '',
|
|
'top': offset + 'px'
|
|
});
|
|
el.unstop();
|
|
};
|
|
|
|
// Extension of sticky object to add behaviours specific to sticking to bottom of window
|
|
var stickAtBottom = new Sticky('.js-stick-at-bottom-when-scrolling');
|
|
stickAtBottom.edge = 'bottom';
|
|
// Store furthest point sticky elements are allowed
|
|
stickAtBottom.getEndOfScrollArea = function () {
|
|
var header = $('.js-header:eq(0)');
|
|
if (header.length === 0) {
|
|
return 0;
|
|
}
|
|
return (header.offset().top + header.outerHeight()) + this.STOP_PADDING;
|
|
};
|
|
// position of the bottom edge when in the page flow
|
|
stickAtBottom.getInPageEdgePosition = function ($el) {
|
|
return $el.offset().top + $el.outerHeight();
|
|
};
|
|
stickAtBottom.getScrolledFrom = function (el) {
|
|
if (_mode === 'dialog') {
|
|
return dialog.getInPageEdgePosition(this);
|
|
} else {
|
|
return el.inPageEdgePosition;
|
|
}
|
|
};
|
|
stickAtBottom.getScrollingTo = function (el) {
|
|
var height = el.height;
|
|
|
|
if (_mode === 'dialog') {
|
|
height = dialog.getHeight(this._els);
|
|
}
|
|
|
|
return this.endOfScrollArea + height;
|
|
};
|
|
stickAtBottom.getStoppingPosition = function (el) {
|
|
var offset = 0;
|
|
|
|
if (_mode === 'dialog') {
|
|
offset = dialog.getOffsetFromEnd(el, this);
|
|
}
|
|
|
|
return this.endOfScrollArea + offset;
|
|
};
|
|
stickAtBottom.windowNotPastScrolledFrom = function (windowPositions, scrolledFrom) {
|
|
return scrolledFrom < windowPositions.bottom;
|
|
};
|
|
stickAtBottom.windowNotPastScrollingTo = function (windowPositions, scrollingTo) {
|
|
return windowPositions.bottom > scrollingTo;
|
|
};
|
|
stickAtBottom.stick = function (el) {
|
|
if (!el.isStuck()) {
|
|
var $el = el.$fixedEl;
|
|
var offset = 0;
|
|
|
|
if (_mode === 'dialog') {
|
|
offset = dialog.getOffsetFromEdge(el, this);
|
|
}
|
|
|
|
el.addShim('after');
|
|
$el.css({
|
|
// element will be absolutely positioned so cannot rely on parent element for width
|
|
'width': $el.width() + 'px',
|
|
'bottom': offset + 'px'
|
|
});
|
|
el.stick(this);
|
|
}
|
|
};
|
|
stickAtBottom.stop = function (el) {
|
|
if (!el.isStopped()) {
|
|
el.$fixedEl.css({
|
|
'position': 'absolute',
|
|
'top': this.getStoppingPosition(el),
|
|
'bottom': 'auto'
|
|
});
|
|
el.stop();
|
|
}
|
|
};
|
|
stickAtBottom.unstop = function (el) {
|
|
var offset = 0;
|
|
|
|
if (_mode === 'dialog') {
|
|
offset = dialog.getOffsetFromEdge(el, this);
|
|
}
|
|
|
|
el.$fixedEl.css({
|
|
'position': '',
|
|
'top': '',
|
|
'bottom': offset + 'px'
|
|
});
|
|
el.unstop();
|
|
};
|
|
|
|
GOVUK.stickAtTopWhenScrolling = stickAtTop;
|
|
GOVUK.stickAtBottomWhenScrolling = stickAtBottom;
|
|
global.GOVUK = GOVUK;
|
|
})(window);
|