121 lines
3.1 KiB
JavaScript
121 lines
3.1 KiB
JavaScript
/**
|
|
* template.js — Custom JavaScript for the Moko Cassiopeia Joomla template
|
|
*
|
|
* @package Joomla.Site
|
|
* @subpackage Templates.Moko-Cassiopeia
|
|
* @file /media/templates/site/moko-cassiopeia/js/template.js
|
|
* @version 2.0
|
|
*
|
|
* @copyright (C) 2025 Moko Consulting
|
|
* @author Jonathan Miller
|
|
* @website https://mokoconsulting.tech
|
|
* @email hello@mokoconsulting.tech
|
|
* @phone +1 (931) 279-6313
|
|
*
|
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
|
*
|
|
* This file is part of a Moko Consulting project.
|
|
*
|
|
* This program is free software: you can redistribute it and/or modify it
|
|
* under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or (at
|
|
* your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
(function (win, doc) {
|
|
"use strict";
|
|
|
|
/**
|
|
* Utility: smooth scroll to top
|
|
*/
|
|
function backToTop() {
|
|
win.scrollTo({ top: 0, behavior: "smooth" });
|
|
}
|
|
|
|
/**
|
|
* Utility: toggle body class on scroll for sticky header styling
|
|
*/
|
|
function handleScroll() {
|
|
if (win.scrollY > 50) {
|
|
doc.body.classList.add("scrolled");
|
|
} else {
|
|
doc.body.classList.remove("scrolled");
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Initialize Bootstrap TOC if #toc element exists.
|
|
* Requires bootstrap-toc.min.js to be loaded.
|
|
*/
|
|
function initTOC() {
|
|
if (typeof win.Toc === "function" && doc.querySelector("#toc")) {
|
|
win.Toc.init({
|
|
$nav: $("#toc"),
|
|
$scope: $("main")
|
|
});
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Initialize offcanvas drawer buttons for left/right drawers.
|
|
* Uses Bootstrap's offcanvas component.
|
|
*/
|
|
function initDrawers() {
|
|
var leftBtn = doc.querySelector(".drawer-toggle-left");
|
|
var rightBtn = doc.querySelector(".drawer-toggle-right");
|
|
if (leftBtn) {
|
|
leftBtn.addEventListener("click", function () {
|
|
var target = doc.querySelector(leftBtn.getAttribute("data-bs-target"));
|
|
if (target) new bootstrap.Offcanvas(target).show();
|
|
});
|
|
}
|
|
if (rightBtn) {
|
|
rightBtn.addEventListener("click", function () {
|
|
var target = doc.querySelector(rightBtn.getAttribute("data-bs-target"));
|
|
if (target) new bootstrap.Offcanvas(target).show();
|
|
});
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Initialize back-to-top link if present
|
|
*/
|
|
function initBackTop() {
|
|
var backTop = doc.getElementById("back-top");
|
|
if (backTop) {
|
|
backTop.addEventListener("click", function (e) {
|
|
e.preventDefault();
|
|
backToTop();
|
|
});
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Run all template JS initializations
|
|
*/
|
|
function init() {
|
|
// Sticky header behavior
|
|
handleScroll();
|
|
win.addEventListener("scroll", handleScroll);
|
|
|
|
// Init features
|
|
initTOC();
|
|
initDrawers();
|
|
initBackTop();
|
|
}
|
|
|
|
if (doc.readyState === "loading") {
|
|
doc.addEventListener("DOMContentLoaded", init);
|
|
} else {
|
|
init();
|
|
}
|
|
})(window, document);
|