Files

331 lines
10 KiB
JavaScript

(function () {
var currentArtId = null;
var currentImageIdx = null;
var currentDocId = null;
var mainContent = document.getElementById('main-content');
var infoOverlay = document.getElementById('info-overlay');
var infoContent = document.getElementById('info-overlay-content');
var closeInfo = document.getElementById('close-info');
var prevArt = document.getElementById('prev-art');
var nextArt = document.getElementById('next-art');
var imageOverlay = document.getElementById('image-overlay');
var fullscreenImage = document.getElementById('fullscreen-image');
var navHome = document.getElementById('nav-home');
var navDocs = document.getElementById('nav-docs');
var navContact = document.getElementById('nav-contact');
var homeLink = document.getElementById('home-link');
function getArtIndex(id) {
for (var i = 0; i < DATA.arts.length; i++) {
if (DATA.arts[i].id === id) return i;
}
return -1;
}
function getArtById(id) {
for (var i = 0; i < DATA.arts.length; i++) {
if (DATA.arts[i].id === id) return DATA.arts[i];
}
return null;
}
function getDocBySlug(slug) {
for (var i = 0; i < DATA.docFiles.length; i++) {
if (DATA.docFiles[i].id === slug) return DATA.docFiles[i];
}
return null;
}
function formatParagraphs(text) {
var parts = text.split('\n\n');
var result = '';
for (var i = 0; i < parts.length; i++) {
var p = parts[i].trim();
if (p) {
result += '<p>' + p.replace(/\n/g, '<br>') + '</p>';
}
}
return result;
}
function updateUrl() {
var params = new URLSearchParams();
if (currentArtId) {
params.set('view', currentArtId);
if (currentImageIdx !== null) {
params.set('img', currentImageIdx);
}
} else if (currentDocId) {
params.set('view', 'docs');
params.set('doc', currentDocId);
} else if (mainContent.querySelector('#contact-view')) {
params.set('view', 'contact');
}
var qs = params.toString();
var url = qs ? '?' + qs : window.location.pathname;
window.history.replaceState(null, '', url);
}
function parseUrl() {
var params = new URLSearchParams(window.location.search);
return {
view: params.get('view'),
img: params.get('img') !== null ? parseInt(params.get('img'), 10) : null,
doc: params.get('doc')
};
}
function renderMainPage() {
var html = '<section id="overview">' + formatParagraphs(DATA.overview) + '</section>';
html += '<section id="gallery"><div class="section-header" style="border-bottom:1px solid #222;padding-bottom:0.5rem;margin-bottom:1.5rem;color:#666;font-size:0.85rem;">ARTWORK</div><div class="art-grid">';
for (var i = 0; i < DATA.arts.length; i++) {
var art = DATA.arts[i];
html += '<div class="art-card" data-art-id="' + art.id + '">' +
'<img src="' + art.images[0] + '" alt="' + art.title.replace(/"/g, '&quot;') + '" loading="lazy">' +
'<div class="art-title">' + art.title + '</div></div>';
}
html += '</div></section>';
mainContent.innerHTML = html;
var cards = document.querySelectorAll('.art-card');
for (var i = 0; i < cards.length; i++) {
cards[i].addEventListener('click', function () {
openInfoOverlay(this.dataset.artId);
});
}
}
function renderInfoOverlay(artId) {
var art = getArtById(artId);
if (!art) return;
var isMulti = art.images.length > 1;
var imagesHtml = '';
for (var i = 0; i < art.images.length; i++) {
imagesHtml += '<img src="' + art.images[i] + '" alt="' + art.title.replace(/"/g, '&quot;') + ' ' + (i + 1) + '" data-img-index="' + i + '">';
}
var html = '<h2>' + art.title + '</h2>' +
'<div class="overlay-image-container' + (isMulti ? ' multi' : '') + '">' +
imagesHtml + '</div>' +
'<div class="description">' + formatParagraphs(art.description) + '</div>';
infoContent.innerHTML = html;
var imgs = infoContent.querySelectorAll('.overlay-image-container img');
for (var i = 0; i < imgs.length; i++) {
imgs[i].addEventListener('click', function (e) {
e.stopPropagation();
openImageOverlay(artId, parseInt(this.dataset.imgIndex, 10));
});
}
var idx = getArtIndex(artId);
prevArt.style.display = idx > 0 ? 'block' : 'none';
nextArt.style.display = idx < DATA.arts.length - 1 ? 'block' : 'none';
}
function renderDocsView(docId) {
currentDocId = docId || null;
var html = '<div id="docs-view">' +
'<div class="docs-tree">' +
'<div class="folder-label">docs/</div><ul>';
for (var i = 0; i < DATA.docFiles.length; i++) {
var doc = DATA.docFiles[i];
var activeClass = (doc.id === docId) ? ' active' : '';
html += '<li data-doc-id="' + doc.id + '" class="' + activeClass + '">' +
doc.filename + '<span class="type-badge">' + (doc.type || '?') + '</span></li>';
}
html += '</ul></div>' +
'<div class="docs-content">' +
'<pre id="docs-content-pre">';
if (docId) {
var found = getDocBySlug(docId);
html += found ? found.raw.replace(/</g, '&lt;').replace(/>/g, '&gt;') : 'File not found';
} else {
html += 'Select a file to view';
}
html += '</pre></div></div>';
mainContent.innerHTML = html;
var items = document.querySelectorAll('.docs-tree li');
for (var i = 0; i < items.length; i++) {
items[i].addEventListener('click', function () {
var active = document.querySelectorAll('.docs-tree li.active');
for (var j = 0; j < active.length; j++) {
active[j].classList.remove('active');
}
this.classList.add('active');
var slug = this.dataset.docId;
currentDocId = slug;
updateUrl();
var found = getDocBySlug(slug);
document.getElementById('docs-content-pre').textContent = found ? found.raw : 'File not found';
});
}
}
function renderContactView() {
var html = '<div id="contact-view">' +
'<div class="section-header" style="border-bottom:1px solid #222;padding-bottom:0.5rem;margin-bottom:2rem;color:#666;font-size:0.85rem;">CONTACT</div>' +
'<div class="contact-list">' +
'<div class="contact-item"><span class="contact-label">email</span><a href="mailto:jonmcc.0723@gmail.com">jonmcc.0723@gmail.com</a></div>' +
'<div class="contact-item"><span class="contact-label">linkedIn</span><a href="https://www.linkedin.com/in/jonmcc/" target="_blank" rel="noopener">Jonathan McCaffrey | LinkedIn</a></div>' +
'<div class="contact-item"><span class="contact-label">git</span><a href="https://git.jonathanmccaffrey.ca/JonathanMcCaffrey" target="_blank" rel="noopener">Jonathan McCaffrey - Gitea: Git with a cup of tea</a></div>' +
'<div class="contact-item"><span class="contact-label">website</span><a href="https://jonathanmccaffrey.ca/" target="_blank" rel="noopener">Ottawa .NET Developer | Jonathan McCaffrey</a></div>' +
'</div></div>';
mainContent.innerHTML = html;
}
function openInfoOverlay(artId, imgIdx) {
currentArtId = artId;
currentImageIdx = (imgIdx !== undefined && imgIdx !== null) ? imgIdx : null;
currentDocId = null;
renderInfoOverlay(artId);
infoOverlay.classList.remove('hidden');
if (currentImageIdx !== null) {
openImageOverlay(artId, currentImageIdx);
} else {
imageOverlay.classList.add('hidden');
}
updateUrl();
}
function closeInfoOverlay() {
currentArtId = null;
currentImageIdx = null;
infoOverlay.classList.add('hidden');
imageOverlay.classList.add('hidden');
updateUrl();
}
function openImageOverlay(artId, idx) {
var art = getArtById(artId);
if (!art || !art.images[idx]) return;
fullscreenImage.src = art.images[idx];
imageOverlay.classList.remove('hidden');
currentImageIdx = idx;
updateUrl();
}
function closeImageOverlay() {
imageOverlay.classList.add('hidden');
currentImageIdx = null;
updateUrl();
}
function navigateArt(direction) {
var idx = getArtIndex(currentArtId);
if (idx === -1) return;
var newIdx = idx + direction;
if (newIdx < 0 || newIdx >= DATA.arts.length) return;
openInfoOverlay(DATA.arts[newIdx].id);
}
function goHome() {
closeInfoOverlay();
currentArtId = null;
currentImageIdx = null;
currentDocId = null;
renderMainPage();
updateUrl();
}
function goDocs(docId) {
closeInfoOverlay();
currentArtId = null;
currentImageIdx = null;
renderDocsView(docId || null);
updateUrl();
}
function goContact() {
closeInfoOverlay();
currentArtId = null;
currentImageIdx = null;
currentDocId = null;
renderContactView();
updateUrl();
}
closeInfo.addEventListener('click', closeInfoOverlay);
prevArt.addEventListener('click', function () { navigateArt(-1); });
nextArt.addEventListener('click', function () { navigateArt(1); });
imageOverlay.addEventListener('click', function (e) {
if (e.target === imageOverlay || e.target === fullscreenImage) {
closeImageOverlay();
}
});
navHome.addEventListener('click', function (e) {
e.preventDefault();
goHome();
});
navDocs.addEventListener('click', function (e) {
e.preventDefault();
goDocs();
});
navContact.addEventListener('click', function (e) {
e.preventDefault();
goContact();
});
homeLink.addEventListener('click', function (e) {
if (!infoOverlay.classList.contains('hidden') ||
mainContent.querySelector('#docs-view') ||
mainContent.querySelector('#contact-view')) {
e.preventDefault();
goHome();
}
});
document.addEventListener('keydown', function (e) {
if (!infoOverlay.classList.contains('hidden')) {
if (e.key === 'Escape') {
if (!imageOverlay.classList.contains('hidden')) {
closeImageOverlay();
} else {
closeInfoOverlay();
}
} else if (e.key === 'ArrowLeft') {
navigateArt(-1);
} else if (e.key === 'ArrowRight') {
navigateArt(1);
}
}
});
function init() {
var params = parseUrl();
if (params.view === 'contact') {
renderContactView();
} else if (params.view === 'docs') {
renderDocsView(params.doc || null);
} else if (params.view && getArtById(params.view)) {
renderMainPage();
openInfoOverlay(params.view, params.img);
} else {
renderMainPage();
}
}
init();
})();