{\n this.setState({ whatsAppAbout: e.target.value });\n }}\n defaultValue={this.state.whatsAppAbout}\n style={{ width: \"calc(100%)\" }}\n />\n );\n };\n\n render() {\n const { classes, assistantId } = this.props;\n const { isoCode, whatsAppAbout, activeKey } = this.state;\n return (\n \n {/* Page name & description container */}\n {!assistantId &&
}\n
\n {/* Layout */}\n
\n \n \n \n \n Set Defaults \n \n \n \n \n Country\n \n \n {isoToName[isoCode] ?? \"Set country\"}\n \n\n \n this.toggleDialog(\"isoCode\")}\n >\n \n \n \n \n \n\n \n\n \n \n About\n \n \n {whatsAppAbout ?? \"Set WhatsApp About\"}\n \n\n \n this.toggleDialog(\"whatsAppAbout\")}\n >\n \n \n \n \n\n \n \n \n \n \n \n
\n
\n \n {this.state.alertMsg}\n \n \n {this.state.openDialog && (\n
\n )}\n
\n );\n }\n}\n\nconst styles = (theme) => ({\n root: {\n width: \"100%\",\n height: \"100%\",\n // background: \"red\",\n // display: \"flex\",\n background: \"rgb(247,247,247)\",\n boxSizing: \"border-box\",\n overflowX: \"hidden\",\n position: \"relative\",\n // paddingBottom: \"80px\"\n },\n pageTitleContainer: {\n position: \"sticky\",\n zIndex: 100,\n top: 0,\n height: 60,\n boxSizing: \"border-box\",\n [theme.breakpoints.down(\"md\")]: {\n // paddingTop: \"40px\",\n },\n [theme.breakpoints.down(\"sm\")]: {\n // paddingTop: \"20px\",\n },\n },\n fullWidth: {\n width: \"100%\",\n background: \"white\",\n },\n detailContainer: {\n marginTop: \"20px\",\n width: \"100%\",\n boxSizing: \"border-box\",\n [theme.breakpoints.down(\"md\")]: {\n paddingLeft: theme.spacing(1),\n paddingRight: theme.spacing(1),\n boxSizing: \"border-box\",\n },\n },\n primaryBackground: {\n background: \"rgb(70 20 134 / 15%)\",\n },\n container: {\n marginTop: 16,\n background: \"white\",\n borderRadius: 8,\n padding: \"24px 36px\",\n },\n containerPlans: {\n marginTop: 24,\n background: \"white\",\n borderRadius: 8,\n padding: \"24px 36px\",\n },\n containerButton: {\n marginTop: 16,\n background: \"white\",\n borderRadius: 8,\n padding: \"15px 15px\",\n },\n viewOldButton: {\n \"&:hover\": {\n color: \"grey\",\n cursor: \"pointer\",\n },\n },\n cells: {\n padding: 24,\n color: \"grey\",\n borderBottom: \"1px solid #ecebeb\",\n },\n lastCell: {\n padding: 24,\n color: \"grey\",\n },\n plandialogContainer: {\n // zIndex: \"3400 !important\",\n \"& .MuiPaper-root\": {\n [theme.breakpoints.down(\"md\")]: {\n width: \"100%\",\n height: \"unset\",\n maxHeight: \"unset\",\n minHeight: \"100px\",\n margin: \"unset\",\n borderRadius: \"6px\",\n },\n [theme.breakpoints.up(\"md\")]: {\n width: \"100%\",\n height: \"unset\",\n maxHeight: \"unset\",\n minHeight: \"100px\",\n margin: \"unset\",\n borderRadius: \"6px\",\n },\n },\n },\n});\n\nconst connectDefaultsPage = connect((state) => ({\n partner: state.partner.partner,\n}))(DefaultsPage);\n\nexport default withStyles(styles)(connectDefaultsPage);\n","import React, { Component } from \"react\";\nimport {\n withStyles,\n Box,\n Grid,\n IconButton,\n Typography,\n Tabs,\n Tab,\n Button,\n TextField,\n Snackbar,\n Dialog,\n DialogContent,\n DialogActions,\n Tooltip,\n Select,\n MenuItem,\n} from \"@material-ui/core\";\nimport { withRouter } from \"react-router-dom\";\nimport FileCopyIcon from \"@material-ui/icons/FileCopy\";\nimport moment from \"moment\";\nimport {\n Add,\n ArrowBack,\n Clear,\n Edit,\n FiberManualRecord,\n Person,\n Info,\n ArrowForward,\n Close,\n} from \"@material-ui/icons\";\nimport BillingPage from \"../BillingProfile/BillingPage\";\nimport TemplateRoute from \"../TemplateMsg/TemplateRoute\";\nimport { connect } from \"react-redux\";\nimport axios from \"axios\";\nimport { URL, projectURL, partnerURL } from \"../../config/config\";\nimport { editAssistants } from \"../../store/assistantState\";\nimport { Alert, Autocomplete } from \"@material-ui/lab\";\nimport { CircularProgress } from \"@material-ui/core\";\nimport WebhookMappingsTable from \"./webhookMappingsTable\";\nimport PlanChangeDialog from \"../PlanDetails/PlanChangeDialog\";\nimport plansHelper from \"../../helpers/plansHelper\";\nimport WccChangeDialog from \"../PlanDetails/WccChangeDialog\";\nimport WabaForm from \"../Dashboard/WabaForm/WabaForm\";\nimport AgentLimitDialog from \"../Manage/AgentLimitDialog\";\nimport WccAnalytics from \"./WccAnalytics/WccAnalytics\";\nimport StepBasedSolutions from \"../../commons/Troubleshoot/StepBasedSolutions\";\nimport SetupMetaAds from \"../MetaAds/SetupMetaAds\";\nimport AddonsManagement from \"./AddonsManagement\";\nimport { fetchPartnerDetails } from \"../../store/partnerState\";\nimport DeleteForeverIcon from \"@material-ui/icons/DeleteForever\";\nimport DefaultsPage from \"../Manage/DefaultsPage\";\nimport Header from \"../../commons/Header/Header\";\nconst verticals = [\n \"UNDEFINED\",\n \"OTHER\",\n \"AUTO\",\n \"BEAUTY\",\n \"APPAREL\",\n \"EDU\",\n \"ENTERTAIN\",\n \"EVENT_PLAN\",\n \"FINANCE\",\n \"GROCERY\",\n \"GOVT\",\n \"HOTEL\",\n \"HEALTH\",\n \"NONPROFIT\",\n \"PROF_SERVICES\",\n \"RETAIL\",\n \"TRAVEL\",\n \"RESTAURANT\",\n \"NOT_A_BIZ\",\n];\nconst PLANS = [\n \"FREE\",\n \"LITE\",\n \"PRO\",\n \"PLUS\",\n \"PREMIUM\",\n \"BASIC_MONTHLY\",\n \"BASIC_QUARTERLY\",\n \"BASIC_YEARLY\",\n \"PRO_MONTHLY\",\n \"PRO_QUARTERLY\",\n \"PRO_YEARLY\",\n \"ENTERPRISE_MONTHLY\",\n \"ENTERPRISE_QUARTERLY\",\n \"ENTERPRISE_YEARLY\",\n \"BASIC_MONTHLY_TIER_1\",\n \"BASIC_MONTHLY_TIER_2\",\n \"BASIC_MONTHLY_TIER_3\",\n \"BASIC_MONTHLY_TIER_4\",\n \"BASIC_MONTHLY_TIER_5\",\n \"BASIC_MONTHLY_TIER_6\",\n \"BASIC_MONTHLY_TIER_7\",\n \"BASIC_MONTHLY_TIER_8\",\n \"BASIC_YEARLY_TIER_1\",\n \"BASIC_YEARLY_TIER_2\",\n];\n\nconst PROJECT_STATUS = [\n \"active\",\n \"unpaid\",\n \"terminated\",\n \"pending\",\n \"suspended\",\n \"stopped\",\n \"archived\",\n];\n\nfunction getStatusColor() {\n const templateStatus = \"PENDING\";\n switch (templateStatus) {\n case \"PENDING\":\n return \"grey\";\n case \"APPROVED\":\n return \"#08CF65\";\n case \"REJECTED\":\n return \"#ff0000\";\n default:\n return \"pink\";\n }\n}\n\nconst allTopics = [\n \"message.created\",\n \"message.status.updated\",\n \"message.sender.user\",\n \"contact.created\",\n \"contact.attribute.updated\",\n \"contact.chat.intervened\",\n \"contact.chat.requesting\",\n \"contact.chat.closed\",\n \"contact.campaign.sent\",\n \"contact.campaign.read\",\n \"contact.first_message.added\",\n \"contact.first_message.removed\",\n \"contact.tag.added\",\n \"contact.tag.removed\",\n];\n\nclass AssistantsDetail extends Component {\n constructor(props) {\n super(props);\n this.state = {\n alertType: \"\",\n alertMsg: \"\",\n tab: \"Details\",\n assistant: null,\n editProfilePicture: false,\n editAboutText: false,\n editAddress: false,\n editDescription: false,\n editEmail: false,\n editWebsite: false,\n editBusinessVertical: false,\n newProfilePicture: \"\",\n newAboutText: \"\",\n newAddress: \"\",\n newDescription: \"\",\n newEmail: \"\",\n newWebsite: \"\",\n newBusinessVertical: \"\",\n loadProfilePicture: false,\n loadAboutText: false,\n loadAddress: false,\n loadDescription: false,\n loadEmail: false,\n loadWebsite: false,\n loadBusinessVertical: false,\n selectedFile: {},\n wasImageChanged: false,\n openPlanDialog: false,\n allPlans: [],\n changePlan: \"\",\n planSubmitLoader: false,\n openWebhookDialog: false,\n webhookSubmitLoader: false,\n selectedTopic: [],\n selectedWebhookurl: \"\",\n webhookResult: [],\n isWebhookLoading: true,\n openWccDialog: false,\n wccAction: \"\",\n wccAmount: \"\",\n wccSubmitLoader: false,\n openPwcDialog: false,\n pwcAction: \"\",\n pwcAmount: \"\",\n pwcSubmitLoader: false,\n linkGenerated: \"\",\n migrationLinkGenerated: \"\",\n manageCatalogLinkGenerated: \"\",\n selectedFamilyId: null,\n currentPlan: \"\",\n email: \"\",\n contact: \"\",\n company: \"\",\n reqBusinessId: \"\",\n planStoppedOn: false,\n selectedValue: \"a\",\n disableDate: Number(localStorage.getItem(\"resetField\")) + 300000,\n openWccChangeDialog: false,\n handleClickDialog: false,\n openPlanDialog: false,\n wabaDialog: false,\n showAgentDialog: false,\n metaError: null,\n isLoading: false,\n openPrivateAppDialog: false,\n privateApps: [],\n password: \"\",\n webhookUrl: \"\",\n SharedSecret: \"\",\n isWebhookLoading: false,\n chatApiWebhook: {},\n reactivationLoader: false,\n deleteProject: false,\n deleteMessage: null,\n deleteStatus: null,\n deleteLoading: false,\n reportLoading: false,\n };\n }\n\n fecthAssistant = () => {\n axios\n .get(URL + `/api/get-project/${this.props.match.params.projectId}`)\n .then((response) => {\n const assistants = response.data.projects;\n this.setState({ assistant: assistants[0] }, () => {\n this.setState({\n addonCheck: this.state?.assistant?.addons?.WEBHOOKS ? true : false,\n });\n });\n if (assistants[0].isWhatsappVerified) this.getDirectApiPartnerProfile();\n })\n .catch((error) => {\n console.error(error);\n });\n };\n\n updateAssistant = (values) => {\n const existing = this.state.assistant;\n this.setState({ assistant: { ...existing, ...values } });\n };\n\n onConfirm = () => {\n this.setState({ deleteLoading: true });\n axios\n .post(URL + `/api/delete-project/${this.props.match.params.projectId}`)\n .then((response) => {\n let { message, status } = response.data;\n this.setState(\n {\n deleteMessage: message,\n deleteStatus: status,\n deleteLoading: false,\n },\n () => {\n const data = { projectDeleted: true };\n if (this.state.deleteStatus == \"#28C153\") {\n this.props.history.push(\"/assistants\", data);\n }\n }\n );\n })\n .catch((error) => {\n this.setState({ deleteLoading: false });\n console.error(error);\n });\n };\n\n componentDidMount() {\n this.props.fetchPartnerDetails();\n this.fecthAssistant();\n // this.fetchBusiness();\n\n if (this.props.partner?.type !== \"DIRECT\") {\n this.fetchPrivateApp();\n plansHelper.setPlanFamilies();\n plansHelper.setWccPlans();\n this.getPartnerPlans();\n this.fetchWebhook();\n }\n }\n\n getPartnerPlans = () => {\n axios\n .get(\n URL +\n `/api/get-partner-plan-family/${this.props.match.params.projectId}`\n )\n .then((response) => {\n const data = response.data.family.plans;\n this.setState({\n selectedFamilyId: response.data.family._id.toString(),\n allPlans: data,\n });\n })\n .catch((error) => {\n console.error(error);\n });\n };\n\n closeHandleDialog = () => {\n this.setState({ handleClickDialog: !this.state.handleClickDialog });\n };\n\n fetchBusiness() {\n // const partnerId = this.props.user.id;\n const _id = this.props.match.params.projectId;\n axios\n .get(URL + `/api/get-business-details/${_id}`)\n .then((response) => {\n const data = response.data;\n this.setState({\n email: data.email,\n contact: data.contact,\n company: data.company,\n reqBusinessId: data._id,\n });\n })\n .catch((error) => {\n console.error(error);\n });\n }\n\n goToBusiness = () => {\n this.props.history.push(`/businesses/${this.state.reqBusinessId}`);\n };\n\n editToggle = (edit) => {\n this.setState({\n [edit]: !this.state[edit],\n });\n };\n\n handleInput = (e) => {\n this.setState({\n [e.target.name]: e.target.value,\n });\n };\n\n syncWaba = () => {\n this.setState({ planSubmitLoader: true });\n const id = this.props.match.params.projectId;\n const partnerId = this.props.partner._id;\n const checkTime = localStorage.getItem(\"resetField\");\n const resetCheck = Number(checkTime) + 300000;\n if (checkTime) {\n if (resetCheck > Date.now()) {\n this.setState({\n alertType: \"error\",\n alertMsg: \"Please try to sync after few miniutes\",\n });\n return;\n }\n }\n axios\n .post(partnerURL + `/partner/${partnerId}/sync-channel-status`, {\n assistantId: id,\n })\n .then((response) => {\n this.setState({\n alertType: \"success\",\n alertMsg: \"Sync successful\",\n planSubmitLoader: false,\n disableDate: Date.now() + 30000,\n });\n localStorage.setItem(\"resetField\", Date.now());\n })\n .catch((error) => {\n this.setState({\n alertType: \"error\",\n alertMsg: \"Sync failed\",\n planSubmitLoader: false,\n });\n });\n };\n\n onEdit = async (item) => {\n try {\n const { assistant, wasImageChanged, selectedFile } = this.state;\n\n const keys = {\n newAboutText: [\"whatsAppAbout\", \"loadAboutText\", \"editAboutText\"],\n newAddress: [\"address\", \"loadAddress\", \"editAddress\"],\n newDescription: [\"description\", \"loadDescription\", \"editDescription\"],\n newEmail: [\"email\", \"loadEmail\", \"editEmail\"],\n newWebsite: [\"websites\", \"loadWebsite\", \"editWebsite\"],\n newBusinessVertical: [\n \"vertical\",\n \"loadBusinessVertical\",\n \"editBusinessVertical\",\n ],\n newProfilePicture: [\n \"whatsAppDisplayImage\",\n \"loadProfilePicture\",\n \"editProfilePicture\",\n ],\n };\n const { whatsAppAbout, whatsAppBusinessProfile } = assistant || {};\n\n const { address, description, email, vertical, websites } =\n whatsAppBusinessProfile || {};\n\n const queryObj = {\n assistantId: this.props.match.params.projectId,\n address: address,\n description: description,\n email: email,\n vertical: vertical,\n wasImageChanged: false,\n filePath: \"\",\n websites: websites,\n whatsAppAbout: whatsAppAbout,\n whatsAppDisplayImage: \"\",\n };\n\n let result;\n\n if (item === \"newProfilePicture\") {\n if (this.state.loadProfilePicture) return;\n } else {\n if (!this.state[item] || this.state[keys[item][1]]) {\n return;\n }\n if (item === \"newWebsite\") result = [this.state[item]];\n else result = this.state[item];\n queryObj[keys[item][0]] = result;\n }\n\n this.setState({\n [keys[item][1]]: true,\n });\n\n if (wasImageChanged && selectedFile) {\n // Get Upload-URL\n const response = await axios.post(\n projectURL + `/project/${queryObj.assistantId}/get-upload-url`,\n {\n assistantId: queryObj.assistantId,\n fileName: \"profile_picture_\" + Date.now(),\n contentType: selectedFile.type,\n fileType: \"IMAGE\",\n }\n );\n let form = new FormData();\n Object.entries(response.data.uploadURL.fields).forEach((entry) => {\n form.append(entry[0], entry[1]);\n });\n form.append(\"file\", selectedFile);\n const config = {\n transformRequest: (data, headers) => {\n delete headers.common[\"Authorization\"];\n return data;\n },\n };\n await axios.post(response.data.uploadURL.url, form, config);\n const fileURL = response.data.fullUrl;\n const filePath = response.data.filePath;\n queryObj.wasImageChanged = true;\n queryObj.whatsAppDisplayImage = fileURL;\n queryObj.filePath = filePath;\n queryObj.type = selectedFile?.type;\n queryObj.size = selectedFile?.size;\n result = fileURL;\n }\n\n axios\n .post(\n projectURL +\n `/project/${queryObj.assistantId}/update-business-profile`,\n queryObj\n )\n .then((response) => {\n this.setState({\n alertType: \"success\",\n alertMsg: \"Successfully Updated\",\n wasImageChanged: false,\n selectedFile: {},\n });\n this.props.editAssistants(\n queryObj.assistantId,\n keys[item][0],\n result\n );\n\n const updateState = {\n ...this.state.assistant,\n };\n\n if (item === \"newAboutText\" || item === \"newProfilePicture\") {\n updateState[keys[item][0]] = result;\n } else {\n const whatsAppBusinessProfile = {\n ...this.state.assistant.whatsAppBusinessProfile,\n [keys[item][0]]: result,\n };\n updateState.whatsAppBusinessProfile = whatsAppBusinessProfile;\n }\n\n this.setState({\n [keys[item][1]]: false,\n [keys[item][2]]: false,\n assistant: updateState,\n });\n })\n .catch((error) => {\n console.log(error);\n this.setState({\n alertType: \"error\",\n alertMsg:\n error?.response?.data?.message ||\n \"something went wrong, tryagain\",\n });\n this.setState({\n [keys[item][1]]: false,\n [keys[item][2]]: false,\n });\n });\n } catch (error) {\n console.log(error);\n this.setState({\n alertType: \"error\",\n alertMsg:\n error?.response?.data?.message || \"something went wrong, tryagain\",\n });\n }\n };\n\n openFileUploader = (action) => {\n this.setState({ editProfilePicture: true }, () => {\n if (this.props.partner?.type !== \"DIRECT\") {\n const input = document.createElement(\"input\");\n input.id = \"file_uploader\";\n input.type = \"file\";\n input.accept = \"image/jpeg, image/png\";\n input.addEventListener(\"change\", this.uploadFile);\n input.click();\n }\n });\n };\n\n uploadFile = (input) => {\n const files = input.target?.files;\n if (files && files[0]) {\n const file = files[0];\n const MB_1 = 1000000;\n\n // Max size of 5MB allowed.\n if (file.size > 5 * MB_1) {\n this.setState({\n alertOpen: true,\n alertType: \"error\",\n alertMsg: \"Maximum size of Image can be 5MB\",\n });\n } else {\n var reader = new FileReader();\n reader.onload = (e) => {\n const img = new Image();\n const updateState = (data) => this.setState(data);\n // const updateForm = (key, value) => this.updateForm(key, value);\n img.onload = function () {\n var height = this.height;\n var width = this.width;\n if (height < 192 || width < 192) {\n updateState({\n alertOpen: true,\n alertType: \"error\",\n alertMsg: \"Image size too small\",\n });\n } else {\n updateState({\n wasImageChanged: true,\n selectedFile: file,\n });\n }\n };\n img.onerror = () => {\n this.setState({\n alertOpen: true,\n alertType: \"error\",\n alertMsg: \"Invalid file format\",\n });\n };\n img.src = e.target.result;\n };\n reader.readAsDataURL(file);\n const input = document.getElementById(\"file_uploader\");\n input.removeEventListener(\"change\", this.uploadFile);\n }\n }\n };\n\n updateProfileViaDirectAPI = (key, value, toggleKey) => {\n axios\n .patch(URL + `/api/direct-api-partner/profile`, {\n projectId: this.props.match.params.projectId,\n [key]: value,\n })\n .then((_) => {\n this.setState({\n alertType: \"success\",\n alertMsg: \"Updated successfully\",\n [toggleKey]: !this.state[toggleKey],\n });\n })\n .catch((error) => {\n console.log(error);\n this.setState({\n alertType: \"error\",\n alertMsg:\n error?.response?.data?.message || \"something went wrong, try again\",\n });\n });\n };\n\n getDirectApiPartnerProfile = () => {\n const projectId = this.props.match.params.projectId;\n axios\n .get(URL + `/api/direct-api-partner/profile/${projectId}`)\n .then(({ data }) => {\n const { businessProfile } = data;\n const [profile] = businessProfile;\n\n const {\n about: newAboutText,\n address: newAddress,\n description: newDescription,\n email: newEmail,\n profile_picture_url: newProfilePicture,\n websites,\n vertical: newBusinessVertical,\n } = profile;\n\n this.setState({\n newProfilePicture,\n newAboutText,\n newAddress,\n newDescription,\n newEmail,\n newWebsite: websites?.join(\", \") || \"\",\n newBusinessVertical,\n });\n })\n .catch((error) => {\n console.log(error);\n this.setState({\n alertType: \"error\",\n alertMsg:\n error?.response?.data?.message || \"something went wrong, try again\",\n });\n });\n };\n\n planDialog = () => {\n this.setState({ openPlanDialog: !this.state.openPlanDialog });\n };\n\n toggleWccPlanChangeDialog = () => {\n this.setState({ openWccChangeDialog: !this.state.openWccChangeDialog });\n };\n\n wccDialog = (event, reason) => {\n reason !== \"backdropClick\" &&\n this.setState({ openWccDialog: !this.state.openWccDialog });\n };\n\n pwcDialog = (event, reason) => {\n reason !== \"backdropClick\" &&\n this.setState({ openPwcDialog: !this.state.openPwcDialog });\n };\n\n webhookDialog = (event, reason) => {\n reason !== \"backdropClick\" &&\n this.setState({ openWebhookDialog: !this.state.openWebhookDialog });\n };\n\n privateAppDialog = (event, reason) => {\n reason !== \"backdropClick\" &&\n this.setState({ openPrivateAppDialog: !this.state.openPrivateAppDialog });\n };\n\n fetchWebhook = () => {\n this.setState({ isWebhookLoading: true });\n axios\n .get(\n projectURL +\n `/project/${this.props.match.params.projectId}/partner/webhook`\n )\n .then((response) => {\n let webhooks = response.data;\n webhooks.forEach((assistant) => {\n if (\n assistant.topics.includes(\"message.sender.user\") ||\n assistant.topics.includes(\"message.status.updated\")\n ) {\n this.setState({ chatApiWebhook: assistant });\n }\n });\n this.setState({ webhookResult: webhooks, isWebhookLoading: false });\n })\n .catch((error) => {\n console.error(error);\n this.setState({ isWebhookLoading: false });\n });\n };\n\n fetchPrivateApp = () => {\n this.setState({ isLoading: true });\n axios\n .get(\n projectURL + `/project/${this.props.match.params.projectId}/private-app`\n )\n .then((response) => {\n let privateApps = response.data;\n this.setState({\n privateApps: privateApps,\n isLoading: false,\n });\n })\n .catch((error) => {\n console.error(error);\n this.setState({\n isLoading: false,\n });\n });\n };\n\n CreatePrivateApp = async () => {\n this.setState({ isLoading: true });\n const { partnerId, clientId } = this.state.assistant;\n axios\n .post(\n projectURL +\n `/project/${this.props.match.params.projectId}/private-app`,\n { partnerId, clientId }\n )\n .then((response) => {\n let privateApps = response.data;\n // let webhooksObj = {};\n // webhooks.forEach((assistant) => {\n // webhooksObj[assistant.project_id] = assistant;\n // });\n this.setState({ password: privateApps, isLoading: false });\n this.setState({\n isLoading: false,\n alertType: \"success\",\n alertMsg: \"Private app created\",\n });\n })\n .catch((error) => {\n console.log(error);\n this.setState({\n isLoading: false,\n alertType: \"error\",\n alertMsg: error.response.data.message || \"Something went wrong\",\n });\n });\n };\n\n updatePrivateApp = async () => {\n this.setState({ isLoading: true });\n const { _id } = this.state.privateApps;\n axios\n .patch(\n projectURL +\n `/project/${this.props.match.params.projectId}/private-app`,\n { privateAppId: _id }\n )\n .then((response) => {\n let reqPassword = response.data;\n this.setState({\n password: reqPassword,\n isLoading: false,\n alertType: \"success\",\n alertMsg: \"Private app updated\",\n });\n })\n .catch((error) => {\n console.error(error);\n this.setState({\n isLoading: false,\n alertType: \"error\",\n alertMsg: error.response.data.message || \"Something went wrong\",\n });\n });\n };\n\n planSubmit = async (newFamilyId, newPlanId) => {\n const newObj = {\n ...this.state.assistant,\n };\n try {\n const resonse = await axios.patch(\n partnerURL +\n `/partner/${this.state.assistant.partnerId}/project/${this.props.match.params.projectId}/billing/switch-plan`,\n {\n newFamilyId,\n newPlanId,\n }\n );\n\n if (resonse.status === 200) {\n newObj.activePlan = newPlanId;\n newObj.familyId = newFamilyId;\n this.props.editAssistants(newObj._id, \"familyId\", newFamilyId);\n this.props.editAssistants(newObj._id, \"activePlan\", newPlanId);\n\n const reqPlan = this.state.changePlan;\n this.setState({\n alertType: \"success\",\n alertMsg: \"Successfully updated plan\",\n planSubmitLoader: false,\n openPlanDialog: false,\n assistant: newObj,\n currentPlan: reqPlan,\n });\n }\n } catch (error) {\n this.setState({\n alertType: \"error\",\n alertMsg:\n error?.response?.data?.message || \"something went wrong, try again\",\n planSubmitLoader: false,\n });\n }\n };\n\n changeWccPlan = async (wccPlanId) => {\n const newObj = {\n ...this.state.assistant,\n };\n try {\n const response = await axios.patch(URL + \"/api/update-project-wcc-plan\", {\n assistantId: this.props.match.params.projectId,\n wccPlanId,\n });\n\n if (response.status === 200) {\n newObj.templateTier = wccPlanId;\n this.props.editAssistants(newObj._id, \"templateTier\", wccPlanId);\n this.setState({\n alertType: \"success\",\n alertMsg: \"Successfully changed wcc plan\",\n planSubmitLoader: false,\n openWccChangeDialog: false,\n assistant: newObj,\n });\n }\n } catch (error) {\n this.setState({\n alertType: \"error\",\n alertMsg:\n error?.response?.data?.message || \"something went wrong, try again\",\n planSubmitLoader: false,\n });\n }\n };\n\n // to stop the plan\n stopPlan = () => {\n this.setState({\n planSubmitLoader: true,\n });\n axios\n .patch(\n partnerURL +\n `/partner/${this.state.assistant.partnerId}/stop-project-billing/${this.props.match.params.projectId}`\n )\n .then((response) => {\n this.setState({\n alertType: \"success\",\n alertMsg: \"Plan stopped!\",\n planStoppedOn: true,\n planSubmitLoader: false,\n });\n this.closeHandleDialog();\n })\n .catch((error) => {\n console.log(error);\n this.setState({\n alertType: \"error\",\n alertMsg:\n error?.response?.data?.message || \"something went wrong, try again\",\n planSubmitLoader: false,\n });\n });\n };\n\n webhookSubmit = () => {\n this.setState({ webhookSubmitLoader: true });\n const newObj = {\n topics: this.state.selectedTopic,\n webhook_url: this.state.selectedWebhookurl,\n };\n\n axios\n .post(\n projectURL + `/project/${this.props.match.params.projectId}/webhook`,\n newObj\n )\n .then((response) => {\n this.setState(\n {\n alertType: \"success\",\n alertMsg: \"Successfully created webhook\",\n webhookSubmitLoader: false,\n openWebhookDialog: false,\n },\n this.fetchWebhook\n );\n })\n .catch((error) => {\n this.setState({\n alertType: \"error\",\n alertMsg:\n error?.response?.data?.message || \"something went wrong, tryagain\",\n webhookSubmitLoader: false,\n });\n });\n };\n\n createWebhook = () => {\n this.setState({ isWebhookLoading: true });\n const newObj = {\n topics: [\"message.sender.user\", \"message.status.updated\"],\n webhook_url: this.state.webhookUrl,\n };\n\n axios\n .post(\n projectURL + `/project/${this.props.match.params.projectId}/webhook`,\n newObj\n )\n .then((response) => {\n this.setState(\n {\n alertType: \"success\",\n alertMsg: \"Successfully created webhook\",\n isWebhookLoading: false,\n },\n this.fetchWebhook\n );\n })\n .catch((error) => {\n this.setState({\n alertType: \"error\",\n alertMsg:\n error?.response?.data?.message || \"something went wrong, tryagain\",\n isWebhookLoading: false,\n });\n });\n };\n\n wccSubmit = () => {\n this.setState({ wccSubmitLoader: true });\n const newObj = {\n ...this.state.assistant,\n };\n if (this.state.wccAmount && this.state.wccAmount > 0)\n axios\n .patch(\n partnerURL +\n `/partner/${this.state.assistant.partnerId}/project/${this.props.match.params.projectId}/billing/wcc`,\n {\n amount: this.state.wccAmount,\n action: this.state.wccAction,\n }\n )\n .then((response) => {\n newObj.templateCredit = response.data.credit;\n this.props.editAssistants(\n newObj._id,\n \"templateCredit\",\n response.data.credit\n );\n this.setState({\n alertType: \"success\",\n alertMsg: \"Successfully updated Wcc\",\n wccSubmitLoader: false,\n openWccDialog: false,\n assistant: newObj,\n });\n })\n .catch((error) => {\n this.setState({\n alertType: \"error\",\n alertMsg:\n error?.response?.data?.message ||\n \"something went wrong, tryagain\",\n wccSubmitLoader: false,\n });\n });\n else\n this.setState({\n alertType: \"error\",\n alertMsg: \"Invalid amount\",\n wccSubmitLoader: false,\n });\n };\n\n pwcSubmit = () => {\n this.setState({ pwcSubmitLoader: true });\n const newObj = {\n ...this.state.assistant,\n };\n if (this.state.pwcAmount && this.state.pwcAmount > 0)\n axios\n .patch(\n partnerURL +\n `/partner/${this.state.assistant.partnerId}/project/${this.props.match.params.projectId}/billing/pwc`,\n {\n amount: this.state.pwcAmount,\n action: this.state.pwcAction,\n }\n )\n .then((response) => {\n newObj.partnerWhatsappCredit = response.data.partner_credit;\n this.props.editAssistants(\n newObj._id,\n \"partnerWhatsappCredit\",\n response.data.partner_credit\n );\n this.setState({\n alertType: \"success\",\n alertMsg: \"Successfully updated Wcc\",\n pwcSubmitLoader: false,\n openPwcDialog: false,\n assistant: newObj,\n });\n })\n .catch((error) => {\n this.setState({\n alertType: \"error\",\n alertMsg:\n error?.response?.data?.message ||\n \"something went wrong, tryagain\",\n pwcSubmitLoader: false,\n });\n });\n else\n this.setState({\n alertType: \"error\",\n alertMsg: \"Enter a valid amount value\",\n pwcSubmitLoader: false,\n });\n };\n\n onSnackbarClose = () => {\n this.setState({\n alertType: \"\",\n alertMsg: \"\",\n });\n };\n\n setSnackbar = ({ alert, alertType, alertMsg }) => {\n this.setState({\n alert,\n alertType,\n alertMsg,\n });\n };\n\n generateWabaLink = () => {\n const assistant = this.state.assistant;\n const { clientId, partnerId, _id } = assistant;\n axios\n .post(URL + `/api/generate-waba-link/`, {\n assistantId: _id,\n businessId: clientId,\n partnerId,\n })\n .then((response) => {\n const { link } = response.data;\n this.setState({\n linkGenerated: link,\n });\n })\n .catch((error) => {\n console.error(error);\n });\n };\n\n generateWabaMigrationLink = () => {\n const assistant = this.state.assistant;\n const { partnerId, _id } = assistant;\n axios\n .post(URL + `/api/generate-waba-migration-link/`, {\n assistantId: _id,\n partnerId,\n })\n .then((response) => {\n const { link } = response.data;\n this.setState({\n migrationLinkGenerated: link,\n });\n })\n .catch((error) => {\n console.error(error);\n });\n };\n\n generateManageCatalogLink = () => {\n const assistant = this.state.assistant;\n const { partnerId, _id } = assistant;\n axios\n .post(URL + `/api/generate-connect-catalog-link/`, {\n assistantId: _id,\n partnerId,\n })\n .then((response) => {\n const { link } = response.data;\n this.setState({\n manageCatalogLinkGenerated: link,\n });\n })\n .catch((error) => {\n console.error(error);\n });\n };\n\n handleChange = (event) => {\n this.setState({\n selectedValue: event.target.value,\n });\n };\n\n toggleWABADialog = () => {\n this.setState({ wabaDialog: !this.state.wabaDialog });\n };\n\n openWabaDialog = () => {\n if (!this.state.assistant.apiFormDetails) {\n this.setState({\n alertType: \"error\",\n alertMsg: \"API form details are not available\",\n });\n } else {\n this.setState({\n wabaDialog: true,\n });\n }\n };\n\n toggleAgentDialog = () => {\n this.setState({ showAgentDialog: !this.state.showAgentDialog });\n };\n\n updateAgentLimit = (agentSize) => {\n const assistant = this.state.assistant;\n const { _id } = assistant;\n axios\n .patch(URL + `/api/update-project-agent-size`, {\n projectId: _id,\n agentSize,\n })\n .then((response) => {\n this.setState({\n alertType: \"success\",\n alertMsg: \"Agent size updated successfully\",\n showAgentDialog: false,\n assistant: { ...this.state.assistant, maxAgents: agentSize },\n });\n })\n .catch((error) => {\n this.setState({\n alertType: \"error\",\n alertMsg: \"Agent size updation failed! Something went wrong.\",\n });\n console.error(error);\n });\n };\n\n EmbeddedSignup = () => {\n const { classes } = this.props;\n return (\n \n \n \n \n Embedded SignUp URL\n \n \n \n {this.state.linkGenerated ? (\n \n \n {this.state.linkGenerated}\n \n {\n this.setState({\n alertType: \"success\",\n alertMsg: \"URL copied to clipboard\",\n });\n navigator.clipboard.writeText(this.state.linkGenerated);\n }}\n >\n Copy Link\n \n \n ) : (\n \n Generate URL\n \n )}\n \n \n \n );\n };\n\n EmbeddedMigration = () => {\n const { classes } = this.props;\n return (\n \n \n \n \n WhatsApp Business API Account Migration URL\n \n \n \n {this.state.migrationLinkGenerated ? (\n \n \n {this.state.migrationLinkGenerated}\n \n {\n this.setState({\n alertType: \"success\",\n alertMsg: \"URL copied to clipboard\",\n });\n navigator.clipboard.writeText(\n this.state.migrationLinkGenerated\n );\n }}\n >\n Copy Link\n \n \n ) : (\n \n Generate URL\n \n )}\n \n \n \n );\n };\n\n SyncProgress = () => {\n const { classes } = this.props;\n return (\n \n \n \n Sync Waba Procurement Progress\n \n \n = 0 ||\n this.state.planSubmitLoader\n ? \"Retry again a in few mins\"\n : \"Sync WABA\"\n }\n >\n \n = 0 ||\n this.state.planSubmitLoader\n }\n endIcon={\n this.state.planSubmitLoader ? (\n \n ) : (\n \"\"\n )\n }\n >\n Sync\n \n \n \n \n \n \n );\n };\n\n setMetaError = (error) => {\n this.setState({ metaError: error });\n };\n\n embeddedHelper = () => {\n const { assistant, metaError } = this.state;\n let { wabaProgressStep, procurementError, isWhatsappVerified } =\n assistant || {};\n if (this.state.assistant && !wabaProgressStep) {\n wabaProgressStep = 0;\n }\n\n let { process, error } = metaError || procurementError || {};\n if (metaError && metaError?.response?.data?.error?.code) {\n error = metaError;\n }\n let steps = [];\n let customObject = { component: <>>, title: \"\", message: \"\" };\n\n if (wabaProgressStep === 9 && isWhatsappVerified) {\n steps = [\"register_pin\", \"otp_verification\"];\n }\n\n if (wabaProgressStep <= 7) {\n steps = [\n \"health_check\",\n \"register_pin\",\n \"otp_verification\",\n \"embedded_signup\",\n \"sync_progress\",\n // \"migrate_partner_project\"\n ];\n\n const code = error?.response?.data?.error?.code;\n const subCode = error?.response?.data?.error?.error_subcode;\n // If we have error available then use it to suggest further steps\n if (wabaProgressStep < 7) {\n switch (process) {\n case \"partner_embedded_signup\":\n //Facebook Authentication failed!\n steps = [\"health_check\", \"embedded_signup\"];\n break;\n case \"partner_waba_migration\":\n steps = [\"health_check\", \"migrate_partner_project\"];\n break;\n default:\n break;\n }\n }\n // If we have error available then use it to suggest further steps\n else if (code === 100) {\n // \"Phone numbers count exceeded limit per business.\n if (subCode === 2388112) {\n steps = [\n \"health_check\",\n process === \"migrate_partner_project\"\n ? \"migrate_partner_project\"\n : \"embedded_signup\",\n ];\n }\n // Incorrect/Unsupported APP ID.\n else if (subCode === 33) {\n steps = [\n \"custom_msg\",\n process === \"migrate_partner_project\"\n ? \"migrate_partner_project\"\n : \"embedded_signup\",\n ];\n customObject = {\n component: <>>,\n title: \"Incorrect/Unsupported WhatsApp Business Account ID\",\n message:\n \"The WhatsApp Business Account ID used during the onboarding process is either Incorrect or Unsupported. Kindly go through onboarding again with a new Account ID\",\n };\n }\n // Fallback\n else {\n steps = [];\n }\n } else if (code === 133005) {\n //2 Factor Authentication is enabled\n steps = [\n \"custom_msg\",\n \"register_pin\",\n \"otp_verification\",\n \"health_check\",\n \"sync_progress\",\n ];\n customObject = {\n component: <>>,\n title: \"Disable 2FA\",\n message:\n \"Your 2 Factor Authentication is enabled. Please disable 2 Factor Authentication first and then proceed to register your phone number in the next step.\",\n };\n } else if (code === 133006) {\n //Phone number re-verification needed\n if (subCode === 2494081) {\n steps = [\n \"custom_msg\",\n \"otp_verification\",\n \"register_pin\",\n \"health_check\",\n \"sync_progress\",\n ];\n }\n\n customObject = {\n component: <>>,\n title: \"Phone number re-verification needed\",\n message:\n \"Phone number needs to be verified before registering. Press next to proceed for OTP verification.\",\n };\n } else if (code === 136024) {\n // You have already verified ownership of this phone number.\n if (subCode === 2388091) {\n steps = [\"register_pin\", \"health_check\", \"sync_progress\"];\n }\n }\n // If error isn't available then suggest general solutions for troubleshooting\n else {\n steps = [\n \"health_check\",\n process === \"migrate_partner_project\"\n ? \"migrate_partner_project\"\n : \"embedded_signup\",\n ];\n }\n }\n\n return { troubleshootSteps: steps, customObject };\n };\n\n updateWebhookUrl = () => {\n this.setState({ isWebhookLoading: true });\n axios\n .patch(\n projectURL +\n `/project/${this.state.assistant._id}/webhook/${this.state.chatApiWebhook.id}`,\n {\n webhook_url: this.state.chatApiWebhook.webhook_url,\n }\n )\n .then((response) => {\n this.setState(\n {\n isWebhookLoading: false,\n alertType: \"success\",\n alertMsg: \"Successfully Updated webhook\",\n },\n this.props.fetchWebhook\n );\n })\n .catch((error) => {\n console.error(error);\n this.setState({\n isWebhookLoading: false,\n alertType: \"error\",\n alertMsg:\n error?.response?.data?.message || \"something went wrong, try again\",\n });\n });\n };\n\n copySnack = () => {\n this.setState({\n alertType: \"success\",\n alertMsg: \"API key copied\",\n });\n };\n\n reactivatePlan = async () => {\n this.setState({\n reactivationLoader: true,\n });\n const {\n familyId,\n activePlan: defaultPlan,\n partnerId,\n } = this.state.assistant;\n axios\n .patch(\n partnerURL +\n `/partner/${partnerId}/project/${this.state.assistant._id}/billing/reactivate-project`,\n { familyId, defaultPlan }\n )\n .then((response) => {\n this.setState({\n planStoppedOn: false,\n reactivationLoader: false,\n alertType: \"success\",\n alertMsg: response.data.message || \"plan reactivated Successfully\",\n });\n this.props.editAssistants(\n this.state.assistant._id,\n \"planStoppedOn\",\n false\n );\n })\n .catch((error) => {\n this.setState({\n reactivationLoader: false,\n alertType: \"error\",\n alertMsg: error?.response?.data?.message || \"something went wrong\",\n });\n });\n };\n\n toggleDialog = () => {\n this.props.history.push(\"/assistants\");\n };\n\n getConvoReport = async (startDate, endDate) => {\n const assistant = this.state.assistant;\n const assistantId = assistant?._id;\n const partnerId = this.props.partner._id;\n if (!assistantId) return null;\n\n this.setState({ reportLoading: true });\n\n try {\n const response = await axios.post(\n partnerURL +\n `/partner/${partnerId}/project/${assistantId}/project-convo-report`,\n { startDate, endDate },\n { responseType: \"stream\" }\n );\n\n const report = response.data?.logs || [];\n const headers = Array.from(\n new Set(report.flatMap((entry) => Object.keys(entry)))\n );\n let csvContent = headers.join(\",\") + \"\\n\";\n\n report.forEach((entry) => {\n headers.forEach((header) => {\n const cellContent = entry[header]\n ? `\"${entry[header].replace(/\\n/g, \"\\n\")}\"`\n : \"\";\n csvContent += cellContent + \",\";\n });\n csvContent = csvContent.slice(0, -1);\n csvContent += \"\\n\";\n });\n\n // Download the file as CSV\n var downloadLink = document.createElement(\"a\");\n const blob = new Blob([csvContent], { type: \"text/csv\" });\n var url = window.URL.createObjectURL(blob);\n downloadLink.href = url;\n\n // Format file name\n let start = moment(startDate, \"YYYY-MM-DD\").isValid()\n ? moment(startDate).format(\"DD-MM-YYYY\")\n : null;\n let end = moment(endDate, \"YYYY-MM-DD\").isValid()\n ? moment(endDate).format(\"DD-MM-YYYY\")\n : null;\n\n if (startDate && endDate) {\n downloadLink.download = `Project_Conversation_Report_${start}_to_${end}.csv`;\n } else {\n downloadLink.download = `Project_Conversation_Report.csv`;\n }\n\n document.body.appendChild(downloadLink);\n downloadLink.click();\n document.body.removeChild(downloadLink);\n this.setState({\n reportLoading: false,\n alertType: \"success\",\n alertMsg: \"Successfully Downloaded!\",\n });\n } catch (error) {\n this.setState({\n reportLoading: false,\n alertType: \"error\",\n alertMsg: error?.response?.data?.message || \"Error downloading report!\",\n });\n console.log(error);\n }\n };\n\n render() {\n // console.log(this.state.privateApps);\n const { assistant, tab } = this.state;\n const { classes, planFamilies, wccPlans, partner } = this.props;\n const { type, addons, centralBalance, minAllowedBalance } = partner || {};\n const {\n assistantName,\n isWhatsappVerified,\n currency,\n templateCredit,\n whatsappNumber,\n whatsAppDisplayImage,\n whatsAppAbout,\n whatsAppDisplayName,\n displayNameVerified,\n dailyTemplateLimit,\n qualityRating,\n whatsAppBusinessProfile,\n activePlan,\n familyId,\n planStoppedOn,\n templateTier,\n chargebeePlan,\n status,\n maxAgents,\n fbManagerVerified,\n partnerId,\n addons: assistantAddons,\n isoCode,\n wabaNumberId,\n wabaAppId,\n wabaProgressStep,\n } = assistant || {};\n\n const family = planFamilies[familyId];\n const familyName = family?.name ?? \"NA FAMILY\";\n const name = family?.plans[activePlan]?.name ?? \"NA PLAN\";\n const wccPlanName = wccPlans[templateTier]?.name ?? \"NA PLAN\";\n const { address, description, email, vertical, websites } =\n whatsAppBusinessProfile || {};\n const { troubleshootSteps, customObject } = this.embeddedHelper();\n const lowBalance = centralBalance < (minAllowedBalance || 0);\n return !this.state.assistant ? (\n \n ) : (\n \n {/* Page name & description container */}\n
\n \n \n \n \n\n \n\n \n \n \n \n \n \n {assistantName}\n \n \n Project Name\n \n \n \n \n \n {isWhatsappVerified ? \"LIVE\" : \"DRAFT\"}\n \n \n Account Status\n \n \n {[\"verified\", \"VERIFIED\"].includes(fbManagerVerified) &&\n [\"ready\", \"verified\", \"APPROVED\", \"DEFERRED\"].includes(\n displayNameVerified\n ) && (\n \n \n \n {[\"Low\", \"FLAGGED\"].includes(qualityRating)\n ? \"LOW\"\n : [\"UNFLAGGED\", \"Medium\", \"RESTRICTED\"].includes(\n qualityRating\n )\n ? \"MEDIUM\"\n : \"HIGH\"}\n \n \n Quality Rating\n \n \n )}\n \n \n {`${!currency || currency === \"INR\" ? \"₹\" : \"$\"} ${\n !templateCredit\n ? 0\n : Number(templateCredit / 100000).toFixed(2)\n }`}\n \n \n WhatsApp Credits\n \n \n \n
\n \n {/* \n \n \n \n \n Business Details\n \n \n Go To Business \n \n \n\n \n \n \n \n \n \n Company Name\n \n \n \n \n \n \n \n \n \n \n {this.state.company}\n \n \n \n \n\n \n \n \n Mobile Number\n \n \n \n \n \n \n \n \n \n \n {this.state.contact}\n \n \n \n \n\n \n \n \n Admin Email\n \n \n \n \n \n \n \n \n \n \n {this.state.email}\n \n \n \n \n \n
\n \n \n \n */}\n \n \n\n \n {\n this.setState({\n tab: newValue,\n });\n }}\n indicatorColor=\"primary\"\n textColor=\"primary\"\n variant=\"scrollable\"\n scrollButtons=\"auto\"\n aria-label=\"scrollable auto tabs example\"\n TabIndicatorProps={{\n className: classes.tabIndicator,\n }}\n >\n {/* */}\n \n \n {/* */}\n {/* */}\n {/* */}\n\n {/* */}\n\n {/* {isWhatsappVerified &&\n Object.keys(addons || {}).length > 0 && (\n \n )} */}\n {/* {assistantAddons?.WEBHOOKS?.status === \"active\" &&\n [\"WHITELABEL\"].includes(this.props.partner.type) &&\n ![\"BASIC_MONTHLY\", \"BASIC_YEARLY\", \"NONE\"].includes(\n this.state.assistant.activePlan\n ) && }\n {[\"WHITELABEL\"].includes(this.props.partner.type) &&\n ![\"BASIC_MONTHLY\", \"BASIC_YEARLY\", \"NONE\"].includes(\n this.state.assistant.activePlan\n ) &&\n [\"DIAMOND\"].includes(partner.planType) && (\n \n )}\n {addons?.META_ADS?.status === \"active\" &&\n addons?.META_ADS?.dashboardKey &&\n [\"WHITELABEL\"].includes(this.props.partner.type) &&\n this.state.assistant.activePlan &&\n this.state.assistant.activePlan !== \"NONE\" && (\n \n )} */}\n {/* */}\n \n \n \n {tab === \"Setup\" ? (\n
\n \n Onboarding\n \n\n \n <>\n {type !== \"DIRECT\" && (\n \n \n Agents Limit\n \n \n {maxAgents ?? \"Set agent limit\"}\n \n\n \n \n \n \n \n \n )}\n {type !== \"DIRECT\" &&\n partner?.onboardingMethod ===\n \"WHATSAPP_API_FORM\" && (\n \n \n Whatsapp Application Form Status\n \n = 768 && \"flex-end\",\n }}\n className={classes.cells}\n >\n {assistant?.apiFormDetails\n ? \"Completed\"\n : \"Incomplete\"}\n \n \n )}\n {type !== \"DIRECT\" &&\n partner?.onboardingMethod ===\n \"WHATSAPP_API_FORM\" && (\n \n \n WhatsApp API Form Details\n \n = 768 && \"flex-end\",\n }}\n className={classes.cells}\n >\n this.openWabaDialog()}\n >\n View Details\n \n \n \n )}\n\n {type === \"DIRECT\" && (\n \n \n Embedded FB Catalog Management\n \n = 768 && \"flex-end\",\n }}\n className={classes.cells}\n >\n {this.state.manageCatalogLinkGenerated ? (\n \n \n {this.state.manageCatalogLinkGenerated}\n \n {\n this.setState({\n alertType: \"success\",\n alertMsg: \"URL copied to clipboard\",\n });\n navigator.clipboard.writeText(\n this.state.manageCatalogLinkGenerated\n );\n }}\n >\n Copy Link\n \n \n ) : (\n \n \n \n Generate URL\n \n \n \n )}\n \n \n )}\n\n {!isWhatsappVerified && (\n \n \n WhatsApp Business API Account Migration URL\n \n = 768 && \"flex-end\",\n }}\n className={classes.cells}\n >\n {this.state.migrationLinkGenerated ? (\n \n \n {this.state.migrationLinkGenerated}\n \n {\n this.setState({\n alertType: \"success\",\n alertMsg: \"URL copied to clipboard\",\n });\n navigator.clipboard.writeText(\n this.state.migrationLinkGenerated\n );\n }}\n >\n Copy Link\n \n \n ) : (\n \n \n \n Generate URL\n \n \n \n )}\n \n \n )}\n {!isWhatsappVerified && (\n \n \n Embedded SignUp URL\n \n = 768 && \"flex-end\",\n }}\n className={classes.cells}\n >\n {this.state.linkGenerated ? (\n \n \n {this.state.linkGenerated}\n \n {\n this.setState({\n alertType: \"success\",\n alertMsg: \"URL copied to clipboard\",\n });\n navigator.clipboard.writeText(\n this.state.linkGenerated\n );\n }}\n >\n Copy Link\n \n \n ) : (\n \n \n \n Generate URL\n \n \n \n )}\n \n \n )}\n {assistant && !assistant.planTier && (\n \n \n Sync Waba Progress\n \n = 768 && \"flex-end\",\n }}\n className={classes.cells}\n >\n =\n 0 || this.state.planSubmitLoader\n ? \"Retry again a in few mins\"\n : \"Sync WABA\"\n }\n >\n \n =\n 0 || this.state.planSubmitLoader\n }\n endIcon={\n this.state.planSubmitLoader ? (\n \n ) : (\n \"\"\n )\n }\n >\n Sync\n \n \n \n \n \n )}\n >\n \n\n {troubleshootSteps.length > 0 && (\n \n )}\n \n ) : tab === \"Details\" ? (\n
\n \n \n WhatsApp Channel\n \n \n \n \n Phone Number\n \n \n {whatsappNumber}\n \n \n \n {/* DONE */}\n {/* */}\n \n \n\n \n WhatsApp Display Name\n \n \n {whatsAppDisplayName}\n \n \n \n {displayNameVerified}\n {/* */}\n \n \n\n \n Messaging Limit\n \n \n {dailyTemplateLimit}\n \n \n\n \n Quality Rating\n \n \n {qualityRating}\n \n \n {/* \n WABA Number ID\n \n \n {wabaNumberId}\n */}\n {/* \n \n WABA App ID\n \n \n {wabaAppId}\n */}\n {/* \n Embedded SignUp URL\n {this.state.linkGenerated ? (\n \n \n {this.state.linkGenerated}\n \n {\n this.setState({\n alertType: \"success\",\n alertMsg: \"URL copied to clipboard\",\n });\n navigator.clipboard.writeText(\n this.state.linkGenerated\n );\n }}\n >\n Copy Link\n \n \n ) : (\n \n Generate URL\n \n )}\n */}\n\n {/* \n Sync Waba Progress\n \n Sync\n \n */}\n \n \n \n ) : tab === \"Analytics\" ? (\n
\n \n \n ) : tab === \"Templates\" ? (\n
\n \n \n ) : tab === \"Profile\" ? (\n
\n {isWhatsappVerified ? (\n \n \n WhatsApp Business Profile\n \n \n \n \n Profile Picture\n \n \n {this.state.editProfilePicture ? (\n type === \"DIRECT\" ? (\n \n ) : (\n \n )\n ) : this.state.newProfilePicture ||\n whatsAppDisplayImage ? (\n \n ) : (\n \n )}\n \n\n \n {this.state.editProfilePicture ? (\n <>\n {\n if (type === \"DIRECT\")\n this.updateProfileViaDirectAPI(\n \"whatsAppDisplayImage\",\n this.state.newProfilePicture,\n \"editProfilePicture\"\n );\n else this.onEdit(\"newProfilePicture\");\n }}\n style={{ marginRight: 16 }}\n disabled={this.state.loadProfilePicture}\n endIcon={\n this.state.loadProfilePicture ? (\n \n ) : (\n \"\"\n )\n }\n >\n Save\n \n {\n this.editToggle(\"editProfilePicture\");\n this.setState({\n wasImageChanged: false,\n selectedFile: {},\n });\n }}\n >\n Cancel\n \n >\n ) : (\n \n Change Profile Picture\n \n )}\n \n\n \n About text\n \n \n {this.state.editAboutText ? (\n \n ) : (\n this.state.newAboutText || whatsAppAbout\n )}\n \n \n {this.state.editAboutText ? (\n <>\n {\n if (type === \"DIRECT\")\n this.updateProfileViaDirectAPI(\n \"whatsAppAbout\",\n this.state.newAboutText,\n \"editAboutText\"\n );\n else this.onEdit(\"newAboutText\");\n }}\n style={{ marginRight: 16 }}\n disabled={this.state.loadAboutText}\n endIcon={\n this.state.loadAboutText ? (\n \n ) : (\n \"\"\n )\n }\n >\n Save\n \n {\n this.editToggle(\"editAboutText\");\n }}\n >\n Cancel\n \n >\n ) : (\n {\n this.editToggle(\"editAboutText\");\n }}\n >\n \n \n )}\n \n \n \n \n Business Profile\n \n\n \n \n Address\n \n \n {this.state.editAddress ? (\n \n ) : this.state.newAddress || address ? (\n this.state.newAddress || address\n ) : (\n N/A \n )}\n \n \n {this.state.editAddress ? (\n <>\n {\n if (type === \"DIRECT\")\n this.updateProfileViaDirectAPI(\n \"address\",\n this.state.newAddress,\n \"editAddress\"\n );\n else this.onEdit(\"newAddress\");\n }}\n disabled={this.state.loadAddress}\n style={{ marginRight: 16 }}\n endIcon={\n this.state.loadAddress ? (\n \n ) : (\n \"\"\n )\n }\n >\n Save\n \n {\n this.editToggle(\"editAddress\");\n }}\n >\n Cancel\n \n >\n ) : (\n {\n this.editToggle(\"editAddress\");\n }}\n >\n \n \n )}\n \n \n Description\n \n \n {this.state.editDescription ? (\n \n ) : this.state.newDescription || description ? (\n this.state.newDescription || description\n ) : (\n N/A \n )}\n \n \n {this.state.editDescription ? (\n <>\n {\n if (type === \"DIRECT\")\n this.updateProfileViaDirectAPI(\n \"description\",\n this.state.newDescription,\n \"editDescription\"\n );\n else this.onEdit(\"newDescription\");\n }}\n disabled={this.state.loadDescription}\n style={{ marginRight: 16 }}\n endIcon={\n this.state.loadDescription ? (\n \n ) : (\n \"\"\n )\n }\n >\n Save\n \n {\n this.editToggle(\"editDescription\");\n }}\n >\n Cancel\n \n >\n ) : (\n {\n this.editToggle(\"editDescription\");\n }}\n >\n \n \n )}\n \n \n Email\n \n \n {this.state.editEmail ? (\n \n ) : this.state.newEmail || email ? (\n this.state.newEmail || email\n ) : (\n N/A \n )}\n \n \n {this.state.editEmail ? (\n <>\n {\n if (type === \"DIRECT\")\n this.updateProfileViaDirectAPI(\n \"email\",\n this.state.newEmail,\n \"editEmail\"\n );\n else this.onEdit(\"newEmail\");\n }}\n disabled={this.state.loadEmail}\n style={{ marginRight: 16 }}\n endIcon={\n this.state.loadEmail ? (\n \n ) : (\n \"\"\n )\n }\n >\n Save\n \n {\n this.editToggle(\"editEmail\");\n }}\n >\n Cancel\n \n >\n ) : (\n {\n this.editToggle(\"editEmail\");\n }}\n >\n \n \n )}\n \n\n \n Website\n \n \n {this.state.editWebsite ? (\n \n ) : this.state.newWebsite || websites ? (\n this.state.newWebsite || websites.join(\", \")\n ) : (\n N/A \n )}\n \n \n {this.state.editWebsite ? (\n <>\n {\n if (type === \"DIRECT\")\n this.updateProfileViaDirectAPI(\n \"websites\",\n this.state.newWebsite\n .split(\",\")\n .map(function (item) {\n return item.trim();\n }),\n \"editWebsite\"\n );\n else this.onEdit(\"newWebsite\");\n }}\n style={{ marginRight: 16 }}\n disabled={this.state.loadWebsite}\n endIcon={\n this.state.loadWebsite ? (\n \n ) : (\n \"\"\n )\n }\n >\n Save\n \n {\n this.editToggle(\"editWebsite\");\n }}\n >\n Cancel\n \n >\n ) : (\n {\n this.editToggle(\"editWebsite\");\n }}\n >\n \n \n )}\n \n\n \n Business vertical\n \n \n {this.state.editBusinessVertical ? (\n {\n this.setState({\n newBusinessVertical: event.target.value,\n });\n }}\n >\n {verticals.map((e) => (\n {e} \n ))}\n \n ) : this.state.newBusinessVertical || vertical ? (\n this.state.newBusinessVertical || vertical\n ) : (\n N/A \n )}\n \n \n {this.state.editBusinessVertical ? (\n <>\n {\n if (type === \"DIRECT\")\n this.updateProfileViaDirectAPI(\n \"vertical\",\n this.state.newBusinessVertical,\n \"editBusinessVertical\"\n );\n else this.onEdit(\"newBusinessVertical\");\n }}\n style={{ marginRight: 16 }}\n disabled={this.state.loadBusinessVertical}\n endIcon={\n this.state.loadBusinessVertical ? (\n \n ) : (\n \"\"\n )\n }\n >\n Save\n \n {\n this.editToggle(\"editBusinessVertical\");\n }}\n >\n Cancel\n \n >\n ) : (\n {\n this.editToggle(\"editBusinessVertical\");\n }}\n >\n \n \n )}\n \n \n \n ) : (\n <>\n \n WhatsApp Business Profile\n \n \n \n Profile Picture\n \n \n {whatsAppDisplayImage ? (\n \n ) : (\n \n )}\n \n \n \n\n \n Change Profile Picture\n \n \n\n \n About text\n \n \n <>>\n \n \n \n \n \n \n \n \n\n \n \n Waba is not live yet!\n \n \n >\n )}\n \n ) : tab === \"Billings\" ? (\n
\n \n \n ) : tab === \"Plan Setup\" ? (\n
\n {isWhatsappVerified ? (\n <>\n \n {!chargebeePlan && type !== \"DIRECT\" && (\n <>\n \n \n Subscription Plan Family\n \n \n \n \n Active Plan Family\n \n \n {familyName}\n \n\n \n {!planStoppedOn &&\n Object.keys(planFamilies).length >= 1 && (\n \n \n \n )}\n \n \n \n \n Active Plan\n \n \n {name}\n \n\n \n {!planStoppedOn &&\n Object.keys(planFamilies).length >= 1 && (\n \n \n \n )}\n \n \n \n \n WhatsApp Conversation Charges (WCC) Family\n \n \n \n \n Active Plan\n \n \n {wccPlanName}\n \n\n \n {!planStoppedOn &&\n Object.keys(wccPlans).length > 1 && (\n \n \n \n )}\n \n \n {partner._id === \"62839c2733581a0fb5e71df6\" && (\n <>\n \n \n PWC Credit\n \n \n \n \n Amount\n \n \n {Number(\n this.state.assistant\n .partnerWhatsappCredit / 100000\n ).toFixed(2)}\n {/* {parseInt(this.state.assistant.partnerWhatsappCredit.toString().slice(0,-5))} */}\n \n \n \n \n \n \n \n >\n )}\n {partner._id !== \"62839c2733581a0fb5e71df6\" && (\n <>\n \n \n WCC Credit\n \n \n \n \n Amount\n \n \n {Number(\n this.state.assistant.templateCredit /\n 100000\n ).toFixed(2)}\n {/* {parseInt(this.state.assistant.templateCredit.toString().slice(0,-5))} */}\n \n \n \n \n \n \n \n >\n )}\n >\n )}\n {chargebeePlan && (\n \n \n \n \n This project's billing is being managed\n through ChargeBee\n \n \n \n \n )}\n \n \n \n Settings\n \n \n \n \n Delete Project\n \n\n \n {/* {\n \"Un-register this project from your patner dashboard\"\n } */}\n \n \n {\n console.log(\"qwertyui\");\n console.log(this.state.deleteProject);\n this.editToggle(\"deleteProject\");\n }}\n >\n \n \n \n \n \n {!chargebeePlan && (\n \n \n {planStoppedOn || this.state.planStoppedOn ? (\n <>\n \n {status !== \"stopped\" ? (\n \n This project was requested to be\n stopped on{\" \"}\n \n {moment(planStoppedOn).format(\n \"LL\"\n )}\n {\" \"}\n all the services of this project\n will be disabled on{\" \"}\n \n {moment(\n assistant.planRenewalOn\n ).format(\"LL\")}\n \n \n ) : (\n \n This project was stopped on{\" \"}\n \n {moment(planStoppedOn).format(\n \"LL\"\n )}\n {\" \"}\n all the services of this project are\n be disabled\n \n )}\n \n \n )\n }\n >\n Reactivate Plan\n \n >\n ) : (\n <>\n \n Disable Plan\n \n \n Stop\n \n >\n )}\n \n \n )}\n \n >\n ) : (\n <>\n \n \n Settings\n \n\n \n \n \n Delete Project\n \n\n \n {/* {\n \"Un-register this project from your patner dashboard\"\n } */}\n \n \n {\n console.log(\"qwertyui\");\n console.log(this.state.deleteProject);\n this.editToggle(\"deleteProject\");\n }}\n >\n \n \n \n \n \n\n \n \n Waba is not live yet!\n \n \n >\n )}\n \n ) : tab === \"Addons Management\" ? (\n
\n ) : tab === \"Webhook\" ? (\n
\n {isWhatsappVerified ? (\n \n \n \n \n \n webhooks\n \n \n \n = 5\n ? \"Maximux(5) Webhook Is Created\"\n : \"\"\n }\n >\n \n }\n style={{ marginRight: 8, height: 42 }}\n onClick={this.webhookDialog}\n disabled={\n this.state.webhookResult.length >= 5\n ? true\n : false\n }\n >\n Create New Webhook\n \n \n \n \n \n <>\n \n \n >\n \n ) : (\n \n \n \n \n \n webhooks\n \n \n \n }\n style={{ marginRight: 8, height: 42 }}\n onClick={this.webhookDialog}\n disabled={!isWhatsappVerified ? true : false}\n >\n Create New Webhook\n \n \n \n \n Waba is not live yet!\n \n \n )}\n \n ) : tab === \"Project Api\" ? (\n
\n \n \n \n \n Project API\n \n \n \n \n \n \n Project API Password\n \n \n \n \n \n The password shown here will be lost after being\n copied.\n \n \n \n \n {\n navigator.clipboard.writeText(\n this.state.password\n );\n this.setState({ password: \"\" });\n }}\n placeholder=\"Click Reset to get a new password\"\n fullWidth\n value={this.state.password}\n InputProps={{\n endAdornment: (\n \n \n {\n navigator.clipboard.writeText(\n this.state.password\n );\n this.setState({ password: \"\" });\n this.copySnack();\n }}\n >\n \n \n \n \n ),\n }}\n />\n \n \n {\n if (\n Object.keys(this.state.privateApps).length >\n 0\n ) {\n this.updatePrivateApp();\n } else {\n this.CreatePrivateApp();\n }\n }}\n endIcon={\n this.state.isLoading ? (\n \n ) : (\n \"\"\n )\n }\n >\n {Object.keys(this.state.privateApps).length > 0\n ? \"Reset\"\n : \"Create\"}\n \n \n \n \n \n \n \n \n \n Chat API URL\n \n \n \n \n \n \n WebHook URL\n \n \n \n \n \n Endpoint URL to receive webhook events\n \n \n \n \n {\n if (this.state.chatApiWebhook.webhook_url) {\n this.setState({\n chatApiWebhook: {\n ...this.state.chatApiWebhook,\n webhook_url: event.target.value,\n },\n });\n } else {\n this.setState({\n webhookUrl: event.target.value,\n });\n }\n }}\n // InputProps={{\n // endAdornment: (\n // \n // \n // {\n // navigator.clipboard.writeText(\n // this.state.password\n // );\n // this.copySnack();\n // }}\n // >\n // \n // \n // \n // \n // ),\n // }}\n />\n \n \n {\n if (\n Object.keys(this.state.chatApiWebhook)\n .length > 0\n ) {\n this.updateWebhookUrl();\n } else {\n this.createWebhook();\n }\n }}\n endIcon={\n this.state.isWebhookLoading ? (\n \n ) : (\n \"\"\n )\n }\n >\n {Object.keys(this.state.chatApiWebhook).length >\n 0\n ? \"Update\"\n : \"Create\"}\n \n \n \n \n \n \n \n \n \n Shared Secret\n \n \n \n \n \n \n Shared Secret\n \n \n \n \n \n Shared secret to validate the received webhook\n event\n \n \n \n \n {\n navigator.clipboard.writeText(\n this.state.SharedSecret\n );\n this.setState({\n SharedSecret: \"\",\n });\n }}\n value={this.state.SharedSecret}\n InputProps={{\n endAdornment: (\n \n \n {\n navigator.clipboard.writeText(\n this.state.SharedSecret\n );\n this.setState({\n SharedSecret: \"\",\n });\n this.copySnack();\n }}\n >\n \n \n \n \n ),\n }}\n />\n \n \n {\n if (this.state.chatApiWebhook.shared_secret) {\n this.setState({\n SharedSecret:\n this.state.chatApiWebhook.shared_secret,\n });\n } else {\n this.setState({\n alertType: \"error\",\n alertMsg: \"subscribe to webhook first\",\n });\n }\n }}\n >\n Generate\n \n \n \n \n \n \n ) : tab === \"Meta Ad\" ? (\n
\n \n \n ) : (\n tab === \"Defaults Config\" && (\n
\n \n \n )\n )}\n
\n \n \n \n \n {this.state.openPlanDialog && (\n
\n )}\n {this.state.openWccChangeDialog && (\n
\n )}\n {this.state.showAgentDialog && (\n
\n )}\n
\n \n \n \n \n \n \n \n \n \n Topics \n \n \n {\n if (reason === \"select-option\") {\n this.setState({ selectedTopic: value });\n }\n }}\n renderInput={(params) => (\n \n )}\n />\n \n \n WebHook Url \n \n \n {\n this.setState({ selectedWebhookurl: e.target.value });\n }}\n />\n \n \n \n \n \n \n \n ) : (\n \"\"\n )\n }\n >\n Submit\n \n \n \n \n
\n \n <>\n \n Generate Project API Key \n \n \n \n \n \n \n Click generate to create Api key and on copy button to copy it\n \n \n \n \n \n \n {\n navigator.clipboard.writeText(\n this.state.appPassword\n );\n this.copySnack();\n }}\n >\n \n \n \n \n ),\n }}\n />\n \n \n \n Generate\n \n \n \n >\n \n \n \n \n ) : (\n \"\"\n )\n }\n >\n Close\n \n \n \n \n
\n \n \n \n \n \n \n \n \n \n action \n \n \n {\n if (reason === \"select-option\") {\n this.setState({ wccAction: value });\n }\n }}\n renderInput={(params) => (\n \n )}\n style={{ width: \"calc(100%)\", paddingTop: \"0px\" }}\n />\n \n \n Amount \n \n \n {\n const value = Number(e.target.value ?? 0);\n if (value > 0)\n this.setState({\n wccAmount: value * 100000,\n });\n }}\n />\n \n \n \n \n \n \n \n ) : (\n \"\"\n )\n }\n >\n Submit\n \n \n \n \n
\n \n \n \n \n \n \n \n \n \n Action \n \n \n {\n if (reason === \"select-option\") {\n this.setState({ pwcAction: value });\n }\n }}\n renderInput={(params) => (\n \n )}\n style={{ width: \"calc(100%)\", paddingTop: \"0px\" }}\n />\n \n \n Amount \n \n \n {\n const value = Number(e.target.value ?? 0);\n if (value > 0)\n this.setState({\n pwcAmount: value * 100000,\n });\n }}\n />\n \n \n \n \n \n \n \n ) : (\n \"\"\n )\n }\n >\n Submit\n \n \n \n \n
\n \n Stop Project \n \n \n \n \n \n \n \n \n Are you sure you want disable this project? If you disable\n this project then all of its services will be terminated\n permanently.\n \n \n \n \n \n {\n this.closeHandleDialog();\n }}\n >\n Cancel\n \n\n \n ) : (\n \"\"\n )\n }\n >\n Stop Plan\n \n \n \n \n
{\n // this.editToggle(\"deleteProject\");\n // }}\n style={{ zIndex: 3002 }}\n >\n \n \n \n \n Are you sure that you want to delete this project?\n \n \n\n \n {this.state.deleteMessage}\n \n \n \n ) : (\n \"\"\n )\n }\n >\n Confirm\n \n {\n this.editToggle(\"deleteProject\");\n\n this.setState({\n deleteMessage: null,\n });\n }}\n >\n Cancel\n \n \n \n \n \n {this.state.wabaDialog && (\n
\n )}\n\n
\n \n {this.state.alertMsg}\n \n \n
\n );\n }\n}\n\nconst styles = (theme) => ({\n root: {\n width: \"100%\",\n height: \"100%\",\n background: \"rgb(249,249,249)\",\n boxSizing: \"border-box\",\n overflowX: \"hidden\",\n position: \"relative\",\n },\n tabContainer: {\n background: \"white\",\n boxShadow: \"0 0 12px rgb(171 170 170 / 21%)\",\n },\n tableContainer: {\n background: \"white\",\n borderRadius: 8,\n padding: 24,\n },\n detailContainer: {\n top: \"60px\",\n height: \"calc(100vh - 260px)\",\n width: \"100%\",\n // overflowY: \"scroll\",\n // background: \"red\",\n boxSizing: \"border-box\",\n position: \"sticky\",\n [theme.breakpoints.down(\"sm\")]: {\n // position top + bottombar height + bottom padding + navbar\n height: \"calc(100vh - 70px - 50px - 10px - 50px)\",\n paddingLeft: theme.spacing(1),\n paddingRight: theme.spacing(1),\n boxSizing: \"border-box\",\n },\n },\n tabIndicator: {\n borderRadius: \"3px 3px 0px 0px\",\n height: 3,\n },\n cells: {\n padding: 24,\n color: \"grey\",\n borderBottom: \"1px solid #ecebeb\",\n },\n textField: {\n width: 200,\n marginRight: 8,\n height: 42,\n },\n descriptionTextField: {\n width: 240,\n marginRight: 8,\n whiteSpace: \"pre-line\",\n },\n plandialogContainer: {\n // zIndex: \"3400 !important\",\n \"& .MuiPaper-root\": {\n [theme.breakpoints.down(\"md\")]: {\n width: \"100%\",\n // height: \"100%\",\n // maxHeight: \"100%\",\n margin: \"0\",\n borderRadius: \"0px\",\n },\n [theme.breakpoints.up(\"md\")]: {\n width: \"100%\",\n // height: \"unset\",\n // maxHeight: \"unset\",\n // minHeight: \"100px\",\n margin: \"unset\",\n borderRadius: \"6px\",\n },\n },\n },\n urlWrapper: {\n display: \"flex\",\n alignItems: \"center\",\n background: \"white\",\n overflow: \"hidden\",\n },\n urlLink: {\n border: `1px solid ${theme.palette.primary.main}`,\n overflowX: \"scroll\",\n whiteSpace: \"nowrap\",\n padding: \"9px 16px\",\n fontWeight: \"600\",\n background: \"rgb(70, 20, 134, 0.1)\",\n color: theme.palette.primary.main,\n borderTopLeftRadius: \"8px\",\n borderBottomLeftRadius: \"8px\",\n width: \"200px\",\n fontSize: \"14px\",\n \"&::-webkit-scrollbar\": {\n display: \"none\",\n },\n },\n copyLink: {\n background: theme.palette.primary.main,\n padding: \"10px 16px\",\n color: \"white\",\n fontWeight: \"600\",\n borderTopRightRadius: \"8px\",\n borderBottomRightRadius: \"8px\",\n fontSize: \"14px\",\n \"&:hover\": {\n cursor: \"pointer\",\n opacity: 0.9,\n },\n },\n roundedBox2: {\n // width: \"100%\",\n [theme.breakpoints.down(\"sm\")]: {\n // height: \"auto\",\n borderRadius: \"12px\",\n marginRight: \"0px\",\n },\n height: \"100%\",\n backgroundColor: \"white\",\n border: \"1px black\",\n borderRadius: \"8px\",\n // margin: \"0px 8px\",\n padding: \"1.5em 1.5em 1.5em 1.5em\",\n marginBottom: \"2em\",\n },\n roundedBox3: {\n // height: \"45px\",\n borderRadius: \"12px\",\n backgroundColor: \"rgb(237, 226, 252)\",\n border: \"1px black\",\n margin: \"1em 0em\",\n padding: \"1em 2em\",\n flexWrap: \"nowrap\",\n lineBreak: \"anywhere\",\n },\n container: {\n background: \"rgb(70 20 134/0.1)\",\n borderRadius: \"15px\",\n },\n dialogTopbar: {\n position: \"sticky\",\n top: 0,\n left: 0,\n background: \"white\",\n zIndex: 100,\n },\n});\n\n// export default withStyles(styles)(withRouter(AssistantsDetail));\n\nconst connectAssistantsDetail = connect(\n (state) => ({\n assistants: state.assistants.assistants,\n partner: state.partner.partner,\n planFamilies: state.planFamilies.allPlanFamilies,\n wccPlans: state.wccPlans.allWccPlans,\n }),\n {\n editAssistants,\n fetchPartnerDetails,\n }\n)(AssistantsDetail);\n\nexport default withStyles(styles)(withRouter(connectAssistantsDetail));\n","import React, { Component } from \"react\";\nimport {\n withStyles,\n Grid,\n Button,\n Box,\n Typography,\n IconButton,\n TextField,\n Snackbar,\n Switch,\n CircularProgress,\n} from \"@material-ui/core\";\nimport { ArrowBackOutlined } from \"@material-ui/icons\";\nimport { Autocomplete, Alert } from \"@material-ui/lab\";\nimport axios from \"axios\";\n\nimport { URL } from \"../../config/config\";\nimport { withRouter } from \"react-router-dom\";\n\nconst PLANS = [\n \"FREE\",\n \"LITE\",\n \"PRO\",\n \"PLUS\",\n \"PREMIUM\",\n \"BASIC_MONTHLY_TIER_1\",\n \"BASIC_MONTHLY_TIER_2\",\n \"BASIC_MONTHLY_TIER_3\",\n \"BASIC_MONTHLY_TIER_4\",\n \"BASIC_MONTHLY_TIER_5\",\n \"BASIC_MONTHLY_TIER_6\",\n \"BASIC_MONTHLY_TIER_7\",\n \"BASIC_MONTHLY_TIER_8\",\n \"BASIC_YEARLY_TIER_1\",\n \"BASIC_YEARLY_TIER_2\",\n];\n\nconst PROJECT_STATUS = [\n \"active\",\n \"unpaid\",\n \"terminated\",\n \"pending\",\n \"suspended\",\n \"stopped\",\n \"archived\",\n];\n\nfunction getStatusColor() {\n const templateStatus = \"PENDING\";\n switch (templateStatus) {\n case \"PENDING\":\n return \"grey\";\n case \"APPROVED\":\n return \"#08CF65\";\n case \"REJECTED\":\n return \"#ff0000\";\n default:\n return \"pink\";\n }\n}\n\nclass Assistant extends Component {\n constructor(props) {\n super(props);\n this.state = {\n status: \"PENDING\",\n alert: false,\n alertMsg: \"\",\n alertSeverity: \"\",\n remainingCredit: 0,\n activePlan: this.props.template,\n whatsappNumber: \"\",\n appName: \"\",\n templateCredit: 0,\n apiKey360: \"\",\n cachedAssistant: null,\n projectStatus: \"\",\n // verifiedStatus: null,\n };\n }\n handleInput = (e) => {\n this.setState({ [e.target.name]: parseInt(e.target.value) });\n };\n handleNumberInput = (e) => {\n this.setState({ [e.target.name]: e.target.value });\n };\n updateTemplateCredit = (e) => {\n let template = this.props.templates[this.props.templateIndex];\n // const amount = template.remainingCredit + this.state.remainingCredit*100\n // console.log({status: e.target.checked});\n const amount = this.state.templateCredit * 100000;\n\n const totalTemplateCredit = template.templateCredit\n ? template.templateCredit + amount\n : amount;\n this.setTemplateInDb({\n templateCredit: totalTemplateCredit,\n })\n .then((response) => {\n template.templateCredit = totalTemplateCredit;\n this.props.setTemplate(template);\n this.setState({\n alert: true,\n alertMsg: \"Template credit updated!\",\n alertSeverity: \"success\",\n });\n })\n .catch((error) => {\n this.setState({\n alert: true,\n alertMsg: \"Unable to add template credit\",\n alertSeverity: \"error\",\n });\n });\n };\n updateApiKey = (e) => {\n let template = this.props.templates[this.props.templateIndex];\n // const amount = template.remainingCredit + this.state.remainingCredit*100\n const apiKey360 = this.state.apiKey360;\n\n this.setTemplateInDb({\n apiKey360: apiKey360,\n })\n .then((response) => {\n template.apiKey360 = apiKey360;\n this.props.setTemplate(template);\n this.setState({\n alert: true,\n alertMsg: \"API Key updated!\",\n alertSeverity: \"success\",\n });\n })\n .catch((error) => {\n this.setState({\n alert: true,\n alertMsg: \"Unable to add API Key\",\n alertSeverity: \"error\",\n });\n });\n };\n updateVerificationStatus = (e) => {\n let template = this.props.templates[this.props.templateIndex];\n // const amount = template.remainingCredit + this.state.remainingCredit*100\n // console.log({status: e.target.checked});\n const checked = e.target.checked;\n\n this.setTemplateInDb({\n isWhatsappVerified: checked,\n })\n .then((response) => {\n template.isWhatsappVerified = checked;\n this.props.setTemplate(template);\n this.setState({\n alert: true,\n alertMsg: \"Assistant verified status updated!\",\n alertSeverity: \"success\",\n });\n })\n .catch((error) => {\n this.setState({\n alert: true,\n alertMsg: \"Unable to update verified status\",\n alertSeverity: \"error\",\n });\n });\n };\n updateVerificationStatus = (e) => {\n let template = this.props.templates[this.props.templateIndex];\n // const amount = template.remainingCredit + this.state.remainingCredit*100\n // console.log({status: e.target.checked});\n const checked = e.target.checked;\n\n this.setTemplateInDb({\n isWhatsappVerified: checked,\n })\n .then((response) => {\n template.isWhatsappVerified = checked;\n this.props.setTemplate(template);\n this.setState({\n alert: true,\n alertMsg: \"Assistant verified status updated!\",\n alertSeverity: \"success\",\n });\n })\n .catch((error) => {\n this.setState({\n alert: true,\n alertMsg: \"Unable to update verified status\",\n alertSeverity: \"error\",\n });\n });\n };\n updateWhatsAppNumber = () => {\n let template = this.props.templates[this.props.templateIndex];\n // const amount = template.remainingCredit + this.state.remainingCredit*100\n // console.log({status: e.target.checked});\n const whatsappNumber = this.state.whatsappNumber;\n template.whatsappNumber = whatsappNumber;\n\n this.setTemplateInDb({\n whatsappNumber,\n })\n .then((response) => {\n this.props.setTemplate(template);\n this.setState({\n alert: true,\n alertMsg: \"Whatsapp number updated!\",\n alertSeverity: \"success\",\n });\n })\n .catch((error) => {\n this.setState({\n alert: true,\n alertMsg: \"Unable to update whatsapp number\",\n alertSeverity: \"error\",\n });\n });\n };\n updateAppName = () => {\n let template = this.props.templates[this.props.templateIndex];\n // const amount = template.remainingCredit + this.state.remainingCredit*100\n // console.log({status: e.target.checked});\n const appName = this.state.appName;\n template.appName = appName;\n\n this.setTemplateInDb({\n appName,\n })\n .then((response) => {\n this.props.setTemplate(template);\n this.setState({\n alert: true,\n alertMsg: \"Whatsapp app name updated!\",\n alertSeverity: \"success\",\n });\n })\n .catch((error) => {\n this.setState({\n alert: true,\n alertMsg: \"Unable to update whatsapp app name\",\n alertSeverity: \"error\",\n });\n });\n };\n updateProjectStatus = () => {\n let template = this.props.templates[this.props.templateIndex];\n // const amount = template.remainingCredit + this.state.remainingCredit*100\n // console.log({status: e.target.checked});\n const projectStatus = this.state.projectStatus;\n template.status = projectStatus;\n\n this.setTemplateInDb({\n status: projectStatus,\n })\n .then((response) => {\n this.props.setTemplate(template);\n this.setState({\n alert: true,\n alertMsg: \"Project status updated!\",\n alertSeverity: \"success\",\n });\n })\n .catch((error) => {\n this.setState({\n alert: true,\n alertMsg: \"Unable to update project status\",\n alertSeverity: \"error\",\n });\n });\n };\n updateCredit = () => {\n let template = { ...this.props.templates[this.props.templateIndex] };\n const amount =\n template.remainingCredit + this.state.remainingCredit * 100000;\n const activePlan = this.state.activePlan || template.activePlan;\n // console.log(template.remainingCredit, this.state.remainingCredit)\n // if (this.state.activePlan) {\n const oldPlan = template.activePlan;\n\n // Set plan name\n template.activePlan = activePlan;\n\n // Check plan type\n if (activePlan.includes(\"BASIC_\")) {\n if (!oldPlan.includes(\"BASIC_\")) {\n template.planActivatedOn = null;\n }\n // Check if its renewal or first time\n if (template.planActivatedOn) {\n template.planActivatedOn = new Date(template.planActivatedOn);\n\n // console.log(template.planActivatedOn);\n\n // template.planActivatedOn.setMonth(\n // template.planActivatedOn.getMonth() + 8\n // );\n // console.log(template.planActivatedOn);\n // Get the current date\n const currentDate = template.planActivatedOn.getDate();\n // Set to day 1 to avoid forward\n template.planActivatedOn.setDate(1);\n // Increase month by 1\n template.planActivatedOn.setMonth(\n template.planActivatedOn.getMonth() + 1\n );\n // Get max # of days in this new month\n var daysInMonth = new Date(\n template.planActivatedOn.getYear(),\n template.planActivatedOn.getMonth() + 1,\n 0\n ).getDate();\n // Set the date to the minimum of current date of days in month\n template.planActivatedOn.setDate(Math.min(currentDate, daysInMonth));\n } else {\n template.planActivatedOn = new Date();\n }\n } else {\n // v1 pricing\n template.planActivatedOn = new Date();\n }\n // }\n\n template.remainingCredit = amount;\n\n this.setTemplateInDb({\n remainingCredit: amount,\n activePlan: template.activePlan,\n planActivatedOn: template.planActivatedOn,\n })\n .then((response) => {\n this.props.setTemplate(template);\n this.setState({\n alert: true,\n alertMsg: \"Assistant status updated!\",\n alertSeverity: \"success\",\n });\n })\n .catch((error) => {\n this.setState({\n alert: true,\n alertMsg: \"Unable to update template status\",\n alertSeverity: \"error\",\n });\n });\n };\n handleAutoComplete = (fieldName, value, reason) => {\n if (reason == \"select-option\") {\n this.setState({ [fieldName]: value });\n }\n };\n handleMark = (mark, temp) => {\n const template = { ...temp };\n if (mark) {\n // set process to new Date()\n this.setTemplateInDb({ processed: new Date() })\n .then((response) => {\n template.processed = new Date();\n this.props.setTemplate(template);\n this.setState({\n alert: true,\n alertMsg: \"Template marked processed!\",\n alertSeverity: \"success\",\n });\n })\n .catch((error) => {\n this.setState({\n alert: true,\n alertMsg: \"Unable to mark, Check your internet connection.\",\n alertSeverity: \"error\",\n });\n });\n } else {\n this.setTemplateInDb({ processed: null })\n .then((response) => {\n template.processed = null;\n this.props.setTemplate(template);\n this.setState({\n alert: true,\n alertMsg: \"Template marked un-processed!\",\n alertSeverity: \"success\",\n });\n })\n .catch((error) => {\n this.setState({\n alert: true,\n alertMsg:\n \"Unable to fulfill request, Check your internet connection.\",\n alertSeverity: \"error\",\n });\n });\n }\n };\n setTemplateInDb = (fields) => {\n // try {\n const t = this.props.templates[this.props.templateIndex];\n const queryObj = {\n assistantId: t._id,\n fields,\n dailyTemplateLimit: t.dailyTemplateLimit,\n };\n return axios.post(URL + \"/superadmin/set-assistant-field\", { ...queryObj });\n };\n closeSnackbar = () => {\n this.setState({ alert: false, alertMsg: \"\", alertSeverity: \"\" });\n };\n getAssistantDetailsFromCache = async () => {\n try {\n const { templateIndex, templates } = this.props;\n const assistant = templates[templateIndex];\n if (!assistant) {\n return;\n }\n const redisKey = \"assistants/\" + assistant._id;\n\n const { data: cachedAssistant } = await axios.post(\n URL + \"/superadmin/get-credits-from-cache\",\n {\n assistantId: assistant._id,\n }\n );\n\n this.setState({ cachedAssistant });\n } catch (err) {\n console.log(err);\n }\n };\n syncTemplateMsg = async () => {\n try {\n const { templateIndex, templates } = this.props;\n const assistant = templates[templateIndex];\n if (!assistant) {\n return;\n }\n const { data: cachedAssistant } = await axios.post(\n URL + \"/superadmin/sync-template-messages\",\n {\n assistantId: assistant._id,\n }\n );\n\n this.setState({\n alert: true,\n alertMsg: \"Template msg synched\",\n alertSeverity: \"success\",\n });\n } catch (err) {\n console.log(err);\n this.setState({\n alert: true,\n alertMsg: \"Unable update template msg\",\n alertSeverity: \"error\",\n });\n }\n };\n migrateTemplateMsg = async () => {\n try {\n const { templateIndex, templates } = this.props;\n const assistant = templates[templateIndex];\n if (!assistant) {\n return;\n }\n const { data: cachedAssistant } = await axios.post(\n URL + \"/superadmin/migrate-all-templates\",\n {\n assistantId: assistant._id,\n }\n );\n\n this.setState({\n alert: true,\n alertMsg: \"Template msg migrated\",\n alertSeverity: \"success\",\n });\n } catch (err) {\n console.log(err);\n this.setState({\n alert: true,\n alertMsg: \"Unable migrate template msg\",\n alertSeverity: \"error\",\n });\n }\n };\n componentDidMount() {\n this.getAssistantDetailsFromCache();\n }\n render() {\n //changes\n const { classes, toggleDialog, templateIndex, templates, setTemplate } =\n this.props || {};\n const template = templates?.[templateIndex];\n return this.props?.templates ? (\n \n \n \n \n \n \n \n {template.assistantName} \n \n \n
\n \n \n \n \n \n \n \n \n Remaining Credit\n \n \n For v3, Keep the excessMac reading as New Plan will\n reset it\n \n \n Default currency in INR for Dollar Billing contact\n developers\n \n Max Agents 5 for v3 pricing \n 35p template msg pricing for v3 pricing \n \n v3 monthly pricing & renewal can be done from here\n \n \n v3 yearly plan can be added but can't be renewed from\n here\n \n To upgrade monthly to yearly - Contact devs \n \n \n \n {template.remainingCredit / 100000}\n \n \n \n \n \n Add Credit\n \n \n \n \n \n \n \n Change Plan\n \n \n \n option)}\n onChange={(e, value, reason) =>\n this.handleAutoComplete(\"activePlan\", value, reason)\n }\n value={template.activePlan}\n renderInput={(params) => (\n \n )}\n />\n \n \n \n Update\n \n \n \n \n \n \n \n WhatsApp Verification Status\n \n \n \n \n \n \n \n For GS only - WhatsApp Number - Enter number with dial code\n \n \n Number in use - {template.whatsappNumber}\n \n \n \n \n \n \n \n \n Update Number\n \n \n \n \n \n \n \n WhatsApp App Name - Enter name (default to AiSensy)\n \n \n Number in use - {template.appName}\n \n \n \n \n \n \n \n \n Update App Name\n \n \n \n \n \n \n \n Template Credit (v2 & v3) - Enter amount in +/-\n \n \n Current Balance -{\" \"}\n {template.templateCredit\n ? template.templateCredit / 100000\n : 0}\n \n \n \n \n \n \n \n \n Update Template Credit\n \n \n \n \n {\" \"}\n \n \n 360 API Key (Do this at the end)\n \n \n API Key in use - {template.apiKey360}\n \n \n \n \n \n \n \n \n Update Key\n \n \n \n \n \n \n \n Cached Assistant\n \n \n \n
\n {this.state.cachedAssistant\n ? JSON.stringify(this.state.cachedAssistant, null, 2)\n : \"\"}\n \n
\n \n \n \n \n Sync Template Msg ( for migration from GS to 360D only)\n \n \n \n Sync Template Messages\n \n \n \n \n \n Suspend/Activate Project\n \n \n \n option)}\n onChange={(e, value, reason) =>\n this.handleAutoComplete(\"projectStatus\", value, reason)\n }\n value={template.status}\n renderInput={(params) => (\n \n )}\n />\n \n \n \n \n Update Status\n \n \n \n \n \n \n \n Migrate Template Msg ( for migration of new clients only)\n \n \n \n Migrate Template Messages\n \n \n \n \n \n
\n \n \n {this.state.alertMsg}\n \n \n \n ) : (\n \n \n \n );\n }\n}\n\nfunction getFormattedDate(date) {\n const d = new Date(date);\n const options = { year: \"numeric\", month: \"long\", day: \"numeric\" };\n return d.toLocaleDateString(\"en-US\", options);\n}\nconst styles = (theme) => ({\n root: {\n // width: \"100%\",\n paddingTop: 20,\n overflowX: \"hidden\",\n },\n dialogContainer: {\n overflowX: \"hidden\",\n height: \"100%\",\n position: \"relative\",\n color: \"black\",\n },\n dialogTopbar: {\n position: \"sticky\",\n top: 0,\n left: 0,\n background: \"white\",\n zIndex: 100,\n },\n container: {\n background: \"rgb(70 20 134 / 15%)\",\n borderRadius: \"15px\",\n },\n textField: {\n width: 200,\n },\n});\n\nexport default withStyles(styles)(withRouter(Assistant));\n","import React, { Component } from \"react\";\nimport {\n withStyles,\n Grid,\n Box,\n Checkbox,\n Typography,\n Dialog,\n DialogActions,\n DialogContent,\n DialogContentText,\n DialogTitle,\n Button,\n TextField,\n IconButton,\n CircularProgress,\n} from \"@material-ui/core\";\n\nimport clsx from \"clsx\";\nimport AssistantsDetails from \"./AssistantsDetails\";\nimport { withRouter } from \"react-router-dom\";\nimport { connect } from \"react-redux\";\nimport { Skeleton } from \"@material-ui/lab\";\n\nconst leads = [\n {\n name: \"Name\",\n size: 2,\n key: \"agentName\",\n },\n {\n name: \"Project Name\",\n size: 2,\n key: \"assistantName\",\n },\n {\n name: \"Phone Number\",\n size: 2,\n key: \"agentPhone\",\n },\n {\n name: \"Company Name\",\n size: 2,\n key: \"agentCompanyName\",\n },\n {\n name: \"Industry\",\n size: 2,\n key: \"agentIndustry\",\n },\n {\n name: \"Created At\",\n size: 2,\n key: \"createdAt\",\n },\n];\n\nconst wabaLive = [\n {\n name: \"Name\",\n size: 2,\n key: \"agentName\",\n },\n {\n name: \"Project Name\",\n size: 2,\n key: \"assistantName\",\n },\n {\n name: \"Phone Number\",\n size: 2,\n key: \"agentPhone\",\n },\n {\n name: \"WABA Number\",\n size: 2,\n key: \"whatsappNumber\",\n },\n {\n name: \"Company Name\",\n size: 2,\n key: \"agentCompanyName\",\n },\n {\n name: \"Industry\",\n size: 2,\n key: \"agentIndustry\",\n },\n {\n name: \"Created At\",\n size: 2,\n key: \"createdAt\",\n },\n];\n\nconst conversions = [\n {\n name: \"Project Name\",\n key: \"assistantName\",\n size: 2,\n },\n {\n name: \"Plan Purchased\",\n key: \"planPurchased\",\n size: 2,\n },\n {\n name: \"Currency\",\n key: \"currency\",\n size: 2,\n },\n {\n name: \"Amount\",\n key: \"amountPaid\",\n size: 2,\n },\n {\n name: \"Commision Earned\",\n key: \"commision\",\n size: 2,\n },\n {\n name: \"Commision Percentage\",\n key: \"commisionPercent\",\n size: 2,\n },\n {\n name: \"Purchase Date\",\n key: \"createdAt\",\n size: 2,\n },\n];\n\nclass AssistantsTable extends Component {\n constructor(props) {\n super(props);\n this.state = {\n openDialog: false,\n selectedTemplate: null,\n };\n }\n\n listScrolled = (event) => {\n if (!this.offsetPosition) {\n this.offsetPosition = this.rootEle.current.offsetTop;\n }\n const container = event.currentTarget;\n const currentPos = container.scrollTop;\n this.setState({\n sticky: currentPos > this.offsetPosition,\n });\n };\n toggleDialog = () => {\n this.setState({ openDialog: !this.state.openDialog });\n };\n openTemplate = (e) => {\n const templates = [...this.props.templates];\n const foundIndex = templates.findIndex((x) => x._id == e._id);\n this.props.history.push(\"/assistants/\" + templates[foundIndex]._id);\n // this.setState({ openDialog: true, selectedTemplate: foundIndex });\n };\n render() {\n const { classes, templates, tab } = this.props;\n\n const fields =\n tab === \"Signups\" ? leads : tab === \"WABA Live\" ? wabaLive : conversions;\n\n return (\n <>\n \n \n \n \n \n \n \n {fields.map((e) => (\n \n \n \n {e.name}\n \n \n \n ))}\n \n {templates.map((e) => (\n \n {fields.map((field) => (\n this.openTemplate(e)\n // }\n >\n | \n \n ))}\n \n ))}\n \n \n \n \n \n
\n\n \n {/* Assistants Details */}\n \n \n >\n );\n }\n}\n\nfunction Cell({\n classes,\n item,\n field,\n allBusiness,\n openCampaign,\n partner,\n ...props\n}) {\n switch (field) {\n case \"commision\":\n return (\n \n \n \n {item.currency === \"USD\" ? \"$\" : \"₹\"}\n {(item[field] ? item[field] / 100000 : 0).toFixed(2)}\n \n \n \n );\n break;\n\n case \"amountPaid\":\n return (\n \n \n \n {item.currency === \"USD\" ? \"$\" : \"₹\"}\n {(item[field] ? item[field] / 100000 : 0).toFixed(2)}\n \n \n \n );\n break;\n\n case \"commisionPercent\":\n return (\n \n \n \n {item[field] ? item[field] / 100000 : 0}%\n \n \n \n );\n\n case \"displayNameVerified\":\n return (\n \n \n \n {item[field] ? \"Verified\" : \"Not Verified\"}\n \n \n \n );\n\n case \"assistantName\":\n return (\n \n \n {item[field]} \n \n \n );\n break;\n case \"createdAt\":\n return (\n \n \n \n {getFormattedDate(item[field])}\n \n \n \n );\n\n break;\n\n case \"Loading\":\n return (\n \n \n {Array.from(Array(20).keys()).map((i) => {\n return (\n \n \n
\n );\n })}\n \n \n );\n break;\n\n case \"agentPhone\":\n return (\n \n \n \n {partner?.isChildAffiliate\n ? maskPhoneNumber(item[field])\n : item[field]}\n \n \n \n );\n\n default:\n return (\n \n \n {item[field]} \n \n \n );\n }\n}\n\nexport function getFormattedDate(date) {\n const d = new Date(date);\n const options = { year: \"numeric\", month: \"long\", day: \"numeric\" };\n return d.toLocaleDateString(\"en-US\", options);\n}\n\nexport const maskPhoneNumber = (phone) => {\n if (phone) return phone?.replace(/.(?=.{4})/g, \"*\");\n};\n\nconst styles = (theme) => ({\n root: {\n width: \"100%\",\n height: \"100%\",\n // background: \"red\",\n // display: \"flex\",\n background: \"rgb(249,249,249)\",\n boxSizing: \"border-box\",\n overflowX: \"hidden\",\n position: \"relative\",\n // marginBottom: \"5em\",\n // paddingBottom: \"80px\"\n },\n tableWrapper: {\n overflowX: \"auto\",\n width: \"100%\",\n },\n\n tableContainer: {\n marginTop: \"1.5em\",\n minWidth: 1800,\n },\n\n tableItem: {\n background: \"white\",\n marginBottom: \"1em\",\n padding: 4,\n flexWrap: \"nowrap\",\n borderRadius: 8,\n // cursor: \"pointer\",\n // \"&:hover\": {\n // background: \"#efefef\",\n // },\n },\n tableItemNoHover: {\n background: \"white\",\n marginBottom: \"1em\",\n padding: 4,\n flexWrap: \"nowrap\",\n borderRadius: 8,\n },\n tableItem1: {\n marginBottom: \"1em\",\n padding: 4,\n flexWrap: \"nowrap\",\n borderRadius: 8,\n },\n\n tableHeader: {\n background: \"white\",\n padding: 4,\n borderRadius: 8,\n marginBottom: \"1em\",\n flexWrap: \"nowrap\",\n },\n row: {\n background: \"white\",\n display: \"flex\",\n },\n alterRow: {\n background: \"rgb(246,246,246)\",\n },\n dataRow: {\n cursor: \"pointer\",\n transition: \"0.3s\",\n \"&:hover\": {\n background: \"rgb(70 20 134 / 15%)\",\n },\n },\n topbar: {\n background: \"rgb(252,252,252)\",\n position: \"sticky\",\n top: \"0\",\n zIndex: \"100\",\n },\n column: {\n // width: \"4400px\",\n fontWeight: \"normal\",\n height: \"50px\",\n display: \"flex\",\n alignItems: \"center\",\n textAlign: \"center\",\n paddingRight: theme.spacing(2),\n paddingLeft: theme.spacing(2),\n paddingTop: theme.spacing(2),\n paddingBottom: theme.spacing(2),\n boxSizing: \"border-box\",\n },\n});\n\n// export default withStyles(styles)(withRouter(AssistantsTable));\n\nconst connectAssistantsTable = connect((state) => ({\n allBusiness: state.business.allBusiness,\n isLoading: state.assistants.isLoading,\n partner: state.partner.partner,\n}))(AssistantsTable);\n\nexport default withStyles(styles)(withRouter(connectAssistantsTable));\n","export const countries = [\n { code: \"AD\", label: \"Andorra\", phone: \"376\" },\n { code: \"AE\", label: \"United Arab Emirates\", phone: \"971\" },\n { code: \"AF\", label: \"Afghanistan\", phone: \"93\" },\n { code: \"AG\", label: \"Antigua and Barbuda\", phone: \"1-268\" },\n { code: \"AI\", label: \"Anguilla\", phone: \"1-264\" },\n { code: \"AL\", label: \"Albania\", phone: \"355\" },\n { code: \"AM\", label: \"Armenia\", phone: \"374\" },\n { code: \"AO\", label: \"Angola\", phone: \"244\" },\n { code: \"AQ\", label: \"Antarctica\", phone: \"672\" },\n { code: \"AR\", label: \"Argentina\", phone: \"54\" },\n { code: \"AS\", label: \"American Samoa\", phone: \"1-684\" },\n { code: \"AT\", label: \"Austria\", phone: \"43\" },\n { code: \"AU\", label: \"Australia\", phone: \"61\", suggested: true },\n { code: \"AW\", label: \"Aruba\", phone: \"297\" },\n { code: \"AX\", label: \"Alland Islands\", phone: \"358\" },\n { code: \"AZ\", label: \"Azerbaijan\", phone: \"994\" },\n { code: \"BA\", label: \"Bosnia and Herzegovina\", phone: \"387\" },\n { code: \"BB\", label: \"Barbados\", phone: \"1-246\" },\n { code: \"BD\", label: \"Bangladesh\", phone: \"880\" },\n { code: \"BE\", label: \"Belgium\", phone: \"32\" },\n { code: \"BF\", label: \"Burkina Faso\", phone: \"226\" },\n { code: \"BG\", label: \"Bulgaria\", phone: \"359\" },\n { code: \"BH\", label: \"Bahrain\", phone: \"973\" },\n { code: \"BI\", label: \"Burundi\", phone: \"257\" },\n { code: \"BJ\", label: \"Benin\", phone: \"229\" },\n { code: \"BL\", label: \"Saint Barthelemy\", phone: \"590\" },\n { code: \"BM\", label: \"Bermuda\", phone: \"1-441\" },\n { code: \"BN\", label: \"Brunei Darussalam\", phone: \"673\" },\n { code: \"BO\", label: \"Bolivia\", phone: \"591\" },\n { code: \"BR\", label: \"Brazil\", phone: \"55\" },\n { code: \"BS\", label: \"Bahamas\", phone: \"1-242\" },\n { code: \"BT\", label: \"Bhutan\", phone: \"975\" },\n { code: \"BV\", label: \"Bouvet Island\", phone: \"47\" },\n { code: \"BW\", label: \"Botswana\", phone: \"267\" },\n { code: \"BY\", label: \"Belarus\", phone: \"375\" },\n { code: \"BZ\", label: \"Belize\", phone: \"501\" },\n { code: \"CA\", label: \"Canada\", phone: \"1\", suggested: true },\n { code: \"CC\", label: \"Cocos (Keeling) Islands\", phone: \"61\" },\n { code: \"CD\", label: \"Congo, Democratic Republic of the\", phone: \"243\" },\n { code: \"CF\", label: \"Central African Republic\", phone: \"236\" },\n { code: \"CG\", label: \"Congo, Republic of the\", phone: \"242\" },\n { code: \"CH\", label: \"Switzerland\", phone: \"41\" },\n { code: \"CI\", label: \"Cote d'Ivoire\", phone: \"225\" },\n { code: \"CK\", label: \"Cook Islands\", phone: \"682\" },\n { code: \"CL\", label: \"Chile\", phone: \"56\" },\n { code: \"CM\", label: \"Cameroon\", phone: \"237\" },\n { code: \"CN\", label: \"China\", phone: \"86\" },\n { code: \"CO\", label: \"Colombia\", phone: \"57\" },\n { code: \"CR\", label: \"Costa Rica\", phone: \"506\" },\n { code: \"CU\", label: \"Cuba\", phone: \"53\" },\n { code: \"CV\", label: \"Cape Verde\", phone: \"238\" },\n { code: \"CW\", label: \"Curacao\", phone: \"599\" },\n { code: \"CX\", label: \"Christmas Island\", phone: \"61\" },\n { code: \"CY\", label: \"Cyprus\", phone: \"357\" },\n { code: \"CZ\", label: \"Czech Republic\", phone: \"420\" },\n { code: \"DE\", label: \"Germany\", phone: \"49\", suggested: true },\n { code: \"DJ\", label: \"Djibouti\", phone: \"253\" },\n { code: \"DK\", label: \"Denmark\", phone: \"45\" },\n { code: \"DM\", label: \"Dominica\", phone: \"1-767\" },\n { code: \"DO\", label: \"Dominican Republic\", phone: \"1-809\" },\n { code: \"DZ\", label: \"Algeria\", phone: \"213\" },\n { code: \"EC\", label: \"Ecuador\", phone: \"593\" },\n { code: \"EE\", label: \"Estonia\", phone: \"372\" },\n { code: \"EG\", label: \"Egypt\", phone: \"20\" },\n { code: \"EH\", label: \"Western Sahara\", phone: \"212\" },\n { code: \"ER\", label: \"Eritrea\", phone: \"291\" },\n { code: \"ES\", label: \"Spain\", phone: \"34\" },\n { code: \"ET\", label: \"Ethiopia\", phone: \"251\" },\n { code: \"FI\", label: \"Finland\", phone: \"358\" },\n { code: \"FJ\", label: \"Fiji\", phone: \"679\" },\n { code: \"FK\", label: \"Falkland Islands (Malvinas)\", phone: \"500\" },\n { code: \"FM\", label: \"Micronesia, Federated States of\", phone: \"691\" },\n { code: \"FO\", label: \"Faroe Islands\", phone: \"298\" },\n { code: \"FR\", label: \"France\", phone: \"33\", suggested: true },\n { code: \"GA\", label: \"Gabon\", phone: \"241\" },\n { code: \"GB\", label: \"United Kingdom\", phone: \"44\" },\n { code: \"GD\", label: \"Grenada\", phone: \"1-473\" },\n { code: \"GE\", label: \"Georgia\", phone: \"995\" },\n { code: \"GF\", label: \"French Guiana\", phone: \"594\" },\n { code: \"GG\", label: \"Guernsey\", phone: \"44\" },\n { code: \"GH\", label: \"Ghana\", phone: \"233\" },\n { code: \"GI\", label: \"Gibraltar\", phone: \"350\" },\n { code: \"GL\", label: \"Greenland\", phone: \"299\" },\n { code: \"GM\", label: \"Gambia\", phone: \"220\" },\n { code: \"GN\", label: \"Guinea\", phone: \"224\" },\n { code: \"GP\", label: \"Guadeloupe\", phone: \"590\" },\n { code: \"GQ\", label: \"Equatorial Guinea\", phone: \"240\" },\n { code: \"GR\", label: \"Greece\", phone: \"30\" },\n {\n code: \"GS\",\n label: \"South Georgia and the South Sandwich Islands\",\n phone: \"500\",\n },\n { code: \"GT\", label: \"Guatemala\", phone: \"502\" },\n { code: \"GU\", label: \"Guam\", phone: \"1-671\" },\n { code: \"GW\", label: \"Guinea-Bissau\", phone: \"245\" },\n { code: \"GY\", label: \"Guyana\", phone: \"592\" },\n { code: \"HK\", label: \"Hong Kong\", phone: \"852\" },\n { code: \"HM\", label: \"Heard Island and McDonald Islands\", phone: \"672\" },\n { code: \"HN\", label: \"Honduras\", phone: \"504\" },\n { code: \"HR\", label: \"Croatia\", phone: \"385\" },\n { code: \"HT\", label: \"Haiti\", phone: \"509\" },\n { code: \"HU\", label: \"Hungary\", phone: \"36\" },\n { code: \"ID\", label: \"Indonesia\", phone: \"62\" },\n { code: \"IE\", label: \"Ireland\", phone: \"353\" },\n { code: \"IL\", label: \"Israel\", phone: \"972\" },\n { code: \"IM\", label: \"Isle of Man\", phone: \"44\" },\n { code: \"IN\", label: \"India\", phone: \"91\" },\n { code: \"IO\", label: \"British Indian Ocean Territory\", phone: \"246\" },\n { code: \"IQ\", label: \"Iraq\", phone: \"964\" },\n { code: \"IR\", label: \"Iran, Islamic Republic of\", phone: \"98\" },\n { code: \"IS\", label: \"Iceland\", phone: \"354\" },\n { code: \"IT\", label: \"Italy\", phone: \"39\" },\n { code: \"JE\", label: \"Jersey\", phone: \"44\" },\n { code: \"JM\", label: \"Jamaica\", phone: \"1-876\" },\n { code: \"JO\", label: \"Jordan\", phone: \"962\" },\n { code: \"JP\", label: \"Japan\", phone: \"81\", suggested: true },\n { code: \"KE\", label: \"Kenya\", phone: \"254\" },\n { code: \"KG\", label: \"Kyrgyzstan\", phone: \"996\" },\n { code: \"KH\", label: \"Cambodia\", phone: \"855\" },\n { code: \"KI\", label: \"Kiribati\", phone: \"686\" },\n { code: \"KM\", label: \"Comoros\", phone: \"269\" },\n { code: \"KN\", label: \"Saint Kitts and Nevis\", phone: \"1-869\" },\n {\n code: \"KP\",\n label: \"Korea, Democratic People's Republic of\",\n phone: \"850\",\n },\n { code: \"KR\", label: \"Korea, Republic of\", phone: \"82\" },\n { code: \"KW\", label: \"Kuwait\", phone: \"965\" },\n { code: \"KY\", label: \"Cayman Islands\", phone: \"1-345\" },\n { code: \"KZ\", label: \"Kazakhstan\", phone: \"7\" },\n { code: \"LA\", label: \"Lao People's Democratic Republic\", phone: \"856\" },\n { code: \"LB\", label: \"Lebanon\", phone: \"961\" },\n { code: \"LC\", label: \"Saint Lucia\", phone: \"1-758\" },\n { code: \"LI\", label: \"Liechtenstein\", phone: \"423\" },\n { code: \"LK\", label: \"Sri Lanka\", phone: \"94\" },\n { code: \"LR\", label: \"Liberia\", phone: \"231\" },\n { code: \"LS\", label: \"Lesotho\", phone: \"266\" },\n { code: \"LT\", label: \"Lithuania\", phone: \"370\" },\n { code: \"LU\", label: \"Luxembourg\", phone: \"352\" },\n { code: \"LV\", label: \"Latvia\", phone: \"371\" },\n { code: \"LY\", label: \"Libya\", phone: \"218\" },\n { code: \"MA\", label: \"Morocco\", phone: \"212\" },\n { code: \"MC\", label: \"Monaco\", phone: \"377\" },\n { code: \"MD\", label: \"Moldova, Republic of\", phone: \"373\" },\n { code: \"ME\", label: \"Montenegro\", phone: \"382\" },\n { code: \"MF\", label: \"Saint Martin (French part)\", phone: \"590\" },\n { code: \"MG\", label: \"Madagascar\", phone: \"261\" },\n { code: \"MH\", label: \"Marshall Islands\", phone: \"692\" },\n {\n code: \"MK\",\n label: \"Macedonia, the Former Yugoslav Republic of\",\n phone: \"389\",\n },\n { code: \"ML\", label: \"Mali\", phone: \"223\" },\n { code: \"MM\", label: \"Myanmar\", phone: \"95\" },\n { code: \"MN\", label: \"Mongolia\", phone: \"976\" },\n { code: \"MO\", label: \"Macao\", phone: \"853\" },\n { code: \"MP\", label: \"Northern Mariana Islands\", phone: \"1-670\" },\n { code: \"MQ\", label: \"Martinique\", phone: \"596\" },\n { code: \"MR\", label: \"Mauritania\", phone: \"222\" },\n { code: \"MS\", label: \"Montserrat\", phone: \"1-664\" },\n { code: \"MT\", label: \"Malta\", phone: \"356\" },\n { code: \"MU\", label: \"Mauritius\", phone: \"230\" },\n { code: \"MV\", label: \"Maldives\", phone: \"960\" },\n { code: \"MW\", label: \"Malawi\", phone: \"265\" },\n { code: \"MX\", label: \"Mexico\", phone: \"52\" },\n { code: \"MY\", label: \"Malaysia\", phone: \"60\" },\n { code: \"MZ\", label: \"Mozambique\", phone: \"258\" },\n { code: \"NA\", label: \"Namibia\", phone: \"264\" },\n { code: \"NC\", label: \"New Caledonia\", phone: \"687\" },\n { code: \"NE\", label: \"Niger\", phone: \"227\" },\n { code: \"NF\", label: \"Norfolk Island\", phone: \"672\" },\n { code: \"NG\", label: \"Nigeria\", phone: \"234\" },\n { code: \"NI\", label: \"Nicaragua\", phone: \"505\" },\n { code: \"NL\", label: \"Netherlands\", phone: \"31\" },\n { code: \"NO\", label: \"Norway\", phone: \"47\" },\n { code: \"NP\", label: \"Nepal\", phone: \"977\" },\n { code: \"NR\", label: \"Nauru\", phone: \"674\" },\n { code: \"NU\", label: \"Niue\", phone: \"683\" },\n { code: \"NZ\", label: \"New Zealand\", phone: \"64\" },\n { code: \"OM\", label: \"Oman\", phone: \"968\" },\n { code: \"PA\", label: \"Panama\", phone: \"507\" },\n { code: \"PE\", label: \"Peru\", phone: \"51\" },\n { code: \"PF\", label: \"French Polynesia\", phone: \"689\" },\n { code: \"PG\", label: \"Papua New Guinea\", phone: \"675\" },\n { code: \"PH\", label: \"Philippines\", phone: \"63\" },\n { code: \"PK\", label: \"Pakistan\", phone: \"92\" },\n { code: \"PL\", label: \"Poland\", phone: \"48\" },\n { code: \"PM\", label: \"Saint Pierre and Miquelon\", phone: \"508\" },\n { code: \"PN\", label: \"Pitcairn\", phone: \"870\" },\n { code: \"PR\", label: \"Puerto Rico\", phone: \"1\" },\n { code: \"PS\", label: \"Palestine, State of\", phone: \"970\" },\n { code: \"PT\", label: \"Portugal\", phone: \"351\" },\n { code: \"PW\", label: \"Palau\", phone: \"680\" },\n { code: \"PY\", label: \"Paraguay\", phone: \"595\" },\n { code: \"QA\", label: \"Qatar\", phone: \"974\" },\n { code: \"RE\", label: \"Reunion\", phone: \"262\" },\n { code: \"RO\", label: \"Romania\", phone: \"40\" },\n { code: \"RS\", label: \"Serbia\", phone: \"381\" },\n { code: \"RU\", label: \"Russian Federation\", phone: \"7\" },\n { code: \"RW\", label: \"Rwanda\", phone: \"250\" },\n { code: \"SA\", label: \"Saudi Arabia\", phone: \"966\" },\n { code: \"SB\", label: \"Solomon Islands\", phone: \"677\" },\n { code: \"SC\", label: \"Seychelles\", phone: \"248\" },\n { code: \"SD\", label: \"Sudan\", phone: \"249\" },\n { code: \"SE\", label: \"Sweden\", phone: \"46\" },\n { code: \"SG\", label: \"Singapore\", phone: \"65\" },\n { code: \"SH\", label: \"Saint Helena\", phone: \"290\" },\n { code: \"SI\", label: \"Slovenia\", phone: \"386\" },\n { code: \"SJ\", label: \"Svalbard and Jan Mayen\", phone: \"47\" },\n { code: \"SK\", label: \"Slovakia\", phone: \"421\" },\n { code: \"SL\", label: \"Sierra Leone\", phone: \"232\" },\n { code: \"SM\", label: \"San Marino\", phone: \"378\" },\n { code: \"SN\", label: \"Senegal\", phone: \"221\" },\n { code: \"SO\", label: \"Somalia\", phone: \"252\" },\n { code: \"SR\", label: \"Suriname\", phone: \"597\" },\n { code: \"SS\", label: \"South Sudan\", phone: \"211\" },\n { code: \"ST\", label: \"Sao Tome and Principe\", phone: \"239\" },\n { code: \"SV\", label: \"El Salvador\", phone: \"503\" },\n { code: \"SX\", label: \"Sint Maarten (Dutch part)\", phone: \"1-721\" },\n { code: \"SY\", label: \"Syrian Arab Republic\", phone: \"963\" },\n { code: \"SZ\", label: \"Swaziland\", phone: \"268\" },\n { code: \"TC\", label: \"Turks and Caicos Islands\", phone: \"1-649\" },\n { code: \"TD\", label: \"Chad\", phone: \"235\" },\n { code: \"TF\", label: \"French Southern Territories\", phone: \"262\" },\n { code: \"TG\", label: \"Togo\", phone: \"228\" },\n { code: \"TH\", label: \"Thailand\", phone: \"66\" },\n { code: \"TJ\", label: \"Tajikistan\", phone: \"992\" },\n { code: \"TK\", label: \"Tokelau\", phone: \"690\" },\n { code: \"TL\", label: \"Timor-Leste\", phone: \"670\" },\n { code: \"TM\", label: \"Turkmenistan\", phone: \"993\" },\n { code: \"TN\", label: \"Tunisia\", phone: \"216\" },\n { code: \"TO\", label: \"Tonga\", phone: \"676\" },\n { code: \"TR\", label: \"Turkey\", phone: \"90\" },\n { code: \"TT\", label: \"Trinidad and Tobago\", phone: \"1-868\" },\n { code: \"TV\", label: \"Tuvalu\", phone: \"688\" },\n { code: \"TW\", label: \"Taiwan, Province of China\", phone: \"886\" },\n { code: \"TZ\", label: \"United Republic of Tanzania\", phone: \"255\" },\n { code: \"UA\", label: \"Ukraine\", phone: \"380\" },\n { code: \"UG\", label: \"Uganda\", phone: \"256\" },\n { code: \"US\", label: \"United States\", phone: \"1\", suggested: true },\n { code: \"UY\", label: \"Uruguay\", phone: \"598\" },\n { code: \"UZ\", label: \"Uzbekistan\", phone: \"998\" },\n { code: \"VA\", label: \"Holy See (Vatican City State)\", phone: \"379\" },\n { code: \"VC\", label: \"Saint Vincent and the Grenadines\", phone: \"1-784\" },\n { code: \"VE\", label: \"Venezuela\", phone: \"58\" },\n { code: \"VG\", label: \"British Virgin Islands\", phone: \"1-284\" },\n { code: \"VI\", label: \"US Virgin Islands\", phone: \"1-340\" },\n { code: \"VN\", label: \"Vietnam\", phone: \"84\" },\n { code: \"VU\", label: \"Vanuatu\", phone: \"678\" },\n { code: \"WF\", label: \"Wallis and Futuna\", phone: \"681\" },\n { code: \"WS\", label: \"Samoa\", phone: \"685\" },\n { code: \"XK\", label: \"Kosovo\", phone: \"383\" },\n { code: \"YE\", label: \"Yemen\", phone: \"967\" },\n { code: \"YT\", label: \"Mayotte\", phone: \"262\" },\n { code: \"ZA\", label: \"South Africa\", phone: \"27\" },\n { code: \"ZM\", label: \"Zambia\", phone: \"260\" },\n { code: \"ZW\", label: \"Zimbabwe\", phone: \"263\" },\n];\n","import React, { useState } from \"react\";\nimport {\n Grid,\n TextField,\n Typography,\n Button,\n IconButton,\n DialogContent,\n Dialog,\n DialogActions,\n CircularProgress,\n withStyles,\n Checkbox,\n} from \"@material-ui/core\";\nimport { Autocomplete } from \"@material-ui/lab\";\nimport { Clear } from \"@material-ui/icons\";\nimport {\n CountryDropdown,\n RegionDropdown,\n CountryRegionData,\n} from \"react-country-region-selector-material-ui-new\";\nimport { countries } from \"../countries/countries.js\";\nimport { DialogTitle } from \"@material-ui/core\";\n\nfunction CreateProjectDialog(props) {\n const {\n showDialog,\n closeDialog,\n classes,\n allBusiness,\n selectBusiness,\n setProjectName,\n createBusiness,\n onSubmit,\n disableSubmit,\n reqId4Business,\n planFamilies,\n defaultPlanFamily,\n defaultPlanName,\n wccPlans,\n defaultWccPlan,\n type,\n } = props;\n const states = [\n \"Andaman and Nicobar Islands\",\n \"Andhra Pradesh\",\n \"Arunachal Pradesh\",\n \"Assam\",\n \"Bihar\",\n \"Chandigarh\",\n \"Chhattisgarh\",\n \"Dadra and Nagar Haveli\",\n \"Daman and Diu\",\n \"Delhi\",\n \"Foreign Country\",\n \"Goa\",\n \"Gujarat\",\n \"Haryana\",\n \"Himachal Pradesh\",\n \"Jammu and Kashmir\",\n \"Jharkhand\",\n \"Karnataka\",\n \"Kerala\",\n \"Ladakh\",\n \"Lakshadweep\",\n \"Madhya Pradesh\",\n \"Maharashtra\",\n \"Manipur\",\n \"Meghalaya\",\n \"Mizoram\",\n \"Nagaland\",\n \"Odisha\",\n \"Other Territory\",\n \"Puducherry\",\n \"Punjab\",\n \"Rajasthan\",\n \"Sikkim\",\n \"Tamil Nadu\",\n \"Telangana\",\n \"Tripura\",\n \"Uttar Pradesh\",\n \"Uttarakhand\",\n \"West Bengal\",\n ];\n\n // const [country, setCountry] = useState(\"\");\n const [billingAddress, setBillingAddress] = useState({\n country: \"\",\n street: \"\",\n state: \"\",\n city: \"\",\n zip: \"\",\n });\n const [shippingAddress, setShippingAddress] = useState({\n country: \"\",\n street: \"\",\n state: \"\",\n city: \"\",\n zip: \"\",\n });\n const [gstNumber, setGstNumber] = useState(\"\");\n const [checked, setChecked] = useState(true);\n const [isLoading, setIsLoading] = useState(false);\n\n const copyBillingAddress = (event) => {\n setChecked(event.target.checked);\n if (event.target.checked) {\n setShippingAddress(billingAddress);\n }\n };\n\n const countryToFlag = (isoCode) => {\n return typeof String.fromCodePoint !== \"undefined\"\n ? isoCode\n .toUpperCase()\n .replace(/./g, (char) =>\n String.fromCodePoint(char.charCodeAt(0) + 127397)\n )\n : isoCode;\n };\n\n const defualtBusiness = Object.values(allBusiness).filter((item) => {\n if (item._id === reqId4Business) {\n return item;\n }\n })[0];\n console.log(Object.keys(planFamilies)[0]);\n // <--- Set Plan Family and Plan Name --->\n let [newFamilyId, newPlanId] = [defaultPlanFamily, defaultPlanName];\n if (type !== \"DIRECT\" && (!defaultPlanFamily || !defaultPlanName)) {\n newFamilyId = Object.keys(planFamilies)[0];\n const plans = planFamilies[newFamilyId].plans;\n newPlanId = Object.keys(plans)[0];\n }\n const [plan, setActivePlan] = useState({\n newFamilyId,\n newPlanId,\n });\n console.log(plan);\n const plans = type !== \"DIRECT\" && planFamilies[plan.newFamilyId].plans;\n // <--- Set Wcc Plan --->\n const [wccPlan, setWccPlan] = useState(\n type !== \"DIRECT\" && (defaultWccPlan ?? Object.keys(wccPlans)[0])\n );\n const [name, setName] = useState(selectBusiness ?? \"\");\n console.log(props.partner);\n return (\n \n \n \n Create project \n \n \n \n \n \n \n \n {/* Header */}\n\n {/* Body */}\n \n {/* \n Business \n \n \n {\n return (\n (item.company || item.displayName) + \" - \" + item.name\n );\n })\n }\n // value={this.state.selectBusiness}\n onChange={selectBusiness}\n defaultValue={\n defualtBusiness &&\n (defualtBusiness.company || defualtBusiness.displayName) +\n \" - \" +\n defualtBusiness.name\n }\n renderInput={(params) => (\n \n )}\n style={{ width: \"calc(100%)\", paddingTop: \"0px\" }}\n />\n */}\n\n {/* Body */}\n \n \n \n Business \n \n \n {\n return item.company + \" - \" + item.name;\n })\n }\n // value={this.state.selectBusiness}\n onChange={selectBusiness}\n defaultValue={\n defualtBusiness &&\n defualtBusiness.company + \" - \" + defualtBusiness.name\n }\n renderInput={(params) => (\n \n )}\n style={{ width: \"calc(100%)\", paddingTop: \"0px\" }}\n />\n \n\n \n Project Name \n \n \n setName(e.target.value)}\n fullWidth\n disabled={\n allBusiness && Object.values(allBusiness).length === 0\n }\n />\n \n {type !== \"DIRECT\" && (\n <>\n \n Plan Family \n \n \n option\n )}\n getOptionLabel={(option) => {\n return planFamilies && planFamilies[option].name;\n }}\n onChange={(e, value, reason) => {\n if (reason === \"select-option\") {\n setActivePlan((prev) => ({\n ...prev,\n newFamilyId: value,\n }));\n }\n }}\n defaultValue={plan.newFamilyId}\n renderInput={(params) => (\n \n )}\n style={{ width: \"calc(100%)\", paddingTop: \"0px\" }}\n />\n \n \n Plan \n \n \n option)\n }\n getOptionLabel={(option) => {\n return plans && plans[option]?.name;\n }}\n onChange={(e, value, reason) => {\n if (reason === \"select-option\") {\n setActivePlan((prev) => ({\n ...prev,\n newPlanId: value,\n }));\n }\n }}\n defaultValue={plan.newPlanId}\n renderInput={(params) => (\n \n )}\n style={{ width: \"calc(100%)\", paddingTop: \"0px\" }}\n />\n \n \n Wcc Plan Family \n \n \n option)}\n getOptionLabel={(option) => {\n return wccPlans[option]?.name;\n }}\n onChange={(e, value, reason) => {\n if (reason === \"select-option\") {\n setWccPlan(value);\n }\n }}\n defaultValue={wccPlan}\n renderInput={(params) => (\n \n )}\n style={{ width: \"calc(100%)\", paddingTop: \"0px\" }}\n />\n \n >\n )}\n {props?.partner?.isZohoEnabled === true && (\n <>\n \n \n Enter GST Number\n \n \n \n setGstNumber(event.target.value)}\n disabled={isLoading}\n />\n \n \n \n Billing Address\n \n \n \n \n Select Country\n \n \n \n option.label}\n onChange={(e, value, reason) => {\n console.log(e, value, reason);\n setBillingAddress((prev) => ({\n ...prev,\n country: value.label,\n }));\n }}\n renderOption={(option) => (\n \n {countryToFlag(option.code)} \n {option.label} ({option.code}) +{option.phone}\n \n )}\n renderInput={(params) => (\n \n )}\n style={{ width: \"calc(100%)\", paddingTop: \"0px\" }}\n />\n \n\n \n \n Select state\n \n \n \n {billingAddress?.country === \"India\" && (\n option)}\n onChange={(e, value, reason) => {\n setBillingAddress((prev) => ({\n ...prev,\n state: value,\n }));\n }}\n value={billingAddress?.state}\n renderInput={(params) => (\n \n )}\n style={{ width: \"calc(100%)\", paddingTop: \"0px\" }}\n />\n )}\n\n {billingAddress?.country !== \"India\" && (\n \n setBillingAddress((prev) => ({\n ...prev,\n state: e.target.value,\n }))\n }\n className={classes.textFieldRoot}\n // onBlur={() => {\n // trackEvent({\n // event: !isOtpVerified\n // ? \"signup_company_name_filled\"\n // : \"new_signup_company_name_filled\",\n // properties: {\n // company,\n // },\n // });\n // }}\n />\n )}\n \n \n \n Enter Address\n \n \n \n \n setBillingAddress({\n ...billingAddress,\n street: event.target.value,\n })\n }\n disabled={isLoading}\n />\n \n \n \n Enter City\n \n \n \n \n setBillingAddress({\n ...billingAddress,\n city: event.target.value,\n })\n }\n disabled={isLoading}\n />\n \n\n \n \n Enter Zip\n \n \n \n \n setBillingAddress({\n ...billingAddress,\n zip: event.target.value,\n })\n }\n disabled={isLoading}\n />\n \n\n \n \n Shipping Address\n \n \n \n \n \n Use shipping address same as billing address\n \n copyBillingAddress(e)}\n inputProps={{ \"aria-label\": \"primary checkbox\" }}\n />\n \n \n\n {!checked && (\n <>\n \n \n Select Country\n \n \n \n option.label}\n onChange={(e, value, reason) => {\n console.log(e, value, reason);\n setShippingAddress({\n ...shippingAddress,\n country: value.label,\n });\n }}\n renderOption={(option) => (\n \n {this.countryToFlag(option.code)} \n {option.label} ({option.code}) +{option.phone}\n \n )}\n renderInput={(params) => (\n \n )}\n style={{ width: \"calc(100%)\", paddingTop: \"0px\" }}\n />\n \n \n \n Select state\n \n \n \n {shippingAddress?.country === \"India\" && (\n option)}\n onChange={(e, value, reason) => {\n setShippingAddress((prev) => ({\n ...prev,\n state: value,\n }));\n }}\n value={billingAddress?.state}\n renderInput={(params) => (\n \n )}\n style={{ width: \"calc(100%)\", paddingTop: \"0px\" }}\n />\n )}\n\n {billingAddress?.country !== \"India\" && (\n \n setShippingAddress((prev) => ({\n ...prev,\n state: e.target.value,\n }))\n }\n className={classes.textFieldRoot}\n // onBlur={() => {\n // trackEvent({\n // event: !isOtpVerified\n // ? \"signup_company_name_filled\"\n // : \"new_signup_company_name_filled\",\n // properties: {\n // company,\n // },\n // });\n // }}\n />\n )}\n \n \n \n Enter Address\n \n \n \n \n setShippingAddress({\n ...shippingAddress,\n street: event.target.value,\n })\n }\n disabled={isLoading}\n />\n \n \n \n Enter City\n \n \n \n \n setShippingAddress({\n ...shippingAddress,\n city: event.target.value,\n })\n }\n disabled={isLoading}\n />\n \n \n \n Enter Zip\n \n \n \n \n setShippingAddress({\n ...shippingAddress,\n zip: event.target.value,\n })\n }\n disabled={isLoading}\n />\n \n >\n )}\n >\n )}\n \n \n \n \n {/* Buttons here */}\n \n \n \n \n \n Create New Business\n \n \n \n \n onSubmit({\n name,\n planFamilyId: plan.newFamilyId,\n planName: plan.newPlanId,\n wccId: wccPlan,\n gstNumber,\n shippingAddress,\n billingAddress,\n checked,\n })\n }\n disabled={disableSubmit}\n endIcon={disableSubmit && }\n >\n Submit\n \n \n \n \n \n );\n}\n\nconst styles = (theme) => ({\n textField: {\n width: \"100%\",\n height: 42,\n },\n dialogContainer: {\n // zIndex: \"3400 !important\",\n \"& .MuiPaper-root\": {\n [theme.breakpoints.down(\"md\")]: {\n width: \"100%\",\n // height: \"100%\",\n // maxHeight: \"100%\",\n margin: \"0\",\n borderRadius: \"0px\",\n },\n [theme.breakpoints.up(\"md\")]: {\n width: \"100%\",\n // height: \"unset\",\n // maxHeight: \"unset\",\n // minHeight: \"600px\",\n margin: \"unset\",\n borderRadius: \"6px\",\n },\n },\n },\n projectdialogContainer: {\n // zIndex: \"3400 !important\",\n \"& .MuiPaper-root\": {\n [theme.breakpoints.down(\"md\")]: {\n width: \"100%\",\n // height: \"100%\",\n maxHeight: \"100%\",\n margin: \"0\",\n borderRadius: \"0px\",\n },\n [theme.breakpoints.up(\"md\")]: {\n width: \"100%\",\n // height: \"unset\",\n // maxHeight: \"unset\",\n // minHeight: \"100px\",\n margin: \"unset\",\n borderRadius: \"6px\",\n },\n },\n },\n});\n\nexport default withStyles(styles)(CreateProjectDialog);\n","import React, { Component } from \"react\";\nimport { DateRangePicker } from \"react-dates\";\nimport { connect } from \"react-redux\";\nimport moment from \"moment\";\nimport { withRouter } from \"react-router-dom\";\nimport {\n withStyles,\n Button,\n Box,\n Typography,\n Grid,\n IconButton,\n TextField,\n Tabs,\n Tab,\n DialogContent,\n Dialog,\n DialogActions,\n FormControl,\n Select,\n MenuItem,\n InputLabel,\n Snackbar,\n CircularProgress,\n Popper,\n InputAdornment,\n Tooltip,\n} from \"@material-ui/core\";\nimport {\n NavigateNext,\n NavigateBefore,\n Clear,\n FilterList,\n Add,\n Visibility,\n VisibilityOff,\n InfoOutlined,\n CheckCircleOutlined,\n GetApp,\n} from \"@material-ui/icons\";\nimport \"react-dates/initialize\";\nimport \"react-dates/lib/css/_datepicker.css\";\nimport \"../../react-dates-custom.css\";\n\nimport AssistantsTable, { fields, getFormattedDate } from \"./AssistantsTable\";\nimport { TablePagination } from \"@material-ui/core\";\n// import { assistants } from \"../../config/assistants\";\nimport { URL, partnerURL } from \"../../config/config\";\nimport axios from \"axios\";\nimport {\n fetchAssistants,\n loadNextAssistants,\n loadPrevAssistants,\n changeRowsPerPage,\n fetchAssistantsCounts,\n} from \"../../store/assistantState\";\nimport { fetchBusiness } from \"../../store/businessState\";\nimport { Alert, Autocomplete, Skeleton } from \"@material-ui/lab\";\nimport { timeZones } from \"../../config/timezone\";\nimport dialCodes, {\n countryMap,\n isoMap,\n isoToDialCode,\n} from \"../../config/dialCodes\";\nimport Header from \"../../commons/Header/Header\";\nimport CreateProjectDialog from \"../../commons/Dialog/CreateProjectDialog\";\nimport plansHelper from \"../../helpers/plansHelper\";\nimport withMediaQuery from \"../../helpers/mediaQueryHelper\";\n\nclass AssistantsPage extends Component {\n constructor(props) {\n super(props);\n this.state = {\n alertType: \"\",\n alertMsg: \"\",\n rowsPerPage: 15,\n totalCount: null,\n skip: 0,\n timeStamp: new Date(),\n search: \"\",\n searchDone: false,\n templates: [],\n backupTemplates: [],\n backupSearchTemplates: [],\n err: null,\n isLoading: false,\n page: 0,\n applied: {\n createdAt: {\n startDate: null,\n endDate: null,\n focus: null,\n },\n planActivatedOn: {\n startDate: null,\n endDate: null,\n focus: null,\n },\n },\n tab: \"Signups\",\n openDialog: false,\n assistantFilter: {\n filter: \"Signups\",\n sort: \"dec\",\n },\n allCount: 0,\n wabaCount: 0,\n countryCode: isoToDialCode[this.props.partner?.isoCode] || \"+91\",\n liveCount: 0,\n draftCount: 0,\n trialPlanCount: 0,\n monthlyPlanCount: 0,\n yearlyPlanCount: 0,\n deletedCount: 0,\n stoppedCount: 0,\n openProjectDialog: false,\n selectBusiness: \"\",\n newProjectName: \"\",\n projectSubmitLoader: false,\n clientDialog: false,\n company: \"\",\n companySize: \"\",\n industry: \"\",\n display_name: \"\",\n contact: \"\",\n buttonDisable: false,\n timezone: \"Asia/Calcutta GMT+05:30\",\n currency: this.props.partner?.currency || \"INR\",\n };\n }\n componentDidMount() {\n const { partner } = this.props;\n const { isSuperAffiliate } = partner || {};\n this.props.fetchAssistantsCounts(isSuperAffiliate);\n this.props.fetchAssistants(this.state.assistantFilter, isSuperAffiliate);\n }\n\n toggleDialog = (event, reason) => {\n reason !== \"backdropClick\" &&\n this.setState({ openDialog: !this.state.openDialog });\n };\n\n handleChangePage = (event, newPage) => {\n const { partner } = this.props;\n const { isSuperAffiliate } = partner || {};\n const { page } = this.props;\n const step = newPage - page;\n if (step === -1) {\n this.props.loadPrevAssistants();\n } else if (step === 1) {\n this.props.loadNextAssistants(\n this.state.assistantFilter,\n isSuperAffiliate\n );\n }\n };\n\n handleChangeRowsPerPage = (event) => {\n const rowsPerPage = parseInt(event.target.value, 10);\n this.props.changeRowsPerPage(rowsPerPage);\n this.onTabChange();\n };\n\n setTemplate = (template) => {\n const templates = [...this.state.templates];\n const foundIndex = templates.findIndex((x) => x._id == template._id);\n // console.log({foundIndex})\n // if(foundIndex > -1) {\n templates[foundIndex] = template;\n this.setState({ templates });\n // } else {\n // }\n // templates.forEach\n };\n\n handleBusinessInput = (e) => {\n this.setState({ [e.target.name]: e.target.value });\n };\n\n handleInput = (e) => {\n this.setState({ [e.target.name]: e.target.value });\n };\n\n handleEnter = (e) => {\n // const { backupSearchTemplates, search } = this.state;\n if (e.key === \"Enter\") {\n const { assistantFilter } = this.state;\n const newObj = { ...assistantFilter };\n if (e.target.value) {\n newObj.assistantName = e.target.value;\n } else {\n delete newObj.assistantName;\n }\n this.setState(\n { assistantFilter: newObj, searchDone: true },\n this.onTabChange\n );\n }\n };\n handleSearch = (e) => {\n const { assistantFilter } = this.state;\n const newObj = { ...assistantFilter, assistantName: e.target.value };\n this.setState({\n assistantFilter: newObj,\n search: e.target.value,\n });\n };\n handleClear = () => {\n const { assistantFilter } = this.state;\n delete assistantFilter.assistantName;\n this.setState(\n {\n search: \"\",\n searchDone: false,\n assistantFilter,\n },\n this.onTabChange\n );\n };\n\n // methods for date filter\n clearDate = (filterKey) => {\n const applied = { ...this.state.applied };\n applied[filterKey].startDate = null;\n applied[filterKey].endDate = null;\n this.setState({ applied });\n };\n checkDateInput = (filterKey) => {\n const applied = { ...this.state.applied };\n const s = applied[filterKey];\n if (s.startDate && s.endDate) {\n return;\n } else {\n this.clearDate(filterKey);\n }\n };\n setFocus = (filterKey, focusedInput) => {\n const applied = { ...this.state.applied };\n applied[filterKey].focusedInput = focusedInput;\n this.setState({ applied });\n };\n setDate = (filterKey, startDate, endDate) => {\n const applied = { ...this.state.applied };\n applied[filterKey].startDate = startDate;\n applied[filterKey].endDate = endDate;\n this.setState({ applied });\n };\n applyFilter = () => {\n const { applied, assistantFilter } = this.state;\n const filterObj = { ...assistantFilter };\n if (applied.createdAt.startDate && applied.createdAt.endDate) {\n filterObj.fromDate = applied.createdAt.startDate._d;\n filterObj.toDate = applied.createdAt.endDate._d;\n } else {\n delete filterObj.fromDate;\n delete filterObj.toDate;\n }\n\n if (applied.planActivatedOn.startDate && applied.planActivatedOn.endDate) {\n filterObj.fromPlanActivatedDate = applied.planActivatedOn.startDate._d;\n filterObj.toPlanActivatedDate = applied.planActivatedOn.endDate._d;\n } else {\n delete filterObj.fromPlanActivatedDate;\n delete filterObj.toPlanActivatedDate;\n }\n this.setState(\n {\n assistantFilter: filterObj,\n openDialog: false,\n },\n () => {\n this.onTabChange();\n }\n );\n };\n\n onTabChange = () => {\n const { partner } = this.props;\n const { isSuperAffiliate } = partner || {};\n this.props.fetchAssistants(this.state.assistantFilter, isSuperAffiliate);\n };\n\n projectDialog = (event) => {\n this.setState({ openProjectDialog: !this.state.openProjectDialog });\n };\n\n toggleclientDialog = () => {\n this.setState({ clientDialog: !this.state.clientDialog });\n };\n\n buttondisbale = () => {\n this.setState({ buttonDisable: !this.state.buttonDisable });\n };\n\n createclient = async () => {\n try {\n const partnerId = this.props.user.id;\n const {\n email,\n display_name,\n company,\n contact,\n currency,\n timezone,\n password,\n } = this.state;\n const queryObj = {\n email,\n password,\n display_name,\n company,\n contact,\n timezone,\n currency,\n };\n\n let response = await axios.post(\n partnerURL + `/partner/${partnerId}/business/`,\n queryObj\n );\n // this.props.createnewClient(response.data)\n // this.props.fetchBusiness();\n this.setState({\n clientDialog: !this.state.clientDialog,\n openProjectDialog: true,\n });\n this.setState({\n status: \"success\",\n statusMessage: \"Business Created Succesfully\",\n });\n this.setState({\n email: null,\n display_name: null,\n company: null,\n contact: null,\n currency: this.props.partner?.currency || \"INR\",\n timezone: \"Asia/Calcutta GMT+05:30\",\n countryCode: isoToDialCode[this.props.partner?.isoCode] || \"+91\",\n industry: null,\n companySize: null,\n password: null,\n });\n } catch (error) {\n this.setState({\n buttonDisable: false,\n });\n const errorResponse = error.response;\n console.error(error);\n const errorMessage = errorResponse?.data?.message;\n this.setState({\n status: \"error\",\n statusMessage: errorMessage,\n });\n }\n };\n\n onSnackbarClose = () => {\n this.setState({\n alertType: \"\",\n alertMsg: \"\",\n });\n };\n\n selectBusiness = (e, value, reason) => {\n if (reason === \"select-option\") {\n const assistantValue = Object.keys(this.props.allBusiness).find(\n (key) => this.props.allBusiness[key].name === value.split(\" - \")[1]\n );\n this.setState({ selectBusiness: assistantValue });\n }\n };\n\n render() {\n const {\n classes,\n partner,\n partnerPlanFamily,\n partnerWccFamily,\n mediaQuery,\n partnerId,\n user,\n } = this.props;\n const { backupTemplates, tab, assistantFilter, applied } = this.state;\n const { assistants, page, rowsPerPage, totalAssistants } = this.props;\n\n const affiliateTabs = [\n {\n display: \"Signups\",\n state: \"draftCount\",\n theme: \"#00b4bf59\",\n list: backupTemplates.filter((i) => !i.isWhatsappVerified),\n },\n {\n display: \"WABA Live\",\n state: \"wabaCount\",\n theme: \"#f2c14e3d\",\n list: backupTemplates,\n },\n {\n display: \"Conversions\",\n state: \"liveCount\",\n theme: \"#08cf6533\",\n list: backupTemplates.filter((i) => !!i.isWhatsappVerified),\n },\n ];\n\n const tabs = affiliateTabs;\n\n const passwordRegex = /^(?=.*[A-Za-z])(?=.*\\d)[A-Za-z\\d]{8,}$/;\n const isPasswordValid = passwordRegex.test(this.state.password);\n\n return (\n \n {/* Page name & description container */}\n
\n \n \n }\n // buttonTitle={user.isReferral ? \"\" : \"Create New Project\"}\n onButtonClick={this.projectDialog}\n showSearch={true}\n searchPlaceholder={\"Search project name\"}\n handleSearch={this.handleSearch}\n handleEnter={this.handleEnter}\n searchDone={this.state.searchDone}\n handleClear={this.handleClear}\n filter={this.state.search}\n toggleFilterDialog={this.toggleDialog}\n filterColor={\n (!!applied.createdAt.startDate &&\n !!applied.createdAt.endDate) ||\n (!!applied.planActivatedOn.endDate &&\n !!applied.planActivatedOn.startDate)\n }\n />\n \n {\n if (this.state.tab === newValue) return;\n const tab = tabs.find((i) => i.display === newValue);\n this.setState(\n {\n tab: newValue,\n templates: tab.list,\n totalCount: tab.list.length,\n assistantFilter: {\n ...this.state.assistantFilter,\n filter: newValue,\n },\n },\n this.onTabChange\n );\n }}\n indicatorColor=\"primary\"\n textColor=\"primary\"\n variant=\"scrollable\"\n scrollButtons=\"auto\"\n aria-label=\"scrollable auto tabs example\"\n TabIndicatorProps={{\n className: classes.tabIndicator,\n }}\n >\n {tabs.map((i) => (\n \n \n \n \n {this.props[i.state]}\n \n \n \n \n {`${i.display}`}\n \n >\n }\n value={i.display}\n style={{\n textTransform: \"none\",\n alignItems: \"end\",\n minWidth: 190,\n // marginLeft: 16,\n paddingBottom: 8,\n background: `linear-gradient(180deg, white, ${\n this.state.tab === i.display ? i.theme : \"white\"\n })`,\n }}\n />\n ))}\n \n \n \n \n \n
\n {/* Layout */}\n
\n \n {/* First section */}\n {/* Scrollable (x & y) table container */}\n \n \n \n {/* Table footer fixed at bottom */}\n
\n {/* Filter dialog */}\n
\n \n \n \n Filter \n \n \n \n \n \n \n \n \n \n \n \n \n \n Apply\n \n \n \n \n\n {/* project dialog */}\n {this.state.openProjectDialog && (\n
\n this.setState({ newProjectName: e.target.value })\n }\n createBusiness={() =>\n this.setState({\n openProjectDialog: false,\n clientDialog: true,\n })\n }\n onSubmit={this.projectSubmit}\n disableSubmit={this.state.projectSubmitLoader}\n planFamilies={partnerPlanFamily}\n defaultPlanFamily={partner?.defaultPlanFamily}\n defaultPlanName={partner?.defaultPlanName}\n wccPlans={partnerWccFamily}\n defaultWccPlan={partner?.defaultWccPlan}\n type={partner?.type}\n partner={partner}\n />\n )}\n\n \n \n {this.state.alertMsg}\n \n \n \n );\n }\n}\n\n// Navigattion icons used in date fliters\nfunction NavigationWrapper(props) {\n const { classes } = props;\n return (\n \n {props.children} \n
\n );\n}\n\nfunction DateFilter({\n filterKey,\n applied,\n setFocus,\n setDate,\n checkDateInput,\n clearDate,\n ...props\n}) {\n const { classes, filterName } = props;\n const A = applied[filterKey];\n return (\n \n \n {A.startDate && A.endDate ? (\n \n clearDate(filterKey)}\n >\n \n \n \n ) : (\n \"\"\n )}\n \n \n {filterName}\n \n \n \n checkDateInput(filterKey)}\n startDateId=\"startDate\"\n endDateId=\"endDate\"\n startDate={A.startDate}\n endDate={A.endDate}\n onDatesChange={({ startDate, endDate }) =>\n setDate(filterKey, startDate, endDate)\n }\n focusedInput={A.focusedInput}\n onFocusChange={(focusedInput) => setFocus(filterKey, focusedInput)}\n navPosition=\"navPositionTop\"\n numberOfMonths={1}\n navPrev={\n \n \n \n }\n navNext={\n \n \n \n }\n hideKeyboardShortcutsPanel\n customArrowIcon={null}\n screenReaderInputMessage={\" \"}\n small\n readOnly\n isOutsideRange={(day) => moment().diff(day) < 0}\n />\n \n \n \n );\n\n // function setToday()\n}\n\nconst styles = (theme) => ({\n root: {\n width: \"100%\",\n height: \"100%\",\n // background: \"red\",\n // display: \"flex\",\n background: \"rgb(249,249,249)\",\n boxSizing: \"border-box\",\n overflowX: \"hidden\",\n position: \"relative\",\n // paddingBottom: \"80px\"\n },\n pageTitleContainer: {\n position: \"sticky\",\n zIndex: 100,\n top: 0,\n height: 60,\n boxSizing: \"border-box\",\n [theme.breakpoints.down(\"md\")]: {\n // paddingTop: \"40px\",\n },\n [theme.breakpoints.down(\"sm\")]: {\n // paddingTop: \"20px\",\n },\n },\n fullWidth: {\n width: \"100%\",\n background: \"white\",\n },\n container: {\n background: \"white\",\n borderRadius: \"8px\",\n },\n fixedBottomContainer: {\n position: \"fixed\",\n height: \"60px\",\n borderTop: \"1px solid lightgrey\",\n background: \"white\",\n bottom: \"0\",\n left: \"71px\",\n right: \"0\",\n overflow: \"hidden\",\n display: \"flex\",\n alignItems: \"center\",\n justifyContent: \"center\",\n [theme.breakpoints.down(\"sm\")]: {\n left: \"0px\",\n height: \"50px\",\n },\n },\n tableContainer: {\n top: \"60px\",\n height: \"calc(100vh - 260px)\",\n width: \"100%\",\n overflow: \"hidden\",\n // background: \"red\",\n boxSizing: \"border-box\",\n position: \"sticky\",\n [theme.breakpoints.down(\"xs\")]: {\n // position top + bottombar height + bottom padding + navbar\n marginTop: 48,\n height: \"calc(100vh - 70px - 50px - 10px - 50px)\",\n paddingLeft: theme.spacing(1),\n paddingRight: theme.spacing(1),\n boxSizing: \"border-box\",\n },\n },\n textField: {\n width: \"100%\",\n height: 42,\n },\n sortFormControl: {\n width: 120,\n marginRight: 8,\n height: 32,\n borderRadius: \"4px\",\n border: \"1px solid lightgrey\",\n },\n textFieldPopper: {\n [theme.breakpoints.down(\"md\")]: {\n left: \"12px !important\",\n minWidth: \"calc(100% - 35px)\",\n },\n [theme.breakpoints.up(\"md\")]: {\n left: \"calc(12.5vw - 16px) !important\",\n minWidth: \"calc(25vw + 54px)\",\n },\n },\n tabContainer: {\n background: \"white\",\n boxShadow: \"0 0 12px rgb(171 170 170)\",\n // padding: \"24px\",\n },\n tabIndicator: {\n borderRadius: \"3px 3px 0px 0px\",\n height: 3,\n },\n textFieldRootAutocomplete: {\n width: \"100%\",\n marginTop: \"4px\",\n marginBottom: \"20px\",\n height: 32,\n background: \"white\",\n border: \"2px solid grey\",\n \"& input\": {\n padding: \"0px 12px!important\",\n },\n \"& label\": {\n background: \"white\",\n color: \"#b2b3b3 !important\",\n padding: \"0 6px\",\n marginTop: -4,\n },\n },\n dialogContainer: {\n // zIndex: \"3400 !important\",\n \"& .MuiPaper-root\": {\n [theme.breakpoints.down(\"md\")]: {\n width: \"100%\",\n // height: \"100%\",\n // maxHeight: \"100%\",\n margin: \"0\",\n borderRadius: \"0px\",\n },\n [theme.breakpoints.up(\"md\")]: {\n width: \"100%\",\n // height: \"unset\",\n // maxHeight: \"unset\",\n // minHeight: \"600px\",\n margin: \"unset\",\n borderRadius: \"6px\",\n },\n },\n },\n projectdialogContainer: {\n // zIndex: \"3400 !important\",\n \"& .MuiPaper-root\": {\n [theme.breakpoints.down(\"md\")]: {\n width: \"100%\",\n // height: \"100%\",\n // maxHeight: \"100%\",\n margin: \"0\",\n borderRadius: \"0px\",\n },\n [theme.breakpoints.up(\"md\")]: {\n width: \"100%\",\n // height: \"unset\",\n // maxHeight: \"unset\",\n // minHeight: \"100px\",\n margin: \"unset\",\n borderRadius: \"6px\",\n },\n },\n },\n\n inputRoot: {\n '&[class*=\"MuiOutlinedInput-root\"].MuiAutocomplete-input': {\n padding: 12,\n },\n '&&[class*=\"MuiOutlinedInput-root\"]': {\n padding: 0,\n },\n },\n popper: {\n width: \"fit-content\",\n zIndex: 10000,\n height: \"10px\",\n },\n snackbar: {\n // marginTop:\"20px\",\n // [theme.breakpoints.down(\"sm\")]: {\n // marginTop:\"40px\",\n // },\n },\n textFieldRootAutocomplete: {\n width: \"100%\",\n padding: \"2px 8px\",\n marginTop: \"4px\",\n marginBottom: \"20px\",\n background: \"white\",\n border: \"2px solid grey\",\n \"& label\": {\n background: \"white\",\n color: \"#b2b3b3 !important\",\n padding: \"0 6px\",\n marginTop: -4,\n },\n },\n phoneTextFieldRoot: {\n width: \"100%\",\n padding: \"8px 20px\",\n marginTop: \"4px\",\n marginBottom: \"20px\",\n \"& input\": { width: \"calc(100% - 80px)\", marginLeft: \"80px\" },\n background: \"white\",\n border: \"2px solid grey\",\n },\n codeTextFieldRoot: {\n width: \"90px\",\n padding: \"0 0 0 14px\",\n position: \"absolute\",\n zIndex: 1,\n margin: \"10px 0 0 2px\",\n background: \"white\",\n \"& input\": {\n padding: \"10px 0 !important\",\n },\n \"& button\": {\n width: \"12px\",\n position: \"relative\",\n left: \"7px\",\n background: \"white\",\n borderRadius: 0,\n \"&:hover\": {\n background: \"white\",\n },\n },\n \"& div\": {\n paddingRight: \"0px !important\",\n },\n },\n textFieldRoot: {\n width: \"100%\",\n padding: \"8px 20px 10px\",\n marginTop: \"4px\",\n marginBottom: \"20px\",\n background: \"white\",\n border: \"2px solid grey\",\n \"& input\": {\n padding: \"6px 12px!important\",\n },\n \"& label\": {\n background: \"white\",\n color: \"#b2b3b3 !important\",\n padding: \"0 6px\",\n marginTop: -4,\n },\n },\n});\n\nconst connectAssistants = connect(\n (state) => ({\n user: state.login.user,\n assistants: state.assistants.assistants,\n page: state.assistants.page,\n isAssistantsLoading: state.assistants.isLoading,\n assistantError: state.assistants.error,\n rowsPerPage: state.assistants.rowsPerPage,\n totalAssistants: state.assistants.total,\n allCount: state.assistants.allCount,\n liveCount: state.assistants.liveCount,\n wabaCount: state.assistants.wabaCount,\n draftCount: state.assistants.draftCount,\n trialPlanCount: state.assistants.trialPlanCount,\n monthlyPlanCount: state.assistants.monthlyPlanCount,\n yearlyPlanCount: state.assistants.yearlyPlanCount,\n deletedCount: state.assistants.deletedCount,\n stoppedCount: state.assistants.stoppedCount,\n allBusiness: state.business.allBusiness,\n partnerId: state.partner.partner._id,\n partner: state.partner.partner,\n partnerSecretHash: state.partner.partner.passwordHash,\n partnerPlanFamily: state.planFamilies.allPlanFamilies,\n partnerWccFamily: state.wccPlans.allWccPlans,\n }),\n {\n fetchAssistants,\n loadNextAssistants,\n loadPrevAssistants,\n changeRowsPerPage,\n fetchAssistantsCounts,\n fetchBusiness,\n }\n)(AssistantsPage);\n\nexport default withStyles(styles)(\n withMediaQuery(\"(max-width:600px)\")(withRouter(connectAssistants))\n);\n","import React, { Component } from \"react\";\nimport { Route, Switch } from \"react-router-dom\";\nimport ErrorPage from \"../../commons/ErrorPage/ErrorPage\";\nimport AssistantsDetail from \"./AssistantsDetail\";\n// import AssistantsDetails from \"./AssistantsDetails\";\nimport AssistantsPage from \"./AssistantsPage\";\n\nclass AssistantsRoute extends Component {\n constructor(props) {\n super(props);\n this.state = {};\n }\n render() {\n return (\n \n \n \n \n \n );\n }\n}\n\nexport default AssistantsRoute;\n","import React, { Component } from \"react\";\nimport axios from \"axios\";\nimport { URL } from \"../../config/config.js\";\nimport {\n withStyles,\n Grid,\n Box,\n Typography,\n CircularProgress,\n} from \"@material-ui/core\";\nimport { connect } from \"react-redux\";\nimport { withRouter } from \"react-router-dom\";\nimport billingsHelper from \"../../helpers/billingsHelper.js\";\nimport Header from \"../../commons/Header/Header.js\";\n\nclass BillingDetails extends Component {\n constructor(props) {\n super(props);\n this.state = {\n commision: 0,\n amountPaid: 0,\n currency: \"\",\n planPurchased: \"\",\n createdAt: \"\",\n isLoading: true,\n assistantName: \"\",\n commisionPercent: 0,\n };\n }\n\n getBillingLog = () => {\n const partnerId = this.props.user.id;\n const _id = this.props.match.params.billingId;\n const { partnerPlanFamily, isPlansLoading, partner } = this.props;\n const { type } = partner || {};\n if (!isPlansLoading)\n axios\n .get(URL + `/api/${partnerId}/partnerbilling-lists/${_id}`)\n .then((response) => {\n const assistantName = response.data[1]?.assistantName;\n const modifiedBill =\n type !== \"DIRECT\"\n ? billingsHelper.mouldBill(response.data[0], partnerPlanFamily)\n : billingsHelper.mouldDirectBill(response.data[0]);\n const { commision, amountPaid, currency, planPurchased, createdAt } =\n modifiedBill;\n\n this.setState({\n commision,\n amountPaid,\n currency,\n planPurchased,\n createdAt,\n isLoading: false,\n assistantName: assistantName,\n });\n })\n .catch((error) => {\n console.error(error);\n });\n };\n //merge error\n componentDidMount() {\n // if (!this.props.location?.state?.log) this.getBillingLog();\n const log = this.props.location?.state;\n\n if (log) {\n const {\n commision,\n amountPaid,\n currency,\n planPurchased,\n createdAt,\n assistantName,\n commisionPercent,\n } = log;\n this.setState({\n commision,\n amountPaid,\n currency,\n planPurchased,\n createdAt,\n isLoading: false,\n assistantName,\n commisionPercent,\n });\n }\n }\n render() {\n const { classes } = this.props;\n const toggleDialog = () => {\n this.props.history.goBack();\n };\n\n const log = this.props.location?.state;\n\n const { currency } = log;\n\n return (\n \n \n <>\n {\" \"}\n {this.state.isLoading ? (\n \n \n \n ) : (\n \n \n \n {this.state.assistantName && (\n \n \n Project Name\n \n \n {this.state.assistantName}\n \n \n )}\n \n \n COMMISION\n \n \n {`${\n currency === \"INR\"\n ? `₹ `\n : currency === \"USD\"\n ? `$ `\n : \"\"\n } ${this.state.commision / 100000}`}\n \n \n \n \n COMMISION PERCENTAGE\n \n \n {this.state.commisionPercent / 100000} {\" %\"}\n \n \n \n \n AMOUNT PAID\n \n \n {`${\n currency === \"INR\"\n ? `₹ `\n : currency === \"USD\"\n ? `$ `\n : \"\"\n } ${this.state.amountPaid / 100000}`}\n \n \n \n \n CURRENCY\n \n \n {this.state.currency}\n \n \n \n \n PLAN PURCHASED\n \n \n {this.state.planPurchased}\n \n \n \n \n Created At\n \n \n {getFormattedDate(this.state.createdAt)}\n \n \n\n \n \n \n
\n )}\n >\n \n );\n }\n}\n\nfunction getFormattedDate(date) {\n const d = new Date(date);\n const options = { year: \"numeric\", month: \"long\", day: \"numeric\" };\n return d.toLocaleDateString(\"en-US\", options);\n}\nconst styles = (theme) => ({\n root: {\n // width: \"100%\",\n paddingTop: 20,\n overflowX: \"hidden\",\n \"& pre\": {\n whiteSpace: \"pre-wrap\",\n // whiteSpace: \"-moz-pre-wrap\",\n // whiteSpace: \"-pre-wrap\",\n // whiteSpace: \"-o-pre-wrap\",\n wordWrap: \"break-word\",\n wordBreak: \"keep-all\",\n },\n },\n dialogContainer: {\n overflowX: \"hidden\",\n height: \"100%\",\n position: \"relative\",\n color: \"black\",\n },\n dialogTopbar: {\n position: \"sticky\",\n top: 0,\n left: 0,\n background: \"white\",\n zIndex: 100,\n },\n container: {\n background: \"rgb(237, 226, 252)\",\n borderRadius: \"15px\",\n },\n textField: {\n width: 200,\n },\n});\nconst billingconnect = connect(\n (state) => ({\n user: state.login.user,\n partnerPlanFamily: state.planFamilies.allPlanFamilies,\n isPlansLoading: state.planFamilies.isLoading,\n partner: state.partner.partner,\n }),\n {}\n)(BillingDetails);\nexport default withStyles(styles)(withRouter(billingconnect));\n","import React, { Component } from \"react\";\nimport { Route, Switch } from \"react-router-dom\";\nimport ErrorPage from \"../../commons/ErrorPage/ErrorPage\";\nimport BillingPage from \"./BillingPage\";\nimport BillingDetails from \"./BillingDetails\";\nclass BillingRoute extends Component {\n render() {\n return (\n \n \n {/* */}\n \n );\n }\n}\n\nexport default BillingRoute;\n","import React, { Component } from \"react\";\nimport {\n withStyles,\n Grid,\n Box,\n Typography,\n Button,\n IconButton,\n Snackbar,\n CircularProgress,\n Dialog,\n DialogContent,\n DialogActions,\n TextField,\n} from \"@material-ui/core\";\nimport {\n FiberManualRecord,\n FileCopyOutlined,\n FilterNone,\n VisibilityOutlined,\n Edit,\n Clear,\n Flag,\n} from \"@material-ui/icons\";\nimport { Alert, Autocomplete } from \"@material-ui/lab\";\nimport Header from \"../../commons/Header/Header\";\nimport { connect } from \"react-redux\";\nimport { getUnavailablePlanFamilies } from \"../../store/planFamilyState\";\nimport { toHaveStyle } from \"@testing-library/jest-dom\";\nimport { URL } from \"../../config/config\";\nimport axios from \"axios\";\nimport plansHelper from \"../../helpers/plansHelper\";\nclass PlanFamilyPage extends Component {\n constructor(props) {\n super(props);\n this.state = {\n alert: false,\n alertMsg: \"\",\n alertSeverity: \"\",\n isLoading: true,\n openPlanDialog: false,\n planFamily: {},\n changePlan: \"\",\n slectedPlans: \"\",\n openSelectPlanDialog: false,\n openWccDialog: false,\n changePlanFamily: \"\",\n wccPlanFamily: {},\n changeWccPlanFamily: \"\",\n planSubmitLoader: false,\n defaultWccPlan: \"\",\n defaultPlanName: \"\",\n defaultPlanFamily: \"\",\n };\n }\n\n componentDidMount() {\n plansHelper.setPlanFamilies();\n plansHelper.setWccPlans();\n this.setState({\n isLoading: false,\n });\n }\n\n planSubmit = () => {\n this.setState({\n planSubmitLoader: true,\n });\n const firstPlan =\n this.state.planFamily &&\n this.state.planFamily[Object.keys(this.state.planFamily)[0]]?.plans;\n const renderFirstPlan = firstPlan && Object.keys(firstPlan)[0];\n\n const updatePlanFamily = this.state.changePlanFamily\n ? this.state.changePlanFamily\n : this.state.defaultPlanFamily\n ? this.state.defaultPlanFamily\n : Object.keys(this.state.planFamily)[0];\n\n const updateDefualtPlan = this.state.changePlan\n ? this.state.changePlan\n : this.state.defaultPlanName\n ? this.state.defaultPlanName\n : renderFirstPlan;\n\n axios\n .patch(URL + `/api/update-partner-config`, {\n defaultPlanFamily: updatePlanFamily,\n defaultPlanName: updateDefualtPlan,\n })\n .then((response) => {\n this.setState({\n alert: true,\n alertSeverity: \"success\",\n alertMsg: \"Plan family updated successfully\",\n planSubmitLoader: false,\n openPlanDialog: false,\n defaultPlanFamily: updatePlanFamily,\n defaultPlanName: updateDefualtPlan,\n });\n })\n .catch((error) => {\n this.setState({\n alert: true,\n alertSeverity: \"error\",\n alertMsg: \"Plan family update failed\",\n planSubmitLoader: false,\n });\n });\n };\n\n wccPlanSubmit = () => {\n this.setState({\n planSubmitLoader: true,\n });\n\n const updateDefaultWcc = this.state.changeWccPlanFamily\n ? this.state.changeWccPlanFamily\n : this.state.defaultWccPlan\n ? this.state.defaultWccPlan\n : Object.keys(this.state?.wccPlanFamily)[0];\n\n axios\n .patch(URL + `/api/update-partner-config`, {\n defaultWccPlan: updateDefaultWcc,\n })\n .then((response) => {\n this.setState({\n alert: true,\n alertSeverity: \"success\",\n alertMsg: \"WCC Plan family updated successfully\",\n planSubmitLoader: false,\n openPlanDialog: false,\n defaultWccPlan: updateDefaultWcc,\n });\n this.props.updateWccFamily({ defaultWccPlan: updateDefaultWcc });\n })\n .catch((error) => {\n this.setState({\n alert: true,\n alertSeverity: \"error\",\n alertMsg: \"WCC Plan family update failed\",\n planSubmitLoader: false,\n });\n });\n };\n\n componentDidUpdate(prevProps) {\n if (this.props !== prevProps) {\n this.setState(\n {\n planFamily: { ...this.props?.planFamilies },\n defaultPlanFamily: this.props?.partner?.defaultPlanFamily\n ? this.props?.partner?.defaultPlanFamily\n : Object.keys(this.props?.planFamilies)[0],\n defaultPlanName: this.props?.partner?.defaultPlanName,\n defaultWccPlan: this.props?.partner?.defaultWccPlan,\n wccPlanFamily: { ...this.props.wccPlans },\n },\n () => {\n this.setState({\n plans:\n this.props?.planFamilies?.[this?.state?.defaultPlanFamily]?.[\n \"plans\"\n ],\n });\n }\n );\n }\n }\n\n closeSnackbar = () => {\n this.setState({ alert: false, alertMsg: \"\", alertSeverity: \"\" });\n };\n\n planDialog = (event, reason) => {\n reason !== \"backdropClick\" &&\n this.setState({ openPlanDialog: !this.state.openPlanDialog });\n };\n\n SelectWccDialog = (event, reason) => {\n reason !== \"backdropClick\" &&\n this.setState({ openWccDialog: !this.state.openWccDialog });\n };\n\n selectPlanDialog = (event, reason) => {\n reason !== \"backdropClick\" &&\n this.setState({ openSelectPlanDialog: !this.state.openSelectPlanDialog });\n };\n\n render() {\n const { classes, match } = this.props;\n const {\n planFamily,\n changePlanFamily,\n changePlan,\n plans,\n changeWccPlanFamily,\n defaultWccPlan,\n wccPlanFamily,\n defaultPlanFamily,\n defaultPlanName,\n } = this.state;\n const firstPlan =\n planFamily && planFamily[Object.keys(planFamily)[0]]?.plans;\n\n const renderFirstPlan = firstPlan && Object.keys(firstPlan)[0];\n\n const renderPlanName =\n planFamily?.[defaultPlanFamily]?.[\"plans\"]?.[defaultPlanName]?.[\"name\"];\n\n return this.state.isLoading ? (\n \n ) : (\n \n
\n
\n
\n \n \n \n \n \n Select Default Subscription Plan Family\n \n \n \n \n \n Active Plan Family\n \n \n {planFamily?.[defaultPlanFamily]?.[\"name\"] ??\n \"Select a plan family\"}\n \n \n \n \n \n \n \n \n \n \n \n Active Plan\n \n \n {renderPlanName ?? \"Select a plan\"}\n \n\n \n \n \n \n \n \n \n \n \n \n \n \n \n Select Default WhatsApp Conversation Charges (WCC) Family\n \n \n \n \n \n Active Plan\n \n \n {wccPlanFamily[defaultWccPlan]?.name ??\n \"Select a plan first\"}\n \n \n \n \n \n \n \n \n \n \n \n \n \n \n
\n\n {this.state.openPlanDialog && (\n
\n \n Select Plan Family \n \n \n \n \n \n \n \n \n Plan Family \n \n \n option)}\n getOptionLabel={(option) => {\n return planFamily[option].name;\n }}\n onChange={(e, value, reason) => {\n if (reason === \"select-option\") {\n this.setState({\n changePlanFamily: value,\n plans: planFamily[value]?.plans,\n });\n }\n }}\n defaultValue={defaultPlanFamily}\n renderInput={(params) => (\n \n )}\n style={{ width: \"calc(100%)\", paddingTop: \"0px\" }}\n />\n \n \n Plans \n \n \n option)\n }\n getOptionLabel={(option) => {\n return plans && plans[option]?.name;\n }}\n onChange={(e, value, reason) => {\n if (reason === \"select-option\") {\n this.setState({ changePlan: value });\n }\n }}\n defaultValue={\n defaultPlanName ? defaultPlanName : renderFirstPlan\n }\n renderInput={(params) => (\n \n )}\n style={{ width: \"calc(100%)\", paddingTop: \"0px\" }}\n />\n \n \n \n \n \n \n (\n this.planDialog(),\n this.setState({\n planSubmitLoader: false,\n })\n )}\n style={{ marginRight: 8 }}\n >\n Cancel\n \n \n ) : (\n \"\"\n )\n }\n >\n Change Plan\n \n \n \n \n )}\n {/* {this.state.openSelectPlanDialog && (\n
\n \n \n \n \n \n \n \n \n \n PLANS \n \n \n option)}\n getOptionLabel={(option) => {\n return plans[option]?.name;\n }}\n onChange={(e, value, reason) => {\n if (reason === \"select-option\") {\n this.setState({ changePlan: value });\n }\n }}\n defaultValue={changePlan ?? renderFirstPlan}\n renderInput={(params) => (\n \n )}\n style={{ width: \"calc(100%)\", paddingTop: \"0px\" }}\n />\n \n \n \n \n \n \n \n Cancel\n \n \n ) : (\n \"\"\n )\n }\n >\n Change Plan\n \n \n \n \n )} */}\n {this.state.openWccDialog && (\n
\n \n Select Wcc Plan Family \n \n \n \n \n \n \n \n \n Plans \n \n \n option\n )}\n getOptionLabel={(option) => {\n return wccPlanFamily[option]?.name;\n }}\n onChange={(e, value, reason) => {\n if (reason === \"select-option\") {\n this.setState({ changeWccPlanFamily: value });\n }\n }}\n defaultValue={\n defaultWccPlan ?? Object.keys(wccPlanFamily)[0]\n }\n renderInput={(params) => (\n \n )}\n style={{ width: \"calc(100%)\", paddingTop: \"0px\" }}\n />\n \n \n \n \n \n \n \n Cancel\n \n \n ) : (\n \"\"\n )\n }\n >\n Change Plan\n \n \n \n \n )}\n\n
\n \n {this.state.alertMsg}\n \n \n
\n );\n }\n}\n\nconst styles = (theme) => ({\n root: {\n width: \"100%\",\n height: \"100%\",\n // background: \"red\",\n // display: \"flex\",\n background: \"rgb(247,247,247)\",\n boxSizing: \"border-box\",\n overflowX: \"hidden\",\n position: \"relative\",\n // paddingBottom: \"80px\"\n },\n pageTitleContainer: {\n position: \"sticky\",\n zIndex: 100,\n top: 0,\n height: 60,\n boxSizing: \"border-box\",\n [theme.breakpoints.down(\"md\")]: {\n // paddingTop: \"40px\",\n },\n [theme.breakpoints.down(\"sm\")]: {\n // paddingTop: \"20px\",\n },\n },\n cells: {\n padding: 24,\n color: \"grey\",\n borderBottom: \"1px solid #ecebeb\",\n },\n fullWidth: {\n width: \"100%\",\n background: \"white\",\n },\n detailContainer: {\n marginTop: \"20px\",\n width: \"100%\",\n boxSizing: \"border-box\",\n [theme.breakpoints.down(\"md\")]: {\n paddingLeft: theme.spacing(1),\n paddingRight: theme.spacing(1),\n boxSizing: \"border-box\",\n },\n },\n primaryBackground: {\n background: \"rgb(70 20 134 / 15%)\",\n },\n container: {\n marginTop: 16,\n background: \"white\",\n borderRadius: 8,\n padding: \"24px 36px\",\n },\n containerPlans: {\n marginTop: 24,\n background: \"white\",\n borderRadius: 8,\n padding: \"24px 36px\",\n },\n containerButton: {\n marginTop: 16,\n background: \"white\",\n borderRadius: 8,\n padding: \"15px 15px\",\n },\n viewOldButton: {\n \"&:hover\": {\n color: \"grey\",\n cursor: \"pointer\",\n },\n },\n plandialogContainer: {\n // zIndex: \"3400 !important\",\n \"& .MuiPaper-root\": {\n [theme.breakpoints.down(\"md\")]: {\n width: \"100%\",\n height: \"unset\",\n maxHeight: \"unset\",\n minHeight: \"100px\",\n margin: \"unset\",\n borderRadius: \"6px\",\n },\n [theme.breakpoints.up(\"md\")]: {\n width: \"100%\",\n height: \"unset\",\n maxHeight: \"unset\",\n minHeight: \"100px\",\n margin: \"unset\",\n borderRadius: \"6px\",\n },\n },\n },\n});\n\nconst connectPlanFamilyPage = connect((state) => ({\n partner: state.partner.partner,\n planFamilies: state.planFamilies.allPlanFamilies,\n wccPlans: state.wccPlans.allWccPlans,\n}))(PlanFamilyPage);\n\nexport default withStyles(styles)(connectPlanFamilyPage);\n","import React from \"react\";\nimport { useHistory, useLocation, useParams, Link } from \"react-router-dom\";\nimport {\n Box,\n Typography,\n withStyles,\n Grid,\n ButtonBase,\n} from \"@material-ui/core\";\nimport { connect } from \"react-redux\";\nimport { renderMetaAdsRoute } from \"../Routes/ProjectRoutes\";\n\nfunction AdsNavBarDesktop(props) {\n const location = useLocation();\n const history = useHistory();\n const params = useParams();\n const { classes, agent, partner } = props;\n const { type } = partner || {};\n\n return (\n \n \n \n Meta Ads \n \n {renderMetaAdsRoute.map((route, index) => {\n let className = \"inactive\";\n if (location.pathname.includes(route.to)) {\n className = \"active\";\n }\n return (\n {\n history.push(\"/meta-ads\" + route.to);\n }}\n >\n \n \n \n {route.icon}\n \n \n \n \n {route.name}\n \n \n \n \n );\n })}\n \n
\n );\n}\n\nconst styles = (theme) => ({\n root: {\n width: \"220px\",\n height: \"100vh\",\n background: \"rgb(255, 255, 255)\",\n borderRight: \"1px solid rgb(240,240,240)\",\n display: \"flex\",\n flexDirection: \"column\",\n justifyContent: \"space-between\",\n color: \"gray\",\n [theme.breakpoints.down(\"md\")]: {\n width: \"220px\",\n },\n [theme.breakpoints.down(\"sm\")]: {\n display: \"none\",\n },\n },\n logo: {\n height: 50,\n marginTop: 15,\n },\n link_inactive: {\n display: \"flex\",\n justifyContent: \"center\",\n alignItems: \"center\",\n height: \"35px\",\n width: \"35px\",\n margin: \"auto\",\n borderRadius: \"50%\",\n cursor: \"pointer\",\n transition: \"0.5s\",\n },\n link_active: {\n display: \"flex\",\n justifyContent: \"center\",\n alignItems: \"center\",\n height: \"35px\",\n width: \"35px\",\n margin: \"auto\",\n borderRadius: \"50%\",\n backgroundColor: \"white\",\n color: \"white\",\n cursor: \"pointer\",\n transition: \"0.5s\",\n },\n icon_active: {\n color: theme.palette.primary.main,\n },\n icon_inactive: {\n color: \"rgb(20,20,20)\",\n },\n name_active: {\n color: theme.palette.primary.main,\n },\n name_inactive: {\n color: \"rgb(20,20,20)\",\n },\n linkButton: {\n margin: \"3px 0em\",\n width: \"100%\",\n borderRadius: \"5px\",\n padding: \"10px\",\n },\n active: {\n backgroundColor: \"#002d621f\",\n },\n inactive: {\n backgroundColor: \"transparent\",\n },\n});\n\nconst connectedAdsNavBarDesktop = connect((state) => ({\n agent: state.login.user,\n partner: state.partner.partner,\n}))(AdsNavBarDesktop);\n\nexport default withStyles(styles)(connectedAdsNavBarDesktop);\n","import React, { Component } from \"react\";\nimport { withStyles, Grid, Typography, Box } from \"@material-ui/core\";\nimport { connect } from \"react-redux\";\nimport { Redirect, Route, Switch } from \"react-router-dom\";\nimport AdsNavBarDesktop from \"./Navbar/AdsNavBarDesktop\";\nimport { renderMetaAdsRoute } from \"./Routes/ProjectRoutes\";\nimport { fetchPartnerDetails } from \"../../store/partnerState\";\n\nclass MetaAdsLayout extends Component {\n constructor(props) {\n super(props);\n this.state = {};\n }\n componentDidMount() {\n this.props.fetchPartnerDetails();\n }\n\n render() {\n const { classes, agent, partner, user } = this.props;\n const { type } = partner || {};\n const baseroute = \"/meta-ads\";\n\n return (\n \n \n \n \n \n \n \n (\n \n )}\n />\n {renderMetaAdsRoute.map((route, index) => {\n return (\n \n );\n })}\n } />\n \n
\n \n \n );\n }\n}\n\nconst styles = (theme) => ({\n root: {\n width: \"100%\",\n height: \"100vh\",\n overflow: \"hidden\",\n boxSizing: \"border-box\",\n },\n routeWrapper: {\n height: \"100vh\",\n overflow: \"hidden\",\n boxSizing: \"border-box\",\n background: \"white\",\n },\n routeContainer: {\n width: \"100%\",\n height: \"100vh\",\n transition: \"1s\",\n overflow: \"hidden\",\n [theme.breakpoints.down(\"sm\")]: {},\n },\n});\n\nconst connectedMetaAdsLayout = connect(\n (state) => ({\n user: state.login.user,\n agent: state.login.user,\n partner: state.partner.partner,\n }),\n { fetchPartnerDetails }\n)(MetaAdsLayout);\n\nexport default withStyles(styles)(connectedMetaAdsLayout);\n","import React, { Component } from \"react\";\nimport {\n withStyles,\n Grid,\n Box,\n Checkbox,\n Typography,\n Dialog,\n DialogActions,\n DialogContent,\n DialogContentText,\n DialogTitle,\n Button,\n TextField,\n IconButton,\n CircularProgress,\n} from \"@material-ui/core\";\n\nimport { withRouter } from \"react-router-dom\";\nimport { Skeleton } from \"@material-ui/lab\";\n\nexport const adfields = [\n {\n name: \"Ad Name\",\n size: 12,\n key: \"adName\",\n },\n {\n name: \"Agent Name\",\n size: 12,\n key: \"affiliateName\",\n },\n {\n name: \"Team Name\",\n size: 12,\n key: \"teamName\",\n },\n {\n name: \"Business Name\",\n size: 12,\n key: \"businessName\",\n },\n {\n name: \"Project Name\",\n size: 12,\n key: \"projectName\",\n },\n {\n name: \"Ad Status\",\n size: 12,\n key: \"adStatus\",\n },\n {\n name: \"Quality Ranking\",\n size: 12,\n key: \"qualityRanking\",\n },\n {\n name: \"Ad Start Date\",\n size: 12,\n key: \"adStartDate\",\n },\n {\n name: \"Ad End Date\",\n size: 12,\n key: \"adStopDate\",\n },\n {\n name: \"Daily Ad Budget\",\n size: 12,\n key: \"dailyBudget\",\n },\n {\n name: \"Impressions\",\n size: 12,\n key: \"impressions\",\n },\n {\n name: \"Reach\",\n size: 12,\n key: \"reach\",\n },\n {\n name: \"Unique Clicks\",\n size: 12,\n key: \"uniqueClicks\",\n },\n {\n name: \"Clicks\",\n size: 12,\n key: \"clicks\",\n },\n {\n name: \"Total Spend\",\n size: 12,\n key: \"totalBudget\",\n },\n {\n name: \"Cost Per Click\",\n size: 12,\n key: \"costPerClick\",\n },\n // {\n // name: \"Messages\",\n // size: 12,\n // key: \"totalMessages\",\n // },\n {\n name: \"New messaging contacts\",\n size: 12,\n key: \"messagingFirstReply\",\n },\n {\n name: \"Messaging conversations started\",\n size: 12,\n key: \"totalMessagingConnection\",\n },\n {\n name: \"Engagement Ranking\",\n size: 12,\n key: \"engagementRateRanking\",\n },\n];\n\nexport const adAccountFields = [\n {\n name: \"Agent Name\",\n size: 12,\n key: \"affiliateName\",\n },\n {\n name: \"Team Name\",\n size: 12,\n key: \"teamName\",\n },\n {\n name: \"Business Name\",\n size: 12,\n key: \"businessName\",\n },\n {\n name: \"Project Name\",\n size: 12,\n key: \"projectName\",\n },\n {\n name: \"Connection Stage\",\n size: 12,\n key: \"connectionStage\",\n },\n {\n name: \"Active Ads\",\n size: 12,\n key: \"activeAds\",\n },\n {\n name: \"Cost per lead\",\n size: 12,\n key: \"costPerLead\",\n },\n {\n name: \"Number of leads\",\n size: 12,\n key: \"numberOfLeads\",\n },\n {\n name: \"Click Through Rate\",\n size: 12,\n key: \"clickThroughRate\",\n },\n // {\n // name: \"Today's spend\",\n // size: 12,\n // key: \"todaySpend\",\n // },\n {\n name: \"Total spend\",\n size: 12,\n key: \"totalSpend\",\n },\n];\n\nclass AdsTable extends Component {\n constructor(props) {\n super(props);\n this.state = {};\n }\n listScrolled = (event) => {\n if (!this.offsetPosition) {\n this.offsetPosition = this.rootEle.current.offsetTop;\n }\n const container = event.currentTarget;\n const currentPos = container.scrollTop;\n this.setState({\n sticky: currentPos > this.offsetPosition,\n });\n };\n\n render() {\n const { classes, ads, adAccounts, isLoading, page } = this.props;\n\n const fields = page == \"Project\" ? adAccountFields : adfields;\n\n const data = page == \"Project\" ? ads : adAccounts;\n\n return (\n <>\n \n \n \n \n \n \n \n {fields.map((e) => (\n \n \n \n {e.name}\n \n \n \n ))}\n \n {ads.map((e) => (\n \n {fields.map((field) => (\n \n | \n \n ))}\n \n ))}\n \n \n \n \n \n
\n >\n );\n }\n}\n\nfunction Cell({ classes, item, field, allAds, allAdAccounts }) {\n switch (field) {\n case \"activeAds\":\n return (\n \n \n {item[field] || 0} \n \n \n );\n break;\n case \"adStartDate\":\n return (\n \n \n \n {getFormattedDate(item[field])}\n \n \n \n );\n\n break;\n case \"adStopDate\":\n return (\n \n \n \n {getFormattedDate(item[field])}\n \n \n \n );\n\n break;\n case \"Loading\":\n return (\n \n \n {Array.from(Array(20).keys()).map((i) => {\n return (\n \n \n
\n );\n })}\n \n \n );\n break;\n\n case \"numberOfLeads\":\n const nol = item?.totalLeadsGenerated || \"0\";\n\n return (\n \n \n {nol} \n \n \n );\n\n break;\n default:\n return (\n \n \n {item[field]} \n \n \n );\n }\n}\n\nexport function getFormattedDate(date) {\n if (date) {\n const d = new Date(date);\n const options = { year: \"numeric\", month: \"long\", day: \"numeric\" };\n return d.toLocaleDateString(\"en-US\", options);\n }\n return \"\";\n}\nconst styles = (theme) => ({\n root: {\n width: \"100%\",\n height: \"100%\",\n // background: \"red\",\n // display: \"flex\",\n background: \"rgb(249,249,249)\",\n boxSizing: \"border-box\",\n overflowX: \"hidden\",\n position: \"relative\",\n // marginBottom: \"5em\",\n // paddingBottom: \"80px\"\n },\n tableWrapper: {\n overflowX: \"auto\",\n width: \"100%\",\n },\n\n tableContainer: {\n marginTop: \"1.5em\",\n minWidth: \"220%\",\n },\n\n tableItem: {\n background: \"white\",\n marginBottom: \"1em\",\n padding: 4,\n flexWrap: \"nowrap\",\n borderRadius: 8,\n cursor: \"pointer\",\n \"&:hover\": {\n background: \"#efefef\",\n },\n },\n tableItemNoHover: {\n background: \"white\",\n marginBottom: \"1em\",\n padding: 4,\n flexWrap: \"nowrap\",\n borderRadius: 8,\n },\n tableItem1: {\n marginBottom: \"1em\",\n padding: 4,\n flexWrap: \"nowrap\",\n borderRadius: 8,\n },\n\n tableHeader: {\n background: \"white\",\n padding: 4,\n borderRadius: 8,\n marginBottom: \"1em\",\n flexWrap: \"nowrap\",\n },\n row: {\n background: \"white\",\n display: \"flex\",\n },\n alterRow: {\n background: \"rgb(246,246,246)\",\n },\n dataRow: {\n cursor: \"pointer\",\n transition: \"0.3s\",\n \"&:hover\": {\n background: \"rgb(70 20 134 / 15%)\",\n },\n },\n topbar: {\n background: \"rgb(252,252,252)\",\n position: \"sticky\",\n top: \"0\",\n zIndex: \"100\",\n },\n column: {\n // width: \"4400px\",\n fontWeight: \"normal\",\n height: \"50px\",\n display: \"flex\",\n alignItems: \"center\",\n textAlign: \"center\",\n paddingRight: theme.spacing(2),\n paddingLeft: theme.spacing(2),\n paddingTop: theme.spacing(2),\n paddingBottom: theme.spacing(2),\n boxSizing: \"border-box\",\n },\n});\n\nexport default withStyles(styles)(AdsTable);\n","import React, { Component, useEffect, useState } from \"react\";\nimport {\n withStyles,\n Grid,\n Box,\n Typography,\n Button,\n IconButton,\n Snackbar,\n CircularProgress,\n Dialog,\n DialogContent,\n DialogActions,\n TextField,\n Switch,\n Tabs,\n Tab,\n} from \"@material-ui/core\";\nimport { DateRangePicker } from \"react-dates\";\nimport { TablePagination } from \"@material-ui/core\";\nimport { URL } from \"../../config/config\";\nimport {\n NavigateNext,\n NavigateBefore,\n FiberManualRecord,\n FileCopyOutlined,\n FilterNone,\n VisibilityOutlined,\n Edit,\n Clear,\n Flag,\n GetApp,\n} from \"@material-ui/icons\";\nimport { Alert, Autocomplete } from \"@material-ui/lab\";\nimport Header from \"../../commons/Header/Header\";\nimport { connect } from \"react-redux\";\nimport axios from \"axios\";\nimport { fetchPartnerDetails } from \"../../store/partnerState\";\nimport { useHistory, useLocation, useParams, Link } from \"react-router-dom\";\nimport AdsTable from \"./table/AdsTable\";\nimport withMediaQuery from \"../../helpers/mediaQueryHelper\";\nimport moment from \"moment\";\n\nconst MetaAds = (props) => {\n const { id: childAffiliateId } = useParams();\n const { classes, mediaQuery } = props;\n const [tab, setTab] = useState(\"All\");\n const [alertObj, setAlertObj] = useState({\n alertType: \"\",\n alertMsg: \"\",\n });\n const [reportLoading, setReportLoading] = useState(false);\n const [isLoading, setIsLoading] = useState(false);\n const [ads, setAds] = useState([]);\n const [counts, setCounts] = useState({});\n const [paginationCount, setPaginationCount] = useState(0);\n const [page, setPage] = useState(\"\");\n const [currentPage, setCurrentPage] = useState(0);\n const [rowsPerPage, setRowsPerPage] = useState(10);\n const [openDialog, setOpenDialog] = useState(false);\n const [applied, setApplied] = useState({\n adStartDate: {\n startDate: null,\n endDate: null,\n focus: null,\n },\n adStopDate: {\n startDate: null,\n endDate: null,\n focus: null,\n },\n });\n const [assistantFilter, setAssistantFilter] = useState({});\n const location = useLocation();\n\n const tabs =\n page == \"Project\"\n ? [\n {\n display: \"All\",\n state: \"allCount\",\n // icon: ,\n theme: \"#00b4bf59\",\n },\n {\n display: \"FB not connected\",\n state: \"fbNotConnectedCount\",\n theme: \"#08cf6533\",\n },\n {\n display: \"Page not selected\",\n state: \"pageNotSelectedCount\",\n // icon: ,\n theme: \"#f2c14e3d\",\n },\n {\n display: \"WA Number not linked\",\n state: \"waNumberNotLinkedCount\",\n // icon: ,\n theme: \"#ff000059\",\n },\n {\n display: \"Connected\",\n state: \"completedCount\",\n theme: \"#08cf6533\",\n },\n ]\n : [\n {\n display: \"All\",\n state: \"all\",\n // icon: ,\n theme: \"#00b4bf59\",\n },\n {\n display: \"Live\",\n state: \"liveCount\",\n theme: \"#08cf6533\",\n },\n {\n display: \"Paused\",\n state: \"pausedCount\",\n // icon: ,\n theme: \"#f2c14e3d\",\n },\n {\n display: \"Stopped\",\n state: \"stoppedCount\",\n // icon: ,\n theme: \"#ff808059\",\n },\n {\n display: \"Others\",\n state: \"othersCount\",\n // icon: ,\n theme: \"#ff000059\",\n },\n ];\n\n const toggleDialog = (event, reason) => {\n reason !== \"backdropClick\" && setOpenDialog(!openDialog);\n };\n\n useEffect(() => {\n const fetchCounts = async () => {\n try {\n const { partner = {} } = props;\n const { isSuperAffiliate } = partner;\n const response = await axios.post(`${URL}/api/meta-ad-counts`, {\n childAffiliateId: childAffiliateId,\n isSuperAffiliate,\n });\n setCounts(response.data[0]);\n if (!assistantFilter?.filter) {\n setPaginationCount(response.data[0][\"all\"]);\n }\n } catch (error) {\n console.error(error);\n }\n };\n\n const fetchProjectAdsInsightData = async () => {\n try {\n const { partner = {} } = props;\n const { isSuperAffiliate } = partner;\n const response = await axios.post(`${URL}/api/meta-ad-account-counts`, {\n childAffiliateId: childAffiliateId,\n isSuperAffiliate,\n });\n setCounts(response.data);\n if (!assistantFilter?.filter) {\n setPaginationCount(response.data.allCount);\n }\n } catch (error) {\n console.error(error);\n }\n };\n\n if (location.pathname.includes(\"project\")) {\n setPage(\"Project\");\n fetchProjectAdsInsightData();\n } else {\n fetchCounts();\n setPage(\"Ads\");\n }\n }, []);\n\n useEffect(() => {\n if (location.pathname.includes(\"project\")) {\n fetchMetaAdAccountLists(assistantFilter);\n } else {\n fetchMetaAdLists(assistantFilter);\n }\n }, [tab, currentPage, rowsPerPage]);\n\n const DownloadReportHandler = async () => {\n if (location.pathname.includes(\"project\")) {\n await fetchMetaAdAccountLists(assistantFilter, \"report\");\n } else {\n await fetchMetaAdLists(assistantFilter, \"report\");\n }\n };\n\n const makeCsvDataReport = (data) => {\n try {\n const headers = Array.from(\n new Set(data.flatMap((entry) => Object.keys(entry)))\n );\n // Convert to CSV\n const csvContent = [\n headers.join(\",\"),\n ...data.map((entry) =>\n headers\n .map((header) =>\n entry[header] !== undefined && entry[header] !== null\n ? `\"${entry[header].toString().replace(/\\n/g, \"\\\\n\")}\"`\n : \"\"\n )\n .join(\",\")\n ),\n ].join(\"\\n\");\n let start;\n let end;\n if (assistantFilter?.filter) {\n start = moment(\n assistantFilter?.fromAdStartDate\n ? assistantFilter.fromAdStartDate\n : assistantFilter.fromAdStopDate,\n \"YYYY-MM-DD\"\n ).isValid()\n ? moment(\n assistantFilter?.fromAdStartDate\n ? assistantFilter.fromAdStartDate\n : assistantFilter.fromAdStopDate\n ).format(\"DD-MM-YYYY\")\n : null;\n end = moment(\n assistantFilter?.toAdStartDate\n ? assistantFilter.toAdStartDate\n : assistantFilter.toAdStopDate,\n \"YYYY-MM-DD\"\n ).isValid()\n ? moment(\n assistantFilter?.toAdStartDate\n ? assistantFilter.toAdStartDate\n : assistantFilter.toAdStopDate\n ).format(\"DD-MM-YYYY\")\n : null;\n }\n\n // Create download link\n var downloadLink = document.createElement(\"a\");\n const blob = new Blob([csvContent], { type: \"text/csv\" });\n var url = window.URL.createObjectURL(blob);\n downloadLink.href = url;\n downloadLink.download = `Meta${\n location.pathname.includes(\"project\") ? \"-AdAccounts\" : \"-Ads\"\n }${\n assistantFilter?.search\n ? `-${assistantFilter?.search}`\n : assistantFilter?.filter\n ? `-${start}-${end}`\n : `-${tab}`\n }-Report`;\n\n document.body.appendChild(downloadLink);\n downloadLink.click();\n document.body.removeChild(downloadLink);\n setAlertObj({\n alertType: \"success\",\n alertMsg: \"Report successfully downloaded\",\n });\n } catch (error) {\n setAlertObj({\n alertType: \"error\",\n alertMsg: \"Unable to download report\",\n });\n }\n };\n\n const fetchMetaAdLists = async (filterObj, purpose) => {\n purpose == \"report\" ? setReportLoading(true) : setIsLoading(true);\n const { partner = {} } = props;\n const { isSuperAffiliate } = partner;\n const endPoint = `${URL}/api/meta-ad-lists?sort=dec`;\n const headers = {\n accept: \"application/json\",\n };\n const data = {\n skip: purpose == \"report\" ? 0 : currentPage * rowsPerPage,\n rowsPerPage: purpose == \"report\" ? paginationCount : rowsPerPage,\n tab,\n filterObj,\n childAffiliateId: childAffiliateId,\n isSuperAffiliate,\n };\n\n try {\n const response = await axios.post(endPoint, data, { headers });\n if (purpose != \"report\") {\n setAds(response?.data?.logs || []);\n if (filterObj?.filter || filterObj?.search) {\n setPaginationCount(response.data?.metaAdsCount);\n } else {\n const newTab = tabs.find((item) => item.display == tab);\n if (counts[newTab.state]) {\n setPaginationCount(counts[newTab.state]);\n }\n }\n } else {\n makeCsvDataReport(response?.data?.logs || []);\n }\n purpose == \"report\" ? setReportLoading(false) : setIsLoading(false);\n } catch (error) {\n console.error(\"Error:\", error);\n setAlertObj({\n alertType: \"error\",\n alertMsg: \"Unable to fetch data\",\n });\n setIsLoading(false);\n }\n };\n\n const fetchMetaAdAccountLists = async (filterObj, purpose) => {\n purpose == \"report\" ? setReportLoading(true) : setIsLoading(true);\n const endpoint = `${URL}/api/meta-ad-acccount-lists?sort=dec`;\n const headers = {\n accept: \"application/json\",\n };\n const { partner = {} } = props;\n const { isSuperAffiliate } = partner;\n const data = {\n skip: purpose == \"report\" ? 0 : currentPage * rowsPerPage,\n rowsPerPage: purpose == \"report\" ? paginationCount : rowsPerPage,\n tab,\n filterObj,\n childAffiliateId: childAffiliateId,\n isSuperAffiliate,\n };\n\n try {\n const response = await axios.post(endpoint, data, { headers });\n\n if (purpose != \"report\") {\n setAds(response?.data?.logs || []);\n if (filterObj?.filter || filterObj?.search) {\n setPaginationCount(response.data?.totalCount);\n } else {\n const newTab = tabs.find((item) => item.display == tab);\n if (counts[newTab.state]) {\n setPaginationCount(counts[newTab.state]);\n }\n }\n } else {\n makeCsvDataReport(response?.data?.logs || []);\n }\n\n purpose == \"report\" ? setReportLoading(false) : setIsLoading(false);\n } catch (error) {\n console.error(\"Error:\", error);\n setAlertObj({\n alertType: \"error\",\n alertMsg: \"Unable to fetch data\",\n });\n purpose == \"report\" ? setReportLoading(false) : setIsLoading(false);\n }\n };\n\n const handleChangePage = (event, newPage) => {\n setCurrentPage(newPage);\n };\n\n const handleChangeRowsPerPage = (event) => {\n setRowsPerPage(parseInt(event.target.value, 10));\n setCurrentPage(0);\n };\n\n const setDate = (filterKey, startDate, endDate) => {\n const appliedData = { ...applied };\n appliedData[filterKey].startDate = startDate;\n appliedData[filterKey].endDate = endDate;\n\n setApplied(applied);\n };\n\n const setFocus = (filterKey, focusedInput) => {\n const appliedData = { ...applied };\n appliedData[filterKey].focusedInput = focusedInput;\n setApplied(appliedData);\n };\n\n const checkDateInput = (filterKey) => {\n const appliedData = { ...applied };\n const s = appliedData[filterKey];\n if (s.startDate && s.endDate) {\n return;\n } else {\n clearDate(filterKey);\n }\n };\n const clearDate = (filterKey) => {\n const appliedData = { ...applied };\n appliedData[filterKey].startDate = null;\n appliedData[filterKey].endDate = null;\n setApplied(appliedData);\n };\n\n const applyFilter = async () => {\n const filterObj = {};\n if (applied.adStartDate.startDate && applied.adStartDate.endDate) {\n filterObj.filter = true;\n filterObj.fromAdStartDate = applied.adStartDate.startDate._d;\n filterObj.toAdStartDate = applied.adStartDate.endDate._d;\n } else {\n delete filterObj.fromAdStartDate;\n delete filterObj.toAdStartDate;\n }\n if (applied.adStopDate.startDate && applied.adStopDate.endDate) {\n filterObj.filter = true;\n filterObj.fromAdStopDate = applied.adStopDate.startDate._d;\n filterObj.toAdStopDate = applied.adStopDate.endDate._d;\n } else {\n delete filterObj.fromAdStopDate;\n delete filterObj.toAdStopDate;\n }\n setAssistantFilter(filterObj);\n await fetchMetaAdLists(filterObj);\n setOpenDialog(!openDialog);\n };\n\n const onSnackbarClose = () => {\n setAlertObj({\n alertType: \"\",\n alertMsg: \"\",\n });\n };\n return (\n \n
\n ) : (\n
\n )\n }\n onButtonClick={() => DownloadReportHandler()}\n filter={assistantFilter?.search}\n searchPlaceholder={`Search ${page}`}\n handleSearch={(e) => {\n setAssistantFilter({ ...assistantFilter, search: e.target.value });\n }}\n handleEnter={(e) => {\n if (e.key === \"Enter\") {\n const filterObj = {};\n setRowsPerPage(10);\n setCurrentPage(0);\n filterObj.search = e.target.value;\n setAssistantFilter({ ...assistantFilter, filterObj });\n if (location.pathname.includes(\"project\")) {\n fetchMetaAdAccountLists(filterObj);\n } else {\n fetchMetaAdLists(filterObj);\n }\n }\n }}\n searchDone={assistantFilter?.search}\n handleClear={() => {\n if (location.pathname.includes(\"project\")) {\n fetchMetaAdAccountLists();\n } else {\n fetchMetaAdLists();\n }\n\n setAssistantFilter({});\n setRowsPerPage(10);\n setCurrentPage(0);\n }}\n toggleFilterDialog={\n location.pathname.includes(\"project\") ? \"\" : toggleDialog\n }\n buttonTitle={\n props?.partner?.isChildAffiliate ? null : \"Download Report\"\n }\n showSearch={true}\n filterColor={false}\n />\n
\n {\n if (tab === newValue) return;\n\n const newTab = tabs.find((item) => item.display == newValue);\n setPaginationCount(counts[newTab.state]);\n setCurrentPage(0);\n setRowsPerPage(10);\n const filterObj = {};\n filterObj.filter = true;\n filterObj.search = \"\";\n setAssistantFilter(filterObj);\n setTab(newValue);\n setAssistantFilter({});\n setApplied({\n adStartDate: {\n startDate: null,\n endDate: null,\n focus: null,\n },\n adStopDate: {\n startDate: null,\n endDate: null,\n focus: null,\n },\n });\n }}\n indicatorColor=\"primary\"\n textColor=\"primary\"\n variant=\"scrollable\"\n scrollButtons=\"auto\"\n aria-label=\"scrollable auto tabs example\"\n TabIndicatorProps={{ className: classes.tabIndicator }}\n >\n {tabs.map((i) => (\n \n \n \n \n {(assistantFilter?.filter || assistantFilter?.search) &&\n tab === i.display\n ? paginationCount\n : counts?.[i.state] || 0}\n \n \n \n \n \n {(assistantFilter?.filter || assistantFilter?.search) &&\n tab === i.display\n ? \"Result\"\n : i.display}\n \n >\n }\n value={i.display}\n style={{\n textTransform: \"none\",\n alignItems: \"end\",\n minWidth: 190,\n paddingBottom: 8,\n background: `linear-gradient(180deg, white, ${\n tab === i.display ? i.theme : \"white\"\n })`,\n }}\n />\n ))}\n \n \n
\n
\n \n \n \n {page == \"projects\"}\n
\n
\n \n \n \n
\n
\n `${from}-${to} of ${count !== -1 ? count : \"more than \" + to}`\n }\n />\n \n\n
\n \n \n \n Filter \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n Apply\n \n \n \n \n
\n \n {alertObj.alertMsg}\n \n \n
\n );\n};\n\nconst styles = (theme) => ({\n root: {\n width: \"100%\",\n height: \"100%\",\n // background: \"red\",\n // display: \"flex\",\n background: \"rgb(247,247,247)\",\n boxSizing: \"border-box\",\n overflowX: \"hidden\",\n position: \"relative\",\n // paddingBottom: \"80px\"\n },\n dialogContainer: {\n // zIndex: \"3400 !important\",\n \"& .MuiPaper-root\": {\n [theme.breakpoints.down(\"md\")]: {\n width: \"100%\",\n // height: \"100%\",\n // maxHeight: \"100%\",\n margin: \"0\",\n borderRadius: \"0px\",\n },\n [theme.breakpoints.up(\"md\")]: {\n width: \"100%\",\n // height: \"unset\",\n // maxHeight: \"unset\",\n // minHeight: \"600px\",\n margin: \"unset\",\n borderRadius: \"6px\",\n },\n },\n },\n pageTitleContainer: {\n position: \"sticky\",\n // zIndex: 100,\n top: 0,\n height: 60,\n boxSizing: \"border-box\",\n [theme.breakpoints.down(\"md\")]: {\n // paddingTop: \"40px\",\n },\n [theme.breakpoints.down(\"sm\")]: {\n // paddingTop: \"20px\",\n },\n },\n fullWidth: {\n width: \"100%\",\n background: \"white\",\n height: \"71px\",\n },\n tabIndicator: {\n borderRadius: \"3px 3px 0px 0px\",\n height: 3,\n },\n tableContainer: {\n top: \"60px\",\n height: \"calc(100vh - 260px)\",\n width: \"100%\",\n overflow: \"hidden\",\n // background: \"red\",\n boxSizing: \"border-box\",\n position: \"sticky\",\n [theme.breakpoints.down(\"xs\")]: {\n // position top + bottombar height + bottom padding + navbar\n marginTop: 48,\n height: \"calc(100vh - 70px - 50px - 10px - 50px)\",\n paddingLeft: theme.spacing(1),\n paddingRight: theme.spacing(1),\n boxSizing: \"border-box\",\n },\n },\n fixedBottomContainer: {\n position: \"fixed\",\n height: \"60px\",\n borderTop: \"1px solid lightgrey\",\n background: \"white\",\n bottom: \"0\",\n left: \"71px\",\n right: \"0\",\n overflow: \"hidden\",\n display: \"flex\",\n alignItems: \"center\",\n justifyContent: \"center\",\n [theme.breakpoints.down(\"sm\")]: {\n left: \"0px\",\n height: \"50px\",\n },\n },\n});\n\nfunction DateFilter({\n filterKey,\n applied,\n setFocus,\n setDate,\n checkDateInput,\n clearDate,\n ...props\n}) {\n const { classes, filterName } = props;\n const A = applied[filterKey];\n return (\n \n \n {A.startDate && A.endDate ? (\n \n clearDate(filterKey)}\n >\n \n \n \n ) : (\n \"\"\n )}\n \n \n {filterName}\n \n \n \n checkDateInput(filterKey)}\n startDateId=\"startDate\"\n endDateId=\"endDate\"\n startDate={A.startDate}\n endDate={A.endDate}\n onDatesChange={({ startDate, endDate }) =>\n setDate(filterKey, startDate, endDate)\n }\n focusedInput={A.focusedInput}\n onFocusChange={(focusedInput) => setFocus(filterKey, focusedInput)}\n navPosition=\"navPositionTop\"\n numberOfMonths={1}\n navPrev={\n \n \n \n }\n navNext={\n \n \n \n }\n hideKeyboardShortcutsPanel\n customArrowIcon={null}\n screenReaderInputMessage={\" \"}\n small\n readOnly\n isOutsideRange={(day) =>\n filterKey === \"adStopDate\" ? false : moment().diff(day) < 0\n }\n />\n \n \n \n );\n}\n\nfunction NavigationWrapper(props) {\n const { classes } = props;\n return (\n \n {props.children} \n
\n );\n}\n\nconst connectMetaAds = connect(\n (state) => ({\n partner: state.partner.partner,\n user: state.login.user,\n }),\n {\n fetchPartnerDetails,\n }\n)(MetaAds);\n\nexport default withStyles(styles)(\n withMediaQuery(\"(max-width:600px)\")(connectMetaAds)\n);\n","import React, { Component } from \"react\";\nimport {\n withStyles,\n Grid,\n Box,\n Typography,\n CircularProgress,\n Snackbar,\n} from \"@material-ui/core\";\nimport { connect } from \"react-redux\";\nimport { fetchPartnerDetails } from \"../../store/partnerState\";\nimport Header from \"../../commons/Header/Header\";\nimport axios from \"axios\";\nimport { Alert } from \"@material-ui/lab\";\n\nconst { APIURL } = require(\"../../config/config\");\n\nclass SuperDashboard extends Component {\n constructor(props) {\n super(props);\n this.state = {\n isLoading: false,\n totalClick: 0,\n totalLeadsGenerated: 0,\n totalConversions: 0,\n conversionRate: 0,\n termsDialogOpen: true,\n selectedLink: \"\",\n snackbarOpen: false,\n uniqueClicks: 0,\n analyticsDialog: false,\n data: [\n { name: \"Total Clicks\", value: 400 },\n { name: \"Unique Clicks\", value: 300 },\n ],\n totalFosCount: 0,\n totalBusiness: 0,\n totalSpentAmount: 0,\n totalAdCreditRecharged: 0,\n };\n }\n\n componentDidMount() {\n this.props.fetchPartnerDetails();\n this.fetchDashboardData();\n }\n\n fetchDashboardData = () => {\n axios\n .get(APIURL + \"/affiliates/dashboard-analytics-super-affiliate\")\n .then((response) => {\n const { data } = response;\n this.setState({\n totalFosCount: data.affiliateCount,\n totalBusiness: data.projectCount,\n totalSpentAmount: data.totalSpentAmount,\n totalAdCreditRecharged: data.totalAdCreditRecharged,\n });\n })\n .catch((error) => {\n console.error(error);\n });\n };\n\n render() {\n const { classes, isPartnerDetailLoading, partnerDetail } = this.props;\n\n const isPartnerDetailEmpty =\n !partnerDetail || Object.keys(partnerDetail).length === 0;\n\n if (isPartnerDetailLoading || isPartnerDetailEmpty) {\n return (\n \n \n \n );\n }\n\n return (\n \n
\n
\n
\n
\n \n \n \n \n \n \n Overview\n \n \n \n \n \n \n Total Business Onboarded \n \n \n {this.state.totalBusiness || 0} \n \n \n \n\n \n \n \n Total Ads Spend \n \n \n \n {this.state.totalSpentAmount > 0 &&\n `₹ ${this.state.totalSpentAmount}`}\n {this.state.totalSpentAmount === 0 && <>₹ 0.00>}\n \n \n \n \n\n \n \n \n Total Revenue Generated \n \n \n \n {(partnerDetail.totalCommisionPaidINR ?? 0) > 0 &&\n `₹ ${(\n partnerDetail.totalCommisionPaidINR / 100000\n ).toFixed(2)}`}\n {\n (partnerDetail.totalCommisionPaidINR ?? 0) >\n 0 &&\n (partnerDetail.totalCommisionPaidUSD ?? 0) >\n 0 &&\n \" | \" // Display a bar only if both INR and USD have values\n }\n {(partnerDetail.totalCommisionPaidUSD ?? 0) > 0 &&\n `$ ${(\n partnerDetail.totalCommisionPaidUSD / 100000\n ).toFixed(2)}`}\n {(partnerDetail.totalCommisionPaidINR ?? 0) ===\n 0 &&\n (partnerDetail.totalCommisionPaidUSD ?? 0) ===\n 0 && <>₹ 0.00>}\n \n \n \n \n \n \n \n \n \n Total Commission \n \n \n {this.state.totalLeadsGenerated || 0} \n \n \n \n \n \n \n Total FOS Agents \n \n \n {this.state.totalFosCount || 0} \n \n \n \n \n \n \n Total Ad Credit Recharge \n \n \n {this.state.totalAdCreditRecharged || 0} \n \n \n \n \n \n \n \n \n \n
\n
\n \n {this.state.alertMsg}\n \n \n
\n
\n );\n }\n}\n\nconst styles = (theme) => ({\n root: {\n width: \"100%\",\n height: \"100%\",\n background: \"rgb(247,247,247)\",\n boxSizing: \"border-box\",\n overflowX: \"hidden\",\n position: \"relative\",\n },\n pageTitleContainer: {\n position: \"sticky\",\n zIndex: 100,\n top: 0,\n height: 60,\n boxSizing: \"border-box\",\n },\n fullWidth: {\n width: \"100%\",\n background: \"white\",\n },\n primaryBackground: {\n background: \"rgb(70 20 134 / 15%)\",\n },\n container: {\n // marginTop: 16,\n background: \"white\",\n borderRadius: 8,\n padding: \"24px 36px\",\n marginTop: 10,\n marginBottom: 10,\n },\n analyticsContainer: {\n background: \"white\",\n padding: 0,\n margin: 0,\n },\n snackbar: {\n top: theme.spacing(9),\n },\n viewOldButton: {\n \"&:hover\": {\n color: \"grey\",\n cursor: \"pointer\",\n },\n },\n textFieldRootAutocomplete: {\n display: \"flex\",\n width: \"100%\",\n maxWidth: \"100%\",\n padding: \"2px 8px\",\n marginTop: \"10px\",\n marginBottom: \"20px\",\n background: \"white\",\n border: \"2px solid grey\",\n justifyContent: \"center\",\n alignItems: \"center\",\n \"& label\": {\n background: \"white\",\n color: \"#b2b3b3 !important\",\n padding: \"0 6px\",\n marginTop: -4,\n },\n },\n\n helpLink: {\n cursor: \"pointer !important\",\n },\n\n textFieldRootAutocompleteDropDown: {\n display: \"flex\",\n width: \"100%\",\n maxWidth: \"100%\",\n padding: \"2px 8px\",\n marginTop: \"7px\",\n marginBottom: \"20px\",\n background: \"white\",\n border: \"2px solid grey\",\n justifyContent: \"center\",\n alignItems: \"center\",\n \"& label\": {\n background: \"white\",\n color: \"#b2b3b3 !important\",\n padding: \"0 6px\",\n marginTop: -4,\n },\n },\n chip: {\n margin: \"2px\",\n fontSize: \"0.5rem\",\n display: \"flex\",\n flexDirection: \"row\",\n },\n tagList: {\n display: \"flex\",\n flexWrap: \"wrap\",\n },\n});\n\nconst connectSuperDashboard = connect(\n (state) => ({\n isPartnerDetailLoading: state.partner.isLoading,\n partnerDetail: state.partner.partner,\n }),\n {\n fetchPartnerDetails,\n }\n)(SuperDashboard);\n\nexport default withStyles(styles)(connectSuperDashboard);\n","import React, { useState, useEffect } from \"react\";\nimport {\n Dialog,\n DialogTitle,\n DialogContent,\n DialogActions,\n Button,\n TextField,\n Grid,\n Typography,\n FormControl,\n Select,\n MenuItem,\n InputLabel,\n Box,\n IconButton,\n} from \"@material-ui/core\";\nimport { Close } from \"@material-ui/icons\";\n\nconst EditAgentDialog = ({ open, onClose, agent, onSave }) => {\n const [contact, setContact] = useState(\"\");\n const [status, setStatus] = useState(\"\");\n const [isSubmitting, setIsSubmitting] = useState(false);\n\n useEffect(() => {\n if (agent) {\n setContact(agent.contact || \"\");\n setStatus(agent.status || \"ACTIVE\");\n }\n }, [agent]);\n\n const handleSubmit = () => {\n setIsSubmitting(true);\n\n onSave({\n ...agent,\n contact,\n status,\n })\n .then(() => {\n setIsSubmitting(false);\n onClose();\n })\n .catch((error) => {\n setIsSubmitting(false);\n // Handle error\n console.error(\"Error updating agent:\", error);\n });\n };\n\n if (!agent) return null;\n\n return (\n \n \n \n \n Edit Affiliate Agent\n \n \n \n \n \n \n \n \n \n \n \n Agent Name\n \n {agent.name || \"NA\"} \n \n\n \n \n Email\n \n {agent.email} \n \n\n \n setContact(e.target.value)}\n margin=\"normal\"\n />\n \n\n \n \n Status \n setStatus(e.target.value)}\n label=\"Status\"\n >\n Active \n Inactive \n \n \n \n \n \n \n \n \n Cancel\n \n \n {isSubmitting ? \"Saving...\" : \"Save Changes\"}\n \n \n \n );\n};\n\nexport default EditAgentDialog;\n","import axios from \"axios\";\nimport { APIURL, URL } from \"../../config/config\";\nimport { StaticRouter } from \"react-router\";\n\nconst initialState = {\n // \"business\": [],\n business: Array(10)\n .fill(0)\n .map((item) => ({ name: \"Loading\" })),\n skip: 0,\n totalBusinesses: 0,\n isLoading: true,\n error: null,\n page: 0,\n rowsPerPage: 10,\n allCount: 0,\n archivedCount: 0,\n};\n\nconst CREATE_CLIENT = \"Client/CREATE_CLIENT\";\nconst LOAD_START = \"Client/LOAD_START\";\nconst LOAD_SUCCESS = \"Client/LOAD_SUCCESS\";\nconst LOAD_MORE_SUCCESS = \"Client/LOAD_MORE_SUCCESS\";\nconst LOAD_FAILURE = \"Client/LOAD_FAILURE\";\nconst ADD_CLIENT = \"Client/ADD_CLIENT\";\nconst CHANGE_PAGE = \"Client/CHANGE_PAGE\";\nconst CHANGE_ROWS_PER_PAGE = \"Clients/CHANGE_ROWS_PER_PAGE\";\nconst CHANGE_QUERY = \"Clients/CHANGE_QUERY\";\nconst CHANGE_STATUS = \"Clients/CHANGE_STATUS\";\nconst EXIT_CLIENTS = \"TEMPLATE/EXIT_CLIENTS\";\nconst LOAD_DATE = \"LOAD_DATE/CLIENTS\";\nconst LOAD_APPLIED = \"LOAD_APPLIED/CLIENT\";\nconst LOAD_BUSINESS_COUNTS = \"LOAD_BUSINESS_COUNTS/CLIENT\";\n\nconst loadStart = () => ({\n type: LOAD_START,\n});\n\nconst loadSuccess = ({ business, skip, totalBusinesses }) => ({\n type: LOAD_SUCCESS,\n payload: { business, skip, totalBusinesses },\n});\n\nconst createClient = ({ business, totalBusinesses, createdAt }) => ({\n type: CREATE_CLIENT,\n payload: { business, totalBusinesses, createdAt },\n});\n\nconst loadmoresuccess = ({ business, newSkip }) => ({\n type: LOAD_MORE_SUCCESS,\n payload: { business, newSkip },\n});\n\nconst loadFailure = (error) => ({\n type: LOAD_FAILURE,\n payload: error,\n});\n\nconst loadBusinessCount = ({ all, archived }) => ({\n type: LOAD_BUSINESS_COUNTS,\n payload: {\n all,\n archived,\n },\n});\n\nexport const loadClient = (filters) => (dispatch, getState) => {\n const { rowsPerPage } = getState().client;\n const { user } = getState().login;\n const newFilter = { ...filters };\n console.log({ user });\n dispatch(loadStart);\n axios\n .post(APIURL + \"/affiliates/fetch-child-affiliates\", {\n skip: 0,\n rowsPerPage,\n ...newFilter,\n })\n .then((response) => {\n const { business, newSkip, totalBusinesses } = response.data;\n dispatch(\n loadSuccess({\n business: business,\n skip: newSkip,\n totalBusinesses: totalBusinesses,\n })\n );\n })\n .catch((error) => {\n console.error(error);\n dispatch(loadFailure(error));\n });\n};\n\nexport const loadSearch = (filter) => (dispatch, getState) => {\n const skip = 0;\n const rowsPerPage = 0;\n const clientSearch = true;\n const { user } = getState().login;\n dispatch(loadStart);\n axios\n .post(APIURL + `${`/affiliates/fetch-child-affiliates`}`, {\n ...(user.isReferral ? { partnerReferralId: user.id } : {}),\n skip,\n rowsPerPage,\n filter,\n clientSearch,\n })\n .then((response) => {\n const { business, newSkip, totalBusinesses } = response.data;\n dispatch(\n loadSuccess({\n business: business,\n skip: newSkip,\n totalBusinesses: totalBusinesses,\n })\n );\n })\n .catch((error) => {\n console.error(error);\n dispatch(loadFailure(error));\n });\n};\n\nexport const loadNext = (event, newPage, filters) => (dispatch, getState) => {\n const { rowsPerPage, skip, business } = getState().client;\n const { user } = getState().login;\n const sort = filters.sort;\n const newFilter = { ...filters };\n delete newFilter.sort;\n\n if (business.length > newPage * rowsPerPage) {\n dispatch({ type: CHANGE_PAGE, payload: newPage });\n } else {\n dispatch({ type: CHANGE_PAGE, payload: newPage });\n axios\n .post(APIURL + `${`/affiliates/fetch-child-affiliates`}?sort=${sort}`, {\n skip,\n rowsPerPage,\n ...(user.isReferral ? { partnerReferralId: user.id } : {}),\n ...newFilter,\n })\n .then((response) => {\n const { business, newSkip } = response.data;\n dispatch(loadmoresuccess({ business: business, newSkip: newSkip }));\n })\n .catch((error) => {\n console.error(error);\n dispatch(loadFailure(error));\n });\n }\n};\n\nexport const createnewClient = (business) => (dispatch, getState) => {\n const totalBusinesses = getState().client.totalBusinesses;\n dispatch(\n createClient({\n business: { ...business, _id: business.id },\n totalBusinesses: totalBusinesses,\n createdAt: Date.now(),\n })\n );\n};\n\nexport const addClient = (client) => (dispatch) => {\n dispatch({ type: ADD_CLIENT, payload: client });\n};\n\nexport const changeQuery = (nameQuery) => (dispatch) => {\n dispatch({ type: CHANGE_QUERY, payload: nameQuery });\n};\n\nexport const changeRowsPerPage = (rowsPerPage) => (dispatch) => {\n dispatch({ type: CHANGE_ROWS_PER_PAGE, payload: rowsPerPage });\n};\n\nexport const unfetchclients = () => (dispatch) => {\n dispatch({ type: EXIT_CLIENTS, payload: null });\n};\n\nexport const fetchChildAffiliateCount = () => (dispatch, getState) => {\n const { user } = getState().login;\n\n axios\n .get(APIURL + \"/affiliates/fetch-child-affiliates-count\")\n .then((response) => {\n const { allCount } = response.data;\n dispatch(\n loadBusinessCount({\n all: allCount || 0,\n })\n );\n })\n .catch((error) => {\n console.error(error);\n dispatch(loadFailure(error));\n });\n};\n\nexport default function ClientReducer(state = initialState, { type, payload }) {\n switch (type) {\n case LOAD_START:\n return { ...state, isLoading: true, error: null };\n\n case LOAD_SUCCESS:\n return {\n ...state,\n business: [...payload.business],\n skip: payload.skip,\n totalBusinesses: payload.totalBusinesses,\n page: 0,\n isLoading: false,\n error: null,\n };\n case LOAD_DATE:\n return {\n ...state,\n applied: {\n ...state.applied,\n createdAt: {\n ...state.applied.createdAt,\n startDate: payload.fromDate,\n endDate: payload.toDate,\n },\n },\n };\n case CREATE_CLIENT:\n const businessList = [...state.business];\n if (businessList.length > 9) {\n businessList.pop();\n }\n return {\n ...state,\n business: [\n {\n ...payload.business,\n displayName: payload.business.display_name,\n createdOn: payload.createdAt,\n },\n ...businessList,\n ],\n totalBusinesses: payload.totalBusinesses + 1,\n };\n\n case LOAD_MORE_SUCCESS:\n return {\n ...state,\n business: [...state.business, ...payload.business],\n skip: payload.newSkip,\n isLoading: false,\n };\n\n case CHANGE_ROWS_PER_PAGE:\n return { ...state, rowsPerPage: payload, page: 0 };\n\n case CHANGE_PAGE:\n return { ...state, page: payload };\n\n case LOAD_FAILURE:\n return { ...state, isLoading: true, error: payload };\n\n case ADD_CLIENT:\n return {\n ...state,\n business: { [payload.name]: payload, ...state.business },\n };\n\n case LOAD_APPLIED:\n return {\n ...state,\n applied: { ...state.applied, ...payload.applied },\n };\n\n case LOAD_BUSINESS_COUNTS:\n return {\n ...state,\n allCount: payload.all,\n archivedCount: payload.archived,\n };\n\n case CHANGE_QUERY:\n return {\n ...state,\n nameQuery: payload,\n };\n\n case EXIT_CLIENTS:\n return {\n ...initialState,\n };\n\n case CHANGE_STATUS:\n return {\n ...state,\n business: {\n ...state.business,\n [payload.displayName]: {\n ...state.business[payload.displayName],\n status: payload.status,\n },\n },\n };\n\n default:\n return state;\n }\n}\n","import React, { Component } from \"react\";\nimport { connect } from \"react-redux\";\nimport {\n withStyles,\n Grid,\n Box,\n Typography,\n IconButton,\n Tooltip,\n} from \"@material-ui/core\";\nimport { withRouter } from \"react-router-dom\";\nimport { Skeleton } from \"@material-ui/lab\";\nimport { Edit } from \"@material-ui/icons\";\nimport EditAgentDialog from \"./EditAgentDialog\";\nimport axios from \"axios\";\nimport { APIURL } from \"../../config/config\";\nimport { loadClient } from \"./AffiliateAgentsStore\";\n\n// Updated fields with edit action column\nconst fields = [\n {\n name: \"Name\",\n size: 2,\n key: \"name\",\n },\n {\n name: \"Email\",\n size: 3,\n key: \"email\",\n },\n {\n name: \"Contact\",\n size: 1,\n key: \"contact\",\n },\n {\n name: \"Created On\",\n size: 2,\n key: \"createdAt\",\n },\n {\n name: \"Team\",\n size: 2,\n key: \"team\",\n },\n {\n name: \"Status\",\n size: 1,\n key: \"status\",\n },\n {\n name: \"Actions\",\n size: 1,\n key: \"actions\",\n },\n];\n\nclass SuperAffiliateAgentsTable extends Component {\n constructor(props) {\n super(props);\n this.state = {\n openDialog: false,\n selectedTemplate: null,\n editDialogOpen: false,\n selectedAgent: null,\n };\n }\n\n listScrolled = (event) => {\n if (!this.offsetPosition) {\n this.offsetPosition = this.rootEle?.current?.offsetTop;\n }\n const container = event.currentTarget;\n const currentPos = container.scrollTop;\n this.setState({\n sticky: currentPos > this.offsetPosition,\n });\n };\n\n toggleDialog = () => {\n this.setState({ openDialog: !this.state.openDialog });\n };\n\n openClient = (e) => {\n if (!e?._id) return;\n this.props.history.push(`/agents/meta-ads/projects/${e._id}`);\n };\n\n handleEditClick = (e, agent) => {\n e.stopPropagation(); // Prevent row click from triggering\n this.setState({\n editDialogOpen: true,\n selectedAgent: agent,\n });\n };\n\n handleCloseEditDialog = () => {\n this.setState({\n editDialogOpen: false,\n selectedAgent: null,\n });\n };\n\n handleSaveAgent = async (updatedAgent) => {\n axios\n .post(APIURL + \"/affiliates/edit-child-affiliate\", { ...updatedAgent })\n .then((response) => {\n this.props.loadClient({});\n this.setState({\n editDialogOpen: false,\n selectedAgent: null,\n });\n this.props.showSnackbar(\"success\", \"Updated successfully\");\n })\n .catch((error) => {\n this.setState({\n editDialogOpen: false,\n selectedAgent: null,\n });\n this.props.showSnackbar(\n \"error\",\n error?.response?.data?.message ||\n \"Error updating agent. Please try again.\"\n );\n });\n };\n\n render() {\n const { classes, business, isLoading } = this.props;\n const { editDialogOpen, selectedAgent } = this.state;\n\n return (\n \n \n \n \n \n \n \n {fields.map((e) => (\n \n \n \n {e.name}\n \n \n \n ))}\n \n {business.map((agent) => (\n {\n this.openClient(agent);\n }}\n >\n {fields.map((field) => (\n \n | \n \n ))}\n \n ))}\n \n \n \n \n \n\n {/* Edit Dialog */}\n \n
\n );\n }\n}\n\nfunction Cell({ classes, item, field, onEditClick, ...props }) {\n switch (field) {\n case \"displayName\":\n case \"name\":\n return (\n \n \n \n {item.name || item.displayName}\n \n \n \n );\n\n case \"email\":\n return (\n \n \n \n {item[field]?.startsWith(\"__archived__\")\n ? item[field].replace(\"__archived__\", \"\")\n : item[field]}\n \n \n \n );\n\n case \"createdAt\":\n return (\n \n \n \n {getFormattedDate(item[field])}\n \n \n \n );\n\n case \"contact\":\n return (\n \n \n \n {item[field]}\n \n \n \n );\n\n case \"status\":\n return (\n \n \n \n {item[field] || \"Active\"}\n \n \n \n );\n\n case \"actions\":\n return (\n \n \n \n onEditClick(e, item)}\n className={classes.editButton}\n size=\"small\"\n >\n \n \n \n \n \n );\n\n case \"Loading\":\n return (\n \n \n \n \n \n );\n\n default:\n return (\n \n \n \n {item[field]}\n \n \n \n );\n }\n}\n\nfunction getFormattedDate(date) {\n const d = new Date(date);\n const options = { year: \"numeric\", month: \"long\", day: \"numeric\" };\n return d.toLocaleDateString(\"en-US\", options);\n}\n\nfunction getStatusColor(status) {\n switch (status.toLowerCase()) {\n case \"active\":\n return \"#4caf50\"; // Green\n case \"inactive\":\n return \"#ff9800\"; // Orange\n case \"suspended\":\n return \"#f44336\"; // Red\n default:\n return \"inherit\";\n }\n}\n\nconst styles = (theme) => ({\n root: {\n width: \"100%\",\n height: \"100%\",\n background: \"rgb(249,249,249)\",\n boxSizing: \"border-box\",\n overflowX: \"hidden\",\n position: \"relative\",\n },\n tableWrapper: {\n overflowX: \"auto\",\n width: \"100%\",\n },\n tableContainer: {\n marginTop: \"1.5em\",\n minWidth: 1100,\n },\n tableItem: {\n background: \"white\",\n marginBottom: \"1em\",\n padding: 4,\n borderRadius: 8,\n cursor: \"pointer\",\n \"&:hover\": {\n background: \"#efefef\",\n },\n },\n tableHeader: {\n background: \"white\",\n padding: 4,\n borderRadius: 8,\n marginBottom: \"1em\",\n },\n editButton: {\n padding: 8,\n \"&:hover\": {\n backgroundColor: \"rgba(0, 0, 0, 0.08)\",\n },\n },\n});\n\nconst SuperAffiliateAgentsTableConnect = connect(\n (state) => ({\n isLoading: state.client.isLoading,\n }),\n { loadClient }\n)(SuperAffiliateAgentsTable);\n\nexport default withStyles(styles)(withRouter(SuperAffiliateAgentsTableConnect));\n","import React, { Component } from \"react\";\nimport { connect } from \"react-redux\";\nimport { withRouter } from \"react-router-dom\";\nimport {\n withStyles,\n TextField,\n Grid,\n Typography,\n Box,\n} from \"@material-ui/core\";\nimport Autocomplete from \"@material-ui/lab/Autocomplete\";\nimport dialCodes, { countryMap, isoMap } from \"../../config/dialCodes\";\n\nclass AdDetailStep extends Component {\n handleInput = (e) => {\n this.props.updateState(e.target.name, e.target.value);\n };\n\n render() {\n const { classes } = this.props;\n\n const getflag = (langcode) => {\n var first = langcode.charCodeAt(0) + 127397;\n var second = langcode.charCodeAt(1) + 127397;\n var flag = `${first};${second};`;\n const x = document.createElement(\"p\");\n x.innerHTML = flag;\n return x.innerText;\n };\n\n return (\n \n \n \n Create Your FOS Here\n \n\n \n\n \n i.dialCode)}\n classes={{ popper: classes.textFieldPopper }}\n getOptionLabel={(option) => {\n return `${getflag(isoMap[option])} ${option}`;\n }}\n onChange={(e, value, reason) => {\n const countryCode = \"countryCode\";\n this.props.updateState(countryCode, value);\n }}\n value={this.props.countryCode}\n renderInput={(params) => (\n \n )}\n />\n\n \n \n\n \n\n \n \n \n );\n }\n}\n\nconst styles = (theme) => ({\n root: {\n width: \"100%\",\n padding: \"20px 10px\",\n display: \"flex\",\n },\n codeTextFieldRoot: {\n width: \"90px\",\n padding: \"0 0 0 14px\",\n position: \"absolute\",\n zIndex: 1,\n margin: \"10px 0 0 2px\",\n background: \"white\",\n \"& input\": {\n padding: \"10px 0 !important\",\n },\n \"& button\": {\n width: \"12px\",\n position: \"relative\",\n left: \"7px\",\n background: \"white\",\n borderRadius: 0,\n \"&:hover\": {\n background: \"white\",\n },\n },\n \"& div\": {\n paddingRight: \"0px !important\",\n },\n },\n bottomMargin: {\n marginBottom: theme.spacing(4),\n },\n TextFieldRoot: {\n width: \"100%\",\n padding: \"2px\",\n marginTop: \"4px\",\n // marginBottom: \"20px\",\n \"& input\": { width: \"calc(100% - 80px)\" },\n background: \"rgb(240,240,240)\",\n },\n textFieldRoot: {\n width: \"100%\",\n padding: \"8px 20px 10px\",\n marginTop: \"4px\",\n marginBottom: \"20px\",\n background: \"white\",\n border: \"2px solid grey\",\n \"& input\": {\n padding: \"6px 12px!important\",\n },\n \"& label\": {\n background: \"white\",\n color: \"#b2b3b3 !important\",\n padding: \"0 6px\",\n marginTop: -4,\n },\n },\n textFieldRootAutocomplete: {\n width: \"100%\",\n padding: \"2px 8px\",\n marginTop: \"4px\",\n marginBottom: \"20px\",\n background: \"white\",\n border: \"2px solid grey\",\n \"& label\": {\n background: \"white\",\n color: \"#b2b3b3 !important\",\n padding: \"0 6px\",\n marginTop: -4,\n },\n },\n textFieldPopper: {\n [theme.breakpoints.down(\"md\")]: {\n left: \"12px !important\",\n minWidth: \"calc(100% - 35px)\",\n },\n [theme.breakpoints.up(\"md\")]: {\n left: \"calc(12.5vw - 16px) !important\",\n minWidth: \"calc(25vw + 54px)\",\n },\n },\n phoneTextFieldRoot: {\n width: \"100%\",\n padding: \"8px 20px\",\n marginTop: \"4px\",\n marginBottom: \"20px\",\n \"& input\": { width: \"calc(100% - 80px)\", marginLeft: \"80px\" },\n background: \"white\",\n border: \"2px solid grey\",\n },\n passwordIcon: {\n cursor: \"pointer\",\n fontSize: 20,\n },\n});\n\nconst Adsconnect = connect((state) => ({\n partner: state.partner.partner,\n}))(AdDetailStep);\n\nexport default withStyles(styles)(withRouter(Adsconnect));\n","import React from \"react\";\nimport { Button } from \"@material-ui/core\";\nimport { GetApp } from \"@material-ui/icons\";\n\nconst CSVTemplateDownloader = () => {\n const generateTemplate = () => {\n const headers = [\"Name\", \"Mobile No\", \"Email ID\", \"Team Name\"];\n const sampleData = [\n [\"John Doe\", \"9876543210\", \"john.doe@example.com\", \"Marketing\"],\n [\"Jane Smith\", \"8765432109\", \"jane.smith@example.com\", \"Sales\"],\n ];\n\n // Generate CSV content\n let csvContent = headers.join(\",\") + \"\\n\";\n sampleData.forEach((row) => {\n csvContent += row.join(\",\") + \"\\n\";\n });\n\n // Create a Blob and download\n const blob = new Blob([csvContent], { type: \"text/csv;charset=utf-8;\" });\n const url = URL.createObjectURL(blob);\n const link = document.createElement(\"a\");\n link.href = url;\n link.setAttribute(\"download\", \"affiliate_agents_template.csv\");\n document.body.appendChild(link);\n link.click();\n document.body.removeChild(link);\n };\n\n return (\n }\n variant=\"outlined\"\n color=\"primary\"\n onClick={generateTemplate}\n size=\"small\"\n >\n Download CSV Template\n \n );\n};\n\nexport default CSVTemplateDownloader;\n","import React, { useState } from \"react\";\nimport {\n Box,\n Typography,\n Paper,\n Divider,\n Chip,\n Grid,\n Button,\n List,\n ListItem,\n ListItemText,\n Collapse,\n} from \"@material-ui/core\";\nimport {\n ExpandMore,\n ExpandLess,\n GetApp,\n ArrowUpward,\n ArrowDownward,\n} from \"@material-ui/icons\";\nimport { makeStyles } from \"@material-ui/core/styles\";\n\nconst useStyles = makeStyles((theme) => ({\n root: {\n padding: theme.spacing(2),\n marginTop: theme.spacing(2),\n maxWidth: \"100%\",\n boxSizing: \"border-box\",\n },\n summaryContainer: {\n display: \"flex\",\n justifyContent: \"space-around\",\n marginBottom: theme.spacing(2),\n padding: theme.spacing(2),\n backgroundColor: \"#f5f5f5\",\n borderRadius: theme.shape.borderRadius,\n },\n statItem: {\n display: \"flex\",\n flexDirection: \"column\",\n alignItems: \"center\",\n },\n statValue: {\n fontSize: \"1.5rem\",\n fontWeight: \"bold\",\n },\n statLabel: {\n fontSize: \"0.875rem\",\n color: theme.palette.text.secondary,\n },\n successValue: {\n color: theme.palette.success.main,\n },\n errorValue: {\n color: theme.palette.error.main,\n },\n divider: {\n margin: theme.spacing(2, 0),\n },\n errorList: {\n maxHeight: \"300px\",\n overflowY: \"auto\",\n overflowX: \"hidden\",\n marginTop: theme.spacing(2),\n width: \"100%\",\n boxSizing: \"border-box\",\n },\n errorItem: {\n backgroundColor: \"#fff4f4\",\n marginBottom: theme.spacing(1),\n borderRadius: theme.shape.borderRadius,\n },\n errorSummary: {\n display: \"flex\",\n justifyContent: \"space-between\",\n alignItems: \"center\",\n width: \"100%\",\n },\n errorDetails: {\n paddingLeft: theme.spacing(2),\n paddingRight: theme.spacing(2),\n backgroundColor: \"#fff\",\n borderTop: `1px solid ${theme.palette.divider}`,\n },\n actionButton: {\n marginTop: theme.spacing(2),\n },\n chip: {\n marginRight: theme.spacing(1),\n },\n errorMessage: {\n color: theme.palette.error.main,\n fontWeight: \"bold\",\n },\n}));\n\nconst UploadResultsSummary = ({ stats, errors, onClose }) => {\n const classes = useStyles();\n const [expandedError, setExpandedError] = useState(null);\n const [sortOrder, setSortOrder] = useState(\"asc\"); // 'asc' or 'desc'\n\n const toggleErrorExpansion = (errorIndex) => {\n setExpandedError(expandedError === errorIndex ? null : errorIndex);\n };\n\n const toggleSortOrder = () => {\n setSortOrder(sortOrder === \"asc\" ? \"desc\" : \"asc\");\n };\n\n const sortedErrors = [...errors].sort((a, b) => {\n if (sortOrder === \"asc\") {\n return a.row - b.row;\n } else {\n return b.row - a.row;\n }\n });\n\n // Generate downloadable CSV with errors\n const downloadErrorsCSV = () => {\n // Create headers\n let csvContent = \"Row,Name,Email,Errors\\n\";\n\n // Add error data\n errors.forEach((error) => {\n const errorString = error.errors.join(\"; \").replace(/\"/g, '\"\"'); // Escape quotes\n csvContent += `${error.row},\"${error.name}\",\"${error.email}\",\"${errorString}\"\\n`;\n });\n\n // Create a blob and download\n const blob = new Blob([csvContent], { type: \"text/csv;charset=utf-8;\" });\n const url = URL.createObjectURL(blob);\n const link = document.createElement(\"a\");\n link.href = url;\n link.setAttribute(\"download\", \"affiliate_upload_errors.csv\");\n document.body.appendChild(link);\n link.click();\n document.body.removeChild(link);\n };\n\n return (\n \n \n Upload Results Summary\n \n\n \n \n \n Total Records\n \n {stats.total} \n \n\n \n \n Successfully Created\n \n \n {stats.success}\n \n \n\n \n \n Failed\n \n \n {stats.failed}\n \n \n \n\n {errors.length > 0 && (\n <>\n \n\n \n \n Errors Found ({errors.length})\n \n\n \n {/* : \n }\n size=\"small\"\n onClick={toggleSortOrder}\n >\n {sortOrder === \"asc\" ? \"Oldest First\" : \"Newest First\"}\n */}\n\n }\n variant=\"outlined\"\n size=\"small\"\n color=\"primary\"\n onClick={downloadErrorsCSV}\n style={{ marginLeft: 8 }}\n >\n Export Errors\n \n \n \n\n \n {sortedErrors.map((error, index) => (\n \n toggleErrorExpansion(index)}>\n \n \n \n Row {error.row}: {error.name}\n \n \n {error.email}\n \n \n\n \n \n {expandedError === index ? (\n \n ) : (\n \n )}\n \n \n \n\n \n \n \n {error.errors.map((err, errIndex) => (\n \n \n \n ))}\n
\n \n \n \n ))}\n
\n >\n )}\n\n \n \n {stats.success > 0 ? \"Done\" : \"Close\"}\n \n \n \n );\n};\n\nexport default UploadResultsSummary;\n","import React, { useState } from \"react\";\nimport {\n withStyles,\n Button,\n Grid,\n Typography,\n Box,\n CircularProgress,\n IconButton,\n LinearProgress,\n Paper,\n} from \"@material-ui/core\";\nimport { CloudUpload, GetApp } from \"@material-ui/icons\";\nimport axios from \"axios\";\nimport Papa from \"papaparse\";\nimport { APIURL } from \"../../config/config\";\nimport CSVTemplateDownloader from \"./CSVTemplateDownloader\";\nimport UploadResultsSummary from \"./UploadResultSummary\";\n\nconst CSVUpload = (props) => {\n const {\n classes,\n toggleCreateAffiliateAgents,\n showSnackbar,\n fetchChildAffiliateCount,\n loadClient,\n partner,\n } = props;\n const [file, setFile] = useState(null);\n const [isLoading, setIsLoading] = useState(false);\n const [uploadStats, setUploadStats] = useState({\n total: 0,\n success: 0,\n failed: 0,\n });\n const [errors, setErrors] = useState([]);\n const [showResults, setShowResults] = useState(false);\n const [progress, setProgress] = useState(0);\n const [progressMessage, setProgressMessage] = useState(\"\");\n\n const handleFileChange = (e) => {\n const selectedFile = e.target.files[0];\n if (selectedFile && selectedFile.type === \"text/csv\") {\n setFile(selectedFile);\n setErrors([]);\n setShowResults(false);\n } else {\n showSnackbar(\"error\", \"Please upload a valid CSV file\");\n }\n };\n\n const validateRow = (row) => {\n const errors = [];\n\n if (!row[\"Name\"] || row[\"Name\"].trim() === \"\") {\n errors.push(\"Name is required\");\n }\n\n if (!row[\"Mobile No\"] || !/^\\d{10,}$/.test(row[\"Mobile No\"])) {\n errors.push(\"Valid Mobile No is required (at least 10 digits)\");\n }\n\n if (\n !row[\"Email ID\"] ||\n !/^[^\\s@]+@[^\\s@]+\\.[^\\s@]+$/.test(row[\"Email ID\"])\n ) {\n errors.push(\"Valid Email ID is required\");\n }\n\n return errors;\n };\n\n const createAffiliateLink = async (affiliateId) => {\n try {\n await axios.post(APIURL + `/affiliates/create-new-affiliate-link`, {\n linkTitle: \"link1\",\n domain: \"app.aisensy.com\",\n affiliateId,\n });\n return true;\n } catch (error) {\n console.error(\"Error creating affiliate link:\", error);\n return false;\n }\n };\n\n const processCSV = async () => {\n if (!file) {\n showSnackbar(\"error\", \"Please select a CSV file first\");\n return;\n }\n\n setIsLoading(true);\n setShowResults(false);\n setErrors([]);\n setProgress(0);\n setProgressMessage(\"Preparing to process CSV...\");\n\n // Parse the CSV file\n Papa.parse(file, {\n header: true,\n skipEmptyLines: true,\n complete: async (results) => {\n const { data } = results;\n if (!data || data.length === 0) {\n setIsLoading(false);\n showSnackbar(\"error\", \"The CSV file is empty or has invalid format\");\n return;\n }\n\n // Check if required headers exist\n const requiredHeaders = [\"Name\", \"Mobile No\", \"Email ID\"];\n const headers = Object.keys(data[0]);\n const missingHeaders = requiredHeaders.filter(\n (header) => !headers.includes(header)\n );\n\n if (missingHeaders.length > 0) {\n setIsLoading(false);\n showSnackbar(\n \"error\",\n `Missing required headers: ${missingHeaders.join(\", \")}`\n );\n return;\n }\n\n // Process records\n const stats = {\n total: data.length,\n success: 0,\n failed: 0,\n };\n\n const errorList = [];\n\n // Get partner ID\n const superAffiliateId = partner?._id || \"67c29170603e6b0bf4f5849b\";\n\n for (let i = 0; i < data.length; i++) {\n // Update progress\n const percent = Math.floor((i / data.length) * 100);\n setProgress(percent);\n setProgressMessage(\n `Processing ${i + 1} of ${data.length} records (${percent}%)...`\n );\n\n const row = data[i];\n const rowErrors = validateRow(row);\n\n if (rowErrors.length > 0) {\n stats.failed++;\n errorList.push({\n row: i + 2, // +2 for header row and 1-indexing\n name: row[\"Name\"] || \"N/A\",\n email: row[\"Email ID\"] || \"N/A\",\n errors: rowErrors,\n });\n\n // Small delay to allow UI to update\n await new Promise((resolve) => setTimeout(resolve, 10));\n continue;\n }\n\n try {\n const response = await axios.post(\n APIURL + \"/affiliates/create-super-affiliate-agent\",\n {\n userName: row[\"Name\"],\n name: row[\"Name\"],\n team: row[\"Team Name\"] || \"\",\n companyName: \"\",\n contact: row[\"Mobile No\"],\n email: row[\"Email ID\"],\n loginPassword: null,\n isChildAffiliate: true,\n superAffiliateId: superAffiliateId,\n isAdsCommissionEnabled: true,\n }\n );\n\n await createAffiliateLink(response.data.affiliate._id);\n\n stats.success++;\n } catch (error) {\n stats.failed++;\n errorList.push({\n row: i + 2,\n name: row[\"Name\"] || \"N/A\",\n email: row[\"Email ID\"] || \"N/A\",\n errors: [\n error.response?.data?.message || \"Unknown error occurred\",\n ],\n });\n }\n\n // Small delay to allow UI to update\n await new Promise((resolve) => setTimeout(resolve, 10));\n }\n\n // Complete\n setProgress(100);\n setProgressMessage(\"Processing complete!\");\n setUploadStats(stats);\n setErrors(errorList);\n setIsLoading(false);\n setShowResults(true);\n\n // Refresh the list if any successful uploads\n if (stats.success > 0) {\n loadClient();\n fetchChildAffiliateCount();\n showSnackbar(\n \"success\",\n `Successfully created ${stats.success} affiliate agents`\n );\n } else if (stats.total > 0) {\n showSnackbar(\n \"error\",\n \"No records were successfully processed. Please check the errors and try again.\"\n );\n }\n },\n error: (error) => {\n setIsLoading(false);\n showSnackbar(\"error\", \"Error parsing CSV: \" + error.message);\n },\n });\n };\n\n return (\n \n \n \n \n \n \n }\n disabled={isLoading}\n className={classes.uploadButton}\n >\n Select CSV File\n \n \n {file && (\n \n {file.name}\n \n )}\n \n \n\n \n \n \n The CSV should have these headers: \"Name\", \"Mobile No\", \"Email\n ID\", \"Team Name\"\n \n \n \n \n \n \n\n \n \n \n {isLoading ? (\n \n ) : (\n \"Upload and Process\"\n )}\n \n \n\n {isLoading && (\n \n \n {progressMessage}\n \n \n \n )}\n \n\n {showResults && (\n \n {\n if (uploadStats.success > 0) {\n // If we had successful uploads, close the dialog\n toggleCreateAffiliateAgents();\n } else {\n // Otherwise, just hide the results and let the user try again\n setShowResults(false);\n }\n }}\n />\n \n )}\n \n
\n );\n};\n\nconst styles = (theme) => ({\n root: {\n padding: theme.spacing(2),\n },\n input: {\n display: \"none\",\n },\n uploadBox: {\n display: \"flex\",\n flexDirection: \"column\",\n alignItems: \"center\",\n padding: theme.spacing(3),\n border: \"2px dashed #ccc\",\n borderRadius: 8,\n backgroundColor: \"#f9f9f9\",\n },\n uploadButton: {\n marginBottom: theme.spacing(2),\n },\n fileName: {\n marginTop: theme.spacing(1),\n },\n processButton: {\n minWidth: 150,\n },\n progressBar: {\n height: 8,\n borderRadius: 4,\n },\n resultsPaper: {\n padding: theme.spacing(3),\n marginTop: theme.spacing(2),\n },\n errorList: {\n maxHeight: 200,\n overflowY: \"auto\",\n border: \"1px solid #ddd\",\n borderRadius: 4,\n padding: theme.spacing(1),\n },\n errorItem: {\n padding: theme.spacing(1),\n backgroundColor: \"#fff4f4\",\n borderRadius: 4,\n marginBottom: theme.spacing(1),\n },\n errorSubList: {\n margin: 0,\n paddingLeft: theme.spacing(2),\n },\n});\n\nexport default withStyles(styles)(CSVUpload);\n","import axios from \"axios\";\nimport { connect } from \"react-redux\";\nimport React, { Component } from \"react\";\nimport { withRouter } from \"react-router-dom\";\nimport { APIURL, URL, partnerURL } from \"../../config/config\";\nimport { Clear } from \"@material-ui/icons\";\nimport {\n withStyles,\n Button,\n Grid,\n IconButton,\n Tooltip,\n Typography,\n Box,\n Tabs,\n Tab,\n} from \"@material-ui/core\";\nimport { isoToDialCode } from \"../../config/dialCodes\";\nimport CreateAgentForm from \"./CreateAgentForm\";\nimport {\n fetchChildAffiliateCount,\n loadClient,\n} from \"../../pages/SuperAffiliate/AffiliateAgentsStore\";\nimport CSVUpload from \"./CSVUpload\";\n\nclass CreateAffiliateAgent extends Component {\n constructor(props) {\n super(props);\n this.state = {\n activeStep: 0,\n error: null,\n snack: false,\n snackMessage: \"\",\n snackStatus: \"\",\n email: null,\n name: \"\",\n team: \"\",\n contact: \"\",\n company: \"\",\n currency: this.props?.partner?.currency || \"INR\",\n password: null,\n confirmPassword: null,\n buttonDisable: false,\n countryCode: isoToDialCode[this.props.partner?.isoCode] || \"+91\",\n showPassword: false,\n tabValue: 0,\n };\n }\n\n handleTabChange = (event, newValue) => {\n this.setState({ tabValue: newValue });\n };\n\n createAffiliateLink = (affiliateId) => {\n axios\n .post(APIURL + `/affiliates/create-new-affiliate-link`, {\n linkTitle: \"link1\",\n domain: \"app.aisensy.com\",\n affiliateId,\n })\n .then((response) => {})\n .catch((error) => {\n this.setState({ isLoading: false });\n const snackStatus = \"error\";\n const snackMessage = error?.response?.data?.message;\n this.props.businessSnackDialog(snackStatus, snackMessage);\n });\n };\n\n createChildAffiliate = () => {\n this.setState({ isLoading: true });\n const partnerId = this.props.partner._id;\n const { name, team, company, contact, email, password, confirmPassword } =\n this.state;\n axios\n .post(APIURL + \"/affiliates/create-super-affiliate-agent\", {\n userName: name,\n name,\n team,\n companyName: company,\n contact,\n email,\n loginPassword: password,\n isChildAffiliate: true,\n superAffiliateId: partnerId,\n isAdsCommissionEnabled: true,\n })\n .then((response) => {\n const snackStatus = \"success\";\n const snackMessage = \"Successfully created\";\n this.props.toggleCreateAffiliateAgents();\n this.props.loadClient();\n this.props.fetchChildAffiliateCount();\n this.props.showSnackbar(snackStatus, snackMessage);\n const { data } = response;\n const affiliateId = data?.affiliate?._id;\n\n this.createAffiliateLink(affiliateId);\n this.setState({\n countryCode: \"+91\",\n company: \"\",\n password: null,\n confirmPassword: null,\n isLoading: false,\n name: \"\",\n contact: \"\",\n email: \"\",\n team: \"\",\n });\n })\n .catch((error) => {\n this.setState({ isLoading: false });\n const snackStatus = \"error\";\n const snackMessage = error?.response?.data?.message;\n this.props.showSnackbar(snackStatus, snackMessage);\n });\n };\n\n updateState = (keys, value) => {\n const returnObj = {};\n returnObj[keys] = value;\n this.setState(returnObj);\n };\n\n render() {\n const { classes, tags, partner } = this.props;\n const { activeStep, snackStatus, tabValue } = this.state;\n\n return (\n \n \n \n \n \n Add Affiliate Agents\n \n \n \n this.props.toggleCreateAffiliateAgents()}\n >\n \n \n \n \n \n\n \n \n \n \n\n {tabValue === 0 ? (\n \n \n \n {this.getStepContent(activeStep, tags)}\n \n\n \n \n \n \n \n {this.state.isLoading ? \"Creating...\" : \"Create\"}\n \n \n \n \n \n \n \n ) : (\n \n \n \n )}\n
\n );\n }\n\n getStepContent = (step, tags) => {\n const { partner } = this.props;\n const { type } = partner || {};\n\n switch (step) {\n case 0:\n return (\n \n );\n case 1:\n return <>>;\n\n default:\n return <>>;\n }\n };\n\n nextBtnDisableStatus = () => {\n const error = this.getValidationError();\n return !!error;\n };\n\n getValidationError = () => {\n const validateEmail = (email) => {\n const emailRegex = /^[^\\s@]+@[^\\s@]+\\.[^\\s@]+$/;\n return emailRegex.test(email);\n };\n\n const validatePhone = (phone) => {\n const phoneRegex = /^\\d{10,}$/;\n return phoneRegex.test(phone);\n };\n\n const { email, name, team, company, contact, password, confirmPassword } =\n this.state;\n\n if (!name || name.trim() === \"\") return \"Name is required\";\n if (name.trim().length < 3) return \"Name should be at least 3 characters\";\n\n // if (!company || company.trim() === \"\") return \"Company name is required\";\n // if (company.trim().length < 2)\n // return \"Company name should be at least 2 characters\";\n\n if (!contact) return \"Phone number is required\";\n if (!validatePhone(contact))\n return \"Phone number should be at least 10 digits\";\n\n if (!email) return \"Email is required\";\n if (!validateEmail(email)) return \"Invalid email format\";\n\n // if (!password) return \"Password is required\";\n // if (password.length < 6) return \"Password should be at least 6 characters\";\n\n // if (!confirmPassword) return \"Confirm password is required\";\n // if (password !== confirmPassword) return \"Passwords do not match\";\n\n return null;\n };\n}\n\nconst styles = (theme) => ({\n root: {\n height: \"100%\",\n overflowY: \"auto\",\n overflowX: \"hidden\", \n width: \"100%\",\n boxSizing: \"border-box\",\n },\n\n formContainer: {\n borderRadius: \"8px\",\n // minHeight: 200,\n justifyContent: \"center\",\n boxSizing: \"border-box\",\n [theme.breakpoints.down(\"sm\")]: {\n paddingLeft: \"10px\",\n paddingRight: \"10px\",\n },\n },\n textFieldRoot: {\n width: \"100%\",\n padding: \"8px 20px 10px\",\n marginTop: \"4px\",\n marginBottom: \"20px\",\n background: \"disabled\",\n border: \"2px solid disabled\",\n \"& input\": {\n padding: \"6px 12px!important\",\n },\n \"& label\": {\n background: \"disabled\",\n // color: \"#b2b3b3 !important\",\n padding: \"0 6px\",\n marginTop: -4,\n },\n },\n\n formContainer1: {\n background: \"white\",\n borderRadius: \"8px\",\n // minHeight: 200,\n boxSizing: \"border-box\",\n paddingLeft: \"10px\",\n paddingRight: \"10px\",\n [theme.breakpoints.down(\"sm\")]: {\n paddingLeft: \"10px\",\n paddingRight: \"10px\",\n },\n marginBottom: \"20px\",\n },\n\n stepContainer: {\n overflow: \"auto\",\n marginBottom: \"1em\",\n display: \"block\",\n },\n\n buttonContainer: {\n // margin: \"0% 8%\",\n width: \"100%\",\n padding: \"1em 0em\",\n overflowX: \"none\",\n },\n\n pageTitleContainer: {\n position: \"sticky\",\n zIndex: 100,\n top: 0,\n height: 60,\n boxSizing: \"border-box\",\n background: \"white\",\n marginBottom: \"20px\",\n },\n option: {\n // backgroundColor: \"white\",\n fontSize: 15,\n [theme.breakpoints.down(\"sm\")]: {\n width: \"95%\",\n },\n \"& > span\": {\n marginRight: 10,\n fontSize: 18,\n },\n myTextfield: {\n \"& MuiOutlinedInput-adornedEnd\": {\n paddingRight: 0,\n },\n },\n },\n myStateClass: {\n \"&::before\": {\n content: '\"some content\"',\n display: \"block\",\n height: 60,\n marginTop: -60,\n },\n \"&::after\": {\n content: '\"some content\"',\n display: \"block\",\n height: 60,\n marginTop: -60,\n },\n },\n tooltipError: {\n backgroundColor: theme.palette.error.main,\n color: theme.palette.error.contrastText,\n fontSize: \"12px\",\n maxWidth: \"300px\",\n },\n csvUploadContainer: {\n padding: theme.spacing(2),\n border: `1px dashed ${theme.palette.grey[400]}`,\n borderRadius: theme.spacing(1),\n marginTop: theme.spacing(2),\n },\n fileInput: {\n display: \"none\",\n },\n uploadButton: {\n margin: theme.spacing(1),\n },\n progressContainer: {\n display: \"flex\",\n alignItems: \"center\",\n marginTop: theme.spacing(2),\n },\n progressText: {\n marginLeft: theme.spacing(2),\n },\n});\n\nconst CreateAffiliateAgentConnect = connect(\n (state) => ({\n user: state.login.user,\n partner: state.partner.partner,\n }),\n {\n loadClient,\n fetchChildAffiliateCount,\n }\n)(CreateAffiliateAgent);\n\nexport default withStyles(styles)(withRouter(CreateAffiliateAgentConnect));\n","/* eslint-disable no-undef */\nimport React, { Component } from \"react\";\nimport { connect } from \"react-redux\";\nimport { Alert, Autocomplete } from \"@material-ui/lab\";\nimport axios from \"axios\";\nimport { withRouter } from \"react-router-dom\";\nimport {\n withStyles,\n Button,\n Box,\n Typography,\n Grid,\n IconButton,\n TextField,\n Dialog,\n DialogContent,\n DialogActions,\n Snackbar,\n TablePagination,\n CircularProgress,\n Tabs,\n Tab,\n} from \"@material-ui/core\";\nimport { DateRangePicker } from \"react-dates\";\nimport { partnerURL } from \"../../config/config\";\nimport moment from \"moment\";\nimport { Add, Clear } from \"@material-ui/icons\";\nimport \"react-dates/initialize\";\nimport \"react-dates/lib/css/_datepicker.css\";\nimport \"../../react-dates-custom.css\";\nimport ClientsTable from \"./SuperAffiliateAgentsTable\";\nimport {\n loadNext,\n loadClient,\n unfetchclients,\n createnewClient,\n loadSearch,\n fetchChildAffiliateCount,\n} from \"./AffiliateAgentsStore\";\nimport Header from \"../../commons/Header/Header\";\nimport withMediaQuery from \"../../helpers/mediaQueryHelper\";\nimport { isoToDialCode } from \"../../config/dialCodes\";\nimport CreateAffiliateAgents from \"./CreateAffiliateAgents\";\nimport { fetchPartnerDetails } from \"../../store/partnerState\";\nclass SuperAffiliateAgents extends Component {\n constructor(props) {\n super(props);\n this.state = {\n openDialog: false,\n agentDialog: false,\n email: null,\n display_name: \"\",\n company: \"\",\n contact: \"\",\n currency: this.props.partner?.currency || \"INR\",\n password: null,\n timezone: \"Asia/Calcutta GMT+05:30\",\n countryCode: isoToDialCode[this.props.partner?.isoCode] || \"+91\",\n industry: \"\",\n snack: false,\n companySize: \"\",\n showPassword: false,\n showConfirmPassword: false,\n status: null,\n statusMessage: \"\",\n buttonDisable: false,\n filter: \"\",\n searchdone: false,\n applied: {\n createdAt: {\n startDate: null,\n endDate: null,\n focus: null,\n },\n },\n clientFilter: {\n filter: \"All\",\n sort: \"dec\",\n },\n tab: \"All\",\n allCount: 0,\n archivedCount: 0,\n };\n }\n\n businessSnackDialog = (val, mess) => {\n this.setState({\n status: val,\n statusMessage: mess,\n snack: true,\n });\n };\n componentDidMount() {\n // Always fetch partner details on mount\n this.props\n .fetchPartnerDetails()\n .then(() => {\n // Only fetch other data after partner details are loaded\n this.props.fetchChildAffiliateCount();\n this.props.unfetchclients();\n this.props.loadClient(this.state.clientFilter);\n })\n .catch((error) => {\n console.error(\"Failed to fetch partner details:\", error);\n // Handle error - maybe show a snackbar\n this.setState({\n snack: true,\n status: \"error\",\n statusMessage: \"Failed to load data. Please refresh the page.\",\n });\n });\n }\n\n componentDidUpdate(prevProps) {\n if (!prevProps.partner && this.props.partner) {\n // Partner data was loaded, update state that depends on partner\n this.setState({\n currency: this.props.partner.currency || \"INR\",\n countryCode: isoToDialCode[this.props.partner.isoCode] || \"+91\",\n });\n }\n }\n\n handleInput = (e) => {\n this.setState({ [e.target.name]: e.target.value });\n };\n\n clearDate = (filterKey) => {\n const applied = { ...this.state.applied };\n applied[filterKey].startDate = null;\n applied[filterKey].endDate = null;\n this.setState({ applied });\n };\n checkDateInput = (filterKey) => {\n const applied = { ...this.state.applied };\n const s = applied[filterKey];\n if (s.startDate && s.endDate) {\n return;\n } else {\n this.clearDate(filterKey);\n }\n };\n\n setFocus = (filterKey, focusedInput) => {\n const applied = { ...this.state.applied };\n applied[filterKey].focusedInput = focusedInput;\n this.setState({ applied });\n };\n setDate = (filterKey, startDate, endDate) => {\n const applied = { ...this.state.applied };\n applied[filterKey].startDate = startDate;\n applied[filterKey].endDate = endDate;\n this.setState({ applied });\n };\n\n applyFilter = () => {\n const { applied, clientFilter } = this.state;\n const filterObj = { ...clientFilter };\n if (applied.createdAt.startDate && applied.createdAt.endDate) {\n filterObj.fromDate = applied.createdAt.startDate._d;\n filterObj.toDate = applied.createdAt.endDate._d;\n } else {\n delete filterObj.fromDate;\n delete filterObj.toDate;\n }\n this.setState(\n {\n clientFilter: filterObj,\n openDialog: false,\n },\n () => {\n this.onTabChange();\n }\n );\n };\n onTabChange = () => {\n this.props.loadClient(this.state.clientFilter);\n };\n\n showSnackbar = (status, message) => {\n this.setState({\n status: status,\n statusMessage: message,\n snack: true,\n });\n };\n\n handlesearch = (e) => {\n this.setState({ filter: e.target.value });\n };\n onSnackbarClose = (event) => {\n this.setState({ snack: false });\n };\n\n handleClickShowConfirmPassword = () => {\n this.setState({ showConfirmPassword: !this.state.showConfirmPassword });\n };\n\n handleClickShowPassword = () => {\n this.setState({ showPassword: !this.state.showPassword });\n };\n\n toggleDialog = () => {\n this.setState({ openDialog: !this.state.openDialog });\n };\n toggleCreateAffiliateAgents = () => {\n this.setState({ agentDialog: !this.state.agentDialog });\n };\n buttondisbale = () => {\n this.setState({ buttonDisable: !this.state.buttonDisable });\n };\n\n handleEnter = (e) => {\n if (e.key === \"Enter\") {\n if (this.state.filter === \"\") {\n this.props.unfetchclients();\n this.props.loadClient(this.state.clientFilter);\n this.setState({\n searchdone: false,\n });\n }\n this.props.loadSearch(this.state.filter);\n if (this.state.filter !== \"\") {\n this.setState({\n searchdone: true,\n });\n }\n }\n };\n\n handleClear = () => {\n this.props.unfetchclients();\n this.setState({\n filter: \"\",\n });\n this.setState({\n searchdone: false,\n });\n this.props.loadClient(this.state.clientFilter);\n };\n\n render() {\n const {\n classes,\n business,\n skip,\n totalBusinesses,\n loadNext,\n loadClient,\n isLoading,\n page,\n unfetchclients,\n rowsPerPage,\n createnewClient,\n fetchChildAffiliateCount,\n user,\n loadSearch,\n } = this.props;\n\n const { clientFilter, applied, tab } = this.state;\n const { partnerPlanFamily, partnerWccFamily, mediaQuery, partner } =\n this.props;\n\n const passwordRegex = /^(?=.*[A-Za-z])(?=.*\\d)[A-Za-z\\d]{8,}$/;\n const isPasswordValid = passwordRegex.test(this.state.password);\n const clients = this.props.business.slice(\n page * rowsPerPage,\n page * rowsPerPage + rowsPerPage\n );\n const SIZE = [\n \"1 - 10 Employees\",\n \"10 - 20 Employees\",\n \"20 - 50 Employees\",\n \"50 - 200 Employees\",\n \"200 + Employees\",\n ];\n\n const SORT = [\"dec\", \"asc\"];\n const VERTICALS = [\n \"Ecommerce\",\n \"Education\",\n \"Automotive\",\n \"IT Services\",\n \"Real Estate\",\n \"SAAS/Apps\",\n \"Gaming\",\n \"Entertainment\",\n \"Finance and Banking\",\n \"Medical and Health\",\n \"Hotel and Lodging\",\n \"Beauty, Spa and Salon\",\n \"Clothing and Apparel\",\n \"Event Planning and Service\",\n \"Food and Grocery\",\n \"Professional Services\",\n \"Public Service\",\n \"Non-profit\",\n \"Shopping and Retail\",\n \"Travel and Transportation\",\n \"Restaurant\",\n \"Other\",\n ];\n\n const tabs = [\n {\n display: \"All\",\n state: \"allCount\",\n theme: \"#f4f4f4\",\n },\n // {\n // display: \"Archived\",\n // state: \"archivedCount\",\n // theme: \"#f2c14e3d\",\n // },\n ];\n\n const getflag = (langcode) => {\n var first = langcode.charCodeAt(0) + 127397;\n var second = langcode.charCodeAt(1) + 127397;\n var flag = `${first};${second};`;\n const x = document.createElement(\"p\");\n x.innerHTML = flag;\n return x.innerText;\n };\n\n return (\n \n {/* Page name & description container */}\n
\n \n \n }\n buttonTitle={\"Create New Agent\"}\n onButtonClick={this.toggleCreateAffiliateAgents}\n showSearch={true}\n searchPlaceholder={\"Search business name\"}\n handleSearch={this.handlesearch}\n handleEnter={this.handleEnter}\n handleClear={this.handleClear}\n searchDone={this.state.searchdone}\n filter={this.state.filter}\n toggleFilterDialog={this.toggleDialog}\n filterColor={\n !!applied.createdAt.endDate && !!applied.createdAt.startDate\n }\n section={\"All Businesses\"}\n />\n\n \n {\n if (this.state.tab === newValue) return;\n // const tab = tabs.find((i) => i.display === newValue);\n this.setState(\n {\n tab: newValue,\n // templates: tab.list,\n // totalCount: tab.list.length,\n clientFilter: {\n ...this.state.clientFilter,\n filter: newValue,\n },\n },\n this.onTabChange\n );\n }}\n indicatorColor=\"primary\"\n textColor=\"primary\"\n variant=\"scrollable\"\n scrollButtons=\"auto\"\n aria-label=\"scrollable auto tabs example\"\n TabIndicatorProps={{\n className: classes.tabIndicator,\n }}\n >\n {tabs.map((i) => (\n \n \n \n \n {this.props[i.state]}\n \n \n \n \n {`${i.display}`}\n \n >\n }\n value={i.display}\n style={{\n textTransform: \"none\",\n alignItems: \"end\",\n minWidth: 190,\n marginLeft: 35,\n paddingBottom: 8,\n background: `linear-gradient(180deg, white, ${\n this.state.tab === i.display ? i.theme : \"white\"\n })`,\n }}\n />\n ))}\n \n \n \n \n \n
\n {/* Layout */}\n
\n \n {/* First section */}\n {/* Scrollable (x & y) table container */}\n\n \n \n
\n \n {\" \"}\n {/* Table footer fixed at bottom */}\n
\n
{\n this.props.loadNext(event, newPage, this.state.clientFilter);\n }}\n rowsPerPage={this.props.rowsPerPage}\n rowsPerPageOptions={[25]}\n />\n {\" \"}\n {/* Create Affiliate Agents Dialog */}\n
\n \n \n \n \n {/* Filter Dialog */}\n
\n \n \n \n Filter \n \n \n \n \n \n \n \n\n \n Sort \n \n {\n const filterObj = {\n ...this.state.clientFilter,\n sort: value === \"Descending\" ? \"dec\" : \"asc\",\n };\n this.setState({ clientFilter: filterObj });\n }}\n name=\"sort\"\n options={[\"Descending\", \"Ascending\"]}\n renderInput={(params) => (\n \n )}\n />\n \n \n \n \n this.clearDate(\"createdAt\")}\n style={{ marginRight: 8 }}\n >\n Clear Filter\n \n \n Apply\n \n \n \n \n
\n \n {this.state.statusMessage}\n \n \n
\n );\n }\n}\n\nfunction DateFilter({\n filterKey,\n applied,\n setFocus,\n setDate,\n checkDateInput,\n clearDate,\n ...props\n}) {\n const { classes, filterName } = props;\n const A = applied[filterKey];\n return (\n \n \n {A.startDate && A.endDate ? (\n \n clearDate(filterKey)}\n >\n \n \n \n ) : (\n \"\"\n )}\n \n \n {filterName}\n \n \n \n checkDateInput(filterKey)}\n startDateId=\"startDate\"\n endDateId=\"endDate\"\n startDate={A.startDate}\n endDate={A.endDate}\n onDatesChange={({ startDate, endDate }) =>\n setDate(filterKey, startDate, endDate)\n }\n focusedInput={A.focusedInput}\n onFocusChange={(focusedInput) => setFocus(filterKey, focusedInput)}\n navPosition=\"navPositionTop\"\n numberOfMonths={1}\n hideKeyboardShortcutsPanel\n customArrowIcon={null}\n screenReaderInputMessage={\" \"}\n small\n readOnly\n isOutsideRange={(day) => moment().diff(day) < 0}\n />\n \n \n \n );\n\n // function setToday()\n}\n\n//\nconst styles = (theme) => ({\n root: {\n width: \"100%\",\n height: \"100%\",\n // background: \"red\",\n // display: \"flex\",\n background: \"rgb(249,249,249)\",\n boxSizing: \"border-box\",\n overflowX: \"hidden\",\n position: \"relative\",\n // overflowY: \"scroll\"\n // paddingBottom: \"80px\"\n },\n pageTitleContainer: {\n // position: \"sticky\",\n // zIndex: 100,\n // top: 0,\n // height: 80,\n // boxSizing: \"border-box\",\n // [theme.breakpoints.down(\"md\")]: {\n // // paddingTop: \"40px\",\n // },\n // [theme.breakpoints.down(\"sm\")]: {\n // // paddingTop: \"20px\",\n // },\n },\n fullWidth: {\n width: \"100%\",\n background: \"white\",\n },\n container: {\n background: \"white\",\n borderRadius: \"8px\",\n },\n attachTooltipPopper: {\n \"& .MuiTooltip-tooltip\": {\n padding: 0,\n backgroundColor: \"white\",\n color: \"#222\",\n boxShadow: \"0 0 11px rgb(230,230,230)\",\n },\n \"& .MuiTooltip-arrow\": {\n color: \"rgb(70 20 134 / 15%)\",\n filter: \"drop-shadow(0px -2px 1px #ddd)\",\n },\n },\n dialogContainer: {\n // zIndex: \"3400 !important\",\n \"& .MuiPaper-root\": {\n [theme.breakpoints.down(\"md\")]: {\n width: \"100%\",\n // height: \"100%\",\n // maxHeight: \"100%\",\n margin: \"0\",\n borderRadius: \"6px\",\n },\n [theme.breakpoints.up(\"md\")]: {\n width: \"100%\",\n height: \"unset\",\n // maxHeight: \"unset\",\n // minHeight: \"600px\",\n margin: \"unset\",\n borderRadius: \"6px\",\n },\n },\n },\n fixedBottomContainer: {\n position: \"fixed\",\n height: \"60px\",\n borderTop: \"1px solid lightgrey\",\n background: \"white\",\n bottom: \"0\",\n left: \"71px\",\n right: \"0\",\n overflow: \"hidden\",\n display: \"flex\",\n alignItems: \"center\",\n justifyContent: \"center\",\n [theme.breakpoints.down(\"sm\")]: {\n left: \"0px\",\n height: \"50px\",\n },\n },\n tableContainer: {\n top: \"60px\",\n height: \"calc(100vh - 180px)\",\n width: \"100%\",\n overflow: \"hidden\",\n // background: \"red\",\n boxSizing: \"border-box\",\n position: \"sticky\",\n [theme.breakpoints.down(\"sm\")]: {\n // position top + bottombar height + bottom padding + navbar\n height: \"calc(100vh - 70px - 50px - 10px - 50px)\",\n paddingLeft: theme.spacing(1),\n paddingRight: theme.spacing(1),\n boxSizing: \"border-box\",\n },\n },\n textFieldRoot: {\n width: \"100%\",\n padding: \"8px 20px 10px\",\n marginTop: \"4px\",\n marginBottom: \"20px\",\n background: \"white\",\n border: \"2px solid grey\",\n \"& input\": {\n padding: \"6px 12px!important\",\n },\n \"& label\": {\n background: \"white\",\n color: \"#b2b3b3 !important\",\n padding: \"0 6px\",\n marginTop: -4,\n },\n },\n phoneTextFieldRoot: {\n width: \"100%\",\n padding: \"8px 20px\",\n marginTop: \"4px\",\n marginBottom: \"20px\",\n \"& input\": { width: \"calc(100% - 80px)\", marginLeft: \"80px\" },\n background: \"white\",\n border: \"2px solid grey\",\n },\n sortFormControl: {\n width: 120,\n marginRight: 8,\n height: 32,\n borderRadius: \"4px\",\n border: \"1px solid lightgrey\",\n },\n\n codeTextFieldRoot: {\n width: \"90px\",\n padding: \"0 0 0 14px\",\n position: \"absolute\",\n zIndex: 1,\n margin: \"10px 0 0 2px\",\n background: \"white\",\n \"& input\": {\n padding: \"10px 0 !important\",\n },\n \"& button\": {\n width: \"12px\",\n position: \"relative\",\n left: \"7px\",\n background: \"white\",\n borderRadius: 0,\n \"&:hover\": {\n background: \"white\",\n },\n },\n \"& div\": {\n paddingRight: \"0px !important\",\n },\n },\n passwordIcon: {\n cursor: \"pointer\",\n fontSize: 20,\n },\n\n textFieldRootAutocomplete: {\n width: \"100%\",\n padding: \"2px 8px\",\n marginTop: \"4px\",\n marginBottom: \"20px\",\n background: \"white\",\n border: \"2px solid grey\",\n \"& label\": {\n background: \"white\",\n color: \"#b2b3b3 !important\",\n padding: \"0 6px\",\n marginTop: -4,\n },\n },\n\n textFieldSortAutocomplete: {\n width: \"100%\",\n padding: \"2px 8px\",\n marginTop: \"4px\",\n marginBottom: \"20px\",\n background: \"white\",\n border: \"2px solid grey\",\n \"& label\": {\n background: \"white\",\n color: \"#b2b3b3 !important\",\n padding: \"0 6px\",\n marginTop: -4,\n },\n },\n textFieldPopper: {\n [theme.breakpoints.down(\"md\")]: {\n left: \"12px !important\",\n minWidth: \"calc(100% - 35px)\",\n },\n [theme.breakpoints.up(\"md\")]: {\n left: \"calc(12.5vw - 16px) !important\",\n minWidth: \"calc(25vw + 54px)\",\n },\n },\n textField: {\n width: 300,\n marginRight: 8,\n height: 42,\n },\n\n tabIndicator: {\n borderRadius: \"3px 3px 0px 0px\",\n height: 3,\n },\n\n tabContainer: {\n background: \"white\",\n boxShadow: \"0 0 12px rgb(171 170 170)\",\n },\n\n textFieldSortAutocomplete: {\n width: \"120\",\n height: 32,\n marginTop: \"4px\",\n marginBottom: \"20px\",\n \"& input\": {\n padding: \"0px 12px!important\",\n },\n background: \"white\",\n borderRadius: \"4px\",\n border: \"1px solid #d3d3d3\",\n \"& label\": {\n background: \"white\",\n color: \"#b2b3b3 !important\",\n padding: \"0 6px\",\n marginTop: -4,\n },\n },\n});\n\nconst SuperAffiliateAgentsConnect = connect(\n (state) => ({\n user: state.login.user,\n business: state.client.business,\n skip: state.client.skip,\n totalBusinesses: state.client.totalBusinesses,\n allCount: state.client.allCount,\n archivedCount: state.client.archivedCount,\n isLoading: state.client.isLoading,\n error: state.client.error,\n page: state.client.page,\n rowsPerPage: state.client.rowsPerPage,\n partnerPlanFamily: state.planFamilies.allPlanFamilies,\n partnerWccFamily: state.wccPlans.allWccPlans,\n partner: state.partner.partner,\n }),\n {\n loadNext,\n loadClient,\n unfetchclients,\n createnewClient,\n loadSearch,\n fetchChildAffiliateCount,\n fetchPartnerDetails,\n }\n)(SuperAffiliateAgents);\nexport default withStyles(styles)(\n withMediaQuery(\"(max-width:600px)\")(withRouter(SuperAffiliateAgentsConnect))\n);\n","import React, { Component } from \"react\";\nimport { Route, Switch } from \"react-router-dom\";\nimport ErrorPage from \"../../commons/ErrorPage/ErrorPage\";\nimport SuperAffiliateAgents from \"./SuperAffiliateAgents\";\nimport MetaAdsLayout from \"../../commons/ProjectsLayout/MetaAdsLayout\";\n\nclass SuperAffiliateRoute extends Component {\n constructor(props) {\n super(props);\n this.state = {};\n }\n render() {\n return (\n \n \n (\n \n )}\n />\n \n \n );\n }\n}\n\nexport default SuperAffiliateRoute;\n","import React from \"react\";\nimport { useHistory, useLocation, useParams, Link } from \"react-router-dom\";\nimport {\n Box,\n Typography,\n withStyles,\n Grid,\n ButtonBase,\n} from \"@material-ui/core\";\nimport { connect } from \"react-redux\";\nimport { renderAgnetsMetaAdsRoute } from \"../Routes/ProjectRoutes\";\nimport AccountTreeOutlinedIcon from \"@material-ui/icons/AccountTreeOutlined\";\nimport BarChartOutlinedIcon from \"@material-ui/icons/BarChartOutlined\";\n\nfunction AgentsAdsNavBarDesktop(props) {\n const location = useLocation();\n const history = useHistory();\n const params = useParams();\n const { classes, agent, partner, agentId } = props;\n\n const getNavItems = () => [\n {\n name: \"Ads Setup\",\n path: `/agents/meta-ads/projects/${agentId}`,\n icon: ,\n },\n {\n name: \"Ads Overview\",\n path: `/agents/meta-ads/ads/${agentId}`,\n icon: ,\n },\n ];\n\n return (\n \n \n \n Meta Ads \n \n {getNavItems().map((item, index) => {\n const isActive = location.pathname === item.path;\n return (\n history.push(item.path)}\n >\n \n \n \n {item.icon}\n \n \n \n \n {item.name}\n \n \n \n \n );\n })}\n \n
\n );\n}\n\nconst styles = (theme) => ({\n root: {\n width: \"220px\",\n height: \"100vh\",\n background: \"rgb(255, 255, 255)\",\n borderRight: \"1px solid rgb(240,240,240)\",\n display: \"flex\",\n flexDirection: \"column\",\n justifyContent: \"space-between\",\n color: \"gray\",\n [theme.breakpoints.down(\"md\")]: {\n width: \"220px\",\n },\n [theme.breakpoints.down(\"sm\")]: {\n display: \"none\",\n },\n },\n logo: {\n height: 50,\n marginTop: 15,\n },\n link_inactive: {\n display: \"flex\",\n justifyContent: \"center\",\n alignItems: \"center\",\n height: \"35px\",\n width: \"35px\",\n margin: \"auto\",\n borderRadius: \"50%\",\n cursor: \"pointer\",\n transition: \"0.5s\",\n },\n link_active: {\n display: \"flex\",\n justifyContent: \"center\",\n alignItems: \"center\",\n height: \"35px\",\n width: \"35px\",\n margin: \"auto\",\n borderRadius: \"50%\",\n backgroundColor: \"white\",\n color: \"white\",\n cursor: \"pointer\",\n transition: \"0.5s\",\n },\n icon_active: {\n color: theme.palette.primary.main,\n },\n icon_inactive: {\n color: \"rgb(20,20,20)\",\n },\n name_active: {\n color: theme.palette.primary.main,\n },\n name_inactive: {\n color: \"rgb(20,20,20)\",\n },\n linkButton: {\n margin: \"3px 0em\",\n width: \"100%\",\n borderRadius: \"5px\",\n padding: \"10px\",\n },\n active: {\n backgroundColor: \"#002d621f\",\n },\n inactive: {\n backgroundColor: \"transparent\",\n },\n});\n\nconst connectedAgentsAdsNavBarDesktop = connect((state) => ({\n agent: state.login.user,\n partner: state.partner.partner,\n}))(AgentsAdsNavBarDesktop);\n\nexport default withStyles(styles)(connectedAgentsAdsNavBarDesktop);\n","import React, { Component } from \"react\";\nimport { withStyles, Grid, Typography, Box } from \"@material-ui/core\";\nimport { connect } from \"react-redux\";\nimport { Redirect, Route, Switch } from \"react-router-dom\";\nimport { fetchPartnerDetails } from \"../../store/partnerState\";\nimport AgentsAdsNavBarDesktop from \"./Navbar/AgnetAdsNavBarDesktop\";\nimport MetaAds from \"../../pages/MetaAds/MetaAds\";\n\nclass AgentsMetaAdsLayout extends Component {\n constructor(props) {\n super(props);\n this.state = {\n agentId: props.match?.params?.id,\n view: props.location?.pathname?.includes(\"/ads/\") ? \"ads\" : \"projects\",\n };\n }\n componentDidMount() {\n this.props.fetchPartnerDetails();\n }\n\n render() {\n const { classes, agent, partner, user } = this.props;\n const { type } = partner || {};\n const { agentId } = this.state;\n const baseroute = `/agents/meta-ads`;\n\n return (\n \n \n \n \n \n \n \n {/* Default route */}\n (\n \n )}\n />\n\n (\n \n )}\n />\n\n (\n \n )}\n />\n \n
\n \n \n );\n }\n}\n\nconst styles = (theme) => ({\n root: {\n width: \"100%\",\n height: \"100vh\",\n overflow: \"hidden\",\n boxSizing: \"border-box\",\n },\n routeWrapper: {\n height: \"100vh\",\n overflow: \"hidden\",\n boxSizing: \"border-box\",\n background: \"white\",\n },\n routeContainer: {\n width: \"100%\",\n height: \"100vh\",\n transition: \"1s\",\n overflow: \"hidden\",\n [theme.breakpoints.down(\"sm\")]: {},\n },\n});\n\nconst connectedAgentsMetaAdsLayout = connect(\n (state) => ({\n user: state.login.user,\n agent: state.login.user,\n partner: state.partner.partner,\n }),\n { fetchPartnerDetails }\n)(AgentsMetaAdsLayout);\n\nexport default withStyles(styles)(connectedAgentsMetaAdsLayout);\n","import React, { Component } from \"react\";\nimport {\n withStyles,\n Grid,\n Box,\n Checkbox,\n Typography,\n Dialog,\n DialogActions,\n DialogContent,\n DialogContentText,\n DialogTitle,\n Button,\n TextField,\n IconButton,\n CircularProgress,\n} from \"@material-ui/core\";\n\nimport clsx from \"clsx\";\n// import AssistantsDetails from \"./AssistantsDetails\";\nimport { withRouter } from \"react-router-dom\";\nimport { connect } from \"react-redux\";\nimport { Skeleton } from \"@material-ui/lab\";\n\nconst leads = [\n {\n name: \"Name\",\n size: 2,\n key: \"agentName\",\n },\n {\n name: \"Project Name\",\n size: 2,\n key: \"assistantName\",\n },\n {\n name: \"Aggregated Ad Recharge\",\n size: 2,\n key: \"aggAdCreditRecharged\",\n },\n {\n name: \"Phone Number\",\n size: 2,\n key: \"agentPhone\",\n },\n {\n name: \"Company Name\",\n size: 2,\n key: \"agentCompanyName\",\n },\n {\n name: \"Industry\",\n size: 2,\n key: \"agentIndustry\",\n },\n {\n name: \"Created At\",\n size: 2,\n key: \"createdAt\",\n },\n];\n\nconst wabaLive = [\n {\n name: \"Name\",\n size: 2,\n key: \"agentName\",\n },\n {\n name: \"Project Name\",\n size: 2,\n key: \"assistantName\",\n },\n {\n name: \"Phone Number\",\n size: 2,\n key: \"agentPhone\",\n },\n {\n name: \"WABA Number\",\n size: 2,\n key: \"whatsappNumber\",\n },\n {\n name: \"Company Name\",\n size: 2,\n key: \"agentCompanyName\",\n },\n {\n name: \"Industry\",\n size: 2,\n key: \"agentIndustry\",\n },\n {\n name: \"Created At\",\n size: 2,\n key: \"createdAt\",\n },\n];\n\nconst conversions = [\n {\n name: \"Project Name\",\n key: \"assistantName\",\n size: 3,\n },\n {\n name: \"Plan Purchased\",\n key: \"planPurchased\",\n size: 3,\n },\n {\n name: \"Currency\",\n key: \"currency\",\n size: 3,\n },\n // {\n // name: \"Amount\",\n // key: \"amountPaid\",\n // size: 2,\n // },\n // {\n // name: \"Commision Earned\",\n // key: \"commision\",\n // size: 2,\n // },\n // {\n // name: \"Commision Percentage\",\n // key: \"commisionPercent\",\n // size: 2,\n // },\n {\n name: \"Purchase Date\",\n key: \"createdAt\",\n size: 3,\n },\n];\n\nclass BusinessesTable extends Component {\n constructor(props) {\n super(props);\n this.state = {\n openDialog: false,\n selectedTemplate: null,\n };\n }\n\n listScrolled = (event) => {\n if (!this.offsetPosition) {\n this.offsetPosition = this.rootEle.current.offsetTop;\n }\n const container = event.currentTarget;\n const currentPos = container.scrollTop;\n this.setState({\n sticky: currentPos > this.offsetPosition,\n });\n };\n toggleDialog = () => {\n this.setState({ openDialog: !this.state.openDialog });\n };\n openTemplate = (e) => {\n const templates = [...this.props.templates];\n const foundIndex = templates.findIndex((x) => x._id == e._id);\n this.props.history.push(\"/assistants/\" + templates[foundIndex]._id);\n // this.setState({ openDialog: true, selectedTemplate: foundIndex });\n };\n render() {\n const { classes, templates, tab } = this.props;\n\n const fields =\n tab === \"Signups\" ? leads : tab === \"WABA Live\" ? wabaLive : conversions;\n\n return (\n <>\n \n \n \n \n \n \n \n {fields.map((e) => (\n \n \n \n {e.name}\n \n \n \n ))}\n \n {templates.map((e) => (\n \n {fields.map((field) => (\n this.openTemplate(e)\n // }\n >\n | \n \n ))}\n \n ))}\n \n \n \n \n \n
\n\n \n {/* Assistants Details */}\n {/* */}\n \n >\n );\n }\n}\n\nfunction Cell({\n classes,\n item,\n field,\n allBusiness,\n openCampaign,\n partner,\n ...props\n}) {\n switch (field) {\n case \"commision\":\n return (\n \n \n \n {item.currency === \"USD\" ? \"$\" : \"₹\"}\n {(item[field] ? item[field] / 100000 : 0).toFixed(2)}\n \n \n \n );\n break;\n\n case \"amountPaid\":\n return (\n \n \n \n {item.currency === \"USD\" ? \"$\" : \"₹\"}\n {(item[field] ? item[field] / 100000 : 0).toFixed(2)}\n \n \n \n );\n break;\n\n case \"commisionPercent\":\n return (\n \n \n \n {item[field] ? item[field] / 100000 : 0}%\n \n \n \n );\n\n case \"displayNameVerified\":\n return (\n \n \n \n {item[field] ? \"Verified\" : \"Not Verified\"}\n \n \n \n );\n\n case \"assistantName\":\n return (\n \n \n {item[field]} \n \n \n );\n break;\n case \"createdAt\":\n return (\n \n \n \n {getFormattedDate(item[field])}\n \n \n \n );\n\n case \"aggAdCreditRecharged\":\n return (\n \n \n \n {item[field] ? item[field] / 100000 : 0}\n \n \n \n );\n\n case \"Loading\":\n return (\n \n \n {Array.from(Array(20).keys()).map((i) => {\n return (\n \n \n
\n );\n })}\n \n \n );\n break;\n\n case \"agentPhone\":\n return (\n \n \n \n {partner?.isChildAffiliate\n ? maskPhoneNumber(item[field])\n : item[field]}\n \n \n \n );\n\n default:\n return (\n \n \n {item[field]} \n \n \n );\n }\n}\n\nexport function getFormattedDate(date) {\n const d = new Date(date);\n const options = { year: \"numeric\", month: \"long\", day: \"numeric\" };\n return d.toLocaleDateString(\"en-US\", options);\n}\n\nexport const maskPhoneNumber = (phone) => {\n if (phone) return phone?.replace(/.(?=.{4})/g, \"*\");\n};\n\nconst styles = (theme) => ({\n root: {\n width: \"100%\",\n height: \"100%\",\n // background: \"red\",\n // display: \"flex\",\n background: \"rgb(249,249,249)\",\n boxSizing: \"border-box\",\n overflowX: \"hidden\",\n position: \"relative\",\n // marginBottom: \"5em\",\n // paddingBottom: \"80px\"\n },\n tableWrapper: {\n overflowX: \"auto\",\n width: \"100%\",\n },\n\n tableContainer: {\n marginTop: \"1.5em\",\n minWidth: 1800,\n },\n\n tableItem: {\n background: \"white\",\n marginBottom: \"1em\",\n padding: 4,\n flexWrap: \"nowrap\",\n borderRadius: 8,\n // cursor: \"pointer\",\n // \"&:hover\": {\n // background: \"#efefef\",\n // },\n },\n tableItemNoHover: {\n background: \"white\",\n marginBottom: \"1em\",\n padding: 4,\n flexWrap: \"nowrap\",\n borderRadius: 8,\n },\n tableItem1: {\n marginBottom: \"1em\",\n padding: 4,\n flexWrap: \"nowrap\",\n borderRadius: 8,\n },\n\n tableHeader: {\n background: \"white\",\n padding: 4,\n borderRadius: 8,\n marginBottom: \"1em\",\n flexWrap: \"nowrap\",\n },\n row: {\n background: \"white\",\n display: \"flex\",\n },\n alterRow: {\n background: \"rgb(246,246,246)\",\n },\n dataRow: {\n cursor: \"pointer\",\n transition: \"0.3s\",\n \"&:hover\": {\n background: \"rgb(70 20 134 / 15%)\",\n },\n },\n topbar: {\n background: \"rgb(252,252,252)\",\n position: \"sticky\",\n top: \"0\",\n zIndex: \"100\",\n },\n column: {\n // width: \"4400px\",\n fontWeight: \"normal\",\n height: \"50px\",\n display: \"flex\",\n alignItems: \"center\",\n textAlign: \"center\",\n paddingRight: theme.spacing(2),\n paddingLeft: theme.spacing(2),\n paddingTop: theme.spacing(2),\n paddingBottom: theme.spacing(2),\n boxSizing: \"border-box\",\n },\n});\n\n// export default withStyles(styles)(withRouter(BusinessesTable));\n\nconst connectAssistantsTable = connect((state) => ({\n allBusiness: state.business.allBusiness,\n isLoading: state.assistants.isLoading,\n partner: state.partner.partner,\n}))(BusinessesTable);\n\nexport default withStyles(styles)(withRouter(connectAssistantsTable));\n","import React, { Component } from \"react\";\nimport { DateRangePicker } from \"react-dates\";\nimport { connect } from \"react-redux\";\nimport moment from \"moment\";\nimport { withRouter } from \"react-router-dom\";\nimport {\n withStyles,\n Button,\n Box,\n Typography,\n Grid,\n IconButton,\n TextField,\n Tabs,\n Tab,\n DialogContent,\n Dialog,\n DialogActions,\n FormControl,\n Select,\n MenuItem,\n InputLabel,\n Snackbar,\n CircularProgress,\n Popper,\n InputAdornment,\n Tooltip,\n} from \"@material-ui/core\";\nimport {\n NavigateNext,\n NavigateBefore,\n Clear,\n FilterList,\n Add,\n Visibility,\n VisibilityOff,\n InfoOutlined,\n CheckCircleOutlined,\n GetApp,\n} from \"@material-ui/icons\";\nimport \"react-dates/initialize\";\nimport \"react-dates/lib/css/_datepicker.css\";\nimport \"../../react-dates-custom.css\";\n\nimport BusinessesTable, { fields, getFormattedDate } from \"./BusinessesTable\";\nimport { TablePagination } from \"@material-ui/core\";\n// import { assistants } from \"../../config/assistants\";\nimport { URL, partnerURL } from \"../../config/config\";\nimport axios from \"axios\";\nimport {\n fetchAssistants,\n loadNextAssistants,\n loadPrevAssistants,\n changeRowsPerPage,\n fetchAssistantsCounts,\n} from \"../../store/assistantState\";\nimport { fetchBusiness } from \"../../store/businessState\";\nimport { Alert, Autocomplete, Skeleton } from \"@material-ui/lab\";\nimport { timeZones } from \"../../config/timezone\";\nimport dialCodes, {\n countryMap,\n isoMap,\n isoToDialCode,\n} from \"../../config/dialCodes\";\nimport Header from \"../../commons/Header/Header\";\nimport CreateProjectDialog from \"../../commons/Dialog/CreateProjectDialog\";\nimport plansHelper from \"../../helpers/plansHelper\";\nimport withMediaQuery from \"../../helpers/mediaQueryHelper\";\n\nclass AllBusinesses extends Component {\n constructor(props) {\n super(props);\n this.state = {\n alertType: \"\",\n alertMsg: \"\",\n rowsPerPage: 15,\n totalCount: null,\n skip: 0,\n timeStamp: new Date(),\n search: \"\",\n searchDone: false,\n templates: [],\n backupTemplates: [],\n backupSearchTemplates: [],\n err: null,\n isLoading: false,\n page: 0,\n applied: {\n createdAt: {\n startDate: null,\n endDate: null,\n focus: null,\n },\n planActivatedOn: {\n startDate: null,\n endDate: null,\n focus: null,\n },\n },\n tab: \"Signups\",\n openDialog: false,\n assistantFilter: {\n filter: \"Signups\",\n sort: \"dec\",\n },\n allCount: 0,\n wabaCount: 0,\n countryCode: isoToDialCode[this.props.partner?.isoCode] || \"+91\",\n liveCount: 0,\n draftCount: 0,\n trialPlanCount: 0,\n monthlyPlanCount: 0,\n yearlyPlanCount: 0,\n deletedCount: 0,\n stoppedCount: 0,\n openProjectDialog: false,\n selectBusiness: \"\",\n newProjectName: \"\",\n projectSubmitLoader: false,\n clientDialog: false,\n company: \"\",\n companySize: \"\",\n industry: \"\",\n display_name: \"\",\n contact: \"\",\n buttonDisable: false,\n timezone: \"Asia/Calcutta GMT+05:30\",\n currency: this.props.partner?.currency || \"INR\",\n };\n }\n componentDidMount() {\n const { partner } = this.props;\n const { isSuperAffiliate } = partner || {};\n this.props.fetchAssistantsCounts(isSuperAffiliate);\n this.props.fetchAssistants(this.state.assistantFilter, isSuperAffiliate);\n }\n\n toggleDialog = (event, reason) => {\n reason !== \"backdropClick\" &&\n this.setState({ openDialog: !this.state.openDialog });\n };\n\n handleChangePage = (event, newPage) => {\n const { page } = this.props;\n const { partner } = this.props;\n const { isSuperAffiliate } = partner || {};\n const step = newPage - page;\n if (step === -1) {\n this.props.loadPrevAssistants();\n } else if (step === 1) {\n this.props.loadNextAssistants(\n this.state.assistantFilter,\n isSuperAffiliate\n );\n }\n };\n\n handleChangeRowsPerPage = (event) => {\n const rowsPerPage = parseInt(event.target.value, 10);\n this.props.changeRowsPerPage(rowsPerPage);\n this.onTabChange();\n };\n\n setTemplate = (template) => {\n const templates = [...this.state.templates];\n const foundIndex = templates.findIndex((x) => x._id == template._id);\n // console.log({foundIndex})\n // if(foundIndex > -1) {\n templates[foundIndex] = template;\n this.setState({ templates });\n // } else {\n // }\n // templates.forEach\n };\n\n handleBusinessInput = (e) => {\n this.setState({ [e.target.name]: e.target.value });\n };\n\n handleInput = (e) => {\n this.setState({ [e.target.name]: e.target.value });\n };\n\n handleEnter = (e) => {\n // const { backupSearchTemplates, search } = this.state;\n if (e.key === \"Enter\") {\n const { assistantFilter } = this.state;\n const newObj = { ...assistantFilter };\n if (e.target.value) {\n newObj.assistantName = e.target.value;\n } else {\n delete newObj.assistantName;\n }\n this.setState(\n { assistantFilter: newObj, searchDone: true },\n this.onTabChange\n );\n }\n };\n handleSearch = (e) => {\n const { assistantFilter } = this.state;\n const newObj = { ...assistantFilter, assistantName: e.target.value };\n this.setState({\n assistantFilter: newObj,\n search: e.target.value,\n });\n };\n handleClear = () => {\n const { assistantFilter } = this.state;\n delete assistantFilter.assistantName;\n this.setState(\n {\n search: \"\",\n searchDone: false,\n assistantFilter,\n },\n this.onTabChange\n );\n };\n\n // methods for date filter\n clearDate = (filterKey) => {\n const applied = { ...this.state.applied };\n applied[filterKey].startDate = null;\n applied[filterKey].endDate = null;\n this.setState({ applied });\n };\n checkDateInput = (filterKey) => {\n const applied = { ...this.state.applied };\n const s = applied[filterKey];\n if (s.startDate && s.endDate) {\n return;\n } else {\n this.clearDate(filterKey);\n }\n };\n setFocus = (filterKey, focusedInput) => {\n const applied = { ...this.state.applied };\n applied[filterKey].focusedInput = focusedInput;\n this.setState({ applied });\n };\n setDate = (filterKey, startDate, endDate) => {\n const applied = { ...this.state.applied };\n applied[filterKey].startDate = startDate;\n applied[filterKey].endDate = endDate;\n this.setState({ applied });\n };\n applyFilter = () => {\n const { applied, assistantFilter } = this.state;\n const filterObj = { ...assistantFilter };\n if (applied.createdAt.startDate && applied.createdAt.endDate) {\n filterObj.fromDate = applied.createdAt.startDate._d;\n filterObj.toDate = applied.createdAt.endDate._d;\n } else {\n delete filterObj.fromDate;\n delete filterObj.toDate;\n }\n\n if (applied.planActivatedOn.startDate && applied.planActivatedOn.endDate) {\n filterObj.fromPlanActivatedDate = applied.planActivatedOn.startDate._d;\n filterObj.toPlanActivatedDate = applied.planActivatedOn.endDate._d;\n } else {\n delete filterObj.fromPlanActivatedDate;\n delete filterObj.toPlanActivatedDate;\n }\n this.setState(\n {\n assistantFilter: filterObj,\n openDialog: false,\n },\n () => {\n this.onTabChange();\n }\n );\n };\n\n onTabChange = () => {\n const { partner } = this.props;\n const { isSuperAffiliate } = partner || {};\n this.props.fetchAssistants(this.state.assistantFilter, isSuperAffiliate);\n };\n\n projectDialog = (event) => {\n this.setState({ openProjectDialog: !this.state.openProjectDialog });\n };\n\n toggleclientDialog = () => {\n this.setState({ clientDialog: !this.state.clientDialog });\n };\n\n buttondisbale = () => {\n this.setState({ buttonDisable: !this.state.buttonDisable });\n };\n\n createclient = async () => {\n try {\n const partnerId = this.props.user.id;\n const {\n email,\n display_name,\n company,\n contact,\n currency,\n timezone,\n password,\n } = this.state;\n const queryObj = {\n email,\n password,\n display_name,\n company,\n contact,\n timezone,\n currency,\n };\n\n let response = await axios.post(\n partnerURL + `/partner/${partnerId}/business/`,\n queryObj\n );\n // this.props.createnewClient(response.data)\n // this.props.fetchBusiness();\n this.setState({\n clientDialog: !this.state.clientDialog,\n openProjectDialog: true,\n });\n this.setState({\n status: \"success\",\n statusMessage: \"Business Created Succesfully\",\n });\n this.setState({\n email: null,\n display_name: null,\n company: null,\n contact: null,\n currency: this.props.partner?.currency || \"INR\",\n timezone: \"Asia/Calcutta GMT+05:30\",\n countryCode: isoToDialCode[this.props.partner?.isoCode] || \"+91\",\n industry: null,\n companySize: null,\n password: null,\n });\n } catch (error) {\n this.setState({\n buttonDisable: false,\n });\n const errorResponse = error.response;\n console.error(error);\n const errorMessage = errorResponse?.data?.message;\n this.setState({\n status: \"error\",\n statusMessage: errorMessage,\n });\n }\n };\n\n onSnackbarClose = () => {\n this.setState({\n alertType: \"\",\n alertMsg: \"\",\n });\n };\n\n selectBusiness = (e, value, reason) => {\n if (reason === \"select-option\") {\n const assistantValue = Object.keys(this.props.allBusiness).find(\n (key) => this.props.allBusiness[key].name === value.split(\" - \")[1]\n );\n this.setState({ selectBusiness: assistantValue });\n }\n };\n\n render() {\n const {\n classes,\n partner,\n partnerPlanFamily,\n partnerWccFamily,\n mediaQuery,\n partnerId,\n user,\n } = this.props;\n const { backupTemplates, tab, assistantFilter, applied } = this.state;\n const { assistants, page, rowsPerPage, totalAssistants } = this.props;\n\n const affiliateTabs = [\n {\n display: \"Businesses\",\n state: \"draftCount\",\n theme: \"#00b4bf59\",\n },\n // {\n // display: \"WABA Live\",\n // state: \"wabaCount\",\n // theme: \"#f2c14e3d\",\n // list: backupTemplates,\n // },\n // {\n // display: \"Conversions\",\n // state: \"liveCount\",\n // theme: \"#08cf6533\",\n // list: backupTemplates.filter((i) => !!i.isWhatsappVerified),\n // },\n ];\n\n const tabs = affiliateTabs;\n\n const passwordRegex = /^(?=.*[A-Za-z])(?=.*\\d)[A-Za-z\\d]{8,}$/;\n const isPasswordValid = passwordRegex.test(this.state.password);\n\n return (\n \n {/* Page name & description container */}\n
\n \n \n }\n // buttonTitle={user.isReferral ? \"\" : \"Create New Project\"}\n onButtonClick={this.projectDialog}\n showSearch={true}\n searchPlaceholder={\"Search business name\"}\n handleSearch={this.handleSearch}\n handleEnter={this.handleEnter}\n searchDone={this.state.searchDone}\n handleClear={this.handleClear}\n filter={this.state.search}\n toggleFilterDialog={this.toggleDialog}\n filterColor={\n (!!applied.createdAt.startDate &&\n !!applied.createdAt.endDate) ||\n (!!applied.planActivatedOn.endDate &&\n !!applied.planActivatedOn.startDate)\n }\n />\n \n {}}\n indicatorColor=\"primary\"\n textColor=\"#000\"\n variant=\"scrollable\"\n scrollButtons=\"auto\"\n aria-label=\"scrollable auto tabs example\"\n TabIndicatorProps={{\n className: classes.tabIndicator,\n }}\n >\n {tabs.map((i) => (\n \n \n \n \n {totalAssistants}\n \n \n \n \n {`${i.display}`}\n \n >\n }\n value={i.display}\n style={{\n textTransform: \"none\",\n alignItems: \"end\",\n minWidth: 190,\n // marginLeft: 16,\n paddingBottom: 8,\n // background: `linear-gradient(180deg, white, ${\"#00b4bf59\"})`,\n }}\n />\n ))}\n \n \n \n \n \n
\n {/* Layout */}\n
\n \n {/* First section */}\n {/* Scrollable (x & y) table container */}\n \n \n
\n \n \n {/* Table footer fixed at bottom */}\n
\n {/* Filter dialog */}\n
\n \n \n \n Filter \n \n \n \n \n \n \n \n \n \n \n \n \n \n Apply\n \n \n \n \n\n {/* project dialog */}\n {this.state.openProjectDialog && (\n
\n this.setState({ newProjectName: e.target.value })\n }\n createBusiness={() =>\n this.setState({\n openProjectDialog: false,\n clientDialog: true,\n })\n }\n onSubmit={this.projectSubmit}\n disableSubmit={this.state.projectSubmitLoader}\n planFamilies={partnerPlanFamily}\n defaultPlanFamily={partner?.defaultPlanFamily}\n defaultPlanName={partner?.defaultPlanName}\n wccPlans={partnerWccFamily}\n defaultWccPlan={partner?.defaultWccPlan}\n type={partner?.type}\n partner={partner}\n />\n )}\n\n \n \n {this.state.alertMsg}\n \n \n \n );\n }\n}\n\n// Navigattion icons used in date fliters\nfunction NavigationWrapper(props) {\n const { classes } = props;\n return (\n \n {props.children} \n
\n );\n}\n\nfunction DateFilter({\n filterKey,\n applied,\n setFocus,\n setDate,\n checkDateInput,\n clearDate,\n ...props\n}) {\n const { classes, filterName } = props;\n const A = applied[filterKey];\n return (\n \n \n {A.startDate && A.endDate ? (\n \n clearDate(filterKey)}\n >\n \n \n \n ) : (\n \"\"\n )}\n \n \n {filterName}\n \n \n \n checkDateInput(filterKey)}\n startDateId=\"startDate\"\n endDateId=\"endDate\"\n startDate={A.startDate}\n endDate={A.endDate}\n onDatesChange={({ startDate, endDate }) =>\n setDate(filterKey, startDate, endDate)\n }\n focusedInput={A.focusedInput}\n onFocusChange={(focusedInput) => setFocus(filterKey, focusedInput)}\n navPosition=\"navPositionTop\"\n numberOfMonths={1}\n navPrev={\n \n \n \n }\n navNext={\n \n \n \n }\n hideKeyboardShortcutsPanel\n customArrowIcon={null}\n screenReaderInputMessage={\" \"}\n small\n readOnly\n isOutsideRange={(day) => moment().diff(day) < 0}\n />\n \n \n \n );\n\n // function setToday()\n}\n\nconst styles = (theme) => ({\n root: {\n width: \"100%\",\n height: \"100%\",\n // background: \"red\",\n // display: \"flex\",\n background: \"rgb(249,249,249)\",\n boxSizing: \"border-box\",\n overflowX: \"hidden\",\n position: \"relative\",\n // paddingBottom: \"80px\"\n },\n pageTitleContainer: {\n position: \"sticky\",\n zIndex: 100,\n top: 0,\n height: 60,\n boxSizing: \"border-box\",\n [theme.breakpoints.down(\"md\")]: {\n // paddingTop: \"40px\",\n },\n [theme.breakpoints.down(\"sm\")]: {\n // paddingTop: \"20px\",\n },\n },\n fullWidth: {\n width: \"100%\",\n background: \"white\",\n },\n container: {\n background: \"white\",\n borderRadius: \"8px\",\n },\n fixedBottomContainer: {\n position: \"fixed\",\n height: \"60px\",\n borderTop: \"1px solid lightgrey\",\n background: \"white\",\n bottom: \"0\",\n left: \"71px\",\n right: \"0\",\n overflow: \"hidden\",\n display: \"flex\",\n alignItems: \"center\",\n justifyContent: \"center\",\n [theme.breakpoints.down(\"sm\")]: {\n left: \"0px\",\n height: \"50px\",\n },\n },\n tableContainer: {\n top: \"60px\",\n height: \"calc(100vh - 260px)\",\n width: \"100%\",\n overflow: \"hidden\",\n // background: \"red\",\n boxSizing: \"border-box\",\n position: \"sticky\",\n [theme.breakpoints.down(\"xs\")]: {\n // position top + bottombar height + bottom padding + navbar\n marginTop: 48,\n height: \"calc(100vh - 70px - 50px - 10px - 50px)\",\n paddingLeft: theme.spacing(1),\n paddingRight: theme.spacing(1),\n boxSizing: \"border-box\",\n },\n },\n textField: {\n width: \"100%\",\n height: 42,\n },\n sortFormControl: {\n width: 120,\n marginRight: 8,\n height: 32,\n borderRadius: \"4px\",\n border: \"1px solid lightgrey\",\n },\n textFieldPopper: {\n [theme.breakpoints.down(\"md\")]: {\n left: \"12px !important\",\n minWidth: \"calc(100% - 35px)\",\n },\n [theme.breakpoints.up(\"md\")]: {\n left: \"calc(12.5vw - 16px) !important\",\n minWidth: \"calc(25vw + 54px)\",\n },\n },\n tabContainer: {\n background: \"white\",\n boxShadow: \"0 0 12px rgb(171 170 170)\",\n // padding: \"24px\",\n },\n tabIndicator: {\n borderRadius: \"3px 3px 0px 0px\",\n height: 3,\n },\n textFieldRootAutocomplete: {\n width: \"100%\",\n marginTop: \"4px\",\n marginBottom: \"20px\",\n height: 32,\n background: \"white\",\n border: \"2px solid grey\",\n \"& input\": {\n padding: \"0px 12px!important\",\n },\n \"& label\": {\n background: \"white\",\n color: \"#b2b3b3 !important\",\n padding: \"0 6px\",\n marginTop: -4,\n },\n },\n dialogContainer: {\n // zIndex: \"3400 !important\",\n \"& .MuiPaper-root\": {\n [theme.breakpoints.down(\"md\")]: {\n width: \"100%\",\n // height: \"100%\",\n // maxHeight: \"100%\",\n margin: \"0\",\n borderRadius: \"0px\",\n },\n [theme.breakpoints.up(\"md\")]: {\n width: \"100%\",\n // height: \"unset\",\n // maxHeight: \"unset\",\n // minHeight: \"600px\",\n margin: \"unset\",\n borderRadius: \"6px\",\n },\n },\n },\n projectdialogContainer: {\n // zIndex: \"3400 !important\",\n \"& .MuiPaper-root\": {\n [theme.breakpoints.down(\"md\")]: {\n width: \"100%\",\n // height: \"100%\",\n // maxHeight: \"100%\",\n margin: \"0\",\n borderRadius: \"0px\",\n },\n [theme.breakpoints.up(\"md\")]: {\n width: \"100%\",\n // height: \"unset\",\n // maxHeight: \"unset\",\n // minHeight: \"100px\",\n margin: \"unset\",\n borderRadius: \"6px\",\n },\n },\n },\n\n inputRoot: {\n '&[class*=\"MuiOutlinedInput-root\"].MuiAutocomplete-input': {\n padding: 12,\n },\n '&&[class*=\"MuiOutlinedInput-root\"]': {\n padding: 0,\n },\n },\n popper: {\n width: \"fit-content\",\n zIndex: 10000,\n height: \"10px\",\n },\n snackbar: {\n // marginTop:\"20px\",\n // [theme.breakpoints.down(\"sm\")]: {\n // marginTop:\"40px\",\n // },\n },\n textFieldRootAutocomplete: {\n width: \"100%\",\n padding: \"2px 8px\",\n marginTop: \"4px\",\n marginBottom: \"20px\",\n background: \"white\",\n border: \"2px solid grey\",\n \"& label\": {\n background: \"white\",\n color: \"#b2b3b3 !important\",\n padding: \"0 6px\",\n marginTop: -4,\n },\n },\n phoneTextFieldRoot: {\n width: \"100%\",\n padding: \"8px 20px\",\n marginTop: \"4px\",\n marginBottom: \"20px\",\n \"& input\": { width: \"calc(100% - 80px)\", marginLeft: \"80px\" },\n background: \"white\",\n border: \"2px solid grey\",\n },\n codeTextFieldRoot: {\n width: \"90px\",\n padding: \"0 0 0 14px\",\n position: \"absolute\",\n zIndex: 1,\n margin: \"10px 0 0 2px\",\n background: \"white\",\n \"& input\": {\n padding: \"10px 0 !important\",\n },\n \"& button\": {\n width: \"12px\",\n position: \"relative\",\n left: \"7px\",\n background: \"white\",\n borderRadius: 0,\n \"&:hover\": {\n background: \"white\",\n },\n },\n \"& div\": {\n paddingRight: \"0px !important\",\n },\n },\n textFieldRoot: {\n width: \"100%\",\n padding: \"8px 20px 10px\",\n marginTop: \"4px\",\n marginBottom: \"20px\",\n background: \"white\",\n border: \"2px solid grey\",\n \"& input\": {\n padding: \"6px 12px!important\",\n },\n \"& label\": {\n background: \"white\",\n color: \"#b2b3b3 !important\",\n padding: \"0 6px\",\n marginTop: -4,\n },\n },\n});\n\nconst connectAssistants = connect(\n (state) => ({\n user: state.login.user,\n assistants: state.assistants.assistants,\n page: state.assistants.page,\n isAssistantsLoading: state.assistants.isLoading,\n assistantError: state.assistants.error,\n rowsPerPage: state.assistants.rowsPerPage,\n totalAssistants: state.assistants.total,\n allCount: state.assistants.allCount,\n liveCount: state.assistants.liveCount,\n wabaCount: state.assistants.wabaCount,\n draftCount: state.assistants.draftCount,\n trialPlanCount: state.assistants.trialPlanCount,\n monthlyPlanCount: state.assistants.monthlyPlanCount,\n yearlyPlanCount: state.assistants.yearlyPlanCount,\n deletedCount: state.assistants.deletedCount,\n stoppedCount: state.assistants.stoppedCount,\n allBusiness: state.business.allBusiness,\n partnerId: state.partner.partner._id,\n partner: state.partner.partner,\n partnerSecretHash: state.partner.partner.passwordHash,\n partnerPlanFamily: state.planFamilies.allPlanFamilies,\n partnerWccFamily: state.wccPlans.allWccPlans,\n }),\n {\n fetchAssistants,\n loadNextAssistants,\n loadPrevAssistants,\n changeRowsPerPage,\n fetchAssistantsCounts,\n fetchBusiness,\n }\n)(AllBusinesses);\n\nexport default withStyles(styles)(\n withMediaQuery(\"(max-width:600px)\")(withRouter(connectAssistants))\n);\n","import React from \"react\";\nimport {\n PersonOutlined,\n DashboardOutlined,\n Build,\n TuneOutlined,\n Facebook,\n} from \"@material-ui/icons\";\nimport AgentProfilePage from \"../../../pages/AgentProfile/AgentProfilePage\";\nimport DashboardPage from \"../../../pages/Dashboard/DashboardPage\";\nimport AssistantsRoute from \"../../../pages/Assistants/AssistantsRoute\";\nimport BillingRoute from \"../../../pages/BillingProfile/BillingRoute\";\nimport PlanFamilyPage from \"../../../pages/Manage/PlanFamilyPage\";\nimport DefaultsPage from \"../../../pages/Manage/DefaultsPage\";\nimport DashboardIcon from \"@material-ui/icons/Dashboard\";\nimport ContactlessIcon from \"@material-ui/icons/Contactless\";\nimport BlurOnIcon from \"@material-ui/icons/BlurOn\";\nimport MetaAdsLayout from \"../MetaAdsLayout\";\nimport MetaAds from \"../../../pages/MetaAds/MetaAds\";\nimport BarChartOutlinedIcon from \"@material-ui/icons/BarChartOutlined\";\nimport AccountTreeOutlinedIcon from \"@material-ui/icons/AccountTreeOutlined\";\nimport SuperDashboard from \"../../../pages/SuperAffiliate/SuperDashboard\";\nimport ViewQuiltIcon from \"@material-ui/icons/ViewQuilt\";\nimport BusinessIcon from \"@material-ui/icons/Business\";\nimport SuperAffiliateRoute from \"../../../pages/SuperAffiliate/SuperAffiliateRoute\";\nimport AgentsMetaAdsLayout from \"../agentsMetaAdLayout\";\nimport GroupIcon from '@material-ui/icons/Group';\nimport AssistantsPage from \"../../../pages/Assistants/AssistantsPage\";\nimport AllBusinessesPage from \"../../../pages/SuperAffiliate/AllBusinessesPage\";\n\nconst topRoutes = [\n {\n name: \"Dashboard\",\n to: \"/\",\n component: DashboardPage,\n icon: ,\n exact: true,\n },\n {\n name: \"Customers\",\n to: \"/customers\",\n component: AssistantsRoute,\n icon: ,\n },\n {\n name: \"Meta Ads\",\n to: \"/meta-ads\",\n component: MetaAdsLayout,\n icon: ,\n },\n {\n name: \"Payouts\",\n to: \"/payouts\",\n component: BillingRoute,\n icon: ,\n },\n];\n\nconst renderMetaAdsRoute = [\n {\n name: \"Ads Setup\",\n to: \"/project\",\n component: MetaAds,\n icon: ,\n },\n {\n name: \"Ads Overview\",\n to: \"/ads\",\n component: MetaAds,\n icon: ,\n },\n {\n name: \"Businesses\",\n to: \"/businesses\",\n component: AllBusinessesPage,\n icon: ,\n },\n];\n\nconst renderAgnetsMetaAdsRoute = [\n {\n name: \"Ads Setup\",\n to: \"/projects/:id\",\n component: MetaAds,\n icon: ,\n },\n {\n name: \"Ads Overview\",\n to: \"/ads/:id\",\n component: MetaAds,\n icon: ,\n },\n];\n\nconst renderSuperAffiliateRoute = [\n {\n name: \"Dashboard\",\n to: \"/\",\n component: SuperDashboard,\n icon: ,\n exact: true,\n },\n {\n name: \"Agents\",\n to: \"/agents\",\n component: SuperAffiliateRoute,\n icon: ,\n exact: true,\n },\n {\n name: \"Meta Ads\",\n to: \"/meta-ads\",\n component: MetaAdsLayout,\n icon: ,\n },\n {\n name: \"Agent Meta Ads\",\n to: \"/agents/meta-ads/projects/:id\",\n component: AgentsMetaAdsLayout,\n icon: ,\n exact: true,\n hidden: true,\n },\n {\n name: \"Agent Meta Ads\",\n to: \"/agents/meta-ads/ads/:id\",\n component: AgentsMetaAdsLayout,\n icon: ,\n exact: true,\n hidden: true,\n },\n];\n\nconst renderManageRoute = [\n {\n name: \"Configure Plan\",\n to: \"/config\",\n component: PlanFamilyPage,\n icon: ,\n },\n {\n name: \"Defaults\",\n to: \"/defaults\",\n component: DefaultsPage,\n icon: ,\n },\n];\n\nconst bottomRoutes = [\n {\n name: \"Profile\",\n to: \"/agentProfile\",\n component: AgentProfilePage,\n icon: ,\n },\n];\n\nexport {\n topRoutes,\n bottomRoutes,\n renderManageRoute,\n renderMetaAdsRoute,\n renderSuperAffiliateRoute,\n renderAgnetsMetaAdsRoute,\n};\n","import React from \"react\";\nimport { useHistory, useLocation, useParams, Link } from \"react-router-dom\";\nimport { Box, Typography, withStyles } from \"@material-ui/core\";\nimport { connect } from \"react-redux\";\nimport LOGO from \"../../../static/logo.jpg\";\nimport {\n topRoutes,\n bottomRoutes,\n renderSuperAffiliateRoute,\n} from \"../Routes/ProjectRoutes\";\n\nfunction NavBarDesktop(props) {\n const location = useLocation();\n const history = useHistory();\n const params = useParams();\n const { classes, partner } = props;\n const { isAdsCommissionEnabled, isSuperAffiliate, isChildAffiliate } =\n partner || {};\n\n const filteredRoutes = topRoutes.filter((route) => {\n if (route.to === \"/meta-ads\") {\n return isAdsCommissionEnabled;\n }\n\n if (route.to === \"/payouts\" && isChildAffiliate) {\n return false;\n }\n return true;\n });\n\n const routes = isSuperAffiliate\n ? renderSuperAffiliateRoute.filter((route) => !route.hidden)\n : filteredRoutes;\n\n return (\n \n
\n \n \n \n\n {routes.map((route, index) => {\n let className = \"inactive\";\n if (location.pathname.split(\"/\")[1] === route.to.split(\"/\")[1]) {\n className = \"active\";\n }\n\n return (\n {\n history.push(route.to);\n }}\n >\n \n \n {route.icon}\n \n \n \n {route.name}\n \n
\n );\n })}\n \n
\n {bottomRoutes.map((route, index) => {\n let className = \"inactive\";\n if (location.pathname.includes(route.to)) {\n className = \"active\";\n }\n return (\n {\n history.push(route.to);\n }}\n >\n \n \n {route.icon}\n \n \n
\n );\n })}\n \n
\n );\n}\n\nconst styles = (theme) => ({\n root: {\n width: \"70px\",\n height: \"100vh\",\n position: \"fixed\",\n // top: 0,\n left: 0,\n background: \"#002D62\",\n display: \"flex\",\n flexDirection: \"column\",\n justifyContent: \"space-between\",\n color: \"gray\",\n [theme.breakpoints.down(\"sm\")]: {\n display: \"none\",\n },\n overflow: \"auto\",\n },\n logo: {\n height: 50,\n marginTop: 15,\n },\n link_inactive: {\n display: \"flex\",\n justifyContent: \"center\",\n alignItems: \"center\",\n height: \"35px\",\n width: \"35px\",\n margin: \"auto\",\n borderRadius: \"50%\",\n cursor: \"pointer\",\n transition: \"0.5s\",\n },\n link_active: {\n display: \"flex\",\n justifyContent: \"center\",\n alignItems: \"center\",\n height: \"35px\",\n width: \"35px\",\n margin: \"auto\",\n borderRadius: \"50%\",\n backgroundColor: \"white\",\n color: \"white\",\n cursor: \"pointer\",\n transition: \"0.5s\",\n },\n // profile_active: {\n\n // },\n // profile_inactive: {\n\n // }\n icon_active: {\n color: theme.palette.primary.main,\n },\n icon_inactive: {\n color: \"#fbfbfb\",\n },\n name_active: {\n color: \"white\",\n },\n name_inactive: {\n color: \"#fbfbfb\",\n },\n active: {\n margin: \"0.9em 0em\",\n borderLeft: \"2px solid\",\n borderColor: \"white\",\n },\n inactive: {\n margin: \"0.9em 0em\",\n borderLeft: \"2px solid\",\n borderColor: \"transparent\",\n },\n});\n\nconst connectedNavBarDesktop = connect((state) => ({\n agent: state.login.user,\n partner: state.partner.partner,\n}))(NavBarDesktop);\n\nexport default withStyles(styles)(connectedNavBarDesktop);\n","import React from \"react\";\nimport { useHistory, useLocation, useParams } from \"react-router-dom\";\nimport {\n Drawer,\n Box,\n Grid,\n Typography,\n ButtonBase,\n withStyles,\n Accordion,\n AccordionDetails,\n AccordionSummary,\n} from \"@material-ui/core\";\nimport { SettingsOutlined, ExpandMore } from \"@material-ui/icons\";\nimport { KeyboardArrowDown } from \"@material-ui/icons\";\nimport { connect } from \"react-redux\";\nimport LOGO from \"../../../static/logo.jpg\";\nimport {\n topRoutes,\n bottomRoutes,\n renderSuperAffiliateRoute,\n} from \"../Routes/ProjectRoutes\";\n\nconst NavDrawerMobile = (props) => {\n const location = useLocation();\n const history = useHistory();\n const params = useParams();\n const { classes, drawerOpen, onDrawerClose, partner } = props;\n const { type, isAdsCommissionEnabled } = partner || {};\n const openRoute = (route) => {\n history.push(route.to);\n onDrawerClose({});\n };\n const openMangeRoute = (route) => {\n history.push(\"/manage\" + route.to);\n onDrawerClose({});\n };\n\n const { isSuperAffiliate, isChildAffiliate } = partner || {};\n\n const filteredRoutes = topRoutes.filter((route) => {\n if (route.to === \"/meta-ads\") {\n return isAdsCommissionEnabled;\n }\n\n if (route.to === \"/payouts\" && isChildAffiliate) {\n return false;\n }\n\n return true;\n });\n\n const routes = isSuperAffiliate\n ? renderSuperAffiliateRoute.filter((route) => !route.hidden)\n : filteredRoutes;\n\n return (\n \n \n \n \n \n \n \n \n AiSensy\n \n \n \n {routes.map((route, index) => {\n let className = \"inactive\";\n if (location.pathname == route.to) {\n className = \"active\";\n }\n if (route.to == \"/manage\") {\n return;\n }\n return (\n openRoute(route)}\n >\n \n \n \n {route.icon}\n \n \n \n \n {route.name}\n \n \n \n \n );\n })}\n {/* {\n \n }\n aria-controls=\"panel1a-content\"\n id=\"panel1a-header\"\n className={classes.accordianSummary}\n >\n \n \n \n \n { }\n \n \n \n \n Manage\n \n \n \n
\n \n \n \n {[...renderFilteredManageRoutes].map((route, index) => {\n let className = \"inactive\";\n if (location.pathname.includes(route.to)) {\n className = \"active\";\n }\n return (\n !route.excludedTypes?.includes(type) && (\n openMangeRoute(route)}\n >\n \n \n \n {route.icon}\n \n \n \n \n {route.name}\n \n \n \n \n )\n );\n })}\n \n \n \n } */}\n \n \n {bottomRoutes.map((route, index) => {\n return (\n openRoute(route)}>\n {route.icon} \n \n );\n })}\n \n \n );\n};\n\nconst styles = (theme) => ({\n container: {\n width: \"100%\",\n height: \"100%\",\n overflowY: \"auto\",\n boxSizing: \"border-box\",\n overflowX: \"hidden\",\n // minWidth: \"300px\",\n // flexGrow: 1,\n },\n drawerPaper: {\n width: \"calc(100vw - 70px)\",\n height: \"100%\",\n position: \"relative\",\n overflow: \"hidden\",\n },\n logo: {\n height: \"50px\",\n marginRight: \"10px\",\n },\n link_inactive: {\n display: \"flex\",\n justifyContent: \"center\",\n alignItems: \"center\",\n height: \"35px\",\n width: \"35px\",\n margin: \"auto\",\n borderRadius: \"50%\",\n cursor: \"pointer\",\n transition: \"0.5s\",\n },\n link_active: {\n display: \"flex\",\n justifyContent: \"center\",\n alignItems: \"center\",\n height: \"35px\",\n width: \"35px\",\n margin: \"auto\",\n borderRadius: \"50%\",\n backgroundColor: \"white\",\n color: \"white\",\n cursor: \"pointer\",\n transition: \"0.5s\",\n },\n // profile_active: {\n\n // },\n // profile_inactive: {\n\n // }\n icon_active: {\n color: theme.palette.primary.main,\n },\n icon_inactive: {\n color: \"rgb(20,20,20)\",\n },\n name_active: {\n color: theme.palette.primary.main,\n },\n name_inactive: {\n color: \"rgb(20,20,20)\",\n },\n linkButton: {\n margin: \"3px 0em\",\n width: \"100%\",\n borderRadius: \"5px\",\n padding: \"10px\",\n },\n active: {\n backgroundColor: \"rgb(70 20 134 / 15%)\",\n },\n inactive: {\n backgroundColor: \"transparent\",\n },\n fullWidth: {\n width: \"100%\",\n display: \"flex\",\n alignItems: \"flex-start\",\n },\n profileWrapper: {\n position: \"absolute\",\n left: 0,\n bottom: 0,\n width: \"100%\",\n backgroundColor: \"white\",\n paddingTop: theme.spacing(2) * 1.5,\n paddingBottom: theme.spacing(2) * 1.5,\n },\n profileIcon: {\n backgroundColor: theme.palette.primary.main,\n borderRadius: \"50%\",\n color: \"white\",\n width: \"40px\",\n height: \"40px\",\n margin: \"0px 10px\",\n display: \"flex\",\n alignItems: \"center\",\n justifyContent: \"center\",\n },\n});\n\nconst connectedNavDrawerMobile = connect((state) => ({\n agent: state.login.user,\n partner: state.partner.partner,\n}))(NavDrawerMobile);\n\nexport default withStyles(styles)(connectedNavDrawerMobile);\n","import React, { Component } from \"react\";\nimport { withRouter } from \"react-router-dom\";\nimport {\n withStyles,\n IconButton,\n Typography,\n Box,\n Hidden,\n} from \"@material-ui/core\";\nimport { connect } from \"react-redux\";\nimport { FilterList, Search, Menu } from \"@material-ui/icons\";\nimport clsx from \"clsx\";\n\nimport NavDrawerMobile from \"./NavDrawerMobile\";\n\nclass NavBarMobile extends Component {\n state = {\n drawerOpen: false,\n };\n\n onDrawerClose = (event) => {\n if (\n event.type === \"keydown\" &&\n (event.key === \"Tab\" || event.key === \"Shift\")\n ) {\n return;\n }\n this.setState({ drawerOpen: false });\n };\n\n onDrawerOpen = () => {\n this.setState({ drawerOpen: true });\n };\n render() {\n const { drawerOpen } = this.state;\n const { classes, location, agent } = this.props;\n const livePage = false;\n\n return (\n \n \n \n \n \n \n \n AiSensy\n \n \n \n
\n \n );\n }\n}\n\nconst styles = (theme) => ({\n root: {\n // position: \"sticky\",\n // top: \"0\",\n width: \"100%\",\n height: \"50px\",\n overflow: \"hidden\",\n transition: \"0.3s\",\n zIndex: 1000,\n },\n sticky: {\n position: \"sticky\",\n top: 0,\n },\n icon: {\n fontSize: \"1em\",\n // color: \"white\"\n },\n hideNavBar: {\n height: 0,\n },\n});\n\nconst connectedNavBarMobile = connect((state) => ({\n agent: state.login.user,\n}))(NavBarMobile);\n\nexport default withStyles(styles)(withRouter(connectedNavBarMobile));\n","import { Typography, useMediaQuery, useTheme } from \"@material-ui/core\";\nimport { Clock } from \"lucide-react\";\nexport default function WarningBanner({ text, backgroundColor }) {\n const theme = useTheme();\n const isMobile = useMediaQuery(theme.breakpoints.down(\"sm\"));\n\n return (\n \n \n \n {text}\n \n
\n );\n}\n","import React from \"react\";\nimport NavBarDesktop from \"./NavBarDesktop\";\nimport NavBarMobile from \"./NavBarMobile\";\nimport WarningBanner from \"../../WarningBanner/WarningBanner\";\nimport { connect } from \"react-redux\";\n\nfunction NavBar(props) {\n const ignoredId = props.partnerDetail._id;\n\n return (\n \n {ignoredId !== \"67c29170603e6b0bf4f5849b\" && (\n \n )}\n \n \n \n );\n}\n\nexport default connect((state) => ({\n isPartnerDetailLoading: state.partner.isLoading,\n partnerDetail: state.partner.partner,\n}))(NavBar);\n","import React, { Component } from \"react\";\n\nclass ErrorBoundary extends Component {\n state = { hasError: false };\n\n static getDerivedStateFromError(error) {\n // Update state so the next render will show the fallback UI.\n return { hasError: true };\n }\n\n componentDidCatch(error, errorInfo) {\n // You can also log the error to an error reporting service\n console.error(error, errorInfo);\n this.setState({ errorInfo });\n }\n\n render() {\n if (this.state.hasError) {\n // You can render any custom fallback UI\n return (\n \n
Something went wrong. \n \n );\n }\n\n return this.props.children;\n }\n}\n\nexport default ErrorBoundary;\n","import React, { Component } from \"react\";\nimport {\n withStyles,\n Grid,\n Snackbar,\n CircularProgress,\n} from \"@material-ui/core\";\nimport { Alert } from \"@material-ui/lab\";\nimport { Redirect, Route, Switch } from \"react-router-dom\";\nimport { connect } from \"react-redux\";\nimport NavBar from \"./Navbar/NavBar\";\nimport ErrorBoundary from \"../ErrorBoundary/ErrorBoundary\";\nimport {\n topRoutes,\n bottomRoutes,\n renderSuperAffiliateRoute,\n} from \"./Routes/ProjectRoutes\";\n\nclass ProjectsLayout extends Component {\n constructor(props) {\n super(props);\n this.state = {\n isLoading: true,\n };\n }\n\n componentDidMount() {\n if (!this.props.partner?._id) {\n this.props\n .fetchPartnerDetails()\n .then(() => {\n this.setState({ isLoading: false });\n })\n .catch((error) => {\n console.error(\"Failed to fetch partner details:\", error);\n this.setState({ isLoading: false });\n });\n } else {\n this.setState({ isLoading: false });\n }\n }\n\n render() {\n const { classes, partner } = this.props;\n const { isLoading } = this.state;\n const { type, isSuperAffiliate } = partner || {};\n const routes = isSuperAffiliate ? renderSuperAffiliateRoute : topRoutes;\n\n if (isLoading) {\n return (\n \n \n
\n );\n }\n\n return (\n \n
\n
\n \n \n {routes.map((route, index) => {\n if (!route.excludedTypes?.includes(type)) {\n return (\n \n );\n }\n })}\n {bottomRoutes.map((route) => {\n return (\n \n );\n })}\n } />\n \n
\n \n
\n );\n }\n}\n\nconst styles = (theme) => ({\n root: {\n width: \"100vw\",\n height: \"100vh\",\n overflow: \"hidden\",\n [theme.breakpoints.down(\"sm\")]: {\n overflowY: \"auto\",\n position: \"relative\",\n },\n },\n routeContainer: {\n width: \"calc(100% - 70px)\",\n height: \"100vh\",\n transition: \"1s\",\n overflow: \"hidden\",\n marginLeft: 70,\n // background: \"green\",\n [theme.breakpoints.down(\"sm\")]: {\n width: \"100%\",\n height: \"calc(100vh - 50px)\",\n // height: \"100vh\",\n marginLeft: 0,\n },\n },\n fullHeight: {\n [theme.breakpoints.down(\"sm\")]: {\n height: \"100vh\",\n },\n },\n preloader: {\n height: 150,\n },\n});\n\nconst connectedProjectslayout = connect((state) => ({\n agent: state.login.user,\n partner: state.partner.partner,\n isPartnerDetailLoading: state.partner.isLoading,\n tenantDetails: state.tenant.tenant,\n}))(ProjectsLayout);\n\nexport default withStyles(styles)(connectedProjectslayout);\n","import React from \"react\";\nimport { Alert } from \"@material-ui/lab\";\nimport { Snackbar } from \"@material-ui/core\";\n\nexport default function Toast(props) {\n const { open, closeSnackbar, severity, msg, vertical, horizontal, autoHideDuration } = props;\n return (\n \n \n {msg}\n \n \n );\n}\n","export default __webpack_public_path__ + \"static/media/partnerBG.439262e9.png\";","import React, { Component } from \"react\";\nimport { connect } from \"react-redux\";\nimport \"./pages.css\";\nimport { loginUser, loginWithToken } from \"./loginstate\";\nimport { createTheme } from \"@material-ui/core/styles\";\nimport {\n Grid,\n Container,\n TextField,\n ThemeProvider,\n Box,\n Typography,\n Button,\n Link,\n} from \"@material-ui/core\";\nimport { Redirect } from \"react-router-dom\";\nimport Toast from \"../../commons/Toast/Toast\";\nimport partnerBG from \"../../static/partnerBG.png\";\n\nclass Login extends Component {\n constructor(props) {\n super(props);\n this.state = {\n userName: \"\",\n password: \"\",\n status: \"\",\n statusMessage: \"\",\n showSnackbar: false,\n alertMsg: \"\",\n alertSeverity: \"\",\n };\n }\n\n componentDidMount() {\n this.tokenLogin();\n }\n\n componentDidUpdate(prevProps) {\n if (prevProps !== this.props && this.props.error) {\n this.setAlert(\"Invalid login credentials!\", \"error\");\n }\n }\n\n handleChange = (event) => {\n this.setState({ [event.target.name]: event.target.value });\n };\n\n closeSnackbar = () => {\n this.setState({ showSnackbar: false });\n };\n\n setAlert = (msg, severity) => {\n this.setState({\n alertMsg: msg,\n alertSeverity: severity,\n showSnackbar: true,\n });\n };\n\n tokenLogin() {\n const search = this.props.location.search;\n const params = new URLSearchParams(search);\n const token = params.get(\"token\");\n if (!token) return;\n\n this.props.loginWithToken(token);\n }\n\n render() {\n const { userName, password } = this.state;\n const { isAuthenticated, loginUser } = this.props;\n if (isAuthenticated) {\n return ;\n } else {\n return (\n \n \n \n \n \n Affiliate Dashboard\n \n \n \n \n Sign In\n \n \n {\n if (event.key === \"Enter\")\n loginUser(userName, password);\n }}\n style={{\n marginBottom: \"20px\",\n backgroundColor: \"#F1F1F1\",\n borderRadius: \"8px\",\n width: \"calc(100% - 16px)\",\n padding: \"8px\",\n }}\n />\n\n {\n loginUser(userName, password);\n }}\n >\n Continue\n \n \n \n \n \n \n \n Don't have an account?{\" \"}\n \n Let's SignUp\n \n \n \n \n\n \n
\n );\n }\n }\n}\n\nfunction Signup() {\n return (\n <>\n \n OR{\" \"}\n \n \n Continue With Google\n \n \n \n \n Sign up for new account\n \n \n \n >\n );\n}\n\nconst theme = createTheme({\n palette: {\n gray: {\n main: \"#bdbdbd\",\n darker: \"#616161\",\n },\n black: {\n main: \"#212121\",\n contrastText: \"#424242\",\n },\n },\n});\n\nexport default connect(\n (state) => ({\n isAuthenticated: state.login.isAuthenticated,\n isLoading: state.login.isLoading,\n error: state.login.error,\n errmsg: state.login.errmsg,\n }),\n { loginUser, loginWithToken }\n)(Login);\n","export default __webpack_public_path__ + \"static/media/f0d373c302b5c689e4445a1c3a9691a3.f0d373c3.svg\";","import React, { Component } from \"react\";\nimport {\n withStyles,\n Grid,\n Box,\n Typography,\n Button,\n IconButton,\n Snackbar,\n TextField,\n Checkbox,\n FormControlLabel,\n CircularProgress,\n} from \"@material-ui/core\";\nimport {\n Close,\n Facebook,\n FiberManualRecord,\n FileCopyOutlined,\n FilterNone,\n Send,\n VisibilityOutlined,\n} from \"@material-ui/icons\";\nimport { Alert, Autocomplete } from \"@material-ui/lab\";\n// import CallhippoLogo from \"../../static/callhippo_logo.png\";\nimport LOGO from \"../../static/logo.jpg\";\nimport WhatsAppLogo from \"../../static/f0d373c302b5c689e4445a1c3a9691a3.svg\";\nimport dialCodes from \"../../config/dialCodes\";\nimport axios from \"axios\";\nimport { URL } from \"../../config/config\";\nimport { timeZones } from \"../../config/timezone\";\n\nclass Preview extends Component {\n constructor(props) {\n super(props);\n this.state = {\n alert: false,\n alertMsg: \"\",\n alertSeverity: \"\",\n step: 1,\n hasRead: false,\n countryCode: \"+91\",\n contact: \"\",\n otp: \"\",\n otpSent: false,\n fetchLoading: false,\n partnerLogos: {},\n name: \"\",\n company: \"\",\n contact: \"\",\n email: \"\",\n timezone: \"\",\n };\n }\n\n launchWhatsAppSignup = async () => {\n // const { showSnackbar, assistantId, startConfetti } = this.props;\n const assistantId = \"123\";\n const { company, contact, email, timezone } = this.state;\n this.setState({ fbLoading: true });\n // trackEvent({\n // event: \"went_for_whatsapp_signup\",\n // properties: {},\n // });\n window.launchWhatsAppSignup && window.launchWhatsAppSignup();\n window.FB?.login &&\n window.FB.login(\n (response) => {\n if (response.authResponse) {\n // trackEvent({\n // event: \"whatsapp_access_token_success\",\n // properties: {},\n // });\n const accessToken = response.authResponse.accessToken;\n axios\n .post(URL + \"/api/submit-facebook-access-token\", {\n assistantId,\n accessToken,\n })\n .then((data) => {\n if (data.data?.success) {\n // trackEvent({\n // event: \"whatsapp_verification_success\",\n // properties: {},\n // });\n this.setState({\n alertMsg: \"WABA is Live!\",\n alert: true,\n alertSeverity: \"success\",\n });\n // showSnackbar({\n // alertMsg: \"WABA is Live!\",\n // severity: \"success\",\n // alertOpen: true,\n // });\n // startConfetti(assistantId, \"v3\");\n } else {\n // trackEvent({\n // event: \"whatsapp_verification_pending\",\n // properties: {},\n // });\n this.setState({\n alertMsg: \"WABA in progress! Please Sync after 10 minutes\",\n alert: true,\n alertSeverity: \"info\",\n });\n // showSnackbar({\n // alertMsg: \"WABA in progress! Please Sync after 10 minutes\",\n // severity: \"info\",\n // alertOpen: true,\n // });\n }\n })\n .catch((err) => {\n console.log(err);\n this.setState({\n alertMsg: err?.response?.data || \"Something went wrong!\",\n alert: true,\n alertSeverity: \"error\",\n });\n // showSnackbar({\n // alertMsg: err?.response?.data || \"Something went wrong!\",\n // severity: \"error\",\n // alertOpen: true,\n // });\n })\n .finally(() => {\n this.setState({\n fbLoading: false,\n });\n });\n } else {\n // trackEvent({\n // event: \"whatsapp_access_token_failed\",\n // properties: {},\n // });\n this.setState({\n alert: false,\n alertMsg: \"Connection Failed!\",\n alertSeverity: \"error\",\n });\n // showSnackbar({\n // alertMsg: \"Connection Failed!\",\n // severity: \"error\",\n // alert: true,\n // });\n this.setState({\n fbLoading: false,\n });\n }\n },\n {\n scope:\n \"business_management,whatsapp_business_management,public_profile,email\",\n extras: {\n feature: \"whatsapp_embedded_signup\",\n setup: {\n business: {\n name: company || \"\",\n email: email || \"\",\n phone: {\n code: 91,\n number: contact || \"\",\n },\n website: \"\",\n address: {\n streetAddress1: \"\",\n city: \"\",\n state: \"\",\n zipPostal: \"\",\n country: \"\",\n },\n timezone: `UTC${\n timeZones.find((i) => i.label.includes(timezone))?.value ||\n \"+05:30\"\n }`,\n },\n phone: {\n displayName: company || \"\",\n category: \"\",\n description: \"\",\n },\n },\n },\n }\n );\n };\n\n fetchPartnerLogos = () => {\n this.setState({ fetchLoading: true });\n axios\n .get(URL + `/api/${this.props.match.params.partnerId}/get-partner-logos`)\n .then((response) => {\n this.setState({ fetchLoading: false, partnerLogos: response.data });\n })\n .catch((error) => {\n console.error(error);\n this.setState({ fetchLoading: false });\n });\n };\n\n handleInput = (e) => {\n this.setState({\n [e.target.name]: e.target.value,\n });\n };\n\n closeSnackbar = () => {\n this.setState({ alert: false, alertMsg: \"\", alertSeverity: \"\" });\n };\n\n componentDidMount = () => {\n this.fetchPartnerLogos();\n };\n\n render() {\n const { classes, match } = this.props;\n const { step, hasRead, countryCode, contact, otpSent, otp } = this.state;\n const assistantId = match.params.projectId;\n return (\n \n {/* Page name & description container */}\n {this.state.fetchLoading ? (\n
\n \n \n ) : (\n <>\n
\n \n \n \n \n AiSensy is a verified WhatsApp Solution Provider and has\n been given the right by Facebook/WhatsApp to grant their\n clients access to the WhatsApp Business API.\n \n \n \n \n \n
\n {/* Layout */}\n
\n \n \n \n \n \n \n {`Step ${step} of 2`}\n \n {step === 1 && (\n <>\n \n \n \n WhatsApp Business API Account Submission\n \n \n Please provide information about the company and phone\n number that you would like to connect to the WhatsApp\n Business API. Please make sure that your company\n follows Facebook's compliance policy.\n \n\n \n \n PERSONAL INFORMATION\n \n \n \n \n \n \n BUSINESS INFORMATION\n \n \n\n \n \n Terms & Conditions\n \n {\n this.setState({\n hasRead: e.target.checked,\n });\n }}\n color=\"primary\"\n />\n }\n label={\n \n I agree to the{\" \"}\n \n AiSensy Terms of Service\n \n \n }\n />\n \n }\n onClick={() => {\n // let hubUrl = window.location.origin + \"/facebook-verify\";\n // console.log(hubUrl);\n // const redirect = window.open(hubUrl, \"_blank\");\n // redirect.focus();\n this.launchWhatsAppSignup();\n // this.setState({\n // step: 2,\n // });\n }}\n style={{\n margin: \"24px 0\",\n fontSize: 18,\n color: \"white\",\n background: hasRead ? \"#1877F2\" : \"lightgrey\",\n fontWeight: 600,\n }}\n variant=\"contained\"\n color=\"primary\"\n >\n Connect to Facebook\n \n >\n )}\n {step === 2 && (\n <>\n \n \n \n Configure WhatsApp Number\n \n \n Please provide information about the company and phone\n number that you would like to connect to the WhatsApp\n Business API. Please make sure that your company\n follows Facebook's compliance policy.\n \n\n \n\n \n i.dialCode)}\n onChange={(e, value, reason) => {\n this.setState({\n countryCode: value,\n });\n }}\n value={countryCode}\n renderInput={(params) => (\n \n )}\n />\n\n {\n this.setState({\n contact: e.target.value,\n });\n }}\n className={classes.phoneTextFieldRoot}\n />\n \n \n \n\n {otpSent && (\n <>\n \n {[1, 2, 3, 4].map((i) => (\n \n \n \n ))}\n \n \n\n \n Didn't received code?{\" \"}\n {\n this.setState({\n alertMsg: \"OTP Resent Successfully!\",\n alert: true,\n alertSeverity: \"success\",\n });\n }}\n >\n Regenerate\n \n \n >\n )}\n\n \n {\n this.setState({\n otpSent: true,\n step: otpSent ? 3 : 2,\n alertMsg: \"OTP Sent Successfully!\",\n alert: otpSent ? false : true,\n alertSeverity: \"success\",\n });\n }}\n style={{\n margin: \"24px 0\",\n // fontSize: 18,\n // fontWeight: 600,\n }}\n variant=\"contained\"\n color=\"primary\"\n >\n {otpSent ? \"Submit OTP\" : \"Verify Phone Number\"}\n \n \n >\n )}\n {step === 3 && (\n <>\n \n \n \n Setup Complete\n \n\n \n\n \n \n \n }\n disabled={!hasRead}\n onClick={() => {\n this.setState({\n alertMsg: \"Message Sent Successfully!\",\n alert: true,\n alertSeverity: \"success\",\n });\n }}\n style={{\n margin: \"24px 0\",\n }}\n variant=\"contained\"\n color=\"primary\"\n >\n Send Test Message\n \n \n >\n )}\n \n \n \n\n
\n \n Powered⚡ by\n \n \n \n \n \n \n AiSensy \n \n \n \n
\n >\n )}\n
\n \n {this.state.alertMsg}\n \n \n
\n );\n }\n}\n\nconst styles = (theme) => ({\n root: {\n width: \"100%\",\n height: \"100%\",\n background: \"#e8ece1\",\n boxSizing: \"border-box\",\n overflowX: \"hidden\",\n position: \"fixed\",\n left: 0,\n },\n pageTitleContainer: {\n position: \"sticky\",\n zIndex: 100,\n top: 0,\n height: 70,\n boxSizing: \"border-box\",\n [theme.breakpoints.down(\"md\")]: {\n // paddingTop: \"40px\",\n },\n [theme.breakpoints.down(\"sm\")]: {\n // paddingTop: \"20px\",\n },\n },\n fullWidth: {\n width: \"100%\",\n background: \"#0a474c\",\n },\n primaryBackground: {\n background: \"rgb(70 20 134 / 15%)\",\n },\n container: {\n marginTop: 16,\n background: \"white\",\n borderRadius: 8,\n padding: \"24px 40px\",\n },\n viewOldButton: {\n \"&:hover\": {\n color: \"grey\",\n cursor: \"pointer\",\n },\n },\n TextField: {\n marginTop: 12,\n },\n codeTextFieldRoot: {\n width: \"80px\",\n padding: \"0 0 0 8px\",\n position: \"absolute\",\n zIndex: 1,\n marginTop: 6,\n },\n phoneTextFieldRoot: {\n width: \"100%\",\n padding: \"8px 20px\",\n marginTop: \"4px\",\n marginBottom: \"20px\",\n \"& input\": { width: \"75%\", marginLeft: \"15%\" },\n },\n otpInputField: {\n background: \"transparent\",\n margin: \"auto\",\n display: \"flex\",\n bottom: \"60px\",\n left: \"33px\",\n maxWidth: \"300px\",\n \"& input\": {\n fontSize: \"24px\",\n letterSpacing: \"50px\",\n },\n },\n otpFieldOverlay: {\n height: \"60px\",\n width: \"10px\",\n background: \"white\",\n position: \"absolute\",\n right: \"calc(50% - 160px)\",\n zIndex: 1,\n },\n logo: {\n height: 40,\n },\n});\n\nexport default withStyles(styles)(Preview);\n","export default __webpack_public_path__ + \"static/media/9978094-new.5cc2de71.png\";","export default __webpack_public_path__ + \"static/media/AiSensy_Logo_Dark_PNG.4a604bb0.png\";","import React, { Component } from \"react\";\nimport axios from \"axios\";\nimport {\n Grid,\n Typography,\n withStyles,\n Box,\n CircularProgress,\n TextField,\n Button,\n Link,\n Snackbar,\n Tooltip,\n IconButton,\n Fade,\n InputAdornment,\n} from \"@material-ui/core\";\nimport { Autocomplete, Alert } from \"@material-ui/lab\";\nimport ErrorOutlineIcon from \"@material-ui/icons/ErrorOutline\";\nimport Visibility from \"@material-ui/icons/Visibility\";\nimport VisibilityOff from \"@material-ui/icons/VisibilityOff\";\nimport { connect } from \"react-redux\";\nimport { Redirect } from \"react-router-dom\";\nimport { loginUser } from \"../Login/loginstate\";\nimport leftImage from \"../../static/9978094-new.png\";\nimport logo from \"../../static/AiSensy_Logo_Dark_PNG.png\";\nimport { APIURL as url } from \"../../config/config\";\nimport dialCodes, { countryMap, isoMap } from \"../../config/dialCodes\";\n\nconst getflag = (langcode) => {\n var first = langcode.charCodeAt(0) + 127397;\n var second = langcode.charCodeAt(1) + 127397;\n var flag = `${first};${second};`;\n const x = document.createElement(\"p\");\n x.innerHTML = flag;\n return x.innerText;\n};\n\nconst styles = (theme) => ({\n root: {\n height: \"100vh\",\n width: \"100%\",\n display: \"flex\",\n justifyContent: \"center\",\n alignItems: \"center\",\n },\n loginContainer: {\n display: \"flex\",\n height: \"100%\",\n width: \"100%\",\n margin: 0,\n },\n leftContainer: {\n backgroundColor: \"#cbe7e6\",\n display: \"flex\",\n flexDirection: \"column\",\n textAlign: \"center\",\n justifyContent: \"center\",\n position: \"relative\",\n width: \"40%\",\n height: \"100vh\",\n overflow: \"hidden\",\n padding: \"0 20px\",\n boxSizing: \"border-box\",\n [theme.breakpoints.down(960)]: {\n display: \"none\",\n },\n },\n logoLink: {\n position: \"absolute\",\n top: 20,\n left: 20,\n },\n rightLogoLink: {\n position: \"absolute\",\n top: \"-1.5%\",\n left: 5,\n [theme.breakpoints.up(960)]: {\n display: \"none\",\n },\n },\n logo: {\n width: 250,\n [theme.breakpoints.down(960)]: {\n width: 150,\n },\n },\n leftImageContainer: {\n width: \"100%\",\n height: \"auto\",\n overflow: \"hidden\",\n justifyContent: \"center\",\n alignItems: \"center\",\n flexGrow: 1,\n position: \"relative\",\n },\n leftImage: {\n width: \"60%\",\n maxWidth: \"100%\",\n height: \"auto\",\n marginTop: \"5%\",\n position: \"relative\",\n },\n centeredWrapper: {\n display: \"flex\",\n justifyContent: \"center\",\n margin: \"24px 0 0\",\n alignItems: \"center\",\n width: \"100%\",\n },\n rightContainer: {\n justifyContent: \"center\",\n alignItems: \"center\",\n display: \"flex\",\n flexDirection: \"column\",\n overflowY: \"auto\",\n height: \"100vh\",\n position: \"relative\",\n padding: \"20px\",\n width: \"60%\",\n [theme.breakpoints.down(\"md\")]: {\n width: \"100%\",\n height: \"auto\",\n padding: \"10px\",\n },\n },\n formContainer: {\n position: \"relative\",\n width: \"100%\",\n maxWidth: \"400px\",\n alignItems: \"center\",\n padding: \"0 50px 50px 50px\",\n [theme.breakpoints.down(\"sm\")]: {\n padding: \"20px\",\n },\n },\n nextButtonContainer: {\n display: \"flex\",\n justifyContent: \"center\",\n marginTop: \"7%\",\n width: \"100%\",\n },\n nextButton: {\n width: \"100%\",\n },\n googleSignInButton: {\n marginTop: theme.spacing(2),\n marginBottom: theme.spacing(2),\n backgroundColor: \"#0563fa\",\n color: \"white\",\n textTransform: \"none\",\n },\n signInButton: {\n marginTop: theme.spacing(2),\n textTransform: \"none\",\n \"&.disabled\": {\n backgroundColor: \"#ddd\",\n color: \"#888\",\n },\n \"&.enabled\": {\n color: \"#fff\",\n backgroundColor: \"#0a474c\",\n },\n },\n orDivider: {\n display: \"flex\",\n alignItems: \"center\",\n textAlign: \"center\",\n margin: theme.spacing(2, 0),\n \"& span\": {\n flex: 1,\n borderBottom: \"1px solid #ccc\",\n },\n \"& span:first-child\": {\n marginRight: theme.spacing(1),\n },\n \"& span:last-child\": {\n marginLeft: theme.spacing(1),\n },\n },\n signUpLink: {\n position: \"absolute\",\n top: 20,\n right: 20,\n [theme.breakpoints.down(960)]: {\n position: \"absolute\",\n top: \"9%\",\n right: 20,\n },\n [theme.breakpoints.down(538)]: {\n position: \"absolute\",\n top: \"7%\",\n right: 20,\n marginBottom: \"2%\",\n },\n },\n welcomeText: {\n marginBottom: theme.spacing(1),\n [theme.breakpoints.down(538)]: {\n marginBottom: theme.spacing(1),\n marginTop: \"18%\",\n },\n },\n forgotPassword: {\n marginTop: theme.spacing(2),\n textAlign: \"center\",\n },\n resendOtpButton: {\n marginTop: theme.spacing(2),\n display: \"flex\",\n justifyContent: \"center\",\n },\n errorText: {\n color: \"red\",\n textAlign: \"center\",\n },\n phoneTextFieldRoot: {\n width: \"100%\",\n padding: \"8px 20px\",\n marginTop: \"4px\",\n \"& input\": { width: \"calc(100% - 80px)\", marginLeft: \"80px\" },\n background: \"white\",\n border: \"2px solid grey\",\n },\n otpInput: {\n width: \"60px\",\n height: \"60px\",\n background: \"white\",\n padding: \"4px 6px\",\n fontSize: \"24px\",\n textAlign: \"center\",\n margin: \"0 5px\",\n border: \"1px solid #ccc\",\n borderRadius: \"4px\",\n },\n otpContainer: {\n display: \"flex\",\n justifyContent: \"center\",\n marginBottom: theme.spacing(2),\n },\n textFieldRootAutocomplete: {\n width: \"100%\",\n padding: \"2px 2px\",\n marginTop: \"4px\",\n background: \"white\",\n border: \"2px solid grey\",\n \"& label\": {\n background: \"white\",\n color: \"#b2b3b3 !important\",\n padding: \"0 6px\",\n marginTop: -4,\n },\n },\n codeTextFieldRoot: {\n width: \"80px\",\n padding: \"0 0 0 14px\",\n position: \"absolute\",\n zIndex: 1,\n margin: \"10px 0 0 2px\",\n background: \"white\",\n \"& input\": {\n padding: \"10px 0 !important\",\n },\n \"& button\": {\n width: \"12px\",\n position: \"relative\",\n left: \"7px\",\n background: \"white\",\n borderRadius: 0,\n \"&:hover\": {\n background: \"white\",\n },\n },\n \"& div\": {\n paddingRight: \"0px !important\",\n },\n },\n textFieldPopper: {\n [theme.breakpoints.down(\"md\")]: {\n left: \"12px !important\",\n minWidth: \"calc(100% - 35px)\",\n },\n [theme.breakpoints.up(\"md\")]: {\n left: \"calc(12.5vw - 16px) !important\",\n minWidth: \"calc(25vw + 54px)\",\n },\n },\n textFieldRoot: {\n backgroundColor: \"#f1f1f1\",\n outline: \"none\",\n border: \"none\",\n borderRadius: \"8px\",\n width: \"100%\",\n marginBottom: theme.spacing(3),\n padding: \"14px 15px\",\n boxShadow: \"none\",\n boxSizing: \"border-box\",\n fontFamily: \"Roboto, sans-serif\",\n fontSize: \"15px\",\n fontWeight: 500,\n },\n otpButtonContainer: {\n display: \"flex\",\n justifyContent: \"space-between\",\n marginTop: theme.spacing(3),\n },\n passwordFieldContainer: {\n alignItems: \"center\",\n position: \"relative\",\n width: \"100%\",\n },\n passwordTooltip: {\n position: \"absolute\",\n right: 10,\n },\n passwordField: {\n flex: 1,\n width: \"100%\",\n },\n passwordIcon: {\n marginLeft: theme.spacing(1),\n position: \"absolute\",\n right: \"-30px\",\n top: \"20px\",\n },\n passwordIconEmail: {\n marginLeft: theme.spacing(1),\n position: \"absolute\",\n right: \"2%\",\n top: \"48%\",\n },\n passwordIconName: {\n marginLeft: theme.spacing(1),\n position: \"absolute\",\n right: \"2%\",\n top: \"11%\",\n },\n passwordIconCompanyName: {\n marginLeft: theme.spacing(1),\n position: \"absolute\",\n right: \"2%\",\n top: \"23%\",\n },\n passwordIconContact: {\n marginLeft: theme.spacing(1),\n position: \"absolute\",\n right: \"2%\",\n top: \"49%\",\n },\n errorSnackbar: {\n backgroundColor: theme.palette.error.main,\n },\n});\n\nclass AddAffiliatePage extends Component {\n constructor(props) {\n super(props);\n this.state = {\n step: 1,\n alertType: \"\",\n alertMsg: \"\",\n formValues: {\n name: \"\",\n country: \"\",\n state: \"\",\n city: \"\",\n companyName: \"\",\n contact: \"\",\n email: \"\",\n userName: \"\",\n loginPassword: \"\",\n type: \"Affiliate\",\n planType: \"tier0\",\n timezone: \"\",\n currency: \"\",\n partnershipFee: \"0\",\n otp: [\"\", \"\", \"\", \"\"],\n facebookAds: \"\",\n companySize: \"\",\n industry: [],\n websiteURL: \"\",\n confirmPassword: \"\",\n },\n errors: {\n email: \"\",\n userName: \"\",\n loginPassword: \"\",\n otp: \"\",\n confirmPassword: \"\",\n name: \"\",\n contact: \"\",\n companyName: \"\",\n },\n passwordValidation: {\n length: false,\n lowercase: false,\n number: false,\n specialChar: false,\n },\n showPasswordTooltip: false,\n generatedOtp: \"\",\n isOtpVerified: false,\n otpResendTime: 30,\n canResendOtp: false,\n countryCode: \"+91\",\n contact: \"\",\n snackbarOpen: false,\n snackbarMessage: \"\",\n isLoading: false,\n snackbarError: false,\n showPassword: false,\n showConfirmPassword: false,\n };\n }\n\n handleChange = (event, value, name) => {\n const { formValues, errors, passwordValidation } = this.state;\n formValues[name] = value;\n errors[name] = \"\";\n\n if (name === \"loginPassword\" || name === \"confirmPassword\") {\n const length = formValues.loginPassword.length >= 8;\n const lowercase = /[a-z]/.test(formValues.loginPassword);\n const number = /\\d/.test(formValues.loginPassword);\n const specialChar = /[!@#$%^&*(),.?\":{}|<>]/.test(\n formValues.loginPassword\n );\n\n passwordValidation.length = length;\n passwordValidation.lowercase = lowercase;\n passwordValidation.number = number;\n passwordValidation.specialChar = specialChar;\n\n errors.loginPassword =\n !length || !lowercase || !number || !specialChar\n ? \"Password must be 8 characters long and include at least 1 lowercase letter, 1 number, and 1 special character\"\n : \"\";\n\n // Validate confirm password\n errors.confirmPassword =\n formValues.confirmPassword !== formValues.loginPassword\n ? \"Passwords do not match\"\n : \"\";\n }\n\n if (name === \"email\" && !this.validateEmail(value)) {\n errors.email = \"Enter a valid email address\";\n }\n\n if (name === \"name\" && !/^[A-Za-z\\s]+$/.test(value)) {\n errors.name = \"Name should contain only alphabets and spaces\";\n }\n\n if (name === \"contact\" && (!/^\\d+$/.test(value) || value.length <= 7)) {\n errors.contact = \"Contact number should be 7 to 10 digits long\";\n }\n if (name === \"companyName\" && value.length === 0) {\n errors.companyName = \"Company Name can't be empty\";\n }\n this.setState({ formValues, errors, passwordValidation });\n };\n\n handleInputChange = (event) => {\n const { name, value } = event.target;\n const { formValues, errors } = this.state;\n\n if (name === \"contact\") {\n if (/^\\d*$/.test(value)) {\n // Only allow digits\n formValues[name] = value;\n if (value.length <= 7) {\n errors.contact = \"Contact number should be 7 to 10 digits long\";\n } else {\n errors.contact = \"\";\n }\n }\n } else {\n formValues[name] = value;\n }\n\n this.setState({ formValues, errors });\n };\n\n handleOtpChange = (index, event) => {\n const otp = [...this.state.formValues.otp];\n otp[index] = event.target.value;\n if (event.target.value.length === 1 && index < 3) {\n this[`otpInput${index + 1}`].focus();\n }\n this.setState({ formValues: { ...this.state.formValues, otp } });\n };\n\n handleNext = () => {\n const { step } = this.state;\n if (this.isStepComplete()) {\n if (step === 1) {\n this.generateOtp();\n this.setState({ isLoading: true });\n }\n }\n };\n\n handleBack = () => {\n this.setState((prevState) => ({ step: prevState.step - 1 }));\n };\n\n handleSubmit = () => {\n const { formValues, countryCode } = this.state;\n const { userName, loginPassword, email, contact } = formValues;\n const updatedFormValues = { ...formValues, contact: countryCode + contact };\n\n axios\n .post(url + \"/affiliates/create-affiliate\", updatedFormValues)\n .then((response) => {\n if (response.status === 200) {\n this.props.loginUser(userName, loginPassword, email);\n }\n })\n .catch((error) => {\n if (error.response) {\n if (error.response.data.message === \"Email already in use\") {\n this.setState({\n snackbarOpen: true,\n alertType: \"error\",\n snackbarMessage: \"Email already in use\",\n snackbarError: true,\n });\n } else if (error.response.data.message === \"User blocked\") {\n this.setState({\n snackbarOpen: true,\n alertType: \"error\",\n snackbarMessage:\n \"You have tried too many times, please contact customer support\",\n snackbarError: true,\n });\n }\n } else {\n console.error(\"There was an error!\", error);\n }\n });\n };\n\n generateOtp = () => {\n const { email, name, contact } = this.state.formValues;\n axios\n .post(url + \"/affiliates/send-otp\", { email, fullName: name, contact })\n .then((response) => {\n if (response.data.success) {\n this.setState(\n {\n isLoading: false,\n generatedOtp: response.data.otp,\n isOtpVerified: false,\n otpResendTime: 30,\n canResendOtp: false,\n },\n () => {\n this.startOtpTimer();\n this.setState((prevState) => ({ step: prevState.step + 1 })); // Move to OTP step\n }\n );\n } else if (response.data.blocked) {\n this.setState({\n snackbarOpen: true,\n alertType: \"error\",\n snackbarMessage:\n \"You have tried too many times, please contact customer support\",\n isLoading: false,\n snackbarError: true,\n });\n } else {\n this.setState({\n snackbarOpen: true,\n alertType: \"error\",\n snackbarMessage: \"Error Generating OTP\",\n isLoading: false,\n snackbarError: true,\n });\n console.error(\"Error generating OTP\");\n }\n })\n .catch((error) => {\n if (error.response) {\n if (error.response?.data?.message === \"Email already in use!\") {\n this.setState({\n snackbarOpen: true,\n alertType: \"error\",\n snackbarMessage: \"Email already in use\",\n isLoading: false,\n snackbarError: true,\n });\n } else if (error.response?.data?.message === \"blocked\") {\n this.setState({\n snackbarOpen: true,\n alertType: \"error\",\n snackbarMessage:\n \"You have tried too many times, please contact customer support\",\n isLoading: false,\n snackbarError: true,\n });\n }\n } else {\n this.setState({\n snackbarOpen: true,\n alertType: \"error\",\n snackbarMessage: \"There was an error generating otp\",\n isLoading: false,\n snackbarError: true,\n });\n console.error(\"There was an error generating the OTP!\", error);\n }\n });\n };\n\n generateResendOtp = () => {\n const { email, name, contact } = this.state.formValues;\n axios\n .post(url + \"/affiliates/send-otp\", { email, fullName: name, contact })\n .then((response) => {\n if (response.data.success) {\n this.setState(\n {\n isLoading: false,\n generatedOtp: response.data.otp,\n isOtpVerified: false,\n otpResendTime: 30,\n canResendOtp: false,\n },\n () => {\n this.startOtpTimer();\n }\n );\n } else if (response.data.blocked) {\n this.setState({\n snackbarOpen: true,\n alertType: \"error\",\n snackbarMessage:\n \"You have tried too many times, please contact customer support\",\n isLoading: false,\n snackbarError: true,\n });\n } else {\n console.error(\"Error generating OTP\");\n }\n })\n .catch((error) => {\n if (error.response) {\n if (error.response.data === \"Email already in use!\") {\n this.setState({\n snackbarOpen: true,\n alertType: \"error\",\n snackbarMessage: \"Email already in use\",\n isLoading: false,\n snackbarError: true,\n });\n } else if (error.response.data === \"blocked\") {\n this.setState({\n snackbarOpen: true,\n alertType: \"error\",\n snackbarMessage:\n \"You have tried too many times, please contact customer support\",\n isLoading: false,\n snackbarError: true,\n });\n }\n } else {\n console.error(\"There was an error generating the OTP!\", error);\n }\n });\n };\n\n startOtpTimer = () => {\n this.otpTimer = setInterval(() => {\n this.setState((prevState) => {\n if (prevState.otpResendTime <= 1) {\n clearInterval(this.otpTimer);\n return { otpResendTime: 0, canResendOtp: true };\n }\n return { otpResendTime: prevState.otpResendTime - 1 };\n });\n }, 1000);\n };\n\n verifyOtp = () => {\n const { formValues } = this.state;\n const { email, otp } = formValues;\n axios\n .post(url + \"/affiliates/verify-otp\", {\n email,\n otp: formValues.otp.join(\"\"),\n })\n .then((response) => {\n if (response.data) {\n this.setState(\n { isOtpVerified: true, errors: { ...this.state.errors, otp: \"\" } },\n this.handleSubmit\n );\n } else {\n this.setState({\n isOtpVerified: false,\n errors: {\n ...this.state.errors,\n otp: \"Invalid OTP. Please try again.\",\n },\n });\n }\n })\n .catch((error) => {\n console.error(\"There was an error verifying the OTP!\", error);\n this.setState({\n isOtpVerified: false,\n errors: {\n ...this.state.errors,\n otp: \"An error occurred. Please try again.\",\n },\n });\n });\n };\n\n validateEmail = (email) => {\n const emailRegex = /^[^\\s@]+@[^\\s@]+\\.[^\\s@]+$/;\n return emailRegex.test(email);\n };\n\n isStepComplete = () => {\n const { step, formValues, errors } = this.state;\n const stepFields = {\n 1: [\n \"name\",\n \"companyName\",\n \"contact\",\n \"email\",\n \"userName\",\n \"loginPassword\",\n ],\n 2: [\"otp\"],\n };\n\n const areFieldsFilled = stepFields[step]\n ? stepFields[step].every((field) => formValues[field])\n : true;\n const areFieldsValid = Object.values(errors).every((error) => !error);\n\n if (step === 1) {\n return areFieldsFilled && areFieldsValid;\n }\n\n return areFieldsFilled && areFieldsValid;\n };\n\n handleClickShowPassword = () => {\n this.setState((prevState) => ({\n showPassword: !prevState.showPassword,\n }));\n };\n\n handleClickShowConfirmPassword = () => {\n this.setState((prevState) => ({\n showConfirmPassword: !prevState.showConfirmPassword,\n }));\n };\n\n renderStep = () => {\n const {\n step,\n formValues,\n errors,\n passwordValidation,\n showPasswordTooltip,\n otpResendTime,\n canResendOtp,\n countryCode,\n showPassword,\n showConfirmPassword,\n } = this.state;\n const { classes } = this.props;\n\n const passwordTooltipContent = (\n \n \n - Must be 8 characters long\n \n \n - At least 1 lowercase letter\n \n \n - At least 1 number\n \n \n - At least 1 special character\n \n
\n );\n\n switch (step) {\n case 1:\n return (\n \n \n \n \n this.handleChange(event, value, \"name\")\n }\n renderInput={(params) => (\n \n \n \n \n \n ),\n }}\n />\n )}\n />\n \n \n \n this.handleChange(event, value, \"companyName\")\n }\n renderInput={(params) => (\n \n \n \n \n \n ),\n }}\n />\n )}\n />\n \n \n
\n \n
i.dialCode)}\n classes={{ popper: classes.textFieldPopper }}\n getOptionLabel={(option) => {\n return `${getflag(isoMap[option])} ${option} ${\n countryMap[option.substring(1)]\n }`;\n }}\n onChange={(e, value, reason) => {\n this.setState({\n countryCode: value,\n });\n }}\n value={countryCode}\n renderInput={(params) => (\n \n )}\n />\n this.handleInputChange(e)}\n value={formValues.contact}\n className={classes.phoneTextFieldRoot}\n InputProps={{\n endAdornment: errors.contact && (\n \n \n \n \n \n ),\n }}\n />\n \n \n \n {\n this.handleChange(event, value, \"email\");\n this.handleChange(event, value, \"userName\"); // Set both email and username\n }}\n renderInput={(params) => (\n \n \n \n \n \n ),\n }}\n />\n )}\n />\n \n \n \n this.handleChange(event, value, \"loginPassword\")\n }\n renderInput={(params) => (\n \n {errors.loginPassword && (\n \n \n \n )}\n \n this.setState({ showPasswordTooltip: true })\n }\n onMouseLeave={() =>\n this.setState({ showPasswordTooltip: false })\n }\n onClick={this.handleClickShowPassword}\n >\n {showPassword ? (\n \n ) : (\n \n )}\n \n \n ),\n }}\n />\n )}\n />\n \n \n \n this.handleChange(event, value, \"confirmPassword\")\n }\n renderInput={(params) => (\n \n {errors.confirmPassword && (\n \n \n \n )}\n \n {showConfirmPassword ? (\n \n ) : (\n \n )}\n \n \n ),\n }}\n />\n )}\n />\n \n \n \n );\n case 2:\n return (\n \n \n \n {[0, 1, 2, 3].map((index) => (\n this.handleOtpChange(index, event)}\n className={classes.otpInput}\n inputRef={(input) => (this[`otpInput${index}`] = input)}\n />\n ))}\n \n \n \n OTP sent to {formValues.email} Resend OTP after{\" \"}\n {`00:${otpResendTime < 10 ? \"0\" : \"\"}${otpResendTime}`}\n \n {otpResendTime === 0 ? (\n \n Didn't receive the code?{\" \"}\n \n Resend OTP\n \n \n ) : null}\n \n \n \n );\n default:\n return null;\n }\n };\n\n handleSnackbarClose = () => {\n this.setState({ snackbarOpen: false, snackbarError: false });\n };\n\n render() {\n const { isAuthenticated, classes } = this.props;\n const { step, formValues, snackbarOpen, snackbarMessage, snackbarError } =\n this.state;\n\n if (isAuthenticated) {\n return ;\n } else {\n return (\n \n
\n \n \n \n \n \n \n AiSensy Affiliate Dashboard\n \n \n Start with 20% commission and Grow More !!!\n \n \n \n \n\n \n Trusted by 50000 + Brands\n \n \n \n\n \n \n \n \n \n \n Already a member ?{\" \"}\n \n Log in\n \n \n
\n \n \n Create Your AiSensy Affiliate Account\n \n \n Fill in the details below to complete your signup.\n \n
\n \n
\n
\n {this.renderStep()}
\n \n
\n {step > 1 && (\n \n Back\n \n )}\n {step === 2 ? (\n \n Verify OTP\n \n ) : (\n \n {this.state.isLoading ? (\n <>\n Sending OTP \n \n >\n ) : (\n \"Next \"\n )}\n \n )}\n
\n
\n
\n \n \n
\n \n {this.state.snackbarMessage}\n \n \n
\n );\n }\n }\n}\n\nconst mapDispatchToProps = {\n loginUser,\n};\n\nexport default connect(\n (state) => ({\n isAuthenticated: state.login.isAuthenticated,\n isLoading: state.login.isLoading,\n error: state.login.error,\n errmsg: state.login.errmsg,\n }),\n mapDispatchToProps\n)(withStyles(styles)(AddAffiliatePage));\n","import React, { Component } from \"react\";\nimport {\n withStyles,\n Grid,\n Box,\n Typography,\n Button,\n TextField,\n Link,\n InputAdornment,\n Tooltip,\n IconButton,\n Dialog,\n DialogActions,\n DialogContent,\n DialogContentText,\n DialogTitle,\n CircularProgress,\n} from \"@material-ui/core\";\nimport Visibility from \"@material-ui/icons/Visibility\";\nimport VisibilityOff from \"@material-ui/icons/VisibilityOff\";\nimport ErrorOutlineIcon from \"@material-ui/icons/ErrorOutline\";\nimport axios from \"axios\";\nimport { APIURL as url } from \"../../config/config\";\n\nconst styles = (theme) => ({\n formContainer: {\n width: \"100%\",\n maxWidth: 400,\n padding: \"0 50px 50px 0\",\n [theme.breakpoints.down(\"sm\")]: {\n padding: \"20px\",\n },\n },\n backToLoginContainer: {\n display: \"flex\",\n justifyContent: \"center\",\n alignItems: \"center\",\n width: \"100%\",\n marginTop: theme.spacing(2),\n },\n nextButtonContainer: {\n display: \"flex\",\n justifyContent: \"center\",\n marginTop: theme.spacing(3),\n width: \"100%\",\n },\n nextButton: {\n width: \"100%\",\n },\n resetPasswordButton: {\n width: \"100%\",\n },\n textFieldRootAutocomplete: {\n width: \"100%\",\n padding: \"2px 2px\",\n marginTop: \"4px\",\n background: \"white\",\n border: \"2px solid grey\",\n \"& label\": {\n background: \"white\",\n color: \"#b2b3b3 !important\",\n padding: \"0 6px\",\n marginTop: -4,\n },\n },\n forgotPassword: {\n marginTop: theme.spacing(2),\n textAlign: \"center\",\n },\n otpInput: {\n width: \"60px\",\n height: \"60px\",\n background: \"white\",\n padding: \"4px 6px\",\n fontSize: \"24px\",\n textAlign: \"center\",\n margin: \"0 5px\",\n border: \"1px solid #ccc\",\n borderRadius: \"4px\",\n },\n otpContainer: {\n display: \"flex\",\n justifyContent: \"center\",\n marginBottom: theme.spacing(2),\n },\n passwordFieldContainer: {\n alignItems: \"center\",\n position: \"relative\",\n width: \"100%\",\n },\n});\n\nclass ResetPassword extends Component {\n constructor(props) {\n super(props);\n this.state = {\n step: 1,\n formValues: {\n email: \"\",\n loginPassword: \"\",\n confirmPassword: \"\",\n otp: [\"\", \"\", \"\", \"\"],\n },\n errors: {\n email: \"\",\n loginPassword: \"\",\n confirmPassword: \"\",\n otp: \"\",\n },\n showPassword: false,\n showConfirmPassword: false,\n successDialogOpen: false,\n isSubmitting: false,\n };\n }\n\n handleInputChange = (event) => {\n const { name, value } = event.target;\n this.setState((prevState) => ({\n formValues: {\n ...prevState.formValues,\n [name]: value,\n },\n errors: {\n ...prevState.errors,\n [name]: \"\",\n },\n }));\n };\n\n handleOtpChange = (index, event) => {\n const otp = [...this.state.formValues.otp];\n otp[index] = event.target.value;\n if (event.target.value.length === 1 && index < 3) {\n this[`otpInput${index + 1}`].focus();\n }\n this.setState((prevState) => ({\n formValues: {\n ...prevState.formValues,\n otp,\n },\n }));\n };\n\n handleNext = () => {\n const { step } = this.state;\n if (step === 1) {\n this.searchAccount();\n } else {\n this.setState({ step: step + 1 });\n }\n };\n\n handleBack = () => {\n const { step } = this.state;\n this.setState({ step: step - 1 });\n };\n\n handleVerifyOtp = () => {\n const { formValues } = this.state;\n axios\n .post(url + \"/affiliates/verify-otp\", {\n email: formValues.email,\n otp: formValues.otp.join(\"\"),\n })\n .then((response) => {\n if (response.data) {\n this.setState({ step: 4 });\n } else {\n this.setState({\n showSnackbar: true,\n snackbarMessage: \"Invalid OTP. Please try again.\",\n });\n }\n })\n .catch((error) => {\n console.error(\"There was an error verifying the OTP!\", error);\n this.setState({\n showSnackbar: true,\n snackbarMessage: \"Invalid OTP. Please try again.\",\n });\n });\n };\n\n handleSubmit = () => {\n const { formValues } = this.state;\n this.setState({ isSubmitting: true });\n axios\n .post(url + \"/affiliates/reset-password\", formValues)\n .then((response) => {\n if (response.status === 200) {\n this.setState({ successDialogOpen: true });\n }\n })\n .finally(() => {\n this.setState({ isSubmitting: false });\n });\n };\n\n searchAccount = () => {\n const { formValues } = this.state;\n axios\n .post(url + \"/affiliates/search-account\", {\n forgottenDetail: formValues.email,\n })\n .then((response) => {\n if (response.data.email) {\n this.setState({ step: 2 });\n }\n })\n .catch((error) => {\n this.setState((prevState) => ({\n errors: {\n ...prevState.errors,\n email: \"Sorry! No account found. Please contact your project owner.\",\n },\n }));\n });\n };\n\n generateOtp = () => {\n const { formValues } = this.state;\n this.handleNext();\n axios\n .post(url + \"/affiliates/reset-password-otp\", { email: formValues.email })\n .then((response) => {\n if (response.data.success) {\n this.setState({ step: 3 });\n this.startOtpTimer();\n }\n })\n .catch((error) => {\n console.error(\"There was an error generating the OTP!\", error);\n });\n };\n\n generateResendOtp = () => {\n const { formValues } = this.state;\n axios\n .post(url + \"/affiliates/reset-password-otp\", { email: formValues.email })\n .then((response) => {\n if (response.data.success) {\n this.setState({ step: 3 });\n this.startOtpTimer();\n }\n })\n .catch((error) => {\n console.error(\"There was an error generating the OTP!\", error);\n });\n };\n\n handleCloseSuccessDialog = () => {\n window.location.href = \"/login\";\n };\n\n renderStep = () => {\n const { step, formValues, errors, isSubmitting } = this.state;\n const { classes } = this.props;\n\n switch (step) {\n case 1:\n return (\n \n \n \n \n \n \n \n \n ),\n }}\n />\n \n \n \n Enter your email to help us find your account.\n \n\n \n \n Search Account\n \n \n \n \n Back to Login\n \n \n \n );\n case 2:\n return (\n \n \n \n Account Verification\n \n \n Yay! An account has been found. Please verify email for reset\n password. OTP will be sent over email.\n \n \n \n \n Send OTP\n \n \n \n \n Back to Login\n \n \n \n );\n case 3:\n return (\n \n \n \n OTP sent over email.\n \n \n \n {[0, 1, 2, 3].map((index) => (\n this.handleOtpChange(index, event)}\n className={classes.otpInput}\n inputRef={(input) => (this[`otpInput${index}`] = input)}\n />\n ))}\n \n \n \n Didn't receive code?{\" \"}\n Resend OTP\n \n \n \n otp !== \"\")}\n className={classes.nextButton}\n >\n Verify OTP\n \n \n \n \n Back to Login\n \n \n \n );\n case 4:\n return (\n \n \n \n \n this.setState({\n showPassword: !this.state.showPassword,\n })\n }\n >\n {this.state.showPassword ? (\n \n ) : (\n \n )}\n \n \n ),\n }}\n />\n \n \n \n \n this.setState({\n showConfirmPassword: !this.state.showConfirmPassword,\n })\n }\n >\n {this.state.showConfirmPassword ? (\n \n ) : (\n \n )}\n \n \n ),\n }}\n />\n \n \n \n {isSubmitting ? : \"Reset Password\"}\n \n \n \n \n Back to Login\n \n \n \n );\n default:\n return null;\n }\n };\n\n render() {\n const { successDialogOpen } = this.state;\n\n return (\n \n {this.renderStep()}\n \n Password Reset Successful \n \n \n Your password has been successfully reset. Please log in with your\n new password.\n \n \n \n \n OK\n \n \n \n \n );\n }\n}\n\nexport default withStyles(styles)(ResetPassword);\n","import React from \"react\";\nimport { Box, Typography, withStyles } from \"@material-ui/core\";\n\nconst styles = (theme) => ({\n welcomeTextRightContainer: {\n marginBottom: \"5px\",\n width: \"100%\",\n textAlign: \"left\",\n },\n});\n\nconst WelcomeText = ({ classes }) => (\n \n \n \n WELCOME BACK\n \n \n Log in to AiSensy Affiliate Account\n \n \n \n);\n\nexport default withStyles(styles)(WelcomeText);\n","import React, { Component } from \"react\";\nimport { connect } from \"react-redux\";\nimport { loginUser, loginWithToken } from \"./loginstate\";\nimport {\n withStyles,\n Grid,\n Box,\n Typography,\n Button,\n Link,\n TextField,\n} from \"@material-ui/core\";\nimport { Redirect } from \"react-router-dom\";\nimport { Autocomplete } from \"@material-ui/lab\";\nimport Toast from \"../../commons/Toast/Toast\";\nimport leftImage from \"../../static/9978094-new.png\";\nimport logo from \"../../static/AiSensy_Logo_Dark_PNG.png\";\nimport WelcomeTextComponent from \"./welcomeTextComponent\";\nimport ResetPassword from \"../forgotPassword/forgotPassword\";\n\nconst styles = (theme) => ({\n root: {\n display: \"flex\",\n justifyContent: \"center\",\n alignItems: \"center\",\n height: \"100vh\",\n [theme.breakpoints.down(767)]: {\n padding: \"3%\",\n },\n },\n loginContainer: {\n display: \"flex\",\n flexWrap: \"wrap\",\n width: \"100%\",\n margin: 0,\n },\n leftContainer: {\n backgroundColor: \"#cbe7e6\",\n display: \"flex\",\n flexDirection: \"column\",\n textAlign: \"center\",\n justifyContent: \"center\",\n alignItems: \"center\",\n width: \"40%\",\n height: \"100vh\",\n position: \"relative\",\n padding: \"0 20px\",\n boxSizing: \"border-box\",\n [theme.breakpoints.down(767)]: {\n display: \"none\",\n },\n },\n logoLink: {\n position: \"absolute\",\n top: 20,\n left: 20,\n },\n rightLogoLink: {\n width: \"100%\",\n position: \"absolute\",\n top: \"-1.5%\",\n left: \"-5%\",\n [theme.breakpoints.up(767)]: {\n display: \"none\",\n },\n },\n logo: {\n width: 250,\n [theme.breakpoints.down(767)]: {\n width: 150,\n },\n },\n leftImageContainer: {\n width: \"100%\",\n height: \"auto\",\n overflow: \"hidden\",\n display: \"flex\",\n justifyContent: \"center\",\n alignItems: \"center\",\n flexGrow: 1,\n position: \"relative\",\n },\n leftImage: {\n width: \"60%\",\n maxWidth: \"100%\",\n height: \"auto\",\n marginTop: \"5%\",\n position: \"relative\",\n },\n rightContainer: {\n justifyContent: \"center\",\n alignItems: \"center\",\n display: \"flex\",\n flexDirection: \"column\",\n overflowY: \"auto\",\n height: \"100vh\",\n position: \"relative\",\n padding: \"20px\",\n width: \"60%\",\n [theme.breakpoints.down(\"md\")]: {\n width: \"100%\",\n padding: \"10px\",\n },\n [theme.breakpoints.down(\"sm\")]: {\n width: \"100%\",\n padding: \"0px\",\n },\n },\n formContainer: {\n width: \"100%\",\n maxWidth: 400,\n padding: \"0 0 50px\",\n marginTop: \"5%\",\n height: \"50vh\",\n [theme.breakpoints.down(\"sm\")]: {\n padding: \"0px\",\n },\n },\n signUpLink: {\n position: \"absolute\",\n top: 20,\n right: 20,\n [theme.breakpoints.down(767)]: {\n position: \"absolute\",\n top: \"9%\",\n right: 20,\n },\n },\n welcomeText: {\n marginTop: theme.spacing(4),\n marginBottom: theme.spacing(1),\n },\n welcomeTextContainer: {\n width: \"100%\",\n textAlign: \"center\",\n marginBottom: theme.spacing(4),\n [theme.breakpoints.down(767)]: {\n marginBottom: theme.spacing(2),\n },\n },\n nextButtonContainer: {\n display: \"flex\",\n justifyContent: \"center\",\n marginTop: theme.spacing(3),\n width: \"100%\",\n },\n nextButton: {\n width: \"100%\",\n },\n forgotPassword: {\n marginTop: theme.spacing(2),\n textAlign: \"center\",\n },\n textFieldRootAutocomplete: {\n width: \"100%\",\n padding: \"2px 2px\",\n marginTop: \"4px\",\n background: \"white\",\n marginBottom: \"10px\",\n border: \"2px solid grey\",\n \"& label\": {\n background: \"white\",\n color: \"#b2b3b3 !important\",\n padding: \"0 6px\",\n marginTop: -4,\n },\n },\n});\n\n\nclass newLogin extends Component {\n constructor(props) {\n super(props);\n this.state = {\n username: \"\",\n password: \"\",\n status: \"\",\n statusMessage: \"\",\n showSnackbar: false,\n alertMsg: \"\",\n alertSeverity: \"\",\n forgotPassword: false,\n };\n }\n\n componentDidMount() {\n this.tokenLogin();\n }\n\n componentDidUpdate(prevProps) {\n if (prevProps !== this.props && this.props.error) {\n this.setAlert(\"Invalid login credentials!\", \"error\");\n }\n }\n\n handleChange = (event, value, name) => {\n this.setState({ [name]: value });\n };\n\n closeSnackbar = () => {\n this.setState({ showSnackbar: false });\n };\n\n setAlert = (msg, severity) => {\n this.setState({\n alertMsg: msg,\n alertSeverity: severity,\n showSnackbar: true,\n });\n };\n\n tokenLogin() {\n const search = this.props.location.search;\n const params = new URLSearchParams(search);\n const token = params.get(\"token\");\n if (!token) return;\n\n this.props.loginWithToken(token);\n }\n\n handleSubmit = (event) => {\n event.preventDefault();\n const { username, password } = this.state;\n this.props.loginUser(username, password);\n };\n\n handleForgotPassword = () => {\n this.setState({ forgotPassword: true });\n };\n\n handleBackToLogin = () => {\n this.setState({ forgotPassword: false });\n };\n\n render() {\n const { classes, isAuthenticated } = this.props;\n const {\n username,\n password,\n showSnackbar,\n alertMsg,\n alertSeverity,\n forgotPassword,\n } = this.state;\n const isFormValid = username && password;\n\n if (isAuthenticated) {\n return ;\n }\n\n return (\n \n
\n \n \n \n \n \n \n AiSensy Affiliate Dashboard\n \n \n Start with 20% commission and Grow More !!!\n \n \n \n \n\n \n Trusted by 50000 + Brands\n \n \n \n \n \n \n \n \n \n \n \n Not a member yet?{\" \"}\n \n Sign up\n \n \n \n\n \n \n \n \n {forgotPassword ? (\n \n ) : (\n \n \n this.handleChange(event, value, \"username\")\n }\n renderInput={(params) => (\n \n )}\n />\n \n this.handleChange(event, value, \"password\")\n }\n renderInput={(params) => (\n \n )}\n />\n \n Continue\n \n \n \n Forgot Password?\n \n \n \n )}\n \n \n \n
\n
\n );\n }\n}\n\nconst mapStateToProps = (state) => ({\n isAuthenticated: state.login.isAuthenticated,\n error: state.login.error,\n});\n\nexport default connect(mapStateToProps, { loginUser, loginWithToken })(\n withStyles(styles)(newLogin)\n);\n","import React, { Component } from \"react\";\nimport axios from \"axios\";\nimport {\n Grid,\n Typography,\n withStyles,\n Box,\n CircularProgress,\n TextField,\n Button,\n Link,\n Snackbar,\n Tooltip,\n IconButton,\n Fade,\n InputAdornment,\n} from \"@material-ui/core\";\nimport { Autocomplete, Alert } from \"@material-ui/lab\";\nimport ErrorOutlineIcon from \"@material-ui/icons/ErrorOutline\";\nimport Visibility from \"@material-ui/icons/Visibility\";\nimport VisibilityOff from \"@material-ui/icons/VisibilityOff\";\nimport { connect } from \"react-redux\";\nimport { Redirect } from \"react-router-dom\";\nimport { loginUser } from \"../Login/loginstate\";\nimport leftImage from \"../../static/9978094-new.png\";\nimport logo from \"../../static/AiSensy_Logo_Dark_PNG.png\";\nimport { APIURL as url } from \"../../config/config\";\nimport dialCodes, { countryMap, isoMap } from \"../../config/dialCodes\";\n\nconst getflag = (langcode) => {\n var first = langcode.charCodeAt(0) + 127397;\n var second = langcode.charCodeAt(1) + 127397;\n var flag = `${first};${second};`;\n const x = document.createElement(\"p\");\n x.innerHTML = flag;\n return x.innerText;\n};\n\nconst styles = (theme) => ({\n root: {\n height: \"100vh\",\n width: \"100%\",\n display: \"flex\",\n justifyContent: \"center\",\n alignItems: \"center\",\n },\n loginContainer: {\n display: \"flex\",\n height: \"100%\",\n width: \"100%\",\n margin: 0,\n },\n leftContainer: {\n backgroundColor: \"#cbe7e6\",\n display: \"flex\",\n flexDirection: \"column\",\n textAlign: \"center\",\n justifyContent: \"center\",\n position: \"relative\",\n width: \"40%\",\n height: \"100vh\",\n overflow: \"hidden\",\n padding: \"0 20px\",\n boxSizing: \"border-box\",\n [theme.breakpoints.down(960)]: {\n display: \"none\",\n },\n },\n logoLink: {\n position: \"absolute\",\n top: 20,\n left: 20,\n },\n rightLogoLink: {\n position: \"absolute\",\n top: \"-1.5%\",\n left: 5,\n [theme.breakpoints.up(960)]: {\n display: \"none\",\n },\n },\n logo: {\n width: 250,\n [theme.breakpoints.down(960)]: {\n width: 150,\n },\n },\n leftImageContainer: {\n width: \"100%\",\n height: \"auto\",\n overflow: \"hidden\",\n justifyContent: \"center\",\n alignItems: \"center\",\n flexGrow: 1,\n position: \"relative\",\n },\n leftImage: {\n width: \"60%\",\n maxWidth: \"100%\",\n height: \"auto\",\n marginTop: \"5%\",\n position: \"relative\",\n },\n centeredWrapper: {\n display: \"flex\",\n justifyContent: \"center\",\n margin: \"24px 0 0\",\n alignItems: \"center\",\n width: \"100%\",\n },\n rightContainer: {\n justifyContent: \"center\",\n alignItems: \"center\",\n display: \"flex\",\n flexDirection: \"column\",\n overflowY: \"auto\",\n height: \"100vh\",\n position: \"relative\",\n padding: \"20px\",\n width: \"60%\",\n [theme.breakpoints.down(\"md\")]: {\n width: \"100%\",\n height: \"auto\",\n padding: \"10px\",\n },\n },\n formContainer: {\n position: \"relative\",\n width: \"100%\",\n maxWidth: \"400px\",\n alignItems: \"center\",\n padding: \"0 50px 50px 50px\",\n [theme.breakpoints.down(\"sm\")]: {\n padding: \"20px\",\n },\n },\n nextButtonContainer: {\n display: \"flex\",\n justifyContent: \"center\",\n marginTop: \"7%\",\n width: \"100%\",\n },\n nextButton: {\n width: \"100%\",\n },\n googleSignInButton: {\n marginTop: theme.spacing(2),\n marginBottom: theme.spacing(2),\n backgroundColor: \"#0563fa\",\n color: \"white\",\n textTransform: \"none\",\n },\n signInButton: {\n marginTop: theme.spacing(2),\n textTransform: \"none\",\n \"&.disabled\": {\n backgroundColor: \"#ddd\",\n color: \"#888\",\n },\n \"&.enabled\": {\n color: \"#fff\",\n backgroundColor: \"#0a474c\",\n },\n },\n orDivider: {\n display: \"flex\",\n alignItems: \"center\",\n textAlign: \"center\",\n margin: theme.spacing(2, 0),\n \"& span\": {\n flex: 1,\n borderBottom: \"1px solid #ccc\",\n },\n \"& span:first-child\": {\n marginRight: theme.spacing(1),\n },\n \"& span:last-child\": {\n marginLeft: theme.spacing(1),\n },\n },\n signUpLink: {\n position: \"absolute\",\n top: 20,\n right: 20,\n [theme.breakpoints.down(960)]: {\n position: \"absolute\",\n top: \"9%\",\n right: 20,\n },\n [theme.breakpoints.down(538)]: {\n position: \"absolute\",\n top: \"7%\",\n right: 20,\n marginBottom: \"2%\",\n },\n },\n welcomeText: {\n marginBottom: theme.spacing(1),\n [theme.breakpoints.down(538)]: {\n marginBottom: theme.spacing(1),\n marginTop: \"18%\",\n },\n },\n forgotPassword: {\n marginTop: theme.spacing(2),\n textAlign: \"center\",\n },\n resendOtpButton: {\n marginTop: theme.spacing(2),\n display: \"flex\",\n justifyContent: \"center\",\n },\n errorText: {\n color: \"red\",\n textAlign: \"center\",\n },\n phoneTextFieldRoot: {\n width: \"100%\",\n padding: \"8px 20px\",\n marginTop: \"4px\",\n \"& input\": { width: \"calc(100% - 80px)\", marginLeft: \"80px\" },\n background: \"white\",\n border: \"2px solid grey\",\n },\n otpInput: {\n width: \"60px\",\n height: \"60px\",\n background: \"white\",\n padding: \"4px 6px\",\n fontSize: \"24px\",\n textAlign: \"center\",\n margin: \"0 5px\",\n border: \"1px solid #ccc\",\n borderRadius: \"4px\",\n },\n otpContainer: {\n display: \"flex\",\n justifyContent: \"center\",\n marginBottom: theme.spacing(2),\n },\n textFieldRootAutocomplete: {\n width: \"100%\",\n padding: \"2px 2px\",\n marginTop: \"4px\",\n background: \"white\",\n border: \"2px solid grey\",\n \"& label\": {\n background: \"white\",\n color: \"#b2b3b3 !important\",\n padding: \"0 6px\",\n marginTop: -4,\n },\n },\n codeTextFieldRoot: {\n width: \"80px\",\n padding: \"0 0 0 14px\",\n position: \"absolute\",\n zIndex: 1,\n margin: \"10px 0 0 2px\",\n background: \"white\",\n \"& input\": {\n padding: \"10px 0 !important\",\n },\n \"& button\": {\n width: \"12px\",\n position: \"relative\",\n left: \"7px\",\n background: \"white\",\n borderRadius: 0,\n \"&:hover\": {\n background: \"white\",\n },\n },\n \"& div\": {\n paddingRight: \"0px !important\",\n },\n },\n textFieldPopper: {\n [theme.breakpoints.down(\"md\")]: {\n left: \"12px !important\",\n minWidth: \"calc(100% - 35px)\",\n },\n [theme.breakpoints.up(\"md\")]: {\n left: \"calc(12.5vw - 16px) !important\",\n minWidth: \"calc(25vw + 54px)\",\n },\n },\n textFieldRoot: {\n backgroundColor: \"#f1f1f1\",\n outline: \"none\",\n border: \"none\",\n borderRadius: \"8px\",\n width: \"100%\",\n marginBottom: theme.spacing(3),\n padding: \"14px 15px\",\n boxShadow: \"none\",\n boxSizing: \"border-box\",\n fontFamily: \"Roboto, sans-serif\",\n fontSize: \"15px\",\n fontWeight: 500,\n },\n otpButtonContainer: {\n display: \"flex\",\n justifyContent: \"space-between\",\n marginTop: theme.spacing(3),\n },\n passwordFieldContainer: {\n alignItems: \"center\",\n position: \"relative\",\n width: \"100%\",\n },\n passwordTooltip: {\n position: \"absolute\",\n right: 10,\n },\n passwordField: {\n flex: 1,\n width: \"100%\",\n },\n passwordIcon: {\n marginLeft: theme.spacing(1),\n position: \"absolute\",\n right: \"-30px\",\n top: \"20px\",\n },\n passwordIconEmail: {\n marginLeft: theme.spacing(1),\n position: \"absolute\",\n right: \"2%\",\n top: \"48%\",\n },\n passwordIconName: {\n marginLeft: theme.spacing(1),\n position: \"absolute\",\n right: \"2%\",\n top: \"11%\",\n },\n passwordIconCompanyName: {\n marginLeft: theme.spacing(1),\n position: \"absolute\",\n right: \"2%\",\n top: \"23%\",\n },\n passwordIconContact: {\n marginLeft: theme.spacing(1),\n position: \"absolute\",\n right: \"2%\",\n top: \"49%\",\n },\n errorSnackbar: {\n backgroundColor: theme.palette.error.main,\n },\n});\n\nclass AgentLoginPage extends Component {\n constructor(props) {\n super(props);\n this.state = {\n step: 1,\n alertType: \"\",\n alertMsg: \"\",\n formValues: {\n name: \"\",\n country: \"\",\n state: \"\",\n city: \"\",\n companyName: \"\",\n contact: \"\",\n email: \"\",\n userName: \"\",\n loginPassword: \"\",\n type: \"Affiliate\",\n planType: \"tier0\",\n timezone: \"\",\n currency: \"\",\n partnershipFee: \"0\",\n otp: [\"\", \"\", \"\", \"\"],\n facebookAds: \"\",\n companySize: \"\",\n industry: [],\n websiteURL: \"\",\n confirmPassword: \"\",\n },\n errors: {\n email: \"\",\n userName: \"\",\n loginPassword: \"\",\n otp: \"\",\n confirmPassword: \"\",\n name: \"\",\n contact: \"\",\n companyName: \"\",\n },\n passwordValidation: {\n length: false,\n lowercase: false,\n number: false,\n specialChar: false,\n },\n showPasswordTooltip: false,\n generatedOtp: \"\",\n isOtpVerified: false,\n otpResendTime: 30,\n canResendOtp: false,\n countryCode: \"+91\",\n contact: \"\",\n snackbarOpen: false,\n snackbarMessage: \"\",\n isLoading: false,\n snackbarError: false,\n showPassword: false,\n showConfirmPassword: false,\n };\n }\n\n handleInputChange = (event) => {\n const { name, value } = event.target;\n const { formValues, errors } = this.state;\n\n if (name === \"contact\") {\n if (/^\\d*$/.test(value)) {\n // Only allow digits\n formValues[name] = value;\n if (value.length < 10) {\n errors.contact = \"Contact number should be 7 to 10 digits long\";\n } else {\n errors.contact = \"\";\n }\n }\n } else {\n formValues[name] = value;\n }\n\n this.setState({ formValues, errors });\n };\n\n handleOtpChange = (index, event) => {\n const otp = [...this.state.formValues.otp];\n otp[index] = event.target.value;\n if (event.target.value.length === 1 && index < 3) {\n this[`otpInput${index + 1}`].focus();\n }\n this.setState({ formValues: { ...this.state.formValues, otp } });\n };\n\n handleNext = () => {\n const { step } = this.state;\n if (this.isStepComplete()) {\n if (step === 1) {\n this.generateOtp();\n this.setState({ isLoading: true });\n }\n }\n };\n\n handleBack = () => {\n this.setState((prevState) => ({ step: prevState.step - 1 }));\n };\n\n handleSubmit = () => {\n const { formValues, countryCode } = this.state;\n const { userName, loginPassword, email, contact } = formValues;\n const updatedFormValues = { ...formValues, contact: countryCode + contact };\n let isChildAffiliate = true;\n\n this.props.loginUser(userName, null, email, contact, isChildAffiliate);\n };\n\n generateOtp = () => {\n const { email, name, contact } = this.state.formValues;\n axios\n .post(url + \"/affiliates/send-agent-otp\", {\n email,\n fullName: name,\n contact,\n })\n .then((response) => {\n if (response.data.success) {\n this.setState(\n {\n isLoading: false,\n generatedOtp: response.data.otp,\n isOtpVerified: false,\n otpResendTime: 30,\n canResendOtp: false,\n snackbarOpen: true,\n alertType: \"success\",\n snackbarMessage:\n \"OTP has been sent to your registered mobile number\",\n snackbarError: false,\n },\n () => {\n this.startOtpTimer();\n this.setState((prevState) => ({ step: prevState.step + 1 })); // Move to OTP step\n }\n );\n } else {\n this.setState({\n snackbarOpen: true,\n alertType: \"error\",\n snackbarMessage: \"Error Generating OTP\",\n isLoading: false,\n snackbarError: true,\n });\n console.error(\"Error generating OTP\");\n }\n })\n .catch((error) => {\n if (error.response) {\n this.setState({\n snackbarOpen: true,\n alertType: \"error\",\n snackbarMessage: error?.response?.data?.message,\n isLoading: false,\n snackbarError: true,\n });\n console.error(\"There was an error generating the OTP!\", error);\n }\n });\n };\n\n generateResendOtp = () => {\n const { email, name, contact } = this.state.formValues;\n axios\n .post(url + \"/affiliates/send-agent-otp\", {\n email,\n fullName: name,\n contact,\n })\n .then((response) => {\n if (response.data.success) {\n this.setState(\n {\n isLoading: false,\n generatedOtp: response.data.otp,\n isOtpVerified: false,\n otpResendTime: 30,\n canResendOtp: false,\n snackbarOpen: true,\n alertType: \"success\",\n snackbarMessage:\n \"The OTP has been resent to your registered mobile number\",\n snackbarError: false,\n },\n () => {\n this.startOtpTimer();\n }\n );\n } else if (response.data.blocked) {\n this.setState({\n snackbarOpen: true,\n alertType: \"error\",\n snackbarMessage:\n \"You have tried too many times, please contact customer support\",\n isLoading: false,\n snackbarError: true,\n });\n } else {\n console.error(\"Error generating OTP\");\n }\n })\n .catch((error) => {\n if (error.response) {\n if (error.response.data === \"Email already in use!\") {\n this.setState({\n snackbarOpen: true,\n alertType: \"error\",\n snackbarMessage: \"Email already in use\",\n isLoading: false,\n snackbarError: true,\n });\n } else if (error.response.data === \"blocked\") {\n this.setState({\n snackbarOpen: true,\n alertType: \"error\",\n snackbarMessage:\n \"You have tried too many times, please contact customer support\",\n isLoading: false,\n snackbarError: true,\n });\n }\n } else {\n console.error(\"There was an error generating the OTP!\", error);\n }\n });\n };\n\n startOtpTimer = () => {\n this.otpTimer = setInterval(() => {\n this.setState((prevState) => {\n if (prevState.otpResendTime <= 1) {\n clearInterval(this.otpTimer);\n return { otpResendTime: 0, canResendOtp: true };\n }\n return { otpResendTime: prevState.otpResendTime - 1 };\n });\n }, 1000);\n };\n\n verifyOtp = () => {\n const { formValues } = this.state;\n const { contact, otp } = formValues;\n axios\n .post(url + \"/affiliates/verify-agent-otp\", {\n contact,\n otp: otp.join(\"\"),\n })\n .then((response) => {\n if (response.data) {\n this.setState(\n { isOtpVerified: true, errors: { ...this.state.errors, otp: \"\" } },\n this.handleSubmit\n );\n } else {\n this.setState({\n isOtpVerified: false,\n errors: {\n ...this.state.errors,\n otp: \"Invalid OTP. Please try again.\",\n },\n });\n }\n })\n .catch((error) => {\n console.error(\"There was an error verifying the OTP!\", error);\n this.setState({\n isOtpVerified: false,\n errors: {\n ...this.state.errors,\n otp: \"An error occurred. Please try again.\",\n },\n });\n });\n };\n\n validateEmail = (email) => {\n const emailRegex = /^[^\\s@]+@[^\\s@]+\\.[^\\s@]+$/;\n return emailRegex.test(email);\n };\n\n isStepComplete = () => {\n const { step, formValues, errors } = this.state;\n const stepFields = {\n 1: [\"contact\"],\n 2: [\"otp\"],\n };\n\n const areFieldsFilled = stepFields[step]\n ? stepFields[step].every((field) => formValues[field])\n : true;\n const areFieldsValid = Object.values(errors).every((error) => !error);\n\n if (step === 1) {\n return areFieldsFilled && areFieldsValid;\n }\n\n return areFieldsFilled && areFieldsValid;\n };\n\n handleClickShowPassword = () => {\n this.setState((prevState) => ({\n showPassword: !prevState.showPassword,\n }));\n };\n\n handleClickShowConfirmPassword = () => {\n this.setState((prevState) => ({\n showConfirmPassword: !prevState.showConfirmPassword,\n }));\n };\n\n renderStep = () => {\n const {\n step,\n formValues,\n errors,\n passwordValidation,\n showPasswordTooltip,\n otpResendTime,\n canResendOtp,\n countryCode,\n showPassword,\n showConfirmPassword,\n } = this.state;\n const { classes } = this.props;\n\n switch (step) {\n case 1:\n return (\n \n \n \n
\n \n
i.dialCode)}\n classes={{ popper: classes.textFieldPopper }}\n getOptionLabel={(option) => {\n return `${getflag(isoMap[option])} ${option} ${\n countryMap[option.substring(1)]\n }`;\n }}\n onChange={(e, value, reason) => {\n this.setState({\n countryCode: value,\n });\n }}\n value={countryCode}\n renderInput={(params) => (\n \n )}\n />\n this.handleInputChange(e)}\n value={formValues.contact}\n className={classes.phoneTextFieldRoot}\n InputProps={{\n endAdornment: errors.contact && (\n \n \n \n \n \n ),\n }}\n />\n \n \n \n \n );\n case 2:\n return (\n \n \n \n {[0, 1, 2, 3].map((index) => (\n this.handleOtpChange(index, event)}\n className={classes.otpInput}\n inputRef={(input) => (this[`otpInput${index}`] = input)}\n />\n ))}\n \n \n \n OTP sent to {formValues.contact} Resend OTP after{\" \"}\n {`00:${otpResendTime < 10 ? \"0\" : \"\"}${otpResendTime}`}\n \n {otpResendTime === 0 ? (\n \n Didn't receive the code?{\" \"}\n \n Resend OTP\n \n \n ) : null}\n \n \n \n );\n default:\n return null;\n }\n };\n\n handleSnackbarClose = () => {\n this.setState({ snackbarOpen: false, snackbarError: false });\n };\n\n render() {\n const { isAuthenticated, classes } = this.props;\n const { step, formValues, snackbarOpen, snackbarMessage, snackbarError } =\n this.state;\n\n if (isAuthenticated) {\n return ;\n } else {\n return (\n \n
\n \n \n \n \n \n \n AiSensy Affiliate Dashboard\n \n {/* \n Start with 20% commission and Grow More !!!\n */}\n \n \n \n\n \n Trusted by 50000 + Brands\n \n \n \n\n \n \n \n Welcome! Sign in to continue.\n \n {/* \n Fill in the details below to complete your signup.\n */}\n
\n \n
\n
\n {this.renderStep()}
\n \n
\n {step > 1 && (\n \n Back\n \n )}\n {step === 2 ? (\n \n Verify OTP\n \n ) : (\n \n {this.state.isLoading ? (\n <>\n Sending OTP \n \n >\n ) : (\n \"Next \"\n )}\n \n )}\n
\n
\n
\n \n \n
\n \n {this.state.snackbarMessage}\n \n \n
\n );\n }\n }\n}\n\nconst mapDispatchToProps = {\n loginUser,\n};\n\nexport default connect(\n (state) => ({\n isAuthenticated: state.login.isAuthenticated,\n isLoading: state.login.isLoading,\n error: state.login.error,\n errmsg: state.login.errmsg,\n }),\n mapDispatchToProps\n)(withStyles(styles)(AgentLoginPage));\n","import React, { useEffect, useState } from \"react\";\nimport { BrowserRouter, Route, Redirect, Switch } from \"react-router-dom\";\nimport { connect } from \"react-redux\";\n\nimport ErrorPage from \"./commons/ErrorPage/ErrorPage\";\nimport ProjectsLayout from \"./commons/ProjectsLayout/ProjectsLayout\";\nimport Login from \"./pages/Login/Login\";\nimport Embad from \"./pages/Embed/EmbedSignup\";\nimport AffiliateSignUp from \"./pages/SignUp/AffiliateSignUp\";\nimport forgotPassword from \"./pages/forgotPassword/forgotPassword\";\n\nimport { fetchPartnerDetails } from \"./store/partnerState\";\nimport newLogin from \"./pages/Login/newLogin\";\nimport { CircularProgress } from \"@material-ui/core\";\nimport AgentLoginPage from \"./pages/SignUp/agentLogin\";\n\nconst PrivateRoute = ({\n component,\n isAuthenticated,\n fetchPartnerDetails,\n ...rest\n}) => {\n const [isLoading, setIsLoading] = useState(true);\n\n useEffect(() => {\n if (isAuthenticated) {\n fetchPartnerDetails()\n .then(() => setIsLoading(false))\n .catch(() => setIsLoading(false));\n } else {\n setIsLoading(false);\n }\n }, [isAuthenticated, fetchPartnerDetails]);\n\n document.title = \"Affiliate Dashboard\";\n return (\n {\n if (isLoading) {\n return (\n \n \n
\n );\n }\n\n return !isAuthenticated ? (\n \n ) : (\n \n );\n }}\n />\n );\n};\nconst PublicRoute = ({ component, isAuthenticated, ...rest }) => {\n document.title = \"Affiliate Dashboard\";\n return ;\n};\n\nconst App = (props) => {\n return (\n \n \n \n\n \n\n \n\n \n \n \n \n \n \n );\n};\n\nexport default connect(\n (state) => ({\n isAuthenticated: state.login.isAuthenticated,\n }),\n {\n fetchPartnerDetails,\n }\n)(App);\n","// This optional code is used to register a service worker.\n// register() is not called by default.\n\n// This lets the app load faster on subsequent visits in production, and gives\n// it offline capabilities. However, it also means that developers (and users)\n// will only see deployed updates on subsequent visits to a page, after all the\n// existing tabs open on the page have been closed, since previously cached\n// resources are updated in the background.\n\n// To learn more about the benefits of this model and instructions on how to\n// opt-in, read https://bit.ly/CRA-PWA\n\nconst isLocalhost = Boolean(\n window.location.hostname === 'localhost' ||\n // [::1] is the IPv6 localhost address.\n window.location.hostname === '[::1]' ||\n // 127.0.0.0/8 are considered localhost for IPv4.\n window.location.hostname.match(\n /^127(?:\\.(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)){3}$/\n )\n);\n\nexport function register(config) {\n if (process.env.NODE_ENV === 'production' && 'serviceWorker' in navigator) {\n // The URL constructor is available in all browsers that support SW.\n const publicUrl = new URL(process.env.PUBLIC_URL, window.location.href);\n if (publicUrl.origin !== window.location.origin) {\n // Our service worker won't work if PUBLIC_URL is on a different origin\n // from what our page is served on. This might happen if a CDN is used to\n // serve assets; see https://github.com/facebook/create-react-app/issues/2374\n return;\n }\n\n window.addEventListener('load', () => {\n const swUrl = `${process.env.PUBLIC_URL}/service-worker.js`;\n\n if (isLocalhost) {\n // This is running on localhost. Let's check if a service worker still exists or not.\n checkValidServiceWorker(swUrl, config);\n\n // Add some additional logging to localhost, pointing developers to the\n // service worker/PWA documentation.\n navigator.serviceWorker.ready.then(() => {\n console.log(\n 'This web app is being served cache-first by a service ' +\n 'worker. To learn more, visit https://bit.ly/CRA-PWA'\n );\n });\n } else {\n // Is not localhost. Just register service worker\n registerValidSW(swUrl, config);\n }\n });\n }\n}\n\nfunction registerValidSW(swUrl, config) {\n navigator.serviceWorker\n .register(swUrl)\n .then(registration => {\n registration.onupdatefound = () => {\n const installingWorker = registration.installing;\n if (installingWorker == null) {\n return;\n }\n installingWorker.onstatechange = () => {\n if (installingWorker.state === 'installed') {\n if (navigator.serviceWorker.controller) {\n // At this point, the updated precached content has been fetched,\n // but the previous service worker will still serve the older\n // content until all client tabs are closed.\n console.log(\n 'New content is available and will be used when all ' +\n 'tabs for this page are closed. See https://bit.ly/CRA-PWA.'\n );\n\n // Execute callback\n if (config && config.onUpdate) {\n config.onUpdate(registration);\n }\n } else {\n // At this point, everything has been precached.\n // It's the perfect time to display a\n // \"Content is cached for offline use.\" message.\n console.log('Content is cached for offline use.');\n\n // Execute callback\n if (config && config.onSuccess) {\n config.onSuccess(registration);\n }\n }\n }\n };\n };\n })\n .catch(error => {\n console.error('Error during service worker registration:', error);\n });\n}\n\nfunction checkValidServiceWorker(swUrl, config) {\n // Check if the service worker can be found. If it can't reload the page.\n fetch(swUrl, {\n headers: { 'Service-Worker': 'script' },\n })\n .then(response => {\n // Ensure service worker exists, and that we really are getting a JS file.\n const contentType = response.headers.get('content-type');\n if (\n response.status === 404 ||\n (contentType != null && contentType.indexOf('javascript') === -1)\n ) {\n // No service worker found. Probably a different app. Reload the page.\n navigator.serviceWorker.ready.then(registration => {\n registration.unregister().then(() => {\n window.location.reload();\n });\n });\n } else {\n // Service worker found. Proceed as normal.\n registerValidSW(swUrl, config);\n }\n })\n .catch(() => {\n console.log(\n 'No internet connection found. App is running in offline mode.'\n );\n });\n}\n\nexport function unregister() {\n if ('serviceWorker' in navigator) {\n navigator.serviceWorker.ready\n .then(registration => {\n registration.unregister();\n })\n .catch(error => {\n console.error(error?.response?.data?.message);\n });\n }\n}\n","const pending = \"#e65100\";\nconst resolved = \"#388e3c\";\nconst primary = \"#002D62\";\nconst secondary = \"#fec000\";\nconst warning = \"#FFC260\";\nconst success = \"#4caf50\";\nconst error = \"#f44336\";\nconst info = \"#9013FE\";\nconst backdrop = \"#f6f7ff\";\nconst lightenRate = 7.5;\nconst darkenRate = 15;\nconst queryText = \"#e0e0e0\";\n\nexport default {\n palette: {\n pending: {\n main: pending,\n },\n resolved: {\n main: resolved,\n },\n\n primary: {\n main: primary,\n },\n secondary: {\n main: secondary,\n },\n warning: {\n main: warning,\n },\n success: {\n main: success,\n },\n error: {\n main: error,\n },\n info: {\n main: info,\n },\n text: {\n primary: \"#4A4A4A\",\n secondary: \"#6E6E6E\",\n hint: \"#B9B9B9\",\n },\n background: {\n default: \"#F6F7FF\",\n light: \"#F3F5FF\",\n },\n backdrop: {\n default: \"#f6f7ff\",\n },\n },\n breakpoints: {\n values: {\n xs: 0,\n sm: 767,\n md: 959, // 1348\n lg: 1599, //1902\n xl: 1920,\n },\n },\n // shadows: [\"none\"],\n shape: {\n borderRadius: 6,\n },\n typography: {\n h2: {\n fontFamily: \"Roboto\",\n fontSize: 25,\n fontWeight: 600,\n letterSpacing: 2,\n wordSpacing: 5,\n \"@media (max-width:600px)\": {\n fontSize: 20,\n fontWeight: 700,\n wordSpacing: 5,\n letterSpacing: 1,\n },\n },\n\n h3: {\n fontFamily: \"Roboto\",\n fontSize: 20,\n \"@media (max-width:600px)\": {\n fontSize: 18,\n },\n },\n h4: {\n fontFamily: \"Roboto\",\n fontSize: 16,\n },\n h5: {\n fontFamily: \"Roboto\",\n fontSize: 14,\n fontWeight: 400,\n },\n h6: {\n fontFamily: \"Roboto\",\n fontSize: 12,\n fontWeight: 400,\n },\n body1: {\n fontFamily: \"Roboto\",\n fontSize: 14,\n },\n body2: {\n fontFamily: \"Roboto\",\n fontSize: 12,\n },\n },\n overrides: {\n MuiTextField: {\n root: {\n backgroundColor: \"rgb(240,240,240)\",\n borderRadius: \"8px\",\n boxSizing: \"border-box\",\n fontSize: \"15px\",\n fontWeight: 500,\n padding: \"6px 15px\",\n },\n \"&:focus\": {\n border: \"1px solid red\",\n },\n },\n MuiOutlinedInput: {\n root: {\n \"& $notchedOutline\": {\n border: \"none\",\n },\n \"&:hover $notchedOutline\": {\n border: \"none\",\n },\n \"&$focused $notchedOutline\": {\n border: \"none\",\n },\n \"&&& $input\": {\n padding: \"12px\",\n },\n \"&&& $input:first-child\": {\n padding: \"12px\",\n },\n },\n multiline: {\n padding: 0,\n },\n },\n MuiAutocomplete: {\n inputRoot: {\n '&[class*=\"MuiOutlinedInput-root\"].MuiAutocomplete-input': {\n padding: 12,\n },\n '&&[class*=\"MuiOutlinedInput-root\"]': {\n padding: 0,\n },\n },\n },\n MuiTooltip: {\n tooltip: {\n backgroundColor: \"white\",\n color: \"#222\",\n boxShadow: \"0 0 11px rgb(230,230,230)\",\n },\n },\n MuiButton: {\n root: {\n textTransform: \"none\",\n },\n contained: {\n boxShadow: \"none\",\n },\n containedPrimary: {\n color: \"white\",\n\n \"&:hover\": {\n cursor: \"pointer\",\n backgroundColor: \"#034694\",\n },\n },\n },\n MuiGrid: {\n container: {},\n },\n },\n props: {\n MuiInput: {\n disableUnderline: true,\n },\n },\n};\n","import defaultTheme from './default';\n\nexport default {\n default: defaultTheme\n}","import React from \"react\";\nimport ReactDOM from \"react-dom\";\nimport \"./index.css\";\nimport App from \"./App\";\nimport * as serviceWorker from \"./serviceWorker\";\n\nimport themes from \"./themes\";\nimport { MuiThemeProvider, createTheme } from \"@material-ui/core/styles\";\nimport { BrowserRouter } from \"react-router-dom\";\nimport { Provider } from \"react-redux\";\nimport store from \"./store/store\";\nimport setAuthToken from \"./helpers/setAuthToken\";\nimport jwt_decode from \"jwt-decode\";\nimport { compose } from \"redux\";\nimport { fetchPartnerDetails } from \"./store/partnerState\";\n\nconst login = async () => {\n try {\n if (localStorage.getItem(\"jwtToken\")) {\n await setAuthToken(localStorage.getItem(\"jwtToken\"));\n const decoded = jwt_decode(localStorage.getItem(\"jwtToken\"));\n const currentTime = Date.now() / 1000;\n if (decoded.exp < currentTime) {\n localStorage.removeItem(\"jwtToken\");\n await store.dispatch({ type: \"Login/SIGN_OUT_SUCCESS\" });\n } else {\n await store.dispatch({ type: \"Login/LOGIN_SUCCESS\", payload: decoded });\n const newObj = {\n partnerData: {\n ...decoded,\n _id: decoded.id,\n },\n };\n await store.dispatch({ type: \"PARTNER/LOAD_SUCCESS\", payload: newObj });\n }\n }\n } catch (error) {\n console.error({ message: \"invalid JWT token\", jwtError: error });\n await store.dispatch({ type: \"Login/SIGN_OUT_SUCCESS\" });\n }\n};\n\nlogin().then(() => {\n const theme = createTheme({ ...themes.default });\n\n const renderApp = () => {\n ReactDOM.render(\n \n \n \n \n \n \n \n \n ,\n document.getElementById(\"root\")\n );\n };\n\n // Initialize the app\n if (localStorage.getItem(\"jwtToken\")) {\n store.dispatch(fetchPartnerDetails()).then(renderApp).catch(renderApp);\n } else {\n renderApp();\n }\n\n // If you want your app to work offline and load faster, you can change\n // unregister() to register() below. Note this comes with some pitfalls.\n // Learn more about service workers: https://bit.ly/CRA-PWA\n serviceWorker.unregister();\n});\n"],"sourceRoot":""}