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