@pie-element/multiple-choice 9.18.0 → 9.18.1-beta.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +32 -0
- package/configure/CHANGELOG.md +32 -0
- package/configure/lib/__tests__/root.test.js +271 -0
- package/configure/lib/index.js +1 -1
- package/configure/lib/index.js.map +1 -1
- package/configure/lib/main.js +6 -6
- package/configure/lib/main.js.map +1 -1
- package/configure/package.json +4 -2
- package/controller/CHANGELOG.md +32 -0
- package/controller/lib/__tests__/index.test.js +867 -0
- package/controller/lib/__tests__/utils.test.js +8 -0
- package/controller/lib/index.js +1 -1
- package/controller/lib/index.js.map +1 -1
- package/controller/package.json +2 -2
- package/docs/demo/pie.manifest.json +11 -0
- package/lib/__tests__/choice-input-test.js +131 -0
- package/lib/__tests__/index-test.js +183 -0
- package/lib/__tests__/key-events-test.js +98 -0
- package/lib/__tests__/multiple-choice-test.js +235 -0
- package/lib/__tests__/session-updater-test.js +72 -0
- package/lib/choice-input.js +1 -1
- package/lib/choice-input.js.map +1 -1
- package/lib/feedback-tick.js +1 -1
- package/lib/feedback-tick.js.map +1 -1
- package/lib/index.js +2 -2
- package/lib/index.js.map +1 -1
- package/lib/main.js +1 -1
- package/lib/main.js.map +1 -1
- package/lib/multiple-choice.js +4 -4
- package/lib/multiple-choice.js.map +1 -1
- package/lib/print.js +1 -1
- package/lib/print.js.map +1 -1
- package/package.json +6 -3
- package/module/configure.js +0 -11070
- package/module/controller.js +0 -3601
- package/module/demo.js +0 -86
- package/module/element.js +0 -13448
- package/module/index.html +0 -21
- package/module/manifest.json +0 -14
- package/module/print-demo.js +0 -124
- package/module/print.html +0 -18
- package/module/print.js +0 -13204
|
@@ -0,0 +1,8 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
|
|
3
|
+
describe('utils', function () {
|
|
4
|
+
xit('todo', function () {
|
|
5
|
+
expect(true).toEqual(false);
|
|
6
|
+
});
|
|
7
|
+
});
|
|
8
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9fX3Rlc3RzX18vdXRpbHMudGVzdC5qcyJdLCJuYW1lcyI6WyJkZXNjcmliZSIsInhpdCIsImV4cGVjdCIsInRvRXF1YWwiXSwibWFwcGluZ3MiOiI7O0FBQUFBLFFBQVEsQ0FBQyxPQUFELEVBQVUsWUFBTTtBQUN0QkMsRUFBQUEsR0FBRyxDQUFDLE1BQUQsRUFBUyxZQUFNO0FBQ2hCQyxJQUFBQSxNQUFNLENBQUMsSUFBRCxDQUFOLENBQWFDLE9BQWIsQ0FBcUIsS0FBckI7QUFDRCxHQUZFLENBQUg7QUFHRCxDQUpPLENBQVIiLCJzb3VyY2VzQ29udGVudCI6WyJkZXNjcmliZSgndXRpbHMnLCAoKSA9PiB7XG4gIHhpdCgndG9kbycsICgpID0+IHtcbiAgICBleHBlY3QodHJ1ZSkudG9FcXVhbChmYWxzZSk7XG4gIH0pO1xufSk7XG4iXX0=
|
package/controller/lib/index.js
CHANGED
|
@@ -29,7 +29,7 @@ var _utils = require("./utils");
|
|
|
29
29
|
|
|
30
30
|
var _defaults = _interopRequireDefault(require("./defaults"));
|
|
31
31
|
|
|
32
|
-
var _controllerUtils = require("@pie-lib/
|
|
32
|
+
var _controllerUtils = require("@pie-lib/controller-utils");
|
|
33
33
|
|
|
34
34
|
var _excluded = ["verticalMode", "choicesLayout"];
|
|
35
35
|
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/index.js"],"names":["prepareChoice","model","env","defaultFeedback","choice","role","mode","out","label","value","rationale","rationaleEnabled","correct","feedbackEnabled","feedbackType","feedback","type","createDefaultModel","Promise","resolve","defaults","normalize","question","verticalMode","choicesLayout","questionProps","session","updateSession","normalizedQuestion","Object","assign","incorrect","choices","map","lockChoiceOrder","disabled","prompt","promptEnabled","gridColumns","choiceMode","keyMode","choicePrefix","responseCorrect","undefined","language","extraCSSRules","fontSizeFactor","isSelectionButtonBelow","selectedAnswerBackgroundColor","minSelections","maxSelections","keyboardEventsEnabled","autoplayAudioEnabled","completeAudioEnabled","customAudioButton","teacherInstructions","teacherInstructionsEnabled","getScore","config","selectedChoices","correctChoices","filter","ch","score","reduce","acc","selectedChoice","find","length","str","parseFloat","toFixed","outcome","empty","partialScoringEnabled","partialScoring","enabled","createCorrectResponseSession","id","c","getInnerText","html","replaceAll","getContent","replace","validate","minAnswerChoices","maxAnswerChoices","reversedChoices","reverse","choicesErrors","rationaleErrors","errors","forEach","field","required","hasCorrectResponse","index","identicalAnswer","slice","some","nbOfChoices","answerChoices","correctResponse"],"mappings":"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","sourcesContent":["/* eslint-disable no-console */\nimport isEmpty from 'lodash/isEmpty';\nimport { isResponseCorrect } from './utils';\nimport defaults from './defaults';\nimport { lockChoices, partialScoring, getShuffledChoices } from '@pie-lib/pie-toolbox/controller-utils';\n\nconst prepareChoice = (model, env, defaultFeedback) => (choice) => {\n const { role, mode } = env || {};\n const out = {\n label: choice.label,\n value: choice.value,\n };\n\n if (role === 'instructor' && (mode === 'view' || mode === 'evaluate')) {\n out.rationale = model.rationaleEnabled ? choice.rationale : null;\n } else {\n out.rationale = null;\n }\n\n if (mode === 'evaluate') {\n out.correct = !!choice.correct;\n\n if (model.feedbackEnabled) {\n const feedbackType = (choice.feedback && choice.feedback.type) || 'none';\n\n if (feedbackType === 'default') {\n out.feedback = defaultFeedback[choice.correct ? 'correct' : 'incorrect'];\n } else if (feedbackType === 'custom') {\n out.feedback = choice.feedback.value;\n }\n }\n }\n\n return out;\n};\n\nexport function createDefaultModel(model = {}) {\n return new Promise((resolve) => resolve({ ...defaults, ...model }));\n}\n\nexport const normalize = (question) => {\n const { verticalMode, choicesLayout, ...questionProps } = question || {};\n\n return {\n ...defaults,\n ...questionProps,\n // This is used for offering support for old models which have the property verticalMode\n // Same thing is set in authoring : packages/multiple-choice/configure/src/index.jsx - createDefaultModel\n choicesLayout: choicesLayout || (verticalMode === false && 'horizontal') || defaults.choicesLayout,\n };\n};\n\n/**\n *\n * @param {*} question\n * @param {*} session\n * @param {*} env\n * @param {*} updateSession - optional - a function that will set the properties passed into it on the session.\n */\nexport async function model(question, session, env, updateSession) {\n const normalizedQuestion = normalize(question);\n\n const defaultFeedback = Object.assign(\n { correct: 'Correct', incorrect: 'Incorrect' },\n normalizedQuestion.defaultFeedback,\n );\n\n let choices = (normalizedQuestion.choices || []).map(prepareChoice(normalizedQuestion, env, defaultFeedback));\n\n const lockChoiceOrder = lockChoices(normalizedQuestion, session, env);\n\n if (!lockChoiceOrder) {\n choices = await getShuffledChoices(choices, session, updateSession, 'value');\n }\n\n const out = {\n disabled: env.mode !== 'gather',\n mode: env.mode,\n prompt: normalizedQuestion.promptEnabled ? normalizedQuestion.prompt : null,\n choicesLayout: normalizedQuestion.choicesLayout,\n gridColumns: normalizedQuestion.gridColumns,\n choiceMode: normalizedQuestion.choiceMode,\n keyMode: normalizedQuestion.choicePrefix,\n choices,\n responseCorrect: env.mode === 'evaluate' ? isResponseCorrect(normalizedQuestion, session) : undefined,\n language: normalizedQuestion.language,\n extraCSSRules: normalizedQuestion.extraCSSRules,\n fontSizeFactor: normalizedQuestion.fontSizeFactor,\n isSelectionButtonBelow: normalizedQuestion.isSelectionButtonBelow,\n selectedAnswerBackgroundColor: normalizedQuestion.selectedAnswerBackgroundColor || 'initial',\n minSelections: normalizedQuestion.minSelections,\n maxSelections: normalizedQuestion.maxSelections,\n keyboardEventsEnabled: normalizedQuestion.keyboardEventsEnabled,\n autoplayAudioEnabled: normalizedQuestion.autoplayAudioEnabled,\n completeAudioEnabled: normalizedQuestion.completeAudioEnabled,\n customAudioButton: normalizedQuestion.customAudioButton,\n };\n\n const { role, mode } = env || {};\n\n if (role === 'instructor' && (mode === 'view' || mode === 'evaluate')) {\n out.teacherInstructions = normalizedQuestion.teacherInstructionsEnabled\n ? normalizedQuestion.teacherInstructions\n : null;\n } else {\n out.teacherInstructions = null;\n }\n\n return out;\n}\n\nexport const getScore = (config, session) => {\n if (!session || isEmpty(session)) {\n return 0;\n }\n\n const selectedChoices = session.value || [];\n const correctChoices = (config.choices || []).filter((ch) => ch.correct);\n\n let score = selectedChoices.reduce(\n (acc, selectedChoice) => acc + (correctChoices.find((ch) => ch.value === selectedChoice) ? 1 : 0),\n 0,\n );\n\n if (correctChoices.length < selectedChoices.length) {\n score -= selectedChoices.length - correctChoices.length;\n\n if (score < 0) {\n score = 0;\n }\n }\n\n const str = correctChoices.length ? score / correctChoices.length : 0;\n\n return parseFloat(str.toFixed(2));\n};\n\n/**\n *\n * The score is partial by default for checkbox mode, allOrNothing for radio mode.\n * To disable partial scoring for checkbox mode you either set model.partialScoring = false or env.partialScoring = false. the value in `env` will\n * override the value in `model`.\n * @param {Object} model - the main model\n * @param {*} session\n * @param {Object} env\n */\nexport function outcome(model, session, env) {\n return new Promise((resolve) => {\n if (!session || isEmpty(session)) {\n resolve({ score: 0, empty: true });\n } else {\n const partialScoringEnabled = partialScoring.enabled(model, env) && model.choiceMode !== 'radio';\n const score = getScore(model, session);\n\n resolve({ score: partialScoringEnabled ? score : score === 1 ? 1 : 0, empty: false });\n }\n });\n}\n\nexport const createCorrectResponseSession = (question, env) => {\n return new Promise((resolve) => {\n if (env.mode !== 'evaluate' && env.role === 'instructor') {\n const { choices } = question || { choices: [] };\n\n resolve({\n id: '1',\n value: choices.filter((c) => c.correct).map((c) => c.value),\n });\n } else {\n resolve(null);\n }\n });\n};\n\n// remove all html tags\nconst getInnerText = (html) => (html || '').replaceAll(/<[^>]*>/g, '');\n\n// remove all html tags except img, iframe and source tag for audio\nconst getContent = (html) => (html || '').replace(/(<(?!img|iframe|source)([^>]+)>)/gi, '');\n\nexport const validate = (model = {}, config = {}) => {\n const { choices } = model;\n const { minAnswerChoices = 2, maxAnswerChoices } = config;\n const reversedChoices = [...(choices || [])].reverse();\n const choicesErrors = {};\n const rationaleErrors = {};\n const errors = {};\n\n ['teacherInstructions', 'prompt'].forEach((field) => {\n if (config[field]?.required && !getContent(model[field])) {\n errors[field] = 'This field is required.';\n }\n });\n\n let hasCorrectResponse = false;\n\n reversedChoices.forEach((choice, index) => {\n const { correct, value, label, rationale } = choice;\n\n if (correct) {\n hasCorrectResponse = true;\n }\n\n if (!getContent(label)) {\n choicesErrors[value] = 'Content should not be empty.';\n } else {\n const identicalAnswer = reversedChoices.slice(index + 1).some((c) => c.label === label);\n\n if (identicalAnswer) {\n choicesErrors[value] = 'Content should be unique.';\n }\n }\n\n if (config.rationale?.required && !getContent(rationale)) {\n rationaleErrors[value] = 'This field is required.';\n }\n });\n\n const nbOfChoices = (choices || []).length;\n\n if (nbOfChoices < minAnswerChoices) {\n errors.answerChoices = `There should be at least ${minAnswerChoices} choices defined.`;\n } else if (nbOfChoices > maxAnswerChoices) {\n errors.answerChoices = `No more than ${maxAnswerChoices} choices should be defined.`;\n }\n\n if (!hasCorrectResponse) {\n errors.correctResponse = 'No correct response defined.';\n }\n\n if (!isEmpty(choicesErrors)) {\n errors.choices = choicesErrors;\n }\n\n if (!isEmpty(rationaleErrors)) {\n errors.rationale = rationaleErrors;\n }\n\n return errors;\n};\n"],"file":"index.js"}
|
|
1
|
+
{"version":3,"sources":["../src/index.js"],"names":["prepareChoice","model","env","defaultFeedback","choice","role","mode","out","label","value","rationale","rationaleEnabled","correct","feedbackEnabled","feedbackType","feedback","type","createDefaultModel","Promise","resolve","defaults","normalize","question","verticalMode","choicesLayout","questionProps","session","updateSession","normalizedQuestion","Object","assign","incorrect","choices","map","lockChoiceOrder","disabled","prompt","promptEnabled","gridColumns","choiceMode","keyMode","choicePrefix","responseCorrect","undefined","language","extraCSSRules","fontSizeFactor","isSelectionButtonBelow","selectedAnswerBackgroundColor","minSelections","maxSelections","keyboardEventsEnabled","autoplayAudioEnabled","completeAudioEnabled","customAudioButton","teacherInstructions","teacherInstructionsEnabled","getScore","config","selectedChoices","correctChoices","filter","ch","score","reduce","acc","selectedChoice","find","length","str","parseFloat","toFixed","outcome","empty","partialScoringEnabled","partialScoring","enabled","createCorrectResponseSession","id","c","getInnerText","html","replaceAll","getContent","replace","validate","minAnswerChoices","maxAnswerChoices","reversedChoices","reverse","choicesErrors","rationaleErrors","errors","forEach","field","required","hasCorrectResponse","index","identicalAnswer","slice","some","nbOfChoices","answerChoices","correctResponse"],"mappings":"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","sourcesContent":["/* eslint-disable no-console */\nimport isEmpty from 'lodash/isEmpty';\nimport { isResponseCorrect } from './utils';\nimport defaults from './defaults';\nimport { lockChoices, partialScoring, getShuffledChoices } from '@pie-lib/controller-utils';\n\nconst prepareChoice = (model, env, defaultFeedback) => (choice) => {\n const { role, mode } = env || {};\n const out = {\n label: choice.label,\n value: choice.value,\n };\n\n if (role === 'instructor' && (mode === 'view' || mode === 'evaluate')) {\n out.rationale = model.rationaleEnabled ? choice.rationale : null;\n } else {\n out.rationale = null;\n }\n\n if (mode === 'evaluate') {\n out.correct = !!choice.correct;\n\n if (model.feedbackEnabled) {\n const feedbackType = (choice.feedback && choice.feedback.type) || 'none';\n\n if (feedbackType === 'default') {\n out.feedback = defaultFeedback[choice.correct ? 'correct' : 'incorrect'];\n } else if (feedbackType === 'custom') {\n out.feedback = choice.feedback.value;\n }\n }\n }\n\n return out;\n};\n\nexport function createDefaultModel(model = {}) {\n return new Promise((resolve) => resolve({ ...defaults, ...model }));\n}\n\nexport const normalize = (question) => {\n const { verticalMode, choicesLayout, ...questionProps } = question || {};\n\n return {\n ...defaults,\n ...questionProps,\n // This is used for offering support for old models which have the property verticalMode\n // Same thing is set in authoring : packages/multiple-choice/configure/src/index.jsx - createDefaultModel\n choicesLayout: choicesLayout || (verticalMode === false && 'horizontal') || defaults.choicesLayout,\n };\n};\n\n/**\n *\n * @param {*} question\n * @param {*} session\n * @param {*} env\n * @param {*} updateSession - optional - a function that will set the properties passed into it on the session.\n */\nexport async function model(question, session, env, updateSession) {\n const normalizedQuestion = normalize(question);\n\n const defaultFeedback = Object.assign(\n { correct: 'Correct', incorrect: 'Incorrect' },\n normalizedQuestion.defaultFeedback,\n );\n\n let choices = (normalizedQuestion.choices || []).map(prepareChoice(normalizedQuestion, env, defaultFeedback));\n\n const lockChoiceOrder = lockChoices(normalizedQuestion, session, env);\n\n if (!lockChoiceOrder) {\n choices = await getShuffledChoices(choices, session, updateSession, 'value');\n }\n\n const out = {\n disabled: env.mode !== 'gather',\n mode: env.mode,\n prompt: normalizedQuestion.promptEnabled ? normalizedQuestion.prompt : null,\n choicesLayout: normalizedQuestion.choicesLayout,\n gridColumns: normalizedQuestion.gridColumns,\n choiceMode: normalizedQuestion.choiceMode,\n keyMode: normalizedQuestion.choicePrefix,\n choices,\n responseCorrect: env.mode === 'evaluate' ? isResponseCorrect(normalizedQuestion, session) : undefined,\n language: normalizedQuestion.language,\n extraCSSRules: normalizedQuestion.extraCSSRules,\n fontSizeFactor: normalizedQuestion.fontSizeFactor,\n isSelectionButtonBelow: normalizedQuestion.isSelectionButtonBelow,\n selectedAnswerBackgroundColor: normalizedQuestion.selectedAnswerBackgroundColor || 'initial',\n minSelections: normalizedQuestion.minSelections,\n maxSelections: normalizedQuestion.maxSelections,\n keyboardEventsEnabled: normalizedQuestion.keyboardEventsEnabled,\n autoplayAudioEnabled: normalizedQuestion.autoplayAudioEnabled,\n completeAudioEnabled: normalizedQuestion.completeAudioEnabled,\n customAudioButton: normalizedQuestion.customAudioButton,\n };\n\n const { role, mode } = env || {};\n\n if (role === 'instructor' && (mode === 'view' || mode === 'evaluate')) {\n out.teacherInstructions = normalizedQuestion.teacherInstructionsEnabled\n ? normalizedQuestion.teacherInstructions\n : null;\n } else {\n out.teacherInstructions = null;\n }\n\n return out;\n}\n\nexport const getScore = (config, session) => {\n if (!session || isEmpty(session)) {\n return 0;\n }\n\n const selectedChoices = session.value || [];\n const correctChoices = (config.choices || []).filter((ch) => ch.correct);\n\n let score = selectedChoices.reduce(\n (acc, selectedChoice) => acc + (correctChoices.find((ch) => ch.value === selectedChoice) ? 1 : 0),\n 0,\n );\n\n if (correctChoices.length < selectedChoices.length) {\n score -= selectedChoices.length - correctChoices.length;\n\n if (score < 0) {\n score = 0;\n }\n }\n\n const str = correctChoices.length ? score / correctChoices.length : 0;\n\n return parseFloat(str.toFixed(2));\n};\n\n/**\n *\n * The score is partial by default for checkbox mode, allOrNothing for radio mode.\n * To disable partial scoring for checkbox mode you either set model.partialScoring = false or env.partialScoring = false. the value in `env` will\n * override the value in `model`.\n * @param {Object} model - the main model\n * @param {*} session\n * @param {Object} env\n */\nexport function outcome(model, session, env) {\n return new Promise((resolve) => {\n if (!session || isEmpty(session)) {\n resolve({ score: 0, empty: true });\n } else {\n const partialScoringEnabled = partialScoring.enabled(model, env) && model.choiceMode !== 'radio';\n const score = getScore(model, session);\n\n resolve({ score: partialScoringEnabled ? score : score === 1 ? 1 : 0, empty: false });\n }\n });\n}\n\nexport const createCorrectResponseSession = (question, env) => {\n return new Promise((resolve) => {\n if (env.mode !== 'evaluate' && env.role === 'instructor') {\n const { choices } = question || { choices: [] };\n\n resolve({\n id: '1',\n value: choices.filter((c) => c.correct).map((c) => c.value),\n });\n } else {\n resolve(null);\n }\n });\n};\n\n// remove all html tags\nconst getInnerText = (html) => (html || '').replaceAll(/<[^>]*>/g, '');\n\n// remove all html tags except img, iframe and source tag for audio\nconst getContent = (html) => (html || '').replace(/(<(?!img|iframe|source)([^>]+)>)/gi, '');\n\nexport const validate = (model = {}, config = {}) => {\n const { choices } = model;\n const { minAnswerChoices = 2, maxAnswerChoices } = config;\n const reversedChoices = [...(choices || [])].reverse();\n const choicesErrors = {};\n const rationaleErrors = {};\n const errors = {};\n\n ['teacherInstructions', 'prompt'].forEach((field) => {\n if (config[field]?.required && !getContent(model[field])) {\n errors[field] = 'This field is required.';\n }\n });\n\n let hasCorrectResponse = false;\n\n reversedChoices.forEach((choice, index) => {\n const { correct, value, label, rationale } = choice;\n\n if (correct) {\n hasCorrectResponse = true;\n }\n\n if (!getContent(label)) {\n choicesErrors[value] = 'Content should not be empty.';\n } else {\n const identicalAnswer = reversedChoices.slice(index + 1).some((c) => c.label === label);\n\n if (identicalAnswer) {\n choicesErrors[value] = 'Content should be unique.';\n }\n }\n\n if (config.rationale?.required && !getContent(rationale)) {\n rationaleErrors[value] = 'This field is required.';\n }\n });\n\n const nbOfChoices = (choices || []).length;\n\n if (nbOfChoices < minAnswerChoices) {\n errors.answerChoices = `There should be at least ${minAnswerChoices} choices defined.`;\n } else if (nbOfChoices > maxAnswerChoices) {\n errors.answerChoices = `No more than ${maxAnswerChoices} choices should be defined.`;\n }\n\n if (!hasCorrectResponse) {\n errors.correctResponse = 'No correct response defined.';\n }\n\n if (!isEmpty(choicesErrors)) {\n errors.choices = choicesErrors;\n }\n\n if (!isEmpty(rationaleErrors)) {\n errors.rationale = rationaleErrors;\n }\n\n return errors;\n};\n"],"file":"index.js"}
|
package/controller/package.json
CHANGED
|
@@ -1,14 +1,14 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@pie-element/multiple-choice-controller",
|
|
3
3
|
"private": true,
|
|
4
|
-
"version": "5.
|
|
4
|
+
"version": "5.11.0-beta.3",
|
|
5
5
|
"description": "",
|
|
6
6
|
"main": "lib/index.js",
|
|
7
7
|
"module": "src/index.js",
|
|
8
8
|
"author": "",
|
|
9
9
|
"license": "ISC",
|
|
10
10
|
"dependencies": {
|
|
11
|
-
"@pie-lib/
|
|
11
|
+
"@pie-lib/controller-utils": "beta",
|
|
12
12
|
"debug": "^3.1.0",
|
|
13
13
|
"lodash": "^4.17.15"
|
|
14
14
|
}
|
|
@@ -0,0 +1,131 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
|
|
3
|
+
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
|
|
4
|
+
|
|
5
|
+
var _extends2 = _interopRequireDefault(require("@babel/runtime/helpers/extends"));
|
|
6
|
+
|
|
7
|
+
var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty"));
|
|
8
|
+
|
|
9
|
+
var _react = _interopRequireDefault(require("react"));
|
|
10
|
+
|
|
11
|
+
var _enzyme = require("enzyme");
|
|
12
|
+
|
|
13
|
+
var _choiceInput = require("../choice-input");
|
|
14
|
+
|
|
15
|
+
var _enzymeToJson = _interopRequireDefault(require("enzyme-to-json"));
|
|
16
|
+
|
|
17
|
+
function ownKeys(object, enumerableOnly) { var keys = Object.keys(object); if (Object.getOwnPropertySymbols) { var symbols = Object.getOwnPropertySymbols(object); enumerableOnly && (symbols = symbols.filter(function (sym) { return Object.getOwnPropertyDescriptor(object, sym).enumerable; })), keys.push.apply(keys, symbols); } return keys; }
|
|
18
|
+
|
|
19
|
+
function _objectSpread(target) { for (var i = 1; i < arguments.length; i++) { var source = null != arguments[i] ? arguments[i] : {}; i % 2 ? ownKeys(Object(source), !0).forEach(function (key) { (0, _defineProperty2["default"])(target, key, source[key]); }) : Object.getOwnPropertyDescriptors ? Object.defineProperties(target, Object.getOwnPropertyDescriptors(source)) : ownKeys(Object(source)).forEach(function (key) { Object.defineProperty(target, key, Object.getOwnPropertyDescriptor(source, key)); }); } return target; }
|
|
20
|
+
|
|
21
|
+
describe('ChoiceInput', function () {
|
|
22
|
+
var onChange, wrapper;
|
|
23
|
+
|
|
24
|
+
var mkWrapper = function mkWrapper() {
|
|
25
|
+
var opts = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
|
|
26
|
+
var defaultOpts = {
|
|
27
|
+
checked: false,
|
|
28
|
+
disabled: false,
|
|
29
|
+
choiceMode: 'checkbox',
|
|
30
|
+
label: 'label',
|
|
31
|
+
displayKey: '1',
|
|
32
|
+
correctness: 'correct',
|
|
33
|
+
value: 'value',
|
|
34
|
+
classes: {
|
|
35
|
+
label: 'label'
|
|
36
|
+
}
|
|
37
|
+
};
|
|
38
|
+
|
|
39
|
+
var finalOpts = _objectSpread(_objectSpread({}, defaultOpts), opts);
|
|
40
|
+
|
|
41
|
+
return (0, _enzyme.shallow)( /*#__PURE__*/_react["default"].createElement(_choiceInput.ChoiceInput, (0, _extends2["default"])({}, finalOpts, {
|
|
42
|
+
onChange: onChange
|
|
43
|
+
})));
|
|
44
|
+
};
|
|
45
|
+
|
|
46
|
+
beforeEach(function () {
|
|
47
|
+
onChange = jest.fn();
|
|
48
|
+
wrapper = mkWrapper();
|
|
49
|
+
});
|
|
50
|
+
describe('snapshots', function () {
|
|
51
|
+
describe('radio', function () {
|
|
52
|
+
it('renders', function () {
|
|
53
|
+
var wrapper = mkWrapper({
|
|
54
|
+
choiceMode: 'radio'
|
|
55
|
+
});
|
|
56
|
+
expect((0, _enzymeToJson["default"])(wrapper)).toMatchSnapshot();
|
|
57
|
+
});
|
|
58
|
+
});
|
|
59
|
+
describe('checkbox', function () {
|
|
60
|
+
it('renders', function () {
|
|
61
|
+
var wrapper = mkWrapper({
|
|
62
|
+
choiceMode: 'checkbox'
|
|
63
|
+
});
|
|
64
|
+
expect((0, _enzymeToJson["default"])(wrapper)).toMatchSnapshot();
|
|
65
|
+
});
|
|
66
|
+
});
|
|
67
|
+
describe('radio with incorrect', function () {
|
|
68
|
+
it('renders', function () {
|
|
69
|
+
var wrapper = mkWrapper({
|
|
70
|
+
choiceMode: 'radio',
|
|
71
|
+
correctness: 'incorrect'
|
|
72
|
+
});
|
|
73
|
+
expect((0, _enzymeToJson["default"])(wrapper)).toMatchSnapshot();
|
|
74
|
+
});
|
|
75
|
+
});
|
|
76
|
+
describe('rationale', function () {
|
|
77
|
+
it('does not render', function () {
|
|
78
|
+
var wrapper = mkWrapper();
|
|
79
|
+
expect((0, _enzymeToJson["default"])(wrapper)).toMatchSnapshot();
|
|
80
|
+
});
|
|
81
|
+
it('renders', function () {
|
|
82
|
+
var wrapper = mkWrapper({
|
|
83
|
+
rationale: 'This is rationale'
|
|
84
|
+
});
|
|
85
|
+
expect((0, _enzymeToJson["default"])(wrapper)).toMatchSnapshot();
|
|
86
|
+
});
|
|
87
|
+
});
|
|
88
|
+
});
|
|
89
|
+
describe('onToggleChoice', function () {
|
|
90
|
+
it('calls handler', function () {
|
|
91
|
+
wrapper.instance().onToggleChoice({
|
|
92
|
+
target: {
|
|
93
|
+
checked: true
|
|
94
|
+
}
|
|
95
|
+
});
|
|
96
|
+
expect(onChange).toBeCalledWith({
|
|
97
|
+
value: 'value',
|
|
98
|
+
selected: true
|
|
99
|
+
});
|
|
100
|
+
});
|
|
101
|
+
});
|
|
102
|
+
/** we're mocking preview prompt - so this won't pass */
|
|
103
|
+
|
|
104
|
+
describe.skip('clicking on choice text triggers toggle choice', function () {
|
|
105
|
+
it('calls handler', function () {
|
|
106
|
+
var onChange = jest.fn();
|
|
107
|
+
var props = {
|
|
108
|
+
checked: false,
|
|
109
|
+
disabled: false,
|
|
110
|
+
choiceMode: 'checkbox',
|
|
111
|
+
label: 'label',
|
|
112
|
+
displayKey: '1',
|
|
113
|
+
correctness: 'correct',
|
|
114
|
+
value: 'value',
|
|
115
|
+
classes: {
|
|
116
|
+
label: 'label'
|
|
117
|
+
},
|
|
118
|
+
onChange: onChange
|
|
119
|
+
};
|
|
120
|
+
console.log(_choiceInput.ChoiceInput);
|
|
121
|
+
var item = (0, _enzyme.mount)( /*#__PURE__*/_react["default"].createElement(_choiceInput.ChoiceInput, props));
|
|
122
|
+
var label = item.find('.label');
|
|
123
|
+
label.simulate('click');
|
|
124
|
+
expect(onChange).toHaveBeenCalledWith({
|
|
125
|
+
value: 'value',
|
|
126
|
+
selected: true
|
|
127
|
+
});
|
|
128
|
+
});
|
|
129
|
+
});
|
|
130
|
+
});
|
|
131
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/__tests__/choice-input-test.jsx"],"names":["describe","onChange","wrapper","mkWrapper","opts","defaultOpts","checked","disabled","choiceMode","label","displayKey","correctness","value","classes","finalOpts","beforeEach","jest","fn","it","expect","toMatchSnapshot","rationale","instance","onToggleChoice","target","toBeCalledWith","selected","skip","props","console","log","ChoiceInput","item","find","simulate","toHaveBeenCalledWith"],"mappings":"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","sourcesContent":["import React from 'react';\nimport { shallow, mount } from 'enzyme';\nimport { ChoiceInput } from '../choice-input';\nimport toJson from 'enzyme-to-json';\n\ndescribe('ChoiceInput', () => {\n  let onChange, wrapper;\n\n const mkWrapper = (opts = {}) => {\n  const defaultOpts = {\n    checked: false,\n    disabled: false,\n    choiceMode: 'checkbox',\n    label: 'label',\n    displayKey: '1',\n    correctness: 'correct',\n    value: 'value',\n    classes: {\n      label: 'label',\n    },\n  };\n\n  const finalOpts = { ...defaultOpts, ...opts };\n\n  return shallow(<ChoiceInput {...finalOpts} onChange={onChange} />);\n};\n\n  beforeEach(() => {\n    onChange = jest.fn();\n    wrapper = mkWrapper();\n  });\n\n  describe('snapshots', () => {\n    describe('radio', () => {\n      it('renders', () => {\n        const wrapper = mkWrapper({ choiceMode: 'radio' });\n        expect(toJson(wrapper)).toMatchSnapshot();\n      });\n    });\n\n    describe('checkbox', () => {\n      it('renders', () => {\n        const wrapper = mkWrapper({ choiceMode: 'checkbox' });\n        expect(toJson(wrapper)).toMatchSnapshot();\n      });\n    });\n\n    describe('radio with incorrect', () => {\n      it('renders', () => {\n        const wrapper = mkWrapper({\n          choiceMode: 'radio',\n          correctness: 'incorrect',\n        });\n        expect(toJson(wrapper)).toMatchSnapshot();\n      });\n    });\n\n    describe('rationale', () => {\n      it('does not render', () => {\n        const wrapper = mkWrapper();\n        expect(toJson(wrapper)).toMatchSnapshot();\n      });\n\n      it('renders', () => {\n        const wrapper = mkWrapper({ rationale: 'This is rationale' });\n        expect(toJson(wrapper)).toMatchSnapshot();\n      });\n    });\n  });\n\n  describe('onToggleChoice', () => {\n    it('calls handler', () => {\n      wrapper.instance().onToggleChoice({ target: { checked: true } });\n      expect(onChange).toBeCalledWith({\n        value: 'value',\n        selected: true,\n      });\n    });\n  });\n\n  /** we're mocking preview prompt - so this won't pass */\n  describe.skip('clicking on choice text triggers toggle choice', () => {\n    it('calls handler', () => {\n      const onChange = jest.fn();\n      const props = {\n        checked: false,\n        disabled: false,\n        choiceMode: 'checkbox',\n        label: 'label',\n        displayKey: '1',\n        correctness: 'correct',\n        value: 'value',\n        classes: {\n          label: 'label',\n        },\n        onChange,\n      };\n\n      console.log(ChoiceInput);\n      const item = mount(<ChoiceInput {...props} />);\n      const label = item.find('.label');\n\n      label.simulate('click');\n\n      expect(onChange).toHaveBeenCalledWith({ value: 'value', selected: true });\n    });\n  });\n});\n"]}
|
|
@@ -0,0 +1,183 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
|
|
3
|
+
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
|
|
4
|
+
|
|
5
|
+
var _typeof = require("@babel/runtime/helpers/typeof");
|
|
6
|
+
|
|
7
|
+
var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty"));
|
|
8
|
+
|
|
9
|
+
var _taggedTemplateLiteral2 = _interopRequireDefault(require("@babel/runtime/helpers/taggedTemplateLiteral"));
|
|
10
|
+
|
|
11
|
+
var _react = _interopRequireDefault(require("react"));
|
|
12
|
+
|
|
13
|
+
var _enzyme = require("enzyme");
|
|
14
|
+
|
|
15
|
+
var _piePlayerEvents = require("@pie-framework/pie-player-events");
|
|
16
|
+
|
|
17
|
+
var _main = _interopRequireDefault(require("../main"));
|
|
18
|
+
|
|
19
|
+
var _index = _interopRequireWildcard(require("../index"));
|
|
20
|
+
|
|
21
|
+
var _templateObject;
|
|
22
|
+
|
|
23
|
+
function _getRequireWildcardCache(nodeInterop) { if (typeof WeakMap !== "function") return null; var cacheBabelInterop = new WeakMap(); var cacheNodeInterop = new WeakMap(); return (_getRequireWildcardCache = function _getRequireWildcardCache(nodeInterop) { return nodeInterop ? cacheNodeInterop : cacheBabelInterop; })(nodeInterop); }
|
|
24
|
+
|
|
25
|
+
function _interopRequireWildcard(obj, nodeInterop) { if (!nodeInterop && obj && obj.__esModule) { return obj; } if (obj === null || _typeof(obj) !== "object" && typeof obj !== "function") { return { "default": obj }; } var cache = _getRequireWildcardCache(nodeInterop); if (cache && cache.has(obj)) { return cache.get(obj); } var newObj = {}; var hasPropertyDescriptor = Object.defineProperty && Object.getOwnPropertyDescriptor; for (var key in obj) { if (key !== "default" && Object.prototype.hasOwnProperty.call(obj, key)) { var desc = hasPropertyDescriptor ? Object.getOwnPropertyDescriptor(obj, key) : null; if (desc && (desc.get || desc.set)) { Object.defineProperty(newObj, key, desc); } else { newObj[key] = obj[key]; } } } newObj["default"] = obj; if (cache) { cache.set(obj, newObj); } return newObj; }
|
|
26
|
+
|
|
27
|
+
function ownKeys(object, enumerableOnly) { var keys = Object.keys(object); if (Object.getOwnPropertySymbols) { var symbols = Object.getOwnPropertySymbols(object); enumerableOnly && (symbols = symbols.filter(function (sym) { return Object.getOwnPropertyDescriptor(object, sym).enumerable; })), keys.push.apply(keys, symbols); } return keys; }
|
|
28
|
+
|
|
29
|
+
function _objectSpread(target) { for (var i = 1; i < arguments.length; i++) { var source = null != arguments[i] ? arguments[i] : {}; i % 2 ? ownKeys(Object(source), !0).forEach(function (key) { (0, _defineProperty2["default"])(target, key, source[key]); }) : Object.getOwnPropertyDescriptors ? Object.defineProperties(target, Object.getOwnPropertyDescriptors(source)) : ownKeys(Object(source)).forEach(function (key) { Object.defineProperty(target, key, Object.getOwnPropertyDescriptor(source, key)); }); } return target; }
|
|
30
|
+
|
|
31
|
+
jest.useFakeTimers();
|
|
32
|
+
jest.mock('@pie-lib/math-rendering', function () {
|
|
33
|
+
return {
|
|
34
|
+
renderMath: jest.fn()
|
|
35
|
+
};
|
|
36
|
+
});
|
|
37
|
+
jest.mock('lodash/debounce', function () {
|
|
38
|
+
return jest.fn(function (fn) {
|
|
39
|
+
return fn;
|
|
40
|
+
});
|
|
41
|
+
});
|
|
42
|
+
describe('isComplete', function () {
|
|
43
|
+
it.each(_templateObject || (_templateObject = (0, _taggedTemplateLiteral2["default"])(["\n session | expected\n ", " | ", "\n ", " | ", "\n ", " | ", "\n ", " | ", "\n ", " | ", "\n "])), {
|
|
44
|
+
value: [1, 2, 3]
|
|
45
|
+
}, true, {
|
|
46
|
+
value: []
|
|
47
|
+
}, false, {}, false, null, false, undefined, false)('session = $session is complete => $expected', function (_ref) {
|
|
48
|
+
var session = _ref.session,
|
|
49
|
+
expected = _ref.expected;
|
|
50
|
+
expect((0, _index.isComplete)(session)).toEqual(expected);
|
|
51
|
+
});
|
|
52
|
+
});
|
|
53
|
+
describe('multiple-choice', function () {
|
|
54
|
+
describe('renders', function () {
|
|
55
|
+
var wrapper = function wrapper(props) {
|
|
56
|
+
var defaultProps = {
|
|
57
|
+
model: _objectSpread({}, props),
|
|
58
|
+
session: {},
|
|
59
|
+
classes: {}
|
|
60
|
+
};
|
|
61
|
+
return (0, _enzyme.shallow)( /*#__PURE__*/_react["default"].createElement(_main["default"], defaultProps));
|
|
62
|
+
};
|
|
63
|
+
|
|
64
|
+
it('snapshot', function () {
|
|
65
|
+
var w = wrapper();
|
|
66
|
+
expect(w).toMatchSnapshot();
|
|
67
|
+
});
|
|
68
|
+
it('snapshot with rationale', function () {
|
|
69
|
+
var w = wrapper({
|
|
70
|
+
rationale: 'This is rationale'
|
|
71
|
+
});
|
|
72
|
+
expect(w).toMatchSnapshot();
|
|
73
|
+
});
|
|
74
|
+
it('snapshot with teacherInstructions', function () {
|
|
75
|
+
var w = wrapper({
|
|
76
|
+
teacherInstructions: 'These are teacher instructions'
|
|
77
|
+
});
|
|
78
|
+
expect(w).toMatchSnapshot();
|
|
79
|
+
});
|
|
80
|
+
});
|
|
81
|
+
describe('events', function () {
|
|
82
|
+
describe('model', function () {
|
|
83
|
+
it('dispatches model set event', function () {
|
|
84
|
+
var el = new _index["default"]();
|
|
85
|
+
el.tagName = 'mc-el';
|
|
86
|
+
el.model = {};
|
|
87
|
+
expect(el.dispatchEvent).toBeCalledWith(new _piePlayerEvents.ModelSetEvent('mc-el', false, true));
|
|
88
|
+
});
|
|
89
|
+
});
|
|
90
|
+
describe('onChange', function () {
|
|
91
|
+
it('dispatches session changed event - add answer (checkbox)', function () {
|
|
92
|
+
var el = new _index["default"]();
|
|
93
|
+
el.tagName = 'mc-el';
|
|
94
|
+
el.model = {
|
|
95
|
+
choiceMode: 'checkbox'
|
|
96
|
+
};
|
|
97
|
+
el.session = {
|
|
98
|
+
value: []
|
|
99
|
+
};
|
|
100
|
+
|
|
101
|
+
el._onChange({
|
|
102
|
+
value: 'a',
|
|
103
|
+
selected: true
|
|
104
|
+
});
|
|
105
|
+
|
|
106
|
+
expect(el.dispatchEvent).toBeCalledWith(new _piePlayerEvents.SessionChangedEvent('mc-el', true));
|
|
107
|
+
});
|
|
108
|
+
it('dispatches session changed event - remove answer (checkbox)', function () {
|
|
109
|
+
var el = new _index["default"]();
|
|
110
|
+
el.tagName = 'mc-el';
|
|
111
|
+
el.model = {
|
|
112
|
+
choiceMode: 'checkbox'
|
|
113
|
+
};
|
|
114
|
+
el.session = {
|
|
115
|
+
value: ['a']
|
|
116
|
+
};
|
|
117
|
+
|
|
118
|
+
el._onChange({
|
|
119
|
+
value: 'a',
|
|
120
|
+
selected: false
|
|
121
|
+
});
|
|
122
|
+
|
|
123
|
+
expect(el.dispatchEvent).toBeCalledWith(new _piePlayerEvents.SessionChangedEvent('mc-el', false));
|
|
124
|
+
});
|
|
125
|
+
it('dispatches session changed event - add/remove answer (checkbox)', function () {
|
|
126
|
+
var el = new _index["default"]();
|
|
127
|
+
el.tagName = 'mc-el';
|
|
128
|
+
el.model = {
|
|
129
|
+
choiceMode: 'checkbox'
|
|
130
|
+
};
|
|
131
|
+
el.session = {
|
|
132
|
+
value: ['1']
|
|
133
|
+
};
|
|
134
|
+
|
|
135
|
+
el._onChange({
|
|
136
|
+
id: '2',
|
|
137
|
+
selected: true
|
|
138
|
+
});
|
|
139
|
+
|
|
140
|
+
expect(el.dispatchEvent).toBeCalledWith(new _piePlayerEvents.SessionChangedEvent('mc-el', true));
|
|
141
|
+
|
|
142
|
+
el._onChange({
|
|
143
|
+
id: '1',
|
|
144
|
+
selected: false
|
|
145
|
+
});
|
|
146
|
+
|
|
147
|
+
expect(el.dispatchEvent).toBeCalledWith(new _piePlayerEvents.SessionChangedEvent('mc-el', true));
|
|
148
|
+
|
|
149
|
+
el._onChange({
|
|
150
|
+
id: '2',
|
|
151
|
+
selected: false
|
|
152
|
+
});
|
|
153
|
+
|
|
154
|
+
expect(el.dispatchEvent).toBeCalledWith(new _piePlayerEvents.SessionChangedEvent('mc-el', false));
|
|
155
|
+
});
|
|
156
|
+
it('dispatches session changed event - add/change answer (radio)', function () {
|
|
157
|
+
var el = new _index["default"]();
|
|
158
|
+
el.tagName = 'mc-el';
|
|
159
|
+
el.model = {
|
|
160
|
+
choiceMode: 'radio'
|
|
161
|
+
};
|
|
162
|
+
el.session = {
|
|
163
|
+
value: []
|
|
164
|
+
};
|
|
165
|
+
|
|
166
|
+
el._onChange({
|
|
167
|
+
value: 'a',
|
|
168
|
+
selected: true
|
|
169
|
+
});
|
|
170
|
+
|
|
171
|
+
expect(el.dispatchEvent).toBeCalledWith(new _piePlayerEvents.SessionChangedEvent('mc-el', true));
|
|
172
|
+
|
|
173
|
+
el._onChange({
|
|
174
|
+
value: 'b',
|
|
175
|
+
selected: true
|
|
176
|
+
});
|
|
177
|
+
|
|
178
|
+
expect(el.dispatchEvent).toBeCalledWith(new _piePlayerEvents.SessionChangedEvent('mc-el', true));
|
|
179
|
+
});
|
|
180
|
+
});
|
|
181
|
+
});
|
|
182
|
+
});
|
|
183
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/__tests__/index-test.jsx"],"names":["jest","useFakeTimers","mock","renderMath","fn","describe","it","each","value","undefined","session","expected","expect","toEqual","wrapper","props","defaultProps","model","classes","w","toMatchSnapshot","rationale","teacherInstructions","el","MultipleChoice","tagName","dispatchEvent","toBeCalledWith","ModelSetEvent","choiceMode","_onChange","selected","SessionChangedEvent","id"],"mappings":"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","sourcesContent":["import React from 'react';\nimport { shallow } from 'enzyme';\nimport { ModelSetEvent, SessionChangedEvent } from '@pie-framework/pie-player-events';\nimport MultipleChoiceComponent from '../main';\nimport MultipleChoice from '../index';\nimport { isComplete } from '../index';\n\njest.useFakeTimers();\njest.mock('@pie-lib/math-rendering', () => ({ renderMath: jest.fn() }));\njest.mock('lodash/debounce', () => jest.fn((fn) => fn));\n\ndescribe('isComplete', () => {\n  it.each`\n    session                 | expected\n    ${{ value: [1, 2, 3] }} | ${true}\n    ${{ value: [] }}        | ${false}\n    ${{}}                   | ${false}\n    ${null}                 | ${false}\n    ${undefined}            | ${false}\n  `('session = $session is complete => $expected', ({ session, expected }) => {\n    expect(isComplete(session)).toEqual(expected);\n  });\n});\n\ndescribe('multiple-choice', () => {\n  describe('renders', () => {\n    let wrapper = (props) => {\n      let defaultProps = {\n        model: {\n          ...props,\n        },\n        session: {},\n        classes: {},\n      };\n\n      return shallow(<MultipleChoiceComponent {...defaultProps} />);\n    };\n\n    it('snapshot', () => {\n      const w = wrapper();\n      expect(w).toMatchSnapshot();\n    });\n\n    it('snapshot with rationale', () => {\n      const w = wrapper({ rationale: 'This is rationale' });\n      expect(w).toMatchSnapshot();\n    });\n\n    it('snapshot with teacherInstructions', () => {\n      const w = wrapper({ teacherInstructions: 'These are teacher instructions' });\n      expect(w).toMatchSnapshot();\n    });\n  });\n\n  describe('events', () => {\n    describe('model', () => {\n      it('dispatches model set event', () => {\n        const el = new MultipleChoice();\n        el.tagName = 'mc-el';\n        el.model = {};\n        expect(el.dispatchEvent).toBeCalledWith(new ModelSetEvent('mc-el', false, true));\n      });\n    });\n\n    describe('onChange', () => {\n      it('dispatches session changed event - add answer (checkbox)', () => {\n        const el = new MultipleChoice();\n        el.tagName = 'mc-el';\n        el.model = { choiceMode: 'checkbox' };\n        el.session = { value: [] };\n        el._onChange({ value: 'a', selected: true });\n        expect(el.dispatchEvent).toBeCalledWith(new SessionChangedEvent('mc-el', true));\n      });\n\n      it('dispatches session changed event - remove answer (checkbox)', () => {\n        const el = new MultipleChoice();\n        el.tagName = 'mc-el';\n        el.model = { choiceMode: 'checkbox' };\n        el.session = { value: ['a'] };\n        el._onChange({ value: 'a', selected: false });\n        expect(el.dispatchEvent).toBeCalledWith(new SessionChangedEvent('mc-el', false));\n      });\n\n      it('dispatches session changed event - add/remove answer (checkbox)', () => {\n        const el = new MultipleChoice();\n        el.tagName = 'mc-el';\n        el.model = { choiceMode: 'checkbox' };\n        el.session = { value: ['1'] };\n        el._onChange({ id: '2', selected: true });\n        expect(el.dispatchEvent).toBeCalledWith(new SessionChangedEvent('mc-el', true));\n\n        el._onChange({ id: '1', selected: false });\n        expect(el.dispatchEvent).toBeCalledWith(new SessionChangedEvent('mc-el', true));\n\n        el._onChange({ id: '2', selected: false });\n        expect(el.dispatchEvent).toBeCalledWith(new SessionChangedEvent('mc-el', false));\n      });\n\n      it('dispatches session changed event - add/change answer (radio)', () => {\n        const el = new MultipleChoice();\n        el.tagName = 'mc-el';\n        el.model = { choiceMode: 'radio' };\n        el.session = { value: [] };\n        el._onChange({ value: 'a', selected: true });\n        expect(el.dispatchEvent).toBeCalledWith(new SessionChangedEvent('mc-el', true));\n        el._onChange({ value: 'b', selected: true });\n        expect(el.dispatchEvent).toBeCalledWith(new SessionChangedEvent('mc-el', true));\n      });\n    });\n  });\n});\n"]}
|
|
@@ -0,0 +1,98 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
|
|
3
|
+
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
|
|
4
|
+
|
|
5
|
+
var _multipleChoice = _interopRequireDefault(require("../multiple-choice"));
|
|
6
|
+
|
|
7
|
+
describe('MultipleChoice', function () {
|
|
8
|
+
var instance;
|
|
9
|
+
beforeEach(function () {
|
|
10
|
+
instance = new _multipleChoice["default"]();
|
|
11
|
+
instance._model = {
|
|
12
|
+
choices: [{
|
|
13
|
+
value: '1'
|
|
14
|
+
}, {
|
|
15
|
+
value: '2'
|
|
16
|
+
}, {
|
|
17
|
+
value: '3'
|
|
18
|
+
}],
|
|
19
|
+
choiceMode: 'checkbox'
|
|
20
|
+
};
|
|
21
|
+
instance._session = {
|
|
22
|
+
value: []
|
|
23
|
+
};
|
|
24
|
+
instance._onChange = jest.fn();
|
|
25
|
+
});
|
|
26
|
+
describe('handleKeyDown', function () {
|
|
27
|
+
var simulateKeyDown = function simulateKeyDown(key) {
|
|
28
|
+
var event = new KeyboardEvent('keydown', {
|
|
29
|
+
key: key
|
|
30
|
+
});
|
|
31
|
+
instance.handleKeyDown(event);
|
|
32
|
+
};
|
|
33
|
+
|
|
34
|
+
it('selects a choice on valid numeric key (checkbox)', function () {
|
|
35
|
+
simulateKeyDown('1');
|
|
36
|
+
expect(instance._onChange).toHaveBeenCalledWith({
|
|
37
|
+
value: '1',
|
|
38
|
+
selected: true
|
|
39
|
+
});
|
|
40
|
+
});
|
|
41
|
+
it('toggles a choice on valid numeric key (checkbox)', function () {
|
|
42
|
+
instance._session.value = ['1'];
|
|
43
|
+
simulateKeyDown('1');
|
|
44
|
+
expect(instance._onChange).toHaveBeenCalledWith({
|
|
45
|
+
value: '1',
|
|
46
|
+
selected: false
|
|
47
|
+
});
|
|
48
|
+
});
|
|
49
|
+
it('selects a choice on valid alphabetic key (checkbox)', function () {
|
|
50
|
+
simulateKeyDown('a');
|
|
51
|
+
expect(instance._onChange).toHaveBeenCalledWith({
|
|
52
|
+
value: '1',
|
|
53
|
+
selected: true
|
|
54
|
+
});
|
|
55
|
+
});
|
|
56
|
+
it('does nothing on invalid key', function () {
|
|
57
|
+
simulateKeyDown('z');
|
|
58
|
+
expect(instance._onChange).not.toHaveBeenCalled();
|
|
59
|
+
});
|
|
60
|
+
it('selects a choice in radio mode', function () {
|
|
61
|
+
instance._model.choiceMode = 'radio';
|
|
62
|
+
simulateKeyDown('2');
|
|
63
|
+
expect(instance._onChange).toHaveBeenCalledWith({
|
|
64
|
+
value: '2',
|
|
65
|
+
selected: true
|
|
66
|
+
});
|
|
67
|
+
});
|
|
68
|
+
it('replaces the selected choice in radio mode', function () {
|
|
69
|
+
instance._model.choiceMode = 'radio';
|
|
70
|
+
instance._session.value = ['1'];
|
|
71
|
+
simulateKeyDown('3');
|
|
72
|
+
expect(instance._onChange).toHaveBeenCalledWith({
|
|
73
|
+
value: '3',
|
|
74
|
+
selected: true
|
|
75
|
+
});
|
|
76
|
+
});
|
|
77
|
+
it('does not toggle the same choice in radio mode', function () {
|
|
78
|
+
instance._model.choiceMode = 'radio';
|
|
79
|
+
instance._session.value = ['1'];
|
|
80
|
+
simulateKeyDown('1');
|
|
81
|
+
expect(instance._onChange).toHaveBeenCalledWith({
|
|
82
|
+
value: '1',
|
|
83
|
+
selected: true
|
|
84
|
+
});
|
|
85
|
+
});
|
|
86
|
+
it('does nothing if model is missing', function () {
|
|
87
|
+
instance._model = null;
|
|
88
|
+
simulateKeyDown('1');
|
|
89
|
+
expect(instance._onChange).not.toHaveBeenCalled();
|
|
90
|
+
});
|
|
91
|
+
it('does nothing if session is missing', function () {
|
|
92
|
+
instance._session = null;
|
|
93
|
+
simulateKeyDown('1');
|
|
94
|
+
expect(instance._onChange).not.toHaveBeenCalled();
|
|
95
|
+
});
|
|
96
|
+
});
|
|
97
|
+
});
|
|
98
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/__tests__/key-events-test.jsx"],"names":["describe","instance","beforeEach","MultipleChoice","_model","choices","value","choiceMode","_session","_onChange","jest","fn","simulateKeyDown","key","event","KeyboardEvent","handleKeyDown","it","expect","toHaveBeenCalledWith","selected","not","toHaveBeenCalled"],"mappings":"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","sourcesContent":["import MultipleChoice from '../multiple-choice';\n\ndescribe('MultipleChoice', () => {\n  let instance;\n\n  beforeEach(() => {\n    instance = new MultipleChoice();\n    instance._model = {\n      choices: [\n        { value: '1' },\n        { value: '2' },\n        { value: '3' },\n      ],\n      choiceMode: 'checkbox',\n    };\n    instance._session = { value: [] };\n    instance._onChange = jest.fn();\n  });\n\n  describe('handleKeyDown', () => {\n    const simulateKeyDown = (key) => {\n      const event = new KeyboardEvent('keydown', { key });\n      instance.handleKeyDown(event);\n    };\n\n    it('selects a choice on valid numeric key (checkbox)', () => {\n      simulateKeyDown('1');\n      expect(instance._onChange).toHaveBeenCalledWith({\n        value: '1',\n        selected: true,\n      });\n    });\n\n    it('toggles a choice on valid numeric key (checkbox)', () => {\n      instance._session.value = ['1'];\n      simulateKeyDown('1');\n      expect(instance._onChange).toHaveBeenCalledWith({\n        value: '1',\n        selected: false,\n      });\n    });\n\n    it('selects a choice on valid alphabetic key (checkbox)', () => {\n      simulateKeyDown('a');\n      expect(instance._onChange).toHaveBeenCalledWith({\n        value: '1',\n        selected: true,\n      });\n    });\n\n    it('does nothing on invalid key', () => {\n      simulateKeyDown('z');\n      expect(instance._onChange).not.toHaveBeenCalled();\n    });\n\n    it('selects a choice in radio mode', () => {\n      instance._model.choiceMode = 'radio';\n      simulateKeyDown('2');\n      expect(instance._onChange).toHaveBeenCalledWith({\n        value: '2',\n        selected: true,\n      });\n    });\n\n    it('replaces the selected choice in radio mode', () => {\n      instance._model.choiceMode = 'radio';\n      instance._session.value = ['1'];\n      simulateKeyDown('3');\n      expect(instance._onChange).toHaveBeenCalledWith({\n        value: '3',\n        selected: true,\n      });\n    });\n\n    it('does not toggle the same choice in radio mode', () => {\n      instance._model.choiceMode = 'radio';\n      instance._session.value = ['1'];\n      simulateKeyDown('1');\n      expect(instance._onChange).toHaveBeenCalledWith({\n        value: '1',\n        selected: true,\n      });\n    });\n\n    it('does nothing if model is missing', () => {\n      instance._model = null;\n      simulateKeyDown('1');\n      expect(instance._onChange).not.toHaveBeenCalled();\n    });\n\n    it('does nothing if session is missing', () => {\n      instance._session = null;\n      simulateKeyDown('1');\n      expect(instance._onChange).not.toHaveBeenCalled();\n    });\n  });\n});\n"]}
|