Pārlūkot izejas kodu

Merge branch 'feature-chaojie' into dev

username 1 gadu atpakaļ
vecāks
revīzija
972fc09658

+ 51 - 0
src/api/user.ts

@@ -0,0 +1,51 @@
+import { server } from "@/utils/server";
+
+export const getBindPhoneApi = () => {
+    return server.post<any>({
+        url: "/v1/api/user/bind_phone",
+    });
+};
+
+export const getFindPwdApi = () => {
+    return server.post<any>({
+        url: "/v1/api/user/find_pwd"
+    });
+};
+
+export const getGoogleLoginApi = () => {
+    return server.post<any>({
+        url: "/v1/api/user/google_login"
+    });
+};
+
+export const getLoginApi = (data: any) => {
+    return server.post<any>({
+        url: "/v1/api/user/login",
+        data
+    });
+};
+
+export const getLogoutApi = () => {
+    return server.post<any>({
+        url: "/v1/api/user/logout"
+    });
+};
+
+export const getRegisterApi = (data: any) => {
+    return server.post<any>({
+        url: "/v1/api/user/register",
+        data
+    });
+};
+
+export const getSendCodeApi = () => {
+    return server.get<any>({
+        url: "/v1/api/user/send_code"
+    });
+};
+
+export const getUserInfoApi = () => {
+    return server.post<any>({
+        url: "/v1/api/user/user_info"
+    });
+};

+ 10 - 9
src/app/[locale]/(ordinary)/profile/page.scss

@@ -61,14 +61,13 @@
           font-size: .12rem;
         }
       }
+
       .goto {
-        display: -webkit-box;
-        display: -ms-flexbox;
         display: flex;
-        -webkit-box-align: center;
-        -ms-flex-align: center;
         align-items: center;
+        cursor: pointer;
       }
+
       .iconfont {
         font-size: .16rem;
         font-weight: 700;
@@ -78,16 +77,14 @@
         margin-left: .04rem;
       }
     }
