jspsych 6.2.0 → 7.1.0

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 (360) hide show
  1. package/README.md +43 -29
  2. package/css/jspsych.css +39 -39
  3. package/dist/JsPsych.d.ts +112 -0
  4. package/dist/TimelineNode.d.ts +34 -0
  5. package/dist/index.browser.js +3164 -0
  6. package/dist/index.browser.js.map +1 -0
  7. package/dist/index.browser.min.js +2 -0
  8. package/dist/index.browser.min.js.map +1 -0
  9. package/dist/index.cjs +3158 -0
  10. package/dist/index.cjs.map +1 -0
  11. package/dist/index.d.ts +11 -0
  12. package/dist/index.js +3152 -0
  13. package/dist/index.js.map +1 -0
  14. package/dist/migration.d.ts +3 -0
  15. package/dist/modules/data/DataCollection.d.ts +45 -0
  16. package/dist/modules/data/DataColumn.d.ts +15 -0
  17. package/dist/modules/data/index.d.ts +25 -0
  18. package/dist/modules/data/utils.d.ts +3 -0
  19. package/dist/modules/extensions.d.ts +22 -0
  20. package/dist/modules/plugin-api/HardwareAPI.d.ts +15 -0
  21. package/dist/modules/plugin-api/KeyboardListenerAPI.d.ts +34 -0
  22. package/dist/modules/plugin-api/MediaAPI.d.ts +27 -0
  23. package/dist/modules/plugin-api/SimulationAPI.d.ts +41 -0
  24. package/dist/modules/plugin-api/TimeoutAPI.d.ts +5 -0
  25. package/dist/modules/plugin-api/index.d.ts +8 -0
  26. package/dist/modules/plugins.d.ts +129 -0
  27. package/dist/modules/randomization.d.ts +35 -0
  28. package/dist/modules/turk.d.ts +40 -0
  29. package/dist/modules/utils.d.ts +7 -0
  30. package/package.json +32 -15
  31. package/src/JsPsych.ts +884 -0
  32. package/src/TimelineNode.ts +536 -0
  33. package/src/index.ts +71 -0
  34. package/src/migration.ts +37 -0
  35. package/src/modules/data/DataCollection.ts +198 -0
  36. package/src/modules/data/DataColumn.ts +86 -0
  37. package/src/modules/data/index.ts +174 -0
  38. package/src/modules/data/utils.ts +75 -0
  39. package/src/modules/extensions.ts +23 -0
  40. package/src/modules/plugin-api/HardwareAPI.ts +32 -0
  41. package/src/modules/plugin-api/KeyboardListenerAPI.ts +165 -0
  42. package/src/modules/plugin-api/MediaAPI.ts +337 -0
  43. package/src/modules/plugin-api/SimulationAPI.ts +181 -0
  44. package/src/modules/plugin-api/TimeoutAPI.ts +16 -0
  45. package/src/modules/plugin-api/index.ts +28 -0
  46. package/src/modules/plugins.ts +158 -0
  47. package/src/modules/randomization.ts +327 -0
  48. package/src/modules/turk.ts +99 -0
  49. package/src/modules/utils.ts +30 -0
  50. package/.github/workflows/jest.yml +0 -20
  51. package/code-of-conduct.md +0 -56
  52. package/contributors.md +0 -61
  53. package/docs/CNAME +0 -1
  54. package/docs/about/about.md +0 -18
  55. package/docs/about/contributing.md +0 -43
  56. package/docs/about/license.md +0 -25
  57. package/docs/about/support.md +0 -7
  58. package/docs/core_library/jspsych-core.md +0 -661
  59. package/docs/core_library/jspsych-data.md +0 -589
  60. package/docs/core_library/jspsych-pluginAPI.md +0 -510
  61. package/docs/core_library/jspsych-randomization.md +0 -397
  62. package/docs/core_library/jspsych-turk.md +0 -102
  63. package/docs/img/blue.png +0 -0
  64. package/docs/img/folder-setup.png +0 -0
  65. package/docs/img/folder-with-html.png +0 -0
  66. package/docs/img/githubreleases.jpg +0 -0
  67. package/docs/img/jspsych-favicon.png +0 -0
  68. package/docs/img/jspsych-logo-no-text-mono.svg +0 -493
  69. package/docs/img/jspsych-logo.jpg +0 -0
  70. package/docs/img/orange.png +0 -0
  71. package/docs/img/palmer_stim.png +0 -0
  72. package/docs/img/progress_bar.png +0 -0
  73. package/docs/img/visual_search_example.jpg +0 -0
  74. package/docs/index.md +0 -9
  75. package/docs/overview/browser-device-support.md +0 -35
  76. package/docs/overview/callbacks.md +0 -140
  77. package/docs/overview/data.md +0 -281
  78. package/docs/overview/exclude-browser.md +0 -32
  79. package/docs/overview/experiment-options.md +0 -121
  80. package/docs/overview/fullscreen.md +0 -36
  81. package/docs/overview/media-preloading.md +0 -91
  82. package/docs/overview/mturk.md +0 -77
  83. package/docs/overview/progress-bar.md +0 -110
  84. package/docs/overview/record-browser-interactions.md +0 -23
  85. package/docs/overview/running-experiments.md +0 -95
  86. package/docs/overview/timeline.md +0 -387
  87. package/docs/overview/trial.md +0 -142
  88. package/docs/plugins/creating-a-plugin.md +0 -79
  89. package/docs/plugins/jspsych-animation.md +0 -40
  90. package/docs/plugins/jspsych-audio-button-response.md +0 -60
  91. package/docs/plugins/jspsych-audio-keyboard-response.md +0 -58
  92. package/docs/plugins/jspsych-audio-slider-response.md +0 -53
  93. package/docs/plugins/jspsych-call-function.md +0 -81
  94. package/docs/plugins/jspsych-canvas-button-response.md +0 -66
  95. package/docs/plugins/jspsych-canvas-keyboard-response.md +0 -68
  96. package/docs/plugins/jspsych-canvas-slider-response.md +0 -89
  97. package/docs/plugins/jspsych-categorize-animation.md +0 -60
  98. package/docs/plugins/jspsych-categorize-html.md +0 -52
  99. package/docs/plugins/jspsych-categorize-image.md +0 -53
  100. package/docs/plugins/jspsych-cloze.md +0 -45
  101. package/docs/plugins/jspsych-external-html.md +0 -70
  102. package/docs/plugins/jspsych-free-sort.md +0 -55
  103. package/docs/plugins/jspsych-fullscreen.md +0 -57
  104. package/docs/plugins/jspsych-html-button-response.md +0 -42
  105. package/docs/plugins/jspsych-html-keyboard-response.md +0 -51
  106. package/docs/plugins/jspsych-html-slider-response.md +0 -45
  107. package/docs/plugins/jspsych-iat-html.md +0 -64
  108. package/docs/plugins/jspsych-iat-image.md +0 -64
  109. package/docs/plugins/jspsych-image-button-response.md +0 -46
  110. package/docs/plugins/jspsych-image-keyboard-response.md +0 -57
  111. package/docs/plugins/jspsych-image-slider-response.md +0 -52
  112. package/docs/plugins/jspsych-instructions.md +0 -58
  113. package/docs/plugins/jspsych-maxdiff.md +0 -42
  114. package/docs/plugins/jspsych-rdk.md +0 -119
  115. package/docs/plugins/jspsych-reconstruction.md +0 -48
  116. package/docs/plugins/jspsych-resize.md +0 -39
  117. package/docs/plugins/jspsych-same-different-html.md +0 -53
  118. package/docs/plugins/jspsych-same-different-image.md +0 -66
  119. package/docs/plugins/jspsych-serial-reaction-time-mouse.md +0 -50
  120. package/docs/plugins/jspsych-serial-reaction-time.md +0 -57
  121. package/docs/plugins/jspsych-survey-html-form.md +0 -50
  122. package/docs/plugins/jspsych-survey-likert.md +0 -70
  123. package/docs/plugins/jspsych-survey-multi-choice.md +0 -48
  124. package/docs/plugins/jspsych-survey-multi-select.md +0 -53
  125. package/docs/plugins/jspsych-survey-text.md +0 -63
  126. package/docs/plugins/jspsych-video-button-response.md +0 -52
  127. package/docs/plugins/jspsych-video-keyboard-response.md +0 -48
  128. package/docs/plugins/jspsych-video-slider-response.md +0 -58
  129. package/docs/plugins/jspsych-visual-search-circle.md +0 -52
  130. package/docs/plugins/jspsych-vsl-animate-occlusion.md +0 -55
  131. package/docs/plugins/jspsych-vsl-grid-scene.md +0 -62
  132. package/docs/plugins/overview.md +0 -111
  133. package/docs/tutorials/hello-world.md +0 -144
  134. package/docs/tutorials/rt-task.md +0 -1107
  135. package/examples/add-to-end-of-timeline.html +0 -32
  136. package/examples/conditional-and-loop-functions.html +0 -63
  137. package/examples/css/jquery-ui.css +0 -1225
  138. package/examples/data-add-properties.html +0 -40
  139. package/examples/data-as-function.html +0 -36
  140. package/examples/data-from-timeline.html +0 -45
  141. package/examples/data-from-url.html +0 -21
  142. package/examples/demo-flanker.html +0 -108
  143. package/examples/demo-simple-rt-task.html +0 -104
  144. package/examples/demos/demo_1.html +0 -29
  145. package/examples/demos/demo_2.html +0 -43
  146. package/examples/demos/demo_3.html +0 -58
  147. package/examples/display-element-to-embed-experiment.html +0 -73
  148. package/examples/end-active-node.html +0 -52
  149. package/examples/end-experiment.html +0 -43
  150. package/examples/exclusions.html +0 -32
  151. package/examples/external_html/simple_consent.html +0 -4
  152. package/examples/img/1.gif +0 -0
  153. package/examples/img/10.gif +0 -0
  154. package/examples/img/11.gif +0 -0
  155. package/examples/img/12.gif +0 -0
  156. package/examples/img/2.gif +0 -0
  157. package/examples/img/3.gif +0 -0
  158. package/examples/img/4.gif +0 -0
  159. package/examples/img/5.gif +0 -0
  160. package/examples/img/6.gif +0 -0
  161. package/examples/img/7.gif +0 -0
  162. package/examples/img/8.gif +0 -0
  163. package/examples/img/9.gif +0 -0
  164. package/examples/img/age/of1.jpg +0 -0
  165. package/examples/img/age/of2.jpg +0 -0
  166. package/examples/img/age/of3.jpg +0 -0
  167. package/examples/img/age/om1.jpg +0 -0
  168. package/examples/img/age/om2.jpg +0 -0
  169. package/examples/img/age/om3.jpg +0 -0
  170. package/examples/img/age/yf1.jpg +0 -0
  171. package/examples/img/age/yf4.jpg +0 -0
  172. package/examples/img/age/yf5.jpg +0 -0
  173. package/examples/img/age/ym2.jpg +0 -0
  174. package/examples/img/age/ym3.jpg +0 -0
  175. package/examples/img/age/ym5.jpg +0 -0
  176. package/examples/img/backwardN.gif +0 -0
  177. package/examples/img/blue.png +0 -0
  178. package/examples/img/con1.png +0 -0
  179. package/examples/img/con2.png +0 -0
  180. package/examples/img/fixation.gif +0 -0
  181. package/examples/img/happy_face_1.jpg +0 -0
  182. package/examples/img/happy_face_2.jpg +0 -0
  183. package/examples/img/happy_face_3.jpg +0 -0
  184. package/examples/img/happy_face_4.jpg +0 -0
  185. package/examples/img/inc1.png +0 -0
  186. package/examples/img/inc2.png +0 -0
  187. package/examples/img/normalN.gif +0 -0
  188. package/examples/img/orange.png +0 -0
  189. package/examples/img/redX.png +0 -0
  190. package/examples/img/ribbon.jpg +0 -0
  191. package/examples/img/sad_face_1.jpg +0 -0
  192. package/examples/img/sad_face_2.jpg +0 -0
  193. package/examples/img/sad_face_3.jpg +0 -0
  194. package/examples/img/sad_face_4.jpg +0 -0
  195. package/examples/js/snap.svg-min.js +0 -21
  196. package/examples/jspsych-RDK.html +0 -58
  197. package/examples/jspsych-animation.html +0 -33
  198. package/examples/jspsych-audio-button-response.html +0 -52
  199. package/examples/jspsych-audio-keyboard-response.html +0 -62
  200. package/examples/jspsych-audio-slider-response.html +0 -55
  201. package/examples/jspsych-call-function.html +0 -32
  202. package/examples/jspsych-canvas-button-response.html +0 -95
  203. package/examples/jspsych-canvas-keyboard-response.html +0 -78
  204. package/examples/jspsych-canvas-slider-response.html +0 -67
  205. package/examples/jspsych-categorize-animation.html +0 -46
  206. package/examples/jspsych-categorize-html.html +0 -38
  207. package/examples/jspsych-categorize-image.html +0 -38
  208. package/examples/jspsych-cloze.html +0 -42
  209. package/examples/jspsych-free-sort.html +0 -97
  210. package/examples/jspsych-fullscreen.html +0 -44
  211. package/examples/jspsych-html-button-response.html +0 -46
  212. package/examples/jspsych-html-keyboard-response.html +0 -42
  213. package/examples/jspsych-html-slider-response.html +0 -53
  214. package/examples/jspsych-iat.html +0 -510
  215. package/examples/jspsych-image-button-response.html +0 -84
  216. package/examples/jspsych-image-keyboard-response.html +0 -78
  217. package/examples/jspsych-image-slider-response.html +0 -76
  218. package/examples/jspsych-instructions.html +0 -37
  219. package/examples/jspsych-maxdiff.html +0 -33
  220. package/examples/jspsych-reconstruction.html +0 -43
  221. package/examples/jspsych-resize.html +0 -34
  222. package/examples/jspsych-same-different-html.html +0 -28
  223. package/examples/jspsych-same-different-image.html +0 -33
  224. package/examples/jspsych-serial-reaction-time-mouse.html +0 -98
  225. package/examples/jspsych-serial-reaction-time.html +0 -54
  226. package/examples/jspsych-survey-html-form.html +0 -33
  227. package/examples/jspsych-survey-likert.html +0 -42
  228. package/examples/jspsych-survey-multi-choice.html +0 -40
  229. package/examples/jspsych-survey-multi-select.html +0 -42
  230. package/examples/jspsych-survey-text.html +0 -34
  231. package/examples/jspsych-video-button-response.html +0 -57
  232. package/examples/jspsych-video-keyboard-response.html +0 -53
  233. package/examples/jspsych-video-slider-response.html +0 -55
  234. package/examples/jspsych-visual-search-circle.html +0 -58
  235. package/examples/jspsych-vsl-animate-occlusion.html +0 -29
  236. package/examples/jspsych-vsl-grid-scene.html +0 -41
  237. package/examples/lexical-decision.html +0 -132
  238. package/examples/manual-preloading.html +0 -53
  239. package/examples/pause-unpause.html +0 -33
  240. package/examples/progress-bar.html +0 -62
  241. package/examples/sound/hammer.mp3 +0 -0
  242. package/examples/sound/sound.mp3 +0 -0
  243. package/examples/sound/speech_blue.mp3 +0 -0
  244. package/examples/sound/speech_green.mp3 +0 -0
  245. package/examples/sound/speech_joke.mp3 +0 -0
  246. package/examples/sound/speech_red.mp3 +0 -0
  247. package/examples/sound/tone.mp3 +0 -0
  248. package/examples/timeline-variables-sampling.html +0 -50
  249. package/examples/timeline-variables.html +0 -55
  250. package/examples/video/sample_video.mp4 +0 -0
  251. package/jspsych.js +0 -2796
  252. package/license.txt +0 -21
  253. package/mkdocs.yml +0 -104
  254. package/plugins/jspsych-animation.js +0 -189
  255. package/plugins/jspsych-audio-button-response.js +0 -247
  256. package/plugins/jspsych-audio-keyboard-response.js +0 -204
  257. package/plugins/jspsych-audio-slider-response.js +0 -262
  258. package/plugins/jspsych-call-function.js +0 -58
  259. package/plugins/jspsych-canvas-button-response.js +0 -199
  260. package/plugins/jspsych-canvas-keyboard-response.js +0 -155
  261. package/plugins/jspsych-canvas-slider-response.js +0 -207
  262. package/plugins/jspsych-categorize-animation.js +0 -266
  263. package/plugins/jspsych-categorize-html.js +0 -220
  264. package/plugins/jspsych-categorize-image.js +0 -222
  265. package/plugins/jspsych-cloze.js +0 -112
  266. package/plugins/jspsych-external-html.js +0 -112
  267. package/plugins/jspsych-free-sort.js +0 -444
  268. package/plugins/jspsych-fullscreen.js +0 -104
  269. package/plugins/jspsych-html-button-response.js +0 -188
  270. package/plugins/jspsych-html-keyboard-response.js +0 -149
  271. package/plugins/jspsych-html-slider-response.js +0 -202
  272. package/plugins/jspsych-iat-html.js +0 -284
  273. package/plugins/jspsych-iat-image.js +0 -286
  274. package/plugins/jspsych-image-button-response.js +0 -311
  275. package/plugins/jspsych-image-keyboard-response.js +0 -247
  276. package/plugins/jspsych-image-slider-response.js +0 -353
  277. package/plugins/jspsych-instructions.js +0 -237
  278. package/plugins/jspsych-maxdiff.js +0 -174
  279. package/plugins/jspsych-rdk.js +0 -1373
  280. package/plugins/jspsych-reconstruction.js +0 -134
  281. package/plugins/jspsych-resize.js +0 -166
  282. package/plugins/jspsych-same-different-html.js +0 -168
  283. package/plugins/jspsych-same-different-image.js +0 -169
  284. package/plugins/jspsych-serial-reaction-time-mouse.js +0 -213
  285. package/plugins/jspsych-serial-reaction-time.js +0 -247
  286. package/plugins/jspsych-survey-html-form.js +0 -171
  287. package/plugins/jspsych-survey-likert.js +0 -195
  288. package/plugins/jspsych-survey-multi-choice.js +0 -208
  289. package/plugins/jspsych-survey-multi-select.js +0 -232
  290. package/plugins/jspsych-survey-text.js +0 -185
  291. package/plugins/jspsych-video-button-response.js +0 -320
  292. package/plugins/jspsych-video-keyboard-response.js +0 -279
  293. package/plugins/jspsych-video-slider-response.js +0 -351
  294. package/plugins/jspsych-visual-search-circle.js +0 -259
  295. package/plugins/jspsych-vsl-animate-occlusion.js +0 -196
  296. package/plugins/jspsych-vsl-grid-scene.js +0 -103
  297. package/plugins/template/jspsych-plugin-template.js +0 -35
  298. package/tests/README.md +0 -7
  299. package/tests/jsPsych/default-iti.test.js +0 -51
  300. package/tests/jsPsych/default-parameters.test.js +0 -58
  301. package/tests/jsPsych/endexperiment.test.js +0 -49
  302. package/tests/jsPsych/events.test.js +0 -369
  303. package/tests/jsPsych/init.test.js +0 -48
  304. package/tests/jsPsych/loads.test.js +0 -7
  305. package/tests/jsPsych/min-rt.test.js +0 -58
  306. package/tests/jsPsych/progressbar.test.js +0 -202
  307. package/tests/jsPsych/timeline-variables.test.js +0 -254
  308. package/tests/jsPsych/timelines.test.js +0 -498
  309. package/tests/jsPsych.data/datacollection.test.js +0 -116
  310. package/tests/jsPsych.data/datacolumn.test.js +0 -50
  311. package/tests/jsPsych.data/datamodule.test.js +0 -152
  312. package/tests/jsPsych.data/dataparameter.test.js +0 -251
  313. package/tests/jsPsych.data/interactions.test.js +0 -109
  314. package/tests/jsPsych.pluginAPI/pluginapi.test.js +0 -144
  315. package/tests/jsPsych.randomization/randomziation.test.js +0 -27
  316. package/tests/jsPsych.utils/utils.test.js +0 -58
  317. package/tests/media/blue.png +0 -0
  318. package/tests/media/orange.png +0 -0
  319. package/tests/media/sample_video.mp4 +0 -0
  320. package/tests/media/sound.mp3 +0 -0
  321. package/tests/plugins/plugin-animation.test.js +0 -35
  322. package/tests/plugins/plugin-audio-button-response.test.js +0 -15
  323. package/tests/plugins/plugin-audio-keyboard-response.test.js +0 -15
  324. package/tests/plugins/plugin-audio-slider-response.test.js +0 -15
  325. package/tests/plugins/plugin-call-function.test.js +0 -49
  326. package/tests/plugins/plugin-categorize-animation.test.js +0 -274
  327. package/tests/plugins/plugin-categorize-html.test.js +0 -17
  328. package/tests/plugins/plugin-categorize-image.test.js +0 -17
  329. package/tests/plugins/plugin-cloze.test.js +0 -140
  330. package/tests/plugins/plugin-free-sort.test.js +0 -112
  331. package/tests/plugins/plugin-fullscreen.test.js +0 -41
  332. package/tests/plugins/plugin-html-button-response.test.js +0 -161
  333. package/tests/plugins/plugin-html-keyboard-response.test.js +0 -139
  334. package/tests/plugins/plugin-html-slider-response.test.js +0 -155
  335. package/tests/plugins/plugin-iat-html.test.js +0 -328
  336. package/tests/plugins/plugin-iat-image.test.js +0 -308
  337. package/tests/plugins/plugin-image-button-response.test.js +0 -183
  338. package/tests/plugins/plugin-image-keyboard-response.test.js +0 -154
  339. package/tests/plugins/plugin-image-slider-response.test.js +0 -183
  340. package/tests/plugins/plugin-instructions.test.js +0 -66
  341. package/tests/plugins/plugin-maxdiff.test.js +0 -39
  342. package/tests/plugins/plugin-rdk.test.js +0 -17
  343. package/tests/plugins/plugin-reconstruction.test.js +0 -16
  344. package/tests/plugins/plugin-resize.test.js +0 -16
  345. package/tests/plugins/plugin-same-different-html.test.js +0 -17
  346. package/tests/plugins/plugin-same-different-image.test.js +0 -17
  347. package/tests/plugins/plugin-serial-reaction-time-mouse.test.js +0 -42
  348. package/tests/plugins/plugin-serial-reaction-time.test.js +0 -69
  349. package/tests/plugins/plugin-survey-html-form.test.js +0 -44
  350. package/tests/plugins/plugin-survey-likert.test.js +0 -48
  351. package/tests/plugins/plugin-survey-multi-choice.test.js +0 -48
  352. package/tests/plugins/plugin-survey-multi-select.test.js +0 -72
  353. package/tests/plugins/plugin-survey-text.test.js +0 -115
  354. package/tests/plugins/plugin-video-button-response.test.js +0 -35
  355. package/tests/plugins/plugin-video-keyboard-response.test.js +0 -35
  356. package/tests/plugins/plugin-video-slider-response.test.js +0 -34
  357. package/tests/plugins/plugin-visual-search-circle.test.js +0 -16
  358. package/tests/plugins/plugin-vsl-animate-occlusion.test.js +0 -16
  359. package/tests/plugins/plugin-vsl-grid-scene.test.js +0 -16
  360. package/tests/testing-utils.js +0 -13
