\n
\n
\n
\n Create a Batch \n \n \n \n Cancel\n \n \n
\n );\n }\n}\n\nexport default compose(\n withFormik({\n mapPropsToValues: () => ({\n stateIds: [],\n utilityIds: [],\n type: '',\n name: '',\n portfolioIds: [],\n projectIds: []\n }),\n\n validate: values => {\n const errors = {};\n const { stateIds, utilityIds, type, portfolioIds, projectIds } = values;\n\n if (\n !stateIds.length &&\n !utilityIds.length &&\n type === '' &&\n portfolioIds.length === 0 &&\n projectIds.length === 0\n ) {\n errors.selectOne = 'Criteria must be selected to create a batch.';\n }\n\n return errors;\n },\n handleSubmit: (values, { props, resetForm, setSubmitting }) => {\n const { month: forMonth, year: forYear } = props;\n const { name, stateIds, utilityIds, type, portfolioIds, projectIds } = values;\n\n const formatFilters = () => {\n const filters = [\n ...stateIds,\n ...utilityIds,\n type ? { type, label: SUBSCRIBER_TYPES[type].label } : [],\n ...portfolioIds,\n ...projectIds\n ];\n\n const filterNames = filters.map(item => (item.abbr ? item.abbr : item.label)).join(':');\n return filterNames;\n };\n\n const filters = formatFilters();\n\n const data = {\n name,\n utilityIds: utilityIds ? utilityIds.map(u => u.value) : [],\n stateIds: stateIds ? stateIds.map(s => s.value) : [],\n type: type ? [type] : [],\n forMonth,\n forYear,\n portfolioIds: portfolioIds ? portfolioIds.map(item => item.value) : [],\n projectIds: projectIds ? projectIds.map(item => item.value) : [],\n filterNames: filters\n };\n\n props.onCreateFromFilters(data);\n setSubmitting(false);\n resetForm();\n props.onToggle(false);\n }\n })\n)(CreateBatchForm);\n","import { withFormik } from 'formik';\nimport React, { Component } from 'react';\nimport { withRouter } from 'react-router-dom';\nimport { Form, FormGroup } from 'reactstrap';\nimport { compose } from 'recompose';\nimport * as Yup from 'yup';\n\nimport Loading from 'shared/components/Loading';\nimport MonthsDropdown from 'shared/components/MonthsDropdown';\nimport YearsDropdown from 'shared/components/YearsDropdown';\n\nclass PeriodForm extends Component {\n static defaultProps = {\n billingDates: {}\n };\n\n componentDidUpdate({ availableMonths, values: { month, year }, ...rest }) {\n const {\n billingDates,\n history,\n values: { month: nextMonth, year: nextYear }\n } = this.props;\n\n let monthGoTo = parseInt(nextMonth, 0);\n\n if (month !== nextMonth || year !== nextYear) {\n const available = billingDates ? billingDates[nextYear] : null;\n // if our present month doesn't exist in the options\n // navigate to the first month available.\n if (available && !available.includes(monthGoTo)) {\n monthGoTo = available[0];\n }\n\n available && history.push(`/admin/invoices/${monthGoTo}/${nextYear}`);\n }\n }\n\n render() {\n const { isFetching, handleSubmit, availableYears, availableMonths } = this.props;\n\n if (isFetching) return \n {totalInvoiceCount > successfullyGeneratedInvoiceCount && canFinalize ? (\n
\n
\n
\n Some invoices were not created. Please use the Preflight Check to identify errors, or delete the\n subscribers from the batch.\n
\n
\n ) : null}\n
Batch
\n
{name}
\n
\n \n \n \n
\n Total Subscribers\n
\n
{subscriberData.length}
\n
\n \n
\n Total Statements Generated\n
\n
\n \n {!canGenerate || failedToGenerateInvoiceCount > 0 ? successfullyGeneratedInvoiceCount : '--'}\n
\n \n
\n \n
\n Total Statements Sent\n
\n
\n {!canFinalize && !canGenerate ? successfullySentCount : '--'}
\n \n
\n \n
\n Total Invoices\n
\n
\n \n {showTotal ? : '--'}\n
\n \n
\n \n
\n Total Utility Credits\n
\n
\n \n {showTotal ? : '--'}\n
\n \n
\n \n
\n Total Allocation (kWac) \n
\n
\n {showTotal ? this.state.totalAllocationAmount?.toFixed(4) : '--'}
\n \n
\n \n \n
\n {filters?.length > 0 ? (\n
\n \n Batch Filters:
\n \n {filters.map(filter => {\n return (\n
\n {filter}\n
\n );\n })}\n
\n \n
\n ) : null}\n
\n );\n }\n}\n\nconst mapStateToProps = state => ({\n isInvoicingBusy: state.invoices.isInvoicingBusy\n});\n\nexport default connect(mapStateToProps)(InvoicesStatus);\n","import React, { Component } from 'react';\nimport { Alert, Button, Modal, ModalBody } from 'reactstrap';\n\nimport Icon from 'shared/components/Icon';\n\nclass PreflightModal extends Component {\n static displayName = 'PreflightModal';\n\n render() {\n const { onClearPreflight, onDownloadPreflight, month, year, preflightCheck } = this.props;\n\n if (preflightCheck && !!preflightCheck.alerts.length) {\n return (\n \n
\n \n {title}\n {this.renderSearchBar()}\n {actions && actions}\n \n\n {!!results.length ? (\n
\n
\n \n \n
\n \n \n
\n
Customer Name
\n
Projects
\n
Total Invoice
\n
Created
\n
\n Sent\n
\n
\n
\n {results.map(item => {\n const toggleSubscriber = showSubscriberInfo?.find(\n toggle => toggle.id === item.billingCustomerDetailId\n );\n const projectsReferenceActive = item.projectDetails.filter(\n project => project.status === ProjectBillingStatusDetail.InAnotherBillingForGivenPeriod\n ).length;\n const projectsReferenceRemoved = item.projectDetails.filter(\n project => project.status === ProjectBillingStatusDetail.OtherBillingReferenceRemoved\n ).length;\n\n if ((item?.projectDetails?.length === 0 && item.totalProjects > 0) || item?.isDeleted) {\n return null;\n }\n return (\n
\n
\n
\n
\n
\n {item.projectDetails.length > 0 ? (\n
handleSubscriberDetailsToggle(item.billingCustomerDetailId)}\n />\n ) : (\n
\n )}\n \n {item.displayName}\n
\n \n
\n
0 ? sctheme.blues.secondary : sctheme.black,\n cursor: item.projectDetails.length > 0 ? 'pointer' : 'auto',\n pointerEvents: item.projectDetails.length > 0 ? 'auto' : 'none'\n }}\n onClick={() => handleProjectDetailsToggle(item.billingCustomerDetailId)}\n >\n
\n
\n {item.projectDetails.length === 0 ? '—' : item.projectDetails.length}\n <>\n {projectsReferenceActive >= 1 || projectsReferenceRemoved >= 1 ? (\n
\n ) : null}\n {this.state.showProjectInfo === item.billingCustomerDetailId ? (\n
\n ) : null}\n >\n
\n
\n
\n
\n\n
\n {item?.status === BillingCustomerDetailStatus.InvoiceGenerated ||\n item?.status === BillingCustomerDetailStatus.FinalizedCompletely ||\n item?.status === BillingCustomerDetailStatus.FinalizedPartially ||\n item?.status === BillingCustomerDetailStatus.InvoiceRegenerated ? (\n \n ) : item.isInvoiceGenerated === false ? (\n '--'\n ) : null}\n
\n
\n {item.isInvoiceGenerated && this.renderDownloadLink(item, month, year)}\n {item.isInvoiceGenerated === false && (\n \n )}\n
\n
\n \n
\n
\n {!canGenerate && !canFinalize ? null : (\n handleDeleteSubscriber(item.billingCustomerDetailId)}\n />\n )}\n
\n
\n
\n
\n {toggleSubscriber?.open ? (\n
\n
\n
Project(s)
\n
In Existing Batch
\n
Project Invoice
\n
Utility Credit
\n
Allocation (kWac)
\n
\n
\n {item.projectDetails.map(project => (\n
\n ))}\n
\n ) : null}\n
\n );\n })}\n
\n \n \n
\n ) : !!results && this.state.query ? (\n
\n No results found for \"{this.state.query} \n
\n ) : (\n
\n {emptyText} \n
\n )}\n
\n );\n }\n}\n\nconst mapDispatchToProps = {\n fetchInvoice\n};\n\nexport default connect(null, mapDispatchToProps)(ResultsTable);\n","import invert from 'lodash/invert';\n\nexport const BillingProcess = {\n GeneratingInvoices: 0,\n FinalizingInvoices: 1,\n AbortingInvoices: 2,\n NotProcessing: 3\n};\n\nexport default Object.freeze({\n ...BillingProcess,\n ...invert(BillingProcess)\n});\n","import React, { useEffect, useState, useCallback } from 'react';\nimport BillingProcess from './BillingProcess';\nimport { Card } from 'reactstrap';\nimport { FontAwesomeIcon } from '@fortawesome/react-fontawesome';\nimport { faArrowsRotate } from '@fortawesome/pro-solid-svg-icons';\nimport ProgressBar from 'shared/components/ProgressBar';\nimport sctheme from 'app/styles/sctheme';\nimport { useSelector } from 'react-redux';\n\nconst Progress = props => {\n const {\n currentProcess,\n successfullyGeneratedInvoiceCount,\n successfullyFinalizedCount,\n queuedToAbortInvoiceCount,\n totalInvoiceCount\n } = props;\n\n const timestamp = useSelector(state => state.invoices.invoicingTimestamp);\n\n const [variables, setVariables] = useState({ a: null, b: null });\n\n const getProcessVariables = useCallback(() => {\n switch (currentProcess) {\n case BillingProcess.GeneratingInvoices:\n return { a: successfullyGeneratedInvoiceCount, b: totalInvoiceCount };\n case BillingProcess.FinalizingInvoices:\n return { a: successfullyFinalizedCount, b: totalInvoiceCount };\n case BillingProcess.AbortingInvoices:\n return { a: totalInvoiceCount - queuedToAbortInvoiceCount, b: totalInvoiceCount };\n default:\n return { a: null, b: null };\n }\n }, [\n currentProcess,\n successfullyGeneratedInvoiceCount,\n totalInvoiceCount,\n successfullyFinalizedCount,\n queuedToAbortInvoiceCount\n ]);\n\n useEffect(() => {\n const { a, b } = getProcessVariables();\n setVariables({ a, b });\n }, [getProcessVariables]);\n\n const process =\n currentProcess === BillingProcess.AbortingInvoices\n ? 'Deleting Invoices'\n : currentProcess === BillingProcess.GeneratingInvoices\n ? 'Generating Invoices'\n : 'Finalizing Invoices';\n\n const progress = variables.a && variables.a !== 0 ? (variables.a / variables.b) * 100 : 0;\n\n return (\n