Browse Source

商家提交

yjwang 1 week ago
parent
commit
36daa60a44
5 changed files with 490 additions and 95 deletions
  1. 10 0
      src/api/system/user.js
  2. 287 79
      src/views/hisStore/adv/index.vue
  3. 143 16
      src/views/hisStore/store/index.vue
  4. 49 0
      src/views/index.vue
  5. 1 0
      src/views/login.vue

+ 10 - 0
src/api/system/user.js

@@ -150,3 +150,13 @@ export function isAdmin() {
   })
   })
 }
 }
 
 
+
+/**
+ * 数据提示
+ */
+export function noticeInfo() {
+  return request({
+    url: '/system/user/noticeInfo',
+    method: 'get'
+  })
+}

+ 287 - 79
src/views/hisStore/adv/index.vue

@@ -4,21 +4,21 @@
       <el-form-item label="广告位置" prop="advType">
       <el-form-item label="广告位置" prop="advType">
         <el-select v-model="queryParams.advType" placeholder="请选择广告位置" clearable size="small">
         <el-select v-model="queryParams.advType" placeholder="请选择广告位置" clearable size="small">
           <el-option
           <el-option
-                v-for="item in advTypeOptions"
-                :key="item.dictValue"
-                :label="item.dictLabel"
-                :value="item.dictValue"
-              />
+            v-for="item in advTypeOptions"
+            :key="item.dictValue"
+            :label="item.dictLabel"
+            :value="item.dictValue"
+          />
         </el-select>
         </el-select>
       </el-form-item>
       </el-form-item>
       <el-form-item label="显示类型" prop="showType">
       <el-form-item label="显示类型" prop="showType">
         <el-select v-model="queryParams.showType" placeholder="请选择显示类型" clearable size="small">
         <el-select v-model="queryParams.showType" placeholder="请选择显示类型" clearable size="small">
-           <el-option
-                v-for="item in showTypeOptions"
-                :key="item.dictValue"
-                :label="item.dictLabel"
-                :value="item.dictValue"
-              />
+          <el-option
+            v-for="item in showTypeOptions"
+            :key="item.dictValue"
+            :label="item.dictLabel"
+            :value="item.dictValue"
+          />
         </el-select>
         </el-select>
       </el-form-item>
       </el-form-item>
       <el-form-item>
       <el-form-item>
@@ -83,17 +83,17 @@
       <el-table-column label="排序" align="center" prop="sort" />
       <el-table-column label="排序" align="center" prop="sort" />
       <el-table-column  label="广告位置" align="center" prop="advType">
       <el-table-column  label="广告位置" align="center" prop="advType">
         <template slot-scope="scope">
         <template slot-scope="scope">
-            <el-tag prop="advType" v-for="(item, index) in advTypeOptions"    v-if="scope.row.advType==item.dictValue">{{item.dictLabel}}</el-tag>
-          </template>
+          <el-tag prop="advType" v-for="(item, index) in advTypeOptions"    v-if="scope.row.advType==item.dictValue">{{item.dictLabel}}</el-tag>
+        </template>
       </el-table-column>
       </el-table-column>
       <el-table-column  label="显示方式" align="center" prop="showType">
       <el-table-column  label="显示方式" align="center" prop="showType">
         <template slot-scope="scope">
         <template slot-scope="scope">
-             <el-tag prop="showType" v-for="(item, index) in showTypeOptions"    v-if="scope.row.showType==item.dictValue">{{item.dictLabel}}</el-tag>
-          </template>
+          <el-tag prop="showType" v-for="(item, index) in showTypeOptions"    v-if="scope.row.showType==item.dictValue">{{item.dictLabel}}</el-tag>
+        </template>
       </el-table-column>
       </el-table-column>
       <el-table-column  label="状态" align="center" prop="status">
       <el-table-column  label="状态" align="center" prop="status">
         <template slot-scope="scope">
         <template slot-scope="scope">
-            <el-tag prop="status" v-for="(item, index) in statusOptions"    v-if="scope.row.status==item.dictValue">{{item.dictLabel}}</el-tag>
+          <el-tag prop="status" v-for="(item, index) in statusOptions"    v-if="scope.row.status==item.dictValue">{{item.dictLabel}}</el-tag>
         </template>
         </template>
       </el-table-column>
       </el-table-column>
       <el-table-column label="操作" align="center" class-name="small-padding fixed-width">
       <el-table-column label="操作" align="center" class-name="small-padding fixed-width">
