更新网站架构

This commit is contained in:
李志强 2026-03-09 16:35:39 +08:00
parent e291e2c48f
commit b8761ad3e4
6 changed files with 791 additions and 117 deletions

110
src/api/domain.js Normal file
View File

@ -0,0 +1,110 @@
import request from '@/utils/request'
// ==================== 主域名池管理 ====================
// 获取域名池列表
export function getDomainPoolList(params) {
return request({
url: '/admin/domain/pool/index',
method: 'get',
params
})
}
// 获取启用的主域名列表
export function getEnabledDomains() {
return request({
url: '/admin/domain/pool/getEnabledDomains',
method: 'get'
})
}
// 创建主域名
export function createDomainPool(data) {
return request({
url: '/admin/domain/pool/create',
method: 'post',
data
})
}
// 更新主域名
export function updateDomainPool(data) {
return request({
url: '/admin/domain/pool/update',
method: 'post',
data
})
}
// 删除主域名
export function deleteDomainPool(id) {
return request({
url: `/admin/domain/pool/delete/${id}`,
method: 'delete'
})
}
// 切换主域名状态
export function toggleDomainPoolStatus(id) {
return request({
url: '/admin/domain/pool/toggleStatus',
method: 'post',
data: { id }
})
}
// ==================== 租户域名管理 ====================
// 获取租户域名列表(管理员)
export function getTenantDomainList(params) {
return request({
url: '/admin/domain/tenant/index',
method: 'get',
params
})
}
// 获取当前租户的域名列表
export function getMyDomains(params) {
return request({
url: '/admin/domain/tenant/myDomains',
method: 'get',
params
})
}
// 申请二级域名
export function applyTenantDomain(data) {
return request({
url: '/admin/domain/tenant/apply',
method: 'post',
data
})
}
// 审核租户域名
export function auditTenantDomain(data) {
return request({
url: '/admin/domain/tenant/audit',
method: 'post',
data
})
}
// 禁用/启用租户域名
export function toggleTenantDomainStatus(id) {
return request({
url: '/admin/domain/tenant/toggleStatus',
method: 'post',
data: { id }
})
}
// 删除租户域名
export function deleteTenantDomain(id) {
return request({
url: `/admin/domain/tenant/delete/${id}`,
method: 'delete'
})
}

View File

