MelissaPerfil: scroll mobile + ancoras nos badges + email placeholder + 50/50 desktop

1. Mobile scroll fix: .mpr-main ganha min-height: 0 (faltava pra
   permitir shrink dentro do flex column do .mpr-body em mobile, sem
   isso o overflow-y: auto nao engatava).

2. Badges e dicas viraram <button> com @click que rola pra sessao
   correspondente do form (Identidade / Avatar / Bio / Contato /
   Redes). Em mobile o drawer fecha antes do scroll (exceto Avatar,
   que vive na propria sidebar). Cada card .mpr-w ganhou id pra
   ancora (mpr-sec-*).

3. Email readonly recebe placeholder=" " (espaco) — sem ele o
   FloatLabel variant=on ficava em cima do email enquanto o user
   nao tinha foco, pq :placeholder-shown nao aplica.

4. Desktop (>=1024px): .mpr-main vira grid 2 colunas (50/50). Os 4
   cards (Identidade, Contato, Bio, Redes) ficam lado a lado em
   pares. Internal .mpr-grid colapsa pra 1 col nesse modo, e
   .mpr-field--half passa a span 1/-1 — evita ficar cramped em
   metade da largura.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
This commit is contained in:
Leonardo
2026-05-06 15:14:49 -03:00
parent abd4f8f34c
commit f2fd2e4722
+74 -10
View File
@@ -15,7 +15,7 @@
* Logica de load/save espelhada do ProfilePage.vue (mesmas tabelas * Logica de load/save espelhada do ProfilePage.vue (mesmas tabelas
* profiles + user_settings + auth.user_metadata, mesmo bucket avatars). * profiles + user_settings + auth.user_metadata, mesmo bucket avatars).
*/ */
import { ref, reactive, computed, onMounted, onBeforeUnmount } from 'vue'; import { ref, reactive, computed, onMounted, onBeforeUnmount, nextTick } from 'vue';
import { useToast } from 'primevue/usetoast'; import { useToast } from 'primevue/usetoast';
import { useConfirm } from 'primevue/useconfirm'; import { useConfirm } from 'primevue/useconfirm';
import { useRouter } from 'vue-router'; import { useRouter } from 'vue-router';
@@ -402,6 +402,34 @@ async function saveAll() {
} }
} }
// ── Ancoras: badge/dica leva pra sessao do form ────────────
const SECTION_BY_FIELD = {
full_name: 'identidade',
nickname: 'identidade',
work_description: 'identidade',
name: 'identidade',
nick: 'identidade',
work: 'identidade',
avatar: 'avatar',
photo: 'avatar',
bio: 'bio',
phone: 'contato',
social: 'redes'
};
function scrollToSection(field) {
const sec = SECTION_BY_FIELD[field];
if (!sec) return;
// Avatar fica na sidebar; em mobile mantem o drawer aberto (e nao scrolla
// o main, pq a sessao nao existe no main).
const isAvatar = sec === 'avatar';
if (isMobile.value && !isAvatar) drawerOpen.value = false;
nextTick(() => {
const target = document.getElementById('mpr-sec-' + sec);
if (target) target.scrollIntoView({ behavior: 'smooth', block: 'start' });
});
}
// ── Sair da conta ────────────────────────────────────────── // ── Sair da conta ──────────────────────────────────────────
function confirmSignOut() { function confirmSignOut() {
confirm.require({ confirm.require({
@@ -557,34 +585,38 @@ onBeforeUnmount(() => {
<!-- Badges --> <!-- Badges -->
<div class="mpr-badges-head">Conquistas</div> <div class="mpr-badges-head">Conquistas</div>
<div class="mpr-badges"> <div class="mpr-badges">
<div <button
v-for="badge in badges" v-for="badge in badges"
:key="badge.key" :key="badge.key"
type="button"
class="mpr-badge" class="mpr-badge"
:class="badge.earned ? 'is-earned' : 'is-locked'" :class="badge.earned ? 'is-earned' : 'is-locked'"
v-tooltip.top="badge.earned ? badge.label : 'Bloqueado ' + badge.label" v-tooltip.top="badge.earned ? badge.label : 'Bloqueado ' + badge.label"
@click="scrollToSection(badge.key)"
> >
<span class="mpr-badge__icon">{{ badge.icon }}</span> <span class="mpr-badge__icon">{{ badge.icon }}</span>
<span class="mpr-badge__label">{{ badge.label }}</span> <span class="mpr-badge__label">{{ badge.label }}</span>
</div> </button>
</div> </div>
<!-- Dicas (campos faltando) --> <!-- Dicas (campos faltando) -->
<div v-if="progressSuggestions.length" class="mpr-tips"> <div v-if="progressSuggestions.length" class="mpr-tips">
<div class="mpr-tips__head">O que falta</div> <div class="mpr-tips__head">O que falta</div>
<span <button
v-for="(tip, i) in progressSuggestions" v-for="(tip, i) in progressSuggestions"
:key="i" :key="i"
type="button"
class="mpr-tip" class="mpr-tip"
@click="scrollToSection(tip.key)"
> >
<i :class="tip.icon" /> <i :class="tip.icon" />
{{ tip.text }} {{ tip.text }}
</span> </button>
</div> </div>
</div> </div>
<!-- Card: Avatar --> <!-- Card: Avatar -->
<div class="mpr-w mpr-w--side"> <div id="mpr-sec-avatar" class="mpr-w mpr-w--side">
<div class="mpr-w__head"> <div class="mpr-w__head">
<span class="mpr-w__title"> <span class="mpr-w__title">
<i class="pi pi-image" /> Avatar <i class="pi pi-image" /> Avatar
@@ -642,7 +674,7 @@ onBeforeUnmount(() => {
<template v-else> <template v-else>
<!-- Identidade --> <!-- Identidade -->
<div class="mpr-w"> <div id="mpr-sec-identidade" class="mpr-w">
<div class="mpr-w__head"> <div class="mpr-w__head">
<span class="mpr-w__title"> <span class="mpr-w__title">
<i class="pi pi-id-card" /> Identidade <i class="pi pi-id-card" /> Identidade
@@ -712,7 +744,7 @@ onBeforeUnmount(() => {
</div> </div>
<!-- Contato --> <!-- Contato -->
<div class="mpr-w"> <div id="mpr-sec-contato" class="mpr-w">
<div class="mpr-w__head"> <div class="mpr-w__head">
<span class="mpr-w__title"> <span class="mpr-w__title">
<i class="pi pi-phone" /> Contato <i class="pi pi-phone" /> Contato
@@ -740,6 +772,7 @@ onBeforeUnmount(() => {
id="mpr_email" id="mpr_email"
:value="userEmail" :value="userEmail"
readonly readonly
placeholder=" "
class="w-full" class="w-full"
/> />
<label for="mpr_email">E-mail (login)</label> <label for="mpr_email">E-mail (login)</label>
@@ -750,7 +783,7 @@ onBeforeUnmount(() => {
</div> </div>
<!-- Bio --> <!-- Bio -->
<div class="mpr-w"> <div id="mpr-sec-bio" class="mpr-w">
<div class="mpr-w__head"> <div class="mpr-w__head">
<span class="mpr-w__title"> <span class="mpr-w__title">
<i class="pi pi-pencil" /> Bio <i class="pi pi-pencil" /> Bio
@@ -770,7 +803,7 @@ onBeforeUnmount(() => {
</div> </div>
<!-- Sites e Redes --> <!-- Sites e Redes -->
<div class="mpr-w"> <div id="mpr-sec-redes" class="mpr-w">
<div class="mpr-w__head"> <div class="mpr-w__head">
<span class="mpr-w__title"> <span class="mpr-w__title">
<i class="pi pi-share-alt" /> Sites e Redes <i class="pi pi-share-alt" /> Sites e Redes
@@ -1043,6 +1076,7 @@ onBeforeUnmount(() => {
.mpr-main { .mpr-main {
flex: 1; flex: 1;
min-width: 0; min-width: 0;
min-height: 0;
overflow-y: auto; overflow-y: auto;
overflow-x: hidden; overflow-x: hidden;
padding: 16px; padding: 16px;
@@ -1058,6 +1092,23 @@ onBeforeUnmount(() => {
border-radius: 3px; border-radius: 3px;
} }
/* Desktop (>=1024px): cards em 2 colunas (50/50). Internal grid
colapsa pra 1 col pra nao ficar cramped. */
@media (min-width: 1024px) {
.mpr-main {
display: grid;
grid-template-columns: repeat(2, 1fr);
gap: 12px;
align-content: start;
}
.mpr-main .mpr-grid {
grid-template-columns: 1fr;
}
.mpr-main .mpr-field--half {
grid-column: 1 / -1;
}
}
/* ═══════ Card-base (mpr-w) ═══════ */ /* ═══════ Card-base (mpr-w) ═══════ */
.mpr-w { .mpr-w {
background: var(--m-bg-medium); background: var(--m-bg-medium);
@@ -1183,7 +1234,11 @@ onBeforeUnmount(() => {
border-radius: 999px; border-radius: 999px;
border: 1px solid var(--m-border); border: 1px solid var(--m-border);
font-size: 0.7rem; font-size: 0.7rem;
font-family: inherit;
cursor: pointer;
transition: transform 120ms ease, background-color 120ms ease;
} }
.mpr-badge:hover { transform: translateY(-1px); }
.mpr-badge.is-earned { .mpr-badge.is-earned {
background: color-mix(in srgb, var(--p-primary-color) 14%, transparent); background: color-mix(in srgb, var(--p-primary-color) 14%, transparent);
border-color: color-mix(in srgb, var(--p-primary-color) 38%, transparent); border-color: color-mix(in srgb, var(--p-primary-color) 38%, transparent);
@@ -1222,6 +1277,15 @@ onBeforeUnmount(() => {
border: 1px solid var(--m-border); border: 1px solid var(--m-border);
color: var(--m-text-muted); color: var(--m-text-muted);
font-size: 0.72rem; font-size: 0.72rem;
font-family: inherit;
cursor: pointer;
text-align: left;
transition: background-color 120ms ease, color 120ms ease, border-color 120ms ease;
}
.mpr-tip:hover {
background: color-mix(in srgb, var(--p-primary-color) 12%, transparent);
border-color: color-mix(in srgb, var(--p-primary-color) 38%, transparent);
color: var(--p-primary-color);
} }
.mpr-tip > i { font-size: 0.7rem; color: var(--p-primary-color); } .mpr-tip > i { font-size: 0.7rem; color: var(--p-primary-color); }