mirror of
https://github.com/jspsych/jsPsych.git
synced 2025-05-12 08:38:11 +00:00

- needed to change keyboardResponse method to support measuring RT off of the audio context - related to issues #3 and #4
135 lines
3.7 KiB
JavaScript
135 lines
3.7 KiB
JavaScript
/**
|
|
* jspsych-single-audio
|
|
* Josh de Leeuw
|
|
*
|
|
* plugin for playing an audio file and getting a keyboard response
|
|
*
|
|
* documentation: docs.jspsych.org
|
|
*
|
|
**/
|
|
|
|
(function($) {
|
|
jsPsych["single-audio"] = (function() {
|
|
|
|
var plugin = {};
|
|
|
|
var context = new AudioContext();
|
|
|
|
plugin.create = function(params) {
|
|
|
|
//params = jsPsych.pluginAPI.enforceArray(params, ['stimuli', 'choices', 'data']);
|
|
|
|
var trials = new Array(params.stimuli.length);
|
|
|
|
for (var i = 0; i < trials.length; i++) {
|
|
|
|
trials[i] = {};
|
|
trials[i].audio_stim = jsPsych.pluginAPI.loadAudioFile(params.stimuli[i]);
|
|
trials[i].audio_path = params.stimuli[i];
|
|
trials[i].choices = params.choices || [];
|
|
// option to show image for fixed time interval, ignoring key responses
|
|
// true = image will keep displaying after response
|
|
// false = trial will immediately advance when response is recorded
|
|
trials[i].continue_after_response = (typeof params.continue_after_response === 'undefined') ? true : params.continue_after_response;
|
|
// timing parameters
|
|
// trials[i].timing_stim = params.timing_stim || -1; // if -1, then show indefinitely
|
|
trials[i].timing_response = params.timing_response || -1; // if -1, then wait for response forever
|
|
trials[i].prompt = (typeof params.prompt === 'undefined') ? "" : params.prompt;
|
|
|
|
}
|
|
|
|
return trials;
|
|
};
|
|
|
|
plugin.trial = function(display_element, trial) {
|
|
|
|
// if any trial variables are functions
|
|
// this evaluates the function and replaces
|
|
// it with the output of the function
|
|
trial = jsPsych.pluginAPI.normalizeTrialVariables(trial);
|
|
|
|
// this array holds handlers from setTimeout calls
|
|
// that need to be cleared if the trial ends early
|
|
var setTimeoutHandlers = [];
|
|
|
|
// play stimulus
|
|
var source = context.createBufferSource();
|
|
source.buffer = jsPsych.pluginAPI.getAudioBuffer(trial.audio_stim);
|
|
source.connect(context.destination);
|
|
startTime = context.currentTime + 0.1;
|
|
source.start(startTime);
|
|
|
|
// show prompt if there is one
|
|
if (trial.prompt !== "") {
|
|
display_element.append(trial.prompt);
|
|
}
|
|
|
|
// store response
|
|
var response = {rt: -1, key: -1};
|
|
|
|
// function to end trial when it is time
|
|
var end_trial = function() {
|
|
|
|
// kill any remaining setTimeout handlers
|
|
for (var i = 0; i < setTimeoutHandlers.length; i++) {
|
|
clearTimeout(setTimeoutHandlers[i]);
|
|
}
|
|
|
|
// stop the audio file if it is playing
|
|
source.stop();
|
|
|
|
// kill keyboard listeners
|
|
jsPsych.pluginAPI.cancelKeyboardResponse(keyboardListener);
|
|
|
|
// gather the data to store for the trial
|
|
var trial_data = {
|
|
"rt": response.rt,
|
|
"stimulus": trial.audio_path,
|
|
"key_press": response.key
|
|
};
|
|
|
|
jsPsych.data.write($.extend({}, trial_data, trial.data));
|
|
|
|
// clear the display
|
|
display_element.html('');
|
|
|
|
// move on to the next trial
|
|
if (trial.timing_post_trial > 0) {
|
|
setTimeout(function() {
|
|
jsPsych.finishTrial();
|
|
}, trial.timing_post_trial);
|
|
} else {
|
|
jsPsych.finishTrial();
|
|
}
|
|
};
|
|
|
|
// function to handle responses by the subject
|
|
var after_response = function(info) {
|
|
|
|
// only record the first response
|
|
if(response.key == -1){
|
|
response = info;
|
|
}
|
|
|
|
if (trial.continue_after_response) {
|
|
end_trial();
|
|
}
|
|
};
|
|
|
|
// start the response listener
|
|
var keyboardListener = jsPsych.pluginAPI.getKeyboardResponse(after_response, trial.choices, 'audio', false, context, startTime);
|
|
|
|
// end trial if time limit is set
|
|
if (trial.timing_response > 0) {
|
|
var t2 = setTimeout(function() {
|
|
end_trial();
|
|
}, trial.timing_response);
|
|
setTimeoutHandlers.push(t2);
|
|
}
|
|
|
|
};
|
|
|
|
return plugin;
|
|
})();
|
|
})(jQuery);
|