new selfhosted version

This commit is contained in:
antonio
2025-11-28 14:11:51 +01:00
parent afda29997d
commit 951860f67e
1046 changed files with 72586 additions and 574750 deletions

View File

@@ -0,0 +1,210 @@
<script setup lang="ts">
import {
Drawer,
DrawerClose,
DrawerContent,
DrawerDescription,
DrawerFooter,
DrawerHeader,
DrawerTitle,
DrawerTrigger,
} from '@/components/ui/drawer'
import { Funnel, Link, Lock, MessageCircleMoreIcon } from 'lucide-vue-next'
import LiveUsers from '~/components/dashboard/LiveUsers.vue';
const domainStore = useDomainStore();
const premium = usePremiumStore();
const projectStore = useProjectStore();
const sidebarHeaderVisible = computed(() => {
if (domainStore.domains.length == 0) return false;
return true;
});
const { data: plan } = await useAuthFetch('/api/user/plan');
const feedbackText = ref<string>('');
const feedbackOpen = ref<boolean>(false);
function sendFeedback() {
useCatch({
toast: true,
toastTitle: 'Error sending feedback',
async action() {
await useAuthFetchSync('/api/feedback/send', {
method: 'POST',
headers: { 'Content-Type': 'application/json' },
body: { text: feedbackText.value }
});
},
onSuccess(_, showToast) {
feedbackOpen.value = false;
feedbackText.value = '';
showToast('Feedback sent', { description: 'Feedback sent successfully', position: 'top-right' });
},
})
}
</script>
<template>
<SidebarProvider class="h-dvh">
<AppSidebar />
<SidebarInset>
<SidebarTrigger v-if="!projectStore.firstInteraction" class="ml-2 mt-2 lg:hidden" />
<header v-if="projectStore.firstInteraction" class="bg-sidebar border-border py-2 px-2 flex gap-2"
:class="{ 'border-b-2': sidebarHeaderVisible }">
<SidebarTrigger class="ml-2 md:mt-1 lg:hidden" />
<Drawer>
<DrawerTrigger class="lg:hidden"><Button variant="outline">
<Funnel class="size-4" /> Filters
</Button></DrawerTrigger>
<DrawerContent>
<DrawerHeader class="flex flex-row justify-between">
<div>
<DrawerTitle>Filters</DrawerTitle>
<DrawerDescription>Filter between your domains and select your Snapshot
</DrawerDescription>
</div>
<div>
<DrawerClose>
<Button>
Cancel
</Button>
</DrawerClose>
</div>
</DrawerHeader>
<div class="flex flex-row gap-4 justify-center p-4">
<SelectorDomainSelector v-if="sidebarHeaderVisible" class="w-[15rem]" />
<SelectorSnapshotSelector v-if="sidebarHeaderVisible" class="w-[15rem]" />
</div>
<DrawerFooter>
</DrawerFooter>
</DrawerContent>
</Drawer>
<SelectorDomainSelector v-if="sidebarHeaderVisible" class="w-[15rem] hidden lg:flex" />
<SelectorSnapshotSelector v-if="sidebarHeaderVisible" class="w-[15rem] hidden lg:flex" />
<div class="flex items-center justify-end grow pr-4 gap-4">
<NuxtLink to="https://docs.litlyx.com" target="_blank"> Docs </NuxtLink>
<Popover v-if="!isSelfhosted()" v-model:open="feedbackOpen">
<PopoverTrigger as-child>
<Button @click.prevent.stop variant="ghost" size="sm"
class="hover:!bg-sidebar-accent flex justify-start font-normal">
<MessageCircleMoreIcon class="size-4 text-muted-foreground" />
Feedback
</Button>
</PopoverTrigger>
<PopoverContent side="bottom" :side-offset='16'>
<div class="flex flex-col gap-4">
<Label>Share everything with us.</Label>
<Textarea v-model="feedbackText" placeholder="Leave your feedback here"
class="resize-none h-24" />
<Button @click="sendFeedback()">Send</Button>
<div class="text-center">
or email us at
<a class="underline" href="mailto:help@litlyx.com">help@litlyx.com</a>
</div>
</div>
</PopoverContent>
</Popover>
<TooltipProvider v-if="!isSelfhosted()">
<Tooltip>
<TooltipTrigger>
<NuxtLink
:to="[0].includes(premium.planInfo?.ID ?? -1) ? '/plans' : '/shareable_links'">
<Button variant="outline" :disabled="[0].includes(premium.planInfo?.ID ?? -1)">
<Lock v-if="[0].includes(premium.planInfo?.ID ?? -1)"
class="size-4 text-yellow-500" />
<Link v-else class="size-4" />
Share
</Button>
</NuxtLink>
</TooltipTrigger>
<TooltipContent>
<p> Share your project </p>
</TooltipContent>
</Tooltip>
</TooltipProvider>
<!-- <Popover v-model:open="feedbackOpen">
<PopoverTrigger>
<Button variant="outline"> Feedback </Button>
</PopoverTrigger>
<PopoverContent>
<div class="flex flex-col gap-4">
<Label> Share everything with us. </Label>
<Textarea v-model="feedbackText" placeholder="Leave your feedback here"
class="resize-none h-24"></Textarea>
<Button @click="sendFeedback()"> Send </Button>
</div>
</PopoverContent>
</Popover> -->
<!-- <Popover v-model:open="helpOpen">
<PopoverTrigger>
<div class="cursor-pointer"> Help </div>
</PopoverTrigger>
<PopoverContent :side-offset="10">
<div class="flex flex-col gap-4">
<Label> Contact support </Label>
<Label class="text-muted-foreground">
If you have any question or issue we are here to help you
</Label>
<div>
<div class="border-solid border-[1px] rounded-md px-2 py-1 relative">
<CopyIcon @click="copyEmail()"
class="size-4 absolute right-2 top-2 cursor-pointer"></CopyIcon>
<div class="poppins text-[.9rem]"> help@litlyx.com </div>
</div>
</div>
<Label class="text-muted-foreground">
or text us on Discord, we will reply to you personally.
</Label>
<NuxtLink to="https://discord.gg/tg7FHkffR7" target="_blank">
<Button variant="outline" size="sm" class="w-full">
<Icon class="text-2xl" name="ic:baseline-discord"></Icon>
</Button>
</NuxtLink>
</div>
</PopoverContent>
</Popover> -->
</div>
</header>
<div v-if="sidebarHeaderVisible" class="min-h-52 overflow-auto h-full">
<div v-if="plan && !(plan.premium_type === 7999 && !projectStore.activeProject?.guest)"
class="flex flex-1 flex-col gap-4 p-4 overflow-y-auto h-full bg-gray-100 dark:bg-background">
<slot></slot>
</div>
<div v-else>
<LazyFreeTrialEnded></LazyFreeTrialEnded>
</div>
</div>
<div v-if="!sidebarHeaderVisible" class="flex justify-center mt-[20vh] text-[1.3rem] poppins font-medium">
<Unauthorized authorization="At least 1 domain"
v-if="!domainStore.domainPending && domainStore.domains.length == 0">
</Unauthorized>
<div v-else>
<Loader></Loader>
</div>
</div>
</SidebarInset>
</SidebarProvider>
</template>