@@ -0,0 +1,198 @@
1
+ import { deepCopy } from "../utils";
2
+ import { DataColumn } from "./DataColumn";
3
+ import { JSON2CSV, saveTextToFile } from "./utils";
4
+
5
+ export class DataCollection {
6
+ private trials: any[];
7
+
8
+ constructor(data = []) {
9
+ this.trials = data;
10
+ }
11
+
12
+ push(new_data) {
13
+ this.trials.push(new_data);
14
+ return this;
15
+ }
16
+
17
+ join(other_data_collection: DataCollection) {
18
+ this.trials = this.trials.concat(other_data_collection.values());
19
+ return this;
20
+ }
21
+
22
+ top() {
23
+ if (this.trials.length <= 1) {
24
+ return this;
25
+ } else {
26
+ return new DataCollection([this.trials[this.trials.length - 1]]);
27
+ }
28
+ }
29
+
30
+ /**
31
+ * Queries the first n elements in a collection of trials.
32
+ *
33
+ * @param n A positive integer of elements to return. A value of
34
+ * n that is less than 1 will throw an error.
35
+ *
36
+ * @return First n objects of a collection of trials. If fewer than
37
+ * n trials are available, the trials.length elements will
38
+ * be returned.
39
+ *
40
+ */
41
+ first(n = 1) {
42
+ if (n < 1) {
43
+ throw `You must query with a positive nonzero integer. Please use a
44
+ different value for n.`;
45
+ }
46
+ if (this.trials.length === 0) return new DataCollection();
47
+ if (n > this.trials.length) n = this.trials.length;
48
+ return new DataCollection(this.trials.slice(0, n));
49
+ }
50
+
51
+ /**
52
+ * Queries the last n elements in a collection of trials.
53
+ *
54
+ * @param n A positive integer of elements to return. A value of
55
+ * n that is less than 1 will throw an error.
56
+ *
57
+ * @return Last n objects of a collection of trials. If fewer than
58
+ * n trials are available, the trials.length elements will
59
+ * be returned.
60
+ *
61
+ */
62
+ last(n = 1) {
63
+ if (n < 1) {
64
+ throw `You must query with a positive nonzero integer. Please use a
65
+ different value for n.`;
66
+ }
67
+ if (this.trials.length === 0) return new DataCollection();
68
+ if (n > this.trials.length) n = this.trials.length;
69
+ return new DataCollection(this.trials.slice(this.trials.length - n, this.trials.length));
70
+ }
71
+
72
+ values() {
73
+ return this.trials;
74
+ }
75
+
76
+ count() {
77
+ return this.trials.length;
78
+ }
79
+
80
+ readOnly() {
81
+ return new DataCollection(deepCopy(this.trials));
82
+ }
83
+
84
+ addToAll(properties) {
85
+ for (const trial of this.trials) {
86
+ Object.assign(trial, properties);
87
+ }
88
+ return this;
89
+ }
90
+
91
+ addToLast(properties) {
92
+ if (this.trials.length != 0) {
93
+ Object.assign(this.trials[this.trials.length - 1], properties);
94
+ }
95
+ return this;
96
+ }
97
+
98
+ filter(filters) {
99
+ // [{p1: v1, p2:v2}, {p1:v2}]
100
+ // {p1: v1}
101
+ let f;
102
+ if (!Array.isArray(filters)) {
103
+ f = deepCopy([filters]);
104
+ } else {
105
+ f = deepCopy(filters);
106
+ }
107
+
108
+ const filtered_data = [];
109
+ for (const trial of this.trials) {
110
+ let keep = false;
111
+ for (const filter of f) {
112
+ let match = true;
113
+ for (const key of Object.keys(filter)) {
114
+ if (typeof trial[key] !== "undefined" && trial[key] === filter[key]) {
115
+ // matches on this key!
116
+ } else {
117
+ match = false;
118
+ }
119
+ }
120
+ if (match) {
121
+ keep = true;
122
+ break;
123
+ } // can break because each filter is OR.
124
+ }
125
+ if (keep) {
126
+ filtered_data.push(trial);
127
+ }
128
+ }
129
+
130
+ return new DataCollection(filtered_data);
131
+ }
132
+
133
+ filterCustom(fn) {
134
+ return new DataCollection(this.trials.filter(fn));
135
+ }
136
+
137
+ select(column) {
138
+ const values = [];
139
+ for (const trial of this.trials) {
140
+ if (typeof trial[column] !== "undefined") {
141
+ values.push(trial[column]);
142
+ }
143
+ }
144
+ return new DataColumn(values);
145
+ }
146
+
147
+ ignore(columns) {
148
+ if (!Array.isArray(columns)) {
149
+ columns = [columns];
150
+ }
151
+ const o = deepCopy(this.trials);
152
+ for (const trial of o) {
153
+ for (const delete_key of columns) {
154
+ delete trial[delete_key];
155
+ }
156
+ }
157
+ return new DataCollection(o);
158
+ }
159
+
160
+ uniqueNames() {
161
+ const names = [];
162
+
163
+ for (const trial of this.trials) {
164
+ for (const key of Object.keys(trial)) {
165
+ if (!names.includes(key)) {
166
+ names.push(key);
167
+ }
168
+ }
169
+ }
170
+
171
+ return names;
172
+ }
173
+
174
+ csv() {
175
+ return JSON2CSV(this.trials);
176
+ }
177
+
178
+ json(pretty = false) {
179
+ if (pretty) {
180
+ return JSON.stringify(this.trials, null, "\t");
181
+ }
182
+ return JSON.stringify(this.trials);
183
+ }
184
+
185
+ localSave(format, filename) {
186
+ format = format.toLowerCase();
187
+ let data_string;
188
+ if (format === "json") {
189
+ data_string = this.json();
190
+ } else if (format === "csv") {
191
+ data_string = this.csv();
192
+ } else {
193
+ throw new Error('Invalid format specified for localSave. Must be "json" or "csv".');
194
+ }
195
+
196
+ saveTextToFile(data_string, filename);
197
+ }
198
+ }
@@ -0,0 +1,86 @@
1
+ export class DataColumn {
2
+ constructor(public values = []) {}
3
+
4
+ sum() {
5
+ let s = 0;
6
+ for (const v of this.values) {
7
+ s += v;
8
+ }
9
+ return s;
10
+ }
11
+
12
+ mean() {
13
+ return this.sum() / this.count();
14
+ }
15
+
16
+ median() {
17
+ if (this.values.length === 0) {
18
+ return undefined;
19
+ }
20
+ const numbers = this.values.slice(0).sort(function (a, b) {
21
+ return a - b;
22
+ });
23
+ const middle = Math.floor(numbers.length / 2);
24
+ const isEven = numbers.length % 2 === 0;
25
+ return isEven ? (numbers[middle] + numbers[middle - 1]) / 2 : numbers[middle];
26
+ }
27
+
28
+ min() {
29
+ return Math.min.apply(null, this.values);
30
+ }
31
+
32
+ max() {
33
+ return Math.max.apply(null, this.values);
34
+ }
35
+
36
+ count() {
37
+ return this.values.length;
38
+ }
39
+
40
+ variance() {
41
+ const mean = this.mean();
42
+ let sum_square_error = 0;
43
+ for (const x of this.values) {
44
+ sum_square_error += Math.pow(x - mean, 2);
45
+ }
46
+ const mse = sum_square_error / (this.values.length - 1);
47
+ return mse;
48
+ }
49
+
50
+ sd() {
51
+ const mse = this.variance();
52
+ const rmse = Math.sqrt(mse);
53
+ return rmse;
54
+ }
55
+
56
+ frequencies() {
57
+ const unique = {};
58
+ for (const x of this.values) {
59
+ if (typeof unique[x] === "undefined") {
60
+ unique[x] = 1;
61
+ } else {
62
+ unique[x]++;
63
+ }
64
+ }
65
+ return unique;
66
+ }
67
+
68
+ all(eval_fn) {
69
+ for (const x of this.values) {
70
+ if (!eval_fn(x)) {
71
+ return false;
72
+ }
73
+ }
74
+ return true;
75
+ }
76
+
77
+ subset(eval_fn) {
78
+ const out = [];
79
+ for (const x of this.values) {
80
+ if (eval_fn(x)) {
81
+ out.push(x);
82
+ }
83
+ }
84
+ return new DataColumn(out);
85
+ }
86
+ }
@@ -0,0 +1,174 @@
1
+ import { JsPsych } from "../../JsPsych";
2
+ import { DataCollection } from "./DataCollection";
3
+ import { getQueryString } from "./utils";
4
+
5
+ export class JsPsychData {
6
+ // data storage object
7
+ private allData: DataCollection;
8
+
9
+ // browser interaction event data
10
+ private interactionData: DataCollection;
11
+
12
+ // data properties for all trials
13
+ private dataProperties = {};
14
+
15
+ // cache the query_string
16
+ private query_string;
17
+
18
+ constructor(private jsPsych: JsPsych) {
19
+ this.reset();
20
+ }
21
+
22
+ reset() {
23
+ this.allData = new DataCollection();
24
+ this.interactionData = new DataCollection();
25
+ }
26
+
27
+ get() {
28
+ return this.allData;
29
+ }
30
+
31
+ getInteractionData() {
32
+ return this.interactionData;
33
+ }
34
+
35
+ write(data_object) {
36
+ const progress = this.jsPsych.getProgress();
37
+ const trial = this.jsPsych.getCurrentTrial();
38
+
39
+ //var trial_opt_data = typeof trial.data == 'function' ? trial.data() : trial.data;
40
+
41
+ const default_data = {
42
+ trial_type: trial.type.info.name,
43
+ trial_index: progress.current_trial_global,
44
+ time_elapsed: this.jsPsych.getTotalTime(),
45
+ internal_node_id: this.jsPsych.getCurrentTimelineNodeID(),
46
+ };
47
+
48
+ this.allData.push({
49
+ ...data_object,
50
+ ...trial.data,
51
+ ...default_data,
52
+ ...this.dataProperties,
53
+ });
54
+ }
55
+
56
+ addProperties(properties) {
57
+ // first, add the properties to all data that's already stored
58
+ this.allData.addToAll(properties);
59
+
60
+ // now add to list so that it gets appended to all future data
61
+ this.dataProperties = Object.assign({}, this.dataProperties, properties);
62
+ }
63
+
64
+ addDataToLastTrial(data) {
65
+ this.allData.addToLast(data);
66
+ }
67
+
68
+ getDataByTimelineNode(node_id) {
69
+ return this.allData.filterCustom(
70
+ (x) => x.internal_node_id.slice(0, node_id.length) === node_id
71
+ );
72
+ }
73
+
74
+ getLastTrialData() {
75
+ return this.allData.top();
76
+ }
77
+
78
+ getLastTimelineData() {
79
+ const lasttrial = this.getLastTrialData();
80
+ const node_id = lasttrial.select("internal_node_id").values[0];
81
+ if (typeof node_id === "undefined") {
82
+ return new DataCollection();
83
+ } else {
84
+ const parent_node_id = node_id.substr(0, node_id.lastIndexOf("-"));
85
+ const lastnodedata = this.getDataByTimelineNode(parent_node_id);
86
+ return lastnodedata;
87
+ }
88
+ }
89
+
90
+ displayData(format = "json") {
91
+ format = format.toLowerCase();
92
+ if (format != "json" && format != "csv") {
93
+ console.log("Invalid format declared for displayData function. Using json as default.");
94
+ format = "json";
95
+ }
96
+
97
+ const data_string = format === "json" ? this.allData.json(true) : this.allData.csv();
98
+
99
+ const display_element = this.jsPsych.getDisplayElement();
100
+
101
+ display_element.innerHTML = '<pre id="jspsych-data-display"></pre>';
102
+
103
+ document.getElementById("jspsych-data-display").textContent = data_string;
104
+ }
105
+
106
+ urlVariables() {
107
+ if (typeof this.query_string == "undefined") {
108
+ this.query_string = getQueryString();
109
+ }
110
+ return this.query_string;
111
+ }
112
+
113
+ getURLVariable(whichvar) {
114
+ return this.urlVariables()[whichvar];
115
+ }
116
+
117
+ createInteractionListeners() {
118
+ // blur event capture
119
+ window.addEventListener("blur", () => {
120
+ const data = {
121
+ event: "blur",
122
+ trial: this.jsPsych.getProgress().current_trial_global,
123
+ time: this.jsPsych.getTotalTime(),
124
+ };
125
+ this.interactionData.push(data);
126
+ this.jsPsych.getInitSettings().on_interaction_data_update(data);
127
+ });
128
+
129
+ // focus event capture
130
+ window.addEventListener("focus", () => {
131
+ const data = {
132
+ event: "focus",
133
+ trial: this.jsPsych.getProgress().current_trial_global,
134
+ time: this.jsPsych.getTotalTime(),
135
+ };
136
+ this.interactionData.push(data);
137
+ this.jsPsych.getInitSettings().on_interaction_data_update(data);
138
+ });
139
+
140
+ // fullscreen change capture
141
+ const fullscreenchange = () => {
142
+ const data = {
143
+ event:
144
+ // @ts-expect-error
145
+ document.isFullScreen ||
146
+ // @ts-expect-error
147
+ document.webkitIsFullScreen ||
148
+ // @ts-expect-error
149
+ document.mozIsFullScreen ||
150
+ document.fullscreenElement
151
+ ? "fullscreenenter"
152
+ : "fullscreenexit",
153
+ trial: this.jsPsych.getProgress().current_trial_global,
154
+ time: this.jsPsych.getTotalTime(),
155
+ };
156
+ this.interactionData.push(data);
157
+ this.jsPsych.getInitSettings().on_interaction_data_update(data);
158
+ };
159
+
160
+ document.addEventListener("fullscreenchange", fullscreenchange);
161
+ document.addEventListener("mozfullscreenchange", fullscreenchange);
162
+ document.addEventListener("webkitfullscreenchange", fullscreenchange);
163
+ }
164
+
165
+ // public methods for testing purposes. not recommended for use.
166
+ _customInsert(data) {
167
+ this.allData = new DataCollection(data);
168
+ }
169
+
170
+ _fullreset() {
171
+ this.reset();
172
+ this.dataProperties = {};
173
+ }
174
+ }
@@ -0,0 +1,75 @@
1
+ // private function to save text file on local drive
2
+ export function saveTextToFile(textstr: string, filename: string) {
3
+ const blobToSave = new Blob([textstr], {
4
+ type: "text/plain",
5
+ });
6
+ let blobURL = "";
7
+ if (typeof window.webkitURL !== "undefined") {
8
+ blobURL = window.webkitURL.createObjectURL(blobToSave);
9
+ } else {
10
+ blobURL = window.URL.createObjectURL(blobToSave);
11
+ }
12
+
13
+ const link = document.createElement("a");
14
+ link.id = "jspsych-download-as-text-link";
15
+ link.style.display = "none";
16
+ link.download = filename;
17
+ link.href = blobURL;
18
+ link.click();
19
+ }
20
+
21
+ // this function based on code suggested by StackOverflow users:
22
+ // http://stackoverflow.com/users/64741/zachary
23
+ // http://stackoverflow.com/users/317/joseph-sturtevant
24
+
25
+ export function JSON2CSV(objArray) {
26
+ const array = typeof objArray != "object" ? JSON.parse(objArray) : objArray;
27
+ let line = "";
28
+ let result = "";
29
+ const columns = [];
30
+
31
+ for (const row of array) {
32
+ for (const key in row) {
33
+ let keyString = key + "";
34
+ keyString = '"' + keyString.replace(/"/g, '""') + '",';
35
+ if (!columns.includes(key)) {
36
+ columns.push(key);
37
+ line += keyString;
38
+ }
39
+ }
40
+ }
41
+
42
+ line = line.slice(0, -1); // removes last comma
43
+ result += line + "\r\n";
44
+
45
+ for (const row of array) {
46
+ line = "";
47
+ for (const col of columns) {
48
+ let value = typeof row[col] === "undefined" ? "" : row[col];
49
+ if (typeof value == "object") {
50
+ value = JSON.stringify(value);
51
+ }
52
+ const valueString = value + "";
53
+ line += '"' + valueString.replace(/"/g, '""') + '",';
54
+ }
55
+
56
+ line = line.slice(0, -1);
57
+ result += line + "\r\n";
58
+ }
59
+
60
+ return result;
61
+ }
62
+
63
+ // this function is modified from StackOverflow:
64
+ // http://stackoverflow.com/posts/3855394
65
+
66
+ export function getQueryString() {
67
+ const a = window.location.search.substr(1).split("&");
68
+ const b = {};
69
+ for (let i = 0; i < a.length; ++i) {
70
+ const p = a[i].split("=", 2);
71
+ if (p.length == 1) b[p[0]] = "";
72
+ else b[p[0]] = decodeURIComponent(p[1].replace(/\+/g, " "));
73
+ }
74
+ return b;
75
+ }
@@ -0,0 +1,23 @@
1
+ export interface JsPsychExtensionInfo {
2
+ name: string;
3
+ }
4
+
5
+ export interface JsPsychExtension {
6
+ /**
7
+ * Called once at the start of the experiment to initialize the extension
8
+ */
9
+ initialize(params?: Record<string, any>): Promise<void>;
10
+ /**
11
+ * Called at the start of a trial, prior to invoking the plugin's trial method.
12
+ */
13
+ on_start(params?: Record<string, any>): void;
14
+ /**
15
+ * Called during a trial, after the plugin makes initial changes to the DOM.
16
+ */
17
+ on_load(params?: Record<string, any>): void;
18
+ /**
19
+ * Called at the end of the trial.
20
+ * @returns Data to append to the trial's data object.
21
+ */
22
+ on_finish(params?: Record<string, any>): Record<string, any>;
23
+ }
@@ -0,0 +1,32 @@
1
+ export class HardwareAPI {
2
+ /**
3
+ * Indicates whether this instance of jspsych has opened a hardware connection through our browser
4
+ * extension
5
+ **/
6
+ hardwareConnected = false;
7
+
8
+ constructor() {
9
+ //it might be useful to open up a line of communication from the extension back to this page
10
+ //script, again, this will have to pass through DOM events. For now speed is of no concern so I
11
+ //will use jQuery
12
+ document.addEventListener("jspsych-activate", (evt) => {
13
+ this.hardwareConnected = true;
14
+ });
15
+ }
16
+
17
+ /**
18
+ * Allows communication with user hardware through our custom Google Chrome extension + native C++ program
19
+ * @param mess The message to be passed to our extension, see its documentation for the expected members of this object.
20
+ * @author Daniel Rivas
21
+ *
22
+ */
23
+ hardware(mess) {
24
+ //since Chrome extension content-scripts do not share the javascript environment with the page
25
+ //script that loaded jspsych, we will need to use hacky methods like communicating through DOM
26
+ //events.
27
+ const jspsychEvt = new CustomEvent("jspsych", { detail: mess });
28
+ document.dispatchEvent(jspsychEvt);
29
+ //And voila! it will be the job of the content script injected by the extension to listen for
30
+ //the event and do the appropriate actions.
31
+ }
32
+ }