gigafibre-fsm/apps/ops/src/layouts/MainLayout.vue
louispaulb 21e2c846bf feat(ops): Email Queue admin page (view/delete/purge ERPNext outbound)
Surfaces ERPNext's Email Queue in Ops (nav « File courriels ») so ops can see
what's queued — important now that mute_emails=1 + scheduler paused mean nothing
flushes — and delete/purge stale entries without the ERPNext desk.

- hub lib/email-queue.js: GET list (by status, recipients read from each row's
  full doc since ERPNext ignores fields on child-doctype REST), DELETE :name,
  POST /purge {status}. Wired in server.js.
- ops: api/emailQueue.js + EmailQueuePage.vue (status filter, recipients,
  reference, error tooltip, per-row delete + « Purger Not Sent »), route + nav.

Verified live: 13 'Not Sent' (old internal test emails, no invoice refs).

Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
2026-06-02 17:29:33 -04:00

270 lines
12 KiB
Vue

<template>
<q-layout view="lHh LpR fFf">
<!-- Collapsible Sidebar -->
<q-drawer v-model="drawer" :width="sidebarW" :breakpoint="1024" class="ops-sidebar" :class="{ 'ops-sidebar-mini': collapsed }">
<q-list>
<q-item class="q-py-md q-mb-sm" style="pointer-events:none" :class="{ 'justify-center': collapsed }">
<q-item-section avatar><q-icon name="hub" size="28px" color="white" /></q-item-section>
<q-item-section v-if="!collapsed"><q-item-label style="color:#fff;font-size:1.1rem;font-weight:700">Targo Ops</q-item-label></q-item-section>
</q-item>
<q-item v-for="nav in navItems" :key="nav.path" clickable :to="nav.path"
:class="{ 'active-link': isActive(nav.path), 'justify-center': collapsed }"
:title="collapsed ? nav.label : undefined">
<q-item-section avatar><component :is="icons[nav.icon]" :size="20" /></q-item-section>
<q-item-section v-if="!collapsed"><q-item-label>{{ nav.label }}</q-item-label></q-item-section>
<q-item-section side v-if="nav.badge && !collapsed"><q-badge color="red" :label="nav.badge" rounded /></q-item-section>
<q-tooltip v-if="collapsed" anchor="center right" self="center left" :offset="[8, 0]">{{ nav.label }}</q-tooltip>
</q-item>
</q-list>
<div class="ops-sidebar-bottom">
<q-item dense clickable @click="toggleCollapse" class="ops-collapse-btn" :class="{ 'justify-center': collapsed }">
<q-item-section avatar>
<component :is="collapsed ? icons.PanelLeftOpen : icons.PanelLeftClose" :size="16" />
</q-item-section>
<q-item-section v-if="!collapsed"><q-item-label>Réduire</q-item-label></q-item-section>
<q-tooltip v-if="collapsed" anchor="center right" self="center left" :offset="[8, 0]">Agrandir le menu</q-tooltip>
</q-item>
<q-item dense clickable @click="auth.doLogout()" :class="{ 'justify-center': collapsed }">
<q-item-section avatar><component :is="icons.LogOut" :size="16" /></q-item-section>
<q-item-section v-if="!collapsed"><q-item-label>{{ userName || auth.user || 'User' }}</q-item-label></q-item-section>
<q-tooltip v-if="collapsed" anchor="center right" self="center left" :offset="[8, 0]">{{ userName || auth.user || 'Déconnexion' }}</q-tooltip>
</q-item>
</div>
</q-drawer>
<!-- Mobile header -->
<q-header v-if="$q.screen.lt.lg" class="ops-mobile-header">
<q-toolbar>
<q-btn flat round dense icon="menu" color="white" @click="drawer = !drawer" />
<q-toolbar-title class="text-weight-bold" style="font-size:1rem;color:#fff">{{ currentNav?.label || 'Targo Ops' }}</q-toolbar-title>
<q-space />
<q-btn v-if="!isDispatch" flat round dense icon="search" color="white" @click="mobileSearchOpen = !mobileSearchOpen" />
</q-toolbar>
<div v-if="mobileSearchOpen && !isDispatch" class="q-px-sm q-pb-sm" style="background:var(--ops-sidebar-bg)">
<q-input v-model="searchQuery" placeholder="Rechercher client, adresse..." dense outlined dark autofocus class="ops-search-dark"
@keydown.enter.prevent="doSearch" @keydown.escape="closeMobileSearch">
<template #prepend><q-icon name="search" color="grey-5" /></template>
<template #append v-if="searchQuery"><q-icon name="close" class="cursor-pointer" color="grey-5" @click="clearSearch" /></template>
</q-input>
<div v-if="searchResults.length" class="ops-search-results">
<div v-for="r in searchResults" :key="r.id" class="ops-search-result" @mousedown="goToResult(r)">
<q-icon :name="r.icon" size="18px" :color="r.type === 'customer' ? 'indigo-5' : 'teal-5'" class="q-mr-sm" />
<div style="flex:1;min-width:0">
<div class="ops-search-title">{{ r.title }}</div>
<div class="ops-search-sub">{{ r.sub }}</div>
</div>
</div>
</div>
</div>
</q-header>
<!-- Main content -->
<q-page-container>
<!-- Desktop top bar (hidden on dispatch) -->
<div v-if="$q.screen.gt.md && !isDispatch" class="ops-topbar">
<div class="text-h6 text-weight-bold">{{ currentNav?.label || '' }}</div>
<q-space />
<div style="position:relative;width:400px">
<q-input v-model="searchQuery" placeholder="Rechercher client, adresse, ticket..." dense outlined class="ops-search"
@keydown.enter.prevent="doSearch" @keydown.escape="clearSearch"
@keydown.down.prevent="moveHighlight(1)" @keydown.up.prevent="moveHighlight(-1)"
@update:model-value="onSearchInput" @blur="onSearchBlur">
<template #prepend><q-icon name="search" color="grey-6" /></template>
<template #append v-if="searchQuery">
<q-icon name="close" class="cursor-pointer" @click="clearSearch" />
</template>
</q-input>
<div v-if="searchResults.length && searchDropdownOpen" class="ops-search-results ops-search-results-desktop">
<div v-for="(r, i) in searchResults" :key="r.id" class="ops-search-result"
:class="{ 'ops-search-highlighted': i === highlightIdx }"
@mousedown="goToResult(r)">
<q-icon :name="r.icon" size="18px" :color="r.type === 'customer' ? 'indigo-5' : 'teal-5'" class="q-mr-sm" />
<div style="flex:1;min-width:0">
<div class="ops-search-title">{{ r.title }}</div>
<div class="ops-search-sub">{{ r.sub }}</div>
</div>
<div class="ops-search-type">{{ r.typeLabel }}</div>
</div>
</div>
<div v-else-if="searchDropdownOpen && searchQuery.length >= 2 && !searchResults.length && !searchLoading" class="ops-search-results ops-search-results-desktop">
<div class="ops-search-result" style="justify-content:center;color:#94a3b8">Aucun résultat</div>
</div>
</div>
</div>
<router-view />
</q-page-container>
<!-- Conversation panel (right drawer) -->
<ConversationPanel v-if="can('view_clients')" />
<!-- Floating conversation button -->
<q-page-sticky v-if="can('view_clients')" position="bottom-right" :offset="[18, 18]">
<q-btn fab icon="chat" color="indigo-6" @click="toggleConvPanel">
<q-badge v-if="convCount > 0" color="red" floating rounded :label="convCount" />
<q-tooltip>Conversations</q-tooltip>
</q-btn>
</q-page-sticky>
<!-- Global Flow Editor dialog (any page can open it via useFlowEditor) -->
<FlowEditorDialog v-if="can('manage_settings')" />
</q-layout>
</template>
<script setup>
import { ref, computed } from 'vue'
import { useRoute, useRouter } from 'vue-router'
import { useAuthStore } from 'src/stores/auth'
import { usePermissions } from 'src/composables/usePermissions'
import { listDocs } from 'src/api/erp'
import { navItems as allNavItems } from 'src/config/nav'
import {
LayoutDashboard, Users, Truck, Ticket, UsersRound, BarChart3,
Gift, Settings, LogOut, PanelLeftOpen, PanelLeftClose, Mail,
} from 'lucide-vue-next'
import ConversationPanel from 'src/components/shared/ConversationPanel.vue'
import { useConversations } from 'src/composables/useConversations'
import FlowEditorDialog from 'src/components/flow-editor/FlowEditorDialog.vue'
const icons = { LayoutDashboard, Users, Truck, Ticket, UsersRound, BarChart3, Gift, Settings, LogOut, PanelLeftOpen, PanelLeftClose, Mail }
const { panelOpen, activeCount: convCount } = useConversations()
function toggleConvPanel () { panelOpen.value = !panelOpen.value }
const auth = useAuthStore()
const { can, isLoaded, userName } = usePermissions()
// Filter nav items based on user capabilities
const navItems = computed(() =>
allNavItems.filter(n => !n.requires || can(n.requires))
)
const route = useRoute()
const router = useRouter()
const drawer = ref(true)
const collapsed = ref(localStorage.getItem('ops-sidebar-collapsed') !== 'false')
const sidebarW = computed(() => collapsed.value ? 64 : 220)
const isDispatch = computed(() => route.path === '/dispatch')
const currentNav = computed(() =>
navItems.value.find(n => n.path === route.path) || navItems.value.find(n => route.path.startsWith(n.path) && n.path !== '/')
)
function isActive (path) {
if (path === '/') return route.path === '/'
return route.path === path || route.path.startsWith(path + '/')
}
function toggleCollapse () {
collapsed.value = !collapsed.value
localStorage.setItem('ops-sidebar-collapsed', collapsed.value ? 'true' : 'false')
}
// ── Simple inline search (no composable, no external state) ──────────────
const searchQuery = ref('')
const searchResults = ref([])
const searchLoading = ref(false)
const searchDropdownOpen = ref(false)
const highlightIdx = ref(-1)
const mobileSearchOpen = ref(false)
let searchTimer = null
function onSearchInput (val) {
clearTimeout(searchTimer)
highlightIdx.value = -1
if (!val || val.length < 2) {
searchResults.value = []
searchDropdownOpen.value = false
searchLoading.value = false
return
}
searchDropdownOpen.value = true
searchLoading.value = true
searchTimer = setTimeout(() => runSearch(val), 300)
}
async function runSearch (q) {
if (!q || q.length < 2) { searchLoading.value = false; return }
try {
const cf = ['name', 'customer_name', 'customer_type', 'territory', 'disabled']
const lf = ['name', 'address_line', 'city', 'customer', 'customer_name', 'status']
const timeout = new Promise((_, r) => setTimeout(() => r(new Error('timeout')), 4000))
const [cName, cId, lAddr, lCity] = await Promise.race([
Promise.all([
listDocs('Customer', { filters: { customer_name: ['like', '%' + q + '%'] }, fields: cf, limit: 6, orderBy: 'customer_name asc' }).catch(() => []),
listDocs('Customer', { filters: { name: ['like', '%' + q + '%'] }, fields: cf, limit: 4, orderBy: 'name asc' }).catch(() => []),
listDocs('Service Location', { filters: { address_line: ['like', '%' + q + '%'] }, fields: lf, limit: 6, orderBy: 'address_line asc' }).catch(() => []),
listDocs('Service Location', { filters: { city: ['like', '%' + q + '%'] }, fields: lf, limit: 4, orderBy: 'city asc' }).catch(() => []),
]),
timeout,
])
const seen = new Set()
const out = []
for (const c of [...cName, ...cId]) {
if (seen.has(c.name)) continue; seen.add(c.name)
out.push({ id: 'c-' + c.name, type: 'customer', typeLabel: 'Client', icon: 'person', title: c.customer_name, sub: c.name + (c.territory ? ' · ' + c.territory : ''), route: '/clients/' + c.name })
}
for (const l of [...lAddr, ...lCity]) {
if (seen.has(l.name)) continue; seen.add(l.name)
out.push({ id: 'l-' + l.name, type: 'location', typeLabel: 'Adresse', icon: 'location_on', title: l.address_line + (l.city ? ', ' + l.city : ''), sub: (l.customer_name || l.customer) + ' · ' + l.status, route: '/clients/' + l.customer })
}
searchResults.value = out.slice(0, 12)
} catch {
searchResults.value = []
}
searchLoading.value = false
}
function moveHighlight (dir) {
if (!searchResults.value.length) return
searchDropdownOpen.value = true
highlightIdx.value = Math.max(-1, Math.min(searchResults.value.length - 1, highlightIdx.value + dir))
}
async function doSearch () {
const q = searchQuery.value?.trim()
if (!q || q.length < 2) return
// Always cancel debounce and run search immediately on Enter
if (!searchResults.value.length) {
clearTimeout(searchTimer)
searchLoading.value = true
await runSearch(q)
}
if (searchResults.value.length) {
const idx = highlightIdx.value >= 0 ? highlightIdx.value : 0
goToResult(searchResults.value[idx])
} else {
router.push({ path: '/clients', query: { q } })
clearSearch()
}
}
function goToResult (r) {
router.push(r.route)
clearSearch()
}
function clearSearch () {
searchQuery.value = ''
searchResults.value = []
searchDropdownOpen.value = false
searchLoading.value = false
highlightIdx.value = -1
mobileSearchOpen.value = false
clearTimeout(searchTimer)
}
function closeMobileSearch () {
clearSearch()
mobileSearchOpen.value = false
}
function onSearchBlur () {
setTimeout(() => { searchDropdownOpen.value = false }, 200)
}
</script>