mirror of
https://github.com/DavidHDev/vue-bits.git
synced 2026-03-07 14:39:30 -07:00
Added <CardNav /> Component
This commit is contained in:
257
src/content/Components/CardNav/CardNav.vue
Normal file
257
src/content/Components/CardNav/CardNav.vue
Normal file
@@ -0,0 +1,257 @@
|
||||
<script setup lang="ts">
|
||||
import { gsap } from 'gsap';
|
||||
import { nextTick, onBeforeUpdate, onMounted, onUnmounted, ref, watch, type VNodeRef } from 'vue';
|
||||
|
||||
type CardNavLink = {
|
||||
label: string;
|
||||
href?: string;
|
||||
ariaLabel: string;
|
||||
};
|
||||
|
||||
export type CardNavItem = {
|
||||
label: string;
|
||||
bgColor: string;
|
||||
textColor: string;
|
||||
links: CardNavLink[];
|
||||
};
|
||||
|
||||
export interface CardNavProps {
|
||||
logo: string;
|
||||
logoAlt?: string;
|
||||
items: CardNavItem[];
|
||||
className?: string;
|
||||
ease?: string;
|
||||
baseColor?: string;
|
||||
menuColor?: string;
|
||||
buttonBgColor?: string;
|
||||
buttonTextColor?: string;
|
||||
}
|
||||
|
||||
const props = withDefaults(defineProps<CardNavProps>(), {
|
||||
logoAlt: 'Logo',
|
||||
className: '',
|
||||
ease: 'power3.out',
|
||||
baseColor: '#fff'
|
||||
});
|
||||
|
||||
const isHamburgerOpen = ref(false);
|
||||
const isExpanded = ref(false);
|
||||
|
||||
const navRef = ref<HTMLDivElement | null>(null);
|
||||
const cardsRef = ref<HTMLDivElement[]>([]);
|
||||
const tlRef = ref<gsap.core.Timeline | null>(null);
|
||||
|
||||
const setCardRef =
|
||||
(i: number): VNodeRef =>
|
||||
el => {
|
||||
if (el && el instanceof HTMLDivElement) {
|
||||
cardsRef.value[i] = el;
|
||||
}
|
||||
};
|
||||
|
||||
onBeforeUpdate(() => {
|
||||
cardsRef.value = [];
|
||||
});
|
||||
|
||||
const calculateHeight = () => {
|
||||
const navEl = navRef.value;
|
||||
if (!navEl) return 260;
|
||||
|
||||
const isMobile = window.matchMedia('(max-width: 768px)').matches;
|
||||
if (isMobile) {
|
||||
const contentEl = navEl.querySelector('.card-nav-content') as HTMLElement;
|
||||
if (contentEl) {
|
||||
const wasVisible = contentEl.style.visibility;
|
||||
const wasPosition = contentEl.style.position;
|
||||
const wasHeight = contentEl.style.height;
|
||||
|
||||
contentEl.style.visibility = 'visible';
|
||||
contentEl.style.position = 'static';
|
||||
contentEl.style.height = 'auto';
|
||||
|
||||
const topBar = 60;
|
||||
const padding = 16;
|
||||
const contentHeight = contentEl.scrollHeight;
|
||||
|
||||
contentEl.style.visibility = wasVisible;
|
||||
contentEl.style.position = wasPosition;
|
||||
contentEl.style.height = wasHeight;
|
||||
|
||||
return topBar + contentHeight + padding;
|
||||
}
|
||||
}
|
||||
return 260;
|
||||
};
|
||||
|
||||
const createTimeline = () => {
|
||||
const navEl = navRef.value;
|
||||
if (!navEl) return null;
|
||||
|
||||
gsap.set(navEl, { height: 60, overflow: 'hidden' });
|
||||
gsap.set(cardsRef.value, { y: 50, opacity: 0 });
|
||||
|
||||
const tl = gsap.timeline({ paused: true });
|
||||
|
||||
tl.to(navEl, {
|
||||
height: calculateHeight,
|
||||
duration: 0.4,
|
||||
ease: props.ease
|
||||
});
|
||||
|
||||
tl.to(cardsRef.value, { y: 0, opacity: 1, duration: 0.4, ease: props.ease, stagger: 0.08 }, '-=0.1');
|
||||
|
||||
return tl;
|
||||
};
|
||||
|
||||
const toggleMenu = () => {
|
||||
const tl = tlRef.value;
|
||||
if (!tl) return;
|
||||
if (!isExpanded.value) {
|
||||
isHamburgerOpen.value = true;
|
||||
isExpanded.value = true;
|
||||
nextTick(() => {
|
||||
tl.play(0);
|
||||
});
|
||||
} else {
|
||||
isHamburgerOpen.value = false;
|
||||
tl.eventCallback('onReverseComplete', () => {
|
||||
isExpanded.value = false;
|
||||
tl.eventCallback('onReverseComplete', null);
|
||||
});
|
||||
tl.reverse();
|
||||
}
|
||||
};
|
||||
|
||||
const handleResize = () => {
|
||||
if (!tlRef.value) return;
|
||||
|
||||
if (isExpanded.value) {
|
||||
const newHeight = calculateHeight();
|
||||
gsap.set(navRef.value, { height: newHeight });
|
||||
|
||||
tlRef.value.kill();
|
||||
const newTl = createTimeline();
|
||||
if (newTl) {
|
||||
newTl.progress(1);
|
||||
tlRef.value = newTl;
|
||||
}
|
||||
} else {
|
||||
tlRef.value.kill();
|
||||
tlRef.value = createTimeline();
|
||||
}
|
||||
};
|
||||
|
||||
onMounted(() => {
|
||||
tlRef.value = createTimeline();
|
||||
window.addEventListener('resize', handleResize);
|
||||
});
|
||||
|
||||
onUnmounted(() => {
|
||||
tlRef.value?.kill();
|
||||
tlRef.value = null;
|
||||
window.removeEventListener('resize', handleResize);
|
||||
});
|
||||
|
||||
watch(
|
||||
() => [props.ease, props.items],
|
||||
() => {
|
||||
nextTick(() => {
|
||||
if (tlRef.value) tlRef.value.kill();
|
||||
tlRef.value = createTimeline();
|
||||
});
|
||||
}
|
||||
);
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<div
|
||||
:class="`card-nav-container absolute left-1/2 -translate-x-1/2 w-[90%] max-w-[800px] z-[99] top-[1.2em] md:top-[2em] ${props.className}`"
|
||||
>
|
||||
<nav
|
||||
ref="navRef"
|
||||
:class="[
|
||||
'card-nav block h-[60px] p-0 rounded-xl shadow-md relative overflow-hidden will-change-[height]',
|
||||
{ open: isExpanded }
|
||||
]"
|
||||
:style="{ backgroundColor: props.baseColor }"
|
||||
>
|
||||
<div
|
||||
class="card-nav-top top-0 z-[2] absolute inset-x-0 flex justify-between items-center p-2 px-[1.1rem] h-[60px]"
|
||||
>
|
||||
<div
|
||||
:class="[
|
||||
'hamburger-menu group h-full flex flex-col items-center justify-center cursor-pointer gap-[6px] order-2 md:order-none',
|
||||
{ open: isHamburgerOpen }
|
||||
]"
|
||||
@click="toggleMenu"
|
||||
role="button"
|
||||
:aria-label="isExpanded ? 'Close menu' : 'Open menu'"
|
||||
tabindex="0"
|
||||
:style="{ color: props.menuColor || '#000' }"
|
||||
>
|
||||
<div
|
||||
:class="[
|
||||
'hamburger-line w-[30px] h-[2px] bg-current transition-[transform,opacity,margin] duration-300 ease-linear [transform-origin:50%_50%] group-hover:opacity-75',
|
||||
{ 'translate-y-[4px] rotate-45': isHamburgerOpen }
|
||||
]"
|
||||
/>
|
||||
<div
|
||||
:class="[
|
||||
'hamburger-line w-[30px] h-[2px] bg-current transition-[transform,opacity,margin] duration-300 ease-linear [transform-origin:50%_50%] group-hover:opacity-75',
|
||||
{ '-translate-y-[4px] -rotate-45': isHamburgerOpen }
|
||||
]"
|
||||
/>
|
||||
</div>
|
||||
|
||||
<div
|
||||
class="md:top-1/2 md:left-1/2 md:absolute flex items-center order-1 md:order-none md:-translate-x-1/2 md:-translate-y-1/2 logo-container"
|
||||
>
|
||||
<img :src="props.logo" :alt="props.logoAlt" class="h-[28px] logo" />
|
||||
</div>
|
||||
|
||||
<button
|
||||
type="button"
|
||||
class="hidden md:inline-flex px-4 py-2 border-0 rounded-[calc(0.75rem-0.2rem)] h-full font-medium transition-colors duration-300 cursor-pointer card-nav-cta-button"
|
||||
:style="{
|
||||
backgroundColor: props.buttonBgColor,
|
||||
color: props.buttonTextColor
|
||||
}"
|
||||
>
|
||||
Get Started
|
||||
</button>
|
||||
</div>
|
||||
|
||||
<div
|
||||
:class="[
|
||||
'card-nav-content absolute left-0 right-0 top-[60px] bottom-0 p-2 flex flex-col items-stretch gap-2 justify-start z-[1] md:flex-row md:items-end md:gap-[12px]',
|
||||
isExpanded ? 'visible pointer-events-auto' : 'invisible pointer-events-none'
|
||||
]"
|
||||
:aria-hidden="!isExpanded"
|
||||
>
|
||||
<div
|
||||
v-for="(item, idx) in (props.items || []).slice(0, 3)"
|
||||
:key="`${item.label}-${idx}`"
|
||||
:ref="setCardRef(idx)"
|
||||
class="relative flex flex-col flex-[1_1_auto] md:flex-[1_1_0%] gap-2 p-[12px_16px] rounded-[calc(0.75rem-0.2rem)] min-w-0 h-auto md:h-full min-h-[60px] md:min-h-0 select-none nav-card"
|
||||
:style="{ backgroundColor: item.bgColor, color: item.textColor }"
|
||||
>
|
||||
<div class="font-normal text-[18px] md:text-[22px] tracking-[-0.5px] nav-card-label">
|
||||
{{ item.label }}
|
||||
</div>
|
||||
<div class="flex flex-col gap-[2px] mt-auto nav-card-links">
|
||||
<a
|
||||
v-for="(lnk, i) in item.links"
|
||||
:key="`${lnk.label}-${i}`"
|
||||
class="inline-flex items-center gap-[6px] hover:opacity-75 text-[15px] md:text-[16px] no-underline transition-opacity duration-300 cursor-pointer nav-card-link"
|
||||
:href="lnk.href"
|
||||
:aria-label="lnk.ariaLabel"
|
||||
>
|
||||
<v-icon name="go-arrow-up-right" class="nav-card-link-icon shrink-0" aria-hidden="true" />
|
||||
{{ lnk.label }}
|
||||
</a>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</nav>
|
||||
</div>
|
||||
</template>
|
||||
Reference in New Issue
Block a user