element)\n contentText: props.contentText, // For use when the content is strictly text (i.e. elements wrappable within a
element)\n handleClose: props.handleClose,\n emailPackage: props.emailPackage,\n altButton: props.altButton,\n clearBarcode: props.clearBarcode,\n generateBarcode: props.generateBarcode,\n emailPlaceholder: props.emailPlaceholder,\n emailSuccessMessage: props.emailSuccessMessage,\n emailInvalidMessage: props.emailInvalidMessage,\n validEmail: true,\n emailSending: false,\n emailSent: false\n };\n\n this.handleClickOpen = this.handleClickOpen.bind(this);\n this.handleClose = this.handleClose.bind(this);\n this.handleChange = this.handleChange.bind(this);\n this.sendEmail = this.sendEmail.bind(this);\n this.emailInputRef = React.createRef();\n }\n\n handleClickOpen()\n {\n this.setState({open: true})\n }\n\n handleClose = (event, reason) => {\n // Prevent closing the dialog by clicking the background\n if (reason && reason === \"backdropClick\") {\n return;\n }\n\n if (this.state.type === 'email') {\n this.setState({ validEmail: true, emailSent: false });\n }\n\n this.setState({open: false})\n }\n\n componentWillReceiveProps(props)\n {\n this.setState({\n open: props.open,\n title: props.title,\n content: props.content,\n contentText: props.contentText,\n firstButton: props.firstButton,\n handleFirstButtonClick: props.handleFirstButtonClick,\n secondButton: props.secondButton,\n handleSecondButtonClick: props.handleSecondButtonClick,\n altButton: props.altButton,\n });\n }\n\n\n handleChange(event)\n {\n emailAddress = event.target.value;\n }\n\n\n async sendEmail() {\n emailAddress = emailAddress.trim();\n let emailPackage = this.state.emailPackage;\n emailPackage.emailAddress = emailAddress;\n\n if (emailAddress && emailAddress.match(/.+?@{1}.+\\..+/g)) {\n this.setState({\n validEmail: true,\n emailSending: true\n });\n\n await xpressApi.post(EMAIL_DONOR_PASS, emailPackage);\n\n this.setState({\n emailSending: false,\n emailSent: true\n });\n\n await xpressApi.get(RECORD_STAT, {\n params: {\n bloodCenterId: emailPackage.bloodCenterId,\n type: 'email',\n emailAddress: emailAddress\n }\n });\n }\n else {\n this.setState({\n validEmail: false,\n emailSent: false\n });\n }\n }\n\n render()\n {\n let content;\n let actions;\n\n if (this.state.type === 'email')\n {\n if (this.state.emailSent) {\n content =\n \n \n {this.state.emailSuccessMessage}\n \n ;\n\n actions =\n \n \n ;\n }\n else {\n const enter = e => {\n if (e.key === 'Enter') {\n this.sendEmail(e).then();\n }\n }\n\n content =\n \n \n {!this.state.emailSending && }\n {!this.state.validEmail && {this.state.emailInvalidMessage}}\n {this.state.emailSending && }\n \n ;\n\n actions =\n \n {!this.state.emailSending &&\n \n \n\n \n
\n }\n ;\n }\n }\n else if (this.state.type === 'generateBarcode')\n {\n content =\n \n \n {this.state.contentText}\n \n ;\n\n actions =\n \n \n \n \n }\n else {\n content =\n \n {\n /* If the contentText prop was used, wrap it in a DialogContentText element. Otherwise, display the content as-is */\n (this.state.contentText && {this.state.contentText})\n ||\n (this.state.content)\n }\n ;\n\n actions =\n \n \n {this.state.secondButton && \n \n }\n \n }\n\n return (\n
\n \n
\n );\n }\n}\n","import React from 'react';\nimport './Barcode.css';\nimport {xpressApi, RECORD_STAT} from '../../providers/xpress_api';\nimport XpressDialog from '../XpressDialog/XpressDialog';\nimport Header from \"../Common/Header/Header\";\nimport Footer from \"../Common/Footer/Footer\";\n\nexport default class Barcode extends React.Component\n{\n constructor(props)\n {\n super(props);\n\n this.questionnaire = this.props.xpressState.questionnaire;\n this.language = this.props.xpressState.language;\n this.translationConstants = this.props.xpressState.translationConstants;\n this.donorBirthDate = this.props.xpressState.dob;\n this.donorSex = this.props.xpressState.sex;\n this.sexDescription = this.props.xpressState.sexDescription;\n this.bloodCenterProperties = this.props.xpressState.bloodCenterProperties;\n this.barcodeType = this.bloodCenterProperties.barcodeType;\n this.donorPass = this.props.xpressState.donorPass;\n\n this.emailPackage =\n {\n encodedResponseList: this.donorPass.encodedResponseList,\n barcodeType: this.barcodeType,\n translationId: this.language,\n donorBirthDate: this.donorBirthDate,\n donorSex: this.donorSex,\n donorSexDescription: this.sexDescription, // Added the translated sex description (HD #10194)\n bloodCenterId: this.questionnaire.bloodCenterId,\n bloodCenterName: this.bloodCenterProperties.bloodCenterName,\n barcodeImageList: this.donorPass.barcodeImageList,\n showBarcodeInEmail: this.bloodCenterProperties.showBarcode,\n stage: process.env.REACT_APP_DEPLOY_ENV\n }\n\n this.state = {\n emailClicked: false,\n downloadClicked: false\n };\n }\n\n startOver = () => { window.location.href = window.origin + '/' + this.questionnaire.bloodCenterId; }\n\n handlePrint = async () =>\n {\n await xpressApi.get(RECORD_STAT,\n {\n params: {\n bloodCenterId: this.bloodCenterProperties.bloodCenterId,\n type: 'print'\n }\n\n });\n\n window.print();\n }\n\n handleDownload = async () =>\n {\n await xpressApi.get(RECORD_STAT,\n {\n params: {\n bloodCenterId: this.bloodCenterProperties.bloodCenterId,\n type: 'download'\n }\n });\n\n const binaryString = window.atob(this.donorPass.downloadImage);\n let bytes = new Uint8Array(binaryString.length);\n let pdfData = bytes.map((byte, i) => binaryString.charCodeAt(i));\n\n let file = new Blob([pdfData], { type: 'application/pdf' });\n let fileURL = URL.createObjectURL(file);\n\n const anchor = document.createElement('a');\n anchor.href = fileURL;\n anchor.download = 'Donor Pass.pdf';\n\n document.body.appendChild(anchor);\n anchor.click();\n document.body.removeChild(anchor);\n\n this.setState({emailClicked: false, downloadClicked: true});\n }\n\n handleEmail = async () =>\n {\n this.setState({emailClicked: true, downloadClicked: false});\n }\n\n render()\n {\n let donorPassBarcodes = [];\n let donorPassHeader;\n let emailModal, downloadModal;\n let translationConstants = this.translationConstants;\n\n if (this.state.downloadClicked)\n {\n downloadModal =
;\n }\n else\n {\n downloadModal = \"\";\n }\n\n if (this.state.emailClicked)\n {\n emailModal = ;\n }\n else\n {\n emailModal = null;\n }\n\n let barcodeIndex = 0;\n let barcodePackage = this.donorPass;\n let dob = this.donorBirthDate;\n let sex = this.donorSex;\n\n let sexLabel = translationConstants.prescreenPageGender;\n let dobLabel = translationConstants.prescreenPageDOB;\n\n donorPassHeader = (\n \n {this.language === \"1\" && this.questionnaire.thanksPath &&

}\n {this.language === \"2\" && this.questionnaire.thanksPathEs &&

}\n
\n
{sexLabel}: {this.sexDescription}
\n {dobLabel}: {dob}
\n \n
);\n\n barcodePackage.barcodeImageList.forEach(() => {\n let imageText = barcodePackage.encodedResponseList[barcodeIndex];\n let image = barcodePackage.barcodeImageList[barcodeIndex];\n\n //\n // set the sizing css based upon the type of barcode\n //\n\n //\n // default to code128 CSS\n //\n\n let barcodeImageCSS = null;\n if (this.barcodeType === 'qrcode') {\n barcodeImageCSS = {maxWidth: '200px'};\n }\n else if(this.barcodeType === 'pdf417') {\n barcodeImageCSS = {width: '100%', maxWidth: '500px', maxHeight: '150px'};\n }\n else if (this.barcodeType === 'code128') {\n barcodeImageCSS = {width: '100%'};\n }\n\n donorPassBarcodes.push(\n 1 && barcodeIndex + 1 !== barcodePackage.barcodeImageList.length ? '20px' : '')}}>\n
\n
{translationConstants.donorPageBarcode + (barcodePackage.barcodeImageList.length > 1 ? \" \" + (barcodeIndex + 1) : \"\")}:
\n {imageText}
\n \n
\n

\n
\n
\n );\n\n barcodeIndex++;\n });\n\n return (\n \n
\n
\n {emailModal}\n {downloadModal}\n\n \n
\n
\n {/* Display a Thank-You message when the donor pass page is being printed */}\n
{this.translationConstants.donorPassThankYou.replace(\"%BLOOD_CENTER_NAME%\", this.bloodCenterProperties.bloodCenterName)}\n
\n
\n
Donor Pass
\n
\n
\n {donorPassHeader}\n {donorPassBarcodes}\n
{translationConstants.donorPassExpire}
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n
\n {/* Add vertical padding between the buttons if an XS viewport is being used, which causes them to wrap */}\n
\n
\n
\n \n
\n
\n \n
\n
\n
\n
\n
\n
\n
\n \n
\n
\n
\n
\n \n
\n );\n }\n}","import React from 'react';\nimport './Prescreen.css';\nimport {xpressApi, GET_QUESTIONNAIRE_DATA} from '../../providers/xpress_api';\nimport {DotLoader} from 'react-spinners';\nimport moment from 'moment';\nimport XpressDialog from '../XpressDialog/XpressDialog';\nimport {isIOS} from 'react-device-detect';\nimport Header from \"../Common/Header/Header\";\nimport Footer from \"../Common/Footer/Footer\";\nimport {VERSION_NUMBER, PATH_ERROR, PATH_QUESTION, XPRESS_STATE_DATE_OF_BIRTH, XPRESS_STATE_FIRST_TIME_DONOR, XPRESS_STATE_ORIGINAL_QUESTION_ID_LIST, XPRESS_STATE_QUESTIONNAIRE_DATA, \n XPRESS_STATE_QUESTIONNAIRE_NODE_LIST, XPRESS_STATE_RESPONSE_BRANCH_OPTIONS, XPRESS_STATE_RESPONSES, XPRESS_STATE_SEX, XPRESS_STATE_SEX_DESCRIPTION} from \"../../App\";\n\nexport default class Prescreen extends React.Component\n{\n constructor(props)\n {\n super(props);\n\n this.dobInput = React.createRef();\n this.sexInput = React.createRef();\n // this.ftdYesInput = React.createRef();\n // this.ftdNoInput = React.createRef();\n this.continueButton = React.createRef();\n\n\t\tthis.showAlertDialog = false;\n\t\tthis.errors = [];\n\t\tthis.warnings = [];\n\t\t\n //\n // the blood center name and the questionnaire data will be fetched up front, since we need the blood center name for\n // this page.\n //\n\t\tthis.bloodCenterId = this.props.match.params.bcId;\n\t\tthis.translationConstants = this.props.xpressState.translationConstants;\n this.bloodCenterName = '';\n this.questionnaireData = null;\n this.language = this.props.xpressState.language;\n this.sex = this.props.xpressState.sex;\n this.dob = this.props.xpressState.dob;\n this.ftd = this.props.xpressState.ftd;\n this.questionnaire = this.props.xpressState.questionnaire;\n this.bloodCenterProperties = this.props.xpressState.bloodCenterProperties;\n this.sexOptions = this.props.xpressState.sexOptions;\n\n if (!this.bloodCenterProperties) {\n this.props.history.push({\n pathname: PATH_ERROR\n });\n }\n else {\n this.bloodCenterName = this.bloodCenterProperties.bloodCenterName;\n }\n\n this.state = {\n loading: false,\n showAlertDialog: this.showAlertDialog,\n };\n }\n\n componentDidMount() {\n this.sexInput.current.value = (this.props.xpressState.sex ? this.props.xpressState.sex : \"\");\n this.dobInput.current.value = this.props.xpressState.dob;\n /*\n if(this.props.xpressState.ftd !== null) {\n if(this.props.xpressState.ftd) {\n this.ftdYesInput.current.checked = true;\n }\n else {\n this.ftdNoInput.current.checked = true;\n }\n }\n */\n\n this.checkEnableContinueButton();\n }\n\n handleSexChange = event =>\n {\n let sex = event.target.value;\n this.sex = sex;\n\n const sexDescription = event.target.selectedOptions[0].text; // Get the description for the selected sex and store it in the state (HD #10194)\n\n this.setState({sex: sex});\n this.updateXpressStateValue(XPRESS_STATE_SEX, sex);\n this.updateXpressStateValue(XPRESS_STATE_SEX_DESCRIPTION, sexDescription);\n\n //\n // move focus automatically to the DOB field\n //\n // Only set focus with non-iOS devices since iOS automatically opens the keyboard on focus, which is not desired\n if(!isIOS) {\n this.dobInput.current.focus();\n }\n\n this.checkEnableContinueButton();\n };\n\n handleFtdChange = event => {\n let firstTimeDonor = false;\n if (event.target === this.ftdYesInput.current && event.target.checked) {\n firstTimeDonor = true;\n }\n\n this.setState({ftd: firstTimeDonor});\n this.updateXpressStateValue(XPRESS_STATE_FIRST_TIME_DONOR, firstTimeDonor);\n\n this.checkEnableContinueButton();\n }\n\n handleDOBChange = e =>\n {\n let temp = '';\n\n\t\t// Automatically add forward slashes to the birth date upon entering characters\n if (e.target.value.length < 11) {\n if (e.keyCode !== 193 && e.keyCode !== 111)\n {\n if (e.keyCode !== 8)\n {\n if (e.target.value.length === 2 || e.target.value.length === 5) {\n e.target.value = e.target.value + \"/\";\n }\n\n let slashes = e.target.value.split(\"/\");\n //fills in first slash for date validation\n if (e.target.value.length > 2 && slashes.length < 2) {\n const firstPart = e.target.value.substring(0, 2);\n const secondPart = e.target.value.substring(2);\n e.target.value = firstPart + \"/\" + secondPart;\n }\n\n slashes = e.target.value.split(\"/\");\n //fills in second slash for date validation\n if (e.target.value.length > 6 && slashes.length < 3) {\n const firstPart = e.target.value.substring(0, 5);\n const secondPart = e.target.value.substring(5);\n e.target.value = firstPart + \"/\" + secondPart;\n }\n\n if (e.target.value.length > 10) {\n e.target.value = e.target.value.substring(0, 10);\n }\n }\n else\n {\n temp = e.target.value;\n if (e.target.value.length === 5)\n {\n e.target.value = temp.substring(0, 4);\n } else if (e.target.value.length === 2)\n {\n e.target.value = temp.substring(0, 1);\n }\n }\n }\n else\n {\n e.target.value = e.target.value.substring(0, e.target.value.length - 1);\n }\n }\n else {\n e.target.value = e.target.value.substring(0, 10);\n }\n\n let dob = e.target.value;\n this.updateXpressStateValue(XPRESS_STATE_DATE_OF_BIRTH, dob);\n\n this.checkEnableContinueButton();\n }\n\n checkEnableContinueButton = () =>\n {\n let sexUndefined = !this.sexInput.current.value;\n let dobUndefined = this.parseBirthDate(this.dobInput.current.value) === null;\n let ftdUndefined = false; // this.ftdYesInput.current.checked === false && this.ftdNoInput.current.checked === false;\n\n if (sexUndefined || dobUndefined || ftdUndefined) {\n this.continueButton.current.disabled = true;\n }\n else {\n this.continueButton.current.disabled = false;\n }\n }\n\n handleBackClick = () =>\n {\n window.history.back();\n }\n\n\thandleAlertDialogClose = () => {\n\t\tthis.setState({showAlertDialog: false});\n\t}\n\n\thandleAlertDialogContinue = async () => {\n\t\tawait this.launchQuestionnaire(this.bloodCenterId);\n\t}\n\n\tisFormValid = () => {\n let sex = this.props.xpressState.sex;\n let dob = this.props.xpressState.dob;\n // let ftd = this.props.xpressState.ftd;\n\n\t\tlet formValid = true;\n\t\tthis.errors = [];\n\t\tthis.warnings = [];\n\n if (sex === undefined || sex === \"\") {\n formValid = false;\n }\n\n\t\t// Perform birth date validations\n\t\tlet birthDateValid = false;\n\t\t// Verify the value is 10 characters in MM/DD/YYYY format\n\t\tif (dob && dob.length === 10) {\n\t\t\tconst birthDate = this.parseBirthDate(dob);\n\t\t\tif(birthDate) {\n\t\t\t\tlet currentDate = moment(new Date());\n\t\t\t\t// Verify the birth date is prior to the current date and is no more than 200 years in the past\n\t\t\t\tif (birthDate.isBefore(currentDate,'days') && currentDate.diff(birthDate, 'year') < 200) {\n\t\t\t\t\t// Consider the birth date to be valid at this point\n\t\t\t\t\tbirthDateValid = true;\n\n\t\t\t\t\t// Verify the donor's age is between 15 and 99 years old, and generate a warning if not (HD #12577/SCRBX-1053)\n\t\t\t\t\tconst donorAge = currentDate.diff(birthDate, 'year');\n\t\t\t\t\tif(donorAge < 15 || donorAge > 99) {\n\t\t\t\t\t\tthis.warnings.push(this.translationConstants.prescreenPageAgeWarning.replace('%DONOR_AGE%', donorAge));\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\tif(!birthDateValid) {\n\t\t\tthis.errors.push(this.translationConstants.xpressDialogValidDateText);\n\t\t}\n\n\t\t/*\n if (ftd === undefined || ftd === \"\") {\n formValid = false;\n } \n\t\t*/\n\n\t\tif(this.errors.length > 0 || this.warnings.length > 0) {\n\t\t\tformValid = false;\n\t\t}\n\n\t\treturn formValid;\n\t}\n\n handleContinueClick = async () =>\n {\n\t\tconst formValid = this.isFormValid();\n\n if (formValid) {\n await this.launchQuestionnaire(this.bloodCenterId);\n }\n\t\telse if(this.errors.length > 0 || this.warnings.length > 0) {\n\t\t\tthis.setState({ showAlertDialog: true });\n\t\t}\n }\n\n\tparseBirthDate(birthDateString) {\n\t\tlet birthDate = moment(birthDateString, 'MM/DD/YYYY', true);\n\n\t\tif (birthDate.isValid()) {\n\t\t\treturn birthDate;\n }\n else{\n return null;\n }\n\t}\n\n launchQuestionnaire = async () =>\n {\n this.setState({\n loading: true\n });\n\n this.questionnaire.sex = this.sex;\n this.questionnaire.translation = this.language;\n this.questionnaire.bloodCenterId = this.bloodCenterId;\n\n //\n // get the questionnaire data\n //\n let getQuestionnaireDataResponse = await xpressApi.post(GET_QUESTIONNAIRE_DATA, this.questionnaire);\n this.questionnaireData = getQuestionnaireDataResponse.data.questionnaireData;\n\n this.updateXpressStateValue(XPRESS_STATE_QUESTIONNAIRE_DATA, this.questionnaireData);\n this.updateXpressStateValue(XPRESS_STATE_ORIGINAL_QUESTION_ID_LIST, getQuestionnaireDataResponse.data.originalQuestionIdList);\n this.updateXpressStateValue(XPRESS_STATE_QUESTIONNAIRE_NODE_LIST, getQuestionnaireDataResponse.data.questionnaireNodeList);\n this.updateXpressStateValue(XPRESS_STATE_RESPONSE_BRANCH_OPTIONS, getQuestionnaireDataResponse.data.responseBranchOptions);\n\n // Initialize the responses array with blank responses for each question\n let responses = this.props.xpressState.responses;\n if(responses.length === 0) {\n this.questionnaireData.questions.forEach((question) => {\n responses.push({questionId: question.questionId, response: undefined});\n });\n this.updateXpressStateValue(XPRESS_STATE_RESPONSES, responses);\n }\n\n const firstQuestionId = this.questionnaireData.questions[0].questionId;\n\n this.setState({\n loading: false\n });\n\n this.props.history.push({\n pathname: PATH_QUESTION.replace(':bcId', this.bloodCenterId).replace(':qId', firstQuestionId)\n });\n };\n\n updateXpressStateValue = (key, value) => {\n this.props.fnUpdateXpressStateValue(key, value);\n }\n\n render()\n {\n const translationConstants = this.props.xpressState.translationConstants;\n\t\tconst errorsExist = this.errors.length > 0;\n\t\tconst warningsExist = this.warnings.length > 0;\n\n if (this.state.loading) {\n return (\n \n );\n }\n else {\n\t\t\t// Create a modal to display errors or warnings if they exist. If errors exist, only an \"OK\" button will be shown to return to the page. If warnings exist, a\n\t\t\t// \"Continue\" button will be shown to acknowledge the warnings and continue with the questionnaire, as well as a \"Back\" button to return to the page (HD #12577/SCRBX-1053)\n\t\t\tconst alertDialog = \n\t\t\t\t\t\t\t{this.errors.map((errorMessage, index) => )}\n\t\t\t\t\t\t )\n\t\t\t\t\t||\n\t\t\t\t\t(warningsExist && \n\t\t\t\t\t\t