Files
portfolio/app/components/portfolio/Hero.vue
2025-11-04 09:09:29 +03:30

190 lines
8.8 KiB
Vue

<template>
<section id="hero" class="pt-8 pb-6 scroll-mt-20 flex items-center">
<UContainer>
<div class="flex flex-col-reverse sm:flex-row items-center sm:items-center justify-center gap-4 sm:gap-8">
<div class="flex-1 max-w-2xl text-center sm:text-start">
<h1 class="text-primary text-3xl sm:text-4xl font-bold tracking-tight mb-3 sm:mb-4">
{{ portfolio.profile.name }}
</h1>
<p class="text-base sm:text-lg text-gray-600 dark:text-gray-300 mb-4 sm:mb-3">
{{ portfolio.profile.summary }}
</p>
<div v-if="portfolio.profile.location || currentRole"
class="mb-6 flex flex-col items-center gap-2 text-sm text-gray-600 dark:text-gray-300 sm:items-start">
<div v-if="portfolio.profile.location" class="flex items-center gap-2">
<UIcon name="i-twemoji-round-pushpin" class="text-base text-primary-600 me-1 dark:text-primary-300" />
<span class="leading-relaxed">{{ portfolio.profile.location }}</span>
</div>
<div v-if="currentRole" class="flex items-center gap-1 text-base text-gray-700 dark:text-gray-200">
<img v-if="currentRole.companyLogo" :src="currentRole.companyLogo" :alt="`${currentRole.company} logo`"
class="h-7 w-7 rounded-md object-contain" loading="lazy" />
<span class="">{{ t('hero.currently') }}</span>
<span class="font-semibold text-primary-600 dark:text-primary-300">
<a v-if="currentRole.companyLink" :href="currentRole.companyLink" target="_blank" rel="noopener"
class="hover:underline text-primary-600 dark:text-primary-300">
{{ currentRole.company }}
</a>
<span v-else>{{ currentRole.company }}</span>
</span>
</div>
</div>
<div class="flex flex-wrap items-center justify-center gap-3 sm:justify-start">
<ClientTooltip :text="emailTooltip">
<UButton icon="i-twemoji-e-mail" :square="true" color="gray" variant="ghost"
class="chip-button cursor-pointer" aria-label="Email" title="Email" @click="copyEmail" />
</ClientTooltip>
<ClientTooltip text="GitHub" v-if="portfolio.profile.socials?.github">
<UButton :to="portfolio.profile.socials.github" target="_blank" icon="i-mdi-github" :square="true"
color="gray" variant="ghost" class="chip-button" aria-label="GitHub" title="GitHub" />
</ClientTooltip>
<ClientTooltip text="LinkedIn" v-if="portfolio.profile.socials?.linkedin">
<UButton :to="portfolio.profile.socials.linkedin" target="_blank" icon="i-logos-linkedin-icon"
:square="true" color="gray" variant="ghost" class="chip-button" aria-label="LinkedIn"
title="LinkedIn" />
</ClientTooltip>
<ClientTooltip text="X" v-if="portfolio.profile.socials?.x || portfolio.profile.socials?.twitter">
<UButton :to="portfolio.profile.socials?.x || portfolio.profile.socials?.twitter" target="_blank"
icon="i-logos-twitter" :square="true" color="gray" variant="ghost" class="chip-button" aria-label="X"
title="X" />
</ClientTooltip>
<ClientTooltip text="Telegram" v-if="portfolio.profile.socials?.telegram">
<UButton :to="portfolio.profile.socials.telegram" target="_blank" icon="i-logos-telegram" :square="true"
color="gray" variant="ghost" class="chip-button" aria-label="Telegram" title="Telegram" />
</ClientTooltip>
<ClientTooltip text="WhatsApp" v-if="portfolio.profile.socials?.whatsapp">
<UButton :to="portfolio.profile.socials.whatsapp" target="_blank" icon="i-logos-whatsapp-icon"
:square="true" color="gray" variant="ghost" class="chip-button" aria-label="WhatsApp"
title="WhatsApp" />
</ClientTooltip>
<ClientTooltip text="Spotify" v-if="portfolio.profile.socials?.spotify">
<UButton :to="portfolio.profile.socials.spotify" target="_blank" :square="true" color="gray"
variant="ghost" class="chip-button" aria-label="Spotify" title="Spotify">
<UIcon name="i-simple-icons-spotify" class="text-[#1DB954] text-lg" />
</UButton>
</ClientTooltip>
<ClientTooltip text="Bento" v-if="portfolio.profile.socials?.bento">
<UButton :to="portfolio.profile.socials.bento" target="_blank" icon="i-simple-icons-bento" :square="true"
color="gray" variant="ghost" class="chip-button" aria-label="Bento" title="Bento" />
</ClientTooltip>
<ClientTooltip text="Instagram" v-if="portfolio.profile.socials?.instagram">
<UButton :to="portfolio.profile.socials.instagram" target="_blank" :square="true" color="gray"
variant="ghost" class="chip-button" aria-label="Instagram" title="Instagram">
<span
class="inline-flex h-5 w-5 items-center justify-center overflow-hidden rounded-[6px] bg-gradient-to-br from-[#f9ce34] via-[#ee2a7b] to-[#6228d7]">
<UIcon name="i-mdi-instagram" class="text-white text-sm" />
</span>
</UButton>
</ClientTooltip>
<ClientTooltip text="Email" v-if="portfolio.profile.socials?.email">
<UButton :to="portfolio.profile.socials.email" icon="i-twemoji-e-mail" :square="true" color="gray"
variant="ghost" class="chip-button" aria-label="Email" title="Email" />
</ClientTooltip>
</div>
</div>
<div
class="block mx-auto sm:mx-0 shrink-0 w-24 h-24 sm:w-32 sm:h-32 md:w-40 md:h-40 ring-4 ring-primary-400/50 dark:ring-primary-300/40 rounded-full overflow-hidden">
<NuxtImg :src="portfolio.profile.avatar || undefined" :alt="portfolio.profile.name"
sizes="96px sm:128px md:160px" width="160" height="160" class="h-full w-full object-cover" format="webp"
preload />
</div>
</div>
</UContainer>
</section>
</template>
<script setup lang="ts">
import { computed, ref } from 'vue'
import { usePortfolio } from '@/composables/usePortfolio'
import type { CompanyExperience, Experience } from '@/types/portfolio.types'
const { t } = useI18n()
const portfolio = usePortfolio()
const toast = useToast()
const currentRole = computed(() => {
const experiences = portfolio.value.experiences as Array<CompanyExperience | Experience>
const formatLinkLabel = (url?: string) => {
if (!url) return undefined
try {
return new URL(url).hostname.replace(/^www\./, '')
} catch {
return undefined
}
}
for (const exp of experiences) {
if ('positions' in exp) {
const active = exp.positions.find(pos => pos.ongoing)
if (active) {
const companyLink = exp.link || active.link
const projectLink = active.link || undefined
return {
company: exp.company,
title: active.title,
companyLink,
companyLogo: active.logo || exp.logo,
projectLink,
projectLabel: active.linkLabel || formatLinkLabel(projectLink),
}
}
} else if (exp.ongoing) {
return {
company: exp.company,
title: exp.role,
companyLink: exp.link,
companyLogo: exp.logo,
}
}
}
return null
})
/**
* Email copy-to-clipboard for Hero quick action
*/
const emailAddress = 'aliarghyani@gmail.com'
const emailTooltip = ref('Email')
async function copyEmail() {
try {
await navigator.clipboard.writeText(emailAddress)
emailTooltip.value = 'Copied'
setTimeout(() => { emailTooltip.value = 'Email' }, 1500)
// Nuxt UI toast: success
toast.add({
title: t('toasts.emailCopied.title'),
description: t('toasts.emailCopied.desc', { email: emailAddress }),
icon: 'i-mdi-clipboard-check',
color: 'emerald'
})
} catch {
// Nuxt UI toast: failure (clipboard not accessible)
toast.add({
title: t('toasts.copyFailed.title'),
description: t('toasts.copyFailed.desc', { email: emailAddress }),
icon: 'i-mdi-clipboard-alert',
color: 'amber'
})
// Fallback prompt if clipboard API is unavailable
const ok = typeof window !== 'undefined' && window.confirm(`Copy email:\n\n${emailAddress}`)
if (ok) {
emailTooltip.value = 'Copied'
setTimeout(() => { emailTooltip.value = 'Email' }, 1500)
// Show success toast after manual copy confirmation
toast.add({
title: t('toasts.emailCopied.title'),
description: t('toasts.emailCopied.desc', { email: emailAddress }),
icon: 'i-mdi-clipboard-check',
color: 'emerald'
})
}
}
}
</script>