228 lines
6.3 KiB
Vue
228 lines
6.3 KiB
Vue
<template>
|
|
<TransitionRoot :show="sidebarOpened">
|
|
<Dialog as="div" @close="sidebarOpened = false" class="fixed inset-0 z-40">
|
|
<TransitionChild
|
|
as="template"
|
|
enter="transition ease-in-out duration-200 transform"
|
|
enter-from="-translate-x-full"
|
|
enter-to="translate-x-0"
|
|
leave="transition ease-in-out duration-200 transform"
|
|
leave-from="translate-x-0"
|
|
leave-to="-translate-x-full"
|
|
>
|
|
<div
|
|
class="relative z-10 flex h-full w-[260px] flex-col justify-between border-r bg-surface-menu-bar transition-all duration-300 ease-in-out"
|
|
>
|
|
<div>
|
|
<UserDropdown class="p-2" :isCollapsed="!sidebarOpened" />
|
|
</div>
|
|
<div class="flex-1 overflow-y-auto">
|
|
<div class="mb-3 flex flex-col">
|
|
<SidebarLink
|
|
id="notifications-btn"
|
|
:label="__('Notifications')"
|
|
:icon="NotificationsIcon"
|
|
:to="{ name: 'Notifications' }"
|
|
class="relative mx-2 my-0.5"
|
|
>
|
|
<template #right>
|
|
<Badge
|
|
v-if="unreadNotificationsCount"
|
|
:label="unreadNotificationsCount"
|
|
variant="subtle"
|
|
/>
|
|
</template>
|
|
</SidebarLink>
|
|
</div>
|
|
<div v-for="view in allViews" :key="view.label">
|
|
<Section
|
|
:label="view.name"
|
|
:hideLabel="view.hideLabel"
|
|
:opened="view.opened"
|
|
>
|
|
<template #header="{ opened, hide, toggle }">
|
|
<div
|
|
v-if="!hide"
|
|
class="ml-2 mt-4 flex h-7 w-auto cursor-pointer gap-1.5 px-1 text-base font-medium text-ink-gray-5 opacity-100 transition-all duration-300 ease-in-out"
|
|
@click="toggle()"
|
|
>
|
|
<FeatherIcon
|
|
name="chevron-right"
|
|
class="h-4 text-ink-gray-9 transition-all duration-300 ease-in-out"
|
|
:class="{ 'rotate-90': opened }"
|
|
/>
|
|
<span>{{ __(view.name) }}</span>
|
|
</div>
|
|
</template>
|
|
<nav class="flex flex-col">
|
|
<SidebarLink
|
|
v-for="link in view.views"
|
|
:icon="link.icon"
|
|
:label="__(link.label)"
|
|
:to="link.to"
|
|
class="mx-2 my-0.5"
|
|
/>
|
|
</nav>
|
|
</Section>
|
|
</div>
|
|
</div>
|
|
<TrialBanner v-if="isFCSite.data" />
|
|
</div>
|
|
</TransitionChild>
|
|
<TransitionChild
|
|
as="template"
|
|
enter="transition-opacity ease-linear duration-200"
|
|
enter-from="opacity-0"
|
|
enter-to="opacity-100"
|
|
leave="transition-opacity ease-linear duration-200"
|
|
leave-from="opacity-100"
|
|
leave-to="opacity-0"
|
|
>
|
|
<DialogOverlay class="fixed inset-0 bg-gray-600 bg-opacity-50" />
|
|
</TransitionChild>
|
|
</Dialog>
|
|
</TransitionRoot>
|
|
</template>
|
|
<script setup>
|
|
import {
|
|
TransitionRoot,
|
|
TransitionChild,
|
|
Dialog,
|
|
DialogOverlay,
|
|
} from '@headlessui/vue'
|
|
import Section from '@/components/Section.vue'
|
|
import Email2Icon from '@/components/Icons/Email2Icon.vue'
|
|
import PinIcon from '@/components/Icons/PinIcon.vue'
|
|
import UserDropdown from '@/components/UserDropdown.vue'
|
|
import LeadsIcon from '@/components/Icons/LeadsIcon.vue'
|
|
import DealsIcon from '@/components/Icons/DealsIcon.vue'
|
|
import ContactsIcon from '@/components/Icons/ContactsIcon.vue'
|
|
import OrganizationsIcon from '@/components/Icons/OrganizationsIcon.vue'
|
|
import NoteIcon from '@/components/Icons/NoteIcon.vue'
|
|
import TaskIcon from '@/components/Icons/TaskIcon.vue'
|
|
import PhoneIcon from '@/components/Icons/PhoneIcon.vue'
|
|
import NotificationsIcon from '@/components/Icons/NotificationsIcon.vue'
|
|
import SidebarLink from '@/components/SidebarLink.vue'
|
|
import { viewsStore } from '@/stores/views'
|
|
import { unreadNotificationsCount } from '@/stores/notifications'
|
|
import { TrialBanner, createResource } from 'frappe-ui'
|
|
import { computed, h, provide } from 'vue'
|
|
import { mobileSidebarOpened as sidebarOpened } from '@/composables/settings'
|
|
|
|
const { getPinnedViews, getPublicViews } = viewsStore()
|
|
|
|
const links = [
|
|
{
|
|
label: 'Leads',
|
|
icon: LeadsIcon,
|
|
to: 'Leads',
|
|
},
|
|
{
|
|
label: 'Deals',
|
|
icon: DealsIcon,
|
|
to: 'Deals',
|
|
},
|
|
{
|
|
label: 'Contacts',
|
|
icon: ContactsIcon,
|
|
to: 'Contacts',
|
|
},
|
|
{
|
|
label: 'Organizations',
|
|
icon: OrganizationsIcon,
|
|
to: 'Organizations',
|
|
},
|
|
{
|
|
label: 'Notes',
|
|
icon: NoteIcon,
|
|
to: 'Notes',
|
|
},
|
|
{
|
|
label: 'Tasks',
|
|
icon: TaskIcon,
|
|
to: 'Tasks',
|
|
},
|
|
{
|
|
label: 'Call Logs',
|
|
icon: PhoneIcon,
|
|
to: 'Call Logs',
|
|
},
|
|
{
|
|
label: 'Email Templates',
|
|
icon: Email2Icon,
|
|
to: 'Email Templates',
|
|
},
|
|
]
|
|
|
|
const allViews = computed(() => {
|
|
let _views = [
|
|
{
|
|
name: 'All Views',
|
|
hideLabel: true,
|
|
opened: true,
|
|
views: links,
|
|
},
|
|
]
|
|
if (getPublicViews().length) {
|
|
_views.push({
|
|
name: 'Public views',
|
|
opened: true,
|
|
views: parseView(getPublicViews()),
|
|
})
|
|
}
|
|
|
|
if (getPinnedViews().length) {
|
|
_views.push({
|
|
name: 'Pinned views',
|
|
opened: true,
|
|
views: parseView(getPinnedViews()),
|
|
})
|
|
}
|
|
return _views
|
|
})
|
|
|
|
function parseView(views) {
|
|
return views.map((view) => {
|
|
return {
|
|
label: view.label,
|
|
icon: getIcon(view.route_name, view.icon),
|
|
to: {
|
|
name: view.route_name,
|
|
params: { viewType: view.type || 'list' },
|
|
query: { view: view.name },
|
|
},
|
|
}
|
|
})
|
|
}
|
|
|
|
function getIcon(routeName, icon) {
|
|
if (icon) return h('div', { class: 'size-auto' }, icon)
|
|
|
|
switch (routeName) {
|
|
case 'Leads':
|
|
return LeadsIcon
|
|
case 'Deals':
|
|
return DealsIcon
|
|
case 'Contacts':
|
|
return ContactsIcon
|
|
case 'Organizations':
|
|
return OrganizationsIcon
|
|
case 'Notes':
|
|
return NoteIcon
|
|
case 'Call Logs':
|
|
return PhoneIcon
|
|
default:
|
|
return PinIcon
|
|
}
|
|
}
|
|
|
|
const isFCSite = createResource({
|
|
url: 'frappe.integrations.frappe_providers.frappecloud_billing.is_fc_site',
|
|
cache: 'isFCSite',
|
|
auto: true,
|
|
transform: (data) => Boolean(data),
|
|
})
|
|
|
|
provide('isFCSite', isFCSite)
|
|
</script>
|