SecurityCenterTab.vue 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424
  1. <template>
  2. <view class="user-form crm-form">
  3. <view class="card">
  4. <view class="bank-menu card-header">
  5. <view v-for="item in types" :key="item.key" class="bank-menu-item"
  6. :class="{ active: selectedType === item.key }" @click="selectedType = item.key">
  7. <!-- <image class="bank-icon" :src="item.icon" mode="widthFix" />-->
  8. <text>{{ item.label }}</text>
  9. </view>
  10. </view>
  11. <uni-row class="demo-uni-row uni-row1" :gutter="20">
  12. <uni-col v-show="selectedType == 1" :xs="24" :sm="24" :md="24" :lg="12" :xl="12" :offset="6">
  13. <view class=" p-3">
  14. <view class="content-title">
  15. <view v-t="'PersonalManagement.Title.CustomerZonePasswordChange'"></view>
  16. </view>
  17. <hr>
  18. <uni-forms :model="passwordInfo" labelWidth="200" label-position="top">
  19. <uni-row class="demo-uni-row">
  20. <uni-col :xs="24">
  21. <uni-forms-item :label="t('PersonalManagement.Label.OldPassword')">
  22. <uni-easyinput :clearable="false" v-model="passwordInfo.oldPassword"
  23. :placeholder="locale == 'es' ? 'Introduzca el nombre de la red' : t('placeholder.input')" />
  24. </uni-forms-item>
  25. </uni-col>
  26. <uni-col :xs="24">
  27. <uni-forms-item :label="t('PersonalManagement.Label.NewPassword')">
  28. <uni-easyinput :clearable="false" v-model="passwordInfo.newPassword"
  29. :placeholder="locale == 'es' ? 'Introduzca nueva contraseña' : t('placeholder.input')" />
  30. </uni-forms-item>
  31. </uni-col>
  32. <uni-col :xs="24">
  33. <uni-forms-item :label="t('PersonalManagement.Label.NewPasswordConfirmation')">
  34. <uni-easyinput :clearable="false" v-model="passwordInfo.checkPass"
  35. :placeholder="locale == 'es' ? 'Introduzca nueva contraseña' : t('placeholder.input')" />
  36. </uni-forms-item>
  37. </uni-col>
  38. <uni-col :xs="24">
  39. <view class="notice-list">
  40. <view v-for="(item, index) in noticeItems" :key="index"
  41. :class="['notice-item', item.valid ? 'isOK' : '']">
  42. {{ item.label }}
  43. </view>
  44. </view>
  45. </uni-col>
  46. <uni-col :xs="24">
  47. <view class="btn btn-confirm mt-4" @click="passwordUpdate">{{ locale == 'es' ?
  48. 'Actualizarcontraseña' : t('Btn.Application') }}</view>
  49. </uni-col>
  50. </uni-row>
  51. </uni-forms>
  52. </view>
  53. </uni-col>
  54. <uni-col v-show="selectedType == 2" :xs="24" :sm="24" :md="24" :lg="12" :xl="12" :offset="6">
  55. <view class=" p-3">
  56. <view class="content-title">
  57. <view v-t="'PersonalManagement.Title.EmailChange'"></view>
  58. </view>
  59. <hr>
  60. <uni-forms :model="emailInfo" labelWidth="200" label-position="top">
  61. <uni-row class="demo-uni-row">
  62. <uni-col :xs="24">
  63. <uni-forms-item :label="t('PersonalManagement.Label.OldEmail')">
  64. <uni-easyinput :clearable="false" v-model="emailInfo.oldEmail"
  65. :placeholder="t('placeholder.input')" />
  66. </uni-forms-item>
  67. </uni-col>
  68. <uni-col :xs="24">
  69. <uni-forms-item :label="t('PersonalManagement.Label.NewEmail')">
  70. <uni-easyinput :clearable="false" v-model="emailInfo.email"
  71. :placeholder="t('placeholder.input')" />
  72. </uni-forms-item>
  73. </uni-col>
  74. <uni-col :xs="24">
  75. <view class="email-code">
  76. <uni-forms-item :label="t('PersonalManagement.Label.MailboxVerificationCode')"
  77. class="email-code-item">
  78. <uni-easyinput :clearable="false" v-model="emailInfo.emailCode"
  79. :placeholder="locale == 'es' ? 'Código de 6 dígitos' : t('placeholder.input')" />
  80. </uni-forms-item>
  81. <view class="btn btn-code" @click="handleGetCode">{{ getCodeString }}</view>
  82. </view>
  83. </uni-col>
  84. <uni-col :xs="24">
  85. <view class="btn btn-confirm mt-4" @click="emailUpdate">{{ locale == 'es' ?
  86. 'Actualizarcontraseña' : t('Btn.Application') }}</view>
  87. </uni-col>
  88. </uni-row>
  89. </uni-forms>
  90. </view>
  91. </uni-col>
  92. </uni-row>
  93. </view>
  94. </view>
  95. </template>
  96. <script setup lang="ts">
  97. import { computed, ref, onMounted } from 'vue';
  98. import { useI18n } from 'vue-i18n';
  99. import { showToast } from "@/utils/toast";
  100. import { personalApi } from '@/service/personal';
  101. import { useEmailCountdown } from '@/hooks/useEmailCountdown';
  102. import config from '@/config';
  103. const {
  104. time,
  105. text: getCodeString,
  106. canSend,
  107. start,
  108. restore,
  109. clear,
  110. } = useEmailCountdown()
  111. const { t, locale } = useI18n();
  112. interface BankListType {
  113. key: string;
  114. label: string;
  115. icon: string;
  116. }
  117. const passwordInfo = ref({
  118. oldPassword: '',
  119. newPassword: '',
  120. checkPass: ''
  121. });
  122. const emailInfo = ref({
  123. email: '',
  124. });
  125. const selectedType = ref(1)
  126. const types = computed(() => {
  127. return [
  128. { key: '1', label: t('PersonalManagement.Title.CustomerZonePasswordChange') },
  129. {
  130. key: '2',
  131. label: t('PersonalManagement.Title.EmailChange'),
  132. },
  133. ]
  134. })
  135. const rule1 = computed(() => {
  136. if (!passwordInfo.value.newPassword) {
  137. return false;
  138. }
  139. return /^.{8,16}$/.test(passwordInfo.value.newPassword);
  140. });
  141. const rule2 = computed(() => {
  142. return /^(?=.*?[a-z])(?=.*?[A-Z]).*$/.test(passwordInfo.value.newPassword);
  143. });
  144. const rule3 = computed(() => {
  145. return /^(?=.*[A-Z])(?=.*[a-z])(?=.*[0-9])(?!.*([~!@&%$^\\(\\)#_]).*\\1.*\\1)[A-Za-z0-9~!@&%$^\\(\\)#_]{8,16}$/.test(
  146. passwordInfo.value.newPassword
  147. );
  148. });
  149. const rule4 = computed(() => {
  150. console.log(/^(?=.*[A-Z])(?=.*[a-z])(?=.*[0-9])(?=.*[~!@&%$^*./\\(\\)\\+\\=#_-])[A-Za-z0-9~!@&%$^*./\\(\\)\\+\\=#_-]{8,16}$/.test(passwordInfo.value.newPassword))
  151. return /^(?=.*[A-Z])(?=.*[a-z])(?=.*[0-9])(?=.*[~!@&%$^*./\\(\\)\\+\\=#_-])[A-Za-z0-9~!@&%$^*./\\(\\)\\+\\=#_-]{8,16}$/.test(
  152. passwordInfo.value.newPassword
  153. );
  154. });
  155. const noticeItems = computed(() => [
  156. { label: t('signup.form.rules.1st'), valid: rule1.value },
  157. { label: t('signup.form.rules.2nd'), valid: rule2.value },
  158. { label: t('signup.form.rules.4rd'), valid: rule4.value }
  159. ])
  160. // 发送邮箱验证码
  161. async function sendEmailCode() {
  162. try {
  163. const res = await personalApi.customUpdateEmailCode({
  164. email: emailInfo.value.email,
  165. oldEmail: emailInfo.value.oldEmail,
  166. });
  167. if (res.code === 200) {
  168. start()
  169. return true;
  170. } else {
  171. showToast(t("Msg.CodeFail"));
  172. return false;
  173. }
  174. } catch (error: any) {
  175. console.log(error, 12);
  176. showToast(t("Msg.CodeFail"));
  177. return false;
  178. }
  179. }
  180. async function handleGetCode() {
  181. if (!emailInfo.value.email) {
  182. showToast(t("vaildate.email.empty"));
  183. return;
  184. }
  185. if (!config.Pattern.Email.test(emailInfo.value.email)) {
  186. showToast(t("vaildate.email.format"));
  187. return;
  188. }
  189. if (!canSend.value) return
  190. await sendEmailCode();
  191. }
  192. // 提交修改邮箱
  193. async function emailUpdate() {
  194. try {
  195. const res = await personalApi.customUpdateEmail({
  196. ...emailInfo.value
  197. });
  198. if (res.code === 200) {
  199. clear()
  200. emailInfo.value = {}
  201. showToast(t("Msg.Success"));
  202. } else {
  203. showToast(res.msg);
  204. }
  205. } catch (error: any) {
  206. showToast(error.msg);
  207. }
  208. }
  209. // 提交修改密码
  210. async function passwordUpdate() {
  211. try {
  212. if (!rule1.value) {
  213. showToast(t("signup.form.rules.1st"));
  214. return;
  215. }
  216. if (!rule2.value) {
  217. showToast(t("signup.form.rules.2nd"));
  218. return;
  219. }
  220. if (!rule4.value) {
  221. showToast(t("signup.form.rules.4rd"));
  222. return;
  223. }
  224. if (passwordInfo.value.newPassword !== passwordInfo.value.checkPass) {
  225. showToast(t("vaildate.password.same"));
  226. return;
  227. }
  228. const res = await personalApi.customUpdatePassword({
  229. ...passwordInfo.value
  230. });
  231. if (res.code === 200) {
  232. // start()
  233. showToast(t("Msg.Success"));
  234. } else {
  235. showToast(res.msg);
  236. }
  237. } catch (error: any) {
  238. showToast(error.msg);
  239. }
  240. }
  241. onMounted(() => {
  242. restore()
  243. })
  244. </script>
  245. <style scoped lang="scss">
  246. @import "@/uni.scss";
  247. :deep(.uni-row1) {
  248. .uni-col {
  249. }
  250. .uni-forms-item {
  251. min-height: px2rpx(79);
  252. margin-bottom: px2rpx(10);
  253. }
  254. .uni-easyinput__content {
  255. //border: none !important;
  256. background-color: transparent;
  257. //background-color: var(--color-zinc-100) !important;
  258. }
  259. }
  260. .btn {
  261. width: 100%;
  262. display: flex;
  263. align-items: center;
  264. justify-content: center;
  265. padding: px2rpx(12) 0;
  266. background-color: var(--color-error);
  267. color: white;
  268. border: none;
  269. font-size: px2rpx(16);
  270. cursor: pointer;
  271. margin-bottom: px2rpx(24);
  272. }
  273. .notice-list {
  274. margin: 0;
  275. padding: 0 px2rpx(12) px2rpx(12) 0;
  276. .notice-item {
  277. font-size: px2rpx(14);
  278. color: var(--bs-emphasis-color);
  279. line-height: px2rpx(24);
  280. &::before {
  281. content: '●';
  282. display: inline-block;
  283. font-size: px2rpx(10);
  284. margin-right: px2rpx(4);
  285. }
  286. }
  287. .isOK {
  288. color: var(--bs-success);
  289. }
  290. }
  291. .email-code {
  292. display: flex;
  293. align-items: flex-end;
  294. :deep(.email-code-item) {
  295. flex: 1;
  296. .uni-easyinput__content{
  297. border-radius: px2rpx(6) 0 0 px2rpx(6);
  298. }
  299. }
  300. .btn-code {
  301. width: 30%;
  302. margin-bottom: px2rpx(10);
  303. padding: px2rpx(10) px2rpx(16);
  304. background-color: #102047;
  305. color: #fff;
  306. height: 3rem;
  307. text-align: center;
  308. border-radius: 0 px2rpx(6) px2rpx(6) 0;
  309. cursor: pointer;
  310. font-size: 14px;
  311. flex-shrink: 0;
  312. }
  313. }
  314. .bank-menu {
  315. background: #fff;
  316. overflow: hidden;
  317. .bank-menu-item {
  318. display: flex;
  319. align-items: center;
  320. gap: px2rpx(12);
  321. padding: px2rpx(10) px2rpx(16);
  322. cursor: pointer;
  323. border: 1px solid #f3f4f6;
  324. font-size: px2rpx(16);
  325. font-weight: 500;
  326. color: #1f2937;
  327. transition: all 0.3s;
  328. height: px2rpx(50);
  329. border-radius: px2rpx(8);
  330. margin-bottom: px2rpx(8);
  331. .bank-icon {
  332. width: px2rpx(50);
  333. }
  334. &.active {
  335. background: #ea2027;
  336. color: var(--bs-emphasis-color);
  337. border-radius: px2rpx(0);
  338. }
  339. &:hover {
  340. background: #f9fafb;
  341. }
  342. &.active:hover {
  343. background: #d11920;
  344. }
  345. }
  346. }
  347. .bank-menu {
  348. // background: #fff;
  349. display: flex;
  350. flex-wrap: wrap;
  351. overflow: hidden;
  352. &.card-header {
  353. padding: px2rpx(10) px2rpx(20);
  354. }
  355. .bank-menu-item {
  356. flex: 1;
  357. min-width: px2rpx(260);
  358. display: flex;
  359. align-items: center;
  360. justify-content: center;
  361. //gap: px2rpx(12);
  362. padding: 0 px2rpx(16);
  363. cursor: pointer;
  364. font-size: px2rpx(16);
  365. font-weight: bold;
  366. color: var(--bs-emphasis-color);
  367. transition: all 0.3s;
  368. height: px2rpx(36);
  369. border-radius: px2rpx(8);
  370. .bank-icon {
  371. width: px2rpx(50);
  372. }
  373. &.active {
  374. background: #f5f5f5;
  375. color: #000;
  376. border-radius: px2rpx(8);
  377. &:hover {
  378. background: var(--bs-link-hover-color-rgb);
  379. color: #333;
  380. }
  381. }
  382. &:hover {
  383. background: #f5f5f5;
  384. color: #333;
  385. }
  386. &.active:hover {
  387. background: #f5f5f5;
  388. }
  389. }
  390. }
  391. </style>