@ -0,0 +1,185 @@
<template>
<div class="container-box">
<div class="header-bar">
<h2>租户域名审核</h2>
<el-button @click="fetchData" :loading="loading">
<el-icon><Refresh /></el-icon>
刷新
</el-button>
</div>
<el-divider></el-divider>
<!-- 搜索区域 -->
<div class="search-box">
<el-input
v-model="searchForm.sub_domain"
placeholder="搜索二级域名"
clearable
style="width: 200px; margin-right: 10px;"
@keyup.enter="handleSearch"
/>
<el-select v-model="searchForm.status" placeholder="状态" clearable style="width: 120px; margin-right: 10px;">
<el-option label="审核中" :value="0" />
<el-option label="已生效" :value="1" />
<el-option label="已禁用" :value="2" />
</el-select>
<el-button type="primary" @click="handleSearch">搜索</el-button>
</div>
<!-- 表格 -->
<el-table
:data="tableData"
style="width: 100%"
border
v-loading="loading"
element-loading-text="正在加载..."
>
<el-table-column type="index" label="序号" width="60" align="center" />
<el-table-column prop="tenant_name" label="租户名称" min-width="150" />
<el-table-column prop="sub_domain" label="二级域名前缀" width="150" />
<el-table-column prop="main_domain" label="主域名" min-width="150" />
<el-table-column prop="full_domain" label="完整域名" min-width="200" />
<el-table-column prop="status" label="状态" width="100" align="center">
<template #default="scope">
<el-tag v-if="scope.row.status === 0" type="warning">审核中</el-tag>
<el-tag v-else-if="scope.row.status === 1" type="success">已生效</el-tag>
<el-tag v-else type="danger">已禁用</el-tag>
</template>
</el-table-column>
<el-table-column prop="create_time" label="申请时间" width="180" />
<el-table-column label="操作" width="200" fixed="right" align="center">
<template #default="scope">
<template v-if="scope.row.status === 0">
<el-button size="small" type="success" @click="handleAudit(scope.row, 'approve')">
通过
</el-button>
<el-button size="small" type="danger" @click="handleAudit(scope.row, 'reject')">
拒绝
</el-button>
</template>
<template v-else-if="scope.row.status === 1">
<el-button size="small" text type="danger" @click="handleDisable(scope.row)">
禁用
</el-button>
</template>
<span v-else style="color: #999;">-</span>
</template>
</el-table-column>
</el-table>
<!-- 分页 -->
<div class="pagination-wrap">
<el-pagination
v-model:current-page="pagination.page"
v-model:page-size="pagination.pageSize"
:total="pagination.total"
:page-sizes="[10, 20, 50]"
layout="total, sizes, prev, pager, next, jumper"
@size-change="fetchData"
@current-change="fetchData"
/>
</div>
</div>
</template>
<script lang="ts" setup>
import { ref, reactive, onMounted } from 'vue'
import { ElMessage, ElMessageBox } from 'element-plus'
import { Refresh } from '@element-plus/icons-vue'
import { getTenantDomainList, auditTenantDomain, toggleTenantDomainStatus } from '@/api/domain'
const loading = ref(false)
const searchForm = reactive({
sub_domain: '',
status: ''
})
const pagination = reactive({
page: 1,
pageSize: 10,
total: 0
})
const tableData = ref<any[]>([])
const fetchData = async () => {
loading.value = true
try {
const res = await getTenantDomainList({
page: pagination.page,
pageSize: pagination.pageSize,
...searchForm
})
if (res.code === 200) {
tableData.value = res.data.list || []
pagination.total = res.data.total || 0
}
} finally {
loading.value = false
}
}
const handleSearch = () => {
pagination.page = 1
fetchData()
}
const handleAudit = async (row: any, action: string) => {
const msg = action === 'approve' ? '确定要通过该域名申请吗?' : '确定要拒绝该域名申请吗?'
await ElMessageBox.confirm(msg, '提示', { type: 'warning' })
const res = await auditTenantDomain({ id: row.id, action })
if (res.code === 200) {
ElMessage.success(res.msg)
fetchData()
} else {
ElMessage.error(res.msg || '操作失败')
}
}
const handleDisable = async (row: any) => {
await ElMessageBox.confirm('确定要禁用该域名吗?', '提示', { type: 'warning' })
const res = await toggleTenantDomainStatus(row.id)
if (res.code === 200) {
ElMessage.success(res.msg)
fetchData()
} else {
ElMessage.error(res.msg || '操作失败')
}
}
onMounted(() => {
fetchData()
})
</script>
<style lang="less" scoped>
.container-box {
padding: 20px;
}
.header-bar {
display: flex;
justify-content: space-between;
align-items: center;
h2 {
margin: 0;
font-size: 18px;
font-weight: 600;
}
}
.search-box {
margin-bottom: 20px;
}
.pagination-wrap {
margin-top: 20px;
display: flex;
justify-content: flex-end;
}
</style>

View File

@ -0,0 +1,5 @@
<template></template>
<script lang="ts" setup></script>
<style lang="less" scoped></style>

View File

