@doenet/doenetml 0.6.0-alpha1

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.
Files changed (549) hide show
  1. package/.prettierrc +3 -0
  2. package/LICENSE +661 -0
  3. package/README.md +146 -0
  4. package/cypress/e2e/ActivityViewer/activityVariants.cy.js +1770 -0
  5. package/cypress/e2e/ActivityViewer/compiledActivity.cy.js +83 -0
  6. package/cypress/e2e/ActivityViewer/relationshipsAmongPages.cy.js +697 -0
  7. package/cypress/e2e/answerValidation/errorinnumbers.cy.js +2125 -0
  8. package/cypress/e2e/answerValidation/factoring.cy.js +1945 -0
  9. package/cypress/e2e/answerValidation/factoringOldAlgorithm.cy.js +892 -0
  10. package/cypress/e2e/answerValidation/functionanswers.cy.js +314 -0
  11. package/cypress/e2e/answerValidation/matchingpatterns.cy.js +287 -0
  12. package/cypress/e2e/answerValidation/matchpartial.cy.js +6711 -0
  13. package/cypress/e2e/answerValidation/pointlocation.cy.js +3989 -0
  14. package/cypress/e2e/answerValidation/symbolicequality.cy.js +1893 -0
  15. package/cypress/e2e/answerValidation/videoProgress.cy.js +210 -0
  16. package/cypress/e2e/assignNames/basiccopy.cy.js +2376 -0
  17. package/cypress/e2e/assignNames/collections.cy.js +9247 -0
  18. package/cypress/e2e/assignNames/selects.cy.js +105 -0
  19. package/cypress/e2e/assignNames/sequences.cy.js +1964 -0
  20. package/cypress/e2e/baseComponent/basecomponentproperties.cy.js +999 -0
  21. package/cypress/e2e/baseComponent/doenetMLtext.cy.js +427 -0
  22. package/cypress/e2e/chemistry/atom.cy.js +201 -0
  23. package/cypress/e2e/chemistry/ion.cy.js +608 -0
  24. package/cypress/e2e/chemistry/ioniccompound.cy.js +133 -0
  25. package/cypress/e2e/dynamicalsystem/cobwebpolyline.cy.js +2653 -0
  26. package/cypress/e2e/dynamicalsystem/equilibriumcurve.cy.js +311 -0
  27. package/cypress/e2e/dynamicalsystem/equilibriumline.cy.js +279 -0
  28. package/cypress/e2e/dynamicalsystem/equilibriumpoint.cy.js +283 -0
  29. package/cypress/e2e/dynamicalsystem/odesystem.cy.js +1834 -0
  30. package/cypress/e2e/equality/mathexpressions.cy.js +948 -0
  31. package/cypress/e2e/graphing/graphreferences.cy.js +978 -0
  32. package/cypress/e2e/graphing/graphreferences2.cy.js +615 -0
  33. package/cypress/e2e/linearAlgebra/eigenDecomposition.cy.js +401 -0
  34. package/cypress/e2e/tagSpecific/angle.cy.js +3898 -0
  35. package/cypress/e2e/tagSpecific/animatefromsequence.cy.js +2306 -0
  36. package/cypress/e2e/tagSpecific/answer.cy.js +31647 -0
  37. package/cypress/e2e/tagSpecific/bestfitline.cy.js +612 -0
  38. package/cypress/e2e/tagSpecific/blockquote.cy.js +30 -0
  39. package/cypress/e2e/tagSpecific/boolean.cy.js +742 -0
  40. package/cypress/e2e/tagSpecific/booleaninput.cy.js +1283 -0
  41. package/cypress/e2e/tagSpecific/booleanlist.cy.js +588 -0
  42. package/cypress/e2e/tagSpecific/booleanoperators.cy.js +596 -0
  43. package/cypress/e2e/tagSpecific/booleanoperatorsonmath.cy.js +498 -0
  44. package/cypress/e2e/tagSpecific/callaction.cy.js +2835 -0
  45. package/cypress/e2e/tagSpecific/choiceinput.cy.js +3205 -0
  46. package/cypress/e2e/tagSpecific/circle.cy.js +22036 -0
  47. package/cypress/e2e/tagSpecific/codeeditor.cy.js +1995 -0
  48. package/cypress/e2e/tagSpecific/collect.cy.js +5035 -0
  49. package/cypress/e2e/tagSpecific/componentsize.cy.js +502 -0
  50. package/cypress/e2e/tagSpecific/conditionalcontent.cy.js +3495 -0
  51. package/cypress/e2e/tagSpecific/contentBrowser.cy.js +335 -0
  52. package/cypress/e2e/tagSpecific/contentpicker.cy.js +261 -0
  53. package/cypress/e2e/tagSpecific/copy.cy.js +12627 -0
  54. package/cypress/e2e/tagSpecific/copy2.cy.js +5698 -0
  55. package/cypress/e2e/tagSpecific/curve.bezier.cy.js +12440 -0
  56. package/cypress/e2e/tagSpecific/curve.cy.js +1716 -0
  57. package/cypress/e2e/tagSpecific/curve.function.cy.js +1471 -0
  58. package/cypress/e2e/tagSpecific/curve.parametrized.cy.js +920 -0
  59. package/cypress/e2e/tagSpecific/document.cy.js +234 -0
  60. package/cypress/e2e/tagSpecific/endpoint.cy.js +197 -0
  61. package/cypress/e2e/tagSpecific/evaluate.cy.js +8895 -0
  62. package/cypress/e2e/tagSpecific/extract.cy.js +2282 -0
  63. package/cypress/e2e/tagSpecific/feedback.cy.js +2941 -0
  64. package/cypress/e2e/tagSpecific/function.cy.js +9450 -0
  65. package/cypress/e2e/tagSpecific/functioniterates.cy.js +1178 -0
  66. package/cypress/e2e/tagSpecific/functionoperators.cy.js +4047 -0
  67. package/cypress/e2e/tagSpecific/graph.cy.js +2491 -0
  68. package/cypress/e2e/tagSpecific/group.cy.js +683 -0
  69. package/cypress/e2e/tagSpecific/hint.cy.js +204 -0
  70. package/cypress/e2e/tagSpecific/image.cy.js +770 -0
  71. package/cypress/e2e/tagSpecific/integer.cy.js +206 -0
  72. package/cypress/e2e/tagSpecific/label.cy.js +800 -0
  73. package/cypress/e2e/tagSpecific/legend.cy.js +1001 -0
  74. package/cypress/e2e/tagSpecific/line.cy.js +12167 -0
  75. package/cypress/e2e/tagSpecific/linesegment.cy.js +4749 -0
  76. package/cypress/e2e/tagSpecific/lorem.cy.js +289 -0
  77. package/cypress/e2e/tagSpecific/map.cy.js +4476 -0
  78. package/cypress/e2e/tagSpecific/matchespattern.cy.js +693 -0
  79. package/cypress/e2e/tagSpecific/math.cy.js +10990 -0
  80. package/cypress/e2e/tagSpecific/mathdisplay.cy.js +2689 -0
  81. package/cypress/e2e/tagSpecific/mathinput.cy.js +15628 -0
  82. package/cypress/e2e/tagSpecific/mathinputgraph.cy.js +566 -0
  83. package/cypress/e2e/tagSpecific/mathlist.cy.js +4073 -0
  84. package/cypress/e2e/tagSpecific/mathoperators.cy.js +13851 -0
  85. package/cypress/e2e/tagSpecific/matrix.cy.js +8825 -0
  86. package/cypress/e2e/tagSpecific/matrixinput.cy.js +16277 -0
  87. package/cypress/e2e/tagSpecific/module.cy.js +1771 -0
  88. package/cypress/e2e/tagSpecific/number.cy.js +2221 -0
  89. package/cypress/e2e/tagSpecific/numberlist.cy.js +1285 -0
  90. package/cypress/e2e/tagSpecific/p.cy.js +72 -0
  91. package/cypress/e2e/tagSpecific/paginator.cy.js +2983 -0
  92. package/cypress/e2e/tagSpecific/parabola.cy.js +14331 -0
  93. package/cypress/e2e/tagSpecific/paragraphmarkup.cy.js +104 -0
  94. package/cypress/e2e/tagSpecific/periodicset.cy.js +1439 -0
  95. package/cypress/e2e/tagSpecific/piecewisefunction.cy.js +1055 -0
  96. package/cypress/e2e/tagSpecific/pluralize.cy.js +274 -0
  97. package/cypress/e2e/tagSpecific/point.cy.js +8895 -0
  98. package/cypress/e2e/tagSpecific/point2.cy.js +10259 -0
  99. package/cypress/e2e/tagSpecific/polygon.cy.js +5039 -0
  100. package/cypress/e2e/tagSpecific/polyline.cy.js +4704 -0
  101. package/cypress/e2e/tagSpecific/problem.cy.js +2768 -0
  102. package/cypress/e2e/tagSpecific/ray.cy.js +10770 -0
  103. package/cypress/e2e/tagSpecific/rectangle.cy.js +2143 -0
  104. package/cypress/e2e/tagSpecific/ref.cy.js +420 -0
  105. package/cypress/e2e/tagSpecific/regularPolygon.cy.js +1006 -0
  106. package/cypress/e2e/tagSpecific/regularPolygon2.cy.js +100 -0
  107. package/cypress/e2e/tagSpecific/regularPolygon3.cy.js +777 -0
  108. package/cypress/e2e/tagSpecific/samplerandomnumbers.cy.js +3619 -0
  109. package/cypress/e2e/tagSpecific/sectioning.cy.js +3530 -0
  110. package/cypress/e2e/tagSpecific/select.cy.js +5376 -0
  111. package/cypress/e2e/tagSpecific/selectfromsequence.cy.js +3846 -0
  112. package/cypress/e2e/tagSpecific/selectrandomnumbers.cy.js +2914 -0
  113. package/cypress/e2e/tagSpecific/sequence.cy.js +2093 -0
  114. package/cypress/e2e/tagSpecific/shuffle.cy.js +490 -0
  115. package/cypress/e2e/tagSpecific/sidebyside.cy.js +8057 -0
  116. package/cypress/e2e/tagSpecific/singlecharactercomponents.cy.js +72 -0
  117. package/cypress/e2e/tagSpecific/slider.cy.js +914 -0
  118. package/cypress/e2e/tagSpecific/solution.cy.js +109 -0
  119. package/cypress/e2e/tagSpecific/solveequations.cy.js +1026 -0
  120. package/cypress/e2e/tagSpecific/sort.cy.js +1685 -0
  121. package/cypress/e2e/tagSpecific/spreadsheet.cy.js +5971 -0
  122. package/cypress/e2e/tagSpecific/subsetofreals.cy.js +2725 -0
  123. package/cypress/e2e/tagSpecific/substitute.cy.js +2646 -0
  124. package/cypress/e2e/tagSpecific/tabular.cy.js +36 -0
  125. package/cypress/e2e/tagSpecific/text.cy.js +975 -0
  126. package/cypress/e2e/tagSpecific/textinput.cy.js +2177 -0
  127. package/cypress/e2e/tagSpecific/textlist.cy.js +369 -0
  128. package/cypress/e2e/tagSpecific/triangle.cy.js +1936 -0
  129. package/cypress/e2e/tagSpecific/triggerset.cy.js +2023 -0
  130. package/cypress/e2e/tagSpecific/updatevalue.cy.js +3288 -0
  131. package/cypress/e2e/tagSpecific/vector.cy.js +20183 -0
  132. package/cypress/e2e/tagSpecific/video.cy.js +612 -0
  133. package/cypress/e2e/tagSpecific/when.cy.js +202 -0
  134. package/cypress/e2e/variants/specifysinglevariant.cy.js +6726 -0
  135. package/cypress/e2e/variants/uniquevariants.cy.js +4846 -0
  136. package/cypress/fixtures/example.json +5 -0
  137. package/cypress/support/commands.js +32 -0
  138. package/cypress/support/e2e.js +31 -0
  139. package/cypress.config.js +18 -0
  140. package/docs/codeSnippet.jsx +11 -0
  141. package/docs/index.html +133 -0
  142. package/docs/index.jsx +138 -0
  143. package/docs/prism.css +3 -0
  144. package/index.html +14 -0
  145. package/index.js +21 -0
  146. package/media/answer_example.png +0 -0
  147. package/media/graph_example.png +0 -0
  148. package/media/graph_markup_example.png +0 -0
  149. package/package.json +83 -0
  150. package/public/favicon.ico +0 -0
  151. package/public/fonts/files/open-sans-v18-latin-700.woff +0 -0
  152. package/public/fonts/files/open-sans-v18-latin-700.woff2 +0 -0
  153. package/public/fonts/files/open-sans-v18-latin-700italic.woff +0 -0
  154. package/public/fonts/files/open-sans-v18-latin-700italic.woff2 +0 -0
  155. package/public/fonts/files/open-sans-v18-latin-italic.woff +0 -0
  156. package/public/fonts/files/open-sans-v18-latin-italic.woff2 +0 -0
  157. package/public/fonts/files/open-sans-v18-latin-light-italic.woff +0 -0
  158. package/public/fonts/files/open-sans-v18-latin-light-italic.woff2 +0 -0
  159. package/public/fonts/files/open-sans-v18-latin-light.woff +0 -0
  160. package/public/fonts/files/open-sans-v18-latin-light.woff2 +0 -0
  161. package/public/fonts/files/open-sans-v18-latin-regular.woff +0 -0
  162. package/public/fonts/files/open-sans-v18-latin-regular.woff2 +0 -0
  163. package/src/Core/ComponentTypes.js +426 -0
  164. package/src/Core/Core.js +11852 -0
  165. package/src/Core/CoreWorker.js +127 -0
  166. package/src/Core/Dependencies.js +8226 -0
  167. package/src/Core/Numerics.js +473 -0
  168. package/src/Core/ParameterStack.js +36 -0
  169. package/src/Core/ReadOnlyProxyHandler.js +41 -0
  170. package/src/Core/StateProxyHandler.js +88 -0
  171. package/src/Core/components/Aliases.js +67 -0
  172. package/src/Core/components/Angle.js +758 -0
  173. package/src/Core/components/AnimateFromSequence.js +922 -0
  174. package/src/Core/components/Answer.js +2087 -0
  175. package/src/Core/components/AsList.js +83 -0
  176. package/src/Core/components/AttractTo.js +245 -0
  177. package/src/Core/components/AttractToAngles.js +262 -0
  178. package/src/Core/components/AttractToConstraint.js +104 -0
  179. package/src/Core/components/AttractToGrid.js +315 -0
  180. package/src/Core/components/Award.js +906 -0
  181. package/src/Core/components/BestFitLine.js +318 -0
  182. package/src/Core/components/BezierControls.js +719 -0
  183. package/src/Core/components/BlockQuote.js +35 -0
  184. package/src/Core/components/Boolean.js +500 -0
  185. package/src/Core/components/BooleanInput.js +330 -0
  186. package/src/Core/components/BooleanList.js +396 -0
  187. package/src/Core/components/BooleanOperators.js +35 -0
  188. package/src/Core/components/BooleanOperatorsOfMath.js +148 -0
  189. package/src/Core/components/CallAction.js +261 -0
  190. package/src/Core/components/Caption.js +73 -0
  191. package/src/Core/components/Case.js +56 -0
  192. package/src/Core/components/Cell.js +439 -0
  193. package/src/Core/components/CellBlock.js +64 -0
  194. package/src/Core/components/Chart.js +795 -0
  195. package/src/Core/components/Choice.js +266 -0
  196. package/src/Core/components/ChoiceInput.js +1407 -0
  197. package/src/Core/components/Circle.js +2884 -0
  198. package/src/Core/components/CodeEditor.js +647 -0
  199. package/src/Core/components/CodeViewer.js +294 -0
  200. package/src/Core/components/CollaborateGroupSetup.js +46 -0
  201. package/src/Core/components/CollaborateGroups.js +119 -0
  202. package/src/Core/components/Collect.js +850 -0
  203. package/src/Core/components/Column.js +608 -0
  204. package/src/Core/components/ConditionalContent.js +468 -0
  205. package/src/Core/components/ConsiderAsResponses.js +49 -0
  206. package/src/Core/components/ConstrainTo.js +161 -0
  207. package/src/Core/components/ConstrainToAngles.js +244 -0
  208. package/src/Core/components/ConstrainToGraph.js +142 -0
  209. package/src/Core/components/ConstrainToGrid.js +175 -0
  210. package/src/Core/components/ConstraintUnion.js +119 -0
  211. package/src/Core/components/Constraints.js +497 -0
  212. package/src/Core/components/ContentBrowser.js +441 -0
  213. package/src/Core/components/ContentPicker.js +263 -0
  214. package/src/Core/components/ControlVectors.js +25 -0
  215. package/src/Core/components/Coords.js +63 -0
  216. package/src/Core/components/Copy.js +3412 -0
  217. package/src/Core/components/Curve.js +4130 -0
  218. package/src/Core/components/CustomAttribute.js +175 -0
  219. package/src/Core/components/DataFrame.js +357 -0
  220. package/src/Core/components/DiscreteSimulationResultList.js +342 -0
  221. package/src/Core/components/DiscreteSimulationResultPolyline.js +581 -0
  222. package/src/Core/components/Divisions.js +55 -0
  223. package/src/Core/components/Document.js +888 -0
  224. package/src/Core/components/Embed.js +65 -0
  225. package/src/Core/components/Endpoint.js +62 -0
  226. package/src/Core/components/Evaluate.js +321 -0
  227. package/src/Core/components/Extract.js +656 -0
  228. package/src/Core/components/Extrema.js +556 -0
  229. package/src/Core/components/Feedback.js +200 -0
  230. package/src/Core/components/FeedbackDefinitions.js +97 -0
  231. package/src/Core/components/Figure.js +148 -0
  232. package/src/Core/components/Footnote.js +73 -0
  233. package/src/Core/components/Function.js +5344 -0
  234. package/src/Core/components/FunctionIterates.js +306 -0
  235. package/src/Core/components/FunctionOperators.js +702 -0
  236. package/src/Core/components/Graph.js +1679 -0
  237. package/src/Core/components/Group.js +7 -0
  238. package/src/Core/components/HasSameFactoring.js +407 -0
  239. package/src/Core/components/Hint.js +241 -0
  240. package/src/Core/components/Image.js +524 -0
  241. package/src/Core/components/Indexing.js +79 -0
  242. package/src/Core/components/IntComma.js +64 -0
  243. package/src/Core/components/Integer.js +81 -0
  244. package/src/Core/components/Intersection.js +328 -0
  245. package/src/Core/components/Interval.js +29 -0
  246. package/src/Core/components/Label.js +492 -0
  247. package/src/Core/components/Latex.js +104 -0
  248. package/src/Core/components/Legend.js +329 -0
  249. package/src/Core/components/Line.js +2040 -0
  250. package/src/Core/components/LineSegment.js +882 -0
  251. package/src/Core/components/Lists.js +180 -0
  252. package/src/Core/components/Lorem.js +249 -0
  253. package/src/Core/components/MMeMen.js +377 -0
  254. package/src/Core/components/Map.js +873 -0
  255. package/src/Core/components/Markers.js +101 -0
  256. package/src/Core/components/MatchesPattern.js +339 -0
  257. package/src/Core/components/Math.js +2552 -0
  258. package/src/Core/components/MathInput.js +948 -0
  259. package/src/Core/components/MathList.js +828 -0
  260. package/src/Core/components/MathOperators.js +1286 -0
  261. package/src/Core/components/Matrix.js +497 -0
  262. package/src/Core/components/MatrixInput.js +3157 -0
  263. package/src/Core/components/MdMdnMrow.js +394 -0
  264. package/src/Core/components/Module.js +16 -0
  265. package/src/Core/components/Number.js +1031 -0
  266. package/src/Core/components/NumberList.js +550 -0
  267. package/src/Core/components/Option.js +24 -0
  268. package/src/Core/components/P.js +71 -0
  269. package/src/Core/components/Paginator.js +338 -0
  270. package/src/Core/components/Panel.js +126 -0
  271. package/src/Core/components/Parabola.js +1561 -0
  272. package/src/Core/components/ParagraphMarkup.js +59 -0
  273. package/src/Core/components/Pegboard.js +43 -0
  274. package/src/Core/components/PeriodicSet.js +291 -0
  275. package/src/Core/components/PiecewiseFunction.js +832 -0
  276. package/src/Core/components/Pluralize.js +198 -0
  277. package/src/Core/components/Point.js +1295 -0
  278. package/src/Core/components/Polygon.js +408 -0
  279. package/src/Core/components/Polyline.js +841 -0
  280. package/src/Core/components/RandomizedTextList.js +225 -0
  281. package/src/Core/components/Ray.js +1737 -0
  282. package/src/Core/components/Rectangle.js +1535 -0
  283. package/src/Core/components/Ref.js +350 -0
  284. package/src/Core/components/RegionBetweenCurveXAxis.js +124 -0
  285. package/src/Core/components/RegionHalfPlane.js +107 -0
  286. package/src/Core/components/RegularPolygon.js +2118 -0
  287. package/src/Core/components/RenderDoenetML.js +181 -0
  288. package/src/Core/components/Row.js +780 -0
  289. package/src/Core/components/SampleRandomNumbers.js +653 -0
  290. package/src/Core/components/Sectioning.js +303 -0
  291. package/src/Core/components/Select.js +947 -0
  292. package/src/Core/components/SelectFromSequence.js +1242 -0
  293. package/src/Core/components/SelectRandomNumbers.js +225 -0
  294. package/src/Core/components/Sequence.js +444 -0
  295. package/src/Core/components/Setup.js +53 -0
  296. package/src/Core/components/Shuffle.js +470 -0
  297. package/src/Core/components/SideBySide.js +2130 -0
  298. package/src/Core/components/SingleCharacterComponents.js +41 -0
  299. package/src/Core/components/Slider.js +819 -0
  300. package/src/Core/components/SolutionContainer.js +67 -0
  301. package/src/Core/components/Solutions.js +334 -0
  302. package/src/Core/components/SolveEquations.js +568 -0
  303. package/src/Core/components/Sort.js +398 -0
  304. package/src/Core/components/Sources.js +108 -0
  305. package/src/Core/components/Split.js +205 -0
  306. package/src/Core/components/Spreadsheet.js +1507 -0
  307. package/src/Core/components/StyleDefinitions.js +111 -0
  308. package/src/Core/components/SubsetOfReals.js +348 -0
  309. package/src/Core/components/SubsetOfRealsInput.js +1474 -0
  310. package/src/Core/components/Substitute.js +496 -0
  311. package/src/Core/components/SummaryStatistics.js +652 -0
  312. package/src/Core/components/Table.js +145 -0
  313. package/src/Core/components/Tabular.js +384 -0
  314. package/src/Core/components/Template.js +360 -0
  315. package/src/Core/components/Text.js +341 -0
  316. package/src/Core/components/TextInput.js +566 -0
  317. package/src/Core/components/TextList.js +442 -0
  318. package/src/Core/components/TextListFromString.js +137 -0
  319. package/src/Core/components/TextOperatorsOfMath.js +21 -0
  320. package/src/Core/components/Triangle.js +280 -0
  321. package/src/Core/components/TriggerSet.js +189 -0
  322. package/src/Core/components/TupleList.js +43 -0
  323. package/src/Core/components/UpdateValue.js +435 -0
  324. package/src/Core/components/VariantControl.js +36 -0
  325. package/src/Core/components/Vector.js +2478 -0
  326. package/src/Core/components/Verbatim.js +125 -0
  327. package/src/Core/components/Video.js +673 -0
  328. package/src/Core/components/When.js +198 -0
  329. package/src/Core/components/abstract/AngleListComponent.js +140 -0
  330. package/src/Core/components/abstract/BaseComponent.js +1496 -0
  331. package/src/Core/components/abstract/BlockComponent.js +5 -0
  332. package/src/Core/components/abstract/BooleanBaseOperator.js +88 -0
  333. package/src/Core/components/abstract/BooleanBaseOperatorOfMath.js +100 -0
  334. package/src/Core/components/abstract/BooleanBaseOperatorOneInput.js +44 -0
  335. package/src/Core/components/abstract/ComponentSize.js +789 -0
  336. package/src/Core/components/abstract/ComponentWithSelectableType.js +537 -0
  337. package/src/Core/components/abstract/CompositeComponent.js +142 -0
  338. package/src/Core/components/abstract/ConstraintComponent.js +19 -0
  339. package/src/Core/components/abstract/FunctionBaseOperator.js +680 -0
  340. package/src/Core/components/abstract/GraphicalComponent.js +56 -0
  341. package/src/Core/components/abstract/InlineComponent.js +5 -0
  342. package/src/Core/components/abstract/InlineRenderInlineChildren.js +63 -0
  343. package/src/Core/components/abstract/Input.js +192 -0
  344. package/src/Core/components/abstract/IntervalListComponent.js +218 -0
  345. package/src/Core/components/abstract/LineListComponent.js +114 -0
  346. package/src/Core/components/abstract/MathBaseOperator.js +631 -0
  347. package/src/Core/components/abstract/MathBaseOperatorOneInput.js +112 -0
  348. package/src/Core/components/abstract/PointListComponent.js +238 -0
  349. package/src/Core/components/abstract/SectioningComponent.js +1262 -0
  350. package/src/Core/components/abstract/SingleCharacterInline.js +23 -0
  351. package/src/Core/components/abstract/TextBaseOperatorOfMath.js +47 -0
  352. package/src/Core/components/abstract/TextOrInline.js +66 -0
  353. package/src/Core/components/abstract/VariableName.js +31 -0
  354. package/src/Core/components/abstract/VariableNameList.js +83 -0
  355. package/src/Core/components/abstract/VectorListComponent.js +235 -0
  356. package/src/Core/components/chemistry/Atom.js +910 -0
  357. package/src/Core/components/chemistry/ElectronConfiguration.js +36 -0
  358. package/src/Core/components/chemistry/Ion.js +684 -0
  359. package/src/Core/components/chemistry/IonicCompound.js +189 -0
  360. package/src/Core/components/chemistry/OrbitalDiagram.js +175 -0
  361. package/src/Core/components/chemistry/OrbitalDiagramInput.js +753 -0
  362. package/src/Core/components/chemistry/index.js +6 -0
  363. package/src/Core/components/commonsugar/breakstrings.js +627 -0
  364. package/src/Core/components/commonsugar/lists.js +177 -0
  365. package/src/Core/components/dynamicalSystems/CobwebPolyline.js +913 -0
  366. package/src/Core/components/dynamicalSystems/EquilibriumCurve.js +95 -0
  367. package/src/Core/components/dynamicalSystems/EquilibriumLine.js +93 -0
  368. package/src/Core/components/dynamicalSystems/EquilibriumPoint.js +93 -0
  369. package/src/Core/components/dynamicalSystems/ODESystem.js +943 -0
  370. package/src/Core/components/dynamicalSystems/index.js +5 -0
  371. package/src/Core/components/linearAlgebra/EigenDecomposition.js +294 -0
  372. package/src/Core/utils/array.js +30 -0
  373. package/src/Core/utils/booleanLogic.js +965 -0
  374. package/src/Core/utils/checkEquality.js +818 -0
  375. package/src/Core/utils/cid.js +29 -0
  376. package/src/Core/utils/componentInfoObjects.js +100 -0
  377. package/src/Core/utils/constraints.js +23 -0
  378. package/src/Core/utils/copy.js +572 -0
  379. package/src/Core/utils/deepFunctions.js +173 -0
  380. package/src/Core/utils/descendants.js +252 -0
  381. package/src/Core/utils/enumeration.js +234 -0
  382. package/src/Core/utils/feedback.js +84 -0
  383. package/src/Core/utils/function.js +1343 -0
  384. package/src/Core/utils/graphical.js +196 -0
  385. package/src/Core/utils/label.js +396 -0
  386. package/src/Core/utils/math.js +1056 -0
  387. package/src/Core/utils/naming.js +45 -0
  388. package/src/Core/utils/periodicSetEquality.js +403 -0
  389. package/src/Core/utils/randomNumbers.js +70 -0
  390. package/src/Core/utils/retrieveMedia.js +98 -0
  391. package/src/Core/utils/retrieveTextFile.js +140 -0
  392. package/src/Core/utils/returnAllPossibleVariants.js +73 -0
  393. package/src/Core/utils/rounding.js +316 -0
  394. package/src/Core/utils/sequence.js +754 -0
  395. package/src/Core/utils/serializedStateProcessing.js +4049 -0
  396. package/src/Core/utils/size.js +22 -0
  397. package/src/Core/utils/stateVariables.js +138 -0
  398. package/src/Core/utils/style.js +535 -0
  399. package/src/Core/utils/subset-of-reals.js +796 -0
  400. package/src/Core/utils/table.js +41 -0
  401. package/src/Core/utils/text.js +16 -0
  402. package/src/Core/utils/triggering.js +167 -0
  403. package/src/Core/utils/variants.js +477 -0
  404. package/src/DoenetML.css +308 -0
  405. package/src/DoenetML.jsx +201 -0
  406. package/src/Parser/doenet.grammar +90 -0
  407. package/src/Parser/doenet.js +33 -0
  408. package/src/Parser/doenet.terms.js +20 -0
  409. package/src/Parser/parser.js +266 -0
  410. package/src/Parser/tokens.js +129 -0
  411. package/src/Tools/CodeMirror.jsx +440 -0
  412. package/src/Tools/DarkmodeController.jsx +21 -0
  413. package/src/Tools/Footers/MathInputSelector.jsx +34 -0
  414. package/src/Tools/Footers/VirtualKeyboard.jsx +751 -0
  415. package/src/Tools/cypressTest/CypressTest.jsx +341 -0
  416. package/src/Tools/cypressTest/index.html +102 -0
  417. package/src/Tools/cypressTest/index.jsx +40 -0
  418. package/src/Viewer/ActivityViewer.jsx +1461 -0
  419. package/src/Viewer/PageViewer.jsx +1329 -0
  420. package/src/Viewer/renderers/alert.jsx +17 -0
  421. package/src/Viewer/renderers/angle.jsx +209 -0
  422. package/src/Viewer/renderers/answer.jsx +206 -0
  423. package/src/Viewer/renderers/asList.jsx +25 -0
  424. package/src/Viewer/renderers/blockQuote.jsx +41 -0
  425. package/src/Viewer/renderers/boolean.jsx +17 -0
  426. package/src/Viewer/renderers/booleanInput.css +105 -0
  427. package/src/Viewer/renderers/booleanInput.jsx +636 -0
  428. package/src/Viewer/renderers/button.jsx +369 -0
  429. package/src/Viewer/renderers/c.jsx +17 -0
  430. package/src/Viewer/renderers/callAction.jsx +18 -0
  431. package/src/Viewer/renderers/cell.jsx +59 -0
  432. package/src/Viewer/renderers/chart.jsx +83 -0
  433. package/src/Viewer/renderers/choiceInput.css +223 -0
  434. package/src/Viewer/renderers/choiceInput.jsx +535 -0
  435. package/src/Viewer/renderers/circle.jsx +990 -0
  436. package/src/Viewer/renderers/cobwebPolyline.jsx +442 -0
  437. package/src/Viewer/renderers/codeEditor.jsx +248 -0
  438. package/src/Viewer/renderers/codeViewer.jsx +105 -0
  439. package/src/Viewer/renderers/containerBlock.jsx +41 -0
  440. package/src/Viewer/renderers/containerInline.jsx +17 -0
  441. package/src/Viewer/renderers/contentBrowser.jsx +159 -0
  442. package/src/Viewer/renderers/contentPicker.jsx +160 -0
  443. package/src/Viewer/renderers/curve.jsx +1072 -0
  444. package/src/Viewer/renderers/ellipsis.jsx +17 -0
  445. package/src/Viewer/renderers/em.jsx +17 -0
  446. package/src/Viewer/renderers/embed.jsx +110 -0
  447. package/src/Viewer/renderers/feedback.jsx +74 -0
  448. package/src/Viewer/renderers/figure.jsx +131 -0
  449. package/src/Viewer/renderers/footnote.jsx +52 -0
  450. package/src/Viewer/renderers/graph.jsx +925 -0
  451. package/src/Viewer/renderers/hint.jsx +142 -0
  452. package/src/Viewer/renderers/image.jsx +581 -0
  453. package/src/Viewer/renderers/jsxgraph-distrib/jsxgraphcore.mjs +2 -0
  454. package/src/Viewer/renderers/jsxgraph-distrib/jsxgraphcore.mjs.map +1 -0
  455. package/src/Viewer/renderers/label.jsx +470 -0
  456. package/src/Viewer/renderers/legend.jsx +306 -0
  457. package/src/Viewer/renderers/line.jsx +511 -0
  458. package/src/Viewer/renderers/lineSegment.jsx +754 -0
  459. package/src/Viewer/renderers/list.jsx +111 -0
  460. package/src/Viewer/renderers/lq.jsx +12 -0
  461. package/src/Viewer/renderers/lsq.jsx +12 -0
  462. package/src/Viewer/renderers/math.jsx +582 -0
  463. package/src/Viewer/renderers/mathInput.css +10 -0
  464. package/src/Viewer/renderers/mathInput.jsx +425 -0
  465. package/src/Viewer/renderers/mathInputog.jsx +534 -0
  466. package/src/Viewer/renderers/mathList.jsx +39 -0
  467. package/src/Viewer/renderers/matrixInput.jsx +317 -0
  468. package/src/Viewer/renderers/mdash.jsx +12 -0
  469. package/src/Viewer/renderers/nbsp.jsx +12 -0
  470. package/src/Viewer/renderers/ndash.jsx +12 -0
  471. package/src/Viewer/renderers/number.jsx +454 -0
  472. package/src/Viewer/renderers/numberList.jsx +35 -0
  473. package/src/Viewer/renderers/orbitalDiagram.jsx +247 -0
  474. package/src/Viewer/renderers/orbitalDiagramInput.jsx +450 -0
  475. package/src/Viewer/renderers/p.jsx +38 -0
  476. package/src/Viewer/renderers/paginatorControls.jsx +41 -0
  477. package/src/Viewer/renderers/pegboard.jsx +239 -0
  478. package/src/Viewer/renderers/point.jsx +649 -0
  479. package/src/Viewer/renderers/polygon.jsx +612 -0
  480. package/src/Viewer/renderers/polyline.jsx +608 -0
  481. package/src/Viewer/renderers/pre.jsx +34 -0
  482. package/src/Viewer/renderers/q.jsx +17 -0
  483. package/src/Viewer/renderers/ray.jsx +410 -0
  484. package/src/Viewer/renderers/ref.jsx +149 -0
  485. package/src/Viewer/renderers/regionBetweenCurveXAxis.jsx +182 -0
  486. package/src/Viewer/renderers/renderDoenetML.jsx +56 -0
  487. package/src/Viewer/renderers/row.jsx +31 -0
  488. package/src/Viewer/renderers/rq.jsx +12 -0
  489. package/src/Viewer/renderers/rsq.jsx +12 -0
  490. package/src/Viewer/renderers/section.jsx +427 -0
  491. package/src/Viewer/renderers/sideBySide.jsx +80 -0
  492. package/src/Viewer/renderers/slider.jsx +800 -0
  493. package/src/Viewer/renderers/solution.jsx +134 -0
  494. package/src/Viewer/renderers/spreadsheet.jsx +83 -0
  495. package/src/Viewer/renderers/sq.jsx +17 -0
  496. package/src/Viewer/renderers/styles/global.css +14 -0
  497. package/src/Viewer/renderers/subsetOfRealsInput.jsx +392 -0
  498. package/src/Viewer/renderers/summaryStatistics.jsx +83 -0
  499. package/src/Viewer/renderers/table.jsx +78 -0
  500. package/src/Viewer/renderers/tabular.jsx +58 -0
  501. package/src/Viewer/renderers/tag.jsx +26 -0
  502. package/src/Viewer/renderers/text.jsx +439 -0
  503. package/src/Viewer/renderers/textInput.jsx +774 -0
  504. package/src/Viewer/renderers/textList.jsx +30 -0
  505. package/src/Viewer/renderers/triggerSet.jsx +52 -0
  506. package/src/Viewer/renderers/updateValue.jsx +30 -0
  507. package/src/Viewer/renderers/utils/css.js +13 -0
  508. package/src/Viewer/renderers/utils/graph.js +159 -0
  509. package/src/Viewer/renderers/utils/offGraphIndicators.js +91 -0
  510. package/src/Viewer/renderers/vector.jsx +678 -0
  511. package/src/Viewer/renderers/video.jsx +494 -0
  512. package/src/Viewer/useDoenetRenderer.jsx +128 -0
  513. package/src/main.jsx +16 -0
  514. package/src/media/fonts/files/open-sans-v18-latin-700.woff +0 -0
  515. package/src/media/fonts/files/open-sans-v18-latin-700.woff2 +0 -0
  516. package/src/media/fonts/files/open-sans-v18-latin-700italic.woff +0 -0
  517. package/src/media/fonts/files/open-sans-v18-latin-700italic.woff2 +0 -0
  518. package/src/media/fonts/files/open-sans-v18-latin-italic.woff +0 -0
  519. package/src/media/fonts/files/open-sans-v18-latin-italic.woff2 +0 -0
  520. package/src/media/fonts/files/open-sans-v18-latin-light-italic.woff +0 -0
  521. package/src/media/fonts/files/open-sans-v18-latin-light-italic.woff2 +0 -0
  522. package/src/media/fonts/files/open-sans-v18-latin-light.woff +0 -0
  523. package/src/media/fonts/files/open-sans-v18-latin-light.woff2 +0 -0
  524. package/src/media/fonts/files/open-sans-v18-latin-regular.woff +0 -0
  525. package/src/media/fonts/files/open-sans-v18-latin-regular.woff2 +0 -0
  526. package/src/test/testCode.doenet +26 -0
  527. package/src/test/testViewer.jsx +158 -0
  528. package/src/uiComponents/ActionButton.jsx +157 -0
  529. package/src/uiComponents/ActionButtonGroup.jsx +93 -0
  530. package/src/uiComponents/Button.jsx +160 -0
  531. package/src/uiComponents/ButtonGroup.jsx +56 -0
  532. package/src/uiComponents/ToggleButton.jsx +194 -0
  533. package/src/uiComponents/ToggleButtonGroup.jsx +77 -0
  534. package/src/utils/activityUtils.js +713 -0
  535. package/src/utils/array.js +17 -0
  536. package/src/utils/cid.js +34 -0
  537. package/src/utils/componentInfoObjects.js +89 -0
  538. package/src/utils/deepFunctions.js +165 -0
  539. package/src/utils/enumeration.js +226 -0
  540. package/src/utils/math.js +624 -0
  541. package/src/utils/naming.js +44 -0
  542. package/src/utils/retrieveTextFile.js +156 -0
  543. package/src/utils/returnAllPossibleVariants.js +81 -0
  544. package/src/utils/sequence.js +715 -0
  545. package/src/utils/serialize.js +29 -0
  546. package/src/utils/serializedStateProcessing.js +2587 -0
  547. package/src/utils/subset-of-reals.js +783 -0
  548. package/src/utils/url.js +19 -0
  549. package/vite.config.js +14 -0
