loginDialog.vue 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751
  1. <template>
  2. <div>
  3. <el-dialog
  4. :custom-class="
  5. activeName === 'first' ? 'custom-login-form' : 'custom-register-form'
  6. "
  7. :lock-scroll="false"
  8. :visible.sync="dialogVisible"
  9. :width="dialogWidth"
  10. :before-close="close">
  11. <el-tabs v-model="activeName">
  12. <el-tab-pane
  13. label="LOGIN"
  14. name="first">
  15. <GlobalForm
  16. :formData="loginData"
  17. :formConfig="loginFormConfig"
  18. :rules="loginRules"
  19. :submitLoading="submitLoading"
  20. @handleSubmit="submit"
  21. :showDeleteBtn="showDeleteBtn"
  22. :showRegisterBtn="false"
  23. ref="loginForm">
  24. <template #checkbox>
  25. <div class="rememberMe">
  26. <el-checkbox v-model="isRemember">Remember me</el-checkbox>
  27. <div
  28. @click="resetPassword"
  29. class="btn-forgot-pw">
  30. Forgot Password
  31. </div>
  32. </div>
  33. </template>
  34. </GlobalForm>
  35. <div
  36. style="
  37. font-size: 14px;
  38. color: #666;
  39. margin-top: 18px;
  40. text-align: right;
  41. ">
  42. Strictly distributors only. Proof of status will be required
  43. </div>
  44. </el-tab-pane>
  45. <el-tab-pane
  46. label="REGISTER AS A NEW USER"
  47. name="second">
  48. <div class="flex start">
  49. <div style="width: 50%; min-width: 50%; padding-top: 48px">
  50. <div
  51. class="flex center"
  52. style="margin-bottom: 55px">
  53. <img
  54. src="@/assets/img/logo.png"
  55. alt="PromoCollection Logo" />
  56. </div>
  57. <div class="desc-content main-desc">
  58. PromoCollection is Australia's fastest growing
  59. <br />
  60. promotional products supplier, 5 years in a row!
  61. </div>
  62. <p class="desc-content">
  63. We work exclusively with distributors to offer over 2000
  64. catalogue products, as well as a world of sourcing options, with
  65. our 60 person offshore sourcing team,to source anything you
  66. need!
  67. </p>
  68. <div class="desc-content">
  69. To apply for a login, please input the below information
  70. </div>
  71. <div
  72. class="signature-image flex center"
  73. v-if="config.configInfo?.image">
  74. <img
  75. :class="{ pointer: config.configInfo?.link }"
  76. @click="toSignatureLink"
  77. :src="config.configInfo.image"
  78. alt="" />
  79. </div>
  80. </div>
  81. <div class="flex-auto register-form">
  82. <el-form
  83. :inline="true"
  84. :model="registData"
  85. :rules="registRules"
  86. ref="registForm"
  87. label-position="top">
  88. <el-form-item
  89. v-for="item in registFormConfig"
  90. :class="{
  91. 'long-form-item': item.long,
  92. 'short-form-item': !item.long,
  93. }"
  94. :prop="item.prop"
  95. :label="item.placeholder"
  96. :key="item.label">
  97. <el-input
  98. v-if="item.type === 'password'"
  99. :type="showPassword ? 'text' : 'password'"
  100. v-model="registData[item.prop]">
  101. <i
  102. v-if="!showPassword"
  103. slot="suffix"
  104. class="el-input__icon iconfont"
  105. style="font-size: 22px; cursor: pointer; color: #666"
  106. @click="showPassword = true"
  107. >&#xebcc;</i
  108. >
  109. <i
  110. v-else
  111. slot="suffix"
  112. class="el-input__icon iconfont"
  113. style="font-size: 22px; cursor: pointer; color: #666"
  114. @click="showPassword = false"
  115. >&#xebcd;</i
  116. >
  117. </el-input>
  118. <el-input
  119. :id="item.prop + '_input'"
  120. @change="$e => onInputChange($e, item.prop)"
  121. v-else
  122. v-model="registData[item.prop]"></el-input>
  123. </el-form-item>
  124. </el-form>
  125. <div class="flex center">
  126. <el-button
  127. :loading="submitLoading"
  128. @click="checkRegisterForm"
  129. >REGISTER</el-button
  130. >
  131. </div>
  132. <p class="desc-content">* required</p>
  133. </div>
  134. </div>
  135. </el-tab-pane>
  136. </el-tabs>
  137. </el-dialog>
  138. <resetPasswordDialog
  139. ref="resetPasswordDialog"
  140. @openLoginDialog="openLoginDialog" />
  141. <dialog-XX-success
  142. :visible.sync="xxContentVisible"
  143. :content="xxContent"></dialog-XX-success>
  144. </div>
  145. </template>
  146. <script>
  147. import { mapState, mapMutations } from 'vuex'
  148. import GlobalForm from '../components/PcGlobalForm.vue'
  149. import dialogXXSuccess from './DIalogXXSuccess.vue'
  150. const Base64 = require('js-base64').Base64
  151. export default {
  152. name: 'LoginDialog',
  153. components: { GlobalForm, 'dialog-XX-success': dialogXXSuccess },
  154. data() {
  155. return {
  156. showPassword: false,
  157. xxContentVisible: false,
  158. xxContent: 'success',
  159. isRemember: false,
  160. showDeleteBtn: false,
  161. activeName: 'first',
  162. submitLoading: false,
  163. loginData: {
  164. email: '',
  165. password: '',
  166. },
  167. registData: {},
  168. // 表单配置
  169. loginFormConfig: [
  170. {
  171. autoInput: true,
  172. prop: 'email',
  173. type: 'input',
  174. placeholder: 'User Name',
  175. },
  176. {
  177. prop: 'password',
  178. type: 'password',
  179. placeholder: 'Password',
  180. },
  181. {
  182. type: 'slot',
  183. slotName: 'checkbox',
  184. },
  185. ],
  186. registFormConfig: [
  187. {
  188. long: true,
  189. prop: 'email',
  190. type: 'input',
  191. placeholder: 'Email',
  192. },
  193. {
  194. long: true,
  195. prop: 'password',
  196. type: 'password',
  197. placeholder: 'Password',
  198. },
  199. {
  200. long: true,
  201. prop: 'confirm_password',
  202. type: 'password',
  203. placeholder: 'Comfirm Password',
  204. },
  205. {
  206. prop: 'contacts',
  207. type: 'input',
  208. placeholder: 'First Name',
  209. },
  210. {
  211. prop: 'last_name',
  212. type: 'input',
  213. placeholder: 'Last Name',
  214. },
  215. {
  216. long: true,
  217. prop: 'company',
  218. type: 'input',
  219. placeholder: 'Company Name',
  220. },
  221. {
  222. prop: 'phone',
  223. type: 'input',
  224. placeholder: 'Phone No',
  225. },
  226. {
  227. prop: 'url',
  228. type: 'input',
  229. placeholder: 'Website',
  230. },
  231. {
  232. long: true,
  233. prop: 'address',
  234. type: 'input',
  235. placeholder: 'Address',
  236. },
  237. {
  238. prop: 'state',
  239. type: 'input',
  240. placeholder: 'State',
  241. },
  242. {
  243. prop: 'postcode',
  244. type: 'input',
  245. placeholder: 'Post Code',
  246. },
  247. ],
  248. // 表单规则
  249. loginRules: {
  250. email: [
  251. {
  252. required: true,
  253. message: 'Please enter the correct email',
  254. trigger: 'blur',
  255. },
  256. ],
  257. password: [
  258. {
  259. required: true,
  260. message: 'The password length only allows 6-16 bits',
  261. trigger: 'blur',
  262. min: 6,
  263. max: 16,
  264. },
  265. ],
  266. },
  267. registRules: {
  268. email: [
  269. {
  270. required: true,
  271. message: 'Please enter your email address',
  272. trigger: 'blur',
  273. },
  274. ],
  275. password: [
  276. {
  277. required: true,
  278. message: 'The password length only allows 6-16 bits',
  279. trigger: 'blur',
  280. min: 6,
  281. max: 16,
  282. },
  283. ],
  284. confirm_password: [
  285. {
  286. required: true,
  287. message: 'The password length only allows 6-16 bits',
  288. trigger: 'blur',
  289. min: 6,
  290. max: 16,
  291. },
  292. ],
  293. company: [
  294. {
  295. required: true,
  296. message: 'Please enter your company name',
  297. trigger: 'blur',
  298. },
  299. ],
  300. contacts: [
  301. {
  302. required: true,
  303. message: 'Please enter your contact person',
  304. trigger: 'blur',
  305. },
  306. ],
  307. last_name: [
  308. {
  309. required: true,
  310. message: 'Please enter your contact person',
  311. trigger: 'blur',
  312. },
  313. ],
  314. phone: [
  315. {
  316. required: true,
  317. message: 'Please enter your phone number',
  318. trigger: 'blur',
  319. },
  320. ],
  321. },
  322. autoComplete: null,
  323. // 记录谷歌地图API选择地点后拿到的地址名. 这个不能直接更新到registeData上, 不然输入框内容会再次变化.
  324. selectAddress: '',
  325. }
  326. },
  327. computed: {
  328. ...mapState(['dialogVisible', 'config']),
  329. dialogWidth() {
  330. return this.activeName === 'first' ? '506px' : '1108px'
  331. },
  332. },
  333. watch: {
  334. dialogVisible(value) {
  335. if (value) {
  336. document.addEventListener('keydown', this.onEnterClick)
  337. } else {
  338. document.removeEventListener('keydown', this.onEnterClick)
  339. }
  340. },
  341. activeName() {
  342. if (this.activeName === 'second' && this.autoComplete === null) {
  343. if (window.google) {
  344. this.initGoogleMapAutoComplete()
  345. this.getAddrByLocation()
  346. } else {
  347. setTimeout(() => {
  348. this.initGoogleMapAutoComplete()
  349. this.getAddrByLocation()
  350. }, 1500)
  351. }
  352. }
  353. },
  354. },
  355. mounted() {
  356. this.getCookie()
  357. },
  358. methods: {
  359. ...mapMutations(['closeDialog', 'openDialog']),
  360. onEnterClick(e) {
  361. if (e.keyCode === 13) {
  362. console.log('enter')
  363. let target = this.$refs.loginForm
  364. if (this.activeName === 'first') {
  365. // do nothing. login form
  366. }
  367. if (this.activeName === 'second') {
  368. target = this.$refs.registForm
  369. }
  370. if (typeof target.submit === 'function') {
  371. target.submit()
  372. }
  373. }
  374. },
  375. checkRegisterForm() {
  376. this.$refs.registForm.validate(valid => {
  377. if (valid) {
  378. this.submit()
  379. }
  380. })
  381. },
  382. submit() {
  383. if (this.activeName === 'first') {
  384. if (this.isRemember) {
  385. const password = Base64.encode(this.loginData.password)
  386. this.setCookie(this.loginData.email, password, 7)
  387. } else {
  388. this.setCookie('', '', -1)
  389. }
  390. this.login()
  391. } else if (this.activeName === 'second') {
  392. this.regist()
  393. }
  394. },
  395. login() {
  396. this.$store
  397. .dispatch('login', this.loginData)
  398. .then(res => {
  399. if (res.code == 1) {
  400. setTimeout(() => {
  401. this.closeDialog()
  402. }, 200)
  403. this.xxContentVisible = true
  404. this.xxContent = 'Login Successful'
  405. this.submitLoading = false
  406. // this.$confirm("Login Successful", {
  407. // confirmButtonText: "OK",
  408. // confirmButtonClass: 'confirmBtn',
  409. // showCancelButton: false,
  410. // showClose: false,
  411. // type: "success",
  412. // center: true,
  413. // lockScroll: false,
  414. // }).then(() => {
  415. // });
  416. } else if (res.code == 10100) {
  417. this.$confirm(res.msg, {
  418. confirmButtonText: 'OK',
  419. showCancelButton: false,
  420. type: 'error',
  421. center: true,
  422. })
  423. }
  424. })
  425. .catch(() => {
  426. this.closeDialog()
  427. })
  428. },
  429. regist() {
  430. this.submitLoading = true
  431. const formData = Object.assign({}, this.registData)
  432. if (this.selectAddress && this.selectAddress.length) {
  433. formData.address = this.selectAddress
  434. }
  435. this.$axios({
  436. url: '/au/register',
  437. method: 'post',
  438. data: formData,
  439. })
  440. .then(res => {
  441. if (res.code == 1) {
  442. this.closeDialog()
  443. this.submitLoading = false
  444. this.xxContentVisible = true
  445. this.xxContent =
  446. "We've received your login request. Someone will be in touch shortly to confirm."
  447. this.registerEmail()
  448. }
  449. })
  450. .catch(e => {
  451. const res = e.response.data
  452. if (res.code === 10100) {
  453. this.$confirm(res.msg + ', please click OK to jump to login.', {
  454. confirmButtonText: 'OK',
  455. showCancelButton: false,
  456. type: 'error',
  457. center: true,
  458. }).then(() => {
  459. this.submitLoading = false
  460. this.activeName = 'first'
  461. })
  462. }
  463. })
  464. },
  465. registerEmail() {
  466. this.$axios
  467. .get('/member/register_email?email=' + this.registData.email)
  468. .then(res => {})
  469. .catch(error => {
  470. this.$message.error(error.response.data.msg)
  471. })
  472. },
  473. close() {
  474. this.closeDialog()
  475. this.registData = {}
  476. },
  477. resetPassword() {
  478. this.closeDialog()
  479. this.$refs.resetPasswordDialog.resetDialogVisible = true
  480. },
  481. openLoginDialog() {
  482. this.openDialog()
  483. },
  484. // 设置cookie
  485. setCookie(email, password, days) {
  486. const date = new Date()
  487. date.setTime(date.getTime() + 24 * 60 * 60 * 1000 * days)
  488. document.cookie =
  489. 'email' + '=' + email + ';path=/;expires=' + date.toGMTString()
  490. document.cookie =
  491. 'password' + '=' + password + ';path=/;expires=' + date.toGMTString()
  492. },
  493. // 读取cookie 将用户名和密码回显到input框中
  494. getCookie() {
  495. if (document.cookie.length > 0 && this.$cookies.get('can-use-cookie')) {
  496. const arr = document.cookie.split('; ')
  497. for (let i = 0; i < arr.length; i++) {
  498. const arr2 = arr[i].split('=')
  499. if (arr2[0] === 'email') {
  500. this.loginData.email = arr2[1]
  501. } else if (arr2[0] === 'password') {
  502. this.loginData.password = Base64.decode(arr2[1])
  503. this.isRemember = true
  504. }
  505. }
  506. }
  507. },
  508. toSignatureLink() {
  509. const link = this.config.configInfo?.link || ''
  510. if (link) {
  511. window.open(link, '_blank')
  512. }
  513. },
  514. initGoogleMapAutoComplete() {
  515. // console.log(window.google.maps, 'maps')
  516. // https://developers.google.com/maps/documentation/javascript/place-autocomplete
  517. this.autoComplete = new window.google.maps.places.Autocomplete(
  518. document.getElementById('address_input'),
  519. {
  520. types: ['address'],
  521. componentRestrictions: {
  522. country: ['AU'],
  523. },
  524. // 如果不知道需要什么字段, 可以查文档, 或给'all'或不传fields参数, 然后从返回结果里面找出需要数据的字段名, 填到这里
  525. fields: [
  526. 'address_components',
  527. 'place_id',
  528. 'name',
  529. 'formatted_address',
  530. ],
  531. }
  532. )
  533. this.autoComplete.addListener('place_changed', this.onPlaceChange)
  534. },
  535. onPlaceChange() {
  536. const data = this.autoComplete.getPlace()
  537. if (Array.isArray(data.address_components)) {
  538. data.address_components.forEach(i => {
  539. if (i.types.includes('postal_code')) {
  540. this.$set(this.registData, 'postcode', i.long_name || i.short_name)
  541. }
  542. if (i.types.includes('administrative_area_level_1')) {
  543. this.$set(this.registData, 'state', i.short_name || i.long_name)
  544. }
  545. })
  546. }
  547. this.selectAddress = data.name || this.registData.address
  548. },
  549. onInputChange(value, attr) {
  550. if (attr === 'address') {
  551. // 地址输入变更时, 清空这个值, 避免手动输入地址又不选的时候, 这个变量保存了上一次选地址时留下的数据.
  552. this.selectAddress = ''
  553. }
  554. },
  555. getAddrByLocation() {
  556. if (navigator.geolocation) {
  557. let location = {
  558. lat: 0,
  559. lng: 0,
  560. }
  561. navigator.geolocation.getCurrentPosition(
  562. position => {
  563. location = {
  564. lat: position.coords.latitude,
  565. lng: position.coords.longitude,
  566. }
  567. console.log(location, 'position')
  568. const geocoder = new window.google.maps.Geocoder()
  569. geocoder.geocode({ location }).then(response => {
  570. console.log(response, 'response')
  571. if (response.results?.length) {
  572. response.results[0].address_components.forEach(i => {
  573. if (i.types.includes('postal_code')) {
  574. this.$set(
  575. this.registData,
  576. 'postcode',
  577. i.long_name || i.short_name
  578. )
  579. }
  580. if (i.types.includes('administrative_area_level_1')) {
  581. this.$set(
  582. this.registData,
  583. 'state',
  584. i.short_name || i.long_name
  585. )
  586. }
  587. if (i.types.includes('route')) {
  588. this.$set(
  589. this.registData,
  590. 'address',
  591. i.short_name || i.long_name
  592. )
  593. }
  594. this.selectAddress = i.short_name || i.long_name
  595. })
  596. }
  597. })
  598. },
  599. error => {
  600. console.log(error, 'get location error')
  601. }
  602. )
  603. }
  604. },
  605. },
  606. }
  607. </script>
  608. <style lang="scss" scoped>
  609. ::v-deep .custom-login-form {
  610. margin: 15vh auto 0 !important;
  611. }
  612. ::v-deep .custom-register-form {
  613. margin: 6vh auto 0 !important;
  614. }
  615. :deep(.el-dialog__header) {
  616. border-radius: 3px;
  617. border-top: 4px solid #00213b;
  618. position: relative;
  619. padding: 0px 20px;
  620. .el-dialog__headerbtn {
  621. position: absolute;
  622. top: -41px;
  623. right: 0;
  624. .el-icon-close:before {
  625. content: '\e78d';
  626. color: #fff;
  627. font-size: 22px;
  628. }
  629. }
  630. }
  631. :deep(.el-dialog__body) {
  632. padding-top: 20px;
  633. padding-bottom: 17px;
  634. }
  635. :deep(.el-tabs__nav-scroll) {
  636. display: flex;
  637. justify-content: space-around;
  638. font-family: Proxima Nova;
  639. .el-tabs__nav.is-top {
  640. display: flex;
  641. justify-content: space-around;
  642. align-items: center;
  643. .el-tabs__item.is-top {
  644. font-family: Proxima Nova;
  645. font-weight: bold;
  646. color: #aaacb1;
  647. }
  648. .el-tabs__item.is-top.is-active {
  649. color: #00bfe2;
  650. }
  651. .el-tabs__active-bar.is-top {
  652. background-color: #00bfe2;
  653. }
  654. }
  655. .el-tabs__item {
  656. padding: 0px 80px;
  657. }
  658. }
  659. .el-tab-pane {
  660. .desc-content {
  661. box-sizing: border-box;
  662. word-break: break-word;
  663. width: 100%;
  664. text-align: center;
  665. font-size: 14px;
  666. line-height: 20px;
  667. color: #4a596c;
  668. padding: 0 36px;
  669. margin: 8px auto 10px;
  670. }
  671. .main-desc {
  672. font-size: 22px;
  673. line-height: 30px;
  674. padding-left: 5px;
  675. }
  676. .signature-image {
  677. margin-top: 60px;
  678. img {
  679. width: 444px;
  680. height: auto;
  681. }
  682. .pointer {
  683. cursor: pointer;
  684. }
  685. }
  686. }
  687. .rememberMe {
  688. display: flex;
  689. justify-content: space-between;
  690. font-style: italic;
  691. font-size: 12px;
  692. text-decoration: underline;
  693. .btn-forgot-pw {
  694. cursor: pointer;
  695. font-weight: 900;
  696. }
  697. }
  698. .msg-div {
  699. text-align: center;
  700. padding: 25px 0 33px;
  701. img {
  702. width: 40px;
  703. }
  704. .title {
  705. font-weight: bold;
  706. font-size: 26px;
  707. margin: 14px 0 22px;
  708. color: #00213b;
  709. }
  710. .text {
  711. font-size: 16px;
  712. margin: 0 0 20px;
  713. color: #666;
  714. }
  715. .el-button--primary {
  716. //需要更改的按钮类型 type='primary'
  717. background: #00213b !important;
  718. border-color: #00213b !important;
  719. }
  720. }
  721. ::v-deep .register-form {
  722. label {
  723. line-height: 24px;
  724. padding-bottom: 0;
  725. color: #4a596c;
  726. }
  727. .long-form-item {
  728. width: 96%;
  729. }
  730. .short-form-item {
  731. width: 47%;
  732. }
  733. .el-input__inner {
  734. border-color: #828282;
  735. }
  736. .el-button {
  737. color: #fff;
  738. background-color: #e90000;
  739. }
  740. }
  741. </style>