\n {!showRichTextButtons && (\n
\n \n \n )}\n
\n
\n {showRichTextButtons && (\n
\n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n
\n )}\n {showMarkersButtons &&\n this.props.isNotCompleteForStudent() &&\n !showRichTextButtons && (\n
\n
\n \n
\n \n \n \n this.props.startDrawingMode(\n '#ff8f02'\n )\n }\n />\n \n this.props.startDrawingMode(\n '#000'\n )\n }\n />\n \n this.props.startDrawingMode(\n '#2A3DB9'\n )\n }\n />\n \n \n
\n )}\n
\n {this.props.isNotCompleteForStudent() && !showRichTextButtons && (\n
\n )}\n {this.props.isNotCompleteForStudent() && !showRichTextButtons && (\n
\n )}\n {showAdminSaveButtons && (\n
\n )}\n {showStudentSaveButtons &&\n this.props.isNotCompleteForStudent() &&\n !showRichTextButtons && (\n
\n )}\n {showStudentSaveButtons && !showRichTextButtons && !isIOS && (\n
\n )}\n {this.props.isNotCompleteForStudent() && !showRichTextButtons && (\n
\n )}\n
\n );\n }\n}\n\nProjectToolbar.propTypes = {};\n\nexport default ProjectToolbar;\n","export default \"\"","export default \"\"","export default \"\"","export default \"\"","import React from 'react';\nimport FabricUI from './FabricUI';\nimport { round } from 'lodash';\n\nclass BLM extends React.Component {\n constructor(props) {\n super(props);\n this.state = {\n pageContainerWidth: 0,\n pageContainerHeight: 0,\n marginLeft: 0,\n pageClassName: '',\n pageWidth: 0,\n pageHeight: 0\n };\n this.resizeProjectPage = this.resizeProjectPage.bind(this);\n }\n componentDidMount() {\n console.log('Project (BLM) component mounted');\n this.resizeProjectPage();\n window.addEventListener('resize', this.resizeProjectPage);\n }\n componentWillUnmount() {\n window.removeEventListener('resize', this.resizeProjectPage);\n console.log('unmounting BLM');\n }\n componentDidUpdate(prevProps) {\n let shouldResize = false;\n if (\n prevProps.leftPageContainerWidth !==\n this.props.leftPageContainerWidth ||\n prevProps.leftPageContainerHeight !==\n this.props.leftPageContainerHeight\n ) {\n shouldResize = true;\n // console.log('leftPageContainer changed in Project');\n }\n if (\n prevProps.blm.Title !== this.props.blm.Title ||\n prevProps.blmMode !== this.props.blmMode\n ) {\n shouldResize = true;\n // console.log('blm changed in project');\n }\n if (prevProps.pagesVisible !== this.props.pagesVisible) {\n shouldResize = true;\n // console.log('pagesVisible changed in project')\n }\n if (shouldResize) {\n this.resizeProjectPage();\n }\n }\n resizeProjectPage() {\n const topToolbarToggle = 40; // and close button\n const bottomToolbarHeight = 120;\n const widthOfArrow = 60;\n const widthOfProjectToolbar = 50;\n let maxContentHeight = 0;\n let maxContentWidth = 0;\n let scalePercent = 1;\n let pageContainerWidth = 0;\n let pageContainerHeight = 0;\n let marginLeft = 0;\n\n const pageWidth = this.props.blm.bounds[0][0];\n const pageHeight = this.props.blm.bounds[0][1];\n\n // if Project is displaying next to a book page, then subtract the page width and the two arrow buttson from the available space\n if (this.props.pagesVisible === 1) {\n maxContentHeight = window.innerHeight - topToolbarToggle;\n maxContentWidth = window.innerWidth - widthOfProjectToolbar - 80; // plus some extra whitespace\n } else {\n maxContentHeight =\n window.innerHeight - topToolbarToggle - bottomToolbarHeight;\n maxContentWidth =\n window.innerWidth -\n this.props.leftPageContainerWidth -\n widthOfArrow * 2 -\n widthOfProjectToolbar;\n marginLeft = this.props.leftPageContainerWidth + 50;\n }\n const heightDiff = pageHeight - maxContentHeight; // difference between the height of the project and the available hight we can display it in\n let widthDiff = pageWidth - maxContentWidth; // difference between the width of the project and the available width we have to display it in\n\n let pageClassName = this.getProjectClassName();\n // if we are lacking mostly height, then scale down by height\n if (heightDiff < widthDiff) {\n // TODO we might be able to increase the available width to display the wide BLM by adding margin top for the close button\n // limit by width\n scalePercent = round(maxContentWidth / pageWidth, 3);\n pageContainerWidth = pageWidth * scalePercent;\n pageContainerHeight = pageHeight * scalePercent;\n // console.log('resizing blm by width', maxContentHeight, maxContentWidth, pageContainerWidth, pageContainerHeight)\n } else {\n // limit by height\n scalePercent = round(maxContentHeight / pageHeight, 3);\n pageContainerWidth = pageWidth * scalePercent;\n pageContainerHeight = pageHeight * scalePercent;\n // console.log('resizing blm by height', maxContentHeight, maxContentWidth, pageContainerWidth, pageContainerHeight)\n }\n this.setState({\n pageContainerWidth,\n pageContainerHeight,\n pageClassName,\n marginLeft,\n pageWidth,\n pageHeight\n });\n this.props.updateProjectScalePercent(scalePercent);\n }\n\n getProjectClassName() {\n let pageClassName = 'blm ';\n if (this.props.pagesVisible === 1) {\n pageClassName += 'single-page';\n }\n return pageClassName;\n }\n\n render() {\n /*\n * size and position self\n * if single page: 'left page-left single-page'\n * if 2 page: 'right page-right blm-page'\n */\n const pagesVisible = this.props.pagesVisible;\n let projectClassName = '';\n if (pagesVisible === 1) {\n projectClassName = 'page left page-left single-page';\n } else {\n projectClassName = 'page right page-right blm-page';\n }\n if (!this.props.blmHtml) {\n return null;\n }\n const pageStyle = {\n width: `${this.state.pageContainerWidth}px`,\n height: `${this.state.pageContainerHeight}px`,\n marginLeft: `${this.state.marginLeft}px`,\n position: 'absolute'\n }; // the width and height will change as we zoom\n const jpedalStyle = {\n width: `${this.state.pageWidth}px`,\n height: `${this.state.pageHeight}px`,\n position: 'relative',\n display: 'block',\n transform: `translateY(0px) translateX(0px) scale(${this.props.projectScalePercent})`,\n transformOrigin: 'top left'\n }; // the scale will change as we zoom.\n return (\n \n
\n {this.getNotesHTML()}\n
this.bookPageTapped(e, pageNumber)}\n onMouseDown={(e) =>\n this.props.pagesOnMouseDown(e, pageNumber)\n }\n onMouseMove={this.props.pagesOnMouseMove}\n onMouseUp={(e) =>\n this.props.pagesOnMouseUp(e, pageNumber)\n }\n onTouchStart={(e) =>\n this.props.pagesOnMouseDown(e, pageNumber)\n }\n onDragStart={(e) => {\n e.preventDefault();\n }}\n />\n
\n
\n );\n }\n}\n\nBookPage.propTypes = {\n pagesOnMouseDown: PropTypes.func.isRequired,\n pagesOnMouseUp: PropTypes.func.isRequired,\n pagesOnMouseMove: PropTypes.func.isRequired,\n pagesVisible: PropTypes.number.isRequired,\n blmMode: PropTypes.bool.isRequired,\n user: PropTypes.object.isRequired,\n highlighterRight: PropTypes.object,\n highlighterLeft: PropTypes.object,\n wordTapped: PropTypes.func.isRequired,\n location: PropTypes.object,\n page: PropTypes.object.isRequired,\n pagesContainerHeight: PropTypes.number.isRequired,\n pagesContainerWidth: PropTypes.number.isRequired,\n updateLeftPageContainer: PropTypes.func.isRequired,\n leftPageContainerWidth: PropTypes.number.isRequired,\n leftPageContainerHeight: PropTypes.number.isRequired,\n updateBookScalePercent: PropTypes.func.isRequired\n};\n\nconst mapStateToProps = (state, ownProps) => {\n return {\n bookToolbar: state.bookToolbar,\n bookView: state.bookView,\n loading: state.ajaxCallsInProgress > 0,\n book: state.book\n };\n};\nexport default connect(mapStateToProps, {\n saveBookItem,\n setActiveNote,\n deleteBookItem,\n startPointing\n})(withRouter(BookPage));\n","/*\n * Pages component is responsible for containing the book and/or project pages.\n * it is also responsible for gestures and changing pages\n * It centers the pages currently only horrizontally ( in the future we would like it to center vertically as well) It positions the left and right arrow buttons.\n * it does this by adding up the width and height of the elements it contains, then centers itself\n * try simply adding up the width of the elements inside and then setting the pages container width.\n * calculate the width of each page by multiplying their bounds by the scalePercent of that page.\n */\n\nimport * as React from 'react';\n\nimport BLM from '../blm/BLM';\nimport BookPage from './BookPage';\nimport { Button } from 'react-bootstrap';\nimport PropTypes from 'prop-types';\nimport { StyledButton } from '../../constants/styledComponents';\nimport UserAPI from '../../api/userAPI';\nimport constants from '../../constants/constants';\nimport icon_arrowleft from '../../images/icon_arrowleft.png';\nimport icon_arrowright from '../../images/icon_arrowright.png';\nimport { viewerModes } from '../../constants/viewerModes';\nimport {map} from 'lodash';\nclass Pages extends React.Component {\n constructor(props) {\n super(props);\n this.state = {\n transX: '',\n transY: '',\n scaleX: '',\n scaleY: '',\n pagesContainerWidth: 0,\n pagesContainerHeight: 0,\n leftMarginForArrow: 0,\n dragging: false\n };\n this.handleSwipe = this.handleSwipe.bind(this);\n this.resizePagesContainer = this.resizePagesContainer.bind(this);\n this.pagesOnMouseDown = this.pagesOnMouseDown.bind(this);\n this.pagesOnMouseMove = this.pagesOnMouseMove.bind(this);\n this.pagesOnMouseUp = this.pagesOnMouseUp.bind(this);\n this.shouldShowProject = this.shouldShowProject.bind(this);\n\n this.lastClientX = 0;\n this.lastClientY = 0;\n this.theme = constants.themeProvider.activeTheme;\n }\n\n componentDidMount() {\n // window.addEventListener('resize', this.resizePagesContainer);\n // this.resizePagesContainer();\n }\n\n componentWillUnmount() {\n // window.removeEventListener('resize', this.resizePagesContainer);\n }\n\n /*\n * Step 5 in loading pages container\n *\n */\n componentDidUpdate(prevProps) {\n let shouldResize = false;\n if (\n prevProps.leftPageContainerWidth !==\n this.props.leftPageContainerWidth ||\n prevProps.leftPageContainerHeight !==\n this.props.leftPageContainerHeight\n ) {\n shouldResize = true;\n }\n if (prevProps.pagesVisible !== this.props.pagesVisible) {\n shouldResize = true;\n }\n // we must check for BLM mode changes here because the size and scale do not change when closing a BLM on a wide screen\n if (\n prevProps.blm.Title !== this.props.blm.Title ||\n prevProps.blmMode !== this.props.blmMode\n ) {\n shouldResize = true;\n }\n // if (prevProps.book.ID !== this.props.book.ID || prevProps.book.CurrentPage !== this.props.book.CurrentPage) {\n // this.resizePagesContainer();\n // }\n // if (prevProps.bookScalePercent !== this.props.bookScalePercent){\n // this.resizePagesContainer();\n // }\n // when the project is the right page, we need to resize after it has\n if (prevProps.projectScalePercent !== this.props.projectScalePercent) {\n shouldResize = true;\n }\n if (prevProps.book.bookIsReady !== this.props.book.bookIsReady) {\n shouldResize = true;\n }\n if (prevProps.blm.projectIsReady !== this.props.blm.projectIsReady) {\n shouldResize = true;\n }\n if (shouldResize) {\n this.resizePagesContainer();\n }\n }\n\n /*\n * resize the pages container\n *\n */\n resizePagesContainer() {\n // console.log('resizing pages container')\n // const maxContentHeight = window.innerHeight - 120; // height of window - toolbar and top toggle\n let leftW = 0;\n let leftH = 0;\n // let rightW = this.props.book.bounds[rIndex][0];\n // let rightH = this.props.book.bounds[rIndex][1];\n let rightW = 0;\n let rightH = 0;\n const pagesVisible = this.props.pagesVisible;\n let leftScalePercent = this.props.bookScalePercent; // TODO might rename bookScalePercent to leftPageScalePercent. We are Assuming that the pages of the book are all the same size.\n let rightScalePercent = 0;\n let totalWidthOfPages = 0;\n let tallestPageHeight = 0;\n let widthOfArrow = 50;\n let widthOfProjectToolbar = 50;\n let leftMarginForArrow = 0;\n let gapBetweenPages = 5;\n\n // if blmMode then use the bounds of the BLM instead.\n if (this.props.blmMode && this.props.blm.projectIsReady) {\n if (pagesVisible === 1) {\n leftScalePercent = this.props.projectScalePercent;\n leftW = this.props.blm.bounds[0][0] * leftScalePercent;\n leftH = this.props.blm.bounds[0][1] * leftScalePercent;\n totalWidthOfPages = leftW + widthOfProjectToolbar;\n } else {\n leftW = this.props.leftPageContainerWidth;\n leftH = this.props.leftPageContainerHeight;\n // TODO before determining the rightScalePercent we need to determin if we are limiting by height or width.\n // maybe we calculate both the scalePercent for height and width - then use the greater scaling (resulting in smaller page size)\n rightScalePercent = this.props.projectScalePercent;\n // rightScalePercentWidth = maxContentWidth / rightW;\n rightW = this.props.blm.bounds[0][0] * rightScalePercent;\n rightH = this.props.blm.bounds[0][1] * rightScalePercent;\n totalWidthOfPages =\n leftW + rightW + widthOfArrow + gapBetweenPages;\n leftMarginForArrow = leftW;\n // if the Project is wider than it is tall, switch to single page view automatically\n if (leftW > leftH) {\n this.props.automaticUpdatePagesVisible(1);\n }\n }\n } else if (this.props.book.bookIsReady) {\n if (pagesVisible === 1) {\n leftW = this.props.leftPageContainerWidth;\n leftH = this.props.leftPageContainerHeight;\n totalWidthOfPages = leftW;\n leftMarginForArrow = leftW;\n } else {\n // set both left and right to the leftPageContainer because it is easier and we can assume that both pages of the book are the same size.\n leftW = this.props.leftPageContainerWidth;\n leftH = this.props.leftPageContainerHeight;\n rightW = this.props.leftPageContainerWidth;\n rightH = this.props.leftPageContainerHeight;\n totalWidthOfPages = leftW + rightW + gapBetweenPages;\n leftMarginForArrow = leftW + rightW + gapBetweenPages;\n // if the book page is wider than it is tall, switch to single page view automatically\n // if (leftW > leftH){\n // this.props.automaticUpdatePagesVisible(1);\n // }\n }\n }\n\n tallestPageHeight = leftH > rightH ? leftH : rightH;\n\n this.setState({\n pagesContainerHeight: tallestPageHeight,\n pagesContainerWidth: totalWidthOfPages,\n leftMarginForArrow\n });\n }\n\n handleSwipe(direction) {\n this.props.resetActiveTimeout();\n console.log(`you swiped ${direction}`);\n switch (direction) {\n case 'top':\n break;\n case 'bottom':\n break;\n case 'left':\n this.props.nextPage();\n break;\n case 'right':\n this.props.prevPage();\n break;\n default:\n break;\n }\n }\n\n getBookPage(page, pageID) {\n return (\n
\n );\n }\n\n /*\n * do Not show the book if we are in BLMMode and only 1 page visible\n * wait to show the book until prop.pages is defined\n */\n shouldShowBook() {\n let showBook = false;\n if (this.state.pagesContainerWidth === 0) {\n showBook = false;\n }\n if (!this.props.book.bookIsReady) {\n return false;\n }\n\n // wait for the highlighters to be ready - this might block the book page from loading completely\n // if (!this.props.highlighterLeft || !this.props.highlighterRight){\n // return false;\n // }\n if (this.props.blmMode && this.props.pagesVisible === 1) {\n showBook = false;\n } else if (!!this.props.book.cachedPages) {\n showBook = true;\n } else {\n showBook = false;\n }\n return showBook;\n }\n shouldShowProject() {\n let showProject = false;\n if (\n this.props.blmMode &&\n this.props.blm.bounds &&\n this.state.pagesContainerWidth &&\n this.props.blm.projectIsReady\n ) {\n showProject = true;\n }\n return showProject;\n }\n\n /*\n * shouldShowCLoseButton\n */\n shouldShowCloseButton = () => {\n const { RoleID } = this.props.user;\n if (\n // is a student\n UserAPI.isStudent(RoleID) === true ||\n // is generic and not resource mode\n (UserAPI.isGeneric(RoleID) === true &&\n this.props.viewerMode !== viewerModes.MODE_RESOURCE) ||\n // is demo\n UserAPI.isDemo(RoleID) === true\n ) {\n return true;\n } else {\n return false;\n }\n };\n\n /*\n * respond to mouse down\n * if we are not already dragging and the pointer tool is active, then drag scroll\n */\n pagesOnMouseDown(e, pageNumber) {\n e.persist();\n if (!this.state.dragging && this.props.bookToolbar.pointing) {\n this.setState({ dragging: true });\n this.lastClientX = e.clientX || e.touches[0].clientX;\n this.lastClientY = e.clientY || e.touches[0].clientY;\n const tempX = this.lastClientX;\n const tempY = this.lastClientY;\n\n // if they don't move much, then they are highlighting rather than dragging\n setTimeout(() => {\n const moveX = parseInt(tempX - this.lastClientX, 10);\n const moveY = parseInt(tempY - this.lastClientY, 10);\n if (moveX < 30 || moveY < 30) {\n this.setState({ dragging: false });\n }\n }, 200);\n }\n this.props.onMouseDown(e, pageNumber);\n }\n pagesOnMouseMove(e) {\n e.persist();\n if (this.state.dragging) {\n this.pagesContainer.scrollLeft -=\n -this.lastClientX + (this.lastClientX = e.clientX);\n this.pagesContainer.scrollTop -=\n -this.lastClientY + (this.lastClientY = e.clientY);\n this.mouseMove++;\n }\n this.props.onMouseMove(e);\n }\n\n pagesOnMouseUp(e, pageNumber) {\n if (this.state.dragging) {\n this.setState({ dragging: false });\n }\n\n e.persist();\n this.props.onMouseUp(e, pageNumber);\n }\n\n render() {\n let bookReadyClass = this.shouldShowBook() ? 'book-ready' : '';\n let classes = `pages ${this.props.bookToolbar.pagesClassName} ${bookReadyClass}`;\n let showBookArrows = 'block';\n // TODO canvasClassForBLM is defined in Pages as well as BLM - pick one!\n let canvasClassForBlm, bookArrowLeftStyle, bookArrowRightStyle;\n if (this.props.pagesVisible === 1 && this.props.blmMode) {\n showBookArrows = 'none';\n canvasClassForBlm = 'page single-page blm-canvas-container';\n } else if (this.props.pagesVisible === 2 && this.props.blmMode) {\n canvasClassForBlm = 'page blm-page blm-canvas-container';\n }\n if (this.props.blmMode) {\n classes += ' blm-mode';\n }\n if (this.props.pagesVisible === 1) {\n classes += ' one-visible';\n } else {\n classes += ' two-visible';\n }\n bookArrowLeftStyle = { display: `${showBookArrows}` };\n bookArrowRightStyle = {\n display: `${showBookArrows}`,\n marginLeft: `${this.state.leftMarginForArrow}px`\n };\n const pagesContainerStyle = {\n width: `${this.state.pagesContainerWidth}px`,\n height: `${this.state.pagesContainerHeight}px`\n };\n return (\n
\n {this.shouldShowCloseButton() && (\n
this.props.closeBookView()}\n >\n \n \n )}\n
\n this.props.updatePagesVisible(1)}\n >\n \n \n this.props.updatePagesVisible(2)}\n >\n \n \n
\n
\n
\n
\n
\n
\n
\n
\n\n {this.shouldShowBook() &&\n map(this.props.book.cachedPages, (page, pageID) => {\n return this.getBookPage(page, pageID);\n })}\n\n {this.shouldShowProject() && (\n
\n )}\n
\n
\n );\n }\n}\n\nPages.propTypes = {\n onMouseDown: PropTypes.func.isRequired,\n onMouseUp: PropTypes.func.isRequired,\n onMouseMove: PropTypes.func.isRequired,\n nextPage: PropTypes.func.isRequired,\n prevPage: PropTypes.func.isRequired,\n pagesVisible: PropTypes.number.isRequired,\n blmMode: PropTypes.bool.isRequired,\n user: PropTypes.object.isRequired,\n book: PropTypes.object.isRequired,\n highlighterRight: PropTypes.object,\n highlighterLeft: PropTypes.object,\n wordTapped: PropTypes.func.isRequired,\n location: PropTypes.object,\n updateLeftPageContainer: PropTypes.func.isRequired,\n leftPageContainerWidth: PropTypes.number.isRequired,\n leftPageContainerHeight: PropTypes.number.isRequired,\n bookScalePercent: PropTypes.number.isRequired,\n updateBookScalePercent: PropTypes.func.isRequired,\n updateProjectScalePercent: PropTypes.func.isRequired,\n projectScalePercent: PropTypes.number,\n automaticUpdatePagesVisible: PropTypes.func.isRequired,\n bookToolbar: PropTypes.object.isRequired,\n resetActiveTimeout: PropTypes.func.isRequired,\n viewerMode: PropTypes.string.isRequired\n};\n\nexport default React.forwardRef((props, ref) => {\n return
;\n });;\n","export default \"\"","export default \"\"","export default __webpack_public_path__ + \"static/media/move_icon.de0c46bd.png\";","import * as React from 'react';\n\nimport {\n Button,\n Col,\n Dropdown,\n FormControl,\n FormGroup,\n Row\n} from 'react-bootstrap';\nimport { StyledButton, StyledDiv } from '../../constants/styledComponents';\n\nimport CommonModal from '../common/CommonModal';\nimport FontAwesome from 'react-fontawesome';\nimport constants from '../../constants/constants';\nimport icon_eraser from '../../images/icon_eraser.png';\nimport icon_hl_green from '../../images/icon_hl_green.png';\nimport icon_hl_orange from '../../images/icon_hl_orange.png';\nimport icon_hl_teal from '../../images/icon_hl_teal.png';\nimport icon_hl_yellow from '../../images/icon_hl_yellow.png';\nimport icon_pen from '../../images/icon_pen.png';\nimport icon_pointer from '../../images/icon_pointer.png';\nimport move_icon from '../../images/move_icon.png';\nimport { Ibook, IbookToolbar, Iuser, IviewerSettings } from '../../models';\nimport { viewerModes } from '../../constants/viewerModes';\n\ninterface Iprops {\n lastPage: ()=>void;\n firstPage: ()=>void;\n nextPage: ()=>void;\n prevPage: ()=>void;\n user: Iuser;\n book: Ibook;\n startHighlighter: (e: any, color: string)=>void;\n startEraser: ()=>void;\n startPointer: ()=>void;\n startNotes: ()=>void;\n openDrawer: ()=>void;\n viewerSettings: IviewerSettings;\n currentPage: number;\n blmMode: boolean;\n pagesVisible: number;\n bookToolbar: IbookToolbar;\n decreaseBookZoom: ()=>void;\n increaseBookZoom: ()=>void;\n resetActiveTimeout: ()=>void;\n viewerMode?: string;\n goToPage: (pageNumber: number, callback?: ()=>void, shouldCheckLeftPageEven?: boolean)=>void;\n bookIsZooming: boolean;\n}\ninterface Istate {\n showGoToPageModal: boolean;\n goToPage: string;\n showBookMarkupTools: boolean;\n}\nconst theme = constants.themeProvider.activeTheme;\n\nclass Toolbar extends React.Component
{\n private startedHighlighting = false;\n\n constructor(props: Iprops) {\n super(props);\n\n this.state = {\n showGoToPageModal: false,\n goToPage: '',\n showBookMarkupTools: this.shouldShowMarkupTools()\n \n };\n this.startedHighlighting = false;\n }\n/**\n * @return {boolean}\n */\n shouldShowMarkupTools = (): boolean => {\n const hasCorrectViewerMode = this.props.viewerMode !==\n viewerModes.MODE_TEACHER_VIEW_STUDENT_BOOK &&\n this.props.viewerMode !==\n viewerModes.MODE_TEACHER_CLASS_NOTES &&\n this.props.viewerMode !==\n viewerModes.MODE_TEACHER_GROUP_NOTES &&\n this.props.viewerMode !==\n viewerModes.MODE_TEACHER_STUDENT_BLM &&\n this.props.viewerMode !== viewerModes.MODE_RESOURCE;\n\n const {IsEPub} = this.props.book;\n return hasCorrectViewerMode && IsEPub === false;\n }\n\n startHL = (e: any, color: string) => {\n // console.log('starting hl', e.type)\n /*\n * stopPropegation rather than preventDefault because preventDefault on iOS leaves the selected text highlighted\n * and we do not want the text to remain highlighted. Then you can't see the color of the highlight.\n * Unfortunately only dong stopPropagation does not prevent this from being clicked twice... so we have a timeout.\n * after further testing apparently we do not need to preventDefault here... Will keep this around for a bit.\n */\n // if (this.startedHighlighting) return;\n // this.startedHighlighting = true;\n // setTimeout(() => {\n // this.startedHighlighting = false;\n // }, 300)\n\n // if (e.type === 'touchend'){\n // e.stopPropagation();\n // }\n // e.preventDefault();\n\n this.props.startHighlighter(e, color);\n };\n closeGoToPageModal = () => {\n this.setState({ showGoToPageModal: false });\n };\n openGoToPageModal = (e: any) => {\n this.props.resetActiveTimeout();\n e.preventDefault();\n this.setState({ showGoToPageModal: true });\n };\n\n submitGoToPage = () => {\n this.props.resetActiveTimeout();\n this.setState(\n {\n showGoToPageModal: false\n },\n () => {\n this.props.goToPage(parseInt(this.state.goToPage, 10));\n this.setState({ goToPage: '' });\n }\n );\n };\n toggleColors = (isOpen: boolean, e: any) => {\n this.props.resetActiveTimeout();\n if (e) {\n e.stopPropagation();\n e.preventDefault();\n }\n };\n\n handleChange = (e: any) => {\n this.setState({\n goToPage: e.target.value\n });\n };\n\n inputForm = () => {\n return (\n \n \n \n \n \n \n
\n );\n };\n\n render() {\n let icon_hl = icon_hl_yellow;\n let toolbarStyle, pointerIcon;\n if (this.props.bookToolbar.highlightColor === 'hl-green') {\n icon_hl = icon_hl_green;\n } else if (this.props.bookToolbar.highlightColor === 'hl-teal') {\n icon_hl = icon_hl_teal;\n }\n if (this.props.bookToolbar.highlightColor === 'hl-orange') {\n icon_hl = icon_hl_orange;\n }\n if (this.props.viewerSettings.showToolbar) {\n toolbarStyle = 'toolbar row';\n } else {\n toolbarStyle = 'toolbar row d-none';\n }\n if (this.props.bookIsZooming) {\n pointerIcon = move_icon;\n } else {\n pointerIcon = icon_pointer;\n }\n const bookToolbar = this.props.bookToolbar;\n const pointerClassName = bookToolbar.pointing\n ? 'item hl-pointer active'\n : 'item hl-pointer';\n const highlighterClassName =\n bookToolbar.highlighting &&\n bookToolbar.highlightColor !== 'hl-strikethrough' &&\n bookToolbar.highlightColor !== 'hl-underline'\n ? `item-child btn btn-light dropdown-toggle hl-highlight active`\n : 'item-child btn btn-light dropdown-toggle hl-highlight';\n const strikeClassName =\n bookToolbar.highlighting &&\n bookToolbar.highlightColor === 'hl-strikethrough'\n ? 'item icon-btn hl-strike active'\n : 'item icon-btn hl-strike';\n const underlineClassName =\n bookToolbar.highlighting &&\n bookToolbar.highlightColor === 'hl-underline'\n ? 'item icon-btn hl-under active'\n : 'item icon-btn hl-under';\n const eraseClassName = bookToolbar.erasing\n ? 'item hl-erase active'\n : 'item hl-erase';\n let notesClassName = bookToolbar.allowNotes\n ? 'item hl-notes active'\n : 'item hl-notes';\n if (this.props.book.IsEPub){\n notesClassName = 'hide'\n }\n // hide the bottom toolbar if in single page Project mode\n if (this.props.blmMode && this.props.pagesVisible === 1) {\n return null;\n }\n return (\n \n \n
\n {this.state.showBookMarkupTools && (\n
\n \n \n \n \n \n \n )}\n {/* notes button stays outside the span above for teachers to add notes */}\n
\n
\n
\n {/*\n \n
\n \n \n \n \n \n \n \n */}\n \n {\n this.props.resetActiveTimeout();\n this.props.decreaseBookZoom();\n }}\n >\n \n \n {\n this.props.resetActiveTimeout();\n this.props.increaseBookZoom();\n }}\n >\n \n \n
\n \n \n \n \n \n \n
\n this.closeGoToPageModal()}\n submit={this.submitGoToPage}\n />\n \n );\n }\n}\n\nexport default Toolbar;\n","export default \"\"","export default \"\"","export default \"\"","import rangy from 'rangy/lib/rangy-core';\nimport 'rangy/lib/rangy-textrange';\nimport 'rangy/lib/rangy-serializer';\nimport 'rangy/lib/rangy-selectionsaverestore';\nimport 'rangy/lib/rangy-highlighter';\nimport 'rangy/lib/rangy-classapplier';\n\nconst testForSimilarMarkup = (element, highlightColor) => {\n if (\n element.className.includes(highlightColor) &&\n element.textContent !== ' '\n ) {\n // alert('found similar')\n return true;\n } else {\n return false;\n }\n};\n\n// search through all the possible matches, then merge them\nconst findSimilarMarkups = (target, whichSide, pid, highlightColor) => {\n let range = rangy.createRange();\n let currentRange = rangy.createRange();\n let willMergeMarkups = false;\n currentRange.selectNodeContents(target);\n range.selectNode(target); // start with the initial selection of the word\n findPreviousMarkups(target); // increase the range before this markup?\n findNextMarkups(target); // increase the selection range after this markup?\n if (willMergeMarkups) {\n range.select();\n }\n\n function findPreviousMarkups(element) {\n let didFindElement = false; // keep track of if we found one\n let foundElement;\n\n // find elements in non-audio books\n try {\n if (\n testForSimilarMarkup(\n element.previousElementSibling,\n highlightColor\n )\n ) {\n didFindElement = true;\n foundElement = element.previousElementSibling;\n }\n } catch (error) {}\n // find elements that are on different lines\n try {\n if (\n !foundElement &&\n element.parentNode.children.length === 1 &&\n element.parentNode.className.includes('t') &&\n testForSimilarMarkup(\n element.parentNode.previousElementSibling.children[0],\n highlightColor\n )\n ) {\n didFindElement = true;\n foundElement =\n element.parentNode.previousElementSibling.children[0];\n }\n } catch (error) {}\n // find elements in audio books - they are buried in an additional span occassionaly\n try {\n if (\n !foundElement &&\n element.parentNode.children.length === 1 &&\n element.parentNode.className.includes('t') &&\n testForSimilarMarkup(\n element.parentNode.previousElementSibling.children[0]\n .children[0],\n highlightColor\n )\n ) {\n didFindElement = true;\n foundElement =\n element.parentNode.previousElementSibling.children[0]\n .children[0];\n }\n } catch (error) {}\n\n // if we found something, then take that something and see if there is another similar markup before that\n if (didFindElement) {\n willMergeMarkups = true;\n range.setStart(foundElement);\n\n findPreviousMarkups(foundElement);\n }\n }\n\n function findNextMarkups(element) {\n let didFindElement = false; // keep track of if we found one\n let foundElement;\n try {\n if (\n testForSimilarMarkup(element.nextElementSibling, highlightColor)\n ) {\n didFindElement = true;\n foundElement = element.nextElementSibling;\n }\n } catch (error) {}\n\n try {\n if (\n !foundElement &&\n element.parentNode.children.length === 1 &&\n element.parentNode.className.includes('t') &&\n testForSimilarMarkup(\n element.parentNode.nextElementSibling.children[0],\n highlightColor\n )\n ) {\n didFindElement = true;\n foundElement =\n element.parentNode.nextElementSibling.children[0];\n }\n } catch (error) {}\n try {\n if (\n !foundElement &&\n element.parentNode.children.length === 1 &&\n element.parentNode.className.includes('t') &&\n testForSimilarMarkup(\n element.parentNode.nextElementSibling.children[0]\n .children[0],\n highlightColor\n )\n ) {\n didFindElement = true;\n foundElement =\n element.parentNode.nextElementSibling.children[0]\n .children[0];\n }\n } catch (error) {}\n // if we found something, then take that something and see if there is another similar markup AFTER that\n if (didFindElement) {\n willMergeMarkups = true;\n range.setEnd(\n foundElement.firstChild,\n foundElement.firstChild.length\n );\n\n findNextMarkups(foundElement);\n }\n }\n};\n\nexport default findSimilarMarkups;\n","/*\n * Eventually the BookView component will only be responsible for\n */\n\nimport 'rangy/lib/rangy-textrange';\nimport 'rangy/lib/rangy-serializer';\nimport 'rangy/lib/rangy-selectionsaverestore';\nimport 'rangy/lib/rangy-highlighter';\nimport 'rangy/lib/rangy-classapplier';\n\n// import 'react-jplayer/src/less/skins/sleek.less';\n// Styles Play/Pause/Mute etc when icons () are used for them\n// import 'react-jplayer/src/less/controls/iconControls.less';\n\nimport { viewerModes } from '../../constants/viewerModes';\nimport EpubView from './EpubView';\nimport React, { Component } from 'react';\nimport {\n automaticUpdatePagesVisible,\n decreaseBookZoom,\n increaseBookZoom,\n resetBookView,\n setActiveNote,\n updateBookScalePercent,\n updateLeftPageContainer,\n updatePagesVisible,\n updatePendingItem,\n updateProjectScalePercent\n} from '../../actions/bookViewActions';\nimport {\n cacheBookPage,\n deleteBookItem,\n downloadSpeech,\n getBookByID,\n getBookItems,\n getBookItemsByStudentID,\n getBookPage,\n getBookProperties,\n searchBookBagBooks,\n getSpeechMarks,\n getTeacherBookItems,\n nextPage,\n prevPage,\n resetCachedBookPages,\n saveBookItem,\n setBookReady,\n updateBookStatus,\n updateCurrentPage,\n} from '../../actions/bookActions';\nimport {\n emptyBLMs,\n getAssignedBLMs,\n getBLMByID,\n getBLMItems,\n getBLMProperties,\n getBLMStatus,\n getStudentsBLMItems,\n getTeachersComment,\n getTemplateItems,\n setProjectReady\n} from '../../actions/blmActions';\nimport { hashHistory, withRouter } from 'react-router';\nimport {\n manualAjaxEnd,\n manualAjaxStart\n} from '../../actions/ajaxStatusActions';\nimport {\n startErasing,\n startHighlighting,\n startMoving,\n startNotes,\n startPointing,\n startStriking,\n startUnderlining\n} from '../../actions/bookToolbarActions';\n\nimport BlmAPI from '../../api/blmAPI';\nimport CommonModal from '../common/CommonModal';\nimport Drawer from './Drawer';\nimport { HeaderContainer } from '../header/HeaderContainer';\nimport Loading from '../common/Loading';\nimport Pages from './Pages';\nimport Toolbar from './Toolbar';\nimport UserAPI from '../../api/userAPI';\nimport config from '../../api/config';\nimport { connect } from 'react-redux';\nimport constants from '../../constants/constants';\nimport { debounce } from 'lodash';\nimport findSimilarMarkups from '../../utilities/MarkupMergerUtility';\nimport rangy from 'rangy/lib/rangy-core';\nimport { removeQuery } from '../../vendor/utils-router';\nimport { toastr } from 'react-redux-toastr';\nimport { userLogout } from '../../actions/userActions';\nimport { requireSignIn } from '../../routes';\nimport $ from 'jquery';\nimport {\n BOOK_FIRST_PAGE,\n BOOK_LAST_PAGE,\n BOOK_NEXT_PAGE,\n BOOK_PREV_PAGE\n} from './events';\nimport { viewerDrawerTypeEnum } from '../../models-enums';\n\nwindow.rangy = rangy;\n\nclass BookView extends Component {\n highlights;\n activeTimeout;\n activeWarningTimeout;\n\n constructor(props) {\n super(props);\n this.state = {\n highlighterLeft: null,\n highlighterRight: null,\n EPubUrl: '',\n blmHtml: {},\n viewerSettings: {\n showDrawer: false,\n showDrawerHighlightsButton: true,\n showDrawerNotesButton: true,\n drawerType: viewerDrawerTypeEnum.notes,\n highlightingMouseDown: false,\n header: { visible: false, class: 'fa fa-angle-double-down' },\n showToolbar: true,\n showHeader: true,\n highlightingPageNumber: 1,\n loadedProjectAssignmentID: '' // keep track of what Project Assignment the blmData belongs too so we know when to re-load it\n },\n showSaveProjectConfirm: false,\n saveProjectSubmit: () => {\n console.error('saveProjectModal function not replaced');\n }\n };\n this.pagesRef = React.createRef();\n\n this.superDebouncedMarkupUpdate = debounce(this.updateMarkups, 8000);\n this.debouncedMarkupUpdate = debounce(this.updateMarkups, 1000);\n\n // lets store moseMove on the BookView class because we are changing it very quickly as the mouse moves.\n this.mouseMove = 0;\n\n // let speechMarkRaw = constants.testSpeechMarkRaw;\n // this.speechMarksData = speechMarkRaw.split(/\\n/);\n this.highlightedSpeechMarks = [];\n this.speechMarkIndex = 0;\n }\n\n componentDidMount() {\n console.info('bookview mounted');\n if (requireSignIn(this.props.user, this.props.location)) {\n return;\n }\n\n // initialize all rangy stuff\n rangy.init();\n this.setState({\n highlighterRight: this.createHighlighter(),\n highlighterLeft: this.createHighlighter()\n });\n // open the header for testing\n // this.toggleHeader();\n\n /*\n * Initial Viewer Settings\n */\n let initialViewerSettings = {};\n\n // should we show the header?\n initialViewerSettings.showHeader = UserAPI.canAccessBook(\n this.props.user.RoleID\n );\n\n // load list of BLMs and Books?\n if (UserAPI.isStudent(this.props.user.RoleID)) {\n // this.initBooks(); // TODO fix the action in here... it expects a lot of params\n this.initBLMs();\n } else {\n // load blm if you are not a student\n if (this.props.blmMode) {\n if (\n this.props.viewerMode ===\n viewerModes.MODE_TEACHER_STUDENT_BLM\n ) {\n this.loadBLM(\n this.props.projectAssignmentID,\n this.props.blmID\n ).then(() => {\n this.props.setProjectReady(true);\n $('#pages').fadeTo('fast', 1.0);\n // if blmID and No bookID then we will only display a single page blm\n if (!this.props.bookID) {\n this.props.updatePagesVisible(1);\n }\n });\n } else {\n this.loadBLM('', this.props.blmID).then(() => {\n this.props.setProjectReady(true);\n $('#pages').fadeTo('fast', 1.0);\n // if blmID and No bookID then we will only display a single page blm\n if (!this.props.bookID) {\n this.props.updatePagesVisible(1);\n }\n });\n }\n }\n if (\n this.props.viewerMode ===\n viewerModes.MODE_TEACHER_CLASS_NOTES ||\n this.props.viewerMode === viewerModes.MODE_TEACHER_GROUP_NOTES\n ) {\n initialViewerSettings.showDrawerHighlightsButton = false;\n }\n }\n\n initialViewerSettings.hideDrawer = this.shouldHideDrawer();\n\n this.updateViewerSettings(initialViewerSettings, () => {\n console.log('updated initial viewer settings');\n\n // Load Book?\n if (this.props.bookID && this.props.bookID.length) {\n if (this.props.book.ID !== this.props.bookID) {\n this.props.resetCachedBookPages();\n this.loadBook(this.props.bookID).then(() => {\n setTimeout(() => {\n this.props.setBookReady(true);\n // check if the left page is supposed to be even\n this.goToPage(this.props.currentPage, false, false);\n }, 100);\n $('#pages').fadeTo('fast', 1.0);\n });\n return;\n }\n this.loadBook(this.props.bookID).then(() => {\n setTimeout(() => {\n this.props.setBookReady(true);\n // check if the left page is supposed to be even\n this.goToPage(this.props.currentPage, false, false);\n }, 100);\n $('#pages').fadeTo('fast', 1.0);\n });\n }\n });\n\n // listen for active user\n // document.addEventListener(\n // 'visibilitychange',\n // this.handleVisibilityChange,\n // false\n // );\n // this.resetActiveTimeout();\n this.setupEventHandlers();\n }\n\n componentDidUpdate(prevProps) {\n if (prevProps.bookIsZooming !== this.props.bookIsZooming) {\n if (this.props.bookIsZooming) {\n this.props.startMoving();\n } else {\n this.props.startPointing(); // this is a little strange to zoom to normal and then it switches to pointer, but cant think of simple solution\n }\n }\n // if the pages visible changes we need to check if the left page is supposed to be even\n // unless the book or the blm is not ready yet\n if (\n prevProps.pagesVisible !== this.props.pagesVisible &&\n (this.props.blm.projectIsReady || this.props.book.bookIsReady)\n ) {\n this.goToPage(this.props.book.currentPage);\n }\n\n // TODO can we improve this flow by grouping the things to do when a book or blm loads here?\n // if (prevProps.blm.projectIsReady !== this.props.blm.projectIsReady && this.props.blm.projectIsReady){\n // $(\"#pages\").fadeTo(\"fast\", 1.0);\n // }\n // if (prevProps.book.bookIsReady !== this.props.book.bookIsReady && this.props.book.bookIsReady){\n // $(\"#pages\").fadeTo(\"fast\", 1.0);\n // }\n\n if (prevProps.book.ID && !this.props.book.ID) {\n console.log(\n 'book has been reset in bookView, redirecting to bookbag'\n );\n this.goToBag();\n }\n }\n\n componentWillUnmount() {\n setTimeout(() => {\n this.removeAllHighlights();\n }, 100);\n if (this.props.blms.length > 0) {\n this.props.emptyBLMs();\n }\n this.superDebouncedMarkupUpdate.cancel();\n this.debouncedMarkupUpdate.cancel();\n // document.removeEventListener(\n // 'visibilitychange',\n // this.handleVisibilityChange\n // );\n // if (this.activeTimeout) {\n // clearTimeout(this.activeTimeout);\n // }\n // if (this.activeWarningTimeout) {\n // clearTimeout(this.activeWarningTimeout);\n // }\n // if (this.handleTimeoutCloseBook) {\n // clearTimeout(this.handleTimeoutCloseBook);\n // }\n this.removeEventHandlers();\n }\n\n setupEventHandlers() {\n // listen for renewedAzureToken\n document.addEventListener(\n 'renewedAzureToken',\n this.handleRenewedAzureToken\n );\n document.addEventListener('keyup', this.handleKeyPress, false);\n }\n\n removeEventHandlers() {\n document.removeEventListener(\n 'renewedAzureToken',\n this.handleRenewedAzureToken\n );\n document.removeEventListener('keyup', this.handleKeyPress, false);\n }\n\n handleKeyPress = ({ key }) => {\n key && key === 'ArrowRight' && this.nextPage();\n key && key === 'ArrowLeft' && this.prevPage();\n };\n\n /*\n * shouldHideDrawer\n * don't show the left drawer in resource mode or when viewing an ePub\n */\n shouldHideDrawer = () => {\n return (\n this.props.viewerMode === viewerModes.MODE_RESOURCE ||\n this.props.book.IsEPub === true\n );\n };\n\n handleRenewedAzureToken = () => {\n console.log('handling renewed Azure token');\n\n // if( config.System === 'web' && this.props.isOnline){\n // this.props.resetCachedBookPages();\n // this.getBookHTMLPages(this.props.book, this.props.currentPage, true);\n // } else if (config.System !== 'web' && this.props.isOnline){\n\n if (this.props.isOnline) {\n // if on a device we need to re-download the entire book\n this.props.resetCachedBookPages();\n this.loadBook(this.props.bookID).then(() => {\n setTimeout(() => {\n this.props.setBookReady(true);\n // check if the left page is supposed to be even\n this.goToPage(this.props.currentPage);\n }, 100);\n $('#pages').fadeTo('fast', 1.0);\n });\n } else {\n toastr.error(\n `No internet connection.`,\n `Error Loading Book`,\n constants.toastrErrorOptions\n );\n }\n };\n\n // handleVisibilityChange = () => {\n // if (document.hidden) {\n // this.activeTimeout = setTimeout(\n // this.handleTimeoutWarning,\n // constants.closeBookTimeout\n // );\n // console.log('view hidden');\n // } else {\n // console.log('view visible');\n // clearTimeout(this.activeTimeout);\n // clearTimeout(this.activeWarningTimeout);\n // }\n // };\n\n // handleTimeoutWarning = () => {\n // toastr.warning(\n // `Still there?`,\n // `We will be closing the book in 30 seconds due to inactivity`,\n // {\n // ...constants.toastrWarningOptions,\n // id: 'countdownWarning',\n // timeOut: constants.closeBookWarning,\n // onCloseButtonClick: this.resetActiveTimeout\n // }\n // );\n // this.activeWarningTimeout = setTimeout(\n // this.handleTimeoutCloseBook,\n // constants.closeBookWarning\n // );\n // clearTimeout(this.activeTimeout);\n // };\n\n // handleTimeoutCloseBook = () => {\n // clearTimeout(this.activeWarningTimeout);\n // this.props.updateBookStatus(\n // this.props.book.ID,\n // this.props.book.currentPage,\n // this.props.book.pagecount,\n // true,\n // this.props.blmMode\n // );\n // this.closeBookView();\n // toastr.warning(\n // '',\n // \"We have closed the book due to inactivity. Don't worry, we put in a bookmark for ya.\",\n // { ...constants.toastrWarningOptions, timeOut: 0 }\n // );\n // };\n\n resetActiveTimeout = () => {\n console.log('resetting countdown timeout');\n // clearTimeout(this.activeTimeout);\n // clearTimeout(this.activeWarningTimeout);\n // this.activeTimeout = setTimeout(\n // this.handleTimeoutWarning,\n // constants.closeBookTimeout\n // );\n // // toastr.remove('countdownWarning') // not sure why this did not work.\n // toastr.removeByType('warning');\n };\n\n /*\n * function to call from child components to update viewerSettings\n * newViewerSettings = some subset of the viewer settings\n * object that you want to update.\n * callback - setState is asynchronus, so I am allowing an optional callback\n * in case we need to do something AFTER the update completes\n */\n updateViewerSettings = (newViewerSettings, callback) => {\n // let now = new Date().getTime();\n // console.error('UPDATING VIEWER SETTINGS: old, new', now ,this.state.viewerSettings, newViewerSettings );\n\n let viewerSettings = Object.assign(\n {},\n this.state.viewerSettings,\n newViewerSettings\n );\n return this.setState({ viewerSettings }, callback);\n };\n\n /*\n * Gets an array of books\n */\n initBooks = () => {\n if (this.props.books.length === 0) {\n // check if we are online\n if (!this.props.isOnline) {\n // if we do not have books and we are offline, we need to show an error\n console.warn('unable to load bookview while offline');\n toastr.error(\n `No internet connection.`,\n `Error Loading Book`,\n constants.toastrErrorOptions\n );\n return;\n }\n this.props.searchBookBagBooks();\n }\n };\n\n initBLMs = () => {\n console.log('initializing BLMs');\n if (!this.props.isOnline) {\n console.log('offline, skipping getAssigndBlms');\n return;\n }\n this.props\n .getAssignedBLMs(this.props.bookID, this.props.user)\n .then((blms) => {\n // Load BLM?\n if (this.props.projectAssignmentID && this.props.blmMode) {\n this.loadBLM(\n this.props.projectAssignmentID,\n this.props.blmID\n );\n\n // if blmID and No bookID then we will only display a single page blm\n // commented out for now as a student never views a blm by itself\n // if (!this.props.bookID) {\n // initialViewerSettings.pagesVisible = 1;\n // }\n }\n })\n .catch((error) => {\n console.error('Error searching for blm', error);\n if (this.props.isOnline) {\n toastr.error(\n 'Error searching for blm',\n `Error`,\n constants.toastrErrorOptions\n );\n }\n });\n };\n\n /*\n ***************************************** RANGY HIGHLIGHTER ***************************************************\n /*\n\n /*\n * create Rangy highlighter objects for the various markup types and colors\n */\n createHighlighter = () => {\n var newHL = rangy.createHighlighter(this.pagesRef.current, 'TextRange');\n newHL.addClassApplier(rangy.createClassApplier('hl-yellow'));\n newHL.addClassApplier(rangy.createClassApplier('hl-teal'));\n newHL.addClassApplier(rangy.createClassApplier('hl-green'));\n newHL.addClassApplier(rangy.createClassApplier('hl-orange'));\n newHL.addClassApplier(rangy.createClassApplier('hl-underline'));\n newHL.addClassApplier(rangy.createClassApplier('hl-strikethrough'));\n return newHL;\n };\n\n removeAllHighlights = () => {\n this.state.highlighterLeft.removeAllHighlights();\n this.state.highlighterRight.removeAllHighlights();\n };\n\n /*\n * pass in which side then make sure all selections are safe before saving them\n * we have to have two highlighters because this saves all the highlights for the highlighter.\n * this function is performance intensive and blocking\n */\n serializePageMarkups = (side) => {\n let self = this;\n let serializedText = this.state[`highlighter${side}`].serialize({\n serializeHighlightText: true\n });\n /* if we find a \" or a -webkit-transform throw an error */\n if (\n serializedText.indexOf('\"') !== -1 &&\n serializedText.indexOf('-webkit-transform') !== -1\n ) {\n console.error('unable to serialize markup', serializedText);\n alert('Error selecting text, please try again');\n self.state[`highlighter${side}`].unhighlightSelection();\n throw new Error('unable to serialize text');\n } else {\n return serializedText;\n }\n };\n\n /*\n ****************************************************** BOTTOM TOOLBAR *********************************************\n */\n startNotes = (e) => {\n this.resetActiveTimeout();\n if (e) {\n e.preventDefault();\n }\n if (\n !this.props.isOnline &&\n !UserAPI.isStudent(this.props.user.RoleID)\n ) {\n // if we do not have books and we are offline, we need to show an error\n console.warn('unable to create note while offline');\n toastr.error(\n `No internet connection.`,\n `Error Saving Note`,\n constants.toastrErrorOptions\n );\n return;\n }\n this.props.startNotes(true);\n };\n startHighlighter = (e, hlClass) => {\n this.resetActiveTimeout();\n // if this is a touch then we highlight what is selected then turn off the highlight mode.\n // as of 6/2018 we are no longer doing touch highlight mode\n // if (e.type === 'touchend'){\n // this.touchHighlight(e, hlClass);\n // } else {\n this.clickHighlight(e, hlClass);\n // }\n };\n\n // this function is being depricated\n touchHighlight = (e, hlClass) => {\n this.resetActiveTimeout();\n console.log('touch highlight');\n e.preventDefault();\n\n // make sure they selected some text first\n if (window.getSelection().type === 'None') {\n toastr.warning(\n `When using touch, please select text first.`,\n ``,\n constants.toastrWarningOptions\n );\n return;\n }\n this.doTouchHighlight(hlClass);\n };\n\n clickHighlight = (e, hlClass) => {\n this.resetActiveTimeout();\n console.log('started click highlighter ' + hlClass);\n this.props.startHighlighting(true, hlClass);\n };\n\n startEraser = () => {\n this.resetActiveTimeout();\n console.log('started eraser');\n this.props.startErasing(true);\n };\n\n startPointer = () => {\n this.resetActiveTimeout();\n console.log('started pointer');\n this.props.startPointing();\n };\n\n /*\n * Book Navigation (first, last, next, previous)\n */\n firstPage = (e) => {\n this.resetActiveTimeout();\n if (e) e.preventDefault();\n this.goToPage(1);\n document.dispatchEvent(new Event(BOOK_FIRST_PAGE));\n };\n\n lastPage = (e) => {\n this.resetActiveTimeout();\n if (e) e.preventDefault();\n this.goToPage(this.props.book.totalPage);\n document.dispatchEvent(new Event(BOOK_LAST_PAGE));\n // if (this.props.currentPage !== this.props.book.pagecount) {\n // const pi = this.props.book.pagecount - this.props.pagesVisible + 1;\n // this.goToPage(pi);\n // }\n };\n\n nextPage = (e) => {\n this.resetActiveTimeout();\n if (e) e.preventDefault();\n \n let pagesToChange = 1;\n \n if (!this.props.blmMode && this.props.pagesVisible === 2) {\n pagesToChange = 2;\n }\n \n if (this.props.currentPage >= this.props.totalPage) {\n return;\n }\n this.goToPage(this.props.currentPage + pagesToChange);\n document.dispatchEvent(new Event(BOOK_NEXT_PAGE));\n };\n\n prevPage = (e) => {\n this.resetActiveTimeout();\n if (e) e.preventDefault();\n let pagesToChange = 1;\n\n if (!this.props.blmMode && this.props.pagesVisible === 2) {\n pagesToChange = 2;\n }\n\n if (this.props.currentPage <= 0) {\n return;\n }\n this.goToPage(this.props.currentPage - pagesToChange);\n document.dispatchEvent(new Event(BOOK_PREV_PAGE));\n };\n\n /*\n ************************************************* CLICK SWIPE AND DRAG LISTENERS *****************************************\n */\n\n // TODO create a get relative position util function. accepts pageW, pageH, layerX, layerY, offsetX, offsetY, scaleX, scaleY then returns offsetX and offsetY\n\n /*\n * respond to page taps.\n */\n pagesTapped = (e, pageNumber) => {\n this.resetActiveTimeout();\n // console.log('page tapped: ', e.type);\n if (\n e.target.className.includes('tappable') ||\n e.target.parentNode.className.includes('tappable')\n ) {\n this.wordTapped(e, pageNumber);\n }\n\n // on mobile Safari we use the e.originalEvent.layerX (with react e.nativeEvent.layerX) from the touch event, but on Chrome, touch event's don't have this.\n // we don't need it on desktop browsers because they will trigger a click event right after the touch event.\n if (e.type === 'touchend' && !e.nativeEvent.layerX) {\n return;\n }\n };\n\n updateMarkups = (whichSide, pid) => {\n this.updateHighlights(this.serializePageMarkups(whichSide), pid);\n rangy.getSelection().removeAllRanges(); // this only slightly helps to be debounced and we call it unnecessarily when erasing\n this.props.updatePendingItem(false);\n };\n\n /*\n * respond to word taps\n */\n wordTapped = (e, pageNumber) => {\n this.resetActiveTimeout();\n // console.log('word tapped', e.type);\n // if (config.System === 'mobile'){\n // this.props.manualAjaxStart();\n // }\n // console.log($(e.currentTarget).text(), e);\n // console.log(this.state.highlighter);\n // console.log($('.pages .left').attr('id'));\n let erasing = this.props.bookToolbar.erasing;\n const pid = `page${pageNumber}`;\n const whichSide =\n this.props.book.currentPage === pageNumber ? 'Left' : 'Right';\n\n if (this.props.bookToolbar.highlighting) {\n /*\n * Before we added user-select: all to the .t class we had to manually highlight the\n * characters before applying the highlights with the code below:\n */\n // let sel = rangy.getSelection();\n // sel.selectCharacters(e.target, 0, e.target.textContent.length);\n\n /*\n * find similar markups and merge them in order to help iOS users tap to highlight\n */\n findSimilarMarkups(\n e.target,\n whichSide,\n pid,\n this.props.bookToolbar.highlightColor\n );\n\n // console.log('word tapped, highlighting selection', e.target.textContent.length, sel);\n this.state[`highlighter${whichSide}`].highlightSelection(\n this.props.bookToolbar.highlightColor,\n {\n containerElementId: pid\n }\n );\n\n // this is blocking rendering the view, so we wait for view to show highlight before serializing\n // on iOS we need a much longer delay\n this.props.updatePendingItem(true);\n // if (config.System === 'mobile'){\n // this.superDebouncedMarkupUpdate(whichSide, pid);\n // } else {\n this.debouncedMarkupUpdate(whichSide, pid);\n // }\n } else if (this.props.bookToolbar.erasing) {\n /*\n * since a word was tapped, we must manually\n * highlight the characters before applying highlights. only if it is not Mobile\n * we do not support highlighting a single word on mobile. when on mobile we are erasing the entire selection.\n */\n // var sel = rangy.getSelection();\n // sel.selectCharacters(e.target, 0, e.target.textContent.length);\n this.state[`highlighter${whichSide}`].unhighlightSelection();\n\n // this is blocking rendering the view, so we wait for view to show highlight before serializing\n // on iOS we need a much longer delay\n this.props.updatePendingItem(true);\n // if (config.System === 'mobile'){\n // this.superDebouncedMarkupUpdate(whichSide, pid);\n // } else {\n this.debouncedMarkupUpdate(whichSide, pid);\n // }\n\n // turn off erasing\n erasing = false;\n }\n this.updateViewerSettings({ highlighting: false }, () => {\n // if it was a touch event and we just erased something then automatically turn off erasing\n // commented out while we are only doing single tap markups\n if (e.type === 'touchend' && !erasing) {\n // this.startPointer();\n }\n });\n this.props.manualAjaxEnd();\n };\n\n /*\n * respond to mouse down\n */\n onMouseDown = (e, pageNumber) => {\n this.updateViewerSettings({\n highlightingMouseDown: true,\n highlightingPageNumber: pageNumber\n });\n this.mouseMove = 0;\n };\n\n /*\n * respond to mouse up\n */\n onMouseUp = (e, pageNumber) => {\n const pid = `page${pageNumber}`;\n const whichSide =\n this.props.book.currentPage === pageNumber ? 'Left' : 'Right';\n var sel = rangy.getSelection();\n if (\n this.props.bookToolbar.highlighting &&\n this.state.viewerSettings.highlightingMouseDown &&\n this.mouseMove > 1\n ) {\n this.state[`highlighter${whichSide}`].highlightSelection(\n this.props.bookToolbar.highlightColor,\n {\n containerElementId: pid\n }\n );\n this.updateHighlights(this.serializePageMarkups(whichSide), pid);\n // remove the selection from view\n sel.removeAllRanges();\n } else if (\n this.props.bookToolbar.erasing &&\n this.state.viewerSettings.highlightingMouseDown &&\n this.mouseMove > 5\n ) {\n this.state[`highlighter${whichSide}`].unhighlightSelection();\n this.updateHighlights(this.serializePageMarkups(whichSide), pid);\n }\n this.updateViewerSettings({ highlightingMouseDown: false });\n this.mouseMove = 0;\n };\n\n onMouseMove = (e) => {\n this.mouseMove++;\n };\n\n /*\n * doTouchHighlight highlights the text after a touch devices has highlighted a piece of text and then chosen the highlight color\n */\n doTouchHighlight = (highlightClass) => {\n // get the current selection\n var sel = rangy.getSelection();\n const pid = `page${this.state.viewerSettings.highlightingPageNumber}`;\n const whichSide =\n this.props.book.currentPage ===\n this.state.viewerSettings.highlightingPageNumber\n ? 'Left'\n : 'Right';\n this.state[`highlighter${whichSide}`].highlightSelection(\n highlightClass,\n {\n containerElementId: pid\n }\n );\n this.updateHighlights(this.serializePageMarkups(whichSide), pid);\n // remove the selection from view\n sel.removeAllRanges();\n this.startPointer();\n\n // workaround the iOS bug that does not remove the highlight automatically\n $('.pages .left').click();\n };\n /*\n * save the markups to the server by sending all the highlights and the page id (pid) example: \"page13\"\n */\n updateHighlights = (highlight, pid) => {\n const newHighlights = Object.assign(\n {},\n this.props.book.highlights.Content,\n { [pid]: highlight }\n );\n const bookData = Object.assign(\n {},\n {\n Type: 2,\n Content: newHighlights,\n bookID: this.props.book.ID,\n LastDownloadedFromServer:\n this.props.book.highlights.LastDownloadedFromServer\n }\n );\n this.props\n .saveBookItem(\n bookData,\n this.props.user,\n this.props.viewerMode,\n this.props.book.ID\n )\n .then((bi) => {\n console.log('saved book item');\n })\n .catch((error) => {\n console.error('Error saving book item', error);\n toastr.error(\n 'Unable to save markup. Please try again or contact support.',\n `Error Saving`,\n constants.toastrErrorOptions\n );\n });\n };\n\n /*\n ************************************************* DRAWER *******************************************\n */\n goToNote = (note) => {\n this.resetActiveTimeout();\n const noteID = note.ID || note.TempID;\n this.goToPage(note.Page, () => {\n // give the page time to fade in\n setTimeout(() => {\n this.props.setActiveNote(noteID);\n }, 200);\n });\n };\n\n openDrawer = (drawerType = viewerDrawerTypeEnum.notes) => {\n this.resetActiveTimeout();\n const showDrawer = !this.state.viewerSettings.showDrawer;\n if (drawerType !== this.state.viewerSettings.drawerType) {\n this.updateViewerSettings({\n showDrawer: true,\n drawerType: drawerType\n });\n } else {\n this.updateViewerSettings({\n showDrawer: showDrawer,\n drawerType: drawerType\n });\n }\n };\n closeDrawer = () => {\n this.resetActiveTimeout();\n //close the drawer only if it is open this way we do not call updateViewerSettings too often\n if (this.state.viewerSettings.showDrawer) {\n this.updateViewerSettings({ showDrawer: false });\n }\n };\n\n /*\n ****************************************************** HEADER *************************************************\n */\n\n toggleHeader = () => {\n this.resetActiveTimeout();\n const self = this;\n if (this.state.viewerSettings.header.visible) {\n // TODO refactor to CSS animations\n $('.page-header').animate({ top: '-60px' }, 500);\n $('.toggle').animate({ top: '-25px' }, 500);\n self.updateViewerSettings({\n header: { visible: false, class: 'fa fa-angle-double-down' }\n });\n } else {\n $('.page-header').animate({ top: '0px' }, 500);\n $('.toggle').animate({ top: '35px' }, 500);\n self.updateViewerSettings({\n header: { visible: true, class: 'fa fa-angle-double-up' }\n });\n }\n };\n\n /*\n * toggle BLM mode\n */\n\n toggleBLMMode = (blmMode = false) => {\n this.resetActiveTimeout();\n if (this.props.itemSavePending && blmMode === false) {\n this.setState({\n showSaveProjectConfirm: true,\n saveProjectSubmit: () => {\n removeQuery('blmID');\n removeQuery('projectAssignmentID');\n this.props.updatePendingItem(false);\n }\n });\n } else {\n this.props.updatePendingItem(false);\n if (blmMode === false) {\n removeQuery('blmID');\n removeQuery('projectAssignmentID');\n }\n }\n };\n\n closeBookView = () => {\n // students or .... are able to go to the bookbag, everyone else is logged out\n if (this.props.itemSavePending) {\n this.setState({\n showSaveProjectConfirm: true,\n saveProjectSubmit: () => {\n this.props.updatePendingItem(false);\n this.closeBookView();\n this.setState({ showSaveProjectConfirm: false });\n }\n });\n return;\n }\n this.props.resetBookView();\n this.props.startPointing();\n this.props.setBookReady(false);\n this.props.setProjectReady(false);\n $('#pages').fadeTo('fast', 0.0);\n\n if (\n UserAPI.isStudent(this.props.user.RoleID) ||\n UserAPI.isGeneric(this.props.user.RoleID)\n ) {\n this.goToBag();\n } else {\n this.props.userLogout();\n hashHistory.replace('/');\n }\n };\n\n goToBag = () => {\n // TODO @jfbloom22 why do we empty blms?\n if (!this.props.itemSavePending) {\n this.props.emptyBLMs();\n }\n // let the API know user closed the book by passing -1 in as the page index\n this.props\n .updateBookStatus(\n this.props.book.ID,\n -1,\n this.props.book.pagecount,\n false,\n this.props.blmMode\n )\n .then((resp) => {\n console.log('updated book status - close book');\n })\n .catch((err) => {\n console.error('Error updating book status - close book', err);\n // I don't think we want to show an error if this does not work\n // toastr.error(`Please try again or contact support`, `Error Changing Page`, {\n // closeButton: true,\n // showAnimation: 'animated fadeInDown'\n // });\n });\n\n hashHistory.push({\n pathname: '/bag',\n query: this.props.location.query\n });\n };\n /*\n *\n * update the URL, close the drawer, change to the most recent book page\n * TODO going to need to do this for BLM, they will grab a BLM id\n */\n changeBook = (book) => {\n this.resetActiveTimeout();\n if (this.props.itemSavePending) {\n this.setState({\n showSaveProjectConfirm: true,\n saveProjectSubmit: () => {\n this.changeBook(book);\n this.setState({ showSaveProjectConfirm: false });\n this.props.updatePendingItem(false);\n }\n });\n return;\n }\n if (book.IsExternal) {\n // open the book in the Lerner viewer, not in the DiBS viewer\n const externalURL = `${config.API.Main}/book/openexternalviewer?bookID=${book.ID}`;\n var win = window.open(externalURL, '_blank');\n win.focus();\n return;\n }\n this.props.setProjectReady(false);\n this.props.setBookReady(false);\n this.setState({ pages: {} }, () => {\n const query = { ...this.props.location.query, bookID: book.ID };\n hashHistory.replace({\n pathname: `/viewer`,\n query: query\n });\n this.closeDrawer();\n this.loadBook(book.ID).then(() => {\n this.toggleBLMMode(false);\n setTimeout(() => {\n this.props.setBookReady(true);\n }, 100);\n $('#pages').fadeTo('fast', 1.0);\n this.initBLMs();\n });\n });\n };\n\n changeBlm = (projectAssignmentID, blmID) => {\n this.resetActiveTimeout();\n this.closeDrawer();\n this.props.setProjectReady(false);\n this.loadBLM(projectAssignmentID, blmID).then(() => {\n this.props.setProjectReady(true);\n }); // TODO this will get removed when we add some listeners in did receive props\n };\n\n /*\n **************************************************** GENERAL VIEWER FUNCTIONS *******************************************\n */\n\n fadeOut = (duration, cb) => {\n $('#pages').fadeTo(duration, 0.01, cb);\n // TODO verify that we do not need to handle fading out and in better.\n setTimeout(() => {\n $('#pages').fadeTo('fast', 1.0);\n }, 300);\n };\n /*\n * to support images that span two pages, we make sure we are displaying even or odd pages on the left\n * if left page is supposed to be even, if viewing two pages, if not BLM mode, if pi is odd then subtract one\n * if left page is supposed to be odd, and pi is not odd, then subtract one\n */\n checkLeftPageEven = (pi) => {\n function isOdd(num) {\n return num % 2;\n }\n if (\n this.props.book.LeftPageEven &&\n this.props.pagesVisible === 2 &&\n !this.props.blmMode &&\n isOdd(pi)\n ) {\n return pi - 1;\n } else if (\n !this.props.book.LeftPageEven &&\n this.props.pagesVisible === 2 &&\n !this.props.blmMode &&\n !isOdd(pi)\n ) {\n return pi - 1;\n } else {\n return pi;\n }\n };\n\n checkIfPageExists = (pi) => {\n // if the user tries to view a page and has a letter in the search, it will reject the search\n let re = /^\\d+$/;\n if (!re.test(pi)) {\n return false;\n }\n // if it exceeds the page count\n // if it is 0 and we are viewing a single page or in blm mode\n // if it is -1\n if (\n pi > this.props.book.pagecount ||\n (pi <= 0 &&\n (this.props.pagesVisible === 1 || this.props.blmMode)) ||\n pi < 0\n ) {\n return false;\n }\n return true;\n };\n\n /*\n * go to a specific page\n */\n goToPage = (pi, cb, shouldCheckLeftPageEven = true) => {\n this.resetActiveTimeout();\n // only check the left page even if it is not the the initial load of the book\n if (shouldCheckLeftPageEven) {\n pi = this.checkLeftPageEven(pi); // make sure the correct page is displaying on the left\n }\n\n if (!this.checkIfPageExists(pi)) {\n if (!pi || (pi && pi === 0 && this.props.pagesVisible === 1)) {\n pi = 1;\n } else {\n toastr.error(\n `Page does not exist`,\n `The page you are looking for does not exist.`,\n constants.toastrErrorOptions\n );\n return;\n }\n }\n\n if (pi === this.props.currentPage)\n if (!!cb) {\n return cb();\n } else {\n return;\n }\n // close any notes that happen to be open\n this.props.setActiveNote('');\n if (!!this.state.highlighterRight) {\n this.state.highlighterRight.removeAllHighlights();\n }\n if (!!this.state.highlighterLeft) {\n this.state.highlighterLeft.removeAllHighlights();\n }\n const self = this;\n self.fadeOut(200, function () {\n // update the current page number that we are on\n self.props.updateCurrentPage(pi, self.props.book.ID);\n self.props\n .updateBookStatus(\n self.props.book.ID,\n pi,\n self.props.book.pagecount,\n false,\n self.props.blmMode\n )\n .catch((err) => {\n console.warn('Error updating book status', err);\n });\n // sometimes we go to page 0 in order to display an even page on the left side, don't try to get the html\n if (pi !== 0) {\n self.getBookHTMLPages(self.props.book, pi, true);\n } else if (\n pi === 0 &&\n self.props.pagesVisible === 2 &&\n !self.props.blmMode\n ) {\n self.getBookHTMLPages(self.props.book, 1, true);\n }\n\n if (!!cb) {\n window.setTimeout(() => {\n return cb();\n }, 750);\n }\n });\n };\n\n /*\n * Load the selected book\n * step 1 of 5 in loading a book page\n */\n loadBook = (bookID) => {\n $('#pages').fadeTo('fast', 0.0);\n return this.props\n .getBookByID(bookID, this.props.user)\n .then((book) => {\n console.log(\n `opening book: ISBN: ${book.ISBN}, Title: ${book.Title}, ID: ${book.ID}`\n );\n if (book.IsEPub){\n const httpPrefix = config.Debug ? 'http://' : 'https://';\n this.setState({EPubUrl: `${httpPrefix}${window.location.host}${book.EBookPath}/${book.ISBN}`})\n return Promise.resolve(true);\n }\n return this.getBookParts(book);\n })\n .catch((error) => {\n console.error('error loading book', error);\n let message = `We encountered an error loading the book.`;\n if (error.message) {\n message = error.message;\n }\n this.closeBookView();\n if (error.status === 403) {\n this.handleInvalidSession();\n return;\n } else {\n toastr.error(\n `Error`,\n message,\n constants.toastrErrorOptions\n );\n }\n });\n };\n\n /*\n * step 2 of 5 get the book\n * a book has 3 parts, the properties.json file, HTML pages, and book items\n */\n getBookParts = (book) => {\n let data = {};\n // start step 3\n return this.props\n .getBookProperties(book, this.props.user.AzureToken) // get book properties.json (part 1 of 3)\n .then((bookObj) => {\n // bookReducer UPDATE_BOOK_STATUS_SUCCESS deals with this logic, don't return this since we don't care if it fails\n this.props\n .updateBookStatus(\n bookObj.ID,\n 0,\n bookObj.pagecount,\n false,\n this.props.blmMode\n )\n .then((resp) => {})\n .catch((err) => {\n console.error('Error updating book status', err);\n // do not display an error if updating the status fails\n });\n // we are returning the book obj along with the properties\n // start step 4a\n this.getBookHTMLPages(bookObj, this.props.currentPage, true); // get book html pages (part 2 of 3)\n\n // start step 4b - get student book items if user is a teacher viewing student book\n // MODE: teacher token to view a students items and pull up teacher items with that book\n // action: getTeacherBookItems shows the teacher the items they have made\n // teachers will not have downloaded books as they load in with a token (requiring internet support)\n if (\n this.props.viewerMode ===\n viewerModes.MODE_TEACHER_VIEW_STUDENT_BOOK\n ) {\n Object.assign(data, {\n book: book.ID,\n classID: this.props.location.query.classID,\n groupID: this.props.location.query.groupID,\n studentID: this.props.location.query.studentID\n });\n let promises = [];\n promises.push(\n this.props.getBookItemsByStudentID(\n book.ID,\n this.props.location.query.studentID,\n this.props.user\n )\n );\n promises.push(\n this.props.getTeacherBookItems(\n data,\n this.props.user,\n book.ID\n )\n );\n return Promise.all(promises);\n\n // MODE: teacher token to view a group or class and pull up their teacher items with that class or group\n } else if (\n this.props.viewerMode ===\n viewerModes.MODE_TEACHER_GROUP_NOTES ||\n this.props.viewerMode ===\n viewerModes.MODE_TEACHER_CLASS_NOTES\n ) {\n Object.assign(data, {\n book: book.ID,\n classID: this.props.location.query.classID,\n groupID: this.props.location.query.groupID\n });\n return this.props.getTeacherBookItems(\n data,\n this.props.user,\n book.ID\n );\n\n // MODE: teacher has a token to view a students blm, also grab items for the student\n } else if (\n this.props.viewerMode ===\n viewerModes.MODE_TEACHER_STUDENT_BLM\n ) {\n Object.assign(data, {\n book: book.ID,\n classID: this.props.location.query.classID,\n groupID: this.props.location.query.groupID,\n StudentID: this.props.location.query.studentID\n });\n let promises = [];\n promises.push(\n this.props.getBookItemsByStudentID(\n book.ID,\n this.props.location.query.studentID,\n this.props.user\n )\n );\n promises.push(\n this.props.getTeacherBookItems(\n data,\n this.props.user,\n book.ID\n )\n );\n return Promise.all(promises);\n } else if (UserAPI.isStudent(this.props.user.RoleID)) {\n // TODO 2/16/18 determine this is online\n // start step 4c - get the book items and set them in the redux store\n // action: getBookItems shows the student the items they have made in the book\n return this.props\n .getBookItems(\n book.ID,\n this.props.user,\n this.props.downloadedBooks\n )\n .then(() => {\n // if offline and there are no highlights and no notes - then show an error. TODO add support for doing markups in multiple books while offline\n // if (!this.props.isOnline && !this.props.book.highlights.Content && this.state.notes.length === 0 ){\n // this.updateViewerSettings({showToolbar: false});\n // throw {message: 'Please connect to the internet and re-open this book before doing any markups.'};\n // }\n }); // get book items (part 3 of 3)\n // TODO 2/16/18 create another if statment determining if the user is a student and we are offline -->\n // Load from downloaded books in redux where we have book items\n } else if (UserAPI.isGeneric(this.props.user.RoleID)) {\n // console.log('not getting book items when a generic user.')\n return Promise.resolve(true);\n } else if (this.props.viewerMode === viewerModes.MODE_GENERIC) {\n // console.log('not getting book items when in generic mode.')\n return Promise.resolve(true);\n } else if (\n this.props.viewerMode === viewerModes.MODE_RESOURCE\n ) {\n console.log('resource mode');\n return Promise.resolve(true);\n } else {\n return Promise.reject({\n message: `not getting book items for this user: ${this.props.user.LoginID}`\n });\n }\n });\n };\n\n /*\n * step 4a grab the HTML assets for the page passed in (left page) plus the nextpage (right page)\n * then wait 1.5 seconds and cache the next 2 pages, wait 2 seconds and cache the previous 2 pages\n *\n */\n getBookHTMLPages = (book, pi, cache) => {\n if (book.IsEPub) {\n return;\n }\n const lIndex = pi; // left page index\n const lpageKey = `page${lIndex}`; // left page key\n const rIndex = lIndex + 1; // right page index we manually add 1 to get the right pi\n const rpageKey = `page${rIndex}`; // right page key\n const pageIndexToCache = lIndex + this.props.pagesVisible;\n const previousPageIndexToCache = lIndex - this.props.pagesVisible;\n const pageIndexToCacheTwo = lIndex + this.props.pagesVisible * 2;\n\n // if left page does not already exist in state, then grab it from the server\n if (!this.props.book.cachedPages[lpageKey]) {\n this.getBookHTMLPage(lpageKey, book, lIndex);\n }\n // if right page does not already exist in state and the right page index is less then or equal to the total pages in the book\n // then grab the book page for the right side\n // TODO how does this work when we are in single page view?\n if (\n !this.props.book.cachedPages[rpageKey] &&\n rIndex <= book.pagecount\n ) {\n this.getBookHTMLPage(rpageKey, book, rIndex);\n }\n\n /*\n * should we cache and is the page index less than the total pages in the book\n * if yes and yes then cache them\n */\n if (cache && lIndex < book.pagecount - 1) {\n setTimeout(() => {\n this.getBookHTMLPages(this.props.book, pageIndexToCache, false);\n }, 1500);\n }\n\n /*\n * should we cache and is the page index more than 2\n * this caches the previous two pages\n */\n if (cache && lIndex > 2) {\n setTimeout(() => {\n this.getBookHTMLPages(\n this.props.book,\n previousPageIndexToCache,\n false\n );\n }, 2500);\n }\n\n /*\n * should we cache and is the page index less than 4 pages ahead of the current index and less than the total pages in the book\n * this caches up to 4 pages ahead of the current viewer page index\n */\n if (\n cache &&\n lIndex - this.props.currentPage < 4 &&\n pageIndexToCacheTwo <= book.pagecount - 1\n ) {\n setTimeout(() => {\n this.getBookHTMLPages(\n this.props.book,\n pageIndexToCacheTwo,\n false\n );\n }, 2000);\n }\n };\n\n /*\n * Get the assets for a single book HTML page and store them in cachedPages\n */\n getBookHTMLPage = (pageKey, bookObj, index) => {\n if (index === 0) {\n this.props.cacheBookPage({\n pageKey,\n pageHTML: '
',\n pageNumber: 0,\n width: bookObj.bounds[0][0],\n height: bookObj.bounds[0][1]\n });\n return;\n }\n // since the book pages are so temporary we do not store them in Redux\n return this.props\n .getSpeechMarks(bookObj, index, this.props.user.AzureToken, pageKey)\n .then(() => {\n return this.props\n .getBookPage(\n this.props.book,\n index,\n this.props.user.AzureToken,\n pageKey\n )\n .then((pageHTML) => {\n // console.info('retrieved book page', pageData);\n if (!pageHTML) {\n console.error('failed to process page', pageHTML);\n throw new Error({\n message: 'failed to process page'\n });\n }\n this.props.cacheBookPage({\n pageKey,\n pageHTML,\n pageNumber: index,\n width: bookObj.bounds[index - 1][0],\n height: bookObj.bounds[index - 1][1]\n });\n });\n })\n .catch((error) => {\n // check if we are online\n if (!this.props.isOnline) {\n // if we do not have books and we are offline, we need to show an error\n console.warn('unable to load page while offline');\n toastr.error(\n `No internet connection.`,\n `Error Loading Book Page`,\n constants.toastrErrorOptions\n );\n return;\n }\n console.error('Error trying to get book page', error);\n if (error.status === 403) {\n this.handleInvalidSession();\n } else {\n toastr.error(\n `We encountered an error getting the book page.`,\n `Error Getting Page`,\n constants.toastrErrorOptions\n );\n }\n });\n };\n\n loadBLM = (projectAssignmentID, blmID) => {\n // if the projectAssignmentID is empty, the current object is the blmID or vice versa\n // For example: blmCreator does not have a projectAssignmentID, only a blmID\n // A student does not use a blmID only a projectAssignmentID\n let currentObjectID;\n if (this.props.itemSavePending) {\n this.setState({\n showSaveProjectConfirm: true,\n saveProjectSubmit: () => {\n this.loadBLM(projectAssignmentID, blmID);\n removeQuery('blmID');\n this.setState({ showSaveProjectConfirm: false });\n this.props.updatePendingItem(false);\n }\n });\n return Promise.reject({ message: 'not saved' });\n }\n\n const query = {\n ...this.props.location.query,\n blmID,\n projectAssignmentID\n };\n hashHistory.replace({\n pathname: `/viewer`,\n query: query\n });\n\n // projectAssignmentID is empty here for a creator\n if (projectAssignmentID === '') {\n currentObjectID = blmID;\n } else {\n currentObjectID = projectAssignmentID;\n }\n return this.getBLMObject(currentObjectID)\n .then((blm) => {\n // need to not fade out if we have already loaded the blm\n if (\n this.props.projectAssignmentID &&\n this.props.blm.projectAssignmentID !==\n blm.projectAssignmentID\n ) {\n this.fadeOut(600, () => {});\n return this.getBLMParts(blm);\n } else {\n return this.getBLMParts(blm);\n }\n })\n .catch((err) => {\n console.error('unable to load BLM', err);\n toastr.error(\n ``,\n `Error Loading BLM`,\n constants.toastrErrorOptions\n );\n throw err;\n });\n };\n\n getBLMObject = (projectAssignmentID) => {\n let blm = this.props.blm;\n\n // if the book is already in redux as the active book (this happens after token login and page refresh)\n // For a student, we check to see if projectAssignmentID equals the paID on the blm\n // For an admin making a blm, we do not have a projectAssignmentID, so we pass a blmID\n // For a teacher, we do not have a projectAssignmentID on the blm but we have it in props\n if (this.props.viewerMode === viewerModes.MODE_TEACHER_STUDENT_BLM) {\n if (this.props.projectAssignmentID === projectAssignmentID) {\n return Promise.resolve(blm);\n }\n } else {\n if (\n blm.projectAssignmentID === projectAssignmentID ||\n blm.ID === projectAssignmentID\n ) {\n return Promise.resolve(blm);\n }\n }\n\n blm = this.props.blms.filter((b) => {\n return b.projectAssignmentID === projectAssignmentID;\n })[0];\n if (blm) {\n return Promise.resolve(blm);\n } else {\n // console.error('unable to find the project in the array of projects');\n // return reject(false);\n // TODO enable downloaded BLMs ????\n // if we are on mobile or chrome then try to find it in the array of downloaded blms\n // if (config.System === 'chrome' || config.System === 'mobile'){\n // blm = this.props.downloadedBlms.filter((b) => { return b.ID === blmID })[0];\n // if (blm){\n // console.log('have blm in downloadedBooks');\n // resolve(blm);\n // }\n }\n if (this.props.isOnline) {\n // we are online so grab it from the API\n return this.props\n .getBLMByID(this.props.blmID, this.props.user)\n .then((result) => {\n // this.loadBook(bookSaved);\n // book = result;\n return result;\n });\n } else {\n return Promise.reject(false);\n }\n };\n\n getBLMParts = (blm) => {\n return this.props\n .getBLMProperties(blm, this.props.user.AzureToken) // step 1, get blm properties\n .then((blmObj) => {\n if (!this.props.blmMode) {\n this.toggleBLMMode(true);\n }\n this.getBlmHtml(blmObj); // step 2, get blm html\n\n if (UserAPI.isStudent(this.props.user.RoleID)) {\n return this.props\n .getBLMStatus(blm, this.props.user)\n .then(() => {\n return this.props.getTeachersComment(\n blm.projectAssignmentID,\n blm.TeacherID,\n this.props.user\n );\n })\n .then(() => {\n return this.props.getBLMItems(\n blmObj.ID,\n blm.projectAssignmentID,\n this.props.user\n );\n })\n .then(() => {\n this.updateViewerSettings({\n loadedProjectAssignmentID:\n blm.projectAssignmentID\n });\n });\n }\n\n // MODE: teacher has token to view a students blm, this call grabs the students blm items\n if (\n this.props.viewerMode ===\n viewerModes.MODE_TEACHER_STUDENT_BLM\n ) {\n return this.props\n .getStudentsBLMItems(\n blm.ID,\n this.props.projectAssignmentID,\n this.props.location.query.studentID,\n this.props.user\n )\n .then(() => {\n this.updateViewerSettings({\n loadedProjectAssignmentID:\n blm.projectAssignmentID\n });\n });\n\n // MODE: teacher has token to create a blm template. They find any items they have previously made with the template\n } else if (\n this.props.viewerMode ===\n viewerModes.MODE_ADMIN_CREATE_TEMPLATE\n ) {\n return this.props\n .getTemplateItems(blmObj.ID, this.props.user)\n .then(() => {\n this.updateViewerSettings({\n loadedProjectAssignmentID:\n blm.projectAssignmentID\n });\n });\n }\n })\n .catch((error) => {\n console.error('Error getting BLM', error);\n if (error.status === 403) {\n this.handleInvalidSession();\n } else {\n toastr.error(\n `We encountered an error loading the BLM.`,\n `Error Loading BLM`,\n constants.toastrErrorOptions\n );\n }\n });\n };\n handleInvalidSession = () => {\n toastr.warning(\n `Trying to fix an issue with the book.`,\n `Please wait a moment`,\n constants.toastrWarningOptions\n );\n };\n\n getBlmHtml = (blm) => {\n // console.log(blm, 'BLM getting html');\n // passing in blm, 1 (because there is always only 1 page of the BLM, and the azure token)\n BlmAPI.getBlmPageRemote(blm, 1, this.props.user.AzureToken)\n .then((blmPage) => {\n this.setState({ blmHtml: blmPage });\n })\n .catch((error) => {\n // check if we are online\n if (!this.props.isOnline) {\n // if we do not have books and we are offline, we need to show an error\n console.warn('unable to load BLM while offline');\n toastr.error(\n `No internet connection.`,\n `Error Loading BLM`,\n constants.toastrErrorOptions\n );\n return;\n }\n console.error('Error trying to get BLM', error);\n if (error.status === 403) {\n this.handleInvalidSession();\n } else {\n toastr.error(\n `We encountered an error getting the BLM.`,\n `Error Getting BLM`,\n constants.toastrErrorOptions\n );\n }\n });\n };\n\n render() {\n if (requireSignIn(this.props.user, this.props.location)) {\n return null;\n }\n let books;\n if (this.props.isOnline) {\n books = this.props.books;\n } else {\n books = this.props.downloadedBooks;\n }\n return (\n \n {this.props.book.IsEPub && this.state.EPubUrl && (\n
\n \n console.log(epubcifi)\n }\n tocChanged={(toc) => console.log(toc)}\n closeBookView={this.closeBookView}\n // audioChanged={this.props.updateBookAudioUrl}\n />\n
\n )}\n {this.props.book.IsEPub === false && (\n
\n )}\n
\n {this.state.viewerSettings.showHeader && (\n \n )}\n\n \n \n
\n
\n
{\n this.setState({ showSaveProjectConfirm: false });\n }}\n submit={this.state.saveProjectSubmit}\n />\n \n );\n }\n}\n\nconst mapStateToProps = (state, ownProps) => {\n return {\n user: state.user,\n highlights: state.highlights,\n book: state.book,\n books: state.books,\n completedBLMs: state.completedBLMs,\n blms: state.blms,\n blm: state.blm,\n blmProperties: state.blmProperties,\n isOnline: state.offlineQueue.isOnline && navigator.onLine,\n loading: state.ajaxCallsInProgress > 0,\n downloadedBooks: state.downloadedBooks,\n currentPage: state.book.currentPage,\n totalPage: state.book.pagecount,\n leftPageContainerWidth: state.bookView.leftPageContainerWidth,\n leftPageContainerHeight: state.bookView.leftPageContainerHeight,\n bookScalePercent: state.bookView.bookScalePercent,\n projectScalePercent: state.bookView.projectScalePercent,\n blmMode: !!(\n ownProps.location.query.blmID &&\n ownProps.location.query.blmID.length > 0\n ),\n pagesVisible: state.bookView.pagesVisible,\n bookID: ownProps.location.query.bookID,\n viewerMode: ownProps.location.query.viewerMode,\n projectAssignmentID: ownProps.location.query.projectAssignmentID,\n blmID: ownProps.location.query.blmID,\n bookToolbar: state.bookToolbar,\n bookIsZooming: state.bookView.bookManualZoomLevel !== 1,\n itemSavePending: state.bookView.itemSavePending\n };\n};\n\nexport default connect(mapStateToProps, {\n userLogout,\n getBookProperties,\n getBookItems,\n getBookItemsByStudentID,\n getTeacherBookItems,\n getBookPage,\n getSpeechMarks,\n searchBookBagBooks,\n saveBookItem,\n deleteBookItem,\n updateBookStatus,\n getAssignedBLMs,\n getBLMProperties,\n getBLMStatus,\n getTeachersComment,\n emptyBLMs,\n getBLMItems,\n getStudentsBLMItems,\n getTemplateItems,\n getBookByID,\n getBLMByID,\n manualAjaxStart,\n manualAjaxEnd,\n updateCurrentPage,\n nextPage,\n prevPage,\n cacheBookPage,\n updateLeftPageContainer,\n updateBookScalePercent,\n updateProjectScalePercent,\n updatePagesVisible,\n resetBookView,\n automaticUpdatePagesVisible,\n setBookReady,\n setProjectReady,\n startNotes,\n startPointing,\n startHighlighting,\n startErasing,\n startUnderlining,\n startStriking,\n setActiveNote,\n increaseBookZoom,\n decreaseBookZoom,\n startMoving,\n downloadSpeech,\n updatePendingItem,\n resetCachedBookPages,\n})(withRouter(BookView));\n","import * as React from 'react';\nimport { FormGroup, FormControl, Row, Col } from 'react-bootstrap';\n\nclass UserProfileForm extends React.Component {\n render() {\n return (\n \n \n {this.props.forms.map((form) => {\n return (\n \n \n \n );\n })}\n \n \n Need to update your password?\n \n
\n \n Enter the new one twice above, otherwise leave it\n blank.\n \n
\n \n
\n );\n }\n}\n\nexport default UserProfileForm;\n","import * as types from '../actionTypes';\n\nexport const toggleUserProfileModal = () => ({\n type: types.TOGGLE_USER_PROFILE_MODAL\n});\n\nexport const toggleLoggoutConfirmModal = () => ({\n type: types.TOGGLE_LOGGOUT_CONFIRM_MODAL\n});\n","import React from 'react';\nimport { Col, FormControl, FormGroup, Row, FormLabel } from 'react-bootstrap';\nimport { connect } from 'react-redux';\nimport { addClassCode, toggleClassCodeModal } from '../../actions/userActions';\nimport CommonModal from '../common/CommonModal';\n\n\ninterface Iprops {\n show: boolean;\n submit: typeof addClassCode;\n cancel: ()=> void;\n addCode?: string;\n}\n\ninterface Istate {\n code: string;\n}\n\nclass ClassCodeModal extends React.Component {\n constructor(props: Iprops) {\n super(props);\n this.state = {\n code: this.props.addCode || ''\n }\n }\n render() {\n return (\n \n
{e.preventDefault(); this.props.submit(this.state.code)}}\n >\n \n \n \n \n \n Class Code or Group\n \n this.setState({code: e.currentTarget.value})}\n />\n \n \n
\n
\n\n \n
\n );\n }\n}\n\n\nfunction mapStateToProps(state: any, ownProps: any) {\n return {\n loading: state.ajaxCallsInProgress > 0,\n show: state.dashboard.showClassCodeModal\n };\n}\n\nexport default connect(mapStateToProps, {\n submit: addClassCode,\n cancel: toggleClassCodeModal\n})(ClassCodeModal);\n","import * as React from 'react';\nimport { connect } from 'react-redux';\nimport { toastr } from 'react-redux-toastr';\nimport { hashHistory } from 'react-router';\nimport UserProfileForm from '../common/UserProfileForm';\nimport CommonModal from '../common/CommonModal';\nimport { Nav, Navbar, NavItem } from 'react-bootstrap';\n\nimport {\n userUpdateProfile,\n userUpdatePW,\n userLogout,\n toggleClassCodeModal\n} from '../../actions/userActions';\nimport {\n toggleUserProfileModal,\n toggleLoggoutConfirmModal\n} from '../../actions/dashboard/dashboardActions';\n\nimport constants from '../../constants/constants';\nimport UserAPI from '../../api/userAPI';\nimport { StyledNavbar } from '../../constants/styledComponents';\nimport ClassCodeModal from './ClassCodeModal';\n\nconst FontAwesome = require('react-fontawesome');\n\nclass BookBagHeader extends React.Component {\n constructor(props) {\n super(props);\n this.state = {\n firstName: this.props.user.First,\n lastName: this.props.user.Last,\n password: '',\n confirmPassword: '',\n loginID: this.props.user.LoginID,\n logoutWarning:\n 'All of your downloaded books will be removed and you will have to redownload them. Are you sure you want to continue?'\n };\n this.theme = constants.themeProvider.activeTheme;\n this.logo = this.theme.logo;\n }\n\n componentDidMount() {\n this.setState({ logoutWarning: '' });\n }\n\n closeLoggoutConfirmModal = () => {\n this.setState({ showLoggoutConfirmModal: false });\n };\n\n handleLoggout = () => {\n this.props.userLogout();\n hashHistory.push('/login');\n };\n\n handleSelect = (e) => {\n switch (e) {\n case '1':\n this.props.toggleUserProfileModal();\n break;\n case '2':\n this.signOut();\n break;\n case '3':\n this.props.toggleClassCodeModal();\n break;\n default:\n }\n };\n\n handleChange = (e) => {\n this.setState({ [e.target.name]: e.target.value });\n };\n\n signOut = () => {\n // prevent user from logging out if there is no internet connection\n if (!this.props.isOnline) {\n toastr.error(\n `You must be connected to the internet before logging out.`,\n `Warning`,\n constants.toastrWarningOptions\n );\n return;\n }\n this.props.toggleLoggoutConfirmModal();\n };\n\n handleSubmit = () => {\n if (!this.props.isOnline) {\n toastr.error(\n `You must be connected to the internet before updating your profile or changing your password.`,\n `No Internet Connection`,\n {\n closeButton: true,\n showAnimation: 'animated fadeInDown'\n }\n );\n return;\n }\n if (\n this.state.firstName !== '' &&\n this.state.lastName !== '' &&\n this.state.loginID !== ''\n ) {\n const firstName = this.state.firstName;\n const lastName = this.state.lastName;\n const loginID = this.state.loginID;\n this.props.toggleUserProfileModal();\n\n this.props\n .userUpdateProfile(\n this.props.user,\n firstName,\n lastName,\n loginID\n )\n .then(() => {\n toastr.success(\n `Updated profile info.`,\n `Success`,\n constants.toastrSuccessOptions\n );\n })\n .catch((error) => {\n console.error('Error updating profile', error);\n toastr.error(\n `Unable to update profile. ${error.statusText}`,\n `Error`,\n constants.toastrErrorOptions\n );\n });\n }\n\n if (\n this.state.password === this.state.confirmPassword &&\n this.state.password !== ''\n ) {\n const password = this.state.password;\n this.props\n .userUpdatePW(this.props.user, password)\n .then(() => {\n toastr.success(\n `Updated password.`,\n `Success`,\n constants.toastrSuccessOptions\n );\n })\n .catch((error) => {\n console.error('Error updating password', error);\n toastr.error(\n `Unable to update password. ${error.statusText}`,\n `Error`,\n constants.toastrErrorOptions\n );\n });\n this.setState({ password: '', confirmPassword: '' });\n this.props.toggleUserProfileModal();\n }\n };\n buildUserProfileForm = () => {\n return (\n \n );\n };\n\n render() {\n return (\n \n
\n {/* */}\n \n
\n \n {/* */}\n \n \n \n \n\n
\n\n
\n
\n
\n );\n }\n}\n\nfunction mapStateToProps(state, ownProps) {\n return {\n user: state.user,\n showUserProfileModal: state.dashboard.showUserProfileModal,\n showLoggoutConfirmModal: state.dashboard.showLoggoutConfirmModal\n };\n}\n\nexport default connect(mapStateToProps, {\n userUpdateProfile,\n userUpdatePW,\n userLogout,\n toggleUserProfileModal,\n toggleLoggoutConfirmModal,\n toggleClassCodeModal\n})(BookBagHeader);\n","export default \"\"","export default \"\"","import * as types from './actionTypes';\n\nexport const saveAdvancedSearchFilter = (filters) => ({\n type: types.SAVE_ADVANCED_SEARCH_FILTERS,\n filters\n});\n\nexport const clearAdvancedSearchFilter = () => ({\n type: types.CLEAR_ADVANCED_SEARCH_FILTERS\n});\n","export default \"\"","import React from 'react';\nimport { connect } from 'react-redux';\nimport config from '../../api/config';\nimport {\n saveAdvancedSearchFilter,\n clearAdvancedSearchFilter\n} from '../../actions/bookbagFiltersActions';\nimport { Col, Row, FormGroup, FormControl, Form } from 'react-bootstrap';\nimport PropTypes from 'prop-types';\nimport debounce from 'throttle-debounce/debounce';\nimport CommonModal from '../common/CommonModal';\nimport { MultiSelect } from 'react-selectize';\nimport { reject } from 'lodash';\nimport UserAPI from '../../api/userAPI';\nimport Select from 'react-select';\n\nclass AdvancedSearchModal extends React.Component {\n constructor(props) {\n super(props);\n this.state = {\n showAdvancedSearch: false,\n advancedSearch: ''\n };\n this.debouncedAdvancedSearch = debounce(800, this.doAdvancedSearch);\n }\n\n clearAdvancedSearch = () => {\n this.props.clearAdvancedSearchFilter();\n this.props.cancel();\n };\n handleReadingLevelChange = (selectedReadingLevelTypeOption) => {\n this.props.saveAdvancedSearchFilter({ selectedReadingLevelTypeOption });\n };\n\n advancedSearchForm() {\n return (\n \n
\n \n \n {this.state.loading && (\n \n )}\n
\n \n
\n
\n \n \n {\n this.props.saveAdvancedSearchFilter({\n tags: value\n });\n }}\n closeOnSelect={true}\n onSearchChange={(advancedSearch) => {\n if (\n this.state.advancedSearch ===\n advancedSearch\n )\n return;\n this.setState({ advancedSearch });\n\n if (advancedSearch.length > 0) {\n this.setState({ loading: true });\n if (!!this.req) {\n this.req.abort();\n }\n this.handleAdvancedSearchChange(\n advancedSearch\n );\n }\n }}\n uid={(item) => {\n return item.TagID;\n }}\n filterOptions={(options, search) => {\n return options;\n }}\n renderOption={(item) => {\n return (\n \n
\n \n {item.name}\n \n \n {'@' +\n item.Name +\n ' (' +\n item.TagCount +\n ')'}\n \n
\n
\n );\n }}\n renderValue={(item) => {\n return (\n \n {\n let rejectedSearch = reject(\n this.props\n .bookbagFilters,\n (search) => {\n return (\n search.TagID ===\n item.TagID\n );\n }\n );\n this.props.saveAdvancedSearchFilter(\n rejectedSearch\n );\n }}\n >\n x\n \n\n \n {item.name}\n \n \n {'@' +\n item.Name +\n ' (' +\n item.TagCount +\n ')'}\n \n
\n );\n }}\n renderNoResultsFound={(value, search) => {\n return (\n \n {typeof this.req === 'undefined' &&\n this.state.advancedSearch.length ===\n 0\n ? 'type a few characters to kick off remote search'\n : 'No results found'}\n
\n );\n }}\n />\n \n \n
\n\n
\n \n \n \n \n \n \n \n \n this.props.saveAdvancedSearchFilter({\n readingMin: e.target.value\n })\n }\n />\n \n \n \n \n \n this.props.saveAdvancedSearchFilter({\n readingMax: e.target.value\n })\n }\n />\n \n \n
\n
\n \n \n \n this.props.saveAdvancedSearchFilter({\n spanishOnly: e.target.checked\n })\n }\n label=\"Spanish Only\"\n />\n \n \n
\n
\n );\n }\n\n doAdvancedSearch(value) {\n UserAPI.getAdvancedSearch(value, this.props.user)\n .then((advancedSearchResults) => {\n this.setState({\n advancedSearchResults: advancedSearchResults,\n loading: false\n });\n })\n .catch((error) => {\n this.setState({ loading: false });\n console.error(error, 'Error with advanced search');\n });\n }\n\n handleAdvancedSearchChange(value) {\n this.debouncedAdvancedSearch(value);\n }\n render() {\n return (\n \n \n
\n );\n }\n}\n\nAdvancedSearchModal.propTypes = {\n show: PropTypes.bool.isRequired,\n cancel: PropTypes.func.isRequired\n};\n\nfunction mapStateToProps(state, ownProps) {\n return {\n user: state.user,\n bookbagFilters: state.bookbagFilters,\n loading: state.ajaxCallsInProgress > 0,\n isOnline: state.offlineQueue.isOnline && navigator.onLine\n };\n}\n\nexport default connect(mapStateToProps, {\n saveAdvancedSearchFilter,\n clearAdvancedSearchFilter\n})(AdvancedSearchModal);\n","import * as React from 'react';\nimport allSquare from '../../images/NandF_Square.png';\nimport backpackImg from '../../images/backpack.png';\nimport bookshelfImg from '../../images/bookshelf.png';\nimport languageSquare from '../../images/Language_Square.png';\nimport mathSquare from '../../images/Math_Square.png';\nimport nonFictionSquare from '../../images/Nonfiction_Square.png';\nimport AdvancedSearchModal from './AdvancedSearchModal';\nimport scienceSquare from '../../images/Science_Square.png';\nimport socialStudiesSquare from '../../images/SocialStudies_Square.png';\nimport fictionSquare from '../../images/Fiction_Square.png';\nimport { FormGroup, Row, Col, FormControl, Button } from 'react-bootstrap';\nimport { StyledButton } from '../../constants/styledComponents';\nimport { addQuery, removeQuery } from '../../vendor/utils-router';\nconst FontAwesome = require('react-fontawesome');\n\nimport {\n checkForUpdatedBookFiles,\n getBookItemsList,\n initialDashboardQuery,\n searchBookBagBooks\n} from '../../actions/bookActions';\nimport { clearAdvancedSearchFilter } from '../../actions/bookbagFiltersActions';\n\ninterface Iprops {\n query: typeof initialDashboardQuery;\n isOnline: boolean;\n bookbagFilters: {\n tags: any;\n spanishOnly: any;\n selectedReadingLevelTypeOption: any;\n readingMax: any;\n readingMin: any;\n };\n showSearchBar: boolean;\n theme: any;\n books: Array;\n searchBookBagBooks: typeof searchBookBagBooks;\n isFiltersActive: boolean;\n user: any;\n downloadedBooks: Array;\n checkForUpdatedBookFiles: typeof checkForUpdatedBookFiles;\n manualAjaxEnd: () => void;\n handleInvalidSession: () => void;\n clearAdvancedSearchFilter: typeof clearAdvancedSearchFilter;\n loading: boolean;\n isStudent: boolean;\n isGeneric: boolean;\n getBookItemsList: typeof getBookItemsList;\n}\n\ninterface Istate {\n showAdvancedSearch: boolean;\n}\n\nclass BookBagSearch extends React.Component {\n constructor(props: Iprops) {\n super(props);\n this.state = {\n showAdvancedSearch: false\n };\n }\n componentDidMount() {\n this.initializeComponent();\n }\n componentDidUpdate(prevProps: Iprops) {\n if (prevProps.isOnline !== this.props.isOnline && this.props.isOnline) {\n setTimeout(() => {\n this.updateBookItems();\n }, 300);\n }\n }\n /*\n * initial setup of the component\n * call getBookListItems to get the latest book items for DOWNLOADED books only\n * this.props.downloadedBooks, loop through array to get all bookIDs\n * Action: save the book contents to downloaded books\n * Potential Issue: Being offline, saving a book item, going online and going straight to book bag:\n * Solution: check to see if there are any ajaxCallsInProgress\n */\n initializeComponent = () => {\n if (!this.props.query.searchMode) {\n addQuery({ searchMode: 'local' });\n }\n this.updateBookItems();\n\n if (this.props.isGeneric) {\n addQuery({ searchMode: 'remote' });\n setTimeout(() => {\n this.props.searchBookBagBooks(\n this.props.isFiltersActive,\n this.props.query\n );\n }, 100);\n } else if (this.props.isStudent) {\n // if local then always load books. If remote the only load books if there is a search\n if (\n !this.props.query.searchMode ||\n this.props.query.searchMode === 'local'\n ) {\n this.props.searchBookBagBooks(\n this.props.isFiltersActive,\n this.props.query\n );\n } else if (\n this.props.query.searchMode === 'remote' &&\n this.props.query.search.length > 0 &&\n this.props.books.length === 0\n ) {\n this.props.searchBookBagBooks(\n this.props.isFiltersActive,\n this.props.query\n );\n }\n if (this.props.isOnline) {\n console.log('****** we are online');\n } else {\n console.log('****** we are offline');\n }\n } else {\n // if you are not a student or a Generic user, they will be sent to the login screen via the auth check in the render function.\n }\n // TODO load all assigned blms and save this to redux\n // TODO start with including it into props\n };\n /*\n * If we are on a device and downloaded books, check to see if there are updated Book Items or updated Book files\n */\n updateBookItems = () => {\n console.log('about to update book items');\n if (this.props.isOnline && this.props.downloadedBooks.length > 0) {\n // Make call to grab all book items if ajax calls are not currently in progress\n if (!this.props.loading) {\n this.props.getBookItemsList(\n this.props.downloadedBooks,\n this.props.user\n );\n } else {\n //wait 5 seconds then try again\n setTimeout(() => {\n this.updateBookItems();\n }, 5000);\n }\n }\n };\n\n isActiveAdvancedSearch = () => {\n const {\n tags,\n spanishOnly,\n selectedReadingLevelTypeOption,\n readingMax,\n readingMin\n } = this.props.bookbagFilters;\n if (\n (tags && tags.length) ||\n spanishOnly ||\n (selectedReadingLevelTypeOption &&\n selectedReadingLevelTypeOption.value !== 'grl') ||\n readingMax ||\n readingMin\n ) {\n return true;\n } else {\n return false;\n }\n };\n\n /*\n * Switch to Remote Books\n */\n switchToRemote = () => {\n // this.setState({ searchMode: 'remote' }, () => {\n // // only load remote books if there is a search paramater\n // // do we want to search if there is a tag, type, or reading level selected??? Lets stick with search for now\n // // if (this.state.search.length > 0 || this.state.tag !== 'all-tags' || this.state.type !== 'all' || this.state.readingLeve.length > 0){\n // });\n addQuery({ searchMode: 'remote' });\n removeQuery('activePage');\n };\n\n /*\n * Switch to Local Books\n */\n switchToLocal = () => {\n addQuery({ searchMode: 'local' });\n removeQuery('activePage');\n };\n\n searchForBooks = (event: React.ChangeEvent) => {\n if (event) {\n addQuery({ search: event.target.value });\n }\n };\n\n searchForReadingLevel = (event: React.ChangeEvent) => {\n if (event) {\n addQuery({ grl: event.target.value });\n }\n };\n\n handleTagChange = (val: string) => {\n addQuery({ tag: val });\n };\n\n handleTypeChange = (val: string) => {\n addQuery({ type: val });\n };\n\n showAdvancedSearch = () => {\n this.setState({ showAdvancedSearch: true });\n };\n\n closeAdvancedSearch = () => {\n this.setState({ showAdvancedSearch: false });\n };\n clearAdvancedSearchForm = () => {\n removeQuery('search', 'type', 'tag', 'grl');\n addQuery({ activePage: 1 });\n this.props.clearAdvancedSearchFilter();\n };\n\n handleSubmit = (\n event: React.FormEvent,\n activePage: number\n ) => {\n if (event) {\n event.preventDefault();\n }\n console.log(`initiating a bookbag search: ${this.props.query.search}`);\n addQuery({ activePage });\n this.props.searchBookBagBooks(this.props.isFiltersActive, {\n ...this.props.query,\n activePage\n });\n };\n\n render() {\n let backpack = '';\n let bookshelf = '';\n let bookbagSearchMode = 'bookbag-searchmode';\n if (this.props.isGeneric) {\n bookbagSearchMode = 'd-none';\n }\n if (\n !this.props.query.searchMode ||\n this.props.query.searchMode === 'local'\n ) {\n backpack += 'backpack';\n bookshelf += '';\n } else {\n backpack += '';\n bookshelf += 'bookshelf';\n }\n\n let advancedSearchButton = this.isActiveAdvancedSearch()\n ? 'advanced-search-active'\n : 'advanced-search';\n // tags\n let languageBC =\n this.props.query.tag === 'languageArts' ? 'language-bc' : '';\n let socialStudiesBC =\n this.props.query.tag === 'socialStudies' ? 'social-science-bc' : '';\n let mathBC = this.props.query.tag === 'math' ? 'math-bc' : '';\n let scienceSearchBC =\n this.props.query.tag === 'science' ? 'science-bc' : '';\n let allTags = this.props.query.tag === 'allTags' ? 'all-tags' : '';\n // types\n let allBC = this.props.query.type === 'All' ? 'all-bc' : '';\n let fictionBC = this.props.query.type === 'Fiction' ? 'fiction-bc' : '';\n let nonFictionBC =\n this.props.query.type === 'Non-Fiction' ? 'non-fiction-bc' : '';\n\n return (\n \n {this.props.showSearchBar && this.props.isOnline && (\n \n )}\n \n );\n }\n}\n\nexport default BookBagSearch;\n","export default \"\"","export default \"\"","export default \"\"","export default \"\"","export default \"\"","export default \"\"","export default \"\"","export default \"\"","import { connect } from 'react-redux';\nimport { manualAjaxEnd } from '../../actions/ajaxStatusActions';\nimport {\n checkForUpdatedBookFiles,\n getBookItemsList,\n initialDashboardQuery,\n searchBookBagBooks\n} from '../../actions/bookActions';\nimport { clearAdvancedSearchFilter } from '../../actions/bookbagFiltersActions';\nimport UserAPI from '../../api/userAPI';\nimport BookBagSearch from './BookBagSearch';\n\ninterface Iprops {\n query: typeof initialDashboardQuery;\n theme: any;\n isFiltersActive: boolean;\n handleInvalidSession: () => void;\n}\n\nconst mapStateToProps = (state: any, ownProps: Iprops) => {\n const showSearchBar =\n UserAPI.isStudent(state.user.RoleID) ||\n UserAPI.isGeneric(state.user.RoleID);\n return {\n loading: state.ajaxCallsInProgress > 0,\n books: state.books,\n downloadedBooks: state.downloadedBooks,\n isGeneric: UserAPI.isGeneric(state.user.RoleID),\n isStudent: UserAPI.isStudent(state.user.RoleID),\n user: state.user,\n isOnline: state.offlineQueue.isOnline && navigator.onLine,\n bookbagFilters: state.bookbagFilters,\n showSearchBar\n };\n};\n\nexport const BookBagSearchContainer = connect(mapStateToProps, {\n getBookItemsList,\n checkForUpdatedBookFiles,\n clearAdvancedSearchFilter,\n searchBookBagBooks: searchBookBagBooks,\n manualAjaxEnd: manualAjaxEnd\n})(BookBagSearch);\n","import { Col, Container, Row } from 'react-bootstrap';\nimport { StyledButton, StyledDiv } from '../../constants/styledComponents';\nimport {\n addBook,\n searchBookBagBooks,\n initialDashboardQuery,\n removeBook,\n downloadBook,\n deleteDownloadedBook\n} from '../../actions/bookActions';\nimport { addQuery, removeQuery } from '../../vendor/utils-router';\nimport { manualAjaxStart } from '../../actions/ajaxStatusActions';\n\nimport BookBagHeader from './BookBagHeader';\nimport Loading from '../common/Loading';\nimport React from 'react';\n\nimport config from '../../api/config';\nimport { connect } from 'react-redux';\nimport constants from '../../constants/constants';\nimport defaultPhoto from '../../images/default.jpg';\nimport downloadedImage from '../../images/downloaded.png';\nimport { forEach } from 'lodash';\nimport { hashHistory } from 'react-router';\nimport { indexOf } from 'lodash';\n\nimport readBook from '../../images/read.png';\nimport { requireSignIn } from '../../routes';\n\nimport { sortBy } from 'lodash';\nimport { toastr } from 'react-redux-toastr';\nimport { toggleUserProfileModal } from '../../actions/dashboard/dashboardActions';\nimport unReadBook from '../../images/unread.png';\nimport { toggleClassCodeModal, userLogout } from '../../actions/userActions';\nimport UltimatePagination from 'react-ultimate-pagination-bootstrap-4';\nimport { BookBagSearchContainer } from './BookBagSearchContainer';\nimport UserAPI from '../../api/userAPI';\n\nlet bookState = unReadBook;\n\nclass BookBag extends React.Component {\n constructor(props) {\n super(props);\n this.state = {\n prevSelected: [],\n selectedTypes: [],\n selectedBookID: false,\n loading: false,\n emptySearchMessage:\n \"Please enter your search criteria above and press 'Go'\"\n };\n\n this.openingBook = false;\n this.theme = constants.themeProvider.activeTheme;\n }\n\n componentDidMount() {\n console.log('bookbag is mounted');\n // this.handleTypeChange(this.props.query.tag); // TODO what is this????\n // this.handleTypeChange(this.props.query.type);\n // remove the BLM ID and projectAssignmentID from query so that we do not stay in BLM mode when switching books\n removeQuery('blmID');\n removeQuery('projectAssignmentID');\n this.checkAddCode();\n this.checkCanAccessBookBag();\n }\n\n componentDidUpdate(prevProps) {\n if (prevProps.books !== this.props.books) {\n // set the appropriate empty search message. If this is the local bag and they have no search criteria, then they have no books in their bag\n let emptySearchMessage = '';\n if (this.props.books.length <= 0 && !this.filtersActive()) {\n emptySearchMessage = `You currently do not have any books in your book bag.`;\n }\n this.setState({ emptySearchMessage });\n }\n if (prevProps.query !== this.props.query) {\n this.setState({ prevSelected: [] });\n }\n }\n\n /*\n * only students, generic, and demo can access the bookbag. otherwise log them out\n */\n checkCanAccessBookBag = () => {\n if (this.props.user.SchoolID && (this.props.isStudent || this.props.isDemo || this.props.isGeneric) === false){\n toastr.warning('Warning', 'User does not have access to the bookbag. Logging out...', constants.toastrWarningOptions);\n setTimeout(() => {\n this.props.userLogout();\n hashHistory.replace('/');\n }, 5000);\n }\n\n\n }\n\n checkAddCode = () =>{\n console.log('checking addcode')\n if (this.props.addCode){\n setTimeout(() => {\n this.props.toggleClassCodeModal(true);\n }, 1000);\n }\n }\n\n handleBookClick = (e, book) => {\n if (e) {\n e.preventDefault();\n }\n if (this.state.selectedBookID === book.ID) {\n this.openBook(book);\n } else {\n this.setState({ selectedBookID: book.ID });\n }\n };\n\n openBook = (book) => {\n // prevent double click on open book\n // If we are offline:\n if (this.openingBook) return;\n\n this.openingBook = true;\n setTimeout(() => {\n this.openingBook = false;\n }, 300);\n\n if (book.IsExternal) {\n // open the book in the Lerner viewer, not in the DiBS viewer\n const externalURL = `${config.API.Main}/book/openexternalviewer?bookID=${book.ID}`;\n var win = window.open(externalURL, '_blank');\n win.focus();\n } else {\n // addQuery({bookID: book.ID});\n const location = Object.assign(\n {},\n hashHistory.getCurrentLocation()\n );\n Object.assign(location.query, { bookID: book.ID });\n hashHistory.push({\n pathname: `/viewer`,\n query: location.query\n });\n }\n };\n\n handleSubmit = (e, activePage) => {\n if (e) {\n e.preventDefault();\n }\n console.log(`initiating a bookbag search: ${this.props.query.search}`);\n addQuery({ activePage });\n this.props.searchBookBagBooks(this.filtersActive(), {\n ...this.props.query,\n activePage\n });\n };\n\n // are any book bag filters active? returns a boolean\n filtersActive = () => {\n if (\n this.props.query.search.length === 0 &&\n this.props.query.type.length === 0 &&\n this.props.query.tag.length === 0 &&\n this.props.query.grl.length === 0 &&\n this.props.bookbagFilters.length === 0\n ) {\n return false;\n } else {\n return true;\n }\n };\n\n handleInvalidSession = () => {\n toastr.error(\n `Please login again.`,\n `Session Expired`,\n constants.toastrErrorOptions\n );\n setTimeout(() => {\n this.props.userLogout();\n hashHistory.replace('/');\n }, 3000);\n };\n\n displayBookReadingLevel = (book) => {\n let grl, atos, lex;\n let hideBookReadingLevel = 'book-reading-level';\n if (\n book.GuidedReadingLevel === '' ||\n book.GuidedReadingLevel === null\n ) {\n grl = 'grl-hide';\n } else if (book.ATOS === '' || book.ATOS === null) {\n atos = 'atos-hide';\n } else if (book.LEX === '' || book.LEX === null) {\n lex = 'lex-hide';\n }\n\n if (!book.GuidedReadingLevel && !book.ATOS && !book.LEX) {\n hideBookReadingLevel = '';\n return;\n }\n\n return (\n \n {book.ATOS !== null && book.ATOS !== '' && (\n A: {book.ATOS}, \n )}\n {book.LEX !== null && book.LEX !== '' && (\n L: {book.LEX}, \n )}\n {book.GuidedReadingLevel !== null &&\n book.GuidedReadingLevel !== '' && (\n \n GRL: {book.GuidedReadingLevel}\n \n )}\n
\n );\n };\n\n displayBookImage = (book) => {\n let img = !!book.OfficialImage ? book.OfficialImage : defaultPhoto;\n const bookTitle = book.Title;\n let downloadedIcon;\n forEach(this.props.downloadedBooks, (dlBook) => {\n if (book.ID === dlBook.ID) {\n downloadedIcon = downloadedImage;\n }\n });\n\n let photo = {\n backgroundImage: `url(${img})`,\n backgroundPosition: 'center center',\n backgroundSize: 'contain',\n backgroundRepeat: 'no-repeat'\n };\n if (!!book.OfficialImage) {\n return (\n \n
{bookTitle}
\n
this.handleBookClick(e, book)}\n >\n

\n {book.IsExternal && (\n
\n \n
\n )}\n {this.displayBookReadingLevel(book)}\n
\n
\n );\n } else {\n return (\n this.handleBookClick(e, book)}\n >\n

\n {book.IsExternal && (\n
\n \n
\n )}\n
{bookTitle}
\n {this.displayBookReadingLevel(book)}\n
\n );\n }\n };\n\n render() {\n if (requireSignIn(this.props.user, this.props.location)) {\n return null;\n }\n let books = !!this.props.books\n ? this.props.books.filter((book) => {\n return book;\n })\n : [];\n const downloadedBooks = this.props.downloadedBooks;\n let filteredBooks = this.props.isOnline\n ? sortBy(books, [\n function (o) {\n return o.Title;\n }\n ])\n : sortBy(downloadedBooks, [\n function (o) {\n return o.Title;\n }\n ]);\n\n return (\n \n \n \n\n {!this.props.isOnline && (\n \n \n You are offline\n \n
\n )}\n \n {filteredBooks.length <= 0 && (\n \n \n {this.state.emptySearchMessage}\n
\n \n )}\n {filteredBooks.map((book) => {\n let selected = 'd-none';\n let glow = 'book-cover book-shadow';\n let bookHeight = '';\n\n if (this.state.selectedBookID === book.ID) {\n selected = 'visible animated fadeInDown';\n bookHeight = ' pulse raise-book';\n glow = 'book-cover glow';\n if (book.CurrentPage !== null && book.CurrentPage !== 1) {\n bookState = readBook;\n }\n } else if (\n indexOf(this.state.prevSelected, book.ID) !== -1\n ) {\n bookHeight = ' ';\n }\n if (book.CurrentPage !== null && book.CurrentPage !== 1) {\n bookState = readBook;\n } else {\n bookState = unReadBook;\n }\n const isDownloaded = this.props.downloadedBooks.find(\n (downloadedBook) => downloadedBook.ID === book.ID\n )\n ? true\n : false;\n return (\n \n \n
\n {this.displayBookImage(book)}\n

\n
\n
\n this.openBook(book)}\n >\n READ\n \n {book.isInBookBag === true &&\n isDownloaded === false && (\n \n this.props\n .downloadBook(book)\n .then(() => {\n toastr.success('Succcess', 'Book downloaded for viewing offline.', constants.toastrSuccessOptions)\n })\n }\n >\n \n DOWNLOAD\n \n \n )}\n {book.isInBookBag === true &&\n isDownloaded === true && (\n \n this.props.deleteDownloadedBook(\n book.ISBN\n )\n }\n >\n \n REMOVE DOWNLOAD\n \n \n )}\n\n {book.isInBookBag === false && (\n \n this.props.addBook(book)\n }\n >\n \n ADD TO BOOK BAG\n \n \n )}\n {book.IsStudentAdded === true &&\n book.isInBookBag === true && (\n \n this.props.removeBook(\n book.ID, book.ISBN\n )\n }\n >\n \n REMOVE FROM BAG\n \n \n )}\n
\n
\n \n );\n })}\n
\n {this.props.books.length > 0 && (\n \n \n \n \n this.handleSubmit('', pageNumber)\n }\n />\n \n \n
\n )}\n \n \n );\n }\n}\n\nfunction mapStateToProps(state, ownProps) {\n const addCode = ownProps.location.query.addCode || ownProps.location.query.addcode || ownProps.location.query.AddCode;\n return {\n completedBLMs: state.completedBLMs,\n user: state.user,\n book: state.book,\n books: state.books,\n bookbagFilters: state.bookbagFilters,\n loading: state.ajaxCallsInProgress > 0,\n isOnline: state.offlineQueue.isOnline && navigator.onLine,\n downloadedBooks: state.downloadedBooks,\n pageResults: state.dashboardPageResults,\n query: { ...initialDashboardQuery, ...ownProps.location.query },\n addCode,\n isGeneric: UserAPI.isGeneric(state.user.RoleID),\n isStudent: UserAPI.isStudent(state.user.RoleID),\n isDemo: UserAPI.isDemo(state.user.RoleID)\n };\n}\n\nexport default connect(mapStateToProps, {\n addBook,\n removeBook,\n searchBookBagBooks,\n userLogout,\n manualAjaxStart,\n toggleUserProfileModal,\n downloadBook,\n deleteDownloadedBook,\n toggleClassCodeModal\n})(BookBag);\n","export default \"\"","export default \"\"","import { Container, Nav, NavItem, Navbar } from 'react-bootstrap';\nimport { ToastContainer, ToastMessage } from 'react-toastr';\nimport {\n manualAjaxEnd,\n manualAjaxStart\n} from '../../actions/ajaxStatusActions';\nimport {\n userLogout,\n userUpdatePW,\n userUpdateProfile\n} from '../../actions/userActions';\n\nimport FabricUI from './FabricUI';\nimport FontAwesome from 'react-fontawesome';\nimport React from 'react';\nimport { StyledNavbar } from '../../constants/styledComponents';\nimport { connect } from 'react-redux';\nimport constants from '../../constants/constants';\nimport { hashHistory } from 'react-router';\nimport logo from '../../images/logo.png';\nimport { requireSignIn } from '../../routes';\n\nconst ToastMessageFactory = React.createFactory(ToastMessage.animation);\n\nclass BLMView extends React.Component {\n toastCont;\n\n constructor(props) {\n super(props);\n this.state = {};\n\n this.handleSelect = this.handleSelect.bind(this);\n this.theme = constants.themeProvider.activeTheme;\n }\n\n signOut() {\n this.props.userLogout();\n hashHistory.push('/login');\n }\n\n handleSelect(e) {\n switch (e) {\n case 2:\n this.signOut();\n break;\n default:\n }\n }\n\n render() {\n if (requireSignIn(this.props.user, this.props.location)) {\n return null;\n }\n\n return (\n \n \n \n \n
\n \n \n \n \n \n \n \n {\n // \n }\n {\n this.toastCont = toast;\n }}\n className=\"toast-top-right\"\n />\n \n );\n }\n}\n\n/*\n \n
*/\n\nfunction mapStateToProps(state, ownProps) {\n return {\n user: state.user,\n book: state.book,\n books: state.books,\n loading: state.ajaxCallsInProgress > 0\n };\n}\n\nexport default connect(mapStateToProps, {\n userUpdateProfile,\n userUpdatePW,\n userLogout,\n manualAjaxStart,\n manualAjaxEnd\n})(BLMView);\n","import React, { Component } from 'react';\nimport {\n manualAjaxEnd,\n manualAjaxStart\n} from '../../actions/ajaxStatusActions';\nimport { tokenLogin, userLogout } from '../../actions/userActions';\n\nimport { Container } from 'react-bootstrap';\nimport Loading from '../common/Loading';\nimport { Navbar } from 'react-bootstrap';\nimport { StyledNavbar } from '../../constants/styledComponents';\nimport { connect } from 'react-redux';\nimport constants from '../../constants/constants';\nimport { getBLMByID } from '../../actions/blmActions';\nimport { getBookByID } from '../../actions/bookActions';\nimport { hashHistory } from 'react-router';\nimport logo from '../../images/logo.png';\nimport { toastr } from 'react-redux-toastr';\nimport { updateViewerMode } from '../../actions/modeActions';\nimport { viewerModes } from '../../constants/viewerModes';\n\n/*\n *** IMPORTANT ***\n - Tokens are only applicable for web deployments\n - (will not be used for iOS and Android)\n *** IMPORTANT ***\n*/\n\nclass Token extends Component {\n constructor(props) {\n super(props);\n\n this.state = {\n showLoading: true\n };\n\n this.theme = constants.themeProvider.activeTheme;\n }\n\n componentDidMount() {\n console.log('opening book with token login');\n // return;\n let params = this.props.location.query;\n if (!params || !params.token) {\n // fail\n this.setState({ showLoading: false });\n toastr.error(\n `Error`,\n `Missing token/bookID, unable to load book`,\n constants.toastrWarningOptions\n );\n } else {\n // console.log(params);\n this.doLogin(params);\n }\n }\n\n doLogin = (params) => {\n this.props\n .userLogout()\n .then(() => {\n return this.props.tokenLogin(params.token);\n })\n .then((user) => {\n console.log('***USER', user.LoginID);\n if (user.APIKey === null) {\n toastr.error(\n `Error`,\n 'Unable to load book. Please contact support. User missing API Key.',\n constants.toastrWarningOptions\n );\n return;\n }\n // we need to account for multiple possibilities now...\n // if we have a bookID only, we should just load up the book\n if (params.viewerMode === viewerModes.MODE_GENERIC) {\n return this.props\n .getBookByID(params.bookID, user)\n .then((bookSaved) => {\n this.setState({ loading: false });\n // hashHistory.push('/viewer/' + params.bookID);\n hashHistory.push({\n pathname: `/viewer`,\n query: this.props.location.query\n });\n });\n // TODO: error handling here?\n }\n // if we have a bookID and a studentID, we should load the book\n // and load the student's highlights\n else if (\n params.viewerMode ===\n viewerModes.MODE_TEACHER_VIEW_STUDENT_BOOK\n ) {\n console.log(\n 'TOKEN: LOADING TEACHER VIEW STUDENT HIHGLIGHTS'\n );\n return this.props\n .getBookByID(params.bookID, user)\n .then((bookSaved) => {\n this.setState({ loading: false });\n hashHistory.push({\n pathname: `/viewer`,\n query: this.props.location.query\n });\n });\n // TODO: error handling here?\n } else if (\n params.viewerMode === viewerModes.MODE_TEACHER_CLASS_NOTES\n ) {\n console.log('TOKEN: LOADING TEACHER VIEW CLASS NOTES');\n return this.props\n .getBookByID(params.bookID, user)\n .then((bookSaved) => {\n this.setState({ loading: false });\n hashHistory.push({\n pathname: `/viewer`,\n query: this.props.location.query\n });\n });\n // TODO: error handling here?\n } else if (\n params.viewerMode === viewerModes.MODE_TEACHER_GROUP_NOTES\n ) {\n console.log('TOKEN: LOADING TEACHER VIEW Group NOTES');\n return this.props\n .getBookByID(params.bookID, user)\n .then((bookSaved) => {\n this.setState({ loading: false });\n hashHistory.push({\n pathname: `/viewer`,\n query: this.props.location.query\n });\n });\n // TODO: error handling here?\n }\n // if we have a blmID, we should load up the blm in admin\n // create template mode\n else if (\n params.viewerMode === viewerModes.MODE_ADMIN_CREATE_TEMPLATE\n ) {\n // TODO: load admin create mode\n return this.props\n .getBLMByID(params.blmID, user)\n .then((blmSaved) => {\n this.setState({ loading: false });\n hashHistory.push({\n pathname: `/viewer`,\n query: this.props.location.query\n });\n });\n } else if (\n params.viewerMode === viewerModes.MODE_TEACHER_STUDENT_BLM\n ) {\n return this.props\n .getBLMByID(params.blmID, user)\n .then((blmSaved) => {\n this.setState({ loading: false });\n // hashHistory.push(`/viewer/${blmSaved.BookID}/${params.blmID}`);\n hashHistory.push({\n pathname: `/viewer`,\n query: this.props.location.query\n });\n });\n } else if (params.viewerMode === viewerModes.MODE_RESOURCE) {\n return this.props\n .getBookByID(params.bookID, user)\n .then((bookSaved) => {\n this.setState({ loading: false });\n // hashHistory.push('/viewer/' + params.bookID);\n hashHistory.push({\n pathname: `/viewer`,\n query: this.props.location.query\n });\n });\n } else if (params.viewerMode === viewerModes.MODE_STUDENT) {\n hashHistory.push({\n pathname: `/bag?searchMode=local&tag=allTags`,\n query: ''\n });\n }\n })\n .catch((error) => {\n this.setState({ showLoading: false });\n console.error(\n `error logging in or loading book with token`,\n error\n );\n let message = `Unable to login or load book with token`;\n if (error === undefined) {\n message = `The provided login token is invalid.`;\n }\n toastr.error(`Error`, message, constants.toastrWarningOptions);\n });\n };\n\n render() {\n return (\n \n \n {' '}\n \n \n
\n \n \n \n \n \n
\n \n );\n }\n}\n\nconst mapStateToProps = (state, ownProps) => {\n return {\n user: state.user,\n loading: state.ajaxCallsInProgress > 0\n };\n};\n\nexport default connect(mapStateToProps, {\n userLogout,\n tokenLogin,\n getBookByID,\n getBLMByID,\n updateViewerMode,\n manualAjaxStart,\n manualAjaxEnd\n})(Token);\n","import * as types from './actionTypes';\n\nexport function updateViewerMode(viewerMode) {\n return function (dispatch) {\n dispatch({ type: types.UPDATE_MODE_SUCCESS, viewerMode });\n };\n}\n","import * as React from 'react';\nimport { Route, IndexRoute } from 'react-router';\nimport App from './App';\nimport LoginForm from './components/auth/LoginForm';\nimport RegisterForm from './components/auth/RegisterForm';\nimport BookView from './components/bookview/BookView';\nimport BookBag from './components/bookbag/BookBag';\nimport BLMView from './components/blm/BLMView';\nimport Token from './components/auth/Token';\nimport { hashHistory } from 'react-router';\n\nexport const requireSignIn = (user, location) => {\n const redirect = `${location.pathname}${location.search}`;\n if (\n !user ||\n (user && !user.APIKey) ||\n (user && user.APIKey && user.APIKey.length === 0)\n ) {\n console.info('sign in required, redirecting user', redirect, user);\n hashHistory.push(`/?redirect=${redirect}`);\n return true;\n }\n return false;\n};\n\nexport default (\n \n \n \n \n \n {/* // All params for the BookBag are GET params... managed within the component */}\n \n \n {/* // All params for the Token component are GET params... managed within the component */}\n \n \n);\n","import config from '../api/config';\n\n/*\n * load state localStorage\n */\nexport const loadState = (key) => {\n var statePromise = new Promise((resolve, reject) => {\n try {\n const serializedState = localStorage.getItem(`state-${key}`);\n if (serializedState === null) {\n resolve(undefined);\n } else {\n resolve(JSON.parse(serializedState));\n }\n } catch (error) {\n console.error('Error loading state', error);\n resolve(undefined);\n }\n });\n return statePromise;\n};\n\n/*\n * Save state\n */\nexport const saveState = (key, state) => {\n try {\n const serializedState = JSON.stringify(state);\n localStorage.setItem(`state-${key}`, serializedState);\n } catch (error) {\n // ignore write error\n console.error('Error saving state', error);\n }\n};\n","import config from './config';\nimport { loadState } from '../store/localStorage';\n\nexport const CL_DEBUG = 1;\nexport const CL_VERBOSE = 2;\nexport const CL_INFO = 3;\nexport const CL_WARN = 4;\nexport const CL_ERROR = 5;\nexport const CL_CRITICAL = 6;\n\nexport class Logger {\n static log(severity, ...args) {\n return loadState('dibs').then((state) => {\n let text = { ...args };\n if (severity === CL_ERROR || severity === CL_CRITICAL) {\n text = { ...args, reduxStore: state };\n }\n var obj = {\n privateKey: config.Coralogix.key,\n applicationName: 'DiBS.' + config.Environment,\n subsystemName: config.Coralogix.subsystemName,\n computerName:\n state && state.user\n ? `${state.user.First} ${state.user.Last}`\n : 'not logged in',\n logEntries: [\n {\n timestamp: new Date().getTime(),\n severity: severity,\n text: text,\n category:\n state && state.user\n ? state.user.APIKey\n : 'no api key'\n }\n ]\n };\n\n return this.sendToCoralogix(obj);\n });\n }\n\n static sendToCoralogix(obj) {\n if (process.env.NODE_ENV !== 'production') {\n return true;\n }\n return fetch('https://api.coralogix.com/api/v1/logs', {\n method: 'POST',\n headers: new Headers({ 'content-type': 'application/json' }),\n body: JSON.stringify(obj)\n })\n .then(function (response) {\n return response.text();\n })\n .then(function (data) {\n return data;\n })\n .catch(function (ex) {\n // if logging fails... what can ya do?\n throw ex;\n });\n }\n}\n","import { map, pickBy, find } from 'lodash';\nimport * as types from '../actions/actionTypes';\nimport initialState from './initialState';\n\nconst normalizeBook = (rawBook) => {\n return {\n ...initialState.book,\n ...pickBy(rawBook, (property) => property !== null)\n };\n};\n\n/*\n * Book Reducer\n */\nexport function bookReducer(state = initialState.book, action) {\n switch (action.type) {\n case types.FETCH_BOOK_SUCCESS:\n return normalizeBook(action.book);\n case types.FETCH_BOOK_FAILED: {\n return initialState.book;\n }\n case types.LOAD_BOOKS_SUCCESS:{\n const foundBook = find(action.books, {ID: state.ID})\n if (foundBook){\n return {...state, ...normalizeBook(foundBook)}\n }\n return state;\n }\n\n case types.LOAD_BOOK_SUCCESS:\n return Object.assign({}, state, action.book);\n\n case types.FETCH_BOOKITEMS_STUDENT_SUCCESS:\n // when connecting back online, dispatch actions might have been triggered for a different book\n if (state.ID !== action.bookID) {\n return state;\n }\n return Object.assign({}, state, {\n studentBookmarks: action.bookmarks,\n studentHighlights: action.highlights,\n studentNotes: action.notes\n });\n\n case types.FETCH_BOOKITEMS_SUCCESS:\n // when connecting back online, dispatch actions might have been triggered for a different book\n if (state.ID !== action.bookID) {\n return state;\n }\n return Object.assign({}, state, {\n bookmarks: action.bookmarks,\n highlights: action.highlights,\n notes: action.notes\n });\n\n case types.SAVE_BOOKITEM_BOOKMARK_SUCCESS:\n // when connecting back online, dispatch actions might have been triggered for a different book\n if (state.ID !== action.bookID) {\n return state;\n }\n let bookmarks = [];\n if (!state.bookmarks) {\n Object.assign({}, state, {\n bookmarks: bookmarks\n });\n }\n bookmarks = [\n ...state.bookmarks.filter(\n (bookmark) => bookmark.ID !== action.bookmark.ID\n ),\n Object.assign({}, action.bookmark)\n ];\n return Object.assign({}, state, {\n bookmarks: bookmarks\n });\n\n case types.SAVE_BOOKITEM_NOTE_SUCCESS:\n // when connecting back online, dispatch actions might have been triggered for a different book\n if (state.ID !== action.bookID) {\n return state;\n }\n let notes = [];\n if (!state.notes) {\n Object.assign({}, state, {\n notes: notes\n });\n }\n\n if (action.note.ID === undefined) {\n // this is for non-students who only save these notes locally - thus they never get a real ID\n notes = [\n ...state.notes.filter(\n (note) => note.TempID !== action.note.TempID\n ),\n Object.assign({}, action.note)\n ];\n } else {\n notes = [\n ...state.notes.filter((note) => note.ID !== action.note.ID),\n Object.assign({}, action.note)\n ];\n }\n notes.sort((a, b) => {\n return a.Page > b.Page;\n });\n return Object.assign({}, state, {\n notes: notes\n });\n\n case types.SAVE_BOOKITEM_HIGHLIGHT_SUCCESS:\n // when connecting back online, dispatch actions might have been triggered for a different book\n if (state.ID !== action.bookID) {\n return state;\n }\n return Object.assign({}, state, {\n highlights: action.highlights\n });\n\n case types.DELETE_BOOKITEM_BOOKMARK_SUCCESS:\n // when connecting back online, dispatch actions might have been triggered for a different book\n if (state.ID !== action.bookID) {\n return state;\n }\n return Object.assign({}, state, {\n bookmarks: state.bookmarks.filter(\n (bookmark) => bookmark.ID !== action.bookmark.ID\n )\n });\n\n case types.DELETE_BOOKITEM_NOTE_SUCCESS:\n // when connecting back online, dispatch actions might have been triggered for a different book\n if (state.ID !== action.bookID) {\n return state;\n }\n\n return Object.assign({}, state, {\n notes: state.notes.filter((note) => note.ID !== action.note.ID)\n });\n // TODO do we need to support teachers deleting student notes?\n // case types.TEACHER_DELETE_BOOKITEM_NOTE_SUCCESS:\n // if (state.ID !== action.bookID){\n // return state;\n // }\n // return Object.assign({}, state, {\n // 'studentNotes': state.studentNotes.filter((note) => note.ID !== action.note.ID)\n // });\n\n // case types.UPDATE_BOOKSTATUS_SUCCESS:\n // if (state.ID !== action.bookID) {\n // return state;\n // } else {\n // // we send 0 when initially opening the book and -1 when closing it. Don't actually set this to the current page\n // if (action.pageIndex === 0 || action.pageIndex === -1){\n // action.pageIndex = state.currentPage ? state.currentPage : 0;\n // }\n // const book = Object.assign({}, state, { currentPage: action.pageIndex });\n // return Object.assign({}, state, book);\n // }\n\n case types.PURGE_TEMP_ID_NOTES:\n // when connecting back online, dispatch actions might have been triggered for a different book\n if (state.ID !== action.bookID) {\n return state;\n }\n return Object.assign({}, state, {\n notes: state.notes.filter((note) => !note.TempID)\n });\n\n case types.CACHE_BOOK_PAGE:\n // if (!state.cachedPages[action.page.pageKey].pageKey){\n const newCachedPage = Object.assign(\n {},\n state.cachedPages[action.page.pageKey],\n { ...action.page, pageReady: true }\n );\n return Object.assign({}, state, {\n cachedPages: {\n ...state.cachedPages,\n [action.page.pageKey]: newCachedPage\n }\n });\n // } else {\n // return state;\n // }\n case types.RESET_CACHED_BOOK_PAGES:\n return { ...state, cachedPages: {} };\n case types.NEXT_PAGE:\n return { ...state, currentPage: state.currentPage + 1 };\n case types.PREV_PAGE:\n return { ...state, currentPage: state.currentPage - 1 };\n case types.UPDATE_CURRENT_PAGE:\n return { ...state, currentPage: action.currentPage };\n case types.SET_BOOK_READY:\n return { ...state, bookIsReady: action.ready };\n case types.TOTAL_PAGE:\n return { ...state, pagecount: action.totalPage };\n case types.GET_SPEECH_MARKS:\n if (!state.cachedPages[action.pageKey]) {\n const newPage = Object.assign(\n {},\n { speechMarks: action.parsedMarks, pageReady: false }\n );\n const newCachedPages = {\n ...state.cachedPages,\n [action.pageKey]: newPage\n };\n return Object.assign({}, state, {\n cachedPages: newCachedPages\n });\n } else {\n return state;\n }\n case types.USER_LOGOUT_SUCCESS:\n return initialState.book;\n case types.RESET_BOOK:\n return initialState.book;\n\n default:\n return state;\n }\n}\n/*\n * Books Reducer\n */\nexport function booksReducer(state = initialState.books, action) {\n switch (action.type) {\n case types.LOAD_BOOKS_SUCCESS:\n return map(action.books, normalizeBook);\n case types.ADD_BOOK_TO_BOOKBAG_SUCCESS:\n // return [...state, normalizeBook(action.book)]; // don't do anything because the user will switch back and hit search again\n case types.REMOVE_BOOK_TO_BOOKBAG_SUCCESS:\n return state.filter((book) => book.ID !== action.bookID);\n case types.UPDATE_CURRENT_PAGE:\n let newBook = state.filter((b) => {\n return b.ID === action.bookID;\n })[0];\n if (!newBook) {\n return state;\n }\n return [\n ...state.filter((b) => {\n return b.ID !== action.bookID;\n }),\n Object.assign({}, newBook, { currentPage: action.currentPage })\n ];\n\n case types.USER_LOGOUT_SUCCESS:\n return initialState.books;\n\n default:\n return state;\n }\n}\n\n/*\n * Downloaded Books Reducer\n */\nexport function downloadedBooksReducer(\n state = initialState.downloadedBooks,\n action\n) {\n switch (action.type) {\n case types.DOWNLOADED_BOOK_SUCCESS:\n // filter the array for this book already on the array, it it is not there then add it\n console.error('why is it downloading')\n return [\n ...state.filter((db) => {\n return db.ID !== action.book.ID;\n }),\n Object.assign({}, normalizeBook(action.book))\n ];\n case types.LOAD_BOOKS_SUCCESS:{\n const newBooks = map(action.books, normalizeBook);\n return map(state, (book)=>{\n const foundNewBook = find(newBooks, {ID: book.ID});\n if (foundNewBook){\n return {...book, ...foundNewBook}\n } else {\n return book;\n }\n })\n }\n case types.DELETE_DOWNLOADED_BOOK: {\n return [...state.filter((db) => db.ISBN !== action.payload)];\n }\n\n case types.FETCH_BOOKITEMS_SUCCESS:\n let downloadedBook = state.filter((b) => {\n return b.ID === action.bookID;\n })[0];\n if (!downloadedBook) {\n return state;\n }\n downloadedBook = Object.assign({}, downloadedBook, {\n bookmarks: action.bookmarks,\n highlights: action.highlights,\n notes: action.notes\n });\n if (!downloadedBook.ID) {\n console.error(\n 'Error: undefined book ID in saving bookitem highlight.'\n );\n return state;\n }\n return [\n ...state.filter((b) => {\n return b.ID !== action.bookID;\n }),\n downloadedBook\n ];\n\n case types.SAVE_BOOKITEM_HIGHLIGHT_SUCCESS:\n let book = state.filter((b) => {\n return b.ID === action.bookID;\n })[0];\n if (!book) {\n return state;\n }\n book = Object.assign({}, book, { highlights: action.highlights });\n if (!book.ID) {\n console.error(\n 'Error: undefined book ID in saving bookitem highlight.'\n );\n return state;\n }\n\n // check to see if state has any length, if not return\n\n return [\n ...state.filter((b) => {\n return b.ID !== action.bookID;\n }),\n book\n ];\n\n case types.SAVE_BOOKITEM_NOTE_SUCCESS:\n let bookNote = state.filter((b) => {\n return b.ID === action.bookID;\n })[0];\n // console.log('book note', bookNote, action.bookID, state)\n if (!bookNote || (bookNote && !bookNote.notes)) {\n return state;\n }\n let notes = [];\n if (action.note.ID === undefined) {\n // this is for non-students who only save these notes locally - thus they never get a real ID\n notes = [\n ...bookNote.notes.filter(\n (note) => note.TempID !== action.note.TempID\n ),\n Object.assign({}, action.note)\n ];\n } else {\n notes = [\n ...bookNote.notes.filter(\n (note) => note.ID !== action.note.ID\n ),\n Object.assign({}, action.note)\n ];\n }\n\n notes.sort((a, b) => {\n return a.Page > b.Page;\n });\n bookNote = Object.assign({}, bookNote, { notes: notes });\n\n return [\n ...state.filter((b) => {\n return b.ID !== action.bookID;\n }),\n bookNote\n ];\n\n case types.UPDATE_CURRENT_PAGE:\n let newBook = state.filter((b) => {\n return b.ID === action.bookID;\n })[0];\n if (!newBook) {\n return state;\n }\n return [\n ...state.filter((b) => {\n return b.ID !== action.bookID;\n }),\n Object.assign({}, newBook, { currentPage: action.currentPage })\n ];\n\n case types.DELETE_BOOKITEM_BOOKMARK_SUCCESS:\n // dlBookBookmark is a downloaded book object and in this use case, we are deleting a bookmark\n let dlBookBookmark = state.filter((b) => {\n return b.ID === action.bookID;\n })[0];\n if (!dlBookBookmark) {\n return state;\n }\n Object.assign({}, dlBookBookmark, {\n bookmarks: dlBookBookmark.bookmarks.filter(\n (bookmark) => bookmark.ID !== action.bookmark.ID\n )\n });\n if (!dlBookBookmark.ID) {\n console.error(\n 'Error: undefined book ID in saving bookitem highlight.'\n );\n return state;\n }\n // if something unexpected happens return state\n return state;\n\n case types.DELETE_BOOKITEM_NOTE_SUCCESS:\n // dlBookNote is a downloaded book object and in this use case, we are deleting the note\n let dlBookNote = state.filter((b) => {\n return b.ID === action.bookID;\n })[0];\n if (!dlBookNote) {\n return state;\n }\n\n if (action.note.ID === undefined) {\n // this is for non-students who only save these notes locally - thus they never get a real ID\n notes = [\n ...dlBookNote.notes.filter(\n (note) => note.TempID !== action.note.TempID\n ),\n Object.assign({}, action.note)\n ];\n } else {\n notes = [\n ...dlBookNote.notes.filter(\n (note) => note.ID !== action.note.ID\n ),\n Object.assign({}, action.note)\n ];\n }\n if (!dlBookNote.ID) {\n console.error(\n 'Error: undefined book ID in saving bookitem highlight.'\n );\n return state;\n }\n // if something unexpected happens, return state\n return state;\n\n case types.PURGE_TEMP_ID_NOTES:\n // dlPurgeTempNotes is a downloaded book object and in this use case, we are purging any temp ID's\n let dlPurgeTempNotes = state.filter((b) => {\n return b.ID === action.bookID;\n })[0];\n if (!dlPurgeTempNotes) {\n return state;\n }\n dlPurgeTempNotes = Object.assign({}, dlPurgeTempNotes, {\n notes: dlPurgeTempNotes.notes.filter((note) => !note.TempID)\n });\n\n return [\n ...state.filter((b) => {\n return b.ID !== action.bookID;\n }),\n dlPurgeTempNotes\n ];\n\n case types.DELETE_BOOKITEM_HIGHLIGHT_SUCCESS:\n // dlBookHighlight is a downloaded book object and in this use case, we are deleting the highlight\n let dlBookHighlight = state.filter((b) => {\n return b.ID === action.bookID;\n })[0];\n if (!dlBookHighlight) {\n return state;\n }\n return Object.assign({}, dlBookHighlight, {\n highlights: dlBookHighlight.highlights\n });\n\n case types.USER_LOGOUT_SUCCESS:\n return initialState.downloadedBooks;\n\n default:\n return state;\n }\n}\n\nexport function dashboardPageResultsReducer(state = 1, action) {\n switch (action.type) {\n case types.DASHBOARD_PAGE_RESULTS:\n if (action.pageResults && action.pageResults > 0) {\n return action.pageResults;\n }\n return state;\n case types.USER_LOGOUT_SUCCESS:\n return 1;\n default:\n return state;\n }\n}\n","import * as types from '../actions/actionTypes';\nimport initialState from './initialState';\n\nfunction actionTypeEndsInSuccess(type) {\n return type.substring(type.length - 8) === '_SUCCESS';\n}\n\nfunction actionTypeEndsInFailed(type) {\n return type.substring(type.length - 7) === '_FAILED';\n}\n\nexport default function ajaxStatusReducer(\n state = initialState.ajaxCallsInProgress,\n action\n) {\n if (action.type === types.BEGIN_AJAX_CALL) {\n return state + 1;\n } else if (\n state >= 1 &&\n (actionTypeEndsInSuccess(action.type) ||\n actionTypeEndsInFailed(action.type))\n ) {\n return state - 1;\n } else if (state >= 1 && action.type === types.END_AJAX_CALL) {\n return state - 1;\n }\n\n return state;\n}\n","import { reducer as offlineQueue } from 'redux-queue-offline';\nimport { combineReducers } from 'redux';\nimport { routerReducer as routing } from 'react-router-redux';\nimport { reducer as toastr, ToastrState } from 'react-redux-toastr';\nimport user from './userReducer';\nimport { bookbagFiltersReducer as bookbagFilters } from './bookbagFiltersReducer';\nimport { dashboard } from './dashboardReducer';\n\nimport { reducer as jPlayers } from 'react-jplayer';\nimport {\n blmReducer as blm,\n blmsReducer as blms,\n completedBLMsReducer as completedBLMs\n} from './blmReducer';\nimport {\n bookReducer as book,\n booksReducer as books,\n downloadedBooksReducer as downloadedBooks,\n dashboardPageResultsReducer as dashboardPageResults\n} from './bookReducer';\nimport bookView from './bookViewReducer';\nimport { projectToolbar } from './projectToolbarReducer';\nimport bookToolbar from './bookToolbarReducer';\nimport ajaxCallsInProgress from './ajaxStatusReducer';\nimport viewerMode from './modeReducer';\nimport readium from './readiumReducer';\n\nconst rootReducer = combineReducers({\n appVersion: (state = {}) => state,\n offlineQueue,\n toastr,\n user,\n blms,\n completedBLMs,\n blm,\n book,\n books,\n downloadedBooks,\n routing,\n viewerMode,\n bookbagFilters,\n dashboard,\n ajaxCallsInProgress,\n jPlayers,\n projectToolbar,\n bookView,\n bookToolbar,\n dashboardPageResults,\n readium,\n});\n\nexport type IinitialState = ReturnType &\n { toastr: ToastrState };\n\nexport default rootReducer;\n","/*\n * Reducer for the user and user account\n */\n\nimport * as types from '../actions/actionTypes';\nimport initialState from './initialState';\n\nexport default function userReducer(state = initialState.user, action) {\n switch (action.type) {\n case types.USER_LOGIN_SUCCESS:\n case types.USER_UPDATE_SUCCESS:\n case types.USER_UPDATE_PASSWORD_SUCCESS:\n const lastCheckSession = Date.now();\n return {...state, ...action.user, lastCheckSession}\n case types.USER_LOGOUT_SUCCESS:\n return initialState.user;\n\n default:\n return state;\n }\n}\n","import * as types from '../actions/actionTypes';\nimport initialState from './initialState';\n\nexport function blmsReducer(state = initialState.blms, action) {\n switch (action.type) {\n case types.LOAD_BLMS_SUCCESS:\n return action.blms;\n\n default:\n return state;\n }\n}\n\nexport function completedBLMsReducer(\n state = initialState.completedBLMs,\n action\n) {\n switch (action.type) {\n case types.GET_COMPLETED_BLMS_SUCCESS:\n return action.completedBLMs;\n\n default:\n return state;\n }\n}\n\nexport function blmReducer(state = initialState.blm, action) {\n switch (action.type) {\n case types.FETCH_BLM_SUCCESS:\n return action.blm;\n\n case types.GET_BLM_ITEMS_SUCCESS:\n return Object.assign({}, state, {\n blmData: action.blmData,\n fabricItems: action.fabricItems,\n originalBLMTemplate: action.originalBLMTemplate\n });\n\n case types.SAVE_TEACHERS_COMMENT_SUCCESS:\n return Object.assign({}, state, {\n teachersComment: action.teachersComment\n });\n\n case types.GET_TEACHERS_BLM_COMMENT_SUCCESS:\n return Object.assign({}, state, {\n teachersComment: action.teachersComment\n });\n\n case types.GET_BLM_STATUS_SUCCESS:\n return Object.assign({}, state, {\n isComplete: action.isComplete\n });\n\n case types.SAVE_BLMITEM_FABRIC_SUCCESS:\n return Object.assign({}, state, {\n blmData: action.blmData\n });\n\n case types.UPDATE_STUDENT_ITEM_SUCCESS:\n return Object.assign({}, state, {\n fabricItemsStudent: action.fabricItemsStudent\n });\n case types.UPDATE_TEMPLATE_ITEM_SUCCESS:\n return Object.assign({}, state, {\n fabricItemsTemplate: action.fabricItemsTemplate\n });\n\n case types.UPDATE_BLM_STATUS_SUCCESS:\n return Object.assign({}, state, {\n isComplete: action.isComplete\n });\n\n case types.REMOVE_BLM_DATA_SUCCESS:\n return action.blm;\n case types.SET_PROJECT_READY:\n return { ...state, projectIsReady: action.ready };\n\n case types.USER_LOGOUT_SUCCESS:\n return initialState.blms;\n\n default:\n return state;\n }\n}\n","import * as types from '../actions/actionTypes';\nimport initialState from './initialState';\n\nexport default function modeReducer(state = initialState.viewerMode, action) {\n switch (action.type) {\n case types.UPDATE_MODE_SUCCESS:\n return action.viewerMode;\n\n case types.USER_LOGOUT_SUCCESS:\n return initialState.viewerMode;\n\n default:\n return state;\n }\n}\n","import * as types from '../actions/actionTypes';\nimport initialState from './initialState';\n\nexport function bookbagFiltersReducer(\n state = initialState.bookbagFilters,\n action\n) {\n switch (action.type) {\n case types.SAVE_ADVANCED_SEARCH_FILTERS:\n return { ...state, ...action.filters };\n // return Array.from(new Set(action.filters.tags));\n case types.CLEAR_ADVANCED_SEARCH_FILTERS:\n return initialState.bookbagFilters;\n case types.USER_LOGOUT_SUCCESS:\n return initialState.bookbagFilters;\n\n default:\n return state;\n }\n}\n","/*\n * This reducer contains all the visual state of the dashboard\n */\n\nimport * as types from '../actions/actionTypes';\nimport initialState from './initialState';\n\nexport function dashboard(state = initialState.dashboard, action) {\n switch (action.type) {\n case types.TOGGLE_USER_PROFILE_MODAL:\n return {\n ...state,\n showUserProfileModal: !state.showUserProfileModal\n };\n\n case types.TOGGLE_LOGGOUT_CONFIRM_MODAL:\n return {\n ...state,\n showLoggoutConfirmModal: !state.showLoggoutConfirmModal\n };\n case types.TOGGLE_CLASS_CODE_MODAL:\n let showClassCodeModal = !state.showClassCodeModal;\n if (action.show && typeof action.show === 'boolean'){\n showClassCodeModal = action.show;\n }\n return {\n ...state,\n showClassCodeModal\n };\n case types.USER_LOGOUT_SUCCESS:\n return initialState.dashboard;\n\n default:\n return state;\n }\n}\n","import * as types from '../actions/actionTypes';\nimport initialState from './initialState';\n\nexport function projectToolbar(state = initialState.projectToolbar, action) {\n switch (action.type) {\n case types.TOOLBAR_ERASER_ON:\n return Object.assign({}, state, action.projectToolbar);\n case types.TOOLBAR_POINTER_ON:\n return Object.assign({}, state, action.projectToolbar);\n case types.TOOLBAR_DRAWING_MODE_ON:\n return Object.assign({}, state, action.projectToolbar);\n case types.TOOLBAR_ADD_TEXT_ON:\n return Object.assign({}, state, action.projectToolbar);\n case types.SHOW_RICH_TEXT_BUTTONS:\n return Object.assign({}, state, action.projectToolbar);\n default:\n return state;\n }\n}\n","/*\n * This reducer contains all the visual state of the viewer\n */\n\nimport * as types from '../actions/actionTypes';\nimport initialState from './initialState';\nimport constants from '../constants/constants';\n\nexport default function bookViewReducer(state = initialState.bookView, action) {\n // dispatch all redux events as document events - probably a bad idea\n document.dispatchEvent(new CustomEvent(action.type, {detail: action}))\n switch (action.type) {\n case types.USER_LOGOUT_SUCCESS:\n return initialState.bookView;\n case types.RESET_VIEW:\n return initialState.bookView;\n case types.UPDATE_BOOK_SCALE_PERCENT:\n return { ...state, bookScalePercent: action.scale };\n case types.UPDATE_PROJECT_SCALE_PERCENT:\n return { ...state, projectScalePercent: action.scale };\n case types.UPDATE_LEFT_PAGE_CONTAINER:\n return {\n ...state,\n leftPageContainerWidth: action.width,\n leftPageContainerHeight: action.height\n };\n case types.UPDATE_PAGES_VISIBLE:\n return { ...state, pagesVisible: action.pagesVisible };\n case types.AUTOMATIC_UPDATE_PAGES_VISIBLE:\n if (state.automaticUpdatedPagesVisible) {\n return state;\n }\n return {\n ...state,\n pagesVisible: action.pagesVisible,\n automaticUpdatedPagesVisible: true\n };\n\n case types.BOOK_TOOLBAR_UPDATE_MODE:\n return { ...state, bookToolbar: action.bookToolbar };\n case types.SET_ACTIVE_NOTE:\n return { ...state, activeNoteID: action.noteID };\n case types.INCREASE_BOOK_ZOOM:\n const newZoomLevelIndex = state.bookManualZoomIndex + 1;\n const newZoomLevel = constants.bookZoomLevels[newZoomLevelIndex];\n if (newZoomLevel === undefined) {\n return state;\n }\n if (\n newZoomLevelIndex === initialState.bookView.bookManualZoomIndex\n ) {\n return {\n ...state,\n bookManualZooming: false,\n bookManualZoomIndex: newZoomLevelIndex,\n bookManualZoomLevel: newZoomLevel / 100\n };\n } else {\n return {\n ...state,\n bookManualZooming: true,\n bookManualZoomIndex: newZoomLevelIndex,\n bookManualZoomLevel: newZoomLevel / 100\n };\n }\n case types.DECREASE_BOOK_ZOOM:\n const newZoomLevelIndexD = state.bookManualZoomIndex - 1;\n const newZoomLevelD = constants.bookZoomLevels[newZoomLevelIndexD];\n if (newZoomLevelD === undefined) {\n return state;\n }\n if (\n newZoomLevelIndexD === initialState.bookView.bookManualZoomIndex\n ) {\n return {\n ...state,\n bookManualZooming: false,\n bookManualZoomIndex: newZoomLevelIndexD,\n bookManualZoomLevel: newZoomLevelD / 100\n };\n } else {\n return {\n ...state,\n bookManualZooming: true,\n bookManualZoomIndex: newZoomLevelIndexD,\n bookManualZoomLevel: newZoomLevelD / 100\n };\n }\n case types.UPDATE_PENDING_ITEM:\n return { ...state, itemSavePending: action.status };\n\n // case types.BOOK_UPDATE_AUDIO:\n // return { ...state, audioUrl: action.audioUrl}\n\n default:\n return state;\n }\n}\n","/*\n * This reducer contains all the state for managing the book toolbar\n */\n\nimport * as types from '../actions/actionTypes';\nimport initialState from './initialState';\n\nexport default function bookToolbar(state = initialState.bookToolbar, action) {\n switch (action.type) {\n case types.USER_LOGOUT_SUCCESS:\n return initialState.bookToolbar;\n case types.BOOK_TOOLBAR_POINTING:\n return initialState.bookToolbar;\n case types.BOOK_TOOLBAR_NOTES:\n return {\n ...initialState.bookToolbar,\n allowNotes: action.allowNotes,\n pointing: false,\n pagesClassName: 'note-mode'\n };\n case types.BOOK_TOOLBAR_HIGHLIGHTING:\n return {\n ...initialState.bookToolbar,\n highlighting: action.highlighting,\n highlightColor: action.highlightColor,\n pointing: false,\n pagesClassName: `hl-mode-${action.highlightColor}`\n };\n case types.BOOK_TOOLBAR_ERASING:\n return {\n ...initialState.bookToolbar,\n erasing: action.erasing,\n pointing: false,\n pagesClassName: 'eraser-mode'\n };\n case types.BOOK_TOOLBAR_UNDERLINING:\n return {\n ...initialState.bookToolbar,\n underlining: action.underlining,\n pointing: false,\n pagesClassName: 'hl-underline'\n };\n case types.BOOK_TOOLBAR_STRIKING:\n return {\n ...initialState.bookToolbar,\n striking: action.striking,\n pointing: false,\n pagesClassName: 'hl-strikethrough'\n };\n case types.BOOK_TOOLBAR_UPDATE_MODE:\n return { ...state, bookToolbar: action.bookToolbar };\n case types.BOOK_TOOLBAR_MOVING:\n return {\n ...initialState.bookToolbar,\n moving: true,\n pointing: true,\n pagesClassName: 'move-mode'\n };\n default:\n return state;\n }\n}\n","import { READIUM_INITIALIZED } from '../actions/actionTypes';\nimport initialState from './initialState';\n\nexport default function readiumReducer(state = initialState.readium, action) {\n switch (action.type) {\n case READIUM_INITIALIZED:\n return { ...state, readiumClass: action.readiumClass };\n default:\n return state;\n }\n}\n","import { IinitialState } from '../reducers/index';\nimport initialState from '../reducers/initialState';\n\n/*\n * Migration 1 runs when upgrading from 0 to 1\n */\n\nconst typedMigrations = {\n 162: (state: IinitialState): IinitialState => {\n console.log('running migration 162', state.appVersion);\n try {\n const serializedState = localStorage.getItem(`state-dibs`);\n if (serializedState !== null) {\n console.log(\"migrating previous state\")\n localStorage.removeItem('state-dibs');\n return {\n ...JSON.parse(serializedState),\n ...state\n };\n }\n } catch {\n console.log('no previous state');\n }\n const { downloadedBooks } = initialState;\n return {\n ...state,\n downloadedBooks\n };\n }\n} as { [key: number]: (state: IinitialState) => IinitialState };\n\nconst untypedMigrations = typedMigrations as any;\nexport const migrations = untypedMigrations;\n","import { middleware as offlineQueueMiddleware } from 'redux-queue-offline';\nimport { createStore, applyMiddleware, compose } from 'redux';\nimport rootReducer from '../reducers';\nimport thunk from 'redux-thunk';\nimport * as createMigration from 'redux-persist-migrate';\nimport initialState from '../reducers/initialState';\nimport {persistStore, autoRehydrate} from 'redux-persist';\n\nimport {\n saveBookItem,\n updateBookStatus,\n deleteBookItem\n} from '../actions/bookActions';\nimport { saveBLMItem } from '../actions/blmActions';\nimport { composeWithDevTools } from 'redux-devtools-extension/developmentOnly';\nimport BookFS from '../api/bookFS';\nimport { migrations } from './migrations';\n\nexport const persistConfig = {\n // keyPrefix: 'dibs',\n debounce: 300,\n storage: BookFS.fs,\n blacklist: ['showEditProfileModal', 'ajaxCallsInProgress', 'toastr']\n // transforms: [createTransform(encode, decode)]\n};\n// see https://github.com/diegoddox/react-redux-toastr/issues/249 about blacklisting toastr\nconst migration = createMigration.default(migrations, 'appVersion');\n\nconst offlineActionCreators = {\n saveBookItem: saveBookItem,\n updateBookStatus: updateBookStatus,\n deleteBookItem: deleteBookItem,\n saveBLMItem: saveBLMItem\n};\n\n\nexport const configureStore = (persistCallback) => {\n if (process.env.NODE_ENV !== 'production') {\n // if (false){ // comment in for testing localling without redux dev tools\n\n const store = createStore(\n rootReducer,\n initialState,\n composeWithDevTools(\n applyMiddleware(\n offlineQueueMiddleware(\n 'offlineQueue',\n ['payload.promise'],\n offlineActionCreators\n ),\n thunk\n ),\n compose(autoRehydrate(), migration) // , require('redux-immutable-state-invariant').default()\n )\n );\n const persistor = persistStore(store, persistConfig, persistCallback);\n return {store, persistor};\n } else {\n const store = createStore(\n rootReducer,\n initialState,\n compose(applyMiddleware(\n offlineQueueMiddleware(\n 'offlineQueue',\n ['payload.promise'],\n offlineActionCreators\n ),\n thunk,\n ),\n autoRehydrate(), migration)\n );\n const persistor = persistStore(store, persistConfig, persistCallback);\n return {store, persistor};\n }\n}\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://cra.link/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://cra.link/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://cra.link/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 &&\n 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.message);\n });\n }\n}\n","const reportWebVitals = (onPerfEntry) => {\n if (onPerfEntry && onPerfEntry instanceof Function) {\n import('web-vitals').then(\n ({ getCLS, getFID, getFCP, getLCP, getTTFB }) => {\n getCLS(onPerfEntry);\n getFID(onPerfEntry);\n getFCP(onPerfEntry);\n getLCP(onPerfEntry);\n getTTFB(onPerfEntry);\n }\n );\n }\n};\n\nexport default reportWebVitals;\n","import { READIUM_INITIALIZED } from './actionTypes';\n\nexport const readiumInitialized = (readiumClass) => {\n return { type: READIUM_INITIALIZED, readiumClass };\n};\n","/* eslint-disable */\nimport * as React from 'react';\nimport * as ReactDOM from 'react-dom';\nimport constants from './constants/constants';\n\nimport { NetworkListener } from './utilities/NetworkListener';\nimport { Provider } from 'react-redux';\nimport { Router, hashHistory } from 'react-router';\nimport routes from './routes';\nimport { Logger, CL_WARN, CL_ERROR } from './api/logger';\nimport config from './api/config';\nimport ReduxToastr from 'react-redux-toastr';\nimport {configureStore} from './store/configureStore';\n\nimport 'whatwg-fetch';\n// import 'bootstrap/dist/css/bootstrap.min.css';\nimport 'font-awesome/css/font-awesome.min.css';\nimport './App.scss';\nimport 'bootstrap/dist/js/bootstrap.min.js';\nimport './vendor/animate.min.css';\nimport './vendor/toastr.min.css';\nimport './vendor/reactReduxToaster.min.css';\nimport './vendor/fonts.css';\nimport * as serviceWorkerRegistration from './serviceWorkerRegistration';\nimport reportWebVitals from './reportWebVitals';\nimport { readiumInitialized } from './actions/readiumActions';\n\n// print version\nconsole.log(\n `***** Version: ${config.Version} **** Environment: ${config.Environment}`\n);\n\nconst NetworkListenerProvider = NetworkListener(Provider);\n\nconstants.themeProvider\n.initTheme()\n\nclass IndexClass extends React.Component {\n constructor(props) {\n super(props);\n this.state = { isRehydrated: false };\n const {store, persistor} = configureStore(() => {\n this.setState({ isRehydrated: true });\n });\n this.store = store;\n this.persistor = persistor;\n this.listenToReadiumInit();\n }\n render() {\n if (!this.state.isRehydrated) {\n return null;\n }\n return (\n // \n \n \n \n \n
\n \n // \n );\n }\n\n listenToReadiumInit() {\n if (window.readiumClass) {\n this.store.dispatch(readiumInitialized(window.readiumClass));\n return;\n }\n const handler = (ev) => {\n if (ev.data === 'readium-initialized') {\n window.removeEventListener('message', handler);\n this.store.dispatch(readiumInitialized(window.readiumClass));\n }\n };\n window.addEventListener('message', handler);\n }\n}\n\nReactDOM.render(\n ,\n document.getElementById('root')\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://cra.link/PWA\nserviceWorkerRegistration.register({\n onUpdate: (registration) => {\n document.dispatchEvent(new Event('newVersionAvailable'));\n if (registration && registration.waiting) {\n registration.waiting.postMessage({ type: 'SKIP_WAITING' });\n }\n }\n});\n// If you want to start measuring performance in your app, pass a function\n// to log results (for example: reportWebVitals(console.log))\n// or send to an analytics endpoint. Learn more: https://bit.ly/CRA-vitals\nreportWebVitals();\n\nwindow.console = (function (origConsole) {\n if (!window.console || !origConsole) origConsole = {};\n var isDebug = false,\n logArray = {\n logs: [],\n errors: [],\n warns: [],\n infos: []\n };\n return {\n log: function () {\n logArray.logs.push(arguments);\n // Logger.log(CL_DEBUG, ...arguments);\n isDebug &&\n origConsole.log &&\n origConsole.log.apply(origConsole, arguments);\n },\n warn: function () {\n logArray.warns.push(arguments);\n Logger.log(CL_WARN, ...arguments);\n isDebug &&\n origConsole.warn &&\n origConsole.warn.apply(origConsole, arguments);\n },\n error: function () {\n logArray.errors.push(arguments);\n Logger.log(CL_ERROR, ...arguments);\n isDebug &&\n origConsole.error &&\n origConsole.error.apply(origConsole, arguments);\n },\n info: function (v) {\n logArray.infos.push(arguments);\n // Logger.log(CL_INFO, ...arguments);\n isDebug &&\n origConsole.info &&\n origConsole.info.apply(origConsole, arguments);\n },\n debug: function (bool) {\n isDebug = bool;\n },\n logArray: function () {\n return logArray;\n }\n };\n})(window.console);\nwindow.console.debug(true);\n","import * as React from 'react';\nimport { ONLINE, OFFLINE } from 'redux-queue-offline';\n\nexport const NetworkListener = (Provider) => {\n return class NetworkListenerComponent extends React.Component {\n defaultProps = { store: {} };\n componentDidMount() {\n if (window && window.addEventListener) {\n window.addEventListener('online', this._onlineListener);\n window.addEventListener('offline', this._offlineListener);\n }\n }\n componentWillUnmount() {\n if (window && window.removeEventListener) {\n window.removeEventListener('online', this._onlineListener);\n window.removeEventListener('offline', this._offlineListener);\n }\n }\n render() {\n return React.createElement(Provider, this.props);\n }\n _onlineListener = () => {\n this.props.store.dispatch({\n type: ONLINE\n });\n };\n _offlineListener = () => {\n this.props.store.dispatch({\n type: OFFLINE\n });\n };\n };\n};\n"],"sourceRoot":""}