View File

@@ -1,122 +0,0 @@
<script setup lang="ts">
import type { Section } from '~/components/layout/VerticalNavigation.vue';
import { Lit } from 'litlyx-js';
import { DialogFeedback } from '#components';
const { userRoles, isLogged } = useLoggedUser();
const { project } = useProject();
const modal = useModal();
const selfhosted = useSelfhosted();
const sections: Section[] = [
{
title: '',
entries: [
{ label: 'Web Analytics', to: '/', icon: 'fal fa-table-layout' },
{ label: 'Custom Events', to: '/events', icon: 'fal fa-square-bolt' },
{ label: 'Members', to: '/members', icon: 'fal fa-users' },
{ label: 'Ask AI', to: '/analyst', icon: 'fal fa-sparkles' },
// { label: 'Security', to: '/security', icon: 'fal fa-shield', disabled: selfhosted },
// { label: 'Insights (soon)', to: '#', icon: 'fal fa-lightbulb', disabled: true },
// { label: 'Links (soon)', to: '#', icon: 'fal fa-globe-pointer', disabled: true },
// { label: 'Integrations (soon)', to: '/integrations', icon: 'fal fa-cube', disabled: true },
{ grow: true, label: 'Settings', to: '/settings', icon: 'fal fa-gear' },
// {
// grow: true,
// label: 'Leave a Feedback', icon: 'fal fa-message',
// action() {
// modal.open(DialogFeedback, {});
// },
// disabled: selfhosted
// },
// {
// grow: true,
// label: 'Documentation', to: 'https://docs.litlyx.com', icon: 'fal fa-book', external: true,
// action() { Lit.event('docs_clicked') },
// },
// {
// grow: true,
// label: 'Discord support', icon: 'fab fa-discord',
// to: 'https://discord.gg/9cQykjsmWX',
// external: true,
// },
// {
// label: 'Slack support', icon: 'fab fa-slack',
// to: '#',
// premiumOnly: true,
// action() {
// if (isLogged.value === true) return;
// if (userRoles.isPremium.value === true) {
// window.open('https://join.slack.com/t/litlyx/shared_invite/zt-2q3oawn29-hZlu_fBUBlc4052Ooe3FZg', '_blank');
// } else {
// pricingDrawer.visible.value = true;
// }
// },
// },
]
}
];
const { showDialog, closeDialog } = useBarCardDialog();
const { isOpen, close, open } = useMenu();
</script>
<template>
<div class="layout relative flex flex-col min-h-[100dvh] h-dvh w-dvw overflow-hidden">
<div
class="px-6 py-3 flex items-center justify-center dark:bg-lyx-background-light z-[20] rounded-xl mx-2 my-2 lg:hidden">
<i @click="open()" class="fas fa-bars text-[1.2rem] absolute left-6"></i>
<!-- <div class="nunito font-semibold text-[1.2rem]">
Litlyx
</div> -->
</div>
<div class="flex h-full overflow-y-hidden">
<div v-if="isOpen" @click="close()"
class="lg:hidden barrier bg-black/40 backdrop-blur-[2px] w-full h-full absolute left-0 top-0 z-[40]">
</div>
<LayoutVerticalNavigation :sections="sections">
</LayoutVerticalNavigation>
<div class="flex flex-col overflow-hidden w-full bg-lyx-lightmode-background dark:bg-lyx-background relative h-full">
<div v-if="showDialog" class="barrier w-full h-full z-[34] absolute bg-black/50 backdrop-blur-[2px]">
<i
class="z-[40] absolute right-12 top-8 fas fa-times text-lyx-lightmode-text-dark dark:text-text-sub text-[1.8rem] lg:text-[3rem]"></i>
</div>
<div @click="closeDialog()" class="w-full h-full z-[35] absolute top-0 left-0 px-4 lg:px-60 py-20"
v-if="showDialog">
<DashboardDialogBarCard @click.stop="null" class="z-[36]"></DashboardDialogBarCard>
</div>
<LayoutTopNavigation class="flex shrink-0"></LayoutTopNavigation>
<div class="flex-1 overflow-auto">
<slot></slot>
</div>
</div>
</div>
</div>
</template>

View File

@@ -1,21 +0,0 @@
<script setup lang="ts">
const { showDialog, closeDialog } = useBarCardDialog();
</script>
<template>
<div class="w-full h-full">
<div v-if="showDialog" class="barrier w-full h-full z-[34] absolute bg-black/50 backdrop-blur-[2px]">
<i class="z-[40] absolute right-12 top-8 fas fa-times text-text-sub text-[1.8rem] lg:text-[3rem]"></i>
</div>
<div @click="closeDialog()" class="w-full h-full z-[35] absolute top-0 left-0 px-4 lg:px-60 py-20"
v-if="showDialog">
<DashboardDialogBarCard @click.stop="null" class="z-[36]"></DashboardDialogBarCard>
</div>
<slot></slot>
</div>
</template>
<style scoped lang="scss"></style>