diff --git a/scripts/www.js b/scripts/www.js
index 92d0a779..99eed55c 100644
--- a/scripts/www.js
+++ b/scripts/www.js
@@ -14,9 +14,9 @@ fs.copySync("node_modules/sinon/pkg/sinon.js", currentReleaseRoot + "/node_modul
fs.copySync("node_modules/mock-socket/dist/mock-socket.js", currentReleaseRoot + "/node_modules/mock-socket/dist/mock-socket.js");
fs.copySync("test/", currentReleaseRoot + "/test");
fs.copySync("src/", currentReleaseRoot + "/src");
-fs.copySync("src/htmx.js", "www/js/htmx.js");
-fs.copySync("src/ext/class-tools.js", "www/js/class-tools.js");
-fs.copySync("src/ext/preload.js", "www/js/preload.js");
+fs.copySync("src/htmx.js", "www/themes/htmx-theme/static/js/htmx.js");
+fs.copySync("src/ext/class-tools.js", "www/themes/htmx-theme/static/js/class-tools.js");
+fs.copySync("src/ext/preload.js", "www/themes/htmx-theme/static/js/preload.js");
var testHTML = "
HTMX TESTS
\n"
fs.readdirSync(testRoot).reverse().forEach(function (file) {
diff --git a/www/js/class-tools.js b/www/js/class-tools.js
deleted file mode 100644
index 1cf4b426..00000000
--- a/www/js/class-tools.js
+++ /dev/null
@@ -1,92 +0,0 @@
-(function () {
-
- function splitOnWhitespace(trigger) {
- return trigger.split(/\s+/);
- }
-
- function parseClassOperation(trimmedValue) {
- var split = splitOnWhitespace(trimmedValue);
- if (split.length > 1) {
- var operation = split[0];
- var classDef = split[1].trim();
- var cssClass;
- var delay;
- if (classDef.indexOf(":") > 0) {
- var splitCssClass = classDef.split(':');
- cssClass = splitCssClass[0];
- delay = htmx.parseInterval(splitCssClass[1]);
- } else {
- cssClass = classDef;
- delay = 100;
- }
- return {
- operation: operation,
- cssClass: cssClass,
- delay: delay
- }
- } else {
- return null;
- }
- }
-
- function performOperation(elt, classOperation, classList, currentRunTime) {
- setTimeout(function () {
- elt.classList[classOperation.operation].call(elt.classList, classOperation.cssClass);
- }, currentRunTime)
- }
-
- function toggleOperation(elt, classOperation, classList, currentRunTime) {
- setTimeout(function () {
- setInterval(function () {
- elt.classList[classOperation.operation].call(elt.classList, classOperation.cssClass);
- }, classOperation.delay);
- }, currentRunTime)
- }
-
- function processClassList(elt, classList) {
- var runs = classList.split("&");
- for (var i = 0; i < runs.length; i++) {
- var run = runs[i];
- var currentRunTime = 0;
- var classOperations = run.split(",");
- for (var j = 0; j < classOperations.length; j++) {
- var value = classOperations[j];
- var trimmedValue = value.trim();
- var classOperation = parseClassOperation(trimmedValue);
- if (classOperation) {
- if (classOperation.operation === "toggle") {
- toggleOperation(elt, classOperation, classList, currentRunTime);
- currentRunTime = currentRunTime + classOperation.delay;
- } else {
- currentRunTime = currentRunTime + classOperation.delay;
- performOperation(elt, classOperation, classList, currentRunTime);
- }
- }
- }
- }
- }
-
- function maybeProcessClasses(elt) {
- if (elt.getAttribute) {
- var classList = elt.getAttribute("classes") || elt.getAttribute("data-classes");
- if (classList) {
- processClassList(elt, classList);
- }
- }
- }
-
- htmx.defineExtension('class-tools', {
- onEvent: function (name, evt) {
- if (name === "htmx:afterProcessNode") {
- var elt = evt.detail.elt;
- maybeProcessClasses(elt);
- if (elt.querySelectorAll) {
- var children = elt.querySelectorAll("[classes], [data-classes]");
- for (var i = 0; i < children.length; i++) {
- maybeProcessClasses(children[i]);
- }
- }
- }
- }
- });
-})();
\ No newline at end of file
diff --git a/www/js/htmx.js b/www/js/htmx.js
deleted file mode 100644
index be02b740..00000000
--- a/www/js/htmx.js
+++ /dev/null
@@ -1,3538 +0,0 @@
-// UMD insanity
-// This code sets up support for (in order) AMD, ES6 modules, and globals.
-(function (root, factory) {
- //@ts-ignore
- if (typeof define === 'function' && define.amd) {
- // AMD. Register as an anonymous module.
- //@ts-ignore
- define([], factory);
- } else if (typeof module === 'object' && module.exports) {
- // Node. Does not work with strict CommonJS, but
- // only CommonJS-like environments that support module.exports,
- // like Node.
- module.exports = factory();
- } else {
- // Browser globals
- root.htmx = root.htmx || factory();
- }
-}(typeof self !== 'undefined' ? self : this, function () {
-return (function () {
- 'use strict';
-
- // Public API
- //** @type {import("./htmx").HtmxApi} */
- // TODO: list all methods in public API
- var htmx = {
- onLoad: onLoadHelper,
- process: processNode,
- on: addEventListenerImpl,
- off: removeEventListenerImpl,
- trigger : triggerEvent,
- ajax : ajaxHelper,
- find : find,
- findAll : findAll,
- closest : closest,
- values : function(elt, type){
- var inputValues = getInputValues(elt, type || "post");
- return inputValues.values;
- },
- remove : removeElement,
- addClass : addClassToElement,
- removeClass : removeClassFromElement,
- toggleClass : toggleClassOnElement,
- takeClass : takeClassForElement,
- defineExtension : defineExtension,
- removeExtension : removeExtension,
- logAll : logAll,
- logger : null,
- config : {
- historyEnabled:true,
- historyCacheSize:10,
- refreshOnHistoryMiss:false,
- defaultSwapStyle:'innerHTML',
- defaultSwapDelay:0,
- defaultSettleDelay:20,
- includeIndicatorStyles:true,
- indicatorClass:'htmx-indicator',
- requestClass:'htmx-request',
- addedClass:'htmx-added',
- settlingClass:'htmx-settling',
- swappingClass:'htmx-swapping',
- allowEval:true,
- inlineScriptNonce:'',
- attributesToSettle:["class", "style", "width", "height"],
- withCredentials:false,
- timeout:0,
- wsReconnectDelay: 'full-jitter',
- wsBinaryType: 'blob',
- disableSelector: "[hx-disable], [data-hx-disable]",
- useTemplateFragments: false,
- scrollBehavior: 'smooth',
- defaultFocusScroll: false,
- getCacheBusterParam: false,
- globalViewTransitions: false,
- },
- parseInterval:parseInterval,
- _:internalEval,
- createEventSource: function(url){
- return new EventSource(url, {withCredentials:true})
- },
- createWebSocket: function(url){
- var sock = new WebSocket(url, []);
- sock.binaryType = htmx.config.wsBinaryType;
- return sock;
- },
- version: "1.9.0"
- };
-
- /** @type {import("./htmx").HtmxInternalApi} */
- var internalAPI = {
- addTriggerHandler: addTriggerHandler,
- bodyContains: bodyContains,
- canAccessLocalStorage: canAccessLocalStorage,
- filterValues: filterValues,
- hasAttribute: hasAttribute,
- getAttributeValue: getAttributeValue,
- getClosestMatch: getClosestMatch,
- getExpressionVars: getExpressionVars,
- getHeaders: getHeaders,
- getInputValues: getInputValues,
- getInternalData: getInternalData,
- getSwapSpecification: getSwapSpecification,
- getTriggerSpecs: getTriggerSpecs,
- getTarget: getTarget,
- makeFragment: makeFragment,
- mergeObjects: mergeObjects,
- makeSettleInfo: makeSettleInfo,
- oobSwap: oobSwap,
- selectAndSwap: selectAndSwap,
- settleImmediately: settleImmediately,
- shouldCancel: shouldCancel,
- triggerEvent: triggerEvent,
- triggerErrorEvent: triggerErrorEvent,
- withExtensions: withExtensions,
- }
-
- var VERBS = ['get', 'post', 'put', 'delete', 'patch'];
- var VERB_SELECTOR = VERBS.map(function(verb){
- return "[hx-" + verb + "], [data-hx-" + verb + "]"
- }).join(", ");
-
- //====================================================================
- // Utilities
- //====================================================================
-
- function parseInterval(str) {
- if (str == undefined) {
- return undefined
- }
- if (str.slice(-2) == "ms") {
- return parseFloat(str.slice(0,-2)) || undefined
- }
- if (str.slice(-1) == "s") {
- return (parseFloat(str.slice(0,-1)) * 1000) || undefined
- }
- if (str.slice(-1) == "m") {
- return (parseFloat(str.slice(0,-1)) * 1000 * 60) || undefined
- }
- return parseFloat(str) || undefined
- }
-
- /**
- * @param {HTMLElement} elt
- * @param {string} name
- * @returns {(string | null)}
- */
- function getRawAttribute(elt, name) {
- return elt.getAttribute && elt.getAttribute(name);
- }
-
- // resolve with both hx and data-hx prefixes
- function hasAttribute(elt, qualifiedName) {
- return elt.hasAttribute && (elt.hasAttribute(qualifiedName) ||
- elt.hasAttribute("data-" + qualifiedName));
- }
-
- /**
- *
- * @param {HTMLElement} elt
- * @param {string} qualifiedName
- * @returns {(string | null)}
- */
- function getAttributeValue(elt, qualifiedName) {
- return getRawAttribute(elt, qualifiedName) || getRawAttribute(elt, "data-" + qualifiedName);
- }
-
- /**
- * @param {HTMLElement} elt
- * @returns {HTMLElement | null}
- */
- function parentElt(elt) {
- return elt.parentElement;
- }
-
- /**
- * @returns {Document}
- */
- function getDocument() {
- return document;
- }
-
- /**
- * @param {HTMLElement} elt
- * @param {(e:HTMLElement) => boolean} condition
- * @returns {HTMLElement | null}
- */
- function getClosestMatch(elt, condition) {
- while (elt && !condition(elt)) {
- elt = parentElt(elt);
- }
-
- return elt ? elt : null;
- }
-
- function getAttributeValueWithDisinheritance(initialElement, ancestor, attributeName){
- var attributeValue = getAttributeValue(ancestor, attributeName);
- var disinherit = getAttributeValue(ancestor, "hx-disinherit");
- if (initialElement !== ancestor && disinherit && (disinherit === "*" || disinherit.split(" ").indexOf(attributeName) >= 0)) {
- return "unset";
- } else {
- return attributeValue
- }
- }
-
- /**
- * @param {HTMLElement} elt
- * @param {string} attributeName
- * @returns {string | null}
- */
- function getClosestAttributeValue(elt, attributeName) {
- var closestAttr = null;
- getClosestMatch(elt, function (e) {
- return closestAttr = getAttributeValueWithDisinheritance(elt, e, attributeName);
- });
- if (closestAttr !== "unset") {
- return closestAttr;
- }
- }
-
- /**
- * @param {HTMLElement} elt
- * @param {string} selector
- * @returns {boolean}
- */
- function matches(elt, selector) {
- // @ts-ignore: non-standard properties for browser compatability
- // noinspection JSUnresolvedVariable
- var matchesFunction = elt.matches || elt.matchesSelector || elt.msMatchesSelector || elt.mozMatchesSelector || elt.webkitMatchesSelector || elt.oMatchesSelector;
- return matchesFunction && matchesFunction.call(elt, selector);
- }
-
- /**
- * @param {string} str
- * @returns {string}
- */
- function getStartTag(str) {
- var tagMatcher = /<([a-z][^\/\0>\x20\t\r\n\f]*)/i
- var match = tagMatcher.exec( str );
- if (match) {
- return match[1].toLowerCase();
- } else {
- return "";
- }
- }
-
- /**
- *
- * @param {string} resp
- * @param {number} depth
- * @returns {Element}
- */
- function parseHTML(resp, depth) {
- var parser = new DOMParser();
- var responseDoc = parser.parseFromString(resp, "text/html");
-
- /** @type {Element} */
- var responseNode = responseDoc.body;
- while (depth > 0) {
- depth--;
- // @ts-ignore
- responseNode = responseNode.firstChild;
- }
- if (responseNode == null) {
- // @ts-ignore
- responseNode = getDocument().createDocumentFragment();
- }
- return responseNode;
- }
-
- function aFullPageResponse(resp) {
- return resp.match(/" + resp + "", 0);
- // @ts-ignore type mismatch between DocumentFragment and Element.
- // TODO: Are these close enough for htmx to use interchangably?
- return documentFragment.querySelector('template').content;
- } else {
- var startTag = getStartTag(resp);
- switch (startTag) {
- case "thead":
- case "tbody":
- case "tfoot":
- case "colgroup":
- case "caption":
- return parseHTML("", 1);
- case "col":
- return parseHTML("", 2);
- case "tr":
- return parseHTML("", 2);
- case "td":
- case "th":
- return parseHTML("", 3);
- case "script":
- return parseHTML("" + resp + "
", 1);
- default:
- return parseHTML(resp, 0);
- }
- }
- }
-
- /**
- * @param {Function} func
- */
- function maybeCall(func){
- if(func) {
- func();
- }
- }
-
- /**
- * @param {any} o
- * @param {string} type
- * @returns
- */
- function isType(o, type) {
- return Object.prototype.toString.call(o) === "[object " + type + "]";
- }
-
- /**
- * @param {*} o
- * @returns {o is Function}
- */
- function isFunction(o) {
- return isType(o, "Function");
- }
-
- /**
- * @param {*} o
- * @returns {o is Object}
- */
- function isRawObject(o) {
- return isType(o, "Object");
- }
-
- /**
- * getInternalData retrieves "private" data stored by htmx within an element
- * @param {HTMLElement} elt
- * @returns {*}
- */
- function getInternalData(elt) {
- var dataProp = 'htmx-internal-data';
- var data = elt[dataProp];
- if (!data) {
- data = elt[dataProp] = {};
- }
- return data;
- }
-
- /**
- * toArray converts an ArrayLike object into a real array.
- * @param {ArrayLike} arr
- * @returns {any[]}
- */
- function toArray(arr) {
- var returnArr = [];
- if (arr) {
- for (var i = 0; i < arr.length; i++) {
- returnArr.push(arr[i]);
- }
- }
- return returnArr
- }
-
- function forEach(arr, func) {
- if (arr) {
- for (var i = 0; i < arr.length; i++) {
- func(arr[i]);
- }
- }
- }
-
- function isScrolledIntoView(el) {
- var rect = el.getBoundingClientRect();
- var elemTop = rect.top;
- var elemBottom = rect.bottom;
- return elemTop < window.innerHeight && elemBottom >= 0;
- }
-
- function bodyContains(elt) {
- // IE Fix
- if (elt.getRootNode && elt.getRootNode() instanceof ShadowRoot) {
- return getDocument().body.contains(elt.getRootNode().host);
- } else {
- return getDocument().body.contains(elt);
- }
- }
-
- function splitOnWhitespace(trigger) {
- return trigger.trim().split(/\s+/);
- }
-
- /**
- * mergeObjects takes all of the keys from
- * obj2 and duplicates them into obj1
- * @param {Object} obj1
- * @param {Object} obj2
- * @returns {Object}
- */
- function mergeObjects(obj1, obj2) {
- for (var key in obj2) {
- if (obj2.hasOwnProperty(key)) {
- obj1[key] = obj2[key];
- }
- }
- return obj1;
- }
-
- function parseJSON(jString) {
- try {
- return JSON.parse(jString);
- } catch(error) {
- logError(error);
- return null;
- }
- }
-
- function canAccessLocalStorage() {
- var test = 'htmx:localStorageTest';
- try {
- localStorage.setItem(test, test);
- localStorage.removeItem(test);
- return true;
- } catch(e) {
- return false;
- }
- }
-
- //==========================================================================================
- // public API
- //==========================================================================================
-
- function internalEval(str){
- return maybeEval(getDocument().body, function () {
- return eval(str);
- });
- }
-
- function onLoadHelper(callback) {
- var value = htmx.on("htmx:load", function(evt) {
- callback(evt.detail.elt);
- });
- return value;
- }
-
- function logAll(){
- htmx.logger = function(elt, event, data) {
- if(console) {
- console.log(event, elt, data);
- }
- }
- }
-
- function find(eltOrSelector, selector) {
- if (selector) {
- return eltOrSelector.querySelector(selector);
- } else {
- return find(getDocument(), eltOrSelector);
- }
- }
-
- function findAll(eltOrSelector, selector) {
- if (selector) {
- return eltOrSelector.querySelectorAll(selector);
- } else {
- return findAll(getDocument(), eltOrSelector);
- }
- }
-
- function removeElement(elt, delay) {
- elt = resolveTarget(elt);
- if (delay) {
- setTimeout(function(){
- removeElement(elt);
- elt = null;
- }, delay);
- } else {
- elt.parentElement.removeChild(elt);
- }
- }
-
- function addClassToElement(elt, clazz, delay) {
- elt = resolveTarget(elt);
- if (delay) {
- setTimeout(function(){
- addClassToElement(elt, clazz);
- elt = null;
- }, delay);
- } else {
- elt.classList && elt.classList.add(clazz);
- }
- }
-
- function removeClassFromElement(elt, clazz, delay) {
- elt = resolveTarget(elt);
- if (delay) {
- setTimeout(function(){
- removeClassFromElement(elt, clazz);
- elt = null;
- }, delay);
- } else {
- if (elt.classList) {
- elt.classList.remove(clazz);
- // if there are no classes left, remove the class attribute
- if (elt.classList.length === 0) {
- elt.removeAttribute("class");
- }
- }
- }
- }
-
- function toggleClassOnElement(elt, clazz) {
- elt = resolveTarget(elt);
- elt.classList.toggle(clazz);
- }
-
- function takeClassForElement(elt, clazz) {
- elt = resolveTarget(elt);
- forEach(elt.parentElement.children, function(child){
- removeClassFromElement(child, clazz);
- })
- addClassToElement(elt, clazz);
- }
-
- function closest(elt, selector) {
- elt = resolveTarget(elt);
- if (elt.closest) {
- return elt.closest(selector);
- } else {
- // TODO remove when IE goes away
- do{
- if (elt == null || matches(elt, selector)){
- return elt;
- }
- }
- while (elt = elt && parentElt(elt));
- return null;
- }
- }
-
- function normalizeSelector(selector) {
- var trimmedSelector = selector.trim();
- if (trimmedSelector.startsWith("<") && trimmedSelector.endsWith("/>")) {
- return trimmedSelector.substring(1, trimmedSelector.length - 2);
- } else {
- return trimmedSelector;
- }
- }
-
- function querySelectorAllExt(elt, selector) {
- if (selector.indexOf("closest ") === 0) {
- return [closest(elt, normalizeSelector(selector.substr(8)))];
- } else if (selector.indexOf("find ") === 0) {
- return [find(elt, normalizeSelector(selector.substr(5)))];
- } else if (selector.indexOf("next ") === 0) {
- return [scanForwardQuery(elt, normalizeSelector(selector.substr(5)))];
- } else if (selector.indexOf("previous ") === 0) {
- return [scanBackwardsQuery(elt, normalizeSelector(selector.substr(9)))];
- } else if (selector === 'document') {
- return [document];
- } else if (selector === 'window') {
- return [window];
- } else {
- return getDocument().querySelectorAll(normalizeSelector(selector));
- }
- }
-
- var scanForwardQuery = function(start, match) {
- var results = getDocument().querySelectorAll(match);
- for (var i = 0; i < results.length; i++) {
- var elt = results[i];
- if (elt.compareDocumentPosition(start) === Node.DOCUMENT_POSITION_PRECEDING) {
- return elt;
- }
- }
- }
-
- var scanBackwardsQuery = function(start, match) {
- var results = getDocument().querySelectorAll(match);
- for (var i = results.length - 1; i >= 0; i--) {
- var elt = results[i];
- if (elt.compareDocumentPosition(start) === Node.DOCUMENT_POSITION_FOLLOWING) {
- return elt;
- }
- }
- }
-
- function querySelectorExt(eltOrSelector, selector) {
- if (selector) {
- return querySelectorAllExt(eltOrSelector, selector)[0];
- } else {
- return querySelectorAllExt(getDocument().body, eltOrSelector)[0];
- }
- }
-
- function resolveTarget(arg2) {
- if (isType(arg2, 'String')) {
- return find(arg2);
- } else {
- return arg2;
- }
- }
-
- function processEventArgs(arg1, arg2, arg3) {
- if (isFunction(arg2)) {
- return {
- target: getDocument().body,
- event: arg1,
- listener: arg2
- }
- } else {
- return {
- target: resolveTarget(arg1),
- event: arg2,
- listener: arg3
- }
- }
-
- }
-
- function addEventListenerImpl(arg1, arg2, arg3) {
- ready(function(){
- var eventArgs = processEventArgs(arg1, arg2, arg3);
- eventArgs.target.addEventListener(eventArgs.event, eventArgs.listener);
- })
- var b = isFunction(arg2);
- return b ? arg2 : arg3;
- }
-
- function removeEventListenerImpl(arg1, arg2, arg3) {
- ready(function(){
- var eventArgs = processEventArgs(arg1, arg2, arg3);
- eventArgs.target.removeEventListener(eventArgs.event, eventArgs.listener);
- })
- return isFunction(arg2) ? arg2 : arg3;
- }
-
- //====================================================================
- // Node processing
- //====================================================================
-
- var DUMMY_ELT = getDocument().createElement("output"); // dummy element for bad selectors
- function findAttributeTargets(elt, attrName) {
- var attrTarget = getClosestAttributeValue(elt, attrName);
- if (attrTarget) {
- if (attrTarget === "this") {
- return [findThisElement(elt, attrName)];
- } else {
- var result = querySelectorAllExt(elt, attrTarget);
- if (result.length === 0) {
- logError('The selector "' + attrTarget + '" on ' + attrName + " returned no matches!");
- return [DUMMY_ELT]
- } else {
- return result;
- }
- }
- }
- }
-
- function findThisElement(elt, attribute){
- return getClosestMatch(elt, function (elt) {
- return getAttributeValue(elt, attribute) != null;
- })
- }
-
- function getTarget(elt) {
- var targetStr = getClosestAttributeValue(elt, "hx-target");
- if (targetStr) {
- if (targetStr === "this") {
- return findThisElement(elt,'hx-target');
- } else {
- return querySelectorExt(elt, targetStr)
- }
- } else {
- var data = getInternalData(elt);
- if (data.boosted) {
- return getDocument().body;
- } else {
- return elt;
- }
- }
- }
-
- function shouldSettleAttribute(name) {
- var attributesToSettle = htmx.config.attributesToSettle;
- for (var i = 0; i < attributesToSettle.length; i++) {
- if (name === attributesToSettle[i]) {
- return true;
- }
- }
- return false;
- }
-
- function cloneAttributes(mergeTo, mergeFrom) {
- forEach(mergeTo.attributes, function (attr) {
- if (!mergeFrom.hasAttribute(attr.name) && shouldSettleAttribute(attr.name)) {
- mergeTo.removeAttribute(attr.name)
- }
- });
- forEach(mergeFrom.attributes, function (attr) {
- if (shouldSettleAttribute(attr.name)) {
- mergeTo.setAttribute(attr.name, attr.value);
- }
- });
- }
-
- function isInlineSwap(swapStyle, target) {
- var extensions = getExtensions(target);
- for (var i = 0; i < extensions.length; i++) {
- var extension = extensions[i];
- try {
- if (extension.isInlineSwap(swapStyle)) {
- return true;
- }
- } catch(e) {
- logError(e);
- }
- }
- return swapStyle === "outerHTML";
- }
-
- /**
- *
- * @param {string} oobValue
- * @param {HTMLElement} oobElement
- * @param {*} settleInfo
- * @returns
- */
- function oobSwap(oobValue, oobElement, settleInfo) {
- var selector = "#" + oobElement.id;
- var swapStyle = "outerHTML";
- if (oobValue === "true") {
- // do nothing
- } else if (oobValue.indexOf(":") > 0) {
- swapStyle = oobValue.substr(0, oobValue.indexOf(":"));
- selector = oobValue.substr(oobValue.indexOf(":") + 1, oobValue.length);
- } else {
- swapStyle = oobValue;
- }
-
- var targets = getDocument().querySelectorAll(selector);
- if (targets) {
- forEach(
- targets,
- function (target) {
- var fragment;
- var oobElementClone = oobElement.cloneNode(true);
- fragment = getDocument().createDocumentFragment();
- fragment.appendChild(oobElementClone);
- if (!isInlineSwap(swapStyle, target)) {
- fragment = oobElementClone; // if this is not an inline swap, we use the content of the node, not the node itself
- }
-
- var beforeSwapDetails = {shouldSwap: true, target: target, fragment:fragment };
- if (!triggerEvent(target, 'htmx:oobBeforeSwap', beforeSwapDetails)) return;
-
- target = beforeSwapDetails.target; // allow re-targeting
- if (beforeSwapDetails['shouldSwap']){
- swap(swapStyle, target, target, fragment, settleInfo);
- }
- forEach(settleInfo.elts, function (elt) {
- triggerEvent(elt, 'htmx:oobAfterSwap', beforeSwapDetails);
- });
- }
- );
- oobElement.parentNode.removeChild(oobElement);
- } else {
- oobElement.parentNode.removeChild(oobElement);
- triggerErrorEvent(getDocument().body, "htmx:oobErrorNoTarget", {content: oobElement});
- }
- return oobValue;
- }
-
- function handleOutOfBandSwaps(elt, fragment, settleInfo) {
- var oobSelects = getClosestAttributeValue(elt, "hx-select-oob");
- if (oobSelects) {
- var oobSelectValues = oobSelects.split(",");
- for (let i = 0; i < oobSelectValues.length; i++) {
- var oobSelectValue = oobSelectValues[i].split(":", 2);
- var id = oobSelectValue[0].trim();
- if (id.indexOf("#") === 0) {
- id = id.substring(1);
- }
- var oobValue = oobSelectValue[1] || "true";
- var oobElement = fragment.querySelector("#" + id);
- if (oobElement) {
- oobSwap(oobValue, oobElement, settleInfo);
- }
- }
- }
- forEach(findAll(fragment, '[hx-swap-oob], [data-hx-swap-oob]'), function (oobElement) {
- var oobValue = getAttributeValue(oobElement, "hx-swap-oob");
- if (oobValue != null) {
- oobSwap(oobValue, oobElement, settleInfo);
- }
- });
- }
-
- function handlePreservedElements(fragment) {
- forEach(findAll(fragment, '[hx-preserve], [data-hx-preserve]'), function (preservedElt) {
- var id = getAttributeValue(preservedElt, "id");
- var oldElt = getDocument().getElementById(id);
- if (oldElt != null) {
- preservedElt.parentNode.replaceChild(oldElt, preservedElt);
- }
- });
- }
-
- function handleAttributes(parentNode, fragment, settleInfo) {
- forEach(fragment.querySelectorAll("[id]"), function (newNode) {
- if (newNode.id && newNode.id.length > 0) {
- var normalizedId = newNode.id.replace("'", "\\'");
- var normalizedTag = newNode.tagName.replace(':', '\\:');
- var oldNode = parentNode.querySelector(normalizedTag + "[id='" + normalizedId + "']");
- if (oldNode && oldNode !== parentNode) {
- var newAttributes = newNode.cloneNode();
- cloneAttributes(newNode, oldNode);
- settleInfo.tasks.push(function () {
- cloneAttributes(newNode, newAttributes);
- });
- }
- }
- });
- }
-
- function makeAjaxLoadTask(child) {
- return function () {
- removeClassFromElement(child, htmx.config.addedClass);
- processNode(child);
- processScripts(child);
- processFocus(child)
- triggerEvent(child, 'htmx:load');
- };
- }
-
- function processFocus(child) {
- var autofocus = "[autofocus]";
- var autoFocusedElt = matches(child, autofocus) ? child : child.querySelector(autofocus)
- if (autoFocusedElt != null) {
- autoFocusedElt.focus();
- }
- }
-
- function insertNodesBefore(parentNode, insertBefore, fragment, settleInfo) {
- handleAttributes(parentNode, fragment, settleInfo);
- while(fragment.childNodes.length > 0){
- var child = fragment.firstChild;
- addClassToElement(child, htmx.config.addedClass);
- parentNode.insertBefore(child, insertBefore);
- if (child.nodeType !== Node.TEXT_NODE && child.nodeType !== Node.COMMENT_NODE) {
- settleInfo.tasks.push(makeAjaxLoadTask(child));
- }
- }
- }
-
- // based on https://gist.github.com/hyamamoto/fd435505d29ebfa3d9716fd2be8d42f0,
- // derived from Java's string hashcode implementation
- function stringHash(string, hash) {
- var char = 0;
- while (char < string.length){
- hash = (hash << 5) - hash + string.charCodeAt(char++) | 0; // bitwise or ensures we have a 32-bit int
- }
- return hash;
- }
-
- function attributeHash(elt) {
- var hash = 0;
- // IE fix
- if (elt.attributes) {
- for (var i = 0; i < elt.attributes.length; i++) {
- var attribute = elt.attributes[i];
- if(attribute.value){ // only include attributes w/ actual values (empty is same as non-existent)
- hash = stringHash(attribute.name, hash);
- hash = stringHash(attribute.value, hash);
- }
- }
- }
- return hash;
- }
-
- function deInitNode(element) {
- var internalData = getInternalData(element);
- if (internalData.timeout) {
- clearTimeout(internalData.timeout);
- }
- if (internalData.webSocket) {
- internalData.webSocket.close();
- }
- if (internalData.sseEventSource) {
- internalData.sseEventSource.close();
- }
- if (internalData.listenerInfos) {
- forEach(internalData.listenerInfos, function (info) {
- if (info.on) {
- info.on.removeEventListener(info.trigger, info.listener);
- }
- });
- }
- if (internalData.onHandlers) {
- for (var eventName of internalData.onHandlers) {
- element.removeEventListener(eventName, internalData.onHandlers[eventName]);
- }
- }
- }
-
- function cleanUpElement(element) {
- triggerEvent(element, "htmx:beforeCleanupElement")
- deInitNode(element);
- if (element.children) { // IE
- forEach(element.children, function(child) { cleanUpElement(child) });
- }
- }
-
- function swapOuterHTML(target, fragment, settleInfo) {
- if (target.tagName === "BODY") {
- return swapInnerHTML(target, fragment, settleInfo);
- } else {
- // @type {HTMLElement}
- var newElt
- var eltBeforeNewContent = target.previousSibling;
- insertNodesBefore(parentElt(target), target, fragment, settleInfo);
- if (eltBeforeNewContent == null) {
- newElt = parentElt(target).firstChild;
- } else {
- newElt = eltBeforeNewContent.nextSibling;
- }
- getInternalData(target).replacedWith = newElt; // tuck away so we can fire events on it later
- settleInfo.elts = [] // clear existing elements
- while(newElt && newElt !== target) {
- if (newElt.nodeType === Node.ELEMENT_NODE) {
- settleInfo.elts.push(newElt);
- }
- newElt = newElt.nextElementSibling;
- }
- cleanUpElement(target);
- parentElt(target).removeChild(target);
- }
- }
-
- function swapAfterBegin(target, fragment, settleInfo) {
- return insertNodesBefore(target, target.firstChild, fragment, settleInfo);
- }
-
- function swapBeforeBegin(target, fragment, settleInfo) {
- return insertNodesBefore(parentElt(target), target, fragment, settleInfo);
- }
-
- function swapBeforeEnd(target, fragment, settleInfo) {
- return insertNodesBefore(target, null, fragment, settleInfo);
- }
-
- function swapAfterEnd(target, fragment, settleInfo) {
- return insertNodesBefore(parentElt(target), target.nextSibling, fragment, settleInfo);
- }
- function swapDelete(target, fragment, settleInfo) {
- cleanUpElement(target);
- return parentElt(target).removeChild(target);
- }
-
- function swapInnerHTML(target, fragment, settleInfo) {
- var firstChild = target.firstChild;
- insertNodesBefore(target, firstChild, fragment, settleInfo);
- if (firstChild) {
- while (firstChild.nextSibling) {
- cleanUpElement(firstChild.nextSibling)
- target.removeChild(firstChild.nextSibling);
- }
- cleanUpElement(firstChild)
- target.removeChild(firstChild);
- }
- }
-
- function maybeSelectFromResponse(elt, fragment) {
- var selector = getClosestAttributeValue(elt, "hx-select");
- if (selector) {
- var newFragment = getDocument().createDocumentFragment();
- forEach(fragment.querySelectorAll(selector), function (node) {
- newFragment.appendChild(node);
- });
- fragment = newFragment;
- }
- return fragment;
- }
-
- function swap(swapStyle, elt, target, fragment, settleInfo) {
- switch (swapStyle) {
- case "none":
- return;
- case "outerHTML":
- swapOuterHTML(target, fragment, settleInfo);
- return;
- case "afterbegin":
- swapAfterBegin(target, fragment, settleInfo);
- return;
- case "beforebegin":
- swapBeforeBegin(target, fragment, settleInfo);
- return;
- case "beforeend":
- swapBeforeEnd(target, fragment, settleInfo);
- return;
- case "afterend":
- swapAfterEnd(target, fragment, settleInfo);
- return;
- case "delete":
- swapDelete(target, fragment, settleInfo);
- return;
- default:
- var extensions = getExtensions(elt);
- for (var i = 0; i < extensions.length; i++) {
- var ext = extensions[i];
- try {
- var newElements = ext.handleSwap(swapStyle, target, fragment, settleInfo);
- if (newElements) {
- if (typeof newElements.length !== 'undefined') {
- // if handleSwap returns an array (like) of elements, we handle them
- for (var j = 0; j < newElements.length; j++) {
- var child = newElements[j];
- if (child.nodeType !== Node.TEXT_NODE && child.nodeType !== Node.COMMENT_NODE) {
- settleInfo.tasks.push(makeAjaxLoadTask(child));
- }
- }
- }
- return;
- }
- } catch (e) {
- logError(e);
- }
- }
- if (swapStyle === "innerHTML") {
- swapInnerHTML(target, fragment, settleInfo);
- } else {
- swap(htmx.config.defaultSwapStyle, elt, target, fragment, settleInfo);
- }
- }
- }
-
- function findTitle(content) {
- if (content.indexOf(' -1) {
- var contentWithSvgsRemoved = content.replace(/