melissa/templates: drawer mobile com templates do sistema

Mobile (<1024px) agora segue padrao MelissaBloqueios:
- Coluna esquerda (Templates do sistema) eh teleportada pra um
  drawer fixed que abre via botao "Templates do sistema" no header.
- Botao .mdt-menu-btn--mobile-only substitui o titulo no mobile
  (mais legivel + acao clara).
- Backdrop escuro com blur fecha o drawer ao clicar fora.
- Auto-fecha quando o user seleciona um template (libera viewport
  pra ver o preview no main).

Script:
- drawerOpen + isMobile refs + matchMedia listener
- toggleDrawer/fecharDrawer helpers
- onMounted setup + onBeforeUnmount cleanup

Template:
- <Transition name="mdt-drawer-fade"> wrap (slide horizontal +
  fade do backdrop)
- <Teleport to="#mdt-mobile-drawer-target" :disabled="!isMobile">
  envolvendo a <aside class="mdt-side">
- Botao "Menu" no header com class mdt-menu-btn--mobile-only

CSS:
- .mdt-mobile-drawer fixed left, transform translateX, 250ms cubic
- .mdt-mobile-drawer__backdrop overlay com blur
- @media (max-width: 1023px): cols vira 1-col, sidebar inline some,
  botao menu aparece, titulo canonico some, acções viram icone-only

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
This commit is contained in:
Leonardo
2026-05-21 17:19:17 -03:00
parent 17f114f32f
commit bbbb08ba9d
@@ -9,7 +9,7 @@
* Lógica idêntica à DocumentTemplatesPage (composable * Lógica idêntica à DocumentTemplatesPage (composable
* useDocumentTemplates + DocumentTemplateEditor reusado). * useDocumentTemplates + DocumentTemplateEditor reusado).
*/ */
import { ref, computed, onMounted } from 'vue'; import { ref, computed, onMounted, onBeforeUnmount } from 'vue';
import { useToast } from 'primevue/usetoast'; import { useToast } from 'primevue/usetoast';
import { useConfirm } from 'primevue/useconfirm'; import { useConfirm } from 'primevue/useconfirm';
@@ -67,6 +67,8 @@ function openEdit(tpl) {
function openPreview(tpl) { function openPreview(tpl) {
previewTemplate.value = tpl; previewTemplate.value = tpl;
view.value = 'preview'; view.value = 'preview';
// No mobile, fecha o drawer pra dar espaço ao preview
if (isMobile.value) drawerOpen.value = false;
} }
// Monta HTML completo do template (cabeçalho + corpo + rodapé) com // Monta HTML completo do template (cabeçalho + corpo + rodapé) com
@@ -198,16 +200,68 @@ function getCardMenuItems(tpl) {
return items; return items;
} }
// ── Mobile drawer (espelha padrão MelissaBloqueios) ─────
const drawerOpen = ref(false);
const isMobile = ref(false);
let _mqMobile = null;
function _onMqMobileChange(e) {
isMobile.value = e.matches;
if (!e.matches) drawerOpen.value = false;
}
function toggleDrawer() { drawerOpen.value = !drawerOpen.value; }
function fecharDrawer() { drawerOpen.value = false; }
onMounted(() => { onMounted(() => {
fetchTemplates(true); fetchTemplates(true);
if (typeof window !== 'undefined' && window.matchMedia) {
_mqMobile = window.matchMedia('(max-width: 1023px)');
isMobile.value = _mqMobile.matches;
try { _mqMobile.addEventListener('change', _onMqMobileChange); }
catch { _mqMobile.addListener(_onMqMobileChange); }
}
});
onBeforeUnmount(() => {
if (_mqMobile) {
try { _mqMobile.removeEventListener('change', _onMqMobileChange); }
catch { _mqMobile.removeListener(_onMqMobileChange); }
}
}); });
</script> </script>
<template> <template>
<!-- Mobile drawer (templates do sistema) -->
<Transition name="mdt-drawer-fade">
<div
v-show="isMobile && drawerOpen"
class="mdt-mobile-drawer"
:class="{ 'is-open': drawerOpen }"
>
<div id="mdt-mobile-drawer-target" class="mdt-mobile-drawer__scroll" />
</div>
</Transition>
<Transition name="mdt-drawer-fade">
<div
v-show="isMobile && drawerOpen"
class="mdt-mobile-drawer__backdrop"
@click="fecharDrawer"
/>
</Transition>
<ConfirmDialog /> <ConfirmDialog />
<section class="mdt-page"> <section class="mdt-page">
<header class="mdt-page__head"> <header class="mdt-page__head">
<!-- Botão "Menu" mobile-only (vira título no mobile, abre drawer com templates do sistema) -->
<button
v-if="view === 'list' || view === 'preview'"
class="mdt-menu-btn mdt-menu-btn--mobile-only"
v-tooltip.bottom="'Templates do sistema'"
@click="toggleDrawer"
>
<i class="pi pi-bars" />
<span>Templates do sistema</span>
</button>
<div class="mdt-page__title"> <div class="mdt-page__title">
<button <button
v-if="view !== 'list'" v-if="view !== 'list'"
@@ -301,7 +355,8 @@ onMounted(() => {
<!-- Layout 2-col: sidebar (globais) + main (do tenant) --> <!-- Layout 2-col: sidebar (globais) + main (do tenant) -->
<div v-else class="mdt-cols"> <div v-else class="mdt-cols">
<!-- COL 1 Sidebar: Templates do sistema --> <!-- COL 1 Sidebar: Templates do sistema (teleporta pro drawer no mobile) -->
<Teleport to="#mdt-mobile-drawer-target" :disabled="!isMobile">
<aside class="mdt-side"> <aside class="mdt-side">
<header class="mdt-side__head"> <header class="mdt-side__head">
<div class="mdt-side__title"> <div class="mdt-side__title">
@@ -340,6 +395,7 @@ onMounted(() => {
</li> </li>
</ul> </ul>
</aside> </aside>
</Teleport>
<!-- COL 2 Main: Seus documentos OU Preview do sistema --> <!-- COL 2 Main: Seus documentos OU Preview do sistema -->
<main class="mdt-main"> <main class="mdt-main">
@@ -1152,11 +1208,96 @@ onMounted(() => {
color: var(--p-primary-color) !important; color: var(--p-primary-color) !important;
} }
/* Mobile (<1024px) */ /* ═══════ Botão "Menu" mobile-only (abre drawer com templates do sistema) ═══════ */
.mdt-menu-btn {
display: none;
align-items: center;
gap: 6px;
padding: 7px 12px;
border-radius: 8px;
border: 1px solid var(--m-border);
background: var(--m-bg-soft);
color: var(--m-text);
cursor: pointer;
font-size: 0.82rem;
font-weight: 600;
flex-shrink: 0;
}
.mdt-menu-btn:hover { background: var(--m-bg-soft-hover); }
/* ═══════ Mobile drawer (templates do sistema teleportados) ═══════ */
.mdt-mobile-drawer {
position: fixed;
top: 0; left: 0;
height: 100vh;
height: 100dvh;
width: min(360px, 88vw);
z-index: 80;
background: var(--m-bg-medium);
backdrop-filter: blur(28px) saturate(160%);
-webkit-backdrop-filter: blur(28px) saturate(160%);
border-right: 1px solid var(--m-border);
transform: translateX(-100%);
transition: transform 250ms cubic-bezier(0.4, 0, 0.2, 1);
color: var(--m-text);
font-family: 'Segoe UI', system-ui, -apple-system, sans-serif;
display: flex;
flex-direction: column;
}
.mdt-mobile-drawer.is-open { transform: translateX(0); }
.mdt-mobile-drawer__scroll {
flex: 1;
min-height: 0;
overflow-y: auto;
overflow-x: hidden;
padding: 12px;
display: flex;
flex-direction: column;
gap: 12px;
scrollbar-width: thin;
scrollbar-color: var(--m-border-strong) transparent;
}
.mdt-mobile-drawer__scroll::-webkit-scrollbar { width: 5px; }
.mdt-mobile-drawer__scroll::-webkit-scrollbar-thumb {
background: var(--m-border-strong);
border-radius: 3px;
}
/* No mobile a .mdt-side é teleportada pra dentro do drawer scroll */
.mdt-mobile-drawer__scroll .mdt-side {
width: 100%;
height: 100%;
border: 1px solid var(--m-border);
}
.mdt-mobile-drawer__backdrop {
position: fixed;
inset: 0;
background: rgba(0, 0, 0, 0.45);
backdrop-filter: blur(4px);
-webkit-backdrop-filter: blur(4px);
z-index: 79;
}
.mdt-drawer-fade-enter-active,
.mdt-drawer-fade-leave-active { transition: opacity 200ms ease; }
.mdt-drawer-fade-enter-from,
.mdt-drawer-fade-leave-to { opacity: 0; }
/* ═══════ Mobile (<1024px) — ajustes ═══════ */
@media (max-width: 1023px) { @media (max-width: 1023px) {
.mdt-page__title > span:nth-child(2):not(.mdt-page__count) { /* Esconde a sidebar inline (templates do sistema viram drawer) */
font-size: 0.92rem; .mdt-cols {
grid-template-columns: 1fr;
overflow: hidden;
} }
.mdt-cols > .mdt-side { display: none; }
/* Mostra botão Menu, esconde título canônico (vira sub-info) */
.mdt-menu-btn--mobile-only { display: inline-flex; }
.mdt-page__title > span:not(.mdt-page__count) { display: none; }
.mdt-page__title-icon { display: none; }
.mdt-page__count { display: none; }
/* Compacta botões de ação */
.mdt-act-btn span { display: none; } .mdt-act-btn span { display: none; }
.mdt-act-btn { width: 32px; padding: 0; justify-content: center; } .mdt-act-btn { width: 32px; padding: 0; justify-content: center; }
.mdt-grid { grid-template-columns: 1fr; } .mdt-grid { grid-template-columns: 1fr; }