Files
BarangaySystem/resources/js/Pages/ViewStoreMarket.vue
2026-06-06 18:43:00 +08:00

581 lines
20 KiB
Vue

<script setup>
import { usePageTitle } from '../composables/Core/usePageTitle';
usePageTitle('View Store Market');
import { ref, computed, onMounted } from 'vue';
import axios from 'axios';
import { useNavigate } from '../composables/Core/useNavigate';
import LoadingSpinner from '../Components/LoadingSpinner.vue';
import ProductCard from '../Components/Market/ProductCard.vue';
import FileImage from '../Components/Core/FileImage.vue';
import { useAuth } from '../composables/Core/useAuth';
import { useModal } from '../composables/Core/useModal';
import BackButton from '../Components/Core/BackButton.vue';
const props = defineProps({
target: { type: String, required: true }
});
const { navigate } = useNavigate();
const { isUltimate, hasRole, UserTypes } = useAuth();
const modal = useModal();
const store = ref(null);
const loading = ref(true);
const error = ref(null);
const showGallery = ref(false);
const galleryIndex = ref(0);
const isBig3 = computed(() => isUltimate.value || hasRole('super operator') || hasRole('operator'));
const photos = computed(() => store.value?.resolved_photos ?? []);
const hasPhotos = computed(() => photos.value.length > 0);
const fetchStoreDetails = async () => {
loading.value = true;
error.value = null;
try {
const response = await axios.post('/View/Store/Details/data', {
target: props.target
});
if (response.data && response.data.success && response.data.data) {
store.value = response.data.data;
} else {
error.value = 'Store not found';
}
} catch (e) {
console.error('Failed to fetch store details:', e);
error.value = 'Failed to load store details';
} finally {
loading.value = false;
}
};
const goBack = () => {
navigate({ page: 'ListStores' });
};
const viewProduct = (product) => {
navigate({
page: 'BuyViewProductMarket',
props: {
payload: {
product_hash: product.hashkey,
store_hash: props.target
}
}
});
};
const viewAllPhotos = () => {
if (hasPhotos.value) {
galleryIndex.value = 0;
showGallery.value = true;
}
};
const openGallery = (index = 0) => {
if (hasPhotos.value) {
galleryIndex.value = index;
showGallery.value = true;
}
};
const prevPhoto = () => {
if (photos.value.length === 0) return;
galleryIndex.value = (galleryIndex.value - 1 + photos.value.length) % photos.value.length;
};
const nextPhoto = () => {
if (photos.value.length === 0) return;
galleryIndex.value = (galleryIndex.value + 1) % photos.value.length;
};
const editStore = () => {
navigate({
page: 'EditStoreUltimate',
props: { target: props.target }
});
};
const navigateToPOS = () => {
navigate({
page: 'PosMain',
props: { target: props.target }
});
};
const navigateToPosHistory = () => {
navigate({
page: 'PosHistory',
props: {
target: props.target,
storeName: store.value?.name
}
});
};
const reportStore = () => {
// Placeholder for report functionality
modal.open({
title: 'Report Store',
body: `Reporting store ${store.value?.name}`
});
};
const addProduct = () => {
navigate({
page: 'AddProductsToStore',
props: { target: props.target }
});
};
const assignProduct = () => {
navigate({
page: 'AddProductsToStore',
props: { target: props.target }
});
};
const toggleDescription = () => {
const desc = document.querySelector('.description');
if (desc) {
desc.classList.toggle('text-truncate-3');
}
};
onMounted(() => {
fetchStoreDetails();
});
</script>
<template>
<div class="store-details-page pb-5">
<div v-if="loading" class="text-center py-5">
<LoadingSpinner />
<p class="mt-3 text-muted">Loading store...</p>
</div>
<div v-else-if="error" class="tf-container mt-5 text-center">
<div class="alert alert-danger">{{ error }}</div>
<BackButton to="ListStores" text="Go Back" />
</div>
<template v-else-if="store">
<!-- Store Profile Header -->
<div class="store-header shadow-sm">
<div class="store-banner" @click="hasPhotos ? openGallery(0) : null" :style="{ cursor: hasPhotos ? 'pointer' : 'default' }">
<BackButton
to="ListStores"
className="banner-back-btn"
/>
<FileImage
:src="hasPhotos ? photos[0] : 'https://cdn.jsdelivr.net/gh/telemagnadon/obj-vault-3a@v2026.05.14-vendor-2/a/85605eacd4c8.bin'"
class="banner-img"
fallback="https://cdn.jsdelivr.net/gh/telemagnadon/obj-vault-3a@v2026.05.14-vendor-2/a/85605eacd4c8.bin"
/>
</div>
<div class="tf-container store-profile">
<div class="profile-card glass-card">
<div class="d-flex align-items-center">
<div class="store-avatar shadow" @click="hasPhotos ? openGallery(0) : null" :style="{ cursor: hasPhotos ? 'pointer' : 'default' }">
<FileImage
:src="hasPhotos ? photos[0] : 'https://cdn.jsdelivr.net/gh/telemagnadon/obj-vault-3a@v2026.05.14-vendor-2/a/85605eacd4c8.bin'"
class="avatar-img"
fallback="https://cdn.jsdelivr.net/gh/telemagnadon/obj-vault-3a@v2026.05.14-vendor-2/a/85605eacd4c8.bin"
/>
</div>
<div class="ms-3 pt-2">
<h3 class="fw_7 mb-0">{{ store.name }}</h3>
<div class="d-flex align-items-center flex-wrap">
<div class="text-muted small me-3">
<i class="fas fa-map-marker-alt me-1"></i> {{ store.address }}
</div>
<div v-if="store.category" class="badge bg-soft-secondary px-2 py-1 rounded-pill text-secondary small me-2">
{{ store.category }}
</div>
<div v-if="store.subcategory" class="badge bg-soft-info px-2 py-1 rounded-pill text-info small">
{{ store.subcategory }}
</div>
</div>
</div>
</div>
<div class="mt-3 text-muted description text-truncate-3" id="storeDescription">
{{ store.description }}
</div>
<!-- Store Actions -->
<div class="store-actions mt-4 pt-3 border-top">
<div class="row g-2 justify-content-center">
<div class="col-6 col-sm-3">
<button @click="toggleDescription" class="action-btn">
<div class="icon-wrapper">
<img src="https://cdn.jsdelivr.net/gh/telemagnadon/obj-vault-3a@v2026.05.14-vendor-2/a/a72b90763114.bin" alt="Details">
</div>
<span>Details</span>
</button>
</div>
<div class="col-6 col-sm-3">
<button @click="reportStore" class="action-btn">
<div class="icon-wrapper">
<img src="https://cdn.jsdelivr.net/gh/telemagnadon/obj-vault-3a@v2026.05.14-vendor-2/a/5b3227c8ca50.bin" alt="Report">
</div>
<span>Report</span>
</button>
</div>
<div v-if="hasPhotos" class="col-6 col-sm-3">
<button @click="openGallery(0)" class="action-btn photos-btn">
<div class="icon-wrapper">
<i class="fas fa-images" style="font-size: 22px; color: #8b5cf6;"></i>
</div>
<span>Photos</span>
</button>
</div>
<div v-if="isBig3 || store.can_add_product" class="col-6 col-sm-3">
<button @click="addProduct" class="action-btn add-product-btn">
<div class="icon-wrapper">
<img src="https://cdn.jsdelivr.net/gh/telemagnadon/obj-vault-3a@v2026.05.14-vendor-2/a/2596d5468e30.bin" alt="Add Product">
</div>
<span>Add Product</span>
</button>
</div>
<div v-if="store.can_assign_product" class="col-6 col-sm-3">
<button @click="assignProduct" class="action-btn assign-btn">
<div class="icon-wrapper">
<i class="fas fa-link" style="font-size: 22px; color: #f59e0b;"></i>
</div>
<span>Assign Product</span>
</button>
</div>
<div v-if="isBig3" class="col-6 col-sm-3">
<button @click="editStore" class="action-btn manage-btn">
<div class="icon-wrapper">
<img src="https://cdn.jsdelivr.net/gh/telemagnadon/obj-vault-3a@v2026.05.14-vendor-2/a/d3ac5ee9c253.bin" alt="Manage">
</div>
<span>Manage</span>
</button>
</div>
<div v-if="store.can_access_pos" class="col-6 col-sm-3">
<button @click="navigateToPOS" class="action-btn pos-btn">
<div class="icon-wrapper">
<i class="fas fa-cash-register" style="font-size: 22px; color: #10b981;"></i>
</div>
<span>POS</span>
</button>
</div>
<div v-if="store.can_access_pos" class="col-6 col-sm-3">
<button @click="navigateToPosHistory" class="action-btn history-btn">
<div class="icon-wrapper">
<i class="fas fa-history" style="font-size: 22px; color: #6366f1;"></i>
</div>
<span>History</span>
</button>
</div>
</div>
</div>
</div>
</div>
</div>
<div class="tf-container mt-4">
<div class="d-flex align-items-center justify-content-between mb-4 mt-2">
<h4 class="fw_7 mb-0">Our Products</h4>
<span class="badge bg-soft-primary px-3 py-2 rounded-pill text-primary">
{{ store.products?.length || 0 }} Items
</span>
</div>
<div v-if="!store.products || store.products.length === 0" class="text-center py-5 bg-light rounded-20">
<p class="text-muted mb-0">No products available in this store yet.</p>
</div>
<div v-else class="row g-3">
<div v-for="product in store.products" :key="product.hashkey" class="col-6 col-md-4 col-lg-3">
<ProductCard :name="product.name" :price="product.store_price || product.price"
:unit="product.unitname" :image="product.photourl ? product.photourl[0] : ''"
@click="viewProduct(product)" />
</div>
</div>
</div>
</template>
<!-- Photo Gallery Modal -->
<Teleport to="body">
<div v-if="showGallery" class="photo-gallery-overlay" @click.self="showGallery = false">
<button class="gallery-close" @click="showGallery = false">
<i class="fas fa-times"></i>
</button>
<button v-if="photos.length > 1" class="gallery-nav gallery-prev" @click="prevPhoto">
<i class="fas fa-chevron-left"></i>
</button>
<div class="gallery-img-wrap">
<img :src="photos[galleryIndex]" class="gallery-img" :alt="store.name + ' photo ' + (galleryIndex + 1)" />
<div class="gallery-counter">{{ galleryIndex + 1 }} / {{ photos.length }}</div>
</div>
<button v-if="photos.length > 1" class="gallery-nav gallery-next" @click="nextPhoto">
<i class="fas fa-chevron-right"></i>
</button>
<div v-if="photos.length > 1" class="gallery-thumbs">
<img
v-for="(url, i) in photos"
:key="i"
:src="url"
class="gallery-thumb"
:class="{ active: i === galleryIndex }"
@click="galleryIndex = i"
/>
</div>
</div>
</Teleport>
</div>
</template>
<style scoped>
.store-banner {
position: relative;
width: 100%;
height: 180px;
}
.banner-img {
width: 100%;
height: 100%;
object-fit: cover;
filter: brightness(0.8);
}
.back-btn {
position: absolute;
top: 20px;
left: 20px;
z-index: 10;
width: 36px;
height: 36px;
border-radius: 50%;
border: none;
background: white;
display: flex;
align-items: center;
justify-content: center;
color: #333;
}
.store-profile {
margin-top: -50px;
position: relative;
z-index: 20;
}
.profile-card {
background: var(--bg-card);
border-radius: 20px;
padding: 20px;
box-shadow: 0 4px 15px rgba(0, 0, 0, 0.05);
}
.store-avatar {
width: 80px;
height: 80px;
border-radius: 15px;
overflow: hidden;
border: 4px solid white;
background: white;
}
.avatar-img {
width: 100%;
height: 100%;
object-fit: cover;
}
.description {
line-height: 1.5;
font-size: 0.9rem;
}
.rounded-20 {
border-radius: 20px;
}
.badge.bg-soft-primary {
background-color: rgba(66, 185, 131, 0.1);
}
.text-truncate-3 {
display: -webkit-box;
-webkit-line-clamp: 3;
line-clamp: 3;
-webkit-box-orient: vertical;
overflow: hidden;
}
.banner-back-btn {
position: absolute;
top: 20px;
left: 20px;
z-index: 10;
}
:global(.dark-mode) .profile-card {
background: #24272c;
}
:global(.dark-mode) .back-btn {
background: #24272c;
color: #fff;
}
:global(.dark-mode) .store-avatar {
border-color: #24272c;
}
:global(.dark-mode) .bg-light {
background-color: #1a1c20 !important;
}
.store-actions .action-btn {
width: 100%;
background: transparent;
border: none;
display: flex;
flex-direction: column;
align-items: center;
gap: 8px;
padding: 10px;
border-radius: 12px;
transition: all 0.2s ease;
}
.store-actions .action-btn:hover {
background: rgba(0, 0, 0, 0.03);
}
.store-actions .icon-wrapper {
width: 44px;
height: 44px;
background: var(--accent-soft);
border-radius: 12px;
display: flex;
align-items: center;
justify-content: center;
transition: all 0.3s ease;
box-shadow: 0 2px 5px rgba(0,0,0,0.05);
border: 1px solid var(--border-color);
}
.store-actions .icon-wrapper img {
width: 24px;
height: 24px;
object-fit: contain;
}
.store-actions span {
font-size: 0.75rem;
font-weight: 600;
color: #64748b;
}
.manage-btn .icon-wrapper {
background: rgba(37, 99, 235, 0.1);
}
.manage-btn span {
color: #2563eb;
}
.pos-btn .icon-wrapper {
background: rgba(16, 185, 129, 0.1);
}
.pos-btn span {
color: #10b981;
}
.history-btn .icon-wrapper {
background: rgba(99, 102, 241, 0.1);
}
.history-btn span {
color: #6366f1;
}
.assign-btn .icon-wrapper {
background: rgba(245, 158, 11, 0.1);
}
.assign-btn span {
color: #f59e0b;
}
.action-btn:active .icon-wrapper {
transform: scale(0.9);
}
.bg-soft-secondary {
background-color: rgba(100, 116, 139, 0.1);
}
.bg-soft-info {
background-color: rgba(14, 165, 233, 0.1);
}
:global(.dark-mode) .store-actions .icon-wrapper {
background: #2d333b;
}
:global(.dark-mode) .store-actions .action-btn:hover {
background: rgba(255, 255, 255, 0.03);
}
:global(.dark-mode) .store-actions span {
color: #94a3b8;
}
:global(.dark-mode) .manage-btn span {
color: #60a5fa;
}
.photos-btn .icon-wrapper { background: rgba(139, 92, 246, 0.1); }
.photos-btn span { color: #8b5cf6; }
.photo-gallery-overlay {
position: fixed; inset: 0; z-index: 9999;
background: rgba(0,0,0,0.92);
display: flex; flex-direction: column;
align-items: center; justify-content: center;
}
.gallery-close {
position: absolute; top: 16px; right: 16px;
background: rgba(255,255,255,0.15); border: none;
color: #fff; border-radius: 50%; width: 40px; height: 40px;
font-size: 18px; display: flex; align-items: center; justify-content: center;
cursor: pointer; z-index: 10000;
}
.gallery-img-wrap { position: relative; max-width: 90vw; max-height: 70vh; display: flex; align-items: center; justify-content: center; }
.gallery-img { max-width: 90vw; max-height: 70vh; object-fit: contain; border-radius: 10px; }
.gallery-counter {
position: absolute; bottom: -28px; left: 50%; transform: translateX(-50%);
color: rgba(255,255,255,0.7); font-size: 0.8rem;
}
.gallery-nav {
position: absolute; top: 50%; transform: translateY(-50%);
background: rgba(255,255,255,0.15); border: none; color: #fff;
border-radius: 50%; width: 44px; height: 44px; font-size: 18px;
display: flex; align-items: center; justify-content: center;
cursor: pointer; z-index: 10000;
}
.gallery-prev { left: 16px; }
.gallery-next { right: 16px; }
.gallery-thumbs {
position: absolute; bottom: 16px; left: 50%; transform: translateX(-50%);
display: flex; gap: 6px; max-width: 90vw; overflow-x: auto;
padding: 4px 8px;
}
.gallery-thumb {
width: 52px; height: 52px; object-fit: cover; border-radius: 8px;
opacity: 0.5; cursor: pointer; border: 2px solid transparent;
transition: opacity 0.2s, border-color 0.2s; flex-shrink: 0;
}
.gallery-thumb.active { opacity: 1; border-color: #fff; }
</style>