Переглянути джерело

feat(physical): 新增门店管理功能

- 实现门店信息的增删改查接口
- 开发门店列表页面及操作功能
- 集成高德地图选址功能
- 支持门店图标和背景图上传
- 添加门店自定义标签管理
- 实现门店营业时间设置
- 集成门店类型下拉选择
- 添加门店状态控制功能
fugui001 2 тижнів тому
батько
коміт
5318e35348

+ 1 - 0
package.json

@@ -20,6 +20,7 @@
     "url": "https://gitee.com/JavaLionLi/plus-ui.git"
   },
   "dependencies": {
+    "@amap/amap-jsapi-loader": "^1.0.1",
     "@element-plus/icons-vue": "2.3.1",
     "@highlightjs/vue-plugin": "2.1.0",
     "@tinymce/tinymce-vue": "^6.2.0",

+ 63 - 0
src/api/system/physical/store/index.ts

@@ -0,0 +1,63 @@
+import request from '@/utils/request';
+import { AxiosPromise } from 'axios';
+import { StoreVO, StoreForm, StoreQuery } from '@/api/system/physical/store/types';
+
+/**
+ * 查询门店信息列表
+ * @param query
+ * @returns {*}
+ */
+
+export const listStore = (query?: StoreQuery): AxiosPromise<StoreVO[]> => {
+  return request({
+    url: '/physical/store/list',
+    method: 'get',
+    params: query
+  });
+};
+
+/**
+ * 查询门店信息详细
+ * @param id
+ */
+export const getStore = (id: string | number): AxiosPromise<StoreVO> => {
+  return request({
+    url: '/physical/store/' + id,
+    method: 'get'
+  });
+};
+
+/**
+ * 新增门店信息
+ * @param data
+ */
+export const addStore = (data: StoreForm) => {
+  return request({
+    url: '/physical/store',
+    method: 'post',
+    data: data
+  });
+};
+
+/**
+ * 修改门店信息
+ * @param data
+ */
+export const updateStore = (data: StoreForm) => {
+  return request({
+    url: '/physical/store',
+    method: 'put',
+    data: data
+  });
+};
+
+/**
+ * 删除门店信息
+ * @param id
+ */
+export const delStore = (id: string | number | Array<string | number>) => {
+  return request({
+    url: '/physical/store/' + id,
+    method: 'delete'
+  });
+};

+ 219 - 0
src/api/system/physical/store/types.ts

@@ -0,0 +1,219 @@
+export interface StoreVO {
+  /**
+   * 主键ID
+   */
+  id: string | number;
+
+  /**
+   * 门店编号,如M0001
+   */
+  storeCode: string;
+
+  /**
+   * 门店名称,最多20字
+   */
+  name: string;
+
+  /**
+   * 门店类型ID,关联service_type表
+   */
+  storeTypeId: string | number;
+
+  /**
+   * 所属赛区分组ID,关联zone表
+   */
+  zoneId: string | number;
+
+  /**
+   * 详细地址
+   */
+  address: string;
+
+  /**
+   * 门店图标URL
+   */
+  iconUrl: string;
+
+  /**
+   * 门店背景图URL
+   */
+  backgroundUrl: string;
+
+  /**
+   * 营业开始时间
+   */
+  businessStartTime: string;
+
+  /**
+   * 营业结束时间
+   */
+  businessEndTime: string;
+
+  /**
+   * 状态:1=开业中,2=闭店中
+   */
+  status: number;
+
+  /**
+   *
+   */
+  createdAt: string;
+
+  /**
+   *
+   */
+  updatedAt: string;
+
+  /**
+   * 软删除时间
+   */
+  deletedAt: string;
+
+  customTags: string[]; // 初始化为空数组
+}
+
+export interface StoreForm extends BaseEntity {
+  /**
+   * 主键ID
+   */
+  id?: string | number;
+
+  /**
+   * 门店编号,如M0001
+   */
+  storeCode?: string;
+
+  /**
+   * 门店名称,最多20字
+   */
+  name?: string;
+
+  /**
+   * 门店类型ID,关联service_type表
+   */
+  storeTypeId?: string | number;
+
+  /**
+   * 所属赛区分组ID,关联zone表
+   */
+  zoneId?: string | number;
+
+  /**
+   * 详细地址
+   */
+  address?: string;
+
+  /**
+   * 门店图标URL
+   */
+  iconUrl?: string;
+
+  /**
+   * 门店背景图URL
+   */
+  backgroundUrl?: string;
+  phone?: string;
+
+  /**
+   * 营业开始时间
+   */
+  businessStartTime?: string;
+
+  /**
+   * 营业结束时间
+   */
+  businessEndTime?: string;
+
+  /**
+   * 状态:1=开业中,2=闭店中
+   */
+  status?: number;
+
+  /**
+   *
+   */
+  createdAt?: string;
+
+  /**
+   *
+   */
+  updatedAt?: string;
+
+  /**
+   * 软删除时间
+   */
+  deletedAt?: string;
+  customTags: string[]; // 初始化为空数组
+}
+
+export interface StoreQuery extends PageQuery {
+  /**
+   * 门店编号,如M0001
+   */
+  storeCode?: string;
+
+  /**
+   * 门店名称,最多20字
+   */
+  name?: string;
+
+  /**
+   * 门店类型ID,关联service_type表
+   */
+  storeTypeId?: string | number;
+
+  /**
+   * 所属赛区分组ID,关联zone表
+   */
+  zoneId?: string | number;
+
+  /**
+   * 详细地址
+   */
+  address?: string;
+
+  /**
+   * 门店图标URL
+   */
+  iconUrl?: string;
+
+  /**
+   * 门店背景图URL
+   */
+  backgroundUrl?: string;
+
+  /**
+   * 营业开始时间
+   */
+  businessStartTime?: string;
+
+  /**
+   * 营业结束时间
+   */
+  businessEndTime?: string;
+
+  /**
+   * 状态:1=开业中,2=闭店中
+   */
+  status?: number;
+
+  /**
+   *
+   */
+  createdAt?: string;
+
+  /**
+   *
+   */
+  updatedAt?: string;
+
+  /**
+   * 软删除时间
+   */
+  deletedAt?: string;
+
+  /**
+   * 日期范围参数
+   */
+  params?: any;
+}

+ 6 - 0
src/api/system/physical/tag/index.ts

@@ -61,3 +61,9 @@ export const delTag = (id: string | number | Array<string | number>) => {
     method: 'delete'
   });
 };