@ -0,0 +1,287 @@
<template>
<div class="container-box">
<div class="header-bar">
<h2>主域名池管理</h2>
<div class="header-actions">
<el-button type="primary" @click="handleAdd">
<el-icon><Plus /></el-icon>
添加主域名
</el-button>
<el-button @click="fetchData" :loading="loading">
<el-icon><Refresh /></el-icon>
刷新
</el-button>
</div>
</div>
<el-divider></el-divider>
<!-- 搜索区域 -->
<div class="search-box">
<el-input
v-model="searchForm.main_domain"
placeholder="搜索主域名"
clearable
style="width: 200px; margin-right: 10px;"
@keyup.enter="handleSearch"
/>
<el-select v-model="searchForm.status" placeholder="状态" clearable style="width: 120px; margin-right: 10px;">
<el-option label="禁用" :value="0" />
<el-option label="启用" :value="1" />
</el-select>
<el-button type="primary" @click="handleSearch">搜索</el-button>
</div>
<!-- 表格 -->
<el-table
:data="tableData"
style="width: 100%"
border
v-loading="loading"
element-loading-text="正在加载..."
>
<el-table-column type="index" label="序号" width="60" align="center" />
<el-table-column prop="main_domain" label="主域名" min-width="200" />
<el-table-column prop="status" label="状态" width="100" align="center">
<template #default="scope">
<el-tag :type="scope.row.status === 1 ? 'success' : 'danger'">
{{ scope.row.status === 1 ? '启用' : '禁用' }}
</el-tag>
</template>
</el-table-column>
<el-table-column prop="create_time" label="创建时间" width="180" />
<el-table-column label="操作" width="240" fixed="right" align="center">
<template #default="scope">
<el-button size="small" text @click="handleEdit(scope.row)">
<el-icon><Edit /></el-icon>
<span>编辑</span>
</el-button>
<el-button size="small" text @click="handleToggleStatus(scope.row)">
<el-icon><Switch /></el-icon>
<span>{{ scope.row.status === 1 ? '禁用' : '启用' }}</span>
</el-button>
<el-button size="small" text type="danger" @click="handleDelete(scope.row)">
<el-icon><Delete /></el-icon>
<span>删除</span>
</el-button>
</template>
</el-table-column>
</el-table>
<!-- 分页 -->
<div class="pagination-wrap">
<el-pagination
v-model:current-page="pagination.page"
v-model:page-size="pagination.pageSize"
:total="pagination.total"
:page-sizes="[10, 20, 50]"
layout="total, sizes, prev, pager, next, jumper"
@size-change="fetchData"
@current-change="fetchData"
/>
</div>
<!-- 添加/编辑弹窗 -->
<el-dialog
v-model="dialogVisible"
:title="isEdit ? '编辑主域名' : '添加主域名'"
width="500px"
>
<el-form :model="form" :rules="rules" ref="formRef" label-width="100px">
<el-form-item label="主域名" prop="main_domain">
<el-input v-model="form.main_domain" placeholder="请输入主域名,如: example.com" />
</el-form-item>
<el-form-item label="状态" prop="status">
<el-switch v-model="form.status" :active-value="1" :inactive-value="0" />
</el-form-item>
</el-form>
<template #footer>
<el-button @click="dialogVisible = false">取消</el-button>
<el-button type="primary" @click="handleSubmit" :loading="submitLoading">确定</el-button>
</template>
</el-dialog>
</div>
</template>
<script lang="ts" setup>
import { ref, reactive, onMounted } from 'vue'
import { ElMessage, ElMessageBox } from 'element-plus'
import { Plus, Edit, Delete, Refresh, Switch } from '@element-plus/icons-vue'
import {
getDomainPoolList,
createDomainPool,
updateDomainPool,
deleteDomainPool,
toggleDomainPoolStatus
} from '@/api/domain'
const loading = ref(false)
const submitLoading = ref(false)
const dialogVisible = ref(false)
const isEdit = ref(false)
const formRef = ref()
//
const searchForm = reactive({
main_domain: '',
status: ''
})
//
const pagination = reactive({
page: 1,
pageSize: 10,
total: 0
})
//
const tableData = ref<any[]>([])
//
const form = reactive({
id: 0,
main_domain: '',
status: 1
})
//
const rules = {
main_domain: [
{ required: true, message: '请输入主域名', trigger: 'blur' }
]
}
//
const fetchData = async () => {
loading.value = true
try {
const res = await getDomainPoolList({
page: pagination.page,
pageSize: pagination.pageSize,
...searchForm
})
if (res.code === 200) {
tableData.value = res.data.list || []
pagination.total = res.data.total || 0
}
} finally {
loading.value = false
}
}
//
const handleSearch = () => {
pagination.page = 1
fetchData()
}
//
const handleAdd = () => {
isEdit.value = false
form.id = 0
form.main_domain = ''
form.status = 1
dialogVisible.value = true
}
//
const handleEdit = (row: any) => {
isEdit.value = true
form.id = row.id
form.main_domain = row.main_domain
form.status = row.status
dialogVisible.value = true
}
//
const handleSubmit = async () => {
await formRef.value.validate()
submitLoading.value = true
try {
if (isEdit.value) {
const res = await updateDomainPool(form)
if (res.code === 200) {
ElMessage.success('更新成功')
dialogVisible.value = false
fetchData()
} else {
ElMessage.error(res.msg || '更新失败')
}
} else {
const res = await createDomainPool(form)
if (res.code === 200) {
ElMessage.success('创建成功')
dialogVisible.value = false
fetchData()
} else {
ElMessage.error(res.msg || '创建失败')
}
}
} finally {
submitLoading.value = false
}
}
//
const handleDelete = (row: any) => {
ElMessageBox.confirm('确定要删除该主域名吗?', '提示', {
type: 'warning'
}).then(async () => {
const res = await deleteDomainPool(row.id)
if (res.code === 200) {
ElMessage.success('删除成功')
fetchData()
} else {
ElMessage.error(res.msg || '删除失败')
}
})
}
//
const handleToggleStatus = async (row: any) => {
const res = await toggleDomainPoolStatus(row.id)
if (res.code === 200) {
ElMessage.success('状态更新成功')
fetchData()
} else {
ElMessage.error(res.msg || '操作失败')
}
}
onMounted(() => {
fetchData()
})
</script>
<style lang="less" scoped>
.container-box {
padding: 20px;
}
.header-bar {
display: flex;
justify-content: space-between;
align-items: center;
h2 {
margin: 0;
font-size: 18px;
font-weight: 600;
}
.header-actions {
display: flex;
gap: 10px;
}
}
.search-box {
margin-bottom: 20px;
}
.pagination-wrap {
margin-top: 20px;
display: flex;
justify-content: flex-end;
}
</style>

