@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,3495 @@
1
+ import { cesc, cesc2 } from "../../../../src/utils/url";
2
+
3
+ describe("Conditional Content Tag Tests", function () {
4
+ beforeEach(() => {
5
+ cy.clearIndexedDB();
6
+ cy.visit("/src/Tools/cypressTest/");
7
+ });
8
+
9
+ // tests without cases or else
10
+
11
+ it("inline content containing sign of number", () => {
12
+ cy.window().then(async (win) => {
13
+ win.postMessage(
14
+ {
15
+ doenetML: `
16
+ <text>a</text>
17
+ <mathinput name="n" />
18
+
19
+ <p>You typed
20
+ <conditionalcontent condition="$n > 0">
21
+ a positive number.
22
+ </conditionalcontent>
23
+ <conditionalcontent condition="$n < 0">
24
+ a negative number.
25
+ </conditionalcontent>
26
+ <conditionalcontent condition="$n=0">
27
+ zero.
28
+ </conditionalcontent>
29
+ <conditionalcontent condition="not ($n>0 or $n<0 or $n=0)" >
30
+ something else.
31
+ </conditionalcontent>
32
+ </p>
33
+ `,
34
+ },
35
+ "*",
36
+ );
37
+ });
38
+
39
+ cy.get(cesc("#\\/_text1")).should("have.text", "a");
40
+
41
+ cy.get("p" + cesc("#\\/_p1"))
42
+ .invoke("text")
43
+ .then((text) => {
44
+ expect(text.replace(/\s+/g, " ").trim()).equal(
45
+ "You typed something else.",
46
+ );
47
+ });
48
+
49
+ cy.get(cesc("#\\/n") + " textarea").type("10{enter}", { force: true });
50
+
51
+ cy.get("p" + cesc("#\\/_p1")).should("contain.text", "a positive number.");
52
+ cy.get("p" + cesc("#\\/_p1"))
53
+ .invoke("text")
54
+ .then((text) => {
55
+ expect(text.replace(/\s+/g, " ").trim()).equal(
56
+ "You typed a positive number.",
57
+ );
58
+ });
59
+
60
+ cy.get(cesc("#\\/n") + " textarea").type(
61
+ "{end}{backspace}{backspace}-5/9{enter}",
62
+ { force: true },
63
+ );
64
+ cy.get("p" + cesc("#\\/_p1")).should("contain.text", "a negative number.");
65
+ cy.get("p" + cesc("#\\/_p1"))
66
+ .invoke("text")
67
+ .then((text) => {
68
+ expect(text.replace(/\s+/g, " ").trim()).equal(
69
+ "You typed a negative number.",
70
+ );
71
+ });
72
+
73
+ cy.get(cesc("#\\/n") + " textarea").type(
74
+ "{ctrl+home}{shift+end}{backspace}5-5{enter}",
75
+ { force: true },
76
+ );
77
+ cy.get("p" + cesc("#\\/_p1")).should("contain.text", "zero.");
78
+ cy.get("p" + cesc("#\\/_p1"))
79
+ .invoke("text")
80
+ .then((text) => {
81
+ expect(text.replace(/\s+/g, " ").trim()).equal("You typed zero.");
82
+ });
83
+
84
+ cy.get(cesc("#\\/n") + " textarea").type(
85
+ "{ctrl+home}{shift+end}{backspace}-x{enter}",
86
+ { force: true },
87
+ );
88
+ cy.get("p" + cesc("#\\/_p1")).should("contain.text", "something else.");
89
+ cy.get("p" + cesc("#\\/_p1"))
90
+ .invoke("text")
91
+ .then((text) => {
92
+ expect(text.replace(/\s+/g, " ").trim()).equal(
93
+ "You typed something else.",
94
+ );
95
+ });
96
+ });
97
+
98
+ it("inline content containing sign of number, use XML entities", () => {
99
+ cy.window().then(async (win) => {
100
+ win.postMessage(
101
+ {
102
+ doenetML: `
103
+ <text>a</text>
104
+ <mathinput name="n" />
105
+
106
+ <p>You typed
107
+ <conditionalcontent condition="$n &gt; 0">
108
+ a positive number.
109
+ </conditionalcontent>
110
+ <conditionalcontent condition="$n &lt; 0">
111
+ a negative number.
112
+ </conditionalcontent>
113
+ <conditionalcontent condition="$n=0">
114
+ zero.
115
+ </conditionalcontent>
116
+ <conditionalcontent condition="not ($n&gt;0 or $n&lt;0 or $n=0)" >
117
+ something else.
118
+ </conditionalcontent>
119
+ </p>
120
+ `,
121
+ },
122
+ "*",
123
+ );
124
+ });
125
+
126
+ cy.get(cesc("#\\/_text1")).should("have.text", "a");
127
+
128
+ cy.get("p" + cesc("#\\/_p1"))
129
+ .invoke("text")
130
+ .then((text) => {
131
+ expect(text.replace(/\s+/g, " ").trim()).equal(
132
+ "You typed something else.",
133
+ );
134
+ });
135
+
136
+ cy.get(cesc("#\\/n") + " textarea").type("10{enter}", { force: true });
137
+ cy.get("p" + cesc("#\\/_p1")).should("contain.text", "a positive number.");
138
+ cy.get("p" + cesc("#\\/_p1"))
139
+ .invoke("text")
140
+ .then((text) => {
141
+ expect(text.replace(/\s+/g, " ").trim()).equal(
142
+ "You typed a positive number.",
143
+ );
144
+ });
145
+
146
+ cy.get(cesc("#\\/n") + " textarea").type(
147
+ "{end}{backspace}{backspace}-5/9{enter}",
148
+ { force: true },
149
+ );
150
+ cy.get("p" + cesc("#\\/_p1")).should("contain.text", "a negative number.");
151
+ cy.get("p" + cesc("#\\/_p1"))
152
+ .invoke("text")
153
+ .then((text) => {
154
+ expect(text.replace(/\s+/g, " ").trim()).equal(
155
+ "You typed a negative number.",
156
+ );
157
+ });
158
+
159
+ cy.get(cesc("#\\/n") + " textarea").type(
160
+ "{ctrl+home}{shift+end}{backspace}5-5{enter}",
161
+ { force: true },
162
+ );
163
+ cy.get("p" + cesc("#\\/_p1")).should("contain.text", "zero.");
164
+ cy.get("p" + cesc("#\\/_p1"))
165
+ .invoke("text")
166
+ .then((text) => {
167
+ expect(text.replace(/\s+/g, " ").trim()).equal("You typed zero.");
168
+ });
169
+
170
+ cy.get(cesc("#\\/n") + " textarea").type(
171
+ "{ctrl+home}{shift+end}{backspace}-x{enter}",
172
+ { force: true },
173
+ );
174
+ cy.get("p" + cesc("#\\/_p1")).should("contain.text", "something else.");
175
+ cy.get("p" + cesc("#\\/_p1"))
176
+ .invoke("text")
177
+ .then((text) => {
178
+ expect(text.replace(/\s+/g, " ").trim()).equal(
179
+ "You typed something else.",
180
+ );
181
+ });
182
+ });
183
+
184
+ it("block content containing sign of number", () => {
185
+ cy.window().then(async (win) => {
186
+ win.postMessage(
187
+ {
188
+ doenetML: `
189
+ <text>a</text>
190
+ <mathinput name="n" />
191
+
192
+ <section>
193
+ <conditionalcontent condition="$n>0" >
194
+ <p>You typed a positive number.</p>
195
+ </conditionalcontent>
196
+ </section>
197
+ <section>
198
+ <conditionalcontent condition="$n<0" >
199
+ <p>You typed a negative number.</p>
200
+ </conditionalcontent>
201
+ </section>
202
+ <section>
203
+ <conditionalcontent condition="$n=0" >
204
+ <p>You typed zero.</p>
205
+ </conditionalcontent>
206
+ </section>
207
+ <section>
208
+ <conditionalcontent condition="not ($n>0 or $n<0 or $n=0)" >
209
+ <p>You typed something else.</p>
210
+ </conditionalcontent>
211
+ </section>
212
+ `,
213
+ },
214
+ "*",
215
+ );
216
+ });
217
+
218
+ cy.get(cesc("#\\/_text1")).should("have.text", "a");
219
+
220
+ cy.get(cesc("#\\/_section1") + " p").should("not.exist");
221
+ cy.get(cesc("#\\/_section2") + " p").should("not.exist");
222
+ cy.get(cesc("#\\/_section3") + " p").should("not.exist");
223
+ cy.get(cesc("#\\/_section4") + " p")
224
+ .invoke("text")
225
+ .then((text) => {
226
+ expect(text.replace(/\s+/g, " ").trim()).equal(
227
+ "You typed something else.",
228
+ );
229
+ });
230
+
231
+ cy.get(cesc("#\\/n") + " textarea").type("10{enter}", { force: true });
232
+ cy.get(cesc("#\\/_section1") + " p").should(
233
+ "contain.text",
234
+ "a positive number.",
235
+ );
236
+ cy.get(cesc("#\\/_section1") + " p")
237
+ .invoke("text")
238
+ .then((text) => {
239
+ expect(text.replace(/\s+/g, " ").trim()).equal(
240
+ "You typed a positive number.",
241
+ );
242
+ });
243
+ cy.get(cesc("#\\/_section2") + " p").should("not.exist");
244
+ cy.get(cesc("#\\/_section3") + " p").should("not.exist");
245
+ cy.get(cesc("#\\/_section4") + " p").should("not.exist");
246
+
247
+ cy.get(cesc("#\\/n") + " textarea").type(
248
+ "{end}{backspace}{backspace}-5/9{enter}",
249
+ { force: true },
250
+ );
251
+ cy.get(cesc("#\\/_section2") + " p").should(
252
+ "contain.text",
253
+ "a negative number.",
254
+ );
255
+ cy.get(cesc("#\\/_section1") + " p").should("not.exist");
256
+ cy.get(cesc("#\\/_section2") + " p")
257
+ .invoke("text")
258
+ .then((text) => {
259
+ expect(text.replace(/\s+/g, " ").trim()).equal(
260
+ "You typed a negative number.",
261
+ );
262
+ });
263
+ cy.get(cesc("#\\/_section3") + " p").should("not.exist");
264
+ cy.get(cesc("#\\/_section4") + " p").should("not.exist");
265
+
266
+ cy.get(cesc("#\\/n") + " textarea").type(
267
+ "{ctrl+home}{shift+end}{backspace}5-5{enter}",
268
+ { force: true },
269
+ );
270
+ cy.get(cesc("#\\/_section3") + " p").should("contain.text", "zero.");
271
+ cy.get(cesc("#\\/_section1") + " p").should("not.exist");
272
+ cy.get(cesc("#\\/_section2") + " p").should("not.exist");
273
+ cy.get(cesc("#\\/_section3") + " p")
274
+ .invoke("text")
275
+ .then((text) => {
276
+ expect(text.replace(/\s+/g, " ").trim()).equal("You typed zero.");
277
+ });
278
+ cy.get(cesc("#\\/_section4") + " p").should("not.exist");
279
+
280
+ cy.get(cesc("#\\/n") + " textarea").type(
281
+ "{ctrl+home}{shift+end}{backspace}-x{enter}",
282
+ { force: true },
283
+ );
284
+ cy.get(cesc("#\\/_section4") + " p").should(
285
+ "contain.text",
286
+ "something else.",
287
+ );
288
+ cy.get(cesc("#\\/_section1") + " p").should("not.exist");
289
+ cy.get(cesc("#\\/_section2") + " p").should("not.exist");
290
+ cy.get(cesc("#\\/_section3") + " p").should("not.exist");
291
+ cy.get(cesc("#\\/_section4") + " p")
292
+ .invoke("text")
293
+ .then((text) => {
294
+ expect(text.replace(/\s+/g, " ").trim()).equal(
295
+ "You typed something else.",
296
+ );
297
+ });
298
+ });
299
+
300
+ it("conditional text used as correct answer", () => {
301
+ cy.window().then(async (win) => {
302
+ win.postMessage(
303
+ {
304
+ doenetML: `
305
+ <text>a</text>
306
+ <p>Enter a slope: <mathinput name="m" /></p>
307
+
308
+ <p>If this is the slope at an equilibrium of a discrete dynamical system, the equilibrium is
309
+ <answer>
310
+ <choiceinput inline="true" shuffleOrder><choice>stable</choice><choice>unstable</choice></choiceinput>
311
+ <award><when>
312
+ <copy prop="selectedvalue" target="_choiceinput1" />
313
+ =
314
+ <text>
315
+ <conditionalcontent condition="abs($m) < 1" >
316
+ stable
317
+ </conditionalcontent>
318
+ <conditionalcontent condition="abs($m) > 1" >
319
+ unstable
320
+ </conditionalcontent>
321
+ </text>
322
+ </when></award>
323
+ </answer>
324
+ </p>
325
+ `,
326
+ },
327
+ "*",
328
+ );
329
+ });
330
+
331
+ cy.get(cesc("#\\/_text1")).should("have.text", "a");
332
+
333
+ cy.get(cesc("#\\/_choiceinput1")).select(`stable`);
334
+ cy.get(cesc("#\\/_choiceinput1_submit")).click();
335
+ cy.get(cesc("#\\/_choiceinput1_incorrect")).should("be.visible");
336
+ cy.get(cesc("#\\/_choiceinput1")).select(`unstable`);
337
+ cy.get(cesc("#\\/_choiceinput1_submit")).click();
338
+ cy.get(cesc("#\\/_choiceinput1_incorrect")).should("be.visible");
339
+
340
+ cy.get(cesc("#\\/m") + " textarea").type("3{enter}", { force: true });
341
+ cy.get(cesc("#\\/_choiceinput1")).select(`stable`);
342
+ cy.get(cesc("#\\/_choiceinput1_submit")).click();
343
+ cy.get(cesc("#\\/_choiceinput1_incorrect")).should("be.visible");
344
+ cy.get(cesc("#\\/_choiceinput1")).select(`unstable`);
345
+ cy.get(cesc("#\\/_choiceinput1_submit")).click();
346
+ cy.get(cesc("#\\/_choiceinput1_correct")).should("be.visible");
347
+
348
+ cy.get(cesc("#\\/m") + " textarea").type("{end}{backspace}-0.8{enter}", {
349
+ force: true,
350
+ });
351
+ cy.get(cesc("#\\/_choiceinput1")).select(`stable`);
352
+ cy.get(cesc("#\\/_choiceinput1_submit")).click();
353
+ cy.get(cesc("#\\/_choiceinput1_correct")).should("be.visible");
354
+ cy.get(cesc("#\\/_choiceinput1")).select(`unstable`);
355
+ cy.get(cesc("#\\/_choiceinput1_submit")).click();
356
+ cy.get(cesc("#\\/_choiceinput1_incorrect")).should("be.visible");
357
+
358
+ cy.get(cesc("#\\/m") + " textarea").type(
359
+ "{ctrl+home}{shift+end}{backspace}1/3{enter}",
360
+ { force: true },
361
+ );
362
+ cy.get(cesc("#\\/_choiceinput1")).select(`stable`);
363
+ cy.get(cesc("#\\/_choiceinput1_submit")).click();
364
+ cy.get(cesc("#\\/_choiceinput1_correct")).should("be.visible");
365
+ cy.get(cesc("#\\/_choiceinput1")).select(`unstable`);
366
+ cy.get(cesc("#\\/_choiceinput1_submit")).click();
367
+ cy.get(cesc("#\\/_choiceinput1_incorrect")).should("be.visible");
368
+
369
+ cy.get(cesc("#\\/m") + " textarea").type(
370
+ "{ctrl+home}{shift+end}{backspace}-7/5{enter}",
371
+ { force: true },
372
+ );
373
+ cy.get(cesc("#\\/_choiceinput1")).select(`stable`);
374
+ cy.get(cesc("#\\/_choiceinput1_submit")).click();
375
+ cy.get(cesc("#\\/_choiceinput1_incorrect")).should("be.visible");
376
+ cy.get(cesc("#\\/_choiceinput1")).select(`unstable`);
377
+ cy.get(cesc("#\\/_choiceinput1_submit")).click();
378
+ cy.get(cesc("#\\/_choiceinput1_correct")).should("be.visible");
379
+
380
+ cy.get(cesc("#\\/m") + " textarea").type(
381
+ "{ctrl+home}{shift+end}{backspace}1{enter}",
382
+ { force: true },
383
+ );
384
+ cy.get(cesc("#\\/_choiceinput1")).select(`stable`);
385
+ cy.get(cesc("#\\/_choiceinput1_submit")).click();
386
+ cy.get(cesc("#\\/_choiceinput1_incorrect")).should("be.visible");
387
+ cy.get(cesc("#\\/_choiceinput1")).select(`unstable`);
388
+ cy.get(cesc("#\\/_choiceinput1_submit")).click();
389
+ cy.get(cesc("#\\/_choiceinput1_incorrect")).should("be.visible");
390
+ });
391
+
392
+ it("conditional math used as correct answer", () => {
393
+ cy.window().then(async (win) => {
394
+ win.postMessage(
395
+ {
396
+ doenetML: `
397
+ <text>a</text>
398
+ <p>Require <choiceinput inline="true" name="c" shuffleOrder><choice>positive</choice><choice>negative</choice></choiceinput>.</p>
399
+
400
+ <p>Condition on <m>x</m>:
401
+ <answer>
402
+ <mathinput name="x" />
403
+ <award><when>
404
+ <copy prop="immediateValue" target="x" />
405
+ =
406
+ <math>
407
+ <conditionalcontent condition="$c = positive" >
408
+ x > 0
409
+ </conditionalcontent>
410
+ <conditionalcontent condition="$c = negative" >
411
+ x < 0
412
+ </conditionalcontent>
413
+ </math>
414
+ </when></award>
415
+ </answer>
416
+ </p>
417
+ `,
418
+ },
419
+ "*",
420
+ );
421
+ });
422
+
423
+ cy.get(cesc("#\\/_text1")).should("have.text", "a");
424
+ cy.get(cesc("#\\/x_submit")).click();
425
+ cy.get(cesc("#\\/x_incorrect")).should("be.visible");
426
+
427
+ cy.get(cesc("#\\/x") + " textarea").type("x > 0{enter}", { force: true });
428
+ cy.get(cesc("#\\/x_incorrect")).should("be.visible");
429
+
430
+ cy.get(cesc("#\\/x") + " textarea").type(
431
+ "{end}{backspace}{backspace}{backspace}< 0",
432
+ { force: true },
433
+ );
434
+ cy.get(cesc("#\\/x_submit")).should("be.visible");
435
+ cy.get(cesc("#\\/x") + " textarea").type("{enter}", { force: true });
436
+ cy.get(cesc("#\\/x_incorrect")).should("be.visible");
437
+
438
+ cy.get(cesc("#\\/c")).select(`negative`);
439
+ cy.get(cesc("#\\/x_submit")).click();
440
+ cy.get(cesc("#\\/x_correct")).should("be.visible");
441
+
442
+ cy.get(cesc("#\\/x") + " textarea").type(
443
+ "{end}{backspace}{backspace}{backspace}> 0",
444
+ { force: true },
445
+ );
446
+ cy.get(cesc("#\\/x_submit")).should("be.visible");
447
+ cy.get(cesc("#\\/x") + " textarea").type("{enter}", { force: true });
448
+ cy.get(cesc("#\\/x_incorrect")).should("be.visible");
449
+
450
+ cy.get(cesc("#\\/c")).select(`positive`);
451
+ cy.get(cesc("#\\/x_submit")).click();
452
+ cy.get(cesc("#\\/x_correct")).should("be.visible");
453
+
454
+ cy.get(cesc("#\\/x") + " textarea").type(
455
+ "{end}{backspace}{backspace}{backspace}< 0",
456
+ { force: true },
457
+ );
458
+ cy.get(cesc("#\\/x_submit")).should("be.visible");
459
+ cy.get(cesc("#\\/x") + " textarea").type("{enter}", { force: true });
460
+ cy.get(cesc("#\\/x_incorrect")).should("be.visible");
461
+ });
462
+
463
+ it("include blank string between tags", () => {
464
+ cy.window().then(async (win) => {
465
+ win.postMessage(
466
+ {
467
+ doenetML: `
468
+ <text>a</text>
469
+ <setup>
470
+ <text name="animal">fox</text><text name="verb">jumps</text>
471
+ </setup>
472
+ <booleaninput name="b" >
473
+ <label>animal phrase</label>
474
+ </booleaninput>
475
+
476
+ <p name="p"><conditionalContent condition="$b">The $animal $verb.</conditionalcontent></p>
477
+ `,
478
+ },
479
+ "*",
480
+ );
481
+ });
482
+
483
+ cy.get(cesc("#\\/_text1")).should("have.text", "a");
484
+
485
+ cy.get(cesc("#\\/p")).should("have.text", "");
486
+
487
+ cy.get(cesc("#\\/b")).click();
488
+
489
+ cy.get(cesc("#\\/p")).should("have.text", "The fox jumps.");
490
+ });
491
+
492
+ it("assignNames skips strings but strings still displayed", () => {
493
+ cy.window().then(async (win) => {
494
+ win.postMessage(
495
+ {
496
+ doenetML: `
497
+ <text>a</text>
498
+ <mathinput name="n" />
499
+ <p name="p1"><conditionalContent condition="$n > 0" assignNames="a b">
500
+ <text>dog</text> mouse <text>cat</text>
501
+ </conditionalContent></p>
502
+
503
+ <p name="pa">$a</p>
504
+
505
+ <p name="pb">$b</p>
506
+
507
+ <p name="p2" ><copy target="_conditionalcontent1" assignNames="c d" /></p>
508
+
509
+ <p name="pc">$c</p>
510
+
511
+ <p name="pd">$d</p>
512
+
513
+
514
+ `,
515
+ },
516
+ "*",
517
+ );
518
+ });
519
+
520
+ cy.get(cesc("#\\/_text1")).should("have.text", "a"); // to wait until loaded
521
+
522
+ cy.get(cesc("#\\/p1")).should("have.text", "");
523
+ cy.get(cesc("#\\/pa")).should("have.text", "");
524
+ cy.get(cesc("#\\/pb")).should("have.text", "");
525
+ cy.get(cesc("#\\/p2")).should("have.text", "");
526
+ cy.get(cesc("#\\/pc")).should("have.text", "");
527
+ cy.get(cesc("#\\/pd")).should("have.text", "");
528
+
529
+ cy.log("enter 1");
530
+ cy.get(cesc("#\\/n") + " textarea").type("1{enter}", { force: true });
531
+
532
+ cy.get(cesc("#\\/p1")).should("contain.text", "dog mouse cat");
533
+ cy.get(cesc("#\\/pa")).should("have.text", "dog");
534
+ cy.get(cesc("#\\/pb")).should("have.text", "cat");
535
+ cy.get(cesc("#\\/p2")).should("contain.text", "dog mouse cat");
536
+ cy.get(cesc("#\\/pc")).should("have.text", "dog");
537
+ cy.get(cesc("#\\/pd")).should("have.text", "cat");
538
+
539
+ cy.log("enter 0");
540
+ cy.get(cesc("#\\/n") + " textarea").type("{end}{backspace}0{enter}", {
541
+ force: true,
542
+ });
543
+
544
+ cy.get(cesc("#\\/p1")).should("have.text", "");
545
+ cy.get(cesc("#\\/pa")).should("have.text", "");
546
+ cy.get(cesc("#\\/pb")).should("have.text", "");
547
+ cy.get(cesc("#\\/p2")).should("have.text", "");
548
+ cy.get(cesc("#\\/pc")).should("have.text", "");
549
+ cy.get(cesc("#\\/pd")).should("have.text", "");
550
+ });
551
+
552
+ it("correctly withhold replacements when shadowing", () => {
553
+ cy.window().then(async (win) => {
554
+ win.postMessage(
555
+ {
556
+ doenetML: `
557
+ <p>Hide greeting:
558
+ <booleanInput name="hide" />
559
+ </p>
560
+
561
+ <p name="p">Greeting is hidden: $hide. Greeting: <conditionalContent condition="not $hide">Hello!</conditionalContent></p>
562
+
563
+ <p>Show copy:
564
+ <booleanInput name="show_copy" />
565
+ </p>
566
+ <conditionalContent condition="$show_copy" assignNames="(p2)">
567
+ $p
568
+ </conditionalContent>
569
+
570
+ `,
571
+ },
572
+ "*",
573
+ );
574
+ });
575
+
576
+ cy.get(cesc("#\\/p")).should(
577
+ "have.text",
578
+ "Greeting is hidden: false. Greeting: Hello!",
579
+ );
580
+ cy.get(cesc("#\\/p2")).should("not.exist");
581
+
582
+ cy.get(cesc("#\\/hide")).click();
583
+
584
+ cy.get(cesc("#\\/p")).should(
585
+ "have.text",
586
+ "Greeting is hidden: true. Greeting: ",
587
+ );
588
+ cy.get(cesc("#\\/p2")).should("not.exist");
589
+
590
+ cy.get(cesc("#\\/show_copy")).click();
591
+ cy.get(cesc("#\\/p2")).should(
592
+ "have.text",
593
+ "Greeting is hidden: true. Greeting: ",
594
+ );
595
+
596
+ cy.get(cesc("#\\/hide")).click();
597
+
598
+ cy.get(cesc("#\\/p")).should(
599
+ "have.text",
600
+ "Greeting is hidden: false. Greeting: Hello!",
601
+ );
602
+ cy.get(cesc("#\\/p2")).should(
603
+ "have.text",
604
+ "Greeting is hidden: false. Greeting: Hello!",
605
+ );
606
+ });
607
+
608
+ // tests with cases or else
609
+
610
+ it("case/else with single text, assign sub on copy", () => {
611
+ cy.window().then(async (win) => {
612
+ win.postMessage(
613
+ {
614
+ doenetML: `
615
+ <text>a</text>
616
+ <mathinput name="n" />
617
+ <p name="pa">a: <conditionalContent assignNames="a">
618
+ <case condition="$n < 0"><text>dog</text></case>
619
+ <case condition="$n <=1"><text>cat</text></case>
620
+ <else><text>mouse</text></else>
621
+ </conditionalContent></p>
622
+
623
+ <p name="pa1">a1: <copy target="a" assignNames="(a1)" /></p>
624
+
625
+ <p name="pb" >b: <copy target="_conditionalcontent1" assignNames="(b)" /></p>
626
+
627
+ <p name="pb1">b1: <copy target="b" assignNames="b1" /></p>
628
+
629
+ `,
630
+ },
631
+ "*",
632
+ );
633
+ });
634
+
635
+ cy.get(cesc("#\\/_text1")).should("have.text", "a"); // to wait until loaded
636
+
637
+ cy.get(cesc("#\\/pa")).should("have.text", "a: mouse");
638
+ cy.get(cesc("#\\/pa1")).should("have.text", "a1: mouse");
639
+ cy.get(cesc("#\\/pb")).should("have.text", "b: mouse");
640
+ cy.get(cesc("#\\/pb1")).should("have.text", "b1: mouse");
641
+
642
+ cy.get(cesc("#\\/a1")).should("have.text", "mouse");
643
+ cy.get(cesc("#\\/b")).should("have.text", "mouse");
644
+ cy.get(cesc("#\\/b1")).should("have.text", "mouse");
645
+
646
+ cy.log("enter 1");
647
+ cy.get(cesc("#\\/n") + " textarea").type("1{enter}", { force: true });
648
+
649
+ cy.get(cesc("#\\/pa")).should("have.text", "a: cat");
650
+ cy.get(cesc("#\\/pa1")).should("have.text", "a1: cat");
651
+ cy.get(cesc("#\\/pb")).should("have.text", "b: cat");
652
+ cy.get(cesc("#\\/pb1")).should("have.text", "b1: cat");
653
+
654
+ cy.get(cesc("#\\/a1")).should("have.text", "cat");
655
+ cy.get(cesc("#\\/b")).should("have.text", "cat");
656
+ cy.get(cesc("#\\/b1")).should("have.text", "cat");
657
+
658
+ cy.log("enter 10");
659
+ cy.get(cesc("#\\/n") + " textarea").type(
660
+ "{end}{backspace}{backspace}10{enter}",
661
+ { force: true },
662
+ );
663
+
664
+ cy.get(cesc("#\\/pa")).should("have.text", "a: mouse");
665
+ cy.get(cesc("#\\/pa1")).should("have.text", "a1: mouse");
666
+ cy.get(cesc("#\\/pb")).should("have.text", "b: mouse");
667
+ cy.get(cesc("#\\/pb1")).should("have.text", "b1: mouse");
668
+
669
+ cy.get(cesc("#\\/a1")).should("have.text", "mouse");
670
+ cy.get(cesc("#\\/b")).should("have.text", "mouse");
671
+ cy.get(cesc("#\\/b1")).should("have.text", "mouse");
672
+
673
+ cy.log("enter -1");
674
+ cy.get(cesc("#\\/n") + " textarea").type(
675
+ "{ctrl+home}{shift+end}{backspace}-1{enter}",
676
+ { force: true },
677
+ );
678
+
679
+ cy.get(cesc("#\\/pa")).should("have.text", "a: dog");
680
+ cy.get(cesc("#\\/pa1")).should("have.text", "a1: dog");
681
+ cy.get(cesc("#\\/pb")).should("have.text", "b: dog");
682
+ cy.get(cesc("#\\/pb1")).should("have.text", "b1: dog");
683
+
684
+ cy.get(cesc("#\\/a1")).should("have.text", "dog");
685
+ cy.get(cesc("#\\/b")).should("have.text", "dog");
686
+ cy.get(cesc("#\\/b1")).should("have.text", "dog");
687
+
688
+ cy.log("enter x");
689
+ cy.get(cesc("#\\/n") + " textarea").type(
690
+ "{ctrl+home}{shift+end}{backspace}x{enter}",
691
+ { force: true },
692
+ );
693
+
694
+ cy.get(cesc("#\\/pa")).should("have.text", "a: mouse");
695
+ cy.get(cesc("#\\/pa1")).should("have.text", "a1: mouse");
696
+ cy.get(cesc("#\\/pb")).should("have.text", "b: mouse");
697
+ cy.get(cesc("#\\/pb1")).should("have.text", "b1: mouse");
698
+
699
+ cy.get(cesc("#\\/a1")).should("have.text", "mouse");
700
+ cy.get(cesc("#\\/b")).should("have.text", "mouse");
701
+ cy.get(cesc("#\\/b1")).should("have.text", "mouse");
702
+ });
703
+
704
+ it("case/else with single text, initially assign sub", () => {
705
+ cy.window().then(async (win) => {
706
+ win.postMessage(
707
+ {
708
+ doenetML: `
709
+ <text>a</text>
710
+ <mathinput name="n" />
711
+ <p name="pa">a: <conditionalContent assignNames="(a)">
712
+ <case condition="$n < 0"><text>dog</text></case>
713
+ <case condition="$n <=1"><text>cat</text></case>
714
+ <else><text>mouse</text></else>
715
+ </conditionalContent></p>
716
+
717
+ <p name="pa1">a1: <copy target="a" assignNames="a1" /></p>
718
+
719
+ <p name="pb" >b: <copy target="_conditionalcontent1" assignNames="b" /></p>
720
+
721
+ <p name="pb1">b1: <copy target="b" assignNames="(b1)" /></p>
722
+
723
+ `,
724
+ },
725
+ "*",
726
+ );
727
+ });
728
+
729
+ cy.get(cesc("#\\/_text1")).should("have.text", "a"); // to wait until loaded
730
+
731
+ cy.get(cesc("#\\/pa")).should("have.text", "a: mouse");
732
+ cy.get(cesc("#\\/pa1")).should("have.text", "a1: mouse");
733
+ cy.get(cesc("#\\/pb")).should("have.text", "b: mouse");
734
+ cy.get(cesc("#\\/pb1")).should("have.text", "b1: mouse");
735
+
736
+ cy.get(cesc("#\\/a")).should("have.text", "mouse");
737
+ cy.get(cesc("#\\/a1")).should("have.text", "mouse");
738
+ cy.get(cesc("#\\/b1")).should("have.text", "mouse");
739
+
740
+ cy.log("enter 1");
741
+ cy.get(cesc("#\\/n") + " textarea").type("1{enter}", { force: true });
742
+
743
+ cy.get(cesc("#\\/pa")).should("have.text", "a: cat");
744
+ cy.get(cesc("#\\/pa1")).should("have.text", "a1: cat");
745
+ cy.get(cesc("#\\/pb")).should("have.text", "b: cat");
746
+ cy.get(cesc("#\\/pb1")).should("have.text", "b1: cat");
747
+
748
+ cy.get(cesc("#\\/a")).should("have.text", "cat");
749
+ cy.get(cesc("#\\/a1")).should("have.text", "cat");
750
+ cy.get(cesc("#\\/b1")).should("have.text", "cat");
751
+
752
+ cy.log("enter 10");
753
+ cy.get(cesc("#\\/n") + " textarea").type("{end}{backspace}10{enter}", {
754
+ force: true,
755
+ });
756
+
757
+ cy.get(cesc("#\\/pa")).should("have.text", "a: mouse");
758
+ cy.get(cesc("#\\/pa1")).should("have.text", "a1: mouse");
759
+ cy.get(cesc("#\\/pb")).should("have.text", "b: mouse");
760
+ cy.get(cesc("#\\/pb1")).should("have.text", "b1: mouse");
761
+
762
+ cy.get(cesc("#\\/a")).should("have.text", "mouse");
763
+ cy.get(cesc("#\\/a1")).should("have.text", "mouse");
764
+ cy.get(cesc("#\\/b1")).should("have.text", "mouse");
765
+
766
+ cy.log("enter -11");
767
+ cy.get(cesc("#\\/n") + " textarea").type(
768
+ "{end}{backspace}{backspace}-1{enter}",
769
+ { force: true },
770
+ );
771
+
772
+ cy.get(cesc("#\\/pa")).should("have.text", "a: dog");
773
+ cy.get(cesc("#\\/pa1")).should("have.text", "a1: dog");
774
+ cy.get(cesc("#\\/pb")).should("have.text", "b: dog");
775
+ cy.get(cesc("#\\/pb1")).should("have.text", "b1: dog");
776
+
777
+ cy.get(cesc("#\\/a")).should("have.text", "dog");
778
+ cy.get(cesc("#\\/a1")).should("have.text", "dog");
779
+ cy.get(cesc("#\\/b1")).should("have.text", "dog");
780
+
781
+ cy.log("enter x");
782
+ cy.get(cesc("#\\/n") + " textarea").type(
783
+ "{end}{backspace}{backspace}x{enter}",
784
+ { force: true },
785
+ );
786
+
787
+ cy.get(cesc("#\\/pa")).should("have.text", "a: mouse");
788
+ cy.get(cesc("#\\/pa1")).should("have.text", "a1: mouse");
789
+ cy.get(cesc("#\\/pb")).should("have.text", "b: mouse");
790
+ cy.get(cesc("#\\/pb1")).should("have.text", "b1: mouse");
791
+
792
+ cy.get(cesc("#\\/a")).should("have.text", "mouse");
793
+ cy.get(cesc("#\\/a1")).should("have.text", "mouse");
794
+ cy.get(cesc("#\\/b1")).should("have.text", "mouse");
795
+ });
796
+
797
+ it("case/else with text, math, and optional", () => {
798
+ cy.window().then(async (win) => {
799
+ win.postMessage(
800
+ {
801
+ doenetML: `
802
+ <text>a</text>
803
+ <mathinput name="n" />
804
+ <p>original: <conditionalContent assignNames="(a b c d)">
805
+ <case condition="$n<0" ><text>dog</text> <math>x</math>
806
+ <text>optional text!</text>
807
+ </case>
808
+ <case condition="$n <= 1" ><text>cat</text> <math>y</math>
809
+ </case>
810
+ <else><text>mouse</text> <math>z</math>
811
+ </else>
812
+ </conditionalContent></p>
813
+
814
+ <p>a1: <copy target="a" assignNames="a1" /></p>
815
+ <p>b1: <copy target="b" assignNames="b1" /></p>
816
+ <p>c1: <copy target="c" assignNames="c1" /></p>
817
+ <p>d1: <copy target="d" assignNames="d1" /></p>
818
+
819
+ <p>copy: <copy name="cp1" target="_conditionalcontent1" assignNames="(e f g h i)" /></p>
820
+
821
+ <p>e1: <copy target="e" assignNames="e1" /></p>
822
+ <p>f1: <copy target="f" assignNames="f1" /></p>
823
+ <p>g1: <copy target="g" assignNames="g1" /></p>
824
+ <p>h1: <copy target="h" assignNames="h1" /></p>
825
+ <p>i1: <copy target="i" assignNames="i1" /></p>
826
+
827
+ <p>copied copy: <copy target="cp1" assignNames="(j k l)" /></p>
828
+
829
+ <p>j1: <copy target="j" assignNames="j1" /></p>
830
+ <p>k1: <copy target="k" assignNames="k1" /></p>
831
+ <p>l1: <copy target="l" assignNames="l1" /></p>
832
+ `,
833
+ },
834
+ "*",
835
+ );
836
+ });
837
+
838
+ cy.get(cesc("#\\/_text1")).should("have.text", "a"); // to wait until loaded
839
+
840
+ cy.get(cesc("#\\/a")).should("have.text", "mouse");
841
+ cy.get(cesc("#\\/a1")).should("have.text", "mouse");
842
+ cy.get(cesc("#\\/e")).should("have.text", "mouse");
843
+ cy.get(cesc("#\\/e1")).should("have.text", "mouse");
844
+ cy.get(cesc("#\\/j")).should("have.text", "mouse");
845
+ cy.get(cesc("#\\/j1")).should("have.text", "mouse");
846
+
847
+ cy.get(cesc("#\\/b"))
848
+ .find(".mjx-mrow")
849
+ .eq(0)
850
+ .invoke("text")
851
+ .then((text) => {
852
+ expect(text.trim()).equal("z");
853
+ });
854
+ cy.get(cesc("#\\/b1"))
855
+ .find(".mjx-mrow")
856
+ .eq(0)
857
+ .invoke("text")
858
+ .then((text) => {
859
+ expect(text.trim()).equal("z");
860
+ });
861
+ cy.get(cesc("#\\/f"))
862
+ .find(".mjx-mrow")
863
+ .eq(0)
864
+ .invoke("text")
865
+ .then((text) => {
866
+ expect(text.trim()).equal("z");
867
+ });
868
+ cy.get(cesc("#\\/f1"))
869
+ .find(".mjx-mrow")
870
+ .eq(0)
871
+ .invoke("text")
872
+ .then((text) => {
873
+ expect(text.trim()).equal("z");
874
+ });
875
+ cy.get(cesc("#\\/k"))
876
+ .find(".mjx-mrow")
877
+ .eq(0)
878
+ .invoke("text")
879
+ .then((text) => {
880
+ expect(text.trim()).equal("z");
881
+ });
882
+ cy.get(cesc("#\\/k1"))
883
+ .find(".mjx-mrow")
884
+ .eq(0)
885
+ .invoke("text")
886
+ .then((text) => {
887
+ expect(text.trim()).equal("z");
888
+ });
889
+
890
+ cy.get(cesc("#\\/c")).should("not.exist");
891
+ cy.get(cesc("#\\/c1")).should("not.exist");
892
+ cy.get(cesc("#\\/g")).should("not.exist");
893
+ cy.get(cesc("#\\/g1")).should("not.exist");
894
+ cy.get(cesc("#\\/l")).should("not.exist");
895
+ cy.get(cesc("#\\/l1")).should("not.exist");
896
+
897
+ cy.get(cesc("#\\/d")).should("not.exist");
898
+ cy.get(cesc("#\\/d1")).should("not.exist");
899
+ cy.get(cesc("#\\/h")).should("not.exist");
900
+ cy.get(cesc("#\\/h1")).should("not.exist");
901
+ cy.get(cesc("#\\/i")).should("not.exist");
902
+ cy.get(cesc("#\\/i1")).should("not.exist");
903
+
904
+ cy.log("enter 1");
905
+ cy.get(cesc("#\\/n") + " textarea").type("1{enter}", { force: true });
906
+
907
+ cy.get(cesc("#\\/a")).should("have.text", "cat");
908
+ cy.get(cesc("#\\/a1")).should("have.text", "cat");
909
+ cy.get(cesc("#\\/e")).should("have.text", "cat");
910
+ cy.get(cesc("#\\/e1")).should("have.text", "cat");
911
+ cy.get(cesc("#\\/j")).should("have.text", "cat");
912
+ cy.get(cesc("#\\/j1")).should("have.text", "cat");
913
+
914
+ cy.get(cesc("#\\/b"))
915
+ .find(".mjx-mrow")
916
+ .eq(0)
917
+ .invoke("text")
918
+ .then((text) => {
919
+ expect(text.trim()).equal("y");
920
+ });
921
+ cy.get(cesc("#\\/b1"))
922
+ .find(".mjx-mrow")
923
+ .eq(0)
924
+ .invoke("text")
925
+ .then((text) => {
926
+ expect(text.trim()).equal("y");
927
+ });
928
+ cy.get(cesc("#\\/f"))
929
+ .find(".mjx-mrow")
930
+ .eq(0)
931
+ .invoke("text")
932
+ .then((text) => {
933
+ expect(text.trim()).equal("y");
934
+ });
935
+ cy.get(cesc("#\\/f1"))
936
+ .find(".mjx-mrow")
937
+ .eq(0)
938
+ .invoke("text")
939
+ .then((text) => {
940
+ expect(text.trim()).equal("y");
941
+ });
942
+ cy.get(cesc("#\\/k"))
943
+ .find(".mjx-mrow")
944
+ .eq(0)
945
+ .invoke("text")
946
+ .then((text) => {
947
+ expect(text.trim()).equal("y");
948
+ });
949
+ cy.get(cesc("#\\/k1"))
950
+ .find(".mjx-mrow")
951
+ .eq(0)
952
+ .invoke("text")
953
+ .then((text) => {
954
+ expect(text.trim()).equal("y");
955
+ });
956
+
957
+ cy.get(cesc("#\\/c")).should("not.exist");
958
+ cy.get(cesc("#\\/c1")).should("not.exist");
959
+ cy.get(cesc("#\\/g")).should("not.exist");
960
+ cy.get(cesc("#\\/g1")).should("not.exist");
961
+ cy.get(cesc("#\\/l")).should("not.exist");
962
+ cy.get(cesc("#\\/l1")).should("not.exist");
963
+
964
+ cy.get(cesc("#\\/d")).should("not.exist");
965
+ cy.get(cesc("#\\/d1")).should("not.exist");
966
+ cy.get(cesc("#\\/h")).should("not.exist");
967
+ cy.get(cesc("#\\/h1")).should("not.exist");
968
+ cy.get(cesc("#\\/i")).should("not.exist");
969
+ cy.get(cesc("#\\/i1")).should("not.exist");
970
+
971
+ cy.log("enter 10");
972
+ cy.get(cesc("#\\/n") + " textarea").type("{end}{backspace}10{enter}", {
973
+ force: true,
974
+ });
975
+
976
+ cy.get(cesc("#\\/a")).should("have.text", "mouse");
977
+ cy.get(cesc("#\\/a1")).should("have.text", "mouse");
978
+ cy.get(cesc("#\\/e")).should("have.text", "mouse");
979
+ cy.get(cesc("#\\/e1")).should("have.text", "mouse");
980
+ cy.get(cesc("#\\/j")).should("have.text", "mouse");
981
+ cy.get(cesc("#\\/j1")).should("have.text", "mouse");
982
+
983
+ cy.get(cesc("#\\/b"))
984
+ .find(".mjx-mrow")
985
+ .eq(0)
986
+ .invoke("text")
987
+ .then((text) => {
988
+ expect(text.trim()).equal("z");
989
+ });
990
+ cy.get(cesc("#\\/b1"))
991
+ .find(".mjx-mrow")
992
+ .eq(0)
993
+ .invoke("text")
994
+ .then((text) => {
995
+ expect(text.trim()).equal("z");
996
+ });
997
+ cy.get(cesc("#\\/f"))
998
+ .find(".mjx-mrow")
999
+ .eq(0)
1000
+ .invoke("text")
1001
+ .then((text) => {
1002
+ expect(text.trim()).equal("z");
1003
+ });
1004
+ cy.get(cesc("#\\/f1"))
1005
+ .find(".mjx-mrow")
1006
+ .eq(0)
1007
+ .invoke("text")
1008
+ .then((text) => {
1009
+ expect(text.trim()).equal("z");
1010
+ });
1011
+ cy.get(cesc("#\\/k"))
1012
+ .find(".mjx-mrow")
1013
+ .eq(0)
1014
+ .invoke("text")
1015
+ .then((text) => {
1016
+ expect(text.trim()).equal("z");
1017
+ });
1018
+ cy.get(cesc("#\\/k1"))
1019
+ .find(".mjx-mrow")
1020
+ .eq(0)
1021
+ .invoke("text")
1022
+ .then((text) => {
1023
+ expect(text.trim()).equal("z");
1024
+ });
1025
+
1026
+ cy.get(cesc("#\\/c")).should("not.exist");
1027
+ cy.get(cesc("#\\/c1")).should("not.exist");
1028
+ cy.get(cesc("#\\/g")).should("not.exist");
1029
+ cy.get(cesc("#\\/g1")).should("not.exist");
1030
+ cy.get(cesc("#\\/l")).should("not.exist");
1031
+ cy.get(cesc("#\\/l1")).should("not.exist");
1032
+
1033
+ cy.get(cesc("#\\/d")).should("not.exist");
1034
+ cy.get(cesc("#\\/d1")).should("not.exist");
1035
+ cy.get(cesc("#\\/h")).should("not.exist");
1036
+ cy.get(cesc("#\\/h1")).should("not.exist");
1037
+ cy.get(cesc("#\\/i")).should("not.exist");
1038
+ cy.get(cesc("#\\/i1")).should("not.exist");
1039
+
1040
+ cy.log("enter -11");
1041
+ cy.get(cesc("#\\/n") + " textarea").type(
1042
+ "{end}{backspace}{backspace}-1{enter}",
1043
+ { force: true },
1044
+ );
1045
+
1046
+ cy.get(cesc("#\\/a")).should("have.text", "dog");
1047
+ cy.get(cesc("#\\/a1")).should("have.text", "dog");
1048
+ cy.get(cesc("#\\/e")).should("have.text", "dog");
1049
+ cy.get(cesc("#\\/e1")).should("have.text", "dog");
1050
+ cy.get(cesc("#\\/j")).should("have.text", "dog");
1051
+ cy.get(cesc("#\\/j1")).should("have.text", "dog");
1052
+
1053
+ cy.get(cesc("#\\/b"))
1054
+ .find(".mjx-mrow")
1055
+ .eq(0)
1056
+ .invoke("text")
1057
+ .then((text) => {
1058
+ expect(text.trim()).equal("x");
1059
+ });
1060
+ cy.get(cesc("#\\/b1"))
1061
+ .find(".mjx-mrow")
1062
+ .eq(0)
1063
+ .invoke("text")
1064
+ .then((text) => {
1065
+ expect(text.trim()).equal("x");
1066
+ });
1067
+ cy.get(cesc("#\\/f"))
1068
+ .find(".mjx-mrow")
1069
+ .eq(0)
1070
+ .invoke("text")
1071
+ .then((text) => {
1072
+ expect(text.trim()).equal("x");
1073
+ });
1074
+ cy.get(cesc("#\\/f1"))
1075
+ .find(".mjx-mrow")
1076
+ .eq(0)
1077
+ .invoke("text")
1078
+ .then((text) => {
1079
+ expect(text.trim()).equal("x");
1080
+ });
1081
+ cy.get(cesc("#\\/k"))
1082
+ .find(".mjx-mrow")
1083
+ .eq(0)
1084
+ .invoke("text")
1085
+ .then((text) => {
1086
+ expect(text.trim()).equal("x");
1087
+ });
1088
+ cy.get(cesc("#\\/k1"))
1089
+ .find(".mjx-mrow")
1090
+ .eq(0)
1091
+ .invoke("text")
1092
+ .then((text) => {
1093
+ expect(text.trim()).equal("x");
1094
+ });
1095
+
1096
+ cy.get(cesc("#\\/c")).should("have.text", "optional text!");
1097
+ cy.get(cesc("#\\/c1")).should("have.text", "optional text!");
1098
+ cy.get(cesc("#\\/g")).should("have.text", "optional text!");
1099
+ cy.get(cesc("#\\/g1")).should("have.text", "optional text!");
1100
+ cy.get(cesc("#\\/l")).should("have.text", "optional text!");
1101
+ cy.get(cesc("#\\/l1")).should("have.text", "optional text!");
1102
+
1103
+ cy.get(cesc("#\\/d")).should("not.exist");
1104
+ cy.get(cesc("#\\/d1")).should("not.exist");
1105
+ cy.get(cesc("#\\/h")).should("not.exist");
1106
+ cy.get(cesc("#\\/h1")).should("not.exist");
1107
+ cy.get(cesc("#\\/i")).should("not.exist");
1108
+ cy.get(cesc("#\\/i1")).should("not.exist");
1109
+
1110
+ cy.log("enter x");
1111
+ cy.get(cesc("#\\/n") + " textarea").type(
1112
+ "{end}{backspace}{backspace}x{enter}",
1113
+ { force: true },
1114
+ );
1115
+
1116
+ cy.get(cesc("#\\/a")).should("have.text", "mouse");
1117
+ cy.get(cesc("#\\/a1")).should("have.text", "mouse");
1118
+ cy.get(cesc("#\\/e")).should("have.text", "mouse");
1119
+ cy.get(cesc("#\\/e1")).should("have.text", "mouse");
1120
+ cy.get(cesc("#\\/j")).should("have.text", "mouse");
1121
+ cy.get(cesc("#\\/j1")).should("have.text", "mouse");
1122
+
1123
+ cy.get(cesc("#\\/b"))
1124
+ .find(".mjx-mrow")
1125
+ .eq(0)
1126
+ .invoke("text")
1127
+ .then((text) => {
1128
+ expect(text.trim()).equal("z");
1129
+ });
1130
+ cy.get(cesc("#\\/b1"))
1131
+ .find(".mjx-mrow")
1132
+ .eq(0)
1133
+ .invoke("text")
1134
+ .then((text) => {
1135
+ expect(text.trim()).equal("z");
1136
+ });
1137
+ cy.get(cesc("#\\/f"))
1138
+ .find(".mjx-mrow")
1139
+ .eq(0)
1140
+ .invoke("text")
1141
+ .then((text) => {
1142
+ expect(text.trim()).equal("z");
1143
+ });
1144
+ cy.get(cesc("#\\/f1"))
1145
+ .find(".mjx-mrow")
1146
+ .eq(0)
1147
+ .invoke("text")
1148
+ .then((text) => {
1149
+ expect(text.trim()).equal("z");
1150
+ });
1151
+ cy.get(cesc("#\\/k"))
1152
+ .find(".mjx-mrow")
1153
+ .eq(0)
1154
+ .invoke("text")
1155
+ .then((text) => {
1156
+ expect(text.trim()).equal("z");
1157
+ });
1158
+ cy.get(cesc("#\\/k1"))
1159
+ .find(".mjx-mrow")
1160
+ .eq(0)
1161
+ .invoke("text")
1162
+ .then((text) => {
1163
+ expect(text.trim()).equal("z");
1164
+ });
1165
+
1166
+ cy.get(cesc("#\\/c")).should("not.exist");
1167
+ cy.get(cesc("#\\/c1")).should("not.exist");
1168
+ cy.get(cesc("#\\/g")).should("not.exist");
1169
+ cy.get(cesc("#\\/g1")).should("not.exist");
1170
+ cy.get(cesc("#\\/l")).should("not.exist");
1171
+ cy.get(cesc("#\\/l1")).should("not.exist");
1172
+
1173
+ cy.get(cesc("#\\/d")).should("not.exist");
1174
+ cy.get(cesc("#\\/d1")).should("not.exist");
1175
+ cy.get(cesc("#\\/h")).should("not.exist");
1176
+ cy.get(cesc("#\\/h1")).should("not.exist");
1177
+ cy.get(cesc("#\\/i")).should("not.exist");
1178
+ cy.get(cesc("#\\/i1")).should("not.exist");
1179
+ });
1180
+
1181
+ it("case/else with text, math, and optional, new namespace", () => {
1182
+ cy.window().then(async (win) => {
1183
+ win.postMessage(
1184
+ {
1185
+ doenetML: `
1186
+ <text>a</text>
1187
+ <mathinput name="n" />
1188
+ <p>original: <conditionalContent assignNames="(a b c d)" name="s1" newnamespace>
1189
+ <case condition="$(../n)<0" ><text>dog</text> <math>x</math>
1190
+ <text>optional text!</text>
1191
+ </case>
1192
+ <case condition="$(../n) <= 1" ><text>cat</text> <math>y</math>
1193
+ </case>
1194
+ <else><text>mouse</text> <math>z</math>
1195
+ </else>
1196
+ </conditionalContent></p>
1197
+
1198
+ <p>a1: <copy target="s1/a" assignNames="a1" /></p>
1199
+ <p>b1: <copy target="s1/b" assignNames="b1" /></p>
1200
+ <p>c1: <copy target="s1/c" assignNames="c1" /></p>
1201
+ <p>d1: <copy target="s1/d" assignNames="d1" /></p>
1202
+
1203
+ <p>copy: <copy name="s2" target="s1" assignNames="(e f g h i)" /></p>
1204
+
1205
+ <p>e1: <copy target="e" assignNames="e1" /></p>
1206
+ <p>f1: <copy target="f" assignNames="f1" /></p>
1207
+ <p>g1: <copy target="g" assignNames="g1" /></p>
1208
+ <p>h1: <copy target="h" assignNames="h1" /></p>
1209
+ <p>i1: <copy target="i" assignNames="i1" /></p>
1210
+
1211
+ <p>copied copy: <copy name="s3" target="s2" assignNames="(j k l)" newNamespace /></p>
1212
+
1213
+ <p>j1: <copy target="s3/j" assignNames="j1" /></p>
1214
+ <p>k1: <copy target="s3/k" assignNames="k1" /></p>
1215
+ <p>l1: <copy target="s3/l" assignNames="l1" /></p>
1216
+ `,
1217
+ },
1218
+ "*",
1219
+ );
1220
+ });
1221
+
1222
+ cy.get(cesc("#\\/_text1")).should("have.text", "a"); // to wait until loaded
1223
+
1224
+ cy.get(cesc("#\\/s1\\/a")).should("have.text", "mouse");
1225
+ cy.get(cesc("#\\/a1")).should("have.text", "mouse");
1226
+ cy.get(cesc("#\\/e")).should("have.text", "mouse");
1227
+ cy.get(cesc("#\\/e1")).should("have.text", "mouse");
1228
+ cy.get(cesc("#\\/s3\\/j")).should("have.text", "mouse");
1229
+ cy.get(cesc("#\\/j1")).should("have.text", "mouse");
1230
+
1231
+ cy.get(cesc("#\\/s1\\/b"))
1232
+ .find(".mjx-mrow")
1233
+ .eq(0)
1234
+ .invoke("text")
1235
+ .then((text) => {
1236
+ expect(text.trim()).equal("z");
1237
+ });
1238
+ cy.get(cesc("#\\/b1"))
1239
+ .find(".mjx-mrow")
1240
+ .eq(0)
1241
+ .invoke("text")
1242
+ .then((text) => {
1243
+ expect(text.trim()).equal("z");
1244
+ });
1245
+ cy.get(cesc("#\\/f"))
1246
+ .find(".mjx-mrow")
1247
+ .eq(0)
1248
+ .invoke("text")
1249
+ .then((text) => {
1250
+ expect(text.trim()).equal("z");
1251
+ });
1252
+ cy.get(cesc("#\\/f1"))
1253
+ .find(".mjx-mrow")
1254
+ .eq(0)
1255
+ .invoke("text")
1256
+ .then((text) => {
1257
+ expect(text.trim()).equal("z");
1258
+ });
1259
+ cy.get(cesc("#\\/s3\\/k"))
1260
+ .find(".mjx-mrow")
1261
+ .eq(0)
1262
+ .invoke("text")
1263
+ .then((text) => {
1264
+ expect(text.trim()).equal("z");
1265
+ });
1266
+ cy.get(cesc("#\\/k1"))
1267
+ .find(".mjx-mrow")
1268
+ .eq(0)
1269
+ .invoke("text")
1270
+ .then((text) => {
1271
+ expect(text.trim()).equal("z");
1272
+ });
1273
+
1274
+ cy.get(cesc("#\\/s1\\/c")).should("not.exist");
1275
+ cy.get(cesc("#\\/c1")).should("not.exist");
1276
+ cy.get(cesc("#\\/g")).should("not.exist");
1277
+ cy.get(cesc("#\\/g1")).should("not.exist");
1278
+ cy.get(cesc("#\\/s3\\/l")).should("not.exist");
1279
+ cy.get(cesc("#\\/l1")).should("not.exist");
1280
+
1281
+ cy.get(cesc("#\\/s1\\/d")).should("not.exist");
1282
+ cy.get(cesc("#\\/d1")).should("not.exist");
1283
+ cy.get(cesc("#\\/h")).should("not.exist");
1284
+ cy.get(cesc("#\\/h1")).should("not.exist");
1285
+ cy.get(cesc("#\\/i")).should("not.exist");
1286
+ cy.get(cesc("#\\/i1")).should("not.exist");
1287
+
1288
+ cy.log("enter 1");
1289
+ cy.get(cesc("#\\/n") + " textarea").type("1{enter}", { force: true });
1290
+
1291
+ cy.get(cesc("#\\/s1\\/a")).should("have.text", "cat");
1292
+ cy.get(cesc("#\\/a1")).should("have.text", "cat");
1293
+ cy.get(cesc("#\\/e")).should("have.text", "cat");
1294
+ cy.get(cesc("#\\/e1")).should("have.text", "cat");
1295
+ cy.get(cesc("#\\/s3\\/j")).should("have.text", "cat");
1296
+ cy.get(cesc("#\\/j1")).should("have.text", "cat");
1297
+
1298
+ cy.get(cesc("#\\/s1\\/b"))
1299
+ .find(".mjx-mrow")
1300
+ .eq(0)
1301
+ .invoke("text")
1302
+ .then((text) => {
1303
+ expect(text.trim()).equal("y");
1304
+ });
1305
+ cy.get(cesc("#\\/b1"))
1306
+ .find(".mjx-mrow")
1307
+ .eq(0)
1308
+ .invoke("text")
1309
+ .then((text) => {
1310
+ expect(text.trim()).equal("y");
1311
+ });
1312
+ cy.get(cesc("#\\/f"))
1313
+ .find(".mjx-mrow")
1314
+ .eq(0)
1315
+ .invoke("text")
1316
+ .then((text) => {
1317
+ expect(text.trim()).equal("y");
1318
+ });
1319
+ cy.get(cesc("#\\/f1"))
1320
+ .find(".mjx-mrow")
1321
+ .eq(0)
1322
+ .invoke("text")
1323
+ .then((text) => {
1324
+ expect(text.trim()).equal("y");
1325
+ });
1326
+ cy.get(cesc("#\\/s3\\/k"))
1327
+ .find(".mjx-mrow")
1328
+ .eq(0)
1329
+ .invoke("text")
1330
+ .then((text) => {
1331
+ expect(text.trim()).equal("y");
1332
+ });
1333
+ cy.get(cesc("#\\/k1"))
1334
+ .find(".mjx-mrow")
1335
+ .eq(0)
1336
+ .invoke("text")
1337
+ .then((text) => {
1338
+ expect(text.trim()).equal("y");
1339
+ });
1340
+
1341
+ cy.get(cesc("#\\/s1\\/c")).should("not.exist");
1342
+ cy.get(cesc("#\\/c1")).should("not.exist");
1343
+ cy.get(cesc("#\\/g")).should("not.exist");
1344
+ cy.get(cesc("#\\/g1")).should("not.exist");
1345
+ cy.get(cesc("#\\/s3\\/l")).should("not.exist");
1346
+ cy.get(cesc("#\\/l1")).should("not.exist");
1347
+
1348
+ cy.get(cesc("#\\/s1\\/d")).should("not.exist");
1349
+ cy.get(cesc("#\\/d1")).should("not.exist");
1350
+ cy.get(cesc("#\\/h")).should("not.exist");
1351
+ cy.get(cesc("#\\/h1")).should("not.exist");
1352
+ cy.get(cesc("#\\/i")).should("not.exist");
1353
+ cy.get(cesc("#\\/i1")).should("not.exist");
1354
+
1355
+ cy.log("enter 10");
1356
+ cy.get(cesc("#\\/n") + " textarea").type(
1357
+ "{end}{backspace}{backspace}10{enter}",
1358
+ { force: true },
1359
+ );
1360
+
1361
+ cy.get(cesc("#\\/s1\\/a")).should("have.text", "mouse");
1362
+ cy.get(cesc("#\\/a1")).should("have.text", "mouse");
1363
+ cy.get(cesc("#\\/e")).should("have.text", "mouse");
1364
+ cy.get(cesc("#\\/e1")).should("have.text", "mouse");
1365
+ cy.get(cesc("#\\/s3\\/j")).should("have.text", "mouse");
1366
+ cy.get(cesc("#\\/j1")).should("have.text", "mouse");
1367
+
1368
+ cy.get(cesc("#\\/s1\\/b"))
1369
+ .find(".mjx-mrow")
1370
+ .eq(0)
1371
+ .invoke("text")
1372
+ .then((text) => {
1373
+ expect(text.trim()).equal("z");
1374
+ });
1375
+ cy.get(cesc("#\\/b1"))
1376
+ .find(".mjx-mrow")
1377
+ .eq(0)
1378
+ .invoke("text")
1379
+ .then((text) => {
1380
+ expect(text.trim()).equal("z");
1381
+ });
1382
+ cy.get(cesc("#\\/f"))
1383
+ .find(".mjx-mrow")
1384
+ .eq(0)
1385
+ .invoke("text")
1386
+ .then((text) => {
1387
+ expect(text.trim()).equal("z");
1388
+ });
1389
+ cy.get(cesc("#\\/f1"))
1390
+ .find(".mjx-mrow")
1391
+ .eq(0)
1392
+ .invoke("text")
1393
+ .then((text) => {
1394
+ expect(text.trim()).equal("z");
1395
+ });
1396
+ cy.get(cesc("#\\/s3\\/k"))
1397
+ .find(".mjx-mrow")
1398
+ .eq(0)
1399
+ .invoke("text")
1400
+ .then((text) => {
1401
+ expect(text.trim()).equal("z");
1402
+ });
1403
+ cy.get(cesc("#\\/k1"))
1404
+ .find(".mjx-mrow")
1405
+ .eq(0)
1406
+ .invoke("text")
1407
+ .then((text) => {
1408
+ expect(text.trim()).equal("z");
1409
+ });
1410
+
1411
+ cy.get(cesc("#\\/s1\\/c")).should("not.exist");
1412
+ cy.get(cesc("#\\/c1")).should("not.exist");
1413
+ cy.get(cesc("#\\/g")).should("not.exist");
1414
+ cy.get(cesc("#\\/g1")).should("not.exist");
1415
+ cy.get(cesc("#\\/s3\\/l")).should("not.exist");
1416
+ cy.get(cesc("#\\/l1")).should("not.exist");
1417
+
1418
+ cy.get(cesc("#\\/s1\\/d")).should("not.exist");
1419
+ cy.get(cesc("#\\/d1")).should("not.exist");
1420
+ cy.get(cesc("#\\/h")).should("not.exist");
1421
+ cy.get(cesc("#\\/h1")).should("not.exist");
1422
+ cy.get(cesc("#\\/i")).should("not.exist");
1423
+ cy.get(cesc("#\\/i1")).should("not.exist");
1424
+
1425
+ cy.log("enter -1");
1426
+ cy.get(cesc("#\\/n") + " textarea").type(
1427
+ "{ctrl+home}{shift+end}{backspace}-1{enter}",
1428
+ { force: true },
1429
+ );
1430
+
1431
+ cy.get(cesc("#\\/s1\\/a")).should("have.text", "dog");
1432
+ cy.get(cesc("#\\/a1")).should("have.text", "dog");
1433
+ cy.get(cesc("#\\/e")).should("have.text", "dog");
1434
+ cy.get(cesc("#\\/e1")).should("have.text", "dog");
1435
+ cy.get(cesc("#\\/s3\\/j")).should("have.text", "dog");
1436
+ cy.get(cesc("#\\/j1")).should("have.text", "dog");
1437
+
1438
+ cy.get(cesc("#\\/s1\\/b"))
1439
+ .find(".mjx-mrow")
1440
+ .eq(0)
1441
+ .invoke("text")
1442
+ .then((text) => {
1443
+ expect(text.trim()).equal("x");
1444
+ });
1445
+ cy.get(cesc("#\\/b1"))
1446
+ .find(".mjx-mrow")
1447
+ .eq(0)
1448
+ .invoke("text")
1449
+ .then((text) => {
1450
+ expect(text.trim()).equal("x");
1451
+ });
1452
+ cy.get(cesc("#\\/f"))
1453
+ .find(".mjx-mrow")
1454
+ .eq(0)
1455
+ .invoke("text")
1456
+ .then((text) => {
1457
+ expect(text.trim()).equal("x");
1458
+ });
1459
+ cy.get(cesc("#\\/f1"))
1460
+ .find(".mjx-mrow")
1461
+ .eq(0)
1462
+ .invoke("text")
1463
+ .then((text) => {
1464
+ expect(text.trim()).equal("x");
1465
+ });
1466
+ cy.get(cesc("#\\/s3\\/k"))
1467
+ .find(".mjx-mrow")
1468
+ .eq(0)
1469
+ .invoke("text")
1470
+ .then((text) => {
1471
+ expect(text.trim()).equal("x");
1472
+ });
1473
+ cy.get(cesc("#\\/k1"))
1474
+ .find(".mjx-mrow")
1475
+ .eq(0)
1476
+ .invoke("text")
1477
+ .then((text) => {
1478
+ expect(text.trim()).equal("x");
1479
+ });
1480
+
1481
+ cy.get(cesc("#\\/s1\\/c")).should("have.text", "optional text!");
1482
+ cy.get(cesc("#\\/c1")).should("have.text", "optional text!");
1483
+ cy.get(cesc("#\\/g")).should("have.text", "optional text!");
1484
+ cy.get(cesc("#\\/g1")).should("have.text", "optional text!");
1485
+ cy.get(cesc("#\\/s3\\/l")).should("have.text", "optional text!");
1486
+ cy.get(cesc("#\\/l1")).should("have.text", "optional text!");
1487
+
1488
+ cy.get(cesc("#\\/s1\\/d")).should("not.exist");
1489
+ cy.get(cesc("#\\/d1")).should("not.exist");
1490
+ cy.get(cesc("#\\/h")).should("not.exist");
1491
+ cy.get(cesc("#\\/h1")).should("not.exist");
1492
+ cy.get(cesc("#\\/i")).should("not.exist");
1493
+ cy.get(cesc("#\\/i1")).should("not.exist");
1494
+
1495
+ cy.log("enter x");
1496
+ cy.get(cesc("#\\/n") + " textarea").type(
1497
+ "{ctrl+home}{shift+end}{backspace}x{enter}",
1498
+ { force: true },
1499
+ );
1500
+
1501
+ cy.get(cesc("#\\/s1\\/a")).should("have.text", "mouse");
1502
+ cy.get(cesc("#\\/a1")).should("have.text", "mouse");
1503
+ cy.get(cesc("#\\/e")).should("have.text", "mouse");
1504
+ cy.get(cesc("#\\/e1")).should("have.text", "mouse");
1505
+ cy.get(cesc("#\\/s3\\/j")).should("have.text", "mouse");
1506
+ cy.get(cesc("#\\/j1")).should("have.text", "mouse");
1507
+
1508
+ cy.get(cesc("#\\/s1\\/b"))
1509
+ .find(".mjx-mrow")
1510
+ .eq(0)
1511
+ .invoke("text")
1512
+ .then((text) => {
1513
+ expect(text.trim()).equal("z");
1514
+ });
1515
+ cy.get(cesc("#\\/b1"))
1516
+ .find(".mjx-mrow")
1517
+ .eq(0)
1518
+ .invoke("text")
1519
+ .then((text) => {
1520
+ expect(text.trim()).equal("z");
1521
+ });
1522
+ cy.get(cesc("#\\/f"))
1523
+ .find(".mjx-mrow")
1524
+ .eq(0)
1525
+ .invoke("text")
1526
+ .then((text) => {
1527
+ expect(text.trim()).equal("z");
1528
+ });
1529
+ cy.get(cesc("#\\/f1"))
1530
+ .find(".mjx-mrow")
1531
+ .eq(0)
1532
+ .invoke("text")
1533
+ .then((text) => {
1534
+ expect(text.trim()).equal("z");
1535
+ });
1536
+ cy.get(cesc("#\\/s3\\/k"))
1537
+ .find(".mjx-mrow")
1538
+ .eq(0)
1539
+ .invoke("text")
1540
+ .then((text) => {
1541
+ expect(text.trim()).equal("z");
1542
+ });
1543
+ cy.get(cesc("#\\/k1"))
1544
+ .find(".mjx-mrow")
1545
+ .eq(0)
1546
+ .invoke("text")
1547
+ .then((text) => {
1548
+ expect(text.trim()).equal("z");
1549
+ });
1550
+
1551
+ cy.get(cesc("#\\/s1\\/c")).should("not.exist");
1552
+ cy.get(cesc("#\\/c1")).should("not.exist");
1553
+ cy.get(cesc("#\\/g")).should("not.exist");
1554
+ cy.get(cesc("#\\/g1")).should("not.exist");
1555
+ cy.get(cesc("#\\/s3\\/l")).should("not.exist");
1556
+ cy.get(cesc("#\\/l1")).should("not.exist");
1557
+
1558
+ cy.get(cesc("#\\/s1\\/d")).should("not.exist");
1559
+ cy.get(cesc("#\\/d1")).should("not.exist");
1560
+ cy.get(cesc("#\\/h")).should("not.exist");
1561
+ cy.get(cesc("#\\/h1")).should("not.exist");
1562
+ cy.get(cesc("#\\/i")).should("not.exist");
1563
+ cy.get(cesc("#\\/i1")).should("not.exist");
1564
+ });
1565
+
1566
+ it("references to internal and external components", () => {
1567
+ cy.window().then(async (win) => {
1568
+ win.postMessage(
1569
+ {
1570
+ doenetML: `
1571
+ <text>a</text>
1572
+ <text name="x1">dog</text>
1573
+ <text name="x2">cat</text>
1574
+ <text name="x3">mouse</text>
1575
+
1576
+ <mathinput name="n" />
1577
+ <p>original: <conditionalContent assignNames="((a) (b) c)">
1578
+ <case condition="$n<0" >
1579
+ <copy target="x1" />
1580
+ <copy target="y1" />
1581
+ <math simplify>3<math name="a1">x</math><math name="b1">a</math> + <copy target="a1" /><copy target="b1" /></math>
1582
+ </case>
1583
+ <case condition="$n <= 1" >
1584
+ <copy target="x2" />
1585
+ <copy target="y2" />
1586
+ <math simplify>4<math name="a2">y</math><math name="b2">b</math> + <copy target="a2" /><copy target="b2" /></math>
1587
+ </case>
1588
+ <else>
1589
+ <copy target="x3" />
1590
+ <copy target="y3" />
1591
+ <math simplify>5<math name="a3">z</math><math name="b3">c</math> + <copy target="a3" /><copy target="b3" /></math>
1592
+ </else>
1593
+ </conditionalContent></p>
1594
+
1595
+ <text name="y1">tree</text>
1596
+ <text name="y2">shrub</text>
1597
+ <text name="y3">bush</text>
1598
+
1599
+ <p>Selected options repeated</p>
1600
+ <copy assignNames="aa" target="a" />
1601
+ <copy assignNames="bb" target="b" />
1602
+ <copy assignNames="cc" target="c" />
1603
+
1604
+ <p>Whole thing repeated</p>
1605
+ <copy target="_conditionalcontent1" assignNames="((d) (e) f)" />
1606
+
1607
+ <p>Selected options repeated from copy</p>
1608
+ <copy assignNames="dd" target="d" />
1609
+ <copy assignNames="ee" target="e" />
1610
+ <copy assignNames="ff" target="f" />
1611
+
1612
+
1613
+ `,
1614
+ },
1615
+ "*",
1616
+ );
1617
+ });
1618
+
1619
+ cy.get(cesc("#\\/_text1")).should("have.text", "a"); // to wait until loaded
1620
+
1621
+ cy.get(cesc(`#\\/a`)).should("have.text", "mouse");
1622
+ cy.get(cesc(`#\\/b`)).should("have.text", "bush");
1623
+ cy.get(cesc(`#\\/c`))
1624
+ .find(".mjx-mrow")
1625
+ .eq(0)
1626
+ .invoke("text")
1627
+ .then((text) => {
1628
+ expect(text.trim()).equal("6cz");
1629
+ });
1630
+ cy.get(cesc(`#\\/aa`)).should("have.text", "mouse");
1631
+ cy.get(cesc(`#\\/bb`)).should("have.text", "bush");
1632
+ cy.get(cesc(`#\\/cc`))
1633
+ .find(".mjx-mrow")
1634
+ .eq(0)
1635
+ .invoke("text")
1636
+ .then((text) => {
1637
+ expect(text.trim()).equal("6cz");
1638
+ });
1639
+
1640
+ cy.get(cesc(`#\\/d`)).should("have.text", "mouse");
1641
+ cy.get(cesc(`#\\/e`)).should("have.text", "bush");
1642
+ cy.get(cesc(`#\\/f`))
1643
+ .find(".mjx-mrow")
1644
+ .eq(0)
1645
+ .invoke("text")
1646
+ .then((text) => {
1647
+ expect(text.trim()).equal("6cz");
1648
+ });
1649
+ cy.get(cesc(`#\\/dd`)).should("have.text", "mouse");
1650
+ cy.get(cesc(`#\\/ee`)).should("have.text", "bush");
1651
+ cy.get(cesc(`#\\/ff`))
1652
+ .find(".mjx-mrow")
1653
+ .eq(0)
1654
+ .invoke("text")
1655
+ .then((text) => {
1656
+ expect(text.trim()).equal("6cz");
1657
+ });
1658
+
1659
+ cy.log("enter 1");
1660
+ cy.get(cesc("#\\/n") + " textarea").type("{end}{backspace}1{enter}", {
1661
+ force: true,
1662
+ });
1663
+
1664
+ cy.get(cesc(`#\\/a`)).should("have.text", "cat");
1665
+ cy.get(cesc(`#\\/b`)).should("have.text", "shrub");
1666
+ cy.get(cesc(`#\\/c`))
1667
+ .find(".mjx-mrow")
1668
+ .eq(0)
1669
+ .invoke("text")
1670
+ .then((text) => {
1671
+ expect(text.trim()).equal("5by");
1672
+ });
1673
+ cy.get(cesc(`#\\/aa`)).should("have.text", "cat");
1674
+ cy.get(cesc(`#\\/bb`)).should("have.text", "shrub");
1675
+ cy.get(cesc(`#\\/cc`))
1676
+ .find(".mjx-mrow")
1677
+ .eq(0)
1678
+ .invoke("text")
1679
+ .then((text) => {
1680
+ expect(text.trim()).equal("5by");
1681
+ });
1682
+
1683
+ cy.get(cesc(`#\\/d`)).should("have.text", "cat");
1684
+ cy.get(cesc(`#\\/e`)).should("have.text", "shrub");
1685
+ cy.get(cesc(`#\\/f`))
1686
+ .find(".mjx-mrow")
1687
+ .eq(0)
1688
+ .invoke("text")
1689
+ .then((text) => {
1690
+ expect(text.trim()).equal("5by");
1691
+ });
1692
+ cy.get(cesc(`#\\/dd`)).should("have.text", "cat");
1693
+ cy.get(cesc(`#\\/ee`)).should("have.text", "shrub");
1694
+ cy.get(cesc(`#\\/ff`))
1695
+ .find(".mjx-mrow")
1696
+ .eq(0)
1697
+ .invoke("text")
1698
+ .then((text) => {
1699
+ expect(text.trim()).equal("5by");
1700
+ });
1701
+
1702
+ cy.log("enter -1");
1703
+ cy.get(cesc("#\\/n") + " textarea").type("{end}{backspace}-1{enter}", {
1704
+ force: true,
1705
+ });
1706
+
1707
+ cy.get(cesc(`#\\/a`)).should("have.text", "dog");
1708
+ cy.get(cesc(`#\\/b`)).should("have.text", "tree");
1709
+ cy.get(cesc(`#\\/c`))
1710
+ .find(".mjx-mrow")
1711
+ .eq(0)
1712
+ .invoke("text")
1713
+ .then((text) => {
1714
+ expect(text.trim()).equal("4ax");
1715
+ });
1716
+ cy.get(cesc(`#\\/aa`)).should("have.text", "dog");
1717
+ cy.get(cesc(`#\\/bb`)).should("have.text", "tree");
1718
+ cy.get(cesc(`#\\/cc`))
1719
+ .find(".mjx-mrow")
1720
+ .eq(0)
1721
+ .invoke("text")
1722
+ .then((text) => {
1723
+ expect(text.trim()).equal("4ax");
1724
+ });
1725
+
1726
+ cy.get(cesc(`#\\/d`)).should("have.text", "dog");
1727
+ cy.get(cesc(`#\\/e`)).should("have.text", "tree");
1728
+ cy.get(cesc(`#\\/f`))
1729
+ .find(".mjx-mrow")
1730
+ .eq(0)
1731
+ .invoke("text")
1732
+ .then((text) => {
1733
+ expect(text.trim()).equal("4ax");
1734
+ });
1735
+ cy.get(cesc(`#\\/dd`)).should("have.text", "dog");
1736
+ cy.get(cesc(`#\\/ee`)).should("have.text", "tree");
1737
+ cy.get(cesc(`#\\/ff`))
1738
+ .find(".mjx-mrow")
1739
+ .eq(0)
1740
+ .invoke("text")
1741
+ .then((text) => {
1742
+ expect(text.trim()).equal("4ax");
1743
+ });
1744
+
1745
+ cy.log("enter 10");
1746
+ cy.get(cesc("#\\/n") + " textarea").type(
1747
+ "{end}{backspace}{backspace}10{enter}",
1748
+ { force: true },
1749
+ );
1750
+
1751
+ cy.get(cesc(`#\\/a`)).should("have.text", "mouse");
1752
+ cy.get(cesc(`#\\/b`)).should("have.text", "bush");
1753
+ cy.get(cesc(`#\\/c`))
1754
+ .find(".mjx-mrow")
1755
+ .eq(0)
1756
+ .invoke("text")
1757
+ .then((text) => {
1758
+ expect(text.trim()).equal("6cz");
1759
+ });
1760
+ cy.get(cesc(`#\\/aa`)).should("have.text", "mouse");
1761
+ cy.get(cesc(`#\\/bb`)).should("have.text", "bush");
1762
+ cy.get(cesc(`#\\/cc`))
1763
+ .find(".mjx-mrow")
1764
+ .eq(0)
1765
+ .invoke("text")
1766
+ .then((text) => {
1767
+ expect(text.trim()).equal("6cz");
1768
+ });
1769
+
1770
+ cy.get(cesc(`#\\/d`)).should("have.text", "mouse");
1771
+ cy.get(cesc(`#\\/e`)).should("have.text", "bush");
1772
+ cy.get(cesc(`#\\/f`))
1773
+ .find(".mjx-mrow")
1774
+ .eq(0)
1775
+ .invoke("text")
1776
+ .then((text) => {
1777
+ expect(text.trim()).equal("6cz");
1778
+ });
1779
+ cy.get(cesc(`#\\/dd`)).should("have.text", "mouse");
1780
+ cy.get(cesc(`#\\/ee`)).should("have.text", "bush");
1781
+ cy.get(cesc(`#\\/ff`))
1782
+ .find(".mjx-mrow")
1783
+ .eq(0)
1784
+ .invoke("text")
1785
+ .then((text) => {
1786
+ expect(text.trim()).equal("6cz");
1787
+ });
1788
+ });
1789
+
1790
+ it("references to internal and external components, new namespace", () => {
1791
+ cy.window().then(async (win) => {
1792
+ win.postMessage(
1793
+ {
1794
+ doenetML: `
1795
+ <text>a</text>
1796
+ <text name="x1">dog</text>
1797
+ <text name="x2">cat</text>
1798
+ <text name="x3">mouse</text>
1799
+
1800
+ <mathinput name="n" />
1801
+ <p>original: <conditionalContent assignNames="a">
1802
+ <case condition="$n<0" newNamespace >
1803
+ <copy target="../x1" assignNames="animal" />
1804
+ <copy target="../y1" assignNames="plant" />
1805
+ <math simplify name="p">3<math name="x">x</math><math name="a">a</math> + <copy target="x" /><copy target="a" /></math>
1806
+ </case>
1807
+ <case condition="$n <= 1" newNamespace >
1808
+ <copy target="../x2" assignNames="animal" />
1809
+ <copy target="../y2" assignNames="plant" />
1810
+ <math simplify name="p">4<math name="x">y</math><math name="a">b</math> + <copy target="x" /><copy target="a" /></math>
1811
+ </case>
1812
+ <else newNamespace>
1813
+ <copy target="../x3" assignNames="animal" />
1814
+ <copy target="../y3" assignNames="plant" />
1815
+ <math simplify name="p">5<math name="x">z</math><math name="a">c</math> + <copy target="x" /><copy target="a" /></math>
1816
+ </else>
1817
+ </conditionalContent></p>
1818
+
1819
+ <text name="y1">tree</text>
1820
+ <text name="y2">shrub</text>
1821
+ <text name="y3">bush</text>
1822
+
1823
+ <p>Selected options repeated</p>
1824
+ <copy assignNames="animal" target="a/animal" />
1825
+ <copy assignNames="plant" target="a/plant" />
1826
+ <copy assignNames="p" target="a/p" />
1827
+ <copy assignNames="xx" target="a/x" />
1828
+ <copy assignNames="aa" target="a/a" />
1829
+
1830
+ <p>Whole thing repeated</p>
1831
+ <copy target="_conditionalcontent1" assignNames="b" />
1832
+
1833
+ <p>Selected options repeated from copy</p>
1834
+ <copy assignNames="animalcopy" target="b/animal" />
1835
+ <copy assignNames="plantcopy" target="b/plant" />
1836
+ <copy assignNames="pcopy" target="b/p" />
1837
+ <copy assignNames="xxcopy" target="b/x" />
1838
+ <copy assignNames="aacopy" target="b/a" />
1839
+
1840
+ `,
1841
+ },
1842
+ "*",
1843
+ );
1844
+ });
1845
+
1846
+ cy.get(cesc("#\\/_text1")).should("have.text", "a"); // to wait until loaded
1847
+
1848
+ cy.get(cesc(`#\\/a\\/animal`)).should("have.text", "mouse");
1849
+ cy.get(cesc(`#\\/a\\/plant`)).should("have.text", "bush");
1850
+ cy.get(cesc(`#\\/a\\/p`))
1851
+ .find(".mjx-mrow")
1852
+ .eq(0)
1853
+ .invoke("text")
1854
+ .then((text) => {
1855
+ expect(text.trim()).equal("6cz");
1856
+ });
1857
+
1858
+ cy.get(cesc(`#\\/animal`)).should("have.text", "mouse");
1859
+ cy.get(cesc(`#\\/plant`)).should("have.text", "bush");
1860
+ cy.get(cesc(`#\\/p`))
1861
+ .find(".mjx-mrow")
1862
+ .eq(0)
1863
+ .invoke("text")
1864
+ .then((text) => {
1865
+ expect(text.trim()).equal("6cz");
1866
+ });
1867
+ cy.get(cesc(`#\\/xx`))
1868
+ .find(".mjx-mrow")
1869
+ .eq(0)
1870
+ .invoke("text")
1871
+ .then((text) => {
1872
+ expect(text.trim()).equal("z");
1873
+ });
1874
+ cy.get(cesc(`#\\/aa`))
1875
+ .find(".mjx-mrow")
1876
+ .eq(0)
1877
+ .invoke("text")
1878
+ .then((text) => {
1879
+ expect(text.trim()).equal("c");
1880
+ });
1881
+
1882
+ cy.get(cesc(`#\\/b\\/animal`)).should("have.text", "mouse");
1883
+ cy.get(cesc(`#\\/b\\/plant`)).should("have.text", "bush");
1884
+ cy.get(cesc(`#\\/b\\/p`))
1885
+ .find(".mjx-mrow")
1886
+ .eq(0)
1887
+ .invoke("text")
1888
+ .then((text) => {
1889
+ expect(text.trim()).equal("6cz");
1890
+ });
1891
+
1892
+ cy.get(cesc(`#\\/animalcopy`)).should("have.text", "mouse");
1893
+ cy.get(cesc(`#\\/plantcopy`)).should("have.text", "bush");
1894
+ cy.get(cesc(`#\\/pcopy`))
1895
+ .find(".mjx-mrow")
1896
+ .eq(0)
1897
+ .invoke("text")
1898
+ .then((text) => {
1899
+ expect(text.trim()).equal("6cz");
1900
+ });
1901
+ cy.get(cesc(`#\\/xxcopy`))
1902
+ .find(".mjx-mrow")
1903
+ .eq(0)
1904
+ .invoke("text")
1905
+ .then((text) => {
1906
+ expect(text.trim()).equal("z");
1907
+ });
1908
+ cy.get(cesc(`#\\/aacopy`))
1909
+ .find(".mjx-mrow")
1910
+ .eq(0)
1911
+ .invoke("text")
1912
+ .then((text) => {
1913
+ expect(text.trim()).equal("c");
1914
+ });
1915
+
1916
+ cy.log("enter 1");
1917
+ cy.get(cesc("#\\/n") + " textarea").type(
1918
+ "{end}{backspace}{backspace}1{enter}",
1919
+ { force: true },
1920
+ );
1921
+
1922
+ cy.get(cesc(`#\\/a\\/animal`)).should("have.text", "cat");
1923
+ cy.get(cesc(`#\\/a\\/plant`)).should("have.text", "shrub");
1924
+ cy.get(cesc(`#\\/a\\/p`))
1925
+ .find(".mjx-mrow")
1926
+ .eq(0)
1927
+ .invoke("text")
1928
+ .then((text) => {
1929
+ expect(text.trim()).equal("5by");
1930
+ });
1931
+
1932
+ cy.get(cesc(`#\\/animal`)).should("have.text", "cat");
1933
+ cy.get(cesc(`#\\/plant`)).should("have.text", "shrub");
1934
+ cy.get(cesc(`#\\/p`))
1935
+ .find(".mjx-mrow")
1936
+ .eq(0)
1937
+ .invoke("text")
1938
+ .then((text) => {
1939
+ expect(text.trim()).equal("5by");
1940
+ });
1941
+ cy.get(cesc(`#\\/xx`))
1942
+ .find(".mjx-mrow")
1943
+ .eq(0)
1944
+ .invoke("text")
1945
+ .then((text) => {
1946
+ expect(text.trim()).equal("y");
1947
+ });
1948
+ cy.get(cesc(`#\\/aa`))
1949
+ .find(".mjx-mrow")
1950
+ .eq(0)
1951
+ .invoke("text")
1952
+ .then((text) => {
1953
+ expect(text.trim()).equal("b");
1954
+ });
1955
+
1956
+ cy.get(cesc(`#\\/b\\/animal`)).should("have.text", "cat");
1957
+ cy.get(cesc(`#\\/b\\/plant`)).should("have.text", "shrub");
1958
+ cy.get(cesc(`#\\/b\\/p`))
1959
+ .find(".mjx-mrow")
1960
+ .eq(0)
1961
+ .invoke("text")
1962
+ .then((text) => {
1963
+ expect(text.trim()).equal("5by");
1964
+ });
1965
+
1966
+ cy.get(cesc(`#\\/animalcopy`)).should("have.text", "cat");
1967
+ cy.get(cesc(`#\\/plantcopy`)).should("have.text", "shrub");
1968
+ cy.get(cesc(`#\\/pcopy`))
1969
+ .find(".mjx-mrow")
1970
+ .eq(0)
1971
+ .invoke("text")
1972
+ .then((text) => {
1973
+ expect(text.trim()).equal("5by");
1974
+ });
1975
+ cy.get(cesc(`#\\/xxcopy`))
1976
+ .find(".mjx-mrow")
1977
+ .eq(0)
1978
+ .invoke("text")
1979
+ .then((text) => {
1980
+ expect(text.trim()).equal("y");
1981
+ });
1982
+ cy.get(cesc(`#\\/aacopy`))
1983
+ .find(".mjx-mrow")
1984
+ .eq(0)
1985
+ .invoke("text")
1986
+ .then((text) => {
1987
+ expect(text.trim()).equal("b");
1988
+ });
1989
+
1990
+ cy.log("enter -1");
1991
+ cy.get(cesc("#\\/n") + " textarea").type(
1992
+ "{end}{backspace}{backspace}-1{enter}",
1993
+ { force: true },
1994
+ );
1995
+
1996
+ cy.get(cesc(`#\\/a\\/animal`)).should("have.text", "dog");
1997
+ cy.get(cesc(`#\\/a\\/plant`)).should("have.text", "tree");
1998
+ cy.get(cesc(`#\\/a\\/p`))
1999
+ .find(".mjx-mrow")
2000
+ .eq(0)
2001
+ .invoke("text")
2002
+ .then((text) => {
2003
+ expect(text.trim()).equal("4ax");
2004
+ });
2005
+
2006
+ cy.get(cesc(`#\\/animal`)).should("have.text", "dog");
2007
+ cy.get(cesc(`#\\/plant`)).should("have.text", "tree");
2008
+ cy.get(cesc(`#\\/p`))
2009
+ .find(".mjx-mrow")
2010
+ .eq(0)
2011
+ .invoke("text")
2012
+ .then((text) => {
2013
+ expect(text.trim()).equal("4ax");
2014
+ });
2015
+ cy.get(cesc(`#\\/xx`))
2016
+ .find(".mjx-mrow")
2017
+ .eq(0)
2018
+ .invoke("text")
2019
+ .then((text) => {
2020
+ expect(text.trim()).equal("x");
2021
+ });
2022
+ cy.get(cesc(`#\\/aa`))
2023
+ .find(".mjx-mrow")
2024
+ .eq(0)
2025
+ .invoke("text")
2026
+ .then((text) => {
2027
+ expect(text.trim()).equal("a");
2028
+ });
2029
+
2030
+ cy.get(cesc(`#\\/b\\/animal`)).should("have.text", "dog");
2031
+ cy.get(cesc(`#\\/b\\/plant`)).should("have.text", "tree");
2032
+ cy.get(cesc(`#\\/b\\/p`))
2033
+ .find(".mjx-mrow")
2034
+ .eq(0)
2035
+ .invoke("text")
2036
+ .then((text) => {
2037
+ expect(text.trim()).equal("4ax");
2038
+ });
2039
+
2040
+ cy.get(cesc(`#\\/animalcopy`)).should("have.text", "dog");
2041
+ cy.get(cesc(`#\\/plantcopy`)).should("have.text", "tree");
2042
+ cy.get(cesc(`#\\/pcopy`))
2043
+ .find(".mjx-mrow")
2044
+ .eq(0)
2045
+ .invoke("text")
2046
+ .then((text) => {
2047
+ expect(text.trim()).equal("4ax");
2048
+ });
2049
+ cy.get(cesc(`#\\/xxcopy`))
2050
+ .find(".mjx-mrow")
2051
+ .eq(0)
2052
+ .invoke("text")
2053
+ .then((text) => {
2054
+ expect(text.trim()).equal("x");
2055
+ });
2056
+ cy.get(cesc(`#\\/aacopy`))
2057
+ .find(".mjx-mrow")
2058
+ .eq(0)
2059
+ .invoke("text")
2060
+ .then((text) => {
2061
+ expect(text.trim()).equal("a");
2062
+ });
2063
+
2064
+ cy.log("enter 10");
2065
+ cy.get(cesc("#\\/n") + " textarea").type(
2066
+ "{ctrl+home}{shift+end}{backspace}10{enter}",
2067
+ { force: true },
2068
+ );
2069
+
2070
+ cy.get(cesc(`#\\/a\\/animal`)).should("have.text", "mouse");
2071
+ cy.get(cesc(`#\\/a\\/plant`)).should("have.text", "bush");
2072
+ cy.get(cesc(`#\\/a\\/p`))
2073
+ .find(".mjx-mrow")
2074
+ .eq(0)
2075
+ .invoke("text")
2076
+ .then((text) => {
2077
+ expect(text.trim()).equal("6cz");
2078
+ });
2079
+
2080
+ cy.get(cesc(`#\\/animal`)).should("have.text", "mouse");
2081
+ cy.get(cesc(`#\\/plant`)).should("have.text", "bush");
2082
+ cy.get(cesc(`#\\/p`))
2083
+ .find(".mjx-mrow")
2084
+ .eq(0)
2085
+ .invoke("text")
2086
+ .then((text) => {
2087
+ expect(text.trim()).equal("6cz");
2088
+ });
2089
+ cy.get(cesc(`#\\/xx`))
2090
+ .find(".mjx-mrow")
2091
+ .eq(0)
2092
+ .invoke("text")
2093
+ .then((text) => {
2094
+ expect(text.trim()).equal("z");
2095
+ });
2096
+ cy.get(cesc(`#\\/aa`))
2097
+ .find(".mjx-mrow")
2098
+ .eq(0)
2099
+ .invoke("text")
2100
+ .then((text) => {
2101
+ expect(text.trim()).equal("c");
2102
+ });
2103
+
2104
+ cy.get(cesc(`#\\/b\\/animal`)).should("have.text", "mouse");
2105
+ cy.get(cesc(`#\\/b\\/plant`)).should("have.text", "bush");
2106
+ cy.get(cesc(`#\\/b\\/p`))
2107
+ .find(".mjx-mrow")
2108
+ .eq(0)
2109
+ .invoke("text")
2110
+ .then((text) => {
2111
+ expect(text.trim()).equal("6cz");
2112
+ });
2113
+
2114
+ cy.get(cesc(`#\\/animalcopy`)).should("have.text", "mouse");
2115
+ cy.get(cesc(`#\\/plantcopy`)).should("have.text", "bush");
2116
+ cy.get(cesc(`#\\/pcopy`))
2117
+ .find(".mjx-mrow")
2118
+ .eq(0)
2119
+ .invoke("text")
2120
+ .then((text) => {
2121
+ expect(text.trim()).equal("6cz");
2122
+ });
2123
+ cy.get(cesc(`#\\/xxcopy`))
2124
+ .find(".mjx-mrow")
2125
+ .eq(0)
2126
+ .invoke("text")
2127
+ .then((text) => {
2128
+ expect(text.trim()).equal("z");
2129
+ });
2130
+ cy.get(cesc(`#\\/aacopy`))
2131
+ .find(".mjx-mrow")
2132
+ .eq(0)
2133
+ .invoke("text")
2134
+ .then((text) => {
2135
+ expect(text.trim()).equal("c");
2136
+ });
2137
+ });
2138
+
2139
+ it("references to internal and external components, multiple layers of new namespaces", () => {
2140
+ cy.window().then(async (win) => {
2141
+ win.postMessage(
2142
+ {
2143
+ doenetML: `
2144
+ <text>a</text>
2145
+ <text name="x1">dog</text>
2146
+ <text name="x2">cat</text>
2147
+ <text name="x3">mouse</text>
2148
+
2149
+ <mathinput name="n" />
2150
+ <p>original: <conditionalContent name="s" assignNames="a" newNamespace>
2151
+ <case newNamespace condition="$(../n) < 0" >
2152
+ <copy target="../../x1" assignNames="animal" />
2153
+ <copy target="../../y1" assignNames="plant" />
2154
+ <math simplify name="p">3<math name="x">x</math><math name="a">a</math> + <copy target="x" /><copy target="a" /></math>
2155
+ </case>
2156
+ <case newNamespace condition="$(../n) <= 1" >
2157
+ <copy target="../../x2" assignNames="animal" />
2158
+ <copy target="../../y2" assignNames="plant" />
2159
+ <math simplify name="p">4<math name="x">y</math><math name="a">b</math> + <copy target="x" /><copy target="a" /></math>
2160
+ </case>
2161
+ <else newNamespace>
2162
+ <copy target="../../x3" assignNames="animal" />
2163
+ <copy target="../../y3" assignNames="plant" />
2164
+ <math simplify name="p">5<math name="x">z</math><math name="a">c</math> + <copy target="x" /><copy target="a" /></math>
2165
+ </else>
2166
+ </conditionalContent></p>
2167
+
2168
+ <text name="y1">tree</text>
2169
+ <text name="y2">shrub</text>
2170
+ <text name="y3">bush</text>
2171
+
2172
+ <p>Selected options repeated</p>
2173
+ <copy assignNames="animal" target="s/a/animal" />
2174
+ <copy assignNames="plant" target="s/a/plant" />
2175
+ <copy assignNames="p" target="s/a/p" />
2176
+ <copy assignNames="xx" target="s/a/x" />
2177
+ <copy assignNames="aa" target="s/a/a" />
2178
+
2179
+ <p>Whole thing repeated</p>
2180
+ <copy target="s" assignNames="b" />
2181
+
2182
+ <p>Selected options repeated from copy</p>
2183
+ <copy assignNames="animalcopy" target="b/animal" />
2184
+ <copy assignNames="plantcopy" target="b/plant" />
2185
+ <copy assignNames="pcopy" target="b/p" />
2186
+ <copy assignNames="xxcopy" target="b/x" />
2187
+ <copy assignNames="aacopy" target="b/a" />
2188
+
2189
+ `,
2190
+ },
2191
+ "*",
2192
+ );
2193
+ });
2194
+
2195
+ cy.get(cesc("#\\/_text1")).should("have.text", "a"); // to wait until loaded
2196
+
2197
+ cy.get(cesc(`#\\/s\\/a\\/animal`)).should("have.text", "mouse");
2198
+ cy.get(cesc(`#\\/s\\/a\\/plant`)).should("have.text", "bush");
2199
+ cy.get(cesc(`#\\/s\\/a\\/p`))
2200
+ .find(".mjx-mrow")
2201
+ .eq(0)
2202
+ .invoke("text")
2203
+ .then((text) => {
2204
+ expect(text.trim()).equal("6cz");
2205
+ });
2206
+
2207
+ cy.get(cesc(`#\\/animal`)).should("have.text", "mouse");
2208
+ cy.get(cesc(`#\\/plant`)).should("have.text", "bush");
2209
+ cy.get(cesc(`#\\/p`))
2210
+ .find(".mjx-mrow")
2211
+ .eq(0)
2212
+ .invoke("text")
2213
+ .then((text) => {
2214
+ expect(text.trim()).equal("6cz");
2215
+ });
2216
+ cy.get(cesc(`#\\/xx`))
2217
+ .find(".mjx-mrow")
2218
+ .eq(0)
2219
+ .invoke("text")
2220
+ .then((text) => {
2221
+ expect(text.trim()).equal("z");
2222
+ });
2223
+ cy.get(cesc(`#\\/aa`))
2224
+ .find(".mjx-mrow")
2225
+ .eq(0)
2226
+ .invoke("text")
2227
+ .then((text) => {
2228
+ expect(text.trim()).equal("c");
2229
+ });
2230
+
2231
+ cy.get(cesc(`#\\/b\\/animal`)).should("have.text", "mouse");
2232
+ cy.get(cesc(`#\\/b\\/plant`)).should("have.text", "bush");
2233
+ cy.get(cesc(`#\\/b\\/p`))
2234
+ .find(".mjx-mrow")
2235
+ .eq(0)
2236
+ .invoke("text")
2237
+ .then((text) => {
2238
+ expect(text.trim()).equal("6cz");
2239
+ });
2240
+
2241
+ cy.get(cesc(`#\\/animalcopy`)).should("have.text", "mouse");
2242
+ cy.get(cesc(`#\\/plantcopy`)).should("have.text", "bush");
2243
+ cy.get(cesc(`#\\/pcopy`))
2244
+ .find(".mjx-mrow")
2245
+ .eq(0)
2246
+ .invoke("text")
2247
+ .then((text) => {
2248
+ expect(text.trim()).equal("6cz");
2249
+ });
2250
+ cy.get(cesc(`#\\/xxcopy`))
2251
+ .find(".mjx-mrow")
2252
+ .eq(0)
2253
+ .invoke("text")
2254
+ .then((text) => {
2255
+ expect(text.trim()).equal("z");
2256
+ });
2257
+ cy.get(cesc(`#\\/aacopy`))
2258
+ .find(".mjx-mrow")
2259
+ .eq(0)
2260
+ .invoke("text")
2261
+ .then((text) => {
2262
+ expect(text.trim()).equal("c");
2263
+ });
2264
+
2265
+ cy.log("enter 1");
2266
+ cy.get(cesc("#\\/n") + " textarea").type(
2267
+ "{end}{backspace}{backspace}1{enter}",
2268
+ { force: true },
2269
+ );
2270
+
2271
+ cy.get(cesc(`#\\/s\\/a\\/animal`)).should("have.text", "cat");
2272
+ cy.get(cesc(`#\\/s\\/a\\/plant`)).should("have.text", "shrub");
2273
+ cy.get(cesc(`#\\/s\\/a\\/p`))
2274
+ .find(".mjx-mrow")
2275
+ .eq(0)
2276
+ .invoke("text")
2277
+ .then((text) => {
2278
+ expect(text.trim()).equal("5by");
2279
+ });
2280
+
2281
+ cy.get(cesc(`#\\/animal`)).should("have.text", "cat");
2282
+ cy.get(cesc(`#\\/plant`)).should("have.text", "shrub");
2283
+ cy.get(cesc(`#\\/p`))
2284
+ .find(".mjx-mrow")
2285
+ .eq(0)
2286
+ .invoke("text")
2287
+ .then((text) => {
2288
+ expect(text.trim()).equal("5by");
2289
+ });
2290
+ cy.get(cesc(`#\\/xx`))
2291
+ .find(".mjx-mrow")
2292
+ .eq(0)
2293
+ .invoke("text")
2294
+ .then((text) => {
2295
+ expect(text.trim()).equal("y");
2296
+ });
2297
+ cy.get(cesc(`#\\/aa`))
2298
+ .find(".mjx-mrow")
2299
+ .eq(0)
2300
+ .invoke("text")
2301
+ .then((text) => {
2302
+ expect(text.trim()).equal("b");
2303
+ });
2304
+
2305
+ cy.get(cesc(`#\\/b\\/animal`)).should("have.text", "cat");
2306
+ cy.get(cesc(`#\\/b\\/plant`)).should("have.text", "shrub");
2307
+ cy.get(cesc(`#\\/b\\/p`))
2308
+ .find(".mjx-mrow")
2309
+ .eq(0)
2310
+ .invoke("text")
2311
+ .then((text) => {
2312
+ expect(text.trim()).equal("5by");
2313
+ });
2314
+
2315
+ cy.get(cesc(`#\\/animalcopy`)).should("have.text", "cat");
2316
+ cy.get(cesc(`#\\/plantcopy`)).should("have.text", "shrub");
2317
+ cy.get(cesc(`#\\/pcopy`))
2318
+ .find(".mjx-mrow")
2319
+ .eq(0)
2320
+ .invoke("text")
2321
+ .then((text) => {
2322
+ expect(text.trim()).equal("5by");
2323
+ });
2324
+ cy.get(cesc(`#\\/xxcopy`))
2325
+ .find(".mjx-mrow")
2326
+ .eq(0)
2327
+ .invoke("text")
2328
+ .then((text) => {
2329
+ expect(text.trim()).equal("y");
2330
+ });
2331
+ cy.get(cesc(`#\\/aacopy`))
2332
+ .find(".mjx-mrow")
2333
+ .eq(0)
2334
+ .invoke("text")
2335
+ .then((text) => {
2336
+ expect(text.trim()).equal("b");
2337
+ });
2338
+
2339
+ cy.log("enter -1");
2340
+ cy.get(cesc("#\\/n") + " textarea").type(
2341
+ "{end}{backspace}{backspace}-1{enter}",
2342
+ { force: true },
2343
+ );
2344
+
2345
+ cy.get(cesc(`#\\/s\\/a\\/animal`)).should("have.text", "dog");
2346
+ cy.get(cesc(`#\\/s\\/a\\/plant`)).should("have.text", "tree");
2347
+ cy.get(cesc(`#\\/s\\/a\\/p`))
2348
+ .find(".mjx-mrow")
2349
+ .eq(0)
2350
+ .invoke("text")
2351
+ .then((text) => {
2352
+ expect(text.trim()).equal("4ax");
2353
+ });
2354
+
2355
+ cy.get(cesc(`#\\/animal`)).should("have.text", "dog");
2356
+ cy.get(cesc(`#\\/plant`)).should("have.text", "tree");
2357
+ cy.get(cesc(`#\\/p`))
2358
+ .find(".mjx-mrow")
2359
+ .eq(0)
2360
+ .invoke("text")
2361
+ .then((text) => {
2362
+ expect(text.trim()).equal("4ax");
2363
+ });
2364
+ cy.get(cesc(`#\\/xx`))
2365
+ .find(".mjx-mrow")
2366
+ .eq(0)
2367
+ .invoke("text")
2368
+ .then((text) => {
2369
+ expect(text.trim()).equal("x");
2370
+ });
2371
+ cy.get(cesc(`#\\/aa`))
2372
+ .find(".mjx-mrow")
2373
+ .eq(0)
2374
+ .invoke("text")
2375
+ .then((text) => {
2376
+ expect(text.trim()).equal("a");
2377
+ });
2378
+
2379
+ cy.get(cesc(`#\\/b\\/animal`)).should("have.text", "dog");
2380
+ cy.get(cesc(`#\\/b\\/plant`)).should("have.text", "tree");
2381
+ cy.get(cesc(`#\\/b\\/p`))
2382
+ .find(".mjx-mrow")
2383
+ .eq(0)
2384
+ .invoke("text")
2385
+ .then((text) => {
2386
+ expect(text.trim()).equal("4ax");
2387
+ });
2388
+
2389
+ cy.get(cesc(`#\\/animalcopy`)).should("have.text", "dog");
2390
+ cy.get(cesc(`#\\/plantcopy`)).should("have.text", "tree");
2391
+ cy.get(cesc(`#\\/pcopy`))
2392
+ .find(".mjx-mrow")
2393
+ .eq(0)
2394
+ .invoke("text")
2395
+ .then((text) => {
2396
+ expect(text.trim()).equal("4ax");
2397
+ });
2398
+ cy.get(cesc(`#\\/xxcopy`))
2399
+ .find(".mjx-mrow")
2400
+ .eq(0)
2401
+ .invoke("text")
2402
+ .then((text) => {
2403
+ expect(text.trim()).equal("x");
2404
+ });
2405
+ cy.get(cesc(`#\\/aacopy`))
2406
+ .find(".mjx-mrow")
2407
+ .eq(0)
2408
+ .invoke("text")
2409
+ .then((text) => {
2410
+ expect(text.trim()).equal("a");
2411
+ });
2412
+
2413
+ cy.log("enter 10");
2414
+ cy.get(cesc("#\\/n") + " textarea").type(
2415
+ "{ctrl+home}{shift+end}{backspace}10{enter}",
2416
+ { force: true },
2417
+ );
2418
+
2419
+ cy.get(cesc(`#\\/s\\/a\\/animal`)).should("have.text", "mouse");
2420
+ cy.get(cesc(`#\\/s\\/a\\/plant`)).should("have.text", "bush");
2421
+ cy.get(cesc(`#\\/s\\/a\\/p`))
2422
+ .find(".mjx-mrow")
2423
+ .eq(0)
2424
+ .invoke("text")
2425
+ .then((text) => {
2426
+ expect(text.trim()).equal("6cz");
2427
+ });
2428
+
2429
+ cy.get(cesc(`#\\/animal`)).should("have.text", "mouse");
2430
+ cy.get(cesc(`#\\/plant`)).should("have.text", "bush");
2431
+ cy.get(cesc(`#\\/p`))
2432
+ .find(".mjx-mrow")
2433
+ .eq(0)
2434
+ .invoke("text")
2435
+ .then((text) => {
2436
+ expect(text.trim()).equal("6cz");
2437
+ });
2438
+ cy.get(cesc(`#\\/xx`))
2439
+ .find(".mjx-mrow")
2440
+ .eq(0)
2441
+ .invoke("text")
2442
+ .then((text) => {
2443
+ expect(text.trim()).equal("z");
2444
+ });
2445
+ cy.get(cesc(`#\\/aa`))
2446
+ .find(".mjx-mrow")
2447
+ .eq(0)
2448
+ .invoke("text")
2449
+ .then((text) => {
2450
+ expect(text.trim()).equal("c");
2451
+ });
2452
+
2453
+ cy.get(cesc(`#\\/b\\/animal`)).should("have.text", "mouse");
2454
+ cy.get(cesc(`#\\/b\\/plant`)).should("have.text", "bush");
2455
+ cy.get(cesc(`#\\/b\\/p`))
2456
+ .find(".mjx-mrow")
2457
+ .eq(0)
2458
+ .invoke("text")
2459
+ .then((text) => {
2460
+ expect(text.trim()).equal("6cz");
2461
+ });
2462
+
2463
+ cy.get(cesc(`#\\/animalcopy`)).should("have.text", "mouse");
2464
+ cy.get(cesc(`#\\/plantcopy`)).should("have.text", "bush");
2465
+ cy.get(cesc(`#\\/pcopy`))
2466
+ .find(".mjx-mrow")
2467
+ .eq(0)
2468
+ .invoke("text")
2469
+ .then((text) => {
2470
+ expect(text.trim()).equal("6cz");
2471
+ });
2472
+ cy.get(cesc(`#\\/xxcopy`))
2473
+ .find(".mjx-mrow")
2474
+ .eq(0)
2475
+ .invoke("text")
2476
+ .then((text) => {
2477
+ expect(text.trim()).equal("z");
2478
+ });
2479
+ cy.get(cesc(`#\\/aacopy`))
2480
+ .find(".mjx-mrow")
2481
+ .eq(0)
2482
+ .invoke("text")
2483
+ .then((text) => {
2484
+ expect(text.trim()).equal("c");
2485
+ });
2486
+ });
2487
+
2488
+ it("references to internal and external components, inconsistent new namespaces", () => {
2489
+ // not sure why would want to do this, as give inconsistent behavior
2490
+ // depending on which option is chosen
2491
+ // but, we handle it gracefully
2492
+ cy.window().then(async (win) => {
2493
+ win.postMessage(
2494
+ {
2495
+ doenetML: `
2496
+ <text>a</text>
2497
+ <text name="x1">dog</text>
2498
+ <text name="x2">cat</text>
2499
+ <text name="x3">mouse</text>
2500
+
2501
+ <mathinput name="n" />
2502
+ <p>original: <conditionalContent assignNames="a">
2503
+ <case condition="$n<0" >
2504
+ <copy target="x1" assignNames="theanimal" />
2505
+ <copy target="y1" assignNames="theplant" />
2506
+ <math simplify name="thep">3<math name="thex">x</math><math name="thea">a</math> + <copy target="thex" /><copy target="thea" /></math>
2507
+ </case>
2508
+ <case newNamespace condition="$n <= 1" >
2509
+ <copy target="../x2" assignNames="animal" />
2510
+ <copy target="../y2" assignNames="plant" />
2511
+ <math simplify name="p">4<math name="x">y</math><math name="a">b</math> + <copy target="x" /><copy target="a" /></math>
2512
+ </case>
2513
+ <else newNamespace>
2514
+ <copy target="../x3" assignNames="animal" />
2515
+ <copy target="../y3" assignNames="plant" />
2516
+ <math simplify name="p">5<math name="x">z</math><math name="a">c</math> + <copy target="x" /><copy target="a" /></math>
2517
+ </else>
2518
+ </conditionalContent></p>
2519
+
2520
+ <text name="y1">tree</text>
2521
+ <text name="y2">shrub</text>
2522
+ <text name="y3">bush</text>
2523
+
2524
+ <p>Selected options repeated</p>
2525
+ <copy assignNames="animal" target="a/animal" />
2526
+ <copy assignNames="plant" target="a/plant" />
2527
+ <copy assignNames="p" target="a/p" />
2528
+ <copy assignNames="xx" target="a/x" />
2529
+ <copy assignNames="aa" target="a/a" />
2530
+
2531
+ <p>Whole thing repeated</p>
2532
+ <p name="repeat"><copy target="_conditionalcontent1" assignNames="b" /></p>
2533
+
2534
+ <p>Selected options repeated from copy</p>
2535
+ <copy assignNames="animalcopy" target="b/animal" />
2536
+ <copy assignNames="plantcopy" target="b/plant" />
2537
+ <copy assignNames="pcopy" target="b/p" />
2538
+ <copy assignNames="xxcopy" target="b/x" />
2539
+ <copy assignNames="aacopy" target="b/a" />
2540
+
2541
+ `,
2542
+ },
2543
+ "*",
2544
+ );
2545
+ });
2546
+
2547
+ cy.get(cesc("#\\/_text1")).should("have.text", "a"); // to wait until loaded
2548
+
2549
+ cy.get(cesc(`#\\/a\\/animal`)).should("have.text", "mouse");
2550
+ cy.get(cesc(`#\\/a\\/plant`)).should("have.text", "bush");
2551
+ cy.get(cesc(`#\\/a\\/p`))
2552
+ .find(".mjx-mrow")
2553
+ .eq(0)
2554
+ .invoke("text")
2555
+ .then((text) => {
2556
+ expect(text.trim()).equal("6cz");
2557
+ });
2558
+
2559
+ cy.get(cesc(`#\\/animal`)).should("have.text", "mouse");
2560
+ cy.get(cesc(`#\\/plant`)).should("have.text", "bush");
2561
+ cy.get(cesc(`#\\/p`))
2562
+ .find(".mjx-mrow")
2563
+ .eq(0)
2564
+ .invoke("text")
2565
+ .then((text) => {
2566
+ expect(text.trim()).equal("6cz");
2567
+ });
2568
+ cy.get(cesc(`#\\/xx`))
2569
+ .find(".mjx-mrow")
2570
+ .eq(0)
2571
+ .invoke("text")
2572
+ .then((text) => {
2573
+ expect(text.trim()).equal("z");
2574
+ });
2575
+ cy.get(cesc(`#\\/aa`))
2576
+ .find(".mjx-mrow")
2577
+ .eq(0)
2578
+ .invoke("text")
2579
+ .then((text) => {
2580
+ expect(text.trim()).equal("c");
2581
+ });
2582
+
2583
+ cy.get(cesc(`#\\/b\\/animal`)).should("have.text", "mouse");
2584
+ cy.get(cesc(`#\\/b\\/plant`)).should("have.text", "bush");
2585
+ cy.get(cesc(`#\\/b\\/p`))
2586
+ .find(".mjx-mrow")
2587
+ .eq(0)
2588
+ .invoke("text")
2589
+ .then((text) => {
2590
+ expect(text.trim()).equal("6cz");
2591
+ });
2592
+
2593
+ cy.get(cesc(`#\\/animalcopy`)).should("have.text", "mouse");
2594
+ cy.get(cesc(`#\\/plantcopy`)).should("have.text", "bush");
2595
+ cy.get(cesc(`#\\/pcopy`))
2596
+ .find(".mjx-mrow")
2597
+ .eq(0)
2598
+ .invoke("text")
2599
+ .then((text) => {
2600
+ expect(text.trim()).equal("6cz");
2601
+ });
2602
+ cy.get(cesc(`#\\/xxcopy`))
2603
+ .find(".mjx-mrow")
2604
+ .eq(0)
2605
+ .invoke("text")
2606
+ .then((text) => {
2607
+ expect(text.trim()).equal("z");
2608
+ });
2609
+ cy.get(cesc(`#\\/aacopy`))
2610
+ .find(".mjx-mrow")
2611
+ .eq(0)
2612
+ .invoke("text")
2613
+ .then((text) => {
2614
+ expect(text.trim()).equal("c");
2615
+ });
2616
+
2617
+ cy.log("enter 1");
2618
+ cy.get(cesc("#\\/n") + " textarea").type(
2619
+ "{end}{backspace}{backspace}1{enter}",
2620
+ { force: true },
2621
+ );
2622
+
2623
+ cy.get(cesc(`#\\/a\\/animal`)).should("have.text", "cat");
2624
+ cy.get(cesc(`#\\/a\\/plant`)).should("have.text", "shrub");
2625
+ cy.get(cesc(`#\\/a\\/p`))
2626
+ .find(".mjx-mrow")
2627
+ .eq(0)
2628
+ .invoke("text")
2629
+ .then((text) => {
2630
+ expect(text.trim()).equal("5by");
2631
+ });
2632
+
2633
+ cy.get(cesc(`#\\/animal`)).should("have.text", "cat");
2634
+ cy.get(cesc(`#\\/plant`)).should("have.text", "shrub");
2635
+ cy.get(cesc(`#\\/p`))
2636
+ .find(".mjx-mrow")
2637
+ .eq(0)
2638
+ .invoke("text")
2639
+ .then((text) => {
2640
+ expect(text.trim()).equal("5by");
2641
+ });
2642
+ cy.get(cesc(`#\\/xx`))
2643
+ .find(".mjx-mrow")
2644
+ .eq(0)
2645
+ .invoke("text")
2646
+ .then((text) => {
2647
+ expect(text.trim()).equal("y");
2648
+ });
2649
+ cy.get(cesc(`#\\/aa`))
2650
+ .find(".mjx-mrow")
2651
+ .eq(0)
2652
+ .invoke("text")
2653
+ .then((text) => {
2654
+ expect(text.trim()).equal("b");
2655
+ });
2656
+
2657
+ cy.get(cesc(`#\\/b\\/animal`)).should("have.text", "cat");
2658
+ cy.get(cesc(`#\\/b\\/plant`)).should("have.text", "shrub");
2659
+ cy.get(cesc(`#\\/b\\/p`))
2660
+ .find(".mjx-mrow")
2661
+ .eq(0)
2662
+ .invoke("text")
2663
+ .then((text) => {
2664
+ expect(text.trim()).equal("5by");
2665
+ });
2666
+
2667
+ cy.get(cesc(`#\\/animalcopy`)).should("have.text", "cat");
2668
+ cy.get(cesc(`#\\/plantcopy`)).should("have.text", "shrub");
2669
+ cy.get(cesc(`#\\/pcopy`))
2670
+ .find(".mjx-mrow")
2671
+ .eq(0)
2672
+ .invoke("text")
2673
+ .then((text) => {
2674
+ expect(text.trim()).equal("5by");
2675
+ });
2676
+ cy.get(cesc(`#\\/xxcopy`))
2677
+ .find(".mjx-mrow")
2678
+ .eq(0)
2679
+ .invoke("text")
2680
+ .then((text) => {
2681
+ expect(text.trim()).equal("y");
2682
+ });
2683
+ cy.get(cesc(`#\\/aacopy`))
2684
+ .find(".mjx-mrow")
2685
+ .eq(0)
2686
+ .invoke("text")
2687
+ .then((text) => {
2688
+ expect(text.trim()).equal("b");
2689
+ });
2690
+
2691
+ cy.log("enter -1");
2692
+ cy.get(cesc("#\\/n") + " textarea").type(
2693
+ "{end}{backspace}{backspace}-1{enter}",
2694
+ { force: true },
2695
+ );
2696
+
2697
+ cy.get(cesc(`#\\/a\\/animal`)).should("not.exist");
2698
+ cy.get(cesc(`#\\/a\\/plant`)).should("not.exist");
2699
+ cy.get(cesc(`#\\/a\\/p`)).should("not.exist");
2700
+
2701
+ cy.window().then(async (win) => {
2702
+ let stateVariables = await win.returnAllStateVariables1();
2703
+ let p1Chidren = stateVariables["/_p1"].activeChildren;
2704
+ let theAnimalAnchor = cesc2("#" + p1Chidren[2].componentName);
2705
+ let thePlantAnchor = cesc2("#" + p1Chidren[4].componentName);
2706
+ let thePAnchor = cesc2("#" + p1Chidren[6].componentName);
2707
+
2708
+ let repeatChildren = stateVariables["/_p1"].activeChildren;
2709
+ let theAnimalCopyAnchor = cesc2("#" + repeatChildren[2].componentName);
2710
+ let thePlantCopyAnchor = cesc2("#" + repeatChildren[4].componentName);
2711
+ let thePCopyAnchor = cesc2("#" + repeatChildren[6].componentName);
2712
+
2713
+ cy.get(cesc(`#\\/_p1`))
2714
+ .invoke("text")
2715
+ .then((text) => {
2716
+ let words = text.split(/\s+/).slice(1);
2717
+ expect(words[0]).eq("dog");
2718
+ expect(words[1]).eq("tree");
2719
+ });
2720
+ cy.get(cesc(`#\\/_p1`))
2721
+ .find(".mjx-mrow")
2722
+ .eq(0)
2723
+ .invoke("text")
2724
+ .then((text) => {
2725
+ expect(text.trim()).equal("4ax");
2726
+ });
2727
+
2728
+ cy.get(theAnimalAnchor).should("have.text", "dog");
2729
+ cy.get(thePlantAnchor).should("have.text", "tree");
2730
+ cy.get(thePAnchor)
2731
+ .find(".mjx-mrow")
2732
+ .eq(0)
2733
+ .invoke("text")
2734
+ .then((text) => {
2735
+ expect(text.trim()).equal("4ax");
2736
+ });
2737
+
2738
+ cy.get(cesc(`#\\/animal`)).should("not.exist");
2739
+ cy.get(cesc(`#\\/plant`)).should("not.exist");
2740
+ cy.get(cesc(`#\\/p`)).should("not.exist");
2741
+ cy.get(cesc(`#\\/xx`)).should("not.exist");
2742
+ cy.get(cesc(`#\\/aa`)).should("not.exist");
2743
+
2744
+ cy.get(cesc(`#\\/b\\/animal`)).should("not.exist");
2745
+ cy.get(cesc(`#\\/b\\/plant`)).should("not.exist");
2746
+ cy.get(cesc(`#\\/b\\/p`)).should("not.exist");
2747
+
2748
+ cy.get(cesc(`#\\/repeat`))
2749
+ .invoke("text")
2750
+ .then((text) => {
2751
+ let words = text.split(/\s+/).slice(1);
2752
+ expect(words[0]).eq("dog");
2753
+ expect(words[1]).eq("tree");
2754
+ });
2755
+ cy.get(cesc(`#\\/repeat`))
2756
+ .find(".mjx-mrow")
2757
+ .eq(0)
2758
+ .invoke("text")
2759
+ .then((text) => {
2760
+ expect(text.trim()).equal("4ax");
2761
+ });
2762
+
2763
+ cy.get(theAnimalCopyAnchor).should("have.text", "dog");
2764
+ cy.get(thePlantCopyAnchor).should("have.text", "tree");
2765
+ cy.get(thePCopyAnchor)
2766
+ .find(".mjx-mrow")
2767
+ .eq(0)
2768
+ .invoke("text")
2769
+ .then((text) => {
2770
+ expect(text.trim()).equal("4ax");
2771
+ });
2772
+
2773
+ cy.get(cesc(`#\\/animalcopy`)).should("not.exist");
2774
+ cy.get(cesc(`#\\/plantcopy`)).should("not.exist");
2775
+ cy.get(cesc(`#\\/pcopy`)).should("not.exist");
2776
+ cy.get(cesc(`#\\/xxcopy`)).should("not.exist");
2777
+ cy.get(cesc(`#\\/aacopy`)).should("not.exist");
2778
+ });
2779
+
2780
+ cy.log("enter 10");
2781
+ cy.get(cesc("#\\/n") + " textarea").type(
2782
+ "{end}{backspace}{backspace}10{enter}",
2783
+ { force: true },
2784
+ );
2785
+
2786
+ cy.get(cesc(`#\\/a\\/animal`)).should("have.text", "mouse");
2787
+ cy.get(cesc(`#\\/a\\/plant`)).should("have.text", "bush");
2788
+ cy.get(cesc(`#\\/a\\/p`))
2789
+ .find(".mjx-mrow")
2790
+ .eq(0)
2791
+ .invoke("text")
2792
+ .then((text) => {
2793
+ expect(text.trim()).equal("6cz");
2794
+ });
2795
+
2796
+ cy.get(cesc(`#\\/animal`)).should("have.text", "mouse");
2797
+ cy.get(cesc(`#\\/plant`)).should("have.text", "bush");
2798
+ cy.get(cesc(`#\\/p`))
2799
+ .find(".mjx-mrow")
2800
+ .eq(0)
2801
+ .invoke("text")
2802
+ .then((text) => {
2803
+ expect(text.trim()).equal("6cz");
2804
+ });
2805
+ cy.get(cesc(`#\\/xx`))
2806
+ .find(".mjx-mrow")
2807
+ .eq(0)
2808
+ .invoke("text")
2809
+ .then((text) => {
2810
+ expect(text.trim()).equal("z");
2811
+ });
2812
+ cy.get(cesc(`#\\/aa`))
2813
+ .find(".mjx-mrow")
2814
+ .eq(0)
2815
+ .invoke("text")
2816
+ .then((text) => {
2817
+ expect(text.trim()).equal("c");
2818
+ });
2819
+
2820
+ cy.get(cesc(`#\\/b\\/animal`)).should("have.text", "mouse");
2821
+ cy.get(cesc(`#\\/b\\/plant`)).should("have.text", "bush");
2822
+ cy.get(cesc(`#\\/b\\/p`))
2823
+ .find(".mjx-mrow")
2824
+ .eq(0)
2825
+ .invoke("text")
2826
+ .then((text) => {
2827
+ expect(text.trim()).equal("6cz");
2828
+ });
2829
+
2830
+ cy.get(cesc(`#\\/animalcopy`)).should("have.text", "mouse");
2831
+ cy.get(cesc(`#\\/plantcopy`)).should("have.text", "bush");
2832
+ cy.get(cesc(`#\\/pcopy`))
2833
+ .find(".mjx-mrow")
2834
+ .eq(0)
2835
+ .invoke("text")
2836
+ .then((text) => {
2837
+ expect(text.trim()).equal("6cz");
2838
+ });
2839
+ cy.get(cesc(`#\\/xxcopy`))
2840
+ .find(".mjx-mrow")
2841
+ .eq(0)
2842
+ .invoke("text")
2843
+ .then((text) => {
2844
+ expect(text.trim()).equal("z");
2845
+ });
2846
+ cy.get(cesc(`#\\/aacopy`))
2847
+ .find(".mjx-mrow")
2848
+ .eq(0)
2849
+ .invoke("text")
2850
+ .then((text) => {
2851
+ expect(text.trim()).equal("c");
2852
+ });
2853
+ });
2854
+
2855
+ it("dynamic internal references", () => {
2856
+ cy.window().then(async (win) => {
2857
+ win.postMessage(
2858
+ {
2859
+ doenetML: `
2860
+ <text>a</text>
2861
+ <mathinput name="n" prefill="1" />
2862
+ <conditionalContent assignNames="a">
2863
+ <case condition="$n<0" newNamespace>
2864
+ <p>What is your favorite animal? <textinput name="response" /></p>
2865
+ <p>I like <copy prop="value" target="response" />, too.</p>
2866
+ </case>
2867
+ <case condition="$n <= 1" newNamespace >
2868
+ <p>What is your name? <textinput name="response" /></p>
2869
+ <p>Hello, <copy prop="value" target="response" />!</p>
2870
+ </case>
2871
+ <else newNamespace>
2872
+ <p>Anything else? <textinput name="response" /></p>
2873
+ <p>To repeat: <copy prop="value" target="response" />.</p>
2874
+ </else>
2875
+ </conditionalContent>
2876
+
2877
+ <p>The response: <copy target="a/response" prop="value" /></p>
2878
+
2879
+ <copy name="sc2" target="_conditionalcontent1" assignNames="b" />
2880
+
2881
+ <p>The response one more time: <copy target="b/response" prop="value" /></p>
2882
+ `,
2883
+ },
2884
+ "*",
2885
+ );
2886
+ });
2887
+
2888
+ cy.get(cesc("#\\/_text1")).should("have.text", "a"); // to wait until loaded
2889
+
2890
+ cy.get(cesc(`#\\/a\\/_p1`)).should("have.text", "What is your name? ");
2891
+ cy.get(cesc(`#\\/a\\/_p2`)).should("have.text", "Hello, !");
2892
+ cy.get(cesc(`#\\/_p1`)).should("have.text", "The response: ");
2893
+ cy.get(cesc(`#\\/b\\/_p1`)).should("have.text", "What is your name? ");
2894
+ cy.get(cesc(`#\\/b\\/_p2`)).should("have.text", "Hello, !");
2895
+ cy.get(cesc(`#\\/_p2`)).should("have.text", "The response one more time: ");
2896
+
2897
+ cy.get(cesc(`#\\/a\\/response_input`)).clear().type("Fred{enter}");
2898
+ cy.get(cesc(`#\\/a\\/_p2`)).should("have.text", "Hello, Fred!");
2899
+ cy.get(cesc(`#\\/_p1`)).should("have.text", "The response: Fred");
2900
+ cy.get(cesc(`#\\/b\\/_p2`)).should("have.text", "Hello, Fred!");
2901
+ cy.get(cesc(`#\\/_p2`)).should(
2902
+ "have.text",
2903
+ "The response one more time: Fred",
2904
+ );
2905
+
2906
+ cy.get(cesc("#\\/n") + " textarea").type("{end}{backspace}-1{enter}", {
2907
+ force: true,
2908
+ });
2909
+ cy.get(cesc(`#\\/a\\/_p1`)).should(
2910
+ "have.text",
2911
+ "What is your favorite animal? ",
2912
+ );
2913
+ cy.get(cesc(`#\\/a\\/_p2`)).should("have.text", "I like , too.");
2914
+ cy.get(cesc(`#\\/_p1`)).should("have.text", "The response: ");
2915
+ cy.get(cesc(`#\\/b\\/_p1`)).should(
2916
+ "have.text",
2917
+ "What is your favorite animal? ",
2918
+ );
2919
+ cy.get(cesc(`#\\/b\\/_p2`)).should("have.text", "I like , too.");
2920
+ cy.get(cesc(`#\\/_p2`)).should("have.text", "The response one more time: ");
2921
+
2922
+ cy.get(cesc(`#\\/a\\/response_input`)).clear().type("dogs{enter}");
2923
+ cy.get(cesc(`#\\/a\\/_p2`)).should("have.text", "I like dogs, too.");
2924
+ cy.get(cesc(`#\\/_p1`)).should("have.text", "The response: dogs");
2925
+ cy.get(cesc(`#\\/b\\/_p2`)).should("have.text", "I like dogs, too.");
2926
+ cy.get(cesc(`#\\/_p2`)).should(
2927
+ "have.text",
2928
+ "The response one more time: dogs",
2929
+ );
2930
+
2931
+ cy.get(cesc("#\\/n") + " textarea").type(
2932
+ "{end}{backspace}{backspace}3{enter}",
2933
+ { force: true },
2934
+ );
2935
+ cy.get(cesc(`#\\/a\\/_p1`)).should("have.text", "Anything else? ");
2936
+ cy.get(cesc(`#\\/a\\/_p2`)).should("have.text", "To repeat: .");
2937
+ cy.get(cesc(`#\\/_p1`)).should("have.text", "The response: ");
2938
+ cy.get(cesc(`#\\/b\\/_p1`)).should("have.text", "Anything else? ");
2939
+ cy.get(cesc(`#\\/b\\/_p2`)).should("have.text", "To repeat: .");
2940
+ cy.get(cesc(`#\\/_p2`)).should("have.text", "The response one more time: ");
2941
+
2942
+ cy.get(cesc(`#\\/a\\/response_input`)).clear().type("Goodbye{enter}");
2943
+ cy.get(cesc(`#\\/a\\/_p2`)).should("have.text", "To repeat: Goodbye.");
2944
+ cy.get(cesc(`#\\/_p1`)).should("have.text", "The response: Goodbye");
2945
+ cy.get(cesc(`#\\/b\\/_p2`)).should("have.text", "To repeat: Goodbye.");
2946
+ cy.get(cesc(`#\\/_p2`)).should(
2947
+ "have.text",
2948
+ "The response one more time: Goodbye",
2949
+ );
2950
+ });
2951
+
2952
+ it("dynamic internal references, assign pieces", () => {
2953
+ cy.window().then(async (win) => {
2954
+ win.postMessage(
2955
+ {
2956
+ doenetML: `
2957
+ <text>a</text>
2958
+ <mathinput name="n" prefill="1" />
2959
+ <conditionalContent assignNames="(a b)">
2960
+ <case condition="$n<0" >
2961
+ <p newNamespace name="panimal">What is your favorite animal? <textinput name="response" /></p>
2962
+ <p newNamespace>I like <copy prop="value" target="../panimal/response" />, too.</p>
2963
+ </case>
2964
+ <case condition="$n <= 1" >
2965
+ <p newNamespace name="pname">What is your name? <textinput name="response" /></p>
2966
+ <p newNamespace>Hello, <copy prop="value" target="../pname/response" />!</p>
2967
+ </case>
2968
+ <else>
2969
+ <p newNamespace name="pelse">Anything else? <textinput name="response" /></p>
2970
+ <p newNamespace>To repeat: <copy prop="value" target="../pelse/response" />.</p>
2971
+ </else>
2972
+ </conditionalContent>
2973
+
2974
+ <p name="pResponse">The response: <copy target="a/response" prop="value" /></p>
2975
+
2976
+ <copy name="sc2" target="_conditionalcontent1" assignNames="(c d)" />
2977
+
2978
+ <p name="pResponse2">The response one more time: <copy target="c/response" prop="value" /></p>
2979
+ `,
2980
+ },
2981
+ "*",
2982
+ );
2983
+ });
2984
+
2985
+ cy.get(cesc("#\\/_text1")).should("have.text", "a"); // to wait until loaded
2986
+
2987
+ cy.get(cesc(`#\\/a`)).should("have.text", "What is your name? ");
2988
+ cy.get(cesc(`#\\/b`)).should("have.text", "Hello, !");
2989
+ cy.get(cesc(`#\\/pResponse`)).should("have.text", "The response: ");
2990
+ cy.get(cesc(`#\\/c`)).should("have.text", "What is your name? ");
2991
+ cy.get(cesc(`#\\/d`)).should("have.text", "Hello, !");
2992
+ cy.get(cesc(`#\\/pResponse2`)).should(
2993
+ "have.text",
2994
+ "The response one more time: ",
2995
+ );
2996
+
2997
+ cy.get(cesc(`#\\/a\\/response_input`)).clear().type("Fred{enter}");
2998
+ cy.get(cesc(`#\\/b`)).should("have.text", "Hello, Fred!");
2999
+ cy.get(cesc(`#\\/pResponse`)).should("have.text", "The response: Fred");
3000
+ cy.get(cesc(`#\\/d`)).should("have.text", "Hello, Fred!");
3001
+ cy.get(cesc(`#\\/pResponse2`)).should(
3002
+ "have.text",
3003
+ "The response one more time: Fred",
3004
+ );
3005
+
3006
+ cy.get(cesc("#\\/n") + " textarea").type("{end}{backspace}-1{enter}", {
3007
+ force: true,
3008
+ });
3009
+ cy.get(cesc(`#\\/a`)).should("have.text", "What is your favorite animal? ");
3010
+ cy.get(cesc(`#\\/b`)).should("have.text", "I like , too.");
3011
+ cy.get(cesc(`#\\/pResponse`)).should("have.text", "The response: ");
3012
+ cy.get(cesc(`#\\/c`)).should("have.text", "What is your favorite animal? ");
3013
+ cy.get(cesc(`#\\/d`)).should("have.text", "I like , too.");
3014
+ cy.get(cesc(`#\\/pResponse2`)).should(
3015
+ "have.text",
3016
+ "The response one more time: ",
3017
+ );
3018
+
3019
+ cy.get(cesc(`#\\/a\\/response_input`)).clear().type("dogs{enter}");
3020
+ cy.get(cesc(`#\\/b`)).should("have.text", "I like dogs, too.");
3021
+ cy.get(cesc(`#\\/pResponse`)).should("have.text", "The response: dogs");
3022
+ cy.get(cesc(`#\\/d`)).should("have.text", "I like dogs, too.");
3023
+ cy.get(cesc(`#\\/pResponse2`)).should(
3024
+ "have.text",
3025
+ "The response one more time: dogs",
3026
+ );
3027
+
3028
+ cy.get(cesc("#\\/n") + " textarea").type(
3029
+ "{end}{backspace}{backspace}3{enter}",
3030
+ { force: true },
3031
+ );
3032
+ cy.get(cesc(`#\\/a`)).should("have.text", "Anything else? ");
3033
+ cy.get(cesc(`#\\/b`)).should("have.text", "To repeat: .");
3034
+ cy.get(cesc(`#\\/pResponse`)).should("have.text", "The response: ");
3035
+ cy.get(cesc(`#\\/c`)).should("have.text", "Anything else? ");
3036
+ cy.get(cesc(`#\\/d`)).should("have.text", "To repeat: .");
3037
+ cy.get(cesc(`#\\/pResponse2`)).should(
3038
+ "have.text",
3039
+ "The response one more time: ",
3040
+ );
3041
+
3042
+ cy.get(cesc(`#\\/a\\/response_input`)).clear().type("Goodbye{enter}");
3043
+ cy.get(cesc(`#\\/b`)).should("have.text", "To repeat: Goodbye.");
3044
+ cy.get(cesc(`#\\/pResponse`)).should("have.text", "The response: Goodbye");
3045
+ cy.get(cesc(`#\\/d`)).should("have.text", "To repeat: Goodbye.");
3046
+ cy.get(cesc(`#\\/pResponse2`)).should(
3047
+ "have.text",
3048
+ "The response one more time: Goodbye",
3049
+ );
3050
+ });
3051
+
3052
+ it("copy case", () => {
3053
+ cy.window().then(async (win) => {
3054
+ win.postMessage(
3055
+ {
3056
+ doenetML: `
3057
+ <text>a</text>
3058
+ <mathinput name="n" />
3059
+
3060
+ <p><conditionalContent>
3061
+ <case name="positiveCase" condition="$n>0" ><text>positive</text></case>
3062
+ <else><text>non-positive</text></else>
3063
+ </conditionalContent></p>
3064
+
3065
+ <p><conditionalContent>
3066
+ <copy target="positiveCase" createComponentOfType="case" />
3067
+ <case condition="$n<0" ><text>negative</text></case>
3068
+ <else><text>neither</text></else>
3069
+ </conditionalContent></p>
3070
+
3071
+
3072
+ <p><copy target="_conditionalcontent1" /></p>
3073
+
3074
+ <p><copy target="_conditionalcontent2" /></p>
3075
+
3076
+ `,
3077
+ },
3078
+ "*",
3079
+ );
3080
+ });
3081
+
3082
+ cy.get(cesc("#\\/_text1")).should("have.text", "a"); // to wait until loaded
3083
+
3084
+ cy.get(cesc("#\\/_p1")).should("have.text", "non-positive");
3085
+ cy.get(cesc("#\\/_p3")).should("have.text", "non-positive");
3086
+ cy.get(cesc("#\\/_p2")).should("have.text", "neither");
3087
+ cy.get(cesc("#\\/_p4")).should("have.text", "neither");
3088
+
3089
+ cy.log("enter 1");
3090
+ cy.get(cesc("#\\/n") + " textarea").type("{end}{backspace}1{enter}", {
3091
+ force: true,
3092
+ });
3093
+
3094
+ cy.get(cesc("#\\/_p1")).should("have.text", "positive");
3095
+ cy.get(cesc("#\\/_p3")).should("have.text", "positive");
3096
+ cy.get(cesc("#\\/_p2")).should("have.text", "positive");
3097
+ cy.get(cesc("#\\/_p4")).should("have.text", "positive");
3098
+
3099
+ cy.log("enter -1");
3100
+ cy.get(cesc("#\\/n") + " textarea").type(
3101
+ "{end}{backspace}{backspace}-1{enter}",
3102
+ { force: true },
3103
+ );
3104
+
3105
+ cy.get(cesc("#\\/_p1")).should("have.text", "non-positive");
3106
+ cy.get(cesc("#\\/_p3")).should("have.text", "non-positive");
3107
+ cy.get(cesc("#\\/_p2")).should("have.text", "negative");
3108
+ cy.get(cesc("#\\/_p4")).should("have.text", "negative");
3109
+
3110
+ cy.log("enter 0");
3111
+ cy.get(cesc("#\\/n") + " textarea").type(
3112
+ "{ctrl+home}{shift+end}{backspace}0{enter}",
3113
+ { force: true },
3114
+ );
3115
+
3116
+ cy.get(cesc("#\\/_p1")).should("have.text", "non-positive");
3117
+ cy.get(cesc("#\\/_p3")).should("have.text", "non-positive");
3118
+ cy.get(cesc("#\\/_p2")).should("have.text", "neither");
3119
+ cy.get(cesc("#\\/_p4")).should("have.text", "neither");
3120
+ });
3121
+
3122
+ it("copy else", () => {
3123
+ cy.window().then(async (win) => {
3124
+ win.postMessage(
3125
+ {
3126
+ doenetML: `
3127
+ <text>a</text>
3128
+ <mathinput name="n" />
3129
+
3130
+ <p><conditionalContent>
3131
+ <case condition="$n>0" ><text>hello</text></case>
3132
+ <else name="bye"><text>bye</text></else>
3133
+ </conditionalContent></p>
3134
+
3135
+ <p><conditionalContent>
3136
+ <case condition="$n<0" ><text>hello</text></case>
3137
+ <case condition="$n>0" ><text>oops</text></case>
3138
+ <copy target="bye" createComponentOfType="else" />
3139
+ </conditionalContent></p>
3140
+
3141
+ <p><copy target="_conditionalcontent1" /></p>
3142
+
3143
+ <p><copy target="_conditionalcontent2" /></p>
3144
+
3145
+ `,
3146
+ },
3147
+ "*",
3148
+ );
3149
+ });
3150
+
3151
+ cy.get(cesc("#\\/_text1")).should("have.text", "a"); // to wait until loaded
3152
+
3153
+ cy.get(cesc("#\\/_p1")).should("have.text", "bye");
3154
+ cy.get(cesc("#\\/_p3")).should("have.text", "bye");
3155
+ cy.get(cesc("#\\/_p2")).should("have.text", "bye");
3156
+ cy.get(cesc("#\\/_p4")).should("have.text", "bye");
3157
+
3158
+ cy.log("enter 1");
3159
+ cy.get(cesc("#\\/n") + " textarea").type("{end}{backspace}1{enter}", {
3160
+ force: true,
3161
+ });
3162
+
3163
+ cy.get(cesc("#\\/_p1")).should("have.text", "hello");
3164
+ cy.get(cesc("#\\/_p3")).should("have.text", "hello");
3165
+ cy.get(cesc("#\\/_p2")).should("have.text", "oops");
3166
+ cy.get(cesc("#\\/_p4")).should("have.text", "oops");
3167
+
3168
+ cy.log("enter -1");
3169
+ cy.get(cesc("#\\/n") + " textarea").type(
3170
+ "{end}{backspace}{backspace}-1{enter}",
3171
+ { force: true },
3172
+ );
3173
+
3174
+ cy.get(cesc("#\\/_p1")).should("have.text", "bye");
3175
+ cy.get(cesc("#\\/_p3")).should("have.text", "bye");
3176
+ cy.get(cesc("#\\/_p2")).should("have.text", "hello");
3177
+ cy.get(cesc("#\\/_p4")).should("have.text", "hello");
3178
+
3179
+ cy.log("enter 0");
3180
+ cy.get(cesc("#\\/n") + " textarea").type(
3181
+ "{ctrl+home}{shift+end}{backspace}0{enter}",
3182
+ { force: true },
3183
+ );
3184
+
3185
+ cy.get(cesc("#\\/_p1")).should("have.text", "bye");
3186
+ cy.get(cesc("#\\/_p3")).should("have.text", "bye");
3187
+ cy.get(cesc("#\\/_p2")).should("have.text", "bye");
3188
+ cy.get(cesc("#\\/_p4")).should("have.text", "bye");
3189
+ });
3190
+
3191
+ it("conditionalcontents hide dynamically", () => {
3192
+ cy.window().then(async (win) => {
3193
+ win.postMessage(
3194
+ {
3195
+ doenetML: `
3196
+ <text>a</text>
3197
+
3198
+ <booleaninput name='h1' prefill="false" >
3199
+ <label>Hide first conditionalContent</label>
3200
+ </booleaninput>
3201
+ <booleaninput name='h2' prefill="true" >
3202
+ <label>Hide second conditionalContent</label>
3203
+ </booleaninput>
3204
+ <mathinput name="n" />
3205
+ <p name="pa">a: <conditionalContent assignNames="a" hide="$h1">
3206
+ <case condition="$n<0"><text>dog</text></case>
3207
+ <case condition="$n<=1"><text>cat</text></case>
3208
+ <else><text>mouse</text></else>
3209
+ </conditionalContent></p>
3210
+ <p name="pb">b: <conditionalContent assignNames="b" hide="$h2">
3211
+ <case condition="$n<0"><text>dog</text></case>
3212
+ <case condition="$n<=1"><text>cat</text></case>
3213
+ <else><text>mouse</text></else>
3214
+ </conditionalContent></p>
3215
+ <p name="pa1">a1: <copy target="a" assignNames="(a1)" /></p>
3216
+ <p name="pb1">b1: <copy target="b" assignNames="(b1)" /></p>
3217
+ `,
3218
+ },
3219
+ "*",
3220
+ );
3221
+ });
3222
+
3223
+ cy.get(cesc("#\\/_text1")).should("have.text", "a"); // to wait for page to load
3224
+
3225
+ cy.get(cesc("#\\/pa")).should("have.text", "a: mouse");
3226
+ cy.get(cesc("#\\/pa1")).should("have.text", "a1: mouse");
3227
+ cy.get(cesc("#\\/pb")).should("have.text", "b: ");
3228
+ cy.get(cesc("#\\/pb1")).should("have.text", "b1: mouse");
3229
+
3230
+ cy.log("enter 1");
3231
+ cy.get(cesc("#\\/n") + " textarea").type("1{enter}", { force: true });
3232
+
3233
+ cy.get(cesc("#\\/pa")).should("have.text", "a: cat");
3234
+ cy.get(cesc("#\\/pa1")).should("have.text", "a1: cat");
3235
+ cy.get(cesc("#\\/pb")).should("have.text", "b: ");
3236
+ cy.get(cesc("#\\/pb1")).should("have.text", "b1: cat");
3237
+
3238
+ cy.get(cesc("#\\/h1")).click();
3239
+ cy.get(cesc("#\\/h2")).click();
3240
+
3241
+ cy.get(cesc("#\\/pa")).should("have.text", "a: ");
3242
+ cy.get(cesc("#\\/pa1")).should("have.text", "a1: cat");
3243
+ cy.get(cesc("#\\/pb")).should("have.text", "b: cat");
3244
+ cy.get(cesc("#\\/pb1")).should("have.text", "b1: cat");
3245
+
3246
+ cy.log("enter -3");
3247
+ cy.get(cesc("#\\/n") + " textarea").type("{end}{backspace}-3{enter}", {
3248
+ force: true,
3249
+ });
3250
+
3251
+ cy.get(cesc("#\\/pa")).should("have.text", "a: ");
3252
+ cy.get(cesc("#\\/pa1")).should("have.text", "a1: dog");
3253
+ cy.get(cesc("#\\/pb")).should("have.text", "b: dog");
3254
+ cy.get(cesc("#\\/pb1")).should("have.text", "b1: dog");
3255
+
3256
+ cy.get(cesc("#\\/h1")).click();
3257
+ cy.get(cesc("#\\/h2")).click();
3258
+
3259
+ cy.get(cesc("#\\/pa")).should("have.text", "a: dog");
3260
+ cy.get(cesc("#\\/pa1")).should("have.text", "a1: dog");
3261
+ cy.get(cesc("#\\/pb")).should("have.text", "b: ");
3262
+ cy.get(cesc("#\\/pb1")).should("have.text", "b1: dog");
3263
+ });
3264
+
3265
+ it("string and blank strings in case and else", () => {
3266
+ cy.window().then(async (win) => {
3267
+ win.postMessage(
3268
+ {
3269
+ doenetML: `
3270
+ <text>a</text>
3271
+ <setup>
3272
+ <text name="animal1">fox</text><text name="verb1">jumps</text>
3273
+ <text name="animal2">elephant</text><text name="verb2">trumpets</text>
3274
+ </setup>
3275
+
3276
+ <mathinput name="n" />
3277
+ <p name="pa">a: <conditionalContent assignNames="a">
3278
+ <case condition="$n > 0">The $animal1 $verb1.</case>
3279
+ <else>The $animal2 $verb2.</else>
3280
+ </conditionalContent></p>
3281
+
3282
+ <p name="pa1">a1: <copy target="a" assignNames="((a11) (a12))" /></p>
3283
+
3284
+ <p name="ppieces" >pieces: <copy target="_conditionalcontent1" assignNames="(b c)" /></p>
3285
+
3286
+ <p name="pb1">b1: <copy target="b" assignNames="b1" /></p>
3287
+ <p name="pc1">c1: <copy target="c" assignNames="c1" /></p>
3288
+
3289
+ `,
3290
+ },
3291
+ "*",
3292
+ );
3293
+ });
3294
+
3295
+ cy.get(cesc("#\\/_text1")).should("have.text", "a");
3296
+
3297
+ cy.get(cesc("#\\/pa")).should("have.text", "a: The elephant trumpets.");
3298
+ cy.get(cesc("#\\/pa1")).should("have.text", "a1: The elephant trumpets.");
3299
+ cy.get(cesc("#\\/ppieces")).should(
3300
+ "have.text",
3301
+ "pieces: The elephant trumpets.",
3302
+ );
3303
+ cy.get(cesc("#\\/pb1")).should("have.text", "b1: elephant");
3304
+ cy.get(cesc("#\\/pc1")).should("have.text", "c1: trumpets");
3305
+
3306
+ cy.get(cesc("#\\/a11")).should("have.text", "elephant");
3307
+ cy.get(cesc("#\\/a12")).should("have.text", "trumpets");
3308
+ cy.get(cesc("#\\/b1")).should("have.text", "elephant");
3309
+ cy.get(cesc("#\\/c1")).should("have.text", "trumpets");
3310
+
3311
+ cy.log("enter 1");
3312
+ cy.get(cesc("#\\/n") + " textarea").type("1{enter}", { force: true });
3313
+
3314
+ cy.get(cesc("#\\/pa")).should("have.text", "a: The fox jumps.");
3315
+ cy.get(cesc("#\\/pa1")).should("have.text", "a1: The fox jumps.");
3316
+ cy.get(cesc("#\\/ppieces")).should("have.text", "pieces: The fox jumps.");
3317
+ cy.get(cesc("#\\/pb1")).should("have.text", "b1: fox");
3318
+ cy.get(cesc("#\\/pc1")).should("have.text", "c1: jumps");
3319
+
3320
+ cy.get(cesc("#\\/a11")).should("have.text", "fox");
3321
+ cy.get(cesc("#\\/a12")).should("have.text", "jumps");
3322
+ cy.get(cesc("#\\/b1")).should("have.text", "fox");
3323
+ cy.get(cesc("#\\/c1")).should("have.text", "jumps");
3324
+
3325
+ cy.log("enter 0");
3326
+ cy.get(cesc("#\\/n") + " textarea").type("{end}{backspace}0{enter}", {
3327
+ force: true,
3328
+ });
3329
+
3330
+ cy.get(cesc("#\\/pa")).should("have.text", "a: The elephant trumpets.");
3331
+ cy.get(cesc("#\\/pa1")).should("have.text", "a1: The elephant trumpets.");
3332
+ cy.get(cesc("#\\/ppieces")).should(
3333
+ "have.text",
3334
+ "pieces: The elephant trumpets.",
3335
+ );
3336
+ cy.get(cesc("#\\/pb1")).should("have.text", "b1: elephant");
3337
+ cy.get(cesc("#\\/pc1")).should("have.text", "c1: trumpets");
3338
+
3339
+ cy.get(cesc("#\\/a11")).should("have.text", "elephant");
3340
+ cy.get(cesc("#\\/a12")).should("have.text", "trumpets");
3341
+ cy.get(cesc("#\\/b1")).should("have.text", "elephant");
3342
+ cy.get(cesc("#\\/c1")).should("have.text", "trumpets");
3343
+ });
3344
+
3345
+ it("copy with invalid target gets expanded", () => {
3346
+ cy.window().then(async (win) => {
3347
+ win.postMessage(
3348
+ {
3349
+ doenetML: `
3350
+ <text>a</text>
3351
+ <mathinput name="n" />
3352
+ before
3353
+ <conditionalContent assignNames='a'>
3354
+ <case condition="$n=1" newNamespace>nothing: <copy target="nada" name="nothing" /></case>
3355
+ </conditionalContent>
3356
+ after
3357
+ `,
3358
+ },
3359
+ "*",
3360
+ );
3361
+ });
3362
+
3363
+ cy.get(cesc("#\\/_text1")).should("have.text", "a");
3364
+
3365
+ cy.get(cesc("#\\/n") + " textarea")
3366
+ .type("1", { force: true })
3367
+ .blur();
3368
+
3369
+ cy.get(cesc("#\\/_document1")).should(
3370
+ "contain.text",
3371
+ "\n a\n 1\n before\n nothing: \n after\n ",
3372
+ );
3373
+
3374
+ cy.window().then(async (win) => {
3375
+ let stateVariables = await win.returnAllStateVariables1();
3376
+ // don't currently have a way to check for isExpanded or replacements
3377
+ // expect(stateVariables["/a/nothing"].isExpanded).eq(true)
3378
+ // expect(stateVariables["/a/nothing"].replacements).eqls([])
3379
+ expect(
3380
+ stateVariables["/_document1"].activeChildren.filter(
3381
+ (x) => x.componentType === "copy",
3382
+ ),
3383
+ ).eqls([]);
3384
+ });
3385
+ });
3386
+
3387
+ it("use original names if no assignNames", () => {
3388
+ cy.window().then(async (win) => {
3389
+ win.postMessage(
3390
+ {
3391
+ doenetML: `
3392
+ <text>a</text>
3393
+ <mathinput name="n" />
3394
+
3395
+ <conditionalContent condition="$n > 0">
3396
+ <p>We have a <text name="winner1">first winner</text>!</p>
3397
+ </conditionalContent>
3398
+
3399
+ <conditionalContent>
3400
+ <case condition="$n > 0 && $n<=1">
3401
+ <p>Just emphasizing that we have that <text name="winner1b">first winner</text>!</p>
3402
+ </case>
3403
+ <case condition="$n > 1 && $n <= 2">
3404
+ <p>We have a <text name="winner2">second winner</text>!</p>
3405
+ </case>
3406
+ <case condition="$n > 2">
3407
+ <p>We have a <text name="winner3">third winner</text>!</p>
3408
+ </case>
3409
+ <else>
3410
+ <p>We have <text name="winner0">no winner</text>.</p>
3411
+ </else>
3412
+ </conditionalContent>
3413
+ `,
3414
+ },
3415
+ "*",
3416
+ );
3417
+ });
3418
+
3419
+ cy.get(cesc("#\\/_text1")).should("have.text", "a");
3420
+
3421
+ cy.get(cesc("#\\/winner1")).should("not.exist");
3422
+ cy.get(cesc("#\\/_p1")).should("not.exist");
3423
+ cy.get(cesc("#\\/winner1b")).should("not.exist");
3424
+ cy.get(cesc("#\\/_p2")).should("not.exist");
3425
+ cy.get(cesc("#\\/winner2")).should("not.exist");
3426
+ cy.get(cesc("#\\/_p3")).should("not.exist");
3427
+ cy.get(cesc("#\\/winner3")).should("not.exist");
3428
+ cy.get(cesc("#\\/_p4")).should("not.exist");
3429
+ cy.get(cesc("#\\/winner0")).should("have.text", "no winner");
3430
+ cy.get(cesc("#\\/_p5")).should("have.text", "We have no winner.");
3431
+
3432
+ cy.get(cesc("#\\/n") + " textarea")
3433
+ .type("1", { force: true })
3434
+ .blur();
3435
+
3436
+ cy.get(cesc("#\\/winner1")).should("have.text", "first winner");
3437
+ cy.get(cesc("#\\/_p1")).should("have.text", "We have a first winner!");
3438
+ cy.get(cesc("#\\/winner1b")).should("have.text", "first winner");
3439
+ cy.get(cesc("#\\/_p2")).should(
3440
+ "have.text",
3441
+ "Just emphasizing that we have that first winner!",
3442
+ );
3443
+ cy.get(cesc("#\\/winner2")).should("not.exist");
3444
+ cy.get(cesc("#\\/_p3")).should("not.exist");
3445
+ cy.get(cesc("#\\/winner3")).should("not.exist");
3446
+ cy.get(cesc("#\\/_p4")).should("not.exist");
3447
+ cy.get(cesc("#\\/winner0")).should("not.exist");
3448
+ cy.get(cesc("#\\/_p5")).should("not.exist");
3449
+
3450
+ cy.get(cesc("#\\/n") + " textarea")
3451
+ .type("{end}{backspace}2", { force: true })
3452
+ .blur();
3453
+
3454
+ cy.get(cesc("#\\/winner1")).should("have.text", "first winner");
3455
+ cy.get(cesc("#\\/_p1")).should("have.text", "We have a first winner!");
3456
+ cy.get(cesc("#\\/winner1b")).should("not.exist");
3457
+ cy.get(cesc("#\\/_p2")).should("not.exist");
3458
+ cy.get(cesc("#\\/winner2")).should("have.text", "second winner");
3459
+ cy.get(cesc("#\\/_p3")).should("have.text", "We have a second winner!");
3460
+ cy.get(cesc("#\\/winner3")).should("not.exist");
3461
+ cy.get(cesc("#\\/_p4")).should("not.exist");
3462
+ cy.get(cesc("#\\/winner0")).should("not.exist");
3463
+ cy.get(cesc("#\\/_p5")).should("not.exist");
3464
+
3465
+ cy.get(cesc("#\\/n") + " textarea")
3466
+ .type("{end}{backspace}3", { force: true })
3467
+ .blur();
3468
+
3469
+ cy.get(cesc("#\\/winner1")).should("have.text", "first winner");
3470
+ cy.get(cesc("#\\/_p1")).should("have.text", "We have a first winner!");
3471
+ cy.get(cesc("#\\/winner1b")).should("not.exist");
3472
+ cy.get(cesc("#\\/_p2")).should("not.exist");
3473
+ cy.get(cesc("#\\/winner2")).should("not.exist");
3474
+ cy.get(cesc("#\\/_p3")).should("not.exist");
3475
+ cy.get(cesc("#\\/winner3")).should("have.text", "third winner");
3476
+ cy.get(cesc("#\\/_p4")).should("have.text", "We have a third winner!");
3477
+ cy.get(cesc("#\\/winner0")).should("not.exist");
3478
+ cy.get(cesc("#\\/_p5")).should("not.exist");
3479
+
3480
+ cy.get(cesc("#\\/n") + " textarea")
3481
+ .type("{end}{backspace}x", { force: true })
3482
+ .blur();
3483
+
3484
+ cy.get(cesc("#\\/winner1")).should("not.exist");
3485
+ cy.get(cesc("#\\/_p1")).should("not.exist");
3486
+ cy.get(cesc("#\\/winner1b")).should("not.exist");
3487
+ cy.get(cesc("#\\/_p2")).should("not.exist");
3488
+ cy.get(cesc("#\\/winner2")).should("not.exist");
3489
+ cy.get(cesc("#\\/_p3")).should("not.exist");
3490
+ cy.get(cesc("#\\/winner3")).should("not.exist");
3491
+ cy.get(cesc("#\\/_p4")).should("not.exist");
3492
+ cy.get(cesc("#\\/winner0")).should("have.text", "no winner");
3493
+ cy.get(cesc("#\\/_p5")).should("have.text", "We have no winner.");
3494
+ });
3495
+ });