@@ -136,17 +136,25 @@
             <el-radio :label="2">视频</el-radio>
             <el-radio :label="2">视频</el-radio>
           </el-radio-group>
           </el-radio-group>
         </el-form-item>
         </el-form-item>
-         <el-form-item label="广告图片" prop="imageUrl" v-if="form.urlType==1">
-            <el-upload
-              v-model="form.icon"
-              class="avatar-uploader"
-              :action="uploadUrl"
-              :show-file-list="false"
-              :on-success="handleAvatarSuccess"
-              :before-upload="beforeAvatarUpload">
-              <img v-if="form.imageUrl" :src="form.imageUrl" class="avatar">
-              <i v-else class="el-icon-plus avatar-uploader-icon"></i>
-            </el-upload>
+        <el-form-item label="广告图片" prop="imageUrl" v-if="form.urlType==1">
+          <el-upload
+            v-model="form.icon"
+            class="avatar-uploader"
+            :action="uploadUrl"
+            :show-file-list="false"
+            :on-success="handleAvatarSuccess"
+            :before-upload="beforeAvatarUpload">
+            <img v-if="form.imageUrl" :src="form.imageUrl" class="avatar">
+            <i v-else class="el-icon-plus avatar-uploader-icon"></i>
+          </el-upload>
+          <div class="image-size-tip">
+            <i class="el-icon-info"></i>
+            <span>建议尺寸:280×90px(比例28:9)</span>
+          </div>
+          <div v-if="imageValidation.show" :class="['validation-message', imageValidation.type]">
+            <i :class="imageValidation.type === 'warning' ? 'el-icon-warning' : 'el-icon-success'"></i>
+            <span>{{ imageValidation.message }}</span>
+          </div>
         </el-form-item>
         </el-form-item>
         <el-form-item label="广告视频" prop="video" v-if="form.urlType==2">
         <el-form-item label="广告视频" prop="video" v-if="form.urlType==2">
           <div>
           <div>
@@ -190,7 +198,7 @@
           </el-radio-group>
           </el-radio-group>
         </el-form-item>
         </el-form-item>
         <el-form-item label="排序" prop="sort">
         <el-form-item label="排序" prop="sort">
-           <el-input-number v-model="form.sort"  :min="1" :max="99" ></el-input-number>
+          <el-input-number v-model="form.sort"  :min="1" :max="99" ></el-input-number>
         </el-form-item>
         </el-form-item>
       </el-form>
       </el-form>
       <div slot="footer" class="dialog-footer">
       <div slot="footer" class="dialog-footer">
