mirror of
https://github.com/mmahdium/portfolio.git
synced 2025-12-20 09:23:54 +01:00
190 lines
8.8 KiB
Vue
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>
|