mirror of
https://github.com/twinpath/app.git
synced 2026-01-26 05:15:28 +07:00
171 lines
9.0 KiB
PHP
171 lines
9.0 KiB
PHP
{{-- Notification Dropdown Component --}}
|
|
<div class="relative" x-data="{
|
|
dropdownOpen: false,
|
|
notifying: false,
|
|
notifications: [],
|
|
init() {
|
|
this.fetchNotifications();
|
|
|
|
// Listen for realtime notifications
|
|
if (window.Echo) {
|
|
window.Echo.private('App.Models.User.{{ auth()->id() }}')
|
|
.notification((notification) => {
|
|
this.notifications.unshift({
|
|
id: notification.id,
|
|
data: {
|
|
title: notification.data?.title || notification.title || 'Notification',
|
|
body: notification.data?.body || notification.body || '',
|
|
icon: notification.data?.icon || notification.icon,
|
|
action_url: notification.data?.action_url || notification.action_url
|
|
},
|
|
created_at_human: 'Just now',
|
|
read_url: notification.read_url || '#'
|
|
});
|
|
this.notifying = true;
|
|
|
|
// Dispatch Global Event for Toast Alerts
|
|
window.dispatchEvent(new CustomEvent('reverb-notification', {
|
|
detail: notification.data?.title ? notification.data : notification
|
|
}));
|
|
});
|
|
}
|
|
},
|
|
fetchNotifications() {
|
|
fetch('{{ route('notifications.unread') }}')
|
|
.then(response => response.json())
|
|
.then(data => {
|
|
this.notifications = data.notifications;
|
|
this.notifying = data.count > 0;
|
|
});
|
|
},
|
|
toggleDropdown() {
|
|
this.dropdownOpen = !this.dropdownOpen;
|
|
},
|
|
closeDropdown() {
|
|
this.dropdownOpen = false;
|
|
}
|
|
}" @click.away="closeDropdown()">
|
|
<!-- Notification Button -->
|
|
<button
|
|
class="relative flex items-center justify-center text-gray-500 transition-colors bg-white border border-gray-200 rounded-full hover:text-dark-900 h-11 w-11 hover:bg-gray-100 hover:text-gray-700 dark:border-gray-800 dark:bg-gray-900 dark:text-gray-400 dark:hover:bg-gray-800 dark:hover:text-white"
|
|
@click="toggleDropdown()"
|
|
type="button"
|
|
>
|
|
<!-- Notification Badge -->
|
|
<span
|
|
x-show="notifying"
|
|
class="absolute right-0 top-0.5 z-1 h-2 w-2 rounded-full bg-orange-400"
|
|
style="display: none;"
|
|
>
|
|
<span
|
|
class="absolute inline-flex w-full h-full bg-orange-400 rounded-full opacity-75 -z-1 animate-ping"
|
|
></span>
|
|
</span>
|
|
|
|
<!-- Bell Icon -->
|
|
<svg
|
|
class="fill-current"
|
|
width="20"
|
|
height="20"
|
|
viewBox="0 0 20 20"
|
|
fill="none"
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
>
|
|
<path
|
|
fill-rule="evenodd"
|
|
clip-rule="evenodd"
|
|
d="M10.75 2.29248C10.75 1.87827 10.4143 1.54248 10 1.54248C9.58583 1.54248 9.25004 1.87827 9.25004 2.29248V2.83613C6.08266 3.20733 3.62504 5.9004 3.62504 9.16748V14.4591H3.33337C2.91916 14.4591 2.58337 14.7949 2.58337 15.2091C2.58337 15.6234 2.91916 15.9591 3.33337 15.9591H4.37504H15.625H16.6667C17.0809 15.9591 17.4167 15.6234 17.4167 15.2091C17.4167 14.7949 17.0809 14.4591 16.6667 14.4591H16.375V9.16748C16.375 5.9004 13.9174 3.20733 10.75 2.83613V2.29248ZM14.875 14.4591V9.16748C14.875 6.47509 12.6924 4.29248 10 4.29248C7.30765 4.29248 5.12504 6.47509 5.12504 9.16748V14.4591H14.875ZM8.00004 17.7085C8.00004 18.1228 8.33583 18.4585 8.75004 18.4585H11.25C11.6643 18.4585 12 18.1228 12 17.7085C12 17.2943 11.6643 16.9585 11.25 16.9585H8.75004C8.33583 16.9585 8.00004 17.2943 8.00004 17.7085Z"
|
|
fill=""
|
|
/>
|
|
</svg>
|
|
</button>
|
|
|
|
<!-- Dropdown Start -->
|
|
<div
|
|
x-show="dropdownOpen"
|
|
x-transition:enter="transition ease-out duration-100"
|
|
x-transition:enter-start="transform opacity-0 scale-95"
|
|
x-transition:enter-end="transform opacity-100 scale-100"
|
|
x-transition:leave="transition ease-in duration-75"
|
|
x-transition:leave-start="transform opacity-100 scale-100"
|
|
x-transition:leave-end="transform opacity-0 scale-95"
|
|
class="absolute -right-[240px] mt-[17px] flex h-auto max-h-[480px] w-[350px] flex-col rounded-2xl border border-gray-200 bg-white p-3 shadow-theme-lg dark:border-gray-800 dark:bg-gray-dark sm:w-[361px] lg:right-0"
|
|
style="display: none;"
|
|
>
|
|
<!-- Dropdown Header -->
|
|
<div class="flex items-center justify-between pb-3 mb-3 border-b border-gray-100 dark:border-gray-800">
|
|
<h5 class="text-lg font-semibold text-gray-800 dark:text-white/90">Notification</h5>
|
|
|
|
<button @click="closeDropdown()" class="text-gray-500 dark:text-gray-400" type="button">
|
|
<svg
|
|
class="fill-current"
|
|
width="24"
|
|
height="24"
|
|
viewBox="0 0 24 24"
|
|
fill="none"
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
>
|
|
<path
|
|
fill-rule="evenodd"
|
|
clip-rule="evenodd"
|
|
d="M6.21967 7.28131C5.92678 6.98841 5.92678 6.51354 6.21967 6.22065C6.51256 5.92775 6.98744 5.92775 7.28033 6.22065L11.999 10.9393L16.7176 6.22078C17.0105 5.92789 17.4854 5.92788 17.7782 6.22078C18.0711 6.51367 18.0711 6.98855 17.7782 7.28144L13.0597 12L17.7782 16.7186C18.0711 17.0115 18.0711 17.4863 17.7782 17.7792C17.4854 18.0721 17.0105 18.0721 16.7176 17.7792L11.999 13.0607L7.28033 17.7794C6.98744 18.0722 6.51356 18.0722 6.21967 17.7794C5.92678 17.4865 5.92678 17.0116 6.21967 16.7187L10.9384 12L6.21967 7.28131Z"
|
|
fill=""
|
|
/>
|
|
</svg>
|
|
</button>
|
|
</div>
|
|
|
|
<!-- Notification List -->
|
|
<ul class="flex flex-col h-auto overflow-y-auto custom-scrollbar max-h-[300px]">
|
|
<template x-for="notification in notifications" :key="notification.id">
|
|
<li>
|
|
<a
|
|
class="flex gap-3 rounded-lg border-b border-gray-100 p-3 px-4.5 py-3 hover:bg-gray-100 dark:border-gray-800 dark:hover:bg-white/5"
|
|
:href="notification.read_url"
|
|
>
|
|
<div class="flex-shrink-0">
|
|
<div class="w-10 h-10 rounded-full bg-brand-100 text-brand-500 flex items-center justify-center">
|
|
<template x-if="notification.data.icon">
|
|
<svg class="w-6 h-6" fill="none" stroke="currentColor" viewBox="0 0 24 24"><path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M15 17h5l-1.405-1.405A2.032 2.032 0 0118 14.158V11a6.002 6.002 0 00-4-5.659V5a2 2 0 10-4 0v.341C7.67 6.165 6 8.388 6 11v3.159c0 .538-.214 1.055-.595 1.436L4 17h5m6 0v1a3 3 0 11-6 0v-1m6 0H9"></path></svg>
|
|
</template>
|
|
<template x-if="!notification.data.icon">
|
|
<svg class="w-6 h-6" fill="none" stroke="currentColor" viewBox="0 0 24 24"><path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M13 16h-1v-4h-1m1-4h.01M21 12a9 9 0 11-18 0 9 9 0 0118 0z"></path></svg>
|
|
</template>
|
|
</div>
|
|
</div>
|
|
|
|
<span class="block">
|
|
<span class="mb-1.5 block text-theme-sm text-gray-500 dark:text-gray-400">
|
|
<span class="font-medium text-gray-800 dark:text-white/90" x-text="notification.data.title"></span>
|
|
<span class="block text-xs mt-1" x-text="notification.data.body"></span>
|
|
</span>
|
|
|
|
<span class="flex items-center gap-2 text-gray-500 text-theme-xs dark:text-gray-400">
|
|
<span x-text="notification.created_at_human"></span>
|
|
</span>
|
|
</span>
|
|
</a>
|
|
</li>
|
|
</template>
|
|
|
|
<li x-show="notifications.length === 0" class="p-6 text-center text-gray-500 dark:text-gray-400">
|
|
<p>No new notifications.</p>
|
|
</li>
|
|
</ul>
|
|
|
|
<div x-show="notifications.length > 0">
|
|
<!-- Mark All Read Button -->
|
|
<form action="{{ route('notifications.readAll') }}" method="POST" class="mt-3">
|
|
@csrf
|
|
<button
|
|
type="submit"
|
|
class="flex w-full justify-center rounded-lg border border-gray-300 bg-white p-3 text-theme-sm font-medium text-gray-700 shadow-theme-xs hover:bg-gray-50 hover:text-gray-800 dark:border-gray-700 dark:bg-gray-800 dark:text-gray-400 dark:hover:bg-white/[0.03] dark:hover:text-gray-200"
|
|
>
|
|
Mark All as Read
|
|
</button>
|
|
</form>
|
|
</div>
|
|
</div>
|
|
<!-- Dropdown End -->
|
|
</div>
|