-
Notifications
You must be signed in to change notification settings - Fork 0
/
jspsych-webgazer-visual-world.js
148 lines (124 loc) · 3.9 KB
/
jspsych-webgazer-visual-world.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
/**
* jspsych-webgazer-visual-world
* Josh de Leeuw
*
* documentation: docs.jspsych.org
*
**/
jsPsych.plugins["webgazer-visual-world"] = (function() {
var plugin = {};
jsPsych.pluginAPI.registerPreload('webgazer-visual-world', 'audio', 'audio');
plugin.info = {
name: 'webgazer-visual-world',
description: '',
parameters: {
audio: {
type: jsPsych.plugins.parameterType.AUDIO,
pretty_name: 'Audio Stimulus',
default: undefined,
description: 'The audio to be played.'
},
choices: {
type: jsPsych.plugins.parameterType.KEYCODE,
pretty_name: 'Choices',
array: true,
default: jsPsych.ALL_KEYS,
description: 'The keys the subject is allowed to press to respond to the stimulus.'
},
prompt: {
type: jsPsych.plugins.parameterType.STRING,
pretty_name: 'Prompt',
default: null,
description: 'Any content here will be displayed below the stimulus.'
},
trial_duration: {
type: jsPsych.plugins.parameterType.INT,
pretty_name: 'Trial duration',
default: null,
description: 'The maximum duration to wait for a response.'
},
response_ends_trial: {
type: jsPsych.plugins.parameterType.BOOL,
pretty_name: 'Response ends trial',
default: true,
description: 'If true, the trial will end when user makes a response.'
},
trial_ends_after_audio: {
type: jsPsych.plugins.parameterType.BOOL,
pretty_name: 'Trial ends after audio',
default: false,
description: 'If true, then the trial will end as soon as the audio file finishes playing.'
},
}
}
plugin.trial = function(display_element, trial) {
var gaze_data = [];
var audio_start = null;
// setup eye tracking callback
webgazer.setGazeListener(function(data){
if(data == null){
return;
}
gaze_data.push({
x: data.x - document.querySelector('#visual-world-target').offsetLeft,
y: data.y - document.querySelector('#visual-world-target').offsetTop,
t: performance.now() - audio_start
})
})
// setup audio stimulus
var context = jsPsych.pluginAPI.audioContext();
if(context !== null){
var source = context.createBufferSource();
source.buffer = jsPsych.pluginAPI.getAudioBuffer(trial.audio);
source.connect(context.destination);
} else {
var audio = jsPsych.pluginAPI.getAudioBuffer(trial.audio);
audio.currentTime = 0;
}
// set up end event if trial needs it
if(context !== null){
source.onended = function() {
end_trial();
}
} else {
audio.addEventListener('ended', end_trial);
}
// show visual scene
display_element.innerHTML = "<img id='visual-world-target' src='"+trial.image+"'>";
// function to end trial when it is time
function end_trial() {
webgazer.pause();
webgazer.clearGazeListener();
// kill any remaining setTimeout handlers
jsPsych.pluginAPI.clearAllTimeouts();
// stop the audio file if it is playing
// remove end event listeners if they exist
if(context !== null){
source.stop();
source.onended = function() { }
} else {
audio.pause();
audio.removeEventListener('ended', end_trial);
}
// kill keyboard listeners
jsPsych.pluginAPI.cancelAllKeyboardResponses();
var trial_data = {
"gaze_data": JSON.stringify(gaze_data)
};
// clear the display
display_element.innerHTML = '';
// move on to the next trial
jsPsych.finishTrial(trial_data);
};
// start audio
if(context !== null){
startTime = context.currentTime;
source.start(startTime);
} else {
audio.play();
}
audio_start = performance.now();
webgazer.resume();
};
return plugin;
})();