@@ -0,0 +1,3205 @@
1
+ import me from "math-expressions";
2
+ import { cesc, cesc2 } from "../../../../src/utils/url";
3
+
4
+ describe("ChoiceInput Tag Tests", function () {
5
+ beforeEach(() => {
6
+ cy.clearIndexedDB();
7
+ cy.visit("/src/Tools/cypressTest/");
8
+ });
9
+
10
+ it("default is block format, shuffleOrder", () => {
11
+ cy.window().then(async (win) => {
12
+ win.postMessage(
13
+ {
14
+ doenetML: `
15
+ <text>a</text>
16
+ <choiceinput shuffleOrder>
17
+ <choice>cat</choice>
18
+ <choice>dog</choice>
19
+ <choice>monkey</choice>
20
+ <choice>mouse</choice>
21
+ </choiceinput>
22
+
23
+ <p>Selected value: <copy prop='selectedvalue' target="_choiceinput1" /></p>
24
+ <p>Selected index: <copy prop='selectedindex' target="_choiceinput1" /></p>
25
+
26
+ <p name="pCat">Selected cat: $_choice1.selected</p>
27
+ <p name="pDog">Selected dog: $_choice2.selected</p>
28
+ <p name="pMonkey">Selected monkey: $_choice3.selected</p>
29
+ <p name="pMouse">Selected mouse: $_choice4.selected</p>
30
+ `,
31
+ requestedVariantIndex: 8,
32
+ },
33
+ "*",
34
+ );
35
+ });
36
+
37
+ cy.get(cesc("#\\/_text1")).should("have.text", "a"); // to wait for page to load
38
+
39
+ let originalChoices = ["cat", "dog", "monkey", "mouse"];
40
+ cy.get(cesc("#\\/_p1")).should("have.text", "Selected value: ");
41
+ cy.get(cesc("#\\/_p2")).should("have.text", "Selected index: ");
42
+
43
+ cy.get(cesc("#\\/pCat")).should("have.text", "Selected cat: false");
44
+ cy.get(cesc("#\\/pDog")).should("have.text", "Selected dog: false");
45
+ cy.get(cesc("#\\/pMonkey")).should("have.text", "Selected monkey: false");
46
+ cy.get(cesc("#\\/pMouse")).should("have.text", "Selected mouse: false");
47
+
48
+ let choices, choiceOrder;
49
+ cy.window().then(async (win) => {
50
+ let stateVariables = await win.returnAllStateVariables1();
51
+ choices = [...stateVariables["/_choiceinput1"].stateValues.choiceTexts];
52
+ choiceOrder = [
53
+ ...stateVariables["/_choiceinput1"].stateValues.choiceOrder,
54
+ ];
55
+ expect(choices.length).eq(4);
56
+ expect(originalChoices.includes(choices[0])).eq(true);
57
+ expect(originalChoices.includes(choices[1])).eq(true);
58
+ expect(originalChoices.includes(choices[2])).eq(true);
59
+ expect(originalChoices.includes(choices[3])).eq(true);
60
+ expect(choices[1]).not.eq(choices[0]);
61
+ expect(choices[2]).not.eq(choices[0]);
62
+ expect(choices[2]).not.eq(choices[1]);
63
+ expect(choices[3]).not.eq(choices[0]);
64
+ expect(choices[3]).not.eq(choices[1]);
65
+ expect(choices[3]).not.eq(choices[2]);
66
+ expect(stateVariables["/_choiceinput1"].stateValues.selectedValues).eqls(
67
+ [],
68
+ );
69
+ expect(stateVariables["/_choiceinput1"].stateValues.selectedIndices).eqls(
70
+ [],
71
+ );
72
+ expect(stateVariables["/_choiceinput1"].stateValues.inline).eq(false);
73
+ expect(stateVariables["/_choiceinput1"].stateValues.shuffleOrder).eq(
74
+ true,
75
+ );
76
+ expect(stateVariables["/_choice1"].stateValues.selected).eq(false);
77
+ expect(stateVariables["/_choice2"].stateValues.selected).eq(false);
78
+ expect(stateVariables["/_choice3"].stateValues.selected).eq(false);
79
+ expect(stateVariables["/_choice4"].stateValues.selected).eq(false);
80
+ });
81
+
82
+ cy.log("select options in order");
83
+
84
+ for (let i = 0; i < 4; i++) {
85
+ cy.get(cesc(`#\\/_choiceinput1_choice${i + 1}_input`))
86
+ .click()
87
+ .then(() => {
88
+ // make this asynchronous so that choices is populated before line is executed
89
+ cy.get(cesc("#\\/_p1")).should(
90
+ "have.text",
91
+ "Selected value: " + choices[i],
92
+ );
93
+ cy.get(cesc("#\\/_p2")).should(
94
+ "have.text",
95
+ "Selected index: " + (i + 1),
96
+ );
97
+
98
+ cy.get(cesc("#\\/pCat")).should(
99
+ "have.text",
100
+ `Selected cat: ${choiceOrder[i] === 1}`,
101
+ );
102
+ cy.get(cesc("#\\/pDog")).should(
103
+ "have.text",
104
+ `Selected dog: ${choiceOrder[i] === 2}`,
105
+ );
106
+ cy.get(cesc("#\\/pMonkey")).should(
107
+ "have.text",
108
+ `Selected monkey: ${choiceOrder[i] === 3}`,
109
+ );
110
+ cy.get(cesc("#\\/pMouse")).should(
111
+ "have.text",
112
+ `Selected mouse: ${choiceOrder[i] === 4}`,
113
+ );
114
+ });
115
+
116
+ cy.window().then(async (win) => {
117
+ let stateVariables = await win.returnAllStateVariables1();
118
+ expect(
119
+ stateVariables["/_choiceinput1"].stateValues.selectedValues,
120
+ ).eqls([choices[i]]);
121
+ expect(
122
+ stateVariables["/_choiceinput1"].stateValues.selectedIndices,
123
+ ).eqls([i + 1]);
124
+ expect(stateVariables["/_choice1"].stateValues.selected).eq(
125
+ choiceOrder[i] === 1,
126
+ );
127
+ expect(stateVariables["/_choice2"].stateValues.selected).eq(
128
+ choiceOrder[i] === 2,
129
+ );
130
+ expect(stateVariables["/_choice3"].stateValues.selected).eq(
131
+ choiceOrder[i] === 3,
132
+ );
133
+ expect(stateVariables["/_choice4"].stateValues.selected).eq(
134
+ choiceOrder[i] === 4,
135
+ );
136
+ });
137
+ }
138
+ });
139
+
140
+ it("inline choiceinput, shuffleOrder", () => {
141
+ cy.window().then(async (win) => {
142
+ win.postMessage(
143
+ {
144
+ doenetML: `
145
+ <text>a</text>
146
+ <choiceinput inline shuffleOrder>
147
+ <choice>cat</choice>
148
+ <choice>dog</choice>
149
+ <choice>monkey</choice>
150
+ <choice>mouse</choice>
151
+ </choiceinput>
152
+
153
+ <p>Selected value: <copy prop='selectedvalue' target="_choiceinput1" /></p>
154
+ <p>Selected index: <copy prop='selectedindex' target="_choiceinput1" /></p>
155
+
156
+ <p name="pCat">Selected cat: $_choice1.selected</p>
157
+ <p name="pDog">Selected dog: $_choice2.selected</p>
158
+ <p name="pMonkey">Selected monkey: $_choice3.selected</p>
159
+ <p name="pMouse">Selected mouse: $_choice4.selected</p>
160
+ `,
161
+ requestedVariantIndex: 8,
162
+ },
163
+ "*",
164
+ );
165
+ });
166
+
167
+ cy.get(cesc("#\\/_text1")).should("have.text", "a"); // to wait for page to load
168
+
169
+ let originalChoices = ["cat", "dog", "monkey", "mouse"];
170
+ cy.get(cesc("#\\/_p1")).should("have.text", "Selected value: ");
171
+ cy.get(cesc("#\\/_p2")).should("have.text", "Selected index: ");
172
+
173
+ cy.get(cesc("#\\/pCat")).should("have.text", "Selected cat: false");
174
+ cy.get(cesc("#\\/pDog")).should("have.text", "Selected dog: false");
175
+ cy.get(cesc("#\\/pMonkey")).should("have.text", "Selected monkey: false");
176
+ cy.get(cesc("#\\/pMouse")).should("have.text", "Selected mouse: false");
177
+
178
+ cy.get(cesc("#\\/_choiceinput1")).should("have.value", "");
179
+
180
+ let choices, choiceOrder;
181
+ cy.window().then(async (win) => {
182
+ let stateVariables = await win.returnAllStateVariables1();
183
+ choices = [...stateVariables["/_choiceinput1"].stateValues.choiceTexts];
184
+ choiceOrder = [
185
+ ...stateVariables["/_choiceinput1"].stateValues.choiceOrder,
186
+ ];
187
+ expect(choices.length).eq(4);
188
+ expect(originalChoices.includes(choices[0])).eq(true);
189
+ expect(originalChoices.includes(choices[1])).eq(true);
190
+ expect(originalChoices.includes(choices[2])).eq(true);
191
+ expect(originalChoices.includes(choices[3])).eq(true);
192
+ expect(choices[1]).not.eq(choices[0]);
193
+ expect(choices[2]).not.eq(choices[0]);
194
+ expect(choices[2]).not.eq(choices[1]);
195
+ expect(choices[3]).not.eq(choices[0]);
196
+ expect(choices[3]).not.eq(choices[1]);
197
+ expect(choices[3]).not.eq(choices[2]);
198
+ expect(stateVariables["/_choiceinput1"].stateValues.selectedValues).eqls(
199
+ [],
200
+ );
201
+ expect(stateVariables["/_choiceinput1"].stateValues.selectedIndices).eqls(
202
+ [],
203
+ );
204
+ expect(stateVariables["/_choiceinput1"].stateValues.inline).eq(true);
205
+ expect(stateVariables["/_choiceinput1"].stateValues.shuffleOrder).eq(
206
+ true,
207
+ );
208
+ expect(stateVariables["/_choice1"].stateValues.selected).eq(false);
209
+ expect(stateVariables["/_choice2"].stateValues.selected).eq(false);
210
+ expect(stateVariables["/_choice3"].stateValues.selected).eq(false);
211
+ expect(stateVariables["/_choice4"].stateValues.selected).eq(false);
212
+ });
213
+
214
+ cy.log("select options in order");
215
+
216
+ for (let i = 0; i < 4; i++) {
217
+ cy.get(cesc(`#\\/_choiceinput1`)).select(`${i + 1}`);
218
+ cy.get(cesc("#\\/_choiceinput1"))
219
+ .should("have.value", i + 1)
220
+ .then(() => {
221
+ // make this asynchronous so that choices is populated before line is executed
222
+ cy.get(cesc("#\\/_p1")).should(
223
+ "have.text",
224
+ "Selected value: " + choices[i],
225
+ );
226
+ cy.get(cesc("#\\/_p2")).should(
227
+ "have.text",
228
+ "Selected index: " + (i + 1),
229
+ );
230
+
231
+ cy.get(cesc("#\\/pCat")).should(
232
+ "have.text",
233
+ `Selected cat: ${choiceOrder[i] === 1}`,
234
+ );
235
+ cy.get(cesc("#\\/pDog")).should(
236
+ "have.text",
237
+ `Selected dog: ${choiceOrder[i] === 2}`,
238
+ );
239
+ cy.get(cesc("#\\/pMonkey")).should(
240
+ "have.text",
241
+ `Selected monkey: ${choiceOrder[i] === 3}`,
242
+ );
243
+ cy.get(cesc("#\\/pMouse")).should(
244
+ "have.text",
245
+ `Selected mouse: ${choiceOrder[i] === 4}`,
246
+ );
247
+ });
248
+
249
+ cy.window().then(async (win) => {
250
+ let stateVariables = await win.returnAllStateVariables1();
251
+ expect(
252
+ stateVariables["/_choiceinput1"].stateValues.selectedValues,
253
+ ).eqls([choices[i]]);
254
+ expect(
255
+ stateVariables["/_choiceinput1"].stateValues.selectedIndices,
256
+ ).eqls([i + 1]);
257
+ expect(stateVariables["/_choice1"].stateValues.selected).eq(
258
+ choiceOrder[i] === 1,
259
+ );
260
+ expect(stateVariables["/_choice2"].stateValues.selected).eq(
261
+ choiceOrder[i] === 2,
262
+ );
263
+ expect(stateVariables["/_choice3"].stateValues.selected).eq(
264
+ choiceOrder[i] === 3,
265
+ );
266
+ expect(stateVariables["/_choice4"].stateValues.selected).eq(
267
+ choiceOrder[i] === 4,
268
+ );
269
+ });
270
+ }
271
+ });
272
+
273
+ it("fixed order", () => {
274
+ cy.window().then(async (win) => {
275
+ win.postMessage(
276
+ {
277
+ doenetML: `
278
+ <text>a</text>
279
+ <choiceinput>
280
+ <choice>cat</choice>
281
+ <choice>dog</choice>
282
+ <choice>monkey</choice>
283
+ <choice>mouse</choice>
284
+ </choiceinput>
285
+
286
+ <p>Selected value: <copy prop='selectedvalue' target="_choiceinput1" /></p>
287
+ <p>Selected index: <copy prop='selectedindex' target="_choiceinput1" /></p>
288
+ `,
289
+ },
290
+ "*",
291
+ );
292
+ });
293
+
294
+ cy.get(cesc("#\\/_text1")).should("have.text", "a"); // to wait for page to load
295
+
296
+ let originalChoices = ["cat", "dog", "monkey", "mouse"];
297
+ cy.get(cesc("#\\/_p1")).should("have.text", "Selected value: ");
298
+ cy.get(cesc("#\\/_p2")).should("have.text", "Selected index: ");
299
+
300
+ let choices;
301
+ cy.window().then(async (win) => {
302
+ let stateVariables = await win.returnAllStateVariables1();
303
+ choices = [...stateVariables["/_choiceinput1"].stateValues.choiceTexts];
304
+ expect(choices).eqls(originalChoices);
305
+ expect(stateVariables["/_choiceinput1"].stateValues.selectedValues).eqls(
306
+ [],
307
+ );
308
+ expect(stateVariables["/_choiceinput1"].stateValues.selectedIndices).eqls(
309
+ [],
310
+ );
311
+ // expect(stateVariables['/_choiceinput1'].stateValues.selectedoriginalindices).eqls([])
312
+ expect(stateVariables["/_choiceinput1"].stateValues.inline).eq(false);
313
+ expect(stateVariables["/_choiceinput1"].stateValues.shuffleOrder).eq(
314
+ false,
315
+ );
316
+ expect(stateVariables["/_choice1"].stateValues.selected).eq(false);
317
+ expect(stateVariables["/_choice2"].stateValues.selected).eq(false);
318
+ expect(stateVariables["/_choice3"].stateValues.selected).eq(false);
319
+ expect(stateVariables["/_choice4"].stateValues.selected).eq(false);
320
+ });
321
+
322
+ cy.log("select options in order");
323
+
324
+ for (let i = 0; i < 4; i++) {
325
+ cy.get(cesc(`#\\/_choiceinput1_choice${i + 1}_input`))
326
+ .click()
327
+ .then(() => {
328
+ // make this asynchronous so that choices is populated before line is executed
329
+ cy.get(cesc("#\\/_p1")).should(
330
+ "have.text",
331
+ "Selected value: " + choices[i],
332
+ );
333
+ cy.get(cesc("#\\/_p2")).should(
334
+ "have.text",
335
+ "Selected index: " + (i + 1),
336
+ );
337
+ });
338
+
339
+ cy.window().then(async (win) => {
340
+ let stateVariables = await win.returnAllStateVariables1();
341
+ expect(
342
+ stateVariables["/_choiceinput1"].stateValues.selectedValues,
343
+ ).eqls([choices[i]]);
344
+ expect(
345
+ stateVariables["/_choiceinput1"].stateValues.selectedIndices,
346
+ ).eqls([i + 1]);
347
+ // expect(stateVariables['/_choiceinput1'].stateValues.selectedoriginalindices).eqls([i + 1])
348
+ expect(stateVariables["/_choice1"].stateValues.selected).eq(i === 0);
349
+ expect(stateVariables["/_choice2"].stateValues.selected).eq(i === 1);
350
+ expect(stateVariables["/_choice3"].stateValues.selected).eq(i === 2);
351
+ expect(stateVariables["/_choice4"].stateValues.selected).eq(i === 3);
352
+ });
353
+ }
354
+ });
355
+
356
+ it("choiceinput references", () => {
357
+ cy.window().then(async (win) => {
358
+ win.postMessage(
359
+ {
360
+ doenetML: `
361
+ <text>a</text>
362
+ <choiceinput inline shuffleOrder name="ci1">
363
+ <choice>a</choice>
364
+ <choice>b</choice>
365
+ <choice>c</choice>
366
+ <choice>d</choice>
367
+ <choice>e</choice>
368
+ <choice>f</choice>
369
+ </choiceinput>
370
+ <copy name="copy" target="ci1" assignNames="ci2" createComponentOfType="choiceinput" />
371
+ <copy name="copy2" inline="false" target="ci1" assignNames="ci3" createComponentOfType="choiceinput" />
372
+ <copy name="copy3" inline="false" target="copy" assignNames="ci4" createComponentOfType="choiceinput" />
373
+
374
+ <p>Selected values: <aslist>
375
+ <copy prop='selectedvalue' target="ci1" />
376
+ <copy prop='selectedvalue' target="copy" />
377
+ <copy prop='selectedvalue' target="copy2" />
378
+ <copy prop='selectedvalue' target="copy3" />
379
+ </aslist></p>
380
+ <p>Selected indices: <aslist>
381
+ <copy prop='selectedindex' target="ci1" />
382
+ <copy prop='selectedindex' target="copy" />
383
+ <copy prop='selectedindex' target="copy2" />
384
+ <copy prop='selectedindex' target="copy3" />
385
+ </aslist></p>
386
+
387
+ `,
388
+ },
389
+ "*",
390
+ );
391
+ });
392
+
393
+ cy.get(cesc("#\\/_text1")).should("have.text", "a"); // to wait for page to load
394
+
395
+ let originalChoices = ["a", "b", "c", "d", "e", "f"];
396
+ cy.get(cesc("#\\/_p1")).should("have.text", "Selected values: ");
397
+ cy.get(cesc("#\\/_p2")).should("have.text", "Selected indices: ");
398
+
399
+ let choices;
400
+ cy.window().then(async (win) => {
401
+ let stateVariables = await win.returnAllStateVariables1();
402
+ choices = [...stateVariables["/ci1"].stateValues.choiceTexts];
403
+ expect(choices.length).eq(6);
404
+ expect(originalChoices.includes(choices[0])).eq(true);
405
+ expect(originalChoices.includes(choices[1])).eq(true);
406
+ expect(originalChoices.includes(choices[2])).eq(true);
407
+ expect(originalChoices.includes(choices[3])).eq(true);
408
+ expect(originalChoices.includes(choices[4])).eq(true);
409
+ expect(originalChoices.includes(choices[5])).eq(true);
410
+ expect(choices[1]).not.eq(choices[0]);
411
+ expect(choices[2]).not.eq(choices[0]);
412
+ expect(choices[2]).not.eq(choices[1]);
413
+ expect(choices[3]).not.eq(choices[0]);
414
+ expect(choices[3]).not.eq(choices[1]);
415
+ expect(choices[3]).not.eq(choices[2]);
416
+ expect(choices[4]).not.eq(choices[0]);
417
+ expect(choices[4]).not.eq(choices[1]);
418
+ expect(choices[4]).not.eq(choices[2]);
419
+ expect(choices[4]).not.eq(choices[3]);
420
+ expect(choices[5]).not.eq(choices[0]);
421
+ expect(choices[5]).not.eq(choices[1]);
422
+ expect(choices[5]).not.eq(choices[2]);
423
+ expect(choices[5]).not.eq(choices[3]);
424
+ expect(choices[5]).not.eq(choices[4]);
425
+ expect(stateVariables["/ci2"].stateValues.choiceTexts).eqls(choices);
426
+ expect(stateVariables["/ci3"].stateValues.choiceTexts).eqls(choices);
427
+ expect(stateVariables["/ci4"].stateValues.choiceTexts).eqls(choices);
428
+
429
+ expect(stateVariables["/ci1"].stateValues.selectedValues).eqls([]);
430
+ expect(stateVariables["/ci1"].stateValues.selectedIndices).eqls([]);
431
+ expect(stateVariables["/ci2"].stateValues.selectedValues).eqls([]);
432
+ expect(stateVariables["/ci2"].stateValues.selectedIndices).eqls([]);
433
+ expect(stateVariables["/ci3"].stateValues.selectedValues).eqls([]);
434
+ expect(stateVariables["/ci3"].stateValues.selectedIndices).eqls([]);
435
+ expect(stateVariables["/ci4"].stateValues.selectedValues).eqls([]);
436
+ expect(stateVariables["/ci4"].stateValues.selectedIndices).eqls([]);
437
+
438
+ expect(stateVariables["/ci1"].stateValues.inline).eq(true);
439
+ expect(stateVariables["/ci1"].stateValues.shuffleOrder).eq(true);
440
+ expect(stateVariables["/ci2"].stateValues.inline).eq(true);
441
+ expect(stateVariables["/ci2"].stateValues.shuffleOrder).eq(true);
442
+ expect(stateVariables["/ci3"].stateValues.inline).eq(false);
443
+ expect(stateVariables["/ci3"].stateValues.shuffleOrder).eq(true);
444
+ expect(stateVariables["/ci4"].stateValues.inline).eq(false);
445
+ expect(stateVariables["/ci4"].stateValues.shuffleOrder).eq(true);
446
+ });
447
+
448
+ cy.log("select options in order from first input");
449
+ for (let i = 0; i < 6; i++) {
450
+ cy.get(cesc(`#\\/ci1`))
451
+ .select(`${i + 1}`)
452
+ .then(() => {
453
+ // make this asynchronous so that choices is populated before line is executed
454
+ cy.get(cesc("#\\/_p1")).should(
455
+ "have.text",
456
+ `Selected values: ${choices[i]}, ${choices[i]}, ${choices[i]}, ${choices[i]}`,
457
+ );
458
+ cy.get(cesc("#\\/_p2")).should(
459
+ "have.text",
460
+ `Selected indices: ${i + 1}, ${i + 1}, ${i + 1}, ${i + 1}`,
461
+ );
462
+
463
+ cy.window().then(async (win) => {
464
+ let stateVariables = await win.returnAllStateVariables1();
465
+ expect(stateVariables["/ci1"].stateValues.selectedValues).eqls([
466
+ choices[i],
467
+ ]);
468
+ expect(stateVariables["/ci1"].stateValues.selectedIndices).eqls([
469
+ i + 1,
470
+ ]);
471
+ expect(stateVariables["/ci2"].stateValues.selectedValues).eqls([
472
+ choices[i],
473
+ ]);
474
+ expect(stateVariables["/ci2"].stateValues.selectedIndices).eqls([
475
+ i + 1,
476
+ ]);
477
+ expect(stateVariables["/ci3"].stateValues.selectedValues).eqls([
478
+ choices[i],
479
+ ]);
480
+ expect(stateVariables["/ci3"].stateValues.selectedIndices).eqls([
481
+ i + 1,
482
+ ]);
483
+ expect(stateVariables["/ci4"].stateValues.selectedValues).eqls([
484
+ choices[i],
485
+ ]);
486
+ expect(stateVariables["/ci4"].stateValues.selectedIndices).eqls([
487
+ i + 1,
488
+ ]);
489
+ });
490
+ });
491
+ }
492
+
493
+ cy.log("select options in order from second input");
494
+ for (let i = 0; i < 6; i++) {
495
+ cy.get(cesc(`#\\/ci2`))
496
+ .select(`${i + 1}`)
497
+ .then(() => {
498
+ // make this asynchronous so that choices is populated before line is executed
499
+ cy.get(cesc("#\\/_p1")).should(
500
+ "have.text",
501
+ `Selected values: ${choices[i]}, ${choices[i]}, ${choices[i]}, ${choices[i]}`,
502
+ );
503
+ cy.get(cesc("#\\/_p2")).should(
504
+ "have.text",
505
+ `Selected indices: ${i + 1}, ${i + 1}, ${i + 1}, ${i + 1}`,
506
+ );
507
+
508
+ cy.window().then(async (win) => {
509
+ let stateVariables = await win.returnAllStateVariables1();
510
+ expect(stateVariables["/ci1"].stateValues.selectedValues).eqls([
511
+ choices[i],
512
+ ]);
513
+ expect(stateVariables["/ci1"].stateValues.selectedIndices).eqls([
514
+ i + 1,
515
+ ]);
516
+ expect(stateVariables["/ci2"].stateValues.selectedValues).eqls([
517
+ choices[i],
518
+ ]);
519
+ expect(stateVariables["/ci2"].stateValues.selectedIndices).eqls([
520
+ i + 1,
521
+ ]);
522
+ expect(stateVariables["/ci3"].stateValues.selectedValues).eqls([
523
+ choices[i],
524
+ ]);
525
+ expect(stateVariables["/ci3"].stateValues.selectedIndices).eqls([
526
+ i + 1,
527
+ ]);
528
+ expect(stateVariables["/ci4"].stateValues.selectedValues).eqls([
529
+ choices[i],
530
+ ]);
531
+ expect(stateVariables["/ci4"].stateValues.selectedIndices).eqls([
532
+ i + 1,
533
+ ]);
534
+ });
535
+ });
536
+ }
537
+
538
+ cy.log("select options in order from third input");
539
+ for (let i = 0; i < 6; i++) {
540
+ cy.get(`${cesc(`#\\/ci3`)}_choice${i + 1}_input`)
541
+ .click()
542
+ .then(() => {
543
+ // make this asynchronous so that choices is populated before line is executed
544
+ cy.get(cesc("#\\/_p1")).should(
545
+ "have.text",
546
+ `Selected values: ${choices[i]}, ${choices[i]}, ${choices[i]}, ${choices[i]}`,
547
+ );
548
+ cy.get(cesc("#\\/_p2")).should(
549
+ "have.text",
550
+ `Selected indices: ${i + 1}, ${i + 1}, ${i + 1}, ${i + 1}`,
551
+ );
552
+
553
+ cy.window().then(async (win) => {
554
+ let stateVariables = await win.returnAllStateVariables1();
555
+ expect(stateVariables["/ci1"].stateValues.selectedValues).eqls([
556
+ choices[i],
557
+ ]);
558
+ expect(stateVariables["/ci1"].stateValues.selectedIndices).eqls([
559
+ i + 1,
560
+ ]);
561
+ expect(stateVariables["/ci2"].stateValues.selectedValues).eqls([
562
+ choices[i],
563
+ ]);
564
+ expect(stateVariables["/ci2"].stateValues.selectedIndices).eqls([
565
+ i + 1,
566
+ ]);
567
+ expect(stateVariables["/ci3"].stateValues.selectedValues).eqls([
568
+ choices[i],
569
+ ]);
570
+ expect(stateVariables["/ci3"].stateValues.selectedIndices).eqls([
571
+ i + 1,
572
+ ]);
573
+ expect(stateVariables["/ci4"].stateValues.selectedValues).eqls([
574
+ choices[i],
575
+ ]);
576
+ expect(stateVariables["/ci4"].stateValues.selectedIndices).eqls([
577
+ i + 1,
578
+ ]);
579
+ });
580
+ });
581
+ }
582
+
583
+ cy.log("select options in order from fourth input");
584
+ for (let i = 0; i < 6; i++) {
585
+ cy.get(`${cesc(`#\\/ci4`)}_choice${i + 1}_input`)
586
+ .click()
587
+ .then(() => {
588
+ // make this asynchronous so that choices is populated before line is executed
589
+ cy.get(cesc("#\\/_p1")).should(
590
+ "have.text",
591
+ `Selected values: ${choices[i]}, ${choices[i]}, ${choices[i]}, ${choices[i]}`,
592
+ );
593
+ cy.get(cesc("#\\/_p2")).should(
594
+ "have.text",
595
+ `Selected indices: ${i + 1}, ${i + 1}, ${i + 1}, ${i + 1}`,
596
+ );
597
+
598
+ cy.window().then(async (win) => {
599
+ let stateVariables = await win.returnAllStateVariables1();
600
+ expect(stateVariables["/ci1"].stateValues.selectedValues).eqls([
601
+ choices[i],
602
+ ]);
603
+ expect(stateVariables["/ci1"].stateValues.selectedIndices).eqls([
604
+ i + 1,
605
+ ]);
606
+ expect(stateVariables["/ci2"].stateValues.selectedValues).eqls([
607
+ choices[i],
608
+ ]);
609
+ expect(stateVariables["/ci2"].stateValues.selectedIndices).eqls([
610
+ i + 1,
611
+ ]);
612
+ expect(stateVariables["/ci3"].stateValues.selectedValues).eqls([
613
+ choices[i],
614
+ ]);
615
+ expect(stateVariables["/ci3"].stateValues.selectedIndices).eqls([
616
+ i + 1,
617
+ ]);
618
+ expect(stateVariables["/ci4"].stateValues.selectedValues).eqls([
619
+ choices[i],
620
+ ]);
621
+ expect(stateVariables["/ci4"].stateValues.selectedIndices).eqls([
622
+ i + 1,
623
+ ]);
624
+ });
625
+ });
626
+ }
627
+ });
628
+
629
+ it("math inside choices", () => {
630
+ cy.window().then(async (win) => {
631
+ win.postMessage(
632
+ {
633
+ doenetML: `
634
+ <text>a</text>
635
+ <choiceinput shuffleOrder name="ci1">
636
+ <choice>The function is <m>f(\\xi)=\\sin(\\xi)</m>.</choice>
637
+ <choice>The sum of <math name="lambda2">lambda^2</math> and <math name="twice">2 lambda^2</math> is <math simplify><copy target="lambda2" />+<copy target="twice" /></math>.</choice>
638
+ <choice>The sequence is <aslist><sequence from="1" to="5" /></aslist>.</choice>
639
+ <choice>Can't convert this latex: <m>\\bar{x}^i</m>.</choice>
640
+ </choiceinput>
641
+
642
+ <copy name="copy" inline target="ci1" assignNames="ci2" createComponentOfType="choiceinput" />
643
+
644
+ <p>Selected values: <aslist>
645
+ <copy prop='selectedvalue' target="ci1" />
646
+ <copy prop='selectedvalue' target="copy" />
647
+ </aslist></p>
648
+ <p>Selected indices: <aslist>
649
+ <copy prop='selectedindex' target="ci1" />
650
+ <copy prop='selectedindex' target="copy" />
651
+ </aslist></p>
652
+
653
+ `,
654
+ },
655
+ "*",
656
+ );
657
+ });
658
+
659
+ cy.get(cesc("#\\/_text1")).should("have.text", "a"); // to wait for page to load
660
+
661
+ let originalChoices = [
662
+ "The function is f(ξ) = sin(ξ).",
663
+ "The sum of λ² and 2 λ² is 3 λ².",
664
+ "The sequence is 1, 2, 3, 4, 5.",
665
+ "Can't convert this latex: \\bar{x}^i.",
666
+ ];
667
+ cy.get(cesc("#\\/_p1")).should("have.text", "Selected values: ");
668
+ cy.get(cesc("#\\/_p2")).should("have.text", "Selected indices: ");
669
+ // cy.get(cesc('#\\/_p3')).should('have.text', 'Selected original indices: ')
670
+
671
+ let choices;
672
+ cy.window().then(async (win) => {
673
+ let stateVariables = await win.returnAllStateVariables1();
674
+ choices = [...stateVariables["/ci1"].stateValues.choiceTexts];
675
+ expect(choices.length).eq(4);
676
+ expect(originalChoices.includes(choices[0])).eq(true);
677
+ expect(originalChoices.includes(choices[1])).eq(true);
678
+ expect(originalChoices.includes(choices[2])).eq(true);
679
+ expect(originalChoices.includes(choices[3])).eq(true);
680
+ expect(choices[1]).not.eq(choices[0]);
681
+ expect(choices[2]).not.eq(choices[0]);
682
+ expect(choices[2]).not.eq(choices[1]);
683
+ expect(choices[3]).not.eq(choices[0]);
684
+ expect(choices[3]).not.eq(choices[1]);
685
+ expect(choices[3]).not.eq(choices[2]);
686
+
687
+ expect(stateVariables["/ci2"].stateValues.choiceTexts).eqls(choices);
688
+
689
+ expect(stateVariables["/ci1"].stateValues.selectedValues).eqls([]);
690
+ expect(stateVariables["/ci1"].stateValues.selectedIndices).eqls([]);
691
+ expect(stateVariables["/ci2"].stateValues.selectedValues).eqls([]);
692
+ expect(stateVariables["/ci2"].stateValues.selectedIndices).eqls([]);
693
+
694
+ expect(stateVariables["/ci1"].stateValues.inline).eq(false);
695
+ expect(stateVariables["/ci1"].stateValues.shuffleOrder).eq(true);
696
+ expect(stateVariables["/ci2"].stateValues.inline).eq(true);
697
+ expect(stateVariables["/ci2"].stateValues.shuffleOrder).eq(true);
698
+ });
699
+
700
+ cy.log("select options in order from first input");
701
+ for (let i = 0; i < 4; i++) {
702
+ cy.get(cesc(`#\\/ci1_choice${i + 1}_input`))
703
+ .click()
704
+ .then(() => {
705
+ // make this asynchronous so that choices is populated before line is executed
706
+ cy.get(cesc("#\\/_p1")).should(
707
+ "have.text",
708
+ `Selected values: ${choices[i]}, ${choices[i]}`,
709
+ );
710
+ cy.get(cesc("#\\/_p2")).should(
711
+ "have.text",
712
+ `Selected indices: ${i + 1}, ${i + 1}`,
713
+ );
714
+
715
+ cy.window().then(async (win) => {
716
+ let stateVariables = await win.returnAllStateVariables1();
717
+ expect(stateVariables["/ci1"].stateValues.selectedValues).eqls([
718
+ choices[i],
719
+ ]);
720
+ expect(stateVariables["/ci1"].stateValues.selectedIndices).eqls([
721
+ i + 1,
722
+ ]);
723
+ expect(stateVariables["/ci2"].stateValues.selectedValues).eqls([
724
+ choices[i],
725
+ ]);
726
+ expect(stateVariables["/ci2"].stateValues.selectedIndices).eqls([
727
+ i + 1,
728
+ ]);
729
+ });
730
+ });
731
+ }
732
+
733
+ cy.log("select options in order from second input");
734
+ for (let i = 0; i < 4; i++) {
735
+ cy.get(cesc(`#\\/ci2`))
736
+ .select(`${i + 1}`)
737
+ .then(() => {
738
+ // make this asynchronous so that choices is populated before line is executed
739
+ cy.get(cesc("#\\/_p1")).should(
740
+ "have.text",
741
+ `Selected values: ${choices[i]}, ${choices[i]}`,
742
+ );
743
+ cy.get(cesc("#\\/_p2")).should(
744
+ "have.text",
745
+ `Selected indices: ${i + 1}, ${i + 1}`,
746
+ );
747
+
748
+ cy.window().then(async (win) => {
749
+ let stateVariables = await win.returnAllStateVariables1();
750
+ expect(stateVariables["/ci1"].stateValues.selectedValues).eqls([
751
+ choices[i],
752
+ ]);
753
+ expect(stateVariables["/ci1"].stateValues.selectedIndices).eqls([
754
+ i + 1,
755
+ ]);
756
+ expect(stateVariables["/ci2"].stateValues.selectedValues).eqls([
757
+ choices[i],
758
+ ]);
759
+ expect(stateVariables["/ci2"].stateValues.selectedIndices).eqls([
760
+ i + 1,
761
+ ]);
762
+ });
763
+ });
764
+ }
765
+ });
766
+
767
+ it("bind value to textinput", () => {
768
+ cy.window().then(async (win) => {
769
+ win.postMessage(
770
+ {
771
+ doenetML: `
772
+ <text>a</text>
773
+ <choiceinput bindValueTo="$_textinput1" shuffleOrder name="ci1">
774
+ <choice>caT</choice>
775
+ <choice> dog </choice>
776
+ <choice><text>Monkey</text></choice>
777
+ </choiceinput>
778
+
779
+ <p>Select by typing: <textinput prefill="monkey" /></p>
780
+
781
+ <copy name="copy" inline target="ci1" assignNames="ci2" createComponentOfType="choiceinput" />
782
+
783
+ <p>Selected values: <aslist>
784
+ <copy prop='selectedvalue' target="ci1" />
785
+ <copy prop='selectedvalue' target="copy" />
786
+ </aslist></p>
787
+ <p>Selected indices: <aslist>
788
+ <copy prop='selectedindex' target="ci1" />
789
+ <copy prop='selectedindex' target="copy" />
790
+ </aslist></p>
791
+
792
+ `,
793
+ },
794
+ "*",
795
+ );
796
+ });
797
+
798
+ cy.get(cesc("#\\/_text1")).should("have.text", "a"); // to wait for page to load
799
+
800
+ let originalChoices = ["caT", " dog ", "Monkey"];
801
+
802
+ cy.window().then(async (win) => {
803
+ let stateVariables = await win.returnAllStateVariables1();
804
+
805
+ let choiceOrder = stateVariables["/ci1"].stateValues.choiceOrder;
806
+ let choices = choiceOrder.map((x) => originalChoices[x - 1]);
807
+
808
+ let checkChoices = function (selectedChoice, inputText) {
809
+ let choiceArray, indexArray;
810
+ if (selectedChoice === null) {
811
+ choiceArray = indexArray = [];
812
+
813
+ for (let i = 1; i <= 3; i++) {
814
+ cy.get(cesc(`#\\/ci1_choice${i}_input`)).should("not.be.checked");
815
+ }
816
+ cy.get(cesc(`#\\/ci2`)).should("have.value", "");
817
+
818
+ cy.get(cesc("#\\/_p2")).should("have.text", `Selected values: `);
819
+ cy.get(cesc("#\\/_p3")).should("have.text", `Selected indices: `);
820
+ } else {
821
+ let selectedIndex = choices.indexOf(selectedChoice) + 1;
822
+ choiceArray = [selectedChoice];
823
+ indexArray = [selectedIndex];
824
+
825
+ for (let i = 1; i <= 3; i++) {
826
+ if (i === selectedIndex) {
827
+ cy.get(cesc(`#\\/ci1_choice${i}_input`)).should("be.checked");
828
+ } else {
829
+ cy.get(cesc(`#\\/ci1_choice${i}_input`)).should("not.be.checked");
830
+ }
831
+ }
832
+ cy.get(cesc(`#\\/ci2`)).should("have.value", String(selectedIndex));
833
+ cy.get(cesc("#\\/_p2")).should(
834
+ "have.text",
835
+ `Selected values: ${selectedChoice}, ${selectedChoice}`,
836
+ );
837
+ cy.get(cesc("#\\/_p3")).should(
838
+ "have.text",
839
+ `Selected indices: ${selectedIndex}, ${selectedIndex}`,
840
+ );
841
+ }
842
+
843
+ cy.get(cesc("#\\/_textinput1_input")).should("have.value", inputText);
844
+
845
+ cy.window().then(async (win) => {
846
+ let stateVariables = await win.returnAllStateVariables1();
847
+ expect(stateVariables["/ci1"].stateValues.selectedValues).eqls(
848
+ choiceArray,
849
+ );
850
+ expect(stateVariables["/ci1"].stateValues.selectedIndices).eqls(
851
+ indexArray,
852
+ );
853
+ expect(stateVariables["/ci2"].stateValues.selectedValues).eqls(
854
+ choiceArray,
855
+ );
856
+ expect(stateVariables["/ci2"].stateValues.selectedIndices).eqls(
857
+ indexArray,
858
+ );
859
+ });
860
+ };
861
+
862
+ checkChoices("Monkey", "monkey");
863
+
864
+ cy.log("select cat from first input");
865
+ let selectedChoice = "caT";
866
+ let selectedIndex = choices.indexOf(selectedChoice) + 1;
867
+ let inputText = selectedChoice;
868
+ cy.get(cesc(`#\\/ci1_choice${selectedIndex}_input`)).click();
869
+ checkChoices(selectedChoice, inputText);
870
+
871
+ cy.log("Type Dog");
872
+ selectedChoice = " dog ";
873
+ selectedIndex = choices.indexOf(selectedChoice) + 1;
874
+ inputText = "Dog";
875
+ cy.get(cesc("#\\/_textinput1_input")).clear().type(`${inputText}{enter}`);
876
+ checkChoices(selectedChoice, inputText);
877
+
878
+ cy.log("select monkey from second input");
879
+ selectedChoice = "Monkey";
880
+ selectedIndex = choices.indexOf(selectedChoice) + 1;
881
+ inputText = selectedChoice;
882
+ cy.get(cesc(`#\\/ci2`)).select(`${selectedIndex}`);
883
+ checkChoices(selectedChoice, inputText);
884
+
885
+ cy.log("type no cat");
886
+ selectedChoice = null;
887
+ inputText = "no cat";
888
+ cy.get(cesc("#\\/_textinput1_input")).clear().type(`${inputText}{enter}`);
889
+ checkChoices(selectedChoice, inputText);
890
+
891
+ cy.log("select cat from second input");
892
+ selectedChoice = "caT";
893
+ selectedIndex = choices.indexOf(selectedChoice) + 1;
894
+ inputText = selectedChoice;
895
+ cy.get(cesc(`#\\/ci2`)).select(`${selectedIndex}`);
896
+ checkChoices(selectedChoice, inputText);
897
+
898
+ cy.log("type no dog");
899
+ selectedChoice = null;
900
+ inputText = "no dog";
901
+ cy.get(cesc("#\\/_textinput1_input")).clear().type(`${inputText}{enter}`);
902
+ checkChoices(selectedChoice, inputText);
903
+
904
+ cy.log("select dog from first input");
905
+ selectedChoice = " dog ";
906
+ selectedIndex = choices.indexOf(selectedChoice) + 1;
907
+ inputText = selectedChoice;
908
+ cy.get(cesc(`#\\/ci1_choice${selectedIndex}_input`)).click();
909
+ checkChoices(selectedChoice, inputText);
910
+
911
+ cy.log("type no monkey");
912
+ selectedChoice = null;
913
+ inputText = "no monkey";
914
+ cy.get(cesc("#\\/_textinput1_input")).clear().type(`${inputText}{enter}`);
915
+ checkChoices(selectedChoice, inputText);
916
+
917
+ cy.log("type monKey ");
918
+ selectedChoice = "Monkey";
919
+ inputText = " monKey ";
920
+ cy.get(cesc("#\\/_textinput1_input")).clear().type(`${inputText}{enter}`);
921
+ checkChoices(selectedChoice, inputText);
922
+ });
923
+ });
924
+
925
+ it("bind value to textinput, select multiple", () => {
926
+ cy.window().then(async (win) => {
927
+ win.postMessage(
928
+ {
929
+ doenetML: `
930
+ <text>a</text>
931
+ <choiceinput bindValueTo="$_textinput1" shuffleOrder selectMultiple name="ci1">
932
+ <choice><text>caT</text></choice>
933
+ <choice> dog </choice>
934
+ <choice>Monkey</choice>
935
+ </choiceinput>
936
+
937
+ <p>Select by typing: <textinput prefill="monkey" /></p>
938
+
939
+ <copy name="copy" inline target="ci1" assignNames="ci2" createComponentOfType="choiceinput" />
940
+
941
+ <p>Selected values: <aslist>
942
+ <copy prop='selectedvalues' target="ci1" />
943
+ <copy prop='selectedvalues' target="copy" />
944
+ </aslist></p>
945
+ <p>Selected indices: <aslist>
946
+ <copy prop='selectedindices' target="ci1" />
947
+ <copy prop='selectedindices' target="copy" />
948
+ </aslist></p>
949
+
950
+ `,
951
+ },
952
+ "*",
953
+ );
954
+ });
955
+
956
+ cy.get(cesc("#\\/_text1")).should("have.text", "a"); // to wait for page to load
957
+
958
+ let originalChoices = ["caT", " dog ", "Monkey"];
959
+
960
+ cy.window().then(async (win) => {
961
+ let stateVariables = await win.returnAllStateVariables1();
962
+
963
+ let choiceOrder = stateVariables["/ci1"].stateValues.choiceOrder;
964
+ let choices = choiceOrder.map((x) => originalChoices[x - 1]);
965
+
966
+ let checkChoices = function (selectedChoices, inputText) {
967
+ selectedChoices.sort((a, b) => choices.indexOf(a) - choices.indexOf(b));
968
+
969
+ let selectedIndices = selectedChoices.map(
970
+ (x) => choices.indexOf(x) + 1,
971
+ );
972
+
973
+ for (let i = 1; i <= 3; i++) {
974
+ if (selectedIndices.includes(i)) {
975
+ cy.get(cesc(`#\\/ci1_choice${i}_input`)).should("be.checked");
976
+ } else {
977
+ cy.get(cesc(`#\\/ci1_choice${i}_input`)).should("not.be.checked");
978
+ }
979
+ }
980
+ cy.get(cesc(`#\\/ci2`))
981
+ .invoke("val")
982
+ .should(
983
+ "deep.equal",
984
+ selectedIndices.map((x) => String(x)),
985
+ );
986
+ let selectedChoicesString = [
987
+ ...selectedChoices,
988
+ ...selectedChoices,
989
+ ].join(", ");
990
+ let selectedIndicesString = [
991
+ ...selectedIndices,
992
+ ...selectedIndices,
993
+ ].join(", ");
994
+ cy.get(cesc("#\\/_p2")).should(
995
+ "have.text",
996
+ `Selected values: ${selectedChoicesString}`,
997
+ );
998
+ cy.get(cesc("#\\/_p3")).should(
999
+ "have.text",
1000
+ `Selected indices: ${selectedIndicesString}`,
1001
+ );
1002
+
1003
+ cy.get(cesc("#\\/_textinput1_input")).should("have.value", inputText);
1004
+
1005
+ cy.window().then(async (win) => {
1006
+ let stateVariables = await win.returnAllStateVariables1();
1007
+ expect(stateVariables["/ci1"].stateValues.selectedValues).eqls(
1008
+ selectedChoices,
1009
+ );
1010
+ expect(stateVariables["/ci1"].stateValues.selectedIndices).eqls(
1011
+ selectedIndices,
1012
+ );
1013
+ expect(stateVariables["/ci2"].stateValues.selectedValues).eqls(
1014
+ selectedChoices,
1015
+ );
1016
+ expect(stateVariables["/ci2"].stateValues.selectedIndices).eqls(
1017
+ selectedIndices,
1018
+ );
1019
+ });
1020
+ };
1021
+
1022
+ checkChoices(["Monkey"], "monkey");
1023
+
1024
+ cy.log("select cat from first input");
1025
+ let selectedChoices = ["caT", "Monkey"];
1026
+ let selectedIndex = choices.indexOf(selectedChoices[0]) + 1;
1027
+ selectedChoices.sort((a, b) => choices.indexOf(a) - choices.indexOf(b));
1028
+ let inputText = selectedChoices.join(", ");
1029
+ cy.get(cesc(`#\\/ci1_choice${selectedIndex}_input`)).click();
1030
+ checkChoices(selectedChoices, inputText);
1031
+
1032
+ cy.log("Type Dog");
1033
+ selectedChoices = [" dog "];
1034
+ inputText = "Dog";
1035
+ cy.get(cesc("#\\/_textinput1_input")).clear().type(`${inputText}{enter}`);
1036
+ checkChoices(selectedChoices, inputText);
1037
+
1038
+ cy.log("Type cat ,DOG");
1039
+ selectedChoices = [" dog ", "caT"];
1040
+ inputText = "cat ,DOG";
1041
+ cy.get(cesc("#\\/_textinput1_input")).clear().type(`${inputText}{enter}`);
1042
+ checkChoices(selectedChoices, inputText);
1043
+
1044
+ cy.log("select monkey, dog from second input");
1045
+ selectedChoices = [" dog ", "Monkey"];
1046
+ let selectedIndices = selectedChoices.map((x) => choices.indexOf(x) + 1);
1047
+ selectedChoices.sort((a, b) => choices.indexOf(a) - choices.indexOf(b));
1048
+ inputText = selectedChoices.join(", ");
1049
+ cy.get(cesc(`#\\/ci2`)).select(selectedIndices.map(String));
1050
+ checkChoices(selectedChoices, inputText);
1051
+
1052
+ cy.log("type no cat");
1053
+ selectedChoices = [];
1054
+ inputText = "no cat";
1055
+ cy.get(cesc("#\\/_textinput1_input")).clear().type(`${inputText}{enter}`);
1056
+ checkChoices(selectedChoices, inputText);
1057
+
1058
+ cy.log("type cat, no dog");
1059
+ selectedChoices = ["caT"];
1060
+ inputText = "cat, no dog";
1061
+ cy.get(cesc("#\\/_textinput1_input")).clear().type(`${inputText}{enter}`);
1062
+ checkChoices(selectedChoices, inputText);
1063
+
1064
+ cy.log("type dog, no monkey, CAT ");
1065
+ selectedChoices = [" dog ", "caT"];
1066
+ inputText = "dog, no monkey, CAT ";
1067
+ cy.get(cesc("#\\/_textinput1_input")).clear().type(`${inputText}{enter}`);
1068
+ checkChoices(selectedChoices, inputText);
1069
+
1070
+ cy.log("select all from second input");
1071
+ selectedChoices = ["Monkey", " dog ", "caT"];
1072
+ selectedIndices = selectedChoices.map((x) => choices.indexOf(x) + 1);
1073
+ selectedChoices.sort((a, b) => choices.indexOf(a) - choices.indexOf(b));
1074
+ inputText = selectedChoices.join(", ");
1075
+
1076
+ // for some reason, the html <select> tag is ignoring the onChange event
1077
+ // unless first select option 3 before selecting them all!
1078
+ // No idea what's going on
1079
+ cy.get(cesc("#\\/ci2")).select(["3"]);
1080
+ cy.get(cesc(`#\\/ci2`)).select(selectedIndices.map(String));
1081
+
1082
+ checkChoices(selectedChoices, inputText);
1083
+
1084
+ cy.log("type no dog at end");
1085
+ inputText += ", no dog";
1086
+ cy.get(cesc("#\\/_textinput1_input")).type(`{end}, no dog{enter}`);
1087
+ checkChoices(selectedChoices, inputText);
1088
+
1089
+ cy.log("type dog, DOG");
1090
+ selectedChoices = [" dog "];
1091
+ inputText = "dog, DOG";
1092
+ cy.get(cesc("#\\/_textinput1_input")).clear().type(`${inputText}{enter}`);
1093
+ checkChoices(selectedChoices, inputText);
1094
+
1095
+ cy.log("select cat from first input");
1096
+ selectedChoices = [" dog ", "caT"];
1097
+ selectedChoices.sort((a, b) => choices.indexOf(a) - choices.indexOf(b));
1098
+ selectedIndex = choices.indexOf(selectedChoices[0]) + 1;
1099
+ inputText = selectedChoices.join(", ");
1100
+ cy.get(cesc(`#\\/ci1_choice${selectedIndex}_input`)).click();
1101
+ checkChoices(selectedChoices, inputText);
1102
+
1103
+ cy.log("deselect dog from first input");
1104
+ selectedIndex = choices.indexOf(selectedChoices[1]) + 1;
1105
+ selectedChoices = ["caT"];
1106
+ inputText = selectedChoices.join(", ");
1107
+ cy.get(cesc(`#\\/ci1_choice${selectedIndex}_input`)).click();
1108
+ checkChoices(selectedChoices, inputText);
1109
+ });
1110
+ });
1111
+
1112
+ it("bind value to fixed text, choiceinput reverts to fixed value", () => {
1113
+ cy.window().then(async (win) => {
1114
+ win.postMessage(
1115
+ {
1116
+ doenetML: `
1117
+ <text>a</text>
1118
+ <choiceinput bindValueTo="$alwaysMonkey" name="ci1">
1119
+ <choice>cat</choice>
1120
+ <choice>dog</choice>
1121
+ <choice>monkey</choice>
1122
+ </choiceinput>
1123
+
1124
+ <p>Fixed to be: <text name="alwaysMonkey" fixed>monkey</text></p>
1125
+
1126
+ <copy name="copy" inline target="ci1" assignNames="ci2" createComponentOfType="choiceinput" />
1127
+
1128
+ <p>Selected values: <aslist>
1129
+ <copy prop='selectedvalue' target="ci1" />
1130
+ <copy prop='selectedvalue' target="copy" />
1131
+ </aslist></p>
1132
+ <p>Selected indices: <aslist>
1133
+ <copy prop='selectedindex' target="ci1" />
1134
+ <copy prop='selectedindex' target="copy" />
1135
+ </aslist></p>
1136
+
1137
+ <p>Check for core round trip: <booleaninput name="bi" /> <copy prop="value" target="bi" assignNames="b" /></p>
1138
+ `,
1139
+ },
1140
+ "*",
1141
+ );
1142
+ });
1143
+
1144
+ cy.get(cesc("#\\/_text1")).should("have.text", "a"); // to wait for page to load
1145
+
1146
+ function checkStillMonkey() {
1147
+ for (let i = 1; i <= 3; i++) {
1148
+ if (i === 3) {
1149
+ cy.get(cesc(`#\\/ci1_choice${i}_input`)).should("be.checked");
1150
+ } else {
1151
+ cy.get(cesc(`#\\/ci1_choice${i}_input`)).should("not.be.checked");
1152
+ }
1153
+ }
1154
+ cy.get(cesc(`#\\/ci2`)).should("have.value", "3");
1155
+ cy.get(cesc("#\\/_p2")).should(
1156
+ "have.text",
1157
+ `Selected values: monkey, monkey`,
1158
+ );
1159
+ cy.get(cesc("#\\/_p3")).should("have.text", `Selected indices: 3, 3`);
1160
+ }
1161
+
1162
+ checkStillMonkey();
1163
+
1164
+ cy.get(cesc(`#\\/ci1_choice1_input`)).click();
1165
+ cy.get(cesc(`#\\/bi`)).click();
1166
+ cy.get(cesc("#\\/b")).should("have.text", "true");
1167
+ checkStillMonkey();
1168
+
1169
+ cy.get(cesc(`#\\/ci1_choice2_input`)).click();
1170
+ cy.get(cesc(`#\\/bi`)).click();
1171
+ cy.get(cesc("#\\/b")).should("have.text", "false");
1172
+ checkStillMonkey();
1173
+
1174
+ cy.get(cesc(`#\\/ci2`)).select(`1`);
1175
+ cy.get(cesc(`#\\/bi`)).click();
1176
+ cy.get(cesc("#\\/b")).should("have.text", "true");
1177
+ checkStillMonkey();
1178
+
1179
+ cy.get(cesc(`#\\/ci2`)).select(`2`);
1180
+ cy.get(cesc(`#\\/bi`)).click();
1181
+ cy.get(cesc("#\\/b")).should("have.text", "false");
1182
+ checkStillMonkey();
1183
+ });
1184
+
1185
+ it("bind value to mathinput", () => {
1186
+ cy.window().then(async (win) => {
1187
+ win.postMessage(
1188
+ {
1189
+ doenetML: `
1190
+ <text>a</text>
1191
+ <choiceinput bindValueTo="$_mathinput1" name="ci1">
1192
+ <choice><math>x^2/2</math></choice>
1193
+ <choice><m>y</m></choice>
1194
+ <choice><math format="latex">\\frac{\\partial f}{\\partial x}</math></choice>
1195
+ <choice>3</choice>
1196
+ <choice><text>1/(e^x)</text></choice>
1197
+ </choiceinput>
1198
+
1199
+ <p>Select by typing: <mathinput prefill="y" /></p>
1200
+
1201
+ <copy name="copy" inline target="ci1" assignNames="ci2" createComponentOfType="choiceinput" />
1202
+
1203
+ <p>Selected values: <aslist>
1204
+ <copy prop='selectedvalue' target="ci1" />
1205
+ <copy prop='selectedvalue' target="copy" />
1206
+ </aslist></p>
1207
+ <p>Selected indices: <aslist>
1208
+ <copy prop='selectedindex' target="ci1" />
1209
+ <copy prop='selectedindex' target="copy" />
1210
+ </aslist></p>
1211
+
1212
+ `,
1213
+ },
1214
+ "*",
1215
+ );
1216
+ });
1217
+
1218
+ cy.get(cesc("#\\/_text1")).should("have.text", "a"); // to wait for page to load
1219
+
1220
+ let textOrder = ["(x²)/2", "y", "∂f/∂x", "3", "1/(e^x)"];
1221
+
1222
+ let checkChoices = function (selectedIndex, inputText, inputMath) {
1223
+ let choiceArray, indexArray;
1224
+
1225
+ let selectedChoice = null;
1226
+
1227
+ if (selectedIndex === null) {
1228
+ choiceArray = indexArray = [];
1229
+
1230
+ for (let i = 1; i <= 3; i++) {
1231
+ cy.get(cesc(`#\\/ci1_choice${i}_input`)).should("not.be.checked");
1232
+ }
1233
+ cy.get(cesc(`#\\/ci2`)).should("have.value", "");
1234
+
1235
+ cy.get(cesc("#\\/_p2")).should("have.text", `Selected values: `);
1236
+ cy.get(cesc("#\\/_p3")).should("have.text", `Selected indices: `);
1237
+ } else {
1238
+ selectedChoice = textOrder[selectedIndex - 1];
1239
+ choiceArray = [selectedChoice];
1240
+ indexArray = [selectedIndex];
1241
+
1242
+ for (let i = 1; i <= 3; i++) {
1243
+ if (i === selectedIndex) {
1244
+ cy.get(cesc(`#\\/ci1_choice${i}_input`)).should("be.checked");
1245
+ } else {
1246
+ cy.get(cesc(`#\\/ci1_choice${i}_input`)).should("not.be.checked");
1247
+ }
1248
+ }
1249
+ cy.get(cesc(`#\\/ci2`)).should("have.value", String(selectedIndex));
1250
+ cy.get(cesc("#\\/_p2")).should(
1251
+ "have.text",
1252
+ `Selected values: ${selectedChoice}, ${selectedChoice}`,
1253
+ );
1254
+ cy.get(cesc("#\\/_p3")).should(
1255
+ "have.text",
1256
+ `Selected indices: ${selectedIndex}, ${selectedIndex}`,
1257
+ );
1258
+ }
1259
+
1260
+ cy.get(cesc(`#\\/_mathinput1`) + ` .mq-editable-field`)
1261
+ .invoke("text")
1262
+ .then((text) => {
1263
+ expect(text.replace(/[\s\u200B-\u200D\uFEFF]/g, "")).equal(inputText);
1264
+ });
1265
+ cy.window().then(async (win) => {
1266
+ let stateVariables = await win.returnAllStateVariables1();
1267
+ expect(stateVariables["/ci1"].stateValues.selectedValues).eqls(
1268
+ choiceArray,
1269
+ );
1270
+ expect(stateVariables["/ci1"].stateValues.selectedIndices).eqls(
1271
+ indexArray,
1272
+ );
1273
+ expect(stateVariables["/ci2"].stateValues.selectedValues).eqls(
1274
+ choiceArray,
1275
+ );
1276
+ expect(stateVariables["/ci2"].stateValues.selectedIndices).eqls(
1277
+ indexArray,
1278
+ );
1279
+ expect(stateVariables["/_mathinput1"].stateValues.value).eqls(
1280
+ inputMath,
1281
+ );
1282
+ });
1283
+ };
1284
+
1285
+ cy.get(cesc("#\\/_math1")).should("contain.text", "x22");
1286
+
1287
+ checkChoices(2, "y", "y");
1288
+
1289
+ cy.log("select x^2/2 from first input");
1290
+ let selectedIndex = 1;
1291
+ let inputText = "x22";
1292
+ cy.get(cesc(`#\\/ci1_choice${selectedIndex}_input`))
1293
+ .should("not.be.visible")
1294
+ .click({ force: true }); // input is invisible (covered by text), but click it anyway
1295
+ checkChoices(selectedIndex, inputText, ["/", ["^", "x", 2], 2]);
1296
+
1297
+ cy.log("Type 3");
1298
+ selectedIndex = 4;
1299
+ inputText = "3";
1300
+ cy.get(cesc("#\\/_mathinput1") + " textarea").type(
1301
+ `{ctrl+home}{shift+end}{backspace}${inputText}{enter}`,
1302
+ { force: true },
1303
+ );
1304
+ checkChoices(selectedIndex, inputText, 3);
1305
+
1306
+ cy.log("select ∂f/∂x from second input");
1307
+ selectedIndex = 3;
1308
+ inputText = "∂f∂x";
1309
+ cy.get(cesc(`#\\/ci2`)).select(`${selectedIndex}`);
1310
+ checkChoices(selectedIndex, inputText, [
1311
+ "partial_derivative_leibniz",
1312
+ "f",
1313
+ ["tuple", "x"],
1314
+ ]);
1315
+
1316
+ cy.log("type e^{-x}");
1317
+ selectedIndex = null;
1318
+ inputText = "e−x";
1319
+ cy.get(cesc("#\\/_mathinput1") + " textarea").type(
1320
+ `{ctrl+home}{shift+end}{backspace}e^-x{enter}`,
1321
+ { force: true },
1322
+ );
1323
+ checkChoices(selectedIndex, inputText, ["^", "e", ["-", "x"]]);
1324
+
1325
+ cy.log("type 1/e^{x}");
1326
+ selectedIndex = 5;
1327
+ inputText = "1ex";
1328
+ cy.get(cesc("#\\/_mathinput1") + " textarea").type(
1329
+ `{ctrl+home}{shift+end}{backspace}1/e^x{enter}`,
1330
+ { force: true },
1331
+ );
1332
+ checkChoices(selectedIndex, inputText, ["/", 1, ["^", "e", "x"]]);
1333
+
1334
+ cy.log("select y from second input");
1335
+ selectedIndex = 2;
1336
+ inputText = "y";
1337
+ cy.get(cesc(`#\\/ci2`)).select(`${selectedIndex}`);
1338
+ checkChoices(selectedIndex, inputText, "y");
1339
+ });
1340
+
1341
+ it("preselect choices", () => {
1342
+ cy.window().then(async (win) => {
1343
+ win.postMessage(
1344
+ {
1345
+ doenetML: `
1346
+ <text>a</text>
1347
+ <choiceinput name="c1" preselectChoice="2">
1348
+ <choice>cat</choice>
1349
+ <choice>dog</choice>
1350
+ <choice>monkey</choice>
1351
+ <choice>mouse</choice>
1352
+ <choice>rabbit</choice>
1353
+ <choice>emu</choice>
1354
+ <choice>giraffe</choice>
1355
+ <choice>aardvark</choice>
1356
+ </choiceinput>
1357
+
1358
+ <choiceinput name="c2" inline shuffleOrder preselectChoice="2">
1359
+ <choice>cat</choice>
1360
+ <choice>dog</choice>
1361
+ <choice>monkey</choice>
1362
+ <choice>mouse</choice>
1363
+ <choice>rabbit</choice>
1364
+ <choice>emu</choice>
1365
+ <choice>giraffe</choice>
1366
+ <choice>aardvark</choice>
1367
+ </choiceinput>
1368
+
1369
+ <choiceinput name="c3" inline>
1370
+ <choice>cat</choice>
1371
+ <choice>dog</choice>
1372
+ <choice>monkey</choice>
1373
+ <choice preselect>mouse</choice>
1374
+ <choice>rabbit</choice>
1375
+ <choice>emu</choice>
1376
+ <choice>giraffe</choice>
1377
+ <choice>aardvark</choice>
1378
+ </choiceinput>
1379
+
1380
+
1381
+ <choiceinput name="c4" shuffleOrder>
1382
+ <choice>cat</choice>
1383
+ <choice>dog</choice>
1384
+ <choice>monkey</choice>
1385
+ <choice preselect>mouse</choice>
1386
+ <choice>rabbit</choice>
1387
+ <choice>emu</choice>
1388
+ <choice>giraffe</choice>
1389
+ <choice>aardvark</choice>
1390
+ </choiceinput>
1391
+
1392
+ <choiceinput name="c5" inline>
1393
+ <choice>cat</choice>
1394
+ <choice preselect>dog</choice>
1395
+ <choice>monkey</choice>
1396
+ <choice preselect>mouse</choice>
1397
+ <choice>rabbit</choice>
1398
+ <choice>emu</choice>
1399
+ <choice>giraffe</choice>
1400
+ <choice>aardvark</choice>
1401
+ </choiceinput>
1402
+
1403
+
1404
+ <choiceinput name="c6" shuffleOrder>
1405
+ <choice>cat</choice>
1406
+ <choice preselect>dog</choice>
1407
+ <choice>monkey</choice>
1408
+ <choice preselect>mouse</choice>
1409
+ <choice>rabbit</choice>
1410
+ <choice>emu</choice>
1411
+ <choice>giraffe</choice>
1412
+ <choice>aardvark</choice>
1413
+ </choiceinput>
1414
+
1415
+
1416
+ <choiceinput name="c7" preselectChoice="2">
1417
+ <choice>cat</choice>
1418
+ <choice>dog</choice>
1419
+ <choice>monkey</choice>
1420
+ <choice preselect>mouse</choice>
1421
+ <choice>rabbit</choice>
1422
+ <choice>emu</choice>
1423
+ <choice>giraffe</choice>
1424
+ <choice>aardvark</choice>
1425
+ </choiceinput>
1426
+
1427
+
1428
+ <choiceinput name="c8" shuffleOrder inline preselectChoice="2">
1429
+ <choice>cat</choice>
1430
+ <choice>dog</choice>
1431
+ <choice>monkey</choice>
1432
+ <choice preselect>mouse</choice>
1433
+ <choice>rabbit</choice>
1434
+ <choice>emu</choice>
1435
+ <choice>giraffe</choice>
1436
+ <choice>aardvark</choice>
1437
+ </choiceinput>
1438
+
1439
+
1440
+ `,
1441
+ },
1442
+ "*",
1443
+ );
1444
+ });
1445
+
1446
+ cy.get(cesc("#\\/_text1")).should("have.text", "a"); // to wait for page to load
1447
+
1448
+ cy.window().then(async (win) => {
1449
+ let stateVariables = await win.returnAllStateVariables1();
1450
+
1451
+ cy.get(cesc("#\\/c1_choice2_input")).should("be.checked");
1452
+
1453
+ cy.get(cesc("#\\/c2")).should("have.value", "2");
1454
+
1455
+ cy.get(cesc("#\\/c3")).should("have.value", "4");
1456
+
1457
+ let mouseInd4 =
1458
+ stateVariables["/c4"].stateValues.choiceTexts.indexOf("mouse");
1459
+ cy.get(cesc(`#\\/c4_choice${mouseInd4 + 1}_input`)).should("be.checked");
1460
+
1461
+ cy.get(cesc("#\\/c5")).should("have.value", "2");
1462
+
1463
+ let dogInd6 =
1464
+ stateVariables["/c6"].stateValues.choiceTexts.indexOf("dog");
1465
+ let mouseInd6 =
1466
+ stateVariables["/c6"].stateValues.choiceTexts.indexOf("mouse");
1467
+ let selectedInd6 = Math.min(dogInd6, mouseInd6);
1468
+
1469
+ cy.get(cesc(`#\\/c6_choice${selectedInd6 + 1}_input`)).should(
1470
+ "be.checked",
1471
+ );
1472
+
1473
+ cy.get(cesc("#\\/c7_choice4_input")).should("be.checked");
1474
+
1475
+ let mouseInd8 =
1476
+ stateVariables["/c8"].stateValues.choiceTexts.indexOf("mouse");
1477
+ cy.get(cesc("#\\/c8")).should("have.value", `${mouseInd8 + 1}`);
1478
+ });
1479
+ });
1480
+
1481
+ it("disabled choice with inline choiceinput", () => {
1482
+ cy.window().then(async (win) => {
1483
+ win.postMessage(
1484
+ {
1485
+ doenetML: `
1486
+ <text>a</text>
1487
+ <choiceinput inline placeholder="Choose animal">
1488
+ <choice>cat</choice>
1489
+ <choice>dog</choice>
1490
+ <choice disabled>monkey</choice>
1491
+ <choice>mouse</choice>
1492
+ </choiceinput>
1493
+
1494
+ <p>Selected value: <copy prop='selectedvalue' target="_choiceinput1" /></p>
1495
+ <p>Selected index: <copy prop='selectedindex' target="_choiceinput1" /></p>
1496
+ `,
1497
+ },
1498
+ "*",
1499
+ );
1500
+ });
1501
+
1502
+ cy.get(cesc("#\\/_text1")).should("have.text", "a"); // to wait for page to load
1503
+
1504
+ let originalChoices = ["cat", "dog", "monkey", "mouse"];
1505
+ cy.get(cesc("#\\/_p1")).should("have.text", "Selected value: ");
1506
+ cy.get(cesc("#\\/_p2")).should("have.text", "Selected index: ");
1507
+
1508
+ cy.get(cesc("#\\/_choiceinput1")).should("have.value", "");
1509
+
1510
+ let choices, choiceOrder;
1511
+ cy.window().then(async (win) => {
1512
+ let stateVariables = await win.returnAllStateVariables1();
1513
+ choices = [...stateVariables["/_choiceinput1"].stateValues.choiceTexts];
1514
+ choiceOrder = [
1515
+ ...stateVariables["/_choiceinput1"].stateValues.choiceOrder,
1516
+ ];
1517
+ expect(choices.length).eq(4);
1518
+ expect(originalChoices.includes(choices[0])).eq(true);
1519
+ expect(originalChoices.includes(choices[1])).eq(true);
1520
+ expect(originalChoices.includes(choices[2])).eq(true);
1521
+ expect(originalChoices.includes(choices[3])).eq(true);
1522
+ expect(choices[1]).not.eq(choices[0]);
1523
+ expect(choices[2]).not.eq(choices[0]);
1524
+ expect(choices[2]).not.eq(choices[1]);
1525
+ expect(choices[3]).not.eq(choices[0]);
1526
+ expect(choices[3]).not.eq(choices[1]);
1527
+ expect(choices[3]).not.eq(choices[2]);
1528
+ expect(stateVariables["/_choiceinput1"].stateValues.selectedValues).eqls(
1529
+ [],
1530
+ );
1531
+ expect(stateVariables["/_choiceinput1"].stateValues.selectedIndices).eqls(
1532
+ [],
1533
+ );
1534
+ // expect(stateVariables['/_choiceinput1'].stateValues.selectedoriginalindices).eqls([])
1535
+ expect(stateVariables["/_choiceinput1"].stateValues.inline).eq(true);
1536
+ expect(stateVariables["/_choiceinput1"].stateValues.shuffleOrder).eq(
1537
+ false,
1538
+ );
1539
+ expect(stateVariables["/_choice1"].stateValues.selected).eq(false);
1540
+ expect(stateVariables["/_choice2"].stateValues.selected).eq(false);
1541
+ expect(stateVariables["/_choice3"].stateValues.selected).eq(false);
1542
+ expect(stateVariables["/_choice4"].stateValues.selected).eq(false);
1543
+ });
1544
+
1545
+ cy.log("select options in order");
1546
+
1547
+ for (let i = 0; i < 4; i++) {
1548
+ if (i === 2) {
1549
+ cy.get(cesc(`#\\/_choiceinput1`))
1550
+ .get('[value="3"]')
1551
+ .should("be.disabled");
1552
+ } else {
1553
+ cy.get(cesc(`#\\/_choiceinput1`))
1554
+ .select(`${i + 1}`)
1555
+ .then(() => {
1556
+ // make this asynchronous so that choices is populated before line is executed
1557
+ cy.get(cesc("#\\/_p1")).should(
1558
+ "have.text",
1559
+ "Selected value: " + choices[i],
1560
+ );
1561
+ cy.get(cesc("#\\/_p2")).should(
1562
+ "have.text",
1563
+ "Selected index: " + (i + 1),
1564
+ );
1565
+
1566
+ cy.window().then(async (win) => {
1567
+ let stateVariables = await win.returnAllStateVariables1();
1568
+ expect(
1569
+ stateVariables["/_choiceinput1"].stateValues.selectedValues,
1570
+ ).eqls([choices[i]]);
1571
+ expect(
1572
+ stateVariables["/_choiceinput1"].stateValues.selectedIndices,
1573
+ ).eqls([i + 1]);
1574
+ expect(stateVariables["/_choice1"].stateValues.selected).eq(
1575
+ choiceOrder[i] === 1,
1576
+ );
1577
+ expect(stateVariables["/_choice2"].stateValues.selected).eq(
1578
+ choiceOrder[i] === 2,
1579
+ );
1580
+ expect(stateVariables["/_choice3"].stateValues.selected).eq(
1581
+ choiceOrder[i] === 3,
1582
+ );
1583
+ expect(stateVariables["/_choice4"].stateValues.selected).eq(
1584
+ choiceOrder[i] === 4,
1585
+ );
1586
+ });
1587
+ });
1588
+ }
1589
+ }
1590
+ });
1591
+
1592
+ it("hidden choice with inline choiceinput", () => {
1593
+ cy.window().then(async (win) => {
1594
+ win.postMessage(
1595
+ {
1596
+ doenetML: `
1597
+ <text>a</text>
1598
+ <choiceinput inline placeholder="Choose animal">
1599
+ <choice>cat</choice>
1600
+ <choice>dog</choice>
1601
+ <choice hide>monkey</choice>
1602
+ <choice>mouse</choice>
1603
+ </choiceinput>
1604
+
1605
+ <p>Selected value: <copy prop='selectedvalue' target="_choiceinput1" /></p>
1606
+ <p>Selected index: <copy prop='selectedindex' target="_choiceinput1" /></p>
1607
+ `,
1608
+ },
1609
+ "*",
1610
+ );
1611
+ });
1612
+
1613
+ cy.get(cesc("#\\/_text1")).should("have.text", "a"); // to wait for page to load
1614
+
1615
+ let originalChoices = ["cat", "dog", "monkey", "mouse"];
1616
+ cy.get(cesc("#\\/_p1")).should("have.text", "Selected value: ");
1617
+ cy.get(cesc("#\\/_p2")).should("have.text", "Selected index: ");
1618
+
1619
+ cy.get(cesc("#\\/_choiceinput1")).should("have.value", "");
1620
+
1621
+ let choices, choiceOrder;
1622
+ cy.window().then(async (win) => {
1623
+ let stateVariables = await win.returnAllStateVariables1();
1624
+ choices = [...stateVariables["/_choiceinput1"].stateValues.choiceTexts];
1625
+ choiceOrder = [
1626
+ ...stateVariables["/_choiceinput1"].stateValues.choiceOrder,
1627
+ ];
1628
+ expect(choices.length).eq(4);
1629
+ expect(originalChoices.includes(choices[0])).eq(true);
1630
+ expect(originalChoices.includes(choices[1])).eq(true);
1631
+ expect(originalChoices.includes(choices[2])).eq(true);
1632
+ expect(originalChoices.includes(choices[3])).eq(true);
1633
+ expect(choices[1]).not.eq(choices[0]);
1634
+ expect(choices[2]).not.eq(choices[0]);
1635
+ expect(choices[2]).not.eq(choices[1]);
1636
+ expect(choices[3]).not.eq(choices[0]);
1637
+ expect(choices[3]).not.eq(choices[1]);
1638
+ expect(choices[3]).not.eq(choices[2]);
1639
+ expect(stateVariables["/_choiceinput1"].stateValues.selectedValues).eqls(
1640
+ [],
1641
+ );
1642
+ expect(stateVariables["/_choiceinput1"].stateValues.selectedIndices).eqls(
1643
+ [],
1644
+ );
1645
+ expect(stateVariables["/_choiceinput1"].stateValues.inline).eq(true);
1646
+ expect(stateVariables["/_choiceinput1"].stateValues.shuffleOrder).eq(
1647
+ false,
1648
+ );
1649
+ expect(stateVariables["/_choice1"].stateValues.selected).eq(false);
1650
+ expect(stateVariables["/_choice2"].stateValues.selected).eq(false);
1651
+ expect(stateVariables["/_choice3"].stateValues.selected).eq(false);
1652
+ expect(stateVariables["/_choice4"].stateValues.selected).eq(false);
1653
+ });
1654
+
1655
+ cy.log("select options in order");
1656
+
1657
+ for (let i = 0; i < 4; i++) {
1658
+ if (i === 2) {
1659
+ cy.get(cesc(`#\\/_choiceinput1`))
1660
+ .get('[value="3"]')
1661
+ .should("not.exist");
1662
+ } else {
1663
+ cy.get(cesc(`#\\/_choiceinput1`))
1664
+ .select(`${i + 1}`)
1665
+ .then(() => {
1666
+ // make this asynchronous so that choices is populated before line is executed
1667
+ cy.get(cesc("#\\/_p1")).should(
1668
+ "have.text",
1669
+ "Selected value: " + choices[i],
1670
+ );
1671
+ cy.get(cesc("#\\/_p2")).should(
1672
+ "have.text",
1673
+ "Selected index: " + (i + 1),
1674
+ );
1675
+
1676
+ cy.window().then(async (win) => {
1677
+ let stateVariables = await win.returnAllStateVariables1();
1678
+ expect(
1679
+ stateVariables["/_choiceinput1"].stateValues.selectedValues,
1680
+ ).eqls([choices[i]]);
1681
+ expect(
1682
+ stateVariables["/_choiceinput1"].stateValues.selectedIndices,
1683
+ ).eqls([i + 1]);
1684
+ expect(stateVariables["/_choice1"].stateValues.selected).eq(
1685
+ choiceOrder[i] === 1,
1686
+ );
1687
+ expect(stateVariables["/_choice2"].stateValues.selected).eq(
1688
+ choiceOrder[i] === 2,
1689
+ );
1690
+ expect(stateVariables["/_choice3"].stateValues.selected).eq(
1691
+ choiceOrder[i] === 3,
1692
+ );
1693
+ expect(stateVariables["/_choice4"].stateValues.selected).eq(
1694
+ choiceOrder[i] === 4,
1695
+ );
1696
+ });
1697
+ });
1698
+ }
1699
+ }
1700
+ });
1701
+
1702
+ it("disabled choice with block choiceinput", () => {
1703
+ cy.window().then(async (win) => {
1704
+ win.postMessage(
1705
+ {
1706
+ doenetML: `
1707
+ <text>a</text>
1708
+ <choiceinput>
1709
+ <choice>cat</choice>
1710
+ <choice>dog</choice>
1711
+ <choice disabled>monkey</choice>
1712
+ <choice>mouse</choice>
1713
+ </choiceinput>
1714
+
1715
+ <p>Selected value: <copy prop='selectedvalue' target="_choiceinput1" /></p>
1716
+ <p>Selected index: <copy prop='selectedindex' target="_choiceinput1" /></p>
1717
+ `,
1718
+ },
1719
+ "*",
1720
+ );
1721
+ });
1722
+
1723
+ cy.get(cesc("#\\/_text1")).should("have.text", "a"); // to wait for page to load
1724
+
1725
+ let originalChoices = ["cat", "dog", "monkey", "mouse"];
1726
+ cy.get(cesc("#\\/_p1")).should("have.text", "Selected value: ");
1727
+ cy.get(cesc("#\\/_p2")).should("have.text", "Selected index: ");
1728
+
1729
+ let choices;
1730
+ cy.window().then(async (win) => {
1731
+ let stateVariables = await win.returnAllStateVariables1();
1732
+ choices = [...stateVariables["/_choiceinput1"].stateValues.choiceTexts];
1733
+ expect(choices).eqls(originalChoices);
1734
+ expect(stateVariables["/_choiceinput1"].stateValues.selectedValues).eqls(
1735
+ [],
1736
+ );
1737
+ expect(stateVariables["/_choiceinput1"].stateValues.selectedIndices).eqls(
1738
+ [],
1739
+ );
1740
+ expect(stateVariables["/_choiceinput1"].stateValues.inline).eq(false);
1741
+ expect(stateVariables["/_choiceinput1"].stateValues.shuffleOrder).eq(
1742
+ false,
1743
+ );
1744
+ expect(stateVariables["/_choice1"].stateValues.selected).eq(false);
1745
+ expect(stateVariables["/_choice2"].stateValues.selected).eq(false);
1746
+ expect(stateVariables["/_choice3"].stateValues.selected).eq(false);
1747
+ expect(stateVariables["/_choice4"].stateValues.selected).eq(false);
1748
+ });
1749
+
1750
+ cy.log("select options in order");
1751
+
1752
+ for (let i = 0; i < 4; i++) {
1753
+ if (i === 2) {
1754
+ cy.get(cesc(`#\\/_choiceinput1_choice${i + 1}_input`)).should(
1755
+ "be.disabled",
1756
+ );
1757
+ } else {
1758
+ cy.get(cesc(`#\\/_choiceinput1_choice${i + 1}_input`))
1759
+ .click()
1760
+ .then(() => {
1761
+ // make this asynchronous so that choices is populated before line is executed
1762
+ cy.get(cesc("#\\/_p1")).should(
1763
+ "have.text",
1764
+ "Selected value: " + choices[i],
1765
+ );
1766
+ cy.get(cesc("#\\/_p2")).should(
1767
+ "have.text",
1768
+ "Selected index: " + (i + 1),
1769
+ );
1770
+
1771
+ cy.window().then(async (win) => {
1772
+ let stateVariables = await win.returnAllStateVariables1();
1773
+ expect(
1774
+ stateVariables["/_choiceinput1"].stateValues.selectedValues,
1775
+ ).eqls([choices[i]]);
1776
+ expect(
1777
+ stateVariables["/_choiceinput1"].stateValues.selectedIndices,
1778
+ ).eqls([i + 1]);
1779
+ expect(stateVariables["/_choice1"].stateValues.selected).eq(
1780
+ i === 0,
1781
+ );
1782
+ expect(stateVariables["/_choice2"].stateValues.selected).eq(
1783
+ i === 1,
1784
+ );
1785
+ expect(stateVariables["/_choice3"].stateValues.selected).eq(
1786
+ i === 2,
1787
+ );
1788
+ expect(stateVariables["/_choice4"].stateValues.selected).eq(
1789
+ i === 3,
1790
+ );
1791
+ });
1792
+ });
1793
+ }
1794
+ }
1795
+ });
1796
+
1797
+ it("hidden choice with block choiceinput", () => {
1798
+ cy.window().then(async (win) => {
1799
+ win.postMessage(
1800
+ {
1801
+ doenetML: `
1802
+ <text>a</text>
1803
+ <choiceinput>
1804
+ <choice>cat</choice>
1805
+ <choice>dog</choice>
1806
+ <choice hide>monkey</choice>
1807
+ <choice>mouse</choice>
1808
+ </choiceinput>
1809
+
1810
+ <p>Selected value: <copy prop='selectedvalue' target="_choiceinput1" /></p>
1811
+ <p>Selected index: <copy prop='selectedindex' target="_choiceinput1" /></p>
1812
+ `,
1813
+ },
1814
+ "*",
1815
+ );
1816
+ });
1817
+
1818
+ cy.get(cesc("#\\/_text1")).should("have.text", "a"); // to wait for page to load
1819
+
1820
+ let originalChoices = ["cat", "dog", "monkey", "mouse"];
1821
+ cy.get(cesc("#\\/_p1")).should("have.text", "Selected value: ");
1822
+ cy.get(cesc("#\\/_p2")).should("have.text", "Selected index: ");
1823
+
1824
+ let choices;
1825
+ cy.window().then(async (win) => {
1826
+ let stateVariables = await win.returnAllStateVariables1();
1827
+ choices = [...stateVariables["/_choiceinput1"].stateValues.choiceTexts];
1828
+ expect(choices).eqls(originalChoices);
1829
+ expect(stateVariables["/_choiceinput1"].stateValues.selectedValues).eqls(
1830
+ [],
1831
+ );
1832
+ expect(stateVariables["/_choiceinput1"].stateValues.selectedIndices).eqls(
1833
+ [],
1834
+ );
1835
+ expect(stateVariables["/_choiceinput1"].stateValues.inline).eq(false);
1836
+ expect(stateVariables["/_choiceinput1"].stateValues.shuffleOrder).eq(
1837
+ false,
1838
+ );
1839
+ expect(stateVariables["/_choice1"].stateValues.selected).eq(false);
1840
+ expect(stateVariables["/_choice2"].stateValues.selected).eq(false);
1841
+ expect(stateVariables["/_choice3"].stateValues.selected).eq(false);
1842
+ expect(stateVariables["/_choice4"].stateValues.selected).eq(false);
1843
+ });
1844
+
1845
+ cy.log("select options in order");
1846
+
1847
+ for (let i = 0; i < 4; i++) {
1848
+ if (i === 2) {
1849
+ cy.get(cesc(`#\\/_choiceinput1_choice${i + 1}_input`)).should(
1850
+ "not.exist",
1851
+ );
1852
+ } else {
1853
+ cy.get(cesc(`#\\/_choiceinput1_choice${i + 1}_input`))
1854
+ .click()
1855
+ .then(() => {
1856
+ // make this asynchronous so that choices is populated before line is executed
1857
+ cy.get(cesc("#\\/_p1")).should(
1858
+ "have.text",
1859
+ "Selected value: " + choices[i],
1860
+ );
1861
+ cy.get(cesc("#\\/_p2")).should(
1862
+ "have.text",
1863
+ "Selected index: " + (i + 1),
1864
+ );
1865
+
1866
+ cy.window().then(async (win) => {
1867
+ let stateVariables = await win.returnAllStateVariables1();
1868
+ expect(
1869
+ stateVariables["/_choiceinput1"].stateValues.selectedValues,
1870
+ ).eqls([choices[i]]);
1871
+ expect(
1872
+ stateVariables["/_choiceinput1"].stateValues.selectedIndices,
1873
+ ).eqls([i + 1]);
1874
+ expect(stateVariables["/_choice1"].stateValues.selected).eq(
1875
+ i === 0,
1876
+ );
1877
+ expect(stateVariables["/_choice2"].stateValues.selected).eq(
1878
+ i === 1,
1879
+ );
1880
+ expect(stateVariables["/_choice3"].stateValues.selected).eq(
1881
+ i === 2,
1882
+ );
1883
+ expect(stateVariables["/_choice4"].stateValues.selected).eq(
1884
+ i === 3,
1885
+ );
1886
+ });
1887
+ });
1888
+ }
1889
+ }
1890
+ });
1891
+
1892
+ it("select multiple with block choiceinput", () => {
1893
+ cy.window().then(async (win) => {
1894
+ win.postMessage(
1895
+ {
1896
+ doenetML: `
1897
+ <text>a</text>
1898
+ <choiceinput shuffleOrder selectMultiple>
1899
+ <choice>cat</choice>
1900
+ <choice>dog</choice>
1901
+ <choice>monkey</choice>
1902
+ <choice>mouse</choice>
1903
+ </choiceinput>
1904
+
1905
+ <p>Selected values: <aslist><copy prop='selectedvalues' target="_choiceinput1" /></aslist></p>
1906
+ <p>Selected indices: <aslist><copy prop='selectedindices' target="_choiceinput1" /></aslist></p>
1907
+ `,
1908
+ },
1909
+ "*",
1910
+ );
1911
+ });
1912
+
1913
+ cy.get(cesc("#\\/_text1")).should("have.text", "a"); // to wait for page to load
1914
+
1915
+ let originalChoices = ["cat", "dog", "monkey", "mouse"];
1916
+ cy.get(cesc("#\\/_p1")).should("have.text", "Selected values: ");
1917
+ cy.get(cesc("#\\/_p2")).should("have.text", "Selected indices: ");
1918
+
1919
+ let choices, choiceOrder;
1920
+ cy.window().then(async (win) => {
1921
+ let stateVariables = await win.returnAllStateVariables1();
1922
+ choices = [...stateVariables["/_choiceinput1"].stateValues.choiceTexts];
1923
+ choiceOrder = [
1924
+ ...stateVariables["/_choiceinput1"].stateValues.choiceOrder,
1925
+ ];
1926
+ expect(choices.length).eq(4);
1927
+ expect(originalChoices.includes(choices[0])).eq(true);
1928
+ expect(originalChoices.includes(choices[1])).eq(true);
1929
+ expect(originalChoices.includes(choices[2])).eq(true);
1930
+ expect(originalChoices.includes(choices[3])).eq(true);
1931
+ expect(choices[1]).not.eq(choices[0]);
1932
+ expect(choices[2]).not.eq(choices[0]);
1933
+ expect(choices[2]).not.eq(choices[1]);
1934
+ expect(choices[3]).not.eq(choices[0]);
1935
+ expect(choices[3]).not.eq(choices[1]);
1936
+ expect(choices[3]).not.eq(choices[2]);
1937
+ expect(stateVariables["/_choiceinput1"].stateValues.selectedValues).eqls(
1938
+ [],
1939
+ );
1940
+ expect(stateVariables["/_choiceinput1"].stateValues.selectedIndices).eqls(
1941
+ [],
1942
+ );
1943
+ expect(stateVariables["/_choiceinput1"].stateValues.inline).eq(false);
1944
+ expect(stateVariables["/_choiceinput1"].stateValues.shuffleOrder).eq(
1945
+ true,
1946
+ );
1947
+ expect(stateVariables["/_choiceinput1"].stateValues.selectMultiple).eq(
1948
+ true,
1949
+ );
1950
+ expect(stateVariables["/_choice1"].stateValues.selected).eq(false);
1951
+ expect(stateVariables["/_choice2"].stateValues.selected).eq(false);
1952
+ expect(stateVariables["/_choice3"].stateValues.selected).eq(false);
1953
+ expect(stateVariables["/_choice4"].stateValues.selected).eq(false);
1954
+ });
1955
+
1956
+ cy.log("select options in order");
1957
+
1958
+ for (let i = 0; i < 4; i++) {
1959
+ cy.get(cesc(`#\\/_choiceinput1_choice${i + 1}_input`))
1960
+ .click()
1961
+ .then(() => {
1962
+ // make this asynchronous so that choices is populated before line is executed
1963
+ cy.get(cesc("#\\/_p1")).should(
1964
+ "have.text",
1965
+ "Selected values: " + choices.slice(0, i + 1).join(", "),
1966
+ );
1967
+ cy.get(cesc("#\\/_p2")).should(
1968
+ "have.text",
1969
+ "Selected indices: " +
1970
+ [...Array(i + 1).keys()].map((x) => x + 1).join(", "),
1971
+ );
1972
+ });
1973
+
1974
+ cy.window().then(async (win) => {
1975
+ let stateVariables = await win.returnAllStateVariables1();
1976
+ expect(
1977
+ stateVariables["/_choiceinput1"].stateValues.selectedValues,
1978
+ ).eqls(choices.slice(0, i + 1));
1979
+ expect(
1980
+ stateVariables["/_choiceinput1"].stateValues.selectedIndices,
1981
+ ).eqls([...Array(i + 1).keys()].map((x) => x + 1));
1982
+ expect(stateVariables["/_choice1"].stateValues.selected).eq(
1983
+ choiceOrder.indexOf(1) <= i,
1984
+ );
1985
+ expect(stateVariables["/_choice2"].stateValues.selected).eq(
1986
+ choiceOrder.indexOf(2) <= i,
1987
+ );
1988
+ expect(stateVariables["/_choice3"].stateValues.selected).eq(
1989
+ choiceOrder.indexOf(3) <= i,
1990
+ );
1991
+ expect(stateVariables["/_choice4"].stateValues.selected).eq(
1992
+ choiceOrder.indexOf(4) <= i,
1993
+ );
1994
+ });
1995
+ }
1996
+
1997
+ cy.log("deselect options in order");
1998
+
1999
+ for (let i = 0; i < 4; i++) {
2000
+ cy.get(cesc(`#\\/_choiceinput1_choice${i + 1}_input`))
2001
+ .click()
2002
+ .then(() => {
2003
+ // make this asynchronous so that choices is populated before line is executed
2004
+ cy.get(cesc("#\\/_p1")).should(
2005
+ "have.text",
2006
+ "Selected values: " + choices.slice(i + 1).join(", "),
2007
+ );
2008
+ cy.get(cesc("#\\/_p2")).should(
2009
+ "have.text",
2010
+ "Selected indices: " +
2011
+ [...Array(3 - i).keys()].map((x) => x + 2 + i).join(", "),
2012
+ );
2013
+ });
2014
+
2015
+ cy.window().then(async (win) => {
2016
+ let stateVariables = await win.returnAllStateVariables1();
2017
+ expect(
2018
+ stateVariables["/_choiceinput1"].stateValues.selectedValues,
2019
+ ).eqls(choices.slice(i + 1));
2020
+ expect(
2021
+ stateVariables["/_choiceinput1"].stateValues.selectedIndices,
2022
+ ).eqls([...Array(3 - i).keys()].map((x) => x + 2 + i));
2023
+ expect(stateVariables["/_choice1"].stateValues.selected).eq(
2024
+ choiceOrder.indexOf(1) > i,
2025
+ );
2026
+ expect(stateVariables["/_choice2"].stateValues.selected).eq(
2027
+ choiceOrder.indexOf(2) > i,
2028
+ );
2029
+ expect(stateVariables["/_choice3"].stateValues.selected).eq(
2030
+ choiceOrder.indexOf(3) > i,
2031
+ );
2032
+ expect(stateVariables["/_choice4"].stateValues.selected).eq(
2033
+ choiceOrder.indexOf(4) > i,
2034
+ );
2035
+ });
2036
+ }
2037
+
2038
+ cy.log("select options in reverse order");
2039
+
2040
+ for (let i = 3; i >= 0; i--) {
2041
+ cy.get(cesc(`#\\/_choiceinput1_choice${i + 1}_input`))
2042
+ .click()
2043
+ .then(() => {
2044
+ // make this asynchronous so that choices is populated before line is executed
2045
+ cy.get(cesc("#\\/_p1")).should(
2046
+ "have.text",
2047
+ "Selected values: " + choices.slice(i).join(", "),
2048
+ );
2049
+ cy.get(cesc("#\\/_p2")).should(
2050
+ "have.text",
2051
+ "Selected indices: " +
2052
+ [...Array(4 - i).keys()].map((x) => x + 1 + i).join(", "),
2053
+ );
2054
+ });
2055
+
2056
+ cy.window().then(async (win) => {
2057
+ let stateVariables = await win.returnAllStateVariables1();
2058
+ expect(
2059
+ stateVariables["/_choiceinput1"].stateValues.selectedValues,
2060
+ ).eqls(choices.slice(i));
2061
+ expect(
2062
+ stateVariables["/_choiceinput1"].stateValues.selectedIndices,
2063
+ ).eqls([...Array(4 - i).keys()].map((x) => x + 1 + i));
2064
+ expect(stateVariables["/_choice1"].stateValues.selected).eq(
2065
+ choiceOrder.indexOf(1) >= i,
2066
+ );
2067
+ expect(stateVariables["/_choice2"].stateValues.selected).eq(
2068
+ choiceOrder.indexOf(2) >= i,
2069
+ );
2070
+ expect(stateVariables["/_choice3"].stateValues.selected).eq(
2071
+ choiceOrder.indexOf(3) >= i,
2072
+ );
2073
+ expect(stateVariables["/_choice4"].stateValues.selected).eq(
2074
+ choiceOrder.indexOf(4) >= i,
2075
+ );
2076
+ });
2077
+ }
2078
+ });
2079
+
2080
+ it("select multiple with inline choiceinput", () => {
2081
+ cy.window().then(async (win) => {
2082
+ win.postMessage(
2083
+ {
2084
+ doenetML: `
2085
+ <text>a</text>
2086
+ <choiceinput inline shuffleOrder selectMultiple>
2087
+ <choice>cat</choice>
2088
+ <choice>dog</choice>
2089
+ <choice>monkey</choice>
2090
+ <choice>mouse</choice>
2091
+ </choiceinput>
2092
+
2093
+ <p>Selected values: <aslist><copy prop='selectedvalues' target="_choiceinput1" /></aslist></p>
2094
+ <p>Selected indices: <aslist><copy prop='selectedindices' target="_choiceinput1" /></aslist></p>
2095
+ `,
2096
+ },
2097
+ "*",
2098
+ );
2099
+ });
2100
+
2101
+ cy.get(cesc("#\\/_text1")).should("have.text", "a"); // to wait for page to load
2102
+
2103
+ cy.get(cesc("#\\/_choiceinput1")).invoke("val").should("deep.equal", []);
2104
+
2105
+ let originalChoices = ["cat", "dog", "monkey", "mouse"];
2106
+ cy.get(cesc("#\\/_p1")).should("have.text", "Selected values: ");
2107
+ cy.get(cesc("#\\/_p2")).should("have.text", "Selected indices: ");
2108
+
2109
+ let choices, choiceOrder;
2110
+ cy.window().then(async (win) => {
2111
+ let stateVariables = await win.returnAllStateVariables1();
2112
+ choices = [...stateVariables["/_choiceinput1"].stateValues.choiceTexts];
2113
+ choiceOrder = [
2114
+ ...stateVariables["/_choiceinput1"].stateValues.choiceOrder,
2115
+ ];
2116
+ expect(choices.length).eq(4);
2117
+ expect(originalChoices.includes(choices[0])).eq(true);
2118
+ expect(originalChoices.includes(choices[1])).eq(true);
2119
+ expect(originalChoices.includes(choices[2])).eq(true);
2120
+ expect(originalChoices.includes(choices[3])).eq(true);
2121
+ expect(choices[1]).not.eq(choices[0]);
2122
+ expect(choices[2]).not.eq(choices[0]);
2123
+ expect(choices[2]).not.eq(choices[1]);
2124
+ expect(choices[3]).not.eq(choices[0]);
2125
+ expect(choices[3]).not.eq(choices[1]);
2126
+ expect(choices[3]).not.eq(choices[2]);
2127
+ expect(stateVariables["/_choiceinput1"].stateValues.selectedValues).eqls(
2128
+ [],
2129
+ );
2130
+ expect(stateVariables["/_choiceinput1"].stateValues.selectedIndices).eqls(
2131
+ [],
2132
+ );
2133
+ expect(stateVariables["/_choiceinput1"].stateValues.inline).eq(true);
2134
+ expect(stateVariables["/_choiceinput1"].stateValues.shuffleOrder).eq(
2135
+ true,
2136
+ );
2137
+ expect(stateVariables["/_choiceinput1"].stateValues.selectMultiple).eq(
2138
+ true,
2139
+ );
2140
+ expect(stateVariables["/_choice1"].stateValues.selected).eq(false);
2141
+ expect(stateVariables["/_choice2"].stateValues.selected).eq(false);
2142
+ expect(stateVariables["/_choice3"].stateValues.selected).eq(false);
2143
+ expect(stateVariables["/_choice4"].stateValues.selected).eq(false);
2144
+ });
2145
+
2146
+ cy.log("select options in order");
2147
+
2148
+ for (let i = 0; i < 4; i++) {
2149
+ // TODO: the onChange handler wasn't triggering when didn't first deselect
2150
+ // so, as a stopgap, we're deselecting all here.
2151
+ // We shouldn't need to do this!
2152
+
2153
+ cy.get(cesc(`#\\/_choiceinput1`)).select([]);
2154
+
2155
+ cy.get(cesc(`#\\/_choiceinput1`))
2156
+ .select([...Array(i + 1).keys()].map((x) => String(x + 1)))
2157
+ .then(() => {
2158
+ let selectedInds = [...Array(i + 1).keys()].map((x) => String(x + 1));
2159
+ cy.get(cesc("#\\/_choiceinput1"))
2160
+ .invoke("val")
2161
+ .should("deep.equal", selectedInds);
2162
+
2163
+ // make this asynchronous so that choices is populated before line is executed
2164
+ cy.get(cesc("#\\/_p1")).should(
2165
+ "have.text",
2166
+ "Selected values: " + choices.slice(0, i + 1).join(", "),
2167
+ );
2168
+ cy.get(cesc("#\\/_p2")).should(
2169
+ "have.text",
2170
+ "Selected indices: " +
2171
+ [...Array(i + 1).keys()].map((x) => x + 1).join(", "),
2172
+ );
2173
+ });
2174
+
2175
+ cy.window().then(async (win) => {
2176
+ let stateVariables = await win.returnAllStateVariables1();
2177
+ expect(
2178
+ stateVariables["/_choiceinput1"].stateValues.selectedValues,
2179
+ ).eqls(choices.slice(0, i + 1));
2180
+ expect(
2181
+ stateVariables["/_choiceinput1"].stateValues.selectedIndices,
2182
+ ).eqls([...Array(i + 1).keys()].map((x) => x + 1));
2183
+ expect(stateVariables["/_choice1"].stateValues.selected).eq(
2184
+ choiceOrder.indexOf(1) <= i,
2185
+ );
2186
+ expect(stateVariables["/_choice2"].stateValues.selected).eq(
2187
+ choiceOrder.indexOf(2) <= i,
2188
+ );
2189
+ expect(stateVariables["/_choice3"].stateValues.selected).eq(
2190
+ choiceOrder.indexOf(3) <= i,
2191
+ );
2192
+ expect(stateVariables["/_choice4"].stateValues.selected).eq(
2193
+ choiceOrder.indexOf(4) <= i,
2194
+ );
2195
+ });
2196
+ }
2197
+
2198
+ cy.log("deselect options in order");
2199
+
2200
+ for (let i = 0; i < 4; i++) {
2201
+ cy.window().then(async (win) => {
2202
+ let indicesToSelect = [...Array(3 - i).keys()].map((x) =>
2203
+ String(x + 2 + i),
2204
+ );
2205
+ if (i === 3) {
2206
+ indicesToSelect = [""];
2207
+ }
2208
+
2209
+ cy.get(cesc(`#\\/_choiceinput1`))
2210
+ .select(indicesToSelect)
2211
+ .then(() => {
2212
+ let selectedInds = [...Array(3 - i).keys()].map((x) =>
2213
+ String(x + 2 + i),
2214
+ );
2215
+ cy.get(cesc("#\\/_choiceinput1"))
2216
+ .invoke("val")
2217
+ .should("deep.equal", selectedInds);
2218
+
2219
+ // make this asynchronous so that choices is populated before line is executed
2220
+ cy.get(cesc("#\\/_p1")).should(
2221
+ "have.text",
2222
+ "Selected values: " + choices.slice(i + 1).join(", "),
2223
+ );
2224
+ cy.get(cesc("#\\/_p2")).should(
2225
+ "have.text",
2226
+ "Selected indices: " +
2227
+ [...Array(3 - i).keys()].map((x) => x + 2 + i).join(", "),
2228
+ );
2229
+ });
2230
+
2231
+ cy.window().then(async (win) => {
2232
+ let stateVariables = await win.returnAllStateVariables1();
2233
+ expect(
2234
+ stateVariables["/_choiceinput1"].stateValues.selectedValues,
2235
+ ).eqls(choices.slice(i + 1));
2236
+ expect(
2237
+ stateVariables["/_choiceinput1"].stateValues.selectedIndices,
2238
+ ).eqls([...Array(3 - i).keys()].map((x) => x + 2 + i));
2239
+ expect(stateVariables["/_choice1"].stateValues.selected).eq(
2240
+ choiceOrder.indexOf(1) > i,
2241
+ );
2242
+ expect(stateVariables["/_choice2"].stateValues.selected).eq(
2243
+ choiceOrder.indexOf(2) > i,
2244
+ );
2245
+ expect(stateVariables["/_choice3"].stateValues.selected).eq(
2246
+ choiceOrder.indexOf(3) > i,
2247
+ );
2248
+ expect(stateVariables["/_choice4"].stateValues.selected).eq(
2249
+ choiceOrder.indexOf(4) > i,
2250
+ );
2251
+ });
2252
+ });
2253
+ }
2254
+
2255
+ cy.log("select options in reverse order");
2256
+
2257
+ for (let i = 3; i >= 0; i--) {
2258
+ cy.get(cesc(`#\\/_choiceinput1`))
2259
+ .select([...Array(4 - i).keys()].map((x) => String(x + 1 + i)))
2260
+ .then(() => {
2261
+ let selectedInds = [...Array(4 - i).keys()].map((x) =>
2262
+ String(x + 1 + i),
2263
+ );
2264
+ cy.get(cesc("#\\/_choiceinput1"))
2265
+ .invoke("val")
2266
+ .should("deep.equal", selectedInds);
2267
+
2268
+ // make this asynchronous so that choices is populated before line is executed
2269
+ cy.get(cesc("#\\/_p1")).should(
2270
+ "have.text",
2271
+ "Selected values: " + choices.slice(i).join(", "),
2272
+ );
2273
+ cy.get(cesc("#\\/_p2")).should(
2274
+ "have.text",
2275
+ "Selected indices: " +
2276
+ [...Array(4 - i).keys()].map((x) => x + 1 + i).join(", "),
2277
+ );
2278
+ });
2279
+
2280
+ cy.window().then(async (win) => {
2281
+ let stateVariables = await win.returnAllStateVariables1();
2282
+ expect(
2283
+ stateVariables["/_choiceinput1"].stateValues.selectedValues,
2284
+ ).eqls(choices.slice(i));
2285
+ expect(
2286
+ stateVariables["/_choiceinput1"].stateValues.selectedIndices,
2287
+ ).eqls([...Array(4 - i).keys()].map((x) => x + 1 + i));
2288
+ expect(stateVariables["/_choice1"].stateValues.selected).eq(
2289
+ choiceOrder.indexOf(1) >= i,
2290
+ );
2291
+ expect(stateVariables["/_choice2"].stateValues.selected).eq(
2292
+ choiceOrder.indexOf(2) >= i,
2293
+ );
2294
+ expect(stateVariables["/_choice3"].stateValues.selected).eq(
2295
+ choiceOrder.indexOf(3) >= i,
2296
+ );
2297
+ expect(stateVariables["/_choice4"].stateValues.selected).eq(
2298
+ choiceOrder.indexOf(4) >= i,
2299
+ );
2300
+ });
2301
+ }
2302
+ });
2303
+
2304
+ it("chain update off choiceinput", () => {
2305
+ cy.window().then(async (win) => {
2306
+ win.postMessage(
2307
+ {
2308
+ doenetML: `
2309
+ <text>a</text>
2310
+ <choiceinput name="ci" >
2311
+ <choice>red</choice>
2312
+ <choice>orange</choice>
2313
+ <choice>yellow</choice>
2314
+ <choice>green</choice>
2315
+ <choice>blue</choice>
2316
+ </choiceinput>
2317
+
2318
+ <text name="t"></text>
2319
+ <updateValue triggerWith="ci" target="t" newValue="$t $ci" type="text" />
2320
+ `,
2321
+ },
2322
+ "*",
2323
+ );
2324
+ });
2325
+
2326
+ cy.get(cesc("#\\/_text1")).should("have.text", "a"); // to wait until loaded
2327
+
2328
+ cy.get(cesc("#\\/t")).should("have.text", "");
2329
+
2330
+ cy.get(cesc(`#\\/ci_choice2_input`)).click();
2331
+ cy.get(cesc("#\\/t")).should("have.text", " orange");
2332
+
2333
+ cy.get(cesc(`#\\/ci_choice5_input`)).click();
2334
+ cy.get(cesc("#\\/t")).should("have.text", " orange blue");
2335
+
2336
+ cy.get(cesc(`#\\/ci_choice1_input`)).click();
2337
+ cy.get(cesc("#\\/t")).should("have.text", " orange blue red");
2338
+ });
2339
+
2340
+ // verify fixed bug where shuffle order was recalculated
2341
+ // causing a copy with no link to have a different shuffle order
2342
+ it("shuffleOrder is not recalculated when copy with no link", () => {
2343
+ cy.window().then(async (win) => {
2344
+ win.postMessage(
2345
+ {
2346
+ doenetML: `
2347
+ <text>a</text>
2348
+ <variantControl uniqueVariants="false" />
2349
+
2350
+ <group name="g" newNamespace>
2351
+ <choiceinput shuffleOrder name="ci">
2352
+ <choice>a</choice>
2353
+ <choice>b</choice>
2354
+ <choice>c</choice>
2355
+ <choice>d</choice>
2356
+ <choice>e</choice>
2357
+ </choiceinput>
2358
+ </group>
2359
+
2360
+ <copy target="g" assignNames="g2" link="false" />
2361
+ `,
2362
+ },
2363
+ "*",
2364
+ );
2365
+ });
2366
+
2367
+ cy.get(cesc("#\\/_text1")).should("have.text", "a"); // to wait until loaded
2368
+
2369
+ let choices = ["a", "b", "c", "d", "e"];
2370
+
2371
+ cy.window().then(async (win) => {
2372
+ let stateVariables = await win.returnAllStateVariables1();
2373
+
2374
+ let choiceOrder = stateVariables["/g/ci"].stateValues.choiceOrder;
2375
+ let choiceOrder2 = stateVariables["/g2/ci"].stateValues.choiceOrder;
2376
+
2377
+ expect(choiceOrder2).eqls(choiceOrder);
2378
+
2379
+ cy.get(`label[for=${cesc2("/g/ci_choice1_input")}]`).should(
2380
+ "have.text",
2381
+ choices[choiceOrder[0] - 1],
2382
+ );
2383
+ cy.get(`label[for=${cesc2("/g/ci_choice2_input")}]`).should(
2384
+ "have.text",
2385
+ choices[choiceOrder[1] - 1],
2386
+ );
2387
+ cy.get(`label[for=${cesc2("/g/ci_choice3_input")}]`).should(
2388
+ "have.text",
2389
+ choices[choiceOrder[2] - 1],
2390
+ );
2391
+ cy.get(`label[for=${cesc2("/g/ci_choice4_input")}]`).should(
2392
+ "have.text",
2393
+ choices[choiceOrder[3] - 1],
2394
+ );
2395
+ cy.get(`label[for=${cesc2("/g/ci_choice5_input")}]`).should(
2396
+ "have.text",
2397
+ choices[choiceOrder[4] - 1],
2398
+ );
2399
+ cy.get(`label[for=${cesc2("/g2/ci_choice1_input")}]`).should(
2400
+ "have.text",
2401
+ choices[choiceOrder[0] - 1],
2402
+ );
2403
+ cy.get(`label[for=${cesc2("/g2/ci_choice2_input")}]`).should(
2404
+ "have.text",
2405
+ choices[choiceOrder[1] - 1],
2406
+ );
2407
+ cy.get(`label[for=${cesc2("/g2/ci_choice3_input")}]`).should(
2408
+ "have.text",
2409
+ choices[choiceOrder[2] - 1],
2410
+ );
2411
+ cy.get(`label[for=${cesc2("/g2/ci_choice4_input")}]`).should(
2412
+ "have.text",
2413
+ choices[choiceOrder[3] - 1],
2414
+ );
2415
+ cy.get(`label[for=${cesc2("/g2/ci_choice5_input")}]`).should(
2416
+ "have.text",
2417
+ choices[choiceOrder[4] - 1],
2418
+ );
2419
+ });
2420
+ });
2421
+
2422
+ it("math choices", () => {
2423
+ cy.window().then(async (win) => {
2424
+ win.postMessage(
2425
+ {
2426
+ doenetML: `
2427
+ <text>a</text>
2428
+ <choiceinput>
2429
+ <choice><math>x + x</math></choice>
2430
+ <choice><m>y+y</m></choice>
2431
+ <choice><me>z+z</me></choice>
2432
+ <choice><men>u+u</men></choice>
2433
+ </choiceinput>
2434
+
2435
+ <p>Selected value: <copy prop='selectedvalue' source="_choiceinput1" /></p>
2436
+ <p>Selected index: <copy prop='selectedindex' source="_choiceinput1" /></p>
2437
+ <p>Selected value: $_choiceinput1</p>
2438
+ <p>Selected value simplified: $_choiceinput1{simplify}</p>
2439
+
2440
+ `,
2441
+ },
2442
+ "*",
2443
+ );
2444
+ });
2445
+
2446
+ cy.get(cesc("#\\/_text1")).should("have.text", "a"); // to wait for page to load
2447
+
2448
+ cy.get(cesc("#\\/_p1")).should("have.text", "Selected value: ");
2449
+ cy.get(cesc("#\\/_p2")).should("have.text", "Selected index: ");
2450
+ cy.get(cesc("#\\/_p3")).should("have.text", "Selected value: ");
2451
+ cy.get(cesc("#\\/_p4")).should("have.text", "Selected value simplified: ");
2452
+
2453
+ cy.get(cesc(`#\\/_choiceinput1_choice1_input`)).click({ force: true });
2454
+
2455
+ cy.get(cesc("#\\/_p1") + " .mjx-mrow").should("contain.text", "x+x");
2456
+ cy.get(cesc("#\\/_p1") + " .mjx-mrow")
2457
+ .eq(0)
2458
+ .should("have.text", "x+x");
2459
+ cy.get(cesc("#\\/_p2")).should("have.text", "Selected index: 1");
2460
+ cy.get(cesc("#\\/_p3") + " .mjx-mrow")
2461
+ .eq(0)
2462
+ .should("have.text", "x+x");
2463
+ cy.get(cesc("#\\/_p4") + " .mjx-mrow")
2464
+ .eq(0)
2465
+ .should("have.text", "2x");
2466
+
2467
+ cy.get(cesc(`#\\/_choiceinput1_choice2_input`)).click({ force: true });
2468
+
2469
+ cy.get(cesc("#\\/_p1") + " .mjx-mrow").should("contain.text", "y+y");
2470
+ cy.get(cesc("#\\/_p1") + " .mjx-mrow")
2471
+ .eq(0)
2472
+ .should("have.text", "y+y");
2473
+ cy.get(cesc("#\\/_p2")).should("have.text", "Selected index: 2");
2474
+ cy.get(cesc("#\\/_p3") + " .mjx-mrow")
2475
+ .eq(0)
2476
+ .should("have.text", "y+y");
2477
+ cy.get(cesc("#\\/_p4") + " .mjx-mrow")
2478
+ .eq(0)
2479
+ .should("have.text", "2y");
2480
+
2481
+ cy.get(cesc(`#\\/_choiceinput1_choice3_input`)).click({ force: true });
2482
+
2483
+ cy.get(cesc("#\\/_p1") + " .mjx-mrow").should("contain.text", "z+z");
2484
+ cy.get(cesc("#\\/_p1") + " .mjx-mrow")
2485
+ .eq(0)
2486
+ .should("have.text", "z+z");
2487
+ cy.get(cesc("#\\/_p2")).should("have.text", "Selected index: 3");
2488
+ cy.get(cesc("#\\/_p3") + " .mjx-mrow")
2489
+ .eq(0)
2490
+ .should("have.text", "z+z");
2491
+ cy.get(cesc("#\\/_p4") + " .mjx-mrow")
2492
+ .eq(0)
2493
+ .should("have.text", "2z");
2494
+
2495
+ cy.get(cesc(`#\\/_choiceinput1_choice4_input`)).click({ force: true });
2496
+
2497
+ cy.get(cesc("#\\/_p1") + " .mjx-mrow").should("contain.text", "u+u");
2498
+ cy.get(cesc("#\\/_p1") + " .mjx-mrow")
2499
+ .eq(0)
2500
+ .should("have.text", "u+u");
2501
+ cy.get(cesc("#\\/_p2")).should("have.text", "Selected index: 4");
2502
+ cy.get(cesc("#\\/_p3") + " .mjx-mrow")
2503
+ .eq(0)
2504
+ .should("have.text", "u+u");
2505
+ cy.get(cesc("#\\/_p4") + " .mjx-mrow")
2506
+ .eq(0)
2507
+ .should("have.text", "2u");
2508
+ });
2509
+
2510
+ it("consistent order for n elements for given variant", () => {
2511
+ cy.window().then(async (win) => {
2512
+ win.postMessage(
2513
+ {
2514
+ doenetML: `
2515
+ <p>m: <mathinput prefill="1" name="m" /> <number name="m2" copySource="m" /></p>
2516
+ <p>n: <mathinput prefill="6" name="n" /> <number name="n2" copySource="n" /></p>
2517
+ <choiceinput name="ci" shuffleOrder>
2518
+ <map>
2519
+ <template>
2520
+ <choice>$v</choice>
2521
+ </template>
2522
+ <sources alias="v">
2523
+ <sequence from="$m" to="$n" />
2524
+ </sources>
2525
+ </map>
2526
+ </choiceinput>
2527
+ `,
2528
+ requestedVariantIndex: 1,
2529
+ },
2530
+ "*",
2531
+ );
2532
+ });
2533
+
2534
+ cy.get(cesc("#\\/n2")).should("have.text", "6");
2535
+
2536
+ let orders = {};
2537
+
2538
+ cy.window().then(async (win) => {
2539
+ let m = 1,
2540
+ n = 6;
2541
+
2542
+ let stateVariables = await win.returnAllStateVariables1();
2543
+ let choiceOrder = stateVariables["/ci"].stateValues.choiceOrder;
2544
+
2545
+ expect([...choiceOrder].sort((a, b) => a - b)).eqls(
2546
+ [...Array(n - m + 1).keys()].map((x) => x + m),
2547
+ );
2548
+
2549
+ orders[[m, n]] = choiceOrder;
2550
+ });
2551
+
2552
+ cy.log("switch n to 8");
2553
+
2554
+ cy.get(cesc("#\\/n") + " textarea").type("{end}{backspace}8{enter}", {
2555
+ force: true,
2556
+ });
2557
+ cy.get(cesc("#\\/n2")).should("have.text", "8");
2558
+
2559
+ cy.window().then(async (win) => {
2560
+ let m = 1,
2561
+ n = 8;
2562
+
2563
+ let stateVariables = await win.returnAllStateVariables1();
2564
+ let choiceOrder = stateVariables["/ci"].stateValues.choiceOrder;
2565
+
2566
+ expect([...choiceOrder].sort((a, b) => a - b)).eqls(
2567
+ [...Array(n - m + 1).keys()].map((x) => x + m),
2568
+ );
2569
+
2570
+ orders[[m, n]] = choiceOrder;
2571
+ });
2572
+
2573
+ cy.log("get another list of length 6 by setting m to 3");
2574
+
2575
+ cy.get(cesc("#\\/m") + " textarea").type("{end}{backspace}3{enter}", {
2576
+ force: true,
2577
+ });
2578
+ cy.get(cesc("#\\/m2")).should("have.text", "3");
2579
+
2580
+ cy.window().then(async (win) => {
2581
+ let m = 3,
2582
+ n = 8;
2583
+
2584
+ let stateVariables = await win.returnAllStateVariables1();
2585
+ let choiceOrder = stateVariables["/ci"].stateValues.choiceOrder;
2586
+
2587
+ expect(choiceOrder).eqls(orders[[1, 6]]);
2588
+
2589
+ orders[[m, n]] = choiceOrder;
2590
+ });
2591
+
2592
+ cy.log("get another list of length 8 by setting n to 10");
2593
+
2594
+ cy.get(cesc("#\\/n") + " textarea").type("{end}{backspace}10{enter}", {
2595
+ force: true,
2596
+ });
2597
+ cy.get(cesc("#\\/n2")).should("have.text", "10");
2598
+
2599
+ cy.window().then(async (win) => {
2600
+ let m = 3,
2601
+ n = 10;
2602
+
2603
+ let stateVariables = await win.returnAllStateVariables1();
2604
+ let choiceOrder = stateVariables["/ci"].stateValues.choiceOrder;
2605
+
2606
+ expect(choiceOrder).eqls(orders[[1, 8]]);
2607
+
2608
+ orders[[m, n]] = choiceOrder;
2609
+ });
2610
+
2611
+ cy.log("values change with another variant");
2612
+
2613
+ cy.window().then(async (win) => {
2614
+ win.postMessage(
2615
+ {
2616
+ doenetML: `
2617
+ <p>m: <mathinput prefill="1" name="m" /> <number name="m2" copySource="m" /></p>
2618
+ <p>n: <mathinput prefill="6" name="n" /> <number name="n2" copySource="n" /></p>
2619
+ <choiceinput name="ci" shuffleOrder>
2620
+ <map>
2621
+ <template>
2622
+ <choice>$v</choice>
2623
+ </template>
2624
+ <sources alias="v">
2625
+ <sequence from="$m" to="$n" />
2626
+ </sources>
2627
+ </map>
2628
+ </choiceinput>
2629
+ `,
2630
+ requestedVariantIndex: 2,
2631
+ },
2632
+ "*",
2633
+ );
2634
+ });
2635
+
2636
+ cy.get(cesc("#\\/m2")).should("have.text", "1");
2637
+ cy.get(cesc("#\\/n2")).should("have.text", "6");
2638
+
2639
+ cy.window().then(async (win) => {
2640
+ let m = 1,
2641
+ n = 6;
2642
+
2643
+ let stateVariables = await win.returnAllStateVariables1();
2644
+ let choiceOrder = stateVariables["/ci"].stateValues.choiceOrder;
2645
+
2646
+ expect(choiceOrder).not.eqls(orders[[m, n]]);
2647
+
2648
+ expect([...choiceOrder].sort((a, b) => a - b)).eqls(
2649
+ [...Array(n - m + 1).keys()].map((x) => x + m),
2650
+ );
2651
+
2652
+ orders[[m, n]] = choiceOrder;
2653
+ });
2654
+
2655
+ cy.log("switch n to 8");
2656
+
2657
+ cy.get(cesc("#\\/n") + " textarea").type("{end}{backspace}8{enter}", {
2658
+ force: true,
2659
+ });
2660
+ cy.get(cesc("#\\/n2")).should("have.text", "8");
2661
+
2662
+ cy.window().then(async (win) => {
2663
+ let m = 1,
2664
+ n = 8;
2665
+
2666
+ let stateVariables = await win.returnAllStateVariables1();
2667
+ let choiceOrder = stateVariables["/ci"].stateValues.choiceOrder;
2668
+
2669
+ expect(choiceOrder).not.eqls(orders[[m, n]]);
2670
+
2671
+ expect([...choiceOrder].sort((a, b) => a - b)).eqls(
2672
+ [...Array(n - m + 1).keys()].map((x) => x + m),
2673
+ );
2674
+
2675
+ orders[[m, n]] = choiceOrder;
2676
+ });
2677
+
2678
+ cy.log("get another list of length 6 by setting m to 3");
2679
+
2680
+ cy.get(cesc("#\\/m") + " textarea").type("{end}{backspace}3{enter}", {
2681
+ force: true,
2682
+ });
2683
+ cy.get(cesc("#\\/m2")).should("have.text", "3");
2684
+
2685
+ cy.window().then(async (win) => {
2686
+ let m = 3,
2687
+ n = 8;
2688
+
2689
+ let stateVariables = await win.returnAllStateVariables1();
2690
+ let choiceOrder = stateVariables["/ci"].stateValues.choiceOrder;
2691
+
2692
+ expect(choiceOrder).eqls(orders[[1, 6]]);
2693
+
2694
+ orders[[m, n]] = choiceOrder;
2695
+ });
2696
+
2697
+ cy.log("get another list of length 8 by setting n to 10");
2698
+
2699
+ cy.get(cesc("#\\/n") + " textarea").type("{end}{backspace}10{enter}", {
2700
+ force: true,
2701
+ });
2702
+ cy.get(cesc("#\\/n2")).should("have.text", "10");
2703
+
2704
+ cy.window().then(async (win) => {
2705
+ let m = 3,
2706
+ n = 10;
2707
+
2708
+ let stateVariables = await win.returnAllStateVariables1();
2709
+ let choiceOrder = stateVariables["/ci"].stateValues.choiceOrder;
2710
+
2711
+ expect(choiceOrder).eqls(orders[[1, 8]]);
2712
+
2713
+ orders[[m, n]] = choiceOrder;
2714
+ });
2715
+ });
2716
+
2717
+ it("shuffle all but last choice", () => {
2718
+ cy.window().then(async (win) => {
2719
+ win.postMessage(
2720
+ {
2721
+ doenetML: `
2722
+ <text>a</text>
2723
+ <choiceinput>
2724
+ <shuffle>
2725
+ <choice>cat</choice>
2726
+ <choice>dog</choice>
2727
+ <choice>monkey</choice>
2728
+ </shuffle>
2729
+ <choice>none of the above</choice>
2730
+ </choiceinput>
2731
+
2732
+ <p>Selected value: <copy prop='selectedvalue' target="_choiceinput1" /></p>
2733
+ <p>Selected index: <copy prop='selectedindex' target="_choiceinput1" /></p>
2734
+
2735
+ <p name="pCat">Selected cat: $_choice1.selected</p>
2736
+ <p name="pDog">Selected dog: $_choice2.selected</p>
2737
+ <p name="pMonkey">Selected monkey: $_choice3.selected</p>
2738
+ <p name="pNone">Selected none of the above: $_choice4.selected</p>
2739
+ `,
2740
+ requestedVariantIndex: 4,
2741
+ },
2742
+ "*",
2743
+ );
2744
+ });
2745
+
2746
+ cy.get(cesc("#\\/_text1")).should("have.text", "a"); // to wait for page to load
2747
+
2748
+ let originalChoices = ["cat", "dog", "monkey", "none of the above"];
2749
+ cy.get(cesc("#\\/_p1")).should("have.text", "Selected value: ");
2750
+ cy.get(cesc("#\\/_p2")).should("have.text", "Selected index: ");
2751
+ cy.get(cesc("#\\/pCat")).should("have.text", "Selected cat: false");
2752
+ cy.get(cesc("#\\/pDog")).should("have.text", "Selected dog: false");
2753
+ cy.get(cesc("#\\/pMonkey")).should("have.text", "Selected monkey: false");
2754
+ cy.get(cesc("#\\/pNone")).should(
2755
+ "have.text",
2756
+ "Selected none of the above: false",
2757
+ );
2758
+
2759
+ let choices, choiceOrder;
2760
+ cy.window().then(async (win) => {
2761
+ let stateVariables = await win.returnAllStateVariables1();
2762
+ choices = [...stateVariables["/_choiceinput1"].stateValues.choiceTexts];
2763
+ choiceOrder = [
2764
+ ...stateVariables["/_shuffle1"].stateValues.componentOrder,
2765
+ 4,
2766
+ ];
2767
+ expect(choices.length).eq(4);
2768
+ expect(originalChoices.includes(choices[0])).eq(true);
2769
+ expect(originalChoices.includes(choices[1])).eq(true);
2770
+ expect(originalChoices.includes(choices[2])).eq(true);
2771
+ expect(originalChoices.includes(choices[3])).eq(true);
2772
+ expect(originalChoices[3]).eq(choices[3]);
2773
+ expect(choices[1]).not.eq(choices[0]);
2774
+ expect(choices[2]).not.eq(choices[0]);
2775
+ expect(choices[2]).not.eq(choices[1]);
2776
+ expect(choices[3]).not.eq(choices[0]);
2777
+ expect(choices[3]).not.eq(choices[1]);
2778
+ expect(choices[3]).not.eq(choices[2]);
2779
+ expect(stateVariables["/_choiceinput1"].stateValues.selectedValues).eqls(
2780
+ [],
2781
+ );
2782
+ expect(stateVariables["/_choiceinput1"].stateValues.selectedIndices).eqls(
2783
+ [],
2784
+ );
2785
+ expect(stateVariables["/_choiceinput1"].stateValues.inline).eq(false);
2786
+ expect(stateVariables["/_choiceinput1"].stateValues.shuffleOrder).eq(
2787
+ false,
2788
+ );
2789
+ expect(stateVariables["/_choice1"].stateValues.selected).eq(false);
2790
+ expect(stateVariables["/_choice2"].stateValues.selected).eq(false);
2791
+ expect(stateVariables["/_choice3"].stateValues.selected).eq(false);
2792
+ expect(stateVariables["/_choice4"].stateValues.selected).eq(false);
2793
+ });
2794
+
2795
+ cy.log("select options in order");
2796
+
2797
+ for (let i = 0; i < 4; i++) {
2798
+ cy.get(cesc(`#\\/_choiceinput1_choice${i + 1}_input`))
2799
+ .click()
2800
+ .then(() => {
2801
+ // make this asynchronous so that choices is populated before line is executed
2802
+ cy.get(cesc("#\\/_p1")).should(
2803
+ "have.text",
2804
+ "Selected value: " + choices[i],
2805
+ );
2806
+ cy.get(cesc("#\\/_p2")).should(
2807
+ "have.text",
2808
+ "Selected index: " + (i + 1),
2809
+ );
2810
+
2811
+ cy.get(cesc("#\\/pCat")).should(
2812
+ "have.text",
2813
+ `Selected cat: ${choiceOrder[i] === 1}`,
2814
+ );
2815
+ cy.get(cesc("#\\/pDog")).should(
2816
+ "have.text",
2817
+ `Selected dog: ${choiceOrder[i] === 2}`,
2818
+ );
2819
+ cy.get(cesc("#\\/pMonkey")).should(
2820
+ "have.text",
2821
+ `Selected monkey: ${choiceOrder[i] === 3}`,
2822
+ );
2823
+ cy.get(cesc("#\\/pNone")).should(
2824
+ "have.text",
2825
+ `Selected none of the above: ${choiceOrder[i] === 4}`,
2826
+ );
2827
+ });
2828
+
2829
+ cy.window().then(async (win) => {
2830
+ let stateVariables = await win.returnAllStateVariables1();
2831
+ expect(
2832
+ stateVariables["/_choiceinput1"].stateValues.selectedValues,
2833
+ ).eqls([choices[i]]);
2834
+ expect(
2835
+ stateVariables["/_choiceinput1"].stateValues.selectedIndices,
2836
+ ).eqls([i + 1]);
2837
+ expect(stateVariables["/_choice1"].stateValues.selected).eq(
2838
+ choiceOrder[i] === 1,
2839
+ );
2840
+ expect(stateVariables["/_choice2"].stateValues.selected).eq(
2841
+ choiceOrder[i] === 2,
2842
+ );
2843
+ expect(stateVariables["/_choice3"].stateValues.selected).eq(
2844
+ choiceOrder[i] === 3,
2845
+ );
2846
+ expect(stateVariables["/_choice4"].stateValues.selected).eq(
2847
+ choiceOrder[i] === 4,
2848
+ );
2849
+ });
2850
+ }
2851
+ });
2852
+
2853
+ it("sorted choices", () => {
2854
+ cy.window().then(async (win) => {
2855
+ win.postMessage(
2856
+ {
2857
+ doenetML: `
2858
+ <text>a</text>
2859
+ <choiceinput>
2860
+ <sort sortByProp="text">
2861
+ <choice>mouse</choice>
2862
+ <choice>dog</choice>
2863
+ <choice>cat</choice>
2864
+ <choice>monkey</choice>
2865
+ </sort>
2866
+ </choiceinput>
2867
+
2868
+ <p>Selected value: <copy prop='selectedvalue' target="_choiceinput1" /></p>
2869
+ <p>Selected index: <copy prop='selectedindex' target="_choiceinput1" /></p>
2870
+
2871
+ <p name="pMouse">Selected mouse: $_choice1.selected</p>
2872
+ <p name="pDog">Selected dog: $_choice2.selected</p>
2873
+ <p name="pCat">Selected cat: $_choice3.selected</p>
2874
+ <p name="pMonkey">Selected monkey: $_choice4.selected</p>
2875
+ `,
2876
+ },
2877
+ "*",
2878
+ );
2879
+ });
2880
+
2881
+ cy.get(cesc("#\\/_text1")).should("have.text", "a"); // to wait for page to load
2882
+
2883
+ let originalChoices = ["mouse", "dog", "cat", "monkey"];
2884
+ cy.get(cesc("#\\/_p1")).should("have.text", "Selected value: ");
2885
+ cy.get(cesc("#\\/_p2")).should("have.text", "Selected index: ");
2886
+ cy.get(cesc("#\\/pMouse")).should("have.text", "Selected mouse: false");
2887
+ cy.get(cesc("#\\/pDog")).should("have.text", "Selected dog: false");
2888
+ cy.get(cesc("#\\/pCat")).should("have.text", "Selected cat: false");
2889
+ cy.get(cesc("#\\/pMonkey")).should("have.text", "Selected monkey: false");
2890
+
2891
+ let choices, choiceOrder;
2892
+ cy.window().then(async (win) => {
2893
+ let stateVariables = await win.returnAllStateVariables1();
2894
+ choices = [...stateVariables["/_choiceinput1"].stateValues.choiceTexts];
2895
+ choiceOrder = [3, 2, 4, 1];
2896
+ expect(choices.length).eq(4);
2897
+ expect(originalChoices.includes(choices[0])).eq(true);
2898
+ expect(originalChoices.includes(choices[1])).eq(true);
2899
+ expect(originalChoices.includes(choices[2])).eq(true);
2900
+ expect(originalChoices.includes(choices[3])).eq(true);
2901
+ expect(choices[1]).not.eq(choices[0]);
2902
+ expect(choices[2]).not.eq(choices[0]);
2903
+ expect(choices[2]).not.eq(choices[1]);
2904
+ expect(choices[3]).not.eq(choices[0]);
2905
+ expect(choices[3]).not.eq(choices[1]);
2906
+ expect(choices[3]).not.eq(choices[2]);
2907
+ expect(stateVariables["/_choiceinput1"].stateValues.selectedValues).eqls(
2908
+ [],
2909
+ );
2910
+ expect(stateVariables["/_choiceinput1"].stateValues.selectedIndices).eqls(
2911
+ [],
2912
+ );
2913
+ expect(stateVariables["/_choiceinput1"].stateValues.inline).eq(false);
2914
+ expect(stateVariables["/_choiceinput1"].stateValues.shuffleOrder).eq(
2915
+ false,
2916
+ );
2917
+ expect(stateVariables["/_choice1"].stateValues.selected).eq(false);
2918
+ expect(stateVariables["/_choice2"].stateValues.selected).eq(false);
2919
+ expect(stateVariables["/_choice3"].stateValues.selected).eq(false);
2920
+ expect(stateVariables["/_choice4"].stateValues.selected).eq(false);
2921
+ });
2922
+
2923
+ cy.log("select options in order");
2924
+
2925
+ for (let i = 0; i < 4; i++) {
2926
+ cy.get(cesc(`#\\/_choiceinput1_choice${i + 1}_input`))
2927
+ .click()
2928
+ .then(() => {
2929
+ // make this asynchronous so that choices is populated before line is executed
2930
+ cy.get(cesc("#\\/_p1")).should(
2931
+ "have.text",
2932
+ "Selected value: " + choices[i],
2933
+ );
2934
+ cy.get(cesc("#\\/_p2")).should(
2935
+ "have.text",
2936
+ "Selected index: " + (i + 1),
2937
+ );
2938
+
2939
+ cy.get(cesc("#\\/pMouse")).should(
2940
+ "have.text",
2941
+ `Selected mouse: ${choiceOrder[i] === 1}`,
2942
+ );
2943
+ cy.get(cesc("#\\/pDog")).should(
2944
+ "have.text",
2945
+ `Selected dog: ${choiceOrder[i] === 2}`,
2946
+ );
2947
+ cy.get(cesc("#\\/pCat")).should(
2948
+ "have.text",
2949
+ `Selected cat: ${choiceOrder[i] === 3}`,
2950
+ );
2951
+ cy.get(cesc("#\\/pMonkey")).should(
2952
+ "have.text",
2953
+ `Selected monkey: ${choiceOrder[i] === 4}`,
2954
+ );
2955
+ });
2956
+
2957
+ cy.window().then(async (win) => {
2958
+ let stateVariables = await win.returnAllStateVariables1();
2959
+ expect(
2960
+ stateVariables["/_choiceinput1"].stateValues.selectedValues,
2961
+ ).eqls([choices[i]]);
2962
+ expect(
2963
+ stateVariables["/_choiceinput1"].stateValues.selectedIndices,
2964
+ ).eqls([i + 1]);
2965
+ expect(stateVariables["/_choice1"].stateValues.selected).eq(
2966
+ choiceOrder[i] === 1,
2967
+ );
2968
+ expect(stateVariables["/_choice2"].stateValues.selected).eq(
2969
+ choiceOrder[i] === 2,
2970
+ );
2971
+ expect(stateVariables["/_choice3"].stateValues.selected).eq(
2972
+ choiceOrder[i] === 3,
2973
+ );
2974
+ expect(stateVariables["/_choice4"].stateValues.selected).eq(
2975
+ choiceOrder[i] === 4,
2976
+ );
2977
+ });
2978
+ }
2979
+ });
2980
+
2981
+ it("copy choices", () => {
2982
+ cy.window().then(async (win) => {
2983
+ win.postMessage(
2984
+ {
2985
+ doenetML: `
2986
+ <setup>
2987
+ <choice name="cat0">cat</choice>
2988
+ </setup>
2989
+ <choiceinput>
2990
+ <choice copySource="cat0" name="cat1" />
2991
+ <choice name="dog1">dog</choice>
2992
+ <choice name="monkey1">monkey</choice>
2993
+ </choiceinput>
2994
+
2995
+ <choiceinput>
2996
+ <choice copySource="cat1" name="cat2" />
2997
+ <choice copySource="dog1" name="dog2" />
2998
+ <choice name="monkey2">monkey</choice>
2999
+ </choiceinput>
3000
+
3001
+ <p name="pSv1">Selected value 1: <copy prop='selectedvalue' target="_choiceinput1" /></p>
3002
+ <p name="pSi1">Selected index 1: <copy prop='selectedindex' target="_choiceinput1" /></p>
3003
+
3004
+ <p name="pSv2">Selected value 2: <copy prop='selectedvalue' target="_choiceinput2" /></p>
3005
+ <p name="pSi2">Selected index 2: <copy prop='selectedindex' target="_choiceinput2" /></p>
3006
+
3007
+ <p name="pCat0">Selected cat0: $cat0.selected</p>
3008
+
3009
+ <p name="pCat1">Selected cat1: $cat1.selected</p>
3010
+ <p name="pDog1">Selected dog1: $dog1.selected</p>
3011
+ <p name="pMonkey1">Selected monkey1: $monkey1.selected</p>
3012
+
3013
+ <p name="pCat2">Selected cat2: $cat2.selected</p>
3014
+ <p name="pDog2">Selected dog2: $dog2.selected</p>
3015
+ <p name="pMonkey2">Selected monkey2: $monkey2.selected</p>
3016
+
3017
+ `,
3018
+ },
3019
+ "*",
3020
+ );
3021
+ });
3022
+
3023
+ let choices = ["cat", "dog", "monkey"];
3024
+ cy.get(cesc("#\\/pSv1")).should("have.text", "Selected value 1: ");
3025
+ cy.get(cesc("#\\/pSi1")).should("have.text", "Selected index 1: ");
3026
+ cy.get(cesc("#\\/pSv2")).should("have.text", "Selected value 2: ");
3027
+ cy.get(cesc("#\\/pSi2")).should("have.text", "Selected index 2: ");
3028
+
3029
+ cy.get(cesc("#\\/pCat1")).should("have.text", "Selected cat1: false");
3030
+ cy.get(cesc("#\\/pDog1")).should("have.text", "Selected dog1: false");
3031
+ cy.get(cesc("#\\/pMonkey1")).should("have.text", "Selected monkey1: false");
3032
+ cy.get(cesc("#\\/pCat2")).should("have.text", "Selected cat2: false");
3033
+ cy.get(cesc("#\\/pDog2")).should("have.text", "Selected dog2: false");
3034
+ cy.get(cesc("#\\/pMonkey2")).should("have.text", "Selected monkey2: false");
3035
+
3036
+ cy.log("select options 1 in order");
3037
+
3038
+ for (let i = 0; i < 3; i++) {
3039
+ cy.get(cesc(`#\\/_choiceinput1_choice${i + 1}_input`)).click();
3040
+
3041
+ cy.get(cesc("#\\/pSv1")).should(
3042
+ "have.text",
3043
+ "Selected value 1: " + choices[i],
3044
+ );
3045
+ cy.get(cesc("#\\/pSi1")).should(
3046
+ "have.text",
3047
+ "Selected index 1: " + (i + 1),
3048
+ );
3049
+
3050
+ cy.get(cesc("#\\/pSv2")).should("have.text", "Selected value 2: ");
3051
+ cy.get(cesc("#\\/pSi2")).should("have.text", "Selected index 2: ");
3052
+
3053
+ cy.get(cesc("#\\/pCat1")).should(
3054
+ "have.text",
3055
+ `Selected cat1: ${i === 0}`,
3056
+ );
3057
+ cy.get(cesc("#\\/pDog1")).should(
3058
+ "have.text",
3059
+ `Selected dog1: ${i === 1}`,
3060
+ );
3061
+ cy.get(cesc("#\\/pMonkey1")).should(
3062
+ "have.text",
3063
+ `Selected monkey1: ${i == 2}`,
3064
+ );
3065
+ cy.get(cesc("#\\/pCat2")).should("have.text", "Selected cat2: false");
3066
+ cy.get(cesc("#\\/pDog2")).should("have.text", "Selected dog2: false");
3067
+ cy.get(cesc("#\\/pMonkey2")).should(
3068
+ "have.text",
3069
+ "Selected monkey2: false",
3070
+ );
3071
+ }
3072
+
3073
+ cy.log("select options 2 in order");
3074
+
3075
+ for (let i = 0; i < 3; i++) {
3076
+ cy.get(cesc(`#\\/_choiceinput2_choice${i + 1}_input`)).click();
3077
+
3078
+ cy.get(cesc("#\\/pSv2")).should(
3079
+ "have.text",
3080
+ "Selected value 2: " + choices[i],
3081
+ );
3082
+ cy.get(cesc("#\\/pSi2")).should(
3083
+ "have.text",
3084
+ "Selected index 2: " + (i + 1),
3085
+ );
3086
+
3087
+ cy.get(cesc("#\\/pSv1")).should("have.text", "Selected value 1: monkey");
3088
+ cy.get(cesc("#\\/pSi1")).should("have.text", "Selected index 1: 3");
3089
+
3090
+ cy.get(cesc("#\\/pCat1")).should("have.text", "Selected cat1: false");
3091
+ cy.get(cesc("#\\/pDog1")).should("have.text", "Selected dog1: false");
3092
+ cy.get(cesc("#\\/pMonkey1")).should(
3093
+ "have.text",
3094
+ "Selected monkey1: true",
3095
+ );
3096
+
3097
+ cy.get(cesc("#\\/pCat2")).should(
3098
+ "have.text",
3099
+ `Selected cat2: ${i === 0}`,
3100
+ );
3101
+ cy.get(cesc("#\\/pDog2")).should(
3102
+ "have.text",
3103
+ `Selected dog2: ${i === 1}`,
3104
+ );
3105
+ cy.get(cesc("#\\/pMonkey2")).should(
3106
+ "have.text",
3107
+ `Selected monkey2: ${i == 2}`,
3108
+ );
3109
+ }
3110
+ });
3111
+
3112
+ it("valueChanged", () => {
3113
+ let doenetML = `
3114
+ <p><choiceInput name="ci1" inline><choice>Yes</choice><choice>No</choice></choiceInput>
3115
+ <text copySource="ci1" name="ci1a" /> <boolean copysource="ci1.valueChanged" name="ci1changed" /></p>
3116
+ <p><choiceInput name="ci2" preselectChoice="2"><choice>Yes</choice><choice>No</choice></choiceInput>
3117
+ <text copySource="ci2" name="ci2a" /> <boolean copysource="ci2.valueChanged" name="ci2changed" /></p>
3118
+ <p><choiceInput name="ci3" bindValueTo="$ci1.values" ><choice>Yes</choice><choice>No</choice></choiceInput>
3119
+ <text copySource="ci3" name="ci3a" /> <boolean copysource="ci3.valueChanged" name="ci3changed" /></p>
3120
+ <p><choiceInput name="ci4" inline bindValueTo="$ci2.values"><choice>Yes</choice><choice>No</choice></choiceInput
3121
+ <text copySource="ci4" name="ci4a" /> <boolean copysource="ci4.valueChanged" name="ci4changed" /></p>
3122
+
3123
+ `;
3124
+
3125
+ cy.window().then(async (win) => {
3126
+ win.postMessage(
3127
+ {
3128
+ doenetML,
3129
+ },
3130
+ "*",
3131
+ );
3132
+ });
3133
+
3134
+ cy.get(cesc2("#/ci1changed")).should("have.text", "false");
3135
+ cy.get(cesc2("#/ci2changed")).should("have.text", "false");
3136
+ cy.get(cesc2("#/ci3changed")).should("have.text", "false");
3137
+ cy.get(cesc2("#/ci4changed")).should("have.text", "false");
3138
+
3139
+ cy.get(cesc2("#/ci1a")).should("have.text", "");
3140
+ cy.get(cesc2("#/ci2a")).should("have.text", "No");
3141
+ cy.get(cesc2("#/ci3a")).should("have.text", "");
3142
+ cy.get(cesc2("#/ci4a")).should("have.text", "No");
3143
+
3144
+ cy.log("selecting from first and third marks only them as changed");
3145
+
3146
+ cy.get(cesc2("#/ci1")).select("Yes");
3147
+ cy.get(cesc2("#/ci2") + "_choice1_input").click();
3148
+
3149
+ cy.get(cesc2("#/ci1a")).should("have.text", "Yes");
3150
+ cy.get(cesc2("#/ci2a")).should("have.text", "Yes");
3151
+ cy.get(cesc2("#/ci3a")).should("have.text", "Yes");
3152
+ cy.get(cesc2("#/ci4a")).should("have.text", "Yes");
3153
+
3154
+ cy.get(cesc2("#/ci1changed")).should("have.text", "true");
3155
+ cy.get(cesc2("#/ci2changed")).should("have.text", "true");
3156
+ cy.get(cesc2("#/ci3changed")).should("have.text", "false");
3157
+ cy.get(cesc2("#/ci4changed")).should("have.text", "false");
3158
+
3159
+ // TODO: cannot change choiceInputs from bound values, so can't change third and fourth
3160
+ // Should we add this features?
3161
+ });
3162
+
3163
+ it("label", () => {
3164
+ let doenetML = `
3165
+ <p><choiceInput name="ci1" inline><label>Select an option</label><choice>Yes</choice><choice>No</choice></choiceInput>
3166
+ <copy copySource="ci1" assignNames="ci1a" /> </p>
3167
+ <p><choiceInput name="ci2"><choice>Yes</choice><choice>No</choice><label>Select another option</label></choiceInput>
3168
+ <copy copySource="ci2" assignNames="ci2a" /> </p>
3169
+
3170
+ <p><updateValue target="_label1.hide" newValue="!$_label1.hide" type="boolean" name="toggleLabels"><label>Toggle labels</label></updateValue>
3171
+ <updateValue triggerWith="toggleLabels" target="_label2.hide" newValue="!$_label2.hide" type="boolean" /></p>
3172
+ `;
3173
+
3174
+ cy.window().then(async (win) => {
3175
+ win.postMessage(
3176
+ {
3177
+ doenetML,
3178
+ },
3179
+ "*",
3180
+ );
3181
+ });
3182
+
3183
+ cy.get(cesc2("#/ci1-label")).should("contain.text", "Select an option");
3184
+ cy.get(cesc2("#/ci2-label")).should(
3185
+ "contain.text",
3186
+ "Select another option",
3187
+ );
3188
+
3189
+ cy.log("hide labels");
3190
+ cy.get(cesc2("#/toggleLabels")).click();
3191
+ cy.get(cesc2("#/ci1-label")).should("not.contain.text", "Select an option");
3192
+ cy.get(cesc2("#/ci2-label")).should(
3193
+ "not.contain.text",
3194
+ "Select another option",
3195
+ );
3196
+
3197
+ cy.log("show labels again");
3198
+ cy.get(cesc2("#/toggleLabels")).click();
3199
+ cy.get(cesc2("#/ci1-label")).should("contain.text", "Select an option");
3200
+ cy.get(cesc2("#/ci2-label")).should(
3201
+ "contain.text",
3202
+ "Select another option",
3203
+ );
3204
+ });
3205
+ });