+export const selectAllPhysicalTagsSelList = (): AxiosPromise<TagVO[]> => {
+  return request({
+    url: '/physical/tag/selectAllPhysicalTagsSelList',
+    method: 'get'
+  });
+};

+ 1 - 1
src/views/system/business/tournaments/index.vue

@@ -1799,7 +1799,7 @@ const isProdEnvironment = computed(() => {
 const selectOptions = ref<TournamentsVO[]>([]);
 const selectLoading = ref(false);
 const selectPageNum = ref(1);
-const selectPageSize = ref(30); // 每页数量
+const selectPageSize = ref(100); // 每页数量
 const selectTotal = ref(0);
 // 远程搜索方法
 const remoteMethod = async (query: string) => {

+ 865 - 0
src/views/system/physical/store/index.vue

@@ -0,0 +1,865 @@
+<template>
+  <div class="p-2">
+    <transition :enter-active-class="proxy?.animate.searchAnimate.enter" :leave-active-class="proxy?.animate.searchAnimate.leave">
+      <div v-show="showSearch" class="mb-[10px]">
+        <el-card shadow="hover">
+          <el-form ref="queryFormRef" :model="queryParams" :inline="true">
+            <el-form-item label="门店编号" prop="storeCode">
+              <el-input v-model="queryParams.storeCode" placeholder="请输入门店编号" clearable @keyup.enter="handleQuery" />
+            </el-form-item>
+            <el-form-item label="门店名称" prop="name">
+              <el-input v-model="queryParams.name" placeholder="请输入门店名称" clearable @keyup.enter="handleQuery" />
+            </el-form-item>
+            <el-form-item label="详细地址" prop="address">
+              <el-input v-model="queryParams.address" placeholder="请输入详细地址" clearable @keyup.enter="handleQuery" />
+            </el-form-item>
+            <el-form-item>
+              <el-button type="primary" icon="Search" @click="handleQuery">搜索</el-button>
+              <el-button icon="Refresh" @click="resetQuery">重置</el-button>
+            </el-form-item>
+          </el-form>
+        </el-card>
+      </div>
+    </transition>
+
+    <el-card shadow="never">
+      <template #header>
+        <el-row :gutter="10" class="mb8">
+          <el-col :span="1.5">
+            <el-button type="primary" plain icon="Plus" @click="handleAdd" v-hasPermi="['physical:store:add']">新增</el-button>
+          </el-col>
+          <el-col :span="1.5">
+            <el-button type="success" plain icon="Edit" :disabled="single" @click="handleUpdate()" v-hasPermi="['physical:store:edit']"
+              >修改</el-button
+            >
+          </el-col>
+          <el-col :span="1.5">
+            <el-button type="danger" plain icon="Delete" :disabled="multiple" @click="handleDelete()" v-hasPermi="['physical:store:remove']"
+              >删除</el-button
+            >
+          </el-col>
+          <el-col :span="1.5">
+            <el-button type="warning" plain icon="Download" @click="handleExport" v-hasPermi="['physical:store:export']">导出</el-button>
+          </el-col>
+          <right-toolbar v-model:showSearch="showSearch" @queryTable="getList"></right-toolbar>
+        </el-row>
+      </template>
+
+      <el-table v-loading="loading" border :data="storeList" @selection-change="handleSelectionChange">
+        <el-table-column type="selection" width="55" align="center" />
+        <el-table-column label="编号" align="center" prop="id" v-if="true" />
+        <el-table-column label="门店编号" align="center" prop="storeCode" />
+        <el-table-column label="门店名称" align="center" prop="name" />
+        <el-table-column label="门店类型" align="center" prop="storeTypeName" />
+        <el-table-column label="所属赛区" align="center" prop="zoneId" />
+        <el-table-column label="详细地址" align="center" prop="address" />
+        <el-table-column label="门店图标" align="center" prop="iconUrl" width="120">
+          <template #default="scope">
+            <img v-if="scope.row.iconUrl" :src="scope.row.iconUrl" alt="" style="width: 40px; height: 40px; object-fit: cover; border-radius: 4px" />
+            <span v-else>无</span>
+          </template>
+        </el-table-column>
+
+        <el-table-column label="门店背景图" align="center" prop="backgroundUrl" width="120">
+          <template #default="scope">
+            <img
+              v-if="scope.row.backgroundUrl"
+              :src="scope.row.backgroundUrl"
+              alt=""
+              style="width: 40px; height: 40px; object-fit: cover; border-radius: 4px"
+            />
+            <span v-else>无</span>
+          </template>
+        </el-table-column>
+        <el-table-column label="营业开始时间" align="center" prop="businessStartTime" width="180">
+          <template #default="scope">
+            <span>{{ parseTime(scope.row.businessStartTime, '{h}:{i}:{s}') }}</span>
+          </template>
+        </el-table-column>
+
+        <el-table-column label="营业结束时间" align="center" prop="businessEndTime" width="180">
+          <template #default="scope">
+            <span>{{ parseTime(scope.row.businessEndTime, '{h}:{i}:{s}') }}</span>
+          </template>
+        </el-table-column>
+        <el-table-column label="状态" align="center" prop="status">
+          <template #default="scope">
+            <el-tag :type="scope.row.status === 1 ? 'success' : 'info'">
+              {{ scope.row.status === 1 ? '开业中' : '闭店中' }}
+            </el-tag>
+          </template>
+        </el-table-column>
+        <el-table-column label="创建时间" align="center" prop="createdAt" width="180"></el-table-column>
+        <el-table-column label="操作" align="center" class-name="small-padding fixed-width">
+          <template #default="scope">
+            <el-tooltip content="修改" placement="top">
+              <el-button link type="primary" icon="Edit" @click="handleUpdate(scope.row)" v-hasPermi="['physical:store:edit']"></el-button>
+            </el-tooltip>
+            <el-tooltip content="删除" placement="top">
+              <el-button link type="primary" icon="Delete" @click="handleDelete(scope.row)" v-hasPermi="['physical:store:remove']"></el-button>
+            </el-tooltip>
+          </template>
+        </el-table-column>
+      </el-table>
+
+      <pagination v-show="total > 0" :total="total" v-model:page="queryParams.pageNum" v-model:limit="queryParams.pageSize" @pagination="getList" />
+    </el-card>
+    <!-- 添加或修改门店信息对话框 -->
+    <el-dialog :title="dialog.title" v-model="dialog.visible" width="600px" append-to-body>
+      <el-form ref="storeFormRef" :model="form" :rules="rules" label-width="120px">
+        <el-form-item label="门店编号" prop="storeCode">
+          <el-input v-model="form.storeCode" placeholder="请输入门店编号,如M0001" />
+        </el-form-item>
+        <el-form-item label="门店名称" prop="name">
+          <el-input v-model="form.name" placeholder="请输入门店名称,最多20字" />
+        </el-form-item>
+        <el-form-item label="门店类型" prop="storeTypeId">
+          <el-select v-model="form.storeTypeId" placeholder="请选择门店类型" style="width: 100%">
+            <el-option v-for="item in tagOptions" :key="item.id" :label="item.serviceName" :value="item.id" />
+          </el-select>
+        </el-form-item>
+        <el-form-item label="所属赛区" prop="zoneId">
+          <el-input v-model="form.zoneId" placeholder="请输入所属赛区" />
+        </el-form-item>
+        <el-form-item label="详细地址" prop="address">
+          <el-input v-model="form.address" type="textarea" placeholder="请输入内容" />
+          <!-- 添加地图选择按钮 -->
+          <el-button type="primary" @click="openMapSelector">选择地图位置</el-button>
+        </el-form-item>
+        <el-form-item label="门店图标" prop="iconUrl">
+          <div class="upload-container">
+            <el-upload
+              class="upload-icon"
+              action="#"
+              :on-change="handleIconChange"
+              :on-remove="handleIconRemove"
+              :file-list="fileList"
+              :auto-upload="false"
+              :limit="1"
+              accept="image/*"
+            >
+              <template #trigger>
+                <el-button type="primary">点击选择图标</el-button>
+              </template>
+
+              <!-- 预览图区域 -->
+              <template #default>
+                <div class="preview-area" @click="handlePreviewClick">
+                  <img
+                    v-if="iconPreviewUrl || form.iconUrl"
+                    :src="iconPreviewUrl || form.iconUrl"
+                    alt="预览图"
+                    style="max-width: 100px; max-height: 100px; margin-top: 10px; cursor: pointer"
+                  />
+                  <div v-else style="margin-top: 10px; color: #999">无</div>
+                </div>
+              </template>
+
+              <template #tip>
+                <div class="el-upload__tip">
+                  <span v-if="fileList.length > 0">当前已选文件:{{ fileList[0].name }}</span>
+                </div>
+              </template>
+            </el-upload>
+          </div>
+        </el-form-item>
+
+        <el-form-item label="门店背景" prop="backgroundUrl">
+          <div class="upload-container">
+            <el-upload
+              class="upload-icon"
+              action="#"
+              :on-change="handleIconChange2"
+              :on-remove="handleIconRemove2"
+              :file-list="fileList2"
+              :auto-upload="false"
+              :limit="1"
+              accept="image/*"
+            >
+              <template #trigger>
+                <el-button type="primary">点击选择背景</el-button>
+              </template>
+
+              <!-- 预览图区域 -->
+              <template #default>
+                <div class="preview-area" @click="handlePreviewClick2">
+                  <img
+                    v-if="iconPreviewUrl2 || form.backgroundUrl"
+                    :src="iconPreviewUrl2 || form.backgroundUrl"
+                    alt="预览图"
+                    style="max-width: 100px; max-height: 100px; margin-top: 10px; cursor: pointer"
+                  />
+                  <div v-else style="margin-top: 10px; color: #999">无</div>
+                </div>
+              </template>
+
+              <template #tip>
+                <div class="el-upload__tip">
+                  <span v-if="fileList2.length > 0">当前已选文件:{{ fileList2[0].name }}</span>
+                </div>
+              </template>
+            </el-upload>
+          </div>
+        </el-form-item>
+
+        <el-form-item label="门店标签" prop="customTags">
+          <div class="tag-container" style="display: flex; align-items: center; gap: 8px">
+            <!-- 已选标签 -->
+            <el-tag
+              v-for="(tag, index) in customTags"
+              :key="index"
+              closable
+              @close="handleTagClose(index)"
+              style="margin-right: 8px; margin-bottom: 8px"
+            >
+              {{ tag }}
+            </el-tag>
+
+            <!-- 添加新标签按钮 -->
+            <el-button type="primary" size="small" plain @click="openAddTagDialog" style="margin-top: 4px"> + 添加标签 </el-button>
+          </div>
+        </el-form-item>
+
+        <el-form-item label="营业时间" prop="businessStartTime">
+          <el-time-picker
+            v-model="form.businessStartTime"
+            format="HH:mm:ss"
+            value-format="HH:mm:ss"
+            placeholder="请选择营业开始时间"
+            :picker-options="{
+              start: '00:00',
+              end: '23:59',
+              step: '00:30'
+            }"
+          />
+        </el-form-item>
+
+        <el-form-item label="闭店时间" prop="businessEndTime">
+          <el-time-picker
+            v-model="form.businessEndTime"
+            format="HH:mm:ss"
+            value-format="HH:mm:ss"
+            placeholder="请选择营业结束时间"
+            :picker-options="{
+              start: '00:00',
+              end: '23:59',
+              step: '00:30'
+            }"
+          />
+        </el-form-item>
+
+        <el-form-item label="联系方式" prop="backgroundUrl">
+          <el-input v-model="form.phone" type="textarea" placeholder="请输入内容" />
+        </el-form-item>
+        <el-form-item label="状态" prop="status">
+          <el-radio-group v-model="form.status">
+            <el-radio :label="1">开业中</el-radio>
+            <el-radio :label="2">闭店中</el-radio>
+          </el-radio-group>
+        </el-form-item>
+      </el-form>
+      <template #footer>
+        <div class="dialog-footer">
+          <el-button :loading="buttonLoading" type="primary" @click="submitForm">确 定</el-button>
+          <el-button @click="cancel">取 消</el-button>
+        </div>
+      </template>
+    </el-dialog>
+
+    <!-- 地图选择对话框 -->
+    <el-dialog title="选择地理位置" v-model="mapDialogVisible" width="800px" append-to-body>
+      <div style="height: 500px">
+        <!-- 搜索栏 -->
+        <div style="margin-bottom: 10px; display: flex">
+          <el-input v-model="searchKeyword" placeholder="请输入地址关键字搜索" style="flex: 1" @keyup.enter="searchAddress">
+            <template #append>
+              <el-button @click="searchAddress">搜索</el-button>
+            </template>
+          </el-input>
+          <el-button @click="clearSearch" style="margin-left: 10px">清空</el-button>
+        </div>
+
+        <!-- 地图容器 -->
+        <div id="amap-container" style="width: 100%; height: 400px"></div>
+
+        <!-- 选中地址显示 -->
+        <div style="margin-top: 10px">
+          <el-input v-model="selectedAddress" placeholder="点击地图选择位置或搜索地址" readonly>
+            <template #prepend>选中地址:</template>
+          </el-input>
+        </div>
+      </div>
+
+      <template #footer>
+        <div class="dialog-footer">
+          <el-button @click="cancelMapSelection">取 消</el-button>
+          <el-button type="primary" @click="confirmMapSelection" :disabled="!selectedAddress">确 定</el-button>
+        </div>
+      </template>
+    </el-dialog>
+    <!-- 图片预览弹窗 -->
+    <el-dialog v-model="dialogVisible" title="图片预览" width="50%">
+      <img :src="previewSrc || iconPreviewUrl || form.iconUrl" alt="预览图片" style="max-width: 100%; max-height: 80vh" />
+    </el-dialog>
+
+    <el-dialog v-model="dialogVisible2" title="图片预览" width="50%">
+      <img :src="previewSrc2 || iconPreviewUrl2 || form.backgroundUrl" alt="预览图片" style="max-width: 100%; max-height: 80vh" />
+    </el-dialog>
+
+    <!-- 添加标签弹窗 -->
+    <el-dialog title="添加门店标签" v-model="addTagDialogVisible" width="400px">
+      <el-input v-model="newTagName" placeholder="请输入标签名称" clearable @keyup.enter="confirmAddTag" style="width: 100%; margin-bottom: 16px" />
+      <template #footer>
+        <el-button @click="addTagDialogVisible = false">取消</el-button>
+        <el-button type="primary" @click="confirmAddTag">确定</el-button>
+      </template>
+    </el-dialog>
+  </div>
+</template>
+
+<script setup name="Store" lang="ts">
+import { listStore, getStore, delStore, addStore, updateStore } from '@/api/system/physical/store';
+import { StoreVO, StoreQuery, StoreForm } from '@/api/system/physical/store/types';
+import { uploadTournament } from '@/api/system/business/tournaments';
+import { selectAllPhysicalTagsSelList } from '@/api/system/physical/tag';
+
+// 引入高德地图加载器
+import AMapLoader from '@amap/amap-jsapi-loader';
+import { TagVO } from '@/api/system/physical/tag/types';
+const { proxy } = getCurrentInstance() as ComponentInternalInstance;
+
+const storeList = ref<StoreVO[]>([]);
+const buttonLoading = ref(false);
+const loading = ref(true);
+const showSearch = ref(true);
+const ids = ref<Array<string | number>>([]);
+const single = ref(true);
+const multiple = ref(true);
+const total = ref(0);
+
+const queryFormRef = ref<ElFormInstance>();
+const storeFormRef = ref<ElFormInstance>();
+
+const dialog = reactive<DialogOption>({
+  visible: false,
+  title: ''
+});
+
+const initFormData: StoreForm = {
+  id: undefined,
+  storeCode: undefined,
+  name: undefined,
+  storeTypeId: undefined,
+  zoneId: undefined,
+  address: undefined,
+  iconUrl: undefined,
+  backgroundUrl: undefined,
+  businessStartTime: undefined,
+  businessEndTime: undefined,
+  status: 1,
+  phone: '',
+  customTags: [], // 初始化为空数组
+  createdAt: undefined,
+  updatedAt: undefined,
+  deletedAt: undefined
+};
+const data = reactive<PageData<StoreForm, StoreQuery>>({
+  form: { ...initFormData },
+  queryParams: {
+    pageNum: 1,
+    pageSize: 10,
+    storeCode: undefined,
+    name: undefined,
+    storeTypeId: undefined,
+    zoneId: undefined,
+    address: undefined,
+    iconUrl: undefined,
+    backgroundUrl: undefined,
+    businessStartTime: undefined,
+    businessEndTime: undefined,
+    status: undefined,
+    createdAt: undefined,
+    updatedAt: undefined,
+    deletedAt: undefined,
+    params: {}
+  },
+  rules: {
+    id: [{ required: true, message: '主键ID不能为空', trigger: 'blur' }],
+    storeCode: [{ required: true, message: '门店编号,如M0001不能为空', trigger: 'blur' }],
+    name: [{ required: true, message: '门店名称,最多20字不能为空', trigger: 'blur' }],
+    storeTypeId: [{ required: true, message: '门店类型ID,关联service_type表不能为空', trigger: 'blur' }],
+    zoneId: [{ required: true, message: '所属赛区分组ID,关联zone表不能为空', trigger: 'blur' }],
+    address: [{ required: true, message: '详细地址不能为空', trigger: 'blur' }],
+    businessStartTime: [{ required: true, message: '营业开始时间不能为空', trigger: 'blur' }],
+    businessEndTime: [{ required: true, message: '营业结束时间不能为空', trigger: 'blur' }],
+    status: [{ required: true, message: '状态:1=开业中,2=闭店中不能为空', trigger: 'change' }]
+  }
+});
+
+const { queryParams, form, rules } = toRefs(data);
+
+/** 查询门店信息列表 */
+const getList = async () => {
+  loading.value = true;
+  const res = await listStore(queryParams.value);
+  storeList.value = res.rows;
+  total.value = res.total;
+  loading.value = false;
+};
+
+/** 取消按钮 */
+const cancel = () => {
+  reset();
+  dialog.visible = false;
+};
+
+/** 表单重置 */
+const reset = () => {
+  form.value = { ...initFormData };
+  storeFormRef.value?.resetFields();
+
+  // 显式重置 customTags
+  customTags.value = [];
+
+  // 清空上传相关状态
+  fileList.value = [];
+  fileList2.value = [];
+  iconPreviewUrl.value = '';
+  iconPreviewUrl2.value = '';
+  form.value.iconUrl = ''; // 清除表单中的URL
+  form.value.backgroundUrl = ''; // 清除表单中的URL
+};
+/** 搜索按钮操作 */
+const handleQuery = () => {
+  queryParams.value.pageNum = 1;
+  getList();
+};
+
+/** 重置按钮操作 */
+const resetQuery = () => {
+  queryFormRef.value?.resetFields();
+  handleQuery();
+};
+
+/** 多选框选中数据 */
+const handleSelectionChange = (selection: StoreVO[]) => {
+  ids.value = selection.map((item) => item.id);
+  single.value = selection.length != 1;
+  multiple.value = !selection.length;
+};
+
+/** 新增按钮操作 */
+const handleAdd = () => {
+  reset();
+  dialog.visible = true;
+  dialog.title = '添加门店信息';
+  // 清空上传相关状态
+  fileList.value = [];
+  fileList2.value = [];
+  iconPreviewUrl.value = '';
+  iconPreviewUrl2.value = '';
+  form.value.iconUrl = ''; // 清除表单中的URL
+  form.value.backgroundUrl = ''; // 清除表单中的URL
+  // 确保 customTags 被清空
+  customTags.value = [];
+};
+
+/** 修改按钮操作 */
+const handleUpdate = async (row?: StoreVO) => {
+  reset();
+  const _id = row?.id || ids.value[0];
+  const res = await getStore(_id);
+  Object.assign(form.value, res.data);
+
+  // 保留已有的图片URL,但清除上传组件的状态
+  if (form.value.iconUrl) {
+    iconPreviewUrl.value = form.value.iconUrl; // 设置预览图
+  }
+  if (form.value.backgroundUrl) {
+    iconPreviewUrl2.value = form.value.backgroundUrl; // 设置预览图
+  }
+  // 从后端获取的标签数据
+  if (res.data.customTags && Array.isArray(res.data.customTags)) {
+    customTags.value = [...res.data.customTags];
+  } else {
+    customTags.value = [];
+  }
+  dialog.visible = true;
+  dialog.title = '修改门店信息';
+};
+/** 提交按钮 */
+const submitForm = () => {
+  storeFormRef.value?.validate(async (valid: boolean) => {
+    if (valid) {
+      buttonLoading.value = true;
+      if (form.value.id) {
+        await updateStore(form.value).finally(() => (buttonLoading.value = false));
+      } else {
+        await addStore(form.value).finally(() => (buttonLoading.value = false));
+      }
+      proxy?.$modal.msgSuccess('操作成功');
+      dialog.visible = false;
+      await getList();
+    }
+  });
+};
+
+/** 删除按钮操作 */
+const handleDelete = async (row?: StoreVO) => {
+  const _ids = row?.id || ids.value;
+  await proxy?.$modal.confirm('是否确认删除门店信息编号为"' + _ids + '"的数据项?').finally(() => (loading.value = false));
+  await delStore(_ids);
+  proxy?.$modal.msgSuccess('删除成功');
+  await getList();
+};
+
+/** 导出按钮操作 */
+const handleExport = () => {
+  proxy?.download(
+    'system/store/export',
+    {
+      ...queryParams.value
+    },
+    `store_${new Date().getTime()}.xlsx`
+  );
+};
+
+onMounted(() => {
+  getList();
+  loadTagOptions(); // 加载门店类型选项
+});
+// 响应式变量
+const iconPreviewUrl = ref('');
+const iconPreviewUrl2 = ref('');
+const fileList = ref([]);
+const fileList2 = ref([]);
+const dialogVisible = ref(false);
+const dialogVisible2 = ref(false);
+const previewSrc = ref('');
+const previewSrc2 = ref('');
+
+/** 处理图标上传 */
+const handleIconChange = async (file) => {
+  const index = fileList.value.findIndex((f) => f.uid === file.uid);
+  fileList.value = [];
+
+  if (file.raw) {
+    iconPreviewUrl.value = URL.createObjectURL(file.raw);
+  }
+
+  if (index === -1) {
+    // 如果文件不在列表中,则添加进去
+    fileList.value.push(file);
+  }
+
+  try {
+    const rawFile = file.raw;
+    const res = await uploadTournament(rawFile);
+    if (res.code === 200) {
+      // 更新文件状态为成功
+      fileList.value[index] = {
+        ...file,
+        status: 'success',
+        response: res.data.url
+      };
+      form.value.iconUrl = fileList.value[index].response;
+      iconPreviewUrl.value = fileList.value[index].response;
+      ElMessage.success('上传成功');
+    } else {
+      throw new Error(res.msg);
+    }
+  } catch (error) {
+    // 更新文件状态为失败
+    fileList.value[index] = {
+      ...file,
+      status: 'fail',
+      error: '上传失败'
+    };
+    ElMessage.error('上传失败,请重试');
+  }
+};
+
+/** 处理图标删除 */
+const handleIconRemove = (file, updatedFileList) => {
+  fileList.value = updatedFileList;
+  // 清除预览图和临时链接
+  iconPreviewUrl.value = '';
+  form.value.iconUrl = ''; // 清除表单中的URL
+};
+
+/** 点击预览图触发放大 */
+const handlePreviewClick = () => {
+  const currentSrc = iconPreviewUrl.value || form.value.iconUrl;
+  if (currentSrc) {
+    dialogVisible.value = true;
+  }
+};
+
+/** 处理背景图上传 */
+const handleIconChange2 = async (file) => {
+  const index = fileList2.value.findIndex((f) => f.uid === file.uid);
+  fileList2.value = [];
+
+  if (file.raw) {
+    iconPreviewUrl2.value = URL.createObjectURL(file.raw);
+  }
+
+  if (index === -1) {
+    // 如果文件不在列表中,则添加进去
+    fileList2.value.push(file);
+  }
+
+  try {
+    const rawFile = file.raw;
+    const res = await uploadTournament(rawFile);
+    if (res.code === 200) {
+      // 更新文件状态为成功
+      fileList2.value[index] = {
+        ...file,
+        status: 'success',
+        response: res.data.url
+      };
+      form.value.backgroundUrl = fileList2.value[index].response;
+      iconPreviewUrl2.value = fileList2.value[index].response;
+      ElMessage.success('上传成功');
+    } else {
+      throw new Error(res.msg);
+    }
+  } catch (error) {
+    // 更新文件状态为失败
+    fileList2.value[index] = {
+      ...file,
+      status: 'fail',
+      error: '上传失败'
+    };
+    ElMessage.error('上传失败,请重试');
+  }
+};
+
+/** 处理背景图删除 */
+const handleIconRemove2 = (file, updatedFileList) => {
+  fileList2.value = updatedFileList;
+  // 清除预览图和临时链接
+  iconPreviewUrl2.value = '';
+  form.value.backgroundUrl = ''; // 清除表单中的URL
+};
+
+/** 点击预览图触发放大 */
+const handlePreviewClick2 = () => {
+  const currentSrc = iconPreviewUrl2.value || form.value.backgroundUrl;
+  if (currentSrc) {
+    dialogVisible2.value = true;
+  }
+};
+// 响应式变量
+const tagOptions = ref<TagVO[]>([]);
+// 加载门店类型选项
+const loadTagOptions = async () => {
+  try {
+    const res = await selectAllPhysicalTagsSelList();
+    if (res.code === 200) {
+      tagOptions.value = res.data;
+    } else {
+      ElMessage.error('加载门店类型失败:' + res.msg);
+    }
+  } catch (error) {
+    console.error('请求出错:', error);
+    ElMessage.error('请求失败,请检查网络');
+  }
+};
+// 自定义标签相关状态
+const customTags = ref<string[]>([]);
+const addTagDialogVisible = ref(false);
+const newTagName = ref('');
+// 添加标签
+const confirmAddTag = () => {
+  if (!newTagName.value.trim()) {
+    ElMessage.warning('标签名称不能为空');
+    return;
+  }
+  // 检查是否已达到最大数量限制
+  if (customTags.value.length >= 3) {
+    ElMessage.warning('最多只能添加3个标签');
+    return;
+  }
+  // 检查是否已存在
+  if (customTags.value.includes(newTagName.value)) {
+    ElMessage.warning('该标签已存在');
+    return;
+  }
+
+  customTags.value.push(newTagName.value);
+  newTagName.value = '';
+  addTagDialogVisible.value = false;
+};
+
+// 删除标签
+const handleTagClose = (index: number) => {
+  customTags.value.splice(index, 1);
+};
+
+// 表单提交时,将标签数组赋值给 form.customTags
+watch(
+  customTags,
+  () => {
+    // 确保赋值的是一个数组
+    if (Array.isArray(customTags.value)) {
+      form.value.customTags = customTags.value;
+    }
+  },
+  { deep: true }
+);
+const openAddTagDialog = () => {
+  newTagName.value = '';
+  addTagDialogVisible.value = true;
+};
+// 添加响应式变量
+const mapDialogVisible = ref(false);
+const selectedAddress = ref('');
+const selectedLocation = ref({ lng: 0, lat: 0 });
+const searchKeyword = ref('');
+let amapInstance: any = null;
+let marker: any = null;
+
+/** 打开地图选择器 */
+const openMapSelector = () => {
+  mapDialogVisible.value = true;
+  nextTick(() => {
+    initAMap(); // 等待 el-dialog 渲染完成再初始化
+  });
+};
+
+/** 初始化高德地图 */
+const initAMap = () => {
+  AMapLoader.load({
+    key: 'ed2e29e4cbcc0f96905acb2106e82b42', // 请替换为您的高德地图key
+    version: '2.0',
+    plugins: ['AMap.Geocoder', 'AMap.PlaceSearch', 'AMap.AutoComplete', 'AMap.Marker', 'AMap.ToolBar', 'AMap.Scale', 'AMap.Geolocation']
+  })
+    .then((AMap) => {
+      amapInstance = AMap;
+
+      // 确保容器存在后再创建地图
+      const container = document.getElementById('amap-container');
+      if (!container) return;
+
+      const map = new AMap.Map(container, {
+        resizeEnable: true,
+        zoom: 12,
+        center: [116.397428, 39.90923] // 默认中心点
+      });
+
+      // 添加定位功能
+      map.addControl(
+        new AMap.Geolocation({
+          zoomToCurrent: true,
+          position: 'rtb'
+        })
+      );
+
+      // 添加点击事件
+      map.on('click', function (e) {
+        const lnglat = e.lnglat;
+        setSelectedLocation(lnglat, AMap);
+      });
+
+      // 添加地图控件
+      map.addControl(new AMap.ToolBar());
+      map.addControl(new AMap.Scale());
+
+      // 初始化标记
+      marker = new AMap.Marker({
+        map: map,
+        draggable: true,
+        cursor: 'move'
+      });
+
+      // 标记拖拽结束事件
+      marker.on('dragend', function (e) {
+        setSelectedLocation(e.lnglat, AMap);
+      });
+
+      // 将map实例保存到DOM元素上
+      (container as any).__map__ = map;
+    })
+    .catch((e) => {
+      console.error(e);
+      proxy?.$modal.msgError('地图加载失败: ' + e.message);
+    });
+};
+
+/** 设置选中位置 */
+const setSelectedLocation = (lnglat: any, AMap: any) => {
+  selectedLocation.value = {
+    lng: lnglat.getLng(),
+    lat: lnglat.getLat()
+  };
+
+  marker.setPosition(lnglat);
+
+  // 逆地理编码获取地址
+  const geocoder = new AMap.Geocoder();
+  geocoder.getAddress(lnglat, function (status: string, result: any) {
+    debugger;
+    if (status === 'complete' && result.regeocode) {
+      selectedAddress.value = result.regeocode.formattedAddress;
+    }
+  });
+};
+
+/** 搜索地址 */
+const searchAddress = () => {
+  if (!searchKeyword.value || !amapInstance) return;
+
+  amapInstance.plugin('AMap.PlaceSearch', () => {
+    const placeSearch = new amapInstance.PlaceSearch({
+      pageSize: 5,
+      pageIndex: 1,
+      city: '全国', // 搜索范围
+      map: document.getElementById('amap-container')?.__map__ // 使用正确的map引用
+    });
+
+    placeSearch.search(searchKeyword.value, (status: string, result: any) => {
+      if (status === 'complete' && result.info === 'OK') {
+        if (result.poiList && result.poiList.pois.length > 0) {
+          const firstPoi = result.poiList.pois[0];
+          const lnglat = [firstPoi.location.lng, firstPoi.location.lat];
+
+          // 更新地图中心点
+          const mapElement = document.getElementById('amap-container');
+          if (mapElement) {
+            const map = mapElement.__map__;
+            if (map) {
+              map.setCenter(lnglat);
+              setSelectedLocation(new amapInstance.LngLat(firstPoi.location.lng, firstPoi.location.lat), amapInstance);
+            }
+          }
+        }
+      } else {
+        proxy?.$modal.msgWarning('未找到相关地址');
+      }
+    });
+  });
+};
+
+/** 确认选择地图位置 */
+const confirmMapSelection = () => {
+  if (selectedAddress.value) {
+    form.value.address = selectedAddress.value;
+    // 如果需要保存经纬度,可以在form中添加相应字段
+    // form.value.longitude = selectedLocation.value.lng;
+    // form.value.latitude = selectedLocation.value.lat;
+  }
+  mapDialogVisible.value = false;
+};
+
+/** 取消地图选择 */
+const cancelMapSelection = () => {
+  mapDialogVisible.value = false;
+};
+
+/** 清除搜索关键词 */
+const clearSearch = () => {
+  searchKeyword.value = '';
+};
+</script>