You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
OneAuth/ui/page/sys/org/index.html

473 lines
13 KiB
HTML

<!DOCTYPE html>
<html>
<head>
<meta name="description" content="Org Management">
<title>{{ $t('nav.org') }}</title>
<style>
body {
display: flex;
flex-direction: column;
gap: var(--spacing-lg);
padding: var(--spacing-lg);
box-sizing: border-box;
overflow: hidden;
background-color: var(--bg-color);
}
.page-header {
display: flex;
justify-content: space-between;
align-items: center;
flex-wrap: wrap;
gap: var(--spacing-md);
padding-bottom: var(--spacing-md);
border-bottom: 1px solid var(--border-color);
}
.page-title {
font-size: var(--font-size-2xl);
font-weight: var(--font-weight-bold);
color: var(--text-color);
display: flex;
align-items: center;
gap: var(--spacing-sm);
}
.search-box {
display: flex;
align-items: center;
gap: var(--spacing-sm);
background: var(--bg-color-secondary);
padding: 0 var(--spacing-md);
border-radius: var(--radius-full);
border: 1px solid var(--border-color);
min-width: 320px;
height: 40px;
transition: all var(--transition-base);
}
.search-box:focus-within {
border-color: var(--color-primary);
box-shadow: 0 0 0 3px color-mix(in srgb, var(--color-primary), transparent 85%);
}
.search-box input {
border: none;
background: transparent;
outline: none;
font-size: var(--font-size-md);
color: var(--text-color);
width: 100%;
height: 100%;
}
.search-box input::placeholder {
color: var(--text-color-tertiary);
}
.org-grid {
display: grid;
grid-template-columns: repeat(auto-fill, minmax(300px, 1fr));
gap: var(--spacing-lg);
overflow-y: auto;
padding: 4px; /* Prevent shadow clipping */
flex: 1;
}
.org-card {
background: var(--bg-color-secondary);
border-radius: var(--radius-lg);
box-shadow: var(--shadow-sm);
padding: var(--spacing-lg);
display: flex;
flex-direction: column;
gap: var(--spacing-md);
cursor: pointer;
transition: all var(--transition-base);
border: 1px solid var(--border-color);
position: relative;
overflow: hidden;
}
.org-card:hover {
transform: translateY(-4px);
box-shadow: var(--shadow-lg);
border-color: var(--color-primary);
}
.org-header {
display: flex;
align-items: flex-start;
justify-content: space-between;
gap: var(--spacing-md);
}
.org-icon {
width: 56px;
height: 56px;
border-radius: var(--radius-lg);
background: linear-gradient(135deg, var(--color-primary), var(--color-secondary));
display: flex;
align-items: center;
justify-content: center;
color: var(--color-primary-text);
font-size: var(--font-size-xl);
font-weight: bold;
flex-shrink: 0;
overflow: hidden;
}
.org-icon img {
width: 100%;
height: 100%;
object-fit: cover;
}
.org-info {
flex: 1;
min-width: 0;
}
.org-name {
font-weight: var(--font-weight-bold);
font-size: var(--font-size-lg);
color: var(--text-color);
margin-bottom: var(--spacing-xs);
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
}
.org-code {
font-size: var(--font-size-xs);
color: var(--text-color-tertiary);
background: var(--bg-color-tertiary);
padding: 2px 6px;
border-radius: var(--radius-sm);
display: inline-block;
font-family: monospace;
}
.org-status {
position: absolute;
top: var(--spacing-md);
right: var(--spacing-md);
font-size: var(--font-size-xs);
padding: 2px 8px;
border-radius: var(--radius-full);
background: var(--bg-color-tertiary);
color: var(--text-color-secondary);
}
.org-status.active {
background: color-mix(in srgb, var(--color-success), transparent 85%);
color: var(--color-success);
}
.org-desc {
font-size: var(--font-size-sm);
color: var(--text-color-secondary);
line-height: 1.6;
height: 42px; /* 2 lines */
display: -webkit-box;
-webkit-line-clamp: 2;
-webkit-box-orient: vertical;
overflow: hidden;
margin-top: var(--spacing-xs);
}
.org-meta {
display: flex;
justify-content: space-between;
align-items: center;
font-size: var(--font-size-xs);
color: var(--text-color-tertiary);
padding-top: var(--spacing-md);
border-top: 1px solid var(--border-color);
margin-top: auto;
}
.org-meta-item {
display: flex;
align-items: center;
gap: 6px;
}
.org-actions {
display: flex;
gap: var(--spacing-sm);
opacity: 0;
transition: opacity var(--transition-fast);
position: absolute;
top: var(--spacing-md);
right: var(--spacing-md);
background: var(--bg-color-secondary);
padding-left: var(--spacing-sm);
}
.org-card:hover .org-actions {
opacity: 1;
}
/* Hide status when hovering if actions overlap, or better, move status */
.org-card:hover .org-status {
opacity: 0;
}
.loading-state, .empty-state {
display: flex;
flex-direction: column;
align-items: center;
justify-content: center;
padding: var(--spacing-xl);
gap: var(--spacing-md);
color: var(--text-color-secondary);
flex: 1;
text-align: center;
}
.spinner {
width: 40px;
height: 40px;
border: 3px solid var(--border-color);
border-top-color: var(--color-primary);
border-radius: 50%;
animation: spin 1s linear infinite;
}
@keyframes spin {
to { transform: rotate(360deg); }
}
.empty-icon {
font-size: 64px;
color: var(--border-color);
margin-bottom: var(--spacing-md);
}
</style>
</head>
<body>
<div class="page-header">
<div class="page-title">
<i class="fas fa-sitemap" style="color: var(--color-primary);"></i>
{{ $t('nav.org') }}
</div>
<div style="display: flex; gap: var(--spacing-md); align-items: center;">
<div class="search-box">
<i class="fas fa-search" style="color: var(--text-color-tertiary);"></i>
<input type="text" v:value="searchQuery"
:placeholder="$t('org.search_placeholder')">
</div>
<v-btn color="primary" :click="openCreateModal">
<i class="fas fa-plus"></i>
{{ $t('org.create') }}
</v-btn>
</div>
</div>
<!-- Loading State -->
<div class="loading-state" v-if="loading">
<div class="spinner"></div>
<span>{{ $t('common.loading') }}</span>
</div>
<!-- Empty State -->
<div class="empty-state" v-if="!loading && filteredOrgs().length === 0">
<i class="fas fa-building empty-icon"></i>
<h3>{{ $t('org.no_orgs') }}</h3>
<p>{{ $t('org.no_orgs_desc') }}</p>
<v-btn color="primary" :click="openCreateModal" v-if="searchQuery === ''" style="margin-top: var(--spacing-md);">
<i class="fas fa-plus"></i>
{{ $t('org.create_first') }}
</v-btn>
</div>
<!-- Org Grid -->
<div class="org-grid" v-if="!loading && filteredOrgs().length > 0">
<div class="org-card" v-for="org in filteredOrgs()" @click="goToDetail(org.id)">
<div class="org-status" :class="org.status === 1 ? 'active' : ''">
{{ org.status === 1 ? 'Active' : 'Inactive' }}
</div>
<div class="org-header">
<div class="org-icon">
<img v-if="org.logo && org.logo.startsWith('http')" :src="org.logo" alt="logo" onerror="this.style.display='none'">
<span v-else>{{ org.name ? org.name.charAt(0).toUpperCase() : 'O' }}</span>
</div>
<div class="org-info">
<div class="org-name">{{ org.name }}</div>
<div class="org-code">{{ org.code }}</div>
</div>
</div>
<div class="org-desc">{{ org.description || $t('org.no_description') }}</div>
<div class="org-meta">
<div class="org-meta-item" title="Members Limit">
<i class="fas fa-users"></i>
<span>Max: {{ org.max_members }}</span>
</div>
<div class="org-meta-item" title="Created At">
<i class="fas fa-calendar-alt"></i>
<span>{{ formatDate(org.created_at) }}</span>
</div>
</div>
<div class="org-actions" @click.stop>
<v-btn icon size="sm" variant="outline" :click="() => openEditModal(org)" title="Edit">
<i class="fas fa-edit"></i>
</v-btn>
<v-btn icon size="sm" color="danger" variant="outline" :click="() => deleteOrg(org)" title="Delete">
<i class="fas fa-trash"></i>
</v-btn>
</div>
</div>
</div>
<!-- Create/Edit Dialog -->
<v-dialog v:visible="showModal"
:title="isEdit ? $t('org.edit') : $t('org.create')">
<form @submit.prevent="saveOrg" style="display: grid; gap: 16px;">
<v-input v-for="item in formItems" :type="item.type || 'text'" :label="$t(item.labelKey)"
:required="item.required" :placeholder="$t(item.placeholderKey)"
v:value="formData[item.name]" :disabled="item.name === 'code' && isEdit">
</v-input>
</form>
<div vslot="footer">
<v-btn variant="outline" :click="closeModal">{{ $t('common.cancel') }}</v-btn>
<v-btn color="primary" :click="saveOrg">
{{ isEdit ? $t('common.save') : $t('common.create') }}
</v-btn>
</div>
</v-dialog>
</body>
<script setup>
// State
orgs = [];
loading = false;
searchQuery = "";
showModal = false;
isEdit = false;
formData = {
id: null,
name: "",
code: "",
description: "",
logo: ""
};
formItems = [
{name: 'name', labelKey: 'org.name', label: 'Organization Name', required: true, placeholderKey: 'org.name_placeholder', placeholder: 'Enter organization name'},
{name: 'code', labelKey: 'org.code', label: 'Organization Code', required: true, placeholderKey: 'org.code_placeholder', placeholder: 'Enter organization code (unique)'},
{name: 'logo', labelKey: 'org.logo', label: 'Logo URL', required: true, placeholderKey: 'org.logo_placeholder', placeholder: 'Enter logo URL'},
{name: 'description', type: 'textarea', labelKey: 'org.description', label: 'Description', placeholderKey: 'org.desc_placeholder', placeholder: 'Enter organization description (optional)'}
];
// Helper: Format Date
formatDate = (dateStr) => {
if (!dateStr || dateStr.startsWith('0001')) return '-';
try {
const date = new Date(dateStr);
return date.toLocaleDateString();
} catch (e) {
return dateStr;
}
};
// Computed filtered orgs
filteredOrgs = () => {
if (!searchQuery) return orgs;
const query = searchQuery.toLowerCase();
return orgs.filter(org =>
org.name.toLowerCase().includes(query) ||
(org.description && org.description.toLowerCase().includes(query)) ||
(org.code && org.code.toLowerCase().includes(query))
);
};
// Load orgs
loadOrgs = async () => {
loading = true;
try {
const res = await $axios.get('/api/orgs');
orgs = res.items || [];
} catch (e) {
$message.error(e.message);
} finally {
loading = false;
}
};
// Modal operations
openCreateModal = () => {
isEdit = false;
formData = {id: null, name: "", code: "", description: "", logo: ""};
showModal = true;
};
openEditModal = (org) => {
isEdit = true;
formData = {...org};
showModal = true;
};
closeModal = () => {
showModal = false;
};
// Save (create or update)
saveOrg = async () => {
if (!formData.name || !formData.code) {
$message.error($t('org.required_fields'));
return;
}
try {
if (isEdit) {
await $axios.patch(`/api/orgs/${formData.id}`, {
name: formData.name,
description: formData.description,
logo: formData.logo
});
$message.success($t('org.updated'));
} else {
await $axios.post('/api/orgs', {
name: formData.name,
code: formData.code,
description: formData.description,
logo: formData.logo
});
$message.success($t('org.created'));
}
closeModal();
loadOrgs();
} catch (e) {
console.warn(e)
$message.error(e.message);
}
};
// Delete
deleteOrg = async (org) => {
try {
await $message.confirm($t('org.delete_confirm'));
await $axios.delete(`/api/orgs/${org.id}`);
$message.success($t('org.deleted'));
loadOrgs();
} catch (e) {
// Cancelled
}
};
// Navigate to detail
goToDetail = (id) => {
$router.push('/org/' + id);
};
</script>
<script>
$data.loadOrgs();
</script>
</html>