-
Notifications
You must be signed in to change notification settings - Fork 0
/
pupil_extract_saccade_frames.m
231 lines (167 loc) · 7.46 KB
/
pupil_extract_saccade_frames.m
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
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
% pupil_extract_saccade_frames.m
%
% Description:
% This script identifies frames in which saccades occur using DeepLabCut
% pupil-tracking data, distinguishing nasal and temporal movements. Saccades
% are detected based on the central difference of the pupil position and
% deviations from the mean. Optionally, the script saves the detected saccade
% frames and directions to a CSV file.
%
% Inputs:
% - session_id (string): Unique identifier for the recording session.
% - fname (string): Path to the DeepLabCut (DLC) tracked pupil data file.
% - lh_threshold (numeric): Likelihood threshold for filtering low-confidence data.
% - fs (numeric): Sampling rate of the camera in Hz.
% - n_sds (numeric): Number of standard deviations for saccade detection.
% - min_event_interval (numeric): Minimum interval (seconds) to avoid detecting
% events occurring too close in time.
% - save_csv (boolean): Flag to save the detected saccades in a CSV file.
%
% Outputs:
% - Figures showing pupil position, saccade events, and average pupil position
% around each saccade.
% - (Optional) CSV file containing the frame numbers and direction (nasal/temporal)
% of each saccade event.
%
% Usage:
% Set the parameters and run the script. Ensure the DLC data file is accessible.
session_id = 'CAA-1110264_rec1_001';
fname = ctl.file.camera0_dlc_pupil_slow(session_id);
lh_threshold = 0.7; % DLC likelihood threshold
fs = 60; % sampling rate of camera
n_sds = 4; % number of S.D.s to use for detecting events
min_event_interval = 0.2; % seconds, remove events which occur within this time
save_csv = false; % whether to save the saccades .csv file
%% read the DLC .csv and rename the variables
tbl = readtable(fname);
nasal.x = tbl.DLC_resnet50_pupil_trackingJan12shuffle1_350000;
nasal.y = tbl.DLC_resnet50_pupil_trackingJan12shuffle1_350000_1;
nasal.lh = tbl.DLC_resnet50_pupil_trackingJan12shuffle1_350000_2;
temporal.x = tbl.DLC_resnet50_pupil_trackingJan12shuffle1_350000_3;
temporal.y = tbl.DLC_resnet50_pupil_trackingJan12shuffle1_350000_4;
temporal.lh = tbl.DLC_resnet50_pupil_trackingJan12shuffle1_350000_5;
%%
dt = 1/fs;
% position of middle of pupil
pupil = ([nasal.x, nasal.y] + [temporal.x, temporal.y]) / 2;
% remove uncertain values
valid = nasal.lh > lh_threshold & temporal.lh > lh_threshold;
pupil(~valid, :) = nan;
% timebase of the traces
timebase = (0:size(pupil, 1)-1) * dt;
frame_n = 1:size(pupil, 1);
% central difference
t_diff = timebase(2:end-1);
frame_n_diff = frame_n(2:end-1);
pupil_diff = (pupil(3:end, 1) - pupil(1:end-2, 1))/(2*dt);
% take average and standard deviation of difference trace
mean_diff = nanmean(pupil_diff);
sd_diff = nanstd(pupil_diff);
% find where the difference trace goes negative (nasal) or positive
% (temporal)
nasal_fast_move = pupil_diff < mean_diff - n_sds*sd_diff;
temporal_fast_move = pupil_diff > mean_diff + n_sds*sd_diff;
%
nasal_fast_move_onset = [false; diff(nasal_fast_move) == 1];
temporal_fast_move_onset = [false; diff(temporal_fast_move) == 1];
% restrict to onset movements followed by another fast move
nasal_fast_move_onset = [nasal_fast_move_onset(1:end-1) & nasal_fast_move(2:end); false];
temporal_fast_move_onset = [temporal_fast_move_onset(1:end-1) & temporal_fast_move(2:end); false];
% get the frame on which the fast move occurs
nasal_fast_move_frames = frame_n_diff(nasal_fast_move_onset);
temporal_fast_move_frames = frame_n_diff(temporal_fast_move_onset);
% the times
nasal_times = timebase(nasal_fast_move_frames);
temporal_times = timebase(temporal_fast_move_frames);
% remove events which happen within 200ms of each other
delta_times = nasal_times(:) - temporal_times(:)';
delta_times = abs(delta_times) < min_event_interval;
nasal_rm = sum(delta_times, 2) > 0;
temporal_rm = sum(delta_times, 1) > 0;
nasal_fast_move_frames(nasal_rm) = [];
temporal_fast_move_frames(temporal_rm) = [];
nasal_times(nasal_rm) = [];
temporal_times(temporal_rm) = [];
fprintf('Removing %i/%i events within 200ms of each other\n', sum(nasal_rm) + sum(temporal_rm), length(nasal_rm)+length(temporal_rm));
%% plot pupil position and events in different colours
diff_centre = 392.5;
figure;
h_ax = axes();
hold on;
plot(h_ax, timebase, pupil(:, 1));
for ii = 1 : length(nasal_times)
line(h_ax, nasal_times([ii, ii]), get(h_ax, 'ylim'), 'color', 'r');
end
for ii = 1 : length(temporal_times)
line(h_ax, temporal_times([ii, ii]), get(h_ax, 'ylim'), 'color', 'g');
end
set(h_ax, 'plotboxaspectratio', [3, 1, 1])
plot(h_ax, t_diff, diff_centre + (pupil_diff - mean_diff) / (2*n_sds*sd_diff));
line(h_ax, get(h_ax, 'xlim'), diff_centre + [0.5, 0.5], 'color', 'k')
line(h_ax, get(h_ax, 'xlim'), diff_centre - [0.5, 0.5], 'color', 'k')
plot(h_ax, t_diff, diff_centre - 2.5 + nasal_fast_move);
plot(h_ax, t_diff, diff_centre - 1.5 + temporal_fast_move);
xlabel(h_ax, 'Time (s)');
ylabel(h_ax, 'Pupil position (pixels)');
%% plot pupil trace around events
frames_around_event = -30:30;
baseline_frames = -5:-1;
nasal_events = [];
for ii = 1 : length(nasal_fast_move_frames)
idx = nasal_fast_move_frames(ii) + frames_around_event;
baseline_idx = nasal_fast_move_frames(ii) + baseline_frames;
nasal_events(:, ii) = pupil(idx, 1) - mean(pupil(baseline_idx, 1));
end
figure
h_ax = axes();
hold on;
plot(h_ax, frames_around_event, nasal_events);
line(h_ax, [0, 0], get(gca, 'ylim'));
temporal_events = [];
for ii = 1 : length(temporal_fast_move_frames)
idx = temporal_fast_move_frames(ii) + frames_around_event;
baseline_idx = temporal_fast_move_frames(ii) + baseline_frames;
temporal_events(:, ii) = pupil(idx, 1) - mean(pupil(baseline_idx, 1));
end
figure
h_ax = axes();
hold on;
plot(h_ax, frames_around_event, temporal_events);
line(h_ax, [0, 0], get(gca, 'ylim'));
xlabel(h_ax, 'Frame #');
ylabel(h_ax, 'Pupil position (\Delta pixels)');
%% plot average pupil position around events
nasal_event_avg = nanmean(nasal_events, 2);
temporal_event_avg = nanmean(temporal_events, 2);
figure
h_ax = axes();
hold on;
plot(h_ax, frames_around_event, nasal_event_avg, 'color', 'b');
plot(h_ax, frames_around_event, temporal_event_avg, 'color', 'r');
line(h_ax, [0, 0], get(gca, 'ylim'));
xlabel(h_ax, 'Frame #');
ylabel(h_ax, 'Pupil position (\Delta pixels)');
%% write csv
n_nasal_events = length(nasal_fast_move_frames);
n_temporal_events = length(temporal_fast_move_frames);
all_frames = [nasal_fast_move_frames(:); temporal_fast_move_frames(:)];
group = [ones(n_nasal_events, 1); 2*ones(n_temporal_events, 1)];
[all_frames, sort_idx] = sort(all_frames);
group = group(sort_idx);
% check
assert(isequal(all_frames(group == 1), nasal_fast_move_frames(:)), 'something wrong');
assert(isequal(all_frames(group == 2), temporal_fast_move_frames(:)), 'something wrong');
direction = cell(length(group), 1);
for ii = 1 : length(group)
if group(ii) == 1
direction{ii} = 'nasal';
elseif group(ii) == 2
direction{ii} = 'temporal';
end
end
% create table
tbl = table(all_frames, direction, 'VariableNames', {'saccade_frame', 'direction'});
% write to csv
if save_csv
writetable(tbl, 'camera0_saccades.csv');
end