@@ -211,6 +219,18 @@ export default {
   },
   },
   data() {
   data() {
     return {
     return {
+      // 广告图片验证相关
+      imageValidation: {
+        show: false,
+        type: 'success', // success 或 warning
+        message: ''
+      },
+      // 广告图片尺寸配置
+      imageSizeConfig: {
+        width: 280,
+        height: 90,
+        tolerance: 0.15 // 15%容差
+      },
       uploadUrl:process.env.VUE_APP_BASE_API+"/common/uploadOSS",
       uploadUrl:process.env.VUE_APP_BASE_API+"/common/uploadOSS",
       baseUrl: process.env.VUE_APP_BASE_API,
       baseUrl: process.env.VUE_APP_BASE_API,
       videoAccept:"video/*",
       videoAccept:"video/*",
@@ -313,21 +333,142 @@ export default {
       this.form.content=text
       this.form.content=text
     },
     },
     handleAvatarSuccess(res, file) {
     handleAvatarSuccess(res, file) {
-        if(res.code==200){
-          this.form.imageUrl=res.url;
-        }
-        else{
-          this.msgError(res.msg);
-        }
+      if(res.code==200){
+        this.form.imageUrl=res.url;
+        // 验证广告图片尺寸
+        this.validateAdvertImage(res.url);
+      }
+      else{
+        this.msgError(res.msg);
+      }
 
 
     },
     },
     beforeAvatarUpload(file) {
     beforeAvatarUpload(file) {
       const isLt1M = file.size / 1024 / 1024 < 200;
       const isLt1M = file.size / 1024 / 1024 < 200;
       if (!isLt1M) {
       if (!isLt1M) {
         this.$message.error('上传图片大小不能超过 200MB!');
         this.$message.error('上传图片大小不能超过 200MB!');
+        return false;
+      }
+      const isPic = file.type === 'image/jpeg' || file.type === 'image/png' || file.type === 'image/gif' || file.type === 'image/jpg';
+      if (!isPic) {
+        this.$message.error('只能上传JPG、PNG、GIF格式的图片!');
+        return false;
+      }
+      return true;
+    },
+
+    // 验证广告图片尺寸
+    validateAdvertImage(imageUrl) {
+      if (!imageUrl) {
+        this.imageValidation.show = false;
+        return;
+      }
+
+      console.log('开始验证广告图片尺寸:', imageUrl);
+
+      const img = new Image();
+      const config = this.imageSizeConfig;
+
+      // 设置超时处理
+      const timeoutId = setTimeout(() => {
+        this.imageValidation = {
+          show: true,
+          type: 'warning',
+          message: '图片加载超时,无法验证尺寸,请检查网络连接'
+        };
+        console.log('图片加载超时');
+      }, 10000);
+
+      img.onload = () => {
+        clearTimeout(timeoutId);
+        const { width: actualWidth, height: actualHeight } = img;
+        const { width: expectedWidth, height: expectedHeight, tolerance } = config;
+
+        console.log(`实际尺寸: ${actualWidth}x${actualHeight}px, 期望尺寸: ${expectedWidth}x${expectedHeight}px`);
+
+        // 计算比例差异
+        const expectedRatio = expectedWidth / expectedHeight;
+        const actualRatio = actualWidth / actualHeight;
+        const ratioDiff = Math.abs(actualRatio - expectedRatio) / expectedRatio;
+
+        console.log(`期望比例: ${expectedRatio.toFixed(3)}, 实际比例: ${actualRatio.toFixed(3)}, 差异: ${(ratioDiff * 100).toFixed(2)}%`);
+
+        this.imageValidation.show = true;
+
+        if (ratioDiff <= tolerance) {
+          this.imageValidation.type = 'success';
+          this.imageValidation.message = `✓ 尺寸符合要求 (实际: ${actualWidth}x${actualHeight}px)`;
+          console.log('尺寸验证通过');
+        } else {
+          this.imageValidation.type = 'warning';
+          this.imageValidation.message = `⚠ 尺寸不符合要求!实际: ${actualWidth}x${actualHeight}px,建议: ${expectedWidth}x${expectedHeight}px (比例${expectedRatio.toFixed(1)}:1)`;
+          console.log('尺寸验证失败');
+        }
+      };
+
+      img.onerror = (error) => {
+        clearTimeout(timeoutId);
+        console.error('图片加载失败:', error);
+
+        // 检查URL格式和可访问性
+        this.checkImageUrl(imageUrl).then(isValid => {
+          if (!isValid) {
+            this.imageValidation = {
+              show: true,
+              type: 'warning',
+              message: '图片URL无效或无法访问,请检查图片地址是否正确'
+            };
+          } else {
+            this.imageValidation = {
+              show: true,
+              type: 'warning',
+              message: '图片加载失败,可能是跨域问题或格式不支持,请重新上传'
+            };
+          }
+        });
+      };
+
+      // 处理完整的图片URL
+      const fullImageUrl = this.getFullImageUrl(imageUrl);
+      img.crossOrigin = 'anonymous'; // 尝试处理跨域
+      img.src = fullImageUrl;
+    },
+
+    // 获取完整的图片URL
+    getFullImageUrl(imageUrl) {
+      if (!imageUrl) return '';
+
+      // 如果已经是完整URL(http/https开头)
+      if (imageUrl.startsWith('http://') || imageUrl.startsWith('https://')) {
+        return imageUrl;
+      }
+
+      // 如果是base64
+      if (imageUrl.startsWith('data:image/')) {
+        return imageUrl;
+      }
+
+      // 如果是相对路径,拼接基础URL
+      const baseUrl = process.env.VUE_APP_BASE_API || '';
+      if (imageUrl.startsWith('/')) {
+        return baseUrl + imageUrl;
+      } else {
+        return baseUrl + '/' + imageUrl;
+      }
+    },
+
+    // 检查图片URL有效性
+    async checkImageUrl(imageUrl) {
+      try {
+        const fullUrl = this.getFullImageUrl(imageUrl);
+        const response = await fetch(fullUrl, { method: 'HEAD' });
+        return response.ok;
+      } catch (error) {
+        console.error('URL检查失败:', error);
+        return false;
       }
       }
-      return   isLt1M;
     },
     },
+
     /** 查询广告列表 */
     /** 查询广告列表 */
     getList() {
     getList() {
       this.loading = true;
       this.loading = true;
@@ -358,6 +499,8 @@ export default {
         advType: null,
         advType: null,
         showType: null
         showType: null
       };
       };
+      // 重置广告图片验证状态
+      this.imageValidation = { show: false, type: 'success', message: '' };
       this.resetForm("form");
       this.resetForm("form");
     },
     },
     /** 搜索按钮操作 */
     /** 搜索按钮操作 */
@@ -392,16 +535,24 @@ export default {
         this.form.status = response.data.status.toString();
         this.form.status = response.data.status.toString();
         this.form.advType = response.data.advType.toString();
         this.form.advType = response.data.advType.toString();
         this.form.showType = response.data.showType ? response.data.showType.toString() : "";
         this.form.showType = response.data.showType ? response.data.showType.toString() : "";
+
+        // 如果有广告图片且是图片类型,验证其尺寸
+        if (this.form.imageUrl && this.form.urlType === 1) {
+          setTimeout(() => {
+            this.validateAdvertImage(this.form.imageUrl);
+          }, 1000);
+        }
+
         this.open = true;
         this.open = true;
         this.title = "修改广告";
         this.title = "修改广告";
         setTimeout(() => {
         setTimeout(() => {
           if(this.form.showType=="3"){
           if(this.form.showType=="3"){
             console.log(that.form.content)
             console.log(that.form.content)
             if(this.form.content==null){
             if(this.form.content==null){
-                that.$refs.myeditor.setText("");
+              that.$refs.myeditor.setText("");
             }
             }
             else{
             else{
-                that.$refs.myeditor.setText(that.form.content);
+              that.$refs.myeditor.setText(that.form.content);
             }
             }
           }
           }
         }, 200);
         }, 200);
@@ -431,62 +582,119 @@ export default {
     handleDelete(row) {
     handleDelete(row) {
       const advIds = row.advId || this.ids;
       const advIds = row.advId || this.ids;
       this.$confirm('是否确认删除广告编号为"' + advIds + '"的数据项?', "警告", {
       this.$confirm('是否确认删除广告编号为"' + advIds + '"的数据项?', "警告", {
-          confirmButtonText: "确定",
-          cancelButtonText: "取消",
-          type: "warning"
-        }).then(function() {
-          return delAdv(advIds);
-        }).then(() => {
-          this.getList();
-          this.msgSuccess("删除成功");
-        }).catch(() => {});
+        confirmButtonText: "确定",
+        cancelButtonText: "取消",
+        type: "warning"
+      }).then(function() {
+        return delAdv(advIds);
+      }).then(() => {
+        this.getList();
+        this.msgSuccess("删除成功");
+      }).catch(() => {});
     },
     },
     /** 导出按钮操作 */
     /** 导出按钮操作 */
     handleExport() {
     handleExport() {
       const queryParams = this.queryParams;
       const queryParams = this.queryParams;
       this.$confirm('是否确认导出所有广告数据项?', "警告", {
       this.$confirm('是否确认导出所有广告数据项?', "警告", {
-          confirmButtonText: "确定",
-          cancelButtonText: "取消",
-          type: "warning"
-        }).then(() => {
-          this.exportLoading = true;
-          return exportAdv(queryParams);
-        }).then(response => {
-          this.download(response.msg);
-          this.exportLoading = false;
-        }).catch(() => {});
+        confirmButtonText: "确定",
+        cancelButtonText: "取消",
+        type: "warning"
+      }).then(() => {
+        this.exportLoading = true;
+        return exportAdv(queryParams);
+      }).then(response => {
+        this.download(response.msg);
+        this.exportLoading = false;
+      }).catch(() => {});
     }
     }
   }
   }
 };
 };
 </script>
 </script>
