merchant.vue 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634
  1. <template>
  2. <s-layout class="set-wrap" title="商家入驻" :bgStyle="{ color: '#FFF' }">
  3. <template v-if="state.canUse">
  4. <uni-forms :model="state.model" :rules="state.rules" validateTrigger="bind" labelPosition="left" border
  5. class="form-box" labelWidth='200' ref="merchantFormRef">
  6. <view class="bg-white ss-p-x-30">
  7. <uni-forms-item name="name" label="商家名称" :required="!state.formStatus">
  8. <uni-easyinput v-model="state.model.name" type="name" placeholder="请输入商家名称" :inputBorder="false"
  9. :placeholderStyle="placeholderStyle" :clearable="false" :disabled="state.formStatus" />
  10. </uni-forms-item>
  11. <uni-forms-item name="contact" label="联络人" :required="!state.formStatus">
  12. <uni-easyinput v-model="state.model.contact" type="contact" placeholder="请输入联络人"
  13. :inputBorder="false" :placeholderStyle="placeholderStyle" :clearable="false"
  14. :disabled="state.formStatus" />
  15. </uni-forms-item>
  16. <uni-forms-item name="contactNumber" label="联络人手机号" :required="!state.formStatus">
  17. <uni-easyinput v-model="state.model.contactNumber" type="contactNumber" placeholder="请输入联络人手机号"
  18. :inputBorder="false" :placeholderStyle="placeholderStyle" :clearable="false"
  19. :disabled="state.formStatus" />
  20. </uni-forms-item>
  21. <uni-forms-item name="legalPerson" label="法人" :required="!state.formStatus">
  22. <uni-easyinput v-model="state.model.legalPerson" type="legalPerson" placeholder="请输入法人"
  23. :inputBorder="false" :placeholderStyle="placeholderStyle" :clearable="false"
  24. :disabled="state.formStatus" />
  25. </uni-forms-item>
  26. <uni-forms-item name="legalPersonNumber" label="法人手机号" :required="!state.formStatus">
  27. <uni-easyinput v-model="state.model.legalPersonNumber" type="legalPersonNumber"
  28. placeholder="请输入法人手机号" :inputBorder="false" :placeholderStyle="placeholderStyle"
  29. :clearable="false" :disabled="state.formStatus" />
  30. </uni-forms-item>
  31. <uni-forms-item name="identityCardFront" label="法人身份证正面" :required="!state.formStatus"
  32. :errorMessage="state.identityCardFrontError">
  33. <s-uploader v-model:url="state.model.identityCardFront" fileMediatype="image" limit="1"
  34. mode="grid" :imageStyles="{ width: '299rpx', height: '168rpx' }"
  35. :readonly="state.formStatus" />
  36. </uni-forms-item>
  37. <uni-forms-item name="identityCardReverseSide" label="法人身份证反面" :required="!state.formStatus"
  38. :errorMessage="identityCardReverseSideError">
  39. <s-uploader v-model:url="state.model.identityCardReverseSide" fileMediatype="image" limit="1"
  40. mode="grid" :imageStyles="{ width: '299rpx', height: '168rpx' }"
  41. :readonly="state.formStatus" />
  42. </uni-forms-item>
  43. <uni-forms-item name="areaId" label="所在地" :required="!state.formStatus"
  44. :errorMessage="state.areaIdError">
  45. <uni-data-picker v-model="state.model.areaId" :localdata="areaTree"
  46. :readonly="state.formStatus" />
  47. </uni-forms-item>
  48. <uni-forms-item name="email" label="邮箱">
  49. <uni-easyinput v-model="state.model.email" type="email" placeholder="请输入邮箱" :inputBorder="false"
  50. :placeholderStyle="placeholderStyle" :clearable="false" :disabled="state.formStatus" />
  51. </uni-forms-item>
  52. <uni-forms-item name="address" label="办公地址">
  53. <uni-easyinput v-model="state.model.address" type="address" placeholder="请输入办公地址"
  54. :inputBorder="false" :placeholderStyle="placeholderStyle" :clearable="false"
  55. :disabled="state.formStatus" />
  56. </uni-forms-item>
  57. <uni-forms-item name="complaintsHotline" label="维权电话">
  58. <uni-easyinput v-model="state.model.complaintsHotline" type="complaintsHotline"
  59. placeholder="请输入维权电话" :inputBorder="false" :placeholderStyle="placeholderStyle"
  60. :clearable="false" :disabled="state.formStatus" />
  61. </uni-forms-item>
  62. <uni-forms-item name="customerServiceHotline" label="客服电话" :required="!state.formStatus">
  63. <uni-easyinput v-model="state.model.customerServiceHotline" type="customerServiceHotline"
  64. placeholder="请输入客服电话" :inputBorder="false" :placeholderStyle="placeholderStyle"
  65. :clearable="false" :disabled="state.formStatus" />
  66. </uni-forms-item>
  67. <uni-forms-item name="website" label="官网">
  68. <uni-easyinput v-model="state.model.website" type="website" placeholder="请输入官网"
  69. :inputBorder="false" :placeholderStyle="placeholderStyle" :clearable="false"
  70. :disabled="state.formStatus" />
  71. </uni-forms-item>
  72. <uni-forms-item name="bankName" label="开户银行" :required="!state.formStatus">
  73. <uni-easyinput v-model="state.model.bankName" type="bankName" placeholder="请输入开户银行"
  74. :inputBorder="false" :placeholderStyle="placeholderStyle" :clearable="false"
  75. :disabled="state.formStatus" />
  76. </uni-forms-item>
  77. <uni-forms-item name="accountName" label="账户名称" :required="!state.formStatus">
  78. <uni-easyinput v-model="state.model.accountName" type="accountName" placeholder="请输入账户名称"
  79. :inputBorder="false" :placeholderStyle="placeholderStyle" :clearable="false"
  80. :disabled="state.formStatus" />
  81. </uni-forms-item>
  82. <uni-forms-item name="accountNumber" label="账户号码" :required="!state.formStatus">
  83. <uni-easyinput v-model="state.model.accountNumber" type="number" placeholder="请输入账户号码"
  84. :inputBorder="false" :placeholderStyle="placeholderStyle" :clearable="false"
  85. :disabled="state.formStatus" />
  86. </uni-forms-item>
  87. <uni-forms-item name="logoUrl" label="Logo">
  88. <s-uploader v-model:url="state.model.logoUrl" fileMediatype="image" limit="1" mode="grid"
  89. :imageStyles="{ width: '299rpx', height: '168rpx' }" :readonly="state.formStatus" />
  90. </uni-forms-item>
  91. <uni-forms-item name="businessLicensePicture" label="营业执照" :required="!state.formStatus"
  92. :errorMessage="state.businessLicensePictureError">
  93. <s-uploader v-model:url="state.model.businessLicensePicture" fileMediatype="image" limit="1"
  94. mode="grid" :imageStyles="{ width: '299rpx', height: '168rpx' }"
  95. :readonly="state.formStatus" />
  96. </uni-forms-item>
  97. <uni-forms-item name="brandLicensing" label="品牌授权书" :required="!state.formStatus"
  98. :errorMessage="state.brandLicensingError">
  99. <s-uploader v-model:url="state.model.brandLicensing" fileMediatype="image" limit="1" mode="grid"
  100. :imageStyles="{ width: '299rpx', height: '168rpx' }" :readonly="state.formStatus" />
  101. <view class="ss-m-t-20 " style="color: var(--ui-BG-Main);" @click="downloadTemplate"
  102. v-if="!state.model.brandLicensing">
  103. 下载模板
  104. </view>
  105. </uni-forms-item>
  106. <uni-forms-item name="otherCertificate" label="其他证书">
  107. <s-uploader v-model:url="state.model.otherCertificate" fileMediatype="image" limit="9"
  108. mode="grid" :imageStyles="{ width: '299rpx', height: '168rpx' }"
  109. :readonly="state.formStatus" />
  110. </uni-forms-item>
  111. <uni-forms-item name="description" label="简介" :required="!state.formStatus">
  112. <uni-easyinput v-model="state.model.description" type="description" placeholder="请输入简介"
  113. :inputBorder="false" :placeholderStyle="placeholderStyle" :clearable="false"
  114. :disabled="state.formStatus" />
  115. </uni-forms-item>
  116. <uni-forms-item name="description" label="状态" v-if="state.userApplyStatus">
  117. <uni-easyinput v-model="statusName" type="description" placeholder="请输入简介" :inputBorder="false"
  118. :placeholderStyle="placeholderStyle" :clearable="false" :disabled="state.formStatus" />
  119. </uni-forms-item>
  120. </view>
  121. </uni-forms>
  122. <su-fixed bottom placeholder bg="none">
  123. <view class="footer-box ss-p-20 ss-flex">
  124. <!-- 审核中不允许改 -->
  125. <!-- 通过后 变动 -->
  126. <!-- {{"用户是否申请过:"+state.userApplyStatus}}
  127. {{"当前状态"+state.model.checkStatus}}
  128. {{"当前状态是否能改动:" + !state.formStatus}} -->
  129. <button class="ss-rest-button btn" @tap="onSubmit" v-if="!state.userApplyStatus">提交</button>
  130. <template v-if="[0].includes(state.model.checkStatus)">
  131. <button class="ss-rest-button btn">审核中</button>
  132. </template>
  133. <template v-if="[1,2].includes(state.model.checkStatus)">
  134. <template v-if="!state.changeIng">
  135. <button class="ss-rest-button btn-two" @tap="openList">审核记录</button>
  136. <button class="ss-rest-button btn-two" @tap="onChange">变动</button>
  137. </template>
  138. <template v-else>
  139. <button class="ss-rest-button btn-two" @tap="onCancel">取消</button>
  140. <button class="ss-rest-button btn-two" @tap="onSave">保存并提交</button>
  141. </template>
  142. </template>
  143. </view>
  144. </su-fixed>
  145. </template>
  146. <template v-if="!state.canUse">
  147. <view style="position: fixed;top: 50%;left: 50%;transform: translate(-50%,-50%);text-align:center">当前用户({{userInfo.mobile}})<br/>
  148. 已存在商户账号<br/>
  149. 请更换账号申请!
  150. </view>
  151. </template>
  152. </s-layout>
  153. </template>
  154. <script setup>
  155. import {
  156. computed,
  157. reactive,
  158. onBeforeMount,
  159. ref,
  160. unref,
  161. } from 'vue';
  162. import sheep from '@/sheep';
  163. import {
  164. clone
  165. } from 'lodash';
  166. import {
  167. onLoad
  168. } from '@dcloudio/uni-app';
  169. import SaleApi from '@/sheep/api/sale/sale';
  170. import AreaApi from '@/sheep/api/system/area'
  171. import {
  172. email
  173. } from '@/sheep/validate/form';
  174. const userInfo = computed(() => sheep.$store('user').userInfo);
  175. const state = reactive({
  176. canUse: true, //用户是否可以打开页面true 可以,false不行
  177. userApplyStatus: false, //用户是否已经申请过false未无 true为有
  178. formStatus: false, // 当前表单是否可以修改
  179. changeIng: false, //当前是否在修改
  180. model: {
  181. checkStatus: undefined,
  182. id: undefined,
  183. name: undefined,
  184. status: undefined,
  185. description: undefined,
  186. contact: undefined,
  187. address: undefined,
  188. contactNumber: undefined,
  189. website: undefined,
  190. createTime: undefined,
  191. complaintsHotline: undefined,
  192. customerServiceHotline: undefined,
  193. email: undefined,
  194. businessLicensePicture: undefined,
  195. expireTime: undefined,
  196. areaId: undefined,
  197. accountName: undefined,
  198. accountNumber: undefined,
  199. bankName: undefined,
  200. logoUrl: undefined,
  201. brandLicensing: undefined,
  202. otherCertificate: [],
  203. legalPerson: undefined,
  204. legalPersonNumber: undefined,
  205. identityCardFront: undefined,
  206. identityCardReverseSide: undefined,
  207. },
  208. areaIdError: '',
  209. businessLicensePictureError: '',
  210. brandLicensingError: '',
  211. identityCardFrontError: '',
  212. identityCardReverseSideError: '',
  213. rules: {
  214. name: {
  215. rules: [{
  216. required: true,
  217. errorMessage: '商户名称不能为空',
  218. }, ],
  219. },
  220. description: {
  221. rules: [{
  222. required: true,
  223. errorMessage: '简介不能为空',
  224. }, ],
  225. },
  226. contact: {
  227. rules: [{
  228. required: true,
  229. errorMessage: '联络人不能为空',
  230. }, ],
  231. },
  232. contactNumber: {
  233. rules: [{
  234. required: true,
  235. errorMessage: '联络人手机号不能为空',
  236. }, ],
  237. },
  238. customerServiceHotline: {
  239. rules: [{
  240. required: true,
  241. errorMessage: '客服电话不能为空',
  242. }, ],
  243. },
  244. email,
  245. legalPerson: {
  246. rules: [{
  247. required: true,
  248. errorMessage: '法人不能为空',
  249. }, ],
  250. },
  251. legalPersonNumber: {
  252. rules: [{
  253. required: true,
  254. errorMessage: '法人电话不能为空',
  255. }, ],
  256. },
  257. accountName: {
  258. rules: [{
  259. required: true,
  260. errorMessage: '账户名称不能为空',
  261. }, ],
  262. },
  263. accountNumber: {
  264. rules: [{
  265. required: true,
  266. errorMessage: '账户号码不能为空',
  267. }, ],
  268. },
  269. bankName: {
  270. rules: [{
  271. required: true,
  272. errorMessage: '开户银行不能为空',
  273. }, ],
  274. },
  275. }
  276. });
  277. const placeholderStyle = 'color:#BBBBBB;font-size:28rpx;line-height:normal';
  278. const statusName = computed(() => {
  279. const status = state.model.checkStatus;
  280. if (status === 0) {
  281. return '审核中';
  282. } else if (status === 1) {
  283. return '通过';
  284. } else if (status === 2) {
  285. return '拒绝';
  286. } else {
  287. return '';
  288. }
  289. });
  290. // 下载模板
  291. const downloadTemplate = () => {
  292. // console.log('下载模板')
  293. const fileUrl =
  294. 'https://zxgz.newfeifan.cn/static/file/%E9%9D%9E%E7%8B%AC%E5%AE%B6%E5%93%81%E7%89%8C%E6%8E%88%E6%9D%83%E4%B9%A6.docx'
  295. // 创建并触发下载链接
  296. const a = document.createElement('a');
  297. a.href = fileUrl;
  298. a.download = '非独家品牌授权书.docx'; // 下载文件的名称(可以修改为你希望的名称)
  299. document.body.appendChild(a);
  300. a.click();
  301. // 移除下载链接
  302. document.body.removeChild(a);
  303. }
  304. // 校验联络人是否已经申请入驻,如果已经入驻,那就不给提交
  305. const checkNumber = () => {
  306. const {
  307. data,
  308. code
  309. } = SaleApi.checkContactNumber({
  310. mobile: state.model.contactNumber
  311. });
  312. }
  313. // 提交审核
  314. const merchantFormRef = ref(null);
  315. const onSubmit = async () => {
  316. // 参数校验
  317. const validate = await unref(merchantFormRef)
  318. .validate()
  319. .catch((error) => {
  320. console.log('error: ', error);
  321. });
  322. if (!validate) {
  323. return;
  324. }
  325. if (!state.model.areaId) {
  326. state.areaIdError = '请选择所在地';
  327. return;
  328. }
  329. if (!state.model.identityCardFront) {
  330. state.identityCardFrontError = '请选择上传法人身份证正面(国徽面)';
  331. return;
  332. }
  333. if (!state.model.identityCardReverseSide) {
  334. state.identityCardReverseSideError = '请选择上传法人身份证背面(人像面)';
  335. return;
  336. }
  337. if (!state.model.businessLicensePicture) {
  338. state.businessLicensePictureError = '请上传营业执照';
  339. return;
  340. }
  341. if (!state.model.brandLicensing) {
  342. state.brandLicensingError = '请上传品牌授权书';
  343. return;
  344. }
  345. const {
  346. data,
  347. code
  348. } = await SaleApi.createMerchant(state.model);
  349. if (code === 0) {
  350. await getInfo()
  351. }
  352. }
  353. // 修改
  354. const onChange = () => {
  355. state.formStatus = false;
  356. state.changeIng = true;
  357. }
  358. // 取消修改
  359. const onCancel = () => {
  360. state.formStatus = true;
  361. state.changeIng = false;
  362. }
  363. const openList = () => {
  364. sheep.$router.go('/pages/public/merchantApplyList', {
  365. merchantApplyId: state.model.id
  366. })
  367. }
  368. // 保存并提交
  369. const onSave = async () => {
  370. // 参数校验
  371. const validate = await unref(merchantFormRef)
  372. .validate()
  373. .catch((error) => {
  374. console.log('error: ', error);
  375. });
  376. if (!validate) {
  377. return;
  378. }
  379. const {
  380. data,
  381. code
  382. } = await SaleApi.updateMerchant(state.model);
  383. if (code === 0) {
  384. state.formStatus = true;
  385. state.changeIng = false;
  386. await getInfo()
  387. }
  388. }
  389. // 地区树
  390. const areaTree = ref([])
  391. function format(data) {
  392. return data.map((item) => ({
  393. text: item.name, // 显示的文本
  394. value: item.id, // 值
  395. children: item.children.length ? format(item.children) : null // 子项
  396. }));
  397. }
  398. const getInfo = async () => {
  399. const {
  400. data,
  401. code
  402. } = await SaleApi.getMerchant();
  403. if (code === 0 && data) {
  404. // 如果能查到东西就证明当前用户已经申请了商户,只需要展示即可
  405. state.model = data;
  406. state.userApplyStatus = true;
  407. state.formStatus = true;
  408. } else {
  409. state.userApplyStatus = false;
  410. state.formStatus = false;
  411. }
  412. }
  413. onLoad(async (options) => {
  414. await SaleApi.checkSystemUser().then(res => {
  415. console.log(!res.data)
  416. state.canUse = !res.data;
  417. })
  418. await AreaApi.getAreaTree().then(res => {
  419. areaTree.value = format(res.data)
  420. })
  421. await getInfo();
  422. });
  423. </script>
  424. <style lang="scss" scoped>
  425. .icon {
  426. display: flex;
  427. align-items: center;
  428. margin-right: 7rpx
  429. }
  430. .icon image {
  431. width: 35rpx;
  432. height: 35rpx;
  433. }
  434. :deep() {
  435. .file-picker__progress {
  436. height: 0 !important;
  437. }
  438. .uni-list-item__content-title {
  439. font-size: 28rpx !important;
  440. color: #333333 !important;
  441. line-height: normal !important;
  442. }
  443. .uni-icons {
  444. font-size: 40rpx !important;
  445. }
  446. .is-disabled {
  447. color: #333333;
  448. }
  449. }
  450. :deep(.disabled) {
  451. opacity: 1;
  452. }
  453. .gender-name {
  454. font-size: 28rpx;
  455. font-weight: 500;
  456. line-height: normal;
  457. color: #333333;
  458. }
  459. .title-box {
  460. font-size: 28rpx;
  461. font-weight: 500;
  462. color: #666666;
  463. line-height: 100rpx;
  464. }
  465. .btn {
  466. width: 710rpx;
  467. height: 80rpx;
  468. background: linear-gradient(90deg, var(--ui-BG-Main), var(--ui-BG-Main-gradient));
  469. border-radius: 40rpx;
  470. font-size: 30rpx;
  471. font-weight: 500;
  472. color: $white;
  473. }
  474. .btn-two {
  475. width: 310rpx;
  476. height: 80rpx;
  477. background: linear-gradient(90deg, var(--ui-BG-Main), var(--ui-BG-Main-gradient));
  478. border-radius: 40rpx;
  479. font-size: 30rpx;
  480. font-weight: 500;
  481. color: $white;
  482. }
  483. .radio-dark {
  484. filter: grayscale(100%);
  485. filter: gray;
  486. opacity: 0.4;
  487. }
  488. .content-img {
  489. border-radius: 50%;
  490. }
  491. .header-box-content {
  492. position: relative;
  493. width: 160rpx;
  494. height: 160rpx;
  495. overflow: hidden;
  496. border-radius: 50%;
  497. }
  498. .avatar-action {
  499. position: absolute;
  500. left: 50%;
  501. transform: translateX(-50%);
  502. bottom: 0;
  503. z-index: 1;
  504. width: 160rpx;
  505. height: 46rpx;
  506. background: rgba(#000000, 0.3);
  507. .avatar-action-btn {
  508. width: 160rpx;
  509. height: 46rpx;
  510. font-weight: 500;
  511. font-size: 24rpx;
  512. color: #ffffff;
  513. }
  514. }
  515. // 绑定项
  516. .account-list {
  517. background-color: $white;
  518. height: 100rpx;
  519. padding: 0 20rpx;
  520. .list-img {
  521. width: 40rpx;
  522. height: 40rpx;
  523. margin-right: 10rpx;
  524. }
  525. .list-name {
  526. font-size: 28rpx;
  527. color: #333333;
  528. }
  529. .info {
  530. .avatar {
  531. width: 38rpx;
  532. height: 38rpx;
  533. border-radius: 50%;
  534. overflow: hidden;
  535. }
  536. .name {
  537. font-size: 28rpx;
  538. font-weight: 400;
  539. color: $dark-9;
  540. }
  541. }
  542. .bind-box {
  543. width: 100rpx;
  544. height: 50rpx;
  545. line-height: normal;
  546. display: flex;
  547. justify-content: center;
  548. align-items: center;
  549. font-size: 24rpx;
  550. .bind-btn {
  551. width: 100%;
  552. height: 100%;
  553. border-radius: 25rpx;
  554. background: #f4f4f4;
  555. color: #999999;
  556. }
  557. .relieve-btn {
  558. width: 100%;
  559. height: 100%;
  560. border-radius: 25rpx;
  561. background: var(--ui-BG-Main-opacity-1);
  562. color: var(--ui-BG-Main);
  563. }
  564. }
  565. }
  566. .list-border {
  567. font-size: 28rpx;
  568. font-weight: 400;
  569. color: #333333;
  570. border-bottom: 2rpx solid #eeeeee;
  571. }
  572. image {
  573. width: 100%;
  574. height: 100%;
  575. }
  576. </style>