+
     .coin {
-      display: -webkit-box;
-      display: -ms-flexbox;
       display: flex;
-      -webkit-box-pack: justify;
-      -ms-flex-pack: justify;
       justify-content: space-between;
       padding: .07rem 0 .3rem;
       margin: 0 .18rem;
       border-top: .01rem dotted #d9a801;
+
       & > div {
         display: flex;
         -webkit-box-align: center;
@@ -96,18 +93,22 @@
         width: 50%;
         -webkit-box-sizing: border-box;
         box-sizing: border-box;
+
         &:first-child {
           border-right: .01rem dotted #d9a801;
         }
+
         .iconfont {
           margin: 0 .08rem;
           font-size: .18rem;
         }
+
         & > div {
           flex: 1;
           display: flex;
           flex-direction: column;
         }
+        
         div {
           span {
             font-size: .12rem;
@@ -185,7 +186,7 @@
             .iconfont {
               margin-right: .04rem;
               margin-left: .1rem;
-              font-size: .18rem;
+              font-size: .16rem;
             }
           }
         }

+ 12 - 6
src/app/[locale]/(ordinary)/profile/page.tsx

@@ -1,5 +1,6 @@
 "use client";
 import { FC, PropsWithChildren } from "react";
+import { useRouter } from "@/i18n";
 // import ButtonOwn from "@/components/ButtonOwn";
 import './page.scss'
 
@@ -10,6 +11,11 @@ const Profile: FC<PropsWithChildren<Props>> = (props) => {
     let amount = 50
     let amountList = [10,20,50,100,200,500,1000,5000,10000,500,1000,5000,10000]
 
+    const router = useRouter();
+    const goPage = (path = '/') => {
+        router.push(path)
+    }
+
     return (
         <div className="main">
             <div className="userContent">
@@ -21,9 +27,9 @@ const Profile: FC<PropsWithChildren<Props>> = (props) => {
                             <span className="phone">5516982013895</span>
                         </div>
                     </div>
-                    <div className="goto">
+                    <div className="goto" onClick={() => goPage('/login')}>
                         <span>Login</span>
-                        <span className="iconfont icon-to"></span>
+                        <span className="iconfont icon-xiangzuo1"></span>
                     </div>
                 </div>
                 <div className="coin">
@@ -51,8 +57,8 @@ const Profile: FC<PropsWithChildren<Props>> = (props) => {
             </div>
 
             <div className="link">
-                <span>Depósito</span>
-                <span>Sacar</span>
+                <span  onClick={() => goPage('/deposit')}>Depósito</span>
+                <span  onClick={() => goPage('/withdraw')}>Sacar</span>
                 {/* <ButtonOwn active={true}>Depósito</ButtonOwn>
                 <ButtonOwn active={true}>Sacar</ButtonOwn> */}
             </div>
@@ -65,13 +71,13 @@ const Profile: FC<PropsWithChildren<Props>> = (props) => {
                                 <span data-v-5c42ece6="" className="iconfont icon-hot"></span>
                                 <div data-v-5c42ece6=""></div>
                             </div>
-                            <div><span className="iconfont icon-to"></span></div>
+                            <div><span className="iconfont icon-xiangyou1"></span></div>
                         </li>
                     ))
                 }
             </ul>
 
-            <span className="logOut">Login</span>
+            <span className="logOut" onClick={() => goPage('/login')}>Login</span>
         </div>
     );
 };

+ 144 - 0
src/app/[locale]/(ordinary)/withdraw/page.scss

@@ -0,0 +1,144 @@
+.deposit-box {
+    width: 100%;
+    min-height: 100vh;
+    padding: 0 .14rem;
+    padding: .44rem .18rem;
+    background-color: #1f1f1f;
+    display: flex;
+    flex-direction: column;
+    .img-box {
+      width: .8rem;
+      height: .27rem;
+      margin: .12rem 0;
+      background: url('/img/pixImg.png') no-repeat center;
+      background-size: 100% 100%;
+    }
+    .btn-box {
+      margin: .1rem 0;
+      height: .3rem;
+      margin-right: .06rem;
+      width: .98rem !important;
+      -ms-flex-negative: 0;
+      flex-shrink: 0;
+      border-radius: .05rem;
+      line-height: .28rem;
+      align-items: center;
+      background: transparent;
+      border: .01px solid #ff9323;
+      border-radius: .05rem;
+      color: #868686;
+      font-size: .1rem;
+      line-height: .28rem;
+      cursor: pointer;
+      text-align: center;
+      font-family: -apple-system, BlinkMacSystemFont, Helvetica Neue, Helvetica, Segoe UI, Arial, Roboto, PingFang SC, miui, Hiragino Sans GB, Microsoft Yahei, sans-serif;
+    }
+    .amount-box {
+      width: 100%;
+      height: auto;
+      margin: .05rem 0;
+      display: flex;
+      flex-direction: column;
+      span {
+        font-size: .13rem;
+        color: #fff;
+      }
+      input {
+        font-size: .16rem;
+        width: 100%;
+        margin: .12rem 0 .04rem;
+        background-color: #494949;
+        border-radius: .05rem;
+        height: .4rem;
+        color: #868686;
+        text-indent: .14rem;
+        outline: none;
+        &::placeholder {
+          color: #868686;
+        }
+      }
+    }
+    .ul-box {
+      width: 100%;
+      height: auto;
+      -webkit-box-pack: justify;
+      -ms-flex-pack: justify;
+      justify-content: space-between;
+      flex-wrap: wrap;
+      margin-top: .09rem;
+      display: flex;
+      -webkit-box-align: center;
+      -ms-flex-align: center;
+      align-items: center;
+      li {
+        width: 30%;
+        height: .48rem;
+        border-radius: .06rem;
+        color: #fff;
+        font-size: .18rem;
+        background: #494949;
+        display: flex;
+        -webkit-box-align: center;
+        -ms-flex-align: center;
+        align-items: center;
+        -webkit-box-orient: vertical;
+        -webkit-box-direction: normal;
+        -ms-flex-direction: column;
+        flex-direction: column;
+        -webkit-box-pack: center;
+        -ms-flex-pack: center;
+        justify-content: center;
+        margin-top: .12rem;
+        position: relative;
+        overflow: hidden;
+        cursor: pointer;
+        &.active {
+          background-color: #ff6a01;
+          .hot {
+            background: url();
+            width: .33rem;
+            height: .325rem;
+            background-size: 100% 100%;
+          }
+          .amountTips {
+            color: #fff;
+          }
+        }
+        .hot {
+          display: inline-block;
+          position: absolute;
+          top: 0;
+          left: 0;
+          background: url();
+          width: .33rem;
+          height: .325rem;
+          background-size: 100% 100%;
+        }
+        .amountContent {
+          width: 100%;
+          display: flex;
+          align-items: center;
+          justify-content: center;
+          
+          .iconfont {
+            margin-right: .03rem;
+            font-size: .14rem;
+            font-family: SWISSC-BT;
+            color: #fff;
+            font-weight: bold;
+            font-style: oblique;
+          }
+        }
+        .amountTips {
+          font-size: .1rem;
+          color: #ff9500;
+        }
+      }
+    }
+    .topUp {
+      width: 100%;
+      height: auto;
+      margin: .24rem 0 .19rem;
+    }
+  }
+  

+ 27 - 0
src/app/[locale]/(ordinary)/withdraw/page.tsx

@@ -0,0 +1,27 @@
+import { FC, PropsWithChildren } from "react";
+import clsx from "clsx";
+import ButtonOwn from "@/components/ButtonOwn";
+import './page.scss'
+
+interface Props {}
+
+const Withdraw: FC<PropsWithChildren<Props>> = (props) => {
+    let amount = 50
+    let amountList = [10,20,50,100,200,500,1000,5000,10000]
+
+    return (
+        <div className="deposit-box">
+            <div className="img-box"></div>
+            <p className="btn-box" color="primary">PIX Ⅰ</p>
+            <div className="amount-box">
+                <span>Montante (BRL):</span>
+                <input type="number" placeholder="Mín. 10.00"/>
+            </div>
+            <div className="topUp">
+                <ButtonOwn active={amount>0?true:false}>Saque</ButtonOwn>
+            </div>
+        </div>
+    );
+};
+
+export default Withdraw;

+ 43 - 12
src/app/[locale]/login/component/FromCom/index.tsx

@@ -1,5 +1,5 @@
 "use client";
-import { FC, PropsWithChildren, useState } from "react";
+import { ChangeEvent, FC, PropsWithChildren, useMemo, useState } from "react";
 import clsx from "clsx";
 import Link from "next/link";
 import ButtonOwn from "@/components/ButtonOwn";
@@ -8,35 +8,66 @@ import "./style.scss";
 /**
  * @description 登录注册From表单
  * @param {string} type 使用类型
- * @param {() => void} callbackFun 回调方法
+ * @param {(params: any) => void} callbackFun 回调方法
  */
 export interface FromComProps {
     type?: string;
     text?: string;
-    callbackFun?: () => void;
+    callbackFun?: (params: any) => void;
 }
 
 const FromCom: FC<PropsWithChildren<FromComProps>> = ({type = 'login', callbackFun}) => {
-    let [visible, setVisible] = useState(false)
-
+    let [pwdVisible, setPwdVisible] = useState(false)
     const spanClassName = clsx("iconfont", {
-        "icon-kejian": visible,
-        "icon-bukejian": !visible,
+        "icon-kejian": pwdVisible,
+        "icon-bukejian": !pwdVisible,
     });
 
+    let [fromParam, setFromParam] = useState({
+        userPhone: '',
+        pwd: ''
+    })
+
+    const activeCls = useMemo(() => {
+        let { userPhone, pwd } = fromParam
+        if (userPhone && userPhone.length==11 && pwd && pwd.length>6) {
+            return true
+        }
+        return false
+    }, [fromParam]);
+
+
+    const setInputVal = (e: ChangeEvent<HTMLInputElement>, propsName: string) => {
+        setFromParam({
+            ...fromParam,
+            [propsName]: e.target.value
+        })
+    }
+
+    const verifyPwd = (e: any) => {
+        let pwd = e.target.value || '';
+        pwd.replaceAll(/[^a-zA-Z0-9_-]/g, '')
+        setFromParam({ ...fromParam, pwd })
+        console.log('fromParam', fromParam.pwd)
+    }
+
+    const submitRequest = () => {
+        activeCls && callbackFun!(fromParam)
+    }
+
     return (
         <div className="FromCom">
             <div className="phoneInput">
                 <span className="after">+55</span>
-                <input type="tel" placeholder="Número de Celular" maxLength={11}/>
+                <input type="tel" value={fromParam.userPhone} onChange={(e) => setInputVal(e, 'userPhone') } placeholder="Número de Celular" maxLength={11} />
             </div>
             <div className="passwordInput">
-                <input type={visible?'text':'password'} placeholder="Senha" />
-                <span className={spanClassName} onClick={() => setVisible(!visible)}></span>
+                <input type={pwdVisible?'text':'password'} value={fromParam.pwd} onChange={(e) => setInputVal(e, 'pwd') } onInput={verifyPwd} placeholder="Senha" maxLength={12}/>
+                <span className={spanClassName} onClick={() => setPwdVisible(!pwdVisible)}></span>
             </div>
             <div className="btnContent">
                 <div className="tips"> O número de telefone não existe. </div>
-                <ButtonOwn active={true}>{type == 'login'? 'Login' : 'Criar conta'}</ButtonOwn>
+                <ButtonOwn active={activeCls} callbackFun={submitRequest}>{type == 'login'? 'Login' : 'Criar conta'}</ButtonOwn>
             </div>
             <div className="link">
                 {
@@ -45,7 +76,7 @@ const FromCom: FC<PropsWithChildren<FromComProps>> = ({type = 'login', callbackF
                             <Link href="/br/resetPhone">Esqueci minha senha?</Link>
                             <Link href="/br/register">Criar Conta Nova</Link>
                         </>
-                    ) : <Link href="/br/login" className="active">Já tem uma conta? Log in</Link>
+                    ) : <Link href="/br/login" className="active" replace>Já tem uma conta? Log in</Link>
                 }     
             </div>
         </div>

+ 7 - 1
src/app/[locale]/login/page.tsx

@@ -5,15 +5,21 @@ import GoogleCom from "./component/GoogleCom";
 import FromCom from "./component/FromCom";
 import DomainFooter from "@/components/DomainFooter";
 import './page.scss'
+import {getLoginApi} from "@/api/user";
 
 interface Props {}
 
 const Login: FC<PropsWithChildren<Props>> = () => {
+
+    const loginRequest = async (params: any) => {
+        let data = await getLoginApi(params)
+    }
+
     return (
         <div className="login-box">
             <HeaderBack />
             <GoogleCom />
-            <FromCom />
+            <FromCom callbackFun={loginRequest}/>
             <DomainFooter />
         </div>
     );

+ 6 - 1
src/app/[locale]/register/page.tsx

@@ -5,15 +5,20 @@ import GoogleCom from "../login/component/GoogleCom";
 import FromCom from "../login/component/FromCom";
 import DomainFooter from "@/components/DomainFooter";
 import '../login/page.scss'
+import {getRegisterApi} from "@/api/user";
 
 interface Props {}
 
 const Register: FC<PropsWithChildren<Props>> = () => {
+
+    const registerRequest = async (params: any) => {
+        let data = await getRegisterApi(params)
+    }
     return (
         <div className="register-box">
             <HeaderBack />
             <GoogleCom title="Bem-vindo à 9F.COM" text="Registre-se com Google+"/>
-            <FromCom type="register"/>
+            <FromCom type="register" callbackFun={registerRequest} />
             <DomainFooter />
         </div>
     );

+ 1 - 1
src/components/Footer/index.tsx

@@ -1,6 +1,6 @@
 "use client";
 import { usePathname, useRouter} from "@/i18n";
-import { FC, PropsWithChildren, ReactNode, useEffect } from "react";
+import { FC, PropsWithChildren, ReactNode } from "react";
 import clsx from "clsx";
 import "./style.scss";
 

+ 6 - 0
src/utils/index.ts

@@ -18,3 +18,9 @@ export const setHtmlFontSize = () => {
         recalc();
     })(document, window);
 };
+
+// 密码正则 6到12位(字母,数字,下划线,减号)
+export const pwdRegex = (pwd = '') => {
+	let regex = /^[a-zA-Z0-9_-]{6,12}$/;
+	return regex.test(pwd)
+}