综述: 优化平台端web界面样式和功能模块

- 在摊位列表和摊位创建页面中增加绑定已存在供货商账号的功能选项
- 优化平台端web整体界面样式,统一样式规范,改善用户界面体验
- 在菜市场列表页面中优化详情查看和创建表单,增加地图定位功能和营业时间配置交互
- 重构权限管理模块的界面样式,提升视觉效果和操作体验
This commit is contained in:
linbin 2025-10-21 01:40:00 +08:00
parent f1df5b450b
commit e2f5333831
6 changed files with 657 additions and 692 deletions

View File

@ -485,6 +485,7 @@
<option value="">请选择操作类型</option> <option value="">请选择操作类型</option>
<option value="bind_market_manager">绑定收益人是市场经营者</option> <option value="bind_market_manager">绑定收益人是市场经营者</option>
<option value="create_supplier">创建新收益人(供货商)</option> <option value="create_supplier">创建新收益人(供货商)</option>
<option value="bind_existing_supplier">绑定已存在的供货商账号</option>
</select> </select>
</div> </div>
@ -650,6 +651,10 @@
// 创建新收益人(供货商) - 显示账号和密码 // 创建新收益人(供货商) - 显示账号和密码
supplierAccountGroup.style.display = 'block'; supplierAccountGroup.style.display = 'block';
supplierPasswordGroup.style.display = 'block'; supplierPasswordGroup.style.display = 'block';
} else if (operationType === 'bind_existing_supplier') {
// 绑定已存在的供货商账号 - 只显示账号,不显示密码
supplierAccountGroup.style.display = 'block';
supplierPasswordGroup.style.display = 'none';
} else { } else {
// 绑定收益人是市场经营者 - 隐藏账号和密码 // 绑定收益人是市场经营者 - 隐藏账号和密码
supplierAccountGroup.style.display = 'none'; supplierAccountGroup.style.display = 'none';
@ -689,6 +694,17 @@
beneficiaryData.supplierAccount = supplierAccount; beneficiaryData.supplierAccount = supplierAccount;
beneficiaryData.password = password; beneficiaryData.password = password;
} }
// 如果选择了"绑定已存在的供货商账号",只需要验证供货商账号
else if (operationType === 'bind_existing_supplier') {
const supplierAccount = document.getElementById('modalSupplierAccount').value;
if (!supplierAccount) {
alert('请输入供货商账号');
return;
}
beneficiaryData.supplierAccount = supplierAccount;
}
console.log('保存收益人配置:', beneficiaryData); console.log('保存收益人配置:', beneficiaryData);
@ -696,7 +712,7 @@
alert('收益人配置已保存!'); alert('收益人配置已保存!');
// 可选:更新表格中的收益人信息 // 可选:更新表格中的收益人信息
if (currentBoothRow && operationType === 'create_supplier') { if (currentBoothRow && (operationType === 'create_supplier' || operationType === 'bind_existing_supplier')) {
const cells = currentBoothRow.querySelectorAll('td'); const cells = currentBoothRow.querySelectorAll('td');
// cells[2].textContent = '新收益人'; // 可以根据实际需求更新 // cells[2].textContent = '新收益人'; // 可以根据实际需求更新
// cells[3].textContent = beneficiaryData.supplierAccount; // cells[3].textContent = beneficiaryData.supplierAccount;

View File

@ -405,6 +405,7 @@
<option value="">请选择操作类型</option> <option value="">请选择操作类型</option>
<option value="bind_market_manager">绑定收益人是市场经营者</option> <option value="bind_market_manager">绑定收益人是市场经营者</option>
<option value="create_supplier">创建新收益人(供货商)</option> <option value="create_supplier">创建新收益人(供货商)</option>
<option value="bind_existing_supplier">绑定已存在的供货商账号</option>
</select> </select>
</div> </div>
</div> </div>
@ -732,6 +733,10 @@
// 创建新收益人(供货商)- 显示账号和密码 // 创建新收益人(供货商)- 显示账号和密码
supplierAccountGroup.style.display = 'flex'; supplierAccountGroup.style.display = 'flex';
supplierPasswordGroup.style.display = 'flex'; supplierPasswordGroup.style.display = 'flex';
} else if (operationType === 'bind_existing_supplier') {
// 绑定已存在的供货商账号 - 只显示账号,不显示密码
supplierAccountGroup.style.display = 'flex';
supplierPasswordGroup.style.display = 'none';
} else { } else {
// 绑定收益人是市场经营者 - 隐藏账号和密码 // 绑定收益人是市场经营者 - 隐藏账号和密码
supplierAccountGroup.style.display = 'none'; supplierAccountGroup.style.display = 'none';
@ -810,6 +815,17 @@
formData.supplierAccount = supplierAccount; formData.supplierAccount = supplierAccount;
formData.password = password; formData.password = password;
} }
// 如果选择了"绑定已存在的供货商账号",只需要验证供货商账号
else if (operationType === 'bind_existing_supplier') {
const supplierAccount = document.getElementById('supplierAccount').value;
if (!supplierAccount) {
alert('请输入供货商账号');
return;
}
formData.supplierAccount = supplierAccount;
}
} }
if (!formData.boothName) { if (!formData.boothName) {

View File

@ -17,6 +17,7 @@
color: #333; color: #333;
height: 100vh; height: 100vh;
overflow: hidden; overflow: hidden;
background: #f5f5f5;
} }
.admin-layout { .admin-layout {
@ -269,6 +270,7 @@
border-bottom-color: #fff; border-bottom-color: #fff;
z-index: 1; z-index: 1;
margin-bottom: -1px; margin-bottom: -1px;
color: #1890ff;
} }
.tab-item:hover { .tab-item:hover {
@ -384,32 +386,30 @@
<a href="./权限管理/市场经营者创建.html" class="submenu-item" onclick="openTabWithIframe(this, '市场经营者创建')">市场经营者创建</a> <a href="./权限管理/市场经营者创建.html" class="submenu-item" onclick="openTabWithIframe(this, '市场经营者创建')">市场经营者创建</a>
</div> </div>
</li> </li>
<li class="menu-item">
<button class="menu-item-link" onclick="toggleSubmenu(this)">
<span class="menu-icon">
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 1024 1024">
<path fill="currentColor" d="M160 448a32 32 0 0 1-32-32V160.064a32 32 0 0 1 32-32h256a32 32 0 0 1 32 32V416a32 32 0 0 1-32 32zm448 0a32 32 0 0 1-32-32V160.064a32 32 0 0 1 32-32h255.936a32 32 0 0 1 32 32V416a32 32 0 0 1-32 32zM160 896a32 32 0 0 1-32-32V608a32 32 0 0 1 32-32h256a32 32 0 0 1 32 32v256a32 32 0 0 1-32 32zm448 0a32 32 0 0 1-32-32V608a32 32 0 0 1 32-32h255.936a32 32 0 0 1 32 32v256a32 32 0 0 1-32 32z"></path>
</svg>
</span>
<span class="menu-text">菜市场管理</span>
<span class="menu-arrow">
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 1024 1024">
<path fill="currentColor" d="M831.872 340.864 512 652.672 192.128 340.864a30.592 30.592 0 0 0-42.752 0 29.12 29.12 0 0 0 0 41.6L489.664 714.24a32 32 0 0 0 44.672 0l340.288-331.712a29.12 29.12 0 0 0 0-41.728 30.592 30.592 0 0 0-42.752 0z"></path>
</svg>
</span>
</button>
<div class="submenu">
<a href="./菜市场管理/菜市场列表.html" class="submenu-item" onclick="openTabWithIframe(this, '菜市场列表')">菜市场列表</a>
</div>
</li>
</ul> </ul>
</nav> </nav>
</aside> </aside>
<!-- 右侧主内容区 --> <!-- 右侧主内容区 -->
<main class="main-content"> <main class="main-content">
<!-- 顶部导航栏 -->
<header class="header">
<div class="header-left">
<div class="menu-trigger" onclick="toggleSidebar()"></div>
<nav class="breadcrumb">
<a href="#/dashboard" class="breadcrumb-item">首页</a>
<span class="breadcrumb-separator">/</span>
<a href="#" class="breadcrumb-item">权限管理</a>
<span class="breadcrumb-separator">/</span>
<span class="breadcrumb-item">权限编辑</span>
</nav>
</div>
<div class="header-right">
<div class="user-info">
<div class="user-avatar">A</div>
<span>admin</span>
</div>
</div>
</header>
<!-- 标签页区域 --> <!-- 标签页区域 -->
<div class="tabs-container"> <div class="tabs-container">
<div class="tab-item active" data-tab="home"> <div class="tab-item active" data-tab="home">

View File

@ -12,178 +12,178 @@
} }
body { body {
font-family: 'Segoe UI', Tahoma, Geneva, Verdana, sans-serif; font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', 'PingFang SC', 'Hiragino Sans GB', 'Microsoft YaHei', 'Helvetica Neue', Helvetica, Arial, sans-serif;
background-color: #f5f7fa; background-color: #f5f5f5;
color: #333; color: #333;
height: 100vh; min-height: 100vh;
margin: 0; padding: 20px;
padding: 0;
} }
.container { .container {
width: 100%; max-width: 1600px;
height: 100vh; margin: 0 auto;
margin: 0;
background-color: #fff;
border-radius: 0;
box-shadow: none;
overflow: hidden;
} }
.header { /* 页面头部 */
background: linear-gradient(135deg, #2c3e50, #3498db); .page-header {
color: #fff; background: white;
padding: 20px 30px; padding: 16px 24px;
margin-bottom: 16px;
border-radius: 8px;
box-shadow: 0 2px 8px rgba(0, 0, 0, 0.1);
display: flex; display: flex;
justify-content: space-between; justify-content: space-between;
align-items: center; align-items: center;
} }
.header h1 { .page-title {
font-size: 1.8rem; font-size: 24px;
font-weight: 600; font-weight: 600;
color: #262626;
} }
.header-actions { /* 按钮样式 */
display: flex;
gap: 10px;
}
.btn { .btn {
padding: 10px 20px; height: 40px;
padding: 0 16px;
border: none; border: none;
border-radius: 5px; border-radius: 6px;
cursor: pointer;
font-size: 14px; font-size: 14px;
font-weight: 500; cursor: pointer;
transition: all 0.3s ease;
display: inline-flex; display: inline-flex;
align-items: center; align-items: center;
justify-content: center;
transition: all 0.3s;
gap: 5px; gap: 5px;
} }
.btn-primary { .btn-primary {
background-color: #3498db; background: #1890ff;
color: #fff; color: white;
} }
.btn-primary:hover { .btn-primary:hover {
background-color: #2980b9; background: #40a9ff;
transform: translateY(-2px);
} }
.btn-success { .btn-success {
background-color: #27ae60; background: #52c41a;
color: #fff; color: white;
} }
.btn-success:hover { .btn-success:hover {
background-color: #229954; background: #73d13d;
} }
.btn-warning { .btn-default {
background-color: #f39c12; background: white;
color: #fff; color: #666;
border: 1px solid #ddd;
} }
.btn-warning:hover { .btn-default:hover {
background-color: #e67e22; border-color: #1890ff;
color: #1890ff;
} }
/* 内容区域 */ .button-group {
.content { display: flex;
padding: 30px; gap: 12px;
} }
.section { /* 表单区域 */
margin-bottom: 40px; .form-section {
background: white;
padding: 24px;
margin-bottom: 16px;
border-radius: 8px;
box-shadow: 0 2px 8px rgba(0, 0, 0, 0.1);
} }
.section-title { .section-title {
font-size: 1.4rem; font-size: 18px;
font-weight: 600; font-weight: 600;
color: #2c3e50; color: #262626;
margin-bottom: 20px; margin-bottom: 20px;
padding-bottom: 10px; padding-bottom: 12px;
border-bottom: 2px solid #ecf0f1; border-bottom: 1px solid #f0f0f0;
}
/* 账号创建区域 */
.account-form {
background-color: #f8f9fa;
padding: 25px;
border-radius: 8px;
border: 1px solid #dee2e6;
} }
.form-row { .form-row {
display: grid;
grid-template-columns: repeat(auto-fit, minmax(250px, 1fr));
gap: 16px;
margin-bottom: 16px;
}
.form-item {
display: flex; display: flex;
gap: 20px; flex-direction: column;
align-items: end; gap: 8px;
} }
.form-group { .form-label {
flex: 1;
margin-bottom: 20px;
}
.form-group label {
display: block;
margin-bottom: 8px;
font-weight: 500;
color: #2c3e50;
}
.form-group input,
.form-group select {
width: 100%;
padding: 12px 15px;
border: 1px solid #ddd;
border-radius: 5px;
font-size: 14px; font-size: 14px;
transition: border-color 0.3s ease; color: #666;
font-weight: 500;
} }
.form-group input:focus, .form-input,
.form-group select:focus { .form-select {
height: 40px;
padding: 0 12px;
border: 1px solid #ddd;
border-radius: 6px;
font-size: 14px;
transition: all 0.3s;
}
.form-input:focus,
.form-select:focus {
outline: none; outline: none;
border-color: #3498db; border-color: #1890ff;
box-shadow: 0 0 5px rgba(52, 152, 219, 0.3); box-shadow: 0 0 0 2px rgba(24, 144, 255, 0.2);
} }
/* 菜市场选择框样式 */ /* 菜市场选择下拉框 */
.market-select-container { .market-select-container {
position: relative; position: relative;
} }
.market-select {
cursor: pointer;
}
.market-dropdown { .market-dropdown {
position: absolute; position: absolute;
top: 100%; top: 100%;
left: 0; left: 0;
right: 0; right: 0;
background: #fff; background: white;
border: 1px solid #ddd; border: 1px solid #f0f0f0;
border-top: none; border-radius: 6px;
border-radius: 0 0 5px 5px; max-height: 300px;
max-height: 200px;
overflow-y: auto; overflow-y: auto;
z-index: 1000; z-index: 1000;
display: none; display: none;
box-shadow: 0 4px 12px rgba(0, 0, 0, 0.1); box-shadow: 0 4px 12px rgba(0, 0, 0, 0.1);
margin-top: 4px;
} }
.market-dropdown.show { .market-dropdown.show {
display: block; display: block;
} }
.market-filter {
padding: 12px;
border: none;
border-bottom: 1px solid #f0f0f0;
width: 100%;
outline: none;
font-size: 14px;
}
.market-option { .market-option {
padding: 10px 15px; padding: 10px 12px;
cursor: pointer; cursor: pointer;
transition: background-color 0.2s ease; transition: background-color 0.2s;
font-size: 14px;
} }
.market-option:hover { .market-option:hover {
@ -191,24 +191,12 @@
} }
.market-option.selected { .market-option.selected {
background-color: #3498db; background-color: #e6f7ff;
color: #fff; color: #1890ff;
}
.market-filter {
padding: 10px 15px;
border: none;
border-bottom: 1px solid #eee;
width: 100%;
outline: none;
} }
/* 权限分配区域 */ /* 权限分配区域 */
.permissions-container { .permissions-container {
background-color: #f8f9fa;
padding: 25px;
border-radius: 8px;
border: 1px solid #dee2e6;
max-height: 500px; max-height: 500px;
overflow-y: auto; overflow-y: auto;
} }
@ -224,24 +212,22 @@
.permission-node { .permission-node {
display: flex; display: flex;
align-items: center; align-items: center;
padding: 12px 15px; padding: 12px 16px;
border-radius: 6px; border-radius: 6px;
transition: all 0.3s ease; transition: all 0.3s;
cursor: pointer; cursor: pointer;
background-color: #fff; background-color: white;
border: 1px solid #e9ecef; border: 1px solid #f0f0f0;
margin-bottom: 5px;
} }
.permission-node:hover { .permission-node:hover {
background-color: #f8f9fa; background-color: #f8f9fa;
border-color: #3498db; border-color: #1890ff;
box-shadow: 0 2px 8px rgba(52, 152, 219, 0.15);
} }
.permission-node.selected { .permission-node.selected {
background-color: #e3f2fd; background-color: #e6f7ff;
border-color: #3498db; border-color: #1890ff;
} }
.permission-checkbox { .permission-checkbox {
@ -249,23 +235,24 @@
width: 18px; width: 18px;
height: 18px; height: 18px;
cursor: pointer; cursor: pointer;
accent-color: #3498db; accent-color: #1890ff;
} }
.expand-btn { .expand-btn {
background: #3498db; background: none;
border: none; border: none;
cursor: pointer; cursor: pointer;
padding: 2px 8px; padding: 4px;
margin-right: 12px; margin-right: 12px;
color: #fff; color: #1890ff;
font-size: 14px; font-size: 14px;
transition: all 0.3s ease; transition: all 0.3s;
min-width: 28px; min-width: 24px;
border-radius: 4px; border-radius: 4px;
display: flex; }
align-items: center;
justify-content: center; .expand-btn:hover {
background: #e6f7ff;
} }
.expand-btn.collapsed { .expand-btn.collapsed {
@ -274,140 +261,132 @@
.permission-label { .permission-label {
font-size: 14px; font-size: 14px;
color: #2c3e50; color: #333;
user-select: none; user-select: none;
flex: 1; flex: 1;
font-weight: 500; font-weight: 500;
} }
.permission-children { .permission-children {
margin-left: 35px; margin-left: 40px;
margin-top: 8px; margin-top: 8px;
border-left: 2px solid #3498db; border-left: 2px solid #e6f7ff;
padding-left: 20px; padding-left: 16px;
display: none; display: none;
position: relative;
}
.permission-children::before {
content: '';
position: absolute;
left: 0;
top: -8px;
height: calc(100% + 16px);
width: 2px;
background-color: #3498db;
} }
.permission-children.expanded { .permission-children.expanded {
display: block; display: block;
animation: fadeIn 0.3s ease-in-out;
}
@keyframes fadeIn {
from { opacity: 0; transform: translateY(-5px); }
to { opacity: 1; transform: translateY(0); }
} }
.permission-item.parent > .permission-node { .permission-item.parent > .permission-node {
background-color: #edf7ff; background-color: #fafafa;
border-color: #bbdefb;
font-weight: 600; font-weight: 600;
} }
.permission-item.child .permission-node {
margin-left: 15px;
background-color: #f8f9fa;
}
/* 操作区域 */ /* 操作区域 */
.actions { .actions {
display: flex; display: flex;
justify-content: flex-end; justify-content: flex-end;
gap: 10px; gap: 12px;
padding-top: 20px; padding-top: 20px;
border-top: 1px solid #dee2e6;
margin-top: 30px;
} }
/* 响应式设计 */ /* 响应式设计 */
@media (max-width: 768px) { @media (max-width: 768px) {
.container { body {
margin: 0; padding: 10px;
border-radius: 0;
} }
.header { .page-header {
padding: 15px 20px; padding: 12px 16px;
flex-direction: column; flex-direction: column;
gap: 15px; gap: 12px;
align-items: flex-start;
} }
.content { .form-section {
padding: 20px; padding: 16px;
} }
.form-row { .form-row {
flex-direction: column; grid-template-columns: 1fr;
gap: 0;
} }
} }
/* 滚动条样式 */
.permissions-container::-webkit-scrollbar,
.market-dropdown::-webkit-scrollbar {
width: 6px;
}
.permissions-container::-webkit-scrollbar-track,
.market-dropdown::-webkit-scrollbar-track {
background: #f0f0f0;
border-radius: 3px;
}
.permissions-container::-webkit-scrollbar-thumb,
.market-dropdown::-webkit-scrollbar-thumb {
background: #ddd;
border-radius: 3px;
}
.permissions-container::-webkit-scrollbar-thumb:hover,
.market-dropdown::-webkit-scrollbar-thumb:hover {
background: #bbb;
}
</style> </style>
</head> </head>
<body> <body>
<div class="container"> <div class="container">
<!-- 头部 --> <!-- 页面头部 -->
<div class="header"> <div class="page-header">
<h1>市场经营者创建</h1> <h1 class="page-title">市场经营者创建</h1>
<div class="header-actions"> <div class="button-group">
<button class="btn btn-warning" onclick="resetForm()"> <button class="btn btn-default" onclick="resetForm()">重置</button>
🔄 重置
</button>
</div> </div>
</div> </div>
<!-- 内容区域 --> <!-- 账号信息 -->
<div class="content"> <div class="form-section">
<!-- 账号创建区域 --> <h2 class="section-title">账号信息</h2>
<div class="section"> <div class="form-row">
<h2 class="section-title">账号信息</h2> <div class="form-item">
<div class="account-form"> <label class="form-label">菜市场选择 *</label>
<div class="form-row"> <div class="market-select-container">
<div class="form-group"> <input type="text" id="marketSelect" class="form-input"
<label for="marketSelect">菜市场选择 *</label> placeholder="请选择菜市场" readonly onclick="toggleMarketDropdown()">
<div class="market-select-container"> <div id="marketDropdown" class="market-dropdown">
<input type="text" id="marketSelect" class="market-select" <input type="text" id="marketFilter" class="market-filter"
placeholder="请选择菜市场" readonly onclick="toggleMarketDropdown()"> placeholder="搜索菜市场..."
<div id="marketDropdown" class="market-dropdown"> oninput="filterMarkets(this.value)">
<input type="text" id="marketFilter" class="market-filter" <div id="marketOptions"></div>
placeholder="输入菜市场名称搜索..."
oninput="filterMarkets(this.value)">
<div id="marketOptions"></div>
</div>
</div>
</div>
<div class="form-group">
<label for="phoneNumber">账号手机号 *</label>
<input type="tel" id="phoneNumber" placeholder="请输入手机号" maxlength="11">
</div> </div>
</div> </div>
</div> </div>
</div> <div class="form-item">
<label class="form-label">账号手机号 *</label>
<!-- 权限分配区域 --> <input type="tel" id="phoneNumber" class="form-input" placeholder="请输入手机号" maxlength="11">
<div class="section">
<h2 class="section-title">权限分配</h2>
<div class="permissions-container">
<ul id="permissionTree" class="permission-tree">
<!-- 权限树将通过JavaScript动态生成 -->
</ul>
</div> </div>
</div> </div>
</div>
<!-- 操作按钮 --> <!-- 权限分配 -->
<div class="form-section">
<h2 class="section-title">权限分配</h2>
<div class="permissions-container">
<ul id="permissionTree" class="permission-tree">
<!-- 权限树将通过JavaScript动态生成 -->
</ul>
</div>
</div>
<!-- 操作按钮 -->
<div class="form-section">
<div class="actions"> <div class="actions">
<button class="btn btn-warning" onclick="cancelAssignment()">取消</button> <button class="btn btn-default" onclick="cancelAssignment()">取消</button>
<button class="btn btn-success" onclick="saveAssignment()">💾 保存分配</button> <button class="btn btn-success" onclick="saveAssignment()">保存分配</button>
</div> </div>
</div> </div>
</div> </div>
@ -425,7 +404,7 @@
{ id: 8, name: '建设菜市场', code: 'JIANSHE_MARKET' } { id: 8, name: '建设菜市场', code: 'JIANSHE_MARKET' }
]; ];
// 权限数据(从权限编辑页面获取的示例数据) // 权限数据
const permissions = [ const permissions = [
{ {
id: 1, id: 1,
@ -491,7 +470,7 @@
function initMarketOptions() { function initMarketOptions() {
const optionsContainer = document.getElementById('marketOptions'); const optionsContainer = document.getElementById('marketOptions');
optionsContainer.innerHTML = ''; optionsContainer.innerHTML = '';
markets.forEach(market => { markets.forEach(market => {
const option = document.createElement('div'); const option = document.createElement('div');
option.className = 'market-option'; option.className = 'market-option';
@ -507,7 +486,7 @@
selectedMarket = market; selectedMarket = market;
document.getElementById('marketSelect').value = market.name; document.getElementById('marketSelect').value = market.name;
document.getElementById('marketDropdown').classList.remove('show'); document.getElementById('marketDropdown').classList.remove('show');
// 更新选中状态 // 更新选中状态
document.querySelectorAll('.market-option').forEach(option => { document.querySelectorAll('.market-option').forEach(option => {
option.classList.toggle('selected', option.dataset.marketId == market.id); option.classList.toggle('selected', option.dataset.marketId == market.id);
@ -518,7 +497,7 @@
function toggleMarketDropdown() { function toggleMarketDropdown() {
const dropdown = document.getElementById('marketDropdown'); const dropdown = document.getElementById('marketDropdown');
dropdown.classList.toggle('show'); dropdown.classList.toggle('show');
if (dropdown.classList.contains('show')) { if (dropdown.classList.contains('show')) {
document.getElementById('marketFilter').focus(); document.getElementById('marketFilter').focus();
} }
@ -528,7 +507,7 @@
function filterMarkets(keyword) { function filterMarkets(keyword) {
const options = document.querySelectorAll('.market-option'); const options = document.querySelectorAll('.market-option');
keyword = keyword.toLowerCase(); keyword = keyword.toLowerCase();
options.forEach(option => { options.forEach(option => {
const marketName = option.textContent.toLowerCase(); const marketName = option.textContent.toLowerCase();
option.style.display = marketName.includes(keyword) ? 'block' : 'none'; option.style.display = marketName.includes(keyword) ? 'block' : 'none';
@ -540,7 +519,7 @@
const tree = buildPermissionTree(permissions); const tree = buildPermissionTree(permissions);
const treeContainer = document.getElementById('permissionTree'); const treeContainer = document.getElementById('permissionTree');
treeContainer.innerHTML = ''; treeContainer.innerHTML = '';
tree.forEach(item => { tree.forEach(item => {
treeContainer.appendChild(createPermissionNode(item)); treeContainer.appendChild(createPermissionNode(item));
}); });
@ -550,12 +529,12 @@
function buildPermissionTree(data) { function buildPermissionTree(data) {
const map = {}; const map = {};
const roots = []; const roots = [];
// 创建映射 // 创建映射
data.forEach(item => { data.forEach(item => {
map[item.code] = { ...item, children: [] }; map[item.code] = { ...item, children: [] };
}); });
// 构建树结构 // 构建树结构
data.forEach(item => { data.forEach(item => {
if (item.parentCode && map[item.parentCode]) { if (item.parentCode && map[item.parentCode]) {
@ -564,7 +543,7 @@
roots.push(map[item.code]); roots.push(map[item.code]);
} }
}); });
return roots; return roots;
} }
@ -572,10 +551,10 @@
function createPermissionNode(item) { function createPermissionNode(item) {
const li = document.createElement('li'); const li = document.createElement('li');
li.className = 'permission-item ' + (item.children.length > 0 ? 'parent' : 'child'); li.className = 'permission-item ' + (item.children.length > 0 ? 'parent' : 'child');
const nodeDiv = document.createElement('div'); const nodeDiv = document.createElement('div');
nodeDiv.className = 'permission-node'; nodeDiv.className = 'permission-node';
// 展开/收起按钮 // 展开/收起按钮
if (item.children.length > 0) { if (item.children.length > 0) {
const expandBtn = document.createElement('button'); const expandBtn = document.createElement('button');
@ -589,11 +568,11 @@
} else { } else {
// 占位符保持对齐 // 占位符保持对齐
const placeholder = document.createElement('span'); const placeholder = document.createElement('span');
placeholder.style.width = '28px'; placeholder.style.width = '24px';
placeholder.style.display = 'inline-block'; placeholder.style.display = 'inline-block';
nodeDiv.appendChild(placeholder); nodeDiv.appendChild(placeholder);
} }
// 复选框 // 复选框
const checkbox = document.createElement('input'); const checkbox = document.createElement('input');
checkbox.type = 'checkbox'; checkbox.type = 'checkbox';
@ -601,28 +580,28 @@
checkbox.dataset.permissionCode = item.code; checkbox.dataset.permissionCode = item.code;
checkbox.onchange = (e) => handlePermissionChange(e, item); checkbox.onchange = (e) => handlePermissionChange(e, item);
nodeDiv.appendChild(checkbox); nodeDiv.appendChild(checkbox);
// 权限标签 // 权限标签
const label = document.createElement('span'); const label = document.createElement('span');
label.className = 'permission-label'; label.className = 'permission-label';
label.textContent = `${item.name}${item.description || item.code}`; label.textContent = `${item.name}(${item.description || item.code})`;
label.onclick = () => checkbox.click(); label.onclick = () => checkbox.click();
nodeDiv.appendChild(label); nodeDiv.appendChild(label);
li.appendChild(nodeDiv); li.appendChild(nodeDiv);
// 子权限 // 子权限
if (item.children.length > 0) { if (item.children.length > 0) {
const childrenContainer = document.createElement('ul'); const childrenContainer = document.createElement('ul');
childrenContainer.className = 'permission-children'; childrenContainer.className = 'permission-children';
item.children.forEach(child => { item.children.forEach(child => {
childrenContainer.appendChild(createPermissionNode(child)); childrenContainer.appendChild(createPermissionNode(child));
}); });
li.appendChild(childrenContainer); li.appendChild(childrenContainer);
} }
return li; return li;
} }
@ -630,7 +609,7 @@
function togglePermissionChildren(btn, li) { function togglePermissionChildren(btn, li) {
const children = li.querySelector('.permission-children'); const children = li.querySelector('.permission-children');
if (!children) return; if (!children) return;
const isExpanded = children.classList.contains('expanded'); const isExpanded = children.classList.contains('expanded');
children.classList.toggle('expanded'); children.classList.toggle('expanded');
btn.textContent = isExpanded ? '▶' : '▼'; btn.textContent = isExpanded ? '▶' : '▼';
@ -643,52 +622,50 @@
const isChecked = checkbox.checked; const isChecked = checkbox.checked;
const permissionCode = permission.code; const permissionCode = permission.code;
const node = checkbox.closest('.permission-node'); const node = checkbox.closest('.permission-node');
// 更新节点选中样式 // 更新节点选中样式
if (node) { if (node) {
if (isChecked) { node.classList.toggle('selected', isChecked);
node.classList.add('selected');
} else {
node.classList.remove('selected');
}
} }
if (isChecked) { if (isChecked) {
selectedPermissions.add(permissionCode); selectedPermissions.add(permissionCode);
// 选中子级权限时自动选中所有父级权限 // 选中子级权限时,自动选中所有父级权限
selectParentPermissions(permission); selectParentPermissions(permission);
// 如果是父权限自动选中所有子权限 // 如果是父权限,自动选中所有子权限
if (permission.children && permission.children.length > 0) { if (permission.children && permission.children.length > 0) {
selectChildrenPermissions(permission, true); selectChildrenPermissions(permission, true);
} }
} else { } else {
selectedPermissions.delete(permissionCode); selectedPermissions.delete(permissionCode);
// 如果是父权限自动取消所有子权限 // 如果是父权限,自动取消所有子权限
if (permission.children && permission.children.length > 0) { if (permission.children && permission.children.length > 0) {
selectChildrenPermissions(permission, false); selectChildrenPermissions(permission, false);
} }
// 如果是子权限检查是否需要取消父权限 // 如果是子权限,检查是否需要取消父权限
uncheckParentIfNeeded(permission); uncheckParentIfNeeded(permission);
} }
updatePermissionDisplay(); updatePermissionDisplay();
} }
// 选择所有父级权限 // 选择所有父级权限
function selectParentPermissions(permission) { function selectParentPermissions(permission) {
if (!permission.parentCode) return; if (!permission.parentCode) return;
// 查找父权限 // 查找父权限
const parent = permissions.find(p => p.code === permission.parentCode); const parent = permissions.find(p => p.code === permission.parentCode);
if (!parent) return; if (!parent) return;
// 选中父权限 // 选中父权限
const parentCheckbox = document.querySelector(`[data-permission-code="${parent.code}"]`); const parentCheckbox = document.querySelector(`[data-permission-code="${parent.code}"]`);
if (parentCheckbox && !parentCheckbox.checked) { if (parentCheckbox && !parentCheckbox.checked) {
parentCheckbox.checked = true; parentCheckbox.checked = true;
selectedPermissions.add(parent.code); selectedPermissions.add(parent.code);
const parentNode = parentCheckbox.closest('.permission-node');
if (parentNode) parentNode.classList.add('selected');
} }
// 递归选中上级父权限 // 递归选中上级父权限
selectParentPermissions(parent); selectParentPermissions(parent);
} }
@ -696,17 +673,20 @@
// 选择/取消子权限 // 选择/取消子权限
function selectChildrenPermissions(parent, select) { function selectChildrenPermissions(parent, select) {
if (!parent.children) return; if (!parent.children) return;
parent.children.forEach(child => { parent.children.forEach(child => {
const childCheckbox = document.querySelector(`[data-permission-code="${child.code}"]`); const childCheckbox = document.querySelector(`[data-permission-code="${child.code}"]`);
if (childCheckbox) { if (childCheckbox) {
childCheckbox.checked = select; childCheckbox.checked = select;
const childNode = childCheckbox.closest('.permission-node');
if (childNode) childNode.classList.toggle('selected', select);
if (select) { if (select) {
selectedPermissions.add(child.code); selectedPermissions.add(child.code);
} else { } else {
selectedPermissions.delete(child.code); selectedPermissions.delete(child.code);
} }
// 递归处理子权限 // 递归处理子权限
selectChildrenPermissions(child, select); selectChildrenPermissions(child, select);
} }
@ -716,20 +696,22 @@
// 检查是否需要取消父权限选择 // 检查是否需要取消父权限选择
function uncheckParentIfNeeded(permission) { function uncheckParentIfNeeded(permission) {
if (!permission.parentCode) return; if (!permission.parentCode) return;
// 查找父权限 // 查找父权限
const parent = permissions.find(p => p.code === permission.parentCode); const parent = permissions.find(p => p.code === permission.parentCode);
if (!parent) return; if (!parent) return;
// 检查父权限的所有子权限是否都未选中 // 检查父权限的所有子权限是否都未选中
const siblings = permissions.filter(p => p.parentCode === parent.code); const siblings = permissions.filter(p => p.parentCode === parent.code);
const hasSelectedSibling = siblings.some(sibling => selectedPermissions.has(sibling.code)); const hasSelectedSibling = siblings.some(sibling => selectedPermissions.has(sibling.code));
if (!hasSelectedSibling) { if (!hasSelectedSibling) {
const parentCheckbox = document.querySelector(`[data-permission-code="${parent.code}"]`); const parentCheckbox = document.querySelector(`[data-permission-code="${parent.code}"]`);
if (parentCheckbox) { if (parentCheckbox) {
parentCheckbox.checked = false; parentCheckbox.checked = false;
selectedPermissions.delete(parent.code); selectedPermissions.delete(parent.code);
const parentNode = parentCheckbox.closest('.permission-node');
if (parentNode) parentNode.classList.remove('selected');
// 递归检查上级父权限 // 递归检查上级父权限
uncheckParentIfNeeded(parent); uncheckParentIfNeeded(parent);
} }
@ -738,7 +720,6 @@
// 更新权限显示 // 更新权限显示
function updatePermissionDisplay() { function updatePermissionDisplay() {
// 这里可以添加一些UI反馈比如显示已选权限数量等
console.log('已选择权限:', Array.from(selectedPermissions)); console.log('已选择权限:', Array.from(selectedPermissions));
} }
@ -748,7 +729,7 @@
document.addEventListener('click', function(e) { document.addEventListener('click', function(e) {
const dropdown = document.getElementById('marketDropdown'); const dropdown = document.getElementById('marketDropdown');
const selectContainer = document.querySelector('.market-select-container'); const selectContainer = document.querySelector('.market-select-container');
if (!selectContainer.contains(e.target)) { if (!selectContainer.contains(e.target)) {
dropdown.classList.remove('show'); dropdown.classList.remove('show');
} }
@ -766,23 +747,26 @@
// 重置表单 // 重置表单
function resetForm() { function resetForm() {
if (confirm('确定要重置所有信息吗')) { if (confirm('确定要重置所有信息吗?')) {
// 重置菜市场选择 // 重置菜市场选择
selectedMarket = null; selectedMarket = null;
document.getElementById('marketSelect').value = ''; document.getElementById('marketSelect').value = '';
document.querySelectorAll('.market-option').forEach(option => { document.querySelectorAll('.market-option').forEach(option => {
option.classList.remove('selected'); option.classList.remove('selected');
}); });
// 重置手机号 // 重置手机号
document.getElementById('phoneNumber').value = ''; document.getElementById('phoneNumber').value = '';
// 重置权限选择 // 重置权限选择
selectedPermissions.clear(); selectedPermissions.clear();
document.querySelectorAll('.permission-checkbox').forEach(checkbox => { document.querySelectorAll('.permission-checkbox').forEach(checkbox => {
checkbox.checked = false; checkbox.checked = false;
}); });
document.querySelectorAll('.permission-node').forEach(node => {
node.classList.remove('selected');
});
// 重置搜索 // 重置搜索
document.getElementById('marketFilter').value = ''; document.getElementById('marketFilter').value = '';
filterMarkets(''); filterMarkets('');
@ -791,7 +775,7 @@
// 取消分配 // 取消分配
function cancelAssignment() { function cancelAssignment() {
if (confirm('确定要取消权限分配吗未保存的更改将丢失。')) { if (confirm('确定要取消权限分配吗?未保存的更改将丢失。')) {
window.history.back(); window.history.back();
} }
} }
@ -800,40 +784,40 @@
function saveAssignment() { function saveAssignment() {
// 验证菜市场 // 验证菜市场
if (!selectedMarket) { if (!selectedMarket) {
alert('请选择菜市场'); alert('请选择菜市场!');
return; return;
} }
// 验证手机号 // 验证手机号
const phoneNumber = document.getElementById('phoneNumber').value.trim(); const phoneNumber = document.getElementById('phoneNumber').value.trim();
if (!phoneNumber) { if (!phoneNumber) {
alert('请输入手机号'); alert('请输入手机号!');
return; return;
} }
if (!/^1[3-9]\d{9}$/.test(phoneNumber)) { if (!/^1[3-9]\d{9}$/.test(phoneNumber)) {
alert('请输入正确的手机号格式'); alert('请输入正确的手机号格式!');
return; return;
} }
// 验证权限选择 // 验证权限选择
if (selectedPermissions.size === 0) { if (selectedPermissions.size === 0) {
alert('请至少选择一个权限'); alert('请至少选择一个权限!');
return; return;
} }
// 构建保存数据 // 构建保存数据
const assignmentData = { const assignmentData = {
market: selectedMarket, market: selectedMarket,
phoneNumber: phoneNumber, phoneNumber: phoneNumber,
permissions: Array.from(selectedPermissions) permissions: Array.from(selectedPermissions)
}; };
console.log('保存权限分配数据:', assignmentData); console.log('保存权限分配数据:', assignmentData);
// 这里应该调用API保存数据 // 这里应该调用API保存数据
alert('权限分配保存成功'); alert('权限分配保存成功!');
} }
</script> </script>
</body> </body>
</html> </html>

View File

@ -12,8 +12,8 @@
} }
body { body {
font-family: 'Segoe UI', Tahoma, Geneva, Verdana, sans-serif; font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', 'PingFang SC', 'Hiragino Sans GB', 'Microsoft YaHei', 'Helvetica Neue', Helvetica, Arial, sans-serif;
background-color: #f5f7fa; background-color: #f5f5f5;
color: #333; color: #333;
height: 100vh; height: 100vh;
margin: 0; margin: 0;
@ -21,142 +21,171 @@
} }
.container { .container {
width: 100%; max-width: 1600px;
height: 100vh; margin: 0 auto;
margin: 0; padding: 20px;
background-color: #fff;
border-radius: 0;
box-shadow: none;
overflow: hidden;
} }
.header { /* 页面头部 */
background: linear-gradient(135deg, #2c3e50, #3498db); .page-header {
color: #fff; background: white;
padding: 20px 30px; padding: 16px 24px;
margin-bottom: 16px;
border-radius: 8px;
box-shadow: 0 2px 8px rgba(0, 0, 0, 0.1);
display: flex; display: flex;
justify-content: space-between; justify-content: space-between;
align-items: center; align-items: center;
} }
.header h1 { .page-title {
font-size: 1.8rem; font-size: 24px;
font-weight: 600; font-weight: 600;
color: #262626;
} }
.header-actions { /* 按钮样式 */
display: flex;
gap: 10px;
}
.btn { .btn {
padding: 10px 20px; height: 40px;
padding: 0 16px;
border: none; border: none;
border-radius: 5px; border-radius: 6px;
cursor: pointer;
font-size: 14px; font-size: 14px;
font-weight: 500; cursor: pointer;
transition: all 0.3s ease;
display: inline-flex; display: inline-flex;
align-items: center; align-items: center;
justify-content: center;
transition: all 0.3s;
gap: 5px; gap: 5px;
} }
.btn-primary { .btn-primary {
background-color: #3498db; background: #1890ff;
color: #fff; color: white;
} }
.btn-primary:hover { .btn-primary:hover {
background-color: #2980b9; background: #40a9ff;
transform: translateY(-2px);
} }
.btn-success { .btn-success {
background-color: #27ae60; background: #52c41a;
color: #fff; color: white;
} }
.btn-success:hover { .btn-success:hover {
background-color: #229954; background: #73d13d;
} }
.btn-danger { .btn-danger {
background-color: #e74c3c; background: #ff4d4f;
color: #fff; color: white;
} }
.btn-danger:hover { .btn-danger:hover {
background-color: #c0392b; background: #ff7875;
} }
.btn-warning { .btn-default {
background-color: #f39c12; background: white;
color: #fff; color: #666;
border: 1px solid #ddd;
} }
.btn-warning:hover { .btn-default:hover {
background-color: #e67e22; border-color: #1890ff;
color: #1890ff;
} }
/* 搜索栏 */ .btn-sm {
.search-bar { height: 32px;
padding: 20px 30px; padding: 0 12px;
background-color: #f8f9fa; font-size: 12px;
border-bottom: 1px solid #dee2e6; }
.button-group {
display: flex;
gap: 12px;
}
/* 搜索区域 */
.search-section {
background: white;
padding: 24px;
margin-bottom: 16px;
border-radius: 8px;
box-shadow: 0 2px 8px rgba(0, 0, 0, 0.1);
} }
.search-form { .search-form {
display: flex; display: grid;
gap: 15px; grid-template-columns: repeat(auto-fit, minmax(250px, 1fr));
align-items: center; gap: 16px;
margin-bottom: 16px;
} }
.search-input { .form-item {
flex: 1; display: flex;
padding: 10px 15px; flex-direction: column;
gap: 8px;
}
.form-label {
font-size: 14px;
color: #666;
font-weight: 500;
}
.form-input {
height: 40px;
padding: 0 12px;
border: 1px solid #ddd; border: 1px solid #ddd;
border-radius: 5px; border-radius: 6px;
font-size: 14px; font-size: 14px;
} }
.search-input:focus { .form-input:focus {
outline: none; outline: none;
border-color: #3498db; border-color: #1890ff;
box-shadow: 0 0 5px rgba(52, 152, 219, 0.3); box-shadow: 0 0 0 2px rgba(24, 144, 255, 0.2);
} }
/* 表格样式 */ /* 表格样式 */
.table-container { .table-section {
padding: 0 30px 30px 30px; background: white;
border-radius: 8px;
box-shadow: 0 2px 8px rgba(0, 0, 0, 0.1);
overflow: hidden;
}
.table-wrapper {
overflow-x: auto; overflow-x: auto;
} }
.permissions-table { .permissions-table {
width: 100%; width: 100%;
border-collapse: collapse; border-collapse: collapse;
margin-top: 20px; font-size: 14px;
background-color: #fff;
}
.permissions-table th,
.permissions-table td {
padding: 12px 15px;
text-align: left;
border-bottom: 1px solid #eee;
} }
.permissions-table th { .permissions-table th {
background-color: #f8f9fa; background: #fafafa;
padding: 12px 16px;
text-align: left;
font-weight: 600; font-weight: 600;
color: #2c3e50; color: #666;
position: sticky; border-bottom: 1px solid #f0f0f0;
top: 0; white-space: nowrap;
z-index: 10;
} }
.permissions-table tr:hover { .permissions-table td {
background-color: #f8f9fa; padding: 12px 16px;
border-bottom: 1px solid #f9f9f9;
vertical-align: top;
}
.permissions-table tbody tr:hover {
background: #f8f9fa;
} }
.permission-row { .permission-row {
@ -169,7 +198,7 @@
} }
.permission-row.child-row { .permission-row.child-row {
background-color: #fafbfc; background-color: #fafafa;
} }
.permission-row.level-1 td:first-child { .permission-row.level-1 td:first-child {
@ -210,38 +239,45 @@
.permission-code { .permission-code {
font-family: monospace; font-family: monospace;
background-color: #ecf0f1; background-color: #f0f0f0;
padding: 2px 6px; padding: 4px 8px;
border-radius: 3px; border-radius: 4px;
font-size: 13px; font-size: 12px;
} }
.permission-type { .permission-type {
padding: 4px 8px; padding: 4px 8px;
border-radius: 12px; border-radius: 4px;
font-size: 12px; font-size: 12px;
font-weight: 500; font-weight: 500;
} }
.type-menu { .type-menu {
background-color: #3498db; background: #e6f7ff;
color: #fff; color: #1890ff;
border: 1px solid #91d5ff;
} }
.type-button { .type-button {
background-color: #27ae60; background: #f6ffed;
color: #fff; color: #52c41a;
border: 1px solid #b7eb8f;
} }
.expand-btn { .expand-btn {
background: none; background: none;
border: none; border: none;
cursor: pointer; cursor: pointer;
padding: 2px 5px; padding: 4px;
margin-right: 8px; margin-right: 8px;
color: #3498db; color: #1890ff;
font-size: 12px; font-size: 12px;
transition: transform 0.3s ease; transition: all 0.3s;
border-radius: 4px;
}
.expand-btn:hover {
background: #e6f7ff;
} }
.expand-btn.collapsed { .expand-btn.collapsed {
@ -253,11 +289,6 @@
gap: 5px; gap: 5px;
} }
.btn-sm {
padding: 5px 10px;
font-size: 12px;
}
/* 弹窗样式 */ /* 弹窗样式 */
.modal { .modal {
display: none; display: none;
@ -293,23 +324,24 @@
} }
.modal-header { .modal-header {
background: linear-gradient(135deg, #2c3e50, #3498db); background: #fafafa;
color: #fff; padding: 16px 24px;
padding: 20px 25px;
display: flex; display: flex;
justify-content: space-between; justify-content: space-between;
align-items: center; align-items: center;
border-bottom: 1px solid #f0f0f0;
} }
.modal-header h3 { .modal-header h3 {
font-size: 1.2rem; font-size: 18px;
font-weight: 600; font-weight: 600;
color: #262626;
} }
.close { .close {
background: none; background: none;
border: none; border: none;
color: #fff; color: #666;
font-size: 24px; font-size: 24px;
cursor: pointer; cursor: pointer;
padding: 0; padding: 0;
@ -319,15 +351,16 @@
align-items: center; align-items: center;
justify-content: center; justify-content: center;
border-radius: 50%; border-radius: 50%;
transition: background-color 0.3s ease; transition: all 0.3s;
} }
.close:hover { .close:hover {
background-color: rgba(255, 255, 255, 0.2); background-color: #f0f0f0;
color: #333;
} }
.modal-body { .modal-body {
padding: 25px; padding: 24px;
} }
.form-group { .form-group {
@ -338,26 +371,27 @@
display: block; display: block;
margin-bottom: 8px; margin-bottom: 8px;
font-weight: 500; font-weight: 500;
color: #2c3e50; color: #333;
font-size: 14px;
} }
.form-group input, .form-group input,
.form-group select, .form-group select,
.form-group textarea { .form-group textarea {
width: 100%; width: 100%;
padding: 10px 15px; padding: 10px 12px;
border: 1px solid #ddd; border: 1px solid #ddd;
border-radius: 5px; border-radius: 6px;
font-size: 14px; font-size: 14px;
transition: border-color 0.3s ease; transition: all 0.3s;
} }
.form-group input:focus, .form-group input:focus,
.form-group select:focus, .form-group select:focus,
.form-group textarea:focus { .form-group textarea:focus {
outline: none; outline: none;
border-color: #3498db; border-color: #1890ff;
box-shadow: 0 0 5px rgba(52, 152, 219, 0.3); box-shadow: 0 0 0 2px rgba(24, 144, 255, 0.2);
} }
.form-group textarea { .form-group textarea {
@ -365,36 +399,44 @@
min-height: 80px; min-height: 80px;
} }
.form-group small {
color: #666;
font-size: 12px;
margin-top: 4px;
display: block;
}
.modal-footer { .modal-footer {
background-color: #f8f9fa; background-color: #fafafa;
padding: 15px 25px; padding: 12px 24px;
display: flex; display: flex;
justify-content: flex-end; justify-content: flex-end;
gap: 10px; gap: 12px;
border-top: 1px solid #dee2e6; border-top: 1px solid #f0f0f0;
} }
/* 右键菜单样式 */ /* 右键菜单样式 */
.context-menu { .context-menu {
position: fixed; position: fixed;
background-color: #fff; background-color: #fff;
border: 1px solid #ddd; border: 1px solid #f0f0f0;
border-radius: 5px; border-radius: 6px;
box-shadow: 0 4px 12px rgba(0, 0, 0, 0.1); box-shadow: 0 4px 12px rgba(0, 0, 0, 0.1);
z-index: 2000; z-index: 2000;
padding: 5px 0; padding: 4px 0;
min-width: 120px; min-width: 120px;
display: none; display: none;
} }
.context-menu-item { .context-menu-item {
padding: 8px 15px; padding: 8px 16px;
cursor: pointer; cursor: pointer;
font-size: 14px; font-size: 14px;
transition: background-color 0.2s ease; transition: background-color 0.2s ease;
display: flex; display: flex;
align-items: center; align-items: center;
gap: 8px; gap: 8px;
color: #333;
} }
.context-menu-item:hover { .context-menu-item:hover {
@ -405,44 +447,46 @@
.empty-state { .empty-state {
text-align: center; text-align: center;
padding: 60px 20px; padding: 60px 20px;
color: #7f8c8d; color: #999;
} }
.empty-state h3 { .empty-state h3 {
margin-bottom: 10px; margin-bottom: 10px;
color: #95a5a6; color: #666;
font-size: 16px;
font-weight: 500;
}
.empty-state p {
font-size: 14px;
} }
/* 响应式设计 */ /* 响应式设计 */
@media (max-width: 768px) { @media (max-width: 768px) {
.container { .container {
margin: 0; padding: 10px;
border-radius: 0;
} }
.header { .page-header {
padding: 15px 20px; padding: 12px 16px;
flex-direction: column; flex-direction: column;
gap: 15px; gap: 12px;
align-items: flex-start;
} }
.search-bar { .search-section {
padding: 15px 20px; padding: 16px;
} }
.table-container {
padding: 0 20px 20px 20px;
}
.permissions-table { .permissions-table {
font-size: 14px; font-size: 12px;
} }
.permissions-table th, .permissions-table th,
.permissions-table td { .permissions-table td {
padding: 8px 10px; padding: 8px 10px;
} }
.modal-content { .modal-content {
width: 95vw; width: 95vw;
margin: 10px; margin: 10px;
@ -452,46 +496,52 @@
</head> </head>
<body> <body>
<div class="container"> <div class="container">
<!-- 头部 --> <!-- 页面头部 -->
<div class="header"> <div class="page-header">
<h1>权限编辑</h1> <h1 class="page-title">权限编辑</h1>
<div class="header-actions"> <div class="button-group">
<button class="btn btn-success" onclick="openAddModal()"> <button class="btn btn-success" onclick="openAddModal()">
新增权限 新增权限
</button> </button>
</div> </div>
</div> </div>
<!-- 搜索 --> <!-- 搜索区域 -->
<div class="search-bar"> <div class="search-section">
<div class="search-form"> <div class="search-form">
<input type="text" class="search-input" id="searchInput" placeholder="搜索权限编码、名称或描述..."> <div class="form-item">
<button class="btn btn-primary" onclick="searchPermissions()">🔍 搜索</button> <input type="text" class="form-input" id="searchInput" placeholder="搜索权限编码、名称或描述...">
<button class="btn btn-warning" onclick="resetSearch()">🔄 重置</button> </div>
</div>
<div class="button-group">
<button class="btn btn-primary" onclick="searchPermissions()">搜索</button>
<button class="btn btn-default" onclick="resetSearch()">重置</button>
</div> </div>
</div> </div>
<!-- 表格容器 --> <!-- 表格区域 -->
<div class="table-container"> <div class="table-section">
<table class="permissions-table"> <div class="table-wrapper">
<thead> <table class="permissions-table">
<tr> <thead>
<th>权限编码</th> <tr>
<th>权限名称</th> <th>权限编码</th>
<th>权限描述</th> <th>权限名称</th>
<th>权限类型</th> <th>权限描述</th>
<th>父级权限</th> <th>权限类型</th>
<th>可见身份</th> <th>父级权限</th>
<th>操作</th> <th>可见身份</th>
</tr> <th>操作</th>
</thead> </tr>
<tbody id="permissionsTableBody"> </thead>
<!-- 数据将通过JavaScript动态加载 --> <tbody id="permissionsTableBody">
</tbody> <!-- 数据将通过JavaScript动态加载 -->
</table> </tbody>
<div class="empty-state" id="emptyState" style="display: none;"> </table>
<h3>暂无权限数据</h3> <div class="empty-state" id="emptyState" style="display: none;">
<p>点击"新增权限"按钮开始添加权限信息</p> <h3>暂无权限数据</h3>
<p>点击"新增权限"按钮开始添加权限信息</p>
</div>
</div> </div>
</div> </div>
</div> </div>
@ -515,7 +565,7 @@
</div> </div>
<div class="form-group"> <div class="form-group">
<label for="permissionDesc">权限描述</label> <label for="permissionDesc">权限描述</label>
<textarea id="permissionDesc" name="description" placeholder="请输入权限描述(可选)"></textarea> <textarea id="permissionDesc" name="description" placeholder="请输入权限描述(可选)"></textarea>
</div> </div>
<div class="form-group"> <div class="form-group">
<label for="parentCode">父级权限编码</label> <label for="parentCode">父级权限编码</label>
@ -546,7 +596,7 @@
</form> </form>
</div> </div>
<div class="modal-footer"> <div class="modal-footer">
<button type="button" class="btn btn-warning" onclick="closeModal()">取消</button> <button type="button" class="btn btn-default" onclick="closeModal()">取消</button>
<button type="button" class="btn btn-success" onclick="savePermission()">保存</button> <button type="button" class="btn btn-success" onclick="savePermission()">保存</button>
</div> </div>
</div> </div>
@ -555,13 +605,13 @@
<!-- 右键菜单 --> <!-- 右键菜单 -->
<div id="contextMenu" class="context-menu"> <div id="contextMenu" class="context-menu">
<div class="context-menu-item" onclick="addChildPermission()"> <div class="context-menu-item" onclick="addChildPermission()">
添加子权限 添加子权限
</div> </div>
<div class="context-menu-item" onclick="editPermission()"> <div class="context-menu-item" onclick="editPermission()">
✏️ 编辑权限 编辑权限
</div> </div>
<div class="context-menu-item" onclick="deletePermission()"> <div class="context-menu-item" onclick="deletePermission()">
🗑️ 删除权限 删除权限
</div> </div>
</div> </div>
@ -636,23 +686,23 @@
function renderPermissionsTable(data = permissions) { function renderPermissionsTable(data = permissions) {
const tableBody = document.getElementById('permissionsTableBody'); const tableBody = document.getElementById('permissionsTableBody');
const emptyState = document.getElementById('emptyState'); const emptyState = document.getElementById('emptyState');
if (data.length === 0) { if (data.length === 0) {
tableBody.innerHTML = ''; tableBody.innerHTML = '';
emptyState.style.display = 'block'; emptyState.style.display = 'block';
return; return;
} }
emptyState.style.display = 'none'; emptyState.style.display = 'none';
// 构建树状结构 // 构建树状结构
const tree = buildPermissionTree(data); const tree = buildPermissionTree(data);
let html = ''; let html = '';
tree.forEach(item => { tree.forEach(item => {
html += renderPermissionRow(item, 0); html += renderPermissionRow(item, 0);
}); });
tableBody.innerHTML = html; tableBody.innerHTML = html;
} }
@ -660,12 +710,12 @@
function buildPermissionTree(data) { function buildPermissionTree(data) {
const map = {}; const map = {};
const roots = []; const roots = [];
// 创建映射 // 创建映射
data.forEach(item => { data.forEach(item => {
map[item.code] = { ...item, children: [] }; map[item.code] = { ...item, children: [] };
}); });
// 构建树结构 // 构建树结构
data.forEach(item => { data.forEach(item => {
if (item.parentCode && map[item.parentCode]) { if (item.parentCode && map[item.parentCode]) {
@ -674,7 +724,7 @@
roots.push(map[item.code]); roots.push(map[item.code]);
} }
}); });
return roots; return roots;
} }
@ -683,16 +733,16 @@
const hasChildren = item.children && item.children.length > 0; const hasChildren = item.children && item.children.length > 0;
const isParent = level === 0; const isParent = level === 0;
let rowClass = 'permission-row'; let rowClass = 'permission-row';
if (isParent) { if (isParent) {
rowClass += ' parent-row'; rowClass += ' parent-row';
} else { } else {
rowClass += ' child-row level-' + level; rowClass += ' child-row level-' + level;
} }
let html = ` let html = `
<tr class="${rowClass}" <tr class="${rowClass}"
data-id="${item.id}" data-id="${item.id}"
data-code="${item.code}" data-code="${item.code}"
oncontextmenu="showContextMenu(event, this)"> oncontextmenu="showContextMenu(event, this)">
<td> <td>
@ -710,14 +760,14 @@
</td> </td>
</tr> </tr>
`; `;
// 添加子权限行 // 添加子权限行
if (hasChildren) { if (hasChildren) {
item.children.forEach(child => { item.children.forEach(child => {
html += renderPermissionRow(child, level + 1); html += renderPermissionRow(child, level + 1);
}); });
} }
return html; return html;
} }
@ -726,35 +776,35 @@
const row = btn.closest('tr'); const row = btn.closest('tr');
const parentCode = row.dataset.code; const parentCode = row.dataset.code;
let nextRow = row.nextElementSibling; let nextRow = row.nextElementSibling;
const isExpanded = btn.textContent === '▼'; const isExpanded = btn.textContent === '▼';
btn.textContent = isExpanded ? '▶' : '▼'; btn.textContent = isExpanded ? '▶' : '▼';
btn.classList.toggle('collapsed', isExpanded); btn.classList.toggle('collapsed', isExpanded);
// 找到这个权限的所有直接和间接子权限 // 找到这个权限的所有直接和间接子权限
const childCodes = getChildPermissionCodes(parentCode); const childCodes = getChildPermissionCodes(parentCode);
// 切换相关子行的显示状态 // 切换相关子行的显示状态
while (nextRow) { while (nextRow) {
const nextRowCode = nextRow.dataset.code; const nextRowCode = nextRow.dataset.code;
const nextRowParent = permissions.find(p => p.code === nextRowCode)?.parentCode; const nextRowParent = permissions.find(p => p.code === nextRowCode)?.parentCode;
// 如果当前行是目标权限的子权限(直接或间接) // 如果当前行是目标权限的子权限(直接或间接)
if (childCodes.includes(nextRowCode)) { if (childCodes.includes(nextRowCode)) {
nextRow.style.display = isExpanded ? 'none' : 'table-row'; nextRow.style.display = isExpanded ? 'none' : 'table-row';
} else if (!isChildOfAny(nextRowCode, childCodes)) { } else if (!isChildOfAny(nextRowCode, childCodes)) {
// 如果不是子权限,说明已到达同级或上级权限,停止处理 // 如果不是子权限,说明已到达同级或上级权限,停止处理
break; break;
} }
nextRow = nextRow.nextElementSibling; nextRow = nextRow.nextElementSibling;
} }
} }
// 获取某个权限的所有子权限编码(递归) // 获取某个权限的所有子权限编码(递归)
function getChildPermissionCodes(parentCode) { function getChildPermissionCodes(parentCode) {
const children = []; const children = [];
function findChildren(code) { function findChildren(code) {
permissions.forEach(p => { permissions.forEach(p => {
if (p.parentCode === code) { if (p.parentCode === code) {
@ -763,17 +813,17 @@
} }
}); });
} }
findChildren(parentCode); findChildren(parentCode);
return children; return children;
} }
// 检查某个权限是否为指定权限列表中任一权限的子权限 // 检查某个权限是否为指定权限列表中任一权限的子权限
function isChildOfAny(code, parentCodes) { function isChildOfAny(code, parentCodes) {
const permission = permissions.find(p => p.code === code); const permission = permissions.find(p => p.code === code);
if (!permission) return false; if (!permission) return false;
return parentCodes.includes(permission.parentCode) || return parentCodes.includes(permission.parentCode) ||
(permission.parentCode && isChildOfAny(permission.parentCode, parentCodes)); (permission.parentCode && isChildOfAny(permission.parentCode, parentCodes));
} }
@ -781,7 +831,7 @@
function showContextMenu(event, row) { function showContextMenu(event, row) {
event.preventDefault(); event.preventDefault();
contextMenuTargetRow = row; contextMenuTargetRow = row;
const menu = document.getElementById('contextMenu'); const menu = document.getElementById('contextMenu');
menu.style.display = 'block'; menu.style.display = 'block';
menu.style.left = event.pageX + 'px'; menu.style.left = event.pageX + 'px';
@ -809,7 +859,7 @@
// 添加子权限 // 添加子权限
function addChildPermission() { function addChildPermission() {
if (!contextMenuTargetRow) return; if (!contextMenuTargetRow) return;
const parentCode = contextMenuTargetRow.dataset.code; const parentCode = contextMenuTargetRow.dataset.code;
openAddModal(); openAddModal();
document.getElementById('parentCode').value = parentCode; document.getElementById('parentCode').value = parentCode;
@ -819,7 +869,7 @@
// 编辑权限 // 编辑权限
function editPermission() { function editPermission() {
if (!contextMenuTargetRow) return; if (!contextMenuTargetRow) return;
const id = parseInt(contextMenuTargetRow.dataset.id); const id = parseInt(contextMenuTargetRow.dataset.id);
editPermissionById(id); editPermissionById(id);
hideContextMenu(); hideContextMenu();
@ -829,30 +879,30 @@
function editPermissionById(id) { function editPermissionById(id) {
const permission = permissions.find(p => p.id === id); const permission = permissions.find(p => p.id === id);
if (!permission) return; if (!permission) return;
currentEditingId = id; currentEditingId = id;
document.getElementById('modalTitle').textContent = '编辑权限'; document.getElementById('modalTitle').textContent = '编辑权限';
document.getElementById('permissionCode').value = permission.code; document.getElementById('permissionCode').value = permission.code;
document.getElementById('permissionName').value = permission.name; document.getElementById('permissionName').value = permission.name;
document.getElementById('permissionDesc').value = permission.description || ''; document.getElementById('permissionDesc').value = permission.description || '';
document.getElementById('permissionType').value = permission.type; document.getElementById('permissionType').value = permission.type;
// 设置可见身份选项 // 设置可见身份选项
const visibleRolesSelect = document.getElementById('visibleRoles'); const visibleRolesSelect = document.getElementById('visibleRoles');
Array.from(visibleRolesSelect.options).forEach(option => { Array.from(visibleRolesSelect.options).forEach(option => {
option.selected = permission.visibleRoles && permission.visibleRoles.includes(option.value); option.selected = permission.visibleRoles && permission.visibleRoles.includes(option.value);
}); });
updateParentCodeOptions(permission.code); updateParentCodeOptions(permission.code);
document.getElementById('parentCode').value = permission.parentCode || ''; document.getElementById('parentCode').value = permission.parentCode || '';
document.getElementById('permissionModal').classList.add('show'); document.getElementById('permissionModal').classList.add('show');
} }
// 删除权限 // 删除权限
function deletePermission() { function deletePermission() {
if (!contextMenuTargetRow) return; if (!contextMenuTargetRow) return;
const id = parseInt(contextMenuTargetRow.dataset.id); const id = parseInt(contextMenuTargetRow.dataset.id);
deletePermissionById(id); deletePermissionById(id);
hideContextMenu(); hideContextMenu();
@ -862,15 +912,15 @@
function deletePermissionById(id) { function deletePermissionById(id) {
const permission = permissions.find(p => p.id === id); const permission = permissions.find(p => p.id === id);
if (!permission) return; if (!permission) return;
// 检查是否有子权限 // 检查是否有子权限
const hasChildren = permissions.some(p => p.parentCode === permission.code); const hasChildren = permissions.some(p => p.parentCode === permission.code);
if (hasChildren) { if (hasChildren) {
alert('该权限下有子权限,请先删除子权限!'); alert('该权限下有子权限,请先删除子权限!');
return; return;
} }
if (confirm(`确定要删除权限"${permission.name}"吗`)) { if (confirm(`确定要删除权限"${permission.name}"吗?`)) {
permissions = permissions.filter(p => p.id !== id); permissions = permissions.filter(p => p.id !== id);
renderPermissionsTable(); renderPermissionsTable();
} }
@ -880,7 +930,7 @@
function updateParentCodeOptions(excludeCode = '') { function updateParentCodeOptions(excludeCode = '') {
const select = document.getElementById('parentCode'); const select = document.getElementById('parentCode');
select.innerHTML = '<option value="">无父级权限</option>'; select.innerHTML = '<option value="">无父级权限</option>';
permissions.forEach(permission => { permissions.forEach(permission => {
if (permission.code !== excludeCode && permission.type === 'menu') { if (permission.code !== excludeCode && permission.type === 'menu') {
const option = document.createElement('option'); const option = document.createElement('option');
@ -895,43 +945,43 @@
function savePermission() { function savePermission() {
const form = document.getElementById('permissionForm'); const form = document.getElementById('permissionForm');
const formData = new FormData(form); const formData = new FormData(form);
const code = formData.get('code').trim(); const code = formData.get('code').trim();
const name = formData.get('name').trim(); const name = formData.get('name').trim();
const description = formData.get('description').trim(); const description = formData.get('description').trim();
const parentCode = formData.get('parentCode'); const parentCode = formData.get('parentCode');
const type = formData.get('type'); const type = formData.get('type');
// 获取可见身份选项 // 获取可见身份选项
const visibleRolesSelect = document.getElementById('visibleRoles'); const visibleRolesSelect = document.getElementById('visibleRoles');
const selectedRoles = Array.from(visibleRolesSelect.selectedOptions).map(option => option.value); const selectedRoles = Array.from(visibleRolesSelect.selectedOptions).map(option => option.value);
// 验证 // 验证
if (!code || !name || !type) { if (!code || !name || !type) {
alert('请填写必填项'); alert('请填写必填项!');
return; return;
} }
// 检查权限编码是否重复(编辑时排除自身) // 检查权限编码是否重复(编辑时排除自身)
const existingPermission = permissions.find(p => p.code === code && p.id !== currentEditingId); const existingPermission = permissions.find(p => p.code === code && p.id !== currentEditingId);
if (existingPermission) { if (existingPermission) {
alert('权限编码已存在'); alert('权限编码已存在!');
return; return;
} }
// 验证父级权限 // 验证父级权限
if (parentCode) { if (parentCode) {
const parentPermission = permissions.find(p => p.code === parentCode); const parentPermission = permissions.find(p => p.code === parentCode);
if (!parentPermission) { if (!parentPermission) {
alert('父级权限不存在'); alert('父级权限不存在!');
return; return;
} }
if (parentPermission.type !== 'menu') { if (parentPermission.type !== 'menu') {
alert('只能选择菜单类型的权限作为父级权限'); alert('只能选择菜单类型的权限作为父级权限!');
return; return;
} }
} }
const permissionData = { const permissionData = {
code, code,
name, name,
@ -940,7 +990,7 @@
type, type,
visibleRoles: selectedRoles visibleRoles: selectedRoles
}; };
if (currentEditingId) { if (currentEditingId) {
// 编辑权限 // 编辑权限
const index = permissions.findIndex(p => p.id === currentEditingId); const index = permissions.findIndex(p => p.id === currentEditingId);
@ -952,7 +1002,7 @@
const newId = Math.max(...permissions.map(p => p.id), 0) + 1; const newId = Math.max(...permissions.map(p => p.id), 0) + 1;
permissions.push({ id: newId, ...permissionData }); permissions.push({ id: newId, ...permissionData });
} }
closeModal(); closeModal();
renderPermissionsTable(); renderPermissionsTable();
} }
@ -970,13 +1020,13 @@
renderPermissionsTable(); renderPermissionsTable();
return; return;
} }
const filteredPermissions = permissions.filter(permission => const filteredPermissions = permissions.filter(permission =>
permission.code.toLowerCase().includes(keyword) || permission.code.toLowerCase().includes(keyword) ||
permission.name.toLowerCase().includes(keyword) || permission.name.toLowerCase().includes(keyword) ||
(permission.description && permission.description.toLowerCase().includes(keyword)) (permission.description && permission.description.toLowerCase().includes(keyword))
); );
renderPermissionsTable(filteredPermissions); renderPermissionsTable(filteredPermissions);
} }
@ -1014,4 +1064,4 @@
}); });
</script> </script>
</body> </body>
</html> </html>

View File

@ -84,6 +84,10 @@
.btn-small { .btn-small {
padding: 5px 12px; padding: 5px 12px;
font-size: 12px; font-size: 12px;
max-width: 90px;
white-space: normal;
word-break: break-all;
line-height: 1.4;
} }
.btn-danger { .btn-danger {
@ -477,6 +481,11 @@
background-color: #f5f7fa; background-color: #f5f7fa;
border-color: #c0c4cc; border-color: #c0c4cc;
} }
/* 隐藏元素 */
.hidden {
display: none !important;
}
</style> </style>
</head> </head>
<body> <body>
@ -649,26 +658,9 @@
</div> </div>
<div class="modal-body"> <div class="modal-body">
<form id="detailMarketForm"> <form id="detailMarketForm">
<!-- 经营者信息 -->
<div class="form-section">
<div class="form-section-title">经营者信息</div>
<div class="form-group">
<div class="form-label">经营者账号</div>
<div class="form-input-wrapper">
<input type="text" class="form-input" id="detailOperatorAccount" readonly>
</div>
</div>
<div class="form-group">
<div class="form-label">经营者姓名</div>
<div class="form-input-wrapper">
<input type="text" class="form-input" id="detailOperatorName" readonly>
</div>
</div>
</div>
<!-- 基础配置 --> <!-- 基础配置 -->
<div class="form-section"> <div class="form-section">
<div class="form-section-title">基础配置</div> <div class="form-section-title">基础信息</div>
<div class="form-group"> <div class="form-group">
<div class="form-label">菜市场名称</div> <div class="form-label">菜市场名称</div>
@ -677,47 +669,6 @@
</div> </div>
</div> </div>
<div class="form-group">
<div class="form-label">商品类目类型</div>
<div class="form-input-wrapper">
<input type="text" class="form-input" id="detailCategoryType" readonly>
</div>
</div>
<div class="form-group">
<div class="form-label">菜市场背景图</div>
<div class="form-input-wrapper">
<div class="upload-box" style="pointer-events: none;">
<span class="upload-box-text">暂无图片</span>
</div>
</div>
</div>
<div class="form-group">
<div class="form-label">菜市场营业执照</div>
<div class="form-input-wrapper">
<div class="upload-box" style="pointer-events: none;">
<span class="upload-box-text">暂无图片</span>
</div>
</div>
</div>
<div class="form-group">
<div class="form-label">菜市场实拍图</div>
<div class="form-input-wrapper">
<div class="upload-box" style="pointer-events: none;">
<span class="upload-box-text">暂无图片</span>
</div>
</div>
</div>
<div class="form-group">
<div class="form-label">菜市场地址</div>
<div class="form-input-wrapper">
<input type="text" class="form-input" id="detailAddress" readonly>
</div>
</div>
<div class="form-group"> <div class="form-group">
<div class="form-label">详细地址</div> <div class="form-label">详细地址</div>
<div class="form-input-wrapper"> <div class="form-input-wrapper">
@ -726,93 +677,14 @@
</div> </div>
<div class="form-group"> <div class="form-group">
<div class="form-label">菜市场定位</div> <div class="form-label">定位信息</div>
<div class="form-input-wrapper"> <div class="form-input-wrapper">
<input type="text" class="form-input" id="detailLocation" readonly> <div style="width: 100%; height: 300px; border: 1px solid #dcdfe6; border-radius: 4px; background-color: #f5f7fa; display: flex; align-items: center; justify-content: center; color: #909399; font-size: 14px;">
</div> <div style="text-align: center;">
</div> <div style="font-size: 48px; margin-bottom: 10px;">📍</div>
<div>地图定位显示区域</div>
<div class="form-group"> <div style="font-size: 12px; margin-top: 5px;" id="detailLocationCoords"></div>
<div class="form-label">菜市场营业时间</div>
<div class="form-input-wrapper">
<div class="switch-group">
<span class="switch-label">手动营业</span>
<label class="switch">
<input type="checkbox" id="detailAutoBusinessTime" disabled>
<span class="switch-slider"></span>
</label>
<span class="switch-label active">自动营业</span>
</div>
<div class="checkbox-group" style="margin-top: 10px;">
<div class="checkbox-option">
<input type="checkbox" id="detailEveryday" disabled>
<label for="detailEveryday">每天</label>
</div> </div>
<div class="checkbox-option">
<input type="checkbox" id="detailWorkday" disabled>
<label for="detailWorkday">工作日</label>
</div>
<div class="checkbox-option">
<input type="checkbox" id="detailCustom" disabled>
<label for="detailCustom">自定义</label>
</div>
</div>
<div class="time-group" style="margin-top: 10px;">
<input type="time" class="form-input time-input" id="detailStartTime" readonly>
<input type="time" class="form-input time-input" id="detailEndTime" readonly>
</div>
</div>
</div>
<div class="form-group">
<div class="form-label">菜市场状态</div>
<div class="form-input-wrapper">
<div class="switch-group">
<span class="switch-label">禁用</span>
<label class="switch">
<input type="checkbox" id="detailMarketStatus" disabled>
<span class="switch-slider"></span>
</label>
<span class="switch-label active">启用</span>
</div>
</div>
</div>
<div class="form-group">
<div class="form-label">摊位数</div>
<div class="form-input-wrapper">
<input type="text" class="form-input" id="detailBoothCount" readonly>
</div>
</div>
<div class="form-group">
<div class="form-label">营业状态</div>
<div class="form-input-wrapper">
<input type="text" class="form-input" id="detailBusinessStatus" readonly>
</div>
</div>
</div>
<!-- 配送设置 -->
<div class="form-section">
<div class="form-section-title">配送设置</div>
<div class="form-group">
<div class="form-label"></div>
<div class="form-input-wrapper">
<div class="form-hint"> 开启后,将允许该菜市场经营者/摊主(无经营者)自由配置是否呼叫第三方骑手进行配送;</div>
<div class="form-hint">第三方配送费用因距离、天气、时段等因素会产生不同程度的浮动;将根据实际配送费用进行扣除。</div>
</div>
</div>
<div class="form-group">
<div class="form-label">是否允许第三方骑手配送</div>
<div class="form-input-wrapper">
<div class="switch-group">
<span class="switch-label">禁用</span>
<label class="switch">
<input type="checkbox" id="detailThirdPartyDelivery" disabled>
<span class="switch-slider"></span>
</label>
<span class="switch-label active">启用</span>
</div> </div>
</div> </div>
</div> </div>
@ -945,11 +817,15 @@
<div class="form-group"> <div class="form-group">
<div class="form-label required">菜市场定位</div> <div class="form-label required">菜市场定位</div>
<div class="form-input-wrapper"> <div class="form-input-wrapper">
<div class="location-input"> <div style="width: 100%; height: 300px; border: 1px solid #dcdfe6; border-radius: 4px; background-color: #f5f7fa; display: flex; align-items: center; justify-content: center; color: #909399; font-size: 14px; cursor: pointer;" onclick="selectLocation()">
<input type="text" class="form-input" placeholder="请在地图上选择定位"> <div style="text-align: center;">
<span class="location-icon">📍</span> <div style="font-size: 48px; margin-bottom: 10px;">📍</div>
<div>点击选择地图定位</div>
<div style="font-size: 12px; margin-top: 5px; color: #f56c6c;" id="selectedLocationText">请选择定位</div>
</div>
</div> </div>
<div class="form-hint" style="color: #f56c6c;">请选择定位</div> <input type="hidden" id="locationLat" value="">
<input type="hidden" id="locationLng" value="">
</div> </div>
</div> </div>
@ -959,12 +835,12 @@
<div class="switch-group"> <div class="switch-group">
<span class="switch-label">手动营业</span> <span class="switch-label">手动营业</span>
<label class="switch"> <label class="switch">
<input type="checkbox" id="autoBusinessTime" checked> <input type="checkbox" id="autoBusinessTime" checked onchange="toggleBusinessTimeOptions()">
<span class="switch-slider"></span> <span class="switch-slider"></span>
</label> </label>
<span class="switch-label active">自动营业</span> <span class="switch-label active">自动营业</span>
</div> </div>
<div class="checkbox-group" style="margin-top: 10px;"> <div class="checkbox-group" id="businessDayOptions" style="margin-top: 10px;">
<div class="checkbox-option"> <div class="checkbox-option">
<input type="checkbox" id="everyday"> <input type="checkbox" id="everyday">
<label for="everyday">每天</label> <label for="everyday">每天</label>
@ -978,7 +854,7 @@
<label for="custom">自定义</label> <label for="custom">自定义</label>
</div> </div>
</div> </div>
<div class="time-group" style="margin-top: 10px;"> <div class="time-group" id="businessTimeRange" style="margin-top: 10px;">
<input type="time" class="form-input time-input" placeholder="开始营业时间"> <input type="time" class="form-input time-input" placeholder="开始营业时间">
<input type="time" class="form-input time-input" placeholder="结束营业时间"> <input type="time" class="form-input time-input" placeholder="结束营业时间">
</div> </div>
@ -1039,6 +915,39 @@
let countdown = 0; let countdown = 0;
let countdownTimer = null; let countdownTimer = null;
// 切换营业时间选项的显示/隐藏
function toggleBusinessTimeOptions() {
const autoBusinessTime = document.getElementById('autoBusinessTime');
const businessDayOptions = document.getElementById('businessDayOptions');
const businessTimeRange = document.getElementById('businessTimeRange');
if (autoBusinessTime.checked) {
// 自动营业,显示时间范围和日期选项
businessDayOptions.classList.remove('hidden');
businessTimeRange.classList.remove('hidden');
} else {
// 手动营业,隐藏时间范围和日期选项
businessDayOptions.classList.add('hidden');
businessTimeRange.classList.add('hidden');
}
}
// 选择地图定位
function selectLocation() {
// 这里应该打开地图选择器实际应调用地图API
// 模拟选择定位
alert('此处应打开地图选择器实际开发时需要集成地图API如高德地图、百度地图等');
// 模拟选择了一个位置
const mockLat = 31.230416;
const mockLng = 121.473701;
document.getElementById('locationLat').value = mockLat;
document.getElementById('locationLng').value = mockLng;
document.getElementById('selectedLocationText').textContent = '已选择: ' + mockLat + ', ' + mockLng;
document.getElementById('selectedLocationText').style.color = '#67c23a';
}
// 发送验证码 // 发送验证码
function sendVerifyCode() { function sendVerifyCode() {
const account = document.getElementById('operatorAccount').value; const account = document.getElementById('operatorAccount').value;
@ -1108,6 +1017,11 @@
} }
// 重置表单 // 重置表单
document.getElementById('addMarketForm').reset(); document.getElementById('addMarketForm').reset();
// 重置定位信息
document.getElementById('locationLat').value = '';
document.getElementById('locationLng').value = '';
document.getElementById('selectedLocationText').textContent = '请选择定位';
document.getElementById('selectedLocationText').style.color = '#f56c6c';
} }
// 提交表单 // 提交表单
@ -1237,31 +1151,16 @@
// 打开详情模态框 // 打开详情模态框
function openDetailModal(rowData) { function openDetailModal(rowData) {
// 填充详情数据 // 填充详情数据
document.getElementById('detailOperatorAccount').value = rowData.operatorAccount || '';
document.getElementById('detailOperatorName').value = rowData.operatorName || '';
document.getElementById('detailMarketName').value = rowData.marketName || ''; document.getElementById('detailMarketName').value = rowData.marketName || '';
document.getElementById('detailDetailAddress').value = rowData.address || '';
// 填充商品类目类型,转换为中文显示 // 显示定位坐标信息(模拟数据,实际应从后端获取)
let categoryTypeText = ''; const coordsElement = document.getElementById('detailLocationCoords');
if (rowData.categoryType === 'offline') { if (rowData.location) {
categoryTypeText = '线下菜市场'; coordsElement.textContent = '经纬度: ' + rowData.location;
} else if (rowData.categoryType === 'cloud') {
categoryTypeText = '云店';
} else { } else {
categoryTypeText = rowData.categoryType || ''; coordsElement.textContent = '暂无定位信息';
} }
document.getElementById('detailCategoryType').value = categoryTypeText;
document.getElementById('detailAddress').value = rowData.address || '';
document.getElementById('detailDetailAddress').value = rowData.detailAddress || '';
document.getElementById('detailLocation').value = rowData.location || '';
document.getElementById('detailBoothCount').value = rowData.boothCount || '';
document.getElementById('detailBusinessStatus').value = rowData.businessStatus || '';
// 设置开关状态
document.getElementById('detailAutoBusinessTime').checked = rowData.autoBusinessTime || false;
document.getElementById('detailMarketStatus').checked = rowData.marketStatus || false;
document.getElementById('detailThirdPartyDelivery').checked = rowData.thirdPartyDelivery || false;
// 显示模态框 // 显示模态框
document.getElementById('detailMarketModal').classList.add('active'); document.getElementById('detailMarketModal').classList.add('active');