View File

@ -35,7 +35,7 @@
<div v-else class="template-grid"> <div v-else class="template-grid">
<div v-for="item in templateList" :key="item.key" class="template-card" :class="{ active: item.key === currentTheme }"> <div v-for="item in templateList" :key="item.key" class="template-card" :class="{ active: item.key === currentTheme }">
<div class="card-preview"> <div class="card-preview">
<img :src="item.preview" alt="preview" @error="handleImageError($event)" /> <img :src="getPreviewUrl(item.preview)" alt="preview" @error="handleImageError($event)" />
<div v-if="item.key === currentTheme" class="current-tag"> <div v-if="item.key === currentTheme" class="current-tag">
<el-tag type="success" size="small">使用中</el-tag> <el-tag type="success" size="small">使用中</el-tag>
</div> </div>
@ -61,44 +61,11 @@
<el-button v-else type="info" size="small" disabled> <el-button v-else type="info" size="small" disabled>
当前使用 当前使用
</el-button> </el-button>
<el-button size="small" @click="handlePreview(item)">预览</el-button>
<el-button size="small" @click="handleEdit(item)">编辑数据</el-button>
</div> </div>
</div> </div>
</div> </div>
</div> </div>
</main> </main>
<!-- 预览弹窗 -->
<el-dialog v-model="previewVisible" title="模板预览" width="90%" top="5vh">
<iframe :src="previewUrl" class="preview-iframe" frameborder="0"></iframe>
</el-dialog>
<!-- 编辑数据弹窗 -->
<el-dialog v-model="editVisible" title="编辑模板数据" width="800px">
<el-form :model="editForm" label-width="100px">
<el-form-item label="模板">
<el-input v-model="editForm.theme_key" disabled />
</el-form-item>
<el-form-item label="选择字段">
<el-select v-model="selectedField" placeholder="选择要编辑的字段" @change="handleFieldChange">
<el-option v-for="(label, key) in editForm.fields" :key="key" :label="label" :value="key" />
</el-select>
</el-form-item>
<el-form-item v-if="selectedField" label="字段值">
<el-input
v-model="fieldValue"
type="textarea"
:rows="10"
placeholder="输入 JSON 格式或普通文本"
/>
</el-form-item>
</el-form>
<template #footer>
<el-button @click="editVisible = false">取消</el-button>
<el-button type="primary" @click="handleSaveData" :loading="saving">保存</el-button>
</template>
</el-dialog>
</div> </div>
</template> </template>
@ -106,20 +73,22 @@
import { ref, onMounted } from 'vue' import { ref, onMounted } from 'vue'
import { ElMessage } from 'element-plus' import { ElMessage } from 'element-plus'
import { Refresh, Loading } from '@element-plus/icons-vue' import { Refresh, Loading } from '@element-plus/icons-vue'
import { getThemeList, switchTheme, getThemeData, saveThemeData } from '@/api/theme' import { getThemeList, switchTheme } from '@/api/theme'
const API_BASE_URL = import.meta.env.VITE_API_BASE_URL
// //
const loading = ref(false) const loading = ref(false)
const switching = ref('') const switching = ref('')
const saving = ref(false)
const templateList = ref<any[]>([]) const templateList = ref<any[]>([])
const currentTheme = ref('default') const currentTheme = ref('default')
const previewVisible = ref(false)
const previewUrl = ref('') // URL
const editVisible = ref(false) const getPreviewUrl = (path: string) => {
const editForm = ref<any>({}) if (!path) return ''
const selectedField = ref('') if (path.startsWith('http')) return path
const fieldValue = ref('') return API_BASE_URL + path
}
// //
const fetchTemplates = async () => { const fetchTemplates = async () => {
@ -157,75 +126,6 @@ const handleUse = async (item: any) => {
} }
} }
//
const handlePreview = (item: any) => {
previewUrl.value = item.path
previewVisible.value = true
}
//
const handleEdit = async (item: any) => {
editForm.value = { ...item, fields: item.fields || {} }
selectedField.value = ''
fieldValue.value = ''
//
try {
const res = await getThemeData({ theme_key: item.key })
if (res.code === 200 && res.data.data) {
//
const savedData = res.data.data
//
for (const key in savedData) {
if (savedData[key]) {
selectedField.value = key
fieldValue.value = typeof savedData[key] === 'object'
? JSON.stringify(savedData[key], null, 2)
: savedData[key]
break
}
}
}
} catch (error) {
console.error('获取模板数据失败', error)
}
editVisible.value = true
}
//
const handleFieldChange = (field: string) => {
fieldValue.value = ''
}
//
const handleSaveData = async () => {
if (!selectedField.value) {
ElMessage.warning('请选择要编辑的字段')
return
}
saving.value = true
try {
const res = await saveThemeData({
theme_key: editForm.value.key,
field_key: selectedField.value,
field_value: fieldValue.value
})
if (res.code === 200) {
ElMessage.success('保存成功')
editVisible.value = false
} else {
ElMessage.error(res.msg || '保存失败')
}
} catch (error) {
ElMessage.error('保存失败')
} finally {
saving.value = false
}
}
// //
const handleImageError = (event: Event) => { const handleImageError = (event: Event) => {
const img = event.target as HTMLImageElement const img = event.target as HTMLImageElement
@ -370,10 +270,4 @@ onMounted(() => {
} }
} }
} }
.preview-iframe {
width: 100%;
height: 70vh;
border: none;
}
</style> </style>