-<style >
- .avatar-uploader .el-upload {
-    border: 1px dashed #d9d9d9;
-    border-radius: 6px;
-    cursor: pointer;
-    position: relative;
-    overflow: hidden;
-  }
-  .avatar-uploader .el-upload:hover {
-    border-color: #409EFF;
-  }
-</style>
 <style scoped>
 <style scoped>
+.image-size-tip {
+  margin-top: 5px;
+  font-size: 12px;
+  color: #909399;
+  display: flex;
+  align-items: center;
+}
+
+.image-size-tip i {
+  margin-right: 5px;
+  color: #409EFF;
+}
 
 
-  .avatar-uploader-icon {
-    font-size: 28px;
-    color: #8c939d;
-    width: 300px;
-    height: 150px;
-    line-height: 150px;
-    text-align: center;
+.validation-message {
+  margin-top: 8px;
+  padding: 6px 10px;
+  border-radius: 4px;
+  font-size: 12px;
+  display: flex;
+  align-items: center;
+  animation: fadeIn 0.3s ease-in;
+}
+
+.validation-message i {
+  margin-right: 6px;
+  font-size: 14px;
+}
+
+.validation-message.success {
+  background-color: #f0f9ff;
+  border: 1px solid #b3d8ff;
+  color: #67C23A;
+}
+
+.validation-message.success i {
+  color: #67C23A;
+}
+
+.validation-message.warning {
+  background-color: #fdf6ec;
+  border: 1px solid #f5dab1;
+  color: #E6A23C;
+}
+
+.validation-message.warning i {
+  color: #E6A23C;
+}
+
+@keyframes fadeIn {
+  from {
+    opacity: 0;
+    transform: translateY(-5px);
   }
   }
-  .avatar {
-    width: 300px;
-    height: 150px;
-    display: block;
+  to {
+    opacity: 1;
+    transform: translateY(0);
   }
   }
+}
 
 
+.avatar-uploader-icon {
+  font-size: 28px;
+  color: #8c939d;
+  width: 300px;
+  height: 150px;
+  line-height: 150px;
+  text-align: center;
+}
 
 
+.avatar {
+  width: 300px;
+  height: 150px;
+  display: block;
+}
+</style>
+<style >
+.avatar-uploader .el-upload {
+  border: 1px dashed #d9d9d9;
+  border-radius: 6px;
+  cursor: pointer;
+  position: relative;
+  overflow: hidden;
+}
+.avatar-uploader .el-upload:hover {
+  border-color: #409EFF;
+}
 </style>
 </style>
 
 

