378 lines
19 KiB
Vue
378 lines
19 KiB
Vue
<script setup>
|
|
import { computed, onMounted, ref, watch } from 'vue';
|
|
import { Head, Link, router, usePage } from '@inertiajs/vue3';
|
|
import ApplicationMark from '@/Components/ApplicationMark.vue';
|
|
import Banner from '@/Components/Banner.vue';
|
|
import Dropdown from '@/Components/Dropdown.vue';
|
|
import DropdownLink from '@/Components/DropdownLink.vue';
|
|
import NavLink from '@/Components/NavLink.vue';
|
|
import ResponsiveNavLink from '@/Components/ResponsiveNavLink.vue';
|
|
import Breadcrumbs from '@/Components/Breadcrumbs.vue';
|
|
import axios from 'axios';
|
|
import { debounce } from 'lodash';
|
|
import { FwbButton, FwbInput, FwbListGroup, FwbListGroupItem } from 'flowbite-vue';
|
|
import { AdjustmentIcon, SearchIcon } from '@/Utilities/Icons';
|
|
|
|
const props = defineProps({
|
|
title: String,
|
|
});
|
|
|
|
const showingNavigationDropdown = ref(false);
|
|
const querySearchDiv = ref(null);
|
|
|
|
const querySearch = ref('');
|
|
|
|
const querySearchList = ref(true);
|
|
const querySearchResult = ref([]);
|
|
|
|
const logout = () => {
|
|
router.post(route('logout'));
|
|
};
|
|
|
|
const setRoute = (index) => {
|
|
return route('client.show', index);
|
|
}
|
|
|
|
const onSearchFocus = () => {
|
|
if(querySearch.value.length > 0) {
|
|
querySearchList.value = false;
|
|
}
|
|
}
|
|
|
|
const onSearchBlue = () => {
|
|
setTimeout(() => querySearchList.value = true, 300)
|
|
}
|
|
|
|
const searchDebounce = debounce((value) => {
|
|
axios.get(
|
|
route('search'),
|
|
{
|
|
params: {
|
|
query: value,
|
|
limit: 5,
|
|
tag: ''
|
|
}
|
|
}
|
|
)
|
|
.then(function(res) {
|
|
querySearchResult.value = res.data
|
|
querySearchList.value = false;
|
|
console.log(res);
|
|
})
|
|
.catch(function(error){
|
|
console.log(error)
|
|
})
|
|
.finally(function(){
|
|
|
|
});
|
|
}, 300);
|
|
|
|
watch(querySearch, (value) => {
|
|
searchDebounce(value);
|
|
});
|
|
|
|
|
|
</script>
|
|
|
|
<template>
|
|
<div>
|
|
<Head :title="title" />
|
|
|
|
<Banner />
|
|
|
|
<div class="min-h-screen bg-gray-100">
|
|
<nav class="bg-white border-b border-gray-100">
|
|
<!-- Primary Navigation Menu -->
|
|
<div class="max-w-7xl mx-auto px-4 sm:px-6 lg:px-8">
|
|
<div class="flex justify-between h-16">
|
|
<div class="flex">
|
|
<!-- Logo -->
|
|
<div class="shrink-0 flex items-center">
|
|
<Link :href="route('dashboard')">
|
|
<ApplicationMark class="block h-9 w-auto" />
|
|
</Link>
|
|
</div>
|
|
|
|
<!-- Navigation Links -->
|
|
<div class="hidden space-x-8 sm:-my-px sm:ms-10 sm:flex">
|
|
<NavLink :href="route('dashboard')" :active="route().current('dashboard')">
|
|
Nadzorna plošča
|
|
</NavLink>
|
|
</div>
|
|
|
|
<div class="hidden space-x-8 sm:-my-px sm:ms-10 sm:flex">
|
|
<NavLink :href="route('client')" :active="route().current('client') || route().current('client.*')">
|
|
Naročniki
|
|
</NavLink>
|
|
</div>
|
|
<div class="hidden space-x-8 sm:-my-px sm:ms-10 sm:flex">
|
|
<NavLink :href="route('clientCase')" :active="route().current('clientCase') || route().current('clientCase.*')">
|
|
Primeri
|
|
</NavLink>
|
|
</div>
|
|
<div class="hidden space-x-8 sm:-my-px sm:ms-10 sm:flex">
|
|
<NavLink :href="route('settings')" :active="route().current('settings') || route().current('settings.*')">
|
|
Nastavitve
|
|
</NavLink>
|
|
</div>
|
|
</div>
|
|
|
|
<div class="hidden sm:flex sm:items-center sm:ms-6">
|
|
<!-- Settings Dropdown -->
|
|
<div>
|
|
<fwb-input
|
|
v-model="querySearch"
|
|
placeholder="Iskalnik..."
|
|
size="md"
|
|
@focus="onSearchFocus"
|
|
@blur="onSearchBlue"
|
|
@input.stop="true"
|
|
>
|
|
<template #prefix>
|
|
<SearchIcon />
|
|
</template>
|
|
|
|
</fwb-input>
|
|
<fwb-list-group ref="querySearchDiv" class="absolute" :hidden="querySearchList">
|
|
<fwb-list-group-item class="px-0 flex flex-col">
|
|
|
|
<p>Naročniki:</p>
|
|
<fwb-list-group>
|
|
<fwb-list-group-item hover v-for="client in querySearchResult.clients">
|
|
<a :href="route('client.show', {uuid: client.client_uuid})">{{ client.full_name }}</a>
|
|
</fwb-list-group-item>
|
|
</fwb-list-group>
|
|
|
|
</fwb-list-group-item>
|
|
<fwb-list-group-item class="px-0 flex flex-col">
|
|
|
|
<p>Primeri:</p>
|
|
<fwb-list-group>
|
|
<fwb-list-group-item hover v-for="clientCase in querySearchResult.client_cases">
|
|
<a :href="route('clientCase.show', {uuid: clientCase.case_uuid})">{{ clientCase.full_name }}</a>
|
|
</fwb-list-group-item>
|
|
</fwb-list-group>
|
|
|
|
</fwb-list-group-item>
|
|
</fwb-list-group>
|
|
</div>
|
|
<div class="ms-3 relative">
|
|
<Dropdown align="right" width="48">
|
|
<template #trigger>
|
|
<button v-if="$page.props.jetstream.managesProfilePhotos" class="flex text-sm border-2 border-transparent rounded-full focus:outline-none focus:border-gray-300 transition">
|
|
<img class="h-8 w-8 rounded-full object-cover" :src="$page.props.auth.user.profile_photo_url" :alt="$page.props.auth.user.name">
|
|
</button>
|
|
|
|
<span v-else class="inline-flex rounded-md">
|
|
<button type="button" class="inline-flex items-center px-3 py-2 border border-transparent text-sm leading-4 font-medium rounded-md text-gray-500 bg-white hover:text-gray-700 focus:outline-none focus:bg-gray-50 active:bg-gray-50 transition ease-in-out duration-150">
|
|
{{ $page.props.auth.user.name }}
|
|
<svg class="ms-2 -me-0.5 h-4 w-4" xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor">
|
|
<path stroke-linecap="round" stroke-linejoin="round" d="M19.5 8.25l-7.5 7.5-7.5-7.5" />
|
|
</svg>
|
|
</button>
|
|
</span>
|
|
</template>
|
|
|
|
<template #content>
|
|
<!-- Account Management -->
|
|
<div class="block px-4 py-2 text-xs text-gray-400">
|
|
Nastavitve računa
|
|
</div>
|
|
|
|
<DropdownLink :href="route('profile.show')">
|
|
Profil
|
|
</DropdownLink>
|
|
|
|
<DropdownLink v-if="$page.props.jetstream.hasApiFeatures" :href="route('api-tokens.index')">
|
|
API Tokens
|
|
</DropdownLink>
|
|
|
|
<div class="border-t border-gray-200" />
|
|
|
|
<!-- Authentication -->
|
|
<form @submit.prevent="logout">
|
|
<DropdownLink as="button">
|
|
Izpis
|
|
</DropdownLink>
|
|
</form>
|
|
</template>
|
|
</Dropdown>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Hamburger -->
|
|
<div class="-me-2 flex items-center sm:hidden">
|
|
<button class="inline-flex items-center justify-center p-2 rounded-md text-gray-400 hover:text-gray-500 hover:bg-gray-100 focus:outline-none focus:bg-gray-100 focus:text-gray-500 transition duration-150 ease-in-out" @click="showingNavigationDropdown = ! showingNavigationDropdown">
|
|
<svg
|
|
class="h-6 w-6"
|
|
stroke="currentColor"
|
|
fill="none"
|
|
viewBox="0 0 24 24"
|
|
>
|
|
<path
|
|
:class="{'hidden': showingNavigationDropdown, 'inline-flex': ! showingNavigationDropdown }"
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
stroke-width="2"
|
|
d="M4 6h16M4 12h16M4 18h16"
|
|
/>
|
|
<path
|
|
:class="{'hidden': ! showingNavigationDropdown, 'inline-flex': showingNavigationDropdown }"
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
stroke-width="2"
|
|
d="M6 18L18 6M6 6l12 12"
|
|
/>
|
|
</svg>
|
|
</button>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Responsive Navigation Menu -->
|
|
<div :class="{'block': showingNavigationDropdown, 'hidden': ! showingNavigationDropdown}" class="sm:hidden">
|
|
<div class="pt-2 pb-3 space-y-1">
|
|
<ResponsiveNavLink :href="route('dashboard')" :active="route().current('dashboard')">
|
|
Nadzorna plošča
|
|
</ResponsiveNavLink>
|
|
<ResponsiveNavLink :href="route('client')" :active="route().current('client')">
|
|
Naročniki
|
|
</ResponsiveNavLink>
|
|
<ResponsiveNavLink :href="route('clientCase')" :active="route().current('clientCase')">
|
|
Primeri
|
|
</ResponsiveNavLink>
|
|
<ResponsiveNavLink :href="route('settings')" :active="route().current('settings')">
|
|
Nastavitve
|
|
</ResponsiveNavLink>
|
|
</div>
|
|
<!-- Responsive Settings Options -->
|
|
<div class="pt-4 pb-1 border-t border-gray-200">
|
|
<div class="flex items-center px-4">
|
|
<div v-if="$page.props.jetstream.managesProfilePhotos" class="shrink-0 me-3">
|
|
<img class="h-10 w-10 rounded-full object-cover" :src="$page.props.auth.user.profile_photo_url" :alt="$page.props.auth.user.name">
|
|
</div>
|
|
|
|
<div>
|
|
<div class="font-medium text-base text-gray-800">
|
|
{{ $page.props.auth.user.name }}
|
|
</div>
|
|
<div class="font-medium text-sm text-gray-500">
|
|
{{ $page.props.auth.user.email }}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<div class="mt-3 space-y-1">
|
|
<ResponsiveNavLink :href="route('profile.show')" :active="route().current('profile.show')">
|
|
Profil
|
|
</ResponsiveNavLink>
|
|
|
|
<ResponsiveNavLink v-if="$page.props.jetstream.hasApiFeatures" :href="route('api-tokens.index')" :active="route().current('api-tokens.index')">
|
|
API Tokens
|
|
</ResponsiveNavLink>
|
|
|
|
<!-- Authentication -->
|
|
<form method="POST" @submit.prevent="logout">
|
|
<ResponsiveNavLink as="button">
|
|
Izpis
|
|
</ResponsiveNavLink>
|
|
</form>
|
|
|
|
<!-- Team Management -->
|
|
<template v-if="$page.props.jetstream.hasTeamFeatures">
|
|
<div class="border-t border-gray-200" />
|
|
|
|
<div class="block px-4 py-2 text-xs text-gray-400">
|
|
Manage Team
|
|
</div>
|
|
|
|
<!-- Team Settings -->
|
|
<ResponsiveNavLink :href="route('teams.show', $page.props.auth.user.current_team)" :active="route().current('teams.show')">
|
|
Team Settings
|
|
</ResponsiveNavLink>
|
|
|
|
<ResponsiveNavLink v-if="$page.props.jetstream.canCreateTeams" :href="route('teams.create')" :active="route().current('teams.create')">
|
|
Create New Team
|
|
</ResponsiveNavLink>
|
|
|
|
<!-- Team Switcher -->
|
|
<template v-if="$page.props.auth.user.all_teams.length > 1">
|
|
<div class="border-t border-gray-200" />
|
|
|
|
<div class="block px-4 py-2 text-xs text-gray-400">
|
|
Switch Teams
|
|
</div>
|
|
|
|
<template v-for="team in $page.props.auth.user.all_teams" :key="team.id">
|
|
<form @submit.prevent="switchToTeam(team)">
|
|
<ResponsiveNavLink as="button">
|
|
<div class="flex items-center">
|
|
<svg v-if="team.id == $page.props.auth.user.current_team_id" class="me-2 h-5 w-5 text-green-400" xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor">
|
|
<path stroke-linecap="round" stroke-linejoin="round" d="M9 12.75L11.25 15 15 9.75M21 12a9 9 0 11-18 0 9 9 0 0118 0z" />
|
|
</svg>
|
|
<div>{{ team.name }}</div>
|
|
</div>
|
|
</ResponsiveNavLink>
|
|
</form>
|
|
</template>
|
|
</template>
|
|
</template>
|
|
</div>
|
|
<!--div class="px-2 w-full relative">
|
|
<fwb-input
|
|
v-model="querySearch"
|
|
placeholder="Iskalnik..."
|
|
size="md"
|
|
@focus="onSearchFocus"
|
|
@blur="onSearchBlue"
|
|
@input.stop="true"
|
|
>
|
|
<template #prefix>
|
|
<SearchIcon />
|
|
</template>
|
|
|
|
</fwb-input>
|
|
<fwb-list-group ref="querySearchDiv" class="absolute px-2 w-full mx-auto" :hidden="querySearchList">
|
|
<fwb-list-group-item class="px-0 flex flex-col">
|
|
|
|
<p>Naročniki:</p>
|
|
<fwb-list-group class="w-full">
|
|
<fwb-list-group-item hover v-for="client in querySearchResult.clients">
|
|
<a :href="route('client.show', {uuid: client.client_uuid})">{{ client.full_name }}</a>
|
|
</fwb-list-group-item>
|
|
</fwb-list-group>
|
|
|
|
</fwb-list-group-item>
|
|
<fwb-list-group-item class="px-0 flex flex-col">
|
|
|
|
<p>Primeri:</p>
|
|
<fwb-list-group class="w-full">
|
|
<fwb-list-group-item hover v-for="clientCase in querySearchResult.client_cases">
|
|
<a :href="route('clientCase.show', {uuid: clientCase.case_uuid})">{{ clientCase.full_name }}</a>
|
|
</fwb-list-group-item>
|
|
</fwb-list-group>
|
|
|
|
</fwb-list-group-item>
|
|
</fwb-list-group>
|
|
</div-->
|
|
</div>
|
|
</div>
|
|
</nav>
|
|
|
|
<!-- Page Heading -->
|
|
<header v-if="$slots.header" class="bg-white shadow ">
|
|
<div class="max-w-7xl mx-auto py-6 px-4 sm:px-6 lg:px-8">
|
|
<Breadcrumbs :breadcrumbs="$page.props.breadcrumbs"></Breadcrumbs>
|
|
|
|
</div>
|
|
|
|
</header>
|
|
|
|
<!-- Page Content -->
|
|
<main>
|
|
<slot />
|
|
</main>
|
|
</div>
|
|
</div>
|
|
</template>
|