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:
@@ -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); }
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user