+ 143 - 16
src/views/hisStore/store/index.vue

@@ -477,7 +477,90 @@
             </el-col>
             </el-col>
           </el-row>
           </el-row>
         </div>
         </div>
-
+        <div v-hasPermi="['his:store:AgreementSigned']">
+          <el-divider content-position="left">签署协议与特殊资质信息</el-divider>
+          <el-row>
+            <el-col :span="12">
+              <el-form-item label="其它资质-入驻协议">
+                <el-upload
+                  class="avatar-uploader"
+                  :action="uploadUrl"
+                  :show-file-list="false"
+                  :on-success="(response, file) => handleFileSuccess(response, file, 'settlementAgreement')"
+                  :before-upload="beforeAvatarUpload">
+                  <img v-if="form.settlementAgreement" :src="form.settlementAgreement" class="avatar" width="100px" />
+                  <i v-else class="el-icon-plus avatar-uploader-icon"></i>
+                </el-upload>
+              </el-form-item>
+            </el-col>
+            <el-col :span="12">
+              <el-form-item label="其它资质-入驻协议有效期">
+                <el-date-picker
+                  v-model="form.settlementAgreementExpiry"
+                  type="daterange"
+                  value-format="yyyy-MM-dd"
+                  range-separator="至"
+                  start-placeholder="开始日期"
+                  end-placeholder="结束日期">
+                </el-date-picker>
+              </el-form-item>
+            </el-col>
+          </el-row>
+          <el-row>
+            <el-col :span="12">
+              <el-form-item label="质量保证协议">
+                <el-upload
+                  class="avatar-uploader"
+                  :action="uploadUrl"
+                  :show-file-list="false"
+                  :on-success="(response, file) => handleFileSuccess(response, file, 'qualityAssuranceAgreement')"
+                  :before-upload="beforeAvatarUpload">
+                  <img v-if="form.qualityAssuranceAgreement" :src="form.qualityAssuranceAgreement" class="avatar" width="100px">
+                  <i v-else class="el-icon-plus avatar-uploader-icon"></i>
+                </el-upload>
+              </el-form-item>
+            </el-col>
+            <el-col :span="12">
+              <el-form-item label="质量保证协议有效期">
+                <el-date-picker
+                  v-model="form.qualityAssuranceAgreementExpiry"
+                  type="daterange"
+                  value-format="yyyy-MM-dd"
+                  range-separator="至"
+                  start-placeholder="开始日期"
+                  end-placeholder="结束日期">
+                </el-date-picker>
+              </el-form-item>
+            </el-col>
+          </el-row>
+          <el-row>
+            <el-col :span="12">
+              <el-form-item label="其它特殊资质">
+                <el-upload
+                  class="avatar-uploader"
+                  :action="uploadUrl"
+                  :show-file-list="false"
+                  :on-success="(response, file) => handleFileSuccess(response, file, 'otherSpecialQualification')"
+                  :before-upload="beforeAvatarUpload">
+                  <img v-if="form.otherSpecialQualification" :src="form.otherSpecialQualification" class="avatar" width="100px">
+                  <i v-else class="el-icon-plus avatar-uploader-icon"></i>
+                </el-upload>
+              </el-form-item>
+            </el-col>
+            <el-col :span="12">
+              <el-form-item label="其它特殊资质有效期">
+                <el-date-picker
+                  v-model="form.otherSpecialQualificationExpiry"
+                  type="daterange"
+                  value-format="yyyy-MM-dd"
+                  range-separator="至"
+                  start-placeholder="开始日期"
+                  end-placeholder="结束日期">
+                </el-date-picker>
+              </el-form-item>
+            </el-col>
+          </el-row>
+        </div>
         <el-divider content-position="left">店铺配置信息</el-divider>
         <el-divider content-position="left">店铺配置信息</el-divider>
         <el-form-item label="退货地址" prop="refundAddress">
         <el-form-item label="退货地址" prop="refundAddress">
           <el-input v-model="form.refundAddress" placeholder="请输入退货地址" />
           <el-input v-model="form.refundAddress" placeholder="请输入退货地址" />