View File

@ -0,0 +1,193 @@
<template>
<div class="container-box">
<div class="header-bar">
<h2>我的域名</h2>
<el-button type="primary" @click="dialogVisible = true">
<el-icon><Plus /></el-icon>
申请二级域名
</el-button>
</div>
<el-divider></el-divider>
<!-- 我的域名列表 -->
<el-table
:data="tableData"
style="width: 100%"
border
v-loading="loading"
element-loading-text="正在加载..."
>
<el-table-column type="index" label="序号" width="60" align="center" />
<el-table-column prop="sub_domain" label="二级域名前缀" width="150" />
<el-table-column prop="main_domain" label="主域名" min-width="150" />
<el-table-column prop="full_domain" label="完整域名" min-width="200">
<template #default="scope">
<el-link type="primary" :href="'http://' + scope.row.full_domain" target="_blank">
{{ scope.row.full_domain }}
</el-link>
</template>
</el-table-column>
<el-table-column prop="status" label="状态" width="100" align="center">
<template #default="scope">
<el-tag v-if="scope.row.status === 0" type="warning">审核中</el-tag>
<el-tag v-else-if="scope.row.status === 1" type="success">已生效</el-tag>
<el-tag v-else type="danger">已禁用</el-tag>
</template>
</el-table-column>
<el-table-column prop="create_time" label="申请时间" width="180" />
<el-table-column label="操作" width="100" fixed="right" align="center">
<template #default="scope">
<el-button size="small" text type="primary" @click="handleCopy(scope.row.full_domain)">
复制
</el-button>
</template>
</el-table-column>
</el-table>
<!-- 申请域名弹窗 -->
<el-dialog v-model="dialogVisible" title="申请二级域名" width="500px">
<el-form :model="form" :rules="rules" ref="formRef" label-width="100px">
<el-form-item label="选择主域名" prop="main_domain">
<el-select v-model="form.main_domain" placeholder="请选择主域名" style="width: 100%;">
<el-option
v-for="item in domainList"
:key="item.main_domain"
:label="item.main_domain"
:value="item.main_domain"
/>
</el-select>
</el-form-item>
<el-form-item label="二级前缀" prop="sub_domain">
<el-input v-model="form.sub_domain" placeholder="请输入二级域名前缀">
<template #append>{{ form.main_domain ? '.' + form.main_domain : '' }}</template>
</el-input>
<div class="form-tip">只能包含字母数字和连字符不能以连字符开头或结尾</div>
</el-form-item>
<el-form-item label="预览">
<div class="domain-preview">
{{ form.sub_domain ? form.sub_domain + '.' + (form.main_domain || 'example.com') : '请填写上方信息' }}
</div>
</el-form-item>
</el-form>
<template #footer>
<el-button @click="dialogVisible = false">取消</el-button>
<el-button type="primary" @click="handleSubmit" :loading="submitLoading">提交申请</el-button>
</template>
</el-dialog>
</div>
</template>
<script lang="ts" setup>
import { ref, reactive, onMounted } from 'vue'
import { ElMessage } from 'element-plus'
import { Plus } from '@element-plus/icons-vue'
import { getMyDomains, applyTenantDomain, getEnabledDomains } from '@/api/domain'
import { useAuthStore } from '@/stores/auth'
const authStore = useAuthStore()
const loading = ref(false)
const submitLoading = ref(false)
const dialogVisible = ref(false)
const formRef = ref()
const tableData = ref<any[]>([])
const domainList = ref<any[]>([])
const form = reactive({
main_domain: '',
sub_domain: ''
})
const rules = {
main_domain: [
{ required: true, message: '请选择主域名', trigger: 'change' }
],
sub_domain: [
{ required: true, message: '请输入二级域名前缀', trigger: 'blur' },
{ pattern: /^[a-zA-Z0-9][a-zA-Z0-9-]{0,61}[a-zA-Z0-9]$/, message: '格式不正确', trigger: 'blur' }
]
}
const fetchDomains = async () => {
loading.value = true
try {
//
const res = await getMyDomains({ tid: authStore.user.tenant_id })
if (res.code === 200) {
tableData.value = res.data || []
}
//
const domainRes = await getEnabledDomains()
if (domainRes.code === 200) {
domainList.value = domainRes.data || []
}
} finally {
loading.value = false
}
}
const handleSubmit = async () => {
await formRef.value.validate()
submitLoading.value = true
try {
const res = await applyTenantDomain({
tenant_id: authStore.user.tenant_id,
main_domain: form.main_domain,
sub_domain: form.sub_domain
})
if (res.code === 200) {
ElMessage.success(res.msg)
dialogVisible.value = false
form.sub_domain = ''
fetchDomains()
} else {
ElMessage.error(res.msg || '申请失败')
}
} finally {
submitLoading.value = false
}
}
const handleCopy = (domain: string) => {
navigator.clipboard.writeText('http://' + domain)
ElMessage.success('链接已复制到剪贴板')
}
onMounted(() => {
fetchDomains()
})
</script>
<style lang="less" scoped>
.container-box {
padding: 20px;
}
.header-bar {
display: flex;
justify-content: space-between;
align-items: center;
h2 {
margin: 0;
font-size: 18px;
font-weight: 600;
}
}
.form-tip {
font-size: 12px;
color: #999;
margin-top: 5px;
}
.domain-preview {
padding: 10px;
background: #f5f7fa;
border-radius: 4px;
font-size: 14px;
color: #409eff;
}
</style>