login.vue 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354
  1. <template>
  2. <view class="containers" style="min-height: 100vh;">
  3. <view class="header" style="width: 100vw;height: 122rpx">
  4. <image class="backFn" src="../../static/images/common/back.png" @click="$switchTab('/pages/index/index')"></image>
  5. </view>
  6. <view class="logoText">
  7. <view class="texts" style="font-weight: bold;">
  8. <view class="txt">欢迎来到</view>
  9. <view class="txt">团蜂社区</view>
  10. </view>
  11. <view class="" style="margin-top: 48rpx;margin-left: -6rpx;">
  12. <text class="appTitle">年轻的社区生活元宇宙</text>
  13. </view>
  14. </view>
  15. <view class="login-form">
  16. <tui-form ref="refLoginForm" :show-message="false">
  17. <view>
  18. <tui-input
  19. v-model="loginQuery.phone" label="+ 86" type="number" padding="26rpx 20rpx 26rpx 0"
  20. placeholder-style="color: #f3c1c4;font-size: 32rpx;" label-color="#ffffff" border-color="#EA5B1D"
  21. placeholder="请输入手机号码" background-color="transparent" :border-top="false" color="#ffffff"
  22. style="border-bottom: 2rpx solid #ffffff;"
  23. ></tui-input>
  24. </view>
  25. <view v-if="loginType === 'password'">
  26. <tui-input
  27. v-model="loginQuery.password" type="password" padding="26rpx 20rpx 26rpx 0"
  28. placeholder-style="color: #f3c1c4;font-size: 32rpx;" background-color="transparent" :border-top="false"
  29. border-color="#EA5B1D" label-color="#ffffff" placeholder="请输入密码" color="#ffffff"
  30. style="border-bottom: 2rpx solid #ffffff;"
  31. >
  32. <template #right>
  33. <text style="font-size: 30rpx;color: #dddddd;" @click="isShowResettingPasswordDialog = true">忘记密码</text>
  34. </template>
  35. </tui-input>
  36. </view>
  37. <view v-if="loginType === 'verificationCode'">
  38. <tui-input
  39. v-model="loginQuery.verificationCode" padding="26rpx 20rpx 26rpx 0"
  40. placeholder-style="color: #f3c1c4;font-size: 32rpx;" background-color="transparent" :border-top="false"
  41. border-color="#EA5B1D" label-color="#ffffff" placeholder="请输入验证码" color="#ffffff"
  42. style="border-bottom: 2rpx solid #ffffff;"
  43. >
  44. <template #right>
  45. <tui-countdown-verify
  46. v-if="loginType === 'verificationCode'" ref="refLoginVerify" width="188rpx"
  47. height="48rpx" border-width="0" text="获取验证码" :size="30"
  48. color="#dddddd"
  49. @send="handleSendVerify"
  50. ></tui-countdown-verify>
  51. </template>
  52. </tui-input>
  53. </view>
  54. </tui-form>
  55. </view>
  56. <view style="display: flex;flex-direction: column;justify-content: center;align-items: center;margin-top: 80rpx;">
  57. <view style="text-align: center;">
  58. <tui-button
  59. type="white" width="640rpx" height="82rpx" :size="38"
  60. margin="40rpx 0 0" shape="circle"
  61. style="font-weight: bold;color: #ea5b1d!important;" @click="handleLogin"
  62. >
  63. 登录
  64. </tui-button>
  65. </view>
  66. </view>
  67. <view style="margin-top: 120rpx;display: flex;flex-direction: column;align-items: center;">
  68. <view style="display: flex;justify-content: space-around;white-space: nowrap;">
  69. <view style="display: flex;flex-direction: column;align-items: center;">
  70. <view style="width: fit-content;padding: 14rpx;border: 1rpx solid #ffffff;border-radius: 48rpx;">
  71. <tui-icon name="friendadd-fill" color="#ffffff" :size="22" @click="go('/pages/login/register')"></tui-icon>
  72. </view>
  73. <view style="margin-top: 12rpx;font-size: 26rpx;color: #ffffff;">密码注册</view>
  74. </view>
  75. <view
  76. v-if="!(loginType === 'password')"
  77. style="display: flex;flex-direction: column;align-items: center;padding-left: 48rpx;"
  78. >
  79. <view style="width: fit-content;padding: 14rpx;border: 1rpx solid #ffffff;border-radius: 48rpx;">
  80. <tui-icon name="pwd" color="#ffffff" :size="22" @click="loginType = 'password'"></tui-icon>
  81. </view>
  82. <view style="margin-top: 12rpx;font-size: 26rpx;color: #ffffff;">密码登录</view>
  83. </view>
  84. <view
  85. v-if="!(loginType === 'verificationCode')"
  86. style="display: flex;flex-direction: column;align-items: center;padding-left: 48rpx;"
  87. >
  88. <view style="width: fit-content;padding: 14rpx;border: 1rpx solid #ffffff;border-radius: 48rpx;">
  89. <tui-icon name="mobile" color="#ffffff" :size="22" @click="loginType = 'verificationCode'"></tui-icon>
  90. </view>
  91. <view style="margin-top: 12rpx;font-size: 26rpx;color: #ffffff;">短信登录</view>
  92. </view>
  93. <!-- #ifdef MP-ALIPAY -->
  94. <view
  95. v-if="!(loginType === 'verificationCode')"
  96. style="display: flex;flex-direction: column;align-items: center;padding-left: 48rpx;"
  97. >
  98. <view style="width: fit-content;padding: 14rpx;border: 1rpx solid #ffffff;border-radius: 48rpx;">
  99. <tui-icon name="mobile" color="#ffffff" :size="22" @click="handleAliPayLogin"></tui-icon>
  100. </view>
  101. <view style="margin-top: 12rpx;font-size: 26rpx;color: #ffffff;">支付宝登录</view>
  102. </view>
  103. <!-- #endif -->
  104. <view
  105. v-if="(terminal === 3) || (terminal === 2)"
  106. style="display: flex;flex-direction: column;align-items: center;padding-left: 48rpx;"
  107. >
  108. <view style="width: fit-content;padding: 14rpx;border: 1rpx solid #ffffff;border-radius: 48rpx;">
  109. <tui-icon name="wechat" color="#ffffff" :size="22" @click="handleWXLogin"></tui-icon>
  110. </view>
  111. <view style="margin-top: 12rpx;font-size: 26rpx;color: #ffffff;">微信登录</view>
  112. </view>
  113. </view>
  114. </view>
  115. <tui-dialog
  116. style="position: relative;z-index: 888;" :buttons="[{ text: '取消' }, { text: '确定', color: '#586c94' }]"
  117. :show="isShowResettingPasswordDialog" title="重置密码" @click="handleResettingPassword"
  118. >
  119. <template #content>
  120. <tui-input v-model="resettingFormData.phone" label="手机号" type="number" placeholder="请输入手机号"></tui-input>
  121. <tui-input v-model="resettingFormData.password" label="密码" type="password" placeholder="请输入密码"></tui-input>
  122. <tui-input
  123. v-model="resettingFormData.passwordAgain" label="确认密码" type="password"
  124. placeholder="请再次输入密码"
  125. ></tui-input>
  126. <tui-input v-model="resettingFormData.verificationCode" label="验证码" type="number" placeholder="请输入验证码">
  127. <template #right>
  128. <tui-countdown-verify
  129. ref="refResettingPasswordVerify" width="144rpx"
  130. @send="handleSendVerifyResettingPassword"
  131. ></tui-countdown-verify>
  132. </template>
  133. </tui-input>
  134. </template>
  135. </tui-dialog>
  136. </view>
  137. </template>
  138. <script>
  139. import { mapGetters } from 'vuex'
  140. import { T_REDIRECT_TYPE, USER_ID, T_STORAGE_KEY } from '../../constant'
  141. import { getVerifyCodeApi, updatePasswordUserApi } from '../../api/anotherTFInterface'
  142. import { CHANGE_IS_IN_MINIPROGRAM } from '../../store/modules/type'
  143. import { getUrlCode } from '../../utils'
  144. export default {
  145. name: 'Login',
  146. data() {
  147. return {
  148. loginType: 'verificationCode', // password,verificationCode
  149. loginQuery: {
  150. phone: '',
  151. verificationCode: '',
  152. password: ''
  153. },
  154. resettingFormData: {
  155. phone: '',
  156. password: '',
  157. passwordAgain: '',
  158. verificationCode: ''
  159. },
  160. isShowResettingPasswordDialog: false
  161. }
  162. },
  163. onLoad(options) {
  164. if (options.to) uni.setStorageSync(T_REDIRECT_TYPE, options.to)
  165. if (options.miniProgram) {
  166. this.$store.commit(`app/${CHANGE_IS_IN_MINIPROGRAM}`, !!options.miniProgram)
  167. }
  168. },
  169. onShow() {
  170. const userId = uni.getStorageSync(USER_ID)
  171. const userInfo = uni.getStorageSync(T_STORAGE_KEY)
  172. if (userId && userInfo && userInfo.token) {
  173. uni.switchTab({
  174. url: '/pages/index/index'
  175. })
  176. } else if (this.$store.state.app.terminal === 3) {
  177. const code = getUrlCode().code
  178. if (code) this.handleWXLogin()
  179. }
  180. },
  181. computed: {
  182. ...mapGetters([ 'terminal' ])
  183. },
  184. methods: {
  185. // 获取验证码
  186. handleSendVerify() {
  187. if (!this.loginQuery.phone) {
  188. this.$refs.refLoginVerify.reset()
  189. return this.$showToast('请填写手机号')
  190. }
  191. if (!/^1[3-9]\d{9}$/.test(this.loginQuery.phone)) {
  192. this.$refs.refLoginVerify.reset()
  193. return this.$showToast('请输入正确的手机号')
  194. }
  195. getVerifyCodeApi({ phone: this.loginQuery.phone })
  196. .then((res) => {
  197. this.$refs.refLoginVerify.success()
  198. this.$showToast('发送成功,请注意查看手机短信')
  199. })
  200. .catch(() => {
  201. this.$refs.refLoginVerify.reset()
  202. })
  203. },
  204. // 点击登录
  205. handleLogin() {
  206. const validateRules = [ {
  207. name: 'phone',
  208. rule: ['required', 'isMobile'],
  209. msg: ['请输入手机号', '请输入正确的手机号']
  210. } ]
  211. let loginFilterQuery
  212. if (this.loginType === 'verificationCode') {
  213. validateRules.push({
  214. name: 'verificationCode',
  215. rule: ['required', 'isNum'],
  216. msg: [ '请输入验证码' ]
  217. })
  218. loginFilterQuery = {
  219. phone: this.loginQuery.phone,
  220. verificationCode: this.loginQuery.verificationCode
  221. }
  222. } else if (this.loginType === 'password') {
  223. validateRules.push({
  224. name: 'password',
  225. rule: ['required', 'isEnAndNo'],
  226. msg: ['请输入密码', '密码为8~20位英文和数字组合']
  227. })
  228. loginFilterQuery = {
  229. phone: this.loginQuery.phone,
  230. password: this.loginQuery.password
  231. }
  232. }
  233. this.$refs.refLoginForm
  234. .validate(this.loginQuery, validateRules)
  235. .then(() => {
  236. this.$store.dispatch('auth/phoneLoginRegisterAction', {
  237. isAfter: true,
  238. loginData: {
  239. type: 2, // 1注册,2登录
  240. ...loginFilterQuery
  241. }
  242. })
  243. })
  244. .catch((e) => {
  245. this.$showToast(JSON.stringify(e.errorMsg))
  246. })
  247. },
  248. async handleAliPayLogin() {
  249. await this.$store.dispatch('auth/aliPayLoginAction', { isAfter: true })
  250. },
  251. async handleWXLogin() {
  252. await this.$store.dispatch('auth/wxLoginAction', { isAfter: true })
  253. },
  254. handleSendVerifyResettingPassword() {
  255. if (!this.resettingFormData.phone) {
  256. this.$refs.refResettingPasswordVerify.reset()
  257. return this.$showToast('请填写手机号')
  258. }
  259. if (!/^1[3-9]\d{9}$/.test(this.resettingFormData.phone)) {
  260. this.$refs.refResettingPasswordVerify.reset()
  261. return this.$showToast('请输入正确的手机号')
  262. }
  263. getVerifyCodeApi({ phone: this.resettingFormData.phone })
  264. .then((res) => {
  265. this.$refs.refResettingPasswordVerify.success()
  266. this.$showToast('发送成功,请注意查看手机短信')
  267. })
  268. .catch(() => {
  269. this.$refs.refResettingPasswordVerify.reset()
  270. })
  271. },
  272. handleResettingPassword(e) {
  273. if (e.index === 0) { } else if (e.index === 1) {
  274. if (!this.resettingFormData.phone) return this.$showToast('请填写手机号')
  275. if (!this.resettingFormData.verificationCode) return this.$showToast('请填写验证码')
  276. if (!this.resettingFormData.password) return this.$showToast('请设置密码!')
  277. if (this.resettingFormData.password !== this.resettingFormData.passwordAgain) return this.$showToast('密码不一致')
  278. uni.showLoading({
  279. title: '操作中'
  280. })
  281. updatePasswordUserApi({ ...this.resettingFormData })
  282. .then(({ data }) => {
  283. uni.hideLoading()
  284. this.$showToast('重置成功')
  285. })
  286. .catch(() => {
  287. uni.hideLoading()
  288. })
  289. }
  290. this.resettingFormData.phone = ''
  291. this.resettingFormData.verificationCode = ''
  292. this.resettingFormData.password = ''
  293. this.resettingFormData.passwordAgain = ''
  294. this.isShowResettingPasswordDialog = false
  295. }
  296. }
  297. }
  298. </script>
  299. <style lang="less" scoped>
  300. .containers {
  301. display: flex;
  302. flex-direction: column;
  303. position: relative;
  304. box-sizing: border-box;
  305. padding-bottom: 66rpx;
  306. background-color: #EA5B1D;
  307. .backFn {
  308. padding: 30rpx 30rpx;
  309. width: 62rpx;
  310. height: 62rpx;
  311. }
  312. .logoText {
  313. width: 100vw;
  314. height: 280rpx;
  315. box-sizing: border-box;
  316. padding: 30rpx 0 0 56rpx;
  317. color: #ffffff;
  318. clear: both;
  319. .texts {
  320. /* font-style: oblique; */
  321. .txt {
  322. transform: skewX(-6deg);
  323. }
  324. line-height: 80rpx;
  325. font-size: 74rpx;
  326. letter-spacing: 6rpx;
  327. -webkit-box-reflect: below 2px -webkit-linear-gradient(top, rgba(250, 250, 250, 0), rgba(250, 250, 250, 0) 70%, rgba(255, 255, 255, 0.644));
  328. box-reflect: below 0px linear-gradient(top, rgba(250, 250, 250, 0), rgba(250, 250, 250, .05) 70%, rgba(250, 250, 250, 0.3));
  329. }
  330. }
  331. .appTitle {
  332. font-family: 思源黑体;
  333. font-size: 34rpx;
  334. font-weight: normal;
  335. line-height: 42rpx;
  336. color: rgba(255, 255, 255, 0.914);
  337. }
  338. .login-form {
  339. margin-top: 20rpx;
  340. padding: 146rpx 48rpx 10rpx 48rpx;
  341. }
  342. }
  343. </style>