@@ -893,7 +976,13 @@ export default {
         foodLicense: null,               // 食品经营许可证文件
         foodLicense: null,               // 食品经营许可证文件
         foodLicenseExpiry: null,             // 食品经营许可证有效期
         foodLicenseExpiry: null,             // 食品经营许可证有效期
         medicalLicense: null,  // 医疗机构执业许可证文件
         medicalLicense: null,  // 医疗机构执业许可证文件
-        medicalLicenseExpiry: null // 医疗机构执业许可证有效期
+        medicalLicenseExpiry: null, // 医疗机构执业许可证有效期
+        settlementAgreement: null,  //其它资质-入驻协议
+        settlementAgreementExpiry: null,//其它资质-有效期
+        qualityAssuranceAgreement: null,//质量保证协议
+        qualityAssuranceAgreementExpiry: null,//质量保证协议有效期
+        otherSpecialQualification: null,//其它特殊资质
+        otherSpecialQualificationExpiry: null,//其它特殊资质有效期
       };
       };
       this.resetForm("form");
       this.resetForm("form");
     },
     },
@@ -928,17 +1017,11 @@ export default {
         this.form = response.data;
         this.form = response.data;
         this.open = true;
         this.open = true;
         this.title = "修改店铺";
         this.title = "修改店铺";
+
         let str = this.form.shippingType
         let str = this.form.shippingType
         this.form.shippingType = str.split(",")
         this.form.shippingType = str.split(",")
         this.form.cityIds = ((this.form.cityIds).split(",")).map(Number)
         this.form.cityIds = ((this.form.cityIds).split(",")).map(Number)
-        if (this.form.brokerageType != null) {
-          this.form.brokerageType = JSON.stringify(this.form.brokerageType)
-        }
-        if (this.form.deliveryType != null) {
-          this.form.deliveryType = JSON.stringify(this.form.deliveryType)
-        }
 
 
-        // 确保正确处理所有日期范围字段
         const dateFields = [
         const dateFields = [
           'drugLicenseExpiry',
           'drugLicenseExpiry',
           'medicalDevice1Expiry',
           'medicalDevice1Expiry',
@@ -948,20 +1031,37 @@ export default {
           'medicalLicenseExpiry',
           'medicalLicenseExpiry',
           'businessLicenseExpire'
           'businessLicenseExpire'
         ];
         ];
-
         dateFields.forEach(field => {
         dateFields.forEach(field => {
           const startField = field + 'Start';
           const startField = field + 'Start';
           const endField = field + 'End';
           const endField = field + 'End';
-
           if (this.form[startField] && this.form[endField]) {
           if (this.form[startField] && this.form[endField]) {
-            // 使用 $set 确保响应式更新
-            this.$set(this.form, field, [
-              this.form[startField],
-              this.form[endField]
-            ]);
+            this.$set(this.form, field, [this.form[startField], this.form[endField]]);
           }
           }
         });
         });
 
 
+
+        // 入驻协议有效期
+        if (this.form.settlementAgreementStart && this.form.settlementAgreementEnd) {
+          this.$set(this.form, 'settlementAgreementExpiry', [
+            this.form.settlementAgreementStart,
+            this.form.settlementAgreementEnd
+          ]);
+        }
+
+        // 质量保证协议有效期
+        if (this.form.qualityAssuranceAgreementStart && this.form.qualityAssuranceAgreementEnd) {
+          this.$set(this.form, 'qualityAssuranceAgreementExpiry', [
+            this.form.qualityAssuranceAgreementStart,
+            this.form.qualityAssuranceAgreementEnd
+          ]);
+        }
+        // 其它特殊资质有效期
+        if (this.form.otherSpecialQualificationStart && this.form.otherSpecialQualificationEnd) {
+          this.$set(this.form, 'otherSpecialQualificationExpiry', [
+            this.form.otherSpecialQualificationStart,
+            this.form.otherSpecialQualificationEnd
+          ]);
+        }
       });
       });
     },
     },
     /** 提交按钮 */
     /** 提交按钮 */
@@ -1016,6 +1116,33 @@ export default {
             formData.businessLicenseExpireEnd = formData.businessLicenseExpire[1];
             formData.businessLicenseExpireEnd = formData.businessLicenseExpire[1];
           }
           }
 
 
+          if (formData.settlementAgreementExpiry && formData.settlementAgreementExpiry.length === 2) {
+            formData.settlementAgreementStart = formData.settlementAgreementExpiry[0];
+            formData.settlementAgreementEnd = formData.settlementAgreementExpiry[1];
+          }else {
+            if(formData.settlementAgreement != null){
+              return this.$message.warning("其它资质-入驻协议,有效期开始日期或结束日期不能为空!");
+            }
+          }
+
+          if (formData.qualityAssuranceAgreementExpiry && formData.qualityAssuranceAgreementExpiry.length === 2) {
+            formData.qualityAssuranceAgreementStart = formData.qualityAssuranceAgreementExpiry[0];
+            formData.qualityAssuranceAgreementEnd = formData.qualityAssuranceAgreementExpiry[1];
+          }else {
+            if(formData.qualityAssuranceAgreement != null){
+              return this.$message.warning("质量保证协议,有效期开始日期或结束日期不能为空!");
+            }
+          }
+
+          if (formData.otherSpecialQualificationExpiry && formData.otherSpecialQualificationExpiry.length === 2) {
+            formData.otherSpecialQualificationStart = formData.otherSpecialQualificationExpiry[0];
+            formData.otherSpecialQualificationEnd = formData.otherSpecialQualificationExpiry[1];
+          }else {
+            if(formData.otherSpecialQualification != null){
+              return this.$message.warning("其它特殊资质,有效期开始日期或结束日期不能为空!");
+            }
+          }
+
           if (formData.storeId != null) {
           if (formData.storeId != null) {
             updateStore(formData).then(response => {
             updateStore(formData).then(response => {
               this.msgSuccess("修改成功");
               this.msgSuccess("修改成功");

+ 49 - 0
src/views/index.vue

@@ -533,6 +533,18 @@
       </el-col>
       </el-col>
     </el-row>
     </el-row>
     <br/>
     <br/>
+
+    <el-dialog
+      title="店铺协议消息提示"
+      :visible.sync="dialogVisible"
+      width="40%"
+      :before-close="handleClose"
+      center>
+      <div v-for="(item,index) in promptList">({{index+1}})、{{item}}</div>
+      <span slot="footer" class="dialog-footer">
+       <el-button type="primary" @click="dialogVisible = false">已 知 晓</el-button>
+      </span>
+    </el-dialog>
   </div>
   </div>
 </template>
 </template>
 
 
@@ -547,6 +559,7 @@ import {
   watchCourseTopTen, watchEndPlayTrend
   watchCourseTopTen, watchEndPlayTrend
 } from "@/api/statistics/statistics";
 } from "@/api/statistics/statistics";
 import dayjs from 'dayjs';
 import dayjs from 'dayjs';
+import {noticeInfo} from '@/api/system/user'
 import { listDept } from '@/api/system/dept'
 import { listDept } from '@/api/system/dept'
 import { listCompany } from '@/api/his/company'
 import { listCompany } from '@/api/his/company'
 import { getConfigByKey } from '@/api/system/config'
 import { getConfigByKey } from '@/api/system/config'
@@ -884,6 +897,8 @@ export default {
   components: {CountTo},
   components: {CountTo},
   data() {
   data() {
     return {
     return {
+      promptList:[],
+      dialogVisible: false,
       medicalMallConfig:{},
       medicalMallConfig:{},
       deptInitOptions:[],
       deptInitOptions:[],
       deptOptions:[],
       deptOptions:[],
@@ -1008,6 +1023,13 @@ export default {
         console.log(this.medicalMallConfig)
         console.log(this.medicalMallConfig)
       }
       }
     });
     });
+
+    this.$nextTick(()=>{
+      const needExecute = localStorage.getItem('needExecuteLoginEvent') === 'true';
+      if (needExecute) {
+        this.handleNoticeInfo();
+      }
+    })
   },
   },
   methods: {
   methods: {
     getDeptOptions(deptId) {
     getDeptOptions(deptId) {
@@ -1565,6 +1587,33 @@ export default {
       this.answerRedPackMoneyViewerChart = echarts.init(this.$refs.answerRedPackMoneyViewerChart)
       this.answerRedPackMoneyViewerChart = echarts.init(this.$refs.answerRedPackMoneyViewerChart)
 
 
       this.answerRedPackMoneyViewerChart.setOption(lineChartOption)
       this.answerRedPackMoneyViewerChart.setOption(lineChartOption)
+    },
+    handleNoticeInfo() {
+      noticeInfo()
+        .then(response => {
+          console.log("noticeInfo接口完整响应:", response);
+          if (response.code === 200) {
+            this.promptList = response.data;
+            if (this.promptList && this.promptList.length > 0) {
+              this.dialogVisible = true;
+            }
+            console.log("通知信息请求完成!", this.promptList);
+            localStorage.setItem('needExecuteLoginEvent', 'false');
+          } else {
+            console.warn("接口返回非成功状态:", response.code);
+          }
+        })
+        .catch(err => {
+          console.error("获取通知信息失败:", err);
+          this.$message.error("加载通知信息失败,请稍后重试");
+        });
+    },
+    handleClose(done) {
+      this.$confirm('确定是否已知晓?')
+        .then(_ => {
+          done();
+        })
+        .catch(_ => {});
     }
     }
   },
   },
 
 

+ 1 - 0
src/views/login.vue

@@ -166,6 +166,7 @@ export default {
             .dispatch("Login", this.loginForm)
             .dispatch("Login", this.loginForm)
             .then(() => {
             .then(() => {
               this.$router.push({ path: this.redirect || "/" });
               this.$router.push({ path: this.redirect || "/" });
+              localStorage.setItem('needExecuteLoginEvent', 'true');
             })
             })
             .catch(() => {
             .catch(() => {
               this.loading = false;
               this.loading = false;