Added navigation bar to tags page.
This commit is contained in:
parent
e79e34ac29
commit
fbcae6b7d1
|
@ -1,12 +1,38 @@
|
||||||
<template>
|
<template>
|
||||||
<div class="page">
|
<div class="page">
|
||||||
<div
|
<ul
|
||||||
v-for="(tags, category) in categories"
|
ref="categories"
|
||||||
|
class="categories nolist"
|
||||||
|
>
|
||||||
|
<li
|
||||||
|
v-for="(tags, category) in showcase"
|
||||||
:key="`category-${category}`"
|
:key="`category-${category}`"
|
||||||
>
|
>
|
||||||
<h3 class="category">{{ category }}</h3>
|
<a
|
||||||
|
:href="`#${category}`"
|
||||||
|
class="category nolink"
|
||||||
|
:class="{ active: activeCategory === category }"
|
||||||
|
>{{ category }}</a>
|
||||||
|
</li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
<ul class="tags nolist">
|
<div
|
||||||
|
ref="content"
|
||||||
|
class="content"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
v-for="(tags, category) in showcase"
|
||||||
|
:key="`tags-${category}`"
|
||||||
|
>
|
||||||
|
<h3
|
||||||
|
:id="category"
|
||||||
|
class="category-heading"
|
||||||
|
>{{ category }}</h3>
|
||||||
|
|
||||||
|
<ul
|
||||||
|
class="tags nolist"
|
||||||
|
:data-category="category"
|
||||||
|
>
|
||||||
<li
|
<li
|
||||||
v-for="tag in tags"
|
v-for="tag in tags"
|
||||||
:key="`tag-${tag.slug}`"
|
:key="`tag-${tag.slug}`"
|
||||||
|
@ -48,16 +74,99 @@
|
||||||
</ul>
|
</ul>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script setup>
|
<script setup>
|
||||||
import { inject } from 'vue';
|
import { ref, onMounted, inject } from 'vue';
|
||||||
|
import navigate from '#/src/navigate.js';
|
||||||
|
|
||||||
const pageContext = inject('pageContext');
|
const pageContext = inject('pageContext');
|
||||||
const categories = pageContext.pageProps.tagShowcase;
|
const showcase = pageContext.pageProps.tagShowcase;
|
||||||
|
|
||||||
|
const categories = ref(null);
|
||||||
|
const content = ref(null);
|
||||||
|
|
||||||
|
const activeCategory = ref(null);
|
||||||
|
|
||||||
|
function calculateActiveCategory() {
|
||||||
|
activeCategory.value = Array.from(document.querySelectorAll('.tags')).reduce((closest, element) => {
|
||||||
|
const { top } = element.getBoundingClientRect();
|
||||||
|
|
||||||
|
if (!closest || Math.abs(top) < Math.abs(closest.top)) {
|
||||||
|
return { category: element.dataset.category, top };
|
||||||
|
}
|
||||||
|
|
||||||
|
return closest;
|
||||||
|
}, null).category;
|
||||||
|
|
||||||
|
const activeLink = document.querySelector(`a[href="#${activeCategory.value}"]`);
|
||||||
|
|
||||||
|
activeLink.scrollIntoView();
|
||||||
|
|
||||||
|
navigate(`#${activeCategory.value}`, null, { replace: true });
|
||||||
|
}
|
||||||
|
|
||||||
|
onMounted(() => {
|
||||||
|
// div doesn't scroll automatically on page load, reset hash to scroll
|
||||||
|
if (window.location.hash) {
|
||||||
|
const hash = window.location.hash;
|
||||||
|
|
||||||
|
window.location.hash = undefined;
|
||||||
|
window.location.hash = hash;
|
||||||
|
}
|
||||||
|
|
||||||
|
categories.value.addEventListener('wheel', (event) => {
|
||||||
|
categories.value.scrollLeft += event.deltaY;
|
||||||
|
});
|
||||||
|
|
||||||
|
content.value.addEventListener('scroll', calculateActiveCategory);
|
||||||
|
|
||||||
|
calculateActiveCategory();
|
||||||
|
});
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style scoped>
|
<style scoped>
|
||||||
|
.page {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
.content {
|
||||||
|
flex-grow: 1;
|
||||||
|
overflow-y: auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
.categories {
|
||||||
|
display: flex;
|
||||||
|
gap: .25rem;
|
||||||
|
flex-shrink: 0;
|
||||||
|
padding: .5rem 1rem;
|
||||||
|
background: var(--grey-dark-40);
|
||||||
|
overflow-x: auto;
|
||||||
|
scrollbar-width: none;
|
||||||
|
|
||||||
|
&::-webkit-scrollbar {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.category {
|
||||||
|
padding: .25rem .5rem;
|
||||||
|
border-radius: .25rem;
|
||||||
|
background: var(--grey-dark-30);
|
||||||
|
color: var(--highlight-strong-30);
|
||||||
|
font-weight: bold;
|
||||||
|
font-size: .9rem;
|
||||||
|
|
||||||
|
&:hover,
|
||||||
|
&.active {
|
||||||
|
color: var(--text-light);
|
||||||
|
background: var(--primary);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
.tags {
|
.tags {
|
||||||
display: grid;
|
display: grid;
|
||||||
grid-template-columns: repeat(auto-fill, minmax(20rem, 1fr));
|
grid-template-columns: repeat(auto-fill, minmax(20rem, 1fr));
|
||||||
|
@ -65,7 +174,7 @@ const categories = pageContext.pageProps.tagShowcase;
|
||||||
padding: .75rem 1rem;
|
padding: .75rem 1rem;
|
||||||
}
|
}
|
||||||
|
|
||||||
.category {
|
.category-heading {
|
||||||
padding: 1rem 1rem 0 1.5rem;
|
padding: 1rem 1rem 0 1.5rem;
|
||||||
margin: 0;
|
margin: 0;
|
||||||
text-transform: capitalize;
|
text-transform: capitalize;
|
||||||
|
|
|
@ -12,6 +12,8 @@ export default function navigate(path, query, options = {}) {
|
||||||
|
|
||||||
if (options.redirect) {
|
if (options.redirect) {
|
||||||
window.location.href = url;
|
window.location.href = url;
|
||||||
|
} else if (options.replace) {
|
||||||
|
history.replaceState({}, '', url); // eslint-disable-line no-restricted-globals
|
||||||
} else {
|
} else {
|
||||||
history.pushState({}, '', url); // eslint-disable-line no-restricted-globals
|
history.pushState({}, '', url); // eslint-disable-line no-restricted-globals
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue