jspsych 6.3.0 → 7.1.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +36 -37
- package/css/jspsych.css +39 -39
- package/dist/JsPsych.d.ts +112 -0
- package/dist/TimelineNode.d.ts +34 -0
- package/dist/index.browser.js +3164 -0
- package/dist/index.browser.js.map +1 -0
- package/dist/index.browser.min.js +2 -0
- package/dist/index.browser.min.js.map +1 -0
- package/dist/index.cjs +3158 -0
- package/dist/index.cjs.map +1 -0
- package/dist/index.d.ts +11 -0
- package/dist/index.js +3152 -0
- package/dist/index.js.map +1 -0
- package/dist/migration.d.ts +3 -0
- package/dist/modules/data/DataCollection.d.ts +45 -0
- package/dist/modules/data/DataColumn.d.ts +15 -0
- package/dist/modules/data/index.d.ts +25 -0
- package/dist/modules/data/utils.d.ts +3 -0
- package/dist/modules/extensions.d.ts +22 -0
- package/dist/modules/plugin-api/HardwareAPI.d.ts +15 -0
- package/dist/modules/plugin-api/KeyboardListenerAPI.d.ts +34 -0
- package/dist/modules/plugin-api/MediaAPI.d.ts +27 -0
- package/dist/modules/plugin-api/SimulationAPI.d.ts +41 -0
- package/dist/modules/plugin-api/TimeoutAPI.d.ts +5 -0
- package/dist/modules/plugin-api/index.d.ts +8 -0
- package/dist/modules/plugins.d.ts +129 -0
- package/dist/modules/randomization.d.ts +35 -0
- package/dist/modules/turk.d.ts +40 -0
- package/dist/modules/utils.d.ts +7 -0
- package/package.json +32 -15
- package/src/JsPsych.ts +884 -0
- package/src/TimelineNode.ts +536 -0
- package/src/index.ts +71 -0
- package/src/migration.ts +37 -0
- package/src/modules/data/DataCollection.ts +198 -0
- package/src/modules/data/DataColumn.ts +86 -0
- package/src/modules/data/index.ts +174 -0
- package/src/modules/data/utils.ts +75 -0
- package/src/modules/extensions.ts +23 -0
- package/src/modules/plugin-api/HardwareAPI.ts +32 -0
- package/src/modules/plugin-api/KeyboardListenerAPI.ts +165 -0
- package/src/modules/plugin-api/MediaAPI.ts +337 -0
- package/src/modules/plugin-api/SimulationAPI.ts +181 -0
- package/src/modules/plugin-api/TimeoutAPI.ts +16 -0
- package/src/modules/plugin-api/index.ts +28 -0
- package/src/modules/plugins.ts +158 -0
- package/src/modules/randomization.ts +327 -0
- package/src/modules/turk.ts +99 -0
- package/src/modules/utils.ts +30 -0
- package/.github/workflows/jest.yml +0 -20
- package/code-of-conduct.md +0 -56
- package/contributors.md +0 -61
- package/docs/CNAME +0 -1
- package/docs/about/about.md +0 -18
- package/docs/about/contributing.md +0 -43
- package/docs/about/license.md +0 -25
- package/docs/about/support.md +0 -7
- package/docs/core_library/jspsych-core.md +0 -734
- package/docs/core_library/jspsych-data.md +0 -589
- package/docs/core_library/jspsych-pluginAPI.md +0 -610
- package/docs/core_library/jspsych-randomization.md +0 -397
- package/docs/core_library/jspsych-turk.md +0 -102
- package/docs/extensions/extensions.md +0 -83
- package/docs/extensions/jspsych-ext-webgazer.md +0 -106
- package/docs/img/blue.png +0 -0
- package/docs/img/devtools-change-css.png +0 -0
- package/docs/img/devtools-css-errors.png +0 -0
- package/docs/img/devtools-inspect-element.png +0 -0
- package/docs/img/folder-setup.png +0 -0
- package/docs/img/folder-with-html.png +0 -0
- package/docs/img/githubreleases.jpg +0 -0
- package/docs/img/jspsych-favicon.png +0 -0
- package/docs/img/jspsych-logo-no-text-mono.svg +0 -493
- package/docs/img/jspsych-logo.jpg +0 -0
- package/docs/img/orange.png +0 -0
- package/docs/img/palmer_stim.png +0 -0
- package/docs/img/progress_bar.png +0 -0
- package/docs/img/prolific-study-completion.png +0 -0
- package/docs/img/prolific-study-link.png +0 -0
- package/docs/img/visual_search_example.jpg +0 -0
- package/docs/index.md +0 -9
- package/docs/overview/browser-device-support.md +0 -35
- package/docs/overview/callbacks.md +0 -184
- package/docs/overview/data.md +0 -281
- package/docs/overview/dynamic-parameters.md +0 -147
- package/docs/overview/exclude-browser.md +0 -32
- package/docs/overview/experiment-options.md +0 -149
- package/docs/overview/eye-tracking.md +0 -237
- package/docs/overview/fullscreen.md +0 -36
- package/docs/overview/media-preloading.md +0 -369
- package/docs/overview/mturk.md +0 -77
- package/docs/overview/plugins.md +0 -320
- package/docs/overview/progress-bar.md +0 -110
- package/docs/overview/prolific.md +0 -78
- package/docs/overview/record-browser-interactions.md +0 -23
- package/docs/overview/running-experiments.md +0 -95
- package/docs/overview/style.md +0 -293
- package/docs/overview/timeline.md +0 -457
- package/docs/plugins/jspsych-animation.md +0 -40
- package/docs/plugins/jspsych-audio-button-response.md +0 -60
- package/docs/plugins/jspsych-audio-keyboard-response.md +0 -58
- package/docs/plugins/jspsych-audio-slider-response.md +0 -53
- package/docs/plugins/jspsych-call-function.md +0 -81
- package/docs/plugins/jspsych-canvas-button-response.md +0 -66
- package/docs/plugins/jspsych-canvas-keyboard-response.md +0 -68
- package/docs/plugins/jspsych-canvas-slider-response.md +0 -89
- package/docs/plugins/jspsych-categorize-animation.md +0 -60
- package/docs/plugins/jspsych-categorize-html.md +0 -53
- package/docs/plugins/jspsych-categorize-image.md +0 -53
- package/docs/plugins/jspsych-cloze.md +0 -45
- package/docs/plugins/jspsych-external-html.md +0 -70
- package/docs/plugins/jspsych-free-sort.md +0 -56
- package/docs/plugins/jspsych-fullscreen.md +0 -57
- package/docs/plugins/jspsych-html-button-response.md +0 -42
- package/docs/plugins/jspsych-html-keyboard-response.md +0 -51
- package/docs/plugins/jspsych-html-slider-response.md +0 -45
- package/docs/plugins/jspsych-iat-html.md +0 -64
- package/docs/plugins/jspsych-iat-image.md +0 -64
- package/docs/plugins/jspsych-image-button-response.md +0 -48
- package/docs/plugins/jspsych-image-keyboard-response.md +0 -58
- package/docs/plugins/jspsych-image-slider-response.md +0 -54
- package/docs/plugins/jspsych-instructions.md +0 -58
- package/docs/plugins/jspsych-maxdiff.md +0 -41
- package/docs/plugins/jspsych-preload.md +0 -128
- package/docs/plugins/jspsych-rdk.md +0 -119
- package/docs/plugins/jspsych-reconstruction.md +0 -48
- package/docs/plugins/jspsych-resize.md +0 -39
- package/docs/plugins/jspsych-same-different-html.md +0 -53
- package/docs/plugins/jspsych-same-different-image.md +0 -66
- package/docs/plugins/jspsych-serial-reaction-time-mouse.md +0 -52
- package/docs/plugins/jspsych-serial-reaction-time.md +0 -57
- package/docs/plugins/jspsych-survey-html-form.md +0 -50
- package/docs/plugins/jspsych-survey-likert.md +0 -70
- package/docs/plugins/jspsych-survey-multi-choice.md +0 -48
- package/docs/plugins/jspsych-survey-multi-select.md +0 -53
- package/docs/plugins/jspsych-survey-text.md +0 -63
- package/docs/plugins/jspsych-video-button-response.md +0 -54
- package/docs/plugins/jspsych-video-keyboard-response.md +0 -50
- package/docs/plugins/jspsych-video-slider-response.md +0 -60
- package/docs/plugins/jspsych-virtual-chinrest.md +0 -105
- package/docs/plugins/jspsych-visual-search-circle.md +0 -52
- package/docs/plugins/jspsych-vsl-animate-occlusion.md +0 -55
- package/docs/plugins/jspsych-vsl-grid-scene.md +0 -62
- package/docs/plugins/jspsych-webgazer-calibrate.md +0 -60
- package/docs/plugins/jspsych-webgazer-init-camera.md +0 -31
- package/docs/plugins/jspsych-webgazer-validate.md +0 -43
- package/docs/plugins/list-of-plugins.md +0 -54
- package/docs/tutorials/hello-world.md +0 -162
- package/docs/tutorials/rt-task.md +0 -1334
- package/docs/tutorials/video-tutorials.md +0 -11
- package/examples/add-to-end-of-timeline.html +0 -38
- package/examples/case-sensitive-responses.html +0 -45
- package/examples/conditional-and-loop-functions.html +0 -64
- package/examples/css/jquery-ui.css +0 -1225
- package/examples/css-classes-parameter.html +0 -145
- package/examples/data-add-properties.html +0 -44
- package/examples/data-as-function.html +0 -39
- package/examples/data-from-timeline.html +0 -52
- package/examples/data-from-url.html +0 -21
- package/examples/demo-flanker.html +0 -117
- package/examples/demo-simple-rt-task.html +0 -120
- package/examples/demos/demo_1.html +0 -35
- package/examples/demos/demo_2.html +0 -50
- package/examples/demos/demo_3.html +0 -63
- package/examples/display-element-to-embed-experiment.html +0 -79
- package/examples/end-active-node.html +0 -52
- package/examples/end-experiment.html +0 -45
- package/examples/exclusions.html +0 -32
- package/examples/external_html/simple_consent.html +0 -4
- package/examples/img/1.gif +0 -0
- package/examples/img/10.gif +0 -0
- package/examples/img/11.gif +0 -0
- package/examples/img/12.gif +0 -0
- package/examples/img/2.gif +0 -0
- package/examples/img/3.gif +0 -0
- package/examples/img/4.gif +0 -0
- package/examples/img/5.gif +0 -0
- package/examples/img/6.gif +0 -0
- package/examples/img/7.gif +0 -0
- package/examples/img/8.gif +0 -0
- package/examples/img/9.gif +0 -0
- package/examples/img/age/of1.jpg +0 -0
- package/examples/img/age/of2.jpg +0 -0
- package/examples/img/age/of3.jpg +0 -0
- package/examples/img/age/om1.jpg +0 -0
- package/examples/img/age/om2.jpg +0 -0
- package/examples/img/age/om3.jpg +0 -0
- package/examples/img/age/yf1.jpg +0 -0
- package/examples/img/age/yf4.jpg +0 -0
- package/examples/img/age/yf5.jpg +0 -0
- package/examples/img/age/ym2.jpg +0 -0
- package/examples/img/age/ym3.jpg +0 -0
- package/examples/img/age/ym5.jpg +0 -0
- package/examples/img/backwardN.gif +0 -0
- package/examples/img/blue.png +0 -0
- package/examples/img/card.png +0 -0
- package/examples/img/con1.png +0 -0
- package/examples/img/con2.png +0 -0
- package/examples/img/fixation.gif +0 -0
- package/examples/img/happy_face_1.jpg +0 -0
- package/examples/img/happy_face_2.jpg +0 -0
- package/examples/img/happy_face_3.jpg +0 -0
- package/examples/img/happy_face_4.jpg +0 -0
- package/examples/img/inc1.png +0 -0
- package/examples/img/inc2.png +0 -0
- package/examples/img/normalN.gif +0 -0
- package/examples/img/orange.png +0 -0
- package/examples/img/redX.png +0 -0
- package/examples/img/ribbon.jpg +0 -0
- package/examples/img/sad_face_1.jpg +0 -0
- package/examples/img/sad_face_2.jpg +0 -0
- package/examples/img/sad_face_3.jpg +0 -0
- package/examples/img/sad_face_4.jpg +0 -0
- package/examples/js/snap.svg-min.js +0 -21
- package/examples/js/webgazer.js +0 -88886
- package/examples/jspsych-RDK.html +0 -58
- package/examples/jspsych-animation.html +0 -39
- package/examples/jspsych-audio-button-response.html +0 -58
- package/examples/jspsych-audio-keyboard-response.html +0 -68
- package/examples/jspsych-audio-slider-response.html +0 -61
- package/examples/jspsych-call-function.html +0 -32
- package/examples/jspsych-canvas-button-response.html +0 -95
- package/examples/jspsych-canvas-keyboard-response.html +0 -78
- package/examples/jspsych-canvas-slider-response.html +0 -67
- package/examples/jspsych-categorize-animation.html +0 -49
- package/examples/jspsych-categorize-html.html +0 -33
- package/examples/jspsych-categorize-image.html +0 -44
- package/examples/jspsych-cloze.html +0 -37
- package/examples/jspsych-free-sort.html +0 -109
- package/examples/jspsych-fullscreen.html +0 -45
- package/examples/jspsych-html-button-response.html +0 -43
- package/examples/jspsych-html-keyboard-response.html +0 -42
- package/examples/jspsych-html-slider-response.html +0 -53
- package/examples/jspsych-iat.html +0 -520
- package/examples/jspsych-image-button-response.html +0 -91
- package/examples/jspsych-image-keyboard-response.html +0 -85
- package/examples/jspsych-image-slider-response.html +0 -85
- package/examples/jspsych-instructions.html +0 -37
- package/examples/jspsych-maxdiff.html +0 -33
- package/examples/jspsych-preload.html +0 -140
- package/examples/jspsych-reconstruction.html +0 -43
- package/examples/jspsych-resize.html +0 -34
- package/examples/jspsych-same-different-html.html +0 -28
- package/examples/jspsych-same-different-image.html +0 -39
- package/examples/jspsych-serial-reaction-time-mouse.html +0 -98
- package/examples/jspsych-serial-reaction-time.html +0 -54
- package/examples/jspsych-survey-html-form.html +0 -33
- package/examples/jspsych-survey-likert.html +0 -42
- package/examples/jspsych-survey-multi-choice.html +0 -40
- package/examples/jspsych-survey-multi-select.html +0 -42
- package/examples/jspsych-survey-text.html +0 -34
- package/examples/jspsych-video-button-response.html +0 -65
- package/examples/jspsych-video-keyboard-response.html +0 -61
- package/examples/jspsych-video-slider-response.html +0 -63
- package/examples/jspsych-virtual-chinrest.html +0 -69
- package/examples/jspsych-visual-search-circle.html +0 -64
- package/examples/jspsych-vsl-animate-occlusion.html +0 -35
- package/examples/jspsych-vsl-grid-scene.html +0 -47
- package/examples/lexical-decision.html +0 -134
- package/examples/manual-preloading.html +0 -59
- package/examples/pause-unpause.html +0 -33
- package/examples/progress-bar.html +0 -68
- package/examples/save-trial-parameters.html +0 -98
- package/examples/sound/hammer.mp3 +0 -0
- package/examples/sound/sound.mp3 +0 -0
- package/examples/sound/speech_blue.mp3 +0 -0
- package/examples/sound/speech_green.mp3 +0 -0
- package/examples/sound/speech_joke.mp3 +0 -0
- package/examples/sound/speech_red.mp3 +0 -0
- package/examples/sound/tone.mp3 +0 -0
- package/examples/timeline-variables-sampling.html +0 -50
- package/examples/timeline-variables.html +0 -64
- package/examples/video/sample_video.mp4 +0 -0
- package/examples/webgazer.html +0 -162
- package/examples/webgazer_image.html +0 -60
- package/extensions/jspsych-ext-webgazer.js +0 -185
- package/jspsych.js +0 -3015
- package/license.txt +0 -21
- package/mkdocs.yml +0 -118
- package/plugins/jspsych-animation.js +0 -189
- package/plugins/jspsych-audio-button-response.js +0 -269
- package/plugins/jspsych-audio-keyboard-response.js +0 -212
- package/plugins/jspsych-audio-slider-response.js +0 -278
- package/plugins/jspsych-call-function.js +0 -58
- package/plugins/jspsych-canvas-button-response.js +0 -199
- package/plugins/jspsych-canvas-keyboard-response.js +0 -155
- package/plugins/jspsych-canvas-slider-response.js +0 -207
- package/plugins/jspsych-categorize-animation.js +0 -266
- package/plugins/jspsych-categorize-html.js +0 -220
- package/plugins/jspsych-categorize-image.js +0 -222
- package/plugins/jspsych-cloze.js +0 -112
- package/plugins/jspsych-external-html.js +0 -112
- package/plugins/jspsych-free-sort.js +0 -478
- package/plugins/jspsych-fullscreen.js +0 -106
- package/plugins/jspsych-html-button-response.js +0 -188
- package/plugins/jspsych-html-keyboard-response.js +0 -149
- package/plugins/jspsych-html-slider-response.js +0 -202
- package/plugins/jspsych-iat-html.js +0 -284
- package/plugins/jspsych-iat-image.js +0 -286
- package/plugins/jspsych-image-button-response.js +0 -327
- package/plugins/jspsych-image-keyboard-response.js +0 -263
- package/plugins/jspsych-image-slider-response.js +0 -369
- package/plugins/jspsych-instructions.js +0 -237
- package/plugins/jspsych-maxdiff.js +0 -173
- package/plugins/jspsych-preload.js +0 -345
- package/plugins/jspsych-rdk.js +0 -1373
- package/plugins/jspsych-reconstruction.js +0 -134
- package/plugins/jspsych-resize.js +0 -166
- package/plugins/jspsych-same-different-html.js +0 -168
- package/plugins/jspsych-same-different-image.js +0 -169
- package/plugins/jspsych-serial-reaction-time-mouse.js +0 -212
- package/plugins/jspsych-serial-reaction-time.js +0 -247
- package/plugins/jspsych-survey-html-form.js +0 -171
- package/plugins/jspsych-survey-likert.js +0 -195
- package/plugins/jspsych-survey-multi-choice.js +0 -208
- package/plugins/jspsych-survey-multi-select.js +0 -232
- package/plugins/jspsych-survey-text.js +0 -185
- package/plugins/jspsych-video-button-response.js +0 -335
- package/plugins/jspsych-video-keyboard-response.js +0 -279
- package/plugins/jspsych-video-slider-response.js +0 -351
- package/plugins/jspsych-virtual-chinrest.js +0 -471
- package/plugins/jspsych-visual-search-circle.js +0 -259
- package/plugins/jspsych-vsl-animate-occlusion.js +0 -196
- package/plugins/jspsych-vsl-grid-scene.js +0 -103
- package/plugins/jspsych-webgazer-calibrate.js +0 -166
- package/plugins/jspsych-webgazer-init-camera.js +0 -95
- package/plugins/jspsych-webgazer-validate.js +0 -304
- package/plugins/template/jspsych-plugin-template.js +0 -35
- package/tests/README.md +0 -7
- package/tests/jsPsych/case-sensitive-responses.test.js +0 -53
- package/tests/jsPsych/css-classes-parameter.test.js +0 -107
- package/tests/jsPsych/default-iti.test.js +0 -51
- package/tests/jsPsych/default-parameters.test.js +0 -58
- package/tests/jsPsych/endexperiment.test.js +0 -49
- package/tests/jsPsych/events.test.js +0 -606
- package/tests/jsPsych/functions-as-parameters.test.js +0 -210
- package/tests/jsPsych/init.test.js +0 -48
- package/tests/jsPsych/loads.test.js +0 -7
- package/tests/jsPsych/min-rt.test.js +0 -58
- package/tests/jsPsych/progressbar.test.js +0 -202
- package/tests/jsPsych/timeline-variables.test.js +0 -531
- package/tests/jsPsych/timelines.test.js +0 -569
- package/tests/jsPsych.data/data-csv-conversion.test.js +0 -85
- package/tests/jsPsych.data/data-json-conversion.test.js +0 -120
- package/tests/jsPsych.data/datacollection.test.js +0 -117
- package/tests/jsPsych.data/datacolumn.test.js +0 -50
- package/tests/jsPsych.data/datamodule.test.js +0 -152
- package/tests/jsPsych.data/dataparameter.test.js +0 -251
- package/tests/jsPsych.data/interactions.test.js +0 -109
- package/tests/jsPsych.data/trialparameters.test.js +0 -175
- package/tests/jsPsych.extensions/extensions.test.js +0 -207
- package/tests/jsPsych.extensions/test-extension.js +0 -42
- package/tests/jsPsych.pluginAPI/pluginapi.test.js +0 -341
- package/tests/jsPsych.pluginAPI/preloads.test.js +0 -43
- package/tests/jsPsych.randomization/randomziation.test.js +0 -27
- package/tests/jsPsych.utils/utils.test.js +0 -58
- package/tests/plugins/plugin-animation.test.js +0 -34
- package/tests/plugins/plugin-audio-button-response.test.js +0 -15
- package/tests/plugins/plugin-audio-keyboard-response.test.js +0 -15
- package/tests/plugins/plugin-audio-slider-response.test.js +0 -15
- package/tests/plugins/plugin-call-function.test.js +0 -49
- package/tests/plugins/plugin-categorize-animation.test.js +0 -263
- package/tests/plugins/plugin-categorize-html.test.js +0 -17
- package/tests/plugins/plugin-categorize-image.test.js +0 -17
- package/tests/plugins/plugin-cloze.test.js +0 -157
- package/tests/plugins/plugin-free-sort.test.js +0 -106
- package/tests/plugins/plugin-fullscreen.test.js +0 -41
- package/tests/plugins/plugin-html-button-response.test.js +0 -161
- package/tests/plugins/plugin-html-keyboard-response.test.js +0 -139
- package/tests/plugins/plugin-html-slider-response.test.js +0 -155
- package/tests/plugins/plugin-iat-html.test.js +0 -299
- package/tests/plugins/plugin-iat-image.test.js +0 -298
- package/tests/plugins/plugin-image-button-response.test.js +0 -174
- package/tests/plugins/plugin-image-keyboard-response.test.js +0 -147
- package/tests/plugins/plugin-image-slider-response.test.js +0 -174
- package/tests/plugins/plugin-instructions.test.js +0 -85
- package/tests/plugins/plugin-maxdiff.test.js +0 -39
- package/tests/plugins/plugin-preload.test.js +0 -916
- package/tests/plugins/plugin-rdk.test.js +0 -61
- package/tests/plugins/plugin-reconstruction.test.js +0 -16
- package/tests/plugins/plugin-resize.test.js +0 -16
- package/tests/plugins/plugin-same-different-html.test.js +0 -17
- package/tests/plugins/plugin-same-different-image.test.js +0 -17
- package/tests/plugins/plugin-serial-reaction-time-mouse.test.js +0 -42
- package/tests/plugins/plugin-serial-reaction-time.test.js +0 -109
- package/tests/plugins/plugin-survey-html-form.test.js +0 -44
- package/tests/plugins/plugin-survey-likert.test.js +0 -48
- package/tests/plugins/plugin-survey-multi-choice.test.js +0 -47
- package/tests/plugins/plugin-survey-multi-select.test.js +0 -71
- package/tests/plugins/plugin-survey-text.test.js +0 -115
- package/tests/plugins/plugin-video-button-response.test.js +0 -32
- package/tests/plugins/plugin-video-keyboard-response.test.js +0 -32
- package/tests/plugins/plugin-video-slider-response.test.js +0 -31
- package/tests/plugins/plugin-visual-search-circle.test.js +0 -16
- package/tests/plugins/plugin-vsl-animate-occlusion.test.js +0 -16
- package/tests/plugins/plugin-vsl-grid-scene.test.js +0 -16
- 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
|
+
}
|