\r\n
\r\n
請選擇證件 \r\n \r\n\r\n {/* {userRef.IPRegion !== \"CN\" && ( */}\r\n
\r\n
changeIDCardOrigin(\"CHN\")}\r\n >\r\n \r\n
\r\n
\r\n
中國內地身份證 \r\n
\r\n \r\n
\r\n {/* )} */}\r\n\r\n
\r\n {/*
*/}\r\n
changeIDCardOrigin(\"HKG\")}\r\n >\r\n \r\n
\r\n
\r\n
中國香港身份證 \r\n
\r\n \r\n
\r\n\r\n \r\n
\r\n
changeIDCardOrigin(\"OTHER\")}\r\n >\r\n \r\n
\r\n
\r\n
其他國家及地區證件 \r\n
\r\n \r\n
\r\n \r\n\r\n {/*
\r\n changeIDCardOrigin(\"OTHER\")}\r\n >\r\n 其他國家證件 \r\n \r\n
*/}\r\n
\r\n \r\n );\r\n}\r\n\r\nexport default UserClassification;\r\n","export default \"\"","import React from \"react\";\r\n\r\nexport const PrettyPrintJson = ({ data }) => {\r\n return (
{JSON.stringify(data, null, 2)} );\r\n};\r\n\r\nexport const ReqFieldIcon = () => {\r\n return (
* )\r\n};\r\n\r\nexport const FBLabel = (props) => {\r\n return (
{props.text} {props.required && } {props.children} )\r\n};\r\n\r\nexport const PropIsExist = (obj, PropName) => PropName && PropName.split('.').reduce((a, b) => (a || {})[b], obj) !== undefined;\r\n\r\n\r\nexport const cssFormInput = (ErrMsg) => {\r\n return 'form-control' + (ErrMsg ? ' is-invalid' : '');\r\n};\r\n\r\nexport const InvalidFieldStyle = (ErrorFieldName) => ErrorFieldName ? \"is-invalid\" : \"\";\r\n\r\nexport const GetIDCardOriginName = (Code) => {\r\n switch (Code?.toUpperCase() ?? \"\") {\r\n case \"HKG\": return \"中國香港身份證\";\r\n case \"CHN\": return \"中國內地身份證\";\r\n case \"OTHER\": return \"其他國家證件\";\r\n default:\r\n return \"\";\r\n }\r\n};\r\n","import React, { useState, useRef, useEffect } from \"react\";\r\nimport { library } from '@fortawesome/fontawesome-svg-core';\r\nimport { FontAwesomeIcon } from '@fortawesome/react-fontawesome'\r\nimport { faCircle, faSquare, faTrashAlt, faSpinner, faCheckCircle } from '@fortawesome/free-solid-svg-icons'\r\nimport { faTimesCircle } from '@fortawesome/free-regular-svg-icons'\r\nimport AOSService from \"../../services/AOSServices\";\r\nimport { GetIDCardOriginName } from '../../services/Utils';\r\nimport Compress from \"compress.js\";\r\nimport * as Sentry from \"@sentry/react\";\r\n\r\nimport \"./IDCardUploader.css\";\r\nimport back from '../../assets/IDCard_back.png';\r\nimport front from '../../assets/IDCard_front.png';\r\nimport sam1 from '../../assets/sample_1.png';\r\nimport sam2 from '../../assets/sample_2.png';\r\nimport sam3 from '../../assets/sample_3.png';\r\nimport sam4 from '../../assets/sample_4.png';\r\n\r\nimport { setLocale } from \"yup\";\r\n\r\nconst IDCardUploader = React.memo(({ UserRef, onIDCardReady }) => {\r\n library.add(faCircle, faSquare, faTrashAlt, faSpinner, faCheckCircle, faTimesCircle);\r\n\r\n const DocStatus = useRef({ Front: false, Back: false });\r\n\r\n const FileChangeHandler = ({ ImgName, Status }) => {\r\n if (ImgName.toLowerCase() === \"front\") {\r\n DocStatus.current.Front = Status;\r\n } else if (ImgName.toLowerCase() === \"back\") {\r\n DocStatus.current.Back = Status;\r\n }\r\n onIDCardReady(DocStatus.current.Front && DocStatus.current.Back);\r\n };\r\n\r\n const IDCardUploadButton = ({ UserRef, Side, BGImage, Title, onFileChanged }) => {\r\n const compress = new Compress();\r\n const initialRender = useRef(true);\r\n const [uploadProcess, setUploadProcess] = useState(0);\r\n const [file, setFile] = useState(\"\");\r\n const [loading, setLoading] = useState(false);\r\n const inputFileRef = useRef();\r\n\r\n useEffect(() => {\r\n if (initialRender.current) {\r\n //console.log(`${Side} file or IDCardImage changed`, \"useEffect\");\r\n\r\n if (!file && UserRef.IDCardOrigin && UserRef.RefCode) {\r\n AOSService.GetIDCard(UserRef.IDCardOrigin, UserRef.RefCode, Side)\r\n .then(d => {\r\n const base64 = btoa(\r\n new Uint8Array(d).reduce(\r\n (data, byte) => data + String.fromCharCode(byte),\r\n '',\r\n ),\r\n );\r\n if (base64) {\r\n onFileChanged({ ImgName: Side, Status: true });\r\n setFile(\"data:;base64,\" + base64);\r\n }\r\n });\r\n }\r\n initialRender.current = false;\r\n }\r\n\r\n return () => {\r\n //console.log(\"unmount IDCardUploadButton\", \"useEffect\")\r\n if (file) URL.revokeObjectURL(file);\r\n };\r\n });\r\n\r\n const showFileDialog = (event) => {\r\n inputFileRef.current.click();\r\n };\r\n\r\n const deleteImage = () => {\r\n onFileChanged({ ImgName: Side, Status: false });\r\n setFile(null);\r\n };\r\n\r\n const handleChange = (event) => {\r\n const fileObject = event.target.files[0];\r\n if (!fileObject) return;\r\n renderImage([fileObject]);\r\n };\r\n\r\n const renderImage = (fileObject) => {\r\n compress.compress(fileObject, {\r\n size: 2, // the max size in MB, defaults to 2MB\r\n quality: .90, // the quality of the image, max is 1,\r\n maxWidth: 1920, // the max width of the output image, defaults to 1920px\r\n maxHeight: 1920, // the max height of the output image, defaults to 1920px\r\n resize: true, // defaults to true, set false if you do not want to resize the image width and height\r\n rotate: false, // See the rotation section below\r\n }).then((data) => {\r\n // returns an array of compressed images\r\n //console.log(data[0], \"compress aynce\");\r\n const img = data[0];\r\n const base64str = img.data;\r\n const imgExt = img.ext;\r\n const resizedFile = Compress.convertBase64ToFile(base64str, imgExt);\r\n resizedFile.fileName = img.alt;\r\n setFile(URL.createObjectURL(resizedFile));\r\n\r\n setLoading(true);\r\n AOSService.UploadID(UserRef.IDCardOrigin, UserRef.RefCode, Side, resizedFile,\r\n (event) => {\r\n setUploadProcess(Math.round((100 * event.loaded) / event.total));\r\n }).then(req => {\r\n if (req && !req.Result) {\r\n switch (req.Info) {\r\n case 1:\r\n alert(`上傳失敗檔案格式不支援,請重新上載`);\r\n break;\r\n case 2:\r\n alert(`身份證號已經註冊開戶`);\r\n break;\r\n default:\r\n alert(`上傳失敗或證件文字不清晰,請重新上載`);\r\n }\r\n deleteImage();\r\n } else {\r\n onFileChanged({ ImgName: Side, Status: true });\r\n }\r\n\r\n setLoading(false);\r\n }).catch(error => {\r\n Sentry.captureException({\r\n errorName: 'AOSService.UploadID',\r\n message: error,\r\n });\r\n });\r\n }).catch(error => {\r\n Sentry.captureException({\r\n errorName: 'compress',\r\n message: error,\r\n });\r\n });\r\n\r\n };\r\n\r\n const LoadingPanel = () => {\r\n return (\r\n
\r\n );\r\n }\r\n\r\n return (\r\n <>\r\n {file ? (\r\n
\r\n {loading && (
)}\r\n
\r\n
\r\n
\r\n \r\n \r\n \r\n {uploadProcess > 0 && uploadProcess < 100 && (\r\n
\r\n \r\n {uploadProcess}%\r\n \r\n \r\n )}\r\n
\r\n
\r\n ) : (\r\n
\r\n
\r\n
\r\n
{Title}
\r\n
\r\n
\r\n
\r\n )}\r\n >\r\n );\r\n };\r\n\r\n return (\r\n
\r\n
\r\n
上傳證件 ({GetIDCardOriginName(UserRef.IDCardOrigin)}) \r\n \r\n
\r\n \r\n
\r\n\r\n
\r\n \r\n
\r\n\r\n
\r\n\r\n
\r\n
\r\n
\r\n \r\n 正确示例\r\n
\r\n
\r\n
\r\n
\r\n
\r\n \r\n 边角缺失\r\n
\r\n
\r\n
\r\n
\r\n
\r\n \r\n 信息模糊\r\n
\r\n
\r\n
\r\n
\r\n
\r\n \r\n 闪光强烈\r\n
\r\n
\r\n\r\n\r\n
\r\n {/* {docIDCardFront && (
{docIDCardFront.length} )} */}\r\n
\r\n 1. 拍摄时请将证件平放,手机横向拍摄; \r\n 2. 确保身份证 边框完整、文字清晰可见; \r\n 3. 若点击上传无反应,请先确认是否开启相机权限。\r\n
\r\n\r\n
\r\n
\r\n );\r\n});\r\n\r\nexport default IDCardUploader;","export default \"\"","export default \"\"","export default \"\"","export default \"\"","export default \"\"","export default \"\"","import styled, { css } from 'styled-components'\r\n\r\nexport const PreviewImgWrapper = styled.div`\r\n margin:0.25rem;\r\n padding:auto;\r\n height:80px;\r\n width:80px;\r\n background-color: #fff;\r\n border: 1px solid #dee2e6;\r\n border-radius: 0.25rem;\r\n position: relative;\r\n\r\n & span.Trash {\r\n position: absolute;\r\n right: -8px;\r\n top: -5px;\r\n }\r\n`;\r\n\r\nexport const PreviewImg = styled.img`\r\n padding: 0.25rem;\r\n max-width: 100%;\r\n height: auto;\r\n`;\r\n\r\nexport const DocUploadButton = styled.div`\r\n display:block;\r\n height:100%;\r\n\r\n & input {\r\n position: absolute;\r\n top: 0;\r\n left: 0;\r\n right: 0;\r\n bottom: 0;\r\n width: 1px;\r\n font-size: 0;\r\n opacity: 0;\r\n }\r\n\r\n ${props =>\r\n props.disabled &&\r\n css`\r\n background-color: #ececec;\r\n `};\r\n`;","import React, { Component } from \"react\";\r\nimport AOSServices from \"../../services/AOSServices\";\r\nimport { PreviewImgWrapper, PreviewImg, DocUploadButton } from \"./DocUploaderStyle\";\r\n\r\nimport { library } from '@fortawesome/fontawesome-svg-core';\r\nimport { FontAwesomeIcon } from '@fortawesome/react-fontawesome'\r\nimport { faCircle, faSquare, faTrashAlt, faPlusCircle, faSpinner, faCheckCircle } from '@fortawesome/free-solid-svg-icons'\r\nimport { faTimesCircle } from '@fortawesome/free-regular-svg-icons'\r\nimport Compress from \"compress.js\";\r\n\r\nexport default class DocUploader extends Component {\r\n constructor(props) {\r\n super(props);\r\n\r\n this.selectFiles = this.selectFiles.bind(this);\r\n this.upload = this.upload.bind(this);\r\n this.uploadImages = this.uploadImages.bind(this);\r\n this.SelectFilesRef = this.SelectFilesRef.bind(this);\r\n this.DeleteFile = this.DeleteFile.bind(this);\r\n this.inputFilesRef = React.createRef();\r\n this.compress = new Compress();\r\n\r\n library.add(faCircle, faSquare, faTrashAlt, faSpinner, faCheckCircle, faTimesCircle);\r\n\r\n this.state = {\r\n selectedFiles: undefined,\r\n previewImages: [],\r\n progressInfos: [],\r\n message: [],\r\n\r\n imageInfos: [],\r\n processCount: 0\r\n };\r\n }\r\n\r\n componentDidMount() {\r\n AOSServices.GetDocList(this.props.RefCode, this.props.AType).then((response) => {\r\n this.setState({\r\n imageInfos: response,\r\n });\r\n });\r\n }\r\n\r\n selectFiles(event) {\r\n const files = [...event.target.files]\r\n let images = this.state.previewImages ?? [];\r\n let resizedImgFiles = this.state.selectedFiles ?? [];\r\n\r\n this.compress.compress(files, {\r\n size: 1, // the max size in MB, defaults to 2MB\r\n quality: .90, // the quality of the image, max is 1,\r\n maxWidth: 3840, // the max width of the output image, defaults to 1920px\r\n maxHeight: 3840, // the max height of the output image, defaults to 1920px\r\n resize: true, // defaults to true, set false if you do not want to resize the image width and height\r\n rotate: false, // See the rotation section below\r\n }).then((data) => {\r\n // returns an array of compressed images\r\n\r\n for (let i = 0; i < data.length; i++) {\r\n const img = data[i];\r\n const base64str = img.data;\r\n const imgExt = img.ext;\r\n const resizedFile = Compress.convertBase64ToFile(base64str, imgExt);\r\n resizedFile.name = resizedFile.fileName = img.alt;\r\n\r\n resizedImgFiles.push(resizedFile);\r\n images.push(URL.createObjectURL(resizedFile))\r\n }\r\n\r\n const { progressInfos, message } = this.state;\r\n this.setState({\r\n progressInfos: progressInfos,\r\n message: message,\r\n selectedFiles: resizedImgFiles,\r\n previewImages: images\r\n });\r\n }).then(() => {\r\n // upload all files\r\n this.uploadImages();\r\n });\r\n }\r\n\r\n upload(idx, file) {\r\n let _progressInfos = [...this.state.progressInfos];\r\n\r\n this.setState((p) => { return { processCount: p.processCount + 1 } });\r\n AOSServices.UploadDoc(this.props.RefCode, this.props.AType, file, (event) => {\r\n _progressInfos[idx].percentage = Math.round((100 * event.loaded) / event.total);\r\n this.setState({\r\n progressInfos: _progressInfos,\r\n })\r\n })\r\n .then((r) => {\r\n this.setState((prev) => {\r\n let nextMessage = [...prev.message, \"Uploaded the image successfully: \" + file.fileName];\r\n // let _selectedFiles = prev.selectedFiles;\r\n // let removeIdx = _selectedFiles.indexOf(w => w.fileName === file.fileName);\r\n // if (removeIdx >= 0) {\r\n // _selectedFiles = _selectedFiles.splic(removeIdx, 1);\r\n // console.log(\"remove\");\r\n // }\r\n\r\n return {\r\n message: nextMessage,\r\n //selectedFiles: _selectedFiles\r\n processCount: prev.processCount - 1\r\n };\r\n });\r\n\r\n if (r && !r.Result) {\r\n throw 'Could not upload the image';\r\n }\r\n\r\n //return AOSServices.GetDocList(this.props.RefCode, this.props.AType);\r\n })\r\n .then(() => {\r\n let pc = this.state.processCount;\r\n if (pc === 0) {\r\n this.setState((prev) => {\r\n return {\r\n selectedFiles: [],\r\n previewImages: [],\r\n };\r\n });\r\n\r\n AOSServices.GetDocList(this.props.RefCode, this.props.AType)\r\n .then((files) => {\r\n this.setState({\r\n imageInfos: files,\r\n });\r\n });\r\n }\r\n })\r\n // .then((files) => {\r\n // this.setState({\r\n // imageInfos: files,\r\n // });\r\n // })\r\n .catch(() => {\r\n _progressInfos[idx].percentage = 0;\r\n this.setState((prev) => {\r\n let nextMessage = [...prev.message, \"Could not upload the image: \" + file.fileName];\r\n return {\r\n progressInfos: _progressInfos,\r\n message: nextMessage\r\n };\r\n });\r\n });\r\n };\r\n\r\n DeleteFile = (DocID) => {\r\n AOSServices.DeleteDoc(this.props.RefCode, DocID)\r\n .then((r) => {\r\n if (r && r.Result) {\r\n AOSServices.GetDocList(this.props.RefCode, this.props.AType)\r\n .then((files) => {\r\n this.setState({\r\n imageInfos: files,\r\n });\r\n });\r\n\r\n }\r\n });\r\n };\r\n\r\n SelectFilesRef = (event) => {\r\n this.inputFilesRef.current.click();\r\n };\r\n\r\n uploadImages() {\r\n const selectedFiles = this.state.selectedFiles;\r\n let _progressInfos = [];\r\n\r\n for (let i = 0; i < selectedFiles.length; i++) {\r\n _progressInfos.push({ percentage: 0, fileName: selectedFiles[i].fileName });\r\n }\r\n\r\n this.setState(\r\n {\r\n progressInfos: _progressInfos,\r\n message: [],\r\n },\r\n () => {\r\n for (let i = 0; i < selectedFiles.length; i++) {\r\n this.upload(i, selectedFiles[i]);\r\n }\r\n }\r\n );\r\n }\r\n\r\n render() {\r\n const { selectedFiles, previewImages, progressInfos, message, imageInfos } = this.state;\r\n\r\n return (\r\n
\r\n {/*
\r\n
\r\n \r\n \r\n \r\n
\r\n\r\n
\r\n \r\n Upload\r\n \r\n
\r\n
*/}\r\n\r\n {/* {progressInfos &&\r\n progressInfos.map((progressInfo, index) => (\r\n
\r\n
{progressInfo.fileName} \r\n
\r\n
\r\n {progressInfo.percentage}%\r\n
\r\n
\r\n
\r\n ))} */}\r\n {/* {this.state.processCount},\r\n {this.state.selectedFiles?.length} */}\r\n {previewImages && (\r\n
\r\n\r\n
\r\n 0}>\r\n \r\n 0 ? \"rgb(240, 200, 200)\" : \"rgb(204, 0, 16)\"}\r\n transform=\"shrink-1\" />\r\n \r\n 0} type=\"file\" ref={this.inputFilesRef} multiple accept=\"image/*\" onChange={this.selectFiles} />\r\n \r\n \r\n\r\n {previewImages.map((img, i) => {\r\n return (\r\n
\r\n \r\n\r\n {progressInfos[i] && (\r\n \r\n
\r\n {progressInfos[i].percentage}%\r\n
\r\n
\r\n )}\r\n \r\n )\r\n })}\r\n\r\n {imageInfos &&\r\n imageInfos.map((img, index) => (\r\n
\r\n \r\n\r\n this.DeleteFile(img.ID)}>\r\n \r\n \r\n \r\n \r\n ))}\r\n
\r\n )}\r\n\r\n {/* {message.length > 0 && (\r\n
\r\n
\r\n {message.map((item, i) => {\r\n return {item} ;\r\n })}\r\n \r\n
\r\n )} */}\r\n\r\n\r\n {/*
\r\n
List of Files
\r\n
\r\n {imageInfos &&\r\n imageInfos.map((img, index) => (\r\n \r\n \r\n \r\n ))}\r\n \r\n
*/}\r\n\r\n
\r\n );\r\n }\r\n}","import React, { useState, useEffect, useCallback } from \"react\";\r\nimport { useUserSettings } from '../UserSettingsProvider';\r\nimport StepContainer from '../StepContainer/StepContainer';\r\nimport IDCardUploader from '../IDCardUploader/IDCardUploader';\r\nimport AOSServices from \"../../services/AOSServices\";\r\nimport { FBLabel, InvalidFieldStyle, PrettyPrintJson } from '../../services/Utils';\r\nimport DocUploader from '../DocUploader/DocUploader';\r\n\r\nimport { useForm, Controller } from \"react-hook-form\";\r\nimport { yupResolver } from '@hookform/resolvers/yup';\r\nimport * as Yup from 'yup';\r\n\r\nimport { subMonths, format } from 'date-fns';\r\nimport zh from 'date-fns/locale/zh-CN';\r\nimport ReactDatePicker, { registerLocale } from \"react-datepicker\";\r\nimport \"react-datepicker/dist/react-datepicker.css\";\r\n\r\nimport \"./PersonalInfo.css\";\r\n\r\n\r\nexport default function PersonalInfo(props) {\r\n\r\n // for date picker\r\n registerLocale('zh', zh);\r\n\r\n const { userRef } = useUserSettings();\r\n const [IDCardReady, setIDCardReady] = useState(false);\r\n const [RevisedInfo, setRevisedInfo] = useState(false);\r\n\r\n const validationSchema = Yup.object().shape({\r\n OCROrderNo: Yup.string().nullable(),\r\n ChnName: Yup.string().nullable().required(\"請輸入你的中文姓名\"),\r\n EngName: Yup.string().nullable().required(\"請輸入你的姓名拼音\"),\r\n HKID: Yup.string().nullable().required(\"請輸入你的證件號碼\"),\r\n Birthday: Yup.date().nullable().required(\"請輸入你的出生日期\"),\r\n Gender: Yup.string().nullable().required(\"請選擇你的稱呼\"),\r\n IsSameAddress: Yup.boolean().nullable(),\r\n ResidentAddress1: Yup.string().nullable().required(\"請輸入你的住宅地址\"),\r\n CommunicateAddress1: Yup.string().nullable()\r\n .when('IsSameAddress', (IsSameAddress, schema) => {\r\n return !IsSameAddress ? schema.required(\"請輸入你的通訊地址\") : schema.notRequired();\r\n }),\r\n Revised: Yup.boolean(),\r\n Region: Yup.string().nullable(),\r\n });\r\n\r\n const { register, trigger, watch, setValue, getValues, reset, control, formState: { isValid, errors }, clearErrors } = useForm({\r\n mode: \"all\",\r\n shouldFocusError: true,\r\n resolver: yupResolver(validationSchema),\r\n defaultValues: {\r\n IsSameAddress: true,\r\n Revised: (userRef.IDCardOrigin === \"OTHER\" ? true : false)\r\n }\r\n });\r\n\r\n const watchAllFields = watch();\r\n\r\n const IDCardReadyhandler = useCallback((val) => {\r\n if (val) {\r\n AOSServices.GetInfo(userRef.RefCode, userRef.IDCardOrigin)\r\n .then((resp) => {\r\n //console.log(`resp:` + resp);\r\n if (resp && resp.Result) {\r\n //console.log(`restore info`);\r\n //if (resp.Info.HKID != null && resp.Info.ChnName != null) {\r\n reset({\r\n\r\n Birthday: new Date(resp.Info.Birthday),\r\n ChnName: resp.Info.ChnName,\r\n CommunicateAddress1: resp.Info.CommunicateAddress1,\r\n CommunicateAddress2: resp.Info.CommunicateAddress2,\r\n CommunicateAddress3: resp.Info.CommunicateAddress3,\r\n EngName: resp.Info.EngName,\r\n Gender: resp.Info.Gender,\r\n HKID: resp.Info.HKID,\r\n IsSameAddress: resp.Info.IsSameAddress,\r\n OCROrderNo: resp.Info.OCROrderNo,\r\n RefID: resp.Info.RefID,\r\n Region: resp.Info.Region,\r\n ResidentAddress1: resp.Info.ResidentAddress1,\r\n ResidentAddress2: resp.Info.ResidentAddress2,\r\n ResidentAddress3: resp.Info.ResidentAddress3,\r\n Revised: resp.Info.Revised,\r\n });\r\n //}\r\n\r\n setRevisedInfo((userRef.IDCardOrigin === \"OTHER\") ? true : resp.Info.Revised);\r\n //setRevisedInfo(resp.Info.Revised);\r\n }\r\n }).finally(() => setIDCardReady(val));\r\n } else {\r\n setIDCardReady(val);\r\n }\r\n }, []);\r\n\r\n\r\n\r\n //const manualAdj = watchAllFields.PInfo.Revised;\r\n const EnableManualAdjust = () => {\r\n setValue(\"Revised\", true);\r\n setRevisedInfo(true);\r\n };\r\n\r\n const goNext = () => {\r\n if (!isValid) {\r\n trigger(undefined, { shouldFocus: true });\r\n return new Promise(resolve => resolve(false));\r\n } else {\r\n const { Birthday, ...PData } = getValues();\r\n return new Promise(resolve => {\r\n AOSServices.SavePersonalInfo(userRef.RefCode, {\r\n \"Birthday\": format(watchAllFields.Birthday, \"yyyy-MM-dd\"), ...PData\r\n }, \"SettlementInfo\")\r\n .then(res => {\r\n if (!res || !res.Result) {\r\n alert(\"提交資料失敗,請重新再試\")\r\n }\r\n resolve(res.Result);\r\n });\r\n });\r\n }\r\n };\r\n\r\n\r\n\r\n // useEffect(() => {\r\n // // load initial data\r\n // })\r\n\r\n\r\n const ManualAdjustButton = () => {\r\n return (userRef.IDCardOrigin !== \"OTHER\") && !watchAllFields.Revised && (
\r\n
\r\n 如上述個人信息不正確或地址與身份證不同,請 按此修改 \r\n
\r\n
);\r\n };\r\n\r\n return (\r\n
\r\n\r\n {/* */}\r\n\r\n \r\n\r\n {IDCardReady && (\r\n \r\n )}\r\n\r\n
\r\n\r\n\r\n\r\n \r\n );\r\n}","import React from \"react\";\r\nimport { useId, nextId } from \"react-id-generator\";\r\nimport styled from 'styled-components'\r\n\r\nconst Wrapper = styled.div`\r\n display: inline-flex;\r\n position: relative;\r\n padding: 4px;\r\n margin: 0px;\r\n -webkit-user-select: none;\r\n -moz-user-select: none;\r\n -ms-user-select: none;\r\n user-select: none;\r\n\r\n .checkmark {\r\n position: absolute;\r\n top: 23px;\r\n left: 16px;\r\n height: 20px;\r\n width: 20px;\r\n background-color: #ced4da;\r\n border-radius: 50%;\r\n }\r\n\r\n & input:checked ~ label > .checkmark:after {\r\n display: block;\r\n }\r\n\r\n & input:checked ~ label > .checkmark {\r\n background-color: rgb(204,0,16);\r\n }\r\n\r\n &:hover input:enabled ~ label {\r\n color: rgb(204,0,16);\r\n border: 1px solid rgb(204,0,16);\r\n }\r\n\r\n .checkmark:after {\r\n content: \"\";\r\n position: absolute;\r\n display: none;\r\n left: 6px;\r\n top: 0px;\r\n width: 8px;\r\n height: 16px;\r\n border: solid white;\r\n border-width: 0 3px 3px 0;\r\n -webkit-transform: rotate(45deg);\r\n -ms-transform: rotate(45deg);\r\n transform: rotate(45deg);\r\n }\r\n`;\r\n\r\nconst Label = styled.label`\r\n display: inline-grid;\r\n width: 100%;\r\n cursor: pointer;\r\n padding: 16px !important;\r\n padding-left: 40px !important;\r\n border: 1px solid #ced4da;\r\n border-radius: 30px;\r\n`;\r\n\r\nconst Input = styled.input`\r\n position: absolute;\r\n opacity: 0;\r\n cursor: pointer;\r\n height: 0;\r\n width: 0;\r\n\r\n &:checked + label {\r\n color: rgb(204,0,16);\r\n border: 1px solid rgb(204,0,16);\r\n }\r\n`;\r\n\r\n\r\nexport const FFGRadio = ({ register, name, value, className, checked, children, ...rest }) => {\r\n //export const FFGRadio = ({ name, value, defaultValue, onChange, ...rest }) => {\r\n //export const FFGRadio = React.forwardRef(\r\n // ({ name, value, onChange, defaultValue, ...rest }, forwardedRef) => {\r\n const [htmlId] = useId();\r\n\r\n // const toggleCheck = (e) => {\r\n // console.log('toggle');\r\n // const { value } = e.currentTarget;\r\n // console.log(e.currentTarget, \"currentTarget\");\r\n // onChange(value);\r\n // };\r\n\r\n // React.useEffect(() => {\r\n // console.log('useEffect');\r\n // if(defaultValue) {\r\n // console.log('checked');\r\n // onChange(value);\r\n // }\r\n // }, []);\r\n\r\n return (\r\n
\r\n \r\n \r\n {children}\r\n \r\n \r\n \r\n );\r\n};\r\n\r\n\r\nexport const FFGCheckbox = ({register, className, children, checked, ...rest}) => {\r\n const htmlId = useId();\r\n\r\n return (\r\n
\r\n \r\n \r\n {children}\r\n \r\n \r\n \r\n );\r\n};\r\n\r\n\r\nexport const RadioButtonList = props => {\r\n const { Items, defaultValue, className, name, onChange, setValueAs } = props;\r\n const [radioState, setRadioState] = React.useState();\r\n const htmlId = useId(Items.length);\r\n\r\n React.useEffect(() => {\r\n setRadioState(defaultValue);\r\n //console.log(\"radioState:\", radioState, \"name:\", name);\r\n }, [defaultValue])\r\n\r\n return (<>\r\n {\r\n Items.map((item, idx) => {\r\n return (\r\n
\r\n {\r\n setRadioState(e.target.value);\r\n if (setValueAs) {\r\n //console.log(setValueAs(e.target.value), \"setValueAs\");\r\n onChange(setValueAs(e.target.value));\r\n } else {\r\n //console.log(e.target.value, \"default\");\r\n onChange(e.target.value);\r\n }\r\n }}\r\n />\r\n \r\n {item.label}\r\n \r\n \r\n \r\n );\r\n }\r\n )\r\n }\r\n >);\r\n};","import React, { useState, useEffect, useCallback } from \"react\";\r\nimport { useUserSettings } from '../UserSettingsProvider';\r\nimport StepContainer from '../StepContainer/StepContainer';\r\nimport AOSServices from \"../../services/AOSServices\";\r\nimport { FBLabel, InvalidFieldStyle, PrettyPrintJson, GetIDCardOriginName } from '../../services/Utils';\r\nimport { RadioButtonList } from \"../FFGCheckbox/FFGCheckbox\";\r\n\r\nimport { useForm, Controller } from \"react-hook-form\";\r\nimport { yupResolver } from '@hookform/resolvers/yup';\r\nimport * as Yup from 'yup';\r\nimport Select, { components, createFilter } from 'react-select';\r\n\r\n\r\nconst SettlementInfo = (props) => {\r\n const { userRef, setUserRef } = useUserSettings();\r\n const [BankList, setBankList] = useState([]);\r\n const [CommonBankList, setCommonBankList] = useState([]);\r\n\r\n const validationSchema = Yup.object().shape({\r\n\r\n Email: Yup.string().nullable().email(\"你的電子郵箱格式不正確\").required(\"請輸入你的電子郵箱\"),\r\n HasHKGBankAcc: Yup.boolean().nullable(),\r\n UnionPayCardNo: Yup.string().nullable()\r\n .when('HasHKGBankAcc', (HasHKGBankAcc, schema) => {\r\n return (userRef.IDCardOrigin === \"CHN\" && userRef.IPRegion !== \"CN\" && !HasHKGBankAcc) ?\r\n schema.nullable().required(\"請輸入國內銀行卡號\") : schema.notRequired();\r\n }),\r\n UnionPayMobileNo: Yup.string().nullable()\r\n .when('HasHKGBankAcc', (HasHKGBankAcc, schema) => {\r\n return (userRef.IDCardOrigin === \"CHN\" && !HasHKGBankAcc) ?\r\n schema.required(\"請輸入國內銀行卡綁定的手機號碼\") : schema.notRequired();\r\n }),\r\n MobileCountry: Yup.string().nullable(),\r\n Mobile: Yup.string().nullable(),\r\n PhoneCountry: Yup.string().nullable(),\r\n Phone: Yup.string().nullable(),\r\n Statement: Yup.string().nullable().oneOf([\"E\", \"M\"], \"請選擇收取結單方式\"),\r\n BankCode: Yup.string().nullable()\r\n .when('HasHKGBankAcc', (HasHKGBankAcc, schema) => {\r\n return userRef.IDCardOrigin !== \"CHN\" || (userRef.IDCardOrigin === \"CHN\" && HasHKGBankAcc) ?\r\n schema.required(\"請輸入香港銀行\") : schema.notRequired();\r\n }),\r\n BankName: Yup.string().nullable(),\r\n BankNumber: Yup.string().nullable()\r\n .when('HasHKGBankAcc', (HasHKGBankAcc, schema) => {\r\n return userRef.IDCardOrigin !== \"CHN\" || (userRef.IDCardOrigin === \"CHN\" && HasHKGBankAcc) ?\r\n schema.required(\"請輸入香港銀行賬戶\") : schema.notRequired();\r\n }),\r\n AuthorizeEDDA: Yup.boolean().nullable()\r\n .when('HasHKGBankAcc', (HasHKGBankAcc, schema) => {\r\n return userRef.IDCardOrigin !== \"CHN\" || (userRef.IDCardOrigin === \"CHN\" && HasHKGBankAcc) ?\r\n schema.required(\"請選擇是否授權於帳戶建立後自動綁定EDDA\") : schema.notRequired();\r\n }),\r\n AuthorizeEDDAAmount: Yup.number().positive(\"授權金額格式不正確\").integer(\"授權金額格式不正確\").nullable()\r\n .when('AuthorizeEDDA', (AuthorizeEDDA, schema) => {\r\n return AuthorizeEDDA ? schema.required(\"請輸入授權金額\") : schema.notRequired();\r\n }),\r\n });\r\n\r\n const { register, trigger, watch, getValues, setValue, handleSubmit, reset, control, formState: { isValid, errors }, clearErrors } = useForm({\r\n mode: \"all\",\r\n resolver: yupResolver(validationSchema),\r\n defaultValues: {\r\n HasHKGBankAcc: true\r\n }\r\n });\r\n\r\n const watchAllFields = watch();\r\n\r\n useEffect(() => {\r\n AOSServices.GetSettlementInfo(userRef.RefCode)\r\n .then((resp) => {\r\n if (resp && resp.Result) {\r\n //if(userRef.IPRegion === \"CN\")\r\n reset(resp.Info);\r\n }\r\n });\r\n }, [reset]);\r\n\r\n useEffect(() => {\r\n AOSServices.GetBankList().then(resp => {\r\n if (resp) {\r\n let banks = resp.map(m => {\r\n return {\r\n value: m.BankCode,\r\n label: `(${m.BankCode}) ${m.CName || m.EName}`,\r\n CName: m.CName,\r\n EName: m.EName,\r\n }\r\n });\r\n setBankList(banks);\r\n setCommonBankList(banks.filter((x) => [\"003\",\"004\",\"012\",\"024\",\"072\"].includes(x.value)));\r\n }\r\n });\r\n }, []);\r\n\r\n //const PropIsExist = (obj, PropName) => PropName && PropName.split('.').reduce((a, b) => (a || {})[b], obj) !== undefined;\r\n\r\n // const ErrorMsg = ({ PropName }) => {\r\n // //console.log(PropIsExist(errors, PropName), \"PropIsExist\");\r\n // if (errors && PropIsExist(errors, PropName)) {\r\n // let msg = eval(`errors.${PropName}`);\r\n // //console.log(msg, \"msg\");\r\n // return (
{msg.message}
);\r\n // } else\r\n // return (<>>);\r\n // };\r\n\r\n // const cssFormInput = (ErrProp) => {\r\n // let Props = ErrProp.split('.');\r\n // let PropIsExist = Props.reduce((a, b) => (a || {})[b], errors) !== undefined;\r\n // return 'form-control' + (PropIsExist ? ' is-invalid' : '');\r\n // };\r\n\r\n // const UnionPayIdentification = () => {\r\n // return (\r\n // <>\r\n //
\r\n //
內地銀行卡認證 \r\n // 該銀行卡僅用於實名認證,與出入資金無關。 \r\n // \r\n\r\n //
\r\n //
\r\n //
\r\n //
{errors?.UnionPayCardNo?.message}
\r\n //
\r\n\r\n //
\r\n //
\r\n //
\r\n //
{errors?.UnionPayCardNo?.message}
\r\n //
\r\n // >\r\n // );\r\n // }\r\n\r\n const cssFormLabelCol = \"col-sm-3 col-xs-12 col-form-label my-0 my-md-2\";\r\n const cssFormInputCol = \"col-sm-9 col-xs-12 my-0 my-md-2\";\r\n\r\n const goNext = () => {\r\n if (!isValid) {\r\n //trigger(\"Info\", { shouldFocus: true });\r\n trigger(undefined, { shouldFocus: true });\r\n return new Promise(resolve => resolve(false));\r\n } else {\r\n return new Promise(resolve => {\r\n const data = getValues();\r\n\r\n if (data.HasHKGBankAcc == null) {\r\n data.HasHKGBankAcc = true;\r\n }\r\n\r\n AOSServices.SaveSettlementInfo(userRef.RefCode, data, \"FFGAccountInfo\")\r\n .then(res => {\r\n if (!res || !res.Result) {\r\n if (res.ErrorCode === -1)\r\n alert(\"內地銀行卡驗證失敗,請檢查姓名﹑身份證號碼﹑銀行卡號碼﹑手機號碼是否正確。\");\r\n else\r\n alert(\"提交資料失敗,請重新再試\");\r\n } else {\r\n //console.log(`watchallfields.hashkbankacc` + watchAllFields.HasHKGBankAcc);\r\n //console.log(`hashkgbankacc:` + data.HasHKGBankAcc)\r\n setUserRef({ ...userRef, HasHKGBankAcc: watchAllFields.HasHKGBankAcc})\r\n }\r\n resolve(res.Result);\r\n });\r\n });\r\n }\r\n };\r\n\r\n\r\n return (\r\n
\r\n\r\n {/* */}\r\n {/* */}\r\n\r\n <>\r\n \r\n
\r\n
信息驗證 ({GetIDCardOriginName(userRef.IDCardOrigin)}) \r\n \r\n\r\n
\r\n
聯繫方式 \r\n \r\n
\r\n\r\n \r\n\r\n\r\n {/* { formRef.current.dispatchEvent(new Event('submit', { cancelable: true, bubbles: true }))}}>Submit */}\r\n >\r\n \r\n );\r\n}\r\n\r\nexport default SettlementInfo;","import styled, { css } from 'styled-components'\r\n\r\nexport const ButtonWrapper = styled.div`\r\ndisplay: inline-flex;\r\nposition: relative;\r\npadding: 4px;\r\nmargin: 0px;\r\n-webkit-user-select: none;\r\n-moz-user-select: none;\r\n-ms-user-select: none;\r\nuser-select: none;\r\n\r\n.checkmark {\r\n position: absolute;\r\n top: 33%;\r\n left: 20px;\r\n height: 30px;\r\n width: 30px;\r\n background-color: #ced4da;\r\n border-radius: 50%;\r\n}\r\n\r\n& input:checked ~ label > .checkmark:after {\r\n display: block;\r\n}\r\n\r\n& input:checked ~ label > .checkmark {\r\n background-color: rgb(204,0,16);\r\n}\r\n\r\n&:hover input:enabled ~ label {\r\n color: rgb(204,0,16);\r\n border: 1px solid rgb(204,0,16);\r\n}\r\n\r\n.checkmark:after {\r\n content: \"\";\r\n position: absolute;\r\n display: none;\r\n left: 10px;\r\n top: 0px;\r\n width: 10px;\r\n height: 25px;\r\n border: solid rgb(255,248,31);\r\n border-width: 0px 5px 7px 0px;\r\n -webkit-transform: rotate(45deg);\r\n -ms-transform: rotate(45deg);\r\n transform: rotate(45deg);\r\n }\r\n\r\n ${props =>\r\n props.Lite &&\r\n css`\r\n .checkmark {\r\n top: 28%;\r\n left: 20px;\r\n height: 30px;\r\n width: 30px;\r\n background-color: #ced4da;\r\n border-radius: 4px;\r\n }\r\n\r\n & input:checked ~ label > .checkmark {\r\n \r\n }\r\n\r\n &:hover input:enabled ~ label {\r\n color: rgb(204,0,16);\r\n border: 0px;\r\n } \r\n\r\n .checkmark:after {\r\n border: solid rgb(255,255,255);\r\n border-width: 0px 5px 7px 0px;\r\n }\r\n `};\r\n`;\r\n\r\nexport const ButtonLabel = styled.label`\r\ndisplay: inline-grid;\r\nwidth: 100%;\r\ncursor: pointer;\r\npadding: 16px !important;\r\npadding-left: 55px !important;\r\nborder: 1px solid #ced4da;\r\nborder-radius: 35px;\r\nbackground-color: rgb(255,248,252);\r\n\r\n${props =>\r\n props.Lite &&\r\n css`\r\n border: 0px;\r\n background-color: transparent;\r\n `};\r\n`;\r\n\r\nexport const ButtonCheckbox = styled.input`\r\nposition: absolute;\r\nopacity: 0;\r\ncursor: pointer;\r\nheight: 0;\r\nwidth: 0;\r\n\r\n&:checked ~ label {\r\n color: rgb(204,0,16);\r\n border: 1px solid rgb(204,0,16);\r\n}\r\n\r\n&:disabled ~ label {\r\n color: rgb(160,160,160);\r\n cursor: not-allowed;\r\n}\r\n\r\n${props =>\r\n props.Lite &&\r\n css`\r\n &:checked ~ label {\r\n border: 0px;\r\n }\r\n `};\r\n`;\r\n\r\n","import React, { useState, useEffect, useCallback } from \"react\";\r\nimport { useUserSettings } from '../UserSettingsProvider';\r\nimport StepContainer from '../StepContainer/StepContainer';\r\nimport { FFGCheckbox, RadioButtonList } from \"../FFGCheckbox/FFGCheckbox\";\r\nimport { ButtonWrapper, ButtonLabel, ButtonCheckbox } from \"./FFGAccountStyle\";\r\nimport AOSServices from \"../../services/AOSServices\";\r\nimport { FBLabel, PrettyPrintJson, cssFormInput } from '../../services/Utils';\r\nimport { useForm, Controller } from \"react-hook-form\";\r\nimport { yupResolver } from '@hookform/resolvers/yup';\r\nimport * as Yup from 'yup';\r\nimport { array } from \"yup/lib/locale\";\r\n\r\n\r\nconst FFGAccountInfo = (props) => {\r\n const { userRef, setUserRef } = useUserSettings();\r\n\r\n const validationSchema = Yup.object().shape({\r\n IntegratedSecAC: Yup.boolean(),\r\n IntegratedFutAC: Yup.boolean(),\r\n AccTypeStockOptions: Yup.boolean(),\r\n AccTypeUSOptions: Yup.boolean(),\r\n //HasReferrer: Yup.boolean(),\r\n ReferrerName: Yup.string().nullable(),\r\n //.when('HasReferrer', (val, schema) => val ? schema.required(\"請填上相關客戶經理編號或名稱\") : schema.notRequired()),\r\n // TradingPreference: Yup.array().nullable()\r\n // .when('HasReferrer', (val, schema) => {\r\n // return !val ?\r\n // schema.of(Yup.string()).min(1, \"請選擇將會主要參與哪項交易\") : schema.nullable().notRequired();\r\n // }),\r\n DirectMarketing: Yup.string().nullable().required(\"請選擇是不否同意接收優惠及推廣資訊\"),\r\n MarketingChannelAll: Yup.boolean(),\r\n MarketingChannelEmail: Yup.boolean(),\r\n MarketingChannelMail: Yup.boolean(),\r\n MarketingChannelSMS: Yup.boolean(),\r\n MarketingChannelPhone: Yup.boolean(),\r\n }).test('CustomAccValidation', null, (obj) =>{\r\n if ( obj.IntegratedSecAC || obj.IntegratedFutAC || obj.AccTypeStockOptions || obj.AccTypeUSOptions ) {\r\n return true; // everything is fine\r\n }\r\n return new Yup.ValidationError('請選擇開戶類型', null, 'AccountTypes');\r\n });\r\n\r\n const { register, trigger, watch, getValues, setValue, setError, handleSubmit, reset, control,\r\n formState: { isValid, errors }, clearErrors } = useForm({\r\n mode: \"all\",\r\n resolver: yupResolver(validationSchema),\r\n defaultValues: {\r\n IntegratedSecAC: true,\r\n IntegratedFutAC: false,\r\n AccTypeStockOptions: false,\r\n AccTypeUSOptions: false,\r\n AccTypeOTCOptions: false,\r\n //HasReferrer: false,\r\n TradingPreference: [],\r\n DirectMarketing: \"Y\",\r\n MarketingChannelAll: true,\r\n MarketingChannelEmail: false,\r\n MarketingChannelMail: false,\r\n MarketingChannelSMS: false,\r\n MarketingChannelPhone: false,\r\n }\r\n });\r\n\r\n useEffect(() => {\r\n AOSServices.GetFFGAccountInfo(userRef.RefCode)\r\n .then((resp) => {\r\n if (resp && resp.Result) {\r\n reset(resp.Info);\r\n }\r\n });\r\n }, [reset]);\r\n\r\n\r\n const FBCheckBox = ({ register, children, className = \"col-sm-3 col-xs-12\", ...props }) => {\r\n let id = register.name;\r\n return (\r\n
\r\n \r\n \r\n {children}\r\n \r\n \r\n \r\n );\r\n }\r\n\r\n const goNext = () => {\r\n clearErrors();\r\n if (!isValid) {\r\n trigger(undefined, { shouldFocus: true });\r\n return new Promise(resolve => resolve(false));\r\n }\r\n else {\r\n const data = getValues();\r\n return new Promise(resolve => {\r\n if (!data.IntegratedSecAC && !data.IntegratedFutAC) {\r\n alert(\"請選擇開戶類型\");\r\n resolve(false);\r\n } else if (\r\n data.DirectMarketing === 'Y' &&\r\n !data.MarketingChannelAll &&\r\n !data.MarketingChannelEmail &&\r\n !data.MarketingChannelMail &&\r\n !data.MarketingChannelSMS &&\r\n !data.MarketingChannelPhone\r\n ) {\r\n setError(\"MarketingChannel\", {\r\n type: \"manual\",\r\n message: \"請選擇接收方式\"\r\n });\r\n resolve(false);\r\n } else {\r\n if(!Array.isArray(data.TradingPreference)){\r\n let tp = new Array();\r\n tp.concat(data.TradingPreference);\r\n data.TradingPreference = tp;\r\n //console.log(data.TradingPreference);\r\n }\r\n\r\n AOSServices.SaveFFGAccountInfo(userRef.RefCode, data, \"EmploymentInfo\")\r\n .then(res => {\r\n if (!res || !res.Result) {\r\n alert(\"提交資料失敗,請重新再試\");\r\n } else {\r\n let FBAcc = [];\r\n if (data.IntegratedSecAC) FBAcc.push(\"S\");\r\n if (data.IntegratedFutAC) FBAcc.push(\"F\");\r\n if (data.AccTypeStockOptions) FBAcc.push(\"O\");\r\n if (data.AccTypeUSOptions) FBAcc.push(\"UO\");\r\n if (data.AccTypeOTCOptions) FBAcc.push(\"OTC\");\r\n setUserRef({ ...userRef, FBAcc });\r\n }\r\n resolve(res.Result);\r\n });\r\n }\r\n });\r\n }\r\n };\r\n\r\n const watchAllFields = watch();\r\n const cssFormLabelCol = \"col-sm-3 col-xs-12 col-form-label my-0 my-md-2\";\r\n const cssFormInputCol = \"col-sm-9 col-xs-12 my-0 my-md-2\";\r\n\r\n const handleSecACChanged = (v) => {\r\n //console.log(v);\r\n if (!v.target.checked) {\r\n if (watchAllFields.AccTypeStockOptions) setValue(\"AccTypeStockOptions\", false);\r\n if (watchAllFields.AccTypeUSOptions) setValue(\"AccTypeUSOptions\", false);\r\n }\r\n };\r\n\r\n return (\r\n
\r\n\r\n \r\n
\r\n
請選擇開戶類型 \r\n \r\n
\r\n\r\n {/* */}\r\n\r\n \r\n\r\n \r\n\r\n\r\n
\r\n\r\n );\r\n};\r\n\r\nexport default FFGAccountInfo;","import styled, { css } from 'styled-components'\r\n\r\nexport const SwitchBoxWapper = styled.div`\r\ndisplay: flex;\r\nposition: relative;\r\npadding: 4px;\r\nmargin: 0px;\r\n-webkit-user-select: none;\r\n-moz-user-select: none;\r\n-ms-user-select: none;\r\nuser-select: none;\r\n\r\n& .checkmark {\r\n position: absolute;\r\n top: 5px;\r\n left: 0px;\r\n height: 20px;\r\n width: 20px;\r\n background-color: rgb(255, 255, 255);\r\n border: 1px solid rgb(191, 119, 124);\r\n border-radius: 50%;\r\n}\r\n\r\n& input:enabled ~ label {\r\n color: rgb(0,0,0);\r\n}\r\n\r\n&:hover input:enabled ~ label {\r\n color: rgb(204,0,16);\r\n}\r\n\r\n& .checkmark:after {\r\n content: \"\";\r\n position: absolute;\r\n display: none;\r\n left: 6px;\r\n top: 1px;\r\n width: 7px;\r\n height: 15px;\r\n border: solid rgb(255,255,255);\r\n border-width: 0px 3px 4px 0px;\r\n -webkit-transform: rotate(45deg);\r\n -ms-transform: rotate(45deg);\r\n transform: rotate(45deg);\r\n }\r\n`;\r\n\r\nexport const SwitchBoxLabel = styled.label`\r\ndisplay: inline-grid;\r\nwidth: 100%;\r\ncursor: pointer;\r\npadding-left: 28px !important;\r\ncolor: rgb(0,0,0);\r\n`;\r\n\r\nexport const SwitchBox = styled.input`\r\nposition: absolute;\r\nopacity: 0;\r\ncursor: pointer;\r\nheight: 0;\r\nwidth: 0;\r\n\r\n&:checked ~ label {\r\n color: rgb(0,0,0);\r\n}\r\n\r\n&:checked ~ label > .checkmark {\r\n background-color: rgb(204,0,16);\r\n}\r\n\r\n&:checked ~ label > .checkmark:after {\r\n display: block;\r\n}\r\n\r\n&:disabled ~ label {\r\n color: rgb(160,160,160);\r\n cursor: not-allowed;\r\n}\r\n`;","import React, { useState, useEffect, useCallback } from \"react\";\r\nimport { useUserSettings } from '../UserSettingsProvider';\r\nimport StepContainer from '../StepContainer/StepContainer';\r\nimport AOSServices from \"../../services/AOSServices\";\r\nimport { FBLabel, PrettyPrintJson } from '../../services/Utils';\r\nimport { SwitchBoxWapper, SwitchBox, SwitchBoxLabel } from '../SelfDisclosure/SwitchBoxStyle';\r\n\r\nimport { format } from 'date-fns';\r\nimport AudioPlayer, { RHAP_UI } from 'react-h5-audio-player';\r\nimport 'react-h5-audio-player/lib/styles.css';\r\nimport { useForm, Controller } from \"react-hook-form\";\r\nimport { yupResolver } from '@hookform/resolvers/yup';\r\nimport * as Yup from 'yup';\r\n\r\n\r\nconst RiskDisclosure = (props) => {\r\n const { userRef } = useUserSettings();\r\n const [AudioSource, setAudioSource] = useState();\r\n const [PlayList, setPlayList] = useState([]);\r\n const [PlaybackRate, setPlaybackRate] = useState(1);\r\n const [RiskLanguage, setRiskLanguage] = useState(\"zh_hk\");\r\n const [CurrentAudioIndex, setCurrentAudioIndex] = useState(0);\r\n const [RiskCompleted, setRiskCompleted] = useState(false);\r\n\r\n const Player = React.createRef();\r\n\r\n const validationSchema = Yup.object().shape({\r\n UnstandandRiskDisclosure: Yup.boolean().oneOf([true], \"請閱讀及同意以上有關聲明\"),\r\n AcceptRiskDisclosure: Yup.boolean().oneOf([true], \"請閱讀及同意以上有關聲明\"),\r\n PersonalInfoCollection: Yup.boolean().oneOf([true], \"請閱讀及同意以上有關聲明\"),\r\n AcceptDisclaimer: Yup.boolean().oneOf([true], \"請閱讀及同意以上有關聲明\"),\r\n SelectedLang: Yup.string().nullable(),\r\n StartAt: Yup.date().nullable(),\r\n CompletedAt: Yup.date().nullable(),\r\n });\r\n\r\n\r\n const { register, trigger, watch, getValues, setValue, control,\r\n formState: { isValid, errors }, clearErrors } = useForm({\r\n mode: \"all\",\r\n resolver: yupResolver(validationSchema),\r\n defaultValues: {\r\n }\r\n });\r\n\r\n const watchAllFields = watch();\r\n\r\n useEffect(() => {\r\n AOSServices.GetRiskDisclosure(userRef.RefCode)\r\n .then((resp) => {\r\n if (resp && resp.Result) {\r\n setAudioSource(resp.Info);\r\n // initial value\r\n setValue(\"SelectedLang\", RiskLanguage);\r\n setValue(\"StaffName\", resp.Info.StaffName);\r\n setValue(\"Licence\", resp.Info.Licence);\r\n setValue(\"PlayList\", resp.Info.PlayList);\r\n }\r\n });\r\n }, []);\r\n\r\n useEffect(() => {\r\n ChangeRiskLanaguage(RiskLanguage);\r\n }, [AudioSource]);\r\n\r\n useEffect(() => {\r\n //console.log(\"change PlaybackRate\")\r\n if (Player.current?.audio?.current) {\r\n Player.current.audio.current.playbackRate = PlaybackRate;\r\n }\r\n }, [PlaybackRate, CurrentAudioIndex, RiskLanguage]);\r\n\r\n const ChangePlaybackRate = () => {\r\n //console.log(Player.current?.audio?.current?.playbackRate, \"playbackRate\");\r\n if (Player.current?.audio?.current) {\r\n setPlaybackRate(PlaybackRate === 1 ? 1.6 : 1);\r\n }\r\n };\r\n\r\n const GetDisclosureType = () => PlayList[CurrentAudioIndex]?.DisclosureType;\r\n\r\n const GetDisclosureTypeList = () => PlayList?.map(a => a.DisclosureType);\r\n\r\n const HandlePlayerStart = async (e) => {\r\n const s = getValues(\"StartAt\");\r\n if (!s) setValue(\"StartAt\", new Date());\r\n\r\n // Allow client to procceed after 15s\r\n setTimeout(() => {\r\n setRiskCompleted(true);\r\n\r\n const end = getValues(\"CompletedAt\");\r\n if (!end)\r\n setValue(\"CompletedAt\", new Date()); \r\n \r\n }, 15000)\r\n };\r\n\r\n const HandleChangeRiskLanguage = (e) => {\r\n const lang = e.target.value;\r\n ChangeRiskLanaguage(lang);\r\n setValue(\"SelectedLang\", lang);\r\n }\r\n\r\n const ChangeRiskLanaguage = (lang) => {\r\n setRiskLanguage(lang);\r\n const playlist = [];\r\n // console.log(lang, \"lang\");\r\n // console.log(RiskLanguage, \"RiskLanguage\");\r\n // console.log(AudioSource, \"AudioSource\");\r\n AudioSource?.PlayList.forEach(obj => {\r\n if (obj.LangCode === lang) {\r\n playlist.push(obj);\r\n }\r\n });\r\n setPlayList(playlist);\r\n setCurrentAudioIndex(0);\r\n };\r\n\r\n const GetRiskTitle = (RType) => {\r\n switch (RType) {\r\n case \"sec\": return \"證券風險聲明\";\r\n case \"ft\": return \"期貨風險聲明\";\r\n case \"opt\": return \"期權風險聲明\";\r\n default: return \"\";\r\n }\r\n }\r\n\r\n const HandlePlayerFinished = (e) => {\r\n //console.log(e, \"HandlePlayerFinished\");\r\n\r\n const idx = CurrentAudioIndex + 1;\r\n //console.log(PlayList.length, \"PlayList.length\");\r\n if (PlayList.length > idx) {\r\n setCurrentAudioIndex(idx);\r\n } else if (idx >= PlayList.length) {\r\n setRiskCompleted(true);\r\n\r\n const end = getValues(\"CompletedAt\");\r\n if (!end)\r\n setValue(\"CompletedAt\", new Date());\r\n }\r\n };\r\n\r\n const HandlePlayerLoadedData = (e) => {\r\n //console.log(e, \"HandlePlayerLoadedData\");\r\n };\r\n\r\n const ProfilePic = () => {\r\n const dst = GetDisclosureType();\r\n //console.log(dst);\r\n\r\n //return dst ? (dst === \"sec\" ? \"/rd/20210608.jpg\" : \"/rd/20200417.jpg\") : \"\";\r\n //return dst ? (RiskLanguage === \"zh_hk\" ? \"/rd/20230927.jpg\" : \"/rd/anonymous.png\") : \"\";\r\n return dst ? \"/rd/anonymous.png\" : \"\";\r\n };\r\n\r\n const goNext = () => {\r\n if (!isValid) {\r\n trigger();\r\n return new Promise(resolve => resolve(false));\r\n } else {\r\n const data = getValues();\r\n //console.log(data);\r\n //return new Promise(resolve => resolve(false));\r\n\r\n return new Promise(resolve => {\r\n AOSServices.SaveRiskDisclosure(userRef.RefCode, data, \"ClientSignature\")\r\n .then(res => {\r\n if (!res || !res.Result) {\r\n alert(\"提交資料失敗,請重新再試\");\r\n }\r\n resolve(res.Result);\r\n });\r\n });\r\n }\r\n };\r\n\r\n\r\n const cssFormLabelCol = \"col-sm-3 col-xs-12 col-form-label my-0 my-md-2\";\r\n const cssFormInputCol = \"col-sm-9 col-xs-12 my-0 my-md-2\";\r\n\r\n return (\r\n
\r\n\r\n {/* */}\r\n\r\n \r\n
\r\n
風險聲明 \r\n \r\n\r\n
\r\n
{GetRiskTitle(GetDisclosureType())} {`${CurrentAudioIndex + 1}/${PlayList?.length}`} \r\n \r\n\r\n
\r\n
\r\n {/*
{AudioSource?.StaffName}
*/}\r\n
\r\n
\r\n\r\n
\r\n
\r\n\r\n
\r\n 廣東話 \r\n
\r\n\r\n
\r\n 普通話 \r\n
\r\n\r\n
,\r\n RHAP_UI.DURATION\r\n ]\r\n }\r\n customControlsSection={\r\n [\r\n RHAP_UI.MAIN_CONTROLS,\r\n RHAP_UI.VOLUME_CONTROLS,\r\n
{PlaybackRate}x
,\r\n ]\r\n }\r\n />\r\n
\r\n\r\n
\r\n\r\n
\r\n \r\n\r\n