{"version":3,"file":"proofreader-DuTniLFn.js","sources":["../../../client/app/bundles/Proofreader/helpers/getParameterByName.ts","../../../client/app/bundles/Proofreader/actions/actionTypes.ts","../../../client/app/bundles/Proofreader/actions/session.ts","../../../client/app/bundles/Proofreader/components/Header.tsx","../../../client/app/bundles/Proofreader/components/PageLayout.tsx","../../../client/app/bundles/Proofreader/lib/proofreader_activities_api.ts","../../../client/app/bundles/Proofreader/actions/proofreaderActivities.ts","../../../client/app/bundles/Proofreader/components/admin/tabLink.tsx","../../../client/app/bundles/Proofreader/actions/concepts.ts","../../../client/app/bundles/Proofreader/components/concepts/concepts.tsx","../../../client/app/bundles/Proofreader/components/lessons/editGenerator.tsx","../../../client/app/bundles/Proofreader/components/lessons/lessonForm.tsx","../../../client/node_modules/sbd/lib/sanitize-html-browser.js","../../../client/node_modules/sbd/lib/stringHelper.js","../../../client/node_modules/sbd/lib/Match.js","../../../client/node_modules/sbd/lib/tokenizer.js","../../../client/app/bundles/Proofreader/components/proofreaderActivities/earlySubmitModal.tsx","../../../client/app/bundles/Proofreader/components/proofreaderActivities/followupModal.tsx","../../../client/app/bundles/Proofreader/components/proofreaderActivities/formatInitialPassage.ts","../../../client/app/bundles/Proofreader/components/proofreaderActivities/editInput.tsx","../../../client/app/bundles/Proofreader/components/proofreaderActivities/paragraph.tsx","../../../client/app/bundles/Proofreader/helpers/determineUnnecessaryEditType.ts","../../../client/app/bundles/Proofreader/components/proofreaderActivities/editTooltip.tsx","../../../client/app/bundles/Proofreader/components/proofreaderActivities/edit.tsx","../../../client/app/bundles/Proofreader/components/proofreaderActivities/sharedRegexes.ts","../../../client/app/bundles/Proofreader/components/proofreaderActivities/passageReviewer.tsx","../../../client/app/bundles/Proofreader/components/proofreaderActivities/progressBar.tsx","../../../client/app/bundles/Proofreader/components/proofreaderActivities/resetModal.tsx","../../../client/app/bundles/Proofreader/components/proofreaderActivities/reviewModal.tsx","../../../client/app/bundles/Proofreader/components/proofreaderActivities/welcomePage.tsx","../../../client/app/bundles/Proofreader/helpers/EditCaretPositioning.ts","../../../client/app/bundles/Proofreader/components/shared/loading_spinner.tsx","../../../client/app/bundles/Proofreader/components/proofreaderActivities/container.tsx","../../../client/app/bundles/Proofreader/components/lessons/lesson.tsx","../../../client/app/bundles/Proofreader/components/shared/linkListItem.tsx","../../../client/app/bundles/Proofreader/components/lessons/lessons.tsx","../../../client/app/bundles/Proofreader/components/admin/admin.tsx","../../../client/app/bundles/Proofreader/routes.tsx","../../../client/app/bundles/Proofreader/actions/actions.ts","../../../client/app/bundles/Proofreader/reducers/conceptsReducer.ts","../../../client/app/bundles/Proofreader/reducers/proofreaderActivitiesReducer.ts","../../../client/app/bundles/Proofreader/reducers/sessionReducer.ts","../../../client/app/bundles/Proofreader/reducers/rootReducer.ts","../../../client/app/bundles/Proofreader/store/configStore.ts","../../../client/app/bundles/Proofreader/App.tsx","../../../client/app/bundles/Proofreader/clientRegistration.js"],"sourcesContent":["export default function getParameterByName(name: string, url: string) {\n if (!url) { url = window.location.href; }\n name = name.replace(/[\\[\\]]/g, '\\\\$&');\n const regex = new RegExp(`[?&]${name}(=([^]*)|&|#|$)`),\n results = regex.exec(url);\n if (!results) { return null; }\n if (!results[2]) { return ''; }\n return decodeURIComponent(results[2].replace(/\\+/g, ' '));\n}\n","export const ActionTypes = {\n // INIT STORE\n INIT_STORE: 'INIT_STORE',\n\n // PROOFREADER ACTIVITIES\n RECEIVE_PROOFREADER_ACTIVITY_DATA: 'RECEIVE_PROOFREADER_ACTIVITY_DATA',\n NO_PROOFREADER_ACTIVITY_FOUND: 'NO_PROOFREADER_ACTIVITY_FOUND',\n RECEIVE_PROOFREADER_ACTIVITIES_DATA: 'RECEIVE_PROOFREADER_ACTIVITIES_DATA',\n NO_PROOFREADER_ACTIVITIES_FOUND: 'NO_PROOFREADER_ACTIVITIES_FOUND',\n TOGGLE_LESSON_FORM: 'TOGGLE_LESSON_FORM',\n AWAIT_NEW_LESSON_RESPONSE: 'AWAIT_NEW_LESSON_RESPONSE',\n RECEIVE_NEW_LESSON_RESPONSE: 'RECEIVE_NEW_LESSON_RESPONSE',\n START_LESSON_EDIT: 'START_LESSON_EDIT',\n SUBMIT_LESSON_EDIT: 'SUBMIT_LESSON_EDIT',\n FINISH_LESSON_EDIT: 'FINISH_LESSON_EDIT',\n EDITING_LESSON: 'EDITING_LESSON',\n SUBMITTING_LESSON: 'SUBMITTING_LESSON',\n\n // QUESTIONS\n SUBMIT_EDIT: 'SUBMIT_RESPONSE',\n\n // SESSIONS\n SET_FIREBASE_PASSAGE: 'SET_FIREBASE_PASSAGE',\n SET_PASSAGE: 'SET_PASSAGE',\n SET_TIMETRACKING: 'SET_TIMETRACKING',\n\n // CONCEPTS,\n RECEIVE_CONCEPTS_DATA: 'RECEIVE_CONCEPTS_DATA',\n\n // DISPLAY\n DISPLAY_ERROR: 'DISPLAY_ERROR',\n DISPLAY_MESSAGE: 'DISPLAY_MESSAGE',\n CLEAR_DISPLAY_MESSAGE_AND_ERROR: 'CLEAR_DISPLAY_MESSAGE_AND_ERROR',\n};\n","import { v4 as uuid } from 'uuid';\n\nimport { ActionTypes } from './actionTypes';\n\nimport { ConceptResultObject, WordObject } from '../interfaces/proofreaderActivities';\nimport { SessionApi } from '../../Shared';\n\nexport const updateSessionOnFirebase = (sessionID: string, session: { passage: Array \n \n {this.state.generatedEdit}\n \n \n \n \n \n If there are multiple correct edits, type them into the following field separated by a tilde (~). Example: if \"loves\" and \"adores\" are both correct, you would enter \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n loves~adores
.\n \n
')\n submit({\n title,\n description,\n flag,\n passage: formattedPassage,\n underlineErrorsInProofreader,\n readingLevel\n });\n }\n\n handleTitleChange = (e: React.ChangeEvent<{value: string}>) => {\n this.setState({ title: e.target.value, });\n }\n\n handleReadingLevelChange = (e: React.ChangeEvent<{value: string}>) => {\n this.setState({ readingLevel: e.target.value, });\n }\n\n handleFlagSelect = (e: React.ChangeEvent<{value: string}>) => {\n this.setState({ flag: e.target.value, });\n }\n\n handleToggleUnderline = () => {\n const { underlineErrorsInProofreader } = this.state;\n this.setState({ underlineErrorsInProofreader: !underlineErrorsInProofreader });\n }\n\n onHandleDescriptionChange = (e: string) => {\n this.setState({ description: e, });\n }\n\n handlePassageChange = (e: React.ChangeEvent<{value: string}>) => {\n this.setState({ passage: e.target.value, });\n }\n\n render() {\n const { currentValues, returnToView, stateSpecificClass } = this.props;\n const { title, readingLevel, description, flag, underlineErrorsInProofreader, passage } = this.state;\n const addOrEdit = currentValues ? 'Edit' : 'Add';\n const buttonText = currentValues ? 'Return To Activity' : 'Return To Activities';\n return (\n {addOrEdit} Activity
\n
\n
|<\\\\/(\" + options.html_boundaries_tags.join(\"|\") + \")>)\";\r\n var re = new RegExp(html_boundaries_regexp, \"g\");\r\n text = text.replace(re, \"$1\" + newline_placeholder);\r\n }\r\n\r\n if (options.sanitize || options.allowed_tags) {\r\n if (! options.allowed_tags) {\r\n options.allowed_tags = [\"\"];\r\n }\r\n\r\n text = sanitizeHtml(text, { \"allowedTags\" : options.allowed_tags });\r\n }\r\n\r\n\r\n // Split the text into words\r\n var words;\r\n var tokens;\r\n\r\n // Split the text into words\r\n if (options.preserve_whitespace) {\r\n //
tags are the odd man out, as whitespace is allowed inside the tag\r\n tokens = text.split(/(
|\\S+|\\n+)/);\r\n\r\n // every other token is a word\r\n words = tokens.filter(function (token, ii) {\r\n return ii % 2;\r\n });\r\n }\r\n else {\r\n // - see http://blog.tompawlak.org/split-string-into-tokens-javascript\r\n words = text.trim().match(splitIntoWords);\r\n }\r\n\r\n\r\n var wordCount = 0;\r\n var index = 0;\r\n var temp = [];\r\n var sentences = [];\r\n var current = [];\r\n\r\n // If given text is only whitespace (or nothing of \\S+)\r\n if (!words || !words.length) {\r\n return [];\r\n }\r\n\r\n for (var i=0, L=words.length; i < L; i++) {\r\n wordCount++;\r\n\r\n // Add the word to current sentence\r\n current.push(words[i]);\r\n\r\n // Sub-sentences, reset counter\r\n if (~words[i].indexOf(\",\")) {\r\n wordCount = 0;\r\n }\r\n\r\n if (Match.isBoundaryChar(words[i]) || stringHelper.endsWithChar(words[i], \"?!\") || words[i] === newline_placeholder_t) {\r\n if ((options.newline_boundaries || options.html_boundaries) && words[i] === newline_placeholder_t) {\r\n current.pop();\r\n }\r\n\r\n sentences.push(current);\r\n\r\n wordCount = 0;\r\n current = [];\r\n\r\n continue;\r\n }\r\n\r\n\r\n if (stringHelper.endsWithChar(words[i], \"\\\"\") || stringHelper.endsWithChar(words[i], \"”\")) {\r\n words[i] = words[i].slice(0, -1);\r\n }\r\n\r\n // A dot might indicate the end sentences\r\n // Exception: The next sentence starts with a word (non abbreviation)\r\n // that has a capital letter.\r\n if (stringHelper.endsWithChar(words[i], \".\")) {\r\n // Check if there is a next word\r\n // This probably needs to be improved with machine learning\r\n if (i+1 < L) {\r\n // Single character abbr.\r\n if (words[i].length === 2 && isNaN(words[i].charAt(0))) {\r\n continue;\r\n }\r\n\r\n // Common abbr. that often do not end sentences\r\n if (Match.isCommonAbbreviation(words[i])) {\r\n continue;\r\n }\r\n\r\n // Next word starts with capital word, but current sentence is\r\n // quite short\r\n if (Match.isSentenceStarter(words[i+1])) {\r\n if (Match.isTimeAbbreviation(words[i], words[i+1])) {\r\n continue;\r\n }\r\n\r\n // Dealing with names at the start of sentences\r\n if (Match.isNameAbbreviation(wordCount, words.slice(i, 6))) {\r\n continue;\r\n }\r\n\r\n if (Match.isNumber(words[i+1])) {\r\n if (Match.isCustomAbbreviation(words[i])) {\r\n continue;\r\n }\r\n }\r\n }\r\n else {\r\n // Skip ellipsis\r\n if (stringHelper.endsWith(words[i], \"..\")) {\r\n continue;\r\n }\r\n\r\n //// Skip abbreviations\r\n // Short words + dot or a dot after each letter\r\n if (Match.isDottedAbbreviation(words[i])) {\r\n continue;\r\n }\r\n\r\n if (Match.isNameAbbreviation(wordCount, words.slice(i, 5))) {\r\n continue;\r\n }\r\n }\r\n }\r\n\r\n sentences.push(current);\r\n current = [];\r\n wordCount = 0;\r\n\r\n continue;\r\n }\r\n\r\n // Check if the word has a dot in it\r\n if ((index = words[i].indexOf(\".\")) > -1) {\r\n if (Match.isNumber(words[i], index)) {\r\n continue;\r\n }\r\n\r\n // Custom dotted abbreviations (like K.L.M or I.C.T)\r\n if (Match.isDottedAbbreviation(words[i])) {\r\n continue;\r\n }\r\n\r\n // Skip urls / emails and the like\r\n if (Match.isURL(words[i]) || Match.isPhoneNr(words[i])) {\r\n continue;\r\n }\r\n }\r\n\r\n if (temp = Match.isConcatenated(words[i])) {\r\n current.pop();\r\n current.push(temp[0]);\r\n sentences.push(current);\r\n\r\n current = [];\r\n wordCount = 0;\r\n current.push(temp[1]);\r\n }\r\n }\r\n\r\n if (current.length) {\r\n sentences.push(current);\r\n }\r\n\r\n\r\n // Clear \"empty\" sentences\r\n sentences = sentences.filter(function(s) {\r\n return s.length > 0;\r\n });\r\n\r\n var result = sentences.slice(1).reduce(function (out, sentence) {\r\n var lastSentence = out[out.length - 1];\r\n\r\n // Single words, could be \"enumeration lists\"\r\n if (lastSentence.length === 1 && /^.{1,2}[.]$/.test(lastSentence[0])) {\r\n // Check if there is a next sentence\r\n // It should not be another list item\r\n if (!/[.]/.test(sentence[0])) {\r\n out.pop()\r\n out.push(lastSentence.concat(sentence));\r\n return out;\r\n }\r\n }\r\n\r\n out.push(sentence);\r\n\r\n return out;\r\n }, [ sentences[0] ]);\r\n\r\n // join tokens back together\r\n return result.map(function (sentence, ii) {\r\n if (options.preserve_whitespace && !options.newline_boundaries && !options.html_boundaries) {\r\n // tokens looks like so: [leading-space token, non-space token, space\r\n // token, non-space token, space token... ]. In other words, the first\r\n // item is the leading space (or the empty string), and the rest of\r\n // the tokens are [non-space, space] token pairs.\r\n var tokenCount = sentence.length * 2;\r\n\r\n if (ii === 0) {\r\n tokenCount += 1;\r\n }\r\n\r\n return tokens.splice(0, tokenCount).join(\"\");\r\n }\r\n\r\n return sentence.join(\" \");\r\n });\r\n};\r\n","import * as React from 'react'\n\nimport useModalAccessibility from '../../../Shared/hooks/useModalAccessibility'\n\ninterface EarlySubmitModalProps {\n closeModal: any,\n requiredEditCount: number\n}\n\nconst EarlySubmitModal = ({ closeModal, requiredEditCount, }: EarlySubmitModalProps) => {\n const { modalRef } = useModalAccessibility(closeModal);\n\n return (\n Keep looking! You must make at least {requiredEditCount} edits.
\n \n That's the end of this passage! Now let's do some follow-up practice.
\n
', '
').replace(/
|<\\/p>/g, '').split(' Explanation {correctAnswers} Correct {explanation} The correct text was {displayText}. {incorrectText && `You submitted ${incorrectText === ' ' ? 'An empty space.' : incorrectText}.`}
')\n let necessaryEditCounter = 0\n let paragraphIndex = 0\n const passageArray = paragraphs.map((paragraph: string) => {\n if (paragraph.length === 0) {\n return null\n }\n let i = 0\n const paragraphArray = paragraph.split(/{|}/).map((text) => {\n let wordObj, wordArray\n if (necessaryEditRegex.test(text)) {\n wordObj = {\n originalText: text.match(originalTextRegex) ? text.match(originalTextRegex)[1] : '',\n currentText: text.match(originalTextRegex) ? text.match(originalTextRegex)[1] : '',\n necessaryEditIndex: necessaryEditCounter,\n conceptUID: text.match(conceptUIDRegex) ? text.match(conceptUIDRegex)[1] : '',\n correctText: text.match(correctEditRegex) ? text.match(correctEditRegex)[1] : '',\n underlined: true,\n wordIndex: i,\n paragraphIndex\n }\n wordArray = [wordObj]\n necessaryEditCounter+=1\n i+=1\n } else {\n wordArray = text.split(/\\s+/).map(word => {\n if (word.length === 0) {\n return null\n }\n wordObj = {\n originalText: word,\n currentText: word,\n correctText: word,\n underlined: false,\n wordIndex: i,\n paragraphIndex\n }\n i+=1\n return wordObj\n })\n }\n return wordArray.filter(Boolean)\n })\n paragraphIndex+=1\n return _.flatten(paragraphArray)\n })\n return {passage: passageArray.filter(Boolean), necessaryEdits}\n}\n\nexport default formatInitialPassage\n","import * as React from 'react';\nimport ContentEditable from 'react-contenteditable';\n\nimport { WordObject } from '../../interfaces/proofreaderActivities';\n\ntype EditInputProps = WordObject & { onWordChange: Function, numberOfResets: number, underlineErrors: boolean }\n\nexport default class EditInput extends React.Component\n
{headerText}
\n
')\n const { activeIndex, numberOfEdits } = this.state\n let index = 0\n return paragraphs.map((paragraph: string, paragraphIndex: number) => {\n const parts: Array |<\\/p>/g, '').split(/{|}/g)\n for (let i = 0; i < parts.length; i +=1) {\n if (typeof parts[i] === \"string\" && parts[i][0] === '+') {\n const plusMatch = parts[i].match(/\\+([^-]+)-/m)\n const plus = plusMatch ? plusMatch[1] : ''\n const conceptUIDMatch = parts[i].match(/\\|(.+)/m)\n const conceptUID = conceptUIDMatch ? conceptUIDMatch[1] : ''\n const negativeMatch = parts[i].match(negativeMatchRegex)\n const negative = negativeMatch ? negativeMatch[1] : null\n const concept = concepts.find(c => c.uid === conceptUID)\n const indexToPass = index\n let state = 'correct'\n if (unnecessaryArray.includes(conceptUID)) {\n state = conceptUID\n } else if (negative) {\n state = 'incorrect'\n }\n index+=1\n parts[i] = ( {parts} No passage {answeredQuestionCount} of {questionCount} edits made This will undo all the edits you made and reset the passage to its original state. Read the passage and correct the grammar and punctuation errors you find. Get feedback on each correction you made. Work on a follow-up activity that focuses on one of the errors from the passage. ').concat(joinWords(words.filter(word => word.length))).concat(' Screenreader users: once you have finished reading the passage, use the tab keys to navigate between words and make changes to ones that have errors. {currentActivity.underlineErrorsInProofreader && 'Words that contain errors will be described as underlined.'} Words that you have already changed will be described as bolded. There are {necessaryEditsLength} errors to find and fix. When you are done, navigate to the "Get Feedback" button after the passage and select it. Loading... \n \n \n Note: the activity below functions exactly like the real activity, except that you will not be directed to a results page or to a Grammar activity upon finishing the review. Instead, the activity will reset. 404: No Concept FoundReset the passage?
\n {highScoreMessage} You found {numberOfCorrectChanges} of {numberOfErrors} errors. Let's review your edits.
\n \n Welcome to Quill Proofreader!
\n Read
\n \n
Review
\n \n
Improve
\n \n
\n
{currentActivity.title}
\n \n \n
\n {data[lessonID].title}
\n Flag: {data[lessonID].flag}
\n Play Proofreader Activity
\n